Files
speckle-server/packages/server/modules/webhooks/services/webhooks.js
T
Gergő Jedlicska 3d6653f73b hotfix/2.9.2 (#1175)
* Update to new specklepy (#1173)

* Publish images for all branches but limit tagging
* only tag 'latest' and '2' when 'SHOULD_PUBLISH' variable is 'true'

* Publishing helm chart should check for `SHOULD_PUBLISH`

* Move blocking step to publish-helm chart, and allow images to be published

* Pin python requirements and bump to latest versions

* Fix EOL whitespace

* use valid version for psycopg2-binary (the clue is in the 2!)

* fix(fileimports): add exception printing to file imports

* fix(fileimports): bump specklepy version

move to a specklepy version that contains a fix for send without writing to disk

Co-authored-by: Iain Sproat <68657+iainsproat@users.noreply.github.com>

* Fixes liveness and readiness checks to prevent CSRF error message (#1169)

- provides content-type header
- check that status code is 200

* Fixes broken helm template by adding quotation marks around liveness probe command (#1171)

* fix(server activities): make sure the stream events are properly dispatched

* feat(server webhooks): add scheduled orphaned webhook cleanup

* test(server webhooks): add test to webhook cleanup service

* feat(server webhooks): drop foreign key reference for webhooks schema to streams

* refactor(server req context): refactor req context to have the ip attribute for all requests

* feat(server objects rest api): add ratelimits to objects rest api endpoints

* fix(server rest api): properly handle returning 419

Co-authored-by: Iain Sproat <68657+iainsproat@users.noreply.github.com>
2022-11-09 13:23:32 +01:00

153 lines
4.4 KiB
JavaScript

'use strict'
const knex = require('@/db/knex')
const { getStream } = require('@/modules/core/repositories/streams')
const crs = require('crypto-random-string')
const WebhooksConfig = () => knex('webhooks_config')
const WebhooksEvents = () => knex('webhooks_events')
const Users = () => knex('users')
const { getServerInfo } = require('../../core/services/generic')
const MAX_STREAM_WEBHOOKS = 100
module.exports = {
async createWebhook({ streamId, url, description, secret, enabled, triggers }) {
const streamWebhookCount = await module.exports.getStreamWebhooksCount({ streamId })
if (streamWebhookCount >= MAX_STREAM_WEBHOOKS) {
throw new Error(
`Maximum number of webhooks for a stream reached (${MAX_STREAM_WEBHOOKS})`
)
}
const triggersObj = Object.assign({}, ...triggers.map((x) => ({ [x]: true })))
const [{ id }] = await WebhooksConfig()
.returning('id')
.insert({
id: crs({ length: 10 }),
streamId,
url,
description,
secret,
enabled,
triggers: triggersObj
})
return id
},
async getWebhook({ id }) {
const webhook = await WebhooksConfig().select('*').where({ id }).first()
if (webhook) {
webhook.triggers = Object.keys(webhook.triggers)
}
return webhook
},
async updateWebhook({ id, url, description, secret, enabled, triggers }) {
const fieldsToUpdate = {}
if (url !== undefined) fieldsToUpdate.url = url
if (description !== undefined) fieldsToUpdate.description = description
if (secret !== undefined) fieldsToUpdate.secret = secret
if (enabled !== undefined) fieldsToUpdate.enabled = enabled
if (triggers !== undefined) {
const triggersObj = Object.assign({}, ...triggers.map((x) => ({ [x]: true })))
fieldsToUpdate.triggers = triggersObj
}
const [{ id: res }] = await WebhooksConfig()
.returning('id')
.where({ id })
.update(fieldsToUpdate)
return res
},
async deleteWebhook({ id }) {
return await WebhooksConfig().where({ id }).del()
},
async getStreamWebhooks({ streamId }) {
const webhooks = await WebhooksConfig().select('*').where({ streamId })
for (const webhook of webhooks) {
webhook.triggers = Object.keys(webhook.triggers)
}
return webhooks
},
async getStreamWebhooksCount({ streamId }) {
const [res] = await WebhooksConfig().count().where({ streamId })
return parseInt(res.count)
},
async dispatchStreamEvent({ streamId, event, eventPayload }) {
// Add server info
eventPayload.server = await getServerInfo()
eventPayload.server.canonicalUrl = process.env.CANONICAL_URL
delete eventPayload.server.id
// Add stream info
if (eventPayload.streamId) {
eventPayload.stream = await getStream({
streamId: eventPayload.streamId,
userId: eventPayload.userId
})
}
// Add user info (except email and pwd)
if (eventPayload.userId) {
eventPayload.user = await Users()
.where({ id: eventPayload.userId })
.select('*')
.first()
if (eventPayload.user) {
delete eventPayload.user.passwordDigest
delete eventPayload.user.email
}
}
// with this select, we must have the streamid available on the webhook config,
// even when the stream is deleted, to dispatch the stream deleted webhook events
const { rows } = await knex.raw(
`
SELECT * FROM webhooks_config WHERE "streamId" = ?
`,
[streamId]
)
for (const wh of rows) {
if (!wh.enabled) continue
if (!(event in wh.triggers)) continue
// Add webhook info (the key `webhook` will be replaced for each webhook configured, before serializing the payload and storing it)
eventPayload.webhook = wh
eventPayload.webhook.triggers = Object.keys(eventPayload.webhook.triggers)
delete eventPayload.webhook.secret
await WebhooksEvents().insert({
id: crs({ length: 20 }),
webhookId: wh.id,
payload: JSON.stringify(eventPayload)
})
}
},
async getLastWebhookEvents({ webhookId, limit }) {
if (!limit) {
limit = 100
}
return await WebhooksEvents()
.select('*')
.where({ webhookId })
.orderBy('lastUpdate', 'desc')
.limit(limit)
},
async getWebhookEventsCount({ webhookId }) {
const [res] = await WebhooksEvents().count().where({ webhookId })
return parseInt(res.count)
}
}