construct-hub 0.4.430 → 0.4.432

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 (47) hide show
  1. package/.jsii +9 -9
  2. package/docs/operator-runbook.md +96 -0
  3. package/lib/backend/orchestration/index.d.ts +5 -1
  4. package/lib/backend/orchestration/index.js +9 -2
  5. package/lib/backend/orchestration/read-uninstallable-report.d.ts +7 -0
  6. package/lib/backend/orchestration/read-uninstallable-report.js +20 -0
  7. package/lib/backend/orchestration/read-uninstallable-report.lambda.bundle/index.js +60016 -0
  8. package/lib/backend/orchestration/read-uninstallable-report.lambda.bundle/index.js.map +7 -0
  9. package/lib/backend/orchestration/read-uninstallable-report.lambda.d.ts +13 -0
  10. package/lib/backend/orchestration/read-uninstallable-report.lambda.js +33 -0
  11. package/lib/backend/orchestration/retry-uninstallable-packages.d.ts +19 -0
  12. package/lib/backend/orchestration/retry-uninstallable-packages.js +100 -0
  13. package/lib/backend/package-stats/index.d.ts +15 -2
  14. package/lib/backend/package-stats/index.js +79 -21
  15. package/lib/backend/package-stats/package-stats-aggregator.d.ts +7 -0
  16. package/lib/backend/package-stats/package-stats-aggregator.js +20 -0
  17. package/lib/backend/package-stats/package-stats-aggregator.lambda.bundle/index.js +62489 -0
  18. package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-aggregator.lambda.bundle}/index.js.map +4 -4
  19. package/lib/backend/package-stats/package-stats-aggregator.lambda.d.ts +13 -0
  20. package/lib/backend/package-stats/package-stats-aggregator.lambda.js +78 -0
  21. package/lib/backend/package-stats/package-stats-chunker.d.ts +7 -0
  22. package/lib/backend/package-stats/package-stats-chunker.js +20 -0
  23. package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js +60075 -0
  24. package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js.map +7 -0
  25. package/lib/backend/package-stats/package-stats-chunker.lambda.d.ts +6 -0
  26. package/lib/backend/package-stats/package-stats-chunker.lambda.js +25 -0
  27. package/lib/backend/package-stats/package-stats-processor.d.ts +7 -0
  28. package/lib/backend/package-stats/package-stats-processor.js +20 -0
  29. package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-processor.lambda.bundle}/index.js +36 -2556
  30. package/lib/backend/package-stats/package-stats-processor.lambda.bundle/index.js.map +7 -0
  31. package/lib/backend/package-stats/package-stats-processor.lambda.d.ts +10 -0
  32. package/lib/backend/package-stats/package-stats-processor.lambda.js +41 -0
  33. package/lib/backend-dashboard.js +11 -6
  34. package/lib/construct-hub.d.ts +1 -1
  35. package/lib/construct-hub.js +12 -4
  36. package/lib/package-sources/code-artifact.js +1 -1
  37. package/lib/package-sources/npmjs.js +1 -1
  38. package/lib/package-tag/index.js +3 -3
  39. package/lib/package-tag-group/index.js +2 -2
  40. package/lib/preload-file/index.js +1 -1
  41. package/lib/s3/storage.js +1 -1
  42. package/lib/spdx-license.js +1 -1
  43. package/package.json +9 -3
  44. package/lib/backend/package-stats/package-stats.d.ts +0 -7
  45. package/lib/backend/package-stats/package-stats.js +0 -20
  46. package/lib/backend/package-stats/package-stats.lambda.d.ts +0 -25
  47. package/lib/backend/package-stats/package-stats.lambda.js +0 -79
