construct-hub 0.4.9 → 0.4.12

Sign up to get free protection for your applications and to get access to all the features.
@@ -1 +1 @@
1
- {"version":3,"file":"code-artifact-forwarder.lambda.d.ts","sourceRoot":"","sources":["../../../src/package-sources/codeartifact/code-artifact-forwarder.lambda.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAkB5D,eAAO,MAAM,OAAO,+PA6InB,CAAC;AAEF;;GAEG;AACH,UAAU,kBAAkB;IAC1B;;OAEG;IACH,QAAQ,CAAC,UAAU,EAAE,MAAM,CAAC;IAE5B;;OAEG;IACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAE7B;;OAEG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;OAEG;IACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;IAE/B;;OAEG;IACH,QAAQ,CAAC,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAEnC;;OAEG;IACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAE7B;;OAEG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;OAEG;IACH,QAAQ,CAAC,mBAAmB,EACxB,YAAY,GACZ,WAAW,GACX,UAAU,GACV,UAAU,GACV,UAAU,CAAC;IAEf;;;;OAIG;IACH,QAAQ,CAAC,sBAAsB,EAAE,MAAM,CAAC;IAExC,QAAQ,CAAC,OAAO,EAAE;QAChB;;;WAGG;QACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;QAE7B;;WAEG;QACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;QAE/B;;WAEG;QACH,QAAQ,CAAC,eAAe,EAAE,OAAO,CAAC;QAElC;;;WAGG;QACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;QAE/B;;;WAGG;QACH,QAAQ,CAAC,aAAa,EAAE,OAAO,CAAC;KACjC,CAAC;IAEF;;OAEG;IACH,QAAQ,CAAC,aAAa,EAAE,SAAS,GAAG,SAAS,GAAG,SAAS,CAAC;IAE1D;;;OAGG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;;;OAIG;IACH,QAAQ,CAAC,oBAAoB,EAAE,MAAM,CAAC;CACvC"}
1
+ {"version":3,"file":"code-artifact-forwarder.lambda.d.ts","sourceRoot":"","sources":["../../../src/package-sources/codeartifact/code-artifact-forwarder.lambda.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAkB5D,eAAO,MAAM,OAAO,+PA4InB,CAAC;AAEF;;GAEG;AACH,UAAU,kBAAkB;IAC1B;;OAEG;IACH,QAAQ,CAAC,UAAU,EAAE,MAAM,CAAC;IAE5B;;OAEG;IACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAE7B;;OAEG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;OAEG;IACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;IAE/B;;OAEG;IACH,QAAQ,CAAC,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAEnC;;OAEG;IACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAE7B;;OAEG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;OAEG;IACH,QAAQ,CAAC,mBAAmB,EACxB,YAAY,GACZ,WAAW,GACX,UAAU,GACV,UAAU,GACV,UAAU,CAAC;IAEf;;;;OAIG;IACH,QAAQ,CAAC,sBAAsB,EAAE,MAAM,CAAC;IAExC,QAAQ,CAAC,OAAO,EAAE;QAChB;;;WAGG;QACH,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;QAE7B;;WAEG;QACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;QAE/B;;WAEG;QACH,QAAQ,CAAC,eAAe,EAAE,OAAO,CAAC;QAElC;;;WAGG;QACH,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;QAE/B;;;WAGG;QACH,QAAQ,CAAC,aAAa,EAAE,OAAO,CAAC;KACjC,CAAC;IAEF;;OAEG;IACH,QAAQ,CAAC,aAAa,EAAE,SAAS,GAAG,SAAS,GAAG,SAAS,CAAC;IAE1D;;;OAGG;IACH,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;IAEhC;;;;OAIG;IACH,QAAQ,CAAC,oBAAoB,EAAE,MAAM,CAAC;CACvC"}
@@ -105,9 +105,8 @@ exports.handler = aws_embedded_metrics_1.metricScope((metrics) => async (event,
105
105
  },
106
106
  },
107
107
  MessageBody: JSON.stringify(message),
108
- MessageDeduplicationId: event.detail.eventDeduplicationId,
109
108
  QueueUrl: queueUrl,
110
109
  })
