construct-hub 0.4.8 → 0.4.11

Sign up to get free protection for your applications and to get access to all the features.
Files changed (29) hide show
  1. package/.jsii +2 -2
  2. package/changelog.md +1 -1
  3. package/lib/construct-hub.js +1 -1
  4. package/lib/package-sources/code-artifact.js +1 -1
  5. package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.bundle/index.js +0 -1
  6. package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.bundle/index.js.map +2 -2
  7. package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.d.ts.map +1 -1
  8. package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.js +1 -2
  9. package/lib/package-sources/npmjs/canary/constants.d.ts +7 -1
  10. package/lib/package-sources/npmjs/canary/constants.d.ts.map +1 -1
  11. package/lib/package-sources/npmjs/canary/constants.js +1 -1
  12. package/lib/package-sources/npmjs/canary/index.d.ts +6 -0
  13. package/lib/package-sources/npmjs/canary/index.d.ts.map +1 -1
  14. package/lib/package-sources/npmjs/canary/index.js +15 -1
  15. package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js +106 -54
  16. package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js.map +2 -2
  17. package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.d.ts +7 -0
  18. package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.d.ts.map +1 -1
  19. package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.js +142 -77
  20. package/lib/package-sources/npmjs.d.ts.map +1 -1
  21. package/lib/package-sources/npmjs.js +21 -2
  22. package/lib/package-tag/index.js +3 -3
  23. package/lib/package-tag-group/index.js +2 -2
  24. package/lib/preload-file/index.js +1 -1
  25. package/lib/s3/storage.js +1 -1
  26. package/lib/spdx-license.js +1 -1
  27. package/package.json +1 -1
  28. package/releasetag.txt +1 -1
  29. package/version.txt +1 -1
