construct-hub 0.4.2 → 0.4.5

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.
@@ -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
- const handler = new npmjs_package_canary_1.NpmjsPackageCanary(this, 'Resource', {
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvcGFja2FnZS1zb3VyY2VzL25wbWpzL2NhbmFyeS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2Q0FBdUM7QUFDdkMsK0RBQThFO0FBQzlFLHVEQUF3RDtBQUN4RCx1RUFBZ0U7QUFFaEUsMkNBQXVDO0FBQ3ZDLGdGQUFrRjtBQUNsRiwyQ0FLcUI7QUFDckIsaUVBQXVFO0FBUXZFLE1BQWEsa0JBQW1CLFNBQVEsc0JBQVM7SUFDL0MsWUFDRSxLQUFnQixFQUNoQixFQUFVLEVBQ1YsS0FBOEI7UUFFOUIsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixNQUFNLE9BQU8sR0FBRyxJQUFJLHlDQUFPLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUM1QyxZQUFZLEVBQUUsZ0RBQXlCLENBQUMsSUFBSSxDQUFDO1lBQzdDLFdBQVcsRUFBRSxJQUFJLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSw0QkFBNEIsS0FBSyxDQUFDLFdBQVcsd0JBQXdCO1lBQ3JHLFdBQVcsRUFBRTtnQkFDWCxtQkFBbUIsRUFBRSxPQUFPO2dCQUM1Qix1REFBb0MsRUFBRSxLQUFLLENBQUMsbUJBQW1CO2dCQUMvRCwrREFBd0MsRUFBRSxLQUFLLENBQUMsTUFBTSxDQUFDLFVBQVU7Z0JBQ2pFLG1DQUEwQixFQUFFLEtBQUssQ0FBQyxXQUFXO2FBQzlDO1lBQ0QsVUFBVSxFQUFFLEtBQU07WUFDbEIsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztTQUM3QixDQUFDLENBQUM7UUFDSCxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDLGNBQWMsQ0FDdkMsT0FBTyxFQUNQLEdBQUcsb0NBQXNCLElBQUksZ0NBQXNCLEVBQUUsQ0FDdEQsQ0FBQztRQUVGLE1BQU0sUUFBUSxHQUFHLElBQUksaUJBQUksQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQzFDLFdBQVcsRUFBRSxrREFBa0Q7WUFDL0QsUUFBUSxFQUFFLHFCQUFRLENBQUMsSUFBSSxDQUFDLHNCQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzVDLE9BQU8sRUFBRSxDQUFDLElBQUksbUNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztTQUN2QyxDQUFDLENBQUM7UUFDSCx1RUFBdUU7UUFDdkUsUUFBUSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVNLGVBQWUsQ0FBQyxJQUFvQjtRQUN6QyxPQUFPLElBQUksdUJBQU0sQ0FBQztZQUNoQixNQUFNLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1lBQzNCLFNBQVMsRUFBRSwwQkFBUyxDQUFDLE9BQU87WUFDNUIsR0FBRyxJQUFJO1lBQ1AsVUFBVSw4QkFBdUI7WUFDakMsU0FBUyxFQUFFLDZCQUFpQjtTQUM3QixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRU0sbUJBQW1CLENBQUMsSUFBb0I7UUFDN0MsT0FBTyxJQUFJLHVCQUFNLENBQUM7WUFDaEIsTUFBTSxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztZQUMzQixTQUFTLEVBQUUsMEJBQVMsQ0FBQyxPQUFPO1lBQzVCLEdBQUcsSUFBSTtZQUNQLFVBQVUsdUNBQTRCO1lBQ3RDLFNBQVMsRUFBRSw2QkFBaUI7U0FDN0IsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVNLHlCQUF5QixDQUFDLElBQW9CO1FBQ25ELE9BQU8sSUFBSSx1QkFBTSxDQUFDO1lBQ2hCLE1BQU0sRUFBRSxzQkFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7WUFDM0IsU0FBUyxFQUFFLDBCQUFTLENBQUMsT0FBTztZQUM1QixHQUFHLElBQUk7WUFDUCxVQUFVLG1EQUFrQztZQUM1QyxTQUFTLEVBQUUsNkJBQWlCO1NBQzdCLENBQUMsQ0FBQztJQUNMLENBQUM7Q0FDRjtBQS9ERCxnREErREMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEdXJhdGlvbiB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IE1ldHJpY09wdGlvbnMsIE1ldHJpYywgU3RhdGlzdGljIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gnO1xuaW1wb3J0IHsgUnVsZSwgU2NoZWR1bGUgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtZXZlbnRzJztcbmltcG9ydCB7IExhbWJkYUZ1bmN0aW9uIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWV2ZW50cy10YXJnZXRzJztcbmltcG9ydCB7IElCdWNrZXQgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtczMnO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBncmF2aXRvbkxhbWJkYUlmQXZhaWxhYmxlIH0gZnJvbSAnLi4vLi4vLi4vYmFja2VuZC9fbGFtYmRhLWFyY2hpdGVjdHVyZSc7XG5pbXBvcnQge1xuICBFbnZpcm9ubWVudCxcbiAgT2JqZWN0S2V5LFxuICBNRVRSSUNTX05BTUVTUEFDRSxcbiAgTWV0cmljTmFtZSxcbn0gZnJvbSAnLi9jb25zdGFudHMnO1xuaW1wb3J0IHsgTnBtanNQYWNrYWdlQ2FuYXJ5IGFzIEhhbmRsZXIgfSBmcm9tICcuL25wbWpzLXBhY2thZ2UtY2FuYXJ5JztcblxuZXhwb3J0IGludGVyZmFjZSBOcG1Kc1BhY2thZ2VDYW5hcnlQcm9wcyB7XG4gIHJlYWRvbmx5IGJ1Y2tldDogSUJ1Y2tldDtcbiAgcmVhZG9ubHkgY29uc3RydWN0SHViQmFzZVVybDogc3RyaW5nO1xuICByZWFkb25seSBwYWNrYWdlTmFtZTogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgTnBtSnNQYWNrYWdlQ2FuYXJ5IGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgcHVibGljIGNvbnN0cnVjdG9yKFxuICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogTnBtSnNQYWNrYWdlQ2FuYXJ5UHJvcHNcbiAgKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIGNvbnN0IGhhbmRsZXIgPSBuZXcgSGFuZGxlcih0aGlzLCAnUmVzb3VyY2UnLCB7XG4gICAgICBhcmNoaXRlY3R1cmU6IGdyYXZpdG9uTGFtYmRhSWZBdmFpbGFibGUodGhpcyksXG4gICAgICBkZXNjcmlwdGlvbjogYFske3Njb3BlLm5vZGUucGF0aH0vUGFja2FnZUNhbmFyeV0gTW9uaXRvcnMgJHtwcm9wcy5wYWNrYWdlTmFtZX0gdmVyc2lvbnMgYXZhaWxhYmlsaXR5YCxcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIEFXU19FTUZfRU5WSVJPTk1FTlQ6ICdMb2NhbCcsXG4gICAgICAgIFtFbnZpcm9ubWVudC5DT05TVFJVQ1RfSFVCX0JBU0VfVVJMXTogcHJvcHMuY29uc3RydWN0SHViQmFzZVVybCxcbiAgICAgICAgW0Vudmlyb25tZW50LlBBQ0tBR0VfQ0FOQVJZX0JVQ0tFVF9OQU1FXTogcHJvcHMuYnVja2V0LmJ1Y2tldE5hbWUsXG4gICAgICAgIFtFbnZpcm9ubWVudC5QQUNLQUdFX05BTUVdOiBwcm9wcy5wYWNrYWdlTmFtZSxcbiAgICAgIH0sXG4gICAgICBtZW1vcnlTaXplOiAxMF8wMjQsXG4gICAgICB0aW1lb3V0OiBEdXJhdGlvbi5taW51dGVzKDEpLFxuICAgIH0pO1xuICAgIGNvbnN0IGdyYW50ID0gcHJvcHMuYnVja2V0LmdyYW50UmVhZFdyaXRlKFxuICAgICAgaGFuZGxlcixcbiAgICAgIGAke09iamVjdEtleS5TVEFURV9QUkVGSVh9KiR7T2JqZWN0S2V5LlNUQVRFX1NVRkZJWH1gXG4gICAgKTtcblxuICAgIGNvbnN0IHNjaGVkdWxlID0gbmV3IFJ1bGUodGhpcywgJ1NjaGVkdWxlJywge1xuICAgICAgZGVzY3JpcHRpb246ICdTY2hlZHVsZWQgZXhlY3V0aW9ucyBvZiB0aGUgTnBtSlMgcGFja2FnZSBjYW5hcnknLFxuICAgICAgc2NoZWR1bGU6IFNjaGVkdWxlLnJhdGUoRHVyYXRpb24ubWludXRlcygxKSksXG4gICAgICB0YXJnZXRzOiBbbmV3IExhbWJkYUZ1bmN0aW9uKGhhbmRsZXIpXSxcbiAgICB9KTtcbiAgICAvLyBFbnN1cmUgd2UgZG9uJ3QgYXR0ZW1wdCB0byBydW4gYmVmb3JlIHBlcm1pc3Npb25zIGhhdmUgYmVlbiBncmFudGVkLlxuICAgIHNjaGVkdWxlLm5vZGUuYWRkRGVwZW5kZW5jeShncmFudCk7XG4gIH1cblxuICBwdWJsaWMgbWV0cmljRHdlbGxUaW1lKG9wdHM/OiBNZXRyaWNPcHRpb25zKTogTWV0cmljIHtcbiAgICByZXR1cm4gbmV3IE1ldHJpYyh7XG4gICAgICBwZXJpb2Q6IER1cmF0aW9uLm1pbnV0ZXMoMSksXG4gICAgICBzdGF0aXN0aWM6IFN0YXRpc3RpYy5NQVhJTVVNLFxuICAgICAgLi4ub3B0cyxcbiAgICAgIG1ldHJpY05hbWU6IE1ldHJpY05hbWUuRFdFTExfVElNRSxcbiAgICAgIG5hbWVzcGFjZTogTUVUUklDU19OQU1FU1BBQ0UsXG4gICAgfSk7XG4gIH1cblxuICBwdWJsaWMgbWV0cmljVGltZVRvQ2F0YWxvZyhvcHRzPzogTWV0cmljT3B0aW9ucyk6IE1ldHJpYyB7XG4gICAgcmV0dXJuIG5ldyBNZXRyaWMoe1xuICAgICAgcGVyaW9kOiBEdXJhdGlvbi5taW51dGVzKDEpLFxuICAgICAgc3RhdGlzdGljOiBTdGF0aXN0aWMuTUFYSU1VTSxcbiAgICAgIC4uLm9wdHMsXG4gICAgICBtZXRyaWNOYW1lOiBNZXRyaWNOYW1lLlRJTUVfVE9fQ0FUQUxPRyxcbiAgICAgIG5hbWVzcGFjZTogTUVUUklDU19OQU1FU1BBQ0UsXG4gICAgfSk7XG4gIH1cblxuICBwdWJsaWMgbWV0cmljVHJhY2tlZFZlcnNpb25Db3VudChvcHRzPzogTWV0cmljT3B0aW9ucyk6IE1ldHJpYyB7XG4gICAgcmV0dXJuIG5ldyBNZXRyaWMoe1xuICAgICAgcGVyaW9kOiBEdXJhdGlvbi5taW51dGVzKDEpLFxuICAgICAgc3RhdGlzdGljOiBTdGF0aXN0aWMuTUFYSU1VTSxcbiAgICAgIC4uLm9wdHMsXG4gICAgICBtZXRyaWNOYW1lOiBNZXRyaWNOYW1lLlRSQUNLRURfVkVSU0lPTl9DT1VOVCxcbiAgICAgIG5hbWVzcGFjZTogTUVUUklDU19OQU1FU1BBQ0UsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
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