@nocobase/plugin-idp-oauth 2.1.0-alpha.17 → 2.1.0-alpha.19

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 (221) hide show
  1. package/build.config.ts +1 -1
  2. package/dist/externalVersion.js +6 -4
  3. package/dist/node_modules/light-my-request/package.json +1 -1
  4. package/dist/node_modules/undici/LICENSE +21 -0
  5. package/dist/node_modules/undici/README.md +741 -0
  6. package/dist/node_modules/undici/docs/docs/api/Agent.md +84 -0
  7. package/dist/node_modules/undici/docs/docs/api/BalancedPool.md +99 -0
  8. package/dist/node_modules/undici/docs/docs/api/CacheStorage.md +30 -0
  9. package/dist/node_modules/undici/docs/docs/api/CacheStore.md +164 -0
  10. package/dist/node_modules/undici/docs/docs/api/Client.md +285 -0
  11. package/dist/node_modules/undici/docs/docs/api/ClientStats.md +27 -0
  12. package/dist/node_modules/undici/docs/docs/api/Connector.md +115 -0
  13. package/dist/node_modules/undici/docs/docs/api/ContentType.md +57 -0
  14. package/dist/node_modules/undici/docs/docs/api/Cookies.md +101 -0
  15. package/dist/node_modules/undici/docs/docs/api/Debug.md +62 -0
  16. package/dist/node_modules/undici/docs/docs/api/DiagnosticsChannel.md +315 -0
  17. package/dist/node_modules/undici/docs/docs/api/Dispatcher.md +1392 -0
  18. package/dist/node_modules/undici/docs/docs/api/EnvHttpProxyAgent.md +159 -0
  19. package/dist/node_modules/undici/docs/docs/api/Errors.md +49 -0
  20. package/dist/node_modules/undici/docs/docs/api/EventSource.md +45 -0
  21. package/dist/node_modules/undici/docs/docs/api/Fetch.md +60 -0
  22. package/dist/node_modules/undici/docs/docs/api/GlobalInstallation.md +139 -0
  23. package/dist/node_modules/undici/docs/docs/api/H2CClient.md +263 -0
  24. package/dist/node_modules/undici/docs/docs/api/MockAgent.md +603 -0
  25. package/dist/node_modules/undici/docs/docs/api/MockCallHistory.md +197 -0
  26. package/dist/node_modules/undici/docs/docs/api/MockCallHistoryLog.md +43 -0
  27. package/dist/node_modules/undici/docs/docs/api/MockClient.md +81 -0
  28. package/dist/node_modules/undici/docs/docs/api/MockErrors.md +12 -0
  29. package/dist/node_modules/undici/docs/docs/api/MockPool.md +555 -0
  30. package/dist/node_modules/undici/docs/docs/api/Pool.md +84 -0
  31. package/dist/node_modules/undici/docs/docs/api/PoolStats.md +35 -0
  32. package/dist/node_modules/undici/docs/docs/api/ProxyAgent.md +229 -0
  33. package/dist/node_modules/undici/docs/docs/api/RedirectHandler.md +93 -0
  34. package/dist/node_modules/undici/docs/docs/api/RetryAgent.md +50 -0
  35. package/dist/node_modules/undici/docs/docs/api/RetryHandler.md +118 -0
  36. package/dist/node_modules/undici/docs/docs/api/RoundRobinPool.md +145 -0
  37. package/dist/node_modules/undici/docs/docs/api/SnapshotAgent.md +616 -0
  38. package/dist/node_modules/undici/docs/docs/api/Socks5ProxyAgent.md +274 -0
  39. package/dist/node_modules/undici/docs/docs/api/Util.md +25 -0
  40. package/dist/node_modules/undici/docs/docs/api/WebSocket.md +141 -0
  41. package/dist/node_modules/undici/docs/docs/api/api-lifecycle.md +91 -0
  42. package/dist/node_modules/undici/docs/docs/best-practices/client-certificate.md +64 -0
  43. package/dist/node_modules/undici/docs/docs/best-practices/crawling.md +58 -0
  44. package/dist/node_modules/undici/docs/docs/best-practices/mocking-request.md +190 -0
  45. package/dist/node_modules/undici/docs/docs/best-practices/proxy.md +127 -0
  46. package/dist/node_modules/undici/docs/docs/best-practices/undici-vs-builtin-fetch.md +224 -0
  47. package/dist/node_modules/undici/docs/docs/best-practices/writing-tests.md +63 -0
  48. package/dist/node_modules/undici/index-fetch.js +65 -0
  49. package/dist/node_modules/undici/index.d.ts +3 -0
  50. package/dist/node_modules/undici/index.js +234 -0
  51. package/dist/node_modules/undici/lib/api/abort-signal.js +59 -0
  52. package/dist/node_modules/undici/lib/api/api-connect.js +110 -0
  53. package/dist/node_modules/undici/lib/api/api-pipeline.js +252 -0
  54. package/dist/node_modules/undici/lib/api/api-request.js +214 -0
  55. package/dist/node_modules/undici/lib/api/api-stream.js +209 -0
  56. package/dist/node_modules/undici/lib/api/api-upgrade.js +111 -0
  57. package/dist/node_modules/undici/lib/api/index.js +7 -0
  58. package/dist/node_modules/undici/lib/api/readable.js +580 -0
  59. package/dist/node_modules/undici/lib/cache/memory-cache-store.js +234 -0
  60. package/dist/node_modules/undici/lib/cache/sqlite-cache-store.js +461 -0
  61. package/dist/node_modules/undici/lib/core/connect.js +137 -0
  62. package/dist/node_modules/undici/lib/core/constants.js +143 -0
  63. package/dist/node_modules/undici/lib/core/diagnostics.js +227 -0
  64. package/dist/node_modules/undici/lib/core/errors.js +477 -0
  65. package/dist/node_modules/undici/lib/core/request.js +438 -0
  66. package/dist/node_modules/undici/lib/core/socks5-client.js +407 -0
  67. package/dist/node_modules/undici/lib/core/socks5-utils.js +203 -0
  68. package/dist/node_modules/undici/lib/core/symbols.js +75 -0
  69. package/dist/node_modules/undici/lib/core/tree.js +160 -0
  70. package/dist/node_modules/undici/lib/core/util.js +992 -0
  71. package/dist/node_modules/undici/lib/dispatcher/agent.js +158 -0
  72. package/dist/node_modules/undici/lib/dispatcher/balanced-pool.js +219 -0
  73. package/dist/node_modules/undici/lib/dispatcher/client-h1.js +1610 -0
  74. package/dist/node_modules/undici/lib/dispatcher/client-h2.js +995 -0
  75. package/dist/node_modules/undici/lib/dispatcher/client.js +659 -0
  76. package/dist/node_modules/undici/lib/dispatcher/dispatcher-base.js +165 -0
  77. package/dist/node_modules/undici/lib/dispatcher/dispatcher.js +48 -0
  78. package/dist/node_modules/undici/lib/dispatcher/env-http-proxy-agent.js +146 -0
  79. package/dist/node_modules/undici/lib/dispatcher/fixed-queue.js +135 -0
  80. package/dist/node_modules/undici/lib/dispatcher/h2c-client.js +51 -0
  81. package/dist/node_modules/undici/lib/dispatcher/pool-base.js +214 -0
  82. package/dist/node_modules/undici/lib/dispatcher/pool.js +118 -0
  83. package/dist/node_modules/undici/lib/dispatcher/proxy-agent.js +318 -0
  84. package/dist/node_modules/undici/lib/dispatcher/retry-agent.js +35 -0
  85. package/dist/node_modules/undici/lib/dispatcher/round-robin-pool.js +137 -0
  86. package/dist/node_modules/undici/lib/dispatcher/socks5-proxy-agent.js +249 -0
  87. package/dist/node_modules/undici/lib/encoding/index.js +33 -0
  88. package/dist/node_modules/undici/lib/global.js +50 -0
  89. package/dist/node_modules/undici/lib/handler/cache-handler.js +578 -0
  90. package/dist/node_modules/undici/lib/handler/cache-revalidation-handler.js +124 -0
  91. package/dist/node_modules/undici/lib/handler/decorator-handler.js +67 -0
  92. package/dist/node_modules/undici/lib/handler/deduplication-handler.js +460 -0
  93. package/dist/node_modules/undici/lib/handler/redirect-handler.js +238 -0
  94. package/dist/node_modules/undici/lib/handler/retry-handler.js +394 -0
  95. package/dist/node_modules/undici/lib/handler/unwrap-handler.js +100 -0
  96. package/dist/node_modules/undici/lib/handler/wrap-handler.js +105 -0
  97. package/dist/node_modules/undici/lib/interceptor/cache.js +495 -0
  98. package/dist/node_modules/undici/lib/interceptor/decompress.js +259 -0
  99. package/dist/node_modules/undici/lib/interceptor/deduplicate.js +117 -0
  100. package/dist/node_modules/undici/lib/interceptor/dns.js +571 -0
  101. package/dist/node_modules/undici/lib/interceptor/dump.js +112 -0
  102. package/dist/node_modules/undici/lib/interceptor/redirect.js +21 -0
  103. package/dist/node_modules/undici/lib/interceptor/response-error.js +95 -0
  104. package/dist/node_modules/undici/lib/interceptor/retry.js +19 -0
  105. package/dist/node_modules/undici/lib/llhttp/.gitkeep +0 -0
  106. package/dist/node_modules/undici/lib/llhttp/constants.d.ts +195 -0
  107. package/dist/node_modules/undici/lib/llhttp/constants.js +531 -0
  108. package/dist/node_modules/undici/lib/llhttp/llhttp-wasm.js +15 -0
  109. package/dist/node_modules/undici/lib/llhttp/llhttp_simd-wasm.js +15 -0
  110. package/dist/node_modules/undici/lib/llhttp/utils.d.ts +2 -0
  111. package/dist/node_modules/undici/lib/llhttp/utils.js +12 -0
  112. package/dist/node_modules/undici/lib/mock/mock-agent.js +232 -0
  113. package/dist/node_modules/undici/lib/mock/mock-call-history.js +248 -0
  114. package/dist/node_modules/undici/lib/mock/mock-client.js +68 -0
  115. package/dist/node_modules/undici/lib/mock/mock-errors.js +29 -0
  116. package/dist/node_modules/undici/lib/mock/mock-interceptor.js +209 -0
  117. package/dist/node_modules/undici/lib/mock/mock-pool.js +68 -0
  118. package/dist/node_modules/undici/lib/mock/mock-symbols.js +32 -0
  119. package/dist/node_modules/undici/lib/mock/mock-utils.js +486 -0
  120. package/dist/node_modules/undici/lib/mock/pending-interceptors-formatter.js +43 -0
  121. package/dist/node_modules/undici/lib/mock/snapshot-agent.js +353 -0
  122. package/dist/node_modules/undici/lib/mock/snapshot-recorder.js +588 -0
  123. package/dist/node_modules/undici/lib/mock/snapshot-utils.js +158 -0
  124. package/dist/node_modules/undici/lib/util/cache.js +407 -0
  125. package/dist/node_modules/undici/lib/util/date.js +653 -0
  126. package/dist/node_modules/undici/lib/util/promise.js +28 -0
  127. package/dist/node_modules/undici/lib/util/runtime-features.js +124 -0
  128. package/dist/node_modules/undici/lib/util/stats.js +32 -0
  129. package/dist/node_modules/undici/lib/util/timers.js +425 -0
  130. package/dist/node_modules/undici/lib/web/cache/cache.js +864 -0
  131. package/dist/node_modules/undici/lib/web/cache/cachestorage.js +152 -0
  132. package/dist/node_modules/undici/lib/web/cache/util.js +45 -0
  133. package/dist/node_modules/undici/lib/web/cookies/constants.js +12 -0
  134. package/dist/node_modules/undici/lib/web/cookies/index.js +199 -0
  135. package/dist/node_modules/undici/lib/web/cookies/parse.js +322 -0
  136. package/dist/node_modules/undici/lib/web/cookies/util.js +282 -0
  137. package/dist/node_modules/undici/lib/web/eventsource/eventsource-stream.js +399 -0
  138. package/dist/node_modules/undici/lib/web/eventsource/eventsource.js +501 -0
  139. package/dist/node_modules/undici/lib/web/eventsource/util.js +29 -0
  140. package/dist/node_modules/undici/lib/web/fetch/LICENSE +21 -0
  141. package/dist/node_modules/undici/lib/web/fetch/body.js +509 -0
  142. package/dist/node_modules/undici/lib/web/fetch/constants.js +131 -0
  143. package/dist/node_modules/undici/lib/web/fetch/data-url.js +596 -0
  144. package/dist/node_modules/undici/lib/web/fetch/formdata-parser.js +575 -0
  145. package/dist/node_modules/undici/lib/web/fetch/formdata.js +259 -0
  146. package/dist/node_modules/undici/lib/web/fetch/global.js +40 -0
  147. package/dist/node_modules/undici/lib/web/fetch/headers.js +719 -0
  148. package/dist/node_modules/undici/lib/web/fetch/index.js +2397 -0
  149. package/dist/node_modules/undici/lib/web/fetch/request.js +1115 -0
  150. package/dist/node_modules/undici/lib/web/fetch/response.js +641 -0
  151. package/dist/node_modules/undici/lib/web/fetch/util.js +1520 -0
  152. package/dist/node_modules/undici/lib/web/infra/index.js +229 -0
  153. package/dist/node_modules/undici/lib/web/subresource-integrity/Readme.md +9 -0
  154. package/dist/node_modules/undici/lib/web/subresource-integrity/subresource-integrity.js +307 -0
  155. package/dist/node_modules/undici/lib/web/webidl/index.js +1006 -0
  156. package/dist/node_modules/undici/lib/web/websocket/connection.js +329 -0
  157. package/dist/node_modules/undici/lib/web/websocket/constants.js +126 -0
  158. package/dist/node_modules/undici/lib/web/websocket/events.js +331 -0
  159. package/dist/node_modules/undici/lib/web/websocket/frame.js +133 -0
  160. package/dist/node_modules/undici/lib/web/websocket/permessage-deflate.js +118 -0
  161. package/dist/node_modules/undici/lib/web/websocket/receiver.js +450 -0
  162. package/dist/node_modules/undici/lib/web/websocket/sender.js +109 -0
  163. package/dist/node_modules/undici/lib/web/websocket/stream/websocketerror.js +104 -0
  164. package/dist/node_modules/undici/lib/web/websocket/stream/websocketstream.js +497 -0
  165. package/dist/node_modules/undici/lib/web/websocket/util.js +347 -0
  166. package/dist/node_modules/undici/lib/web/websocket/websocket.js +751 -0
  167. package/dist/node_modules/undici/package.json +152 -0
  168. package/dist/node_modules/undici/scripts/strip-comments.js +10 -0
  169. package/dist/node_modules/undici/types/README.md +6 -0
  170. package/dist/node_modules/undici/types/agent.d.ts +32 -0
  171. package/dist/node_modules/undici/types/api.d.ts +43 -0
  172. package/dist/node_modules/undici/types/balanced-pool.d.ts +30 -0
  173. package/dist/node_modules/undici/types/cache-interceptor.d.ts +179 -0
  174. package/dist/node_modules/undici/types/cache.d.ts +36 -0
  175. package/dist/node_modules/undici/types/client-stats.d.ts +15 -0
  176. package/dist/node_modules/undici/types/client.d.ts +123 -0
  177. package/dist/node_modules/undici/types/connector.d.ts +36 -0
  178. package/dist/node_modules/undici/types/content-type.d.ts +21 -0
  179. package/dist/node_modules/undici/types/cookies.d.ts +30 -0
  180. package/dist/node_modules/undici/types/diagnostics-channel.d.ts +74 -0
  181. package/dist/node_modules/undici/types/dispatcher.d.ts +273 -0
  182. package/dist/node_modules/undici/types/env-http-proxy-agent.d.ts +22 -0
  183. package/dist/node_modules/undici/types/errors.d.ts +177 -0
  184. package/dist/node_modules/undici/types/eventsource.d.ts +66 -0
  185. package/dist/node_modules/undici/types/fetch.d.ts +231 -0
  186. package/dist/node_modules/undici/types/formdata.d.ts +114 -0
  187. package/dist/node_modules/undici/types/global-dispatcher.d.ts +9 -0
  188. package/dist/node_modules/undici/types/global-origin.d.ts +7 -0
  189. package/dist/node_modules/undici/types/h2c-client.d.ts +73 -0
  190. package/dist/node_modules/undici/types/handlers.d.ts +14 -0
  191. package/dist/node_modules/undici/types/header.d.ts +160 -0
  192. package/dist/node_modules/undici/types/index.d.ts +91 -0
  193. package/dist/node_modules/undici/types/interceptors.d.ts +80 -0
  194. package/dist/node_modules/undici/types/mock-agent.d.ts +68 -0
  195. package/dist/node_modules/undici/types/mock-call-history.d.ts +111 -0
  196. package/dist/node_modules/undici/types/mock-client.d.ts +27 -0
  197. package/dist/node_modules/undici/types/mock-errors.d.ts +12 -0
  198. package/dist/node_modules/undici/types/mock-interceptor.d.ts +94 -0
  199. package/dist/node_modules/undici/types/mock-pool.d.ts +27 -0
  200. package/dist/node_modules/undici/types/patch.d.ts +29 -0
  201. package/dist/node_modules/undici/types/pool-stats.d.ts +19 -0
  202. package/dist/node_modules/undici/types/pool.d.ts +41 -0
  203. package/dist/node_modules/undici/types/proxy-agent.d.ts +29 -0
  204. package/dist/node_modules/undici/types/readable.d.ts +68 -0
  205. package/dist/node_modules/undici/types/retry-agent.d.ts +8 -0
  206. package/dist/node_modules/undici/types/retry-handler.d.ts +125 -0
  207. package/dist/node_modules/undici/types/round-robin-pool.d.ts +41 -0
  208. package/dist/node_modules/undici/types/snapshot-agent.d.ts +109 -0
  209. package/dist/node_modules/undici/types/socks5-proxy-agent.d.ts +25 -0
  210. package/dist/node_modules/undici/types/util.d.ts +18 -0
  211. package/dist/node_modules/undici/types/utility.d.ts +7 -0
  212. package/dist/node_modules/undici/types/webidl.d.ts +347 -0
  213. package/dist/node_modules/undici/types/websocket.d.ts +188 -0
  214. package/dist/server/collections/oidcStates.d.ts +10 -0
  215. package/dist/server/collections/oidcStates.js +96 -0
  216. package/dist/server/db-adapter.d.ts +25 -0
  217. package/dist/server/db-adapter.js +156 -0
  218. package/dist/server/service.js +11 -10
  219. package/package.json +2 -2
  220. package/dist/server/cache-adapter.d.ts +0 -33
  221. package/dist/server/cache-adapter.js +0 -159
