@data-client/core 0.15.3 → 0.15.4
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/CHANGELOG.md +34 -0
- package/dist/index.js +16 -12
- package/dist/index.umd.min.js +1 -1
- package/dist/mock.js +16 -12
- package/legacy/controller/Controller.js +16 -12
- package/legacy/state/reducer/setResponseReducer.js +2 -2
- package/lib/controller/Controller.d.ts.map +1 -1
- package/lib/controller/Controller.js +16 -12
- package/lib/state/reducer/setResponseReducer.d.ts.map +1 -1
- package/lib/state/reducer/setResponseReducer.js +2 -2
- package/package.json +3 -3
- package/src/controller/Controller.ts +16 -17
- package/src/state/reducer/setResponseReducer.ts +1 -3
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,39 @@
|
|
|
1
1
|
# @data-client/core
|
|
2
2
|
|
|
3
|
+
## 0.15.4
|
|
4
|
+
|
|
5
|
+
### Patch Changes
|
|
6
|
+
|
|
7
|
+
- [#3703](https://github.com/reactive/data-client/pull/3703) [`4fe8779`](https://github.com/reactive/data-client/commit/4fe8779706cb14d9018b3375d07b486a758ccb57) Thanks [@ntucker](https://github.com/ntucker)! - Improve normalize/denormalize performance 10-15%
|
|
8
|
+
- Replace `Object.keys().forEach()` with indexed for loops
|
|
9
|
+
- Replace `reduce()` with spreading to direct object mutation
|
|
10
|
+
- Cache getter results to avoid repeated property lookups
|
|
11
|
+
- Centralize arg extraction with pre-allocated loop
|
|
12
|
+
- Eliminate Map double-get pattern
|
|
13
|
+
|
|
14
|
+
#### Microbenchmark Results
|
|
15
|
+
|
|
16
|
+
| # | Optimization | Before | After | Improvement |
|
|
17
|
+
| --- | ---------------------------- | ----------------- | ----------------- | ---------------- |
|
|
18
|
+
| 1 | **forEach → forLoop** | 7,164 ops/sec | 7,331 ops/sec | **+2.3%** |
|
|
19
|
+
| 2 | **reduce+spread → mutation** | 912 ops/sec | 7,468 ops/sec | **+719% (8.2x)** |
|
|
20
|
+
| 3 | **getter repeated → cached** | 1,652,211 ops/sec | 4,426,994 ops/sec | **+168% (2.7x)** |
|
|
21
|
+
| 4 | **slice+map → indexed** | 33,221 ops/sec | 54,701 ops/sec | **+65% (1.65x)** |
|
|
22
|
+
| 5 | **Map double-get → single** | 23,046 ops/sec | 23,285 ops/sec | **+1%** |
|
|
23
|
+
|
|
24
|
+
#### Impact Summary by Codepath
|
|
25
|
+
|
|
26
|
+
| Codepath | Optimizations Applied | Expected Improvement |
|
|
27
|
+
| ------------------------------------------ | --------------------- | -------------------- |
|
|
28
|
+
| **normalize** (setResponse) | 1, 2, 4 | 10-15% |
|
|
29
|
+
| **denormalize** (getResponse) | 1, 2, 4 | 10-15% |
|
|
30
|
+
| **Controller queries** (get, getQueryMeta) | 5, 6 | 5-10% |
|
|
31
|
+
|
|
32
|
+
- [`09056b0`](https://github.com/reactive/data-client/commit/09056b0adf1375e0aa17df6c1db6f73f721c518f) Thanks [@ntucker](https://github.com/ntucker)! - Simplify endpoint.update() error message
|
|
33
|
+
|
|
34
|
+
- Updated dependencies [[`4fe8779`](https://github.com/reactive/data-client/commit/4fe8779706cb14d9018b3375d07b486a758ccb57)]:
|
|
35
|
+
- @data-client/normalizr@0.15.4
|
|
36
|
+
|
|
3
37
|
## 0.15.3
|
|
4
38
|
|
|
5
39
|
### Patch Changes
|
package/dist/index.js
CHANGED
|
@@ -220,7 +220,7 @@ function setResponseReducer(state, action, controller) {
|
|
|
220
220
|
// no reason to completely fail because of user-code error
|
|
221
221
|
// integrity of this state update is still guaranteed
|
|
222
222
|
} catch (error) {
|
|
223
|
-
console.error(`
|
|
223
|
+
console.error(`Endpoint.update() error: ${action.key}`);
|
|
224
224
|
console.error(error);
|
|
225
225
|
}
|
|
226
226
|
return {
|
|
@@ -884,11 +884,7 @@ class Controller {
|
|
|
884
884
|
*/
|
|
885
885
|
|
|
886
886
|
getResponseMeta(endpoint, ...rest) {
|
|
887
|
-
const state = rest
|
|
888
|
-
// this is typescript generics breaking
|
|
889
|
-
const args = rest.slice(0, rest.length - 1)
|
|
890
|
-
// handle FormData
|
|
891
|
-
.map(ensurePojo);
|
|
887
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
892
888
|
const isActive = args.length !== 1 || args[0] !== null;
|
|
893
889
|
const key = isActive ? endpoint.key(...args) : '';
|
|
894
890
|
const cacheEndpoints = isActive ? state.endpoints[key] : undefined;
|
|
@@ -949,9 +945,7 @@ class Controller {
|
|
|
949
945
|
* @see https://dataclient.io/docs/api/Controller#get
|
|
950
946
|
*/
|
|
951
947
|
get(schema, ...rest) {
|
|
952
|
-
const state = rest
|
|
953
|
-
// this is typescript generics breaking
|
|
954
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
948
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
955
949
|
const {
|
|
956
950
|
data
|
|
957
951
|
} = this.memo.query(schema, args, state);
|
|
@@ -963,9 +957,7 @@ class Controller {
|
|
|
963
957
|
* @see https://dataclient.io/docs/api/Controller#getQueryMeta
|
|
964
958
|
*/
|
|
965
959
|
getQueryMeta(schema, ...rest) {
|
|
966
|
-
const state = rest
|
|
967
|
-
// this is typescript generics breaking
|
|
968
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
960
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
969
961
|
const {
|
|
970
962
|
data,
|
|
971
963
|
paths
|
|
@@ -1065,6 +1057,18 @@ class Snapshot {
|
|
|
1065
1057
|
}
|
|
1066
1058
|
}
|
|
1067
1059
|
|
|
1060
|
+
/** Extract state and args from rest params, applying ensurePojo to args */
|
|
1061
|
+
function extractStateAndArgs(rest) {
|
|
1062
|
+
const l = rest.length;
|
|
1063
|
+
const args = new Array(l - 1);
|
|
1064
|
+
for (let i = 0; i < l - 1; i++) {
|
|
1065
|
+
// handle FormData
|
|
1066
|
+
args[i] = ensurePojo(rest[i]);
|
|
1067
|
+
}
|
|
1068
|
+
// this is typescript generics breaking
|
|
1069
|
+
return [rest[l - 1], args];
|
|
1070
|
+
}
|
|
1071
|
+
|
|
1068
1072
|
class ResetError extends Error {
|
|
1069
1073
|
name = 'ResetError';
|
|
1070
1074
|
constructor() {
|
package/dist/index.umd.min.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t(((e="undefined"!=typeof globalThis?globalThis:e||self).RDC=e.RDC||{},e.RDC.Core={}))}(this,(function(e){"use strict";class t{constructor(){this.localCache=new Map}getEntity(e,t,n,i){const s=t.key;this.localCache.has(s)||this.localCache.set(s,new Map);const r=this.localCache.get(s);return r.get(e)||i(r),r.get(e)}getResults(e,t,n){return{data:n(),paths:[]}}}const n=Symbol("INVALID"),i={};function s(e){return null!==e&&void 0!==e.pk}const r=e=>e[0],o=e=>void 0!==e&&"symbol"!=typeof e,a=(e,t,n,i,s,o)=>{e=r(e);const a=(e=>Array.isArray(e)?e:Object.keys(e).map((t=>e[t])))(t);return a.map((t=>o(e,t,n,i,s)))},c=(e,t,n,i)=>(e=r(e),t.map?t.map((t=>i(e,t))).filter(o):t);function h(){}function l(){return l=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var i in n)({}).hasOwnProperty.call(n,i)&&(e[i]=n[i])}return e},l.apply(null,arguments)}const u=(e,t,n,i,s,r)=>{const o=l({},t);return Object.keys(e).forEach((n=>{const i=e[n],a=r(i,t[n],t,n,s);void 0===a?delete o[n]:o[n]=a})),o},d=(e,t,i,s)=>{if(function(e){return!("function"!=typeof e.hasOwnProperty||!(Object.hasOwnProperty.call(e,"__ownerID")||e._map&&Object.hasOwnProperty.call(e._map,"__ownerID")))}(t))return function(e,t,i,s){let r=!1;const o=Object.keys(e).reduce(((t,n)=>{const i=`${n}`,o=s(e[i],t.get(i));return"symbol"==typeof o&&(r=!0),t.has(i)?t.set(i,o):t}),t);return r?n:o}(e,t,0,s);const r=l({},t);let o=!1;return Object.keys(e).forEach((t=>{const n=s(e[t],r[t]);void 0!==r[t]&&(r[t]=n),"symbol"==typeof n&&(o=!0)})),o?n:r};function p(e,t,n,i){const s={};for(const r of Object.keys(e))s[r]=n(e[r],t,i);return s}const f=(e,t,n,s)=>function(r,o){const a="object"!=typeof o,c=a?e({key:r.key,pk:o}):o;if("symbol"==typeof c)return r.denormalize(c,n,s);if(void 0===c&&a&&""!==o&&"undefined"!==o)return t.getEntity(o,r,i,(e=>{e.set(o,void 0)}));if("object"!=typeof c||null===c)return c;let h=a?o:r.pk(c,void 0,void 0,n);return void 0===h||""===h||"undefined"===h?function(e){const t=new Map;return e(t),t.get("")}((e=>y(r,c,"",e,n,s))):("string"!=typeof h&&(h=`${h}`),t.getEntity(h,r,c,(e=>y(r,c,h,e,n,s))))};function y(e,t,i,s,r,o){const a=e.createIfValid(t);void 0===a?s.set(i,n):(s.set(i,a),s.set(i,e.denormalize(a,r,o)))}const g=(e,t,n)=>{const i=f(e,t,n,r);function r(e,t){if(!e)return t;if(null==t)return t;if("function"==typeof e.denormalize)return s(e)?i(e,t):e.denormalize(t,n,r);if("function"==typeof e)return e(t);if("object"==typeof e){return(Array.isArray(e)?c:d)(e,t,n,r)}return t}return(e,n)=>{const i=Object(n)===n&&Object(e)===e;return t.getResults(n,i,(()=>r(e,n)))}};class m{constructor({entities:e,indexes:t}){this.entities=e,this.indexes=t}tracked(e){const t=this,i=[{path:[""],entity:e}];return[{INVALID:n,getIndex(...e){const n=t.getIndex(...e);return i.push({path:e,entity:n}),t.getIndexEnd(n,e[2])},getEntity(...e){const n=t.getEntity(...e);return i.push({path:e,entity:n}),n},getEntities(e){const n=t.getEntitiesObject(e);return i.push({path:[e],entity:n}),t.getEntities(e)}},i]}}class v extends m{constructor(e){super(e)}getEntitiesObject(e){return this.entities[e]}getEntities(e){const t=this.entities[e];if(void 0!==t)return{keys:()=>Object.keys(t),entries:()=>Object.entries(t)}}getEntity(e,t){var n;return null==(n=this.entities[e])?void 0:n[t]}getIndex(e,t){var n;return null==(n=this.indexes[e])?void 0:n[t]}getIndexEnd(e,t){return null==e?void 0:e[t]}}const E={QueryDelegate:v,getEntities:e=>({key:t,pk:n})=>{var i;return null==(i=e[t])?void 0:i[n]}};class w{constructor(){this.next=new WeakMap,this.nextPath=void 0}get(e,t){let n=this.next.get(e);if(!n)return x;for(;n.nextPath;){var s;const e=null!=(s=t(n.nextPath))?s:i;if(n=n.next.get(e),!n)return x}return[n.value,n.journey]}set(e,t){if(e.length<1)throw new k;let n=this;for(const{path:t,entity:s}of e){let e=n.next.get(s);e||(e=new b,n.next.set(null!=s?s:i,e)),n.nextPath=t,n=e}n.nextPath=void 0,n.value=t,n.journey=e.map((e=>e.path))}}const x=[void 0,void 0];class b{constructor(){this.next=new WeakMap,this.nextPath=void 0,this.value=void 0,this.journey=[]}}class k extends Error{constructor(...e){super(...e),this.message="Keys must include at least one member"}}class I extends v{constructor(e,t){super(e),this.newEntities=new Map,this.newIndexes=new Map,this.entitiesMeta=e.entitiesMeta,this.meta=t,this.checkLoop=function(){const e=new Map;return function(t,n,i){let s=e.get(t);s||(s=new Map,e.set(t,s));let r=s.get(n);return r||(r=new Set,s.set(n,r)),!!r.has(i)||(r.add(i),!1)}}()}getNewEntity(e,t){return this.getNewEntities(e).get(t)}getNewEntities(e){return this.newEntities.has(e)||(this.newEntities.set(e,new Map),this.entities[e]=l({},this.entities[e]),this.entitiesMeta[e]=l({},this.entitiesMeta[e])),this.newEntities.get(e)}getNewIndexes(e){return this.newIndexes.has(e)||(this.newIndexes.set(e,new Map),this.indexes[e]=l({},this.indexes[e])),this.newIndexes.get(e)}mergeEntity(e,t,n){const i=e.key;let s=n,r=this.meta,o=this.getNewEntity(i,t);if(o)s=e.merge(o,n);else if(o=this.getEntity(i,t),o){const a=this.getMeta(i,t);s=e.mergeWithStore(a,r,o,n),r=e.mergeMetaWithStore(a,r,o,n)}this.setEntity(e,t,s,r)}setEntity(e,t,i,s=this.meta){const r=e.key,o=this.getNewEntities(r),a=!o.has(t);o.set(t,i),e.indexes&&function(e,t,i,s,r,o){for(const a of t){i.has(a)||i.set(a,s[a]={});const t=i.get(a);o[e]&&delete t[o[e][a]],o&&o[e]&&o[e][a]!==r[a]&&(t[o[e][a]]=n),a in r&&(t[r[a]]=e)}}(t,e.indexes,this.getNewIndexes(r),this.indexes[r],i,this.entities[r]),this._setEntity(r,t,i),a&&this._setMeta(r,t,s)}invalidate({key:e},t){this.setEntity({key:e},t,n)}_setEntity(e,t,n){this.entities[e][t]=n}_setMeta(e,t,n){this.entitiesMeta[e][t]=n}getMeta(e,t){return this.entitiesMeta[e][t]}}const A=(e,t,n=[],{entities:i,indexes:s,entitiesMeta:r}=L,o={fetchedAt:0,date:Date.now(),expiresAt:1/0})=>{if(null==e)return{result:t,entities:i,indexes:s,entitiesMeta:r};const c=function(e){return["object","function"].includes(typeof e)?"object":typeof e}(e);if(null===t||typeof t!==c&&(void 0===e.key||void 0!==e.pk||"string"!=typeof t))throw new Error(`Unexpected input given to normalize. Expected type to be "${c}", found "${null===t?"null":typeof t}".`);const h={result:"",entities:l({},i),indexes:l({},s),entitiesMeta:l({},r)},d=(e=>{const t=(n,i,s,r,o)=>i&&n?n.normalize&&"function"==typeof n.normalize?"object"!=typeof i?n.pk?`${i}`:i:n.normalize(i,s,r,o,t,e):"object"!=typeof i||"object"!=typeof n?i:(Array.isArray(n)?a:u)(n,i,s,r,o,t):i;return t})(new I(h,o));return h.result=d(e,t,t,void 0,n),h};const L={entities:{},indexes:{},entitiesMeta:{}};class S{constructor(e,t,n){this.dependencies=[],this.cycleCache=new Map,this.cycleIndex=-1,this.localCache=new Map,this._getEntity=e,this._getCache=t,this._resultCache=n}getEntity(e,t,n,i){const s=t.key,{localCacheKey:r,cycleCacheKey:o}=this.getCacheKey(s);if(r.get(e))o.has(e)?this.cycleIndex=o.get(e):this.dependencies.push({path:{key:s,pk:e},entity:n});else{const a=this._getCache(e,t),[c,h]=a.get(n,this._getEntity);if(h)return r.set(e,c.value),this.dependencies.push(...c.dependencies),c.value;{const t=this.dependencies.length;o.set(e,t),this.dependencies.push({path:{key:s,pk:e},entity:n}),i(r),o.delete(e);const c=this.dependencies.slice(-1===this.cycleIndex?t:this.cycleIndex),h={dependencies:c,value:r.get(e)};a.set(c,h),this.cycleIndex===t&&(this.cycleIndex=-1)}}return r.get(e)}getCacheKey(e){this.localCache.has(e)||this.localCache.set(e,new Map),this.cycleCache.has(e)||this.cycleCache.set(e,new Map);return{localCacheKey:this.localCache.get(e),cycleCacheKey:this.cycleCache.get(e)}}getResults(e,t,n){if(!t)return{data:n(),paths:this.paths()};let[i,s]=this._resultCache.get(e,this._getEntity);return void 0===s?(i=n(),s=this.paths(),this.dependencies.unshift({path:{key:"",pk:""},entity:e}),this._resultCache.set(this.dependencies,i)):s.shift(),{data:i,paths:s}}paths(){return this.dependencies.map((e=>e.path))}}function M(e){return void 0!==e&&(!(e&&"object"==typeof e&&!Array.isArray(e))||Object.values(e).every(M))}class C{constructor(e=E){var t;this.endpoints=new w,this.queryKeys=new Map,this.policy=e,this._getCache=(t=new Map,(e,n)=>{var i;const s=n.key,r=null!=(i=n.cacheWith)?i:n;t.has(s)||t.set(s,new Map);const o=t.get(s);o.has(e)||o.set(e,new WeakMap);const a=o.get(e);let c=a.get(r);return c||(c=new w,a.set(r,c)),c})}denormalize(e,t,n,i=[]){if(void 0===e)return{data:t,paths:[]};if(void 0===t)return{data:void 0,paths:[]};const s=this.policy.getEntities(n);return g(s,new S(s,this._getCache,this.endpoints),i)(e,t)}query(e,t,n,i=JSON.stringify(t)){const s=this.buildQueryKey(e,t,n,i);return s?this.denormalize(e,s,n.entities,t):{data:void 0,paths:[]}}buildQueryKey(e,t,n,i=JSON.stringify(t)){if("object"!=typeof e&&"function"!=typeof e.queryKey||!e)return e;this.queryKeys.get(i)||this.queryKeys.set(i,new w);const s=this.queryKeys.get(i),r=new this.policy.QueryDelegate(n);let[o,a]=s.get(e,(c=r,e=>c[["","getEntitiesObject","getEntity","getIndex"][e.length]](...e)));var c;if(!a){const[n,i]=r.tracked(e);o=function(e){return function t(n,i){return function(e){return!!e&&"function"==typeof e.queryKey}(n)?n.queryKey(i,t,e):"object"==typeof n&&n?(Array.isArray(n)?h:p)(n,i,t,e):n}}(n)(e,t),s.set(i,o)}return o}}var O={Invalid:1,InvalidIfStale:2,Valid:3};function _(e,t){const n=Date.now();return{fetchedAt:null!=t?t:n,date:n,expiresAt:n+e}}const R="rdc/fetch",P="rdc/set",j="rdc/setresponse",T="rdc/optimistic",q="rdc/reset",D="rdc/subscribe",N="rdc/unsubscribe",K="rdc/invalidate",z="rdc/invalidateall",Q="rdc/expireall",H="rdc/gc",F=R,V=P,B=j,U=T,Y=q,$=D,W=N,X=K,G=z,J=Q,Z=H;var ee=Object.freeze({__proto__:null,EXPIREALL:Q,EXPIREALL_TYPE:J,FETCH:R,FETCH_TYPE:F,GC:H,GC_TYPE:Z,INVALIDATE:K,INVALIDATEALL:z,INVALIDATEALL_TYPE:G,INVALIDATE_TYPE:X,OPTIMISTIC:T,OPTIMISTIC_TYPE:U,RESET:q,RESET_TYPE:Y,SET:P,SET_RESPONSE:j,SET_RESPONSE_TYPE:B,SET_TYPE:V,SUBSCRIBE:D,SUBSCRIBE_TYPE:$,UNSUBSCRIBE:N,UNSUBSCRIBE_TYPE:W});function te(e,t,n){var i;return{type:T,key:e.key(...t),args:t,endpoint:e,meta:_(null!=(i=e.dataExpiryLength)?i:6e4,n)}}class ne extends Error{}function ie(e,t,n){return"AbortError"===n.name?l({},e,{optimistic:se(e,t)}):l({},e,{meta:l({},e.meta,{[t.key]:{date:t.meta.date,fetchedAt:t.meta.fetchedAt,expiresAt:t.meta.expiresAt,error:n,errorPolicy:null==t.endpoint.errorPolicy?void 0:t.endpoint.errorPolicy(n)}}),optimistic:se(e,t)})}function se(e,t){return e.optimistic.filter((e=>e.key!==t.key||(e.type===T?e.meta.fetchedAt!==t.meta.fetchedAt:e.meta.date>t.meta.date)))}const re={entities:{},endpoints:{},indexes:{},meta:{},entitiesMeta:{},optimistic:[],lastReset:0};var oe=Object.freeze({__proto__:null,INVALID:n,MemoCache:C,initialState:re});function ae(e,{args:t}){return{type:D,key:e.key(...t),args:t,endpoint:e}}function ce(e,{args:t}){return{type:N,key:e.key(...t),args:t,endpoint:e}}const he="undefined"!=typeof FormData?e=>e instanceof FormData?Object.fromEntries(e.entries()):e:e=>e;function le(e,{args:t,fetchedAt:n,response:i,error:s=!1}){var r,o;const a=s?null!=(r=e.errorExpiryLength)?r:1e3:null!=(o=e.dataExpiryLength)?o:6e4;return{type:j,key:e.key(...t),response:i,args:t.map(he),endpoint:e,meta:_(a,n),error:s}}function ue(e,{args:t,fetchedAt:n,value:i}){return{type:P,value:i,args:t.map(he),schema:e,meta:_(6e4,n)}}function de(){return{type:q,date:Date.now()}}function pe(e){return{type:z,testKey:e}}function fe(e,{args:t}){return{type:K,key:e.key(...t)}}function ye(e,{args:t}){let n=0,i=0;const s=new Promise(((e,t)=>{[n,i]=[e,t]})),r={fetchedAt:Date.now(),resolve:n,reject:i,promise:s};return{type:R,key:e.key(...t),args:t,endpoint:e,meta:r}}function ge(e){return{type:Q,testKey:e}}var me=Object.freeze({__proto__:null,createExpireAll:ge,createFetch:ye,createInvalidate:fe,createInvalidateAll:pe,createMeta:_,createOptimistic:te,createReset:de,createSet:ue,createSetResponse:le,createSubscription:ae,createUnsubscription:ce});class ve{init(){}cleanup(){}createCountRef(){return()=>()=>{}}}function Ee(e,t){return e.meta[t]}const we=e=>{throw new Error("Dispatching while constructing your middleware is not allowed. Other middleware would not be applied to this dispatch.")},xe=()=>re;class be{constructor({dispatch:e=we,getState:n=xe,memo:i=new C,gcPolicy:s=new ve}={}){this.fetch=(e,...n)=>{const i=ye(e,{args:n});return this.dispatch(i),e.schema?i.meta.promise.then((i=>function(e,n,i,s=[]){return void 0===e||void 0===n?n:g(E.getEntities(i),new t,s)(e,n).data}(e.schema,i,{},n))):i.meta.promise},this.fetchIfStale=(e,...t)=>{const{data:n,expiresAt:i,expiryStatus:s}=this.getResponseMeta(e,...t,this.getState());return s!==O.Invalid&&Date.now()<=i?n:this.fetch(e,...t)},this.invalidate=(e,...t)=>null!==t[0]?this.dispatch(fe(e,{args:t})):Promise.resolve(),this.invalidateAll=e=>this.dispatch(pe((t=>e.testKey(t)))),this.expireAll=e=>this.dispatch(ge((t=>e.testKey(t)))),this.resetEntireStore=()=>this.dispatch(de()),this.setResponse=(e,...t)=>{const n=t[t.length-1],i=le(e,{args:t.slice(0,t.length-1),response:n});return this.dispatch(i)},this.setError=(e,...t)=>{const n=t[t.length-1],i=le(e,{args:t.slice(0,t.length-1),response:n,error:!0});return this.dispatch(i)},this.resolve=(e,t)=>this.dispatch(le(e,t)),this.subscribe=(e,...t)=>null!==t[0]?this.dispatch(ae(e,{args:t})):Promise.resolve(),this.unsubscribe=(e,...t)=>null!==t[0]?this.dispatch(ce(e,{args:t})):Promise.resolve(),this.snapshot=(e,t)=>new Ie(this,e,t),this._dispatch=e,this.getState=n,this.memo=i,this.gcPolicy=s}set dispatch(e){this._dispatch=e}get dispatch(){return this._dispatch}bindMiddleware({dispatch:e,getState:t}){this._dispatch=e,this.getState=t}set(e,...t){const n=t[t.length-1],i=ue(e,{args:t.slice(0,t.length-1),value:n});return this.dispatch(i)}getError(e,...t){if(null===t[0])return;const n=t[t.length-1],i=t.slice(0,t.length-1),s=e.key(...i),r=Ee(n,s);return void 0===n.endpoints[s]||"soft"!==(null==r?void 0:r.errorPolicy)?null==r?void 0:r.error:void 0}getResponse(e,...t){return this.getResponseMeta(e,...t)}getResponseMeta(e,...t){const n=t[t.length-1],i=t.slice(0,t.length-1).map(he),s=1!==i.length||null!==i[0],r=s?e.key(...i):"",o=s?n.endpoints[r]:void 0,a=e.schema,c=Ee(n,r);let h=null==c?void 0:c.expiresAt;const l=void 0===o&&void 0!==a,u=l?this.memo.buildQueryKey(a,i,n,r):o;if(!s)return{data:u,expiryStatus:O.Valid,expiresAt:1/0,countRef:()=>()=>{}};let d=!1;if(l)d=!M(u);else if(!a||!ke(a))return{data:o,expiryStatus:this.getExpiryStatus(!o,!!e.invalidIfStale,c),expiresAt:h||0,countRef:this.gcPolicy.createCountRef({key:r})};const{data:p,paths:f}=this.memo.denormalize(a,u,n.entities,i);return h||(h=d?1:function(e,t){let n=1/0;for(const{key:s,pk:r}of e){var i;const e=null==(i=t[s])||null==(i=i[r])?void 0:i.expiresAt;e<n&&(n=e)}return n}(f,n.entitiesMeta)),{data:p,expiryStatus:this.getExpiryStatus("symbol"==typeof p,!!e.invalidIfStale||d,c),expiresAt:h,countRef:this.gcPolicy.createCountRef({key:r,paths:f})}}get(e,...t){const n=t[t.length-1],i=t.slice(0,t.length-1).map(he),{data:s}=this.memo.query(e,i,n);return"symbol"==typeof s?void 0:s}getQueryMeta(e,...t){const n=t[t.length-1],i=t.slice(0,t.length-1).map(he),{data:s,paths:r}=this.memo.query(e,i,n);return{data:"symbol"==typeof s?void 0:s,countRef:this.gcPolicy.createCountRef({paths:r})}}getExpiryStatus(e,t,n={}){return n.invalidated||e&&!n.error?O.Invalid:e||t?O.InvalidIfStale:O.Valid}}function ke(e){if(s(e))return!0;if(Array.isArray(e))return 0!==e.length&&ke(e[0]);if(e&&("object"==typeof e||"function"==typeof e)){const t="schema"in e?e.schema:e;return"function"==typeof t?ke(t):Object.values(t).some((e=>ke(e)))}return!1}class Ie{constructor(e,t,n=0){this.state=void 0,this.controller=void 0,this.fetchedAt=void 0,this.abort=Ie.abort,this.state=t,this.controller=e,this.fetchedAt=n}getResponse(e,...t){return this.controller.getResponse(e,...t,this.state)}getResponseMeta(e,...t){return this.controller.getResponseMeta(e,...t,this.state)}getError(e,...t){return this.controller.getError(e,...t,this.state)}get(e,...t){return this.controller.get(e,...t,this.state)}getQueryMeta(e,...t){return this.controller.getQueryMeta(e,...t,this.state)}}Ie.abort=new ne;class Ae extends Error{constructor(){super("Aborted due to RESET"),this.name="ResetError"}}class Le{isOnline(){return void 0===navigator.onLine||navigator.onLine}addOnlineListener(e){addEventListener("online",e)}removeOnlineListener(e){removeEventListener("online",e)}addOfflineListener(e){addEventListener("offline",e)}removeOfflineListener(e){removeEventListener("offline",e)}}class Se{isOnline(){return!0}addOnlineListener(){}removeOnlineListener(){}addOfflineListener(){}removeOfflineListener(){}}let Me;Me="undefined"!=typeof navigator&&"function"==typeof addEventListener?Le:Se;var Ce=Me;let Oe={};class _e{constructor(e,t){this.started=!1,this.actions=[],this.maxBufferLength=100,this.devTools="undefined"!=typeof window&&window.__REDUX_DEVTOOLS_EXTENSION__&&window.__REDUX_DEVTOOLS_EXTENSION__.connect(l({},Oe,e)),null!=e&&e.maxAge&&(this.maxBufferLength=2*e.maxAge),t&&(this.skipLogging=t)}handleAction(e,t){this.started?this.devTools.send(e,t):(this.actions.length>this.maxBufferLength&&(this.actions=this.actions.slice(this.maxBufferLength/2)),this.actions.push([e,t]))}init(e){}cleanup(){}}_e.prototype.middleware=()=>e=>t=>e(t);e.Controller=be,e.DefaultConnectionListener=Ce,e.DevToolsManager=_e,e.ExpiryStatus=O,e.GCPolicy=class{constructor({intervalMS:e=3e5,expiryMultiplier:t=2,expiresAt:n}={}){this.endpointCount=new Map,this.entityCount=new Map,this.endpointsQ=new Set,this.entitiesQ=[],n&&(this.expiresAt=n.bind(this)),this.options={intervalMS:e,expiryMultiplier:t}}init(e){this.controller=e,this.intervalId=setInterval((()=>{this.idleCallback((()=>this.runSweep()),{timeout:1e3})}),this.options.intervalMS)}cleanup(){clearInterval(this.intervalId)}createCountRef({key:e,paths:t=[]}){return()=>{var n;return e&&this.endpointCount.set(e,(null!=(n=this.endpointCount.get(e))?n:0)+1),t.forEach((e=>{var t;const{key:n,pk:i}=e;this.entityCount.has(n)||this.entityCount.set(n,new Map);const s=this.entityCount.get(n);s.set(i,(null!=(t=s.get(i))?t:0)+1)})),()=>{if(e){const t=this.endpointCount.get(e);void 0!==t&&(t<=1?(this.endpointCount.delete(e),this.endpointsQ.add(e)):this.endpointCount.set(e,t-1))}t.forEach((e=>{const{key:t,pk:n}=e;if(!this.entityCount.has(t))return;const i=this.entityCount.get(t),s=i.get(n);void 0!==s&&(s<=1?(i.delete(n),this.entitiesQ.push(e)):i.set(n,s-1))}))}}}expiresAt({fetchedAt:e,expiresAt:t}){return Math.max((t-e)*this.options.expiryMultiplier,12e4)+e}runSweep(){const e=this.controller.getState(),t=[],n=[],i=Date.now(),s=new Set;for(const t of this.endpointsQ){var r;!this.endpointCount.has(t)&&this.expiresAt(null!=(r=e.meta[t])?r:{fetchedAt:0,date:0,expiresAt:0})<i?n.push(t):s.add(t)}this.endpointsQ=s;const o=[];for(const n of this.entitiesQ){var a,c,h;(null==(a=this.entityCount.get(n.key))||!a.has(n.pk))&&this.expiresAt(null!=(c=null==(h=e.entitiesMeta[n.key])?void 0:h[n.pk])?c:{fetchedAt:0,date:0,expiresAt:0})<i?t.push(n):o.push(n)}this.entitiesQ=o,(t.length||n.length)&&this.controller.dispatch({type:H,entities:t,endpoints:n})}idleCallback(e,t){"function"==typeof requestIdleCallback?requestIdleCallback(e,t):e()}},e.ImmortalGCPolicy=ve,e.LogoutManager=class{constructor({handleLogout:e,shouldLogout:t}={}){this.middleware=e=>t=>async n=>{await t(n),n.type===j&&n.error&&this.shouldLogout(n.response)&&this.handleLogout(e)},e&&(this.handleLogout=e),t&&(this.shouldLogout=t)}cleanup(){}shouldLogout(e){return 401===e.status}handleLogout(e){e.resetEntireStore()}},e.NetworkManager=class{constructor({dataExpiryLength:e=6e4,errorExpiryLength:t=1e3}={}){this.fetching=new Map,this.controller=new be,this.middleware=e=>(this.controller=e,t=>n=>{switch(n.type){case R:return this.handleFetch(n),void 0!==n.endpoint.getOptimisticResponse&&n.endpoint.sideEffect?t(n):Promise.resolve();case j:return t(n).then((()=>{if(this.fetching.has(n.key)){var t;const i=null==(t=e.getState().meta[n.key])?void 0:t.error;i?this.handleSet(le(n.endpoint,{args:n.args,response:i,fetchedAt:n.meta.fetchedAt,error:!0})):this.handleSet(n)}}));case q:{const e=Array.from(this.fetching.values());return this.clearAll(),t(n).then((()=>{for(const{reject:t}of e)t(new Ae)}))}default:return t(n)}}),this.dataExpiryLength=e,this.errorExpiryLength=t}init(){delete this.cleanupDate}cleanup(){this.cleanupDate=Date.now()}skipLogging(e){return e.type===R&&this.fetching.has(e.key)}allSettled(){if(this.fetching.size)return Promise.allSettled(this.fetching.values().map((({promise:e})=>e)))}clearAll(){for(const e of this.fetching.keys())this.clear(e)}clear(e){this.fetching.has(e)&&(this.fetching.get(e).promise.catch((()=>{})),this.fetching.delete(e))}getLastReset(){return this.cleanupDate?this.cleanupDate:this.controller.getState().lastReset}handleFetch(e){const{resolve:t,reject:n,fetchedAt:i}=e.meta,s=!e.endpoint.sideEffect,r=()=>{let r=e.endpoint(...e.args);return s||(r=(e=>e.then((e=>(t(e),e))).catch((e=>{throw n(e),e})))(r)),r=r.then((t=>{let n=this.getLastReset();return i>=n&&this.controller.resolve(e.endpoint,{args:e.args,response:t,fetchedAt:i}),t})).catch((t=>{const n=this.getLastReset();throw i>=n&&this.controller.resolve(e.endpoint,{args:e.args,response:t,fetchedAt:i,error:!0}),t})),r};return s?this.throttle(e.key,r,i).then((e=>t(e))).catch((e=>n(e))):r().catch((()=>{}))}handleSet(e){if(this.fetching.has(e.key)){const{reject:t,resolve:n}=this.fetching.get(e.key);e.error?t(e.response):n(e.response),this.clear(e.key)}}throttle(e,t,n){const i=this.getLastReset();let s=this.fetching.get(e);return s&&s.fetchedAt>i||(s=function(e){const t={fetchedAt:e};return t.promise=new Promise(((e,n)=>{t.resolve=e,t.reject=n})),t}(n),this.fetching.set(e,s),this.idleCallback((()=>{t().catch((()=>null))}),{timeout:500})),s.promise}idleCallback(e,t){e()}},e.PollingSubscription=class{constructor(e,t,n){if(this.frequencyHistogram=new Map,this.offlineListener=()=>{this.cleanup(),this.connectionListener.addOnlineListener(this.onlineListener)},this.onlineListener=()=>{this.connectionListener.removeOnlineListener(this.onlineListener);const e=Date.now();this.startId=setTimeout((()=>{this.startId&&(delete this.startId,this.update(),this.run())}),Math.max(0,this.lastFetchTime()-e+this.frequency)),this.connectionListener.addOfflineListener(this.offlineListener)},void 0===e.endpoint.pollFrequency)throw new Error("frequency needed for polling subscription");this.endpoint=e.endpoint,this.frequency=e.endpoint.pollFrequency,this.args=e.args,this.key=e.key,this.frequencyHistogram.set(this.frequency,1),this.controller=t,this.connectionListener=n||new Ce,this.connectionListener.isOnline()?this.onlineListener():this.offlineListener()}add(e){void 0!==e&&(this.frequencyHistogram.has(e)?this.frequencyHistogram.set(e,this.frequencyHistogram.get(e)+1):(this.frequencyHistogram.set(e,1),e<this.frequency&&(this.frequency=e,this.run())))}remove(e){if(void 0===e)return!1;if(this.frequencyHistogram.has(e)&&(this.frequencyHistogram.set(e,this.frequencyHistogram.get(e)-1),this.frequencyHistogram.get(e)<1)){if(this.frequencyHistogram.delete(e),0===this.frequencyHistogram.size)return this.cleanup(),!0;e<=this.frequency&&(this.frequency=Math.min(...this.frequencyHistogram.keys()),this.run())}return!1}cleanup(){this.intervalId&&(clearInterval(this.intervalId),delete this.intervalId),this.lastIntervalId&&(clearInterval(this.lastIntervalId),delete this.lastIntervalId),this.startId&&(clearTimeout(this.startId),delete this.startId),this.connectionListener.removeOnlineListener(this.onlineListener),this.connectionListener.removeOfflineListener(this.offlineListener)}update(){const e=this.endpoint,t=function(...t){return e.call(this,...t)};Object.assign(t,this.endpoint),t.dataExpiryLength=this.frequency/2,t.errorExpiryLength=this.frequency/10,t.errorPolicy=()=>"soft",t.key=()=>this.key,this.controller.fetch(t,...this.args).catch((()=>null))}run(){this.startId||(this.intervalId&&(this.lastIntervalId=this.intervalId),this.intervalId=setInterval((()=>{this.lastIntervalId&&(clearInterval(this.lastIntervalId),delete this.lastIntervalId),this.intervalId&&this.update()}),this.frequency))}lastFetchTime(){var e,t;return null!=(e=null==(t=this.controller.getState().meta[this.key])?void 0:t.date)?e:0}},e.ResetError=Ae,e.SubscriptionManager=class{constructor(e){this.subscriptions={},this.controller=new be,this.middleware=e=>(this.controller=e,e=>t=>{switch(t.type){case D:try{this.handleSubscribe(t)}catch(e){console.error(e)}return Promise.resolve();case N:return this.handleUnsubscribe(t),Promise.resolve();default:return e(t)}}),this.Subscription=e}cleanup(){for(const e in this.subscriptions)this.subscriptions[e].cleanup()}handleSubscribe(e){const t=e.key;if(t in this.subscriptions){const n=e.endpoint.pollFrequency;this.subscriptions[t].add(n)}else this.subscriptions[t]=new this.Subscription(e,this.controller)}handleUnsubscribe(e){const t=e.key;if(t in this.subscriptions){const n=e.endpoint.pollFrequency;this.subscriptions[t].remove(n)&&delete this.subscriptions[t]}}},e.__INTERNAL__=oe,e.actionTypes=ee,e.actions=me,e.applyManager=function(e,t){return e.map(((e,n)=>(e.middleware||(e.middleware=null==e.getMiddleware?void 0:e.getMiddleware()),i=>(0===n&&t.bindMiddleware(i),e.middleware(t)))))},e.createReducer=function(e){return function(t,n){switch(t||(t=re),n.type){case H:return n.entities.forEach((({key:e,pk:n})=>{var i,s;null==(i=t.entities[e])||delete i[n],null==(s=t.entitiesMeta[e])||delete s[n]})),n.endpoints.forEach((e=>{delete t.endpoints[e],delete t.meta[e]})),t;case R:return function(e,t){if(t.endpoint.getOptimisticResponse&&t.endpoint.sideEffect){const n=te(t.endpoint,t.args,t.meta.fetchedAt);return l({},e,{optimistic:[...e.optimistic,n]})}return e}(t,n);case T:case j:return function(e,t,n){if(t.error)return ie(e,t,t.response);try{var i;let s;if(t.type===T){if(!t.endpoint.getOptimisticResponse)return e;try{s=t.endpoint.getOptimisticResponse.call(t.endpoint,n.snapshot(e,t.meta.fetchedAt),...t.args)}catch(t){if(t.constructor===ne)return e;throw t}}else s=t.response;const{result:r,entities:o,indexes:a,entitiesMeta:c}=A(t.endpoint.schema,s,t.args,e,t.meta),h=l({},e.endpoints,{[t.key]:r});try{if(t.endpoint.update){const e=t.endpoint.update(r,...t.args);Object.keys(e).forEach((t=>{h[t]=e[t](h[t])}))}}catch(e){console.error(`The following error occured during Endpoint.update() for ${t.key}`),console.error(e)}return{entities:o,endpoints:h,indexes:a,meta:l({},e.meta,{[t.key]:{date:t.meta.date,fetchedAt:t.meta.fetchedAt,expiresAt:t.meta.expiresAt,prevExpiresAt:null==(i=e.meta[t.key])?void 0:i.expiresAt}}),entitiesMeta:c,optimistic:se(e,t),lastReset:e.lastReset}}catch(n){return"object"==typeof n&&(n.message=`Error processing ${t.key}\n\nFull Schema: ${JSON.stringify(t.endpoint.schema,void 0,2)}\n\nError:\n${n.message}`,"response"in t&&(n.response=t.response),n.status=400),ie(e,t,n)}}(t,n,e);case P:return function(e,t,n){let i;if("function"==typeof t.value){const s=n.get(t.schema,...t.args,e);if(void 0===s)return e;i=t.value(s)}else i=t.value;try{const{entities:n,indexes:s,entitiesMeta:r}=A(t.schema,i,t.args,e,t.meta);return{entities:n,endpoints:e.endpoints,indexes:s,meta:e.meta,entitiesMeta:r,optimistic:e.optimistic,lastReset:e.lastReset}}catch(t){return e}}(t,n,e);case z:case K:return function(e,t){const n=l({},e.endpoints),i=l({},e.meta),s=e=>{delete n[e];const t=l({},i[e],{expiresAt:0,invalidated:!0});delete t.error,i[e]=t};return t.type===K?s(t.key):Object.keys(n).forEach((e=>{t.testKey(e)&&s(e)})),l({},e,{endpoints:n,meta:i})}(t,n);case Q:return function(e,t){const n=l({},e.meta);return Object.keys(n).forEach((e=>{t.testKey(e)&&(n[e]=l({},n[e],{expiresAt:1}))})),l({},e,{meta:n})}(t,n);case q:return l({},re,{lastReset:n.date});default:return t}}},e.initManager=function(e,t,n){return()=>(e.forEach((e=>{null==e.init||e.init(n)})),t.gcPolicy.init(t),()=>{e.forEach((e=>{e.cleanup()})),t.gcPolicy.cleanup()})},e.initialState=re}));
|
|
1
|
+
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t(((e="undefined"!=typeof globalThis?globalThis:e||self).RDC=e.RDC||{},e.RDC.Core={}))}(this,(function(e){"use strict";class t{constructor(){this.localCache=new Map}getEntity(e,t,n,i){const s=t.key;this.localCache.has(s)||this.localCache.set(s,new Map);const r=this.localCache.get(s);return r.get(e)||i(r),r.get(e)}getResults(e,t,n){return{data:n(),paths:[]}}}const n=Symbol("INVALID"),i={};function s(e){return null!==e&&void 0!==e.pk}const r=e=>e[0],o=e=>void 0!==e&&"symbol"!=typeof e,a=(e,t,n,i,s,o)=>{e=r(e);const a=(e=>Array.isArray(e)?e:Object.keys(e).map((t=>e[t])))(t);return a.map((t=>o(e,t,n,i,s)))},c=(e,t,n,i)=>(e=r(e),t.map?t.map((t=>i(e,t))).filter(o):t);function h(){}function l(){return l=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var i in n)({}).hasOwnProperty.call(n,i)&&(e[i]=n[i])}return e},l.apply(null,arguments)}const u=(e,t,n,i,s,r)=>{const o=l({},t),a=Object.keys(e);for(let n=0;n<a.length;n++){const i=a[n],c=r(e[i],t[i],t,i,s);void 0===c?delete o[i]:o[i]=c}return o},d=(e,t,i,s)=>{if(function(e){return!("function"!=typeof e.hasOwnProperty||!(Object.hasOwnProperty.call(e,"__ownerID")||e._map&&Object.hasOwnProperty.call(e._map,"__ownerID")))}(t))return function(e,t,i,s){let r=!1;const o=Object.keys(e).reduce(((t,n)=>{const i=`${n}`,o=s(e[i],t.get(i));return"symbol"==typeof o&&(r=!0),t.has(i)?t.set(i,o):t}),t);return r?n:o}(e,t,0,s);const r=l({},t);let o=!1;const a=Object.keys(e);for(let t=0;t<a.length;t++){const n=a[t],i=s(e[n],r[n]);void 0!==r[n]&&(r[n]=i),"symbol"==typeof i&&(o=!0)}return o?n:r};function p(e,t,n,i){const s={};for(const r of Object.keys(e))s[r]=n(e[r],t,i);return s}const f=(e,t,n,s)=>function(r,o){const a="object"!=typeof o,c=a?e({key:r.key,pk:o}):o;if("symbol"==typeof c)return r.denormalize(c,n,s);if(void 0===c&&a&&""!==o&&"undefined"!==o)return t.getEntity(o,r,i,(e=>{e.set(o,void 0)}));if("object"!=typeof c||null===c)return c;let h=a?o:r.pk(c,void 0,void 0,n);return void 0===h||""===h||"undefined"===h?function(e){const t=new Map;return e(t),t.get("")}((e=>y(r,c,"",e,n,s))):("string"!=typeof h&&(h=`${h}`),t.getEntity(h,r,c,(e=>y(r,c,h,e,n,s))))};function y(e,t,i,s,r,o){const a=e.createIfValid(t);void 0===a?s.set(i,n):(s.set(i,a),s.set(i,e.denormalize(a,r,o)))}const g=(e,t,n)=>{const i=f(e,t,n,r);function r(e,t){if(!e)return t;if(null==t)return t;if("function"==typeof e.denormalize)return s(e)?i(e,t):e.denormalize(t,n,r);if("function"==typeof e)return e(t);if("object"==typeof e){return(Array.isArray(e)?c:d)(e,t,n,r)}return t}return(e,n)=>{const i=Object(n)===n&&Object(e)===e;return t.getResults(n,i,(()=>r(e,n)))}};class m{constructor({entities:e,indexes:t}){this.entities=e,this.indexes=t}tracked(e){const t=this,i=[{path:[""],entity:e}];return[{INVALID:n,getIndex(...e){const n=t.getIndex(...e);return i.push({path:e,entity:n}),t.getIndexEnd(n,e[2])},getEntity(...e){const n=t.getEntity(...e);return i.push({path:e,entity:n}),n},getEntities(e){const n=t.getEntitiesObject(e);return i.push({path:[e],entity:n}),t.getEntities(e)}},i]}}class v extends m{constructor(e){super(e)}getEntitiesObject(e){return this.entities[e]}getEntities(e){const t=this.entities[e];if(void 0!==t)return{keys:()=>Object.keys(t),entries:()=>Object.entries(t)}}getEntity(e,t){var n;return null==(n=this.entities[e])?void 0:n[t]}getIndex(e,t){var n;return null==(n=this.indexes[e])?void 0:n[t]}getIndexEnd(e,t){return null==e?void 0:e[t]}}const E={QueryDelegate:v,getEntities:e=>({key:t,pk:n})=>{var i;return null==(i=e[t])?void 0:i[n]}};class w{constructor(){this.next=new WeakMap,this.nextPath=void 0}get(e,t){let n=this.next.get(e);if(!n)return x;for(;n.nextPath;){var s;const e=null!=(s=t(n.nextPath))?s:i;if(n=n.next.get(e),!n)return x}return[n.value,n.journey]}set(e,t){if(e.length<1)throw new k;let n=this;for(const{path:t,entity:s}of e){let e=n.next.get(s);e||(e=new b,n.next.set(null!=s?s:i,e)),n.nextPath=t,n=e}n.nextPath=void 0,n.value=t,n.journey=e.map((e=>e.path))}}const x=[void 0,void 0];class b{constructor(){this.next=new WeakMap,this.nextPath=void 0,this.value=void 0,this.journey=[]}}class k extends Error{constructor(...e){super(...e),this.message="Keys must include at least one member"}}class I extends v{constructor(e,t){super(e),this.newEntities=new Map,this.newIndexes=new Map,this.entitiesMeta=e.entitiesMeta,this.meta=t,this.checkLoop=function(){const e=new Map;return function(t,n,i){let s=e.get(t);s||(s=new Map,e.set(t,s));let r=s.get(n);return r||(r=new Set,s.set(n,r)),!!r.has(i)||(r.add(i),!1)}}()}getNewEntity(e,t){return this.getNewEntities(e).get(t)}getNewEntities(e){return this.newEntities.has(e)||(this.newEntities.set(e,new Map),this.entities[e]=l({},this.entities[e]),this.entitiesMeta[e]=l({},this.entitiesMeta[e])),this.newEntities.get(e)}getNewIndexes(e){return this.newIndexes.has(e)||(this.newIndexes.set(e,new Map),this.indexes[e]=l({},this.indexes[e])),this.newIndexes.get(e)}mergeEntity(e,t,n){const i=e.key;let s=n,r=this.meta,o=this.getNewEntity(i,t);if(o)s=e.merge(o,n);else if(o=this.getEntity(i,t),o){const a=this.getMeta(i,t);s=e.mergeWithStore(a,r,o,n),r=e.mergeMetaWithStore(a,r,o,n)}this.setEntity(e,t,s,r)}setEntity(e,t,i,s=this.meta){const r=e.key,o=this.getNewEntities(r),a=!o.has(t);o.set(t,i),e.indexes&&function(e,t,i,s,r,o){for(const a of t){i.has(a)||i.set(a,s[a]={});const t=i.get(a);o[e]&&delete t[o[e][a]],o&&o[e]&&o[e][a]!==r[a]&&(t[o[e][a]]=n),a in r&&(t[r[a]]=e)}}(t,e.indexes,this.getNewIndexes(r),this.indexes[r],i,this.entities[r]),this._setEntity(r,t,i),a&&this._setMeta(r,t,s)}invalidate({key:e},t){this.setEntity({key:e},t,n)}_setEntity(e,t,n){this.entities[e][t]=n}_setMeta(e,t,n){this.entitiesMeta[e][t]=n}getMeta(e,t){return this.entitiesMeta[e][t]}}const A=(e,t,n=[],{entities:i,indexes:s,entitiesMeta:r}=L,o={fetchedAt:0,date:Date.now(),expiresAt:1/0})=>{if(null==e)return{result:t,entities:i,indexes:s,entitiesMeta:r};const c=function(e){return["object","function"].includes(typeof e)?"object":typeof e}(e);if(null===t||typeof t!==c&&(void 0===e.key||void 0!==e.pk||"string"!=typeof t))throw new Error(`Unexpected input given to normalize. Expected type to be "${c}", found "${null===t?"null":typeof t}".`);const h={result:"",entities:l({},i),indexes:l({},s),entitiesMeta:l({},r)},d=(e=>{const t=(n,i,s,r,o)=>i&&n?n.normalize&&"function"==typeof n.normalize?"object"!=typeof i?n.pk?`${i}`:i:n.normalize(i,s,r,o,t,e):"object"!=typeof i||"object"!=typeof n?i:(Array.isArray(n)?a:u)(n,i,s,r,o,t):i;return t})(new I(h,o));return h.result=d(e,t,t,void 0,n),h};const L={entities:{},indexes:{},entitiesMeta:{}};class S{constructor(e,t,n){this.dependencies=[],this.cycleCache=new Map,this.cycleIndex=-1,this.localCache=new Map,this._getEntity=e,this._getCache=t,this._resultCache=n}getEntity(e,t,n,i){const s=t.key,{localCacheKey:r,cycleCacheKey:o}=this.getCacheKey(s);if(r.get(e))o.has(e)?this.cycleIndex=o.get(e):this.dependencies.push({path:{key:s,pk:e},entity:n});else{const a=this._getCache(e,t),[c,h]=a.get(n,this._getEntity);if(h)return r.set(e,c.value),this.dependencies.push(...c.dependencies),c.value;{const t=this.dependencies.length;o.set(e,t),this.dependencies.push({path:{key:s,pk:e},entity:n}),i(r),o.delete(e);const c=this.dependencies.slice(-1===this.cycleIndex?t:this.cycleIndex),h={dependencies:c,value:r.get(e)};a.set(c,h),this.cycleIndex===t&&(this.cycleIndex=-1)}}return r.get(e)}getCacheKey(e){this.localCache.has(e)||this.localCache.set(e,new Map),this.cycleCache.has(e)||this.cycleCache.set(e,new Map);return{localCacheKey:this.localCache.get(e),cycleCacheKey:this.cycleCache.get(e)}}getResults(e,t,n){if(!t)return{data:n(),paths:this.paths()};let[i,s]=this._resultCache.get(e,this._getEntity);return void 0===s?(i=n(),s=this.paths(),this.dependencies.unshift({path:{key:"",pk:""},entity:e}),this._resultCache.set(this.dependencies,i)):s.shift(),{data:i,paths:s}}paths(){return this.dependencies.map((e=>e.path))}}function M(e){return void 0!==e&&(!(e&&"object"==typeof e&&!Array.isArray(e))||Object.values(e).every(M))}class C{constructor(e=E){var t;this.endpoints=new w,this.queryKeys=new Map,this.policy=e,this._getCache=(t=new Map,(e,n)=>{var i;const s=n.key,r=null!=(i=n.cacheWith)?i:n;t.has(s)||t.set(s,new Map);const o=t.get(s);o.has(e)||o.set(e,new WeakMap);const a=o.get(e);let c=a.get(r);return c||(c=new w,a.set(r,c)),c})}denormalize(e,t,n,i=[]){if(void 0===e)return{data:t,paths:[]};if(void 0===t)return{data:void 0,paths:[]};const s=this.policy.getEntities(n);return g(s,new S(s,this._getCache,this.endpoints),i)(e,t)}query(e,t,n,i=JSON.stringify(t)){const s=this.buildQueryKey(e,t,n,i);return s?this.denormalize(e,s,n.entities,t):{data:void 0,paths:[]}}buildQueryKey(e,t,n,i=JSON.stringify(t)){if("object"!=typeof e&&"function"!=typeof e.queryKey||!e)return e;let s=this.queryKeys.get(i);s||(s=new w,this.queryKeys.set(i,s));const r=new this.policy.QueryDelegate(n);let[o,a]=s.get(e,(c=r,e=>c[["","getEntitiesObject","getEntity","getIndex"][e.length]](...e)));var c;if(!a){const[n,i]=r.tracked(e);o=function(e){return function t(n,i){return function(e){return!!e&&"function"==typeof e.queryKey}(n)?n.queryKey(i,t,e):"object"==typeof n&&n?(Array.isArray(n)?h:p)(n,i,t,e):n}}(n)(e,t),s.set(i,o)}return o}}var O={Invalid:1,InvalidIfStale:2,Valid:3};function _(e,t){const n=Date.now();return{fetchedAt:null!=t?t:n,date:n,expiresAt:n+e}}const R="rdc/fetch",P="rdc/set",j="rdc/setresponse",T="rdc/optimistic",q="rdc/reset",D="rdc/subscribe",N="rdc/unsubscribe",K="rdc/invalidate",z="rdc/invalidateall",Q="rdc/expireall",H="rdc/gc",F=R,V=P,B=j,U=T,Y=q,$=D,W=N,X=K,G=z,J=Q,Z=H;var ee=Object.freeze({__proto__:null,EXPIREALL:Q,EXPIREALL_TYPE:J,FETCH:R,FETCH_TYPE:F,GC:H,GC_TYPE:Z,INVALIDATE:K,INVALIDATEALL:z,INVALIDATEALL_TYPE:G,INVALIDATE_TYPE:X,OPTIMISTIC:T,OPTIMISTIC_TYPE:U,RESET:q,RESET_TYPE:Y,SET:P,SET_RESPONSE:j,SET_RESPONSE_TYPE:B,SET_TYPE:V,SUBSCRIBE:D,SUBSCRIBE_TYPE:$,UNSUBSCRIBE:N,UNSUBSCRIBE_TYPE:W});function te(e,t,n){var i;return{type:T,key:e.key(...t),args:t,endpoint:e,meta:_(null!=(i=e.dataExpiryLength)?i:6e4,n)}}class ne extends Error{}function ie(e,t,n){return"AbortError"===n.name?l({},e,{optimistic:se(e,t)}):l({},e,{meta:l({},e.meta,{[t.key]:{date:t.meta.date,fetchedAt:t.meta.fetchedAt,expiresAt:t.meta.expiresAt,error:n,errorPolicy:null==t.endpoint.errorPolicy?void 0:t.endpoint.errorPolicy(n)}}),optimistic:se(e,t)})}function se(e,t){return e.optimistic.filter((e=>e.key!==t.key||(e.type===T?e.meta.fetchedAt!==t.meta.fetchedAt:e.meta.date>t.meta.date)))}const re={entities:{},endpoints:{},indexes:{},meta:{},entitiesMeta:{},optimistic:[],lastReset:0};var oe=Object.freeze({__proto__:null,INVALID:n,MemoCache:C,initialState:re});function ae(e,{args:t}){return{type:D,key:e.key(...t),args:t,endpoint:e}}function ce(e,{args:t}){return{type:N,key:e.key(...t),args:t,endpoint:e}}const he="undefined"!=typeof FormData?e=>e instanceof FormData?Object.fromEntries(e.entries()):e:e=>e;function le(e,{args:t,fetchedAt:n,response:i,error:s=!1}){var r,o;const a=s?null!=(r=e.errorExpiryLength)?r:1e3:null!=(o=e.dataExpiryLength)?o:6e4;return{type:j,key:e.key(...t),response:i,args:t.map(he),endpoint:e,meta:_(a,n),error:s}}function ue(e,{args:t,fetchedAt:n,value:i}){return{type:P,value:i,args:t.map(he),schema:e,meta:_(6e4,n)}}function de(){return{type:q,date:Date.now()}}function pe(e){return{type:z,testKey:e}}function fe(e,{args:t}){return{type:K,key:e.key(...t)}}function ye(e,{args:t}){let n=0,i=0;const s=new Promise(((e,t)=>{[n,i]=[e,t]})),r={fetchedAt:Date.now(),resolve:n,reject:i,promise:s};return{type:R,key:e.key(...t),args:t,endpoint:e,meta:r}}function ge(e){return{type:Q,testKey:e}}var me=Object.freeze({__proto__:null,createExpireAll:ge,createFetch:ye,createInvalidate:fe,createInvalidateAll:pe,createMeta:_,createOptimistic:te,createReset:de,createSet:ue,createSetResponse:le,createSubscription:ae,createUnsubscription:ce});class ve{init(){}cleanup(){}createCountRef(){return()=>()=>{}}}function Ee(e,t){return e.meta[t]}const we=e=>{throw new Error("Dispatching while constructing your middleware is not allowed. Other middleware would not be applied to this dispatch.")},xe=()=>re;class be{constructor({dispatch:e=we,getState:n=xe,memo:i=new C,gcPolicy:s=new ve}={}){this.fetch=(e,...n)=>{const i=ye(e,{args:n});return this.dispatch(i),e.schema?i.meta.promise.then((i=>function(e,n,i,s=[]){return void 0===e||void 0===n?n:g(E.getEntities(i),new t,s)(e,n).data}(e.schema,i,{},n))):i.meta.promise},this.fetchIfStale=(e,...t)=>{const{data:n,expiresAt:i,expiryStatus:s}=this.getResponseMeta(e,...t,this.getState());return s!==O.Invalid&&Date.now()<=i?n:this.fetch(e,...t)},this.invalidate=(e,...t)=>null!==t[0]?this.dispatch(fe(e,{args:t})):Promise.resolve(),this.invalidateAll=e=>this.dispatch(pe((t=>e.testKey(t)))),this.expireAll=e=>this.dispatch(ge((t=>e.testKey(t)))),this.resetEntireStore=()=>this.dispatch(de()),this.setResponse=(e,...t)=>{const n=t[t.length-1],i=le(e,{args:t.slice(0,t.length-1),response:n});return this.dispatch(i)},this.setError=(e,...t)=>{const n=t[t.length-1],i=le(e,{args:t.slice(0,t.length-1),response:n,error:!0});return this.dispatch(i)},this.resolve=(e,t)=>this.dispatch(le(e,t)),this.subscribe=(e,...t)=>null!==t[0]?this.dispatch(ae(e,{args:t})):Promise.resolve(),this.unsubscribe=(e,...t)=>null!==t[0]?this.dispatch(ce(e,{args:t})):Promise.resolve(),this.snapshot=(e,t)=>new Ie(this,e,t),this._dispatch=e,this.getState=n,this.memo=i,this.gcPolicy=s}set dispatch(e){this._dispatch=e}get dispatch(){return this._dispatch}bindMiddleware({dispatch:e,getState:t}){this._dispatch=e,this.getState=t}set(e,...t){const n=t[t.length-1],i=ue(e,{args:t.slice(0,t.length-1),value:n});return this.dispatch(i)}getError(e,...t){if(null===t[0])return;const n=t[t.length-1],i=t.slice(0,t.length-1),s=e.key(...i),r=Ee(n,s);return void 0===n.endpoints[s]||"soft"!==(null==r?void 0:r.errorPolicy)?null==r?void 0:r.error:void 0}getResponse(e,...t){return this.getResponseMeta(e,...t)}getResponseMeta(e,...t){const[n,i]=Ae(t),s=1!==i.length||null!==i[0],r=s?e.key(...i):"",o=s?n.endpoints[r]:void 0,a=e.schema,c=Ee(n,r);let h=null==c?void 0:c.expiresAt;const l=void 0===o&&void 0!==a,u=l?this.memo.buildQueryKey(a,i,n,r):o;if(!s)return{data:u,expiryStatus:O.Valid,expiresAt:1/0,countRef:()=>()=>{}};let d=!1;if(l)d=!M(u);else if(!a||!ke(a))return{data:o,expiryStatus:this.getExpiryStatus(!o,!!e.invalidIfStale,c),expiresAt:h||0,countRef:this.gcPolicy.createCountRef({key:r})};const{data:p,paths:f}=this.memo.denormalize(a,u,n.entities,i);return h||(h=d?1:function(e,t){let n=1/0;for(const{key:s,pk:r}of e){var i;const e=null==(i=t[s])||null==(i=i[r])?void 0:i.expiresAt;e<n&&(n=e)}return n}(f,n.entitiesMeta)),{data:p,expiryStatus:this.getExpiryStatus("symbol"==typeof p,!!e.invalidIfStale||d,c),expiresAt:h,countRef:this.gcPolicy.createCountRef({key:r,paths:f})}}get(e,...t){const[n,i]=Ae(t),{data:s}=this.memo.query(e,i,n);return"symbol"==typeof s?void 0:s}getQueryMeta(e,...t){const[n,i]=Ae(t),{data:s,paths:r}=this.memo.query(e,i,n);return{data:"symbol"==typeof s?void 0:s,countRef:this.gcPolicy.createCountRef({paths:r})}}getExpiryStatus(e,t,n={}){return n.invalidated||e&&!n.error?O.Invalid:e||t?O.InvalidIfStale:O.Valid}}function ke(e){if(s(e))return!0;if(Array.isArray(e))return 0!==e.length&&ke(e[0]);if(e&&("object"==typeof e||"function"==typeof e)){const t="schema"in e?e.schema:e;return"function"==typeof t?ke(t):Object.values(t).some((e=>ke(e)))}return!1}class Ie{constructor(e,t,n=0){this.state=void 0,this.controller=void 0,this.fetchedAt=void 0,this.abort=Ie.abort,this.state=t,this.controller=e,this.fetchedAt=n}getResponse(e,...t){return this.controller.getResponse(e,...t,this.state)}getResponseMeta(e,...t){return this.controller.getResponseMeta(e,...t,this.state)}getError(e,...t){return this.controller.getError(e,...t,this.state)}get(e,...t){return this.controller.get(e,...t,this.state)}getQueryMeta(e,...t){return this.controller.getQueryMeta(e,...t,this.state)}}function Ae(e){const t=e.length,n=new Array(t-1);for(let i=0;i<t-1;i++)n[i]=he(e[i]);return[e[t-1],n]}Ie.abort=new ne;class Le extends Error{constructor(){super("Aborted due to RESET"),this.name="ResetError"}}class Se{isOnline(){return void 0===navigator.onLine||navigator.onLine}addOnlineListener(e){addEventListener("online",e)}removeOnlineListener(e){removeEventListener("online",e)}addOfflineListener(e){addEventListener("offline",e)}removeOfflineListener(e){removeEventListener("offline",e)}}class Me{isOnline(){return!0}addOnlineListener(){}removeOnlineListener(){}addOfflineListener(){}removeOfflineListener(){}}let Ce;Ce="undefined"!=typeof navigator&&"function"==typeof addEventListener?Se:Me;var Oe=Ce;let _e={};class Re{constructor(e,t){this.started=!1,this.actions=[],this.maxBufferLength=100,this.devTools="undefined"!=typeof window&&window.__REDUX_DEVTOOLS_EXTENSION__&&window.__REDUX_DEVTOOLS_EXTENSION__.connect(l({},_e,e)),null!=e&&e.maxAge&&(this.maxBufferLength=2*e.maxAge),t&&(this.skipLogging=t)}handleAction(e,t){this.started?this.devTools.send(e,t):(this.actions.length>this.maxBufferLength&&(this.actions=this.actions.slice(this.maxBufferLength/2)),this.actions.push([e,t]))}init(e){}cleanup(){}}Re.prototype.middleware=()=>e=>t=>e(t);e.Controller=be,e.DefaultConnectionListener=Oe,e.DevToolsManager=Re,e.ExpiryStatus=O,e.GCPolicy=class{constructor({intervalMS:e=3e5,expiryMultiplier:t=2,expiresAt:n}={}){this.endpointCount=new Map,this.entityCount=new Map,this.endpointsQ=new Set,this.entitiesQ=[],n&&(this.expiresAt=n.bind(this)),this.options={intervalMS:e,expiryMultiplier:t}}init(e){this.controller=e,this.intervalId=setInterval((()=>{this.idleCallback((()=>this.runSweep()),{timeout:1e3})}),this.options.intervalMS)}cleanup(){clearInterval(this.intervalId)}createCountRef({key:e,paths:t=[]}){return()=>{var n;return e&&this.endpointCount.set(e,(null!=(n=this.endpointCount.get(e))?n:0)+1),t.forEach((e=>{var t;const{key:n,pk:i}=e;this.entityCount.has(n)||this.entityCount.set(n,new Map);const s=this.entityCount.get(n);s.set(i,(null!=(t=s.get(i))?t:0)+1)})),()=>{if(e){const t=this.endpointCount.get(e);void 0!==t&&(t<=1?(this.endpointCount.delete(e),this.endpointsQ.add(e)):this.endpointCount.set(e,t-1))}t.forEach((e=>{const{key:t,pk:n}=e;if(!this.entityCount.has(t))return;const i=this.entityCount.get(t),s=i.get(n);void 0!==s&&(s<=1?(i.delete(n),this.entitiesQ.push(e)):i.set(n,s-1))}))}}}expiresAt({fetchedAt:e,expiresAt:t}){return Math.max((t-e)*this.options.expiryMultiplier,12e4)+e}runSweep(){const e=this.controller.getState(),t=[],n=[],i=Date.now(),s=new Set;for(const t of this.endpointsQ){var r;!this.endpointCount.has(t)&&this.expiresAt(null!=(r=e.meta[t])?r:{fetchedAt:0,date:0,expiresAt:0})<i?n.push(t):s.add(t)}this.endpointsQ=s;const o=[];for(const n of this.entitiesQ){var a,c,h;(null==(a=this.entityCount.get(n.key))||!a.has(n.pk))&&this.expiresAt(null!=(c=null==(h=e.entitiesMeta[n.key])?void 0:h[n.pk])?c:{fetchedAt:0,date:0,expiresAt:0})<i?t.push(n):o.push(n)}this.entitiesQ=o,(t.length||n.length)&&this.controller.dispatch({type:H,entities:t,endpoints:n})}idleCallback(e,t){"function"==typeof requestIdleCallback?requestIdleCallback(e,t):e()}},e.ImmortalGCPolicy=ve,e.LogoutManager=class{constructor({handleLogout:e,shouldLogout:t}={}){this.middleware=e=>t=>async n=>{await t(n),n.type===j&&n.error&&this.shouldLogout(n.response)&&this.handleLogout(e)},e&&(this.handleLogout=e),t&&(this.shouldLogout=t)}cleanup(){}shouldLogout(e){return 401===e.status}handleLogout(e){e.resetEntireStore()}},e.NetworkManager=class{constructor({dataExpiryLength:e=6e4,errorExpiryLength:t=1e3}={}){this.fetching=new Map,this.controller=new be,this.middleware=e=>(this.controller=e,t=>n=>{switch(n.type){case R:return this.handleFetch(n),void 0!==n.endpoint.getOptimisticResponse&&n.endpoint.sideEffect?t(n):Promise.resolve();case j:return t(n).then((()=>{if(this.fetching.has(n.key)){var t;const i=null==(t=e.getState().meta[n.key])?void 0:t.error;i?this.handleSet(le(n.endpoint,{args:n.args,response:i,fetchedAt:n.meta.fetchedAt,error:!0})):this.handleSet(n)}}));case q:{const e=Array.from(this.fetching.values());return this.clearAll(),t(n).then((()=>{for(const{reject:t}of e)t(new Le)}))}default:return t(n)}}),this.dataExpiryLength=e,this.errorExpiryLength=t}init(){delete this.cleanupDate}cleanup(){this.cleanupDate=Date.now()}skipLogging(e){return e.type===R&&this.fetching.has(e.key)}allSettled(){if(this.fetching.size)return Promise.allSettled(this.fetching.values().map((({promise:e})=>e)))}clearAll(){for(const e of this.fetching.keys())this.clear(e)}clear(e){this.fetching.has(e)&&(this.fetching.get(e).promise.catch((()=>{})),this.fetching.delete(e))}getLastReset(){return this.cleanupDate?this.cleanupDate:this.controller.getState().lastReset}handleFetch(e){const{resolve:t,reject:n,fetchedAt:i}=e.meta,s=!e.endpoint.sideEffect,r=()=>{let r=e.endpoint(...e.args);return s||(r=(e=>e.then((e=>(t(e),e))).catch((e=>{throw n(e),e})))(r)),r=r.then((t=>{let n=this.getLastReset();return i>=n&&this.controller.resolve(e.endpoint,{args:e.args,response:t,fetchedAt:i}),t})).catch((t=>{const n=this.getLastReset();throw i>=n&&this.controller.resolve(e.endpoint,{args:e.args,response:t,fetchedAt:i,error:!0}),t})),r};return s?this.throttle(e.key,r,i).then((e=>t(e))).catch((e=>n(e))):r().catch((()=>{}))}handleSet(e){if(this.fetching.has(e.key)){const{reject:t,resolve:n}=this.fetching.get(e.key);e.error?t(e.response):n(e.response),this.clear(e.key)}}throttle(e,t,n){const i=this.getLastReset();let s=this.fetching.get(e);return s&&s.fetchedAt>i||(s=function(e){const t={fetchedAt:e};return t.promise=new Promise(((e,n)=>{t.resolve=e,t.reject=n})),t}(n),this.fetching.set(e,s),this.idleCallback((()=>{t().catch((()=>null))}),{timeout:500})),s.promise}idleCallback(e,t){e()}},e.PollingSubscription=class{constructor(e,t,n){if(this.frequencyHistogram=new Map,this.offlineListener=()=>{this.cleanup(),this.connectionListener.addOnlineListener(this.onlineListener)},this.onlineListener=()=>{this.connectionListener.removeOnlineListener(this.onlineListener);const e=Date.now();this.startId=setTimeout((()=>{this.startId&&(delete this.startId,this.update(),this.run())}),Math.max(0,this.lastFetchTime()-e+this.frequency)),this.connectionListener.addOfflineListener(this.offlineListener)},void 0===e.endpoint.pollFrequency)throw new Error("frequency needed for polling subscription");this.endpoint=e.endpoint,this.frequency=e.endpoint.pollFrequency,this.args=e.args,this.key=e.key,this.frequencyHistogram.set(this.frequency,1),this.controller=t,this.connectionListener=n||new Oe,this.connectionListener.isOnline()?this.onlineListener():this.offlineListener()}add(e){void 0!==e&&(this.frequencyHistogram.has(e)?this.frequencyHistogram.set(e,this.frequencyHistogram.get(e)+1):(this.frequencyHistogram.set(e,1),e<this.frequency&&(this.frequency=e,this.run())))}remove(e){if(void 0===e)return!1;if(this.frequencyHistogram.has(e)&&(this.frequencyHistogram.set(e,this.frequencyHistogram.get(e)-1),this.frequencyHistogram.get(e)<1)){if(this.frequencyHistogram.delete(e),0===this.frequencyHistogram.size)return this.cleanup(),!0;e<=this.frequency&&(this.frequency=Math.min(...this.frequencyHistogram.keys()),this.run())}return!1}cleanup(){this.intervalId&&(clearInterval(this.intervalId),delete this.intervalId),this.lastIntervalId&&(clearInterval(this.lastIntervalId),delete this.lastIntervalId),this.startId&&(clearTimeout(this.startId),delete this.startId),this.connectionListener.removeOnlineListener(this.onlineListener),this.connectionListener.removeOfflineListener(this.offlineListener)}update(){const e=this.endpoint,t=function(...t){return e.call(this,...t)};Object.assign(t,this.endpoint),t.dataExpiryLength=this.frequency/2,t.errorExpiryLength=this.frequency/10,t.errorPolicy=()=>"soft",t.key=()=>this.key,this.controller.fetch(t,...this.args).catch((()=>null))}run(){this.startId||(this.intervalId&&(this.lastIntervalId=this.intervalId),this.intervalId=setInterval((()=>{this.lastIntervalId&&(clearInterval(this.lastIntervalId),delete this.lastIntervalId),this.intervalId&&this.update()}),this.frequency))}lastFetchTime(){var e,t;return null!=(e=null==(t=this.controller.getState().meta[this.key])?void 0:t.date)?e:0}},e.ResetError=Le,e.SubscriptionManager=class{constructor(e){this.subscriptions={},this.controller=new be,this.middleware=e=>(this.controller=e,e=>t=>{switch(t.type){case D:try{this.handleSubscribe(t)}catch(e){console.error(e)}return Promise.resolve();case N:return this.handleUnsubscribe(t),Promise.resolve();default:return e(t)}}),this.Subscription=e}cleanup(){for(const e in this.subscriptions)this.subscriptions[e].cleanup()}handleSubscribe(e){const t=e.key;if(t in this.subscriptions){const n=e.endpoint.pollFrequency;this.subscriptions[t].add(n)}else this.subscriptions[t]=new this.Subscription(e,this.controller)}handleUnsubscribe(e){const t=e.key;if(t in this.subscriptions){const n=e.endpoint.pollFrequency;this.subscriptions[t].remove(n)&&delete this.subscriptions[t]}}},e.__INTERNAL__=oe,e.actionTypes=ee,e.actions=me,e.applyManager=function(e,t){return e.map(((e,n)=>(e.middleware||(e.middleware=null==e.getMiddleware?void 0:e.getMiddleware()),i=>(0===n&&t.bindMiddleware(i),e.middleware(t)))))},e.createReducer=function(e){return function(t,n){switch(t||(t=re),n.type){case H:return n.entities.forEach((({key:e,pk:n})=>{var i,s;null==(i=t.entities[e])||delete i[n],null==(s=t.entitiesMeta[e])||delete s[n]})),n.endpoints.forEach((e=>{delete t.endpoints[e],delete t.meta[e]})),t;case R:return function(e,t){if(t.endpoint.getOptimisticResponse&&t.endpoint.sideEffect){const n=te(t.endpoint,t.args,t.meta.fetchedAt);return l({},e,{optimistic:[...e.optimistic,n]})}return e}(t,n);case T:case j:return function(e,t,n){if(t.error)return ie(e,t,t.response);try{var i;let s;if(t.type===T){if(!t.endpoint.getOptimisticResponse)return e;try{s=t.endpoint.getOptimisticResponse.call(t.endpoint,n.snapshot(e,t.meta.fetchedAt),...t.args)}catch(t){if(t.constructor===ne)return e;throw t}}else s=t.response;const{result:r,entities:o,indexes:a,entitiesMeta:c}=A(t.endpoint.schema,s,t.args,e,t.meta),h=l({},e.endpoints,{[t.key]:r});try{if(t.endpoint.update){const e=t.endpoint.update(r,...t.args);Object.keys(e).forEach((t=>{h[t]=e[t](h[t])}))}}catch(e){console.error(`Endpoint.update() error: ${t.key}`),console.error(e)}return{entities:o,endpoints:h,indexes:a,meta:l({},e.meta,{[t.key]:{date:t.meta.date,fetchedAt:t.meta.fetchedAt,expiresAt:t.meta.expiresAt,prevExpiresAt:null==(i=e.meta[t.key])?void 0:i.expiresAt}}),entitiesMeta:c,optimistic:se(e,t),lastReset:e.lastReset}}catch(n){return"object"==typeof n&&(n.message=`Error processing ${t.key}\n\nFull Schema: ${JSON.stringify(t.endpoint.schema,void 0,2)}\n\nError:\n${n.message}`,"response"in t&&(n.response=t.response),n.status=400),ie(e,t,n)}}(t,n,e);case P:return function(e,t,n){let i;if("function"==typeof t.value){const s=n.get(t.schema,...t.args,e);if(void 0===s)return e;i=t.value(s)}else i=t.value;try{const{entities:n,indexes:s,entitiesMeta:r}=A(t.schema,i,t.args,e,t.meta);return{entities:n,endpoints:e.endpoints,indexes:s,meta:e.meta,entitiesMeta:r,optimistic:e.optimistic,lastReset:e.lastReset}}catch(t){return e}}(t,n,e);case z:case K:return function(e,t){const n=l({},e.endpoints),i=l({},e.meta),s=e=>{delete n[e];const t=l({},i[e],{expiresAt:0,invalidated:!0});delete t.error,i[e]=t};return t.type===K?s(t.key):Object.keys(n).forEach((e=>{t.testKey(e)&&s(e)})),l({},e,{endpoints:n,meta:i})}(t,n);case Q:return function(e,t){const n=l({},e.meta);return Object.keys(n).forEach((e=>{t.testKey(e)&&(n[e]=l({},n[e],{expiresAt:1}))})),l({},e,{meta:n})}(t,n);case q:return l({},re,{lastReset:n.date});default:return t}}},e.initManager=function(e,t,n){return()=>(e.forEach((e=>{null==e.init||e.init(n)})),t.gcPolicy.init(t),()=>{e.forEach((e=>{e.cleanup()})),t.gcPolicy.cleanup()})},e.initialState=re}));
|
package/dist/mock.js
CHANGED
|
@@ -184,7 +184,7 @@ function setResponseReducer(state, action, controller) {
|
|
|
184
184
|
// no reason to completely fail because of user-code error
|
|
185
185
|
// integrity of this state update is still guaranteed
|
|
186
186
|
} catch (error) {
|
|
187
|
-
console.error(`
|
|
187
|
+
console.error(`Endpoint.update() error: ${action.key}`);
|
|
188
188
|
console.error(error);
|
|
189
189
|
}
|
|
190
190
|
return {
|
|
@@ -685,11 +685,7 @@ class Controller {
|
|
|
685
685
|
*/
|
|
686
686
|
|
|
687
687
|
getResponseMeta(endpoint, ...rest) {
|
|
688
|
-
const state = rest
|
|
689
|
-
// this is typescript generics breaking
|
|
690
|
-
const args = rest.slice(0, rest.length - 1)
|
|
691
|
-
// handle FormData
|
|
692
|
-
.map(ensurePojo);
|
|
688
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
693
689
|
const isActive = args.length !== 1 || args[0] !== null;
|
|
694
690
|
const key = isActive ? endpoint.key(...args) : '';
|
|
695
691
|
const cacheEndpoints = isActive ? state.endpoints[key] : undefined;
|
|
@@ -750,9 +746,7 @@ class Controller {
|
|
|
750
746
|
* @see https://dataclient.io/docs/api/Controller#get
|
|
751
747
|
*/
|
|
752
748
|
get(schema, ...rest) {
|
|
753
|
-
const state = rest
|
|
754
|
-
// this is typescript generics breaking
|
|
755
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
749
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
756
750
|
const {
|
|
757
751
|
data
|
|
758
752
|
} = this.memo.query(schema, args, state);
|
|
@@ -764,9 +758,7 @@ class Controller {
|
|
|
764
758
|
* @see https://dataclient.io/docs/api/Controller#getQueryMeta
|
|
765
759
|
*/
|
|
766
760
|
getQueryMeta(schema, ...rest) {
|
|
767
|
-
const state = rest
|
|
768
|
-
// this is typescript generics breaking
|
|
769
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
761
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
770
762
|
const {
|
|
771
763
|
data,
|
|
772
764
|
paths
|
|
@@ -866,6 +858,18 @@ class Snapshot {
|
|
|
866
858
|
}
|
|
867
859
|
}
|
|
868
860
|
|
|
861
|
+
/** Extract state and args from rest params, applying ensurePojo to args */
|
|
862
|
+
function extractStateAndArgs(rest) {
|
|
863
|
+
const l = rest.length;
|
|
864
|
+
const args = new Array(l - 1);
|
|
865
|
+
for (let i = 0; i < l - 1; i++) {
|
|
866
|
+
// handle FormData
|
|
867
|
+
args[i] = ensurePojo(rest[i]);
|
|
868
|
+
}
|
|
869
|
+
// this is typescript generics breaking
|
|
870
|
+
return [rest[l - 1], args];
|
|
871
|
+
}
|
|
872
|
+
|
|
869
873
|
var _DevToolsManager;
|
|
870
874
|
let DEFAULT_CONFIG = {};
|
|
871
875
|
if (process.env.NODE_ENV !== 'production') {
|
|
@@ -228,11 +228,7 @@ export default class Controller {
|
|
|
228
228
|
*/
|
|
229
229
|
|
|
230
230
|
getResponseMeta(endpoint, ...rest) {
|
|
231
|
-
const state = rest
|
|
232
|
-
// this is typescript generics breaking
|
|
233
|
-
const args = rest.slice(0, rest.length - 1)
|
|
234
|
-
// handle FormData
|
|
235
|
-
.map(ensurePojo);
|
|
231
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
236
232
|
const isActive = args.length !== 1 || args[0] !== null;
|
|
237
233
|
const key = isActive ? endpoint.key(...args) : '';
|
|
238
234
|
const cacheEndpoints = isActive ? state.endpoints[key] : undefined;
|
|
@@ -293,9 +289,7 @@ export default class Controller {
|
|
|
293
289
|
* @see https://dataclient.io/docs/api/Controller#get
|
|
294
290
|
*/
|
|
295
291
|
get(schema, ...rest) {
|
|
296
|
-
const state = rest
|
|
297
|
-
// this is typescript generics breaking
|
|
298
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
292
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
299
293
|
const {
|
|
300
294
|
data
|
|
301
295
|
} = this.memo.query(schema, args, state);
|
|
@@ -307,9 +301,7 @@ export default class Controller {
|
|
|
307
301
|
* @see https://dataclient.io/docs/api/Controller#getQueryMeta
|
|
308
302
|
*/
|
|
309
303
|
getQueryMeta(schema, ...rest) {
|
|
310
|
-
const state = rest
|
|
311
|
-
// this is typescript generics breaking
|
|
312
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
304
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
313
305
|
const {
|
|
314
306
|
data,
|
|
315
307
|
paths
|
|
@@ -407,5 +399,17 @@ class Snapshot {
|
|
|
407
399
|
return this.controller.getQueryMeta(schema, ...args, this.state);
|
|
408
400
|
}
|
|
409
401
|
}
|
|
402
|
+
|
|
403
|
+
/** Extract state and args from rest params, applying ensurePojo to args */
|
|
410
404
|
Snapshot.abort = new AbortOptimistic();
|
|
411
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ExpiryStatus","MemoCache","isEntity","denormalize","validateQueryKey","AbortOptimistic","createUnsubscription","createSubscription","createExpireAll","createFetch","createInvalidate","createInvalidateAll","createReset","createSet","createSetResponse","ensurePojo","ImmortalGCPolicy","initialState","selectMeta","unsetDispatch","action","Error","unsetState","Controller","constructor","dispatch","getState","memo","gcPolicy","fetch","endpoint","args","schema","meta","promise","then","input","fetchIfStale","data","expiresAt","expiryStatus","getResponseMeta","Invalid","Date","now","invalidate","Promise","resolve","invalidateAll","options","key","testKey","expireAll","resetEntireStore","setResponse","rest","response","length","slice","setError","error","subscribe","unsubscribe","snapshot","state","fetchedAt","Snapshot","_dispatch","bindMiddleware","set","value","getError","endpoints","undefined","errorPolicy","getResponse","map","isActive","cacheEndpoints","shouldQuery","buildQueryKey","Valid","Infinity","countRef","isInvalid","schemaHasEntity","getExpiryStatus","invalidIfStale","createCountRef","paths","entities","entityExpiresAt","entitiesMeta","get","query","getQueryMeta","invalidData","invalidated","InvalidIfStale","pk","_entitiesMeta$key","entityExpiry","Array","isArray","nestedSchema","Object","values","some","x","controller","abort"],"sources":["../../src/controller/Controller.ts"],"sourcesContent":["import type {\n  ErrorTypes,\n  SnapshotInterface,\n  Schema,\n  Denormalize,\n  Queryable,\n  SchemaArgs,\n} from '@data-client/normalizr';\nimport {\n  ExpiryStatus,\n  EndpointInterface,\n  FetchFunction,\n  ResolveType,\n  DenormalizeNullable,\n  EntityPath,\n  MemoCache,\n  isEntity,\n  denormalize,\n  validateQueryKey,\n} from '@data-client/normalizr';\n\nimport AbortOptimistic from './AbortOptimistic.js';\nimport {\n  createUnsubscription,\n  createSubscription,\n} from './actions/createSubscription.js';\nimport {\n  createExpireAll,\n  createFetch,\n  createInvalidate,\n  createInvalidateAll,\n  createReset,\n  createSet,\n  createSetResponse,\n} from './actions/index.js';\nimport ensurePojo from './ensurePojo.js';\nimport type { EndpointUpdateFunction } from './types.js';\nimport { ReduxMiddlewareAPI } from '../manager/applyManager.js';\nimport type { GCInterface } from '../state/GCPolicy.js';\nimport { ImmortalGCPolicy } from '../state/GCPolicy.js';\nimport { initialState } from '../state/reducer/createReducer.js';\nimport selectMeta from '../state/selectMeta.js';\nimport type { ActionTypes, Dispatch, State } from '../types.js';\n\nexport type GenericDispatch = (value: any) => Promise<void>;\nexport type DataClientDispatch = (value: ActionTypes) => Promise<void>;\n\nexport interface ControllerConstructorProps<\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  dispatch?: D;\n  getState?: () => State<unknown>;\n  memo?: Pick<MemoCache, 'denormalize' | 'query' | 'buildQueryKey'>;\n  gcPolicy?: GCInterface;\n}\n\nconst unsetDispatch = (action: unknown): Promise<void> => {\n  throw new Error(\n    `Dispatching while constructing your middleware is not allowed. ` +\n      `Other middleware would not be applied to this dispatch.`,\n  );\n};\nconst unsetState = (): State<unknown> => {\n  // This is only the value until it is set by the DataProvider\n  /* istanbul ignore next */\n  return initialState;\n};\n\n/**\n * Imperative control of Reactive Data Client store\n * @see https://dataclient.io/docs/api/Controller\n */\nexport default class Controller<\n  // NOTE: We template on entire dispatch, so we can be contravariant on ActionTypes\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  /**\n   * Dispatches an action to Reactive Data Client reducer.\n   *\n   * @see https://dataclient.io/docs/api/Controller#dispatch\n   */\n  declare protected _dispatch: D;\n  /**\n   * Gets the latest state snapshot that is fully committed.\n   *\n   * This can be useful for imperative use-cases like event handlers.\n   * This should *not* be used to render; instead useSuspense() or useCache()\n   * @see https://dataclient.io/docs/api/Controller#getState\n   */\n  declare getState: () => State<unknown>;\n  /**\n   * Singleton to maintain referential equality between calls\n   */\n  declare readonly memo: Pick<\n    MemoCache,\n    'denormalize' | 'query' | 'buildQueryKey'\n  >;\n\n  /**\n   * Handles garbage collection\n   */\n  declare readonly gcPolicy: GCInterface;\n\n  constructor({\n    dispatch = unsetDispatch as any,\n    getState = unsetState,\n    memo = new MemoCache(),\n    gcPolicy = new ImmortalGCPolicy(),\n  }: ControllerConstructorProps<D> = {}) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n    this.memo = memo;\n    this.gcPolicy = gcPolicy;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  set dispatch(dispatch: D) {\n    /* istanbul ignore next */\n    this._dispatch = dispatch;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  get dispatch(): D {\n    return this._dispatch;\n  }\n\n  bindMiddleware({\n    dispatch,\n    getState,\n  }: {\n    dispatch: D;\n    getState: ReduxMiddlewareAPI['getState'];\n  }) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n  }\n\n  /*************** Action Dispatchers ***************/\n\n  /**\n   * Fetches the endpoint with given args, updating the Reactive Data Client cache with the response or error upon completion.\n   * @see https://dataclient.io/docs/api/Controller#fetch\n   */\n  fetch = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E>\n  : Promise<Denormalize<E['schema']>> => {\n    const action = createFetch(endpoint, {\n      args,\n    });\n    this.dispatch(action);\n\n    if (endpoint.schema) {\n      return action.meta.promise.then(input =>\n        denormalize(endpoint.schema, input, {}, args),\n      ) as any;\n    }\n    return action.meta.promise as any;\n  };\n\n  /**\n   * Fetches only if endpoint is considered 'stale'; otherwise returns undefined\n   * @see https://dataclient.io/docs/api/Controller#fetchIfStale\n   */\n  fetchIfStale = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E> | ResolveType<E>\n  : Promise<Denormalize<E['schema']>> | Denormalize<E['schema']> => {\n    const { data, expiresAt, expiryStatus } = this.getResponseMeta(\n      endpoint,\n      ...args,\n      this.getState(),\n    );\n    if (expiryStatus !== ExpiryStatus.Invalid && Date.now() <= expiresAt)\n      return data as any;\n    return this.fetch(endpoint, ...args);\n  };\n\n  /**\n   * Forces refetching and suspense on useSuspense with the same Endpoint and parameters.\n   * @see https://dataclient.io/docs/api/Controller#invalidate\n   */\n  invalidate = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createInvalidate(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Forces refetching and suspense on useSuspense on all matching endpoint result keys.\n   * @see https://dataclient.io/docs/api/Controller#invalidateAll\n   * @returns Promise that resolves when invalidation is commited.\n   */\n  invalidateAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createInvalidateAll((key: string) => options.testKey(key)));\n\n  /**\n   * Sets all matching endpoint result keys to be STALE.\n   * @see https://dataclient.io/docs/api/Controller#expireAll\n   * @returns Promise that resolves when expiry is commited. *NOT* fetch promise\n   */\n  expireAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createExpireAll((key: string) => options.testKey(key)));\n\n  /**\n   * Resets the entire Reactive Data Client cache. All inflight requests will not resolve.\n   * @see https://dataclient.io/docs/api/Controller#resetEntireStore\n   */\n  resetEntireStore = (): Promise<void> => this.dispatch(createReset());\n\n  /**\n   * Sets value for the Queryable and args.\n   * @see https://dataclient.io/docs/api/Controller#set\n   */\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, (previousValue: Denormalize<S>) => {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, any]\n  ): Promise<void> {\n    const value = rest[rest.length - 1];\n    const action = createSet(schema, {\n      args: rest.slice(0, rest.length - 1) as SchemaArgs<S>,\n      value,\n    });\n    // TODO: reject with error if this fails in reducer\n    return this.dispatch(action);\n  }\n\n  /**\n   * Sets response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setResponse\n   */\n  setResponse = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, any]\n  ): Promise<void> => {\n    const response: ResolveType<E> = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Sets an error response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setError\n   */\n  setError = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, Error]\n  ): Promise<void> => {\n    const response: Error = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n      error: true,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Resolves an inflight fetch.\n   * @see https://dataclient.io/docs/api/Controller#resolve\n   */\n  resolve = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    meta:\n      | {\n          args: readonly [...Parameters<E>];\n          response: Error;\n          fetchedAt: number;\n          error: true;\n        }\n      | {\n          args: readonly [...Parameters<E>];\n          response: any;\n          fetchedAt: number;\n          error?: false | undefined;\n        },\n  ): Promise<void> => {\n    return this.dispatch(createSetResponse(endpoint, meta as any));\n  };\n\n  /**\n   * Marks a new subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#subscribe\n   */\n  subscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createSubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Marks completion of subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#unsubscribe\n   */\n  unsubscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createUnsubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /*************** More ***************/\n\n  /* TODO:\n  abort = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): Promise<void>\n  */\n\n  /**\n   * Gets a snapshot (https://dataclient.io/docs/api/Snapshot)\n   * @see https://dataclient.io/docs/api/Controller#snapshot\n   */\n  snapshot = (state: State<unknown>, fetchedAt?: number): Snapshot<unknown> => {\n    return new Snapshot(this, state, fetchedAt);\n  };\n\n  /**\n   * Gets the error, if any, for a given endpoint. Returns undefined for no errors.\n   * @see https://dataclient.io/docs/api/Controller#getError\n   */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E['key']>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): ErrorTypes | undefined {\n    if (rest[0] === null) return;\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest.slice(0, rest.length - 1);\n    const key = endpoint.key(...args);\n\n    const meta = selectMeta(state, key);\n    const error = state.endpoints[key];\n\n    if (error !== undefined && meta?.errorPolicy === 'soft') return;\n\n    return meta?.error as any;\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponse\n   */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    // TODO: breaking: only return data\n    return this.getResponseMeta(endpoint, ...rest);\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponseMeta\n   */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      // handle FormData\n      .map(ensurePojo);\n    const isActive = args.length !== 1 || args[0] !== null;\n    const key = isActive ? endpoint.key(...args) : '';\n    const cacheEndpoints = isActive ? state.endpoints[key] : undefined;\n    const schema = endpoint.schema;\n    const meta = selectMeta(state, key);\n    let expiresAt = meta?.expiresAt;\n    // if we have no endpoint entry, and our endpoint has a schema - try querying the store\n    const shouldQuery = cacheEndpoints === undefined && schema !== undefined;\n\n    const input =\n      shouldQuery ?\n        // nothing in endpoints cache, so try querying if we have a schema to do so\n        this.memo.buildQueryKey(schema, args, state, key)\n      : cacheEndpoints;\n\n    if (!isActive) {\n      // when not active simply return the query input without denormalizing\n      return {\n        data: input as any,\n        expiryStatus: ExpiryStatus.Valid,\n        expiresAt: Infinity,\n        countRef: () => () => undefined,\n      };\n    }\n\n    let isInvalid = false;\n    if (shouldQuery) {\n      isInvalid = !validateQueryKey(input);\n      // endpoint without entities\n    } else if (!schema || !schemaHasEntity(schema)) {\n      return {\n        data: cacheEndpoints,\n        expiryStatus: this.getExpiryStatus(\n          !cacheEndpoints,\n          !!endpoint.invalidIfStale,\n          meta,\n        ),\n        expiresAt: expiresAt || 0,\n        countRef: this.gcPolicy.createCountRef({ key }),\n      };\n    }\n\n    const { data, paths } = this.memo.denormalize(\n      schema,\n      input,\n      state.entities,\n      args,\n    ) as { data: any; paths: EntityPath[] };\n\n    if (!expiresAt) {\n      // note: isInvalid can only be true if shouldQuery is true\n      if (isInvalid) expiresAt = 1;\n      // fallback to entity expiry time\n      else expiresAt = entityExpiresAt(paths, state.entitiesMeta);\n    }\n\n    return {\n      data,\n      expiryStatus: this.getExpiryStatus(\n        typeof data === 'symbol',\n        !!endpoint.invalidIfStale || isInvalid,\n        meta,\n      ),\n      expiresAt,\n      countRef: this.gcPolicy.createCountRef({ key, paths }),\n    };\n  }\n\n  /**\n   * Queries the store for a Querable schema\n   * @see https://dataclient.io/docs/api/Controller#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): DenormalizeNullable<S> | undefined {\n    const state = rest[rest.length - 1] as State<any>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      .map(ensurePojo) as SchemaArgs<S>;\n\n    const { data } = this.memo.query(schema, args, state);\n    return typeof data === 'symbol' ? undefined : data;\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Controller#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    const state = rest[rest.length - 1] as State<any>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      .map(ensurePojo) as SchemaArgs<S>;\n\n    const { data, paths } = this.memo.query(schema, args, state);\n\n    return {\n      data: typeof data === 'symbol' ? undefined : data,\n      countRef: this.gcPolicy.createCountRef({ paths }),\n    };\n  }\n\n  private getExpiryStatus(\n    invalidData: boolean,\n    invalidIfStale: boolean,\n    meta: { error?: unknown; invalidated?: unknown } = {},\n  ) {\n    // https://dataclient.io/docs/concepts/expiry-policy#expiry-status\n    // we don't track the difference between stale or fresh because that is tied to triggering\n    // conditions\n    return (\n      meta.invalidated || (invalidData && !meta.error) ? ExpiryStatus.Invalid\n      : invalidData || invalidIfStale ? ExpiryStatus.InvalidIfStale\n      : ExpiryStatus.Valid\n    );\n  }\n}\n\n// benchmark: https://www.measurethat.net/Benchmarks/Show/24691/0/min-reducer-vs-imperative-with-paths\n// earliest expiry dictates age\nfunction entityExpiresAt(\n  paths: EntityPath[],\n  entitiesMeta: {\n    readonly [entityKey: string]: {\n      readonly [pk: string]: {\n        readonly date: number;\n        readonly expiresAt: number;\n        readonly fetchedAt: number; // This is only the value until it is set by the DataProvider\n      };\n    };\n  },\n) {\n  let expiresAt = Infinity;\n  for (const { key, pk } of paths) {\n    const entityExpiry = entitiesMeta[key]?.[pk]?.expiresAt;\n    // expiresAt will always resolve to false with any comparison\n    if (entityExpiry < expiresAt) expiresAt = entityExpiry;\n  }\n  return expiresAt;\n}\n\n/** Determine whether the schema has any entities.\n *\n * Without entities, denormalization is not needed, and results should not be queried.\n */\nfunction schemaHasEntity(schema: Schema): boolean {\n  if (isEntity(schema)) return true;\n  if (Array.isArray(schema))\n    return schema.length !== 0 && schemaHasEntity(schema[0]);\n  if (schema && (typeof schema === 'object' || typeof schema === 'function')) {\n    const nestedSchema =\n      'schema' in schema ? (schema.schema as Record<string, Schema>) : schema;\n    if (typeof nestedSchema === 'function') {\n      return schemaHasEntity(nestedSchema);\n    }\n    return Object.values(nestedSchema).some(x => schemaHasEntity(x));\n  }\n  return false;\n}\n\nexport type { ErrorTypes };\n\nclass Snapshot<T = unknown> implements SnapshotInterface {\n  static readonly abort = new AbortOptimistic();\n\n  private state: State<T>;\n  private controller: Controller;\n  readonly fetchedAt: number;\n  readonly abort = Snapshot.abort;\n\n  constructor(controller: Controller, state: State<T>, fetchedAt = 0) {\n    this.state = state;\n    this.controller = controller;\n    this.fetchedAt = fetchedAt;\n  }\n\n  /*************** Data Access ***************/\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponse */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponse(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponseMeta */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponseMeta(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getError */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined {\n    return this.controller.getError(endpoint, ...args, this.state);\n  }\n\n  /**\n   * Retrieved memoized value for any Querable schema\n   * @see https://dataclient.io/docs/api/Snapshot#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): DenormalizeNullable<S> | undefined {\n    return this.controller.get(schema, ...args, this.state);\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Snapshot#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    return this.controller.getQueryMeta(schema, ...args, this.state);\n  }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
405
|
+
function extractStateAndArgs(rest) {
|
|
406
|
+
const l = rest.length;
|
|
407
|
+
const args = new Array(l - 1);
|
|
408
|
+
for (let i = 0; i < l - 1; i++) {
|
|
409
|
+
// handle FormData
|
|
410
|
+
args[i] = ensurePojo(rest[i]);
|
|
411
|
+
}
|
|
412
|
+
// this is typescript generics breaking
|
|
413
|
+
return [rest[l - 1], args];
|
|
414
|
+
}
|
|
415
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ExpiryStatus","MemoCache","isEntity","denormalize","validateQueryKey","AbortOptimistic","createUnsubscription","createSubscription","createExpireAll","createFetch","createInvalidate","createInvalidateAll","createReset","createSet","createSetResponse","ensurePojo","ImmortalGCPolicy","initialState","selectMeta","unsetDispatch","action","Error","unsetState","Controller","constructor","dispatch","getState","memo","gcPolicy","fetch","endpoint","args","schema","meta","promise","then","input","fetchIfStale","data","expiresAt","expiryStatus","getResponseMeta","Invalid","Date","now","invalidate","Promise","resolve","invalidateAll","options","key","testKey","expireAll","resetEntireStore","setResponse","rest","response","length","slice","setError","error","subscribe","unsubscribe","snapshot","state","fetchedAt","Snapshot","_dispatch","bindMiddleware","set","value","getError","endpoints","undefined","errorPolicy","getResponse","extractStateAndArgs","isActive","cacheEndpoints","shouldQuery","buildQueryKey","Valid","Infinity","countRef","isInvalid","schemaHasEntity","getExpiryStatus","invalidIfStale","createCountRef","paths","entities","entityExpiresAt","entitiesMeta","get","query","getQueryMeta","invalidData","invalidated","InvalidIfStale","pk","_entitiesMeta$key","entityExpiry","Array","isArray","nestedSchema","Object","values","some","x","controller","abort","l","i"],"sources":["../../src/controller/Controller.ts"],"sourcesContent":["import type {\n  ErrorTypes,\n  SnapshotInterface,\n  Schema,\n  Denormalize,\n  Queryable,\n  SchemaArgs,\n} from '@data-client/normalizr';\nimport {\n  ExpiryStatus,\n  EndpointInterface,\n  FetchFunction,\n  ResolveType,\n  DenormalizeNullable,\n  EntityPath,\n  MemoCache,\n  isEntity,\n  denormalize,\n  validateQueryKey,\n} from '@data-client/normalizr';\n\nimport AbortOptimistic from './AbortOptimistic.js';\nimport {\n  createUnsubscription,\n  createSubscription,\n} from './actions/createSubscription.js';\nimport {\n  createExpireAll,\n  createFetch,\n  createInvalidate,\n  createInvalidateAll,\n  createReset,\n  createSet,\n  createSetResponse,\n} from './actions/index.js';\nimport ensurePojo from './ensurePojo.js';\nimport type { EndpointUpdateFunction } from './types.js';\nimport { ReduxMiddlewareAPI } from '../manager/applyManager.js';\nimport type { GCInterface } from '../state/GCPolicy.js';\nimport { ImmortalGCPolicy } from '../state/GCPolicy.js';\nimport { initialState } from '../state/reducer/createReducer.js';\nimport selectMeta from '../state/selectMeta.js';\nimport type { ActionTypes, State } from '../types.js';\n\nexport type GenericDispatch = (value: any) => Promise<void>;\nexport type DataClientDispatch = (value: ActionTypes) => Promise<void>;\n\nexport interface ControllerConstructorProps<\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  dispatch?: D;\n  getState?: () => State<unknown>;\n  memo?: Pick<MemoCache, 'denormalize' | 'query' | 'buildQueryKey'>;\n  gcPolicy?: GCInterface;\n}\n\nconst unsetDispatch = (action: unknown): Promise<void> => {\n  throw new Error(\n    `Dispatching while constructing your middleware is not allowed. ` +\n      `Other middleware would not be applied to this dispatch.`,\n  );\n};\nconst unsetState = (): State<unknown> => {\n  // This is only the value until it is set by the DataProvider\n  /* istanbul ignore next */\n  return initialState;\n};\n\n/**\n * Imperative control of Reactive Data Client store\n * @see https://dataclient.io/docs/api/Controller\n */\nexport default class Controller<\n  // NOTE: We template on entire dispatch, so we can be contravariant on ActionTypes\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  /**\n   * Dispatches an action to Reactive Data Client reducer.\n   *\n   * @see https://dataclient.io/docs/api/Controller#dispatch\n   */\n  declare protected _dispatch: D;\n  /**\n   * Gets the latest state snapshot that is fully committed.\n   *\n   * This can be useful for imperative use-cases like event handlers.\n   * This should *not* be used to render; instead useSuspense() or useCache()\n   * @see https://dataclient.io/docs/api/Controller#getState\n   */\n  declare getState: () => State<unknown>;\n  /**\n   * Singleton to maintain referential equality between calls\n   */\n  declare readonly memo: Pick<\n    MemoCache,\n    'denormalize' | 'query' | 'buildQueryKey'\n  >;\n\n  /**\n   * Handles garbage collection\n   */\n  declare readonly gcPolicy: GCInterface;\n\n  constructor({\n    dispatch = unsetDispatch as any,\n    getState = unsetState,\n    memo = new MemoCache(),\n    gcPolicy = new ImmortalGCPolicy(),\n  }: ControllerConstructorProps<D> = {}) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n    this.memo = memo;\n    this.gcPolicy = gcPolicy;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  set dispatch(dispatch: D) {\n    /* istanbul ignore next */\n    this._dispatch = dispatch;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  get dispatch(): D {\n    return this._dispatch;\n  }\n\n  bindMiddleware({\n    dispatch,\n    getState,\n  }: {\n    dispatch: D;\n    getState: ReduxMiddlewareAPI['getState'];\n  }) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n  }\n\n  /*************** Action Dispatchers ***************/\n\n  /**\n   * Fetches the endpoint with given args, updating the Reactive Data Client cache with the response or error upon completion.\n   * @see https://dataclient.io/docs/api/Controller#fetch\n   */\n  fetch = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E>\n  : Promise<Denormalize<E['schema']>> => {\n    const action = createFetch(endpoint, {\n      args,\n    });\n    this.dispatch(action);\n\n    if (endpoint.schema) {\n      return action.meta.promise.then(input =>\n        denormalize(endpoint.schema, input, {}, args),\n      ) as any;\n    }\n    return action.meta.promise as any;\n  };\n\n  /**\n   * Fetches only if endpoint is considered 'stale'; otherwise returns undefined\n   * @see https://dataclient.io/docs/api/Controller#fetchIfStale\n   */\n  fetchIfStale = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E> | ResolveType<E>\n  : Promise<Denormalize<E['schema']>> | Denormalize<E['schema']> => {\n    const { data, expiresAt, expiryStatus } = this.getResponseMeta(\n      endpoint,\n      ...args,\n      this.getState(),\n    );\n    if (expiryStatus !== ExpiryStatus.Invalid && Date.now() <= expiresAt)\n      return data as any;\n    return this.fetch(endpoint, ...args);\n  };\n\n  /**\n   * Forces refetching and suspense on useSuspense with the same Endpoint and parameters.\n   * @see https://dataclient.io/docs/api/Controller#invalidate\n   */\n  invalidate = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createInvalidate(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Forces refetching and suspense on useSuspense on all matching endpoint result keys.\n   * @see https://dataclient.io/docs/api/Controller#invalidateAll\n   * @returns Promise that resolves when invalidation is commited.\n   */\n  invalidateAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createInvalidateAll((key: string) => options.testKey(key)));\n\n  /**\n   * Sets all matching endpoint result keys to be STALE.\n   * @see https://dataclient.io/docs/api/Controller#expireAll\n   * @returns Promise that resolves when expiry is commited. *NOT* fetch promise\n   */\n  expireAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createExpireAll((key: string) => options.testKey(key)));\n\n  /**\n   * Resets the entire Reactive Data Client cache. All inflight requests will not resolve.\n   * @see https://dataclient.io/docs/api/Controller#resetEntireStore\n   */\n  resetEntireStore = (): Promise<void> => this.dispatch(createReset());\n\n  /**\n   * Sets value for the Queryable and args.\n   * @see https://dataclient.io/docs/api/Controller#set\n   */\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, (previousValue: Denormalize<S>) => {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, any]\n  ): Promise<void> {\n    const value = rest[rest.length - 1];\n    const action = createSet(schema, {\n      args: rest.slice(0, rest.length - 1) as SchemaArgs<S>,\n      value,\n    });\n    // TODO: reject with error if this fails in reducer\n    return this.dispatch(action);\n  }\n\n  /**\n   * Sets response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setResponse\n   */\n  setResponse = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, any]\n  ): Promise<void> => {\n    const response: ResolveType<E> = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Sets an error response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setError\n   */\n  setError = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, Error]\n  ): Promise<void> => {\n    const response: Error = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n      error: true,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Resolves an inflight fetch.\n   * @see https://dataclient.io/docs/api/Controller#resolve\n   */\n  resolve = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    meta:\n      | {\n          args: readonly [...Parameters<E>];\n          response: Error;\n          fetchedAt: number;\n          error: true;\n        }\n      | {\n          args: readonly [...Parameters<E>];\n          response: any;\n          fetchedAt: number;\n          error?: false | undefined;\n        },\n  ): Promise<void> => {\n    return this.dispatch(createSetResponse(endpoint, meta as any));\n  };\n\n  /**\n   * Marks a new subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#subscribe\n   */\n  subscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createSubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Marks completion of subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#unsubscribe\n   */\n  unsubscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createUnsubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /*************** More ***************/\n\n  /* TODO:\n  abort = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): Promise<void>\n  */\n\n  /**\n   * Gets a snapshot (https://dataclient.io/docs/api/Snapshot)\n   * @see https://dataclient.io/docs/api/Controller#snapshot\n   */\n  snapshot = (state: State<unknown>, fetchedAt?: number): Snapshot<unknown> => {\n    return new Snapshot(this, state, fetchedAt);\n  };\n\n  /**\n   * Gets the error, if any, for a given endpoint. Returns undefined for no errors.\n   * @see https://dataclient.io/docs/api/Controller#getError\n   */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E['key']>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): ErrorTypes | undefined {\n    if (rest[0] === null) return;\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest.slice(0, rest.length - 1);\n    const key = endpoint.key(...args);\n\n    const meta = selectMeta(state, key);\n    const error = state.endpoints[key];\n\n    if (error !== undefined && meta?.errorPolicy === 'soft') return;\n\n    return meta?.error as any;\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponse\n   */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    // TODO: breaking: only return data\n    return this.getResponseMeta(endpoint, ...rest);\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponseMeta\n   */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    const [state, args] = extractStateAndArgs(rest);\n    const isActive = args.length !== 1 || args[0] !== null;\n    const key = isActive ? endpoint.key(...args) : '';\n    const cacheEndpoints = isActive ? state.endpoints[key] : undefined;\n    const schema = endpoint.schema;\n    const meta = selectMeta(state, key);\n    let expiresAt = meta?.expiresAt;\n    // if we have no endpoint entry, and our endpoint has a schema - try querying the store\n    const shouldQuery = cacheEndpoints === undefined && schema !== undefined;\n\n    const input =\n      shouldQuery ?\n        // nothing in endpoints cache, so try querying if we have a schema to do so\n        this.memo.buildQueryKey(schema, args, state, key)\n      : cacheEndpoints;\n\n    if (!isActive) {\n      // when not active simply return the query input without denormalizing\n      return {\n        data: input as any,\n        expiryStatus: ExpiryStatus.Valid,\n        expiresAt: Infinity,\n        countRef: () => () => undefined,\n      };\n    }\n\n    let isInvalid = false;\n    if (shouldQuery) {\n      isInvalid = !validateQueryKey(input);\n      // endpoint without entities\n    } else if (!schema || !schemaHasEntity(schema)) {\n      return {\n        data: cacheEndpoints,\n        expiryStatus: this.getExpiryStatus(\n          !cacheEndpoints,\n          !!endpoint.invalidIfStale,\n          meta,\n        ),\n        expiresAt: expiresAt || 0,\n        countRef: this.gcPolicy.createCountRef({ key }),\n      };\n    }\n\n    const { data, paths } = this.memo.denormalize(\n      schema,\n      input,\n      state.entities,\n      args,\n    ) as { data: any; paths: EntityPath[] };\n\n    if (!expiresAt) {\n      // note: isInvalid can only be true if shouldQuery is true\n      if (isInvalid) expiresAt = 1;\n      // fallback to entity expiry time\n      else expiresAt = entityExpiresAt(paths, state.entitiesMeta);\n    }\n\n    return {\n      data,\n      expiryStatus: this.getExpiryStatus(\n        typeof data === 'symbol',\n        !!endpoint.invalidIfStale || isInvalid,\n        meta,\n      ),\n      expiresAt,\n      countRef: this.gcPolicy.createCountRef({ key, paths }),\n    };\n  }\n\n  /**\n   * Queries the store for a Querable schema\n   * @see https://dataclient.io/docs/api/Controller#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): DenormalizeNullable<S> | undefined {\n    const [state, args] = extractStateAndArgs(rest);\n\n    const { data } = this.memo.query(schema, args, state);\n    return typeof data === 'symbol' ? undefined : data;\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Controller#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    const [state, args] = extractStateAndArgs(rest);\n\n    const { data, paths } = this.memo.query(schema, args, state);\n\n    return {\n      data: typeof data === 'symbol' ? undefined : data,\n      countRef: this.gcPolicy.createCountRef({ paths }),\n    };\n  }\n\n  private getExpiryStatus(\n    invalidData: boolean,\n    invalidIfStale: boolean,\n    meta: { error?: unknown; invalidated?: unknown } = {},\n  ) {\n    // https://dataclient.io/docs/concepts/expiry-policy#expiry-status\n    // we don't track the difference between stale or fresh because that is tied to triggering\n    // conditions\n    return (\n      meta.invalidated || (invalidData && !meta.error) ? ExpiryStatus.Invalid\n      : invalidData || invalidIfStale ? ExpiryStatus.InvalidIfStale\n      : ExpiryStatus.Valid\n    );\n  }\n}\n\n// benchmark: https://www.measurethat.net/Benchmarks/Show/24691/0/min-reducer-vs-imperative-with-paths\n// earliest expiry dictates age\nfunction entityExpiresAt(\n  paths: EntityPath[],\n  entitiesMeta: {\n    readonly [entityKey: string]: {\n      readonly [pk: string]: {\n        readonly date: number;\n        readonly expiresAt: number;\n        readonly fetchedAt: number; // This is only the value until it is set by the DataProvider\n      };\n    };\n  },\n) {\n  let expiresAt = Infinity;\n  for (const { key, pk } of paths) {\n    const entityExpiry = entitiesMeta[key]?.[pk]?.expiresAt;\n    // expiresAt will always resolve to false with any comparison\n    if (entityExpiry < expiresAt) expiresAt = entityExpiry;\n  }\n  return expiresAt;\n}\n\n/** Determine whether the schema has any entities.\n *\n * Without entities, denormalization is not needed, and results should not be queried.\n */\nfunction schemaHasEntity(schema: Schema): boolean {\n  if (isEntity(schema)) return true;\n  if (Array.isArray(schema))\n    return schema.length !== 0 && schemaHasEntity(schema[0]);\n  if (schema && (typeof schema === 'object' || typeof schema === 'function')) {\n    const nestedSchema =\n      'schema' in schema ? (schema.schema as Record<string, Schema>) : schema;\n    if (typeof nestedSchema === 'function') {\n      return schemaHasEntity(nestedSchema);\n    }\n    return Object.values(nestedSchema).some(x => schemaHasEntity(x));\n  }\n  return false;\n}\n\nexport type { ErrorTypes };\n\nclass Snapshot<T = unknown> implements SnapshotInterface {\n  static readonly abort = new AbortOptimistic();\n\n  private state: State<T>;\n  private controller: Controller;\n  readonly fetchedAt: number;\n  readonly abort = Snapshot.abort;\n\n  constructor(controller: Controller, state: State<T>, fetchedAt = 0) {\n    this.state = state;\n    this.controller = controller;\n    this.fetchedAt = fetchedAt;\n  }\n\n  /*************** Data Access ***************/\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponse */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponse(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponseMeta */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponseMeta(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getError */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined {\n    return this.controller.getError(endpoint, ...args, this.state);\n  }\n\n  /**\n   * Retrieved memoized value for any Querable schema\n   * @see https://dataclient.io/docs/api/Snapshot#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): DenormalizeNullable<S> | undefined {\n    return this.controller.get(schema, ...args, this.state);\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Snapshot#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    return this.controller.getQueryMeta(schema, ...args, this.state);\n  }\n}\n\n/** Extract state and args from rest params, applying ensurePojo to args */\nfunction extractStateAndArgs(rest: readonly unknown[]): [State<any>, any[]] {\n  const l = rest.length;\n  const args: any = new Array(l - 1);\n  for (let i = 0; i < l - 1; i++) {\n    // handle FormData\n    args[i] = ensurePojo(rest[i]);\n  }\n  // this is typescript generics breaking\n  return [rest[l - 1] as State<any>, args];\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -46,7 +46,7 @@ export function setResponseReducer(state, action, controller) {
|
|
|
46
46
|
// no reason to completely fail because of user-code error
|
|
47
47
|
// integrity of this state update is still guaranteed
|
|
48
48
|
} catch (error) {
|
|
49
|
-
console.error(`
|
|
49
|
+
console.error(`Endpoint.update() error: ${action.key}`);
|
|
50
50
|
console.error(error);
|
|
51
51
|
}
|
|
52
52
|
return {
|
|
@@ -107,4 +107,4 @@ function reduceError(state, action, error) {
|
|
|
107
107
|
function filterOptimistic(state, resolvingAction) {
|
|
108
108
|
return state.optimistic.filter(optimisticAction => optimisticAction.key !== resolvingAction.key || (optimisticAction.type === OPTIMISTIC ? optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt : optimisticAction.meta.date > resolvingAction.meta.date));
|
|
109
109
|
}
|
|
110
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["normalize","OPTIMISTIC","AbortOptimistic","setResponseReducer","state","action","controller","error","reduceError","response","_state$meta$action$ke","type","endpoint","getOptimisticResponse","call","snapshot","meta","fetchedAt","args","e","constructor","result","entities","indexes","entitiesMeta","schema","endpoints","_extends","key","update","updaters","Object","keys","forEach","console","date","expiresAt","prevExpiresAt","optimistic","filterOptimistic","lastReset","message","JSON","stringify","undefined","status","process","env","NODE_ENV","name","errorPolicy","resolvingAction","filter","optimisticAction"],"sources":["../../../src/state/reducer/setResponseReducer.ts"],"sourcesContent":["import { normalize } from '@data-client/normalizr';\n\nimport { OPTIMISTIC } from '../../actionTypes.js';\nimport AbortOptimistic from '../../controller/AbortOptimistic.js';\nimport type Controller from '../../controller/Controller.js';\nimport type {\n  State,\n  SetResponseAction,\n  OptimisticAction,\n} from '../../types.js';\n\nexport function setResponseReducer(\n  state: State<unknown>,\n  action: OptimisticAction | SetResponseAction,\n  controller: Controller,\n) {\n  if (action.error) {\n    return reduceError(state, action, action.response);\n  }\n  try {\n    let response: any;\n    // for true set's response is contained in action\n    if (action.type === OPTIMISTIC) {\n      // this should never happen\n      /* istanbul ignore if */\n      if (!action.endpoint.getOptimisticResponse) return state;\n      try {\n        // compute optimistic response based on current state\n        response = action.endpoint.getOptimisticResponse.call(\n          action.endpoint,\n          controller.snapshot(state, action.meta.fetchedAt),\n          ...action.args,\n        );\n      } catch (e: any) {\n        // AbortOptimistic means 'do nothing', otherwise we count the exception as endpoint failure\n        if (e.constructor === AbortOptimistic) {\n          return state;\n        }\n        throw e;\n      }\n    } else {\n      response = action.response;\n    }\n    const { result, entities, indexes, entitiesMeta } = normalize(\n      action.endpoint.schema,\n      response,\n      action.args,\n      state,\n      action.meta,\n    );\n    const endpoints: Record<string, unknown> = {\n      ...state.endpoints,\n      [action.key]: result,\n    };\n    try {\n      if (action.endpoint.update) {\n        const updaters = action.endpoint.update(result, ...action.args);\n        Object.keys(updaters).forEach(key => {\n          endpoints[key] = updaters[key](endpoints[key]);\n        });\n      }\n      // no reason to completely fail because of user-code error\n      // integrity of this state update is still guaranteed\n    } catch (error) {\n      console.error(\n        `The following error occured during Endpoint.update() for ${action.key}`,\n      );\n      console.error(error);\n    }\n    return {\n      entities,\n      endpoints,\n      indexes,\n      meta: {\n        ...state.meta,\n        [action.key]: {\n          date: action.meta.date,\n          fetchedAt: action.meta.fetchedAt,\n          expiresAt: action.meta.expiresAt,\n          prevExpiresAt: state.meta[action.key]?.expiresAt,\n        },\n      },\n      entitiesMeta,\n      optimistic: filterOptimistic(state, action),\n      lastReset: state.lastReset,\n    };\n    // reducer must update the state, so in case of processing errors we simply compute the endpoints inline\n  } catch (error: any) {\n    if (typeof error === 'object') {\n      error.message = `Error processing ${\n        action.key\n      }\\n\\nFull Schema: ${JSON.stringify(\n        action.endpoint.schema,\n        undefined,\n        2,\n      )}\\n\\nError:\\n${error.message}`;\n      if ('response' in action) error.response = action.response;\n      error.status = 400;\n    }\n\n    // this is not always bubbled up, so let's double sure this doesn't fail silently\n    /* istanbul ignore else */\n    if (process.env.NODE_ENV !== 'production') {\n      console.error(error);\n    }\n    return reduceError(state, action, error);\n  }\n}\n\nfunction reduceError(\n  state: State<unknown>,\n  action: SetResponseAction | OptimisticAction,\n  error: any,\n): State<unknown> {\n  if (error.name === 'AbortError') {\n    // In case we abort simply undo the optimistic update and act like no fetch even occured\n    // We still want those watching promises from fetch directly to observed the abort, but we don't want to\n    // Trigger errors in this case. This means theoretically improperly built aborts useSuspense() could suspend forever.\n    return {\n      ...state,\n      optimistic: filterOptimistic(state, action),\n    };\n  }\n  return {\n    ...state,\n    meta: {\n      ...state.meta,\n      [action.key]: {\n        date: action.meta.date,\n        fetchedAt: action.meta.fetchedAt,\n        expiresAt: action.meta.expiresAt,\n        error,\n        errorPolicy: action.endpoint.errorPolicy?.(error),\n      },\n    },\n    optimistic: filterOptimistic(state, action),\n  };\n}\n/** Filter all requests with same serialization that did not start after the resolving request */\nfunction filterOptimistic(\n  state: State<unknown>,\n  resolvingAction: SetResponseAction | OptimisticAction,\n) {\n  return state.optimistic.filter(\n    optimisticAction =>\n      optimisticAction.key !== resolvingAction.key ||\n      (optimisticAction.type === OPTIMISTIC ?\n        optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt\n      : optimisticAction.meta.date > resolvingAction.meta.date),\n  );\n}\n"],"mappings":";AAAA,SAASA,SAAS,QAAQ,wBAAwB;AAElD,SAASC,UAAU,QAAQ,sBAAsB;AACjD,OAAOC,eAAe,MAAM,qCAAqC;AAQjE,OAAO,SAASC,kBAAkBA,CAChCC,KAAqB,EACrBC,MAA4C,EAC5CC,UAAsB,EACtB;EACA,IAAID,MAAM,CAACE,KAAK,EAAE;IAChB,OAAOC,WAAW,CAACJ,KAAK,EAAEC,MAAM,EAAEA,MAAM,CAACI,QAAQ,CAAC;EACpD;EACA,IAAI;IAAA,IAAAC,qBAAA;IACF,IAAID,QAAa;IACjB;IACA,IAAIJ,MAAM,CAACM,IAAI,KAAKV,UAAU,EAAE;MAC9B;MACA;MACA,IAAI,CAACI,MAAM,CAACO,QAAQ,CAACC,qBAAqB,EAAE,OAAOT,KAAK;MACxD,IAAI;QACF;QACAK,QAAQ,GAAGJ,MAAM,CAACO,QAAQ,CAACC,qBAAqB,CAACC,IAAI,CACnDT,MAAM,CAACO,QAAQ,EACfN,UAAU,CAACS,QAAQ,CAACX,KAAK,EAAEC,MAAM,CAACW,IAAI,CAACC,SAAS,CAAC,EACjD,GAAGZ,MAAM,CAACa,IACZ,CAAC;MACH,CAAC,CAAC,OAAOC,CAAM,EAAE;QACf;QACA,IAAIA,CAAC,CAACC,WAAW,KAAKlB,eAAe,EAAE;UACrC,OAAOE,KAAK;QACd;QACA,MAAMe,CAAC;MACT;IACF,CAAC,MAAM;MACLV,QAAQ,GAAGJ,MAAM,CAACI,QAAQ;IAC5B;IACA,MAAM;MAAEY,MAAM;MAAEC,QAAQ;MAAEC,OAAO;MAAEC;IAAa,CAAC,GAAGxB,SAAS,CAC3DK,MAAM,CAACO,QAAQ,CAACa,MAAM,EACtBhB,QAAQ,EACRJ,MAAM,CAACa,IAAI,EACXd,KAAK,EACLC,MAAM,CAACW,IACT,CAAC;IACD,MAAMU,SAAkC,GAAAC,QAAA,KACnCvB,KAAK,CAACsB,SAAS;MAClB,CAACrB,MAAM,CAACuB,GAAG,GAAGP;IAAM,EACrB;IACD,IAAI;MACF,IAAIhB,MAAM,CAACO,QAAQ,CAACiB,MAAM,EAAE;QAC1B,MAAMC,QAAQ,GAAGzB,MAAM,CAACO,QAAQ,CAACiB,MAAM,CAACR,MAAM,EAAE,GAAGhB,MAAM,CAACa,IAAI,CAAC;QAC/Da,MAAM,CAACC,IAAI,CAACF,QAAQ,CAAC,CAACG,OAAO,CAACL,GAAG,IAAI;UACnCF,SAAS,CAACE,GAAG,CAAC,GAAGE,QAAQ,CAACF,GAAG,CAAC,CAACF,SAAS,CAACE,GAAG,CAAC,CAAC;QAChD,CAAC,CAAC;MACJ;MACA;MACA;IACF,CAAC,CAAC,OAAOrB,KAAK,EAAE;MACd2B,OAAO,CAAC3B,KAAK,CACX,4DAA4DF,MAAM,CAACuB,GAAG,EACxE,CAAC;MACDM,OAAO,CAAC3B,KAAK,CAACA,KAAK,CAAC;IACtB;IACA,OAAO;MACLe,QAAQ;MACRI,SAAS;MACTH,OAAO;MACPP,IAAI,EAAAW,QAAA,KACCvB,KAAK,CAACY,IAAI;QACb,CAACX,MAAM,CAACuB,GAAG,GAAG;UACZO,IAAI,EAAE9B,MAAM,CAACW,IAAI,CAACmB,IAAI;UACtBlB,SAAS,EAAEZ,MAAM,CAACW,IAAI,CAACC,SAAS;UAChCmB,SAAS,EAAE/B,MAAM,CAACW,IAAI,CAACoB,SAAS;UAChCC,aAAa,GAAA3B,qBAAA,GAAEN,KAAK,CAACY,IAAI,CAACX,MAAM,CAACuB,GAAG,CAAC,qBAAtBlB,qBAAA,CAAwB0B;QACzC;MAAC,EACF;MACDZ,YAAY;MACZc,UAAU,EAAEC,gBAAgB,CAACnC,KAAK,EAAEC,MAAM,CAAC;MAC3CmC,SAAS,EAAEpC,KAAK,CAACoC;IACnB,CAAC;IACD;EACF,CAAC,CAAC,OAAOjC,KAAU,EAAE;IACnB,IAAI,OAAOA,KAAK,KAAK,QAAQ,EAAE;MAC7BA,KAAK,CAACkC,OAAO,GAAG,oBACdpC,MAAM,CAACuB,GAAG,oBACQc,IAAI,CAACC,SAAS,CAChCtC,MAAM,CAACO,QAAQ,CAACa,MAAM,EACtBmB,SAAS,EACT,CACF,CAAC,eAAerC,KAAK,CAACkC,OAAO,EAAE;MAC/B,IAAI,UAAU,IAAIpC,MAAM,EAAEE,KAAK,CAACE,QAAQ,GAAGJ,MAAM,CAACI,QAAQ;MAC1DF,KAAK,CAACsC,MAAM,GAAG,GAAG;IACpB;;IAEA;IACA;IACA,IAAIC,OAAO,CAACC,GAAG,CAACC,QAAQ,KAAK,YAAY,EAAE;MACzCd,OAAO,CAAC3B,KAAK,CAACA,KAAK,CAAC;IACtB;IACA,OAAOC,WAAW,CAACJ,KAAK,EAAEC,MAAM,EAAEE,KAAK,CAAC;EAC1C;AACF;AAEA,SAASC,WAAWA,CAClBJ,KAAqB,EACrBC,MAA4C,EAC5CE,KAAU,EACM;EAChB,IAAIA,KAAK,CAAC0C,IAAI,KAAK,YAAY,EAAE;IAC/B;IACA;IACA;IACA,OAAAtB,QAAA,KACKvB,KAAK;MACRkC,UAAU,EAAEC,gBAAgB,CAACnC,KAAK,EAAEC,MAAM;IAAC;EAE/C;EACA,OAAAsB,QAAA,KACKvB,KAAK;IACRY,IAAI,EAAAW,QAAA,KACCvB,KAAK,CAACY,IAAI;MACb,CAACX,MAAM,CAACuB,GAAG,GAAG;QACZO,IAAI,EAAE9B,MAAM,CAACW,IAAI,CAACmB,IAAI;QACtBlB,SAAS,EAAEZ,MAAM,CAACW,IAAI,CAACC,SAAS;QAChCmB,SAAS,EAAE/B,MAAM,CAACW,IAAI,CAACoB,SAAS;QAChC7B,KAAK;QACL2C,WAAW,EAAE7C,MAAM,CAACO,QAAQ,CAACsC,WAAW,oBAA3B7C,MAAM,CAACO,QAAQ,CAACsC,WAAW,CAAG3C,KAAK;MAClD;IAAC,EACF;IACD+B,UAAU,EAAEC,gBAAgB,CAACnC,KAAK,EAAEC,MAAM;EAAC;AAE/C;AACA;AACA,SAASkC,gBAAgBA,CACvBnC,KAAqB,EACrB+C,eAAqD,EACrD;EACA,OAAO/C,KAAK,CAACkC,UAAU,CAACc,MAAM,CAC5BC,gBAAgB,IACdA,gBAAgB,CAACzB,GAAG,KAAKuB,eAAe,CAACvB,GAAG,KAC3CyB,gBAAgB,CAAC1C,IAAI,KAAKV,UAAU,GACnCoD,gBAAgB,CAACrC,IAAI,CAACC,SAAS,KAAKkC,eAAe,CAACnC,IAAI,CAACC,SAAS,GAClEoC,gBAAgB,CAACrC,IAAI,CAACmB,IAAI,GAAGgB,eAAe,CAACnC,IAAI,CAACmB,IAAI,CAC5D,CAAC;AACH","ignoreList":[]}
|
|
110
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["normalize","OPTIMISTIC","AbortOptimistic","setResponseReducer","state","action","controller","error","reduceError","response","_state$meta$action$ke","type","endpoint","getOptimisticResponse","call","snapshot","meta","fetchedAt","args","e","constructor","result","entities","indexes","entitiesMeta","schema","endpoints","_extends","key","update","updaters","Object","keys","forEach","console","date","expiresAt","prevExpiresAt","optimistic","filterOptimistic","lastReset","message","JSON","stringify","undefined","status","process","env","NODE_ENV","name","errorPolicy","resolvingAction","filter","optimisticAction"],"sources":["../../../src/state/reducer/setResponseReducer.ts"],"sourcesContent":["import { normalize } from '@data-client/normalizr';\n\nimport { OPTIMISTIC } from '../../actionTypes.js';\nimport AbortOptimistic from '../../controller/AbortOptimistic.js';\nimport type Controller from '../../controller/Controller.js';\nimport type {\n  State,\n  SetResponseAction,\n  OptimisticAction,\n} from '../../types.js';\n\nexport function setResponseReducer(\n  state: State<unknown>,\n  action: OptimisticAction | SetResponseAction,\n  controller: Controller,\n) {\n  if (action.error) {\n    return reduceError(state, action, action.response);\n  }\n  try {\n    let response: any;\n    // for true set's response is contained in action\n    if (action.type === OPTIMISTIC) {\n      // this should never happen\n      /* istanbul ignore if */\n      if (!action.endpoint.getOptimisticResponse) return state;\n      try {\n        // compute optimistic response based on current state\n        response = action.endpoint.getOptimisticResponse.call(\n          action.endpoint,\n          controller.snapshot(state, action.meta.fetchedAt),\n          ...action.args,\n        );\n      } catch (e: any) {\n        // AbortOptimistic means 'do nothing', otherwise we count the exception as endpoint failure\n        if (e.constructor === AbortOptimistic) {\n          return state;\n        }\n        throw e;\n      }\n    } else {\n      response = action.response;\n    }\n    const { result, entities, indexes, entitiesMeta } = normalize(\n      action.endpoint.schema,\n      response,\n      action.args,\n      state,\n      action.meta,\n    );\n    const endpoints: Record<string, unknown> = {\n      ...state.endpoints,\n      [action.key]: result,\n    };\n    try {\n      if (action.endpoint.update) {\n        const updaters = action.endpoint.update(result, ...action.args);\n        Object.keys(updaters).forEach(key => {\n          endpoints[key] = updaters[key](endpoints[key]);\n        });\n      }\n      // no reason to completely fail because of user-code error\n      // integrity of this state update is still guaranteed\n    } catch (error) {\n      console.error(`Endpoint.update() error: ${action.key}`);\n      console.error(error);\n    }\n    return {\n      entities,\n      endpoints,\n      indexes,\n      meta: {\n        ...state.meta,\n        [action.key]: {\n          date: action.meta.date,\n          fetchedAt: action.meta.fetchedAt,\n          expiresAt: action.meta.expiresAt,\n          prevExpiresAt: state.meta[action.key]?.expiresAt,\n        },\n      },\n      entitiesMeta,\n      optimistic: filterOptimistic(state, action),\n      lastReset: state.lastReset,\n    };\n    // reducer must update the state, so in case of processing errors we simply compute the endpoints inline\n  } catch (error: any) {\n    if (typeof error === 'object') {\n      error.message = `Error processing ${\n        action.key\n      }\\n\\nFull Schema: ${JSON.stringify(\n        action.endpoint.schema,\n        undefined,\n        2,\n      )}\\n\\nError:\\n${error.message}`;\n      if ('response' in action) error.response = action.response;\n      error.status = 400;\n    }\n\n    // this is not always bubbled up, so let's double sure this doesn't fail silently\n    /* istanbul ignore else */\n    if (process.env.NODE_ENV !== 'production') {\n      console.error(error);\n    }\n    return reduceError(state, action, error);\n  }\n}\n\nfunction reduceError(\n  state: State<unknown>,\n  action: SetResponseAction | OptimisticAction,\n  error: any,\n): State<unknown> {\n  if (error.name === 'AbortError') {\n    // In case we abort simply undo the optimistic update and act like no fetch even occured\n    // We still want those watching promises from fetch directly to observed the abort, but we don't want to\n    // Trigger errors in this case. This means theoretically improperly built aborts useSuspense() could suspend forever.\n    return {\n      ...state,\n      optimistic: filterOptimistic(state, action),\n    };\n  }\n  return {\n    ...state,\n    meta: {\n      ...state.meta,\n      [action.key]: {\n        date: action.meta.date,\n        fetchedAt: action.meta.fetchedAt,\n        expiresAt: action.meta.expiresAt,\n        error,\n        errorPolicy: action.endpoint.errorPolicy?.(error),\n      },\n    },\n    optimistic: filterOptimistic(state, action),\n  };\n}\n/** Filter all requests with same serialization that did not start after the resolving request */\nfunction filterOptimistic(\n  state: State<unknown>,\n  resolvingAction: SetResponseAction | OptimisticAction,\n) {\n  return state.optimistic.filter(\n    optimisticAction =>\n      optimisticAction.key !== resolvingAction.key ||\n      (optimisticAction.type === OPTIMISTIC ?\n        optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt\n      : optimisticAction.meta.date > resolvingAction.meta.date),\n  );\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Controller.d.ts","sourceRoot":"","sources":["../../src/controller/Controller.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,UAAU,EACV,iBAAiB,EACjB,MAAM,EACN,WAAW,EACX,SAAS,EACT,UAAU,EACX,MAAM,wBAAwB,CAAC;AAChC,OAAO,EACL,YAAY,EACZ,iBAAiB,EACjB,aAAa,EACb,WAAW,EACX,mBAAmB,EAEnB,SAAS,EAIV,MAAM,wBAAwB,CAAC;AAEhC,OAAO,eAAe,MAAM,sBAAsB,CAAC;AAenD,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,YAAY,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAChE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAIxD,OAAO,KAAK,EAAE,WAAW,
|
|
1
|
+
{"version":3,"file":"Controller.d.ts","sourceRoot":"","sources":["../../src/controller/Controller.ts"],"names":[],"mappings":"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"}
|
|
@@ -228,11 +228,7 @@ export default class Controller {
|
|
|
228
228
|
*/
|
|
229
229
|
|
|
230
230
|
getResponseMeta(endpoint, ...rest) {
|
|
231
|
-
const state = rest
|
|
232
|
-
// this is typescript generics breaking
|
|
233
|
-
const args = rest.slice(0, rest.length - 1)
|
|
234
|
-
// handle FormData
|
|
235
|
-
.map(ensurePojo);
|
|
231
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
236
232
|
const isActive = args.length !== 1 || args[0] !== null;
|
|
237
233
|
const key = isActive ? endpoint.key(...args) : '';
|
|
238
234
|
const cacheEndpoints = isActive ? state.endpoints[key] : undefined;
|
|
@@ -293,9 +289,7 @@ export default class Controller {
|
|
|
293
289
|
* @see https://dataclient.io/docs/api/Controller#get
|
|
294
290
|
*/
|
|
295
291
|
get(schema, ...rest) {
|
|
296
|
-
const state = rest
|
|
297
|
-
// this is typescript generics breaking
|
|
298
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
292
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
299
293
|
const {
|
|
300
294
|
data
|
|
301
295
|
} = this.memo.query(schema, args, state);
|
|
@@ -307,9 +301,7 @@ export default class Controller {
|
|
|
307
301
|
* @see https://dataclient.io/docs/api/Controller#getQueryMeta
|
|
308
302
|
*/
|
|
309
303
|
getQueryMeta(schema, ...rest) {
|
|
310
|
-
const state = rest
|
|
311
|
-
// this is typescript generics breaking
|
|
312
|
-
const args = rest.slice(0, rest.length - 1).map(ensurePojo);
|
|
304
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
313
305
|
const {
|
|
314
306
|
data,
|
|
315
307
|
paths
|
|
@@ -407,5 +399,17 @@ class Snapshot {
|
|
|
407
399
|
return this.controller.getQueryMeta(schema, ...args, this.state);
|
|
408
400
|
}
|
|
409
401
|
}
|
|
402
|
+
|
|
403
|
+
/** Extract state and args from rest params, applying ensurePojo to args */
|
|
410
404
|
Snapshot.abort = new AbortOptimistic();
|
|
411
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ExpiryStatus","MemoCache","isEntity","denormalize","validateQueryKey","AbortOptimistic","createUnsubscription","createSubscription","createExpireAll","createFetch","createInvalidate","createInvalidateAll","createReset","createSet","createSetResponse","ensurePojo","ImmortalGCPolicy","initialState","selectMeta","unsetDispatch","action","Error","unsetState","Controller","constructor","dispatch","getState","memo","gcPolicy","fetch","endpoint","args","schema","meta","promise","then","input","fetchIfStale","data","expiresAt","expiryStatus","getResponseMeta","Invalid","Date","now","invalidate","Promise","resolve","invalidateAll","options","key","testKey","expireAll","resetEntireStore","setResponse","rest","response","length","slice","setError","error","subscribe","unsubscribe","snapshot","state","fetchedAt","Snapshot","_dispatch","bindMiddleware","set","value","getError","endpoints","undefined","errorPolicy","getResponse","map","isActive","cacheEndpoints","shouldQuery","buildQueryKey","Valid","Infinity","countRef","isInvalid","schemaHasEntity","getExpiryStatus","invalidIfStale","createCountRef","paths","entities","entityExpiresAt","entitiesMeta","get","query","getQueryMeta","invalidData","invalidated","InvalidIfStale","pk","_entitiesMeta$key","entityExpiry","Array","isArray","nestedSchema","Object","values","some","x","controller","abort"],"sources":["../../src/controller/Controller.ts"],"sourcesContent":["import type {\n  ErrorTypes,\n  SnapshotInterface,\n  Schema,\n  Denormalize,\n  Queryable,\n  SchemaArgs,\n} from '@data-client/normalizr';\nimport {\n  ExpiryStatus,\n  EndpointInterface,\n  FetchFunction,\n  ResolveType,\n  DenormalizeNullable,\n  EntityPath,\n  MemoCache,\n  isEntity,\n  denormalize,\n  validateQueryKey,\n} from '@data-client/normalizr';\n\nimport AbortOptimistic from './AbortOptimistic.js';\nimport {\n  createUnsubscription,\n  createSubscription,\n} from './actions/createSubscription.js';\nimport {\n  createExpireAll,\n  createFetch,\n  createInvalidate,\n  createInvalidateAll,\n  createReset,\n  createSet,\n  createSetResponse,\n} from './actions/index.js';\nimport ensurePojo from './ensurePojo.js';\nimport type { EndpointUpdateFunction } from './types.js';\nimport { ReduxMiddlewareAPI } from '../manager/applyManager.js';\nimport type { GCInterface } from '../state/GCPolicy.js';\nimport { ImmortalGCPolicy } from '../state/GCPolicy.js';\nimport { initialState } from '../state/reducer/createReducer.js';\nimport selectMeta from '../state/selectMeta.js';\nimport type { ActionTypes, Dispatch, State } from '../types.js';\n\nexport type GenericDispatch = (value: any) => Promise<void>;\nexport type DataClientDispatch = (value: ActionTypes) => Promise<void>;\n\nexport interface ControllerConstructorProps<\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  dispatch?: D;\n  getState?: () => State<unknown>;\n  memo?: Pick<MemoCache, 'denormalize' | 'query' | 'buildQueryKey'>;\n  gcPolicy?: GCInterface;\n}\n\nconst unsetDispatch = (action: unknown): Promise<void> => {\n  throw new Error(\n    `Dispatching while constructing your middleware is not allowed. ` +\n      `Other middleware would not be applied to this dispatch.`,\n  );\n};\nconst unsetState = (): State<unknown> => {\n  // This is only the value until it is set by the DataProvider\n  /* istanbul ignore next */\n  return initialState;\n};\n\n/**\n * Imperative control of Reactive Data Client store\n * @see https://dataclient.io/docs/api/Controller\n */\nexport default class Controller<\n  // NOTE: We template on entire dispatch, so we can be contravariant on ActionTypes\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  /**\n   * Dispatches an action to Reactive Data Client reducer.\n   *\n   * @see https://dataclient.io/docs/api/Controller#dispatch\n   */\n  declare protected _dispatch: D;\n  /**\n   * Gets the latest state snapshot that is fully committed.\n   *\n   * This can be useful for imperative use-cases like event handlers.\n   * This should *not* be used to render; instead useSuspense() or useCache()\n   * @see https://dataclient.io/docs/api/Controller#getState\n   */\n  declare getState: () => State<unknown>;\n  /**\n   * Singleton to maintain referential equality between calls\n   */\n  declare readonly memo: Pick<\n    MemoCache,\n    'denormalize' | 'query' | 'buildQueryKey'\n  >;\n\n  /**\n   * Handles garbage collection\n   */\n  declare readonly gcPolicy: GCInterface;\n\n  constructor({\n    dispatch = unsetDispatch as any,\n    getState = unsetState,\n    memo = new MemoCache(),\n    gcPolicy = new ImmortalGCPolicy(),\n  }: ControllerConstructorProps<D> = {}) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n    this.memo = memo;\n    this.gcPolicy = gcPolicy;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  set dispatch(dispatch: D) {\n    /* istanbul ignore next */\n    this._dispatch = dispatch;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  get dispatch(): D {\n    return this._dispatch;\n  }\n\n  bindMiddleware({\n    dispatch,\n    getState,\n  }: {\n    dispatch: D;\n    getState: ReduxMiddlewareAPI['getState'];\n  }) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n  }\n\n  /*************** Action Dispatchers ***************/\n\n  /**\n   * Fetches the endpoint with given args, updating the Reactive Data Client cache with the response or error upon completion.\n   * @see https://dataclient.io/docs/api/Controller#fetch\n   */\n  fetch = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E>\n  : Promise<Denormalize<E['schema']>> => {\n    const action = createFetch(endpoint, {\n      args,\n    });\n    this.dispatch(action);\n\n    if (endpoint.schema) {\n      return action.meta.promise.then(input =>\n        denormalize(endpoint.schema, input, {}, args),\n      ) as any;\n    }\n    return action.meta.promise as any;\n  };\n\n  /**\n   * Fetches only if endpoint is considered 'stale'; otherwise returns undefined\n   * @see https://dataclient.io/docs/api/Controller#fetchIfStale\n   */\n  fetchIfStale = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E> | ResolveType<E>\n  : Promise<Denormalize<E['schema']>> | Denormalize<E['schema']> => {\n    const { data, expiresAt, expiryStatus } = this.getResponseMeta(\n      endpoint,\n      ...args,\n      this.getState(),\n    );\n    if (expiryStatus !== ExpiryStatus.Invalid && Date.now() <= expiresAt)\n      return data as any;\n    return this.fetch(endpoint, ...args);\n  };\n\n  /**\n   * Forces refetching and suspense on useSuspense with the same Endpoint and parameters.\n   * @see https://dataclient.io/docs/api/Controller#invalidate\n   */\n  invalidate = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createInvalidate(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Forces refetching and suspense on useSuspense on all matching endpoint result keys.\n   * @see https://dataclient.io/docs/api/Controller#invalidateAll\n   * @returns Promise that resolves when invalidation is commited.\n   */\n  invalidateAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createInvalidateAll((key: string) => options.testKey(key)));\n\n  /**\n   * Sets all matching endpoint result keys to be STALE.\n   * @see https://dataclient.io/docs/api/Controller#expireAll\n   * @returns Promise that resolves when expiry is commited. *NOT* fetch promise\n   */\n  expireAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createExpireAll((key: string) => options.testKey(key)));\n\n  /**\n   * Resets the entire Reactive Data Client cache. All inflight requests will not resolve.\n   * @see https://dataclient.io/docs/api/Controller#resetEntireStore\n   */\n  resetEntireStore = (): Promise<void> => this.dispatch(createReset());\n\n  /**\n   * Sets value for the Queryable and args.\n   * @see https://dataclient.io/docs/api/Controller#set\n   */\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, (previousValue: Denormalize<S>) => {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, any]\n  ): Promise<void> {\n    const value = rest[rest.length - 1];\n    const action = createSet(schema, {\n      args: rest.slice(0, rest.length - 1) as SchemaArgs<S>,\n      value,\n    });\n    // TODO: reject with error if this fails in reducer\n    return this.dispatch(action);\n  }\n\n  /**\n   * Sets response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setResponse\n   */\n  setResponse = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, any]\n  ): Promise<void> => {\n    const response: ResolveType<E> = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Sets an error response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setError\n   */\n  setError = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, Error]\n  ): Promise<void> => {\n    const response: Error = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n      error: true,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Resolves an inflight fetch.\n   * @see https://dataclient.io/docs/api/Controller#resolve\n   */\n  resolve = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    meta:\n      | {\n          args: readonly [...Parameters<E>];\n          response: Error;\n          fetchedAt: number;\n          error: true;\n        }\n      | {\n          args: readonly [...Parameters<E>];\n          response: any;\n          fetchedAt: number;\n          error?: false | undefined;\n        },\n  ): Promise<void> => {\n    return this.dispatch(createSetResponse(endpoint, meta as any));\n  };\n\n  /**\n   * Marks a new subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#subscribe\n   */\n  subscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createSubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Marks completion of subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#unsubscribe\n   */\n  unsubscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createUnsubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /*************** More ***************/\n\n  /* TODO:\n  abort = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): Promise<void>\n  */\n\n  /**\n   * Gets a snapshot (https://dataclient.io/docs/api/Snapshot)\n   * @see https://dataclient.io/docs/api/Controller#snapshot\n   */\n  snapshot = (state: State<unknown>, fetchedAt?: number): Snapshot<unknown> => {\n    return new Snapshot(this, state, fetchedAt);\n  };\n\n  /**\n   * Gets the error, if any, for a given endpoint. Returns undefined for no errors.\n   * @see https://dataclient.io/docs/api/Controller#getError\n   */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E['key']>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): ErrorTypes | undefined {\n    if (rest[0] === null) return;\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest.slice(0, rest.length - 1);\n    const key = endpoint.key(...args);\n\n    const meta = selectMeta(state, key);\n    const error = state.endpoints[key];\n\n    if (error !== undefined && meta?.errorPolicy === 'soft') return;\n\n    return meta?.error as any;\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponse\n   */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    // TODO: breaking: only return data\n    return this.getResponseMeta(endpoint, ...rest);\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponseMeta\n   */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      // handle FormData\n      .map(ensurePojo);\n    const isActive = args.length !== 1 || args[0] !== null;\n    const key = isActive ? endpoint.key(...args) : '';\n    const cacheEndpoints = isActive ? state.endpoints[key] : undefined;\n    const schema = endpoint.schema;\n    const meta = selectMeta(state, key);\n    let expiresAt = meta?.expiresAt;\n    // if we have no endpoint entry, and our endpoint has a schema - try querying the store\n    const shouldQuery = cacheEndpoints === undefined && schema !== undefined;\n\n    const input =\n      shouldQuery ?\n        // nothing in endpoints cache, so try querying if we have a schema to do so\n        this.memo.buildQueryKey(schema, args, state, key)\n      : cacheEndpoints;\n\n    if (!isActive) {\n      // when not active simply return the query input without denormalizing\n      return {\n        data: input as any,\n        expiryStatus: ExpiryStatus.Valid,\n        expiresAt: Infinity,\n        countRef: () => () => undefined,\n      };\n    }\n\n    let isInvalid = false;\n    if (shouldQuery) {\n      isInvalid = !validateQueryKey(input);\n      // endpoint without entities\n    } else if (!schema || !schemaHasEntity(schema)) {\n      return {\n        data: cacheEndpoints,\n        expiryStatus: this.getExpiryStatus(\n          !cacheEndpoints,\n          !!endpoint.invalidIfStale,\n          meta,\n        ),\n        expiresAt: expiresAt || 0,\n        countRef: this.gcPolicy.createCountRef({ key }),\n      };\n    }\n\n    const { data, paths } = this.memo.denormalize(\n      schema,\n      input,\n      state.entities,\n      args,\n    ) as { data: any; paths: EntityPath[] };\n\n    if (!expiresAt) {\n      // note: isInvalid can only be true if shouldQuery is true\n      if (isInvalid) expiresAt = 1;\n      // fallback to entity expiry time\n      else expiresAt = entityExpiresAt(paths, state.entitiesMeta);\n    }\n\n    return {\n      data,\n      expiryStatus: this.getExpiryStatus(\n        typeof data === 'symbol',\n        !!endpoint.invalidIfStale || isInvalid,\n        meta,\n      ),\n      expiresAt,\n      countRef: this.gcPolicy.createCountRef({ key, paths }),\n    };\n  }\n\n  /**\n   * Queries the store for a Querable schema\n   * @see https://dataclient.io/docs/api/Controller#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): DenormalizeNullable<S> | undefined {\n    const state = rest[rest.length - 1] as State<any>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      .map(ensurePojo) as SchemaArgs<S>;\n\n    const { data } = this.memo.query(schema, args, state);\n    return typeof data === 'symbol' ? undefined : data;\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Controller#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    const state = rest[rest.length - 1] as State<any>;\n    // this is typescript generics breaking\n    const args: any = rest\n      .slice(0, rest.length - 1)\n      .map(ensurePojo) as SchemaArgs<S>;\n\n    const { data, paths } = this.memo.query(schema, args, state);\n\n    return {\n      data: typeof data === 'symbol' ? undefined : data,\n      countRef: this.gcPolicy.createCountRef({ paths }),\n    };\n  }\n\n  private getExpiryStatus(\n    invalidData: boolean,\n    invalidIfStale: boolean,\n    meta: { error?: unknown; invalidated?: unknown } = {},\n  ) {\n    // https://dataclient.io/docs/concepts/expiry-policy#expiry-status\n    // we don't track the difference between stale or fresh because that is tied to triggering\n    // conditions\n    return (\n      meta.invalidated || (invalidData && !meta.error) ? ExpiryStatus.Invalid\n      : invalidData || invalidIfStale ? ExpiryStatus.InvalidIfStale\n      : ExpiryStatus.Valid\n    );\n  }\n}\n\n// benchmark: https://www.measurethat.net/Benchmarks/Show/24691/0/min-reducer-vs-imperative-with-paths\n// earliest expiry dictates age\nfunction entityExpiresAt(\n  paths: EntityPath[],\n  entitiesMeta: {\n    readonly [entityKey: string]: {\n      readonly [pk: string]: {\n        readonly date: number;\n        readonly expiresAt: number;\n        readonly fetchedAt: number; // This is only the value until it is set by the DataProvider\n      };\n    };\n  },\n) {\n  let expiresAt = Infinity;\n  for (const { key, pk } of paths) {\n    const entityExpiry = entitiesMeta[key]?.[pk]?.expiresAt;\n    // expiresAt will always resolve to false with any comparison\n    if (entityExpiry < expiresAt) expiresAt = entityExpiry;\n  }\n  return expiresAt;\n}\n\n/** Determine whether the schema has any entities.\n *\n * Without entities, denormalization is not needed, and results should not be queried.\n */\nfunction schemaHasEntity(schema: Schema): boolean {\n  if (isEntity(schema)) return true;\n  if (Array.isArray(schema))\n    return schema.length !== 0 && schemaHasEntity(schema[0]);\n  if (schema && (typeof schema === 'object' || typeof schema === 'function')) {\n    const nestedSchema =\n      'schema' in schema ? (schema.schema as Record<string, Schema>) : schema;\n    if (typeof nestedSchema === 'function') {\n      return schemaHasEntity(nestedSchema);\n    }\n    return Object.values(nestedSchema).some(x => schemaHasEntity(x));\n  }\n  return false;\n}\n\nexport type { ErrorTypes };\n\nclass Snapshot<T = unknown> implements SnapshotInterface {\n  static readonly abort = new AbortOptimistic();\n\n  private state: State<T>;\n  private controller: Controller;\n  readonly fetchedAt: number;\n  readonly abort = Snapshot.abort;\n\n  constructor(controller: Controller, state: State<T>, fetchedAt = 0) {\n    this.state = state;\n    this.controller = controller;\n    this.fetchedAt = fetchedAt;\n  }\n\n  /*************** Data Access ***************/\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponse */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponse(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponseMeta */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponseMeta(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getError */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined {\n    return this.controller.getError(endpoint, ...args, this.state);\n  }\n\n  /**\n   * Retrieved memoized value for any Querable schema\n   * @see https://dataclient.io/docs/api/Snapshot#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): DenormalizeNullable<S> | undefined {\n    return this.controller.get(schema, ...args, this.state);\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Snapshot#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    return this.controller.getQueryMeta(schema, ...args, this.state);\n  }\n}\n"],"mappings":"AAQA,SACEA,YAAY,EAMZC,SAAS,EACTC,QAAQ,EACRC,WAAW,EACXC,gBAAgB,QACX,wBAAwB;AAE/B,OAAOC,eAAe,MAAM,sBAAsB;AAClD,SACEC,oBAAoB,EACpBC,kBAAkB,QACb,iCAAiC;AACxC,SACEC,eAAe,EACfC,WAAW,EACXC,gBAAgB,EAChBC,mBAAmB,EACnBC,WAAW,EACXC,SAAS,EACTC,iBAAiB,QACZ,oBAAoB;AAC3B,OAAOC,UAAU,MAAM,iBAAiB;AAIxC,SAASC,gBAAgB,QAAQ,sBAAsB;AACvD,SAASC,YAAY,QAAQ,mCAAmC;AAChE,OAAOC,UAAU,MAAM,wBAAwB;AAe/C,MAAMC,aAAa,GAAIC,MAAe,IAAoB;EACxD,MAAM,IAAIC,KAAK,CACb,iEAAiE,GAC/D,yDACJ,CAAC;AACH,CAAC;AACD,MAAMC,UAAU,GAAGA,CAAA,KAAsB;EACvC;EACA;EACA,OAAOL,YAAY;AACrB,CAAC;;AAED;AACA;AACA;AACA;AACA,eAAe,MAAMM,UAAU,CAG7B;EACA;AACF;AACA;AACA;AACA;;EAEE;AACF;AACA;AACA;AACA;AACA;AACA;;EAEE;AACF;AACA;;EAME;AACF;AACA;;EAGEC,WAAWA,CAAC;IACVC,QAAQ,GAAGN,aAAoB;IAC/BO,QAAQ,GAAGJ,UAAU;IACrBK,IAAI,GAAG,IAAI1B,SAAS,CAAC,CAAC;IACtB2B,QAAQ,GAAG,IAAIZ,gBAAgB,CAAC;EACH,CAAC,GAAG,CAAC,CAAC,EAAE;IA6BvC;IAEA;AACF;AACA;AACA;IAHE,KAIAa,KAAK,GAAG,CAGNC,QAAW,EACX,GAAGC,IAAiC,KAEC;MACrC,MAAMX,MAAM,GAAGX,WAAW,CAACqB,QAAQ,EAAE;QACnCC;MACF,CAAC,CAAC;MACF,IAAI,CAACN,QAAQ,CAACL,MAAM,CAAC;MAErB,IAAIU,QAAQ,CAACE,MAAM,EAAE;QACnB,OAAOZ,MAAM,CAACa,IAAI,CAACC,OAAO,CAACC,IAAI,CAACC,KAAK,IACnCjC,WAAW,CAAC2B,QAAQ,CAACE,MAAM,EAAEI,KAAK,EAAE,CAAC,CAAC,EAAEL,IAAI,CAC9C,CAAC;MACH;MACA,OAAOX,MAAM,CAACa,IAAI,CAACC,OAAO;IAC5B,CAAC;IAED;AACF;AACA;AACA;IAHE,KAIAG,YAAY,GAAG,CAGbP,QAAW,EACX,GAAGC,IAAiC,KAE4B;MAChE,MAAM;QAAEO,IAAI;QAAEC,SAAS;QAAEC;MAAa,CAAC,GAAG,IAAI,CAACC,eAAe,CAC5DX,QAAQ,EACR,GAAGC,IAAI,EACP,IAAI,CAACL,QAAQ,CAAC,CAChB,CAAC;MACD,IAAIc,YAAY,KAAKxC,YAAY,CAAC0C,OAAO,IAAIC,IAAI,CAACC,GAAG,CAAC,CAAC,IAAIL,SAAS,EAClE,OAAOD,IAAI;MACb,OAAO,IAAI,CAACT,KAAK,CAACC,QAAQ,EAAE,GAAGC,IAAI,CAAC;IACtC,CAAC;IAED;AACF;AACA;AACA;IAHE,KAIAc,UAAU,GAAG,CACXf,QAAW,EACX,GAAGC,IAAmD,KAEtDA,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,GACd,IAAI,CAACN,QAAQ,CACXf,gBAAgB,CAACoB,QAAQ,EAAE;MACzBC,IAAI,EAAEA;IACR,CAAC,CACH,CAAC,GACDe,OAAO,CAACC,OAAO,CAAC,CAAC;IAErB;AACF;AACA;AACA;AACA;IAJE,KAKAC,aAAa,GAAIC,OAA8C,IAC7D,IAAI,CAACxB,QAAQ,CAACd,mBAAmB,CAAEuC,GAAW,IAAKD,OAAO,CAACE,OAAO,CAACD,GAAG,CAAC,CAAC,CAAC;IAE3E;AACF;AACA;AACA;AACA;IAJE,KAKAE,SAAS,GAAIH,OAA8C,IACzD,IAAI,CAACxB,QAAQ,CAACjB,eAAe,CAAE0C,GAAW,IAAKD,OAAO,CAACE,OAAO,CAACD,GAAG,CAAC,CAAC,CAAC;IAEvE;AACF;AACA;AACA;IAHE,KAIAG,gBAAgB,GAAG,MAAqB,IAAI,CAAC5B,QAAQ,CAACb,WAAW,CAAC,CAAC,CAAC;IA6BpE;AACF;AACA;AACA;IAHE,KAIA0C,WAAW,GAAG,CAKZxB,QAAW,EACX,GAAGyB,IAAsC,KACvB;MAClB,MAAMC,QAAwB,GAAGD,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC;MACtD,MAAMrC,MAAM,GAAGN,iBAAiB,CAACgB,QAAQ,EAAE;QACzCC,IAAI,EAAEwB,IAAI,CAACG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAkB;QACrDD;MACF,CAAC,CAAC;MACF,OAAO,IAAI,CAAC/B,QAAQ,CAACL,MAAM,CAAC;IAC9B,CAAC;IAED;AACF;AACA;AACA;IAHE,KAIAuC,QAAQ,GAAG,CAKT7B,QAAW,EACX,GAAGyB,IAAwC,KACzB;MAClB,MAAMC,QAAe,GAAGD,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC;MAC7C,MAAMrC,MAAM,GAAGN,iBAAiB,CAACgB,QAAQ,EAAE;QACzCC,IAAI,EAAEwB,IAAI,CAACG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAkB;QACrDD,QAAQ;QACRI,KAAK,EAAE;MACT,CAAC,CAAC;MACF,OAAO,IAAI,CAACnC,QAAQ,CAACL,MAAM,CAAC;IAC9B,CAAC;IAED;AACF;AACA;AACA;IAHE,KAIA2B,OAAO,GAAG,CAKRjB,QAAW,EACXG,IAYK,KACa;MAClB,OAAO,IAAI,CAACR,QAAQ,CAACX,iBAAiB,CAACgB,QAAQ,EAAEG,IAAW,CAAC,CAAC;IAChE,CAAC;IAED;AACF;AACA;AACA;IAHE,KAIA4B,SAAS,GAAG,CAOV/B,QAAW,EACX,GAAGC,IAAmD,KAEtDA,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,GACd,IAAI,CAACN,QAAQ,CACXlB,kBAAkB,CAACuB,QAAQ,EAAE;MAC3BC,IAAI,EAAEA;IACR,CAAC,CACH,CAAC,GACDe,OAAO,CAACC,OAAO,CAAC,CAAC;IAErB;AACF;AACA;AACA;IAHE,KAIAe,WAAW,GAAG,CAOZhC,QAAW,EACX,GAAGC,IAAmD,KAEtDA,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,GACd,IAAI,CAACN,QAAQ,CACXnB,oBAAoB,CAACwB,QAAQ,EAAE;MAC7BC,IAAI,EAAEA;IACR,CAAC,CACH,CAAC,GACDe,OAAO,CAACC,OAAO,CAAC,CAAC;IAErB;IAEA;AACF;AACA;AACA;AACA;AACA;IAEE;AACF;AACA;AACA;IAHE,KAIAgB,QAAQ,GAAG,CAACC,KAAqB,EAAEC,SAAkB,KAAwB;MAC3E,OAAO,IAAIC,QAAQ,CAAC,IAAI,EAAEF,KAAK,EAAEC,SAAS,CAAC;IAC7C,CAAC;IA3QC,IAAI,CAACE,SAAS,GAAG1C,QAAQ;IACzB,IAAI,CAACC,QAAQ,GAAGA,QAAQ;IACxB,IAAI,CAACC,IAAI,GAAGA,IAAI;IAChB,IAAI,CAACC,QAAQ,GAAGA,QAAQ;EAC1B;;EAEA;EACA,IAAIH,QAAQA,CAACA,QAAW,EAAE;IACxB;IACA,IAAI,CAAC0C,SAAS,GAAG1C,QAAQ;EAC3B;;EAEA;EACA,IAAIA,QAAQA,CAAA,EAAM;IAChB,OAAO,IAAI,CAAC0C,SAAS;EACvB;EAEAC,cAAcA,CAAC;IACb3C,QAAQ;IACRC;EAIF,CAAC,EAAE;IACD,IAAI,CAACyC,SAAS,GAAG1C,QAAQ;IACzB,IAAI,CAACC,QAAQ,GAAGA,QAAQ;EAC1B;EAuFA;AACF;AACA;AACA;EAWE2C,GAAGA,CACDrC,MAAS,EACT,GAAGuB,IAAsC,EAC1B;IACf,MAAMe,KAAK,GAAGf,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC;IACnC,MAAMrC,MAAM,GAAGP,SAAS,CAACmB,MAAM,EAAE;MAC/BD,IAAI,EAAEwB,IAAI,CAACG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAkB;MACrDa;IACF,CAAC,CAAC;IACF;IACA,OAAO,IAAI,CAAC7C,QAAQ,CAACL,MAAM,CAAC;EAC9B;EAmIA;AACF;AACA;AACA;EAeEmD,QAAQA,CACNzC,QAA2B,EAC3B,GAAGyB,IAA6C,EACxB;IACxB,IAAIA,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE;IACtB,MAAMS,KAAK,GAAGT,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAmB;IACrD;IACA,MAAM1B,IAAS,GAAGwB,IAAI,CAACG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC;IAChD,MAAMP,GAAG,GAAGpB,QAAQ,CAACoB,GAAG,CAAC,GAAGnB,IAAI,CAAC;IAEjC,MAAME,IAAI,GAAGf,UAAU,CAAC8C,KAAK,EAAEd,GAAG,CAAC;IACnC,MAAMU,KAAK,GAAGI,KAAK,CAACQ,SAAS,CAACtB,GAAG,CAAC;IAElC,IAAIU,KAAK,KAAKa,SAAS,IAAI,CAAAxC,IAAI,oBAAJA,IAAI,CAAEyC,WAAW,MAAK,MAAM,EAAE;IAEzD,OAAOzC,IAAI,oBAAJA,IAAI,CAAE2B,KAAK;EACpB;;EAEA;AACF;AACA;AACA;;EA4BEe,WAAWA,CACT7C,QAA2B,EAC3B,GAAGyB,IAA6C,EAMhD;IACA;IACA,OAAO,IAAI,CAACd,eAAe,CAACX,QAAQ,EAAE,GAAGyB,IAAI,CAAC;EAChD;;EAEA;AACF;AACA;AACA;;EA4BEd,eAAeA,CACbX,QAA2B,EAC3B,GAAGyB,IAA6C,EAMhD;IACA,MAAMS,KAAK,GAAGT,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAmB;IACrD;IACA,MAAM1B,IAAS,GAAGwB,IAAI,CACnBG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC;IACzB;IAAA,CACCmB,GAAG,CAAC7D,UAAU,CAAC;IAClB,MAAM8D,QAAQ,GAAG9C,IAAI,CAAC0B,MAAM,KAAK,CAAC,IAAI1B,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI;IACtD,MAAMmB,GAAG,GAAG2B,QAAQ,GAAG/C,QAAQ,CAACoB,GAAG,CAAC,GAAGnB,IAAI,CAAC,GAAG,EAAE;IACjD,MAAM+C,cAAc,GAAGD,QAAQ,GAAGb,KAAK,CAACQ,SAAS,CAACtB,GAAG,CAAC,GAAGuB,SAAS;IAClE,MAAMzC,MAAM,GAAGF,QAAQ,CAACE,MAAM;IAC9B,MAAMC,IAAI,GAAGf,UAAU,CAAC8C,KAAK,EAAEd,GAAG,CAAC;IACnC,IAAIX,SAAS,GAAGN,IAAI,oBAAJA,IAAI,CAAEM,SAAS;IAC/B;IACA,MAAMwC,WAAW,GAAGD,cAAc,KAAKL,SAAS,IAAIzC,MAAM,KAAKyC,SAAS;IAExE,MAAMrC,KAAK,GACT2C,WAAW;IACT;IACA,IAAI,CAACpD,IAAI,CAACqD,aAAa,CAAChD,MAAM,EAAED,IAAI,EAAEiC,KAAK,EAAEd,GAAG,CAAC,GACjD4B,cAAc;IAElB,IAAI,CAACD,QAAQ,EAAE;MACb;MACA,OAAO;QACLvC,IAAI,EAAEF,KAAY;QAClBI,YAAY,EAAExC,YAAY,CAACiF,KAAK;QAChC1C,SAAS,EAAE2C,QAAQ;QACnBC,QAAQ,EAAEA,CAAA,KAAM,MAAMV;MACxB,CAAC;IACH;IAEA,IAAIW,SAAS,GAAG,KAAK;IACrB,IAAIL,WAAW,EAAE;MACfK,SAAS,GAAG,CAAChF,gBAAgB,CAACgC,KAAK,CAAC;MACpC;IACF,CAAC,MAAM,IAAI,CAACJ,MAAM,IAAI,CAACqD,eAAe,CAACrD,MAAM,CAAC,EAAE;MAC9C,OAAO;QACLM,IAAI,EAAEwC,cAAc;QACpBtC,YAAY,EAAE,IAAI,CAAC8C,eAAe,CAChC,CAACR,cAAc,EACf,CAAC,CAAChD,QAAQ,CAACyD,cAAc,EACzBtD,IACF,CAAC;QACDM,SAAS,EAAEA,SAAS,IAAI,CAAC;QACzB4C,QAAQ,EAAE,IAAI,CAACvD,QAAQ,CAAC4D,cAAc,CAAC;UAAEtC;QAAI,CAAC;MAChD,CAAC;IACH;IAEA,MAAM;MAAEZ,IAAI;MAAEmD;IAAM,CAAC,GAAG,IAAI,CAAC9D,IAAI,CAACxB,WAAW,CAC3C6B,MAAM,EACNI,KAAK,EACL4B,KAAK,CAAC0B,QAAQ,EACd3D,IACF,CAAuC;IAEvC,IAAI,CAACQ,SAAS,EAAE;MACd;MACA,IAAI6C,SAAS,EAAE7C,SAAS,GAAG,CAAC;MAC5B;MAAA,KACKA,SAAS,GAAGoD,eAAe,CAACF,KAAK,EAAEzB,KAAK,CAAC4B,YAAY,CAAC;IAC7D;IAEA,OAAO;MACLtD,IAAI;MACJE,YAAY,EAAE,IAAI,CAAC8C,eAAe,CAChC,OAAOhD,IAAI,KAAK,QAAQ,EACxB,CAAC,CAACR,QAAQ,CAACyD,cAAc,IAAIH,SAAS,EACtCnD,IACF,CAAC;MACDM,SAAS;MACT4C,QAAQ,EAAE,IAAI,CAACvD,QAAQ,CAAC4D,cAAc,CAAC;QAAEtC,GAAG;QAAEuC;MAAM,CAAC;IACvD,CAAC;EACH;;EAEA;AACF;AACA;AACA;EACEI,GAAGA,CACD7D,MAAS,EACT,GAAGuB,IAGF,EACmC;IACpC,MAAMS,KAAK,GAAGT,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAe;IACjD;IACA,MAAM1B,IAAS,GAAGwB,IAAI,CACnBG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC,CACzBmB,GAAG,CAAC7D,UAAU,CAAkB;IAEnC,MAAM;MAAEuB;IAAK,CAAC,GAAG,IAAI,CAACX,IAAI,CAACmE,KAAK,CAAC9D,MAAM,EAAED,IAAI,EAAEiC,KAAK,CAAC;IACrD,OAAO,OAAO1B,IAAI,KAAK,QAAQ,GAAGmC,SAAS,GAAGnC,IAAI;EACpD;;EAEA;AACF;AACA;AACA;EACEyD,YAAYA,CACV/D,MAAS,EACT,GAAGuB,IAGF,EAID;IACA,MAAMS,KAAK,GAAGT,IAAI,CAACA,IAAI,CAACE,MAAM,GAAG,CAAC,CAAe;IACjD;IACA,MAAM1B,IAAS,GAAGwB,IAAI,CACnBG,KAAK,CAAC,CAAC,EAAEH,IAAI,CAACE,MAAM,GAAG,CAAC,CAAC,CACzBmB,GAAG,CAAC7D,UAAU,CAAkB;IAEnC,MAAM;MAAEuB,IAAI;MAAEmD;IAAM,CAAC,GAAG,IAAI,CAAC9D,IAAI,CAACmE,KAAK,CAAC9D,MAAM,EAAED,IAAI,EAAEiC,KAAK,CAAC;IAE5D,OAAO;MACL1B,IAAI,EAAE,OAAOA,IAAI,KAAK,QAAQ,GAAGmC,SAAS,GAAGnC,IAAI;MACjD6C,QAAQ,EAAE,IAAI,CAACvD,QAAQ,CAAC4D,cAAc,CAAC;QAAEC;MAAM,CAAC;IAClD,CAAC;EACH;EAEQH,eAAeA,CACrBU,WAAoB,EACpBT,cAAuB,EACvBtD,IAAgD,GAAG,CAAC,CAAC,EACrD;IACA;IACA;IACA;IACA,OACEA,IAAI,CAACgE,WAAW,IAAKD,WAAW,IAAI,CAAC/D,IAAI,CAAC2B,KAAM,GAAG5D,YAAY,CAAC0C,OAAO,GACrEsD,WAAW,IAAIT,cAAc,GAAGvF,YAAY,CAACkG,cAAc,GAC3DlG,YAAY,CAACiF,KAAK;EAExB;AACF;;AAEA;AACA;AACA,SAASU,eAAeA,CACtBF,KAAmB,EACnBG,YAQC,EACD;EACA,IAAIrD,SAAS,GAAG2C,QAAQ;EACxB,KAAK,MAAM;IAAEhC,GAAG;IAAEiD;EAAG,CAAC,IAAIV,KAAK,EAAE;IAAA,IAAAW,iBAAA;IAC/B,MAAMC,YAAY,IAAAD,iBAAA,GAAGR,YAAY,CAAC1C,GAAG,CAAC,cAAAkD,iBAAA,GAAjBA,iBAAA,CAAoBD,EAAE,CAAC,qBAAvBC,iBAAA,CAAyB7D,SAAS;IACvD;IACA,IAAI8D,YAAY,GAAG9D,SAAS,EAAEA,SAAS,GAAG8D,YAAY;EACxD;EACA,OAAO9D,SAAS;AAClB;;AAEA;AACA;AACA;AACA;AACA,SAAS8C,eAAeA,CAACrD,MAAc,EAAW;EAChD,IAAI9B,QAAQ,CAAC8B,MAAM,CAAC,EAAE,OAAO,IAAI;EACjC,IAAIsE,KAAK,CAACC,OAAO,CAACvE,MAAM,CAAC,EACvB,OAAOA,MAAM,CAACyB,MAAM,KAAK,CAAC,IAAI4B,eAAe,CAACrD,MAAM,CAAC,CAAC,CAAC,CAAC;EAC1D,IAAIA,MAAM,KAAK,OAAOA,MAAM,KAAK,QAAQ,IAAI,OAAOA,MAAM,KAAK,UAAU,CAAC,EAAE;IAC1E,MAAMwE,YAAY,GAChB,QAAQ,IAAIxE,MAAM,GAAIA,MAAM,CAACA,MAAM,GAA8BA,MAAM;IACzE,IAAI,OAAOwE,YAAY,KAAK,UAAU,EAAE;MACtC,OAAOnB,eAAe,CAACmB,YAAY,CAAC;IACtC;IACA,OAAOC,MAAM,CAACC,MAAM,CAACF,YAAY,CAAC,CAACG,IAAI,CAACC,CAAC,IAAIvB,eAAe,CAACuB,CAAC,CAAC,CAAC;EAClE;EACA,OAAO,KAAK;AACd;AAIA,MAAM1C,QAAQ,CAA2C;EAQvD1C,WAAWA,CAACqF,UAAsB,EAAE7C,KAAe,EAAEC,SAAS,GAAG,CAAC,EAAE;IAAA,KAL5DD,KAAK;IAAA,KACL6C,UAAU;IAAA,KACT5C,SAAS;IAAA,KACT6C,KAAK,GAAG5C,QAAQ,CAAC4C,KAAK;IAG7B,IAAI,CAAC9C,KAAK,GAAGA,KAAK;IAClB,IAAI,CAAC6C,UAAU,GAAGA,UAAU;IAC5B,IAAI,CAAC5C,SAAS,GAAGA,SAAS;EAC5B;;EAEA;EACA;;EA8BAU,WAAWA,CAGT7C,QAAW,EACX,GAAGC,IAA0D,EAK7D;IACA,OAAO,IAAI,CAAC8E,UAAU,CAAClC,WAAW,CAAC7C,QAAQ,EAAE,GAAGC,IAAI,EAAE,IAAI,CAACiC,KAAK,CAAC;EACnE;;EAEA;;EA8BAvB,eAAeA,CAGbX,QAAW,EACX,GAAGC,IAA0D,EAK7D;IACA,OAAO,IAAI,CAAC8E,UAAU,CAACpE,eAAe,CAACX,QAAQ,EAAE,GAAGC,IAAI,EAAE,IAAI,CAACiC,KAAK,CAAC;EACvE;;EAEA;;EAWAO,QAAQA,CACNzC,QAAW,EACX,GAAGC,IAA0D,EACrC;IACxB,OAAO,IAAI,CAAC8E,UAAU,CAACtC,QAAQ,CAACzC,QAAQ,EAAE,GAAGC,IAAI,EAAE,IAAI,CAACiC,KAAK,CAAC;EAChE;;EAEA;AACF;AACA;AACA;EACE6B,GAAGA,CACD7D,MAAS,EACT,GAAGD,IAAmB,EACc;IACpC,OAAO,IAAI,CAAC8E,UAAU,CAAChB,GAAG,CAAC7D,MAAM,EAAE,GAAGD,IAAI,EAAE,IAAI,CAACiC,KAAK,CAAC;EACzD;;EAEA;AACF;AACA;AACA;EACE+B,YAAYA,CACV/D,MAAS,EACT,GAAGD,IAAmB,EAItB;IACA,OAAO,IAAI,CAAC8E,UAAU,CAACd,YAAY,CAAC/D,MAAM,EAAE,GAAGD,IAAI,EAAE,IAAI,CAACiC,KAAK,CAAC;EAClE;AACF;AA/IME,QAAQ,CACI4C,KAAK,GAAG,IAAIzG,eAAe,CAAC,CAAC","ignoreList":[]}
|
|
405
|
+
function extractStateAndArgs(rest) {
|
|
406
|
+
const l = rest.length;
|
|
407
|
+
const args = new Array(l - 1);
|
|
408
|
+
for (let i = 0; i < l - 1; i++) {
|
|
409
|
+
// handle FormData
|
|
410
|
+
args[i] = ensurePojo(rest[i]);
|
|
411
|
+
}
|
|
412
|
+
// this is typescript generics breaking
|
|
413
|
+
return [rest[l - 1], args];
|
|
414
|
+
}
|
|
415
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ExpiryStatus","MemoCache","isEntity","denormalize","validateQueryKey","AbortOptimistic","createUnsubscription","createSubscription","createExpireAll","createFetch","createInvalidate","createInvalidateAll","createReset","createSet","createSetResponse","ensurePojo","ImmortalGCPolicy","initialState","selectMeta","unsetDispatch","action","Error","unsetState","Controller","constructor","dispatch","getState","memo","gcPolicy","fetch","endpoint","args","schema","meta","promise","then","input","fetchIfStale","data","expiresAt","expiryStatus","getResponseMeta","Invalid","Date","now","invalidate","Promise","resolve","invalidateAll","options","key","testKey","expireAll","resetEntireStore","setResponse","rest","response","length","slice","setError","error","subscribe","unsubscribe","snapshot","state","fetchedAt","Snapshot","_dispatch","bindMiddleware","set","value","getError","endpoints","undefined","errorPolicy","getResponse","extractStateAndArgs","isActive","cacheEndpoints","shouldQuery","buildQueryKey","Valid","Infinity","countRef","isInvalid","schemaHasEntity","getExpiryStatus","invalidIfStale","createCountRef","paths","entities","entityExpiresAt","entitiesMeta","get","query","getQueryMeta","invalidData","invalidated","InvalidIfStale","pk","_entitiesMeta$key","entityExpiry","Array","isArray","nestedSchema","Object","values","some","x","controller","abort","l","i"],"sources":["../../src/controller/Controller.ts"],"sourcesContent":["import type {\n  ErrorTypes,\n  SnapshotInterface,\n  Schema,\n  Denormalize,\n  Queryable,\n  SchemaArgs,\n} from '@data-client/normalizr';\nimport {\n  ExpiryStatus,\n  EndpointInterface,\n  FetchFunction,\n  ResolveType,\n  DenormalizeNullable,\n  EntityPath,\n  MemoCache,\n  isEntity,\n  denormalize,\n  validateQueryKey,\n} from '@data-client/normalizr';\n\nimport AbortOptimistic from './AbortOptimistic.js';\nimport {\n  createUnsubscription,\n  createSubscription,\n} from './actions/createSubscription.js';\nimport {\n  createExpireAll,\n  createFetch,\n  createInvalidate,\n  createInvalidateAll,\n  createReset,\n  createSet,\n  createSetResponse,\n} from './actions/index.js';\nimport ensurePojo from './ensurePojo.js';\nimport type { EndpointUpdateFunction } from './types.js';\nimport { ReduxMiddlewareAPI } from '../manager/applyManager.js';\nimport type { GCInterface } from '../state/GCPolicy.js';\nimport { ImmortalGCPolicy } from '../state/GCPolicy.js';\nimport { initialState } from '../state/reducer/createReducer.js';\nimport selectMeta from '../state/selectMeta.js';\nimport type { ActionTypes, State } from '../types.js';\n\nexport type GenericDispatch = (value: any) => Promise<void>;\nexport type DataClientDispatch = (value: ActionTypes) => Promise<void>;\n\nexport interface ControllerConstructorProps<\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  dispatch?: D;\n  getState?: () => State<unknown>;\n  memo?: Pick<MemoCache, 'denormalize' | 'query' | 'buildQueryKey'>;\n  gcPolicy?: GCInterface;\n}\n\nconst unsetDispatch = (action: unknown): Promise<void> => {\n  throw new Error(\n    `Dispatching while constructing your middleware is not allowed. ` +\n      `Other middleware would not be applied to this dispatch.`,\n  );\n};\nconst unsetState = (): State<unknown> => {\n  // This is only the value until it is set by the DataProvider\n  /* istanbul ignore next */\n  return initialState;\n};\n\n/**\n * Imperative control of Reactive Data Client store\n * @see https://dataclient.io/docs/api/Controller\n */\nexport default class Controller<\n  // NOTE: We template on entire dispatch, so we can be contravariant on ActionTypes\n  D extends GenericDispatch = DataClientDispatch,\n> {\n  /**\n   * Dispatches an action to Reactive Data Client reducer.\n   *\n   * @see https://dataclient.io/docs/api/Controller#dispatch\n   */\n  declare protected _dispatch: D;\n  /**\n   * Gets the latest state snapshot that is fully committed.\n   *\n   * This can be useful for imperative use-cases like event handlers.\n   * This should *not* be used to render; instead useSuspense() or useCache()\n   * @see https://dataclient.io/docs/api/Controller#getState\n   */\n  declare getState: () => State<unknown>;\n  /**\n   * Singleton to maintain referential equality between calls\n   */\n  declare readonly memo: Pick<\n    MemoCache,\n    'denormalize' | 'query' | 'buildQueryKey'\n  >;\n\n  /**\n   * Handles garbage collection\n   */\n  declare readonly gcPolicy: GCInterface;\n\n  constructor({\n    dispatch = unsetDispatch as any,\n    getState = unsetState,\n    memo = new MemoCache(),\n    gcPolicy = new ImmortalGCPolicy(),\n  }: ControllerConstructorProps<D> = {}) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n    this.memo = memo;\n    this.gcPolicy = gcPolicy;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  set dispatch(dispatch: D) {\n    /* istanbul ignore next */\n    this._dispatch = dispatch;\n  }\n\n  // TODO: drop when drop support for destructuring (0.14 and below)\n  get dispatch(): D {\n    return this._dispatch;\n  }\n\n  bindMiddleware({\n    dispatch,\n    getState,\n  }: {\n    dispatch: D;\n    getState: ReduxMiddlewareAPI['getState'];\n  }) {\n    this._dispatch = dispatch;\n    this.getState = getState;\n  }\n\n  /*************** Action Dispatchers ***************/\n\n  /**\n   * Fetches the endpoint with given args, updating the Reactive Data Client cache with the response or error upon completion.\n   * @see https://dataclient.io/docs/api/Controller#fetch\n   */\n  fetch = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E>\n  : Promise<Denormalize<E['schema']>> => {\n    const action = createFetch(endpoint, {\n      args,\n    });\n    this.dispatch(action);\n\n    if (endpoint.schema) {\n      return action.meta.promise.then(input =>\n        denormalize(endpoint.schema, input, {}, args),\n      ) as any;\n    }\n    return action.meta.promise as any;\n  };\n\n  /**\n   * Fetches only if endpoint is considered 'stale'; otherwise returns undefined\n   * @see https://dataclient.io/docs/api/Controller#fetchIfStale\n   */\n  fetchIfStale = <\n    E extends EndpointInterface & { update?: EndpointUpdateFunction<E> },\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): E['schema'] extends undefined | null ? ReturnType<E> | ResolveType<E>\n  : Promise<Denormalize<E['schema']>> | Denormalize<E['schema']> => {\n    const { data, expiresAt, expiryStatus } = this.getResponseMeta(\n      endpoint,\n      ...args,\n      this.getState(),\n    );\n    if (expiryStatus !== ExpiryStatus.Invalid && Date.now() <= expiresAt)\n      return data as any;\n    return this.fetch(endpoint, ...args);\n  };\n\n  /**\n   * Forces refetching and suspense on useSuspense with the same Endpoint and parameters.\n   * @see https://dataclient.io/docs/api/Controller#invalidate\n   */\n  invalidate = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createInvalidate(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Forces refetching and suspense on useSuspense on all matching endpoint result keys.\n   * @see https://dataclient.io/docs/api/Controller#invalidateAll\n   * @returns Promise that resolves when invalidation is commited.\n   */\n  invalidateAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createInvalidateAll((key: string) => options.testKey(key)));\n\n  /**\n   * Sets all matching endpoint result keys to be STALE.\n   * @see https://dataclient.io/docs/api/Controller#expireAll\n   * @returns Promise that resolves when expiry is commited. *NOT* fetch promise\n   */\n  expireAll = (options: { testKey: (key: string) => boolean }) =>\n    this.dispatch(createExpireAll((key: string) => options.testKey(key)));\n\n  /**\n   * Resets the entire Reactive Data Client cache. All inflight requests will not resolve.\n   * @see https://dataclient.io/docs/api/Controller#resetEntireStore\n   */\n  resetEntireStore = (): Promise<void> => this.dispatch(createReset());\n\n  /**\n   * Sets value for the Queryable and args.\n   * @see https://dataclient.io/docs/api/Controller#set\n   */\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, (previousValue: Denormalize<S>) => {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, {}]\n  ): Promise<void>;\n\n  set<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [...SchemaArgs<S>, any]\n  ): Promise<void> {\n    const value = rest[rest.length - 1];\n    const action = createSet(schema, {\n      args: rest.slice(0, rest.length - 1) as SchemaArgs<S>,\n      value,\n    });\n    // TODO: reject with error if this fails in reducer\n    return this.dispatch(action);\n  }\n\n  /**\n   * Sets response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setResponse\n   */\n  setResponse = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, any]\n  ): Promise<void> => {\n    const response: ResolveType<E> = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Sets an error response for the Endpoint and args.\n   * @see https://dataclient.io/docs/api/Controller#setError\n   */\n  setError = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    ...rest: readonly [...Parameters<E>, Error]\n  ): Promise<void> => {\n    const response: Error = rest[rest.length - 1];\n    const action = createSetResponse(endpoint, {\n      args: rest.slice(0, rest.length - 1) as Parameters<E>,\n      response,\n      error: true,\n    });\n    return this.dispatch(action);\n  };\n\n  /**\n   * Resolves an inflight fetch.\n   * @see https://dataclient.io/docs/api/Controller#resolve\n   */\n  resolve = <\n    E extends EndpointInterface & {\n      update?: EndpointUpdateFunction<E>;\n    },\n  >(\n    endpoint: E,\n    meta:\n      | {\n          args: readonly [...Parameters<E>];\n          response: Error;\n          fetchedAt: number;\n          error: true;\n        }\n      | {\n          args: readonly [...Parameters<E>];\n          response: any;\n          fetchedAt: number;\n          error?: false | undefined;\n        },\n  ): Promise<void> => {\n    return this.dispatch(createSetResponse(endpoint, meta as any));\n  };\n\n  /**\n   * Marks a new subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#subscribe\n   */\n  subscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createSubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /**\n   * Marks completion of subscription to a given Endpoint.\n   * @see https://dataclient.io/docs/api/Controller#unsubscribe\n   */\n  unsubscribe = <\n    E extends EndpointInterface<\n      FetchFunction,\n      Schema | undefined,\n      undefined | false\n    >,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): Promise<void> =>\n    args[0] !== null ?\n      this.dispatch(\n        createUnsubscription(endpoint, {\n          args: args as Parameters<E>,\n        }),\n      )\n    : Promise.resolve();\n\n  /*************** More ***************/\n\n  /* TODO:\n  abort = <E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): Promise<void>\n  */\n\n  /**\n   * Gets a snapshot (https://dataclient.io/docs/api/Snapshot)\n   * @see https://dataclient.io/docs/api/Controller#snapshot\n   */\n  snapshot = (state: State<unknown>, fetchedAt?: number): Snapshot<unknown> => {\n    return new Snapshot(this, state, fetchedAt);\n  };\n\n  /**\n   * Gets the error, if any, for a given endpoint. Returns undefined for no errors.\n   * @see https://dataclient.io/docs/api/Controller#getError\n   */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E['key']>, State<unknown>]\n  ): ErrorTypes | undefined;\n\n  getError(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): ErrorTypes | undefined {\n    if (rest[0] === null) return;\n    const state = rest[rest.length - 1] as State<unknown>;\n    // this is typescript generics breaking\n    const args: any = rest.slice(0, rest.length - 1);\n    const key = endpoint.key(...args);\n\n    const meta = selectMeta(state, key);\n    const error = state.endpoints[key];\n\n    if (error !== undefined && meta?.errorPolicy === 'soft') return;\n\n    return meta?.error as any;\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponse\n   */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponse(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    // TODO: breaking: only return data\n    return this.getResponseMeta(endpoint, ...rest);\n  }\n\n  /**\n   * Gets the (globally referentially stable) response for a given endpoint/args pair from state given.\n   * @see https://dataclient.io/docs/api/Controller#getResponseMeta\n   */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...rest:\n      | readonly [null, State<unknown>]\n      | readonly [...Parameters<E>, State<unknown>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...rest: readonly [\n      ...(readonly [...Parameters<E['key']>] | readonly [null]),\n      State<unknown>,\n    ]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  };\n\n  getResponseMeta(\n    endpoint: EndpointInterface,\n    ...rest: readonly [...unknown[], State<unknown>]\n  ): {\n    data: unknown;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n    countRef: () => () => void;\n  } {\n    const [state, args] = extractStateAndArgs(rest);\n    const isActive = args.length !== 1 || args[0] !== null;\n    const key = isActive ? endpoint.key(...args) : '';\n    const cacheEndpoints = isActive ? state.endpoints[key] : undefined;\n    const schema = endpoint.schema;\n    const meta = selectMeta(state, key);\n    let expiresAt = meta?.expiresAt;\n    // if we have no endpoint entry, and our endpoint has a schema - try querying the store\n    const shouldQuery = cacheEndpoints === undefined && schema !== undefined;\n\n    const input =\n      shouldQuery ?\n        // nothing in endpoints cache, so try querying if we have a schema to do so\n        this.memo.buildQueryKey(schema, args, state, key)\n      : cacheEndpoints;\n\n    if (!isActive) {\n      // when not active simply return the query input without denormalizing\n      return {\n        data: input as any,\n        expiryStatus: ExpiryStatus.Valid,\n        expiresAt: Infinity,\n        countRef: () => () => undefined,\n      };\n    }\n\n    let isInvalid = false;\n    if (shouldQuery) {\n      isInvalid = !validateQueryKey(input);\n      // endpoint without entities\n    } else if (!schema || !schemaHasEntity(schema)) {\n      return {\n        data: cacheEndpoints,\n        expiryStatus: this.getExpiryStatus(\n          !cacheEndpoints,\n          !!endpoint.invalidIfStale,\n          meta,\n        ),\n        expiresAt: expiresAt || 0,\n        countRef: this.gcPolicy.createCountRef({ key }),\n      };\n    }\n\n    const { data, paths } = this.memo.denormalize(\n      schema,\n      input,\n      state.entities,\n      args,\n    ) as { data: any; paths: EntityPath[] };\n\n    if (!expiresAt) {\n      // note: isInvalid can only be true if shouldQuery is true\n      if (isInvalid) expiresAt = 1;\n      // fallback to entity expiry time\n      else expiresAt = entityExpiresAt(paths, state.entitiesMeta);\n    }\n\n    return {\n      data,\n      expiryStatus: this.getExpiryStatus(\n        typeof data === 'symbol',\n        !!endpoint.invalidIfStale || isInvalid,\n        meta,\n      ),\n      expiresAt,\n      countRef: this.gcPolicy.createCountRef({ key, paths }),\n    };\n  }\n\n  /**\n   * Queries the store for a Querable schema\n   * @see https://dataclient.io/docs/api/Controller#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): DenormalizeNullable<S> | undefined {\n    const [state, args] = extractStateAndArgs(rest);\n\n    const { data } = this.memo.query(schema, args, state);\n    return typeof data === 'symbol' ? undefined : data;\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Controller#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...rest: readonly [\n      ...SchemaArgs<S>,\n      Pick<State<unknown>, 'entities' | 'indexes'>,\n    ]\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    const [state, args] = extractStateAndArgs(rest);\n\n    const { data, paths } = this.memo.query(schema, args, state);\n\n    return {\n      data: typeof data === 'symbol' ? undefined : data,\n      countRef: this.gcPolicy.createCountRef({ paths }),\n    };\n  }\n\n  private getExpiryStatus(\n    invalidData: boolean,\n    invalidIfStale: boolean,\n    meta: { error?: unknown; invalidated?: unknown } = {},\n  ) {\n    // https://dataclient.io/docs/concepts/expiry-policy#expiry-status\n    // we don't track the difference between stale or fresh because that is tied to triggering\n    // conditions\n    return (\n      meta.invalidated || (invalidData && !meta.error) ? ExpiryStatus.Invalid\n      : invalidData || invalidIfStale ? ExpiryStatus.InvalidIfStale\n      : ExpiryStatus.Valid\n    );\n  }\n}\n\n// benchmark: https://www.measurethat.net/Benchmarks/Show/24691/0/min-reducer-vs-imperative-with-paths\n// earliest expiry dictates age\nfunction entityExpiresAt(\n  paths: EntityPath[],\n  entitiesMeta: {\n    readonly [entityKey: string]: {\n      readonly [pk: string]: {\n        readonly date: number;\n        readonly expiresAt: number;\n        readonly fetchedAt: number; // This is only the value until it is set by the DataProvider\n      };\n    };\n  },\n) {\n  let expiresAt = Infinity;\n  for (const { key, pk } of paths) {\n    const entityExpiry = entitiesMeta[key]?.[pk]?.expiresAt;\n    // expiresAt will always resolve to false with any comparison\n    if (entityExpiry < expiresAt) expiresAt = entityExpiry;\n  }\n  return expiresAt;\n}\n\n/** Determine whether the schema has any entities.\n *\n * Without entities, denormalization is not needed, and results should not be queried.\n */\nfunction schemaHasEntity(schema: Schema): boolean {\n  if (isEntity(schema)) return true;\n  if (Array.isArray(schema))\n    return schema.length !== 0 && schemaHasEntity(schema[0]);\n  if (schema && (typeof schema === 'object' || typeof schema === 'function')) {\n    const nestedSchema =\n      'schema' in schema ? (schema.schema as Record<string, Schema>) : schema;\n    if (typeof nestedSchema === 'function') {\n      return schemaHasEntity(nestedSchema);\n    }\n    return Object.values(nestedSchema).some(x => schemaHasEntity(x));\n  }\n  return false;\n}\n\nexport type { ErrorTypes };\n\nclass Snapshot<T = unknown> implements SnapshotInterface {\n  static readonly abort = new AbortOptimistic();\n\n  private state: State<T>;\n  private controller: Controller;\n  readonly fetchedAt: number;\n  readonly abort = Snapshot.abort;\n\n  constructor(controller: Controller, state: State<T>, fetchedAt = 0) {\n    this.state = state;\n    this.controller = controller;\n    this.fetchedAt = fetchedAt;\n  }\n\n  /*************** Data Access ***************/\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponse */\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponse<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponse(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getResponseMeta */\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  };\n\n  getResponseMeta<\n    E extends Pick<EndpointInterface, 'key' | 'schema' | 'invalidIfStale'>,\n  >(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): {\n    data: DenormalizeNullable<E['schema']>;\n    expiryStatus: ExpiryStatus;\n    expiresAt: number;\n  } {\n    return this.controller.getResponseMeta(endpoint, ...args, this.state);\n  }\n\n  /** @see https://dataclient.io/docs/api/Snapshot#getError */\n  getError<E extends EndpointInterface>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined;\n\n  getError<E extends Pick<EndpointInterface, 'key'>>(\n    endpoint: E,\n    ...args: readonly [...Parameters<E['key']>] | readonly [null]\n  ): ErrorTypes | undefined {\n    return this.controller.getError(endpoint, ...args, this.state);\n  }\n\n  /**\n   * Retrieved memoized value for any Querable schema\n   * @see https://dataclient.io/docs/api/Snapshot#get\n   */\n  get<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): DenormalizeNullable<S> | undefined {\n    return this.controller.get(schema, ...args, this.state);\n  }\n\n  /**\n   * Queries the store for a Querable schema; providing related metadata\n   * @see https://dataclient.io/docs/api/Snapshot#getQueryMeta\n   */\n  getQueryMeta<S extends Queryable>(\n    schema: S,\n    ...args: SchemaArgs<S>\n  ): {\n    data: DenormalizeNullable<S> | undefined;\n    countRef: () => () => void;\n  } {\n    return this.controller.getQueryMeta(schema, ...args, this.state);\n  }\n}\n\n/** Extract state and args from rest params, applying ensurePojo to args */\nfunction extractStateAndArgs(rest: readonly unknown[]): [State<any>, any[]] {\n  const l = rest.length;\n  const args: any = new Array(l - 1);\n  for (let i = 0; i < l - 1; i++) {\n    // handle FormData\n    args[i] = ensurePojo(rest[i]);\n  }\n  // this is typescript generics breaking\n  return [rest[l - 1] as State<any>, args];\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"setResponseReducer.d.ts","sourceRoot":"","sources":["../../../src/state/reducer/setResponseReducer.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,UAAU,MAAM,gCAAgC,CAAC;AAC7D,OAAO,KAAK,EACV,KAAK,EACL,iBAAiB,EACjB,gBAAgB,EACjB,MAAM,gBAAgB,CAAC;AAExB,wBAAgB,kBAAkB,CAChC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,EACrB,MAAM,EAAE,gBAAgB,GAAG,iBAAiB,EAC5C,UAAU,EAAE,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
1
|
+
{"version":3,"file":"setResponseReducer.d.ts","sourceRoot":"","sources":["../../../src/state/reducer/setResponseReducer.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,UAAU,MAAM,gCAAgC,CAAC;AAC7D,OAAO,KAAK,EACV,KAAK,EACL,iBAAiB,EACjB,gBAAgB,EACjB,MAAM,gBAAgB,CAAC;AAExB,wBAAgB,kBAAkB,CAChC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,EACrB,MAAM,EAAE,gBAAgB,GAAG,iBAAiB,EAC5C,UAAU,EAAE,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA2FvB"}
|
|
@@ -46,7 +46,7 @@ export function setResponseReducer(state, action, controller) {
|
|
|
46
46
|
// no reason to completely fail because of user-code error
|
|
47
47
|
// integrity of this state update is still guaranteed
|
|
48
48
|
} catch (error) {
|
|
49
|
-
console.error(`
|
|
49
|
+
console.error(`Endpoint.update() error: ${action.key}`);
|
|
50
50
|
console.error(error);
|
|
51
51
|
}
|
|
52
52
|
return {
|
|
@@ -111,4 +111,4 @@ function reduceError(state, action, error) {
|
|
|
111
111
|
function filterOptimistic(state, resolvingAction) {
|
|
112
112
|
return state.optimistic.filter(optimisticAction => optimisticAction.key !== resolvingAction.key || (optimisticAction.type === OPTIMISTIC ? optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt : optimisticAction.meta.date > resolvingAction.meta.date));
|
|
113
113
|
}
|
|
114
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["normalize","OPTIMISTIC","AbortOptimistic","setResponseReducer","state","action","controller","error","reduceError","response","_state$meta$action$ke","type","endpoint","getOptimisticResponse","call","snapshot","meta","fetchedAt","args","e","constructor","result","entities","indexes","entitiesMeta","schema","endpoints","key","update","updaters","Object","keys","forEach","console","date","expiresAt","prevExpiresAt","optimistic","filterOptimistic","lastReset","message","JSON","stringify","undefined","status","process","env","NODE_ENV","name","errorPolicy","resolvingAction","filter","optimisticAction"],"sources":["../../../src/state/reducer/setResponseReducer.ts"],"sourcesContent":["import { normalize } from '@data-client/normalizr';\n\nimport { OPTIMISTIC } from '../../actionTypes.js';\nimport AbortOptimistic from '../../controller/AbortOptimistic.js';\nimport type Controller from '../../controller/Controller.js';\nimport type {\n  State,\n  SetResponseAction,\n  OptimisticAction,\n} from '../../types.js';\n\nexport function setResponseReducer(\n  state: State<unknown>,\n  action: OptimisticAction | SetResponseAction,\n  controller: Controller,\n) {\n  if (action.error) {\n    return reduceError(state, action, action.response);\n  }\n  try {\n    let response: any;\n    // for true set's response is contained in action\n    if (action.type === OPTIMISTIC) {\n      // this should never happen\n      /* istanbul ignore if */\n      if (!action.endpoint.getOptimisticResponse) return state;\n      try {\n        // compute optimistic response based on current state\n        response = action.endpoint.getOptimisticResponse.call(\n          action.endpoint,\n          controller.snapshot(state, action.meta.fetchedAt),\n          ...action.args,\n        );\n      } catch (e: any) {\n        // AbortOptimistic means 'do nothing', otherwise we count the exception as endpoint failure\n        if (e.constructor === AbortOptimistic) {\n          return state;\n        }\n        throw e;\n      }\n    } else {\n      response = action.response;\n    }\n    const { result, entities, indexes, entitiesMeta } = normalize(\n      action.endpoint.schema,\n      response,\n      action.args,\n      state,\n      action.meta,\n    );\n    const endpoints: Record<string, unknown> = {\n      ...state.endpoints,\n      [action.key]: result,\n    };\n    try {\n      if (action.endpoint.update) {\n        const updaters = action.endpoint.update(result, ...action.args);\n        Object.keys(updaters).forEach(key => {\n          endpoints[key] = updaters[key](endpoints[key]);\n        });\n      }\n      // no reason to completely fail because of user-code error\n      // integrity of this state update is still guaranteed\n    } catch (error) {\n      console.error(\n        `The following error occured during Endpoint.update() for ${action.key}`,\n      );\n      console.error(error);\n    }\n    return {\n      entities,\n      endpoints,\n      indexes,\n      meta: {\n        ...state.meta,\n        [action.key]: {\n          date: action.meta.date,\n          fetchedAt: action.meta.fetchedAt,\n          expiresAt: action.meta.expiresAt,\n          prevExpiresAt: state.meta[action.key]?.expiresAt,\n        },\n      },\n      entitiesMeta,\n      optimistic: filterOptimistic(state, action),\n      lastReset: state.lastReset,\n    };\n    // reducer must update the state, so in case of processing errors we simply compute the endpoints inline\n  } catch (error: any) {\n    if (typeof error === 'object') {\n      error.message = `Error processing ${\n        action.key\n      }\\n\\nFull Schema: ${JSON.stringify(\n        action.endpoint.schema,\n        undefined,\n        2,\n      )}\\n\\nError:\\n${error.message}`;\n      if ('response' in action) error.response = action.response;\n      error.status = 400;\n    }\n\n    // this is not always bubbled up, so let's double sure this doesn't fail silently\n    /* istanbul ignore else */\n    if (process.env.NODE_ENV !== 'production') {\n      console.error(error);\n    }\n    return reduceError(state, action, error);\n  }\n}\n\nfunction reduceError(\n  state: State<unknown>,\n  action: SetResponseAction | OptimisticAction,\n  error: any,\n): State<unknown> {\n  if (error.name === 'AbortError') {\n    // In case we abort simply undo the optimistic update and act like no fetch even occured\n    // We still want those watching promises from fetch directly to observed the abort, but we don't want to\n    // Trigger errors in this case. This means theoretically improperly built aborts useSuspense() could suspend forever.\n    return {\n      ...state,\n      optimistic: filterOptimistic(state, action),\n    };\n  }\n  return {\n    ...state,\n    meta: {\n      ...state.meta,\n      [action.key]: {\n        date: action.meta.date,\n        fetchedAt: action.meta.fetchedAt,\n        expiresAt: action.meta.expiresAt,\n        error,\n        errorPolicy: action.endpoint.errorPolicy?.(error),\n      },\n    },\n    optimistic: filterOptimistic(state, action),\n  };\n}\n/** Filter all requests with same serialization that did not start after the resolving request */\nfunction filterOptimistic(\n  state: State<unknown>,\n  resolvingAction: SetResponseAction | OptimisticAction,\n) {\n  return state.optimistic.filter(\n    optimisticAction =>\n      optimisticAction.key !== resolvingAction.key ||\n      (optimisticAction.type === OPTIMISTIC ?\n        optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt\n      : optimisticAction.meta.date > resolvingAction.meta.date),\n  );\n}\n"],"mappings":"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","ignoreList":[]}
|
|
114
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["normalize","OPTIMISTIC","AbortOptimistic","setResponseReducer","state","action","controller","error","reduceError","response","_state$meta$action$ke","type","endpoint","getOptimisticResponse","call","snapshot","meta","fetchedAt","args","e","constructor","result","entities","indexes","entitiesMeta","schema","endpoints","key","update","updaters","Object","keys","forEach","console","date","expiresAt","prevExpiresAt","optimistic","filterOptimistic","lastReset","message","JSON","stringify","undefined","status","process","env","NODE_ENV","name","errorPolicy","resolvingAction","filter","optimisticAction"],"sources":["../../../src/state/reducer/setResponseReducer.ts"],"sourcesContent":["import { normalize } from '@data-client/normalizr';\n\nimport { OPTIMISTIC } from '../../actionTypes.js';\nimport AbortOptimistic from '../../controller/AbortOptimistic.js';\nimport type Controller from '../../controller/Controller.js';\nimport type {\n  State,\n  SetResponseAction,\n  OptimisticAction,\n} from '../../types.js';\n\nexport function setResponseReducer(\n  state: State<unknown>,\n  action: OptimisticAction | SetResponseAction,\n  controller: Controller,\n) {\n  if (action.error) {\n    return reduceError(state, action, action.response);\n  }\n  try {\n    let response: any;\n    // for true set's response is contained in action\n    if (action.type === OPTIMISTIC) {\n      // this should never happen\n      /* istanbul ignore if */\n      if (!action.endpoint.getOptimisticResponse) return state;\n      try {\n        // compute optimistic response based on current state\n        response = action.endpoint.getOptimisticResponse.call(\n          action.endpoint,\n          controller.snapshot(state, action.meta.fetchedAt),\n          ...action.args,\n        );\n      } catch (e: any) {\n        // AbortOptimistic means 'do nothing', otherwise we count the exception as endpoint failure\n        if (e.constructor === AbortOptimistic) {\n          return state;\n        }\n        throw e;\n      }\n    } else {\n      response = action.response;\n    }\n    const { result, entities, indexes, entitiesMeta } = normalize(\n      action.endpoint.schema,\n      response,\n      action.args,\n      state,\n      action.meta,\n    );\n    const endpoints: Record<string, unknown> = {\n      ...state.endpoints,\n      [action.key]: result,\n    };\n    try {\n      if (action.endpoint.update) {\n        const updaters = action.endpoint.update(result, ...action.args);\n        Object.keys(updaters).forEach(key => {\n          endpoints[key] = updaters[key](endpoints[key]);\n        });\n      }\n      // no reason to completely fail because of user-code error\n      // integrity of this state update is still guaranteed\n    } catch (error) {\n      console.error(`Endpoint.update() error: ${action.key}`);\n      console.error(error);\n    }\n    return {\n      entities,\n      endpoints,\n      indexes,\n      meta: {\n        ...state.meta,\n        [action.key]: {\n          date: action.meta.date,\n          fetchedAt: action.meta.fetchedAt,\n          expiresAt: action.meta.expiresAt,\n          prevExpiresAt: state.meta[action.key]?.expiresAt,\n        },\n      },\n      entitiesMeta,\n      optimistic: filterOptimistic(state, action),\n      lastReset: state.lastReset,\n    };\n    // reducer must update the state, so in case of processing errors we simply compute the endpoints inline\n  } catch (error: any) {\n    if (typeof error === 'object') {\n      error.message = `Error processing ${\n        action.key\n      }\\n\\nFull Schema: ${JSON.stringify(\n        action.endpoint.schema,\n        undefined,\n        2,\n      )}\\n\\nError:\\n${error.message}`;\n      if ('response' in action) error.response = action.response;\n      error.status = 400;\n    }\n\n    // this is not always bubbled up, so let's double sure this doesn't fail silently\n    /* istanbul ignore else */\n    if (process.env.NODE_ENV !== 'production') {\n      console.error(error);\n    }\n    return reduceError(state, action, error);\n  }\n}\n\nfunction reduceError(\n  state: State<unknown>,\n  action: SetResponseAction | OptimisticAction,\n  error: any,\n): State<unknown> {\n  if (error.name === 'AbortError') {\n    // In case we abort simply undo the optimistic update and act like no fetch even occured\n    // We still want those watching promises from fetch directly to observed the abort, but we don't want to\n    // Trigger errors in this case. This means theoretically improperly built aborts useSuspense() could suspend forever.\n    return {\n      ...state,\n      optimistic: filterOptimistic(state, action),\n    };\n  }\n  return {\n    ...state,\n    meta: {\n      ...state.meta,\n      [action.key]: {\n        date: action.meta.date,\n        fetchedAt: action.meta.fetchedAt,\n        expiresAt: action.meta.expiresAt,\n        error,\n        errorPolicy: action.endpoint.errorPolicy?.(error),\n      },\n    },\n    optimistic: filterOptimistic(state, action),\n  };\n}\n/** Filter all requests with same serialization that did not start after the resolving request */\nfunction filterOptimistic(\n  state: State<unknown>,\n  resolvingAction: SetResponseAction | OptimisticAction,\n) {\n  return state.optimistic.filter(\n    optimisticAction =>\n      optimisticAction.key !== resolvingAction.key ||\n      (optimisticAction.type === OPTIMISTIC ?\n        optimisticAction.meta.fetchedAt !== resolvingAction.meta.fetchedAt\n      : optimisticAction.meta.date > resolvingAction.meta.date),\n  );\n}\n"],"mappings":"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","ignoreList":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@data-client/core",
|
|
3
|
-
"version": "0.15.
|
|
3
|
+
"version": "0.15.4",
|
|
4
4
|
"description": "Async State Management without the Management. REST, GraphQL, SSE, Websockets, Fetch",
|
|
5
5
|
"sideEffects": false,
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -134,12 +134,12 @@
|
|
|
134
134
|
},
|
|
135
135
|
"dependencies": {
|
|
136
136
|
"@babel/runtime": "^7.20.0",
|
|
137
|
-
"@data-client/normalizr": "^0.15.
|
|
137
|
+
"@data-client/normalizr": "^0.15.4",
|
|
138
138
|
"flux-standard-action": "^2.1.1"
|
|
139
139
|
},
|
|
140
140
|
"devDependencies": {
|
|
141
141
|
"@anansi/browserslist-config": "^1.4.2",
|
|
142
|
-
"@data-client/endpoint": "0.15.
|
|
142
|
+
"@data-client/endpoint": "0.15.4",
|
|
143
143
|
"@types/jest": "30.0.0",
|
|
144
144
|
"@types/node": "^24.0.0",
|
|
145
145
|
"rollup-plugins": "1.0.0"
|
|
@@ -40,7 +40,7 @@ import type { GCInterface } from '../state/GCPolicy.js';
|
|
|
40
40
|
import { ImmortalGCPolicy } from '../state/GCPolicy.js';
|
|
41
41
|
import { initialState } from '../state/reducer/createReducer.js';
|
|
42
42
|
import selectMeta from '../state/selectMeta.js';
|
|
43
|
-
import type { ActionTypes,
|
|
43
|
+
import type { ActionTypes, State } from '../types.js';
|
|
44
44
|
|
|
45
45
|
export type GenericDispatch = (value: any) => Promise<void>;
|
|
46
46
|
export type DataClientDispatch = (value: ActionTypes) => Promise<void>;
|
|
@@ -496,12 +496,7 @@ export default class Controller<
|
|
|
496
496
|
expiresAt: number;
|
|
497
497
|
countRef: () => () => void;
|
|
498
498
|
} {
|
|
499
|
-
const state = rest
|
|
500
|
-
// this is typescript generics breaking
|
|
501
|
-
const args: any = rest
|
|
502
|
-
.slice(0, rest.length - 1)
|
|
503
|
-
// handle FormData
|
|
504
|
-
.map(ensurePojo);
|
|
499
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
505
500
|
const isActive = args.length !== 1 || args[0] !== null;
|
|
506
501
|
const key = isActive ? endpoint.key(...args) : '';
|
|
507
502
|
const cacheEndpoints = isActive ? state.endpoints[key] : undefined;
|
|
@@ -581,11 +576,7 @@ export default class Controller<
|
|
|
581
576
|
Pick<State<unknown>, 'entities' | 'indexes'>,
|
|
582
577
|
]
|
|
583
578
|
): DenormalizeNullable<S> | undefined {
|
|
584
|
-
const state = rest
|
|
585
|
-
// this is typescript generics breaking
|
|
586
|
-
const args: any = rest
|
|
587
|
-
.slice(0, rest.length - 1)
|
|
588
|
-
.map(ensurePojo) as SchemaArgs<S>;
|
|
579
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
589
580
|
|
|
590
581
|
const { data } = this.memo.query(schema, args, state);
|
|
591
582
|
return typeof data === 'symbol' ? undefined : data;
|
|
@@ -605,11 +596,7 @@ export default class Controller<
|
|
|
605
596
|
data: DenormalizeNullable<S> | undefined;
|
|
606
597
|
countRef: () => () => void;
|
|
607
598
|
} {
|
|
608
|
-
const state = rest
|
|
609
|
-
// this is typescript generics breaking
|
|
610
|
-
const args: any = rest
|
|
611
|
-
.slice(0, rest.length - 1)
|
|
612
|
-
.map(ensurePojo) as SchemaArgs<S>;
|
|
599
|
+
const [state, args] = extractStateAndArgs(rest);
|
|
613
600
|
|
|
614
601
|
const { data, paths } = this.memo.query(schema, args, state);
|
|
615
602
|
|
|
@@ -823,3 +810,15 @@ class Snapshot<T = unknown> implements SnapshotInterface {
|
|
|
823
810
|
return this.controller.getQueryMeta(schema, ...args, this.state);
|
|
824
811
|
}
|
|
825
812
|
}
|
|
813
|
+
|
|
814
|
+
/** Extract state and args from rest params, applying ensurePojo to args */
|
|
815
|
+
function extractStateAndArgs(rest: readonly unknown[]): [State<any>, any[]] {
|
|
816
|
+
const l = rest.length;
|
|
817
|
+
const args: any = new Array(l - 1);
|
|
818
|
+
for (let i = 0; i < l - 1; i++) {
|
|
819
|
+
// handle FormData
|
|
820
|
+
args[i] = ensurePojo(rest[i]);
|
|
821
|
+
}
|
|
822
|
+
// this is typescript generics breaking
|
|
823
|
+
return [rest[l - 1] as State<any>, args];
|
|
824
|
+
}
|
|
@@ -62,9 +62,7 @@ export function setResponseReducer(
|
|
|
62
62
|
// no reason to completely fail because of user-code error
|
|
63
63
|
// integrity of this state update is still guaranteed
|
|
64
64
|
} catch (error) {
|
|
65
|
-
console.error(
|
|
66
|
-
`The following error occured during Endpoint.update() for ${action.key}`,
|
|
67
|
-
);
|
|
65
|
+
console.error(`Endpoint.update() error: ${action.key}`);
|
|
68
66
|
console.error(error);
|
|
69
67
|
}
|
|
70
68
|
return {
|