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.
- package/.jsii +2 -2
- package/changelog.md +1 -1
- package/lib/construct-hub.js +1 -1
- package/lib/package-sources/code-artifact.js +1 -1
- package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.bundle/index.js +0 -1
- package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.bundle/index.js.map +2 -2
- package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.d.ts.map +1 -1
- package/lib/package-sources/codeartifact/code-artifact-forwarder.lambda.js +1 -2
- package/lib/package-sources/npmjs/canary/constants.d.ts +7 -1
- package/lib/package-sources/npmjs/canary/constants.d.ts.map +1 -1
- package/lib/package-sources/npmjs/canary/constants.js +1 -1
- package/lib/package-sources/npmjs/canary/index.d.ts +6 -0
- package/lib/package-sources/npmjs/canary/index.d.ts.map +1 -1
- package/lib/package-sources/npmjs/canary/index.js +15 -1
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js +106 -54
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js.map +2 -2
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.d.ts +7 -0
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.d.ts.map +1 -1
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.js +142 -77
- package/lib/package-sources/npmjs.d.ts.map +1 -1
- package/lib/package-sources/npmjs.js +21 -2
- package/lib/package-tag/index.js +3 -3
- package/lib/package-tag-group/index.js +2 -2
- package/lib/preload-file/index.js +1 -1
- package/lib/s3/storage.js +1 -1
- package/lib/spdx-license.js +1 -1
- package/package.json +1 -1
- package/releasetag.txt +1 -1
- package/version.txt +1 -1
@@ -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,+
|
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;
|
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,
|
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;
|
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
|
8134
|
-
|
8135
|
-
|
8136
|
-
|
8137
|
-
|
8138
|
-
|
8139
|
-
|
8140
|
-
|
8141
|
-
|
8142
|
-
|
8143
|
-
|
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.
|
8150
|
-
metrics.
|
8151
|
-
|
8152
|
-
|
8153
|
-
|
8154
|
-
|
8155
|
-
|
8156
|
-
|
8157
|
-
|
8158
|
-
|
8159
|
-
|
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
|
-
|
8164
|
-
|
8165
|
-
|
8166
|
-
|
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
|
-
|
8169
|
-
|
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
|
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
|
-
|
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: {
|
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
|
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
|