construct-hub 0.4.4 → 0.4.5
Sign up to get free protection for your applications and to get access to all the features.
- package/.jsii +21 -21
- package/API.md +4 -4
- package/changelog.md +1 -1
- package/docs/operator-runbook.md +32 -0
- package/lib/backend/catalog-builder/index.js +5 -1
- package/lib/construct-hub.js +1 -1
- package/lib/monitoring/api.d.ts +3 -3
- package/lib/monitoring/api.js +1 -1
- package/lib/monitoring/index.d.ts +4 -4
- package/lib/monitoring/index.js +5 -5
- package/lib/package-sources/code-artifact.js +1 -1
- package/lib/package-sources/npmjs/canary/constants.d.ts +8 -1
- package/lib/package-sources/npmjs/canary/constants.js +1 -1
- package/lib/package-sources/npmjs/canary/index.d.ts +16 -0
- package/lib/package-sources/npmjs/canary/index.js +35 -4
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js +20 -1
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js.map +2 -2
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.d.ts +58 -0
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.js +24 -3
- package/lib/package-sources/npmjs.js +63 -7
- 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 +1 -1
- package/releasetag.txt +1 -1
- package/version.txt +1 -1
@@ -12,7 +12,7 @@ const npmjs_package_canary_1 = require("./npmjs-package-canary");
|
|
12
12
|
class NpmJsPackageCanary extends constructs_1.Construct {
|
13
13
|
constructor(scope, id, props) {
|
14
14
|
super(scope, id);
|
15
|
-
|
15
|
+
this.handler = new npmjs_package_canary_1.NpmjsPackageCanary(this, 'Resource', {
|
16
16
|
architecture: _lambda_architecture_1.gravitonLambdaIfAvailable(this),
|
17
17
|
description: `[${scope.node.path}/PackageCanary] Monitors ${props.packageName} versions availability`,
|
18
18
|
environment: {
|
@@ -24,11 +24,11 @@ class NpmJsPackageCanary extends constructs_1.Construct {
|
|
24
24
|
memorySize: 10024,
|
25
25
|
timeout: aws_cdk_lib_1.Duration.minutes(1),
|
26
26
|
});
|
27
|
-
const grant = props.bucket.grantReadWrite(handler, `${"package-canary/" /* STATE_PREFIX */}*${".state.json" /* STATE_SUFFIX */}`);
|
27
|
+
const grant = props.bucket.grantReadWrite(this.handler, `${"package-canary/" /* STATE_PREFIX */}*${".state.json" /* STATE_SUFFIX */}`);
|
28
28
|
const schedule = new aws_events_1.Rule(this, 'Schedule', {
|
29
29
|
description: 'Scheduled executions of the NpmJS package canary',
|
30
30
|
schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(1)),
|
31
|
-
targets: [new aws_events_targets_1.LambdaFunction(handler)],
|
31
|
+
targets: [new aws_events_targets_1.LambdaFunction(this.handler)],
|
32
32
|
});
|
33
33
|
// Ensure we don't attempt to run before permissions have been granted.
|
34
34
|
schedule.node.addDependency(grant);
|
@@ -60,6 +60,37 @@ class NpmJsPackageCanary extends constructs_1.Construct {
|
|
60
60
|
namespace: constants_1.METRICS_NAMESPACE,
|
61
61
|
});
|
62
62
|
}
|
63
|
+
/**
|
64
|
+
* The estimated lag between the npm registry replica (replcate.npmjs.com) and
|
65
|
+
* the primary registry (registry.npmjs.com).
|
66
|
+
*
|
67
|
+
* IMPORTANT NOTE: This is based on the difference in modified timestamps for
|
68
|
+
* the probe package between the two and hence has a granularly no better than
|
69
|
+
* the publishing interval of this. Since the construct-hub-probe package is
|
70
|
+
* only published every 3 hours approximately, this metric has a resolution
|
71
|
+
* that is strictly worse than 3 hours.
|
72
|
+
*/
|
73
|
+
metricEstimatedNpmReplicaLag(opts) {
|
74
|
+
return new aws_cloudwatch_1.Metric({
|
75
|
+
period: aws_cdk_lib_1.Duration.minutes(1),
|
76
|
+
statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
|
77
|
+
...opts,
|
78
|
+
metricName: "EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */,
|
79
|
+
namespace: constants_1.METRICS_NAMESPACE,
|
80
|
+
});
|
81
|
+
}
|
82
|
+
metricErrors(opts) {
|
83
|
+
return this.handler.metricErrors(opts);
|
84
|
+
}
|
85
|
+
metricInvocations(opts) {
|
86
|
+
return this.handler.metricInvocations(opts);
|
87
|
+
}
|
88
|
+
metricThrottles(opts) {
|
89
|
+
return this.handler.metricThrottles(opts);
|
90
|
+
}
|
91
|
+
metricDuration(opts) {
|
92
|
+
return this.handler.metricDuration(opts);
|
93
|
+
}
|
63
94
|
}
|
64
95
|
exports.NpmJsPackageCanary = NpmJsPackageCanary;
|
65
|
-
//# sourceMappingURL=data:application/json;base64,
|
96
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport { MetricOptions, Metric, Statistic } from 'aws-cdk-lib/aws-cloudwatch';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { LambdaFunction } from 'aws-cdk-lib/aws-events-targets';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { gravitonLambdaIfAvailable } from '../../../backend/_lambda-architecture';\nimport {\n  Environment,\n  ObjectKey,\n  METRICS_NAMESPACE,\n  MetricName,\n} from './constants';\nimport { NpmjsPackageCanary as Handler } from './npmjs-package-canary';\n\nexport interface NpmJsPackageCanaryProps {\n  readonly bucket: IBucket;\n  readonly constructHubBaseUrl: string;\n  readonly packageName: string;\n}\n\nexport class NpmJsPackageCanary extends Construct {\n  private readonly handler: Handler;\n\n  public constructor(\n    scope: Construct,\n    id: string,\n    props: NpmJsPackageCanaryProps\n  ) {\n    super(scope, id);\n\n    this.handler = new Handler(this, 'Resource', {\n      architecture: gravitonLambdaIfAvailable(this),\n      description: `[${scope.node.path}/PackageCanary] Monitors ${props.packageName} versions availability`,\n      environment: {\n        AWS_EMF_ENVIRONMENT: 'Local',\n        [Environment.CONSTRUCT_HUB_BASE_URL]: props.constructHubBaseUrl,\n        [Environment.PACKAGE_CANARY_BUCKET_NAME]: props.bucket.bucketName,\n        [Environment.PACKAGE_NAME]: props.packageName,\n      },\n      memorySize: 10_024,\n      timeout: Duration.minutes(1),\n    });\n    const grant = props.bucket.grantReadWrite(\n      this.handler,\n      `${ObjectKey.STATE_PREFIX}*${ObjectKey.STATE_SUFFIX}`\n    );\n\n    const schedule = new Rule(this, 'Schedule', {\n      description: 'Scheduled executions of the NpmJS package canary',\n      schedule: Schedule.rate(Duration.minutes(1)),\n      targets: [new LambdaFunction(this.handler)],\n    });\n    // Ensure we don't attempt to run before permissions have been granted.\n    schedule.node.addDependency(grant);\n  }\n\n  public metricDwellTime(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.DWELL_TIME,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricTimeToCatalog(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.TIME_TO_CATALOG,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricTrackedVersionCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.TRACKED_VERSION_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The estimated lag between the npm registry replica (replcate.npmjs.com) and\n   * the primary registry (registry.npmjs.com).\n   *\n   * IMPORTANT NOTE: This is based on the difference in modified timestamps for\n   * the probe package between the two and hence has a granularly no better than\n   * the publishing interval of this. Since the construct-hub-probe package is\n   * only published every 3 hours approximately, this metric has a resolution\n   * that is strictly worse than 3 hours.\n   */\n  public metricEstimatedNpmReplicaLag(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.NPM_REPLICA_LAG,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricErrors(opts?: MetricOptions): Metric {\n    return this.handler.metricErrors(opts);\n  }\n\n  public metricInvocations(opts?: MetricOptions): Metric {\n    return this.handler.metricInvocations(opts);\n  }\n\n  public metricThrottles(opts?: MetricOptions): Metric {\n    return this.handler.metricThrottles(opts);\n  }\n\n  public metricDuration(opts?: MetricOptions): Metric {\n    return this.handler.metricDuration(opts);\n  }\n}\n"]}
|
@@ -8075,6 +8075,7 @@ var require_lib2 = __commonJS({
|
|
8075
8075
|
// src/package-sources/npmjs/canary/npmjs-package-canary.lambda.ts
|
8076
8076
|
var npmjs_package_canary_lambda_exports = {};
|
8077
8077
|
__export(npmjs_package_canary_lambda_exports, {
|
8078
|
+
CanaryStateService: () => CanaryStateService,
|
8078
8079
|
handler: () => handler
|
8079
8080
|
});
|
8080
8081
|
module.exports = __toCommonJS(npmjs_package_canary_lambda_exports);
|
@@ -8129,9 +8130,10 @@ async function handler(event) {
|
|
8129
8130
|
console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);
|
8130
8131
|
updateLatestIfNeeded(state, latest);
|
8131
8132
|
try {
|
8132
|
-
await (0, import_aws_embedded_metrics.metricScope)((metrics) => () => {
|
8133
|
+
await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
|
8133
8134
|
metrics.setDimensions();
|
8134
8135
|
metrics.putMetric("TrackedVersionCount" /* TRACKED_VERSION_COUNT */, Object.keys(state.pending).length + 1, import_aws_embedded_metrics.Unit.Count);
|
8136
|
+
metrics.putMetric("EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */, await stateService.npmReplicaLagSeconds(packageName), import_aws_embedded_metrics.Unit.Seconds);
|
8135
8137
|
})();
|
8136
8138
|
for (const versionState of [
|
8137
8139
|
state.latest,
|
@@ -8243,6 +8245,22 @@ var CanaryStateService = class {
|
|
8243
8245
|
console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
|
8244
8246
|
return { version, publishedAt: new Date(publishedAt) };
|
8245
8247
|
}
|
8248
|
+
async npmReplicaLagSeconds(packageName) {
|
8249
|
+
const encodedPackageName = encodeURIComponent(packageName);
|
8250
|
+
console.log(`Measuring NPM replica lag using ${packageName}...`);
|
8251
|
+
const [primaryDate, replicaDate] = await Promise.all([
|
8252
|
+
getModifiedTimestamp(`registry.npmjs.org`),
|
8253
|
+
getModifiedTimestamp(`replicate.npmjs.com/registry`)
|
8254
|
+
]);
|
8255
|
+
const deltaMs = primaryDate.getTime() - replicaDate.getTime();
|
8256
|
+
console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
|
8257
|
+
console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 36e5} hours behind)`);
|
8258
|
+
return deltaMs / 1e3;
|
8259
|
+
async function getModifiedTimestamp(baseUrl) {
|
8260
|
+
const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, ["time", "modified"]);
|
8261
|
+
return new Date(isoDate);
|
8262
|
+
}
|
8263
|
+
}
|
8246
8264
|
key(packageName) {
|
8247
8265
|
return `${"package-canary/" /* STATE_PREFIX */}${packageName}${".state.json" /* STATE_SUFFIX */}`;
|
8248
8266
|
}
|
@@ -8287,6 +8305,7 @@ function gunzip(readable) {
|
|
8287
8305
|
}
|
8288
8306
|
// Annotate the CommonJS export names for ESM import in node:
|
8289
8307
|
0 && (module.exports = {
|
8308
|
+
CanaryStateService,
|
8290
8309
|
handler
|
8291
8310
|
});
|
8292
8311
|
//# sourceMappingURL=index.js.map
|