athena-query-execution-waiter 0.3.13 → 0.4.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 CHANGED
@@ -1,7 +1,17 @@
1
1
  # Athena Query Execution Waiter
2
2
 
3
+ [![npm version](https://img.shields.io/npm/v/athena-query-execution-waiter.svg)](https://www.npmjs.com/package/athena-query-execution-waiter)
4
+ [![License](https://img.shields.io/npm/l/athena-query-execution-waiter.svg)](https://github.com/gammarers-aws-sdk-extensions/athena-query-execution-waiter/blob/main/LICENSE)
5
+
3
6
  A small library that waits for an AWS Athena query execution to complete. It polls the Athena API until the execution reaches a terminal state: **SUCCEEDED**, **FAILED**, or **CANCELLED**.
4
7
 
8
+ ## Features
9
+
10
+ - Polls `GetQueryExecution` until the run finishes or an **overall wall-clock timeout** is exceeded (separate from **polling interval**).
11
+ - Configurable **overall timeout** and **poll spacing** via `wait()` options; default overall cap is **`DEFAULT_TIMEOUT_MS`** (2 minutes).
12
+ - Typed errors: **`AthenaQueryExecutionWaiterTimeoutError`**, **`AthenaQueryExecutionWaiterStateError`** (failed or cancelled runs).
13
+ - Built for **AWS SDK for JavaScript v3** (`@aws-sdk/client-athena`).
14
+
5
15
  ## Installation
6
16
 
7
17
  **@aws-sdk/client-athena** is a normal **dependency** of this package: installing `athena-query-execution-waiter` pulls in a compatible AWS SDK v3 Athena client. If your app also depends on `@aws-sdk/client-athena`, npm/yarn will dedupe when versions are compatible; otherwise you may have two copies under different semver ranges.
@@ -29,6 +39,7 @@ npm install athena-query-execution-waiter
29
39
  import { AthenaClient } from '@aws-sdk/client-athena';
30
40
  import {
31
41
  AthenaQueryExecutionWaiter,
42
+ DEFAULT_TIMEOUT_MS,
32
43
  AthenaQueryExecutionWaiterTimeoutError,
33
44
  AthenaQueryExecutionWaiterStateError,
34
45
  } from 'athena-query-execution-waiter';
@@ -36,7 +47,7 @@ import {
36
47
  const client = new AthenaClient({ region: 'us-east-1' });
37
48
  const waiter = new AthenaQueryExecutionWaiter(client);
38
49
 
39
- // Start a query with StartQueryExecution, then wait for it to complete
50
+ // After StartQueryExecution, wait until the execution completes
40
51
  const queryExecutionId = 'your-query-execution-id';
41
52
 
42
53
  try {
@@ -53,59 +64,70 @@ try {
53
64
  }
54
65
  ```
55
66
 
56
- ### Custom timeout
67
+ ### Overall timeout vs polling interval
68
+
69
+ | | Meaning |
70
+ |---|--------|
71
+ | **`waitOptions.timeoutMs` / `DEFAULT_TIMEOUT_MS`** | **Overall** wall-clock limit from when `wait()` starts until **SUCCEEDED**, **FAILED**, or **CANCELLED** (or this cap is exceeded). Omit `timeoutMs` to use `DEFAULT_TIMEOUT_MS`. This is **not** how often Athena is polled. |
72
+ | **`pollIntervalMs`** | Delay **between** `GetQueryExecution` calls. Independent of the overall timeout; a long poll interval still respects `waitOptions.timeoutMs` / `DEFAULT_TIMEOUT_MS`. |
57
73
 
58
- The default timeout is 10 seconds. You can pass a custom timeout in milliseconds:
74
+ Long-running jobs should pass a higher `timeoutMs` when needed:
59
75
 
60
76
  ```typescript
61
- const state = await waiter.wait(queryExecutionId, 60_000); // 60 seconds
77
+ const state = await waiter.wait(queryExecutionId, {
78
+ timeoutMs: 15 * 60_000, // 15 minutes overall
79
+ });
62
80
  ```
63
81
 
64
- ### Polling interval
65
-
66
- The default polling interval is 1 second. For long-running queries, you can increase it to reduce API calls. Set it at construction time or per `wait()` call:
82
+ Default polling interval is **1 second**. Increase it to reduce API calls (constructor or per `wait()`):
67
83
 
68
84
  ```typescript
69
- // Waiter-wide: poll every 5 seconds
70
85
  const waiter = new AthenaQueryExecutionWaiter(client, { pollIntervalMs: 5000 });
71
86
 
72
- // Or per call (overrides the waiter default)
73
- const state = await waiter.wait(queryExecutionId, 60_000, { pollIntervalMs: 3000 });
87
+ const state = await waiter.wait(queryExecutionId, {
88
+ timeoutMs: 60_000,
89
+ pollIntervalMs: 3000,
90
+ });
74
91
  ```
75
92
 
76
- ## API
93
+ ## Options
94
+
95
+ ### `AthenaQueryExecutionWaiterOptions` (constructor)
96
+
97
+ Passed to `new AthenaQueryExecutionWaiter(client, options?)`.
98
+
99
+ | Option | Type | Description |
100
+ |--------|------|-------------|
101
+ | `pollIntervalMs` | `number` (optional) | Default milliseconds **between** `GetQueryExecution` calls when `wait()` omits `pollIntervalMs`. Default: `1000`. |
102
+
103
+ ### `AthenaQueryExecutionWaitOptions` (`wait()`)
104
+
105
+ Passed to `wait(queryExecutionId, waitOptions?)`.
106
+
107
+ | Option | Type | Description |
108
+ |--------|------|-------------|
109
+ | `timeoutMs` | `number` (optional) | **Overall** wall-clock timeout in ms from the start of `wait()` until a terminal state. Default: `DEFAULT_TIMEOUT_MS` (**2 minutes**). |
110
+ | `pollIntervalMs` | `number` (optional) | Milliseconds **between** polls for this call. Default: constructor’s `pollIntervalMs` or `1000`. |
111
+
112
+ ## API reference
77
113
 
78
114
  ### `AthenaQueryExecutionWaiter`
79
115
 
80
116
  - **Constructor:** `new AthenaQueryExecutionWaiter(client: AthenaClient, options?: AthenaQueryExecutionWaiterOptions)`
81
- - `client` An AWS SDK v3 `AthenaClient` instance.
82
- - `options.pollIntervalMs` Optional. Polling interval in ms (default: 1000).
83
-
84
- - **`wait(queryExecutionId: string, timeoutMs?: number, waitOptions?: AthenaQueryExecutionWaitOptions): Promise<QueryExecutionState>`**
85
- - Polls the query execution status until it completes.
86
- - **Returns** the execution state (`SUCCEEDED`) on success.
87
- - **Throws** `AthenaQueryExecutionWaiterTimeoutError` if the timeout is exceeded.
117
+ - **`wait(queryExecutionId: string, waitOptions?: AthenaQueryExecutionWaitOptions): Promise<QueryExecutionState>`**
118
+ - **Returns** `SUCCEEDED` on success.
119
+ - **Throws** `AthenaQueryExecutionWaiterTimeoutError` if overall wait exceeds the effective timeout.
88
120
  - **Throws** `AthenaQueryExecutionWaiterStateError` when the state is `FAILED` or `CANCELLED`.
89
- - `timeoutMs` defaults to 10,000 ms when omitted.
90
- - `waitOptions.pollIntervalMs` – Optional. Overrides the polling interval for this call (default: use constructor value or 1000).
91
121
 
92
- ### Types
122
+ ### Constants
93
123
 
94
- - **`AthenaQueryExecutionWaiterOptions`** `{ pollIntervalMs?: number }` (constructor options).
95
- - **`AthenaQueryExecutionWaitOptions`** – `{ pollIntervalMs?: number }` (options for `wait()`).
124
+ - **`DEFAULT_TIMEOUT_MS`** Default overall wait cap in milliseconds (2 minutes) when `waitOptions.timeoutMs` is omitted. Safe to import for your own guards or logging.
96
125
 
97
126
  ### Errors
98
127
 
99
- - **`AthenaQueryExecutionWaiterError`** Base class for all waiter errors.
100
-
101
- - **`AthenaQueryExecutionWaiterTimeoutError`** Thrown when the wait duration exceeds `timeoutMs`.
102
- - Extends `AthenaQueryExecutionWaiterError`.
103
- - Constructor: `(elapsedTime: number)`.
104
-
105
- - **`AthenaQueryExecutionWaiterStateError`** – Thrown when the query ends in `FAILED` or `CANCELLED`.
106
- - Extends `AthenaQueryExecutionWaiterError`.
107
- - Properties: `state` (`QueryExecutionState`), `reason` (string).
108
- - Constructor: `(state: QueryExecutionState, reason?: string)`.
128
+ - **`AthenaQueryExecutionWaiterError`** Base class for waiter errors.
129
+ - **`AthenaQueryExecutionWaiterTimeoutError`** — Overall elapsed time since `wait()` started exceeded `waitOptions.timeoutMs` or `DEFAULT_TIMEOUT_MS`. Constructor: `(elapsedTime: number)`.
130
+ - **`AthenaQueryExecutionWaiterStateError`** Query ended in `FAILED` or `CANCELLED`. Properties: `state`, `reason`. Constructor: `(state: QueryExecutionState, reason?: string)`.
109
131
 
110
132
  ## License
111
133
 
package/lib/index.d.ts CHANGED
@@ -1,4 +1,17 @@
1
1
  import { QueryExecutionState, AthenaClient } from '@aws-sdk/client-athena';
2
+ /**
3
+ * Default overall wait timeout (milliseconds) when `wait()` is called without
4
+ * `waitOptions.timeoutMs`.
5
+ * This caps total wall-clock time from the start of `wait()` until a terminal state
6
+ * (or error)—it is not the delay between polls (`pollIntervalMs`).
7
+ *
8
+ * **Why 2 minutes:** Athena often exceeds a few seconds (queueing, cold start, moderate
9
+ * scans). Ten seconds fails too often as a library default; unbounded or very large
10
+ * defaults risk hanging callers. Two minutes is a practical middle ground—tight enough
11
+ * to surface stuck work, long enough for many interactive workloads. Use a larger
12
+ * `wait(..., { timeoutMs })` for heavy analytics or ETL.
13
+ */
14
+ export declare const DEFAULT_TIMEOUT_MS: number;
2
15
  /** Options for AthenaQueryExecutionWaiter constructor. */
3
16
  export interface AthenaQueryExecutionWaiterOptions {
4
17
  /**
@@ -10,6 +23,12 @@ export interface AthenaQueryExecutionWaiterOptions {
10
23
  }
11
24
  /** Options for wait(). */
12
25
  export interface AthenaQueryExecutionWaitOptions {
26
+ /**
27
+ * Overall wall-clock timeout in milliseconds for this wait (from the start of `wait()`
28
+ * until a terminal state). Not the delay between polls.
29
+ * Defaults to {@link DEFAULT_TIMEOUT_MS} when omitted.
30
+ */
31
+ timeoutMs?: number;
13
32
  /**
14
33
  * Polling interval in milliseconds for this wait.
15
34
  * Overrides the waiter's default poll interval when specified.
@@ -19,26 +38,27 @@ export interface AthenaQueryExecutionWaitOptions {
19
38
  /**
20
39
  * Waits for Athena query execution to complete.
21
40
  * Polls execution status until it becomes SUCCEEDED, FAILED, or CANCELLED.
41
+ * Overall time is bounded by `waitOptions.timeoutMs` (or {@link DEFAULT_TIMEOUT_MS});
42
+ * spacing between polls is controlled separately by `pollIntervalMs`.
22
43
  */
23
44
  export declare class AthenaQueryExecutionWaiter {
24
45
  private readonly client;
25
46
  private readonly defaultPollIntervalMs;
26
47
  /**
27
48
  * @param client Athena API client
28
- * @param options Optional settings (e.g. pollIntervalMs for polling interval)
49
+ * @param options Optional settings (e.g. pollIntervalMs)
29
50
  */
30
51
  constructor(client: AthenaClient, options?: AthenaQueryExecutionWaiterOptions);
31
52
  /**
32
53
  * Waits until the given query execution completes (or fails/cancels).
33
54
  *
34
55
  * @param queryExecutionId Query execution ID to wait for
35
- * @param timeoutMs Timeout in milliseconds. Defaults to DEFAULT_TIMEOUT_MS when omitted
36
- * @param waitOptions Optional per-call options (e.g. pollIntervalMs overrides default)
56
+ * @param waitOptions Optional per-call settings (`timeoutMs`, `pollIntervalMs`)
37
57
  * @returns Execution state on success (SUCCEEDED)
38
58
  * @throws AthenaQueryExecutionWaiterTimeoutError On timeout
39
59
  * @throws AthenaQueryExecutionWaiterStateError When state is FAILED or CANCELLED
40
60
  */
41
- wait(queryExecutionId: string, timeoutMs?: number, waitOptions?: AthenaQueryExecutionWaitOptions): Promise<QueryExecutionState>;
61
+ wait(queryExecutionId: string, waitOptions?: AthenaQueryExecutionWaitOptions): Promise<QueryExecutionState>;
42
62
  }
43
63
  /**
44
64
  * Base error for Athena query execution waiter.
package/lib/index.js CHANGED
@@ -1,19 +1,32 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.AthenaQueryExecutionWaiterStateError = exports.AthenaQueryExecutionWaiterTimeoutError = exports.AthenaQueryExecutionWaiterError = exports.AthenaQueryExecutionWaiter = void 0;
3
+ exports.AthenaQueryExecutionWaiterStateError = exports.AthenaQueryExecutionWaiterTimeoutError = exports.AthenaQueryExecutionWaiterError = exports.AthenaQueryExecutionWaiter = exports.DEFAULT_TIMEOUT_MS = void 0;
4
4
  const client_athena_1 = require("@aws-sdk/client-athena");
5
5
  /** Default polling interval (milliseconds) for query execution status. */
6
6
  const DEFAULT_POLL_INTERVAL_MS = 1000;
7
- /** Default timeout (milliseconds) for waiting for query execution to complete. */
8
- const DEFAULT_TIMEOUT_MS = 1000 * 10;
7
+ /**
8
+ * Default overall wait timeout (milliseconds) when `wait()` is called without
9
+ * `waitOptions.timeoutMs`.
10
+ * This caps total wall-clock time from the start of `wait()` until a terminal state
11
+ * (or error)—it is not the delay between polls (`pollIntervalMs`).
12
+ *
13
+ * **Why 2 minutes:** Athena often exceeds a few seconds (queueing, cold start, moderate
14
+ * scans). Ten seconds fails too often as a library default; unbounded or very large
15
+ * defaults risk hanging callers. Two minutes is a practical middle ground—tight enough
16
+ * to surface stuck work, long enough for many interactive workloads. Use a larger
17
+ * `wait(..., { timeoutMs })` for heavy analytics or ETL.
18
+ */
19
+ exports.DEFAULT_TIMEOUT_MS = 2 * 60 * 1000;
9
20
  /**
10
21
  * Waits for Athena query execution to complete.
11
22
  * Polls execution status until it becomes SUCCEEDED, FAILED, or CANCELLED.
23
+ * Overall time is bounded by `waitOptions.timeoutMs` (or {@link DEFAULT_TIMEOUT_MS});
24
+ * spacing between polls is controlled separately by `pollIntervalMs`.
12
25
  */
13
26
  class AthenaQueryExecutionWaiter {
14
27
  /**
15
28
  * @param client Athena API client
16
- * @param options Optional settings (e.g. pollIntervalMs for polling interval)
29
+ * @param options Optional settings (e.g. pollIntervalMs)
17
30
  */
18
31
  constructor(client, options) {
19
32
  this.client = client;
@@ -23,18 +36,18 @@ class AthenaQueryExecutionWaiter {
23
36
  * Waits until the given query execution completes (or fails/cancels).
24
37
  *
25
38
  * @param queryExecutionId Query execution ID to wait for
26
- * @param timeoutMs Timeout in milliseconds. Defaults to DEFAULT_TIMEOUT_MS when omitted
27
- * @param waitOptions Optional per-call options (e.g. pollIntervalMs overrides default)
39
+ * @param waitOptions Optional per-call settings (`timeoutMs`, `pollIntervalMs`)
28
40
  * @returns Execution state on success (SUCCEEDED)
29
41
  * @throws AthenaQueryExecutionWaiterTimeoutError On timeout
30
42
  * @throws AthenaQueryExecutionWaiterStateError When state is FAILED or CANCELLED
31
43
  */
32
- async wait(queryExecutionId, timeoutMs = DEFAULT_TIMEOUT_MS, waitOptions) {
44
+ async wait(queryExecutionId, waitOptions) {
45
+ const effectiveTimeoutMs = waitOptions?.timeoutMs ?? exports.DEFAULT_TIMEOUT_MS;
33
46
  const pollIntervalMs = waitOptions?.pollIntervalMs ?? this.defaultPollIntervalMs;
34
47
  const startTime = Date.now();
35
48
  do {
36
49
  const elapsedTime = Date.now() - startTime;
37
- if (elapsedTime > timeoutMs) {
50
+ if (elapsedTime > effectiveTimeoutMs) {
38
51
  throw new AthenaQueryExecutionWaiterTimeoutError(elapsedTime);
39
52
  }
40
53
  const res = await this.client.send(new client_athena_1.GetQueryExecutionCommand({
@@ -96,4 +109,4 @@ class AthenaQueryExecutionWaiterStateError extends AthenaQueryExecutionWaiterErr
96
109
  }
97
110
  }
98
111
  exports.AthenaQueryExecutionWaiterStateError = AthenaQueryExecutionWaiterStateError;
99
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,0DAAqG;AAErG,0EAA0E;AAC1E,MAAM,wBAAwB,GAAG,IAAI,CAAC;AAEtC,kFAAkF;AAClF,MAAM,kBAAkB,GAAG,IAAI,GAAG,EAAE,CAAC;AAqBrC;;;GAGG;AACH,MAAa,0BAA0B;IAIrC;;;OAGG;IACH,YACmB,MAAoB,EACrC,OAA2C;QAD1B,WAAM,GAAN,MAAM,CAAc;QAGrC,IAAI,CAAC,qBAAqB,GAAG,OAAO,EAAE,cAAc,IAAI,wBAAwB,CAAC;IACnF,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK,CAAC,IAAI,CACR,gBAAwB,EACxB,YAAoB,kBAAkB,EACtC,WAA6C;QAE7C,MAAM,cAAc,GAAG,WAAW,EAAE,cAAc,IAAI,IAAI,CAAC,qBAAqB,CAAC;QACjF,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,GAAG,CAAC;YACF,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YAC3C,IAAI,WAAW,GAAG,SAAS,EAAE,CAAC;gBAC5B,MAAM,IAAI,sCAAsC,CAAC,WAAW,CAAC,CAAC;YAChE,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,wCAAwB,CAAC;gBAC9D,gBAAgB,EAAE,gBAAgB;aACnC,CAAC,CAAC,CAAC;YACJ,MAAM,EAAE,GAAG,GAAG,CAAC,cAAc,EAAE,MAAM,CAAC;YACtC,MAAM,KAAK,GAAG,EAAE,EAAE,KAAwC,CAAC;YAC3D,MAAM,MAAM,GAAG,EAAE,EAAE,iBAAiB,CAAC;YAErC,IAAI,KAAK,KAAK,mCAAmB,CAAC,SAAS,EAAE,CAAC;gBAC5C,OAAO,KAAK,CAAC;YACf,CAAC;YACD,IAAI,KAAK,KAAK,mCAAmB,CAAC,MAAM,IAAI,KAAK,KAAK,mCAAmB,CAAC,SAAS,EAAE,CAAC;gBACpF,MAAM,IAAI,oCAAoC,CAAC,KAAK,EAAE,MAAM,IAAI,SAAS,CAAC,CAAC;YAC7E,CAAC;YACD,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,cAAc,CAAC,CAAC,CAAC;QAC1D,CAAC,QAAQ,IAAI,EAAE;IACjB,CAAC;CACF;AAtDD,gEAsDC;AAED;;GAEG;AACH,MAAa,+BAAgC,SAAQ,KAAK;IAExD;;OAEG;IACH,YAAY,OAAe;QACzB,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,iCAAiC,CAAC;IAChD,CAAC;CACF;AATD,0EASC;AAED;;GAEG;AACH,MAAa,sCAAuC,SAAQ,+BAA+B;IAEzF;;OAEG;IACH,YAAY,WAAmB;QAC7B,KAAK,CAAC,gCAAgC,WAAW,IAAI,CAAC,CAAC;QACvD,IAAI,CAAC,IAAI,GAAG,wCAAwC,CAAC;IACvD,CAAC;CACF;AATD,wFASC;AAED;;GAEG;AACH,MAAa,oCAAqC,SAAQ,+BAA+B;IAEvF;;;OAGG;IACH,YAA4B,KAA0B,EAAkB,SAAiB,SAAS;QAChG,KAAK,CAAC,4CAA4C,KAAK,KAAK,MAAM,EAAE,CAAC,CAAC;QAD5C,UAAK,GAAL,KAAK,CAAqB;QAAkB,WAAM,GAAN,MAAM,CAAoB;QAEhG,IAAI,CAAC,IAAI,GAAG,sCAAsC,CAAC;IACrD,CAAC;CACF;AAVD,oFAUC","sourcesContent":["import { GetQueryExecutionCommand, QueryExecutionState, AthenaClient } from '@aws-sdk/client-athena';\n\n/** Default polling interval (milliseconds) for query execution status. */\nconst DEFAULT_POLL_INTERVAL_MS = 1000;\n\n/** Default timeout (milliseconds) for waiting for query execution to complete. */\nconst DEFAULT_TIMEOUT_MS = 1000 * 10;\n\n/** Options for AthenaQueryExecutionWaiter constructor. */\nexport interface AthenaQueryExecutionWaiterOptions {\n  /**\n   * Polling interval in milliseconds.\n   * Increase for long-running queries to reduce API calls.\n   * Defaults to DEFAULT_POLL_INTERVAL_MS (1000) when omitted.\n   */\n  pollIntervalMs?: number;\n}\n\n/** Options for wait(). */\nexport interface AthenaQueryExecutionWaitOptions {\n  /**\n   * Polling interval in milliseconds for this wait.\n   * Overrides the waiter's default poll interval when specified.\n   */\n  pollIntervalMs?: number;\n}\n\n/**\n * Waits for Athena query execution to complete.\n * Polls execution status until it becomes SUCCEEDED, FAILED, or CANCELLED.\n */\nexport class AthenaQueryExecutionWaiter {\n\n  private readonly defaultPollIntervalMs: number;\n\n  /**\n   * @param client Athena API client\n   * @param options Optional settings (e.g. pollIntervalMs for polling interval)\n   */\n  constructor(\n    private readonly client: AthenaClient,\n    options?: AthenaQueryExecutionWaiterOptions,\n  ) {\n    this.defaultPollIntervalMs = options?.pollIntervalMs ?? DEFAULT_POLL_INTERVAL_MS;\n  }\n\n  /**\n   * Waits until the given query execution completes (or fails/cancels).\n   *\n   * @param queryExecutionId Query execution ID to wait for\n   * @param timeoutMs Timeout in milliseconds. Defaults to DEFAULT_TIMEOUT_MS when omitted\n   * @param waitOptions Optional per-call options (e.g. pollIntervalMs overrides default)\n   * @returns Execution state on success (SUCCEEDED)\n   * @throws AthenaQueryExecutionWaiterTimeoutError On timeout\n   * @throws AthenaQueryExecutionWaiterStateError When state is FAILED or CANCELLED\n   */\n  async wait(\n    queryExecutionId: string,\n    timeoutMs: number = DEFAULT_TIMEOUT_MS,\n    waitOptions?: AthenaQueryExecutionWaitOptions,\n  ): Promise<QueryExecutionState> {\n    const pollIntervalMs = waitOptions?.pollIntervalMs ?? this.defaultPollIntervalMs;\n    const startTime = Date.now();\n    do {\n      const elapsedTime = Date.now() - startTime;\n      if (elapsedTime > timeoutMs) {\n        throw new AthenaQueryExecutionWaiterTimeoutError(elapsedTime);\n      }\n\n      const res = await this.client.send(new GetQueryExecutionCommand({\n        QueryExecutionId: queryExecutionId,\n      }));\n      const st = res.QueryExecution?.Status;\n      const state = st?.State as QueryExecutionState | undefined;\n      const reason = st?.StateChangeReason;\n\n      if (state === QueryExecutionState.SUCCEEDED) {\n        return state;\n      }\n      if (state === QueryExecutionState.FAILED || state === QueryExecutionState.CANCELLED) {\n        throw new AthenaQueryExecutionWaiterStateError(state, reason ?? 'unknown');\n      }\n      await new Promise((r) => setTimeout(r, pollIntervalMs));\n    } while (true);\n  }\n}\n\n/**\n * Base error for Athena query execution waiter.\n */\nexport class AthenaQueryExecutionWaiterError extends Error {\n\n  /**\n   * @param message Error message\n   */\n  constructor(message: string) {\n    super(message);\n    this.name = 'AthenaQueryExecutionWaiterError';\n  }\n}\n\n/**\n * Thrown when waiting for query execution times out.\n */\nexport class AthenaQueryExecutionWaiterTimeoutError extends AthenaQueryExecutionWaiterError {\n\n  /**\n   * @param elapsedTime Elapsed time in milliseconds until timeout\n   */\n  constructor(elapsedTime: number) {\n    super(`Athena query timed out after ${elapsedTime}ms`);\n    this.name = 'AthenaQueryExecutionWaiterTimeoutError';\n  }\n}\n\n/**\n * Thrown when the query ends in FAILED or CANCELLED state.\n */\nexport class AthenaQueryExecutionWaiterStateError extends AthenaQueryExecutionWaiterError {\n\n  /**\n   * @param state Final execution state (FAILED or CANCELLED)\n   * @param reason Reason for the state change (e.g. error details). Defaults to 'unknown' when omitted\n   */\n  constructor(public readonly state: QueryExecutionState, public readonly reason: string = 'unknown') {\n    super(`Athena query execution failed with state ${state}: ${reason}`);\n    this.name = 'AthenaQueryExecutionWaiterStateError';\n  }\n}\n"]}
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,0DAAqG;AAErG,0EAA0E;AAC1E,MAAM,wBAAwB,GAAG,IAAI,CAAC;AAEtC;;;;;;;;;;;GAWG;AACU,QAAA,kBAAkB,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC;AA2BhD;;;;;GAKG;AACH,MAAa,0BAA0B;IAIrC;;;OAGG;IACH,YACmB,MAAoB,EACrC,OAA2C;QAD1B,WAAM,GAAN,MAAM,CAAc;QAGrC,IAAI,CAAC,qBAAqB,GAAG,OAAO,EAAE,cAAc,IAAI,wBAAwB,CAAC;IACnF,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,IAAI,CACR,gBAAwB,EACxB,WAA6C;QAE7C,MAAM,kBAAkB,GAAG,WAAW,EAAE,SAAS,IAAI,0BAAkB,CAAC;QACxE,MAAM,cAAc,GAAG,WAAW,EAAE,cAAc,IAAI,IAAI,CAAC,qBAAqB,CAAC;QACjF,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,GAAG,CAAC;YACF,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YAC3C,IAAI,WAAW,GAAG,kBAAkB,EAAE,CAAC;gBACrC,MAAM,IAAI,sCAAsC,CAAC,WAAW,CAAC,CAAC;YAChE,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,wCAAwB,CAAC;gBAC9D,gBAAgB,EAAE,gBAAgB;aACnC,CAAC,CAAC,CAAC;YACJ,MAAM,EAAE,GAAG,GAAG,CAAC,cAAc,EAAE,MAAM,CAAC;YACtC,MAAM,KAAK,GAAG,EAAE,EAAE,KAAwC,CAAC;YAC3D,MAAM,MAAM,GAAG,EAAE,EAAE,iBAAiB,CAAC;YAErC,IAAI,KAAK,KAAK,mCAAmB,CAAC,SAAS,EAAE,CAAC;gBAC5C,OAAO,KAAK,CAAC;YACf,CAAC;YACD,IAAI,KAAK,KAAK,mCAAmB,CAAC,MAAM,IAAI,KAAK,KAAK,mCAAmB,CAAC,SAAS,EAAE,CAAC;gBACpF,MAAM,IAAI,oCAAoC,CAAC,KAAK,EAAE,MAAM,IAAI,SAAS,CAAC,CAAC;YAC7E,CAAC;YACD,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,cAAc,CAAC,CAAC,CAAC;QAC1D,CAAC,QAAQ,IAAI,EAAE;IACjB,CAAC;CACF;AArDD,gEAqDC;AAED;;GAEG;AACH,MAAa,+BAAgC,SAAQ,KAAK;IAExD;;OAEG;IACH,YAAY,OAAe;QACzB,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,iCAAiC,CAAC;IAChD,CAAC;CACF;AATD,0EASC;AAED;;GAEG;AACH,MAAa,sCAAuC,SAAQ,+BAA+B;IAEzF;;OAEG;IACH,YAAY,WAAmB;QAC7B,KAAK,CAAC,gCAAgC,WAAW,IAAI,CAAC,CAAC;QACvD,IAAI,CAAC,IAAI,GAAG,wCAAwC,CAAC;IACvD,CAAC;CACF;AATD,wFASC;AAED;;GAEG;AACH,MAAa,oCAAqC,SAAQ,+BAA+B;IAEvF;;;OAGG;IACH,YAA4B,KAA0B,EAAkB,SAAiB,SAAS;QAChG,KAAK,CAAC,4CAA4C,KAAK,KAAK,MAAM,EAAE,CAAC,CAAC;QAD5C,UAAK,GAAL,KAAK,CAAqB;QAAkB,WAAM,GAAN,MAAM,CAAoB;QAEhG,IAAI,CAAC,IAAI,GAAG,sCAAsC,CAAC;IACrD,CAAC;CACF;AAVD,oFAUC","sourcesContent":["import { GetQueryExecutionCommand, QueryExecutionState, AthenaClient } from '@aws-sdk/client-athena';\n\n/** Default polling interval (milliseconds) for query execution status. */\nconst DEFAULT_POLL_INTERVAL_MS = 1000;\n\n/**\n * Default overall wait timeout (milliseconds) when `wait()` is called without\n * `waitOptions.timeoutMs`.\n * This caps total wall-clock time from the start of `wait()` until a terminal state\n * (or error)—it is not the delay between polls (`pollIntervalMs`).\n *\n * **Why 2 minutes:** Athena often exceeds a few seconds (queueing, cold start, moderate\n * scans). Ten seconds fails too often as a library default; unbounded or very large\n * defaults risk hanging callers. Two minutes is a practical middle ground—tight enough\n * to surface stuck work, long enough for many interactive workloads. Use a larger\n * `wait(..., { timeoutMs })` for heavy analytics or ETL.\n */\nexport const DEFAULT_TIMEOUT_MS = 2 * 60 * 1000;\n\n/** Options for AthenaQueryExecutionWaiter constructor. */\nexport interface AthenaQueryExecutionWaiterOptions {\n  /**\n   * Polling interval in milliseconds.\n   * Increase for long-running queries to reduce API calls.\n   * Defaults to DEFAULT_POLL_INTERVAL_MS (1000) when omitted.\n   */\n  pollIntervalMs?: number;\n}\n\n/** Options for wait(). */\nexport interface AthenaQueryExecutionWaitOptions {\n  /**\n   * Overall wall-clock timeout in milliseconds for this wait (from the start of `wait()`\n   * until a terminal state). Not the delay between polls.\n   * Defaults to {@link DEFAULT_TIMEOUT_MS} when omitted.\n   */\n  timeoutMs?: number;\n  /**\n   * Polling interval in milliseconds for this wait.\n   * Overrides the waiter's default poll interval when specified.\n   */\n  pollIntervalMs?: number;\n}\n\n/**\n * Waits for Athena query execution to complete.\n * Polls execution status until it becomes SUCCEEDED, FAILED, or CANCELLED.\n * Overall time is bounded by `waitOptions.timeoutMs` (or {@link DEFAULT_TIMEOUT_MS});\n * spacing between polls is controlled separately by `pollIntervalMs`.\n */\nexport class AthenaQueryExecutionWaiter {\n\n  private readonly defaultPollIntervalMs: number;\n\n  /**\n   * @param client Athena API client\n   * @param options Optional settings (e.g. pollIntervalMs)\n   */\n  constructor(\n    private readonly client: AthenaClient,\n    options?: AthenaQueryExecutionWaiterOptions,\n  ) {\n    this.defaultPollIntervalMs = options?.pollIntervalMs ?? DEFAULT_POLL_INTERVAL_MS;\n  }\n\n  /**\n   * Waits until the given query execution completes (or fails/cancels).\n   *\n   * @param queryExecutionId Query execution ID to wait for\n   * @param waitOptions Optional per-call settings (`timeoutMs`, `pollIntervalMs`)\n   * @returns Execution state on success (SUCCEEDED)\n   * @throws AthenaQueryExecutionWaiterTimeoutError On timeout\n   * @throws AthenaQueryExecutionWaiterStateError When state is FAILED or CANCELLED\n   */\n  async wait(\n    queryExecutionId: string,\n    waitOptions?: AthenaQueryExecutionWaitOptions,\n  ): Promise<QueryExecutionState> {\n    const effectiveTimeoutMs = waitOptions?.timeoutMs ?? DEFAULT_TIMEOUT_MS;\n    const pollIntervalMs = waitOptions?.pollIntervalMs ?? this.defaultPollIntervalMs;\n    const startTime = Date.now();\n    do {\n      const elapsedTime = Date.now() - startTime;\n      if (elapsedTime > effectiveTimeoutMs) {\n        throw new AthenaQueryExecutionWaiterTimeoutError(elapsedTime);\n      }\n\n      const res = await this.client.send(new GetQueryExecutionCommand({\n        QueryExecutionId: queryExecutionId,\n      }));\n      const st = res.QueryExecution?.Status;\n      const state = st?.State as QueryExecutionState | undefined;\n      const reason = st?.StateChangeReason;\n\n      if (state === QueryExecutionState.SUCCEEDED) {\n        return state;\n      }\n      if (state === QueryExecutionState.FAILED || state === QueryExecutionState.CANCELLED) {\n        throw new AthenaQueryExecutionWaiterStateError(state, reason ?? 'unknown');\n      }\n      await new Promise((r) => setTimeout(r, pollIntervalMs));\n    } while (true);\n  }\n}\n\n/**\n * Base error for Athena query execution waiter.\n */\nexport class AthenaQueryExecutionWaiterError extends Error {\n\n  /**\n   * @param message Error message\n   */\n  constructor(message: string) {\n    super(message);\n    this.name = 'AthenaQueryExecutionWaiterError';\n  }\n}\n\n/**\n * Thrown when waiting for query execution times out.\n */\nexport class AthenaQueryExecutionWaiterTimeoutError extends AthenaQueryExecutionWaiterError {\n\n  /**\n   * @param elapsedTime Elapsed time in milliseconds until timeout\n   */\n  constructor(elapsedTime: number) {\n    super(`Athena query timed out after ${elapsedTime}ms`);\n    this.name = 'AthenaQueryExecutionWaiterTimeoutError';\n  }\n}\n\n/**\n * Thrown when the query ends in FAILED or CANCELLED state.\n */\nexport class AthenaQueryExecutionWaiterStateError extends AthenaQueryExecutionWaiterError {\n\n  /**\n   * @param state Final execution state (FAILED or CANCELLED)\n   * @param reason Reason for the state change (e.g. error details). Defaults to 'unknown' when omitted\n   */\n  constructor(public readonly state: QueryExecutionState, public readonly reason: string = 'unknown') {\n    super(`Athena query execution failed with state ${state}: ${reason}`);\n    this.name = 'AthenaQueryExecutionWaiterStateError';\n  }\n}\n"]}
package/package.json CHANGED
@@ -65,7 +65,7 @@
65
65
  "publishConfig": {
66
66
  "access": "public"
67
67
  },
68
- "version": "0.3.13",
68
+ "version": "0.4.0",
69
69
  "jest": {
70
70
  "coverageProvider": "v8",
71
71
  "testMatch": [