aws-lambda-secret-fetcher 0.2.2 โ†’ 0.3.1

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/README.md CHANGED
@@ -1,21 +1,32 @@
1
1
  # AWS Lambda Secret Fetcher
2
2
 
3
+ [![npm version](https://img.shields.io/npm/v/aws-lambda-secret-fetcher.svg)](https://www.npmjs.com/package/aws-lambda-secret-fetcher)
4
+ [![License](https://img.shields.io/npm/l/aws-lambda-secret-fetcher.svg)](https://www.npmjs.com/package/aws-lambda-secret-fetcher)
5
+
3
6
  A lightweight TypeScript library for fetching secrets from AWS Secrets Manager using the [AWS Parameters and Secrets Lambda Extension](https://docs.aws.amazon.com/secretsmanager/latest/userguide/retrieving-secrets_lambda.html).
4
7
 
5
8
  ## Features
6
9
 
7
- - ๐Ÿš€ Uses the local Lambda Extension API (no SDK required)
8
- - ๐Ÿ”„ Built-in retry with exponential backoff (Full Jitter)
9
- - โฑ๏ธ Configurable timeout
10
- - ๐Ÿ“ฆ Automatic JSON parsing for secret values
11
- - ๐Ÿ’ช TypeScript support with generics
10
+ - Uses the local Lambda Extension API (no AWS SDK required)
11
+ - Retry with timeout and full jitter backoff via [fetch-retrier](https://www.npmjs.com/package/fetch-retrier)
12
+ - Configurable timeout, retries, and base backoff
13
+ - Automatic JSON parsing for secret values
14
+ - TypeScript support with generics
12
15
 
13
16
  ## Installation
14
17
 
18
+ **npm**
19
+
15
20
  ```bash
16
21
  npm install aws-lambda-secret-fetcher
17
22
  ```
18
23
 
24
+ **yarn**
25
+
26
+ ```bash
27
+ yarn add aws-lambda-secret-fetcher
28
+ ```
29
+
19
30
  ## Prerequisites
20
31
 
21
32
  Your Lambda function must have the [AWS Parameters and Secrets Lambda Extension](https://docs.aws.amazon.com/secretsmanager/latest/userguide/retrieving-secrets_lambda.html) layer attached.
@@ -47,14 +58,22 @@ console.log(credentials.username); // Type-safe access
47
58
  import { secretFetcher, type GetSecretValueOptions } from 'aws-lambda-secret-fetcher';
48
59
 
49
60
  const options: GetSecretValueOptions = {
50
- timeoutMs: 3000, // Timeout per request (default: 2000)
51
- retries: 5, // Number of retry attempts (default: 3)
52
- baseBackoffMs: 500, // Base backoff time for retries (default: 300)
61
+ timeoutMs: 3000,
62
+ retries: 5,
63
+ baseBackoffMs: 500,
53
64
  };
54
65
 
55
66
  const secret = await secretFetcher.getSecretValue('my-secret', options);
56
67
  ```
57
68
 
69
+ ## Options
70
+
71
+ | Option | Type | Default | Description |
72
+ |--------|------|---------|-------------|
73
+ | `timeoutMs` | `number` | `2000` | Request timeout in milliseconds per attempt |
74
+ | `retries` | `number` | `3` | Maximum number of attempts (including the first request) |
75
+ | `baseBackoffMs` | `number` | `300` | Base delay in milliseconds for backoff between retries |
76
+
58
77
  ## API
59
78
 
60
79
  The package exports `secretFetcher`, an object that provides:
@@ -68,28 +87,19 @@ Fetches a secret value from AWS Secrets Manager via the Lambda Extension.
68
87
  | Parameter | Type | Description |
69
88
  |-----------|------|-------------|
70
89
  | `name` | `string` | The name or ARN of the secret |
71
- | `options` | `GetSecretValueOptions` | Optional configuration |
72
-
73
- #### Options
74
-
75
- | Option | Type | Default | Description |
76
- |--------|------|---------|-------------|
77
- | `timeoutMs` | `number` | `2000` | Request timeout in milliseconds |
78
- | `retries` | `number` | `3` | Number of retry attempts |
79
- | `baseBackoffMs` | `number` | `300` | Base backoff time for exponential retry |
90
+ | `options` | `GetSecretValueOptions` | Optional timeout, retries, and backoff settings |
80
91
 
81
92
  #### Returns
82
93
 
83
- - `Promise<T>` - The secret value. If the secret is a JSON string, it will be automatically parsed.
94
+ - `Promise<T>` โ€” The secret value. If the secret is a JSON string, it is automatically parsed as `T`.
84
95
 
85
96
  #### Throws
86
97
 
87
- - `Error` - If the secret cannot be retrieved after all retries
88
- - `Error` - If the response format is invalid
98
+ - `Error` โ€” If the secret cannot be retrieved after all retries, or if the response format is invalid.
89
99
 
90
100
  ## Retry Behavior
91
101
 
92
- The library implements AWS-recommended Full Jitter exponential backoff for retries. It will retry on:
102
+ Retries use full jitter exponential backoff. The library retries on:
93
103
 
94
104
  - HTTP status codes: 429, 500, 502, 503, 504
95
105
  - Lambda Extension not ready (400 with "not ready to serve traffic")
@@ -103,4 +113,4 @@ The library implements AWS-recommended Full Jitter exponential backoff for retri
103
113
 
104
114
  ## License
105
115
 
106
- Apache-2.0
116
+ This project is licensed under the Apache-2.0 License.
package/lib/index.d.ts CHANGED
@@ -1,11 +1,17 @@
1
1
  /**
2
- * getSecretValue Options
2
+ * Options for fetching a secret from the Secrets Manager Extension.
3
3
  */
4
4
  export interface GetSecretValueOptions {
5
+ /** Request timeout in milliseconds. Default: 2000 */
5
6
  timeoutMs?: number;
7
+ /** Maximum number of attempts (including the first request). Default: 3 */
6
8
  retries?: number;
9
+ /** Base delay in milliseconds for backoff between retries. Default: 300 */
7
10
  baseBackoffMs?: number;
8
11
  }
12
+ /**
13
+ * Client for fetching secrets from the AWS Lambda Secrets Manager Extension.
14
+ */
9
15
  export declare const secretFetcher: {
10
16
  getSecretValue: <T = string>(name: string, options?: GetSecretValueOptions) => Promise<T>;
11
17
  };
package/lib/index.js CHANGED
@@ -1,24 +1,36 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.secretFetcher = void 0;
4
+ const fetch_retrier_1 = require("fetch-retrier");
4
5
  /**
5
- * Get Secret Value from Secrets Manager Extension
6
- * @param name Secret Name
7
- * @param options GetSecretValue Options
8
- * @returns SecretString as T
6
+ * Fetches a secret value from the AWS Lambda Secrets Manager Extension (localhost:2773).
7
+ * Uses retries with full jitter backoff for transient errors (e.g. 5xx, 429, or extension "not ready").
8
+ *
9
+ * @param name - Secret name (identifier) to fetch
10
+ * @param options - Optional timeout, retry, and backoff settings
11
+ * @returns The secret value as string, or parsed as T if the stored value is JSON
12
+ * @throws Error if the response format is invalid or the request fails after retries
9
13
  */
10
14
  const getSecretValue = async (name, options = {}) => {
11
- // default options
12
15
  const { timeoutMs = 2000, retries = 3, baseBackoffMs = 300 } = options;
13
16
  const url = `http://localhost:2773/secretsmanager/get?secretId=${encodeURIComponent(name)}`;
14
- const response = await request(url, {
17
+ const requestOptions = {
15
18
  headers: {
16
19
  'X-Aws-Parameters-Secrets-Token': process.env.AWS_SESSION_TOKEN ?? '',
17
20
  },
18
21
  retries,
19
22
  timeoutMs,
20
23
  baseBackoffMs,
21
- });
24
+ shouldRetry: (res, body) => {
25
+ if ([429, 500, 502, 503, 504].includes(res.status))
26
+ return true;
27
+ // Extension may return 400 + "not ready to serve traffic" while initializing; retry in that case
28
+ if (res.status === 400 && /not\s+ready.*traffic/i.test(body))
29
+ return true;
30
+ return false;
31
+ },
32
+ };
33
+ const response = await (0, fetch_retrier_1.fetchRetrier)(url, requestOptions);
22
34
  const raw = await response.json();
23
35
  if (!isSecretResponse(raw)) {
24
36
  throw new Error('Invalid secret response format');
@@ -31,71 +43,11 @@ const getSecretValue = async (name, options = {}) => {
31
43
  return secretString;
32
44
  };
33
45
  /**
34
- * retry + timeout + Full Jitter
35
- * @param url - The URL to fetch
36
- * @param options - The options for the fetch
37
- * @returns The response
46
+ * Type guard for the Secrets Manager Extension response shape.
47
+ *
48
+ * @param value - Value to check
49
+ * @returns True if value has the shape of SecretResponse
38
50
  */
39
- const request = async (url, options) => {
40
- const { headers, retries, timeoutMs, baseBackoffMs } = options;
41
- for (let attempt = 1; attempt <= retries; attempt++) {
42
- const controller = new AbortController();
43
- const timer = setTimeout(() => controller.abort(), timeoutMs);
44
- try {
45
- const res = await fetch(url, {
46
- headers,
47
- signal: controller.signal,
48
- });
49
- clearTimeout(timer);
50
- if (res.ok) {
51
- return res;
52
- }
53
- const text = await res.text();
54
- const isContinue = (() => {
55
- // ็‰นๅฎšใฎใ‚นใƒ†ใƒผใ‚ฟใ‚นใ‚ณใƒผใƒ‰ใฎๅ ดๅˆใฏใƒชใƒˆใƒฉใ‚ค
56
- if ([429, 500, 502, 503, 504].includes(res.status)) {
57
- console.log('Specific status code, retrying...');
58
- return true;
59
- }
60
- // AWS Lambda Extension API ใฎ็‰นๆฎŠใ‚ฑใƒผใ‚น๏ผš
61
- // ๆ‹กๅผตๆฉŸ่ƒฝใฎๅˆๆœŸๅŒ–ไธญใซ 400 + 'not ready to serve traffic' ใ‚’่ฟ”ใ™
62
- // ใ“ใ‚Œใฏไธ€ๆ™‚็š„ใช็Šถๆ…‹ใชใฎใงใ€ใƒชใƒˆใƒฉใ‚คใŒ้ฉๅˆ‡
63
- // ๅ‚่€ƒ: https://docs.aws.amazon.com/lambda/latest/dg/runtimes-extensions-api.html
64
- if (res.status === 400 && /not\s+ready.*traffic/i.test(text)) {
65
- console.log('Extension not ready, retrying...');
66
- return true;
67
- }
68
- return false;
69
- })();
70
- if (isContinue) {
71
- if (attempt === retries) {
72
- throw new Error(`HTTP ${res.status}`);
73
- }
74
- await wait(fullJitter(baseBackoffMs, attempt));
75
- }
76
- else {
77
- throw new Error(`Non-retriable HTTP error: ${res.status}`);
78
- }
79
- }
80
- catch (err) {
81
- clearTimeout(timer);
82
- if (err instanceof Error && err.name === 'AbortError') {
83
- if (attempt === retries)
84
- throw err;
85
- await wait(fullJitter(baseBackoffMs, attempt));
86
- continue;
87
- }
88
- if (err instanceof TypeError) {
89
- if (attempt === retries)
90
- throw err;
91
- await wait(fullJitter(baseBackoffMs, attempt));
92
- continue;
93
- }
94
- throw err;
95
- }
96
- }
97
- throw new Error('Unreachable'); // TS ็”จ๏ผˆๅฎŸ้š›ใซใฏๅˆฐ้”ใ—ใชใ„๏ผ‰
98
- };
99
51
  const isSecretResponse = (value) => {
100
52
  if (typeof value !== 'object' || value === null)
101
53
  return false;
@@ -106,30 +58,18 @@ const isSecretResponse = (value) => {
106
58
  (v.VersionId === undefined || typeof v.VersionId === 'string');
107
59
  };
108
60
  /**
109
- * Wait for a given time
110
- * @param ms - The time to wait in milliseconds
111
- * @returns A promise that resolves when the time has elapsed
112
- */
113
- const wait = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
114
- /**
115
- * AWS recommended Full Jitter
116
- * @param base - The base time in milliseconds
117
- * @param attempt - The attempt number
118
- * @returns The time to wait in milliseconds
119
- */
120
- const fullJitter = (base, attempt) => {
121
- const cap = base * Math.pow(2, attempt);
122
- return Math.floor(Math.random() * cap);
123
- };
124
- /**
125
- * Roughly JSON check
126
- * @param str - The string to check
127
- * @returns True if the string looks like JSON
61
+ * Heuristic check whether a string looks like JSON (starts with `{` after trim).
62
+ *
63
+ * @param str - String to check
64
+ * @returns True if the string appears to be JSON
128
65
  */
129
66
  const looksLikeJson = (str) => {
130
67
  return typeof str === 'string' && str.trim().startsWith('{');
131
68
  };
69
+ /**
70
+ * Client for fetching secrets from the AWS Lambda Secrets Manager Extension.
71
+ */
132
72
  exports.secretFetcher = {
133
73
  getSecretValue,
134
74
  };
135
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * getSecretValue Options\n */\nexport interface GetSecretValueOptions {\n  timeoutMs?: number; // default: 2000\n  retries?: number; // default: 3\n  baseBackoffMs?: number; // default: 300\n}\n\n/**\n * fetchWithRetry Options\n */\ninterface RequestOptions {\n  headers?: Record<string, string>;\n  retries: number;\n  timeoutMs: number;\n  baseBackoffMs: number;\n}\n\n/**\n * Secrets Manager Extension Response\n */\ninterface SecretResponse {\n  ARN: string;\n  Name: string;\n  SecretString: string;\n  VersionId?: string;\n}\n\n/**\n * Get Secret Value from Secrets Manager Extension\n * @param name Secret Name\n * @param options GetSecretValue Options\n * @returns SecretString as T\n */\nconst getSecretValue = async <T = string>(name: string, options: GetSecretValueOptions = {}): Promise<T> => {\n  // default options\n  const { timeoutMs = 2000, retries = 3, baseBackoffMs = 300 } = options;\n\n  const url = `http://localhost:2773/secretsmanager/get?secretId=${encodeURIComponent(name)}`;\n\n  const response = await request(url, {\n    headers: {\n      'X-Aws-Parameters-Secrets-Token': process.env.AWS_SESSION_TOKEN ?? '',\n    },\n    retries,\n    timeoutMs,\n    baseBackoffMs,\n  });\n\n  const raw = await response.json();\n\n  if (!isSecretResponse(raw)) {\n    throw new Error('Invalid secret response format');\n  }\n\n  const data: SecretResponse = raw;\n\n  const secretString = data.SecretString;\n\n  if (looksLikeJson(secretString)) {\n    return JSON.parse(secretString) as T;\n  }\n\n  return secretString as T;\n};\n\n/**\n * retry + timeout + Full Jitter\n * @param url - The URL to fetch\n * @param options - The options for the fetch\n * @returns The response\n */\nconst request = async (url: string, options: RequestOptions): Promise<Response> => {\n  const { headers, retries, timeoutMs, baseBackoffMs } = options;\n\n  for (let attempt = 1; attempt <= retries; attempt++) {\n    const controller = new AbortController();\n    const timer = setTimeout(() => controller.abort(), timeoutMs);\n\n    try {\n      const res = await fetch(url, {\n        headers,\n        signal: controller.signal,\n      });\n\n      clearTimeout(timer);\n\n      if (res.ok) {\n        return res;\n      }\n\n      const text = await res.text();\n\n      const isContinue = ((): boolean => {\n        // 特定のステータスコードの場合はリトライ\n        if ([429, 500, 502, 503, 504].includes(res.status)) {\n          console.log('Specific status code, retrying...');\n          return true;\n        }\n        // AWS Lambda Extension API の特殊ケース：\n        // 拡張機能の初期化中に 400 + 'not ready to serve traffic' を返す\n        // これは一時的な状態なので、リトライが適切\n        // 参考: https://docs.aws.amazon.com/lambda/latest/dg/runtimes-extensions-api.html\n        if (res.status === 400 && /not\\s+ready.*traffic/i.test(text)) {\n          console.log('Extension not ready, retrying...');\n          return true;\n        }\n        return false;\n      })();\n\n      if (isContinue) {\n        if (attempt === retries) {\n          throw new Error(`HTTP ${res.status}`);\n        }\n        await wait(fullJitter(baseBackoffMs, attempt));\n      } else {\n        throw new Error(`Non-retriable HTTP error: ${res.status}`);\n      }\n    } catch (err: unknown) {\n      clearTimeout(timer);\n\n      if (err instanceof Error && err.name === 'AbortError') {\n        if (attempt === retries) throw err;\n        await wait(fullJitter(baseBackoffMs, attempt));\n        continue;\n      }\n\n      if (err instanceof TypeError) {\n        if (attempt === retries) throw err;\n        await wait(fullJitter(baseBackoffMs, attempt));\n        continue;\n      }\n\n      throw err;\n    }\n  }\n\n  throw new Error('Unreachable'); // TS 用（実際には到達しない）\n};\n\nconst isSecretResponse = (value: unknown): value is SecretResponse => {\n  if (typeof value !== 'object' || value === null) return false;\n  const v = value as Record<string, unknown>;\n\n  return typeof v.SecretString === 'string' &&\n         typeof v.Name === 'string' &&\n         typeof v.ARN === 'string' &&\n         (v.VersionId === undefined || typeof v.VersionId === 'string');\n};\n\n/**\n * Wait for a given time\n * @param ms - The time to wait in milliseconds\n * @returns A promise that resolves when the time has elapsed\n */\nconst wait = (ms: number): Promise<void> =>\n  new Promise((resolve) => setTimeout(resolve, ms));\n\n/**\n * AWS recommended Full Jitter\n * @param base - The base time in milliseconds\n * @param attempt - The attempt number\n * @returns The time to wait in milliseconds\n */\nconst fullJitter = (base: number, attempt: number): number => {\n  const cap = base * Math.pow(2, attempt);\n  return Math.floor(Math.random() * cap);\n};\n\n/**\n * Roughly JSON check\n * @param str - The string to check\n * @returns True if the string looks like JSON\n */\nconst looksLikeJson = (str: string): boolean => {\n  return typeof str === 'string' && str.trim().startsWith('{');\n};\n\n\nexport const secretFetcher = {\n  getSecretValue,\n};\n"]}
75
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsaURBQWtFO0FBNEJsRTs7Ozs7Ozs7R0FRRztBQUNILE1BQU0sY0FBYyxHQUFHLEtBQUssRUFBYyxJQUFZLEVBQUUsVUFBaUMsRUFBRSxFQUFjLEVBQUU7SUFDekcsTUFBTSxFQUFFLFNBQVMsR0FBRyxJQUFJLEVBQUUsT0FBTyxHQUFHLENBQUMsRUFBRSxhQUFhLEdBQUcsR0FBRyxFQUFFLEdBQUcsT0FBTyxDQUFDO0lBRXZFLE1BQU0sR0FBRyxHQUFHLHFEQUFxRCxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO0lBRTVGLE1BQU0sY0FBYyxHQUFtQjtRQUNyQyxPQUFPLEVBQUU7WUFDUCxnQ0FBZ0MsRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLGlCQUFpQixJQUFJLEVBQUU7U0FDdEU7UUFDRCxPQUFPO1FBQ1AsU0FBUztRQUNULGFBQWE7UUFDYixXQUFXLEVBQUUsQ0FBQyxHQUFHLEVBQUUsSUFBSSxFQUFFLEVBQUU7WUFDekIsSUFBSSxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQztnQkFBRSxPQUFPLElBQUksQ0FBQztZQUNoRSxpR0FBaUc7WUFDakcsSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLEdBQUcsSUFBSSx1QkFBdUIsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDO2dCQUFFLE9BQU8sSUFBSSxDQUFDO1lBQzFFLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztLQUNGLENBQUM7SUFFRixNQUFNLFFBQVEsR0FBRyxNQUFNLElBQUEsNEJBQVksRUFBQyxHQUFHLEVBQUUsY0FBYyxDQUFDLENBQUM7SUFFekQsTUFBTSxHQUFHLEdBQUcsTUFBTSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7SUFFbEMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDM0IsTUFBTSxJQUFJLEtBQUssQ0FBQyxnQ0FBZ0MsQ0FBQyxDQUFDO0lBQ3BELENBQUM7SUFFRCxNQUFNLElBQUksR0FBbUIsR0FBRyxDQUFDO0lBRWpDLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUM7SUFFdkMsSUFBSSxhQUFhLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztRQUNoQyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFNLENBQUM7SUFDdkMsQ0FBQztJQUVELE9BQU8sWUFBaUIsQ0FBQztBQUMzQixDQUFDLENBQUM7QUFFRjs7Ozs7R0FLRztBQUNILE1BQU0sZ0JBQWdCLEdBQUcsQ0FBQyxLQUFjLEVBQTJCLEVBQUU7SUFDbkUsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLElBQUksS0FBSyxLQUFLLElBQUk7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUM5RCxNQUFNLENBQUMsR0FBRyxLQUFnQyxDQUFDO0lBRTNDLE9BQU8sT0FBTyxDQUFDLENBQUMsWUFBWSxLQUFLLFFBQVE7UUFDbEMsT0FBTyxDQUFDLENBQUMsSUFBSSxLQUFLLFFBQVE7UUFDMUIsT0FBTyxDQUFDLENBQUMsR0FBRyxLQUFLLFFBQVE7UUFDekIsQ0FBQyxDQUFDLENBQUMsU0FBUyxLQUFLLFNBQVMsSUFBSSxPQUFPLENBQUMsQ0FBQyxTQUFTLEtBQUssUUFBUSxDQUFDLENBQUM7QUFDeEUsQ0FBQyxDQUFDO0FBRUY7Ozs7O0dBS0c7QUFDSCxNQUFNLGFBQWEsR0FBRyxDQUFDLEdBQVcsRUFBVyxFQUFFO0lBQzdDLE9BQU8sT0FBTyxHQUFHLEtBQUssUUFBUSxJQUFJLEdBQUcsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLENBQUM7QUFDL0QsQ0FBQyxDQUFDO0FBR0Y7O0dBRUc7QUFDVSxRQUFBLGFBQWEsR0FBRztJQUMzQixjQUFjO0NBQ2YsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGZldGNoUmV0cmllciwgdHlwZSBSZXF1ZXN0T3B0aW9ucyB9IGZyb20gJ2ZldGNoLXJldHJpZXInO1xuXG4vKipcbiAqIE9wdGlvbnMgZm9yIGZldGNoaW5nIGEgc2VjcmV0IGZyb20gdGhlIFNlY3JldHMgTWFuYWdlciBFeHRlbnNpb24uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgR2V0U2VjcmV0VmFsdWVPcHRpb25zIHtcbiAgLyoqIFJlcXVlc3QgdGltZW91dCBpbiBtaWxsaXNlY29uZHMuIERlZmF1bHQ6IDIwMDAgKi9cbiAgdGltZW91dE1zPzogbnVtYmVyO1xuICAvKiogTWF4aW11bSBudW1iZXIgb2YgYXR0ZW1wdHMgKGluY2x1ZGluZyB0aGUgZmlyc3QgcmVxdWVzdCkuIERlZmF1bHQ6IDMgKi9cbiAgcmV0cmllcz86IG51bWJlcjtcbiAgLyoqIEJhc2UgZGVsYXkgaW4gbWlsbGlzZWNvbmRzIGZvciBiYWNrb2ZmIGJldHdlZW4gcmV0cmllcy4gRGVmYXVsdDogMzAwICovXG4gIGJhc2VCYWNrb2ZmTXM/OiBudW1iZXI7XG59XG5cbi8qKlxuICogUmVzcG9uc2Ugc2hhcGUgcmV0dXJuZWQgYnkgdGhlIFNlY3JldHMgTWFuYWdlciBFeHRlbnNpb24gQVBJLlxuICovXG5pbnRlcmZhY2UgU2VjcmV0UmVzcG9uc2Uge1xuICAvKiogQVJOIG9mIHRoZSBzZWNyZXQgKi9cbiAgQVJOOiBzdHJpbmc7XG4gIC8qKiBOYW1lIG9mIHRoZSBzZWNyZXQgKi9cbiAgTmFtZTogc3RyaW5nO1xuICAvKiogU2VjcmV0IHZhbHVlIGFzIHN0cmluZyAobWF5IGJlIEpTT04pICovXG4gIFNlY3JldFN0cmluZzogc3RyaW5nO1xuICAvKiogT3B0aW9uYWwgdmVyc2lvbiBpZGVudGlmaWVyICovXG4gIFZlcnNpb25JZD86IHN0cmluZztcbn1cblxuLyoqXG4gKiBGZXRjaGVzIGEgc2VjcmV0IHZhbHVlIGZyb20gdGhlIEFXUyBMYW1iZGEgU2VjcmV0cyBNYW5hZ2VyIEV4dGVuc2lvbiAobG9jYWxob3N0OjI3NzMpLlxuICogVXNlcyByZXRyaWVzIHdpdGggZnVsbCBqaXR0ZXIgYmFja29mZiBmb3IgdHJhbnNpZW50IGVycm9ycyAoZS5nLiA1eHgsIDQyOSwgb3IgZXh0ZW5zaW9uIFwibm90IHJlYWR5XCIpLlxuICpcbiAqIEBwYXJhbSBuYW1lIC0gU2VjcmV0IG5hbWUgKGlkZW50aWZpZXIpIHRvIGZldGNoXG4gKiBAcGFyYW0gb3B0aW9ucyAtIE9wdGlvbmFsIHRpbWVvdXQsIHJldHJ5LCBhbmQgYmFja29mZiBzZXR0aW5nc1xuICogQHJldHVybnMgVGhlIHNlY3JldCB2YWx1ZSBhcyBzdHJpbmcsIG9yIHBhcnNlZCBhcyBUIGlmIHRoZSBzdG9yZWQgdmFsdWUgaXMgSlNPTlxuICogQHRocm93cyBFcnJvciBpZiB0aGUgcmVzcG9uc2UgZm9ybWF0IGlzIGludmFsaWQgb3IgdGhlIHJlcXVlc3QgZmFpbHMgYWZ0ZXIgcmV0cmllc1xuICovXG5jb25zdCBnZXRTZWNyZXRWYWx1ZSA9IGFzeW5jIDxUID0gc3RyaW5nPihuYW1lOiBzdHJpbmcsIG9wdGlvbnM6IEdldFNlY3JldFZhbHVlT3B0aW9ucyA9IHt9KTogUHJvbWlzZTxUPiA9PiB7XG4gIGNvbnN0IHsgdGltZW91dE1zID0gMjAwMCwgcmV0cmllcyA9IDMsIGJhc2VCYWNrb2ZmTXMgPSAzMDAgfSA9IG9wdGlvbnM7XG5cbiAgY29uc3QgdXJsID0gYGh0dHA6Ly9sb2NhbGhvc3Q6Mjc3My9zZWNyZXRzbWFuYWdlci9nZXQ/c2VjcmV0SWQ9JHtlbmNvZGVVUklDb21wb25lbnQobmFtZSl9YDtcblxuICBjb25zdCByZXF1ZXN0T3B0aW9uczogUmVxdWVzdE9wdGlvbnMgPSB7XG4gICAgaGVhZGVyczoge1xuICAgICAgJ1gtQXdzLVBhcmFtZXRlcnMtU2VjcmV0cy1Ub2tlbic6IHByb2Nlc3MuZW52LkFXU19TRVNTSU9OX1RPS0VOID8/ICcnLFxuICAgIH0sXG4gICAgcmV0cmllcyxcbiAgICB0aW1lb3V0TXMsXG4gICAgYmFzZUJhY2tvZmZNcyxcbiAgICBzaG91bGRSZXRyeTogKHJlcywgYm9keSkgPT4ge1xuICAgICAgaWYgKFs0MjksIDUwMCwgNTAyLCA1MDMsIDUwNF0uaW5jbHVkZXMocmVzLnN0YXR1cykpIHJldHVybiB0cnVlO1xuICAgICAgLy8gRXh0ZW5zaW9uIG1heSByZXR1cm4gNDAwICsgXCJub3QgcmVhZHkgdG8gc2VydmUgdHJhZmZpY1wiIHdoaWxlIGluaXRpYWxpemluZzsgcmV0cnkgaW4gdGhhdCBjYXNlXG4gICAgICBpZiAocmVzLnN0YXR1cyA9PT0gNDAwICYmIC9ub3RcXHMrcmVhZHkuKnRyYWZmaWMvaS50ZXN0KGJvZHkpKSByZXR1cm4gdHJ1ZTtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9LFxuICB9O1xuXG4gIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2hSZXRyaWVyKHVybCwgcmVxdWVzdE9wdGlvbnMpO1xuXG4gIGNvbnN0IHJhdyA9IGF3YWl0IHJlc3BvbnNlLmpzb24oKTtcblxuICBpZiAoIWlzU2VjcmV0UmVzcG9uc2UocmF3KSkge1xuICAgIHRocm93IG5ldyBFcnJvcignSW52YWxpZCBzZWNyZXQgcmVzcG9uc2UgZm9ybWF0Jyk7XG4gIH1cblxuICBjb25zdCBkYXRhOiBTZWNyZXRSZXNwb25zZSA9IHJhdztcblxuICBjb25zdCBzZWNyZXRTdHJpbmcgPSBkYXRhLlNlY3JldFN0cmluZztcblxuICBpZiAobG9va3NMaWtlSnNvbihzZWNyZXRTdHJpbmcpKSB7XG4gICAgcmV0dXJuIEpTT04ucGFyc2Uoc2VjcmV0U3RyaW5nKSBhcyBUO1xuICB9XG5cbiAgcmV0dXJuIHNlY3JldFN0cmluZyBhcyBUO1xufTtcblxuLyoqXG4gKiBUeXBlIGd1YXJkIGZvciB0aGUgU2VjcmV0cyBNYW5hZ2VyIEV4dGVuc2lvbiByZXNwb25zZSBzaGFwZS5cbiAqXG4gKiBAcGFyYW0gdmFsdWUgLSBWYWx1ZSB0byBjaGVja1xuICogQHJldHVybnMgVHJ1ZSBpZiB2YWx1ZSBoYXMgdGhlIHNoYXBlIG9mIFNlY3JldFJlc3BvbnNlXG4gKi9cbmNvbnN0IGlzU2VjcmV0UmVzcG9uc2UgPSAodmFsdWU6IHVua25vd24pOiB2YWx1ZSBpcyBTZWNyZXRSZXNwb25zZSA9PiB7XG4gIGlmICh0eXBlb2YgdmFsdWUgIT09ICdvYmplY3QnIHx8IHZhbHVlID09PSBudWxsKSByZXR1cm4gZmFsc2U7XG4gIGNvbnN0IHYgPSB2YWx1ZSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcblxuICByZXR1cm4gdHlwZW9mIHYuU2VjcmV0U3RyaW5nID09PSAnc3RyaW5nJyAmJlxuICAgICAgICAgdHlwZW9mIHYuTmFtZSA9PT0gJ3N0cmluZycgJiZcbiAgICAgICAgIHR5cGVvZiB2LkFSTiA9PT0gJ3N0cmluZycgJiZcbiAgICAgICAgICh2LlZlcnNpb25JZCA9PT0gdW5kZWZpbmVkIHx8IHR5cGVvZiB2LlZlcnNpb25JZCA9PT0gJ3N0cmluZycpO1xufTtcblxuLyoqXG4gKiBIZXVyaXN0aWMgY2hlY2sgd2hldGhlciBhIHN0cmluZyBsb29rcyBsaWtlIEpTT04gKHN0YXJ0cyB3aXRoIGB7YCBhZnRlciB0cmltKS5cbiAqXG4gKiBAcGFyYW0gc3RyIC0gU3RyaW5nIHRvIGNoZWNrXG4gKiBAcmV0dXJucyBUcnVlIGlmIHRoZSBzdHJpbmcgYXBwZWFycyB0byBiZSBKU09OXG4gKi9cbmNvbnN0IGxvb2tzTGlrZUpzb24gPSAoc3RyOiBzdHJpbmcpOiBib29sZWFuID0+IHtcbiAgcmV0dXJuIHR5cGVvZiBzdHIgPT09ICdzdHJpbmcnICYmIHN0ci50cmltKCkuc3RhcnRzV2l0aCgneycpO1xufTtcblxuXG4vKipcbiAqIENsaWVudCBmb3IgZmV0Y2hpbmcgc2VjcmV0cyBmcm9tIHRoZSBBV1MgTGFtYmRhIFNlY3JldHMgTWFuYWdlciBFeHRlbnNpb24uXG4gKi9cbmV4cG9ydCBjb25zdCBzZWNyZXRGZXRjaGVyID0ge1xuICBnZXRTZWNyZXRWYWx1ZSxcbn07XG4iXX0=
package/package.json CHANGED
@@ -37,11 +37,14 @@
37
37
  "eslint-plugin-import": "^2.32.0",
38
38
  "jest": "^30.2.0",
39
39
  "jest-junit": "^16",
40
- "projen": "^0.99.12",
40
+ "projen": "^0.99.16",
41
41
  "ts-jest": "^29.4.6",
42
42
  "ts-node": "^10.9.2",
43
43
  "typescript": "5.9.x"
44
44
  },
45
+ "dependencies": {
46
+ "fetch-retrier": "^0.1"
47
+ },
45
48
  "engines": {
46
49
  "node": ">= 20.0.0"
47
50
  },
@@ -50,7 +53,7 @@
50
53
  "publishConfig": {
51
54
  "access": "public"
52
55
  },
53
- "version": "0.2.2",
56
+ "version": "0.3.1",
54
57
  "jest": {
55
58
  "coverageProvider": "v8",
56
59
  "testMatch": [