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.
- package/.jsii +9 -9
- package/docs/operator-runbook.md +45 -0
- package/lib/backend/package-stats/index.d.ts +15 -2
- package/lib/backend/package-stats/index.js +79 -21
- package/lib/backend/package-stats/package-stats-aggregator.d.ts +7 -0
- package/lib/backend/package-stats/package-stats-aggregator.js +20 -0
- package/lib/backend/package-stats/package-stats-aggregator.lambda.bundle/index.js +62489 -0
- package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-aggregator.lambda.bundle}/index.js.map +4 -4
- package/lib/backend/package-stats/package-stats-aggregator.lambda.d.ts +13 -0
- package/lib/backend/package-stats/package-stats-aggregator.lambda.js +78 -0
- package/lib/backend/package-stats/package-stats-chunker.d.ts +7 -0
- package/lib/backend/package-stats/package-stats-chunker.js +20 -0
- package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js +60075 -0
- package/lib/backend/package-stats/package-stats-chunker.lambda.bundle/index.js.map +7 -0
- package/lib/backend/package-stats/package-stats-chunker.lambda.d.ts +6 -0
- package/lib/backend/package-stats/package-stats-chunker.lambda.js +25 -0
- package/lib/backend/package-stats/package-stats-processor.d.ts +7 -0
- package/lib/backend/package-stats/package-stats-processor.js +20 -0
- package/lib/backend/package-stats/{package-stats.lambda.bundle → package-stats-processor.lambda.bundle}/index.js +36 -2556
- package/lib/backend/package-stats/package-stats-processor.lambda.bundle/index.js.map +7 -0
- package/lib/backend/package-stats/package-stats-processor.lambda.d.ts +10 -0
- package/lib/backend/package-stats/package-stats-processor.lambda.js +41 -0
- package/lib/backend-dashboard.js +8 -6
- package/lib/construct-hub.d.ts +1 -1
- package/lib/construct-hub.js +12 -4
- package/lib/package-sources/code-artifact.js +1 -1
- package/lib/package-sources/npmjs.js +1 -1
- package/lib/package-tag/index.js +3 -3
- package/lib/package-tag-group/index.js +2 -2
- package/lib/preload-file/index.js +1 -1
- package/lib/s3/storage.js +1 -1
- package/lib/spdx-license.js +1 -1
- package/package.json +7 -3
- package/lib/backend/package-stats/package-stats.d.ts +0 -7
- package/lib/backend/package-stats/package-stats.js +0 -20
- package/lib/backend/package-stats/package-stats.lambda.d.ts +0 -25
- 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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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":
|
|
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.
|
|
21429
|
-
"fingerprint": "
|
|
21428
|
+
"version": "0.4.431",
|
|
21429
|
+
"fingerprint": "ZVXHWcBRyngGUZrAvJBbqN3X3fT2wtg9JozdKP9G6Sg="
|
|
21430
21430
|
}
|
package/docs/operator-runbook.md
CHANGED
|
@@ -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
|
|
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
|
|
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
|
|
13
|
-
const
|
|
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
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
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
|
-
|
|
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.
|
|
43
|
-
|
|
44
|
-
const failureAlarm = this.
|
|
45
|
-
.
|
|
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
|
|
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==
|