const ghostBookshelf = require('./base'); const uuid = require('uuid'); const _ = require('lodash'); const {sequence} = require('@tryghost/promise'); const config = require('../../shared/config'); const crypto = require('crypto'); const Member = ghostBookshelf.Model.extend({ tableName: 'members', defaults() { return { subscribed: true, uuid: uuid.v4(), email_count: 0, email_opened_count: 0 }; }, relationships: ['labels', 'stripeCustomers', 'email_recipients'], // do not delete email_recipients records when a member is destroyed. Recipient // records are used for analytics and historical records relationshipConfig: { email_recipients: { destroyRelated: false } }, relationshipBelongsTo: { labels: 'labels', stripeCustomers: 'members_stripe_customers', email_recipients: 'email_recipients' }, labels: function labels() { return this.belongsToMany('Label', 'members_labels', 'member_id', 'label_id') .withPivot('sort_order') .query('orderBy', 'sort_order', 'ASC') .query((qb) => { // avoids bookshelf adding a `DISTINCT` to the query // we know the result set will already be unique and DISTINCT hurts query performance qb.columns('labels.*'); }); }, stripeCustomers() { return this.hasMany('MemberStripeCustomer', 'member_id', 'id'); }, stripeSubscriptions() { return this.belongsToMany( 'StripeCustomerSubscription', 'members_stripe_customers', 'member_id', 'customer_id', 'id', 'customer_id' ); }, email_recipients() { return this.hasMany('EmailRecipient', 'member_id', 'id'); }, serialize(options) { const defaultSerializedObject = ghostBookshelf.Model.prototype.serialize.call(this, options); if (defaultSerializedObject.stripeSubscriptions) { defaultSerializedObject.stripe = { subscriptions: defaultSerializedObject.stripeSubscriptions }; delete defaultSerializedObject.stripeSubscriptions; } return defaultSerializedObject; }, emitChange: function emitChange(event, options) { const eventToTrigger = 'member' + '.' + event; ghostBookshelf.Model.prototype.emitChange.bind(this)(this, eventToTrigger, options); }, onCreated: function onCreated(model, attrs, options) { ghostBookshelf.Model.prototype.onCreated.apply(this, arguments); model.emitChange('added', options); }, onUpdated: function onUpdated(model, attrs, options) { ghostBookshelf.Model.prototype.onUpdated.apply(this, arguments); model.emitChange('edited', options); }, onDestroyed: function onDestroyed(model, options) { ghostBookshelf.Model.prototype.onDestroyed.apply(this, arguments); model.emitChange('deleted', options); }, onDestroying: function onDestroyed(model) { ghostBookshelf.Model.prototype.onDestroying.apply(this, arguments); this.handleAttachedModels(model); }, onSaving: function onSaving(model, attr, options) { let labelsToSave = []; let ops = []; // CASE: detect lowercase/uppercase label slugs if (!_.isUndefined(this.get('labels')) && !_.isNull(this.get('labels'))) { labelsToSave = []; // and deduplicate upper/lowercase tags _.each(this.get('labels'), function each(item) { item.name = item.name && item.name.trim(); for (let i = 0; i < labelsToSave.length; i = i + 1) { if (labelsToSave[i].name && item.name && labelsToSave[i].name.toLocaleLowerCase() === item.name.toLocaleLowerCase()) { return; } } labelsToSave.push(item); }); this.set('labels', labelsToSave); } // CASE: Detect existing labels with same case-insensitive name and replace ops.push(function updateLabels() { return ghostBookshelf.model('Label') .findAll(Object.assign({ columns: ['id', 'name'] }, _.pick(options, 'transacting'))) .then((labels) => { labelsToSave.forEach((label) => { let existingLabel = labels.find((lab) => { return label.name.toLowerCase() === lab.get('name').toLowerCase(); }); label.name = (existingLabel && existingLabel.get('name')) || label.name; }); model.set('labels', labelsToSave); }); }); this.handleAttachedModels(model); return sequence(ops); }, handleAttachedModels: function handleAttachedModels(model) { /** * @NOTE: * Bookshelf only exposes the object that is being detached on `detaching`. * For the reason above, `detached` handler is using the scope of `detaching` * to access the models that are not present in `detached`. */ model.related('labels').once('detaching', function onDetaching(collection, label) { model.related('labels').once('detached', function onDetached(detachedCollection, response, options) { label.emitChange('detached', options); model.emitChange('label.detached', options); }); }); model.related('labels').once('attaching', function onDetaching(collection, labels) { model.related('labels').once('attached', function onDetached(detachedCollection, response, options) { labels.forEach((label) => { label.emitChange('attached', options); model.emitChange('label.attached', options); }); }); }); }, /** * The base model keeps only the columns, which are defined in the schema. * We have to add the relations on top, otherwise bookshelf-relations * has no access to the nested relations, which should be updated. */ permittedAttributes: function permittedAttributes() { let filteredKeys = ghostBookshelf.Model.prototype.permittedAttributes.apply(this, arguments); this.relationships.forEach((key) => { filteredKeys.push(key); }); return filteredKeys; }, /** * We have to ensure consistency. If you listen on model events (e.g. `member.added`), you can expect that you always * receive all fields including relations. Otherwise you can't rely on a consistent flow. And we want to avoid * that event listeners have to re-fetch a resource. This function is used in the context of inserting * and updating resources. We won't return the relations by default for now. */ defaultRelations: function defaultRelations(methodName, options) { if (['edit', 'add', 'destroy'].indexOf(methodName) !== -1) { options.withRelated = _.union(['labels'], options.withRelated || []); } return options; }, searchQuery: function searchQuery(queryBuilder, query) { queryBuilder.where('members.name', 'like', `%${query}%`); queryBuilder.orWhere('members.email', 'like', `%${query}%`); }, // TODO: hacky way to filter by members with an active subscription, // replace with a proper way to do this via filter param. // NOTE: assumes members will have a single subscription customQuery: function customQuery(queryBuilder, options) { if (options.paid === true) { queryBuilder.innerJoin( 'members_stripe_customers', 'members.id', 'members_stripe_customers.member_id' ); queryBuilder.innerJoin( 'members_stripe_customers_subscriptions', function () { this.on( 'members_stripe_customers.customer_id', 'members_stripe_customers_subscriptions.customer_id' ).onIn( 'members_stripe_customers_subscriptions.status', ['active', 'trialing', 'past_due', 'unpaid'] ); } ); } if (options.paid === false) { queryBuilder.leftJoin( 'members_stripe_customers', 'members.id', 'members_stripe_customers.member_id' ); queryBuilder.whereNull('members_stripe_customers.member_id'); } }, orderRawQuery(field, direction) { if (field === 'email_open_rate') { return { orderByRaw: `members.email_open_rate IS NOT NULL DESC, members.email_open_rate ${direction}` }; } }, toJSON(unfilteredOptions) { const options = Member.filterOptions(unfilteredOptions, 'toJSON'); const attrs = ghostBookshelf.Model.prototype.toJSON.call(this, options); // Inject a computed avatar url. Uses gravatar's default ?d= query param // to serve a blank image if there is no gravatar for the member's email. // Will not use gravatar if privacy.useGravatar is false in config attrs.avatar_image = null; if (attrs.email && !config.isPrivacyDisabled('useGravatar')) { const emailHash = crypto.createHash('md5').update(attrs.email.toLowerCase().trim()).digest('hex'); attrs.avatar_image = `https://gravatar.com/avatar/${emailHash}?s=250&d=blank`; } return attrs; } }, { /** * Returns an array of keys permitted in a method's `options` hash, depending on the current method. * @param {String} methodName The name of the method to check valid options for. * @return {Array} Keys allowed in the `options` hash of the model's method. */ permittedOptions: function permittedOptions(methodName) { let options = ghostBookshelf.Model.permittedOptions.call(this, methodName); if (['findPage', 'findAll'].includes(methodName)) { // TODO: remove 'paid' once it's possible to use in a filter options = options.concat(['search', 'paid']); } return options; }, add(data, unfilteredOptions = {}) { if (!unfilteredOptions.transacting) { return ghostBookshelf.transaction((transacting) => { return this.add(data, Object.assign({transacting}, unfilteredOptions)); }); } return ghostBookshelf.Model.add.call(this, data, unfilteredOptions); }, edit(data, unfilteredOptions = {}) { if (!unfilteredOptions.transacting) { return ghostBookshelf.transaction((transacting) => { return this.edit(data, Object.assign({transacting}, unfilteredOptions)); }); } return ghostBookshelf.Model.edit.call(this, data, unfilteredOptions); }, destroy(unfilteredOptions = {}) { if (!unfilteredOptions.transacting) { return ghostBookshelf.transaction((transacting) => { return this.destroy(Object.assign({transacting}, unfilteredOptions)); }); } return ghostBookshelf.Model.destroy.call(this, unfilteredOptions); } }); const Members = ghostBookshelf.Collection.extend({ model: Member }); module.exports = { Member: ghostBookshelf.model('Member', Member), Members: ghostBookshelf.collection('Members', Members) };