velocious 1.0.430 → 1.0.431

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 (811) hide show
  1. package/bin/velocious.js +48 -0
  2. package/build/bin/velocious.js +39 -34
  3. package/build/index.js +1 -2
  4. package/build/src/application.js +214 -187
  5. package/build/src/authorization/ability.d.ts +24 -23
  6. package/build/src/authorization/ability.d.ts.map +1 -1
  7. package/build/src/authorization/ability.js +300 -252
  8. package/build/src/authorization/base-resource.d.ts +20 -26
  9. package/build/src/authorization/base-resource.d.ts.map +1 -1
  10. package/build/src/authorization/base-resource.js +136 -118
  11. package/build/src/background-jobs/client.js +47 -43
  12. package/build/src/background-jobs/cron-expression.js +166 -127
  13. package/build/src/background-jobs/forked-runner-child.js +47 -37
  14. package/build/src/background-jobs/job-record.js +10 -8
  15. package/build/src/background-jobs/job-registry.js +84 -72
  16. package/build/src/background-jobs/job-runner.js +81 -74
  17. package/build/src/background-jobs/job.js +72 -62
  18. package/build/src/background-jobs/json-socket.js +70 -65
  19. package/build/src/background-jobs/main.js +900 -841
  20. package/build/src/background-jobs/normalize-error.js +11 -12
  21. package/build/src/background-jobs/scheduler.js +247 -205
  22. package/build/src/background-jobs/socket-request.js +65 -60
  23. package/build/src/background-jobs/status-reporter.js +96 -86
  24. package/build/src/background-jobs/store.js +980 -862
  25. package/build/src/background-jobs/types.js +3 -2
  26. package/build/src/background-jobs/web/authorization.js +50 -38
  27. package/build/src/background-jobs/web/controller.js +268 -232
  28. package/build/src/background-jobs/web/index.js +40 -36
  29. package/build/src/background-jobs/web/path-matcher.js +48 -45
  30. package/build/src/background-jobs/web/registry.js +14 -9
  31. package/build/src/background-jobs/worker.js +639 -585
  32. package/build/src/beacon/client.js +293 -264
  33. package/build/src/beacon/in-process-broker.js +25 -20
  34. package/build/src/beacon/in-process-client.js +116 -104
  35. package/build/src/beacon/server.js +126 -110
  36. package/build/src/beacon/types.js +8 -2
  37. package/build/src/cli/base-command.js +57 -49
  38. package/build/src/cli/browser-cli.js +42 -37
  39. package/build/src/cli/commands/background-jobs-main.js +5 -5
  40. package/build/src/cli/commands/background-jobs-runner.js +5 -5
  41. package/build/src/cli/commands/background-jobs-worker.js +5 -5
  42. package/build/src/cli/commands/beacon.js +5 -5
  43. package/build/src/cli/commands/console.js +10 -10
  44. package/build/src/cli/commands/db/base-command.js +76 -71
  45. package/build/src/cli/commands/db/create.js +61 -53
  46. package/build/src/cli/commands/db/drop.js +71 -62
  47. package/build/src/cli/commands/db/migrate.js +15 -13
  48. package/build/src/cli/commands/db/reset.js +19 -16
  49. package/build/src/cli/commands/db/rollback.js +13 -12
  50. package/build/src/cli/commands/db/schema/dump.js +9 -9
  51. package/build/src/cli/commands/db/schema/load.js +9 -9
  52. package/build/src/cli/commands/db/seed.js +9 -9
  53. package/build/src/cli/commands/db/tenants/check.js +35 -32
  54. package/build/src/cli/commands/db/tenants/create.js +29 -26
  55. package/build/src/cli/commands/db/tenants/migrate.js +44 -40
  56. package/build/src/cli/commands/destroy/migration.js +5 -5
  57. package/build/src/cli/commands/generate/base-models.js +5 -5
  58. package/build/src/cli/commands/generate/frontend-models.js +9 -9
  59. package/build/src/cli/commands/generate/migration.js +5 -5
  60. package/build/src/cli/commands/generate/model.js +5 -5
  61. package/build/src/cli/commands/init.js +9 -7
  62. package/build/src/cli/commands/routes.js +6 -6
  63. package/build/src/cli/commands/run-script.js +9 -9
  64. package/build/src/cli/commands/runner.js +9 -9
  65. package/build/src/cli/commands/server.js +6 -6
  66. package/build/src/cli/commands/test.js +7 -6
  67. package/build/src/cli/index.js +141 -127
  68. package/build/src/cli/tenant-database-command-helper.js +185 -154
  69. package/build/src/cli/use-browser-cli.js +20 -15
  70. package/build/src/configuration-resolver.js +54 -47
  71. package/build/src/configuration-types.d.ts +21 -2
  72. package/build/src/configuration-types.d.ts.map +1 -1
  73. package/build/src/configuration-types.js +60 -3
  74. package/build/src/configuration.js +2547 -2240
  75. package/build/src/controller.js +407 -363
  76. package/build/src/current-configuration.js +12 -9
  77. package/build/src/current.js +75 -70
  78. package/build/src/database/annotations-async-hooks.js +22 -16
  79. package/build/src/database/annotations.js +18 -12
  80. package/build/src/database/drivers/base-column.js +179 -155
  81. package/build/src/database/drivers/base-columns-index.js +78 -69
  82. package/build/src/database/drivers/base-foreign-key.js +101 -89
  83. package/build/src/database/drivers/base-table.js +149 -124
  84. package/build/src/database/drivers/base.js +1489 -1306
  85. package/build/src/database/drivers/mssql/column.js +50 -39
  86. package/build/src/database/drivers/mssql/columns-index.js +3 -2
  87. package/build/src/database/drivers/mssql/connect-connection.js +9 -11
  88. package/build/src/database/drivers/mssql/foreign-key.js +9 -8
  89. package/build/src/database/drivers/mssql/index.js +587 -507
  90. package/build/src/database/drivers/mssql/options.js +75 -68
  91. package/build/src/database/drivers/mssql/query-parser.js +3 -2
  92. package/build/src/database/drivers/mssql/sql/alter-table.js +2 -2
  93. package/build/src/database/drivers/mssql/sql/create-database.js +31 -24
  94. package/build/src/database/drivers/mssql/sql/create-index.js +2 -2
  95. package/build/src/database/drivers/mssql/sql/create-table.js +2 -2
  96. package/build/src/database/drivers/mssql/sql/delete.js +16 -14
  97. package/build/src/database/drivers/mssql/sql/drop-database.js +31 -24
  98. package/build/src/database/drivers/mssql/sql/drop-table.js +2 -2
  99. package/build/src/database/drivers/mssql/sql/insert.js +2 -2
  100. package/build/src/database/drivers/mssql/sql/update.js +28 -24
  101. package/build/src/database/drivers/mssql/sql/upsert.js +20 -18
  102. package/build/src/database/drivers/mssql/structure-sql.js +114 -102
  103. package/build/src/database/drivers/mssql/table.js +96 -81
  104. package/build/src/database/drivers/mysql/column.js +92 -75
  105. package/build/src/database/drivers/mysql/columns-index.js +19 -16
  106. package/build/src/database/drivers/mysql/foreign-key.js +9 -8
  107. package/build/src/database/drivers/mysql/index.js +457 -396
  108. package/build/src/database/drivers/mysql/options.js +30 -26
  109. package/build/src/database/drivers/mysql/query-parser.js +3 -2
  110. package/build/src/database/drivers/mysql/query.js +29 -26
  111. package/build/src/database/drivers/mysql/sql/alter-table.js +3 -2
  112. package/build/src/database/drivers/mysql/sql/create-database.js +28 -23
  113. package/build/src/database/drivers/mysql/sql/create-index.js +3 -2
  114. package/build/src/database/drivers/mysql/sql/create-table.js +3 -2
  115. package/build/src/database/drivers/mysql/sql/delete.js +17 -14
  116. package/build/src/database/drivers/mysql/sql/drop-database.js +3 -2
  117. package/build/src/database/drivers/mysql/sql/drop-table.js +3 -2
  118. package/build/src/database/drivers/mysql/sql/insert.js +3 -2
  119. package/build/src/database/drivers/mysql/sql/update.js +29 -24
  120. package/build/src/database/drivers/mysql/sql/upsert.js +10 -8
  121. package/build/src/database/drivers/mysql/structure-sql.js +88 -79
  122. package/build/src/database/drivers/mysql/table.js +98 -83
  123. package/build/src/database/drivers/pgsql/column.js +72 -56
  124. package/build/src/database/drivers/pgsql/columns-index.js +3 -2
  125. package/build/src/database/drivers/pgsql/foreign-key.js +9 -8
  126. package/build/src/database/drivers/pgsql/index.js +438 -377
  127. package/build/src/database/drivers/pgsql/options.js +28 -25
  128. package/build/src/database/drivers/pgsql/query-parser.js +3 -2
  129. package/build/src/database/drivers/pgsql/sql/alter-table.js +3 -2
  130. package/build/src/database/drivers/pgsql/sql/create-database.js +23 -19
  131. package/build/src/database/drivers/pgsql/sql/create-index.js +3 -2
  132. package/build/src/database/drivers/pgsql/sql/create-table.js +3 -2
  133. package/build/src/database/drivers/pgsql/sql/delete.js +17 -14
  134. package/build/src/database/drivers/pgsql/sql/drop-database.js +3 -2
  135. package/build/src/database/drivers/pgsql/sql/drop-table.js +3 -2
  136. package/build/src/database/drivers/pgsql/sql/insert.js +3 -2
  137. package/build/src/database/drivers/pgsql/sql/update.js +29 -24
  138. package/build/src/database/drivers/pgsql/sql/upsert.js +11 -9
  139. package/build/src/database/drivers/pgsql/structure-sql.js +120 -108
  140. package/build/src/database/drivers/pgsql/table.js +77 -60
  141. package/build/src/database/drivers/sqlite/base.js +478 -405
  142. package/build/src/database/drivers/sqlite/column.js +69 -54
  143. package/build/src/database/drivers/sqlite/columns-index.js +27 -22
  144. package/build/src/database/drivers/sqlite/connection-sql-js.js +42 -35
  145. package/build/src/database/drivers/sqlite/foreign-key.js +21 -18
  146. package/build/src/database/drivers/sqlite/index.js +373 -330
  147. package/build/src/database/drivers/sqlite/index.native.js +64 -55
  148. package/build/src/database/drivers/sqlite/index.web.js +87 -69
  149. package/build/src/database/drivers/sqlite/options.js +28 -25
  150. package/build/src/database/drivers/sqlite/query-parser.js +3 -2
  151. package/build/src/database/drivers/sqlite/query.js +24 -21
  152. package/build/src/database/drivers/sqlite/query.native.js +25 -20
  153. package/build/src/database/drivers/sqlite/query.web.js +37 -30
  154. package/build/src/database/drivers/sqlite/sql/alter-table.js +179 -159
  155. package/build/src/database/drivers/sqlite/sql/create-index.js +3 -2
  156. package/build/src/database/drivers/sqlite/sql/create-table.js +3 -2
  157. package/build/src/database/drivers/sqlite/sql/delete.js +22 -17
  158. package/build/src/database/drivers/sqlite/sql/drop-table.js +3 -2
  159. package/build/src/database/drivers/sqlite/sql/insert.js +3 -2
  160. package/build/src/database/drivers/sqlite/sql/update.js +29 -24
  161. package/build/src/database/drivers/sqlite/sql/upsert.js +11 -9
  162. package/build/src/database/drivers/sqlite/structure-sql.js +52 -49
  163. package/build/src/database/drivers/sqlite/table-rebuilder.js +75 -62
  164. package/build/src/database/drivers/sqlite/table.js +125 -102
  165. package/build/src/database/drivers/structure-sql/utils.js +17 -14
  166. package/build/src/database/handler.js +10 -9
  167. package/build/src/database/initializer-from-require-context.js +87 -76
  168. package/build/src/database/migration/index.js +395 -332
  169. package/build/src/database/migrator/files-finder.js +50 -40
  170. package/build/src/database/migrator/types.js +30 -2
  171. package/build/src/database/migrator.js +526 -454
  172. package/build/src/database/pool/async-tracked-multi-connection.js +1147 -997
  173. package/build/src/database/pool/base-methods-forward.js +43 -40
  174. package/build/src/database/pool/base.js +343 -298
  175. package/build/src/database/pool/single-multi-use.js +110 -93
  176. package/build/src/database/query/alter-table-base.js +99 -84
  177. package/build/src/database/query/base.js +46 -39
  178. package/build/src/database/query/create-database-base.js +30 -25
  179. package/build/src/database/query/create-index-base.js +94 -75
  180. package/build/src/database/query/create-table-base.js +193 -151
  181. package/build/src/database/query/delete-base.js +16 -14
  182. package/build/src/database/query/drop-database-base.js +28 -23
  183. package/build/src/database/query/drop-table-base.js +53 -42
  184. package/build/src/database/query/from-base.js +33 -30
  185. package/build/src/database/query/from-plain.js +13 -11
  186. package/build/src/database/query/from-table.js +15 -13
  187. package/build/src/database/query/index.js +472 -410
  188. package/build/src/database/query/insert-base.js +164 -143
  189. package/build/src/database/query/join-base.js +40 -35
  190. package/build/src/database/query/join-object.js +153 -128
  191. package/build/src/database/query/join-plain.js +15 -13
  192. package/build/src/database/query/join-tracker.js +90 -76
  193. package/build/src/database/query/model-class-query.js +1370 -1134
  194. package/build/src/database/query/order-base.js +30 -27
  195. package/build/src/database/query/order-column.js +53 -44
  196. package/build/src/database/query/order-plain.js +24 -20
  197. package/build/src/database/query/preloader/belongs-to.js +258 -210
  198. package/build/src/database/query/preloader/ensure-model-class-initialized.js +9 -8
  199. package/build/src/database/query/preloader/has-many.js +301 -240
  200. package/build/src/database/query/preloader/has-one.js +117 -91
  201. package/build/src/database/query/preloader/selection.js +129 -117
  202. package/build/src/database/query/preloader.js +185 -160
  203. package/build/src/database/query/query-data.js +201 -157
  204. package/build/src/database/query/select-base.js +27 -25
  205. package/build/src/database/query/select-plain.js +15 -13
  206. package/build/src/database/query/select-table-and-column.js +25 -21
  207. package/build/src/database/query/update-base.js +38 -35
  208. package/build/src/database/query/upsert-base.js +100 -93
  209. package/build/src/database/query/where-base.js +35 -32
  210. package/build/src/database/query/where-combinator.d.ts.map +1 -1
  211. package/build/src/database/query/where-combinator.js +28 -26
  212. package/build/src/database/query/where-hash.js +68 -61
  213. package/build/src/database/query/where-model-class-hash.js +469 -414
  214. package/build/src/database/query/where-not.js +20 -18
  215. package/build/src/database/query/where-plain.js +17 -15
  216. package/build/src/database/query/with-count.js +159 -125
  217. package/build/src/database/query-parser/base-query-parser.js +37 -32
  218. package/build/src/database/query-parser/from-parser.js +45 -36
  219. package/build/src/database/query-parser/group-parser.js +50 -42
  220. package/build/src/database/query-parser/joins-parser.js +33 -28
  221. package/build/src/database/query-parser/limit-parser.js +70 -67
  222. package/build/src/database/query-parser/options.js +82 -75
  223. package/build/src/database/query-parser/order-parser.js +40 -36
  224. package/build/src/database/query-parser/select-parser.js +60 -49
  225. package/build/src/database/query-parser/where-parser.js +41 -36
  226. package/build/src/database/record/acts-as-list.js +273 -235
  227. package/build/src/database/record/attachments/download.js +45 -44
  228. package/build/src/database/record/attachments/handle.js +161 -141
  229. package/build/src/database/record/attachments/normalize-input.js +138 -128
  230. package/build/src/database/record/attachments/storage-drivers/filesystem.js +91 -77
  231. package/build/src/database/record/attachments/storage-drivers/native.js +121 -112
  232. package/build/src/database/record/attachments/storage-drivers/s3.js +208 -177
  233. package/build/src/database/record/attachments/store.d.ts +1 -1
  234. package/build/src/database/record/attachments/store.d.ts.map +1 -1
  235. package/build/src/database/record/attachments/store.js +540 -468
  236. package/build/src/database/record/index.d.ts +17 -15
  237. package/build/src/database/record/index.d.ts.map +1 -1
  238. package/build/src/database/record/index.js +3894 -3361
  239. package/build/src/database/record/instance-relationships/base.js +268 -234
  240. package/build/src/database/record/instance-relationships/belongs-to.js +73 -58
  241. package/build/src/database/record/instance-relationships/has-many.js +264 -225
  242. package/build/src/database/record/instance-relationships/has-one.js +105 -85
  243. package/build/src/database/record/record-not-found-error.js +2 -3
  244. package/build/src/database/record/relationships/base.d.ts +2 -2
  245. package/build/src/database/record/relationships/base.d.ts.map +1 -1
  246. package/build/src/database/record/relationships/base.js +167 -145
  247. package/build/src/database/record/relationships/belongs-to.js +51 -44
  248. package/build/src/database/record/relationships/has-many.js +40 -32
  249. package/build/src/database/record/relationships/has-one.js +40 -32
  250. package/build/src/database/record/state-machine.js +208 -156
  251. package/build/src/database/record/user-module.js +38 -32
  252. package/build/src/database/record/validators/base.js +24 -22
  253. package/build/src/database/record/validators/format.js +46 -36
  254. package/build/src/database/record/validators/presence.js +20 -18
  255. package/build/src/database/record/validators/uniqueness.js +117 -99
  256. package/build/src/database/table-data/index.js +231 -199
  257. package/build/src/database/table-data/table-column.js +382 -338
  258. package/build/src/database/table-data/table-foreign-key.js +66 -57
  259. package/build/src/database/table-data/table-index.js +36 -29
  260. package/build/src/database/table-data/table-reference.js +10 -10
  261. package/build/src/database/use-database.js +40 -32
  262. package/build/src/environment-handlers/base.js +544 -484
  263. package/build/src/environment-handlers/browser.js +294 -241
  264. package/build/src/environment-handlers/node/cli/commands/background-jobs-main.js +19 -16
  265. package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.js +21 -18
  266. package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.js +29 -22
  267. package/build/src/environment-handlers/node/cli/commands/beacon.js +19 -16
  268. package/build/src/environment-handlers/node/cli/commands/cli-command-context.js +15 -14
  269. package/build/src/environment-handlers/node/cli/commands/console.js +120 -99
  270. package/build/src/environment-handlers/node/cli/commands/db/schema/dump.js +39 -34
  271. package/build/src/environment-handlers/node/cli/commands/db/schema/load.js +63 -57
  272. package/build/src/environment-handlers/node/cli/commands/db/seed.js +63 -51
  273. package/build/src/environment-handlers/node/cli/commands/destroy/migration.js +40 -32
  274. package/build/src/environment-handlers/node/cli/commands/generate/base-models.d.ts.map +1 -1
  275. package/build/src/environment-handlers/node/cli/commands/generate/base-models.js +353 -298
  276. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +844 -729
  277. package/build/src/environment-handlers/node/cli/commands/generate/migration.js +38 -34
  278. package/build/src/environment-handlers/node/cli/commands/generate/model.js +38 -34
  279. package/build/src/environment-handlers/node/cli/commands/init.js +61 -56
  280. package/build/src/environment-handlers/node/cli/commands/routes.js +59 -51
  281. package/build/src/environment-handlers/node/cli/commands/run-script.js +68 -54
  282. package/build/src/environment-handlers/node/cli/commands/runner.js +74 -56
  283. package/build/src/environment-handlers/node/cli/commands/server.js +106 -93
  284. package/build/src/environment-handlers/node/cli/commands/test.js +113 -97
  285. package/build/src/environment-handlers/node.js +874 -753
  286. package/build/src/error-logger.js +21 -22
  287. package/build/src/frontend-model-controller.d.ts +6 -6
  288. package/build/src/frontend-model-controller.d.ts.map +1 -1
  289. package/build/src/frontend-model-controller.js +3288 -2788
  290. package/build/src/frontend-model-resource/base-resource.d.ts +18 -17
  291. package/build/src/frontend-model-resource/base-resource.d.ts.map +1 -1
  292. package/build/src/frontend-model-resource/base-resource.js +869 -759
  293. package/build/src/frontend-models/base.d.ts +19 -12
  294. package/build/src/frontend-models/base.d.ts.map +1 -1
  295. package/build/src/frontend-models/base.js +3602 -3114
  296. package/build/src/frontend-models/clear-pending-debounced-callback.js +8 -7
  297. package/build/src/frontend-models/event-hook-models.js +21 -16
  298. package/build/src/frontend-models/model-registry.js +11 -9
  299. package/build/src/frontend-models/outgoing-event-buffer.js +17 -10
  300. package/build/src/frontend-models/preloader.d.ts +6 -6
  301. package/build/src/frontend-models/preloader.d.ts.map +1 -1
  302. package/build/src/frontend-models/preloader.js +149 -131
  303. package/build/src/frontend-models/query.d.ts.map +1 -1
  304. package/build/src/frontend-models/query.js +1855 -1560
  305. package/build/src/frontend-models/resource-config-validation.js +37 -27
  306. package/build/src/frontend-models/resource-definition.js +288 -234
  307. package/build/src/frontend-models/transport-serialization.js +266 -203
  308. package/build/src/frontend-models/use-created-event.js +7 -5
  309. package/build/src/frontend-models/use-destroyed-event.js +93 -80
  310. package/build/src/frontend-models/use-model-class-event.js +91 -79
  311. package/build/src/frontend-models/use-updated-event.js +97 -84
  312. package/build/src/frontend-models/websocket-channel.js +441 -381
  313. package/build/src/frontend-models/websocket-publishers.js +173 -140
  314. package/build/src/http-client/header.js +14 -13
  315. package/build/src/http-client/index.js +132 -116
  316. package/build/src/http-client/request.js +87 -71
  317. package/build/src/http-client/response.js +140 -122
  318. package/build/src/http-client/websocket-client.js +17 -15
  319. package/build/src/http-server/client/index.js +465 -409
  320. package/build/src/http-server/client/params-to-object.js +135 -124
  321. package/build/src/http-server/client/request-buffer/form-data-part.js +132 -111
  322. package/build/src/http-server/client/request-buffer/header.js +16 -15
  323. package/build/src/http-server/client/request-buffer/index.js +506 -446
  324. package/build/src/http-server/client/request-parser.js +186 -163
  325. package/build/src/http-server/client/request-runner.js +259 -226
  326. package/build/src/http-server/client/request-timing.js +151 -132
  327. package/build/src/http-server/client/request.js +108 -96
  328. package/build/src/http-server/client/response.js +235 -213
  329. package/build/src/http-server/client/uploaded-file/memory-uploaded-file.js +29 -25
  330. package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.js +29 -25
  331. package/build/src/http-server/client/uploaded-file/uploaded-file.js +33 -33
  332. package/build/src/http-server/client/websocket-request.js +137 -114
  333. package/build/src/http-server/client/websocket-session.js +1657 -1452
  334. package/build/src/http-server/cookie.js +236 -216
  335. package/build/src/http-server/development-reloader.js +221 -190
  336. package/build/src/http-server/index.js +525 -451
  337. package/build/src/http-server/remote-address.js +50 -38
  338. package/build/src/http-server/server-client.js +208 -181
  339. package/build/src/http-server/server-lock.js +167 -153
  340. package/build/src/http-server/websocket-channel-subscribers.js +93 -81
  341. package/build/src/http-server/websocket-channel.js +117 -104
  342. package/build/src/http-server/websocket-connection.js +104 -96
  343. package/build/src/http-server/websocket-event-log-store.js +404 -350
  344. package/build/src/http-server/websocket-events-host.js +164 -145
  345. package/build/src/http-server/websocket-events.js +47 -47
  346. package/build/src/http-server/worker-handler/channel-subscriber-dispatch.js +14 -13
  347. package/build/src/http-server/worker-handler/in-process.js +141 -123
  348. package/build/src/http-server/worker-handler/index.js +349 -313
  349. package/build/src/http-server/worker-handler/worker-script.js +5 -4
  350. package/build/src/http-server/worker-handler/worker-thread.js +269 -240
  351. package/build/src/initializer.js +36 -31
  352. package/build/src/jobs/mail-delivery.js +15 -13
  353. package/build/src/logger/base-logger.js +26 -24
  354. package/build/src/logger/console-logger.js +23 -21
  355. package/build/src/logger/file-logger.js +31 -29
  356. package/build/src/logger/outputs/array-output.js +42 -37
  357. package/build/src/logger/outputs/console-output.js +24 -20
  358. package/build/src/logger/outputs/file-output.js +48 -43
  359. package/build/src/logger/outputs/stdout-output.js +48 -39
  360. package/build/src/logger.js +394 -338
  361. package/build/src/mailer/backends/smtp.js +163 -134
  362. package/build/src/mailer/base.js +251 -211
  363. package/build/src/mailer/delivery.js +64 -56
  364. package/build/src/mailer/index.js +22 -4
  365. package/build/src/mailer.js +13 -4
  366. package/build/src/plugins/sqljs-wasm-route-controller.js +52 -42
  367. package/build/src/plugins/sqljs-wasm-route.js +38 -28
  368. package/build/src/record-payload-values.js +28 -25
  369. package/build/src/routes/app-routes.js +14 -12
  370. package/build/src/routes/base-route.js +130 -112
  371. package/build/src/routes/basic-route.js +102 -83
  372. package/build/src/routes/built-in/debug/controller.js +10 -10
  373. package/build/src/routes/built-in/errors/controller.js +5 -5
  374. package/build/src/routes/get-route.js +63 -50
  375. package/build/src/routes/hooks/frontend-model-command-route-hook.js +80 -66
  376. package/build/src/routes/index.js +43 -36
  377. package/build/src/routes/namespace-route.js +47 -38
  378. package/build/src/routes/plugin-routes.js +124 -107
  379. package/build/src/routes/post-route.js +62 -51
  380. package/build/src/routes/resolver.js +494 -422
  381. package/build/src/routes/resource-route.js +143 -124
  382. package/build/src/routes/root-route.js +8 -7
  383. package/build/src/testing/base-expect.js +14 -13
  384. package/build/src/testing/browser-frontend-model-event-hook-scenarios.js +405 -329
  385. package/build/src/testing/browser-test-app.js +29 -23
  386. package/build/src/testing/expect-to-change.js +50 -41
  387. package/build/src/testing/expect-utils.js +184 -139
  388. package/build/src/testing/expect.js +731 -638
  389. package/build/src/testing/request-client.js +85 -70
  390. package/build/src/testing/test-files-finder.js +339 -285
  391. package/build/src/testing/test-filter-parser.js +155 -124
  392. package/build/src/testing/test-runner.js +1020 -883
  393. package/build/src/testing/test-suite-splitter.js +142 -114
  394. package/build/src/testing/test.js +256 -216
  395. package/build/src/utils/backtrace-cleaner-node.js +69 -62
  396. package/build/src/utils/backtrace-cleaner.js +216 -188
  397. package/build/src/utils/ensure-error.js +7 -7
  398. package/build/src/utils/event-emitter.js +6 -4
  399. package/build/src/utils/file-exists.js +10 -9
  400. package/build/src/utils/format-value.js +76 -67
  401. package/build/src/utils/model-scope.js +31 -27
  402. package/build/src/utils/nest-callbacks.js +13 -10
  403. package/build/src/utils/plain-object.js +6 -5
  404. package/build/src/utils/ransack.d.ts.map +1 -1
  405. package/build/src/utils/ransack.js +563 -449
  406. package/build/src/utils/rest-args-error.js +6 -5
  407. package/build/src/utils/singularize-model-name.js +11 -9
  408. package/build/src/utils/split-sql-statements.js +79 -68
  409. package/build/src/utils/to-import-specifier.js +30 -24
  410. package/build/src/utils/with-tracked-stack-async-hooks.js +74 -60
  411. package/build/src/utils/with-tracked-stack.js +18 -14
  412. package/build/src/velocious-error.js +30 -27
  413. package/index.js +1 -0
  414. package/package.json +10 -4
  415. package/scripts/clean-build.js +8 -0
  416. package/scripts/ensure-bin-executable.js +13 -0
  417. package/scripts/run-tests.js +37 -0
  418. package/scripts/test-browser.js +486 -0
  419. package/src/application.js +229 -0
  420. package/src/authorization/ability.js +329 -0
  421. package/src/authorization/base-resource.js +143 -0
  422. package/src/background-jobs/client.js +50 -0
  423. package/src/background-jobs/cron-expression.js +277 -0
  424. package/src/background-jobs/forked-runner-child.js +86 -0
  425. package/src/background-jobs/job-record.js +13 -0
  426. package/src/background-jobs/job-registry.js +92 -0
  427. package/src/background-jobs/job-runner.js +107 -0
  428. package/src/background-jobs/job.js +77 -0
  429. package/src/background-jobs/json-socket.js +78 -0
  430. package/src/background-jobs/main.js +926 -0
  431. package/src/background-jobs/normalize-error.js +26 -0
  432. package/src/background-jobs/scheduler.js +274 -0
  433. package/src/background-jobs/socket-request.js +68 -0
  434. package/src/background-jobs/status-reporter.js +101 -0
  435. package/src/background-jobs/store.js +994 -0
  436. package/src/background-jobs/types.js +70 -0
  437. package/src/background-jobs/web/authorization.js +89 -0
  438. package/src/background-jobs/web/controller.js +280 -0
  439. package/src/background-jobs/web/index.js +57 -0
  440. package/src/background-jobs/web/path-matcher.js +74 -0
  441. package/src/background-jobs/web/registry.js +49 -0
  442. package/src/background-jobs/worker.js +683 -0
  443. package/src/beacon/client.js +330 -0
  444. package/src/beacon/in-process-broker.js +71 -0
  445. package/src/beacon/in-process-client.js +139 -0
  446. package/src/beacon/server.js +148 -0
  447. package/src/beacon/types.js +55 -0
  448. package/src/cli/base-command.js +67 -0
  449. package/src/cli/browser-cli.js +45 -0
  450. package/src/cli/commands/background-jobs-main.js +7 -0
  451. package/src/cli/commands/background-jobs-runner.js +7 -0
  452. package/src/cli/commands/background-jobs-worker.js +7 -0
  453. package/src/cli/commands/beacon.js +7 -0
  454. package/src/cli/commands/console.js +12 -0
  455. package/src/cli/commands/db/base-command.js +82 -0
  456. package/src/cli/commands/db/create.js +64 -0
  457. package/src/cli/commands/db/drop.js +75 -0
  458. package/src/cli/commands/db/migrate.js +17 -0
  459. package/src/cli/commands/db/reset.js +22 -0
  460. package/src/cli/commands/db/rollback.js +15 -0
  461. package/src/cli/commands/db/schema/dump.js +12 -0
  462. package/src/cli/commands/db/schema/load.js +12 -0
  463. package/src/cli/commands/db/seed.js +12 -0
  464. package/src/cli/commands/db/tenants/check.js +38 -0
  465. package/src/cli/commands/db/tenants/create.js +33 -0
  466. package/src/cli/commands/db/tenants/migrate.js +49 -0
  467. package/src/cli/commands/destroy/migration.js +7 -0
  468. package/src/cli/commands/generate/base-models.js +7 -0
  469. package/src/cli/commands/generate/frontend-models.js +12 -0
  470. package/src/cli/commands/generate/migration.js +7 -0
  471. package/src/cli/commands/generate/model.js +7 -0
  472. package/src/cli/commands/init.js +11 -0
  473. package/src/cli/commands/routes.js +7 -0
  474. package/src/cli/commands/run-script.js +12 -0
  475. package/src/cli/commands/runner.js +12 -0
  476. package/src/cli/commands/server.js +7 -0
  477. package/src/cli/commands/test.js +9 -0
  478. package/src/cli/index.js +152 -0
  479. package/src/cli/tenant-database-command-helper.js +198 -0
  480. package/src/cli/use-browser-cli.js +30 -0
  481. package/src/configuration-resolver.js +65 -0
  482. package/src/configuration-types.js +429 -0
  483. package/src/configuration.js +2590 -0
  484. package/src/controller.js +421 -0
  485. package/src/current-configuration.js +31 -0
  486. package/src/current.js +80 -0
  487. package/src/database/annotations-async-hooks.js +47 -0
  488. package/src/database/annotations.js +40 -0
  489. package/src/database/drivers/base-column.js +182 -0
  490. package/src/database/drivers/base-columns-index.js +81 -0
  491. package/src/database/drivers/base-foreign-key.js +104 -0
  492. package/src/database/drivers/base-table.js +156 -0
  493. package/src/database/drivers/base.js +1609 -0
  494. package/src/database/drivers/mssql/column.js +74 -0
  495. package/src/database/drivers/mssql/columns-index.js +6 -0
  496. package/src/database/drivers/mssql/connect-connection.js +16 -0
  497. package/src/database/drivers/mssql/foreign-key.js +12 -0
  498. package/src/database/drivers/mssql/index.js +590 -0
  499. package/src/database/drivers/mssql/options.js +79 -0
  500. package/src/database/drivers/mssql/query-parser.js +6 -0
  501. package/src/database/drivers/mssql/sql/alter-table.js +4 -0
  502. package/src/database/drivers/mssql/sql/create-database.js +36 -0
  503. package/src/database/drivers/mssql/sql/create-index.js +4 -0
  504. package/src/database/drivers/mssql/sql/create-table.js +4 -0
  505. package/src/database/drivers/mssql/sql/delete.js +19 -0
  506. package/src/database/drivers/mssql/sql/drop-database.js +36 -0
  507. package/src/database/drivers/mssql/sql/drop-table.js +4 -0
  508. package/src/database/drivers/mssql/sql/insert.js +4 -0
  509. package/src/database/drivers/mssql/sql/update.js +31 -0
  510. package/src/database/drivers/mssql/sql/upsert.js +23 -0
  511. package/src/database/drivers/mssql/structure-sql.js +120 -0
  512. package/src/database/drivers/mssql/table.js +145 -0
  513. package/src/database/drivers/mysql/column.js +112 -0
  514. package/src/database/drivers/mysql/columns-index.js +22 -0
  515. package/src/database/drivers/mysql/foreign-key.js +12 -0
  516. package/src/database/drivers/mysql/index.js +473 -0
  517. package/src/database/drivers/mysql/options.js +34 -0
  518. package/src/database/drivers/mysql/query-parser.js +6 -0
  519. package/src/database/drivers/mysql/query.js +37 -0
  520. package/src/database/drivers/mysql/sql/alter-table.js +6 -0
  521. package/src/database/drivers/mysql/sql/create-database.js +39 -0
  522. package/src/database/drivers/mysql/sql/create-index.js +6 -0
  523. package/src/database/drivers/mysql/sql/create-table.js +6 -0
  524. package/src/database/drivers/mysql/sql/delete.js +21 -0
  525. package/src/database/drivers/mysql/sql/drop-database.js +6 -0
  526. package/src/database/drivers/mysql/sql/drop-table.js +6 -0
  527. package/src/database/drivers/mysql/sql/insert.js +6 -0
  528. package/src/database/drivers/mysql/sql/update.js +33 -0
  529. package/src/database/drivers/mysql/sql/upsert.js +13 -0
  530. package/src/database/drivers/mysql/structure-sql.js +93 -0
  531. package/src/database/drivers/mysql/table.js +121 -0
  532. package/src/database/drivers/pgsql/column.js +90 -0
  533. package/src/database/drivers/pgsql/columns-index.js +6 -0
  534. package/src/database/drivers/pgsql/foreign-key.js +12 -0
  535. package/src/database/drivers/pgsql/index.js +441 -0
  536. package/src/database/drivers/pgsql/options.js +32 -0
  537. package/src/database/drivers/pgsql/query-parser.js +6 -0
  538. package/src/database/drivers/pgsql/sql/alter-table.js +6 -0
  539. package/src/database/drivers/pgsql/sql/create-database.js +38 -0
  540. package/src/database/drivers/pgsql/sql/create-index.js +6 -0
  541. package/src/database/drivers/pgsql/sql/create-table.js +6 -0
  542. package/src/database/drivers/pgsql/sql/delete.js +21 -0
  543. package/src/database/drivers/pgsql/sql/drop-database.js +6 -0
  544. package/src/database/drivers/pgsql/sql/drop-table.js +6 -0
  545. package/src/database/drivers/pgsql/sql/insert.js +6 -0
  546. package/src/database/drivers/pgsql/sql/update.js +33 -0
  547. package/src/database/drivers/pgsql/sql/upsert.js +14 -0
  548. package/src/database/drivers/pgsql/structure-sql.js +126 -0
  549. package/src/database/drivers/pgsql/table.js +135 -0
  550. package/src/database/drivers/sqlite/base.js +509 -0
  551. package/src/database/drivers/sqlite/column.js +75 -0
  552. package/src/database/drivers/sqlite/columns-index.js +30 -0
  553. package/src/database/drivers/sqlite/connection-sql-js.js +46 -0
  554. package/src/database/drivers/sqlite/foreign-key.js +24 -0
  555. package/src/database/drivers/sqlite/index.js +394 -0
  556. package/src/database/drivers/sqlite/index.native.js +72 -0
  557. package/src/database/drivers/sqlite/index.web.js +99 -0
  558. package/src/database/drivers/sqlite/options.js +32 -0
  559. package/src/database/drivers/sqlite/query-parser.js +6 -0
  560. package/src/database/drivers/sqlite/query.js +35 -0
  561. package/src/database/drivers/sqlite/query.native.js +35 -0
  562. package/src/database/drivers/sqlite/query.web.js +49 -0
  563. package/src/database/drivers/sqlite/sql/alter-table.js +187 -0
  564. package/src/database/drivers/sqlite/sql/create-index.js +6 -0
  565. package/src/database/drivers/sqlite/sql/create-table.js +6 -0
  566. package/src/database/drivers/sqlite/sql/delete.js +26 -0
  567. package/src/database/drivers/sqlite/sql/drop-table.js +6 -0
  568. package/src/database/drivers/sqlite/sql/insert.js +6 -0
  569. package/src/database/drivers/sqlite/sql/update.js +33 -0
  570. package/src/database/drivers/sqlite/sql/upsert.js +14 -0
  571. package/src/database/drivers/sqlite/structure-sql.js +56 -0
  572. package/src/database/drivers/sqlite/table-rebuilder.js +96 -0
  573. package/src/database/drivers/sqlite/table.js +131 -0
  574. package/src/database/drivers/structure-sql/utils.js +35 -0
  575. package/src/database/handler.js +13 -0
  576. package/src/database/initializer-from-require-context.js +101 -0
  577. package/src/database/migration/index.js +438 -0
  578. package/src/database/migrator/files-finder.js +55 -0
  579. package/src/database/migrator/types.js +31 -0
  580. package/src/database/migrator.js +557 -0
  581. package/src/database/pool/async-tracked-multi-connection.js +1164 -0
  582. package/src/database/pool/base-methods-forward.js +52 -0
  583. package/src/database/pool/base.js +380 -0
  584. package/src/database/pool/single-multi-use.js +118 -0
  585. package/src/database/query/alter-table-base.js +104 -0
  586. package/src/database/query/base.js +49 -0
  587. package/src/database/query/create-database-base.js +42 -0
  588. package/src/database/query/create-index-base.js +117 -0
  589. package/src/database/query/create-table-base.js +205 -0
  590. package/src/database/query/delete-base.js +19 -0
  591. package/src/database/query/drop-database-base.js +38 -0
  592. package/src/database/query/drop-table-base.js +58 -0
  593. package/src/database/query/from-base.js +36 -0
  594. package/src/database/query/from-plain.js +16 -0
  595. package/src/database/query/from-table.js +18 -0
  596. package/src/database/query/index.js +533 -0
  597. package/src/database/query/insert-base.js +172 -0
  598. package/src/database/query/join-base.js +43 -0
  599. package/src/database/query/join-object.js +167 -0
  600. package/src/database/query/join-plain.js +18 -0
  601. package/src/database/query/join-tracker.js +93 -0
  602. package/src/database/query/model-class-query.js +1577 -0
  603. package/src/database/query/order-base.js +33 -0
  604. package/src/database/query/order-column.js +77 -0
  605. package/src/database/query/order-plain.js +28 -0
  606. package/src/database/query/preloader/belongs-to.js +267 -0
  607. package/src/database/query/preloader/ensure-model-class-initialized.js +18 -0
  608. package/src/database/query/preloader/has-many.js +316 -0
  609. package/src/database/query/preloader/has-one.js +123 -0
  610. package/src/database/query/preloader/selection.js +152 -0
  611. package/src/database/query/preloader.js +201 -0
  612. package/src/database/query/query-data.js +305 -0
  613. package/src/database/query/select-base.js +30 -0
  614. package/src/database/query/select-plain.js +18 -0
  615. package/src/database/query/select-table-and-column.js +28 -0
  616. package/src/database/query/update-base.js +41 -0
  617. package/src/database/query/upsert-base.js +103 -0
  618. package/src/database/query/where-base.js +38 -0
  619. package/src/database/query/where-combinator.js +31 -0
  620. package/src/database/query/where-hash.js +77 -0
  621. package/src/database/query/where-model-class-hash.js +505 -0
  622. package/src/database/query/where-not.js +23 -0
  623. package/src/database/query/where-plain.js +20 -0
  624. package/src/database/query/with-count.js +219 -0
  625. package/src/database/query-parser/base-query-parser.js +40 -0
  626. package/src/database/query-parser/from-parser.js +49 -0
  627. package/src/database/query-parser/group-parser.js +55 -0
  628. package/src/database/query-parser/joins-parser.js +37 -0
  629. package/src/database/query-parser/limit-parser.js +77 -0
  630. package/src/database/query-parser/options.js +94 -0
  631. package/src/database/query-parser/order-parser.js +45 -0
  632. package/src/database/query-parser/select-parser.js +67 -0
  633. package/src/database/query-parser/where-parser.js +46 -0
  634. package/src/database/record/acts-as-list.js +374 -0
  635. package/src/database/record/attachments/download.js +49 -0
  636. package/src/database/record/attachments/handle.js +188 -0
  637. package/src/database/record/attachments/normalize-input.js +213 -0
  638. package/src/database/record/attachments/storage-drivers/filesystem.js +114 -0
  639. package/src/database/record/attachments/storage-drivers/native.js +146 -0
  640. package/src/database/record/attachments/storage-drivers/s3.js +245 -0
  641. package/src/database/record/attachments/store.js +591 -0
  642. package/src/database/record/index.js +3970 -0
  643. package/src/database/record/instance-relationships/base.js +289 -0
  644. package/src/database/record/instance-relationships/belongs-to.js +84 -0
  645. package/src/database/record/instance-relationships/has-many.js +284 -0
  646. package/src/database/record/instance-relationships/has-one.js +117 -0
  647. package/src/database/record/record-not-found-error.js +3 -0
  648. package/src/database/record/relationships/base.js +195 -0
  649. package/src/database/record/relationships/belongs-to.js +57 -0
  650. package/src/database/record/relationships/has-many.js +46 -0
  651. package/src/database/record/relationships/has-one.js +46 -0
  652. package/src/database/record/state-machine.js +278 -0
  653. package/src/database/record/user-module.js +43 -0
  654. package/src/database/record/validators/base.js +27 -0
  655. package/src/database/record/validators/format.js +50 -0
  656. package/src/database/record/validators/presence.js +24 -0
  657. package/src/database/record/validators/uniqueness.js +124 -0
  658. package/src/database/table-data/index.js +241 -0
  659. package/src/database/table-data/table-column.js +416 -0
  660. package/src/database/table-data/table-foreign-key.js +69 -0
  661. package/src/database/table-data/table-index.js +46 -0
  662. package/src/database/table-data/table-reference.js +13 -0
  663. package/src/database/use-database.js +48 -0
  664. package/src/environment-handlers/base.js +561 -0
  665. package/src/environment-handlers/browser.js +338 -0
  666. package/src/environment-handlers/node/cli/commands/background-jobs-main.js +21 -0
  667. package/src/environment-handlers/node/cli/commands/background-jobs-runner.js +24 -0
  668. package/src/environment-handlers/node/cli/commands/background-jobs-worker.js +47 -0
  669. package/src/environment-handlers/node/cli/commands/beacon.js +21 -0
  670. package/src/environment-handlers/node/cli/commands/cli-command-context.js +31 -0
  671. package/src/environment-handlers/node/cli/commands/console.js +149 -0
  672. package/src/environment-handlers/node/cli/commands/db/schema/dump.js +43 -0
  673. package/src/environment-handlers/node/cli/commands/db/schema/load.js +69 -0
  674. package/src/environment-handlers/node/cli/commands/db/seed.js +79 -0
  675. package/src/environment-handlers/node/cli/commands/destroy/migration.js +47 -0
  676. package/src/environment-handlers/node/cli/commands/generate/base-models.js +367 -0
  677. package/src/environment-handlers/node/cli/commands/generate/frontend-models.js +872 -0
  678. package/src/environment-handlers/node/cli/commands/generate/migration.js +45 -0
  679. package/src/environment-handlers/node/cli/commands/generate/model.js +45 -0
  680. package/src/environment-handlers/node/cli/commands/init.js +68 -0
  681. package/src/environment-handlers/node/cli/commands/routes.js +63 -0
  682. package/src/environment-handlers/node/cli/commands/run-script.js +85 -0
  683. package/src/environment-handlers/node/cli/commands/runner.js +84 -0
  684. package/src/environment-handlers/node/cli/commands/server.js +151 -0
  685. package/src/environment-handlers/node/cli/commands/test.js +118 -0
  686. package/src/environment-handlers/node.js +887 -0
  687. package/src/error-logger.js +30 -0
  688. package/src/frontend-model-controller.js +3491 -0
  689. package/src/frontend-model-resource/base-resource.js +935 -0
  690. package/src/frontend-models/base.js +4004 -0
  691. package/src/frontend-models/clear-pending-debounced-callback.js +16 -0
  692. package/src/frontend-models/event-hook-models.js +49 -0
  693. package/src/frontend-models/model-registry.js +28 -0
  694. package/src/frontend-models/outgoing-event-buffer.js +51 -0
  695. package/src/frontend-models/preloader.js +169 -0
  696. package/src/frontend-models/query.js +2245 -0
  697. package/src/frontend-models/resource-config-validation.js +56 -0
  698. package/src/frontend-models/resource-definition.js +399 -0
  699. package/src/frontend-models/transport-serialization.js +369 -0
  700. package/src/frontend-models/use-created-event.js +21 -0
  701. package/src/frontend-models/use-destroyed-event.js +148 -0
  702. package/src/frontend-models/use-model-class-event.js +164 -0
  703. package/src/frontend-models/use-updated-event.js +152 -0
  704. package/src/frontend-models/websocket-channel.js +494 -0
  705. package/src/frontend-models/websocket-publishers.js +224 -0
  706. package/src/http-client/header.js +17 -0
  707. package/src/http-client/index.js +139 -0
  708. package/src/http-client/request.js +94 -0
  709. package/src/http-client/response.js +151 -0
  710. package/src/http-client/websocket-client.js +27 -0
  711. package/src/http-server/client/index.js +507 -0
  712. package/src/http-server/client/params-to-object.js +152 -0
  713. package/src/http-server/client/request-buffer/form-data-part.js +139 -0
  714. package/src/http-server/client/request-buffer/header.js +19 -0
  715. package/src/http-server/client/request-buffer/index.js +535 -0
  716. package/src/http-server/client/request-parser.js +195 -0
  717. package/src/http-server/client/request-runner.js +321 -0
  718. package/src/http-server/client/request-timing.js +171 -0
  719. package/src/http-server/client/request.js +114 -0
  720. package/src/http-server/client/response.js +251 -0
  721. package/src/http-server/client/uploaded-file/memory-uploaded-file.js +32 -0
  722. package/src/http-server/client/uploaded-file/temporary-uploaded-file.js +32 -0
  723. package/src/http-server/client/uploaded-file/uploaded-file.js +36 -0
  724. package/src/http-server/client/websocket-request.js +147 -0
  725. package/src/http-server/client/websocket-session.js +1755 -0
  726. package/src/http-server/cookie.js +245 -0
  727. package/src/http-server/development-reloader.js +240 -0
  728. package/src/http-server/index.js +561 -0
  729. package/src/http-server/remote-address.js +77 -0
  730. package/src/http-server/server-client.js +222 -0
  731. package/src/http-server/server-lock.js +178 -0
  732. package/src/http-server/websocket-channel-subscribers.js +110 -0
  733. package/src/http-server/websocket-channel.js +137 -0
  734. package/src/http-server/websocket-connection.js +118 -0
  735. package/src/http-server/websocket-event-log-store.js +433 -0
  736. package/src/http-server/websocket-events-host.js +170 -0
  737. package/src/http-server/websocket-events.js +50 -0
  738. package/src/http-server/worker-handler/channel-subscriber-dispatch.js +28 -0
  739. package/src/http-server/worker-handler/in-process.js +155 -0
  740. package/src/http-server/worker-handler/index.js +370 -0
  741. package/src/http-server/worker-handler/worker-script.js +6 -0
  742. package/src/http-server/worker-handler/worker-thread.js +286 -0
  743. package/src/initializer.js +39 -0
  744. package/src/jobs/.gitkeep +1 -0
  745. package/src/jobs/mail-delivery.js +22 -0
  746. package/src/logger/base-logger.js +34 -0
  747. package/src/logger/console-logger.js +28 -0
  748. package/src/logger/file-logger.js +36 -0
  749. package/src/logger/outputs/array-output.js +50 -0
  750. package/src/logger/outputs/console-output.js +32 -0
  751. package/src/logger/outputs/file-output.js +55 -0
  752. package/src/logger/outputs/stdout-output.js +64 -0
  753. package/src/logger.js +507 -0
  754. package/src/mailer/backends/smtp.js +197 -0
  755. package/src/mailer/base.js +337 -0
  756. package/src/mailer/delivery.js +70 -0
  757. package/src/mailer/index.js +24 -0
  758. package/src/mailer.js +15 -0
  759. package/src/plugins/sqljs-wasm-route-controller.js +70 -0
  760. package/src/plugins/sqljs-wasm-route.js +71 -0
  761. package/src/record-payload-values.js +83 -0
  762. package/src/routes/app-routes.js +17 -0
  763. package/src/routes/base-route.js +133 -0
  764. package/src/routes/basic-route.js +109 -0
  765. package/src/routes/built-in/debug/controller.js +12 -0
  766. package/src/routes/built-in/errors/controller.js +7 -0
  767. package/src/routes/built-in/errors/not-found.ejs +1 -0
  768. package/src/routes/get-route.js +75 -0
  769. package/src/routes/hooks/frontend-model-command-route-hook.js +100 -0
  770. package/src/routes/index.js +50 -0
  771. package/src/routes/namespace-route.js +51 -0
  772. package/src/routes/plugin-routes.js +141 -0
  773. package/src/routes/post-route.js +74 -0
  774. package/src/routes/resolver.js +535 -0
  775. package/src/routes/resource-route.js +154 -0
  776. package/src/routes/root-route.js +11 -0
  777. package/src/templates/configuration.js +61 -0
  778. package/src/templates/generate-migration.js +11 -0
  779. package/src/templates/generate-model.js +6 -0
  780. package/src/templates/routes.js +11 -0
  781. package/src/testing/base-expect.js +17 -0
  782. package/src/testing/browser-frontend-model-event-hook-scenarios.js +520 -0
  783. package/src/testing/browser-test-app.js +32 -0
  784. package/src/testing/expect-to-change.js +55 -0
  785. package/src/testing/expect-utils.js +269 -0
  786. package/src/testing/expect.js +763 -0
  787. package/src/testing/request-client.js +90 -0
  788. package/src/testing/test-files-finder.js +364 -0
  789. package/src/testing/test-filter-parser.js +198 -0
  790. package/src/testing/test-runner.js +1168 -0
  791. package/src/testing/test-suite-splitter.js +177 -0
  792. package/src/testing/test.js +370 -0
  793. package/src/types/external-modules.d.ts +57 -0
  794. package/src/utils/backtrace-cleaner-node.js +87 -0
  795. package/src/utils/backtrace-cleaner.js +266 -0
  796. package/src/utils/ensure-error.js +15 -0
  797. package/src/utils/event-emitter.js +8 -0
  798. package/src/utils/file-exists.js +18 -0
  799. package/src/utils/format-value.js +101 -0
  800. package/src/utils/model-scope.js +56 -0
  801. package/src/utils/nest-callbacks.js +22 -0
  802. package/src/utils/plain-object.js +14 -0
  803. package/src/utils/ransack.js +859 -0
  804. package/src/utils/rest-args-error.js +14 -0
  805. package/src/utils/singularize-model-name.js +18 -0
  806. package/src/utils/split-sql-statements.js +88 -0
  807. package/src/utils/to-import-specifier.js +53 -0
  808. package/src/utils/with-tracked-stack-async-hooks.js +103 -0
  809. package/src/utils/with-tracked-stack.js +38 -0
  810. package/src/velocious-error.js +34 -0
  811. package/tsconfig.json +16 -0
