accessio 1.2.0 → 1.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 +21 -21
- package/cjs/accessio.cjs +68 -84
- package/cjs/accessio.cjs.map +1 -1
- package/cjs/core/accessioError.cjs +49 -3
- package/cjs/core/accessioError.cjs.map +1 -1
- package/cjs/core/buildURL.cjs +10 -4
- package/cjs/core/buildURL.cjs.map +1 -1
- package/cjs/core/fetchAdapter.cjs +134 -111
- package/cjs/core/fetchAdapter.cjs.map +1 -1
- package/cjs/core/request.cjs +97 -24
- package/cjs/core/request.cjs.map +1 -1
- package/cjs/core/retry.cjs +25 -0
- package/cjs/core/retry.cjs.map +1 -1
- package/cjs/helpers/debug.cjs +7 -1
- package/cjs/helpers/debug.cjs.map +1 -1
- package/cjs/helpers/flattenHeaders.cjs +37 -0
- package/cjs/helpers/flattenHeaders.cjs.map +1 -1
- package/cjs/helpers/rateLimiter.cjs +11 -22
- package/cjs/helpers/rateLimiter.cjs.map +1 -1
- package/cjs/helpers/settle.cjs +1 -1
- package/cjs/helpers/settle.cjs.map +1 -1
- package/cjs/helpers/transformData.cjs +2 -2
- package/cjs/helpers/transformData.cjs.map +1 -1
- package/cjs/interceptors/interceptorManager.cjs +25 -18
- package/cjs/interceptors/interceptorManager.cjs.map +1 -1
- package/index.d.ts +89 -21
- package/package.json +2 -2
- package/src/accessio.ts +104 -98
- package/src/core/accessioError.ts +50 -1
- package/src/core/buildURL.ts +14 -4
- package/src/core/fetchAdapter.ts +166 -130
- package/src/core/request.ts +115 -28
- package/src/core/retry.ts +19 -1
- package/src/helpers/debug.ts +7 -2
- package/src/helpers/flattenHeaders.ts +30 -0
- package/src/helpers/rateLimiter.ts +11 -24
- package/src/helpers/settle.ts +1 -1
- package/src/helpers/transformData.ts +2 -1
- package/src/interceptors/interceptorManager.ts +26 -19
- package/src/types.ts +1 -0
|
@@ -21,10 +21,7 @@ export function createRateLimiter(
|
|
|
21
21
|
}
|
|
22
22
|
let active = 0;
|
|
23
23
|
let destroyed = false;
|
|
24
|
-
|
|
25
|
-
let tail = 0;
|
|
26
|
-
let pendingCount = 0;
|
|
27
|
-
const queue: Record<number, QueueItem> = {};
|
|
24
|
+
const queue: QueueItem[] = [];
|
|
28
25
|
|
|
29
26
|
function acquire(): Promise<void> {
|
|
30
27
|
if (destroyed) {
|
|
@@ -36,44 +33,34 @@ export function createRateLimiter(
|
|
|
36
33
|
return Promise.resolve();
|
|
37
34
|
}
|
|
38
35
|
|
|
39
|
-
if (
|
|
36
|
+
if (queue.length >= maxQueueSize) {
|
|
40
37
|
return Promise.reject(
|
|
41
38
|
new Error(`[Accessio] Rate limiter queue size exceeded maxQueueSize (${maxQueueSize})`),
|
|
42
39
|
);
|
|
43
40
|
}
|
|
44
41
|
|
|
45
42
|
return new Promise((resolve, reject) => {
|
|
46
|
-
queue
|
|
47
|
-
pendingCount++;
|
|
43
|
+
queue.push({ resolve, reject });
|
|
48
44
|
});
|
|
49
45
|
}
|
|
50
46
|
|
|
51
47
|
function release(): void {
|
|
52
48
|
if (destroyed) return;
|
|
53
|
-
|
|
54
49
|
if (active <= 0) return;
|
|
55
50
|
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
const next = queue[head];
|
|
61
|
-
delete queue[head];
|
|
62
|
-
head++;
|
|
63
|
-
pendingCount--;
|
|
64
|
-
next?.resolve();
|
|
51
|
+
const next = queue.shift();
|
|
52
|
+
if (next) {
|
|
53
|
+
next.resolve();
|
|
54
|
+
return;
|
|
65
55
|
}
|
|
56
|
+
active--;
|
|
66
57
|
}
|
|
67
58
|
|
|
68
59
|
function destroy(): void {
|
|
69
60
|
destroyed = true;
|
|
70
61
|
const reason = new Error('[Accessio] Rate limiter destroyed — pending request cancelled');
|
|
71
|
-
while (
|
|
72
|
-
|
|
73
|
-
delete queue[head];
|
|
74
|
-
head++;
|
|
75
|
-
pendingCount--;
|
|
76
|
-
next?.reject(reason);
|
|
62
|
+
while (queue.length > 0) {
|
|
63
|
+
queue.shift()!.reject(reason);
|
|
77
64
|
}
|
|
78
65
|
}
|
|
79
66
|
|
|
@@ -82,7 +69,7 @@ export function createRateLimiter(
|
|
|
82
69
|
release,
|
|
83
70
|
destroy,
|
|
84
71
|
get pending() {
|
|
85
|
-
return
|
|
72
|
+
return queue.length;
|
|
86
73
|
},
|
|
87
74
|
get active() {
|
|
88
75
|
return active;
|
package/src/helpers/settle.ts
CHANGED
|
@@ -9,7 +9,7 @@ export default function settle(
|
|
|
9
9
|
): void {
|
|
10
10
|
const validateStatus = config.validateStatus;
|
|
11
11
|
|
|
12
|
-
if (!
|
|
12
|
+
if (!validateStatus || validateStatus(response.status)) {
|
|
13
13
|
resolve(response);
|
|
14
14
|
} else {
|
|
15
15
|
const error = new AccessioError(
|
|
@@ -6,6 +6,7 @@ export default async function transformData(
|
|
|
6
6
|
data: unknown,
|
|
7
7
|
headers: Record<string, string | string[]>,
|
|
8
8
|
config?: AccessioRequestConfig,
|
|
9
|
+
direction: 'request' | 'response' = 'request',
|
|
9
10
|
): Promise<unknown> {
|
|
10
11
|
if (!transforms || !Array.isArray(transforms)) {
|
|
11
12
|
return data;
|
|
@@ -20,7 +21,7 @@ export default async function transformData(
|
|
|
20
21
|
} catch (err) {
|
|
21
22
|
throw AccessioError.from(
|
|
22
23
|
err instanceof Error ? err : new Error(String(err)),
|
|
23
|
-
AccessioError.ERR_BAD_REQUEST,
|
|
24
|
+
direction === 'response' ? AccessioError.ERR_BAD_RESPONSE : AccessioError.ERR_BAD_REQUEST,
|
|
24
25
|
config ?? null,
|
|
25
26
|
null,
|
|
26
27
|
null,
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import type { TransformFunction, InterceptorHandler, InterceptorOptions } from '../types';
|
|
2
2
|
|
|
3
3
|
export class InterceptorManager {
|
|
4
|
-
|
|
5
|
-
private
|
|
4
|
+
private _handlers: Map<number, InterceptorHandler>;
|
|
5
|
+
private _nextId: number;
|
|
6
6
|
|
|
7
7
|
constructor() {
|
|
8
|
-
this.
|
|
9
|
-
this.
|
|
8
|
+
this._handlers = new Map();
|
|
9
|
+
this._nextId = 0;
|
|
10
10
|
}
|
|
11
11
|
|
|
12
12
|
use(
|
|
@@ -14,39 +14,46 @@ export class InterceptorManager {
|
|
|
14
14
|
rejected?: ((error: unknown) => unknown) | null,
|
|
15
15
|
options: InterceptorOptions = {},
|
|
16
16
|
): number {
|
|
17
|
-
this.
|
|
17
|
+
const id = this._nextId++;
|
|
18
|
+
this._handlers.set(id, {
|
|
18
19
|
fulfilled: fulfilled || null,
|
|
19
20
|
rejected: rejected || null,
|
|
20
21
|
synchronous: options.synchronous || false,
|
|
21
22
|
runWhen: options.runWhen || null,
|
|
22
23
|
});
|
|
23
|
-
|
|
24
|
-
this._activeCount++;
|
|
25
|
-
return this.handlers.length - 1;
|
|
24
|
+
return id;
|
|
26
25
|
}
|
|
27
26
|
|
|
28
27
|
eject(id: number): void {
|
|
29
|
-
|
|
30
|
-
this.handlers[id] = null;
|
|
31
|
-
this._activeCount--;
|
|
32
|
-
}
|
|
28
|
+
this._handlers.delete(id);
|
|
33
29
|
}
|
|
34
30
|
|
|
35
31
|
clear(): void {
|
|
36
|
-
this.
|
|
37
|
-
this._activeCount = 0;
|
|
32
|
+
this._handlers.clear();
|
|
38
33
|
}
|
|
39
34
|
|
|
40
35
|
forEach(fn: (handler: InterceptorHandler) => void): void {
|
|
41
|
-
for (const handler of this.
|
|
42
|
-
|
|
43
|
-
fn(handler);
|
|
44
|
-
}
|
|
36
|
+
for (const handler of this._handlers.values()) {
|
|
37
|
+
fn(handler);
|
|
45
38
|
}
|
|
46
39
|
}
|
|
47
40
|
|
|
48
41
|
get size(): number {
|
|
49
|
-
return this.
|
|
42
|
+
return this._handlers.size;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Snapshot view for backward-compat introspection. Slot index = interceptor ID;
|
|
47
|
+
* ejected IDs appear as `null`. Reading this builds a fresh array each time —
|
|
48
|
+
* prefer `forEach`/`size` in hot paths.
|
|
49
|
+
*/
|
|
50
|
+
get handlers(): Array<InterceptorHandler | null> {
|
|
51
|
+
const max = this._nextId;
|
|
52
|
+
const out: Array<InterceptorHandler | null> = new Array(max);
|
|
53
|
+
for (let i = 0; i < max; i++) {
|
|
54
|
+
out[i] = this._handlers.get(i) ?? null;
|
|
55
|
+
}
|
|
56
|
+
return out;
|
|
50
57
|
}
|
|
51
58
|
}
|
|
52
59
|
|