@milaboratories/pl-client 2.17.7 → 2.17.9
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/dist/_virtual/_rolldown/runtime.cjs +43 -0
- package/dist/_virtual/_rolldown/runtime.js +18 -0
- package/dist/core/PromiseTracker.cjs +33 -33
- package/dist/core/PromiseTracker.cjs.map +1 -1
- package/dist/core/PromiseTracker.d.ts +10 -7
- package/dist/core/PromiseTracker.js +33 -31
- package/dist/core/PromiseTracker.js.map +1 -1
- package/dist/core/StatefulPromise.cjs +60 -61
- package/dist/core/StatefulPromise.cjs.map +1 -1
- package/dist/core/StatefulPromise.js +60 -60
- package/dist/core/StatefulPromise.js.map +1 -1
- package/dist/core/abstract_stream.d.ts +18 -15
- package/dist/core/advisory_locks.cjs +42 -49
- package/dist/core/advisory_locks.cjs.map +1 -1
- package/dist/core/advisory_locks.js +42 -48
- package/dist/core/advisory_locks.js.map +1 -1
- package/dist/core/auth.cjs +10 -15
- package/dist/core/auth.cjs.map +1 -1
- package/dist/core/auth.d.ts +7 -3
- package/dist/core/auth.js +10 -13
- package/dist/core/auth.js.map +1 -1
- package/dist/core/cache.d.ts +11 -7
- package/dist/core/client.cjs +255 -306
- package/dist/core/client.cjs.map +1 -1
- package/dist/core/client.d.ts +72 -68
- package/dist/core/client.js +253 -285
- package/dist/core/client.js.map +1 -1
- package/dist/core/config.cjs +81 -99
- package/dist/core/config.cjs.map +1 -1
- package/dist/core/config.d.ts +93 -90
- package/dist/core/config.js +81 -98
- package/dist/core/config.js.map +1 -1
- package/dist/core/default_client.cjs +84 -125
- package/dist/core/default_client.cjs.map +1 -1
- package/dist/core/default_client.d.ts +9 -6
- package/dist/core/default_client.js +78 -103
- package/dist/core/default_client.js.map +1 -1
- package/dist/core/driver.cjs +12 -16
- package/dist/core/driver.cjs.map +1 -1
- package/dist/core/driver.d.ts +18 -14
- package/dist/core/driver.js +12 -15
- package/dist/core/driver.js.map +1 -1
- package/dist/core/error_resource.cjs +5 -4
- package/dist/core/error_resource.cjs.map +1 -1
- package/dist/core/error_resource.js +5 -3
- package/dist/core/error_resource.js.map +1 -1
- package/dist/core/errors.cjs +104 -140
- package/dist/core/errors.cjs.map +1 -1
- package/dist/core/errors.d.ts +34 -30
- package/dist/core/errors.js +102 -137
- package/dist/core/errors.js.map +1 -1
- package/dist/core/final.cjs +63 -89
- package/dist/core/final.cjs.map +1 -1
- package/dist/core/final.d.ts +8 -4
- package/dist/core/final.js +63 -87
- package/dist/core/final.js.map +1 -1
- package/dist/core/ll_client.cjs +416 -521
- package/dist/core/ll_client.cjs.map +1 -1
- package/dist/core/ll_client.d.ts +100 -97
- package/dist/core/ll_client.js +415 -519
- package/dist/core/ll_client.js.map +1 -1
- package/dist/core/ll_transaction.cjs +206 -240
- package/dist/core/ll_transaction.cjs.map +1 -1
- package/dist/core/ll_transaction.d.ts +50 -52
- package/dist/core/ll_transaction.js +205 -238
- package/dist/core/ll_transaction.js.map +1 -1
- package/dist/core/stat.cjs +64 -63
- package/dist/core/stat.cjs.map +1 -1
- package/dist/core/stat.d.ts +35 -36
- package/dist/core/stat.js +64 -62
- package/dist/core/stat.js.map +1 -1
- package/dist/core/transaction.cjs +613 -650
- package/dist/core/transaction.cjs.map +1 -1
- package/dist/core/transaction.d.ts +165 -162
- package/dist/core/transaction.js +612 -648
- package/dist/core/transaction.js.map +1 -1
- package/dist/core/type_conversion.cjs +62 -83
- package/dist/core/type_conversion.cjs.map +1 -1
- package/dist/core/type_conversion.js +61 -81
- package/dist/core/type_conversion.js.map +1 -1
- package/dist/core/types.cjs +56 -86
- package/dist/core/types.cjs.map +1 -1
- package/dist/core/types.d.ts +63 -62
- package/dist/core/types.js +54 -83
- package/dist/core/types.js.map +1 -1
- package/dist/core/unauth_client.cjs +35 -41
- package/dist/core/unauth_client.cjs.map +1 -1
- package/dist/core/unauth_client.d.ts +18 -14
- package/dist/core/unauth_client.js +34 -39
- package/dist/core/unauth_client.js.map +1 -1
- package/dist/core/websocket_stream.cjs +280 -349
- package/dist/core/websocket_stream.cjs.map +1 -1
- package/dist/core/websocket_stream.js +278 -347
- package/dist/core/websocket_stream.js.map +1 -1
- package/dist/core/wire.d.ts +21 -17
- package/dist/helpers/pl.cjs +71 -73
- package/dist/helpers/pl.cjs.map +1 -1
- package/dist/helpers/pl.d.ts +40 -41
- package/dist/helpers/pl.js +66 -46
- package/dist/helpers/pl.js.map +1 -1
- package/dist/helpers/poll.cjs +99 -134
- package/dist/helpers/poll.cjs.map +1 -1
- package/dist/helpers/poll.d.ts +37 -34
- package/dist/helpers/poll.js +97 -113
- package/dist/helpers/poll.js.map +1 -1
- package/dist/helpers/retry_strategy.cjs +82 -87
- package/dist/helpers/retry_strategy.cjs.map +1 -1
- package/dist/helpers/retry_strategy.js +83 -86
- package/dist/helpers/retry_strategy.js.map +1 -1
- package/dist/helpers/tx_helpers.cjs +21 -20
- package/dist/helpers/tx_helpers.cjs.map +1 -1
- package/dist/helpers/tx_helpers.d.ts +11 -7
- package/dist/helpers/tx_helpers.js +20 -18
- package/dist/helpers/tx_helpers.js.map +1 -1
- package/dist/index.cjs +117 -106
- package/dist/index.d.ts +17 -17
- package/dist/index.js +17 -19
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.cjs +72 -66
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.cjs.map +1 -1
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.d.ts +35 -37
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.js +71 -64
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.js.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.cjs +12611 -12866
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.cjs.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.cjs +226 -226
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.cjs.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.d.ts +281 -330
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.js +225 -224
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.js.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.d.ts +2640 -4294
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.js +12608 -12706
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.js.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.cjs +1230 -1089
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.cjs.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.d.ts +393 -420
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.js +1228 -1083
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.js.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.cjs +142 -143
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.cjs.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.d.ts +62 -64
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.js +140 -141
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.js.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.cjs +572 -487
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.cjs.map +1 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.d.ts +125 -228
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.js +572 -485
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.js.map +1 -1
- package/dist/proto-grpc/google/protobuf/any.cjs +131 -146
- package/dist/proto-grpc/google/protobuf/any.cjs.map +1 -1
- package/dist/proto-grpc/google/protobuf/any.d.ts +78 -84
- package/dist/proto-grpc/google/protobuf/any.js +130 -144
- package/dist/proto-grpc/google/protobuf/any.js.map +1 -1
- package/dist/proto-grpc/google/protobuf/duration.cjs +92 -100
- package/dist/proto-grpc/google/protobuf/duration.cjs.map +1 -1
- package/dist/proto-grpc/google/protobuf/duration.d.ts +38 -43
- package/dist/proto-grpc/google/protobuf/duration.js +91 -98
- package/dist/proto-grpc/google/protobuf/duration.js.map +1 -1
- package/dist/proto-grpc/google/protobuf/timestamp.cjs +117 -128
- package/dist/proto-grpc/google/protobuf/timestamp.cjs.map +1 -1
- package/dist/proto-grpc/google/protobuf/timestamp.d.ts +50 -55
- package/dist/proto-grpc/google/protobuf/timestamp.js +116 -126
- package/dist/proto-grpc/google/protobuf/timestamp.js.map +1 -1
- package/dist/proto-grpc/google/rpc/code.cjs +223 -238
- package/dist/proto-grpc/google/rpc/code.cjs.map +1 -1
- package/dist/proto-grpc/google/rpc/code.d.ts +209 -206
- package/dist/proto-grpc/google/rpc/code.js +221 -237
- package/dist/proto-grpc/google/rpc/code.js.map +1 -1
- package/dist/proto-rest/index.cjs +67 -66
- package/dist/proto-rest/index.cjs.map +1 -1
- package/dist/proto-rest/index.d.ts +24 -18
- package/dist/proto-rest/index.js +61 -65
- package/dist/proto-rest/index.js.map +1 -1
- package/dist/proto-rest/plapi.d.ts +1400 -1477
- package/dist/test/tcp-proxy.cjs +100 -126
- package/dist/test/tcp-proxy.cjs.map +1 -1
- package/dist/test/tcp-proxy.d.ts +17 -13
- package/dist/test/tcp-proxy.js +97 -104
- package/dist/test/tcp-proxy.js.map +1 -1
- package/dist/test/test_config.cjs +145 -194
- package/dist/test/test_config.cjs.map +1 -1
- package/dist/test/test_config.d.ts +34 -30
- package/dist/test/test_config.js +138 -166
- package/dist/test/test_config.js.map +1 -1
- package/dist/util/pl.cjs +4 -3
- package/dist/util/pl.cjs.map +1 -1
- package/dist/util/pl.js +4 -2
- package/dist/util/pl.js.map +1 -1
- package/dist/util/util.cjs +7 -10
- package/dist/util/util.cjs.map +1 -1
- package/dist/util/util.js +7 -9
- package/dist/util/util.js.map +1 -1
- package/package.json +8 -8
- package/src/core/errors.ts +1 -1
- package/src/core/transaction.ts +123 -133
- package/src/core/websocket_stream.ts +10 -1
- package/dist/__external/.pnpm/@rollup_plugin-typescript@12.3.0_rollup@4.52.4_tslib@2.8.1_typescript@5.6.3/__external/tslib/tslib.es6.cjs +0 -61
- package/dist/__external/.pnpm/@rollup_plugin-typescript@12.3.0_rollup@4.52.4_tslib@2.8.1_typescript@5.6.3/__external/tslib/tslib.es6.cjs.map +0 -1
- package/dist/__external/.pnpm/@rollup_plugin-typescript@12.3.0_rollup@4.52.4_tslib@2.8.1_typescript@5.6.3/__external/tslib/tslib.es6.js +0 -58
- package/dist/__external/.pnpm/@rollup_plugin-typescript@12.3.0_rollup@4.52.4_tslib@2.8.1_typescript@5.6.3/__external/tslib/tslib.es6.js.map +0 -1
- package/dist/core/PromiseTracker.d.ts.map +0 -1
- package/dist/core/StatefulPromise.d.ts +0 -39
- package/dist/core/StatefulPromise.d.ts.map +0 -1
- package/dist/core/abstract_stream.d.ts.map +0 -1
- package/dist/core/advisory_locks.d.ts +0 -10
- package/dist/core/advisory_locks.d.ts.map +0 -1
- package/dist/core/auth.d.ts.map +0 -1
- package/dist/core/cache.d.ts.map +0 -1
- package/dist/core/client.d.ts.map +0 -1
- package/dist/core/config.d.ts.map +0 -1
- package/dist/core/default_client.d.ts.map +0 -1
- package/dist/core/driver.d.ts.map +0 -1
- package/dist/core/error_resource.d.ts +0 -6
- package/dist/core/error_resource.d.ts.map +0 -1
- package/dist/core/errors.d.ts.map +0 -1
- package/dist/core/final.d.ts.map +0 -1
- package/dist/core/ll_client.d.ts.map +0 -1
- package/dist/core/ll_transaction.d.ts.map +0 -1
- package/dist/core/stat.d.ts.map +0 -1
- package/dist/core/transaction.d.ts.map +0 -1
- package/dist/core/type_conversion.d.ts +0 -8
- package/dist/core/type_conversion.d.ts.map +0 -1
- package/dist/core/types.d.ts.map +0 -1
- package/dist/core/unauth_client.d.ts.map +0 -1
- package/dist/core/websocket_stream.d.ts +0 -67
- package/dist/core/websocket_stream.d.ts.map +0 -1
- package/dist/core/wire.d.ts.map +0 -1
- package/dist/helpers/pl.d.ts.map +0 -1
- package/dist/helpers/poll.d.ts.map +0 -1
- package/dist/helpers/retry_strategy.d.ts +0 -24
- package/dist/helpers/retry_strategy.d.ts.map +0 -1
- package/dist/helpers/state_helpers.d.ts +0 -3
- package/dist/helpers/state_helpers.d.ts.map +0 -1
- package/dist/helpers/tx_helpers.d.ts.map +0 -1
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/proto-grpc/github.com/googleapis/googleapis/google/rpc/status.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.client.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/api_types.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/base_types.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/import.d.ts +0 -106
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/import.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/resource_types.d.ts.map +0 -1
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/ws-test.d.ts +0 -73
- package/dist/proto-grpc/github.com/milaboratory/pl/plapi/plapiproto/ws-test.d.ts.map +0 -1
- package/dist/proto-grpc/google/api/http.d.ts +0 -456
- package/dist/proto-grpc/google/api/http.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/any.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/descriptor.d.ts +0 -2340
- package/dist/proto-grpc/google/protobuf/descriptor.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/duration.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/empty.d.ts +0 -32
- package/dist/proto-grpc/google/protobuf/empty.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/struct.d.ts +0 -187
- package/dist/proto-grpc/google/protobuf/struct.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/timestamp.d.ts.map +0 -1
- package/dist/proto-grpc/google/protobuf/wrappers.d.ts +0 -308
- package/dist/proto-grpc/google/protobuf/wrappers.d.ts.map +0 -1
- package/dist/proto-grpc/google/rpc/code.d.ts.map +0 -1
- package/dist/proto-grpc/google/rpc/error_details.d.ts +0 -654
- package/dist/proto-grpc/google/rpc/error_details.d.ts.map +0 -1
- package/dist/proto-grpc/google/rpc/http.d.ts +0 -121
- package/dist/proto-grpc/google/rpc/http.d.ts.map +0 -1
- package/dist/proto-grpc/google/rpc/status.d.ts +0 -55
- package/dist/proto-grpc/google/rpc/status.d.ts.map +0 -1
- package/dist/proto-rest/index.d.ts.map +0 -1
- package/dist/proto-rest/plapi.d.ts.map +0 -1
- package/dist/test/tcp-proxy.d.ts.map +0 -1
- package/dist/test/test_config.d.ts.map +0 -1
- package/dist/util/branding.d.ts +0 -7
- package/dist/util/branding.d.ts.map +0 -1
- package/dist/util/pl.d.ts +0 -9
- package/dist/util/pl.d.ts.map +0 -1
- package/dist/util/util.d.ts +0 -2
- package/dist/util/util.d.ts.map +0 -1
package/dist/helpers/poll.js
CHANGED
|
@@ -1,123 +1,107 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
3
|
-
import * as tp from
|
|
1
|
+
import { isNotNullResourceId, isNullResourceId, resourceIdToString } from "../core/types.js";
|
|
2
|
+
import { createRetryState, nextRetryStateOrError, notEmpty } from "@milaboratories/ts-helpers";
|
|
3
|
+
import * as tp from "node:timers/promises";
|
|
4
4
|
|
|
5
|
+
//#region src/helpers/poll.ts
|
|
5
6
|
/** This error tells state assertion mechanism that required state is not yet ready */
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
}
|
|
7
|
+
var ContinuePolling = class extends Error {
|
|
8
|
+
name = "ContinuePolling";
|
|
9
|
+
};
|
|
9
10
|
const DefaultPollFieldTraverseOps = {
|
|
10
|
-
|
|
11
|
-
|
|
11
|
+
failOnError: true,
|
|
12
|
+
onlyFinal: false
|
|
13
|
+
};
|
|
14
|
+
var PollResourceAccessor = class {
|
|
15
|
+
constructor(tx, data, path) {
|
|
16
|
+
this.tx = tx;
|
|
17
|
+
this.data = data;
|
|
18
|
+
this.path = path;
|
|
19
|
+
}
|
|
20
|
+
final() {
|
|
21
|
+
if (!this.data.final) throw new ContinuePolling();
|
|
22
|
+
return this;
|
|
23
|
+
}
|
|
24
|
+
async requireNoError() {
|
|
25
|
+
if (isNullResourceId(this.data.error)) return this;
|
|
26
|
+
await this.tx.throwError(this.data.error, this.path);
|
|
27
|
+
return this;
|
|
28
|
+
}
|
|
29
|
+
getFieldData(name, expectedType) {
|
|
30
|
+
const fieldData = this.data.fields.find((f) => f.name === name);
|
|
31
|
+
if (fieldData !== void 0) {
|
|
32
|
+
if (expectedType !== void 0 && fieldData.type !== expectedType) throw new Error(`Unexpected field type. Expected ${expectedType}, found ${fieldData.type}`);
|
|
33
|
+
return fieldData;
|
|
34
|
+
}
|
|
35
|
+
if ((expectedType === "Input" || expectedType === "Service") && this.data.inputsLocked || expectedType === "Output" && this.data.outputsLocked) throw new Error(`Field "${name}" not found. Expected type: ${expectedType}, state: ${this.data}`);
|
|
36
|
+
throw new ContinuePolling();
|
|
37
|
+
}
|
|
38
|
+
async get(name, ops = {}) {
|
|
39
|
+
const { expectedType, failOnError } = {
|
|
40
|
+
...DefaultPollFieldTraverseOps,
|
|
41
|
+
...ops
|
|
42
|
+
};
|
|
43
|
+
const path = [...this.path, name];
|
|
44
|
+
const fieldData = this.getFieldData(name, expectedType);
|
|
45
|
+
if (isNotNullResourceId(fieldData.error) && (failOnError || isNullResourceId(fieldData.value))) await this.tx.throwError(fieldData.error, path);
|
|
46
|
+
if (isNullResourceId(fieldData.value)) throw new ContinuePolling();
|
|
47
|
+
return await this.tx.get(fieldData.value, failOnError, path);
|
|
48
|
+
}
|
|
49
|
+
async getMulti(ops, ...names) {
|
|
50
|
+
return await Promise.all(names.map((name) => this.get(name, ops)));
|
|
51
|
+
}
|
|
52
|
+
async getMultiObj(ops, ...names) {
|
|
53
|
+
return Object.fromEntries(await Promise.all(names.map(async (name) => [name, await this.get(name, ops)])));
|
|
54
|
+
}
|
|
55
|
+
async getAllFinal(ops = {}) {
|
|
56
|
+
return await this.getMultiObj(ops, ...this.data.fields.filter((f) => f.valueIsFinal || isNotNullResourceId(f.error)).map((f) => f.name));
|
|
57
|
+
}
|
|
58
|
+
async getKValue(key) {
|
|
59
|
+
const value = await this.tx.tx.getKValueStringIfExists(this.data.id, key);
|
|
60
|
+
if (value === void 0) throw new ContinuePolling();
|
|
61
|
+
return value;
|
|
62
|
+
}
|
|
63
|
+
async getKValueObj(key) {
|
|
64
|
+
return JSON.parse(await this.getKValue(key));
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
var PollTxAccessor = class {
|
|
68
|
+
constructor(tx) {
|
|
69
|
+
this.tx = tx;
|
|
70
|
+
}
|
|
71
|
+
async get(rid, failOnError = true, path = []) {
|
|
72
|
+
const data = await this.tx.getResourceData(rid, true);
|
|
73
|
+
const accessor = new PollResourceAccessor(this, data, [...path, resourceIdToString(rid)]);
|
|
74
|
+
if (failOnError) await accessor.requireNoError();
|
|
75
|
+
return accessor;
|
|
76
|
+
}
|
|
77
|
+
async throwError(error, path = []) {
|
|
78
|
+
const errorRes = await this.get(error);
|
|
79
|
+
const errorText = Buffer.from(notEmpty(errorRes.data.data)).toString();
|
|
80
|
+
throw new Error(`${path.join(" -> ")} = ${errorText}`);
|
|
81
|
+
}
|
|
12
82
|
};
|
|
13
|
-
class PollResourceAccessor {
|
|
14
|
-
tx;
|
|
15
|
-
data;
|
|
16
|
-
path;
|
|
17
|
-
constructor(tx, data, path) {
|
|
18
|
-
this.tx = tx;
|
|
19
|
-
this.data = data;
|
|
20
|
-
this.path = path;
|
|
21
|
-
}
|
|
22
|
-
final() {
|
|
23
|
-
if (!this.data.final)
|
|
24
|
-
throw new ContinuePolling();
|
|
25
|
-
return this;
|
|
26
|
-
}
|
|
27
|
-
async requireNoError() {
|
|
28
|
-
if (isNullResourceId(this.data.error))
|
|
29
|
-
return this;
|
|
30
|
-
await this.tx.throwError(this.data.error, this.path);
|
|
31
|
-
// hmm... https://github.com/microsoft/TypeScript/issues/34955
|
|
32
|
-
return this;
|
|
33
|
-
}
|
|
34
|
-
getFieldData(name, expectedType) {
|
|
35
|
-
const fieldData = this.data.fields.find((f) => f.name === name);
|
|
36
|
-
if (fieldData !== undefined) {
|
|
37
|
-
if (expectedType !== undefined && fieldData.type !== expectedType)
|
|
38
|
-
throw new Error(`Unexpected field type. Expected ${expectedType}, found ${fieldData.type}`);
|
|
39
|
-
return fieldData;
|
|
40
|
-
}
|
|
41
|
-
if (((expectedType === "Input" || expectedType === "Service") && this.data.inputsLocked) ||
|
|
42
|
-
(expectedType === "Output" && this.data.outputsLocked))
|
|
43
|
-
throw new Error(
|
|
44
|
-
// eslint-disable-next-line @typescript-eslint/no-base-to-string, @typescript-eslint/restrict-template-expressions
|
|
45
|
-
`Field "${name}" not found. Expected type: ${expectedType}, state: ${this.data}`);
|
|
46
|
-
throw new ContinuePolling();
|
|
47
|
-
}
|
|
48
|
-
async get(name, ops = {}) {
|
|
49
|
-
const { expectedType, failOnError } = { ...DefaultPollFieldTraverseOps, ...ops };
|
|
50
|
-
const path = [...this.path, name];
|
|
51
|
-
const fieldData = this.getFieldData(name, expectedType);
|
|
52
|
-
if (isNotNullResourceId(fieldData.error) && (failOnError || isNullResourceId(fieldData.value)))
|
|
53
|
-
await this.tx.throwError(fieldData.error, path);
|
|
54
|
-
if (isNullResourceId(fieldData.value))
|
|
55
|
-
throw new ContinuePolling();
|
|
56
|
-
return await this.tx.get(fieldData.value, failOnError, path);
|
|
57
|
-
}
|
|
58
|
-
async getMulti(ops, ...names) {
|
|
59
|
-
return await Promise.all(names.map((name) => this.get(name, ops)));
|
|
60
|
-
}
|
|
61
|
-
async getMultiObj(ops, ...names) {
|
|
62
|
-
return Object.fromEntries(await Promise.all(names.map(async (name) => [name, await this.get(name, ops)])));
|
|
63
|
-
}
|
|
64
|
-
async getAllFinal(ops = {}) {
|
|
65
|
-
return await this.getMultiObj(ops, ...this.data.fields
|
|
66
|
-
.filter((f) => f.valueIsFinal || isNotNullResourceId(f.error))
|
|
67
|
-
.map((f) => f.name));
|
|
68
|
-
}
|
|
69
|
-
async getKValue(key) {
|
|
70
|
-
const value = await this.tx.tx.getKValueStringIfExists(this.data.id, key);
|
|
71
|
-
if (value === undefined)
|
|
72
|
-
throw new ContinuePolling();
|
|
73
|
-
return value;
|
|
74
|
-
}
|
|
75
|
-
async getKValueObj(key) {
|
|
76
|
-
return JSON.parse(await this.getKValue(key));
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
class PollTxAccessor {
|
|
80
|
-
tx;
|
|
81
|
-
constructor(tx) {
|
|
82
|
-
this.tx = tx;
|
|
83
|
-
}
|
|
84
|
-
async get(rid, failOnError = true, path = []) {
|
|
85
|
-
const data = await this.tx.getResourceData(rid, true);
|
|
86
|
-
const accessor = new PollResourceAccessor(this, data, [...path, resourceIdToString(rid)]);
|
|
87
|
-
if (failOnError)
|
|
88
|
-
await accessor.requireNoError();
|
|
89
|
-
return accessor;
|
|
90
|
-
}
|
|
91
|
-
async throwError(error, path = []) {
|
|
92
|
-
const errorRes = await this.get(error);
|
|
93
|
-
const errorText = Buffer.from(notEmpty(errorRes.data.data)).toString();
|
|
94
|
-
throw new Error(`${path.join(" -> ")} = ${errorText}`);
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
83
|
const DefaultPollingRetryOptions = {
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
84
|
+
type: "linearBackoff",
|
|
85
|
+
jitter: 0,
|
|
86
|
+
maxAttempts: 100,
|
|
87
|
+
backoffStep: 10,
|
|
88
|
+
initialDelay: 10
|
|
103
89
|
};
|
|
104
90
|
async function poll(cl, cb, retryOptions = DefaultPollingRetryOptions, txName = "polling") {
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
await tp.setTimeout(retryState.nextDelay);
|
|
118
|
-
retryState = nextRetryStateOrError(retryState);
|
|
119
|
-
}
|
|
91
|
+
let retryState = createRetryState(retryOptions);
|
|
92
|
+
while (true) {
|
|
93
|
+
try {
|
|
94
|
+
return await cl.withReadTx(txName, async (tx) => {
|
|
95
|
+
return await cb(new PollTxAccessor(tx));
|
|
96
|
+
});
|
|
97
|
+
} catch (e) {
|
|
98
|
+
if (!(e instanceof ContinuePolling)) throw e;
|
|
99
|
+
}
|
|
100
|
+
await tp.setTimeout(retryState.nextDelay);
|
|
101
|
+
retryState = nextRetryStateOrError(retryState);
|
|
102
|
+
}
|
|
120
103
|
}
|
|
121
104
|
|
|
105
|
+
//#endregion
|
|
122
106
|
export { ContinuePolling, DefaultPollingRetryOptions, PollResourceAccessor, PollTxAccessor, poll };
|
|
123
|
-
//# sourceMappingURL=poll.js.map
|
|
107
|
+
//# sourceMappingURL=poll.js.map
|
package/dist/helpers/poll.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"poll.js","sources":["../../src/helpers/poll.ts"],"sourcesContent":["import type { PlClient } from \"../core/client\";\nimport type { RetryOptions } from \"@milaboratories/ts-helpers\";\nimport { createRetryState, nextRetryStateOrError, notEmpty } from \"@milaboratories/ts-helpers\";\nimport type { FieldData, FieldType, ResourceData, ResourceId } from \"../core/types\";\nimport { isNotNullResourceId, isNullResourceId, resourceIdToString } from \"../core/types\";\nimport type { PlTransaction } from \"../core/transaction\";\nimport * as tp from \"node:timers/promises\";\n\n/** This error tells state assertion mechanism that required state is not yet ready */\nexport class ContinuePolling extends Error {\n name = \"ContinuePolling\";\n}\n\nexport type PollFieldTraverseOps = {\n expectedType?: FieldType;\n /** Fail if error present along with the value, if value not present,\n * but error do, exception will be thrown anyway. */\n failOnError: boolean;\n /** Traverse only if field report its value as final. */\n onlyFinal: boolean;\n};\n\nconst DefaultPollFieldTraverseOps: PollFieldTraverseOps = {\n failOnError: true,\n onlyFinal: false,\n};\n\nexport class PollResourceAccessor {\n constructor(\n public readonly tx: PollTxAccessor,\n public readonly data: ResourceData,\n public readonly path: string[],\n ) {}\n\n public final(): PollResourceAccessor {\n if (!this.data.final) throw new ContinuePolling();\n return this;\n }\n\n public async requireNoError(): Promise<PollResourceAccessor> {\n if (isNullResourceId(this.data.error)) return this;\n await this.tx.throwError(this.data.error, this.path);\n // hmm... https://github.com/microsoft/TypeScript/issues/34955\n return this;\n }\n\n public getFieldData(name: string, expectedType?: FieldType): FieldData {\n const fieldData = this.data.fields.find((f) => f.name === name);\n\n if (fieldData !== undefined) {\n if (expectedType !== undefined && fieldData.type !== expectedType)\n throw new Error(`Unexpected field type. Expected ${expectedType}, found ${fieldData.type}`);\n return fieldData;\n }\n\n if (\n ((expectedType === \"Input\" || expectedType === \"Service\") && this.data.inputsLocked) ||\n (expectedType === \"Output\" && this.data.outputsLocked)\n )\n throw new Error(\n // eslint-disable-next-line @typescript-eslint/no-base-to-string, @typescript-eslint/restrict-template-expressions\n `Field \"${name}\" not found. Expected type: ${expectedType}, state: ${this.data}`,\n );\n\n throw new ContinuePolling();\n }\n\n public async get(\n name: string,\n ops: Partial<PollFieldTraverseOps> = {},\n ): Promise<PollResourceAccessor> {\n const { expectedType, failOnError } = { ...DefaultPollFieldTraverseOps, ...ops };\n const path = [...this.path, name];\n\n const fieldData = this.getFieldData(name, expectedType);\n if (isNotNullResourceId(fieldData.error) && (failOnError || isNullResourceId(fieldData.value)))\n await this.tx.throwError(fieldData.error, path);\n\n if (isNullResourceId(fieldData.value)) throw new ContinuePolling();\n\n return await this.tx.get(fieldData.value, failOnError, path);\n }\n\n public async getMulti(\n ops: Partial<PollFieldTraverseOps>,\n ...names: string[]\n ): Promise<PollResourceAccessor[]> {\n return await Promise.all(names.map((name) => this.get(name, ops)));\n }\n\n public async getMultiObj<Key extends string>(\n ops: Partial<PollFieldTraverseOps>,\n ...names: Key[]\n ): Promise<Record<Key, PollResourceAccessor>> {\n return Object.fromEntries(\n await Promise.all(names.map(async (name) => [name, await this.get(name, ops)])),\n );\n }\n\n public async getAllFinal(\n ops: Partial<PollFieldTraverseOps> = {},\n ): Promise<Record<string, PollResourceAccessor>> {\n return await this.getMultiObj(\n ops,\n ...this.data.fields\n .filter((f) => f.valueIsFinal || isNotNullResourceId(f.error))\n .map((f) => f.name),\n );\n }\n\n public async getKValue(key: string): Promise<string> {\n const value = await this.tx.tx.getKValueStringIfExists(this.data.id, key);\n if (value === undefined) throw new ContinuePolling();\n return value;\n }\n\n public async getKValueObj<T>(key: string): Promise<T> {\n return JSON.parse(await this.getKValue(key)) as T;\n }\n}\n\nexport class PollTxAccessor {\n constructor(public readonly tx: PlTransaction) {}\n\n public async get(\n rid: ResourceId,\n failOnError: boolean = true,\n path: string[] = [],\n ): Promise<PollResourceAccessor> {\n const data = await this.tx.getResourceData(rid, true);\n const accessor = new PollResourceAccessor(this, data, [...path, resourceIdToString(rid)]);\n if (failOnError) await accessor.requireNoError();\n return accessor;\n }\n\n async throwError(error: ResourceId, path: string[] = []): Promise<never> {\n const errorRes = await this.get(error);\n const errorText = Buffer.from(notEmpty(errorRes.data.data)).toString();\n throw new Error(`${path.join(\" -> \")} = ${errorText}`);\n }\n}\n\nexport const DefaultPollingRetryOptions: RetryOptions = {\n type: \"linearBackoff\",\n jitter: 0,\n maxAttempts: 100,\n backoffStep: 10,\n initialDelay: 10,\n};\n\nexport async function poll<T>(\n cl: PlClient,\n cb: (tx: PollTxAccessor) => Promise<T>,\n retryOptions: RetryOptions = DefaultPollingRetryOptions,\n txName: string = \"polling\",\n): Promise<T> {\n let retryState = createRetryState(retryOptions);\n while (true) {\n try {\n return await cl.withReadTx(txName, async (tx) => {\n return await cb(new PollTxAccessor(tx));\n });\n } catch (e: any) {\n // Rethrowing any error except the \"not ready yet\"\n if (!(e instanceof ContinuePolling)) throw e;\n }\n await tp.setTimeout(retryState.nextDelay);\n retryState = nextRetryStateOrError(retryState);\n }\n}\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"poll.js","names":[],"sources":["../../src/helpers/poll.ts"],"sourcesContent":["import type { PlClient } from \"../core/client\";\nimport type { RetryOptions } from \"@milaboratories/ts-helpers\";\nimport { createRetryState, nextRetryStateOrError, notEmpty } from \"@milaboratories/ts-helpers\";\nimport type { FieldData, FieldType, ResourceData, ResourceId } from \"../core/types\";\nimport { isNotNullResourceId, isNullResourceId, resourceIdToString } from \"../core/types\";\nimport type { PlTransaction } from \"../core/transaction\";\nimport * as tp from \"node:timers/promises\";\n\n/** This error tells state assertion mechanism that required state is not yet ready */\nexport class ContinuePolling extends Error {\n name = \"ContinuePolling\";\n}\n\nexport type PollFieldTraverseOps = {\n expectedType?: FieldType;\n /** Fail if error present along with the value, if value not present,\n * but error do, exception will be thrown anyway. */\n failOnError: boolean;\n /** Traverse only if field report its value as final. */\n onlyFinal: boolean;\n};\n\nconst DefaultPollFieldTraverseOps: PollFieldTraverseOps = {\n failOnError: true,\n onlyFinal: false,\n};\n\nexport class PollResourceAccessor {\n constructor(\n public readonly tx: PollTxAccessor,\n public readonly data: ResourceData,\n public readonly path: string[],\n ) {}\n\n public final(): PollResourceAccessor {\n if (!this.data.final) throw new ContinuePolling();\n return this;\n }\n\n public async requireNoError(): Promise<PollResourceAccessor> {\n if (isNullResourceId(this.data.error)) return this;\n await this.tx.throwError(this.data.error, this.path);\n // hmm... https://github.com/microsoft/TypeScript/issues/34955\n return this;\n }\n\n public getFieldData(name: string, expectedType?: FieldType): FieldData {\n const fieldData = this.data.fields.find((f) => f.name === name);\n\n if (fieldData !== undefined) {\n if (expectedType !== undefined && fieldData.type !== expectedType)\n throw new Error(`Unexpected field type. Expected ${expectedType}, found ${fieldData.type}`);\n return fieldData;\n }\n\n if (\n ((expectedType === \"Input\" || expectedType === \"Service\") && this.data.inputsLocked) ||\n (expectedType === \"Output\" && this.data.outputsLocked)\n )\n throw new Error(\n // eslint-disable-next-line @typescript-eslint/no-base-to-string, @typescript-eslint/restrict-template-expressions\n `Field \"${name}\" not found. Expected type: ${expectedType}, state: ${this.data}`,\n );\n\n throw new ContinuePolling();\n }\n\n public async get(\n name: string,\n ops: Partial<PollFieldTraverseOps> = {},\n ): Promise<PollResourceAccessor> {\n const { expectedType, failOnError } = { ...DefaultPollFieldTraverseOps, ...ops };\n const path = [...this.path, name];\n\n const fieldData = this.getFieldData(name, expectedType);\n if (isNotNullResourceId(fieldData.error) && (failOnError || isNullResourceId(fieldData.value)))\n await this.tx.throwError(fieldData.error, path);\n\n if (isNullResourceId(fieldData.value)) throw new ContinuePolling();\n\n return await this.tx.get(fieldData.value, failOnError, path);\n }\n\n public async getMulti(\n ops: Partial<PollFieldTraverseOps>,\n ...names: string[]\n ): Promise<PollResourceAccessor[]> {\n return await Promise.all(names.map((name) => this.get(name, ops)));\n }\n\n public async getMultiObj<Key extends string>(\n ops: Partial<PollFieldTraverseOps>,\n ...names: Key[]\n ): Promise<Record<Key, PollResourceAccessor>> {\n return Object.fromEntries(\n await Promise.all(names.map(async (name) => [name, await this.get(name, ops)])),\n );\n }\n\n public async getAllFinal(\n ops: Partial<PollFieldTraverseOps> = {},\n ): Promise<Record<string, PollResourceAccessor>> {\n return await this.getMultiObj(\n ops,\n ...this.data.fields\n .filter((f) => f.valueIsFinal || isNotNullResourceId(f.error))\n .map((f) => f.name),\n );\n }\n\n public async getKValue(key: string): Promise<string> {\n const value = await this.tx.tx.getKValueStringIfExists(this.data.id, key);\n if (value === undefined) throw new ContinuePolling();\n return value;\n }\n\n public async getKValueObj<T>(key: string): Promise<T> {\n return JSON.parse(await this.getKValue(key)) as T;\n }\n}\n\nexport class PollTxAccessor {\n constructor(public readonly tx: PlTransaction) {}\n\n public async get(\n rid: ResourceId,\n failOnError: boolean = true,\n path: string[] = [],\n ): Promise<PollResourceAccessor> {\n const data = await this.tx.getResourceData(rid, true);\n const accessor = new PollResourceAccessor(this, data, [...path, resourceIdToString(rid)]);\n if (failOnError) await accessor.requireNoError();\n return accessor;\n }\n\n async throwError(error: ResourceId, path: string[] = []): Promise<never> {\n const errorRes = await this.get(error);\n const errorText = Buffer.from(notEmpty(errorRes.data.data)).toString();\n throw new Error(`${path.join(\" -> \")} = ${errorText}`);\n }\n}\n\nexport const DefaultPollingRetryOptions: RetryOptions = {\n type: \"linearBackoff\",\n jitter: 0,\n maxAttempts: 100,\n backoffStep: 10,\n initialDelay: 10,\n};\n\nexport async function poll<T>(\n cl: PlClient,\n cb: (tx: PollTxAccessor) => Promise<T>,\n retryOptions: RetryOptions = DefaultPollingRetryOptions,\n txName: string = \"polling\",\n): Promise<T> {\n let retryState = createRetryState(retryOptions);\n while (true) {\n try {\n return await cl.withReadTx(txName, async (tx) => {\n return await cb(new PollTxAccessor(tx));\n });\n } catch (e: any) {\n // Rethrowing any error except the \"not ready yet\"\n if (!(e instanceof ContinuePolling)) throw e;\n }\n await tp.setTimeout(retryState.nextDelay);\n retryState = nextRetryStateOrError(retryState);\n }\n}\n"],"mappings":"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"}
|
|
@@ -1,92 +1,87 @@
|
|
|
1
|
-
'use strict';
|
|
2
1
|
|
|
2
|
+
//#region src/helpers/retry_strategy.ts
|
|
3
3
|
const DEFAULT_RETRY_CONFIG = {
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
4
|
+
maxAttempts: 10,
|
|
5
|
+
initialDelay: 100,
|
|
6
|
+
maxDelay: 3e4
|
|
7
|
+
};
|
|
8
|
+
var RetryStrategy = class {
|
|
9
|
+
attempts = 0;
|
|
10
|
+
timer = null;
|
|
11
|
+
config;
|
|
12
|
+
callbacks;
|
|
13
|
+
backoff;
|
|
14
|
+
constructor(config, callbacks) {
|
|
15
|
+
this.config = {
|
|
16
|
+
...DEFAULT_RETRY_CONFIG,
|
|
17
|
+
...config
|
|
18
|
+
};
|
|
19
|
+
this.callbacks = callbacks;
|
|
20
|
+
this.backoff = new ExponentialBackoff({
|
|
21
|
+
initialDelay: this.config.initialDelay,
|
|
22
|
+
maxDelay: this.config.maxDelay,
|
|
23
|
+
factor: 2,
|
|
24
|
+
jitter: .1
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
schedule() {
|
|
28
|
+
if (this.timer) return;
|
|
29
|
+
if (this.hasExceededLimit()) {
|
|
30
|
+
this.notifyMaxAttemptsReached();
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
this.timer = setTimeout(() => {
|
|
34
|
+
this.timer = null;
|
|
35
|
+
this.attempts++;
|
|
36
|
+
this.callbacks.onRetry();
|
|
37
|
+
}, this.backoff.delay());
|
|
38
|
+
}
|
|
39
|
+
cancel() {
|
|
40
|
+
if (this.timer) {
|
|
41
|
+
clearTimeout(this.timer);
|
|
42
|
+
this.timer = null;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
reset() {
|
|
46
|
+
this.attempts = 0;
|
|
47
|
+
this.backoff.reset();
|
|
48
|
+
}
|
|
49
|
+
hasExceededLimit() {
|
|
50
|
+
return this.attempts >= this.config.maxAttempts;
|
|
51
|
+
}
|
|
52
|
+
notifyMaxAttemptsReached() {
|
|
53
|
+
const error = /* @__PURE__ */ new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);
|
|
54
|
+
this.callbacks.onMaxAttemptsReached(error);
|
|
55
|
+
}
|
|
56
|
+
};
|
|
57
|
+
var ExponentialBackoff = class {
|
|
58
|
+
initialDelay;
|
|
59
|
+
maxDelay;
|
|
60
|
+
currentDelay;
|
|
61
|
+
factor;
|
|
62
|
+
jitter;
|
|
63
|
+
constructor(config) {
|
|
64
|
+
this.initialDelay = config.initialDelay;
|
|
65
|
+
this.maxDelay = config.maxDelay;
|
|
66
|
+
this.factor = config.factor;
|
|
67
|
+
this.jitter = config.jitter;
|
|
68
|
+
this.currentDelay = config.initialDelay;
|
|
69
|
+
}
|
|
70
|
+
delay() {
|
|
71
|
+
if (this.currentDelay >= this.maxDelay) return this.applyJitter(this.maxDelay);
|
|
72
|
+
this.currentDelay = this.currentDelay * this.factor;
|
|
73
|
+
if (this.currentDelay > this.maxDelay) this.currentDelay = this.maxDelay;
|
|
74
|
+
return this.applyJitter(this.currentDelay);
|
|
75
|
+
}
|
|
76
|
+
reset() {
|
|
77
|
+
this.currentDelay = this.initialDelay;
|
|
78
|
+
}
|
|
79
|
+
applyJitter(delay) {
|
|
80
|
+
if (delay === 0 || this.jitter === 0) return delay;
|
|
81
|
+
return delay * (1 - this.jitter / 2 + Math.random() * this.jitter);
|
|
82
|
+
}
|
|
7
83
|
};
|
|
8
|
-
class RetryStrategy {
|
|
9
|
-
attempts = 0;
|
|
10
|
-
timer = null;
|
|
11
|
-
config;
|
|
12
|
-
callbacks;
|
|
13
|
-
backoff;
|
|
14
|
-
constructor(config, callbacks) {
|
|
15
|
-
this.config = { ...DEFAULT_RETRY_CONFIG, ...config };
|
|
16
|
-
this.callbacks = callbacks;
|
|
17
|
-
this.backoff = new ExponentialBackoff({
|
|
18
|
-
initialDelay: this.config.initialDelay,
|
|
19
|
-
maxDelay: this.config.maxDelay,
|
|
20
|
-
factor: 2,
|
|
21
|
-
jitter: 0.1,
|
|
22
|
-
});
|
|
23
|
-
}
|
|
24
|
-
schedule() {
|
|
25
|
-
if (this.timer)
|
|
26
|
-
return;
|
|
27
|
-
if (this.hasExceededLimit()) {
|
|
28
|
-
this.notifyMaxAttemptsReached();
|
|
29
|
-
return;
|
|
30
|
-
}
|
|
31
|
-
this.timer = setTimeout(() => {
|
|
32
|
-
this.timer = null;
|
|
33
|
-
this.attempts++;
|
|
34
|
-
this.callbacks.onRetry();
|
|
35
|
-
}, this.backoff.delay());
|
|
36
|
-
}
|
|
37
|
-
cancel() {
|
|
38
|
-
if (this.timer) {
|
|
39
|
-
clearTimeout(this.timer);
|
|
40
|
-
this.timer = null;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
reset() {
|
|
44
|
-
this.attempts = 0;
|
|
45
|
-
this.backoff.reset();
|
|
46
|
-
}
|
|
47
|
-
hasExceededLimit() {
|
|
48
|
-
return this.attempts >= this.config.maxAttempts;
|
|
49
|
-
}
|
|
50
|
-
notifyMaxAttemptsReached() {
|
|
51
|
-
const error = new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);
|
|
52
|
-
this.callbacks.onMaxAttemptsReached(error);
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
class ExponentialBackoff {
|
|
56
|
-
initialDelay;
|
|
57
|
-
maxDelay;
|
|
58
|
-
currentDelay;
|
|
59
|
-
factor;
|
|
60
|
-
jitter;
|
|
61
|
-
constructor(config) {
|
|
62
|
-
this.initialDelay = config.initialDelay;
|
|
63
|
-
this.maxDelay = config.maxDelay;
|
|
64
|
-
this.factor = config.factor;
|
|
65
|
-
this.jitter = config.jitter;
|
|
66
|
-
this.currentDelay = config.initialDelay;
|
|
67
|
-
}
|
|
68
|
-
delay() {
|
|
69
|
-
if (this.currentDelay >= this.maxDelay) {
|
|
70
|
-
return this.applyJitter(this.maxDelay);
|
|
71
|
-
}
|
|
72
|
-
this.currentDelay = this.currentDelay * this.factor;
|
|
73
|
-
if (this.currentDelay > this.maxDelay) {
|
|
74
|
-
this.currentDelay = this.maxDelay;
|
|
75
|
-
}
|
|
76
|
-
return this.applyJitter(this.currentDelay);
|
|
77
|
-
}
|
|
78
|
-
reset() {
|
|
79
|
-
this.currentDelay = this.initialDelay;
|
|
80
|
-
}
|
|
81
|
-
applyJitter(delay) {
|
|
82
|
-
if (delay === 0 || this.jitter === 0) {
|
|
83
|
-
return delay;
|
|
84
|
-
}
|
|
85
|
-
const delayFactor = 1 - this.jitter / 2 + Math.random() * this.jitter;
|
|
86
|
-
return delay * delayFactor;
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
84
|
|
|
90
|
-
|
|
85
|
+
//#endregion
|
|
91
86
|
exports.RetryStrategy = RetryStrategy;
|
|
92
|
-
//# sourceMappingURL=retry_strategy.cjs.map
|
|
87
|
+
//# sourceMappingURL=retry_strategy.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"retry_strategy.cjs","sources":["../../src/helpers/retry_strategy.ts"],"sourcesContent":["export interface RetryConfig {\n maxAttempts: number;\n initialDelay: number;\n maxDelay: number;\n}\n\nexport const DEFAULT_RETRY_CONFIG: RetryConfig = {\n maxAttempts: 10,\n initialDelay: 100,\n maxDelay: 30000,\n};\n\nexport interface RetryCallbacks {\n onRetry: () => void;\n onMaxAttemptsReached: (error: Error) => void;\n}\n\nexport class RetryStrategy {\n private attempts = 0;\n private timer: ReturnType<typeof setTimeout> | null = null;\n private readonly config: RetryConfig;\n private readonly callbacks: RetryCallbacks;\n private readonly backoff: ExponentialBackoff;\n\n constructor(config: Partial<RetryConfig>, callbacks: RetryCallbacks) {\n this.config = { ...DEFAULT_RETRY_CONFIG, ...config };\n this.callbacks = callbacks;\n this.backoff = new ExponentialBackoff({\n initialDelay: this.config.initialDelay,\n maxDelay: this.config.maxDelay,\n factor: 2,\n jitter: 0.1,\n });\n }\n\n schedule(): void {\n if (this.timer) return;\n if (this.hasExceededLimit()) {\n this.notifyMaxAttemptsReached();\n return;\n }\n\n this.timer = setTimeout(() => {\n this.timer = null;\n this.attempts++;\n this.callbacks.onRetry();\n }, this.backoff.delay());\n }\n\n cancel(): void {\n if (this.timer) {\n clearTimeout(this.timer);\n this.timer = null;\n }\n }\n\n reset(): void {\n this.attempts = 0;\n this.backoff.reset();\n }\n\n private hasExceededLimit(): boolean {\n return this.attempts >= this.config.maxAttempts;\n }\n\n private notifyMaxAttemptsReached(): void {\n const error = new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);\n this.callbacks.onMaxAttemptsReached(error);\n }\n}\n\ninterface ExponentialBackoffConfig {\n initialDelay: number;\n maxDelay: number;\n factor: number;\n jitter: number;\n}\n\nclass ExponentialBackoff {\n private readonly initialDelay: number;\n private readonly maxDelay: number;\n\n private currentDelay: number;\n\n private readonly factor: number;\n private readonly jitter: number;\n\n constructor(config: ExponentialBackoffConfig) {\n this.initialDelay = config.initialDelay;\n this.maxDelay = config.maxDelay;\n this.factor = config.factor;\n this.jitter = config.jitter;\n this.currentDelay = config.initialDelay;\n }\n\n delay(): number {\n if (this.currentDelay >= this.maxDelay) {\n return this.applyJitter(this.maxDelay);\n }\n\n this.currentDelay = this.currentDelay * this.factor;\n\n if (this.currentDelay > this.maxDelay) {\n this.currentDelay = this.maxDelay;\n }\n\n return this.applyJitter(this.currentDelay);\n }\n\n reset(): void {\n this.currentDelay = this.initialDelay;\n }\n\n private applyJitter(delay: number): number {\n if (delay === 0 || this.jitter === 0) {\n return delay;\n }\n const delayFactor = 1 - this.jitter / 2 + Math.random() * this.jitter;\n return delay * delayFactor;\n }\n}\n"],"
|
|
1
|
+
{"version":3,"file":"retry_strategy.cjs","names":[],"sources":["../../src/helpers/retry_strategy.ts"],"sourcesContent":["export interface RetryConfig {\n maxAttempts: number;\n initialDelay: number;\n maxDelay: number;\n}\n\nexport const DEFAULT_RETRY_CONFIG: RetryConfig = {\n maxAttempts: 10,\n initialDelay: 100,\n maxDelay: 30000,\n};\n\nexport interface RetryCallbacks {\n onRetry: () => void;\n onMaxAttemptsReached: (error: Error) => void;\n}\n\nexport class RetryStrategy {\n private attempts = 0;\n private timer: ReturnType<typeof setTimeout> | null = null;\n private readonly config: RetryConfig;\n private readonly callbacks: RetryCallbacks;\n private readonly backoff: ExponentialBackoff;\n\n constructor(config: Partial<RetryConfig>, callbacks: RetryCallbacks) {\n this.config = { ...DEFAULT_RETRY_CONFIG, ...config };\n this.callbacks = callbacks;\n this.backoff = new ExponentialBackoff({\n initialDelay: this.config.initialDelay,\n maxDelay: this.config.maxDelay,\n factor: 2,\n jitter: 0.1,\n });\n }\n\n schedule(): void {\n if (this.timer) return;\n if (this.hasExceededLimit()) {\n this.notifyMaxAttemptsReached();\n return;\n }\n\n this.timer = setTimeout(() => {\n this.timer = null;\n this.attempts++;\n this.callbacks.onRetry();\n }, this.backoff.delay());\n }\n\n cancel(): void {\n if (this.timer) {\n clearTimeout(this.timer);\n this.timer = null;\n }\n }\n\n reset(): void {\n this.attempts = 0;\n this.backoff.reset();\n }\n\n private hasExceededLimit(): boolean {\n return this.attempts >= this.config.maxAttempts;\n }\n\n private notifyMaxAttemptsReached(): void {\n const error = new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);\n this.callbacks.onMaxAttemptsReached(error);\n }\n}\n\ninterface ExponentialBackoffConfig {\n initialDelay: number;\n maxDelay: number;\n factor: number;\n jitter: number;\n}\n\nclass ExponentialBackoff {\n private readonly initialDelay: number;\n private readonly maxDelay: number;\n\n private currentDelay: number;\n\n private readonly factor: number;\n private readonly jitter: number;\n\n constructor(config: ExponentialBackoffConfig) {\n this.initialDelay = config.initialDelay;\n this.maxDelay = config.maxDelay;\n this.factor = config.factor;\n this.jitter = config.jitter;\n this.currentDelay = config.initialDelay;\n }\n\n delay(): number {\n if (this.currentDelay >= this.maxDelay) {\n return this.applyJitter(this.maxDelay);\n }\n\n this.currentDelay = this.currentDelay * this.factor;\n\n if (this.currentDelay > this.maxDelay) {\n this.currentDelay = this.maxDelay;\n }\n\n return this.applyJitter(this.currentDelay);\n }\n\n reset(): void {\n this.currentDelay = this.initialDelay;\n }\n\n private applyJitter(delay: number): number {\n if (delay === 0 || this.jitter === 0) {\n return delay;\n }\n const delayFactor = 1 - this.jitter / 2 + Math.random() * this.jitter;\n return delay * delayFactor;\n }\n}\n"],"mappings":";;AAMA,MAAa,uBAAoC;CAC/C,aAAa;CACb,cAAc;CACd,UAAU;CACX;AAOD,IAAa,gBAAb,MAA2B;CACzB,AAAQ,WAAW;CACnB,AAAQ,QAA8C;CACtD,AAAiB;CACjB,AAAiB;CACjB,AAAiB;CAEjB,YAAY,QAA8B,WAA2B;AACnE,OAAK,SAAS;GAAE,GAAG;GAAsB,GAAG;GAAQ;AACpD,OAAK,YAAY;AACjB,OAAK,UAAU,IAAI,mBAAmB;GACpC,cAAc,KAAK,OAAO;GAC1B,UAAU,KAAK,OAAO;GACtB,QAAQ;GACR,QAAQ;GACT,CAAC;;CAGJ,WAAiB;AACf,MAAI,KAAK,MAAO;AAChB,MAAI,KAAK,kBAAkB,EAAE;AAC3B,QAAK,0BAA0B;AAC/B;;AAGF,OAAK,QAAQ,iBAAiB;AAC5B,QAAK,QAAQ;AACb,QAAK;AACL,QAAK,UAAU,SAAS;KACvB,KAAK,QAAQ,OAAO,CAAC;;CAG1B,SAAe;AACb,MAAI,KAAK,OAAO;AACd,gBAAa,KAAK,MAAM;AACxB,QAAK,QAAQ;;;CAIjB,QAAc;AACZ,OAAK,WAAW;AAChB,OAAK,QAAQ,OAAO;;CAGtB,AAAQ,mBAA4B;AAClC,SAAO,KAAK,YAAY,KAAK,OAAO;;CAGtC,AAAQ,2BAAiC;EACvC,MAAM,wBAAQ,IAAI,MAAM,uBAAuB,KAAK,OAAO,YAAY,WAAW;AAClF,OAAK,UAAU,qBAAqB,MAAM;;;AAW9C,IAAM,qBAAN,MAAyB;CACvB,AAAiB;CACjB,AAAiB;CAEjB,AAAQ;CAER,AAAiB;CACjB,AAAiB;CAEjB,YAAY,QAAkC;AAC5C,OAAK,eAAe,OAAO;AAC3B,OAAK,WAAW,OAAO;AACvB,OAAK,SAAS,OAAO;AACrB,OAAK,SAAS,OAAO;AACrB,OAAK,eAAe,OAAO;;CAG7B,QAAgB;AACd,MAAI,KAAK,gBAAgB,KAAK,SAC5B,QAAO,KAAK,YAAY,KAAK,SAAS;AAGxC,OAAK,eAAe,KAAK,eAAe,KAAK;AAE7C,MAAI,KAAK,eAAe,KAAK,SAC3B,MAAK,eAAe,KAAK;AAG3B,SAAO,KAAK,YAAY,KAAK,aAAa;;CAG5C,QAAc;AACZ,OAAK,eAAe,KAAK;;CAG3B,AAAQ,YAAY,OAAuB;AACzC,MAAI,UAAU,KAAK,KAAK,WAAW,EACjC,QAAO;AAGT,SAAO,SADa,IAAI,KAAK,SAAS,IAAI,KAAK,QAAQ,GAAG,KAAK"}
|
|
@@ -1,89 +1,86 @@
|
|
|
1
|
+
//#region src/helpers/retry_strategy.ts
|
|
1
2
|
const DEFAULT_RETRY_CONFIG = {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
maxAttempts: 10,
|
|
4
|
+
initialDelay: 100,
|
|
5
|
+
maxDelay: 3e4
|
|
6
|
+
};
|
|
7
|
+
var RetryStrategy = class {
|
|
8
|
+
attempts = 0;
|
|
9
|
+
timer = null;
|
|
10
|
+
config;
|
|
11
|
+
callbacks;
|
|
12
|
+
backoff;
|
|
13
|
+
constructor(config, callbacks) {
|
|
14
|
+
this.config = {
|
|
15
|
+
...DEFAULT_RETRY_CONFIG,
|
|
16
|
+
...config
|
|
17
|
+
};
|
|
18
|
+
this.callbacks = callbacks;
|
|
19
|
+
this.backoff = new ExponentialBackoff({
|
|
20
|
+
initialDelay: this.config.initialDelay,
|
|
21
|
+
maxDelay: this.config.maxDelay,
|
|
22
|
+
factor: 2,
|
|
23
|
+
jitter: .1
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
schedule() {
|
|
27
|
+
if (this.timer) return;
|
|
28
|
+
if (this.hasExceededLimit()) {
|
|
29
|
+
this.notifyMaxAttemptsReached();
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
this.timer = setTimeout(() => {
|
|
33
|
+
this.timer = null;
|
|
34
|
+
this.attempts++;
|
|
35
|
+
this.callbacks.onRetry();
|
|
36
|
+
}, this.backoff.delay());
|
|
37
|
+
}
|
|
38
|
+
cancel() {
|
|
39
|
+
if (this.timer) {
|
|
40
|
+
clearTimeout(this.timer);
|
|
41
|
+
this.timer = null;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
reset() {
|
|
45
|
+
this.attempts = 0;
|
|
46
|
+
this.backoff.reset();
|
|
47
|
+
}
|
|
48
|
+
hasExceededLimit() {
|
|
49
|
+
return this.attempts >= this.config.maxAttempts;
|
|
50
|
+
}
|
|
51
|
+
notifyMaxAttemptsReached() {
|
|
52
|
+
const error = /* @__PURE__ */ new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);
|
|
53
|
+
this.callbacks.onMaxAttemptsReached(error);
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
var ExponentialBackoff = class {
|
|
57
|
+
initialDelay;
|
|
58
|
+
maxDelay;
|
|
59
|
+
currentDelay;
|
|
60
|
+
factor;
|
|
61
|
+
jitter;
|
|
62
|
+
constructor(config) {
|
|
63
|
+
this.initialDelay = config.initialDelay;
|
|
64
|
+
this.maxDelay = config.maxDelay;
|
|
65
|
+
this.factor = config.factor;
|
|
66
|
+
this.jitter = config.jitter;
|
|
67
|
+
this.currentDelay = config.initialDelay;
|
|
68
|
+
}
|
|
69
|
+
delay() {
|
|
70
|
+
if (this.currentDelay >= this.maxDelay) return this.applyJitter(this.maxDelay);
|
|
71
|
+
this.currentDelay = this.currentDelay * this.factor;
|
|
72
|
+
if (this.currentDelay > this.maxDelay) this.currentDelay = this.maxDelay;
|
|
73
|
+
return this.applyJitter(this.currentDelay);
|
|
74
|
+
}
|
|
75
|
+
reset() {
|
|
76
|
+
this.currentDelay = this.initialDelay;
|
|
77
|
+
}
|
|
78
|
+
applyJitter(delay) {
|
|
79
|
+
if (delay === 0 || this.jitter === 0) return delay;
|
|
80
|
+
return delay * (1 - this.jitter / 2 + Math.random() * this.jitter);
|
|
81
|
+
}
|
|
5
82
|
};
|
|
6
|
-
class RetryStrategy {
|
|
7
|
-
attempts = 0;
|
|
8
|
-
timer = null;
|
|
9
|
-
config;
|
|
10
|
-
callbacks;
|
|
11
|
-
backoff;
|
|
12
|
-
constructor(config, callbacks) {
|
|
13
|
-
this.config = { ...DEFAULT_RETRY_CONFIG, ...config };
|
|
14
|
-
this.callbacks = callbacks;
|
|
15
|
-
this.backoff = new ExponentialBackoff({
|
|
16
|
-
initialDelay: this.config.initialDelay,
|
|
17
|
-
maxDelay: this.config.maxDelay,
|
|
18
|
-
factor: 2,
|
|
19
|
-
jitter: 0.1,
|
|
20
|
-
});
|
|
21
|
-
}
|
|
22
|
-
schedule() {
|
|
23
|
-
if (this.timer)
|
|
24
|
-
return;
|
|
25
|
-
if (this.hasExceededLimit()) {
|
|
26
|
-
this.notifyMaxAttemptsReached();
|
|
27
|
-
return;
|
|
28
|
-
}
|
|
29
|
-
this.timer = setTimeout(() => {
|
|
30
|
-
this.timer = null;
|
|
31
|
-
this.attempts++;
|
|
32
|
-
this.callbacks.onRetry();
|
|
33
|
-
}, this.backoff.delay());
|
|
34
|
-
}
|
|
35
|
-
cancel() {
|
|
36
|
-
if (this.timer) {
|
|
37
|
-
clearTimeout(this.timer);
|
|
38
|
-
this.timer = null;
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
reset() {
|
|
42
|
-
this.attempts = 0;
|
|
43
|
-
this.backoff.reset();
|
|
44
|
-
}
|
|
45
|
-
hasExceededLimit() {
|
|
46
|
-
return this.attempts >= this.config.maxAttempts;
|
|
47
|
-
}
|
|
48
|
-
notifyMaxAttemptsReached() {
|
|
49
|
-
const error = new Error(`Max retry attempts (${this.config.maxAttempts}) reached`);
|
|
50
|
-
this.callbacks.onMaxAttemptsReached(error);
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
class ExponentialBackoff {
|
|
54
|
-
initialDelay;
|
|
55
|
-
maxDelay;
|
|
56
|
-
currentDelay;
|
|
57
|
-
factor;
|
|
58
|
-
jitter;
|
|
59
|
-
constructor(config) {
|
|
60
|
-
this.initialDelay = config.initialDelay;
|
|
61
|
-
this.maxDelay = config.maxDelay;
|
|
62
|
-
this.factor = config.factor;
|
|
63
|
-
this.jitter = config.jitter;
|
|
64
|
-
this.currentDelay = config.initialDelay;
|
|
65
|
-
}
|
|
66
|
-
delay() {
|
|
67
|
-
if (this.currentDelay >= this.maxDelay) {
|
|
68
|
-
return this.applyJitter(this.maxDelay);
|
|
69
|
-
}
|
|
70
|
-
this.currentDelay = this.currentDelay * this.factor;
|
|
71
|
-
if (this.currentDelay > this.maxDelay) {
|
|
72
|
-
this.currentDelay = this.maxDelay;
|
|
73
|
-
}
|
|
74
|
-
return this.applyJitter(this.currentDelay);
|
|
75
|
-
}
|
|
76
|
-
reset() {
|
|
77
|
-
this.currentDelay = this.initialDelay;
|
|
78
|
-
}
|
|
79
|
-
applyJitter(delay) {
|
|
80
|
-
if (delay === 0 || this.jitter === 0) {
|
|
81
|
-
return delay;
|
|
82
|
-
}
|
|
83
|
-
const delayFactor = 1 - this.jitter / 2 + Math.random() * this.jitter;
|
|
84
|
-
return delay * delayFactor;
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
83
|
|
|
88
|
-
|
|
89
|
-
|
|
84
|
+
//#endregion
|
|
85
|
+
export { RetryStrategy };
|
|
86
|
+
//# sourceMappingURL=retry_strategy.js.map
|