2018-12-10 04:20:54 -05:00
|
|
|
const ghostBookshelf = require('./base');
|
2019-11-05 05:07:30 -05:00
|
|
|
const uuid = require('uuid');
|
2020-02-14 04:33:10 -05:00
|
|
|
const _ = require('lodash');
|
|
|
|
const sequence = require('../lib/promise/sequence');
|
2020-05-27 12:47:53 -05:00
|
|
|
const config = require('../../shared/config');
|
2020-02-19 05:08:12 -05:00
|
|
|
const crypto = require('crypto');
|
2018-12-10 04:20:54 -05:00
|
|
|
|
|
|
|
const Member = ghostBookshelf.Model.extend({
|
2019-10-11 02:15:39 -05:00
|
|
|
tableName: 'members',
|
|
|
|
|
2019-11-05 05:07:30 -05:00
|
|
|
defaults() {
|
|
|
|
return {
|
|
|
|
subscribed: true,
|
|
|
|
uuid: uuid.v4()
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2020-02-14 04:33:10 -05:00
|
|
|
relationships: ['labels'],
|
|
|
|
|
|
|
|
relationshipBelongsTo: {
|
|
|
|
labels: 'labels'
|
|
|
|
},
|
|
|
|
|
|
|
|
labels: function labels() {
|
|
|
|
return this.belongsToMany('Label', 'members_labels', 'member_id', 'label_id')
|
|
|
|
.withPivot('sort_order')
|
|
|
|
.query('orderBy', 'sort_order', 'ASC');
|
|
|
|
},
|
|
|
|
|
2019-10-11 02:15:39 -05:00
|
|
|
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);
|
|
|
|
},
|
|
|
|
|
2019-10-31 09:34:50 -05:00
|
|
|
onUpdated: function onUpdated(model, attrs, options) {
|
|
|
|
ghostBookshelf.Model.prototype.onUpdated.apply(this, arguments);
|
|
|
|
|
|
|
|
model.emitChange('edited', options);
|
|
|
|
},
|
|
|
|
|
2019-10-11 02:15:39 -05:00
|
|
|
onDestroyed: function onDestroyed(model, options) {
|
|
|
|
ghostBookshelf.Model.prototype.onDestroyed.apply(this, arguments);
|
|
|
|
|
|
|
|
model.emitChange('deleted', options);
|
2020-02-14 04:33:10 -05:00
|
|
|
},
|
2020-02-19 05:08:12 -05:00
|
|
|
|
2020-02-14 04:33:10 -05:00
|
|
|
onDestroying: function onDestroyed(model) {
|
|
|
|
ghostBookshelf.Model.prototype.onDestroying.apply(this, arguments);
|
|
|
|
|
|
|
|
this.handleAttachedModels(model);
|
|
|
|
},
|
2020-02-19 05:08:12 -05:00
|
|
|
|
2020-02-14 04:33:10 -05:00
|
|
|
onSaving: function onSaving(model, attr, options) {
|
2020-02-14 05:36:25 -05:00
|
|
|
let labelsToSave = [];
|
2020-02-14 04:33:10 -05:00
|
|
|
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 onDetached(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 onDetached(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;
|
2020-02-19 05:08:12 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
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;
|
2019-10-11 02:15:39 -05:00
|
|
|
}
|
2018-12-10 04:20:54 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
const Members = ghostBookshelf.Collection.extend({
|
|
|
|
model: Member
|
|
|
|
});
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
Member: ghostBookshelf.model('Member', Member),
|
|
|
|
Members: ghostBookshelf.collection('Members', Members)
|
|
|
|
};
|