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
+ import type { Context } from 'aws-lambda';
2
+ interface ProcessorResult {
3
+ chunkIndex: number;
4
+ chunkKey: string;
5
+ packageCount: number;
6
+ }
7
+ interface StepFunctionEvent {
8
+ processResults: Array<{
9
+ Payload: ProcessorResult;
10
+ }>;
11
+ }
12
+ export declare function handler(event: StepFunctionEvent, context: Context): Promise<void>;
13
+ export {};
@@ -0,0 +1,78 @@
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_embedded_metrics_1 = require("aws-embedded-metrics");
6
+ const constants_1 = require("./constants");
7
+ const caching_1 = require("../../caching");
8
+ const aws_lambda_shared_1 = require("../shared/aws.lambda-shared");
9
+ const env_lambda_shared_1 = require("../shared/env.lambda-shared");
10
+ async function handler(event, context) {
11
+ const STATS_BUCKET_NAME = (0, env_lambda_shared_1.requireEnv)('STATS_BUCKET_NAME');
12
+ const STATS_OBJECT_KEY = (0, env_lambda_shared_1.requireEnv)('STATS_OBJECT_KEY');
13
+ const results = event.processResults.map((r) => r.Payload);
14
+ console.log(`Aggregating ${results.length} chunks`);
15
+ const currentDate = new Date().toISOString();
16
+ const aggregatedStats = {};
17
+ let failedChunks = 0;
18
+ // Read and aggregate all chunk results
19
+ for (const result of results) {
20
+ // Skip results that have errors (failed chunks)
21
+ if ('error' in result) {
22
+ console.warn(`Skipping failed chunk ${result.chunkIndex}`);
23
+ failedChunks++;
24
+ continue;
25
+ }
26
+ try {
27
+ const response = await aws_lambda_shared_1.S3_CLIENT.send(new client_s3_1.GetObjectCommand({
28
+ Bucket: STATS_BUCKET_NAME,
29
+ Key: result.chunkKey,
30
+ }));
31
+ const chunkData = JSON.parse(await response.Body.transformToString());
32
+ Object.assign(aggregatedStats, chunkData);
33
+ // Clean up chunk file
34
+ await aws_lambda_shared_1.S3_CLIENT.send(new client_s3_1.DeleteObjectCommand({
35
+ Bucket: STATS_BUCKET_NAME,
36
+ Key: result.chunkKey,
37
+ }));
38
+ }
39
+ catch (error) {
40
+ console.error(`Failed to process chunk ${result.chunkIndex}:`, error);
41
+ failedChunks++;
42
+ }
43
+ }
44
+ const finalStats = {
45
+ packages: aggregatedStats,
46
+ updated: currentDate,
47
+ };
48
+ const statsCount = Object.keys(aggregatedStats).length;
49
+ console.log(`Aggregated stats for ${statsCount} packages (${failedChunks} chunks failed)`);
50
+ if (failedChunks > 0) {
51
+ console.warn(`Warning: ${failedChunks} chunks failed, stats may be incomplete`);
52
+ }
53
+ // Update metrics
54
+ await (0, aws_embedded_metrics_1.metricScope)((metrics) => async () => {
55
+ metrics.setDimensions({});
56
+ metrics.setNamespace(constants_1.METRICS_NAMESPACE);
57
+ metrics.putMetric("RegisteredPackagesWithStats" /* MetricName.REGISTERED_PACKAGES_WITH_STATS */, statsCount, aws_embedded_metrics_1.Unit.Count);
58
+ })();
59
+ // Upload final stats to S3
60
+ await aws_lambda_shared_1.S3_CLIENT.send(new client_s3_1.PutObjectCommand({
61
+ Bucket: STATS_BUCKET_NAME,
62
+ Key: STATS_OBJECT_KEY,
63
+ Body: JSON.stringify(finalStats, null, 2),
64
+ ContentType: 'application/json',
65
+ CacheControl: caching_1.CacheStrategy.default().toString(),
66
+ Metadata: {
67
+ 'Lambda-Log-Group': context.logGroupName,
68
+ 'Lambda-Log-Stream': context.logStreamName,
69
+ 'Lambda-Run-Id': context.awsRequestId,
70
+ 'Package-Stats-Count': `${statsCount}`,
71
+ },
72
+ }));
73
+ // Fail the state machine if any chunks failed
74
+ if (failedChunks > 0) {
75
+ throw new Error(`${failedChunks} chunks failed during processing`);
76
+ }
77
+ }
78
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"package-stats-aggregator.lambda.js","sourceRoot":"","sources":["../../../src/backend/package-stats/package-stats-aggregator.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  GetObjectCommand,\n  PutObjectCommand,\n  DeleteObjectCommand,\n} from '@aws-sdk/client-s3';\nimport { metricScope, Unit } from 'aws-embedded-metrics';\nimport type { Context } from 'aws-lambda';\nimport { MetricName, METRICS_NAMESPACE } from './constants';\nimport { CacheStrategy } from '../../caching';\nimport { S3_CLIENT } from '../shared/aws.lambda-shared';\nimport { requireEnv } from '../shared/env.lambda-shared';\n\ninterface ProcessorResult {\n  chunkIndex: number;\n  chunkKey: string;\n  packageCount: number;\n}\n\ninterface StepFunctionEvent {\n  processResults: Array<{\n    Payload: ProcessorResult;\n  }>;\n}\n\nexport async function handler(event: StepFunctionEvent, context: Context) {\n  const STATS_BUCKET_NAME = requireEnv('STATS_BUCKET_NAME');\n  const STATS_OBJECT_KEY = requireEnv('STATS_OBJECT_KEY');\n\n  const results = event.processResults.map((r) => r.Payload);\n  console.log(`Aggregating ${results.length} chunks`);\n\n  const currentDate = new Date().toISOString();\n  const aggregatedStats: { [key: string]: { downloads: { npm: number } } } = {};\n  let failedChunks = 0;\n\n  // Read and aggregate all chunk results\n  for (const result of results) {\n    // Skip results that have errors (failed chunks)\n    if ('error' in result) {\n      console.warn(`Skipping failed chunk ${result.chunkIndex}`);\n      failedChunks++;\n      continue;\n    }\n\n    try {\n      const response = await S3_CLIENT.send(\n        new GetObjectCommand({\n          Bucket: STATS_BUCKET_NAME,\n          Key: result.chunkKey,\n        })\n      );\n\n      const chunkData = JSON.parse(await response.Body!.transformToString());\n      Object.assign(aggregatedStats, chunkData);\n\n      // Clean up chunk file\n      await S3_CLIENT.send(\n        new DeleteObjectCommand({\n          Bucket: STATS_BUCKET_NAME,\n          Key: result.chunkKey,\n        })\n      );\n    } catch (error) {\n      console.error(`Failed to process chunk ${result.chunkIndex}:`, error);\n      failedChunks++;\n    }\n  }\n\n  const finalStats = {\n    packages: aggregatedStats,\n    updated: currentDate,\n  };\n\n  const statsCount = Object.keys(aggregatedStats).length;\n  console.log(\n    `Aggregated stats for ${statsCount} packages (${failedChunks} chunks failed)`\n  );\n\n  if (failedChunks > 0) {\n    console.warn(\n      `Warning: ${failedChunks} chunks failed, stats may be incomplete`\n    );\n  }\n\n  // Update metrics\n  await metricScope((metrics) => async () => {\n    metrics.setDimensions({});\n    metrics.setNamespace(METRICS_NAMESPACE);\n    metrics.putMetric(\n      MetricName.REGISTERED_PACKAGES_WITH_STATS,\n      statsCount,\n      Unit.Count\n    );\n  })();\n\n  // Upload final stats to S3\n  await S3_CLIENT.send(\n    new PutObjectCommand({\n      Bucket: STATS_BUCKET_NAME,\n      Key: STATS_OBJECT_KEY,\n      Body: JSON.stringify(finalStats, null, 2),\n      ContentType: 'application/json',\n      CacheControl: CacheStrategy.default().toString(),\n      Metadata: {\n        'Lambda-Log-Group': context.logGroupName,\n        'Lambda-Log-Stream': context.logStreamName,\n        'Lambda-Run-Id': context.awsRequestId,\n        'Package-Stats-Count': `${statsCount}`,\n      },\n    })\n  );\n\n  // Fail the state machine if any chunks failed\n  if (failedChunks > 0) {\n    throw new Error(`${failedChunks} chunks failed during processing`);\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 PackageStatsChunkerProps extends lambda.FunctionOptions {
4
+ }
5
+ export declare class PackageStatsChunker extends lambda.Function {
6
+ constructor(scope: Construct, id: string, props?: PackageStatsChunkerProps);
7
+ }
@@ -0,0 +1,20 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PackageStatsChunker = 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 PackageStatsChunker extends lambda.Function {
8
+ constructor(scope, id, props) {
9
+ super(scope, id, {
10
+ description: 'backend/package-stats/package-stats-chunker.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-chunker.lambda.bundle')),
16
+ });
17
+ }
18
+ }
19
+ exports.PackageStatsChunker = PackageStatsChunker;
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFja2FnZS1zdGF0cy1jaHVua2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2JhY2tlbmQvcGFja2FnZS1zdGF0cy9wYWNrYWdlLXN0YXRzLWNodW5rZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkVBQTZFO0FBQzdFLDZCQUE2QjtBQUM3QixpREFBaUQ7QUFNakQsTUFBYSxtQkFBb0IsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUN0RCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWdDO1FBQ3hFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLHVEQUF1RDtZQUNwRSxHQUFHLEtBQUs7WUFDUixZQUFZLEVBQUUsTUFBTSxDQUFDLFlBQVksQ0FBQyxNQUFNO1lBQ3hDLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLGVBQWU7WUFDeEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLHNDQUFzQyxDQUFDLENBQUM7U0FDMUYsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBWEQsa0RBV0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyB+fiBHZW5lcmF0ZWQgYnkgcHJvamVuLiBUbyBtb2RpZnksIGVkaXQgLnByb2plbnJjLnRzIGFuZCBydW4gXCJucHggcHJvamVuXCIuXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFja2FnZVN0YXRzQ2h1bmtlclByb3BzIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uT3B0aW9ucyB7XG59XG5cbmV4cG9ydCBjbGFzcyBQYWNrYWdlU3RhdHNDaHVua2VyIGV4dGVuZHMgbGFtYmRhLkZ1bmN0aW9uIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBQYWNrYWdlU3RhdHNDaHVua2VyUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnYmFja2VuZC9wYWNrYWdlLXN0YXRzL3BhY2thZ2Utc3RhdHMtY2h1bmtlci5sYW1iZGEudHMnLFxuICAgICAgLi4ucHJvcHMsXG4gICAgICBhcmNoaXRlY3R1cmU6IGxhbWJkYS5BcmNoaXRlY3R1cmUuQVJNXzY0LFxuICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuTk9ERUpTXzIyX1gsXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQocGF0aC5qb2luKF9fZGlybmFtZSwgJy9wYWNrYWdlLXN0YXRzLWNodW5rZXIubGFtYmRhLmJ1bmRsZScpKSxcbiAgICB9KTtcbiAgfVxufSJdfQ==