aws-lambda-secret-fetcher 0.2.1 โ 0.3.0
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 +32 -22
- package/lib/index.d.ts +7 -1
- package/lib/index.js +31 -91
- package/package.json +5 -2
package/README.md
CHANGED
|
@@ -1,21 +1,32 @@
|
|
|
1
1
|
# AWS Lambda Secret Fetcher
|
|
2
2
|
|
|
3
|
+
[](https://www.npmjs.com/package/aws-lambda-secret-fetcher)
|
|
4
|
+
[](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
|
-
-
|
|
8
|
-
-
|
|
9
|
-
-
|
|
10
|
-
-
|
|
11
|
-
-
|
|
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,
|
|
51
|
-
retries: 5,
|
|
52
|
-
baseBackoffMs: 500,
|
|
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
|
|
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>`
|
|
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`
|
|
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
|
-
|
|
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
|
-
*
|
|
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
|
-
*
|
|
6
|
-
*
|
|
7
|
-
*
|
|
8
|
-
* @
|
|
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
|
|
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
|
-
*
|
|
35
|
-
*
|
|
36
|
-
* @param
|
|
37
|
-
* @returns
|
|
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
|
-
*
|
|
110
|
-
*
|
|
111
|
-
* @
|
|
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.
|
|
40
|
+
"projen": "^0.99.12",
|
|
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.
|
|
56
|
+
"version": "0.3.0",
|
|
54
57
|
"jest": {
|
|
55
58
|
"coverageProvider": "v8",
|
|
56
59
|
"testMatch": [
|