deepv-code 1.0.182

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 (223) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +3 -0
  3. package/bundle/assets/help/README.md +113 -0
  4. package/bundle/assets/help/cli-help-knowledge.md +1382 -0
  5. package/bundle/assets/sounds/README.md +74 -0
  6. package/bundle/assets/sounds/confirmation-required.wav +0 -0
  7. package/bundle/assets/sounds/response-complete.wav +0 -0
  8. package/bundle/assets/sounds/selection-made.wav +0 -0
  9. package/bundle/dvcode.js +4442 -0
  10. package/bundle/fix-binary-permissions.js +215 -0
  11. package/bundle/login/templates/authSelectPage.html +870 -0
  12. package/bundle/login/templates/deepv.ico +0 -0
  13. package/bundle/login/templates/feishu.ico +0 -0
  14. package/bundle/node_modules/@vscode/ripgrep/bin/darwin-arm64-rg +0 -0
  15. package/bundle/node_modules/@vscode/ripgrep/bin/darwin-x64-rg +0 -0
  16. package/bundle/node_modules/@vscode/ripgrep/bin/linux-arm-rg +0 -0
  17. package/bundle/node_modules/@vscode/ripgrep/bin/linux-arm64-rg +0 -0
  18. package/bundle/node_modules/@vscode/ripgrep/bin/linux-x64-rg +0 -0
  19. package/bundle/node_modules/@vscode/ripgrep/bin/rg.exe +0 -0
  20. package/bundle/node_modules/@vscode/ripgrep/bin/win32-arm64-rg.exe +0 -0
  21. package/bundle/node_modules/@vscode/ripgrep/bin/win32-ia32-rg.exe +0 -0
  22. package/bundle/node_modules/@vscode/ripgrep/bin/win32-x64-rg.exe +0 -0
  23. package/bundle/node_modules/@vscode/ripgrep/lib/download.js +357 -0
  24. package/bundle/node_modules/@vscode/ripgrep/lib/index.d.ts +1 -0
  25. package/bundle/node_modules/@vscode/ripgrep/lib/index.js +42 -0
  26. package/bundle/node_modules/@vscode/ripgrep/lib/postinstall.js +121 -0
  27. package/bundle/node_modules/@vscode/ripgrep/package.json +24 -0
  28. package/bundle/node_modules/undici/LICENSE +21 -0
  29. package/bundle/node_modules/undici/README.md +472 -0
  30. package/bundle/node_modules/undici/docs/docs/api/Agent.md +83 -0
  31. package/bundle/node_modules/undici/docs/docs/api/BalancedPool.md +99 -0
  32. package/bundle/node_modules/undici/docs/docs/api/CacheStorage.md +30 -0
  33. package/bundle/node_modules/undici/docs/docs/api/CacheStore.md +151 -0
  34. package/bundle/node_modules/undici/docs/docs/api/Client.md +281 -0
  35. package/bundle/node_modules/undici/docs/docs/api/ClientStats.md +27 -0
  36. package/bundle/node_modules/undici/docs/docs/api/Connector.md +115 -0
  37. package/bundle/node_modules/undici/docs/docs/api/ContentType.md +57 -0
  38. package/bundle/node_modules/undici/docs/docs/api/Cookies.md +101 -0
  39. package/bundle/node_modules/undici/docs/docs/api/Debug.md +62 -0
  40. package/bundle/node_modules/undici/docs/docs/api/DiagnosticsChannel.md +204 -0
  41. package/bundle/node_modules/undici/docs/docs/api/Dispatcher.md +1200 -0
  42. package/bundle/node_modules/undici/docs/docs/api/EnvHttpProxyAgent.md +159 -0
  43. package/bundle/node_modules/undici/docs/docs/api/Errors.md +49 -0
  44. package/bundle/node_modules/undici/docs/docs/api/EventSource.md +45 -0
  45. package/bundle/node_modules/undici/docs/docs/api/Fetch.md +52 -0
  46. package/bundle/node_modules/undici/docs/docs/api/H2CClient.md +262 -0
  47. package/bundle/node_modules/undici/docs/docs/api/MockAgent.md +603 -0
  48. package/bundle/node_modules/undici/docs/docs/api/MockCallHistory.md +197 -0
  49. package/bundle/node_modules/undici/docs/docs/api/MockCallHistoryLog.md +43 -0
  50. package/bundle/node_modules/undici/docs/docs/api/MockClient.md +77 -0
  51. package/bundle/node_modules/undici/docs/docs/api/MockErrors.md +12 -0
  52. package/bundle/node_modules/undici/docs/docs/api/MockPool.md +548 -0
  53. package/bundle/node_modules/undici/docs/docs/api/Pool.md +84 -0
  54. package/bundle/node_modules/undici/docs/docs/api/PoolStats.md +35 -0
  55. package/bundle/node_modules/undici/docs/docs/api/ProxyAgent.md +227 -0
  56. package/bundle/node_modules/undici/docs/docs/api/RedirectHandler.md +96 -0
  57. package/bundle/node_modules/undici/docs/docs/api/RetryAgent.md +45 -0
  58. package/bundle/node_modules/undici/docs/docs/api/RetryHandler.md +117 -0
  59. package/bundle/node_modules/undici/docs/docs/api/Util.md +25 -0
  60. package/bundle/node_modules/undici/docs/docs/api/WebSocket.md +85 -0
  61. package/bundle/node_modules/undici/docs/docs/api/api-lifecycle.md +91 -0
  62. package/bundle/node_modules/undici/docs/docs/best-practices/client-certificate.md +64 -0
  63. package/bundle/node_modules/undici/docs/docs/best-practices/mocking-request.md +190 -0
  64. package/bundle/node_modules/undici/docs/docs/best-practices/proxy.md +127 -0
  65. package/bundle/node_modules/undici/docs/docs/best-practices/writing-tests.md +20 -0
  66. package/bundle/node_modules/undici/index-fetch.js +35 -0
  67. package/bundle/node_modules/undici/index.d.ts +3 -0
  68. package/bundle/node_modules/undici/index.js +183 -0
  69. package/bundle/node_modules/undici/lib/api/abort-signal.js +59 -0
  70. package/bundle/node_modules/undici/lib/api/api-connect.js +110 -0
  71. package/bundle/node_modules/undici/lib/api/api-pipeline.js +252 -0
  72. package/bundle/node_modules/undici/lib/api/api-request.js +199 -0
  73. package/bundle/node_modules/undici/lib/api/api-stream.js +209 -0
  74. package/bundle/node_modules/undici/lib/api/api-upgrade.js +110 -0
  75. package/bundle/node_modules/undici/lib/api/index.js +7 -0
  76. package/bundle/node_modules/undici/lib/api/readable.js +558 -0
  77. package/bundle/node_modules/undici/lib/api/util.js +95 -0
  78. package/bundle/node_modules/undici/lib/cache/memory-cache-store.js +234 -0
  79. package/bundle/node_modules/undici/lib/cache/sqlite-cache-store.js +461 -0
  80. package/bundle/node_modules/undici/lib/core/connect.js +164 -0
  81. package/bundle/node_modules/undici/lib/core/constants.js +143 -0
  82. package/bundle/node_modules/undici/lib/core/diagnostics.js +196 -0
  83. package/bundle/node_modules/undici/lib/core/errors.js +244 -0
  84. package/bundle/node_modules/undici/lib/core/request.js +397 -0
  85. package/bundle/node_modules/undici/lib/core/symbols.js +68 -0
  86. package/bundle/node_modules/undici/lib/core/tree.js +160 -0
  87. package/bundle/node_modules/undici/lib/core/util.js +988 -0
  88. package/bundle/node_modules/undici/lib/dispatcher/agent.js +135 -0
  89. package/bundle/node_modules/undici/lib/dispatcher/balanced-pool.js +206 -0
  90. package/bundle/node_modules/undici/lib/dispatcher/client-h1.js +1615 -0
  91. package/bundle/node_modules/undici/lib/dispatcher/client-h2.js +798 -0
  92. package/bundle/node_modules/undici/lib/dispatcher/client.js +614 -0
  93. package/bundle/node_modules/undici/lib/dispatcher/dispatcher-base.js +161 -0
  94. package/bundle/node_modules/undici/lib/dispatcher/dispatcher.js +48 -0
  95. package/bundle/node_modules/undici/lib/dispatcher/env-http-proxy-agent.js +151 -0
  96. package/bundle/node_modules/undici/lib/dispatcher/fixed-queue.js +159 -0
  97. package/bundle/node_modules/undici/lib/dispatcher/h2c-client.js +122 -0
  98. package/bundle/node_modules/undici/lib/dispatcher/pool-base.js +191 -0
  99. package/bundle/node_modules/undici/lib/dispatcher/pool.js +118 -0
  100. package/bundle/node_modules/undici/lib/dispatcher/proxy-agent.js +275 -0
  101. package/bundle/node_modules/undici/lib/dispatcher/retry-agent.js +35 -0
  102. package/bundle/node_modules/undici/lib/global.js +32 -0
  103. package/bundle/node_modules/undici/lib/handler/cache-handler.js +448 -0
  104. package/bundle/node_modules/undici/lib/handler/cache-revalidation-handler.js +124 -0
  105. package/bundle/node_modules/undici/lib/handler/decorator-handler.js +67 -0
  106. package/bundle/node_modules/undici/lib/handler/redirect-handler.js +227 -0
  107. package/bundle/node_modules/undici/lib/handler/retry-handler.js +342 -0
  108. package/bundle/node_modules/undici/lib/handler/unwrap-handler.js +96 -0
  109. package/bundle/node_modules/undici/lib/handler/wrap-handler.js +95 -0
  110. package/bundle/node_modules/undici/lib/interceptor/cache.js +372 -0
  111. package/bundle/node_modules/undici/lib/interceptor/dns.js +432 -0
  112. package/bundle/node_modules/undici/lib/interceptor/dump.js +111 -0
  113. package/bundle/node_modules/undici/lib/interceptor/redirect.js +21 -0
  114. package/bundle/node_modules/undici/lib/interceptor/response-error.js +95 -0
  115. package/bundle/node_modules/undici/lib/interceptor/retry.js +19 -0
  116. package/bundle/node_modules/undici/lib/llhttp/.gitkeep +0 -0
  117. package/bundle/node_modules/undici/lib/llhttp/constants.d.ts +97 -0
  118. package/bundle/node_modules/undici/lib/llhttp/constants.js +498 -0
  119. package/bundle/node_modules/undici/lib/llhttp/constants.js.map +1 -0
  120. package/bundle/node_modules/undici/lib/llhttp/llhttp-wasm.js +15 -0
  121. package/bundle/node_modules/undici/lib/llhttp/llhttp_simd-wasm.js +15 -0
  122. package/bundle/node_modules/undici/lib/llhttp/utils.d.ts +2 -0
  123. package/bundle/node_modules/undici/lib/llhttp/utils.js +15 -0
  124. package/bundle/node_modules/undici/lib/llhttp/utils.js.map +1 -0
  125. package/bundle/node_modules/undici/lib/mock/mock-agent.js +224 -0
  126. package/bundle/node_modules/undici/lib/mock/mock-call-history.js +248 -0
  127. package/bundle/node_modules/undici/lib/mock/mock-client.js +64 -0
  128. package/bundle/node_modules/undici/lib/mock/mock-errors.js +19 -0
  129. package/bundle/node_modules/undici/lib/mock/mock-interceptor.js +209 -0
  130. package/bundle/node_modules/undici/lib/mock/mock-pool.js +64 -0
  131. package/bundle/node_modules/undici/lib/mock/mock-symbols.js +31 -0
  132. package/bundle/node_modules/undici/lib/mock/mock-utils.js +433 -0
  133. package/bundle/node_modules/undici/lib/mock/pending-interceptors-formatter.js +43 -0
  134. package/bundle/node_modules/undici/lib/util/cache.js +368 -0
  135. package/bundle/node_modules/undici/lib/util/date.js +259 -0
  136. package/bundle/node_modules/undici/lib/util/stats.js +32 -0
  137. package/bundle/node_modules/undici/lib/util/timers.js +423 -0
  138. package/bundle/node_modules/undici/lib/web/cache/cache.js +862 -0
  139. package/bundle/node_modules/undici/lib/web/cache/cachestorage.js +152 -0
  140. package/bundle/node_modules/undici/lib/web/cache/util.js +45 -0
  141. package/bundle/node_modules/undici/lib/web/cookies/constants.js +12 -0
  142. package/bundle/node_modules/undici/lib/web/cookies/index.js +199 -0
  143. package/bundle/node_modules/undici/lib/web/cookies/parse.js +322 -0
  144. package/bundle/node_modules/undici/lib/web/cookies/util.js +282 -0
  145. package/bundle/node_modules/undici/lib/web/eventsource/eventsource-stream.js +399 -0
  146. package/bundle/node_modules/undici/lib/web/eventsource/eventsource.js +484 -0
  147. package/bundle/node_modules/undici/lib/web/eventsource/util.js +37 -0
  148. package/bundle/node_modules/undici/lib/web/fetch/LICENSE +21 -0
  149. package/bundle/node_modules/undici/lib/web/fetch/body.js +532 -0
  150. package/bundle/node_modules/undici/lib/web/fetch/constants.js +131 -0
  151. package/bundle/node_modules/undici/lib/web/fetch/data-url.js +744 -0
  152. package/bundle/node_modules/undici/lib/web/fetch/dispatcher-weakref.js +46 -0
  153. package/bundle/node_modules/undici/lib/web/fetch/formdata-parser.js +501 -0
  154. package/bundle/node_modules/undici/lib/web/fetch/formdata.js +263 -0
  155. package/bundle/node_modules/undici/lib/web/fetch/global.js +40 -0
  156. package/bundle/node_modules/undici/lib/web/fetch/headers.js +719 -0
  157. package/bundle/node_modules/undici/lib/web/fetch/index.js +2258 -0
  158. package/bundle/node_modules/undici/lib/web/fetch/request.js +1099 -0
  159. package/bundle/node_modules/undici/lib/web/fetch/response.js +636 -0
  160. package/bundle/node_modules/undici/lib/web/fetch/util.js +1782 -0
  161. package/bundle/node_modules/undici/lib/web/fetch/webidl.js +740 -0
  162. package/bundle/node_modules/undici/lib/web/websocket/connection.js +325 -0
  163. package/bundle/node_modules/undici/lib/web/websocket/constants.js +126 -0
  164. package/bundle/node_modules/undici/lib/web/websocket/events.js +331 -0
  165. package/bundle/node_modules/undici/lib/web/websocket/frame.js +138 -0
  166. package/bundle/node_modules/undici/lib/web/websocket/permessage-deflate.js +70 -0
  167. package/bundle/node_modules/undici/lib/web/websocket/receiver.js +454 -0
  168. package/bundle/node_modules/undici/lib/web/websocket/sender.js +109 -0
  169. package/bundle/node_modules/undici/lib/web/websocket/stream/websocketerror.js +83 -0
  170. package/bundle/node_modules/undici/lib/web/websocket/stream/websocketstream.js +485 -0
  171. package/bundle/node_modules/undici/lib/web/websocket/util.js +338 -0
  172. package/bundle/node_modules/undici/lib/web/websocket/websocket.js +686 -0
  173. package/bundle/node_modules/undici/package.json +149 -0
  174. package/bundle/node_modules/undici/scripts/strip-comments.js +10 -0
  175. package/bundle/node_modules/undici/types/README.md +6 -0
  176. package/bundle/node_modules/undici/types/agent.d.ts +35 -0
  177. package/bundle/node_modules/undici/types/api.d.ts +43 -0
  178. package/bundle/node_modules/undici/types/balanced-pool.d.ts +29 -0
  179. package/bundle/node_modules/undici/types/cache-interceptor.d.ts +172 -0
  180. package/bundle/node_modules/undici/types/cache.d.ts +36 -0
  181. package/bundle/node_modules/undici/types/client-stats.d.ts +15 -0
  182. package/bundle/node_modules/undici/types/client.d.ts +110 -0
  183. package/bundle/node_modules/undici/types/connector.d.ts +34 -0
  184. package/bundle/node_modules/undici/types/content-type.d.ts +21 -0
  185. package/bundle/node_modules/undici/types/cookies.d.ts +30 -0
  186. package/bundle/node_modules/undici/types/diagnostics-channel.d.ts +66 -0
  187. package/bundle/node_modules/undici/types/dispatcher.d.ts +281 -0
  188. package/bundle/node_modules/undici/types/env-http-proxy-agent.d.ts +21 -0
  189. package/bundle/node_modules/undici/types/errors.d.ts +171 -0
  190. package/bundle/node_modules/undici/types/eventsource.d.ts +61 -0
  191. package/bundle/node_modules/undici/types/fetch.d.ts +210 -0
  192. package/bundle/node_modules/undici/types/formdata.d.ts +108 -0
  193. package/bundle/node_modules/undici/types/global-dispatcher.d.ts +9 -0
  194. package/bundle/node_modules/undici/types/global-origin.d.ts +7 -0
  195. package/bundle/node_modules/undici/types/h2c-client.d.ts +75 -0
  196. package/bundle/node_modules/undici/types/handlers.d.ts +15 -0
  197. package/bundle/node_modules/undici/types/header.d.ts +160 -0
  198. package/bundle/node_modules/undici/types/index.d.ts +75 -0
  199. package/bundle/node_modules/undici/types/interceptors.d.ts +34 -0
  200. package/bundle/node_modules/undici/types/mock-agent.d.ts +68 -0
  201. package/bundle/node_modules/undici/types/mock-call-history.d.ts +111 -0
  202. package/bundle/node_modules/undici/types/mock-client.d.ts +25 -0
  203. package/bundle/node_modules/undici/types/mock-errors.d.ts +12 -0
  204. package/bundle/node_modules/undici/types/mock-interceptor.d.ts +93 -0
  205. package/bundle/node_modules/undici/types/mock-pool.d.ts +25 -0
  206. package/bundle/node_modules/undici/types/patch.d.ts +29 -0
  207. package/bundle/node_modules/undici/types/pool-stats.d.ts +19 -0
  208. package/bundle/node_modules/undici/types/pool.d.ts +41 -0
  209. package/bundle/node_modules/undici/types/proxy-agent.d.ts +29 -0
  210. package/bundle/node_modules/undici/types/readable.d.ts +68 -0
  211. package/bundle/node_modules/undici/types/retry-agent.d.ts +8 -0
  212. package/bundle/node_modules/undici/types/retry-handler.d.ts +116 -0
  213. package/bundle/node_modules/undici/types/util.d.ts +18 -0
  214. package/bundle/node_modules/undici/types/utility.d.ts +7 -0
  215. package/bundle/node_modules/undici/types/webidl.d.ts +266 -0
  216. package/bundle/node_modules/undici/types/websocket.d.ts +184 -0
  217. package/bundle/sandbox-macos-permissive-closed.sb +26 -0
  218. package/bundle/sandbox-macos-permissive-open.sb +19 -0
  219. package/bundle/sandbox-macos-permissive-proxied.sb +31 -0
  220. package/bundle/sandbox-macos-restrictive-closed.sb +87 -0
  221. package/bundle/sandbox-macos-restrictive-open.sb +90 -0
  222. package/bundle/sandbox-macos-restrictive-proxied.sb +92 -0
  223. package/package.json +137 -0