111
110
  .promise();
112
111
  });
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-artifact-forwarder.lambda.js","sourceRoot":"","sources":["../../../src/package-sources/codeartifact/code-artifact-forwarder.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { metricScope, Unit } from 'aws-embedded-metrics';\nimport type { Context, EventBridgeEvent } from 'aws-lambda';\n\nimport { DenyListClient } from '../../backend/deny-list/client.lambda-shared';\nimport { LicenseListClient } from '../../backend/license-list/client.lambda-shared';\nimport * as aws from '../../backend/shared/aws.lambda-shared';\nimport { requireEnv } from '../../backend/shared/env.lambda-shared';\nimport { integrity } from '../../backend/shared/integrity.lambda-shared';\nimport { extractObjects } from '../../backend/shared/tarball.lambda-shared';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  DOMAIN_OWNER_DIMENSION,\n  DOMAIN_NAME_DIMENSION,\n  REPOSITORY_NAME_DIMENSION,\n} from './constants.lambda-shared';\n\nconst DETAIL_TYPE = 'CodeArtifact Package Version State Change' as const;\n\nexport const handler = metricScope(\n  (metrics) =>\n    async (\n      event: EventBridgeEvent<typeof DETAIL_TYPE, CodeArtifactDetail>,\n      context: Context\n    ) => {\n      console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n      const stagingBucket = requireEnv('BUCKET_NAME');\n      const queueUrl = requireEnv('QUEUE_URL');\n\n      metrics.setNamespace(METRICS_NAMESPACE);\n      metrics.setDimensions({\n        [DOMAIN_OWNER_DIMENSION]: event.detail.domainOwner,\n        [DOMAIN_NAME_DIMENSION]: event.detail.domainName,\n        [REPOSITORY_NAME_DIMENSION]: event.detail.repositoryName,\n      });\n\n      const packageName = event.detail.packageNamespace\n        ? `@${event.detail.packageNamespace}/${event.detail.packageName}`\n        : event.detail.packageName;\n\n      if (event.detail.operationType === 'Deleted') {\n        console.log(\n          `Operation type is \"Deleted\" for ${packageName}@${event.detail.packageVersion}`\n        );\n        metrics.putMetric(MetricName.DELETED_COUNT, 1, Unit.Count);\n        return;\n      }\n\n      const denyList = await DenyListClient.newClient();\n      const denied = denyList.lookup(packageName, event.detail.packageVersion);\n      if (denied) {\n        console.log(\n          `Package ${packageName}@${\n            event.detail.packageVersion\n          } denied: ${JSON.stringify(denied, null, 2)}`\n        );\n        metrics.putMetric(MetricName.DENY_LISTED_COUNT, 1, Unit.Count);\n        return;\n      }\n\n      const { asset, packageVersionRevision } = await aws\n        .codeArtifact()\n        .getPackageVersionAsset({\n          asset: 'package.tgz', // Always named this way for npm packages!\n          domainOwner: event.detail.domainOwner,\n          domain: event.detail.domainName,\n          repository: event.detail.repositoryName,\n          format: event.detail.packageFormat,\n          namespace: event.detail.packageNamespace,\n          package: event.detail.packageName,\n          packageVersion: event.detail.packageVersion,\n        })\n        .promise();\n      const tarball = Buffer.from(asset! as any);\n\n      const { assemblyJson, packageJson } = await extractObjects(tarball, {\n        assemblyJson: { path: 'package/.jsii' },\n        packageJson: { path: 'package/package.json', required: true },\n      });\n      metrics.putMetric(\n        MetricName.NOT_JSII_ENABLED_COUNT,\n        assemblyJson ? 0 : 1,\n        Unit.Count\n      );\n      if (assemblyJson == null) {\n        console.log(\n          `Package \"${packageName}@${event.detail.packageVersion}\" does not contain a .jsii assembly`\n        );\n        return;\n      }\n\n      const metadata = JSON.parse(packageJson.toString('utf-8'));\n      const licenseList = await LicenseListClient.newClient();\n      const eligibleLicense = licenseList.lookup(\n        metadata.license ?? 'UNLICENSED'\n      );\n      metrics.putMetric(\n        MetricName.INELIGIBLE_LICENSE,\n        eligibleLicense ? 0 : 1,\n        Unit.Count\n      );\n      if (!eligibleLicense) {\n        console.log(\n          `Package \"${packageName}@${\n            event.detail.packageVersion\n          }\" does not use allow-listed license: ${\n            metadata.license ?? 'UNLICENSED'\n          }`\n        );\n        return;\n      }\n\n      const stagingKey = `${packageName}/${event.detail.packageVersion}/${packageVersionRevision}/package.tgz`;\n      await aws\n        .s3()\n        .putObject({\n          Bucket: stagingBucket,\n          Key: stagingKey,\n          Body: asset,\n          ContentType: 'application/octet-stream',\n          Metadata: {\n            'Lambda-Log-Group': context.logGroupName,\n            'Lambda-Log-Stream': context.logStreamName,\n            'Lambda-Run-Id': context.awsRequestId,\n          },\n        })\n        .promise();\n\n      const message = integrity(\n        {\n          tarballUri: `s3://${stagingBucket}/${stagingKey}`,\n          metadata: { resources: event.resources.join(', ') },\n          time: event.time,\n        },\n        tarball\n      );\n      return aws\n        .sqs()\n        .sendMessage({\n          MessageAttributes: {\n            AWS_REQUEST_ID: {\n              DataType: 'String',\n              StringValue: context.awsRequestId,\n            },\n            LOG_GROUP_NAME: {\n              DataType: 'String',\n              StringValue: context.logGroupName,\n            },\n            LOG_STREAM_NAME: {\n              DataType: 'String',\n              StringValue: context.logStreamName,\n            },\n          },\n          MessageBody: JSON.stringify(message),\n          MessageDeduplicationId: event.detail.eventDeduplicationId,\n          QueueUrl: queueUrl,\n        })\n        .promise();\n    }\n);\n\n/**\n * @see https://docs.aws.amazon.com/codeartifact/latest/ug/service-event-format-example.html\n */\ninterface CodeArtifactDetail {\n  /**\n   * The domain that contains the repository that contains the package.\n   */\n  readonly domainName: string;\n\n  /**\n   * The AWS account ID of the owner of the domain.\n   */\n  readonly domainOwner: string;\n\n  /**\n   * The repository that contains the package.\n   */\n  readonly repositoryName: string;\n\n  /**\n   * The format of the package that triggered the event.\n   */\n  readonly packageFormat: string;\n\n  /**\n   * The namespace of the package that triggered the event.\n   */\n  readonly packageNamespace?: string;\n\n  /**\n   * The name of the package that triggered the event.\n   */\n  readonly packageName: string;\n\n  /**\n   * The version of the package that triggered the event.\n   */\n  readonly packageVersion: string;\n\n  /**\n   * The state of the package version when the event was triggered.\n   */\n  readonly packageVersionState:\n    | 'Unfinished'\n    | 'Published'\n    | 'Unlisted'\n    | 'Archived'\n    | 'Disposed';\n\n  /**\n   * A value that uniquely identifies the state of the assets and metadata of the package version when the event was\n   * triggered. If the package version is modified (for example, by adding another JAR file to a Maven package), the\n   * packageVersionRevision changes.\n   */\n  readonly packageVersionRevision: string;\n\n  readonly changes: {\n    /**\n     * The number of assets added to a package that triggered an event. Examples of an asset are a Maven JAR file or a\n     * Python wheel.\n     */\n    readonly assetsAdded: number;\n\n    /**\n     * The number of assets removed from a package that triggered an event.\n     */\n    readonly assetsRemoved: number;\n\n    /**\n     * The number of assets modified in the package that triggered the event.\n     */\n    readonly metadataUpdated: boolean;\n\n    /**\n     * A boolean value that is set to true if the event includes modified package-level metadata. For example, an event\n     * might modify a Maven pom.xml file.\n     */\n    readonly assetsUpdated: number;\n\n    /**\n     * A boolean value that is set to true if the event's packageVersionStatus is modified(for example, if\n     * packageVersionStatus changes from Unfinished to Published).\n     */\n    readonly statusChanged: boolean;\n  };\n\n  /**\n   * Describes the high-level type of the package version change.\n   */\n  readonly operationType: 'Created' | 'Updated' | 'Deleted';\n\n  /**\n   * An integer that specifies an event number for a package. Each event on a package increments the sequenceNumber so\n   * events can be arranged sequentially. An event can increment the sequenceNumber by any integer number.\n   */\n  readonly sequenceNumber: number;\n\n  /**\n   * An ID used to differentiate duplicate EventBridge events. In rare cases, EventBridge might trigger the same rule\n   * more than once for a single event or scheduled time. Or, it might invoke the same target more than once for a given\n   * triggered rule.\n   */\n  readonly eventDeduplicationId: string;\n}\n"]}
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-artifact-forwarder.lambda.js","sourceRoot":"","sources":["../../../src/package-sources/codeartifact/code-artifact-forwarder.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { metricScope, Unit } from 'aws-embedded-metrics';\nimport type { Context, EventBridgeEvent } from 'aws-lambda';\n\nimport { DenyListClient } from '../../backend/deny-list/client.lambda-shared';\nimport { LicenseListClient } from '../../backend/license-list/client.lambda-shared';\nimport * as aws from '../../backend/shared/aws.lambda-shared';\nimport { requireEnv } from '../../backend/shared/env.lambda-shared';\nimport { integrity } from '../../backend/shared/integrity.lambda-shared';\nimport { extractObjects } from '../../backend/shared/tarball.lambda-shared';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  DOMAIN_OWNER_DIMENSION,\n  DOMAIN_NAME_DIMENSION,\n  REPOSITORY_NAME_DIMENSION,\n} from './constants.lambda-shared';\n\nconst DETAIL_TYPE = 'CodeArtifact Package Version State Change' as const;\n\nexport const handler = metricScope(\n  (metrics) =>\n    async (\n      event: EventBridgeEvent<typeof DETAIL_TYPE, CodeArtifactDetail>,\n      context: Context\n    ) => {\n      console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n      const stagingBucket = requireEnv('BUCKET_NAME');\n      const queueUrl = requireEnv('QUEUE_URL');\n\n      metrics.setNamespace(METRICS_NAMESPACE);\n      metrics.setDimensions({\n        [DOMAIN_OWNER_DIMENSION]: event.detail.domainOwner,\n        [DOMAIN_NAME_DIMENSION]: event.detail.domainName,\n        [REPOSITORY_NAME_DIMENSION]: event.detail.repositoryName,\n      });\n\n      const packageName = event.detail.packageNamespace\n        ? `@${event.detail.packageNamespace}/${event.detail.packageName}`\n        : event.detail.packageName;\n\n      if (event.detail.operationType === 'Deleted') {\n        console.log(\n          `Operation type is \"Deleted\" for ${packageName}@${event.detail.packageVersion}`\n        );\n        metrics.putMetric(MetricName.DELETED_COUNT, 1, Unit.Count);\n        return;\n      }\n\n      const denyList = await DenyListClient.newClient();\n      const denied = denyList.lookup(packageName, event.detail.packageVersion);\n      if (denied) {\n        console.log(\n          `Package ${packageName}@${\n            event.detail.packageVersion\n          } denied: ${JSON.stringify(denied, null, 2)}`\n        );\n        metrics.putMetric(MetricName.DENY_LISTED_COUNT, 1, Unit.Count);\n        return;\n      }\n\n      const { asset, packageVersionRevision } = await aws\n        .codeArtifact()\n        .getPackageVersionAsset({\n          asset: 'package.tgz', // Always named this way for npm packages!\n          domainOwner: event.detail.domainOwner,\n          domain: event.detail.domainName,\n          repository: event.detail.repositoryName,\n          format: event.detail.packageFormat,\n          namespace: event.detail.packageNamespace,\n          package: event.detail.packageName,\n          packageVersion: event.detail.packageVersion,\n        })\n        .promise();\n      const tarball = Buffer.from(asset! as any);\n\n      const { assemblyJson, packageJson } = await extractObjects(tarball, {\n        assemblyJson: { path: 'package/.jsii' },\n        packageJson: { path: 'package/package.json', required: true },\n      });\n      metrics.putMetric(\n        MetricName.NOT_JSII_ENABLED_COUNT,\n        assemblyJson ? 0 : 1,\n        Unit.Count\n      );\n      if (assemblyJson == null) {\n        console.log(\n          `Package \"${packageName}@${event.detail.packageVersion}\" does not contain a .jsii assembly`\n        );\n        return;\n      }\n\n      const metadata = JSON.parse(packageJson.toString('utf-8'));\n      const licenseList = await LicenseListClient.newClient();\n      const eligibleLicense = licenseList.lookup(\n        metadata.license ?? 'UNLICENSED'\n      );\n      metrics.putMetric(\n        MetricName.INELIGIBLE_LICENSE,\n        eligibleLicense ? 0 : 1,\n        Unit.Count\n      );\n      if (!eligibleLicense) {\n        console.log(\n          `Package \"${packageName}@${\n            event.detail.packageVersion\n          }\" does not use allow-listed license: ${\n            metadata.license ?? 'UNLICENSED'\n          }`\n        );\n        return;\n      }\n\n      const stagingKey = `${packageName}/${event.detail.packageVersion}/${packageVersionRevision}/package.tgz`;\n      await aws\n        .s3()\n        .putObject({\n          Bucket: stagingBucket,\n          Key: stagingKey,\n          Body: asset,\n          ContentType: 'application/octet-stream',\n          Metadata: {\n            'Lambda-Log-Group': context.logGroupName,\n            'Lambda-Log-Stream': context.logStreamName,\n            'Lambda-Run-Id': context.awsRequestId,\n          },\n        })\n        .promise();\n\n      const message = integrity(\n        {\n          tarballUri: `s3://${stagingBucket}/${stagingKey}`,\n          metadata: { resources: event.resources.join(', ') },\n          time: event.time,\n        },\n        tarball\n      );\n      return aws\n        .sqs()\n        .sendMessage({\n          MessageAttributes: {\n            AWS_REQUEST_ID: {\n              DataType: 'String',\n              StringValue: context.awsRequestId,\n            },\n            LOG_GROUP_NAME: {\n              DataType: 'String',\n              StringValue: context.logGroupName,\n            },\n            LOG_STREAM_NAME: {\n              DataType: 'String',\n              StringValue: context.logStreamName,\n            },\n          },\n          MessageBody: JSON.stringify(message),\n          QueueUrl: queueUrl,\n        })\n        .promise();\n    }\n);\n\n/**\n * @see https://docs.aws.amazon.com/codeartifact/latest/ug/service-event-format-example.html\n */\ninterface CodeArtifactDetail {\n  /**\n   * The domain that contains the repository that contains the package.\n   */\n  readonly domainName: string;\n\n  /**\n   * The AWS account ID of the owner of the domain.\n   */\n  readonly domainOwner: string;\n\n  /**\n   * The repository that contains the package.\n   */\n  readonly repositoryName: string;\n\n  /**\n   * The format of the package that triggered the event.\n   */\n  readonly packageFormat: string;\n\n  /**\n   * The namespace of the package that triggered the event.\n   */\n  readonly packageNamespace?: string;\n\n  /**\n   * The name of the package that triggered the event.\n   */\n  readonly packageName: string;\n\n  /**\n   * The version of the package that triggered the event.\n   */\n  readonly packageVersion: string;\n\n  /**\n   * The state of the package version when the event was triggered.\n   */\n  readonly packageVersionState:\n    | 'Unfinished'\n    | 'Published'\n    | 'Unlisted'\n    | 'Archived'\n    | 'Disposed';\n\n  /**\n   * A value that uniquely identifies the state of the assets and metadata of the package version when the event was\n   * triggered. If the package version is modified (for example, by adding another JAR file to a Maven package), the\n   * packageVersionRevision changes.\n   */\n  readonly packageVersionRevision: string;\n\n  readonly changes: {\n    /**\n     * The number of assets added to a package that triggered an event. Examples of an asset are a Maven JAR file or a\n     * Python wheel.\n     */\n    readonly assetsAdded: number;\n\n    /**\n     * The number of assets removed from a package that triggered an event.\n     */\n    readonly assetsRemoved: number;\n\n    /**\n     * The number of assets modified in the package that triggered the event.\n     */\n    readonly metadataUpdated: boolean;\n\n    /**\n     * A boolean value that is set to true if the event includes modified package-level metadata. For example, an event\n     * might modify a Maven pom.xml file.\n     */\n    readonly assetsUpdated: number;\n\n    /**\n     * A boolean value that is set to true if the event's packageVersionStatus is modified(for example, if\n     * packageVersionStatus changes from Unfinished to Published).\n     */\n    readonly statusChanged: boolean;\n  };\n\n  /**\n   * Describes the high-level type of the package version change.\n   */\n  readonly operationType: 'Created' | 'Updated' | 'Deleted';\n\n  /**\n   * An integer that specifies an event number for a package. Each event on a package increments the sequenceNumber so\n   * events can be arranged sequentially. An event can increment the sequenceNumber by any integer number.\n   */\n  readonly sequenceNumber: number;\n\n  /**\n   * An ID used to differentiate duplicate EventBridge events. In rare cases, EventBridge might trigger the same rule\n   * more than once for a single event or scheduled time. Or, it might invoke the same target more than once for a given\n   * triggered rule.\n   */\n  readonly eventDeduplicationId: string;\n}\n"]}
@@ -22,12 +22,12 @@ class NpmJsPackageCanary extends constructs_1.Construct {
22
22
  ["PACKAGE_NAME" /* PACKAGE_NAME */]: props.packageName,
23
23
  },
24
24
  memorySize: 10024,
25
- timeout: aws_cdk_lib_1.Duration.minutes(1),
25
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
26
26
  });
27
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
- schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(1)),
30
+ schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(5)),
31
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.
@@ -35,7 +35,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
35
35
  }
36
36
  metricDwellTime(opts) {
37
37
  return new aws_cloudwatch_1.Metric({
38
- period: aws_cdk_lib_1.Duration.minutes(1),
38
+ period: aws_cdk_lib_1.Duration.minutes(5),
39
39
  statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
40
40
  ...opts,
41
41
  metricName: "DwellTime" /* DWELL_TIME */,
@@ -44,7 +44,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
44
44
  }
45
45
  metricTimeToCatalog(opts) {
46
46
  return new aws_cloudwatch_1.Metric({
47
- period: aws_cdk_lib_1.Duration.minutes(1),
47
+ period: aws_cdk_lib_1.Duration.minutes(5),
48
48
  statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
49
49
  ...opts,
50
50
  metricName: "TimeToCatalog" /* TIME_TO_CATALOG */,
@@ -53,7 +53,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
53
53
  }
54
54
  metricTrackedVersionCount(opts) {
55
55
  return new aws_cloudwatch_1.Metric({
56
- period: aws_cdk_lib_1.Duration.minutes(1),
56
+ period: aws_cdk_lib_1.Duration.minutes(5),
57
57
  statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
58
58
  ...opts,
59
59
  metricName: "TrackedVersionCount" /* TRACKED_VERSION_COUNT */,
@@ -72,7 +72,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
72
72
  */
73
73
  metricEstimatedNpmReplicaLag(opts) {
74
74
  return new aws_cloudwatch_1.Metric({
75
- period: aws_cdk_lib_1.Duration.minutes(1),
75
+ period: aws_cdk_lib_1.Duration.minutes(5),
76
76
  statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
77
77
  ...opts,
78
78
  metricName: "EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */,
@@ -86,7 +86,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
86
86
  */
87
87
  metricNpmReplicaIsDown(opts) {
88
88
  return new aws_cloudwatch_1.Metric({
89
- period: aws_cdk_lib_1.Duration.minutes(1),
89
+ period: aws_cdk_lib_1.Duration.minutes(5),
90
90
  statistic: aws_cloudwatch_1.Statistic.MAXIMUM,
91
91
  ...opts,
92
92
  metricName: "NpmReplicaIsDown" /* NPM_REPLICA_DOWN */,
@@ -100,7 +100,7 @@ class NpmJsPackageCanary extends constructs_1.Construct {
100
100
  */
101
101
  metricHttpGatewayErrors(opts) {
102
102
  return new aws_cloudwatch_1.Metric({
103
- period: aws_cdk_lib_1.Duration.minutes(1),
103
+ period: aws_cdk_lib_1.Duration.minutes(5),
104
104
  statistic: aws_cloudwatch_1.Statistic.SUM,
105
105
  ...opts,
106
106
  metricName: "HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */,
@@ -121,4 +121,4 @@ class NpmJsPackageCanary extends constructs_1.Construct {
121
121
  }
122
122
  }
123
123
  exports.NpmJsPackageCanary = NpmJsPackageCanary;
124
- //# 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  /**\n   * A metric tracking HTTP Gateway errors experienced while the canary is\n   * running. Those are typically caused by the npm registry servers being\n   * overloaded or otherwise impaired, and would cause false alarms.\n   */\n  public metricHttpGatewayErrors(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.HTTP_GATEWAY_ERRORS,\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"]}
124
+ //# 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(5),\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(5)),\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(5),\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(5),\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(5),\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(5),\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(5),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.NPM_REPLICA_DOWN,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * A metric tracking HTTP Gateway errors experienced while the canary is\n   * running. Those are typically caused by the npm registry servers being\n   * overloaded or otherwise impaired, and would cause false alarms.\n   */\n  public metricHttpGatewayErrors(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.HTTP_GATEWAY_ERRORS,\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"]}
@@ -8077,6 +8077,7 @@ var npmjs_package_canary_lambda_exports = {};
8077
8077
  __export(npmjs_package_canary_lambda_exports, {
8078
8078
  CanaryStateService: () => CanaryStateService,
8079
8079
  HTTPError: () => HTTPError,
8080
+ TimeoutError: () => TimeoutError,
8080
8081
  handler: () => handler
8081
8082
  });
8082
8083
  module.exports = __toCommonJS(npmjs_package_canary_lambda_exports);
@@ -8114,6 +8115,7 @@ var METRICS_NAMESPACE = "ConstructHub/PackageCanary";
8114
8115
 
8115
8116
  // src/package-sources/npmjs/canary/npmjs-package-canary.lambda.ts
8116
8117
  import_aws_embedded_metrics.Configuration.namespace = METRICS_NAMESPACE;
8118
+ var REPLICA_REQUEST_TIMEOUT_MS = 3e4;
8117
8119
  async function handler(event) {
8118
8120
  console.log(`Event: ${JSON.stringify(event, null, 2)}`);
8119
8121
  const packageName = requireEnv("PACKAGE_NAME" /* PACKAGE_NAME */);
@@ -8185,6 +8187,14 @@ async function handler(event) {
8185
8187
  metrics.setProperty("ErrorMessage", error.message);
8186
8188
  metrics.putMetric("HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */, 1, import_aws_embedded_metrics.Unit.Count);
8187
8189
  })();
8190
+ } else if (error instanceof TimeoutError) {
8191
+ console.error(`Request timeout from a dependency, assuming this is transient:`, error);
8192
+ await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8193
+ metrics.setDimensions();
8194
+ metrics.setProperty("ErrorCode", "REQUEST_TIMEOUT");
8195
+ metrics.setProperty("ErrorMessage", error.message);
8196
+ metrics.putMetric("HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */, 1, import_aws_embedded_metrics.Unit.Count);
8197
+ })();
8188
8198
  } else {
8189
8199
  throw error;
8190
8200
  }
@@ -8260,14 +8270,16 @@ var CanaryStateService = class {
8260
8270
  }
8261
8271
  async latest(packageName) {
8262
8272
  console.log(`Fetching latest version information from NPM: ${packageName}`);
8263
- const version = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`, ["version"]);
8264
- const publishedAt = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}`, ["time", version]);
8273
+ const version = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`, { jsonPath: ["version"] });
8274
+ const publishedAt = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}`, { jsonPath: ["time", version] });
8265
8275
  console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
8266
8276
  return { version, publishedAt: new Date(publishedAt) };
8267
8277
  }
8268
8278
  async isNpmReplicaDown() {
8269
8279
  try {
8270
- await getJSON("https://replicate.npmjs.com/");
8280
+ await getJSON("https://replicate.npmjs.com/", {
8281
+ timeoutMillis: REPLICA_REQUEST_TIMEOUT_MS
8282
+ });
8271
8283
  return false;
8272
8284
  } catch (e) {
8273
8285
  return true;
@@ -8279,7 +8291,7 @@ var CanaryStateService = class {
8279
8291
  const primaryDate = await getModifiedTimestamp(`registry.npmjs.org`);
8280
8292
  let replicaDate;
8281
8293
  try {
8282
- replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`);
8294
+ replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`, REPLICA_REQUEST_TIMEOUT_MS);
8283
8295
  } catch (e) {
8284
8296
  if (e instanceof Error && e.message.includes("HTTP 504")) {
8285
8297
  console.log(`Warning: error fetching replicate.npmjs.com: ${e.toString()}`);
@@ -8292,8 +8304,8 @@ var CanaryStateService = class {
8292
8304
  console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
8293
8305
  console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 36e5} hours behind)`);
8294
8306
  return deltaMs / 1e3;
8295
- async function getModifiedTimestamp(baseUrl) {
8296
- const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, ["time", "modified"]);
8307
+ async function getModifiedTimestamp(baseUrl, timeoutMillis) {
8308
+ const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, { jsonPath: ["time", "modified"], timeoutMillis });
8297
8309
  return new Date(isoDate);
8298
8310
  }
8299
8311
  }
@@ -8311,10 +8323,23 @@ var HTTPError = class extends Error {
8311
8323
  Error.captureStackTrace(this, HTTPError);
8312
8324
  }
8313
8325
  };
8314
- function getJSON(url, jsonPath) {
8326
+ var TimeoutError = class extends Error {
8327
+ constructor(message) {
8328
+ super(message);
8329
+ Error.captureStackTrace(this, TimeoutError);
8330
+ }
8331
+ };
8332
+ function getJSON(url, {
8333
+ jsonPath,
8334
+ timeoutMillis
8335
+ } = {}) {
8315
8336
  return new Promise((ok, ko) => {
8316
8337
  https.get(url, {
8317
- headers: { Accept: "application/json", "Accept-Encoding": "identity" }
8338
+ headers: {
8339
+ Accept: "application/json",
8340
+ "Accept-Encoding": "identity"
8341
+ },
8342
+ timeout: timeoutMillis
8318
8343
  }, (res) => {
8319
8344
  if (res.statusCode !== 200) {
8320
8345
  const error = new HTTPError(res.statusCode, `GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`);
@@ -8322,11 +8347,16 @@ function getJSON(url, jsonPath) {
8322
8347
  return ko(error);
8323
8348
  }
8324
8349
  res.once("error", ko);
8350
+ res.once("timeout", () => {
8351
+ ko(new TimeoutError(`Request timed out (after ${timeoutMillis ?? "N/A"} ms): GET ${url}`));
8352
+ });
8325
8353
  const json = JSONStream.parse(jsonPath);
8326
8354
  json.once("data", ok);
8327
8355
  json.once("error", ko);
8328
8356
  const plainPayload = res.headers["content-encoding"] === "gzip" ? gunzip(res) : res;
8329
8357
  plainPayload.pipe(json, { end: true });
8358
+ }).once("timeout", () => {
8359
+ ko(new TimeoutError(`Request timed out (after ${timeoutMillis ?? "N/A"} ms): GET ${url}`));
8330
8360
  });
8331
8361
  });
8332
8362
  }
@@ -8350,6 +8380,7 @@ function gunzip(readable) {
8350
8380
  0 && (module.exports = {
8351
8381
  CanaryStateService,
8352
8382
  HTTPError,
8383
+ TimeoutError,
8353
8384
  handler
8354
8385
  });
8355
8386
  //# sourceMappingURL=index.js.map