@@ -0,0 +1,158 @@
1
+ 'use strict'
2
+
3
+ const { InvalidArgumentError, MaxOriginsReachedError } = require('../core/errors')
4
+ const { kClients, kRunning, kClose, kDestroy, kDispatch, kUrl } = require('../core/symbols')
5
+ const DispatcherBase = require('./dispatcher-base')
6
+ const Pool = require('./pool')
7
+ const Client = require('./client')
8
+ const util = require('../core/util')
9
+
10
+ const kOnConnect = Symbol('onConnect')
11
+ const kOnDisconnect = Symbol('onDisconnect')
12
+ const kOnConnectionError = Symbol('onConnectionError')
13
+ const kOnDrain = Symbol('onDrain')
14
+ const kFactory = Symbol('factory')
15
+ const kOptions = Symbol('options')
16
+ const kOrigins = Symbol('origins')
17
+
18
+ function defaultFactory (origin, opts) {
19
+ return opts && opts.connections === 1
20
+ ? new Client(origin, opts)
21
+ : new Pool(origin, opts)
22
+ }
23
+
24
+ class Agent extends DispatcherBase {
25
+ constructor ({ factory = defaultFactory, maxOrigins = Infinity, connect, ...options } = {}) {
26
+ if (typeof factory !== 'function') {
27
+ throw new InvalidArgumentError('factory must be a function.')
28
+ }
29
+
30
+ if (connect != null && typeof connect !== 'function' && typeof connect !== 'object') {
31
+ throw new InvalidArgumentError('connect must be a function or an object')
32
+ }
33
+
34
+ if (typeof maxOrigins !== 'number' || Number.isNaN(maxOrigins) || maxOrigins <= 0) {
35
+ throw new InvalidArgumentError('maxOrigins must be a number greater than 0')
36
+ }
37
+
38
+ super()
39
+
40
+ if (connect && typeof connect !== 'function') {
41
+ connect = { ...connect }
42
+ }
43
+
44
+ this[kOptions] = { ...util.deepClone(options), maxOrigins, connect }
45
+ this[kFactory] = factory
46
+ this[kClients] = new Map()
47
+ this[kOrigins] = new Set()
48
+
49
+ this[kOnDrain] = (origin, targets) => {
50
+ this.emit('drain', origin, [this, ...targets])
51
+ }
52
+
53
+ this[kOnConnect] = (origin, targets) => {
54
+ this.emit('connect', origin, [this, ...targets])
55
+ }
56
+
57
+ this[kOnDisconnect] = (origin, targets, err) => {
58
+ this.emit('disconnect', origin, [this, ...targets], err)
59
+ }
60
+
61
+ this[kOnConnectionError] = (origin, targets, err) => {
62
+ this.emit('connectionError', origin, [this, ...targets], err)
63
+ }
64
+ }
65
+
66
+ get [kRunning] () {
67
+ let ret = 0
68
+ for (const { dispatcher } of this[kClients].values()) {
69
+ ret += dispatcher[kRunning]
70
+ }
71
+ return ret
72
+ }
73
+
74
+ [kDispatch] (opts, handler) {
75
+ let key
76
+ if (opts.origin && (typeof opts.origin === 'string' || opts.origin instanceof URL)) {
77
+ key = String(opts.origin)
78
+ } else {
79
+ throw new InvalidArgumentError('opts.origin must be a non-empty string or URL.')
80
+ }
81
+
82
+ if (this[kOrigins].size >= this[kOptions].maxOrigins && !this[kOrigins].has(key)) {
83
+ throw new MaxOriginsReachedError()
84
+ }
85
+
86
+ const result = this[kClients].get(key)
87
+ let dispatcher = result && result.dispatcher
88
+ if (!dispatcher) {
89
+ const closeClientIfUnused = (connected) => {
90
+ const result = this[kClients].get(key)
91
+ if (result) {
92
+ if (connected) result.count -= 1
93
+ if (result.count <= 0) {
94
+ this[kClients].delete(key)
95
+ if (!result.dispatcher.destroyed) {
96
+ result.dispatcher.close()
97
+ }
98
+ }
99
+ this[kOrigins].delete(key)
100
+ }
101
+ }
102
+ dispatcher = this[kFactory](opts.origin, this[kOptions])
103
+ .on('drain', this[kOnDrain])
104
+ .on('connect', (origin, targets) => {
105
+ const result = this[kClients].get(key)
106
+ if (result) {
107
+ result.count += 1
108
+ }
109
+ this[kOnConnect](origin, targets)
110
+ })
111
+ .on('disconnect', (origin, targets, err) => {
112
+ closeClientIfUnused(true)
113
+ this[kOnDisconnect](origin, targets, err)
114
+ })
115
+ .on('connectionError', (origin, targets, err) => {
116
+ closeClientIfUnused(false)
117
+ this[kOnConnectionError](origin, targets, err)
118
+ })
119
+
120
+ this[kClients].set(key, { count: 0, dispatcher })
121
+ this[kOrigins].add(key)
122
+ }
123
+
124
+ return dispatcher.dispatch(opts, handler)
125
+ }
126
+
127
+ [kClose] () {
128
+ const closePromises = []
129
+ for (const { dispatcher } of this[kClients].values()) {
130
+ closePromises.push(dispatcher.close())
131
+ }
132
+ this[kClients].clear()
133
+
134
+ return Promise.all(closePromises)
135
+ }
136
+
137
+ [kDestroy] (err) {
138
+ const destroyPromises = []
139
+ for (const { dispatcher } of this[kClients].values()) {
140
+ destroyPromises.push(dispatcher.destroy(err))
141
+ }
142
+ this[kClients].clear()
143
+
144
+ return Promise.all(destroyPromises)
145
+ }
146
+
147
+ get stats () {
148
+ const allClientStats = {}
149
+ for (const { dispatcher } of this[kClients].values()) {
150
+ if (dispatcher.stats) {
151
+ allClientStats[dispatcher[kUrl].origin] = dispatcher.stats
152
+ }
153
+ }
154
+ return allClientStats
155
+ }
156
+ }
157
+
158
+ module.exports = Agent
@@ -0,0 +1,219 @@
1
+ 'use strict'
2
+
3
+ const {
4
+ BalancedPoolMissingUpstreamError,
5
+ InvalidArgumentError
6
+ } = require('../core/errors')
7
+ const {
8
+ PoolBase,
9
+ kClients,
10
+ kNeedDrain,
11
+ kAddClient,
12
+ kRemoveClient,
13
+ kGetDispatcher
14
+ } = require('./pool-base')
15
+ const Pool = require('./pool')
16
+ const { kUrl } = require('../core/symbols')
17
+ const util = require('../core/util')
18
+ const kFactory = Symbol('factory')
19
+
20
+ const kOptions = Symbol('options')
21
+ const kGreatestCommonDivisor = Symbol('kGreatestCommonDivisor')
22
+ const kCurrentWeight = Symbol('kCurrentWeight')
23
+ const kIndex = Symbol('kIndex')
24
+ const kWeight = Symbol('kWeight')
25
+ const kMaxWeightPerServer = Symbol('kMaxWeightPerServer')
26
+ const kErrorPenalty = Symbol('kErrorPenalty')
27
+
28
+ /**
29
+ * Calculate the greatest common divisor of two numbers by
30
+ * using the Euclidean algorithm.
31
+ *
32
+ * @param {number} a
33
+ * @param {number} b
34
+ * @returns {number}
35
+ */
36
+ function getGreatestCommonDivisor (a, b) {
37
+ if (a === 0) return b
38
+
39
+ while (b !== 0) {
40
+ const t = b
41
+ b = a % b
42
+ a = t
43
+ }
44
+ return a
45
+ }
46
+
47
+ function defaultFactory (origin, opts) {
48
+ return new Pool(origin, opts)
49
+ }
50
+
51
+ class BalancedPool extends PoolBase {
52
+ constructor (upstreams = [], { factory = defaultFactory, ...opts } = {}) {
53
+ if (typeof factory !== 'function') {
54
+ throw new InvalidArgumentError('factory must be a function.')
55
+ }
56
+
57
+ super()
58
+
59
+ this[kOptions] = { ...util.deepClone(opts) }
60
+ this[kOptions].interceptors = opts.interceptors
61
+ ? { ...opts.interceptors }
62
+ : undefined
63
+ this[kIndex] = -1
64
+ this[kCurrentWeight] = 0
65
+
66
+ this[kMaxWeightPerServer] = this[kOptions].maxWeightPerServer || 100
67
+ this[kErrorPenalty] = this[kOptions].errorPenalty || 15
68
+
69
+ if (!Array.isArray(upstreams)) {
70
+ upstreams = [upstreams]
71
+ }
72
+
73
+ this[kFactory] = factory
74
+
75
+ for (const upstream of upstreams) {
76
+ this.addUpstream(upstream)
77
+ }
78
+ this._updateBalancedPoolStats()
79
+ }
80
+
81
+ addUpstream (upstream) {
82
+ const upstreamOrigin = util.parseOrigin(upstream).origin
83
+
84
+ if (this[kClients].find((pool) => (
85
+ pool[kUrl].origin === upstreamOrigin &&
86
+ pool.closed !== true &&
87
+ pool.destroyed !== true
88
+ ))) {
89
+ return this
90
+ }
91
+ const pool = this[kFactory](upstreamOrigin, this[kOptions])
92
+
93
+ this[kAddClient](pool)
94
+ pool.on('connect', () => {
95
+ pool[kWeight] = Math.min(this[kMaxWeightPerServer], pool[kWeight] + this[kErrorPenalty])
96
+ })
97
+
98
+ pool.on('connectionError', () => {
99
+ pool[kWeight] = Math.max(1, pool[kWeight] - this[kErrorPenalty])
100
+ this._updateBalancedPoolStats()
101
+ })
102
+
103
+ pool.on('disconnect', (...args) => {
104
+ const err = args[2]
105
+ if (err && err.code === 'UND_ERR_SOCKET') {
106
+ // decrease the weight of the pool.
107
+ pool[kWeight] = Math.max(1, pool[kWeight] - this[kErrorPenalty])
108
+ this._updateBalancedPoolStats()
109
+ }
110
+ })
111
+
112
+ for (const client of this[kClients]) {
113
+ client[kWeight] = this[kMaxWeightPerServer]
114
+ }
115
+
116
+ this._updateBalancedPoolStats()
117
+
118
+ return this
119
+ }
120
+
121
+ _updateBalancedPoolStats () {
122
+ let result = 0
123
+ for (let i = 0; i < this[kClients].length; i++) {
124
+ result = getGreatestCommonDivisor(this[kClients][i][kWeight], result)
125
+ }
126
+
127
+ this[kGreatestCommonDivisor] = result
128
+ }
129
+
130
+ removeUpstream (upstream) {
131
+ const upstreamOrigin = util.parseOrigin(upstream).origin
132
+
133
+ const pool = this[kClients].find((pool) => (
134
+ pool[kUrl].origin === upstreamOrigin &&
135
+ pool.closed !== true &&
136
+ pool.destroyed !== true
137
+ ))
138
+
139
+ if (pool) {
140
+ this[kRemoveClient](pool)
141
+ }
142
+
143
+ return this
144
+ }
145
+
146
+ getUpstream (upstream) {
147
+ const upstreamOrigin = util.parseOrigin(upstream).origin
148
+
149
+ return this[kClients].find((pool) => (
150
+ pool[kUrl].origin === upstreamOrigin &&
151
+ pool.closed !== true &&
152
+ pool.destroyed !== true
153
+ ))
154
+ }
155
+
156
+ get upstreams () {
157
+ return this[kClients]
158
+ .filter(dispatcher => dispatcher.closed !== true && dispatcher.destroyed !== true)
159
+ .map((p) => p[kUrl].origin)
160
+ }
161
+
162
+ [kGetDispatcher] () {
163
+ // We validate that pools is greater than 0,
164
+ // otherwise we would have to wait until an upstream
165
+ // is added, which might never happen.
166
+ if (this[kClients].length === 0) {
167
+ throw new BalancedPoolMissingUpstreamError()
168
+ }
169
+
170
+ const dispatcher = this[kClients].find(dispatcher => (
171
+ !dispatcher[kNeedDrain] &&
172
+ dispatcher.closed !== true &&
173
+ dispatcher.destroyed !== true
174
+ ))
175
+
176
+ if (!dispatcher) {
177
+ return
178
+ }
179
+
180
+ const allClientsBusy = this[kClients].map(pool => pool[kNeedDrain]).reduce((a, b) => a && b, true)
181
+
182
+ if (allClientsBusy) {
183
+ return
184
+ }
185
+
186
+ let counter = 0
187
+
188
+ let maxWeightIndex = this[kClients].findIndex(pool => !pool[kNeedDrain])
189
+
190
+ while (counter++ < this[kClients].length) {
191
+ this[kIndex] = (this[kIndex] + 1) % this[kClients].length
192
+ const pool = this[kClients][this[kIndex]]
193
+
194
+ // find pool index with the largest weight
195
+ if (pool[kWeight] > this[kClients][maxWeightIndex][kWeight] && !pool[kNeedDrain]) {
196
+ maxWeightIndex = this[kIndex]
197
+ }
198
+
199
+ // decrease the current weight every `this[kClients].length`.
200
+ if (this[kIndex] === 0) {
201
+ // Set the current weight to the next lower weight.
202
+ this[kCurrentWeight] = this[kCurrentWeight] - this[kGreatestCommonDivisor]
203
+
204
+ if (this[kCurrentWeight] <= 0) {
205
+ this[kCurrentWeight] = this[kMaxWeightPerServer]
206
+ }
207
+ }
208
+ if (pool[kWeight] >= this[kCurrentWeight] && (!pool[kNeedDrain])) {
209
+ return pool
210
+ }
211
+ }
212
+
213
+ this[kCurrentWeight] = this[kClients][maxWeightIndex][kWeight]
214
+ this[kIndex] = maxWeightIndex
215
+ return this[kClients][maxWeightIndex]
216
+ }
217
+ }
218
+
219
+ module.exports = BalancedPool