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.
Files changed (40) hide show
  1. package/README.md +21 -21
  2. package/cjs/accessio.cjs +68 -84
  3. package/cjs/accessio.cjs.map +1 -1
  4. package/cjs/core/accessioError.cjs +49 -3
  5. package/cjs/core/accessioError.cjs.map +1 -1
  6. package/cjs/core/buildURL.cjs +10 -4
  7. package/cjs/core/buildURL.cjs.map +1 -1
  8. package/cjs/core/fetchAdapter.cjs +134 -111
  9. package/cjs/core/fetchAdapter.cjs.map +1 -1
  10. package/cjs/core/request.cjs +97 -24
  11. package/cjs/core/request.cjs.map +1 -1
  12. package/cjs/core/retry.cjs +25 -0
  13. package/cjs/core/retry.cjs.map +1 -1
  14. package/cjs/helpers/debug.cjs +7 -1
  15. package/cjs/helpers/debug.cjs.map +1 -1
  16. package/cjs/helpers/flattenHeaders.cjs +37 -0
  17. package/cjs/helpers/flattenHeaders.cjs.map +1 -1
  18. package/cjs/helpers/rateLimiter.cjs +11 -22
  19. package/cjs/helpers/rateLimiter.cjs.map +1 -1
  20. package/cjs/helpers/settle.cjs +1 -1
  21. package/cjs/helpers/settle.cjs.map +1 -1
  22. package/cjs/helpers/transformData.cjs +2 -2
  23. package/cjs/helpers/transformData.cjs.map +1 -1
  24. package/cjs/interceptors/interceptorManager.cjs +25 -18
  25. package/cjs/interceptors/interceptorManager.cjs.map +1 -1
  26. package/index.d.ts +89 -21
  27. package/package.json +2 -2
  28. package/src/accessio.ts +104 -98
  29. package/src/core/accessioError.ts +50 -1
  30. package/src/core/buildURL.ts +14 -4
  31. package/src/core/fetchAdapter.ts +166 -130
  32. package/src/core/request.ts +115 -28
  33. package/src/core/retry.ts +19 -1
  34. package/src/helpers/debug.ts +7 -2
  35. package/src/helpers/flattenHeaders.ts +30 -0
  36. package/src/helpers/rateLimiter.ts +11 -24
  37. package/src/helpers/settle.ts +1 -1
  38. package/src/helpers/transformData.ts +2 -1
  39. package/src/interceptors/interceptorManager.ts +26 -19
  40. 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
- let head = 0;
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 (pendingCount >= maxQueueSize) {
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[tail++] = { resolve, reject };
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
- active--;
57
-
58
- if (pendingCount > 0 && active < maxConcurrent) {
59
- active++;
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 (pendingCount > 0) {
72
- const next = queue[head];
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 pendingCount;
72
+ return queue.length;
86
73
  },
87
74
  get active() {
88
75
  return active;
@@ -9,7 +9,7 @@ export default function settle(
9
9
  ): void {
10
10
  const validateStatus = config.validateStatus;
11
11
 
12
- if (!response.status || !validateStatus || validateStatus(response.status)) {
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
- handlers: Array<InterceptorHandler | null>;
5
- private _activeCount: number;
4
+ private _handlers: Map<number, InterceptorHandler>;
5
+ private _nextId: number;
6
6
 
7
7
  constructor() {
8
- this.handlers = [];
9
- this._activeCount = 0;
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.handlers.push({
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
- if (this.handlers[id]) {
30
- this.handlers[id] = null;
31
- this._activeCount--;
32
- }
28
+ this._handlers.delete(id);
33
29
  }
34
30
 
35
31
  clear(): void {
36
- this.handlers = [];
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.handlers) {
42
- if (handler !== null) {
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._activeCount;
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
 
package/src/types.ts CHANGED
@@ -85,6 +85,7 @@ export interface AccessioRequestConfig {
85
85
  rateLimiter?: RateLimiter;
86
86
  _builtUrl?: string;
87
87
  maxContentLength?: number;
88
+ allowedProtocols?: string[] | null;
88
89
  dispatcher?: unknown;
89
90
  agent?: unknown;
90
91
  dedupe?: boolean;