construct-hub 0.4.406 → 0.4.407

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.
@@ -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.406" };
281
+ CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.407" };
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"]}
@@ -120,34 +120,20 @@ class CouchChanges extends events_1.EventEmitter {
120
120
  let maxDelay = 100;
121
121
  while (true) {
122
122
  try {
123
- return await new Promise((ok, ko) => {
124
- const req = (0, https_1.request)(url, requestOptions, (res) => {
125
- if (res.statusCode == null) {
126
- throw new RetryableError('No status code available');
127
- }
128
- // Server errors. We can't know whether these are really retryable but we usually pretend that they are.
129
- if (res.statusCode >= 500 && res.statusCode < 600) {
130
- throw new RetryableError(`HTTP ${res.statusCode} ${res.statusMessage}`);
131
- }
132
- // Permanent (client) errors:
133
- if (res.statusCode >= 400 && res.statusCode < 500) {
134
- throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);
135
- }
136
- console.log(`Response: ${method.toUpperCase()} ${url} => HTTP ${res.statusCode} (${res.statusMessage})`);
137
- res.once('error', ko);
138
- const json = JSONStream.parse(true);
139
- json.once('data', ok);
140
- json.once('error', ko);
141
- const plainPayload = res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;
142
- plainPayload.pipe(json, { end: true });
143
- plainPayload.once('error', ko);
144
- });
145
- req.on('error', ko);
146
- req.on('timeout', () => {
147
- req.destroy(new RetryableError(`Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`));
148
- });
149
- req.end(body && JSON.stringify(body, null, 2));
150
- });
123
+ const res = await requestPromise(url, requestOptions, body);
124
+ if (res.statusCode == null) {
125
+ throw new RetryableError('No status code available');
126
+ }
127
+ // Server errors. We can't know whether these are really retryable but we usually pretend that they are.
128
+ if (res.statusCode >= 500 && res.statusCode < 600) {
129
+ throw new RetryableError(`HTTP ${res.statusCode} ${res.statusMessage}`);
130
+ }
131
+ // Permanent (client) errors:
132
+ if (res.statusCode >= 400 && res.statusCode < 500) {
133
+ throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);
134
+ }
135
+ console.log(`Response: ${method.toUpperCase()} ${url} => HTTP ${res.statusCode} (${res.statusMessage})`);
136
+ return await readResponseJson(res);
151
137
  }
