-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.ts
executable file
·131 lines (120 loc) · 3.67 KB
/
main.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
#!/usr/bin/env -S node --disable-warning=ExperimentalWarning --disable-warning=MODULE_TYPELESS_PACKAGE_JSON --experimental-strip-types
import axios from "axios";
import auth from "basic-auth";
import fastify, { type FastifyReply } from "fastify";
import fastifyRawBody from "fastify-raw-body";
import Mustache from "mustache";
import { createHmac } from "node:crypto";
import type FromSchema from "./json-schema-to-ts";
const app = fastify();
const slackWebhook = process.env["SLACK_WEBHOOK"];
const hubSecret = process.env["SHA1_HUB_SECRET"];
const basicUser = process.env["BASIC_USERNAME"];
const basicPassword = process.env["BASIC_PASSWORD"];
const format =
process.env["MESSAGE_FORMAT"] ??
"Zammad ticket {{ticket.id}}, state {{ticket.state}} from a user at {{customerDomain}}";
app.get("/", (_req, res) => {
res.code(200).send("Nothing to see here, move on");
});
const sendToSlack = async (message: string) => {
try {
if (slackWebhook) {
await axios.post(slackWebhook, { text: message });
}
} catch (e) {
console.error(e);
}
};
const getSenderDomain = (
replyTo: string | undefined,
customerEmail: string | undefined,
): string => {
const replyToEmail = replyTo ? /<([^>]+)>/i.exec(replyTo)?.[1] : undefined;
const domain = (replyToEmail ?? customerEmail)?.split("@")?.[1] ?? "??";
return domain;
};
const end = (res: FastifyReply, error?: string): void => {
if (error) {
console.error(error);
}
res.status(200);
};
const body = {
type: "object",
properties: {
article: {
type: "object",
properties: { sender: { type: "string" }, reply_to: { type: "string" } },
},
ticket: {
type: "object",
properties: {
customer: { type: "object", properties: { email: { type: "string" } } },
},
},
},
required: ["article"],
} as const;
type Body = FromSchema<typeof body>;
app.register(fastifyRawBody).then(() => {
app.post<{ Body: Body }>(
"/zammad",
{ schema: { body }, attachValidation: true },
async (req, res) => {
if (req.validationError !== undefined) {
console.error(req.validationError);
return end(res);
}
if (hubSecret) {
if (req.rawBody === undefined) {
return end(
res,
"Unable to check signature when rawBody is undefined",
);
}
const hmac = createHmac("sha1", hubSecret);
hmac.update(req.rawBody);
const token = `sha1=${hmac.digest("hex")}`;
if (req.headers["x-hub-signature"] !== token) {
return end(
res,
`Received signature ${req.headers["x-hub-signature"]} does not match calculated signature ${token}`,
);
}
}
if (basicUser && basicPassword) {
const credentials = auth(req);
if (
credentials?.name !== basicUser ||
credentials.pass !== basicPassword
) {
return end(
res,
"Received username and password did not match those in the environment",
);
}
}
const body = req.body;
// Only send triggers to Slack if the action is not taken by an Agent.
if (body.article.sender !== "Agent") {
const customerDomain = getSenderDomain(
body.article.reply_to,
body.ticket?.customer?.email,
);
const message = Mustache.render(format, { ...body, customerDomain });
await sendToSlack(message);
}
return end(res);
},
);
});
const HOST = "::";
const PORT = 8000;
app.listen({ port: PORT, host: HOST }, function (err, address) {
if (err) {
console.error(err);
process.exit(1);
}
console.info(`Web server started on ${address}`);
});