construct-hub 0.4.397 → 0.4.398

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.
package/.jsii CHANGED
@@ -20923,6 +20923,6 @@
20923
20923
  "symbolId": "src/package-sources/npmjs:NpmJsProps"
20924
20924
  }
20925
20925
  },
20926
- "version": "0.4.397",
20927
- "fingerprint": "6YUfXnpOqCDWjh7nJ6l86x8NaXs5qI8b3Saa4hldtn0="
20926
+ "version": "0.4.398",
20927
+ "fingerprint": "spKM6o8+bIrYc435knKy3BHBbvbEsG6CogHQEyFL5Us="
20928
20928
  }
@@ -393,7 +393,7 @@ class ConstructHub extends constructs_1.Construct {
393
393
  }
394
394
  exports.ConstructHub = ConstructHub;
395
395
  _a = JSII_RTTI_SYMBOL_1;
396
- ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.397" };
396
+ ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.398" };
397
397
  /**
398
398
  * How possibly risky operations (such as doc-generation, which requires
399
399
  * installing the indexed packages in order to trans-literate sample code) are
@@ -278,5 +278,5 @@ class CodeArtifact {
278
278
  }
279
279
  exports.CodeArtifact = CodeArtifact;
280
280
  _a = JSII_RTTI_SYMBOL_1;
281
- CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.397" };
281
+ CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.398" };
282
282
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-artifact.js","sourceRoot":"","sources":["../../src/package-sources/code-artifact.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnFormat, Aws, Duration, Stack } from 'aws-cdk-lib';\nimport {\n  ComparisonOperator,\n  GraphWidget,\n  MathExpression,\n  Metric,\n  MetricOptions,\n  Statistic,\n  TreatMissingData,\n} from 'aws-cdk-lib/aws-cloudwatch';\nimport { CfnRepository } from 'aws-cdk-lib/aws-codeartifact';\nimport { Rule } from 'aws-cdk-lib/aws-events';\nimport { LambdaFunction } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Tracing } from 'aws-cdk-lib/aws-lambda';\nimport { BlockPublicAccess, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { Construct } from 'constructs';\nimport {\n  codeArtifactRepositoryUrl,\n  lambdaFunctionUrl,\n  lambdaSearchLogGroupUrl,\n  sqsQueueUrl,\n} from '../deep-link';\nimport { fillMetric } from '../metric-utils';\nimport type {\n  IPackageSource,\n  PackageSourceBindOptions,\n  PackageSourceBindResult,\n} from '../package-source';\nimport { CodeArtifactForwarder } from './codeartifact/code-artifact-forwarder';\nimport {\n  METRICS_NAMESPACE,\n  MetricName,\n  DOMAIN_NAME_DIMENSION,\n  DOMAIN_OWNER_DIMENSION,\n  REPOSITORY_NAME_DIMENSION,\n} from './codeartifact/constants.lambda-shared';\nimport { S3StorageFactory } from '../s3/storage';\n\nexport interface CodeArtifactProps {\n  /**\n   * The CodeArtifact repository where packages are obtained from.\n   */\n  readonly repository: CfnRepository;\n\n  /**\n   * The S3 bucket where packages will be staged.\n   */\n  readonly bucket?: IBucket;\n}\n\n/**\n * A package source that obtains package data from an npm CodeArtifact\n * repository.\n */\nexport class CodeArtifact implements IPackageSource {\n  public constructor(private readonly props: CodeArtifactProps) {}\n\n  public bind(\n    scope: Construct,\n    {\n      denyList,\n      ingestion,\n      licenseList,\n      monitoring,\n      overviewDashboard,\n      queue,\n    }: PackageSourceBindOptions\n  ): PackageSourceBindResult {\n    const idPrefix = this.props.repository.node.path;\n    const repositoryId = `${this.props.repository.attrDomainOwner}:${this.props.repository.attrDomainName}/${this.props.repository.attrName}`;\n\n    const storageFactory = S3StorageFactory.getOrCreate(scope);\n    const bucket =\n      this.props.bucket ||\n      storageFactory.newBucket(scope, `${idPrefix}/StagingBucket`, {\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        enforceSSL: true,\n        lifecycleRules: [{ expiration: Duration.days(30) }],\n      });\n    bucket.grantRead(ingestion);\n\n    const dlq = new Queue(scope, `${idPrefix}/DLQ`, {\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: Duration.days(14),\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    const forwarder = new CodeArtifactForwarder(\n      scope,\n      `${idPrefix}/Forwarder`,\n      {\n        deadLetterQueue: dlq,\n        description: `[${scope.node.path}/CodeArtifact/${repositoryId}] Handle CodeArtifact EventBridge events`,\n        environment: {\n          AWS_EMF_ENVIRONMENT: 'Local',\n          BUCKET_NAME: bucket.bucketName,\n          QUEUE_URL: queue.queueUrl,\n        },\n        memorySize: 1024,\n        timeout: Duration.seconds(60),\n        tracing: Tracing.ACTIVE,\n      }\n    );\n    bucket.grantReadWrite(forwarder);\n    denyList?.grantRead(forwarder);\n    licenseList.grantRead(forwarder);\n    queue.grantSendMessages(forwarder);\n    forwarder.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['codeartifact:GetPackageVersionAsset'],\n        resources: [\n          Stack.of(scope).formatArn({\n            service: 'codeartifact',\n            resource: 'package',\n            arnFormat: ArnFormat.SLASH_RESOURCE_NAME,\n            resourceName: [\n              this.props.repository.attrDomainName,\n              this.props.repository.attrName,\n              'npm', // package format\n              '*', // namespace/package-name\n            ].join('/'),\n          }),\n        ],\n      })\n    );\n\n    const rule = new Rule(scope, `${idPrefix}/EventBridge`, {\n      description: `${scope.node.path}/CodeArtifact/${repositoryId}/EventBridge`,\n      eventPattern: {\n        source: ['aws.codeartifact'],\n        detailType: ['CodeArtifact Package Version State Change'],\n        detail: {\n          domainOwner: [this.props.repository.attrDomainOwner],\n          domainName: [this.props.repository.attrDomainName],\n          repositoryName: [this.props.repository.attrName],\n          packageFormat: ['npm'],\n        },\n      },\n      targets: [new LambdaFunction(forwarder)],\n    });\n\n    const failureAlarm = forwarder\n      .metricErrors()\n      .createAlarm(scope, `${idPrefix}/Forwarder/Failures`, {\n        alarmName: `${scope.node.path}/CodeArtifact/${repositoryId}/Forwarder`,\n        alarmDescription: [\n          `The CodeArtifact fowarder for ${repositoryId} is failing`,\n          '',\n          `Link to the lambda function: ${lambdaFunctionUrl(forwarder)}`,\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.addHighSeverityAlarm(\n      `CodeArtifact:${this.props.repository.attrName} Failures`,\n      failureAlarm\n    );\n\n    const dlqNotEmptyAlarm = new MathExpression({\n      expression: 'mVisible + mHidden',\n      usingMetrics: {\n        mVisible: dlq.metricApproximateNumberOfMessagesVisible({\n          period: Duration.minutes(1),\n        }),\n        mHidden: dlq.metricApproximateNumberOfMessagesNotVisible({\n          period: Duration.minutes(1),\n        }),\n      },\n    }).createAlarm(scope, `${idPrefix}/Forwarder/DLQNotEmpty`, {\n      alarmName: `${scope.node.path}/CodeArtifact/${repositoryId}/DLQNotEmpty`,\n      alarmDescription: [\n        `The CodeArtifact fowarder for ${repositoryId} is failing`,\n        '',\n        `Link to the lambda function: ${lambdaFunctionUrl(forwarder)}`,\n        `Link to the dead letter queue: ${sqsQueueUrl(dlq)}`,\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      `CodeArtifact/${repositoryId} DLQ Not Empty`,\n      dlqNotEmptyAlarm\n    );\n\n    rule.node.addDependency(failureAlarm, dlqNotEmptyAlarm);\n\n    overviewDashboard.addDLQMetricToDashboard(\n      `CodeArtifact/${repositoryId} DLQ`,\n      dlq\n    );\n    overviewDashboard.addConcurrentExecutionMetricToDashboard(\n      forwarder,\n      `${idPrefix}/ForwarderLambda`\n    );\n\n    return {\n      name: `CodeArtifact: ${repositoryId}`,\n      links: [\n        {\n          name: 'CodeArtifact',\n          url: codeArtifactRepositoryUrl(this.props.repository),\n          primary: true,\n        },\n        {\n          name: 'Forwarder Function',\n          url: lambdaFunctionUrl(forwarder),\n        },\n        {\n          name: 'Search Log group',\n          url: lambdaSearchLogGroupUrl(forwarder),\n        },\n        {\n          name: 'DLQ',\n          url: sqsQueueUrl(dlq),\n        },\n      ],\n      dashboardWidgets: [\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Function Health',\n            left: [\n              fillMetric(forwarder.metricInvocations({ label: 'Invocations' })),\n              fillMetric(forwarder.metricErrors({ label: 'Errors' })),\n            ],\n            leftYAxis: { min: 0 },\n            right: [forwarder.metricDuration({ label: 'Duration' })],\n            rightYAxis: { min: 0 },\n            period: Duration.minutes(15),\n          }),\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Dead Letter Queue',\n            left: [\n              dlq.metricApproximateNumberOfMessagesVisible({\n                label: 'Visible Messages',\n                period: Duration.minutes(1),\n              }),\n              dlq.metricApproximateNumberOfMessagesNotVisible({\n                label: 'Hidden Messages',\n                period: Duration.minutes(1),\n              }),\n            ],\n            leftYAxis: { min: 0 },\n            right: [\n              dlq.metricApproximateAgeOfOldestMessage({\n                label: 'Oldest Message Age',\n                period: Duration.minutes(1),\n              }),\n            ],\n            rightYAxis: { min: 0 },\n          }),\n        ],\n        [\n          new GraphWidget({\n            height: 6,\n            width: 12,\n            title: 'Quality Metrics',\n            left: [\n              fillMetric(\n                this.metricNotJsiiEnabledCount({ label: 'Not a jsii package' }),\n                0\n              ),\n              fillMetric(\n                this.metricIneligibleLicense({ label: 'Ineligible License' }),\n                0\n              ),\n              fillMetric(\n                this.metricDenyListedCount({ label: 'Deny Listed' }),\n                0\n              ),\n              fillMetric(\n                this.metricDeletedCount({ label: 'Deletion Events' }),\n                0\n              ),\n            ],\n            leftYAxis: { min: 0 },\n          }),\n        ],\n      ],\n    };\n  }\n\n  /**\n   * The count of package versions that were ignored due to being in the deny list.\n   */\n  public metricDenyListedCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]: this.props.repository.attrDomainOwner,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.DENY_LISTED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that were ignored due to using an ineloigible license.\n   */\n  public metricIneligibleLicense(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.INELIGIBLE_LICENSE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that were deleted from CodeArtifact (those events are not\n   * handled currently).\n   */\n  public metricDeletedCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.DELETED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * The number of package versions that do not have a jsii assembly in the package.\n   */\n  public metricNotJsiiEnabledCount(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      dimensionsMap: {\n        [DOMAIN_OWNER_DIMENSION]:\n          this.props.repository.attrDomainOwner ?? Aws.ACCOUNT_ID,\n        [DOMAIN_NAME_DIMENSION]: this.props.repository.attrDomainName,\n        [REPOSITORY_NAME_DIMENSION]: this.props.repository.attrName,\n      },\n      metricName: MetricName.NOT_JSII_ENABLED_COUNT,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
@@ -63,9 +63,18 @@ class CouchChanges extends events_1.EventEmitter {
63
63
  }
64
64
  const metadataUrl = new url_1.URL(change.id, NPM_REGISTRY_URL);
65
65
  console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);
66
- const meta = await this.https('get', metadataUrl);
67
- change.doc = meta; // add metadata to the change object
68
- return change;
66
+ try {
67
+ const meta = await this.https('get', metadataUrl);
68
+ change.doc = meta; // add metadata to the change object
69
+ return change;
70
+ }
71
+ catch (e) {
72
+ if (e.errorMessage.includes('HTTP 404')) {
73
+ console.log(`Skipping ${change.id} because of 404 error:\n${e}`);
74
+ return;
75
+ }
76
+ throw e;
77
+ }
69
78
  }
70
79
  async fetchAndFilterAllMetadata(changes) {
71
80
  return (await Promise.all(changes.map((change) => this.fetchAndFilterMetadata(change)))).filter((change) => change !== undefined);
@@ -99,7 +108,6 @@ class CouchChanges extends events_1.EventEmitter {
99
108
  if (body) {
100
109
  headers['Content-Type'] = 'application/json';
101
110
  }
102
- console.log(`Request: ${method.toUpperCase()} ${url}, ${JSON.stringify(headers)}`);
103
111
  const req = (0, https_1.request)(url, {
104
112
  agent: this.agent,
105
113
  headers,
@@ -155,4 +163,4 @@ function gunzip(readable) {
155
163
  readable.pipe(gz, { end: true });
156
164
  return gz;
157
165
  }
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"couch-changes.lambda-shared.js","sourceRoot":"","sources":["../../../src/package-sources/npmjs/couch-changes.lambda-shared.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport { OutgoingHttpHeaders } from 'http';\nimport { Agent, request } from 'https';\nimport { Readable } from 'stream';\nimport { URL } from 'url';\nimport { createGunzip } from 'zlib';\nimport * as JSONStream from 'JSONStream';\n\nconst NPM_REGISTRY_URL = 'https://registry.npmjs.org/';\n\n/**\n * A utility class that helps with traversing CouchDB database changes streams\n * in a promise-based, page-by-page manner.\n */\nexport class CouchChanges extends EventEmitter {\n  private readonly agent: Agent;\n  private readonly baseUrl: URL;\n  private readonly databaseUrl: URL;\n\n  /**\n   * @param baseUrl  the CouchDB endpoint URL.\n   * @param database the name of the database for which changes are fetched.\n   */\n  public constructor(baseUrl: string, database: string) {\n    super();\n    // Setting up for keep-alive connections.\n    this.agent = new Agent({\n      keepAlive: true,\n      keepAliveMsecs: 5_000,\n      maxSockets: 4,\n      timeout: 60_000,\n    });\n    this.baseUrl = new URL(baseUrl);\n    this.databaseUrl = new URL(database, baseUrl);\n  }\n\n  /**\n   * @returns summary informations about the database.\n   */\n  public async info(): Promise<DatabaseInfos> {\n    return (await this.https('get', this.baseUrl)) as any;\n  }\n\n  /**\n   * Obtains a batch of changes from the database.\n   *\n   * @param since     the sequence value since when history should be fetched.\n   * @param batchSize the maximum amount of changes to return in a single page.\n   *\n   * @returns a page of changes.\n   */\n  public async changes(\n    since: string | number,\n    opts?: { readonly batchSize?: number }\n  ): Promise<DatabaseChanges> {\n    const batchSize = opts?.batchSize ?? 100;\n\n    const changesUrl = new URL('_changes', this.databaseUrl);\n    changesUrl.searchParams.set('limit', batchSize.toFixed());\n    changesUrl.searchParams.set('since', since.toString());\n\n    const result = (await this.https('get', changesUrl)) as any;\n\n    const last_seq = result.last_seq;\n    const results = await this.fetchAndFilterAllMetadata(result.results);\n\n    return {\n      last_seq,\n      results,\n    };\n  }\n\n  private async fetchAndFilterMetadata(change: DatabaseChange) {\n    // Filter out deleted packages or null ids\n    if (change.deleted || !change.id) {\n      console.log(`Skipping ${change.id}: deleted or null id`);\n      return;\n    }\n\n    const metadataUrl = new URL(change.id, NPM_REGISTRY_URL);\n    console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);\n    const meta = await this.https('get', metadataUrl);\n    change.doc = meta; // add metadata to the change object\n    return change;\n  }\n\n  private async fetchAndFilterAllMetadata(\n    changes: DatabaseChange[]\n  ): Promise<DatabaseChange[]> {\n    return (\n      await Promise.all(\n        changes.map((change) => this.fetchAndFilterMetadata(change))\n      )\n    ).filter((change): change is DatabaseChange => change !== undefined);\n  }\n\n  /**\n   * Makes an HTTPs request using the provided method, url, and optionally payload. This function\n   * properly handles input that is received with `Content-Type: gzip` and automatically retries\n   * typical transient errors (HTTP 5XX, ECONNRESET, etc...) with linear back-off and no maximum\n   * retry count (this is used in Lambda functions, which de-facto caps the amount of attempts\n   * that will be made due to the function time out).\n   *\n   * @param method the HTTP method used for the request (e.g: 'get', 'post', ...).\n   * @param url    the URL to request.\n   * @param body   an optional HTTP request payload, which will be JSON-encoded.\n   *\n   * @param attempt the request attempt number (used to determine back-off / retry).\n   *\n   * @returns the JSON-decoded response body.\n   */\n  private https(\n    method: string,\n    url: URL,\n    body?: { [key: string]: unknown },\n    attempt = 1\n  ): Promise<{ [key: string]: unknown }> {\n    return new Promise((ok, ko) => {\n      const retry = () =>\n        setTimeout(() => {\n          console.log(`Retrying ${method.toUpperCase()} ${url}`);\n          this.https(method, url, body, attempt + 1).then(ok, ko);\n        }, Math.min(500 * attempt, 5_000));\n\n      const headers: OutgoingHttpHeaders = {\n        Accept: 'application/json',\n        'Accept-Encoding': 'gzip',\n        'npm-replication-opt-in': 'true', // can be deleted after May 29: https://github.com/orgs/community/discussions/152515\n      };\n      if (body) {\n        headers['Content-Type'] = 'application/json';\n      }\n      console.log(\n        `Request: ${method.toUpperCase()} ${url}, ${JSON.stringify(headers)}`\n      );\n      const req = request(\n        url,\n        {\n          agent: this.agent,\n          headers,\n          method,\n          port: 443,\n          servername: url.hostname,\n        },\n        (res) => {\n          if (res.statusCode == null) {\n            const error = new Error(\n              `[FATAL] Request failed: ${method.toUpperCase()} ${url}`\n            );\n            Error.captureStackTrace(error);\n            return ko(error);\n          }\n\n          console.log(\n            `Response: ${method.toUpperCase()} ${url} => HTTP ${\n              res.statusCode\n            } (${res.statusMessage})`\n          );\n\n          // Transient (server) errors:\n          if (res.statusCode >= 500 && res.statusCode < 600) {\n            console.error(\n              `[RETRYABLE] HTTP ${res.statusCode} (${\n                res.statusMessage\n              }) - ${method.toUpperCase()} ${url}`\n            );\n            // Call again after a short back-off\n            return retry();\n          }\n          // Permanent (client) errors:\n          if (res.statusCode >= 400 && res.statusCode < 500) {\n            const error = new Error(\n              `[FATAL] HTTP ${res.statusCode} (${\n                res.statusMessage\n              }) - ${method.toUpperCase()} ${url}`\n            );\n            Error.captureStackTrace(error);\n            return ko(error);\n          }\n\n          const onError = (err: Error & { code?: string }) => {\n            if (err.code === 'ECONNRESET') {\n              // Transient networking problem?\n              console.error(\n                `[RETRYABLE] ${err.code} - ${method.toUpperCase()} ${url}`\n              );\n              retry();\n            } else {\n              Error.captureStackTrace(err);\n              console.log('[NON-RETRYABLE]', err);\n              ko(err);\n            }\n          };\n\n          res.once('error', onError);\n\n          const json = JSONStream.parse(true);\n          json.once('data', ok);\n          json.once('error', onError);\n\n          const plainPayload =\n            res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n          plainPayload.pipe(json, { end: true });\n          plainPayload.once('error', onError);\n        }\n      );\n      req.end(body && JSON.stringify(body, null, 2));\n    });\n  }\n}\n\nexport interface DatabaseChanges {\n  /**\n   * The last sequence ID from this change set. This is the value that should be\n   * passed to the subsequent `.changes` call to fetch the next page.\n   */\n  readonly last_seq: string | number;\n\n  /**\n   * The amount of pending changes from the server. This value is not always\n   * returned by the servers.\n   */\n  readonly pending?: number;\n\n  /**\n   * The changes that are part of this batch.\n   */\n  readonly results: readonly DatabaseChange[];\n}\n\nexport interface DatabaseChange {\n  /**\n   * The set of revisions to the object that were resolved as part of this\n   * change.\n   */\n  readonly changes: ReadonlyArray<{ readonly rev: string }>;\n\n  /**\n   * The ID of the document that has changed.\n   */\n  readonly id: string;\n\n  /**\n   * The sequence ID for this change in the stream. It may not be present for\n   * all (or any) entries in the result.\n   */\n  readonly seq?: string | number;\n\n  /**\n   * Whether this change corresponds to this document being deleted.\n   */\n  readonly deleted: boolean;\n\n  /**\n   * If present, the resolved document after the change has been applied.\n   */\n  doc?: { readonly [key: string]: unknown };\n}\n\nexport interface DatabaseInfos {\n  readonly db_name: string;\n  readonly disk_format_version: number;\n  readonly doc_count: number;\n  readonly doc_del_count: number;\n  readonly instance_start_time: string;\n  readonly purge_seq: string | number;\n  readonly sizes: {\n    readonly active: number;\n    readonly external: number;\n    readonly file: number;\n  };\n  readonly update_seq: string | number;\n}\n\nfunction gunzip(readable: Readable): Readable {\n  const gz = createGunzip();\n  readable.pipe(gz, { end: true });\n  return gz;\n}\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"couch-changes.lambda-shared.js","sourceRoot":"","sources":["../../../src/package-sources/npmjs/couch-changes.lambda-shared.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport { OutgoingHttpHeaders } from 'http';\nimport { Agent, request } from 'https';\nimport { Readable } from 'stream';\nimport { URL } from 'url';\nimport { createGunzip } from 'zlib';\nimport * as JSONStream from 'JSONStream';\n\nconst NPM_REGISTRY_URL = 'https://registry.npmjs.org/';\n\n/**\n * A utility class that helps with traversing CouchDB database changes streams\n * in a promise-based, page-by-page manner.\n */\nexport class CouchChanges extends EventEmitter {\n  private readonly agent: Agent;\n  private readonly baseUrl: URL;\n  private readonly databaseUrl: URL;\n\n  /**\n   * @param baseUrl  the CouchDB endpoint URL.\n   * @param database the name of the database for which changes are fetched.\n   */\n  public constructor(baseUrl: string, database: string) {\n    super();\n    // Setting up for keep-alive connections.\n    this.agent = new Agent({\n      keepAlive: true,\n      keepAliveMsecs: 5_000,\n      maxSockets: 4,\n      timeout: 60_000,\n    });\n    this.baseUrl = new URL(baseUrl);\n    this.databaseUrl = new URL(database, baseUrl);\n  }\n\n  /**\n   * @returns summary informations about the database.\n   */\n  public async info(): Promise<DatabaseInfos> {\n    return (await this.https('get', this.baseUrl)) as any;\n  }\n\n  /**\n   * Obtains a batch of changes from the database.\n   *\n   * @param since     the sequence value since when history should be fetched.\n   * @param batchSize the maximum amount of changes to return in a single page.\n   *\n   * @returns a page of changes.\n   */\n  public async changes(\n    since: string | number,\n    opts?: { readonly batchSize?: number }\n  ): Promise<DatabaseChanges> {\n    const batchSize = opts?.batchSize ?? 100;\n\n    const changesUrl = new URL('_changes', this.databaseUrl);\n    changesUrl.searchParams.set('limit', batchSize.toFixed());\n    changesUrl.searchParams.set('since', since.toString());\n\n    const result = (await this.https('get', changesUrl)) as any;\n\n    const last_seq = result.last_seq;\n    const results = await this.fetchAndFilterAllMetadata(result.results);\n\n    return {\n      last_seq,\n      results,\n    };\n  }\n\n  private async fetchAndFilterMetadata(change: DatabaseChange) {\n    // Filter out deleted packages or null ids\n    if (change.deleted || !change.id) {\n      console.log(`Skipping ${change.id}: deleted or null id`);\n      return;\n    }\n\n    const metadataUrl = new URL(change.id, NPM_REGISTRY_URL);\n    console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);\n\n    try {\n      const meta = await this.https('get', metadataUrl);\n      change.doc = meta; // add metadata to the change object\n      return change;\n    } catch (e: any) {\n      if (e.errorMessage.includes('HTTP 404')) {\n        console.log(`Skipping ${change.id} because of 404 error:\\n${e}`);\n        return;\n      }\n      throw e;\n    }\n  }\n\n  private async fetchAndFilterAllMetadata(\n    changes: DatabaseChange[]\n  ): Promise<DatabaseChange[]> {\n    return (\n      await Promise.all(\n        changes.map((change) => this.fetchAndFilterMetadata(change))\n      )\n    ).filter((change): change is DatabaseChange => change !== undefined);\n  }\n\n  /**\n   * Makes an HTTPs request using the provided method, url, and optionally payload. This function\n   * properly handles input that is received with `Content-Type: gzip` and automatically retries\n   * typical transient errors (HTTP 5XX, ECONNRESET, etc...) with linear back-off and no maximum\n   * retry count (this is used in Lambda functions, which de-facto caps the amount of attempts\n   * that will be made due to the function time out).\n   *\n   * @param method the HTTP method used for the request (e.g: 'get', 'post', ...).\n   * @param url    the URL to request.\n   * @param body   an optional HTTP request payload, which will be JSON-encoded.\n   *\n   * @param attempt the request attempt number (used to determine back-off / retry).\n   *\n   * @returns the JSON-decoded response body.\n   */\n  private https(\n    method: string,\n    url: URL,\n    body?: { [key: string]: unknown },\n    attempt = 1\n  ): Promise<{ [key: string]: unknown }> {\n    return new Promise((ok, ko) => {\n      const retry = () =>\n        setTimeout(() => {\n          console.log(`Retrying ${method.toUpperCase()} ${url}`);\n          this.https(method, url, body, attempt + 1).then(ok, ko);\n        }, Math.min(500 * attempt, 5_000));\n\n      const headers: OutgoingHttpHeaders = {\n        Accept: 'application/json',\n        'Accept-Encoding': 'gzip',\n        'npm-replication-opt-in': 'true', // can be deleted after May 29: https://github.com/orgs/community/discussions/152515\n      };\n      if (body) {\n        headers['Content-Type'] = 'application/json';\n      }\n\n      const req = request(\n        url,\n        {\n          agent: this.agent,\n          headers,\n          method,\n          port: 443,\n          servername: url.hostname,\n        },\n        (res) => {\n          if (res.statusCode == null) {\n            const error = new Error(\n              `[FATAL] Request failed: ${method.toUpperCase()} ${url}`\n            );\n            Error.captureStackTrace(error);\n            return ko(error);\n          }\n\n          console.log(\n            `Response: ${method.toUpperCase()} ${url} => HTTP ${\n              res.statusCode\n            } (${res.statusMessage})`\n          );\n\n          // Transient (server) errors:\n          if (res.statusCode >= 500 && res.statusCode < 600) {\n            console.error(\n              `[RETRYABLE] HTTP ${res.statusCode} (${\n                res.statusMessage\n              }) - ${method.toUpperCase()} ${url}`\n            );\n            // Call again after a short back-off\n            return retry();\n          }\n          // Permanent (client) errors:\n          if (res.statusCode >= 400 && res.statusCode < 500) {\n            const error = new Error(\n              `[FATAL] HTTP ${res.statusCode} (${\n                res.statusMessage\n              }) - ${method.toUpperCase()} ${url}`\n            );\n            Error.captureStackTrace(error);\n            return ko(error);\n          }\n\n          const onError = (err: Error & { code?: string }) => {\n            if (err.code === 'ECONNRESET') {\n              // Transient networking problem?\n              console.error(\n                `[RETRYABLE] ${err.code} - ${method.toUpperCase()} ${url}`\n              );\n              retry();\n            } else {\n              Error.captureStackTrace(err);\n              console.log('[NON-RETRYABLE]', err);\n              ko(err);\n            }\n          };\n\n          res.once('error', onError);\n\n          const json = JSONStream.parse(true);\n          json.once('data', ok);\n          json.once('error', onError);\n\n          const plainPayload =\n            res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n          plainPayload.pipe(json, { end: true });\n          plainPayload.once('error', onError);\n        }\n      );\n      req.end(body && JSON.stringify(body, null, 2));\n    });\n  }\n}\n\nexport interface DatabaseChanges {\n  /**\n   * The last sequence ID from this change set. This is the value that should be\n   * passed to the subsequent `.changes` call to fetch the next page.\n   */\n  readonly last_seq: string | number;\n\n  /**\n   * The amount of pending changes from the server. This value is not always\n   * returned by the servers.\n   */\n  readonly pending?: number;\n\n  /**\n   * The changes that are part of this batch.\n   */\n  readonly results: readonly DatabaseChange[];\n}\n\nexport interface DatabaseChange {\n  /**\n   * The set of revisions to the object that were resolved as part of this\n   * change.\n   */\n  readonly changes: ReadonlyArray<{ readonly rev: string }>;\n\n  /**\n   * The ID of the document that has changed.\n   */\n  readonly id: string;\n\n  /**\n   * The sequence ID for this change in the stream. It may not be present for\n   * all (or any) entries in the result.\n   */\n  readonly seq?: string | number;\n\n  /**\n   * Whether this change corresponds to this document being deleted.\n   */\n  readonly deleted: boolean;\n\n  /**\n   * If present, the resolved document after the change has been applied.\n   */\n  doc?: { readonly [key: string]: unknown };\n}\n\nexport interface DatabaseInfos {\n  readonly db_name: string;\n  readonly disk_format_version: number;\n  readonly doc_count: number;\n  readonly doc_del_count: number;\n  readonly instance_start_time: string;\n  readonly purge_seq: string | number;\n  readonly sizes: {\n    readonly active: number;\n    readonly external: number;\n    readonly file: number;\n  };\n  readonly update_seq: string | number;\n}\n\nfunction gunzip(readable: Readable): Readable {\n  const gz = createGunzip();\n  readable.pipe(gz, { end: true });\n  return gz;\n}\n"]}
@@ -61367,9 +61367,18 @@ var CouchChanges = class extends import_events.EventEmitter {
61367
61367
  }
61368
61368
  const metadataUrl = new import_url.URL(change.id, NPM_REGISTRY_URL);
61369
61369
  console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);
61370
- const meta = await this.https("get", metadataUrl);
61371
- change.doc = meta;
61372
- return change;
61370
+ try {
61371
+ const meta = await this.https("get", metadataUrl);
61372
+ change.doc = meta;
61373
+ return change;
61374
+ } catch (e3) {
61375
+ if (e3.errorMessage.includes("HTTP 404")) {
61376
+ console.log(`Skipping ${change.id} because of 404 error:
61377
+ ${e3}`);
61378
+ return;
61379
+ }
61380
+ throw e3;
61381
+ }
61373
61382
  }
61374
61383
  async fetchAndFilterAllMetadata(changes) {
61375
61384
  return (await Promise.all(
@@ -61406,9 +61415,6 @@ var CouchChanges = class extends import_events.EventEmitter {
61406
61415
  if (body) {
61407
61416
  headers["Content-Type"] = "application/json";
61408
61417
  }
61409
- console.log(
61410
- `Request: ${method.toUpperCase()} ${url}, ${JSON.stringify(headers)}`
61411
- );
61412
61418
  const req = (0, import_https.request)(
61413
61419
  url,
61414
61420
  {