wrapito 13.5.0-beta6 → 13.5.0-beta8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.cjs ADDED
@@ -0,0 +1,18 @@
1
+ "use strict";var ye=Object.create;var O=Object.defineProperty;var Re=Object.getOwnPropertyDescriptor;var be=Object.getOwnPropertyNames;var we=Object.getPrototypeOf,ke=Object.prototype.hasOwnProperty;var xe=(e,t)=>{for(var n in t)O(e,n,{get:t[n],enumerable:!0})},N=(e,t,n,s)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of be(t))!ke.call(e,r)&&r!==n&&O(e,r,{get:()=>t[r],enumerable:!(s=Re(t,r))||s.enumerable});return e};var E=(e,t,n)=>(n=e!=null?ye(we(e)):{},N(t||!e||!e.__esModule?O(n,"default",{value:e,enumerable:!0}):n,e)),Ie=e=>N(O({},"__esModule",{value:!0}),e);var at={};xe(at,{assertions:()=>ot,configure:()=>Me,getConfig:()=>m,matchers:()=>ot,wrap:()=>He});module.exports=Ie(at);var K=E(require("react"),1);var h=E(require("chalk"),1);var W=E(require("object-hash"),1);var j=require("@testing-library/react"),A={defaultHost:"",extend:{},mount:j.render,changeRoute:e=>window.history.replaceState(null,"",e)};function Me(e){A={...A,...e}}var m=()=>({...A});var $=e=>t=>{let{method:n="GET",path:s,host:r=m().defaultHost,requestBody:o=void 0,catchParams:a}=t,l=r+s,i=!m().handleQueryParams||a,u=(0,W.default)({url:i?l:l.split("?")[0],method:n.toUpperCase(),requestBody:o}),c;return o!==void 0&&"_bodyInit"in e&&e._bodyInit!==void 0&&(c=JSON.parse(e._bodyInit)),(0,W.default)({url:i?e.url:e.url.split("?")[0],method:e.method,requestBody:c})===u};var _=async()=>{let e={json:()=>Promise.resolve(),status:200,ok:!0,headers:new Headers({"Content-Type":"application/json"})};return Promise.resolve(e)},Te=e=>{let{streamBody:t,status:n=200,headers:s={}}=e;return{body:t,status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"text/event-stream",...s})}},Oe=e=>{let{responseBody:t,status:n=200,headers:s={}}=e;return{json:()=>Promise.resolve(t),status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"application/json",...s})}},L=async e=>{let t=e.streamBody?Te(e):Oe(e);return e.delay?new Promise(n=>setTimeout(()=>n(t),e.delay)):Promise.resolve(t)},Ee=e=>console.warn(`
2
+ ${h.default.white.bold.bgRed("wrapito")} ${h.default.redBright.bold("cannot find any mock matching:")}
3
+ ${h.default.greenBright(`URL: ${e.url}`)}
4
+ ${h.default.greenBright(`METHOD: ${e.method?.toLowerCase()}`)}
5
+ ${h.default.greenBright(`REQUEST BODY: ${e._bodyInit}`)}
6
+ `),F=async(e,t,n)=>{let s=e.find($(t));if(!s)return n&&Ee(t),_();let{multipleResponses:r}=s;if(!r)return L(s);let o=r.find(a=>!a.hasBeenReturned);if(!o){n&&qe(s);return}return o.hasBeenReturned=!0,L(o)},G=(e=[],t=!1)=>{global.fetch.mockImplementation((s,r)=>{if(typeof s=="string"){let a=new Request(s,r);return F(e,a,t)}return F(e,s,t)})},qe=e=>{let t=`\u{1F32F} Wrapito: Missing response in the multipleResponses array for path ${e.path} and method ${e.method}.`,n=h.default.greenBright(t);console.warn(n)},V=e=>{let t=global.fetch.getMockImplementation();global.fetch.mockImplementation((n,s)=>{let r=typeof n=="string"?n:n.url,o=s?.method||(typeof n!="string"?n.method:"GET");return console.warn(`
7
+ ${h.default.white.bold.bgYellow(" \u{1F32F} wrapito ")}
8
+ ${h.default.yellowBright.bold("\u26A0\uFE0F pending request detected after test finished:")}
9
+ ${h.default.greenBright(`URL: ${r}`)}
10
+ ${h.default.greenBright(`METHOD: ${o?.toLowerCase()}`)}
11
+ ${e?` ${h.default.greenBright(`TEST: ${e}`)}
12
+ `:""}`),t?t(n,s):_()})};var S,y=e=>{S={...S,...e}},R=()=>({...S});function q(e,t){if(!e)throw new Error(t)}function k(e,t){return typeof t===e}function Ce(e){return e instanceof Promise}function v(e,t,n){Object.defineProperty(e,t,n)}function x(e,t,n){Object.defineProperty(e,t,{value:n})}var M=Symbol.for("tinyspy:spy"),Pe=new Set,Ue=e=>{e.called=!1,e.callCount=0,e.calls=[],e.results=[],e.next=[]},Ae=e=>(v(e,M,{value:{reset:()=>Ue(e[M])}}),e[M]),C=e=>e[M]||Ae(e);function We(e){q(k("function",e)||k("undefined",e),"cannot spy on a non-function value");let t=function(...s){let r=C(t);r.called=!0,r.callCount++,r.calls.push(s);let o=r.next.shift();if(o){r.results.push(o);let[u,c]=o;if(u==="ok")return c;throw c}let a,l="ok";if(r.impl)try{new.target?a=Reflect.construct(r.impl,s,new.target):a=r.impl.apply(this,s),l="ok"}catch(u){throw a=u,l="error",r.results.push([l,u]),u}let i=[l,a];if(Ce(a)){let u=a.then(c=>i[1]=c).catch(c=>{throw i[0]="error",i[1]=c,c});Object.assign(u,a),a=u}return r.results.push(i),a};x(t,"_isMockFunction",!0),x(t,"length",e?e.length:0),x(t,"name",e&&e.name||"spy");let n=C(t);return n.reset(),n.impl=e,t}var J=(e,t)=>Object.getOwnPropertyDescriptor(e,t),Q=(e,t)=>{t!=null&&typeof t=="function"&&t.prototype!=null&&Object.setPrototypeOf(e.prototype,t.prototype)};function Y(e,t,n){q(!k("undefined",e),"spyOn could not find an object to spy upon"),q(k("object",e)||k("function",e),"cannot spyOn on a primitive value");let[s,r]=(()=>{if(!k("object",t))return[t,"value"];if("getter"in t&&"setter"in t)throw new Error("cannot spy on both getter and setter");if("getter"in t)return[t.getter,"get"];if("setter"in t)return[t.setter,"set"];throw new Error("specify getter or setter to spy on")})(),o=J(e,s),a=Object.getPrototypeOf(e),l=a&&J(a,s),i=o||l;q(i||s in e,`${String(s)} does not exist`);let u=!1;r==="value"&&i&&!i.value&&i.get&&(r="get",u=!0,n=i.get());let c;i?c=i[r]:r!=="value"?c=()=>e[s]:c=e[s],n||(n=c);let d=We(n);r==="value"&&Q(d,c);let p=f=>{let{value:I,...U}=i||{configurable:!0,writable:!0};r!=="value"&&delete U.writable,U[r]=f,v(e,s,U)},b=()=>i?v(e,s,i):p(c),g=d[M];return x(g,"restore",b),x(g,"getOriginal",()=>u?c():c),x(g,"willCall",f=>(g.impl=f,d)),p(u?()=>(Q(d,n),d):d),Pe.add(d),d}var ve=new Set,Be=0;function De(e){let t=e,n,s=[],r=[],o=C(e),a={get calls(){return o.calls},get instances(){return s},get invocationCallOrder(){return r},get results(){return o.results.map(([p,b])=>({type:p==="error"?"throw":"return",value:b}))},get lastCall(){return o.calls[o.calls.length-1]}},l=[],i=!1;function u(...p){return s.push(this),r.push(++Be),(i?n:l.shift()||n||o.getOriginal()||(()=>{})).apply(this,p)}let c=t.name;t.getMockName=()=>c||"vi.fn()",t.mockName=p=>(c=p,t),t.mockClear=()=>(o.reset(),s=[],r=[],t),t.mockReset=()=>(t.mockClear(),n=()=>{},l=[],t),t.mockRestore=()=>(t.mockReset(),o.restore(),n=void 0,t),t.getMockImplementation=()=>n,t.mockImplementation=p=>(n=p,o.willCall(u),t),t.mockImplementationOnce=p=>(l.push(p),t);function d(p,b){let g=n;n=p,o.willCall(u),i=!0;let f=()=>{n=g,i=!1},I=b();return I instanceof Promise?I.then(()=>(f(),t)):(f(),t)}return t.withImplementation=d,t.mockReturnThis=()=>t.mockImplementation(function(){return this}),t.mockReturnValue=p=>t.mockImplementation(()=>p),t.mockReturnValueOnce=p=>t.mockImplementationOnce(()=>p),t.mockResolvedValue=p=>t.mockImplementation(()=>Promise.resolve(p)),t.mockResolvedValueOnce=p=>t.mockImplementationOnce(()=>Promise.resolve(p)),t.mockRejectedValue=p=>t.mockImplementation(()=>Promise.reject(p)),t.mockRejectedValueOnce=p=>t.mockImplementationOnce(()=>Promise.reject(p)),Object.defineProperty(t,"mock",{get:()=>a}),o.willCall(u),ve.add(t),t}var z=e=>De(Y({spy:e||(()=>{})},"spy"));beforeEach(()=>{global.fetch=z()});afterEach(()=>{let{warnOnPendingRequests:e}=m();if(e){let t=expect.getState?.()?.currentTestName;V(t)}});var He=e=>(y({Component:e,responses:[],props:{},path:"",hasPath:!1,interactionConfig:void 0,debug:process.env.npm_config_debugRequests==="true"}),w()),w=()=>{let e=Le();return{withProps:Ge,withNetwork:Je,withStreamingNetwork:_e,withInteraction:Ve,atPath:Qe,debugRequests:Ye,mount:Xe,...e}},Ne=e=>{let t=R(),n=[...t.responses,...e];y({...t,responses:n})},je=(e,t)=>(t({addResponses:Ne},e),w()),$e=(e,t)=>{let{extend:n}=m(),s=n[t];return{...e,[t]:(...r)=>je(r,s)}},Le=()=>{let{extend:e}=m();return Object.keys(e).reduce($e,{})},Fe=(e,t,n)=>{let s=new TextEncoder;return new ReadableStream({async start(r){for(let o of e){let a=typeof o=="string"?o:o.text,l=typeof o=="object"&&o.delay!==void 0?o.delay:t;l>0&&await new Promise(i=>setTimeout(i,l)),r.enqueue(s.encode(a))}n||r.close()}})},_e=e=>{let{path:t,host:n,method:s="GET",chunks:r,delayBetweenChunks:o=0,keepOpen:a=!1}=e,l=R();return y({...l,responses:[...l.responses,{path:t,host:n,method:s,streamBody:Fe(r,o,a)}]}),w()},Ge=e=>{let t=R();return y({...t,props:e}),w()},Ve=e=>{let t=R();return y({...t,interactionConfig:e}),w()},Je=(e=[])=>{let t=R(),n=Array.isArray(e)?e:[e];return y({...t,responses:[...t.responses,...n]}),w()},Qe=(e,t)=>{let n=R();return y({...n,historyState:t,path:e,hasPath:!0}),w()},Ye=()=>{let e=R();return y({...e,debug:!0}),w()},ze=e=>{let{defaultResponses:t=[]}=m();return[...e,...t]},Ke=(e,t,n,s,r)=>{if(e){if(s){console.warn("wrapito WARNING: history is DEPRECATED. Pass a changeRoute function to the config instead."),console.warn("Read about changeRoute in: https://github.com/mercadona/wrapito#changeRoute"),s.push(t,n);return}r(t)}},Xe=()=>{let{portal:e,portals:t,changeRoute:n,history:s,mount:r,interaction:o}=m(),{Component:a,props:l,responses:i,path:u,hasPath:c,debug:d,historyState:p,interactionConfig:b}=R(),g=a;e&&X(e),t&&Ze(t),Ke(c,u,p,s,n),G(ze(i),d);let f=r(K.createElement(g,{...l}));if(o){let I=o.setup?o.setup(o.userLib,b):o.userLib;return{...f,user:I}}return{...f,user:void 0}},X=e=>{if(document.getElementById(e))return;let t=document.createElement("div");t.setAttribute("id",e),t.setAttribute("data-testid",e),document.body.appendChild(t)},Ze=e=>{e.forEach(t=>{X(t)})};var B=require("jest-diff"),D=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t?.host}${e} ain't got called`:`\u{1F32F} Wrapito: ${e} ain't got called`;return{pass:!1,message:()=>n}},Z=(e,t,n)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: ${e} is called ${n} times, you expected ${t} times`}),ee=(e,t)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch method does not match, expected ${e} received ${t??"none"}`}),te=(e,t)=>{let n=t.map(s=>(0,B.diff)(e,s)).join(`
13
+
14
+ `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch body does not match.
15
+ ${n}`}},ne=(e,t)=>{let n=t.find(s=>e!==s);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Host request does not match, expected ${e} received ${n}`}},se=(e,t)=>{let n=t.map(s=>(0,B.diff)(e,s)).join(`
16
+
17
+ `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch headers do not match.
18
+ ${n}`}},re=()=>({pass:!1,message:()=>"\u{1F32F} Wrapito: Unable to find body."}),H=()=>({pass:!0,message:()=>"Test passing"}),oe=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t.host}${e} is called`:`\u{1F32F} Wrapito: ${e} is called`;return{pass:!0,message:()=>n}};var ae=E(require("deep-equal"),1);var ie=()=>{let e=m().defaultHost;return e?.includes("http")?e:"https://default.com"},et=(e="",t,n)=>t.includes(n)?t:e+t,T=e=>e instanceof Request,tt=e=>T(e)?e.url:e,P=(e,t={method:"GET"})=>fetch.mock.calls.filter(([s])=>{let r=tt(s),o=ie(),a=new URL(r,o),l=et(t?.host,e,o),i=t?.host||o,u=new URL(l,i),c=a.pathname===u.pathname,d=a.search===u.search,p=a.host===u.host;return u.search?c&&d:t?.host?c&&p:c}),ce=e=>e.flat(1).filter(T).map(t=>t.method),pe=e=>e.flat(1).filter(T).map(t=>t._bodyInit?JSON.parse(t._bodyInit):{}),ue=e=>e.flat(1).filter(T).map(t=>new URL(t.url,ie()).hostname),le=e=>e.flat(1).filter(T).map(t=>{let n={};return t.headers.forEach((s,r)=>{n[r]=s}),n}),de=(e,t)=>e&&!t.includes(e),me=(e,t)=>t.map(s=>(0,ae.default)(e,s)).every(s=>s===!1),he=(e,t)=>t.every(s=>s!==e),ge=(e,t)=>t.every(s=>Object.entries(e).some(([r,o])=>s[r.toLowerCase()]!==o)),fe=e=>e.length===0;var nt=(e,t)=>{let n=P(e);if(fe(n))return D(e);let s=ce(n),r=t?.method;if(de(r,s))return ee(r,s);let o=pe(n),a=t?.body;if(!a)return re();if(me(a,o))return te(a,o);let l=ue(n),i=t?.host;if(i&&he(i,l))return ne(i,l);let u=le(n),c=t?.headers;return c&&ge(c,u)?se(c,u):H()},st=(e,t={method:"GET"})=>P(e,t).length?oe(e,t):D(e,t),rt=(e,t,n={method:"GET"})=>{let s=P(e,n);return s.length!==t?Z(e,t,s.length):H()},ot={toHaveBeenFetched:st,toHaveBeenFetchedWith:nt,toHaveBeenFetchedTimes:rt};0&&(module.exports={assertions,configure,getConfig,matchers,wrap});
package/dist/index.d.cts CHANGED
@@ -29,8 +29,21 @@ interface WrapResponse extends Partial<Response> {
29
29
  catchParams?: boolean;
30
30
  delay?: number;
31
31
  hasBeenReturned?: boolean;
32
+ streamBody?: ReadableStream<Uint8Array<ArrayBuffer>>;
32
33
  }
33
34
  type NetworkResponses = WrapResponse | WrapResponse[];
35
+ type StreamChunk = string | {
36
+ text: string;
37
+ delay?: number;
38
+ };
39
+ interface StreamingNetworkConfig {
40
+ path: string;
41
+ host?: string;
42
+ method?: HttpMethod;
43
+ chunks: StreamChunk[];
44
+ delayBetweenChunks?: number;
45
+ keepOpen?: boolean;
46
+ }
34
47
  type DefaultUserLib = unknown;
35
48
  type DefaultUserInstance = unknown;
36
49
  type DefaultUserSetupOptions = unknown;
@@ -41,6 +54,7 @@ interface InteractionDescriptor<UserLib = DefaultUserLib, UserInstance = Default
41
54
  }
42
55
  interface Wrap<UserInteraction extends InteractionDescriptor = InteractionDescriptor> {
43
56
  withNetwork: (responses?: NetworkResponses) => Wrap<UserInteraction>;
57
+ withStreamingNetwork: (config: StreamingNetworkConfig) => Wrap<UserInteraction>;
44
58
  atPath: (path: string, historyState?: object) => Wrap<UserInteraction>;
45
59
  withProps: (props: object) => Wrap<UserInteraction>;
46
60
  withInteraction: (config: UserInteraction['UserSetupOptions']) => Wrap<UserInteraction>;
@@ -62,9 +76,9 @@ interface WrapOptions<SetupOptions = DefaultUserSetupOptions> {
62
76
  interface WrapExtensionAPI {
63
77
  addResponses: (responses: Array<WrapResponse>) => unknown;
64
78
  }
65
- type Extension<Args extends unknown[] = unknown[], UserInteraction extends InteractionDescriptor = InteractionDescriptor> = (extensionAPI: WrapExtensionAPI, args: Args) => Wrap<UserInteraction> | void;
79
+ type Extension<UserInteraction extends InteractionDescriptor = InteractionDescriptor> = <T>(extensionAPI: WrapExtensionAPI, args: T) => Wrap<UserInteraction>;
66
80
  type Extensions<UserInteraction extends InteractionDescriptor = InteractionDescriptor> = {
67
- [key: string]: Extension<any, UserInteraction>;
81
+ [key: string]: Extension<UserInteraction>;
68
82
  };
69
83
  type Component = React.ReactElement<any, any>;
70
84
  type RenderResult = RenderResult$1;
@@ -110,4 +124,4 @@ declare const matchers: {
110
124
  };
111
125
  };
112
126
 
113
- export { type BrowserHistory, type Component, type Config, type DefaultUserInstance, type DefaultUserLib, type DefaultUserSetupOptions, type Extension, type Extensions, type HttpMethod, type InteractionDescriptor, type InteractionOptions, type Mount, type NetworkResponses, type RenderResult, type RequestOptions, type WrapResponse as Response, type Wrap, type WrapExtensionAPI, type WrapOptions, type WrapRequest, type WrapResponse, matchers as assertions, configure, getConfig, matchers, wrap };
127
+ export { type BrowserHistory, type Component, type Config, type DefaultUserInstance, type DefaultUserLib, type DefaultUserSetupOptions, type Extension, type Extensions, type HttpMethod, type InteractionDescriptor, type InteractionOptions, type Mount, type NetworkResponses, type RenderResult, type RequestOptions, type WrapResponse as Response, type StreamChunk, type StreamingNetworkConfig, type Wrap, type WrapExtensionAPI, type WrapOptions, type WrapRequest, type WrapResponse, matchers as assertions, configure, getConfig, matchers, wrap };
package/dist/index.d.ts CHANGED
@@ -29,8 +29,21 @@ interface WrapResponse extends Partial<Response> {
29
29
  catchParams?: boolean;
30
30
  delay?: number;
31
31
  hasBeenReturned?: boolean;
32
+ streamBody?: ReadableStream<Uint8Array<ArrayBuffer>>;
32
33
  }
33
34
  type NetworkResponses = WrapResponse | WrapResponse[];
35
+ type StreamChunk = string | {
36
+ text: string;
37
+ delay?: number;
38
+ };
39
+ interface StreamingNetworkConfig {
40
+ path: string;
41
+ host?: string;
42
+ method?: HttpMethod;
43
+ chunks: StreamChunk[];
44
+ delayBetweenChunks?: number;
45
+ keepOpen?: boolean;
46
+ }
34
47
  type DefaultUserLib = unknown;
35
48
  type DefaultUserInstance = unknown;
36
49
  type DefaultUserSetupOptions = unknown;
@@ -41,6 +54,7 @@ interface InteractionDescriptor<UserLib = DefaultUserLib, UserInstance = Default
41
54
  }
42
55
  interface Wrap<UserInteraction extends InteractionDescriptor = InteractionDescriptor> {
43
56
  withNetwork: (responses?: NetworkResponses) => Wrap<UserInteraction>;
57
+ withStreamingNetwork: (config: StreamingNetworkConfig) => Wrap<UserInteraction>;
44
58
  atPath: (path: string, historyState?: object) => Wrap<UserInteraction>;
45
59
  withProps: (props: object) => Wrap<UserInteraction>;
46
60
  withInteraction: (config: UserInteraction['UserSetupOptions']) => Wrap<UserInteraction>;
@@ -62,9 +76,9 @@ interface WrapOptions<SetupOptions = DefaultUserSetupOptions> {
62
76
  interface WrapExtensionAPI {
63
77
  addResponses: (responses: Array<WrapResponse>) => unknown;
64
78
  }
65
- type Extension<Args extends unknown[] = unknown[], UserInteraction extends InteractionDescriptor = InteractionDescriptor> = (extensionAPI: WrapExtensionAPI, args: Args) => Wrap<UserInteraction> | void;
79
+ type Extension<UserInteraction extends InteractionDescriptor = InteractionDescriptor> = <T>(extensionAPI: WrapExtensionAPI, args: T) => Wrap<UserInteraction>;
66
80
  type Extensions<UserInteraction extends InteractionDescriptor = InteractionDescriptor> = {
67
- [key: string]: Extension<any, UserInteraction>;
81
+ [key: string]: Extension<UserInteraction>;
68
82
  };
69
83
  type Component = React.ReactElement<any, any>;
70
84
  type RenderResult = RenderResult$1;
@@ -110,4 +124,4 @@ declare const matchers: {
110
124
  };
111
125
  };
112
126
 
113
- export { type BrowserHistory, type Component, type Config, type DefaultUserInstance, type DefaultUserLib, type DefaultUserSetupOptions, type Extension, type Extensions, type HttpMethod, type InteractionDescriptor, type InteractionOptions, type Mount, type NetworkResponses, type RenderResult, type RequestOptions, type WrapResponse as Response, type Wrap, type WrapExtensionAPI, type WrapOptions, type WrapRequest, type WrapResponse, matchers as assertions, configure, getConfig, matchers, wrap };
127
+ export { type BrowserHistory, type Component, type Config, type DefaultUserInstance, type DefaultUserLib, type DefaultUserSetupOptions, type Extension, type Extensions, type HttpMethod, type InteractionDescriptor, type InteractionOptions, type Mount, type NetworkResponses, type RenderResult, type RequestOptions, type WrapResponse as Response, type StreamChunk, type StreamingNetworkConfig, type Wrap, type WrapExtensionAPI, type WrapOptions, type WrapRequest, type WrapResponse, matchers as assertions, configure, getConfig, matchers, wrap };
package/dist/index.mjs CHANGED
@@ -1,18 +1,18 @@
1
- import*as V from"react";import g from"chalk";import D from"object-hash";import{render as de}from"@testing-library/react";var U={defaultHost:"",extend:{},mount:de,changeRoute:e=>window.history.replaceState(null,"",e)};function _e(e){U={...U,...e}}var m=()=>({...U});var H=e=>t=>{let{method:n="GET",path:s,host:r=m().defaultHost,requestBody:o=void 0,catchParams:a}=t,l=r+s,c=!m().handleQueryParams||a,u=D({url:c?l:l.split("?")[0],method:n.toUpperCase(),requestBody:o}),i;return"_bodyInit"in e&&e._bodyInit!==void 0&&(i=JSON.parse(e._bodyInit)),D({url:c?e.url:e.url.split("?")[0],method:e.method,requestBody:i})===u};var j=async()=>{let e={json:()=>Promise.resolve(),status:200,ok:!0,headers:new Headers({"Content-Type":"application/json"})};return Promise.resolve(e)},B=async e=>{let{responseBody:t,status:n=200,headers:s={},delay:r}=e,o={json:()=>Promise.resolve(t),status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"application/json",...s})};return r?new Promise(a=>setTimeout(()=>a(o),r)):Promise.resolve(o)},me=e=>console.warn(`
2
- ${g.white.bold.bgRed("wrapito")} ${g.redBright.bold("cannot find any mock matching:")}
3
- ${g.greenBright(`URL: ${e.url}`)}
4
- ${g.greenBright(`METHOD: ${e.method?.toLowerCase()}`)}
5
- ${g.greenBright(`REQUEST BODY: ${e._bodyInit}`)}
6
- `),S=async(e,t,n)=>{let s=e.find(H(t));if(!s)return n&&me(t),j();let{multipleResponses:r}=s;if(!r)return B(s);let o=r.find(a=>!a.hasBeenReturned);if(!o){n&&ge(s);return}return o.hasBeenReturned=!0,B(o)},N=(e=[],t=!1)=>{global.window.fetch.mockImplementation((s,r)=>{if(typeof s=="string"){let a=new Request(s,r);return S(e,a,t)}return S(e,s,t)})},ge=e=>{let t=`\u{1F32F} Wrapito: Missing response in the multipleResponses array for path ${e.path} and method ${e.method}.`,n=g.greenBright(t);console.warn(n)},$=e=>{let t=global.window.fetch.getMockImplementation();global.window.fetch.mockImplementation((n,s)=>{let r=typeof n=="string"?n:n.url,o=s?.method||(typeof n!="string"?n.method:"GET");return console.warn(`
7
- ${g.white.bold.bgYellow(" \u{1F32F} wrapito ")}
8
- ${g.yellowBright.bold("\u26A0\uFE0F pending request detected after test finished:")}
9
- ${g.greenBright(`URL: ${r}`)}
10
- ${g.greenBright(`METHOD: ${o?.toLowerCase()}`)}
11
- ${e?` ${g.greenBright(`TEST: ${e}`)}
12
- `:""}`),t?t(n,s):j()})};var W,R=e=>{W={...W,...e}},w=()=>({...W});function E(e,t){if(!e)throw new Error(t)}function I(e,t){return typeof t===e}function he(e){return e instanceof Promise}function A(e,t,n){Object.defineProperty(e,t,n)}function x(e,t,n){Object.defineProperty(e,t,{value:n})}var M=Symbol.for("tinyspy:spy"),fe=new Set,ye=e=>{e.called=!1,e.callCount=0,e.calls=[],e.results=[],e.next=[]},Re=e=>(A(e,M,{value:{reset:()=>ye(e[M])}}),e[M]),T=e=>e[M]||Re(e);function we(e){E(I("function",e)||I("undefined",e),"cannot spy on a non-function value");let t=function(...s){let r=T(t);r.called=!0,r.callCount++,r.calls.push(s);let o=r.next.shift();if(o){r.results.push(o);let[u,i]=o;if(u==="ok")return i;throw i}let a,l="ok";if(r.impl)try{new.target?a=Reflect.construct(r.impl,s,new.target):a=r.impl.apply(this,s),l="ok"}catch(u){throw a=u,l="error",r.results.push([l,u]),u}let c=[l,a];if(he(a)){let u=a.then(i=>c[1]=i).catch(i=>{throw c[0]="error",c[1]=i,i});Object.assign(u,a),a=u}return r.results.push(c),a};x(t,"_isMockFunction",!0),x(t,"length",e?e.length:0),x(t,"name",e&&e.name||"spy");let n=T(t);return n.reset(),n.impl=e,t}var L=(e,t)=>Object.getOwnPropertyDescriptor(e,t),F=(e,t)=>{t!=null&&typeof t=="function"&&t.prototype!=null&&Object.setPrototypeOf(e.prototype,t.prototype)};function _(e,t,n){E(!I("undefined",e),"spyOn could not find an object to spy upon"),E(I("object",e)||I("function",e),"cannot spyOn on a primitive value");let[s,r]=(()=>{if(!I("object",t))return[t,"value"];if("getter"in t&&"setter"in t)throw new Error("cannot spy on both getter and setter");if("getter"in t)return[t.getter,"get"];if("setter"in t)return[t.setter,"set"];throw new Error("specify getter or setter to spy on")})(),o=L(e,s),a=Object.getPrototypeOf(e),l=a&&L(a,s),c=o||l;E(c||s in e,`${String(s)} does not exist`);let u=!1;r==="value"&&c&&!c.value&&c.get&&(r="get",u=!0,n=c.get());let i;c?i=c[r]:r!=="value"?i=()=>e[s]:i=e[s],n||(n=i);let d=we(n);r==="value"&&F(d,i);let p=f=>{let{value:k,...P}=c||{configurable:!0,writable:!0};r!=="value"&&delete P.writable,P[r]=f,A(e,s,P)},y=()=>c?A(e,s,c):p(i),h=d[M];return x(h,"restore",y),x(h,"getOriginal",()=>u?i():i),x(h,"willCall",f=>(h.impl=f,d)),p(u?()=>(F(d,n),d):d),fe.add(d),d}var Ie=new Set,xe=0;function ke(e){let t=e,n,s=[],r=[],o=T(e),a={get calls(){return o.calls},get instances(){return s},get invocationCallOrder(){return r},get results(){return o.results.map(([p,y])=>({type:p==="error"?"throw":"return",value:y}))},get lastCall(){return o.calls[o.calls.length-1]}},l=[],c=!1;function u(...p){return s.push(this),r.push(++xe),(c?n:l.shift()||n||o.getOriginal()||(()=>{})).apply(this,p)}let i=t.name;t.getMockName=()=>i||"vi.fn()",t.mockName=p=>(i=p,t),t.mockClear=()=>(o.reset(),s=[],r=[],t),t.mockReset=()=>(t.mockClear(),n=()=>{},l=[],t),t.mockRestore=()=>(t.mockReset(),o.restore(),n=void 0,t),t.getMockImplementation=()=>n,t.mockImplementation=p=>(n=p,o.willCall(u),t),t.mockImplementationOnce=p=>(l.push(p),t);function d(p,y){let h=n;n=p,o.willCall(u),c=!0;let f=()=>{n=h,c=!1},k=y();return k instanceof Promise?k.then(()=>(f(),t)):(f(),t)}return t.withImplementation=d,t.mockReturnThis=()=>t.mockImplementation(function(){return this}),t.mockReturnValue=p=>t.mockImplementation(()=>p),t.mockReturnValueOnce=p=>t.mockImplementationOnce(()=>p),t.mockResolvedValue=p=>t.mockImplementation(()=>Promise.resolve(p)),t.mockResolvedValueOnce=p=>t.mockImplementationOnce(()=>Promise.resolve(p)),t.mockRejectedValue=p=>t.mockImplementation(()=>Promise.reject(p)),t.mockRejectedValueOnce=p=>t.mockImplementationOnce(()=>Promise.reject(p)),Object.defineProperty(t,"mock",{get:()=>a}),o.willCall(u),Ie.add(t),t}var G=e=>ke(_({spy:e||(()=>{})},"spy"));beforeEach(()=>{global.fetch=G()});afterEach(()=>{let{warnOnPendingRequests:e}=m();if(e){let t=expect.getState?.()?.currentTestName;$(t)}});var rt=e=>(R({Component:e,responses:[],props:{},path:"",hasPath:!1,interactionConfig:void 0,debug:process.env.npm_config_debugRequests==="true"}),b()),b=()=>{let e=Te();return{withProps:qe,withNetwork:Ue,withInteraction:Pe,atPath:We,debugRequests:Ae,mount:De,...e}},Me=e=>{let t=w(),n=[...t.responses,...e];R({...t,responses:n})},Oe=(e,t)=>(t({addResponses:Me},e),b()),Ee=(e,t)=>{let{extend:n}=m(),s=n[t];return{...e,[t]:(...r)=>Oe(r,s)}},Te=()=>{let{extend:e}=m();return Object.keys(e).reduce(Ee,{})},qe=e=>{let t=w();return R({...t,props:e}),b()},Pe=e=>{let t=w();return R({...t,interactionConfig:e}),b()},Ue=(e=[])=>{let t=w(),n=Array.isArray(e)?e:[e];return R({...t,responses:[...t.responses,...n]}),b()},We=(e,t)=>{let n=w();return R({...n,historyState:t,path:e,hasPath:!0}),b()},Ae=()=>{let e=w();return R({...e,debug:!0}),b()},ve=e=>{let{defaultResponses:t=[]}=m();return[...e,...t]},Ce=(e,t,n,s,r)=>{if(e){if(s){console.warn("wrapito WARNING: history is DEPRECATED. Pass a changeRoute function to the config instead."),console.warn("Read about changeRoute in: https://github.com/mercadona/wrapito#changeRoute"),s.push(t,n);return}r(t)}},De=()=>{let{portal:e,portals:t,changeRoute:n,history:s,mount:r,interaction:o}=m(),{Component:a,props:l,responses:c,path:u,hasPath:i,debug:d,historyState:p,interactionConfig:y}=w(),h=a;e&&Q(e),t&&He(t),Ce(i,u,p,s,n),N(ve(c),d);let f=r(V.createElement(h,{...l}));if(o){let k=o.setup?o.setup(o.userLib,y):o.userLib;return{...f,user:k}}return{...f,user:void 0}},Q=e=>{if(document.getElementById(e))return;let t=document.createElement("div");t.setAttribute("id",e),t.setAttribute("data-testid",e),document.body.appendChild(t)},He=e=>{e.forEach(t=>{Q(t)})};import{diff as J}from"jest-diff";var v=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t?.host}${e} ain't got called`:`\u{1F32F} Wrapito: ${e} ain't got called`;return{pass:!1,message:()=>n}},Y=(e,t,n)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: ${e} is called ${n} times, you expected ${t} times`}),z=(e,t)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch method does not match, expected ${e} received ${t??"none"}`}),K=(e,t)=>{let n=t.map(s=>J(e,s)).join(`
1
+ import*as V from"react";import h from"chalk";import v from"object-hash";import{render as de}from"@testing-library/react";var P={defaultHost:"",extend:{},mount:de,changeRoute:e=>window.history.replaceState(null,"",e)};function Qe(e){P={...P,...e}}var m=()=>({...P});var B=e=>t=>{let{method:n="GET",path:s,host:r=m().defaultHost,requestBody:o=void 0,catchParams:a}=t,l=r+s,i=!m().handleQueryParams||a,u=v({url:i?l:l.split("?")[0],method:n.toUpperCase(),requestBody:o}),c;return o!==void 0&&"_bodyInit"in e&&e._bodyInit!==void 0&&(c=JSON.parse(e._bodyInit)),v({url:i?e.url:e.url.split("?")[0],method:e.method,requestBody:c})===u};var N=async()=>{let e={json:()=>Promise.resolve(),status:200,ok:!0,headers:new Headers({"Content-Type":"application/json"})};return Promise.resolve(e)},me=e=>{let{streamBody:t,status:n=200,headers:s={}}=e;return{body:t,status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"text/event-stream",...s})}},he=e=>{let{responseBody:t,status:n=200,headers:s={}}=e;return{json:()=>Promise.resolve(t),status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"application/json",...s})}},D=async e=>{let t=e.streamBody?me(e):he(e);return e.delay?new Promise(n=>setTimeout(()=>n(t),e.delay)):Promise.resolve(t)},ge=e=>console.warn(`
2
+ ${h.white.bold.bgRed("wrapito")} ${h.redBright.bold("cannot find any mock matching:")}
3
+ ${h.greenBright(`URL: ${e.url}`)}
4
+ ${h.greenBright(`METHOD: ${e.method?.toLowerCase()}`)}
5
+ ${h.greenBright(`REQUEST BODY: ${e._bodyInit}`)}
6
+ `),H=async(e,t,n)=>{let s=e.find(B(t));if(!s)return n&&ge(t),N();let{multipleResponses:r}=s;if(!r)return D(s);let o=r.find(a=>!a.hasBeenReturned);if(!o){n&&fe(s);return}return o.hasBeenReturned=!0,D(o)},j=(e=[],t=!1)=>{global.fetch.mockImplementation((s,r)=>{if(typeof s=="string"){let a=new Request(s,r);return H(e,a,t)}return H(e,s,t)})},fe=e=>{let t=`\u{1F32F} Wrapito: Missing response in the multipleResponses array for path ${e.path} and method ${e.method}.`,n=h.greenBright(t);console.warn(n)},$=e=>{let t=global.fetch.getMockImplementation();global.fetch.mockImplementation((n,s)=>{let r=typeof n=="string"?n:n.url,o=s?.method||(typeof n!="string"?n.method:"GET");return console.warn(`
7
+ ${h.white.bold.bgYellow(" \u{1F32F} wrapito ")}
8
+ ${h.yellowBright.bold("\u26A0\uFE0F pending request detected after test finished:")}
9
+ ${h.greenBright(`URL: ${r}`)}
10
+ ${h.greenBright(`METHOD: ${o?.toLowerCase()}`)}
11
+ ${e?` ${h.greenBright(`TEST: ${e}`)}
12
+ `:""}`),t?t(n,s):N()})};var U,y=e=>{U={...U,...e}},R=()=>({...U});function O(e,t){if(!e)throw new Error(t)}function k(e,t){return typeof t===e}function ye(e){return e instanceof Promise}function A(e,t,n){Object.defineProperty(e,t,n)}function x(e,t,n){Object.defineProperty(e,t,{value:n})}var M=Symbol.for("tinyspy:spy"),Re=new Set,be=e=>{e.called=!1,e.callCount=0,e.calls=[],e.results=[],e.next=[]},we=e=>(A(e,M,{value:{reset:()=>be(e[M])}}),e[M]),E=e=>e[M]||we(e);function ke(e){O(k("function",e)||k("undefined",e),"cannot spy on a non-function value");let t=function(...s){let r=E(t);r.called=!0,r.callCount++,r.calls.push(s);let o=r.next.shift();if(o){r.results.push(o);let[u,c]=o;if(u==="ok")return c;throw c}let a,l="ok";if(r.impl)try{new.target?a=Reflect.construct(r.impl,s,new.target):a=r.impl.apply(this,s),l="ok"}catch(u){throw a=u,l="error",r.results.push([l,u]),u}let i=[l,a];if(ye(a)){let u=a.then(c=>i[1]=c).catch(c=>{throw i[0]="error",i[1]=c,c});Object.assign(u,a),a=u}return r.results.push(i),a};x(t,"_isMockFunction",!0),x(t,"length",e?e.length:0),x(t,"name",e&&e.name||"spy");let n=E(t);return n.reset(),n.impl=e,t}var L=(e,t)=>Object.getOwnPropertyDescriptor(e,t),F=(e,t)=>{t!=null&&typeof t=="function"&&t.prototype!=null&&Object.setPrototypeOf(e.prototype,t.prototype)};function _(e,t,n){O(!k("undefined",e),"spyOn could not find an object to spy upon"),O(k("object",e)||k("function",e),"cannot spyOn on a primitive value");let[s,r]=(()=>{if(!k("object",t))return[t,"value"];if("getter"in t&&"setter"in t)throw new Error("cannot spy on both getter and setter");if("getter"in t)return[t.getter,"get"];if("setter"in t)return[t.setter,"set"];throw new Error("specify getter or setter to spy on")})(),o=L(e,s),a=Object.getPrototypeOf(e),l=a&&L(a,s),i=o||l;O(i||s in e,`${String(s)} does not exist`);let u=!1;r==="value"&&i&&!i.value&&i.get&&(r="get",u=!0,n=i.get());let c;i?c=i[r]:r!=="value"?c=()=>e[s]:c=e[s],n||(n=c);let d=ke(n);r==="value"&&F(d,c);let p=f=>{let{value:I,...C}=i||{configurable:!0,writable:!0};r!=="value"&&delete C.writable,C[r]=f,A(e,s,C)},b=()=>i?A(e,s,i):p(c),g=d[M];return x(g,"restore",b),x(g,"getOriginal",()=>u?c():c),x(g,"willCall",f=>(g.impl=f,d)),p(u?()=>(F(d,n),d):d),Re.add(d),d}var Ie=new Set,Me=0;function Te(e){let t=e,n,s=[],r=[],o=E(e),a={get calls(){return o.calls},get instances(){return s},get invocationCallOrder(){return r},get results(){return o.results.map(([p,b])=>({type:p==="error"?"throw":"return",value:b}))},get lastCall(){return o.calls[o.calls.length-1]}},l=[],i=!1;function u(...p){return s.push(this),r.push(++Me),(i?n:l.shift()||n||o.getOriginal()||(()=>{})).apply(this,p)}let c=t.name;t.getMockName=()=>c||"vi.fn()",t.mockName=p=>(c=p,t),t.mockClear=()=>(o.reset(),s=[],r=[],t),t.mockReset=()=>(t.mockClear(),n=()=>{},l=[],t),t.mockRestore=()=>(t.mockReset(),o.restore(),n=void 0,t),t.getMockImplementation=()=>n,t.mockImplementation=p=>(n=p,o.willCall(u),t),t.mockImplementationOnce=p=>(l.push(p),t);function d(p,b){let g=n;n=p,o.willCall(u),i=!0;let f=()=>{n=g,i=!1},I=b();return I instanceof Promise?I.then(()=>(f(),t)):(f(),t)}return t.withImplementation=d,t.mockReturnThis=()=>t.mockImplementation(function(){return this}),t.mockReturnValue=p=>t.mockImplementation(()=>p),t.mockReturnValueOnce=p=>t.mockImplementationOnce(()=>p),t.mockResolvedValue=p=>t.mockImplementation(()=>Promise.resolve(p)),t.mockResolvedValueOnce=p=>t.mockImplementationOnce(()=>Promise.resolve(p)),t.mockRejectedValue=p=>t.mockImplementation(()=>Promise.reject(p)),t.mockRejectedValueOnce=p=>t.mockImplementationOnce(()=>Promise.reject(p)),Object.defineProperty(t,"mock",{get:()=>a}),o.willCall(u),Ie.add(t),t}var G=e=>Te(_({spy:e||(()=>{})},"spy"));beforeEach(()=>{global.fetch=G()});afterEach(()=>{let{warnOnPendingRequests:e}=m();if(e){let t=expect.getState?.()?.currentTestName;$(t)}});var ct=e=>(y({Component:e,responses:[],props:{},path:"",hasPath:!1,interactionConfig:void 0,debug:process.env.npm_config_debugRequests==="true"}),w()),w=()=>{let e=Ce();return{withProps:Ae,withNetwork:Se,withStreamingNetwork:Ue,withInteraction:We,atPath:ve,debugRequests:Be,mount:Ne,...e}},Oe=e=>{let t=R(),n=[...t.responses,...e];y({...t,responses:n})},Ee=(e,t)=>(t({addResponses:Oe},e),w()),qe=(e,t)=>{let{extend:n}=m(),s=n[t];return{...e,[t]:(...r)=>Ee(r,s)}},Ce=()=>{let{extend:e}=m();return Object.keys(e).reduce(qe,{})},Pe=(e,t,n)=>{let s=new TextEncoder;return new ReadableStream({async start(r){for(let o of e){let a=typeof o=="string"?o:o.text,l=typeof o=="object"&&o.delay!==void 0?o.delay:t;l>0&&await new Promise(i=>setTimeout(i,l)),r.enqueue(s.encode(a))}n||r.close()}})},Ue=e=>{let{path:t,host:n,method:s="GET",chunks:r,delayBetweenChunks:o=0,keepOpen:a=!1}=e,l=R();return y({...l,responses:[...l.responses,{path:t,host:n,method:s,streamBody:Pe(r,o,a)}]}),w()},Ae=e=>{let t=R();return y({...t,props:e}),w()},We=e=>{let t=R();return y({...t,interactionConfig:e}),w()},Se=(e=[])=>{let t=R(),n=Array.isArray(e)?e:[e];return y({...t,responses:[...t.responses,...n]}),w()},ve=(e,t)=>{let n=R();return y({...n,historyState:t,path:e,hasPath:!0}),w()},Be=()=>{let e=R();return y({...e,debug:!0}),w()},De=e=>{let{defaultResponses:t=[]}=m();return[...e,...t]},He=(e,t,n,s,r)=>{if(e){if(s){console.warn("wrapito WARNING: history is DEPRECATED. Pass a changeRoute function to the config instead."),console.warn("Read about changeRoute in: https://github.com/mercadona/wrapito#changeRoute"),s.push(t,n);return}r(t)}},Ne=()=>{let{portal:e,portals:t,changeRoute:n,history:s,mount:r,interaction:o}=m(),{Component:a,props:l,responses:i,path:u,hasPath:c,debug:d,historyState:p,interactionConfig:b}=R(),g=a;e&&J(e),t&&je(t),He(c,u,p,s,n),j(De(i),d);let f=r(V.createElement(g,{...l}));if(o){let I=o.setup?o.setup(o.userLib,b):o.userLib;return{...f,user:I}}return{...f,user:void 0}},J=e=>{if(document.getElementById(e))return;let t=document.createElement("div");t.setAttribute("id",e),t.setAttribute("data-testid",e),document.body.appendChild(t)},je=e=>{e.forEach(t=>{J(t)})};import{diff as Q}from"jest-diff";var W=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t?.host}${e} ain't got called`:`\u{1F32F} Wrapito: ${e} ain't got called`;return{pass:!1,message:()=>n}},Y=(e,t,n)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: ${e} is called ${n} times, you expected ${t} times`}),z=(e,t)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch method does not match, expected ${e} received ${t??"none"}`}),K=(e,t)=>{let n=t.map(s=>Q(e,s)).join(`
13
13
 
14
14
  `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch body does not match.
15
- ${n}`}},X=(e,t)=>{let n=t.find(s=>e!==s);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Host request does not match, expected ${e} received ${n}`}},Z=(e,t)=>{let n=t.map(s=>J(e,s)).join(`
15
+ ${n}`}},X=(e,t)=>{let n=t.find(s=>e!==s);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Host request does not match, expected ${e} received ${n}`}},Z=(e,t)=>{let n=t.map(s=>Q(e,s)).join(`
16
16
 
17
17
  `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch headers do not match.
18
- ${n}`}},ee=()=>({pass:!1,message:()=>"\u{1F32F} Wrapito: Unable to find body."}),C=()=>({pass:!0,message:()=>"Test passing"}),te=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t.host}${e} is called`:`\u{1F32F} Wrapito: ${e} is called`;return{pass:!0,message:()=>n}};import Be from"deep-equal";var ne=()=>{let e=m().defaultHost;return e?.includes("http")?e:"https://default.com"},Se=(e="",t,n)=>t.includes(n)?t:e+t,O=e=>e instanceof Request,je=e=>O(e)?e.url:e,q=(e,t={method:"GET"})=>fetch.mock.calls.filter(([s])=>{let r=je(s),o=ne(),a=new URL(r,o),l=Se(t?.host,e,o),c=t?.host||o,u=new URL(l,c),i=a.pathname===u.pathname,d=a.search===u.search,p=a.host===u.host;return u.search?i&&d:t?.host?i&&p:i}),se=e=>e.flat(1).filter(O).map(t=>t.method),re=e=>e.flat(1).filter(O).map(t=>t._bodyInit?JSON.parse(t._bodyInit):{}),oe=e=>e.flat(1).filter(O).map(t=>new URL(t.url,ne()).hostname),ae=e=>e.flat(1).filter(O).map(t=>{let n={};return t.headers.forEach((s,r)=>{n[r]=s}),n}),ie=(e,t)=>e&&!t.includes(e),ce=(e,t)=>t.map(s=>Be(e,s)).every(s=>s===!1),pe=(e,t)=>t.every(s=>s!==e),ue=(e,t)=>t.every(s=>Object.entries(e).some(([r,o])=>s[r.toLowerCase()]!==o)),le=e=>e.length===0;var Ne=(e,t)=>{let n=q(e);if(le(n))return v(e);let s=se(n),r=t?.method;if(ie(r,s))return z(r,s);let o=re(n),a=t?.body;if(!a)return ee();if(ce(a,o))return K(a,o);let l=oe(n),c=t?.host;if(c&&pe(c,l))return X(c,l);let u=ae(n),i=t?.headers;return i&&ue(i,u)?Z(i,u):C()},$e=(e,t={method:"GET"})=>q(e,t).length?te(e,t):v(e,t),Le=(e,t,n={method:"GET"})=>{let s=q(e,n);return s.length!==t?Y(e,t,s.length):C()},mt={toHaveBeenFetched:$e,toHaveBeenFetchedWith:Ne,toHaveBeenFetchedTimes:Le};export{mt as assertions,_e as configure,m as getConfig,mt as matchers,rt as wrap};
18
+ ${n}`}},ee=()=>({pass:!1,message:()=>"\u{1F32F} Wrapito: Unable to find body."}),S=()=>({pass:!0,message:()=>"Test passing"}),te=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t.host}${e} is called`:`\u{1F32F} Wrapito: ${e} is called`;return{pass:!0,message:()=>n}};import $e from"deep-equal";var ne=()=>{let e=m().defaultHost;return e?.includes("http")?e:"https://default.com"},Le=(e="",t,n)=>t.includes(n)?t:e+t,T=e=>e instanceof Request,Fe=e=>T(e)?e.url:e,q=(e,t={method:"GET"})=>fetch.mock.calls.filter(([s])=>{let r=Fe(s),o=ne(),a=new URL(r,o),l=Le(t?.host,e,o),i=t?.host||o,u=new URL(l,i),c=a.pathname===u.pathname,d=a.search===u.search,p=a.host===u.host;return u.search?c&&d:t?.host?c&&p:c}),se=e=>e.flat(1).filter(T).map(t=>t.method),re=e=>e.flat(1).filter(T).map(t=>t._bodyInit?JSON.parse(t._bodyInit):{}),oe=e=>e.flat(1).filter(T).map(t=>new URL(t.url,ne()).hostname),ae=e=>e.flat(1).filter(T).map(t=>{let n={};return t.headers.forEach((s,r)=>{n[r]=s}),n}),ie=(e,t)=>e&&!t.includes(e),ce=(e,t)=>t.map(s=>$e(e,s)).every(s=>s===!1),pe=(e,t)=>t.every(s=>s!==e),ue=(e,t)=>t.every(s=>Object.entries(e).some(([r,o])=>s[r.toLowerCase()]!==o)),le=e=>e.length===0;var _e=(e,t)=>{let n=q(e);if(le(n))return W(e);let s=se(n),r=t?.method;if(ie(r,s))return z(r,s);let o=re(n),a=t?.body;if(!a)return ee();if(ce(a,o))return K(a,o);let l=oe(n),i=t?.host;if(i&&pe(i,l))return X(i,l);let u=ae(n),c=t?.headers;return c&&ue(c,u)?Z(c,u):S()},Ge=(e,t={method:"GET"})=>q(e,t).length?te(e,t):W(e,t),Ve=(e,t,n={method:"GET"})=>{let s=q(e,n);return s.length!==t?Y(e,t,s.length):S()},yt={toHaveBeenFetched:Ge,toHaveBeenFetchedWith:_e,toHaveBeenFetchedTimes:Ve};export{yt as assertions,Qe as configure,m as getConfig,yt as matchers,ct as wrap};
package/package.json CHANGED
@@ -1,17 +1,21 @@
1
1
  {
2
2
  "name": "wrapito",
3
- "version": "13.5.0-beta6",
4
- "packageManager": "npm@10.8.2",
3
+ "version": "13.5.0-beta8",
5
4
  "description": "🌯 🌯 Wrap you tests so that you can test both behaviour and components with less effort.",
6
5
  "type": "module",
7
6
  "exports": {
8
7
  ".": {
9
- "require": "./dist/index.js",
10
- "import": "./dist/index.mjs",
11
- "types": "./dist/index.d.ts"
8
+ "import": {
9
+ "types": "./dist/index.d.ts",
10
+ "default": "./dist/index.mjs"
11
+ },
12
+ "require": {
13
+ "types": "./dist/index.d.cts",
14
+ "default": "./dist/index.cjs"
15
+ }
12
16
  }
13
17
  },
14
- "main": "./dist/index.js",
18
+ "main": "./dist/index.cjs",
15
19
  "module": "./dist/index.mjs",
16
20
  "types": "./dist/index.d.ts",
17
21
  "files": [
@@ -24,10 +28,13 @@
24
28
  "test": "vitest",
25
29
  "test:coverage": "vitest --coverage",
26
30
  "test:publish": "npm publish --dry-run",
31
+ "lint": "eslint .",
27
32
  "build:types": "tsc --build",
28
33
  "publish-package": "npm run build && npm publish",
29
34
  "publish-package:beta": "npm run build && npm publish --tag beta",
30
- "typecheck": "tsc --noEmit dist/index.d.ts"
35
+ "typecheck": "tsc --noEmit dist/index.d.ts",
36
+ "check:exports": "attw --pack .",
37
+ "prepublishOnly": "npm run build && npm run check:exports"
31
38
  },
32
39
  "repository": {
33
40
  "type": "git",
@@ -53,39 +60,41 @@
53
60
  },
54
61
  "homepage": "https://github.com/mercadona/wrapito#readme",
55
62
  "dependencies": {
56
- "chalk": "^5.6.2",
63
+ "chalk": "^4.1.2",
57
64
  "deep-equal": "^2.2.3",
58
65
  "jest-diff": "^29.7.0",
59
66
  "object-hash": "^3.0.0",
60
67
  "whatwg-fetch": "^3.6.20"
61
68
  },
62
69
  "devDependencies": {
63
- "@testing-library/jest-dom": "^6.6.4",
64
- "@testing-library/react": "^16.3.0",
70
+ "@arethetypeswrong/cli": "0.18.3",
71
+ "@eslint/js": "^9.39.4",
72
+ "@reduxjs/toolkit": "^2.12.0",
73
+ "@testing-library/jest-dom": "^6.9.1",
74
+ "@testing-library/react": "^16.3.2",
65
75
  "@types/deep-equal": "^1.0.4",
66
- "@types/node": "^20.10.5",
76
+ "@types/node": "^24.13.1",
67
77
  "@types/object-hash": "^3.0.6",
68
- "@types/react": "^19.1.8",
78
+ "@types/react": "^19.2.16",
69
79
  "@types/react-router-dom": "^5.3.3",
70
- "@vitest/coverage-istanbul": "^1.1.0",
71
- "eslint": "^8.56.0",
80
+ "@vitest/coverage-istanbul": "^4.1.5",
81
+ "eslint": "^9.39.4",
72
82
  "eslint-config-prettier": "^10.1.8",
73
- "eslint-config-react-app": "^7.0.1",
74
- "eslint-plugin-prettier": "^5.1.1",
75
- "eslint-plugin-react": "^7.33.2",
76
- "history": "^4.10.1",
77
- "jsdom": "^26.1.0",
78
- "prettier": "^3.1.1",
79
- "react": "^19.1.0",
80
- "react-dom": "^19.2.0",
81
- "react-redux": "^9.2.0",
82
- "react-router-dom": "^5.3.4",
83
- "redux": "^5.0.1",
84
- "redux-thunk": "^3.1.0",
83
+ "eslint-plugin-prettier": "^5.5.6",
84
+ "eslint-plugin-react": "^7.37.5",
85
+ "globals": "^17.6.0",
86
+ "history": "^5.3.0",
87
+ "jsdom": "^29.1.1",
88
+ "prettier": "^3.8.3",
89
+ "react": "^19.2.7",
90
+ "react-dom": "^19.2.7",
91
+ "react-redux": "^9.3.0",
92
+ "react-router-dom": "^7.17.0",
85
93
  "tinyspy": "^2.2.1",
86
94
  "tsup": "^8.5.1",
87
95
  "typescript": "^5.9.3",
88
- "vitest": "^1.1.0"
96
+ "typescript-eslint": "^8.61.0",
97
+ "vitest": "^4.1.5"
89
98
  },
90
99
  "peerDependencies": {
91
100
  "@testing-library/jest-dom": ">=5.16.4",
package/dist/index.js DELETED
@@ -1,18 +0,0 @@
1
- "use strict";var ye=Object.create;var E=Object.defineProperty;var Re=Object.getOwnPropertyDescriptor;var we=Object.getOwnPropertyNames;var be=Object.getPrototypeOf,Ie=Object.prototype.hasOwnProperty;var xe=(e,t)=>{for(var n in t)E(e,n,{get:t[n],enumerable:!0})},j=(e,t,n,s)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of we(t))!Ie.call(e,r)&&r!==n&&E(e,r,{get:()=>t[r],enumerable:!(s=Re(t,r))||s.enumerable});return e};var T=(e,t,n)=>(n=e!=null?ye(be(e)):{},j(t||!e||!e.__esModule?E(n,"default",{value:e,enumerable:!0}):n,e)),ke=e=>j(E({},"__esModule",{value:!0}),e);var nt={};xe(nt,{assertions:()=>tt,configure:()=>Me,getConfig:()=>m,matchers:()=>tt,wrap:()=>He});module.exports=ke(nt);var K=T(require("react"),1);var g=T(require("chalk"),1);var v=T(require("object-hash"),1);var N=require("@testing-library/react"),A={defaultHost:"",extend:{},mount:N.render,changeRoute:e=>window.history.replaceState(null,"",e)};function Me(e){A={...A,...e}}var m=()=>({...A});var $=e=>t=>{let{method:n="GET",path:s,host:r=m().defaultHost,requestBody:o=void 0,catchParams:a}=t,l=r+s,c=!m().handleQueryParams||a,u=(0,v.default)({url:c?l:l.split("?")[0],method:n.toUpperCase(),requestBody:o}),i;return"_bodyInit"in e&&e._bodyInit!==void 0&&(i=JSON.parse(e._bodyInit)),(0,v.default)({url:c?e.url:e.url.split("?")[0],method:e.method,requestBody:i})===u};var _=async()=>{let e={json:()=>Promise.resolve(),status:200,ok:!0,headers:new Headers({"Content-Type":"application/json"})};return Promise.resolve(e)},L=async e=>{let{responseBody:t,status:n=200,headers:s={},delay:r}=e,o={json:()=>Promise.resolve(t),status:n,ok:n>=200&&n<=299,headers:new Headers({"Content-Type":"application/json",...s})};return r?new Promise(a=>setTimeout(()=>a(o),r)):Promise.resolve(o)},Oe=e=>console.warn(`
2
- ${g.default.white.bold.bgRed("wrapito")} ${g.default.redBright.bold("cannot find any mock matching:")}
3
- ${g.default.greenBright(`URL: ${e.url}`)}
4
- ${g.default.greenBright(`METHOD: ${e.method?.toLowerCase()}`)}
5
- ${g.default.greenBright(`REQUEST BODY: ${e._bodyInit}`)}
6
- `),F=async(e,t,n)=>{let s=e.find($(t));if(!s)return n&&Oe(t),_();let{multipleResponses:r}=s;if(!r)return L(s);let o=r.find(a=>!a.hasBeenReturned);if(!o){n&&Ee(s);return}return o.hasBeenReturned=!0,L(o)},G=(e=[],t=!1)=>{global.window.fetch.mockImplementation((s,r)=>{if(typeof s=="string"){let a=new Request(s,r);return F(e,a,t)}return F(e,s,t)})},Ee=e=>{let t=`\u{1F32F} Wrapito: Missing response in the multipleResponses array for path ${e.path} and method ${e.method}.`,n=g.default.greenBright(t);console.warn(n)},V=e=>{let t=global.window.fetch.getMockImplementation();global.window.fetch.mockImplementation((n,s)=>{let r=typeof n=="string"?n:n.url,o=s?.method||(typeof n!="string"?n.method:"GET");return console.warn(`
7
- ${g.default.white.bold.bgYellow(" \u{1F32F} wrapito ")}
8
- ${g.default.yellowBright.bold("\u26A0\uFE0F pending request detected after test finished:")}
9
- ${g.default.greenBright(`URL: ${r}`)}
10
- ${g.default.greenBright(`METHOD: ${o?.toLowerCase()}`)}
11
- ${e?` ${g.default.greenBright(`TEST: ${e}`)}
12
- `:""}`),t?t(n,s):_()})};var C,R=e=>{C={...C,...e}},w=()=>({...C});function q(e,t){if(!e)throw new Error(t)}function I(e,t){return typeof t===e}function Te(e){return e instanceof Promise}function D(e,t,n){Object.defineProperty(e,t,n)}function x(e,t,n){Object.defineProperty(e,t,{value:n})}var M=Symbol.for("tinyspy:spy"),qe=new Set,Pe=e=>{e.called=!1,e.callCount=0,e.calls=[],e.results=[],e.next=[]},Ue=e=>(D(e,M,{value:{reset:()=>Pe(e[M])}}),e[M]),P=e=>e[M]||Ue(e);function We(e){q(I("function",e)||I("undefined",e),"cannot spy on a non-function value");let t=function(...s){let r=P(t);r.called=!0,r.callCount++,r.calls.push(s);let o=r.next.shift();if(o){r.results.push(o);let[u,i]=o;if(u==="ok")return i;throw i}let a,l="ok";if(r.impl)try{new.target?a=Reflect.construct(r.impl,s,new.target):a=r.impl.apply(this,s),l="ok"}catch(u){throw a=u,l="error",r.results.push([l,u]),u}let c=[l,a];if(Te(a)){let u=a.then(i=>c[1]=i).catch(i=>{throw c[0]="error",c[1]=i,i});Object.assign(u,a),a=u}return r.results.push(c),a};x(t,"_isMockFunction",!0),x(t,"length",e?e.length:0),x(t,"name",e&&e.name||"spy");let n=P(t);return n.reset(),n.impl=e,t}var Q=(e,t)=>Object.getOwnPropertyDescriptor(e,t),J=(e,t)=>{t!=null&&typeof t=="function"&&t.prototype!=null&&Object.setPrototypeOf(e.prototype,t.prototype)};function Y(e,t,n){q(!I("undefined",e),"spyOn could not find an object to spy upon"),q(I("object",e)||I("function",e),"cannot spyOn on a primitive value");let[s,r]=(()=>{if(!I("object",t))return[t,"value"];if("getter"in t&&"setter"in t)throw new Error("cannot spy on both getter and setter");if("getter"in t)return[t.getter,"get"];if("setter"in t)return[t.setter,"set"];throw new Error("specify getter or setter to spy on")})(),o=Q(e,s),a=Object.getPrototypeOf(e),l=a&&Q(a,s),c=o||l;q(c||s in e,`${String(s)} does not exist`);let u=!1;r==="value"&&c&&!c.value&&c.get&&(r="get",u=!0,n=c.get());let i;c?i=c[r]:r!=="value"?i=()=>e[s]:i=e[s],n||(n=i);let d=We(n);r==="value"&&J(d,i);let p=f=>{let{value:k,...W}=c||{configurable:!0,writable:!0};r!=="value"&&delete W.writable,W[r]=f,D(e,s,W)},y=()=>c?D(e,s,c):p(i),h=d[M];return x(h,"restore",y),x(h,"getOriginal",()=>u?i():i),x(h,"willCall",f=>(h.impl=f,d)),p(u?()=>(J(d,n),d):d),qe.add(d),d}var ve=new Set,Ce=0;function De(e){let t=e,n,s=[],r=[],o=P(e),a={get calls(){return o.calls},get instances(){return s},get invocationCallOrder(){return r},get results(){return o.results.map(([p,y])=>({type:p==="error"?"throw":"return",value:y}))},get lastCall(){return o.calls[o.calls.length-1]}},l=[],c=!1;function u(...p){return s.push(this),r.push(++Ce),(c?n:l.shift()||n||o.getOriginal()||(()=>{})).apply(this,p)}let i=t.name;t.getMockName=()=>i||"vi.fn()",t.mockName=p=>(i=p,t),t.mockClear=()=>(o.reset(),s=[],r=[],t),t.mockReset=()=>(t.mockClear(),n=()=>{},l=[],t),t.mockRestore=()=>(t.mockReset(),o.restore(),n=void 0,t),t.getMockImplementation=()=>n,t.mockImplementation=p=>(n=p,o.willCall(u),t),t.mockImplementationOnce=p=>(l.push(p),t);function d(p,y){let h=n;n=p,o.willCall(u),c=!0;let f=()=>{n=h,c=!1},k=y();return k instanceof Promise?k.then(()=>(f(),t)):(f(),t)}return t.withImplementation=d,t.mockReturnThis=()=>t.mockImplementation(function(){return this}),t.mockReturnValue=p=>t.mockImplementation(()=>p),t.mockReturnValueOnce=p=>t.mockImplementationOnce(()=>p),t.mockResolvedValue=p=>t.mockImplementation(()=>Promise.resolve(p)),t.mockResolvedValueOnce=p=>t.mockImplementationOnce(()=>Promise.resolve(p)),t.mockRejectedValue=p=>t.mockImplementation(()=>Promise.reject(p)),t.mockRejectedValueOnce=p=>t.mockImplementationOnce(()=>Promise.reject(p)),Object.defineProperty(t,"mock",{get:()=>a}),o.willCall(u),ve.add(t),t}var z=e=>De(Y({spy:e||(()=>{})},"spy"));beforeEach(()=>{global.fetch=z()});afterEach(()=>{let{warnOnPendingRequests:e}=m();if(e){let t=expect.getState?.()?.currentTestName;V(t)}});var He=e=>(R({Component:e,responses:[],props:{},path:"",hasPath:!1,interactionConfig:void 0,debug:process.env.npm_config_debugRequests==="true"}),b()),b=()=>{let e=Ne();return{withProps:$e,withNetwork:Fe,withInteraction:Le,atPath:_e,debugRequests:Ge,mount:Je,...e}},Be=e=>{let t=w(),n=[...t.responses,...e];R({...t,responses:n})},Se=(e,t)=>(t({addResponses:Be},e),b()),je=(e,t)=>{let{extend:n}=m(),s=n[t];return{...e,[t]:(...r)=>Se(r,s)}},Ne=()=>{let{extend:e}=m();return Object.keys(e).reduce(je,{})},$e=e=>{let t=w();return R({...t,props:e}),b()},Le=e=>{let t=w();return R({...t,interactionConfig:e}),b()},Fe=(e=[])=>{let t=w(),n=Array.isArray(e)?e:[e];return R({...t,responses:[...t.responses,...n]}),b()},_e=(e,t)=>{let n=w();return R({...n,historyState:t,path:e,hasPath:!0}),b()},Ge=()=>{let e=w();return R({...e,debug:!0}),b()},Ve=e=>{let{defaultResponses:t=[]}=m();return[...e,...t]},Qe=(e,t,n,s,r)=>{if(e){if(s){console.warn("wrapito WARNING: history is DEPRECATED. Pass a changeRoute function to the config instead."),console.warn("Read about changeRoute in: https://github.com/mercadona/wrapito#changeRoute"),s.push(t,n);return}r(t)}},Je=()=>{let{portal:e,portals:t,changeRoute:n,history:s,mount:r,interaction:o}=m(),{Component:a,props:l,responses:c,path:u,hasPath:i,debug:d,historyState:p,interactionConfig:y}=w(),h=a;e&&X(e),t&&Ye(t),Qe(i,u,p,s,n),G(Ve(c),d);let f=r(K.createElement(h,{...l}));if(o){let k=o.setup?o.setup(o.userLib,y):o.userLib;return{...f,user:k}}return{...f,user:void 0}},X=e=>{if(document.getElementById(e))return;let t=document.createElement("div");t.setAttribute("id",e),t.setAttribute("data-testid",e),document.body.appendChild(t)},Ye=e=>{e.forEach(t=>{X(t)})};var H=require("jest-diff"),B=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t?.host}${e} ain't got called`:`\u{1F32F} Wrapito: ${e} ain't got called`;return{pass:!1,message:()=>n}},Z=(e,t,n)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: ${e} is called ${n} times, you expected ${t} times`}),ee=(e,t)=>({pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch method does not match, expected ${e} received ${t??"none"}`}),te=(e,t)=>{let n=t.map(s=>(0,H.diff)(e,s)).join(`
13
-
14
- `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch body does not match.
15
- ${n}`}},ne=(e,t)=>{let n=t.find(s=>e!==s);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Host request does not match, expected ${e} received ${n}`}},se=(e,t)=>{let n=t.map(s=>(0,H.diff)(e,s)).join(`
16
-
17
- `);return{pass:!1,message:()=>`\u{1F32F} Wrapito: Fetch headers do not match.
18
- ${n}`}},re=()=>({pass:!1,message:()=>"\u{1F32F} Wrapito: Unable to find body."}),S=()=>({pass:!0,message:()=>"Test passing"}),oe=(e,t)=>{let n=t?.host?`\u{1F32F} Wrapito: ${t.host}${e} is called`:`\u{1F32F} Wrapito: ${e} is called`;return{pass:!0,message:()=>n}};var ae=T(require("deep-equal"),1);var ie=()=>{let e=m().defaultHost;return e?.includes("http")?e:"https://default.com"},ze=(e="",t,n)=>t.includes(n)?t:e+t,O=e=>e instanceof Request,Ke=e=>O(e)?e.url:e,U=(e,t={method:"GET"})=>fetch.mock.calls.filter(([s])=>{let r=Ke(s),o=ie(),a=new URL(r,o),l=ze(t?.host,e,o),c=t?.host||o,u=new URL(l,c),i=a.pathname===u.pathname,d=a.search===u.search,p=a.host===u.host;return u.search?i&&d:t?.host?i&&p:i}),ce=e=>e.flat(1).filter(O).map(t=>t.method),pe=e=>e.flat(1).filter(O).map(t=>t._bodyInit?JSON.parse(t._bodyInit):{}),ue=e=>e.flat(1).filter(O).map(t=>new URL(t.url,ie()).hostname),le=e=>e.flat(1).filter(O).map(t=>{let n={};return t.headers.forEach((s,r)=>{n[r]=s}),n}),de=(e,t)=>e&&!t.includes(e),me=(e,t)=>t.map(s=>(0,ae.default)(e,s)).every(s=>s===!1),ge=(e,t)=>t.every(s=>s!==e),he=(e,t)=>t.every(s=>Object.entries(e).some(([r,o])=>s[r.toLowerCase()]!==o)),fe=e=>e.length===0;var Xe=(e,t)=>{let n=U(e);if(fe(n))return B(e);let s=ce(n),r=t?.method;if(de(r,s))return ee(r,s);let o=pe(n),a=t?.body;if(!a)return re();if(me(a,o))return te(a,o);let l=ue(n),c=t?.host;if(c&&ge(c,l))return ne(c,l);let u=le(n),i=t?.headers;return i&&he(i,u)?se(i,u):S()},Ze=(e,t={method:"GET"})=>U(e,t).length?oe(e,t):B(e,t),et=(e,t,n={method:"GET"})=>{let s=U(e,n);return s.length!==t?Z(e,t,s.length):S()},tt={toHaveBeenFetched:Ze,toHaveBeenFetchedWith:Xe,toHaveBeenFetchedTimes:et};0&&(module.exports={assertions,configure,getConfig,matchers,wrap});