construct-hub 0.4.450 → 0.4.451

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
@@ -21425,6 +21425,6 @@
21425
21425
  "symbolId": "src/package-sources/npmjs:NpmJsProps"
21426
21426
  }
21427
21427
  },
21428
- "version": "0.4.450",
21429
- "fingerprint": "dXIl71PI9bHRvLU018Y9Tfur7mPH87UXYgGTeAPcaDs="
21428
+ "version": "0.4.451",
21429
+ "fingerprint": "5J71fe+EqhtFNakbqh6GFyO7M83wpHsWehFfFbdrVcc="
21430
21430
  }
@@ -412,7 +412,7 @@ class ConstructHub extends constructs_1.Construct {
412
412
  }
413
413
  exports.ConstructHub = ConstructHub;
414
414
  _a = JSII_RTTI_SYMBOL_1;
415
- ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.450" };
415
+ ConstructHub[_a] = { fqn: "construct-hub.ConstructHub", version: "0.4.451" };
416
416
  /**
417
417
  * How possibly risky operations (such as doc-generation, which requires
418
418
  * installing the indexed packages in order to trans-literate sample code) are
@@ -272,5 +272,5 @@ class CodeArtifact {
272
272
  }
273
273
  exports.CodeArtifact = CodeArtifact;
274
274
  _a = JSII_RTTI_SYMBOL_1;
275
- CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.450" };
275
+ CodeArtifact[_a] = { fqn: "construct-hub.sources.CodeArtifact", version: "0.4.451" };
276
276
  //# 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  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 = dlq\n      .metricApproximateNumberOfMessagesVisible({\n        period: Duration.minutes(1),\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:\n          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"]}
@@ -27,6 +27,12 @@ export declare class CouchChanges extends EventEmitter {
27
27
  changes(since: string | number, opts?: {
28
28
  readonly batchSize?: number;
29
29
  }): Promise<DatabaseChanges>;
30
+ /**
31
+ * Fetch the metadata associated with a change. The change comes associated with a revision number,
32
+ * which can be compared to the revision number of the metadata to determine if the replica is
33
+ * lagging behind the changes stream. If so, we retry until the replica is up-to-date or until
34
+ * 30 seconds elapsed after which we return the potentially stale metadata.
35
+ */
30
36
  private fetchAndFilterMetadata;
31
37
  private fetchAndFilterAllMetadata;
32
38
  /**
@@ -11,6 +11,7 @@ const REQUEST_DEADLINE_MS = 30000;
11
11
  const REQUEST_ATTEMPT_TIMEOUT_MS = 5000;
12
12
  const DEFAULT_BATCH_SIZE = 100;
13
13
  const MAX_CONNS_PER_HOST = 100;
14
+ const MAX_PACKAGE_SERVER_LAG_MS = 30000; // 30 seconds
14
15
  /**
15
16
  * A utility class that helps with traversing CouchDB database changes streams
16
17
  * in a promise-based, page-by-page manner.
@@ -62,26 +63,52 @@ class CouchChanges extends events_1.EventEmitter {
62
63
  totalCount: result.results.length,
63
64
  };
64
65
  }
66
+ /**
67
+ * Fetch the metadata associated with a change. The change comes associated with a revision number,
68
+ * which can be compared to the revision number of the metadata to determine if the replica is
69
+ * lagging behind the changes stream. If so, we retry until the replica is up-to-date or until
70
+ * 30 seconds elapsed after which we return the potentially stale metadata.
71
+ */
65
72
  async fetchAndFilterMetadata(change) {
66
73
  // Filter out deleted packages or null ids
67
74
  if (change.deleted || !change.id) {
68
75
  console.log(`Skipping ${change.id}: deleted or null id`);
69
76
  return;
70
77
  }
78
+ const latestChangesRev = getMaxSequentialRevision(change);
71
79
  const metadataUrl = new url_1.URL(change.id, NPM_REGISTRY_URL);
72
80
  console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);
