init source
This commit is contained in:
+280
@@ -0,0 +1,280 @@
|
||||
'use strict';
|
||||
|
||||
const Utils = require('./../utils');
|
||||
const Helpers = require('./helpers');
|
||||
const _ = require('lodash');
|
||||
const Association = require('./base');
|
||||
const Op = require('../operators');
|
||||
|
||||
/**
|
||||
* One-to-one association
|
||||
*
|
||||
* In the API reference below, add the name of the association to the method, e.g. for `User.hasOne(Project)` the getter will be `user.getProject()`.
|
||||
* This is almost the same as `belongsTo` with one exception - The foreign key will be defined on the target model.
|
||||
*
|
||||
* @see {@link Model.hasOne}
|
||||
*/
|
||||
class HasOne extends Association {
|
||||
constructor(source, target, options) {
|
||||
super(source, target, options);
|
||||
|
||||
this.associationType = 'HasOne';
|
||||
this.isSingleAssociation = true;
|
||||
this.foreignKeyAttribute = {};
|
||||
|
||||
if (this.as) {
|
||||
this.isAliased = true;
|
||||
this.options.name = {
|
||||
singular: this.as
|
||||
};
|
||||
} else {
|
||||
this.as = this.target.options.name.singular;
|
||||
this.options.name = this.target.options.name;
|
||||
}
|
||||
|
||||
if (_.isObject(this.options.foreignKey)) {
|
||||
this.foreignKeyAttribute = this.options.foreignKey;
|
||||
this.foreignKey = this.foreignKeyAttribute.name || this.foreignKeyAttribute.fieldName;
|
||||
} else if (this.options.foreignKey) {
|
||||
this.foreignKey = this.options.foreignKey;
|
||||
}
|
||||
|
||||
if (!this.foreignKey) {
|
||||
this.foreignKey = Utils.camelize(
|
||||
[
|
||||
Utils.singularize(this.options.as || this.source.name),
|
||||
this.source.primaryKeyAttribute
|
||||
].join('_')
|
||||
);
|
||||
}
|
||||
|
||||
if (
|
||||
this.options.sourceKey
|
||||
&& !this.source.rawAttributes[this.options.sourceKey]
|
||||
) {
|
||||
throw new Error(`Unknown attribute "${this.options.sourceKey}" passed as sourceKey, define this attribute on model "${this.source.name}" first`);
|
||||
}
|
||||
|
||||
this.sourceKey = this.sourceKeyAttribute = this.options.sourceKey || this.source.primaryKeyAttribute;
|
||||
this.sourceKeyField = this.source.rawAttributes[this.sourceKey].field || this.sourceKey;
|
||||
this.sourceKeyIsPrimary = this.sourceKey === this.source.primaryKeyAttribute;
|
||||
|
||||
this.associationAccessor = this.as;
|
||||
this.options.useHooks = options.useHooks;
|
||||
|
||||
if (this.target.rawAttributes[this.foreignKey]) {
|
||||
this.identifierField = this.target.rawAttributes[this.foreignKey].field || this.foreignKey;
|
||||
}
|
||||
|
||||
// Get singular name, trying to uppercase the first letter, unless the model forbids it
|
||||
const singular = _.upperFirst(this.options.name.singular);
|
||||
|
||||
this.accessors = {
|
||||
get: `get${singular}`,
|
||||
set: `set${singular}`,
|
||||
create: `create${singular}`
|
||||
};
|
||||
}
|
||||
|
||||
// the id is in the target table
|
||||
_injectAttributes() {
|
||||
const newAttributes = {};
|
||||
|
||||
newAttributes[this.foreignKey] = _.defaults({}, this.foreignKeyAttribute, {
|
||||
type: this.options.keyType || this.source.rawAttributes[this.sourceKey].type,
|
||||
allowNull: true
|
||||
});
|
||||
|
||||
if (this.options.constraints !== false) {
|
||||
const target = this.target.rawAttributes[this.foreignKey] || newAttributes[this.foreignKey];
|
||||
this.options.onDelete = this.options.onDelete || (target.allowNull ? 'SET NULL' : 'CASCADE');
|
||||
this.options.onUpdate = this.options.onUpdate || 'CASCADE';
|
||||
}
|
||||
|
||||
Helpers.addForeignKeyConstraints(newAttributes[this.foreignKey], this.source, this.target, this.options, this.sourceKeyField);
|
||||
Utils.mergeDefaults(this.target.rawAttributes, newAttributes);
|
||||
|
||||
this.target.refreshAttributes();
|
||||
|
||||
this.identifierField = this.target.rawAttributes[this.foreignKey].field || this.foreignKey;
|
||||
|
||||
Helpers.checkNamingCollision(this);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
mixin(obj) {
|
||||
const methods = ['get', 'set', 'create'];
|
||||
|
||||
Helpers.mixinMethods(this, obj, methods);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the associated instance.
|
||||
*
|
||||
* @param {Model|Array<Model>} instances source instances
|
||||
* @param {Object} [options] find options
|
||||
* @param {string|boolean} [options.scope] Apply a scope on the related model, or remove its default scope by passing false
|
||||
* @param {string} [options.schema] Apply a schema on the related model
|
||||
*
|
||||
* @see
|
||||
* {@link Model.findOne} for a full explanation of options
|
||||
*
|
||||
* @returns {Promise<Model>}
|
||||
*/
|
||||
get(instances, options) {
|
||||
const where = {};
|
||||
|
||||
let Target = this.target;
|
||||
let instance;
|
||||
|
||||
options = Utils.cloneDeep(options);
|
||||
|
||||
if (Object.prototype.hasOwnProperty.call(options, 'scope')) {
|
||||
if (!options.scope) {
|
||||
Target = Target.unscoped();
|
||||
} else {
|
||||
Target = Target.scope(options.scope);
|
||||
}
|
||||
}
|
||||
|
||||
if (Object.prototype.hasOwnProperty.call(options, 'schema')) {
|
||||
Target = Target.schema(options.schema, options.schemaDelimiter);
|
||||
}
|
||||
|
||||
if (!Array.isArray(instances)) {
|
||||
instance = instances;
|
||||
instances = undefined;
|
||||
}
|
||||
|
||||
if (instances) {
|
||||
where[this.foreignKey] = {
|
||||
[Op.in]: instances.map(instance => instance.get(this.sourceKey))
|
||||
};
|
||||
} else {
|
||||
where[this.foreignKey] = instance.get(this.sourceKey);
|
||||
}
|
||||
|
||||
if (this.scope) {
|
||||
Object.assign(where, this.scope);
|
||||
}
|
||||
|
||||
options.where = options.where ?
|
||||
{ [Op.and]: [where, options.where] } :
|
||||
where;
|
||||
|
||||
if (instances) {
|
||||
return Target.findAll(options).then(results => {
|
||||
const result = {};
|
||||
for (const instance of instances) {
|
||||
result[instance.get(this.sourceKey, { raw: true })] = null;
|
||||
}
|
||||
|
||||
for (const instance of results) {
|
||||
result[instance.get(this.foreignKey, { raw: true })] = instance;
|
||||
}
|
||||
|
||||
return result;
|
||||
});
|
||||
}
|
||||
|
||||
return Target.findOne(options);
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the associated model.
|
||||
*
|
||||
* @param {Model} sourceInstance the source instance
|
||||
* @param {?<Model>|string|number} [associatedInstance] An persisted instance or the primary key of an instance to associate with this. Pass `null` or `undefined` to remove the association.
|
||||
* @param {Object} [options] Options passed to getAssociation and `target.save`
|
||||
*
|
||||
* @returns {Promise}
|
||||
*/
|
||||
set(sourceInstance, associatedInstance, options) {
|
||||
let alreadyAssociated;
|
||||
|
||||
options = Object.assign({}, options, {
|
||||
scope: false
|
||||
});
|
||||
|
||||
return sourceInstance[this.accessors.get](options).then(oldInstance => {
|
||||
// TODO Use equals method once #5605 is resolved
|
||||
alreadyAssociated = oldInstance && associatedInstance && this.target.primaryKeyAttributes.every(attribute =>
|
||||
oldInstance.get(attribute, { raw: true }) === (associatedInstance.get ? associatedInstance.get(attribute, { raw: true }) : associatedInstance)
|
||||
);
|
||||
|
||||
if (oldInstance && !alreadyAssociated) {
|
||||
oldInstance[this.foreignKey] = null;
|
||||
return oldInstance.save(Object.assign({}, options, {
|
||||
fields: [this.foreignKey],
|
||||
allowNull: [this.foreignKey],
|
||||
association: true
|
||||
}));
|
||||
}
|
||||
}).then(() => {
|
||||
if (associatedInstance && !alreadyAssociated) {
|
||||
if (!(associatedInstance instanceof this.target)) {
|
||||
const tmpInstance = {};
|
||||
tmpInstance[this.target.primaryKeyAttribute] = associatedInstance;
|
||||
associatedInstance = this.target.build(tmpInstance, {
|
||||
isNewRecord: false
|
||||
});
|
||||
}
|
||||
|
||||
Object.assign(associatedInstance, this.scope);
|
||||
associatedInstance.set(this.foreignKey, sourceInstance.get(this.sourceKeyAttribute));
|
||||
|
||||
return associatedInstance.save(options);
|
||||
}
|
||||
|
||||
return null;
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new instance of the associated model and associate it with this.
|
||||
*
|
||||
* @param {Model} sourceInstance the source instance
|
||||
* @param {Object} [values={}] values to create associated model instance with
|
||||
* @param {Object} [options] Options passed to `target.create` and setAssociation.
|
||||
*
|
||||
* @see
|
||||
* {@link Model#create} for a full explanation of options
|
||||
*
|
||||
* @returns {Promise<Model>} The created target model
|
||||
*/
|
||||
create(sourceInstance, values, options) {
|
||||
values = values || {};
|
||||
options = options || {};
|
||||
|
||||
if (this.scope) {
|
||||
for (const attribute of Object.keys(this.scope)) {
|
||||
values[attribute] = this.scope[attribute];
|
||||
if (options.fields) {
|
||||
options.fields.push(attribute);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
values[this.foreignKey] = sourceInstance.get(this.sourceKeyAttribute);
|
||||
if (options.fields) {
|
||||
options.fields.push(this.foreignKey);
|
||||
}
|
||||
|
||||
return this.target.create(values, options);
|
||||
}
|
||||
|
||||
verifyAssociationAlias(alias) {
|
||||
if (typeof alias === 'string') {
|
||||
return this.as === alias;
|
||||
}
|
||||
|
||||
if (alias && alias.singular) {
|
||||
return this.as === alias.singular;
|
||||
}
|
||||
|
||||
return !this.isAliased;
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = HasOne;
|
||||
Reference in New Issue
Block a user