@@ -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":";;;AAAA,+DAAyD;AAGzD,uFAA8E;AAC9E,0FAAoF;AACpF,8DAA8D;AAC9D,8EAAoE;AACpE,0FAAyE;AACzE,sFAA4E;AAC5E,uEAMmC;AAEnC,MAAM,WAAW,GAAG,2CAAoD,CAAC;AAE5D,QAAA,OAAO,GAAG,kCAAW,CAChC,CAAC,OAAO,EAAE,EAAE,CACV,KAAK,EACH,KAA+D,EAC/D,OAAgB,EAChB,EAAE;;IACF,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;IAExD,MAAM,aAAa,GAAG,8BAAU,CAAC,aAAa,CAAC,CAAC;IAChD,MAAM,QAAQ,GAAG,8BAAU,CAAC,WAAW,CAAC,CAAC;IAEzC,OAAO,CAAC,YAAY,CAAC,2CAAiB,CAAC,CAAC;IACxC,OAAO,CAAC,aAAa,CAAC;QACpB,CAAC,gDAAsB,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,WAAW;QAClD,CAAC,+CAAqB,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,UAAU;QAChD,CAAC,mDAAyB,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,cAAc;KACzD,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,gBAAgB;QAC/C,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,gBAAgB,IAAI,KAAK,CAAC,MAAM,CAAC,WAAW,EAAE;QACjE,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC;IAE7B,IAAI,KAAK,CAAC,MAAM,CAAC,aAAa,KAAK,SAAS,EAAE;QAC5C,OAAO,CAAC,GAAG,CACT,mCAAmC,WAAW,IAAI,KAAK,CAAC,MAAM,CAAC,cAAc,EAAE,CAChF,CAAC;QACF,OAAO,CAAC,SAAS,qCAA2B,CAAC,EAAE,2BAAI,CAAC,KAAK,CAAC,CAAC;QAC3D,OAAO;KACR;IAED,MAAM,QAAQ,GAAG,MAAM,qCAAc,CAAC,SAAS,EAAE,CAAC;IAClD,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC,WAAW,EAAE,KAAK,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;IACzE,IAAI,MAAM,EAAE;QACV,OAAO,CAAC,GAAG,CACT,WAAW,WAAW,IACpB,KAAK,CAAC,MAAM,CAAC,cACf,YAAY,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAC9C,CAAC;QACF,OAAO,CAAC,SAAS,4CAA+B,CAAC,EAAE,2BAAI,CAAC,KAAK,CAAC,CAAC;QAC/D,OAAO;KACR;IAED,MAAM,EAAE,KAAK,EAAE,sBAAsB,EAAE,GAAG,MAAM,GAAG;SAChD,YAAY,EAAE;SACd,sBAAsB,CAAC;QACtB,KAAK,EAAE,aAAa;QACpB,WAAW,EAAE,KAAK,CAAC,MAAM,CAAC,WAAW;QACrC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,UAAU;QAC/B,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,cAAc;QACvC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,aAAa;QAClC,SAAS,EAAE,KAAK,CAAC,MAAM,CAAC,gBAAgB;QACxC,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC,WAAW;QACjC,cAAc,EAAE,KAAK,CAAC,MAAM,CAAC,cAAc;KAC5C,CAAC;SACD,OAAO,EAAE,CAAC;IACb,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,KAAa,CAAC,CAAC;IAE3C,MAAM,EAAE,YAAY,EAAE,WAAW,EAAE,GAAG,MAAM,sCAAc,CAAC,OAAO,EAAE;QAClE,YAAY,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE;QACvC,WAAW,EAAE,EAAE,IAAI,EAAE,sBAAsB,EAAE,QAAQ,EAAE,IAAI,EAAE;KAC9D,CAAC,CAAC;IACH,OAAO,CAAC,SAAS,qDAEf,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EACpB,2BAAI,CAAC,KAAK,CACX,CAAC;IACF,IAAI,YAAY,IAAI,IAAI,EAAE;QACxB,OAAO,CAAC,GAAG,CACT,YAAY,WAAW,IAAI,KAAK,CAAC,MAAM,CAAC,cAAc,qCAAqC,CAC5F,CAAC;QACF,OAAO;KACR;IAED,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;IAC3D,MAAM,WAAW,GAAG,MAAM,wCAAiB,CAAC,SAAS,EAAE,CAAC;IACxD,MAAM,eAAe,GAAG,WAAW,CAAC,MAAM,OACxC,QAAQ,CAAC,OAAO,mCAAI,YAAY,CACjC,CAAC;IACF,OAAO,CAAC,SAAS,oDAEf,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EACvB,2BAAI,CAAC,KAAK,CACX,CAAC;IACF,IAAI,CAAC,eAAe,EAAE;QACpB,OAAO,CAAC,GAAG,CACT,YAAY,WAAW,IACrB,KAAK,CAAC,MAAM,CAAC,cACf,wCACE,MAAA,QAAQ,CAAC,OAAO,mCAAI,YACtB,EAAE,CACH,CAAC;QACF,OAAO;KACR;IAED,MAAM,UAAU,GAAG,GAAG,WAAW,IAAI,KAAK,CAAC,MAAM,CAAC,cAAc,IAAI,sBAAsB,cAAc,CAAC;IACzG,MAAM,GAAG;SACN,EAAE,EAAE;SACJ,SAAS,CAAC;QACT,MAAM,EAAE,aAAa;QACrB,GAAG,EAAE,UAAU;QACf,IAAI,EAAE,KAAK;QACX,WAAW,EAAE,0BAA0B;QACvC,QAAQ,EAAE;YACR,kBAAkB,EAAE,OAAO,CAAC,YAAY;YACxC,mBAAmB,EAAE,OAAO,CAAC,aAAa;YAC1C,eAAe,EAAE,OAAO,CAAC,YAAY;SACtC;KACF,CAAC;SACD,OAAO,EAAE,CAAC;IAEb,MAAM,OAAO,GAAG,mCAAS,CACvB;QACE,UAAU,EAAE,QAAQ,aAAa,IAAI,UAAU,EAAE;QACjD,QAAQ,EAAE,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;QACnD,IAAI,EAAE,KAAK,CAAC,IAAI;KACjB,EACD,OAAO,CACR,CAAC;IACF,OAAO,GAAG;SACP,GAAG,EAAE;SACL,WAAW,CAAC;QACX,iBAAiB,EAAE;YACjB,cAAc,EAAE;gBACd,QAAQ,EAAE,QAAQ;gBAClB,WAAW,EAAE,OAAO,CAAC,YAAY;aAClC;YACD,cAAc,EAAE;gBACd,QAAQ,EAAE,QAAQ;gBAClB,WAAW,EAAE,OAAO,CAAC,YAAY;aAClC;YACD,eAAe,EAAE;gBACf,QAAQ,EAAE,QAAQ;gBAClB,WAAW,EAAE,OAAO,CAAC,aAAa;aACnC;SACF;QACD,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;QACpC,sBAAsB,EAAE,KAAK,CAAC,MAAM,CAAC,oBAAoB;QACzD,QAAQ,EAAE,QAAQ;KACnB,CAAC;SACD,OAAO,EAAE,CAAC;AACf,CAAC,CACJ,CAAC","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"]}
@@ -36,7 +36,13 @@ export declare const enum MetricName {
36
36
  * is down. The value is 1 when the replica is detected to be down, and 0
37
37
  * when the replica is detected to be up.
38
38
  */
39
- NPM_REPLICA_DOWN = "NpmReplicaIsDown"
39
+ NPM_REPLICA_DOWN = "NpmReplicaIsDown",
40
+ /**
41
+ * A metric tracking HTTP 502 and HTTP 504 errors received while processing.
42
+ * Those are often encountered when the npm servers are overloaded, or
43
+ * otherwise impaired, and could cause alarms we cannot do anything about.
44
+ */
45
+ HTTP_GATEWAY_ERRORS = "HttpGatewayErrors"
40
46
  }
