@ledgerhq/device-management-kit 1.6.0 → 1.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/lib/cjs/package.json +1 -2
  2. package/lib/cjs/src/api/logger-subscriber/service/WebLogsExporterLogger.js.map +2 -2
  3. package/lib/cjs/src/api/network/DmkNetworkClient.js +1 -1
  4. package/lib/cjs/src/api/network/DmkNetworkClient.js.map +3 -3
  5. package/lib/cjs/src/api/network/DmkNetworkClient.test.js +1 -1
  6. package/lib/cjs/src/api/network/DmkNetworkClient.test.js.map +3 -3
  7. package/lib/cjs/src/api/network/DmkNetworkClientError.js +1 -1
  8. package/lib/cjs/src/api/network/DmkNetworkClientError.js.map +2 -2
  9. package/lib/cjs/src/api/network/DmkNetworkClientHelpers.js +1 -1
  10. package/lib/cjs/src/api/network/DmkNetworkClientHelpers.js.map +3 -3
  11. package/lib/cjs/src/api/network/DmkNetworkClientHelpers.test.js +1 -1
  12. package/lib/cjs/src/api/network/DmkNetworkClientHelpers.test.js.map +3 -3
  13. package/lib/cjs/src/api/utils/Base64String.js +1 -1
  14. package/lib/cjs/src/api/utils/Base64String.js.map +3 -3
  15. package/lib/cjs/src/api/utils/Base64String.test.js +3 -1
  16. package/lib/cjs/src/api/utils/Base64String.test.js.map +3 -3
  17. package/lib/cjs/src/internal/secure-channel/data/DefaultSecureChannelDataSource.js +1 -1
  18. package/lib/cjs/src/internal/secure-channel/data/DefaultSecureChannelDataSource.js.map +3 -3
  19. package/lib/cjs/src/internal/secure-channel/data/DefaultSecureChannelDataSource.test.js +1 -1
  20. package/lib/cjs/src/internal/secure-channel/data/DefaultSecureChannelDataSource.test.js.map +2 -2
  21. package/lib/esm/package.json +1 -2
  22. package/lib/esm/src/api/logger-subscriber/service/WebLogsExporterLogger.js.map +2 -2
  23. package/lib/esm/src/api/network/DmkNetworkClient.js +1 -1
  24. package/lib/esm/src/api/network/DmkNetworkClient.js.map +3 -3
  25. package/lib/esm/src/api/network/DmkNetworkClient.test.js +1 -1
  26. package/lib/esm/src/api/network/DmkNetworkClient.test.js.map +3 -3
  27. package/lib/esm/src/api/network/DmkNetworkClientError.js +1 -1
  28. package/lib/esm/src/api/network/DmkNetworkClientError.js.map +2 -2
  29. package/lib/esm/src/api/network/DmkNetworkClientHelpers.js +1 -1
  30. package/lib/esm/src/api/network/DmkNetworkClientHelpers.js.map +3 -3
  31. package/lib/esm/src/api/network/DmkNetworkClientHelpers.test.js +1 -1
  32. package/lib/esm/src/api/network/DmkNetworkClientHelpers.test.js.map +3 -3
  33. package/lib/esm/src/api/utils/Base64String.js +1 -1
  34. package/lib/esm/src/api/utils/Base64String.js.map +3 -3
  35. package/lib/esm/src/api/utils/Base64String.test.js +3 -1
  36. package/lib/esm/src/api/utils/Base64String.test.js.map +3 -3
  37. package/lib/esm/src/internal/secure-channel/data/DefaultSecureChannelDataSource.js +1 -1
  38. package/lib/esm/src/internal/secure-channel/data/DefaultSecureChannelDataSource.js.map +3 -3
  39. package/lib/esm/src/internal/secure-channel/data/DefaultSecureChannelDataSource.test.js +1 -1
  40. package/lib/esm/src/internal/secure-channel/data/DefaultSecureChannelDataSource.test.js.map +2 -2
  41. package/lib/types/src/api/logger-subscriber/service/WebLogsExporterLogger.d.ts.map +1 -1
  42. package/lib/types/src/api/network/DmkNetworkClient.d.ts +1 -6
  43. package/lib/types/src/api/network/DmkNetworkClient.d.ts.map +1 -1
  44. package/lib/types/src/api/network/DmkNetworkClientError.d.ts +2 -6
  45. package/lib/types/src/api/network/DmkNetworkClientError.d.ts.map +1 -1
  46. package/lib/types/src/api/network/DmkNetworkClientHelpers.d.ts +10 -7
  47. package/lib/types/src/api/network/DmkNetworkClientHelpers.d.ts.map +1 -1
  48. package/lib/types/src/api/utils/Base64String.d.ts.map +1 -1
  49. package/lib/types/src/internal/secure-channel/data/DefaultSecureChannelDataSource.d.ts.map +1 -1
  50. package/lib/types/tsconfig.prod.tsbuildinfo +1 -1
  51. package/package.json +4 -5
@@ -7,7 +7,6 @@
7
7
  "purify-ts": "catalog:",
8
8
  "reflect-metadata": "catalog:",
9
9
  "semver": "catalog:",
10
- "url": "catalog:",
11
10
  "uuid": "catalog:",
12
11
  "ws": "catalog:",
13
12
  "xstate": "catalog:"
@@ -63,5 +62,5 @@
63
62
  "watch:builds": "pnpm ldmk-tool watch --entryPoints index.ts,src/**/*.ts --tsconfig tsconfig.prod.json",
64
63
  "watch:types": "concurrently \"tsc --watch -p tsconfig.prod.json\" \"tsc-alias --watch -p tsconfig.prod.json\""
65
64
  },
66
- "version": "1.6.0"
65
+ "version": "1.7.0"
67
66
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/api/logger-subscriber/service/WebLogsExporterLogger.ts"],
4
- "sourcesContent": ["import { LogLevel } from \"@api/logger-subscriber/model/LogLevel\";\nimport { type LogSubscriberOptions } from \"@api/types\";\nimport { DeviceSession } from \"@internal/device-session/model/DeviceSession\";\n\nimport { type LoggerSubscriberService } from \"./LoggerSubscriberService\";\n\n/**\n * This function is used to format the logs to JSON format,\n * remove circular dependencies and do some extra formatting.\n * */\nexport function getJSONStringifyReplacer(): (\n key: string,\n value: unknown,\n) => unknown {\n const ancestors: unknown[] = [];\n return function (_: string, value: unknown): unknown {\n // format Uint8Array values to more readable format\n if (value instanceof Uint8Array) {\n const bytesHex = Array.from(value).map((x) =>\n x.toString(16).padStart(2, \"0\"),\n );\n return {\n hex: \"0x\" + bytesHex.join(\"\"),\n readableHex: bytesHex.join(\" \"),\n value: value.toString(),\n };\n }\n\n // format DeviceSession values to avoid huge object in logs\n if (value instanceof DeviceSession) {\n const {\n connectedDevice: { deviceModel, type, id },\n } = value;\n return {\n id: value.id,\n connectedDevice: {\n deviceModel,\n type,\n id,\n },\n };\n }\n\n // format circular references to \"[Circular]\"\n // Taken from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Errors/Cyclic_object_value#circular_references\n if (typeof value !== \"object\" || value === null) {\n return value;\n }\n // `this` is the object that value is contained in,\n // i.e., its direct parent.\n // @ts-expect-error cf. comment above\n while (ancestors.length > 0 && ancestors.at(-1) !== (this as unknown)) {\n ancestors.pop();\n }\n if (ancestors.includes(value)) {\n return \"[Circular]\";\n }\n ancestors.push(value);\n return value;\n };\n}\n\nexport class WebLogsExporterLogger implements LoggerSubscriberService {\n private logs: Array<\n [level: LogLevel, message: string, options: LogSubscriberOptions]\n > = [];\n\n log(level: LogLevel, message: string, options: LogSubscriberOptions): void {\n this.logs.push([level, message, options]);\n }\n\n private formatLogsToJSON(): string {\n const remappedLogs = this.logs.map(([level, message, options]) => {\n const { timestamp, ...restOptions } = options;\n return {\n level: LogLevel[level],\n message,\n options: {\n ...restOptions,\n date: new Date(options.timestamp),\n },\n };\n });\n\n return JSON.stringify(remappedLogs, getJSONStringifyReplacer(), 2);\n }\n\n /**\n * Export logs to JSON file.\n */\n public exportLogsToJSON(): void {\n const logs = this.formatLogsToJSON();\n const blob = new Blob([logs], { type: \"application/json\" });\n const url = URL.createObjectURL(blob);\n const a = document.createElement(\"a\");\n a.href = url;\n a.download = `ledger-device-management-kit-logs-${new Date().toISOString()}.json`;\n a.click();\n }\n}\n"],
5
- "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,2BAAAE,EAAA,6BAAAC,IAAA,eAAAC,EAAAJ,GAAA,IAAAK,EAAyB,iDAEzBC,EAA8B,wDAQvB,SAASH,GAGH,CACX,MAAMI,EAAuB,CAAC,EAC9B,OAAO,SAAUC,EAAWC,EAAyB,CAEnD,GAAIA,aAAiB,WAAY,CAC/B,MAAMC,EAAW,MAAM,KAAKD,CAAK,EAAE,IAAKE,GACtCA,EAAE,SAAS,EAAE,EAAE,SAAS,EAAG,GAAG,CAChC,EACA,MAAO,CACL,IAAK,KAAOD,EAAS,KAAK,EAAE,EAC5B,YAAaA,EAAS,KAAK,GAAG,EAC9B,MAAOD,EAAM,SAAS,CACxB,CACF,CAGA,GAAIA,aAAiB,gBAAe,CAClC,KAAM,CACJ,gBAAiB,CAAE,YAAAG,EAAa,KAAAC,EAAM,GAAAC,CAAG,CAC3C,EAAIL,EACJ,MAAO,CACL,GAAIA,EAAM,GACV,gBAAiB,CACf,YAAAG,EACA,KAAAC,EACA,GAAAC,CACF,CACF,CACF,CAIA,GAAI,OAAOL,GAAU,UAAYA,IAAU,KACzC,OAAOA,EAKT,KAAOF,EAAU,OAAS,GAAKA,EAAU,GAAG,EAAE,IAAO,MACnDA,EAAU,IAAI,EAEhB,OAAIA,EAAU,SAASE,CAAK,EACnB,cAETF,EAAU,KAAKE,CAAK,EACbA,EACT,CACF,CAEO,MAAMP,CAAyD,CAC5D,KAEJ,CAAC,EAEL,IAAIa,EAAiBC,EAAiBC,EAAqC,CACzE,KAAK,KAAK,KAAK,CAACF,EAAOC,EAASC,CAAO,CAAC,CAC1C,CAEQ,kBAA2B,CACjC,MAAMC,EAAe,KAAK,KAAK,IAAI,CAAC,CAACH,EAAOC,EAASC,CAAO,IAAM,CAChE,KAAM,CAAE,UAAAE,EAAW,GAAGC,CAAY,EAAIH,EACtC,MAAO,CACL,MAAO,WAASF,CAAK,EACrB,QAAAC,EACA,QAAS,CACP,GAAGI,EACH,KAAM,IAAI,KAAKH,EAAQ,SAAS,CAClC,CACF,CACF,CAAC,EAED,OAAO,KAAK,UAAUC,EAAcf,EAAyB,EAAG,CAAC,CACnE,CAKO,kBAAyB,CAC9B,MAAMkB,EAAO,KAAK,iBAAiB,EAC7BC,EAAO,IAAI,KAAK,CAACD,CAAI,EAAG,CAAE,KAAM,kBAAmB,CAAC,EACpDE,EAAM,IAAI,gBAAgBD,CAAI,EAC9BE,EAAI,SAAS,cAAc,GAAG,EACpCA,EAAE,KAAOD,EACTC,EAAE,SAAW,qCAAqC,IAAI,KAAK,EAAE,YAAY,CAAC,QAC1EA,EAAE,MAAM,CACV,CACF",
4
+ "sourcesContent": ["import { LogLevel } from \"@api/logger-subscriber/model/LogLevel\";\nimport { type LogSubscriberOptions } from \"@api/types\";\nimport { DeviceSession } from \"@internal/device-session/model/DeviceSession\";\n\nimport { type LoggerSubscriberService } from \"./LoggerSubscriberService\";\n\n/**\n * This function is used to format the logs to JSON format,\n * remove circular dependencies and do some extra formatting.\n * */\nexport function getJSONStringifyReplacer(): (\n key: string,\n value: unknown,\n) => unknown {\n const ancestors: unknown[] = [];\n return function (_: string, value: unknown): unknown {\n // format Uint8Array values to more readable format\n if (value instanceof Uint8Array) {\n const bytesHex = Array.from(value).map((x) =>\n x.toString(16).padStart(2, \"0\"),\n );\n return {\n hex: \"0x\" + bytesHex.join(\"\"),\n readableHex: bytesHex.join(\" \"),\n value: value.toString(),\n };\n }\n\n // format DeviceSession values to avoid huge object in logs\n if (value instanceof DeviceSession) {\n const {\n connectedDevice: { deviceModel, type, id },\n } = value;\n return {\n id: value.id,\n connectedDevice: {\n deviceModel,\n type,\n id,\n },\n };\n }\n\n // format circular references to \"[Circular]\"\n // Taken from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Errors/Cyclic_object_value#circular_references\n if (typeof value !== \"object\" || value === null) {\n return value;\n }\n // `this` is the object that value is contained in,\n // i.e., its direct parent.\n // @ts-expect-error cf. comment above\n while (ancestors.length > 0 && ancestors.at(-1) !== (this as unknown)) {\n ancestors.pop();\n }\n if (ancestors.includes(value)) {\n return \"[Circular]\";\n }\n ancestors.push(value);\n return value;\n };\n}\n\nexport class WebLogsExporterLogger implements LoggerSubscriberService {\n private logs: Array<\n [level: LogLevel, message: string, options: LogSubscriberOptions]\n > = [];\n\n log(level: LogLevel, message: string, options: LogSubscriberOptions): void {\n this.logs.push([level, message, options]);\n }\n\n private formatLogsToJSON(): string {\n const remappedLogs = this.logs.map(([level, message, options]) => {\n const { timestamp, ...restOptions } = options;\n return {\n level: LogLevel[level],\n message,\n options: {\n ...restOptions,\n date: new Date(options.timestamp),\n },\n };\n });\n\n return JSON.stringify(remappedLogs, getJSONStringifyReplacer(), 2);\n }\n\n /**\n * Export logs to JSON file.\n */\n public exportLogsToJSON(): void {\n const logs = this.formatLogsToJSON();\n /* eslint-disable no-restricted-globals -- Browser-only log export path uses Blob URLs to trigger a JSON file download. */\n const blob = new Blob([logs], { type: \"application/json\" });\n const url = URL.createObjectURL(blob);\n /* eslint-enable no-restricted-globals */\n const a = document.createElement(\"a\");\n a.href = url;\n a.download = `ledger-device-management-kit-logs-${new Date().toISOString()}.json`;\n a.click();\n }\n}\n"],
5
+ "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,2BAAAE,EAAA,6BAAAC,IAAA,eAAAC,EAAAJ,GAAA,IAAAK,EAAyB,iDAEzBC,EAA8B,wDAQvB,SAASH,GAGH,CACX,MAAMI,EAAuB,CAAC,EAC9B,OAAO,SAAUC,EAAWC,EAAyB,CAEnD,GAAIA,aAAiB,WAAY,CAC/B,MAAMC,EAAW,MAAM,KAAKD,CAAK,EAAE,IAAKE,GACtCA,EAAE,SAAS,EAAE,EAAE,SAAS,EAAG,GAAG,CAChC,EACA,MAAO,CACL,IAAK,KAAOD,EAAS,KAAK,EAAE,EAC5B,YAAaA,EAAS,KAAK,GAAG,EAC9B,MAAOD,EAAM,SAAS,CACxB,CACF,CAGA,GAAIA,aAAiB,gBAAe,CAClC,KAAM,CACJ,gBAAiB,CAAE,YAAAG,EAAa,KAAAC,EAAM,GAAAC,CAAG,CAC3C,EAAIL,EACJ,MAAO,CACL,GAAIA,EAAM,GACV,gBAAiB,CACf,YAAAG,EACA,KAAAC,EACA,GAAAC,CACF,CACF,CACF,CAIA,GAAI,OAAOL,GAAU,UAAYA,IAAU,KACzC,OAAOA,EAKT,KAAOF,EAAU,OAAS,GAAKA,EAAU,GAAG,EAAE,IAAO,MACnDA,EAAU,IAAI,EAEhB,OAAIA,EAAU,SAASE,CAAK,EACnB,cAETF,EAAU,KAAKE,CAAK,EACbA,EACT,CACF,CAEO,MAAMP,CAAyD,CAC5D,KAEJ,CAAC,EAEL,IAAIa,EAAiBC,EAAiBC,EAAqC,CACzE,KAAK,KAAK,KAAK,CAACF,EAAOC,EAASC,CAAO,CAAC,CAC1C,CAEQ,kBAA2B,CACjC,MAAMC,EAAe,KAAK,KAAK,IAAI,CAAC,CAACH,EAAOC,EAASC,CAAO,IAAM,CAChE,KAAM,CAAE,UAAAE,EAAW,GAAGC,CAAY,EAAIH,EACtC,MAAO,CACL,MAAO,WAASF,CAAK,EACrB,QAAAC,EACA,QAAS,CACP,GAAGI,EACH,KAAM,IAAI,KAAKH,EAAQ,SAAS,CAClC,CACF,CACF,CAAC,EAED,OAAO,KAAK,UAAUC,EAAcf,EAAyB,EAAG,CAAC,CACnE,CAKO,kBAAyB,CAC9B,MAAMkB,EAAO,KAAK,iBAAiB,EAE7BC,EAAO,IAAI,KAAK,CAACD,CAAI,EAAG,CAAE,KAAM,kBAAmB,CAAC,EACpDE,EAAM,IAAI,gBAAgBD,CAAI,EAE9BE,EAAI,SAAS,cAAc,GAAG,EACpCA,EAAE,KAAOD,EACTC,EAAE,SAAW,qCAAqC,IAAI,KAAK,EAAE,YAAY,CAAC,QAC1EA,EAAE,MAAM,CACV,CACF",
6
6
  "names": ["WebLogsExporterLogger_exports", "__export", "WebLogsExporterLogger", "getJSONStringifyReplacer", "__toCommonJS", "import_LogLevel", "import_DeviceSession", "ancestors", "_", "value", "bytesHex", "x", "deviceModel", "type", "id", "level", "message", "options", "remappedLogs", "timestamp", "restOptions", "logs", "blob", "url", "a"]