@@ -0,0 +1,234 @@
1
+ 'use strict'
2
+
3
+ const { Writable } = require('node:stream')
4
+ const { EventEmitter } = require('node:events')
5
+ const { assertCacheKey, assertCacheValue } = require('../util/cache.js')
6
+
7
+ /**
8
+ * @typedef {import('../../types/cache-interceptor.d.ts').default.CacheKey} CacheKey
9
+ * @typedef {import('../../types/cache-interceptor.d.ts').default.CacheValue} CacheValue
10
+ * @typedef {import('../../types/cache-interceptor.d.ts').default.CacheStore} CacheStore
11
+ * @typedef {import('../../types/cache-interceptor.d.ts').default.GetResult} GetResult
12
+ */
13
+
14
+ /**
15
+ * @implements {CacheStore}
16
+ * @extends {EventEmitter}
17
+ */
18
+ class MemoryCacheStore extends EventEmitter {
19
+ #maxCount = Infinity
20
+ #maxSize = Infinity
21
+ #maxEntrySize = Infinity
22
+
23
+ #size = 0
24
+ #count = 0
25
+ #entries = new Map()
26
+ #hasEmittedMaxSizeEvent = false
27
+
28
+ /**
29
+ * @param {import('../../types/cache-interceptor.d.ts').default.MemoryCacheStoreOpts | undefined} [opts]
30
+ */
31
+ constructor (opts) {
32
+ super()
33
+ if (opts) {
34
+ if (typeof opts !== 'object') {
35
+ throw new TypeError('MemoryCacheStore options must be an object')
36
+ }
37
+
38
+ if (opts.maxCount !== undefined) {
39
+ if (
40
+ typeof opts.maxCount !== 'number' ||
41
+ !Number.isInteger(opts.maxCount) ||
42
+ opts.maxCount < 0
43
+ ) {
44
+ throw new TypeError('MemoryCacheStore options.maxCount must be a non-negative integer')
45
+ }
46
+ this.#maxCount = opts.maxCount
47
+ }
48
+
49
+ if (opts.maxSize !== undefined) {
50
+ if (
51
+ typeof opts.maxSize !== 'number' ||
52
+ !Number.isInteger(opts.maxSize) ||
53
+ opts.maxSize < 0
54
+ ) {
55
+ throw new TypeError('MemoryCacheStore options.maxSize must be a non-negative integer')
56
+ }
57
+ this.#maxSize = opts.maxSize
58
+ }
59
+
60
+ if (opts.maxEntrySize !== undefined) {
61
+ if (
62
+ typeof opts.maxEntrySize !== 'number' ||
63
+ !Number.isInteger(opts.maxEntrySize) ||
64
+ opts.maxEntrySize < 0
65
+ ) {
66
+ throw new TypeError('MemoryCacheStore options.maxEntrySize must be a non-negative integer')
67
+ }
68
+ this.#maxEntrySize = opts.maxEntrySize
69
+ }
70
+ }
71
+ }
72
+
73
+ /**
74
+ * Get the current size of the cache in bytes
75
+ * @returns {number} The current size of the cache in bytes
76
+ */
77
+ get size () {
78
+ return this.#size
79
+ }
80
+
81
+ /**
82
+ * Check if the cache is full (either max size or max count reached)
83
+ * @returns {boolean} True if the cache is full, false otherwise
84
+ */
85
+ isFull () {
86
+ return this.#size >= this.#maxSize || this.#count >= this.#maxCount
87
+ }
88
+
89
+ /**
90
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} req
91
+ * @returns {import('../../types/cache-interceptor.d.ts').default.GetResult | undefined}
92
+ */
93
+ get (key) {
94
+ assertCacheKey(key)
95
+
96
+ const topLevelKey = `${key.origin}:${key.path}`
97
+
98
+ const now = Date.now()
99
+ const entries = this.#entries.get(topLevelKey)
100
+
101
+ const entry = entries ? findEntry(key, entries, now) : null
102
+
103
+ return entry == null
104
+ ? undefined
105
+ : {
106
+ statusMessage: entry.statusMessage,
107
+ statusCode: entry.statusCode,
108
+ headers: entry.headers,
109
+ body: entry.body,
110
+ vary: entry.vary ? entry.vary : undefined,
111
+ etag: entry.etag,
112
+ cacheControlDirectives: entry.cacheControlDirectives,
113
+ cachedAt: entry.cachedAt,
114
+ staleAt: entry.staleAt,
115
+ deleteAt: entry.deleteAt
116
+ }
117
+ }
118
+
119
+ /**
120
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
121
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheValue} val
122
+ * @returns {Writable | undefined}
123
+ */
124
+ createWriteStream (key, val) {
125
+ assertCacheKey(key)
126
+ assertCacheValue(val)
127
+
128
+ const topLevelKey = `${key.origin}:${key.path}`
129
+
130
+ const store = this
131
+ const entry = { ...key, ...val, body: [], size: 0 }
132
+
133
+ return new Writable({
134
+ write (chunk, encoding, callback) {
135
+ if (typeof chunk === 'string') {
136
+ chunk = Buffer.from(chunk, encoding)
137
+ }
138
+
139
+ entry.size += chunk.byteLength
140
+
141
+ if (entry.size >= store.#maxEntrySize) {
142
+ this.destroy()
143
+ } else {
144
+ entry.body.push(chunk)
145
+ }
146
+
147
+ callback(null)
148
+ },
149
+ final (callback) {
150
+ let entries = store.#entries.get(topLevelKey)
151
+ if (!entries) {
152
+ entries = []
153
+ store.#entries.set(topLevelKey, entries)
154
+ }
155
+ const previousEntry = findEntry(key, entries, Date.now())
156
+ if (previousEntry) {
157
+ const index = entries.indexOf(previousEntry)
158
+ entries.splice(index, 1, entry)
159
+ store.#size -= previousEntry.size
160
+ } else {
161
+ entries.push(entry)
162
+ store.#count += 1
163
+ }
164
+
165
+ store.#size += entry.size
166
+
167
+ // Check if cache is full and emit event if needed
168
+ if (store.#size > store.#maxSize || store.#count > store.#maxCount) {
169
+ // Emit maxSizeExceeded event if we haven't already
170
+ if (!store.#hasEmittedMaxSizeEvent) {
171
+ store.emit('maxSizeExceeded', {
172
+ size: store.#size,
173
+ maxSize: store.#maxSize,
174
+ count: store.#count,
175
+ maxCount: store.#maxCount
176
+ })
177
+ store.#hasEmittedMaxSizeEvent = true
178
+ }
179
+
180
+ // Perform eviction
181
+ for (const [key, entries] of store.#entries) {
182
+ for (const entry of entries.splice(0, entries.length / 2)) {
183
+ store.#size -= entry.size
184
+ store.#count -= 1
185
+ }
186
+ if (entries.length === 0) {
187
+ store.#entries.delete(key)
188
+ }
189
+ }
190
+
191
+ // Reset the event flag after eviction
192
+ if (store.#size < store.#maxSize && store.#count < store.#maxCount) {
193
+ store.#hasEmittedMaxSizeEvent = false
194
+ }
195
+ }
196
+
197
+ callback(null)
198
+ }
199
+ })
200
+ }
201
+
202
+ /**
203
+ * @param {CacheKey} key
204
+ */
205
+ delete (key) {
206
+ if (typeof key !== 'object') {
207
+ throw new TypeError(`expected key to be object, got ${typeof key}`)
208
+ }
209
+
210
+ const topLevelKey = `${key.origin}:${key.path}`
211
+
212
+ for (const entry of this.#entries.get(topLevelKey) ?? []) {
213
+ this.#size -= entry.size
214
+ this.#count -= 1
215
+ }
216
+ this.#entries.delete(topLevelKey)
217
+ }
218
+ }
219
+
220
+ function findEntry (key, entries, now) {
221
+ return entries.find((entry) => (
222
+ entry.deleteAt > now &&
223
+ entry.method === key.method &&
224
+ (entry.vary == null || Object.keys(entry.vary).every(headerName => {
225
+ if (entry.vary[headerName] === null) {
226
+ return key.headers[headerName] === undefined
227
+ }
228
+
229
+ return entry.vary[headerName] === key.headers[headerName]
230
+ }))
231
+ ))
232
+ }
233
+
234
+ module.exports = MemoryCacheStore
@@ -0,0 +1,461 @@
1
+ 'use strict'
2
+
3
+ const { Writable } = require('stream')
4
+ const { assertCacheKey, assertCacheValue } = require('../util/cache.js')
5
+
6
+ let DatabaseSync
7
+
8
+ const VERSION = 3
9
+
10
+ // 2gb
11
+ const MAX_ENTRY_SIZE = 2 * 1000 * 1000 * 1000
12
+
13
+ /**
14
+ * @typedef {import('../../types/cache-interceptor.d.ts').default.CacheStore} CacheStore
15
+ * @implements {CacheStore}
16
+ *
17
+ * @typedef {{
18
+ * id: Readonly<number>,
19
+ * body?: Uint8Array
20
+ * statusCode: number
21
+ * statusMessage: string
22
+ * headers?: string
23
+ * vary?: string
24
+ * etag?: string
25
+ * cacheControlDirectives?: string
26
+ * cachedAt: number
27
+ * staleAt: number
28
+ * deleteAt: number
29
+ * }} SqliteStoreValue
30
+ */
31
+ module.exports = class SqliteCacheStore {
32
+ #maxEntrySize = MAX_ENTRY_SIZE
33
+ #maxCount = Infinity
34
+
35
+ /**
36
+ * @type {import('node:sqlite').DatabaseSync}
37
+ */
38
+ #db
39
+
40
+ /**
41
+ * @type {import('node:sqlite').StatementSync}
42
+ */
43
+ #getValuesQuery
44
+
45
+ /**
46
+ * @type {import('node:sqlite').StatementSync}
47
+ */
48
+ #updateValueQuery
49
+
50
+ /**
51
+ * @type {import('node:sqlite').StatementSync}
52
+ */
53
+ #insertValueQuery
54
+
55
+ /**
56
+ * @type {import('node:sqlite').StatementSync}
57
+ */
58
+ #deleteExpiredValuesQuery
59
+
60
+ /**
61
+ * @type {import('node:sqlite').StatementSync}
62
+ */
63
+ #deleteByUrlQuery
64
+
65
+ /**
66
+ * @type {import('node:sqlite').StatementSync}
67
+ */
68
+ #countEntriesQuery
69
+
70
+ /**
71
+ * @type {import('node:sqlite').StatementSync | null}
72
+ */
73
+ #deleteOldValuesQuery
74
+
75
+ /**
76
+ * @param {import('../../types/cache-interceptor.d.ts').default.SqliteCacheStoreOpts | undefined} opts
77
+ */
78
+ constructor (opts) {
79
+ if (opts) {
80
+ if (typeof opts !== 'object') {
81
+ throw new TypeError('SqliteCacheStore options must be an object')
82
+ }
83
+
84
+ if (opts.maxEntrySize !== undefined) {
85
+ if (
86
+ typeof opts.maxEntrySize !== 'number' ||
87
+ !Number.isInteger(opts.maxEntrySize) ||
88
+ opts.maxEntrySize < 0
89
+ ) {
90
+ throw new TypeError('SqliteCacheStore options.maxEntrySize must be a non-negative integer')
91
+ }
92
+
93
+ if (opts.maxEntrySize > MAX_ENTRY_SIZE) {
94
+ throw new TypeError('SqliteCacheStore options.maxEntrySize must be less than 2gb')
95
+ }
96
+
97
+ this.#maxEntrySize = opts.maxEntrySize
98
+ }
99
+
100
+ if (opts.maxCount !== undefined) {
101
+ if (
102
+ typeof opts.maxCount !== 'number' ||
103
+ !Number.isInteger(opts.maxCount) ||
104
+ opts.maxCount < 0
105
+ ) {
106
+ throw new TypeError('SqliteCacheStore options.maxCount must be a non-negative integer')
107
+ }
108
+ this.#maxCount = opts.maxCount
109
+ }
110
+ }
111
+
112
+ if (!DatabaseSync) {
113
+ DatabaseSync = require('node:sqlite').DatabaseSync
114
+ }
115
+ this.#db = new DatabaseSync(opts?.location ?? ':memory:')
116
+
117
+ this.#db.exec(`
118
+ PRAGMA journal_mode = WAL;
119
+ PRAGMA synchronous = NORMAL;
120
+ PRAGMA temp_store = memory;
121
+ PRAGMA optimize;
122
+
123
+ CREATE TABLE IF NOT EXISTS cacheInterceptorV${VERSION} (
124
+ -- Data specific to us
125
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
126
+ url TEXT NOT NULL,
127
+ method TEXT NOT NULL,
128
+
129
+ -- Data returned to the interceptor
130
+ body BUF NULL,
131
+ deleteAt INTEGER NOT NULL,
132
+ statusCode INTEGER NOT NULL,
133
+ statusMessage TEXT NOT NULL,
134
+ headers TEXT NULL,
135
+ cacheControlDirectives TEXT NULL,
136
+ etag TEXT NULL,
137
+ vary TEXT NULL,
138
+ cachedAt INTEGER NOT NULL,
139
+ staleAt INTEGER NOT NULL
140
+ );
141
+
142
+ CREATE INDEX IF NOT EXISTS idx_cacheInterceptorV${VERSION}_getValuesQuery ON cacheInterceptorV${VERSION}(url, method, deleteAt);
143
+ CREATE INDEX IF NOT EXISTS idx_cacheInterceptorV${VERSION}_deleteByUrlQuery ON cacheInterceptorV${VERSION}(deleteAt);
144
+ `)
145
+
146
+ this.#getValuesQuery = this.#db.prepare(`
147
+ SELECT
148
+ id,
149
+ body,
150
+ deleteAt,
151
+ statusCode,
152
+ statusMessage,
153
+ headers,
154
+ etag,
155
+ cacheControlDirectives,
156
+ vary,
157
+ cachedAt,
158
+ staleAt
159
+ FROM cacheInterceptorV${VERSION}
160
+ WHERE
161
+ url = ?
162
+ AND method = ?
163
+ ORDER BY
164
+ deleteAt ASC
165
+ `)
166
+
167
+ this.#updateValueQuery = this.#db.prepare(`
168
+ UPDATE cacheInterceptorV${VERSION} SET
169
+ body = ?,
170
+ deleteAt = ?,
171
+ statusCode = ?,
172
+ statusMessage = ?,
173
+ headers = ?,
174
+ etag = ?,
175
+ cacheControlDirectives = ?,
176
+ cachedAt = ?,
177
+ staleAt = ?
178
+ WHERE
179
+ id = ?
180
+ `)
181
+
182
+ this.#insertValueQuery = this.#db.prepare(`
183
+ INSERT INTO cacheInterceptorV${VERSION} (
184
+ url,
185
+ method,
186
+ body,
187
+ deleteAt,
188
+ statusCode,
189
+ statusMessage,
190
+ headers,
191
+ etag,
192
+ cacheControlDirectives,
193
+ vary,
194
+ cachedAt,
195
+ staleAt
196
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
197
+ `)
198
+
199
+ this.#deleteByUrlQuery = this.#db.prepare(
200
+ `DELETE FROM cacheInterceptorV${VERSION} WHERE url = ?`
201
+ )
202
+
203
+ this.#countEntriesQuery = this.#db.prepare(
204
+ `SELECT COUNT(*) AS total FROM cacheInterceptorV${VERSION}`
205
+ )
206
+
207
+ this.#deleteExpiredValuesQuery = this.#db.prepare(
208
+ `DELETE FROM cacheInterceptorV${VERSION} WHERE deleteAt <= ?`
209
+ )
210
+
211
+ this.#deleteOldValuesQuery = this.#maxCount === Infinity
212
+ ? null
213
+ : this.#db.prepare(`
214
+ DELETE FROM cacheInterceptorV${VERSION}
215
+ WHERE id IN (
216
+ SELECT
217
+ id
218
+ FROM cacheInterceptorV${VERSION}
219
+ ORDER BY cachedAt DESC
220
+ LIMIT ?
221
+ )
222
+ `)
223
+ }
224
+
225
+ close () {
226
+ this.#db.close()
227
+ }
228
+
229
+ /**
230
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
231
+ * @returns {(import('../../types/cache-interceptor.d.ts').default.GetResult & { body?: Buffer }) | undefined}
232
+ */
233
+ get (key) {
234
+ assertCacheKey(key)
235
+
236
+ const value = this.#findValue(key)
237
+ return value
238
+ ? {
239
+ body: value.body ? Buffer.from(value.body.buffer, value.body.byteOffset, value.body.byteLength) : undefined,
240
+ statusCode: value.statusCode,
241
+ statusMessage: value.statusMessage,
242
+ headers: value.headers ? JSON.parse(value.headers) : undefined,
243
+ etag: value.etag ? value.etag : undefined,
244
+ vary: value.vary ? JSON.parse(value.vary) : undefined,
245
+ cacheControlDirectives: value.cacheControlDirectives
246
+ ? JSON.parse(value.cacheControlDirectives)
247
+ : undefined,
248
+ cachedAt: value.cachedAt,
249
+ staleAt: value.staleAt,
250
+ deleteAt: value.deleteAt
251
+ }
252
+ : undefined
253
+ }
254
+
255
+ /**
256
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
257
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheValue & { body: null | Buffer | Array<Buffer>}} value
258
+ */
259
+ set (key, value) {
260
+ assertCacheKey(key)
261
+
262
+ const url = this.#makeValueUrl(key)
263
+ const body = Array.isArray(value.body) ? Buffer.concat(value.body) : value.body
264
+ const size = body?.byteLength
265
+
266
+ if (size && size > this.#maxEntrySize) {
267
+ return
268
+ }
269
+
270
+ const existingValue = this.#findValue(key, true)
271
+ if (existingValue) {
272
+ // Updating an existing response, let's overwrite it
273
+ this.#updateValueQuery.run(
274
+ body,
275
+ value.deleteAt,
276
+ value.statusCode,
277
+ value.statusMessage,
278
+ value.headers ? JSON.stringify(value.headers) : null,
279
+ value.etag ? value.etag : null,
280
+ value.cacheControlDirectives ? JSON.stringify(value.cacheControlDirectives) : null,
281
+ value.cachedAt,
282
+ value.staleAt,
283
+ existingValue.id
284
+ )
285
+ } else {
286
+ this.#prune()
287
+ // New response, let's insert it
288
+ this.#insertValueQuery.run(
289
+ url,
290
+ key.method,
291
+ body,
292
+ value.deleteAt,
293
+ value.statusCode,
294
+ value.statusMessage,
295
+ value.headers ? JSON.stringify(value.headers) : null,
296
+ value.etag ? value.etag : null,
297
+ value.cacheControlDirectives ? JSON.stringify(value.cacheControlDirectives) : null,
298
+ value.vary ? JSON.stringify(value.vary) : null,
299
+ value.cachedAt,
300
+ value.staleAt
301
+ )
302
+ }
303
+ }
304
+
305
+ /**
306
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
307
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheValue} value
308
+ * @returns {Writable | undefined}
309
+ */
310
+ createWriteStream (key, value) {
311
+ assertCacheKey(key)
312
+ assertCacheValue(value)
313
+
314
+ let size = 0
315
+ /**
316
+ * @type {Buffer[] | null}
317
+ */
318
+ const body = []
319
+ const store = this
320
+
321
+ return new Writable({
322
+ decodeStrings: true,
323
+ write (chunk, encoding, callback) {
324
+ size += chunk.byteLength
325
+
326
+ if (size < store.#maxEntrySize) {
327
+ body.push(chunk)
328
+ } else {
329
+ this.destroy()
330
+ }
331
+
332
+ callback()
333
+ },
334
+ final (callback) {
335
+ store.set(key, { ...value, body })
336
+ callback()
337
+ }
338
+ })
339
+ }
340
+
341
+ /**
342
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
343
+ */
344
+ delete (key) {
345
+ if (typeof key !== 'object') {
346
+ throw new TypeError(`expected key to be object, got ${typeof key}`)
347
+ }
348
+
349
+ this.#deleteByUrlQuery.run(this.#makeValueUrl(key))
350
+ }
351
+
352
+ #prune () {
353
+ if (Number.isFinite(this.#maxCount) && this.size <= this.#maxCount) {
354
+ return 0
355
+ }
356
+
357
+ {
358
+ const removed = this.#deleteExpiredValuesQuery.run(Date.now()).changes
359
+ if (removed) {
360
+ return removed
361
+ }
362
+ }
363
+
364
+ {
365
+ const removed = this.#deleteOldValuesQuery?.run(Math.max(Math.floor(this.#maxCount * 0.1), 1)).changes
366
+ if (removed) {
367
+ return removed
368
+ }
369
+ }
370
+
371
+ return 0
372
+ }
373
+
374
+ /**
375
+ * Counts the number of rows in the cache
376
+ * @returns {Number}
377
+ */
378
+ get size () {
379
+ const { total } = this.#countEntriesQuery.get()
380
+ return total
381
+ }
382
+
383
+ /**
384
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
385
+ * @returns {string}
386
+ */
387
+ #makeValueUrl (key) {
388
+ return `${key.origin}/${key.path}`
389
+ }
390
+
391
+ /**
392
+ * @param {import('../../types/cache-interceptor.d.ts').default.CacheKey} key
393
+ * @param {boolean} [canBeExpired=false]
394
+ * @returns {SqliteStoreValue | undefined}
395
+ */
396
+ #findValue (key, canBeExpired = false) {
397
+ const url = this.#makeValueUrl(key)
398
+ const { headers, method } = key
399
+
400
+ /**
401
+ * @type {SqliteStoreValue[]}
402
+ */
403
+ const values = this.#getValuesQuery.all(url, method)
404
+
405
+ if (values.length === 0) {
406
+ return undefined
407
+ }
408
+
409
+ const now = Date.now()
410
+ for (const value of values) {
411
+ if (now >= value.deleteAt && !canBeExpired) {
412
+ return undefined
413
+ }
414
+
415
+ let matches = true
416
+
417
+ if (value.vary) {
418
+ const vary = JSON.parse(value.vary)
419
+
420
+ for (const header in vary) {
421
+ if (!headerValueEquals(headers[header], vary[header])) {
422
+ matches = false
423
+ break
424
+ }
425
+ }
426
+ }
427
+
428
+ if (matches) {
429
+ return value
430
+ }
431
+ }
432
+
433
+ return undefined
434
+ }
435
+ }
436
+
437
+ /**
438
+ * @param {string|string[]|null|undefined} lhs
439
+ * @param {string|string[]|null|undefined} rhs
440
+ * @returns {boolean}
441
+ */
442
+ function headerValueEquals (lhs, rhs) {
443
+ if (lhs == null && rhs == null) {
444
+ return true
445
+ }
446
+
447
+ if ((lhs == null && rhs != null) ||
448
+ (lhs != null && rhs == null)) {
449
+ return false
450
+ }
451
+
452
+ if (Array.isArray(lhs) && Array.isArray(rhs)) {
453
+ if (lhs.length !== rhs.length) {
454
+ return false
455
+ }
456
+
457
+ return lhs.every((x, i) => x === rhs[i])
458
+ }
459
+
460
+ return lhs === rhs
461
+ }