'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 = { [this.foreignKey]: { type: this.options.keyType || this.source.rawAttributes[this.sourceKey].type, allowNull: true, ...this.foreignKeyAttribute } }; 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} 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} */ async 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) { const results = await Target.findAll(options); 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 {?|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} */ async set(sourceInstance, associatedInstance, options) { options = { ...options, scope: false }; const oldInstance = await sourceInstance[this.accessors.get](options); // TODO Use equals method once #5605 is resolved const 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; await oldInstance.save({ ...options, fields: [this.foreignKey], allowNull: [this.foreignKey], association: true }); } 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} The created target model */ async 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 await 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;