152
138
  catch (e) {
153
139
  if (Date.now() > deadline || !isRetryableError(e)) {
@@ -161,6 +147,30 @@ class CouchChanges extends events_1.EventEmitter {
161
147
  }
162
148
  }
163
149
  exports.CouchChanges = CouchChanges;
150
+ /**
151
+ * A Promisified version of `https.request()` that also handles timeout events
152
+ */
153
+ function requestPromise(url, options, body) {
154
+ return new Promise((ok, ko) => {
155
+ const req = (0, https_1.request)(url, options ?? {}, ok);
156
+ req.on('error', ko);
157
+ req.on('timeout', () => {
158
+ req.destroy(new RetryableError(`Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`));
159
+ });
160
+ req.end(body && JSON.stringify(body, null, 2));
161
+ });
162
+ }
163
+ function readResponseJson(res) {
164
+ return new Promise((ok, ko) => {
165
+ res.once('error', ko);
166
+ const json = JSONStream.parse(true);
167
+ json.once('data', ok);
168
+ json.once('error', ko);
169
+ const plainPayload = res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;
170
+ plainPayload.pipe(json, { end: true });
171
+ plainPayload.once('error', ko);
172
+ });
173
+ }
164
174
  class RetryableError extends Error {
165
175
  }
166
176
  function isRetryableError(e) {
@@ -174,4 +184,4 @@ function gunzip(readable) {
174
184
  readable.pipe(gz, { end: true });
175
185
  return gz;
176
186
  }
177
- //# 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, RequestOptions } 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\nconst REQUEST_DEADLINE_MS = 30_000;\n\nconst REQUEST_ATTEMPT_TIMEOUT_MS = 5_000;\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 database: string;\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\n      // This timeout is separate from the request timeout, and is here to\n      // prevent stalled/idle connections\n      timeout: 60_000,\n    });\n    this.baseUrl = new URL(baseUrl);\n    this.database = database;\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(this.database, this.baseUrl);\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.message?.includes('HTTP 404')) {\n        console.log(\n          `Skipping ${change.id} because of HTTP 404 (Not Found) error`\n        );\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 async https(\n    method: 'get' | 'post',\n    url: URL,\n    body?: { [key: string]: unknown }\n  ): Promise<{ [key: string]: unknown }> {\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 requestOptions: RequestOptions = {\n      agent: this.agent,\n      headers,\n      method,\n      port: 443,\n      servername: url.hostname,\n      // This just leads to a 'timeout' event\n      timeout: REQUEST_ATTEMPT_TIMEOUT_MS,\n    };\n\n    const deadline = Date.now() + REQUEST_DEADLINE_MS;\n    let maxDelay = 100;\n    while (true) {\n      try {\n        return await new Promise((ok, ko) => {\n          const req = request(url, requestOptions, (res) => {\n            if (res.statusCode == null) {\n              throw new RetryableError('No status code available');\n            }\n\n            // Server errors. We can't know whether these are really retryable but we usually pretend that they are.\n            if (res.statusCode >= 500 && res.statusCode < 600) {\n              throw new RetryableError(\n                `HTTP ${res.statusCode} ${res.statusMessage}`\n              );\n            }\n\n            // Permanent (client) errors:\n            if (res.statusCode >= 400 && res.statusCode < 500) {\n              throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);\n            }\n\n            console.log(\n              `Response: ${method.toUpperCase()} ${url} => HTTP ${\n                res.statusCode\n              } (${res.statusMessage})`\n            );\n\n            res.once('error', ko);\n\n            const json = JSONStream.parse(true);\n            json.once('data', ok);\n            json.once('error', ko);\n\n            const plainPayload =\n              res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n            plainPayload.pipe(json, { end: true });\n            plainPayload.once('error', ko);\n          });\n\n          req.on('error', ko);\n          req.on('timeout', () => {\n            req.destroy(\n              new RetryableError(\n                `Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`\n              )\n            );\n          });\n\n          req.end(body && JSON.stringify(body, null, 2));\n        });\n      } catch (e: any) {\n        if (Date.now() > deadline || !isRetryableError(e)) {\n          throw e;\n        }\n\n        console.error(`[RETRYABLE] ${method} ${url}: ${e}`);\n\n        await sleep(Math.floor(Math.random() * maxDelay));\n        maxDelay *= 2;\n      }\n    }\n  }\n}\n\nclass RetryableError extends Error {}\n\nfunction isRetryableError(e: Error): boolean {\n  return e instanceof RetryableError || (e as any).code === 'ECONNRESET';\n}\n\nasync function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\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"]}
187
+ //# 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 { IncomingMessage, OutgoingHttpHeaders } from 'http';\nimport { Agent, request, RequestOptions } 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\nconst REQUEST_DEADLINE_MS = 30_000;\n\nconst REQUEST_ATTEMPT_TIMEOUT_MS = 5_000;\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 database: string;\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\n      // This timeout is separate from the request timeout, and is here to\n      // prevent stalled/idle connections\n      timeout: 60_000,\n    });\n    this.baseUrl = new URL(baseUrl);\n    this.database = database;\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(this.database, this.baseUrl);\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.message?.includes('HTTP 404')) {\n        console.log(\n          `Skipping ${change.id} because of HTTP 404 (Not Found) error`\n        );\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 async https(\n    method: 'get' | 'post',\n    url: URL,\n    body?: { [key: string]: unknown }\n  ): Promise<{ [key: string]: unknown }> {\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 requestOptions: RequestOptions = {\n      agent: this.agent,\n      headers,\n      method,\n      port: 443,\n      servername: url.hostname,\n      // This just leads to a 'timeout' event\n      timeout: REQUEST_ATTEMPT_TIMEOUT_MS,\n    };\n\n    const deadline = Date.now() + REQUEST_DEADLINE_MS;\n    let maxDelay = 100;\n    while (true) {\n      try {\n        const res = await requestPromise(url, requestOptions, body);\n        if (res.statusCode == null) {\n          throw new RetryableError('No status code available');\n        }\n\n        // Server errors. We can't know whether these are really retryable but we usually pretend that they are.\n        if (res.statusCode >= 500 && res.statusCode < 600) {\n          throw new RetryableError(\n            `HTTP ${res.statusCode} ${res.statusMessage}`\n          );\n        }\n\n        // Permanent (client) errors:\n        if (res.statusCode >= 400 && res.statusCode < 500) {\n          throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);\n        }\n\n        console.log(\n          `Response: ${method.toUpperCase()} ${url} => HTTP ${\n            res.statusCode\n          } (${res.statusMessage})`\n        );\n\n        return await readResponseJson(res);\n      } catch (e: any) {\n        if (Date.now() > deadline || !isRetryableError(e)) {\n          throw e;\n        }\n\n        console.error(`[RETRYABLE] ${method} ${url}: ${e}`);\n\n        await sleep(Math.floor(Math.random() * maxDelay));\n        maxDelay *= 2;\n      }\n    }\n  }\n}\n\n/**\n * A Promisified version of `https.request()` that also handles timeout events\n */\nfunction requestPromise(\n  url: URL,\n  options: RequestOptions,\n  body?: Record<string, unknown>\n) {\n  return new Promise<IncomingMessage>((ok, ko) => {\n    const req = request(url, options ?? {}, ok);\n    req.on('error', ko);\n    req.on('timeout', () => {\n      req.destroy(\n        new RetryableError(\n          `Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`\n        )\n      );\n    });\n    req.end(body && JSON.stringify(body, null, 2));\n  });\n}\n\nfunction readResponseJson(\n  res: IncomingMessage\n): Promise<Record<string, unknown>> {\n  return new Promise((ok, ko) => {\n    res.once('error', ko);\n\n    const json = JSONStream.parse(true);\n    json.once('data', ok);\n    json.once('error', ko);\n\n    const plainPayload =\n      res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n    plainPayload.pipe(json, { end: true });\n    plainPayload.once('error', ko);\n  });\n}\n\nclass RetryableError extends Error {}\n\nfunction isRetryableError(e: Error): boolean {\n  return e instanceof RetryableError || (e as any).code === 'ECONNRESET';\n}\n\nasync function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\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"]}
@@ -61722,40 +61722,22 @@ var CouchChanges = class extends import_events.EventEmitter {
61722
61722
  let maxDelay = 100;
61723
61723
  while (true) {
61724
61724
  try {
61725
- return await new Promise((ok, ko) => {
61726
- const req = (0, import_https.request)(url, requestOptions, (res) => {
61727
- if (res.statusCode == null) {
61728
- throw new RetryableError("No status code available");
61729
- }
61730
- if (res.statusCode >= 500 && res.statusCode < 600) {
61731
- throw new RetryableError(
61732
- `HTTP ${res.statusCode} ${res.statusMessage}`
61733
- );
61734
- }
61735
- if (res.statusCode >= 400 && res.statusCode < 500) {
61736
- throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);
61737
- }
61738
- console.log(
61739
- `Response: ${method.toUpperCase()} ${url} => HTTP ${res.statusCode} (${res.statusMessage})`
61740
- );
61741
- res.once("error", ko);
61742
- const json = JSONStream.parse(true);
61743
- json.once("data", ok);
61744
- json.once("error", ko);
61745
- const plainPayload = res.headers["content-encoding"] === "gzip" ? gunzip(res) : res;
61746
- plainPayload.pipe(json, { end: true });
61747
- plainPayload.once("error", ko);
61748
- });
61749
- req.on("error", ko);
61750
- req.on("timeout", () => {
61751
- req.destroy(
61752
- new RetryableError(
61753
- `Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`
61754
- )
61755
- );
61756
- });
61757
- req.end(body && JSON.stringify(body, null, 2));
61758
- });
61725
+ const res = await requestPromise(url, requestOptions, body);
61726
+ if (res.statusCode == null) {
61727
+ throw new RetryableError("No status code available");
61728
+ }
61729
+ if (res.statusCode >= 500 && res.statusCode < 600) {
61730
+ throw new RetryableError(
61731
+ `HTTP ${res.statusCode} ${res.statusMessage}`
61732
+ );
61733
+ }
61734
+ if (res.statusCode >= 400 && res.statusCode < 500) {
61735
+ throw new Error(`HTTP ${res.statusCode} ${res.statusMessage}`);
61736
+ }
61737
+ console.log(
61738
+ `Response: ${method.toUpperCase()} ${url} => HTTP ${res.statusCode} (${res.statusMessage})`
61739
+ );
61740
+ return await readResponseJson(res);
61759
61741
  } catch (e3) {
61760
61742
  if (Date.now() > deadline || !isRetryableError(e3)) {
61761
61743
  throw e3;
@@ -61767,6 +61749,31 @@ var CouchChanges = class extends import_events.EventEmitter {
61767
61749
  }
61768
61750
  }
61769
61751
  };
61752
+ function requestPromise(url, options, body) {
61753
+ return new Promise((ok, ko) => {
61754
+ const req = (0, import_https.request)(url, options ?? {}, ok);
61755
+ req.on("error", ko);
61756
+ req.on("timeout", () => {
61757
+ req.destroy(
61758
+ new RetryableError(
61759
+ `Timeout after ${REQUEST_ATTEMPT_TIMEOUT_MS}ms, aborting request`
61760
+ )
61761
+ );
61762
+ });
61763
+ req.end(body && JSON.stringify(body, null, 2));
61764
+ });
61765
+ }
61766
+ function readResponseJson(res) {
61767
+ return new Promise((ok, ko) => {
61768
+ res.once("error", ko);
61769
+ const json = JSONStream.parse(true);
61770
+ json.once("data", ok);
61771
+ json.once("error", ko);
61772
+ const plainPayload = res.headers["content-encoding"] === "gzip" ? gunzip(res) : res;
61773
+ plainPayload.pipe(json, { end: true });
61774
+ plainPayload.once("error", ko);
61775
+ });
61776
+ }
61770
61777
  var RetryableError = class extends Error {
61771
61778
  };
61772
61779
  function isRetryableError(e3) {