@@ -1,18 +1,21 @@
1
1
  // @ts-check
2
- import { digg } from "diggerize";
3
- import DevelopmentReloader from "./development-reloader.js";
4
- import EventEmitter from "../utils/event-emitter.js";
5
- import InProcessHandler from "./worker-handler/in-process.js";
6
- import Logger from "../logger.js";
7
- import Net from "net";
8
- import ServerClient from "./server-client.js";
9
- import WorkerHandler from "./worker-handler/index.js";
2
+
3
+ import {digg} from "diggerize"
4
+ import DevelopmentReloader from "./development-reloader.js"
5
+ import EventEmitter from "../utils/event-emitter.js"
6
+ import InProcessHandler from "./worker-handler/in-process.js"
7
+ import Logger from "../logger.js"
8
+ import Net from "net"
9
+ import ServerClient from "./server-client.js"
10
+ import WorkerHandler from "./worker-handler/index.js"
11
+
10
12
  /**
11
13
  * Defines this typedef.
12
14
  @typedef {{start: () => Promise<void>, stop: () => Promise<void>}} DevelopmentReloaderLike */
13
15
  /**
14
16
  * Defines this typedef.
15
17
  @typedef {function({configuration: import("../configuration.js").default, workerCount: number}) : (WorkerHandler | InProcessHandler)} WorkerHandlerFactory */
