headroom-cms 0.1.1

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 (237) hide show
  1. package/README.md +282 -0
  2. package/admin/assets/AdminsPage-CnrQqwKA.js +1 -0
  3. package/admin/assets/AllContentPage-ByN1h3PP.js +1 -0
  4. package/admin/assets/ApiKeysPage-FgNHZPBS.js +1 -0
  5. package/admin/assets/AuditPage-DAPpo-sj.js +1 -0
  6. package/admin/assets/BlockEditor-CZTwex-o.js +179 -0
  7. package/admin/assets/BlockEditor-Cp_wZ2xN.css +1 -0
  8. package/admin/assets/BlockTypeEditPage-Buuwbx1P.js +1 -0
  9. package/admin/assets/BlockTypesPage-Dj0qmsqX.js +1 -0
  10. package/admin/assets/BulkActionBar-BMcUBJSH.js +1 -0
  11. package/admin/assets/CollectionEditPage-CLgQu2HS.js +1 -0
  12. package/admin/assets/CollectionsPage-BnCaxALz.js +1 -0
  13. package/admin/assets/ContentCreatePage-CJI326o-.js +1 -0
  14. package/admin/assets/ContentEditPage-A4i8P2Jd.js +3 -0
  15. package/admin/assets/ContentListPage-Bc4mBIkB.js +1 -0
  16. package/admin/assets/CustomBlockPreview-CCssn6vF.js +479 -0
  17. package/admin/assets/FieldBuilder-YJGSk0nY.js +3 -0
  18. package/admin/assets/LoginPage-Jrne8-Wr.js +1 -0
  19. package/admin/assets/MediaPage-DfPQBmNf.css +1 -0
  20. package/admin/assets/MediaPage-_qNXqsZg.js +1 -0
  21. package/admin/assets/SiteSettingsPage-CoZnavij.js +1 -0
  22. package/admin/assets/SitesPage-ETqFT3nO.js +1 -0
  23. package/admin/assets/TagsPage-BGpp0XZM.js +1 -0
  24. package/admin/assets/UsersPage-CKRJpAb6.js +1 -0
  25. package/admin/assets/WebhookEditPage-BOcLe5OJ.js +1 -0
  26. package/admin/assets/WebhooksPage-Czco583Y.js +1 -0
  27. package/admin/assets/badge-0Z1nL6DI.js +1 -0
  28. package/admin/assets/card-D1-S-QZ6.js +1 -0
  29. package/admin/assets/check-BGA0ADyt.js +1 -0
  30. package/admin/assets/checkbox-BPqrj_XS.js +1 -0
  31. package/admin/assets/command-ChD319uJ.js +1 -0
  32. package/admin/assets/contentStatus-DfWHjFVB.js +1 -0
  33. package/admin/assets/copy-BqH9rXYM.js +1 -0
  34. package/admin/assets/core.esm-Csvubn5Q.js +5 -0
  35. package/admin/assets/format-CZ9bpk32.js +1 -0
  36. package/admin/assets/index-DOqKbrpW.css +1 -0
  37. package/admin/assets/index-Ds50UTAc.js +18 -0
  38. package/admin/assets/lib-BrI1UB_t.js +38 -0
  39. package/admin/assets/media-url-DIg_vSyf.js +1 -0
  40. package/admin/assets/module-RjUF93sV.js +716 -0
  41. package/admin/assets/native-48B9X9Wg.js +1 -0
  42. package/admin/assets/plus-BgHSYWJN.js +1 -0
  43. package/admin/assets/radix-DQ3amgxj.js +51 -0
  44. package/admin/assets/react-vendor-DNVhVxD7.js +4 -0
  45. package/admin/assets/search-DIzcfCVh.js +1 -0
  46. package/admin/assets/select-CJXZv4wv.js +1 -0
  47. package/admin/assets/sortable.esm-Zh-9QRSf.js +1 -0
  48. package/admin/assets/table-B3EHrN_H.js +1 -0
  49. package/admin/assets/tanstack-BO6c-AOu.js +1 -0
  50. package/admin/assets/trash-2-Gny2Upn-.js +1 -0
  51. package/admin/assets/useAdminResolver-BsQc_N4z.js +1 -0
  52. package/admin/assets/useContent-CSobIico.js +1 -0
  53. package/admin/assets/useDebouncedValue-Bf8UizjU.js +1 -0
  54. package/admin/assets/useMedia-CQnmMz4N.js +1 -0
  55. package/admin/assets/useTags-CYqbj5cK.js +1 -0
  56. package/admin/assets/useWebhooks-DXgtQ3aU.js +1 -0
  57. package/admin/index.html +21 -0
  58. package/admin/vite.svg +1 -0
  59. package/dist/admin-site.d.ts +30 -0
  60. package/dist/admin-site.d.ts.map +1 -0
  61. package/dist/admin-site.js +80 -0
  62. package/dist/admin-site.js.map +1 -0
  63. package/dist/api.d.ts +26 -0
  64. package/dist/api.d.ts.map +1 -0
  65. package/dist/api.js +91 -0
  66. package/dist/api.js.map +1 -0
  67. package/dist/auth.d.ts +27 -0
  68. package/dist/auth.d.ts.map +1 -0
  69. package/dist/auth.js +86 -0
  70. package/dist/auth.js.map +1 -0
  71. package/dist/cdn.d.ts +27 -0
  72. package/dist/cdn.d.ts.map +1 -0
  73. package/dist/cdn.js +382 -0
  74. package/dist/cdn.js.map +1 -0
  75. package/dist/image.d.ts +21 -0
  76. package/dist/image.d.ts.map +1 -0
  77. package/dist/image.js +48 -0
  78. package/dist/image.js.map +1 -0
  79. package/dist/index.d.ts +85 -0
  80. package/dist/index.d.ts.map +1 -0
  81. package/dist/index.js +124 -0
  82. package/dist/index.js.map +1 -0
  83. package/dist/storage.d.ts +21 -0
  84. package/dist/storage.d.ts.map +1 -0
  85. package/dist/storage.js +125 -0
  86. package/dist/storage.js.map +1 -0
  87. package/dist/webhooks.d.ts +23 -0
  88. package/dist/webhooks.d.ts.map +1 -0
  89. package/dist/webhooks.js +91 -0
  90. package/dist/webhooks.js.map +1 -0
  91. package/lambda/api/bootstrap +0 -0
  92. package/lambda/api/resource.enc +0 -0
  93. package/lambda/functions/custom-message/index.mjs +112 -0
  94. package/lambda/functions/custom-message/resource.enc +1 -0
  95. package/lambda/image-lambda/index.mjs +188 -0
  96. package/lambda/image-lambda/node_modules/.package-lock.json +160 -0
  97. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/README.md +46 -0
  98. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/lib/glib-2.0/include/glibconfig.h +220 -0
  99. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/lib/index.js +1 -0
  100. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/lib/libvips-cpp.so.42 +0 -0
  101. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/package.json +42 -0
  102. package/lambda/image-lambda/node_modules/@img/sharp-libvips-linux-arm64/versions.json +30 -0
  103. package/lambda/image-lambda/node_modules/@img/sharp-linux-arm64/LICENSE +191 -0
  104. package/lambda/image-lambda/node_modules/@img/sharp-linux-arm64/README.md +18 -0
  105. package/lambda/image-lambda/node_modules/@img/sharp-linux-arm64/lib/sharp-linux-arm64.node +0 -0
  106. package/lambda/image-lambda/node_modules/@img/sharp-linux-arm64/package.json +46 -0
  107. package/lambda/image-lambda/node_modules/color/LICENSE +21 -0
  108. package/lambda/image-lambda/node_modules/color/README.md +123 -0
  109. package/lambda/image-lambda/node_modules/color/index.js +496 -0
  110. package/lambda/image-lambda/node_modules/color/package.json +47 -0
  111. package/lambda/image-lambda/node_modules/color-convert/CHANGELOG.md +54 -0
  112. package/lambda/image-lambda/node_modules/color-convert/LICENSE +21 -0
  113. package/lambda/image-lambda/node_modules/color-convert/README.md +68 -0
  114. package/lambda/image-lambda/node_modules/color-convert/conversions.js +839 -0
  115. package/lambda/image-lambda/node_modules/color-convert/index.js +81 -0
  116. package/lambda/image-lambda/node_modules/color-convert/package.json +48 -0
  117. package/lambda/image-lambda/node_modules/color-convert/route.js +97 -0
  118. package/lambda/image-lambda/node_modules/color-name/LICENSE +8 -0
  119. package/lambda/image-lambda/node_modules/color-name/README.md +11 -0
  120. package/lambda/image-lambda/node_modules/color-name/index.js +152 -0
  121. package/lambda/image-lambda/node_modules/color-name/package.json +28 -0
  122. package/lambda/image-lambda/node_modules/color-string/LICENSE +21 -0
  123. package/lambda/image-lambda/node_modules/color-string/README.md +62 -0
  124. package/lambda/image-lambda/node_modules/color-string/index.js +242 -0
  125. package/lambda/image-lambda/node_modules/color-string/package.json +39 -0
  126. package/lambda/image-lambda/node_modules/detect-libc/LICENSE +201 -0
  127. package/lambda/image-lambda/node_modules/detect-libc/README.md +163 -0
  128. package/lambda/image-lambda/node_modules/detect-libc/index.d.ts +14 -0
  129. package/lambda/image-lambda/node_modules/detect-libc/lib/detect-libc.js +313 -0
  130. package/lambda/image-lambda/node_modules/detect-libc/lib/elf.js +39 -0
  131. package/lambda/image-lambda/node_modules/detect-libc/lib/filesystem.js +51 -0
  132. package/lambda/image-lambda/node_modules/detect-libc/lib/process.js +24 -0
  133. package/lambda/image-lambda/node_modules/detect-libc/package.json +44 -0
  134. package/lambda/image-lambda/node_modules/is-arrayish/LICENSE +21 -0
  135. package/lambda/image-lambda/node_modules/is-arrayish/README.md +16 -0
  136. package/lambda/image-lambda/node_modules/is-arrayish/index.js +9 -0
  137. package/lambda/image-lambda/node_modules/is-arrayish/package.json +45 -0
  138. package/lambda/image-lambda/node_modules/semver/LICENSE +15 -0
  139. package/lambda/image-lambda/node_modules/semver/README.md +665 -0
  140. package/lambda/image-lambda/node_modules/semver/bin/semver.js +191 -0
  141. package/lambda/image-lambda/node_modules/semver/classes/comparator.js +143 -0
  142. package/lambda/image-lambda/node_modules/semver/classes/index.js +7 -0
  143. package/lambda/image-lambda/node_modules/semver/classes/range.js +557 -0
  144. package/lambda/image-lambda/node_modules/semver/classes/semver.js +333 -0
  145. package/lambda/image-lambda/node_modules/semver/functions/clean.js +8 -0
  146. package/lambda/image-lambda/node_modules/semver/functions/cmp.js +54 -0
  147. package/lambda/image-lambda/node_modules/semver/functions/coerce.js +62 -0
  148. package/lambda/image-lambda/node_modules/semver/functions/compare-build.js +9 -0
  149. package/lambda/image-lambda/node_modules/semver/functions/compare-loose.js +5 -0
  150. package/lambda/image-lambda/node_modules/semver/functions/compare.js +7 -0
  151. package/lambda/image-lambda/node_modules/semver/functions/diff.js +60 -0
  152. package/lambda/image-lambda/node_modules/semver/functions/eq.js +5 -0
  153. package/lambda/image-lambda/node_modules/semver/functions/gt.js +5 -0
  154. package/lambda/image-lambda/node_modules/semver/functions/gte.js +5 -0
  155. package/lambda/image-lambda/node_modules/semver/functions/inc.js +21 -0
  156. package/lambda/image-lambda/node_modules/semver/functions/lt.js +5 -0
  157. package/lambda/image-lambda/node_modules/semver/functions/lte.js +5 -0
  158. package/lambda/image-lambda/node_modules/semver/functions/major.js +5 -0
  159. package/lambda/image-lambda/node_modules/semver/functions/minor.js +5 -0
  160. package/lambda/image-lambda/node_modules/semver/functions/neq.js +5 -0
  161. package/lambda/image-lambda/node_modules/semver/functions/parse.js +18 -0
  162. package/lambda/image-lambda/node_modules/semver/functions/patch.js +5 -0
  163. package/lambda/image-lambda/node_modules/semver/functions/prerelease.js +8 -0
  164. package/lambda/image-lambda/node_modules/semver/functions/rcompare.js +5 -0
  165. package/lambda/image-lambda/node_modules/semver/functions/rsort.js +5 -0
  166. package/lambda/image-lambda/node_modules/semver/functions/satisfies.js +12 -0
  167. package/lambda/image-lambda/node_modules/semver/functions/sort.js +5 -0
  168. package/lambda/image-lambda/node_modules/semver/functions/valid.js +8 -0
  169. package/lambda/image-lambda/node_modules/semver/index.js +91 -0
  170. package/lambda/image-lambda/node_modules/semver/internal/constants.js +37 -0
  171. package/lambda/image-lambda/node_modules/semver/internal/debug.js +11 -0
  172. package/lambda/image-lambda/node_modules/semver/internal/identifiers.js +29 -0
  173. package/lambda/image-lambda/node_modules/semver/internal/lrucache.js +42 -0
  174. package/lambda/image-lambda/node_modules/semver/internal/parse-options.js +17 -0
  175. package/lambda/image-lambda/node_modules/semver/internal/re.js +223 -0
  176. package/lambda/image-lambda/node_modules/semver/package.json +78 -0
  177. package/lambda/image-lambda/node_modules/semver/preload.js +4 -0
  178. package/lambda/image-lambda/node_modules/semver/range.bnf +16 -0
  179. package/lambda/image-lambda/node_modules/semver/ranges/gtr.js +6 -0
  180. package/lambda/image-lambda/node_modules/semver/ranges/intersects.js +9 -0
  181. package/lambda/image-lambda/node_modules/semver/ranges/ltr.js +6 -0
  182. package/lambda/image-lambda/node_modules/semver/ranges/max-satisfying.js +27 -0
  183. package/lambda/image-lambda/node_modules/semver/ranges/min-satisfying.js +26 -0
  184. package/lambda/image-lambda/node_modules/semver/ranges/min-version.js +63 -0
  185. package/lambda/image-lambda/node_modules/semver/ranges/outside.js +82 -0
  186. package/lambda/image-lambda/node_modules/semver/ranges/simplify.js +49 -0
  187. package/lambda/image-lambda/node_modules/semver/ranges/subset.js +249 -0
  188. package/lambda/image-lambda/node_modules/semver/ranges/to-comparators.js +10 -0
  189. package/lambda/image-lambda/node_modules/semver/ranges/valid.js +13 -0
  190. package/lambda/image-lambda/node_modules/sharp/LICENSE +191 -0
  191. package/lambda/image-lambda/node_modules/sharp/README.md +118 -0
  192. package/lambda/image-lambda/node_modules/sharp/install/check.js +41 -0
  193. package/lambda/image-lambda/node_modules/sharp/lib/channel.js +174 -0
  194. package/lambda/image-lambda/node_modules/sharp/lib/colour.js +180 -0
  195. package/lambda/image-lambda/node_modules/sharp/lib/composite.js +210 -0
  196. package/lambda/image-lambda/node_modules/sharp/lib/constructor.js +452 -0
  197. package/lambda/image-lambda/node_modules/sharp/lib/index.d.ts +1754 -0
  198. package/lambda/image-lambda/node_modules/sharp/lib/index.js +16 -0
  199. package/lambda/image-lambda/node_modules/sharp/lib/input.js +658 -0
  200. package/lambda/image-lambda/node_modules/sharp/lib/is.js +169 -0
  201. package/lambda/image-lambda/node_modules/sharp/lib/libvips.js +203 -0
  202. package/lambda/image-lambda/node_modules/sharp/lib/operation.js +958 -0
  203. package/lambda/image-lambda/node_modules/sharp/lib/output.js +1587 -0
  204. package/lambda/image-lambda/node_modules/sharp/lib/resize.js +587 -0
  205. package/lambda/image-lambda/node_modules/sharp/lib/sharp.js +114 -0
  206. package/lambda/image-lambda/node_modules/sharp/lib/utility.js +296 -0
  207. package/lambda/image-lambda/node_modules/sharp/package.json +222 -0
  208. package/lambda/image-lambda/node_modules/sharp/src/binding.gyp +280 -0
  209. package/lambda/image-lambda/node_modules/sharp/src/common.cc +1091 -0
  210. package/lambda/image-lambda/node_modules/sharp/src/common.h +393 -0
  211. package/lambda/image-lambda/node_modules/sharp/src/metadata.cc +320 -0
  212. package/lambda/image-lambda/node_modules/sharp/src/metadata.h +85 -0
  213. package/lambda/image-lambda/node_modules/sharp/src/operations.cc +475 -0
  214. package/lambda/image-lambda/node_modules/sharp/src/operations.h +125 -0
  215. package/lambda/image-lambda/node_modules/sharp/src/pipeline.cc +1758 -0
  216. package/lambda/image-lambda/node_modules/sharp/src/pipeline.h +393 -0
  217. package/lambda/image-lambda/node_modules/sharp/src/sharp.cc +40 -0
  218. package/lambda/image-lambda/node_modules/sharp/src/stats.cc +183 -0
  219. package/lambda/image-lambda/node_modules/sharp/src/stats.h +59 -0
  220. package/lambda/image-lambda/node_modules/sharp/src/utilities.cc +269 -0
  221. package/lambda/image-lambda/node_modules/sharp/src/utilities.h +19 -0
  222. package/lambda/image-lambda/node_modules/simple-swizzle/LICENSE +21 -0
  223. package/lambda/image-lambda/node_modules/simple-swizzle/README.md +43 -0
  224. package/lambda/image-lambda/node_modules/simple-swizzle/index.js +29 -0
  225. package/lambda/image-lambda/node_modules/simple-swizzle/package.json +36 -0
  226. package/lambda/webhook-worker/bootstrap +0 -0
  227. package/lambda/webhook-worker/resource.enc +0 -0
  228. package/package.json +50 -0
  229. package/src/admin-site.ts +108 -0
  230. package/src/api.ts +113 -0
  231. package/src/auth.ts +110 -0
  232. package/src/cdn.ts +449 -0
  233. package/src/image.ts +62 -0
  234. package/src/index.ts +216 -0
  235. package/src/sst-env.d.ts +143 -0
  236. package/src/storage.ts +138 -0
  237. package/src/webhooks.ts +114 -0