41
47
  export declare const enum ObjectKey {
42
48
  STATE_PREFIX = "package-canary/",
@@ -1 +1 @@
1
- {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/constants.ts"],"names":[],"mappings":"AAAA,0BAAkB,WAAW;IAC3B,YAAY,iBAAiB;IAC7B,0BAA0B,+BAA+B;IACzD,sBAAsB,2BAA2B;CAClD;AAED,eAAO,MAAM,iBAAiB,+BAA+B,CAAC;AAE9D,0BAAkB,UAAU;IAC1B;;;;;OAKG;IACH,UAAU,cAAc;IAExB;;;;;OAKG;IACH,eAAe,kBAAkB;IAEjC;;;OAGG;IACH,qBAAqB,wBAAwB;IAE7C;;;;;OAKG;IACH,eAAe,2BAA2B;IAE1C;;;;OAIG;IACH,gBAAgB,qBAAqB;CACtC;AAED,0BAAkB,SAAS;IACzB,YAAY,oBAAoB;IAChC,YAAY,gBAAgB;CAC7B"}
1
+ {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/constants.ts"],"names":[],"mappings":"AAAA,0BAAkB,WAAW;IAC3B,YAAY,iBAAiB;IAC7B,0BAA0B,+BAA+B;IACzD,sBAAsB,2BAA2B;CAClD;AAED,eAAO,MAAM,iBAAiB,+BAA+B,CAAC;AAE9D,0BAAkB,UAAU;IAC1B;;;;;OAKG;IACH,UAAU,cAAc;IAExB;;;;;OAKG;IACH,eAAe,kBAAkB;IAEjC;;;OAGG;IACH,qBAAqB,wBAAwB;IAE7C;;;;;OAKG;IACH,eAAe,2BAA2B;IAE1C;;;;OAIG;IACH,gBAAgB,qBAAqB;IAErC;;;;OAIG;IACH,mBAAmB,sBAAsB;CAC1C;AAED,0BAAkB,SAAS;IACzB,YAAY,oBAAoB;IAChC,YAAY,gBAAgB;CAC7B"}
@@ -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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3BhY2thZ2Utc291cmNlcy9ucG1qcy9jYW5hcnkvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQU1hLFFBQUEsaUJBQWlCLEdBQUcsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgZW51bSBFbnZpcm9ubWVudCB7XG4gIFBBQ0tBR0VfTkFNRSA9ICdQQUNLQUdFX05BTUUnLFxuICBQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRSA9ICdQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRScsXG4gIENPTlNUUlVDVF9IVUJfQkFTRV9VUkwgPSAnQ09OU1RSVUNUX0hVQl9CQVNFX1VSTCcsXG59XG5cbmV4cG9ydCBjb25zdCBNRVRSSUNTX05BTUVTUEFDRSA9ICdDb25zdHJ1Y3RIdWIvUGFja2FnZUNhbmFyeSc7XG5cbmV4cG9ydCBjb25zdCBlbnVtIE1ldHJpY05hbWUge1xuICAvKipcbiAgICogVGhlIHRpbWUgZWxhcHNlZCBzaW5jZSBhIHBhY2thZ2Ugd2FzIHB1Ymxpc2hlZCB0byBucG1qcy5jb20sIHdoaWxlIGl0IGhhc1xuICAgKiBub3QgYmVlbiBkZXRlY3RlZCBpbiB0aGUgY2F0YWxvZyBieSB0aGUgcGFja2FnZSBjYW5hcnkuXG4gICAqXG4gICAqIFRoaXMgbWV0cmljcyBpcyBvbmx5IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBEV0VMTF9USU1FID0gJ0R3ZWxsVGltZScsXG5cbiAgLyoqXG4gICAqIFRoZSB0aW1lIGl0IHRvb2sgYmV0d2VlbiBhIHBhY2thZ2UncyBwdWJsaWNhdGlvbiB0byBucG1qcy5jb20gYW5kIHdoZW4gdGhlXG4gICAqIHBhY2thZ2UgY2FuYXJ5IGRldGVjdGVkIHRoZSBwYWNrYWdlIGFzIGF2YWlsYWJsZSBpbiB0aGUgY2F0YWxvZy5cbiAgICpcbiAgICogVGhpcyBtZXRyaWMgaXMgbm90IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGFjdHVhbGx5IGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBUSU1FX1RPX0NBVEFMT0cgPSAnVGltZVRvQ2F0YWxvZycsXG5cbiAgLyoqXG4gICAqIFRoZSBudW1iZXIgb2YgcGFja2FnZSB2ZXJzaW9ucyB0aGF0IHdlcmUgdHJhY2tlZCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZVxuICAgKiBwYWNrYWdlIGNhbmFyeSBleGVjdXRpb24gdGhhdCBwcm9kdWNlZCB0aGUgZGF0YSBwb2ludC5cbiAgICovXG4gIFRSQUNLRURfVkVSU0lPTl9DT1VOVCA9ICdUcmFja2VkVmVyc2lvbkNvdW50JyxcblxuICAvKipcbiAgICogVGhlIGVzdGltYXRlZCBsYWcgYmV0d2VlbiB0aGUgbnBtIHJlZ2lzdHJ5IHJlcGxpY2EgKHJlcGxpY2F0ZS5ucG1qcy5jb20pXG4gICAqIGFuZCB0aGUgcHJpbWFyeSByZWdpc3RyeSAocmVnaXN0cnkubnBtanMuY29tKS4gVGhpcyBjYW5ub3QgYmUgbWVhc3VyZWRcbiAgICogZGlyZWN0bHkgYmVjYXVzZSB0aGUgcHJpbWFyeSBkb2VzIG5vdCBleHBvc2UgdGhlIHJlbGV2YW50IENvdWNoREIgZW5kcG9pbnRzLFxuICAgKiBzbyB3ZSB1c2UgdGhlIHByb2JlIHBhY2thZ2UgdG8gZ2V0IGEgbG93LXJlc29sdXRpb24gdmlldyBvZiB0aGlzLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfTEFHID0gJ0VzdGltYXRlZE5wbVJlcGxpY2FMYWcnLFxuXG4gIC8qKlxuICAgKiBBIG1ldHJpYyB0cmFja2luZyB3aGV0aGVyIHRoZSBucG0gcmVnaXN0cnkgcmVwbGljYSAocmVwbGljYXRlLm5wbWpzLmNvbSlcbiAgICogaXMgZG93bi4gVGhlIHZhbHVlIGlzIDEgd2hlbiB0aGUgcmVwbGljYSBpcyBkZXRlY3RlZCB0byBiZSBkb3duLCBhbmQgMFxuICAgKiB3aGVuIHRoZSByZXBsaWNhIGlzIGRldGVjdGVkIHRvIGJlIHVwLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfRE9XTiA9ICdOcG1SZXBsaWNhSXNEb3duJyxcbn1cblxuZXhwb3J0IGNvbnN0IGVudW0gT2JqZWN0S2V5IHtcbiAgU1RBVEVfUFJFRklYID0gJ3BhY2thZ2UtY2FuYXJ5LycsXG4gIFNUQVRFX1NVRkZJWCA9ICcuc3RhdGUuanNvbicsXG59XG4iXX0=
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3BhY2thZ2Utc291cmNlcy9ucG1qcy9jYW5hcnkvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQU1hLFFBQUEsaUJBQWlCLEdBQUcsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgZW51bSBFbnZpcm9ubWVudCB7XG4gIFBBQ0tBR0VfTkFNRSA9ICdQQUNLQUdFX05BTUUnLFxuICBQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRSA9ICdQQUNLQUdFX0NBTkFSWV9CVUNLRVRfTkFNRScsXG4gIENPTlNUUlVDVF9IVUJfQkFTRV9VUkwgPSAnQ09OU1RSVUNUX0hVQl9CQVNFX1VSTCcsXG59XG5cbmV4cG9ydCBjb25zdCBNRVRSSUNTX05BTUVTUEFDRSA9ICdDb25zdHJ1Y3RIdWIvUGFja2FnZUNhbmFyeSc7XG5cbmV4cG9ydCBjb25zdCBlbnVtIE1ldHJpY05hbWUge1xuICAvKipcbiAgICogVGhlIHRpbWUgZWxhcHNlZCBzaW5jZSBhIHBhY2thZ2Ugd2FzIHB1Ymxpc2hlZCB0byBucG1qcy5jb20sIHdoaWxlIGl0IGhhc1xuICAgKiBub3QgYmVlbiBkZXRlY3RlZCBpbiB0aGUgY2F0YWxvZyBieSB0aGUgcGFja2FnZSBjYW5hcnkuXG4gICAqXG4gICAqIFRoaXMgbWV0cmljcyBpcyBvbmx5IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBEV0VMTF9USU1FID0gJ0R3ZWxsVGltZScsXG5cbiAgLyoqXG4gICAqIFRoZSB0aW1lIGl0IHRvb2sgYmV0d2VlbiBhIHBhY2thZ2UncyBwdWJsaWNhdGlvbiB0byBucG1qcy5jb20gYW5kIHdoZW4gdGhlXG4gICAqIHBhY2thZ2UgY2FuYXJ5IGRldGVjdGVkIHRoZSBwYWNrYWdlIGFzIGF2YWlsYWJsZSBpbiB0aGUgY2F0YWxvZy5cbiAgICpcbiAgICogVGhpcyBtZXRyaWMgaXMgbm90IGVtaXR0ZWQgdW50aWwgdGhlIHBhY2thZ2UgaGFzIGFjdHVhbGx5IGJlZW4gZGV0ZWN0ZWQuXG4gICAqL1xuICBUSU1FX1RPX0NBVEFMT0cgPSAnVGltZVRvQ2F0YWxvZycsXG5cbiAgLyoqXG4gICAqIFRoZSBudW1iZXIgb2YgcGFja2FnZSB2ZXJzaW9ucyB0aGF0IHdlcmUgdHJhY2tlZCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZVxuICAgKiBwYWNrYWdlIGNhbmFyeSBleGVjdXRpb24gdGhhdCBwcm9kdWNlZCB0aGUgZGF0YSBwb2ludC5cbiAgICovXG4gIFRSQUNLRURfVkVSU0lPTl9DT1VOVCA9ICdUcmFja2VkVmVyc2lvbkNvdW50JyxcblxuICAvKipcbiAgICogVGhlIGVzdGltYXRlZCBsYWcgYmV0d2VlbiB0aGUgbnBtIHJlZ2lzdHJ5IHJlcGxpY2EgKHJlcGxpY2F0ZS5ucG1qcy5jb20pXG4gICAqIGFuZCB0aGUgcHJpbWFyeSByZWdpc3RyeSAocmVnaXN0cnkubnBtanMuY29tKS4gVGhpcyBjYW5ub3QgYmUgbWVhc3VyZWRcbiAgICogZGlyZWN0bHkgYmVjYXVzZSB0aGUgcHJpbWFyeSBkb2VzIG5vdCBleHBvc2UgdGhlIHJlbGV2YW50IENvdWNoREIgZW5kcG9pbnRzLFxuICAgKiBzbyB3ZSB1c2UgdGhlIHByb2JlIHBhY2thZ2UgdG8gZ2V0IGEgbG93LXJlc29sdXRpb24gdmlldyBvZiB0aGlzLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfTEFHID0gJ0VzdGltYXRlZE5wbVJlcGxpY2FMYWcnLFxuXG4gIC8qKlxuICAgKiBBIG1ldHJpYyB0cmFja2luZyB3aGV0aGVyIHRoZSBucG0gcmVnaXN0cnkgcmVwbGljYSAocmVwbGljYXRlLm5wbWpzLmNvbSlcbiAgICogaXMgZG93bi4gVGhlIHZhbHVlIGlzIDEgd2hlbiB0aGUgcmVwbGljYSBpcyBkZXRlY3RlZCB0byBiZSBkb3duLCBhbmQgMFxuICAgKiB3aGVuIHRoZSByZXBsaWNhIGlzIGRldGVjdGVkIHRvIGJlIHVwLlxuICAgKi9cbiAgTlBNX1JFUExJQ0FfRE9XTiA9ICdOcG1SZXBsaWNhSXNEb3duJyxcblxuICAvKipcbiAgICogQSBtZXRyaWMgdHJhY2tpbmcgSFRUUCA1MDIgYW5kIEhUVFAgNTA0IGVycm9ycyByZWNlaXZlZCB3aGlsZSBwcm9jZXNzaW5nLlxuICAgKiBUaG9zZSBhcmUgb2Z0ZW4gZW5jb3VudGVyZWQgd2hlbiB0aGUgbnBtIHNlcnZlcnMgYXJlIG92ZXJsb2FkZWQsIG9yXG4gICAqIG90aGVyd2lzZSBpbXBhaXJlZCwgYW5kIGNvdWxkIGNhdXNlIGFsYXJtcyB3ZSBjYW5ub3QgZG8gYW55dGhpbmcgYWJvdXQuXG4gICAqL1xuICBIVFRQX0dBVEVXQVlfRVJST1JTID0gJ0h0dHBHYXRld2F5RXJyb3JzJyxcbn1cblxuZXhwb3J0IGNvbnN0IGVudW0gT2JqZWN0S2V5IHtcbiAgU1RBVEVfUFJFRklYID0gJ3BhY2thZ2UtY2FuYXJ5LycsXG4gIFNUQVRFX1NVRkZJWCA9ICcuc3RhdGUuanNvbicsXG59XG4iXX0=
@@ -29,6 +29,12 @@ export declare class NpmJsPackageCanary extends Construct {
29
29
  * when the replica is detected to be up.
30
30
  */
31
31
  metricNpmReplicaIsDown(opts?: MetricOptions): Metric;
32
+ /**
33
+ * A metric tracking HTTP Gateway errors experienced while the canary is
34
+ * running. Those are typically caused by the npm registry servers being
35
+ * overloaded or otherwise impaired, and would cause false alarms.
36
+ */
37
+ metricHttpGatewayErrors(opts?: MetricOptions): Metric;
32
38
  metricErrors(opts?: MetricOptions): Metric;
33
39
  metricInvocations(opts?: MetricOptions): Metric;
34
40
  metricThrottles(opts?: MetricOptions): Metric;
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,EAAa,MAAM,4BAA4B,CAAC;AAG9E,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAUvC,MAAM,WAAW,uBAAuB;IACtC,QAAQ,CAAC,MAAM,EAAE,OAAO,CAAC;IACzB,QAAQ,CAAC,mBAAmB,EAAE,MAAM,CAAC;IACrC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;CAC9B;AAED,qBAAa,kBAAmB,SAAQ,SAAS;IAC/C,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAU;gBAGhC,KAAK,EAAE,SAAS,EAChB,EAAE,EAAE,MAAM,EACV,KAAK,EAAE,uBAAuB;IA8BzB,eAAe,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAU7C,mBAAmB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUjD,yBAAyB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAU9D;;;;;;;;;OASG;IACI,4BAA4B,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUjE;;;;OAIG;IACI,sBAAsB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUpD,YAAY,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI1C,iBAAiB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI/C,eAAe,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI7C,cAAc,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;CAGpD"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,EAAa,MAAM,4BAA4B,CAAC;AAG9E,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAUvC,MAAM,WAAW,uBAAuB;IACtC,QAAQ,CAAC,MAAM,EAAE,OAAO,CAAC;IACzB,QAAQ,CAAC,mBAAmB,EAAE,MAAM,CAAC;IACrC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;CAC9B;AAED,qBAAa,kBAAmB,SAAQ,SAAS;IAC/C,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAU;gBAGhC,KAAK,EAAE,SAAS,EAChB,EAAE,EAAE,MAAM,EACV,KAAK,EAAE,uBAAuB;IA8BzB,eAAe,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAU7C,mBAAmB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUjD,yBAAyB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAU9D;;;;;;;;;OASG;IACI,4BAA4B,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUjE;;;;OAIG;IACI,sBAAsB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAU3D;;;;OAIG;IACI,uBAAuB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAUrD,YAAY,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI1C,iBAAiB,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI/C,eAAe,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;IAI7C,cAAc,CAAC,IAAI,CAAC,EAAE,aAAa,GAAG,MAAM;CAGpD"}
@@ -93,6 +93,20 @@ class NpmJsPackageCanary extends constructs_1.Construct {
93
93
  namespace: constants_1.METRICS_NAMESPACE,
94
94
  });
95
95
  }
96
+ /**
97
+ * A metric tracking HTTP Gateway errors experienced while the canary is
98
+ * running. Those are typically caused by the npm registry servers being
99
+ * overloaded or otherwise impaired, and would cause false alarms.
100
+ */
101
+ metricHttpGatewayErrors(opts) {
102
+ return new aws_cloudwatch_1.Metric({
103
+ period: aws_cdk_lib_1.Duration.minutes(1),
104
+ statistic: aws_cloudwatch_1.Statistic.SUM,
105
+ ...opts,
106
+ metricName: "HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */,
107
+ namespace: constants_1.METRICS_NAMESPACE,
108
+ });
109
+ }
96
110
  metricErrors(opts) {
97
111
  return this.handler.metricErrors(opts);
98
112
  }
@@ -107,4 +121,4 @@ class NpmJsPackageCanary extends constructs_1.Construct {
107
121
  }
108
122
  }
