@data-client/core 0.15.4 → 0.16.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/CHANGELOG.md +39 -0
  2. package/dist/index.js +28 -11
  3. package/dist/index.umd.min.js +1 -1
  4. package/dist/mock.js +24 -11
  5. package/legacy/controller/actions/createOptimistic.js +3 -2
  6. package/legacy/manager/ConnectionListener.js +1 -1
  7. package/legacy/manager/DevtoolsManager.js +18 -4
  8. package/legacy/manager/NetworkManager.js +2 -2
  9. package/legacy/state/GCPolicy.js +4 -1
  10. package/legacy/state/reducer/setReducer.js +1 -1
  11. package/legacy/state/reducer/setResponseReducer.js +1 -1
  12. package/lib/controller/actions/createOptimistic.d.ts.map +1 -1
  13. package/lib/controller/actions/createOptimistic.js +3 -2
  14. package/lib/manager/ConnectionListener.d.ts +12 -0
  15. package/lib/manager/ConnectionListener.d.ts.map +1 -1
  16. package/lib/manager/ConnectionListener.js +1 -1
  17. package/lib/manager/DevtoolsManager.d.ts +1 -0
  18. package/lib/manager/DevtoolsManager.d.ts.map +1 -1
  19. package/lib/manager/DevtoolsManager.js +19 -5
  20. package/lib/manager/NetworkManager.d.ts +1 -1
  21. package/lib/manager/NetworkManager.js +2 -2
  22. package/lib/state/GCPolicy.d.ts.map +1 -1
  23. package/lib/state/GCPolicy.js +4 -1
  24. package/lib/state/reducer/expireReducer.d.ts +2 -2
  25. package/lib/state/reducer/invalidateReducer.d.ts +2 -2
  26. package/lib/state/reducer/setReducer.d.ts +1 -27
  27. package/lib/state/reducer/setReducer.d.ts.map +1 -1
  28. package/lib/state/reducer/setReducer.js +1 -1
  29. package/lib/state/reducer/setResponseReducer.d.ts +1 -30
  30. package/lib/state/reducer/setResponseReducer.d.ts.map +1 -1
  31. package/lib/state/reducer/setResponseReducer.js +1 -1
  32. package/package.json +3 -3
  33. package/src/controller/__tests__/get.ts +8 -8
  34. package/src/controller/actions/createOptimistic.ts +2 -1
  35. package/src/manager/ConnectionListener.ts +12 -0
  36. package/src/manager/DevtoolsManager.ts +21 -6
  37. package/src/manager/NetworkManager.ts +1 -1
  38. package/src/state/GCPolicy.ts +3 -0
  39. package/src/state/__tests__/GCPolicy.test.ts +2 -0
  40. package/src/state/reducer/setReducer.ts +1 -1
  41. package/src/state/reducer/setResponseReducer.ts +1 -1
  42. package/ts3.4/manager/ConnectionListener.d.ts +12 -0
  43. package/ts3.4/manager/DevtoolsManager.d.ts +1 -0
  44. package/ts3.4/manager/NetworkManager.d.ts +1 -1
  45. package/ts3.4/state/reducer/expireReducer.d.ts +2 -2
  46. package/ts3.4/state/reducer/invalidateReducer.d.ts +2 -2
  47. package/ts3.4/state/reducer/setReducer.d.ts +1 -27
  48. package/ts3.4/state/reducer/setResponseReducer.d.ts +1 -30
@@ -220,7 +220,7 @@ export default class NetworkManager {
220
220
  * create a new promise and call fetch.
221
221
  *
222
222
  * Note: The new promise is not actually tied to fetch at all,
223
- * but is resolved when the expected 'recieve' action is processed.
223
+ * but is resolved when the expected 'receive' action is processed.
224
224
  * This ensures promises are resolved only once their data is processed
225
225
  * by the reducer.
226
226
  */
@@ -264,4 +264,4 @@ function newFetchMeta(fetchedAt) {
264
264
  });
265
265
  return fetchMeta;
266
266
  }
