cloudstructs 0.2.8 → 0.4.0

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.
Files changed (37) hide show
  1. package/.jsii +1477 -149
  2. package/API.md +346 -4
  3. package/lib/codecommit-mirror/index.js +4 -5
  4. package/lib/ecs-service-roller/index.js +2 -2
  5. package/lib/email-receiver/receiver.js +3 -4
  6. package/lib/saml-identity-provider/index.js +2 -2
  7. package/lib/slack-app/index.d.ts +2 -95
  8. package/lib/slack-app/index.js +13 -47
  9. package/lib/slack-app/manifest.d.ts +482 -0
  10. package/lib/slack-app/manifest.js +198 -0
  11. package/lib/slack-app/provider.handler.js +4 -3
  12. package/lib/slack-app/provider.js +1 -3
  13. package/lib/slack-app/slack-app.d.ts +127 -0
  14. package/lib/slack-app/slack-app.js +103 -0
  15. package/lib/slack-events/index.js +2 -4
  16. package/lib/slack-textract/index.js +2 -4
  17. package/lib/state-machine-cr-provider/index.js +4 -4
  18. package/lib/static-website/index.d.ts +9 -5
  19. package/lib/static-website/index.js +36 -20
  20. package/lib/url-shortener/index.js +2 -4
  21. package/node_modules/@types/cacheable-request/node_modules/@types/node/README.md +1 -1
  22. package/node_modules/@types/cacheable-request/node_modules/@types/node/http2.d.ts +1 -1
  23. package/node_modules/@types/cacheable-request/node_modules/@types/node/package.json +2 -2
  24. package/node_modules/@types/is-stream/node_modules/@types/node/README.md +1 -1
  25. package/node_modules/@types/is-stream/node_modules/@types/node/http2.d.ts +1 -1
  26. package/node_modules/@types/is-stream/node_modules/@types/node/package.json +2 -2
  27. package/node_modules/@types/keyv/node_modules/@types/node/README.md +1 -1
  28. package/node_modules/@types/keyv/node_modules/@types/node/http2.d.ts +1 -1
  29. package/node_modules/@types/keyv/node_modules/@types/node/package.json +2 -2
  30. package/node_modules/@types/responselike/node_modules/@types/node/README.md +1 -1
  31. package/node_modules/@types/responselike/node_modules/@types/node/http2.d.ts +1 -1
  32. package/node_modules/@types/responselike/node_modules/@types/node/package.json +2 -2
  33. package/package.json +8 -2
  34. package/lib/static-website/origin-response-handler/index.d.ts +0 -1
  35. package/lib/static-website/origin-response-handler/index.js +0 -25
  36. package/lib/static-website/security-headers.d.ts +0 -10
  37. package/lib/static-website/security-headers.js +0 -14
