construct-hub 0.4.430 → 0.4.431

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/.jsii +9 -9
  2. package/docs/operator-runbook.md +45 -0
  3. package/lib/backend/package-stats/index.d.ts +15 -2
  4. package/lib/backend/package-stats/index.js +79 -21
  5. package/lib/backend/package-stats/package-stats-aggregator.d.ts +7 -0
  6. package/lib/backend/package-stats/package-stats-aggregator.js +20 -0
  7. package/lib/backend/package-stats/package-stats-aggregator.lambda.bundle/index.js +62489 -0
  8. package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-aggregator.lambda.bundle}/index.js.map +4 -4
  9. package/lib/backend/package-stats/package-stats-aggregator.lambda.d.ts +13 -0
  10. package/lib/backend/package-stats/package-stats-aggregator.lambda.js +78 -0
  11. package/lib/backend/package-stats/package-stats-chunker.d.ts +7 -0
  12. package/lib/backend/package-stats/package-stats-chunker.js +20 -0
  13. package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js +60075 -0
  14. package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js.map +7 -0
  15. package/lib/backend/package-stats/package-stats-chunker.lambda.d.ts +6 -0
  16. package/lib/backend/package-stats/package-stats-chunker.lambda.js +25 -0
  17. package/lib/backend/package-stats/package-stats-processor.d.ts +7 -0
  18. package/lib/backend/package-stats/package-stats-processor.js +20 -0
  19. package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-processor.lambda.bundle}/index.js +36 -2556
  20. package/lib/backend/package-stats/package-stats-processor.lambda.bundle/index.js.map +7 -0
  21. package/lib/backend/package-stats/package-stats-processor.lambda.d.ts +10 -0
  22. package/lib/backend/package-stats/package-stats-processor.lambda.js +41 -0
  23. package/lib/backend-dashboard.js +8 -6
  24. package/lib/construct-hub.d.ts +1 -1
  25. package/lib/construct-hub.js +12 -4
  26. package/lib/package-sources/code-artifact.js +1 -1
  27. package/lib/package-sources/npmjs.js +1 -1
  28. package/lib/package-tag/index.js +3 -3
  29. package/lib/package-tag-group/index.js +2 -2
  30. package/lib/preload-file/index.js +1 -1
  31. package/lib/s3/storage.js +1 -1
  32. package/lib/spdx-license.js +1 -1
  33. package/package.json +7 -3
  34. package/lib/backend/package-stats/package-stats.d.ts +0 -7
  35. package/lib/backend/package-stats/package-stats.js +0 -20
  36. package/lib/backend/package-stats/package-stats.lambda.d.ts +0 -25
  37. package/lib/backend/package-stats/package-stats.lambda.js +0 -79
package/.jsii CHANGED
@@ -4438,7 +4438,7 @@
4438
4438
  "immutable": true,
4439
4439
  "locationInModule": {
4440
4440
  "filename": "src/construct-hub.ts",
4441
- "line": 591
4441
+ "line": 596
4442
4442
  },
4443
4443
  "name": "allAlarms",
