construct-hub 0.4.6 → 0.4.7

Sign up to get free protection for your applications and to get access to all the features.
@@ -30,7 +30,12 @@ export declare class CanaryStateService {
30
30
  * Create a state from the latest version of the package.
31
31
  */
32
32
  latest(packageName: string): Promise<CanaryState['latest']>;
33
- npmReplicaLagSeconds(packageName: string): Promise<number>;
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>;
34
39
  private key;
35
40
  private url;
36
41
  }
@@ -66,11 +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
+ const replicaLag = await stateService.npmReplicaLagSeconds(packageName);
69
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);
73
- metrics.putMetric("EstimatedNpmReplicaLag" /* NPM_REPLICA_LAG */, await stateService.npmReplicaLagSeconds(packageName), aws_embedded_metrics_1.Unit.Seconds);
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
+ }
74
80
  })();
75
81
  for (const versionState of [
76
82
  state.latest,
@@ -236,13 +242,37 @@ class CanaryStateService {
236
242
  console.log(`Package: ${packageName} | Version : ${version} | Published At: ${publishedAt}`);
237
243
  return { version, publishedAt: new Date(publishedAt) };
238
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
+ */
239
258
  async npmReplicaLagSeconds(packageName) {
240
259
  const encodedPackageName = encodeURIComponent(packageName);
241
260
  console.log(`Measuring NPM replica lag using ${packageName}...`);
242
- const [primaryDate, replicaDate] = await Promise.all([
243
- getModifiedTimestamp(`registry.npmjs.org`),
244
- getModifiedTimestamp(`replicate.npmjs.com/registry`),
245
- ]);
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
+ }
246
276
  const deltaMs = primaryDate.getTime() - replicaDate.getTime();
247
277
  console.log(`Timestamp on primary: ${primaryDate.toISOString()}`);
248
278
  console.log(`Timestamp on replica: ${replicaDate.toISOString()} (${deltaMs / 3600000} hours behind)`);
@@ -319,4 +349,4 @@ function gunzip(readable) {
319
349
  readable.pipe(gz, { end: true });
320
350
  return gz;
321
351
  }
322
- //# 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) => 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_LAG,\n        await stateService.npmReplicaLagSeconds(packageName),\n        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 npmReplicaLagSeconds(packageName: string): Promise<number> {\n    const encodedPackageName = encodeURIComponent(packageName);\n\n    console.log(`Measuring NPM replica lag using ${packageName}...`);\n\n    const [primaryDate, replicaDate] = await Promise.all([\n      getModifiedTimestamp(`registry.npmjs.org`),\n      getModifiedTimestamp(`replicate.npmjs.com/registry`),\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"]}
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"]}
@@ -397,7 +397,11 @@ class NpmJs {
397
397
  notRunningAlarm.node.addDependency(schedule);
398
398
  schedule.node.addDependency(failureAlarm);
399
399
  }
400
- registerCanary(scope, packageName, visibilitySla, bucket, constructHubBaseUrl, monitoring) {
400
+ registerCanary(scope, packageName,
401
+ // A duration specifying how long we expect the probe package to appear on
402
+ // Construct Hub after it gets published to npm, assuming the npm replica
403
+ // is up to date etc.
404
+ visibilitySla, bucket, constructHubBaseUrl, monitoring) {
401
405
  const canary = new canary_1.NpmJsPackageCanary(scope, 'Canary', {
402
406
  bucket,
403
407
  constructHubBaseUrl,
@@ -410,7 +414,7 @@ class NpmJs {
410
414
  // nothing that can be done except for waiting until the replica has finally caught up. We
411
415
  // hence suppress the alarm if the replica lag is getting within 3 evaluation periods of the
412
416
  // visbility SLA.
413
- expression: `IF(FILL(mLag, 0) < ${Math.max(visibilitySla.toSeconds() - 3 * period.toSeconds(), 3 * period.toSeconds())}, MAX([mDwell, mTTC]))`,
417
+ expression: `IF(FILL(mLag, REPEAT) < ${Math.max(visibilitySla.toSeconds() - 3 * period.toSeconds(), 3 * period.toSeconds())}, MAX([mDwell, mTTC]))`,
414
418
  period,
415
419
  usingMetrics: {
416
420
  mDwell: canary.metricDwellTime(),
@@ -506,11 +510,11 @@ class NpmJs {
506
510
  }
507
511
  exports.NpmJs = NpmJs;
508
512
  _a = JSII_RTTI_SYMBOL_1;
509
- NpmJs[_a] = { fqn: "construct-hub.sources.NpmJs", version: "0.4.6" };
513
+ NpmJs[_a] = { fqn: "construct-hub.sources.NpmJs", version: "0.4.7" };
510
514
  /**
511
515
  * How often 'rate' goes into 'duration' (rounded up)
512
516
  */
513
517
  function howOften(rate, duration) {
514
518
  return Math.ceil(duration.toSeconds() / rate.toSeconds());
515
519
  }
516
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npmjs.js","sourceRoot":"","sources":["../../src/package-sources/npmjs.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport {\n  AlarmRule,\n  ComparisonOperator,\n  CompositeAlarm,\n  GraphWidget,\n  IWidget,\n  MathExpression,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} 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 { Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { SqsEventSource } from 'aws-cdk-lib/aws-lambda-event-sources';\nimport { BlockPublicAccess, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { Construct } from 'constructs';\nimport { lambdaFunctionUrl, s3ObjectUrl, sqsQueueUrl } from '../deep-link';\nimport { fillMetric } from '../metric-utils';\nimport { IMonitoring } from '../monitoring/api';\nimport type {\n  IPackageSource,\n  PackageSourceBindOptions,\n  PackageSourceBindResult,\n} from '../package-source';\nimport { RUNBOOK_URL } from '../runbook-url';\nimport { S3StorageFactory } from '../s3/storage';\nimport { NpmJsPackageCanary } from './npmjs/canary';\nimport {\n  MARKER_FILE_NAME,\n  METRICS_NAMESPACE,\n  MetricName,\n  S3KeyPrefix,\n} from './npmjs/constants.lambda-shared';\n\nimport { NpmJsFollower } from './npmjs/npm-js-follower';\nimport { StageAndNotify } from './npmjs/stage-and-notify';\n\n/**\n * The periodicity at which the NpmJs follower will run. This MUST be a valid\n * CloudWatch Metric grain, as this will also be the period of the CloudWatch\n * alarm that montiors the health of the follower.\n */\nconst FOLLOWER_RUN_RATE = Duration.minutes(5);\n\n/**\n * Alarm if we haven't seen changes over this time\n *\n * The CouchDB leader occasionally just starts tossing out timeouts and they\n * may last for a good while. We need to be conservative with our alarms\n * otherwise we're just going to be too spammy.\n */\nconst NO_CHANGES_ALARM_DURATION = Duration.hours(1);\n\nexport interface NpmJsProps {\n  /**\n   * The bucket to use for staging npm packages.\n   *\n   * @default - a new bucket will be created.\n   */\n  readonly stagingBucket?: IBucket;\n\n  /**\n   * Registers a package canary, which will track availability of a canary\n   * package in ConstructHub, and emit dedicated metrics.\n   *\n   * @default true\n   */\n  readonly enableCanary?: boolean;\n\n  /**\n   * The package that is monitored by the package canary, if enabled by\n   * `enableCanary`.\n   *\n   * @default 'construct-hub-probe'\n   */\n  readonly canaryPackage?: string;\n\n  /**\n   * The maximum amount of time it is supposed to take for packages to become\n   * visible in this ConstructHub instance. If `enableCanary` is enabled, an\n   * alarm will trigger if this SLA is breached by the `canaryPackage`.\n   *\n   * @default Duration.minutes(5)\n   */\n  readonly canarySla?: Duration;\n}\n\n/**\n * A package source that gets package data from the npmjs.com package registry.\n */\nexport class NpmJs implements IPackageSource {\n  public constructor(private readonly props: NpmJsProps = {}) {}\n\n  public bind(\n    scope: Construct,\n    {\n      baseUrl,\n      denyList,\n      ingestion,\n      licenseList,\n      monitoring,\n      queue,\n      repository,\n      overviewDashboard,\n    }: PackageSourceBindOptions\n  ): PackageSourceBindResult {\n    repository?.addExternalConnection('public:npmjs');\n\n    const storageFactory = S3StorageFactory.getOrCreate(scope);\n    const bucket =\n      this.props.stagingBucket ||\n      storageFactory.newBucket(scope, 'NpmJs/StagingBucket', {\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        enforceSSL: true,\n        lifecycleRules: [\n          {\n            prefix: S3KeyPrefix.STAGED_KEY_PREFIX,\n            expiration: Duration.days(30),\n          },\n        ],\n      });\n    bucket.grantRead(ingestion);\n\n    const stager = new StageAndNotify(scope, 'NpmJs-StageAndNotify', {\n      deadLetterQueue: new Queue(scope, 'StagerDLQ', {\n        encryption: QueueEncryption.KMS_MANAGED,\n        retentionPeriod: Duration.days(14),\n        visibilityTimeout: Duration.minutes(15),\n      }),\n      description: `[${scope.node.path}/NpmJS-StageAndNotify] Stages tarballs to S3 and notifies ConstructHub`,\n      environment: {\n        AWS_EMF_ENVIRONMENT: 'Local',\n        BUCKET_NAME: bucket.bucketName,\n        QUEUE_URL: queue.queueUrl,\n      },\n      memorySize: 10_024, // 10GiB\n      retryAttempts: 2,\n      timeout: Duration.minutes(5),\n      tracing: Tracing.ACTIVE,\n    });\n\n    bucket.grantReadWrite(stager);\n    denyList?.grantRead(stager);\n    queue.grantSendMessages(stager);\n\n    stager.addEventSource(\n      new SqsEventSource(stager.deadLetterQueue!, {\n        batchSize: 1,\n        enabled: false,\n      })\n    );\n\n    const follower = new NpmJsFollower(scope, 'NpmJs', {\n      description: `[${scope.node.path}/NpmJs] Periodically query npmjs.com index for new packages`,\n      environment: {\n        AWS_EMF_ENVIRONMENT: 'Local',\n        BUCKET_NAME: bucket.bucketName,\n        FUNCTION_NAME: stager.functionName,\n      },\n      memorySize: 10_024, // 10 GiB\n      reservedConcurrentExecutions: 1, // Only one execution at a time, to avoid race conditions on the S3 marker object\n      timeout: FOLLOWER_RUN_RATE,\n      tracing: Tracing.ACTIVE,\n    });\n\n    bucket.grantReadWrite(follower, MARKER_FILE_NAME);\n    denyList?.grantRead(follower);\n    licenseList.grantRead(follower);\n    stager.grantInvoke(follower);\n\n    const rule = new Rule(scope, 'NpmJs/Schedule', {\n      description: `${scope.node.path}/NpmJs/Schedule`,\n      schedule: Schedule.rate(FOLLOWER_RUN_RATE),\n      targets: [new LambdaFunction(follower)],\n    });\n\n    this.registerAlarms(scope, follower, stager, monitoring, rule);\n\n    stager.deadLetterQueue &&\n      overviewDashboard.addDLQMetricToDashboard(\n        'NPM JS Stager DLQ',\n        stager.deadLetterQueue\n      );\n    follower.deadLetterQueue &&\n      overviewDashboard.addDLQMetricToDashboard(\n        'NPM JS Follower DLQ',\n        follower.deadLetterQueue\n      );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      follower,\n      'NpmJsLambda'\n    );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      stager,\n      'NpmJs-StageAndNotifyLambda'\n    );\n\n    return {\n      name: follower.node.path,\n      links: [\n        {\n          name: 'NpmJs Follower',\n          url: lambdaFunctionUrl(follower),\n          primary: true,\n        },\n        { name: 'Marker Object', url: s3ObjectUrl(bucket, MARKER_FILE_NAME) },\n        { name: 'Stager', url: lambdaFunctionUrl(stager) },\n        { name: 'Stager DLQ', url: sqsQueueUrl(stager.deadLetterQueue!) },\n      ],\n      dashboardWidgets: [\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Follower Health',\n            left: [\n              fillMetric(follower.metricInvocations({ label: 'Invocations' })),\n              fillMetric(follower.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [this.metricRemainingTime({ label: 'Remaining Time' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(5),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Stager Health',\n            left: [\n              fillMetric(stager.metricInvocations({ label: 'Invocations' })),\n              fillMetric(stager.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [stager.metricDuration({ label: 'Duration' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(5),\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'CouchDB Follower',\n            left: [\n              fillMetric(this.metricChangeCount({ label: 'Change Count' }), 0),\n              fillMetric(\n                this.metricUnprocessableEntity({ label: 'Unprocessable' }),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              fillMetric(\n                this.metricNpmJsChangeAge({ label: 'Lag to npmjs.com' }),\n                'REPEAT'\n              ),\n              fillMetric(\n                this.metricPackageVersionAge({ label: 'Package Version Age' }),\n                'REPEAT'\n              ),\n            ],\n            rightYAxis: { label: 'Milliseconds', min: 0, showUnits: false },\n            period: Duration.minutes(5),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'CouchDB Changes',\n            left: [\n              fillMetric(\n                this.metricLastSeq({ label: 'Last Sequence Number' }),\n                'REPEAT'\n              ),\n            ],\n            period: Duration.minutes(5),\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Stager Dead-Letter Queue',\n            left: [\n              fillMetric(\n                stager.deadLetterQueue!.metricApproximateNumberOfMessagesVisible(\n                  { label: 'Visible Messages' }\n                ),\n                0\n              ),\n              fillMetric(\n                stager.deadLetterQueue!.metricApproximateNumberOfMessagesNotVisible(\n                  { label: 'Invisible Messages' }\n                ),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              stager.deadLetterQueue!.metricApproximateAgeOfOldestMessage({\n                label: 'Oldest Message',\n              }),\n            ],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(1),\n          }),\n          ...(this.props.enableCanary ?? true\n            ? this.registerCanary(\n                follower,\n                this.props.canaryPackage ?? 'construct-hub-probe',\n                this.props.canarySla ?? Duration.minutes(5),\n                bucket,\n                baseUrl,\n                monitoring\n              )\n            : []),\n        ],\n      ],\n    };\n  }\n\n  /**\n   * The average time it took to process a changes batch.\n   */\n  public metricBatchProcessingTime(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.AVERAGE,\n      ...opts,\n      metricName: MetricName.BATCH_PROCESSING_TIME,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of changes that were processed.\n   */\n  public metricChangeCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.CHANGE_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The last sequence number that was processed. This metric can be used to\n   * discover when a sequence reset has happened in the CouchDB instance.\n   */\n  public metricLastSeq(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.LAST_SEQ,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricNpmJsChangeAge(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MINIMUM,\n      ...opts,\n      metricName: MetricName.NPMJS_CHANGE_AGE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The age of the oldest package version that was processed.\n   */\n  public metricPackageVersionAge(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.PACKAGE_VERSION_AGE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of package versions that were inspected.\n   */\n  public metricPackageVersionCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.PACKAGE_VERSION_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of package versions that were deemed relevant.\n   */\n  public metricRelevantPackageVersions(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.RELEVANT_PACKAGE_VERSIONS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The amount of time that was remaining when the lambda returned in order to\n   * avoid hitting a timeout.\n   */\n  public metricRemainingTime(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.MINIMUM,\n      ...opts,\n      metricName: MetricName.REMAINING_TIME,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The amount of changes that were not processed due to having an invalid\n   * format.\n   */\n  public metricUnprocessableEntity(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.UNPROCESSABLE_ENTITY,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  private registerAlarms(\n    scope: Construct,\n    follower: NpmJsFollower,\n    stager: StageAndNotify,\n    monitoring: IMonitoring,\n    schedule: Rule\n  ) {\n    const failureAlarm = follower\n      .metricErrors()\n      .createAlarm(scope, 'NpmJs/Follower/Failures', {\n        alarmName: `${scope.node.path}/NpmJs/Follower/Failures`,\n        alarmDescription: [\n          'The NpmJs follower function failed!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n        ].join('\\n'),\n        comparisonOperator:\n          ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 3,\n        threshold: 1,\n        treatMissingData: TreatMissingData.MISSING,\n      });\n    monitoring.addLowSeverityAlarm('NpmJs/Follower Failures', failureAlarm);\n\n    const notRunningAlarm = follower\n      .metricInvocations({ period: FOLLOWER_RUN_RATE })\n      .createAlarm(scope, 'NpmJs/Follower/NotRunning', {\n        alarmName: `${scope.node.path}/NpmJs/Follower/NotRunning`,\n        alarmDescription: [\n          'The NpmJs follower function is not running!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n        evaluationPeriods: 2,\n        threshold: 1,\n        treatMissingData: TreatMissingData.BREACHING,\n      });\n    monitoring.addHighSeverityAlarm(\n      'NpmJs/Follower Not Running',\n      notRunningAlarm\n    );\n\n    // The period for this alarm needs to match the scheduling interval of the\n    // follower, otherwise the metric will be too sparse to properly detect\n    // problems.\n    const noChangeAlarm = this.metricChangeCount({\n      period: FOLLOWER_RUN_RATE,\n    }).createAlarm(scope, 'NpmJs/Follower/NoChanges', {\n      alarmName: `${scope.node.path}/NpmJs/Follower/NoChanges`,\n      alarmDescription: [\n        'The NpmJs follower function is not discovering any changes from CouchDB!',\n        '',\n        `RunBook: ${RUNBOOK_URL}`,\n        '',\n        `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n      ].join('\\n'),\n      comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: howOften(FOLLOWER_RUN_RATE, NO_CHANGES_ALARM_DURATION),\n      threshold: 1,\n      // If the metric is not emitted, it can be assumed to be zero.\n      treatMissingData: TreatMissingData.BREACHING,\n    });\n    monitoring.addLowSeverityAlarm(\n      'Np npmjs.com changes discovered',\n      noChangeAlarm\n    );\n\n    const dlqNotEmptyAlarm = new MathExpression({\n      expression: 'mVisible + mHidden',\n      usingMetrics: {\n        mVisible:\n          stager.deadLetterQueue!.metricApproximateNumberOfMessagesVisible({\n            period: Duration.minutes(1),\n          }),\n        mHidden:\n          stager.deadLetterQueue!.metricApproximateNumberOfMessagesNotVisible({\n            period: Duration.minutes(1),\n          }),\n      },\n    }).createAlarm(scope, `${scope.node.path}/NpmJs/Stager/DLQNotEmpty`, {\n      alarmName: `${scope.node.path}/NpmJs/Stager/DLQNotEmpty`,\n      alarmDescription: [\n        'The NpmJS package stager is failing - its dead letter queue is not empty',\n        '',\n        `Link to the lambda function: ${lambdaFunctionUrl(stager)}`,\n        `Link to the dead letter queue: ${sqsQueueUrl(\n          stager.deadLetterQueue!\n        )}`,\n        '',\n        `Runbook: ${RUNBOOK_URL}`,\n      ].join('/n'),\n      comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold: 1,\n      treatMissingData: TreatMissingData.NOT_BREACHING,\n    });\n    monitoring.addLowSeverityAlarm(\n      'NpmJs/Stager DLQ Not Empty',\n      dlqNotEmptyAlarm\n    );\n\n    // Finally - the \"not running\" alarm depends on the schedule (it won't run until the schedule\n    // exists!), and the schedule depends on the failure alarm existing (we don't want it to run\n    // before we can know it is failing). This means the returned `IDependable` effectively ensures\n    // all alarms have been provisionned already! Isn't it nice!\n    notRunningAlarm.node.addDependency(schedule);\n    schedule.node.addDependency(failureAlarm);\n  }\n\n  private registerCanary(\n    scope: Construct,\n    packageName: string,\n    visibilitySla: Duration,\n    bucket: IBucket,\n    constructHubBaseUrl: string,\n    monitoring: IMonitoring\n  ): IWidget[] {\n    const canary = new NpmJsPackageCanary(scope, 'Canary', {\n      bucket,\n      constructHubBaseUrl,\n      packageName,\n    });\n\n    const period = Duration.minutes(1);\n    const alarm = new MathExpression({\n      // When the npm replica is sufficiently behind the primary, the package source will not be\n      // able to register new canary package versions within the SLA. In such cases, there is\n      // nothing that can be done except for waiting until the replica has finally caught up. We\n      // hence suppress the alarm if the replica lag is getting within 3 evaluation periods of the\n      // visbility SLA.\n      expression: `IF(FILL(mLag, 0) < ${Math.max(\n        visibilitySla.toSeconds() - 3 * period.toSeconds(),\n        3 * period.toSeconds()\n      )}, MAX([mDwell, mTTC]))`,\n      period,\n      usingMetrics: {\n        mDwell: canary.metricDwellTime(),\n        mTTC: canary.metricTimeToCatalog(),\n        mLag: canary.metricEstimatedNpmReplicaLag(),\n      },\n    }).createAlarm(canary, 'Alarm', {\n      alarmName: `${canary.node.path}/SLA-Breached`,\n      alarmDescription: [\n        `New versions of ${packageName} have been published over ${visibilitySla.toHumanString()} ago and are still not visible in construct hub`,\n        `Runbook: ${RUNBOOK_URL}`,\n      ].join('\\n'),\n      comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 2,\n      treatMissingData: TreatMissingData.NOT_BREACHING,\n      threshold: visibilitySla.toSeconds(),\n    });\n    // This is deemed low severity, because the npm registry replica (replicate.npmjs.com) can\n    // occasionally lag several hours behind the primary (registry.npmjs.com), and we cannot easily\n    // tell about that. Someone should have a look, but in virtually all cases we have seen so far,\n    // there is nothing that can be done from our end, besides waiting for the replica to be all\n    // caught up.\n    monitoring.addLowSeverityAlarm(\n      'New version visibility SLA breached',\n      alarm\n    );\n\n    const notRunningOrFailingAlarm = new CompositeAlarm(\n      canary,\n      'NotRunningOrFailing',\n      {\n        alarmRule: AlarmRule.anyOf(\n          canary\n            .metricErrors({ period, statistic: Statistic.SUM })\n            .createAlarm(canary, 'Failing', {\n              alarmName: `${canary.node.path}/Failing`,\n              comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n              evaluationPeriods: 2,\n              threshold: 0,\n              treatMissingData: TreatMissingData.BREACHING,\n            }),\n          canary\n            .metricInvocations({ period, statistic: Statistic.SUM })\n            .createAlarm(canary, 'NotRunning', {\n              alarmName: `${canary.node.path}/NotRunning`,\n              comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n              evaluationPeriods: 2,\n              threshold: 1,\n              treatMissingData: TreatMissingData.BREACHING,\n            })\n        ),\n        alarmDescription: [\n          'The NpmJs package canary is not running or is failing. This prevents alarming when this instance of',\n          'ConstructHub falls out of SLA for new package ingestion!',\n          '',\n          `Runbook: ${RUNBOOK_URL}`,\n        ].join('\\n'),\n        compositeAlarmName: `${canary.node.path}/NotRunningOrFailing`,\n      }\n    );\n    monitoring.addHighSeverityAlarm(\n      'NpmJs Follower Canary is not running or fails',\n      notRunningOrFailingAlarm\n    );\n\n    return [\n      new GraphWidget({\n        height: 6,\n        width: 12,\n        title: 'Package Canary',\n        left: [\n          canary.metricDwellTime({ label: 'Dwell Time' }),\n          canary.metricTimeToCatalog({ label: 'Time to Catalog' }),\n        ],\n        leftAnnotations: [\n          {\n            color: '#ff0000',\n            label: `SLA (${visibilitySla.toHumanString()})`,\n            value: visibilitySla.toSeconds(),\n          },\n        ],\n        leftYAxis: { min: 0 },\n        right: [\n          canary.metricTrackedVersionCount({ label: 'Tracked Version Count' }),\n        ],\n        rightYAxis: { min: 0 },\n      }),\n      new GraphWidget({\n        height: 6,\n        width: 12,\n        title: 'Observed lag of replicate.npmjs.com',\n        left: [\n          canary.metricEstimatedNpmReplicaLag({\n            label: `Replica lag (${packageName})`,\n          }),\n        ],\n        leftAnnotations: [\n          {\n            color: '#ffa500',\n            label: visibilitySla.toHumanString(),\n            value: visibilitySla.toSeconds(),\n          },\n        ],\n        leftYAxis: { min: 0 },\n      }),\n    ];\n  }\n}\n\n/**\n * How often 'rate' goes into 'duration' (rounded up)\n */\nfunction howOften(rate: Duration, duration: Duration) {\n  return Math.ceil(duration.toSeconds() / rate.toSeconds());\n}\n"]}
520
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npmjs.js","sourceRoot":"","sources":["../../src/package-sources/npmjs.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport {\n  AlarmRule,\n  ComparisonOperator,\n  CompositeAlarm,\n  GraphWidget,\n  IWidget,\n  MathExpression,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} 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 { Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { SqsEventSource } from 'aws-cdk-lib/aws-lambda-event-sources';\nimport { BlockPublicAccess, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { Construct } from 'constructs';\nimport { lambdaFunctionUrl, s3ObjectUrl, sqsQueueUrl } from '../deep-link';\nimport { fillMetric } from '../metric-utils';\nimport { IMonitoring } from '../monitoring/api';\nimport type {\n  IPackageSource,\n  PackageSourceBindOptions,\n  PackageSourceBindResult,\n} from '../package-source';\nimport { RUNBOOK_URL } from '../runbook-url';\nimport { S3StorageFactory } from '../s3/storage';\nimport { NpmJsPackageCanary } from './npmjs/canary';\nimport {\n  MARKER_FILE_NAME,\n  METRICS_NAMESPACE,\n  MetricName,\n  S3KeyPrefix,\n} from './npmjs/constants.lambda-shared';\n\nimport { NpmJsFollower } from './npmjs/npm-js-follower';\nimport { StageAndNotify } from './npmjs/stage-and-notify';\n\n/**\n * The periodicity at which the NpmJs follower will run. This MUST be a valid\n * CloudWatch Metric grain, as this will also be the period of the CloudWatch\n * alarm that montiors the health of the follower.\n */\nconst FOLLOWER_RUN_RATE = Duration.minutes(5);\n\n/**\n * Alarm if we haven't seen changes over this time\n *\n * The CouchDB leader occasionally just starts tossing out timeouts and they\n * may last for a good while. We need to be conservative with our alarms\n * otherwise we're just going to be too spammy.\n */\nconst NO_CHANGES_ALARM_DURATION = Duration.hours(1);\n\nexport interface NpmJsProps {\n  /**\n   * The bucket to use for staging npm packages.\n   *\n   * @default - a new bucket will be created.\n   */\n  readonly stagingBucket?: IBucket;\n\n  /**\n   * Registers a package canary, which will track availability of a canary\n   * package in ConstructHub, and emit dedicated metrics.\n   *\n   * @default true\n   */\n  readonly enableCanary?: boolean;\n\n  /**\n   * The package that is monitored by the package canary, if enabled by\n   * `enableCanary`.\n   *\n   * @default 'construct-hub-probe'\n   */\n  readonly canaryPackage?: string;\n\n  /**\n   * The maximum amount of time it is supposed to take for packages to become\n   * visible in this ConstructHub instance. If `enableCanary` is enabled, an\n   * alarm will trigger if this SLA is breached by the `canaryPackage`.\n   *\n   * @default Duration.minutes(5)\n   */\n  readonly canarySla?: Duration;\n}\n\n/**\n * A package source that gets package data from the npmjs.com package registry.\n */\nexport class NpmJs implements IPackageSource {\n  public constructor(private readonly props: NpmJsProps = {}) {}\n\n  public bind(\n    scope: Construct,\n    {\n      baseUrl,\n      denyList,\n      ingestion,\n      licenseList,\n      monitoring,\n      queue,\n      repository,\n      overviewDashboard,\n    }: PackageSourceBindOptions\n  ): PackageSourceBindResult {\n    repository?.addExternalConnection('public:npmjs');\n\n    const storageFactory = S3StorageFactory.getOrCreate(scope);\n    const bucket =\n      this.props.stagingBucket ||\n      storageFactory.newBucket(scope, 'NpmJs/StagingBucket', {\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        enforceSSL: true,\n        lifecycleRules: [\n          {\n            prefix: S3KeyPrefix.STAGED_KEY_PREFIX,\n            expiration: Duration.days(30),\n          },\n        ],\n      });\n    bucket.grantRead(ingestion);\n\n    const stager = new StageAndNotify(scope, 'NpmJs-StageAndNotify', {\n      deadLetterQueue: new Queue(scope, 'StagerDLQ', {\n        encryption: QueueEncryption.KMS_MANAGED,\n        retentionPeriod: Duration.days(14),\n        visibilityTimeout: Duration.minutes(15),\n      }),\n      description: `[${scope.node.path}/NpmJS-StageAndNotify] Stages tarballs to S3 and notifies ConstructHub`,\n      environment: {\n        AWS_EMF_ENVIRONMENT: 'Local',\n        BUCKET_NAME: bucket.bucketName,\n        QUEUE_URL: queue.queueUrl,\n      },\n      memorySize: 10_024, // 10GiB\n      retryAttempts: 2,\n      timeout: Duration.minutes(5),\n      tracing: Tracing.ACTIVE,\n    });\n\n    bucket.grantReadWrite(stager);\n    denyList?.grantRead(stager);\n    queue.grantSendMessages(stager);\n\n    stager.addEventSource(\n      new SqsEventSource(stager.deadLetterQueue!, {\n        batchSize: 1,\n        enabled: false,\n      })\n    );\n\n    const follower = new NpmJsFollower(scope, 'NpmJs', {\n      description: `[${scope.node.path}/NpmJs] Periodically query npmjs.com index for new packages`,\n      environment: {\n        AWS_EMF_ENVIRONMENT: 'Local',\n        BUCKET_NAME: bucket.bucketName,\n        FUNCTION_NAME: stager.functionName,\n      },\n      memorySize: 10_024, // 10 GiB\n      reservedConcurrentExecutions: 1, // Only one execution at a time, to avoid race conditions on the S3 marker object\n      timeout: FOLLOWER_RUN_RATE,\n      tracing: Tracing.ACTIVE,\n    });\n\n    bucket.grantReadWrite(follower, MARKER_FILE_NAME);\n    denyList?.grantRead(follower);\n    licenseList.grantRead(follower);\n    stager.grantInvoke(follower);\n\n    const rule = new Rule(scope, 'NpmJs/Schedule', {\n      description: `${scope.node.path}/NpmJs/Schedule`,\n      schedule: Schedule.rate(FOLLOWER_RUN_RATE),\n      targets: [new LambdaFunction(follower)],\n    });\n\n    this.registerAlarms(scope, follower, stager, monitoring, rule);\n\n    stager.deadLetterQueue &&\n      overviewDashboard.addDLQMetricToDashboard(\n        'NPM JS Stager DLQ',\n        stager.deadLetterQueue\n      );\n    follower.deadLetterQueue &&\n      overviewDashboard.addDLQMetricToDashboard(\n        'NPM JS Follower DLQ',\n        follower.deadLetterQueue\n      );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      follower,\n      'NpmJsLambda'\n    );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      stager,\n      'NpmJs-StageAndNotifyLambda'\n    );\n\n    return {\n      name: follower.node.path,\n      links: [\n        {\n          name: 'NpmJs Follower',\n          url: lambdaFunctionUrl(follower),\n          primary: true,\n        },\n        { name: 'Marker Object', url: s3ObjectUrl(bucket, MARKER_FILE_NAME) },\n        { name: 'Stager', url: lambdaFunctionUrl(stager) },\n        { name: 'Stager DLQ', url: sqsQueueUrl(stager.deadLetterQueue!) },\n      ],\n      dashboardWidgets: [\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Follower Health',\n            left: [\n              fillMetric(follower.metricInvocations({ label: 'Invocations' })),\n              fillMetric(follower.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [this.metricRemainingTime({ label: 'Remaining Time' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(5),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Stager Health',\n            left: [\n              fillMetric(stager.metricInvocations({ label: 'Invocations' })),\n              fillMetric(stager.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [stager.metricDuration({ label: 'Duration' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(5),\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'CouchDB Follower',\n            left: [\n              fillMetric(this.metricChangeCount({ label: 'Change Count' }), 0),\n              fillMetric(\n                this.metricUnprocessableEntity({ label: 'Unprocessable' }),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              fillMetric(\n                this.metricNpmJsChangeAge({ label: 'Lag to npmjs.com' }),\n                'REPEAT'\n              ),\n              fillMetric(\n                this.metricPackageVersionAge({ label: 'Package Version Age' }),\n                'REPEAT'\n              ),\n            ],\n            rightYAxis: { label: 'Milliseconds', min: 0, showUnits: false },\n            period: Duration.minutes(5),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'CouchDB Changes',\n            left: [\n              fillMetric(\n                this.metricLastSeq({ label: 'Last Sequence Number' }),\n                'REPEAT'\n              ),\n            ],\n            period: Duration.minutes(5),\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Stager Dead-Letter Queue',\n            left: [\n              fillMetric(\n                stager.deadLetterQueue!.metricApproximateNumberOfMessagesVisible(\n                  { label: 'Visible Messages' }\n                ),\n                0\n              ),\n              fillMetric(\n                stager.deadLetterQueue!.metricApproximateNumberOfMessagesNotVisible(\n                  { label: 'Invisible Messages' }\n                ),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              stager.deadLetterQueue!.metricApproximateAgeOfOldestMessage({\n                label: 'Oldest Message',\n              }),\n            ],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(1),\n          }),\n          ...(this.props.enableCanary ?? true\n            ? this.registerCanary(\n                follower,\n                this.props.canaryPackage ?? 'construct-hub-probe',\n                this.props.canarySla ?? Duration.minutes(5),\n                bucket,\n                baseUrl,\n                monitoring\n              )\n            : []),\n        ],\n      ],\n    };\n  }\n\n  /**\n   * The average time it took to process a changes batch.\n   */\n  public metricBatchProcessingTime(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.AVERAGE,\n      ...opts,\n      metricName: MetricName.BATCH_PROCESSING_TIME,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of changes that were processed.\n   */\n  public metricChangeCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.CHANGE_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The last sequence number that was processed. This metric can be used to\n   * discover when a sequence reset has happened in the CouchDB instance.\n   */\n  public metricLastSeq(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.LAST_SEQ,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricNpmJsChangeAge(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MINIMUM,\n      ...opts,\n      metricName: MetricName.NPMJS_CHANGE_AGE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The age of the oldest package version that was processed.\n   */\n  public metricPackageVersionAge(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.MAXIMUM,\n      ...opts,\n      metricName: MetricName.PACKAGE_VERSION_AGE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of package versions that were inspected.\n   */\n  public metricPackageVersionCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.PACKAGE_VERSION_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The total count of package versions that were deemed relevant.\n   */\n  public metricRelevantPackageVersions(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.RELEVANT_PACKAGE_VERSIONS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The amount of time that was remaining when the lambda returned in order to\n   * avoid hitting a timeout.\n   */\n  public metricRemainingTime(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.MINIMUM,\n      ...opts,\n      metricName: MetricName.REMAINING_TIME,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The amount of changes that were not processed due to having an invalid\n   * format.\n   */\n  public metricUnprocessableEntity(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(1),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.UNPROCESSABLE_ENTITY,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  private registerAlarms(\n    scope: Construct,\n    follower: NpmJsFollower,\n    stager: StageAndNotify,\n    monitoring: IMonitoring,\n    schedule: Rule\n  ) {\n    const failureAlarm = follower\n      .metricErrors()\n      .createAlarm(scope, 'NpmJs/Follower/Failures', {\n        alarmName: `${scope.node.path}/NpmJs/Follower/Failures`,\n        alarmDescription: [\n          'The NpmJs follower function failed!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n        ].join('\\n'),\n        comparisonOperator:\n          ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 3,\n        threshold: 1,\n        treatMissingData: TreatMissingData.MISSING,\n      });\n    monitoring.addLowSeverityAlarm('NpmJs/Follower Failures', failureAlarm);\n\n    const notRunningAlarm = follower\n      .metricInvocations({ period: FOLLOWER_RUN_RATE })\n      .createAlarm(scope, 'NpmJs/Follower/NotRunning', {\n        alarmName: `${scope.node.path}/NpmJs/Follower/NotRunning`,\n        alarmDescription: [\n          'The NpmJs follower function is not running!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n        evaluationPeriods: 2,\n        threshold: 1,\n        treatMissingData: TreatMissingData.BREACHING,\n      });\n    monitoring.addHighSeverityAlarm(\n      'NpmJs/Follower Not Running',\n      notRunningAlarm\n    );\n\n    // The period for this alarm needs to match the scheduling interval of the\n    // follower, otherwise the metric will be too sparse to properly detect\n    // problems.\n    const noChangeAlarm = this.metricChangeCount({\n      period: FOLLOWER_RUN_RATE,\n    }).createAlarm(scope, 'NpmJs/Follower/NoChanges', {\n      alarmName: `${scope.node.path}/NpmJs/Follower/NoChanges`,\n      alarmDescription: [\n        'The NpmJs follower function is not discovering any changes from CouchDB!',\n        '',\n        `RunBook: ${RUNBOOK_URL}`,\n        '',\n        `Direct link to Lambda function: ${lambdaFunctionUrl(follower)}`,\n      ].join('\\n'),\n      comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: howOften(FOLLOWER_RUN_RATE, NO_CHANGES_ALARM_DURATION),\n      threshold: 1,\n      // If the metric is not emitted, it can be assumed to be zero.\n      treatMissingData: TreatMissingData.BREACHING,\n    });\n    monitoring.addLowSeverityAlarm(\n      'Np npmjs.com changes discovered',\n      noChangeAlarm\n    );\n\n    const dlqNotEmptyAlarm = new MathExpression({\n      expression: 'mVisible + mHidden',\n      usingMetrics: {\n        mVisible:\n          stager.deadLetterQueue!.metricApproximateNumberOfMessagesVisible({\n            period: Duration.minutes(1),\n          }),\n        mHidden:\n          stager.deadLetterQueue!.metricApproximateNumberOfMessagesNotVisible({\n            period: Duration.minutes(1),\n          }),\n      },\n    }).createAlarm(scope, `${scope.node.path}/NpmJs/Stager/DLQNotEmpty`, {\n      alarmName: `${scope.node.path}/NpmJs/Stager/DLQNotEmpty`,\n      alarmDescription: [\n        'The NpmJS package stager is failing - its dead letter queue is not empty',\n        '',\n        `Link to the lambda function: ${lambdaFunctionUrl(stager)}`,\n        `Link to the dead letter queue: ${sqsQueueUrl(\n          stager.deadLetterQueue!\n        )}`,\n        '',\n        `Runbook: ${RUNBOOK_URL}`,\n      ].join('/n'),\n      comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold: 1,\n      treatMissingData: TreatMissingData.NOT_BREACHING,\n    });\n    monitoring.addLowSeverityAlarm(\n      'NpmJs/Stager DLQ Not Empty',\n      dlqNotEmptyAlarm\n    );\n\n    // Finally - the \"not running\" alarm depends on the schedule (it won't run until the schedule\n    // exists!), and the schedule depends on the failure alarm existing (we don't want it to run\n    // before we can know it is failing). This means the returned `IDependable` effectively ensures\n    // all alarms have been provisionned already! Isn't it nice!\n    notRunningAlarm.node.addDependency(schedule);\n    schedule.node.addDependency(failureAlarm);\n  }\n\n  private registerCanary(\n    scope: Construct,\n    packageName: string,\n    // A duration specifying how long we expect the probe package to appear on\n    // Construct Hub after it gets published to npm, assuming the npm replica\n    // is up to date etc.\n    visibilitySla: Duration,\n    bucket: IBucket,\n    constructHubBaseUrl: string,\n    monitoring: IMonitoring\n  ): IWidget[] {\n    const canary = new NpmJsPackageCanary(scope, 'Canary', {\n      bucket,\n      constructHubBaseUrl,\n      packageName,\n    });\n\n    const period = Duration.minutes(1);\n\n    const alarm = new MathExpression({\n      // When the npm replica is sufficiently behind the primary, the package source will not be\n      // able to register new canary package versions within the SLA. In such cases, there is\n      // nothing that can be done except for waiting until the replica has finally caught up. We\n      // hence suppress the alarm if the replica lag is getting within 3 evaluation periods of the\n      // visbility SLA.\n      expression: `IF(FILL(mLag, REPEAT) < ${Math.max(\n        visibilitySla.toSeconds() - 3 * period.toSeconds(),\n        3 * period.toSeconds()\n      )}, MAX([mDwell, mTTC]))`,\n      period,\n      usingMetrics: {\n        mDwell: canary.metricDwellTime(),\n        mTTC: canary.metricTimeToCatalog(),\n        mLag: canary.metricEstimatedNpmReplicaLag(),\n      },\n    }).createAlarm(canary, 'Alarm', {\n      alarmName: `${canary.node.path}/SLA-Breached`,\n      alarmDescription: [\n        `New versions of ${packageName} have been published over ${visibilitySla.toHumanString()} ago and are still not visible in construct hub`,\n        `Runbook: ${RUNBOOK_URL}`,\n      ].join('\\n'),\n      comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: 2,\n      treatMissingData: TreatMissingData.NOT_BREACHING,\n      threshold: visibilitySla.toSeconds(),\n    });\n    // This is deemed low severity, because the npm registry replica (replicate.npmjs.com) can\n    // occasionally lag several hours behind the primary (registry.npmjs.com), and we cannot easily\n    // tell about that. Someone should have a look, but in virtually all cases we have seen so far,\n    // there is nothing that can be done from our end, besides waiting for the replica to be all\n    // caught up.\n    monitoring.addLowSeverityAlarm(\n      'New version visibility SLA breached',\n      alarm\n    );\n\n    const notRunningOrFailingAlarm = new CompositeAlarm(\n      canary,\n      'NotRunningOrFailing',\n      {\n        alarmRule: AlarmRule.anyOf(\n          canary\n            .metricErrors({ period, statistic: Statistic.SUM })\n            .createAlarm(canary, 'Failing', {\n              alarmName: `${canary.node.path}/Failing`,\n              comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n              evaluationPeriods: 2,\n              threshold: 0,\n              treatMissingData: TreatMissingData.BREACHING,\n            }),\n          canary\n            .metricInvocations({ period, statistic: Statistic.SUM })\n            .createAlarm(canary, 'NotRunning', {\n              alarmName: `${canary.node.path}/NotRunning`,\n              comparisonOperator: ComparisonOperator.LESS_THAN_THRESHOLD,\n              evaluationPeriods: 2,\n              threshold: 1,\n              treatMissingData: TreatMissingData.BREACHING,\n            })\n        ),\n        alarmDescription: [\n          'The NpmJs package canary is not running or is failing. This prevents alarming when this instance of',\n          'ConstructHub falls out of SLA for new package ingestion!',\n          '',\n          `Runbook: ${RUNBOOK_URL}`,\n        ].join('\\n'),\n        compositeAlarmName: `${canary.node.path}/NotRunningOrFailing`,\n      }\n    );\n    monitoring.addHighSeverityAlarm(\n      'NpmJs Follower Canary is not running or fails',\n      notRunningOrFailingAlarm\n    );\n\n    return [\n      new GraphWidget({\n        height: 6,\n        width: 12,\n        title: 'Package Canary',\n        left: [\n          canary.metricDwellTime({ label: 'Dwell Time' }),\n          canary.metricTimeToCatalog({ label: 'Time to Catalog' }),\n        ],\n        leftAnnotations: [\n          {\n            color: '#ff0000',\n            label: `SLA (${visibilitySla.toHumanString()})`,\n            value: visibilitySla.toSeconds(),\n          },\n        ],\n        leftYAxis: { min: 0 },\n        right: [\n          canary.metricTrackedVersionCount({ label: 'Tracked Version Count' }),\n        ],\n        rightYAxis: { min: 0 },\n      }),\n      new GraphWidget({\n        height: 6,\n        width: 12,\n        title: 'Observed lag of replicate.npmjs.com',\n        left: [\n          canary.metricEstimatedNpmReplicaLag({\n            label: `Replica lag (${packageName})`,\n          }),\n        ],\n        leftAnnotations: [\n          {\n            color: '#ffa500',\n            label: visibilitySla.toHumanString(),\n            value: visibilitySla.toSeconds(),\n          },\n        ],\n        leftYAxis: { min: 0 },\n      }),\n    ];\n  }\n}\n\n/**\n * How often 'rate' goes into 'duration' (rounded up)\n */\nfunction howOften(rate: Duration, duration: Duration) {\n  return Math.ceil(duration.toSeconds() / rate.toSeconds());\n}\n"]}
@@ -50,7 +50,7 @@ class TagCondition {
50
50
  }
51
51
  exports.TagCondition = TagCondition;
52
52
  _a = JSII_RTTI_SYMBOL_1;
53
- TagCondition[_a] = { fqn: "construct-hub.TagCondition", version: "0.4.6" };
53
+ TagCondition[_a] = { fqn: "construct-hub.TagCondition", version: "0.4.7" };
54
54
  /**
55
55
  * Logic operators for performing specific conditional logic.
56
56
  */
@@ -130,7 +130,7 @@ class TagConditionField {
130
130
  }
131
131
  exports.TagConditionField = TagConditionField;
132
132
  _b = JSII_RTTI_SYMBOL_1;
133
- TagConditionField[_b] = { fqn: "construct-hub.TagConditionField", version: "0.4.6" };
133
+ TagConditionField[_b] = { fqn: "construct-hub.TagConditionField", version: "0.4.7" };
134
134
  /**
135
135
  * Target the README of the package to dictate whether a tag is relevant.
136
136
  */
@@ -147,5 +147,5 @@ class TagConditionReadme {
147
147
  }
148
148
  exports.TagConditionReadme = TagConditionReadme;
149
149
  _c = JSII_RTTI_SYMBOL_1;
150
- TagConditionReadme[_c] = { fqn: "construct-hub.TagConditionReadme", version: "0.4.6" };
150
+ TagConditionReadme[_c] = { fqn: "construct-hub.TagConditionReadme", version: "0.4.7" };
151
151
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/package-tag/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { PackageTagGroup } from '../package-tag-group';\n\ninterface PackageTagPresentationBase {\n  /**\n   * The label for the tag being applied\n   */\n  readonly label: string;\n\n  /**\n   * The hex value string for the color of the tag when displayed\n   */\n  readonly color?: string;\n}\n\nexport interface Keyword extends PackageTagPresentationBase {}\n\nexport interface Highlight extends PackageTagPresentationBase {\n  /**\n   * Icon displayed next to highlight on package card\n   */\n  readonly icon?: string;\n}\n\nexport interface SearchFilter {\n  /**\n   * Display name for filter\n   */\n  readonly display: string;\n\n  /**\n   * Name of group to include filter in\n   * @deprecated use `group` instead\n   */\n  readonly groupBy?: string;\n\n  /**\n   * PackageTagGroup to include filter in\n   */\n  readonly group?: PackageTagGroup;\n}\n\nexport interface PackageTagBase {\n  /**\n   * Identifier for tag, used for search. Must be unique amongst tags.\n   */\n  readonly id: string;\n\n  /**\n   * Configuration for higlighting tag on package card\n   * @default don't highlight tag\n   */\n  readonly highlight?: Highlight;\n\n  /**\n   * Configuration for showing tag as keyword\n   * @default don't show tag in keyword list\n   */\n  readonly keyword?: Keyword;\n\n  /**\n   * Configuration for showing tag as search filter\n   * @default don't show tag in search filters\n   */\n  readonly searchFilter?: SearchFilter;\n}\n\n/**\n * Configuration for applying custom tags to relevant packages. Custom tags are\n * displayed on the package details page, and can be used for searching.\n */\nexport interface PackageTag extends PackageTagBase {\n  /**\n   * The description of the logic that dictates whether the\n   * package has the tag applied.\n   */\n  readonly condition: TagCondition;\n}\n\nexport enum TagConditionSource {\n  PACKAGE_JSON = 'PACKAGE_JSON',\n  README = 'README',\n}\n\n/**\n * Serialized config for a tag condition\n */\nexport interface TagConditionConfig {\n  readonly type: TagConditionLogicType;\n  readonly source?: TagConditionSource;\n  readonly key?: string[];\n  readonly value?: string;\n  readonly options?: { readonly [key: string]: any };\n  readonly children?: TagConditionConfig[];\n}\n\n/**\n * Serialized tag declaration to be passed to lambdas via environment\n * variables.\n */\nexport interface PackageTagConfig extends PackageTagBase {\n  readonly condition: TagConditionConfig;\n}\n\n/**\n * Condition for applying a custom tag to a package.\n */\nexport abstract class TagCondition {\n  /**\n   * Create an && condition which applies only when all condition arguments are\n   * true.\n   */\n  static and(...conds: TagCondition[]): TagCondition {\n    return new TagConditionLogic(TagConditionLogicType.AND, conds);\n  }\n\n  /**\n   * Create an || condition which applies if any of the condition arguments are\n   * true.\n   */\n  static or(...conds: TagCondition[]): TagCondition {\n    return new TagConditionLogic(TagConditionLogicType.OR, conds);\n  }\n\n  /**\n   * Create a ! condition which applies if the condition argument is false\n   */\n  static not(...conds: TagCondition[]): TagCondition {\n    return new TagConditionLogic(TagConditionLogicType.NOT, conds);\n  }\n\n  /**\n   * Target a field within the `package.json` to assert against. Nested fields\n   * can be accessed by passing multiple keys.\n   * `TagCondition.field('key1', 'key2')` will access\n   * `packageJson?.key1?.key2`.\n   */\n  static field(...keys: string[]): TagConditionField {\n    return new TagConditionField(keys);\n  }\n\n  /**\n   * Create a condition with logic targeting the README of the package.\n   */\n  static readme(): TagConditionReadme {\n    return new TagConditionReadme();\n  }\n\n  public abstract bind(): TagConditionConfig;\n}\n\n/**\n * Logic operators for performing specific conditional logic.\n */\nexport enum TagConditionLogicType {\n  AND = 'AND',\n  OR = 'OR',\n  NOT = 'NOT',\n  EQUALS = 'EQUALS',\n  INCLUDES = 'INCLUDES',\n  STARTS_WITH = 'STARTS_WITH',\n}\n\nclass TagConditionLogic extends TagCondition {\n  public readonly isLogic = true;\n  public constructor(\n    private readonly type: TagConditionLogicType,\n    private readonly children: TagCondition[]\n  ) {\n    super();\n  }\n\n  public bind(): TagConditionConfig {\n    return {\n      type: this.type,\n      children: this.children.map((cond) => cond.bind()),\n    };\n  }\n}\n\nclass TagConditionPredicate extends TagCondition {\n  public readonly isPredicate = true;\n  public constructor(\n    private readonly type: TagConditionLogicType,\n    private readonly source?: TagConditionSource,\n    private readonly key?: string[],\n    private readonly value?: string,\n    private readonly options?: TagConditionIncludesOptions\n  ) {\n    super();\n  }\n\n  public bind(): TagConditionConfig {\n    return {\n      type: this.type,\n      source: this.source,\n      key: this.key,\n      value: this.value,\n      options: this.options,\n    };\n  }\n}\n\n/**\n * Target a field to use in logic to dictate whether a tag is relevant.\n */\nexport class TagConditionField {\n  public constructor(private readonly field: string[]) {}\n\n  /**\n   * Create a === condition which applies if the specified field within the\n   * package's package.json is equal to the passed value.\n   */\n  public eq(value: any): TagCondition {\n    return new TagConditionPredicate(\n      TagConditionLogicType.EQUALS,\n      TagConditionSource.PACKAGE_JSON,\n      this.field,\n      value\n    );\n  }\n\n  /**\n   * Create a `field.includes(value)` condition which applies if the specified\n   * field within the package's package.json includes the value. This works for\n   * arrays or strings.\n   */\n  public includes(\n    value: any,\n    options: TagConditionIncludesOptions = {}\n  ): TagCondition {\n    return new TagConditionPredicate(\n      TagConditionLogicType.INCLUDES,\n      TagConditionSource.PACKAGE_JSON,\n      this.field,\n      value,\n      options\n    );\n  }\n\n  /**\n   * Create a `field.startsWith(value)` condition which applies if the specified\n   * field within the package's package.json begins with the value. This works\n   * only for string values.\n   */\n  public startsWith(value: string): TagCondition {\n    return new TagConditionPredicate(\n      TagConditionLogicType.STARTS_WITH,\n      TagConditionSource.PACKAGE_JSON,\n      this.field,\n      value\n    );\n  }\n}\n\n/**\n * Target the README of the package to dictate whether a tag is relevant.\n */\nexport class TagConditionReadme {\n  public constructor() {}\n\n  /**\n   * Create a `readme.includes(value)` condition which applies if the README\n   * includes the specified string.\n   */\n  public includes(\n    value: string,\n    options: TagConditionIncludesOptions = {}\n  ): TagCondition {\n    return new TagConditionPredicate(\n      TagConditionLogicType.INCLUDES,\n      TagConditionSource.README,\n      undefined, // no key\n      value,\n      options\n    );\n  }\n}\n\n/**\n * Options for `includes` operator.\n */\nexport interface TagConditionIncludesOptions {\n  /**\n   * The value must appear at least this many times.\n   * @default 1\n   */\n  readonly atLeast?: number;\n\n  /**\n   * String matches must match the casing of the original string. This option\n   * is ignored if the value we are checking is an array.\n   * @default false\n   */\n  readonly caseSensitive?: boolean;\n}\n"]}
@@ -13,7 +13,7 @@ class FilterType {
13
13
  }
14
14
  exports.FilterType = FilterType;
15
15
  _a = JSII_RTTI_SYMBOL_1;
16
- FilterType[_a] = { fqn: "construct-hub.FilterType", version: "0.4.6" };
16
+ FilterType[_a] = { fqn: "construct-hub.FilterType", version: "0.4.7" };
17
17
  class FilterTypeRadio extends FilterType {
18
18
  bind() {
19
19
  return { type: 'radio' };
@@ -46,5 +46,5 @@ class PackageTagGroup {
46
46
  }
47
47
  exports.PackageTagGroup = PackageTagGroup;
48
48
  _b = JSII_RTTI_SYMBOL_1;
49
- PackageTagGroup[_b] = { fqn: "construct-hub.PackageTagGroup", version: "0.4.6" };
49
+ PackageTagGroup[_b] = { fqn: "construct-hub.PackageTagGroup", version: "0.4.7" };
50
50
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGFja2FnZS10YWctZ3JvdXAvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFJQSxNQUFzQixVQUFVO0lBQzlCLE1BQU0sQ0FBQyxRQUFRO1FBQ2IsT0FBTyxJQUFJLGtCQUFrQixFQUFFLENBQUM7SUFDbEMsQ0FBQztJQUVELE1BQU0sQ0FBQyxLQUFLO1FBQ1YsT0FBTyxJQUFJLGVBQWUsRUFBRSxDQUFDO0lBQy9CLENBQUM7O0FBUEgsZ0NBVUM7OztBQUVELE1BQU0sZUFBZ0IsU0FBUSxVQUFVO0lBQy9CLElBQUk7UUFDVCxPQUFPLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxDQUFDO0lBQzNCLENBQUM7Q0FDRjtBQUVELE1BQU0sa0JBQW1CLFNBQVEsVUFBVTtJQUNsQyxJQUFJO1FBQ1QsT0FBTyxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsQ0FBQztJQUM5QixDQUFDO0NBQ0Y7QUF3QkQ7O0dBRUc7QUFDSCxNQUFhLGVBQWU7SUFLMUIsWUFBNEIsRUFBVSxFQUFFLEtBQTRCOztRQUF4QyxPQUFFLEdBQUYsRUFBRSxDQUFRO1FBQ3BDLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLEtBQUssQ0FBQztRQUMxQixJQUFJLENBQUMsT0FBTyxHQUFHLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxPQUFPLENBQUM7UUFDOUIsSUFBSSxDQUFDLFVBQVUsR0FBRyxPQUFDLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxVQUFVLG1DQUFJLFVBQVUsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLElBQUksQ0FBQztJQUM3RSxDQUFDO0lBRU0sSUFBSTtRQUNULE9BQU87WUFDTCxFQUFFLEVBQUUsSUFBSSxDQUFDLEVBQUU7WUFDWCxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7WUFDakIsT0FBTyxFQUFFLElBQUksQ0FBQyxPQUFPO1lBQ3JCLFVBQVUsRUFBRSxJQUFJLENBQUMsVUFBVTtTQUM1QixDQUFDO0lBQ0osQ0FBQzs7QUFsQkgsMENBbUJDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBGaWx0ZXJUeXBlVmFsdWUge1xuICByZWFkb25seSB0eXBlOiAnY2hlY2tib3gnIHwgJ3JhZGlvJztcbn1cblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIEZpbHRlclR5cGUge1xuICBzdGF0aWMgY2hlY2tib3goKTogRmlsdGVyVHlwZSB7XG4gICAgcmV0dXJuIG5ldyBGaWx0ZXJUeXBlQ2hlY2tib3goKTtcbiAgfVxuXG4gIHN0YXRpYyByYWRpbygpOiBGaWx0ZXJUeXBlIHtcbiAgICByZXR1cm4gbmV3IEZpbHRlclR5cGVSYWRpbygpO1xuICB9XG5cbiAgcHVibGljIGFic3RyYWN0IGJpbmQoKTogRmlsdGVyVHlwZVZhbHVlO1xufVxuXG5jbGFzcyBGaWx0ZXJUeXBlUmFkaW8gZXh0ZW5kcyBGaWx0ZXJUeXBlIHtcbiAgcHVibGljIGJpbmQoKTogRmlsdGVyVHlwZVZhbHVlIHtcbiAgICByZXR1cm4geyB0eXBlOiAncmFkaW8nIH07XG4gIH1cbn1cblxuY2xhc3MgRmlsdGVyVHlwZUNoZWNrYm94IGV4dGVuZHMgRmlsdGVyVHlwZSB7XG4gIHB1YmxpYyBiaW5kKCk6IEZpbHRlclR5cGVWYWx1ZSB7XG4gICAgcmV0dXJuIHsgdHlwZTogJ2NoZWNrYm94JyB9O1xuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFja2FnZVRhZ0dyb3VwUHJvcHMge1xuICAvKipcbiAgICogR3JvdXAgbGFiZWwgdG8gZGlzcGxheS4gRmFsbHMgYmFjayB0byBpZCBpZiBub3QgcHJvdmlkZWRcbiAgICovXG4gIHJlYWRvbmx5IGxhYmVsPzogc3RyaW5nO1xuICAvKipcbiAgICogT3B0aW9uYWwgbWVzc2FnZSB0byBzaG93IHdpdGhpbiBhIHRvb2x0aXAgbmV4dCB0byB0aGUgZmlsdGVyIGxhYmVsXG4gICAqL1xuICByZWFkb25seSB0b29sdGlwPzogc3RyaW5nO1xuICAvKipcbiAgICogQWxsb3dzIHRvIHNwZWNpZnkgdGhlIGdyb3VwIGZpbHRlciB0eXBlLiBEZWZhdWx0cyB0byBjaGVja2JveCBpZiBub3Qgc3BlY2lmaWVkXG4gICAqL1xuICByZWFkb25seSBmaWx0ZXJUeXBlPzogRmlsdGVyVHlwZTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQYWNrYWdlVGFnR3JvdXBDb25maWcge1xuICByZWFkb25seSBpZDogc3RyaW5nO1xuICByZWFkb25seSBsYWJlbD86IHN0cmluZztcbiAgcmVhZG9ubHkgdG9vbHRpcD86IHN0cmluZztcbiAgcmVhZG9ubHkgZmlsdGVyVHlwZT86IHN0cmluZztcbn1cblxuLyoqXG4gKiBEZWZpbmVzIGEgY3VzdG9tIHBhY2thZ2UgdGFnIGdyb3VwXG4gKi9cbmV4cG9ydCBjbGFzcyBQYWNrYWdlVGFnR3JvdXAge1xuICBwdWJsaWMgcmVhZG9ubHkgbGFiZWw/OiBzdHJpbmc7XG4gIHB1YmxpYyByZWFkb25seSB0b29sdGlwPzogc3RyaW5nO1xuICBwdWJsaWMgcmVhZG9ubHkgZmlsdGVyVHlwZT86IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgaWQ6IHN0cmluZywgcHJvcHM/OiBQYWNrYWdlVGFnR3JvdXBQcm9wcykge1xuICAgIHRoaXMubGFiZWwgPSBwcm9wcz8ubGFiZWw7XG4gICAgdGhpcy50b29sdGlwID0gcHJvcHM/LnRvb2x0aXA7XG4gICAgdGhpcy5maWx0ZXJUeXBlID0gKHByb3BzPy5maWx0ZXJUeXBlID8/IEZpbHRlclR5cGUuY2hlY2tib3goKSkuYmluZCgpLnR5cGU7XG4gIH1cblxuICBwdWJsaWMgYmluZCgpOiBQYWNrYWdlVGFnR3JvdXBDb25maWcge1xuICAgIHJldHVybiB7XG4gICAgICBpZDogdGhpcy5pZCxcbiAgICAgIGxhYmVsOiB0aGlzLmxhYmVsLFxuICAgICAgdG9vbHRpcDogdGhpcy50b29sdGlwLFxuICAgICAgZmlsdGVyVHlwZTogdGhpcy5maWx0ZXJUeXBlLFxuICAgIH07XG4gIH1cbn1cbiJdfQ==
@@ -31,5 +31,5 @@ class PreloadFile {
31
31
  }
32
32
  exports.PreloadFile = PreloadFile;
33
33
  _a = JSII_RTTI_SYMBOL_1;
34
- PreloadFile[_a] = { fqn: "construct-hub.PreloadFile", version: "0.4.6" };
34
+ PreloadFile[_a] = { fqn: "construct-hub.PreloadFile", version: "0.4.7" };
35
35
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJlbG9hZC1maWxlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsMkJBQWtDO0FBRWxDOzs7R0FHRztBQUNILE1BQWEsV0FBVztJQWdCdEIsWUFBNkIsSUFBWTtRQUFaLFNBQUksR0FBSixJQUFJLENBQVE7SUFBRyxDQUFDO0lBZjdDOztPQUVHO0lBQ0gsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFZO1FBQzFCLE1BQU0sSUFBSSxHQUFHLGlCQUFZLENBQUMsSUFBSSxFQUFFLEVBQUUsUUFBUSxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFDdkQsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNLENBQUMsUUFBUSxDQUFDLElBQVk7UUFDMUIsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBSU0sSUFBSTtRQUNULE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQztJQUNuQixDQUFDOztBQXBCSCxrQ0FxQkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByZWFkRmlsZVN5bmMgfSBmcm9tICdmcyc7XG5cbi8qKlxuICogUmVwcmVzZW50cyBhIGphdmFzY3JpcHQgZmlsZSB0byBsb2FkIGJlZm9yZSB0aGUgd2ViYXBwLlxuICogVGhpcyBjYW4gYWxsb3cgb3BlcmF0b3JzIHRvIGFkZCB0aGVpciBvd24gY2xpZW50IG1vbml0b3JzIG9yIGFuYWx5dGljcyBpZiB0aGV5IHdpc2hcbiAqL1xuZXhwb3J0IGNsYXNzIFByZWxvYWRGaWxlIHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBQcmVsb2FkRmlsZSBpbnN0YW5jZSBmcm9tIGEgZmlsZXBhdGggdG8gbG9hZFxuICAgKi9cbiAgc3RhdGljIGZyb21GaWxlKHBhdGg6IHN0cmluZyk6IFByZWxvYWRGaWxlIHtcbiAgICBjb25zdCBkYXRhID0gcmVhZEZpbGVTeW5jKHBhdGgsIHsgZW5jb2Rpbmc6ICd1dGYtOCcgfSk7XG4gICAgcmV0dXJuIG5ldyBQcmVsb2FkRmlsZShkYXRhKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgUHJlbG9hZEZpbGUgaW5zdGFuY2UgZGlyZWN0bHkgZnJvbSBzb3VyY2UgY29kZVxuICAgKi9cbiAgc3RhdGljIGZyb21Db2RlKGNvZGU6IHN0cmluZyk6IFByZWxvYWRGaWxlIHtcbiAgICByZXR1cm4gbmV3IFByZWxvYWRGaWxlKGNvZGUpO1xuICB9XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBkYXRhOiBzdHJpbmcpIHt9XG5cbiAgcHVibGljIGJpbmQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy5kYXRhO1xuICB9XG59XG4iXX0=