@@ -0,0 +1,198 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.SlackAppManifest = exports.SlackAppManifestShortcutType = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const nodeUrl = require("url");
7
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
8
+ /**
9
+ * Type of shortcuts.
10
+ *
11
+ * @see https://api.slack.com/interactivity/shortcuts
12
+ * @stability stable
13
+ */
14
+ var SlackAppManifestShortcutType;
15
+ (function (SlackAppManifestShortcutType) {
16
+ SlackAppManifestShortcutType["MESSAGE"] = "message";
17
+ SlackAppManifestShortcutType["GLOBAL"] = "global";
18
+ })(SlackAppManifestShortcutType = exports.SlackAppManifestShortcutType || (exports.SlackAppManifestShortcutType = {}));
19
+ ;
20
+ /**
21
+ * A Slack app manifest.
22
+ *
23
+ * @see https://api.slack.com/reference/manifests
24
+ * @stability stable
25
+ */
26
+ class SlackAppManifest {
27
+ /**
28
+ * @stability stable
29
+ */
30
+ constructor(props) {
31
+ var _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q;
32
+ this.props = props;
33
+ validateLength('app name', 35, props.name);
34
+ validateLength('app description', 140, props.description);
35
+ validateLength('app long description', 4000, props.longDescription);
36
+ validateColor(props.backgroundColor);
37
+ validateUrl((_b = props.eventSubscriptions) === null || _b === void 0 ? void 0 : _b.requestUrl);
38
+ validateItems('bot events', 100, (_c = props.eventSubscriptions) === null || _c === void 0 ? void 0 : _c.botEvents);
39
+ validateItems('users events', 100, (_d = props.eventSubscriptions) === null || _d === void 0 ? void 0 : _d.userEvents);
40
+ validateUrl((_e = props.interactivity) === null || _e === void 0 ? void 0 : _e.requestUrl);
41
+ validateUrl((_f = props.interactivity) === null || _f === void 0 ? void 0 : _f.messageMenuOptionsUrl);
42
+ validateLength('bot display name', 80, (_g = props.botUser) === null || _g === void 0 ? void 0 : _g.displayName);
43
+ validateItems('shortcuts', 5, props.shortcuts);
44
+ (_h = props.shortcuts) === null || _h === void 0 ? void 0 : _h.forEach((shortcut) => {
45
+ validateLength('shortcut callback ID', 255, shortcut.callbackId);
46
+ validateLength('shortcut description', 150, shortcut.description);
47
+ });
48
+ validateItems('slash commands', 5, props.slashCommands);
49
+ (_j = props.slashCommands) === null || _j === void 0 ? void 0 : _j.forEach((command) => {
50
+ validateLength('slash command', 32, command.command);
51
+ validateLength('slash command description', 2000, command.description),
52
+ validateUrl(command.url);
53
+ validateLength('slash command use hint', 1000, command.usageHint);
54
+ });
55
+ validateItems('workflow steps', 10, props.workflowSteps);
56
+ (_k = props.workflowSteps) === null || _k === void 0 ? void 0 : _k.forEach((step) => {
57
+ validateLength('workflow step name', 50, step.name);
58
+ validateLength('workflow step callback ID', 50, step.callbackId);
59
+ });
60
+ validateItems('unfurls domains', 5, props.unfurlDomains);
61
+ validateItems('OAuth redirect URLs', 1000, (_l = props.oauthConfig) === null || _l === void 0 ? void 0 : _l.redirectUrls);
62
+ (_o = (_m = props.oauthConfig) === null || _m === void 0 ? void 0 : _m.redirectUrls) === null || _o === void 0 ? void 0 : _o.forEach((url) => {
63
+ validateUrl(url, false);
64
+ });
65
+ validateItems('bot scopes', 255, (_p = props.oauthConfig) === null || _p === void 0 ? void 0 : _p.botScopes);
66
+ validateItems('user scopes', 255, (_q = props.oauthConfig) === null || _q === void 0 ? void 0 : _q.userScopes);
67
+ }
68
+ /**
69
+ * @stability stable
70
+ */
71
+ render(construct) {
72
+ var _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s, _t, _u, _v;
73
+ const schema = {
74
+ _metadata: {
75
+ major_version: this.props.majorVersion,
76
+ minor_version: this.props.minorVersion,
77
+ },
78
+ display_information: {
79
+ name: this.props.name,
80
+ description: this.props.description,
81
+ long_description: this.props.longDescription,
82
+ background_color: (_b = prefixWith('#', this.props.backgroundColor)) === null || _b === void 0 ? void 0 : _b.toLowerCase(),
83
+ },
84
+ settings: {
85
+ allowed_ip_address_ranges: this.props.allowedIpAddressRanges,
86
+ event_subscriptions: {
87
+ request_url: (_c = this.props.eventSubscriptions) === null || _c === void 0 ? void 0 : _c.requestUrl,
88
+ bot_events: (_d = this.props.eventSubscriptions) === null || _d === void 0 ? void 0 : _d.botEvents,
89
+ user_events: (_e = this.props.eventSubscriptions) === null || _e === void 0 ? void 0 : _e.userEvents,
90
+ },
91
+ interactivity: this.props.interactivity
92
+ ? {
93
+ is_enabled: (_g = (_f = this.props.interactivity) === null || _f === void 0 ? void 0 : _f.enabled) !== null && _g !== void 0 ? _g : true,
94
+ request_url: (_h = this.props.interactivity) === null || _h === void 0 ? void 0 : _h.requestUrl,
95
+ message_menu_options_url: (_j = this.props.interactivity) === null || _j === void 0 ? void 0 : _j.messageMenuOptionsUrl,
96
+ }
97
+ : undefined,
98
+ org_deploy_enabled: this.props.orgDeploy,
99
+ socket_mode_enabled: this.props.socketMode,
100
+ },
101
+ features: {
102
+ app_home: {
103
+ home_tab_enabled: (_k = this.props.appHome) === null || _k === void 0 ? void 0 : _k.homeTab,
104
+ messages_tab_enabled: (_l = this.props.appHome) === null || _l === void 0 ? void 0 : _l.messagesTab,
105
+ messages_tab_read_only_enabled: (_m = this.props.appHome) === null || _m === void 0 ? void 0 : _m.messagesTabReadOnly,
106
+ },
107
+ bot_user: this.props.botUser
108
+ ? {
109
+ display_name: (_o = this.props.botUser) === null || _o === void 0 ? void 0 : _o.displayName,
110
+ always_online: (_p = this.props.botUser) === null || _p === void 0 ? void 0 : _p.alwaysOnline,
111
+ }
112
+ : undefined,
113
+ shortcuts: (_q = this.props.shortcuts) === null || _q === void 0 ? void 0 : _q.map((shortcut) => ({
114
+ name: shortcut.name,
115
+ type: shortcut.type,
116
+ callback_id: shortcut.callbackId,
117
+ description: shortcut.description,
118
+ })),
119
+ slash_commands: (_r = this.props.slashCommands) === null || _r === void 0 ? void 0 : _r.map((command) => ({
120
+ command: prefixWith('/', command.command),
121
+ description: command.description,
122
+ url: command.url,
123
+ usage_hint: command.usageHint,
124
+ should_escape: command.shouldEscape,
125
+ })),
126
+ workflow_steps: (_s = this.props.workflowSteps) === null || _s === void 0 ? void 0 : _s.map((step) => ({
127
+ name: step.name,
128
+ callback_id: step.callbackId,
129
+ })),
130
+ unfurl_domains: this.props.unfurlDomains,
131
+ },
132
+ oauth_config: {
133
+ redirect_urls: (_t = this.props.oauthConfig) === null || _t === void 0 ? void 0 : _t.redirectUrls,
134
+ scopes: {
135
+ bot: (_u = this.props.oauthConfig) === null || _u === void 0 ? void 0 : _u.botScopes,
136
+ users: (_v = this.props.oauthConfig) === null || _v === void 0 ? void 0 : _v.userScopes,
137
+ },
138
+ },
139
+ };
140
+ return aws_cdk_lib_1.Stack.of(construct).toJsonString(removeUndefined(schema));
141
+ }
142
+ }
143
+ exports.SlackAppManifest = SlackAppManifest;
144
+ _a = JSII_RTTI_SYMBOL_1;
145
+ SlackAppManifest[_a] = { fqn: "cloudstructs.SlackAppManifest", version: "0.4.0" };
146
+ function prefixWith(prefix, string) {
147
+ if (!string) {
148
+ return undefined;
149
+ }
150
+ if (string.startsWith(prefix)) {
151
+ return string;
152
+ }
153
+ return `${prefix}${string}`;
154
+ }
155
+ function validateLength(description, max, xs) {
156
+ if (xs && !aws_cdk_lib_1.Token.isUnresolved(xs) && xs.length > max) {
157
+ throw new Error(`Maximum length for ${description} is ${max}, got ${xs.length}: ${xs}`);
158
+ }
159
+ }
160
+ function validateItems(description, max, xs) {
161
+ if (xs && !aws_cdk_lib_1.Token.isUnresolved(xs) && xs.length > max) {
162
+ throw new Error(`Maximum number of items for ${description} is ${max}, got ${xs.length}`);
163
+ }
164
+ }
165
+ function validateColor(color) {
166
+ if (color && !aws_cdk_lib_1.Token.isUnresolved(color) && !/^#?([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(color)) {
167
+ throw new Error(`Invalid hex color: ${color}`);
168
+ }
169
+ }
170
+ function validateUrl(url, https = true) {
171
+ if (url && !aws_cdk_lib_1.Token.isUnresolved(url)) {
172
+ try {
173
+ const parsed = new nodeUrl.URL(url);
174
+ if (https && parsed.protocol !== 'https:') {
175
+ throw new Error('Invalid protocol');
176
+ }
177
+ }
178
+ catch (err) {
179
+ throw new Error(`${url} is not a valid${https ? ' HTTPS' : ''} url`);
180
+ }
181
+ }
182
+ }
183
+ function removeUndefined(obj) {
184
+ if (typeof obj === 'string') {
185
+ return obj;
186
+ }
187
+ if (Array.isArray(obj)) {
188
+ const ret = obj
189
+ .map(v => (v && typeof v === 'object') ? removeUndefined(v) : v)
190
+ .filter(v => !(v == null));
191
+ return ret.length !== 0 ? ret : undefined;
192
+ }
193
+ const ret = Object.entries(obj)
194
+ .map(([k, v]) => [k, v && typeof v === 'object' ? removeUndefined(v) : v])
195
+ .reduce((a, [k, v]) => (v == null ? a : (a[k] = v, a)), {});
196
+ return Object.keys(ret).length !== 0 ? ret : undefined;
197
+ }
198
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"manifest.js","sourceRoot":"","sources":["../../src/slack-app/manifest.ts"],"names":[],"mappings":"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","sourcesContent":["import * as nodeUrl from 'url';\nimport { Stack, Token } from 'aws-cdk-lib';\nimport { IConstruct } from 'constructs';\n\n                                                                                                   \nexport interface SlackAppManifestProps {\n                                                                                \n  readonly name: string;\n\n                                                                                                                                                           \n  readonly description?: string;\n\n                                                                                                             \n  readonly longDescription?: string;\n\n                                                                                                                                                                                                                         \n  readonly backgroundColor?: string;\n\n                                                                                                                                \n  readonly majorVersion?: number;\n\n                                                                                                                               \n  readonly minorVersion?: number;\n\n                                                                                                                                                                          \n  readonly allowedIpAddressRanges?: string[];\n\n                                                                                                     \n  readonly eventSubscriptions?: SlackAppManifestEventSubscriptions;\n\n                                                        \n  readonly interactivity?: SlackAppManifestInteractivity;\n\n                                                                                                                                  \n  readonly orgDeploy?: boolean;\n\n                                                                                                                                     \n  readonly socketMode?: boolean;\n\n                                                                                          \n  readonly appHome?: SlackAppManifestAppHome;\n\n                                                                                      \n  readonly botUser?: SlackkAppManifestBotUser;\n\n                                                                                                                                                          \n  readonly shortcuts?: SlackAppManifestShortcut[];\n\n                                                                                                                                                                         \n  readonly slashCommands?: SlackAppManifestSlashCommand[];\n\n                                                                                                                                               \n  readonly workflowSteps?: SlackAppManifestWorkflowStep[];\n\n                                                                                                                                                                                                       \n  readonly unfurlDomains?: string[];\n\n                                                \n  readonly oauthConfig?: SlackAppManifestOauthConfig;\n}\n\n                                                   \nexport interface SlackAppManifestSettings {\n                                                                                                                                                                          \n  readonly allowedIpAddressRanges?: string[];\n\n                                                                                                     \n  readonly eventSubscriptions?: SlackAppManifestEventSubscriptions;\n\n                                                        \n  readonly interactivity?: SlackAppManifestInteractivity;\n\n                                                                                                                                  \n  readonly orgDeploy?: boolean;\n\n                                                                                                                                     \n  readonly socketMode?: boolean;\n}\n\n                                                                                           \nexport interface SlackAppManifestEventSubscriptions {\n                                                                                                                                                                                               \n  readonly requestUrl: string;\n\n                                                                                                                                                             \n  readonly botEvents?: string[];\n\n                                                                                                                                                \n  readonly userEvents?: string[];\n}\n\n                                                                                                                \nexport interface SlackAppManifestInteractivity {\n                                                                                          \n  readonly enabled?: boolean;\n\n                                                                            \n  readonly requestUrl?: string;\n\n                                                                                \n  readonly messageMenuOptionsUrl?: string;\n}\n\n                                                                                \nexport interface SlackAppManifestAppHome {\n                                                                        \n  readonly homeTab?: boolean;\n\n                                                                        \n  readonly messagesTab?: boolean;\n\n                                                                                                                                   \n  readonly messagesTabReadOnly?: boolean;\n}\n\n                                                                            \nexport interface SlackkAppManifestBotUser {\n                                                                                             \n  readonly displayName: string;\n\n                                                                                              \n  readonly alwaysOnline?: boolean;\n}\n\n                                                                                          \nexport interface SlackAppManifestShortcut {\n                                         \n  readonly name: string;\n\n                                                                                                  \n  readonly type: SlackAppManifestShortcutType;\n\n                                                                                             \n  readonly callbackId: string;\n\n                                                                                               \n  readonly description: string;\n}\n\n                                                                                     \nexport enum SlackAppManifestShortcutType {\n                                                                                                                                                                                        \n  MESSAGE = 'message',\n\n                                                                                                                                                                                                                          \n  GLOBAL = 'global',\n}\n\n                                                                                                    \nexport interface SlackAppManifestSlashCommand {\n                                                                                    \n  readonly command: string;\n\n                                                                                                   \n  readonly description: string;\n\n                                                                                                                                                                    \n  readonly url?: string;\n\n                                                                                                                     \n  readonly usageHint?: string;\n\n                                                                                                                           \n  readonly shouldEscape?: boolean;\n}\n\n                                                                         \nexport interface SlackAppManifestWorkflowStep {\n                                                                                          \n  readonly name: string;\n\n                                                                                                 \n  readonly callbackId: string;\n};\n\n                                          \nexport interface SlackAppManifestOauthConfig {\n                                                                                                                                                                          \n  readonly redirectUrls?: string[];\n\n                                                                                                                                                            \n  readonly botScopes?: string[];\n\n                                                                                                                                                             \n  readonly userScopes?: string[];\n}\n\n\n                                                                                    \nexport class SlackAppManifest {\n  constructor(private readonly props: SlackAppManifestProps) {\n    validateLength('app name', 35, props.name);\n    validateLength('app description', 140, props.description);\n    validateLength('app long description', 4000, props.longDescription);\n    validateColor(props.backgroundColor);\n\n    validateUrl(props.eventSubscriptions?.requestUrl);\n    validateItems('bot events', 100, props.eventSubscriptions?.botEvents);\n    validateItems('users events', 100, props.eventSubscriptions?.userEvents);\n\n    validateUrl(props.interactivity?.requestUrl);\n    validateUrl(props.interactivity?.messageMenuOptionsUrl);\n    validateLength('bot display name', 80, props.botUser?.displayName);\n\n    validateItems('shortcuts', 5, props.shortcuts);\n    props.shortcuts?.forEach((shortcut) => {\n      validateLength('shortcut callback ID', 255, shortcut.callbackId);\n      validateLength('shortcut description', 150, shortcut.description);\n    });\n\n    validateItems('slash commands', 5, props.slashCommands);\n    props.slashCommands?.forEach((command) => {\n      validateLength('slash command', 32, command.command);\n      validateLength('slash command description', 2000, command.description),\n      validateUrl(command.url);\n      validateLength('slash command use hint', 1000, command.usageHint);\n    });\n\n    validateItems('workflow steps', 10, props.workflowSteps);\n    props.workflowSteps?.forEach((step) => {\n      validateLength('workflow step name', 50, step.name);\n      validateLength('workflow step callback ID', 50, step.callbackId);\n    });\n\n    validateItems('unfurls domains', 5, props.unfurlDomains);\n\n    validateItems('OAuth redirect URLs', 1000, props.oauthConfig?.redirectUrls);\n    props.oauthConfig?.redirectUrls?.forEach((url) => {\n      validateUrl(url, false);\n    });\n    validateItems('bot scopes', 255, props.oauthConfig?.botScopes);\n    validateItems('user scopes', 255, props.oauthConfig?.userScopes);\n  }\n\n  public render(construct: IConstruct): string {\n    const schema: SlackAppManifestSchema = {\n      _metadata: {\n        major_version: this.props.majorVersion,\n        minor_version: this.props.minorVersion,\n      },\n      display_information: {\n        name: this.props.name,\n        description: this.props.description,\n        long_description: this.props.longDescription,\n        background_color: prefixWith('#', this.props.backgroundColor)?.toLowerCase(),\n      },\n      settings: {\n        allowed_ip_address_ranges: this.props.allowedIpAddressRanges,\n        event_subscriptions: {\n          request_url: this.props.eventSubscriptions?.requestUrl,\n          bot_events: this.props.eventSubscriptions?.botEvents,\n          user_events: this.props.eventSubscriptions?.userEvents,\n        },\n        interactivity: this.props.interactivity\n          ? {\n            is_enabled: this.props.interactivity?.enabled ?? true,\n            request_url: this.props.interactivity?.requestUrl,\n            message_menu_options_url: this.props.interactivity?.messageMenuOptionsUrl,\n          }\n          : undefined,\n        org_deploy_enabled: this.props.orgDeploy,\n        socket_mode_enabled: this.props.socketMode,\n      },\n      features: {\n        app_home: {\n          home_tab_enabled: this.props.appHome?.homeTab,\n          messages_tab_enabled: this.props.appHome?.messagesTab,\n          messages_tab_read_only_enabled: this.props.appHome?.messagesTabReadOnly,\n        },\n        bot_user: this.props.botUser\n          ? {\n            display_name: this.props.botUser?.displayName,\n            always_online: this.props.botUser?.alwaysOnline,\n          }\n          : undefined,\n        shortcuts: this.props.shortcuts?.map((shortcut) => ({\n          name: shortcut.name,\n          type: shortcut.type,\n          callback_id: shortcut.callbackId,\n          description: shortcut.description,\n        })),\n        slash_commands: this.props.slashCommands?.map((command) => ({\n          command: prefixWith('/', command.command),\n          description: command.description,\n          url: command.url,\n          usage_hint: command.usageHint,\n          should_escape: command.shouldEscape,\n        })),\n        workflow_steps: this.props.workflowSteps?.map((step) => ({\n          name: step.name,\n          callback_id: step.callbackId,\n        })),\n        unfurl_domains: this.props.unfurlDomains,\n      },\n      oauth_config: {\n        redirect_urls: this.props.oauthConfig?.redirectUrls,\n        scopes: {\n          bot: this.props.oauthConfig?.botScopes,\n          users: this.props.oauthConfig?.userScopes,\n        },\n      },\n    };\n\n    return Stack.of(construct).toJsonString(removeUndefined(schema));\n  }\n}\n\ninterface SlackAppManifestSchema {\n  _metadata?: {\n    major_version?: number;\n    minor_version?: number;\n  };\n  display_information: {\n    name: string;\n    description?: string;\n    long_description?: string;\n    background_color?: string;\n  };\n  settings?: {\n    allowed_ip_address_ranges?: string[];\n    event_subscriptions?: {\n      request_url?: string;\n      bot_events?: string[];\n      user_events?: string[];\n    };\n    interactivity?: {\n      is_enabled: boolean;\n      request_url?: string;\n      message_menu_options_url?: string;\n    };\n    org_deploy_enabled?: boolean;\n    socket_mode_enabled?: boolean;\n  };\n  features?: {\n    app_home?: {\n      home_tab_enabled?: boolean;\n      messages_tab_enabled?: boolean;\n      messages_tab_read_only_enabled?: boolean;\n    };\n    bot_user?: {\n      display_name: string;\n      always_online?: boolean;\n    };\n    shortcuts?: {\n      name: string;\n      type: string;\n      callback_id: string;\n      description: string;\n    }[];\n    slash_commands?: {\n      command: string;\n      description: string;\n      url?: string;\n      usage_hint?: string;\n      should_escape?: boolean;\n    }[];\n    workflow_steps?: {\n      name: string;\n      callback_id: string;\n    }[];\n    unfurl_domains?: string[];\n  };\n  oauth_config?: {\n    redirect_urls?: string[];\n    scopes?: {\n      bot?: string[];\n      users?: string[];\n    };\n  };\n}\n\nfunction prefixWith<T extends string | undefined>(prefix: string, string: T): T {\n  if (!string) {\n    return undefined as T;\n  }\n\n  if (string.startsWith(prefix)) {\n    return string;\n  }\n\n  return `${prefix}${string}` as T;\n}\n\nfunction validateLength(description: string, max: number, xs?: string): void {\n  if (xs && !Token.isUnresolved(xs) && xs.length > max) {\n    throw new Error(`Maximum length for ${description} is ${max}, got ${xs.length}: ${xs}`);\n  }\n}\n\nfunction validateItems<T>(description: string, max: number, xs?: T[]): void {\n  if (xs && !Token.isUnresolved(xs) && xs.length > max) {\n    throw new Error(`Maximum number of items for ${description} is ${max}, got ${xs.length}`);\n  }\n}\n\nfunction validateColor(color?: string): void {\n  if (color && !Token.isUnresolved(color) && !/^#?([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(color)) {\n    throw new Error(`Invalid hex color: ${color}`);\n  }\n}\n\nfunction validateUrl(url?: string, https = true): void {\n  if (url && !Token.isUnresolved(url)) {\n    try {\n      const parsed = new nodeUrl.URL(url);\n      if (https && parsed.protocol !== 'https:') {\n        throw new Error('Invalid protocol');\n      }\n    } catch (err) {\n      throw new Error(`${url} is not a valid${https ? ' HTTPS' : ''} url`);\n    }\n  }\n}\n\nfunction removeUndefined(obj: any): any {\n  if (typeof obj === 'string') {\n    return obj;\n  }\n\n  if (Array.isArray(obj)) {\n    const ret = obj\n      .map(v => (v && typeof v === 'object') ? removeUndefined(v) : v)\n      .filter(v => !(v == null));\n    return ret.length !== 0 ? ret : undefined;\n  }\n\n  const ret = Object.entries(obj)\n    .map(([k, v]) => [k, v && typeof v === 'object' ? removeUndefined(v) : v])\n    .reduce((a: any, [k, v]) => (v == null ? a : (a[k]=v, a)), {});\n  return Object.keys(ret).length !== 0 ? ret : undefined;\n}\n"]}
@@ -43,10 +43,11 @@ async function handler(event) {
43
43
  console.log(`Successfully saved access token in secret ${putSecretValue.ARN}`);
44
44
  }