@@ -0,0 +1,13 @@
1
+ export interface ReadUninstallableReportEvent {
2
+ readonly bucket: string;
3
+ readonly key: string;
4
+ }
5
+ export interface PackageInfo {
6
+ readonly originalPackage: string;
7
+ readonly packageName: string;
8
+ readonly packageVersion: string;
9
+ }
10
+ export interface ReadUninstallableReportResult {
11
+ readonly packages: PackageInfo[];
12
+ }
13
+ export declare function handler(event: ReadUninstallableReportEvent): Promise<ReadUninstallableReportResult>;
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handler = handler;
4
+ const client_s3_1 = require("@aws-sdk/client-s3");
5
+ const aws_lambda_shared_1 = require("../shared/aws.lambda-shared");
6
+ const compress_content_lambda_shared_1 = require("../shared/compress-content.lambda-shared");
7
+ function parsePackageName(packageString) {
8
+ const lastAtIndex = packageString.lastIndexOf('@');
9
+ if (lastAtIndex === -1) {
10
+ throw new Error(`Invalid package format: ${packageString}`);
11
+ }
12
+ const packageName = packageString.substring(0, lastAtIndex);
13
+ const packageVersion = packageString.substring(lastAtIndex + 1);
14
+ return {
15
+ originalPackage: packageString,
16
+ packageName,
17
+ packageVersion,
18
+ };
19
+ }
20
+ async function handler(event) {
21
+ const response = await aws_lambda_shared_1.S3_CLIENT.send(new client_s3_1.GetObjectCommand({
22
+ Bucket: event.bucket,
23
+ Key: event.key,
24
+ }));
25
+ if (!response.Body) {
26
+ throw new Error(`Object not found: s3://${event.bucket}/${event.key}`);
27
+ }
28
+ const decompressed = await (0, compress_content_lambda_shared_1.decompressContent)(response.Body, response.ContentEncoding);
29
+ const rawPackages = JSON.parse(decompressed);
30
+ const packages = rawPackages.map(parsePackageName);
31
+ return { packages };
32
+ }
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVhZC11bmluc3RhbGxhYmxlLXJlcG9ydC5sYW1iZGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYmFja2VuZC9vcmNoZXN0cmF0aW9uL3JlYWQtdW5pbnN0YWxsYWJsZS1yZXBvcnQubGFtYmRhLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBbUNBLDBCQXVCQztBQTFERCxrREFBc0Q7QUFDdEQsbUVBQXdEO0FBQ3hELDZGQUE2RTtBQWlCN0UsU0FBUyxnQkFBZ0IsQ0FBQyxhQUFxQjtJQUM3QyxNQUFNLFdBQVcsR0FBRyxhQUFhLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ25ELElBQUksV0FBVyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDdkIsTUFBTSxJQUFJLEtBQUssQ0FBQywyQkFBMkIsYUFBYSxFQUFFLENBQUMsQ0FBQztJQUM5RCxDQUFDO0lBRUQsTUFBTSxXQUFXLEdBQUcsYUFBYSxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsV0FBVyxDQUFDLENBQUM7SUFDNUQsTUFBTSxjQUFjLEdBQUcsYUFBYSxDQUFDLFNBQVMsQ0FBQyxXQUFXLEdBQUcsQ0FBQyxDQUFDLENBQUM7SUFFaEUsT0FBTztRQUNMLGVBQWUsRUFBRSxhQUFhO1FBQzlCLFdBQVc7UUFDWCxjQUFjO0tBQ2YsQ0FBQztBQUNKLENBQUM7QUFFTSxLQUFLLFVBQVUsT0FBTyxDQUMzQixLQUFtQztJQUVuQyxNQUFNLFFBQVEsR0FBRyxNQUFNLDZCQUFTLENBQUMsSUFBSSxDQUNuQyxJQUFJLDRCQUFnQixDQUFDO1FBQ25CLE1BQU0sRUFBRSxLQUFLLENBQUMsTUFBTTtRQUNwQixHQUFHLEVBQUUsS0FBSyxDQUFDLEdBQUc7S0FDZixDQUFDLENBQ0gsQ0FBQztJQUVGLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDbkIsTUFBTSxJQUFJLEtBQUssQ0FBQywwQkFBMEIsS0FBSyxDQUFDLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztJQUN6RSxDQUFDO0lBRUQsTUFBTSxZQUFZLEdBQUcsTUFBTSxJQUFBLGtEQUFpQixFQUMxQyxRQUFRLENBQUMsSUFBVyxFQUNwQixRQUFRLENBQUMsZUFBZSxDQUN6QixDQUFDO0lBRUYsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUM3QyxNQUFNLFFBQVEsR0FBRyxXQUFXLENBQUMsR0FBRyxDQUFDLGdCQUFnQixDQUFDLENBQUM7SUFFbkQsT0FBTyxFQUFFLFFBQVEsRUFBRSxDQUFDO0FBQ3RCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBHZXRPYmplY3RDb21tYW5kIH0gZnJvbSAnQGF3cy1zZGsvY2xpZW50LXMzJztcbmltcG9ydCB7IFMzX0NMSUVOVCB9IGZyb20gJy4uL3NoYXJlZC9hd3MubGFtYmRhLXNoYXJlZCc7XG5pbXBvcnQgeyBkZWNvbXByZXNzQ29udGVudCB9IGZyb20gJy4uL3NoYXJlZC9jb21wcmVzcy1jb250ZW50LmxhbWJkYS1zaGFyZWQnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJlYWRVbmluc3RhbGxhYmxlUmVwb3J0RXZlbnQge1xuICByZWFkb25seSBidWNrZXQ6IHN0cmluZztcbiAgcmVhZG9ubHkga2V5OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFja2FnZUluZm8ge1xuICByZWFkb25seSBvcmlnaW5hbFBhY2thZ2U6IHN0cmluZztcbiAgcmVhZG9ubHkgcGFja2FnZU5hbWU6IHN0cmluZztcbiAgcmVhZG9ubHkgcGFja2FnZVZlcnNpb246IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSZWFkVW5pbnN0YWxsYWJsZVJlcG9ydFJlc3VsdCB7XG4gIHJlYWRvbmx5IHBhY2thZ2VzOiBQYWNrYWdlSW5mb1tdO1xufVxuXG5mdW5jdGlvbiBwYXJzZVBhY2thZ2VOYW1lKHBhY2thZ2VTdHJpbmc6IHN0cmluZyk6IFBhY2thZ2VJbmZvIHtcbiAgY29uc3QgbGFzdEF0SW5kZXggPSBwYWNrYWdlU3RyaW5nLmxhc3RJbmRleE9mKCdAJyk7XG4gIGlmIChsYXN0QXRJbmRleCA9PT0gLTEpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoYEludmFsaWQgcGFja2FnZSBmb3JtYXQ6ICR7cGFja2FnZVN0cmluZ31gKTtcbiAgfVxuXG4gIGNvbnN0IHBhY2thZ2VOYW1lID0gcGFja2FnZVN0cmluZy5zdWJzdHJpbmcoMCwgbGFzdEF0SW5kZXgpO1xuICBjb25zdCBwYWNrYWdlVmVyc2lvbiA9IHBhY2thZ2VTdHJpbmcuc3Vic3RyaW5nKGxhc3RBdEluZGV4ICsgMSk7XG5cbiAgcmV0dXJuIHtcbiAgICBvcmlnaW5hbFBhY2thZ2U6IHBhY2thZ2VTdHJpbmcsXG4gICAgcGFja2FnZU5hbWUsXG4gICAgcGFja2FnZVZlcnNpb24sXG4gIH07XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBoYW5kbGVyKFxuICBldmVudDogUmVhZFVuaW5zdGFsbGFibGVSZXBvcnRFdmVudFxuKTogUHJvbWlzZTxSZWFkVW5pbnN0YWxsYWJsZVJlcG9ydFJlc3VsdD4ge1xuICBjb25zdCByZXNwb25zZSA9IGF3YWl0IFMzX0NMSUVOVC5zZW5kKFxuICAgIG5ldyBHZXRPYmplY3RDb21tYW5kKHtcbiAgICAgIEJ1Y2tldDogZXZlbnQuYnVja2V0LFxuICAgICAgS2V5OiBldmVudC5rZXksXG4gICAgfSlcbiAgKTtcblxuICBpZiAoIXJlc3BvbnNlLkJvZHkpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoYE9iamVjdCBub3QgZm91bmQ6IHMzOi8vJHtldmVudC5idWNrZXR9LyR7ZXZlbnQua2V5fWApO1xuICB9XG5cbiAgY29uc3QgZGVjb21wcmVzc2VkID0gYXdhaXQgZGVjb21wcmVzc0NvbnRlbnQoXG4gICAgcmVzcG9uc2UuQm9keSBhcyBhbnksXG4gICAgcmVzcG9uc2UuQ29udGVudEVuY29kaW5nXG4gICk7XG5cbiAgY29uc3QgcmF3UGFja2FnZXMgPSBKU09OLnBhcnNlKGRlY29tcHJlc3NlZCk7XG4gIGNvbnN0IHBhY2thZ2VzID0gcmF3UGFja2FnZXMubWFwKHBhcnNlUGFja2FnZU5hbWUpO1xuXG4gIHJldHVybiB7IHBhY2thZ2VzIH07XG59XG4iXX0=
@@ -0,0 +1,19 @@
1
+ import { IBucket } from 'aws-cdk-lib/aws-s3';
2
+ import { IStateMachine, StateMachine } from 'aws-cdk-lib/aws-stepfunctions';
3
+ import { Construct } from 'constructs';
4
+ export interface RetryUninstallablePackagesProps {
5
+ readonly bucket: IBucket;
6
+ readonly orchestrationStateMachine: IStateMachine;
7
+ }
8
+ /**
9
+ * State machine that retries processing of uninstallable packages.
10
+ *
11
+ * This workflow:
12
+ * 1. Reads the uninstallable packages report
13
+ * 2. Triggers main orchestration for each package
14
+ * 3. Ends after processing all packages
15
+ */
16
+ export declare class RetryUninstallablePackages extends Construct {
17
+ readonly stateMachine: StateMachine;
18
+ constructor(scope: Construct, id: string, props: RetryUninstallablePackagesProps);
19
+ }
@@ -0,0 +1,100 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.RetryUninstallablePackages = void 0;
4
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
5
+ const aws_logs_1 = require("aws-cdk-lib/aws-logs");
6
+ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
7
+ const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
8
+ const constructs_1 = require("constructs");
9
+ const read_uninstallable_report_1 = require("./read-uninstallable-report");
10
+ /**
11
+ * State machine that retries processing of uninstallable packages.
12
+ *
13
+ * This workflow:
14
+ * 1. Reads the uninstallable packages report
15
+ * 2. Triggers main orchestration for each package
16
+ * 3. Ends after processing all packages
17
+ */
18
+ class RetryUninstallablePackages extends constructs_1.Construct {
19
+ constructor(scope, id, props) {
20
+ super(scope, id);
21
+ const noReportFound = new aws_stepfunctions_1.Fail(this, 'No Report Found', {
22
+ error: 'NoReportFound',
23
+ cause: 'Uninstallable packages report not found at uninstallable-objects/data.json',
24
+ });
25
+ const noPackagesToRetry = new aws_stepfunctions_1.Succeed(this, 'No Packages to Retry');
26
+ const readReportFunction = new read_uninstallable_report_1.ReadUninstallableReport(this, 'ReadReportFunction', {
27
+ logRetention: aws_logs_1.RetentionDays.THREE_MONTHS,
28
+ });
29
+ props.bucket.grantRead(readReportFunction);
30
+ const readReport = new tasks.LambdaInvoke(this, 'Read Uninstallable Report', {
31
+ lambdaFunction: readReportFunction,
32
+ payload: aws_stepfunctions_1.TaskInput.fromObject({
33
+ bucket: props.bucket.bucketName,
34
+ key: 'uninstallable-objects/data.json',
35
+ }),
36
+ resultPath: '$.reportResponse',
37
+ });
38
+ readReport.addRetry({
39
+ errors: ['Lambda.Unknown'],
40
+ interval: aws_cdk_lib_1.Duration.seconds(2),
41
+ maxAttempts: 3,
42
+ backoffRate: 2.0,
43
+ });
44
+ readReport.addCatch(noReportFound, {
45
+ errors: ['States.TaskFailed'],
46
+ });
47
+ const packageRetryFailed = new aws_stepfunctions_1.Pass(this, 'Package Retry Failed', {
48
+ parameters: {
49
+ 'package.$': '$.originalPackage',
50
+ 'prefix.$': "States.Format('data/{}/v{}', $.packageName, $.packageVersion)",
51
+ 'error.$': '$.error',
52
+ },
53
+ });
54
+ const retryPackage = new tasks.StepFunctionsStartExecution(this, 'Retry Package', {
55
+ stateMachine: props.orchestrationStateMachine,
56
+ integrationPattern: aws_stepfunctions_1.IntegrationPattern.RUN_JOB,
57
+ input: aws_stepfunctions_1.TaskInput.fromObject({
58
+ bucket: props.bucket.bucketName,
59
+ assembly: {
60
+ 'key.$': "States.Format('data/{}/v{}/assembly.json', $.packageName, $.packageVersion)",
61
+ },
62
+ metadata: {
63
+ 'key.$': "States.Format('data/{}/v{}/metadata.json', $.packageName, $.packageVersion)",
64
+ },
65
+ package: {
66
+ 'key.$': "States.Format('data/{}/v{}/package.tgz', $.packageName, $.packageVersion)",
67
+ },
68
+ }),
69
+ });
70
+ retryPackage.addRetry({
71
+ errors: ['StepFunctions.ExecutionLimitExceeded'],
72
+ interval: aws_cdk_lib_1.Duration.seconds(60),
73
+ maxAttempts: 3,
74
+ backoffRate: 2.0,
75
+ });
76
+ retryPackage.addCatch(packageRetryFailed, {
77
+ errors: ['States.ALL'],
78
+ resultPath: '$.error',
79
+ });
80
+ const processEachPackage = new aws_stepfunctions_1.Map(this, 'Process Each Package', {
81
+ itemsPath: aws_stepfunctions_1.JsonPath.stringAt('$.reportResponse.Payload.packages'),
82
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
83
+ });
84
+ processEachPackage.itemProcessor(retryPackage);
85
+ const hasPackages = new aws_stepfunctions_1.Choice(this, 'Has Packages?')
86
+ .when(aws_stepfunctions_1.Condition.isPresent('$.reportResponse.Payload.packages[0]'), processEachPackage)
87
+ .otherwise(noPackagesToRetry);
88
+ const definition = readReport.next(hasPackages);
89
+ this.stateMachine = new aws_stepfunctions_1.StateMachine(this, 'Resource', {
90
+ definition,
91
+ stateMachineName: 'RetryUninstallablePackages',
92
+ timeout: aws_cdk_lib_1.Duration.hours(6),
93
+ tracingEnabled: true,
94
+ });
95
+ props.bucket.grantRead(this.stateMachine);
96
+ props.orchestrationStateMachine.grantStartExecution(this.stateMachine);
97
+ }
98
+ }
99
+ exports.RetryUninstallablePackages = RetryUninstallablePackages;
100
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retry-uninstallable-packages.js","sourceRoot":"","sources":["../../../src/backend/orchestration/retry-uninstallable-packages.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport {\n  IStateMachine,\n  StateMachine,\n  Succeed,\n  Fail,\n  Pass,\n  Choice,\n  Condition,\n  Map,\n  JsonPath,\n  IntegrationPattern,\n  TaskInput,\n} from 'aws-cdk-lib/aws-stepfunctions';\nimport * as tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { ReadUninstallableReport } from './read-uninstallable-report';\n\nexport interface RetryUninstallablePackagesProps {\n  readonly bucket: IBucket;\n  readonly orchestrationStateMachine: IStateMachine;\n}\n\n/**\n * State machine that retries processing of uninstallable packages.\n *\n * This workflow:\n * 1. Reads the uninstallable packages report\n * 2. Triggers main orchestration for each package\n * 3. Ends after processing all packages\n */\nexport class RetryUninstallablePackages extends Construct {\n  public readonly stateMachine: StateMachine;\n\n  public constructor(\n    scope: Construct,\n    id: string,\n    props: RetryUninstallablePackagesProps\n  ) {\n    super(scope, id);\n\n    const noReportFound = new Fail(this, 'No Report Found', {\n      error: 'NoReportFound',\n      cause:\n        'Uninstallable packages report not found at uninstallable-objects/data.json',\n    });\n\n    const noPackagesToRetry = new Succeed(this, 'No Packages to Retry');\n\n    const readReportFunction = new ReadUninstallableReport(\n      this,\n      'ReadReportFunction',\n      {\n        logRetention: RetentionDays.THREE_MONTHS,\n      }\n    );\n\n    props.bucket.grantRead(readReportFunction);\n\n    const readReport = new tasks.LambdaInvoke(\n      this,\n      'Read Uninstallable Report',\n      {\n        lambdaFunction: readReportFunction,\n        payload: TaskInput.fromObject({\n          bucket: props.bucket.bucketName,\n          key: 'uninstallable-objects/data.json',\n        }),\n        resultPath: '$.reportResponse',\n      }\n    );\n\n    readReport.addRetry({\n      errors: ['Lambda.Unknown'],\n      interval: Duration.seconds(2),\n      maxAttempts: 3,\n      backoffRate: 2.0,\n    });\n\n    readReport.addCatch(noReportFound, {\n      errors: ['States.TaskFailed'],\n    });\n\n    const packageRetryFailed = new Pass(this, 'Package Retry Failed', {\n      parameters: {\n        'package.$': '$.originalPackage',\n        'prefix.$':\n          \"States.Format('data/{}/v{}', $.packageName, $.packageVersion)\",\n        'error.$': '$.error',\n      },\n    });\n\n    const retryPackage = new tasks.StepFunctionsStartExecution(\n      this,\n      'Retry Package',\n      {\n        stateMachine: props.orchestrationStateMachine,\n        integrationPattern: IntegrationPattern.RUN_JOB,\n        input: TaskInput.fromObject({\n          bucket: props.bucket.bucketName,\n          assembly: {\n            'key.$':\n              \"States.Format('data/{}/v{}/assembly.json', $.packageName, $.packageVersion)\",\n          },\n          metadata: {\n            'key.$':\n              \"States.Format('data/{}/v{}/metadata.json', $.packageName, $.packageVersion)\",\n          },\n          package: {\n            'key.$':\n              \"States.Format('data/{}/v{}/package.tgz', $.packageName, $.packageVersion)\",\n          },\n        }),\n      }\n    );\n\n    retryPackage.addRetry({\n      errors: ['StepFunctions.ExecutionLimitExceeded'],\n      interval: Duration.seconds(60),\n      maxAttempts: 3,\n      backoffRate: 2.0,\n    });\n\n    retryPackage.addCatch(packageRetryFailed, {\n      errors: ['States.ALL'],\n      resultPath: '$.error',\n    });\n\n    const processEachPackage = new Map(this, 'Process Each Package', {\n      itemsPath: JsonPath.stringAt('$.reportResponse.Payload.packages'),\n      resultPath: JsonPath.DISCARD,\n    });\n\n    processEachPackage.itemProcessor(retryPackage);\n\n    const hasPackages = new Choice(this, 'Has Packages?')\n      .when(\n        Condition.isPresent('$.reportResponse.Payload.packages[0]'),\n        processEachPackage\n      )\n      .otherwise(noPackagesToRetry);\n\n    const definition = readReport.next(hasPackages);\n\n    this.stateMachine = new StateMachine(this, 'Resource', {\n      definition,\n      stateMachineName: 'RetryUninstallablePackages',\n      timeout: Duration.hours(6),\n      tracingEnabled: true,\n    });\n\n    props.bucket.grantRead(this.stateMachine);\n    props.orchestrationStateMachine.grantStartExecution(this.stateMachine);\n  }\n}\n"]}
@@ -3,6 +3,7 @@ import { Metric, MetricOptions } from 'aws-cdk-lib/aws-cloudwatch';
3
3
  import { IFunction } from 'aws-cdk-lib/aws-lambda';
4
4
  import { RetentionDays } from 'aws-cdk-lib/aws-logs';
5
5
  import type { IBucket } from 'aws-cdk-lib/aws-s3';
6
+ import * as sfn from 'aws-cdk-lib/aws-stepfunctions';
6
7
  import { Construct } from 'constructs';
7
8
  import { Monitoring } from '../../monitoring';
8
9
  /**
@@ -36,6 +37,12 @@ export interface PackageStatsProps {
36
37
  * The key of the object storing the package stats.
37
38
  */
38
39
  readonly objectKey: string;
40
+ /**
41
+ * Number of packages to process per chunk.
42
+ *
43
+ * @default 100
44
+ */
45
+ readonly chunkSize?: number;
39
46
  }
40
47
  /**
41
48
  * Builds or re-builds the `stats.json` object in the designated bucket.
@@ -47,9 +54,15 @@ export declare class PackageStats extends Construct {
47
54
  */
48
55
  readonly bucket: IBucket;
49
56
  /**
50
- * The Lambda function that periodically updates stats.json.
57
+ * The Step Functions state machine that orchestrates stats processing.
58
+ */
59
+ readonly stateMachine: sfn.StateMachine;
60
+ /**
61
+ * The Lambda functions used in the state machine.
51
62
  */
52
- readonly handler: IFunction;
63
+ readonly chunkerFunction: IFunction;
64
+ readonly processorFunction: IFunction;
65
+ readonly aggregatorFunction: IFunction;
53
66
  /**
54
67
  * The key of the object storing the package stats.
55
68
  */
@@ -7,10 +7,13 @@ const events = require("aws-cdk-lib/aws-events");
7
7
  const targets = require("aws-cdk-lib/aws-events-targets");
8
8
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
9
9
  const aws_logs_1 = require("aws-cdk-lib/aws-logs");
10
+ const sfn = require("aws-cdk-lib/aws-stepfunctions");
11
+ const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
10
12
  const constructs_1 = require("constructs");
11
13
  const constants_1 = require("./constants");
12
- const package_stats_1 = require("./package-stats");
13
- const deep_link_1 = require("../../deep-link");
14
+ const package_stats_aggregator_1 = require("./package-stats-aggregator");
15
+ const package_stats_chunker_1 = require("./package-stats-chunker");
16
+ const package_stats_processor_1 = require("./package-stats-processor");
14
17
  const runbook_url_1 = require("../../runbook-url");
15
18
  const constants_2 = require("../shared/constants");
16
19
  /**
@@ -21,36 +24,91 @@ class PackageStats extends constructs_1.Construct {
21
24
  super(scope, id);
22
25
  this.bucket = props.bucket;
23
26
  this.statsKey = props.objectKey;
24
- this.handler = new package_stats_1.PackageStats(this, 'Default', {
25
- description: `Creates the stats.json object in ${props.bucket.bucketName}`,
26
- environment: {
27
- CATALOG_BUCKET_NAME: this.bucket.bucketName,
28
- CATALOG_OBJECT_KEY: constants_2.CATALOG_KEY,
29
- STATS_BUCKET_NAME: this.bucket.bucketName,
30
- STATS_OBJECT_KEY: props.objectKey,
31
- },
27
+ const commonEnv = {
28
+ CATALOG_BUCKET_NAME: this.bucket.bucketName,
29
+ CATALOG_OBJECT_KEY: constants_2.CATALOG_KEY,
30
+ STATS_BUCKET_NAME: this.bucket.bucketName,
31
+ STATS_OBJECT_KEY: props.objectKey,
32
+ CHUNK_SIZE: (props.chunkSize ?? 100).toString(),
33
+ };
34
+ // Create Lambda functions
35
+ this.chunkerFunction = new package_stats_chunker_1.PackageStatsChunker(this, 'Chunker', {
36
+ description: 'Splits package list into chunks for parallel processing',
37
+ environment: commonEnv,
32
38
  logRetention: props.logRetention ?? aws_logs_1.RetentionDays.TEN_YEARS,
33
- memorySize: 256,
34
- reservedConcurrentExecutions: 1,
35
- timeout: aws_cdk_lib_1.Duration.minutes(15),
39
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
36
40
  tracing: aws_lambda_1.Tracing.PASS_THROUGH,
37
41
  });
42
+ this.processorFunction = new package_stats_processor_1.PackageStatsProcessor(this, 'Processor', {
43
+ description: 'Processes a chunk of packages to get NPM stats',
44
+ environment: commonEnv,
45
+ logRetention: props.logRetention ?? aws_logs_1.RetentionDays.TEN_YEARS,
46
+ memorySize: 1024,
47
+ timeout: aws_cdk_lib_1.Duration.minutes(10),
48
+ tracing: aws_lambda_1.Tracing.PASS_THROUGH,
49
+ });
50
+ this.aggregatorFunction = new package_stats_aggregator_1.PackageStatsAggregator(this, 'Aggregator', {
51
+ description: 'Aggregates processed chunks into final stats.json',
52
+ environment: commonEnv,
53
+ logRetention: props.logRetention ?? aws_logs_1.RetentionDays.TEN_YEARS,
54
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
55
+ tracing: aws_lambda_1.Tracing.PASS_THROUGH,
56
+ });
57
+ // Grant S3 permissions
58
+ this.bucket.grantReadWrite(this.chunkerFunction);
59
+ this.bucket.grantReadWrite(this.processorFunction);
60
+ this.bucket.grantReadWrite(this.aggregatorFunction);
61
+ // Create Step Functions state machine
62
+ const chunkPackages = new tasks.LambdaInvoke(this, 'ChunkPackages', {
63
+ lambdaFunction: this.chunkerFunction,
64
+ resultPath: '$.chunks',
65
+ });
66
+ const processChunksMap = new sfn.Map(this, 'ProcessChunksMap', {
67
+ itemsPath: '$.chunks.Payload.chunks',
68
+ maxConcurrency: 10,
69
+ resultPath: '$.processResults',
70
+ });
71
+ processChunksMap.itemProcessor(new tasks.LambdaInvoke(this, 'ProcessChunk', {
72
+ lambdaFunction: this.processorFunction,
73
+ inputPath: '$',
74
+ })
75
+ .addRetry({
76
+ errors: ['States.ALL'],
77
+ maxAttempts: 1,
78
+ backoffRate: 1,
79
+ interval: aws_cdk_lib_1.Duration.minutes(5),
80
+ })
81
+ .addCatch(new sfn.Pass(this, 'ProcessChunkError', {
82
+ result: sfn.Result.fromObject({ error: 'Failed to process chunk' }),
83
+ }), {
84
+ resultPath: '$.error',
85
+ }));
86
+ const aggregateResults = new tasks.LambdaInvoke(this, 'AggregateResults', {
87
+ lambdaFunction: this.aggregatorFunction,
88
+ inputPath: '$',
89
+ });
90
+ const definition = chunkPackages
91
+ .next(processChunksMap)
92
+ .next(aggregateResults);
93
+ this.stateMachine = new sfn.StateMachine(this, 'StateMachine', {
94
+ definition,
95
+ timeout: aws_cdk_lib_1.Duration.hours(6),
96
+ });
97
+ // Schedule the state machine
38
98
  const updatePeriod = props.updatePeriod ?? aws_cdk_lib_1.Duration.days(1);
39
99
  const rule = new events.Rule(this, 'Rule', {
40
100
  schedule: events.Schedule.rate(updatePeriod),
41
101
  });
42
- rule.addTarget(new targets.LambdaFunction(this.handler));
43
- this.bucket.grantReadWrite(this.handler);
44
- const failureAlarm = this.handler
45
- .metricErrors()
102
+ rule.addTarget(new targets.SfnStateMachine(this.stateMachine));
103
+ // Create alarms
104
+ const failureAlarm = this.stateMachine
105
+ .metricFailed()
46
106
  .createAlarm(scope, 'PackageStats/Failures', {
47
107
  alarmName: `${scope.node.path}/PackageStats/Failures`,
48
108
  alarmDescription: [
49
- 'The package stats function failed!',
109
+ 'The package stats state machine failed!',
50
110
  '',
51
111
  `RunBook: ${runbook_url_1.RUNBOOK_URL}`,
52
- '',
53
- `Direct link to Lambda function: ${(0, deep_link_1.lambdaFunctionUrl)(this.handler)}`,
54
112
  ].join('\n'),
55
113
  comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
56
114
  evaluationPeriods: 1,
@@ -70,4 +128,4 @@ class PackageStats extends constructs_1.Construct {
70
128
  }
71
129
  }
72
130
  exports.PackageStats = PackageStats;
73
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/backend/package-stats/index.ts"],"names":[],"mappings":";;;AAAA,6CAAuC;AACvC,+DAMoC;AACpC,iDAAiD;AACjD,0DAA0D;AAC1D,uDAA4D;AAC5D,mDAAqD;AAErD,2CAAuC;AACvC,2CAA4D;AAC5D,mDAA0D;AAC1D,+CAAoD;AAEpD,mDAAgD;AAChD,mDAAkD;AAuClD;;GAEG;AACH,MAAa,YAAa,SAAQ,sBAAS;IAiBzC,YAAmB,KAAgB,EAAE,EAAU,EAAE,KAAwB;QACvE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC3B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,SAAS,CAAC;QAEhC,IAAI,CAAC,OAAO,GAAG,IAAI,4BAAO,CAAC,IAAI,EAAE,SAAS,EAAE;YAC1C,WAAW,EAAE,oCAAoC,KAAK,CAAC,MAAM,CAAC,UAAU,EAAE;YAC1E,WAAW,EAAE;gBACX,mBAAmB,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU;gBAC3C,kBAAkB,EAAE,uBAAW;gBAC/B,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU;gBACzC,gBAAgB,EAAE,KAAK,CAAC,SAAS;aAClC;YACD,YAAY,EAAE,KAAK,CAAC,YAAY,IAAI,wBAAa,CAAC,SAAS;YAC3D,UAAU,EAAE,GAAG;YACf,4BAA4B,EAAE,CAAC;YAC/B,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,OAAO,EAAE,oBAAO,CAAC,YAAY;SAC9B,CAAC,CAAC;QAEH,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,IAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC5D,MAAM,IAAI,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE;YACzC,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC;SAC7C,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,CAAC,IAAI,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QAEzD,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAEzC,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO;aAC9B,YAAY,EAAE;aACd,WAAW,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC3C,SAAS,EAAE,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,wBAAwB;YACrD,gBAAgB,EAAE;gBAChB,oCAAoC;gBACpC,EAAE;gBACF,YAAY,yBAAW,EAAE;gBACzB,EAAE;gBACF,mCAAmC,IAAA,6BAAiB,EAAC,IAAI,CAAC,OAAO,CAAC,EAAE;aACrE,CAAC,IAAI,CAAC,IAAI,CAAC;YACZ,kBAAkB,EAChB,mCAAkB,CAAC,kCAAkC;YACvD,iBAAiB,EAAE,CAAC;YACpB,SAAS,EAAE,CAAC;YACZ,gBAAgB,EAAE,iCAAgB,CAAC,OAAO;SAC3C,CAAC,CAAC;QACL,KAAK,CAAC,UAAU,CAAC,mBAAmB,CAAC,uBAAuB,EAAE,YAAY,CAAC,CAAC;IAC9E,CAAC;IAEM,mBAAmB,CAAC,IAAoB;QAC7C,OAAO,IAAI,uBAAM,CAAC;YAChB,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC3B,SAAS,EAAE,0BAAS,CAAC,OAAO;YAC5B,GAAG,IAAI;YACP,UAAU,+EAA2C;YACrD,SAAS,EAAE,6BAAiB;SAC7B,CAAC,CAAC;IACL,CAAC;CACF;AA3ED,oCA2EC","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport {\n  ComparisonOperator,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} from 'aws-cdk-lib/aws-cloudwatch';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport { IFunction, Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { MetricName, METRICS_NAMESPACE } from './constants';\nimport { PackageStats as Handler } from './package-stats';\nimport { lambdaFunctionUrl } from '../../deep-link';\nimport { Monitoring } from '../../monitoring';\nimport { RUNBOOK_URL } from '../../runbook-url';\nimport { CATALOG_KEY } from '../shared/constants';\n\n/**\n * Props for `PackageStats`.\n */\nexport interface PackageStatsProps {\n  /**\n   * The package store bucket, which should include both the\n   * catalog and stats.\n   */\n  readonly bucket: IBucket;\n\n  /**\n   * The monitoring handler to register alarms with.\n   */\n  readonly monitoring: Monitoring;\n\n  /**\n   * How long should execution logs be retained?\n   *\n   * @default RetentionDays.TEN_YEARS\n   */\n  readonly logRetention?: RetentionDays;\n\n  /**\n   * How frequently should the stats be updated?\n   *\n   * NPM updates their download stats once a day.\n   *\n   * @default - 1 day\n   */\n  readonly updatePeriod?: Duration;\n\n  /**\n   * The key of the object storing the package stats.\n   */\n  readonly objectKey: string;\n}\n\n/**\n * Builds or re-builds the `stats.json` object in the designated bucket.\n */\nexport class PackageStats extends Construct {\n  /**\n   * The package store bucket, which should include both the\n   * catalog and stats.\n   */\n  public readonly bucket: IBucket;\n\n  /**\n   * The Lambda function that periodically updates stats.json.\n   */\n  public readonly handler: IFunction;\n\n  /**\n   * The key of the object storing the package stats.\n   */\n  public readonly statsKey: string;\n\n  public constructor(scope: Construct, id: string, props: PackageStatsProps) {\n    super(scope, id);\n\n    this.bucket = props.bucket;\n    this.statsKey = props.objectKey;\n\n    this.handler = new Handler(this, 'Default', {\n      description: `Creates the stats.json object in ${props.bucket.bucketName}`,\n      environment: {\n        CATALOG_BUCKET_NAME: this.bucket.bucketName,\n        CATALOG_OBJECT_KEY: CATALOG_KEY,\n        STATS_BUCKET_NAME: this.bucket.bucketName,\n        STATS_OBJECT_KEY: props.objectKey,\n      },\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      memorySize: 256,\n      reservedConcurrentExecutions: 1,\n      timeout: Duration.minutes(15),\n      tracing: Tracing.PASS_THROUGH,\n    });\n\n    const updatePeriod = props.updatePeriod ?? Duration.days(1);\n    const rule = new events.Rule(this, 'Rule', {\n      schedule: events.Schedule.rate(updatePeriod),\n    });\n    rule.addTarget(new targets.LambdaFunction(this.handler));\n\n    this.bucket.grantReadWrite(this.handler);\n\n    const failureAlarm = this.handler\n      .metricErrors()\n      .createAlarm(scope, 'PackageStats/Failures', {\n        alarmName: `${scope.node.path}/PackageStats/Failures`,\n        alarmDescription: [\n          'The package stats function failed!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to Lambda function: ${lambdaFunctionUrl(this.handler)}`,\n        ].join('\\n'),\n        comparisonOperator:\n          ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: 1,\n        treatMissingData: TreatMissingData.MISSING,\n      });\n    props.monitoring.addLowSeverityAlarm('PackageStats Failures', failureAlarm);\n  }\n\n  public metricPackagesCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.REGISTERED_PACKAGES_WITH_STATS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/backend/package-stats/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport {\n  ComparisonOperator,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} from 'aws-cdk-lib/aws-cloudwatch';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport { IFunction, Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { IBucket } from 'aws-cdk-lib/aws-s3';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { MetricName, METRICS_NAMESPACE } from './constants';\nimport { PackageStatsAggregator } from './package-stats-aggregator';\nimport { PackageStatsChunker } from './package-stats-chunker';\nimport { PackageStatsProcessor } from './package-stats-processor';\n\nimport { Monitoring } from '../../monitoring';\nimport { RUNBOOK_URL } from '../../runbook-url';\nimport { CATALOG_KEY } from '../shared/constants';\n\n/**\n * Props for `PackageStats`.\n */\nexport interface PackageStatsProps {\n  /**\n   * The package store bucket, which should include both the\n   * catalog and stats.\n   */\n  readonly bucket: IBucket;\n\n  /**\n   * The monitoring handler to register alarms with.\n   */\n  readonly monitoring: Monitoring;\n\n  /**\n   * How long should execution logs be retained?\n   *\n   * @default RetentionDays.TEN_YEARS\n   */\n  readonly logRetention?: RetentionDays;\n\n  /**\n   * How frequently should the stats be updated?\n   *\n   * NPM updates their download stats once a day.\n   *\n   * @default - 1 day\n   */\n  readonly updatePeriod?: Duration;\n\n  /**\n   * The key of the object storing the package stats.\n   */\n  readonly objectKey: string;\n\n  /**\n   * Number of packages to process per chunk.\n   *\n   * @default 100\n   */\n  readonly chunkSize?: number;\n}\n\n/**\n * Builds or re-builds the `stats.json` object in the designated bucket.\n */\nexport class PackageStats extends Construct {\n  /**\n   * The package store bucket, which should include both the\n   * catalog and stats.\n   */\n  public readonly bucket: IBucket;\n\n  /**\n   * The Step Functions state machine that orchestrates stats processing.\n   */\n  public readonly stateMachine: sfn.StateMachine;\n\n  /**\n   * The Lambda functions used in the state machine.\n   */\n  public readonly chunkerFunction: IFunction;\n  public readonly processorFunction: IFunction;\n  public readonly aggregatorFunction: IFunction;\n\n  /**\n   * The key of the object storing the package stats.\n   */\n  public readonly statsKey: string;\n\n  public constructor(scope: Construct, id: string, props: PackageStatsProps) {\n    super(scope, id);\n\n    this.bucket = props.bucket;\n    this.statsKey = props.objectKey;\n\n    const commonEnv = {\n      CATALOG_BUCKET_NAME: this.bucket.bucketName,\n      CATALOG_OBJECT_KEY: CATALOG_KEY,\n      STATS_BUCKET_NAME: this.bucket.bucketName,\n      STATS_OBJECT_KEY: props.objectKey,\n      CHUNK_SIZE: (props.chunkSize ?? 100).toString(),\n    };\n\n    // Create Lambda functions\n    this.chunkerFunction = new PackageStatsChunker(this, 'Chunker', {\n      description: 'Splits package list into chunks for parallel processing',\n      environment: commonEnv,\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      timeout: Duration.minutes(5),\n      tracing: Tracing.PASS_THROUGH,\n    });\n\n    this.processorFunction = new PackageStatsProcessor(this, 'Processor', {\n      description: 'Processes a chunk of packages to get NPM stats',\n      environment: commonEnv,\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      memorySize: 1024,\n      timeout: Duration.minutes(10),\n      tracing: Tracing.PASS_THROUGH,\n    });\n\n    this.aggregatorFunction = new PackageStatsAggregator(this, 'Aggregator', {\n      description: 'Aggregates processed chunks into final stats.json',\n      environment: commonEnv,\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      timeout: Duration.minutes(5),\n      tracing: Tracing.PASS_THROUGH,\n    });\n\n    // Grant S3 permissions\n    this.bucket.grantReadWrite(this.chunkerFunction);\n    this.bucket.grantReadWrite(this.processorFunction);\n    this.bucket.grantReadWrite(this.aggregatorFunction);\n\n    // Create Step Functions state machine\n    const chunkPackages = new tasks.LambdaInvoke(this, 'ChunkPackages', {\n      lambdaFunction: this.chunkerFunction,\n      resultPath: '$.chunks',\n    });\n\n    const processChunksMap = new sfn.Map(this, 'ProcessChunksMap', {\n      itemsPath: '$.chunks.Payload.chunks',\n      maxConcurrency: 10,\n      resultPath: '$.processResults',\n    });\n\n    processChunksMap.itemProcessor(\n      new tasks.LambdaInvoke(this, 'ProcessChunk', {\n        lambdaFunction: this.processorFunction,\n        inputPath: '$',\n      })\n        .addRetry({\n          errors: ['States.ALL'],\n          maxAttempts: 1,\n          backoffRate: 1,\n          interval: Duration.minutes(5),\n        })\n        .addCatch(\n          new sfn.Pass(this, 'ProcessChunkError', {\n            result: sfn.Result.fromObject({ error: 'Failed to process chunk' }),\n          }),\n          {\n            resultPath: '$.error',\n          }\n        )\n    );\n\n    const aggregateResults = new tasks.LambdaInvoke(this, 'AggregateResults', {\n      lambdaFunction: this.aggregatorFunction,\n      inputPath: '$',\n    });\n\n    const definition = chunkPackages\n      .next(processChunksMap)\n      .next(aggregateResults);\n\n    this.stateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition,\n      timeout: Duration.hours(6),\n    });\n\n    // Schedule the state machine\n    const updatePeriod = props.updatePeriod ?? Duration.days(1);\n    const rule = new events.Rule(this, 'Rule', {\n      schedule: events.Schedule.rate(updatePeriod),\n    });\n    rule.addTarget(new targets.SfnStateMachine(this.stateMachine));\n\n    // Create alarms\n    const failureAlarm = this.stateMachine\n      .metricFailed()\n      .createAlarm(scope, 'PackageStats/Failures', {\n        alarmName: `${scope.node.path}/PackageStats/Failures`,\n        alarmDescription: [\n          'The package stats state machine failed!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n        ].join('\\n'),\n        comparisonOperator:\n          ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: 1,\n        treatMissingData: TreatMissingData.MISSING,\n      });\n    props.monitoring.addLowSeverityAlarm('PackageStats Failures', failureAlarm);\n  }\n\n  public metricPackagesCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.REGISTERED_PACKAGES_WITH_STATS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
@@ -0,0 +1,7 @@
1
+ import * as lambda from 'aws-cdk-lib/aws-lambda';
2
+ import { Construct } from 'constructs';
3
+ export interface PackageStatsAggregatorProps extends lambda.FunctionOptions {
4
+ }
5
+ export declare class PackageStatsAggregator extends lambda.Function {
6
+ constructor(scope: Construct, id: string, props?: PackageStatsAggregatorProps);
7
+ }
@@ -0,0 +1,20 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PackageStatsAggregator = void 0;
4
+ // ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
5
+ const path = require("path");
6
+ const lambda = require("aws-cdk-lib/aws-lambda");
7
+ class PackageStatsAggregator extends lambda.Function {
8
+ constructor(scope, id, props) {
9
+ super(scope, id, {
10
+ description: 'backend/package-stats/package-stats-aggregator.lambda.ts',
11
+ ...props,
12
+ architecture: lambda.Architecture.ARM_64,
13
+ runtime: lambda.Runtime.NODEJS_22_X,
14
+ handler: 'index.handler',
15
+ code: lambda.Code.fromAsset(path.join(__dirname, '/package-stats-aggregator.lambda.bundle')),
16
+ });
17
+ }
18
+ }
19
+ exports.PackageStatsAggregator = PackageStatsAggregator;
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFja2FnZS1zdGF0cy1hZ2dyZWdhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2JhY2tlbmQvcGFja2FnZS1zdGF0cy9wYWNrYWdlLXN0YXRzLWFnZ3JlZ2F0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkVBQTZFO0FBQzdFLDZCQUE2QjtBQUM3QixpREFBaUQ7QUFNakQsTUFBYSxzQkFBdUIsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUN6RCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQW1DO1FBQzNFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLDBEQUEwRDtZQUN2RSxHQUFHLEtBQUs7WUFDUixZQUFZLEVBQUUsTUFBTSxDQUFDLFlBQVksQ0FBQyxNQUFNO1lBQ3hDLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLHlDQUF5QyxDQUFDLENBQUM7U0FDN0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBWEQsd0RBV0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyB+fiBHZW5lcmF0ZWQgYnkgcHJvamVuLiBUbyBtb2RpZnksIGVkaXQgLnByb2plbnJjLnRzIGFuZCBydW4gXCJucHggcHJvamVuXCIuXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFja2FnZVN0YXRzQWdncmVnYXRvclByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbmV4cG9ydCBjbGFzcyBQYWNrYWdlU3RhdHNBZ2dyZWdhdG9yIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBQYWNrYWdlU3RhdHNBZ2dyZWdhdG9yUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnYmFja2VuZC9wYWNrYWdlLXN0YXRzL3BhY2thZ2Utc3RhdHMtYWdncmVnYXRvci5sYW1iZGEudHMnLFxuICAgICAgLi4ucHJvcHMsXG4gICAgICBhcmNoaXRlY3R1cmU6IGxhbWJkYS5BcmNoaXRlY3R1cmUuQVJNXzY0LFxuICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzIyX1gsXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy9wYWNrYWdlLXN0YXRzLWFnZ3JlZ2F0b3IubGFtYmRhLmJ1bmRsZScpKSxcbiAgICB9KTtcbiAgfVxufSJdfQ==