construct-hub 0.4.412 → 0.4.413

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -352,6 +352,7 @@ function getRelevantVersionInfos(changes, metrics, denyList, licenseList, knownV
352
352
  // Parse all the dates to ensure they are comparable
353
353
  .map(([version, isoDate]) => [version, new Date(isoDate)]);
354
354
  metrics.putMetric("PackageVersionCount" /* MetricName.PACKAGE_VERSION_COUNT */, packageVersionUpdates.length, aws_embedded_metrics_1.Unit.Count);
355
+ const unpublishedVersions = [];
355
356
  for (const [version, modified] of packageVersionUpdates) {
356
357
  const knownKey = `${change.doc.name}@${version}`;
357
358
  const known = knownVersions.get(knownKey);
@@ -359,7 +360,7 @@ function getRelevantVersionInfos(changes, metrics, denyList, licenseList, knownV
359
360
  const infos = change.doc.versions[version];
360
361
  if (infos == null) {
361
362
  // Could be the version in question was un-published.
362
- console.log(`[${change.seq}] Could not find info for "${change.doc.name}@${version}". Was it un-published?`);
363
+ unpublishedVersions.push(knownKey);
363
364
  }
364
365
  else if (isConstructLibrary(infos)) {
365
366
  // skip if this package is denied
@@ -384,6 +385,9 @@ function getRelevantVersionInfos(changes, metrics, denyList, licenseList, knownV
384
385
  // Else this is not a construct library, so we'll just ignore it...
385
386
  }
386
387
  }
388
+ if (unpublishedVersions.length > 0) {
389
+ console.log(`[${change.seq}] Could not find info for the following versions. Were they un-published?\n${unpublishedVersions.join(',\n')}`);
390
+ }
387
391
  }
388
392
  return result;
389
393
  /**
@@ -438,4 +442,4 @@ function objFromDateMap(xs) {
438
442
  function dateMapFromObj(xs) {
439
443
  return new Map(Object.entries(xs).map(([k, v]) => [k, new Date(v)]));
440
444
  }
441
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npm-js-follower.lambda.js","sourceRoot":"","sources":["../../../src/package-sources/npmjs/npm-js-follower.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as console from 'console';\nimport { InvokeCommand } from '@aws-sdk/client-lambda';\nimport {\n  GetObjectCommand,\n  NoSuchKey,\n  PutObjectCommand,\n  PutObjectCommandInput,\n} from '@aws-sdk/client-s3';\nimport type {\n  NodeJsRuntimeStreamingBlobPayloadOutputTypes,\n  StreamingBlobPayloadInputTypes,\n} from '@smithy/types';\nimport {\n  metricScope,\n  Configuration,\n  MetricsLogger,\n  Unit,\n} from 'aws-embedded-metrics';\nimport type { Context, ScheduledEvent } from 'aws-lambda';\nimport { captureHTTPsGlobal } from 'aws-xray-sdk-core';\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nimport {\n  MetricName,\n  MARKER_FILE_NAME,\n  METRICS_NAMESPACE,\n  KNOWN_VERSIONS_FILE_NAME,\n} from './constants.lambda-shared';\nimport { CouchChanges, DatabaseChange } from './couch-changes.lambda-shared';\nimport { PackageVersion } from './stage-and-notify.lambda';\nimport { DenyListClient } from '../../backend/deny-list/client.lambda-shared';\nimport { LicenseListClient } from '../../backend/license-list/client.lambda-shared';\nimport {\n  LAMBDA_CLIENT,\n  S3_CLIENT,\n} from '../../backend/shared/aws.lambda-shared';\nimport { decompressContent } from '../../backend/shared/compress-content.lambda-shared';\nimport { requireEnv } from '../../backend/shared/env.lambda-shared';\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst normalizeNPMMetadata = require('normalize-registry-metadata');\n\nconst CONSTRUCT_KEYWORDS: ReadonlySet<string> = new Set([\n  'cdk',\n  'aws-cdk',\n  'awscdk',\n  'cdk8s',\n  'cdktf',\n]);\nconst NPM_REPLICA_REGISTRY_URL = 'https://replicate.npmjs.com/';\n\n/**\n * The release date of `aws-cdk@0.8.0`. Anything earlier than this basically is\n * not a relevant package, as it cannot possibly be a constructs-based package.\n * This is used to fast-forward over boring stuff when the sequence number is\n * reset.\n */\nconst DAWN_OF_CONSTRUCTS = new Date('2018-07-31T13:43:04.615Z');\n\n// Configure embedded metrics format\nConfiguration.namespace = METRICS_NAMESPACE;\n\n// Make sure X-Ray traces will include HTTP(s) calls.\n// eslint-disable-next-line @typescript-eslint/no-require-imports\ncaptureHTTPsGlobal(require('https'));\n// eslint-disable-next-line @typescript-eslint/no-require-imports\ncaptureHTTPsGlobal(require('http'));\n\n/**\n * This function triggers on a fixed schedule and reads a stream of changes from npmjs couchdb _changes endpoint.\n * Upon invocation the function starts reading from a sequence stored in an s3 object - the `marker`.\n * If the marker fails to load (or do not exist), the stream will start from `now` - the latest change.\n * For each change:\n *  - the package version tarball will be copied from the npm registry to a stating bucket.\n *  - a message will be sent to an sqs queue\n * npm registry API docs: https://github.com/npm/registry/blob/master/docs/REGISTRY-API.md\n * @param context a Lambda execution context\n */\nexport async function handler(event: ScheduledEvent, context: Context) {\n  console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n  const stagingBucket = requireEnv('BUCKET_NAME');\n  const stagingFunction = requireEnv('FUNCTION_NAME');\n\n  const denyList = await DenyListClient.newClient();\n  const licenseList = await LicenseListClient.newClient();\n\n  const npm = new CouchChanges(NPM_REPLICA_REGISTRY_URL, 'registry/_changes');\n\n  const {\n    marker: initialMarker,\n    knownVersions,\n    didLoadVersionsFromLegacy,\n  } = await loadFollowerPosition(stagingBucket, npm);\n\n  // If we loaded from the old location, write to the new\n  let writeKnownVersionsFile = didLoadVersionsFromLegacy;\n\n  // The last written marker seq id.\n  let updatedMarker = initialMarker;\n\n  // The slowest batch processing time so far (starts at 30 seconds). This is how much time should\n  // be left before timeout if a new batch is to be fetched.\n  let maxBatchProcessingTime = 30_000;\n  // Whether we should continue reading more items or not... This is set to false when the current\n  // latest change is reached (i.e: next page of changes is empty).\n  let shouldContinue = true;\n\n  do {\n    await metricScope((metrics) => async () => {\n      console.log('Polling changes from npm replica');\n      const changes = await npm.changes(updatedMarker);\n\n      // Clear automatically set dimensions - we don't need them (see https://github.com/awslabs/aws-embedded-metrics-node/issues/73)\n      metrics.setDimensions({});\n\n      // Recording current seq range and updating the `updatedMarker`.\n      metrics.setProperty('StartSeq', updatedMarker);\n      updatedMarker = Number(changes.last_seq);\n      metrics.setProperty('EndSeq', updatedMarker);\n\n      const startTime = Date.now();\n\n      try {\n        const batch = changes.actionableResults as readonly Change[];\n\n        // The most recent \"modified\" timestamp observed in the batch.\n        let lastModified: Date | undefined;\n        // Emit npm.js replication lag\n        for (const { doc } of batch) {\n          if (doc?.time?.modified) {\n            const modified = new Date(doc.time.modified);\n            metrics.putMetric(\n              MetricName.NPMJS_CHANGE_AGE,\n              startTime - modified.getTime(),\n              Unit.Milliseconds\n            );\n            if (lastModified == null || lastModified < modified) {\n              lastModified = modified;\n            }\n          }\n        }\n\n        console.log(\n          `Received a batch of ${changes.totalCount} element(s), ${batch.length} after filtering`\n        );\n        metrics.putMetric(MetricName.CHANGE_COUNT, batch.length, Unit.Count);\n\n        if (lastModified && lastModified < DAWN_OF_CONSTRUCTS) {\n          console.log(\n            `Skipping batch as the latest modification is ${lastModified}, which is pre-Constructs`\n          );\n        } else if (changes.totalCount === 0) {\n          console.log('Received 0 changes, caught up to \"now\", exiting...');\n          shouldContinue = false;\n        } else {\n          // Obtain the modified package version from the update event, and filter\n          // out packages that are not of interest to us (not construct libraries).\n          const versionInfos = getRelevantVersionInfos(\n            batch,\n            metrics,\n            denyList,\n            licenseList,\n            knownVersions\n          );\n          console.log(\n            `Identified ${versionInfos.length} relevant package version update(s)`\n          );\n          metrics.putMetric(\n            MetricName.RELEVANT_PACKAGE_VERSIONS,\n            versionInfos.length,\n            Unit.Count\n          );\n\n          // If we have versionInfos, we already added them to the knownVersions map and we should save it again\n          writeKnownVersionsFile ||= versionInfos.length > 0;\n\n          // Process all remaining updates\n          await Promise.all(\n            versionInfos.map(async ({ infos, modified, seq }) => {\n              const invokeArgs: PackageVersion = {\n                integrity: infos.dist.shasum,\n                modified: modified.toISOString(),\n                name: infos.name,\n                seq: seq?.toString(),\n                tarballUrl: infos.dist.tarball,\n                version: infos.version,\n              };\n              // \"Fire-and-forget\" invocation here.\n              console.log(`Sending ${invokeArgs.tarballUrl} for staging`);\n              await LAMBDA_CLIENT.send(\n                new InvokeCommand({\n                  FunctionName: stagingFunction,\n                  InvocationType: 'Event',\n                  Payload: Buffer.from(JSON.stringify(invokeArgs)),\n                })\n              );\n              // Record that this is now a \"known\" version (no need to re-discover)\n              knownVersions.set(`${infos.name}@${infos.version}`, modified);\n            })\n          );\n        }\n\n        // Updating the S3 stored marker with the new seq id and known versions concurrently\n        await Promise.all([\n          // always save the transaction marker\n          saveLastTransactionMarker(context, stagingBucket, updatedMarker),\n          // conditionally save known versions\n          ...(writeKnownVersionsFile\n            ? [saveLastKnownVersions(context, stagingBucket, knownVersions)]\n            : []),\n        ]);\n      } finally {\n        metrics.putMetric(MetricName.LAST_SEQ, updatedMarker, Unit.None);\n        metrics.putMetric(\n          MetricName.BATCH_PROCESSING_TIME,\n          Date.now() - startTime,\n          Unit.Milliseconds\n        );\n        metrics.putMetric(\n          MetricName.REMAINING_TIME,\n          context.getRemainingTimeInMillis(),\n          Unit.Milliseconds\n        );\n      }\n    })();\n  } while (\n    shouldContinue &&\n    context.getRemainingTimeInMillis() >= maxBatchProcessingTime\n  );\n\n  console.log('All done here, we have success!');\n\n  return { initialMarker, updatedMarker };\n}\n\n//#region Last known versions and marker\n/**\n * Common function to load data from an S3 file with error handling\n *\n * @param stagingBucket The S3 bucket name\n * @param key The file key in the bucket\n * @param warningMessage Message to log when file doesn't exist\n * @returns The decompressed file content as string, or null if file doesn't exist\n */\nasync function loadContentFromS3(\n  stagingBucket: string,\n  key: string,\n  warningMessage: string\n): Promise<string | null> {\n  try {\n    const response = await S3_CLIENT.send(\n      new GetObjectCommand({\n        Bucket: stagingBucket,\n        Key: key,\n      })\n    );\n    if (!response.Body) {\n      throw new Error(`Response Body for ${key} is empty`);\n    }\n    return await decompressContent(\n      response.Body as NodeJsRuntimeStreamingBlobPayloadOutputTypes,\n      response.ContentEncoding\n    );\n  } catch (error: any) {\n    if (error instanceof NoSuchKey || error.name === 'NoSuchKey') {\n      console.warn(warningMessage);\n      return null;\n    }\n    // re-throw unexpected errors\n    throw error;\n  }\n}\n\nasync function loadFollowerPosition(\n  stagingBucket: string,\n  registry: CouchChanges\n): Promise<{\n  marker: number;\n  knownVersions: Map<string, Date>;\n  didLoadVersionsFromLegacy: boolean;\n}> {\n  const [markerFile, knownVersions] = await Promise.all([\n    loadLastTransactionMarker(stagingBucket, registry),\n    loadLastKnownVersions(stagingBucket),\n  ]);\n\n  if (!knownVersions) {\n    // For legacy reasons, knownVersions could come from the transaction marker file\n    // If it does, then we take that as knownVersions but will migrate knownVersions\n    // to the modern file location\n    return {\n      marker: markerFile.marker,\n      knownVersions: markerFile.knownVersions ?? new Map(),\n      didLoadVersionsFromLegacy: true,\n    };\n  }\n\n  return {\n    marker: markerFile.marker,\n    knownVersions,\n    didLoadVersionsFromLegacy: false,\n  };\n}\n\n/**\n * Loads the last known versions from S3.\n *\n * @returns the value of the last known versions.\n */\nasync function loadLastKnownVersions(\n  stagingBucket: string\n): Promise<Map<string, Date> | undefined> {\n  const warningMessage = `Known versions object (s3://${stagingBucket}/${KNOWN_VERSIONS_FILE_NAME}) does not exist, starting from scratch`;\n  const content = await loadContentFromS3(\n    stagingBucket,\n    KNOWN_VERSIONS_FILE_NAME,\n    warningMessage\n  );\n\n  // Known Versions does not exist, starting from scratch\n  if (content === null) {\n    return undefined;\n  }\n\n  // Known Versions exists, generating map from the data\n  const contentsObj: KnownVersionsFileSchema = JSON.parse(content);\n\n  console.log(`Loaded last known versions data`);\n\n  return dateMapFromObj(contentsObj.knownVersions);\n}\n\n/**\n * Loads the last transaction marker from S3.\n *\n * @param registry a Nano database corresponding to the Npmjs.com CouchDB instance.\n *\n * @returns the value of the last transaction marker.\n */\nasync function loadLastTransactionMarker(\n  stagingBucket: string,\n  registry: CouchChanges\n): Promise<{ marker: number; knownVersions?: Map<string, Date> }> {\n  const warningMessage = `Marker object (s3://${stagingBucket}/${MARKER_FILE_NAME}) does not exist, starting from scratch`;\n  const content = await loadContentFromS3(\n    stagingBucket,\n    MARKER_FILE_NAME,\n    warningMessage\n  );\n\n  // Last transaction marker does not exist\n  if (content === null) {\n    return { marker: 0 };\n  }\n\n  const parsedFile: MarkerFileSchema = JSON.parse(content);\n\n  let marker: number;\n  let knownVersions: Map<string, Date> | undefined;\n  if (typeof parsedFile === 'number') {\n    marker = parsedFile;\n  } else {\n    marker = Number(parsedFile.marker);\n    knownVersions = parsedFile.knownVersions\n      ? dateMapFromObj(parsedFile.knownVersions)\n      : undefined;\n  }\n\n  // amazonq-ignore-next-line\n  console.log(`Read last transaction marker: ${marker}`);\n\n  const dbUpdateSeq = Number((await registry.info()).update_seq);\n  if (dbUpdateSeq < marker) {\n    console.warn(\n      `Current DB update_seq (${dbUpdateSeq}) is lower than marker (CouchDB instance was likely replaced), resetting to 0!`\n    );\n    return { marker: 0 };\n  }\n\n  return { marker, knownVersions };\n}\n\n/**\n * Updates the last known versions in S3.\n *\n * @param knownVersions the map of package name + version to last modified timestamp of packages that have been processed.\n */\nasync function saveLastKnownVersions(\n  context: Context,\n  stagingBucket: string,\n  knownVersions: Map<string, Date>\n) {\n  const contentsObj: KnownVersionsFileSchema = {\n    knownVersions: objFromDateMap(knownVersions),\n  };\n\n  console.log(`Known versions changed, updating...`);\n  await putObject(\n    context,\n    stagingBucket,\n    KNOWN_VERSIONS_FILE_NAME,\n    JSON.stringify(contentsObj, undefined, 2),\n    {\n      ContentType: 'application/json',\n    }\n  );\n  console.log('Successfully updated known versions');\n}\n\n/**\n * Updates the last transaction marker in S3.\n *\n * @param marker the last transaction marker value\n */\nasync function saveLastTransactionMarker(\n  context: Context,\n  stagingBucket: string,\n  marker: number\n) {\n  const contents = JSON.stringify({ marker });\n  console.log(`Updating last transaction marker to ${marker}`);\n  await putObject(context, stagingBucket, MARKER_FILE_NAME, contents, {\n    ContentType: 'application/json',\n  });\n  console.log('Successfully updated marker');\n}\n//#endregion\n\n//#region Asynchronous Primitives\n/**\n * Puts an object in the staging bucket, with standardized object metadata.\n *\n * @param key  the key for the object to be put.\n * @param body the body of the object to be put.\n * @param opts any other options to use when sending the S3 request.\n *\n * @returns the result of the S3 request.\n */\nfunction putObject(\n  context: Context,\n  bucket: string,\n  key: string,\n  body: StreamingBlobPayloadInputTypes,\n  opts: Omit<PutObjectCommandInput, 'Bucket' | 'Key' | 'Body'> = {}\n) {\n  return S3_CLIENT.send(\n    new PutObjectCommand({\n      Bucket: bucket,\n      Key: key,\n      Body: body,\n      Metadata: {\n        'Lambda-Log-Group': context.logGroupName,\n        'Lambda-Log-Stream': context.logStreamName,\n        'Lambda-Run-Id': context.awsRequestId,\n        ...opts.Metadata,\n      },\n      ...opts,\n    })\n  );\n}\n//#endregion\n\n/**\n * Obtains the `VersionInfo` corresponding to the modified version(s) in the\n * provided `Change` objects, ensures they are relevant (construct libraries),\n * and returns those only.\n *\n * @param changes the changes to be processed.\n * @param metrics the metrics logger to use.\n * @param denyList deny list client\n *\n * @returns a list of `VersionInfo` objects\n */\nfunction getRelevantVersionInfos(\n  changes: readonly Change[],\n  metrics: MetricsLogger,\n  denyList: DenyListClient,\n  licenseList: LicenseListClient,\n  knownVersions: Map<string, Date>\n): readonly UpdatedVersion[] {\n  const result = new Array<UpdatedVersion>();\n\n  for (const change of changes) {\n    // Filter out all elements that don't have a \"name\" in the document, as\n    // these are schemas, which are not relevant to our business here.\n    if (change.doc.name === undefined) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'name': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // The normalize function change the object in place, if the doc object is invalid it will return undefined\n    if (normalizeNPMMetadata(change.doc) === undefined) {\n      console.error(\n        `[${change.seq}] Changed document invalid, npm normalize returned undefined: ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Sometimes, there are no versions in the document. We skip those.\n    if (change.doc.versions == null) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'versions': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Sometimes, there is no 'time' entry in the document. We skip those.\n    if (change.doc.time == null) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'time': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Get the last modification date from the change\n    const packageVersionUpdates = Object.entries(change.doc.time)\n      // Ignore the \"created\" and \"modified\" keys here\n      .filter(([key]) => key !== 'created' && key !== 'modified')\n      // Parse all the dates to ensure they are comparable\n      .map(([version, isoDate]) => [version, new Date(isoDate)] as const);\n    metrics.putMetric(\n      MetricName.PACKAGE_VERSION_COUNT,\n      packageVersionUpdates.length,\n      Unit.Count\n    );\n\n    for (const [version, modified] of packageVersionUpdates) {\n      const knownKey = `${change.doc.name}@${version}`;\n      const known = knownVersions.get(knownKey);\n      if (known == null || known < modified) {\n        const infos = change.doc.versions[version];\n        if (infos == null) {\n          // Could be the version in question was un-published.\n          console.log(\n            `[${change.seq}] Could not find info for \"${change.doc.name}@${version}\". Was it un-published?`\n          );\n        } else if (isConstructLibrary(infos)) {\n          // skip if this package is denied\n          const denied = denyList.lookup(infos.name, infos.version);\n          if (denied) {\n            console.log(\n              `[${change.seq}] Package denied: ${JSON.stringify(denied)}`\n            );\n            knownVersions.set(knownKey, modified);\n            metrics.putMetric(MetricName.DENY_LISTED_COUNT, 1, Unit.Count);\n            continue;\n          }\n\n          metrics.putMetric(\n            MetricName.PACKAGE_VERSION_AGE,\n            Date.now() - modified.getTime(),\n            Unit.Milliseconds\n          );\n          const isEligible =\n            licenseList.lookup(infos.license ?? 'UNLICENSED') != null;\n          metrics.putMetric(\n            MetricName.INELIGIBLE_LICENSE,\n            isEligible ? 0 : 1,\n            Unit.Count\n          );\n          if (isEligible) {\n            result.push({ infos, modified, seq: change.seq });\n          } else {\n            console.log(\n              `[${change.seq}] Package \"${\n                change.doc.name\n              }@${version}\" does not use allow-listed license: ${\n                infos.license ?? 'UNLICENSED'\n              }`\n            );\n            knownVersions.set(knownKey, modified);\n          }\n        }\n        // Else this is not a construct library, so we'll just ignore it...\n      }\n    }\n  }\n  return result;\n\n  /**\n   * This determines whether a package is \"interesting\" to ConstructHub or not. This is related but\n   * not necessarily identical to the logic in the ingestion process that annotates package metadata\n   * with a construct framework name + version (those could ultimately be re-factored to share more\n   * of the logic/heuristics, though).\n   *\n   * Concretely, it checks for a list of known \"official\" packages for various construct frameworks,\n   * and packages that have a dependency on such a package. It also has a keywords allow-list as a\n   * fall-back (the current dependency-based logic does not consider transitive dependencies and\n   * might hence miss certain rare use-cases, which keywords would rescue).\n   */\n  function isConstructLibrary(infos: VersionInfo): boolean {\n    if (infos.jsii == null) {\n      return false;\n    }\n    // The \"constructs\" package is a sign of a constructs library\n    return (\n      isConstructFrameworkPackage(infos.name) ||\n      // Recursively apply on dependencies\n      Object.keys(infos.dependencies ?? {}).some(isConstructFrameworkPackage) ||\n      Object.keys(infos.devDependencies ?? {}).some(\n        isConstructFrameworkPackage\n      ) ||\n      Object.keys(infos.peerDependencies ?? {}).some(\n        isConstructFrameworkPackage\n      ) ||\n      // Keyword-based fallback\n      infos.keywords?.some((kw) => CONSTRUCT_KEYWORDS.has(kw))\n    );\n  }\n\n  /**\n   * Package is one of the known construct framework's first party packages:\n   * - @aws-cdk/*\n   * - @cdktf/*\n   * - cdk8s or cdk8s-plus\n   */\n  function isConstructFrameworkPackage(name: string): boolean {\n    // IMPORTANT NOTE: Prefix matching should only be used for @scope/ names.\n\n    // The low-level constructs package\n    return (\n      name === 'constructs' ||\n      // AWS CDK Packages\n      name === 'aws-cdk-lib' ||\n      name === 'monocdk' ||\n      name.startsWith('@aws-cdk/') ||\n      // CDK8s packages\n      name === 'cdk8s' ||\n      /^cdk8s-plus(?:-(?:17|20|21|22))?$/.test(name) ||\n      // CDKTf packages\n      name === 'cdktf' ||\n      name.startsWith('@cdktf/')\n    );\n  }\n}\n\nfunction objFromDateMap(xs: Map<string, Date>): Record<string, string> {\n  return Object.fromEntries(\n    Array.from(xs).map(([k, v]) => [k, v.toISOString()])\n  );\n}\n\nfunction dateMapFromObj(\n  xs: Record<string, string | number>\n): Map<string, Date> {\n  return new Map(Object.entries(xs).map(([k, v]) => [k, new Date(v)]));\n}\n\n/**\n * The scheme of a package version in the update. Includes the package.json keys, as well as some additional npm metadata\n * @see https://github.com/npm/registry/blob/master/docs/REGISTRY-API.md#version\n */\nexport interface VersionInfo {\n  readonly dependencies?: { readonly [name: string]: string };\n  readonly devDependencies?: { readonly [name: string]: string };\n  readonly peerDependencies?: { readonly [name: string]: string };\n  readonly jsii: unknown;\n  readonly license?: string;\n  readonly name: string;\n  readonly [key: string]: unknown;\n  readonly keywords: string[];\n  readonly dist: {\n    readonly shasum: string;\n    readonly tarball: string;\n  };\n  readonly version: string;\n}\n\ninterface UpdatedVersion {\n  /**\n   * The `VersionInfo` for the modified package version.\n   */\n  readonly infos: VersionInfo;\n\n  /**\n   * The time at which the `VersionInfo` was last modified.\n   */\n  readonly modified: Date;\n\n  /**\n   * The CouchDB transaction number for the update.\n   */\n  readonly seq?: string | number;\n}\n\n/**\n * We are accounting for a good bit of legacy data modeling here\n *\n * The file can be a just a number, or a combination of a sequence number (which\n * is potentially encoded as a string) and a set of known versions and the dates\n * we first saw them. The date can be encoded as an ISO string, or a timestamp\n * number.\n */\ntype MarkerFileSchema =\n  | number\n  | {\n      marker: number | string;\n      knownVersions?: Record<string, string | number>;\n    };\n\ninterface KnownVersionsFileSchema {\n  knownVersions: Record<string, string>;\n}\n\ninterface Document {\n  /**\n   * a List of all Version objects for the package\n   */\n  readonly versions: { [key: string]: VersionInfo | undefined };\n\n  /**\n   * The package's name.\n   */\n  readonly name: string;\n\n  /**\n   * Timestamps associated with this document. The values are ISO-8601 encoded\n   * timestamps.\n   */\n  readonly time: {\n    readonly created: string;\n    readonly modified: string;\n    readonly [version: string]: string;\n  };\n\n  readonly [key: string]: unknown;\n}\n\ninterface Change extends DatabaseChange {\n  readonly doc: Document;\n}\n"]}
445
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"npm-js-follower.lambda.js","sourceRoot":"","sources":["../../../src/package-sources/npmjs/npm-js-follower.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as console from 'console';\nimport { InvokeCommand } from '@aws-sdk/client-lambda';\nimport {\n  GetObjectCommand,\n  NoSuchKey,\n  PutObjectCommand,\n  PutObjectCommandInput,\n} from '@aws-sdk/client-s3';\nimport type {\n  NodeJsRuntimeStreamingBlobPayloadOutputTypes,\n  StreamingBlobPayloadInputTypes,\n} from '@smithy/types';\nimport {\n  metricScope,\n  Configuration,\n  MetricsLogger,\n  Unit,\n} from 'aws-embedded-metrics';\nimport type { Context, ScheduledEvent } from 'aws-lambda';\nimport { captureHTTPsGlobal } from 'aws-xray-sdk-core';\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nimport {\n  MetricName,\n  MARKER_FILE_NAME,\n  METRICS_NAMESPACE,\n  KNOWN_VERSIONS_FILE_NAME,\n} from './constants.lambda-shared';\nimport { CouchChanges, DatabaseChange } from './couch-changes.lambda-shared';\nimport { PackageVersion } from './stage-and-notify.lambda';\nimport { DenyListClient } from '../../backend/deny-list/client.lambda-shared';\nimport { LicenseListClient } from '../../backend/license-list/client.lambda-shared';\nimport {\n  LAMBDA_CLIENT,\n  S3_CLIENT,\n} from '../../backend/shared/aws.lambda-shared';\nimport { decompressContent } from '../../backend/shared/compress-content.lambda-shared';\nimport { requireEnv } from '../../backend/shared/env.lambda-shared';\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst normalizeNPMMetadata = require('normalize-registry-metadata');\n\nconst CONSTRUCT_KEYWORDS: ReadonlySet<string> = new Set([\n  'cdk',\n  'aws-cdk',\n  'awscdk',\n  'cdk8s',\n  'cdktf',\n]);\nconst NPM_REPLICA_REGISTRY_URL = 'https://replicate.npmjs.com/';\n\n/**\n * The release date of `aws-cdk@0.8.0`. Anything earlier than this basically is\n * not a relevant package, as it cannot possibly be a constructs-based package.\n * This is used to fast-forward over boring stuff when the sequence number is\n * reset.\n */\nconst DAWN_OF_CONSTRUCTS = new Date('2018-07-31T13:43:04.615Z');\n\n// Configure embedded metrics format\nConfiguration.namespace = METRICS_NAMESPACE;\n\n// Make sure X-Ray traces will include HTTP(s) calls.\n// eslint-disable-next-line @typescript-eslint/no-require-imports\ncaptureHTTPsGlobal(require('https'));\n// eslint-disable-next-line @typescript-eslint/no-require-imports\ncaptureHTTPsGlobal(require('http'));\n\n/**\n * This function triggers on a fixed schedule and reads a stream of changes from npmjs couchdb _changes endpoint.\n * Upon invocation the function starts reading from a sequence stored in an s3 object - the `marker`.\n * If the marker fails to load (or do not exist), the stream will start from `now` - the latest change.\n * For each change:\n *  - the package version tarball will be copied from the npm registry to a stating bucket.\n *  - a message will be sent to an sqs queue\n * npm registry API docs: https://github.com/npm/registry/blob/master/docs/REGISTRY-API.md\n * @param context a Lambda execution context\n */\nexport async function handler(event: ScheduledEvent, context: Context) {\n  console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n\n  const stagingBucket = requireEnv('BUCKET_NAME');\n  const stagingFunction = requireEnv('FUNCTION_NAME');\n\n  const denyList = await DenyListClient.newClient();\n  const licenseList = await LicenseListClient.newClient();\n\n  const npm = new CouchChanges(NPM_REPLICA_REGISTRY_URL, 'registry/_changes');\n\n  const {\n    marker: initialMarker,\n    knownVersions,\n    didLoadVersionsFromLegacy,\n  } = await loadFollowerPosition(stagingBucket, npm);\n\n  // If we loaded from the old location, write to the new\n  let writeKnownVersionsFile = didLoadVersionsFromLegacy;\n\n  // The last written marker seq id.\n  let updatedMarker = initialMarker;\n\n  // The slowest batch processing time so far (starts at 30 seconds). This is how much time should\n  // be left before timeout if a new batch is to be fetched.\n  let maxBatchProcessingTime = 30_000;\n  // Whether we should continue reading more items or not... This is set to false when the current\n  // latest change is reached (i.e: next page of changes is empty).\n  let shouldContinue = true;\n\n  do {\n    await metricScope((metrics) => async () => {\n      console.log('Polling changes from npm replica');\n      const changes = await npm.changes(updatedMarker);\n\n      // Clear automatically set dimensions - we don't need them (see https://github.com/awslabs/aws-embedded-metrics-node/issues/73)\n      metrics.setDimensions({});\n\n      // Recording current seq range and updating the `updatedMarker`.\n      metrics.setProperty('StartSeq', updatedMarker);\n      updatedMarker = Number(changes.last_seq);\n      metrics.setProperty('EndSeq', updatedMarker);\n\n      const startTime = Date.now();\n\n      try {\n        const batch = changes.actionableResults as readonly Change[];\n\n        // The most recent \"modified\" timestamp observed in the batch.\n        let lastModified: Date | undefined;\n        // Emit npm.js replication lag\n        for (const { doc } of batch) {\n          if (doc?.time?.modified) {\n            const modified = new Date(doc.time.modified);\n            metrics.putMetric(\n              MetricName.NPMJS_CHANGE_AGE,\n              startTime - modified.getTime(),\n              Unit.Milliseconds\n            );\n            if (lastModified == null || lastModified < modified) {\n              lastModified = modified;\n            }\n          }\n        }\n\n        console.log(\n          `Received a batch of ${changes.totalCount} element(s), ${batch.length} after filtering`\n        );\n        metrics.putMetric(MetricName.CHANGE_COUNT, batch.length, Unit.Count);\n\n        if (lastModified && lastModified < DAWN_OF_CONSTRUCTS) {\n          console.log(\n            `Skipping batch as the latest modification is ${lastModified}, which is pre-Constructs`\n          );\n        } else if (changes.totalCount === 0) {\n          console.log('Received 0 changes, caught up to \"now\", exiting...');\n          shouldContinue = false;\n        } else {\n          // Obtain the modified package version from the update event, and filter\n          // out packages that are not of interest to us (not construct libraries).\n          const versionInfos = getRelevantVersionInfos(\n            batch,\n            metrics,\n            denyList,\n            licenseList,\n            knownVersions\n          );\n          console.log(\n            `Identified ${versionInfos.length} relevant package version update(s)`\n          );\n          metrics.putMetric(\n            MetricName.RELEVANT_PACKAGE_VERSIONS,\n            versionInfos.length,\n            Unit.Count\n          );\n\n          // If we have versionInfos, we already added them to the knownVersions map and we should save it again\n          writeKnownVersionsFile ||= versionInfos.length > 0;\n\n          // Process all remaining updates\n          await Promise.all(\n            versionInfos.map(async ({ infos, modified, seq }) => {\n              const invokeArgs: PackageVersion = {\n                integrity: infos.dist.shasum,\n                modified: modified.toISOString(),\n                name: infos.name,\n                seq: seq?.toString(),\n                tarballUrl: infos.dist.tarball,\n                version: infos.version,\n              };\n              // \"Fire-and-forget\" invocation here.\n              console.log(`Sending ${invokeArgs.tarballUrl} for staging`);\n              await LAMBDA_CLIENT.send(\n                new InvokeCommand({\n                  FunctionName: stagingFunction,\n                  InvocationType: 'Event',\n                  Payload: Buffer.from(JSON.stringify(invokeArgs)),\n                })\n              );\n              // Record that this is now a \"known\" version (no need to re-discover)\n              knownVersions.set(`${infos.name}@${infos.version}`, modified);\n            })\n          );\n        }\n\n        // Updating the S3 stored marker with the new seq id and known versions concurrently\n        await Promise.all([\n          // always save the transaction marker\n          saveLastTransactionMarker(context, stagingBucket, updatedMarker),\n          // conditionally save known versions\n          ...(writeKnownVersionsFile\n            ? [saveLastKnownVersions(context, stagingBucket, knownVersions)]\n            : []),\n        ]);\n      } finally {\n        metrics.putMetric(MetricName.LAST_SEQ, updatedMarker, Unit.None);\n        metrics.putMetric(\n          MetricName.BATCH_PROCESSING_TIME,\n          Date.now() - startTime,\n          Unit.Milliseconds\n        );\n        metrics.putMetric(\n          MetricName.REMAINING_TIME,\n          context.getRemainingTimeInMillis(),\n          Unit.Milliseconds\n        );\n      }\n    })();\n  } while (\n    shouldContinue &&\n    context.getRemainingTimeInMillis() >= maxBatchProcessingTime\n  );\n\n  console.log('All done here, we have success!');\n\n  return { initialMarker, updatedMarker };\n}\n\n//#region Last known versions and marker\n/**\n * Common function to load data from an S3 file with error handling\n *\n * @param stagingBucket The S3 bucket name\n * @param key The file key in the bucket\n * @param warningMessage Message to log when file doesn't exist\n * @returns The decompressed file content as string, or null if file doesn't exist\n */\nasync function loadContentFromS3(\n  stagingBucket: string,\n  key: string,\n  warningMessage: string\n): Promise<string | null> {\n  try {\n    const response = await S3_CLIENT.send(\n      new GetObjectCommand({\n        Bucket: stagingBucket,\n        Key: key,\n      })\n    );\n    if (!response.Body) {\n      throw new Error(`Response Body for ${key} is empty`);\n    }\n    return await decompressContent(\n      response.Body as NodeJsRuntimeStreamingBlobPayloadOutputTypes,\n      response.ContentEncoding\n    );\n  } catch (error: any) {\n    if (error instanceof NoSuchKey || error.name === 'NoSuchKey') {\n      console.warn(warningMessage);\n      return null;\n    }\n    // re-throw unexpected errors\n    throw error;\n  }\n}\n\nasync function loadFollowerPosition(\n  stagingBucket: string,\n  registry: CouchChanges\n): Promise<{\n  marker: number;\n  knownVersions: Map<string, Date>;\n  didLoadVersionsFromLegacy: boolean;\n}> {\n  const [markerFile, knownVersions] = await Promise.all([\n    loadLastTransactionMarker(stagingBucket, registry),\n    loadLastKnownVersions(stagingBucket),\n  ]);\n\n  if (!knownVersions) {\n    // For legacy reasons, knownVersions could come from the transaction marker file\n    // If it does, then we take that as knownVersions but will migrate knownVersions\n    // to the modern file location\n    return {\n      marker: markerFile.marker,\n      knownVersions: markerFile.knownVersions ?? new Map(),\n      didLoadVersionsFromLegacy: true,\n    };\n  }\n\n  return {\n    marker: markerFile.marker,\n    knownVersions,\n    didLoadVersionsFromLegacy: false,\n  };\n}\n\n/**\n * Loads the last known versions from S3.\n *\n * @returns the value of the last known versions.\n */\nasync function loadLastKnownVersions(\n  stagingBucket: string\n): Promise<Map<string, Date> | undefined> {\n  const warningMessage = `Known versions object (s3://${stagingBucket}/${KNOWN_VERSIONS_FILE_NAME}) does not exist, starting from scratch`;\n  const content = await loadContentFromS3(\n    stagingBucket,\n    KNOWN_VERSIONS_FILE_NAME,\n    warningMessage\n  );\n\n  // Known Versions does not exist, starting from scratch\n  if (content === null) {\n    return undefined;\n  }\n\n  // Known Versions exists, generating map from the data\n  const contentsObj: KnownVersionsFileSchema = JSON.parse(content);\n\n  console.log(`Loaded last known versions data`);\n\n  return dateMapFromObj(contentsObj.knownVersions);\n}\n\n/**\n * Loads the last transaction marker from S3.\n *\n * @param registry a Nano database corresponding to the Npmjs.com CouchDB instance.\n *\n * @returns the value of the last transaction marker.\n */\nasync function loadLastTransactionMarker(\n  stagingBucket: string,\n  registry: CouchChanges\n): Promise<{ marker: number; knownVersions?: Map<string, Date> }> {\n  const warningMessage = `Marker object (s3://${stagingBucket}/${MARKER_FILE_NAME}) does not exist, starting from scratch`;\n  const content = await loadContentFromS3(\n    stagingBucket,\n    MARKER_FILE_NAME,\n    warningMessage\n  );\n\n  // Last transaction marker does not exist\n  if (content === null) {\n    return { marker: 0 };\n  }\n\n  const parsedFile: MarkerFileSchema = JSON.parse(content);\n\n  let marker: number;\n  let knownVersions: Map<string, Date> | undefined;\n  if (typeof parsedFile === 'number') {\n    marker = parsedFile;\n  } else {\n    marker = Number(parsedFile.marker);\n    knownVersions = parsedFile.knownVersions\n      ? dateMapFromObj(parsedFile.knownVersions)\n      : undefined;\n  }\n\n  // amazonq-ignore-next-line\n  console.log(`Read last transaction marker: ${marker}`);\n\n  const dbUpdateSeq = Number((await registry.info()).update_seq);\n  if (dbUpdateSeq < marker) {\n    console.warn(\n      `Current DB update_seq (${dbUpdateSeq}) is lower than marker (CouchDB instance was likely replaced), resetting to 0!`\n    );\n    return { marker: 0 };\n  }\n\n  return { marker, knownVersions };\n}\n\n/**\n * Updates the last known versions in S3.\n *\n * @param knownVersions the map of package name + version to last modified timestamp of packages that have been processed.\n */\nasync function saveLastKnownVersions(\n  context: Context,\n  stagingBucket: string,\n  knownVersions: Map<string, Date>\n) {\n  const contentsObj: KnownVersionsFileSchema = {\n    knownVersions: objFromDateMap(knownVersions),\n  };\n\n  console.log(`Known versions changed, updating...`);\n  await putObject(\n    context,\n    stagingBucket,\n    KNOWN_VERSIONS_FILE_NAME,\n    JSON.stringify(contentsObj, undefined, 2),\n    {\n      ContentType: 'application/json',\n    }\n  );\n  console.log('Successfully updated known versions');\n}\n\n/**\n * Updates the last transaction marker in S3.\n *\n * @param marker the last transaction marker value\n */\nasync function saveLastTransactionMarker(\n  context: Context,\n  stagingBucket: string,\n  marker: number\n) {\n  const contents = JSON.stringify({ marker });\n  console.log(`Updating last transaction marker to ${marker}`);\n  await putObject(context, stagingBucket, MARKER_FILE_NAME, contents, {\n    ContentType: 'application/json',\n  });\n  console.log('Successfully updated marker');\n}\n//#endregion\n\n//#region Asynchronous Primitives\n/**\n * Puts an object in the staging bucket, with standardized object metadata.\n *\n * @param key  the key for the object to be put.\n * @param body the body of the object to be put.\n * @param opts any other options to use when sending the S3 request.\n *\n * @returns the result of the S3 request.\n */\nfunction putObject(\n  context: Context,\n  bucket: string,\n  key: string,\n  body: StreamingBlobPayloadInputTypes,\n  opts: Omit<PutObjectCommandInput, 'Bucket' | 'Key' | 'Body'> = {}\n) {\n  return S3_CLIENT.send(\n    new PutObjectCommand({\n      Bucket: bucket,\n      Key: key,\n      Body: body,\n      Metadata: {\n        'Lambda-Log-Group': context.logGroupName,\n        'Lambda-Log-Stream': context.logStreamName,\n        'Lambda-Run-Id': context.awsRequestId,\n        ...opts.Metadata,\n      },\n      ...opts,\n    })\n  );\n}\n//#endregion\n\n/**\n * Obtains the `VersionInfo` corresponding to the modified version(s) in the\n * provided `Change` objects, ensures they are relevant (construct libraries),\n * and returns those only.\n *\n * @param changes the changes to be processed.\n * @param metrics the metrics logger to use.\n * @param denyList deny list client\n *\n * @returns a list of `VersionInfo` objects\n */\nfunction getRelevantVersionInfos(\n  changes: readonly Change[],\n  metrics: MetricsLogger,\n  denyList: DenyListClient,\n  licenseList: LicenseListClient,\n  knownVersions: Map<string, Date>\n): readonly UpdatedVersion[] {\n  const result = new Array<UpdatedVersion>();\n\n  for (const change of changes) {\n    // Filter out all elements that don't have a \"name\" in the document, as\n    // these are schemas, which are not relevant to our business here.\n    if (change.doc.name === undefined) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'name': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // The normalize function change the object in place, if the doc object is invalid it will return undefined\n    if (normalizeNPMMetadata(change.doc) === undefined) {\n      console.error(\n        `[${change.seq}] Changed document invalid, npm normalize returned undefined: ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Sometimes, there are no versions in the document. We skip those.\n    if (change.doc.versions == null) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'versions': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Sometimes, there is no 'time' entry in the document. We skip those.\n    if (change.doc.time == null) {\n      console.error(\n        `[${change.seq}] Changed document contains no 'time': ${change.id}`\n      );\n      metrics.putMetric(MetricName.UNPROCESSABLE_ENTITY, 1, Unit.Count);\n      continue;\n    }\n\n    // Get the last modification date from the change\n    const packageVersionUpdates = Object.entries(change.doc.time)\n      // Ignore the \"created\" and \"modified\" keys here\n      .filter(([key]) => key !== 'created' && key !== 'modified')\n      // Parse all the dates to ensure they are comparable\n      .map(([version, isoDate]) => [version, new Date(isoDate)] as const);\n    metrics.putMetric(\n      MetricName.PACKAGE_VERSION_COUNT,\n      packageVersionUpdates.length,\n      Unit.Count\n    );\n\n    const unpublishedVersions: string[] = [];\n    for (const [version, modified] of packageVersionUpdates) {\n      const knownKey = `${change.doc.name}@${version}`;\n      const known = knownVersions.get(knownKey);\n      if (known == null || known < modified) {\n        const infos = change.doc.versions[version];\n        if (infos == null) {\n          // Could be the version in question was un-published.\n          unpublishedVersions.push(knownKey);\n        } else if (isConstructLibrary(infos)) {\n          // skip if this package is denied\n          const denied = denyList.lookup(infos.name, infos.version);\n          if (denied) {\n            console.log(\n              `[${change.seq}] Package denied: ${JSON.stringify(denied)}`\n            );\n            knownVersions.set(knownKey, modified);\n            metrics.putMetric(MetricName.DENY_LISTED_COUNT, 1, Unit.Count);\n            continue;\n          }\n\n          metrics.putMetric(\n            MetricName.PACKAGE_VERSION_AGE,\n            Date.now() - modified.getTime(),\n            Unit.Milliseconds\n          );\n          const isEligible =\n            licenseList.lookup(infos.license ?? 'UNLICENSED') != null;\n          metrics.putMetric(\n            MetricName.INELIGIBLE_LICENSE,\n            isEligible ? 0 : 1,\n            Unit.Count\n          );\n          if (isEligible) {\n            result.push({ infos, modified, seq: change.seq });\n          } else {\n            console.log(\n              `[${change.seq}] Package \"${\n                change.doc.name\n              }@${version}\" does not use allow-listed license: ${\n                infos.license ?? 'UNLICENSED'\n              }`\n            );\n            knownVersions.set(knownKey, modified);\n          }\n        }\n        // Else this is not a construct library, so we'll just ignore it...\n      }\n    }\n\n    if (unpublishedVersions.length > 0) {\n      console.log(\n        `[${\n          change.seq\n        }] Could not find info for the following versions. Were they un-published?\\n${unpublishedVersions.join(\n          ',\\n'\n        )}`\n      );\n    }\n  }\n  return result;\n\n  /**\n   * This determines whether a package is \"interesting\" to ConstructHub or not. This is related but\n   * not necessarily identical to the logic in the ingestion process that annotates package metadata\n   * with a construct framework name + version (those could ultimately be re-factored to share more\n   * of the logic/heuristics, though).\n   *\n   * Concretely, it checks for a list of known \"official\" packages for various construct frameworks,\n   * and packages that have a dependency on such a package. It also has a keywords allow-list as a\n   * fall-back (the current dependency-based logic does not consider transitive dependencies and\n   * might hence miss certain rare use-cases, which keywords would rescue).\n   */\n  function isConstructLibrary(infos: VersionInfo): boolean {\n    if (infos.jsii == null) {\n      return false;\n    }\n    // The \"constructs\" package is a sign of a constructs library\n    return (\n      isConstructFrameworkPackage(infos.name) ||\n      // Recursively apply on dependencies\n      Object.keys(infos.dependencies ?? {}).some(isConstructFrameworkPackage) ||\n      Object.keys(infos.devDependencies ?? {}).some(\n        isConstructFrameworkPackage\n      ) ||\n      Object.keys(infos.peerDependencies ?? {}).some(\n        isConstructFrameworkPackage\n      ) ||\n      // Keyword-based fallback\n      infos.keywords?.some((kw) => CONSTRUCT_KEYWORDS.has(kw))\n    );\n  }\n\n  /**\n   * Package is one of the known construct framework's first party packages:\n   * - @aws-cdk/*\n   * - @cdktf/*\n   * - cdk8s or cdk8s-plus\n   */\n  function isConstructFrameworkPackage(name: string): boolean {\n    // IMPORTANT NOTE: Prefix matching should only be used for @scope/ names.\n\n    // The low-level constructs package\n    return (\n      name === 'constructs' ||\n      // AWS CDK Packages\n      name === 'aws-cdk-lib' ||\n      name === 'monocdk' ||\n      name.startsWith('@aws-cdk/') ||\n      // CDK8s packages\n      name === 'cdk8s' ||\n      /^cdk8s-plus(?:-(?:17|20|21|22))?$/.test(name) ||\n      // CDKTf packages\n      name === 'cdktf' ||\n      name.startsWith('@cdktf/')\n    );\n  }\n}\n\nfunction objFromDateMap(xs: Map<string, Date>): Record<string, string> {\n  return Object.fromEntries(\n    Array.from(xs).map(([k, v]) => [k, v.toISOString()])\n  );\n}\n\nfunction dateMapFromObj(\n  xs: Record<string, string | number>\n): Map<string, Date> {\n  return new Map(Object.entries(xs).map(([k, v]) => [k, new Date(v)]));\n}\n\n/**\n * The scheme of a package version in the update. Includes the package.json keys, as well as some additional npm metadata\n * @see https://github.com/npm/registry/blob/master/docs/REGISTRY-API.md#version\n */\nexport interface VersionInfo {\n  readonly dependencies?: { readonly [name: string]: string };\n  readonly devDependencies?: { readonly [name: string]: string };\n  readonly peerDependencies?: { readonly [name: string]: string };\n  readonly jsii: unknown;\n  readonly license?: string;\n  readonly name: string;\n  readonly [key: string]: unknown;\n  readonly keywords: string[];\n  readonly dist: {\n    readonly shasum: string;\n    readonly tarball: string;\n  };\n  readonly version: string;\n}\n\ninterface UpdatedVersion {\n  /**\n   * The `VersionInfo` for the modified package version.\n   */\n  readonly infos: VersionInfo;\n\n  /**\n   * The time at which the `VersionInfo` was last modified.\n   */\n  readonly modified: Date;\n\n  /**\n   * The CouchDB transaction number for the update.\n   */\n  readonly seq?: string | number;\n}\n\n/**\n * We are accounting for a good bit of legacy data modeling here\n *\n * The file can be a just a number, or a combination of a sequence number (which\n * is potentially encoded as a string) and a set of known versions and the dates\n * we first saw them. The date can be encoded as an ISO string, or a timestamp\n * number.\n */\ntype MarkerFileSchema =\n  | number\n  | {\n      marker: number | string;\n      knownVersions?: Record<string, string | number>;\n    };\n\ninterface KnownVersionsFileSchema {\n  knownVersions: Record<string, string>;\n}\n\ninterface Document {\n  /**\n   * a List of all Version objects for the package\n   */\n  readonly versions: { [key: string]: VersionInfo | undefined };\n\n  /**\n   * The package's name.\n   */\n  readonly name: string;\n\n  /**\n   * Timestamps associated with this document. The values are ISO-8601 encoded\n   * timestamps.\n   */\n  readonly time: {\n    readonly created: string;\n    readonly modified: string;\n    readonly [version: string]: string;\n  };\n\n  readonly [key: string]: unknown;\n}\n\ninterface Change extends DatabaseChange {\n  readonly doc: Document;\n}\n"]}
@@ -549,7 +549,7 @@ class NpmJs {
549
549
  }
550
550
  exports.NpmJs = NpmJs;
551
551
  _a = JSII_RTTI_SYMBOL_1;
552
- NpmJs[_a] = { fqn: "construct-hub.sources.NpmJs", version: "0.4.412" };
552
+ NpmJs[_a] = { fqn: "construct-hub.sources.NpmJs", version: "0.4.413" };
553
553
  /**
554
554
  * How often 'rate' goes into 'duration' (rounded up)
555
555
  */
@@ -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.412" };
53
+ TagCondition[_a] = { fqn: "construct-hub.TagCondition", version: "0.4.413" };
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.412" };
133
+ TagConditionField[_b] = { fqn: "construct-hub.TagConditionField", version: "0.4.413" };
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.412" };
150
+ TagConditionReadme[_c] = { fqn: "construct-hub.TagConditionReadme", version: "0.4.413" };
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.412" };
16
+ FilterType[_a] = { fqn: "construct-hub.FilterType", version: "0.4.413" };
17
17
  class FilterTypeRadio extends FilterType {
18
18
  bind() {
19
19
  return { type: 'radio' };
@@ -45,5 +45,5 @@ class PackageTagGroup {
45
45
  }
46
46
  exports.PackageTagGroup = PackageTagGroup;
47
47
  _b = JSII_RTTI_SYMBOL_1;
48
- PackageTagGroup[_b] = { fqn: "construct-hub.PackageTagGroup", version: "0.4.412" };
48
+ PackageTagGroup[_b] = { fqn: "construct-hub.PackageTagGroup", version: "0.4.413" };
49
49
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGFja2FnZS10YWctZ3JvdXAvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFJQSxNQUFzQixVQUFVO0lBQzlCLE1BQU0sQ0FBQyxRQUFRO1FBQ2IsT0FBTyxJQUFJLGtCQUFrQixFQUFFLENBQUM7SUFDbEMsQ0FBQztJQUVELE1BQU0sQ0FBQyxLQUFLO1FBQ1YsT0FBTyxJQUFJLGVBQWUsRUFBRSxDQUFDO0lBQy9CLENBQUM7O0FBUEgsZ0NBVUM7OztBQUVELE1BQU0sZUFBZ0IsU0FBUSxVQUFVO0lBQy9CLElBQUk7UUFDVCxPQUFPLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxDQUFDO0lBQzNCLENBQUM7Q0FDRjtBQUVELE1BQU0sa0JBQW1CLFNBQVEsVUFBVTtJQUNsQyxJQUFJO1FBQ1QsT0FBTyxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsQ0FBQztJQUM5QixDQUFDO0NBQ0Y7QUF3QkQ7O0dBRUc7QUFDSCxNQUFhLGVBQWU7SUFLMUIsWUFBNEIsRUFBVSxFQUFFLEtBQTRCO1FBQXhDLE9BQUUsR0FBRixFQUFFLENBQVE7UUFDcEMsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLEVBQUUsS0FBSyxDQUFDO1FBQzFCLElBQUksQ0FBQyxPQUFPLEdBQUcsS0FBSyxFQUFFLE9BQU8sQ0FBQztRQUM5QixJQUFJLENBQUMsVUFBVSxHQUFHLENBQUMsS0FBSyxFQUFFLFVBQVUsSUFBSSxVQUFVLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxJQUFJLENBQUM7SUFDN0UsQ0FBQztJQUVNLElBQUk7UUFDVCxPQUFPO1lBQ0wsRUFBRSxFQUFFLElBQUksQ0FBQyxFQUFFO1lBQ1gsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLO1lBQ2pCLE9BQU8sRUFBRSxJQUFJLENBQUMsT0FBTztZQUNyQixVQUFVLEVBQUUsSUFBSSxDQUFDLFVBQVU7U0FDNUIsQ0FBQztJQUNKLENBQUM7O0FBbEJILDBDQW1CQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgRmlsdGVyVHlwZVZhbHVlIHtcbiAgcmVhZG9ubHkgdHlwZTogJ2NoZWNrYm94JyB8ICdyYWRpbyc7XG59XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBGaWx0ZXJUeXBlIHtcbiAgc3RhdGljIGNoZWNrYm94KCk6IEZpbHRlclR5cGUge1xuICAgIHJldHVybiBuZXcgRmlsdGVyVHlwZUNoZWNrYm94KCk7XG4gIH1cblxuICBzdGF0aWMgcmFkaW8oKTogRmlsdGVyVHlwZSB7XG4gICAgcmV0dXJuIG5ldyBGaWx0ZXJUeXBlUmFkaW8oKTtcbiAgfVxuXG4gIHB1YmxpYyBhYnN0cmFjdCBiaW5kKCk6IEZpbHRlclR5cGVWYWx1ZTtcbn1cblxuY2xhc3MgRmlsdGVyVHlwZVJhZGlvIGV4dGVuZHMgRmlsdGVyVHlwZSB7XG4gIHB1YmxpYyBiaW5kKCk6IEZpbHRlclR5cGVWYWx1ZSB7XG4gICAgcmV0dXJuIHsgdHlwZTogJ3JhZGlvJyB9O1xuICB9XG59XG5cbmNsYXNzIEZpbHRlclR5cGVDaGVja2JveCBleHRlbmRzIEZpbHRlclR5cGUge1xuICBwdWJsaWMgYmluZCgpOiBGaWx0ZXJUeXBlVmFsdWUge1xuICAgIHJldHVybiB7IHR5cGU6ICdjaGVja2JveCcgfTtcbiAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBhY2thZ2VUYWdHcm91cFByb3BzIHtcbiAgLyoqXG4gICAqIEdyb3VwIGxhYmVsIHRvIGRpc3BsYXkuIEZhbGxzIGJhY2sgdG8gaWQgaWYgbm90IHByb3ZpZGVkXG4gICAqL1xuICByZWFkb25seSBsYWJlbD86IHN0cmluZztcbiAgLyoqXG4gICAqIE9wdGlvbmFsIG1lc3NhZ2UgdG8gc2hvdyB3aXRoaW4gYSB0b29sdGlwIG5leHQgdG8gdGhlIGZpbHRlciBsYWJlbFxuICAgKi9cbiAgcmVhZG9ubHkgdG9vbHRpcD86IHN0cmluZztcbiAgLyoqXG4gICAqIEFsbG93cyB0byBzcGVjaWZ5IHRoZSBncm91cCBmaWx0ZXIgdHlwZS4gRGVmYXVsdHMgdG8gY2hlY2tib3ggaWYgbm90IHNwZWNpZmllZFxuICAgKi9cbiAgcmVhZG9ubHkgZmlsdGVyVHlwZT86IEZpbHRlclR5cGU7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFja2FnZVRhZ0dyb3VwQ29uZmlnIHtcbiAgcmVhZG9ubHkgaWQ6IHN0cmluZztcbiAgcmVhZG9ubHkgbGFiZWw/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHRvb2x0aXA/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGZpbHRlclR5cGU/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogRGVmaW5lcyBhIGN1c3RvbSBwYWNrYWdlIHRhZyBncm91cFxuICovXG5leHBvcnQgY2xhc3MgUGFja2FnZVRhZ0dyb3VwIHtcbiAgcHVibGljIHJlYWRvbmx5IGxhYmVsPzogc3RyaW5nO1xuICBwdWJsaWMgcmVhZG9ubHkgdG9vbHRpcD86IHN0cmluZztcbiAgcHVibGljIHJlYWRvbmx5IGZpbHRlclR5cGU/OiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IGlkOiBzdHJpbmcsIHByb3BzPzogUGFja2FnZVRhZ0dyb3VwUHJvcHMpIHtcbiAgICB0aGlzLmxhYmVsID0gcHJvcHM/LmxhYmVsO1xuICAgIHRoaXMudG9vbHRpcCA9IHByb3BzPy50b29sdGlwO1xuICAgIHRoaXMuZmlsdGVyVHlwZSA9IChwcm9wcz8uZmlsdGVyVHlwZSA/PyBGaWx0ZXJUeXBlLmNoZWNrYm94KCkpLmJpbmQoKS50eXBlO1xuICB9XG5cbiAgcHVibGljIGJpbmQoKTogUGFja2FnZVRhZ0dyb3VwQ29uZmlnIHtcbiAgICByZXR1cm4ge1xuICAgICAgaWQ6IHRoaXMuaWQsXG4gICAgICBsYWJlbDogdGhpcy5sYWJlbCxcbiAgICAgIHRvb2x0aXA6IHRoaXMudG9vbHRpcCxcbiAgICAgIGZpbHRlclR5cGU6IHRoaXMuZmlsdGVyVHlwZSxcbiAgICB9O1xuICB9XG59XG4iXX0=
@@ -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.412" };
34
+ PreloadFile[_a] = { fqn: "construct-hub.PreloadFile", version: "0.4.413" };
35
35
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJlbG9hZC1maWxlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsMkJBQWtDO0FBRWxDOzs7R0FHRztBQUNILE1BQWEsV0FBVztJQUN0Qjs7T0FFRztJQUNILE1BQU0sQ0FBQyxRQUFRLENBQUMsSUFBWTtRQUMxQixNQUFNLElBQUksR0FBRyxJQUFBLGlCQUFZLEVBQUMsSUFBSSxFQUFFLEVBQUUsUUFBUSxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFDdkQsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNLENBQUMsUUFBUSxDQUFDLElBQVk7UUFDMUIsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBRUQsWUFBNkIsSUFBWTtRQUFaLFNBQUksR0FBSixJQUFJLENBQVE7SUFBRyxDQUFDO0lBRXRDLElBQUk7UUFDVCxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDbkIsQ0FBQzs7QUFwQkgsa0NBcUJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgcmVhZEZpbGVTeW5jIH0gZnJvbSAnZnMnO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgYSBqYXZhc2NyaXB0IGZpbGUgdG8gbG9hZCBiZWZvcmUgdGhlIHdlYmFwcC5cbiAqIFRoaXMgY2FuIGFsbG93IG9wZXJhdG9ycyB0byBhZGQgdGhlaXIgb3duIGNsaWVudCBtb25pdG9ycyBvciBhbmFseXRpY3MgaWYgdGhleSB3aXNoXG4gKi9cbmV4cG9ydCBjbGFzcyBQcmVsb2FkRmlsZSB7XG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgUHJlbG9hZEZpbGUgaW5zdGFuY2UgZnJvbSBhIGZpbGVwYXRoIHRvIGxvYWRcbiAgICovXG4gIHN0YXRpYyBmcm9tRmlsZShwYXRoOiBzdHJpbmcpOiBQcmVsb2FkRmlsZSB7XG4gICAgY29uc3QgZGF0YSA9IHJlYWRGaWxlU3luYyhwYXRoLCB7IGVuY29kaW5nOiAndXRmLTgnIH0pO1xuICAgIHJldHVybiBuZXcgUHJlbG9hZEZpbGUoZGF0YSk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyBhIFByZWxvYWRGaWxlIGluc3RhbmNlIGRpcmVjdGx5IGZyb20gc291cmNlIGNvZGVcbiAgICovXG4gIHN0YXRpYyBmcm9tQ29kZShjb2RlOiBzdHJpbmcpOiBQcmVsb2FkRmlsZSB7XG4gICAgcmV0dXJuIG5ldyBQcmVsb2FkRmlsZShjb2RlKTtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVhZG9ubHkgZGF0YTogc3RyaW5nKSB7fVxuXG4gIHB1YmxpYyBiaW5kKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YTtcbiAgfVxufVxuIl19
package/lib/s3/storage.js CHANGED
@@ -52,6 +52,6 @@ class S3StorageFactory extends constructs_1.Construct {
52
52
  }
53
53
  exports.S3StorageFactory = S3StorageFactory;
54
54
  _a = JSII_RTTI_SYMBOL_1;
55
- S3StorageFactory[_a] = { fqn: "construct-hub.S3StorageFactory", version: "0.4.412" };
55
+ S3StorageFactory[_a] = { fqn: "construct-hub.S3StorageFactory", version: "0.4.413" };
56
56
  S3StorageFactory.UID = 'S3StorageFactory';
57
57
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zMy9zdG9yYWdlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkNBQXFEO0FBQ3JELHlDQUF5QztBQUN6QywyQ0FBdUM7QUFjdkM7O0dBRUc7QUFDSCxNQUFhLGdCQUFpQixTQUFRLHNCQUFTO0lBQzdDOzs7O09BSUc7SUFDSSxNQUFNLENBQUMsV0FBVyxDQUN2QixLQUFnQixFQUNoQixRQUErQixFQUFFO1FBRWpDLE1BQU0sS0FBSyxHQUFHLG1CQUFLLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzlCLE1BQU0sT0FBTyxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLGdCQUFnQixDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQzlELElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNiLE9BQU8sSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsZ0JBQWdCLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxDQUFDO1FBQ2xFLENBQUM7UUFDRCxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLGdCQUFnQixDQUFDLEdBQUcsQ0FBcUIsQ0FBQztJQUN4RSxDQUFDO0lBTUQsWUFDRSxLQUFnQixFQUNoQixFQUFVLEVBQ1YsUUFBK0IsRUFBRTtRQUVqQyxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ2pCLElBQUksQ0FBQyxjQUFjLEdBQUcsS0FBSyxDQUFDLFFBQVEsSUFBSSxLQUFLLENBQUM7SUFDaEQsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxTQUFTLENBQ2QsS0FBZ0IsRUFDaEIsRUFBVSxFQUNWLEtBQXNCO1FBRXRCLFNBQVMsV0FBVyxDQUFDLE1BQWlCO1lBQ3BDLE1BQU0sU0FBUyxHQUFHLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsV0FBVyxFQUFFLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztZQUMvRCxrQkFBSSxDQUFDLEVBQUUsQ0FBQyxTQUFTLENBQUMsQ0FBQyxHQUFHLENBQUMsVUFBVSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBRTNDLElBQUksdUJBQVMsQ0FBQyxLQUFLLEVBQUUsaUJBQWlCLEVBQUU7Z0JBQ3RDLFdBQVcsRUFBRSxZQUFZLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFO2dCQUMzQyxLQUFLLEVBQUUsb0JBQW9CLE1BQU0sQ0FBQyxVQUFVLFNBQVMsU0FBUyxDQUFDLFVBQVUsRUFBRTthQUM1RSxDQUFDLENBQUM7WUFDSCxPQUFPLFNBQVMsQ0FBQztRQUNuQixDQUFDO1FBRUQsTUFBTSxPQUFPLEdBQUcsSUFBSSxFQUFFLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFFaEQsNkVBQTZFO1FBQzdFLGtGQUFrRjtRQUNsRix5QkFBeUI7UUFDekIsTUFBTSxRQUFRLEdBQUcsV0FBVyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBRXRDLE9BQU8sSUFBSSxDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUM7SUFDbEQsQ0FBQzs7QUE1REgsNENBNkRDOzs7QUEzQ3lCLG9CQUFHLEdBQUcsa0JBQWtCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDZm5PdXRwdXQsIFN0YWNrLCBUYWdzIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0ICogYXMgczMgZnJvbSAnYXdzLWNkay1saWIvYXdzLXMzJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIFByb3BlcnRpZXMgZm9yIGBTM1N0b3JhZ2VGYWN0b3J5YFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFMzU3RvcmFnZUZhY3RvcnlQcm9wcyB7XG4gIC8qKlxuICAgKiBXaGVuIGVuYWJsZWQsIHRoZSBmYWN0b3J5IHdpbGwgcmV0dXJuIHRoZSBmYWlsb3ZlciBidWNrZXRzIGluc3RlYWQgb2YgdGhlIHByaW1hcnkuXG4gICAqXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICByZWFkb25seSBmYWlsb3Zlcj86IGJvb2xlYW47XG59XG5cbi8qKlxuICogQ3JlYXRlIHMzIHN0b3JhZ2UgcmVzb3VyY2VzLlxuICovXG5leHBvcnQgY2xhc3MgUzNTdG9yYWdlRmFjdG9yeSBleHRlbmRzIENvbnN0cnVjdCB7XG4gIC8qKlxuICAgKiBSZXRyaWV2ZSBvciBjcmVhdGUgdGhlIHN0b3JhZ2UgZmFjdG9yeSBmb3IgdGhlIGN1cnJlbnQgc2NvcGUuXG4gICAqXG4gICAqIFRoaXMgaXMgc3RhY2sgc2luZ2xldG9uLlxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBnZXRPckNyZWF0ZShcbiAgICBzY29wZTogQ29uc3RydWN0LFxuICAgIHByb3BzOiBTM1N0b3JhZ2VGYWN0b3J5UHJvcHMgPSB7fVxuICApOiBTM1N0b3JhZ2VGYWN0b3J5IHtcbiAgICBjb25zdCBzdGFjayA9IFN0YWNrLm9mKHNjb3BlKTtcbiAgICBjb25zdCBmYWN0b3J5ID0gc3RhY2subm9kZS50cnlGaW5kQ2hpbGQoUzNTdG9yYWdlRmFjdG9yeS5VSUQpO1xuICAgIGlmICghZmFjdG9yeSkge1xuICAgICAgcmV0dXJuIG5ldyBTM1N0b3JhZ2VGYWN0b3J5KHN0YWNrLCBTM1N0b3JhZ2VGYWN0b3J5LlVJRCwgcHJvcHMpO1xuICAgIH1cbiAgICByZXR1cm4gc3RhY2subm9kZS5maW5kQ2hpbGQoUzNTdG9yYWdlRmFjdG9yeS5VSUQpIGFzIFMzU3RvcmFnZUZhY3Rvcnk7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBVSUQgPSAnUzNTdG9yYWdlRmFjdG9yeSc7XG5cbiAgcHJpdmF0ZSBmYWlsb3ZlckFjdGl2ZTogYm9vbGVhbjtcblxuICBwcml2YXRlIGNvbnN0cnVjdG9yKFxuICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogUzNTdG9yYWdlRmFjdG9yeVByb3BzID0ge31cbiAgKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcbiAgICB0aGlzLmZhaWxvdmVyQWN0aXZlID0gcHJvcHMuZmFpbG92ZXIgPz8gZmFsc2U7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlIGEgbmV3IGJ1Y2tldCBpbiBhIHN0b3JhZ2UgY29uZmlnIGF3YXJlIG1hbm5lci5cbiAgICpcbiAgICogQHJldHVybnMgczMuQnVja2V0XG4gICAqL1xuICBwdWJsaWMgbmV3QnVja2V0KFxuICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wcz86IHMzLkJ1Y2tldFByb3BzXG4gICk6IHMzLkJ1Y2tldCB7XG4gICAgZnVuY3Rpb24gZmFpbG92ZXJGb3IoYnVja2V0OiBzMy5CdWNrZXQpOiBzMy5CdWNrZXQge1xuICAgICAgY29uc3QgX2ZhaWxvdmVyID0gbmV3IHMzLkJ1Y2tldChzY29wZSwgYEZhaWxvdmVyJHtpZH1gLCBwcm9wcyk7XG4gICAgICBUYWdzLm9mKF9mYWlsb3ZlcikuYWRkKCdmYWlsb3ZlcicsICd0cnVlJyk7XG5cbiAgICAgIG5ldyBDZm5PdXRwdXQoc2NvcGUsICdTbmFwc2hvdENvbW1hbmQnLCB7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgU25hcHNob3QgJHtidWNrZXQubm9kZS5wYXRofWAsXG4gICAgICAgIHZhbHVlOiBgYXdzIHMzIHN5bmMgczM6Ly8ke2J1Y2tldC5idWNrZXROYW1lfSBzMzovLyR7X2ZhaWxvdmVyLmJ1Y2tldE5hbWV9YCxcbiAgICAgIH0pO1xuICAgICAgcmV0dXJuIF9mYWlsb3ZlcjtcbiAgICB9XG5cbiAgICBjb25zdCBwcmltYXJ5ID0gbmV3IHMzLkJ1Y2tldChzY29wZSwgaWQsIHByb3BzKTtcblxuICAgIC8vIG5vdGUgdGhhdCB3ZSBjcmVhdGUgdGhlIGZhaWxvdmVyIGJ1Y2tldCBldmVuIGlmIHdlIGRvbid0IGN1cnJlbnRseSB1c2UgaXQuXG4gICAgLy8gdGhpcyBpcyBiZWNhdXNlIGNvbmRpdGlvbmluZyBidWNrZXQgY3JlYXRpb24gd2lsbCBldmVudHVhbGx5IGZhaWwgc2luY2UgYnVja2V0c1xuICAgIC8vIGFyZSBub3JtYWxseSByZXRhaW5lZC5cbiAgICBjb25zdCBmYWlsb3ZlciA9IGZhaWxvdmVyRm9yKHByaW1hcnkpO1xuXG4gICAgcmV0dXJuIHRoaXMuZmFpbG92ZXJBY3RpdmUgPyBmYWlsb3ZlciA6IHByaW1hcnk7XG4gIH1cbn1cbiJdfQ==
@@ -269,7 +269,7 @@ class SpdxLicense {
269
269
  }
270
270
  exports.SpdxLicense = SpdxLicense;
271
271
  _a = JSII_RTTI_SYMBOL_1;
272
- SpdxLicense[_a] = { fqn: "construct-hub.SpdxLicense", version: "0.4.412" };
272
+ SpdxLicense[_a] = { fqn: "construct-hub.SpdxLicense", version: "0.4.413" };
273
273
  SpdxLicense._ALL = new Map();
274
274
  //#region Individual SPDX Licenses
275
275
  /**
package/package.json CHANGED
@@ -209,7 +209,7 @@
209
209
  "publishConfig": {
210
210
  "access": "public"
211
211
  },
212
- "version": "0.4.412",
212
+ "version": "0.4.413",
213
213
  "jest": {
214
214
  "coverageProvider": "v8",
215
215
  "maxConcurrency": 2,