@lambdaimg/alchemy 0.1.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.
package/README.md ADDED
@@ -0,0 +1,26 @@
1
+ # @lambdaimg/alchemy
2
+
3
+ AWS Alchemy stack for LambdaImg.
4
+
5
+ ```ts
6
+ import { ImagesStack } from "@lambdaimg/alchemy";
7
+ import * as Alchemy from "alchemy";
8
+ import * as AWS from "alchemy/AWS";
9
+ import * as Effect from "effect/Effect";
10
+
11
+ export default Alchemy.Stack(
12
+ "images",
13
+ {
14
+ providers: AWS.providers(),
15
+ state: AWS.state(),
16
+ },
17
+ Effect.gen(function* () {
18
+ return yield* ImagesStack("Images", {
19
+ domain: "images.example.com",
20
+ hostedZoneId: "Z0000000000000",
21
+ });
22
+ }),
23
+ );
24
+ ```
25
+
26
+ This package is AWS Lambda only. LambdaImg uses Sharp native binaries for WebP generation, so Cloudflare Workers are intentionally out of scope.
@@ -0,0 +1,5 @@
1
+ import * as S3 from "alchemy/AWS/S3";
2
+ import { Stack } from "alchemy/Stack";
3
+ import * as Effect from "effect/Effect";
4
+ export declare const imagesBucket: Effect.Effect<S3.Bucket, never, import("alchemy/AWS/Providers").Providers | Stack>;
5
+ //# sourceMappingURL=images-bucket.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"images-bucket.d.ts","sourceRoot":"","sources":["../src/images-bucket.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACtC,OAAO,KAAK,MAAM,MAAM,eAAe,CAAC;AAExC,eAAO,MAAM,YAAY,oFAYvB,CAAC"}
@@ -0,0 +1,16 @@
1
+ import * as S3 from "alchemy/AWS/S3";
2
+ import { Stack } from "alchemy/Stack";
3
+ import * as Effect from "effect/Effect";
4
+ export const imagesBucket = Effect.gen(function* () {
5
+ const stack = yield* Stack;
6
+ return yield* S3.Bucket("ImagesBucket", {
7
+ forceDestroy: stack.stage === "dev",
8
+ publicAccessBlock: {
9
+ blockPublicAcls: true,
10
+ ignorePublicAcls: true,
11
+ blockPublicPolicy: true,
12
+ restrictPublicBuckets: true,
13
+ },
14
+ });
15
+ });
16
+ //# sourceMappingURL=images-bucket.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"images-bucket.js","sourceRoot":"","sources":["../src/images-bucket.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACtC,OAAO,KAAK,MAAM,MAAM,eAAe,CAAC;AAExC,MAAM,CAAC,MAAM,YAAY,GAAG,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAC9C,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,KAAK,CAAC;IAE3B,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,cAAc,EAAE;QACtC,YAAY,EAAE,KAAK,CAAC,KAAK,KAAK,KAAK;QACnC,iBAAiB,EAAE;YACjB,eAAe,EAAE,IAAI;YACrB,gBAAgB,EAAE,IAAI;YACtB,iBAAiB,EAAE,IAAI;YACvB,qBAAqB,EAAE,IAAI;SAC5B;KACF,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { ImagesStack, type ImagesStackProps } from "./stack.js";
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,KAAK,gBAAgB,EAAE,MAAM,YAAY,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ export { ImagesStack } from "./stack.js";
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAyB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,11 @@
1
+ import * as AWS from "alchemy/AWS";
2
+ import * as Effect from "effect/Effect";
3
+ import { HttpServerRequest } from "effect/unstable/http/HttpServerRequest";
4
+ import * as HttpServerResponse from "effect/unstable/http/HttpServerResponse";
5
+ declare const ImagesResize_base: Effect.Effect<AWS.Lambda.Function & import("alchemy").Rpc<ImagesResize>, never, any> & import("alchemy").Named<"ImagesResize"> & (new (_: never) => {
6
+ fetch: Effect.Effect<HttpServerResponse.HttpServerResponse, never, HttpServerRequest>;
7
+ } & import("alchemy").Named<"ImagesResize"> & import("alchemy/Named").Tag<"AWS.Lambda.Function">);
8
+ export default class ImagesResize extends ImagesResize_base {
9
+ }
10
+ export {};
11
+ //# sourceMappingURL=resize-function.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resize-function.d.ts","sourceRoot":"","sources":["../src/resize-function.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,GAAG,MAAM,aAAa,CAAC;AAGnC,OAAO,KAAK,MAAM,MAAM,eAAe,CAAC;AAGxC,OAAO,EAAE,iBAAiB,EAAE,MAAM,wCAAwC,CAAC;AAC3E,OAAO,KAAK,kBAAkB,MAAM,yCAAyC,CAAC;;;;AAgB9E,MAAM,CAAC,OAAO,OAAO,YAAa,SAAQ,iBAmGzC;CAAG"}
@@ -0,0 +1,94 @@
1
+ import { NOT_FOUND_CACHE_CONTROL, RESIZED_CACHE_CONTROL, WEBP_CONTENT_TYPE, parseResizedPath, } from "@lambdaimg/core";
2
+ import * as AWS from "alchemy/AWS";
3
+ import * as S3 from "alchemy/AWS/S3";
4
+ import * as Duration from "effect/Duration";
5
+ import * as Effect from "effect/Effect";
6
+ import * as Layer from "effect/Layer";
7
+ import * as Stream from "effect/Stream";
8
+ import { HttpServerRequest } from "effect/unstable/http/HttpServerRequest";
9
+ import * as HttpServerResponse from "effect/unstable/http/HttpServerResponse";
10
+ import { imagesBucket } from "./images-bucket.js";
11
+ import { resizeToWebp } from "./resize.js";
12
+ function collectBodyBytes(chunks) {
13
+ const parts = Array.from(chunks);
14
+ const total = parts.reduce((sum, chunk) => sum + chunk.length, 0);
15
+ const output = new Uint8Array(total);
16
+ let offset = 0;
17
+ for (const chunk of parts) {
18
+ output.set(chunk, offset);
19
+ offset += chunk.length;
20
+ }
21
+ return output;
22
+ }
23
+ export default class ImagesResize extends AWS.Lambda.Function()("ImagesResize", {
24
+ main: import.meta.filename,
25
+ url: { authType: "AWS_IAM" },
26
+ timeout: Duration.seconds(30),
27
+ runtime: "nodejs24.x",
28
+ architecture: "arm64",
29
+ memorySize: 2048,
30
+ build: {
31
+ install: ["sharp", "heic-convert"],
32
+ },
33
+ }, Effect.gen(function* () {
34
+ const bucket = yield* imagesBucket;
35
+ const getObject = yield* S3.GetObject.bind(bucket);
36
+ const putObject = yield* S3.PutObject.bind(bucket);
37
+ return {
38
+ fetch: Effect.gen(function* () {
39
+ const request = yield* HttpServerRequest;
40
+ const parsed = parseResizedPath(new URL(request.originalUrl).pathname);
41
+ if (!parsed) {
42
+ return HttpServerResponse.text("Not Found", {
43
+ status: 404,
44
+ headers: { "cache-control": NOT_FOUND_CACHE_CONTROL },
45
+ });
46
+ }
47
+ const cached = yield* getObject({ Key: parsed.resizedKey }).pipe(Effect.catchTag("NoSuchKey", () => Effect.succeed(undefined)));
48
+ if (cached?.Body) {
49
+ const cachedBytes = yield* Stream.runCollect(cached.Body).pipe(Effect.map(collectBodyBytes));
50
+ return HttpServerResponse.uint8Array(cachedBytes, {
51
+ headers: {
52
+ "content-type": WEBP_CONTENT_TYPE,
53
+ "cache-control": RESIZED_CACHE_CONTROL,
54
+ },
55
+ });
56
+ }
57
+ const object = yield* getObject({ Key: parsed.originalKey }).pipe(Effect.catchTag("NoSuchKey", () => Effect.succeed(undefined)));
58
+ if (!object?.Body) {
59
+ return HttpServerResponse.text("Not Found", {
60
+ status: 404,
61
+ headers: { "cache-control": NOT_FOUND_CACHE_CONTROL },
62
+ });
63
+ }
64
+ const sourceBytes = yield* Stream.runCollect(object.Body).pipe(Effect.map(collectBodyBytes));
65
+ const resizeResult = yield* Effect.tryPromise({
66
+ try: () => resizeToWebp(sourceBytes, parsed.width, {
67
+ originalKey: parsed.originalKey,
68
+ }),
69
+ catch: (cause) => cause,
70
+ }).pipe(Effect.match({
71
+ onFailure: () => ({ ok: false }),
72
+ onSuccess: (webp) => ({ ok: true, webp }),
73
+ }));
74
+ if (!resizeResult.ok) {
75
+ return HttpServerResponse.text("Bad Request", { status: 400 });
76
+ }
77
+ const { webp } = resizeResult;
78
+ yield* putObject({
79
+ Key: parsed.resizedKey,
80
+ Body: webp,
81
+ ContentType: WEBP_CONTENT_TYPE,
82
+ CacheControl: RESIZED_CACHE_CONTROL,
83
+ });
84
+ return HttpServerResponse.uint8Array(webp, {
85
+ headers: {
86
+ "content-type": WEBP_CONTENT_TYPE,
87
+ "cache-control": RESIZED_CACHE_CONTROL,
88
+ },
89
+ });
90
+ }).pipe(Effect.catchCause(() => Effect.succeed(HttpServerResponse.text("Internal Server Error", { status: 500 })))),
91
+ };
92
+ }).pipe(Effect.provide(Layer.mergeAll(S3.GetObjectLive, S3.PutObjectLive)))) {
93
+ }
94
+ //# sourceMappingURL=resize-function.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resize-function.js","sourceRoot":"","sources":["../src/resize-function.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export interface ResizeToWebpOptions {
2
+ originalKey?: string;
3
+ }
4
+ export declare function resizeToWebp(input: Uint8Array | Buffer, width: number, options?: ResizeToWebpOptions): Promise<Uint8Array>;
5
+ //# sourceMappingURL=resize.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resize.d.ts","sourceRoot":"","sources":["../src/resize.ts"],"names":[],"mappings":"AAUA,MAAM,WAAW,mBAAmB;IAClC,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,wBAAsB,YAAY,CAChC,KAAK,EAAE,UAAU,GAAG,MAAM,EAC1B,KAAK,EAAE,MAAM,EACb,OAAO,GAAE,mBAAwB,GAChC,OAAO,CAAC,UAAU,CAAC,CAerB"}
package/dist/resize.js ADDED
@@ -0,0 +1,29 @@
1
+ import { createRequire } from "node:module";
2
+ import sharp from "sharp";
3
+ const HEIC_EXTENSION_RE = /\.(?:heic|heif)$/i;
4
+ const require = createRequire(import.meta.url);
5
+ const heicConvert = require("heic-convert");
6
+ export async function resizeToWebp(input, width, options = {}) {
7
+ try {
8
+ return await resizeSharpInputToWebp(input, width);
9
+ }
10
+ catch (error) {
11
+ if (!options.originalKey || !HEIC_EXTENSION_RE.test(options.originalKey)) {
12
+ throw error;
13
+ }
14
+ const png = await heicConvert({
15
+ buffer: Buffer.from(input),
16
+ format: "PNG",
17
+ });
18
+ return await resizeSharpInputToWebp(Buffer.from(png), width);
19
+ }
20
+ }
21
+ async function resizeSharpInputToWebp(input, width) {
22
+ const output = await sharp(input)
23
+ .rotate()
24
+ .resize({ width, withoutEnlargement: true })
25
+ .webp()
26
+ .toBuffer();
27
+ return new Uint8Array(output);
28
+ }
29
+ //# sourceMappingURL=resize.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resize.js","sourceRoot":"","sources":["../src/resize.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,MAAM,iBAAiB,GAAG,mBAAmB,CAAC;AAC9C,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC;AAC/C,MAAM,WAAW,GAAG,OAAO,CAAC,cAAc,CAGrB,CAAC;AAMtB,MAAM,CAAC,KAAK,UAAU,YAAY,CAChC,KAA0B,EAC1B,KAAa,EACb,OAAO,GAAwB,EAAE;IAEjC,IAAI,CAAC;QACH,OAAO,MAAM,sBAAsB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;IACpD,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,CAAC,OAAO,CAAC,WAAW,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;YACzE,MAAM,KAAK,CAAC;QACd,CAAC;QAED,MAAM,GAAG,GAAG,MAAM,WAAW,CAAC;YAC5B,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC;YAC1B,MAAM,EAAE,KAAK;SACd,CAAC,CAAC;QAEH,OAAO,MAAM,sBAAsB,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;IAC/D,CAAC;AACH,CAAC;AAED,KAAK,UAAU,sBAAsB,CACnC,KAA0B,EAC1B,KAAa;IAEb,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC;SAC9B,MAAM,EAAE;SACR,MAAM,CAAC,EAAE,KAAK,EAAE,kBAAkB,EAAE,IAAI,EAAE,CAAC;SAC3C,IAAI,EAAE;SACN,QAAQ,EAAE,CAAC;IACd,OAAO,IAAI,UAAU,CAAC,MAAM,CAAC,CAAC;AAChC,CAAC"}
@@ -0,0 +1,14 @@
1
+ import * as Output from "alchemy/Output";
2
+ import * as Effect from "effect/Effect";
3
+ export interface ImagesStackProps {
4
+ domain: string;
5
+ hostedZoneId: string;
6
+ }
7
+ export declare const ImagesStack: (id: string, props: ImagesStackProps) => Effect.Effect<{
8
+ domain: string;
9
+ url: Output.Output<string, unknown>;
10
+ distributionDomain: Output.Output<string, never>;
11
+ resizeFunctionUrl: Output.Output<string | undefined, never>;
12
+ bucketName: Output.Output<string, never>;
13
+ }, never, any>;
14
+ //# sourceMappingURL=stack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stack.d.ts","sourceRoot":"","sources":["../src/stack.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,MAAM,MAAM,gBAAgB,CAAC;AACzC,OAAO,KAAK,MAAM,MAAM,eAAe,CAAC;AA2BxC,MAAM,WAAW,gBAAgB;IAC/B,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,eAAO,MAAM,WAAW,OAAQ,MAAM,SAAS,gBAAgB;;;;;;cA0J3D,CAAC"}
package/dist/stack.js ADDED
@@ -0,0 +1,170 @@
1
+ import { ALLOWED_IMAGE_WIDTHS } from "@lambdaimg/core";
2
+ import * as AWS from "alchemy/AWS";
3
+ import * as Output from "alchemy/Output";
4
+ import * as Effect from "effect/Effect";
5
+ import { imagesBucket } from "./images-bucket.js";
6
+ import ImagesResize from "./resize-function.js";
7
+ const CLOUDFRONT_ORIGIN_DEFAULTS = {
8
+ originPath: "",
9
+ customHeaders: {},
10
+ originShield: { enabled: false },
11
+ connectionAttempts: 3,
12
+ connectionTimeout: 10,
13
+ };
14
+ // CloudFront update calls require fully materialized origin/behavior blocks.
15
+ // These values mirror CloudFront defaults so Alchemy updates remain stable.
16
+ const CLOUDFRONT_BEHAVIOR_DEFAULTS = {
17
+ smoothStreaming: false,
18
+ fieldLevelEncryptionId: "",
19
+ trustedSigners: [],
20
+ trustedKeyGroups: [],
21
+ functionAssociations: [],
22
+ lambdaFunctionAssociations: [],
23
+ };
24
+ export const ImagesStack = (id, props) => Effect.gen(function* () {
25
+ const bucket = yield* imagesBucket;
26
+ const { accountId } = yield* AWS.AWSEnvironment.current;
27
+ const s3Oac = yield* AWS.CloudFront.OriginAccessControl(`${id}S3Oac`, {
28
+ originType: "s3",
29
+ description: "Images bucket origin access control",
30
+ });
31
+ const lambdaOac = yield* AWS.CloudFront.OriginAccessControl(`${id}LambdaOac`, {
32
+ originType: "lambda",
33
+ description: "Images resize Lambda function URL origin access control",
34
+ });
35
+ const resize = yield* ImagesResize;
36
+ const certificate = yield* AWS.ACM.Certificate(`${id}Cert`, {
37
+ domainName: props.domain,
38
+ hostedZoneId: props.hostedZoneId,
39
+ });
40
+ const longTtlCachePolicy = yield* AWS.CloudFront.CachePolicy(`${id}LongTtlCache`, {
41
+ name: `${id}-long-ttl`,
42
+ minTTL: 31536000,
43
+ defaultTTL: 31536000,
44
+ maxTTL: 31536000,
45
+ parametersInCacheKeyAndForwardedToOrigin: {
46
+ CookiesConfig: { CookieBehavior: "none" },
47
+ HeadersConfig: { HeaderBehavior: "none" },
48
+ QueryStringsConfig: { QueryStringBehavior: "none" },
49
+ EnableAcceptEncodingGzip: true,
50
+ EnableAcceptEncodingBrotli: true,
51
+ },
52
+ });
53
+ const lambdaOriginDomain = Output.map(resize.functionUrl, (functionUrl) => {
54
+ if (!functionUrl) {
55
+ throw new Error("ImagesResize function URL is required");
56
+ }
57
+ return new URL(functionUrl).hostname;
58
+ });
59
+ const distribution = yield* AWS.CloudFront.Distribution(`${id}Cdn`, {
60
+ aliases: [props.domain],
61
+ priceClass: "PriceClass_All",
62
+ httpVersion: "http2and3",
63
+ logging: { enabled: false },
64
+ defaultRootObject: "",
65
+ webAclId: "",
66
+ originGroups: [
67
+ {
68
+ id: "resized",
69
+ members: ["s3", "lambda"],
70
+ failoverStatusCodes: [403, 404],
71
+ },
72
+ ],
73
+ staging: false,
74
+ origins: [
75
+ {
76
+ ...CLOUDFRONT_ORIGIN_DEFAULTS,
77
+ id: "s3",
78
+ domainName: bucket.bucketRegionalDomainName,
79
+ s3Origin: true,
80
+ originAccessControlId: s3Oac.originAccessControlId,
81
+ },
82
+ {
83
+ ...CLOUDFRONT_ORIGIN_DEFAULTS,
84
+ id: "lambda",
85
+ domainName: lambdaOriginDomain,
86
+ originAccessControlId: lambdaOac.originAccessControlId,
87
+ customOriginConfig: {
88
+ originProtocolPolicy: "https-only",
89
+ originReadTimeout: 30,
90
+ originKeepaliveTimeout: 5,
91
+ },
92
+ },
93
+ ],
94
+ defaultCacheBehavior: {
95
+ ...CLOUDFRONT_BEHAVIOR_DEFAULTS,
96
+ targetOriginId: "s3",
97
+ viewerProtocolPolicy: "redirect-to-https",
98
+ compress: true,
99
+ allowedMethods: ["GET", "HEAD"],
100
+ cachedMethods: ["GET", "HEAD"],
101
+ cachePolicyId: longTtlCachePolicy.cachePolicyId,
102
+ },
103
+ orderedCacheBehaviors: ALLOWED_IMAGE_WIDTHS.map((width) => ({
104
+ ...CLOUDFRONT_BEHAVIOR_DEFAULTS,
105
+ pathPattern: `/_/w${width}/*`,
106
+ targetOriginId: "resized",
107
+ viewerProtocolPolicy: "redirect-to-https",
108
+ compress: true,
109
+ allowedMethods: ["GET", "HEAD"],
110
+ cachedMethods: ["GET", "HEAD"],
111
+ cachePolicyId: longTtlCachePolicy.cachePolicyId,
112
+ })),
113
+ customErrorResponses: [],
114
+ viewerCertificate: {
115
+ acmCertificateArn: certificate.certificateArn,
116
+ sslSupportMethod: "sni-only",
117
+ minimumProtocolVersion: "TLSv1.2_2021",
118
+ },
119
+ });
120
+ yield* AWS.Lambda.Permission(`${id}ResizeCloudFront`, {
121
+ action: "lambda:InvokeFunctionUrl",
122
+ functionName: resize.functionArn,
123
+ principal: "cloudfront.amazonaws.com",
124
+ sourceArn: distribution.distributionArn,
125
+ functionUrlAuthType: "AWS_IAM",
126
+ });
127
+ yield* AWS.Lambda.Permission(`${id}ResizeCloudFrontInvokeFunction`, {
128
+ action: "lambda:InvokeFunction",
129
+ functionName: resize.functionArn,
130
+ principal: "cloudfront.amazonaws.com",
131
+ sourceArn: distribution.distributionArn,
132
+ invokedViaFunctionUrl: true,
133
+ });
134
+ // Avoid a Bucket -> Distribution -> Lambda -> Bucket dependency cycle by
135
+ // scoping OAC access to CloudFront distributions in the same AWS account.
136
+ yield* bucket.bind `AWS.S3.Policy(${bucket})`({
137
+ policyStatements: [
138
+ {
139
+ Effect: "Allow",
140
+ Principal: {
141
+ Service: "cloudfront.amazonaws.com",
142
+ },
143
+ Action: ["s3:GetObject"],
144
+ Resource: [Output.interpolate `${bucket.bucketArn}/*`],
145
+ Condition: {
146
+ ArnLike: {
147
+ "AWS:SourceArn": `arn:aws:cloudfront::${accountId}:distribution/*`,
148
+ },
149
+ },
150
+ },
151
+ ],
152
+ });
153
+ yield* AWS.Route53.Record(`${id}Alias`, {
154
+ hostedZoneId: props.hostedZoneId,
155
+ name: props.domain,
156
+ type: "A",
157
+ aliasTarget: {
158
+ hostedZoneId: distribution.hostedZoneId,
159
+ dnsName: distribution.domainName,
160
+ },
161
+ });
162
+ return {
163
+ domain: props.domain,
164
+ url: Output.interpolate `https://${props.domain}`,
165
+ distributionDomain: distribution.domainName,
166
+ resizeFunctionUrl: resize.functionUrl,
167
+ bucketName: bucket.bucketName,
168
+ };
169
+ });
170
+ //# sourceMappingURL=stack.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stack.js","sourceRoot":"","sources":["../src/stack.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,43 @@
1
+ {
2
+ "name": "@lambdaimg/alchemy",
3
+ "version": "0.1.0",
4
+ "description": "AWS Alchemy stack and Lambda runtime for LambdaImg.",
5
+ "license": "MIT",
6
+ "repository": {
7
+ "type": "git",
8
+ "url": "git+https://github.com/jperezrealini/lambdaimg.git",
9
+ "directory": "packages/alchemy"
10
+ },
11
+ "files": [
12
+ "dist",
13
+ "README.md"
14
+ ],
15
+ "type": "module",
16
+ "sideEffects": false,
17
+ "exports": {
18
+ ".": {
19
+ "default": "./dist/index.js",
20
+ "types": "./dist/index.d.ts"
21
+ }
22
+ },
23
+ "publishConfig": {
24
+ "access": "public"
25
+ },
26
+ "scripts": {
27
+ "build": "tsc -p tsconfig.build.json",
28
+ "check": "oxlint && oxfmt --check && tsc --noEmit -p tsconfig.json",
29
+ "fix": "oxlint --fix && oxfmt",
30
+ "test": "bun test"
31
+ },
32
+ "dependencies": {
33
+ "@lambdaimg/core": "workspace:*",
34
+ "alchemy": "2.0.0-beta.58",
35
+ "effect": "4.0.0-beta.90",
36
+ "heic-convert": "^2.1.0",
37
+ "sharp": "^0.34.5"
38
+ },
39
+ "devDependencies": {
40
+ "@types/bun": "catalog:",
41
+ "@types/node": "catalog:"
42
+ }
43
+ }