267
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SET_RESPONSE","FETCH","RESET","createSetResponse","Controller","ResetError","Error","constructor","name","NetworkManager","dataExpiryLength","errorExpiryLength","fetching","Map","controller","middleware","next","action","type","handleFetch","endpoint","getOptimisticResponse","undefined","sideEffect","Promise","resolve","then","has","key","_controller$getState$","error","getState","meta","handleSet","args","response","fetchedAt","fetches","Array","from","values","clearAll","reject","init","cleanupDate","cleanup","Date","now","skipLogging","allSettled","size","map","promise","k","keys","clear","get","catch","delete","getLastReset","lastReset","throttle","deferedFetch","resolvePromise","data","process","env","NODE_ENV","isNaN","console","fetch","fetchMeta","newFetchMeta","set","idleCallback","timeout","callback","options"],"sources":["../../src/manager/NetworkManager.ts"],"sourcesContent":["import { SET_RESPONSE, FETCH, RESET } from '../actionTypes.js';\nimport { createSetResponse } from '../controller/actions/index.js';\nimport Controller from '../controller/Controller.js';\nimport type {\n  FetchAction,\n  Manager,\n  ActionTypes,\n  Middleware,\n  SetResponseAction,\n} from '../types.js';\n\nexport class ResetError extends Error {\n  name = 'ResetError';\n\n  constructor() {\n    super('Aborted due to RESET');\n  }\n}\n\nexport interface FetchingMeta {\n  promise: Promise<any>;\n  resolve: (value?: any) => void;\n  reject: (value?: any) => void;\n  fetchedAt: number;\n}\n\n/** Handles all async network dispatches\n *\n * Dedupes concurrent requests by keeping track of all fetches in flight\n * and returning existing promises for requests already in flight.\n *\n * Interfaces with store via a redux-compatible middleware.\n *\n * @see https://dataclient.io/docs/api/NetworkManager\n */\nexport default class NetworkManager implements Manager {\n  protected fetching: Map<string, FetchingMeta> = new Map();\n  declare readonly dataExpiryLength: number;\n  declare readonly errorExpiryLength: number;\n  protected controller: Controller = new Controller();\n  declare cleanupDate?: number;\n\n  constructor({ dataExpiryLength = 60000, errorExpiryLength = 1000 } = {}) {\n    this.dataExpiryLength = dataExpiryLength;\n    this.errorExpiryLength = errorExpiryLength;\n  }\n\n  middleware: Middleware = controller => {\n    this.controller = controller;\n    return next => action => {\n      switch (action.type) {\n        case FETCH:\n          this.handleFetch(action);\n          // This is the only case that causes any state change\n          // It's important to intercept other fetches as we don't want to trigger reducers during\n          // render - so we need to stop 'readonly' fetches which can be triggered in render\n          if (\n            action.endpoint.getOptimisticResponse !== undefined &&\n            action.endpoint.sideEffect\n          ) {\n            return next(action);\n          }\n          return Promise.resolve();\n        case SET_RESPONSE:\n          // only set after new state is computed\n          return next(action).then(() => {\n            if (this.fetching.has(action.key)) {\n              // Note: meta *must* be set by reducer so this should be safe\n              const error = controller.getState().meta[action.key]?.error;\n              // processing errors result in state meta having error, so we should reject the promise\n              if (error) {\n                this.handleSet(\n                  createSetResponse(action.endpoint, {\n                    args: action.args,\n                    response: error,\n                    fetchedAt: action.meta.fetchedAt,\n                    error: true,\n                  }),\n                );\n              } else {\n                this.handleSet(action);\n              }\n            }\n          });\n        case RESET: {\n          // take snapshot of rejectors at this point in time\n          // we must use Array.from since iteration does not freeze state at this point in time\n          const fetches = Array.from(this.fetching.values());\n\n          this.clearAll();\n          return next(action).then(() => {\n            // there could be external listeners to the promise\n            // this must happen after commit so our own rejector knows not to dispatch an error based on this\n            for (const { reject } of fetches) {\n              reject(new ResetError());\n            }\n          });\n        }\n        default:\n          return next(action);\n      }\n    };\n  };\n\n  /** On mount */\n  init() {\n    delete this.cleanupDate;\n  }\n\n  /** Ensures all promises are completed by rejecting remaining. */\n  cleanup() {\n    // ensure no dispatches after unmount\n    // this must be reversible (done in init) so useEffect() remains symmetric\n    this.cleanupDate = Date.now();\n  }\n\n  /** Used by DevtoolsManager to determine whether to log an action */\n  skipLogging(action: ActionTypes) {\n    /* istanbul ignore next */\n    return action.type === FETCH && this.fetching.has(action.key);\n  }\n\n  allSettled() {\n    if (this.fetching.size)\n      return Promise.allSettled(\n        this.fetching.values().map(({ promise }) => promise),\n      );\n  }\n\n  /** Clear all promise state */\n  protected clearAll() {\n    for (const k of this.fetching.keys()) {\n      this.clear(k);\n    }\n  }\n\n  /** Clear promise state for a given key */\n  protected clear(key: string) {\n    if (this.fetching.has(key)) {\n      (this.fetching.get(key) as FetchingMeta).promise.catch(() => {});\n      this.fetching.delete(key);\n    }\n  }\n\n  protected getLastReset() {\n    if (this.cleanupDate) return this.cleanupDate;\n    return this.controller.getState().lastReset;\n  }\n\n  /** Called when middleware intercepts 'rdc/fetch' action.\n   *\n   * Will then start a promise for a key and potentially start the network\n   * fetch.\n   *\n   * Uses throttle endpoints without sideEffects. This is valuable\n   * for ensures mutation requests always go through.\n   */\n  protected handleFetch(action: FetchAction) {\n    const { resolve, reject, fetchedAt } = action.meta;\n    const throttle = !action.endpoint.sideEffect;\n\n    const deferedFetch = () => {\n      let promise = action.endpoint(...action.args);\n      const resolvePromise = (\n        promise: Promise<string | number | object | null>,\n      ) =>\n        promise\n          .then(data => {\n            resolve(data);\n            return data;\n          })\n          .catch(error => {\n            reject(error);\n            throw error;\n          });\n      // schedule non-throttled resolutions in a microtask before set\n      // this enables users awaiting their fetch to trigger any react updates needed to deal\n      // with upcoming changes because of the fetch (for instance avoiding suspense if something is deleted)\n      if (!throttle) {\n        promise = resolvePromise(promise);\n      }\n      promise = promise\n        .then(response => {\n          let lastReset = this.getLastReset();\n\n          /* istanbul ignore else */\n          if (process.env.NODE_ENV !== 'production' && isNaN(lastReset)) {\n            console.error(\n              'state.lastReset is NaN. Only positive timestamps are valid.',\n            );\n            lastReset = 0;\n          }\n\n          // don't update state with promises started before last clear\n          if (fetchedAt >= lastReset) {\n            this.controller.resolve(action.endpoint, {\n              args: action.args,\n              response,\n              fetchedAt,\n            });\n          }\n          return response;\n        })\n        .catch(error => {\n          const lastReset = this.getLastReset();\n          // don't update state with promises started before last clear\n          if (fetchedAt >= lastReset) {\n            this.controller.resolve(action.endpoint, {\n              args: action.args,\n              response: error,\n              fetchedAt,\n              error: true,\n            });\n          }\n          throw error;\n        });\n      return promise;\n    };\n\n    if (throttle) {\n      return this.throttle(action.key, deferedFetch, fetchedAt)\n        .then(data => resolve(data))\n        .catch(error => reject(error));\n    } else {\n      return deferedFetch().catch(() => {});\n    }\n  }\n\n  /** Called when middleware intercepts a set action.\n   *\n   * Will resolve the promise associated with set key.\n   */\n  protected handleSet(action: SetResponseAction) {\n    // this can still turn out to be untrue since this is async\n    if (this.fetching.has(action.key)) {\n      const { reject, resolve } = this.fetching.get(action.key) as FetchingMeta;\n      if (action.error) {\n        reject(action.response);\n      } else {\n        resolve(action.response);\n      }\n\n      // since we're resolved we no longer need to keep track of this promise\n      this.clear(action.key);\n    }\n  }\n\n  /** Ensures only one request for a given key is in flight at any time\n   *\n   * Uses key to either retrieve in-flight promise, or if not\n   * create a new promise and call fetch.\n   *\n   * Note: The new promise is not actually tied to fetch at all,\n   * but is resolved when the expected 'recieve' action is processed.\n   * This ensures promises are resolved only once their data is processed\n   * by the reducer.\n   */\n  protected throttle(\n    key: string,\n    fetch: () => Promise<any>,\n    fetchedAt: number,\n  ): Promise<any> {\n    const lastReset = this.getLastReset();\n    let fetchMeta = this.fetching.get(key);\n\n    // we're already fetching so reuse the promise\n    // fetches after reset do not count\n    if (fetchMeta && fetchMeta.fetchedAt > lastReset) {\n      return fetchMeta.promise;\n    }\n\n    fetchMeta = newFetchMeta(fetchedAt);\n    this.fetching.set(key, fetchMeta);\n\n    this.idleCallback(\n      () => {\n        // since our real promise is resolved via the wrapReducer(),\n        // we should just stop all errors here.\n        // TODO: decouple this from useFetcher() (that's what's dispatching the error the resolves in here)\n        fetch().catch(() => null);\n      },\n      { timeout: 500 },\n    );\n\n    return fetchMeta.promise;\n  }\n\n  /** Calls the callback when client is not 'busy' with high priority interaction tasks\n   *\n   * Override for platform-specific implementations\n   */\n  protected idleCallback(\n    callback: (...args: any[]) => void,\n    options?: IdleRequestOptions,\n  ) {\n    callback();\n  }\n}\n\nfunction newFetchMeta(fetchedAt: number): FetchingMeta {\n  const fetchMeta = { fetchedAt } as FetchingMeta;\n  fetchMeta.promise = new Promise((resolve, reject) => {\n    fetchMeta.resolve = resolve;\n    fetchMeta.reject = reject;\n  });\n  return fetchMeta;\n}\n"],"mappings":"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","ignoreList":[]}
267
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SET_RESPONSE","FETCH","RESET","createSetResponse","Controller","ResetError","Error","constructor","name","NetworkManager","dataExpiryLength","errorExpiryLength","fetching","Map","controller","middleware","next","action","type","handleFetch","endpoint","getOptimisticResponse","undefined","sideEffect","Promise","resolve","then","has","key","_controller$getState$","error","getState","meta","handleSet","args","response","fetchedAt","fetches","Array","from","values","clearAll","reject","init","cleanupDate","cleanup","Date","now","skipLogging","allSettled","size","map","promise","k","keys","clear","get","catch","delete","getLastReset","lastReset","throttle","deferedFetch","resolvePromise","data","process","env","NODE_ENV","isNaN","console","fetch","fetchMeta","newFetchMeta","set","idleCallback","timeout","callback","options"],"sources":["../../src/manager/NetworkManager.ts"],"sourcesContent":["import { SET_RESPONSE, FETCH, RESET } from '../actionTypes.js';\nimport { createSetResponse } from '../controller/actions/index.js';\nimport Controller from '../controller/Controller.js';\nimport type {\n  FetchAction,\n  Manager,\n  ActionTypes,\n  Middleware,\n  SetResponseAction,\n} from '../types.js';\n\nexport class ResetError extends Error {\n  name = 'ResetError';\n\n  constructor() {\n    super('Aborted due to RESET');\n  }\n}\n\nexport interface FetchingMeta {\n  promise: Promise<any>;\n  resolve: (value?: any) => void;\n  reject: (value?: any) => void;\n  fetchedAt: number;\n}\n\n/** Handles all async network dispatches\n *\n * Dedupes concurrent requests by keeping track of all fetches in flight\n * and returning existing promises for requests already in flight.\n *\n * Interfaces with store via a redux-compatible middleware.\n *\n * @see https://dataclient.io/docs/api/NetworkManager\n */\nexport default class NetworkManager implements Manager {\n  protected fetching: Map<string, FetchingMeta> = new Map();\n  declare readonly dataExpiryLength: number;\n  declare readonly errorExpiryLength: number;\n  protected controller: Controller = new Controller();\n  declare cleanupDate?: number;\n\n  constructor({ dataExpiryLength = 60000, errorExpiryLength = 1000 } = {}) {\n    this.dataExpiryLength = dataExpiryLength;\n    this.errorExpiryLength = errorExpiryLength;\n  }\n\n  middleware: Middleware = controller => {\n    this.controller = controller;\n    return next => action => {\n      switch (action.type) {\n        case FETCH:\n          this.handleFetch(action);\n          // This is the only case that causes any state change\n          // It's important to intercept other fetches as we don't want to trigger reducers during\n          // render - so we need to stop 'readonly' fetches which can be triggered in render\n          if (\n            action.endpoint.getOptimisticResponse !== undefined &&\n            action.endpoint.sideEffect\n          ) {\n            return next(action);\n          }\n          return Promise.resolve();\n        case SET_RESPONSE:\n          // only set after new state is computed\n          return next(action).then(() => {\n            if (this.fetching.has(action.key)) {\n              // Note: meta *must* be set by reducer so this should be safe\n              const error = controller.getState().meta[action.key]?.error;\n              // processing errors result in state meta having error, so we should reject the promise\n              if (error) {\n                this.handleSet(\n                  createSetResponse(action.endpoint, {\n                    args: action.args,\n                    response: error,\n                    fetchedAt: action.meta.fetchedAt,\n                    error: true,\n                  }),\n                );\n              } else {\n                this.handleSet(action);\n              }\n            }\n          });\n        case RESET: {\n          // take snapshot of rejectors at this point in time\n          // we must use Array.from since iteration does not freeze state at this point in time\n          const fetches = Array.from(this.fetching.values());\n\n          this.clearAll();\n          return next(action).then(() => {\n            // there could be external listeners to the promise\n            // this must happen after commit so our own rejector knows not to dispatch an error based on this\n            for (const { reject } of fetches) {\n              reject(new ResetError());\n            }\n          });\n        }\n        default:\n          return next(action);\n      }\n    };\n  };\n\n  /** On mount */\n  init() {\n    delete this.cleanupDate;\n  }\n\n  /** Ensures all promises are completed by rejecting remaining. */\n  cleanup() {\n    // ensure no dispatches after unmount\n    // this must be reversible (done in init) so useEffect() remains symmetric\n    this.cleanupDate = Date.now();\n  }\n\n  /** Used by DevtoolsManager to determine whether to log an action */\n  skipLogging(action: ActionTypes) {\n    /* istanbul ignore next */\n    return action.type === FETCH && this.fetching.has(action.key);\n  }\n\n  allSettled() {\n    if (this.fetching.size)\n      return Promise.allSettled(\n        this.fetching.values().map(({ promise }) => promise),\n      );\n  }\n\n  /** Clear all promise state */\n  protected clearAll() {\n    for (const k of this.fetching.keys()) {\n      this.clear(k);\n    }\n  }\n\n  /** Clear promise state for a given key */\n  protected clear(key: string) {\n    if (this.fetching.has(key)) {\n      (this.fetching.get(key) as FetchingMeta).promise.catch(() => {});\n      this.fetching.delete(key);\n    }\n  }\n\n  protected getLastReset() {\n    if (this.cleanupDate) return this.cleanupDate;\n    return this.controller.getState().lastReset;\n  }\n\n  /** Called when middleware intercepts 'rdc/fetch' action.\n   *\n   * Will then start a promise for a key and potentially start the network\n   * fetch.\n   *\n   * Uses throttle endpoints without sideEffects. This is valuable\n   * for ensures mutation requests always go through.\n   */\n  protected handleFetch(action: FetchAction) {\n    const { resolve, reject, fetchedAt } = action.meta;\n    const throttle = !action.endpoint.sideEffect;\n\n    const deferedFetch = () => {\n      let promise = action.endpoint(...action.args);\n      const resolvePromise = (\n        promise: Promise<string | number | object | null>,\n      ) =>\n        promise\n          .then(data => {\n            resolve(data);\n            return data;\n          })\n          .catch(error => {\n            reject(error);\n            throw error;\n          });\n      // schedule non-throttled resolutions in a microtask before set\n      // this enables users awaiting their fetch to trigger any react updates needed to deal\n      // with upcoming changes because of the fetch (for instance avoiding suspense if something is deleted)\n      if (!throttle) {\n        promise = resolvePromise(promise);\n      }\n      promise = promise\n        .then(response => {\n          let lastReset = this.getLastReset();\n\n          /* istanbul ignore else */\n          if (process.env.NODE_ENV !== 'production' && isNaN(lastReset)) {\n            console.error(\n              'state.lastReset is NaN. Only positive timestamps are valid.',\n            );\n            lastReset = 0;\n          }\n\n          // don't update state with promises started before last clear\n          if (fetchedAt >= lastReset) {\n            this.controller.resolve(action.endpoint, {\n              args: action.args,\n              response,\n              fetchedAt,\n            });\n          }\n          return response;\n        })\n        .catch(error => {\n          const lastReset = this.getLastReset();\n          // don't update state with promises started before last clear\n          if (fetchedAt >= lastReset) {\n            this.controller.resolve(action.endpoint, {\n              args: action.args,\n              response: error,\n              fetchedAt,\n              error: true,\n            });\n          }\n          throw error;\n        });\n      return promise;\n    };\n\n    if (throttle) {\n      return this.throttle(action.key, deferedFetch, fetchedAt)\n        .then(data => resolve(data))\n        .catch(error => reject(error));\n    } else {\n      return deferedFetch().catch(() => {});\n    }\n  }\n\n  /** Called when middleware intercepts a set action.\n   *\n   * Will resolve the promise associated with set key.\n   */\n  protected handleSet(action: SetResponseAction) {\n    // this can still turn out to be untrue since this is async\n    if (this.fetching.has(action.key)) {\n      const { reject, resolve } = this.fetching.get(action.key) as FetchingMeta;\n      if (action.error) {\n        reject(action.response);\n      } else {\n        resolve(action.response);\n      }\n\n      // since we're resolved we no longer need to keep track of this promise\n      this.clear(action.key);\n    }\n  }\n\n  /** Ensures only one request for a given key is in flight at any time\n   *\n   * Uses key to either retrieve in-flight promise, or if not\n   * create a new promise and call fetch.\n   *\n   * Note: The new promise is not actually tied to fetch at all,\n   * but is resolved when the expected 'receive' action is processed.\n   * This ensures promises are resolved only once their data is processed\n   * by the reducer.\n   */\n  protected throttle(\n    key: string,\n    fetch: () => Promise<any>,\n    fetchedAt: number,\n  ): Promise<any> {\n    const lastReset = this.getLastReset();\n    let fetchMeta = this.fetching.get(key);\n\n    // we're already fetching so reuse the promise\n    // fetches after reset do not count\n    if (fetchMeta && fetchMeta.fetchedAt > lastReset) {\n      return fetchMeta.promise;\n    }\n\n    fetchMeta = newFetchMeta(fetchedAt);\n    this.fetching.set(key, fetchMeta);\n\n    this.idleCallback(\n      () => {\n        // since our real promise is resolved via the wrapReducer(),\n        // we should just stop all errors here.\n        // TODO: decouple this from useFetcher() (that's what's dispatching the error the resolves in here)\n        fetch().catch(() => null);\n      },\n      { timeout: 500 },\n    );\n\n    return fetchMeta.promise;\n  }\n\n  /** Calls the callback when client is not 'busy' with high priority interaction tasks\n   *\n   * Override for platform-specific implementations\n   */\n  protected idleCallback(\n    callback: (...args: any[]) => void,\n    options?: IdleRequestOptions,\n  ) {\n    callback();\n  }\n}\n\nfunction newFetchMeta(fetchedAt: number): FetchingMeta {\n  const fetchMeta = { fetchedAt } as FetchingMeta;\n  fetchMeta.promise = new Promise((resolve, reject) => {\n    fetchMeta.resolve = resolve;\n    fetchMeta.reject = reject;\n  });\n  return fetchMeta;\n}\n"],"mappings":"AAAA,SAASA,YAAY,EAAEC,KAAK,EAAEC,KAAK,QAAQ,mBAAmB;AAC9D,SAASC,iBAAiB,QAAQ,gCAAgC;AAClE,OAAOC,UAAU,MAAM,6BAA6B;AASpD,OAAO,MAAMC,UAAU,SAASC,KAAK,CAAC;EAGpCC,WAAWA,CAAA,EAAG;IACZ,KAAK,CAAC,sBAAsB,CAAC;IAAC,KAHhCC,IAAI,GAAG,YAAY;EAInB;AACF;AASA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,MAAMC,cAAc,CAAoB;EAOrDF,WAAWA,CAAC;IAAEG,gBAAgB,GAAG,KAAK;IAAEC,iBAAiB,GAAG;EAAK,CAAC,GAAG,CAAC,CAAC,EAAE;IAAA,KAN/DC,QAAQ,GAA8B,IAAIC,GAAG,CAAC,CAAC;IAAA,KAG/CC,UAAU,GAAe,IAAIV,UAAU,CAAC,CAAC;IAAA,KAQnDW,UAAU,GAAeD,UAAU,IAAI;MACrC,IAAI,CAACA,UAAU,GAAGA,UAAU;MAC5B,OAAOE,IAAI,IAAIC,MAAM,IAAI;QACvB,QAAQA,MAAM,CAACC,IAAI;UACjB,KAAKjB,KAAK;YACR,IAAI,CAACkB,WAAW,CAACF,MAAM,CAAC;YACxB;YACA;YACA;YACA,IACEA,MAAM,CAACG,QAAQ,CAACC,qBAAqB,KAAKC,SAAS,IACnDL,MAAM,CAACG,QAAQ,CAACG,UAAU,EAC1B;cACA,OAAOP,IAAI,CAACC,MAAM,CAAC;YACrB;YACA,OAAOO,OAAO,CAACC,OAAO,CAAC,CAAC;UAC1B,KAAKzB,YAAY;YACf;YACA,OAAOgB,IAAI,CAACC,MAAM,CAAC,CAACS,IAAI,CAAC,MAAM;cAC7B,IAAI,IAAI,CAACd,QAAQ,CAACe,GAAG,CAACV,MAAM,CAACW,GAAG,CAAC,EAAE;gBAAA,IAAAC,qBAAA;gBACjC;gBACA,MAAMC,KAAK,IAAAD,qBAAA,GAAGf,UAAU,CAACiB,QAAQ,CAAC,CAAC,CAACC,IAAI,CAACf,MAAM,CAACW,GAAG,CAAC,qBAAtCC,qBAAA,CAAwCC,KAAK;gBAC3D;gBACA,IAAIA,KAAK,EAAE;kBACT,IAAI,CAACG,SAAS,CACZ9B,iBAAiB,CAACc,MAAM,CAACG,QAAQ,EAAE;oBACjCc,IAAI,EAAEjB,MAAM,CAACiB,IAAI;oBACjBC,QAAQ,EAAEL,KAAK;oBACfM,SAAS,EAAEnB,MAAM,CAACe,IAAI,CAACI,SAAS;oBAChCN,KAAK,EAAE;kBACT,CAAC,CACH,CAAC;gBACH,CAAC,MAAM;kBACL,IAAI,CAACG,SAAS,CAAChB,MAAM,CAAC;gBACxB;cACF;YACF,CAAC,CAAC;UACJ,KAAKf,KAAK;YAAE;cACV;cACA;cACA,MAAMmC,OAAO,GAAGC,KAAK,CAACC,IAAI,CAAC,IAAI,CAAC3B,QAAQ,CAAC4B,MAAM,CAAC,CAAC,CAAC;cAElD,IAAI,CAACC,QAAQ,CAAC,CAAC;cACf,OAAOzB,IAAI,CAACC,MAAM,CAAC,CAACS,IAAI,CAAC,MAAM;gBAC7B;gBACA;gBACA,KAAK,MAAM;kBAAEgB;gBAAO,CAAC,IAAIL,OAAO,EAAE;kBAChCK,MAAM,CAAC,IAAIrC,UAAU,CAAC,CAAC,CAAC;gBAC1B;cACF,CAAC,CAAC;YACJ;UACA;YACE,OAAOW,IAAI,CAACC,MAAM,CAAC;QACvB;MACF,CAAC;IACH,CAAC;IA3DC,IAAI,CAACP,gBAAgB,GAAGA,gBAAgB;IACxC,IAAI,CAACC,iBAAiB,GAAGA,iBAAiB;EAC5C;EA2DA;EACAgC,IAAIA,CAAA,EAAG;IACL,OAAO,IAAI,CAACC,WAAW;EACzB;;EAEA;EACAC,OAAOA,CAAA,EAAG;IACR;IACA;IACA,IAAI,CAACD,WAAW,GAAGE,IAAI,CAACC,GAAG,CAAC,CAAC;EAC/B;;EAEA;EACAC,WAAWA,CAAC/B,MAAmB,EAAE;IAC/B;IACA,OAAOA,MAAM,CAACC,IAAI,KAAKjB,KAAK,IAAI,IAAI,CAACW,QAAQ,CAACe,GAAG,CAACV,MAAM,CAACW,GAAG,CAAC;EAC/D;EAEAqB,UAAUA,CAAA,EAAG;IACX,IAAI,IAAI,CAACrC,QAAQ,CAACsC,IAAI,EACpB,OAAO1B,OAAO,CAACyB,UAAU,CACvB,IAAI,CAACrC,QAAQ,CAAC4B,MAAM,CAAC,CAAC,CAACW,GAAG,CAAC,CAAC;MAAEC;IAAQ,CAAC,KAAKA,OAAO,CACrD,CAAC;EACL;;EAEA;EACUX,QAAQA,CAAA,EAAG;IACnB,KAAK,MAAMY,CAAC,IAAI,IAAI,CAACzC,QAAQ,CAAC0C,IAAI,CAAC,CAAC,EAAE;MACpC,IAAI,CAACC,KAAK,CAACF,CAAC,CAAC;IACf;EACF;;EAEA;EACUE,KAAKA,CAAC3B,GAAW,EAAE;IAC3B,IAAI,IAAI,CAAChB,QAAQ,CAACe,GAAG,CAACC,GAAG,CAAC,EAAE;MACzB,IAAI,CAAChB,QAAQ,CAAC4C,GAAG,CAAC5B,GAAG,CAAC,CAAkBwB,OAAO,CAACK,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;MAChE,IAAI,CAAC7C,QAAQ,CAAC8C,MAAM,CAAC9B,GAAG,CAAC;IAC3B;EACF;EAEU+B,YAAYA,CAAA,EAAG;IACvB,IAAI,IAAI,CAACf,WAAW,EAAE,OAAO,IAAI,CAACA,WAAW;IAC7C,OAAO,IAAI,CAAC9B,UAAU,CAACiB,QAAQ,CAAC,CAAC,CAAC6B,SAAS;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACYzC,WAAWA,CAACF,MAAmB,EAAE;IACzC,MAAM;MAAEQ,OAAO;MAAEiB,MAAM;MAAEN;IAAU,CAAC,GAAGnB,MAAM,CAACe,IAAI;IAClD,MAAM6B,QAAQ,GAAG,CAAC5C,MAAM,CAACG,QAAQ,CAACG,UAAU;IAE5C,MAAMuC,YAAY,GAAGA,CAAA,KAAM;MACzB,IAAIV,OAAO,GAAGnC,MAAM,CAACG,QAAQ,CAAC,GAAGH,MAAM,CAACiB,IAAI,CAAC;MAC7C,MAAM6B,cAAc,GAClBX,OAAiD,IAEjDA,OAAO,CACJ1B,IAAI,CAACsC,IAAI,IAAI;QACZvC,OAAO,CAACuC,IAAI,CAAC;QACb,OAAOA,IAAI;MACb,CAAC,CAAC,CACDP,KAAK,CAAC3B,KAAK,IAAI;QACdY,MAAM,CAACZ,KAAK,CAAC;QACb,MAAMA,KAAK;MACb,CAAC,CAAC;MACN;MACA;MACA;MACA,IAAI,CAAC+B,QAAQ,EAAE;QACbT,OAAO,GAAGW,cAAc,CAACX,OAAO,CAAC;MACnC;MACAA,OAAO,GAAGA,OAAO,CACd1B,IAAI,CAACS,QAAQ,IAAI;QAChB,IAAIyB,SAAS,GAAG,IAAI,CAACD,YAAY,CAAC,CAAC;;QAEnC;QACA,IAAIM,OAAO,CAACC,GAAG,CAACC,QAAQ,KAAK,YAAY,IAAIC,KAAK,CAACR,SAAS,CAAC,EAAE;UAC7DS,OAAO,CAACvC,KAAK,CACX,6DACF,CAAC;UACD8B,SAAS,GAAG,CAAC;QACf;;QAEA;QACA,IAAIxB,SAAS,IAAIwB,SAAS,EAAE;UAC1B,IAAI,CAAC9C,UAAU,CAACW,OAAO,CAACR,MAAM,CAACG,QAAQ,EAAE;YACvCc,IAAI,EAAEjB,MAAM,CAACiB,IAAI;YACjBC,QAAQ;YACRC;UACF,CAAC,CAAC;QACJ;QACA,OAAOD,QAAQ;MACjB,CAAC,CAAC,CACDsB,KAAK,CAAC3B,KAAK,IAAI;QACd,MAAM8B,SAAS,GAAG,IAAI,CAACD,YAAY,CAAC,CAAC;QACrC;QACA,IAAIvB,SAAS,IAAIwB,SAAS,EAAE;UAC1B,IAAI,CAAC9C,UAAU,CAACW,OAAO,CAACR,MAAM,CAACG,QAAQ,EAAE;YACvCc,IAAI,EAAEjB,MAAM,CAACiB,IAAI;YACjBC,QAAQ,EAAEL,KAAK;YACfM,SAAS;YACTN,KAAK,EAAE;UACT,CAAC,CAAC;QACJ;QACA,MAAMA,KAAK;MACb,CAAC,CAAC;MACJ,OAAOsB,OAAO;IAChB,CAAC;IAED,IAAIS,QAAQ,EAAE;MACZ,OAAO,IAAI,CAACA,QAAQ,CAAC5C,MAAM,CAACW,GAAG,EAAEkC,YAAY,EAAE1B,SAAS,CAAC,CACtDV,IAAI,CAACsC,IAAI,IAAIvC,OAAO,CAACuC,IAAI,CAAC,CAAC,CAC3BP,KAAK,CAAC3B,KAAK,IAAIY,MAAM,CAACZ,KAAK,CAAC,CAAC;IAClC,CAAC,MAAM;MACL,OAAOgC,YAAY,CAAC,CAAC,CAACL,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;IACvC;EACF;;EAEA;AACF;AACA;AACA;EACYxB,SAASA,CAAChB,MAAyB,EAAE;IAC7C;IACA,IAAI,IAAI,CAACL,QAAQ,CAACe,GAAG,CAACV,MAAM,CAACW,GAAG,CAAC,EAAE;MACjC,MAAM;QAAEc,MAAM;QAAEjB;MAAQ,CAAC,GAAG,IAAI,CAACb,QAAQ,CAAC4C,GAAG,CAACvC,MAAM,CAACW,GAAG,CAAiB;MACzE,IAAIX,MAAM,CAACa,KAAK,EAAE;QAChBY,MAAM,CAACzB,MAAM,CAACkB,QAAQ,CAAC;MACzB,CAAC,MAAM;QACLV,OAAO,CAACR,MAAM,CAACkB,QAAQ,CAAC;MAC1B;;MAEA;MACA,IAAI,CAACoB,KAAK,CAACtC,MAAM,CAACW,GAAG,CAAC;IACxB;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACYiC,QAAQA,CAChBjC,GAAW,EACX0C,KAAyB,EACzBlC,SAAiB,EACH;IACd,MAAMwB,SAAS,GAAG,IAAI,CAACD,YAAY,CAAC,CAAC;IACrC,IAAIY,SAAS,GAAG,IAAI,CAAC3D,QAAQ,CAAC4C,GAAG,CAAC5B,GAAG,CAAC;;IAEtC;IACA;IACA,IAAI2C,SAAS,IAAIA,SAAS,CAACnC,SAAS,GAAGwB,SAAS,EAAE;MAChD,OAAOW,SAAS,CAACnB,OAAO;IAC1B;IAEAmB,SAAS,GAAGC,YAAY,CAACpC,SAAS,CAAC;IACnC,IAAI,CAACxB,QAAQ,CAAC6D,GAAG,CAAC7C,GAAG,EAAE2C,SAAS,CAAC;IAEjC,IAAI,CAACG,YAAY,CACf,MAAM;MACJ;MACA;MACA;MACAJ,KAAK,CAAC,CAAC,CAACb,KAAK,CAAC,MAAM,IAAI,CAAC;IAC3B,CAAC,EACD;MAAEkB,OAAO,EAAE;IAAI,CACjB,CAAC;IAED,OAAOJ,SAAS,CAACnB,OAAO;EAC1B;;EAEA;AACF;AACA;AACA;EACYsB,YAAYA,CACpBE,QAAkC,EAClCC,OAA4B,EAC5B;IACAD,QAAQ,CAAC,CAAC;EACZ;AACF;AAEA,SAASJ,YAAYA,CAACpC,SAAiB,EAAgB;EACrD,MAAMmC,SAAS,GAAG;IAAEnC;EAAU,CAAiB;EAC/CmC,SAAS,CAACnB,OAAO,GAAG,IAAI5B,OAAO,CAAC,CAACC,OAAO,EAAEiB,MAAM,KAAK;IACnD6B,SAAS,CAAC9C,OAAO,GAAGA,OAAO;IAC3B8C,SAAS,CAAC7B,MAAM,GAAGA,MAAM;EAC3B,CAAC,CAAC;EACF,OAAO6B,SAAS;AAClB","ignoreList":[]}
@@ -1 +1 @@
1
- {"version":3,"file":"GCPolicy.d.ts","sourceRoot":"","sources":["../../src/state/GCPolicy.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AAGzD,OAAO,UAAU,MAAM,6BAA6B,CAAC;AAErD,qBAAa,QAAS,YAAW,WAAW;IAC1C,SAAS,CAAC,aAAa,sBAA6B;IACpD,SAAS,CAAC,WAAW,mCAA0C;IAC/D,SAAS,CAAC,UAAU,cAAqB;IACzC,SAAS,CAAC,SAAS,EAAE,UAAU,EAAE,CAAM;IAEvC,UAAkB,UAAU,EAAE,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC;IAC7D,UAAkB,UAAU,EAAE,UAAU,CAAC;IACzC,UAAkB,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC,CAAC;gBAEtD,EAEV,UAA0B,EAC1B,gBAAoB,EACpB,SAAS,GACV,GAAE,SAAc;IAUjB,IAAI,CAAC,UAAU,EAAE,UAAU;IAQ3B,OAAO;IAIP,cAAc,CAAC,EAAE,GAAG,EAAE,KAAU,EAAE,EAAE;QAAE,GAAG,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,UAAU,EAAE,CAAA;KAAE;IAiD1E,SAAS,CAAC,SAAS,CAAC,EAClB,SAAS,EACT,SAAS,GACV,EAAE;QACD,SAAS,EAAE,MAAM,CAAC;QAClB,IAAI,EAAE,MAAM,CAAC;QACb,SAAS,EAAE,MAAM,CAAC;KACnB,GAAG,MAAM;IASV,SAAS,CAAC,QAAQ;IAiDlB;;;OAGG;IACH,SAAS,CAAC,YAAY,CACpB,QAAQ,EAAE,CAAC,GAAG,IAAI,EAAE,GAAG,EAAE,KAAK,IAAI,EAClC,OAAO,CAAC,EAAE,kBAAkB;CAQ/B;AAED,qBAAa,gBAAiB,YAAW,WAAW;IAElD,IAAI;IAEJ,OAAO;IACP,cAAc;CAGf;AAED,MAAM,WAAW,SAAS;IACxB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,SAAS,CAAC,EAAE,CAAC,IAAI,EAAE;QACjB,SAAS,EAAE,MAAM,CAAC;QAClB,IAAI,EAAE,MAAM,CAAC;QACb,SAAS,EAAE,MAAM,CAAC;KACnB,KAAK,MAAM,CAAC;CACd;AACD,MAAM,WAAW,cAAc;IAC7B,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QAAE,GAAG,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,UAAU,EAAE,CAAA;KAAE,GAAG,MAAM,MAAM,IAAI,CAAC;CAC5E;AACD,MAAM,WAAW,WAAW;IAC1B,cAAc,EAAE,cAAc,CAAC;IAC/B,IAAI,CAAC,UAAU,EAAE,UAAU,GAAG,IAAI,CAAC;IACnC,OAAO,IAAI,IAAI,CAAC;CACjB"}
1
+ {"version":3,"file":"GCPolicy.d.ts","sourceRoot":"","sources":["../../src/state/GCPolicy.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AAGzD,OAAO,UAAU,MAAM,6BAA6B,CAAC;AAErD,qBAAa,QAAS,YAAW,WAAW;IAC1C,SAAS,CAAC,aAAa,sBAA6B;IACpD,SAAS,CAAC,WAAW,mCAA0C;IAC/D,SAAS,CAAC,UAAU,cAAqB;IACzC,SAAS,CAAC,SAAS,EAAE,UAAU,EAAE,CAAM;IAEvC,UAAkB,UAAU,EAAE,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC;IAC7D,UAAkB,UAAU,EAAE,UAAU,CAAC;IACzC,UAAkB,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC,CAAC;gBAEtD,EAEV,UAA0B,EAC1B,gBAAoB,EACpB,SAAS,GACV,GAAE,SAAc;IAUjB,IAAI,CAAC,UAAU,EAAE,UAAU;IAW3B,OAAO;IAIP,cAAc,CAAC,EAAE,GAAG,EAAE,KAAU,EAAE,EAAE;QAAE,GAAG,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,UAAU,EAAE,CAAA;KAAE;IAiD1E,SAAS,CAAC,SAAS,CAAC,EAClB,SAAS,EACT,SAAS,GACV,EAAE;QACD,SAAS,EAAE,MAAM,CAAC;QAClB,IAAI,EAAE,MAAM,CAAC;QACb,SAAS,EAAE,MAAM,CAAC;KACnB,GAAG,MAAM;IASV,SAAS,CAAC,QAAQ;IAiDlB;;;OAGG;IACH,SAAS,CAAC,YAAY,CACpB,QAAQ,EAAE,CAAC,GAAG,IAAI,EAAE,GAAG,EAAE,KAAK,IAAI,EAClC,OAAO,CAAC,EAAE,kBAAkB;CAQ/B;AAED,qBAAa,gBAAiB,YAAW,WAAW;IAElD,IAAI;IAEJ,OAAO;IACP,cAAc;CAGf;AAED,MAAM,WAAW,SAAS;IACxB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,SAAS,CAAC,EAAE,CAAC,IAAI,EAAE;QACjB,SAAS,EAAE,MAAM,CAAC;QAClB,IAAI,EAAE,MAAM,CAAC;QACb,SAAS,EAAE,MAAM,CAAC;KACnB,KAAK,MAAM,CAAC;CACd;AACD,MAAM,WAAW,cAAc;IAC7B,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QAAE,GAAG,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,UAAU,EAAE,CAAA;KAAE,GAAG,MAAM,MAAM,IAAI,CAAC;CAC5E;AACD,MAAM,WAAW,WAAW;IAC1B,cAAc,EAAE,cAAc,CAAC;IAC/B,IAAI,CAAC,UAAU,EAAE,UAAU,GAAG,IAAI,CAAC;IACnC,OAAO,IAAI,IAAI,CAAC;CACjB"}
@@ -25,6 +25,9 @@ export class GCPolicy {
25
25
  timeout: 1000
26
26
  });
27
27
  }, this.options.intervalMS);
28
+ if (typeof this.intervalId === 'object' && 'unref' in this.intervalId) {
29
+ this.intervalId.unref();
30
+ }
28
31
  }
29
32
  cleanup() {
30
33
  clearInterval(this.intervalId);
@@ -156,4 +159,4 @@ export class ImmortalGCPolicy {
156
159
  return () => () => undefined;
157
160
  }
158
161
  }
159
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["GC","GCPolicy","constructor","intervalMS","expiryMultiplier","expiresAt","endpointCount","Map","entityCount","endpointsQ","Set","entitiesQ","bind","options","init","controller","intervalId","setInterval","idleCallback","runSweep","timeout","cleanup","clearInterval","createCountRef","key","paths","_this$endpointCount$g","set","get","forEach","path","_instanceCount$get","pk","has","instanceCount","currentCount","undefined","delete","add","push","fetchedAt","Math","max","state","getState","entities","endpoints","now","Date","nextEndpointsQ","_state$meta$key","meta","date","nextEntitiesQ","_this$entityCount$get","_state$entitiesMeta$p","_state$entitiesMeta$p2","entitiesMeta","length","dispatch","type","callback","requestIdleCallback","ImmortalGCPolicy"],"sources":["../../src/state/GCPolicy.ts"],"sourcesContent":["import type { EntityPath } from '@data-client/normalizr';\n\nimport { GC } from '../actionTypes.js';\nimport Controller from '../controller/Controller.js';\n\nexport class GCPolicy implements GCInterface {\n  protected endpointCount = new Map<string, number>();\n  protected entityCount = new Map<string, Map<string, number>>();\n  protected endpointsQ = new Set<string>();\n  protected entitiesQ: EntityPath[] = [];\n\n  declare protected intervalId: ReturnType<typeof setInterval>;\n  declare protected controller: Controller;\n  declare protected options: Required<Omit<GCOptions, 'expiresAt'>>;\n\n  constructor({\n    // every 5 min\n    intervalMS = 60 * 1000 * 5,\n    expiryMultiplier = 2,\n    expiresAt,\n  }: GCOptions = {}) {\n    if (expiresAt) {\n      this.expiresAt = expiresAt.bind(this);\n    }\n    this.options = {\n      intervalMS,\n      expiryMultiplier,\n    };\n  }\n\n  init(controller: Controller) {\n    this.controller = controller;\n\n    this.intervalId = setInterval(() => {\n      this.idleCallback(() => this.runSweep(), { timeout: 1000 });\n    }, this.options.intervalMS);\n  }\n\n  cleanup() {\n    clearInterval(this.intervalId);\n  }\n\n  createCountRef({ key, paths = [] }: { key?: string; paths?: EntityPath[] }) {\n    // increment\n    return () => {\n      if (key)\n        this.endpointCount.set(key, (this.endpointCount.get(key) ?? 0) + 1);\n      paths.forEach(path => {\n        const { key, pk } = path;\n        if (!this.entityCount.has(key)) {\n          this.entityCount.set(key, new Map<string, number>());\n        }\n        const instanceCount = this.entityCount.get(key)!;\n        instanceCount.set(pk, (instanceCount.get(pk) ?? 0) + 1);\n      });\n\n      // decrement\n      return () => {\n        if (key) {\n          const currentCount = this.endpointCount.get(key)!;\n          if (currentCount !== undefined) {\n            if (currentCount <= 1) {\n              this.endpointCount.delete(key);\n              // queue for cleanup\n              this.endpointsQ.add(key);\n            } else {\n              this.endpointCount.set(key, currentCount - 1);\n            }\n          }\n        }\n        paths.forEach(path => {\n          const { key, pk } = path;\n          if (!this.entityCount.has(key)) {\n            return;\n          }\n          const instanceCount = this.entityCount.get(key)!;\n          const entityCount = instanceCount.get(pk)!;\n          if (entityCount !== undefined) {\n            if (entityCount <= 1) {\n              instanceCount.delete(pk);\n              // queue for cleanup\n              this.entitiesQ.push(path);\n            } else {\n              instanceCount.set(pk, entityCount - 1);\n            }\n          }\n        });\n      };\n    };\n  }\n\n  protected expiresAt({\n    fetchedAt,\n    expiresAt,\n  }: {\n    expiresAt: number;\n    date: number;\n    fetchedAt: number;\n  }): number {\n    return (\n      Math.max(\n        (expiresAt - fetchedAt) * this.options.expiryMultiplier,\n        120000,\n      ) + fetchedAt\n    );\n  }\n\n  protected runSweep() {\n    const state = this.controller.getState();\n    const entities: EntityPath[] = [];\n    const endpoints: string[] = [];\n    const now = Date.now();\n\n    const nextEndpointsQ = new Set<string>();\n    for (const key of this.endpointsQ) {\n      if (\n        !this.endpointCount.has(key) &&\n        this.expiresAt(\n          state.meta[key] ?? {\n            fetchedAt: 0,\n            date: 0,\n            expiresAt: 0,\n          },\n        ) < now\n      ) {\n        endpoints.push(key);\n      } else {\n        nextEndpointsQ.add(key);\n      }\n    }\n    this.endpointsQ = nextEndpointsQ;\n\n    const nextEntitiesQ: EntityPath[] = [];\n    for (const path of this.entitiesQ) {\n      if (\n        !this.entityCount.get(path.key)?.has(path.pk) &&\n        this.expiresAt(\n          state.entitiesMeta[path.key]?.[path.pk] ?? {\n            fetchedAt: 0,\n            date: 0,\n            expiresAt: 0,\n          },\n        ) < now\n      ) {\n        entities.push(path);\n      } else {\n        nextEntitiesQ.push(path);\n      }\n    }\n    this.entitiesQ = nextEntitiesQ;\n\n    if (entities.length || endpoints.length) {\n      this.controller.dispatch({ type: GC, entities, endpoints });\n    }\n  }\n\n  /** Calls the callback when client is not 'busy' with high priority interaction tasks\n   *\n   * Override for platform-specific implementations\n   */\n  protected idleCallback(\n    callback: (...args: any[]) => void,\n    options?: IdleRequestOptions,\n  ) {\n    if (typeof requestIdleCallback === 'function') {\n      requestIdleCallback(callback, options);\n    } else {\n      callback();\n    }\n  }\n}\n\nexport class ImmortalGCPolicy implements GCInterface {\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  init() {}\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  cleanup() {}\n  createCountRef() {\n    return () => () => undefined;\n  }\n}\n\nexport interface GCOptions {\n  intervalMS?: number;\n  expiryMultiplier?: number;\n  expiresAt?: (meta: {\n    expiresAt: number;\n    date: number;\n    fetchedAt: number;\n  }) => number;\n}\nexport interface CreateCountRef {\n  ({ key, paths }: { key?: string; paths?: EntityPath[] }): () => () => void;\n}\nexport interface GCInterface {\n  createCountRef: CreateCountRef;\n  init(controller: Controller): void;\n  cleanup(): void;\n}\n"],"mappings":"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","ignoreList":[]}
162
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["GC","GCPolicy","constructor","intervalMS","expiryMultiplier","expiresAt","endpointCount","Map","entityCount","endpointsQ","Set","entitiesQ","bind","options","init","controller","intervalId","setInterval","idleCallback","runSweep","timeout","unref","cleanup","clearInterval","createCountRef","key","paths","_this$endpointCount$g","set","get","forEach","path","_instanceCount$get","pk","has","instanceCount","currentCount","undefined","delete","add","push","fetchedAt","Math","max","state","getState","entities","endpoints","now","Date","nextEndpointsQ","_state$meta$key","meta","date","nextEntitiesQ","_this$entityCount$get","_state$entitiesMeta$p","_state$entitiesMeta$p2","entitiesMeta","length","dispatch","type","callback","requestIdleCallback","ImmortalGCPolicy"],"sources":["../../src/state/GCPolicy.ts"],"sourcesContent":["import type { EntityPath } from '@data-client/normalizr';\n\nimport { GC } from '../actionTypes.js';\nimport Controller from '../controller/Controller.js';\n\nexport class GCPolicy implements GCInterface {\n  protected endpointCount = new Map<string, number>();\n  protected entityCount = new Map<string, Map<string, number>>();\n  protected endpointsQ = new Set<string>();\n  protected entitiesQ: EntityPath[] = [];\n\n  declare protected intervalId: ReturnType<typeof setInterval>;\n  declare protected controller: Controller;\n  declare protected options: Required<Omit<GCOptions, 'expiresAt'>>;\n\n  constructor({\n    // every 5 min\n    intervalMS = 60 * 1000 * 5,\n    expiryMultiplier = 2,\n    expiresAt,\n  }: GCOptions = {}) {\n    if (expiresAt) {\n      this.expiresAt = expiresAt.bind(this);\n    }\n    this.options = {\n      intervalMS,\n      expiryMultiplier,\n    };\n  }\n\n  init(controller: Controller) {\n    this.controller = controller;\n\n    this.intervalId = setInterval(() => {\n      this.idleCallback(() => this.runSweep(), { timeout: 1000 });\n    }, this.options.intervalMS);\n    if (typeof this.intervalId === 'object' && 'unref' in this.intervalId) {\n      this.intervalId.unref();\n    }\n  }\n\n  cleanup() {\n    clearInterval(this.intervalId);\n  }\n\n  createCountRef({ key, paths = [] }: { key?: string; paths?: EntityPath[] }) {\n    // increment\n    return () => {\n      if (key)\n        this.endpointCount.set(key, (this.endpointCount.get(key) ?? 0) + 1);\n      paths.forEach(path => {\n        const { key, pk } = path;\n        if (!this.entityCount.has(key)) {\n          this.entityCount.set(key, new Map<string, number>());\n        }\n        const instanceCount = this.entityCount.get(key)!;\n        instanceCount.set(pk, (instanceCount.get(pk) ?? 0) + 1);\n      });\n\n      // decrement\n      return () => {\n        if (key) {\n          const currentCount = this.endpointCount.get(key)!;\n          if (currentCount !== undefined) {\n            if (currentCount <= 1) {\n              this.endpointCount.delete(key);\n              // queue for cleanup\n              this.endpointsQ.add(key);\n            } else {\n              this.endpointCount.set(key, currentCount - 1);\n            }\n          }\n        }\n        paths.forEach(path => {\n          const { key, pk } = path;\n          if (!this.entityCount.has(key)) {\n            return;\n          }\n          const instanceCount = this.entityCount.get(key)!;\n          const entityCount = instanceCount.get(pk)!;\n          if (entityCount !== undefined) {\n            if (entityCount <= 1) {\n              instanceCount.delete(pk);\n              // queue for cleanup\n              this.entitiesQ.push(path);\n            } else {\n              instanceCount.set(pk, entityCount - 1);\n            }\n          }\n        });\n      };\n    };\n  }\n\n  protected expiresAt({\n    fetchedAt,\n    expiresAt,\n  }: {\n    expiresAt: number;\n    date: number;\n    fetchedAt: number;\n  }): number {\n    return (\n      Math.max(\n        (expiresAt - fetchedAt) * this.options.expiryMultiplier,\n        120000,\n      ) + fetchedAt\n    );\n  }\n\n  protected runSweep() {\n    const state = this.controller.getState();\n    const entities: EntityPath[] = [];\n    const endpoints: string[] = [];\n    const now = Date.now();\n\n    const nextEndpointsQ = new Set<string>();\n    for (const key of this.endpointsQ) {\n      if (\n        !this.endpointCount.has(key) &&\n        this.expiresAt(\n          state.meta[key] ?? {\n            fetchedAt: 0,\n            date: 0,\n            expiresAt: 0,\n          },\n        ) < now\n      ) {\n        endpoints.push(key);\n      } else {\n        nextEndpointsQ.add(key);\n      }\n    }\n    this.endpointsQ = nextEndpointsQ;\n\n    const nextEntitiesQ: EntityPath[] = [];\n    for (const path of this.entitiesQ) {\n      if (\n        !this.entityCount.get(path.key)?.has(path.pk) &&\n        this.expiresAt(\n          state.entitiesMeta[path.key]?.[path.pk] ?? {\n            fetchedAt: 0,\n            date: 0,\n            expiresAt: 0,\n          },\n        ) < now\n      ) {\n        entities.push(path);\n      } else {\n        nextEntitiesQ.push(path);\n      }\n    }\n    this.entitiesQ = nextEntitiesQ;\n\n    if (entities.length || endpoints.length) {\n      this.controller.dispatch({ type: GC, entities, endpoints });\n    }\n  }\n\n  /** Calls the callback when client is not 'busy' with high priority interaction tasks\n   *\n   * Override for platform-specific implementations\n   */\n  protected idleCallback(\n    callback: (...args: any[]) => void,\n    options?: IdleRequestOptions,\n  ) {\n    if (typeof requestIdleCallback === 'function') {\n      requestIdleCallback(callback, options);\n    } else {\n      callback();\n    }\n  }\n}\n\nexport class ImmortalGCPolicy implements GCInterface {\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  init() {}\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  cleanup() {}\n  createCountRef() {\n    return () => () => undefined;\n  }\n}\n\nexport interface GCOptions {\n  intervalMS?: number;\n  expiryMultiplier?: number;\n  expiresAt?: (meta: {\n    expiresAt: number;\n    date: number;\n    fetchedAt: number;\n  }) => number;\n}\nexport interface CreateCountRef {\n  ({ key, paths }: { key?: string; paths?: EntityPath[] }): () => () => void;\n}\nexport interface GCInterface {\n  createCountRef: CreateCountRef;\n  init(controller: Controller): void;\n  cleanup(): void;\n}\n"],"mappings":"AAEA,SAASA,EAAE,QAAQ,mBAAmB;AAGtC,OAAO,MAAMC,QAAQ,CAAwB;EAU3CC,WAAWA,CAAC;IACV;IACAC,UAAU,GAAG,EAAE,GAAG,IAAI,GAAG,CAAC;IAC1BC,gBAAgB,GAAG,CAAC;IACpBC;EACS,CAAC,GAAG,CAAC,CAAC,EAAE;IAAA,KAdTC,aAAa,GAAG,IAAIC,GAAG,CAAiB,CAAC;IAAA,KACzCC,WAAW,GAAG,IAAID,GAAG,CAA8B,CAAC;IAAA,KACpDE,UAAU,GAAG,IAAIC,GAAG,CAAS,CAAC;IAAA,KAC9BC,SAAS,GAAiB,EAAE;IAYpC,IAAIN,SAAS,EAAE;MACb,IAAI,CAACA,SAAS,GAAGA,SAAS,CAACO,IAAI,CAAC,IAAI,CAAC;IACvC;IACA,IAAI,CAACC,OAAO,GAAG;MACbV,UAAU;MACVC;IACF,CAAC;EACH;EAEAU,IAAIA,CAACC,UAAsB,EAAE;IAC3B,IAAI,CAACA,UAAU,GAAGA,UAAU;IAE5B,IAAI,CAACC,UAAU,GAAGC,WAAW,CAAC,MAAM;MAClC,IAAI,CAACC,YAAY,CAAC,MAAM,IAAI,CAACC,QAAQ,CAAC,CAAC,EAAE;QAAEC,OAAO,EAAE;MAAK,CAAC,CAAC;IAC7D,CAAC,EAAE,IAAI,CAACP,OAAO,CAACV,UAAU,CAAC;IAC3B,IAAI,OAAO,IAAI,CAACa,UAAU,KAAK,QAAQ,IAAI,OAAO,IAAI,IAAI,CAACA,UAAU,EAAE;MACrE,IAAI,CAACA,UAAU,CAACK,KAAK,CAAC,CAAC;IACzB;EACF;EAEAC,OAAOA,CAAA,EAAG;IACRC,aAAa,CAAC,IAAI,CAACP,UAAU,CAAC;EAChC;EAEAQ,cAAcA,CAAC;IAAEC,GAAG;IAAEC,KAAK,GAAG;EAA2C,CAAC,EAAE;IAC1E;IACA,OAAO,MAAM;MAAA,IAAAC,qBAAA;MACX,IAAIF,GAAG,EACL,IAAI,CAACnB,aAAa,CAACsB,GAAG,CAACH,GAAG,EAAE,EAAAE,qBAAA,GAAC,IAAI,CAACrB,aAAa,CAACuB,GAAG,CAACJ,GAAG,CAAC,YAAAE,qBAAA,GAAI,CAAC,IAAI,CAAC,CAAC;MACrED,KAAK,CAACI,OAAO,CAACC,IAAI,IAAI;QAAA,IAAAC,kBAAA;QACpB,MAAM;UAAEP,GAAG;UAAEQ;QAAG,CAAC,GAAGF,IAAI;QACxB,IAAI,CAAC,IAAI,CAACvB,WAAW,CAAC0B,GAAG,CAACT,GAAG,CAAC,EAAE;UAC9B,IAAI,CAACjB,WAAW,CAACoB,GAAG,CAACH,GAAG,EAAE,IAAIlB,GAAG,CAAiB,CAAC,CAAC;QACtD;QACA,MAAM4B,aAAa,GAAG,IAAI,CAAC3B,WAAW,CAACqB,GAAG,CAACJ,GAAG,CAAE;QAChDU,aAAa,CAACP,GAAG,CAACK,EAAE,EAAE,EAAAD,kBAAA,GAACG,aAAa,CAACN,GAAG,CAACI,EAAE,CAAC,YAAAD,kBAAA,GAAI,CAAC,IAAI,CAAC,CAAC;MACzD,CAAC,CAAC;;MAEF;MACA,OAAO,MAAM;QACX,IAAIP,GAAG,EAAE;UACP,MAAMW,YAAY,GAAG,IAAI,CAAC9B,aAAa,CAACuB,GAAG,CAACJ,GAAG,CAAE;UACjD,IAAIW,YAAY,KAAKC,SAAS,EAAE;YAC9B,IAAID,YAAY,IAAI,CAAC,EAAE;cACrB,IAAI,CAAC9B,aAAa,CAACgC,MAAM,CAACb,GAAG,CAAC;cAC9B;cACA,IAAI,CAAChB,UAAU,CAAC8B,GAAG,CAACd,GAAG,CAAC;YAC1B,CAAC,MAAM;cACL,IAAI,CAACnB,aAAa,CAACsB,GAAG,CAACH,GAAG,EAAEW,YAAY,GAAG,CAAC,CAAC;YAC/C;UACF;QACF;QACAV,KAAK,CAACI,OAAO,CAACC,IAAI,IAAI;UACpB,MAAM;YAAEN,GAAG;YAAEQ;UAAG,CAAC,GAAGF,IAAI;UACxB,IAAI,CAAC,IAAI,CAACvB,WAAW,CAAC0B,GAAG,CAACT,GAAG,CAAC,EAAE;YAC9B;UACF;UACA,MAAMU,aAAa,GAAG,IAAI,CAAC3B,WAAW,CAACqB,GAAG,CAACJ,GAAG,CAAE;UAChD,MAAMjB,WAAW,GAAG2B,aAAa,CAACN,GAAG,CAACI,EAAE,CAAE;UAC1C,IAAIzB,WAAW,KAAK6B,SAAS,EAAE;YAC7B,IAAI7B,WAAW,IAAI,CAAC,EAAE;cACpB2B,aAAa,CAACG,MAAM,CAACL,EAAE,CAAC;cACxB;cACA,IAAI,CAACtB,SAAS,CAAC6B,IAAI,CAACT,IAAI,CAAC;YAC3B,CAAC,MAAM;cACLI,aAAa,CAACP,GAAG,CAACK,EAAE,EAAEzB,WAAW,GAAG,CAAC,CAAC;YACxC;UACF;QACF,CAAC,CAAC;MACJ,CAAC;IACH,CAAC;EACH;EAEUH,SAASA,CAAC;IAClBoC,SAAS;IACTpC;EAKF,CAAC,EAAU;IACT,OACEqC,IAAI,CAACC,GAAG,CACN,CAACtC,SAAS,GAAGoC,SAAS,IAAI,IAAI,CAAC5B,OAAO,CAACT,gBAAgB,EACvD,MACF,CAAC,GAAGqC,SAAS;EAEjB;EAEUtB,QAAQA,CAAA,EAAG;IACnB,MAAMyB,KAAK,GAAG,IAAI,CAAC7B,UAAU,CAAC8B,QAAQ,CAAC,CAAC;IACxC,MAAMC,QAAsB,GAAG,EAAE;IACjC,MAAMC,SAAmB,GAAG,EAAE;IAC9B,MAAMC,GAAG,GAAGC,IAAI,CAACD,GAAG,CAAC,CAAC;IAEtB,MAAME,cAAc,GAAG,IAAIxC,GAAG,CAAS,CAAC;IACxC,KAAK,MAAMe,GAAG,IAAI,IAAI,CAAChB,UAAU,EAAE;MAAA,IAAA0C,eAAA;MACjC,IACE,CAAC,IAAI,CAAC7C,aAAa,CAAC4B,GAAG,CAACT,GAAG,CAAC,IAC5B,IAAI,CAACpB,SAAS,EAAA8C,eAAA,GACZP,KAAK,CAACQ,IAAI,CAAC3B,GAAG,CAAC,YAAA0B,eAAA,GAAI;QACjBV,SAAS,EAAE,CAAC;QACZY,IAAI,EAAE,CAAC;QACPhD,SAAS,EAAE;MACb,CACF,CAAC,GAAG2C,GAAG,EACP;QACAD,SAAS,CAACP,IAAI,CAACf,GAAG,CAAC;MACrB,CAAC,MAAM;QACLyB,cAAc,CAACX,GAAG,CAACd,GAAG,CAAC;MACzB;IACF;IACA,IAAI,CAAChB,UAAU,GAAGyC,cAAc;IAEhC,MAAMI,aAA2B,GAAG,EAAE;IACtC,KAAK,MAAMvB,IAAI,IAAI,IAAI,CAACpB,SAAS,EAAE;MAAA,IAAA4C,qBAAA,EAAAC,qBAAA,EAAAC,sBAAA;MACjC,IACE,GAAAF,qBAAA,GAAC,IAAI,CAAC/C,WAAW,CAACqB,GAAG,CAACE,IAAI,CAACN,GAAG,CAAC,aAA9B8B,qBAAA,CAAgCrB,GAAG,CAACH,IAAI,CAACE,EAAE,CAAC,KAC7C,IAAI,CAAC5B,SAAS,EAAAmD,qBAAA,IAAAC,sBAAA,GACZb,KAAK,CAACc,YAAY,CAAC3B,IAAI,CAACN,GAAG,CAAC,qBAA5BgC,sBAAA,CAA+B1B,IAAI,CAACE,EAAE,CAAC,YAAAuB,qBAAA,GAAI;QACzCf,SAAS,EAAE,CAAC;QACZY,IAAI,EAAE,CAAC;QACPhD,SAAS,EAAE;MACb,CACF,CAAC,GAAG2C,GAAG,EACP;QACAF,QAAQ,CAACN,IAAI,CAACT,IAAI,CAAC;MACrB,CAAC,MAAM;QACLuB,aAAa,CAACd,IAAI,CAACT,IAAI,CAAC;MAC1B;IACF;IACA,IAAI,CAACpB,SAAS,GAAG2C,aAAa;IAE9B,IAAIR,QAAQ,CAACa,MAAM,IAAIZ,SAAS,CAACY,MAAM,EAAE;MACvC,IAAI,CAAC5C,UAAU,CAAC6C,QAAQ,CAAC;QAAEC,IAAI,EAAE7D,EAAE;QAAE8C,QAAQ;QAAEC;MAAU,CAAC,CAAC;IAC7D;EACF;;EAEA;AACF;AACA;AACA;EACY7B,YAAYA,CACpB4C,QAAkC,EAClCjD,OAA4B,EAC5B;IACA,IAAI,OAAOkD,mBAAmB,KAAK,UAAU,EAAE;MAC7CA,mBAAmB,CAACD,QAAQ,EAAEjD,OAAO,CAAC;IACxC,CAAC,MAAM;MACLiD,QAAQ,CAAC,CAAC;IACZ;EACF;AACF;AAEA,OAAO,MAAME,gBAAgB,CAAwB;EACnD;EACAlD,IAAIA,CAAA,EAAG,CAAC;EACR;EACAQ,OAAOA,CAAA,EAAG,CAAC;EACXE,cAAcA,CAAA,EAAG;IACf,OAAO,MAAM,MAAMa,SAAS;EAC9B;AACF","ignoreList":[]}
@@ -6,7 +6,7 @@ export declare function expireReducer(state: State<unknown>, action: ExpireAllAc
6
6
  readonly fetchedAt: number;
7
7
  readonly expiresAt: number;
8
8
  readonly prevExpiresAt?: number;
9
- readonly error?: import("packages/normalizr/lib/index.js").ErrorTypes;
9
+ readonly error?: import("@data-client/normalizr").ErrorTypes;
10
10
  readonly invalidated?: boolean;
11
11
  readonly errorPolicy?: "hard" | "soft" | undefined;
12
12
  };
@@ -19,7 +19,7 @@ export declare function expireReducer(state: State<unknown>, action: ExpireAllAc
19
19
  endpoints: {
20
20
  readonly [key: string]: unknown | import("../../types.js").PK[] | import("../../types.js").PK | undefined;
21
21
  };
22
- indexes: import("packages/normalizr/lib/interface.js").NormalizedIndex;
22
+ indexes: import("@data-client/normalizr").NormalizedIndex;
23
23
  entitiesMeta: {
24
24
  readonly [entityKey: string]: {
25
25
  readonly [pk: string]: {
@@ -9,7 +9,7 @@ export declare function invalidateReducer(state: State<unknown>, action: Invalid
9
9
  readonly fetchedAt: number;
10
10
  readonly expiresAt: number;
11
11
  readonly prevExpiresAt?: number;
12
- readonly error?: import("packages/normalizr/lib/index.js").ErrorTypes;
12
+ readonly error?: import("@data-client/normalizr").ErrorTypes;
13
13
  readonly invalidated?: boolean;
14
14
  readonly errorPolicy?: "hard" | "soft" | undefined;
15
15
  };
@@ -19,7 +19,7 @@ export declare function invalidateReducer(state: State<unknown>, action: Invalid
19
19
  readonly [pk: string]: unknown;
20
20
  } | undefined;
21
21
  };
22
- indexes: import("packages/normalizr/lib/interface.js").NormalizedIndex;
22
+ indexes: import("@data-client/normalizr").NormalizedIndex;
23
23
  entitiesMeta: {
24
24
  readonly [entityKey: string]: {
25
25
  readonly [pk: string]: {
@@ -1,30 +1,4 @@
1
1
  import Controller from '../../controller/Controller.js';
2
2
  import type { State, SetAction } from '../../types.js';
3
- export declare function setReducer(state: State<unknown>, action: SetAction, controller: Controller): State<unknown> | {
4
- entities: {
5
- [x: string]: /*elided*/ any;
6
- };
7
- endpoints: {
8
- readonly [key: string]: unknown;
9
- };
10
- indexes: import("@data-client/normalizr").NormalizedIndex;
11
- meta: {
12
- readonly [key: string]: {
13
- readonly date: number;
14
- readonly fetchedAt: number;
15
- readonly expiresAt: number;
16
- readonly prevExpiresAt?: number;
17
- readonly error?: import("@data-client/normalizr").ErrorTypes;
18
- readonly invalidated?: boolean;
19
- readonly errorPolicy?: "hard" | "soft" | undefined;
20
- };
21
- };
22
- entitiesMeta: import("packages/normalizr/lib/types.js").EntitiesToMeta<{
23
- [x: string]: /*elided*/ any;
24
- }>;
25
- optimistic: (import("../../actions.js").OptimisticAction<import("@data-client/normalizr").EndpointInterface<import("@data-client/normalizr").FetchFunction, import("@data-client/normalizr").Schema | undefined, boolean | undefined> & {
26
- update?: import("../../index.js").EndpointUpdateFunction<import("@data-client/normalizr").EndpointInterface>;
27
- }> | import("../../actions.js").SetResponseAction)[];
28
- lastReset: number;
29
- };
3
+ export declare function setReducer(state: State<unknown>, action: SetAction, controller: Controller): State<unknown>;
30
4
  //# sourceMappingURL=setReducer.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"setReducer.d.ts","sourceRoot":"","sources":["../../../src/state/reducer/setReducer.ts"],"names":[],"mappings":"AAEA,OAAO,UAAU,MAAM,gCAAgC,CAAC;AACxD,OAAO,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAEvD,wBAAgB,UAAU,CACxB,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,EACrB,MAAM,EAAE,SAAS,EACjB,UAAU,EAAE,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;EAoCvB"}
1
+ {"version":3,"file":"setReducer.d.ts","sourceRoot":"","sources":["../../../src/state/reducer/setReducer.ts"],"names":[],"mappings":"AAEA,OAAO,UAAU,MAAM,gCAAgC,CAAC;AACxD,OAAO,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAEvD,wBAAgB,UAAU,CACxB,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,EACrB,MAAM,EAAE,SAAS,EACjB,UAAU,EAAE,UAAU,GACrB,KAAK,CAAC,OAAO,CAAC,CAmChB"}
@@ -33,4 +33,4 @@ export function setReducer(state, action, controller) {
33
33
  return state;
34
34
  }
35
35
  }
36
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJub3JtYWxpemUiLCJzZXRSZWR1Y2VyIiwic3RhdGUiLCJhY3Rpb24iLCJjb250cm9sbGVyIiwidmFsdWUiLCJwcmV2aW91c1ZhbHVlIiwiZ2V0Iiwic2NoZW1hIiwiYXJncyIsInVuZGVmaW5lZCIsImVudGl0aWVzIiwiaW5kZXhlcyIsImVudGl0aWVzTWV0YSIsIm1ldGEiLCJlbmRwb2ludHMiLCJvcHRpbWlzdGljIiwibGFzdFJlc2V0IiwiZXJyb3IiLCJwcm9jZXNzIiwiZW52IiwiTk9ERV9FTlYiLCJjb25zb2xlIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0YXRlL3JlZHVjZXIvc2V0UmVkdWNlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBub3JtYWxpemUgfSBmcm9tICdAZGF0YS1jbGllbnQvbm9ybWFsaXpyJztcblxuaW1wb3J0IENvbnRyb2xsZXIgZnJvbSAnLi4vLi4vY29udHJvbGxlci9Db250cm9sbGVyLmpzJztcbmltcG9ydCB0eXBlIHsgU3RhdGUsIFNldEFjdGlvbiB9IGZyb20gJy4uLy4uL3R5cGVzLmpzJztcblxuZXhwb3J0IGZ1bmN0aW9uIHNldFJlZHVjZXIoXG4gIHN0YXRlOiBTdGF0ZTx1bmtub3duPixcbiAgYWN0aW9uOiBTZXRBY3Rpb24sXG4gIGNvbnRyb2xsZXI6IENvbnRyb2xsZXIsXG4pIHtcbiAgbGV0IHZhbHVlOiBhbnk7XG4gIGlmICh0eXBlb2YgYWN0aW9uLnZhbHVlID09PSAnZnVuY3Rpb24nKSB7XG4gICAgY29uc3QgcHJldmlvdXNWYWx1ZSA9IGNvbnRyb2xsZXIuZ2V0KGFjdGlvbi5zY2hlbWEsIC4uLmFjdGlvbi5hcmdzLCBzdGF0ZSk7XG4gICAgaWYgKHByZXZpb3VzVmFsdWUgPT09IHVuZGVmaW5lZCkgcmV0dXJuIHN0YXRlO1xuICAgIHZhbHVlID0gYWN0aW9uLnZhbHVlKHByZXZpb3VzVmFsdWUpO1xuICB9IGVsc2Uge1xuICAgIHZhbHVlID0gYWN0aW9uLnZhbHVlO1xuICB9XG4gIHRyeSB7XG4gICAgY29uc3QgeyBlbnRpdGllcywgaW5kZXhlcywgZW50aXRpZXNNZXRhIH0gPSBub3JtYWxpemUoXG4gICAgICBhY3Rpb24uc2NoZW1hLFxuICAgICAgdmFsdWUsXG4gICAgICBhY3Rpb24uYXJncyxcbiAgICAgIHN0YXRlLFxuICAgICAgYWN0aW9uLm1ldGEsXG4gICAgKTtcbiAgICByZXR1cm4ge1xuICAgICAgZW50aXRpZXMsXG4gICAgICBlbmRwb2ludHM6IHN0YXRlLmVuZHBvaW50cyxcbiAgICAgIGluZGV4ZXMsXG4gICAgICBtZXRhOiBzdGF0ZS5tZXRhLFxuICAgICAgZW50aXRpZXNNZXRhLFxuICAgICAgb3B0aW1pc3RpYzogc3RhdGUub3B0aW1pc3RpYyxcbiAgICAgIGxhc3RSZXNldDogc3RhdGUubGFzdFJlc2V0LFxuICAgIH07XG4gICAgLy8gcmVkdWNlciBtdXN0IHVwZGF0ZSB0aGUgc3RhdGUsIHNvIGluIGNhc2Ugb2YgcHJvY2Vzc2luZyBlcnJvcnMgd2Ugc2ltcGx5IGNvbXB1dGUgdGhlIGVuZHBvaW50cyBpbmxpbmVcbiAgfSBjYXRjaCAoZXJyb3I6IGFueSkge1xuICAgIC8vIHRoaXMgaXMgbm90IGFsd2F5cyBidWJibGVkIHVwLCBzbyBsZXQncyBkb3VibGUgc3VyZSB0aGlzIGRvZXNuJ3QgZmFpbCBzaWxlbnRseVxuICAgIC8qIGlzdGFuYnVsIGlnbm9yZSBlbHNlICovXG4gICAgaWYgKHByb2Nlc3MuZW52Lk5PREVfRU5WICE9PSAncHJvZHVjdGlvbicpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgIH1cbiAgICByZXR1cm4gc3RhdGU7XG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsU0FBUyxRQUFRLHdCQUF3QjtBQUtsRCxPQUFPLFNBQVNDLFVBQVVBLENBQ3hCQyxLQUFxQixFQUNyQkMsTUFBaUIsRUFDakJDLFVBQXNCLEVBQ3RCO0VBQ0EsSUFBSUMsS0FBVTtFQUNkLElBQUksT0FBT0YsTUFBTSxDQUFDRSxLQUFLLEtBQUssVUFBVSxFQUFFO0lBQ3RDLE1BQU1DLGFBQWEsR0FBR0YsVUFBVSxDQUFDRyxHQUFHLENBQUNKLE1BQU0sQ0FBQ0ssTUFBTSxFQUFFLEdBQUdMLE1BQU0sQ0FBQ00sSUFBSSxFQUFFUCxLQUFLLENBQUM7SUFDMUUsSUFBSUksYUFBYSxLQUFLSSxTQUFTLEVBQUUsT0FBT1IsS0FBSztJQUM3Q0csS0FBSyxHQUFHRixNQUFNLENBQUNFLEtBQUssQ0FBQ0MsYUFBYSxDQUFDO0VBQ3JDLENBQUMsTUFBTTtJQUNMRCxLQUFLLEdBQUdGLE1BQU0sQ0FBQ0UsS0FBSztFQUN0QjtFQUNBLElBQUk7SUFDRixNQUFNO01BQUVNLFFBQVE7TUFBRUMsT0FBTztNQUFFQztJQUFhLENBQUMsR0FBR2IsU0FBUyxDQUNuREcsTUFBTSxDQUFDSyxNQUFNLEVBQ2JILEtBQUssRUFDTEYsTUFBTSxDQUFDTSxJQUFJLEVBQ1hQLEtBQUssRUFDTEMsTUFBTSxDQUFDVyxJQUNULENBQUM7SUFDRCxPQUFPO01BQ0xILFFBQVE7TUFDUkksU0FBUyxFQUFFYixLQUFLLENBQUNhLFNBQVM7TUFDMUJILE9BQU87TUFDUEUsSUFBSSxFQUFFWixLQUFLLENBQUNZLElBQUk7TUFDaEJELFlBQVk7TUFDWkcsVUFBVSxFQUFFZCxLQUFLLENBQUNjLFVBQVU7TUFDNUJDLFNBQVMsRUFBRWYsS0FBSyxDQUFDZTtJQUNuQixDQUFDO0lBQ0Q7RUFDRixDQUFDLENBQUMsT0FBT0MsS0FBVSxFQUFFO0lBQ25CO0lBQ0E7SUFDQSxJQUFJQyxPQUFPLENBQUNDLEdBQUcsQ0FBQ0MsUUFBUSxLQUFLLFlBQVksRUFBRTtNQUN6Q0MsT0FBTyxDQUFDSixLQUFLLENBQUNBLEtBQUssQ0FBQztJQUN0QjtJQUNBLE9BQU9oQixLQUFLO0VBQ2Q7QUFDRiIsImlnbm9yZUxpc3QiOltdfQ==
36
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJub3JtYWxpemUiLCJzZXRSZWR1Y2VyIiwic3RhdGUiLCJhY3Rpb24iLCJjb250cm9sbGVyIiwidmFsdWUiLCJwcmV2aW91c1ZhbHVlIiwiZ2V0Iiwic2NoZW1hIiwiYXJncyIsInVuZGVmaW5lZCIsImVudGl0aWVzIiwiaW5kZXhlcyIsImVudGl0aWVzTWV0YSIsIm1ldGEiLCJlbmRwb2ludHMiLCJvcHRpbWlzdGljIiwibGFzdFJlc2V0IiwiZXJyb3IiLCJwcm9jZXNzIiwiZW52IiwiTk9ERV9FTlYiLCJjb25zb2xlIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0YXRlL3JlZHVjZXIvc2V0UmVkdWNlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBub3JtYWxpemUgfSBmcm9tICdAZGF0YS1jbGllbnQvbm9ybWFsaXpyJztcblxuaW1wb3J0IENvbnRyb2xsZXIgZnJvbSAnLi4vLi4vY29udHJvbGxlci9Db250cm9sbGVyLmpzJztcbmltcG9ydCB0eXBlIHsgU3RhdGUsIFNldEFjdGlvbiB9IGZyb20gJy4uLy4uL3R5cGVzLmpzJztcblxuZXhwb3J0IGZ1bmN0aW9uIHNldFJlZHVjZXIoXG4gIHN0YXRlOiBTdGF0ZTx1bmtub3duPixcbiAgYWN0aW9uOiBTZXRBY3Rpb24sXG4gIGNvbnRyb2xsZXI6IENvbnRyb2xsZXIsXG4pOiBTdGF0ZTx1bmtub3duPiB7XG4gIGxldCB2YWx1ZTogYW55O1xuICBpZiAodHlwZW9mIGFjdGlvbi52YWx1ZSA9PT0gJ2Z1bmN0aW9uJykge1xuICAgIGNvbnN0IHByZXZpb3VzVmFsdWUgPSBjb250cm9sbGVyLmdldChhY3Rpb24uc2NoZW1hLCAuLi5hY3Rpb24uYXJncywgc3RhdGUpO1xuICAgIGlmIChwcmV2aW91c1ZhbHVlID09PSB1bmRlZmluZWQpIHJldHVybiBzdGF0ZTtcbiAgICB2YWx1ZSA9IGFjdGlvbi52YWx1ZShwcmV2aW91c1ZhbHVlKTtcbiAgfSBlbHNlIHtcbiAgICB2YWx1ZSA9IGFjdGlvbi52YWx1ZTtcbiAgfVxuICB0cnkge1xuICAgIGNvbnN0IHsgZW50aXRpZXMsIGluZGV4ZXMsIGVudGl0aWVzTWV0YSB9ID0gbm9ybWFsaXplKFxuICAgICAgYWN0aW9uLnNjaGVtYSxcbiAgICAgIHZhbHVlLFxuICAgICAgYWN0aW9uLmFyZ3MsXG4gICAgICBzdGF0ZSxcbiAgICAgIGFjdGlvbi5tZXRhLFxuICAgICk7XG4gICAgcmV0dXJuIHtcbiAgICAgIGVudGl0aWVzLFxuICAgICAgZW5kcG9pbnRzOiBzdGF0ZS5lbmRwb2ludHMsXG4gICAgICBpbmRleGVzLFxuICAgICAgbWV0YTogc3RhdGUubWV0YSxcbiAgICAgIGVudGl0aWVzTWV0YSxcbiAgICAgIG9wdGltaXN0aWM6IHN0YXRlLm9wdGltaXN0aWMsXG4gICAgICBsYXN0UmVzZXQ6IHN0YXRlLmxhc3RSZXNldCxcbiAgICB9O1xuICAgIC8vIHJlZHVjZXIgbXVzdCB1cGRhdGUgdGhlIHN0YXRlLCBzbyBpbiBjYXNlIG9mIHByb2Nlc3NpbmcgZXJyb3JzIHdlIHNpbXBseSBjb21wdXRlIHRoZSBlbmRwb2ludHMgaW5saW5lXG4gIH0gY2F0Y2ggKGVycm9yOiBhbnkpIHtcbiAgICAvLyB0aGlzIGlzIG5vdCBhbHdheXMgYnViYmxlZCB1cCwgc28gbGV0J3MgZG91YmxlIHN1cmUgdGhpcyBkb2Vzbid0IGZhaWwgc2lsZW50bHlcbiAgICAvKiBpc3RhbmJ1bCBpZ25vcmUgZWxzZSAqL1xuICAgIGlmIChwcm9jZXNzLmVudi5OT0RFX0VOViAhPT0gJ3Byb2R1Y3Rpb24nKSB7XG4gICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICB9XG4gICAgcmV0dXJuIHN0YXRlO1xuICB9XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFNBQVMsUUFBUSx3QkFBd0I7QUFLbEQsT0FBTyxTQUFTQyxVQUFVQSxDQUN4QkMsS0FBcUIsRUFDckJDLE1BQWlCLEVBQ2pCQyxVQUFzQixFQUNOO0VBQ2hCLElBQUlDLEtBQVU7RUFDZCxJQUFJLE9BQU9GLE1BQU0sQ0FBQ0UsS0FBSyxLQUFLLFVBQVUsRUFBRTtJQUN0QyxNQUFNQyxhQUFhLEdBQUdGLFVBQVUsQ0FBQ0csR0FBRyxDQUFDSixNQUFNLENBQUNLLE1BQU0sRUFBRSxHQUFHTCxNQUFNLENBQUNNLElBQUksRUFBRVAsS0FBSyxDQUFDO0lBQzFFLElBQUlJLGFBQWEsS0FBS0ksU0FBUyxFQUFFLE9BQU9SLEtBQUs7SUFDN0NHLEtBQUssR0FBR0YsTUFBTSxDQUFDRSxLQUFLLENBQUNDLGFBQWEsQ0FBQztFQUNyQyxDQUFDLE1BQU07SUFDTEQsS0FBSyxHQUFHRixNQUFNLENBQUNFLEtBQUs7RUFDdEI7RUFDQSxJQUFJO0lBQ0YsTUFBTTtNQUFFTSxRQUFRO01BQUVDLE9BQU87TUFBRUM7SUFBYSxDQUFDLEdBQUdiLFNBQVMsQ0FDbkRHLE1BQU0sQ0FBQ0ssTUFBTSxFQUNiSCxLQUFLLEVBQ0xGLE1BQU0sQ0FBQ00sSUFBSSxFQUNYUCxLQUFLLEVBQ0xDLE1BQU0sQ0FBQ1csSUFDVCxDQUFDO0lBQ0QsT0FBTztNQUNMSCxRQUFRO01BQ1JJLFNBQVMsRUFBRWIsS0FBSyxDQUFDYSxTQUFTO01BQzFCSCxPQUFPO01BQ1BFLElBQUksRUFBRVosS0FBSyxDQUFDWSxJQUFJO01BQ2hCRCxZQUFZO01BQ1pHLFVBQVUsRUFBRWQsS0FBSyxDQUFDYyxVQUFVO01BQzVCQyxTQUFTLEVBQUVmLEtBQUssQ0FBQ2U7SUFDbkIsQ0FBQztJQUNEO0VBQ0YsQ0FBQyxDQUFDLE9BQU9DLEtBQVUsRUFBRTtJQUNuQjtJQUNBO0lBQ0EsSUFBSUMsT0FBTyxDQUFDQyxHQUFHLENBQUNDLFFBQVEsS0FBSyxZQUFZLEVBQUU7TUFDekNDLE9BQU8sQ0FBQ0osS0FBSyxDQUFDQSxLQUFLLENBQUM7SUFDdEI7SUFDQSxPQUFPaEIsS0FBSztFQUNkO0FBQ0YiLCJpZ25vcmVMaXN0IjpbXX0=
@@ -1,33 +1,4 @@
1
1
  import type Controller from '../../controller/Controller.js';
2
2
  import type { State, SetResponseAction, OptimisticAction } from '../../types.js';
3
- export declare function setResponseReducer(state: State<unknown>, action: OptimisticAction | SetResponseAction, controller: Controller): State<unknown> | {
4
- entities: {
5
- [x: string]: /*elided*/ any;
6
- };
7
- endpoints: Record<string, unknown>;
8
- indexes: import("@data-client/normalizr").NormalizedIndex;
9
- meta: {
10
- [x: string]: {
11
- readonly date: number;
12
- readonly fetchedAt: number;
13
- readonly expiresAt: number;
14
- readonly prevExpiresAt?: number;
15
- readonly error?: import("@data-client/normalizr").ErrorTypes;
16
- readonly invalidated?: boolean;
17
- readonly errorPolicy?: "hard" | "soft" | undefined;
18
- } | {
19
- date: number;
20
- fetchedAt: number;
21
- expiresAt: number;
22
- prevExpiresAt: number;
23
- };
24
- };
25
- entitiesMeta: import("packages/normalizr/lib/types.js").EntitiesToMeta<{
26
- [x: string]: /*elided*/ any;
27
- }>;
28
- optimistic: (OptimisticAction<import("@data-client/normalizr").EndpointInterface<import("@data-client/normalizr").FetchFunction, import("@data-client/normalizr").Schema | undefined, boolean | undefined> & {
29
- update?: import("../../index.js").EndpointUpdateFunction<import("@data-client/normalizr").EndpointInterface>;
30
- }> | SetResponseAction)[];
31
- lastReset: number;
32
- };
3
+ export declare function setResponseReducer(state: State<unknown>, action: OptimisticAction | SetResponseAction, controller: Controller): State<unknown>;
33
4
  //# sourceMappingURL=setResponseReducer.d.ts.map
@@ -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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA2FvB"}
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,GACrB,KAAK,CAAC,OAAO,CAAC,CA0FhB"}
@@ -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(`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":[]}
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): State<unknown> {\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.4",
3
+ "version": "0.16.0",
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.4",
137
+ "@data-client/normalizr": "^0.16.0",
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.4",
142
+ "@data-client/endpoint": "0.16.0",
143
143
  "@types/jest": "30.0.0",
144
144
  "@types/node": "^24.0.0",
145
145
  "rollup-plugins": "1.0.0"
@@ -1,4 +1,4 @@
1
- import { Entity, schema } from '@data-client/endpoint';
1
+ import { Entity, Collection, All, Query, Union } from '@data-client/endpoint';
2
2
 
3
3
  import { initialState } from '../../state/reducer/createReducer';
4
4
  import { State } from '../../types';
@@ -8,7 +8,7 @@ class Tacos extends Entity {
8
8
  type = '';
9
9
  id = '';
10
10
  }
11
- const TacoList = new schema.Collection([Tacos]);
11
+ const TacoList = new Collection([Tacos]);
12
12
  const entities = {
13
13
  Tacos: {
14
14
  1: { id: '1', type: 'foo' },
@@ -188,7 +188,7 @@ describe('Controller.get()', () => {
188
188
  ...initialState,
189
189
  entities,
190
190
  };
191
- const AllTacos = new schema.All(Tacos);
191
+ const AllTacos = new All(Tacos);
192
192
 
193
193
  it('should get all entities', () => {
194
194
  const allTacos = controller.get(AllTacos, state);
@@ -229,8 +229,8 @@ describe('Controller.get()', () => {
229
229
  ...initialState,
230
230
  entities,
231
231
  };
232
- const queryTacos = new schema.Query(
233
- new schema.All(Tacos),
232
+ const queryTacos = new Query(
233
+ new All(Tacos),
234
234
  (tacos, { type }: { type?: string } = {}) => {
235
235
  if (!type) return tacos;
236
236
  return tacos.filter(taco => taco.type === type);
@@ -281,7 +281,7 @@ describe('Controller.get()', () => {
281
281
  ...initialState,
282
282
  entities,
283
283
  };
284
- const tacoCount = new schema.Query(TacoList, tacos => {
284
+ const tacoCount = new Query(TacoList, tacos => {
285
285
  return tacos.length ?? 0;
286
286
  });
287
287
 
@@ -303,7 +303,7 @@ describe('Controller.get()', () => {
303
303
  groupname: string = '';
304
304
  memberCount = 0;
305
305
  }
306
- const queryPerson = new schema.Union(
306
+ const queryPerson = new Union(
307
307
  {
308
308
  users: User,
309
309
  groups: Group,
@@ -363,7 +363,7 @@ describe('Controller.get()', () => {
363
363
  groupname: string = '';
364
364
  memberCount = 0;
365
365
  }
366
- const queryPerson = new schema.Union(
366
+ const queryPerson = new Union(
367
367
  {
368
368
  users: User,
369
369
  groups: Group,
@@ -3,6 +3,7 @@ import type { EndpointInterface } from '@data-client/normalizr';
3
3
  import { createMeta } from './createMeta.js';
4
4
  import { OPTIMISTIC } from '../../actionTypes.js';
5
5
  import type { OptimisticAction } from '../../types.js';
6
+ import ensurePojo from '../ensurePojo.js';
6
7
  import type { EndpointUpdateFunction } from '../types.js';
7
8
 
8
9
  export function createOptimistic<
@@ -25,7 +26,7 @@ export function createOptimistic<
25
26
  return {
26
27
  type: OPTIMISTIC,
27
28
  key: endpoint.key(...args),
28
- args,
29
+ args: args.map(ensurePojo),
29
30
  endpoint,
30
31
  meta: createMeta(endpoint.dataExpiryLength ?? 60000, fetchedAt),
31
32
  };
@@ -1,7 +1,19 @@
1
+ /** Listens to online/offline events for triggering re-fetches on reconnect.
2
+ *
3
+ * Implement this interface to provide custom connectivity detection
4
+ * (e.g., for React Native or Node.js environments).
5
+ *
6
+ * @see https://dataclient.io/docs/api/PollingSubscription
7
+ */
1
8
  export default interface ConnectionListener {
9
+ /** Returns whether the client is currently connected to the network. */
2
10
  isOnline: () => boolean;
11
+ /** Register a handler to be called when the client comes back online. */
3
12
  addOnlineListener: (handler: () => void) => void;
13
+ /** Remove a previously registered online handler. */
4
14
  removeOnlineListener: (handler: () => void) => void;
15
+ /** Register a handler to be called when the client goes offline. */
5
16
  addOfflineListener: (handler: () => void) => void;
17
+ /** Remove a previously registered offline handler. */
6
18
  removeOfflineListener: (handler: () => void) => void;
7
19
  }
@@ -94,6 +94,7 @@ export default class DevToolsManager implements Manager {
94
94
  protected actions: [ActionTypes, State<unknown>][] = [];
95
95
  declare protected controller: Controller;
96
96
  declare skipLogging?: (action: ActionTypes) => boolean;
97
+ declare devtoolsName: string;
97
98
  maxBufferLength = 100;
98
99
 
99
100
  constructor(
@@ -101,13 +102,13 @@ export default class DevToolsManager implements Manager {
101
102
  skipLogging?: (action: ActionTypes) => boolean,
102
103
  ) {
103
104
  /* istanbul ignore next */
105
+ const options = { ...DEFAULT_CONFIG, ...config };
106
+ this.devtoolsName =
107
+ options.name ?? `Data Client: ${globalThis.document?.title}`;
104
108
  this.devTools =
105
109
  typeof window !== 'undefined' &&
106
110
  (window as any).__REDUX_DEVTOOLS_EXTENSION__ &&
107
- (window as any).__REDUX_DEVTOOLS_EXTENSION__.connect({
108
- ...DEFAULT_CONFIG,
109
- ...config,
110
- });
111
+ (window as any).__REDUX_DEVTOOLS_EXTENSION__.connect(options);
111
112
  // we cut it in half so we should double so we don't lose
112
113
  if (config?.maxAge) this.maxBufferLength = config.maxAge * 2;
113
114
  if (skipLogging) this.skipLogging = skipLogging;
@@ -118,8 +119,8 @@ export default class DevToolsManager implements Manager {
118
119
  /* istanbul ignore next */
119
120
  if (process.env.NODE_ENV !== 'production') {
120
121
  this.prototype.middleware = function (controller) {
121
- if (!this.devTools) return next => action => next(action);
122
122
  this.controller = controller;
123
+ if (!this.devTools) return next => action => next(action);
123
124
  const reducer = createReducer(controller as any);
124
125
  let state = controller.getState();
125
126
  return next => action => {
@@ -158,6 +159,12 @@ export default class DevToolsManager implements Manager {
158
159
 
159
160
  /** Called when initial state is ready */
160
161
  init(state: State<any>) {
162
+ if (process.env.NODE_ENV !== 'production') {
163
+ ((globalThis as any).__DC_CONTROLLERS__ ??= new Map()).set(
164
+ this.devtoolsName,
165
+ this.controller,
166
+ );
167
+ }
161
168
  if (process.env.NODE_ENV !== 'production' && this.devTools) {
162
169
  this.devTools.init(state);
163
170
  this.devTools.subscribe((msg: any) => {
@@ -186,5 +193,13 @@ export default class DevToolsManager implements Manager {
186
193
  }
187
194
 
188
195
  /** Ensures all subscriptions are cleaned up. */
189
- cleanup() {}
196
+ cleanup() {
197
+ if (process.env.NODE_ENV !== 'production') {
198
+ const map: Map<string, Controller> | undefined = (globalThis as any)
199
+ .__DC_CONTROLLERS__;
200
+ if (map?.get(this.devtoolsName) === this.controller) {
201
+ map.delete(this.devtoolsName);
202
+ }
203
+ }
204
+ }
190
205
  }
@@ -251,7 +251,7 @@ export default class NetworkManager implements Manager {
251
251
  * create a new promise and call fetch.
252
252
  *
253
253
  * Note: The new promise is not actually tied to fetch at all,
254
- * but is resolved when the expected 'recieve' action is processed.
254
+ * but is resolved when the expected 'receive' action is processed.
255
255
  * This ensures promises are resolved only once their data is processed
256
256
  * by the reducer.
257
257
  */
@@ -34,6 +34,9 @@ export class GCPolicy implements GCInterface {
34
34
  this.intervalId = setInterval(() => {
35
35
  this.idleCallback(() => this.runSweep(), { timeout: 1000 });
36
36
  }, this.options.intervalMS);
37
+ if (typeof this.intervalId === 'object' && 'unref' in this.intervalId) {
38
+ this.intervalId.unref();
39
+ }
37
40
  }
38
41
 
39
42
  cleanup() {