mirror of
https://github.com/outline/outline.git
synced 2026-06-13 11:25:03 +03:00
4dbad4e46c
* wip
* stash
* fix: make authenticationId nullable fk
* fix: apply generics to resolve compile time type errors
* fix: loosen integration settings
* chore: refactor into functional component
* feat: pass integrations all the way to embeds
* perf: avoid re-fetching integrations
* fix: change attr name to avoid type overlap
* feat: use hostname from embed settings in matcher
* Revert "feat: use hostname from embed settings in matcher"
This reverts commit e7485d9cda.
* feat: refactor into a class
* chore: refactor url regex formation as a util
* fix: escape regex special chars
* fix: remove in-house escapeRegExp in favor of lodash's
* fix: sanitize url
* perf: memoize embeds
* fix: rename hostname to url and allow spreading entire settings instead of just url
* fix: replace diagrams with drawio
* fix: rename
* fix: support self-hosted and saas both
* fix: assert on settings url
* fix: move embed integrations loading to hook
* fix: address review comments
* fix: use observer in favor of explicit state setters
* fix: refactor useEmbedIntegrations into useEmbeds
* fix: use translations for toasts
Co-authored-by: Tom Moor <tom.moor@gmail.com>
133 lines
3.5 KiB
TypeScript
133 lines
3.5 KiB
TypeScript
import fetch from "fetch-with-proxy";
|
|
import { Op } from "sequelize";
|
|
import { IntegrationType } from "@shared/types";
|
|
import env from "@server/env";
|
|
import { Document, Integration, Collection, Team } from "@server/models";
|
|
import { presentSlackAttachment } from "@server/presenters";
|
|
import {
|
|
DocumentEvent,
|
|
IntegrationEvent,
|
|
RevisionEvent,
|
|
Event,
|
|
} from "@server/types";
|
|
import BaseProcessor from "./BaseProcessor";
|
|
|
|
export default class SlackProcessor extends BaseProcessor {
|
|
static applicableEvents: Event["name"][] = [
|
|
"documents.publish",
|
|
"revisions.create",
|
|
"integrations.create",
|
|
];
|
|
|
|
async perform(event: Event) {
|
|
switch (event.name) {
|
|
case "documents.publish":
|
|
case "revisions.create":
|
|
return this.documentUpdated(event);
|
|
|
|
case "integrations.create":
|
|
return this.integrationCreated(event);
|
|
|
|
default:
|
|
}
|
|
}
|
|
|
|
async integrationCreated(event: IntegrationEvent) {
|
|
const integration = (await Integration.findOne({
|
|
where: {
|
|
id: event.modelId,
|
|
service: "slack",
|
|
type: "post",
|
|
},
|
|
include: [
|
|
{
|
|
model: Collection,
|
|
required: true,
|
|
as: "collection",
|
|
},
|
|
],
|
|
})) as Integration<IntegrationType.Post>;
|
|
if (!integration) {
|
|
return;
|
|
}
|
|
|
|
const collection = integration.collection;
|
|
if (!collection) {
|
|
return;
|
|
}
|
|
|
|
await fetch(integration.settings.url, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify({
|
|
text: `👋 Hey there! When documents are published or updated in the *${collection.name}* collection on Outline they will be posted to this channel!`,
|
|
attachments: [
|
|
{
|
|
color: collection.color,
|
|
title: collection.name,
|
|
title_link: `${env.URL}${collection.url}`,
|
|
text: collection.description,
|
|
},
|
|
],
|
|
}),
|
|
});
|
|
}
|
|
|
|
async documentUpdated(event: DocumentEvent | RevisionEvent) {
|
|
// never send notifications when batch importing documents
|
|
// @ts-expect-error ts-migrate(2339) FIXME: Property 'data' does not exist on type 'DocumentEv... Remove this comment to see the full error message
|
|
if (event.data && event.data.source === "import") {
|
|
return;
|
|
}
|
|
const [document, team] = await Promise.all([
|
|
Document.findByPk(event.documentId),
|
|
Team.findByPk(event.teamId),
|
|
]);
|
|
if (!document || !team) {
|
|
return;
|
|
}
|
|
|
|
// never send notifications for draft documents
|
|
if (!document.publishedAt) {
|
|
return;
|
|
}
|
|
|
|
const integration = (await Integration.findOne({
|
|
where: {
|
|
teamId: document.teamId,
|
|
collectionId: document.collectionId,
|
|
service: "slack",
|
|
type: "post",
|
|
events: {
|
|
[Op.contains]: [
|
|
event.name === "revisions.create" ? "documents.update" : event.name,
|
|
],
|
|
},
|
|
},
|
|
})) as Integration<IntegrationType.Post>;
|
|
if (!integration) {
|
|
return;
|
|
}
|
|
let text = `${document.updatedBy.name} updated a document`;
|
|
|
|
if (event.name === "documents.publish") {
|
|
text = `${document.createdBy.name} published a new document`;
|
|
}
|
|
|
|
await fetch(integration.settings.url, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify({
|
|
text,
|
|
attachments: [
|
|
presentSlackAttachment(document, document.collection, team),
|
|
],
|
|
}),
|
|
});
|
|
}
|
|
}
|