7
7
  }
@@ -1,2 +1,2 @@
1
- "use strict";var i=Object.defineProperty;var l=Object.getOwnPropertyDescriptor;var k=Object.getOwnPropertyNames;var y=Object.prototype.hasOwnProperty;var g=(a,e)=>{for(var t in e)i(a,t,{get:e[t],enumerable:!0})},b=(a,e,t,s)=>{if(e&&typeof e=="object"||typeof e=="function")for(let r of k(e))!y.call(a,r)&&r!==t&&i(a,r,{get:()=>e[r],enumerable:!(s=l(e,r))||s.enumerable});return a};var w=a=>b(i({},"__esModule",{value:!0}),a);var D={};g(D,{DmkNetworkClient:()=>c});module.exports=w(D);var p=require("./DmkNetworkClientError"),o=require("./DmkNetworkClientHelpers");class c{baseUrl;defaultHeaders;fetchImpl;constructor(e={}){this.baseUrl=e.baseUrl,this.defaultHeaders=e.headers??{},this.fetchImpl=e.fetch}getFetch(){return this.fetchImpl??globalThis.fetch.bind(globalThis)}get(e,t){return this.request({...t,method:"GET",url:e}).then(s=>s.data)}post(e,t,s){return this.request({...s,method:"POST",url:e,body:t}).then(r=>r.data)}put(e,t,s){return this.request({...s,method:"PUT",url:e,body:t}).then(r=>r.data)}patch(e,t,s){return this.request({...s,method:"PATCH",url:e,body:t}).then(r=>r.data)}delete(e,t){return this.request({...t,method:"DELETE",url:e}).then(s=>s.data)}head(e,t){return this.request({...t,method:"HEAD",url:e,responseType:"void"}).then(()=>{})}async request(e){const t=(0,o.buildUrl)({url:e.url,params:e.params,baseUrl:this.baseUrl}),{body:s,headers:r}=(0,o.buildBodyAndHeaders)({method:e.method,body:e.body,defaultHeaders:this.defaultHeaders,perRequestHeaders:e.headers}),m=(0,o.buildSignal)({timeoutMs:e.timeoutMs,externalSignal:e.signal}),d=e.throwOnHttpError??!0,h=e.responseType??"json";let n;try{n=await this.getFetch()(t,{method:e.method,headers:r,body:s,signal:m})}catch(u){throw(0,o.wrapFetchError)({cause:u,externalSignal:e.signal,timeoutMs:e.timeoutMs})}if(!n.ok&&d){const u=await(0,o.safeReadText)(n);throw new p.DmkNetworkClientError({message:`HTTP error ${n.status} ${n.statusText}`.trim(),status:n.status,statusText:n.statusText,responseBody:u})}return{data:await(0,o.parseBody)(n,h),status:n.status,statusText:n.statusText,headers:n.headers,ok:n.ok}}}0&&(module.exports={DmkNetworkClient});
1
+ "use strict";var i=Object.defineProperty;var k=Object.getOwnPropertyDescriptor;var y=Object.getOwnPropertyNames;var c=Object.prototype.hasOwnProperty;var b=(a,e)=>{for(var t in e)i(a,t,{get:e[t],enumerable:!0})},w=(a,e,t,s)=>{if(e&&typeof e=="object"||typeof e=="function")for(let r of y(e))!c.call(a,r)&&r!==t&&i(a,r,{get:()=>e[r],enumerable:!(s=k(e,r))||s.enumerable});return a};var g=a=>w(i({},"__esModule",{value:!0}),a);var f={};b(f,{DmkNetworkClient:()=>D});module.exports=g(f);var p=require("./DmkNetworkClientError"),o=require("./DmkNetworkClientHelpers");class D{baseUrl;defaultHeaders;fetchImpl;constructor(e={}){this.baseUrl=e.baseUrl,this.defaultHeaders=e.headers??{},this.fetchImpl=e.fetch}getFetch(){return this.fetchImpl??globalThis.fetch.bind(globalThis)}get(e,t){return this.request({...t,method:"GET",url:e}).then(s=>s.data)}post(e,t,s){return this.request({...s,method:"POST",url:e,body:t}).then(r=>r.data)}put(e,t,s){return this.request({...s,method:"PUT",url:e,body:t}).then(r=>r.data)}patch(e,t,s){return this.request({...s,method:"PATCH",url:e,body:t}).then(r=>r.data)}delete(e,t){return this.request({...t,method:"DELETE",url:e}).then(s=>s.data)}head(e,t){return this.request({...t,method:"HEAD",url:e,responseType:"void"}).then(()=>{})}async request(e){const t=(0,o.buildUrl)({url:e.url,params:e.params,baseUrl:this.baseUrl}),{body:s,headers:r}=(0,o.buildBodyAndHeaders)({method:e.method,body:e.body,defaultHeaders:this.defaultHeaders,perRequestHeaders:e.headers}),{signal:m,cleanup:d}=(0,o.buildSignal)({timeoutMs:e.timeoutMs}),h=e.throwOnHttpError??!0,l=e.responseType??"json";try{let n;try{n=await this.getFetch()(t,{method:e.method,headers:r,body:s,signal:m})}catch(u){throw(0,o.wrapFetchError)({cause:u,timeoutMs:e.timeoutMs})}if(!n.ok&&h){const u=await(0,o.safeReadText)(n);throw new p.DmkNetworkClientError({message:`HTTP error ${n.status} ${n.statusText}`.trim(),status:n.status,statusText:n.statusText,responseBody:u})}return{data:await(0,o.parseBody)(n,l),status:n.status,statusText:n.statusText,headers:n.headers,ok:n.ok}}finally{d()}}}0&&(module.exports={DmkNetworkClient});
2
2
  //# sourceMappingURL=DmkNetworkClient.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/api/network/DmkNetworkClient.ts"],