109
123
  exports.NpmJsPackageCanary = NpmJsPackageCanary;
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"]}
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"]}
@@ -8076,6 +8076,8 @@ var require_lib2 = __commonJS({
8076
8076
  var npmjs_package_canary_lambda_exports = {};
8077
8077
  __export(npmjs_package_canary_lambda_exports, {
8078
8078
  CanaryStateService: () => CanaryStateService,
8079
+ HTTPError: () => HTTPError,
8080
+ TimeoutError: () => TimeoutError,
8079
8081
  handler: () => handler
8080
8082
  });
8081
8083
  module.exports = __toCommonJS(npmjs_package_canary_lambda_exports);
@@ -8120,58 +8122,81 @@ async function handler(event) {
8120
8122
  const constructHubEndpoint = requireEnv("CONSTRUCT_HUB_BASE_URL" /* CONSTRUCT_HUB_BASE_URL */);
8121
8123
  const stateService = new CanaryStateService(stateBucket);
8122
8124
  const constructHub = new ConstructHub(constructHubEndpoint);
8123
- const latest = await stateService.latest(packageName);
8124
- const state = await stateService.load(packageName) ?? {
8125
- latest: __spreadProps(__spreadValues({}, latest), {
8126
- availableAt: await constructHub.isInCatalog(packageName, latest.version) ? latest.publishedAt : void 0
8127
- }),
8128
- pending: {}
8129
- };
8130
- console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);
8131
- updateLatestIfNeeded(state, latest);
8132
8125
  try {
8133
- const replicaLag = await stateService.npmReplicaLagSeconds(packageName);
8134
- await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8135
- metrics.setDimensions();
8136
- metrics.putMetric("TrackedVersionCount" /* TRACKED_VERSION_COUNT */, Object.keys(state.pending).length + 1, import_aws_embedded_metrics.Unit.Count);
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
- }
8141
- })();
8142
- for (const versionState of [
8143
- state.latest,
8144
- ...Object.values(state.pending ?? {})
8145
- ]) {
8146
- console.log(`Checking state of ${versionState.version}, current: ${JSON.stringify(versionState, null, 2)}`);
8126
+ const latest = await stateService.latest(packageName);
8127
+ const state = await stateService.load(packageName) ?? {
8128
+ latest: __spreadProps(__spreadValues({}, latest), {
8129
+ availableAt: await constructHub.isInCatalog(packageName, latest.version) ? latest.publishedAt : void 0
8130
+ }),
8131
+ pending: {}
8132
+ };
8133
+ console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);
8134
+ updateLatestIfNeeded(state, latest);
8135
+ try {
8136
+ const replicaLag = await stateService.npmReplicaLagSeconds(packageName);
8147
8137
  await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8148
8138
  metrics.setDimensions();
8149
- metrics.setProperty("PackageName", packageName);
8150
- metrics.setProperty("PackageVersion", versionState.version);
8151
- metrics.setProperty("IsLatest", state.latest.version === versionState.version);
8152
- if (!versionState.availableAt) {
8153
- if (versionState.version === state.latest.version) {
8154
- if (await constructHub.isInCatalog(packageName, versionState.version)) {
8155
- versionState.availableAt = new Date();
8156
- }
8157
- } else {
8158
- if (await constructHub.hasTypeScriptDocumentation(packageName, versionState.version)) {
8159
- versionState.availableAt = new Date();
8139
+ metrics.putMetric("TrackedVersionCount" /* TRACKED_VERSION_COUNT */, Object.keys(state.pending).length + 1, import_aws_embedded_metrics.Unit.Count);
8140
+ metrics.putMetric("NpmReplicaIsDown" /* NPM_REPLICA_DOWN */, await stateService.isNpmReplicaDown() ? 1 : 0, import_aws_embedded_metrics.Unit.None);
8141
+ if (replicaLag !== void 0) {
8142
+ metrics.putMetric("EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */, replicaLag, import_aws_embedded_metrics.Unit.Seconds);
8143
+ }
8144
+ })();
8145
+ for (const versionState of [
8146
+ state.latest,
8147
+ ...Object.values(state.pending ?? {})
8148
+ ]) {
8149
+ console.log(`Checking state of ${versionState.version}, current: ${JSON.stringify(versionState, null, 2)}`);
8150
+ await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8151
+ metrics.setDimensions();
8152
+ metrics.setProperty("PackageName", packageName);
8153
+ metrics.setProperty("PackageVersion", versionState.version);
8154
+ metrics.setProperty("IsLatest", state.latest.version === versionState.version);
8155
+ if (!versionState.availableAt) {
8156
+ if (versionState.version === state.latest.version) {
8157
+ if (await constructHub.isInCatalog(packageName, versionState.version)) {
8158
+ versionState.availableAt = new Date();
8159
+ }
8160
+ } else {
8161
+ if (await constructHub.hasTypeScriptDocumentation(packageName, versionState.version)) {
8162
+ versionState.availableAt = new Date();
8163
+ }
8160
8164
  }
8161
8165
  }
8162
- }
8163
- if (versionState.availableAt) {
8164
- metrics.putMetric("TimeToCatalog" /* TIME_TO_CATALOG */, (versionState.availableAt.getTime() - versionState.publishedAt.getTime()) / 1e3, import_aws_embedded_metrics.Unit.Seconds);
8165
- if (versionState.version in state.pending) {
8166
- delete state.pending[versionState.version];
8166
+ if (versionState.availableAt) {
8167
+ metrics.putMetric("TimeToCatalog" /* TIME_TO_CATALOG */, (versionState.availableAt.getTime() - versionState.publishedAt.getTime()) / 1e3, import_aws_embedded_metrics.Unit.Seconds);
8168
+ if (versionState.version in state.pending) {
8169
+ delete state.pending[versionState.version];
8170
+ }
8171
+ } else {
8172
+ metrics.putMetric("DwellTime" /* DWELL_TIME */, (Date.now() - versionState.publishedAt.getTime()) / 1e3, import_aws_embedded_metrics.Unit.Seconds);
8167
8173
  }
8168
- } else {
8169
- metrics.putMetric("DwellTime" /* DWELL_TIME */, (Date.now() - versionState.publishedAt.getTime()) / 1e3, import_aws_embedded_metrics.Unit.Seconds);
8170
- }
8174
+ metrics.putMetric("HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */, 0, import_aws_embedded_metrics.Unit.Count);
8175
+ })();
8176
+ }
8177
+ } finally {
8178
+ await stateService.save(packageName, state);
8179
+ }
8180
+ } catch (error) {
8181
+ if (error instanceof HTTPError && (error.httpStatusCode === 502 || error.httpStatusCode === 504)) {
8182
+ console.error("HTTP 5XX from a dependency, assuming this is transient:", error);
8183
+ await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8184
+ metrics.setDimensions();
8185
+ metrics.setProperty("ErrorCode", error.httpStatusCode);
8186
+ metrics.setProperty("ErrorMessage", error.message);
8187
+ metrics.putMetric("HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */, 1, import_aws_embedded_metrics.Unit.Count);
8171
8188
  })();
8189
+ } else if (error instanceof TimeoutError) {
8190
+ console.error(`Request timeout from a dependency, assuming this is transient:`, error);
8191
+ await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {
8192
+ metrics.setDimensions();
8193
+ metrics.setProperty("ErrorCode", "REQUEST_TIMEOUT");
8194
+ metrics.setProperty("ErrorMessage", error.message);
8195
+ metrics.putMetric("HttpGatewayErrors" /* HTTP_GATEWAY_ERRORS */, 1, import_aws_embedded_metrics.Unit.Count);
8196
+ })();
8197
+ } else {
8198
+ throw error;
8172
8199
  }