18
+
16
19
  /**
17
20
  * Runs normalize worker count.
18
21
  * @param {object} args - Options object.
@@ -20,468 +23,539 @@ import WorkerHandler from "./worker-handler/index.js";
20
23
  * @param {number} [args.workers] - Configured worker count.
21
24
  * @returns {number} - Normalized worker count.
22
25
  */
23
- function normalizeWorkerCount({ maxWorkers, workers }) {
24
- const workerCount = workers ?? maxWorkers ?? 1;
25
- if (!Number.isInteger(workerCount) || workerCount < 1) {
26
- throw new Error("HTTP server workers must be a positive integer");
27
- }
28
- return workerCount;
26
+ function normalizeWorkerCount({maxWorkers, workers}) {
27
+ const workerCount = workers ?? maxWorkers ?? 1
28
+
29
+ if (!Number.isInteger(workerCount) || workerCount < 1) {
30
+ throw new Error("HTTP server workers must be a positive integer")
31
+ }
32
+
33
+ return workerCount
29
34
  }
35
+
30
36
  export default class VelociousHttpServer {
31
- clientCount = 0;
32
- _starting = false;
33
- /**
34
- * Narrows the runtime value to the documented type.
35
- @type {DevelopmentReloader | DevelopmentReloaderLike | undefined} */
36
- developmentReloader;
37
- /**
38
- * Narrows the runtime value to the documented type.
39
- @type {import("net").Server | undefined} */
40
- netServer;
41
- /**
42
- * Narrows the runtime value to the documented type.
43
- @type {WorkerHandlerFactory | undefined} */
44
- workerHandlerFactory;
45
- /**
46
- * Clients.
47
- @type {Record<string, ServerClient>} */
48
- clients = {};
49
- /**
50
- * Active sockets.
51
- @type {Set<import("net").Socket>} */
52
- _activeSockets = new Set();
53
- events = new EventEmitter();
54
- workerCount = 0;
55
- /**
56
- * Worker handlers.
57
- @type {Array<WorkerHandler | InProcessHandler>} */
58
- workerHandlers = [];
59
- nextWorkerHandlerIndex = 0;
60
- /**
61
- * Sticky worker handlers.
62
- @type {Map<string, WorkerHandler | InProcessHandler>} */
63
- stickyWorkerHandlers = new Map();
64
- /**
65
- * Runs constructor.
66
- * @param {object} args - Options object.
67
- * @param {import("../configuration.js").default} args.configuration - Configuration instance.
68
- * @param {string} [args.host] - Host.
69
- * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.
70
- * @param {number} [args.port] - Port.
71
- * @param {number} [args.maxWorkers] - Max workers.
72
- * @param {number} [args.workers] - Worker handlers to start.
73
- * @param {function({configuration: import("../configuration.js").default, onReload: function({changedPath: string}) : Promise<void>}) : {start: () => Promise<void>, stop: () => Promise<void>}} [args.developmentReloaderFactory] - Development reloader factory.
74
- * @param {WorkerHandlerFactory} [args.workerHandlerFactory] - Worker handler factory.
75
- */
76
- constructor({ configuration, developmentReloaderFactory, host, inProcess, maxWorkers, port, workerHandlerFactory, workers }) {
77
- this.configuration = configuration;
78
- this.developmentReloaderFactory = developmentReloaderFactory;
79
- this.workerHandlerFactory = workerHandlerFactory;
80
- this.inProcess = inProcess || false;
81
- this.logger = new Logger(this);
82
- this.host = host ?? "0.0.0.0";
83
- this.port = port ?? 3006;
84
- this.workers = normalizeWorkerCount({ maxWorkers, workers });
37
+ clientCount = 0
38
+ _starting = false
39
+
40
+ /**
41
+ * Narrows the runtime value to the documented type.
42
+ @type {DevelopmentReloader | DevelopmentReloaderLike | undefined} */
43
+ developmentReloader
44
+
45
+ /**
46
+ * Narrows the runtime value to the documented type.
47
+ @type {import("net").Server | undefined} */
48
+ netServer
49
+
50
+ /**
51
+ * Narrows the runtime value to the documented type.
52
+ @type {WorkerHandlerFactory | undefined} */
53
+ workerHandlerFactory
54
+
55
+ /**
56
+ * Clients.
57
+ @type {Record<string, ServerClient>} */
58
+ clients = {}
59
+
60
+ /**
61
+ * Active sockets.
62
+ @type {Set<import("net").Socket>} */
63
+ _activeSockets = new Set()
64
+
65
+ events = new EventEmitter()
66
+ workerCount = 0
67
+
68
+ /**
69
+ * Worker handlers.
70
+ @type {Array<WorkerHandler | InProcessHandler>} */
71
+ workerHandlers = []
72
+ nextWorkerHandlerIndex = 0
73
+ /**
74
+ * Sticky worker handlers.
75
+ @type {Map<string, WorkerHandler | InProcessHandler>} */
76
+ stickyWorkerHandlers = new Map()
77
+
78
+ /**
79
+ * Runs constructor.
80
+ * @param {object} args - Options object.
81
+ * @param {import("../configuration.js").default} args.configuration - Configuration instance.
82
+ * @param {string} [args.host] - Host.
83
+ * @param {boolean} [args.inProcess] - Run HTTP handlers in the main thread instead of worker threads.
84
+ * @param {number} [args.port] - Port.
85
+ * @param {number} [args.maxWorkers] - Max workers.
86
+ * @param {number} [args.workers] - Worker handlers to start.
87
+ * @param {function({configuration: import("../configuration.js").default, onReload: function({changedPath: string}) : Promise<void>}) : {start: () => Promise<void>, stop: () => Promise<void>}} [args.developmentReloaderFactory] - Development reloader factory.
88
+ * @param {WorkerHandlerFactory} [args.workerHandlerFactory] - Worker handler factory.
89
+ */
90
+ constructor({configuration, developmentReloaderFactory, host, inProcess, maxWorkers, port, workerHandlerFactory, workers}) {
91
+ this.configuration = configuration
92
+ this.developmentReloaderFactory = developmentReloaderFactory
93
+ this.workerHandlerFactory = workerHandlerFactory
94
+ this.inProcess = inProcess || false
95
+ this.logger = new Logger(this)
96
+ this.host = host ?? "0.0.0.0"
97
+ this.port = port ?? 3006
98
+ this.workers = normalizeWorkerCount({maxWorkers, workers})
99
+ }
100
+
101
+ /**
102
+ * Runs start.
103
+ * @returns {Promise<void>} - Resolves when complete.
104
+ */
105
+ async start() {
106
+ if (this._starting) throw new Error("Velocious HTTP server is already starting")
107
+ if (this.isActive()) throw new Error("Velocious HTTP server is already running")
108
+
109
+ this._starting = true
110
+ const startupState = this._captureStartupState()
111
+
112
+ try {
113
+ await this._ensureWorkers()
114
+ await this._startDevelopmentReloader()
115
+ /**
116
+ * Net server.
117
+ @type {import("net").Server} */
118
+ const netServer = new Net.Server()
119
+ this.netServer = netServer
120
+ netServer.on("close", this.onClose)
121
+ netServer.on("connection", this.onConnection)
122
+ netServer.on("error", this.onServerError)
123
+ await this._netServerListen()
124
+ } catch (error) {
125
+ await this._stopStartupResources(startupState)
126
+ throw error
127
+ } finally {
128
+ this._starting = false
85
129
  }
86
- /**
87
- * Runs start.
88
- * @returns {Promise<void>} - Resolves when complete.
89
- */
90
- async start() {
91
- if (this._starting)
92
- throw new Error("Velocious HTTP server is already starting");
93
- if (this.isActive())
94
- throw new Error("Velocious HTTP server is already running");
95
- this._starting = true;
96
- const startupState = this._captureStartupState();
97
- try {
98
- await this._ensureWorkers();
99
- await this._startDevelopmentReloader();
100
- /**
101
- * Net server.
102
- @type {import("net").Server} */
103
- const netServer = new Net.Server();
104
- this.netServer = netServer;
105
- netServer.on("close", this.onClose);
106
- netServer.on("connection", this.onConnection);
107
- netServer.on("error", this.onServerError);
108
- await this._netServerListen();
109
- }
110
- catch (error) {
111
- await this._stopStartupResources(startupState);
112
- throw error;
113
- }
114
- finally {
115
- this._starting = false;
116
- }
117
- }
118
- /**
119
- * Runs capture startup state.
120
- * @returns {{developmentReloader: DevelopmentReloader | DevelopmentReloaderLike | undefined, netServer: import("net").Server | undefined, workerHandlers: Array<WorkerHandler | InProcessHandler>}} - Startup state.
121
- */
122
- _captureStartupState() {
123
- return {
124
- developmentReloader: this.developmentReloader,
125
- netServer: this.netServer,
126
- workerHandlers: [...this.workerHandlers]
127
- };
128
- }
129
- /**
130
- * Runs stop startup resources.
131
- * @param {ReturnType<VelociousHttpServer["_captureStartupState"]>} startupState - State captured before startup.
132
- * @returns {Promise<void>} - Resolves when cleanup is complete.
133
- */
134
- async _stopStartupResources(startupState) {
135
- /**
136
- * Startup net server.
137
- @type {import("net").Server | undefined} */
138
- const startupNetServer = this.netServer;
139
- if (this.developmentReloader && this.developmentReloader !== startupState.developmentReloader) {
140
- await this.developmentReloader.stop();
141
- }
142
- if (startupNetServer && startupNetServer !== startupState.netServer) {
143
- await this.stopServer(startupNetServer);
144
- }
145
- const startupWorkerHandlers = this.workerHandlers.filter((workerHandler) => !startupState.workerHandlers.includes(workerHandler));
146
- await Promise.all(startupWorkerHandlers.map((handler) => handler.stop()));
147
- this.developmentReloader = startupState.developmentReloader;
148
- this.netServer = startupState.netServer;
149
- this.workerHandlers = startupState.workerHandlers;
150
- this.stickyWorkerHandlers.clear();
130
+ }
131
+
132
+ /**
133
+ * Runs capture startup state.
134
+ * @returns {{developmentReloader: DevelopmentReloader | DevelopmentReloaderLike | undefined, netServer: import("net").Server | undefined, workerHandlers: Array<WorkerHandler | InProcessHandler>}} - Startup state.
135
+ */
136
+ _captureStartupState() {
137
+ return {
138
+ developmentReloader: this.developmentReloader,
139
+ netServer: this.netServer,
140
+ workerHandlers: [...this.workerHandlers]
151
141
  }
152
- /**
153
- * Runs net server listen.
154
- * @returns {Promise<void>} - Resolves when complete.
155
- */
156
- _netServerListen() {
157
- return new Promise((resolve, reject) => {
158
- if (!this.netServer)
159
- throw new Error("No netServer");
160
- /**
161
- * On listen error.
162
- * @param {Error} error - Listen error.
163
- */
164
- const onListenError = (error) => {
165
- this.netServer?.off("error", onListenError);
166
- reject(error);
167
- };
168
- try {
169
- this.netServer.once("error", onListenError);
170
- this.netServer.listen(this.port, this.host, () => {
171
- this.netServer?.off("error", onListenError);
172
- this.logger.debug(`Velocious listening on ${this.host}:${this.port}`);
173
- resolve(undefined);
174
- });
175
- }
176
- catch (error) {
177
- reject(error);
178
- }
179
- });
180
- }
181
- /**
182
- * Runs ensure workers.
183
- * @returns {Promise<void>} - Resolves when complete.
184
- */
185
- async _ensureWorkers() {
186
- while (this.workerHandlers.length < this.workers) {
187
- await this.spawnWorker();
188
- }
189
- }
190
- /**
191
- * Runs is active.
192
- * @returns {boolean} - Whether active.
193
- */
194
- isActive() {
195
- if (this.netServer) {
196
- return this.netServer.listening;
197
- }
198
- return false;
142
+ }
143
+
144
+ /**
145
+ * Runs stop startup resources.
146
+ * @param {ReturnType<VelociousHttpServer["_captureStartupState"]>} startupState - State captured before startup.
147
+ * @returns {Promise<void>} - Resolves when cleanup is complete.
148
+ */
149
+ async _stopStartupResources(startupState) {
150
+ /**
151
+ * Startup net server.
152
+ @type {import("net").Server | undefined} */
153
+ const startupNetServer = this.netServer
154
+
155
+ if (this.developmentReloader && this.developmentReloader !== startupState.developmentReloader) {
156
+ await this.developmentReloader.stop()
199
157
  }
200
- /**
201
- * Runs get debug snapshot.
202
- * @returns {Promise<Record<string, ?>>} - HTTP server worker diagnostics.
203
- */
204
- async getDebugSnapshot() {
205
- return {
206
- active: this.isActive(),
207
- activeSocketCount: this._activeSockets.size,
208
- clientCount: Object.keys(this.clients).length,
209
- configuredWorkerCount: this.workers,
210
- inProcess: this.inProcess,
211
- workerCount: this.workerHandlers.length,
212
- workers: await Promise.all(this.workerHandlers.map((handler) => this.workerDebugSnapshot(handler)))
213
- };
158
+
159
+ if (startupNetServer && startupNetServer !== startupState.netServer) {
160
+ await this.stopServer(startupNetServer)
214
161
  }
215
- /**
216
- * Runs worker debug snapshot.
217
- * @param {WorkerHandler | InProcessHandler} workerHandler - Worker handler to inspect.
218
- * @returns {Promise<Record<string, ?>>} Worker debug snapshot.
219
- */
220
- async workerDebugSnapshot(workerHandler) {
221
- if (workerHandler instanceof WorkerHandler)
222
- return await workerHandler.getDebugSnapshot();
223
- if (workerHandler instanceof InProcessHandler)
224
- return this.inProcessWorkerDebugSnapshot(workerHandler);
225
- return { active: false, error: "Unknown worker handler type" };
162
+
163
+ const startupWorkerHandlers = this.workerHandlers.filter((workerHandler) => !startupState.workerHandlers.includes(workerHandler))
164
+
165
+ await Promise.all(startupWorkerHandlers.map((handler) => handler.stop()))
166
+
167
+ this.developmentReloader = startupState.developmentReloader
168
+ this.netServer = startupState.netServer
169
+ this.workerHandlers = startupState.workerHandlers
170
+ this.stickyWorkerHandlers.clear()
171
+ }
172
+
173
+ /**
174
+ * Runs net server listen.
175
+ * @returns {Promise<void>} - Resolves when complete.
176
+ */
177
+ _netServerListen() {
178
+ return new Promise((resolve, reject) => {
179
+ if (!this.netServer) throw new Error("No netServer")
180
+
181
+ /**
182
+ * On listen error.
183
+ * @param {Error} error - Listen error.
184
+ */
185
+ const onListenError = (error) => {
186
+ this.netServer?.off("error", onListenError)
187
+ reject(error)
188
+ }
189
+
190
+ try {
191
+ this.netServer.once("error", onListenError)
192
+ this.netServer.listen(this.port, this.host, () => {
193
+ this.netServer?.off("error", onListenError)
194
+ this.logger.debug(`Velocious listening on ${this.host}:${this.port}`)
195
+ resolve(undefined)
196
+ })
197
+ } catch (error) {
198
+ reject(error)
199
+ }
200
+ })
201
+ }
202
+
203
+ /**
204
+ * Runs ensure workers.
205
+ * @returns {Promise<void>} - Resolves when complete.
206
+ */
207
+ async _ensureWorkers() {
208
+ while (this.workerHandlers.length < this.workers) {
209
+ await this.spawnWorker()
226
210
  }
227
- /**
228
- * Runs in process worker debug snapshot.
229
- * @param {InProcessHandler} workerHandler - In-process worker handler to inspect.
230
- * @returns {Record<string, ?>} Worker debug snapshot.
231
- */
232
- inProcessWorkerDebugSnapshot(workerHandler) {
233
- return {
234
- active: true,
235
- clientCount: Object.keys(workerHandler.clients).length,
236
- snapshot: workerHandler.configuration.getLocalDebugSnapshot(),
237
- workerCount: workerHandler.workerCount
238
- };
211
+ }
212
+
213
+ /**
214
+ * Runs is active.
215
+ * @returns {boolean} - Whether active.
216
+ */
217
+ isActive() {
218
+ if (this.netServer) {
219
+ return this.netServer.listening
239
220
  }
240
- /**
241
- * Runs stop clients.
242
- * @returns {Promise<void>} - Resolves when complete.
243
- */
244
- async stopClients() {
245
- const promises = [];
246
- for (const clientCount in this.clients) {
247
- const client = this.clients[clientCount];
248
- promises.push(client.end());
249
- }
250
- await Promise.all(promises);
221
+
222
+ return false
223
+ }
224
+
225
+ /**
226
+ * Runs get debug snapshot.
227
+ * @returns {Promise<Record<string, ?>>} - HTTP server worker diagnostics.
228
+ */
229
+ async getDebugSnapshot() {
230
+ return {
231
+ active: this.isActive(),
232
+ activeSocketCount: this._activeSockets.size,
233
+ clientCount: Object.keys(this.clients).length,
234
+ configuredWorkerCount: this.workers,
235
+ inProcess: this.inProcess,
236
+ workerCount: this.workerHandlers.length,
237
+ workers: await Promise.all(this.workerHandlers.map((handler) => this.workerDebugSnapshot(handler)))
251
238
  }
252
- /**
253
- * Runs stop server.
254
- * @param {import("net").Server | undefined} [netServer] - Server to stop.
255
- * @returns {Promise<void>} - Resolves when complete.
256
- */
257
- stopServer(netServer = this.netServer) {
258
- return new Promise((resolve, reject) => {
259
- if (!netServer || !netServer.listening) {
260
- resolve(undefined);
261
- return;
262
- }
263
- if (netServer === this.netServer) {
264
- // Force-close lingering sockets (e.g. WebSocket upgrade
265
- // connections mid-close-handshake) so the port is released
266
- // immediately instead of waiting for graceful drain.
267
- for (const socket of this._activeSockets) {
268
- socket.destroy();
269
- }
270
- this._activeSockets.clear();
271
- }
272
- netServer.close((error) => {
273
- if (error) {
274
- reject(error);
275
- }
276
- else {
277
- resolve(undefined);
278
- }
279
- });
280
- });
239
+ }
240
+
241
+ /**
242
+ * Runs worker debug snapshot.
243
+ * @param {WorkerHandler | InProcessHandler} workerHandler - Worker handler to inspect.
244
+ * @returns {Promise<Record<string, ?>>} Worker debug snapshot.
245
+ */
246
+ async workerDebugSnapshot(workerHandler) {
247
+ if (workerHandler instanceof WorkerHandler) return await workerHandler.getDebugSnapshot()
248
+ if (workerHandler instanceof InProcessHandler) return this.inProcessWorkerDebugSnapshot(workerHandler)
249
+
250
+ return {active: false, error: "Unknown worker handler type"}
251
+ }
252
+
253
+ /**
254
+ * Runs in process worker debug snapshot.
255
+ * @param {InProcessHandler} workerHandler - In-process worker handler to inspect.
256
+ * @returns {Record<string, ?>} Worker debug snapshot.
257
+ */
258
+ inProcessWorkerDebugSnapshot(workerHandler) {
259
+ return {
260
+ active: true,
261
+ clientCount: Object.keys(workerHandler.clients).length,
262
+ snapshot: workerHandler.configuration.getLocalDebugSnapshot(),
263
+ workerCount: workerHandler.workerCount
281
264
  }
282
- /**
283
- * Runs stop.
284
- * @returns {Promise<void>} - Resolves when complete.
285
- */
286
- async stop() {
287
- this._stopping = true;
288
- await this.developmentReloader?.stop();
289
- this.developmentReloader = undefined;
290
- await this.stopClients();
291
- await this.stopServer();
292
- const stopTasks = this.workerHandlers.map((handler) => handler.stop());
293
- await Promise.all(stopTasks);
294
- this.workerHandlers = [];
295
- this.stickyWorkerHandlers.clear();
265
+ }
266
+
267
+ /**
268
+ * Runs stop clients.
269
+ * @returns {Promise<void>} - Resolves when complete.
270
+ */
271
+ async stopClients() {
272
+ const promises = []
273
+
274
+ for (const clientCount in this.clients) {
275
+ const client = this.clients[clientCount]
276
+
277
+ promises.push(client.end())
296
278
  }
297
- /**
298
- * On close.
299
- * @returns {void} - No return value.
300
- */
301
- onClose = () => {
302
- this.events.emit("close");
303
- };
304
- /**
305
- * On server error.
306
- * @param {Error} error - Server socket error.
307
- * @returns {void} - No return value.
308
- */
309
- onServerError = (error) => {
310
- this.logger.error(`Velocious HTTP server socket error on ${this.host}:${this.port}`, error);
311
- };
312
- /**
313
- * On connection.
314
- * @param {import("net").Socket} socket - Socket instance.
315
- * @returns {void} - No return value.
316
- */
317
- onConnection = (socket) => {
318
- const clientCount = this.clientCount;
319
- this._activeSockets.add(socket);
320
- socket.once("close", () => this._activeSockets.delete(socket));
321
- this.logger.debug(() => ["New client", {
322
- clientCount,
323
- remoteAddress: socket.remoteAddress,
324
- remoteFamily: socket.remoteFamily,
325
- remotePort: socket.remotePort
326
- }]);
327
- this.clientCount++;
328
- try {
329
- // Paused WebSocket sessions are worker-local, so reconnects from
330
- // the same client address must return to the same worker.
331
- const workerHandler = this.workerHandlerToUse({ stickyKey: socket.remoteAddress });
332
- const client = new ServerClient({
333
- clientCount,
334
- configuration: this.configuration,
335
- socket
336
- });
337
- client.events.on("close", this.onClientClose);
338
- this.logger.debug(`Gave client ${clientCount} to worker ${workerHandler.workerCount}`);
339
- workerHandler.addSocketConnection(client);
340
- this.clients[clientCount] = client;
279
+
280
+ await Promise.all(promises)
281
+ }
282
+
283
+ /**
284
+ * Runs stop server.
285
+ * @param {import("net").Server | undefined} [netServer] - Server to stop.
286
+ * @returns {Promise<void>} - Resolves when complete.
287
+ */
288
+ stopServer(netServer = this.netServer) {
289
+ return new Promise((resolve, reject) => {
290
+ if (!netServer || !netServer.listening) {
291
+ resolve(undefined)
292
+ return
293
+ }
294
+
295
+ if (netServer === this.netServer) {
296
+ // Force-close lingering sockets (e.g. WebSocket upgrade
297
+ // connections mid-close-handshake) so the port is released
298
+ // immediately instead of waiting for graceful drain.
299
+ for (const socket of this._activeSockets) {
300
+ socket.destroy()
341
301
  }
342
- catch (error) {
343
- this.logger.error(`Failed to initialize client ${clientCount} on new connection`, error);
344
- socket.destroy();
302
+
303
+ this._activeSockets.clear()
304
+ }
305
+
306
+ netServer.close((error) => {
307
+ if (error) {
308
+ reject(error)
309
+ } else {
310
+ resolve(undefined)
345
311
  }
346
- };
347
- /**
348
- * On client close.
349
- * @param {ServerClient} client - Client instance.
350
- * @returns {void} - No return value.
351
- */
352
- onClientClose = (client) => {
353
- const clientCount = digg(client, "clientCount");
354
- const oldClientsLength = Object.keys(this.clients).length;
355
- delete this.clients[clientCount];
356
- const newClientsLength = Object.keys(this.clients).length;
357
- if (newClientsLength != (oldClientsLength - 1)) {
358
- this.logger.error(`Expected client to have been removed but length didn't change from ${oldClientsLength} to ${oldClientsLength - 1}`);
359
- }
360
- };
361
- /**
362
- * Runs spawn worker.
363
- * @returns {Promise<void>} - Resolves when complete.
364
- */
365
- async spawnWorker() {
366
- const workerHandler = await this._buildWorkerHandler();
367
- this.workerHandlers.push(workerHandler);
312
+ })
313
+ })
314
+ }
315
+
316
+ /**
317
+ * Runs stop.
318
+ * @returns {Promise<void>} - Resolves when complete.
319
+ */
320
+ async stop() {
321
+ this._stopping = true
322
+ await this.developmentReloader?.stop()
323
+ this.developmentReloader = undefined
324
+ await this.stopClients()
325
+ await this.stopServer()
326
+
327
+ const stopTasks = this.workerHandlers.map((handler) => handler.stop())
328
+ await Promise.all(stopTasks)
329
+ this.workerHandlers = []
330
+ this.stickyWorkerHandlers.clear()
331
+ }
332
+
333
+ /**
334
+ * On close.
335
+ * @returns {void} - No return value.
336
+ */
337
+ onClose = () => {
338
+ this.events.emit("close")
339
+ }
340
+
341
+ /**
342
+ * On server error.
343
+ * @param {Error} error - Server socket error.
344
+ * @returns {void} - No return value.
345
+ */
346
+ onServerError = (error) => {
347
+ this.logger.error(`Velocious HTTP server socket error on ${this.host}:${this.port}`, error)
348
+ }
349
+
350
+ /**
351
+ * On connection.
352
+ * @param {import("net").Socket} socket - Socket instance.
353
+ * @returns {void} - No return value.
354
+ */
355
+ onConnection = (socket) => {
356
+ const clientCount = this.clientCount
357
+
358
+ this._activeSockets.add(socket)
359
+ socket.once("close", () => this._activeSockets.delete(socket))
360
+
361
+ this.logger.debug(() => ["New client", {
362
+ clientCount,
363
+ remoteAddress: socket.remoteAddress,
364
+ remoteFamily: socket.remoteFamily,
365
+ remotePort: socket.remotePort
366
+ }])
367
+ this.clientCount++
368
+
369
+ try {
370
+ // Paused WebSocket sessions are worker-local, so reconnects from
371
+ // the same client address must return to the same worker.
372
+ const workerHandler = this.workerHandlerToUse({stickyKey: socket.remoteAddress})
373
+ const client = new ServerClient({
374
+ clientCount,
375
+ configuration: this.configuration,
376
+ socket
377
+ })
378
+
379
+ client.events.on("close", this.onClientClose)
380
+
381
+ this.logger.debug(`Gave client ${clientCount} to worker ${workerHandler.workerCount}`)
382
+ workerHandler.addSocketConnection(client)
383
+ this.clients[clientCount] = client
384
+ } catch (error) {
385
+ this.logger.error(`Failed to initialize client ${clientCount} on new connection`, error)
386
+ socket.destroy()
368
387
  }
369
- /**
370
- * Runs build worker handlers.
371
- * @returns {Promise<Array<WorkerHandler | InProcessHandler>>} - Started worker handlers.
372
- */
373
- async _buildWorkerHandlers() {
374
- /**
375
- * Worker handlers.
376
- @type {Array<WorkerHandler | InProcessHandler>} */
377
- const workerHandlers = [];
378
- for (let index = 0; index < this.workers; index += 1) {
379
- workerHandlers.push(await this._buildWorkerHandler());
380
- }
381
- return workerHandlers;
382
- }
383
- /**
384
- * Runs build worker handler.
385
- * @returns {Promise<WorkerHandler | InProcessHandler>} - Started worker handler.
386
- */
387
- async _buildWorkerHandler() {
388
- const workerCount = this.workerCount;
389
- this.workerCount++;
390
- const Handler = this.inProcess ? InProcessHandler : WorkerHandler;
391
- const workerHandler = this.workerHandlerFactory
392
- ? this.workerHandlerFactory({ configuration: this.configuration, workerCount })
393
- : new Handler({
394
- configuration: this.configuration,
395
- workerCount
396
- });
397
- await workerHandler.start();
398
- return workerHandler;
388
+ }
389
+
390
+ /**
391
+ * On client close.
392
+ * @param {ServerClient} client - Client instance.
393
+ * @returns {void} - No return value.
394
+ */
395
+ onClientClose = (client) => {
396
+ const clientCount = digg(client, "clientCount")
397
+ const oldClientsLength = Object.keys(this.clients).length
398
+
399
+ delete this.clients[clientCount]
400
+
401
+ const newClientsLength = Object.keys(this.clients).length
402
+
403
+ if (newClientsLength != (oldClientsLength - 1)) {
404
+ this.logger.error(`Expected client to have been removed but length didn't change from ${oldClientsLength} to ${oldClientsLength - 1}`)
399
405
  }
406
+ }
407
+
408
+ /**
409
+ * Runs spawn worker.
410
+ * @returns {Promise<void>} - Resolves when complete.
411
+ */
412
+ async spawnWorker() {
413
+ const workerHandler = await this._buildWorkerHandler()
414
+
415
+ this.workerHandlers.push(workerHandler)
416
+ }
417
+
418
+ /**
419
+ * Runs build worker handlers.
420
+ * @returns {Promise<Array<WorkerHandler | InProcessHandler>>} - Started worker handlers.
421
+ */
422
+ async _buildWorkerHandlers() {
400
423
  /**
401
- * Runs worker handler to use.
402
- * @param {object} [args] - Options object.
403
- * @param {string} [args.stickyKey] - Stable key that must keep routing to the same worker.
404
- * @returns {WorkerHandler | InProcessHandler} - The worker handler to use.
405
- */
406
- workerHandlerToUse({ stickyKey } = {}) {
407
- if (stickyKey) {
408
- const stickyWorkerHandler = this.stickyWorkerHandlers.get(stickyKey);
409
- if (stickyWorkerHandler && this.workerHandlers.includes(stickyWorkerHandler)) {
410
- return stickyWorkerHandler;
411
- }
412
- const workerHandler = this._nextRoundRobinWorkerHandler();
413
- this.stickyWorkerHandlers.set(stickyKey, workerHandler);
414
- return workerHandler;
415
- }
416
- return this._nextRoundRobinWorkerHandler();
424
+ * Worker handlers.
425
+ @type {Array<WorkerHandler | InProcessHandler>} */
426
+ const workerHandlers = []
427
+
428
+ for (let index = 0; index < this.workers; index += 1) {
429
+ workerHandlers.push(await this._buildWorkerHandler())
417
430
  }
418
- /**
419
- * Runs next round robin worker handler.
420
- * @returns {WorkerHandler | InProcessHandler} - The next round-robin worker handler.
421
- */
422
- _nextRoundRobinWorkerHandler() {
423
- this.logger.debug(`Worker handlers length: ${this.workerHandlers.length}`);
424
- const workerHandlerIndex = this.nextWorkerHandlerIndex % this.workerHandlers.length;
425
- const workerHandler = this.workerHandlers[workerHandlerIndex];
426
- if (!workerHandler) {
427
- throw new Error(`No workerHandler by that number: ${workerHandlerIndex}`);
428
- }
429
- this.nextWorkerHandlerIndex += 1;
430
- return workerHandler;
431
+
432
+ return workerHandlers
433
+ }
434
+
435
+ /**
436
+ * Runs build worker handler.
437
+ * @returns {Promise<WorkerHandler | InProcessHandler>} - Started worker handler.
438
+ */
439
+ async _buildWorkerHandler() {
440
+ const workerCount = this.workerCount
441
+
442
+ this.workerCount++
443
+
444
+ const Handler = this.inProcess ? InProcessHandler : WorkerHandler
445
+ const workerHandler = this.workerHandlerFactory
446
+ ? this.workerHandlerFactory({configuration: this.configuration, workerCount})
447
+ : new Handler({
448
+ configuration: this.configuration,
449
+ workerCount
450
+ })
451
+
452
+ await workerHandler.start()
453
+
454
+ return workerHandler
455
+ }
456
+
457
+ /**
458
+ * Runs worker handler to use.
459
+ * @param {object} [args] - Options object.
460
+ * @param {string} [args.stickyKey] - Stable key that must keep routing to the same worker.
461
+ * @returns {WorkerHandler | InProcessHandler} - The worker handler to use.
462
+ */
463
+ workerHandlerToUse({stickyKey} = {}) {
464
+ if (stickyKey) {
465
+ const stickyWorkerHandler = this.stickyWorkerHandlers.get(stickyKey)
466
+
467
+ if (stickyWorkerHandler && this.workerHandlers.includes(stickyWorkerHandler)) {
468
+ return stickyWorkerHandler
469
+ }
470
+
471
+ const workerHandler = this._nextRoundRobinWorkerHandler()
472
+
473
+ this.stickyWorkerHandlers.set(stickyKey, workerHandler)
474
+
475
+ return workerHandler
431
476
  }
432
- /**
433
- * Runs should use development hot reload.
434
- * @returns {boolean} - Whether development worker hot reload should run.
435
- */
436
- shouldUseDevelopmentHotReload() {
437
- return !this.inProcess && this.configuration.getEnvironment() === "development";
477
+
478
+ return this._nextRoundRobinWorkerHandler()
479
+ }
480
+
481
+ /**
482
+ * Runs next round robin worker handler.
483
+ * @returns {WorkerHandler | InProcessHandler} - The next round-robin worker handler.
484
+ */
485
+ _nextRoundRobinWorkerHandler() {
486
+ this.logger.debug(`Worker handlers length: ${this.workerHandlers.length}`)
487
+
488
+ const workerHandlerIndex = this.nextWorkerHandlerIndex % this.workerHandlers.length
489
+ const workerHandler = this.workerHandlers[workerHandlerIndex]
490
+
491
+ if (!workerHandler) {
492
+ throw new Error(`No workerHandler by that number: ${workerHandlerIndex}`)
438
493
  }
439
- /**
440
- * Runs start development reloader.
441
- * @returns {Promise<void>} - Resolves when watcher setup finishes.
442
- */
443
- async _startDevelopmentReloader() {
444
- if (!this.shouldUseDevelopmentHotReload())
445
- return;
446
- if (this.developmentReloader)
447
- return;
448
- const createDevelopmentReloader = this.developmentReloaderFactory
449
- || ((args) => new DevelopmentReloader(args));
450
- this.developmentReloader = createDevelopmentReloader({
451
- configuration: this.configuration,
452
- onReload: async ({ changedPath }) => {
453
- await this.logger.info(`Development hot reload detected change in ${changedPath}`);
454
- await this.reloadWorkersForDevelopment();
455
- }
456
- });
457
- await this.developmentReloader.start();
494
+
495
+ this.nextWorkerHandlerIndex += 1
496
+
497
+ return workerHandler
498
+ }
499
+
500
+ /**
501
+ * Runs should use development hot reload.
502
+ * @returns {boolean} - Whether development worker hot reload should run.
503
+ */
504
+ shouldUseDevelopmentHotReload() {
505
+ return !this.inProcess && this.configuration.getEnvironment() === "development"
506
+ }
507
+
508
+ /**
509
+ * Runs start development reloader.
510
+ * @returns {Promise<void>} - Resolves when watcher setup finishes.
511
+ */
512
+ async _startDevelopmentReloader() {
513
+ if (!this.shouldUseDevelopmentHotReload()) return
514
+ if (this.developmentReloader) return
515
+
516
+ const createDevelopmentReloader = this.developmentReloaderFactory
517
+ || ((args) => new DevelopmentReloader(args))
518
+
519
+ this.developmentReloader = createDevelopmentReloader({
520
+ configuration: this.configuration,
521
+ onReload: async ({changedPath}) => {
522
+ await this.logger.info(`Development hot reload detected change in ${changedPath}`)
523
+ await this.reloadWorkersForDevelopment()
524
+ }
525
+ })
526
+
527
+ await this.developmentReloader.start()
528
+ }
529
+
530
+ /**
531
+ * Runs reload workers for development.
532
+ * @returns {Promise<void>} - Resolves when workers have been refreshed.
533
+ */
534
+ async reloadWorkersForDevelopment() {
535
+ if (this._stopping) return
536
+
537
+ if (this._reloadingWorkersForDevelopment) {
538
+ this._reloadWorkersForDevelopmentQueued = true
539
+ return
458
540
  }
459
- /**
460
- * Runs reload workers for development.
461
- * @returns {Promise<void>} - Resolves when workers have been refreshed.
462
- */
463
- async reloadWorkersForDevelopment() {
464
- if (this._stopping)
465
- return;
466
- if (this._reloadingWorkersForDevelopment) {
467
- this._reloadWorkersForDevelopmentQueued = true;
468
- return;
469
- }
470
- this._reloadingWorkersForDevelopment = true;
471
- try {
472
- do {
473
- this._reloadWorkersForDevelopmentQueued = false;
474
- const oldWorkerHandlers = [...this.workerHandlers];
475
- const newWorkerHandlers = await this._buildWorkerHandlers();
476
- this.workerHandlers = newWorkerHandlers;
477
- this.nextWorkerHandlerIndex = 0;
478
- this.stickyWorkerHandlers.clear();
479
- await Promise.all(oldWorkerHandlers.map((workerHandler) => workerHandler.stop()));
480
- } while (this._reloadWorkersForDevelopmentQueued && !this._stopping);
481
- }
482
- finally {
483
- this._reloadingWorkersForDevelopment = false;
484
- }
541
+
542
+ this._reloadingWorkersForDevelopment = true
543
+
544
+ try {
545
+ do {
546
+ this._reloadWorkersForDevelopmentQueued = false
547
+
548
+ const oldWorkerHandlers = [...this.workerHandlers]
549
+ const newWorkerHandlers = await this._buildWorkerHandlers()
550
+
551
+ this.workerHandlers = newWorkerHandlers
552
+ this.nextWorkerHandlerIndex = 0
553
+ this.stickyWorkerHandlers.clear()
554
+
555
+ await Promise.all(oldWorkerHandlers.map((workerHandler) => workerHandler.stop()))
556
+ } while (this._reloadWorkersForDevelopmentQueued && !this._stopping)
557
+ } finally {
558
+ this._reloadingWorkersForDevelopment = false
485
559
  }
560
+ }
486
561
  }
487
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaHR0cC1zZXJ2ZXIvaW5kZXguanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsWUFBWTtBQUVaLE9BQU8sRUFBQyxJQUFJLEVBQUMsTUFBTSxXQUFXLENBQUE7QUFDOUIsT0FBTyxtQkFBbUIsTUFBTSwyQkFBMkIsQ0FBQTtBQUMzRCxPQUFPLFlBQVksTUFBTSwyQkFBMkIsQ0FBQTtBQUNwRCxPQUFPLGdCQUFnQixNQUFNLGdDQUFnQyxDQUFBO0FBQzdELE9BQU8sTUFBTSxNQUFNLGNBQWMsQ0FBQTtBQUNqQyxPQUFPLEdBQUcsTUFBTSxLQUFLLENBQUE7QUFDckIsT0FBTyxZQUFZLE1BQU0sb0JBQW9CLENBQUE7QUFDN0MsT0FBTyxhQUFhLE1BQU0sMkJBQTJCLENBQUE7QUFFckQ7OytGQUUrRjtBQUMvRjs7K0pBRStKO0FBRS9KOzs7Ozs7R0FNRztBQUNILFNBQVMsb0JBQW9CLENBQUMsRUFBQyxVQUFVLEVBQUUsT0FBTyxFQUFDO0lBQ2pELE1BQU0sV0FBVyxHQUFHLE9BQU8sSUFBSSxVQUFVLElBQUksQ0FBQyxDQUFBO0lBRTlDLElBQUksQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLFdBQVcsQ0FBQyxJQUFJLFdBQVcsR0FBRyxDQUFDLEVBQUUsQ0FBQztRQUN0RCxNQUFNLElBQUksS0FBSyxDQUFDLGdEQUFnRCxDQUFDLENBQUE7SUFDbkUsQ0FBQztJQUVELE9BQU8sV0FBVyxDQUFBO0FBQ3BCLENBQUM7QUFFRCxNQUFNLENBQUMsT0FBTyxPQUFPLG1CQUFtQjtJQUN0QyxXQUFXLEdBQUcsQ0FBQyxDQUFBO0lBQ2YsU0FBUyxHQUFHLEtBQUssQ0FBQTtJQUVqQjs7MEVBRXNFO0lBQ3RFLG1CQUFtQixDQUFBO0lBRW5COztpREFFNkM7SUFDN0MsU0FBUyxDQUFBO0lBRVQ7O2lEQUU2QztJQUM3QyxvQkFBb0IsQ0FBQTtJQUVwQjs7OENBRTBDO0lBQzFDLE9B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QUFvQixDQUFDLEtBQUssRUFBRSxDQUFBO2dCQUVqQyxNQUFNLE9BQU8sQ0FBQyxHQUFHLENBQUMsaUJBQWlCLENBQUMsR0FBRyxDQUFDLENBQUMsYUFBYSxFQUFFLEVBQUUsQ0FBQyxhQUFhLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyxDQUFBO1lBQ25GLENBQUMsUUFBUSxJQUFJLENBQUMsa0NBQWtDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFDO1FBQ3RFLENBQUM7Z0JBQVMsQ0FBQztZQUNULElBQUksQ0FBQywrQkFBK0IsR0FBRyxLQUFLLENBQUE7UUFDOUMsQ0FBQztJQUNILENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQge2RpZ2d9IGZyb20gXCJkaWdnZXJpemVcIlxuaW1wb3J0IERldmVsb3BtZW50UmVsb2FkZXIgZnJvbSBcIi4vZGV2ZWxvcG1lbnQtcmVsb2FkZXIuanNcIlxuaW1wb3J0IEV2ZW50RW1pdHRlciBmcm9tIFwiLi4vdXRpbHMvZXZlbnQtZW1pdHRlci5qc1wiXG5pbXBvcnQgSW5Qcm9jZXNzSGFuZGxlciBmcm9tIFwiLi93b3JrZXItaGFuZGxlci9pbi1wcm9jZXNzLmpzXCJcbmltcG9ydCBMb2dnZXIgZnJvbSBcIi4uL2xvZ2dlci5qc1wiXG5pbXBvcnQgTmV0IGZyb20gXCJuZXRcIlxuaW1wb3J0IFNlcnZlckNsaWVudCBmcm9tIFwiLi9zZXJ2ZXItY2xpZW50LmpzXCJcbmltcG9ydCBXb3JrZXJIYW5kbGVyIGZyb20gXCIuL3dvcmtlci1oYW5kbGVyL2luZGV4LmpzXCJcblxuLyoqXG4gKiBEZWZpbmVzIHRoaXMgdHlwZWRlZi5cbiAgQHR5cGVkZWYge3tzdGFydDogKCkgPT4gUHJvbWlzZTx2b2lkPiwgc3RvcDogKCkgPT4gUHJvbWlzZTx2b2lkPn19IERldmVsb3BtZW50UmVsb2FkZXJMaWtlICovXG4vKipcbiAqIERlZmluZXMgdGhpcyB0eXBlZGVmLlxuICBAdHlwZWRlZiB7ZnVuY3Rpb24oe2NvbmZpZ3VyYXRpb246IGltcG9ydChcIi4uL2NvbmZpZ3VyYXRpb24uanNcIikuZGVmYXVsdCwgd29ya2VyQ291bnQ6IG51bWJlcn0pIDogKFdvcmtlckhhbmRsZXIgfCBJblByb2Nlc3NIYW5kbGVyKX0gV29ya2VySGFuZGxlckZhY3RvcnkgKi9cblxuLyoqXG4gKiBSdW5zIG5vcm1hbGl6ZSB3b3JrZXIgY291bnQuXG4gKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICogQHBhcmFtIHtudW1iZXJ9IFthcmdzLm1heFdvcmtlcnNdIC0gQmFja3dhcmQtY29tcGF0aWJsZSB3b3JrZXIgY291bnQgYWxpYXMuXG4gKiBAcGFyYW0ge251bWJlcn0gW2FyZ3Mud29ya2Vyc10gLSBDb25maWd1cmVkIHdvcmtlciBjb3VudC5cbiAqIEByZXR1cm5zIHtudW1iZXJ9IC0gTm9ybWFsaXplZCB3b3JrZXIgY291bnQuXG4gKi9cbmZ1bmN0aW9uIG5vcm1hbGl6ZVdvcmtlckNvdW50KHttYXhXb3JrZXJzLCB3b3JrZXJzfSkge1xuICBjb25zdCB3b3JrZXJDb3VudCA9IHdvcmtlcnMgPz8gbWF4V29ya2VycyA/PyAxXG5cbiAgaWYgKCFOdW1iZXIuaXNJbnRlZ2VyKHdvcmtlckNvdW50KSB8fCB3b3JrZXJDb3VudCA8IDEpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXCJIVFRQIHNlcnZlciB3b3JrZXJzIG11c3QgYmUgYSBwb3NpdGl2ZSBpbnRlZ2VyXCIpXG4gIH1cblxuICByZXR1cm4gd29ya2VyQ291bnRcbn1cblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgVmVsb2Npb3VzSHR0cFNlcnZlciB7XG4gIGNsaWVudENvdW50ID0gMFxuICBfc3RhcnRpbmcgPSBmYWxzZVxuXG4gIC8qKlxuICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgQHR5cGUge0RldmVsb3BtZW50UmVsb2FkZXIgfCBEZXZlbG9wbWVudFJlbG9hZGVyTGlrZSB8IHVuZGVmaW5lZH0gKi9cbiAgZGV2ZWxvcG1lbnRSZWxvYWRlclxuXG4gIC8qKlxuICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgQHR5cGUge2ltcG9ydChcIm5ldFwiKS5TZXJ2ZXIgfCB1bmRlZmluZWR9ICovXG4gIG5ldFNlcnZlclxuXG4gIC8qKlxuICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgQHR5cGUge1dvcmtlckhhbmRsZXJGYWN0b3J5IHwgdW5kZWZpbmVkfSAqL1xuICB3b3JrZXJIYW5kbGVyRmFjdG9yeVxuXG4gIC8qKlxuICAgKiBDbGllbnRzLlxuICAgIEB0eXBlIHtSZWNvcmQ8c3RyaW5nLCBTZXJ2ZXJDbGllbnQ+fSAgKi9cbiAgY2xpZW50cyA9IHt9XG5cbiAgLyoqXG4gICAqIEFjdGl2ZSBzb2NrZXRzLlxuICAgIEB0eXBlIHtTZXQ8aW1wb3J0KFwibmV0XCIpLlNvY2tldD59ICovXG4gIF9hY3RpdmVTb2NrZXRzID0gbmV3IFNldCgpXG5cbiAgZXZlbnRzID0gbmV3IEV2ZW50RW1pdHRlcigpXG4gIHdvcmtlckNvdW50ID0gMFxuXG4gIC8qKlxuICAgKiBXb3JrZXIgaGFuZGxlcnMuXG4gICAgQHR5cGUge0FycmF5PFdvcmtlckhhbmRsZXIgfCBJblByb2Nlc3NIYW5kbGVyPn0gKi9cbiAgd29ya2VySGFuZGxlcnMgPSBbXVxuICBuZXh0V29ya2VySGFuZGxlckluZGV4ID0gMFxuICAvKipcbiAgICogU3RpY2t5IHdvcmtlciBoYW5kbGVycy5cbiAgICBAdHlwZSB7TWFwPHN0cmluZywgV29ya2VySGFuZGxlciB8IEluUHJvY2Vzc0hhbmRsZXI+fSAqL1xuICBzdGlja3lXb3JrZXJIYW5kbGVycyA9IG5ldyBNYXAoKVxuXG4gIC8qKlxuICAgKiBSdW5zIGNvbnN0cnVjdG9yLlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2NvbmZpZ3VyYXRpb24uanNcIikuZGVmYXVsdH0gYXJncy5jb25maWd1cmF0aW9uIC0gQ29uZmlndXJhdGlvbiBpbnN0YW5jZS5cbiAgICogQHBhcmFtIHtzdHJpbmd9IFthcmdzLmhvc3RdIC0gSG9zdC5cbiAgICogQHBhcmFtIHtib29sZWFufSBbYXJncy5pblByb2Nlc3NdIC0gUnVuIEhUVFAgaGFuZGxlcnMgaW4gdGhlIG1haW4gdGhyZWFkIGluc3RlYWQgb2Ygd29ya2VyIHRocmVhZHMuXG4gICAqIEBwYXJhbSB7bnVtYmVyfSBbYXJncy5wb3J0XSAtIFBvcnQuXG4gICAqIEBwYXJhbSB7bnVtYmVyfSBbYXJncy5tYXhXb3JrZXJzXSAtIE1heCB3b3JrZXJzLlxuICAgKiBAcGFyYW0ge251bWJlcn0gW2FyZ3Mud29ya2Vyc10gLSBXb3JrZXIgaGFuZGxlcnMgdG8gc3RhcnQuXG4gICAqIEBwYXJhbSB7ZnVuY3Rpb24oe2NvbmZpZ3VyYXRpb246IGltcG9ydChcIi4uL2NvbmZpZ3VyYXRpb24uanNcIikuZGVmYXVsdCwgb25SZWxvYWQ6IGZ1bmN0aW9uKHtjaGFuZ2VkUGF0aDogc3RyaW5nfSkgOiBQcm9taXNlPHZvaWQ+fSkgOiB7c3RhcnQ6ICgpID0+IFByb21pc2U8dm9pZD4sIHN0b3A6ICgpID0+IFByb21pc2U8dm9pZD59fSBbYXJncy5kZXZlbG9wbWVudFJlbG9hZGVyRmFjdG9yeV0gLSBEZXZlbG9wbWVudCByZWxvYWRlciBmYWN0b3J5LlxuICAgKiBAcGFyYW0ge1dvcmtlckhhbmRsZXJGYWN0b3J5fSBbYXJncy53b3JrZXJIYW5kbGVyRmFjdG9yeV0gLSBXb3JrZXIgaGFuZGxlciBmYWN0b3J5LlxuICAgKi9cbiAgY29uc3RydWN0b3Ioe2NvbmZpZ3VyYXRpb24sIGRldmVsb3BtZW50UmVsb2FkZXJGYWN0b3J5LCBob3N0LCBpblByb2Nlc3MsIG1heFdvcmtlcnMsIHBvcnQsIHdvcmtlckhhbmRsZXJGYWN0b3J5LCB3b3JrZXJzfSkge1xuICAgIHRoaXMuY29uZmlndXJhdGlvbiA9IGNvbmZpZ3VyYXRpb25cbiAgICB0aGlzLmRldmVsb3BtZW50UmVsb2FkZXJGYWN0b3J5ID0gZGV2ZWxvcG1lbnRSZWxvYWRlckZhY3RvcnlcbiAgICB0aGlzLndvcmtlckhhbmRsZXJGYWN0b3J5ID0gd29ya2VySGFuZGxlckZhY3RvcnlcbiAgICB0aGlzLmluUHJvY2VzcyA9IGluUHJvY2VzcyB8fCBmYWxzZVxuICAgIHRoaXMubG9nZ2VyID0gbmV3IExvZ2dlcih0aGlzKVxuICAgIHRoaXMuaG9zdCA9IGhvc3QgPz8gXCIwLjAuMC4wXCJcbiAgICB0aGlzLnBvcnQgPSBwb3J0ID8/IDMwMDZcbiAgICB0aGlzLndvcmtlcnMgPSBub3JtYWxpemVXb3JrZXJDb3VudCh7bWF4V29ya2Vycywgd29ya2Vyc30pXG4gIH1cblxuICAvKipcbiAgICogUnVucyBzdGFydC5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiBjb21wbGV0ZS5cbiAgICovXG4gIGFzeW5jIHN0YXJ0KCkge1xuICAgIGlmICh0aGlzLl9zdGFydGluZykgdGhyb3cgbmV3IEVycm9yKFwiVmVsb2Npb3VzIEhUVFAgc2VydmVyIGlzIGFscmVhZHkgc3RhcnRpbmdcIilcbiAgICBpZiAodGhpcy5pc0FjdGl2ZSgpKSB0aHJvdyBuZXcgRXJyb3IoXCJWZWxvY2lvdXMgSFRUUCBzZXJ2ZXIgaXMgYWxyZWFkeSBydW5uaW5nXCIpXG5cbiAgICB0aGlzLl9zdGFydGluZyA9IHRydWVcbiAgICBjb25zdCBzdGFydHVwU3RhdGUgPSB0aGlzLl9jYXB0dXJlU3RhcnR1cFN0YXRlKClcblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLl9lbnN1cmVXb3JrZXJzKClcbiAgICAgIGF3YWl0IHRoaXMuX3N0YXJ0RGV2ZWxvcG1lbnRSZWxvYWRlcigpXG4gICAgICAvKipcbiAgICAgICAqIE5ldCBzZXJ2ZXIuXG4gICAgICAgIEB0eXBlIHtpbXBvcnQoXCJuZXRcIikuU2VydmVyfSAqL1xuICAgICAgY29uc3QgbmV0U2VydmVyID0gbmV3IE5ldC5TZXJ2ZXIoKVxuICAgICAgdGhpcy5uZXRTZXJ2ZXIgPSBuZXRTZXJ2ZXJcbiAgICAgIG5ldFNlcnZlci5vbihcImNsb3NlXCIsIHRoaXMub25DbG9zZSlcbiAgICAgIG5ldFNlcnZlci5vbihcImNvbm5lY3Rpb25cIiwgdGhpcy5vbkNvbm5lY3Rpb24pXG4gICAgICBuZXRTZXJ2ZXIub24oXCJlcnJvclwiLCB0aGlzLm9uU2VydmVyRXJyb3IpXG4gICAgICBhd2FpdCB0aGlzLl9uZXRTZXJ2ZXJMaXN0ZW4oKVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBhd2FpdCB0aGlzLl9zdG9wU3RhcnR1cFJlc291cmNlcyhzdGFydHVwU3RhdGUpXG4gICAgICB0aHJvdyBlcnJvclxuICAgIH0gZmluYWxseSB7XG4gICAgICB0aGlzLl9zdGFydGluZyA9IGZhbHNlXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY2FwdHVyZSBzdGFydHVwIHN0YXRlLlxuICAgKiBAcmV0dXJucyB7e2RldmVsb3BtZW50UmVsb2FkZXI6IERldmVsb3BtZW50UmVsb2FkZXIgfCBEZXZlbG9wbWVudFJlbG9hZGVyTGlrZSB8IHVuZGVmaW5lZCwgbmV0U2VydmVyOiBpbXBvcnQoXCJuZXRcIikuU2VydmVyIHwgdW5kZWZpbmVkLCB3b3JrZXJIYW5kbGVyczogQXJyYXk8V29ya2VySGFuZGxlciB8IEluUHJvY2Vzc0hhbmRsZXI+fX0gLSBTdGFydHVwIHN0YXRlLlxuICAgKi9cbiAgX2NhcHR1cmVTdGFydHVwU3RhdGUoKSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGRldmVsb3BtZW50UmVsb2FkZXI6IHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlcixcbiAgICAgIG5ldFNlcnZlcjogdGhpcy5uZXRTZXJ2ZXIsXG4gICAgICB3b3JrZXJIYW5kbGVyczogWy4uLnRoaXMud29ya2VySGFuZGxlcnNdXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgc3RvcCBzdGFydHVwIHJlc291cmNlcy5cbiAgICogQHBhcmFtIHtSZXR1cm5UeXBlPFZlbG9jaW91c0h0dHBTZXJ2ZXJbXCJfY2FwdHVyZVN0YXJ0dXBTdGF0ZVwiXT59IHN0YXJ0dXBTdGF0ZSAtIFN0YXRlIGNhcHR1cmVkIGJlZm9yZSBzdGFydHVwLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNsZWFudXAgaXMgY29tcGxldGUuXG4gICAqL1xuICBhc3luYyBfc3RvcFN0YXJ0dXBSZXNvdXJjZXMoc3RhcnR1cFN0YXRlKSB7XG4gICAgLyoqXG4gICAgICogU3RhcnR1cCBuZXQgc2VydmVyLlxuICAgICAgQHR5cGUge2ltcG9ydChcIm5ldFwiKS5TZXJ2ZXIgfCB1bmRlZmluZWR9ICovXG4gICAgY29uc3Qgc3RhcnR1cE5ldFNlcnZlciA9IHRoaXMubmV0U2VydmVyXG5cbiAgICBpZiAodGhpcy5kZXZlbG9wbWVudFJlbG9hZGVyICYmIHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlciAhPT0gc3RhcnR1cFN0YXRlLmRldmVsb3BtZW50UmVsb2FkZXIpIHtcbiAgICAgIGF3YWl0IHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlci5zdG9wKClcbiAgICB9XG5cbiAgICBpZiAoc3RhcnR1cE5ldFNlcnZlciAmJiBzdGFydHVwTmV0U2VydmVyICE9PSBzdGFydHVwU3RhdGUubmV0U2VydmVyKSB7XG4gICAgICBhd2FpdCB0aGlzLnN0b3BTZXJ2ZXIoc3RhcnR1cE5ldFNlcnZlcilcbiAgICB9XG5cbiAgICBjb25zdCBzdGFydHVwV29ya2VySGFuZGxlcnMgPSB0aGlzLndvcmtlckhhbmRsZXJzLmZpbHRlcigod29ya2VySGFuZGxlcikgPT4gIXN0YXJ0dXBTdGF0ZS53b3JrZXJIYW5kbGVycy5pbmNsdWRlcyh3b3JrZXJIYW5kbGVyKSlcblxuICAgIGF3YWl0IFByb21pc2UuYWxsKHN0YXJ0dXBXb3JrZXJIYW5kbGVycy5tYXAoKGhhbmRsZXIpID0+IGhhbmRsZXIuc3RvcCgpKSlcblxuICAgIHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlciA9IHN0YXJ0dXBTdGF0ZS5kZXZlbG9wbWVudFJlbG9hZGVyXG4gICAgdGhpcy5uZXRTZXJ2ZXIgPSBzdGFydHVwU3RhdGUubmV0U2VydmVyXG4gICAgdGhpcy53b3JrZXJIYW5kbGVycyA9IHN0YXJ0dXBTdGF0ZS53b3JrZXJIYW5kbGVyc1xuICAgIHRoaXMuc3RpY2t5V29ya2VySGFuZGxlcnMuY2xlYXIoKVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgbmV0IHNlcnZlciBsaXN0ZW4uXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gY29tcGxldGUuXG4gICAqL1xuICBfbmV0U2VydmVyTGlzdGVuKCkge1xuICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICBpZiAoIXRoaXMubmV0U2VydmVyKSB0aHJvdyBuZXcgRXJyb3IoXCJObyBuZXRTZXJ2ZXJcIilcblxuICAgICAgLyoqXG4gICAgICAgKiBPbiBsaXN0ZW4gZXJyb3IuXG4gICAgICAgKiBAcGFyYW0ge0Vycm9yfSBlcnJvciAtIExpc3RlbiBlcnJvci5cbiAgICAgICAqL1xuICAgICAgY29uc3Qgb25MaXN0ZW5FcnJvciA9IChlcnJvcikgPT4ge1xuICAgICAgICB0aGlzLm5ldFNlcnZlcj8ub2ZmKFwiZXJyb3JcIiwgb25MaXN0ZW5FcnJvcilcbiAgICAgICAgcmVqZWN0KGVycm9yKVxuICAgICAgfVxuXG4gICAgICB0cnkge1xuICAgICAgICB0aGlzLm5ldFNlcnZlci5vbmNlKFwiZXJyb3JcIiwgb25MaXN0ZW5FcnJvcilcbiAgICAgICAgdGhpcy5uZXRTZXJ2ZXIubGlzdGVuKHRoaXMucG9ydCwgdGhpcy5ob3N0LCAoKSA9PiB7XG4gICAgICAgICAgdGhpcy5uZXRTZXJ2ZXI/Lm9mZihcImVycm9yXCIsIG9uTGlzdGVuRXJyb3IpXG4gICAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYFZlbG9jaW91cyBsaXN0ZW5pbmcgb24gJHt0aGlzLmhvc3R9OiR7dGhpcy5wb3J0fWApXG4gICAgICAgICAgcmVzb2x2ZSh1bmRlZmluZWQpXG4gICAgICAgIH0pXG4gICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICByZWplY3QoZXJyb3IpXG4gICAgICB9XG4gICAgfSlcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGVuc3VyZSB3b3JrZXJzLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgYXN5bmMgX2Vuc3VyZVdvcmtlcnMoKSB7XG4gICAgd2hpbGUgKHRoaXMud29ya2VySGFuZGxlcnMubGVuZ3RoIDwgdGhpcy53b3JrZXJzKSB7XG4gICAgICBhd2FpdCB0aGlzLnNwYXduV29ya2VyKClcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBpcyBhY3RpdmUuXG4gICAqIEByZXR1cm5zIHtib29sZWFufSAtIFdoZXRoZXIgYWN0aXZlLlxuICAgKi9cbiAgaXNBY3RpdmUoKSB7XG4gICAgaWYgKHRoaXMubmV0U2VydmVyKSB7XG4gICAgICByZXR1cm4gdGhpcy5uZXRTZXJ2ZXIubGlzdGVuaW5nXG4gICAgfVxuXG4gICAgcmV0dXJuIGZhbHNlXG4gIH1cblxuICAvKipcbiAgICogUnVucyBnZXQgZGVidWcgc25hcHNob3QuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPFJlY29yZDxzdHJpbmcsID8+Pn0gLSBIVFRQIHNlcnZlciB3b3JrZXIgZGlhZ25vc3RpY3MuXG4gICAqL1xuICBhc3luYyBnZXREZWJ1Z1NuYXBzaG90KCkge1xuICAgIHJldHVybiB7XG4gICAgICBhY3RpdmU6IHRoaXMuaXNBY3RpdmUoKSxcbiAgICAgIGFjdGl2ZVNvY2tldENvdW50OiB0aGlzLl9hY3RpdmVTb2NrZXRzLnNpemUsXG4gICAgICBjbGllbnRDb3VudDogT2JqZWN0LmtleXModGhpcy5jbGllbnRzKS5sZW5ndGgsXG4gICAgICBjb25maWd1cmVkV29ya2VyQ291bnQ6IHRoaXMud29ya2VycyxcbiAgICAgIGluUHJvY2VzczogdGhpcy5pblByb2Nlc3MsXG4gICAgICB3b3JrZXJDb3VudDogdGhpcy53b3JrZXJIYW5kbGVycy5sZW5ndGgsXG4gICAgICB3b3JrZXJzOiBhd2FpdCBQcm9taXNlLmFsbCh0aGlzLndvcmtlckhhbmRsZXJzLm1hcCgoaGFuZGxlcikgPT4gdGhpcy53b3JrZXJEZWJ1Z1NuYXBzaG90KGhhbmRsZXIpKSlcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyB3b3JrZXIgZGVidWcgc25hcHNob3QuXG4gICAqIEBwYXJhbSB7V29ya2VySGFuZGxlciB8IEluUHJvY2Vzc0hhbmRsZXJ9IHdvcmtlckhhbmRsZXIgLSBXb3JrZXIgaGFuZGxlciB0byBpbnNwZWN0LlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCA/Pj59IFdvcmtlciBkZWJ1ZyBzbmFwc2hvdC5cbiAgICovXG4gIGFzeW5jIHdvcmtlckRlYnVnU25hcHNob3Qod29ya2VySGFuZGxlcikge1xuICAgIGlmICh3b3JrZXJIYW5kbGVyIGluc3RhbmNlb2YgV29ya2VySGFuZGxlcikgcmV0dXJuIGF3YWl0IHdvcmtlckhhbmRsZXIuZ2V0RGVidWdTbmFwc2hvdCgpXG4gICAgaWYgKHdvcmtlckhhbmRsZXIgaW5zdGFuY2VvZiBJblByb2Nlc3NIYW5kbGVyKSByZXR1cm4gdGhpcy5pblByb2Nlc3NXb3JrZXJEZWJ1Z1NuYXBzaG90KHdvcmtlckhhbmRsZXIpXG5cbiAgICByZXR1cm4ge2FjdGl2ZTogZmFsc2UsIGVycm9yOiBcIlVua25vd24gd29ya2VyIGhhbmRsZXIgdHlwZVwifVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgaW4gcHJvY2VzcyB3b3JrZXIgZGVidWcgc25hcHNob3QuXG4gICAqIEBwYXJhbSB7SW5Qcm9jZXNzSGFuZGxlcn0gd29ya2VySGFuZGxlciAtIEluLXByb2Nlc3Mgd29ya2VyIGhhbmRsZXIgdG8gaW5zcGVjdC5cbiAgICogQHJldHVybnMge1JlY29yZDxzdHJpbmcsID8+fSBXb3JrZXIgZGVidWcgc25hcHNob3QuXG4gICAqL1xuICBpblByb2Nlc3NXb3JrZXJEZWJ1Z1NuYXBzaG90KHdvcmtlckhhbmRsZXIpIHtcbiAgICByZXR1cm4ge1xuICAgICAgYWN0aXZlOiB0cnVlLFxuICAgICAgY2xpZW50Q291bnQ6IE9iamVjdC5rZXlzKHdvcmtlckhhbmRsZXIuY2xpZW50cykubGVuZ3RoLFxuICAgICAgc25hcHNob3Q6IHdvcmtlckhhbmRsZXIuY29uZmlndXJhdGlvbi5nZXRMb2NhbERlYnVnU25hcHNob3QoKSxcbiAgICAgIHdvcmtlckNvdW50OiB3b3JrZXJIYW5kbGVyLndvcmtlckNvdW50XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgc3RvcCBjbGllbnRzLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgYXN5bmMgc3RvcENsaWVudHMoKSB7XG4gICAgY29uc3QgcHJvbWlzZXMgPSBbXVxuXG4gICAgZm9yIChjb25zdCBjbGllbnRDb3VudCBpbiB0aGlzLmNsaWVudHMpIHtcbiAgICAgIGNvbnN0IGNsaWVudCA9IHRoaXMuY2xpZW50c1tjbGllbnRDb3VudF1cblxuICAgICAgcHJvbWlzZXMucHVzaChjbGllbnQuZW5kKCkpXG4gICAgfVxuXG4gICAgYXdhaXQgUHJvbWlzZS5hbGwocHJvbWlzZXMpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBzdG9wIHNlcnZlci5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCJuZXRcIikuU2VydmVyIHwgdW5kZWZpbmVkfSBbbmV0U2VydmVyXSAtIFNlcnZlciB0byBzdG9wLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgc3RvcFNlcnZlcihuZXRTZXJ2ZXIgPSB0aGlzLm5ldFNlcnZlcikge1xuICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICBpZiAoIW5ldFNlcnZlciB8fCAhbmV0U2VydmVyLmxpc3RlbmluZykge1xuICAgICAgICByZXNvbHZlKHVuZGVmaW5lZClcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIGlmIChuZXRTZXJ2ZXIgPT09IHRoaXMubmV0U2VydmVyKSB7XG4gICAgICAgIC8vIEZvcmNlLWNsb3NlIGxpbmdlcmluZyBzb2NrZXRzIChlLmcuIFdlYlNvY2tldCB1cGdyYWRlXG4gICAgICAgIC8vIGNvbm5lY3Rpb25zIG1pZC1jbG9zZS1oYW5kc2hha2UpIHNvIHRoZSBwb3J0IGlzIHJlbGVhc2VkXG4gICAgICAgIC8vIGltbWVkaWF0ZWx5IGluc3RlYWQgb2Ygd2FpdGluZyBmb3IgZ3JhY2VmdWwgZHJhaW4uXG4gICAgICAgIGZvciAoY29uc3Qgc29ja2V0IG9mIHRoaXMuX2FjdGl2ZVNvY2tldHMpIHtcbiAgICAgICAgICBzb2NrZXQuZGVzdHJveSgpXG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLl9hY3RpdmVTb2NrZXRzLmNsZWFyKClcbiAgICAgIH1cblxuICAgICAgbmV0U2VydmVyLmNsb3NlKChlcnJvcikgPT4ge1xuICAgICAgICBpZiAoZXJyb3IpIHtcbiAgICAgICAgICByZWplY3QoZXJyb3IpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgcmVzb2x2ZSh1bmRlZmluZWQpXG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgfSlcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHN0b3AuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gY29tcGxldGUuXG4gICAqL1xuICBhc3luYyBzdG9wKCkge1xuICAgIHRoaXMuX3N0b3BwaW5nID0gdHJ1ZVxuICAgIGF3YWl0IHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlcj8uc3RvcCgpXG4gICAgdGhpcy5kZXZlbG9wbWVudFJlbG9hZGVyID0gdW5kZWZpbmVkXG4gICAgYXdhaXQgdGhpcy5zdG9wQ2xpZW50cygpXG4gICAgYXdhaXQgdGhpcy5zdG9wU2VydmVyKClcblxuICAgIGNvbnN0IHN0b3BUYXNrcyA9IHRoaXMud29ya2VySGFuZGxlcnMubWFwKChoYW5kbGVyKSA9PiBoYW5kbGVyLnN0b3AoKSlcbiAgICBhd2FpdCBQcm9taXNlLmFsbChzdG9wVGFza3MpXG4gICAgdGhpcy53b3JrZXJIYW5kbGVycyA9IFtdXG4gICAgdGhpcy5zdGlja3lXb3JrZXJIYW5kbGVycy5jbGVhcigpXG4gIH1cblxuICAvKipcbiAgICogT24gY2xvc2UuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIG9uQ2xvc2UgPSAoKSA9PiB7XG4gICAgdGhpcy5ldmVudHMuZW1pdChcImNsb3NlXCIpXG4gIH1cblxuICAvKipcbiAgICogT24gc2VydmVyIGVycm9yLlxuICAgKiBAcGFyYW0ge0Vycm9yfSBlcnJvciAtIFNlcnZlciBzb2NrZXQgZXJyb3IuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIG9uU2VydmVyRXJyb3IgPSAoZXJyb3IpID0+IHtcbiAgICB0aGlzLmxvZ2dlci5lcnJvcihgVmVsb2Npb3VzIEhUVFAgc2VydmVyIHNvY2tldCBlcnJvciBvbiAke3RoaXMuaG9zdH06JHt0aGlzLnBvcnR9YCwgZXJyb3IpXG4gIH1cblxuICAvKipcbiAgICogT24gY29ubmVjdGlvbi5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCJuZXRcIikuU29ja2V0fSBzb2NrZXQgLSBTb2NrZXQgaW5zdGFuY2UuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIG9uQ29ubmVjdGlvbiA9IChzb2NrZXQpID0+IHtcbiAgICBjb25zdCBjbGllbnRDb3VudCA9IHRoaXMuY2xpZW50Q291bnRcblxuICAgIHRoaXMuX2FjdGl2ZVNvY2tldHMuYWRkKHNvY2tldClcbiAgICBzb2NrZXQub25jZShcImNsb3NlXCIsICgpID0+IHRoaXMuX2FjdGl2ZVNvY2tldHMuZGVsZXRlKHNvY2tldCkpXG5cbiAgICB0aGlzLmxvZ2dlci5kZWJ1ZygoKSA9PiBbXCJOZXcgY2xpZW50XCIsIHtcbiAgICAgIGNsaWVudENvdW50LFxuICAgICAgcmVtb3RlQWRkcmVzczogc29ja2V0LnJlbW90ZUFkZHJlc3MsXG4gICAgICByZW1vdGVGYW1pbHk6IHNvY2tldC5yZW1vdGVGYW1pbHksXG4gICAgICByZW1vdGVQb3J0OiBzb2NrZXQucmVtb3RlUG9ydFxuICAgIH1dKVxuICAgIHRoaXMuY2xpZW50Q291bnQrK1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIFBhdXNlZCBXZWJTb2NrZXQgc2Vzc2lvbnMgYXJlIHdvcmtlci1sb2NhbCwgc28gcmVjb25uZWN0cyBmcm9tXG4gICAgICAvLyB0aGUgc2FtZSBjbGllbnQgYWRkcmVzcyBtdXN0IHJldHVybiB0byB0aGUgc2FtZSB3b3JrZXIuXG4gICAgICBjb25zdCB3b3JrZXJIYW5kbGVyID0gdGhpcy53b3JrZXJIYW5kbGVyVG9Vc2Uoe3N0aWNreUtleTogc29ja2V0LnJlbW90ZUFkZHJlc3N9KVxuICAgICAgY29uc3QgY2xpZW50ID0gbmV3IFNlcnZlckNsaWVudCh7XG4gICAgICAgIGNsaWVudENvdW50LFxuICAgICAgICBjb25maWd1cmF0aW9uOiB0aGlzLmNvbmZpZ3VyYXRpb24sXG4gICAgICAgIHNvY2tldFxuICAgICAgfSlcblxuICAgICAgY2xpZW50LmV2ZW50cy5vbihcImNsb3NlXCIsIHRoaXMub25DbGllbnRDbG9zZSlcblxuICAgICAgdGhpcy5sb2dnZXIuZGVidWcoYEdhdmUgY2xpZW50ICR7Y2xpZW50Q291bnR9IHRvIHdvcmtlciAke3dvcmtlckhhbmRsZXIud29ya2VyQ291bnR9YClcbiAgICAgIHdvcmtlckhhbmRsZXIuYWRkU29ja2V0Q29ubmVjdGlvbihjbGllbnQpXG4gICAgICB0aGlzLmNsaWVudHNbY2xpZW50Q291bnRdID0gY2xpZW50XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIHRoaXMubG9nZ2VyLmVycm9yKGBGYWlsZWQgdG8gaW5pdGlhbGl6ZSBjbGllbnQgJHtjbGllbnRDb3VudH0gb24gbmV3IGNvbm5lY3Rpb25gLCBlcnJvcilcbiAgICAgIHNvY2tldC5kZXN0cm95KClcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogT24gY2xpZW50IGNsb3NlLlxuICAgKiBAcGFyYW0ge1NlcnZlckNsaWVudH0gY2xpZW50IC0gQ2xpZW50IGluc3RhbmNlLlxuICAgKiBAcmV0dXJucyB7dm9pZH0gLSBObyByZXR1cm4gdmFsdWUuXG4gICAqL1xuICBvbkNsaWVudENsb3NlID0gKGNsaWVudCkgPT4ge1xuICAgIGNvbnN0IGNsaWVudENvdW50ID0gZGlnZyhjbGllbnQsIFwiY2xpZW50Q291bnRcIilcbiAgICBjb25zdCBvbGRDbGllbnRzTGVuZ3RoID0gT2JqZWN0LmtleXModGhpcy5jbGllbnRzKS5sZW5ndGhcblxuICAgIGRlbGV0ZSB0aGlzLmNsaWVudHNbY2xpZW50Q291bnRdXG5cbiAgICBjb25zdCBuZXdDbGllbnRzTGVuZ3RoID0gT2JqZWN0LmtleXModGhpcy5jbGllbnRzKS5sZW5ndGhcblxuICAgIGlmIChuZXdDbGllbnRzTGVuZ3RoICE9IChvbGRDbGllbnRzTGVuZ3RoIC0gMSkpIHtcbiAgICAgIHRoaXMubG9nZ2VyLmVycm9yKGBFeHBlY3RlZCBjbGllbnQgdG8gaGF2ZSBiZWVuIHJlbW92ZWQgYnV0IGxlbmd0aCBkaWRuJ3QgY2hhbmdlIGZyb20gJHtvbGRDbGllbnRzTGVuZ3RofSB0byAke29sZENsaWVudHNMZW5ndGggLSAxfWApXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgc3Bhd24gd29ya2VyLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgYXN5bmMgc3Bhd25Xb3JrZXIoKSB7XG4gICAgY29uc3Qgd29ya2VySGFuZGxlciA9IGF3YWl0IHRoaXMuX2J1aWxkV29ya2VySGFuZGxlcigpXG5cbiAgICB0aGlzLndvcmtlckhhbmRsZXJzLnB1c2god29ya2VySGFuZGxlcilcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGJ1aWxkIHdvcmtlciBoYW5kbGVycy5cbiAgICogQHJldHVybnMge1Byb21pc2U8QXJyYXk8V29ya2VySGFuZGxlciB8IEluUHJvY2Vzc0hhbmRsZXI+Pn0gLSBTdGFydGVkIHdvcmtlciBoYW5kbGVycy5cbiAgICovXG4gIGFzeW5jIF9idWlsZFdvcmtlckhhbmRsZXJzKCkge1xuICAgIC8qKlxuICAgICAqIFdvcmtlciBoYW5kbGVycy5cbiAgICAgIEB0eXBlIHtBcnJheTxXb3JrZXJIYW5kbGVyIHwgSW5Qcm9jZXNzSGFuZGxlcj59ICovXG4gICAgY29uc3Qgd29ya2VySGFuZGxlcnMgPSBbXVxuXG4gICAgZm9yIChsZXQgaW5kZXggPSAwOyBpbmRleCA8IHRoaXMud29ya2VyczsgaW5kZXggKz0gMSkge1xuICAgICAgd29ya2VySGFuZGxlcnMucHVzaChhd2FpdCB0aGlzLl9idWlsZFdvcmtlckhhbmRsZXIoKSlcbiAgICB9XG5cbiAgICByZXR1cm4gd29ya2VySGFuZGxlcnNcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGJ1aWxkIHdvcmtlciBoYW5kbGVyLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxXb3JrZXJIYW5kbGVyIHwgSW5Qcm9jZXNzSGFuZGxlcj59IC0gU3RhcnRlZCB3b3JrZXIgaGFuZGxlci5cbiAgICovXG4gIGFzeW5jIF9idWlsZFdvcmtlckhhbmRsZXIoKSB7XG4gICAgY29uc3Qgd29ya2VyQ291bnQgPSB0aGlzLndvcmtlckNvdW50XG5cbiAgICB0aGlzLndvcmtlckNvdW50KytcblxuICAgIGNvbnN0IEhhbmRsZXIgPSB0aGlzLmluUHJvY2VzcyA/IEluUHJvY2Vzc0hhbmRsZXIgOiBXb3JrZXJIYW5kbGVyXG4gICAgY29uc3Qgd29ya2VySGFuZGxlciA9IHRoaXMud29ya2VySGFuZGxlckZhY3RvcnlcbiAgICAgID8gdGhpcy53b3JrZXJIYW5kbGVyRmFjdG9yeSh7Y29uZmlndXJhdGlvbjogdGhpcy5jb25maWd1cmF0aW9uLCB3b3JrZXJDb3VudH0pXG4gICAgICA6IG5ldyBIYW5kbGVyKHtcbiAgICAgICAgY29uZmlndXJhdGlvbjogdGhpcy5jb25maWd1cmF0aW9uLFxuICAgICAgICB3b3JrZXJDb3VudFxuICAgICAgfSlcblxuICAgIGF3YWl0IHdvcmtlckhhbmRsZXIuc3RhcnQoKVxuXG4gICAgcmV0dXJuIHdvcmtlckhhbmRsZXJcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHdvcmtlciBoYW5kbGVyIHRvIHVzZS5cbiAgICogQHBhcmFtIHtvYmplY3R9IFthcmdzXSAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gW2FyZ3Muc3RpY2t5S2V5XSAtIFN0YWJsZSBrZXkgdGhhdCBtdXN0IGtlZXAgcm91dGluZyB0byB0aGUgc2FtZSB3b3JrZXIuXG4gICAqIEByZXR1cm5zIHtXb3JrZXJIYW5kbGVyIHwgSW5Qcm9jZXNzSGFuZGxlcn0gLSBUaGUgd29ya2VyIGhhbmRsZXIgdG8gdXNlLlxuICAgKi9cbiAgd29ya2VySGFuZGxlclRvVXNlKHtzdGlja3lLZXl9ID0ge30pIHtcbiAgICBpZiAoc3RpY2t5S2V5KSB7XG4gICAgICBjb25zdCBzdGlja3lXb3JrZXJIYW5kbGVyID0gdGhpcy5zdGlja3lXb3JrZXJIYW5kbGVycy5nZXQoc3RpY2t5S2V5KVxuXG4gICAgICBpZiAoc3RpY2t5V29ya2VySGFuZGxlciAmJiB0aGlzLndvcmtlckhhbmRsZXJzLmluY2x1ZGVzKHN0aWNreVdvcmtlckhhbmRsZXIpKSB7XG4gICAgICAgIHJldHVybiBzdGlja3lXb3JrZXJIYW5kbGVyXG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHdvcmtlckhhbmRsZXIgPSB0aGlzLl9uZXh0Um91bmRSb2JpbldvcmtlckhhbmRsZXIoKVxuXG4gICAgICB0aGlzLnN0aWNreVdvcmtlckhhbmRsZXJzLnNldChzdGlja3lLZXksIHdvcmtlckhhbmRsZXIpXG5cbiAgICAgIHJldHVybiB3b3JrZXJIYW5kbGVyXG4gICAgfVxuXG4gICAgcmV0dXJuIHRoaXMuX25leHRSb3VuZFJvYmluV29ya2VySGFuZGxlcigpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBuZXh0IHJvdW5kIHJvYmluIHdvcmtlciBoYW5kbGVyLlxuICAgKiBAcmV0dXJucyB7V29ya2VySGFuZGxlciB8IEluUHJvY2Vzc0hhbmRsZXJ9IC0gVGhlIG5leHQgcm91bmQtcm9iaW4gd29ya2VyIGhhbmRsZXIuXG4gICAqL1xuICBfbmV4dFJvdW5kUm9iaW5Xb3JrZXJIYW5kbGVyKCkge1xuICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBXb3JrZXIgaGFuZGxlcnMgbGVuZ3RoOiAke3RoaXMud29ya2VySGFuZGxlcnMubGVuZ3RofWApXG5cbiAgICBjb25zdCB3b3JrZXJIYW5kbGVySW5kZXggPSB0aGlzLm5leHRXb3JrZXJIYW5kbGVySW5kZXggJSB0aGlzLndvcmtlckhhbmRsZXJzLmxlbmd0aFxuICAgIGNvbnN0IHdvcmtlckhhbmRsZXIgPSB0aGlzLndvcmtlckhhbmRsZXJzW3dvcmtlckhhbmRsZXJJbmRleF1cblxuICAgIGlmICghd29ya2VySGFuZGxlcikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBObyB3b3JrZXJIYW5kbGVyIGJ5IHRoYXQgbnVtYmVyOiAke3dvcmtlckhhbmRsZXJJbmRleH1gKVxuICAgIH1cblxuICAgIHRoaXMubmV4dFdvcmtlckhhbmRsZXJJbmRleCArPSAxXG5cbiAgICByZXR1cm4gd29ya2VySGFuZGxlclxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgc2hvdWxkIHVzZSBkZXZlbG9wbWVudCBob3QgcmVsb2FkLlxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gLSBXaGV0aGVyIGRldmVsb3BtZW50IHdvcmtlciBob3QgcmVsb2FkIHNob3VsZCBydW4uXG4gICAqL1xuICBzaG91bGRVc2VEZXZlbG9wbWVudEhvdFJlbG9hZCgpIHtcbiAgICByZXR1cm4gIXRoaXMuaW5Qcm9jZXNzICYmIHRoaXMuY29uZmlndXJhdGlvbi5nZXRFbnZpcm9ubWVudCgpID09PSBcImRldmVsb3BtZW50XCJcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHN0YXJ0IGRldmVsb3BtZW50IHJlbG9hZGVyLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIHdhdGNoZXIgc2V0dXAgZmluaXNoZXMuXG4gICAqL1xuICBhc3luYyBfc3RhcnREZXZlbG9wbWVudFJlbG9hZGVyKCkge1xuICAgIGlmICghdGhpcy5zaG91bGRVc2VEZXZlbG9wbWVudEhvdFJlbG9hZCgpKSByZXR1cm5cbiAgICBpZiAodGhpcy5kZXZlbG9wbWVudFJlbG9hZGVyKSByZXR1cm5cblxuICAgIGNvbnN0IGNyZWF0ZURldmVsb3BtZW50UmVsb2FkZXIgPSB0aGlzLmRldmVsb3BtZW50UmVsb2FkZXJGYWN0b3J5XG4gICAgICB8fCAoKGFyZ3MpID0+IG5ldyBEZXZlbG9wbWVudFJlbG9hZGVyKGFyZ3MpKVxuXG4gICAgdGhpcy5kZXZlbG9wbWVudFJlbG9hZGVyID0gY3JlYXRlRGV2ZWxvcG1lbnRSZWxvYWRlcih7XG4gICAgICBjb25maWd1cmF0aW9uOiB0aGlzLmNvbmZpZ3VyYXRpb24sXG4gICAgICBvblJlbG9hZDogYXN5bmMgKHtjaGFuZ2VkUGF0aH0pID0+IHtcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dnZXIuaW5mbyhgRGV2ZWxvcG1lbnQgaG90IHJlbG9hZCBkZXRlY3RlZCBjaGFuZ2UgaW4gJHtjaGFuZ2VkUGF0aH1gKVxuICAgICAgICBhd2FpdCB0aGlzLnJlbG9hZFdvcmtlcnNGb3JEZXZlbG9wbWVudCgpXG4gICAgICB9XG4gICAgfSlcblxuICAgIGF3YWl0IHRoaXMuZGV2ZWxvcG1lbnRSZWxvYWRlci5zdGFydCgpXG4gIH1cblxuICAvKipcbiAgICogUnVucyByZWxvYWQgd29ya2VycyBmb3IgZGV2ZWxvcG1lbnQuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gd29ya2VycyBoYXZlIGJlZW4gcmVmcmVzaGVkLlxuICAgKi9cbiAgYXN5bmMgcmVsb2FkV29ya2Vyc0ZvckRldmVsb3BtZW50KCkge1xuICAgIGlmICh0aGlzLl9zdG9wcGluZykgcmV0dXJuXG5cbiAgICBpZiAodGhpcy5fcmVsb2FkaW5nV29ya2Vyc0ZvckRldmVsb3BtZW50KSB7XG4gICAgICB0aGlzLl9yZWxvYWRXb3JrZXJzRm9yRGV2ZWxvcG1lbnRRdWV1ZWQgPSB0cnVlXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICB0aGlzLl9yZWxvYWRpbmdXb3JrZXJzRm9yRGV2ZWxvcG1lbnQgPSB0cnVlXG5cbiAgICB0cnkge1xuICAgICAgZG8ge1xuICAgICAgICB0aGlzLl9yZWxvYWRXb3JrZXJzRm9yRGV2ZWxvcG1lbnRRdWV1ZWQgPSBmYWxzZVxuXG4gICAgICAgIGNvbnN0IG9sZFdvcmtlckhhbmRsZXJzID0gWy4uLnRoaXMud29ya2VySGFuZGxlcnNdXG4gICAgICAgIGNvbnN0IG5ld1dvcmtlckhhbmRsZXJzID0gYXdhaXQgdGhpcy5fYnVpbGRXb3JrZXJIYW5kbGVycygpXG5cbiAgICAgICAgdGhpcy53b3JrZXJIYW5kbGVycyA9IG5ld1dvcmtlckhhbmRsZXJzXG4gICAgICAgIHRoaXMubmV4dFdvcmtlckhhbmRsZXJJbmRleCA9IDBcbiAgICAgICAgdGhpcy5zdGlja3lXb3JrZXJIYW5kbGVycy5jbGVhcigpXG5cbiAgICAgICAgYXdhaXQgUHJvbWlzZS5hbGwob2xkV29ya2VySGFuZGxlcnMubWFwKCh3b3JrZXJIYW5kbGVyKSA9PiB3b3JrZXJIYW5kbGVyLnN0b3AoKSkpXG4gICAgICB9IHdoaWxlICh0aGlzLl9yZWxvYWRXb3JrZXJzRm9yRGV2ZWxvcG1lbnRRdWV1ZWQgJiYgIXRoaXMuX3N0b3BwaW5nKVxuICAgIH0gZmluYWxseSB7XG4gICAgICB0aGlzLl9yZWxvYWRpbmdXb3JrZXJzRm9yRGV2ZWxvcG1lbnQgPSBmYWxzZVxuICAgIH1cbiAgfVxufVxuIl19