construct-hub 0.4.4 → 0.4.7
Sign up to get free protection for your applications and to get access to all the features.
- package/.jsii +119 -57
- package/API.md +50 -4
- package/changelog.md +6 -1
- package/docs/operator-runbook.md +32 -0
- package/lib/backend/catalog-builder/index.js +5 -1
- package/lib/construct-hub.d.ts +19 -0
- package/lib/construct-hub.js +37 -12
- package/lib/monitoring/api.d.ts +3 -3
- package/lib/monitoring/api.js +1 -1
- package/lib/monitoring/index.d.ts +4 -4
- package/lib/monitoring/index.js +5 -5
- package/lib/package-sources/code-artifact.js +1 -1
- package/lib/package-sources/npmjs/canary/constants.d.ts +14 -1
- package/lib/package-sources/npmjs/canary/constants.js +1 -1
- package/lib/package-sources/npmjs/canary/index.d.ts +22 -0
- package/lib/package-sources/npmjs/canary/index.js +49 -4
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.bundle/index.js +40 -1
- 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 +63 -0
- package/lib/package-sources/npmjs/canary/npmjs-package-canary.lambda.js +54 -3
- package/lib/package-sources/npmjs.js +68 -8
- 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
@@ -15,3 +15,66 @@
|
|
15
15
|
* will receive one sample per tracked version.
|
16
16
|
*/
|
17
17
|
export declare function handler(event: unknown): Promise<void>;
|
18
|
+
export declare class CanaryStateService {
|
19
|
+
private readonly bucketName;
|
20
|
+
constructor(bucketName: string);
|
21
|
+
/**
|
22
|
+
* Save the state to the bucket.
|
23
|
+
*/
|
24
|
+
save(packageName: string, state: CanaryState): Promise<void>;
|
25
|
+
/**
|
26
|
+
* Load the state file for this package from the bucket.
|
27
|
+
*/
|
28
|
+
load(packageName: string): Promise<CanaryState | undefined>;
|
29
|
+
/**
|
30
|
+
* Create a state from the latest version of the package.
|
31
|
+
*/
|
32
|
+
latest(packageName: string): Promise<CanaryState['latest']>;
|
33
|
+
isNpmReplicaDown(): Promise<boolean>;
|
34
|
+
/**
|
35
|
+
* Estimate how far behind the NPM replica is compared to the live NPM
|
36
|
+
* registry. If the NPM replica is down, return undefined.
|
37
|
+
*/
|
38
|
+
npmReplicaLagSeconds(packageName: string): Promise<number | undefined>;
|
39
|
+
private key;
|
40
|
+
private url;
|
41
|
+
}
|
42
|
+
interface CanaryState {
|
43
|
+
/**
|
44
|
+
* The latest package version, as of the last execution of the canary.
|
45
|
+
*/
|
46
|
+
latest: {
|
47
|
+
/**
|
48
|
+
* The version we are tracking.
|
49
|
+
*/
|
50
|
+
readonly version: string;
|
51
|
+
/**
|
52
|
+
* The publish date of the version.
|
53
|
+
*/
|
54
|
+
readonly publishedAt: Date;
|
55
|
+
/**
|
56
|
+
* The date at which the version is available on the hub.
|
57
|
+
*/
|
58
|
+
availableAt?: Date;
|
59
|
+
};
|
60
|
+
/**
|
61
|
+
* Each existing, but not-yet-found versions that are still tracked.
|
62
|
+
*/
|
63
|
+
pending: {
|
64
|
+
[version: string]: {
|
65
|
+
/**
|
66
|
+
* The version we are tracking.
|
67
|
+
*/
|
68
|
+
readonly version: string;
|
69
|
+
/**
|
70
|
+
* The publish date of the version.
|
71
|
+
*/
|
72
|
+
readonly publishedAt: Date;
|
73
|
+
/**
|
74
|
+
* These pending packages are NEVER available at this point.
|
75
|
+
*/
|
76
|
+
availableAt: undefined;
|
77
|
+
};
|
78
|
+
};
|
79
|
+
}
|
80
|
+
export {};
|
@@ -14,7 +14,7 @@ var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (
|
|
14
14
|
};
|
15
15
|
var _catalog;
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
17
|
-
exports.handler = void 0;
|
17
|
+
exports.CanaryStateService = exports.handler = void 0;
|
18
18
|
const https = require("https");
|
19
19
|
const zlib_1 = require("zlib");
|
20
20
|
const aws_embedded_metrics_1 = require("aws-embedded-metrics");
|
@@ -66,10 +66,17 @@ async function handler(event) {
|
|
66
66
|
// If the current "latest" isn't the one from state, it needs updating.
|
67
67
|
updateLatestIfNeeded(state, latest);
|
68
68
|
try {
|
69
|
-
await
|
69
|
+
const replicaLag = await stateService.npmReplicaLagSeconds(packageName);
|
70
|
+
await aws_embedded_metrics_1.metricScope((metrics) => async () => {
|
70
71
|
// Clear out default dimensions as we don't need those. See https://github.com/awslabs/aws-embedded-metrics-node/issues/73.
|
71
72
|
metrics.setDimensions();
|
72
73
|
metrics.putMetric("TrackedVersionCount" /* TRACKED_VERSION_COUNT */, Object.keys(state.pending).length + 1, aws_embedded_metrics_1.Unit.Count);
|
74
|
+
metrics.putMetric("NpmReplicaIsDown" /* NPM_REPLICA_DOWN */, (await stateService.isNpmReplicaDown()) ? 1 : 0, aws_embedded_metrics_1.Unit.None);
|
75
|
+
// If we weren't able to calculate the replica's lag, then simply
|
76
|
+
// don't report the metric.
|
77
|
+
if (replicaLag !== undefined) {
|
78
|
+
metrics.putMetric("EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */, replicaLag, aws_embedded_metrics_1.Unit.Seconds);
|
79
|
+
}
|
73
80
|
})();
|
74
81
|
for (const versionState of [
|
75
82
|
state.latest,
|
@@ -235,6 +242,49 @@ class CanaryStateService {
|
|
235
242
|
console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
|
236
243
|
return { version, publishedAt: new Date(publishedAt) };
|
237
244
|
}
|
245
|
+
async isNpmReplicaDown() {
|
246
|
+
try {
|
247
|
+
await getJSON('https://replicate.npmjs.com/');
|
248
|
+
return false;
|
249
|
+
}
|
250
|
+
catch (e) {
|
251
|
+
return true;
|
252
|
+
}
|
253
|
+
}
|
254
|
+
/**
|
255
|
+
* Estimate how far behind the NPM replica is compared to the live NPM
|
256
|
+
* registry. If the NPM replica is down, return undefined.
|
257
|
+
*/
|
258
|
+
async npmReplicaLagSeconds(packageName) {
|
259
|
+
const encodedPackageName = encodeURIComponent(packageName);
|
260
|
+
console.log(`Measuring NPM replica lag using ${packageName}...`);
|
261
|
+
const primaryDate = await getModifiedTimestamp(`registry.npmjs.org`);
|
262
|
+
let replicaDate;
|
263
|
+
try {
|
264
|
+
replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`);
|
265
|
+
}
|
266
|
+
catch (e) {
|
267
|
+
if (e instanceof Error && e.message.includes('HTTP 504')) {
|
268
|
+
console.log(`Warning: error fetching replicate.npmjs.com: ${e.toString()}`);
|
269
|
+
// There is no value to report
|
270
|
+
return undefined;
|
271
|
+
}
|
272
|
+
else {
|
273
|
+
throw e;
|
274
|
+
}
|
275
|
+
}
|
276
|
+
const deltaMs = primaryDate.getTime() - replicaDate.getTime();
|
277
|
+
console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
|
278
|
+
console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 3600000} hours behind)`);
|
279
|
+
// We return in seconds... The millisecond resolution is silly here since the probe package is
|
280
|
+
// only published approximately once every three hours. We use seconds only because this is the
|
281
|
+
// largest available time unit in CloudWatch.
|
282
|
+
return deltaMs / 1000;
|
283
|
+
async function getModifiedTimestamp(baseUrl) {
|
284
|
+
const isoDate = await getJSON(`https://${baseUrl}/${encodedPackageName}`, ['time', 'modified']);
|
285
|
+
return new Date(isoDate);
|
286
|
+
}
|
287
|
+
}
|
238
288
|
key(packageName) {
|
239
289
|
return `${"package-canary/" /* STATE_PREFIX */}${packageName}${".state.json" /* STATE_SUFFIX */}`;
|
240
290
|
}
|
@@ -242,6 +292,7 @@ class CanaryStateService {
|
|
242
292
|
return `s3://${this.bucketName}/${this.key(packageName)}`;
|
243
293
|
}
|
244
294
|
}
|
295
|
+
exports.CanaryStateService = CanaryStateService;
|
245
296
|
/**
|
246
297
|
* Makes a request to the provided URL and returns the response after having
|
247
298
|
* parsed it from JSON.
|
@@ -298,4 +349,4 @@ function gunzip(readable) {
|
|
298
349
|
readable.pipe(gz, { end: true });
|
299
350
|
return gz;
|
300
351
|
}
|
301
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npmjs-package-canary.lambda.js","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/npmjs-package-canary.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as https from 'https';\nimport { Readable } from 'stream';\nimport { createGunzip } from 'zlib';\nimport { metricScope, Configuration, Unit } from 'aws-embedded-metrics';\nimport type { AWSError, S3 } from 'aws-sdk';\nimport * as JSONStream from 'JSONStream';\nimport { CatalogModel } from '../../../backend';\nimport * as aws from '../../../backend/shared/aws.lambda-shared';\nimport { requireEnv } from '../../../backend/shared/env.lambda-shared';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  Environment,\n  ObjectKey,\n} from './constants';\n\nConfiguration.namespace = METRICS_NAMESPACE;\n\n/**\n * This package canary monitors the availability of the versions of a specified\n * package in the ConstructHub catalog. It publishes metrics that help\n * understand how much time passes between a pakcage appearing in the public\n * registry and it's availability in the ConstructHub instance.\n *\n * From the moment a package has been published, and until it appeared in\n * catalog, the `MetricName.DWELL_TIME` metric is emitted.\n *\n * Once the package has appeared in catalog, and until a new package version is\n * identified in npmjs.com, the `MetricName.TIME_TO_CATALOG` metric is emitted.\n *\n * If a new package version is published before the previous one has appeared\n * in catalog, both versions will be tracked at the same time, and the metrics\n * will receive one sample per tracked version.\n */\nexport async function handler(event: unknown): Promise<void> {\n  console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n  const packageName = requireEnv(Environment.PACKAGE_NAME);\n  const stateBucket = requireEnv(Environment.PACKAGE_CANARY_BUCKET_NAME);\n  const constructHubEndpoint = requireEnv(Environment.CONSTRUCT_HUB_BASE_URL);\n\n  const stateService = new CanaryStateService(stateBucket);\n  const constructHub = new ConstructHub(constructHubEndpoint);\n\n  const latest = await stateService.latest(packageName);\n  const state: CanaryState = (await stateService.load(packageName)) ?? {\n    // If we did not have any state, we'll bootstrap using the current latest version.\n    latest: {\n      ...latest,\n      // If that latest version is ALREADY in catalog, pretend it was\n      // \"instantaneously\" there, so we avoid possibly reporting an breach of\n      // SLA alarm, when we really just observed presence of the package in\n      // catalog too late, for example on first deployment of the canary.\n      availableAt: (await constructHub.isInCatalog(packageName, latest.version))\n        ? latest.publishedAt\n        : undefined,\n    },\n    pending: {},\n  };\n\n  console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);\n\n  // If the current \"latest\" isn't the one from state, it needs updating.\n  updateLatestIfNeeded(state, latest);\n\n  try {\n    await metricScope((metrics) => () => {\n      // Clear out default dimensions as we don't need those. See https://github.com/awslabs/aws-embedded-metrics-node/issues/73.\n      metrics.setDimensions();\n      metrics.putMetric(\n        MetricName.TRACKED_VERSION_COUNT,\n        Object.keys(state.pending).length + 1,\n        Unit.Count\n      );\n    })();\n\n    for (const versionState of [\n      state.latest,\n      ...Object.values(state.pending ?? {}),\n    ]) {\n      console.log(\n        `Checking state of ${versionState.version}, current: ${JSON.stringify(\n          versionState,\n          null,\n          2\n        )}`\n      );\n\n      await metricScope((metrics) => async () => {\n        // Clear out default dimensions as we don't need those. See https://github.com/awslabs/aws-embedded-metrics-node/issues/73.\n        metrics.setDimensions();\n        metrics.setProperty('PackageName', packageName);\n        metrics.setProperty('PackageVersion', versionState.version);\n        metrics.setProperty(\n          'IsLatest',\n          state.latest.version === versionState.version\n        );\n\n        if (!versionState.availableAt) {\n          if (versionState.version === state.latest.version) {\n            if (\n              await constructHub.isInCatalog(packageName, versionState.version)\n            ) {\n              versionState.availableAt = new Date();\n            }\n          } else {\n            // Non-current versions will probably never make it to catalog (they're older than the\n            // current version), so instead, we check whether they have TypeScript documentation.\n            if (\n              await constructHub.hasTypeScriptDocumentation(\n                packageName,\n                versionState.version\n              )\n            ) {\n              versionState.availableAt = new Date();\n            }\n          }\n        }\n\n        if (versionState.availableAt) {\n          // Tells us how long it's taken for the package to make it to catalog after it was published.\n          metrics.putMetric(\n            MetricName.TIME_TO_CATALOG,\n            (versionState.availableAt.getTime() -\n              versionState.publishedAt.getTime()) /\n              1_000,\n            Unit.Seconds\n          );\n\n          // Stop tracking that version, as it's now available.\n          if (versionState.version in state.pending) {\n            delete state.pending[versionState.version];\n          }\n        } else {\n          // Tells us how long we've been waiting for this version to show up, so far.\n          metrics.putMetric(\n            MetricName.DWELL_TIME,\n            (Date.now() - versionState.publishedAt.getTime()) / 1_000,\n            Unit.Seconds\n          );\n        }\n      })();\n    }\n  } finally {\n    await stateService.save(packageName, state);\n  }\n}\n\nclass ConstructHub {\n  #catalog?: CatalogModel;\n\n  constructor(private readonly baseUrl: string) {}\n\n  /**\n   * Determines whether the specified package version is present in the catalog\n   * object or not.\n   *\n   * @param packageName    the name of the checked package.\n   * @param packageVersion the version of the checked package.\n   *\n   * @returns `true` IIF the exact package version is found in the catalog.\n   */\n  public async isInCatalog(\n    packageName: string,\n    packageVersion: string\n  ): Promise<boolean> {\n    const catalog = await this.getCatalog();\n    const filtered = catalog.packages.filter(\n      (p: any) => p.name === packageName && p.version === packageVersion\n    );\n\n    if (filtered.length > 1) {\n      throw new Error(\n        `Found multiple entries for ${packageName}@${packageVersion} in catalog`\n      );\n    }\n\n    return filtered.length === 1;\n  }\n\n  /**\n   * Checks whether TypeScript documentation exists in ConstructHub for the\n   * specified package version.\n   *\n   * @param packageName    the name of the checked package.\n   * @param packageVersion the version of the checked package.\n   *\n   * @returns `true` IIF the `docs-typescript.md` document exists for the\n   *          specified package.\n   */\n  public async hasTypeScriptDocumentation(\n    packageName: string,\n    packageVersion: string\n  ): Promise<boolean> {\n    return new Promise((ok, ko) => {\n      const url = `${this.baseUrl}/data/${packageName}/v${packageVersion}/docs-typescript.md`;\n      https\n        .request(url, { method: 'HEAD' }, (res) => {\n          if (res.statusCode === 200) {\n            // This returns HTTP 200 with text/html if it's a 404, due to how\n            // we configured CloudFront behaviors.\n            return ok(\n              !!res.headers['content-type']?.startsWith('text/markdown')\n            );\n          }\n          const err = new Error(\n            `HEAD ${url} -- HTTP ${res.statusCode} (${res.statusMessage})`\n          );\n          Error.captureStackTrace(err);\n          ko(err);\n        })\n        .end();\n    });\n  }\n\n  private async getCatalog(): Promise<CatalogModel> {\n    if (this.#catalog) {\n      return this.#catalog;\n    }\n    return (this.#catalog = await getJSON(`${this.baseUrl}/catalog.json`));\n  }\n}\n\nclass CanaryStateService {\n  constructor(private readonly bucketName: string) {}\n\n  /**\n   * Save the state to the bucket.\n   */\n  public async save(packageName: string, state: CanaryState) {\n    const url = this.url(packageName);\n\n    console.log(`Saving to ${url}: ${JSON.stringify(state, null, 2)}`);\n    await aws\n      .s3()\n      .putObject({\n        Bucket: this.bucketName,\n        Key: this.key(packageName),\n        Body: JSON.stringify(state, null, 2),\n        ContentType: 'application/json',\n      })\n      .promise();\n  }\n\n  /**\n   * Load the state file for this package from the bucket.\n   */\n  public async load(packageName: string): Promise<CanaryState | undefined> {\n    console.log(`Loading state for package '${packageName}'`);\n\n    const objectKey = this.key(packageName);\n    const url = this.url(packageName);\n\n    console.log(`Fetching: ${url}`);\n    const data = await aws\n      .s3()\n      .getObject({ Bucket: this.bucketName, Key: objectKey })\n      .promise()\n      .catch((err: AWSError) =>\n        err.code !== 'NoSuchKey'\n          ? Promise.reject(err)\n          : Promise.resolve({\n              /* no data */\n            } as S3.GetObjectOutput)\n      );\n\n    if (!data?.Body) {\n      console.log(`Not found: ${url}`);\n      return undefined;\n    }\n\n    console.log(`Loaded: ${url}`);\n    return JSON.parse(data.Body.toString('utf-8'), (key, value) => {\n      if (key === 'publishedAt' || key === 'availableAt') {\n        return new Date(value);\n      }\n      return value;\n    });\n  }\n\n  /**\n   * Create a state from the latest version of the package.\n   */\n  public async latest(packageName: string): Promise<CanaryState['latest']> {\n    console.log(`Fetching latest version information from NPM: ${packageName}`);\n    const version = await getJSON(\n      `https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`,\n      ['version']\n    );\n    const publishedAt = await getJSON(\n      `https://registry.npmjs.org/${encodeURIComponent(packageName)}`,\n      ['time', version]\n    );\n\n    console.log(\n      `Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`\n    );\n\n    return { version, publishedAt: new Date(publishedAt) };\n  }\n\n  private key(packageName: string): string {\n    return `${ObjectKey.STATE_PREFIX}${packageName}${ObjectKey.STATE_SUFFIX}`;\n  }\n\n  private url(packageName: string) {\n    return `s3://${this.bucketName}/${this.key(packageName)}`;\n  }\n}\n\ninterface CanaryState {\n  /**\n   * The latest package version, as of the last execution of the canary.\n   */\n  latest: {\n    /**\n     * The version we are tracking.\n     */\n    readonly version: string;\n\n    /**\n     * The publish date of the version.\n     */\n    readonly publishedAt: Date;\n\n    /**\n     * The date at which the version is available on the hub.\n     */\n    availableAt?: Date;\n  };\n\n  /**\n   * Each existing, but not-yet-found versions that are still tracked.\n   */\n  pending: {\n    [version: string]: {\n      /**\n       * The version we are tracking.\n       */\n      readonly version: string;\n\n      /**\n       * The publish date of the version.\n       */\n      readonly publishedAt: Date;\n\n      /**\n       * These pending packages are NEVER available at this point.\n       */\n      availableAt: undefined;\n    };\n  };\n}\n\n/**\n * Makes a request to the provided URL and returns the response after having\n * parsed it from JSON.\n *\n * @param url the URL to get.\n * @param jsonPath a JSON path to extract only a subset of the object.\n */\nfunction getJSON(url: string, jsonPath?: string[]): Promise<any> {\n  return new Promise((ok, ko) => {\n    https.get(\n      url,\n      {\n        headers: { Accept: 'application/json', 'Accept-Encoding': 'identity' },\n      },\n      (res) => {\n        if (res.statusCode !== 200) {\n          const error = new Error(\n            `GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`\n          );\n          Error.captureStackTrace(error);\n          return ko(error);\n        }\n\n        res.once('error', ko);\n\n        const json = JSONStream.parse(jsonPath);\n        json.once('data', ok);\n        json.once('error', ko);\n\n        const plainPayload =\n          res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n        plainPayload.pipe(json, { end: true });\n      }\n    );\n  });\n}\n\n/**\n * Updates the `latest` property of `state` ti the provided `latest` value,\n * unless this is already the current latest.\n *\n * If the previous latest version does not have the `availableAt` property, adds\n * that to the `pending` set.\n *\n * @param state  the state to be updated.\n * @param latest the current \"latest\" version of the tracked package.\n */\nfunction updateLatestIfNeeded(\n  state: CanaryState,\n  latest: CanaryState['latest']\n): void {\n  if (state.latest.version === latest.version) {\n    return;\n  }\n\n  // If the current \"latest\" isn't available yet, add it to the `pending` versions.\n  if (state.latest.availableAt == null) {\n    // The TypeScript version of jsii doesn't do control flow analysis well enough here to\n    // determine that the`if` branch guarantees `availableAt` is undefined here.\n    state.pending[state.latest.version] = {\n      ...state.latest,\n      availableAt: undefined,\n    };\n  }\n\n  state.latest = latest;\n}\n\nfunction gunzip(readable: Readable): Readable {\n  const gz = createGunzip();\n  readable.pipe(gz, { end: true });\n  return gz;\n}\n"]}
|
352
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npmjs-package-canary.lambda.js","sourceRoot":"","sources":["../../../../src/package-sources/npmjs/canary/npmjs-package-canary.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as https from 'https';\nimport { Readable } from 'stream';\nimport { createGunzip } from 'zlib';\nimport { metricScope, Configuration, Unit } from 'aws-embedded-metrics';\nimport type { AWSError, S3 } from 'aws-sdk';\nimport * as JSONStream from 'JSONStream';\nimport { CatalogModel } from '../../../backend';\nimport * as aws from '../../../backend/shared/aws.lambda-shared';\nimport { requireEnv } from '../../../backend/shared/env.lambda-shared';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  Environment,\n  ObjectKey,\n} from './constants';\n\nConfiguration.namespace = METRICS_NAMESPACE;\n\n/**\n * This package canary monitors the availability of the versions of a specified\n * package in the ConstructHub catalog. It publishes metrics that help\n * understand how much time passes between a pakcage appearing in the public\n * registry and it's availability in the ConstructHub instance.\n *\n * From the moment a package has been published, and until it appeared in\n * catalog, the `MetricName.DWELL_TIME` metric is emitted.\n *\n * Once the package has appeared in catalog, and until a new package version is\n * identified in npmjs.com, the `MetricName.TIME_TO_CATALOG` metric is emitted.\n *\n * If a new package version is published before the previous one has appeared\n * in catalog, both versions will be tracked at the same time, and the metrics\n * will receive one sample per tracked version.\n */\nexport async function handler(event: unknown): Promise<void> {\n  console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n  const packageName = requireEnv(Environment.PACKAGE_NAME);\n  const stateBucket = requireEnv(Environment.PACKAGE_CANARY_BUCKET_NAME);\n  const constructHubEndpoint = requireEnv(Environment.CONSTRUCT_HUB_BASE_URL);\n\n  const stateService = new CanaryStateService(stateBucket);\n  const constructHub = new ConstructHub(constructHubEndpoint);\n\n  const latest = await stateService.latest(packageName);\n  const state: CanaryState = (await stateService.load(packageName)) ?? {\n    // If we did not have any state, we'll bootstrap using the current latest version.\n    latest: {\n      ...latest,\n      // If that latest version is ALREADY in catalog, pretend it was\n      // \"instantaneously\" there, so we avoid possibly reporting an breach of\n      // SLA alarm, when we really just observed presence of the package in\n      // catalog too late, for example on first deployment of the canary.\n      availableAt: (await constructHub.isInCatalog(packageName, latest.version))\n        ? latest.publishedAt\n        : undefined,\n    },\n    pending: {},\n  };\n\n  console.log(`Initial state: ${JSON.stringify(state, null, 2)}`);\n\n  // If the current \"latest\" isn't the one from state, it needs updating.\n  updateLatestIfNeeded(state, latest);\n\n  try {\n    const replicaLag = await stateService.npmReplicaLagSeconds(packageName);\n\n    await metricScope((metrics) => async () => {\n      // Clear out default dimensions as we don't need those. See https://github.com/awslabs/aws-embedded-metrics-node/issues/73.\n      metrics.setDimensions();\n      metrics.putMetric(\n        MetricName.TRACKED_VERSION_COUNT,\n        Object.keys(state.pending).length + 1,\n        Unit.Count\n      );\n      metrics.putMetric(\n        MetricName.NPM_REPLICA_DOWN,\n        (await stateService.isNpmReplicaDown()) ? 1 : 0,\n        Unit.None\n      );\n\n      // If we weren't able to calculate the replica's lag, then simply\n      // don't report the metric.\n      if (replicaLag !== undefined) {\n        metrics.putMetric(MetricName.NPM_REPLICA_LAG, replicaLag, Unit.Seconds);\n      }\n    })();\n\n    for (const versionState of [\n      state.latest,\n      ...Object.values(state.pending ?? {}),\n    ]) {\n      console.log(\n        `Checking state of ${versionState.version}, current: ${JSON.stringify(\n          versionState,\n          null,\n          2\n        )}`\n      );\n\n      await metricScope((metrics) => async () => {\n        // Clear out default dimensions as we don't need those. See https://github.com/awslabs/aws-embedded-metrics-node/issues/73.\n        metrics.setDimensions();\n        metrics.setProperty('PackageName', packageName);\n        metrics.setProperty('PackageVersion', versionState.version);\n        metrics.setProperty(\n          'IsLatest',\n          state.latest.version === versionState.version\n        );\n\n        if (!versionState.availableAt) {\n          if (versionState.version === state.latest.version) {\n            if (\n              await constructHub.isInCatalog(packageName, versionState.version)\n            ) {\n              versionState.availableAt = new Date();\n            }\n          } else {\n            // Non-current versions will probably never make it to catalog (they're older than the\n            // current version), so instead, we check whether they have TypeScript documentation.\n            if (\n              await constructHub.hasTypeScriptDocumentation(\n                packageName,\n                versionState.version\n              )\n            ) {\n              versionState.availableAt = new Date();\n            }\n          }\n        }\n\n        if (versionState.availableAt) {\n          // Tells us how long it's taken for the package to make it to catalog after it was published.\n          metrics.putMetric(\n            MetricName.TIME_TO_CATALOG,\n            (versionState.availableAt.getTime() -\n              versionState.publishedAt.getTime()) /\n              1_000,\n            Unit.Seconds\n          );\n\n          // Stop tracking that version, as it's now available.\n          if (versionState.version in state.pending) {\n            delete state.pending[versionState.version];\n          }\n        } else {\n          // Tells us how long we've been waiting for this version to show up, so far.\n          metrics.putMetric(\n            MetricName.DWELL_TIME,\n            (Date.now() - versionState.publishedAt.getTime()) / 1_000,\n            Unit.Seconds\n          );\n        }\n      })();\n    }\n  } finally {\n    await stateService.save(packageName, state);\n  }\n}\n\nclass ConstructHub {\n  #catalog?: CatalogModel;\n\n  constructor(private readonly baseUrl: string) {}\n\n  /**\n   * Determines whether the specified package version is present in the catalog\n   * object or not.\n   *\n   * @param packageName    the name of the checked package.\n   * @param packageVersion the version of the checked package.\n   *\n   * @returns `true` IIF the exact package version is found in the catalog.\n   */\n  public async isInCatalog(\n    packageName: string,\n    packageVersion: string\n  ): Promise<boolean> {\n    const catalog = await this.getCatalog();\n    const filtered = catalog.packages.filter(\n      (p: any) => p.name === packageName && p.version === packageVersion\n    );\n\n    if (filtered.length > 1) {\n      throw new Error(\n        `Found multiple entries for ${packageName}@${packageVersion} in catalog`\n      );\n    }\n\n    return filtered.length === 1;\n  }\n\n  /**\n   * Checks whether TypeScript documentation exists in ConstructHub for the\n   * specified package version.\n   *\n   * @param packageName    the name of the checked package.\n   * @param packageVersion the version of the checked package.\n   *\n   * @returns `true` IIF the `docs-typescript.md` document exists for the\n   *          specified package.\n   */\n  public async hasTypeScriptDocumentation(\n    packageName: string,\n    packageVersion: string\n  ): Promise<boolean> {\n    return new Promise((ok, ko) => {\n      const url = `${this.baseUrl}/data/${packageName}/v${packageVersion}/docs-typescript.md`;\n      https\n        .request(url, { method: 'HEAD' }, (res) => {\n          if (res.statusCode === 200) {\n            // This returns HTTP 200 with text/html if it's a 404, due to how\n            // we configured CloudFront behaviors.\n            return ok(\n              !!res.headers['content-type']?.startsWith('text/markdown')\n            );\n          }\n          const err = new Error(\n            `HEAD ${url} -- HTTP ${res.statusCode} (${res.statusMessage})`\n          );\n          Error.captureStackTrace(err);\n          ko(err);\n        })\n        .end();\n    });\n  }\n\n  private async getCatalog(): Promise<CatalogModel> {\n    if (this.#catalog) {\n      return this.#catalog;\n    }\n    return (this.#catalog = await getJSON(`${this.baseUrl}/catalog.json`));\n  }\n}\n\nexport class CanaryStateService {\n  constructor(private readonly bucketName: string) {}\n\n  /**\n   * Save the state to the bucket.\n   */\n  public async save(packageName: string, state: CanaryState) {\n    const url = this.url(packageName);\n\n    console.log(`Saving to ${url}: ${JSON.stringify(state, null, 2)}`);\n    await aws\n      .s3()\n      .putObject({\n        Bucket: this.bucketName,\n        Key: this.key(packageName),\n        Body: JSON.stringify(state, null, 2),\n        ContentType: 'application/json',\n      })\n      .promise();\n  }\n\n  /**\n   * Load the state file for this package from the bucket.\n   */\n  public async load(packageName: string): Promise<CanaryState | undefined> {\n    console.log(`Loading state for package '${packageName}'`);\n\n    const objectKey = this.key(packageName);\n    const url = this.url(packageName);\n\n    console.log(`Fetching: ${url}`);\n    const data = await aws\n      .s3()\n      .getObject({ Bucket: this.bucketName, Key: objectKey })\n      .promise()\n      .catch((err: AWSError) =>\n        err.code !== 'NoSuchKey'\n          ? Promise.reject(err)\n          : Promise.resolve({\n              /* no data */\n            } as S3.GetObjectOutput)\n      );\n\n    if (!data?.Body) {\n      console.log(`Not found: ${url}`);\n      return undefined;\n    }\n\n    console.log(`Loaded: ${url}`);\n    return JSON.parse(data.Body.toString('utf-8'), (key, value) => {\n      if (key === 'publishedAt' || key === 'availableAt') {\n        return new Date(value);\n      }\n      return value;\n    });\n  }\n\n  /**\n   * Create a state from the latest version of the package.\n   */\n  public async latest(packageName: string): Promise<CanaryState['latest']> {\n    console.log(`Fetching latest version information from NPM: ${packageName}`);\n    const version = await getJSON(\n      `https://registry.npmjs.org/${encodeURIComponent(packageName)}/latest`,\n      ['version']\n    );\n    const publishedAt = await getJSON(\n      `https://registry.npmjs.org/${encodeURIComponent(packageName)}`,\n      ['time', version]\n    );\n\n    console.log(\n      `Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`\n    );\n\n    return { version, publishedAt: new Date(publishedAt) };\n  }\n\n  public async isNpmReplicaDown(): Promise<boolean> {\n    try {\n      await getJSON('https://replicate.npmjs.com/');\n      return false;\n    } catch (e) {\n      return true;\n    }\n  }\n\n  /**\n   * Estimate how far behind the NPM replica is compared to the live NPM\n   * registry. If the NPM replica is down, return undefined.\n   */\n  public async npmReplicaLagSeconds(\n    packageName: string\n  ): Promise<number | undefined> {\n    const encodedPackageName = encodeURIComponent(packageName);\n\n    console.log(`Measuring NPM replica lag using ${packageName}...`);\n\n    const primaryDate = await getModifiedTimestamp(`registry.npmjs.org`);\n\n    let replicaDate;\n    try {\n      replicaDate = await getModifiedTimestamp(`replicate.npmjs.com/registry`);\n    } catch (e) {\n      if (e instanceof Error && e.message.includes('HTTP 504')) {\n        console.log(\n          `Warning: error fetching replicate.npmjs.com: ${e.toString()}`\n        );\n        // There is no value to report\n        return undefined;\n      } else {\n        throw e;\n      }\n    }\n\n    const deltaMs = primaryDate.getTime() - replicaDate.getTime();\n\n    console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);\n    console.log(\n      `Timestamp on replica: ${replicaDate.toISOString()} (${\n        deltaMs / 3_600_000\n      } hours behind)`\n    );\n\n    // We return in seconds... The millisecond resolution is silly here since the probe package is\n    // only published approximately once every three hours. We use seconds only because this is the\n    // largest available time unit in CloudWatch.\n    return deltaMs / 1_000;\n\n    async function getModifiedTimestamp(baseUrl: string) {\n      const isoDate = await getJSON(\n        `https://${baseUrl}/${encodedPackageName}`,\n        ['time', 'modified']\n      );\n      return new Date(isoDate);\n    }\n  }\n\n  private key(packageName: string): string {\n    return `${ObjectKey.STATE_PREFIX}${packageName}${ObjectKey.STATE_SUFFIX}`;\n  }\n\n  private url(packageName: string) {\n    return `s3://${this.bucketName}/${this.key(packageName)}`;\n  }\n}\n\ninterface CanaryState {\n  /**\n   * The latest package version, as of the last execution of the canary.\n   */\n  latest: {\n    /**\n     * The version we are tracking.\n     */\n    readonly version: string;\n\n    /**\n     * The publish date of the version.\n     */\n    readonly publishedAt: Date;\n\n    /**\n     * The date at which the version is available on the hub.\n     */\n    availableAt?: Date;\n  };\n\n  /**\n   * Each existing, but not-yet-found versions that are still tracked.\n   */\n  pending: {\n    [version: string]: {\n      /**\n       * The version we are tracking.\n       */\n      readonly version: string;\n\n      /**\n       * The publish date of the version.\n       */\n      readonly publishedAt: Date;\n\n      /**\n       * These pending packages are NEVER available at this point.\n       */\n      availableAt: undefined;\n    };\n  };\n}\n\n/**\n * Makes a request to the provided URL and returns the response after having\n * parsed it from JSON.\n *\n * @param url the URL to get.\n * @param jsonPath a JSON path to extract only a subset of the object.\n */\nfunction getJSON(url: string, jsonPath?: string[]): Promise<any> {\n  return new Promise((ok, ko) => {\n    https.get(\n      url,\n      {\n        headers: { Accept: 'application/json', 'Accept-Encoding': 'identity' },\n      },\n      (res) => {\n        if (res.statusCode !== 200) {\n          const error = new Error(\n            `GET ${url} - HTTP ${res.statusCode} (${res.statusMessage})`\n          );\n          Error.captureStackTrace(error);\n          return ko(error);\n        }\n\n        res.once('error', ko);\n\n        const json = JSONStream.parse(jsonPath);\n        json.once('data', ok);\n        json.once('error', ko);\n\n        const plainPayload =\n          res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n        plainPayload.pipe(json, { end: true });\n      }\n    );\n  });\n}\n\n/**\n * Updates the `latest` property of `state` ti the provided `latest` value,\n * unless this is already the current latest.\n *\n * If the previous latest version does not have the `availableAt` property, adds\n * that to the `pending` set.\n *\n * @param state  the state to be updated.\n * @param latest the current \"latest\" version of the tracked package.\n */\nfunction updateLatestIfNeeded(\n  state: CanaryState,\n  latest: CanaryState['latest']\n): void {\n  if (state.latest.version === latest.version) {\n    return;\n  }\n\n  // If the current \"latest\" isn't available yet, add it to the `pending` versions.\n  if (state.latest.availableAt == null) {\n    // The TypeScript version of jsii doesn't do control flow analysis well enough here to\n    // determine that the`if` branch guarantees `availableAt` is undefined here.\n    state.pending[state.latest.version] = {\n      ...state.latest,\n      availableAt: undefined,\n    };\n  }\n\n  state.latest = latest;\n}\n\nfunction gunzip(readable: Readable): Readable {\n  const gz = createGunzip();\n  readable.pipe(gz, { end: true });\n  return gz;\n}\n"]}
|