Files
habitica/website/server/libs/webhook.js
2019-10-10 20:11:50 +02:00

159 lines
3.4 KiB
JavaScript

import got from 'got';
import { isURL } from 'validator';
import nconf from 'nconf';
import logger from './logger';
import { // eslint-disable-line import/no-cycle
model as User,
} from '../models/user';
const IS_PRODUCTION = nconf.get('IS_PROD');
function sendWebhook (url, body) {
got.post(url, {
body,
json: true,
}).catch(err => logger.error(err));
}
function isValidWebhook (hook) {
return hook.enabled && isURL(hook.url, {
require_tld: !!IS_PRODUCTION, // eslint-disable-line camelcase
});
}
export class WebhookSender {
constructor (options = {}) {
this.type = options.type;
this.transformData = options.transformData || WebhookSender.defaultTransformData;
this.webhookFilter = options.webhookFilter || WebhookSender.defaultWebhookFilter;
}
static defaultTransformData (data) {
return data;
}
static defaultWebhookFilter () {
return true;
}
attachDefaultData (user, body) {
body.webhookType = this.type;
body.user = body.user || {};
body.user._id = user._id;
}
send (user, data) {
const { webhooks } = user;
const hooks = webhooks.filter(hook => {
if (!isValidWebhook(hook)) return false;
if (hook.type === 'globalActivity') return true;
return this.type === hook.type && this.webhookFilter(hook, data);
});
if (hooks.length < 1) {
return; // prevents running the body creation code if there are no webhooks to send
}
const body = this.transformData(data);
this.attachDefaultData(user, body);
hooks.forEach(hook => {
sendWebhook(hook.url, body);
});
}
}
export const taskScoredWebhook = new WebhookSender({
type: 'taskActivity',
webhookFilter (hook) {
const scored = hook.options && hook.options.scored;
return scored;
},
transformData (data) {
const {
user, task, direction, delta,
} = data;
const extendedStats = User.addComputedStatsToJSONObj(user.stats.toJSON(), user);
const userData = {
// _id: user._id, added automatically when the webhook is sent
_tmp: user._tmp,
stats: extendedStats,
};
const dataToSend = {
type: 'scored',
direction,
delta,
task,
user: userData,
};
return dataToSend;
},
});
export const taskActivityWebhook = new WebhookSender({
type: 'taskActivity',
webhookFilter (hook, data) {
const { type } = data;
return hook.options[type];
},
});
export const userActivityWebhook = new WebhookSender({
type: 'userActivity',
webhookFilter (hook, data) {
const { type } = data;
return hook.options[type];
},
});
export const questActivityWebhook = new WebhookSender({
type: 'questActivity',
webhookFilter (hook, data) {
const { type } = data;
return hook.options[type];
},
transformData (data) {
const { group, quest, type } = data;
const dataToSend = {
type,
group: {
id: group.id,
name: group.name,
},
quest: {
key: quest.key,
},
};
return dataToSend;
},
});
export const groupChatReceivedWebhook = new WebhookSender({
type: 'groupChatReceived',
webhookFilter (hook, data) {
return hook.options.groupId === data.group.id;
},
transformData (data) {
const { group, chat } = data;
const dataToSend = {
group: {
id: group.id,
name: group.name,
},
chat,
};
return dataToSend;
},
});