construct-hub 0.4.6 → 0.4.7

Sign up to get free protection for your applications and to get access to all the features.
package/.jsii CHANGED
@@ -16651,6 +16651,6 @@
16651
16651
  "symbolId": "src/package-sources/npmjs:NpmJsProps"
16652
16652
  }
16653
16653
  },
16654
- "version": "0.4.6",
16655
- "fingerprint": "SQlDa8sKOmg48L3FvgMiodfZ2hsxN24ZUk13u9TWIFw="
16654
+ "version": "0.4.7",
16655
+ "fingerprint": "ux0h/QGgJhPbVwSBaDXTAWTOCpnwJSLoo5Ft8fo2scc="
16656
16656
  }
package/changelog.md CHANGED
@@ -1,7 +1,7 @@
1
1
 
2
- ### [0.4.6](https://github.com/cdklabs/construct-hub/compare/v0.4.5...v0.4.6) (2022-05-16)
2
+ ### [0.4.7](https://github.com/cdklabs/construct-hub/compare/v0.4.6...v0.4.7) (2022-05-25)
3
3
 
4
4
 
5
- ### Features
5
+ ### Bug Fixes
6
6
 
7
- * expose alarm sets publicly ([#897](https://github.com/cdklabs/construct-hub/issues/897)) ([dabb6cd](https://github.com/cdklabs/construct-hub/commit/dabb6cd2a664458817f289d69e7ffb74b0ba67ff))
7
+ * **npmjs:** canary alarms when replicate.npmjs.com is down ([#899](https://github.com/cdklabs/construct-hub/issues/899)) ([36867ef](https://github.com/cdklabs/construct-hub/commit/36867ef36993a802e85bd5be423f4ccd387b8732))
@@ -388,7 +388,7 @@ class ConstructHub extends constructs_1.Construct {
388
388
  }
389
389
  exports.ConstructHub = ConstructHub;
390
390
  _a = JSII_RTTI_SYMBOL_1;
391
- ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.6" };
391
+ ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.7" };
392
392
  /**
393
393
  * How possibly risky operations (such as doc-generation, which requires
394
394
  * installing the indexed packages in order to trans-literate sample code) are
@@ -281,5 +281,5 @@ class CodeArtifact {
281
281
  }
282
282
  exports.CodeArtifact = CodeArtifact;
283
283
  _a = JSII_RTTI_SYMBOL_1;
284
- CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.6" };
284
+ CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.7" };
285
285
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-artifact.js","sourceRoot":"","sources":["../../src/package-sources/code-artifact.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnFormat, Aws, Duration, Stack } from 'aws-cdk-lib';\nimport {\n  ComparisonOperator,\n  GraphWidget,\n  MathExpression,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} from 'aws-cdk-lib/aws-cloudwatch';\nimport { CfnRepository } from 'aws-cdk-lib/aws-codeartifact';\nimport { Rule } from 'aws-cdk-lib/aws-events';\nimport { LambdaFunction } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { BlockPublicAccess, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { Construct } from 'constructs';\nimport {\n  codeArtifactRepositoryUrl,\n  lambdaFunctionUrl,\n  lambdaSearchLogGroupUrl,\n  sqsQueueUrl,\n} from '../deep-link';\nimport { fillMetric } from '../metric-utils';\nimport type {\n  IPackageSource,\n  PackageSourceBindOptions,\n  PackageSourceBindResult,\n} from '../package-source';\nimport { S3StorageFactory } from '../s3/storage';\nimport { CodeArtifactForwarder } from './codeartifact/code-artifact-forwarder';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  DOMAIN_NAME_DIMENSION,\n  DOMAIN_OWNER_DIMENSION,\n  REPOSITORY_NAME_DIMENSION,\n} from './codeartifact/constants.lambda-shared';\n\nexport interface CodeArtifactProps {\n  /**\n   * The CodeArtifact repository where packages are obtained from.\n   */\n  readonly repository: CfnRepository;\n\n  /**\n   * The S3 bucket where packages will be staged.\n   */\n  readonly bucket?: IBucket;\n}\n\n/**\n * A package source that obtains package data from an npm CodeArtifact\n * repository.\n */\nexport class CodeArtifact implements IPackageSource {\n  public constructor(private readonly props: CodeArtifactProps) {}\n\n  public bind(\n    scope: Construct,\n    {\n      denyList,\n      ingestion,\n      licenseList,\n      monitoring,\n      overviewDashboard,\n      queue,\n    }: PackageSourceBindOptions\n  ): PackageSourceBindResult {\n    const idPrefix = this.props.repository.node.path;\n    const repositoryId = `${this.props.repository.attrDomainOwner}:${this.props.repository.attrDomainName}/${this.props.repository.attrName}`;\n\n    const storageFactory = S3StorageFactory.getOrCreate(scope);\n    const bucket =\n      this.props.bucket ||\n      storageFactory.newBucket(scope, `${idPrefix}/StagingBucket`, {\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        enforceSSL: true,\n        lifecycleRules: [{ expiration: Duration.days(30) }],\n      });\n    bucket.grantRead(ingestion);\n\n    const dlq = new Queue(scope, `${idPrefix}/DLQ`, {\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: Duration.days(14),\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    const forwarder = new CodeArtifactForwarder(\n      scope,\n      `${idPrefix}/Forwarder`,\n      {\n        deadLetterQueue: dlq,\n        description: `[${scope.node.path}/CodeArtifact/${repositoryId}] Handle CodeArtifact EventBridge events`,\n        environment: {\n          AWS_EMF_ENVIRONMENT: 'Local',\n          BUCKET_NAME: bucket.bucketName,\n          QUEUE_URL: queue.queueUrl,\n        },\n        memorySize: 1024,\n        timeout: Duration.seconds(60),\n        tracing: Tracing.ACTIVE,\n      }\n    );\n    bucket.grantReadWrite(forwarder);\n    denyList?.grantRead(forwarder);\n    licenseList.grantRead(forwarder);\n    queue.grantSendMessages(forwarder);\n    forwarder.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['codeartifact:GetPackageVersionAsset'],\n        resources: [\n          Stack.of(scope).formatArn({\n            service: 'codeartifact',\n            resource: 'package',\n            arnFormat: ArnFormat.SLASH_RESOURCE_NAME,\n            resourceName: [\n              this.props.repository.attrDomainName,\n              this.props.repository.attrName,\n              'npm', // package format\n              '*', // namespace/package-name\n            ].join('/'),\n          }),\n        ],\n      })\n    );\n\n    const rule = new Rule(scope, `${idPrefix}/EventBridge`, {\n      description: `${scope.node.path}/CodeArtifact/${repositoryId}/EventBridge`,\n      eventPattern: {\n        source: ['aws.codeartifact'],\n        detailType: ['CodeArtifact Package Version State Change'],\n        detail: {\n          domainOwner: [this.props.repository.attrDomainOwner],\n          domainName: [this.props.repository.attrDomainName],\n          repositoryName: [this.props.repository.attrName],\n          packageFormat: ['npm'],\n        },\n      },\n      targets: [new LambdaFunction(forwarder)],\n    });\n\n    const failureAlarm = forwarder\n      .metricErrors()\n      .createAlarm(scope, `${idPrefix}/Forwarder/Failures`, {\n        alarmName: `${scope.node.path}/CodeArtifact/${repositoryId}/Forwarder`,\n        alarmDescription: [\n          `The CodeArtifact fowarder for ${repositoryId} is failing`,\n          '',\n          `Link to the lambda function: ${lambdaFunctionUrl(forwarder)}`,\n        ].join('\\n'),\n        comparisonOperator:\n          ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 3,\n        threshold: 1,\n        treatMissingData: TreatMissingData.MISSING,\n      });\n    monitoring.addHighSeverityAlarm(\n      `CodeArtifact:${this.props.repository.attrName} Failures`,\n      failureAlarm\n    );\n\n    const dlqNotEmptyAlarm = new MathExpression({\n      expression: 'mVisible + mHidden',\n      usingMetrics: {\n        mVisible: dlq.metricApproximateNumberOfMessagesVisible({\n          period: Duration.minutes(1),\n        }),\n        mHidden: dlq.metricApproximateNumberOfMessagesNotVisible({\n          period: Duration.minutes(1),\n        }),\n      },\n    }).createAlarm(scope, `${idPrefix}/Forwarder/DLQNotEmpty`, {\n      alarmName: `${scope.node.path}/CodeArtifact/${repositoryId}/DLQNotEmpty`,\n      alarmDescription: [\n        `The CodeArtifact fowarder for ${repositoryId} is failing`,\n        '',\n        `Link to the lambda function: ${lambdaFunctionUrl(forwarder)}`,\n        `Link to the dead letter queue: ${sqsQueueUrl(dlq)}`,\n      ].join('/n'),\n      comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold: 1,\n      treatMissingData: TreatMissingData.NOT_BREACHING,\n    });\n    monitoring.addLowSeverityAlarm(\n      `CodeArtifact/${repositoryId} DLQ Not Empty`,\n      dlqNotEmptyAlarm\n    );\n\n    rule.node.addDependency(failureAlarm, dlqNotEmptyAlarm);\n\n    overviewDashboard.addDLQMetricToDashboard(\n      `CodeArtifact/${repositoryId} DLQ`,\n      dlq\n    );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      forwarder,\n      `${idPrefix}/ForwarderLambda`\n    );\n\n    return {\n      name: `CodeArtifact: ${repositoryId}`,\n      links: [\n        {\n          name: 'CodeArtifact',\n          url: codeArtifactRepositoryUrl(this.props.repository),\n          primary: true,\n        },\n        {\n          name: 'Forwarder Function',\n          url: lambdaFunctionUrl(forwarder),\n        },\n        {\n          name: 'Search Log group',\n          url: lambdaSearchLogGroupUrl(forwarder),\n        },\n        {\n          name: 'DLQ',\n          url: sqsQueueUrl(dlq),\n        },\n      ],\n      dashboardWidgets: [\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Function Health',\n            left: [\n              fillMetric(forwarder.metricInvocations({ label: 'Invocations' })),\n              fillMetric(forwarder.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [forwarder.metricDuration({ label: 'Duration' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(15),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Dead Letter Queue',\n            left: [\n              dlq.metricApproximateNumberOfMessagesVisible({\n                label: 'Visible Messages',\n                period: Duration.minutes(1),\n              }),\n              dlq.metricApproximateNumberOfMessagesNotVisible({\n                label: 'Hidden Messages',\n                period: Duration.minutes(1),\n              }),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              dlq.metricApproximateAgeOfOldestMessage({\n                label: 'Oldest Message Age',\n                period: Duration.minutes(1),\n              }),\n            ],\n            rightYAxis: { min: 0 },\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Quality Metrics',\n            left: [\n              fillMetric(\n                this.metricNotJsiiEnabledCount({ label: 'Not a jsii package' }),\n                0\n              ),\n              fillMetric(\n                this.metricIneligibleLicense({ label: 'Ineligible License' }),\n                0\n              ),\n              fillMetric(\n                this.metricDenyListedCount({ label: 'Deny Listed' }),\n                0\n              ),\n              fillMetric(\n                this.metricDeletedCount({ label: 'Deletion Events' }),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n          }),\n        ],\n      ],\n    };\n  }\n\n  /**\n   * The count of package versions that were ignored due to being in the deny list.\n   */\n  public metricDenyListedCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]: this.props.repository.attrDomainOwner,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.DENY_LISTED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that were ignored due to using an ineloigible license.\n   */\n  public metricIneligibleLicense(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.INELIGIBLE_LICENSE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that were deleted from CodeArtifact (those events are not\n   * handled currently).\n   */\n  public metricDeletedCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.DELETED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that do not have a jsii assembly in the package.\n   */\n  public metricNotJsiiEnabledCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.NOT_JSII_ENABLED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
@@ -30,7 +30,13 @@ export declare const enum MetricName {
30
30
  * directly because the primary does not expose the relevant CouchDB endpoints,
31
31
  * so we use the probe package to get a low-resolution view of this.
32
32
  */
33
- NPM_REPLICA_LAG = "EstimatedNpmReplicaLag"
33
+ NPM_REPLICA_LAG = "EstimatedNpmReplicaLag",
34
+ /**
35
+ * A metric tracking whether the npm registry replica (replicate.npmjs.com)
36
+ * is down. The value is 1 when the replica is detected to be down, and 0
37
+ * when the replica is detected to be up.
38
+ */
39
+ NPM_REPLICA_DOWN = "NpmReplicaIsDown"
34
40
  }
35
41
  export declare const enum ObjectKey {
36
42
  STATE_PREFIX = "package-canary/",
@@ -2,4 +2,4 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.METRICS_NAMESPACE = void 0;
4
4
  exports.METRICS_NAMESPACE = 'ConstructHub/PackageCanary';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3BhY2thZ2Utc291cmNlcy9ucG1qcy9jYW5hcnkvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQU1hLFFBQUEsaUJBQWlCLEdBQUcsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgZW51bSBFbnZpcm9ubWVudCB7XG4gIFBBQ0tBR0VfTkFNRSA9ICdQQUNLQUdFX05BTUUnLFxuICBQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRSA9ICdQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRScsXG4gIENPTlNUUlVDVF9IVUJfQkFTRV9VUkwgPSAnQ09OU1RSVUNUX0hVQl9CQVNFX1VSTCcsXG59XG5cbmV4cG9ydCBjb25zdCBNRVRSSUNTX05BTUVTUEFDRSA9ICdDb25zdHJ1Y3RIdWIvUGFja2FnZUNhbmFyeSc7XG5cbmV4cG9ydCBjb25zdCBlbnVtIE1ldHJpY05hbWUge1xuICAvKipcbiAgICogVGhlIHRpbWUgZWxhcHNlZCBzaW5jZSBhIHBhY2thZ2Ugd2FzIHB1Ymxpc2hlZCB0byBucG1qcy5jb20sIHdoaWxlIGl0IGhhc1xuICAgKiBub3QgYmVlbiBkZXRlY3RlZCBpbiB0aGUgY2F0YWxvZyBieSB0aGUgcGFja2FnZSBjYW5hcnkuXG4gICAqXG4gICAqIFRoaXMgbWV0cmljcyBpcyBvbmx5IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBEV0VMTF9USU1FID0gJ0R3ZWxsVGltZScsXG5cbiAgLyoqXG4gICAqIFRoZSB0aW1lIGl0IHRvb2sgYmV0d2VlbiBhIHBhY2thZ2UncyBwdWJsaWNhdGlvbiB0byBucG1qcy5jb20gYW5kIHdoZW4gdGhlXG4gICAqIHBhY2thZ2UgY2FuYXJ5IGRldGVjdGVkIHRoZSBwYWNrYWdlIGFzIGF2YWlsYWJsZSBpbiB0aGUgY2F0YWxvZy5cbiAgICpcbiAgICogVGhpcyBtZXRyaWMgaXMgbm90IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGFjdHVhbGx5IGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBUSU1FX1RPX0NBVEFMT0cgPSAnVGltZVRvQ2F0YWxvZycsXG5cbiAgLyoqXG4gICAqIFRoZSBudW1iZXIgb2YgcGFja2FnZSB2ZXJzaW9ucyB0aGF0IHdlcmUgdHJhY2tlZCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZVxuICAgKiBwYWNrYWdlIGNhbmFyeSBleGVjdXRpb24gdGhhdCBwcm9kdWNlZCB0aGUgZGF0YSBwb2ludC5cbiAgICovXG4gIFRSQUNLRURfVkVSU0lPTl9DT1VOVCA9ICdUcmFja2VkVmVyc2lvbkNvdW50JyxcblxuICAvKipcbiAgICogVGhlIGVzdGltYXRlZCBsYWcgYmV0d2VlbiB0aGUgbnBtIHJlZ2lzdHJ5IHJlcGxpY2EgKHJlcGxpY2F0ZS5ucG1qcy5jb20pXG4gICAqIGFuZCB0aGUgcHJpbWFyeSByZWdpc3RyeSAocmVnaXN0cnkubnBtanMuY29tKS4gVGhpcyBjYW5ub3QgYmUgbWVhc3VyZWRcbiAgICogZGlyZWN0bHkgYmVjYXVzZSB0aGUgcHJpbWFyeSBkb2VzIG5vdCBleHBvc2UgdGhlIHJlbGV2YW50IENvdWNoREIgZW5kcG9pbnRzLFxuICAgKiBzbyB3ZSB1c2UgdGhlIHByb2JlIHBhY2thZ2UgdG8gZ2V0IGEgbG93LXJlc29sdXRpb24gdmlldyBvZiB0aGlzLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfTEFHID0gJ0VzdGltYXRlZE5wbVJlcGxpY2FMYWcnLFxufVxuXG5leHBvcnQgY29uc3QgZW51bSBPYmplY3RLZXkge1xuICBTVEFURV9QUkVGSVggPSAncGFja2FnZS1jYW5hcnkvJyxcbiAgU1RBVEVfU1VGRklYID0gJy5zdGF0ZS5qc29uJyxcbn1cbiJdfQ==
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3BhY2thZ2Utc291cmNlcy9ucG1qcy9jYW5hcnkvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQU1hLFFBQUEsaUJBQWlCLEdBQUcsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgZW51bSBFbnZpcm9ubWVudCB7XG4gIFBBQ0tBR0VfTkFNRSA9ICdQQUNLQUdFX05BTUUnLFxuICBQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRSA9ICdQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRScsXG4gIENPTlNUUlVDVF9IVUJfQkFTRV9VUkwgPSAnQ09OU1RSVUNUX0hVQl9CQVNFX1VSTCcsXG59XG5cbmV4cG9ydCBjb25zdCBNRVRSSUNTX05BTUVTUEFDRSA9ICdDb25zdHJ1Y3RIdWIvUGFja2FnZUNhbmFyeSc7XG5cbmV4cG9ydCBjb25zdCBlbnVtIE1ldHJpY05hbWUge1xuICAvKipcbiAgICogVGhlIHRpbWUgZWxhcHNlZCBzaW5jZSBhIHBhY2thZ2Ugd2FzIHB1Ymxpc2hlZCB0byBucG1qcy5jb20sIHdoaWxlIGl0IGhhc1xuICAgKiBub3QgYmVlbiBkZXRlY3RlZCBpbiB0aGUgY2F0YWxvZyBieSB0aGUgcGFja2FnZSBjYW5hcnkuXG4gICAqXG4gICAqIFRoaXMgbWV0cmljcyBpcyBvbmx5IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBEV0VMTF9USU1FID0gJ0R3ZWxsVGltZScsXG5cbiAgLyoqXG4gICAqIFRoZSB0aW1lIGl0IHRvb2sgYmV0d2VlbiBhIHBhY2thZ2UncyBwdWJsaWNhdGlvbiB0byBucG1qcy5jb20gYW5kIHdoZW4gdGhlXG4gICAqIHBhY2thZ2UgY2FuYXJ5IGRldGVjdGVkIHRoZSBwYWNrYWdlIGFzIGF2YWlsYWJsZSBpbiB0aGUgY2F0YWxvZy5cbiAgICpcbiAgICogVGhpcyBtZXRyaWMgaXMgbm90IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGFjdHVhbGx5IGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBUSU1FX1RPX0NBVEFMT0cgPSAnVGltZVRvQ2F0YWxvZycsXG5cbiAgLyoqXG4gICAqIFRoZSBudW1iZXIgb2YgcGFja2FnZSB2ZXJzaW9ucyB0aGF0IHdlcmUgdHJhY2tlZCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZVxuICAgKiBwYWNrYWdlIGNhbmFyeSBleGVjdXRpb24gdGhhdCBwcm9kdWNlZCB0aGUgZGF0YSBwb2ludC5cbiAgICovXG4gIFRSQUNLRURfVkVSU0lPTl9DT1VOVCA9ICdUcmFja2VkVmVyc2lvbkNvdW50JyxcblxuICAvKipcbiAgICogVGhlIGVzdGltYXRlZCBsYWcgYmV0d2VlbiB0aGUgbnBtIHJlZ2lzdHJ5IHJlcGxpY2EgKHJlcGxpY2F0ZS5ucG1qcy5jb20pXG4gICAqIGFuZCB0aGUgcHJpbWFyeSByZWdpc3RyeSAocmVnaXN0cnkubnBtanMuY29tKS4gVGhpcyBjYW5ub3QgYmUgbWVhc3VyZWRcbiAgICogZGlyZWN0bHkgYmVjYXVzZSB0aGUgcHJpbWFyeSBkb2VzIG5vdCBleHBvc2UgdGhlIHJlbGV2YW50IENvdWNoREIgZW5kcG9pbnRzLFxuICAgKiBzbyB3ZSB1c2UgdGhlIHByb2JlIHBhY2thZ2UgdG8gZ2V0IGEgbG93LXJlc29sdXRpb24gdmlldyBvZiB0aGlzLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfTEFHID0gJ0VzdGltYXRlZE5wbVJlcGxpY2FMYWcnLFxuXG4gIC8qKlxuICAgKiBBIG1ldHJpYyB0cmFja2luZyB3aGV0aGVyIHRoZSBucG0gcmVnaXN0cnkgcmVwbGljYSAocmVwbGljYXRlLm5wbWpzLmNvbSlcbiAgICogaXMgZG93bi4gVGhlIHZhbHVlIGlzIDEgd2hlbiB0aGUgcmVwbGljYSBpcyBkZXRlY3RlZCB0byBiZSBkb3duLCBhbmQgMFxuICAgKiB3aGVuIHRoZSByZXBsaWNhIGlzIGRldGVjdGVkIHRvIGJlIHVwLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfRE9XTiA9ICdOcG1SZXBsaWNhSXNEb3duJyxcbn1cblxuZXhwb3J0IGNvbnN0IGVudW0gT2JqZWN0S2V5IHtcbiAgU1RBVEVfUFJFRklYID0gJ3BhY2thZ2UtY2FuYXJ5LycsXG4gIFNUQVRFX1NVRkZJWCA9ICcuc3RhdGUuanNvbicsXG59XG4iXX0=
@@ -23,6 +23,12 @@ export declare class NpmJsPackageCanary extends Construct {
23
23
  * that is strictly worse than 3 hours.
24
24
  */
25
25
  metricEstimatedNpmReplicaLag(opts?: MetricOptions): Metric;
26
+ /**
27
+ * A metric tracking whether the npm registry replica (replicate.npmjs.com)
28
+ * is down. The value is 1 when the replica is detected to be down, and 0
29
+ * when the replica is detected to be up.
30
+ */
31
+ metricNpmReplicaIsDown(opts?: MetricOptions): Metric;
26
32
  metricErrors(opts?: MetricOptions): Metric;
27
33
  metricInvocations(opts?: MetricOptions): Metric;
28
34
  metricThrottles(opts?: MetricOptions): Metric;
@@ -79,6 +79,20 @@ class NpmJsPackageCanary extends constructs_1.Construct {
79
79
  namespace: constants_1.METRICS_NAMESPACE,
80
80
  });
81
81
  }
82
+ /**
83
+ * A metric tracking whether the npm registry replica (replicate.npmjs.com)
84
+ * is down. The value is 1 when the replica is detected to be down, and 0
85
+ * when the replica is detected to be up.
86
+ */
87
+ metricNpmReplicaIsDown(opts) {
88
+ return new aws_cloudwatch_1.Metric({
89
+ period: aws_cdk_lib_1.Duration.minutes(1),
90
+ statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
91
+ ...opts,
92
+ metricName: "NpmReplicaIsDown" /* NPM_REPLICA_DOWN */,
93
+ namespace: constants_1.METRICS_NAMESPACE,
94
+ });
95
+ }
82
96
  metricErrors(opts) {
83
97
  return this.handler.metricErrors(opts);
84
98
  }
@@ -93,4 +107,4 @@ class NpmJsPackageCanary extends constructs_1.Construct {
93
107
  }
94
108
  }
95
109
  exports.NpmJsPackageCanary = NpmJsPackageCanary;
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"]}
110
+ //# 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  /**\n   * A metric tracking whether the npm registry replica (replicate.npmjs.com)\n   * is down. The value is 1 when the replica is detected to be down, and 0\n   * when the replica is detected to be up.\n   */\n  public metricNpmReplicaIsDown(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.NPM_REPLICA_DOWN,\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"]}
@@ -8130,10 +8130,14 @@ async function handler(event) {
8130
8130
  console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);
8131
8131
  updateLatestIfNeeded(state, latest);
8132
8132
  try {
8133
+ const replicaLag = await stateService.npmReplicaLagSeconds(packageName);
8133
8134
  await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8134
8135
  metrics.setDimensions();
8135
8136
  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);
8137
+ metrics.putMetric("NpmReplicaIsDown" /* NPM_REPLICA_DOWN */, await stateService.isNpmReplicaDown() ? 1 : 0, import_aws_embedded_metrics.Unit.None);
8138
+ if (replicaLag !== void 0) {
8139
+ metrics.putMetric("EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */, replicaLag, import_aws_embedded_metrics.Unit.Seconds);
8140
+ }
8137
8141
  })();
8138
8142
  for (const versionState of [
8139
8143
  state.latest,
@@ -8245,13 +8249,29 @@ var CanaryStateService = class {
8245
8249
  console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
8246
8250
  return { version, publishedAt: new Date(publishedAt) };
8247
8251
  }
8252
+ async isNpmReplicaDown() {
8253
+ try {
8254
+ await getJSON("https://replicate.npmjs.com/");
8255
+ return false;
8256
+ } catch (e) {
8257
+ return true;
8258
+ }
8259
+ }
8248
8260
  async npmReplicaLagSeconds(packageName) {
8249
8261
  const encodedPackageName = encodeURIComponent(packageName);
8250
8262
  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
- ]);
8263
+ const primaryDate = await getModifiedTimestamp(`registry.npmjs.org`);
8264
+ let replicaDate;
8265
+ try {
8266
+ replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`);
8267
+ } catch (e) {
8268
+ if (e instanceof Error && e.message.includes("HTTP 504")) {
8269
+ console.log(`Warning: error fetching replicate.npmjs.com: ${e.toString()}`);
8270
+ return void 0;
8271
+ } else {
8272
+ throw e;
8273
+ }
8274
+ }
8255
8275
  const deltaMs = primaryDate.getTime() - replicaDate.getTime();
8256
8276
  console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
8257
8277
  console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 36e5} hours behind)`);