45
45
  const operation = event.RequestType.toLowerCase();
46
- console.log('Calling manifest API');
46
+ const request = getManifestRequest(event);
47
+ console.log(`Calling ${operation} manifest API: %j`, request);
47
48
  const response = await slackClient.post(`apps.manifest.${operation}`, {
48
49
  headers: { Authorization: `Bearer ${accessToken}` },
49
- json: getManifestRequest(event),
50
+ json: request,
50
51
  }).json();
51
52
  if (!response.ok) {
52
53
  const errors = response.errors
@@ -97,4 +98,4 @@ function getManifestRequest(event) {
97
98
  };
98
99
  }
99
100
  }
100
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider.handler.js","sourceRoot":"","sources":["../../src/slack-app/provider.handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport { OnEventRequest, OnEventResponse } from 'aws-cdk-lib/custom-resources/lib/provider-framework/types';\nimport { SecretsManager } from 'aws-sdk'; // eslint-disable-line import/no-extraneous-dependencies\nimport got from 'got';\n\ninterface SlackSecret {\n  accessToken?: string;\n  refreshToken?: string;\n  exp?: number;\n}\n\ninterface RotateResponse {\n  ok: boolean;\n  error?: string;\n  token: string;\n  refresh_token: string;\n  team_id: string;\n  user_id: string;\n  iat: number;\n  exp: number;\n}\n\ninterface ManifestCreateRequest {\n  manifest: string;\n}\n\ninterface ManifestUpdateRequest extends ManifestCreateRequest {\n  app_id: string;\n}\n\ninterface ManifestDeleteRequest {\n  app_id: string;\n}\n\ntype ManifestRequest = ManifestCreateRequest | ManifestUpdateRequest | ManifestDeleteRequest;\n\ninterface ManifestResponse {\n  ok: boolean;\n  error?: string;\n  errors?: {\n    message: string;\n    pointer: string;\n  }[];\n  app_id?: string;\n  credentials?: {\n    client_id: string;\n    client_secret: string;\n    verification_token: string;\n    signing_secret: string;\n  };\n}\n\nconst secretsmanager = new SecretsManager({ apiVersion: '2017-10-17' });\n\nconst slackClient = got.extend({\n  prefixUrl: 'https://slack.com/api',\n});\n\nexport async function handler(event: OnEventRequest): Promise<OnEventResponse> {\n  console.log('Event: %j', event);\n\n  const data = await secretsmanager.getSecretValue({\n    SecretId: event.ResourceProperties.configurationTokenSecretArn,\n  }).promise();\n\n  if (!data.SecretString) {\n    throw new Error('No secret string found in configuration token secret');\n  }\n\n  const secret: SlackSecret = JSON.parse(data.SecretString);\n\n  let accessToken = secret.accessToken;\n\n  if (!accessToken || isExpired(secret.exp ?? 0)) {\n    if (!secret.refreshToken) {\n      throw new Error('No refresh token found in configuration token secret');\n    }\n\n    console.log('Refreshing access token');\n    const rotate = await slackClient.get('tooling.tokens.rotate', {\n      searchParams: { refresh_token: secret.refreshToken },\n    }).json<RotateResponse>();\n\n    if (!rotate.ok) {\n      throw new Error(`Failed to refresh access token: ${rotate.error}`);\n    }\n    console.log('Access token refreshed');\n\n    accessToken = rotate.token;\n\n    console.log('Saving access token');\n    const putSecretValue = await secretsmanager.putSecretValue({\n      SecretId: event.ResourceProperties.configurationTokenSecretArn,\n      SecretString: JSON.stringify({\n        accessToken,\n        refreshToken: rotate.refresh_token,\n        exp: rotate.exp,\n      }),\n    }).promise();\n    console.log(`Successfully saved access token in secret ${putSecretValue.ARN}`);\n  }\n\n  const operation = event.RequestType.toLowerCase();\n  console.log('Calling manifest API');\n  const response = await slackClient.post(`apps.manifest.${operation}`, {\n    headers: { Authorization: `Bearer ${accessToken}` },\n    json: getManifestRequest(event),\n  }).json<ManifestResponse>();\n\n  if (!response.ok) {\n    const errors = response.errors\n      ? response.errors.map((err) => `${err.message} at ${err.pointer}`).join('\\n')\n      : '';\n    throw new Error(`Failed to ${operation} manifest: ${response.error}.${errors ? `\\n${errors}}` : ''}`);\n  }\n\n  console.log(`Successfully ${operation}d Slack app ${event.PhysicalResourceId ?? response.app_id}`);\n\n  if (event.RequestType === 'Create' && response.credentials) {\n    console.log('Saving app credentials');\n    const putSecretValue = await secretsmanager.putSecretValue({\n      SecretId: event.ResourceProperties.credentialsSecretArn,\n      SecretString: JSON.stringify({\n        appId: response.app_id,\n        clientId: response.credentials.client_id,\n        clientSecret: response.credentials.client_secret,\n        verificationToken: response.credentials.verification_token,\n        signingSecret: response.credentials.signing_secret,\n      }),\n    }).promise();\n    console.log(`Successfully saved app credentials in secret ${putSecretValue.ARN}`);\n  }\n\n  return {\n    PhysicalResourceId: response.app_id,\n    Data: {\n      appId: response.app_id,\n    },\n  };\n}\n\nfunction isExpired(iat: number) {\n  return (iat - (Date.now() / 1000)) < 0;\n}\n\nfunction getManifestRequest(event: OnEventRequest): ManifestRequest {\n  switch (event.RequestType) {\n    case 'Create':\n      return {\n        manifest: event.ResourceProperties.manifest,\n      };\n    case 'Update':\n      return {\n        app_id: event.PhysicalResourceId,\n        manifest: event.ResourceProperties.manifest,\n      };\n    case 'Delete':\n      return {\n        app_id: event.PhysicalResourceId!,\n      };\n  }\n}\n"]}
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider.handler.js","sourceRoot":"","sources":["../../src/slack-app/provider.handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport { OnEventRequest, OnEventResponse } from 'aws-cdk-lib/custom-resources/lib/provider-framework/types';\nimport { SecretsManager } from 'aws-sdk'; // eslint-disable-line import/no-extraneous-dependencies\nimport got from 'got';\n\ninterface SlackSecret {\n  accessToken?: string;\n  refreshToken?: string;\n  exp?: number;\n}\n\ninterface RotateResponse {\n  ok: boolean;\n  error?: string;\n  token: string;\n  refresh_token: string;\n  team_id: string;\n  user_id: string;\n  iat: number;\n  exp: number;\n}\n\ninterface ManifestCreateRequest {\n  manifest: string;\n}\n\ninterface ManifestUpdateRequest extends ManifestCreateRequest {\n  app_id: string;\n}\n\ninterface ManifestDeleteRequest {\n  app_id: string;\n}\n\ntype ManifestRequest = ManifestCreateRequest | ManifestUpdateRequest | ManifestDeleteRequest;\n\ninterface ManifestResponse {\n  ok: boolean;\n  error?: string;\n  errors?: {\n    message: string;\n    pointer: string;\n  }[];\n  app_id?: string;\n  credentials?: {\n    client_id: string;\n    client_secret: string;\n    verification_token: string;\n    signing_secret: string;\n  };\n}\n\nconst secretsmanager = new SecretsManager({ apiVersion: '2017-10-17' });\n\nconst slackClient = got.extend({\n  prefixUrl: 'https://slack.com/api',\n});\n\nexport async function handler(event: OnEventRequest): Promise<OnEventResponse> {\n  console.log('Event: %j', event);\n\n  const data = await secretsmanager.getSecretValue({\n    SecretId: event.ResourceProperties.configurationTokenSecretArn,\n  }).promise();\n\n  if (!data.SecretString) {\n    throw new Error('No secret string found in configuration token secret');\n  }\n\n  const secret: SlackSecret = JSON.parse(data.SecretString);\n\n  let accessToken = secret.accessToken;\n\n  if (!accessToken || isExpired(secret.exp ?? 0)) {\n    if (!secret.refreshToken) {\n      throw new Error('No refresh token found in configuration token secret');\n    }\n\n    console.log('Refreshing access token');\n    const rotate = await slackClient.get('tooling.tokens.rotate', {\n      searchParams: { refresh_token: secret.refreshToken },\n    }).json<RotateResponse>();\n\n    if (!rotate.ok) {\n      throw new Error(`Failed to refresh access token: ${rotate.error}`);\n    }\n    console.log('Access token refreshed');\n\n    accessToken = rotate.token;\n\n    console.log('Saving access token');\n    const putSecretValue = await secretsmanager.putSecretValue({\n      SecretId: event.ResourceProperties.configurationTokenSecretArn,\n      SecretString: JSON.stringify({\n        accessToken,\n        refreshToken: rotate.refresh_token,\n        exp: rotate.exp,\n      }),\n    }).promise();\n    console.log(`Successfully saved access token in secret ${putSecretValue.ARN}`);\n  }\n\n  const operation = event.RequestType.toLowerCase();\n  const request = getManifestRequest(event);\n\n  console.log(`Calling ${operation} manifest API: %j`, request);\n  const response = await slackClient.post(`apps.manifest.${operation}`, {\n    headers: { Authorization: `Bearer ${accessToken}` },\n    json: request,\n  }).json<ManifestResponse>();\n\n  if (!response.ok) {\n    const errors = response.errors\n      ? response.errors.map((err) => `${err.message} at ${err.pointer}`).join('\\n')\n      : '';\n    throw new Error(`Failed to ${operation} manifest: ${response.error}.${errors ? `\\n${errors}}` : ''}`);\n  }\n\n  console.log(`Successfully ${operation}d Slack app ${event.PhysicalResourceId ?? response.app_id}`);\n\n  if (event.RequestType === 'Create' && response.credentials) {\n    console.log('Saving app credentials');\n    const putSecretValue = await secretsmanager.putSecretValue({\n      SecretId: event.ResourceProperties.credentialsSecretArn,\n      SecretString: JSON.stringify({\n        appId: response.app_id,\n        clientId: response.credentials.client_id,\n        clientSecret: response.credentials.client_secret,\n        verificationToken: response.credentials.verification_token,\n        signingSecret: response.credentials.signing_secret,\n      }),\n    }).promise();\n    console.log(`Successfully saved app credentials in secret ${putSecretValue.ARN}`);\n  }\n\n  return {\n    PhysicalResourceId: response.app_id,\n    Data: {\n      appId: response.app_id,\n    },\n  };\n}\n\nfunction isExpired(iat: number) {\n  return (iat - (Date.now() / 1000)) < 0;\n}\n\nfunction getManifestRequest(event: OnEventRequest): ManifestRequest {\n  switch (event.RequestType) {\n    case 'Create':\n      return {\n        manifest: event.ResourceProperties.manifest,\n      };\n    case 'Update':\n      return {\n        app_id: event.PhysicalResourceId,\n        manifest: event.ResourceProperties.manifest,\n      };\n    case 'Delete':\n      return {\n        app_id: event.PhysicalResourceId!,\n      };\n  }\n}\n"]}
@@ -2,7 +2,6 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.SlackAppProvider = void 0;
4
4
  const aws_cdk_lib_1 = require("aws-cdk-lib");
