objection
Version:
An SQL-friendly ORM for Node.js
502 lines (374 loc) • 48.6 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = undefined;
var _getOwnPropertyDescriptor = require('babel-runtime/core-js/object/get-own-property-descriptor');
var _getOwnPropertyDescriptor2 = _interopRequireDefault(_getOwnPropertyDescriptor);
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 _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
var _inherits2 = require('babel-runtime/helpers/inherits');
var _inherits3 = _interopRequireDefault(_inherits2);
var _desc, _value, _class;
var _lodash = require('lodash');
var _lodash2 = _interopRequireDefault(_lodash);
var _Relation2 = require('../Relation');
var _Relation3 = _interopRequireDefault(_Relation2);
var _inheritModel = require('../../model/inheritModel');
var _inheritModel2 = _interopRequireDefault(_inheritModel);
var _dbUtils = require('../../utils/dbUtils');
var _memoize = require('../../utils/decorators/memoize');
var _memoize2 = _interopRequireDefault(_memoize);
var _ManyToManyFindOperation = require('./ManyToManyFindOperation');
var _ManyToManyFindOperation2 = _interopRequireDefault(_ManyToManyFindOperation);
var _ManyToManyInsertOperation = require('./ManyToManyInsertOperation');
var _ManyToManyInsertOperation2 = _interopRequireDefault(_ManyToManyInsertOperation);
var _ManyToManyRelateOperation = require('./ManyToManyRelateOperation');
var _ManyToManyRelateOperation2 = _interopRequireDefault(_ManyToManyRelateOperation);
var _ManyToManyUnrelateOperation = require('./ManyToManyUnrelateOperation');
var _ManyToManyUnrelateOperation2 = _interopRequireDefault(_ManyToManyUnrelateOperation);
var _ManyToManyUnrelateSqliteOperation = require('./ManyToManyUnrelateSqliteOperation');
var _ManyToManyUnrelateSqliteOperation2 = _interopRequireDefault(_ManyToManyUnrelateSqliteOperation);
var _ManyToManyUpdateOperation = require('./ManyToManyUpdateOperation');
var _ManyToManyUpdateOperation2 = _interopRequireDefault(_ManyToManyUpdateOperation);
var _ManyToManyUpdateSqliteOperation = require('./ManyToManyUpdateSqliteOperation');
var _ManyToManyUpdateSqliteOperation2 = _interopRequireDefault(_ManyToManyUpdateSqliteOperation);
var _ManyToManyDeleteOperation = require('./ManyToManyDeleteOperation');
var _ManyToManyDeleteOperation2 = _interopRequireDefault(_ManyToManyDeleteOperation);
var _ManyToManyDeleteSqliteOperation = require('./ManyToManyDeleteSqliteOperation');
var _ManyToManyDeleteSqliteOperation2 = _interopRequireDefault(_ManyToManyDeleteSqliteOperation);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _applyDecoratedDescriptor(target, property, decorators, descriptor, context) {
var desc = {};
Object['ke' + 'ys'](descriptor).forEach(function (key) {
desc[key] = descriptor[key];
});
desc.enumerable = !!desc.enumerable;
desc.configurable = !!desc.configurable;
if ('value' in desc || desc.initializer) {
desc.writable = true;
}
desc = decorators.slice().reverse().reduce(function (desc, decorator) {
return decorator(target, property, desc) || desc;
}, desc);
if (context && desc.initializer !== void 0) {
desc.value = desc.initializer ? desc.initializer.call(context) : void 0;
desc.initializer = undefined;
}
if (desc.initializer === void 0) {
Object['define' + 'Property'](target, property, desc);
desc = null;
}
return desc;
}
var sqliteBuiltInRowId = '_rowid_';
var ManyToManyRelation = (_class = function (_Relation) {
(0, _inherits3.default)(ManyToManyRelation, _Relation);
function ManyToManyRelation() {
(0, _classCallCheck3.default)(this, ManyToManyRelation);
return (0, _possibleConstructorReturn3.default)(this, _Relation.apply(this, arguments));
}
ManyToManyRelation.prototype.setMapping = function setMapping(mapping) {
var retVal = _Relation.prototype.setMapping.call(this, mapping);
// Avoid require loop and import here.
var Model = require(__dirname + '/../../model/Model').default;
if (!_lodash2.default.isObject(mapping.join.through)) {
this.throwError('join must have the `through` that describes the join table.');
}
if (!mapping.join.through.from || !mapping.join.through.to) {
this.throwError('join.through must be an object that describes the join table. For example: {from: "JoinTable.someId", to: "JoinTable.someOtherId"}');
}
var joinFrom = this.parseReference(mapping.join.from);
var joinTableFrom = this.parseReference(mapping.join.through.from);
var joinTableTo = this.parseReference(mapping.join.through.to);
var joinTableExtra = mapping.join.through.extra || [];
if (!joinTableFrom.table || _lodash2.default.isEmpty(joinTableFrom.columns)) {
this.throwError('join.through.from must have format JoinTable.columnName. For example "JoinTable.someId" or in case of composite key ["JoinTable.a", "JoinTable.b"].');
}
if (!joinTableTo.table || _lodash2.default.isEmpty(joinTableTo.columns)) {
this.throwError('join.through.to must have format JoinTable.columnName. For example "JoinTable.someId" or in case of composite key ["JoinTable.a", "JoinTable.b"].');
}
if (joinTableFrom.table !== joinTableTo.table) {
this.throwError('join.through `from` and `to` must point to the same join table.');
}
this.joinTable = joinTableFrom.table;
if (joinFrom.table === this.ownerModelClass.tableName) {
this.joinTableOwnerCol = joinTableFrom.columns;
this.joinTableRelatedCol = joinTableTo.columns;
} else {
this.joinTableRelatedCol = joinTableFrom.columns;
this.joinTableOwnerCol = joinTableTo.columns;
}
if (mapping.join.through.modelClass) {
this._joinTableModelClass = this.resolveModel(Model, mapping.join.through.modelClass, 'join.through.modelClass');
} else {
this._joinTableModelClass = (0, _inheritModel2.default)(Model);
this._joinTableModelClass.tableName = this.joinTable;
// We cannot know if the join table has a primary key. Therefore we set some
// known column as the idColumn so that inserts will work.
this._joinTableModelClass.idColumn = this.joinTableRelatedCol;
}
this.joinTableOwnerProp = this.propertyName(this.joinTableOwnerCol, this._joinTableModelClass);
this.joinTableRelatedProp = this.propertyName(this.joinTableRelatedCol, this._joinTableModelClass);
this.joinTableExtras = this.parseExtras(joinTableExtra);
return retVal;
};
/**
* @returns {Array.<string>}
*/
ManyToManyRelation.prototype.fullJoinTableOwnerCol = function fullJoinTableOwnerCol() {
var _this2 = this;
return this.joinTableOwnerCol.map(function (col) {
return _this2.joinTable + '.' + col;
});
};
/**
* @returns {Array.<string>}
*/
ManyToManyRelation.prototype.fullJoinTableRelatedCol = function fullJoinTableRelatedCol() {
var _this3 = this;
return this.joinTableRelatedCol.map(function (col) {
return _this3.joinTable + '.' + col;
});
};
/**
* @returns {string}
*/
ManyToManyRelation.prototype.joinTableAlias = function joinTableAlias() {
return this.joinTable + '_rel_' + this.name;
};
/**
* @returns {ManyToManyRelation}
*/
ManyToManyRelation.prototype.bindKnex = function bindKnex(knex) {
var bound = _Relation.prototype.bindKnex.call(this, knex);
bound._joinTableModelClass = this._joinTableModelClass.bindKnex(knex);
return bound;
};
/**
* @returns {QueryBuilder}
*/
ManyToManyRelation.prototype.findQuery = function findQuery(builder, opt) {
var _this4 = this;
builder.join(this.joinTable, function (join) {
var fullRelatedCol = _this4.fullRelatedCol();
var fullJoinTableRelatedCol = _this4.fullJoinTableRelatedCol();
for (var i = 0, l = fullJoinTableRelatedCol.length; i < l; ++i) {
join.on(fullJoinTableRelatedCol[i], fullRelatedCol[i]);
}
});
if (opt.isColumnRef) {
var fullJoinTableOwnerCol = this.fullJoinTableOwnerCol();
for (var i = 0, l = fullJoinTableOwnerCol.length; i < l; ++i) {
builder.whereRef(fullJoinTableOwnerCol[i], opt.ownerIds[i]);
}
} else {
var hasIds = false;
for (var _i = 0, _l = opt.ownerIds.length; _i < _l; ++_i) {
var id = opt.ownerIds[_i];
if (id) {
hasIds = true;
break;
}
}
if (hasIds) {
builder.whereInComposite(this.fullJoinTableOwnerCol(), opt.ownerIds);
} else {
builder.resolve([]);
}
}
return builder.modify(this.modify);
};
/**
* @returns {QueryBuilder}
*/
ManyToManyRelation.prototype.join = function join(builder, opt) {
opt = opt || {};
opt.joinOperation = opt.joinOperation || 'join';
opt.relatedTableAlias = opt.relatedTableAlias || this.relatedTableAlias();
opt.relatedJoinSelectQuery = opt.relatedJoinSelectQuery || this.relatedModelClass.query().childQueryOf(builder);
opt.relatedTable = opt.relatedTable || this.relatedModelClass.tableName;
opt.ownerTable = opt.ownerTable || this.ownerModelClass.tableName;
opt.joinTableAlias = opt.joinTableAlias || opt.relatedTableAlias + '_join';
var joinTableAsAlias = this.joinTable + ' as ' + opt.joinTableAlias;
var joinTableOwnerCol = this.joinTableOwnerCol.map(function (col) {
return opt.joinTableAlias + '.' + col;
});
var joinTableRelatedCol = this.joinTableRelatedCol.map(function (col) {
return opt.joinTableAlias + '.' + col;
});
var relatedCol = this.relatedCol.map(function (col) {
return opt.relatedTableAlias + '.' + col;
});
var ownerCol = this.ownerCol.map(function (col) {
return opt.ownerTable + '.' + col;
});
var relatedJoinSelect = opt.relatedJoinSelectQuery.modify(this.modify).as(opt.relatedTableAlias);
if (relatedJoinSelect.isSelectAll()) {
// No need to join a subquery if the query is `select * from "RelatedTable"`.
relatedJoinSelect = this.relatedModelClass.tableName + ' as ' + opt.relatedTableAlias;
}
return builder[opt.joinOperation](joinTableAsAlias, function (join) {
for (var i = 0, l = joinTableOwnerCol.length; i < l; ++i) {
join.on(joinTableOwnerCol[i], ownerCol[i]);
}
})[opt.joinOperation](relatedJoinSelect, function (join) {
for (var i = 0, l = joinTableRelatedCol.length; i < l; ++i) {
join.on(joinTableRelatedCol[i], relatedCol[i]);
}
});
};
ManyToManyRelation.prototype.find = function find(builder, owners) {
return new _ManyToManyFindOperation2.default('find', {
relation: this,
owners: owners
});
};
ManyToManyRelation.prototype.insert = function insert(builder, owner) {
return new _ManyToManyInsertOperation2.default('insert', {
relation: this,
owner: owner
});
};
ManyToManyRelation.prototype.update = function update(builder, owner) {
if ((0, _dbUtils.isSqlite)(builder.knex())) {
return new _ManyToManyUpdateSqliteOperation2.default('update', {
relation: this,
owner: owner
});
} else {
return new _ManyToManyUpdateOperation2.default('update', {
relation: this,
owner: owner
});
}
};
ManyToManyRelation.prototype.patch = function patch(builder, owner) {
if ((0, _dbUtils.isSqlite)(builder.knex())) {
return new _ManyToManyUpdateSqliteOperation2.default('patch', {
relation: this,
owner: owner,
modelOptions: { patch: true }
});
} else {
return new _ManyToManyUpdateOperation2.default('patch', {
relation: this,
owner: owner,
modelOptions: { patch: true }
});
}
};
ManyToManyRelation.prototype.delete = function _delete(builder, owner) {
if ((0, _dbUtils.isSqlite)(builder.knex())) {
return new _ManyToManyDeleteSqliteOperation2.default('delete', {
relation: this,
owner: owner
});
} else {
return new _ManyToManyDeleteOperation2.default('delete', {
relation: this,
owner: owner
});
}
};
ManyToManyRelation.prototype.relate = function relate(builder, owner) {
return new _ManyToManyRelateOperation2.default('relate', {
relation: this,
owner: owner
});
};
ManyToManyRelation.prototype.unrelate = function unrelate(builder, owner) {
if ((0, _dbUtils.isSqlite)(builder.knex())) {
return new _ManyToManyUnrelateSqliteOperation2.default('unrelate', {
relation: this,
owner: owner
});
} else {
return new _ManyToManyUnrelateOperation2.default('unrelate', {
relation: this,
owner: owner
});
}
};
ManyToManyRelation.prototype.selectForModify = function selectForModify(builder, owner) {
var ownerId = owner.$values(this.ownerProp);
var idQuery = this.joinTableModelClass(builder.knex()).query().childQueryOf(builder).select(this.fullJoinTableRelatedCol()).whereComposite(this.fullJoinTableOwnerCol(), ownerId);
return builder.whereInComposite(this.fullRelatedCol(), idQuery);
};
ManyToManyRelation.prototype.selectForModifySqlite = function selectForModifySqlite(builder, owner) {
var _this5 = this;
var relatedTable = this.relatedModelClass.tableName;
var relatedTableAlias = this.relatedTableAlias();
var relatedTableAsAlias = relatedTable + ' as ' + relatedTableAlias;
var relatedTableAliasRowId = relatedTableAlias + '.' + sqliteBuiltInRowId;
var relatedTableRowId = relatedTable + '.' + sqliteBuiltInRowId;
var selectRelatedQuery = this.joinTableModelClass(builder.knex()).query().childQueryOf(builder).select(relatedTableAliasRowId).whereComposite(this.fullJoinTableOwnerCol(), owner.$values(this.ownerProp)).join(relatedTableAsAlias, function (join) {
var fullJoinTableRelatedCols = _this5.fullJoinTableRelatedCol();
var fullRelatedCol = _this5.fullRelatedCol();
for (var i = 0, l = fullJoinTableRelatedCols.length; i < l; ++i) {
join.on(fullJoinTableRelatedCols[i], fullRelatedCol[i]);
}
});
return builder.whereInComposite(relatedTableRowId, selectRelatedQuery);
};
ManyToManyRelation.prototype.createJoinModels = function createJoinModels(ownerId, related) {
var joinModels = new Array(related.length);
for (var i = 0, lr = related.length; i < lr; ++i) {
var rel = related[i];
var joinModel = {};
for (var j = 0, lp = this.joinTableOwnerProp.length; j < lp; ++j) {
joinModel[this.joinTableOwnerProp[j]] = ownerId[j];
}
for (var _j = 0, _lp = this.joinTableRelatedProp.length; _j < _lp; ++_j) {
joinModel[this.joinTableRelatedProp[_j]] = rel[this.relatedProp[_j]];
}
for (var _j2 = 0, _lp2 = this.joinTableExtras.length; _j2 < _lp2; ++_j2) {
var extra = this.joinTableExtras[_j2];
var extraValue = rel[extra.aliasProp];
if (!_lodash2.default.isUndefined(extraValue)) {
joinModel[extra.joinTableProp] = extraValue;
}
}
joinModels[i] = joinModel;
}
return joinModels;
};
ManyToManyRelation.prototype.omitExtraProps = function omitExtraProps(models) {
if (!_lodash2.default.isEmpty(this.joinTableExtras)) {
var props = this.joinTableExtras.map(function (extra) {
return extra.aliasProp;
});
for (var i = 0, l = models.length; i < l; ++i) {
models[i].$omitFromDatabaseJson(props);
}
}
};
/**
* @protected
*/
ManyToManyRelation.prototype.parseExtras = function parseExtras(extras) {
var _this6 = this;
if (Array.isArray(extras)) {
extras = extras.reduce(function (extras, col) {
extras[col] = col;
return extras;
}, {});
}
return (0, _keys2.default)(extras).map(function (key) {
var val = extras[key];
return {
joinTableCol: val,
joinTableProp: _this6._joinTableModelClass.columnNameToPropertyName(val),
aliasCol: key,
aliasProp: _this6._joinTableModelClass.columnNameToPropertyName(key)
};
});
};
return ManyToManyRelation;
}(_Relation3.default), (_applyDecoratedDescriptor(_class.prototype, 'fullJoinTableOwnerCol', [_memoize2.default], (0, _getOwnPropertyDescriptor2.default)(_class.prototype, 'fullJoinTableOwnerCol'), _class.prototype), _applyDecoratedDescriptor(_class.prototype, 'fullJoinTableRelatedCol', [_memoize2.default], (0, _getOwnPropertyDescriptor2.default)(_class.prototype, 'fullJoinTableRelatedCol'), _class.prototype)), _class);
exports.default = ManyToManyRelation;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ManyToManyRelation.js"],"names":["sqliteBuiltInRowId","ManyToManyRelation","setMapping","mapping","retVal","Model","require","__dirname","default","isObject","join","through","throwError","from","to","joinFrom","parseReference","joinTableFrom","joinTableTo","joinTableExtra","extra","table","isEmpty","columns","joinTable","ownerModelClass","tableName","joinTableOwnerCol","joinTableRelatedCol","modelClass","_joinTableModelClass","resolveModel","idColumn","joinTableOwnerProp","propertyName","joinTableRelatedProp","joinTableExtras","parseExtras","fullJoinTableOwnerCol","map","col","fullJoinTableRelatedCol","joinTableAlias","name","bindKnex","knex","bound","findQuery","builder","opt","fullRelatedCol","i","l","length","on","isColumnRef","whereRef","ownerIds","hasIds","id","whereInComposite","resolve","modify","joinOperation","relatedTableAlias","relatedJoinSelectQuery","relatedModelClass","query","childQueryOf","relatedTable","ownerTable","joinTableAsAlias","relatedCol","ownerCol","relatedJoinSelect","as","isSelectAll","find","owners","relation","insert","owner","update","patch","modelOptions","delete","relate","unrelate","selectForModify","ownerId","$values","ownerProp","idQuery","joinTableModelClass","select","whereComposite","selectForModifySqlite","relatedTableAsAlias","relatedTableAliasRowId","relatedTableRowId","selectRelatedQuery","fullJoinTableRelatedCols","createJoinModels","related","joinModels","Array","lr","rel","joinModel","j","lp","relatedProp","extraValue","aliasProp","isUndefined","joinTableProp","omitExtraProps","models","props","$omitFromDatabaseJson","extras","isArray","reduce","val","key","joinTableCol","columnNameToPropertyName","aliasCol"],"mappings":"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","file":"ManyToManyRelation.js","sourcesContent":["import _ from 'lodash';\nimport Relation from '../Relation';\nimport inheritModel from '../../model/inheritModel';\nimport {isSqlite} from '../../utils/dbUtils';\nimport memoize from '../../utils/decorators/memoize';\n\nimport ManyToManyFindOperation from './ManyToManyFindOperation';\nimport ManyToManyInsertOperation from './ManyToManyInsertOperation';\nimport ManyToManyRelateOperation from './ManyToManyRelateOperation';\nimport ManyToManyUnrelateOperation from './ManyToManyUnrelateOperation';\nimport ManyToManyUnrelateSqliteOperation from './ManyToManyUnrelateSqliteOperation';\nimport ManyToManyUpdateOperation from './ManyToManyUpdateOperation';\nimport ManyToManyUpdateSqliteOperation from './ManyToManyUpdateSqliteOperation';\nimport ManyToManyDeleteOperation from './ManyToManyDeleteOperation';\nimport ManyToManyDeleteSqliteOperation from './ManyToManyDeleteSqliteOperation';\n\nconst sqliteBuiltInRowId = '_rowid_';\n\nexport default class ManyToManyRelation extends Relation {\n\n  setMapping(mapping) {\n    let retVal = super.setMapping(mapping);\n\n    // Avoid require loop and import here.\n    let Model = require(__dirname + '/../../model/Model').default;\n\n    if (!_.isObject(mapping.join.through)) {\n      this.throwError('join must have the `through` that describes the join table.');\n    }\n\n    if (!mapping.join.through.from || !mapping.join.through.to) {\n      this.throwError('join.through must be an object that describes the join table. For example: {from: \"JoinTable.someId\", to: \"JoinTable.someOtherId\"}');\n    }\n\n    let joinFrom = this.parseReference(mapping.join.from);\n    let joinTableFrom = this.parseReference(mapping.join.through.from);\n    let joinTableTo = this.parseReference(mapping.join.through.to);\n    let joinTableExtra = mapping.join.through.extra || [];\n\n    if (!joinTableFrom.table || _.isEmpty(joinTableFrom.columns)) {\n      this.throwError('join.through.from must have format JoinTable.columnName. For example \"JoinTable.someId\" or in case of composite key [\"JoinTable.a\", \"JoinTable.b\"].');\n    }\n\n    if (!joinTableTo.table || _.isEmpty(joinTableTo.columns)) {\n      this.throwError('join.through.to must have format JoinTable.columnName. For example \"JoinTable.someId\" or in case of composite key [\"JoinTable.a\", \"JoinTable.b\"].');\n    }\n\n    if (joinTableFrom.table !== joinTableTo.table) {\n      this.throwError('join.through `from` and `to` must point to the same join table.');\n    }\n\n    this.joinTable = joinTableFrom.table;\n\n    if (joinFrom.table === this.ownerModelClass.tableName) {\n      this.joinTableOwnerCol = joinTableFrom.columns;\n      this.joinTableRelatedCol = joinTableTo.columns;\n    } else {\n      this.joinTableRelatedCol = joinTableFrom.columns;\n      this.joinTableOwnerCol = joinTableTo.columns;\n    }\n\n    if (mapping.join.through.modelClass) {\n      this._joinTableModelClass = this.resolveModel(Model, mapping.join.through.modelClass, 'join.through.modelClass');\n    } else {\n      this._joinTableModelClass = inheritModel(Model);\n      this._joinTableModelClass.tableName = this.joinTable;\n      // We cannot know if the join table has a primary key. Therefore we set some\n      // known column as the idColumn so that inserts will work.\n      this._joinTableModelClass.idColumn = this.joinTableRelatedCol;\n    }\n\n    this.joinTableOwnerProp = this.propertyName(this.joinTableOwnerCol, this._joinTableModelClass);\n    this.joinTableRelatedProp = this.propertyName(this.joinTableRelatedCol, this._joinTableModelClass);\n    this.joinTableExtras = this.parseExtras(joinTableExtra);\n\n    return retVal;\n  }\n\n  /**\n   * @returns {Array.<string>}\n   */\n  @memoize\n  fullJoinTableOwnerCol() {\n    return this.joinTableOwnerCol.map(col => `${this.joinTable}.${col}`);\n  }\n\n  /**\n   * @returns {Array.<string>}\n   */\n  @memoize\n  fullJoinTableRelatedCol() {\n    return this.joinTableRelatedCol.map(col => `${this.joinTable}.${col}`);\n  }\n\n  /**\n   * @returns {string}\n   */\n  joinTableAlias() {\n    return this.joinTable + '_rel_' + this.name;\n  }\n\n  /**\n   * @returns {ManyToManyRelation}\n   */\n  bindKnex(knex) {\n    let bound = super.bindKnex(knex);\n    bound._joinTableModelClass = this._joinTableModelClass.bindKnex(knex);\n    return bound;\n  }\n\n  /**\n   * @returns {QueryBuilder}\n   */\n  findQuery(builder, opt) {\n    builder.join(this.joinTable, join => {\n      const fullRelatedCol = this.fullRelatedCol();\n      const fullJoinTableRelatedCol = this.fullJoinTableRelatedCol();\n\n      for (let i = 0, l = fullJoinTableRelatedCol.length; i < l; ++i) {\n        join.on(fullJoinTableRelatedCol[i], fullRelatedCol[i]);\n      }\n    });\n\n    if (opt.isColumnRef) {\n      const fullJoinTableOwnerCol = this.fullJoinTableOwnerCol();\n\n      for (let i = 0, l = fullJoinTableOwnerCol.length; i < l; ++i) {\n        builder.whereRef(fullJoinTableOwnerCol[i], opt.ownerIds[i]);\n      }\n    } else {\n      let hasIds = false;\n\n      for (let i = 0, l = opt.ownerIds.length; i < l; ++i) {\n        const id = opt.ownerIds[i];\n\n        if (id) {\n          hasIds = true;\n          break;\n        }\n      }\n\n      if (hasIds) {\n        builder.whereInComposite(this.fullJoinTableOwnerCol(), opt.ownerIds);\n      } else {\n        builder.resolve([]);\n      }\n    }\n\n    return builder.modify(this.modify);\n  }\n\n  /**\n   * @returns {QueryBuilder}\n   */\n  join(builder, opt) {\n    opt = opt || {};\n\n    opt.joinOperation = opt.joinOperation || 'join';\n    opt.relatedTableAlias = opt.relatedTableAlias || this.relatedTableAlias();\n    opt.relatedJoinSelectQuery = opt.relatedJoinSelectQuery || this.relatedModelClass.query().childQueryOf(builder);\n    opt.relatedTable = opt.relatedTable || this.relatedModelClass.tableName;\n    opt.ownerTable = opt.ownerTable || this.ownerModelClass.tableName;\n    opt.joinTableAlias = opt.joinTableAlias || `${opt.relatedTableAlias}_join`;\n\n    const joinTableAsAlias = `${this.joinTable} as ${opt.joinTableAlias}`;\n    const joinTableOwnerCol = this.joinTableOwnerCol.map(col => `${opt.joinTableAlias}.${col}`);\n    const joinTableRelatedCol = this.joinTableRelatedCol.map(col => `${opt.joinTableAlias}.${col}`);\n\n    const relatedCol = this.relatedCol.map(col => `${opt.relatedTableAlias}.${col}`);\n    const ownerCol = this.ownerCol.map(col => `${opt.ownerTable}.${col}`);\n\n    let relatedJoinSelect = opt.relatedJoinSelectQuery\n      .modify(this.modify)\n      .as(opt.relatedTableAlias);\n\n    if (relatedJoinSelect.isSelectAll()) {\n      // No need to join a subquery if the query is `select * from \"RelatedTable\"`.\n      relatedJoinSelect = `${this.relatedModelClass.tableName} as ${opt.relatedTableAlias}`\n    }\n\n    return builder\n      [opt.joinOperation](joinTableAsAlias, join => {\n        for (let i = 0, l = joinTableOwnerCol.length; i < l; ++i) {\n          join.on(joinTableOwnerCol[i], ownerCol[i]);\n        }\n      })\n      [opt.joinOperation](relatedJoinSelect, join => {\n        for (let i = 0, l = joinTableRelatedCol.length; i < l; ++i) {\n          join.on(joinTableRelatedCol[i], relatedCol[i]);\n        }\n      });\n  }\n\n  find(builder, owners) {\n    return new ManyToManyFindOperation('find', {\n      relation: this,\n      owners: owners\n    });\n  }\n\n  insert(builder, owner) {\n    return new ManyToManyInsertOperation('insert', {\n      relation: this,\n      owner: owner\n    });\n  }\n\n  update(builder, owner) {\n    if (isSqlite(builder.knex())) {\n      return new ManyToManyUpdateSqliteOperation('update', {\n        relation: this,\n        owner: owner\n      });\n    } else {\n      return new ManyToManyUpdateOperation('update', {\n        relation: this,\n        owner: owner\n      });\n    }\n  }\n\n  patch(builder, owner) {\n    if (isSqlite(builder.knex())) {\n      return new ManyToManyUpdateSqliteOperation('patch', {\n        relation: this,\n        owner: owner,\n        modelOptions: {patch: true}\n      });\n    } else {\n      return new ManyToManyUpdateOperation('patch', {\n        relation: this,\n        owner: owner,\n        modelOptions: {patch: true}\n      });\n    }\n  }\n\n  delete(builder, owner) {\n    if (isSqlite(builder.knex())) {\n      return new ManyToManyDeleteSqliteOperation('delete', {\n        relation: this,\n        owner: owner\n      });\n    } else {\n      return new ManyToManyDeleteOperation('delete', {\n        relation: this,\n        owner: owner\n      });\n    }\n  }\n\n  relate(builder, owner) {\n    return new ManyToManyRelateOperation('relate', {\n      relation: this,\n      owner: owner\n    });\n  }\n\n  unrelate(builder, owner) {\n    if (isSqlite(builder.knex())) {\n      return new ManyToManyUnrelateSqliteOperation('unrelate', {\n        relation: this,\n        owner: owner\n      });\n    } else {\n      return new ManyToManyUnrelateOperation('unrelate', {\n        relation: this,\n        owner: owner\n      });\n    }\n  }\n\n  selectForModify(builder, owner) {\n    let ownerId = owner.$values(this.ownerProp);\n\n    let idQuery = this.joinTableModelClass(builder.knex())\n      .query()\n      .childQueryOf(builder)\n      .select(this.fullJoinTableRelatedCol())\n      .whereComposite(this.fullJoinTableOwnerCol(), ownerId);\n\n    return builder.whereInComposite(this.fullRelatedCol(), idQuery);\n  }\n\n  selectForModifySqlite(builder, owner) {\n    const relatedTable = this.relatedModelClass.tableName;\n    const relatedTableAlias = this.relatedTableAlias();\n    const relatedTableAsAlias = relatedTable + ' as ' + relatedTableAlias;\n    const relatedTableAliasRowId = relatedTableAlias + '.' + sqliteBuiltInRowId;\n    const relatedTableRowId = relatedTable + '.' + sqliteBuiltInRowId;\n\n    const selectRelatedQuery = this.joinTableModelClass(builder.knex())\n      .query()\n      .childQueryOf(builder)\n      .select(relatedTableAliasRowId)\n      .whereComposite(this.fullJoinTableOwnerCol(), owner.$values(this.ownerProp))\n      .join(relatedTableAsAlias, join => {\n        const fullJoinTableRelatedCols = this.fullJoinTableRelatedCol();\n        const fullRelatedCol = this.fullRelatedCol();\n\n        for (let i = 0, l = fullJoinTableRelatedCols.length; i < l; ++i) {\n          join.on(fullJoinTableRelatedCols[i], fullRelatedCol[i]);\n        }\n      });\n\n    return builder.whereInComposite(relatedTableRowId, selectRelatedQuery);\n  }\n\n  createJoinModels(ownerId, related) {\n    const joinModels = new Array(related.length);\n\n    for (let i = 0, lr = related.length; i < lr; ++i) {\n      const rel = related[i];\n      let joinModel = {};\n\n      for (let j = 0, lp = this.joinTableOwnerProp.length; j < lp; ++j) {\n        joinModel[this.joinTableOwnerProp[j]] = ownerId[j];\n      }\n\n      for (let j = 0, lp = this.joinTableRelatedProp.length; j < lp; ++j) {\n        joinModel[this.joinTableRelatedProp[j]] = rel[this.relatedProp[j]];\n      }\n\n      for (let j = 0, lp = this.joinTableExtras.length; j < lp; ++j) {\n        const extra = this.joinTableExtras[j];\n        const extraValue = rel[extra.aliasProp];\n\n        if (!_.isUndefined(extraValue)) {\n          joinModel[extra.joinTableProp] = extraValue;\n        }\n      }\n\n      joinModels[i] = joinModel;\n    }\n\n    return joinModels;\n  }\n\n  omitExtraProps(models) {\n    if (!_.isEmpty(this.joinTableExtras)) {\n      const props = this.joinTableExtras.map(extra => extra.aliasProp);\n\n      for (let i = 0, l = models.length; i < l; ++i) {\n        models[i].$omitFromDatabaseJson(props);\n      }\n    }\n  }\n\n  /**\n   * @protected\n   */\n  parseExtras(extras) {\n    if (Array.isArray(extras)) {\n      extras = extras.reduce((extras, col) => {\n        extras[col] = col;\n        return extras;\n      }, {});\n    }\n\n    return Object.keys(extras).map(key => {\n      const val = extras[key];\n\n      return {\n        joinTableCol: val,\n        joinTableProp: this._joinTableModelClass.columnNameToPropertyName(val),\n        aliasCol: key,\n        aliasProp: this._joinTableModelClass.columnNameToPropertyName(key)\n      };\n    });\n  }\n}\n"]}