@kingsy/objectloader2 2.25.7

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 (256) hide show
  1. package/.tshy/build.json +8 -0
  2. package/.tshy/commonjs.json +17 -0
  3. package/.tshy/esm.json +16 -0
  4. package/dist/commonjs/helpers/aggregateQueue.d.ts +8 -0
  5. package/dist/commonjs/helpers/aggregateQueue.d.ts.map +1 -0
  6. package/dist/commonjs/helpers/aggregateQueue.js +19 -0
  7. package/dist/commonjs/helpers/aggregateQueue.js.map +1 -0
  8. package/dist/commonjs/helpers/asyncGeneratorQueue.d.ts +8 -0
  9. package/dist/commonjs/helpers/asyncGeneratorQueue.d.ts.map +1 -0
  10. package/dist/commonjs/helpers/asyncGeneratorQueue.js +35 -0
  11. package/dist/commonjs/helpers/asyncGeneratorQueue.js.map +1 -0
  12. package/dist/commonjs/helpers/batchedPool.d.ts +12 -0
  13. package/dist/commonjs/helpers/batchedPool.d.ts.map +1 -0
  14. package/dist/commonjs/helpers/batchedPool.js +45 -0
  15. package/dist/commonjs/helpers/batchedPool.js.map +1 -0
  16. package/dist/commonjs/helpers/batchingQueue.d.ts +14 -0
  17. package/dist/commonjs/helpers/batchingQueue.d.ts.map +1 -0
  18. package/dist/commonjs/helpers/batchingQueue.js +77 -0
  19. package/dist/commonjs/helpers/batchingQueue.js.map +1 -0
  20. package/dist/commonjs/helpers/bufferQueue.d.ts +7 -0
  21. package/dist/commonjs/helpers/bufferQueue.d.ts.map +1 -0
  22. package/dist/commonjs/helpers/bufferQueue.js +13 -0
  23. package/dist/commonjs/helpers/bufferQueue.js.map +1 -0
  24. package/dist/commonjs/helpers/cachePump.d.ts +22 -0
  25. package/dist/commonjs/helpers/cachePump.d.ts.map +1 -0
  26. package/dist/commonjs/helpers/cachePump.js +86 -0
  27. package/dist/commonjs/helpers/cachePump.js.map +1 -0
  28. package/dist/commonjs/helpers/cacheReader.d.ts +14 -0
  29. package/dist/commonjs/helpers/cacheReader.d.ts.map +1 -0
  30. package/dist/commonjs/helpers/cacheReader.js +58 -0
  31. package/dist/commonjs/helpers/cacheReader.js.map +1 -0
  32. package/dist/commonjs/helpers/defermentManager.d.ts +28 -0
  33. package/dist/commonjs/helpers/defermentManager.d.ts.map +1 -0
  34. package/dist/commonjs/helpers/defermentManager.js +150 -0
  35. package/dist/commonjs/helpers/defermentManager.js.map +1 -0
  36. package/dist/commonjs/helpers/deferredBase.d.ts +19 -0
  37. package/dist/commonjs/helpers/deferredBase.d.ts.map +1 -0
  38. package/dist/commonjs/helpers/deferredBase.js +51 -0
  39. package/dist/commonjs/helpers/deferredBase.js.map +1 -0
  40. package/dist/commonjs/helpers/keyedQueue.d.ts +11 -0
  41. package/dist/commonjs/helpers/keyedQueue.d.ts.map +1 -0
  42. package/dist/commonjs/helpers/keyedQueue.js +41 -0
  43. package/dist/commonjs/helpers/keyedQueue.js.map +1 -0
  44. package/dist/commonjs/helpers/memoryPump.d.ts +15 -0
  45. package/dist/commonjs/helpers/memoryPump.d.ts.map +1 -0
  46. package/dist/commonjs/helpers/memoryPump.js +34 -0
  47. package/dist/commonjs/helpers/memoryPump.js.map +1 -0
  48. package/dist/commonjs/helpers/pump.d.ts +8 -0
  49. package/dist/commonjs/helpers/pump.d.ts.map +1 -0
  50. package/dist/commonjs/helpers/pump.js +3 -0
  51. package/dist/commonjs/helpers/pump.js.map +1 -0
  52. package/dist/commonjs/helpers/queue.d.ts +4 -0
  53. package/dist/commonjs/helpers/queue.d.ts.map +1 -0
  54. package/dist/commonjs/helpers/queue.js +3 -0
  55. package/dist/commonjs/helpers/queue.js.map +1 -0
  56. package/dist/commonjs/index.d.ts +3 -0
  57. package/dist/commonjs/index.d.ts.map +1 -0
  58. package/dist/commonjs/index.js +8 -0
  59. package/dist/commonjs/index.js.map +1 -0
  60. package/dist/commonjs/operations/databases/indexedDatabase.d.ts +27 -0
  61. package/dist/commonjs/operations/databases/indexedDatabase.d.ts.map +1 -0
  62. package/dist/commonjs/operations/databases/indexedDatabase.js +98 -0
  63. package/dist/commonjs/operations/databases/indexedDatabase.js.map +1 -0
  64. package/dist/commonjs/operations/databases/memoryDatabase.d.ts +13 -0
  65. package/dist/commonjs/operations/databases/memoryDatabase.d.ts.map +1 -0
  66. package/dist/commonjs/operations/databases/memoryDatabase.js +33 -0
  67. package/dist/commonjs/operations/databases/memoryDatabase.js.map +1 -0
  68. package/dist/commonjs/operations/downloaders/memoryDownloader.d.ts +16 -0
  69. package/dist/commonjs/operations/downloaders/memoryDownloader.d.ts.map +1 -0
  70. package/dist/commonjs/operations/downloaders/memoryDownloader.js +35 -0
  71. package/dist/commonjs/operations/downloaders/memoryDownloader.js.map +1 -0
  72. package/dist/commonjs/operations/downloaders/serverDownloader.d.ts +32 -0
  73. package/dist/commonjs/operations/downloaders/serverDownloader.d.ts.map +1 -0
  74. package/dist/commonjs/operations/downloaders/serverDownloader.js +169 -0
  75. package/dist/commonjs/operations/downloaders/serverDownloader.js.map +1 -0
  76. package/dist/commonjs/operations/interfaces.d.ts +19 -0
  77. package/dist/commonjs/operations/interfaces.d.ts.map +1 -0
  78. package/dist/commonjs/operations/interfaces.js +3 -0
  79. package/dist/commonjs/operations/interfaces.js.map +1 -0
  80. package/dist/commonjs/operations/objectLoader2.d.ts +16 -0
  81. package/dist/commonjs/operations/objectLoader2.d.ts.map +1 -0
  82. package/dist/commonjs/operations/objectLoader2.js +101 -0
  83. package/dist/commonjs/operations/objectLoader2.js.map +1 -0
  84. package/dist/commonjs/operations/objectLoader2Factory.d.ts +25 -0
  85. package/dist/commonjs/operations/objectLoader2Factory.d.ts.map +1 -0
  86. package/dist/commonjs/operations/objectLoader2Factory.js +68 -0
  87. package/dist/commonjs/operations/objectLoader2Factory.js.map +1 -0
  88. package/dist/commonjs/operations/options.d.ts +26 -0
  89. package/dist/commonjs/operations/options.d.ts.map +1 -0
  90. package/dist/commonjs/operations/options.js +3 -0
  91. package/dist/commonjs/operations/options.js.map +1 -0
  92. package/dist/commonjs/operations/traverser.d.ts +19 -0
  93. package/dist/commonjs/operations/traverser.d.ts.map +1 -0
  94. package/dist/commonjs/operations/traverser.js +96 -0
  95. package/dist/commonjs/operations/traverser.js.map +1 -0
  96. package/dist/commonjs/package.json +3 -0
  97. package/dist/commonjs/types/errors.d.ts +21 -0
  98. package/dist/commonjs/types/errors.d.ts.map +1 -0
  99. package/dist/commonjs/types/errors.js +28 -0
  100. package/dist/commonjs/types/errors.js.map +1 -0
  101. package/dist/commonjs/types/types.d.ts +25 -0
  102. package/dist/commonjs/types/types.d.ts.map +1 -0
  103. package/dist/commonjs/types/types.js +39 -0
  104. package/dist/commonjs/types/types.js.map +1 -0
  105. package/dist/esm/helpers/aggregateQueue.d.ts +8 -0
  106. package/dist/esm/helpers/aggregateQueue.d.ts.map +1 -0
  107. package/dist/esm/helpers/aggregateQueue.js +16 -0
  108. package/dist/esm/helpers/aggregateQueue.js.map +1 -0
  109. package/dist/esm/helpers/asyncGeneratorQueue.d.ts +8 -0
  110. package/dist/esm/helpers/asyncGeneratorQueue.d.ts.map +1 -0
  111. package/dist/esm/helpers/asyncGeneratorQueue.js +32 -0
  112. package/dist/esm/helpers/asyncGeneratorQueue.js.map +1 -0
  113. package/dist/esm/helpers/batchedPool.d.ts +12 -0
  114. package/dist/esm/helpers/batchedPool.d.ts.map +1 -0
  115. package/dist/esm/helpers/batchedPool.js +42 -0
  116. package/dist/esm/helpers/batchedPool.js.map +1 -0
  117. package/dist/esm/helpers/batchingQueue.d.ts +14 -0
  118. package/dist/esm/helpers/batchingQueue.d.ts.map +1 -0
  119. package/dist/esm/helpers/batchingQueue.js +71 -0
  120. package/dist/esm/helpers/batchingQueue.js.map +1 -0
  121. package/dist/esm/helpers/bufferQueue.d.ts +7 -0
  122. package/dist/esm/helpers/bufferQueue.d.ts.map +1 -0
  123. package/dist/esm/helpers/bufferQueue.js +10 -0
  124. package/dist/esm/helpers/bufferQueue.js.map +1 -0
  125. package/dist/esm/helpers/cachePump.d.ts +22 -0
  126. package/dist/esm/helpers/cachePump.d.ts.map +1 -0
  127. package/dist/esm/helpers/cachePump.js +79 -0
  128. package/dist/esm/helpers/cachePump.js.map +1 -0
  129. package/dist/esm/helpers/cacheReader.d.ts +14 -0
  130. package/dist/esm/helpers/cacheReader.d.ts.map +1 -0
  131. package/dist/esm/helpers/cacheReader.js +51 -0
  132. package/dist/esm/helpers/cacheReader.js.map +1 -0
  133. package/dist/esm/helpers/defermentManager.d.ts +28 -0
  134. package/dist/esm/helpers/defermentManager.d.ts.map +1 -0
  135. package/dist/esm/helpers/defermentManager.js +146 -0
  136. package/dist/esm/helpers/defermentManager.js.map +1 -0
  137. package/dist/esm/helpers/deferredBase.d.ts +19 -0
  138. package/dist/esm/helpers/deferredBase.d.ts.map +1 -0
  139. package/dist/esm/helpers/deferredBase.js +47 -0
  140. package/dist/esm/helpers/deferredBase.js.map +1 -0
  141. package/dist/esm/helpers/keyedQueue.d.ts +11 -0
  142. package/dist/esm/helpers/keyedQueue.d.ts.map +1 -0
  143. package/dist/esm/helpers/keyedQueue.js +38 -0
  144. package/dist/esm/helpers/keyedQueue.js.map +1 -0
  145. package/dist/esm/helpers/memoryPump.d.ts +15 -0
  146. package/dist/esm/helpers/memoryPump.d.ts.map +1 -0
  147. package/dist/esm/helpers/memoryPump.js +30 -0
  148. package/dist/esm/helpers/memoryPump.js.map +1 -0
  149. package/dist/esm/helpers/pump.d.ts +8 -0
  150. package/dist/esm/helpers/pump.d.ts.map +1 -0
  151. package/dist/esm/helpers/pump.js +2 -0
  152. package/dist/esm/helpers/pump.js.map +1 -0
  153. package/dist/esm/helpers/queue.d.ts +4 -0
  154. package/dist/esm/helpers/queue.d.ts.map +1 -0
  155. package/dist/esm/helpers/queue.js +2 -0
  156. package/dist/esm/helpers/queue.js.map +1 -0
  157. package/dist/esm/index.d.ts +3 -0
  158. package/dist/esm/index.d.ts.map +1 -0
  159. package/dist/esm/index.js +3 -0
  160. package/dist/esm/index.js.map +1 -0
  161. package/dist/esm/operations/databases/indexedDatabase.d.ts +27 -0
  162. package/dist/esm/operations/databases/indexedDatabase.d.ts.map +1 -0
  163. package/dist/esm/operations/databases/indexedDatabase.js +93 -0
  164. package/dist/esm/operations/databases/indexedDatabase.js.map +1 -0
  165. package/dist/esm/operations/databases/memoryDatabase.d.ts +13 -0
  166. package/dist/esm/operations/databases/memoryDatabase.d.ts.map +1 -0
  167. package/dist/esm/operations/databases/memoryDatabase.js +29 -0
  168. package/dist/esm/operations/databases/memoryDatabase.js.map +1 -0
  169. package/dist/esm/operations/downloaders/memoryDownloader.d.ts +16 -0
  170. package/dist/esm/operations/downloaders/memoryDownloader.d.ts.map +1 -0
  171. package/dist/esm/operations/downloaders/memoryDownloader.js +31 -0
  172. package/dist/esm/operations/downloaders/memoryDownloader.js.map +1 -0
  173. package/dist/esm/operations/downloaders/serverDownloader.d.ts +32 -0
  174. package/dist/esm/operations/downloaders/serverDownloader.d.ts.map +1 -0
  175. package/dist/esm/operations/downloaders/serverDownloader.js +163 -0
  176. package/dist/esm/operations/downloaders/serverDownloader.js.map +1 -0
  177. package/dist/esm/operations/interfaces.d.ts +19 -0
  178. package/dist/esm/operations/interfaces.d.ts.map +1 -0
  179. package/dist/esm/operations/interfaces.js +2 -0
  180. package/dist/esm/operations/interfaces.js.map +1 -0
  181. package/dist/esm/operations/objectLoader2.d.ts +16 -0
  182. package/dist/esm/operations/objectLoader2.d.ts.map +1 -0
  183. package/dist/esm/operations/objectLoader2.js +94 -0
  184. package/dist/esm/operations/objectLoader2.js.map +1 -0
  185. package/dist/esm/operations/objectLoader2Factory.d.ts +25 -0
  186. package/dist/esm/operations/objectLoader2Factory.d.ts.map +1 -0
  187. package/dist/esm/operations/objectLoader2Factory.js +61 -0
  188. package/dist/esm/operations/objectLoader2Factory.js.map +1 -0
  189. package/dist/esm/operations/options.d.ts +26 -0
  190. package/dist/esm/operations/options.d.ts.map +1 -0
  191. package/dist/esm/operations/options.js +2 -0
  192. package/dist/esm/operations/options.js.map +1 -0
  193. package/dist/esm/operations/traverser.d.ts +19 -0
  194. package/dist/esm/operations/traverser.d.ts.map +1 -0
  195. package/dist/esm/operations/traverser.js +93 -0
  196. package/dist/esm/operations/traverser.js.map +1 -0
  197. package/dist/esm/package.json +3 -0
  198. package/dist/esm/types/errors.d.ts +21 -0
  199. package/dist/esm/types/errors.d.ts.map +1 -0
  200. package/dist/esm/types/errors.js +23 -0
  201. package/dist/esm/types/errors.js.map +1 -0
  202. package/dist/esm/types/types.d.ts +25 -0
  203. package/dist/esm/types/types.d.ts.map +1 -0
  204. package/dist/esm/types/types.js +33 -0
  205. package/dist/esm/types/types.js.map +1 -0
  206. package/eslint.config.mjs +60 -0
  207. package/package.json +76 -0
  208. package/readme.md +42 -0
  209. package/src/helpers/__snapshots__/cachePump.spec.ts.snap +31 -0
  210. package/src/helpers/__snapshots__/cacheReader.spec.ts.snap +8 -0
  211. package/src/helpers/__snapshots__/defermentManager.spec.ts.snap +8 -0
  212. package/src/helpers/aggregateQueue.ts +20 -0
  213. package/src/helpers/asyncGeneratorQueue.ts +35 -0
  214. package/src/helpers/batchedPool.ts +56 -0
  215. package/src/helpers/batchingQueue.ts +85 -0
  216. package/src/helpers/bufferQueue.ts +12 -0
  217. package/src/helpers/cachePump.disposal.spec.ts +51 -0
  218. package/src/helpers/cachePump.spec.ts +104 -0
  219. package/src/helpers/cachePump.ts +107 -0
  220. package/src/helpers/cacheReader.spec.ts +35 -0
  221. package/src/helpers/cacheReader.ts +64 -0
  222. package/src/helpers/defermentManager.defermentTotals.spec.ts +53 -0
  223. package/src/helpers/defermentManager.disposal.spec.ts +28 -0
  224. package/src/helpers/defermentManager.spec.ts +37 -0
  225. package/src/helpers/defermentManager.ts +160 -0
  226. package/src/helpers/deferredBase.ts +55 -0
  227. package/src/helpers/keyedQueue.ts +45 -0
  228. package/src/helpers/memoryPump.ts +40 -0
  229. package/src/helpers/pump.ts +8 -0
  230. package/src/helpers/queue.ts +3 -0
  231. package/src/index.ts +2 -0
  232. package/src/operations/__snapshots__/objectLoader2.spec.ts.snap +149 -0
  233. package/src/operations/__snapshots__/traverser.spec.ts.snap +45 -0
  234. package/src/operations/databases/__snapshots__/indexedDatabase.spec.ts.snap +18 -0
  235. package/src/operations/databases/indexedDatabase.spec.ts +33 -0
  236. package/src/operations/databases/indexedDatabase.ts +126 -0
  237. package/src/operations/databases/memoryDatabase.spec.ts +48 -0
  238. package/src/operations/databases/memoryDatabase.ts +35 -0
  239. package/src/operations/downloaders/__snapshots__/memoryDownloader.spec.ts.snap +12 -0
  240. package/src/operations/downloaders/__snapshots__/serverDownloader.spec.ts.snap +92 -0
  241. package/src/operations/downloaders/memoryDownloader.spec.ts +49 -0
  242. package/src/operations/downloaders/memoryDownloader.ts +39 -0
  243. package/src/operations/downloaders/serverDownloader.spec.ts +161 -0
  244. package/src/operations/downloaders/serverDownloader.ts +217 -0
  245. package/src/operations/interfaces.ts +18 -0
  246. package/src/operations/objectLoader2.spec.ts +229 -0
  247. package/src/operations/objectLoader2.ts +119 -0
  248. package/src/operations/objectLoader2Factory.ts +78 -0
  249. package/src/operations/options.ts +29 -0
  250. package/src/operations/traverser.spec.ts +58 -0
  251. package/src/operations/traverser.ts +112 -0
  252. package/src/test/e2e.spec.ts +43 -0
  253. package/src/types/errors.ts +25 -0
  254. package/src/types/types.ts +71 -0
  255. package/tsconfig.json +103 -0
  256. package/vitest.config.ts +3 -0