4444
4444
  "type": {
@@ -4458,7 +4458,7 @@
4458
4458
  "immutable": true,
4459
4459
  "locationInModule": {
4460
4460
  "filename": "src/construct-hub.ts",
4461
- "line": 595
4461
+ "line": 604
4462
4462
  },
4463
4463
  "name": "grantPrincipal",
4464
4464
  "overrides": "aws-cdk-lib.aws_iam.IGrantable",
@@ -4475,7 +4475,7 @@
4475
4475
  "immutable": true,
4476
4476
  "locationInModule": {
4477
4477
  "filename": "src/construct-hub.ts",
4478
- "line": 561
4478
+ "line": 566
4479
4479
  },
4480
4480
  "name": "highSeverityAlarms",
4481
4481
  "type": {
@@ -4494,7 +4494,7 @@
4494
4494
  "immutable": true,
4495
4495
  "locationInModule": {
4496
4496
  "filename": "src/construct-hub.ts",
4497
- "line": 599
4497
+ "line": 608
4498
4498
  },
4499
4499
  "name": "ingestionQueue",
4500
4500
  "type": {
@@ -4510,7 +4510,7 @@
4510
4510
  "immutable": true,
4511
4511
  "locationInModule": {
4512
4512
  "filename": "src/construct-hub.ts",
4513
- "line": 583
4513
+ "line": 588
4514
4514
  },
4515
4515
  "name": "lowSeverityAlarms",
4516
4516
  "type": {
@@ -4531,7 +4531,7 @@
4531
4531
  "immutable": true,
4532
4532
  "locationInModule": {
4533
4533
  "filename": "src/construct-hub.ts",
4534
- "line": 572
4534
+ "line": 577
4535
4535
  },
4536
4536
  "name": "mediumSeverityAlarms",
4537
4537
  "type": {
@@ -6098,7 +6098,7 @@
6098
6098
  "kind": "enum",
6099
6099
  "locationInModule": {
6100
6100
  "filename": "src/construct-hub.ts",
6101
- "line": 736
6101
+ "line": 745
6102
6102
  },
6103
6103
  "members": [
6104
6104
  {
@@ -21425,6 +21425,6 @@
21425
21425
  "symbolId": "src/package-sources/npmjs:NpmJsProps"
21426
21426
  }
21427
21427
  },
21428
- "version": "0.4.430",
21429
- "fingerprint": "+ECqu94FgNMjmwocDKmv5WFiiKaa+QhGQEAavA0PdmU="
21428
+ "version": "0.4.431",
21429
+ "fingerprint": "ZVXHWcBRyngGUZrAvJBbqN3X3fT2wtg9JozdKP9G6Sg="
21430
21430
  }
@@ -769,6 +769,51 @@ to determine what is happening and resolve the problem.
769
769
  Once the canary starts running normally again, the alarm will clear itself
770
770
  without requiring any further intervention.
771
771
 
772
+ ### `ConstructHub/PackageStats/Failures`
773
+
774
+ #### Description
775
+
776
+ The package stats state machine has failed. This means the `stats.json` file
777
+ containing NPM download statistics was not updated successfully. The stats are
778
+ used to display download counts on package pages.
779
+
780
+ #### Investigation
781
+
782
+ The package stats feature uses a Step Functions state machine that orchestrates
783
+ three Lambda functions:
784
+
785
+ 1. **Chunker** - Splits the package list into chunks for parallel processing
786
+ 2. **Processor** - Fetches NPM download stats for each chunk of packages
787
+ 3. **Aggregator** - Combines all chunks into the final `stats.json` file
788
+
789
+ In the backend dashboard under *Package Stats*, click the *Package Stats State
790
+ Machine* button to access the Step Functions console. Review failed executions
791
+ to identify which step failed:
792
+
793
+ - If the **ChunkPackages** step failed, check the Chunker function logs
794
+ - If the **ProcessChunksMap** step failed, check the Processor function logs
795
+ - If the **AggregateResults** step failed, check the Aggregator function logs
796
+
797
+ Common failure causes:
798
+ - NPM API throttling or unavailability (affects Processor)
799
+ - Large number of packages causing timeout (should be handled by chunking)
800
+ - S3 access issues (affects all functions)
801
+ - Temporary chunk files not cleaned up properly (affects Aggregator)
802
+
803
+ For additional recommendations for diving into CloudWatch Logs, refer to the
804
+ [Diving into Lambda Function logs in CloudWatch Logs][#lambda-log-dive] section.
805
+
806
+ #### Resolution
807
+
808
+ The alarm will automatically go back to green once the next scheduled execution
809
+ of the state machine succeeds. The state machine runs daily by default.
810
+
811
+ If the failure was transient, no action is needed. If the issue persists,
812
+ manually trigger a new execution of the state machine from the Step Functions
813
+ console after addressing the root cause.
814
+
815
+ --------------------------------------------------------------------------------
816
+
772
817
  ## :information_source: General Recommendations
773
818
 
774
819
  ### Diving into Lambda Function logs in CloudWatch Logs
@@ -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==