8173
- } finally {
8174
- await stateService.save(packageName, state);
8175
8200
  }
8176
8201
  }
8177
8202
  var _catalog;
@@ -8196,7 +8221,7 @@ var ConstructHub = class {
8196
8221
  if (res.statusCode === 200) {
8197
8222
  return ok(!!((_a = res.headers["content-type"]) == null ? void 0 : _a.startsWith("text/markdown")));
8198
8223
  }
8199
- const err = new Error(`HEAD ${url} -- HTTP ${res.statusCode} (${res.statusMessage})`);
8224
+ const err = new HTTPError(res.statusCode, `HEAD ${url} -- HTTP ${res.statusCode} (${res.statusMessage})`);
8200
8225
  Error.captureStackTrace(err);
8201
8226
  ko(err);
8202
8227
  }).end();
@@ -8244,14 +8269,14 @@ var CanaryStateService = class {
8244
8269
  }
8245
8270
  async latest(packageName) {
8246
8271
  console.log(`Fetching latest version information from NPM: ${packageName}`);
8247
- const version = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`, ["version"]);
8248
- const publishedAt = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}`, ["time", version]);
8272
+ const version = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`, { jsonPath: ["version"] });
8273
+ const publishedAt = await getJSON(`https://registry.npmjs.org/${encodeURIComponent(packageName)}`, { jsonPath: ["time", version] });
8249
8274
  console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
