@zayne-labs/callapi 1.11.27 → 1.11.28

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.
@@ -91,13 +91,6 @@ const toBlobOrString = (value) => {
91
91
  * user: { name: "John", age: 30 },
92
92
  * settings: { theme: "dark" }
93
93
  * });
94
- *
95
- * // Type-preserving usage with Zod
96
- * const schema = z.object({ name: z.string(), file: z.instanceof(Blob) });
97
- * const data = schema.parse({ name: "John", file: blob });
98
- * const typedFormData = toFormData(data, { returnType: "inputType" });
99
- * // Type is { name: string; file: Blob }, runtime is FormData
100
- * ```
101
94
  */
102
95
  const toFormData = (data) => {
103
96
  const formData = new FormData();
@@ -202,4 +195,4 @@ const isJavascriptError = (error) => {
202
195
 
203
196
  //#endregion
204
197
  export { isReadableStream as _, isValidationErrorInstance as a, isValidJsonString as b, toFormData as c, isBoolean as d, isFunction as f, isQueryString as g, isPromise as h, isValidationError as i, toQueryString as l, isPlainObject as m, isHTTPErrorInstance as n, HTTPError as o, isObject as p, isJavascriptError as r, ValidationError as s, isHTTPError as t, isArray as u, isSerializable as v, isString as y };
205
- //# sourceMappingURL=guards-5Ij_loC1.js.map
198
+ //# sourceMappingURL=guards-DvOwVtmd.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"guards-DvOwVtmd.js","names":["toQueryString: ToQueryStringFn"],"sources":["../src/utils/guards.ts","../src/utils/external/body.ts","../src/utils/external/error.ts","../src/utils/external/guards.ts"],"sourcesContent":["import type { AnyFunction } from \"../types/type-helpers\";\n\nexport const isArray = <TArrayItem>(value: unknown): value is TArrayItem[] => Array.isArray(value);\n\nexport const isBoolean = (value: unknown): value is boolean => typeof value === \"boolean\";\n\nexport const isBlob = (value: unknown): value is Blob => value instanceof Blob;\n\nexport const isObject = <TObject extends object>(value: unknown): value is TObject => {\n\treturn typeof value === \"object\" && value !== null;\n};\n\nconst hasObjectPrototype = (value: unknown) => {\n\treturn Object.prototype.toString.call(value) === \"[object Object]\";\n};\n\n/**\n * @description Copied from TanStack Query's isPlainObject\n * @see https://github.com/TanStack/query/blob/main/packages/query-core/src/utils.ts#L321\n */\nexport const isPlainObject = <TPlainObject extends Record<string, unknown>>(\n\tvalue: unknown\n): value is TPlainObject => {\n\tif (!hasObjectPrototype(value)) {\n\t\treturn false;\n\t}\n\n\t// If has no constructor\n\tconst constructor = (value as object | undefined)?.constructor;\n\tif (constructor === undefined) {\n\t\treturn true;\n\t}\n\n\t// If has modified prototype\n\tconst prototype = constructor.prototype as object;\n\tif (!hasObjectPrototype(prototype)) {\n\t\treturn false;\n\t}\n\n\t// If constructor does not have an Object-specific method\n\tif (!Object.hasOwn(prototype, \"isPrototypeOf\")) {\n\t\treturn false;\n\t}\n\n\t// Handles Objects created by Object.create(<arbitrary prototype>)\n\tif (Object.getPrototypeOf(value) !== Object.prototype) {\n\t\treturn false;\n\t}\n\n\t// It's probably a plain object at this point\n\treturn true;\n};\n\nexport const isValidJsonString = (value: unknown): value is string => {\n\tif (!isString(value)) {\n\t\treturn false;\n\t}\n\n\ttry {\n\t\tJSON.parse(value);\n\t\treturn true;\n\t} catch {\n\t\treturn false;\n\t}\n};\n\nexport const isSerializable = (value: unknown) => {\n\treturn (\n\t\tisPlainObject(value)\n\t\t|| isArray(value)\n\t\t|| typeof (value as { toJSON: unknown } | undefined)?.toJSON === \"function\"\n\t);\n};\n\nexport const isFunction = <TFunction extends AnyFunction>(value: unknown): value is TFunction =>\n\ttypeof value === \"function\";\n\nexport const isQueryString = (value: unknown): value is string => isString(value) && value.includes(\"=\");\n\nexport const isString = (value: unknown) => typeof value === \"string\";\n\nexport const isPromise = (value: unknown) => value instanceof Promise;\n\nexport const isReadableStream = (value: unknown): value is ReadableStream<unknown> => {\n\treturn value instanceof ReadableStream;\n};\n\n// https://github.com/unjs/ofetch/blob/main/src/utils.ts\nexport const isJSONSerializable = (value: unknown) => {\n\tif (value === undefined) {\n\t\treturn false;\n\t}\n\tconst t = typeof value;\n\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- No time to make this more type-safe\n\tif (t === \"string\" || t === \"number\" || t === \"boolean\" || t === null) {\n\t\treturn true;\n\t}\n\tif (t !== \"object\") {\n\t\treturn false;\n\t}\n\tif (isArray(value)) {\n\t\treturn true;\n\t}\n\tif ((value as Buffer | null)?.buffer) {\n\t\treturn false;\n\t}\n\n\treturn (\n\t\tvalue?.constructor.name === \"Object\"\n\t\t|| typeof (value as { toJSON: () => unknown } | null)?.toJSON === \"function\"\n\t);\n};\n","import type { CallApiExtraOptions } from \"../../types/common\";\nimport { isArray, isBlob, isObject } from \"../guards\";\n\ntype ToQueryStringFn = {\n\t(query: CallApiExtraOptions[\"query\"]): string | null;\n\t(query: Required<CallApiExtraOptions>[\"query\"]): string;\n};\n\nexport const toQueryString: ToQueryStringFn = (query) => {\n\tif (!query) {\n\t\tconsole.error(\"toQueryString:\", \"No query params provided!\");\n\n\t\treturn null as never;\n\t}\n\n\treturn new URLSearchParams(query as Record<string, string>).toString();\n};\n\ntype AllowedPrimitives = boolean | number | string | Blob | null | undefined;\n\ntype AllowedValues = AllowedPrimitives | AllowedPrimitives[] | Record<string, AllowedPrimitives>;\n\nconst toBlobOrString = (value: AllowedPrimitives): string | Blob => {\n\treturn isBlob(value) ? value : String(value);\n};\n\n/**\n * @description Converts a plain object to FormData.\n *\n * Handles various data types:\n * - **Primitives** (string, number, boolean): Converted to strings\n * - **Blobs/Files**: Added directly to FormData\n * - **Arrays**: Each item is appended (allows multiple values for same key)\n * - **Objects**: JSON stringified before adding to FormData\n *\n * @example\n * ```ts\n * // Basic usage\n * const formData = toFormData({\n * name: \"John\",\n * age: 30,\n * active: true\n * });\n *\n * // With arrays\n * const formData = toFormData({\n * tags: [\"javascript\", \"typescript\"],\n * name: \"John\"\n * });\n *\n * // With files\n * const formData = toFormData({\n * avatar: fileBlob,\n * name: \"John\"\n * });\n *\n * // With nested objects (one level only)\n * const formData = toFormData({\n * user: { name: \"John\", age: 30 },\n * settings: { theme: \"dark\" }\n * });\n */\nexport const toFormData = (data: Record<string, AllowedValues>) => {\n\tconst formData = new FormData();\n\n\tfor (const [key, value] of Object.entries(data)) {\n\t\tif (isArray(value)) {\n\t\t\tvalue.forEach((innerValue) => formData.append(key, toBlobOrString(innerValue)));\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (isObject(value) && !isBlob(value)) {\n\t\t\tformData.set(key, JSON.stringify(value));\n\t\t\tcontinue;\n\t\t}\n\n\t\tformData.set(key, toBlobOrString(value));\n\t}\n\n\treturn formData;\n};\n","import { extraOptionDefaults } from \"../../constants/defaults\";\nimport type { CallApiExtraOptions } from \"../../types/common\";\nimport type { StandardSchemaV1 } from \"../../types/standard-schema\";\nimport type { CallApiSchema, CallApiSchemaConfig } from \"../../validation\";\nimport { isObject, isString } from \"../guards\";\n\ntype HTTPErrorDetails<TErrorData> = Pick<CallApiExtraOptions, \"defaultHTTPErrorMessage\"> & {\n\terrorData: TErrorData;\n\tresponse: Response;\n};\n\nconst httpErrorSymbol = Symbol(\"HTTPError\");\n\nexport class HTTPError<TErrorData = Record<string, unknown>> extends Error {\n\terrorData: HTTPErrorDetails<TErrorData>[\"errorData\"];\n\n\treadonly httpErrorSymbol = httpErrorSymbol;\n\n\toverride name = \"HTTPError\" as const;\n\n\tresponse: HTTPErrorDetails<TErrorData>[\"response\"];\n\n\tconstructor(errorDetails: HTTPErrorDetails<TErrorData>, errorOptions?: ErrorOptions) {\n\t\tconst { defaultHTTPErrorMessage, errorData, response } = errorDetails;\n\n\t\tconst resolvedDefaultHTTPErrorMessage =\n\t\t\tisString(defaultHTTPErrorMessage) ? defaultHTTPErrorMessage : (\n\t\t\t\tdefaultHTTPErrorMessage?.({ errorData, response })\n\t\t\t);\n\n\t\tconst selectedDefaultErrorMessage =\n\t\t\tresolvedDefaultHTTPErrorMessage\n\t\t\t?? (response.statusText || extraOptionDefaults.defaultHTTPErrorMessage);\n\n\t\tconst message =\n\t\t\t(errorData as { message?: string } | undefined)?.message ?? selectedDefaultErrorMessage;\n\n\t\tsuper(message, errorOptions);\n\n\t\tthis.errorData = errorData;\n\t\tthis.response = response;\n\t}\n\n\t/**\n\t * @description Checks if the given error is an instance of HTTPError\n\t * @param error - The error to check\n\t * @returns true if the error is an instance of HTTPError, false otherwise\n\t */\n\tstatic override isError<TErrorData>(error: unknown): error is HTTPError<TErrorData> {\n\t\tif (!isObject<HTTPError>(error)) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (error instanceof HTTPError) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst actualError = error as HTTPError;\n\n\t\treturn (\n\t\t\tactualError.httpErrorSymbol === httpErrorSymbol\n\t\t\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- Allow\n\t\t\t&& actualError.name === \"HTTPError\"\n\t\t);\n\t}\n}\n\nconst prettifyPath = (path: ValidationError[\"errorData\"][number][\"path\"]) => {\n\tif (!path || path.length === 0) {\n\t\treturn \"\";\n\t}\n\n\tconst pathString = path.map((segment) => (isObject(segment) ? segment.key : segment)).join(\".\");\n\n\treturn ` → at ${pathString}`;\n};\n\nconst prettifyValidationIssues = (issues: ValidationError[\"errorData\"]) => {\n\tconst issuesString = issues.map((issue) => `✖ ${issue.message}${prettifyPath(issue.path)}`).join(\" | \");\n\n\treturn issuesString;\n};\n\ntype SafeExtract<TUnion, TKey extends TUnion> = Extract<TUnion, TKey>;\n\ntype ValidationErrorDetails = {\n\t/**\n\t * The cause of the validation error.\n\t *\n\t * It's either the name the schema for which validation failed, or the name of the schema config option that led to the validation error.\n\t */\n\tissueCause:\n\t\t| \"unknown\"\n\t\t| `schemaConfig-(${SafeExtract<keyof CallApiSchemaConfig, \"strict\">})`\n\t\t| keyof CallApiSchema;\n\n\t/**\n\t * The issues that caused the validation error.\n\t */\n\tissues: readonly StandardSchemaV1.Issue[];\n\n\t/**\n\t * The response from server, if any.\n\t */\n\tresponse: Response | null;\n};\n\nconst validationErrorSymbol = Symbol(\"ValidationErrorSymbol\");\n\nexport class ValidationError extends Error {\n\terrorData: ValidationErrorDetails[\"issues\"];\n\n\tissueCause: ValidationErrorDetails[\"issueCause\"];\n\n\toverride name = \"ValidationError\" as const;\n\n\tresponse: ValidationErrorDetails[\"response\"];\n\n\treadonly validationErrorSymbol = validationErrorSymbol;\n\n\tconstructor(details: ValidationErrorDetails, errorOptions?: ErrorOptions) {\n\t\tconst { issueCause, issues, response } = details;\n\n\t\tconst prettyMessage = prettifyValidationIssues(issues);\n\n\t\tconst message = `(${issueCause.toUpperCase()}) - ${prettyMessage}`;\n\n\t\tsuper(message, errorOptions);\n\n\t\tthis.errorData = issues;\n\t\tthis.response = response;\n\t\tthis.issueCause = issueCause;\n\t}\n\n\t/**\n\t * @description Checks if the given error is an instance of ValidationError\n\t * @param error - The error to check\n\t * @returns true if the error is an instance of ValidationError, false otherwise\n\t */\n\tstatic override isError(error: unknown): error is ValidationError {\n\t\tif (!isObject<ValidationError>(error)) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (error instanceof ValidationError) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst actualError = error as ValidationError;\n\n\t\treturn (\n\t\t\tactualError.validationErrorSymbol === validationErrorSymbol\n\t\t\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- Allow\n\t\t\t&& actualError.name === \"ValidationError\"\n\t\t);\n\t}\n}\n","import type {\n\tCallApiResultErrorVariant,\n\tPossibleHTTPError,\n\tPossibleJavaScriptError,\n\tPossibleValidationError,\n} from \"../../result\";\nimport { isObject } from \"../guards\";\nimport { HTTPError, ValidationError } from \"./error\";\n\nexport const isHTTPError = <TErrorData>(\n\terror: CallApiResultErrorVariant<TErrorData>[\"error\"] | null\n): error is PossibleHTTPError<TErrorData> => {\n\treturn isObject(error) && error.name === \"HTTPError\";\n};\n\nexport const isHTTPErrorInstance = <TErrorData>(error: unknown) => {\n\treturn HTTPError.isError<TErrorData>(error);\n};\n\nexport const isValidationError = (\n\terror: CallApiResultErrorVariant<unknown>[\"error\"] | null\n): error is PossibleValidationError => {\n\treturn isObject(error) && error.name === \"ValidationError\";\n};\n\nexport const isValidationErrorInstance = (error: unknown): error is ValidationError => {\n\treturn ValidationError.isError(error);\n};\n\nexport const isJavascriptError = (\n\terror: CallApiResultErrorVariant<unknown>[\"error\"] | null\n): error is PossibleJavaScriptError => {\n\treturn isObject(error) && !isHTTPError(error) && !isValidationError(error);\n};\n"],"mappings":"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"}
@@ -283,7 +283,7 @@ type DedupeOptions = {
283
283
  //#endregion
284
284
  //#region src/middlewares.d.ts
285
285
  type FetchImpl = UnmaskType<(input: string | Request | URL, init?: RequestInit) => Promise<Response>>;
286
- interface Middlewares {
286
+ interface Middlewares<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> {
287
287
  /**
288
288
  * Wraps the fetch implementation to intercept requests at the network layer.
289
289
  *
@@ -315,7 +315,7 @@ interface Middlewares {
315
315
  * }
316
316
  * ```
317
317
  */
318
- fetchMiddleware?: (context: RequestContext & {
318
+ fetchMiddleware?: (context: RequestContext<TCallApiContext$1> & {
319
319
  fetchImpl: FetchImpl;
320
320
  }) => FetchImpl;
321
321
  }
@@ -845,6 +845,7 @@ type CallApiContext = {
845
845
  Meta?: DefaultMetaObject;
846
846
  ResultMode?: ResultModeType;
847
847
  };
848
+ type GetMergedCallApiContext<TFullCallApiContext extends CallApiContext, TOverrideCallApiContext extends CallApiContext> = Omit<TFullCallApiContext, keyof TOverrideCallApiContext> & TOverrideCallApiContext;
848
849
  type FetchSpecificKeysUnion = Exclude<(typeof fetchSpecificKeys)[number], "body" | "headers" | "method">;
849
850
  type ModifiedRequestInit = RequestInit & {
850
851
  duplex?: "half";
@@ -867,9 +868,9 @@ type CallApiRequestOptions = Prettify<{
867
868
  type CallApiRequestOptionsForHooks = Omit<CallApiRequestOptions, "headers"> & {
868
869
  headers: Record<string, string | undefined>;
869
870
  };
870
- type SharedExtraOptions<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext, TData$1 = DefaultDataType, TErrorData$1 = DefaultDataType, TResultMode$1 extends ResultModeType = ResultModeType, TThrowOnError$1 extends ThrowOnErrorUnion = DefaultThrowOnError, TResponseType$1 extends ResponseTypeType = ResponseTypeType, TPluginArray$1 extends CallApiPlugin[] = DefaultPluginArray, TComputedPluginOptions = InferPluginOptions<TPluginArray$1> & TCallApiContext$1["InferredPluginOptions"], TComputedInferredPluginOptions extends Pick<Required<CallApiContext>, "InferredPluginOptions"> = {
871
- InferredPluginOptions: TComputedPluginOptions;
872
- }> = DedupeOptions & HookConfigOptions & HooksOrHooksArray<TComputedInferredPluginOptions, TData$1, TErrorData$1> & Middlewares & Partial<TComputedInferredPluginOptions["InferredPluginOptions"]> & ResultModeOption<TErrorData$1, TResultMode$1> & RetryOptions<TErrorData$1> & ThrowOnErrorOption<TErrorData$1, TThrowOnError$1> & URLOptions & {
871
+ type SharedExtraOptions<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext, TData$1 = DefaultDataType, TErrorData$1 = DefaultDataType, TResultMode$1 extends ResultModeType = ResultModeType, TThrowOnError$1 extends ThrowOnErrorUnion = DefaultThrowOnError, TResponseType$1 extends ResponseTypeType = ResponseTypeType, TPluginArray$1 extends CallApiPlugin[] = DefaultPluginArray, TComputedMergedPluginOptions = Partial<InferPluginOptions<TPluginArray$1> & TCallApiContext$1["InferredPluginOptions"]>> = DedupeOptions & HookConfigOptions & HooksOrHooksArray<GetMergedCallApiContext<TCallApiContext$1, {
872
+ InferredPluginOptions: TComputedMergedPluginOptions;
873
+ }>> & Middlewares & ResultModeOption<TErrorData$1, TResultMode$1> & RetryOptions<TErrorData$1> & TComputedMergedPluginOptions & ThrowOnErrorOption<TErrorData$1, TThrowOnError$1> & URLOptions & {
873
874
  /**
874
875
  * Automatically add an Authorization header value.
875
876
  *
@@ -1403,12 +1404,6 @@ type ToQueryStringFn = {
1403
1404
  declare const toQueryString: ToQueryStringFn;
1404
1405
  type AllowedPrimitives = boolean | number | string | Blob | null | undefined;
1405
1406
  type AllowedValues = AllowedPrimitives | AllowedPrimitives[] | Record<string, AllowedPrimitives>;
1406
- type ToFormDataFn = {
1407
- (data: Record<string, AllowedValues>): FormData;
1408
- <TData extends Record<string, AllowedValues>>(data: TData, options: {
1409
- returnType: "inputType";
1410
- }): TData;
1411
- };
1412
1407
  /**
1413
1408
  * @description Converts a plain object to FormData.
1414
1409
  *
@@ -1444,15 +1439,8 @@ type ToFormDataFn = {
1444
1439
  * user: { name: "John", age: 30 },
1445
1440
  * settings: { theme: "dark" }
1446
1441
  * });
1447
- *
1448
- * // Type-preserving usage with Zod
1449
- * const schema = z.object({ name: z.string(), file: z.instanceof(Blob) });
1450
- * const data = schema.parse({ name: "John", file: blob });
1451
- * const typedFormData = toFormData(data, { returnType: "inputType" });
1452
- * // Type is { name: string; file: Blob }, runtime is FormData
1453
- * ```
1454
1442
  */
1455
- declare const toFormData: ToFormDataFn;
1443
+ declare const toFormData: (data: Record<string, AllowedValues>) => FormData;
1456
1444
  //#endregion
1457
1445
  //#region src/utils/external/define.d.ts
1458
1446
  declare const defineSchema: <const TBaseSchemaRoutes extends BaseCallApiSchemaRoutes, const TSchemaConfig extends CallApiSchemaConfig>(routes: TBaseSchemaRoutes, config?: Satisfies<TSchemaConfig, CallApiSchemaConfig>) => {
@@ -1504,7 +1492,7 @@ declare global {
1504
1492
  }
1505
1493
  //#endregion
1506
1494
  //#region src/hooks.d.ts
1507
- interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TData$1 = TCallApiContext$1["Data"], TErrorData$1 = TCallApiContext$1["ErrorData"]> {
1495
+ interface Hooks<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> {
1508
1496
  /**
1509
1497
  * Hook called when any error occurs within the request/response lifecycle.
1510
1498
  *
@@ -1515,7 +1503,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1515
1503
  * @param context - Error context containing error details, request info, and response (if available)
1516
1504
  * @returns Promise or void - Hook can be async or sync
1517
1505
  */
1518
- onError?: (context: ErrorContext<TCallApiContext$1, TErrorData$1>) => Awaitable<unknown>;
1506
+ onError?: (context: ErrorContext<TCallApiContext$1>) => Awaitable<unknown>;
1519
1507
  /**
1520
1508
  * Hook called before the HTTP request is sent and before any internal processing of the request object begins.
1521
1509
  *
@@ -1567,7 +1555,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1567
1555
  * @returns Promise or void - Hook can be async or sync
1568
1556
  *
1569
1557
  */
1570
- onResponse?: (context: ResponseContext<TCallApiContext$1, TData$1, TErrorData$1>) => Awaitable<unknown>;
1558
+ onResponse?: (context: ResponseContext<TCallApiContext$1>) => Awaitable<unknown>;
1571
1559
  /**
1572
1560
  * Hook called when an HTTP error response (4xx, 5xx) is received from the API.
1573
1561
  *
@@ -1578,7 +1566,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1578
1566
  * @param context - Response error context with HTTP error details and response
1579
1567
  * @returns Promise or void - Hook can be async or sync
1580
1568
  */
1581
- onResponseError?: (context: ResponseErrorContext<TCallApiContext$1, TErrorData$1>) => Awaitable<unknown>;
1569
+ onResponseError?: (context: ResponseErrorContext<TCallApiContext$1>) => Awaitable<unknown>;
1582
1570
  /**
1583
1571
  * Hook called during download stream progress tracking.
1584
1572
  *
@@ -1602,7 +1590,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1602
1590
  * @returns Promise or void - Hook can be async or sync
1603
1591
  *
1604
1592
  */
1605
- onRetry?: (response: RetryContext<TCallApiContext$1, TErrorData$1>) => Awaitable<unknown>;
1593
+ onRetry?: (response: RetryContext<TCallApiContext$1>) => Awaitable<unknown>;
1606
1594
  /**
1607
1595
  * Hook called when a successful response (2xx status) is received from the API.
1608
1596
  *
@@ -1614,7 +1602,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1614
1602
  * @returns Promise or void - Hook can be async or sync
1615
1603
  *
1616
1604
  */
1617
- onSuccess?: (context: SuccessContext<TCallApiContext$1, TData$1>) => Awaitable<unknown>;
1605
+ onSuccess?: (context: SuccessContext<TCallApiContext$1>) => Awaitable<unknown>;
1618
1606
  /**
1619
1607
  * Hook called when a validation error occurs.
1620
1608
  *
@@ -1628,7 +1616,7 @@ interface Hooks<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorDa
1628
1616
  */
1629
1617
  onValidationError?: (context: ValidationErrorContext<TCallApiContext$1>) => Awaitable<unknown>;
1630
1618
  }
1631
- type HooksOrHooksArray<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext, TData$1 = TCallApiContext$1["Data"], TErrorData$1 = TCallApiContext$1["ErrorData"]> = { [Key in keyof Hooks<TCallApiContext$1, TData$1, TErrorData$1>]: Hooks<TCallApiContext$1, TData$1, TErrorData$1>[Key] | Array<Hooks<TCallApiContext$1, TData$1, TErrorData$1>[Key]> };
1619
+ type HooksOrHooksArray<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> = { [Key in keyof Hooks<TCallApiContext$1>]: Hooks<TCallApiContext$1>[Key] | Array<Hooks<TCallApiContext$1>[Key]> };
1632
1620
  interface HookConfigOptions {
1633
1621
  /**
1634
1622
  * Controls the execution mode of all composed hooks (main + plugin hooks).
@@ -1678,28 +1666,28 @@ type ValidationErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "Infe
1678
1666
  error: PossibleValidationError;
1679
1667
  response: Response | null;
1680
1668
  }>;
1681
- type SuccessContext<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TData$1 = TCallApiContext$1["Data"]> = UnmaskType<RequestContext<TCallApiContext$1> & {
1682
- data: NoInfer<TData$1>;
1669
+ type SuccessContext<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<RequestContext<TCallApiContext$1> & {
1670
+ data: NoInfer<TCallApiContext$1["Data"]>;
1683
1671
  response: Response;
1684
1672
  }>;
1685
- type ResponseContext<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TData$1 = TCallApiContext$1["Data"], TErrorData$1 = TCallApiContext$1["ErrorData"]> = UnmaskType<RequestContext<TCallApiContext$1> & (Prettify<CallApiResultSuccessVariant<TData$1>> | Prettify<Extract<CallApiResultErrorVariant<TErrorData$1>, {
1686
- error: PossibleHTTPError<TErrorData$1>;
1673
+ type ResponseContext<TCallApiContext$1 extends Pick<CallApiContext, "Data" | "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<RequestContext<TCallApiContext$1> & (Prettify<CallApiResultSuccessVariant<TCallApiContext$1["Data"]>> | Prettify<Extract<CallApiResultErrorVariant<TCallApiContext$1["ErrorData"]>, {
1674
+ error: PossibleHTTPError<TCallApiContext$1["ErrorData"]>;
1687
1675
  }>>)>;
1688
1676
  type RequestErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = RequestContext<TCallApiContext$1> & {
1689
1677
  error: PossibleJavaScriptError;
1690
1678
  response: null;
1691
1679
  };
1692
- type ErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TErrorData$1 = TCallApiContext$1["ErrorData"]> = UnmaskType<RequestContext<TCallApiContext$1> & ({
1693
- error: PossibleHTTPError<TErrorData$1>;
1680
+ type ErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<RequestContext<TCallApiContext$1> & ({
1681
+ error: PossibleHTTPError<TCallApiContext$1["ErrorData"]>;
1694
1682
  response: Response;
1695
1683
  } | {
1696
1684
  error: PossibleJavaScriptOrValidationError;
1697
1685
  response: Response | null;
1698
1686
  })>;
1699
- type ResponseErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TErrorData$1 = TCallApiContext$1["ErrorData"]> = UnmaskType<Extract<ErrorContext<TCallApiContext$1, TErrorData$1>, {
1700
- error: PossibleHTTPError<TErrorData$1>;
1687
+ type ResponseErrorContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<Extract<ErrorContext<TCallApiContext$1>, {
1688
+ error: PossibleHTTPError<TCallApiContext$1["ErrorData"]>;
1701
1689
  }> & RequestContext<TCallApiContext$1>>;
1702
- type RetryContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext, TErrorData$1 = TCallApiContext$1["ErrorData"]> = UnmaskType<ErrorContext<TCallApiContext$1, TErrorData$1> & {
1690
+ type RetryContext<TCallApiContext$1 extends Pick<CallApiContext, "ErrorData" | "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<ErrorContext<TCallApiContext$1> & {
1703
1691
  retryAttemptCount: number;
1704
1692
  }>;
1705
1693
  type RequestStreamContext<TCallApiContext$1 extends Pick<CallApiContext, "InferredPluginOptions" | "Meta"> = DefaultCallApiContext> = UnmaskType<RequestContext<TCallApiContext$1> & {
@@ -1715,12 +1703,15 @@ type ResponseStreamContext<TCallApiContext$1 extends Pick<CallApiContext, "Infer
1715
1703
  type PluginSetupContext<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> = RequestContext<TCallApiContext$1> & {
1716
1704
  initURL: string;
1717
1705
  };
1718
- type PluginInitResult = Partial<Omit<PluginSetupContext, "initURL" | "request"> & {
1706
+ type PluginInitResult<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> = Partial<Omit<PluginSetupContext<TCallApiContext$1>, "initURL" | "request"> & {
1719
1707
  initURL: InitURLOrURLObject;
1720
1708
  request: CallApiRequestOptions;
1721
1709
  }>;
1722
- type PluginHooks<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext, TData$1 = (unknown extends TCallApiContext$1["Data"] ? never : TCallApiContext$1["Data"]), TErrorData$1 = (unknown extends TCallApiContext$1["ErrorData"] ? never : TCallApiContext$1["ErrorData"])> = HooksOrHooksArray<TCallApiContext$1, TData$1, TErrorData$1>;
1723
- interface CallApiPlugin {
1710
+ type PluginHooks<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> = HooksOrHooksArray<GetMergedCallApiContext<TCallApiContext$1, {
1711
+ Data: DefaultDataType extends TCallApiContext$1["Data"] ? never : TCallApiContext$1["Data"];
1712
+ ErrorData: DefaultDataType extends TCallApiContext$1["ErrorData"] ? never : TCallApiContext$1["ErrorData"];
1713
+ }>>;
1714
+ interface CallApiPlugin<TCallApiContext$1 extends CallApiContext = DefaultCallApiContext> {
1724
1715
  /**
1725
1716
  * Defines additional options that can be passed to callApi
1726
1717
  */
@@ -1732,7 +1723,7 @@ interface CallApiPlugin {
1732
1723
  /**
1733
1724
  * Hooks for the plugin
1734
1725
  */
1735
- hooks?: PluginHooks | ((context: PluginSetupContext) => Awaitable<PluginHooks>);
1726
+ hooks?: PluginHooks<TCallApiContext$1> | ((context: PluginSetupContext<TCallApiContext$1>) => Awaitable<PluginHooks<TCallApiContext$1>>);
1736
1727
  /**
1737
1728
  * A unique id for the plugin
1738
1729
  */
@@ -1740,7 +1731,7 @@ interface CallApiPlugin {
1740
1731
  /**
1741
1732
  * Middlewares that for the plugin
1742
1733
  */
1743
- middlewares?: Middlewares | ((context: PluginSetupContext) => Awaitable<Middlewares>);
1734
+ middlewares?: Middlewares<TCallApiContext$1> | ((context: PluginSetupContext<TCallApiContext$1>) => Awaitable<Middlewares<TCallApiContext$1>>);
1744
1735
  /**
1745
1736
  * A name for the plugin
1746
1737
  */
@@ -1752,7 +1743,7 @@ interface CallApiPlugin {
1752
1743
  /**
1753
1744
  * A function that will be called when the plugin is initialized. This will be called before the any of the other internal functions.
1754
1745
  */
1755
- setup?: (context: PluginSetupContext) => Awaitable<PluginInitResult> | Awaitable<void>;
1746
+ setup?: (context: PluginSetupContext<TCallApiContext$1>) => Awaitable<PluginInitResult<TCallApiContext$1>> | Awaitable<void>;
1756
1747
  /**
1757
1748
  * A version for the plugin
1758
1749
  */
@@ -1854,14 +1845,38 @@ type BaseCallApiSchemaAndConfig = {
1854
1845
  };
1855
1846
  //#endregion
1856
1847
  //#region src/createFetchClient.d.ts
1857
- declare const createFetchClientWithContext: <TOuterCallApiContext extends CallApiContext = DefaultCallApiContext>() => <TBaseCallApiContext extends CallApiContext = TOuterCallApiContext, TBaseData = TBaseCallApiContext["Data"], TBaseErrorData = TBaseCallApiContext["ErrorData"], TBaseResultMode extends ResultModeType = (TBaseCallApiContext["ResultMode"] extends ResultModeType ? TBaseCallApiContext["ResultMode"] : ResultModeType), TBaseThrowOnError extends ThrowOnErrorUnion = boolean, TBaseResponseType extends ResponseTypeType = ResponseTypeType, const TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig, const TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray, TComputedBaseSchemaConfig extends CallApiSchemaConfig = GetBaseSchemaConfig<TBaseSchemaAndConfig>, TComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes = GetBaseSchemaRoutes<TBaseSchemaAndConfig>>(initBaseConfig?: BaseCallApiConfig<TBaseCallApiContext, TBaseData, TBaseErrorData, TBaseResultMode, TBaseThrowOnError, TBaseResponseType, TBaseSchemaAndConfig, TBasePluginArray>) => <TData = TBaseData, TErrorData = TBaseErrorData, TResultMode extends ResultModeType = TBaseResultMode, TCallApiContext extends CallApiContext = TBaseCallApiContext, TThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError, TResponseType extends ResponseTypeType = TBaseResponseType, const TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig, TInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<TComputedBaseSchemaRoutes, TCurrentRouteSchemaKey>, const TPluginArray extends CallApiPlugin[] = TBasePluginArray, TComputedResult = CallApiResult<InferSchemaOutput<TSchema["data"], TData>, InferSchemaOutput<TSchema["errorData"], TErrorData>, TResultMode, TThrowOnError, TResponseType>>(initURL: TInitURL, initConfig?: CallApiConfig<TCallApiContext, InferSchemaOutput<TSchema["data"], GetResponseType<TData, TResponseType>>, InferSchemaOutput<TSchema["errorData"], GetResponseType<TErrorData, TResponseType>>, TResultMode, TThrowOnError, TResponseType, TComputedBaseSchemaRoutes, TSchema, TComputedBaseSchemaConfig, TSchemaConfig, TInitURL, TCurrentRouteSchemaKey, TBasePluginArray, TPluginArray>) => Promise<TComputedResult>;
1858
- declare const createFetchClient: <TBaseCallApiContext extends CallApiContext = DefaultCallApiContext, TBaseData = TBaseCallApiContext["Data"], TBaseErrorData = TBaseCallApiContext["ErrorData"], TBaseResultMode extends ResultModeType = (TBaseCallApiContext["ResultMode"] extends ResultModeType ? TBaseCallApiContext["ResultMode"] : ResultModeType), TBaseThrowOnError extends ThrowOnErrorUnion = boolean, TBaseResponseType extends ResponseTypeType = ResponseTypeType, const TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig, const TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray, TComputedBaseSchemaConfig extends CallApiSchemaConfig = Writeable<NonNullable<TBaseSchemaAndConfig["config"]>, "deep">, TComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes = Writeable<TBaseSchemaAndConfig["routes"], "deep">>(initBaseConfig?: BaseCallApiConfig<TBaseCallApiContext, TBaseData, TBaseErrorData, TBaseResultMode, TBaseThrowOnError, TBaseResponseType, TBaseSchemaAndConfig, TBasePluginArray>) => <TData = TBaseData, TErrorData = TBaseErrorData, TResultMode extends ResultModeType = TBaseResultMode, TCallApiContext extends CallApiContext = TBaseCallApiContext, TThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError, TResponseType extends ResponseTypeType = TBaseResponseType, const TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig, TInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<TComputedBaseSchemaRoutes, TCurrentRouteSchemaKey, TComputedBaseSchemaRoutes["@default"], TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey], NonNullable<Omit<TComputedBaseSchemaRoutes["@default"], keyof TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey]> & TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey]>>, const TPluginArray extends CallApiPlugin[] = TBasePluginArray, TComputedResult = GetCallApiResult<InferSchemaResult<TSchema["data"], TData, "infer-output">, InferSchemaResult<TSchema["errorData"], TErrorData, "infer-output">, TResultMode, TThrowOnError, TResponseType, {
1848
+ declare const createFetchClientWithContext: <TOuterCallApiContext extends CallApiContext = DefaultCallApiContext>() => <TBaseCallApiContext extends CallApiContext = TOuterCallApiContext, TBaseData = TBaseCallApiContext["Data"], TBaseErrorData = TBaseCallApiContext["ErrorData"], TBaseResultMode extends ResultModeType = (TBaseCallApiContext["ResultMode"] extends ResultModeType ? TBaseCallApiContext["ResultMode"] : ResultModeType), TBaseThrowOnError extends ThrowOnErrorUnion = boolean, TBaseResponseType extends ResponseTypeType = ResponseTypeType, const TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig, const TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray, TComputedBaseCallApiContext extends CallApiContext = GetMergedCallApiContext<TBaseCallApiContext, {
1849
+ Data: TBaseData;
1850
+ ErrorData: TBaseErrorData;
1851
+ ResultMode: TBaseResultMode;
1852
+ }>, TComputedBaseSchemaConfig extends CallApiSchemaConfig = GetBaseSchemaConfig<TBaseSchemaAndConfig>, TComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes = GetBaseSchemaRoutes<TBaseSchemaAndConfig>>(initBaseConfig?: BaseCallApiConfig<TBaseCallApiContext, TBaseData, TBaseErrorData, TBaseResultMode, TBaseThrowOnError, TBaseResponseType, TBaseSchemaAndConfig, TBasePluginArray>) => <TData = TBaseData, TErrorData = TBaseErrorData, TResultMode extends ResultModeType = TBaseResultMode, TCallApiContext extends CallApiContext = GetMergedCallApiContext<TComputedBaseCallApiContext, {
1853
+ Data: TData;
1854
+ ErrorData: TErrorData;
1855
+ ResultMode: TResultMode;
1856
+ }>, TThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError, TResponseType extends ResponseTypeType = TBaseResponseType, const TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig, TInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<TComputedBaseSchemaRoutes, TCurrentRouteSchemaKey>, const TPluginArray extends CallApiPlugin[] = TBasePluginArray, TComputedResult = CallApiResult<InferSchemaOutput<TSchema["data"], TData>, InferSchemaOutput<TSchema["errorData"], TErrorData>, TResultMode, TThrowOnError, TResponseType>>(initURL: TInitURL, initConfig?: CallApiConfig<TCallApiContext, InferSchemaOutput<TSchema["data"], GetResponseType<TData, TResponseType>>, InferSchemaOutput<TSchema["errorData"], GetResponseType<TErrorData, TResponseType>>, TResultMode, TThrowOnError, TResponseType, TComputedBaseSchemaRoutes, TSchema, TComputedBaseSchemaConfig, TSchemaConfig, TInitURL, TCurrentRouteSchemaKey, TBasePluginArray, TPluginArray>) => Promise<TComputedResult>;
1857
+ declare const createFetchClient: <TBaseCallApiContext extends CallApiContext = DefaultCallApiContext, TBaseData = TBaseCallApiContext["Data"], TBaseErrorData = TBaseCallApiContext["ErrorData"], TBaseResultMode extends ResultModeType = (TBaseCallApiContext["ResultMode"] extends ResultModeType ? TBaseCallApiContext["ResultMode"] : ResultModeType), TBaseThrowOnError extends ThrowOnErrorUnion = boolean, TBaseResponseType extends ResponseTypeType = ResponseTypeType, const TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig, const TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray, TComputedBaseCallApiContext extends CallApiContext = GetMergedCallApiContext<TBaseCallApiContext, {
1858
+ Data: TBaseData;
1859
+ ErrorData: TBaseErrorData;
1860
+ ResultMode: TBaseResultMode;
1861
+ }>, TComputedBaseSchemaConfig extends CallApiSchemaConfig = Writeable<NonNullable<TBaseSchemaAndConfig["config"]>, "deep">, TComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes = Writeable<TBaseSchemaAndConfig["routes"], "deep">>(initBaseConfig?: BaseCallApiConfig<TBaseCallApiContext, TBaseData, TBaseErrorData, TBaseResultMode, TBaseThrowOnError, TBaseResponseType, TBaseSchemaAndConfig, TBasePluginArray>) => <TData = TBaseData, TErrorData = TBaseErrorData, TResultMode extends ResultModeType = TBaseResultMode, TCallApiContext extends CallApiContext = GetMergedCallApiContext<TComputedBaseCallApiContext, {
1862
+ Data: TData;
1863
+ ErrorData: TErrorData;
1864
+ ResultMode: TResultMode;
1865
+ }>, TThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError, TResponseType extends ResponseTypeType = TBaseResponseType, const TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig, TInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<TComputedBaseSchemaRoutes, TCurrentRouteSchemaKey, TComputedBaseSchemaRoutes["@default"], TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey], NonNullable<Omit<TComputedBaseSchemaRoutes["@default"], keyof TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey]> & TComputedBaseSchemaRoutes[TCurrentRouteSchemaKey]>>, const TPluginArray extends CallApiPlugin[] = TBasePluginArray, TComputedResult = GetCallApiResult<InferSchemaResult<TSchema["data"], TData, "infer-output">, InferSchemaResult<TSchema["errorData"], TErrorData, "infer-output">, TResultMode, TThrowOnError, TResponseType, {
1859
1866
  all: CallApiResultSuccessVariant<GetResponseType<InferSchemaResult<TSchema["data"], TData, "infer-output">, TResponseType, ResponseTypeMap<InferSchemaResult<TSchema["data"], TData, "infer-output">>>>;
1860
1867
  onlyData: NoInfer<GetResponseType<InferSchemaResult<TSchema["data"], TData, "infer-output">, TResponseType, ResponseTypeMap<InferSchemaResult<TSchema["data"], TData, "infer-output">>>>;
1861
1868
  onlyResponse: Response;
1862
1869
  withoutResponse: Omit<CallApiResultSuccessVariant<GetResponseType<InferSchemaResult<TSchema["data"], TData, "infer-output">, TResponseType, ResponseTypeMap<InferSchemaResult<TSchema["data"], TData, "infer-output">>>>, "response">;
1863
1870
  }, ResultModeMap<InferSchemaResult<TSchema["data"], TData, "infer-output">, InferSchemaResult<TSchema["errorData"], TErrorData, "infer-output">, TResponseType, TThrowOnError>>>(initURL: TInitURL, initConfig?: CallApiConfig<TCallApiContext, InferSchemaResult<TSchema["data"], GetResponseType<TData, TResponseType, ResponseTypeMap<TData>>, "infer-output">, InferSchemaResult<TSchema["errorData"], GetResponseType<TErrorData, TResponseType, ResponseTypeMap<TErrorData>>, "infer-output">, TResultMode, TThrowOnError, TResponseType, TComputedBaseSchemaRoutes, TSchema, TComputedBaseSchemaConfig, TSchemaConfig, TInitURL, TCurrentRouteSchemaKey, TBasePluginArray, TPluginArray>) => Promise<TComputedResult>;
1864
- declare const callApi: <TData = unknown, TErrorData = unknown, TResultMode extends ResultModeType = ResultModeType, TCallApiContext extends CallApiContext = DefaultCallApiContext, TThrowOnError extends ThrowOnErrorUnion = boolean, TResponseType extends ResponseTypeType = ResponseTypeType, const TSchemaConfig extends CallApiSchemaConfig = CallApiSchemaConfig, TInitURL extends ApplyStrictConfig<TSchemaConfig, ApplyURLBasedConfig<TSchemaConfig, AnyString | "@delete/" | "@get/" | "@patch/" | "@post/" | "@put/">> = ApplyStrictConfig<TSchemaConfig, ApplyURLBasedConfig<TSchemaConfig, AnyString | "@delete/" | "@get/" | "@patch/" | "@post/" | "@put/">>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<{
1871
+ declare const callApi: <TData = unknown, TErrorData = unknown, TResultMode extends ResultModeType = ResultModeType, TCallApiContext extends CallApiContext = GetMergedCallApiContext<GetMergedCallApiContext<DefaultCallApiContext, {
1872
+ Data: unknown;
1873
+ ErrorData: unknown;
1874
+ ResultMode: ResultModeType;
1875
+ }>, {
1876
+ Data: TData;
1877
+ ErrorData: TErrorData;
1878
+ ResultMode: TResultMode;
1879
+ }>, TThrowOnError extends ThrowOnErrorUnion = boolean, TResponseType extends ResponseTypeType = ResponseTypeType, const TSchemaConfig extends CallApiSchemaConfig = CallApiSchemaConfig, TInitURL extends ApplyStrictConfig<TSchemaConfig, ApplyURLBasedConfig<TSchemaConfig, AnyString | "@delete/" | "@get/" | "@patch/" | "@post/" | "@put/">> = ApplyStrictConfig<TSchemaConfig, ApplyURLBasedConfig<TSchemaConfig, AnyString | "@delete/" | "@get/" | "@patch/" | "@post/" | "@put/">>, TCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> = GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>, const TSchema extends CallApiSchema = GetCurrentRouteSchema<{
1865
1880
  [x: AnyString]: CallApiSchema | undefined;
1866
1881
  "@default"?: CallApiSchema | undefined;
1867
1882
  "@delete/"?: CallApiSchema | undefined;
@@ -1909,4 +1924,4 @@ declare const callApi: <TData = unknown, TErrorData = unknown, TResultMode exten
1909
1924
  }, TSchema, CallApiSchemaConfig, TSchemaConfig, TInitURL, TCurrentRouteSchemaKey, DefaultPluginArray, TPluginArray>) => Promise<TComputedResult>;
1910
1925
  //#endregion
1911
1926
  export { PossibleHTTPError as $, definePlugin as A, CallApiExtraOptionsForHooks as B, isHTTPError as C, isValidationErrorInstance as D, isValidationError as E, toQueryString as F, GetExtendSchemaConfigContext as G, CallApiRequestOptions as H, BaseCallApiConfig as I, Register as J, InferExtendSchemaContext as K, BaseCallApiExtraOptions as L, defineSchemaConfig as M, defineSchemaRoutes as N, defineBaseConfig as O, toFormData as P, CallApiSuccessOrErrorVariant as Q, CallApiConfig as R, SuccessContext as S, isJavascriptError as T, CallApiRequestOptionsForHooks as U, CallApiParameters as V, CallApiResultLoose as W, CallApiResultErrorVariant as X, RetryOptions as Y, CallApiResultSuccessVariant as Z, RequestContext as _, BaseSchemaRouteKeyPrefixes as a, HTTPError as at, ResponseErrorContext as b, InferSchemaInput as c, URLOptions as ct, CallApiPlugin as d, PossibleJavaScriptError as et, PluginHooks as f, HooksOrHooksArray as g, Hooks as h, BaseCallApiSchemaRoutes as i, ResultModeType as it, defineSchema as j, defineMainSchema as k, InferSchemaOutput as l, DedupeOptions as lt, ErrorContext as m, createFetchClient as n, PossibleValidationError as nt, CallApiSchema as o, ValidationError as ot, PluginSetupContext as p, InstanceContext as q, createFetchClientWithContext as r, ResponseTypeType as rt, CallApiSchemaConfig as s, InferParamsFromRoute as st, callApi as t, PossibleJavaScriptOrValidationError as tt, DefaultCallApiContext as u, RequestStreamContext as v, isHTTPErrorInstance as w, ResponseStreamContext as x, ResponseContext as y, CallApiExtraOptions as z };
1912
- //# sourceMappingURL=index-HHFgKKcn.d.ts.map
1927
+ //# sourceMappingURL=index-j55-O6zR.d.ts.map
package/dist/index.d.ts CHANGED
@@ -1,3 +1,3 @@
1
1
  import "./validation-Dq--Q5zC.js";
2
- import { $ as PossibleHTTPError, B as CallApiExtraOptionsForHooks, G as GetExtendSchemaConfigContext, H as CallApiRequestOptions, I as BaseCallApiConfig, J as Register, K as InferExtendSchemaContext, L as BaseCallApiExtraOptions, Q as CallApiSuccessOrErrorVariant, R as CallApiConfig, S as SuccessContext, U as CallApiRequestOptionsForHooks, V as CallApiParameters, W as CallApiResultLoose, X as CallApiResultErrorVariant, Y as RetryOptions, Z as CallApiResultSuccessVariant, _ as RequestContext, a as BaseSchemaRouteKeyPrefixes, b as ResponseErrorContext, c as InferSchemaInput, ct as URLOptions, d as CallApiPlugin, et as PossibleJavaScriptError, f as PluginHooks, g as HooksOrHooksArray, h as Hooks, i as BaseCallApiSchemaRoutes, it as ResultModeType, l as InferSchemaOutput, lt as DedupeOptions, m as ErrorContext, n as createFetchClient, nt as PossibleValidationError, o as CallApiSchema, p as PluginSetupContext, q as InstanceContext, r as createFetchClientWithContext, rt as ResponseTypeType, s as CallApiSchemaConfig, st as InferParamsFromRoute, t as callApi, tt as PossibleJavaScriptOrValidationError, u as DefaultCallApiContext, v as RequestStreamContext, x as ResponseStreamContext, y as ResponseContext, z as CallApiExtraOptions } from "./index-HHFgKKcn.js";
2
+ import { $ as PossibleHTTPError, B as CallApiExtraOptionsForHooks, G as GetExtendSchemaConfigContext, H as CallApiRequestOptions, I as BaseCallApiConfig, J as Register, K as InferExtendSchemaContext, L as BaseCallApiExtraOptions, Q as CallApiSuccessOrErrorVariant, R as CallApiConfig, S as SuccessContext, U as CallApiRequestOptionsForHooks, V as CallApiParameters, W as CallApiResultLoose, X as CallApiResultErrorVariant, Y as RetryOptions, Z as CallApiResultSuccessVariant, _ as RequestContext, a as BaseSchemaRouteKeyPrefixes, b as ResponseErrorContext, c as InferSchemaInput, ct as URLOptions, d as CallApiPlugin, et as PossibleJavaScriptError, f as PluginHooks, g as HooksOrHooksArray, h as Hooks, i as BaseCallApiSchemaRoutes, it as ResultModeType, l as InferSchemaOutput, lt as DedupeOptions, m as ErrorContext, n as createFetchClient, nt as PossibleValidationError, o as CallApiSchema, p as PluginSetupContext, q as InstanceContext, r as createFetchClientWithContext, rt as ResponseTypeType, s as CallApiSchemaConfig, st as InferParamsFromRoute, t as callApi, tt as PossibleJavaScriptOrValidationError, u as DefaultCallApiContext, v as RequestStreamContext, x as ResponseStreamContext, y as ResponseContext, z as CallApiExtraOptions } from "./index-j55-O6zR.js";
3
3
  export { BaseCallApiConfig, BaseCallApiExtraOptions, BaseCallApiSchemaRoutes, BaseSchemaRouteKeyPrefixes, CallApiConfig, CallApiExtraOptions, CallApiExtraOptionsForHooks, CallApiParameters, CallApiPlugin, CallApiRequestOptions, CallApiRequestOptionsForHooks, CallApiResultLoose as CallApiResult, CallApiResultErrorVariant, CallApiResultSuccessVariant, CallApiSchema, CallApiSchemaConfig, CallApiSuccessOrErrorVariant, DedupeOptions, DefaultCallApiContext, ErrorContext, GetExtendSchemaConfigContext, Hooks, HooksOrHooksArray, InferExtendSchemaContext, InferParamsFromRoute, InferSchemaInput, InferSchemaOutput, InstanceContext, PluginHooks, PluginSetupContext, PossibleHTTPError, PossibleJavaScriptError, PossibleJavaScriptOrValidationError, PossibleValidationError, Register, RequestContext, RequestStreamContext, ResponseContext, ResponseErrorContext, ResponseStreamContext, ResponseTypeType, ResultModeType, RetryOptions, SuccessContext, URLOptions, callApi, createFetchClient, createFetchClientWithContext };
package/dist/index.js CHANGED
@@ -1,5 +1,5 @@
1
1
  import { n as requestOptionDefaults, r as defineEnum, t as extraOptionDefaults } from "./defaults-C6WKIXsf.js";
2
- import { _ as isReadableStream, a as isValidationErrorInstance, b as isValidJsonString, d as isBoolean, f as isFunction, g as isQueryString, h as isPromise, l as toQueryString, m as isPlainObject, n as isHTTPErrorInstance, o as HTTPError, p as isObject, s as ValidationError, u as isArray, v as isSerializable, y as isString } from "./guards-5Ij_loC1.js";
2
+ import { _ as isReadableStream, a as isValidationErrorInstance, b as isValidJsonString, d as isBoolean, f as isFunction, g as isQueryString, h as isPromise, l as toQueryString, m as isPlainObject, n as isHTTPErrorInstance, o as HTTPError, p as isObject, s as ValidationError, u as isArray, v as isSerializable, y as isString } from "./guards-DvOwVtmd.js";
3
3
  import { n as fetchSpecificKeys, t as fallBackRouteSchemaKey } from "./validation-8gwacxHw.js";
4
4
 
5
5
  //#region src/auth.ts
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["validatedResultObject: Prettify<\n\t\tPick<CallApiExtraOptions, (typeof extraOptionsToBeValidated)[number]>\n\t>","validatedResultObject: Prettify<\n\t\tPick<CallApiRequestOptions, (typeof requestOptionsToBeValidated)[number]>\n\t>","headersObject: Record<string, string | undefined>","result: Record<string, unknown>","hookRegistries: HookRegistries","dedupeKey","dedupeCacheScopeKey","$GlobalRequestInfoCache","middlewareRegistries: MiddlewareRegistries","composedMiddleware: Middlewares[keyof Middlewares]","resolvedHooks: Hooks","resolvedMiddlewares: Middlewares","callApi","$GlobalRequestInfoCache: GlobalRequestInfoCache","createFetchClient","$LocalRequestInfoCache: RequestInfoCache","callApi"],"sources":["../src/auth.ts","../src/validation.ts","../src/url.ts","../src/utils/polyfills/combinedSignal.ts","../src/utils/polyfills/timeoutSignal.ts","../src/utils/common.ts","../src/result.ts","../src/hooks.ts","../src/stream.ts","../src/dedupe.ts","../src/middlewares.ts","../src/plugins.ts","../src/retry.ts","../src/createFetchClient.ts"],"sourcesContent":["/* eslint-disable perfectionist/sort-object-types -- Avoid Sorting for now */\n\nimport type { CallApiExtraOptions } from \"./types/common\";\nimport type { Awaitable } from \"./types/type-helpers\";\nimport { isFunction, isObject, isPromise } from \"./utils/guards\";\n\ntype PossibleAuthValue = Awaitable<string | null | undefined>;\n\ntype PossibleAuthValueOrGetter = PossibleAuthValue | (() => PossibleAuthValue);\n\nexport type BearerOrTokenAuth =\n\t| {\n\t\t\ttype?: \"Bearer\";\n\t\t\tbearer?: PossibleAuthValueOrGetter;\n\t\t\ttoken?: never;\n\t }\n\t| {\n\t\t\ttype?: \"Token\";\n\t\t\tbearer?: never;\n\t\t\ttoken?: PossibleAuthValueOrGetter;\n\t };\n\nexport type BasicAuth = {\n\ttype: \"Basic\";\n\tusername: PossibleAuthValueOrGetter;\n\tpassword: PossibleAuthValueOrGetter;\n};\n\n/**\n * Custom auth\n *\n * @param prefix - prefix of the header\n * @param authValue - value of the header\n *\n * @example\n * ```ts\n * {\n * type: \"Custom\",\n * prefix: \"Token\",\n * authValue: \"token\"\n * }\n * ```\n */\nexport type CustomAuth = {\n\ttype: \"Custom\";\n\tprefix: PossibleAuthValueOrGetter;\n\tvalue: PossibleAuthValueOrGetter;\n};\n\n// eslint-disable-next-line perfectionist/sort-union-types -- Let the first one be first\nexport type Auth = PossibleAuthValueOrGetter | BearerOrTokenAuth | BasicAuth | CustomAuth;\n\nconst resolveAuthValue = (value: PossibleAuthValueOrGetter) => (isFunction(value) ? value() : value);\n\ntype AuthHeaderObject = { Authorization: string };\n\nexport const getAuthHeader = async (\n\tauth: CallApiExtraOptions[\"auth\"]\n): Promise<AuthHeaderObject | undefined> => {\n\tif (auth === undefined) return;\n\n\tif (isPromise(auth) || isFunction(auth) || !isObject(auth)) {\n\t\tconst authValue = await resolveAuthValue(auth);\n\n\t\tif (authValue === undefined) return;\n\n\t\treturn {\n\t\t\tAuthorization: `Bearer ${authValue}`,\n\t\t};\n\t}\n\n\tswitch (auth.type) {\n\t\tcase \"Basic\": {\n\t\t\tconst [username, password] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.username),\n\t\t\t\tresolveAuthValue(auth.password),\n\t\t\t]);\n\n\t\t\tif (username === undefined || password === undefined) return;\n\n\t\t\treturn {\n\t\t\t\tAuthorization: `Basic ${globalThis.btoa(`${username}:${password}`)}`,\n\t\t\t};\n\t\t}\n\n\t\tcase \"Custom\": {\n\t\t\tconst [prefix, value] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.prefix),\n\t\t\t\tresolveAuthValue(auth.value),\n\t\t\t]);\n\n\t\t\tif (value === undefined) return;\n\n\t\t\treturn {\n\t\t\t\tAuthorization: `${prefix} ${value}`,\n\t\t\t};\n\t\t}\n\n\t\tdefault: {\n\t\t\tconst [bearer, token] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.bearer),\n\t\t\t\tresolveAuthValue(auth.token),\n\t\t\t]);\n\n\t\t\tif (bearer !== undefined) {\n\t\t\t\treturn { Authorization: `Bearer ${bearer}` };\n\t\t\t}\n\n\t\t\tif (token === undefined) return;\n\n\t\t\treturn { Authorization: `Token ${token}` };\n\t\t}\n\t}\n};\n","import { fallBackRouteSchemaKey, type FallBackRouteSchemaKey } from \"./constants/validation\";\nimport type {\n\tBaseCallApiExtraOptions,\n\tCallApiExtraOptions,\n\tCallApiRequestOptions,\n\tGlobalMeta,\n} from \"./types/common\";\nimport type { Body, HeadersOption, MethodUnion } from \"./types/conditional-types\";\nimport type { StandardSchemaV1 } from \"./types/standard-schema\";\nimport {\n\ttype AnyFunction,\n\ttype AnyString,\n\ttype Awaitable,\n\tdefineEnum,\n\ttype Prettify,\n\ttype UnionToIntersection,\n} from \"./types/type-helpers\";\nimport type { Params, Query } from \"./url\";\nimport { toArray } from \"./utils/common\";\nimport { ValidationError } from \"./utils/external/error\";\nimport { isFunction } from \"./utils/guards\";\n\ntype ResultVariant = \"infer-input\" | \"infer-output\";\n\nexport type InferSchemaResult<TSchema, TFallbackResult, TResultVariant extends ResultVariant> =\n\t// == Checking for undefined first and returning fallback to avoid type errors when passing the config around (weird tbh)\n\tundefined extends TSchema ? TFallbackResult\n\t: TSchema extends StandardSchemaV1 ?\n\t\tTResultVariant extends \"infer-input\" ?\n\t\t\tStandardSchemaV1.InferInput<TSchema>\n\t\t:\tStandardSchemaV1.InferOutput<TSchema>\n\t: TSchema extends AnyFunction<infer TResult> ? Awaited<TResult>\n\t: TFallbackResult;\n\nexport type InferSchemaOutput<TSchema, TFallbackResult = unknown> = InferSchemaResult<\n\tTSchema,\n\tTFallbackResult,\n\t\"infer-output\"\n>;\n\nexport type InferSchemaInput<TSchema, TFallbackResult = unknown> = InferSchemaResult<\n\tTSchema,\n\tTFallbackResult,\n\t\"infer-input\"\n>;\n\nconst handleValidatorFunction = async <TInput>(\n\tvalidator: Extract<CallApiSchema[keyof CallApiSchema], AnyFunction>,\n\tinputData: TInput\n): Promise<StandardSchemaV1.Result<TInput>> => {\n\ttry {\n\t\tconst result = await validator(inputData as never);\n\n\t\treturn { issues: undefined, value: result as never };\n\t} catch (error) {\n\t\treturn { issues: toArray(error) as never, value: undefined };\n\t}\n};\n\nexport const standardSchemaParser = async <\n\tTFullSchema extends CallApiSchema,\n\tTSchemaName extends keyof CallApiSchema,\n\tTSchema extends NonNullable<TFullSchema[TSchemaName]>,\n>(\n\tfullSchema: TFullSchema | undefined,\n\tschemaName: TSchemaName,\n\toptions: { inputValue: InferSchemaInput<TSchema>; response?: Response | null }\n): Promise<InferSchemaOutput<TSchema>> => {\n\tconst { inputValue, response } = options;\n\n\tconst schema = fullSchema?.[schemaName];\n\n\tif (!schema) {\n\t\treturn inputValue as never;\n\t}\n\n\tconst result =\n\t\tisFunction(schema) ?\n\t\t\tawait handleValidatorFunction(schema, inputValue)\n\t\t:\tawait schema[\"~standard\"].validate(inputValue);\n\n\t// == If the `issues` field exists, it means the validation failed\n\n\tif (result.issues) {\n\t\tthrow new ValidationError({\n\t\t\tissueCause: schemaName,\n\t\t\tissues: result.issues,\n\t\t\tresponse: response ?? null,\n\t\t});\n\t}\n\n\treturn result.value as never;\n};\n\nexport interface CallApiSchemaConfig {\n\t/**\n\t * The base url of the schema. By default it's the baseURL of the callApi instance.\n\t */\n\tbaseURL?: string;\n\n\t/**\n\t * Disables runtime validation for the schema.\n\t */\n\tdisableRuntimeValidation?: boolean;\n\n\t/**\n\t * If `true`, the original input value will be used instead of the transformed/validated output.\n\t *\n\t * This is useful when you want to validate the input but don't want any transformations\n\t * applied by the validation schema (e.g., type coercion, default values, etc).\n\t */\n\tdisableValidationOutputApplication?: boolean;\n\n\t/**\n\t * Optional url prefix that will be substituted for the `baseURL` of the schemaConfig at runtime.\n\t *\n\t * This allows you to reuse the same schema against different base URLs (for example,\n\t * swapping between `/api/v1` and `/api/v2`) without redefining the entire schema.\n\t */\n\tprefix?: string;\n\n\t/**\n\t * Controls the strictness of API route validation.\n\t *\n\t * When true:\n\t * - Only routes explicitly defined in the schema will be considered valid to typescript and the runtime.\n\t * - Attempting to call routes not defined in the schema will result in both type errors and runtime validation errors.\n\t * - Useful for ensuring API calls conform exactly to your schema definition\n\t *\n\t * When false or undefined (default):\n\t * - All routes will be allowed, whether they are defined in the schema or not\n\t */\n\tstrict?: boolean;\n}\n\nexport interface CallApiSchema {\n\t/**\n\t * The schema to use for validating the request body.\n\t */\n\tbody?: StandardSchemaV1<Body | undefined> | ((body: Body) => Awaitable<Body | undefined>);\n\n\t/**\n\t * The schema to use for validating the response data.\n\t */\n\tdata?: StandardSchemaV1 | ((data: unknown) => unknown);\n\n\t/**\n\t * The schema to use for validating the response error data.\n\t */\n\terrorData?: StandardSchemaV1 | ((errorData: unknown) => unknown);\n\n\t/**\n\t * The schema to use for validating the request headers.\n\t */\n\theaders?:\n\t\t| StandardSchemaV1<HeadersOption | undefined>\n\t\t| ((headers: HeadersOption) => Awaitable<HeadersOption | undefined>);\n\n\t/**\n\t * The schema to use for validating the meta option.\n\t */\n\tmeta?:\n\t\t| StandardSchemaV1<GlobalMeta | undefined>\n\t\t| ((meta: GlobalMeta) => Awaitable<GlobalMeta | undefined>);\n\n\t/**\n\t * The schema to use for validating the request method.\n\t */\n\tmethod?:\n\t\t| StandardSchemaV1<MethodUnion | undefined>\n\t\t| ((method: MethodUnion) => Awaitable<MethodUnion | undefined>);\n\n\t/**\n\t * The schema to use for validating the request url parameters.\n\t */\n\tparams?: StandardSchemaV1<Params | undefined> | ((params: Params) => Awaitable<Params | undefined>);\n\n\t/**\n\t * The schema to use for validating the request url queries.\n\t */\n\tquery?: StandardSchemaV1<Query | undefined> | ((query: Query) => Awaitable<Query | undefined>);\n}\n\nexport const routeKeyMethods = defineEnum([\"delete\", \"get\", \"patch\", \"post\", \"put\"]);\n\nexport type RouteKeyMethods = (typeof routeKeyMethods)[number];\n\nexport type RouteKeyMethodsURLUnion = `@${RouteKeyMethods}/`;\n\nexport type BaseSchemaRouteKeyPrefixes = FallBackRouteSchemaKey | RouteKeyMethodsURLUnion;\n\nexport type BaseCallApiSchemaRoutes = Partial<\n\tRecord<AnyString | BaseSchemaRouteKeyPrefixes, CallApiSchema>\n>;\n\nexport type BaseCallApiSchemaAndConfig = {\n\tconfig?: CallApiSchemaConfig;\n\troutes: BaseCallApiSchemaRoutes;\n};\n\ntype ValidationOptions<\n\tTSchema extends CallApiSchema[keyof CallApiSchema] = CallApiSchema[keyof CallApiSchema],\n> = {\n\tinputValue: InferSchemaInput<TSchema>;\n\tresponse?: Response | null;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nexport const handleSchemaValidation = async <\n\tTFullSchema extends CallApiSchema,\n\tTSchemaName extends keyof CallApiSchema,\n\tTSchema extends NonNullable<TFullSchema[TSchemaName]>,\n>(\n\tfullSchema: TFullSchema | undefined,\n\tschemaName: TSchemaName,\n\tvalidationOptions: ValidationOptions<TSchema>\n): Promise<InferSchemaOutput<TSchema>> => {\n\tconst { inputValue, response, schemaConfig } = validationOptions;\n\n\tif (schemaConfig?.disableRuntimeValidation) {\n\t\treturn inputValue as never;\n\t}\n\n\tconst validResult = await standardSchemaParser(fullSchema, schemaName, { inputValue, response });\n\n\treturn validResult as never;\n};\n\ntype LastOf<TValue> =\n\tUnionToIntersection<TValue extends unknown ? () => TValue : never> extends () => infer R ? R : never;\n\ntype Push<TArray extends unknown[], TArrayItem> = [...TArray, TArrayItem];\n\ntype UnionToTuple<\n\tTUnion,\n\tTComputedLastUnion = LastOf<TUnion>,\n\tTComputedIsUnionEqualToNever = [TUnion] extends [never] ? true : false,\n> =\n\ttrue extends TComputedIsUnionEqualToNever ? []\n\t:\tPush<UnionToTuple<Exclude<TUnion, TComputedLastUnion>>, TComputedLastUnion>;\n\nexport type Tuple<TTuple, TArray extends TTuple[] = []> =\n\tUnionToTuple<TTuple>[\"length\"] extends TArray[\"length\"] ? [...TArray]\n\t:\tTuple<TTuple, [TTuple, ...TArray]>;\n\nconst extraOptionsToBeValidated = [\"meta\", \"params\", \"query\"] satisfies Tuple<\n\tExtract<keyof CallApiSchema, keyof CallApiExtraOptions>\n>;\n\ntype ExtraOptionsValidationOptions = {\n\toptions: CallApiExtraOptions;\n\tschema: CallApiSchema | undefined;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nconst handleExtraOptionsValidation = async (validationOptions: ExtraOptionsValidationOptions) => {\n\tconst { options, schema, schemaConfig } = validationOptions;\n\n\tconst validationResultArray = await Promise.all(\n\t\textraOptionsToBeValidated.map((schemaName) =>\n\t\t\thandleSchemaValidation(schema, schemaName, {\n\t\t\t\tinputValue: options[schemaName],\n\t\t\t\tschemaConfig,\n\t\t\t})\n\t\t)\n\t);\n\n\tconst validatedResultObject: Prettify<\n\t\tPick<CallApiExtraOptions, (typeof extraOptionsToBeValidated)[number]>\n\t> = {};\n\n\tfor (const [index, schemaName] of extraOptionsToBeValidated.entries()) {\n\t\tconst validationResult = validationResultArray[index];\n\n\t\tif (validationResult === undefined) continue;\n\n\t\tvalidatedResultObject[schemaName] = validationResult as never;\n\t}\n\n\treturn validatedResultObject;\n};\n\nconst requestOptionsToBeValidated = [\"body\", \"headers\", \"method\"] satisfies Tuple<\n\tExtract<keyof CallApiSchema, keyof CallApiRequestOptions>\n>;\n\ntype RequestOptionsValidationOptions = {\n\trequestOptions: CallApiRequestOptions;\n\tschema: CallApiSchema | undefined;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nconst handleRequestOptionsValidation = async (validationOptions: RequestOptionsValidationOptions) => {\n\tconst { requestOptions, schema, schemaConfig } = validationOptions;\n\n\tconst validationResultArray = await Promise.all(\n\t\trequestOptionsToBeValidated.map((schemaName) =>\n\t\t\thandleSchemaValidation(schema, schemaName, {\n\t\t\t\tinputValue: requestOptions[schemaName],\n\t\t\t\tschemaConfig,\n\t\t\t})\n\t\t)\n\t);\n\n\tconst validatedResultObject: Prettify<\n\t\tPick<CallApiRequestOptions, (typeof requestOptionsToBeValidated)[number]>\n\t> = {};\n\n\tfor (const [index, propertyKey] of requestOptionsToBeValidated.entries()) {\n\t\tconst validationResult = validationResultArray[index];\n\n\t\tif (validationResult === undefined) continue;\n\n\t\tvalidatedResultObject[propertyKey] = validationResult as never;\n\t}\n\n\treturn validatedResultObject;\n};\n\nexport const handleConfigValidation = async (\n\tvalidationOptions: GetResolvedSchemaContext\n\t\t& Omit<ExtraOptionsValidationOptions & RequestOptionsValidationOptions, \"schema\" | \"schemaConfig\">\n) => {\n\tconst { baseExtraOptions, currentRouteSchemaKey, extraOptions, options, requestOptions } =\n\t\tvalidationOptions;\n\n\tconst { currentRouteSchema, resolvedSchema } = getResolvedSchema({\n\t\tbaseExtraOptions,\n\t\tcurrentRouteSchemaKey,\n\t\textraOptions,\n\t});\n\n\tconst resolvedSchemaConfig = getResolvedSchemaConfig({ baseExtraOptions, extraOptions });\n\n\tif (resolvedSchemaConfig?.strict === true && !currentRouteSchema) {\n\t\tthrow new ValidationError({\n\t\t\tissueCause: \"schemaConfig-(strict)\",\n\t\t\tissues: [{ message: `Strict Mode - No schema found for route '${currentRouteSchemaKey}' ` }],\n\t\t\tresponse: null,\n\t\t});\n\t}\n\n\tif (resolvedSchemaConfig?.disableRuntimeValidation) {\n\t\treturn {\n\t\t\textraOptionsValidationResult: null,\n\t\t\trequestOptionsValidationResult: null,\n\t\t\tresolvedSchema,\n\t\t\tresolvedSchemaConfig,\n\t\t\tshouldApplySchemaOutput: false,\n\t\t};\n\t}\n\n\tconst [extraOptionsValidationResult, requestOptionsValidationResult] = await Promise.all([\n\t\thandleExtraOptionsValidation({\n\t\t\toptions,\n\t\t\tschema: resolvedSchema,\n\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t}),\n\t\thandleRequestOptionsValidation({\n\t\t\trequestOptions,\n\t\t\tschema: resolvedSchema,\n\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t}),\n\t]);\n\n\tconst shouldApplySchemaOutput =\n\t\t(Boolean(extraOptionsValidationResult) || Boolean(requestOptionsValidationResult))\n\t\t&& !resolvedSchemaConfig?.disableValidationOutputApplication;\n\n\treturn {\n\t\textraOptionsValidationResult,\n\t\trequestOptionsValidationResult,\n\t\tresolvedSchema,\n\t\tresolvedSchemaConfig,\n\t\tshouldApplySchemaOutput,\n\t};\n};\n\ntype GetResolvedSchemaContext = {\n\tbaseExtraOptions: BaseCallApiExtraOptions;\n\tcurrentRouteSchemaKey: string;\n\textraOptions: CallApiExtraOptions;\n};\n\nexport const getResolvedSchema = (context: GetResolvedSchemaContext) => {\n\tconst { baseExtraOptions, currentRouteSchemaKey, extraOptions } = context;\n\n\tconst fallbackRouteSchema = baseExtraOptions.schema?.routes[fallBackRouteSchemaKey];\n\tconst currentRouteSchema = baseExtraOptions.schema?.routes[currentRouteSchemaKey];\n\n\tconst resolvedRouteSchema = {\n\t\t...fallbackRouteSchema,\n\t\t// == Current route schema takes precedence over fallback route schema\n\t\t...currentRouteSchema,\n\t} satisfies CallApiSchema as CallApiSchema | undefined;\n\n\tconst resolvedSchema =\n\t\tisFunction(extraOptions.schema) ?\n\t\t\textraOptions.schema({\n\t\t\t\tbaseSchemaRoutes: baseExtraOptions.schema?.routes ?? {},\n\t\t\t\tcurrentRouteSchema: resolvedRouteSchema ?? {},\n\t\t\t})\n\t\t:\t(extraOptions.schema ?? resolvedRouteSchema);\n\n\treturn { currentRouteSchema, resolvedSchema };\n};\n\nexport const getResolvedSchemaConfig = (\n\tcontext: Omit<GetResolvedSchemaContext, \"currentRouteSchemaKey\">\n) => {\n\tconst { baseExtraOptions, extraOptions } = context;\n\n\tconst resolvedSchemaConfig =\n\t\tisFunction(extraOptions.schemaConfig) ?\n\t\t\textraOptions.schemaConfig({ baseSchemaConfig: baseExtraOptions.schema?.config ?? {} })\n\t\t:\t(extraOptions.schemaConfig ?? baseExtraOptions.schema?.config);\n\n\treturn resolvedSchemaConfig;\n};\n\nexport const getCurrentRouteSchemaKeyAndMainInitURL = (\n\tcontext: Pick<GetResolvedSchemaContext, \"baseExtraOptions\" | \"extraOptions\"> & { initURL: string }\n) => {\n\tconst { baseExtraOptions, extraOptions, initURL } = context;\n\n\tconst schemaConfig = getResolvedSchemaConfig({ baseExtraOptions, extraOptions });\n\n\tlet currentRouteSchemaKey = initURL;\n\tlet mainInitURL = initURL;\n\n\tif (schemaConfig?.prefix && currentRouteSchemaKey.startsWith(schemaConfig.prefix)) {\n\t\tcurrentRouteSchemaKey = currentRouteSchemaKey.replace(schemaConfig.prefix, \"\");\n\n\t\tmainInitURL = mainInitURL.replace(schemaConfig.prefix, schemaConfig.baseURL ?? \"\");\n\t}\n\n\tif (schemaConfig?.baseURL && currentRouteSchemaKey.startsWith(schemaConfig.baseURL)) {\n\t\tcurrentRouteSchemaKey = currentRouteSchemaKey.replace(schemaConfig.baseURL, \"\");\n\t}\n\n\treturn { currentRouteSchemaKey, mainInitURL };\n};\n","import type { CallApiExtraOptions } from \"./types/common\";\nimport type { AnyString, UnmaskType } from \"./types/type-helpers\";\nimport { toQueryString } from \"./utils/external/body\";\nimport { isArray } from \"./utils/guards\";\nimport { type RouteKeyMethodsURLUnion, routeKeyMethods } from \"./validation\";\n\nconst slash = \"/\";\nconst colon = \":\";\nconst openBrace = \"{\";\nconst closeBrace = \"}\";\n\nconst mergeUrlWithParams = (url: string, params: CallApiExtraOptions[\"params\"]) => {\n\tif (!params) {\n\t\treturn url;\n\t}\n\n\tlet newUrl = url;\n\n\tif (isArray(params)) {\n\t\tconst urlParts = newUrl.split(slash);\n\n\t\t// == Find all parameters in order (both :param and {param} patterns)\n\t\tconst matchedParamsArray = urlParts.filter(\n\t\t\t(part) => part.startsWith(colon) || (part.startsWith(openBrace) && part.endsWith(closeBrace))\n\t\t);\n\n\t\tfor (const [paramIndex, matchedParam] of matchedParamsArray.entries()) {\n\t\t\tconst stringParamValue = String(params[paramIndex]);\n\t\t\tnewUrl = newUrl.replace(matchedParam, stringParamValue);\n\t\t}\n\n\t\treturn newUrl;\n\t}\n\n\t// == Handle object params - replace both :param and {param} patterns\n\tfor (const [paramKey, paramValue] of Object.entries(params)) {\n\t\tconst colonPattern = `${colon}${paramKey}` as const;\n\t\tconst bracePattern = `${openBrace}${paramKey}${closeBrace}` as const;\n\t\tconst stringValue = String(paramValue);\n\n\t\tnewUrl = newUrl.replace(colonPattern, stringValue);\n\t\tnewUrl = newUrl.replace(bracePattern, stringValue);\n\t}\n\n\treturn newUrl;\n};\n\nconst questionMark = \"?\";\nconst ampersand = \"&\";\nconst mergeUrlWithQuery = (url: string, query: CallApiExtraOptions[\"query\"]): string => {\n\tif (!query) {\n\t\treturn url;\n\t}\n\n\tconst queryString = toQueryString(query);\n\n\tif (queryString?.length === 0) {\n\t\treturn url;\n\t}\n\n\tif (url.endsWith(questionMark)) {\n\t\treturn `${url}${queryString}`;\n\t}\n\n\tif (url.includes(questionMark)) {\n\t\treturn `${url}${ampersand}${queryString}`;\n\t}\n\n\treturn `${url}${questionMark}${queryString}`;\n};\n\n/**\n * @description Extracts the HTTP method from method-prefixed route patterns.\n *\n * Analyzes URLs that start with method modifiers (e.g., \"@get/\", \"@post/\") and extracts\n * the HTTP method for use in API requests. This enables method specification directly\n * in route definitions.\n *\n * @param initURL - The URL string to analyze for method modifiers\n * @returns The extracted HTTP method (lowercase) if found, otherwise undefined\n *\n * @example\n * ```typescript\n * // Method extraction from prefixed routes\n * extractMethodFromURL(\"@get/users\"); // Returns: \"get\"\n * extractMethodFromURL(\"@post/users\"); // Returns: \"post\"\n * extractMethodFromURL(\"@put/users/:id\"); // Returns: \"put\"\n * extractMethodFromURL(\"@delete/users/:id\"); // Returns: \"delete\"\n * extractMethodFromURL(\"@patch/users/:id\"); // Returns: \"patch\"\n *\n * // No method modifier\n * extractMethodFromURL(\"/users\"); // Returns: undefined\n * extractMethodFromURL(\"users\"); // Returns: undefined\n *\n * // Invalid or unsupported methods\n * extractMethodFromURL(\"@invalid/users\"); // Returns: undefined\n * extractMethodFromURL(\"@/users\"); // Returns: undefined\n *\n * // Edge cases\n * extractMethodFromURL(undefined); // Returns: undefined\n * extractMethodFromURL(\"\"); // Returns: undefined\n * ```\n */\nexport const extractMethodFromURL = (initURL: string | undefined) => {\n\tif (!initURL?.startsWith(\"@\")) return;\n\n\tconst method = initURL.split(\"@\")[1]?.split(\"/\")[0];\n\n\tif (!method || !routeKeyMethods.includes(method as (typeof routeKeyMethods)[number])) return;\n\n\treturn method;\n};\n\nconst normalizeURL = (initURL: string) => {\n\tconst methodFromURL = extractMethodFromURL(initURL);\n\n\tif (!methodFromURL) {\n\t\treturn initURL;\n\t}\n\n\tconst normalizedURL = initURL.replace(`@${methodFromURL}/`, \"/\");\n\n\treturn normalizedURL;\n};\n\ntype GetFullURLOptions = {\n\t/** Base URL to prepend to relative URLs */\n\tbaseURL: string | undefined;\n\t/** Initial URL pattern that may contain parameters and method modifiers */\n\tinitURL: string;\n\t/** Parameters to substitute into the URL path */\n\tparams: CallApiExtraOptions[\"params\"];\n\t/** Query parameters to append to the URL */\n\tquery: CallApiExtraOptions[\"query\"];\n};\n\nexport const getFullAndNormalizedURL = (options: GetFullURLOptions) => {\n\tconst { baseURL, initURL, params, query } = options;\n\n\tconst normalizedInitURL = normalizeURL(initURL);\n\n\tconst urlWithMergedParams = mergeUrlWithParams(normalizedInitURL, params);\n\n\tconst urlWithMergedQueryAndParams = mergeUrlWithQuery(urlWithMergedParams, query);\n\n\tconst shouldPrependBaseURL = !urlWithMergedQueryAndParams.startsWith(\"http\") && baseURL;\n\n\tconst fullURL =\n\t\tshouldPrependBaseURL ? `${baseURL}${urlWithMergedQueryAndParams}` : urlWithMergedQueryAndParams;\n\n\treturn {\n\t\tfullURL,\n\t\tnormalizedInitURL,\n\t};\n};\n\nexport type AllowedQueryParamValues = UnmaskType<boolean | number | string>;\n\nexport type RecordStyleParams = UnmaskType<Record<string, AllowedQueryParamValues>>;\n\nexport type TupleStyleParams = UnmaskType<AllowedQueryParamValues[]>;\n\nexport type Params = UnmaskType<RecordStyleParams | TupleStyleParams>;\n\nexport type Query = UnmaskType<Record<string, AllowedQueryParamValues>>;\n\nexport type InitURLOrURLObject = AnyString | RouteKeyMethodsURLUnion | URL;\n\nexport interface URLOptions {\n\t/**\n\t * Base URL for all API requests. Will only be prepended to relative URLs.\n\t *\n\t * Absolute URLs (starting with http/https) will not be prepended by the baseURL.\n\t *\n\t * @example\n\t * ```ts\n\t * // Set base URL for all requests\n\t * baseURL: \"https://api.example.com/v1\"\n\t *\n\t * // Then use relative URLs in requests\n\t * callApi(\"/users\") // → https://api.example.com/v1/users\n\t * callApi(\"/posts/123\") // → https://api.example.com/v1/posts/123\n\t *\n\t * // Environment-specific base URLs\n\t * baseURL: process.env.NODE_ENV === \"production\"\n\t * ? \"https://api.example.com\"\n\t * : \"http://localhost:3000/api\"\n\t * ```\n\t */\n\tbaseURL?: string;\n\n\t/**\n\t * Resolved request URL after processing baseURL, parameters, and query strings (readonly)\n\t *\n\t * This is the final URL that will be used for the HTTP request, computed from\n\t * baseURL, initURL, params, and query parameters.\n\t *\n\t */\n\treadonly fullURL?: string;\n\n\t/**\n\t * The original URL string passed to the callApi instance (readonly)\n\t *\n\t * This preserves the original URL as provided, including any method modifiers like \"@get/\" or \"@post/\".\n\t *\n\t */\n\treadonly initURL?: string;\n\n\t/**\n\t * The URL string after normalization, with method modifiers removed(readonly)\n\t *\n\t * Method modifiers like \"@get/\", \"@post/\" are stripped to create a clean URL\n\t * for parameter substitution and final URL construction.\n\t *\n\t */\n\treadonly initURLNormalized?: string;\n\n\t/**\n\t * Parameters to be substituted into URL path segments.\n\t *\n\t * Supports both object-style (named parameters) and array-style (positional parameters)\n\t * for flexible URL parameter substitution.\n\t *\n\t * @example\n\t * ```typescript\n\t * // Object-style parameters (recommended)\n\t * const namedParams: URLOptions = {\n\t * initURL: \"/users/:userId/posts/:postId\",\n\t * params: { userId: \"123\", postId: \"456\" }\n\t * };\n\t * // Results in: /users/123/posts/456\n\t *\n\t * // Array-style parameters (positional)\n\t * const positionalParams: URLOptions = {\n\t * initURL: \"/users/:userId/posts/:postId\",\n\t * params: [\"123\", \"456\"] // Maps in order: userId=123, postId=456\n\t * };\n\t * // Results in: /users/123/posts/456\n\t *\n\t * // Single parameter\n\t * const singleParam: URLOptions = {\n\t * initURL: \"/users/:id\",\n\t * params: { id: \"user-123\" }\n\t * };\n\t * // Results in: /users/user-123\n\t * ```\n\t */\n\tparams?: Params;\n\n\t/**\n\t * Query parameters to append to the URL as search parameters.\n\t *\n\t * These will be serialized into the URL query string using standard\n\t * URL encoding practices.\n\t *\n\t * @example\n\t * ```typescript\n\t * // Basic query parameters\n\t * const queryOptions: URLOptions = {\n\t * initURL: \"/users\",\n\t * query: {\n\t * page: 1,\n\t * limit: 10,\n\t * search: \"john doe\",\n\t * active: true\n\t * }\n\t * };\n\t * // Results in: /users?page=1&limit=10&search=john%20doe&active=true\n\t *\n\t * // Filtering and sorting\n\t * const filterOptions: URLOptions = {\n\t * initURL: \"/products\",\n\t * query: {\n\t * category: \"electronics\",\n\t * minPrice: 100,\n\t * maxPrice: 500,\n\t * sortBy: \"price\",\n\t * order: \"asc\"\n\t * }\n\t * };\n\t * // Results in: /products?category=electronics&minPrice=100&maxPrice=500&sortBy=price&order=asc\n\t * ```\n\t */\n\tquery?: Query;\n}\n","export const createCombinedSignalPolyfill = (signals: AbortSignal[]) => {\n\tconst controller = new AbortController();\n\n\tconst handleAbort = (actualSignal: AbortSignal) => {\n\t\tif (controller.signal.aborted) return;\n\n\t\tcontroller.abort(actualSignal.reason);\n\t};\n\n\tfor (const actualSignal of signals) {\n\t\tif (actualSignal.aborted) {\n\t\t\thandleAbort(actualSignal);\n\t\t\tbreak;\n\t\t}\n\n\t\tactualSignal.addEventListener(\"abort\", () => handleAbort(actualSignal), {\n\t\t\tsignal: controller.signal,\n\t\t});\n\t}\n\n\treturn controller.signal;\n};\n","export const createTimeoutSignalPolyfill = (milliseconds: number) => {\n\tconst controller = new AbortController();\n\n\tconst reason = new DOMException(\"Request timed out\", \"TimeoutError\");\n\n\tconst timeout = setTimeout(() => controller.abort(reason), milliseconds);\n\n\tcontroller.signal.addEventListener(\"abort\", () => clearTimeout(timeout));\n\n\treturn controller.signal;\n};\n","import { getAuthHeader } from \"../auth\";\nimport { fetchSpecificKeys } from \"../constants/common\";\nimport { extraOptionDefaults, requestOptionDefaults } from \"../constants/defaults\";\nimport type { RequestContext } from \"../hooks\";\nimport type { Middlewares } from \"../middlewares\";\nimport type { BaseCallApiExtraOptions, CallApiExtraOptions, CallApiRequestOptions } from \"../types/common\";\nimport type { DistributiveOmit } from \"../types/type-helpers\";\nimport { extractMethodFromURL } from \"../url\";\nimport {\n\tisArray,\n\tisFunction,\n\tisPlainObject,\n\tisQueryString,\n\tisSerializable,\n\tisValidJsonString,\n} from \"./guards\";\nimport { createCombinedSignalPolyfill, createTimeoutSignalPolyfill } from \"./polyfills\";\n\nexport const omitKeys = <\n\tTObject extends Record<string, unknown>,\n\tconst TOmitArray extends Array<keyof TObject> | ReadonlyArray<keyof TObject>,\n>(\n\tinitialObject: TObject,\n\tkeysToOmit: TOmitArray\n) => {\n\tconst updatedObject = {} as Record<string, unknown>;\n\n\tconst keysToOmitSet = new Set(keysToOmit);\n\n\tfor (const [key, value] of Object.entries(initialObject)) {\n\t\tif (!keysToOmitSet.has(key)) {\n\t\t\tupdatedObject[key] = value;\n\t\t}\n\t}\n\n\treturn updatedObject as DistributiveOmit<TObject, TOmitArray[number]>;\n};\n\nexport const pickKeys = <\n\tTObject extends Record<string, unknown>,\n\tconst TPickArray extends Array<keyof TObject> | ReadonlyArray<keyof TObject>,\n>(\n\tinitialObject: TObject,\n\tkeysToPick: TPickArray\n) => {\n\tconst updatedObject = {} as Record<string, unknown>;\n\n\tconst keysToPickSet = new Set(keysToPick);\n\n\tfor (const [key, value] of Object.entries(initialObject)) {\n\t\tif (keysToPickSet.has(key)) {\n\t\t\tupdatedObject[key] = value;\n\t\t}\n\t}\n\n\treturn updatedObject as Pick<TObject, TPickArray[number]>;\n};\n\n// eslint-disable-next-line ts-eslint/no-explicit-any -- Any is required here so that one can pass custom function type without type errors\nexport const splitBaseConfig = (baseConfig: Record<string, any>) =>\n\t[\n\t\tpickKeys(baseConfig, fetchSpecificKeys) as CallApiRequestOptions,\n\t\tomitKeys(baseConfig, fetchSpecificKeys) as BaseCallApiExtraOptions,\n\t] as const;\n\n// eslint-disable-next-line ts-eslint/no-explicit-any -- Any is required here so that one can pass custom function type without type errors\nexport const splitConfig = (config: Record<string, any>) =>\n\t[\n\t\tpickKeys(config, fetchSpecificKeys) as CallApiRequestOptions,\n\t\tomitKeys(config, fetchSpecificKeys) as CallApiExtraOptions,\n\t] as const;\n\nexport const objectifyHeaders = (headers: CallApiRequestOptions[\"headers\"]) => {\n\tif (!headers || isPlainObject(headers)) {\n\t\treturn headers;\n\t}\n\n\treturn Object.fromEntries(headers);\n};\n\nexport type GetHeadersOptions = {\n\tauth: CallApiExtraOptions[\"auth\"];\n\tbody: CallApiRequestOptions[\"body\"];\n\theaders: CallApiRequestOptions[\"headers\"];\n};\n\nexport const getHeaders = async (options: GetHeadersOptions) => {\n\tconst { auth, body, headers } = options;\n\n\t// == Return early if any of the following conditions are not met (so that native fetch would auto set the correct headers):\n\tconst shouldResolveHeaders = Boolean(headers) || Boolean(body) || Boolean(auth);\n\n\tif (!shouldResolveHeaders) return;\n\n\tconst headersObject: Record<string, string | undefined> = {\n\t\t...(await getAuthHeader(auth)),\n\t\t...objectifyHeaders(headers),\n\t};\n\n\tif (isQueryString(body)) {\n\t\theadersObject[\"Content-Type\"] = \"application/x-www-form-urlencoded\";\n\n\t\treturn headersObject;\n\t}\n\n\tif (isSerializable(body) || isValidJsonString(body)) {\n\t\theadersObject[\"Content-Type\"] = \"application/json\";\n\t\theadersObject.Accept = \"application/json\";\n\t}\n\n\treturn headersObject;\n};\n\nexport type GetMethodContext = {\n\t/** The URL string that may contain method modifiers like \"@get/\" or \"@post/\" */\n\tinitURL: string | undefined;\n\t/** Explicitly specified HTTP method */\n\tmethod: CallApiRequestOptions[\"method\"];\n};\n\nexport const getMethod = (ctx: GetMethodContext) => {\n\tconst { initURL, method } = ctx;\n\n\treturn (\n\t\tmethod?.toUpperCase() ?? extractMethodFromURL(initURL)?.toUpperCase() ?? requestOptionDefaults.method\n\t);\n};\n\nexport type GetBodyOptions = {\n\tbody: CallApiRequestOptions[\"body\"];\n\tbodySerializer: CallApiExtraOptions[\"bodySerializer\"];\n};\n\nexport const getBody = (options: GetBodyOptions) => {\n\tconst { body, bodySerializer } = options;\n\n\tif (isSerializable(body)) {\n\t\tconst selectedBodySerializer = bodySerializer ?? extraOptionDefaults.bodySerializer;\n\n\t\treturn selectedBodySerializer(body);\n\t}\n\n\treturn body;\n};\n\nexport const getInitFetchImpl = (customFetchImpl: CallApiExtraOptions[\"customFetchImpl\"]) => {\n\tif (customFetchImpl) {\n\t\treturn customFetchImpl;\n\t}\n\n\tif (typeof globalThis !== \"undefined\" && isFunction(globalThis.fetch)) {\n\t\treturn globalThis.fetch;\n\t}\n\n\tthrow new Error(\"No fetch implementation found\");\n};\n\nexport const getFetchImpl = (context: {\n\tcustomFetchImpl: CallApiExtraOptions[\"customFetchImpl\"];\n\tfetchMiddleware: Middlewares[\"fetchMiddleware\"];\n\trequestContext: RequestContext;\n}) => {\n\tconst { customFetchImpl, fetchMiddleware, requestContext } = context;\n\n\tconst initFetchImpl = getInitFetchImpl(customFetchImpl);\n\n\tconst resolvedFetchImpl =\n\t\tfetchMiddleware ? fetchMiddleware({ ...requestContext, fetchImpl: initFetchImpl }) : initFetchImpl;\n\n\treturn resolvedFetchImpl;\n};\n\nexport const waitFor = (delay: number) => {\n\tif (delay === 0) return;\n\n\tconst promise = new Promise((resolve) => setTimeout(resolve, delay));\n\n\treturn promise;\n};\n\nexport const createCombinedSignal = (...signals: Array<AbortSignal | null | undefined>) => {\n\tconst cleanedSignals = signals.filter((signal) => signal != null);\n\n\tif (!(\"any\" in AbortSignal)) {\n\t\treturn createCombinedSignalPolyfill(cleanedSignals);\n\t}\n\n\tconst combinedSignal = AbortSignal.any(cleanedSignals);\n\n\treturn combinedSignal;\n};\n\nexport const createTimeoutSignal = (milliseconds: number | null | undefined) => {\n\tif (milliseconds == null) {\n\t\treturn null;\n\t}\n\n\tif (!(\"timeout\" in AbortSignal)) {\n\t\treturn createTimeoutSignalPolyfill(milliseconds);\n\t}\n\n\treturn AbortSignal.timeout(milliseconds);\n};\n\nexport const deterministicHashFn = (value: unknown): string => {\n\treturn JSON.stringify(value, (_, val: unknown) => {\n\t\tif (!isPlainObject(val)) {\n\t\t\treturn val;\n\t\t}\n\n\t\tconst sortedKeys = Object.keys(val).toSorted();\n\n\t\tconst result: Record<string, unknown> = {};\n\n\t\tfor (const key of sortedKeys) {\n\t\t\tresult[key] = val[key];\n\t\t}\n\n\t\treturn result;\n\t});\n};\n\nexport const toArray = (value: unknown) => (isArray(value) ? value : [value]);\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport type { CallApiExtraOptions } from \"./types/common\";\nimport type { ThrowOnErrorUnion } from \"./types/conditional-types\";\nimport type { DefaultDataType, DefaultThrowOnError } from \"./types/default-types\";\nimport type { AnyString, Awaitable, DistributiveOmit, UnmaskType } from \"./types/type-helpers\";\nimport { omitKeys } from \"./utils/common\";\nimport type { HTTPError, ValidationError } from \"./utils/external/error\";\nimport { isHTTPErrorInstance, isValidationErrorInstance } from \"./utils/external/guards\";\n\nexport type ResponseType = \"blob\" | \"json\" | \"text\";\n\ntype Parser<TData> = (responseString: string) => Awaitable<TData>;\n\nexport const getResponseType = <TResponse>(response: Response, parser: Parser<TResponse>) => ({\n\tarrayBuffer: () => response.arrayBuffer(),\n\tblob: () => response.blob(),\n\tformData: () => response.formData(),\n\tjson: async (): Promise<TResponse> => {\n\t\tconst text = await response.text();\n\t\treturn parser(text);\n\t},\n\tstream: () => response.body,\n\ttext: () => response.text(),\n});\n\ntype InitResponseTypeMap<TResponse = unknown> = ReturnType<typeof getResponseType<TResponse>>;\n\ntype ResponseTypeUnion = keyof InitResponseTypeMap;\n\ntype ResponseTypePlaceholder = null;\n\nexport type ResponseTypeType = ResponseTypePlaceholder | ResponseTypeUnion;\n\nexport type ResponseTypeMap<TResponse> = {\n\t[Key in keyof InitResponseTypeMap<TResponse>]: Awaited<ReturnType<InitResponseTypeMap<TResponse>[Key]>>;\n};\n\nexport type GetResponseType<\n\tTResponse,\n\tTResponseType extends ResponseTypeType,\n\tTComputedResponseTypeMap extends ResponseTypeMap<TResponse> = ResponseTypeMap<TResponse>,\n> =\n\tnull extends TResponseType ? TComputedResponseTypeMap[\"json\"]\n\t: TResponseType extends NonNullable<ResponseTypeType> ? TComputedResponseTypeMap[TResponseType]\n\t: never;\n\nconst textTypes = new Set([\"image/svg\", \"application/xml\", \"application/xhtml\", \"application/html\"]);\nconst JSON_REGEX = /^application\\/(?:[\\w!#$%&*.^`~-]*\\+)?json(;.+)?$/i;\n\nconst detectResponseType = (response: Response): Extract<ResponseTypeType, \"blob\" | \"json\" | \"text\"> => {\n\tconst initContentType = response.headers.get(\"content-type\");\n\n\tif (!initContentType) {\n\t\treturn extraOptionDefaults.responseType;\n\t}\n\n\tconst contentType = initContentType.split(\";\")[0] ?? \"\";\n\n\tif (JSON_REGEX.test(contentType)) {\n\t\treturn \"json\";\n\t}\n\n\tif (textTypes.has(contentType) || contentType.startsWith(\"text/\")) {\n\t\treturn \"text\";\n\t}\n\n\treturn \"blob\";\n};\n\nexport const resolveResponseData = <TResponse>(\n\tresponse: Response,\n\tresponseType: ResponseTypeType | undefined,\n\tparser: Parser<TResponse> | undefined\n) => {\n\tconst selectedParser = parser ?? extraOptionDefaults.responseParser;\n\tconst selectedResponseType = responseType ?? detectResponseType(response);\n\n\tconst RESPONSE_TYPE_LOOKUP = getResponseType<TResponse>(response, selectedParser);\n\n\tif (!Object.hasOwn(RESPONSE_TYPE_LOOKUP, selectedResponseType)) {\n\t\tthrow new Error(`Invalid response type: ${responseType}`);\n\t}\n\n\treturn RESPONSE_TYPE_LOOKUP[selectedResponseType]();\n};\n\nexport type CallApiResultSuccessVariant<TData> = {\n\tdata: NoInfer<TData>;\n\terror: null;\n\tresponse: Response;\n};\nexport type PossibleJavaScriptError = UnmaskType<{\n\terrorData: false;\n\tmessage: string;\n\tname: \"AbortError\" | \"Error\" | \"SyntaxError\" | \"TimeoutError\" | \"TypeError\" | AnyString;\n\toriginalError: DOMException | Error | SyntaxError | TypeError;\n}>;\n\nexport type PossibleHTTPError<TErrorData> = UnmaskType<{\n\terrorData: NoInfer<TErrorData>;\n\tmessage: string;\n\tname: \"HTTPError\";\n\toriginalError: HTTPError;\n}>;\n\nexport type PossibleValidationError = UnmaskType<{\n\terrorData: ValidationError[\"errorData\"];\n\tissueCause: ValidationError[\"issueCause\"];\n\tmessage: string;\n\tname: \"ValidationError\";\n\toriginalError: ValidationError;\n}>;\n\nexport type PossibleJavaScriptOrValidationError = UnmaskType<\n\tPossibleJavaScriptError | PossibleValidationError\n>;\n\nexport type CallApiResultErrorVariant<TErrorData> =\n\t| {\n\t\t\tdata: null;\n\t\t\terror: PossibleHTTPError<TErrorData>;\n\t\t\tresponse: Response;\n\t }\n\t| {\n\t\t\tdata: null;\n\t\t\terror: PossibleJavaScriptOrValidationError;\n\t\t\tresponse: Response | null;\n\t };\n\nexport type CallApiSuccessOrErrorVariant<TData, TError> =\n\t| CallApiResultErrorVariant<TError>\n\t| CallApiResultSuccessVariant<TData>;\n\nexport type ResultModeMapWithoutException<\n\tTData,\n\tTErrorData,\n\tTResponseType extends ResponseTypeType,\n\tTComputedData = GetResponseType<TData, TResponseType>,\n\tTComputedErrorData = GetResponseType<TErrorData, TResponseType>,\n\tTComputedResult extends CallApiSuccessOrErrorVariant<TComputedData, TComputedErrorData> =\n\t\tCallApiSuccessOrErrorVariant<TComputedData, TComputedErrorData>,\n> = UnmaskType<{\n\tall: TComputedResult;\n\tonlyData: TComputedResult[\"data\"];\n\tonlyResponse: TComputedResult[\"response\"];\n\twithoutResponse: DistributiveOmit<TComputedResult, \"response\">;\n}>;\n\ntype ResultModeMapWithException<\n\tTData,\n\tTResponseType extends ResponseTypeType,\n\tTComputedData = GetResponseType<TData, TResponseType>,\n\tTComputedResult extends CallApiResultSuccessVariant<TComputedData> =\n\t\tCallApiResultSuccessVariant<TComputedData>,\n> = {\n\tall: TComputedResult;\n\tonlyData: TComputedResult[\"data\"];\n\tonlyResponse: TComputedResult[\"response\"];\n\twithoutResponse: DistributiveOmit<TComputedResult, \"response\">;\n};\n\nexport type ResultModeMap<\n\tTData = DefaultDataType,\n\tTErrorData = DefaultDataType,\n\tTResponseType extends ResponseTypeType = ResponseTypeType,\n\tTThrowOnError extends ThrowOnErrorUnion = DefaultThrowOnError,\n> =\n\tTThrowOnError extends true ? ResultModeMapWithException<TData, TResponseType>\n\t:\tResultModeMapWithoutException<TData, TErrorData, TResponseType>;\n\ntype ResultModePlaceholder = null;\n\ntype ResultModeUnion = keyof ResultModeMap;\n\n// FIXME - Revisit this idea later. Take inspirations from how zod does it with pick({}) and omit({d})\n// type ResultModeObject = { data?: boolean; error?: boolean; response?: boolean };\n\nexport type ResultModeType = ResultModePlaceholder | ResultModeUnion;\n\nexport type GetCallApiResult<\n\tTData,\n\tTErrorData,\n\tTResultMode extends ResultModeType,\n\tTThrowOnError extends ThrowOnErrorUnion,\n\tTResponseType extends ResponseTypeType,\n\tTComputedResultModeMapWithException extends ResultModeMapWithException<TData, TResponseType> =\n\t\tResultModeMapWithException<TData, TResponseType>,\n\tTComputedResultModeMap extends ResultModeMap<TData, TErrorData, TResponseType, TThrowOnError> =\n\t\tResultModeMap<TData, TErrorData, TResponseType, TThrowOnError>,\n> =\n\tTErrorData extends false ? TComputedResultModeMapWithException[\"onlyData\"]\n\t: TErrorData extends false | undefined ? TComputedResultModeMapWithException[\"onlyData\"]\n\t: ResultModePlaceholder extends TResultMode ? TComputedResultModeMap[\"all\"]\n\t: TResultMode extends ResultModeUnion ? TComputedResultModeMap[TResultMode]\n\t: never;\n\ntype SuccessInfo = Pick<CallApiExtraOptions, \"resultMode\"> & {\n\tresponse: Response;\n};\n\ntype LazyResultModeMap = {\n\t[key in keyof ResultModeMap]: () => ResultModeMap[key];\n};\n\nconst getResultModeMap = (details: ResultModeMap[\"all\"]): LazyResultModeMap => {\n\treturn {\n\t\tall: () => details,\n\t\tonlyData: () => details.data,\n\t\tonlyResponse: () => details.response,\n\t\twithoutResponse: () => omitKeys(details, [\"response\"]),\n\t};\n};\n\ntype SuccessResult = CallApiResultSuccessVariant<unknown> | null;\n\n// The CallApiResult type is used to cast all return statements due to a design limitation in ts.\n// LINK - See https://www.zhenghao.io/posts/type-functions for more info\nexport const resolveSuccessResult = (data: unknown, info: SuccessInfo): SuccessResult => {\n\tconst { response, resultMode } = info;\n\n\tconst details = {\n\t\tdata,\n\t\terror: null,\n\t\tresponse,\n\t} satisfies CallApiResultSuccessVariant<unknown>;\n\n\tconst resultModeMap = getResultModeMap(details);\n\n\tconst successResult = resultModeMap[resultMode ?? \"all\"]();\n\n\treturn successResult as SuccessResult;\n};\n\nexport type ErrorInfo = Omit<SuccessInfo, \"response\">\n\t& Pick<CallApiExtraOptions, \"cloneResponse\"> & {\n\t\tmessage?: string;\n\t};\n\ntype ErrorResult = {\n\terrorDetails: CallApiResultErrorVariant<unknown>;\n\terrorResult: CallApiResultErrorVariant<unknown> | null;\n};\n\nexport const resolveErrorResult = (error: unknown, info: ErrorInfo): ErrorResult => {\n\tconst { cloneResponse, message: customErrorMessage, resultMode } = info;\n\n\tlet errorDetails = {\n\t\tdata: null,\n\t\terror: {\n\t\t\terrorData: false,\n\t\t\tmessage: customErrorMessage ?? (error as Error).message,\n\t\t\tname: (error as Error).name,\n\t\t\toriginalError: error as Error,\n\t\t},\n\t\tresponse: null,\n\t} satisfies CallApiResultErrorVariant<unknown> as CallApiResultErrorVariant<unknown>;\n\n\tif (isValidationErrorInstance(error)) {\n\t\tconst { errorData, message, response } = error;\n\n\t\terrorDetails = {\n\t\t\tdata: null,\n\t\t\terror: {\n\t\t\t\terrorData,\n\t\t\t\tissueCause: error.issueCause,\n\t\t\t\tmessage,\n\t\t\t\tname: \"ValidationError\",\n\t\t\t\toriginalError: error,\n\t\t\t},\n\t\t\tresponse,\n\t\t};\n\t}\n\n\tif (isHTTPErrorInstance<never>(error)) {\n\t\tconst { errorData, message, name, response } = error;\n\n\t\terrorDetails = {\n\t\t\tdata: null,\n\t\t\terror: { errorData, message, name, originalError: error },\n\t\t\tresponse: cloneResponse ? response.clone() : response,\n\t\t};\n\t}\n\n\tconst resultModeMap = getResultModeMap(errorDetails);\n\n\tconst errorResult = resultModeMap[resultMode ?? \"all\"]() as never;\n\n\treturn { errorDetails, errorResult };\n};\n\nexport const getCustomizedErrorResult = (\n\terrorResult: ErrorResult[\"errorResult\"],\n\tcustomErrorInfo: { message: string | undefined }\n): ErrorResult[\"errorResult\"] => {\n\tif (!errorResult) {\n\t\treturn null;\n\t}\n\n\tconst { message = errorResult.error.message } = customErrorInfo;\n\n\treturn {\n\t\t...errorResult,\n\t\terror: {\n\t\t\t...errorResult.error,\n\t\t\tmessage,\n\t\t} satisfies NonNullable<ErrorResult[\"errorResult\"]>[\"error\"] as never,\n\t};\n};\n","import {\n\ttype CallApiResultErrorVariant,\n\ttype CallApiResultSuccessVariant,\n\ttype ErrorInfo,\n\ttype PossibleHTTPError,\n\ttype PossibleJavaScriptError,\n\ttype PossibleJavaScriptOrValidationError,\n\ttype PossibleValidationError,\n\tresolveErrorResult,\n} from \"./result\";\nimport type { StreamProgressEvent } from \"./stream\";\nimport type {\n\tBaseCallApiConfig,\n\tCallApiConfig,\n\tCallApiContext,\n\tCallApiExtraOptionsForHooks,\n\tCallApiRequestOptionsForHooks,\n} from \"./types/common\";\nimport type { DefaultCallApiContext } from \"./types/default-types\";\nimport type { AnyFunction, Awaitable, Prettify, UnmaskType } from \"./types/type-helpers\";\n\nexport interface Hooks<\n\tTCallApiContext extends Pick<CallApiContext, \"Data\" | \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTData = TCallApiContext[\"Data\"],\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> {\n\t/**\n\t * Hook called when any error occurs within the request/response lifecycle.\n\t *\n\t * This is a unified error handler that catches both request errors (network failures,\n\t * timeouts, etc.) and response errors (HTTP error status codes). It's essentially\n\t * a combination of `onRequestError` and `onResponseError` hooks.\n\t *\n\t * @param context - Error context containing error details, request info, and response (if available)\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonError?: (context: ErrorContext<TCallApiContext, TErrorData>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called before the HTTP request is sent and before any internal processing of the request object begins.\n\t *\n\t * This is the ideal place to modify request headers, add authentication,\n\t * implement request logging, or perform any setup before the network call.\n\t *\n\t * @param context - Request context with mutable request object and configuration\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRequest?: (context: RequestContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when an error occurs during the fetch request itself.\n\t *\n\t * This handles network-level errors like connection failures, timeouts,\n\t * DNS resolution errors, or other issues that prevent getting an HTTP response.\n\t * Note that HTTP error status codes (4xx, 5xx) are handled by `onResponseError`.\n\t *\n\t * @param context - Request error context with error details and null response\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonRequestError?: (context: RequestErrorContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called just before the HTTP request is sent and after the request has been processed.\n\t *\n\t * @param context - Request context with mutable request object and configuration\n\t */\n\tonRequestReady?: (context: RequestContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called during upload stream progress tracking.\n\t *\n\t * This hook is triggered when uploading data (like file uploads) and provides\n\t * progress information about the upload. Useful for implementing progress bars\n\t * or upload status indicators.\n\t *\n\t * @param context - Request stream context with progress event and request instance\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRequestStream?: (context: RequestStreamContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when any HTTP response is received from the API.\n\t *\n\t * This hook is triggered for both successful (2xx) and error (4xx, 5xx) responses.\n\t * It's useful for response logging, metrics collection, or any processing that\n\t * should happen regardless of response status.\n\t *\n\t * @param context - Response context with either success data or error information\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonResponse?: (context: ResponseContext<TCallApiContext, TData, TErrorData>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when an HTTP error response (4xx, 5xx) is received from the API.\n\t *\n\t * This handles server-side errors where an HTTP response was successfully received\n\t * but indicates an error condition. Different from `onRequestError` which handles\n\t * network-level failures.\n\t *\n\t * @param context - Response error context with HTTP error details and response\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonResponseError?: (context: ResponseErrorContext<TCallApiContext, TErrorData>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called during download stream progress tracking.\n\t *\n\t * This hook is triggered when downloading data (like file downloads) and provides\n\t * progress information about the download. Useful for implementing progress bars\n\t * or download status indicators.\n\t *\n\t * @param context - Response stream context with progress event and response\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonResponseStream?: (context: ResponseStreamContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a request is being retried.\n\t *\n\t * This hook is triggered before each retry attempt, providing information about\n\t * the previous failure and the current retry attempt number. Useful for implementing\n\t * custom retry logic, exponential backoff, or retry logging.\n\t *\n\t * @param context - Retry context with error details and retry attempt count\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRetry?: (response: RetryContext<TCallApiContext, TErrorData>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a successful response (2xx status) is received from the API.\n\t *\n\t * This hook is triggered only for successful responses and provides access to\n\t * the parsed response data. Ideal for success logging, caching, or post-processing\n\t * of successful API responses.\n\t *\n\t * @param context - Success context with parsed response data and response object\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonSuccess?: (context: SuccessContext<TCallApiContext, TData>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a validation error occurs.\n\t *\n\t * This hook is triggered when request or response data fails validation against\n\t * a defined schema. It provides access to the validation error details and can\n\t * be used for custom error handling, logging, or fallback behavior.\n\t *\n\t * @param context - Validation error context with error details and response (if available)\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonValidationError?: (context: ValidationErrorContext<TCallApiContext>) => Awaitable<unknown>;\n}\n\nexport type HooksOrHooksArray<\n\tTCallApiContext extends CallApiContext = DefaultCallApiContext,\n\tTData = TCallApiContext[\"Data\"],\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> = {\n\t[Key in keyof Hooks<TCallApiContext, TData, TErrorData>]:\n\t\t| Hooks<TCallApiContext, TData, TErrorData>[Key]\n\t\t// eslint-disable-next-line perfectionist/sort-union-types -- I need arrays to be last\n\t\t| Array<Hooks<TCallApiContext, TData, TErrorData>[Key]>;\n};\n\nexport interface HookConfigOptions {\n\t/**\n\t * Controls the execution mode of all composed hooks (main + plugin hooks).\n\t *\n\t * - **\"parallel\"**: All hooks execute simultaneously via Promise.all() for better performance\n\t * - **\"sequential\"**: All hooks execute one by one in registration order via await in a loop\n\t *\n\t * This affects how ALL hooks execute together, regardless of their source (main or plugin).\n\t *\n\t * @default \"parallel\"\n\t */\n\thooksExecutionMode?: \"parallel\" | \"sequential\";\n}\n\nexport type RequestContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = {\n\t/**\n\t * Base configuration object passed to createFetchClient.\n\t *\n\t * Contains the foundational configuration that applies to all requests\n\t * made by this client instance, such as baseURL, default headers, and\n\t * global options.\n\t */\n\tbaseConfig: Exclude<BaseCallApiConfig, AnyFunction>;\n\n\t/**\n\t * Instance-specific configuration object passed to the callApi instance.\n\t *\n\t * Contains configuration specific to this particular API call, which\n\t * can override or extend the base configuration.\n\t */\n\tconfig: CallApiConfig;\n\n\t/**\n\t * Merged options combining base config, instance config, and default options.\n\t *\n\t * This is the final resolved configuration that will be used for the request,\n\t * with proper precedence applied (instance > base > defaults).\n\t */\n\toptions: CallApiExtraOptionsForHooks<TCallApiContext>;\n\n\t/**\n\t * Merged request object ready to be sent.\n\t *\n\t * Contains the final request configuration including URL, method, headers,\n\t * body, and other fetch options. This object can be modified in onRequest\n\t * hooks to customize the outgoing request.\n\t */\n\trequest: CallApiRequestOptionsForHooks;\n};\n\nexport type ValidationErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\terror: PossibleValidationError;\n\t\tresponse: Response | null;\n\t}\n>;\n\nexport type SuccessContext<\n\tTCallApiContext extends Pick<CallApiContext, \"Data\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTData = TCallApiContext[\"Data\"],\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tdata: NoInfer<TData>;\n\t\tresponse: Response;\n\t}\n>;\n\nexport type ResponseContext<\n\tTCallApiContext extends Pick<CallApiContext, \"Data\" | \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTData = TCallApiContext[\"Data\"],\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> = UnmaskType<\n\tRequestContext<TCallApiContext>\n\t\t& (\n\t\t\t| Prettify<CallApiResultSuccessVariant<TData>>\n\t\t\t| Prettify<\n\t\t\t\t\tExtract<CallApiResultErrorVariant<TErrorData>, { error: PossibleHTTPError<TErrorData> }>\n\t\t\t >\n\t\t)\n>;\n\nexport type RequestErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = RequestContext<TCallApiContext> & {\n\terror: PossibleJavaScriptError;\n\tresponse: null;\n};\n\nexport type ErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> = UnmaskType<\n\tRequestContext<TCallApiContext>\n\t\t& (\n\t\t\t| {\n\t\t\t\t\terror: PossibleHTTPError<TErrorData>;\n\t\t\t\t\tresponse: Response;\n\t\t\t }\n\t\t\t| {\n\t\t\t\t\terror: PossibleJavaScriptOrValidationError;\n\t\t\t\t\tresponse: Response | null;\n\t\t\t }\n\t\t)\n>;\n\nexport type ResponseErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> = UnmaskType<\n\tExtract<ErrorContext<TCallApiContext, TErrorData>, { error: PossibleHTTPError<TErrorData> }>\n\t\t& RequestContext<TCallApiContext>\n>;\n\nexport type RetryContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n\tTErrorData = TCallApiContext[\"ErrorData\"],\n> = UnmaskType<\n\tErrorContext<TCallApiContext, TErrorData> & {\n\t\tretryAttemptCount: number;\n\t}\n>;\n\nexport type RequestStreamContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tevent: StreamProgressEvent;\n\t\trequestInstance: Request;\n\t}\n>;\n\nexport type ResponseStreamContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tevent: StreamProgressEvent;\n\t\tresponse: Response;\n\t}\n>;\n\ntype HookRegistries = Required<{\n\t[Key in keyof Hooks]: Set<Hooks[Key]>;\n}>;\n\nexport const getHookRegistriesAndKeys = () => {\n\tconst hookRegistries: HookRegistries = {\n\t\tonError: new Set(),\n\t\tonRequest: new Set(),\n\t\tonRequestError: new Set(),\n\t\tonRequestReady: new Set(),\n\t\tonRequestStream: new Set(),\n\t\tonResponse: new Set(),\n\t\tonResponseError: new Set(),\n\t\tonResponseStream: new Set(),\n\t\tonRetry: new Set(),\n\t\tonSuccess: new Set(),\n\t\tonValidationError: new Set(),\n\t};\n\n\tconst hookRegistryKeys = Object.keys(hookRegistries) as Array<keyof Hooks>;\n\n\treturn { hookRegistries, hookRegistryKeys };\n};\n\nexport const composeHooksFromArray = (\n\thooksArray: Array<Hooks[keyof Hooks] | undefined>,\n\thooksExecutionMode: CallApiExtraOptionsForHooks[\"hooksExecutionMode\"]\n) => {\n\tconst composedHook = async (ctx: unknown) => {\n\t\tswitch (hooksExecutionMode) {\n\t\t\tcase \"parallel\": {\n\t\t\t\tawait Promise.all(hooksArray.map((uniqueHook) => uniqueHook?.(ctx as never)));\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tcase \"sequential\": {\n\t\t\t\tfor (const hook of hooksArray) {\n\t\t\t\t\t// eslint-disable-next-line no-await-in-loop -- This is necessary in this case\n\t\t\t\t\tawait hook?.(ctx as never);\n\t\t\t\t}\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tdefault: {\n\t\t\t\thooksExecutionMode satisfies undefined;\n\t\t\t}\n\t\t}\n\t};\n\n\treturn composedHook;\n};\n\nexport const executeHooks = async (...hookResultsOrPromise: Array<Awaitable<unknown>>) => {\n\tawait Promise.all(hookResultsOrPromise);\n};\n\nexport type ExecuteHookInfo = {\n\terrorInfo: ErrorInfo;\n\tshouldThrowOnError: boolean | undefined;\n};\n\nexport const executeHooksInCatchBlock = async (\n\thookResultsOrPromise: Array<Awaitable<unknown>>,\n\thookInfo: ExecuteHookInfo\n) => {\n\tconst { errorInfo, shouldThrowOnError } = hookInfo;\n\n\ttry {\n\t\tawait Promise.all(hookResultsOrPromise);\n\n\t\treturn null;\n\t} catch (hookError) {\n\t\tif (shouldThrowOnError) {\n\t\t\tthrow hookError;\n\t\t}\n\n\t\tconst { errorResult } = resolveErrorResult(hookError, errorInfo);\n\n\t\treturn errorResult;\n\t}\n};\n","import {\n\texecuteHooks,\n\ttype RequestContext,\n\ttype RequestStreamContext,\n\ttype ResponseStreamContext,\n} from \"./hooks\";\nimport { isObject, isReadableStream } from \"./utils/guards\";\n\nexport type StreamProgressEvent = {\n\t/**\n\t * Current chunk of data being streamed\n\t */\n\tchunk: Uint8Array;\n\t/**\n\t * Progress in percentage\n\t */\n\tprogress: number;\n\t/**\n\t * Total size of data in bytes\n\t */\n\ttotalBytes: number;\n\t/**\n\t * Amount of data transferred so far\n\t */\n\ttransferredBytes: number;\n};\n\ndeclare global {\n\tinterface ReadableStream<R> {\n\t\t[Symbol.asyncIterator]: () => AsyncIterableIterator<R>;\n\t}\n}\n\nconst createProgressEvent = (options: {\n\tchunk: Uint8Array;\n\ttotalBytes: number;\n\ttransferredBytes: number;\n}): StreamProgressEvent => {\n\tconst { chunk, totalBytes, transferredBytes } = options;\n\n\treturn {\n\t\tchunk,\n\t\tprogress: Math.round((transferredBytes / totalBytes) * 100) || 0,\n\t\ttotalBytes,\n\t\ttransferredBytes,\n\t};\n};\n\nconst calculateTotalBytesFromBody = async (\n\trequestBody: Request[\"body\"] | null,\n\texistingTotalBytes: number\n) => {\n\tlet totalBytes = existingTotalBytes;\n\n\tif (!requestBody) {\n\t\treturn totalBytes;\n\t}\n\n\tfor await (const chunk of requestBody) {\n\t\ttotalBytes += chunk.byteLength;\n\t}\n\n\treturn totalBytes;\n};\n\ntype ToStreamableRequestContext = RequestContext;\n\nexport const toStreamableRequest = async (\n\tcontext: ToStreamableRequestContext\n): Promise<Request | RequestInit> => {\n\tconst { baseConfig, config, options, request } = context;\n\n\tif (!options.onRequestStream || !isReadableStream(request.body)) {\n\t\treturn request as RequestInit;\n\t}\n\n\tconst requestInstance = new Request(\n\t\toptions.fullURL as NonNullable<typeof options.fullURL>,\n\t\t{ ...request, duplex: \"half\" } as RequestInit\n\t);\n\n\tconst contentLength = requestInstance.headers.get(\"content-length\");\n\n\tlet totalBytes = Number(contentLength ?? 0);\n\n\tconst shouldForcefullyCalcStreamSize =\n\t\tisObject(options.forcefullyCalculateStreamSize) ?\n\t\t\toptions.forcefullyCalculateStreamSize.request\n\t\t:\toptions.forcefullyCalculateStreamSize;\n\n\t// == If no content length is present, we read the total bytes from the body\n\tif (!contentLength && shouldForcefullyCalcStreamSize) {\n\t\ttotalBytes = await calculateTotalBytesFromBody(requestInstance.clone().body, totalBytes);\n\t}\n\n\tlet transferredBytes = 0;\n\n\tconst stream = new ReadableStream({\n\t\tstart: async (controller) => {\n\t\t\tconst body = requestInstance.body;\n\n\t\t\tif (!body) return;\n\n\t\t\tconst requestStreamContext = {\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tevent: createProgressEvent({ chunk: new Uint8Array(), totalBytes, transferredBytes }),\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t\trequestInstance,\n\t\t\t} satisfies RequestStreamContext;\n\n\t\t\tawait executeHooks(options.onRequestStream?.(requestStreamContext));\n\n\t\t\tfor await (const chunk of body) {\n\t\t\t\ttransferredBytes += chunk.byteLength;\n\n\t\t\t\ttotalBytes = Math.max(totalBytes, transferredBytes);\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onRequestStream?.({\n\t\t\t\t\t\t...requestStreamContext,\n\t\t\t\t\t\tevent: createProgressEvent({ chunk, totalBytes, transferredBytes }),\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\tcontroller.enqueue(chunk);\n\t\t\t}\n\n\t\t\tcontroller.close();\n\t\t},\n\t});\n\n\treturn new Request(requestInstance, { body: stream, duplex: \"half\" } as RequestInit);\n};\n\ntype StreamableResponseContext = RequestContext & { response: Response };\n\nexport const toStreamableResponse = async (context: StreamableResponseContext): Promise<Response> => {\n\tconst { baseConfig, config, options, request, response } = context;\n\n\tif (!options.onResponseStream || !response.body) {\n\t\treturn response;\n\t}\n\n\tconst contentLength = response.headers.get(\"content-length\");\n\n\tlet totalBytes = Number(contentLength ?? 0);\n\n\tconst shouldForceContentLengthCalc =\n\t\tisObject(options.forcefullyCalculateStreamSize) ?\n\t\t\toptions.forcefullyCalculateStreamSize.response\n\t\t:\toptions.forcefullyCalculateStreamSize;\n\n\t// == If no content length is present and `forceContentLengthCalculation` is enabled, we read the total bytes from the body\n\tif (!contentLength && shouldForceContentLengthCalc) {\n\t\ttotalBytes = await calculateTotalBytesFromBody(response.clone().body, totalBytes);\n\t}\n\n\tlet transferredBytes = 0;\n\n\tconst stream = new ReadableStream({\n\t\tstart: async (controller) => {\n\t\t\tconst body = response.body;\n\n\t\t\tif (!body) return;\n\n\t\t\tconst responseStreamContext = {\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tevent: createProgressEvent({ chunk: new Uint8Array(), totalBytes, transferredBytes }),\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t\tresponse,\n\t\t\t} satisfies ResponseStreamContext;\n\n\t\t\tawait executeHooks(options.onResponseStream?.(responseStreamContext));\n\n\t\t\tfor await (const chunk of body) {\n\t\t\t\ttransferredBytes += chunk.byteLength;\n\n\t\t\t\ttotalBytes = Math.max(totalBytes, transferredBytes);\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onResponseStream?.({\n\t\t\t\t\t\t...responseStreamContext,\n\t\t\t\t\t\tevent: createProgressEvent({ chunk, totalBytes, transferredBytes }),\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\tcontroller.enqueue(chunk);\n\t\t\t}\n\n\t\t\tcontroller.close();\n\t\t},\n\t});\n\n\treturn new Response(stream, response);\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport type { RequestContext } from \"./hooks\";\nimport { toStreamableRequest, toStreamableResponse } from \"./stream\";\nimport type { AnyString, RemovePrefix, UnmaskType } from \"./types/type-helpers\";\nimport { deterministicHashFn, waitFor } from \"./utils/common\";\nimport { isFunction } from \"./utils/guards\";\n\ntype RequestInfo = {\n\tcontroller: AbortController;\n\tresponsePromise: Promise<Response>;\n};\n\n/**\n * Cache that stores active request information for deduplication within a specific scope.\n *\n * Maps deduplication keys to their corresponding request information, including the abort controller\n * and response promise. A `null` key represents requests that don't participate in deduplication.\n *\n * **Internal Usage:**\n * This type is primarily used internally by the deduplication system. You typically won't need to\n * interact with it directly unless you're building custom deduplication logic or debugging.\n *\n * @example\n * ```ts\n * // This is handled internally, but conceptually:\n * const cache: RequestInfoCache = new Map([\n * [\"user-123\", { controller: abortController, responsePromise: fetchPromise }],\n * [\"config\", { controller: abortController2, responsePromise: fetchPromise2 }],\n * ]);\n * ```\n */\nexport type RequestInfoCache = Map<string | null, RequestInfo>;\n\n/**\n * Global cache that manages multiple request info caches, organized by scope keys.\n *\n * This enables the global deduplication feature by maintaining separate cache namespaces\n * for different scope keys. Each scope key gets its own `RequestInfoCache` instance.\n *\n * **Cache Lifecycle:**\n * - Caches are created on-demand when first accessed\n * - Automatic cleanup occurs when no references remain\n * - Each scope key maintains independent deduplication state\n *\n * **Memory Considerations:**\n * - Each scope key creates a separate cache instance\n * - Consider the number of different scope keys in your application\n * - Caches are cleaned up automatically when clients are garbage collected\n *\n * @example\n * ```ts\n * // This is managed internally, but conceptually:\n * const globalCache: GlobalRequestInfoCache = new Map([\n * [\"user-service\", new Map([...])], // Cache for user service requests\n * [\"analytics\", new Map([...])], // Cache for analytics requests\n * [\"default\", new Map([...])] // Default cache scope\n * ]);\n * ```\n */\nexport type GlobalRequestInfoCache = Map<DedupeOptions[\"dedupeCacheScopeKey\"], RequestInfoCache>;\n\ntype DedupeContext = RequestContext & {\n\t$GlobalRequestInfoCache: GlobalRequestInfoCache;\n\t$LocalRequestInfoCache: RequestInfoCache;\n\tnewFetchController: AbortController;\n};\n\nexport const createDedupeStrategy = async (context: DedupeContext) => {\n\tconst {\n\t\t$GlobalRequestInfoCache,\n\t\t$LocalRequestInfoCache,\n\t\tbaseConfig,\n\t\tconfig,\n\t\tnewFetchController,\n\t\toptions: globalOptions,\n\t\trequest: globalRequest,\n\t} = context;\n\n\tconst dedupeStrategy =\n\t\tglobalOptions.dedupeStrategy ?? globalOptions.dedupe?.strategy ?? extraOptionDefaults.dedupeStrategy;\n\n\tconst resolvedDedupeStrategy = isFunction(dedupeStrategy) ? dedupeStrategy(context) : dedupeStrategy;\n\n\tconst getDedupeKey = () => {\n\t\tconst shouldHaveDedupeKey =\n\t\t\tresolvedDedupeStrategy === \"cancel\" || resolvedDedupeStrategy === \"defer\";\n\n\t\tif (!shouldHaveDedupeKey) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst dedupeKey = globalOptions.dedupeKey ?? globalOptions.dedupe?.key;\n\n\t\tconst resolvedDedupeKey = isFunction(dedupeKey) ? dedupeKey(context) : dedupeKey;\n\n\t\tif (resolvedDedupeKey === undefined) {\n\t\t\tconst defaultDedupeKey = `${globalOptions.fullURL}-${deterministicHashFn({ options: globalOptions, request: globalRequest })}`;\n\n\t\t\treturn defaultDedupeKey;\n\t\t}\n\n\t\treturn resolvedDedupeKey;\n\t};\n\n\tconst getDedupeCacheScopeKey = () => {\n\t\tconst dedupeCacheScopeKey = globalOptions.dedupeCacheScopeKey ?? globalOptions.dedupe?.cacheScopeKey;\n\n\t\tconst resolvedDedupeCacheScopeKey =\n\t\t\tisFunction(dedupeCacheScopeKey) ? dedupeCacheScopeKey(context) : dedupeCacheScopeKey;\n\n\t\tif (resolvedDedupeCacheScopeKey === undefined) {\n\t\t\treturn extraOptionDefaults.dedupeCacheScopeKey;\n\t\t}\n\n\t\treturn resolvedDedupeCacheScopeKey;\n\t};\n\n\tconst dedupeKey = getDedupeKey();\n\n\tconst dedupeCacheScope =\n\t\tglobalOptions.dedupeCacheScope\n\t\t?? globalOptions.dedupe?.cacheScope\n\t\t?? extraOptionDefaults.dedupeCacheScope;\n\n\tconst dedupeCacheScopeKey = getDedupeCacheScopeKey();\n\n\tif (dedupeCacheScope === \"global\" && !$GlobalRequestInfoCache.has(dedupeCacheScopeKey)) {\n\t\t$GlobalRequestInfoCache.set(dedupeCacheScopeKey, new Map());\n\t}\n\n\tconst $RequestInfoCache =\n\t\tdedupeCacheScope === \"global\" ?\n\t\t\t$GlobalRequestInfoCache.get(dedupeCacheScopeKey)\n\t\t:\t$LocalRequestInfoCache;\n\n\t// == This is to ensure cache operations only occur when key is available\n\tconst $RequestInfoCacheOrNull = dedupeKey !== null ? $RequestInfoCache : null;\n\n\t/**\n\t * Force sequential execution of parallel requests to enable proper cache-based deduplication.\n\t *\n\t * Problem: When Promise.all([callApi(url), callApi(url)]) executes, both requests\n\t * start synchronously and reach this point before either can populate the cache.\n\t *\n\t * Why `await Promise.resolve()` fails:\n\t * - All microtasks in a batch resolve together at the next microtask checkpoint\n\t * - Both requests resume execution simultaneously after the await\n\t * - Both check `prevRequestInfo` at the same time → both see empty cache\n\t * - Both proceed to populate cache → deduplication fails\n\t *\n\t * Why `wait new Promise(()=> setTimeout(resolve, number))` works:\n\t * - Each setTimeout creates a separate task in the task queue\n\t * - Tasks execute sequentially, not simultaneously\n\t * - Request 1's task runs first: checks cache (empty) → continues → populates cache\n\t * - Request 2's task runs after: checks cache (populated) → uses cached promise\n\t * - Deduplication succeeds\n\t *\n\t * IMPORTANT: The delay must be non-zero. setTimeout(fn, 0) fails because JavaScript engines\n\t * may optimize zero-delay timers by batching them together, causing all requests to resume\n\t * simultaneously (same problem as microtasks). Any non-zero value (even 0.0000000001) forces\n\t * proper sequential task queue scheduling, ensuring each request gets its own task slot.\n\t */\n\tif (dedupeKey !== null) {\n\t\tawait waitFor(0.001);\n\t}\n\n\tconst prevRequestInfo = $RequestInfoCacheOrNull?.get(dedupeKey);\n\n\tconst getAbortErrorMessage = () => {\n\t\tif (globalOptions.dedupeKey) {\n\t\t\treturn `Duplicate request detected - Aborted previous request with key '${dedupeKey}'`;\n\t\t}\n\n\t\treturn `Duplicate request aborted - Aborted previous request to '${globalOptions.fullURL}'`;\n\t};\n\n\tconst handleRequestCancelStrategy = () => {\n\t\tconst shouldCancelRequest = prevRequestInfo && resolvedDedupeStrategy === \"cancel\";\n\n\t\tif (!shouldCancelRequest) return;\n\n\t\tconst message = getAbortErrorMessage();\n\n\t\tconst reason = new DOMException(message, \"AbortError\");\n\n\t\tprevRequestInfo.controller.abort(reason);\n\n\t\t// == Adding this just so that eslint forces me put await when calling the function (it looks better that way tbh)\n\t\treturn Promise.resolve();\n\t};\n\n\tconst handleRequestDeferStrategy = async (deferContext: {\n\t\tfetchApi: NonNullable<DedupeContext[\"options\"][\"customFetchImpl\"]>;\n\t\toptions: DedupeContext[\"options\"];\n\t\trequest: DedupeContext[\"request\"];\n\t}) => {\n\t\t// == Local options and request are needed so that transformations are applied can be applied to both from call site\n\t\tconst { fetchApi, options: localOptions, request: localRequest } = deferContext;\n\n\t\tconst shouldUsePromiseFromCache = prevRequestInfo && resolvedDedupeStrategy === \"defer\";\n\n\t\tconst streamableContext = {\n\t\t\tbaseConfig,\n\t\t\tconfig,\n\t\t\toptions: localOptions,\n\t\t\trequest: localRequest,\n\t\t} satisfies RequestContext;\n\n\t\tconst streamableRequest = await toStreamableRequest(streamableContext);\n\n\t\tconst responsePromise =\n\t\t\tshouldUsePromiseFromCache ?\n\t\t\t\tprevRequestInfo.responsePromise\n\t\t\t:\tfetchApi(localOptions.fullURL as NonNullable<typeof localOptions.fullURL>, streamableRequest);\n\n\t\t$RequestInfoCacheOrNull?.set(dedupeKey, { controller: newFetchController, responsePromise });\n\n\t\tconst response = await responsePromise;\n\n\t\treturn toStreamableResponse({ ...streamableContext, response });\n\t};\n\n\tconst removeDedupeKeyFromCache = () => {\n\t\t$RequestInfoCacheOrNull?.delete(dedupeKey);\n\t};\n\n\treturn {\n\t\tgetAbortErrorMessage,\n\t\thandleRequestCancelStrategy,\n\t\thandleRequestDeferStrategy,\n\t\tremoveDedupeKeyFromCache,\n\t\tresolvedDedupeStrategy,\n\t};\n};\n\ntype DedupeStrategyUnion = UnmaskType<\"cancel\" | \"defer\" | \"none\">;\n\ntype DedupeOptionKeys = Exclude<keyof DedupeOptions, \"dedupe\">;\n\ntype InnerDedupeOptions = {\n\t[Key in DedupeOptionKeys as RemovePrefix<\"dedupe\", Key>]?: DedupeOptions[Key];\n};\n\nexport type DedupeOptions = {\n\t/**\n\t * All dedupe options in a single object instead of separate properties\n\t */\n\tdedupe?: InnerDedupeOptions;\n\n\t/**\n\t * Controls the scope of request deduplication caching.\n\t *\n\t * - `\"global\"`: Shares deduplication cache across all `createFetchClient` instances with the same `dedupeCacheScopeKey`.\n\t * Useful for applications with multiple API clients that should share deduplication state.\n\t * - `\"local\"`: Limits deduplication to requests within the same `createFetchClient` instance.\n\t * Provides better isolation and is recommended for most use cases.\n\t *\n\t *\n\t * **Real-world Scenarios:**\n\t * - Use `\"global\"` when you have multiple API clients (user service, auth service, etc.) that might make overlapping requests\n\t * - Use `\"local\"` (default) for single-purpose clients or when you want strict isolation between different parts of your app\n\t *\n\t * @example\n\t * ```ts\n\t * // Local scope - each client has its own deduplication cache\n\t * const userClient = createFetchClient({ baseURL: \"/api/users\" });\n\t * const postClient = createFetchClient({ baseURL: \"/api/posts\" });\n\t * // These clients won't share deduplication state\n\t *\n\t * // Global scope - share cache across related clients\n\t * const userClient = createFetchClient({\n\t * baseURL: \"/api/users\",\n\t * dedupeCacheScope: \"global\",\n\t * });\n\t * const postClient = createFetchClient({\n\t * baseURL: \"/api/posts\",\n\t * dedupeCacheScope: \"global\",\n\t * });\n\t * // These clients will share deduplication state\n\t * ```\n\t *\n\t * @default \"local\"\n\t */\n\tdedupeCacheScope?: \"global\" | \"local\";\n\n\t/**\n\t * Unique namespace for the global deduplication cache when using `dedupeCacheScope: \"global\"`.\n\t *\n\t * This creates logical groupings of deduplication caches. All instances with the same key\n\t * will share the same cache namespace, allowing fine-grained control over which clients\n\t * share deduplication state.\n\t *\n\t * **Best Practices:**\n\t * - Use descriptive names that reflect the logical grouping (e.g., \"user-service\", \"analytics-api\")\n\t * - Keep scope keys consistent across related API clients\n\t * - Consider using different scope keys for different environments (dev, staging, prod)\n\t * - Avoid overly broad scope keys that might cause unintended cache sharing\n\t *\n\t * **Cache Management:**\n\t * - Each scope key maintains its own independent cache\n\t * - Caches are automatically cleaned up when no references remain\n\t * - Consider the memory implications of multiple global scopes\n\t *\n\t * @example\n\t * ```ts\n\t * // Group related API clients together\n\t * const userClient = createFetchClient({\n\t * baseURL: \"/api/users\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"user-service\"\n\t * });\n\t * const profileClient = createFetchClient({\n\t * baseURL: \"/api/profiles\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"user-service\" // Same scope - will share cache\n\t * });\n\t *\n\t * // Separate analytics client with its own cache\n\t * const analyticsClient = createFetchClient({\n\t * baseURL: \"/api/analytics\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"analytics-service\" // Different scope\n\t * });\n\t *\n\t * // Environment-specific scoping\n\t * const apiClient = createFetchClient({\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: `api-${process.env.NODE_ENV}` // \"api-development\", \"api-production\", etc.\n\t * });\n\t * ```\n\t *\n\t * @default \"default\"\n\t */\n\tdedupeCacheScopeKey?: \"default\" | AnyString | ((context: RequestContext) => string | undefined);\n\n\t/**\n\t * Custom key generator for request deduplication.\n\t *\n\t * Override the default key generation strategy to control exactly which requests\n\t * are considered duplicates. The default key combines URL, method, body, and\n\t * relevant headers (excluding volatile ones like 'Date', 'Authorization', etc.).\n\t *\n\t * **Default Key Generation:**\n\t * The auto-generated key includes:\n\t * - Full request URL (including query parameters)\n\t * - HTTP method (GET, POST, etc.)\n\t * - Request body (for POST/PUT/PATCH requests)\n\t * - Stable headers (excludes Date, Authorization, User-Agent, etc.)\n\t *\n\t * **Custom Key Best Practices:**\n\t * - Include only the parts of the request that should affect deduplication\n\t * - Avoid including volatile data (timestamps, random IDs, etc.)\n\t * - Consider performance - simpler keys are faster to compute and compare\n\t * - Ensure keys are deterministic for the same logical request\n\t * - Use consistent key formats across your application\n\t *\n\t * **Performance Considerations:**\n\t * - Function-based keys are computed on every request - keep them lightweight\n\t * - String keys are fastest but least flexible\n\t * - Consider caching expensive key computations if needed\n\t *\n\t * @example\n\t * ```ts\n\t * import { callApi } from \"@zayne-labs/callapi\";\n\t *\n\t * // Simple static key - useful for singleton requests\n\t * const config = callApi(\"/api/config\", {\n\t * dedupeKey: \"app-config\",\n\t * dedupeStrategy: \"defer\" // Share the same config across all requests\n\t * });\n\t *\n\t * // URL and method only - ignore headers and body\n\t * const userData = callApi(\"/api/user/123\", {\n\t * dedupeKey: (context) => `${context.options.method}:${context.options.fullURL}`\n\t * });\n\t *\n\t * // Include specific headers in deduplication\n\t * const apiCall = callApi(\"/api/data\", {\n\t * dedupeKey: (context) => {\n\t * const authHeader = context.request.headers.get(\"Authorization\");\n\t * return `${context.options.fullURL}-${authHeader}`;\n\t * }\n\t * });\n\t *\n\t * // User-specific deduplication\n\t * const userSpecificCall = callApi(\"/api/dashboard\", {\n\t * dedupeKey: (context) => {\n\t * const userId = context.options.fullURL.match(/user\\/(\\d+)/)?.[1];\n\t * return `dashboard-${userId}`;\n\t * }\n\t * });\n\t *\n\t * // Ignore certain query parameters\n\t * const searchCall = callApi(\"/api/search?q=test&timestamp=123456\", {\n\t * dedupeKey: (context) => {\n\t * const url = new URL(context.options.fullURL);\n\t * url.searchParams.delete(\"timestamp\"); // Remove volatile param\n\t * return `search:${url.toString()}`;\n\t * }\n\t * });\n\t * ```\n\t *\n\t * @default Auto-generated from request details\n\t */\n\tdedupeKey?: string | ((context: RequestContext) => string | undefined);\n\n\t/**\n\t * Strategy for handling duplicate requests. Can be a static string or callback function.\n\t *\n\t * **Available Strategies:**\n\t * - `\"cancel\"`: Cancel previous request when new one starts (good for search)\n\t * - `\"defer\"`: Share response between duplicate requests (good for config loading)\n\t * - `\"none\"`: No deduplication, all requests execute independently\n\t *\n\t * @example\n\t * ```ts\n\t * // Static strategies\n\t * const searchClient = createFetchClient({\n\t * dedupeStrategy: \"cancel\" // Cancel previous searches\n\t * });\n\t *\n\t * const configClient = createFetchClient({\n\t * dedupeStrategy: \"defer\" // Share config across components\n\t * });\n\t *\n\t * // Dynamic strategy based on request\n\t * const smartClient = createFetchClient({\n\t * dedupeStrategy: (context) => {\n\t * return context.options.method === \"GET\" ? \"defer\" : \"cancel\";\n\t * }\n\t * });\n\t *\n\t * // Search-as-you-type with cancel strategy\n\t * const handleSearch = async (query: string) => {\n\t * try {\n\t * const { data } = await callApi(\"/api/search\", {\n\t * method: \"POST\",\n\t * body: { query },\n\t * dedupeStrategy: \"cancel\",\n\t * dedupeKey: \"search\" // Cancel previous searches, only latest one goes through\n\t * });\n\t *\n\t * updateSearchResults(data);\n\t * } catch (error) {\n\t * if (error.name === \"AbortError\") {\n\t * // Previous search cancelled - (expected behavior)\n\t * return;\n\t * }\n\t * console.error(\"Search failed:\", error);\n\t * }\n\t * };\n\t *\n\t * ```\n\t *\n\t * @default \"cancel\"\n\t */\n\tdedupeStrategy?: DedupeStrategyUnion | ((context: RequestContext) => DedupeStrategyUnion);\n};\n","import type { RequestContext } from \"./hooks\";\nimport type { UnmaskType } from \"./types/type-helpers\";\n\nexport type FetchImpl = UnmaskType<\n\t(input: string | Request | URL, init?: RequestInit) => Promise<Response>\n>;\n\nexport interface Middlewares {\n\t/**\n\t * Wraps the fetch implementation to intercept requests at the network layer.\n\t *\n\t * Takes a context object containing the current fetch function and returns a new fetch function.\n\t * Use it to cache responses, add logging, handle offline mode, or short-circuit requests etc.\n\t * Multiple middleware compose in order: plugins → base config → per-request.\n\t *\n\t * Unlike `customFetchImpl`, middleware can call through to the original fetch.\n\t *\n\t * @example\n\t * ```ts\n\t * // Cache responses\n\t * const cache = new Map();\n\t * fetchMiddleware: (ctx) => async (input, init) => {\n\t * const key = input.toString();\n\t * if (cache.has(key)) return cache.get(key).clone();\n\t *\n\t * const response = await ctx.fetchImpl(input, init);\n\t * cache.set(key, response.clone());\n\t * return response;\n\t * }\n\t *\n\t * // Handle offline\n\t * fetchMiddleware: (ctx) => async (input, init) => {\n\t * if (!navigator.onLine) {\n\t * return new Response('{\"error\": \"offline\"}', { status: 503 });\n\t * }\n\t * return ctx.fetchImpl(input, init);\n\t * }\n\t * ```\n\t */\n\tfetchMiddleware?: (context: RequestContext & { fetchImpl: FetchImpl }) => FetchImpl;\n}\n\ntype MiddlewareRegistries = Required<{\n\t[Key in keyof Middlewares]: Set<Middlewares[Key]>;\n}>;\n\nexport const getMiddlewareRegistriesAndKeys = () => {\n\tconst middlewareRegistries: MiddlewareRegistries = {\n\t\tfetchMiddleware: new Set(),\n\t};\n\n\tconst middlewareRegistryKeys = Object.keys(middlewareRegistries) as Array<keyof Middlewares>;\n\n\treturn { middlewareRegistries, middlewareRegistryKeys };\n};\n\nexport const composeMiddlewaresFromArray = (\n\tmiddlewareArray: Array<Middlewares[keyof Middlewares] | undefined>\n) => {\n\tlet composedMiddleware: Middlewares[keyof Middlewares];\n\n\tfor (const currentMiddleware of middlewareArray) {\n\t\tif (!currentMiddleware) continue;\n\n\t\tconst previousMiddleware = composedMiddleware;\n\n\t\tif (!previousMiddleware) {\n\t\t\tcomposedMiddleware = currentMiddleware;\n\t\t\tcontinue;\n\t\t}\n\n\t\tcomposedMiddleware = (context) => {\n\t\t\tconst prevFetchImpl = previousMiddleware(context);\n\t\t\tconst fetchImpl = currentMiddleware({ ...context, fetchImpl: prevFetchImpl });\n\n\t\t\treturn fetchImpl;\n\t\t};\n\t}\n\n\treturn composedMiddleware;\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport {\n\tcomposeHooksFromArray,\n\tgetHookRegistriesAndKeys,\n\ttype Hooks,\n\ttype HooksOrHooksArray,\n\ttype RequestContext,\n} from \"./hooks\";\nimport {\n\tcomposeMiddlewaresFromArray,\n\tgetMiddlewareRegistriesAndKeys,\n\ttype Middlewares,\n} from \"./middlewares\";\nimport type { CallApiContext, CallApiRequestOptions, CallApiRequestOptionsForHooks } from \"./types/common\";\nimport type { DefaultCallApiContext } from \"./types/default-types\";\nimport type { Awaitable } from \"./types/type-helpers\";\nimport type { InitURLOrURLObject } from \"./url\";\nimport { isArray, isFunction, isPlainObject, isString } from \"./utils/guards\";\nimport { type BaseCallApiSchemaAndConfig, getCurrentRouteSchemaKeyAndMainInitURL } from \"./validation\";\n\nexport type PluginSetupContext<TCallApiContext extends CallApiContext = DefaultCallApiContext> =\n\tRequestContext<TCallApiContext> & { initURL: string };\n\nexport type PluginInitResult = Partial<\n\tOmit<PluginSetupContext, \"initURL\" | \"request\"> & {\n\t\tinitURL: InitURLOrURLObject;\n\t\trequest: CallApiRequestOptions;\n\t}\n>;\n\nexport type PluginHooks<\n\tTCallApiContext extends CallApiContext = DefaultCallApiContext,\n\tTData = unknown extends TCallApiContext[\"Data\"] ? never : TCallApiContext[\"Data\"],\n\tTErrorData = unknown extends TCallApiContext[\"ErrorData\"] ? never : TCallApiContext[\"ErrorData\"],\n> = HooksOrHooksArray<TCallApiContext, TData, TErrorData>;\n\nexport type PluginMiddlewares = Middlewares;\n\nexport interface CallApiPlugin {\n\t/**\n\t * Defines additional options that can be passed to callApi\n\t */\n\tdefineExtraOptions?: (...params: never[]) => unknown;\n\n\t/**\n\t * A description for the plugin\n\t */\n\tdescription?: string;\n\n\t/**\n\t * Hooks for the plugin\n\t */\n\thooks?: PluginHooks | ((context: PluginSetupContext) => Awaitable<PluginHooks>);\n\n\t/**\n\t * A unique id for the plugin\n\t */\n\tid: string;\n\n\t/**\n\t * Middlewares that for the plugin\n\t */\n\tmiddlewares?: Middlewares | ((context: PluginSetupContext) => Awaitable<Middlewares>);\n\n\t/**\n\t * A name for the plugin\n\t */\n\tname: string;\n\n\t/**\n\t * Base schema for the client.\n\t */\n\tschema?: BaseCallApiSchemaAndConfig;\n\n\t/**\n\t * A function that will be called when the plugin is initialized. This will be called before the any of the other internal functions.\n\t */\n\tsetup?: (context: PluginSetupContext) => Awaitable<PluginInitResult> | Awaitable<void>;\n\n\t/**\n\t * A version for the plugin\n\t */\n\tversion?: string;\n}\n\nexport const getResolvedPlugins = (context: Pick<RequestContext, \"baseConfig\" | \"options\">) => {\n\tconst { baseConfig, options } = context;\n\n\tconst resolvedPlugins =\n\t\tisFunction(options.plugins) ?\n\t\t\toptions.plugins({ basePlugins: baseConfig.plugins ?? [] })\n\t\t:\t(options.plugins ?? []);\n\n\treturn resolvedPlugins;\n};\n\nexport const initializePlugins = async (context: PluginSetupContext) => {\n\tconst { baseConfig, config, initURL, options, request } = context;\n\n\tconst {\n\t\taddMainHooks,\n\t\taddMainMiddlewares,\n\t\taddPluginHooks,\n\t\taddPluginMiddlewares,\n\t\tgetResolvedHooks,\n\t\tgetResolvedMiddlewares,\n\t} = setupHooksAndMiddlewares({ baseConfig, config, options });\n\n\tconst { currentRouteSchemaKey, mainInitURL } = getCurrentRouteSchemaKeyAndMainInitURL({\n\t\tbaseExtraOptions: baseConfig,\n\t\textraOptions: config,\n\t\tinitURL,\n\t});\n\n\tlet resolvedCurrentRouteSchemaKey = currentRouteSchemaKey;\n\tlet resolvedInitURL = mainInitURL;\n\tlet resolvedOptions = options;\n\tlet resolvedRequestOptions = request;\n\n\tconst executePluginSetupFn = async (pluginSetup: CallApiPlugin[\"setup\"]) => {\n\t\tif (!pluginSetup) return;\n\n\t\tconst initResult = await pluginSetup(context);\n\n\t\tif (!isPlainObject(initResult)) return;\n\n\t\tconst urlString = initResult.initURL?.toString();\n\n\t\tif (isString(urlString)) {\n\t\t\tconst newResult = getCurrentRouteSchemaKeyAndMainInitURL({\n\t\t\t\tbaseExtraOptions: baseConfig,\n\t\t\t\textraOptions: config,\n\t\t\t\tinitURL: urlString,\n\t\t\t});\n\n\t\t\tresolvedCurrentRouteSchemaKey = newResult.currentRouteSchemaKey;\n\t\t\tresolvedInitURL = newResult.mainInitURL;\n\t\t}\n\n\t\tif (isPlainObject(initResult.request)) {\n\t\t\tresolvedRequestOptions = {\n\t\t\t\t...resolvedRequestOptions,\n\t\t\t\t...(initResult.request as CallApiRequestOptionsForHooks),\n\t\t\t};\n\t\t}\n\n\t\tif (isPlainObject(initResult.options)) {\n\t\t\tresolvedOptions = { ...resolvedOptions, ...initResult.options };\n\t\t}\n\t};\n\n\tconst resolvedPlugins = getResolvedPlugins({ baseConfig, options });\n\n\tfor (const plugin of resolvedPlugins) {\n\t\t// eslint-disable-next-line no-await-in-loop -- Await is necessary in this case.\n\t\tconst [, pluginHooks, pluginMiddlewares] = await Promise.all([\n\t\t\texecutePluginSetupFn(plugin.setup),\n\t\t\tisFunction(plugin.hooks) ? plugin.hooks(context) : plugin.hooks,\n\t\t\tisFunction(plugin.middlewares) ? plugin.middlewares(context) : plugin.middlewares,\n\t\t]);\n\n\t\tpluginHooks && addPluginHooks(pluginHooks);\n\t\tpluginMiddlewares && addPluginMiddlewares(pluginMiddlewares);\n\t}\n\n\taddMainHooks();\n\n\taddMainMiddlewares();\n\n\tconst resolvedHooks = getResolvedHooks();\n\n\tconst resolvedMiddlewares = getResolvedMiddlewares();\n\n\treturn {\n\t\tresolvedCurrentRouteSchemaKey,\n\t\tresolvedHooks,\n\t\tresolvedInitURL,\n\t\tresolvedMiddlewares,\n\t\tresolvedOptions,\n\t\tresolvedRequestOptions,\n\t};\n};\n\nconst setupHooksAndMiddlewares = (\n\tcontext: Pick<PluginSetupContext, \"baseConfig\" | \"config\" | \"options\">\n) => {\n\tconst { baseConfig, config, options } = context;\n\n\tconst { hookRegistries, hookRegistryKeys } = getHookRegistriesAndKeys();\n\n\tconst { middlewareRegistries, middlewareRegistryKeys } = getMiddlewareRegistriesAndKeys();\n\n\tconst addMainHooks = () => {\n\t\tfor (const hookName of hookRegistryKeys) {\n\t\t\tconst overriddenHook = options[hookName];\n\t\t\tconst baseHook = baseConfig[hookName];\n\t\t\tconst instanceHook = config[hookName];\n\n\t\t\tconst shouldMergeBaseAndInstanceHooks = isArray(baseHook) && instanceHook;\n\n\t\t\tconst mainHook =\n\t\t\t\tshouldMergeBaseAndInstanceHooks ? [baseHook, instanceHook].flat() : overriddenHook;\n\n\t\t\tmainHook && hookRegistries[hookName].add(mainHook as never);\n\t\t}\n\t};\n\n\tconst addPluginHooks = (pluginHooks: PluginHooks) => {\n\t\tfor (const hookName of hookRegistryKeys) {\n\t\t\tconst pluginHook = pluginHooks[hookName];\n\n\t\t\tpluginHook && hookRegistries[hookName].add(pluginHook as never);\n\t\t}\n\t};\n\n\tconst addMainMiddlewares = () => {\n\t\tfor (const middlewareName of middlewareRegistryKeys) {\n\t\t\tconst baseMiddleware = baseConfig[middlewareName];\n\t\t\tconst instanceMiddleware = config[middlewareName];\n\n\t\t\tbaseMiddleware && middlewareRegistries[middlewareName].add(baseMiddleware);\n\n\t\t\tinstanceMiddleware && middlewareRegistries[middlewareName].add(instanceMiddleware);\n\t\t}\n\t};\n\n\tconst addPluginMiddlewares = (pluginMiddlewares: PluginMiddlewares) => {\n\t\tfor (const middlewareName of middlewareRegistryKeys) {\n\t\t\tconst pluginMiddleware = pluginMiddlewares[middlewareName];\n\n\t\t\tif (!pluginMiddleware) continue;\n\n\t\t\tmiddlewareRegistries[middlewareName].add(pluginMiddleware);\n\t\t}\n\t};\n\n\tconst getResolvedHooks = () => {\n\t\tconst resolvedHooks: Hooks = {};\n\n\t\tfor (const [hookName, hookRegistry] of Object.entries(hookRegistries)) {\n\t\t\tif (hookRegistry.size === 0) continue;\n\n\t\t\t// == Flatten the hook registry to remove any nested arrays, incase an array of hooks is passed to any of the hooks\n\t\t\tconst flattenedHookArray = [...hookRegistry].flat();\n\n\t\t\tif (flattenedHookArray.length === 0) continue;\n\n\t\t\tconst hooksExecutionMode = options.hooksExecutionMode ?? extraOptionDefaults.hooksExecutionMode;\n\n\t\t\tconst composedHook = composeHooksFromArray(flattenedHookArray, hooksExecutionMode);\n\n\t\t\tresolvedHooks[hookName as keyof Hooks] = composedHook;\n\t\t}\n\n\t\treturn resolvedHooks;\n\t};\n\n\tconst getResolvedMiddlewares = () => {\n\t\tconst resolvedMiddlewares: Middlewares = {};\n\n\t\tfor (const [middlewareName, middlewareRegistry] of Object.entries(middlewareRegistries)) {\n\t\t\tif (middlewareRegistry.size === 0) continue;\n\n\t\t\tconst middlewareArray = [...middlewareRegistry];\n\n\t\t\tif (middlewareArray.length === 0) continue;\n\n\t\t\tconst composedMiddleware = composeMiddlewaresFromArray(middlewareArray);\n\n\t\t\tresolvedMiddlewares[middlewareName as keyof Middlewares] = composedMiddleware;\n\t\t}\n\n\t\treturn resolvedMiddlewares;\n\t};\n\n\treturn {\n\t\taddMainHooks,\n\t\taddMainMiddlewares,\n\t\taddPluginHooks,\n\t\taddPluginMiddlewares,\n\t\tgetResolvedHooks,\n\t\tgetResolvedMiddlewares,\n\t};\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport {\n\texecuteHooksInCatchBlock,\n\ttype ErrorContext,\n\ttype ExecuteHookInfo,\n\ttype RequestContext,\n\ttype RetryContext,\n} from \"./hooks\";\nimport type { CallApiResultErrorVariant } from \"./result\";\nimport type { CallApiConfig } from \"./types/common\";\nimport type { MethodUnion } from \"./types/conditional-types\";\nimport {\n\tdefineEnum,\n\ttype AnyNumber,\n\ttype Awaitable,\n\ttype RemovePrefix,\n\ttype UnmaskType,\n} from \"./types/type-helpers\";\nimport type { InitURLOrURLObject } from \"./url\";\nimport { waitFor } from \"./utils/common\";\nimport { isBoolean, isFunction, isString } from \"./utils/guards\";\n\n// eslint-disable-next-line ts-eslint/no-unused-vars -- Ignore\nconst defaultRetryStatusCodesLookup = () =>\n\tdefineEnum({\n\t\t408: \"Request Timeout\",\n\t\t409: \"Conflict\",\n\t\t425: \"Too Early\",\n\t\t429: \"Too Many Requests\",\n\t\t500: \"Internal Server Error\",\n\t\t502: \"Bad Gateway\",\n\t\t503: \"Service Unavailable\",\n\t\t504: \"Gateway Timeout\",\n\t});\n\ntype RetryStatusCodes = UnmaskType<AnyNumber | keyof ReturnType<typeof defaultRetryStatusCodesLookup>>;\n\ntype RetryCondition<TErrorData> = (context: ErrorContext<{ ErrorData: TErrorData }>) => Awaitable<boolean>;\n\ntype RetryOptionKeys<TErrorData> = Exclude<keyof RetryOptions<TErrorData>, \"~retryAttemptCount\" | \"retry\">;\n\ntype InnerRetryOptions<TErrorData> = {\n\t[Key in RetryOptionKeys<TErrorData> as RemovePrefix<\"retry\", Key>]?: RetryOptions<TErrorData>[Key];\n};\n\nexport interface RetryOptions<TErrorData> {\n\t/**\n\t * Keeps track of the number of times the request has already been retried\n\t * @internal\n\t * @deprecated **NOTE**: This property is used internally to track retries. Please abstain from modifying it.\n\t */\n\treadonly [\"~retryAttemptCount\"]?: number;\n\n\t/**\n\t * All retry options in a single object instead of separate properties\n\t */\n\tretry?: InnerRetryOptions<TErrorData>;\n\n\t/**\n\t * Number of allowed retry attempts on HTTP errors\n\t * @default 0\n\t */\n\tretryAttempts?: number;\n\n\t/**\n\t * Callback whose return value determines if a request should be retried or not\n\t */\n\tretryCondition?: RetryCondition<TErrorData>;\n\n\t/**\n\t * Delay between retries in milliseconds\n\t * @default 1000\n\t */\n\tretryDelay?: number | ((currentAttemptCount: number) => number);\n\n\t/**\n\t * Maximum delay in milliseconds. Only applies to exponential strategy\n\t * @default 10000\n\t */\n\tretryMaxDelay?: number;\n\n\t/**\n\t * HTTP methods that are allowed to retry\n\t * @default [\"GET\", \"POST\"]\n\t */\n\tretryMethods?: MethodUnion[];\n\n\t/**\n\t * HTTP status codes that trigger a retry\n\t */\n\tretryStatusCodes?: RetryStatusCodes[];\n\n\t/**\n\t * Strategy to use when retrying\n\t * @default \"linear\"\n\t */\n\tretryStrategy?: \"exponential\" | \"linear\";\n}\n\nconst getLinearDelay = (currentAttemptCount: number, options: RetryOptions<unknown>) => {\n\tconst retryDelay = options.retryDelay ?? options.retry?.delay;\n\n\tconst resolveRetryDelay =\n\t\t(isFunction(retryDelay) ? retryDelay(currentAttemptCount) : retryDelay)\n\t\t?? extraOptionDefaults.retryDelay;\n\n\treturn resolveRetryDelay;\n};\n\nconst getExponentialDelay = (currentAttemptCount: number, options: RetryOptions<unknown>) => {\n\tconst retryDelay = options.retryDelay ?? options.retry?.delay ?? extraOptionDefaults.retryDelay;\n\n\tconst resolvedRetryDelay = isFunction(retryDelay) ? retryDelay(currentAttemptCount) : retryDelay;\n\n\tconst maxDelay = options.retryMaxDelay ?? options.retry?.maxDelay ?? extraOptionDefaults.retryMaxDelay;\n\n\tconst exponentialDelay = resolvedRetryDelay * 2 ** currentAttemptCount;\n\n\treturn Math.min(exponentialDelay, maxDelay);\n};\n\nexport const createRetryManager = (ctx: ErrorContext<{ ErrorData: unknown }> & RequestContext) => {\n\tconst { options, request } = ctx;\n\n\t// eslint-disable-next-line ts-eslint/no-deprecated -- Allowed for internal use\n\tconst currentAttemptCount = options[\"~retryAttemptCount\"] ?? 1;\n\n\tconst retryStrategy =\n\t\toptions.retryStrategy ?? options.retry?.strategy ?? extraOptionDefaults.retryStrategy;\n\n\tconst getDelay = () => {\n\t\tswitch (retryStrategy) {\n\t\t\tcase \"exponential\": {\n\t\t\t\treturn getExponentialDelay(currentAttemptCount, options);\n\t\t\t}\n\t\t\tcase \"linear\": {\n\t\t\t\treturn getLinearDelay(currentAttemptCount, options);\n\t\t\t}\n\t\t\tdefault: {\n\t\t\t\tthrow new Error(`Invalid retry strategy: ${String(retryStrategy)}`);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst shouldAttemptRetry = async () => {\n\t\tif (isBoolean(request.signal) && request.signal.aborted) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst retryCondition =\n\t\t\toptions.retryCondition ?? options.retry?.condition ?? extraOptionDefaults.retryCondition;\n\n\t\tconst maximumRetryAttempts =\n\t\t\toptions.retryAttempts ?? options.retry?.attempts ?? extraOptionDefaults.retryAttempts;\n\n\t\tconst customRetryCondition = await retryCondition(ctx);\n\n\t\tconst baseShouldRetry = currentAttemptCount <= maximumRetryAttempts && customRetryCondition;\n\n\t\tif (!baseShouldRetry) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst retryMethods = new Set(\n\t\t\toptions.retryMethods ?? options.retry?.methods ?? extraOptionDefaults.retryMethods\n\t\t);\n\n\t\tconst includesMethod =\n\t\t\tisString(ctx.request.method) && retryMethods.size > 0 ?\n\t\t\t\tretryMethods.has(ctx.request.method)\n\t\t\t:\ttrue;\n\n\t\tconst retryStatusCodes = new Set(\n\t\t\toptions.retryStatusCodes ?? options.retry?.statusCodes ?? extraOptionDefaults.retryStatusCodes\n\t\t);\n\n\t\tconst includesStatusCodes =\n\t\t\tctx.response != null && retryStatusCodes.size > 0 ?\n\t\t\t\tretryStatusCodes.has(ctx.response.status)\n\t\t\t:\ttrue;\n\n\t\tconst shouldRetry = includesMethod && includesStatusCodes;\n\n\t\treturn shouldRetry;\n\t};\n\n\ttype CallApiImpl = (\n\t\tinitURL: never,\n\t\tinit?: CallApiConfig\n\t) => Promise<CallApiResultErrorVariant<unknown>>;\n\n\tconst handleRetry = async <TCallapi extends CallApiImpl>(context: {\n\t\tcallApi: TCallapi;\n\t\tcallApiArgs: { config: CallApiConfig; initURL: InitURLOrURLObject };\n\t\terrorContext: ErrorContext<{ ErrorData: unknown }>;\n\t\thookInfo: ExecuteHookInfo;\n\t}) => {\n\t\tconst { callApi, callApiArgs, errorContext, hookInfo } = context;\n\n\t\tconst retryContext = {\n\t\t\t...errorContext,\n\t\t\tretryAttemptCount: currentAttemptCount,\n\t\t} satisfies RetryContext<{ ErrorData: unknown }>;\n\n\t\tconst hookError = await executeHooksInCatchBlock([options.onRetry?.(retryContext)], hookInfo);\n\n\t\tif (hookError) {\n\t\t\treturn hookError;\n\t\t}\n\n\t\tawait waitFor(getDelay());\n\n\t\tconst updatedOptions = {\n\t\t\t...callApiArgs.config,\n\t\t\t\"~retryAttemptCount\": currentAttemptCount + 1,\n\t\t} satisfies CallApiConfig;\n\n\t\treturn callApi(callApiArgs.initURL as never, updatedOptions);\n\t};\n\n\treturn {\n\t\thandleRetry,\n\t\tshouldAttemptRetry,\n\t};\n};\n","import { createDedupeStrategy, type GlobalRequestInfoCache, type RequestInfoCache } from \"./dedupe\";\nimport {\n\ttype ErrorContext,\n\ttype ExecuteHookInfo,\n\texecuteHooks,\n\texecuteHooksInCatchBlock,\n\ttype RequestContext,\n\ttype SuccessContext,\n} from \"./hooks\";\nimport { type CallApiPlugin, initializePlugins } from \"./plugins\";\nimport {\n\ttype ErrorInfo,\n\tgetCustomizedErrorResult,\n\ttype GetResponseType,\n\tresolveErrorResult,\n\tresolveResponseData,\n\tresolveSuccessResult,\n\ttype ResponseTypeType,\n\ttype ResultModeType,\n} from \"./result\";\nimport { createRetryManager } from \"./retry\";\nimport type {\n\tBaseCallApiConfig,\n\tCallApiConfig,\n\tCallApiContext,\n\tCallApiExtraOptionsForHooks,\n\tCallApiRequestOptions,\n\tCallApiRequestOptionsForHooks,\n\tCallApiResult,\n\tGetBaseSchemaConfig,\n\tGetBaseSchemaRoutes,\n} from \"./types/common\";\nimport type {\n\tGetCurrentRouteSchema,\n\tGetCurrentRouteSchemaKey,\n\tInferHeadersOption,\n\tInferInitURL,\n\tThrowOnErrorUnion,\n} from \"./types/conditional-types\";\nimport type {\n\tDefaultCallApiContext,\n\tDefaultPluginArray,\n\tDefaultThrowOnError,\n} from \"./types/default-types\";\nimport type { AnyFunction } from \"./types/type-helpers\";\nimport { getFullAndNormalizedURL } from \"./url\";\nimport {\n\tcreateCombinedSignal,\n\tcreateTimeoutSignal,\n\tgetBody,\n\tgetFetchImpl,\n\tgetHeaders,\n\tgetMethod,\n\tsplitBaseConfig,\n\tsplitConfig,\n} from \"./utils/common\";\nimport { HTTPError } from \"./utils/external/error\";\nimport { isHTTPErrorInstance, isValidationErrorInstance } from \"./utils/external/guards\";\nimport { isFunction } from \"./utils/guards\";\nimport {\n\ttype BaseCallApiSchemaAndConfig,\n\ttype BaseCallApiSchemaRoutes,\n\ttype CallApiSchema,\n\ttype CallApiSchemaConfig,\n\thandleConfigValidation,\n\thandleSchemaValidation,\n\ttype InferSchemaOutput,\n} from \"./validation\";\n\nconst $GlobalRequestInfoCache: GlobalRequestInfoCache = new Map();\n\nexport const createFetchClientWithContext = <\n\tTOuterCallApiContext extends CallApiContext = DefaultCallApiContext,\n>() => {\n\tconst createFetchClient = <\n\t\tTBaseCallApiContext extends CallApiContext = TOuterCallApiContext,\n\t\tTBaseData = TBaseCallApiContext[\"Data\"],\n\t\tTBaseErrorData = TBaseCallApiContext[\"ErrorData\"],\n\t\tTBaseResultMode extends ResultModeType = TBaseCallApiContext[\"ResultMode\"] extends ResultModeType ?\n\t\t\tTBaseCallApiContext[\"ResultMode\"]\n\t\t:\tDefaultCallApiContext[\"ResultMode\"],\n\t\tTBaseThrowOnError extends ThrowOnErrorUnion = DefaultThrowOnError,\n\t\tTBaseResponseType extends ResponseTypeType = ResponseTypeType,\n\t\tconst TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig,\n\t\tconst TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray,\n\t\tTComputedBaseSchemaConfig extends CallApiSchemaConfig = GetBaseSchemaConfig<TBaseSchemaAndConfig>,\n\t\tTComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes =\n\t\t\tGetBaseSchemaRoutes<TBaseSchemaAndConfig>,\n\t>(\n\t\tinitBaseConfig: BaseCallApiConfig<\n\t\t\tTBaseCallApiContext,\n\t\t\tTBaseData,\n\t\t\tTBaseErrorData,\n\t\t\tTBaseResultMode,\n\t\t\tTBaseThrowOnError,\n\t\t\tTBaseResponseType,\n\t\t\tTBaseSchemaAndConfig,\n\t\t\tTBasePluginArray\n\t\t> = {} as never\n\t) => {\n\t\tconst $LocalRequestInfoCache: RequestInfoCache = new Map();\n\n\t\tconst callApi = async <\n\t\t\tTData = TBaseData,\n\t\t\tTErrorData = TBaseErrorData,\n\t\t\tTResultMode extends ResultModeType = TBaseResultMode,\n\t\t\tTCallApiContext extends CallApiContext = TBaseCallApiContext,\n\t\t\tTThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError,\n\t\t\tTResponseType extends ResponseTypeType = TBaseResponseType,\n\t\t\tconst TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig,\n\t\t\tTInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTSchemaConfig\n\t\t\t>,\n\t\t\tTCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> =\n\t\t\t\tGetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>,\n\t\t\tconst TSchema extends CallApiSchema = GetCurrentRouteSchema<\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTCurrentRouteSchemaKey\n\t\t\t>,\n\t\t\tconst TPluginArray extends CallApiPlugin[] = TBasePluginArray,\n\t\t\tTComputedResult = CallApiResult<\n\t\t\t\tInferSchemaOutput<TSchema[\"data\"], TData>,\n\t\t\t\tInferSchemaOutput<TSchema[\"errorData\"], TErrorData>,\n\t\t\t\tTResultMode,\n\t\t\t\tTThrowOnError,\n\t\t\t\tTResponseType\n\t\t\t>,\n\t\t>(\n\t\t\tinitURL: TInitURL,\n\t\t\tinitConfig: CallApiConfig<\n\t\t\t\tTCallApiContext,\n\t\t\t\tInferSchemaOutput<TSchema[\"data\"], GetResponseType<TData, TResponseType>>,\n\t\t\t\tInferSchemaOutput<TSchema[\"errorData\"], GetResponseType<TErrorData, TResponseType>>,\n\t\t\t\tTResultMode,\n\t\t\t\tTThrowOnError,\n\t\t\t\tTResponseType,\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTSchema,\n\t\t\t\tTComputedBaseSchemaConfig,\n\t\t\t\tTSchemaConfig,\n\t\t\t\tTInitURL,\n\t\t\t\tTCurrentRouteSchemaKey,\n\t\t\t\tTBasePluginArray,\n\t\t\t\tTPluginArray\n\t\t\t> = {} as never\n\t\t): Promise<TComputedResult> => {\n\t\t\tconst [fetchOptions, extraOptions] = splitConfig(initConfig);\n\n\t\t\tconst resolvedBaseConfig =\n\t\t\t\tisFunction(initBaseConfig) ?\n\t\t\t\t\tinitBaseConfig({\n\t\t\t\t\t\tinitURL: initURL.toString(),\n\t\t\t\t\t\toptions: extraOptions,\n\t\t\t\t\t\trequest: fetchOptions,\n\t\t\t\t\t})\n\t\t\t\t:\tinitBaseConfig;\n\n\t\t\tconst baseConfig = resolvedBaseConfig as BaseCallApiConfig;\n\t\t\tconst config = initConfig as CallApiConfig;\n\n\t\t\tconst [baseFetchOptions, baseExtraOptions] = splitBaseConfig(baseConfig);\n\n\t\t\tconst shouldSkipAutoMergeForOptions =\n\t\t\t\tbaseExtraOptions.skipAutoMergeFor === \"all\" || baseExtraOptions.skipAutoMergeFor === \"options\";\n\n\t\t\tconst shouldSkipAutoMergeForRequest =\n\t\t\t\tbaseExtraOptions.skipAutoMergeFor === \"all\" || baseExtraOptions.skipAutoMergeFor === \"request\";\n\n\t\t\t// == Merged Extra Options\n\t\t\tconst mergedExtraOptions = {\n\t\t\t\t...baseExtraOptions,\n\t\t\t\t...(!shouldSkipAutoMergeForOptions && extraOptions),\n\t\t\t};\n\n\t\t\t// == Merged Request Options\n\t\t\tconst mergedRequestOptions = {\n\t\t\t\theaders: {}, // == Making sure headers is always an object\n\t\t\t\t...baseFetchOptions,\n\t\t\t\t...(!shouldSkipAutoMergeForRequest && fetchOptions),\n\t\t\t} satisfies CallApiRequestOptions;\n\n\t\t\tconst {\n\t\t\t\tresolvedCurrentRouteSchemaKey,\n\t\t\t\tresolvedHooks,\n\t\t\t\tresolvedInitURL,\n\t\t\t\tresolvedMiddlewares,\n\t\t\t\tresolvedOptions,\n\t\t\t\tresolvedRequestOptions,\n\t\t\t} = await initializePlugins({\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tinitURL: initURL.toString(),\n\t\t\t\toptions: mergedExtraOptions as CallApiExtraOptionsForHooks,\n\t\t\t\trequest: mergedRequestOptions as CallApiRequestOptionsForHooks,\n\t\t\t});\n\n\t\t\tconst { fullURL, normalizedInitURL } = getFullAndNormalizedURL({\n\t\t\t\tbaseURL: resolvedOptions.baseURL,\n\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\tparams: resolvedOptions.params,\n\t\t\t\tquery: resolvedOptions.query,\n\t\t\t});\n\n\t\t\tconst options = {\n\t\t\t\t...resolvedOptions,\n\t\t\t\t...resolvedHooks,\n\t\t\t\t...resolvedMiddlewares,\n\n\t\t\t\tfullURL,\n\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\tinitURLNormalized: normalizedInitURL,\n\t\t\t} satisfies CallApiExtraOptionsForHooks;\n\n\t\t\tconst newFetchController = new AbortController();\n\n\t\t\tconst timeoutSignal = createTimeoutSignal(options.timeout);\n\n\t\t\tconst combinedSignal = createCombinedSignal(\n\t\t\t\ttimeoutSignal,\n\t\t\t\tresolvedRequestOptions.signal,\n\t\t\t\tnewFetchController.signal\n\t\t\t);\n\n\t\t\tconst initMethod = getMethod({ initURL: resolvedInitURL, method: resolvedRequestOptions.method });\n\n\t\t\tconst request = {\n\t\t\t\t...resolvedRequestOptions,\n\n\t\t\t\tmethod: initMethod,\n\t\t\t\tsignal: combinedSignal,\n\t\t\t} satisfies CallApiRequestOptionsForHooks;\n\n\t\t\tconst {\n\t\t\t\tgetAbortErrorMessage,\n\t\t\t\thandleRequestCancelStrategy,\n\t\t\t\thandleRequestDeferStrategy,\n\t\t\t\tremoveDedupeKeyFromCache,\n\t\t\t\tresolvedDedupeStrategy,\n\t\t\t} = await createDedupeStrategy({\n\t\t\t\t$GlobalRequestInfoCache,\n\t\t\t\t$LocalRequestInfoCache,\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tnewFetchController,\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tawait handleRequestCancelStrategy();\n\n\t\t\t\tawait executeHooks(options.onRequest?.({ baseConfig, config, options, request }));\n\n\t\t\t\tconst {\n\t\t\t\t\textraOptionsValidationResult,\n\t\t\t\t\trequestOptionsValidationResult,\n\t\t\t\t\tresolvedSchema,\n\t\t\t\t\tresolvedSchemaConfig,\n\t\t\t\t\tshouldApplySchemaOutput,\n\t\t\t\t} = await handleConfigValidation({\n\t\t\t\t\tbaseExtraOptions,\n\t\t\t\t\tcurrentRouteSchemaKey: resolvedCurrentRouteSchemaKey,\n\t\t\t\t\textraOptions,\n\t\t\t\t\toptions,\n\t\t\t\t\trequestOptions: request,\n\t\t\t\t});\n\n\t\t\t\t// == Apply Schema Output for Extra Options\n\t\t\t\tif (shouldApplySchemaOutput) {\n\t\t\t\t\tObject.assign(options, extraOptionsValidationResult);\n\t\t\t\t}\n\n\t\t\t\t// == Apply Schema Output for Request Options\n\t\t\t\tconst validMethod = getMethod({\n\t\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\t\tmethod: shouldApplySchemaOutput ? requestOptionsValidationResult?.method : request.method,\n\t\t\t\t});\n\n\t\t\t\tconst validBody = getBody({\n\t\t\t\t\tbody: shouldApplySchemaOutput ? requestOptionsValidationResult?.body : request.body,\n\t\t\t\t\tbodySerializer: options.bodySerializer,\n\t\t\t\t});\n\n\t\t\t\ttype HeaderFn = Extract<InferHeadersOption<CallApiSchema>[\"headers\"], AnyFunction>;\n\n\t\t\t\tconst resolvedHeaders =\n\t\t\t\t\tisFunction<HeaderFn>(fetchOptions.headers) ?\n\t\t\t\t\t\tfetchOptions.headers({ baseHeaders: baseFetchOptions.headers ?? {} })\n\t\t\t\t\t:\t(fetchOptions.headers ?? baseFetchOptions.headers);\n\n\t\t\t\tconst validHeaders = await getHeaders({\n\t\t\t\t\tauth: options.auth,\n\t\t\t\t\tbody: validBody,\n\t\t\t\t\theaders:\n\t\t\t\t\t\tshouldApplySchemaOutput ? requestOptionsValidationResult?.headers : resolvedHeaders,\n\t\t\t\t});\n\n\t\t\t\tObject.assign(request, {\n\t\t\t\t\t...(validBody && { body: validBody }),\n\t\t\t\t\t...(validHeaders && { headers: validHeaders }),\n\t\t\t\t\t...(validMethod && { method: validMethod }),\n\t\t\t\t});\n\n\t\t\t\tconst readyRequestContext = { baseConfig, config, options, request } satisfies RequestContext;\n\n\t\t\t\tawait executeHooks(options.onRequestReady?.(readyRequestContext));\n\n\t\t\t\tconst fetchApi = getFetchImpl({\n\t\t\t\t\tcustomFetchImpl: options.customFetchImpl,\n\t\t\t\t\tfetchMiddleware: options.fetchMiddleware,\n\t\t\t\t\trequestContext: readyRequestContext,\n\t\t\t\t});\n\n\t\t\t\tconst response = await handleRequestDeferStrategy({ fetchApi, options, request });\n\n\t\t\t\t// == Also clone response when dedupeStrategy is set to \"defer\" to avoid error thrown from reading response.(whatever) more than once\n\t\t\t\tconst shouldCloneResponse = resolvedDedupeStrategy === \"defer\" || options.cloneResponse;\n\n\t\t\t\tif (!response.ok) {\n\t\t\t\t\tconst errorData = await resolveResponseData(\n\t\t\t\t\t\tshouldCloneResponse ? response.clone() : response,\n\t\t\t\t\t\toptions.responseType,\n\t\t\t\t\t\toptions.responseParser\n\t\t\t\t\t);\n\n\t\t\t\t\tconst validErrorData = await handleSchemaValidation(resolvedSchema, \"errorData\", {\n\t\t\t\t\t\tinputValue: errorData,\n\t\t\t\t\t\tresponse,\n\t\t\t\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t\t\t\t});\n\n\t\t\t\t\t// == Push all error handling responsibilities to the catch block if not retrying\n\t\t\t\t\tthrow new HTTPError(\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tdefaultHTTPErrorMessage: options.defaultHTTPErrorMessage,\n\t\t\t\t\t\t\terrorData: validErrorData,\n\t\t\t\t\t\t\tresponse,\n\t\t\t\t\t\t},\n\t\t\t\t\t\t{ cause: validErrorData }\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\tconst successData = await resolveResponseData(\n\t\t\t\t\tshouldCloneResponse ? response.clone() : response,\n\t\t\t\t\toptions.responseType,\n\t\t\t\t\toptions.responseParser\n\t\t\t\t);\n\n\t\t\t\tconst validSuccessData = await handleSchemaValidation(resolvedSchema, \"data\", {\n\t\t\t\t\tinputValue: successData,\n\t\t\t\t\tresponse,\n\t\t\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t\t\t});\n\n\t\t\t\tconst successContext = {\n\t\t\t\t\tbaseConfig,\n\t\t\t\t\tconfig,\n\t\t\t\t\tdata: validSuccessData,\n\t\t\t\t\toptions,\n\t\t\t\t\trequest,\n\t\t\t\t\tresponse,\n\t\t\t\t} satisfies SuccessContext;\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onSuccess?.(successContext),\n\n\t\t\t\t\toptions.onResponse?.({ ...successContext, error: null })\n\t\t\t\t);\n\n\t\t\t\tconst successResult = resolveSuccessResult(successContext.data, {\n\t\t\t\t\tresponse: successContext.response,\n\t\t\t\t\tresultMode: options.resultMode,\n\t\t\t\t});\n\n\t\t\t\treturn successResult as never;\n\n\t\t\t\t// == Exhaustive Error handling\n\t\t\t} catch (error) {\n\t\t\t\tconst errorInfo = {\n\t\t\t\t\tcloneResponse: options.cloneResponse,\n\t\t\t\t\tresultMode: options.resultMode,\n\t\t\t\t} satisfies ErrorInfo;\n\n\t\t\t\tconst { errorDetails, errorResult } = resolveErrorResult(error, errorInfo);\n\n\t\t\t\tconst errorContext = {\n\t\t\t\t\tbaseConfig,\n\t\t\t\t\tconfig,\n\t\t\t\t\terror: errorDetails.error as never,\n\t\t\t\t\toptions,\n\t\t\t\t\trequest,\n\t\t\t\t\tresponse: errorDetails.response as never,\n\t\t\t\t} satisfies ErrorContext<{ ErrorData: unknown }>;\n\n\t\t\t\tconst shouldThrowOnError = Boolean(\n\t\t\t\t\tisFunction(options.throwOnError) ? options.throwOnError(errorContext) : options.throwOnError\n\t\t\t\t);\n\n\t\t\t\tconst hookInfo = {\n\t\t\t\t\terrorInfo,\n\t\t\t\t\tshouldThrowOnError,\n\t\t\t\t} satisfies ExecuteHookInfo;\n\n\t\t\t\tconst { handleRetry, shouldAttemptRetry } = createRetryManager(errorContext);\n\n\t\t\t\tconst handleRetryOrGetErrorResult = async () => {\n\t\t\t\t\tconst shouldRetry = await shouldAttemptRetry();\n\n\t\t\t\t\tif (shouldRetry) {\n\t\t\t\t\t\treturn handleRetry({\n\t\t\t\t\t\t\tcallApi,\n\t\t\t\t\t\t\tcallApiArgs: { config, initURL },\n\t\t\t\t\t\t\terrorContext,\n\t\t\t\t\t\t\thookInfo,\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\n\t\t\t\t\tif (shouldThrowOnError) {\n\t\t\t\t\t\tthrow error;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn errorResult;\n\t\t\t\t};\n\n\t\t\t\tif (isValidationErrorInstance(error)) {\n\t\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t\t[options.onValidationError?.(errorContext), options.onError?.(errorContext)],\n\t\t\t\t\t\thookInfo\n\t\t\t\t\t);\n\n\t\t\t\t\treturn (hookError ?? (await handleRetryOrGetErrorResult())) as never;\n\t\t\t\t}\n\n\t\t\t\tif (isHTTPErrorInstance<TErrorData>(error)) {\n\t\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t\t[\n\t\t\t\t\t\t\toptions.onResponseError?.(errorContext),\n\t\t\t\t\t\t\toptions.onError?.(errorContext),\n\t\t\t\t\t\t\toptions.onResponse?.({ ...errorContext, data: null }),\n\t\t\t\t\t\t],\n\t\t\t\t\t\thookInfo\n\t\t\t\t\t);\n\n\t\t\t\t\treturn (hookError ?? (await handleRetryOrGetErrorResult())) as never;\n\t\t\t\t}\n\n\t\t\t\tlet message = (error as Error | undefined)?.message;\n\n\t\t\t\tif (error instanceof DOMException && error.name === \"AbortError\") {\n\t\t\t\t\tmessage = getAbortErrorMessage();\n\n\t\t\t\t\t!shouldThrowOnError && console.error(`${error.name}:`, message);\n\t\t\t\t}\n\n\t\t\t\tif (error instanceof DOMException && error.name === \"TimeoutError\") {\n\t\t\t\t\tmessage = `Request timed out after ${options.timeout}ms`;\n\n\t\t\t\t\t!shouldThrowOnError && console.error(`${error.name}:`, message);\n\t\t\t\t}\n\n\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t[options.onRequestError?.(errorContext), options.onError?.(errorContext)],\n\t\t\t\t\thookInfo\n\t\t\t\t);\n\n\t\t\t\treturn (hookError\n\t\t\t\t\t?? getCustomizedErrorResult(await handleRetryOrGetErrorResult(), { message })) as never;\n\n\t\t\t\t// == Removing the now unneeded AbortController from store\n\t\t\t} finally {\n\t\t\t\tremoveDedupeKeyFromCache();\n\t\t\t}\n\t\t};\n\n\t\treturn callApi;\n\t};\n\n\treturn createFetchClient;\n};\n\nexport const createFetchClient = createFetchClientWithContext();\n\nexport const callApi = createFetchClient();\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["validatedResultObject: Prettify<\n\t\tPick<CallApiExtraOptions, (typeof extraOptionsToBeValidated)[number]>\n\t>","validatedResultObject: Prettify<\n\t\tPick<CallApiRequestOptions, (typeof requestOptionsToBeValidated)[number]>\n\t>","headersObject: Record<string, string | undefined>","result: Record<string, unknown>","hookRegistries: HookRegistries","dedupeKey","dedupeCacheScopeKey","$GlobalRequestInfoCache","middlewareRegistries: MiddlewareRegistries","composedMiddleware: Middlewares[keyof Middlewares]","resolvedHooks: Hooks","resolvedMiddlewares: Middlewares","callApi","$GlobalRequestInfoCache: GlobalRequestInfoCache","createFetchClient","$LocalRequestInfoCache: RequestInfoCache","callApi"],"sources":["../src/auth.ts","../src/validation.ts","../src/url.ts","../src/utils/polyfills/combinedSignal.ts","../src/utils/polyfills/timeoutSignal.ts","../src/utils/common.ts","../src/result.ts","../src/hooks.ts","../src/stream.ts","../src/dedupe.ts","../src/middlewares.ts","../src/plugins.ts","../src/retry.ts","../src/createFetchClient.ts"],"sourcesContent":["/* eslint-disable perfectionist/sort-object-types -- Avoid Sorting for now */\n\nimport type { CallApiExtraOptions } from \"./types/common\";\nimport type { Awaitable } from \"./types/type-helpers\";\nimport { isFunction, isObject, isPromise } from \"./utils/guards\";\n\ntype PossibleAuthValue = Awaitable<string | null | undefined>;\n\ntype PossibleAuthValueOrGetter = PossibleAuthValue | (() => PossibleAuthValue);\n\nexport type BearerOrTokenAuth =\n\t| {\n\t\t\ttype?: \"Bearer\";\n\t\t\tbearer?: PossibleAuthValueOrGetter;\n\t\t\ttoken?: never;\n\t }\n\t| {\n\t\t\ttype?: \"Token\";\n\t\t\tbearer?: never;\n\t\t\ttoken?: PossibleAuthValueOrGetter;\n\t };\n\nexport type BasicAuth = {\n\ttype: \"Basic\";\n\tusername: PossibleAuthValueOrGetter;\n\tpassword: PossibleAuthValueOrGetter;\n};\n\n/**\n * Custom auth\n *\n * @param prefix - prefix of the header\n * @param authValue - value of the header\n *\n * @example\n * ```ts\n * {\n * type: \"Custom\",\n * prefix: \"Token\",\n * authValue: \"token\"\n * }\n * ```\n */\nexport type CustomAuth = {\n\ttype: \"Custom\";\n\tprefix: PossibleAuthValueOrGetter;\n\tvalue: PossibleAuthValueOrGetter;\n};\n\n// eslint-disable-next-line perfectionist/sort-union-types -- Let the first one be first\nexport type Auth = PossibleAuthValueOrGetter | BearerOrTokenAuth | BasicAuth | CustomAuth;\n\nconst resolveAuthValue = (value: PossibleAuthValueOrGetter) => (isFunction(value) ? value() : value);\n\ntype AuthHeaderObject = { Authorization: string };\n\nexport const getAuthHeader = async (\n\tauth: CallApiExtraOptions[\"auth\"]\n): Promise<AuthHeaderObject | undefined> => {\n\tif (auth === undefined) return;\n\n\tif (isPromise(auth) || isFunction(auth) || !isObject(auth)) {\n\t\tconst authValue = await resolveAuthValue(auth);\n\n\t\tif (authValue === undefined) return;\n\n\t\treturn {\n\t\t\tAuthorization: `Bearer ${authValue}`,\n\t\t};\n\t}\n\n\tswitch (auth.type) {\n\t\tcase \"Basic\": {\n\t\t\tconst [username, password] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.username),\n\t\t\t\tresolveAuthValue(auth.password),\n\t\t\t]);\n\n\t\t\tif (username === undefined || password === undefined) return;\n\n\t\t\treturn {\n\t\t\t\tAuthorization: `Basic ${globalThis.btoa(`${username}:${password}`)}`,\n\t\t\t};\n\t\t}\n\n\t\tcase \"Custom\": {\n\t\t\tconst [prefix, value] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.prefix),\n\t\t\t\tresolveAuthValue(auth.value),\n\t\t\t]);\n\n\t\t\tif (value === undefined) return;\n\n\t\t\treturn {\n\t\t\t\tAuthorization: `${prefix} ${value}`,\n\t\t\t};\n\t\t}\n\n\t\tdefault: {\n\t\t\tconst [bearer, token] = await Promise.all([\n\t\t\t\tresolveAuthValue(auth.bearer),\n\t\t\t\tresolveAuthValue(auth.token),\n\t\t\t]);\n\n\t\t\tif (bearer !== undefined) {\n\t\t\t\treturn { Authorization: `Bearer ${bearer}` };\n\t\t\t}\n\n\t\t\tif (token === undefined) return;\n\n\t\t\treturn { Authorization: `Token ${token}` };\n\t\t}\n\t}\n};\n","import { fallBackRouteSchemaKey, type FallBackRouteSchemaKey } from \"./constants/validation\";\nimport type {\n\tBaseCallApiExtraOptions,\n\tCallApiExtraOptions,\n\tCallApiRequestOptions,\n\tGlobalMeta,\n} from \"./types/common\";\nimport type { Body, HeadersOption, MethodUnion } from \"./types/conditional-types\";\nimport type { StandardSchemaV1 } from \"./types/standard-schema\";\nimport {\n\ttype AnyFunction,\n\ttype AnyString,\n\ttype Awaitable,\n\tdefineEnum,\n\ttype Prettify,\n\ttype UnionToIntersection,\n} from \"./types/type-helpers\";\nimport type { Params, Query } from \"./url\";\nimport { toArray } from \"./utils/common\";\nimport { ValidationError } from \"./utils/external/error\";\nimport { isFunction } from \"./utils/guards\";\n\ntype ResultVariant = \"infer-input\" | \"infer-output\";\n\nexport type InferSchemaResult<TSchema, TFallbackResult, TResultVariant extends ResultVariant> =\n\t// == Checking for undefined first and returning fallback to avoid type errors when passing the config around (weird tbh)\n\tundefined extends TSchema ? TFallbackResult\n\t: TSchema extends StandardSchemaV1 ?\n\t\tTResultVariant extends \"infer-input\" ?\n\t\t\tStandardSchemaV1.InferInput<TSchema>\n\t\t:\tStandardSchemaV1.InferOutput<TSchema>\n\t: TSchema extends AnyFunction<infer TResult> ? Awaited<TResult>\n\t: TFallbackResult;\n\nexport type InferSchemaOutput<TSchema, TFallbackResult = unknown> = InferSchemaResult<\n\tTSchema,\n\tTFallbackResult,\n\t\"infer-output\"\n>;\n\nexport type InferSchemaInput<TSchema, TFallbackResult = unknown> = InferSchemaResult<\n\tTSchema,\n\tTFallbackResult,\n\t\"infer-input\"\n>;\n\nconst handleValidatorFunction = async <TInput>(\n\tvalidator: Extract<CallApiSchema[keyof CallApiSchema], AnyFunction>,\n\tinputData: TInput\n): Promise<StandardSchemaV1.Result<TInput>> => {\n\ttry {\n\t\tconst result = await validator(inputData as never);\n\n\t\treturn { issues: undefined, value: result as never };\n\t} catch (error) {\n\t\treturn { issues: toArray(error) as never, value: undefined };\n\t}\n};\n\nexport const standardSchemaParser = async <\n\tTFullSchema extends CallApiSchema,\n\tTSchemaName extends keyof CallApiSchema,\n\tTSchema extends NonNullable<TFullSchema[TSchemaName]>,\n>(\n\tfullSchema: TFullSchema | undefined,\n\tschemaName: TSchemaName,\n\toptions: { inputValue: InferSchemaInput<TSchema>; response?: Response | null }\n): Promise<InferSchemaOutput<TSchema>> => {\n\tconst { inputValue, response } = options;\n\n\tconst schema = fullSchema?.[schemaName];\n\n\tif (!schema) {\n\t\treturn inputValue as never;\n\t}\n\n\tconst result =\n\t\tisFunction(schema) ?\n\t\t\tawait handleValidatorFunction(schema, inputValue)\n\t\t:\tawait schema[\"~standard\"].validate(inputValue);\n\n\t// == If the `issues` field exists, it means the validation failed\n\n\tif (result.issues) {\n\t\tthrow new ValidationError({\n\t\t\tissueCause: schemaName,\n\t\t\tissues: result.issues,\n\t\t\tresponse: response ?? null,\n\t\t});\n\t}\n\n\treturn result.value as never;\n};\n\nexport interface CallApiSchemaConfig {\n\t/**\n\t * The base url of the schema. By default it's the baseURL of the callApi instance.\n\t */\n\tbaseURL?: string;\n\n\t/**\n\t * Disables runtime validation for the schema.\n\t */\n\tdisableRuntimeValidation?: boolean;\n\n\t/**\n\t * If `true`, the original input value will be used instead of the transformed/validated output.\n\t *\n\t * This is useful when you want to validate the input but don't want any transformations\n\t * applied by the validation schema (e.g., type coercion, default values, etc).\n\t */\n\tdisableValidationOutputApplication?: boolean;\n\n\t/**\n\t * Optional url prefix that will be substituted for the `baseURL` of the schemaConfig at runtime.\n\t *\n\t * This allows you to reuse the same schema against different base URLs (for example,\n\t * swapping between `/api/v1` and `/api/v2`) without redefining the entire schema.\n\t */\n\tprefix?: string;\n\n\t/**\n\t * Controls the strictness of API route validation.\n\t *\n\t * When true:\n\t * - Only routes explicitly defined in the schema will be considered valid to typescript and the runtime.\n\t * - Attempting to call routes not defined in the schema will result in both type errors and runtime validation errors.\n\t * - Useful for ensuring API calls conform exactly to your schema definition\n\t *\n\t * When false or undefined (default):\n\t * - All routes will be allowed, whether they are defined in the schema or not\n\t */\n\tstrict?: boolean;\n}\n\nexport interface CallApiSchema {\n\t/**\n\t * The schema to use for validating the request body.\n\t */\n\tbody?: StandardSchemaV1<Body | undefined> | ((body: Body) => Awaitable<Body | undefined>);\n\n\t/**\n\t * The schema to use for validating the response data.\n\t */\n\tdata?: StandardSchemaV1 | ((data: unknown) => unknown);\n\n\t/**\n\t * The schema to use for validating the response error data.\n\t */\n\terrorData?: StandardSchemaV1 | ((errorData: unknown) => unknown);\n\n\t/**\n\t * The schema to use for validating the request headers.\n\t */\n\theaders?:\n\t\t| StandardSchemaV1<HeadersOption | undefined>\n\t\t| ((headers: HeadersOption) => Awaitable<HeadersOption | undefined>);\n\n\t/**\n\t * The schema to use for validating the meta option.\n\t */\n\tmeta?:\n\t\t| StandardSchemaV1<GlobalMeta | undefined>\n\t\t| ((meta: GlobalMeta) => Awaitable<GlobalMeta | undefined>);\n\n\t/**\n\t * The schema to use for validating the request method.\n\t */\n\tmethod?:\n\t\t| StandardSchemaV1<MethodUnion | undefined>\n\t\t| ((method: MethodUnion) => Awaitable<MethodUnion | undefined>);\n\n\t/**\n\t * The schema to use for validating the request url parameters.\n\t */\n\tparams?: StandardSchemaV1<Params | undefined> | ((params: Params) => Awaitable<Params | undefined>);\n\n\t/**\n\t * The schema to use for validating the request url queries.\n\t */\n\tquery?: StandardSchemaV1<Query | undefined> | ((query: Query) => Awaitable<Query | undefined>);\n}\n\nexport const routeKeyMethods = defineEnum([\"delete\", \"get\", \"patch\", \"post\", \"put\"]);\n\nexport type RouteKeyMethods = (typeof routeKeyMethods)[number];\n\nexport type RouteKeyMethodsURLUnion = `@${RouteKeyMethods}/`;\n\nexport type BaseSchemaRouteKeyPrefixes = FallBackRouteSchemaKey | RouteKeyMethodsURLUnion;\n\nexport type BaseCallApiSchemaRoutes = Partial<\n\tRecord<AnyString | BaseSchemaRouteKeyPrefixes, CallApiSchema>\n>;\n\nexport type BaseCallApiSchemaAndConfig = {\n\tconfig?: CallApiSchemaConfig;\n\troutes: BaseCallApiSchemaRoutes;\n};\n\ntype ValidationOptions<\n\tTSchema extends CallApiSchema[keyof CallApiSchema] = CallApiSchema[keyof CallApiSchema],\n> = {\n\tinputValue: InferSchemaInput<TSchema>;\n\tresponse?: Response | null;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nexport const handleSchemaValidation = async <\n\tTFullSchema extends CallApiSchema,\n\tTSchemaName extends keyof CallApiSchema,\n\tTSchema extends NonNullable<TFullSchema[TSchemaName]>,\n>(\n\tfullSchema: TFullSchema | undefined,\n\tschemaName: TSchemaName,\n\tvalidationOptions: ValidationOptions<TSchema>\n): Promise<InferSchemaOutput<TSchema>> => {\n\tconst { inputValue, response, schemaConfig } = validationOptions;\n\n\tif (schemaConfig?.disableRuntimeValidation) {\n\t\treturn inputValue as never;\n\t}\n\n\tconst validResult = await standardSchemaParser(fullSchema, schemaName, { inputValue, response });\n\n\treturn validResult as never;\n};\n\ntype LastOf<TValue> =\n\tUnionToIntersection<TValue extends unknown ? () => TValue : never> extends () => infer R ? R : never;\n\ntype Push<TArray extends unknown[], TArrayItem> = [...TArray, TArrayItem];\n\ntype UnionToTuple<\n\tTUnion,\n\tTComputedLastUnion = LastOf<TUnion>,\n\tTComputedIsUnionEqualToNever = [TUnion] extends [never] ? true : false,\n> =\n\ttrue extends TComputedIsUnionEqualToNever ? []\n\t:\tPush<UnionToTuple<Exclude<TUnion, TComputedLastUnion>>, TComputedLastUnion>;\n\nexport type Tuple<TTuple, TArray extends TTuple[] = []> =\n\tUnionToTuple<TTuple>[\"length\"] extends TArray[\"length\"] ? [...TArray]\n\t:\tTuple<TTuple, [TTuple, ...TArray]>;\n\nconst extraOptionsToBeValidated = [\"meta\", \"params\", \"query\"] satisfies Tuple<\n\tExtract<keyof CallApiSchema, keyof CallApiExtraOptions>\n>;\n\ntype ExtraOptionsValidationOptions = {\n\toptions: CallApiExtraOptions;\n\tschema: CallApiSchema | undefined;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nconst handleExtraOptionsValidation = async (validationOptions: ExtraOptionsValidationOptions) => {\n\tconst { options, schema, schemaConfig } = validationOptions;\n\n\tconst validationResultArray = await Promise.all(\n\t\textraOptionsToBeValidated.map((schemaName) =>\n\t\t\thandleSchemaValidation(schema, schemaName, {\n\t\t\t\tinputValue: options[schemaName],\n\t\t\t\tschemaConfig,\n\t\t\t})\n\t\t)\n\t);\n\n\tconst validatedResultObject: Prettify<\n\t\tPick<CallApiExtraOptions, (typeof extraOptionsToBeValidated)[number]>\n\t> = {};\n\n\tfor (const [index, schemaName] of extraOptionsToBeValidated.entries()) {\n\t\tconst validationResult = validationResultArray[index];\n\n\t\tif (validationResult === undefined) continue;\n\n\t\tvalidatedResultObject[schemaName] = validationResult as never;\n\t}\n\n\treturn validatedResultObject;\n};\n\nconst requestOptionsToBeValidated = [\"body\", \"headers\", \"method\"] satisfies Tuple<\n\tExtract<keyof CallApiSchema, keyof CallApiRequestOptions>\n>;\n\ntype RequestOptionsValidationOptions = {\n\trequestOptions: CallApiRequestOptions;\n\tschema: CallApiSchema | undefined;\n\tschemaConfig: CallApiSchemaConfig | undefined;\n};\n\nconst handleRequestOptionsValidation = async (validationOptions: RequestOptionsValidationOptions) => {\n\tconst { requestOptions, schema, schemaConfig } = validationOptions;\n\n\tconst validationResultArray = await Promise.all(\n\t\trequestOptionsToBeValidated.map((schemaName) =>\n\t\t\thandleSchemaValidation(schema, schemaName, {\n\t\t\t\tinputValue: requestOptions[schemaName],\n\t\t\t\tschemaConfig,\n\t\t\t})\n\t\t)\n\t);\n\n\tconst validatedResultObject: Prettify<\n\t\tPick<CallApiRequestOptions, (typeof requestOptionsToBeValidated)[number]>\n\t> = {};\n\n\tfor (const [index, propertyKey] of requestOptionsToBeValidated.entries()) {\n\t\tconst validationResult = validationResultArray[index];\n\n\t\tif (validationResult === undefined) continue;\n\n\t\tvalidatedResultObject[propertyKey] = validationResult as never;\n\t}\n\n\treturn validatedResultObject;\n};\n\nexport const handleConfigValidation = async (\n\tvalidationOptions: GetResolvedSchemaContext\n\t\t& Omit<ExtraOptionsValidationOptions & RequestOptionsValidationOptions, \"schema\" | \"schemaConfig\">\n) => {\n\tconst { baseExtraOptions, currentRouteSchemaKey, extraOptions, options, requestOptions } =\n\t\tvalidationOptions;\n\n\tconst { currentRouteSchema, resolvedSchema } = getResolvedSchema({\n\t\tbaseExtraOptions,\n\t\tcurrentRouteSchemaKey,\n\t\textraOptions,\n\t});\n\n\tconst resolvedSchemaConfig = getResolvedSchemaConfig({ baseExtraOptions, extraOptions });\n\n\tif (resolvedSchemaConfig?.strict === true && !currentRouteSchema) {\n\t\tthrow new ValidationError({\n\t\t\tissueCause: \"schemaConfig-(strict)\",\n\t\t\tissues: [{ message: `Strict Mode - No schema found for route '${currentRouteSchemaKey}' ` }],\n\t\t\tresponse: null,\n\t\t});\n\t}\n\n\tif (resolvedSchemaConfig?.disableRuntimeValidation) {\n\t\treturn {\n\t\t\textraOptionsValidationResult: null,\n\t\t\trequestOptionsValidationResult: null,\n\t\t\tresolvedSchema,\n\t\t\tresolvedSchemaConfig,\n\t\t\tshouldApplySchemaOutput: false,\n\t\t};\n\t}\n\n\tconst [extraOptionsValidationResult, requestOptionsValidationResult] = await Promise.all([\n\t\thandleExtraOptionsValidation({\n\t\t\toptions,\n\t\t\tschema: resolvedSchema,\n\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t}),\n\t\thandleRequestOptionsValidation({\n\t\t\trequestOptions,\n\t\t\tschema: resolvedSchema,\n\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t}),\n\t]);\n\n\tconst shouldApplySchemaOutput =\n\t\t(Boolean(extraOptionsValidationResult) || Boolean(requestOptionsValidationResult))\n\t\t&& !resolvedSchemaConfig?.disableValidationOutputApplication;\n\n\treturn {\n\t\textraOptionsValidationResult,\n\t\trequestOptionsValidationResult,\n\t\tresolvedSchema,\n\t\tresolvedSchemaConfig,\n\t\tshouldApplySchemaOutput,\n\t};\n};\n\ntype GetResolvedSchemaContext = {\n\tbaseExtraOptions: BaseCallApiExtraOptions;\n\tcurrentRouteSchemaKey: string;\n\textraOptions: CallApiExtraOptions;\n};\n\nexport const getResolvedSchema = (context: GetResolvedSchemaContext) => {\n\tconst { baseExtraOptions, currentRouteSchemaKey, extraOptions } = context;\n\n\tconst fallbackRouteSchema = baseExtraOptions.schema?.routes[fallBackRouteSchemaKey];\n\tconst currentRouteSchema = baseExtraOptions.schema?.routes[currentRouteSchemaKey];\n\n\tconst resolvedRouteSchema = {\n\t\t...fallbackRouteSchema,\n\t\t// == Current route schema takes precedence over fallback route schema\n\t\t...currentRouteSchema,\n\t} satisfies CallApiSchema as CallApiSchema | undefined;\n\n\tconst resolvedSchema =\n\t\tisFunction(extraOptions.schema) ?\n\t\t\textraOptions.schema({\n\t\t\t\tbaseSchemaRoutes: baseExtraOptions.schema?.routes ?? {},\n\t\t\t\tcurrentRouteSchema: resolvedRouteSchema ?? {},\n\t\t\t})\n\t\t:\t(extraOptions.schema ?? resolvedRouteSchema);\n\n\treturn { currentRouteSchema, resolvedSchema };\n};\n\nexport const getResolvedSchemaConfig = (\n\tcontext: Omit<GetResolvedSchemaContext, \"currentRouteSchemaKey\">\n) => {\n\tconst { baseExtraOptions, extraOptions } = context;\n\n\tconst resolvedSchemaConfig =\n\t\tisFunction(extraOptions.schemaConfig) ?\n\t\t\textraOptions.schemaConfig({ baseSchemaConfig: baseExtraOptions.schema?.config ?? {} })\n\t\t:\t(extraOptions.schemaConfig ?? baseExtraOptions.schema?.config);\n\n\treturn resolvedSchemaConfig;\n};\n\nexport const getCurrentRouteSchemaKeyAndMainInitURL = (\n\tcontext: Pick<GetResolvedSchemaContext, \"baseExtraOptions\" | \"extraOptions\"> & { initURL: string }\n) => {\n\tconst { baseExtraOptions, extraOptions, initURL } = context;\n\n\tconst schemaConfig = getResolvedSchemaConfig({ baseExtraOptions, extraOptions });\n\n\tlet currentRouteSchemaKey = initURL;\n\tlet mainInitURL = initURL;\n\n\tif (schemaConfig?.prefix && currentRouteSchemaKey.startsWith(schemaConfig.prefix)) {\n\t\tcurrentRouteSchemaKey = currentRouteSchemaKey.replace(schemaConfig.prefix, \"\");\n\n\t\tmainInitURL = mainInitURL.replace(schemaConfig.prefix, schemaConfig.baseURL ?? \"\");\n\t}\n\n\tif (schemaConfig?.baseURL && currentRouteSchemaKey.startsWith(schemaConfig.baseURL)) {\n\t\tcurrentRouteSchemaKey = currentRouteSchemaKey.replace(schemaConfig.baseURL, \"\");\n\t}\n\n\treturn { currentRouteSchemaKey, mainInitURL };\n};\n","import type { CallApiExtraOptions } from \"./types/common\";\nimport type { AnyString, UnmaskType } from \"./types/type-helpers\";\nimport { toQueryString } from \"./utils/external/body\";\nimport { isArray } from \"./utils/guards\";\nimport { type RouteKeyMethodsURLUnion, routeKeyMethods } from \"./validation\";\n\nconst slash = \"/\";\nconst colon = \":\";\nconst openBrace = \"{\";\nconst closeBrace = \"}\";\n\nconst mergeUrlWithParams = (url: string, params: CallApiExtraOptions[\"params\"]) => {\n\tif (!params) {\n\t\treturn url;\n\t}\n\n\tlet newUrl = url;\n\n\tif (isArray(params)) {\n\t\tconst urlParts = newUrl.split(slash);\n\n\t\t// == Find all parameters in order (both :param and {param} patterns)\n\t\tconst matchedParamsArray = urlParts.filter(\n\t\t\t(part) => part.startsWith(colon) || (part.startsWith(openBrace) && part.endsWith(closeBrace))\n\t\t);\n\n\t\tfor (const [paramIndex, matchedParam] of matchedParamsArray.entries()) {\n\t\t\tconst stringParamValue = String(params[paramIndex]);\n\t\t\tnewUrl = newUrl.replace(matchedParam, stringParamValue);\n\t\t}\n\n\t\treturn newUrl;\n\t}\n\n\t// == Handle object params - replace both :param and {param} patterns\n\tfor (const [paramKey, paramValue] of Object.entries(params)) {\n\t\tconst colonPattern = `${colon}${paramKey}` as const;\n\t\tconst bracePattern = `${openBrace}${paramKey}${closeBrace}` as const;\n\t\tconst stringValue = String(paramValue);\n\n\t\tnewUrl = newUrl.replace(colonPattern, stringValue);\n\t\tnewUrl = newUrl.replace(bracePattern, stringValue);\n\t}\n\n\treturn newUrl;\n};\n\nconst questionMark = \"?\";\nconst ampersand = \"&\";\nconst mergeUrlWithQuery = (url: string, query: CallApiExtraOptions[\"query\"]): string => {\n\tif (!query) {\n\t\treturn url;\n\t}\n\n\tconst queryString = toQueryString(query);\n\n\tif (queryString?.length === 0) {\n\t\treturn url;\n\t}\n\n\tif (url.endsWith(questionMark)) {\n\t\treturn `${url}${queryString}`;\n\t}\n\n\tif (url.includes(questionMark)) {\n\t\treturn `${url}${ampersand}${queryString}`;\n\t}\n\n\treturn `${url}${questionMark}${queryString}`;\n};\n\n/**\n * @description Extracts the HTTP method from method-prefixed route patterns.\n *\n * Analyzes URLs that start with method modifiers (e.g., \"@get/\", \"@post/\") and extracts\n * the HTTP method for use in API requests. This enables method specification directly\n * in route definitions.\n *\n * @param initURL - The URL string to analyze for method modifiers\n * @returns The extracted HTTP method (lowercase) if found, otherwise undefined\n *\n * @example\n * ```typescript\n * // Method extraction from prefixed routes\n * extractMethodFromURL(\"@get/users\"); // Returns: \"get\"\n * extractMethodFromURL(\"@post/users\"); // Returns: \"post\"\n * extractMethodFromURL(\"@put/users/:id\"); // Returns: \"put\"\n * extractMethodFromURL(\"@delete/users/:id\"); // Returns: \"delete\"\n * extractMethodFromURL(\"@patch/users/:id\"); // Returns: \"patch\"\n *\n * // No method modifier\n * extractMethodFromURL(\"/users\"); // Returns: undefined\n * extractMethodFromURL(\"users\"); // Returns: undefined\n *\n * // Invalid or unsupported methods\n * extractMethodFromURL(\"@invalid/users\"); // Returns: undefined\n * extractMethodFromURL(\"@/users\"); // Returns: undefined\n *\n * // Edge cases\n * extractMethodFromURL(undefined); // Returns: undefined\n * extractMethodFromURL(\"\"); // Returns: undefined\n * ```\n */\nexport const extractMethodFromURL = (initURL: string | undefined) => {\n\tif (!initURL?.startsWith(\"@\")) return;\n\n\tconst method = initURL.split(\"@\")[1]?.split(\"/\")[0];\n\n\tif (!method || !routeKeyMethods.includes(method as (typeof routeKeyMethods)[number])) return;\n\n\treturn method;\n};\n\nconst normalizeURL = (initURL: string) => {\n\tconst methodFromURL = extractMethodFromURL(initURL);\n\n\tif (!methodFromURL) {\n\t\treturn initURL;\n\t}\n\n\tconst normalizedURL = initURL.replace(`@${methodFromURL}/`, \"/\");\n\n\treturn normalizedURL;\n};\n\ntype GetFullURLOptions = {\n\t/** Base URL to prepend to relative URLs */\n\tbaseURL: string | undefined;\n\t/** Initial URL pattern that may contain parameters and method modifiers */\n\tinitURL: string;\n\t/** Parameters to substitute into the URL path */\n\tparams: CallApiExtraOptions[\"params\"];\n\t/** Query parameters to append to the URL */\n\tquery: CallApiExtraOptions[\"query\"];\n};\n\nexport const getFullAndNormalizedURL = (options: GetFullURLOptions) => {\n\tconst { baseURL, initURL, params, query } = options;\n\n\tconst normalizedInitURL = normalizeURL(initURL);\n\n\tconst urlWithMergedParams = mergeUrlWithParams(normalizedInitURL, params);\n\n\tconst urlWithMergedQueryAndParams = mergeUrlWithQuery(urlWithMergedParams, query);\n\n\tconst shouldPrependBaseURL = !urlWithMergedQueryAndParams.startsWith(\"http\") && baseURL;\n\n\tconst fullURL =\n\t\tshouldPrependBaseURL ? `${baseURL}${urlWithMergedQueryAndParams}` : urlWithMergedQueryAndParams;\n\n\treturn {\n\t\tfullURL,\n\t\tnormalizedInitURL,\n\t};\n};\n\nexport type AllowedQueryParamValues = UnmaskType<boolean | number | string>;\n\nexport type RecordStyleParams = UnmaskType<Record<string, AllowedQueryParamValues>>;\n\nexport type TupleStyleParams = UnmaskType<AllowedQueryParamValues[]>;\n\nexport type Params = UnmaskType<RecordStyleParams | TupleStyleParams>;\n\nexport type Query = UnmaskType<Record<string, AllowedQueryParamValues>>;\n\nexport type InitURLOrURLObject = AnyString | RouteKeyMethodsURLUnion | URL;\n\nexport interface URLOptions {\n\t/**\n\t * Base URL for all API requests. Will only be prepended to relative URLs.\n\t *\n\t * Absolute URLs (starting with http/https) will not be prepended by the baseURL.\n\t *\n\t * @example\n\t * ```ts\n\t * // Set base URL for all requests\n\t * baseURL: \"https://api.example.com/v1\"\n\t *\n\t * // Then use relative URLs in requests\n\t * callApi(\"/users\") // → https://api.example.com/v1/users\n\t * callApi(\"/posts/123\") // → https://api.example.com/v1/posts/123\n\t *\n\t * // Environment-specific base URLs\n\t * baseURL: process.env.NODE_ENV === \"production\"\n\t * ? \"https://api.example.com\"\n\t * : \"http://localhost:3000/api\"\n\t * ```\n\t */\n\tbaseURL?: string;\n\n\t/**\n\t * Resolved request URL after processing baseURL, parameters, and query strings (readonly)\n\t *\n\t * This is the final URL that will be used for the HTTP request, computed from\n\t * baseURL, initURL, params, and query parameters.\n\t *\n\t */\n\treadonly fullURL?: string;\n\n\t/**\n\t * The original URL string passed to the callApi instance (readonly)\n\t *\n\t * This preserves the original URL as provided, including any method modifiers like \"@get/\" or \"@post/\".\n\t *\n\t */\n\treadonly initURL?: string;\n\n\t/**\n\t * The URL string after normalization, with method modifiers removed(readonly)\n\t *\n\t * Method modifiers like \"@get/\", \"@post/\" are stripped to create a clean URL\n\t * for parameter substitution and final URL construction.\n\t *\n\t */\n\treadonly initURLNormalized?: string;\n\n\t/**\n\t * Parameters to be substituted into URL path segments.\n\t *\n\t * Supports both object-style (named parameters) and array-style (positional parameters)\n\t * for flexible URL parameter substitution.\n\t *\n\t * @example\n\t * ```typescript\n\t * // Object-style parameters (recommended)\n\t * const namedParams: URLOptions = {\n\t * initURL: \"/users/:userId/posts/:postId\",\n\t * params: { userId: \"123\", postId: \"456\" }\n\t * };\n\t * // Results in: /users/123/posts/456\n\t *\n\t * // Array-style parameters (positional)\n\t * const positionalParams: URLOptions = {\n\t * initURL: \"/users/:userId/posts/:postId\",\n\t * params: [\"123\", \"456\"] // Maps in order: userId=123, postId=456\n\t * };\n\t * // Results in: /users/123/posts/456\n\t *\n\t * // Single parameter\n\t * const singleParam: URLOptions = {\n\t * initURL: \"/users/:id\",\n\t * params: { id: \"user-123\" }\n\t * };\n\t * // Results in: /users/user-123\n\t * ```\n\t */\n\tparams?: Params;\n\n\t/**\n\t * Query parameters to append to the URL as search parameters.\n\t *\n\t * These will be serialized into the URL query string using standard\n\t * URL encoding practices.\n\t *\n\t * @example\n\t * ```typescript\n\t * // Basic query parameters\n\t * const queryOptions: URLOptions = {\n\t * initURL: \"/users\",\n\t * query: {\n\t * page: 1,\n\t * limit: 10,\n\t * search: \"john doe\",\n\t * active: true\n\t * }\n\t * };\n\t * // Results in: /users?page=1&limit=10&search=john%20doe&active=true\n\t *\n\t * // Filtering and sorting\n\t * const filterOptions: URLOptions = {\n\t * initURL: \"/products\",\n\t * query: {\n\t * category: \"electronics\",\n\t * minPrice: 100,\n\t * maxPrice: 500,\n\t * sortBy: \"price\",\n\t * order: \"asc\"\n\t * }\n\t * };\n\t * // Results in: /products?category=electronics&minPrice=100&maxPrice=500&sortBy=price&order=asc\n\t * ```\n\t */\n\tquery?: Query;\n}\n","export const createCombinedSignalPolyfill = (signals: AbortSignal[]) => {\n\tconst controller = new AbortController();\n\n\tconst handleAbort = (actualSignal: AbortSignal) => {\n\t\tif (controller.signal.aborted) return;\n\n\t\tcontroller.abort(actualSignal.reason);\n\t};\n\n\tfor (const actualSignal of signals) {\n\t\tif (actualSignal.aborted) {\n\t\t\thandleAbort(actualSignal);\n\t\t\tbreak;\n\t\t}\n\n\t\tactualSignal.addEventListener(\"abort\", () => handleAbort(actualSignal), {\n\t\t\tsignal: controller.signal,\n\t\t});\n\t}\n\n\treturn controller.signal;\n};\n","export const createTimeoutSignalPolyfill = (milliseconds: number) => {\n\tconst controller = new AbortController();\n\n\tconst reason = new DOMException(\"Request timed out\", \"TimeoutError\");\n\n\tconst timeout = setTimeout(() => controller.abort(reason), milliseconds);\n\n\tcontroller.signal.addEventListener(\"abort\", () => clearTimeout(timeout));\n\n\treturn controller.signal;\n};\n","import { getAuthHeader } from \"../auth\";\nimport { fetchSpecificKeys } from \"../constants/common\";\nimport { extraOptionDefaults, requestOptionDefaults } from \"../constants/defaults\";\nimport type { RequestContext } from \"../hooks\";\nimport type { Middlewares } from \"../middlewares\";\nimport type { BaseCallApiExtraOptions, CallApiExtraOptions, CallApiRequestOptions } from \"../types/common\";\nimport type { DistributiveOmit } from \"../types/type-helpers\";\nimport { extractMethodFromURL } from \"../url\";\nimport {\n\tisArray,\n\tisFunction,\n\tisPlainObject,\n\tisQueryString,\n\tisSerializable,\n\tisValidJsonString,\n} from \"./guards\";\nimport { createCombinedSignalPolyfill, createTimeoutSignalPolyfill } from \"./polyfills\";\n\nexport const omitKeys = <\n\tTObject extends Record<string, unknown>,\n\tconst TOmitArray extends Array<keyof TObject> | ReadonlyArray<keyof TObject>,\n>(\n\tinitialObject: TObject,\n\tkeysToOmit: TOmitArray\n) => {\n\tconst updatedObject = {} as Record<string, unknown>;\n\n\tconst keysToOmitSet = new Set(keysToOmit);\n\n\tfor (const [key, value] of Object.entries(initialObject)) {\n\t\tif (!keysToOmitSet.has(key)) {\n\t\t\tupdatedObject[key] = value;\n\t\t}\n\t}\n\n\treturn updatedObject as DistributiveOmit<TObject, TOmitArray[number]>;\n};\n\nexport const pickKeys = <\n\tTObject extends Record<string, unknown>,\n\tconst TPickArray extends Array<keyof TObject> | ReadonlyArray<keyof TObject>,\n>(\n\tinitialObject: TObject,\n\tkeysToPick: TPickArray\n) => {\n\tconst updatedObject = {} as Record<string, unknown>;\n\n\tconst keysToPickSet = new Set(keysToPick);\n\n\tfor (const [key, value] of Object.entries(initialObject)) {\n\t\tif (keysToPickSet.has(key)) {\n\t\t\tupdatedObject[key] = value;\n\t\t}\n\t}\n\n\treturn updatedObject as Pick<TObject, TPickArray[number]>;\n};\n\n// eslint-disable-next-line ts-eslint/no-explicit-any -- Any is required here so that one can pass custom function type without type errors\nexport const splitBaseConfig = (baseConfig: Record<string, any>) =>\n\t[\n\t\tpickKeys(baseConfig, fetchSpecificKeys) as CallApiRequestOptions,\n\t\tomitKeys(baseConfig, fetchSpecificKeys) as BaseCallApiExtraOptions,\n\t] as const;\n\n// eslint-disable-next-line ts-eslint/no-explicit-any -- Any is required here so that one can pass custom function type without type errors\nexport const splitConfig = (config: Record<string, any>) =>\n\t[\n\t\tpickKeys(config, fetchSpecificKeys) as CallApiRequestOptions,\n\t\tomitKeys(config, fetchSpecificKeys) as CallApiExtraOptions,\n\t] as const;\n\nexport const objectifyHeaders = (headers: CallApiRequestOptions[\"headers\"]) => {\n\tif (!headers || isPlainObject(headers)) {\n\t\treturn headers;\n\t}\n\n\treturn Object.fromEntries(headers);\n};\n\nexport type GetHeadersOptions = {\n\tauth: CallApiExtraOptions[\"auth\"];\n\tbody: CallApiRequestOptions[\"body\"];\n\theaders: CallApiRequestOptions[\"headers\"];\n};\n\nexport const getHeaders = async (options: GetHeadersOptions) => {\n\tconst { auth, body, headers } = options;\n\n\t// == Return early if any of the following conditions are not met (so that native fetch would auto set the correct headers):\n\tconst shouldResolveHeaders = Boolean(headers) || Boolean(body) || Boolean(auth);\n\n\tif (!shouldResolveHeaders) return;\n\n\tconst headersObject: Record<string, string | undefined> = {\n\t\t...(await getAuthHeader(auth)),\n\t\t...objectifyHeaders(headers),\n\t};\n\n\tif (isQueryString(body)) {\n\t\theadersObject[\"Content-Type\"] = \"application/x-www-form-urlencoded\";\n\n\t\treturn headersObject;\n\t}\n\n\tif (isSerializable(body) || isValidJsonString(body)) {\n\t\theadersObject[\"Content-Type\"] = \"application/json\";\n\t\theadersObject.Accept = \"application/json\";\n\t}\n\n\treturn headersObject;\n};\n\nexport type GetMethodContext = {\n\t/** The URL string that may contain method modifiers like \"@get/\" or \"@post/\" */\n\tinitURL: string | undefined;\n\t/** Explicitly specified HTTP method */\n\tmethod: CallApiRequestOptions[\"method\"];\n};\n\nexport const getMethod = (ctx: GetMethodContext) => {\n\tconst { initURL, method } = ctx;\n\n\treturn (\n\t\tmethod?.toUpperCase() ?? extractMethodFromURL(initURL)?.toUpperCase() ?? requestOptionDefaults.method\n\t);\n};\n\nexport type GetBodyOptions = {\n\tbody: CallApiRequestOptions[\"body\"];\n\tbodySerializer: CallApiExtraOptions[\"bodySerializer\"];\n};\n\nexport const getBody = (options: GetBodyOptions) => {\n\tconst { body, bodySerializer } = options;\n\n\tif (isSerializable(body)) {\n\t\tconst selectedBodySerializer = bodySerializer ?? extraOptionDefaults.bodySerializer;\n\n\t\treturn selectedBodySerializer(body);\n\t}\n\n\treturn body;\n};\n\nexport const getInitFetchImpl = (customFetchImpl: CallApiExtraOptions[\"customFetchImpl\"]) => {\n\tif (customFetchImpl) {\n\t\treturn customFetchImpl;\n\t}\n\n\tif (typeof globalThis !== \"undefined\" && isFunction(globalThis.fetch)) {\n\t\treturn globalThis.fetch;\n\t}\n\n\tthrow new Error(\"No fetch implementation found\");\n};\n\nexport const getFetchImpl = (context: {\n\tcustomFetchImpl: CallApiExtraOptions[\"customFetchImpl\"];\n\tfetchMiddleware: Middlewares[\"fetchMiddleware\"];\n\trequestContext: RequestContext;\n}) => {\n\tconst { customFetchImpl, fetchMiddleware, requestContext } = context;\n\n\tconst initFetchImpl = getInitFetchImpl(customFetchImpl);\n\n\tconst resolvedFetchImpl =\n\t\tfetchMiddleware ? fetchMiddleware({ ...requestContext, fetchImpl: initFetchImpl }) : initFetchImpl;\n\n\treturn resolvedFetchImpl;\n};\n\nexport const waitFor = (delay: number) => {\n\tif (delay === 0) return;\n\n\tconst promise = new Promise((resolve) => setTimeout(resolve, delay));\n\n\treturn promise;\n};\n\nexport const createCombinedSignal = (...signals: Array<AbortSignal | null | undefined>) => {\n\tconst cleanedSignals = signals.filter((signal) => signal != null);\n\n\tif (!(\"any\" in AbortSignal)) {\n\t\treturn createCombinedSignalPolyfill(cleanedSignals);\n\t}\n\n\tconst combinedSignal = AbortSignal.any(cleanedSignals);\n\n\treturn combinedSignal;\n};\n\nexport const createTimeoutSignal = (milliseconds: number | null | undefined) => {\n\tif (milliseconds == null) {\n\t\treturn null;\n\t}\n\n\tif (!(\"timeout\" in AbortSignal)) {\n\t\treturn createTimeoutSignalPolyfill(milliseconds);\n\t}\n\n\treturn AbortSignal.timeout(milliseconds);\n};\n\nexport const deterministicHashFn = (value: unknown): string => {\n\treturn JSON.stringify(value, (_, val: unknown) => {\n\t\tif (!isPlainObject(val)) {\n\t\t\treturn val;\n\t\t}\n\n\t\tconst sortedKeys = Object.keys(val).toSorted();\n\n\t\tconst result: Record<string, unknown> = {};\n\n\t\tfor (const key of sortedKeys) {\n\t\t\tresult[key] = val[key];\n\t\t}\n\n\t\treturn result;\n\t});\n};\n\nexport const toArray = (value: unknown) => (isArray(value) ? value : [value]);\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport type { CallApiExtraOptions } from \"./types/common\";\nimport type { ThrowOnErrorUnion } from \"./types/conditional-types\";\nimport type { DefaultDataType, DefaultThrowOnError } from \"./types/default-types\";\nimport type { AnyString, Awaitable, DistributiveOmit, UnmaskType } from \"./types/type-helpers\";\nimport { omitKeys } from \"./utils/common\";\nimport type { HTTPError, ValidationError } from \"./utils/external/error\";\nimport { isHTTPErrorInstance, isValidationErrorInstance } from \"./utils/external/guards\";\n\nexport type ResponseType = \"blob\" | \"json\" | \"text\";\n\ntype Parser<TData> = (responseString: string) => Awaitable<TData>;\n\nexport const getResponseType = <TResponse>(response: Response, parser: Parser<TResponse>) => ({\n\tarrayBuffer: () => response.arrayBuffer(),\n\tblob: () => response.blob(),\n\tformData: () => response.formData(),\n\tjson: async (): Promise<TResponse> => {\n\t\tconst text = await response.text();\n\t\treturn parser(text);\n\t},\n\tstream: () => response.body,\n\ttext: () => response.text(),\n});\n\ntype InitResponseTypeMap<TResponse = unknown> = ReturnType<typeof getResponseType<TResponse>>;\n\ntype ResponseTypeUnion = keyof InitResponseTypeMap;\n\ntype ResponseTypePlaceholder = null;\n\nexport type ResponseTypeType = ResponseTypePlaceholder | ResponseTypeUnion;\n\nexport type ResponseTypeMap<TResponse> = {\n\t[Key in keyof InitResponseTypeMap<TResponse>]: Awaited<ReturnType<InitResponseTypeMap<TResponse>[Key]>>;\n};\n\nexport type GetResponseType<\n\tTResponse,\n\tTResponseType extends ResponseTypeType,\n\tTComputedResponseTypeMap extends ResponseTypeMap<TResponse> = ResponseTypeMap<TResponse>,\n> =\n\tnull extends TResponseType ? TComputedResponseTypeMap[\"json\"]\n\t: TResponseType extends NonNullable<ResponseTypeType> ? TComputedResponseTypeMap[TResponseType]\n\t: never;\n\nconst textTypes = new Set([\"image/svg\", \"application/xml\", \"application/xhtml\", \"application/html\"]);\nconst JSON_REGEX = /^application\\/(?:[\\w!#$%&*.^`~-]*\\+)?json(;.+)?$/i;\n\nconst detectResponseType = (response: Response): Extract<ResponseTypeType, \"blob\" | \"json\" | \"text\"> => {\n\tconst initContentType = response.headers.get(\"content-type\");\n\n\tif (!initContentType) {\n\t\treturn extraOptionDefaults.responseType;\n\t}\n\n\tconst contentType = initContentType.split(\";\")[0] ?? \"\";\n\n\tif (JSON_REGEX.test(contentType)) {\n\t\treturn \"json\";\n\t}\n\n\tif (textTypes.has(contentType) || contentType.startsWith(\"text/\")) {\n\t\treturn \"text\";\n\t}\n\n\treturn \"blob\";\n};\n\nexport const resolveResponseData = <TResponse>(\n\tresponse: Response,\n\tresponseType: ResponseTypeType | undefined,\n\tparser: Parser<TResponse> | undefined\n) => {\n\tconst selectedParser = parser ?? extraOptionDefaults.responseParser;\n\tconst selectedResponseType = responseType ?? detectResponseType(response);\n\n\tconst RESPONSE_TYPE_LOOKUP = getResponseType<TResponse>(response, selectedParser);\n\n\tif (!Object.hasOwn(RESPONSE_TYPE_LOOKUP, selectedResponseType)) {\n\t\tthrow new Error(`Invalid response type: ${responseType}`);\n\t}\n\n\treturn RESPONSE_TYPE_LOOKUP[selectedResponseType]();\n};\n\nexport type CallApiResultSuccessVariant<TData> = {\n\tdata: NoInfer<TData>;\n\terror: null;\n\tresponse: Response;\n};\nexport type PossibleJavaScriptError = UnmaskType<{\n\terrorData: false;\n\tmessage: string;\n\tname: \"AbortError\" | \"Error\" | \"SyntaxError\" | \"TimeoutError\" | \"TypeError\" | AnyString;\n\toriginalError: DOMException | Error | SyntaxError | TypeError;\n}>;\n\nexport type PossibleHTTPError<TErrorData> = UnmaskType<{\n\terrorData: NoInfer<TErrorData>;\n\tmessage: string;\n\tname: \"HTTPError\";\n\toriginalError: HTTPError;\n}>;\n\nexport type PossibleValidationError = UnmaskType<{\n\terrorData: ValidationError[\"errorData\"];\n\tissueCause: ValidationError[\"issueCause\"];\n\tmessage: string;\n\tname: \"ValidationError\";\n\toriginalError: ValidationError;\n}>;\n\nexport type PossibleJavaScriptOrValidationError = UnmaskType<\n\tPossibleJavaScriptError | PossibleValidationError\n>;\n\nexport type CallApiResultErrorVariant<TErrorData> =\n\t| {\n\t\t\tdata: null;\n\t\t\terror: PossibleHTTPError<TErrorData>;\n\t\t\tresponse: Response;\n\t }\n\t| {\n\t\t\tdata: null;\n\t\t\terror: PossibleJavaScriptOrValidationError;\n\t\t\tresponse: Response | null;\n\t };\n\nexport type CallApiSuccessOrErrorVariant<TData, TError> =\n\t| CallApiResultErrorVariant<TError>\n\t| CallApiResultSuccessVariant<TData>;\n\nexport type ResultModeMapWithoutException<\n\tTData,\n\tTErrorData,\n\tTResponseType extends ResponseTypeType,\n\tTComputedData = GetResponseType<TData, TResponseType>,\n\tTComputedErrorData = GetResponseType<TErrorData, TResponseType>,\n\tTComputedResult extends CallApiSuccessOrErrorVariant<TComputedData, TComputedErrorData> =\n\t\tCallApiSuccessOrErrorVariant<TComputedData, TComputedErrorData>,\n> = UnmaskType<{\n\tall: TComputedResult;\n\tonlyData: TComputedResult[\"data\"];\n\tonlyResponse: TComputedResult[\"response\"];\n\twithoutResponse: DistributiveOmit<TComputedResult, \"response\">;\n}>;\n\ntype ResultModeMapWithException<\n\tTData,\n\tTResponseType extends ResponseTypeType,\n\tTComputedData = GetResponseType<TData, TResponseType>,\n\tTComputedResult extends CallApiResultSuccessVariant<TComputedData> =\n\t\tCallApiResultSuccessVariant<TComputedData>,\n> = {\n\tall: TComputedResult;\n\tonlyData: TComputedResult[\"data\"];\n\tonlyResponse: TComputedResult[\"response\"];\n\twithoutResponse: DistributiveOmit<TComputedResult, \"response\">;\n};\n\nexport type ResultModeMap<\n\tTData = DefaultDataType,\n\tTErrorData = DefaultDataType,\n\tTResponseType extends ResponseTypeType = ResponseTypeType,\n\tTThrowOnError extends ThrowOnErrorUnion = DefaultThrowOnError,\n> =\n\tTThrowOnError extends true ? ResultModeMapWithException<TData, TResponseType>\n\t:\tResultModeMapWithoutException<TData, TErrorData, TResponseType>;\n\ntype ResultModePlaceholder = null;\n\ntype ResultModeUnion = keyof ResultModeMap;\n\n// FIXME - Revisit this idea later. Take inspirations from how zod does it with pick({}) and omit({d})\n// type ResultModeObject = { data?: boolean; error?: boolean; response?: boolean };\n\nexport type ResultModeType = ResultModePlaceholder | ResultModeUnion;\n\nexport type GetCallApiResult<\n\tTData,\n\tTErrorData,\n\tTResultMode extends ResultModeType,\n\tTThrowOnError extends ThrowOnErrorUnion,\n\tTResponseType extends ResponseTypeType,\n\tTComputedResultModeMapWithException extends ResultModeMapWithException<TData, TResponseType> =\n\t\tResultModeMapWithException<TData, TResponseType>,\n\tTComputedResultModeMap extends ResultModeMap<TData, TErrorData, TResponseType, TThrowOnError> =\n\t\tResultModeMap<TData, TErrorData, TResponseType, TThrowOnError>,\n> =\n\tTErrorData extends false ? TComputedResultModeMapWithException[\"onlyData\"]\n\t: TErrorData extends false | undefined ? TComputedResultModeMapWithException[\"onlyData\"]\n\t: ResultModePlaceholder extends TResultMode ? TComputedResultModeMap[\"all\"]\n\t: TResultMode extends ResultModeUnion ? TComputedResultModeMap[TResultMode]\n\t: never;\n\ntype SuccessInfo = Pick<CallApiExtraOptions, \"resultMode\"> & {\n\tresponse: Response;\n};\n\ntype LazyResultModeMap = {\n\t[key in keyof ResultModeMap]: () => ResultModeMap[key];\n};\n\nconst getResultModeMap = (details: ResultModeMap[\"all\"]): LazyResultModeMap => {\n\treturn {\n\t\tall: () => details,\n\t\tonlyData: () => details.data,\n\t\tonlyResponse: () => details.response,\n\t\twithoutResponse: () => omitKeys(details, [\"response\"]),\n\t};\n};\n\ntype SuccessResult = CallApiResultSuccessVariant<unknown> | null;\n\n// The CallApiResult type is used to cast all return statements due to a design limitation in ts.\n// LINK - See https://www.zhenghao.io/posts/type-functions for more info\nexport const resolveSuccessResult = (data: unknown, info: SuccessInfo): SuccessResult => {\n\tconst { response, resultMode } = info;\n\n\tconst details = {\n\t\tdata,\n\t\terror: null,\n\t\tresponse,\n\t} satisfies CallApiResultSuccessVariant<unknown>;\n\n\tconst resultModeMap = getResultModeMap(details);\n\n\tconst successResult = resultModeMap[resultMode ?? \"all\"]();\n\n\treturn successResult as SuccessResult;\n};\n\nexport type ErrorInfo = Omit<SuccessInfo, \"response\">\n\t& Pick<CallApiExtraOptions, \"cloneResponse\"> & {\n\t\tmessage?: string;\n\t};\n\ntype ErrorResult = {\n\terrorDetails: CallApiResultErrorVariant<unknown>;\n\terrorResult: CallApiResultErrorVariant<unknown> | null;\n};\n\nexport const resolveErrorResult = (error: unknown, info: ErrorInfo): ErrorResult => {\n\tconst { cloneResponse, message: customErrorMessage, resultMode } = info;\n\n\tlet errorDetails = {\n\t\tdata: null,\n\t\terror: {\n\t\t\terrorData: false,\n\t\t\tmessage: customErrorMessage ?? (error as Error).message,\n\t\t\tname: (error as Error).name,\n\t\t\toriginalError: error as Error,\n\t\t},\n\t\tresponse: null,\n\t} satisfies CallApiResultErrorVariant<unknown> as CallApiResultErrorVariant<unknown>;\n\n\tif (isValidationErrorInstance(error)) {\n\t\tconst { errorData, message, response } = error;\n\n\t\terrorDetails = {\n\t\t\tdata: null,\n\t\t\terror: {\n\t\t\t\terrorData,\n\t\t\t\tissueCause: error.issueCause,\n\t\t\t\tmessage,\n\t\t\t\tname: \"ValidationError\",\n\t\t\t\toriginalError: error,\n\t\t\t},\n\t\t\tresponse,\n\t\t};\n\t}\n\n\tif (isHTTPErrorInstance<never>(error)) {\n\t\tconst { errorData, message, name, response } = error;\n\n\t\terrorDetails = {\n\t\t\tdata: null,\n\t\t\terror: { errorData, message, name, originalError: error },\n\t\t\tresponse: cloneResponse ? response.clone() : response,\n\t\t};\n\t}\n\n\tconst resultModeMap = getResultModeMap(errorDetails);\n\n\tconst errorResult = resultModeMap[resultMode ?? \"all\"]() as never;\n\n\treturn { errorDetails, errorResult };\n};\n\nexport const getCustomizedErrorResult = (\n\terrorResult: ErrorResult[\"errorResult\"],\n\tcustomErrorInfo: { message: string | undefined }\n): ErrorResult[\"errorResult\"] => {\n\tif (!errorResult) {\n\t\treturn null;\n\t}\n\n\tconst { message = errorResult.error.message } = customErrorInfo;\n\n\treturn {\n\t\t...errorResult,\n\t\terror: {\n\t\t\t...errorResult.error,\n\t\t\tmessage,\n\t\t} satisfies NonNullable<ErrorResult[\"errorResult\"]>[\"error\"] as never,\n\t};\n};\n","import {\n\ttype CallApiResultErrorVariant,\n\ttype CallApiResultSuccessVariant,\n\ttype ErrorInfo,\n\ttype PossibleHTTPError,\n\ttype PossibleJavaScriptError,\n\ttype PossibleJavaScriptOrValidationError,\n\ttype PossibleValidationError,\n\tresolveErrorResult,\n} from \"./result\";\nimport type { StreamProgressEvent } from \"./stream\";\nimport type {\n\tBaseCallApiConfig,\n\tCallApiConfig,\n\tCallApiContext,\n\tCallApiExtraOptionsForHooks,\n\tCallApiRequestOptionsForHooks,\n} from \"./types/common\";\nimport type { DefaultCallApiContext } from \"./types/default-types\";\nimport type { AnyFunction, Awaitable, Prettify, UnmaskType } from \"./types/type-helpers\";\n\nexport interface Hooks<TCallApiContext extends CallApiContext = DefaultCallApiContext> {\n\t/**\n\t * Hook called when any error occurs within the request/response lifecycle.\n\t *\n\t * This is a unified error handler that catches both request errors (network failures,\n\t * timeouts, etc.) and response errors (HTTP error status codes). It's essentially\n\t * a combination of `onRequestError` and `onResponseError` hooks.\n\t *\n\t * @param context - Error context containing error details, request info, and response (if available)\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonError?: (context: ErrorContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called before the HTTP request is sent and before any internal processing of the request object begins.\n\t *\n\t * This is the ideal place to modify request headers, add authentication,\n\t * implement request logging, or perform any setup before the network call.\n\t *\n\t * @param context - Request context with mutable request object and configuration\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRequest?: (context: RequestContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when an error occurs during the fetch request itself.\n\t *\n\t * This handles network-level errors like connection failures, timeouts,\n\t * DNS resolution errors, or other issues that prevent getting an HTTP response.\n\t * Note that HTTP error status codes (4xx, 5xx) are handled by `onResponseError`.\n\t *\n\t * @param context - Request error context with error details and null response\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonRequestError?: (context: RequestErrorContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called just before the HTTP request is sent and after the request has been processed.\n\t *\n\t * @param context - Request context with mutable request object and configuration\n\t */\n\tonRequestReady?: (context: RequestContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called during upload stream progress tracking.\n\t *\n\t * This hook is triggered when uploading data (like file uploads) and provides\n\t * progress information about the upload. Useful for implementing progress bars\n\t * or upload status indicators.\n\t *\n\t * @param context - Request stream context with progress event and request instance\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRequestStream?: (context: RequestStreamContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when any HTTP response is received from the API.\n\t *\n\t * This hook is triggered for both successful (2xx) and error (4xx, 5xx) responses.\n\t * It's useful for response logging, metrics collection, or any processing that\n\t * should happen regardless of response status.\n\t *\n\t * @param context - Response context with either success data or error information\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonResponse?: (context: ResponseContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when an HTTP error response (4xx, 5xx) is received from the API.\n\t *\n\t * This handles server-side errors where an HTTP response was successfully received\n\t * but indicates an error condition. Different from `onRequestError` which handles\n\t * network-level failures.\n\t *\n\t * @param context - Response error context with HTTP error details and response\n\t * @returns Promise or void - Hook can be async or sync\n\t */\n\tonResponseError?: (context: ResponseErrorContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called during download stream progress tracking.\n\t *\n\t * This hook is triggered when downloading data (like file downloads) and provides\n\t * progress information about the download. Useful for implementing progress bars\n\t * or download status indicators.\n\t *\n\t * @param context - Response stream context with progress event and response\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonResponseStream?: (context: ResponseStreamContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a request is being retried.\n\t *\n\t * This hook is triggered before each retry attempt, providing information about\n\t * the previous failure and the current retry attempt number. Useful for implementing\n\t * custom retry logic, exponential backoff, or retry logging.\n\t *\n\t * @param context - Retry context with error details and retry attempt count\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonRetry?: (response: RetryContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a successful response (2xx status) is received from the API.\n\t *\n\t * This hook is triggered only for successful responses and provides access to\n\t * the parsed response data. Ideal for success logging, caching, or post-processing\n\t * of successful API responses.\n\t *\n\t * @param context - Success context with parsed response data and response object\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonSuccess?: (context: SuccessContext<TCallApiContext>) => Awaitable<unknown>;\n\n\t/**\n\t * Hook called when a validation error occurs.\n\t *\n\t * This hook is triggered when request or response data fails validation against\n\t * a defined schema. It provides access to the validation error details and can\n\t * be used for custom error handling, logging, or fallback behavior.\n\t *\n\t * @param context - Validation error context with error details and response (if available)\n\t * @returns Promise or void - Hook can be async or sync\n\t *\n\t */\n\tonValidationError?: (context: ValidationErrorContext<TCallApiContext>) => Awaitable<unknown>;\n}\n\nexport type HooksOrHooksArray<TCallApiContext extends CallApiContext = DefaultCallApiContext> = {\n\t[Key in keyof Hooks<TCallApiContext>]:\n\t\t| Hooks<TCallApiContext>[Key]\n\t\t// eslint-disable-next-line perfectionist/sort-union-types -- I need arrays to be last\n\t\t| Array<Hooks<TCallApiContext>[Key]>;\n};\n\nexport interface HookConfigOptions {\n\t/**\n\t * Controls the execution mode of all composed hooks (main + plugin hooks).\n\t *\n\t * - **\"parallel\"**: All hooks execute simultaneously via Promise.all() for better performance\n\t * - **\"sequential\"**: All hooks execute one by one in registration order via await in a loop\n\t *\n\t * This affects how ALL hooks execute together, regardless of their source (main or plugin).\n\t *\n\t * @default \"parallel\"\n\t */\n\thooksExecutionMode?: \"parallel\" | \"sequential\";\n}\n\nexport type RequestContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = {\n\t/**\n\t * Base configuration object passed to createFetchClient.\n\t *\n\t * Contains the foundational configuration that applies to all requests\n\t * made by this client instance, such as baseURL, default headers, and\n\t * global options.\n\t */\n\tbaseConfig: Exclude<BaseCallApiConfig, AnyFunction>;\n\n\t/**\n\t * Instance-specific configuration object passed to the callApi instance.\n\t *\n\t * Contains configuration specific to this particular API call, which\n\t * can override or extend the base configuration.\n\t */\n\tconfig: CallApiConfig;\n\n\t/**\n\t * Merged options combining base config, instance config, and default options.\n\t *\n\t * This is the final resolved configuration that will be used for the request,\n\t * with proper precedence applied (instance > base > defaults).\n\t */\n\toptions: CallApiExtraOptionsForHooks<TCallApiContext>;\n\n\t/**\n\t * Merged request object ready to be sent.\n\t *\n\t * Contains the final request configuration including URL, method, headers,\n\t * body, and other fetch options. This object can be modified in onRequest\n\t * hooks to customize the outgoing request.\n\t */\n\trequest: CallApiRequestOptionsForHooks;\n};\n\nexport type ValidationErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\terror: PossibleValidationError;\n\t\tresponse: Response | null;\n\t}\n>;\n\nexport type SuccessContext<\n\tTCallApiContext extends Pick<CallApiContext, \"Data\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tdata: NoInfer<TCallApiContext[\"Data\"]>;\n\t\tresponse: Response;\n\t}\n>;\n\nexport type ResponseContext<\n\tTCallApiContext extends Pick<CallApiContext, \"Data\" | \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext>\n\t\t& (\n\t\t\t| Prettify<CallApiResultSuccessVariant<TCallApiContext[\"Data\"]>>\n\t\t\t| Prettify<\n\t\t\t\t\tExtract<\n\t\t\t\t\t\tCallApiResultErrorVariant<TCallApiContext[\"ErrorData\"]>,\n\t\t\t\t\t\t{ error: PossibleHTTPError<TCallApiContext[\"ErrorData\"]> }\n\t\t\t\t\t>\n\t\t\t >\n\t\t)\n>;\n\nexport type RequestErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = RequestContext<TCallApiContext> & {\n\terror: PossibleJavaScriptError;\n\tresponse: null;\n};\n\nexport type ErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext>\n\t\t& (\n\t\t\t| {\n\t\t\t\t\terror: PossibleHTTPError<TCallApiContext[\"ErrorData\"]>;\n\t\t\t\t\tresponse: Response;\n\t\t\t }\n\t\t\t| {\n\t\t\t\t\terror: PossibleJavaScriptOrValidationError;\n\t\t\t\t\tresponse: Response | null;\n\t\t\t }\n\t\t)\n>;\n\nexport type ResponseErrorContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n> = UnmaskType<\n\tExtract<ErrorContext<TCallApiContext>, { error: PossibleHTTPError<TCallApiContext[\"ErrorData\"]> }>\n\t\t& RequestContext<TCallApiContext>\n>;\n\nexport type RetryContext<\n\tTCallApiContext extends Pick<CallApiContext, \"ErrorData\" | \"InferredPluginOptions\" | \"Meta\"> =\n\t\tDefaultCallApiContext,\n> = UnmaskType<\n\tErrorContext<TCallApiContext> & {\n\t\tretryAttemptCount: number;\n\t}\n>;\n\nexport type RequestStreamContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tevent: StreamProgressEvent;\n\t\trequestInstance: Request;\n\t}\n>;\n\nexport type ResponseStreamContext<\n\tTCallApiContext extends Pick<CallApiContext, \"InferredPluginOptions\" | \"Meta\"> = DefaultCallApiContext,\n> = UnmaskType<\n\tRequestContext<TCallApiContext> & {\n\t\tevent: StreamProgressEvent;\n\t\tresponse: Response;\n\t}\n>;\n\ntype HookRegistries = Required<{\n\t[Key in keyof Hooks]: Set<Hooks[Key]>;\n}>;\n\nexport const getHookRegistriesAndKeys = () => {\n\tconst hookRegistries: HookRegistries = {\n\t\tonError: new Set(),\n\t\tonRequest: new Set(),\n\t\tonRequestError: new Set(),\n\t\tonRequestReady: new Set(),\n\t\tonRequestStream: new Set(),\n\t\tonResponse: new Set(),\n\t\tonResponseError: new Set(),\n\t\tonResponseStream: new Set(),\n\t\tonRetry: new Set(),\n\t\tonSuccess: new Set(),\n\t\tonValidationError: new Set(),\n\t};\n\n\tconst hookRegistryKeys = Object.keys(hookRegistries) as Array<keyof Hooks>;\n\n\treturn { hookRegistries, hookRegistryKeys };\n};\n\nexport const composeHooksFromArray = (\n\thooksArray: Array<Hooks[keyof Hooks] | undefined>,\n\thooksExecutionMode: CallApiExtraOptionsForHooks[\"hooksExecutionMode\"]\n) => {\n\tconst composedHook = async (ctx: unknown) => {\n\t\tswitch (hooksExecutionMode) {\n\t\t\tcase \"parallel\": {\n\t\t\t\tawait Promise.all(hooksArray.map((uniqueHook) => uniqueHook?.(ctx as never)));\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tcase \"sequential\": {\n\t\t\t\tfor (const hook of hooksArray) {\n\t\t\t\t\t// eslint-disable-next-line no-await-in-loop -- This is necessary in this case\n\t\t\t\t\tawait hook?.(ctx as never);\n\t\t\t\t}\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tdefault: {\n\t\t\t\thooksExecutionMode satisfies undefined;\n\t\t\t}\n\t\t}\n\t};\n\n\treturn composedHook;\n};\n\nexport const executeHooks = async (...hookResultsOrPromise: Array<Awaitable<unknown>>) => {\n\tawait Promise.all(hookResultsOrPromise);\n};\n\nexport type ExecuteHookInfo = {\n\terrorInfo: ErrorInfo;\n\tshouldThrowOnError: boolean | undefined;\n};\n\nexport const executeHooksInCatchBlock = async (\n\thookResultsOrPromise: Array<Awaitable<unknown>>,\n\thookInfo: ExecuteHookInfo\n) => {\n\tconst { errorInfo, shouldThrowOnError } = hookInfo;\n\n\ttry {\n\t\tawait Promise.all(hookResultsOrPromise);\n\n\t\treturn null;\n\t} catch (hookError) {\n\t\tif (shouldThrowOnError) {\n\t\t\tthrow hookError;\n\t\t}\n\n\t\tconst { errorResult } = resolveErrorResult(hookError, errorInfo);\n\n\t\treturn errorResult;\n\t}\n};\n","import {\n\texecuteHooks,\n\ttype RequestContext,\n\ttype RequestStreamContext,\n\ttype ResponseStreamContext,\n} from \"./hooks\";\nimport { isObject, isReadableStream } from \"./utils/guards\";\n\nexport type StreamProgressEvent = {\n\t/**\n\t * Current chunk of data being streamed\n\t */\n\tchunk: Uint8Array;\n\t/**\n\t * Progress in percentage\n\t */\n\tprogress: number;\n\t/**\n\t * Total size of data in bytes\n\t */\n\ttotalBytes: number;\n\t/**\n\t * Amount of data transferred so far\n\t */\n\ttransferredBytes: number;\n};\n\ndeclare global {\n\tinterface ReadableStream<R> {\n\t\t[Symbol.asyncIterator]: () => AsyncIterableIterator<R>;\n\t}\n}\n\nconst createProgressEvent = (options: {\n\tchunk: Uint8Array;\n\ttotalBytes: number;\n\ttransferredBytes: number;\n}): StreamProgressEvent => {\n\tconst { chunk, totalBytes, transferredBytes } = options;\n\n\treturn {\n\t\tchunk,\n\t\tprogress: Math.round((transferredBytes / totalBytes) * 100) || 0,\n\t\ttotalBytes,\n\t\ttransferredBytes,\n\t};\n};\n\nconst calculateTotalBytesFromBody = async (\n\trequestBody: Request[\"body\"] | null,\n\texistingTotalBytes: number\n) => {\n\tlet totalBytes = existingTotalBytes;\n\n\tif (!requestBody) {\n\t\treturn totalBytes;\n\t}\n\n\tfor await (const chunk of requestBody) {\n\t\ttotalBytes += chunk.byteLength;\n\t}\n\n\treturn totalBytes;\n};\n\ntype ToStreamableRequestContext = RequestContext;\n\nexport const toStreamableRequest = async (\n\tcontext: ToStreamableRequestContext\n): Promise<Request | RequestInit> => {\n\tconst { baseConfig, config, options, request } = context;\n\n\tif (!options.onRequestStream || !isReadableStream(request.body)) {\n\t\treturn request as RequestInit;\n\t}\n\n\tconst requestInstance = new Request(\n\t\toptions.fullURL as NonNullable<typeof options.fullURL>,\n\t\t{ ...request, duplex: \"half\" } as RequestInit\n\t);\n\n\tconst contentLength = requestInstance.headers.get(\"content-length\");\n\n\tlet totalBytes = Number(contentLength ?? 0);\n\n\tconst shouldForcefullyCalcStreamSize =\n\t\tisObject(options.forcefullyCalculateStreamSize) ?\n\t\t\toptions.forcefullyCalculateStreamSize.request\n\t\t:\toptions.forcefullyCalculateStreamSize;\n\n\t// == If no content length is present, we read the total bytes from the body\n\tif (!contentLength && shouldForcefullyCalcStreamSize) {\n\t\ttotalBytes = await calculateTotalBytesFromBody(requestInstance.clone().body, totalBytes);\n\t}\n\n\tlet transferredBytes = 0;\n\n\tconst stream = new ReadableStream({\n\t\tstart: async (controller) => {\n\t\t\tconst body = requestInstance.body;\n\n\t\t\tif (!body) return;\n\n\t\t\tconst requestStreamContext = {\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tevent: createProgressEvent({ chunk: new Uint8Array(), totalBytes, transferredBytes }),\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t\trequestInstance,\n\t\t\t} satisfies RequestStreamContext;\n\n\t\t\tawait executeHooks(options.onRequestStream?.(requestStreamContext));\n\n\t\t\tfor await (const chunk of body) {\n\t\t\t\ttransferredBytes += chunk.byteLength;\n\n\t\t\t\ttotalBytes = Math.max(totalBytes, transferredBytes);\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onRequestStream?.({\n\t\t\t\t\t\t...requestStreamContext,\n\t\t\t\t\t\tevent: createProgressEvent({ chunk, totalBytes, transferredBytes }),\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\tcontroller.enqueue(chunk);\n\t\t\t}\n\n\t\t\tcontroller.close();\n\t\t},\n\t});\n\n\treturn new Request(requestInstance, { body: stream, duplex: \"half\" } as RequestInit);\n};\n\ntype StreamableResponseContext = RequestContext & { response: Response };\n\nexport const toStreamableResponse = async (context: StreamableResponseContext): Promise<Response> => {\n\tconst { baseConfig, config, options, request, response } = context;\n\n\tif (!options.onResponseStream || !response.body) {\n\t\treturn response;\n\t}\n\n\tconst contentLength = response.headers.get(\"content-length\");\n\n\tlet totalBytes = Number(contentLength ?? 0);\n\n\tconst shouldForceContentLengthCalc =\n\t\tisObject(options.forcefullyCalculateStreamSize) ?\n\t\t\toptions.forcefullyCalculateStreamSize.response\n\t\t:\toptions.forcefullyCalculateStreamSize;\n\n\t// == If no content length is present and `forceContentLengthCalculation` is enabled, we read the total bytes from the body\n\tif (!contentLength && shouldForceContentLengthCalc) {\n\t\ttotalBytes = await calculateTotalBytesFromBody(response.clone().body, totalBytes);\n\t}\n\n\tlet transferredBytes = 0;\n\n\tconst stream = new ReadableStream({\n\t\tstart: async (controller) => {\n\t\t\tconst body = response.body;\n\n\t\t\tif (!body) return;\n\n\t\t\tconst responseStreamContext = {\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tevent: createProgressEvent({ chunk: new Uint8Array(), totalBytes, transferredBytes }),\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t\tresponse,\n\t\t\t} satisfies ResponseStreamContext;\n\n\t\t\tawait executeHooks(options.onResponseStream?.(responseStreamContext));\n\n\t\t\tfor await (const chunk of body) {\n\t\t\t\ttransferredBytes += chunk.byteLength;\n\n\t\t\t\ttotalBytes = Math.max(totalBytes, transferredBytes);\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onResponseStream?.({\n\t\t\t\t\t\t...responseStreamContext,\n\t\t\t\t\t\tevent: createProgressEvent({ chunk, totalBytes, transferredBytes }),\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\tcontroller.enqueue(chunk);\n\t\t\t}\n\n\t\t\tcontroller.close();\n\t\t},\n\t});\n\n\treturn new Response(stream, response);\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport type { RequestContext } from \"./hooks\";\nimport { toStreamableRequest, toStreamableResponse } from \"./stream\";\nimport type { AnyString, RemovePrefix, UnmaskType } from \"./types/type-helpers\";\nimport { deterministicHashFn, waitFor } from \"./utils/common\";\nimport { isFunction } from \"./utils/guards\";\n\ntype RequestInfo = {\n\tcontroller: AbortController;\n\tresponsePromise: Promise<Response>;\n};\n\n/**\n * Cache that stores active request information for deduplication within a specific scope.\n *\n * Maps deduplication keys to their corresponding request information, including the abort controller\n * and response promise. A `null` key represents requests that don't participate in deduplication.\n *\n * **Internal Usage:**\n * This type is primarily used internally by the deduplication system. You typically won't need to\n * interact with it directly unless you're building custom deduplication logic or debugging.\n *\n * @example\n * ```ts\n * // This is handled internally, but conceptually:\n * const cache: RequestInfoCache = new Map([\n * [\"user-123\", { controller: abortController, responsePromise: fetchPromise }],\n * [\"config\", { controller: abortController2, responsePromise: fetchPromise2 }],\n * ]);\n * ```\n */\nexport type RequestInfoCache = Map<string | null, RequestInfo>;\n\n/**\n * Global cache that manages multiple request info caches, organized by scope keys.\n *\n * This enables the global deduplication feature by maintaining separate cache namespaces\n * for different scope keys. Each scope key gets its own `RequestInfoCache` instance.\n *\n * **Cache Lifecycle:**\n * - Caches are created on-demand when first accessed\n * - Automatic cleanup occurs when no references remain\n * - Each scope key maintains independent deduplication state\n *\n * **Memory Considerations:**\n * - Each scope key creates a separate cache instance\n * - Consider the number of different scope keys in your application\n * - Caches are cleaned up automatically when clients are garbage collected\n *\n * @example\n * ```ts\n * // This is managed internally, but conceptually:\n * const globalCache: GlobalRequestInfoCache = new Map([\n * [\"user-service\", new Map([...])], // Cache for user service requests\n * [\"analytics\", new Map([...])], // Cache for analytics requests\n * [\"default\", new Map([...])] // Default cache scope\n * ]);\n * ```\n */\nexport type GlobalRequestInfoCache = Map<DedupeOptions[\"dedupeCacheScopeKey\"], RequestInfoCache>;\n\ntype DedupeContext = RequestContext & {\n\t$GlobalRequestInfoCache: GlobalRequestInfoCache;\n\t$LocalRequestInfoCache: RequestInfoCache;\n\tnewFetchController: AbortController;\n};\n\nexport const createDedupeStrategy = async (context: DedupeContext) => {\n\tconst {\n\t\t$GlobalRequestInfoCache,\n\t\t$LocalRequestInfoCache,\n\t\tbaseConfig,\n\t\tconfig,\n\t\tnewFetchController,\n\t\toptions: globalOptions,\n\t\trequest: globalRequest,\n\t} = context;\n\n\tconst dedupeStrategy =\n\t\tglobalOptions.dedupeStrategy ?? globalOptions.dedupe?.strategy ?? extraOptionDefaults.dedupeStrategy;\n\n\tconst resolvedDedupeStrategy = isFunction(dedupeStrategy) ? dedupeStrategy(context) : dedupeStrategy;\n\n\tconst getDedupeKey = () => {\n\t\tconst shouldHaveDedupeKey =\n\t\t\tresolvedDedupeStrategy === \"cancel\" || resolvedDedupeStrategy === \"defer\";\n\n\t\tif (!shouldHaveDedupeKey) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst dedupeKey = globalOptions.dedupeKey ?? globalOptions.dedupe?.key;\n\n\t\tconst resolvedDedupeKey = isFunction(dedupeKey) ? dedupeKey(context) : dedupeKey;\n\n\t\tif (resolvedDedupeKey === undefined) {\n\t\t\tconst defaultDedupeKey = `${globalOptions.fullURL}-${deterministicHashFn({ options: globalOptions, request: globalRequest })}`;\n\n\t\t\treturn defaultDedupeKey;\n\t\t}\n\n\t\treturn resolvedDedupeKey;\n\t};\n\n\tconst getDedupeCacheScopeKey = () => {\n\t\tconst dedupeCacheScopeKey = globalOptions.dedupeCacheScopeKey ?? globalOptions.dedupe?.cacheScopeKey;\n\n\t\tconst resolvedDedupeCacheScopeKey =\n\t\t\tisFunction(dedupeCacheScopeKey) ? dedupeCacheScopeKey(context) : dedupeCacheScopeKey;\n\n\t\tif (resolvedDedupeCacheScopeKey === undefined) {\n\t\t\treturn extraOptionDefaults.dedupeCacheScopeKey;\n\t\t}\n\n\t\treturn resolvedDedupeCacheScopeKey;\n\t};\n\n\tconst dedupeKey = getDedupeKey();\n\n\tconst dedupeCacheScope =\n\t\tglobalOptions.dedupeCacheScope\n\t\t?? globalOptions.dedupe?.cacheScope\n\t\t?? extraOptionDefaults.dedupeCacheScope;\n\n\tconst dedupeCacheScopeKey = getDedupeCacheScopeKey();\n\n\tif (dedupeCacheScope === \"global\" && !$GlobalRequestInfoCache.has(dedupeCacheScopeKey)) {\n\t\t$GlobalRequestInfoCache.set(dedupeCacheScopeKey, new Map());\n\t}\n\n\tconst $RequestInfoCache =\n\t\tdedupeCacheScope === \"global\" ?\n\t\t\t$GlobalRequestInfoCache.get(dedupeCacheScopeKey)\n\t\t:\t$LocalRequestInfoCache;\n\n\t// == This is to ensure cache operations only occur when key is available\n\tconst $RequestInfoCacheOrNull = dedupeKey !== null ? $RequestInfoCache : null;\n\n\t/**\n\t * Force sequential execution of parallel requests to enable proper cache-based deduplication.\n\t *\n\t * Problem: When Promise.all([callApi(url), callApi(url)]) executes, both requests\n\t * start synchronously and reach this point before either can populate the cache.\n\t *\n\t * Why `await Promise.resolve()` fails:\n\t * - All microtasks in a batch resolve together at the next microtask checkpoint\n\t * - Both requests resume execution simultaneously after the await\n\t * - Both check `prevRequestInfo` at the same time → both see empty cache\n\t * - Both proceed to populate cache → deduplication fails\n\t *\n\t * Why `wait new Promise(()=> setTimeout(resolve, number))` works:\n\t * - Each setTimeout creates a separate task in the task queue\n\t * - Tasks execute sequentially, not simultaneously\n\t * - Request 1's task runs first: checks cache (empty) → continues → populates cache\n\t * - Request 2's task runs after: checks cache (populated) → uses cached promise\n\t * - Deduplication succeeds\n\t *\n\t * IMPORTANT: The delay must be non-zero. setTimeout(fn, 0) fails because JavaScript engines\n\t * may optimize zero-delay timers by batching them together, causing all requests to resume\n\t * simultaneously (same problem as microtasks). Any non-zero value (even 0.0000000001) forces\n\t * proper sequential task queue scheduling, ensuring each request gets its own task slot.\n\t */\n\tif (dedupeKey !== null) {\n\t\tawait waitFor(0.001);\n\t}\n\n\tconst prevRequestInfo = $RequestInfoCacheOrNull?.get(dedupeKey);\n\n\tconst getAbortErrorMessage = () => {\n\t\tif (globalOptions.dedupeKey) {\n\t\t\treturn `Duplicate request detected - Aborted previous request with key '${dedupeKey}'`;\n\t\t}\n\n\t\treturn `Duplicate request aborted - Aborted previous request to '${globalOptions.fullURL}'`;\n\t};\n\n\tconst handleRequestCancelStrategy = () => {\n\t\tconst shouldCancelRequest = prevRequestInfo && resolvedDedupeStrategy === \"cancel\";\n\n\t\tif (!shouldCancelRequest) return;\n\n\t\tconst message = getAbortErrorMessage();\n\n\t\tconst reason = new DOMException(message, \"AbortError\");\n\n\t\tprevRequestInfo.controller.abort(reason);\n\n\t\t// == Adding this just so that eslint forces me put await when calling the function (it looks better that way tbh)\n\t\treturn Promise.resolve();\n\t};\n\n\tconst handleRequestDeferStrategy = async (deferContext: {\n\t\tfetchApi: NonNullable<DedupeContext[\"options\"][\"customFetchImpl\"]>;\n\t\toptions: DedupeContext[\"options\"];\n\t\trequest: DedupeContext[\"request\"];\n\t}) => {\n\t\t// == Local options and request are needed so that transformations are applied can be applied to both from call site\n\t\tconst { fetchApi, options: localOptions, request: localRequest } = deferContext;\n\n\t\tconst shouldUsePromiseFromCache = prevRequestInfo && resolvedDedupeStrategy === \"defer\";\n\n\t\tconst streamableContext = {\n\t\t\tbaseConfig,\n\t\t\tconfig,\n\t\t\toptions: localOptions,\n\t\t\trequest: localRequest,\n\t\t} satisfies RequestContext;\n\n\t\tconst streamableRequest = await toStreamableRequest(streamableContext);\n\n\t\tconst responsePromise =\n\t\t\tshouldUsePromiseFromCache ?\n\t\t\t\tprevRequestInfo.responsePromise\n\t\t\t:\tfetchApi(localOptions.fullURL as NonNullable<typeof localOptions.fullURL>, streamableRequest);\n\n\t\t$RequestInfoCacheOrNull?.set(dedupeKey, { controller: newFetchController, responsePromise });\n\n\t\tconst response = await responsePromise;\n\n\t\treturn toStreamableResponse({ ...streamableContext, response });\n\t};\n\n\tconst removeDedupeKeyFromCache = () => {\n\t\t$RequestInfoCacheOrNull?.delete(dedupeKey);\n\t};\n\n\treturn {\n\t\tgetAbortErrorMessage,\n\t\thandleRequestCancelStrategy,\n\t\thandleRequestDeferStrategy,\n\t\tremoveDedupeKeyFromCache,\n\t\tresolvedDedupeStrategy,\n\t};\n};\n\ntype DedupeStrategyUnion = UnmaskType<\"cancel\" | \"defer\" | \"none\">;\n\ntype DedupeOptionKeys = Exclude<keyof DedupeOptions, \"dedupe\">;\n\ntype InnerDedupeOptions = {\n\t[Key in DedupeOptionKeys as RemovePrefix<\"dedupe\", Key>]?: DedupeOptions[Key];\n};\n\nexport type DedupeOptions = {\n\t/**\n\t * All dedupe options in a single object instead of separate properties\n\t */\n\tdedupe?: InnerDedupeOptions;\n\n\t/**\n\t * Controls the scope of request deduplication caching.\n\t *\n\t * - `\"global\"`: Shares deduplication cache across all `createFetchClient` instances with the same `dedupeCacheScopeKey`.\n\t * Useful for applications with multiple API clients that should share deduplication state.\n\t * - `\"local\"`: Limits deduplication to requests within the same `createFetchClient` instance.\n\t * Provides better isolation and is recommended for most use cases.\n\t *\n\t *\n\t * **Real-world Scenarios:**\n\t * - Use `\"global\"` when you have multiple API clients (user service, auth service, etc.) that might make overlapping requests\n\t * - Use `\"local\"` (default) for single-purpose clients or when you want strict isolation between different parts of your app\n\t *\n\t * @example\n\t * ```ts\n\t * // Local scope - each client has its own deduplication cache\n\t * const userClient = createFetchClient({ baseURL: \"/api/users\" });\n\t * const postClient = createFetchClient({ baseURL: \"/api/posts\" });\n\t * // These clients won't share deduplication state\n\t *\n\t * // Global scope - share cache across related clients\n\t * const userClient = createFetchClient({\n\t * baseURL: \"/api/users\",\n\t * dedupeCacheScope: \"global\",\n\t * });\n\t * const postClient = createFetchClient({\n\t * baseURL: \"/api/posts\",\n\t * dedupeCacheScope: \"global\",\n\t * });\n\t * // These clients will share deduplication state\n\t * ```\n\t *\n\t * @default \"local\"\n\t */\n\tdedupeCacheScope?: \"global\" | \"local\";\n\n\t/**\n\t * Unique namespace for the global deduplication cache when using `dedupeCacheScope: \"global\"`.\n\t *\n\t * This creates logical groupings of deduplication caches. All instances with the same key\n\t * will share the same cache namespace, allowing fine-grained control over which clients\n\t * share deduplication state.\n\t *\n\t * **Best Practices:**\n\t * - Use descriptive names that reflect the logical grouping (e.g., \"user-service\", \"analytics-api\")\n\t * - Keep scope keys consistent across related API clients\n\t * - Consider using different scope keys for different environments (dev, staging, prod)\n\t * - Avoid overly broad scope keys that might cause unintended cache sharing\n\t *\n\t * **Cache Management:**\n\t * - Each scope key maintains its own independent cache\n\t * - Caches are automatically cleaned up when no references remain\n\t * - Consider the memory implications of multiple global scopes\n\t *\n\t * @example\n\t * ```ts\n\t * // Group related API clients together\n\t * const userClient = createFetchClient({\n\t * baseURL: \"/api/users\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"user-service\"\n\t * });\n\t * const profileClient = createFetchClient({\n\t * baseURL: \"/api/profiles\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"user-service\" // Same scope - will share cache\n\t * });\n\t *\n\t * // Separate analytics client with its own cache\n\t * const analyticsClient = createFetchClient({\n\t * baseURL: \"/api/analytics\",\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: \"analytics-service\" // Different scope\n\t * });\n\t *\n\t * // Environment-specific scoping\n\t * const apiClient = createFetchClient({\n\t * dedupeCacheScope: \"global\",\n\t * dedupeCacheScopeKey: `api-${process.env.NODE_ENV}` // \"api-development\", \"api-production\", etc.\n\t * });\n\t * ```\n\t *\n\t * @default \"default\"\n\t */\n\tdedupeCacheScopeKey?: \"default\" | AnyString | ((context: RequestContext) => string | undefined);\n\n\t/**\n\t * Custom key generator for request deduplication.\n\t *\n\t * Override the default key generation strategy to control exactly which requests\n\t * are considered duplicates. The default key combines URL, method, body, and\n\t * relevant headers (excluding volatile ones like 'Date', 'Authorization', etc.).\n\t *\n\t * **Default Key Generation:**\n\t * The auto-generated key includes:\n\t * - Full request URL (including query parameters)\n\t * - HTTP method (GET, POST, etc.)\n\t * - Request body (for POST/PUT/PATCH requests)\n\t * - Stable headers (excludes Date, Authorization, User-Agent, etc.)\n\t *\n\t * **Custom Key Best Practices:**\n\t * - Include only the parts of the request that should affect deduplication\n\t * - Avoid including volatile data (timestamps, random IDs, etc.)\n\t * - Consider performance - simpler keys are faster to compute and compare\n\t * - Ensure keys are deterministic for the same logical request\n\t * - Use consistent key formats across your application\n\t *\n\t * **Performance Considerations:**\n\t * - Function-based keys are computed on every request - keep them lightweight\n\t * - String keys are fastest but least flexible\n\t * - Consider caching expensive key computations if needed\n\t *\n\t * @example\n\t * ```ts\n\t * import { callApi } from \"@zayne-labs/callapi\";\n\t *\n\t * // Simple static key - useful for singleton requests\n\t * const config = callApi(\"/api/config\", {\n\t * dedupeKey: \"app-config\",\n\t * dedupeStrategy: \"defer\" // Share the same config across all requests\n\t * });\n\t *\n\t * // URL and method only - ignore headers and body\n\t * const userData = callApi(\"/api/user/123\", {\n\t * dedupeKey: (context) => `${context.options.method}:${context.options.fullURL}`\n\t * });\n\t *\n\t * // Include specific headers in deduplication\n\t * const apiCall = callApi(\"/api/data\", {\n\t * dedupeKey: (context) => {\n\t * const authHeader = context.request.headers.get(\"Authorization\");\n\t * return `${context.options.fullURL}-${authHeader}`;\n\t * }\n\t * });\n\t *\n\t * // User-specific deduplication\n\t * const userSpecificCall = callApi(\"/api/dashboard\", {\n\t * dedupeKey: (context) => {\n\t * const userId = context.options.fullURL.match(/user\\/(\\d+)/)?.[1];\n\t * return `dashboard-${userId}`;\n\t * }\n\t * });\n\t *\n\t * // Ignore certain query parameters\n\t * const searchCall = callApi(\"/api/search?q=test&timestamp=123456\", {\n\t * dedupeKey: (context) => {\n\t * const url = new URL(context.options.fullURL);\n\t * url.searchParams.delete(\"timestamp\"); // Remove volatile param\n\t * return `search:${url.toString()}`;\n\t * }\n\t * });\n\t * ```\n\t *\n\t * @default Auto-generated from request details\n\t */\n\tdedupeKey?: string | ((context: RequestContext) => string | undefined);\n\n\t/**\n\t * Strategy for handling duplicate requests. Can be a static string or callback function.\n\t *\n\t * **Available Strategies:**\n\t * - `\"cancel\"`: Cancel previous request when new one starts (good for search)\n\t * - `\"defer\"`: Share response between duplicate requests (good for config loading)\n\t * - `\"none\"`: No deduplication, all requests execute independently\n\t *\n\t * @example\n\t * ```ts\n\t * // Static strategies\n\t * const searchClient = createFetchClient({\n\t * dedupeStrategy: \"cancel\" // Cancel previous searches\n\t * });\n\t *\n\t * const configClient = createFetchClient({\n\t * dedupeStrategy: \"defer\" // Share config across components\n\t * });\n\t *\n\t * // Dynamic strategy based on request\n\t * const smartClient = createFetchClient({\n\t * dedupeStrategy: (context) => {\n\t * return context.options.method === \"GET\" ? \"defer\" : \"cancel\";\n\t * }\n\t * });\n\t *\n\t * // Search-as-you-type with cancel strategy\n\t * const handleSearch = async (query: string) => {\n\t * try {\n\t * const { data } = await callApi(\"/api/search\", {\n\t * method: \"POST\",\n\t * body: { query },\n\t * dedupeStrategy: \"cancel\",\n\t * dedupeKey: \"search\" // Cancel previous searches, only latest one goes through\n\t * });\n\t *\n\t * updateSearchResults(data);\n\t * } catch (error) {\n\t * if (error.name === \"AbortError\") {\n\t * // Previous search cancelled - (expected behavior)\n\t * return;\n\t * }\n\t * console.error(\"Search failed:\", error);\n\t * }\n\t * };\n\t *\n\t * ```\n\t *\n\t * @default \"cancel\"\n\t */\n\tdedupeStrategy?: DedupeStrategyUnion | ((context: RequestContext) => DedupeStrategyUnion);\n};\n","import type { RequestContext } from \"./hooks\";\nimport type { CallApiContext } from \"./types/common\";\nimport type { DefaultCallApiContext } from \"./types/default-types\";\nimport type { UnmaskType } from \"./types/type-helpers\";\n\nexport type FetchImpl = UnmaskType<\n\t(input: string | Request | URL, init?: RequestInit) => Promise<Response>\n>;\n\nexport interface Middlewares<TCallApiContext extends CallApiContext = DefaultCallApiContext> {\n\t/**\n\t * Wraps the fetch implementation to intercept requests at the network layer.\n\t *\n\t * Takes a context object containing the current fetch function and returns a new fetch function.\n\t * Use it to cache responses, add logging, handle offline mode, or short-circuit requests etc.\n\t * Multiple middleware compose in order: plugins → base config → per-request.\n\t *\n\t * Unlike `customFetchImpl`, middleware can call through to the original fetch.\n\t *\n\t * @example\n\t * ```ts\n\t * // Cache responses\n\t * const cache = new Map();\n\t * fetchMiddleware: (ctx) => async (input, init) => {\n\t * const key = input.toString();\n\t * if (cache.has(key)) return cache.get(key).clone();\n\t *\n\t * const response = await ctx.fetchImpl(input, init);\n\t * cache.set(key, response.clone());\n\t * return response;\n\t * }\n\t *\n\t * // Handle offline\n\t * fetchMiddleware: (ctx) => async (input, init) => {\n\t * if (!navigator.onLine) {\n\t * return new Response('{\"error\": \"offline\"}', { status: 503 });\n\t * }\n\t * return ctx.fetchImpl(input, init);\n\t * }\n\t * ```\n\t */\n\tfetchMiddleware?: (context: RequestContext<TCallApiContext> & { fetchImpl: FetchImpl }) => FetchImpl;\n}\n\ntype MiddlewareRegistries = Required<{\n\t[Key in keyof Middlewares]: Set<Middlewares[Key]>;\n}>;\n\nexport const getMiddlewareRegistriesAndKeys = () => {\n\tconst middlewareRegistries: MiddlewareRegistries = {\n\t\tfetchMiddleware: new Set(),\n\t};\n\n\tconst middlewareRegistryKeys = Object.keys(middlewareRegistries) as Array<keyof Middlewares>;\n\n\treturn { middlewareRegistries, middlewareRegistryKeys };\n};\n\nexport const composeMiddlewaresFromArray = (\n\tmiddlewareArray: Array<Middlewares[keyof Middlewares] | undefined>\n) => {\n\tlet composedMiddleware: Middlewares[keyof Middlewares];\n\n\tfor (const currentMiddleware of middlewareArray) {\n\t\tif (!currentMiddleware) continue;\n\n\t\tconst previousMiddleware = composedMiddleware;\n\n\t\tif (!previousMiddleware) {\n\t\t\tcomposedMiddleware = currentMiddleware;\n\t\t\tcontinue;\n\t\t}\n\n\t\tcomposedMiddleware = (context) => {\n\t\t\tconst prevFetchImpl = previousMiddleware(context);\n\t\t\tconst fetchImpl = currentMiddleware({ ...context, fetchImpl: prevFetchImpl });\n\n\t\t\treturn fetchImpl;\n\t\t};\n\t}\n\n\treturn composedMiddleware;\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport {\n\tcomposeHooksFromArray,\n\tgetHookRegistriesAndKeys,\n\ttype Hooks,\n\ttype HooksOrHooksArray,\n\ttype RequestContext,\n} from \"./hooks\";\nimport {\n\tcomposeMiddlewaresFromArray,\n\tgetMiddlewareRegistriesAndKeys,\n\ttype Middlewares,\n} from \"./middlewares\";\nimport type {\n\tCallApiContext,\n\tCallApiRequestOptions,\n\tCallApiRequestOptionsForHooks,\n\tGetMergedCallApiContext,\n} from \"./types/common\";\nimport type { DefaultCallApiContext, DefaultDataType } from \"./types/default-types\";\nimport type { Awaitable } from \"./types/type-helpers\";\nimport type { InitURLOrURLObject } from \"./url\";\nimport { isArray, isFunction, isPlainObject, isString } from \"./utils/guards\";\nimport { type BaseCallApiSchemaAndConfig, getCurrentRouteSchemaKeyAndMainInitURL } from \"./validation\";\n\nexport type PluginSetupContext<TCallApiContext extends CallApiContext = DefaultCallApiContext> =\n\tRequestContext<TCallApiContext> & { initURL: string };\n\nexport type PluginInitResult<TCallApiContext extends CallApiContext = DefaultCallApiContext> = Partial<\n\tOmit<PluginSetupContext<TCallApiContext>, \"initURL\" | \"request\"> & {\n\t\tinitURL: InitURLOrURLObject;\n\t\trequest: CallApiRequestOptions;\n\t}\n>;\n\nexport type PluginHooks<TCallApiContext extends CallApiContext = DefaultCallApiContext> =\n\tHooksOrHooksArray<\n\t\tGetMergedCallApiContext<\n\t\t\tTCallApiContext,\n\t\t\t{\n\t\t\t\tData: DefaultDataType extends TCallApiContext[\"Data\"] ? never : TCallApiContext[\"Data\"];\n\t\t\t\tErrorData: DefaultDataType extends TCallApiContext[\"ErrorData\"] ? never\n\t\t\t\t:\tTCallApiContext[\"ErrorData\"];\n\t\t\t}\n\t\t>\n\t>;\n\nexport interface CallApiPlugin<TCallApiContext extends CallApiContext = DefaultCallApiContext> {\n\t/**\n\t * Defines additional options that can be passed to callApi\n\t */\n\tdefineExtraOptions?: (...params: never[]) => unknown;\n\n\t/**\n\t * A description for the plugin\n\t */\n\tdescription?: string;\n\n\t/**\n\t * Hooks for the plugin\n\t */\n\thooks?:\n\t\t| PluginHooks<TCallApiContext>\n\t\t| ((context: PluginSetupContext<TCallApiContext>) => Awaitable<PluginHooks<TCallApiContext>>);\n\n\t/**\n\t * A unique id for the plugin\n\t */\n\tid: string;\n\n\t/**\n\t * Middlewares that for the plugin\n\t */\n\tmiddlewares?:\n\t\t| Middlewares<TCallApiContext>\n\t\t| ((context: PluginSetupContext<TCallApiContext>) => Awaitable<Middlewares<TCallApiContext>>);\n\n\t/**\n\t * A name for the plugin\n\t */\n\tname: string;\n\n\t/**\n\t * Base schema for the client.\n\t */\n\tschema?: BaseCallApiSchemaAndConfig;\n\n\t/**\n\t * A function that will be called when the plugin is initialized. This will be called before the any of the other internal functions.\n\t */\n\tsetup?: (\n\t\tcontext: PluginSetupContext<TCallApiContext>\n\t) => Awaitable<PluginInitResult<TCallApiContext>> | Awaitable<void>;\n\n\t/**\n\t * A version for the plugin\n\t */\n\tversion?: string;\n}\n\nexport const getResolvedPlugins = (context: Pick<RequestContext, \"baseConfig\" | \"options\">) => {\n\tconst { baseConfig, options } = context;\n\n\tconst resolvedPlugins =\n\t\tisFunction(options.plugins) ?\n\t\t\toptions.plugins({ basePlugins: baseConfig.plugins ?? [] })\n\t\t:\t(options.plugins ?? []);\n\n\treturn resolvedPlugins;\n};\n\nexport const initializePlugins = async (context: PluginSetupContext) => {\n\tconst { baseConfig, config, initURL, options, request } = context;\n\n\tconst {\n\t\taddMainHooks,\n\t\taddMainMiddlewares,\n\t\taddPluginHooks,\n\t\taddPluginMiddlewares,\n\t\tgetResolvedHooks,\n\t\tgetResolvedMiddlewares,\n\t} = setupHooksAndMiddlewares({ baseConfig, config, options });\n\n\tconst { currentRouteSchemaKey, mainInitURL } = getCurrentRouteSchemaKeyAndMainInitURL({\n\t\tbaseExtraOptions: baseConfig,\n\t\textraOptions: config,\n\t\tinitURL,\n\t});\n\n\tlet resolvedCurrentRouteSchemaKey = currentRouteSchemaKey;\n\tlet resolvedInitURL = mainInitURL;\n\tlet resolvedOptions = options;\n\tlet resolvedRequestOptions = request;\n\n\tconst executePluginSetupFn = async (pluginSetup: CallApiPlugin[\"setup\"]) => {\n\t\tif (!pluginSetup) return;\n\n\t\tconst initResult = await pluginSetup(context);\n\n\t\tif (!isPlainObject(initResult)) return;\n\n\t\tconst urlString = initResult.initURL?.toString();\n\n\t\tif (isString(urlString)) {\n\t\t\tconst newResult = getCurrentRouteSchemaKeyAndMainInitURL({\n\t\t\t\tbaseExtraOptions: baseConfig,\n\t\t\t\textraOptions: config,\n\t\t\t\tinitURL: urlString,\n\t\t\t});\n\n\t\t\tresolvedCurrentRouteSchemaKey = newResult.currentRouteSchemaKey;\n\t\t\tresolvedInitURL = newResult.mainInitURL;\n\t\t}\n\n\t\tif (isPlainObject(initResult.request)) {\n\t\t\tresolvedRequestOptions = {\n\t\t\t\t...resolvedRequestOptions,\n\t\t\t\t...(initResult.request as CallApiRequestOptionsForHooks),\n\t\t\t};\n\t\t}\n\n\t\tif (isPlainObject(initResult.options)) {\n\t\t\tresolvedOptions = { ...resolvedOptions, ...initResult.options };\n\t\t}\n\t};\n\n\tconst resolvedPlugins = getResolvedPlugins({ baseConfig, options });\n\n\tfor (const plugin of resolvedPlugins) {\n\t\t// eslint-disable-next-line no-await-in-loop -- Await is necessary in this case.\n\t\tconst [, pluginHooks, pluginMiddlewares] = await Promise.all([\n\t\t\texecutePluginSetupFn(plugin.setup),\n\t\t\tisFunction(plugin.hooks) ? plugin.hooks(context) : plugin.hooks,\n\t\t\tisFunction(plugin.middlewares) ? plugin.middlewares(context) : plugin.middlewares,\n\t\t]);\n\n\t\tpluginHooks && addPluginHooks(pluginHooks);\n\t\tpluginMiddlewares && addPluginMiddlewares(pluginMiddlewares);\n\t}\n\n\taddMainHooks();\n\n\taddMainMiddlewares();\n\n\tconst resolvedHooks = getResolvedHooks();\n\n\tconst resolvedMiddlewares = getResolvedMiddlewares();\n\n\treturn {\n\t\tresolvedCurrentRouteSchemaKey,\n\t\tresolvedHooks,\n\t\tresolvedInitURL,\n\t\tresolvedMiddlewares,\n\t\tresolvedOptions,\n\t\tresolvedRequestOptions,\n\t};\n};\n\nconst setupHooksAndMiddlewares = (\n\tcontext: Pick<PluginSetupContext, \"baseConfig\" | \"config\" | \"options\">\n) => {\n\tconst { baseConfig, config, options } = context;\n\n\tconst { hookRegistries, hookRegistryKeys } = getHookRegistriesAndKeys();\n\n\tconst { middlewareRegistries, middlewareRegistryKeys } = getMiddlewareRegistriesAndKeys();\n\n\tconst addMainHooks = () => {\n\t\tfor (const hookName of hookRegistryKeys) {\n\t\t\tconst overriddenHook = options[hookName];\n\t\t\tconst baseHook = baseConfig[hookName];\n\t\t\tconst instanceHook = config[hookName];\n\n\t\t\tconst shouldMergeBaseAndInstanceHooks = isArray(baseHook) && instanceHook;\n\n\t\t\tconst mainHook =\n\t\t\t\tshouldMergeBaseAndInstanceHooks ? [baseHook, instanceHook].flat() : overriddenHook;\n\n\t\t\tmainHook && hookRegistries[hookName].add(mainHook as never);\n\t\t}\n\t};\n\n\tconst addPluginHooks = (pluginHooks: PluginHooks) => {\n\t\tfor (const hookName of hookRegistryKeys) {\n\t\t\tconst pluginHook = pluginHooks[hookName];\n\n\t\t\tpluginHook && hookRegistries[hookName].add(pluginHook as never);\n\t\t}\n\t};\n\n\tconst addMainMiddlewares = () => {\n\t\tfor (const middlewareName of middlewareRegistryKeys) {\n\t\t\tconst baseMiddleware = baseConfig[middlewareName];\n\t\t\tconst instanceMiddleware = config[middlewareName];\n\n\t\t\tbaseMiddleware && middlewareRegistries[middlewareName].add(baseMiddleware);\n\n\t\t\tinstanceMiddleware && middlewareRegistries[middlewareName].add(instanceMiddleware);\n\t\t}\n\t};\n\n\tconst addPluginMiddlewares = (pluginMiddlewares: Middlewares) => {\n\t\tfor (const middlewareName of middlewareRegistryKeys) {\n\t\t\tconst pluginMiddleware = pluginMiddlewares[middlewareName];\n\n\t\t\tif (!pluginMiddleware) continue;\n\n\t\t\tmiddlewareRegistries[middlewareName].add(pluginMiddleware);\n\t\t}\n\t};\n\n\tconst getResolvedHooks = () => {\n\t\tconst resolvedHooks: Hooks = {};\n\n\t\tfor (const [hookName, hookRegistry] of Object.entries(hookRegistries)) {\n\t\t\tif (hookRegistry.size === 0) continue;\n\n\t\t\t// == Flatten the hook registry to remove any nested arrays, incase an array of hooks is passed to any of the hooks\n\t\t\tconst flattenedHookArray = [...hookRegistry].flat();\n\n\t\t\tif (flattenedHookArray.length === 0) continue;\n\n\t\t\tconst hooksExecutionMode = options.hooksExecutionMode ?? extraOptionDefaults.hooksExecutionMode;\n\n\t\t\tconst composedHook = composeHooksFromArray(flattenedHookArray, hooksExecutionMode);\n\n\t\t\tresolvedHooks[hookName as keyof Hooks] = composedHook;\n\t\t}\n\n\t\treturn resolvedHooks;\n\t};\n\n\tconst getResolvedMiddlewares = () => {\n\t\tconst resolvedMiddlewares: Middlewares = {};\n\n\t\tfor (const [middlewareName, middlewareRegistry] of Object.entries(middlewareRegistries)) {\n\t\t\tif (middlewareRegistry.size === 0) continue;\n\n\t\t\tconst middlewareArray = [...middlewareRegistry];\n\n\t\t\tif (middlewareArray.length === 0) continue;\n\n\t\t\tconst composedMiddleware = composeMiddlewaresFromArray(middlewareArray);\n\n\t\t\tresolvedMiddlewares[middlewareName as keyof Middlewares] = composedMiddleware;\n\t\t}\n\n\t\treturn resolvedMiddlewares;\n\t};\n\n\treturn {\n\t\taddMainHooks,\n\t\taddMainMiddlewares,\n\t\taddPluginHooks,\n\t\taddPluginMiddlewares,\n\t\tgetResolvedHooks,\n\t\tgetResolvedMiddlewares,\n\t};\n};\n","import { extraOptionDefaults } from \"./constants/defaults\";\nimport {\n\texecuteHooksInCatchBlock,\n\ttype ErrorContext,\n\ttype ExecuteHookInfo,\n\ttype RequestContext,\n\ttype RetryContext,\n} from \"./hooks\";\nimport type { CallApiResultErrorVariant } from \"./result\";\nimport type { CallApiConfig } from \"./types/common\";\nimport type { MethodUnion } from \"./types/conditional-types\";\nimport {\n\tdefineEnum,\n\ttype AnyNumber,\n\ttype Awaitable,\n\ttype RemovePrefix,\n\ttype UnmaskType,\n} from \"./types/type-helpers\";\nimport type { InitURLOrURLObject } from \"./url\";\nimport { waitFor } from \"./utils/common\";\nimport { isBoolean, isFunction, isString } from \"./utils/guards\";\n\n// eslint-disable-next-line ts-eslint/no-unused-vars -- Ignore\nconst defaultRetryStatusCodesLookup = () =>\n\tdefineEnum({\n\t\t408: \"Request Timeout\",\n\t\t409: \"Conflict\",\n\t\t425: \"Too Early\",\n\t\t429: \"Too Many Requests\",\n\t\t500: \"Internal Server Error\",\n\t\t502: \"Bad Gateway\",\n\t\t503: \"Service Unavailable\",\n\t\t504: \"Gateway Timeout\",\n\t});\n\ntype RetryStatusCodes = UnmaskType<AnyNumber | keyof ReturnType<typeof defaultRetryStatusCodesLookup>>;\n\ntype RetryCondition<TErrorData> = (context: ErrorContext<{ ErrorData: TErrorData }>) => Awaitable<boolean>;\n\ntype RetryOptionKeys<TErrorData> = Exclude<keyof RetryOptions<TErrorData>, \"~retryAttemptCount\" | \"retry\">;\n\ntype InnerRetryOptions<TErrorData> = {\n\t[Key in RetryOptionKeys<TErrorData> as RemovePrefix<\"retry\", Key>]?: RetryOptions<TErrorData>[Key];\n};\n\nexport interface RetryOptions<TErrorData> {\n\t/**\n\t * Keeps track of the number of times the request has already been retried\n\t * @internal\n\t * @deprecated **NOTE**: This property is used internally to track retries. Please abstain from modifying it.\n\t */\n\treadonly [\"~retryAttemptCount\"]?: number;\n\n\t/**\n\t * All retry options in a single object instead of separate properties\n\t */\n\tretry?: InnerRetryOptions<TErrorData>;\n\n\t/**\n\t * Number of allowed retry attempts on HTTP errors\n\t * @default 0\n\t */\n\tretryAttempts?: number;\n\n\t/**\n\t * Callback whose return value determines if a request should be retried or not\n\t */\n\tretryCondition?: RetryCondition<TErrorData>;\n\n\t/**\n\t * Delay between retries in milliseconds\n\t * @default 1000\n\t */\n\tretryDelay?: number | ((currentAttemptCount: number) => number);\n\n\t/**\n\t * Maximum delay in milliseconds. Only applies to exponential strategy\n\t * @default 10000\n\t */\n\tretryMaxDelay?: number;\n\n\t/**\n\t * HTTP methods that are allowed to retry\n\t * @default [\"GET\", \"POST\"]\n\t */\n\tretryMethods?: MethodUnion[];\n\n\t/**\n\t * HTTP status codes that trigger a retry\n\t */\n\tretryStatusCodes?: RetryStatusCodes[];\n\n\t/**\n\t * Strategy to use when retrying\n\t * @default \"linear\"\n\t */\n\tretryStrategy?: \"exponential\" | \"linear\";\n}\n\nconst getLinearDelay = (currentAttemptCount: number, options: RetryOptions<unknown>) => {\n\tconst retryDelay = options.retryDelay ?? options.retry?.delay;\n\n\tconst resolveRetryDelay =\n\t\t(isFunction(retryDelay) ? retryDelay(currentAttemptCount) : retryDelay)\n\t\t?? extraOptionDefaults.retryDelay;\n\n\treturn resolveRetryDelay;\n};\n\nconst getExponentialDelay = (currentAttemptCount: number, options: RetryOptions<unknown>) => {\n\tconst retryDelay = options.retryDelay ?? options.retry?.delay ?? extraOptionDefaults.retryDelay;\n\n\tconst resolvedRetryDelay = isFunction(retryDelay) ? retryDelay(currentAttemptCount) : retryDelay;\n\n\tconst maxDelay = options.retryMaxDelay ?? options.retry?.maxDelay ?? extraOptionDefaults.retryMaxDelay;\n\n\tconst exponentialDelay = resolvedRetryDelay * 2 ** currentAttemptCount;\n\n\treturn Math.min(exponentialDelay, maxDelay);\n};\n\nexport const createRetryManager = (ctx: ErrorContext<{ ErrorData: unknown }> & RequestContext) => {\n\tconst { options, request } = ctx;\n\n\t// eslint-disable-next-line ts-eslint/no-deprecated -- Allowed for internal use\n\tconst currentAttemptCount = options[\"~retryAttemptCount\"] ?? 1;\n\n\tconst retryStrategy =\n\t\toptions.retryStrategy ?? options.retry?.strategy ?? extraOptionDefaults.retryStrategy;\n\n\tconst getDelay = () => {\n\t\tswitch (retryStrategy) {\n\t\t\tcase \"exponential\": {\n\t\t\t\treturn getExponentialDelay(currentAttemptCount, options);\n\t\t\t}\n\t\t\tcase \"linear\": {\n\t\t\t\treturn getLinearDelay(currentAttemptCount, options);\n\t\t\t}\n\t\t\tdefault: {\n\t\t\t\tthrow new Error(`Invalid retry strategy: ${String(retryStrategy)}`);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst shouldAttemptRetry = async () => {\n\t\tif (isBoolean(request.signal) && request.signal.aborted) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst retryCondition =\n\t\t\toptions.retryCondition ?? options.retry?.condition ?? extraOptionDefaults.retryCondition;\n\n\t\tconst maximumRetryAttempts =\n\t\t\toptions.retryAttempts ?? options.retry?.attempts ?? extraOptionDefaults.retryAttempts;\n\n\t\tconst customRetryCondition = await retryCondition(ctx);\n\n\t\tconst baseShouldRetry = currentAttemptCount <= maximumRetryAttempts && customRetryCondition;\n\n\t\tif (!baseShouldRetry) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst retryMethods = new Set(\n\t\t\toptions.retryMethods ?? options.retry?.methods ?? extraOptionDefaults.retryMethods\n\t\t);\n\n\t\tconst includesMethod =\n\t\t\tisString(ctx.request.method) && retryMethods.size > 0 ?\n\t\t\t\tretryMethods.has(ctx.request.method)\n\t\t\t:\ttrue;\n\n\t\tconst retryStatusCodes = new Set(\n\t\t\toptions.retryStatusCodes ?? options.retry?.statusCodes ?? extraOptionDefaults.retryStatusCodes\n\t\t);\n\n\t\tconst includesStatusCodes =\n\t\t\tctx.response != null && retryStatusCodes.size > 0 ?\n\t\t\t\tretryStatusCodes.has(ctx.response.status)\n\t\t\t:\ttrue;\n\n\t\tconst shouldRetry = includesMethod && includesStatusCodes;\n\n\t\treturn shouldRetry;\n\t};\n\n\ttype CallApiImpl = (\n\t\tinitURL: never,\n\t\tinit?: CallApiConfig\n\t) => Promise<CallApiResultErrorVariant<unknown>>;\n\n\tconst handleRetry = async <TCallapi extends CallApiImpl>(context: {\n\t\tcallApi: TCallapi;\n\t\tcallApiArgs: { config: CallApiConfig; initURL: InitURLOrURLObject };\n\t\terrorContext: ErrorContext<{ ErrorData: unknown }>;\n\t\thookInfo: ExecuteHookInfo;\n\t}) => {\n\t\tconst { callApi, callApiArgs, errorContext, hookInfo } = context;\n\n\t\tconst retryContext = {\n\t\t\t...errorContext,\n\t\t\tretryAttemptCount: currentAttemptCount,\n\t\t} satisfies RetryContext<{ ErrorData: unknown }>;\n\n\t\tconst hookError = await executeHooksInCatchBlock([options.onRetry?.(retryContext)], hookInfo);\n\n\t\tif (hookError) {\n\t\t\treturn hookError;\n\t\t}\n\n\t\tawait waitFor(getDelay());\n\n\t\tconst updatedOptions = {\n\t\t\t...callApiArgs.config,\n\t\t\t\"~retryAttemptCount\": currentAttemptCount + 1,\n\t\t} satisfies CallApiConfig;\n\n\t\treturn callApi(callApiArgs.initURL as never, updatedOptions);\n\t};\n\n\treturn {\n\t\thandleRetry,\n\t\tshouldAttemptRetry,\n\t};\n};\n","import { createDedupeStrategy, type GlobalRequestInfoCache, type RequestInfoCache } from \"./dedupe\";\nimport {\n\ttype ErrorContext,\n\ttype ExecuteHookInfo,\n\texecuteHooks,\n\texecuteHooksInCatchBlock,\n\ttype RequestContext,\n\ttype SuccessContext,\n} from \"./hooks\";\nimport { type CallApiPlugin, initializePlugins } from \"./plugins\";\nimport {\n\ttype ErrorInfo,\n\tgetCustomizedErrorResult,\n\ttype GetResponseType,\n\tresolveErrorResult,\n\tresolveResponseData,\n\tresolveSuccessResult,\n\ttype ResponseTypeType,\n\ttype ResultModeType,\n} from \"./result\";\nimport { createRetryManager } from \"./retry\";\nimport type {\n\tBaseCallApiConfig,\n\tCallApiConfig,\n\tCallApiContext,\n\tCallApiExtraOptionsForHooks,\n\tCallApiRequestOptions,\n\tCallApiRequestOptionsForHooks,\n\tCallApiResult,\n\tGetBaseSchemaConfig,\n\tGetBaseSchemaRoutes,\n\tGetMergedCallApiContext,\n} from \"./types/common\";\nimport type {\n\tGetCurrentRouteSchema,\n\tGetCurrentRouteSchemaKey,\n\tInferHeadersOption,\n\tInferInitURL,\n\tThrowOnErrorUnion,\n} from \"./types/conditional-types\";\nimport type {\n\tDefaultCallApiContext,\n\tDefaultPluginArray,\n\tDefaultThrowOnError,\n} from \"./types/default-types\";\nimport type { AnyFunction } from \"./types/type-helpers\";\nimport { getFullAndNormalizedURL } from \"./url\";\nimport {\n\tcreateCombinedSignal,\n\tcreateTimeoutSignal,\n\tgetBody,\n\tgetFetchImpl,\n\tgetHeaders,\n\tgetMethod,\n\tsplitBaseConfig,\n\tsplitConfig,\n} from \"./utils/common\";\nimport { HTTPError } from \"./utils/external/error\";\nimport { isHTTPErrorInstance, isValidationErrorInstance } from \"./utils/external/guards\";\nimport { isFunction } from \"./utils/guards\";\nimport {\n\ttype BaseCallApiSchemaAndConfig,\n\ttype BaseCallApiSchemaRoutes,\n\ttype CallApiSchema,\n\ttype CallApiSchemaConfig,\n\thandleConfigValidation,\n\thandleSchemaValidation,\n\ttype InferSchemaOutput,\n} from \"./validation\";\n\nconst $GlobalRequestInfoCache: GlobalRequestInfoCache = new Map();\n\nexport const createFetchClientWithContext = <\n\tTOuterCallApiContext extends CallApiContext = DefaultCallApiContext,\n>() => {\n\tconst createFetchClient = <\n\t\tTBaseCallApiContext extends CallApiContext = TOuterCallApiContext,\n\t\tTBaseData = TBaseCallApiContext[\"Data\"],\n\t\tTBaseErrorData = TBaseCallApiContext[\"ErrorData\"],\n\t\tTBaseResultMode extends ResultModeType = TBaseCallApiContext[\"ResultMode\"] extends ResultModeType ?\n\t\t\tTBaseCallApiContext[\"ResultMode\"]\n\t\t:\tDefaultCallApiContext[\"ResultMode\"],\n\t\tTBaseThrowOnError extends ThrowOnErrorUnion = DefaultThrowOnError,\n\t\tTBaseResponseType extends ResponseTypeType = ResponseTypeType,\n\t\tconst TBaseSchemaAndConfig extends BaseCallApiSchemaAndConfig = BaseCallApiSchemaAndConfig,\n\t\tconst TBasePluginArray extends CallApiPlugin[] = DefaultPluginArray,\n\t\tTComputedBaseCallApiContext extends CallApiContext = GetMergedCallApiContext<\n\t\t\tTBaseCallApiContext,\n\t\t\t{ Data: TBaseData; ErrorData: TBaseErrorData; ResultMode: TBaseResultMode }\n\t\t>,\n\t\tTComputedBaseSchemaConfig extends CallApiSchemaConfig = GetBaseSchemaConfig<TBaseSchemaAndConfig>,\n\t\tTComputedBaseSchemaRoutes extends BaseCallApiSchemaRoutes =\n\t\t\tGetBaseSchemaRoutes<TBaseSchemaAndConfig>,\n\t>(\n\t\tinitBaseConfig: BaseCallApiConfig<\n\t\t\tTBaseCallApiContext,\n\t\t\tTBaseData,\n\t\t\tTBaseErrorData,\n\t\t\tTBaseResultMode,\n\t\t\tTBaseThrowOnError,\n\t\t\tTBaseResponseType,\n\t\t\tTBaseSchemaAndConfig,\n\t\t\tTBasePluginArray\n\t\t> = {} as never\n\t) => {\n\t\tconst $LocalRequestInfoCache: RequestInfoCache = new Map();\n\n\t\tconst callApi = async <\n\t\t\tTData = TBaseData,\n\t\t\tTErrorData = TBaseErrorData,\n\t\t\tTResultMode extends ResultModeType = TBaseResultMode,\n\t\t\tTCallApiContext extends CallApiContext = GetMergedCallApiContext<\n\t\t\t\tTComputedBaseCallApiContext,\n\t\t\t\t{ Data: TData; ErrorData: TErrorData; ResultMode: TResultMode }\n\t\t\t>,\n\t\t\tTThrowOnError extends ThrowOnErrorUnion = TBaseThrowOnError,\n\t\t\tTResponseType extends ResponseTypeType = TBaseResponseType,\n\t\t\tconst TSchemaConfig extends CallApiSchemaConfig = TComputedBaseSchemaConfig,\n\t\t\tTInitURL extends InferInitURL<TComputedBaseSchemaRoutes, TSchemaConfig> = InferInitURL<\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTSchemaConfig\n\t\t\t>,\n\t\t\tTCurrentRouteSchemaKey extends GetCurrentRouteSchemaKey<TSchemaConfig, TInitURL> =\n\t\t\t\tGetCurrentRouteSchemaKey<TSchemaConfig, TInitURL>,\n\t\t\tconst TSchema extends CallApiSchema = GetCurrentRouteSchema<\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTCurrentRouteSchemaKey\n\t\t\t>,\n\t\t\tconst TPluginArray extends CallApiPlugin[] = TBasePluginArray,\n\t\t\tTComputedResult = CallApiResult<\n\t\t\t\tInferSchemaOutput<TSchema[\"data\"], TData>,\n\t\t\t\tInferSchemaOutput<TSchema[\"errorData\"], TErrorData>,\n\t\t\t\tTResultMode,\n\t\t\t\tTThrowOnError,\n\t\t\t\tTResponseType\n\t\t\t>,\n\t\t>(\n\t\t\tinitURL: TInitURL,\n\t\t\tinitConfig: CallApiConfig<\n\t\t\t\tTCallApiContext,\n\t\t\t\tInferSchemaOutput<TSchema[\"data\"], GetResponseType<TData, TResponseType>>,\n\t\t\t\tInferSchemaOutput<TSchema[\"errorData\"], GetResponseType<TErrorData, TResponseType>>,\n\t\t\t\tTResultMode,\n\t\t\t\tTThrowOnError,\n\t\t\t\tTResponseType,\n\t\t\t\tTComputedBaseSchemaRoutes,\n\t\t\t\tTSchema,\n\t\t\t\tTComputedBaseSchemaConfig,\n\t\t\t\tTSchemaConfig,\n\t\t\t\tTInitURL,\n\t\t\t\tTCurrentRouteSchemaKey,\n\t\t\t\tTBasePluginArray,\n\t\t\t\tTPluginArray\n\t\t\t> = {} as never\n\t\t): Promise<TComputedResult> => {\n\t\t\tconst [fetchOptions, extraOptions] = splitConfig(initConfig);\n\n\t\t\tconst resolvedBaseConfig =\n\t\t\t\tisFunction(initBaseConfig) ?\n\t\t\t\t\tinitBaseConfig({\n\t\t\t\t\t\tinitURL: initURL.toString(),\n\t\t\t\t\t\toptions: extraOptions,\n\t\t\t\t\t\trequest: fetchOptions,\n\t\t\t\t\t})\n\t\t\t\t:\tinitBaseConfig;\n\n\t\t\tconst baseConfig = resolvedBaseConfig as BaseCallApiConfig;\n\t\t\tconst config = initConfig as CallApiConfig;\n\n\t\t\tconst [baseFetchOptions, baseExtraOptions] = splitBaseConfig(baseConfig);\n\n\t\t\tconst shouldSkipAutoMergeForOptions =\n\t\t\t\tbaseExtraOptions.skipAutoMergeFor === \"all\" || baseExtraOptions.skipAutoMergeFor === \"options\";\n\n\t\t\tconst shouldSkipAutoMergeForRequest =\n\t\t\t\tbaseExtraOptions.skipAutoMergeFor === \"all\" || baseExtraOptions.skipAutoMergeFor === \"request\";\n\n\t\t\t// == Merged Extra Options\n\t\t\tconst mergedExtraOptions = {\n\t\t\t\t...baseExtraOptions,\n\t\t\t\t...(!shouldSkipAutoMergeForOptions && extraOptions),\n\t\t\t};\n\n\t\t\t// == Merged Request Options\n\t\t\tconst mergedRequestOptions = {\n\t\t\t\theaders: {}, // == Making sure headers is always an object\n\t\t\t\t...baseFetchOptions,\n\t\t\t\t...(!shouldSkipAutoMergeForRequest && fetchOptions),\n\t\t\t} satisfies CallApiRequestOptions;\n\n\t\t\tconst {\n\t\t\t\tresolvedCurrentRouteSchemaKey,\n\t\t\t\tresolvedHooks,\n\t\t\t\tresolvedInitURL,\n\t\t\t\tresolvedMiddlewares,\n\t\t\t\tresolvedOptions,\n\t\t\t\tresolvedRequestOptions,\n\t\t\t} = await initializePlugins({\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tinitURL: initURL.toString(),\n\t\t\t\toptions: mergedExtraOptions as CallApiExtraOptionsForHooks,\n\t\t\t\trequest: mergedRequestOptions as CallApiRequestOptionsForHooks,\n\t\t\t});\n\n\t\t\tconst { fullURL, normalizedInitURL } = getFullAndNormalizedURL({\n\t\t\t\tbaseURL: resolvedOptions.baseURL,\n\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\tparams: resolvedOptions.params,\n\t\t\t\tquery: resolvedOptions.query,\n\t\t\t});\n\n\t\t\tconst options = {\n\t\t\t\t...resolvedOptions,\n\t\t\t\t...resolvedHooks,\n\t\t\t\t...resolvedMiddlewares,\n\n\t\t\t\tfullURL,\n\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\tinitURLNormalized: normalizedInitURL,\n\t\t\t} satisfies CallApiExtraOptionsForHooks;\n\n\t\t\tconst newFetchController = new AbortController();\n\n\t\t\tconst timeoutSignal = createTimeoutSignal(options.timeout);\n\n\t\t\tconst combinedSignal = createCombinedSignal(\n\t\t\t\ttimeoutSignal,\n\t\t\t\tresolvedRequestOptions.signal,\n\t\t\t\tnewFetchController.signal\n\t\t\t);\n\n\t\t\tconst initMethod = getMethod({ initURL: resolvedInitURL, method: resolvedRequestOptions.method });\n\n\t\t\tconst request = {\n\t\t\t\t...resolvedRequestOptions,\n\n\t\t\t\tmethod: initMethod,\n\t\t\t\tsignal: combinedSignal,\n\t\t\t} satisfies CallApiRequestOptionsForHooks;\n\n\t\t\tconst {\n\t\t\t\tgetAbortErrorMessage,\n\t\t\t\thandleRequestCancelStrategy,\n\t\t\t\thandleRequestDeferStrategy,\n\t\t\t\tremoveDedupeKeyFromCache,\n\t\t\t\tresolvedDedupeStrategy,\n\t\t\t} = await createDedupeStrategy({\n\t\t\t\t$GlobalRequestInfoCache,\n\t\t\t\t$LocalRequestInfoCache,\n\t\t\t\tbaseConfig,\n\t\t\t\tconfig,\n\t\t\t\tnewFetchController,\n\t\t\t\toptions,\n\t\t\t\trequest,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tawait handleRequestCancelStrategy();\n\n\t\t\t\tawait executeHooks(options.onRequest?.({ baseConfig, config, options, request }));\n\n\t\t\t\tconst {\n\t\t\t\t\textraOptionsValidationResult,\n\t\t\t\t\trequestOptionsValidationResult,\n\t\t\t\t\tresolvedSchema,\n\t\t\t\t\tresolvedSchemaConfig,\n\t\t\t\t\tshouldApplySchemaOutput,\n\t\t\t\t} = await handleConfigValidation({\n\t\t\t\t\tbaseExtraOptions,\n\t\t\t\t\tcurrentRouteSchemaKey: resolvedCurrentRouteSchemaKey,\n\t\t\t\t\textraOptions,\n\t\t\t\t\toptions,\n\t\t\t\t\trequestOptions: request,\n\t\t\t\t});\n\n\t\t\t\t// == Apply Schema Output for Extra Options\n\t\t\t\tif (shouldApplySchemaOutput) {\n\t\t\t\t\tObject.assign(options, extraOptionsValidationResult);\n\t\t\t\t}\n\n\t\t\t\t// == Apply Schema Output for Request Options\n\t\t\t\tconst validMethod = getMethod({\n\t\t\t\t\tinitURL: resolvedInitURL,\n\t\t\t\t\tmethod: shouldApplySchemaOutput ? requestOptionsValidationResult?.method : request.method,\n\t\t\t\t});\n\n\t\t\t\tconst validBody = getBody({\n\t\t\t\t\tbody: shouldApplySchemaOutput ? requestOptionsValidationResult?.body : request.body,\n\t\t\t\t\tbodySerializer: options.bodySerializer,\n\t\t\t\t});\n\n\t\t\t\ttype HeaderFn = Extract<InferHeadersOption<CallApiSchema>[\"headers\"], AnyFunction>;\n\n\t\t\t\tconst resolvedHeaders =\n\t\t\t\t\tisFunction<HeaderFn>(fetchOptions.headers) ?\n\t\t\t\t\t\tfetchOptions.headers({ baseHeaders: baseFetchOptions.headers ?? {} })\n\t\t\t\t\t:\t(fetchOptions.headers ?? baseFetchOptions.headers);\n\n\t\t\t\tconst validHeaders = await getHeaders({\n\t\t\t\t\tauth: options.auth,\n\t\t\t\t\tbody: validBody,\n\t\t\t\t\theaders:\n\t\t\t\t\t\tshouldApplySchemaOutput ? requestOptionsValidationResult?.headers : resolvedHeaders,\n\t\t\t\t});\n\n\t\t\t\tObject.assign(request, {\n\t\t\t\t\t...(validBody && { body: validBody }),\n\t\t\t\t\t...(validHeaders && { headers: validHeaders }),\n\t\t\t\t\t...(validMethod && { method: validMethod }),\n\t\t\t\t});\n\n\t\t\t\tconst readyRequestContext = { baseConfig, config, options, request } satisfies RequestContext;\n\n\t\t\t\tawait executeHooks(options.onRequestReady?.(readyRequestContext));\n\n\t\t\t\tconst fetchApi = getFetchImpl({\n\t\t\t\t\tcustomFetchImpl: options.customFetchImpl,\n\t\t\t\t\tfetchMiddleware: options.fetchMiddleware,\n\t\t\t\t\trequestContext: readyRequestContext,\n\t\t\t\t});\n\n\t\t\t\tconst response = await handleRequestDeferStrategy({ fetchApi, options, request });\n\n\t\t\t\t// == Also clone response when dedupeStrategy is set to \"defer\" to avoid error thrown from reading response.(whatever) more than once\n\t\t\t\tconst shouldCloneResponse = resolvedDedupeStrategy === \"defer\" || options.cloneResponse;\n\n\t\t\t\tif (!response.ok) {\n\t\t\t\t\tconst errorData = await resolveResponseData(\n\t\t\t\t\t\tshouldCloneResponse ? response.clone() : response,\n\t\t\t\t\t\toptions.responseType,\n\t\t\t\t\t\toptions.responseParser\n\t\t\t\t\t);\n\n\t\t\t\t\tconst validErrorData = await handleSchemaValidation(resolvedSchema, \"errorData\", {\n\t\t\t\t\t\tinputValue: errorData,\n\t\t\t\t\t\tresponse,\n\t\t\t\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t\t\t\t});\n\n\t\t\t\t\t// == Push all error handling responsibilities to the catch block if not retrying\n\t\t\t\t\tthrow new HTTPError(\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tdefaultHTTPErrorMessage: options.defaultHTTPErrorMessage,\n\t\t\t\t\t\t\terrorData: validErrorData,\n\t\t\t\t\t\t\tresponse,\n\t\t\t\t\t\t},\n\t\t\t\t\t\t{ cause: validErrorData }\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\tconst successData = await resolveResponseData(\n\t\t\t\t\tshouldCloneResponse ? response.clone() : response,\n\t\t\t\t\toptions.responseType,\n\t\t\t\t\toptions.responseParser\n\t\t\t\t);\n\n\t\t\t\tconst validSuccessData = await handleSchemaValidation(resolvedSchema, \"data\", {\n\t\t\t\t\tinputValue: successData,\n\t\t\t\t\tresponse,\n\t\t\t\t\tschemaConfig: resolvedSchemaConfig,\n\t\t\t\t});\n\n\t\t\t\tconst successContext = {\n\t\t\t\t\tbaseConfig,\n\t\t\t\t\tconfig,\n\t\t\t\t\tdata: validSuccessData,\n\t\t\t\t\toptions,\n\t\t\t\t\trequest,\n\t\t\t\t\tresponse,\n\t\t\t\t} satisfies SuccessContext;\n\n\t\t\t\tawait executeHooks(\n\t\t\t\t\toptions.onSuccess?.(successContext),\n\n\t\t\t\t\toptions.onResponse?.({ ...successContext, error: null })\n\t\t\t\t);\n\n\t\t\t\tconst successResult = resolveSuccessResult(successContext.data, {\n\t\t\t\t\tresponse: successContext.response,\n\t\t\t\t\tresultMode: options.resultMode,\n\t\t\t\t});\n\n\t\t\t\treturn successResult as never;\n\n\t\t\t\t// == Exhaustive Error handling\n\t\t\t} catch (error) {\n\t\t\t\tconst errorInfo = {\n\t\t\t\t\tcloneResponse: options.cloneResponse,\n\t\t\t\t\tresultMode: options.resultMode,\n\t\t\t\t} satisfies ErrorInfo;\n\n\t\t\t\tconst { errorDetails, errorResult } = resolveErrorResult(error, errorInfo);\n\n\t\t\t\tconst errorContext = {\n\t\t\t\t\tbaseConfig,\n\t\t\t\t\tconfig,\n\t\t\t\t\terror: errorDetails.error as never,\n\t\t\t\t\toptions,\n\t\t\t\t\trequest,\n\t\t\t\t\tresponse: errorDetails.response as never,\n\t\t\t\t} satisfies ErrorContext<{ ErrorData: unknown }>;\n\n\t\t\t\tconst shouldThrowOnError = Boolean(\n\t\t\t\t\tisFunction(options.throwOnError) ? options.throwOnError(errorContext) : options.throwOnError\n\t\t\t\t);\n\n\t\t\t\tconst hookInfo = {\n\t\t\t\t\terrorInfo,\n\t\t\t\t\tshouldThrowOnError,\n\t\t\t\t} satisfies ExecuteHookInfo;\n\n\t\t\t\tconst { handleRetry, shouldAttemptRetry } = createRetryManager(errorContext);\n\n\t\t\t\tconst handleRetryOrGetErrorResult = async () => {\n\t\t\t\t\tconst shouldRetry = await shouldAttemptRetry();\n\n\t\t\t\t\tif (shouldRetry) {\n\t\t\t\t\t\treturn handleRetry({\n\t\t\t\t\t\t\tcallApi,\n\t\t\t\t\t\t\tcallApiArgs: { config, initURL },\n\t\t\t\t\t\t\terrorContext,\n\t\t\t\t\t\t\thookInfo,\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\n\t\t\t\t\tif (shouldThrowOnError) {\n\t\t\t\t\t\tthrow error;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn errorResult;\n\t\t\t\t};\n\n\t\t\t\tif (isValidationErrorInstance(error)) {\n\t\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t\t[options.onValidationError?.(errorContext), options.onError?.(errorContext)],\n\t\t\t\t\t\thookInfo\n\t\t\t\t\t);\n\n\t\t\t\t\treturn (hookError ?? (await handleRetryOrGetErrorResult())) as never;\n\t\t\t\t}\n\n\t\t\t\tif (isHTTPErrorInstance<TErrorData>(error)) {\n\t\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t\t[\n\t\t\t\t\t\t\toptions.onResponseError?.(errorContext),\n\t\t\t\t\t\t\toptions.onError?.(errorContext),\n\t\t\t\t\t\t\toptions.onResponse?.({ ...errorContext, data: null }),\n\t\t\t\t\t\t],\n\t\t\t\t\t\thookInfo\n\t\t\t\t\t);\n\n\t\t\t\t\treturn (hookError ?? (await handleRetryOrGetErrorResult())) as never;\n\t\t\t\t}\n\n\t\t\t\tlet message = (error as Error | undefined)?.message;\n\n\t\t\t\tif (error instanceof DOMException && error.name === \"AbortError\") {\n\t\t\t\t\tmessage = getAbortErrorMessage();\n\n\t\t\t\t\t!shouldThrowOnError && console.error(`${error.name}:`, message);\n\t\t\t\t}\n\n\t\t\t\tif (error instanceof DOMException && error.name === \"TimeoutError\") {\n\t\t\t\t\tmessage = `Request timed out after ${options.timeout}ms`;\n\n\t\t\t\t\t!shouldThrowOnError && console.error(`${error.name}:`, message);\n\t\t\t\t}\n\n\t\t\t\tconst hookError = await executeHooksInCatchBlock(\n\t\t\t\t\t[options.onRequestError?.(errorContext), options.onError?.(errorContext)],\n\t\t\t\t\thookInfo\n\t\t\t\t);\n\n\t\t\t\treturn (hookError\n\t\t\t\t\t?? getCustomizedErrorResult(await handleRetryOrGetErrorResult(), { message })) as never;\n\n\t\t\t\t// == Removing the now unneeded AbortController from store\n\t\t\t} finally {\n\t\t\t\tremoveDedupeKeyFromCache();\n\t\t\t}\n\t\t};\n\n\t\treturn callApi;\n\t};\n\n\treturn createFetchClient;\n};\n\nexport const createFetchClient = createFetchClientWithContext();\n\nexport const callApi = createFetchClient();\n"],"mappings":"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"}
@@ -1,3 +1,3 @@
1
1
  import "../../validation-Dq--Q5zC.js";
2
- import { A as definePlugin, C as isHTTPError, D as isValidationErrorInstance, E as isValidationError, F as toQueryString, M as defineSchemaConfig, N as defineSchemaRoutes, O as defineBaseConfig, P as toFormData, T as isJavascriptError, at as HTTPError, j as defineSchema, k as defineMainSchema, ot as ValidationError, w as isHTTPErrorInstance } from "../../index-HHFgKKcn.js";
2
+ import { A as definePlugin, C as isHTTPError, D as isValidationErrorInstance, E as isValidationError, F as toQueryString, M as defineSchemaConfig, N as defineSchemaRoutes, O as defineBaseConfig, P as toFormData, T as isJavascriptError, at as HTTPError, j as defineSchema, k as defineMainSchema, ot as ValidationError, w as isHTTPErrorInstance } from "../../index-j55-O6zR.js";
3
3
  export { HTTPError, ValidationError, defineBaseConfig, defineMainSchema, definePlugin, defineSchema, defineSchemaConfig, defineSchemaRoutes, isHTTPError, isHTTPErrorInstance, isJavascriptError, isValidationError, isValidationErrorInstance, toFormData, toQueryString };
@@ -1,5 +1,5 @@
1
1
  import "../../defaults-C6WKIXsf.js";
2
- import { a as isValidationErrorInstance, c as toFormData, i as isValidationError, l as toQueryString, n as isHTTPErrorInstance, o as HTTPError, r as isJavascriptError, s as ValidationError, t as isHTTPError } from "../../guards-5Ij_loC1.js";
2
+ import { a as isValidationErrorInstance, c as toFormData, i as isValidationError, l as toQueryString, n as isHTTPErrorInstance, o as HTTPError, r as isJavascriptError, s as ValidationError, t as isHTTPError } from "../../guards-DvOwVtmd.js";
3
3
 
4
4
  //#region src/utils/external/define.ts
5
5
  const defineSchema = (routes, config) => {
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@zayne-labs/callapi",
3
3
  "type": "module",
4
- "version": "1.11.27",
4
+ "version": "1.11.28",
5
5
  "description": "A lightweight wrapper over fetch with quality of life improvements like built-in request cancellation, retries, interceptors and more",
6
6
  "author": "Ryan Zayne",
7
7
  "license": "MIT",
@@ -1 +0,0 @@
1
- {"version":3,"file":"guards-5Ij_loC1.js","names":["toQueryString: ToQueryStringFn","toFormData: ToFormDataFn"],"sources":["../src/utils/guards.ts","../src/utils/external/body.ts","../src/utils/external/error.ts","../src/utils/external/guards.ts"],"sourcesContent":["import type { AnyFunction } from \"../types/type-helpers\";\n\nexport const isArray = <TArrayItem>(value: unknown): value is TArrayItem[] => Array.isArray(value);\n\nexport const isBoolean = (value: unknown): value is boolean => typeof value === \"boolean\";\n\nexport const isBlob = (value: unknown): value is Blob => value instanceof Blob;\n\nexport const isObject = <TObject extends object>(value: unknown): value is TObject => {\n\treturn typeof value === \"object\" && value !== null;\n};\n\nconst hasObjectPrototype = (value: unknown) => {\n\treturn Object.prototype.toString.call(value) === \"[object Object]\";\n};\n\n/**\n * @description Copied from TanStack Query's isPlainObject\n * @see https://github.com/TanStack/query/blob/main/packages/query-core/src/utils.ts#L321\n */\nexport const isPlainObject = <TPlainObject extends Record<string, unknown>>(\n\tvalue: unknown\n): value is TPlainObject => {\n\tif (!hasObjectPrototype(value)) {\n\t\treturn false;\n\t}\n\n\t// If has no constructor\n\tconst constructor = (value as object | undefined)?.constructor;\n\tif (constructor === undefined) {\n\t\treturn true;\n\t}\n\n\t// If has modified prototype\n\tconst prototype = constructor.prototype as object;\n\tif (!hasObjectPrototype(prototype)) {\n\t\treturn false;\n\t}\n\n\t// If constructor does not have an Object-specific method\n\tif (!Object.hasOwn(prototype, \"isPrototypeOf\")) {\n\t\treturn false;\n\t}\n\n\t// Handles Objects created by Object.create(<arbitrary prototype>)\n\tif (Object.getPrototypeOf(value) !== Object.prototype) {\n\t\treturn false;\n\t}\n\n\t// It's probably a plain object at this point\n\treturn true;\n};\n\nexport const isValidJsonString = (value: unknown): value is string => {\n\tif (!isString(value)) {\n\t\treturn false;\n\t}\n\n\ttry {\n\t\tJSON.parse(value);\n\t\treturn true;\n\t} catch {\n\t\treturn false;\n\t}\n};\n\nexport const isSerializable = (value: unknown) => {\n\treturn (\n\t\tisPlainObject(value)\n\t\t|| isArray(value)\n\t\t|| typeof (value as { toJSON: unknown } | undefined)?.toJSON === \"function\"\n\t);\n};\n\nexport const isFunction = <TFunction extends AnyFunction>(value: unknown): value is TFunction =>\n\ttypeof value === \"function\";\n\nexport const isQueryString = (value: unknown): value is string => isString(value) && value.includes(\"=\");\n\nexport const isString = (value: unknown) => typeof value === \"string\";\n\nexport const isPromise = (value: unknown) => value instanceof Promise;\n\nexport const isReadableStream = (value: unknown): value is ReadableStream<unknown> => {\n\treturn value instanceof ReadableStream;\n};\n\n// https://github.com/unjs/ofetch/blob/main/src/utils.ts\nexport const isJSONSerializable = (value: unknown) => {\n\tif (value === undefined) {\n\t\treturn false;\n\t}\n\tconst t = typeof value;\n\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- No time to make this more type-safe\n\tif (t === \"string\" || t === \"number\" || t === \"boolean\" || t === null) {\n\t\treturn true;\n\t}\n\tif (t !== \"object\") {\n\t\treturn false;\n\t}\n\tif (isArray(value)) {\n\t\treturn true;\n\t}\n\tif ((value as Buffer | null)?.buffer) {\n\t\treturn false;\n\t}\n\n\treturn (\n\t\tvalue?.constructor.name === \"Object\"\n\t\t|| typeof (value as { toJSON: () => unknown } | null)?.toJSON === \"function\"\n\t);\n};\n","import type { CallApiExtraOptions } from \"../../types/common\";\nimport { isArray, isBlob, isObject } from \"../guards\";\n\ntype ToQueryStringFn = {\n\t(query: CallApiExtraOptions[\"query\"]): string | null;\n\t(query: Required<CallApiExtraOptions>[\"query\"]): string;\n};\n\nexport const toQueryString: ToQueryStringFn = (query) => {\n\tif (!query) {\n\t\tconsole.error(\"toQueryString:\", \"No query params provided!\");\n\n\t\treturn null as never;\n\t}\n\n\treturn new URLSearchParams(query as Record<string, string>).toString();\n};\n\ntype AllowedPrimitives = boolean | number | string | Blob | null | undefined;\n\ntype AllowedValues = AllowedPrimitives | AllowedPrimitives[] | Record<string, AllowedPrimitives>;\n\nconst toBlobOrString = (value: AllowedPrimitives): string | Blob => {\n\treturn isBlob(value) ? value : String(value);\n};\n\ntype ToFormDataFn = {\n\t(data: Record<string, AllowedValues>): FormData;\n\n\t<TData extends Record<string, AllowedValues>>(data: TData, options: { returnType: \"inputType\" }): TData;\n};\n\n/**\n * @description Converts a plain object to FormData.\n *\n * Handles various data types:\n * - **Primitives** (string, number, boolean): Converted to strings\n * - **Blobs/Files**: Added directly to FormData\n * - **Arrays**: Each item is appended (allows multiple values for same key)\n * - **Objects**: JSON stringified before adding to FormData\n *\n * @example\n * ```ts\n * // Basic usage\n * const formData = toFormData({\n * name: \"John\",\n * age: 30,\n * active: true\n * });\n *\n * // With arrays\n * const formData = toFormData({\n * tags: [\"javascript\", \"typescript\"],\n * name: \"John\"\n * });\n *\n * // With files\n * const formData = toFormData({\n * avatar: fileBlob,\n * name: \"John\"\n * });\n *\n * // With nested objects (one level only)\n * const formData = toFormData({\n * user: { name: \"John\", age: 30 },\n * settings: { theme: \"dark\" }\n * });\n *\n * // Type-preserving usage with Zod\n * const schema = z.object({ name: z.string(), file: z.instanceof(Blob) });\n * const data = schema.parse({ name: \"John\", file: blob });\n * const typedFormData = toFormData(data, { returnType: \"inputType\" });\n * // Type is { name: string; file: Blob }, runtime is FormData\n * ```\n */\nexport const toFormData: ToFormDataFn = (data: Record<string, AllowedValues>) => {\n\tconst formData = new FormData();\n\n\tfor (const [key, value] of Object.entries(data)) {\n\t\tif (isArray(value)) {\n\t\t\tvalue.forEach((innerValue) => formData.append(key, toBlobOrString(innerValue)));\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (isObject(value) && !isBlob(value)) {\n\t\t\tformData.set(key, JSON.stringify(value));\n\t\t\tcontinue;\n\t\t}\n\n\t\tformData.set(key, toBlobOrString(value));\n\t}\n\n\treturn formData;\n};\n","import { extraOptionDefaults } from \"../../constants/defaults\";\nimport type { CallApiExtraOptions } from \"../../types/common\";\nimport type { StandardSchemaV1 } from \"../../types/standard-schema\";\nimport type { CallApiSchema, CallApiSchemaConfig } from \"../../validation\";\nimport { isObject, isString } from \"../guards\";\n\ntype HTTPErrorDetails<TErrorData> = Pick<CallApiExtraOptions, \"defaultHTTPErrorMessage\"> & {\n\terrorData: TErrorData;\n\tresponse: Response;\n};\n\nconst httpErrorSymbol = Symbol(\"HTTPError\");\n\nexport class HTTPError<TErrorData = Record<string, unknown>> extends Error {\n\terrorData: HTTPErrorDetails<TErrorData>[\"errorData\"];\n\n\treadonly httpErrorSymbol = httpErrorSymbol;\n\n\toverride name = \"HTTPError\" as const;\n\n\tresponse: HTTPErrorDetails<TErrorData>[\"response\"];\n\n\tconstructor(errorDetails: HTTPErrorDetails<TErrorData>, errorOptions?: ErrorOptions) {\n\t\tconst { defaultHTTPErrorMessage, errorData, response } = errorDetails;\n\n\t\tconst resolvedDefaultHTTPErrorMessage =\n\t\t\tisString(defaultHTTPErrorMessage) ? defaultHTTPErrorMessage : (\n\t\t\t\tdefaultHTTPErrorMessage?.({ errorData, response })\n\t\t\t);\n\n\t\tconst selectedDefaultErrorMessage =\n\t\t\tresolvedDefaultHTTPErrorMessage\n\t\t\t?? (response.statusText || extraOptionDefaults.defaultHTTPErrorMessage);\n\n\t\tconst message =\n\t\t\t(errorData as { message?: string } | undefined)?.message ?? selectedDefaultErrorMessage;\n\n\t\tsuper(message, errorOptions);\n\n\t\tthis.errorData = errorData;\n\t\tthis.response = response;\n\t}\n\n\t/**\n\t * @description Checks if the given error is an instance of HTTPError\n\t * @param error - The error to check\n\t * @returns true if the error is an instance of HTTPError, false otherwise\n\t */\n\tstatic override isError<TErrorData>(error: unknown): error is HTTPError<TErrorData> {\n\t\tif (!isObject<HTTPError>(error)) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (error instanceof HTTPError) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst actualError = error as HTTPError;\n\n\t\treturn (\n\t\t\tactualError.httpErrorSymbol === httpErrorSymbol\n\t\t\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- Allow\n\t\t\t&& actualError.name === \"HTTPError\"\n\t\t);\n\t}\n}\n\nconst prettifyPath = (path: ValidationError[\"errorData\"][number][\"path\"]) => {\n\tif (!path || path.length === 0) {\n\t\treturn \"\";\n\t}\n\n\tconst pathString = path.map((segment) => (isObject(segment) ? segment.key : segment)).join(\".\");\n\n\treturn ` → at ${pathString}`;\n};\n\nconst prettifyValidationIssues = (issues: ValidationError[\"errorData\"]) => {\n\tconst issuesString = issues.map((issue) => `✖ ${issue.message}${prettifyPath(issue.path)}`).join(\" | \");\n\n\treturn issuesString;\n};\n\ntype SafeExtract<TUnion, TKey extends TUnion> = Extract<TUnion, TKey>;\n\ntype ValidationErrorDetails = {\n\t/**\n\t * The cause of the validation error.\n\t *\n\t * It's either the name the schema for which validation failed, or the name of the schema config option that led to the validation error.\n\t */\n\tissueCause:\n\t\t| \"unknown\"\n\t\t| `schemaConfig-(${SafeExtract<keyof CallApiSchemaConfig, \"strict\">})`\n\t\t| keyof CallApiSchema;\n\n\t/**\n\t * The issues that caused the validation error.\n\t */\n\tissues: readonly StandardSchemaV1.Issue[];\n\n\t/**\n\t * The response from server, if any.\n\t */\n\tresponse: Response | null;\n};\n\nconst validationErrorSymbol = Symbol(\"ValidationErrorSymbol\");\n\nexport class ValidationError extends Error {\n\terrorData: ValidationErrorDetails[\"issues\"];\n\n\tissueCause: ValidationErrorDetails[\"issueCause\"];\n\n\toverride name = \"ValidationError\" as const;\n\n\tresponse: ValidationErrorDetails[\"response\"];\n\n\treadonly validationErrorSymbol = validationErrorSymbol;\n\n\tconstructor(details: ValidationErrorDetails, errorOptions?: ErrorOptions) {\n\t\tconst { issueCause, issues, response } = details;\n\n\t\tconst prettyMessage = prettifyValidationIssues(issues);\n\n\t\tconst message = `(${issueCause.toUpperCase()}) - ${prettyMessage}`;\n\n\t\tsuper(message, errorOptions);\n\n\t\tthis.errorData = issues;\n\t\tthis.response = response;\n\t\tthis.issueCause = issueCause;\n\t}\n\n\t/**\n\t * @description Checks if the given error is an instance of ValidationError\n\t * @param error - The error to check\n\t * @returns true if the error is an instance of ValidationError, false otherwise\n\t */\n\tstatic override isError(error: unknown): error is ValidationError {\n\t\tif (!isObject<ValidationError>(error)) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (error instanceof ValidationError) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst actualError = error as ValidationError;\n\n\t\treturn (\n\t\t\tactualError.validationErrorSymbol === validationErrorSymbol\n\t\t\t// eslint-disable-next-line ts-eslint/no-unnecessary-condition -- Allow\n\t\t\t&& actualError.name === \"ValidationError\"\n\t\t);\n\t}\n}\n","import type {\n\tCallApiResultErrorVariant,\n\tPossibleHTTPError,\n\tPossibleJavaScriptError,\n\tPossibleValidationError,\n} from \"../../result\";\nimport { isObject } from \"../guards\";\nimport { HTTPError, ValidationError } from \"./error\";\n\nexport const isHTTPError = <TErrorData>(\n\terror: CallApiResultErrorVariant<TErrorData>[\"error\"] | null\n): error is PossibleHTTPError<TErrorData> => {\n\treturn isObject(error) && error.name === \"HTTPError\";\n};\n\nexport const isHTTPErrorInstance = <TErrorData>(error: unknown) => {\n\treturn HTTPError.isError<TErrorData>(error);\n};\n\nexport const isValidationError = (\n\terror: CallApiResultErrorVariant<unknown>[\"error\"] | null\n): error is PossibleValidationError => {\n\treturn isObject(error) && error.name === \"ValidationError\";\n};\n\nexport const isValidationErrorInstance = (error: unknown): error is ValidationError => {\n\treturn ValidationError.isError(error);\n};\n\nexport const isJavascriptError = (\n\terror: CallApiResultErrorVariant<unknown>[\"error\"] | null\n): error is PossibleJavaScriptError => {\n\treturn isObject(error) && !isHTTPError(error) && !isValidationError(error);\n};\n"],"mappings":"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"}