From 5aee444159b29b0be68dfe0c74477131c2176a74 Mon Sep 17 00:00:00 2001 From: evlach Date: Sat, 3 Oct 2015 00:21:12 +0200 Subject: [PATCH] CriteriaBuilder approach --- lib/CriteriaBuilder.js | 75 +++++++++++++++++++ lib/CriteriaQuery.js | 165 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 240 insertions(+) create mode 100644 lib/CriteriaBuilder.js create mode 100644 lib/CriteriaQuery.js diff --git a/lib/CriteriaBuilder.js b/lib/CriteriaBuilder.js new file mode 100644 index 0000000..541daa2 --- /dev/null +++ b/lib/CriteriaBuilder.js @@ -0,0 +1,75 @@ +"use strict"; +//see https://docs.oracle.com/javaee/6/api/javax/persistence/criteria/CriteriaBuilder.html +let CriteriaQuery = require("./CriteriaQuery"); +module.exports = class CriteriaBuilder { + constructor(metaModel) + { + this.metaModel = metaModel; + } + + createQuery(metaModel) + { + metaModel = metaModel || this.metaModel; + return new CriteriaQuery(metaModel); + } + + and(expressions) + { + let q = ""; + if (Array.isArray(expressions)) + { + let count = expressions.length; + for (let expr of expressions) + { + q = q + expr + " "; + if (count > 1) + { + q = q + " AND "; + } + count--; + } + } + return q; + } + + // Tests whether two expressions are equal + equal(x, y) + { + return CriteriaBuilder.doOp(x, y, "="); + } + + //Tests whether the first numeric expression is greater than the second numeric expression + gt(x, y) + { + return CriteriaBuilder.doOp(x, y, ">"); + } + + //Tests whether the first numeric expression is greater than or equal to the second numeric + // expression + ge(x, y) + { + return CriteriaBuilder.doOp(x, y, ">="); + } + + //Tests whether the first numeric expression is less than the second numeric expression + lt(x, y) + { + return CriteriaBuilder.doOp(x, y, "<"); + } + + //Tests whether the first numeric expression is less than or equal to the second numeric + // expression + le(x, y) + { + return CriteriaBuilder.doOp(x, y, "<="); + } + + static doOp(x, y, op) + { + if (typeof y === "string") + { + return "\"" + x + "\"" + op + "'" + y + "'"; + } + return "\"" + x + "\"" + "=" + y; + } +}; \ No newline at end of file diff --git a/lib/CriteriaQuery.js b/lib/CriteriaQuery.js new file mode 100644 index 0000000..6f5781f --- /dev/null +++ b/lib/CriteriaQuery.js @@ -0,0 +1,165 @@ +"use strict"; +// see https://docs.oracle.com/javaee/6/api/javax/persistence/criteria/CriteriaQuery.html +module.exports = class CriteriaQuery { + constructor(metaModel) + { + this.metaModel = metaModel; + } + + static createQuery(metaModel) + { + this.metaModel = metaModel; + return new CriteriaQuery(metaModel); + } + + getFullTableName() + { + return "\"" + this.metaModel.keySpace + "\".\"" + this.metaModel.name + "\""; + } + + from(q, filtering) + { + q = q || ""; + if (typeof filtering !== "undefined" && filtering === false) + { + return "SELECT * FROM " + this.getFullTableName() + " " + q + ";"; + } + else + { + return "SELECT * FROM " + this.getFullTableName() + " " + q + " ALLOW FILTERING;"; + } + } + + where(q) + { + return " WHERE " + q; + } + + truncate() + { + return "TRUNCATE " + this.getFullTableName() + ";"; + } + + update(row, criteriaQuery) + { + let params = []; + let pairsQ = ""; + let propertiesCount = Object.getOwnPropertyNames(row).length - this.metaModel.getRestrictedFields().length; + let count = 0; + for (let key of this.metaModel.fields.keys()) + { + if (row.hasOwnProperty(key) && !this.metaModel.isRestricted(key)) + { + if (count>0) + { + pairsQ = pairsQ + ", "; + } + count++; + var value = row[key]; + params.push(value); + pairsQ = pairsQ + "\"" + key + "\"" + "=? "; + } + } + return { + query: "UPDATE " + this.getFullTableName() + " SET " + pairsQ + criteriaQuery + ";", + params: params + }; + } + + remove(criteriaQuery) + { + return { + query: "DELETE FROM " + this.getFullTableName() + criteriaQuery + ";", + params: [] + }; + } + + insert(row) + { + let params = []; + let keys = []; + let qq = []; + for (let key of this.metaModel.fields.keys()) + { + if (row.hasOwnProperty(key)) + { + var value = row[key]; + qq.push("?"); + keys.push(key); + params.push(value); + } + } + let paramsKeys = JSON.stringify(keys).toString().substring(1); + paramsKeys = " (" + paramsKeys.substring(0, paramsKeys.length - 1) + ") "; + let paramsQ = JSON.stringify(qq).toString().replace(/"/g, '').substring(1); + paramsQ = " (" + paramsQ.substring(0, paramsQ.length - 1) + ") "; + let queryObject = { + query: "INSERT INTO " + this.getFullTableName() + paramsKeys + " VALUES " + paramsQ + " IF NOT EXISTS;", + params: params + }; + return (queryObject); + } + + create() + { + let q = "CREATE TABLE IF NOT EXISTS " + this.getFullTableName() + " ( "; + this.metaModel.fields.forEach(function (value, key) + { + q = q + '"' + key + '"' + " " + value + ", "; + }); + q = q + "PRIMARY KEY (("; + if (this.metaModel.partitionKeys.length > 0) + { + this.metaModel.partitionKeys.forEach(function (item, index, array) + { + q = q + '"' + item + '"'; + if (index < array.length - 1) + { + q = q + ","; + } + }); + } + q = q + ")"; + // clustering keys + if (this.metaModel.clusteringColumns.size > 0) + { + q = q + ", "; + this.metaModel.clusteringColumns.forEach(function (value, key) + { + q = q + '"' + key + '"'; + q = q + ","; + }); + q = q.substring(0, q.length - 1); + q = q + ")) "; + q = q + "WITH CLUSTERING ORDER BY ("; + this.metaModel.clusteringColumns.forEach(function (value, key) + { + q = q + '"' + key + '" '; + q = q + value; + q = q + ","; + }); + q = q.substring(0, q.length - 1); + } + else + { + q = q + ") "; + } + q = q + "); "; + return q; + } + + drop() + { + return "DROP TABLE IF EXISTS " + this.getFullTableName() + ";"; + } + + insertIndex(field) + { + return "CREATE INDEX ON " + this.getFullTableName() + "(" + field + ");"; + } + + dropIndex(indexName) + { + return "DROP INDEX IF EXISTS \"" + this.metaModel.keySpace + "\".\"" + indexName + "\";"; + } +}; \ No newline at end of file