package/dist/api.js ADDED
@@ -0,0 +1,91 @@
1
+ /**
2
+ * API Infrastructure
3
+ *
4
+ * Go Lambda function serving the Headroom API with access to all resources.
5
+ * Supports dev mode (Go source with live reload) and package mode (pre-compiled binary).
6
+ */
7
+ import path from "path";
8
+ export function createApi(name, args) {
9
+ const { storage, auth, webhooks, image } = args;
10
+ const handlerConfig = args.dev
11
+ ? {
12
+ handler: args.dev.handler,
13
+ runtime: "go",
14
+ }
15
+ : {
16
+ bundle: path.join(args.pkgRoot, "lambda/api"),
17
+ handler: "bootstrap",
18
+ runtime: "provided.al2023",
19
+ architecture: "arm64",
20
+ };
21
+ const api = new sst.aws.Function(`${name}Api`, {
22
+ ...handlerConfig,
23
+ timeout: "60 seconds",
24
+ url: {
25
+ cors: false,
26
+ },
27
+ environment: {
28
+ SITES_TABLE: storage.sites.name,
29
+ CONTENT_TABLE: storage.content.name,
30
+ DRAFT_CONTENT_TABLE: storage.draftContent.name,
31
+ BLOCKS_TABLE: storage.blocks.name,
32
+ MEDIA_TABLE: storage.media.name,
33
+ COLLECTIONS_TABLE: storage.collections.name,
34
+ BLOCK_TYPES_TABLE: storage.blockTypes.name,
35
+ ADMIN_AUDIT_TABLE: storage.adminAudit.name,
36
+ CONTENT_BUCKET: storage.contentBucket.name,
37
+ USER_POOL_ID: auth.userPool.id,
38
+ USER_POOL_CLIENT_ID: auth.userPoolClient.id,
39
+ KVS_ARN: storage.kvs.arn,
40
+ AWS_REGION_NAME: aws.getRegionOutput().name,
41
+ WEBHOOKS_TABLE: webhooks.webhooks.name,
42
+ WEBHOOK_DELIVERIES_TABLE: webhooks.webhookDeliveries.name,
43
+ WEBHOOK_QUEUE_URL: webhooks.webhookDeliveryQueue.url,
44
+ IMAGE_SIGNING_SECRET: image.imageSigningSecret.value,
45
+ IMAGE_LAMBDA_NAME: image.imageLambda.name,
46
+ RELATIONSHIPS_TABLE: storage.relationships.name,
47
+ },
48
+ link: [
49
+ storage.sites,
50
+ storage.content,
51
+ storage.draftContent,
52
+ storage.blocks,
53
+ storage.media,
54
+ storage.collections,
55
+ storage.blockTypes,
56
+ storage.adminAudit,
57
+ storage.contentBucket,
58
+ webhooks.webhooks,
59
+ webhooks.webhookDeliveries,
60
+ webhooks.webhookDeliveryQueue,
61
+ image.imageSigningSecret,
62
+ storage.relationships,
63
+ ],
64
+ permissions: [
65
+ {
66
+ actions: [
67
+ "cloudfront-keyvaluestore:DescribeKeyValueStore",
68
+ "cloudfront-keyvaluestore:PutKey",
69
+ "cloudfront-keyvaluestore:DeleteKey",
70
+ "cloudfront-keyvaluestore:GetKey",
71
+ ],
72
+ resources: [storage.kvs.arn],
73
+ },
74
+ {
75
+ actions: [
76
+ "cognito-idp:ListUsers",
77
+ "cognito-idp:AdminCreateUser",
78
+ "cognito-idp:AdminDeleteUser",
79
+ "cognito-idp:AdminGetUser",
80
+ ],
81
+ resources: [auth.userPool.arn],
82
+ },
83
+ {
84
+ actions: ["lambda:InvokeFunction"],
85
+ resources: [image.imageLambda.arn],
86
+ },
87
+ ],
88
+ });
89
+ return { api };
90
+ }
91
+ //# sourceMappingURL=api.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"api.js","sourceRoot":"","sources":["../src/api.ts"],"names":[],"mappings":"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"}
package/dist/auth.d.ts ADDED
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Authentication Infrastructure
3
+ *
4
+ * Cognito User Pool, SES email identity, and custom message Lambda.
5
+ * Supports dev mode (source handler) and package mode (pre-bundled handler).
6
+ */
7
+ export interface AuthArgs {
8
+ senderEmail: string;
9
+ passwordPolicy?: {
10
+ minimumLength?: number;
11
+ requireLowercase?: boolean;
12
+ requireUppercase?: boolean;
13
+ requireNumbers?: boolean;
14
+ requireSymbols?: boolean;
15
+ };
16
+ pkgRoot: string;
17
+ dev?: {
18
+ /** SST handler string, e.g. "packages/functions/custom-message.handler" */
19
+ handler: string;
20
+ };
21
+ }
22
+ export declare function createAuth(name: string, args: AuthArgs): {
23
+ userPool: sst.aws.CognitoUserPool;
24
+ userPoolClient: sst.aws.CognitoUserPoolClient;
25
+ };
26
+ export type AuthResources = ReturnType<typeof createAuth>;
27
+ //# sourceMappingURL=auth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.d.ts","sourceRoot":"","sources":["../src/auth.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,MAAM,WAAW,QAAQ;IACvB,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,CAAC,EAAE;QACf,aAAa,CAAC,EAAE,MAAM,CAAC;QACvB,gBAAgB,CAAC,EAAE,OAAO,CAAC;QAC3B,gBAAgB,CAAC,EAAE,OAAO,CAAC;QAC3B,cAAc,CAAC,EAAE,OAAO,CAAC;QACzB,cAAc,CAAC,EAAE,OAAO,CAAC;KAC1B,CAAC;IACF,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,CAAC,EAAE;QACJ,2EAA2E;QAC3E,OAAO,EAAE,MAAM,CAAC;KACjB,CAAC;CACH;AAED,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ;;;EAkFtD;AAED,MAAM,MAAM,aAAa,GAAG,UAAU,CAAC,OAAO,UAAU,CAAC,CAAC"}
package/dist/auth.js ADDED
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Authentication Infrastructure
3
+ *
4
+ * Cognito User Pool, SES email identity, and custom message Lambda.
5
+ * Supports dev mode (source handler) and package mode (pre-bundled handler).
6
+ */
7
+ import path from "path";
8
+ export function createAuth(name, args) {
9
+ const emailIdentity = new aws.ses.EmailIdentity(`${name}EmailIdentity`, {
10
+ email: args.senderEmail,
11
+ });
12
+ // Custom message Lambda: customizes Cognito invite emails
13
+ let customMessageFn;
14
+ if (args.dev) {
15
+ customMessageFn = new sst.aws.Function(`${name}CustomMessage`, {
16
+ handler: args.dev.handler,
17
+ runtime: "nodejs20.x",
18
+ });
19
+ }
20
+ else {
21
+ customMessageFn = new sst.aws.Function(`${name}CustomMessage`, {
22
+ bundle: path.join(args.pkgRoot, "lambda/functions/custom-message"),
23
+ handler: "index.handler",
24
+ runtime: "nodejs20.x",
25
+ });
26
+ }
27
+ const pp = args.passwordPolicy ?? {};
28
+ const userPool = new sst.aws.CognitoUserPool(`${name}UserPool`, {
29
+ usernames: ["email"],
30
+ transform: {
31
+ userPool: {
32
+ passwordPolicy: {
33
+ minimumLength: pp.minimumLength ?? 12,
34
+ requireLowercase: pp.requireLowercase ?? true,
35
+ requireUppercase: pp.requireUppercase ?? true,
36
+ requireNumbers: pp.requireNumbers ?? true,
37
+ requireSymbols: pp.requireSymbols ?? true,
38
+ },
39
+ accountRecoverySetting: {
40
+ recoveryMechanisms: [{ name: "verified_email", priority: 1 }],
41
+ },
42
+ lambdaConfig: {
43
+ customMessage: customMessageFn.arn,
44
+ },
45
+ emailConfiguration: {
46
+ emailSendingAccount: "DEVELOPER",
47
+ sourceArn: emailIdentity.arn,
48
+ fromEmailAddress: `Headroom CMS <${args.senderEmail}>`,
49
+ },
50
+ mfaConfiguration: "OPTIONAL",
51
+ softwareTokenMfaConfiguration: {
52
+ enabled: true,
53
+ },
54
+ },
55
+ },
56
+ });
57
+ // Grant Cognito permission to invoke the custom message Lambda
58
+ new aws.lambda.Permission(`${name}CustomMessagePermission`, {
59
+ action: "lambda:InvokeFunction",
60
+ function: customMessageFn.arn,
61
+ principal: "cognito-idp.amazonaws.com",
62
+ sourceArn: userPool.arn,
63
+ });
64
+ const userPoolClient = userPool.addClient(`${name}Client`, {
65
+ transform: {
66
+ client: {
67
+ explicitAuthFlows: [
68
+ "ALLOW_USER_SRP_AUTH",
69
+ "ALLOW_REFRESH_TOKEN_AUTH",
70
+ "ALLOW_ADMIN_USER_PASSWORD_AUTH",
71
+ ],
72
+ accessTokenValidity: 1,
73
+ idTokenValidity: 1,
74
+ refreshTokenValidity: 365,
75
+ tokenValidityUnits: {
76
+ accessToken: "hours",
77
+ idToken: "hours",
78
+ refreshToken: "days",
79
+ },
80
+ preventUserExistenceErrors: "ENABLED",
81
+ },
82
+ },
83
+ });
84
+ return { userPool, userPoolClient };
85
+ }
86
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../src/auth.ts"],"names":[],"mappings":"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"}
package/dist/cdn.d.ts ADDED
@@ -0,0 +1,27 @@
1
+ /**
2
+ * CDN Infrastructure
3
+ *
4
+ * CloudFront distribution with edge authentication, version-based caching,
5
+ * direct S3 media serving, and Sharp Lambda image transforms.
6
+ */
7
+ import type { StorageResources } from "./storage.js";
8
+ import type { ApiResources } from "./api.js";
9
+ import type { ImageResources } from "./image.js";
10
+ export interface CdnArgs {
11
+ api: ApiResources;
12
+ image: ImageResources;
13
+ contentBucket: StorageResources["contentBucket"];
14
+ kvs: StorageResources["kvs"];
15
+ priceClass?: "PriceClass_100" | "PriceClass_200" | "PriceClass_All";
16
+ apiCacheTtl?: number;
17
+ domain?: {
18
+ name: string;
19
+ certificateArn: string;
20
+ };
21
+ }
22
+ export declare function createCdn(name: string, args: CdnArgs): {
23
+ distribution: aws.cloudfront.Distribution;
24
+ url: PulumiOutput<string>;
25
+ };
26
+ export type CdnResources = ReturnType<typeof createCdn>;
27
+ //# sourceMappingURL=cdn.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cdn.d.ts","sourceRoot":"","sources":["../src/cdn.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AACrD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AAC7C,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAEjD,MAAM,WAAW,OAAO;IACtB,GAAG,EAAE,YAAY,CAAC;IAClB,KAAK,EAAE,cAAc,CAAC;IACtB,aAAa,EAAE,gBAAgB,CAAC,eAAe,CAAC,CAAC;IACjD,GAAG,EAAE,gBAAgB,CAAC,KAAK,CAAC,CAAC;IAC7B,UAAU,CAAC,EAAE,gBAAgB,GAAG,gBAAgB,GAAG,gBAAgB,CAAC;IACpE,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,MAAM,CAAC,EAAE;QACP,IAAI,EAAE,MAAM,CAAC;QACb,cAAc,EAAE,MAAM,CAAC;KACxB,CAAC;CACH;AAED,wBAAgB,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO;;;EAsapD;AAED,MAAM,MAAM,YAAY,GAAG,UAAU,CAAC,OAAO,SAAS,CAAC,CAAC"}
package/dist/cdn.js ADDED
@@ -0,0 +1,382 @@
1
+ /**
2
+ * CDN Infrastructure
3
+ *
4
+ * CloudFront distribution with edge authentication, version-based caching,
5
+ * direct S3 media serving, and Sharp Lambda image transforms.
6
+ */
7
+ export function createCdn(name, args) {
8
+ const { api, image, contentBucket, kvs } = args;
9
+ const apiCacheTtl = args.apiCacheTtl ?? 3600;
10
+ // Extract KVS UUID from ARN (cf.kvs() needs the UUID, not the name)
11
+ const kvsUuid = kvs.arn.apply((arn) => arn.split("/").pop());
12
+ // =========================================================================
13
+ // CloudFront Function: Edge Auth
14
+ // =========================================================================
15
+ const edgeFunction = new aws.cloudfront.Function(`${name}EdgeAuth`, {
16
+ name: $interpolate `${$app.name}-${$app.stage}-edge-auth`,
17
+ runtime: "cloudfront-js-2.0",
18
+ publish: true,
19
+ keyValueStoreAssociations: [kvs.arn],
20
+ code: $interpolate `
21
+ import cf from 'cloudfront';
22
+ import crypto from 'crypto';
23
+
24
+ const kvsId = '${kvsUuid}';
25
+ const kvsHandle = cf.kvs(kvsId);
26
+
27
+ async function handler(event) {
28
+ const request = event.request;
29
+ const uri = request.uri;
30
+ const headers = request.headers;
31
+
32
+ // Skip auth for admin paths (use JWT auth at origin)
33
+ if (uri.startsWith('/v1/admin/')) {
34
+ return request;
35
+ }
36
+
37
+ // Skip auth for health endpoint
38
+ if (uri === '/health') {
39
+ return request;
40
+ }
41
+
42
+ // Skip auth for non-API paths
43
+ if (!uri.startsWith('/v1/')) {
44
+ return request;
45
+ }
46
+
47
+ // Extract site from path: /v1/{site}/...
48
+ var pathParts = uri.split('/');
49
+ if (pathParts.length < 3) {
50
+ return {
51
+ statusCode: 400,
52
+ statusDescription: 'Bad Request',
53
+ body: { encoding: 'text', data: 'Invalid path' }
54
+ };
55
+ }
56
+
57
+ var site = pathParts[2];
58
+
59
+ // Get API key from header
60
+ var apiKeyHeader = headers['x-headroom-key'];
61
+ if (!apiKeyHeader) {
62
+ return {
63
+ statusCode: 401,
64
+ statusDescription: 'Unauthorized',
65
+ headers: { 'content-type': { value: 'application/json' } },
66
+ body: { encoding: 'text', data: JSON.stringify({ error: 'API key required' }) }
67
+ };
68
+ }
69
+
70
+ // Hash the API key (needed for KVS lookup key)
71
+ var keyHash = crypto.createHash('sha256').update(apiKeyHeader.value).digest('hex');
72
+ var hashPrefix = keyHash.substring(0, 8);
73
+
74
+ // Fetch API key hash and version in parallel
75
+ var results = await Promise.all([
76
+ kvsHandle.get('site:' + site + ':key:' + hashPrefix).catch(function() { return null; }),
77
+ kvsHandle.get('site:' + site + ':version').catch(function() { return '0'; }),
78
+ ]);
79
+
80
+ var storedHash = results[0];
81
+ var versionResult = results[1];
82
+
83
+ // Validate API key - compare full hash
84
+ if (!storedHash || storedHash !== keyHash) {
85
+ return {
86
+ statusCode: 403,
87
+ statusDescription: 'Forbidden',
88
+ headers: { 'content-type': { value: 'application/json' } },
89
+ body: { encoding: 'text', data: JSON.stringify({ error: 'Invalid API key' }) }
90
+ };
91
+ }
92
+
93
+ // Add version header for cache key discrimination
94
+ request.headers['x-site-version'] = { value: versionResult || '0' };
95
+
96
+ return request;
97
+ }
98
+ `,
99
+ });
100
+ // =========================================================================
101
+ // CloudFront Function: Media Rewrite
102
+ // =========================================================================
103
+ const mediaRewriteFunction = new aws.cloudfront.Function(`${name}MediaRewrite`, {
104
+ name: $interpolate `${$app.name}-${$app.stage}-media-rewrite`,
105
+ runtime: "cloudfront-js-2.0",
106
+ publish: true,
107
+ code: `
108
+ function handler(event) {
109
+ var request = event.request;
110
+ var uri = request.uri;
111
+
112
+ // Rewrite /media/{site}/{mediaId}/{file} → /sites/{site}/media/{mediaId}/{file}
113
+ var parts = uri.split('/');
114
+ // parts: ['', 'media', '{site}', '{mediaId}', '{file}']
115
+ if (parts.length >= 5 && parts[1] === 'media') {
116
+ var site = parts[2];
117
+ var mediaId = parts[3];
118
+ var rest = parts.slice(4).join('/');
119
+ request.uri = '/sites/' + site + '/media/' + mediaId + '/' + rest;
120
+ }
121
+
122
+ return request;
123
+ }
124
+ `,
125
+ });
126
+ // =========================================================================
127
+ // Cache Policies
128
+ // =========================================================================
129
+ const apiCachePolicy = new aws.cloudfront.CachePolicy(`${name}APICachePolicy`, {
130
+ name: $interpolate `${$app.name}-${$app.stage}-api-cache`,
131
+ comment: "Cache policy for Headroom API with version-based cache busting",
132
+ defaultTtl: apiCacheTtl,
133
+ maxTtl: 86400,
134
+ minTtl: 0,
135
+ parametersInCacheKeyAndForwardedToOrigin: {
136
+ cookiesConfig: { cookieBehavior: "none" },
137
+ headersConfig: {
138
+ headerBehavior: "whitelist",
139
+ headers: { items: ["x-site-version"] },
140
+ },
141
+ queryStringsConfig: {
142
+ queryStringBehavior: "whitelist",
143
+ queryStrings: {
144
+ items: ["collection", "limit", "cursor", "before", "after", "tag"],
145
+ },
146
+ },
147
+ enableAcceptEncodingBrotli: true,
148
+ enableAcceptEncodingGzip: true,
149
+ },
150
+ });
151
+ const imageCachePolicy = new aws.cloudfront.CachePolicy(`${name}ImageCachePolicy`, {
152
+ name: $interpolate `${$app.name}-${$app.stage}-image-cache`,
153
+ comment: "Cache policy for transformed images (immutable)",
154
+ defaultTtl: 31536000,
155
+ maxTtl: 31536000,
156
+ minTtl: 31536000,
157
+ parametersInCacheKeyAndForwardedToOrigin: {
158
+ cookiesConfig: { cookieBehavior: "none" },
159
+ headersConfig: { headerBehavior: "none" },
160
+ queryStringsConfig: {
161
+ queryStringBehavior: "whitelist",
162
+ queryStrings: {
163
+ items: ["w", "h", "fit", "format", "q", "sig"],
164
+ },
165
+ },
166
+ enableAcceptEncodingBrotli: true,
167
+ enableAcceptEncodingGzip: true,
168
+ },
169
+ });
170
+ const mediaCachePolicy = new aws.cloudfront.CachePolicy(`${name}MediaCachePolicy`, {
171
+ name: $interpolate `${$app.name}-${$app.stage}-media-cache`,
172
+ comment: "Cache policy for original media files (immutable)",
173
+ defaultTtl: 31536000,
174
+ maxTtl: 31536000,
175
+ minTtl: 31536000,
176
+ parametersInCacheKeyAndForwardedToOrigin: {
177
+ cookiesConfig: { cookieBehavior: "none" },
178
+ headersConfig: { headerBehavior: "none" },
179
+ queryStringsConfig: { queryStringBehavior: "none" },
180
+ enableAcceptEncodingBrotli: true,
181
+ enableAcceptEncodingGzip: true,
182
+ },
183
+ });
184
+ // =========================================================================
185
+ // Origin Request Policy
186
+ // =========================================================================
187
+ const originRequestPolicy = new aws.cloudfront.OriginRequestPolicy(`${name}OriginRequestPolicy`, {
188
+ name: $interpolate `${$app.name}-${$app.stage}-origin-request`,
189
+ comment: "Forward necessary headers to Lambda origin",
190
+ cookiesConfig: { cookieBehavior: "none" },
191
+ headersConfig: {
192
+ headerBehavior: "whitelist",
193
+ headers: {
194
+ items: [
195
+ "x-headroom-key",
196
+ "content-type",
197
+ "accept",
198
+ "x-site-version",
199
+ ],
200
+ },
201
+ },
202
+ queryStringsConfig: { queryStringBehavior: "all" },
203
+ });
204
+ // Managed AllViewerExceptHostHeader policy for admin paths
205
+ const adminOriginRequestPolicyId = "b689b0a8-53d0-40ab-baf2-68738e2966ac";
206
+ // =========================================================================
207
+ // Origin Access Controls (OAC)
208
+ // =========================================================================
209
+ const mediaOAC = new aws.cloudfront.OriginAccessControl(`${name}MediaOAC`, {
210
+ name: $interpolate `${$app.name}-${$app.stage}-media-oac`,
211
+ description: "OAC for S3 media origin",
212
+ originAccessControlOriginType: "s3",
213
+ signingBehavior: "always",
214
+ signingProtocol: "sigv4",
215
+ });
216
+ const imageOAC = new aws.cloudfront.OriginAccessControl(`${name}ImageOAC`, {
217
+ name: $interpolate `${$app.name}-${$app.stage}-image-oac`,
218
+ description: "OAC for image transform Lambda origin",
219
+ originAccessControlOriginType: "lambda",
220
+ signingBehavior: "always",
221
+ signingProtocol: "sigv4",
222
+ });
223
+ // =========================================================================
224
+ // CloudFront Distribution
225
+ // =========================================================================
226
+ const originDomain = api.api.url.apply((url) => {
227
+ const parsed = new URL(url);
228
+ return parsed.hostname;
229
+ });
230
+ const imageLambdaDomain = image.imageLambda.url.apply((url) => {
231
+ const parsed = new URL(url);
232
+ return parsed.hostname;
233
+ });
234
+ const s3RegionalDomain = $interpolate `${contentBucket.name}.s3.${aws.getRegionOutput().name}.amazonaws.com`;
235
+ const priceClass = args.priceClass ?? "PriceClass_100";
236
+ // Build aliases and certificate config for custom domain
237
+ const aliases = args.domain ? [args.domain.name] : undefined;
238
+ const viewerCertificate = args.domain
239
+ ? {
240
+ acmCertificateArn: args.domain.certificateArn,
241
+ sslSupportMethod: "sni-only",
242
+ minimumProtocolVersion: "TLSv1.2_2021",
243
+ }
244
+ : {
245
+ cloudfrontDefaultCertificate: true,
246
+ };
247
+ const distribution = new aws.cloudfront.Distribution(`${name}Distribution`, {
248
+ enabled: true,
249
+ comment: $interpolate `Headroom CMS API - ${$app.stage}`,
250
+ httpVersion: "http2and3",
251
+ priceClass,
252
+ aliases,
253
+ origins: [
254
+ {
255
+ originId: "api",
256
+ domainName: originDomain,
257
+ customOriginConfig: {
258
+ httpPort: 80,
259
+ httpsPort: 443,
260
+ originProtocolPolicy: "https-only",
261
+ originSslProtocols: ["TLSv1.2"],
262
+ },
263
+ },
264
+ {
265
+ originId: "media-s3",
266
+ domainName: s3RegionalDomain,
267
+ originAccessControlId: mediaOAC.id,
268
+ s3OriginConfig: {
269
+ originAccessIdentity: "",
270
+ },
271
+ },
272
+ {
273
+ originId: "image-lambda",
274
+ domainName: imageLambdaDomain,
275
+ originAccessControlId: imageOAC.id,
276
+ customOriginConfig: {
277
+ httpPort: 80,
278
+ httpsPort: 443,
279
+ originProtocolPolicy: "https-only",
280
+ originSslProtocols: ["TLSv1.2"],
281
+ },
282
+ },
283
+ ],
284
+ defaultCacheBehavior: {
285
+ targetOriginId: "api",
286
+ viewerProtocolPolicy: "redirect-to-https",
287
+ allowedMethods: [
288
+ "GET",
289
+ "HEAD",
290
+ "OPTIONS",
291
+ "PUT",
292
+ "POST",
293
+ "PATCH",
294
+ "DELETE",
295
+ ],
296
+ cachedMethods: ["GET", "HEAD", "OPTIONS"],
297
+ compress: true,
298
+ cachePolicyId: apiCachePolicy.id,
299
+ originRequestPolicyId: originRequestPolicy.id,
300
+ functionAssociations: [
301
+ {
302
+ eventType: "viewer-request",
303
+ functionArn: edgeFunction.arn,
304
+ },
305
+ ],
306
+ },
307
+ orderedCacheBehaviors: [
308
+ // Admin paths: no caching, forwards Authorization header
309
+ {
310
+ pathPattern: "/v1/admin/*",
311
+ targetOriginId: "api",
312
+ viewerProtocolPolicy: "redirect-to-https",
313
+ allowedMethods: [
314
+ "GET",
315
+ "HEAD",
316
+ "OPTIONS",
317
+ "PUT",
318
+ "POST",
319
+ "PATCH",
320
+ "DELETE",
321
+ ],
322
+ cachedMethods: ["GET", "HEAD", "OPTIONS"],
323
+ compress: true,
324
+ cachePolicyId: "4135ea2d-6df8-44a3-9df3-4b5a84be39ad",
325
+ originRequestPolicyId: adminOriginRequestPolicyId,
326
+ },
327
+ // Media originals: served directly from S3 via OAC
328
+ {
329
+ pathPattern: "/media/*",
330
+ targetOriginId: "media-s3",
331
+ viewerProtocolPolicy: "redirect-to-https",
332
+ allowedMethods: ["GET", "HEAD", "OPTIONS"],
333
+ cachedMethods: ["GET", "HEAD", "OPTIONS"],
334
+ compress: true,
335
+ cachePolicyId: mediaCachePolicy.id,
336
+ functionAssociations: [
337
+ {
338
+ eventType: "viewer-request",
339
+ functionArn: mediaRewriteFunction.arn,
340
+ },
341
+ ],
342
+ },
343
+ // Image transforms: served via Sharp Lambda
344
+ {
345
+ pathPattern: "/img/*",
346
+ targetOriginId: "image-lambda",
347
+ viewerProtocolPolicy: "redirect-to-https",
348
+ allowedMethods: ["GET", "HEAD", "OPTIONS"],
349
+ cachedMethods: ["GET", "HEAD", "OPTIONS"],
350
+ compress: true,
351
+ cachePolicyId: imageCachePolicy.id,
352
+ },
353
+ // Health endpoint: no caching, no auth
354
+ {
355
+ pathPattern: "/health",
356
+ targetOriginId: "api",
357
+ viewerProtocolPolicy: "redirect-to-https",
358
+ allowedMethods: ["GET", "HEAD"],
359
+ cachedMethods: ["GET", "HEAD"],
360
+ compress: true,
361
+ cachePolicyId: "4135ea2d-6df8-44a3-9df3-4b5a84be39ad",
362
+ originRequestPolicyId: originRequestPolicy.id,
363
+ },
364
+ ],
365
+ restrictions: {
366
+ geoRestriction: { restrictionType: "none" },
367
+ },
368
+ viewerCertificate,
369
+ });
370
+ // Allow CloudFront to invoke the image Lambda via OAC
371
+ new aws.lambda.Permission(`${name}ImageLambdaCFPermission`, {
372
+ action: "lambda:InvokeFunctionUrl",
373
+ function: image.imageLambda.name,
374
+ principal: "cloudfront.amazonaws.com",
375
+ sourceArn: distribution.arn,
376
+ });
377
+ const url = args.domain
378
+ ? $interpolate `https://${args.domain.name}`
379
+ : $interpolate `https://${distribution.domainName}`;
380
+ return { distribution, url };
381
+ }
382
+ //# sourceMappingURL=cdn.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cdn.js","sourceRoot":"","sources":["../src/cdn.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ /**
2
+ * Image Infrastructure
3
+ *
4
+ * Sharp-based image transformation Lambda with HMAC-signed URLs.
5
+ * Supports dev mode (Node.js source) and package mode (pre-bundled handler).
6
+ */
7
+ import type { StorageResources } from "./storage.js";
8
+ export interface ImageArgs {
9
+ contentBucket: StorageResources["contentBucket"];
10
+ pkgRoot: string;
11
+ dev?: {
12
+ /** SST handler string, e.g. "packages/image-lambda/index.handler" */
13
+ handler: string;
14
+ };
15
+ }
16
+ export declare function createImage(name: string, args: ImageArgs): {
17
+ imageSigningSecret: sst.Secret;
18
+ imageLambda: sst.aws.Function;
19
+ };
20
+ export type ImageResources = ReturnType<typeof createImage>;
21
+ //# sourceMappingURL=image.d.ts.map