@action-llama/action-llama 0.18.3 → 0.18.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/build-info.json +1 -1
- package/dist/cli/commands/add.d.ts +2 -2
- package/dist/cli/commands/add.d.ts.map +1 -1
- package/dist/cli/commands/add.js +13 -4
- package/dist/cli/commands/add.js.map +1 -1
- package/dist/cli/commands/agent.d.ts.map +1 -1
- package/dist/cli/commands/agent.js +156 -33
- package/dist/cli/commands/agent.js.map +1 -1
- package/dist/cli/commands/doctor.d.ts.map +1 -1
- package/dist/cli/commands/doctor.js +55 -42
- package/dist/cli/commands/doctor.js.map +1 -1
- package/dist/cli/commands/env.d.ts.map +1 -1
- package/dist/cli/commands/env.js +13 -3
- package/dist/cli/commands/env.js.map +1 -1
- package/dist/cli/commands/new.js +2 -2
- package/dist/cli/commands/update.js +43 -6
- package/dist/cli/commands/update.js.map +1 -1
- package/dist/cli/main.js +1 -1
- package/dist/cloud/vps/hetzner-api.d.ts +2 -0
- package/dist/cloud/vps/hetzner-api.d.ts.map +1 -1
- package/dist/cloud/vps/hetzner-api.js +7 -0
- package/dist/cloud/vps/hetzner-api.js.map +1 -1
- package/dist/cloud/vps/provision.d.ts +1 -1
- package/dist/cloud/vps/provision.d.ts.map +1 -1
- package/dist/cloud/vps/provision.js +8 -7
- package/dist/cloud/vps/provision.js.map +1 -1
- package/dist/cloud/vps/teardown.d.ts.map +1 -1
- package/dist/cloud/vps/teardown.js +35 -3
- package/dist/cloud/vps/teardown.js.map +1 -1
- package/dist/cloud/vps/vultr-api.d.ts +2 -0
- package/dist/cloud/vps/vultr-api.d.ts.map +1 -1
- package/dist/cloud/vps/vultr-api.js +7 -0
- package/dist/cloud/vps/vultr-api.js.map +1 -1
- package/dist/control/routes/stats.d.ts.map +1 -1
- package/dist/control/routes/stats.js +13 -0
- package/dist/control/routes/stats.js.map +1 -1
- package/dist/credentials/builtins/discord-bot.d.ts +4 -0
- package/dist/credentials/builtins/discord-bot.d.ts.map +1 -0
- package/dist/credentials/builtins/discord-bot.js +34 -0
- package/dist/credentials/builtins/discord-bot.js.map +1 -0
- package/dist/credentials/builtins/index.d.ts.map +1 -1
- package/dist/credentials/builtins/index.js +8 -0
- package/dist/credentials/builtins/index.js.map +1 -1
- package/dist/credentials/builtins/slack-bot-token.d.ts +4 -0
- package/dist/credentials/builtins/slack-bot-token.d.ts.map +1 -0
- package/dist/credentials/builtins/slack-bot-token.js +28 -0
- package/dist/credentials/builtins/slack-bot-token.js.map +1 -0
- package/dist/credentials/builtins/slack-signing-secret.d.ts +4 -0
- package/dist/credentials/builtins/slack-signing-secret.d.ts.map +1 -0
- package/dist/credentials/builtins/slack-signing-secret.js +19 -0
- package/dist/credentials/builtins/slack-signing-secret.js.map +1 -0
- package/dist/credentials/builtins/x-twitter-webhook-secret.d.ts +4 -0
- package/dist/credentials/builtins/x-twitter-webhook-secret.d.ts.map +1 -0
- package/dist/credentials/builtins/x-twitter-webhook-secret.js +12 -0
- package/dist/credentials/builtins/x-twitter-webhook-secret.js.map +1 -0
- package/dist/events/routes/webhooks.d.ts.map +1 -1
- package/dist/events/routes/webhooks.js +57 -0
- package/dist/events/routes/webhooks.js.map +1 -1
- package/dist/events/webhook-setup.d.ts +1 -0
- package/dist/events/webhook-setup.d.ts.map +1 -1
- package/dist/events/webhook-setup.js +42 -2
- package/dist/events/webhook-setup.js.map +1 -1
- package/dist/extensions/loader.d.ts.map +1 -1
- package/dist/extensions/loader.js +2 -1
- package/dist/extensions/loader.js.map +1 -1
- package/dist/frontend/assets/index-D5QqK4BL.js +13 -0
- package/dist/frontend/assets/{index-B3IlFzCF.css → index-GJZXHhf1.css} +1 -1
- package/dist/frontend/index.html +2 -2
- package/dist/setup/scaffold.d.ts.map +1 -1
- package/dist/setup/scaffold.js +1 -25
- package/dist/setup/scaffold.js.map +1 -1
- package/dist/shared/environment.d.ts +6 -0
- package/dist/shared/environment.d.ts.map +1 -1
- package/dist/shared/environment.js +15 -0
- package/dist/shared/environment.js.map +1 -1
- package/dist/stats/store.d.ts +1 -0
- package/dist/stats/store.d.ts.map +1 -1
- package/dist/stats/store.js +8 -4
- package/dist/stats/store.js.map +1 -1
- package/dist/webhooks/definitions/discord.d.ts +3 -0
- package/dist/webhooks/definitions/discord.d.ts.map +1 -0
- package/dist/webhooks/definitions/discord.js +23 -0
- package/dist/webhooks/definitions/discord.js.map +1 -0
- package/dist/webhooks/definitions/registry.d.ts.map +1 -1
- package/dist/webhooks/definitions/registry.js +4 -1
- package/dist/webhooks/definitions/registry.js.map +1 -1
- package/dist/webhooks/definitions/slack.d.ts +3 -0
- package/dist/webhooks/definitions/slack.d.ts.map +1 -0
- package/dist/webhooks/definitions/slack.js +25 -0
- package/dist/webhooks/definitions/slack.js.map +1 -0
- package/dist/webhooks/definitions/twitter.d.ts +3 -0
- package/dist/webhooks/definitions/twitter.d.ts.map +1 -0
- package/dist/webhooks/definitions/twitter.js +30 -0
- package/dist/webhooks/definitions/twitter.js.map +1 -0
- package/dist/webhooks/providers/discord.d.ts +9 -0
- package/dist/webhooks/providers/discord.d.ts.map +1 -0
- package/dist/webhooks/providers/discord.js +112 -0
- package/dist/webhooks/providers/discord.js.map +1 -0
- package/dist/webhooks/providers/index.d.ts +12 -0
- package/dist/webhooks/providers/index.d.ts.map +1 -1
- package/dist/webhooks/providers/index.js +94 -0
- package/dist/webhooks/providers/index.js.map +1 -1
- package/dist/webhooks/providers/slack.d.ts +9 -0
- package/dist/webhooks/providers/slack.d.ts.map +1 -0
- package/dist/webhooks/providers/slack.js +120 -0
- package/dist/webhooks/providers/slack.js.map +1 -0
- package/dist/webhooks/providers/twitter.d.ts +13 -0
- package/dist/webhooks/providers/twitter.d.ts.map +1 -0
- package/dist/webhooks/providers/twitter.js +188 -0
- package/dist/webhooks/providers/twitter.js.map +1 -0
- package/dist/webhooks/types.d.ts +24 -1
- package/dist/webhooks/types.d.ts.map +1 -1
- package/dist/webhooks/validation.d.ts +11 -0
- package/dist/webhooks/validation.d.ts.map +1 -1
- package/dist/webhooks/validation.js +45 -1
- package/dist/webhooks/validation.js.map +1 -1
- package/package.json +1 -1
- package/dist/frontend/assets/index-BotTEisS.js +0 -13
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { truncateEventText as truncate, validateEd25519Signature } from "../validation.js";
|
|
2
|
+
// Discord interaction types
|
|
3
|
+
const INTERACTION_TYPE_PING = 1;
|
|
4
|
+
const INTERACTION_TYPE_APPLICATION_COMMAND = 2;
|
|
5
|
+
const INTERACTION_TYPE_MESSAGE_COMPONENT = 3;
|
|
6
|
+
const INTERACTION_TYPE_AUTOCOMPLETE = 4;
|
|
7
|
+
const INTERACTION_TYPE_MODAL_SUBMIT = 5;
|
|
8
|
+
const INTERACTION_TYPE_NAMES = {
|
|
9
|
+
[INTERACTION_TYPE_APPLICATION_COMMAND]: "application_command",
|
|
10
|
+
[INTERACTION_TYPE_MESSAGE_COMPONENT]: "message_component",
|
|
11
|
+
[INTERACTION_TYPE_AUTOCOMPLETE]: "autocomplete",
|
|
12
|
+
[INTERACTION_TYPE_MODAL_SUBMIT]: "modal_submit",
|
|
13
|
+
};
|
|
14
|
+
export class DiscordWebhookProvider {
|
|
15
|
+
source = "discord";
|
|
16
|
+
getDeliveryId(headers) {
|
|
17
|
+
return headers["x-interaction-id"] ?? null;
|
|
18
|
+
}
|
|
19
|
+
validateRequest(headers, rawBody, secrets, allowUnsigned) {
|
|
20
|
+
return validateEd25519Signature(rawBody, headers["x-signature-timestamp"], headers["x-signature-ed25519"], secrets, allowUnsigned);
|
|
21
|
+
}
|
|
22
|
+
parseEvent(headers, body) {
|
|
23
|
+
if (!body || typeof body !== "object")
|
|
24
|
+
return null;
|
|
25
|
+
const interactionType = body.type;
|
|
26
|
+
if (!interactionType || interactionType === INTERACTION_TYPE_PING)
|
|
27
|
+
return null;
|
|
28
|
+
const event = INTERACTION_TYPE_NAMES[interactionType];
|
|
29
|
+
if (!event)
|
|
30
|
+
return null;
|
|
31
|
+
const data = body.data;
|
|
32
|
+
const user = body.member?.user || body.user;
|
|
33
|
+
const guildId = body.guild_id || "";
|
|
34
|
+
const channelId = body.channel_id || "";
|
|
35
|
+
const base = {
|
|
36
|
+
source: "discord",
|
|
37
|
+
event,
|
|
38
|
+
repo: guildId,
|
|
39
|
+
branch: channelId,
|
|
40
|
+
sender: user?.username || "unknown",
|
|
41
|
+
timestamp: new Date().toISOString(),
|
|
42
|
+
};
|
|
43
|
+
switch (interactionType) {
|
|
44
|
+
case INTERACTION_TYPE_APPLICATION_COMMAND: {
|
|
45
|
+
const commandName = data?.name || "unknown";
|
|
46
|
+
const options = data?.options;
|
|
47
|
+
let bodyText;
|
|
48
|
+
if (options?.length) {
|
|
49
|
+
bodyText = options.map((o) => `${o.name}: ${o.value}`).join(", ");
|
|
50
|
+
}
|
|
51
|
+
return {
|
|
52
|
+
...base,
|
|
53
|
+
action: commandName,
|
|
54
|
+
title: commandName,
|
|
55
|
+
body: truncate(bodyText),
|
|
56
|
+
number: undefined,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
case INTERACTION_TYPE_MESSAGE_COMPONENT: {
|
|
60
|
+
return {
|
|
61
|
+
...base,
|
|
62
|
+
action: String(data?.component_type || "unknown"),
|
|
63
|
+
title: data?.custom_id || "component",
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
case INTERACTION_TYPE_AUTOCOMPLETE: {
|
|
67
|
+
const commandName = data?.name || "unknown";
|
|
68
|
+
return {
|
|
69
|
+
...base,
|
|
70
|
+
action: commandName,
|
|
71
|
+
title: commandName,
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
case INTERACTION_TYPE_MODAL_SUBMIT: {
|
|
75
|
+
return {
|
|
76
|
+
...base,
|
|
77
|
+
action: data?.custom_id || "modal",
|
|
78
|
+
title: data?.custom_id || "modal",
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
default:
|
|
82
|
+
return {
|
|
83
|
+
...base,
|
|
84
|
+
title: event,
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
matchesFilter(context, filter) {
|
|
89
|
+
const f = filter;
|
|
90
|
+
if (f.events?.length && !f.events.includes(context.event)) {
|
|
91
|
+
return false;
|
|
92
|
+
}
|
|
93
|
+
if (f.guilds?.length && !f.guilds.includes(context.repo)) {
|
|
94
|
+
return false;
|
|
95
|
+
}
|
|
96
|
+
// channel_id is stored in context.branch
|
|
97
|
+
if (f.channels?.length && context.branch && !f.channels.includes(context.branch)) {
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
// commands filter: match against title (command name) for command-type events
|
|
101
|
+
if (f.commands?.length) {
|
|
102
|
+
if (context.event === "application_command" || context.event === "autocomplete") {
|
|
103
|
+
if (!context.title || !f.commands.includes(context.title)) {
|
|
104
|
+
return false;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
// For non-command events (components, modals), commands filter does not apply — pass through
|
|
108
|
+
}
|
|
109
|
+
return true;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=discord.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"discord.js","sourceRoot":"","sources":["../../../src/webhooks/providers/discord.ts"],"names":[],"mappings":"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"}
|
|
@@ -18,8 +18,20 @@ export declare const mintlifyWebhookExtension: WebhookExtension;
|
|
|
18
18
|
* Sentry webhook provider extension
|
|
19
19
|
*/
|
|
20
20
|
export declare const sentryWebhookExtension: WebhookExtension;
|
|
21
|
+
/**
|
|
22
|
+
* Slack webhook provider extension
|
|
23
|
+
*/
|
|
24
|
+
export declare const slackWebhookExtension: WebhookExtension;
|
|
21
25
|
/**
|
|
22
26
|
* Test webhook provider extension
|
|
23
27
|
*/
|
|
24
28
|
export declare const testWebhookExtension: WebhookExtension;
|
|
29
|
+
/**
|
|
30
|
+
* Discord webhook provider extension
|
|
31
|
+
*/
|
|
32
|
+
export declare const discordWebhookExtension: WebhookExtension;
|
|
33
|
+
/**
|
|
34
|
+
* Twitter webhook provider extension
|
|
35
|
+
*/
|
|
36
|
+
export declare const twitterWebhookExtension: WebhookExtension;
|
|
25
37
|
//# sourceMappingURL=index.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/webhooks/providers/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/webhooks/providers/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAUlE;;GAEG;AACH,eAAO,MAAM,sBAAsB,EAAE,gBAyBpC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,sBAAsB,EAAE,gBAyBpC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,wBAAwB,EAAE,gBAyBtC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,sBAAsB,EAAE,gBAyBpC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,qBAAqB,EAAE,gBAyBnC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,oBAAoB,EAAE,gBAelC,CAAC;AACF;;GAEG;AACH,eAAO,MAAM,uBAAuB,EAAE,gBA6BrC,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,uBAAuB,EAAE,gBAyBrC,CAAC"}
|
|
@@ -5,7 +5,10 @@ import { GitHubWebhookProvider } from "./github.js";
|
|
|
5
5
|
import { LinearWebhookProvider } from "./linear.js";
|
|
6
6
|
import { MintlifyWebhookProvider } from "./mintlify.js";
|
|
7
7
|
import { SentryWebhookProvider } from "./sentry.js";
|
|
8
|
+
import { DiscordWebhookProvider } from "./discord.js";
|
|
9
|
+
import { SlackWebhookProvider } from "./slack.js";
|
|
8
10
|
import { TestWebhookProvider } from "./test.js";
|
|
11
|
+
import { TwitterWebhookProvider } from "./twitter.js";
|
|
9
12
|
/**
|
|
10
13
|
* GitHub webhook provider extension
|
|
11
14
|
*/
|
|
@@ -122,6 +125,35 @@ export const sentryWebhookExtension = {
|
|
|
122
125
|
// No cleanup required
|
|
123
126
|
}
|
|
124
127
|
};
|
|
128
|
+
/**
|
|
129
|
+
* Slack webhook provider extension
|
|
130
|
+
*/
|
|
131
|
+
export const slackWebhookExtension = {
|
|
132
|
+
metadata: {
|
|
133
|
+
name: "slack",
|
|
134
|
+
version: "1.0.0",
|
|
135
|
+
description: "Slack Events API webhook provider",
|
|
136
|
+
type: "webhook",
|
|
137
|
+
requiredCredentials: [
|
|
138
|
+
{ type: "slack_signing_secret", description: "Slack signing secret for request verification", optional: true }
|
|
139
|
+
],
|
|
140
|
+
providesCredentialTypes: [
|
|
141
|
+
{
|
|
142
|
+
type: "slack_signing_secret",
|
|
143
|
+
fields: ["secret"],
|
|
144
|
+
description: "Slack signing secret",
|
|
145
|
+
envMapping: { secret: "SLACK_SIGNING_SECRET" }
|
|
146
|
+
}
|
|
147
|
+
]
|
|
148
|
+
},
|
|
149
|
+
provider: new SlackWebhookProvider(),
|
|
150
|
+
async init() {
|
|
151
|
+
// No special initialization required
|
|
152
|
+
},
|
|
153
|
+
async shutdown() {
|
|
154
|
+
// No cleanup required
|
|
155
|
+
}
|
|
156
|
+
};
|
|
125
157
|
/**
|
|
126
158
|
* Test webhook provider extension
|
|
127
159
|
*/
|
|
@@ -141,4 +173,66 @@ export const testWebhookExtension = {
|
|
|
141
173
|
// No cleanup required
|
|
142
174
|
}
|
|
143
175
|
};
|
|
176
|
+
/**
|
|
177
|
+
* Discord webhook provider extension
|
|
178
|
+
*/
|
|
179
|
+
export const discordWebhookExtension = {
|
|
180
|
+
metadata: {
|
|
181
|
+
name: "discord",
|
|
182
|
+
version: "1.0.0",
|
|
183
|
+
description: "Discord webhook provider (Interactions Endpoint)",
|
|
184
|
+
type: "webhook",
|
|
185
|
+
requiredCredentials: [
|
|
186
|
+
{ type: "discord_bot", description: "Discord bot credentials for Ed25519 signature validation", optional: true }
|
|
187
|
+
],
|
|
188
|
+
providesCredentialTypes: [
|
|
189
|
+
{
|
|
190
|
+
type: "discord_bot",
|
|
191
|
+
fields: ["application_id", "public_key", "bot_token"],
|
|
192
|
+
description: "Discord bot credentials",
|
|
193
|
+
envMapping: {
|
|
194
|
+
application_id: "DISCORD_APPLICATION_ID",
|
|
195
|
+
public_key: "DISCORD_PUBLIC_KEY",
|
|
196
|
+
bot_token: "DISCORD_BOT_TOKEN",
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
]
|
|
200
|
+
},
|
|
201
|
+
provider: new DiscordWebhookProvider(),
|
|
202
|
+
async init() {
|
|
203
|
+
// No special initialization required
|
|
204
|
+
},
|
|
205
|
+
async shutdown() {
|
|
206
|
+
// No cleanup required
|
|
207
|
+
}
|
|
208
|
+
};
|
|
209
|
+
/**
|
|
210
|
+
* Twitter webhook provider extension
|
|
211
|
+
*/
|
|
212
|
+
export const twitterWebhookExtension = {
|
|
213
|
+
metadata: {
|
|
214
|
+
name: "twitter",
|
|
215
|
+
version: "1.0.0",
|
|
216
|
+
description: "X (Twitter) webhook provider with CRC support",
|
|
217
|
+
type: "webhook",
|
|
218
|
+
requiredCredentials: [
|
|
219
|
+
{ type: "x_twitter_webhook_secret", description: "Consumer secret for CRC handshake and HMAC validation", optional: true }
|
|
220
|
+
],
|
|
221
|
+
providesCredentialTypes: [
|
|
222
|
+
{
|
|
223
|
+
type: "x_twitter_webhook_secret",
|
|
224
|
+
fields: ["secret"],
|
|
225
|
+
description: "X (Twitter) consumer secret",
|
|
226
|
+
envMapping: { secret: "X_TWITTER_WEBHOOK_SECRET" }
|
|
227
|
+
}
|
|
228
|
+
]
|
|
229
|
+
},
|
|
230
|
+
provider: new TwitterWebhookProvider(),
|
|
231
|
+
async init() {
|
|
232
|
+
// No special initialization required
|
|
233
|
+
},
|
|
234
|
+
async shutdown() {
|
|
235
|
+
// No cleanup required
|
|
236
|
+
}
|
|
237
|
+
};
|
|
144
238
|
//# sourceMappingURL=index.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/webhooks/providers/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,EAAE,qBAAqB,EAAE,MAAM,aAAa,CAAC;AACpD,OAAO,EAAE,qBAAqB,EAAE,MAAM,aAAa,CAAC;AACpD,OAAO,EAAE,uBAAuB,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,qBAAqB,EAAE,MAAM,aAAa,CAAC;AACpD,OAAO,EAAE,mBAAmB,EAAE,MAAM,WAAW,CAAC;
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/webhooks/providers/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import type { WebhookProvider, WebhookContext, WebhookFilter } from "../types.js";
|
|
2
|
+
export declare class SlackWebhookProvider implements WebhookProvider {
|
|
3
|
+
source: string;
|
|
4
|
+
validateRequest(headers: Record<string, string | undefined>, rawBody: string, secrets?: Record<string, string>, allowUnsigned?: boolean): string | null;
|
|
5
|
+
parseEvent(headers: Record<string, string | undefined>, body: any): WebhookContext | null;
|
|
6
|
+
matchesFilter(context: WebhookContext, filter: WebhookFilter): boolean;
|
|
7
|
+
handleChallenge(headers: Record<string, string | undefined>, rawBody: string, secrets?: Record<string, string>, allowUnsigned?: boolean): object | null;
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=slack.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../../src/webhooks/providers/slack.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,eAAe,EAAE,cAAc,EAAE,aAAa,EAAsB,MAAM,aAAa,CAAC;AAKtG,qBAAa,oBAAqB,YAAW,eAAe;IAC1D,MAAM,SAAW;IAEjB,eAAe,CACb,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,EAC3C,OAAO,EAAE,MAAM,EACf,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EAChC,aAAa,CAAC,EAAE,OAAO,GACtB,MAAM,GAAG,IAAI;IAgChB,UAAU,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,EAAE,IAAI,EAAE,GAAG,GAAG,cAAc,GAAG,IAAI;IAyDzF,aAAa,CAAC,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,aAAa,GAAG,OAAO;IAwBtE,eAAe,CACb,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,EAC3C,OAAO,EAAE,MAAM,EACf,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EAChC,aAAa,CAAC,EAAE,OAAO,GACtB,MAAM,GAAG,IAAI;CAoBjB"}
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import { createHmac, timingSafeEqual } from "crypto";
|
|
2
|
+
import { truncateEventText as truncate } from "../validation.js";
|
|
3
|
+
const MAX_TIMESTAMP_AGE_SECONDS = 5 * 60; // 5 minutes
|
|
4
|
+
export class SlackWebhookProvider {
|
|
5
|
+
source = "slack";
|
|
6
|
+
validateRequest(headers, rawBody, secrets, allowUnsigned) {
|
|
7
|
+
// If no secrets configured, check allowUnsigned policy
|
|
8
|
+
if (!secrets || Object.keys(secrets).length === 0) {
|
|
9
|
+
return allowUnsigned ? "_unsigned" : null;
|
|
10
|
+
}
|
|
11
|
+
const timestamp = headers["x-slack-request-timestamp"];
|
|
12
|
+
const signature = headers["x-slack-signature"];
|
|
13
|
+
if (!timestamp || !signature)
|
|
14
|
+
return null;
|
|
15
|
+
// Replay protection: reject if timestamp is older than 5 minutes
|
|
16
|
+
const now = Math.floor(Date.now() / 1000);
|
|
17
|
+
if (Math.abs(now - parseInt(timestamp, 10)) > MAX_TIMESTAMP_AGE_SECONDS) {
|
|
18
|
+
return null;
|
|
19
|
+
}
|
|
20
|
+
const signingBase = `v0:${timestamp}:${rawBody}`;
|
|
21
|
+
for (const [instanceName, secret] of Object.entries(secrets)) {
|
|
22
|
+
const expected = "v0=" + createHmac("sha256", secret).update(signingBase).digest("hex");
|
|
23
|
+
if (signature.length === expected.length &&
|
|
24
|
+
timingSafeEqual(Buffer.from(signature), Buffer.from(expected))) {
|
|
25
|
+
return instanceName;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
parseEvent(headers, body) {
|
|
31
|
+
// URL verification challenges are handled by handleChallenge, not dispatched
|
|
32
|
+
if (body.type === "url_verification") {
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
if (body.type !== "event_callback") {
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
const event = body.event;
|
|
39
|
+
if (!event)
|
|
40
|
+
return null;
|
|
41
|
+
const eventType = event.type;
|
|
42
|
+
const teamId = body.team_id;
|
|
43
|
+
const sender = event.user || event.bot_id || "unknown";
|
|
44
|
+
const base = {
|
|
45
|
+
source: "slack",
|
|
46
|
+
event: eventType,
|
|
47
|
+
repo: teamId,
|
|
48
|
+
sender,
|
|
49
|
+
timestamp: new Date().toISOString(),
|
|
50
|
+
};
|
|
51
|
+
switch (eventType) {
|
|
52
|
+
case "message":
|
|
53
|
+
return {
|
|
54
|
+
...base,
|
|
55
|
+
body: truncate(event.text),
|
|
56
|
+
comment: event.channel ? `channel:${event.channel}` : undefined,
|
|
57
|
+
};
|
|
58
|
+
case "app_mention":
|
|
59
|
+
return {
|
|
60
|
+
...base,
|
|
61
|
+
body: truncate(event.text),
|
|
62
|
+
comment: event.channel ? `channel:${event.channel}` : undefined,
|
|
63
|
+
};
|
|
64
|
+
case "reaction_added":
|
|
65
|
+
case "reaction_removed":
|
|
66
|
+
return {
|
|
67
|
+
...base,
|
|
68
|
+
title: event.reaction,
|
|
69
|
+
comment: event.item ? `${event.item.type}:${event.item.channel || ""}` : undefined,
|
|
70
|
+
};
|
|
71
|
+
default:
|
|
72
|
+
return {
|
|
73
|
+
...base,
|
|
74
|
+
title: eventType,
|
|
75
|
+
body: truncate(event.text),
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
matchesFilter(context, filter) {
|
|
80
|
+
const f = filter;
|
|
81
|
+
if (f.events?.length && !f.events.includes(context.event)) {
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
if (f.team_ids?.length && !f.team_ids.includes(context.repo)) {
|
|
85
|
+
return false;
|
|
86
|
+
}
|
|
87
|
+
// Channel matching: stored as comment field with "channel:<id>" format
|
|
88
|
+
if (f.channels?.length) {
|
|
89
|
+
const channelComment = context.comment;
|
|
90
|
+
if (!channelComment)
|
|
91
|
+
return false;
|
|
92
|
+
const match = channelComment.match(/^channel:(.+)$/);
|
|
93
|
+
if (!match)
|
|
94
|
+
return false;
|
|
95
|
+
const channelId = match[1];
|
|
96
|
+
if (!f.channels.includes(channelId))
|
|
97
|
+
return false;
|
|
98
|
+
}
|
|
99
|
+
return true;
|
|
100
|
+
}
|
|
101
|
+
handleChallenge(headers, rawBody, secrets, allowUnsigned) {
|
|
102
|
+
let body;
|
|
103
|
+
try {
|
|
104
|
+
body = JSON.parse(rawBody);
|
|
105
|
+
}
|
|
106
|
+
catch {
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
109
|
+
if (body.type !== "url_verification" || !body.challenge) {
|
|
110
|
+
return null;
|
|
111
|
+
}
|
|
112
|
+
// Validate the request signature before responding
|
|
113
|
+
const validationResult = this.validateRequest(headers, rawBody, secrets, allowUnsigned);
|
|
114
|
+
if (!validationResult) {
|
|
115
|
+
return null;
|
|
116
|
+
}
|
|
117
|
+
return { challenge: body.challenge };
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
//# sourceMappingURL=slack.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.js","sourceRoot":"","sources":["../../../src/webhooks/providers/slack.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { WebhookProvider, WebhookContext, WebhookFilter } from "../types.js";
|
|
2
|
+
export declare class TwitterWebhookProvider implements WebhookProvider {
|
|
3
|
+
source: string;
|
|
4
|
+
validateRequest(headers: Record<string, string | undefined>, rawBody: string, secrets?: Record<string, string>, allowUnsigned?: boolean): string | null;
|
|
5
|
+
handleCrcChallenge(queryParams: Record<string, string>, secrets?: Record<string, string>): {
|
|
6
|
+
status: number;
|
|
7
|
+
body: any;
|
|
8
|
+
} | null;
|
|
9
|
+
parseEvent(headers: Record<string, string | undefined>, body: any): WebhookContext | null;
|
|
10
|
+
private extractContext;
|
|
11
|
+
matchesFilter(context: WebhookContext, filter: WebhookFilter): boolean;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=twitter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"twitter.d.ts","sourceRoot":"","sources":["../../../src/webhooks/providers/twitter.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,eAAe,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAIlF,qBAAa,sBAAuB,YAAW,eAAe;IAC5D,MAAM,SAAa;IAEnB,eAAe,CACb,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,EAC3C,OAAO,EAAE,MAAM,EACf,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EAChC,aAAa,CAAC,EAAE,OAAO,GACtB,MAAM,GAAG,IAAI;IAIhB,kBAAkB,CAChB,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EACnC,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAC/B;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,GAAG,CAAA;KAAE,GAAG,IAAI;IAevC,UAAU,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,EAAE,IAAI,EAAE,GAAG,GAAG,cAAc,GAAG,IAAI;IA2CzF,OAAO,CAAC,cAAc;IAuItB,aAAa,CAAC,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,aAAa,GAAG,OAAO;CAavE"}
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
import { createHmac } from "crypto";
|
|
2
|
+
import { truncateEventText as truncate, validateHmacSignature } from "../validation.js";
|
|
3
|
+
export class TwitterWebhookProvider {
|
|
4
|
+
source = "twitter";
|
|
5
|
+
validateRequest(headers, rawBody, secrets, allowUnsigned) {
|
|
6
|
+
return validateHmacSignature(rawBody, headers["x-twitter-webhooks-signature"], secrets, "sha256=", allowUnsigned);
|
|
7
|
+
}
|
|
8
|
+
handleCrcChallenge(queryParams, secrets) {
|
|
9
|
+
const crcToken = queryParams["crc_token"];
|
|
10
|
+
if (!crcToken)
|
|
11
|
+
return null;
|
|
12
|
+
if (!secrets || Object.keys(secrets).length === 0)
|
|
13
|
+
return null;
|
|
14
|
+
// Use the first available secret
|
|
15
|
+
const secret = Object.values(secrets)[0];
|
|
16
|
+
const hmac = createHmac("sha256", secret).update(crcToken).digest("base64");
|
|
17
|
+
return {
|
|
18
|
+
status: 200,
|
|
19
|
+
body: { response_token: `sha256=${hmac}` },
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
parseEvent(headers, body) {
|
|
23
|
+
if (!body || typeof body !== "object")
|
|
24
|
+
return null;
|
|
25
|
+
const forUserId = body.for_user_id ?? "unknown";
|
|
26
|
+
// Metadata keys that are not event arrays
|
|
27
|
+
const metadataKeys = new Set(["for_user_id", "user_has_blocked"]);
|
|
28
|
+
// Find the first event key in the payload
|
|
29
|
+
let eventKey = null;
|
|
30
|
+
for (const key of Object.keys(body)) {
|
|
31
|
+
if (!metadataKeys.has(key) && Array.isArray(body[key])) {
|
|
32
|
+
eventKey = key;
|
|
33
|
+
break;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
if (!eventKey)
|
|
37
|
+
return null;
|
|
38
|
+
const events = body[eventKey];
|
|
39
|
+
const firstEvent = Array.isArray(events) && events.length > 0 ? events[0] : null;
|
|
40
|
+
// Derive action from event key (e.g. tweet_create_events -> create, favorite_events -> favorite)
|
|
41
|
+
const withoutSuffix = eventKey.replace(/_events$/, "");
|
|
42
|
+
const lastUnderscore = withoutSuffix.lastIndexOf("_");
|
|
43
|
+
const action = lastUnderscore >= 0 ? withoutSuffix.slice(lastUnderscore + 1) : withoutSuffix;
|
|
44
|
+
const base = {
|
|
45
|
+
source: "twitter",
|
|
46
|
+
event: eventKey,
|
|
47
|
+
action,
|
|
48
|
+
repo: forUserId,
|
|
49
|
+
sender: "unknown",
|
|
50
|
+
timestamp: new Date().toISOString(),
|
|
51
|
+
};
|
|
52
|
+
if (!firstEvent) {
|
|
53
|
+
return base;
|
|
54
|
+
}
|
|
55
|
+
return this.extractContext(eventKey, firstEvent, base);
|
|
56
|
+
}
|
|
57
|
+
extractContext(eventKey, event, base) {
|
|
58
|
+
switch (eventKey) {
|
|
59
|
+
case "tweet_create_events": {
|
|
60
|
+
return {
|
|
61
|
+
...base,
|
|
62
|
+
sender: event.user?.screen_name ?? event.user?.id_str ?? "unknown",
|
|
63
|
+
title: truncate(event.text),
|
|
64
|
+
body: truncate(event.text),
|
|
65
|
+
url: event.user?.screen_name && event.id_str
|
|
66
|
+
? `https://twitter.com/${event.user.screen_name}/status/${event.id_str}`
|
|
67
|
+
: undefined,
|
|
68
|
+
timestamp: event.created_at ? new Date(event.created_at).toISOString() : base.timestamp,
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
case "tweet_delete_events": {
|
|
72
|
+
return {
|
|
73
|
+
...base,
|
|
74
|
+
sender: event.user_id ?? "unknown",
|
|
75
|
+
title: `Deleted tweet ${event.status?.id ?? "unknown"}`,
|
|
76
|
+
timestamp: event.timestamp_ms
|
|
77
|
+
? new Date(parseInt(event.timestamp_ms)).toISOString()
|
|
78
|
+
: base.timestamp,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
case "favorite_events": {
|
|
82
|
+
return {
|
|
83
|
+
...base,
|
|
84
|
+
sender: event.user?.screen_name ?? event.user?.id_str ?? "unknown",
|
|
85
|
+
title: `Liked tweet by @${event.favorited_status?.user?.screen_name ?? "unknown"}`,
|
|
86
|
+
body: truncate(event.favorited_status?.text),
|
|
87
|
+
url: event.favorited_status?.user?.screen_name && event.favorited_status?.id_str
|
|
88
|
+
? `https://twitter.com/${event.favorited_status.user.screen_name}/status/${event.favorited_status.id_str}`
|
|
89
|
+
: undefined,
|
|
90
|
+
timestamp: event.created_at ? new Date(event.created_at).toISOString() : base.timestamp,
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
case "follow_events": {
|
|
94
|
+
return {
|
|
95
|
+
...base,
|
|
96
|
+
sender: event.source?.screen_name ?? event.source?.id ?? "unknown",
|
|
97
|
+
title: `@${event.source?.screen_name ?? "unknown"} followed @${event.target?.screen_name ?? "unknown"}`,
|
|
98
|
+
timestamp: event.created_timestamp
|
|
99
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
100
|
+
: base.timestamp,
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
case "unfollow_events": {
|
|
104
|
+
return {
|
|
105
|
+
...base,
|
|
106
|
+
sender: event.source?.screen_name ?? event.source?.id ?? "unknown",
|
|
107
|
+
title: `@${event.source?.screen_name ?? "unknown"} unfollowed @${event.target?.screen_name ?? "unknown"}`,
|
|
108
|
+
timestamp: event.created_timestamp
|
|
109
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
110
|
+
: base.timestamp,
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
case "block_events":
|
|
114
|
+
case "unblock_events": {
|
|
115
|
+
const evtType = eventKey === "block_events" ? "blocked" : "unblocked";
|
|
116
|
+
return {
|
|
117
|
+
...base,
|
|
118
|
+
sender: event.source?.screen_name ?? event.source?.id ?? "unknown",
|
|
119
|
+
title: `@${event.source?.screen_name ?? "unknown"} ${evtType} @${event.target?.screen_name ?? "unknown"}`,
|
|
120
|
+
timestamp: event.created_timestamp
|
|
121
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
122
|
+
: base.timestamp,
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
case "mute_events":
|
|
126
|
+
case "unmute_events": {
|
|
127
|
+
const evtType = eventKey === "mute_events" ? "muted" : "unmuted";
|
|
128
|
+
return {
|
|
129
|
+
...base,
|
|
130
|
+
sender: event.source?.screen_name ?? event.source?.id ?? "unknown",
|
|
131
|
+
title: `@${event.source?.screen_name ?? "unknown"} ${evtType} @${event.target?.screen_name ?? "unknown"}`,
|
|
132
|
+
timestamp: event.created_timestamp
|
|
133
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
134
|
+
: base.timestamp,
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
case "direct_message_events": {
|
|
138
|
+
const msgData = event.message_create?.message_data;
|
|
139
|
+
const senderId = event.message_create?.sender_id ?? "unknown";
|
|
140
|
+
return {
|
|
141
|
+
...base,
|
|
142
|
+
sender: senderId,
|
|
143
|
+
title: `Direct message from ${senderId}`,
|
|
144
|
+
body: truncate(msgData?.text),
|
|
145
|
+
timestamp: event.created_timestamp
|
|
146
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
147
|
+
: base.timestamp,
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
case "direct_message_indicate_typing_events": {
|
|
151
|
+
return {
|
|
152
|
+
...base,
|
|
153
|
+
sender: event.sender_id ?? "unknown",
|
|
154
|
+
title: `Typing indicator from ${event.sender_id ?? "unknown"}`,
|
|
155
|
+
timestamp: event.created_timestamp
|
|
156
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
157
|
+
: base.timestamp,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
case "direct_message_mark_read_events": {
|
|
161
|
+
return {
|
|
162
|
+
...base,
|
|
163
|
+
sender: event.sender_id ?? "unknown",
|
|
164
|
+
title: `Message read by ${event.sender_id ?? "unknown"}`,
|
|
165
|
+
timestamp: event.created_timestamp
|
|
166
|
+
? new Date(parseInt(event.created_timestamp)).toISOString()
|
|
167
|
+
: base.timestamp,
|
|
168
|
+
};
|
|
169
|
+
}
|
|
170
|
+
default:
|
|
171
|
+
return {
|
|
172
|
+
...base,
|
|
173
|
+
title: eventKey,
|
|
174
|
+
};
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
matchesFilter(context, filter) {
|
|
178
|
+
const f = filter;
|
|
179
|
+
if (f.events?.length && !f.events.includes(context.event)) {
|
|
180
|
+
return false;
|
|
181
|
+
}
|
|
182
|
+
if (f.users?.length && !f.users.includes(context.repo)) {
|
|
183
|
+
return false;
|
|
184
|
+
}
|
|
185
|
+
return true;
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
//# sourceMappingURL=twitter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"twitter.js","sourceRoot":"","sources":["../../../src/webhooks/providers/twitter.ts"],"names":[],"mappings":"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"}
|