4
- "sourcesContent": ["import { DmkNetworkClientError } from \"./DmkNetworkClientError\";\nimport {\n buildBodyAndHeaders,\n buildSignal,\n buildUrl,\n type DmkQueryParams,\n type DmkResponseType,\n parseBody,\n safeReadText,\n wrapFetchError,\n} from \"./DmkNetworkClientHelpers\";\n\nexport type { DmkQueryParamValue } from \"./DmkNetworkClientHelpers\";\nexport type { DmkQueryParams, DmkResponseType };\n\n/**\n * Per-request configuration. Everything is optional \u2014 sensible defaults are\n * applied by the client.\n */\nexport type DmkRequestConfig = {\n /** Query params merged into the URL. `null`/`undefined` entries are skipped. */\n params?: DmkQueryParams;\n /** Per-request headers merged on top of the client's default headers. */\n headers?: Record<string, string>;\n /**\n * Per-request timeout in milliseconds. When unset (or `0`), the request\n * has no timeout.\n */\n timeoutMs?: number;\n /**\n * External abort signal. Composed with the internal timeout signal, so\n * either one firing will abort the request.\n */\n signal?: AbortSignal;\n /**\n * How to parse the response body. Defaults to `\"json\"`, except for `head`\n * which always resolves to `void`.\n */\n responseType?: DmkResponseType;\n /**\n * When `true` (default), non-2xx responses throw {@link DmkNetworkClientError}.\n * Set to `false` to resolve normally and inspect {@link DmkNetworkResponse}\n * via {@link DmkNetworkClient.request}.\n */\n throwOnHttpError?: boolean;\n};\n\n/**\n * Full response envelope returned by {@link DmkNetworkClient.request}.\n * The simple method helpers (`get`, `post`, \u2026) unwrap to `data` directly.\n *\n * `data` is intentionally typed as `unknown`: network payloads are untrusted\n * input, so callers must narrow it with a type guard or runtime validator\n * (e.g. the existing DTO mappers) before use.\n */\nexport type DmkNetworkResponse = {\n data: unknown;\n status: number;\n statusText: string;\n headers: Headers;\n ok: boolean;\n};\n\nexport type DmkNetworkClientOptions = {\n /** Base URL prepended to relative request URLs. */\n baseUrl?: string;\n /** Default headers merged into every request. */\n headers?: Record<string, string>;\n /** Injection point for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n};\n\ntype InternalRequestConfig = DmkRequestConfig & {\n method: string;\n url: string;\n body?: unknown;\n};\n\n/**\n * Minimal axios-like wrapper over `fetch`. Handles:\n *\n * - URL composition (base URL + relative path + query params from an object)\n * - Automatic JSON body encoding and `Content-Type` header\n * - Default and per-request headers merging\n * - Request timeout via `AbortSignal.timeout`, composable with a caller signal\n * - Automatic `response.ok` check with a typed {@link DmkNetworkClientError}\n * - Typed JSON / text / blob / arrayBuffer response parsing\n *\n * Use the high-level helpers ({@link DmkNetworkClient.get}, {@link DmkNetworkClient.post},\n * \u2026) for 95% of calls and {@link DmkNetworkClient.request} when you need the\n * full response envelope (status, headers).\n */\nexport class DmkNetworkClient {\n private readonly baseUrl?: string;\n private readonly defaultHeaders: Record<string, string>;\n private readonly fetchImpl?: typeof fetch;\n\n constructor(options: DmkNetworkClientOptions = {}) {\n this.baseUrl = options.baseUrl;\n this.defaultHeaders = options.headers ?? {};\n this.fetchImpl = options.fetch;\n }\n\n private getFetch(): typeof fetch {\n // Resolve `fetch` at call time so that spies installed on `globalThis.fetch`\n // after the client was constructed are still honored.\n return this.fetchImpl ?? globalThis.fetch.bind(globalThis);\n }\n\n public get(url: string, config?: DmkRequestConfig): Promise<unknown> {\n return this.request({ ...config, method: \"GET\", url }).then(\n (res) => res.data,\n );\n }\n\n public post(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"POST\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public put(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"PUT\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public patch(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"PATCH\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public delete(url: string, config?: DmkRequestConfig): Promise<unknown> {\n return this.request({ ...config, method: \"DELETE\", url }).then(\n (res) => res.data,\n );\n }\n\n public head(url: string, config?: DmkRequestConfig): Promise<void> {\n return this.request({\n ...config,\n method: \"HEAD\",\n url,\n responseType: \"void\",\n }).then(() => undefined);\n }\n\n /**\n * Escape hatch returning the full response envelope (status, headers, data).\n * Most callers should prefer {@link get}, {@link post}, etc.\n *\n * `data` is `unknown` by design; validate it with a type guard before use.\n */\n public async request(\n config: InternalRequestConfig,\n ): Promise<DmkNetworkResponse> {\n const url = buildUrl({\n url: config.url,\n params: config.params,\n baseUrl: this.baseUrl,\n });\n const { body, headers } = buildBodyAndHeaders({\n method: config.method,\n body: config.body,\n defaultHeaders: this.defaultHeaders,\n perRequestHeaders: config.headers,\n });\n const signal = buildSignal({\n timeoutMs: config.timeoutMs,\n externalSignal: config.signal,\n });\n const throwOnHttpError = config.throwOnHttpError ?? true;\n const responseType: DmkResponseType = config.responseType ?? \"json\";\n\n let response: Response;\n try {\n response = await this.getFetch()(url, {\n method: config.method,\n headers,\n body,\n signal,\n });\n } catch (cause) {\n throw wrapFetchError({\n cause,\n externalSignal: config.signal,\n timeoutMs: config.timeoutMs,\n });\n }\n\n if (!response.ok && throwOnHttpError) {\n const responseBody = await safeReadText(response);\n throw new DmkNetworkClientError({\n message: `HTTP error ${response.status} ${response.statusText}`.trim(),\n status: response.status,\n statusText: response.statusText,\n responseBody,\n });\n }\n\n const data = await parseBody(response, responseType);\n\n return {\n data,\n status: response.status,\n statusText: response.statusText,\n headers: response.headers,\n ok: response.ok,\n };\n }\n}\n"],
5
- "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,sBAAAE,IAAA,eAAAC,EAAAH,GAAA,IAAAI,EAAsC,mCACtCC,EASO,qCAkFA,MAAMH,CAAiB,CACX,QACA,eACA,UAEjB,YAAYI,EAAmC,CAAC,EAAG,CACjD,KAAK,QAAUA,EAAQ,QACvB,KAAK,eAAiBA,EAAQ,SAAW,CAAC,EAC1C,KAAK,UAAYA,EAAQ,KAC3B,CAEQ,UAAyB,CAG/B,OAAO,KAAK,WAAa,WAAW,MAAM,KAAK,UAAU,CAC3D,CAEO,IAAIC,EAAaC,EAA6C,CACnE,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,MAAO,IAAAD,CAAI,CAAC,EAAE,KACpDE,GAAQA,EAAI,IACf,CACF,CAEO,KACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,OAAQ,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC3DD,GAAQA,EAAI,IACf,CACF,CAEO,IACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,MAAO,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC1DD,GAAQA,EAAI,IACf,CACF,CAEO,MACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,QAAS,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC5DD,GAAQA,EAAI,IACf,CACF,CAEO,OAAOF,EAAaC,EAA6C,CACtE,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,SAAU,IAAAD,CAAI,CAAC,EAAE,KACvDE,GAAQA,EAAI,IACf,CACF,CAEO,KAAKF,EAAaC,EAA0C,CACjE,OAAO,KAAK,QAAQ,CAClB,GAAGA,EACH,OAAQ,OACR,IAAAD,EACA,aAAc,MAChB,CAAC,EAAE,KAAK,IAAG,EAAY,CACzB,CAQA,MAAa,QACXC,EAC6B,CAC7B,MAAMD,KAAM,YAAS,CACnB,IAAKC,EAAO,IACZ,OAAQA,EAAO,OACf,QAAS,KAAK,OAChB,CAAC,EACK,CAAE,KAAAE,EAAM,QAAAC,CAAQ,KAAI,uBAAoB,CAC5C,OAAQH,EAAO,OACf,KAAMA,EAAO,KACb,eAAgB,KAAK,eACrB,kBAAmBA,EAAO,OAC5B,CAAC,EACKI,KAAS,eAAY,CACzB,UAAWJ,EAAO,UAClB,eAAgBA,EAAO,MACzB,CAAC,EACKK,EAAmBL,EAAO,kBAAoB,GAC9CM,EAAgCN,EAAO,cAAgB,OAE7D,IAAIO,EACJ,GAAI,CACFA,EAAW,MAAM,KAAK,SAAS,EAAER,EAAK,CACpC,OAAQC,EAAO,OACf,QAAAG,EACA,KAAAD,EACA,OAAAE,CACF,CAAC,CACH,OAASI,EAAO,CACd,QAAM,kBAAe,CACnB,MAAAA,EACA,eAAgBR,EAAO,OACvB,UAAWA,EAAO,SACpB,CAAC,CACH,CAEA,GAAI,CAACO,EAAS,IAAMF,EAAkB,CACpC,MAAMI,EAAe,QAAM,gBAAaF,CAAQ,EAChD,MAAM,IAAI,wBAAsB,CAC9B,QAAS,cAAcA,EAAS,MAAM,IAAIA,EAAS,UAAU,GAAG,KAAK,EACrE,OAAQA,EAAS,OACjB,WAAYA,EAAS,WACrB,aAAAE,CACF,CAAC,CACH,CAIA,MAAO,CACL,KAHW,QAAM,aAAUF,EAAUD,CAAY,EAIjD,OAAQC,EAAS,OACjB,WAAYA,EAAS,WACrB,QAASA,EAAS,QAClB,GAAIA,EAAS,EACf,CACF,CACF",
6
- "names": ["DmkNetworkClient_exports", "__export", "DmkNetworkClient", "__toCommonJS", "import_DmkNetworkClientError", "import_DmkNetworkClientHelpers", "options", "url", "config", "res", "body", "headers", "signal", "throwOnHttpError", "responseType", "response", "cause", "responseBody"]
4
+ "sourcesContent": ["import { DmkNetworkClientError } from \"./DmkNetworkClientError\";\nimport {\n buildBodyAndHeaders,\n buildSignal,\n buildUrl,\n type DmkQueryParams,\n type DmkResponseType,\n parseBody,\n safeReadText,\n wrapFetchError,\n} from \"./DmkNetworkClientHelpers\";\n\nexport type { DmkQueryParamValue } from \"./DmkNetworkClientHelpers\";\nexport type { DmkQueryParams, DmkResponseType };\n\n/**\n * Per-request configuration. Everything is optional \u2014 sensible defaults are\n * applied by the client.\n */\nexport type DmkRequestConfig = {\n /** Query params merged into the URL. `null`/`undefined` entries are skipped. */\n params?: DmkQueryParams;\n /** Per-request headers merged on top of the client's default headers. */\n headers?: Record<string, string>;\n /**\n * Per-request timeout in milliseconds. When unset (or `0`), the request\n * has no timeout.\n */\n timeoutMs?: number;\n /**\n * How to parse the response body. Defaults to `\"json\"`, except for `head`\n * which always resolves to `void`.\n */\n responseType?: DmkResponseType;\n /**\n * When `true` (default), non-2xx responses throw {@link DmkNetworkClientError}.\n * Set to `false` to resolve normally and inspect {@link DmkNetworkResponse}\n * via {@link DmkNetworkClient.request}.\n */\n throwOnHttpError?: boolean;\n};\n\n/**\n * Full response envelope returned by {@link DmkNetworkClient.request}.\n * The simple method helpers (`get`, `post`, \u2026) unwrap to `data` directly.\n *\n * `data` is intentionally typed as `unknown`: network payloads are untrusted\n * input, so callers must narrow it with a type guard or runtime validator\n * (e.g. the existing DTO mappers) before use.\n */\nexport type DmkNetworkResponse = {\n data: unknown;\n status: number;\n statusText: string;\n headers: Headers;\n ok: boolean;\n};\n\nexport type DmkNetworkClientOptions = {\n /** Base URL prepended to relative request URLs. */\n baseUrl?: string;\n /** Default headers merged into every request. */\n headers?: Record<string, string>;\n /** Injection point for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n};\n\ntype InternalRequestConfig = DmkRequestConfig & {\n method: string;\n url: string;\n body?: unknown;\n};\n\n/**\n * Minimal axios-like wrapper over `fetch`. Handles:\n *\n * - URL composition (base URL + relative path + query params from an object)\n * - Automatic JSON body encoding and `Content-Type` header\n * - Default and per-request headers merging\n * - Request timeout via `AbortController`\n * - Automatic `response.ok` check with a typed {@link DmkNetworkClientError}\n * - Typed JSON / text / blob / arrayBuffer response parsing\n *\n * Use the high-level helpers ({@link DmkNetworkClient.get}, {@link DmkNetworkClient.post},\n * \u2026) for 95% of calls and {@link DmkNetworkClient.request} when you need the\n * full response envelope (status, headers).\n */\nexport class DmkNetworkClient {\n private readonly baseUrl?: string;\n private readonly defaultHeaders: Record<string, string>;\n private readonly fetchImpl?: typeof fetch;\n\n constructor(options: DmkNetworkClientOptions = {}) {\n this.baseUrl = options.baseUrl;\n this.defaultHeaders = options.headers ?? {};\n this.fetchImpl = options.fetch;\n }\n\n private getFetch(): typeof fetch {\n // Resolve `fetch` at call time so that spies installed on `globalThis.fetch`\n // after the client was constructed are still honored.\n return this.fetchImpl ?? globalThis.fetch.bind(globalThis);\n }\n\n public get(url: string, config?: DmkRequestConfig): Promise<unknown> {\n return this.request({ ...config, method: \"GET\", url }).then(\n (res) => res.data,\n );\n }\n\n public post(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"POST\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public put(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"PUT\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public patch(\n url: string,\n body?: unknown,\n config?: DmkRequestConfig,\n ): Promise<unknown> {\n return this.request({ ...config, method: \"PATCH\", url, body }).then(\n (res) => res.data,\n );\n }\n\n public delete(url: string, config?: DmkRequestConfig): Promise<unknown> {\n return this.request({ ...config, method: \"DELETE\", url }).then(\n (res) => res.data,\n );\n }\n\n public head(url: string, config?: DmkRequestConfig): Promise<void> {\n return this.request({\n ...config,\n method: \"HEAD\",\n url,\n responseType: \"void\",\n }).then(() => undefined);\n }\n\n /**\n * Escape hatch returning the full response envelope (status, headers, data).\n * Most callers should prefer {@link get}, {@link post}, etc.\n *\n * `data` is `unknown` by design; validate it with a type guard before use.\n */\n public async request(\n config: InternalRequestConfig,\n ): Promise<DmkNetworkResponse> {\n const url = buildUrl({\n url: config.url,\n params: config.params,\n baseUrl: this.baseUrl,\n });\n const { body, headers } = buildBodyAndHeaders({\n method: config.method,\n body: config.body,\n defaultHeaders: this.defaultHeaders,\n perRequestHeaders: config.headers,\n });\n const { signal, cleanup } = buildSignal({\n timeoutMs: config.timeoutMs,\n });\n const throwOnHttpError = config.throwOnHttpError ?? true;\n const responseType: DmkResponseType = config.responseType ?? \"json\";\n\n try {\n let response: Response;\n try {\n response = await this.getFetch()(url, {\n method: config.method,\n headers,\n body,\n signal,\n });\n } catch (cause) {\n throw wrapFetchError({\n cause,\n timeoutMs: config.timeoutMs,\n });\n }\n\n if (!response.ok && throwOnHttpError) {\n const responseBody = await safeReadText(response);\n throw new DmkNetworkClientError({\n message:\n `HTTP error ${response.status} ${response.statusText}`.trim(),\n status: response.status,\n statusText: response.statusText,\n responseBody,\n });\n }\n\n const data = await parseBody(response, responseType);\n\n return {\n data,\n status: response.status,\n statusText: response.statusText,\n headers: response.headers,\n ok: response.ok,\n };\n } finally {\n cleanup();\n }\n }\n}\n"],
5
+ "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,sBAAAE,IAAA,eAAAC,EAAAH,GAAA,IAAAI,EAAsC,mCACtCC,EASO,qCA6EA,MAAMH,CAAiB,CACX,QACA,eACA,UAEjB,YAAYI,EAAmC,CAAC,EAAG,CACjD,KAAK,QAAUA,EAAQ,QACvB,KAAK,eAAiBA,EAAQ,SAAW,CAAC,EAC1C,KAAK,UAAYA,EAAQ,KAC3B,CAEQ,UAAyB,CAG/B,OAAO,KAAK,WAAa,WAAW,MAAM,KAAK,UAAU,CAC3D,CAEO,IAAIC,EAAaC,EAA6C,CACnE,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,MAAO,IAAAD,CAAI,CAAC,EAAE,KACpDE,GAAQA,EAAI,IACf,CACF,CAEO,KACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,OAAQ,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC3DD,GAAQA,EAAI,IACf,CACF,CAEO,IACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,MAAO,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC1DD,GAAQA,EAAI,IACf,CACF,CAEO,MACLF,EACAG,EACAF,EACkB,CAClB,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,QAAS,IAAAD,EAAK,KAAAG,CAAK,CAAC,EAAE,KAC5DD,GAAQA,EAAI,IACf,CACF,CAEO,OAAOF,EAAaC,EAA6C,CACtE,OAAO,KAAK,QAAQ,CAAE,GAAGA,EAAQ,OAAQ,SAAU,IAAAD,CAAI,CAAC,EAAE,KACvDE,GAAQA,EAAI,IACf,CACF,CAEO,KAAKF,EAAaC,EAA0C,CACjE,OAAO,KAAK,QAAQ,CAClB,GAAGA,EACH,OAAQ,OACR,IAAAD,EACA,aAAc,MAChB,CAAC,EAAE,KAAK,IAAG,EAAY,CACzB,CAQA,MAAa,QACXC,EAC6B,CAC7B,MAAMD,KAAM,YAAS,CACnB,IAAKC,EAAO,IACZ,OAAQA,EAAO,OACf,QAAS,KAAK,OAChB,CAAC,EACK,CAAE,KAAAE,EAAM,QAAAC,CAAQ,KAAI,uBAAoB,CAC5C,OAAQH,EAAO,OACf,KAAMA,EAAO,KACb,eAAgB,KAAK,eACrB,kBAAmBA,EAAO,OAC5B,CAAC,EACK,CAAE,OAAAI,EAAQ,QAAAC,CAAQ,KAAI,eAAY,CACtC,UAAWL,EAAO,SACpB,CAAC,EACKM,EAAmBN,EAAO,kBAAoB,GAC9CO,EAAgCP,EAAO,cAAgB,OAE7D,GAAI,CACF,IAAIQ,EACJ,GAAI,CACFA,EAAW,MAAM,KAAK,SAAS,EAAET,EAAK,CACpC,OAAQC,EAAO,OACf,QAAAG,EACA,KAAAD,EACA,OAAAE,CACF,CAAC,CACH,OAASK,EAAO,CACd,QAAM,kBAAe,CACnB,MAAAA,EACA,UAAWT,EAAO,SACpB,CAAC,CACH,CAEA,GAAI,CAACQ,EAAS,IAAMF,EAAkB,CACpC,MAAMI,EAAe,QAAM,gBAAaF,CAAQ,EAChD,MAAM,IAAI,wBAAsB,CAC9B,QACE,cAAcA,EAAS,MAAM,IAAIA,EAAS,UAAU,GAAG,KAAK,EAC9D,OAAQA,EAAS,OACjB,WAAYA,EAAS,WACrB,aAAAE,CACF,CAAC,CACH,CAIA,MAAO,CACL,KAHW,QAAM,aAAUF,EAAUD,CAAY,EAIjD,OAAQC,EAAS,OACjB,WAAYA,EAAS,WACrB,QAASA,EAAS,QAClB,GAAIA,EAAS,EACf,CACF,QAAE,CACAH,EAAQ,CACV,CACF,CACF",
6
+ "names": ["DmkNetworkClient_exports", "__export", "DmkNetworkClient", "__toCommonJS", "import_DmkNetworkClientError", "import_DmkNetworkClientHelpers", "options", "url", "config", "res", "body", "headers", "signal", "cleanup", "throwOnHttpError", "responseType", "response", "cause", "responseBody"]
7
7
  }
@@ -1,2 +1,2 @@
1
- "use strict";var s=require("./DmkNetworkClient"),i=require("./DmkNetworkClientError");describe("DmkNetworkClient",()=>{const n=(e,o={})=>new Response(JSON.stringify(e),{status:200,headers:{"Content-Type":"application/json"},...o});describe("URL composition",()=>{it("should send requests to an absolute URL as-is",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");const[t]=e.mock.calls[0];expect(typeof t).toBe("string"),expect(t).toBe("https://api.example.com/items")}),it("should prepend baseUrl to relative paths with slash normalization",async()=>{const e=vi.fn().mockImplementation(()=>Promise.resolve(n({ok:!0}))),o=new s.DmkNetworkClient({baseUrl:"https://api.example.com/",fetch:e});await o.get("/items"),await o.get("items"),expect(e.mock.calls[0][0]).toBe("https://api.example.com/items"),expect(e.mock.calls[1][0]).toBe("https://api.example.com/items")}),it("should set URL search params from the config",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{params:{chain:1,contract:"0xabc",active:!0,skip:null,alsoSkip:void 0}});const t=e.mock.calls[0][0];expect(typeof t).toBe("string");const c=new URL(t);expect(c.searchParams.get("chain")).toBe("1"),expect(c.searchParams.get("contract")).toBe("0xabc"),expect(c.searchParams.get("active")).toBe("true"),expect(c.searchParams.has("skip")).toBe(!1),expect(c.searchParams.has("alsoSkip")).toBe(!1)}),it("should pass a plain string URL to fetch with no trailing slash after the query (facebook/react-native#54242)",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).get("https://manager.api.live.ledger.com/api/get_device_version",{params:{target_id:858783748,provider:1}});const[t]=e.mock.calls[0];expect(typeof t).toBe("string"),expect(t).toBe("https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1"),expect(t.endsWith("/")).toBe(!1)})}),describe("headers",()=>{it("should merge default and per-request headers",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e,headers:{"X-Default":"default","X-Shared":"from-default"}}).get("https://api.example.com/items",{headers:{"X-Shared":"overridden","X-Per-Request":"per"}});const t=e.mock.calls[0][1];expect(t.headers).toMatchObject({"X-Default":"default","X-Shared":"overridden","X-Per-Request":"per"})})}),describe("body handling",()=>{it("should JSON-stringify plain-object bodies and set Content-Type",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).post("https://api.example.com/items",{foo:"bar"});const t=e.mock.calls[0][1];expect(t.method).toBe("POST"),expect(t.body).toBe(JSON.stringify({foo:"bar"})),expect(t.headers).toMatchObject({"Content-Type":"application/json"})}),it("should not override an explicit Content-Type header",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).post("https://api.example.com/items",{foo:"bar"},{headers:{"content-type":"application/vnd.custom+json"}});const t=e.mock.calls[0][1];expect(t.headers).toMatchObject({"content-type":"application/vnd.custom+json"})}),it("should pass raw BodyInit values through unchanged",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0})),o=new s.DmkNetworkClient({fetch:e}),t=new FormData;t.set("field","value"),await o.post("https://api.example.com/items",t);const c=e.mock.calls[0][1];expect(c.body).toBe(t),expect(c.headers).not.toMatchObject({"Content-Type":"application/json"})}),it("should not send a body on GET or HEAD",async()=>{const e=vi.fn().mockImplementation(()=>Promise.resolve(n({ok:!0}))),o=new s.DmkNetworkClient({fetch:e});await o.get("https://api.example.com/items"),await o.head("https://api.example.com/items"),expect(e.mock.calls[0][1]).toMatchObject({body:void 0}),expect(e.mock.calls[1][1]).toMatchObject({body:void 0})})}),describe("response parsing",()=>{it("should return the parsed JSON body by default",async()=>{const e=vi.fn().mockResolvedValue(n({hello:"world"})),t=await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");expect(t).toEqual({hello:"world"})}),it("should return text when responseType is 'text'",async()=>{const e=vi.fn().mockResolvedValue(new Response("plain-body",{status:200})),t=await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{responseType:"text"});expect(t).toBe("plain-body")}),it("should resolve HEAD to void without reading the body",async()=>{const e=vi.fn().mockResolvedValue(new Response(null,{status:200})),t=await new s.DmkNetworkClient({fetch:e}).head("https://api.example.com/items");expect(t).toBeUndefined()}),it("should resolve empty JSON body to undefined",async()=>{const e=vi.fn().mockResolvedValue(new Response(null,{status:204})),t=await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");expect(t).toBeUndefined()}),it("should throw a DmkNetworkClientError on malformed JSON",async()=>{const e=vi.fn().mockResolvedValue(new Response("not-json",{status:200})),o=new s.DmkNetworkClient({fetch:e});await expect(o.get("https://api.example.com/items")).rejects.toBeInstanceOf(i.DmkNetworkClientError)})}),describe("error handling",()=>{it("should throw DmkNetworkClientError with status on non-2xx responses",async()=>{const e=vi.fn().mockResolvedValue(new Response("boom",{status:500,statusText:"Server Error"})),t=await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items").catch(a=>a);expect(t).toBeInstanceOf(i.DmkNetworkClientError);const c=t;expect(c.status).toBe(500),expect(c.statusText).toBe("Server Error"),expect(c.responseBody).toBe("boom"),expect(c.isTimeout).toBe(!1),expect(c.isAbort).toBe(!1)}),it("should not throw when throwOnHttpError is disabled",async()=>{const e=vi.fn().mockResolvedValue(new Response("boom",{status:500})),t=await new s.DmkNetworkClient({fetch:e}).request({method:"GET",url:"https://api.example.com/items",responseType:"text",throwOnHttpError:!1});expect(t.status).toBe(500),expect(t.ok).toBe(!1),expect(t.data).toBe("boom")}),it("should wrap generic fetch failures into DmkNetworkClientError",async()=>{const e=new TypeError("network down"),o=vi.fn().mockRejectedValue(e),c=await new s.DmkNetworkClient({fetch:o}).get("https://api.example.com/items").catch(a=>a);expect(c).toBeInstanceOf(i.DmkNetworkClientError),expect(c.cause).toBe(e)})}),describe("timeout",()=>{it("should pass an AbortSignal when timeoutMs is configured",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{timeoutMs:1e3});const t=e.mock.calls[0][1];expect(t.signal).toBeInstanceOf(AbortSignal)}),it("should not pass a signal when no timeout and no external signal are set",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new s.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");const t=e.mock.calls[0][1];expect(t.signal).toBeUndefined()}),it("should mark the error as a timeout when fetch rejects with TimeoutError",async()=>{const e=new Error("The operation was aborted");e.name="TimeoutError";const o=vi.fn().mockRejectedValue(e),c=await new s.DmkNetworkClient({fetch:o}).get("https://api.example.com/items",{timeoutMs:10}).catch(a=>a);expect(c).toBeInstanceOf(i.DmkNetworkClientError),expect(c.isTimeout).toBe(!0),expect(c.isAbort).toBe(!1)}),it("should mark the error as an abort when the caller signal is aborted",async()=>{const e=new AbortController;e.abort();const o=new Error("aborted");o.name="AbortError";const t=vi.fn().mockRejectedValue(o),a=await new s.DmkNetworkClient({fetch:t}).get("https://api.example.com/items",{signal:e.signal}).catch(r=>r);expect(a).toBeInstanceOf(i.DmkNetworkClientError),expect(a.isAbort).toBe(!0),expect(a.isTimeout).toBe(!1)})}),describe("request envelope",()=>{it("should expose full response metadata via request()",async()=>{const e=vi.fn().mockResolvedValue(new Response(JSON.stringify({hello:"world"}),{status:201,statusText:"Created",headers:{"X-Custom":"1","Content-Type":"application/json"}})),t=await new s.DmkNetworkClient({fetch:e}).request({method:"GET",url:"https://api.example.com/items"});expect(t.status).toBe(201),expect(t.statusText).toBe("Created"),expect(t.ok).toBe(!0),expect(t.data).toEqual({hello:"world"}),expect(t.headers.get("X-Custom")).toBe("1")})})});
1
+ "use strict";var c=require("./DmkNetworkClient"),i=require("./DmkNetworkClientError");describe("DmkNetworkClient",()=>{const n=(e,o={})=>new Response(JSON.stringify(e),{status:200,headers:{"Content-Type":"application/json"},...o});describe("URL composition",()=>{it("should send requests to an absolute URL as-is",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");const[t]=e.mock.calls[0];expect(typeof t).toBe("string"),expect(t).toBe("https://api.example.com/items")}),it("should prepend baseUrl to relative paths with slash normalization",async()=>{const e=vi.fn().mockImplementation(()=>Promise.resolve(n({ok:!0}))),o=new c.DmkNetworkClient({baseUrl:"https://api.example.com/",fetch:e});await o.get("/items"),await o.get("items"),expect(e.mock.calls[0][0]).toBe("https://api.example.com/items"),expect(e.mock.calls[1][0]).toBe("https://api.example.com/items")}),it("should set URL search params from the config",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{params:{chain:1,contract:"0xabc",active:!0,skip:null,alsoSkip:void 0}});const t=e.mock.calls[0][0];expect(typeof t).toBe("string");const s=new URL(t);expect(s.searchParams.get("chain")).toBe("1"),expect(s.searchParams.get("contract")).toBe("0xabc"),expect(s.searchParams.get("active")).toBe("true"),expect(s.searchParams.has("skip")).toBe(!1),expect(s.searchParams.has("alsoSkip")).toBe(!1)}),it("should pass a plain string URL to fetch with no trailing slash after the query (facebook/react-native#54242)",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://manager.api.live.ledger.com/api/get_device_version",{params:{target_id:858783748,provider:1}});const[t]=e.mock.calls[0];expect(typeof t).toBe("string"),expect(t).toBe("https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1"),expect(t.endsWith("/")).toBe(!1)})}),describe("headers",()=>{it("should merge default and per-request headers",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e,headers:{"X-Default":"default","X-Shared":"from-default"}}).get("https://api.example.com/items",{headers:{"X-Shared":"overridden","X-Per-Request":"per"}});const t=e.mock.calls[0][1];expect(t.headers).toMatchObject({"X-Default":"default","X-Shared":"overridden","X-Per-Request":"per"})})}),describe("body handling",()=>{it("should JSON-stringify plain-object bodies and set Content-Type",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).post("https://api.example.com/items",{foo:"bar"});const t=e.mock.calls[0][1];expect(t.method).toBe("POST"),expect(t.body).toBe(JSON.stringify({foo:"bar"})),expect(t.headers).toMatchObject({"Content-Type":"application/json"})}),it("should not override an explicit Content-Type header",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).post("https://api.example.com/items",{foo:"bar"},{headers:{"content-type":"application/vnd.custom+json"}});const t=e.mock.calls[0][1];expect(t.headers).toMatchObject({"content-type":"application/vnd.custom+json"})}),it("should pass raw BodyInit values through unchanged",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0})),o=new c.DmkNetworkClient({fetch:e}),t=new FormData;t.set("field","value"),await o.post("https://api.example.com/items",t);const s=e.mock.calls[0][1];expect(s.body).toBe(t),expect(s.headers).not.toMatchObject({"Content-Type":"application/json"})}),it("should not send a body on GET or HEAD",async()=>{const e=vi.fn().mockImplementation(()=>Promise.resolve(n({ok:!0}))),o=new c.DmkNetworkClient({fetch:e});await o.get("https://api.example.com/items"),await o.head("https://api.example.com/items"),expect(e.mock.calls[0][1]).toMatchObject({body:void 0}),expect(e.mock.calls[1][1]).toMatchObject({body:void 0})})}),describe("response parsing",()=>{it("should return the parsed JSON body by default",async()=>{const e=vi.fn().mockResolvedValue(n({hello:"world"})),t=await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");expect(t).toEqual({hello:"world"})}),it("should return text when responseType is 'text'",async()=>{const e=vi.fn().mockResolvedValue(new Response("plain-body",{status:200})),t=await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{responseType:"text"});expect(t).toBe("plain-body")}),it("should resolve HEAD to void without reading the body",async()=>{const e=vi.fn().mockResolvedValue(new Response(null,{status:200})),t=await new c.DmkNetworkClient({fetch:e}).head("https://api.example.com/items");expect(t).toBeUndefined()}),it("should resolve empty JSON body to undefined",async()=>{const e=vi.fn().mockResolvedValue(new Response(null,{status:204})),t=await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");expect(t).toBeUndefined()}),it("should throw a DmkNetworkClientError on malformed JSON",async()=>{const e=vi.fn().mockResolvedValue(new Response("not-json",{status:200})),o=new c.DmkNetworkClient({fetch:e});await expect(o.get("https://api.example.com/items")).rejects.toBeInstanceOf(i.DmkNetworkClientError)})}),describe("error handling",()=>{it("should throw DmkNetworkClientError with status on non-2xx responses",async()=>{const e=vi.fn().mockResolvedValue(new Response("boom",{status:500,statusText:"Server Error"})),t=await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items").catch(a=>a);expect(t).toBeInstanceOf(i.DmkNetworkClientError);const s=t;expect(s.status).toBe(500),expect(s.statusText).toBe("Server Error"),expect(s.responseBody).toBe("boom"),expect(s.isTimeout).toBe(!1)}),it("should not throw when throwOnHttpError is disabled",async()=>{const e=vi.fn().mockResolvedValue(new Response("boom",{status:500})),t=await new c.DmkNetworkClient({fetch:e}).request({method:"GET",url:"https://api.example.com/items",responseType:"text",throwOnHttpError:!1});expect(t.status).toBe(500),expect(t.ok).toBe(!1),expect(t.data).toBe("boom")}),it("should wrap generic fetch failures into DmkNetworkClientError",async()=>{const e=new TypeError("network down"),o=vi.fn().mockRejectedValue(e),s=await new c.DmkNetworkClient({fetch:o}).get("https://api.example.com/items").catch(a=>a);expect(s).toBeInstanceOf(i.DmkNetworkClientError),expect(s.cause).toBe(e)})}),describe("timeout",()=>{afterEach(()=>{vi.useRealTimers()}),it("should pass an AbortSignal when timeoutMs is configured",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{timeoutMs:1e3});const t=e.mock.calls[0][1];expect(t.signal).toBeInstanceOf(AbortSignal)}),it("should not pass a signal when no timeout and no external signal are set",async()=>{const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items");const t=e.mock.calls[0][1];expect(t.signal).toBeUndefined()}),it("GIVEN a successful request with timeout WHEN fetch resolves THEN it clears the timeout",async()=>{vi.useFakeTimers();const e=vi.fn().mockResolvedValue(n({ok:!0}));await new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{timeoutMs:1e3});const t=e.mock.calls[0][1];vi.advanceTimersByTime(1e3),expect(t.signal).toBeInstanceOf(AbortSignal),expect(t.signal?.aborted).toBe(!1)}),it("GIVEN a failed request with timeout WHEN fetch rejects THEN it clears the timeout",async()=>{vi.useFakeTimers();const e=new TypeError("network down"),o=vi.fn().mockRejectedValue(e);await new c.DmkNetworkClient({fetch:o}).get("https://api.example.com/items",{timeoutMs:1e3}).catch(()=>{});const s=o.mock.calls[0][1];vi.advanceTimersByTime(1e3),expect(s.signal).toBeInstanceOf(AbortSignal),expect(s.signal?.aborted).toBe(!1)}),it("GIVEN a pending request with timeout WHEN the timeout signal aborts fetch THEN it throws a timeout error",async()=>{vi.useFakeTimers();const e=vi.fn((a,l)=>new Promise((m,p)=>{l?.signal?.addEventListener("abort",()=>{const r=new Error("The operation was aborted");r.name="AbortError",p(r)})})),t=new c.DmkNetworkClient({fetch:e}).get("https://api.example.com/items",{timeoutMs:1e3}).catch(a=>a);await vi.advanceTimersByTimeAsync(1e3);const s=await t;expect(s).toBeInstanceOf(i.DmkNetworkClientError),expect(s.isTimeout).toBe(!0)}),it("should mark the error as a timeout when fetch rejects with TimeoutError",async()=>{const e=new Error("The operation was aborted");e.name="TimeoutError";const o=vi.fn().mockRejectedValue(e),s=await new c.DmkNetworkClient({fetch:o}).get("https://api.example.com/items",{timeoutMs:10}).catch(a=>a);expect(s).toBeInstanceOf(i.DmkNetworkClientError),expect(s.isTimeout).toBe(!0)})}),describe("request envelope",()=>{it("should expose full response metadata via request()",async()=>{const e=vi.fn().mockResolvedValue(new Response(JSON.stringify({hello:"world"}),{status:201,statusText:"Created",headers:{"X-Custom":"1","Content-Type":"application/json"}})),t=await new c.DmkNetworkClient({fetch:e}).request({method:"GET",url:"https://api.example.com/items"});expect(t.status).toBe(201),expect(t.statusText).toBe("Created"),expect(t.ok).toBe(!0),expect(t.data).toEqual({hello:"world"}),expect(t.headers.get("X-Custom")).toBe("1")})})});
2
2
  //# sourceMappingURL=DmkNetworkClient.test.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/api/network/DmkNetworkClient.test.ts"],
4
- "sourcesContent": ["import { DmkNetworkClient } from \"./DmkNetworkClient\";\nimport { DmkNetworkClientError } from \"./DmkNetworkClientError\";\n\ndescribe(\"DmkNetworkClient\", () => {\n const jsonResponse = (body: unknown, init: ResponseInit = {}) =>\n new Response(JSON.stringify(body), {\n status: 200,\n headers: { \"Content-Type\": \"application/json\" },\n ...init,\n });\n\n describe(\"URL composition\", () => {\n it(\"should send requests to an absolute URL as-is\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n\n const [calledUrl] = fetchMock.mock.calls[0]!;\n expect(typeof calledUrl).toBe(\"string\");\n expect(calledUrl).toBe(\"https://api.example.com/items\");\n });\n\n it(\"should prepend baseUrl to relative paths with slash normalization\", async () => {\n const fetchMock = vi\n .fn()\n .mockImplementation(() => Promise.resolve(jsonResponse({ ok: true })));\n const client = new DmkNetworkClient({\n baseUrl: \"https://api.example.com/\",\n fetch: fetchMock,\n });\n\n await client.get(\"/items\");\n await client.get(\"items\");\n\n expect(fetchMock.mock.calls[0]![0]).toBe(\"https://api.example.com/items\");\n expect(fetchMock.mock.calls[1]![0]).toBe(\"https://api.example.com/items\");\n });\n\n it(\"should set URL search params from the config\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\", {\n params: {\n chain: 1,\n contract: \"0xabc\",\n active: true,\n skip: null,\n alsoSkip: undefined,\n },\n });\n\n const calledUrl = fetchMock.mock.calls[0]![0] as string;\n expect(typeof calledUrl).toBe(\"string\");\n const url = new URL(calledUrl);\n expect(url.searchParams.get(\"chain\")).toBe(\"1\");\n expect(url.searchParams.get(\"contract\")).toBe(\"0xabc\");\n expect(url.searchParams.get(\"active\")).toBe(\"true\");\n expect(url.searchParams.has(\"skip\")).toBe(false);\n expect(url.searchParams.has(\"alsoSkip\")).toBe(false);\n });\n\n it(\"should pass a plain string URL to fetch with no trailing slash after the query (facebook/react-native#54242)\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\n \"https://manager.api.live.ledger.com/api/get_device_version\",\n { params: { target_id: 858783748, provider: 1 } },\n );\n\n const [calledUrl] = fetchMock.mock.calls[0]!;\n expect(typeof calledUrl).toBe(\"string\");\n expect(calledUrl).toBe(\n \"https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1\",\n );\n expect((calledUrl as string).endsWith(\"/\")).toBe(false);\n });\n });\n\n describe(\"headers\", () => {\n it(\"should merge default and per-request headers\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({\n fetch: fetchMock,\n headers: { \"X-Default\": \"default\", \"X-Shared\": \"from-default\" },\n });\n\n await client.get(\"https://api.example.com/items\", {\n headers: { \"X-Shared\": \"overridden\", \"X-Per-Request\": \"per\" },\n });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.headers).toMatchObject({\n \"X-Default\": \"default\",\n \"X-Shared\": \"overridden\",\n \"X-Per-Request\": \"per\",\n });\n });\n });\n\n describe(\"body handling\", () => {\n it(\"should JSON-stringify plain-object bodies and set Content-Type\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.post(\"https://api.example.com/items\", { foo: \"bar\" });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.method).toBe(\"POST\");\n expect(init.body).toBe(JSON.stringify({ foo: \"bar\" }));\n expect(init.headers).toMatchObject({\n \"Content-Type\": \"application/json\",\n });\n });\n\n it(\"should not override an explicit Content-Type header\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.post(\n \"https://api.example.com/items\",\n { foo: \"bar\" },\n { headers: { \"content-type\": \"application/vnd.custom+json\" } },\n );\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.headers).toMatchObject({\n \"content-type\": \"application/vnd.custom+json\",\n });\n });\n\n it(\"should pass raw BodyInit values through unchanged\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n const formData = new FormData();\n formData.set(\"field\", \"value\");\n\n await client.post(\"https://api.example.com/items\", formData);\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.body).toBe(formData);\n expect(init.headers).not.toMatchObject({\n \"Content-Type\": \"application/json\",\n });\n });\n\n it(\"should not send a body on GET or HEAD\", async () => {\n const fetchMock = vi\n .fn()\n .mockImplementation(() => Promise.resolve(jsonResponse({ ok: true })));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n await client.head(\"https://api.example.com/items\");\n\n expect(fetchMock.mock.calls[0]![1]).toMatchObject({ body: undefined });\n expect(fetchMock.mock.calls[1]![1]).toMatchObject({ body: undefined });\n });\n });\n\n describe(\"response parsing\", () => {\n it(\"should return the parsed JSON body by default\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(jsonResponse({ hello: \"world\" }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\");\n\n expect(result).toEqual({ hello: \"world\" });\n });\n\n it(\"should return text when responseType is 'text'\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"plain-body\", { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\", {\n responseType: \"text\",\n });\n\n expect(result).toBe(\"plain-body\");\n });\n\n it(\"should resolve HEAD to void without reading the body\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(null, { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.head(\"https://api.example.com/items\");\n\n expect(result).toBeUndefined();\n });\n\n it(\"should resolve empty JSON body to undefined\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(null, { status: 204 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\");\n\n expect(result).toBeUndefined();\n });\n\n it(\"should throw a DmkNetworkClientError on malformed JSON\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"not-json\", { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await expect(\n client.get(\"https://api.example.com/items\"),\n ).rejects.toBeInstanceOf(DmkNetworkClientError);\n });\n });\n\n describe(\"error handling\", () => {\n it(\"should throw DmkNetworkClientError with status on non-2xx responses\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(\n new Response(\"boom\", {\n status: 500,\n statusText: \"Server Error\",\n }),\n );\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\")\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n const dmkError = error as DmkNetworkClientError;\n expect(dmkError.status).toBe(500);\n expect(dmkError.statusText).toBe(\"Server Error\");\n expect(dmkError.responseBody).toBe(\"boom\");\n expect(dmkError.isTimeout).toBe(false);\n expect(dmkError.isAbort).toBe(false);\n });\n\n it(\"should not throw when throwOnHttpError is disabled\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"boom\", { status: 500 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const response = await client.request({\n method: \"GET\",\n url: \"https://api.example.com/items\",\n responseType: \"text\",\n throwOnHttpError: false,\n });\n\n expect(response.status).toBe(500);\n expect(response.ok).toBe(false);\n expect(response.data).toBe(\"boom\");\n });\n\n it(\"should wrap generic fetch failures into DmkNetworkClientError\", async () => {\n const cause = new TypeError(\"network down\");\n const fetchMock = vi.fn().mockRejectedValue(cause);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\")\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).cause).toBe(cause);\n });\n });\n\n describe(\"timeout\", () => {\n it(\"should pass an AbortSignal when timeoutMs is configured\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\", { timeoutMs: 1000 });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.signal).toBeInstanceOf(AbortSignal);\n });\n\n it(\"should not pass a signal when no timeout and no external signal are set\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.signal).toBeUndefined();\n });\n\n it(\"should mark the error as a timeout when fetch rejects with TimeoutError\", async () => {\n const timeoutError = new Error(\"The operation was aborted\");\n timeoutError.name = \"TimeoutError\";\n const fetchMock = vi.fn().mockRejectedValue(timeoutError);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\", { timeoutMs: 10 })\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).isTimeout).toBe(true);\n expect((error as DmkNetworkClientError).isAbort).toBe(false);\n });\n\n it(\"should mark the error as an abort when the caller signal is aborted\", async () => {\n const abortController = new AbortController();\n abortController.abort();\n const abortError = new Error(\"aborted\");\n abortError.name = \"AbortError\";\n const fetchMock = vi.fn().mockRejectedValue(abortError);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\", {\n signal: abortController.signal,\n })\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).isAbort).toBe(true);\n expect((error as DmkNetworkClientError).isTimeout).toBe(false);\n });\n });\n\n describe(\"request envelope\", () => {\n it(\"should expose full response metadata via request()\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(\n new Response(JSON.stringify({ hello: \"world\" }), {\n status: 201,\n statusText: \"Created\",\n headers: { \"X-Custom\": \"1\", \"Content-Type\": \"application/json\" },\n }),\n );\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const response = await client.request({\n method: \"GET\",\n url: \"https://api.example.com/items\",\n });\n\n expect(response.status).toBe(201);\n expect(response.statusText).toBe(\"Created\");\n expect(response.ok).toBe(true);\n expect(response.data).toEqual({ hello: \"world\" });\n expect(response.headers.get(\"X-Custom\")).toBe(\"1\");\n });\n });\n});\n"],
5
- "mappings": "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",
6
- "names": ["import_DmkNetworkClient", "import_DmkNetworkClientError", "jsonResponse", "body", "init", "fetchMock", "calledUrl", "client", "url", "formData", "result", "error", "e", "dmkError", "response", "cause", "timeoutError", "abortController", "abortError"]
4
+ "sourcesContent": ["import { DmkNetworkClient } from \"./DmkNetworkClient\";\nimport { DmkNetworkClientError } from \"./DmkNetworkClientError\";\n\ndescribe(\"DmkNetworkClient\", () => {\n const jsonResponse = (body: unknown, init: ResponseInit = {}) =>\n new Response(JSON.stringify(body), {\n status: 200,\n headers: { \"Content-Type\": \"application/json\" },\n ...init,\n });\n\n describe(\"URL composition\", () => {\n it(\"should send requests to an absolute URL as-is\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n\n const [calledUrl] = fetchMock.mock.calls[0]!;\n expect(typeof calledUrl).toBe(\"string\");\n expect(calledUrl).toBe(\"https://api.example.com/items\");\n });\n\n it(\"should prepend baseUrl to relative paths with slash normalization\", async () => {\n const fetchMock = vi\n .fn()\n .mockImplementation(() => Promise.resolve(jsonResponse({ ok: true })));\n const client = new DmkNetworkClient({\n baseUrl: \"https://api.example.com/\",\n fetch: fetchMock,\n });\n\n await client.get(\"/items\");\n await client.get(\"items\");\n\n expect(fetchMock.mock.calls[0]![0]).toBe(\"https://api.example.com/items\");\n expect(fetchMock.mock.calls[1]![0]).toBe(\"https://api.example.com/items\");\n });\n\n it(\"should set URL search params from the config\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\", {\n params: {\n chain: 1,\n contract: \"0xabc\",\n active: true,\n skip: null,\n alsoSkip: undefined,\n },\n });\n\n const calledUrl = fetchMock.mock.calls[0]![0] as string;\n expect(typeof calledUrl).toBe(\"string\");\n const url = new URL(calledUrl);\n expect(url.searchParams.get(\"chain\")).toBe(\"1\");\n expect(url.searchParams.get(\"contract\")).toBe(\"0xabc\");\n expect(url.searchParams.get(\"active\")).toBe(\"true\");\n expect(url.searchParams.has(\"skip\")).toBe(false);\n expect(url.searchParams.has(\"alsoSkip\")).toBe(false);\n });\n\n it(\"should pass a plain string URL to fetch with no trailing slash after the query (facebook/react-native#54242)\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\n \"https://manager.api.live.ledger.com/api/get_device_version\",\n { params: { target_id: 858783748, provider: 1 } },\n );\n\n const [calledUrl] = fetchMock.mock.calls[0]!;\n expect(typeof calledUrl).toBe(\"string\");\n expect(calledUrl).toBe(\n \"https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1\",\n );\n expect((calledUrl as string).endsWith(\"/\")).toBe(false);\n });\n });\n\n describe(\"headers\", () => {\n it(\"should merge default and per-request headers\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({\n fetch: fetchMock,\n headers: { \"X-Default\": \"default\", \"X-Shared\": \"from-default\" },\n });\n\n await client.get(\"https://api.example.com/items\", {\n headers: { \"X-Shared\": \"overridden\", \"X-Per-Request\": \"per\" },\n });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.headers).toMatchObject({\n \"X-Default\": \"default\",\n \"X-Shared\": \"overridden\",\n \"X-Per-Request\": \"per\",\n });\n });\n });\n\n describe(\"body handling\", () => {\n it(\"should JSON-stringify plain-object bodies and set Content-Type\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.post(\"https://api.example.com/items\", { foo: \"bar\" });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.method).toBe(\"POST\");\n expect(init.body).toBe(JSON.stringify({ foo: \"bar\" }));\n expect(init.headers).toMatchObject({\n \"Content-Type\": \"application/json\",\n });\n });\n\n it(\"should not override an explicit Content-Type header\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.post(\n \"https://api.example.com/items\",\n { foo: \"bar\" },\n { headers: { \"content-type\": \"application/vnd.custom+json\" } },\n );\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.headers).toMatchObject({\n \"content-type\": \"application/vnd.custom+json\",\n });\n });\n\n it(\"should pass raw BodyInit values through unchanged\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n const formData = new FormData();\n formData.set(\"field\", \"value\");\n\n await client.post(\"https://api.example.com/items\", formData);\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.body).toBe(formData);\n expect(init.headers).not.toMatchObject({\n \"Content-Type\": \"application/json\",\n });\n });\n\n it(\"should not send a body on GET or HEAD\", async () => {\n const fetchMock = vi\n .fn()\n .mockImplementation(() => Promise.resolve(jsonResponse({ ok: true })));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n await client.head(\"https://api.example.com/items\");\n\n expect(fetchMock.mock.calls[0]![1]).toMatchObject({ body: undefined });\n expect(fetchMock.mock.calls[1]![1]).toMatchObject({ body: undefined });\n });\n });\n\n describe(\"response parsing\", () => {\n it(\"should return the parsed JSON body by default\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(jsonResponse({ hello: \"world\" }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\");\n\n expect(result).toEqual({ hello: \"world\" });\n });\n\n it(\"should return text when responseType is 'text'\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"plain-body\", { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\", {\n responseType: \"text\",\n });\n\n expect(result).toBe(\"plain-body\");\n });\n\n it(\"should resolve HEAD to void without reading the body\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(null, { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.head(\"https://api.example.com/items\");\n\n expect(result).toBeUndefined();\n });\n\n it(\"should resolve empty JSON body to undefined\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(null, { status: 204 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const result = await client.get(\"https://api.example.com/items\");\n\n expect(result).toBeUndefined();\n });\n\n it(\"should throw a DmkNetworkClientError on malformed JSON\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"not-json\", { status: 200 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await expect(\n client.get(\"https://api.example.com/items\"),\n ).rejects.toBeInstanceOf(DmkNetworkClientError);\n });\n });\n\n describe(\"error handling\", () => {\n it(\"should throw DmkNetworkClientError with status on non-2xx responses\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(\n new Response(\"boom\", {\n status: 500,\n statusText: \"Server Error\",\n }),\n );\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\")\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n const dmkError = error as DmkNetworkClientError;\n expect(dmkError.status).toBe(500);\n expect(dmkError.statusText).toBe(\"Server Error\");\n expect(dmkError.responseBody).toBe(\"boom\");\n expect(dmkError.isTimeout).toBe(false);\n });\n\n it(\"should not throw when throwOnHttpError is disabled\", async () => {\n const fetchMock = vi\n .fn()\n .mockResolvedValue(new Response(\"boom\", { status: 500 }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const response = await client.request({\n method: \"GET\",\n url: \"https://api.example.com/items\",\n responseType: \"text\",\n throwOnHttpError: false,\n });\n\n expect(response.status).toBe(500);\n expect(response.ok).toBe(false);\n expect(response.data).toBe(\"boom\");\n });\n\n it(\"should wrap generic fetch failures into DmkNetworkClientError\", async () => {\n const cause = new TypeError(\"network down\");\n const fetchMock = vi.fn().mockRejectedValue(cause);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\")\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).cause).toBe(cause);\n });\n });\n\n describe(\"timeout\", () => {\n afterEach(() => {\n vi.useRealTimers();\n });\n\n it(\"should pass an AbortSignal when timeoutMs is configured\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\", { timeoutMs: 1000 });\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.signal).toBeInstanceOf(AbortSignal);\n });\n\n it(\"should not pass a signal when no timeout and no external signal are set\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\");\n\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n expect(init.signal).toBeUndefined();\n });\n\n it(\"GIVEN a successful request with timeout WHEN fetch resolves THEN it clears the timeout\", async () => {\n vi.useFakeTimers();\n const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ ok: true }));\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client.get(\"https://api.example.com/items\", { timeoutMs: 1000 });\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n\n vi.advanceTimersByTime(1000);\n\n expect(init.signal).toBeInstanceOf(AbortSignal);\n expect(init.signal?.aborted).toBe(false);\n });\n\n it(\"GIVEN a failed request with timeout WHEN fetch rejects THEN it clears the timeout\", async () => {\n vi.useFakeTimers();\n const cause = new TypeError(\"network down\");\n const fetchMock = vi.fn().mockRejectedValue(cause);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n await client\n .get(\"https://api.example.com/items\", { timeoutMs: 1000 })\n .catch(() => undefined);\n const init = fetchMock.mock.calls[0]![1] as RequestInit;\n\n vi.advanceTimersByTime(1000);\n\n expect(init.signal).toBeInstanceOf(AbortSignal);\n expect(init.signal?.aborted).toBe(false);\n });\n\n it(\"GIVEN a pending request with timeout WHEN the timeout signal aborts fetch THEN it throws a timeout error\", async () => {\n vi.useFakeTimers();\n const fetchMock = vi.fn(\n (_url: string | URL | Request, init?: RequestInit) =>\n new Promise<Response>((_resolve, reject) => {\n init?.signal?.addEventListener(\"abort\", () => {\n const abortError = new Error(\"The operation was aborted\");\n abortError.name = \"AbortError\";\n reject(abortError);\n });\n }),\n );\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const request = client\n .get(\"https://api.example.com/items\", { timeoutMs: 1000 })\n .catch((e: unknown) => e);\n\n await vi.advanceTimersByTimeAsync(1000);\n const error = await request;\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).isTimeout).toBe(true);\n });\n\n it(\"should mark the error as a timeout when fetch rejects with TimeoutError\", async () => {\n const timeoutError = new Error(\"The operation was aborted\");\n timeoutError.name = \"TimeoutError\";\n const fetchMock = vi.fn().mockRejectedValue(timeoutError);\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const error = await client\n .get(\"https://api.example.com/items\", { timeoutMs: 10 })\n .catch((e: unknown) => e);\n\n expect(error).toBeInstanceOf(DmkNetworkClientError);\n expect((error as DmkNetworkClientError).isTimeout).toBe(true);\n });\n });\n\n describe(\"request envelope\", () => {\n it(\"should expose full response metadata via request()\", async () => {\n const fetchMock = vi.fn().mockResolvedValue(\n new Response(JSON.stringify({ hello: \"world\" }), {\n status: 201,\n statusText: \"Created\",\n headers: { \"X-Custom\": \"1\", \"Content-Type\": \"application/json\" },\n }),\n );\n const client = new DmkNetworkClient({ fetch: fetchMock });\n\n const response = await client.request({\n method: \"GET\",\n url: \"https://api.example.com/items\",\n });\n\n expect(response.status).toBe(201);\n expect(response.statusText).toBe(\"Created\");\n expect(response.ok).toBe(true);\n expect(response.data).toEqual({ hello: \"world\" });\n expect(response.headers.get(\"X-Custom\")).toBe(\"1\");\n });\n });\n});\n"],
5
+ "mappings": "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",
6
+ "names": ["import_DmkNetworkClient", "import_DmkNetworkClientError", "jsonResponse", "body", "init", "fetchMock", "calledUrl", "client", "url", "formData", "result", "error", "e", "dmkError", "response", "cause", "_url", "_resolve", "reject", "abortError", "request", "timeoutError"]
7
7
  }
@@ -1,2 +1,2 @@
1
- "use strict";var r=Object.defineProperty;var n=Object.getOwnPropertyDescriptor;var u=Object.getOwnPropertyNames;var l=Object.prototype.hasOwnProperty;var a=(s,e)=>{for(var o in e)r(s,o,{get:e[o],enumerable:!0})},b=(s,e,o,i)=>{if(e&&typeof e=="object"||typeof e=="function")for(let t of u(e))!l.call(s,t)&&t!==o&&r(s,t,{get:()=>e[t],enumerable:!(i=n(e,t))||i.enumerable});return s};var c=s=>b(r({},"__esModule",{value:!0}),s);var y={};a(y,{DmkNetworkClientError:()=>d});module.exports=c(y);class d extends Error{status;statusText;responseBody;isTimeout;isAbort;cause;constructor(e){super(e.message),this.name="DmkNetworkClientError",this.status=e.status,this.statusText=e.statusText,this.responseBody=e.responseBody,this.isTimeout=e.isTimeout??!1,this.isAbort=e.isAbort??!1,this.cause=e.cause}}0&&(module.exports={DmkNetworkClientError});
1
+ "use strict";var o=Object.defineProperty;var i=Object.getOwnPropertyDescriptor;var u=Object.getOwnPropertyNames;var a=Object.prototype.hasOwnProperty;var l=(s,e)=>{for(var r in e)o(s,r,{get:e[r],enumerable:!0})},c=(s,e,r,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let t of u(e))!a.call(s,t)&&t!==r&&o(s,t,{get:()=>e[t],enumerable:!(n=i(e,t))||n.enumerable});return s};var d=s=>c(o({},"__esModule",{value:!0}),s);var m={};l(m,{DmkNetworkClientError:()=>k});module.exports=d(m);class k extends Error{status;statusText;responseBody;isTimeout;cause;constructor(e){super(e.message),this.name="DmkNetworkClientError",this.status=e.status,this.statusText=e.statusText,this.responseBody=e.responseBody,this.isTimeout=e.isTimeout??!1,this.cause=e.cause}}0&&(module.exports={DmkNetworkClientError});
2
2
  //# sourceMappingURL=DmkNetworkClientError.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/api/network/DmkNetworkClientError.ts"],
4
- "sourcesContent": ["export type DmkNetworkClientErrorParams = {\n message: string;\n status?: number;\n statusText?: string;\n responseBody?: string;\n isTimeout?: boolean;\n isAbort?: boolean;\n cause?: unknown;\n};\n\n/**\n * Error thrown by {@link DmkNetworkClient} for HTTP, timeout, abort or\n * transport-level failures.\n *\n * - When the remote returned a non-2xx response, {@link status} and\n * {@link statusText} are populated and {@link responseBody} contains the\n * raw text body (best effort).\n * - When the request timed out via the client's `timeoutMs`,\n * {@link isTimeout} is `true`.\n * - When the request was aborted through the caller's `signal`,\n * {@link isAbort} is `true`.\n * - For other fetch/network failures, {@link cause} carries the original\n * error.\n */\nexport class DmkNetworkClientError extends Error {\n public readonly status?: number;\n public readonly statusText?: string;\n public readonly responseBody?: string;\n public readonly isTimeout: boolean;\n public readonly isAbort: boolean;\n public override readonly cause?: unknown;\n\n constructor(params: DmkNetworkClientErrorParams) {\n super(params.message);\n this.name = \"DmkNetworkClientError\";\n this.status = params.status;\n this.statusText = params.statusText;\n this.responseBody = params.responseBody;\n this.isTimeout = params.isTimeout ?? false;\n this.isAbort = params.isAbort ?? false;\n this.cause = params.cause;\n }\n}\n"],
5
- "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,2BAAAE,IAAA,eAAAC,EAAAH,GAwBO,MAAME,UAA8B,KAAM,CAC/B,OACA,WACA,aACA,UACA,QACS,MAEzB,YAAYE,EAAqC,CAC/C,MAAMA,EAAO,OAAO,EACpB,KAAK,KAAO,wBACZ,KAAK,OAASA,EAAO,OACrB,KAAK,WAAaA,EAAO,WACzB,KAAK,aAAeA,EAAO,aAC3B,KAAK,UAAYA,EAAO,WAAa,GACrC,KAAK,QAAUA,EAAO,SAAW,GACjC,KAAK,MAAQA,EAAO,KACtB,CACF",
4
+ "sourcesContent": ["export type DmkNetworkClientErrorParams = {\n message: string;\n status?: number;\n statusText?: string;\n responseBody?: string;\n isTimeout?: boolean;\n cause?: unknown;\n};\n\n/**\n * Error thrown by {@link DmkNetworkClient} for HTTP, timeout or transport-level\n * failures.\n *\n * - When the remote returned a non-2xx response, {@link status} and\n * {@link statusText} are populated and {@link responseBody} contains the\n * raw text body (best effort).\n * - When the request timed out via the client's `timeoutMs`,\n * {@link isTimeout} is `true`.\n * - For other fetch/network failures, {@link cause} carries the original\n * error.\n */\nexport class DmkNetworkClientError extends Error {\n public readonly status?: number;\n public readonly statusText?: string;\n public readonly responseBody?: string;\n public readonly isTimeout: boolean;\n public override readonly cause?: unknown;\n\n constructor(params: DmkNetworkClientErrorParams) {\n super(params.message);\n this.name = \"DmkNetworkClientError\";\n this.status = params.status;\n this.statusText = params.statusText;\n this.responseBody = params.responseBody;\n this.isTimeout = params.isTimeout ?? false;\n this.cause = params.cause;\n }\n}\n"],
5
+ "mappings": "yaAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,2BAAAE,IAAA,eAAAC,EAAAH,GAqBO,MAAME,UAA8B,KAAM,CAC/B,OACA,WACA,aACA,UACS,MAEzB,YAAYE,EAAqC,CAC/C,MAAMA,EAAO,OAAO,EACpB,KAAK,KAAO,wBACZ,KAAK,OAASA,EAAO,OACrB,KAAK,WAAaA,EAAO,WACzB,KAAK,aAAeA,EAAO,aAC3B,KAAK,UAAYA,EAAO,WAAa,GACrC,KAAK,MAAQA,EAAO,KACtB,CACF",
6
6
  "names": ["DmkNetworkClientError_exports", "__export", "DmkNetworkClientError", "__toCommonJS", "params"]
7
7
  }
@@ -1,2 +1,2 @@
1
- "use strict";var d=Object.defineProperty;var g=Object.getOwnPropertyDescriptor;var p=Object.getOwnPropertyNames;var y=Object.prototype.hasOwnProperty;var b=(t,e)=>{for(var r in e)d(t,r,{get:e[r],enumerable:!0})},x=(t,e,r,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let o of p(e))!y.call(t,o)&&o!==r&&d(t,o,{get:()=>e[o],enumerable:!(n=g(e,o))||n.enumerable});return t};var w=t=>x(d({},"__esModule",{value:!0}),t);var T={};b(T,{JSON_CONTENT_TYPE:()=>f,buildBodyAndHeaders:()=>k,buildSignal:()=>h,buildUrl:()=>S,hasHeader:()=>l,isRawBody:()=>m,joinPath:()=>c,parseBody:()=>A,safeReadText:()=>B,wrapFetchError:()=>E});module.exports=w(T);var u=require("./DmkNetworkClientError");const f="application/json";function c(t,e){if(!e)return t;const r=t.endsWith("/")?t.slice(0,-1):t,n=e.startsWith("/")?e:`/${e}`;return`${r}${n}`}function R(t){const e=[];for(const[r,n]of Object.entries(t))n!=null&&e.push(`${encodeURIComponent(r)}=${encodeURIComponent(String(n))}`);return e.join("&")}function S(t){const{url:e,params:r,baseUrl:n}=t,i=/^[a-z][a-z0-9+.-]*:\/\//i.test(e)?e:n?c(n,e):e,s=r?R(r):"";if(s.length===0)return i;const a=i.includes("?")?"&":"?";return`${i}${a}${s}`}function l(t,e){const r=e.toLowerCase();return Object.keys(t).some(n=>n.toLowerCase()===r)}function m(t){return!!(typeof t=="string"||t instanceof ArrayBuffer||ArrayBuffer.isView(t)||typeof Blob<"u"&&t instanceof Blob||typeof FormData<"u"&&t instanceof FormData||typeof URLSearchParams<"u"&&t instanceof URLSearchParams||typeof ReadableStream<"u"&&t instanceof ReadableStream)}function k(t){const{method:e,body:r,defaultHeaders:n,perRequestHeaders:o}=t,i={...n,...o};return r===void 0||e==="GET"||e==="HEAD"?{body:void 0,headers:i}:m(r)?{body:r,headers:i}:(l(i,"content-type")||(i["Content-Type"]=f),{body:JSON.stringify(r),headers:i})}function h(t){const{timeoutMs:e,externalSignal:r}=t,n=e&&e>0?AbortSignal.timeout(e):void 0;return n&&r?AbortSignal.any([r,n]):n??r}async function A(t,e){switch(e){case"void":return;case"text":return await t.text();case"blob":return await t.blob();case"arrayBuffer":return await t.arrayBuffer();case"json":default:{const r=await t.text();if(r.length===0)return;try{return JSON.parse(r)}catch(n){throw new u.DmkNetworkClientError({message:"Failed to parse JSON response body",status:t.status,statusText:t.statusText,responseBody:r,cause:n})}}}}async function B(t){try{return await t.text()}catch{return}}function E(t){const{cause:e,externalSignal:r,timeoutMs:n}=t,o=!!(n&&n>0);if(e instanceof Error&&(e.name==="AbortError"||e.name==="TimeoutError")){const s=r?.aborted??!1,a=!s&&(e.name==="TimeoutError"||o);return new u.DmkNetworkClientError({message:a?"Request timed out":"Request aborted",isTimeout:a,isAbort:s,cause:e})}return new u.DmkNetworkClientError({message:e instanceof Error?e.message:"Network request failed",cause:e})}0&&(module.exports={JSON_CONTENT_TYPE,buildBodyAndHeaders,buildSignal,buildUrl,hasHeader,isRawBody,joinPath,parseBody,safeReadText,wrapFetchError});
1
+ "use strict";var a=Object.defineProperty;var p=Object.getOwnPropertyDescriptor;var g=Object.getOwnPropertyNames;var y=Object.prototype.hasOwnProperty;var b=(t,e)=>{for(var r in e)a(t,r,{get:e[r],enumerable:!0})},w=(t,e,r,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let s of g(e))!y.call(t,s)&&s!==r&&a(t,s,{get:()=>e[s],enumerable:!(n=p(e,s))||n.enumerable});return t};var x=t=>w(a({},"__esModule",{value:!0}),t);var S={};b(S,{JSON_CONTENT_TYPE:()=>d,buildBodyAndHeaders:()=>T,buildSignal:()=>h,buildUrl:()=>k,hasHeader:()=>f,isRawBody:()=>m,joinPath:()=>c,parseBody:()=>B,safeReadText:()=>E,wrapFetchError:()=>P});module.exports=x(S);var i=require("./DmkNetworkClientError");const d="application/json";function c(t,e){if(!e)return t;const r=t.endsWith("/")?t.slice(0,-1):t,n=e.startsWith("/")?e:`/${e}`;return`${r}${n}`}function R(t){const e=[];for(const[r,n]of Object.entries(t))n!=null&&e.push(`${encodeURIComponent(r)}=${encodeURIComponent(String(n))}`);return e.join("&")}function k(t){const{url:e,params:r,baseUrl:n}=t,o=/^[a-z][a-z0-9+.-]*:\/\//i.test(e)?e:n?c(n,e):e,u=r?R(r):"";if(u.length===0)return o;const l=o.includes("?")?"&":"?";return`${o}${l}${u}`}function f(t,e){const r=e.toLowerCase();return Object.keys(t).some(n=>n.toLowerCase()===r)}function m(t){return!!(typeof t=="string"||t instanceof ArrayBuffer||ArrayBuffer.isView(t)||typeof Blob<"u"&&t instanceof Blob||typeof FormData<"u"&&t instanceof FormData||typeof URLSearchParams<"u"&&t instanceof URLSearchParams||typeof ReadableStream<"u"&&t instanceof ReadableStream)}function T(t){const{method:e,body:r,defaultHeaders:n,perRequestHeaders:s}=t,o={...n,...s};return r===void 0||e==="GET"||e==="HEAD"?{body:void 0,headers:o}:m(r)?{body:r,headers:o}:(f(o,"content-type")||(o["Content-Type"]=d),{body:JSON.stringify(r),headers:o})}function h(t){const{timeoutMs:e}=t;if(!e||e<=0)return{signal:void 0,cleanup:()=>{}};const r=new AbortController,n=setTimeout(()=>{r.abort()},e);return{signal:r.signal,cleanup:()=>clearTimeout(n)}}async function B(t,e){switch(e){case"void":return;case"text":return await t.text();case"blob":return await t.blob();case"arrayBuffer":return await t.arrayBuffer();case"json":default:{const r=await t.text();if(r.length===0)return;try{return JSON.parse(r)}catch(n){throw new i.DmkNetworkClientError({message:"Failed to parse JSON response body",status:t.status,statusText:t.statusText,responseBody:r,cause:n})}}}}async function E(t){try{return await t.text()}catch{return}}function P(t){const{cause:e,timeoutMs:r}=t,n=!!(r&&r>0);if(e instanceof Error&&(e.name==="AbortError"||e.name==="TimeoutError")){const o=e.name==="TimeoutError"||n;return new i.DmkNetworkClientError({message:o?"Request timed out":"Request aborted",isTimeout:o,cause:e})}return new i.DmkNetworkClientError({message:e instanceof Error?e.message:"Network request failed",cause:e})}0&&(module.exports={JSON_CONTENT_TYPE,buildBodyAndHeaders,buildSignal,buildUrl,hasHeader,isRawBody,joinPath,parseBody,safeReadText,wrapFetchError});
2
2
  //# sourceMappingURL=DmkNetworkClientHelpers.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/api/network/DmkNetworkClientHelpers.ts"],
4
- "sourcesContent": ["import { DmkNetworkClientError } from \"./DmkNetworkClientError\";\n\nexport type DmkQueryParamValue = string | number | boolean | null | undefined;\n\nexport type DmkQueryParams = Record<string, DmkQueryParamValue>;\n\nexport type DmkResponseType = \"json\" | \"text\" | \"blob\" | \"arrayBuffer\" | \"void\";\n\nexport const JSON_CONTENT_TYPE = \"application/json\";\n\n/**\n * Joins a base URL with a relative path, normalizing surrounding slashes so\n * that exactly one `/` appears between them.\n */\nexport function joinPath(base: string, path: string): string {\n if (!path) return base;\n const trimmedBase = base.endsWith(\"/\") ? base.slice(0, -1) : base;\n const trimmedPath = path.startsWith(\"/\") ? path : `/${path}`;\n return `${trimmedBase}${trimmedPath}`;\n}\n\n/**\n * Serializes a {@link DmkQueryParams} object into a percent-encoded\n * (RFC 3986) query string (without the leading `?`). `null`/`undefined`\n * entries are skipped.\n *\n * Note: this is not strictly `application/x-www-form-urlencoded` \u2014 spaces\n * are encoded as `%20` (via `encodeURIComponent`) rather than `+`. Any\n * RFC 3986-compliant server will decode both forms identically.\n *\n * Implemented manually rather than via `URLSearchParams` so it works on\n * runtimes where `URLSearchParams.set` is not implemented (e.g. some\n * React Native versions).\n */\nfunction serializeParams(params: DmkQueryParams): string {\n const parts: string[] = [];\n for (const [key, value] of Object.entries(params)) {\n if (value === null || value === undefined) continue;\n parts.push(\n `${encodeURIComponent(key)}=${encodeURIComponent(String(value))}`,\n );\n }\n return parts.join(\"&\");\n}\n\n/**\n * Builds the final request URL as a plain string. `null`/`undefined` params\n * are skipped; params are appended (not merged) when `url` already has a\n * query string; fragments are not supported alongside `params`.\n *\n * Returns a string (not a `URL`) to avoid React Native's `URL.toString()`,\n * which appends a stray trailing `/` to URLs with a query string on affected\n * versions (facebook/react-native#54242).\n */\nexport function buildUrl(args: {\n url: string;\n params?: DmkQueryParams;\n baseUrl?: string;\n}): string {\n const { url, params, baseUrl } = args;\n const isAbsolute = /^[a-z][a-z0-9+.-]*:\\/\\//i.test(url);\n const composed = isAbsolute ? url : baseUrl ? joinPath(baseUrl, url) : url;\n\n const query = params ? serializeParams(params) : \"\";\n if (query.length === 0) {\n return composed;\n }\n\n const separator = composed.includes(\"?\") ? \"&\" : \"?\";\n return `${composed}${separator}${query}`;\n}\n\n/**\n * Case-insensitive lookup for a header name in a plain record.\n */\nexport function hasHeader(\n headers: Record<string, string>,\n name: string,\n): boolean {\n const lower = name.toLowerCase();\n return Object.keys(headers).some((k) => k.toLowerCase() === lower);\n}\n\n/**\n * Returns `true` when the value is already a `BodyInit` accepted by `fetch`\n * and should be passed through without JSON serialization.\n *\n * Each Web-global `instanceof` check is guarded with a `typeof` test so the\n * function is safe on runtimes (e.g. some React Native versions) where\n * `Blob`, `FormData`, `URLSearchParams` or `ReadableStream` are not defined.\n */\nexport function isRawBody(body: unknown): body is BodyInit {\n if (typeof body === \"string\") return true;\n if (body instanceof ArrayBuffer) return true;\n if (ArrayBuffer.isView(body)) return true;\n if (typeof Blob !== \"undefined\" && body instanceof Blob) return true;\n if (typeof FormData !== \"undefined\" && body instanceof FormData) return true;\n if (typeof URLSearchParams !== \"undefined\" && body instanceof URLSearchParams)\n return true;\n if (typeof ReadableStream !== \"undefined\" && body instanceof ReadableStream)\n return true;\n return false;\n}\n\n/**\n * Computes the final request body and merged headers. Plain-object bodies are\n * JSON-serialized and the `Content-Type` header is set (unless the caller\n * already provided one). Raw `BodyInit` values pass through unchanged.\n */\nexport function buildBodyAndHeaders(args: {\n method: string;\n body: unknown;\n defaultHeaders: Record<string, string>;\n perRequestHeaders?: Record<string, string>;\n}): { body?: BodyInit; headers: Record<string, string> } {\n const { method, body, defaultHeaders, perRequestHeaders } = args;\n const headers: Record<string, string> = {\n ...defaultHeaders,\n ...perRequestHeaders,\n };\n\n if (body === undefined || method === \"GET\" || method === \"HEAD\") {\n return { body: undefined, headers };\n }\n\n if (isRawBody(body)) {\n return { body, headers };\n }\n\n if (!hasHeader(headers, \"content-type\")) {\n headers[\"Content-Type\"] = JSON_CONTENT_TYPE;\n }\n return { body: JSON.stringify(body), headers };\n}\n\n/**\n * Composes the effective abort signal for a request. If both a timeout and an\n * external signal are present, they are merged via `AbortSignal.any` so that\n * either one firing aborts the request.\n */\nexport function buildSignal(args: {\n timeoutMs: number | undefined;\n externalSignal: AbortSignal | undefined;\n}): AbortSignal | undefined {\n const { timeoutMs, externalSignal } = args;\n const timeoutSignal =\n timeoutMs && timeoutMs > 0 ? AbortSignal.timeout(timeoutMs) : undefined;\n\n if (timeoutSignal && externalSignal) {\n return AbortSignal.any([externalSignal, timeoutSignal]);\n }\n return timeoutSignal ?? externalSignal;\n}\n\n/**\n * Parses the response body according to the requested {@link DmkResponseType}.\n * Empty JSON bodies resolve to `undefined`; malformed JSON throws a\n * {@link DmkNetworkClientError}.\n */\nexport async function parseBody(\n response: Response,\n responseType: DmkResponseType,\n): Promise<unknown> {\n switch (responseType) {\n case \"void\":\n return undefined;\n case \"text\":\n return await response.text();\n case \"blob\":\n return await response.blob();\n case \"arrayBuffer\":\n return await response.arrayBuffer();\n case \"json\":\n default: {\n const text = await response.text();\n if (text.length === 0) {\n return undefined;\n }\n try {\n return JSON.parse(text);\n } catch (cause) {\n throw new DmkNetworkClientError({\n message: \"Failed to parse JSON response body\",\n status: response.status,\n statusText: response.statusText,\n responseBody: text,\n cause,\n });\n }\n }\n }\n}\n\n/**\n * Reads the response body as text, swallowing any error so callers can use it\n * for best-effort diagnostics (e.g. building an error payload).\n */\nexport async function safeReadText(\n response: Response,\n): Promise<string | undefined> {\n try {\n return await response.text();\n } catch {\n return undefined;\n }\n}\n\n/**\n * Wraps a `fetch` rejection into a typed {@link DmkNetworkClientError},\n * discriminating between external aborts, timeouts and generic failures.\n */\nexport function wrapFetchError(args: {\n cause: unknown;\n externalSignal: AbortSignal | undefined;\n timeoutMs: number | undefined;\n}): DmkNetworkClientError {\n const { cause, externalSignal, timeoutMs } = args;\n const hasTimeout = Boolean(timeoutMs && timeoutMs > 0);\n const isAbortError =\n cause instanceof Error &&\n (cause.name === \"AbortError\" || cause.name === \"TimeoutError\");\n\n if (isAbortError) {\n const externallyAborted = externalSignal?.aborted ?? false;\n const timedOut =\n !externallyAborted && (cause.name === \"TimeoutError\" || hasTimeout);\n return new DmkNetworkClientError({\n message: timedOut ? `Request timed out` : \"Request aborted\",\n isTimeout: timedOut,\n isAbort: externallyAborted,\n cause,\n });\n }\n\n return new DmkNetworkClientError({\n message: cause instanceof Error ? cause.message : \"Network request failed\",\n cause,\n });\n}\n"],
5
- "mappings": "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",
6
- "names": ["DmkNetworkClientHelpers_exports", "__export", "JSON_CONTENT_TYPE", "buildBodyAndHeaders", "buildSignal", "buildUrl", "hasHeader", "isRawBody", "joinPath", "parseBody", "safeReadText", "wrapFetchError", "__toCommonJS", "import_DmkNetworkClientError", "base", "path", "trimmedBase", "trimmedPath", "serializeParams", "params", "parts", "key", "value", "args", "url", "baseUrl", "composed", "query", "separator", "headers", "name", "lower", "k", "body", "method", "defaultHeaders", "perRequestHeaders", "timeoutMs", "externalSignal", "timeoutSignal", "response", "responseType", "text", "cause", "hasTimeout", "externallyAborted", "timedOut"]
4
+ "sourcesContent": ["import { DmkNetworkClientError } from \"./DmkNetworkClientError\";\n\nexport type DmkQueryParamValue = string | number | boolean | null | undefined;\n\nexport type DmkQueryParams = Record<string, DmkQueryParamValue>;\n\nexport type DmkResponseType = \"json\" | \"text\" | \"blob\" | \"arrayBuffer\" | \"void\";\n\nexport const JSON_CONTENT_TYPE = \"application/json\";\n\n/**\n * Joins a base URL with a relative path, normalizing surrounding slashes so\n * that exactly one `/` appears between them.\n */\nexport function joinPath(base: string, path: string): string {\n if (!path) return base;\n const trimmedBase = base.endsWith(\"/\") ? base.slice(0, -1) : base;\n const trimmedPath = path.startsWith(\"/\") ? path : `/${path}`;\n return `${trimmedBase}${trimmedPath}`;\n}\n\n/**\n * Serializes a {@link DmkQueryParams} object into a percent-encoded\n * (RFC 3986) query string (without the leading `?`). `null`/`undefined`\n * entries are skipped.\n *\n * Note: this is not strictly `application/x-www-form-urlencoded` \u2014 spaces\n * are encoded as `%20` (via `encodeURIComponent`) rather than `+`. Any\n * RFC 3986-compliant server will decode both forms identically.\n *\n * Implemented manually rather than via `URLSearchParams` so it works on\n * runtimes where `URLSearchParams.set` is not implemented (e.g. some\n * React Native versions).\n */\nfunction serializeParams(params: DmkQueryParams): string {\n const parts: string[] = [];\n for (const [key, value] of Object.entries(params)) {\n if (value === null || value === undefined) continue;\n parts.push(\n `${encodeURIComponent(key)}=${encodeURIComponent(String(value))}`,\n );\n }\n return parts.join(\"&\");\n}\n\n/**\n * Builds the final request URL as a plain string. `null`/`undefined` params\n * are skipped; params are appended (not merged) when `url` already has a\n * query string; fragments are not supported alongside `params`.\n *\n * Returns a string (not a `URL`) to avoid React Native's `URL.toString()`,\n * which appends a stray trailing `/` to URLs with a query string on affected\n * versions (facebook/react-native#54242).\n */\nexport function buildUrl(args: {\n url: string;\n params?: DmkQueryParams;\n baseUrl?: string;\n}): string {\n const { url, params, baseUrl } = args;\n const isAbsolute = /^[a-z][a-z0-9+.-]*:\\/\\//i.test(url);\n const composed = isAbsolute ? url : baseUrl ? joinPath(baseUrl, url) : url;\n\n const query = params ? serializeParams(params) : \"\";\n if (query.length === 0) {\n return composed;\n }\n\n const separator = composed.includes(\"?\") ? \"&\" : \"?\";\n return `${composed}${separator}${query}`;\n}\n\n/**\n * Case-insensitive lookup for a header name in a plain record.\n */\nexport function hasHeader(\n headers: Record<string, string>,\n name: string,\n): boolean {\n const lower = name.toLowerCase();\n return Object.keys(headers).some((k) => k.toLowerCase() === lower);\n}\n\n/**\n * Returns `true` when the value is already a `BodyInit` accepted by `fetch`\n * and should be passed through without JSON serialization.\n *\n * Each Web-global `instanceof` check is guarded with a `typeof` test so the\n * function is safe on runtimes (e.g. some React Native versions) where\n * `Blob`, `FormData`, `URLSearchParams` or `ReadableStream` are not defined.\n */\nexport function isRawBody(body: unknown): body is BodyInit {\n if (typeof body === \"string\") return true;\n if (body instanceof ArrayBuffer) return true;\n if (ArrayBuffer.isView(body)) return true;\n /* eslint-disable no-restricted-globals -- Each Web global is guarded so the checks are safe when the global is missing in RN/Hermes. */\n if (typeof Blob !== \"undefined\" && body instanceof Blob) return true;\n if (typeof FormData !== \"undefined\" && body instanceof FormData) return true;\n if (typeof URLSearchParams !== \"undefined\" && body instanceof URLSearchParams)\n return true;\n if (typeof ReadableStream !== \"undefined\" && body instanceof ReadableStream)\n return true;\n /* eslint-enable no-restricted-globals */\n return false;\n}\n\n/**\n * Computes the final request body and merged headers. Plain-object bodies are\n * JSON-serialized and the `Content-Type` header is set (unless the caller\n * already provided one). Raw `BodyInit` values pass through unchanged.\n */\nexport function buildBodyAndHeaders(args: {\n method: string;\n body: unknown;\n defaultHeaders: Record<string, string>;\n perRequestHeaders?: Record<string, string>;\n}): { body?: BodyInit; headers: Record<string, string> } {\n const { method, body, defaultHeaders, perRequestHeaders } = args;\n const headers: Record<string, string> = {\n ...defaultHeaders,\n ...perRequestHeaders,\n };\n\n if (body === undefined || method === \"GET\" || method === \"HEAD\") {\n return { body: undefined, headers };\n }\n\n if (isRawBody(body)) {\n return { body, headers };\n }\n\n if (!hasHeader(headers, \"content-type\")) {\n headers[\"Content-Type\"] = JSON_CONTENT_TYPE;\n }\n return { body: JSON.stringify(body), headers };\n}\n\nexport type TimeoutSignal = {\n signal: AbortSignal | undefined;\n cleanup: () => void;\n};\n\n/**\n * Builds an abort signal for request timeouts using `AbortController`.\n *\n * `AbortSignal.timeout` is intentionally avoided because React Native/Hermes\n * does not reliably provide that static helper.\n */\nexport function buildSignal(args: {\n timeoutMs: number | undefined;\n}): TimeoutSignal {\n const { timeoutMs } = args;\n\n if (!timeoutMs || timeoutMs <= 0) {\n return {\n signal: undefined,\n cleanup: () => {},\n };\n }\n\n const controller = new AbortController();\n const timeoutId = setTimeout(() => {\n controller.abort();\n }, timeoutMs);\n\n return {\n signal: controller.signal,\n cleanup: () => clearTimeout(timeoutId),\n };\n}\n\n/**\n * Parses the response body according to the requested {@link DmkResponseType}.\n * Empty JSON bodies resolve to `undefined`; malformed JSON throws a\n * {@link DmkNetworkClientError}.\n */\nexport async function parseBody(\n response: Response,\n responseType: DmkResponseType,\n): Promise<unknown> {\n switch (responseType) {\n case \"void\":\n return undefined;\n case \"text\":\n return await response.text();\n case \"blob\":\n return await response.blob();\n case \"arrayBuffer\":\n return await response.arrayBuffer();\n case \"json\":\n default: {\n const text = await response.text();\n if (text.length === 0) {\n return undefined;\n }\n try {\n return JSON.parse(text);\n } catch (cause) {\n throw new DmkNetworkClientError({\n message: \"Failed to parse JSON response body\",\n status: response.status,\n statusText: response.statusText,\n responseBody: text,\n cause,\n });\n }\n }\n }\n}\n\n/**\n * Reads the response body as text, swallowing any error so callers can use it\n * for best-effort diagnostics (e.g. building an error payload).\n */\nexport async function safeReadText(\n response: Response,\n): Promise<string | undefined> {\n try {\n return await response.text();\n } catch {\n return undefined;\n }\n}\n\n/**\n * Wraps a `fetch` rejection into a typed {@link DmkNetworkClientError},\n * discriminating between request timeouts and generic failures.\n */\nexport function wrapFetchError(args: {\n cause: unknown;\n timeoutMs: number | undefined;\n}): DmkNetworkClientError {\n const { cause, timeoutMs } = args;\n const hasTimeout = Boolean(timeoutMs && timeoutMs > 0);\n const isAbortError =\n cause instanceof Error &&\n (cause.name === \"AbortError\" || cause.name === \"TimeoutError\");\n\n if (isAbortError) {\n const timedOut = cause.name === \"TimeoutError\" || hasTimeout;\n return new DmkNetworkClientError({\n message: timedOut ? `Request timed out` : \"Request aborted\",\n isTimeout: timedOut,\n cause,\n });\n }\n\n return new DmkNetworkClientError({\n message: cause instanceof Error ? cause.message : \"Network request failed\",\n cause,\n });\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["DmkNetworkClientHelpers_exports", "__export", "JSON_CONTENT_TYPE", "buildBodyAndHeaders", "buildSignal", "buildUrl", "hasHeader", "isRawBody", "joinPath", "parseBody", "safeReadText", "wrapFetchError", "__toCommonJS", "import_DmkNetworkClientError", "base", "path", "trimmedBase", "trimmedPath", "serializeParams", "params", "parts", "key", "value", "args", "url", "baseUrl", "composed", "query", "separator", "headers", "name", "lower", "k", "body", "method", "defaultHeaders", "perRequestHeaders", "timeoutMs", "controller", "timeoutId", "response", "responseType", "text", "cause", "hasTimeout", "timedOut"]
7
7
  }
@@ -1,2 +1,2 @@
1
- "use strict";var n=require("./DmkNetworkClientError"),t=require("./DmkNetworkClientHelpers");describe("DmkNetworkClientHelpers",()=>{describe("joinPath",()=>{it("should return the base when the path is empty",()=>{expect((0,t.joinPath)("https://api.example.com","")).toBe("https://api.example.com")}),it("should insert a single slash when neither side has one",()=>{expect((0,t.joinPath)("https://api.example.com","items")).toBe("https://api.example.com/items")}),it("should collapse a trailing base slash with a leading path slash",()=>{expect((0,t.joinPath)("https://api.example.com/","/items")).toBe("https://api.example.com/items")}),it("should keep a single slash when only one side has one",()=>{expect((0,t.joinPath)("https://api.example.com/","items")).toBe("https://api.example.com/items"),expect((0,t.joinPath)("https://api.example.com","/items")).toBe("https://api.example.com/items")})}),describe("buildUrl",()=>{it("should keep an absolute URL as-is",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items"});expect(e).toBe("https://api.example.com/items")}),it("should prepend the base URL to a relative path",()=>{const e=(0,t.buildUrl)({url:"/items",baseUrl:"https://api.example.com/"});expect(e).toBe("https://api.example.com/items")}),it("should ignore the base URL when the input URL is absolute",()=>{const e=(0,t.buildUrl)({url:"https://other.example.com/items",baseUrl:"https://api.example.com"});expect(e).toBe("https://other.example.com/items")}),it("should append query params and skip null/undefined entries",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1,contract:"0xabc",active:!0,skip:null,alsoSkip:void 0}});expect(typeof e).toBe("string");const o=new URL(e);expect(o.searchParams.get("chain")).toBe("1"),expect(o.searchParams.get("contract")).toBe("0xabc"),expect(o.searchParams.get("active")).toBe("true"),expect(o.searchParams.has("skip")).toBe(!1),expect(o.searchParams.has("alsoSkip")).toBe(!1)}),it("should preserve a pre-existing query string in the input URL",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items?keep=1",params:{chain:2}}),o=new URL(e);expect(o.searchParams.get("keep")).toBe("1"),expect(o.searchParams.get("chain")).toBe("2")}),it("should percent-encode keys and values",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{"a key":"a value&b"}});expect(e).toBe("https://api.example.com/items?a%20key=a%20value%26b")}),it("should not append a trailing slash after the last query value (facebook/react-native#54242)",()=>{const e=(0,t.buildUrl)({url:"https://manager.api.live.ledger.com/api/get_device_version",params:{target_id:858783748,provider:1}});expect(typeof e).toBe("string"),expect(e).toBe("https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1"),expect(e.endsWith("/")).toBe(!1)}),it("should return a plain string (not a URL instance) so RN URL serialization is bypassed",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1}});expect(typeof e).toBe("string"),expect(e).not.toBeInstanceOf(URL)}),it("should still build the URL when URLSearchParams.set is unavailable (React Native regression)",()=>{const e=URLSearchParams.prototype.set;URLSearchParams.prototype.set=function(){throw new Error("URLSearchParams.set is not implemented")};try{const o=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1,contract:"0xabc"}});expect(o).toContain("chain=1"),expect(o).toContain("contract=0xabc")}finally{URLSearchParams.prototype.set=e}})}),describe("hasHeader",()=>{it("should find a header regardless of case",()=>{expect((0,t.hasHeader)({"Content-Type":"json"},"content-type")).toBe(!0),expect((0,t.hasHeader)({"content-type":"json"},"Content-Type")).toBe(!0)}),it("should return false when no matching header exists",()=>{expect((0,t.hasHeader)({Accept:"json"},"content-type")).toBe(!1),expect((0,t.hasHeader)({},"content-type")).toBe(!1)})}),describe("isRawBody",()=>{it("should accept native BodyInit values",()=>{expect((0,t.isRawBody)("raw-string")).toBe(!0),expect((0,t.isRawBody)(new ArrayBuffer(4))).toBe(!0),expect((0,t.isRawBody)(new Uint8Array([1,2,3]))).toBe(!0),expect((0,t.isRawBody)(new Blob(["a"]))).toBe(!0),expect((0,t.isRawBody)(new FormData)).toBe(!0),expect((0,t.isRawBody)(new URLSearchParams({a:"1"}))).toBe(!0),expect((0,t.isRawBody)(new ReadableStream)).toBe(!0)}),it("should reject plain objects and primitives",()=>{expect((0,t.isRawBody)({foo:"bar"})).toBe(!1),expect((0,t.isRawBody)([1,2,3])).toBe(!1),expect((0,t.isRawBody)(42)).toBe(!1),expect((0,t.isRawBody)(null)).toBe(!1),expect((0,t.isRawBody)(void 0)).toBe(!1)}),it("should not throw when optional Web globals are missing (React Native regression)",()=>{const e=globalThis,o=["Blob","FormData","URLSearchParams","ReadableStream"],s={};for(const a of o)s[a]=e[a],delete e[a];try{expect(()=>(0,t.isRawBody)({foo:"bar"})).not.toThrow(),expect((0,t.isRawBody)({foo:"bar"})).toBe(!1),expect((0,t.isRawBody)("raw")).toBe(!0),expect((0,t.isRawBody)(new Uint8Array([1]))).toBe(!0)}finally{for(const a of o)e[a]=s[a]}})}),describe("buildBodyAndHeaders",()=>{it("should merge default and per-request headers with per-request winning",()=>{const{headers:e}=(0,t.buildBodyAndHeaders)({method:"GET",body:void 0,defaultHeaders:{"X-Default":"default","X-Shared":"from-default"},perRequestHeaders:{"X-Shared":"overridden","X-Per":"per"}});expect(e).toEqual({"X-Default":"default","X-Shared":"overridden","X-Per":"per"})}),it("should not emit a body on GET/HEAD even when a body is provided",()=>{const e=(0,t.buildBodyAndHeaders)({method:"GET",body:{foo:"bar"},defaultHeaders:{}});expect(e.body).toBeUndefined(),expect(e.headers["Content-Type"]).toBeUndefined()}),it("should JSON-stringify plain objects and set the Content-Type",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:{foo:"bar"},defaultHeaders:{}});expect(e.body).toBe(JSON.stringify({foo:"bar"})),expect(e.headers["Content-Type"]).toBe(t.JSON_CONTENT_TYPE)}),it("should not override an explicit Content-Type header",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:{foo:"bar"},defaultHeaders:{},perRequestHeaders:{"content-type":"application/vnd.custom+json"}});expect(e.body).toBe(JSON.stringify({foo:"bar"})),expect(e.headers["content-type"]).toBe("application/vnd.custom+json"),expect(e.headers["Content-Type"]).toBeUndefined()}),it("should pass raw BodyInit values through unchanged without JSON headers",()=>{const e=new FormData;e.set("field","value");const o=(0,t.buildBodyAndHeaders)({method:"POST",body:e,defaultHeaders:{}});expect(o.body).toBe(e),expect(o.headers["Content-Type"]).toBeUndefined()}),it("should return an undefined body when the body itself is undefined",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:void 0,defaultHeaders:{"X-Default":"d"}});expect(e.body).toBeUndefined(),expect(e.headers).toEqual({"X-Default":"d"})})}),describe("buildSignal",()=>{it("should return undefined when no timeout and no external signal",()=>{expect((0,t.buildSignal)({timeoutMs:void 0,externalSignal:void 0})).toBeUndefined()}),it("should return undefined when timeout is zero and no external signal",()=>{expect((0,t.buildSignal)({timeoutMs:0,externalSignal:void 0})).toBeUndefined()}),it("should return the external signal alone when no timeout is configured",()=>{const e=new AbortController,o=(0,t.buildSignal)({timeoutMs:void 0,externalSignal:e.signal});expect(o).toBe(e.signal)}),it("should return a timeout signal when only a timeout is configured",()=>{const e=(0,t.buildSignal)({timeoutMs:1e3,externalSignal:void 0});expect(e).toBeInstanceOf(AbortSignal)}),it("should use the provided timeout value for AbortSignal.timeout",()=>{const e=vi.spyOn(AbortSignal,"timeout");(0,t.buildSignal)({timeoutMs:50,externalSignal:void 0}),expect(e).toHaveBeenCalledWith(50),e.mockRestore()}),it("should compose both signals with AbortSignal.any when both are set",()=>{const e=vi.spyOn(AbortSignal,"any"),o=new AbortController,s=(0,t.buildSignal)({timeoutMs:100,externalSignal:o.signal});expect(s).toBeInstanceOf(AbortSignal),expect(e).toHaveBeenCalledTimes(1),e.mockRestore()})}),describe("parseBody",()=>{it("should return undefined for 'void' response type without consuming the body",async()=>{const e=new Response("ignored"),o=await(0,t.parseBody)(e,"void");expect(o).toBeUndefined(),expect(e.bodyUsed).toBe(!1)}),it("should return text for 'text' response type",async()=>{const e=new Response("plain-body");await expect((0,t.parseBody)(e,"text")).resolves.toBe("plain-body")}),it("should return a Blob for 'blob' response type",async()=>{const e=new Response("blob-body"),o=await(0,t.parseBody)(e,"blob");expect(o).toBeInstanceOf(Blob)}),it("should return an ArrayBuffer for 'arrayBuffer' response type",async()=>{const e=new Response(new Uint8Array([1,2,3])),o=await(0,t.parseBody)(e,"arrayBuffer");expect(o).toBeInstanceOf(ArrayBuffer),expect(new Uint8Array(o)).toEqual(new Uint8Array([1,2,3]))}),it("should parse JSON for 'json' response type",async()=>{const e=new Response(JSON.stringify({hello:"world"}));await expect((0,t.parseBody)(e,"json")).resolves.toEqual({hello:"world"})}),it("should resolve to undefined for an empty JSON body",async()=>{const e=new Response("");await expect((0,t.parseBody)(e,"json")).resolves.toBeUndefined()}),it("should throw a DmkNetworkClientError for malformed JSON",async()=>{const e=new Response("not-json",{status:200,statusText:"OK"}),o=await(0,t.parseBody)(e,"json").catch(a=>a);expect(o).toBeInstanceOf(n.DmkNetworkClientError);const s=o;expect(s.status).toBe(200),expect(s.statusText).toBe("OK"),expect(s.responseBody).toBe("not-json"),expect(s.cause).toBeInstanceOf(SyntaxError)})}),describe("safeReadText",()=>{it("should return the text body",async()=>{await expect((0,t.safeReadText)(new Response("hello"))).resolves.toBe("hello")}),it("should return undefined when reading the body throws",async()=>{const e={text:()=>Promise.reject(new Error("read failed"))};await expect((0,t.safeReadText)(e)).resolves.toBeUndefined()})}),describe("wrapFetchError",()=>{it("should wrap a generic Error with its message",()=>{const e=new TypeError("network down"),o=(0,t.wrapFetchError)({cause:e,externalSignal:void 0,timeoutMs:void 0});expect(o).toBeInstanceOf(n.DmkNetworkClientError),expect(o.message).toBe("network down"),expect(o.cause).toBe(e),expect(o.isTimeout).toBe(!1),expect(o.isAbort).toBe(!1)}),it("should fall back to a generic message for non-Error causes",()=>{const e=(0,t.wrapFetchError)({cause:"something",externalSignal:void 0,timeoutMs:void 0});expect(e.message).toBe("Network request failed"),expect(e.cause).toBe("something")}),it("should flag TimeoutError as a timeout",()=>{const e=new Error("timed out");e.name="TimeoutError";const o=(0,t.wrapFetchError)({cause:e,externalSignal:void 0,timeoutMs:10});expect(o.isTimeout).toBe(!0),expect(o.isAbort).toBe(!1),expect(o.message).toBe("Request timed out")}),it("should flag AbortError as a timeout when a timeout is configured and no external abort",()=>{const e=new Error("aborted");e.name="AbortError";const o=(0,t.wrapFetchError)({cause:e,externalSignal:void 0,timeoutMs:1e3});expect(o.isTimeout).toBe(!0),expect(o.isAbort).toBe(!1)}),it("should flag AbortError as an external abort when the caller signal is aborted",()=>{const e=new AbortController;e.abort();const o=new Error("aborted");o.name="AbortError";const s=(0,t.wrapFetchError)({cause:o,externalSignal:e.signal,timeoutMs:void 0});expect(s.isAbort).toBe(!0),expect(s.isTimeout).toBe(!1),expect(s.message).toBe("Request aborted")}),it("should treat AbortError with no timeout and no external abort as a plain abort",()=>{const e=new Error("aborted");e.name="AbortError";const o=(0,t.wrapFetchError)({cause:e,externalSignal:void 0,timeoutMs:void 0});expect(o.isTimeout).toBe(!1),expect(o.isAbort).toBe(!1),expect(o.message).toBe("Request aborted")})})});
1
+ "use strict";var n=require("./DmkNetworkClientError"),t=require("./DmkNetworkClientHelpers");describe("DmkNetworkClientHelpers",()=>{describe("joinPath",()=>{it("should return the base when the path is empty",()=>{expect((0,t.joinPath)("https://api.example.com","")).toBe("https://api.example.com")}),it("should insert a single slash when neither side has one",()=>{expect((0,t.joinPath)("https://api.example.com","items")).toBe("https://api.example.com/items")}),it("should collapse a trailing base slash with a leading path slash",()=>{expect((0,t.joinPath)("https://api.example.com/","/items")).toBe("https://api.example.com/items")}),it("should keep a single slash when only one side has one",()=>{expect((0,t.joinPath)("https://api.example.com/","items")).toBe("https://api.example.com/items"),expect((0,t.joinPath)("https://api.example.com","/items")).toBe("https://api.example.com/items")})}),describe("buildUrl",()=>{it("should keep an absolute URL as-is",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items"});expect(e).toBe("https://api.example.com/items")}),it("should prepend the base URL to a relative path",()=>{const e=(0,t.buildUrl)({url:"/items",baseUrl:"https://api.example.com/"});expect(e).toBe("https://api.example.com/items")}),it("should ignore the base URL when the input URL is absolute",()=>{const e=(0,t.buildUrl)({url:"https://other.example.com/items",baseUrl:"https://api.example.com"});expect(e).toBe("https://other.example.com/items")}),it("should append query params and skip null/undefined entries",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1,contract:"0xabc",active:!0,skip:null,alsoSkip:void 0}});expect(typeof e).toBe("string");const o=new URL(e);expect(o.searchParams.get("chain")).toBe("1"),expect(o.searchParams.get("contract")).toBe("0xabc"),expect(o.searchParams.get("active")).toBe("true"),expect(o.searchParams.has("skip")).toBe(!1),expect(o.searchParams.has("alsoSkip")).toBe(!1)}),it("should preserve a pre-existing query string in the input URL",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items?keep=1",params:{chain:2}}),o=new URL(e);expect(o.searchParams.get("keep")).toBe("1"),expect(o.searchParams.get("chain")).toBe("2")}),it("should percent-encode keys and values",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{"a key":"a value&b"}});expect(e).toBe("https://api.example.com/items?a%20key=a%20value%26b")}),it("should not append a trailing slash after the last query value (facebook/react-native#54242)",()=>{const e=(0,t.buildUrl)({url:"https://manager.api.live.ledger.com/api/get_device_version",params:{target_id:858783748,provider:1}});expect(typeof e).toBe("string"),expect(e).toBe("https://manager.api.live.ledger.com/api/get_device_version?target_id=858783748&provider=1"),expect(e.endsWith("/")).toBe(!1)}),it("should return a plain string (not a URL instance) so RN URL serialization is bypassed",()=>{const e=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1}});expect(typeof e).toBe("string"),expect(e).not.toBeInstanceOf(URL)}),it("should still build the URL when URLSearchParams.set is unavailable (React Native regression)",()=>{const e=URLSearchParams.prototype.set;URLSearchParams.prototype.set=function(){throw new Error("URLSearchParams.set is not implemented")};try{const o=(0,t.buildUrl)({url:"https://api.example.com/items",params:{chain:1,contract:"0xabc"}});expect(o).toContain("chain=1"),expect(o).toContain("contract=0xabc")}finally{URLSearchParams.prototype.set=e}})}),describe("hasHeader",()=>{it("should find a header regardless of case",()=>{expect((0,t.hasHeader)({"Content-Type":"json"},"content-type")).toBe(!0),expect((0,t.hasHeader)({"content-type":"json"},"Content-Type")).toBe(!0)}),it("should return false when no matching header exists",()=>{expect((0,t.hasHeader)({Accept:"json"},"content-type")).toBe(!1),expect((0,t.hasHeader)({},"content-type")).toBe(!1)})}),describe("isRawBody",()=>{it("should accept native BodyInit values",()=>{expect((0,t.isRawBody)("raw-string")).toBe(!0),expect((0,t.isRawBody)(new ArrayBuffer(4))).toBe(!0),expect((0,t.isRawBody)(new Uint8Array([1,2,3]))).toBe(!0),expect((0,t.isRawBody)(new Blob(["a"]))).toBe(!0),expect((0,t.isRawBody)(new FormData)).toBe(!0),expect((0,t.isRawBody)(new URLSearchParams({a:"1"}))).toBe(!0),expect((0,t.isRawBody)(new ReadableStream)).toBe(!0)}),it("should reject plain objects and primitives",()=>{expect((0,t.isRawBody)({foo:"bar"})).toBe(!1),expect((0,t.isRawBody)([1,2,3])).toBe(!1),expect((0,t.isRawBody)(42)).toBe(!1),expect((0,t.isRawBody)(null)).toBe(!1),expect((0,t.isRawBody)(void 0)).toBe(!1)}),it("should not throw when optional Web globals are missing (React Native regression)",()=>{const e=globalThis,o=["Blob","FormData","URLSearchParams","ReadableStream"],a={};for(const s of o)a[s]=e[s],delete e[s];try{expect(()=>(0,t.isRawBody)({foo:"bar"})).not.toThrow(),expect((0,t.isRawBody)({foo:"bar"})).toBe(!1),expect((0,t.isRawBody)("raw")).toBe(!0),expect((0,t.isRawBody)(new Uint8Array([1]))).toBe(!0)}finally{for(const s of o)e[s]=a[s]}})}),describe("buildBodyAndHeaders",()=>{it("should merge default and per-request headers with per-request winning",()=>{const{headers:e}=(0,t.buildBodyAndHeaders)({method:"GET",body:void 0,defaultHeaders:{"X-Default":"default","X-Shared":"from-default"},perRequestHeaders:{"X-Shared":"overridden","X-Per":"per"}});expect(e).toEqual({"X-Default":"default","X-Shared":"overridden","X-Per":"per"})}),it("should not emit a body on GET/HEAD even when a body is provided",()=>{const e=(0,t.buildBodyAndHeaders)({method:"GET",body:{foo:"bar"},defaultHeaders:{}});expect(e.body).toBeUndefined(),expect(e.headers["Content-Type"]).toBeUndefined()}),it("should JSON-stringify plain objects and set the Content-Type",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:{foo:"bar"},defaultHeaders:{}});expect(e.body).toBe(JSON.stringify({foo:"bar"})),expect(e.headers["Content-Type"]).toBe(t.JSON_CONTENT_TYPE)}),it("should not override an explicit Content-Type header",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:{foo:"bar"},defaultHeaders:{},perRequestHeaders:{"content-type":"application/vnd.custom+json"}});expect(e.body).toBe(JSON.stringify({foo:"bar"})),expect(e.headers["content-type"]).toBe("application/vnd.custom+json"),expect(e.headers["Content-Type"]).toBeUndefined()}),it("should pass raw BodyInit values through unchanged without JSON headers",()=>{const e=new FormData;e.set("field","value");const o=(0,t.buildBodyAndHeaders)({method:"POST",body:e,defaultHeaders:{}});expect(o.body).toBe(e),expect(o.headers["Content-Type"]).toBeUndefined()}),it("should return an undefined body when the body itself is undefined",()=>{const e=(0,t.buildBodyAndHeaders)({method:"POST",body:void 0,defaultHeaders:{"X-Default":"d"}});expect(e.body).toBeUndefined(),expect(e.headers).toEqual({"X-Default":"d"})})}),describe("buildSignal",()=>{afterEach(()=>{vi.useRealTimers()}),it("GIVEN no timeout WHEN building a signal THEN it returns no signal",()=>{const e=(0,t.buildSignal)({timeoutMs:void 0});expect(e.signal).toBeUndefined(),expect(e.cleanup).toEqual(expect.any(Function))}),it("GIVEN a zero timeout WHEN building a signal THEN it returns no signal",()=>{const e=(0,t.buildSignal)({timeoutMs:0});expect(e.signal).toBeUndefined(),expect(e.cleanup).toEqual(expect.any(Function))}),it("GIVEN a negative timeout WHEN building a signal THEN it returns no signal",()=>{const e=(0,t.buildSignal)({timeoutMs:-1});expect(e.signal).toBeUndefined(),expect(e.cleanup).toEqual(expect.any(Function))}),it("GIVEN a positive timeout WHEN building a signal THEN it returns an AbortSignal",()=>{const{signal:e,cleanup:o}=(0,t.buildSignal)({timeoutMs:1e3});expect(e).toBeInstanceOf(AbortSignal),expect(e?.aborted).toBe(!1),o()}),it("GIVEN a positive timeout WHEN the delay elapses THEN it aborts the signal",()=>{vi.useFakeTimers();const{signal:e,cleanup:o}=(0,t.buildSignal)({timeoutMs:1e3});expect(e?.aborted).toBe(!1),vi.advanceTimersByTime(999),expect(e?.aborted).toBe(!1),vi.advanceTimersByTime(1),expect(e?.aborted).toBe(!0),o()}),it("GIVEN a timeout signal WHEN cleanup runs before the delay THEN it does not abort",()=>{vi.useFakeTimers();const{signal:e,cleanup:o}=(0,t.buildSignal)({timeoutMs:100});o(),vi.advanceTimersByTime(100),expect(e?.aborted).toBe(!1)})}),describe("parseBody",()=>{it("should return undefined for 'void' response type without consuming the body",async()=>{const e=new Response("ignored"),o=await(0,t.parseBody)(e,"void");expect(o).toBeUndefined(),expect(e.bodyUsed).toBe(!1)}),it("should return text for 'text' response type",async()=>{const e=new Response("plain-body");await expect((0,t.parseBody)(e,"text")).resolves.toBe("plain-body")}),it("should return a Blob for 'blob' response type",async()=>{const e=new Response("blob-body"),o=await(0,t.parseBody)(e,"blob");expect(o).toBeInstanceOf(Blob)}),it("should return an ArrayBuffer for 'arrayBuffer' response type",async()=>{const e=new Response(new Uint8Array([1,2,3])),o=await(0,t.parseBody)(e,"arrayBuffer");expect(o).toBeInstanceOf(ArrayBuffer),expect(new Uint8Array(o)).toEqual(new Uint8Array([1,2,3]))}),it("should parse JSON for 'json' response type",async()=>{const e=new Response(JSON.stringify({hello:"world"}));await expect((0,t.parseBody)(e,"json")).resolves.toEqual({hello:"world"})}),it("should resolve to undefined for an empty JSON body",async()=>{const e=new Response("");await expect((0,t.parseBody)(e,"json")).resolves.toBeUndefined()}),it("should throw a DmkNetworkClientError for malformed JSON",async()=>{const e=new Response("not-json",{status:200,statusText:"OK"}),o=await(0,t.parseBody)(e,"json").catch(s=>s);expect(o).toBeInstanceOf(n.DmkNetworkClientError);const a=o;expect(a.status).toBe(200),expect(a.statusText).toBe("OK"),expect(a.responseBody).toBe("not-json"),expect(a.cause).toBeInstanceOf(SyntaxError)})}),describe("safeReadText",()=>{it("should return the text body",async()=>{await expect((0,t.safeReadText)(new Response("hello"))).resolves.toBe("hello")}),it("should return undefined when reading the body throws",async()=>{const e={text:()=>Promise.reject(new Error("read failed"))};await expect((0,t.safeReadText)(e)).resolves.toBeUndefined()})}),describe("wrapFetchError",()=>{it("should wrap a generic Error with its message",()=>{const e=new TypeError("network down"),o=(0,t.wrapFetchError)({cause:e,timeoutMs:void 0});expect(o).toBeInstanceOf(n.DmkNetworkClientError),expect(o.message).toBe("network down"),expect(o.cause).toBe(e),expect(o.isTimeout).toBe(!1)}),it("should fall back to a generic message for non-Error causes",()=>{const e=(0,t.wrapFetchError)({cause:"something",timeoutMs:void 0});expect(e.message).toBe("Network request failed"),expect(e.cause).toBe("something")}),it("should flag TimeoutError as a timeout",()=>{const e=new Error("timed out");e.name="TimeoutError";const o=(0,t.wrapFetchError)({cause:e,timeoutMs:10});expect(o.isTimeout).toBe(!0),expect(o.message).toBe("Request timed out")}),it("should flag AbortError as a timeout when a timeout is configured and no external abort",()=>{const e=new Error("aborted");e.name="AbortError";const o=(0,t.wrapFetchError)({cause:e,timeoutMs:1e3});expect(o.isTimeout).toBe(!0)}),it("should treat AbortError with no timeout and no external abort as a plain abort",()=>{const e=new Error("aborted");e.name="AbortError";const o=(0,t.wrapFetchError)({cause:e,timeoutMs:void 0});expect(o.isTimeout).toBe(!1),expect(o.message).toBe("Request aborted")})})});
2
2
  //# sourceMappingURL=DmkNetworkClientHelpers.test.js.map