5
- const lambda = require("aws-cdk-lib/aws-lambda");
6
5
  const nodejs = require("aws-cdk-lib/aws-lambda-nodejs");
7
6
  const logs = require("aws-cdk-lib/aws-logs");
8
7
  const cr = require("aws-cdk-lib/custom-resources");
@@ -11,7 +10,6 @@ class SlackAppProvider extends constructs_1.Construct {
11
10
  constructor(scope, id) {
12
11
  super(scope, id);
13
12
  this.handler = new nodejs.NodejsFunction(this, 'handler', {
14
- runtime: lambda.Runtime.NODEJS_12_X,
15
13
  logRetention: logs.RetentionDays.ONE_MONTH,
16
14
  });
17
15
  const provider = new cr.Provider(this, 'Resource', {
@@ -30,4 +28,4 @@ class SlackAppProvider extends constructs_1.Construct {
30
28
  }
31
29
  }
32
30
  exports.SlackAppProvider = SlackAppProvider;
33
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2xhY2stYXBwL3Byb3ZpZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZDQUFvQztBQUNwQyxpREFBaUQ7QUFDakQsd0RBQXdEO0FBQ3hELDZDQUE2QztBQUM3QyxtREFBbUQ7QUFDbkQsMkNBQXVDO0FBRXZDLE1BQWEsZ0JBQWlCLFNBQVEsc0JBQVM7SUFjN0MsWUFBWSxLQUFnQixFQUFFLEVBQVU7UUFDdEMsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLENBQUMsT0FBTyxHQUFHLElBQUksTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQ3hELE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsWUFBWSxFQUFFLElBQUksQ0FBQyxhQUFhLENBQUMsU0FBUztTQUMzQyxDQUFDLENBQUM7UUFFSCxNQUFNLFFBQVEsR0FBRyxJQUFJLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUNqRCxjQUFjLEVBQUUsSUFBSSxDQUFDLE9BQU87U0FDN0IsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFlBQVksR0FBRyxRQUFRLENBQUMsWUFBWSxDQUFDO0lBQzVDLENBQUM7SUExQkQ7O09BRUc7SUFDSSxNQUFNLENBQUMsV0FBVyxDQUFDLEtBQWdCOztRQUN4QyxNQUFNLEtBQUssR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QixNQUFNLEdBQUcsR0FBRyxrQkFBa0IsQ0FBQztRQUMvQixhQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBcUIsbUNBQUksSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDOUYsQ0FBQztDQW9CRjtBQTVCRCw0Q0E0QkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCAqIGFzIG5vZGVqcyBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhLW5vZGVqcyc7XG5pbXBvcnQgKiBhcyBsb2dzIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sb2dzJztcbmltcG9ydCAqIGFzIGNyIGZyb20gJ2F3cy1jZGstbGliL2N1c3RvbS1yZXNvdXJjZXMnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBjbGFzcyBTbGFja0FwcFByb3ZpZGVyIGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBzdGFjay1zaW5nbGV0b24gcmVzb3VyY2UgcHJvdmlkZXJcbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgZ2V0T3JDcmVhdGUoc2NvcGU6IENvbnN0cnVjdCk6IFNsYWNrQXBwUHJvdmlkZXIge1xuICAgIGNvbnN0IHN0YWNrID0gU3RhY2sub2Yoc2NvcGUpO1xuICAgIGNvbnN0IHVpZCA9ICdTbGFja0FwcFByb3ZpZGVyJztcbiAgICByZXR1cm4gc3RhY2subm9kZS50cnlGaW5kQ2hpbGQodWlkKSBhcyBTbGFja0FwcFByb3ZpZGVyID8/IG5ldyBTbGFja0FwcFByb3ZpZGVyKHN0YWNrLCB1aWQpO1xuICB9XG5cbiAgcHVibGljIHJlYWRvbmx5IHNlcnZpY2VUb2tlbjogc3RyaW5nO1xuXG4gIHB1YmxpYyByZWFkb25seSBoYW5kbGVyOiBub2RlanMuTm9kZWpzRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICB0aGlzLmhhbmRsZXIgPSBuZXcgbm9kZWpzLk5vZGVqc0Z1bmN0aW9uKHRoaXMsICdoYW5kbGVyJywge1xuICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzEyX1gsXG4gICAgICBsb2dSZXRlbnRpb246IGxvZ3MuUmV0ZW50aW9uRGF5cy5PTkVfTU9OVEgsXG4gICAgfSk7XG5cbiAgICBjb25zdCBwcm92aWRlciA9IG5ldyBjci5Qcm92aWRlcih0aGlzLCAnUmVzb3VyY2UnLCB7XG4gICAgICBvbkV2ZW50SGFuZGxlcjogdGhpcy5oYW5kbGVyLFxuICAgIH0pO1xuXG4gICAgdGhpcy5zZXJ2aWNlVG9rZW4gPSBwcm92aWRlci5zZXJ2aWNlVG9rZW47XG4gIH1cbn1cbiJdfQ==
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2xhY2stYXBwL3Byb3ZpZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZDQUFvQztBQUNwQyx3REFBd0Q7QUFDeEQsNkNBQTZDO0FBQzdDLG1EQUFtRDtBQUNuRCwyQ0FBdUM7QUFFdkMsTUFBYSxnQkFBaUIsU0FBUSxzQkFBUztJQWM3QyxZQUFZLEtBQWdCLEVBQUUsRUFBVTtRQUN0QyxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxTQUFTLEVBQUU7WUFDeEQsWUFBWSxFQUFFLElBQUksQ0FBQyxhQUFhLENBQUMsU0FBUztTQUMzQyxDQUFDLENBQUM7UUFFSCxNQUFNLFFBQVEsR0FBRyxJQUFJLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUNqRCxjQUFjLEVBQUUsSUFBSSxDQUFDLE9BQU87U0FDN0IsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFlBQVksR0FBRyxRQUFRLENBQUMsWUFBWSxDQUFDO0lBQzVDLENBQUM7SUF6QkQ7O09BRUc7SUFDSSxNQUFNLENBQUMsV0FBVyxDQUFDLEtBQWdCOztRQUN4QyxNQUFNLEtBQUssR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QixNQUFNLEdBQUcsR0FBRyxrQkFBa0IsQ0FBQztRQUMvQixhQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBcUIsbUNBQUksSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDOUYsQ0FBQztDQW1CRjtBQTNCRCw0Q0EyQkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCAqIGFzIG5vZGVqcyBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhLW5vZGVqcyc7XG5pbXBvcnQgKiBhcyBsb2dzIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sb2dzJztcbmltcG9ydCAqIGFzIGNyIGZyb20gJ2F3cy1jZGstbGliL2N1c3RvbS1yZXNvdXJjZXMnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBjbGFzcyBTbGFja0FwcFByb3ZpZGVyIGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBzdGFjay1zaW5nbGV0b24gcmVzb3VyY2UgcHJvdmlkZXJcbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgZ2V0T3JDcmVhdGUoc2NvcGU6IENvbnN0cnVjdCk6IFNsYWNrQXBwUHJvdmlkZXIge1xuICAgIGNvbnN0IHN0YWNrID0gU3RhY2sub2Yoc2NvcGUpO1xuICAgIGNvbnN0IHVpZCA9ICdTbGFja0FwcFByb3ZpZGVyJztcbiAgICByZXR1cm4gc3RhY2subm9kZS50cnlGaW5kQ2hpbGQodWlkKSBhcyBTbGFja0FwcFByb3ZpZGVyID8/IG5ldyBTbGFja0FwcFByb3ZpZGVyKHN0YWNrLCB1aWQpO1xuICB9XG5cbiAgcHVibGljIHJlYWRvbmx5IHNlcnZpY2VUb2tlbjogc3RyaW5nO1xuXG4gIHB1YmxpYyByZWFkb25seSBoYW5kbGVyOiBub2RlanMuTm9kZWpzRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICB0aGlzLmhhbmRsZXIgPSBuZXcgbm9kZWpzLk5vZGVqc0Z1bmN0aW9uKHRoaXMsICdoYW5kbGVyJywge1xuICAgICAgbG9nUmV0ZW50aW9uOiBsb2dzLlJldGVudGlvbkRheXMuT05FX01PTlRILFxuICAgIH0pO1xuXG4gICAgY29uc3QgcHJvdmlkZXIgPSBuZXcgY3IuUHJvdmlkZXIodGhpcywgJ1Jlc291cmNlJywge1xuICAgICAgb25FdmVudEhhbmRsZXI6IHRoaXMuaGFuZGxlcixcbiAgICB9KTtcblxuICAgIHRoaXMuc2VydmljZVRva2VuID0gcHJvdmlkZXIuc2VydmljZVRva2VuO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,127 @@
1
+ import * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';
2
+ import { Construct, IConstruct } from 'constructs';
3
+ import { SlackAppManifestProps } from './manifest';
4
+ /**
5
+ * Properties for a SlackApp.
6
+ *
7
+ * @stability stable
8
+ */
9
+ export interface SlackAppProps {
10
+ /**
11
+ * The definition of the app manifest.
12
+ *
13
+ * @see https://api.slack.com/reference/manifests
14
+ * @stability stable
15
+ */
16
+ readonly manifest: SlackAppManifestDefinition;
17
+ /**
18
+ * An AWS Secrets Manager secret containing the app configuration token.
19
+ *
20
+ * Must use the following JSON format:
21
+ *
22
+ * ```
23
+ * {
24
+ * "refreshToken": "<token>"
25
+ * }
26
+ * ```
27
+ *
28
+ * @stability stable
29
+ */
30
+ readonly configurationTokenSecret: secretsmanager.ISecret;
31
+ /**
32
+ * The AWS Secrets Manager secret where to store the app credentials.
33
+ *
34
+ * @default - a new secret is created
35
+ * @stability stable
36
+ */
37
+ readonly credentialsSecret?: secretsmanager.ISecret;
38
+ }
39
+ /**
40
+ * A Slack app manifest definition.
41
+ *
42
+ * @stability stable
43
+ */
44
+ export declare abstract class SlackAppManifestDefinition {
45
+ /**
46
+ * Create a Slack app manifest from a JSON app manifest encoded as a string.
47
+ *
48
+ * @stability stable
49
+ */
50
+ static fromString(manifest: string): SlackAppManifestDefinition;
51
+ /**
52
+ * Creates a Slack app manifest from a file containg a JSON app manifest.
53
+ *
54
+ * @stability stable
55
+ */
56
+ static fromFile(file: string): SlackAppManifestDefinition;
57
+ /**
58
+ * Creates a Slack app manifest by specifying properties.
59
+ *
60
+ * @stability stable
61
+ */
62
+ static fromManifest(props: SlackAppManifestProps): SlackAppManifestDefinition;
63
+ /**
64
+ * Renders the JSON app manifest encoded as a string.
65
+ *
66
+ * @stability stable
67
+ */
68
+ abstract render(construct: IConstruct): string;
69
+ }
70
+ /**
71
+ * A Slack application deployed with a manifest.
72
+ *
73
+ * @see https://api.slack.com/reference/manifests
74
+ * @stability stable
75
+ */
76
+ export declare class SlackApp extends Construct {
77
+ /**
78
+ * The ID of the application.
79
+ *
80
+ * @stability stable
81
+ */
82
+ readonly appId: string;
83
+ /**
84
+ * An AWS Secrets Manager secret containing the credentials of the application.
85
+ *
86
+ * ```
87
+ * {
88
+ * "appId": "...",
89
+ * "clientId": "...",
90
+ * "clientSecret": "...",
91
+ * "verificationToken": "...",
92
+ * "signingSecret": "..."
93
+ * }
94
+ * ```
95
+ *
96
+ * @stability stable
97
+ */
98
+ readonly credentials: secretsmanager.ISecret;
99
+ /**
100
+ * A dynamic reference to the client ID of the app.
101
+ *
102
+ * @stability stable
103
+ */
104
+ readonly clientId: string;
105
+ /**
106
+ * A dynamic reference to the client secret of the app.
107
+ *
108
+ * @stability stable
109
+ */
110
+ readonly clientSecret: string;
111
+ /**
112
+ * A dynamic reference to the verification token of the app.
113
+ *
114
+ * @stability stable
115
+ */
116
+ readonly verificationToken: string;
117
+ /**
118
+ * A dynamic reference to the signing secret of the app.
119
+ *
120
+ * @stability stable
121
+ */
122
+ readonly signingSecret: string;
123
+ /**
124
+ * @stability stable
125
+ */
126
+ constructor(scope: Construct, id: string, props: SlackAppProps);
127
+ }
@@ -0,0 +1,103 @@
1
+ "use strict";
2
+ var _a, _b;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.SlackApp = exports.SlackAppManifestDefinition = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const fs = require("fs");
7
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
8
+ const secretsmanager = require("aws-cdk-lib/aws-secretsmanager");
9
+ const constructs_1 = require("constructs");
10
+ const manifest_1 = require("./manifest");
11
+ const provider_1 = require("./provider");
12
+ /**
13
+ * A Slack app manifest definition.
14
+ *
15
+ * @stability stable
16
+ */
17
+ class SlackAppManifestDefinition {
18
+ /**
19
+ * Create a Slack app manifest from a JSON app manifest encoded as a string.
20
+ *
21
+ * @stability stable
22
+ */
23
+ static fromString(manifest) {
24
+ return new StringManifest(manifest);
25
+ }
26
+ /**
27
+ * Creates a Slack app manifest from a file containg a JSON app manifest.
28
+ *
29
+ * @stability stable
30
+ */
31
+ static fromFile(file) {
32
+ return new FileManifest(file);
33
+ }
34
+ /**
35
+ * Creates a Slack app manifest by specifying properties.
36
+ *
37
+ * @stability stable
38
+ */
39
+ static fromManifest(props) {
40
+ return new manifest_1.SlackAppManifest(props);
41
+ }
42
+ }
43
+ exports.SlackAppManifestDefinition = SlackAppManifestDefinition;
44
+ _a = JSII_RTTI_SYMBOL_1;
45
+ SlackAppManifestDefinition[_a] = { fqn: "cloudstructs.SlackAppManifestDefinition", version: "0.4.0" };
46
+ class StringManifest extends SlackAppManifestDefinition {
47
+ constructor(manifest) {
48
+ super();
49
+ this.manifest = manifest;
50
+ }
51
+ render(_construct) {
52
+ return this.manifest;
53
+ }
54
+ }
55
+ class FileManifest extends SlackAppManifestDefinition {
56
+ constructor(file) {
57
+ super();
58
+ this.file = file;
59
+ }
60
+ render(_construct) {
61
+ return fs.readFileSync(this.file, 'utf8');
62
+ }
63
+ }
64
+ /**
65
+ * A Slack application deployed with a manifest.
66
+ *
67
+ * @see https://api.slack.com/reference/manifests
68
+ * @stability stable
69
+ */
70
+ class SlackApp extends constructs_1.Construct {
71
+ /**
72
+ * @stability stable
73
+ */
74
+ constructor(scope, id, props) {
75
+ var _c;
76
+ super(scope, id);
77
+ const provider = provider_1.SlackAppProvider.getOrCreate(this);
78
+ props.configurationTokenSecret.grantRead(provider.handler);
79
+ props.configurationTokenSecret.grantWrite(provider.handler);
80
+ this.credentials = (_c = props.credentialsSecret) !== null && _c !== void 0 ? _c : new secretsmanager.Secret(this, 'Credentials', {
81
+ description: `Credentials for Slack App ${this.node.id}`,
82
+ });
83
+ this.credentials.grantWrite(provider.handler);
84
+ const resource = new aws_cdk_lib_1.CustomResource(this, 'Resource', {
85
+ serviceToken: provider.serviceToken,
86
+ resourceType: 'Custom::SlackApp',
87
+ properties: {
88
+ manifest: props.manifest.render(this),
89
+ configurationTokenSecretArn: props.configurationTokenSecret.secretArn,
90
+ credentialsSecretArn: this.credentials.secretArn,
91
+ },
92
+ });
93
+ this.appId = resource.getAttString('appId');
94
+ this.clientId = this.credentials.secretValueFromJson('clientId').toString();
95
+ this.clientSecret = this.credentials.secretValueFromJson('clientSecret').toString();
96
+ this.verificationToken = this.credentials.secretValueFromJson('verificationToken').toString();
97
+ this.signingSecret = this.credentials.secretValueFromJson('signingSecret').toString();
98
+ }
99
+ }
100
+ exports.SlackApp = SlackApp;
101
+ _b = JSII_RTTI_SYMBOL_1;
102
+ SlackApp[_b] = { fqn: "cloudstructs.SlackApp", version: "0.4.0" };
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slack-app.js","sourceRoot":"","sources":["../../src/slack-app/slack-app.ts"],"names":[],"mappings":";;;;;AAAA,yBAAyB;AACzB,6CAA6C;AAC7C,iEAAiE;AACjE,2CAAmD;AACnD,yCAAqE;AACrE,yCAA8C;;;;;;AAe9C,MAAsB,0BAA0B;;;;;;IAEvC,MAAM,CAAC,UAAU,CAAC,QAAgB;QACvC,OAAO,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC;IACtC,CAAC;;;;;;IAGM,MAAM,CAAC,QAAQ,CAAC,IAAY;QACjC,OAAO,IAAI,YAAY,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;;;;;;IAGM,MAAM,CAAC,YAAY,CAAC,KAA4B;QACrD,OAAO,IAAI,2BAAgB,CAAC,KAAK,CAAC,CAAC;IACrC,CAAC;;AAdH,gEAkBC;;;AAED,MAAM,cAAe,SAAQ,0BAA0B;IACrD,YAA6B,QAAgB;QAC3C,KAAK,EAAE,CAAC;QADmB,aAAQ,GAAR,QAAQ,CAAQ;IAE7C,CAAC;IAEM,MAAM,CAAC,UAAsB;QAClC,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;CACF;AAED,MAAM,YAAa,SAAQ,0BAA0B;IACnD,YAA6B,IAAY;QACvC,KAAK,EAAE,CAAC;QADmB,SAAI,GAAJ,IAAI,CAAQ;IAEzC,CAAC;IAEM,MAAM,CAAC,UAAsB;QAClC,OAAO,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IAC5C,CAAC;CACF;;;;;;;AAGD,MAAa,QAAS,SAAQ,sBAAS;;;;IAmBrC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAoB;;QAC5D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,QAAQ,GAAG,2BAAgB,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QACpD,KAAK,CAAC,wBAAwB,CAAC,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAC3D,KAAK,CAAC,wBAAwB,CAAC,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAE5D,IAAI,CAAC,WAAW,SAAG,KAAK,CAAC,iBAAiB,mCAAI,IAAI,cAAc,CAAC,MAAM,CAAC,IAAI,EAAE,aAAa,EAAE;YAC3F,WAAW,EAAE,6BAA6B,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;SACzD,CAAC,CAAC;QACH,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAE9C,MAAM,QAAQ,GAAG,IAAI,4BAAc,CAAC,IAAI,EAAE,UAAU,EAAE;YACpD,YAAY,EAAE,QAAQ,CAAC,YAAY;YACnC,YAAY,EAAE,kBAAkB;YAChC,UAAU,EAAE;gBACV,QAAQ,EAAE,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC;gBACrC,2BAA2B,EAAE,KAAK,CAAC,wBAAwB,CAAC,SAAS;gBACrE,oBAAoB,EAAE,IAAI,CAAC,WAAW,CAAC,SAAS;aACjD;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;QAC5C,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAC,QAAQ,EAAE,CAAC;QAC5E,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,cAAc,CAAC,CAAC,QAAQ,EAAE,CAAC;QACpF,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,mBAAmB,CAAC,CAAC,QAAQ,EAAE,CAAC;QAC9F,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,eAAe,CAAC,CAAC,QAAQ,EAAE,CAAC;IACxF,CAAC;;AA9CH,4BA+CC","sourcesContent":["import * as fs from 'fs';\nimport { CustomResource } from 'aws-cdk-lib';\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\nimport { Construct, IConstruct } from 'constructs';\nimport { SlackAppManifest, SlackAppManifestProps } from './manifest';\nimport { SlackAppProvider } from './provider';\n\n                                    \nexport interface SlackAppProps {\n                                                                                                            \n  readonly manifest: SlackAppManifestDefinition;\n\n                                                                                                                                                                                                         \n  readonly configurationTokenSecret: secretsmanager.ISecret;\n\n                                                                                                                               \n  readonly credentialsSecret?: secretsmanager.ISecret;\n}\n\n                                          \nexport abstract class SlackAppManifestDefinition {\n                                                                                         \n  public static fromString(manifest: string): SlackAppManifestDefinition {\n    return new StringManifest(manifest);\n  }\n\n                                                                                      \n  public static fromFile(file: string): SlackAppManifestDefinition {\n    return new FileManifest(file);\n  }\n\n                                                                      \n  public static fromManifest(props: SlackAppManifestProps): SlackAppManifestDefinition {\n    return new SlackAppManifest(props);\n  }\n\n                                                                  \n  public abstract render(construct: IConstruct): string;\n}\n\nclass StringManifest extends SlackAppManifestDefinition {\n  constructor(private readonly manifest: string) {\n    super();\n  }\n\n  public render(_construct: IConstruct): string {\n    return this.manifest;\n  }\n}\n\nclass FileManifest extends SlackAppManifestDefinition {\n  constructor(private readonly file: string) {\n    super();\n  }\n\n  public render(_construct: IConstruct): string {\n    return fs.readFileSync(this.file, 'utf8');\n  }\n}\n\n                                                                                                            \nexport class SlackApp extends Construct {\n                                          \n  public readonly appId: string;\n\n                                                                                                                                                                                                                                                                                  \n  public readonly credentials: secretsmanager.ISecret;\n\n                                                                \n  public readonly clientId: string;\n\n                                                                    \n  public readonly clientSecret: string;\n\n                                                                         \n  public readonly verificationToken: string;\n\n                                                                     \n  public readonly signingSecret: string;\n\n  constructor(scope: Construct, id: string, props: SlackAppProps) {\n    super(scope, id);\n\n    const provider = SlackAppProvider.getOrCreate(this);\n    props.configurationTokenSecret.grantRead(provider.handler);\n    props.configurationTokenSecret.grantWrite(provider.handler);\n\n    this.credentials = props.credentialsSecret ?? new secretsmanager.Secret(this, 'Credentials', {\n      description: `Credentials for Slack App ${this.node.id}`,\n    });\n    this.credentials.grantWrite(provider.handler);\n\n    const resource = new CustomResource(this, 'Resource', {\n      serviceToken: provider.serviceToken,\n      resourceType: 'Custom::SlackApp',\n      properties: {\n        manifest: props.manifest.render(this),\n        configurationTokenSecretArn: props.configurationTokenSecret.secretArn,\n        credentialsSecretArn: this.credentials.secretArn,\n      },\n    });\n\n    this.appId = resource.getAttString('appId');\n    this.clientId = this.credentials.secretValueFromJson('clientId').toString();\n    this.clientSecret = this.credentials.secretValueFromJson('clientSecret').toString();\n    this.verificationToken = this.credentials.secretValueFromJson('verificationToken').toString();\n    this.signingSecret = this.credentials.secretValueFromJson('signingSecret').toString();\n  }\n}\n"]}