@@ -0,0 +1,160 @@
1
+ import { DeferredBase } from './deferredBase.js'
2
+ import { Base, CustomLogger, Item } from '../types/types.js'
3
+ import { DefermentManagerOptions } from '../operations/options.js'
4
+
5
+ export class DefermentManager {
6
+ private deferments: Map<string, DeferredBase> = new Map()
7
+ private timer?: ReturnType<typeof setTimeout>
8
+ private logger: CustomLogger
9
+ private currentSize = 0
10
+ private disposed = false
11
+ //tracks total deferment requests for each id
12
+ //this is used to prevent cleaning up deferments that are still being requested
13
+ private totalDefermentRequests: Map<string, number> = new Map()
14
+
15
+ constructor(private options: DefermentManagerOptions) {
16
+ this.resetGlobalTimer()
17
+ this.logger = options.logger || ((): void => {})
18
+ }
19
+
20
+ private now(): number {
21
+ return Date.now()
22
+ }
23
+
24
+ isDeferred(id: string): boolean {
25
+ return this.deferments.has(id)
26
+ }
27
+
28
+ get(id: string): DeferredBase | undefined {
29
+ if (this.disposed) throw new Error('DefermentManager is disposed')
30
+ return this.deferments.get(id)
31
+ }
32
+
33
+ async defer(params: { id: string }): Promise<Base> {
34
+ if (this.disposed) throw new Error('DefermentManager is disposed')
35
+ this.trackDefermentRequest(params.id)
36
+ const now = this.now()
37
+ const deferredBase = this.deferments.get(params.id)
38
+ if (deferredBase) {
39
+ deferredBase.setAccess(now)
40
+ return deferredBase.getPromise()
41
+ }
42
+ const notYetFound = new DeferredBase(
43
+ this.options.ttlms,
44
+ params.id,
45
+ now + this.options.ttlms
46
+ )
47
+ this.deferments.set(params.id, notYetFound)
48
+ return notYetFound.getPromise()
49
+ }
50
+
51
+ private trackDefermentRequest(id: string): void {
52
+ const request = this.totalDefermentRequests.get(id)
53
+ if (request) {
54
+ this.totalDefermentRequests.set(id, request + 1)
55
+ } else {
56
+ this.totalDefermentRequests.set(id, 1)
57
+ }
58
+ }
59
+
60
+ undefer(item: Item): void {
61
+ if (this.disposed) throw new Error('DefermentManager is disposed')
62
+ const now = this.now()
63
+ this.currentSize += item.size || 0
64
+ //order matters here with found before undefer
65
+ const deferredBase = this.deferments.get(item.baseId)
66
+ if (deferredBase) {
67
+ deferredBase.found(item)
68
+ deferredBase.setAccess(now)
69
+ } else {
70
+ const existing = new DeferredBase(this.options.ttlms, item.baseId, now)
71
+ existing.found(item)
72
+ this.deferments.set(item.baseId, existing)
73
+ }
74
+ }
75
+
76
+ private resetGlobalTimer(): void {
77
+ const run = (): void => {
78
+ this.cleanDeferments()
79
+ this.timer = setTimeout(run, this.options.ttlms)
80
+ }
81
+ this.timer = setTimeout(run, this.options.ttlms)
82
+ }
83
+
84
+ dispose(): void {
85
+ if (this.disposed) return
86
+ this.disposed = true
87
+ if (this.timer) {
88
+ clearTimeout(this.timer)
89
+ this.timer = undefined
90
+ }
91
+ this.clearDeferments()
92
+ }
93
+
94
+ private clearDeferments(): void {
95
+ let waiting = 0
96
+ for (const deferredBase of this.deferments.values()) {
97
+ deferredBase.done(0)
98
+ if (deferredBase.getItem() === undefined) {
99
+ waiting++
100
+ }
101
+ }
102
+ this.currentSize = 0
103
+ this.deferments.clear()
104
+ this.logger('cleared deferments, left', waiting)
105
+ }
106
+
107
+ private cleanDeferments(): void {
108
+ const maxSizeBytes = this.options.maxSizeInMb * 1024 * 1024
109
+ if (this.currentSize < maxSizeBytes) {
110
+ this.logger(
111
+ 'deferments size is ok, no need to clean',
112
+ this.currentSize,
113
+ maxSizeBytes
114
+ )
115
+ return
116
+ }
117
+ const now = this.now()
118
+ let cleaned = 0
119
+ const start = performance.now()
120
+ for (const deferredBase of Array.from(this.deferments.values())
121
+ .filter((x) => x.isExpired(now))
122
+ .sort((a, b) => this.compareMaybeBasesBySize(a.getItem(), b.getItem()))) {
123
+ if (deferredBase.done(now)) {
124
+ //if the deferment is done but has been requested multiple times,
125
+ //we do not clean it up to allow the requests to resolve
126
+ const requestCount = this.totalDefermentRequests.get(deferredBase.getId())
127
+ if (requestCount && requestCount > 1) {
128
+ return
129
+ }
130
+ this.currentSize -= deferredBase.getItem()?.size || 0
131
+ this.deferments.delete(deferredBase.getId())
132
+ cleaned++
133
+ if (this.currentSize < maxSizeBytes) {
134
+ break
135
+ }
136
+ }
137
+ }
138
+ this.logger(
139
+ 'cleaned deferments: cleaned, left, time',
140
+ cleaned,
141
+ this.deferments.size,
142
+ performance.now() - start
143
+ )
144
+ return
145
+ }
146
+
147
+ compareMaybeBasesBySize(a: Item | undefined, b: Item | undefined): number {
148
+ if (a === undefined && b === undefined) return 0
149
+ if (a === undefined) return -1
150
+ if (b === undefined) return 1
151
+ return this.compareMaybe(a.size, b.size)
152
+ }
153
+
154
+ compareMaybe(a: number | undefined, b: number | undefined): number {
155
+ if (a === undefined && b === undefined) return 0
156
+ if (a === undefined) return -1
157
+ if (b === undefined) return 1
158
+ return a - b
159
+ }
160
+ }
@@ -0,0 +1,55 @@
1
+ import { Base, Item } from '../types/types.js'
2
+
3
+ export class DeferredBase {
4
+ private promise: Promise<Base>
5
+ private resolve!: (value: Base) => void
6
+ private reject!: (reason?: Error) => void
7
+ private item?: Item
8
+
9
+ private readonly id: string
10
+ private expiresAt: number // Timestamp in ms
11
+ private ttl: number // ttl in ms
12
+
13
+ constructor(ttl: number, id: string, expiresAt: number) {
14
+ this.expiresAt = expiresAt
15
+ this.ttl = ttl
16
+ this.id = id
17
+ this.promise = new Promise<Base>((resolve, reject) => {
18
+ this.resolve = resolve
19
+ this.reject = reject
20
+ })
21
+ }
22
+
23
+ getId(): string {
24
+ return this.id
25
+ }
26
+
27
+ getItem(): Item | undefined {
28
+ return this.item
29
+ }
30
+
31
+ getPromise(): Promise<Base> {
32
+ return this.promise
33
+ }
34
+
35
+ isExpired(now: number): boolean {
36
+ return this.item !== undefined && now > this.expiresAt
37
+ }
38
+ setAccess(now: number): void {
39
+ this.expiresAt = now + this.ttl
40
+ }
41
+
42
+ found(value: Item): void {
43
+ this.item = value
44
+ this.resolve(value.base)
45
+ }
46
+ done(now: number): boolean {
47
+ if (this.item) {
48
+ this.resolve(this.item.base)
49
+ }
50
+ if (this.isExpired(now)) {
51
+ return true
52
+ }
53
+ return false
54
+ }
55
+ }
@@ -0,0 +1,45 @@
1
+ export default class KeyedQueue<K, V> {
2
+ private _map: Map<K, V>
3
+ private _order: K[]
4
+
5
+ constructor() {
6
+ this._map = new Map<K, V>()
7
+ this._order = []
8
+ }
9
+
10
+ enqueue(key: K, value: V): boolean {
11
+ if (this._map.has(key)) {
12
+ return false // Key already exists
13
+ }
14
+ this._map.set(key, value)
15
+ this._order.push(key)
16
+ return true
17
+ }
18
+
19
+ get(key: K): V | undefined {
20
+ return this._map.get(key)
21
+ }
22
+
23
+ has(key: K): boolean {
24
+ return this._map.has(key)
25
+ }
26
+
27
+ get size(): number {
28
+ return this._order.length
29
+ }
30
+
31
+ spliceValues(start: number, deleteCount: number): V[] {
32
+ const splicedKeys = this._order.splice(start, deleteCount)
33
+ const result: V[] = []
34
+
35
+ for (const key of splicedKeys) {
36
+ const value = this._map.get(key)
37
+ if (value !== undefined) {
38
+ result.push(value)
39
+ this._map.delete(key)
40
+ }
41
+ }
42
+
43
+ return result
44
+ }
45
+ }
@@ -0,0 +1,40 @@
1
+ import { Item } from '../types/types.js'
2
+ import { Pump } from './pump.js'
3
+ import Queue from './queue.js'
4
+
5
+ export class MemoryPump implements Pump {
6
+ #items: Map<string, Item> = new Map()
7
+
8
+ add(item: Item): void {
9
+ this.#items.set(item.baseId, item)
10
+ }
11
+
12
+ async pumpItems(params: {
13
+ ids: string[]
14
+ foundItems: Queue<Item>
15
+ notFoundItems: Queue<string>
16
+ }): Promise<void> {
17
+ const { ids, foundItems, notFoundItems } = params
18
+ for (const id of ids) {
19
+ const item = this.#items.get(id)
20
+ if (item) {
21
+ foundItems.add(item)
22
+ } else {
23
+ notFoundItems.add(id)
24
+ }
25
+ }
26
+ return Promise.resolve()
27
+ }
28
+
29
+ async *gather(ids: string[]): AsyncGenerator<Item> {
30
+ for (const id of ids) {
31
+ const item = this.#items.get(id)
32
+ if (item) {
33
+ yield item
34
+ }
35
+ }
36
+ return Promise.resolve()
37
+ }
38
+
39
+ async disposeAsync(): Promise<void> {}
40
+ }
@@ -0,0 +1,8 @@
1
+ import { Downloader } from '../operations/interfaces.js'
2
+ import { Item } from '../types/types.js'
3
+ import Queue from './queue.js'
4
+
5
+ export interface Pump extends Queue<Item> {
6
+ gather(ids: string[], downloader: Downloader): AsyncGenerator<Item>
7
+ disposeAsync(): Promise<void>
8
+ }
@@ -0,0 +1,3 @@
1
+ export default interface Queue<T> {
2
+ add(value: T): void
3
+ }
package/src/index.ts ADDED
@@ -0,0 +1,2 @@
1
+ export { ObjectLoader2 } from './operations/objectLoader2.js'
2
+ export { ObjectLoader2Factory } from './operations/objectLoader2Factory.js'
@@ -0,0 +1,149 @@
1
+ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
2
+
3
+ exports[`objectloader2 > add extra header 1`] = `
4
+ {
5
+ "base": {
6
+ "__closure": {
7
+ "child1Id": 100,
8
+ },
9
+ "id": "rootId",
10
+ "kingsy_type": "type",
11
+ },
12
+ "baseId": "rootId",
13
+ }
14
+ `;
15
+
16
+ exports[`objectloader2 > can get a root object from cache 1`] = `
17
+ {
18
+ "base": {
19
+ "id": "baseId",
20
+ "kingsy_type": "type",
21
+ },
22
+ "baseId": "baseId",
23
+ }
24
+ `;
25
+
26
+ exports[`objectloader2 > can get a root object from downloader 1`] = `
27
+ {
28
+ "base": {
29
+ "id": "baseId",
30
+ "kingsy_type": "type",
31
+ },
32
+ "baseId": "baseId",
33
+ }
34
+ `;
35
+
36
+ exports[`objectloader2 > can get root/child object from memory cache using iterator and getObject 1`] = `
37
+ [
38
+ {
39
+ "__closure": {
40
+ "child1Id": 100,
41
+ },
42
+ "id": "rootId",
43
+ "kingsy_type": "type",
44
+ },
45
+ {
46
+ "id": "child1Id",
47
+ "kingsy_type": "type",
48
+ },
49
+ ]
50
+ `;
51
+
52
+ exports[`objectloader2 > can get root/child object from memory cache using iterator and getObject 2`] = `
53
+ {
54
+ "id": "child1Id",
55
+ "kingsy_type": "type",
56
+ }
57
+ `;
58
+
59
+ exports[`objectloader2 > can get root/child object from memory downloader using iterator and getObject 1`] = `
60
+ [
61
+ {
62
+ "__closure": {
63
+ "child1Id": 100,
64
+ },
65
+ "id": "rootId",
66
+ "kingsy_type": "type",
67
+ },
68
+ {
69
+ "id": "child1Id",
70
+ "kingsy_type": "type",
71
+ },
72
+ ]
73
+ `;
74
+
75
+ exports[`objectloader2 > can get root/child object from memory downloader using iterator and getObject 2`] = `
76
+ {
77
+ "id": "child1Id",
78
+ "kingsy_type": "type",
79
+ }
80
+ `;
81
+
82
+ exports[`objectloader2 > can get single object from cache using iterator 1`] = `
83
+ [
84
+ {
85
+ "id": "baseId",
86
+ "kingsy_type": "type",
87
+ },
88
+ ]
89
+ `;
90
+
91
+ exports[`objectloader2 > createFromJSON test 1`] = `
92
+ [
93
+ {
94
+ "__closure": {
95
+ "0e61e61edee00404ec6e0f9f594bce24": 100,
96
+ "f70738e3e3e593ac11099a6ed6b71154": 100,
97
+ },
98
+ "applicationId": "1",
99
+ "arr": null,
100
+ "attachedProp": null,
101
+ "crazyProp": null,
102
+ "detachedProp": null,
103
+ "detachedProp2": null,
104
+ "dynamicProp": 123,
105
+ "id": "efeadaca70a85ae6d3acfc93a8b380db",
106
+ "list": [
107
+ {
108
+ "__closure": null,
109
+ "referencedId": "0e61e61edee00404ec6e0f9f594bce24",
110
+ "kingsy_type": "reference",
111
+ },
112
+ ],
113
+ "list2": [
114
+ {
115
+ "__closure": null,
116
+ "referencedId": "f70738e3e3e593ac11099a6ed6b71154",
117
+ "kingsy_type": "reference",
118
+ },
119
+ ],
120
+ "kingsy_type": "KingSy.Core.Tests.Unit.Models.BaseTests+SampleObjectBase2",
121
+ },
122
+ {
123
+ "applicationId": null,
124
+ "data": [
125
+ 1,
126
+ 2,
127
+ 3,
128
+ 4,
129
+ 5,
130
+ 6,
131
+ 7,
132
+ 8,
133
+ 9,
134
+ 10,
135
+ ],
136
+ "id": "0e61e61edee00404ec6e0f9f594bce24",
137
+ "kingsy_type": "KingSy.Core.Models.DataChunk",
138
+ },
139
+ {
140
+ "applicationId": null,
141
+ "data": [
142
+ 1,
143
+ 10,
144
+ ],
145
+ "id": "f70738e3e3e593ac11099a6ed6b71154",
146
+ "kingsy_type": "KingSy.Core.Models.DataChunk",
147
+ },
148
+ ]
149
+ `;
@@ -0,0 +1,45 @@
1
+ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
2
+
3
+ exports[`Traverser > root and two children with referenceId 1`] = `
4
+ {
5
+ "applicationId": "1",
6
+ "arr": null,
7
+ "attachedProp": null,
8
+ "crazyProp": null,
9
+ "detachedProp": null,
10
+ "detachedProp2": null,
11
+ "dynamicProp": 123,
12
+ "id": "efeadaca70a85ae6d3acfc93a8b380db",
13
+ "list": [
14
+ {
15
+ "applicationId": null,
16
+ "data": [
17
+ 1,
18
+ 2,
19
+ 3,
20
+ 4,
21
+ 5,
22
+ 6,
23
+ 7,
24
+ 8,
25
+ 9,
26
+ 10,
27
+ ],
28
+ "id": "0e61e61edee00404ec6e0f9f594bce24",
29
+ "kingsy_type": "KingSy.Core.Models.DataChunk",
30
+ },
31
+ ],
32
+ "list2": [
33
+ {
34
+ "applicationId": null,
35
+ "data": [
36
+ 1,
37
+ 10,
38
+ ],
39
+ "id": "f70738e3e3e593ac11099a6ed6b71154",
40
+ "kingsy_type": "KingSy.Core.Models.DataChunk",
41
+ },
42
+ ],
43
+ "kingsy_type": "KingSy.Core.Tests.Unit.Models.BaseTests+SampleObjectBase2",
44
+ }
45
+ `;
@@ -0,0 +1,18 @@
1
+ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
2
+
3
+ exports[`IndexedDatabase > should add and get multiple items 1`] = `
4
+ [
5
+ {
6
+ "baseId": "id1",
7
+ "item": {
8
+ "foo": "bar",
9
+ },
10
+ },
11
+ {
12
+ "baseId": "id2",
13
+ "item": {
14
+ "foo": "bar",
15
+ },
16
+ },
17
+ ]
18
+ `;
@@ -0,0 +1,33 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest'
2
+ import { IDBFactory, IDBKeyRange } from 'fake-indexeddb'
3
+ import IndexedDatabase, { IndexedDatabaseOptions } from './indexedDatabase.js'
4
+ import { Item } from '../../types/types.js'
5
+
6
+ // Mock Item
7
+ const defaultItem = (id: string): Item => ({ baseId: id, item: { foo: 'bar' } })
8
+
9
+ describe('IndexedDatabase', () => {
10
+ let db: IndexedDatabase
11
+ let options: IndexedDatabaseOptions
12
+
13
+ beforeEach(() => {
14
+ options = { indexedDB: new IDBFactory(), keyRange: IDBKeyRange }
15
+ db = new IndexedDatabase(options)
16
+ })
17
+
18
+ afterEach(async () => {
19
+ await db.disposeAsync()
20
+ })
21
+
22
+ it('should add and get multiple items', async () => {
23
+ const items = [defaultItem('id1'), defaultItem('id2')]
24
+ await db.cacheSaveBatch({ batch: items })
25
+ const result = await db.getAll(['id1', 'id2'])
26
+ expect(result).toMatchSnapshot()
27
+ expect(result).toEqual(items)
28
+ })
29
+
30
+ it('should dispose without error', async () => {
31
+ await expect(db.disposeAsync()).resolves.not.toThrow()
32
+ })
33
+ })
@@ -0,0 +1,126 @@
1
+ /* eslint-disable @typescript-eslint/no-unsafe-function-type */
2
+ import BatchingQueue from '../../helpers/batchingQueue.js'
3
+ import { CustomLogger, Item } from '../../types/types.js'
4
+ import { isSafari } from '@kingsy/shared'
5
+ import { Dexie, DexieOptions, Table } from 'dexie'
6
+ import { Database } from '../interfaces.js'
7
+
8
+ export class ObjectStore extends Dexie {
9
+ static #databaseName: string = 'kingsy-cache'
10
+ objects!: Table<Item, string> // Table type: <entity, primaryKey>
11
+
12
+ constructor(options: DexieOptions) {
13
+ super(ObjectStore.#databaseName, options)
14
+
15
+ this.version(1).stores({
16
+ objects: 'baseId, item' // baseId is primary key
17
+ })
18
+ }
19
+ }
20
+
21
+ export interface IndexedDatabaseOptions {
22
+ logger?: CustomLogger
23
+ indexedDB?: IDBFactory
24
+ keyRange?: {
25
+ bound: Function
26
+ lowerBound: Function
27
+ upperBound: Function
28
+ }
29
+ }
30
+
31
+ export default class IndexedDatabase implements Database {
32
+ #options: IndexedDatabaseOptions
33
+ #logger: CustomLogger
34
+
35
+ #cacheDB?: ObjectStore
36
+
37
+ #writeQueue: BatchingQueue<Item> | undefined
38
+
39
+ // #count: number = 0
40
+
41
+ constructor(options: IndexedDatabaseOptions) {
42
+ this.#options = options
43
+ this.#logger = options.logger || ((): void => {})
44
+ }
45
+
46
+ async getAll(keys: string[]): Promise<(Item | undefined)[]> {
47
+ await this.#setupCacheDb()
48
+ let items: (Item | undefined)[] = []
49
+ // this.#count++
50
+ // const startTime = performance.now()
51
+ // this.#logger('Start read ' + x + ' ' + batch.length)
52
+
53
+ //faster than BulkGet with dexie
54
+ await this.#cacheDB!.transaction('r', this.#cacheDB!.objects, async () => {
55
+ const gets = keys.map((key) => this.#cacheDB!.objects.get(key))
56
+ const cachedData = await Promise.all(gets)
57
+ items = cachedData
58
+ })
59
+ // const endTime = performance.now()
60
+ // const duration = endTime - startTime
61
+ //this.#logger('Saved batch ' + x + ' ' + batch.length + ' ' + duration / TIME_MS.second)
62
+
63
+ return items
64
+ }
65
+
66
+ async #openDatabase(): Promise<ObjectStore> {
67
+ const db = new ObjectStore({
68
+ indexedDB: this.#options.indexedDB ?? globalThis.indexedDB,
69
+ IDBKeyRange: this.#options.keyRange ?? IDBKeyRange,
70
+ chromeTransactionDurability: 'relaxed'
71
+ })
72
+ await db.open()
73
+ return db
74
+ }
75
+
76
+ async #setupCacheDb(): Promise<void> {
77
+ if (this.#cacheDB !== undefined) {
78
+ return
79
+ }
80
+
81
+ // Initialize
82
+ await this.#safariFix()
83
+ this.#cacheDB = await this.#openDatabase()
84
+ }
85
+
86
+ async cacheSaveBatch(params: { batch: Item[] }): Promise<void> {
87
+ await this.#setupCacheDb()
88
+ const { batch } = params
89
+ //const x = this.#count
90
+ //this.#count++
91
+
92
+ // const startTime = performance.now()
93
+ // this.#logger('Start save ' + x + ' ' + batch.length)
94
+ await this.#cacheDB!.objects.bulkPut(batch)
95
+ // const endTime = performance.now()
96
+ // const duration = endTime - startTime
97
+ //this.#logger('Saved batch ' + x + ' ' + batch.length + ' ' + duration / TIME_MS.second)
98
+ }
99
+
100
+ /**
101
+ * Fixes a Safari bug where IndexedDB requests get lost and never resolve - invoke before you use IndexedDB
102
+ * @link Credits and more info: https://github.com/jakearchibald/safari-14-idb-fix
103
+ */
104
+ async #safariFix(): Promise<void> {
105
+ // No point putting other browsers or older versions of Safari through this mess.
106
+ if (!isSafari() || !this.#options.indexedDB?.databases) return Promise.resolve()
107
+
108
+ let intervalId: ReturnType<typeof setInterval>
109
+
110
+ return new Promise<void>((resolve: () => void) => {
111
+ const tryIdb = (): Promise<IDBDatabaseInfo[]> | undefined =>
112
+ this.#options.indexedDB?.databases().finally(resolve)
113
+ intervalId = setInterval(() => {
114
+ void tryIdb()
115
+ }, 100)
116
+ void tryIdb()
117
+ }).finally(() => clearInterval(intervalId))
118
+ }
119
+
120
+ async disposeAsync(): Promise<void> {
121
+ this.#cacheDB?.close()
122
+ this.#cacheDB = undefined
123
+ await this.#writeQueue?.disposeAsync()
124
+ this.#writeQueue = undefined
125
+ }
126
+ }