UNPKG

objection

Version:
445 lines (335 loc) 33.8 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = undefined; var _create = require('babel-runtime/core-js/object/create'); var _create2 = _interopRequireDefault(_create); var _keys = require('babel-runtime/core-js/object/keys'); var _keys2 = _interopRequireDefault(_keys); var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck'); var _classCallCheck3 = _interopRequireDefault(_classCallCheck2); var _createClass2 = require('babel-runtime/helpers/createClass'); var _createClass3 = _interopRequireDefault(_createClass2); var _lodash = require('lodash'); var _lodash2 = _interopRequireDefault(_lodash); var _relationExpressionParser = require('./parsers/relationExpressionParser'); var _relationExpressionParser2 = _interopRequireDefault(_relationExpressionParser); var _ValidationError = require('../model/ValidationError'); var _ValidationError2 = _interopRequireDefault(_ValidationError); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var RECURSIVE_REGEX = /^\^(\d*)$/; var ALL_RECURSIVE_REGEX = /^\*$/; var RelationExpression = function () { function RelationExpression(node, recursionDepth, filters) { (0, _classCallCheck3.default)(this, RelationExpression); node = node || {}; this.name = node.name || null; this.args = node.args || []; this.numChildren = node.numChildren || 0; this.children = node.children || {}; Object.defineProperty(this, '_recursionDepth', { enumerable: false, value: recursionDepth || 0 }); Object.defineProperty(this, '_filters', { enumerable: false, writable: true, value: filters || {} }); } /** * @param {string|RelationExpression} expr * @returns {RelationExpression} */ RelationExpression.parse = function parse(expr) { if (expr instanceof RelationExpression) { return expr; } else if (!_lodash2.default.isString(expr) || _lodash2.default.isEmpty(expr.trim())) { return new RelationExpression(); } else { try { return new RelationExpression(_relationExpressionParser2.default.parse(expr)); } catch (err) { throw new _ValidationError2.default({ message: 'Invalid relation expression "' + expr + '"', cause: err.message }); } } }; /** * @param {Object|Array} graph */ RelationExpression.fromGraph = function fromGraph(graph) { if (!graph) { return new RelationExpression(); } return new RelationExpression(modelGraphToNode(graph, newNode())); }; /** * @param {string|RelationExpression} expr * @returns {RelationExpression} */ RelationExpression.prototype.merge = function merge(expr) { var merged = this.clone(); expr = RelationExpression.parse(expr); merged.numChildren += expr.numChildren; merged.children = _lodash2.default.merge(merged.children, expr.children); merged.args = _lodash2.default.merge(merged.args, expr.args); merged.filters = _lodash2.default.merge(merged.filters, expr.filters); // Handle recursive and all recursive nodes. visit(merged, function (node, childNames) { var maxName = null; var maxDepth = 0; var recurCount = 0; for (var i = 0, l = childNames.length; i < l; ++i) { var name = childNames[i]; var depth = _maxRecursionDepth(name); if (depth > 0) { recurCount++; } if (depth > maxDepth) { maxDepth = depth; maxName = name; } } if (recurCount > 0) { delete node.children[node.name]; } if (recurCount > 1) { for (var _i = 0, _l = childNames.length; _i < _l; ++_i) { var _name = childNames[_i]; if (_name !== maxName) { delete node.children[_name]; } } } }); return merged; }; /** * @param {string|RelationExpression} expr * @returns {boolean} */ RelationExpression.prototype.isSubExpression = function isSubExpression(expr) { var _this = this; expr = RelationExpression.parse(expr); if (this.isAllRecursive()) { return true; } if (expr.isAllRecursive()) { return this.isAllRecursive(); } if (this.name !== expr.name) { return false; } var maxRecursionDepth = expr.maxRecursionDepth(); if (maxRecursionDepth > 0) { return this.isAllRecursive() || this.maxRecursionDepth() >= maxRecursionDepth; } return _lodash2.default.every(expr.children, function (child, childName) { var ownSubExpression = _this.childExpression(childName); var subExpression = expr.childExpression(childName); return ownSubExpression && ownSubExpression.isSubExpression(subExpression); }); }; /** * @returns {number} */ RelationExpression.prototype.maxRecursionDepth = function maxRecursionDepth() { if (this.numChildren !== 1) { return 0; } var key = (0, _keys2.default)(this.children)[0]; return _maxRecursionDepth(key); }; /** * @returns {boolean} */ RelationExpression.prototype.isAllRecursive = function isAllRecursive() { if (this.numChildren !== 1) { return false; } var key = (0, _keys2.default)(this.children)[0]; return ALL_RECURSIVE_REGEX.test(key); }; /** * @returns {RelationExpression} */ RelationExpression.prototype.childExpression = function childExpression(childName) { if (this.isAllRecursive() || childName === this.name && this._recursionDepth < this.maxRecursionDepth() - 1) { return new RelationExpression(this, this._recursionDepth + 1, this._filters); } if (this.children[childName]) { return new RelationExpression(this.children[childName], 0, this._filters); } else { return null; } }; /** * @returns {RelationExpression} */ RelationExpression.prototype.clone = function clone() { var node = { name: this.name, args: this.args, numChildren: this.numChildren, children: _lodash2.default.cloneDeep(this.children) }; var filters = _lodash2.default.clone(this._filters); return new RelationExpression(node, this._recursionDepth, filters); }; RelationExpression.prototype.forEachChild = function forEachChild(cb) { _lodash2.default.forOwn(this.children, function (child, childName) { if (!ALL_RECURSIVE_REGEX.test(childName) && !RECURSIVE_REGEX.test(childName)) { cb(child, childName); } }); }; /** * @param {string|RelationExpression} path * @param {function(QueryBuilder)} filter */ RelationExpression.prototype.addAnonymousFilterAtPath = function addAnonymousFilterAtPath(path, filter) { var filterNodes = this._nodesAtPath(path); var filters = this.filters; var idx = 0; var filterName = '_efe0_'; while (filters[filterName]) { filterName = '_efe' + ++idx + '_'; } if (!_lodash2.default.isEmpty(filterNodes)) { filters[filterName] = filter; _lodash2.default.each(filterNodes, function (node) { return node.args.push(filterName); }); } }; /** * @returns {string} */ RelationExpression.prototype.toString = function toString() { return _toString(this); }; /** * @private * @return {Array.<Object>} */ RelationExpression.prototype._nodesAtPath = function _nodesAtPath(pathExpression) { var path = RelationExpression.parse(pathExpression); var nodes = []; RelationExpression.nodesAtPath(this, path, nodes); return nodes; }; /** * @private */ RelationExpression.nodesAtPath = function nodesAtPath(target, path, expressions) { var _this2 = this; if (path.numChildren == 0) { expressions.push(target); } else { _lodash2.default.forOwn(path.children, function (child) { var targetChild = target.children[child.name]; if (targetChild) { _this2.nodesAtPath(targetChild, child, expressions); } }); } }; (0, _createClass3.default)(RelationExpression, [{ key: 'filters', get: function get() { return this._filters; }, set: function set(filters) { this._filters = filters || {}; } }]); return RelationExpression; }(); exports.default = RelationExpression; function _maxRecursionDepth(key) { var rec = RECURSIVE_REGEX.exec(key); if (rec) { var maxDepth = rec[1]; if (maxDepth) { return parseInt(maxDepth, 10); } else { return Number.POSITIVE_INFINITY; } } else { return 0; } } function visit(node, visitor) { var keys = (0, _keys2.default)(node.children); visitor(node, keys); for (var i = 0, l = keys.length; i < l; ++i) { var key = keys[i]; var childNode = node.children[key]; if (childNode) { visit(childNode, visitor); } } } function _toString(node) { var childExpr = _lodash2.default.values(node.children).map(_toString); if (childExpr.length > 1) { childExpr = '[' + childExpr.join(', ') + ']'; } else { childExpr = childExpr[0]; } var str = node.name; if (node.args.length) { str += '(' + node.args.join(', ') + ')'; } if (childExpr) { if (str) { return str + '.' + childExpr; } else { return childExpr; } } else { return str; } } function modelGraphToNode(models, node) { if (!models) { return; } if (Array.isArray(models)) { for (var i = 0, l = models.length; i < l; ++i) { modelToNode(models[i], node); } } else { modelToNode(models, node); } return node; } function modelToNode(model, node) { var modelClass = model.constructor; var relations = modelClass.getRelationArray(); for (var r = 0, lr = relations.length; r < lr; ++r) { var relName = relations[r].name; if (model.hasOwnProperty(relName)) { var childNode = node.children[relName]; if (!childNode) { childNode = newNode(relName); node.children[relName] = childNode; node.numChildren++; } modelGraphToNode(model[relName], childNode); } } } function newNode(name) { return { name: name || '', args: [], children: (0, _create2.default)(null), numChildren: 0 }; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["RelationExpression.js"],"names":["RECURSIVE_REGEX","ALL_RECURSIVE_REGEX","RelationExpression","node","recursionDepth","filters","name","args","numChildren","children","Object","defineProperty","enumerable","value","writable","parse","expr","isString","isEmpty","trim","err","message","cause","fromGraph","graph","modelGraphToNode","newNode","merge","merged","clone","visit","childNames","maxName","maxDepth","recurCount","i","l","length","depth","maxRecursionDepth","isSubExpression","isAllRecursive","every","child","childName","ownSubExpression","childExpression","subExpression","key","test","_recursionDepth","_filters","cloneDeep","forEachChild","cb","forOwn","addAnonymousFilterAtPath","path","filter","filterNodes","_nodesAtPath","idx","filterName","each","push","toString","pathExpression","nodes","nodesAtPath","target","expressions","targetChild","rec","exec","parseInt","Number","POSITIVE_INFINITY","visitor","keys","childNode","childExpr","values","map","join","str","models","Array","isArray","modelToNode","model","modelClass","constructor","relations","getRelationArray","r","lr","relName","hasOwnProperty"],"mappings":"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","file":"RelationExpression.js","sourcesContent":["import _ from 'lodash';\nimport parser from './parsers/relationExpressionParser';\nimport ValidationError from '../model/ValidationError';\n\nconst RECURSIVE_REGEX = /^\\^(\\d*)$/;\nconst ALL_RECURSIVE_REGEX = /^\\*$/;\n\nexport default class RelationExpression {\n\n  constructor(node, recursionDepth, filters) {\n    node = node || {};\n\n    this.name = node.name || null;\n    this.args = node.args || [];\n    this.numChildren = node.numChildren || 0;\n    this.children = node.children || {};\n\n    Object.defineProperty(this, '_recursionDepth', {\n      enumerable: false,\n      value: recursionDepth || 0\n    });\n\n    Object.defineProperty(this, '_filters', {\n      enumerable: false,\n      writable: true,\n      value: filters || {}\n    });\n  }\n\n  /**\n   * @param {string|RelationExpression} expr\n   * @returns {RelationExpression}\n   */\n  static parse(expr) {\n    if (expr instanceof RelationExpression) {\n      return expr;\n    } else if (!_.isString(expr) || _.isEmpty(expr.trim())) {\n      return new RelationExpression();\n    } else {\n      try {\n        return new RelationExpression(parser.parse(expr));\n      } catch (err) {\n        throw new ValidationError({\n          message: 'Invalid relation expression \"' + expr + '\"',\n          cause: err.message\n        });\n      }\n    }\n  }\n\n  /**\n   * @param {Object|Array} graph\n   */\n  static fromGraph(graph) {\n    if (!graph) {\n      return new RelationExpression();\n    }\n\n    return new RelationExpression(modelGraphToNode(graph, newNode()));\n  }\n\n  get filters() {\n    return this._filters;\n  }\n\n  set filters(filters) {\n    this._filters = filters || {};\n  }\n\n  /**\n   * @param {string|RelationExpression} expr\n   * @returns {RelationExpression}\n   */\n  merge(expr) {\n    const merged = this.clone();\n    expr = RelationExpression.parse(expr);\n\n    merged.numChildren += expr.numChildren;\n    merged.children = _.merge(merged.children, expr.children);\n    merged.args = _.merge(merged.args, expr.args);\n    merged.filters = _.merge(merged.filters, expr.filters);\n\n    // Handle recursive and all recursive nodes.\n    visit(merged, (node, childNames) => {\n      let maxName = null;\n      let maxDepth = 0;\n      let recurCount = 0;\n\n      for (let i = 0, l = childNames.length; i < l; ++i) {\n        const name = childNames[i];\n        const depth = maxRecursionDepth(name);\n\n        if (depth > 0) {\n          recurCount++;\n        }\n\n        if (depth > maxDepth) {\n          maxDepth = depth;\n          maxName = name;\n        }\n      }\n\n      if (recurCount > 0) {\n        delete node.children[node.name];\n      }\n\n      if (recurCount > 1) {\n        for (let i = 0, l = childNames.length; i < l; ++i) {\n          const name = childNames[i];\n\n          if (name !== maxName) {\n            delete node.children[name];\n          }\n        }\n      }\n    });\n\n    return merged;\n  }\n\n  /**\n   * @param {string|RelationExpression} expr\n   * @returns {boolean}\n   */\n  isSubExpression(expr) {\n    expr = RelationExpression.parse(expr);\n\n    if (this.isAllRecursive()) {\n      return true;\n    }\n\n    if (expr.isAllRecursive()) {\n      return this.isAllRecursive();\n    }\n\n    if (this.name !== expr.name) {\n      return false;\n    }\n\n    const maxRecursionDepth = expr.maxRecursionDepth();\n\n    if (maxRecursionDepth > 0) {\n      return this.isAllRecursive() || this.maxRecursionDepth() >= maxRecursionDepth;\n    }\n\n    return _.every(expr.children, (child, childName) => {\n      var ownSubExpression = this.childExpression(childName);\n      var subExpression = expr.childExpression(childName);\n\n      return ownSubExpression && ownSubExpression.isSubExpression(subExpression);\n    });\n  }\n\n  /**\n   * @returns {number}\n   */\n  maxRecursionDepth() {\n    if (this.numChildren !== 1) {\n      return 0;\n    }\n\n    const key = Object.keys(this.children)[0];\n    return maxRecursionDepth(key);\n  }\n\n  /**\n   * @returns {boolean}\n   */\n  isAllRecursive() {\n    if (this.numChildren !== 1) {\n      return false;\n    }\n\n    const key = Object.keys(this.children)[0];\n    return ALL_RECURSIVE_REGEX.test(key);\n  }\n\n  /**\n   * @returns {RelationExpression}\n   */\n  childExpression(childName) {\n    if (this.isAllRecursive() || (childName === this.name && this._recursionDepth < this.maxRecursionDepth() - 1)) {\n      return new RelationExpression(this, this._recursionDepth + 1, this._filters);\n    }\n\n    if (this.children[childName]) {\n      return new RelationExpression(this.children[childName], 0, this._filters);\n    } else {\n      return null;\n    }\n  }\n\n  /**\n   * @returns {RelationExpression}\n   */\n  clone() {\n    const node = {\n      name: this.name,\n      args: this.args,\n      numChildren: this.numChildren,\n      children: _.cloneDeep(this.children)\n    };\n\n    const filters = _.clone(this._filters);\n    return new RelationExpression(node, this._recursionDepth, filters);\n  }\n\n  forEachChild(cb) {\n    _.forOwn(this.children, (child, childName) => {\n      if (!ALL_RECURSIVE_REGEX.test(childName) && !RECURSIVE_REGEX.test(childName)) {\n        cb(child, childName);\n      }\n    });\n  }\n\n  /**\n   * @param {string|RelationExpression} path\n   * @param {function(QueryBuilder)} filter\n   */\n  addAnonymousFilterAtPath(path, filter) {\n    let filterNodes = this._nodesAtPath(path);\n    let filters = this.filters;\n\n    let idx = 0;\n    let filterName = `_efe0_`;\n\n    while (filters[filterName]) {\n      filterName = `_efe${++idx}_`;\n    }\n\n    if (!_.isEmpty(filterNodes)) {\n      filters[filterName] = filter;\n      _.each(filterNodes, node => node.args.push(filterName));\n    }\n  }\n\n  /**\n   * @returns {string}\n   */\n  toString() {\n    return toString(this);\n  }\n\n  /**\n   * @private\n   * @return {Array.<Object>}\n   */\n  _nodesAtPath(pathExpression) {\n    let path = RelationExpression.parse(pathExpression);\n    let nodes = [];\n\n    RelationExpression.nodesAtPath(this, path, nodes);\n    return nodes;\n  }\n\n  /**\n   * @private\n   */\n  static nodesAtPath(target, path, expressions) {\n    if (path.numChildren == 0) {\n      expressions.push(target);\n    } else {\n      _.forOwn(path.children, child => {\n        const targetChild = target.children[child.name];\n\n        if (targetChild) {\n          this.nodesAtPath(targetChild, child, expressions);\n        }\n      });\n    }\n  }\n}\n\nfunction maxRecursionDepth(key) {\n  const rec = RECURSIVE_REGEX.exec(key);\n\n  if (rec) {\n    const maxDepth = rec[1];\n\n    if (maxDepth) {\n      return parseInt(maxDepth, 10);\n    } else {\n      return Number.POSITIVE_INFINITY;\n    }\n  } else {\n    return 0;\n  }\n}\n\nfunction visit(node, visitor) {\n  const keys = Object.keys(node.children);\n\n  visitor(node, keys);\n\n  for (let i = 0, l = keys.length; i < l; ++i) {\n    const key = keys[i];\n    const childNode = node.children[key];\n\n    if (childNode) {\n      visit(childNode, visitor);\n    }\n  }\n}\n\n\nfunction toString(node) {\n  let childExpr = _.values(node.children).map(toString);\n\n  if (childExpr.length > 1) {\n    childExpr = `[${childExpr.join(', ')}]`;\n  } else {\n    childExpr = childExpr[0];\n  }\n\n  let str = node.name;\n\n  if (node.args.length) {\n    str += `(${node.args.join(', ')})`;\n  }\n\n  if (childExpr) {\n    if (str) {\n      return `${str}.${childExpr}`;\n    } else {\n      return childExpr;\n    }\n  } else {\n    return str;\n  }\n}\n\nfunction modelGraphToNode(models, node) {\n  if (!models) {\n    return;\n  }\n\n  if (Array.isArray(models)) {\n    for (let i = 0, l = models.length; i < l; ++i) {\n      modelToNode(models[i], node);\n    }\n  } else {\n    modelToNode(models, node);\n  }\n\n  return node;\n}\n\nfunction modelToNode(model, node) {\n  const modelClass = model.constructor;\n  const relations = modelClass.getRelationArray();\n\n  for (let r = 0, lr = relations.length; r < lr; ++r) {\n    const relName = relations[r].name;\n\n    if (model.hasOwnProperty(relName)) {\n      let childNode = node.children[relName];\n\n      if (!childNode) {\n        childNode = newNode(relName);\n\n        node.children[relName] = childNode;\n        node.numChildren++;\n      }\n\n      modelGraphToNode(model[relName], childNode);\n    }\n  }\n}\n\nfunction newNode(name) {\n  return {\n    name: name || '',\n    args: [],\n    children: Object.create(null),\n    numChildren: 0\n  };\n}"]}