8250
8275
  return { version, publishedAt: new Date(publishedAt) };
8251
8276
  }
8252
8277
  async isNpmReplicaDown() {
8253
8278
  try {
8254
- await getJSON("https://replicate.npmjs.com/");
8279
+ await getJSON("https://replicate.npmjs.com/", { timeoutMillis: 5e3 });
8255
8280
  return false;
8256
8281
  } catch (e) {
8257
8282
  return true;
@@ -8263,7 +8288,7 @@ var CanaryStateService = class {
8263
8288
  const primaryDate = await getModifiedTimestamp(`registry.npmjs.org`);
8264
8289
  let replicaDate;
8265
8290
  try {
8266
- replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`);
8291
+ replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`, 5e3);
8267
8292
  } catch (e) {
8268
8293
  if (e instanceof Error && e.message.includes("HTTP 504")) {
8269
8294
  console.log(`Warning: error fetching replicate.npmjs.com: ${e.toString()}`);
@@ -8276,8 +8301,8 @@ var CanaryStateService = class {
8276
8301
  console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
8277
8302
  console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 36e5} hours behind)`);
8278
8303
  return deltaMs / 1e3;
8279
- async function getModifiedTimestamp(baseUrl) {
8280
- const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, ["time", "modified"]);
8304
+ async function getModifiedTimestamp(baseUrl, timeoutMillis) {
8305
+ const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, { jsonPath: ["time", "modified"], timeoutMillis });
8281
8306
  return new Date(isoDate);
8282
8307
  }
8283
8308
  }
@@ -8288,22 +8313,47 @@ var CanaryStateService = class {
8288
8313
  return `s3://${this.bucketName}/${this.key(packageName)}`;
8289
8314
  }
8290
8315
  };
8291
- function getJSON(url, jsonPath) {
8316
+ var HTTPError = class extends Error {
8317
+ constructor(httpStatusCode, message) {
8318
+ super(message);
8319
+ this.httpStatusCode = httpStatusCode;
8320
+ Error.captureStackTrace(this, HTTPError);
8321
+ }
8322
+ };
8323
+ var TimeoutError = class extends Error {
8324
+ constructor(message) {
8325
+ super(message);
8326
+ Error.captureStackTrace(this, TimeoutError);
8327
+ }
8328
+ };
8329
+ function getJSON(url, {
8330
+ jsonPath,
8331
+ timeoutMillis
8332
+ } = {}) {
8292
8333
  return new Promise((ok, ko) => {
8293
8334
  https.get(url, {
8294
- headers: { Accept: "application/json", "Accept-Encoding": "identity" }
8335
+ headers: {
8336
+ Accept: "application/json",
8337
+ "Accept-Encoding": "identity"
8338
+ },
8339
+ timeout: timeoutMillis
8295
8340
  }, (res) => {
8296
8341
  if (res.statusCode !== 200) {
8297
- const error = new Error(`GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`);
8342
+ const error = new HTTPError(res.statusCode, `GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`);
8298
8343
  Error.captureStackTrace(error);
8299
8344
  return ko(error);
8300
8345
  }
8301
8346
  res.once("error", ko);
8347
+ res.once("timeout", () => {
8348
+ ko(new TimeoutError(`Request timed out (after ${timeoutMillis ?? "N/A"} ms): GET ${url}`));
8349
+ });
8302
8350
  const json = JSONStream.parse(jsonPath);
8303
8351
  json.once("data", ok);
8304
8352
  json.once("error", ko);
8305
8353
  const plainPayload = res.headers["content-encoding"] === "gzip" ? gunzip(res) : res;
8306
8354
  plainPayload.pipe(json, { end: true });
8355
+ }).once("timeout", () => {
8356
+ ko(new TimeoutError(`Request timed out (after ${timeoutMillis ?? "N/A"} ms): GET ${url}`));
8307
8357
  });
8308
8358
  });
8309
8359
  }
@@ -8326,6 +8376,8 @@ function gunzip(readable) {
8326
8376
  // Annotate the CommonJS export names for ESM import in node:
8327
8377
  0 && (module.exports = {
8328
8378
  CanaryStateService,
8379
+ HTTPError,
8380
+ TimeoutError,
8329
8381
  handler
8330
8382
  });
8331
8383
  //# sourceMappingURL=index.js.map