73
- try {
74
- const meta = await this.https('get', metadataUrl);
75
- change.doc = meta; // add metadata to the change object
76
- return change;
77
- }
78
- catch (e) {
79
- if (e.message?.includes('HTTP 404')) {
80
- console.log(`Skipping ${change.id} because of HTTP 404 (Not Found) error`);
81
- return;
81
+ // Retry configuration
82
+ const baseDelay = 1000; // 1 second
83
+ const maxDelay = 8000; // 8 seconds max
84
+ let attempt = 0;
85
+ const startTime = Date.now();
86
+ while (Date.now() - startTime < MAX_PACKAGE_SERVER_LAG_MS) {
87
+ try {
88
+ const meta = await this.https('get', metadataUrl);
89
+ const latestReplicaRev = parseSequentialRevision(meta._rev);
90
+ change.doc = meta; // add metadata to the change object
91
+ // Happy path: replica is up-to-date
92
+ if (latestReplicaRev >= latestChangesRev) {
93
+ return change;
94
+ }
95
+ // Unhappy path: replica is behind. Calculate delay and retry
96
+ const delay = Math.floor(Math.random() * Math.min(baseDelay * Math.pow(2, attempt), maxDelay));
97
+ console.log(`${change.id}: package _rev ${latestReplicaRev} < expected replication rev ${latestChangesRev}, retrying in ${delay} ms`);
98
+ await new Promise(resolve => setTimeout(resolve, delay));
99
+ attempt++;
100
+ }
101
+ catch (e) {
102
+ if (e.message?.includes('HTTP 404')) {
103
+ console.log(`Skipping ${change.id} because of HTTP 404 (Not Found) error`);
104
+ return;
105
+ }
106
+ throw e;
82
107
  }
83
- throw e;
84
108
  }
109
+ // Timeout reached, proceed with stale data
110
+ console.log(`Timeout reached for ${change.id}, replica may be stale`);
111
+ return change;
85
112
  }
86
113
  async fetchAndFilterAllMetadata(changes) {
87
114
  return (await Promise.all(changes.map((change) => this.fetchAndFilterMetadata(change)))).filter((change) => change !== undefined);
@@ -180,9 +207,17 @@ function isRetryableError(e) {
180
207
  async function sleep(ms) {
181
208
  return new Promise((ok) => setTimeout(ok, ms));
182
209
  }
210
+ function parseSequentialRevision(rev) {
211
+ return parseInt(rev.split('-')[0]);
212
+ }
213
+ function getMaxSequentialRevision(change) {
214
+ return Math.max(...change.changes
215
+ .map(change => parseSequentialRevision(change.rev))
216
+ .filter(num => !isNaN(num)));
217
+ }
183
218
  function gunzip(readable) {
184
219
  const gz = (0, zlib_1.createGunzip)();
185
220
  readable.pipe(gz, { end: true });
186
221
  return gz;
187
222
  }
188
- //# 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 { json } from 'node:stream/consumers';\nimport { Readable } from 'stream';\nimport { URL } from 'url';\nimport { createGunzip } from 'zlib';\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\nconst DEFAULT_BATCH_SIZE = 100;\n\nconst MAX_CONNS_PER_HOST = 100;\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: MAX_CONNS_PER_HOST,\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 information 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 ?? DEFAULT_BATCH_SIZE;\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      actionableResults: results,\n      totalCount: result.results.length,\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 ${options.timeout}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 plainPayload =\n      res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n\n    return json(plainPayload)\n      .then((parsed) => ok(parsed as any))\n      .catch((err) => ko(err));\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 actionable changes that are part of this batch.\n   * This has deleted and unreachable packages removed.\n   */\n  readonly actionableResults: readonly DatabaseChange[];\n\n  /**\n   * The total count of changes in this batch. This includes unprocessable changes.\n   * 0 indicates we are up to date with \"now\".\n   */\n  readonly totalCount: number;\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"]}
223
+ //# 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 { json } from 'node:stream/consumers';\nimport { Readable } from 'stream';\nimport { URL } from 'url';\nimport { createGunzip } from 'zlib';\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\nconst DEFAULT_BATCH_SIZE = 100;\n\nconst MAX_CONNS_PER_HOST = 100;\n\nconst MAX_PACKAGE_SERVER_LAG_MS = 30_000; // 30 seconds\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: MAX_CONNS_PER_HOST,\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 information 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 ?? DEFAULT_BATCH_SIZE;\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      actionableResults: results,\n      totalCount: result.results.length,\n    };\n  }\n\n  /**\n   * Fetch the metadata associated with a change. The change comes associated with a revision number,\n   * which can be compared to the revision number of the metadata to determine if the replica is\n   * lagging behind the changes stream. If so, we retry until the replica is up-to-date or until\n   * 30 seconds elapsed after which we return the potentially stale metadata.\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 latestChangesRev = getMaxSequentialRevision(change);\n    const metadataUrl = new URL(change.id, NPM_REGISTRY_URL);\n    console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);\n\n    // Retry configuration\n    const baseDelay = 1_000; // 1 second\n    const maxDelay = 8_000; // 8 seconds max\n    let attempt = 0;\n    const startTime = Date.now();\n\n    while(Date.now() - startTime < MAX_PACKAGE_SERVER_LAG_MS) {\n      try {\n        const meta = await this.https('get', metadataUrl);\n        const latestReplicaRev = parseSequentialRevision(meta._rev as string);\n\n        change.doc = meta; // add metadata to the change object\n\n        // Happy path: replica is up-to-date\n        if (latestReplicaRev >= latestChangesRev) {\n          return change;\n        }\n\n        // Unhappy path: replica is behind. Calculate delay and retry\n        const delay = Math.floor(Math.random() * Math.min(baseDelay * Math.pow(2, attempt), maxDelay));\n        console.log(`${change.id}: package _rev ${latestReplicaRev} < expected replication rev ${latestChangesRev}, retrying in ${delay} ms`);\n        await new Promise(resolve => setTimeout(resolve, delay));\n        attempt++;\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    // Timeout reached, proceed with stale data\n    console.log(`Timeout reached for ${change.id}, replica may be stale`);\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 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 ${options.timeout}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 plainPayload =\n      res.headers['content-encoding'] === 'gzip' ? gunzip(res) : res;\n\n    return json(plainPayload)\n      .then((parsed) => ok(parsed as any))\n      .catch((err) => ko(err));\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\nfunction parseSequentialRevision(rev: string): number {\n  return parseInt(rev.split('-')[0]);\n}\n\nfunction getMaxSequentialRevision(change: DatabaseChange): number {\n  return Math.max(...change.changes\n    .map(change => parseSequentialRevision(change.rev))\n    .filter(num => !isNaN(num))\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 actionable changes that are part of this batch.\n   * This has deleted and unreachable packages removed.\n   */\n  readonly actionableResults: readonly DatabaseChange[];\n\n  /**\n   * The total count of changes in this batch. This includes unprocessable changes.\n   * 0 indicates we are up to date with \"now\".\n   */\n  readonly totalCount: number;\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"]}
@@ -63442,6 +63442,7 @@ var REQUEST_DEADLINE_MS = 3e4;
63442
63442
  var REQUEST_ATTEMPT_TIMEOUT_MS = 5e3;
63443
63443
  var DEFAULT_BATCH_SIZE = 100;
63444
63444
  var MAX_CONNS_PER_HOST = 100;
63445
+ var MAX_PACKAGE_SERVER_LAG_MS = 3e4;
63445
63446
  var CouchChanges = class extends import_events.EventEmitter {
63446
63447
  /**
63447
63448
  * @param baseUrl the CouchDB endpoint URL.
@@ -63488,26 +63489,48 @@ var CouchChanges = class extends import_events.EventEmitter {
63488
63489
  totalCount: result.results.length
63489
63490
  };
63490
63491
  }
63492
+ /**
63493
+ * Fetch the metadata associated with a change. The change comes associated with a revision number,
63494
+ * which can be compared to the revision number of the metadata to determine if the replica is
63495
+ * lagging behind the changes stream. If so, we retry until the replica is up-to-date or until
63496
+ * 30 seconds elapsed after which we return the potentially stale metadata.
63497
+ */
63491
63498
  async fetchAndFilterMetadata(change) {
63492
63499
  if (change.deleted || !change.id) {
63493
63500
  console.log(`Skipping ${change.id}: deleted or null id`);
63494
63501
  return;
63495
63502
  }
63503
+ const latestChangesRev = getMaxSequentialRevision(change);
63496
63504
  const metadataUrl = new import_url.URL(change.id, NPM_REGISTRY_URL);
63497
63505
  console.log(`Fetching metadata for ${change.id}: ${metadataUrl}`);
63498
- try {
63499
- const meta = await this.https("get", metadataUrl);
63500
- change.doc = meta;
63501
- return change;
63502
- } catch (e4) {
63503
- if (e4.message?.includes("HTTP 404")) {
63504
- console.log(
63505
- `Skipping ${change.id} because of HTTP 404 (Not Found) error`
63506
- );
63507
- return;
63506
+ const baseDelay = 1e3;
63507
+ const maxDelay = 8e3;
63508
+ let attempt = 0;
63509
+ const startTime = Date.now();
63510
+ while (Date.now() - startTime < MAX_PACKAGE_SERVER_LAG_MS) {
63511
+ try {
63512
+ const meta = await this.https("get", metadataUrl);
63513
+ const latestReplicaRev = parseSequentialRevision(meta._rev);
63514
+ change.doc = meta;
63515
+ if (latestReplicaRev >= latestChangesRev) {
63516
+ return change;
63517
+ }
63518
+ const delay = Math.floor(Math.random() * Math.min(baseDelay * Math.pow(2, attempt), maxDelay));
63519
+ console.log(`${change.id}: package _rev ${latestReplicaRev} < expected replication rev ${latestChangesRev}, retrying in ${delay} ms`);
63520
+ await new Promise((resolve) => setTimeout(resolve, delay));
63521
+ attempt++;
63522
+ } catch (e4) {
63523
+ if (e4.message?.includes("HTTP 404")) {
63524
+ console.log(
63525
+ `Skipping ${change.id} because of HTTP 404 (Not Found) error`
63526
+ );
63527
+ return;
63528
+ }
63529
+ throw e4;
63508
63530
  }
63509
- throw e4;
63510
63531
  }
63532
+ console.log(`Timeout reached for ${change.id}, replica may be stale`);
63533
+ return change;
63511
63534
  }
63512
63535
  async fetchAndFilterAllMetadata(changes) {
63513
63536
  return (await Promise.all(
@@ -63608,6 +63631,14 @@ function isRetryableError(e4) {
63608
63631
  async function sleep(ms) {
63609
63632
  return new Promise((ok) => setTimeout(ok, ms));
63610
63633
  }
63634
+ function parseSequentialRevision(rev) {
63635
+ return parseInt(rev.split("-")[0]);
63636
+ }
63637
+ function getMaxSequentialRevision(change) {
63638
+ return Math.max(
63639
+ ...change.changes.map((change2) => parseSequentialRevision(change2.rev)).filter((num) => !isNaN(num))
63640
+ );
63641
+ }
63611
63642
  function gunzip(readable) {
63612
63643
  const gz = (0, import_zlib.createGunzip)();
63613
63644
  readable.pipe(gz, { end: true });
@@ -63823,7 +63854,7 @@ async function handler(event, context) {
63823
63854
  } = await loadFollowerPosition(stagingBucket, npm);
63824
63855
  let writeKnownVersionsFile = didLoadVersionsFromLegacy;
63825
63856
  let updatedMarker = initialMarker;
63826
- let maxBatchProcessingTime = 3e4;
63857
+ let maxBatchProcessingTime = 6e4;
63827
63858
  let shouldContinue = true;
63828
63859
  do {
63829
63860
  await (0, import_aws_embedded_metrics.metricScope)((metrics) => async () => {