@uql/core 1.0.12 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (466) hide show
  1. package/CHANGELOG.md +243 -0
  2. package/LICENSE.md +2 -2
  3. package/dist/CHANGELOG.md +186 -0
  4. package/dist/README.md +413 -0
  5. package/dist/browser/http/bus.d.ts +3 -0
  6. package/dist/browser/http/bus.js +14 -0
  7. package/dist/browser/http/http.d.ts +6 -0
  8. package/dist/browser/http/http.js +45 -0
  9. package/dist/browser/http/index.d.ts +2 -0
  10. package/dist/browser/http/index.js +3 -0
  11. package/dist/browser/index.d.ts +4 -0
  12. package/dist/browser/index.js +5 -0
  13. package/dist/browser/options.d.ts +4 -0
  14. package/dist/browser/options.js +14 -0
  15. package/dist/browser/querier/genericClientRepository.d.ts +17 -0
  16. package/dist/browser/querier/genericClientRepository.js +39 -0
  17. package/dist/browser/querier/httpQuerier.d.ts +20 -0
  18. package/dist/browser/querier/httpQuerier.js +71 -0
  19. package/dist/browser/querier/index.d.ts +3 -0
  20. package/dist/browser/querier/index.js +4 -0
  21. package/dist/browser/querier/querier.util.d.ts +2 -0
  22. package/dist/browser/querier/querier.util.js +17 -0
  23. package/dist/browser/type/clientQuerier.d.ts +16 -0
  24. package/dist/browser/type/clientQuerier.js +2 -0
  25. package/dist/browser/type/clientQuerierPool.d.ts +4 -0
  26. package/dist/browser/type/clientQuerierPool.js +2 -0
  27. package/dist/browser/type/clientRepository.d.ts +13 -0
  28. package/dist/browser/type/clientRepository.js +2 -0
  29. package/dist/browser/type/index.d.ts +4 -0
  30. package/dist/browser/type/index.js +5 -0
  31. package/dist/browser/type/request.d.ts +28 -0
  32. package/dist/browser/type/request.js +2 -0
  33. package/dist/browser/uql-browser.min.js +2150 -0
  34. package/dist/browser/uql-browser.min.js.map +1 -0
  35. package/dist/dialect/abstractDialect.d.ts +16 -0
  36. package/dist/dialect/abstractDialect.js +28 -0
  37. package/dist/dialect/abstractSqlDialect.d.ts +47 -0
  38. package/dist/dialect/abstractSqlDialect.js +650 -0
  39. package/dist/dialect/index.d.ts +3 -0
  40. package/dist/dialect/index.js +4 -0
  41. package/dist/dialect/queryContext.d.ts +48 -0
  42. package/dist/dialect/queryContext.js +65 -0
  43. package/{entity → dist/entity}/decorator/definition.d.ts +3 -3
  44. package/dist/entity/decorator/definition.js +214 -0
  45. package/{entity → dist/entity}/decorator/entity.d.ts +1 -1
  46. package/dist/entity/decorator/entity.js +7 -0
  47. package/{entity → dist/entity}/decorator/field.d.ts +1 -1
  48. package/dist/entity/decorator/field.js +8 -0
  49. package/{entity → dist/entity}/decorator/id.d.ts +1 -1
  50. package/dist/entity/decorator/id.js +8 -0
  51. package/dist/entity/decorator/index.d.ts +5 -0
  52. package/dist/entity/decorator/index.js +6 -0
  53. package/{entity → dist/entity}/decorator/relation.d.ts +1 -1
  54. package/dist/entity/decorator/relation.js +20 -0
  55. package/dist/entity/index.d.ts +1 -0
  56. package/dist/entity/index.js +2 -0
  57. package/dist/express/index.d.ts +2 -0
  58. package/dist/express/index.js +3 -0
  59. package/dist/express/querierMiddleware.d.ts +26 -0
  60. package/dist/express/querierMiddleware.js +190 -0
  61. package/dist/express/query.util.d.ts +2 -0
  62. package/dist/express/query.util.js +19 -0
  63. package/dist/index.d.ts +9 -0
  64. package/dist/index.js +10 -0
  65. package/dist/maria/index.d.ts +3 -0
  66. package/dist/maria/index.js +4 -0
  67. package/dist/maria/mariaDialect.d.ts +8 -0
  68. package/dist/maria/mariaDialect.js +38 -0
  69. package/dist/maria/mariaQuerierPool.test.d.ts +5 -0
  70. package/dist/maria/mariaQuerierPool.test.js +19 -0
  71. package/dist/maria/mariadbQuerier.d.ts +17 -0
  72. package/dist/maria/mariadbQuerier.js +39 -0
  73. package/dist/maria/mariadbQuerier.test.d.ts +4 -0
  74. package/dist/maria/mariadbQuerier.test.js +19 -0
  75. package/dist/maria/mariadbQuerierPool.d.ts +10 -0
  76. package/dist/maria/mariadbQuerierPool.js +17 -0
  77. package/dist/migrate/cli.d.ts +2 -0
  78. package/dist/migrate/cli.js +254 -0
  79. package/dist/migrate/generator/index.d.ts +4 -0
  80. package/dist/migrate/generator/index.js +5 -0
  81. package/dist/migrate/generator/mongoSchemaGenerator.d.ts +12 -0
  82. package/dist/migrate/generator/mongoSchemaGenerator.js +100 -0
  83. package/dist/migrate/generator/mysqlSchemaGenerator.d.ts +14 -0
  84. package/dist/migrate/generator/mysqlSchemaGenerator.js +81 -0
  85. package/dist/migrate/generator/postgresSchemaGenerator.d.ts +18 -0
  86. package/dist/migrate/generator/postgresSchemaGenerator.js +111 -0
  87. package/dist/migrate/generator/sqliteSchemaGenerator.d.ts +14 -0
  88. package/dist/migrate/generator/sqliteSchemaGenerator.js +68 -0
  89. package/dist/migrate/index.d.ts +12 -0
  90. package/dist/migrate/index.js +17 -0
  91. package/dist/migrate/introspection/index.d.ts +4 -0
  92. package/dist/migrate/introspection/index.js +5 -0
  93. package/dist/migrate/introspection/mongoIntrospector.d.ts +8 -0
  94. package/dist/migrate/introspection/mongoIntrospector.js +46 -0
  95. package/dist/migrate/introspection/mysqlIntrospector.d.ts +25 -0
  96. package/dist/migrate/introspection/mysqlIntrospector.js +220 -0
  97. package/dist/migrate/introspection/postgresIntrospector.d.ts +21 -0
  98. package/dist/migrate/introspection/postgresIntrospector.js +269 -0
  99. package/dist/migrate/introspection/sqliteIntrospector.d.ts +23 -0
  100. package/dist/migrate/introspection/sqliteIntrospector.js +212 -0
  101. package/dist/migrate/migrator-mongo.test.d.ts +1 -0
  102. package/dist/migrate/migrator-mongo.test.js +54 -0
  103. package/dist/migrate/migrator.d.ts +133 -0
  104. package/dist/migrate/migrator.js +600 -0
  105. package/dist/migrate/migrator.test.d.ts +1 -0
  106. package/dist/migrate/migrator.test.js +106 -0
  107. package/dist/migrate/schemaGenerator.d.ts +78 -0
  108. package/dist/migrate/schemaGenerator.js +363 -0
  109. package/dist/migrate/storage/databaseStorage.d.ts +24 -0
  110. package/dist/migrate/storage/databaseStorage.js +77 -0
  111. package/dist/migrate/storage/index.d.ts +2 -0
  112. package/dist/migrate/storage/index.js +3 -0
  113. package/dist/migrate/storage/jsonStorage.d.ts +15 -0
  114. package/dist/migrate/storage/jsonStorage.js +51 -0
  115. package/dist/migrate/type.d.ts +1 -0
  116. package/dist/migrate/type.js +2 -0
  117. package/dist/mongo/index.d.ts +3 -0
  118. package/dist/mongo/index.js +4 -0
  119. package/dist/mongo/mongoDialect.d.ts +34 -0
  120. package/dist/mongo/mongoDialect.js +163 -0
  121. package/dist/mongo/mongodbQuerier.d.ts +28 -0
  122. package/dist/mongo/mongodbQuerier.js +204 -0
  123. package/dist/mongo/mongodbQuerier.test.d.ts +1 -0
  124. package/dist/mongo/mongodbQuerier.test.js +36 -0
  125. package/dist/mongo/mongodbQuerierPool.d.ts +10 -0
  126. package/dist/mongo/mongodbQuerierPool.js +20 -0
  127. package/dist/mongo/mongodbQuerierPool.test.d.ts +1 -0
  128. package/dist/mongo/mongodbQuerierPool.test.js +21 -0
  129. package/dist/mysql/index.d.ts +3 -0
  130. package/dist/mysql/index.js +4 -0
  131. package/dist/mysql/mysql2Querier.d.ts +17 -0
  132. package/dist/mysql/mysql2Querier.js +43 -0
  133. package/dist/mysql/mysql2Querier.test.d.ts +4 -0
  134. package/dist/mysql/mysql2Querier.test.js +16 -0
  135. package/dist/mysql/mysql2QuerierPool.d.ts +10 -0
  136. package/dist/mysql/mysql2QuerierPool.js +17 -0
  137. package/dist/mysql/mysql2QuerierPool.test.d.ts +5 -0
  138. package/dist/mysql/mysql2QuerierPool.test.js +16 -0
  139. package/dist/mysql/mysqlDialect.d.ts +5 -0
  140. package/dist/mysql/mysqlDialect.js +15 -0
  141. package/dist/namingStrategy/defaultNamingStrategy.d.ts +9 -0
  142. package/dist/namingStrategy/defaultNamingStrategy.js +15 -0
  143. package/dist/namingStrategy/index.d.ts +2 -0
  144. package/dist/namingStrategy/index.js +3 -0
  145. package/dist/namingStrategy/snakeCaseNamingStrategy.d.ts +8 -0
  146. package/dist/namingStrategy/snakeCaseNamingStrategy.js +14 -0
  147. package/{options.d.ts → dist/options.d.ts} +1 -1
  148. package/dist/options.js +14 -0
  149. package/dist/package.json +131 -0
  150. package/dist/postgres/index.d.ts +3 -0
  151. package/dist/postgres/index.js +4 -0
  152. package/dist/postgres/pgQuerier.d.ts +17 -0
  153. package/dist/postgres/pgQuerier.js +39 -0
  154. package/dist/postgres/pgQuerier.test.d.ts +4 -0
  155. package/dist/postgres/pgQuerier.test.js +20 -0
  156. package/dist/postgres/pgQuerierPool.d.ts +10 -0
  157. package/dist/postgres/pgQuerierPool.js +17 -0
  158. package/dist/postgres/pgQuerierPool.test.d.ts +5 -0
  159. package/dist/postgres/pgQuerierPool.test.js +23 -0
  160. package/dist/postgres/postgresDialect.d.ts +13 -0
  161. package/dist/postgres/postgresDialect.js +110 -0
  162. package/dist/querier/abstractQuerier-test.d.ts +45 -0
  163. package/dist/querier/abstractQuerier-test.js +461 -0
  164. package/dist/querier/abstractQuerier.d.ts +50 -0
  165. package/dist/querier/abstractQuerier.js +278 -0
  166. package/dist/querier/abstractQuerierPool-test.d.ts +9 -0
  167. package/dist/querier/abstractQuerierPool-test.js +18 -0
  168. package/dist/querier/abstractQuerierPool.d.ts +14 -0
  169. package/dist/querier/abstractQuerierPool.js +9 -0
  170. package/dist/querier/abstractSqlQuerier-test.d.ts +9 -0
  171. package/dist/querier/abstractSqlQuerier-test.js +16 -0
  172. package/dist/querier/abstractSqlQuerier.d.ts +28 -0
  173. package/dist/querier/abstractSqlQuerier.js +133 -0
  174. package/dist/querier/decorator/index.d.ts +3 -0
  175. package/dist/querier/decorator/index.js +4 -0
  176. package/dist/querier/decorator/injectQuerier.d.ts +3 -0
  177. package/dist/querier/decorator/injectQuerier.js +33 -0
  178. package/dist/querier/decorator/serialized.d.ts +6 -0
  179. package/dist/querier/decorator/serialized.js +14 -0
  180. package/{querier → dist/querier}/decorator/transactional.d.ts +1 -1
  181. package/dist/querier/decorator/transactional.js +48 -0
  182. package/dist/querier/index.d.ts +4 -0
  183. package/dist/querier/index.js +5 -0
  184. package/dist/repository/genericRepository.d.ts +20 -0
  185. package/dist/repository/genericRepository.js +51 -0
  186. package/dist/repository/index.d.ts +1 -0
  187. package/dist/repository/index.js +2 -0
  188. package/dist/sqlite/index.d.ts +3 -0
  189. package/dist/sqlite/index.js +4 -0
  190. package/dist/sqlite/sqliteDialect.d.ts +10 -0
  191. package/dist/sqlite/sqliteDialect.js +48 -0
  192. package/dist/sqlite/sqliteQuerier.d.ts +15 -0
  193. package/dist/sqlite/sqliteQuerier.js +33 -0
  194. package/dist/sqlite/sqliteQuerier.test.d.ts +5 -0
  195. package/dist/sqlite/sqliteQuerier.test.js +19 -0
  196. package/dist/sqlite/sqliteQuerierPool.d.ts +14 -0
  197. package/dist/sqlite/sqliteQuerierPool.js +34 -0
  198. package/dist/sqlite/sqliteQuerierPool.test.d.ts +5 -0
  199. package/dist/sqlite/sqliteQuerierPool.test.js +10 -0
  200. package/dist/test/entityMock.d.ts +164 -0
  201. package/dist/test/entityMock.js +554 -0
  202. package/dist/test/index.d.ts +3 -0
  203. package/dist/test/index.js +4 -0
  204. package/dist/test/it.util.d.ts +4 -0
  205. package/dist/test/it.util.js +55 -0
  206. package/dist/test/spec.util.d.ts +14 -0
  207. package/dist/test/spec.util.js +50 -0
  208. package/{type → dist/type}/entity.d.ts +97 -4
  209. package/dist/type/entity.js +5 -0
  210. package/dist/type/index.d.ts +9 -0
  211. package/dist/type/index.js +10 -0
  212. package/dist/type/migration.d.ts +213 -0
  213. package/dist/type/migration.js +2 -0
  214. package/dist/type/namingStrategy.d.ts +17 -0
  215. package/dist/type/namingStrategy.js +2 -0
  216. package/dist/type/querier.d.ts +96 -0
  217. package/dist/type/querier.js +11 -0
  218. package/{type → dist/type}/querierPool.d.ts +7 -3
  219. package/dist/type/querierPool.js +2 -0
  220. package/{type → dist/type}/query.d.ts +170 -108
  221. package/dist/type/query.js +9 -0
  222. package/{type → dist/type}/repository.d.ts +42 -35
  223. package/dist/type/repository.js +2 -0
  224. package/{type → dist/type}/universalQuerier.d.ts +50 -28
  225. package/dist/type/universalQuerier.js +2 -0
  226. package/dist/type/utility.d.ts +13 -0
  227. package/dist/type/utility.js +2 -0
  228. package/dist/util/dialect.util.d.ts +12 -0
  229. package/dist/util/dialect.util.js +93 -0
  230. package/dist/util/index.d.ts +5 -0
  231. package/dist/util/index.js +6 -0
  232. package/dist/util/object.util.d.ts +7 -0
  233. package/dist/util/object.util.js +19 -0
  234. package/dist/util/raw.d.ts +8 -0
  235. package/dist/util/raw.js +11 -0
  236. package/dist/util/sql.util.d.ts +13 -0
  237. package/dist/util/sql.util.js +78 -0
  238. package/{util → dist/util}/string.util.d.ts +1 -0
  239. package/dist/util/string.util.js +34 -0
  240. package/package.json +84 -16
  241. package/src/@types/index.d.ts +1 -0
  242. package/src/@types/jest.d.ts +6 -0
  243. package/src/browser/http/bus.spec.ts +22 -0
  244. package/src/browser/http/bus.ts +17 -0
  245. package/src/browser/http/http.spec.ts +70 -0
  246. package/src/browser/http/http.ts +55 -0
  247. package/src/browser/http/index.ts +2 -0
  248. package/src/browser/index.ts +4 -0
  249. package/src/browser/options.spec.ts +37 -0
  250. package/src/browser/options.ts +18 -0
  251. package/src/browser/querier/genericClientRepository.spec.ts +105 -0
  252. package/src/browser/querier/genericClientRepository.ts +49 -0
  253. package/src/browser/querier/httpQuerier.ts +82 -0
  254. package/src/browser/querier/index.ts +3 -0
  255. package/src/browser/querier/querier.util.spec.ts +35 -0
  256. package/src/browser/querier/querier.util.ts +18 -0
  257. package/src/browser/type/clientQuerier.ts +45 -0
  258. package/src/browser/type/clientQuerierPool.ts +5 -0
  259. package/src/browser/type/clientRepository.ts +22 -0
  260. package/src/browser/type/index.ts +4 -0
  261. package/src/browser/type/request.ts +25 -0
  262. package/src/dialect/abstractDialect.ts +28 -0
  263. package/src/dialect/abstractSqlDialect-spec.ts +1309 -0
  264. package/src/dialect/abstractSqlDialect.ts +805 -0
  265. package/src/dialect/index.ts +3 -0
  266. package/src/dialect/namingStrategy.spec.ts +52 -0
  267. package/src/dialect/queryContext.ts +69 -0
  268. package/src/entity/decorator/definition.spec.ts +736 -0
  269. package/src/entity/decorator/definition.ts +265 -0
  270. package/src/entity/decorator/entity.ts +8 -0
  271. package/src/entity/decorator/field.ts +9 -0
  272. package/src/entity/decorator/id.ts +9 -0
  273. package/src/entity/decorator/index.ts +5 -0
  274. package/src/entity/decorator/relation.spec.ts +41 -0
  275. package/src/entity/decorator/relation.ts +34 -0
  276. package/src/entity/index.ts +1 -0
  277. package/src/express/@types/express.d.ts +8 -0
  278. package/src/express/@types/index.d.ts +1 -0
  279. package/src/express/index.ts +2 -0
  280. package/src/express/querierMiddleware.ts +217 -0
  281. package/src/express/query.util.spec.ts +40 -0
  282. package/src/express/query.util.ts +21 -0
  283. package/src/index.ts +9 -0
  284. package/src/maria/index.ts +3 -0
  285. package/src/maria/mariaDialect.spec.ts +207 -0
  286. package/src/maria/mariaDialect.ts +42 -0
  287. package/src/maria/mariaQuerierPool.test.ts +23 -0
  288. package/src/maria/mariadbQuerier.test.ts +23 -0
  289. package/src/maria/mariadbQuerier.ts +45 -0
  290. package/src/maria/mariadbQuerierPool.ts +21 -0
  291. package/src/migrate/cli.ts +301 -0
  292. package/src/migrate/generator/index.ts +4 -0
  293. package/src/migrate/generator/mongoSchemaGenerator.spec.ts +112 -0
  294. package/src/migrate/generator/mongoSchemaGenerator.ts +115 -0
  295. package/src/migrate/generator/mysqlSchemaGenerator.spec.ts +34 -0
  296. package/src/migrate/generator/mysqlSchemaGenerator.ts +92 -0
  297. package/src/migrate/generator/postgresSchemaGenerator.spec.ts +44 -0
  298. package/src/migrate/generator/postgresSchemaGenerator.ts +127 -0
  299. package/src/migrate/generator/sqliteSchemaGenerator.spec.ts +33 -0
  300. package/src/migrate/generator/sqliteSchemaGenerator.ts +81 -0
  301. package/src/migrate/index.ts +41 -0
  302. package/src/migrate/introspection/index.ts +4 -0
  303. package/src/migrate/introspection/mongoIntrospector.spec.ts +75 -0
  304. package/src/migrate/introspection/mongoIntrospector.ts +47 -0
  305. package/src/migrate/introspection/mysqlIntrospector.spec.ts +113 -0
  306. package/src/migrate/introspection/mysqlIntrospector.ts +278 -0
  307. package/src/migrate/introspection/postgresIntrospector.spec.ts +112 -0
  308. package/src/migrate/introspection/postgresIntrospector.ts +329 -0
  309. package/src/migrate/introspection/sqliteIntrospector.spec.ts +112 -0
  310. package/src/migrate/introspection/sqliteIntrospector.ts +296 -0
  311. package/src/migrate/migrator-mongo.test.ts +54 -0
  312. package/src/migrate/migrator.spec.ts +255 -0
  313. package/src/migrate/migrator.test.ts +94 -0
  314. package/src/migrate/migrator.ts +719 -0
  315. package/src/migrate/namingStrategy.spec.ts +22 -0
  316. package/src/migrate/schemaGenerator-advanced.spec.ts +138 -0
  317. package/src/migrate/schemaGenerator.spec.ts +190 -0
  318. package/src/migrate/schemaGenerator.ts +478 -0
  319. package/src/migrate/storage/databaseStorage.spec.ts +69 -0
  320. package/src/migrate/storage/databaseStorage.ts +100 -0
  321. package/src/migrate/storage/index.ts +2 -0
  322. package/src/migrate/storage/jsonStorage.ts +58 -0
  323. package/src/migrate/type.ts +1 -0
  324. package/src/mongo/index.ts +3 -0
  325. package/src/mongo/mongoDialect.spec.ts +251 -0
  326. package/src/mongo/mongoDialect.ts +238 -0
  327. package/src/mongo/mongodbQuerier.test.ts +45 -0
  328. package/src/mongo/mongodbQuerier.ts +256 -0
  329. package/src/mongo/mongodbQuerierPool.test.ts +25 -0
  330. package/src/mongo/mongodbQuerierPool.ts +24 -0
  331. package/src/mysql/index.ts +3 -0
  332. package/src/mysql/mysql2Querier.test.ts +20 -0
  333. package/src/mysql/mysql2Querier.ts +49 -0
  334. package/src/mysql/mysql2QuerierPool.test.ts +20 -0
  335. package/src/mysql/mysql2QuerierPool.ts +21 -0
  336. package/src/mysql/mysqlDialect.spec.ts +20 -0
  337. package/src/mysql/mysqlDialect.ts +16 -0
  338. package/src/namingStrategy/defaultNamingStrategy.ts +18 -0
  339. package/src/namingStrategy/index.spec.ts +36 -0
  340. package/src/namingStrategy/index.ts +2 -0
  341. package/src/namingStrategy/snakeCaseNamingStrategy.ts +15 -0
  342. package/src/options.spec.ts +41 -0
  343. package/src/options.ts +18 -0
  344. package/src/postgres/index.ts +3 -0
  345. package/src/postgres/manual-types.d.ts +4 -0
  346. package/src/postgres/pgQuerier.test.ts +25 -0
  347. package/src/postgres/pgQuerier.ts +45 -0
  348. package/src/postgres/pgQuerierPool.test.ts +28 -0
  349. package/src/postgres/pgQuerierPool.ts +21 -0
  350. package/src/postgres/postgresDialect.spec.ts +428 -0
  351. package/src/postgres/postgresDialect.ts +144 -0
  352. package/src/querier/abstractQuerier-test.ts +584 -0
  353. package/src/querier/abstractQuerier.ts +353 -0
  354. package/src/querier/abstractQuerierPool-test.ts +20 -0
  355. package/src/querier/abstractQuerierPool.ts +18 -0
  356. package/src/querier/abstractSqlQuerier-spec.ts +979 -0
  357. package/src/querier/abstractSqlQuerier-test.ts +21 -0
  358. package/src/querier/abstractSqlQuerier.ts +138 -0
  359. package/src/querier/decorator/index.ts +3 -0
  360. package/src/querier/decorator/injectQuerier.spec.ts +74 -0
  361. package/src/querier/decorator/injectQuerier.ts +45 -0
  362. package/src/querier/decorator/serialized.spec.ts +98 -0
  363. package/src/querier/decorator/serialized.ts +13 -0
  364. package/src/querier/decorator/transactional.spec.ts +240 -0
  365. package/src/querier/decorator/transactional.ts +56 -0
  366. package/src/querier/index.ts +4 -0
  367. package/src/repository/genericRepository.spec.ts +111 -0
  368. package/src/repository/genericRepository.ts +74 -0
  369. package/src/repository/index.ts +1 -0
  370. package/src/sqlite/index.ts +3 -0
  371. package/src/sqlite/manual-types.d.ts +4 -0
  372. package/src/sqlite/sqliteDialect.spec.ts +155 -0
  373. package/src/sqlite/sqliteDialect.ts +76 -0
  374. package/src/sqlite/sqliteQuerier.spec.ts +36 -0
  375. package/src/sqlite/sqliteQuerier.test.ts +21 -0
  376. package/src/sqlite/sqliteQuerier.ts +37 -0
  377. package/src/sqlite/sqliteQuerierPool.test.ts +12 -0
  378. package/src/sqlite/sqliteQuerierPool.ts +38 -0
  379. package/src/test/entityMock.ts +375 -0
  380. package/src/test/index.ts +3 -0
  381. package/src/test/it.util.ts +69 -0
  382. package/src/test/spec.util.ts +57 -0
  383. package/src/type/entity.ts +218 -0
  384. package/src/type/index.ts +9 -0
  385. package/src/type/migration.ts +241 -0
  386. package/src/type/namingStrategy.ts +17 -0
  387. package/src/type/querier.ts +143 -0
  388. package/src/type/querierPool.ts +26 -0
  389. package/src/type/query.ts +506 -0
  390. package/src/type/repository.ts +142 -0
  391. package/src/type/universalQuerier.ts +133 -0
  392. package/src/type/utility.ts +21 -0
  393. package/src/util/dialect.util-extra.spec.ts +96 -0
  394. package/src/util/dialect.util.spec.ts +23 -0
  395. package/src/util/dialect.util.ts +134 -0
  396. package/src/util/index.ts +5 -0
  397. package/src/util/object.util.spec.ts +29 -0
  398. package/src/util/object.util.ts +27 -0
  399. package/src/util/raw.ts +11 -0
  400. package/src/util/sql.util-extra.spec.ts +17 -0
  401. package/src/util/sql.util.spec.ts +208 -0
  402. package/src/util/sql.util.ts +104 -0
  403. package/src/util/string.util.spec.ts +46 -0
  404. package/src/util/string.util.ts +35 -0
  405. package/tsconfig.build.json +5 -0
  406. package/tsconfig.json +8 -0
  407. package/README.md +0 -177
  408. package/dialect/abstractSqlDialect.d.ts +0 -30
  409. package/dialect/abstractSqlDialect.js +0 -365
  410. package/dialect/index.d.ts +0 -4
  411. package/dialect/index.js +0 -8
  412. package/dialect/mysqlDialect.d.ts +0 -6
  413. package/dialect/mysqlDialect.js +0 -21
  414. package/dialect/postgresDialect.d.ts +0 -8
  415. package/dialect/postgresDialect.js +0 -44
  416. package/dialect/sqliteDialect.d.ts +0 -4
  417. package/dialect/sqliteDialect.js +0 -11
  418. package/entity/decorator/definition.js +0 -223
  419. package/entity/decorator/entity.js +0 -11
  420. package/entity/decorator/field.js +0 -12
  421. package/entity/decorator/id.js +0 -12
  422. package/entity/decorator/index.d.ts +0 -5
  423. package/entity/decorator/index.js +0 -12
  424. package/entity/decorator/relation.js +0 -27
  425. package/entity/index.d.ts +0 -1
  426. package/entity/index.js +0 -5
  427. package/index.d.ts +0 -1
  428. package/index.js +0 -5
  429. package/options.js +0 -20
  430. package/querier/abstractQuerier.d.ts +0 -30
  431. package/querier/abstractQuerier.js +0 -230
  432. package/querier/abstractSqlQuerier.d.ts +0 -27
  433. package/querier/abstractSqlQuerier.js +0 -88
  434. package/querier/decorator/index.d.ts +0 -2
  435. package/querier/decorator/index.js +0 -6
  436. package/querier/decorator/injectQuerier.d.ts +0 -3
  437. package/querier/decorator/injectQuerier.js +0 -39
  438. package/querier/decorator/transactional.js +0 -52
  439. package/querier/index.d.ts +0 -3
  440. package/querier/index.js +0 -7
  441. package/repository/genericRepository.d.ts +0 -19
  442. package/repository/genericRepository.js +0 -52
  443. package/repository/index.d.ts +0 -1
  444. package/repository/index.js +0 -5
  445. package/type/entity.js +0 -5
  446. package/type/index.d.ts +0 -7
  447. package/type/index.js +0 -11
  448. package/type/querier.d.ts +0 -53
  449. package/type/querier.js +0 -3
  450. package/type/querierPool.js +0 -3
  451. package/type/query.js +0 -13
  452. package/type/repository.js +0 -3
  453. package/type/universalQuerier.js +0 -3
  454. package/type/utility.d.ts +0 -12
  455. package/type/utility.js +0 -3
  456. package/util/dialect.util.d.ts +0 -17
  457. package/util/dialect.util.js +0 -114
  458. package/util/index.d.ts +0 -5
  459. package/util/index.js +0 -9
  460. package/util/object.util.d.ts +0 -3
  461. package/util/object.util.js +0 -22
  462. package/util/raw.d.ts +0 -2
  463. package/util/raw.js +0 -9
  464. package/util/sql.util.d.ts +0 -2
  465. package/util/sql.util.js +0 -55
  466. package/util/string.util.js +0 -20
@@ -0,0 +1,650 @@
1
+ import { getMeta } from '../entity/index.js';
2
+ import { QueryRaw, } from '../type/index.js';
3
+ import { buildSortMap, buldQueryWhereAsMap, escapeSqlId, fillOnFields, filterFieldKeys, filterRelationKeys, flatObject, getFieldCallbackValue, getFieldKeys, getKeys, hasKeys, isSelectingRelations, raw, } from '../util/index.js';
4
+ import { AbstractDialect } from './abstractDialect.js';
5
+ import { SqlQueryContext } from './queryContext.js';
6
+ export class AbstractSqlDialect extends AbstractDialect {
7
+ escapeIdChar;
8
+ beginTransactionCommand;
9
+ commitTransactionCommand;
10
+ rollbackTransactionCommand;
11
+ constructor(namingStrategy, escapeIdChar = '`', beginTransactionCommand = 'START TRANSACTION', commitTransactionCommand = 'COMMIT', rollbackTransactionCommand = 'ROLLBACK') {
12
+ super(namingStrategy);
13
+ this.escapeIdChar = escapeIdChar;
14
+ this.beginTransactionCommand = beginTransactionCommand;
15
+ this.commitTransactionCommand = commitTransactionCommand;
16
+ this.rollbackTransactionCommand = rollbackTransactionCommand;
17
+ }
18
+ createContext() {
19
+ return new SqlQueryContext(this);
20
+ }
21
+ addValue(values, value) {
22
+ values.push(value ?? null);
23
+ return this.placeholder(values.length);
24
+ }
25
+ placeholder(_index) {
26
+ return '?';
27
+ }
28
+ returningId(entity) {
29
+ const meta = getMeta(entity);
30
+ const idName = this.resolveColumnName(meta.id, meta.fields[meta.id]);
31
+ return `RETURNING ${this.escapeId(idName)} ${this.escapeId('id')}`;
32
+ }
33
+ search(ctx, entity, q = {}, opts = {}) {
34
+ const meta = getMeta(entity);
35
+ const tableName = this.resolveTableName(entity, meta);
36
+ const prefix = (opts.prefix ?? (opts.autoPrefix || isSelectingRelations(meta, q.$select))) ? tableName : undefined;
37
+ opts = { ...opts, prefix };
38
+ this.where(ctx, entity, q.$where, opts);
39
+ this.sort(ctx, entity, q.$sort, opts);
40
+ this.pager(ctx, q);
41
+ }
42
+ selectFields(ctx, entity, select, opts = {}) {
43
+ const meta = getMeta(entity);
44
+ const prefix = opts.prefix ? opts.prefix + '.' : '';
45
+ const escapedPrefix = this.escapeId(opts.prefix, true, true);
46
+ let selectArr;
47
+ if (select) {
48
+ if (Array.isArray(select)) {
49
+ selectArr = select;
50
+ }
51
+ else {
52
+ const selectPositive = getKeys(select).filter((it) => select[it]);
53
+ selectArr = selectPositive.length
54
+ ? selectPositive
55
+ : getFieldKeys(meta.fields).filter((it) => !(it in select));
56
+ }
57
+ selectArr = selectArr.filter((it) => it instanceof QueryRaw || it in meta.fields);
58
+ if (opts.prefix && !selectArr.includes(meta.id)) {
59
+ selectArr = [meta.id, ...selectArr];
60
+ }
61
+ }
62
+ else {
63
+ selectArr = getFieldKeys(meta.fields);
64
+ }
65
+ if (!selectArr.length) {
66
+ ctx.append(escapedPrefix + '*');
67
+ return;
68
+ }
69
+ selectArr.forEach((key, index) => {
70
+ if (index > 0)
71
+ ctx.append(', ');
72
+ if (key instanceof QueryRaw) {
73
+ this.getRawValue(ctx, {
74
+ value: key,
75
+ prefix: opts.prefix,
76
+ escapedPrefix,
77
+ autoPrefixAlias: opts.autoPrefixAlias,
78
+ });
79
+ }
80
+ else {
81
+ const field = meta.fields[key];
82
+ const columnName = this.resolveColumnName(key, field);
83
+ if (field.virtual) {
84
+ this.getRawValue(ctx, {
85
+ value: raw(field.virtual.value, key),
86
+ prefix: opts.prefix,
87
+ escapedPrefix,
88
+ autoPrefixAlias: opts.autoPrefixAlias,
89
+ });
90
+ }
91
+ else {
92
+ ctx.append(escapedPrefix + this.escapeId(columnName));
93
+ }
94
+ if (!field.virtual && (columnName !== key || opts.autoPrefixAlias)) {
95
+ const aliasStr = (prefix + key);
96
+ // Replace dots with underscores for alias to avoid syntax errors
97
+ const safeAlias = aliasStr.replace(/\./g, '_');
98
+ ctx.append(' ' + this.escapeId(safeAlias, true));
99
+ }
100
+ }
101
+ });
102
+ }
103
+ select(ctx, entity, select, opts = {}) {
104
+ const meta = getMeta(entity);
105
+ const tableName = this.resolveTableName(entity, meta);
106
+ const prefix = (opts.prefix ?? (opts.autoPrefix || isSelectingRelations(meta, select))) ? tableName : undefined;
107
+ ctx.append('SELECT ');
108
+ this.selectFields(ctx, entity, select, { prefix });
109
+ // Add related fields BEFORE FROM clause
110
+ this.selectRelationFields(ctx, entity, select, { prefix });
111
+ ctx.append(` FROM ${this.escapeId(tableName)}`);
112
+ // Add JOINs AFTER FROM clause
113
+ this.selectRelationJoins(ctx, entity, select, { prefix });
114
+ }
115
+ selectRelationFields(ctx, entity, select, opts = {}) {
116
+ if (Array.isArray(select)) {
117
+ return;
118
+ }
119
+ const meta = getMeta(entity);
120
+ const tableName = this.resolveTableName(entity, meta);
121
+ const relKeys = filterRelationKeys(meta, select);
122
+ const isSelectArray = Array.isArray(select);
123
+ const prefix = opts.prefix;
124
+ for (const relKey of relKeys) {
125
+ const relOpts = meta.relations[relKey];
126
+ if (relOpts.cardinality === '1m' || relOpts.cardinality === 'mm') {
127
+ continue;
128
+ }
129
+ const isFirstLevel = prefix === tableName;
130
+ const joinRelAlias = isFirstLevel ? relKey : prefix ? prefix + '.' + relKey : relKey;
131
+ const relEntity = relOpts.entity();
132
+ const relSelect = select[relKey];
133
+ const relQuery = isSelectArray ? {} : Array.isArray(relSelect) ? { $select: relSelect } : relSelect;
134
+ ctx.append(', ');
135
+ this.selectFields(ctx, relEntity, relQuery.$select, {
136
+ prefix: joinRelAlias,
137
+ autoPrefixAlias: true,
138
+ });
139
+ // Recursively add nested relation fields
140
+ this.selectRelationFields(ctx, relEntity, relQuery.$select, {
141
+ prefix: joinRelAlias,
142
+ });
143
+ }
144
+ }
145
+ selectRelationJoins(ctx, entity, select, opts = {}) {
146
+ if (Array.isArray(select)) {
147
+ return;
148
+ }
149
+ const meta = getMeta(entity);
150
+ const tableName = this.resolveTableName(entity, meta);
151
+ const relKeys = filterRelationKeys(meta, select);
152
+ const isSelectArray = Array.isArray(select);
153
+ const prefix = opts.prefix;
154
+ for (const relKey of relKeys) {
155
+ const relOpts = meta.relations[relKey];
156
+ if (relOpts.cardinality === '1m' || relOpts.cardinality === 'mm') {
157
+ continue;
158
+ }
159
+ const isFirstLevel = prefix === tableName;
160
+ const joinRelAlias = isFirstLevel ? relKey : prefix ? prefix + '.' + relKey : relKey;
161
+ const relEntity = relOpts.entity();
162
+ const relSelect = select[relKey];
163
+ const relQuery = isSelectArray ? {} : Array.isArray(relSelect) ? { $select: relSelect } : relSelect;
164
+ const relMeta = getMeta(relEntity);
165
+ const relTableName = this.resolveTableName(relEntity, relMeta);
166
+ const relEntityName = this.escapeId(relTableName);
167
+ const relPath = prefix ? this.escapeId(prefix, true) : this.escapeId(tableName);
168
+ const joinType = relQuery.$required ? 'INNER' : 'LEFT';
169
+ const joinAlias = this.escapeId(joinRelAlias, true);
170
+ ctx.append(` ${joinType} JOIN ${relEntityName} ${joinAlias} ON `);
171
+ ctx.append(relOpts.references
172
+ .map((it) => {
173
+ const foreignColumnName = this.resolveColumnName(it.foreign, relMeta.fields[it.foreign]);
174
+ const localColumnName = this.resolveColumnName(it.local, meta.fields[it.local]);
175
+ return `${joinAlias}.${this.escapeId(foreignColumnName)} = ${relPath}.${this.escapeId(localColumnName)}`;
176
+ })
177
+ .join(' AND '));
178
+ if (relQuery.$where) {
179
+ ctx.append(' AND ');
180
+ this.where(ctx, relEntity, relQuery.$where, { prefix: joinRelAlias, clause: false });
181
+ }
182
+ // Recursively add nested relation JOINs
183
+ this.selectRelationJoins(ctx, relEntity, relQuery.$select, {
184
+ prefix: joinRelAlias,
185
+ });
186
+ }
187
+ }
188
+ where(ctx, entity, where = {}, opts = {}) {
189
+ const meta = getMeta(entity);
190
+ const { usePrecedence, clause = 'WHERE', softDelete } = opts;
191
+ where = buldQueryWhereAsMap(meta, where);
192
+ if (meta.softDelete && (softDelete || softDelete === undefined) && !where[meta.softDelete]) {
193
+ where[meta.softDelete] = null;
194
+ }
195
+ const entries = Object.entries(where);
196
+ if (!entries.length) {
197
+ return;
198
+ }
199
+ if (clause) {
200
+ ctx.append(` ${clause} `);
201
+ }
202
+ if (usePrecedence) {
203
+ ctx.append('(');
204
+ }
205
+ entries.forEach(([key, val], index) => {
206
+ if (index > 0) {
207
+ ctx.append(' AND ');
208
+ }
209
+ this.compare(ctx, entity, key, val, {
210
+ ...opts,
211
+ usePrecedence: entries.length > 1,
212
+ });
213
+ });
214
+ if (usePrecedence) {
215
+ ctx.append(')');
216
+ }
217
+ }
218
+ compare(ctx, entity, key, val, opts = {}) {
219
+ const meta = getMeta(entity);
220
+ if (val instanceof QueryRaw) {
221
+ if (key === '$exists' || key === '$nexists') {
222
+ ctx.append(key === '$exists' ? 'EXISTS (' : 'NOT EXISTS (');
223
+ const tableName = this.resolveTableName(entity, meta);
224
+ this.getRawValue(ctx, {
225
+ value: val,
226
+ prefix: tableName,
227
+ escapedPrefix: this.escapeId(tableName, false, true),
228
+ });
229
+ ctx.append(')');
230
+ return;
231
+ }
232
+ this.getComparisonKey(ctx, entity, key, opts);
233
+ ctx.append(' = ');
234
+ this.getRawValue(ctx, { value: val });
235
+ return;
236
+ }
237
+ if (key === '$text') {
238
+ const search = val;
239
+ const fields = search.$fields.map((fKey) => {
240
+ const field = meta.fields[fKey];
241
+ const columnName = this.resolveColumnName(fKey, field);
242
+ return this.escapeId(columnName);
243
+ });
244
+ ctx.append(`MATCH(${fields.join(', ')}) AGAINST(`);
245
+ ctx.addValue(search.$value);
246
+ ctx.append(')');
247
+ return;
248
+ }
249
+ if (key === '$and' || key === '$or' || key === '$not' || key === '$nor') {
250
+ const negateOperatorMap = {
251
+ $not: '$and',
252
+ $nor: '$or',
253
+ };
254
+ const op = negateOperatorMap[key] ?? key;
255
+ const negate = key in negateOperatorMap ? 'NOT' : '';
256
+ const valArr = val;
257
+ const hasManyItems = valArr.length > 1;
258
+ if ((opts.usePrecedence || negate) && hasManyItems) {
259
+ ctx.append((negate ? negate + ' ' : '') + '(');
260
+ }
261
+ else if (negate) {
262
+ ctx.append(negate + ' ');
263
+ }
264
+ valArr.forEach((whereEntry, index) => {
265
+ if (index > 0) {
266
+ ctx.append(op === '$or' ? ' OR ' : ' AND ');
267
+ }
268
+ if (whereEntry instanceof QueryRaw) {
269
+ this.getRawValue(ctx, {
270
+ value: whereEntry,
271
+ prefix: opts.prefix,
272
+ escapedPrefix: this.escapeId(opts.prefix, true, true),
273
+ });
274
+ }
275
+ else {
276
+ this.where(ctx, entity, whereEntry, {
277
+ prefix: opts.prefix,
278
+ usePrecedence: hasManyItems && !Array.isArray(whereEntry) && getKeys(whereEntry).length > 1,
279
+ clause: false,
280
+ });
281
+ }
282
+ });
283
+ if ((opts.usePrecedence || negate) && hasManyItems) {
284
+ ctx.append(')');
285
+ }
286
+ return;
287
+ }
288
+ const value = Array.isArray(val) ? { $in: val } : typeof val === 'object' && val !== null ? val : { $eq: val };
289
+ const operators = getKeys(value);
290
+ if (operators.length > 1) {
291
+ ctx.append('(');
292
+ }
293
+ operators.forEach((op, index) => {
294
+ if (index > 0) {
295
+ ctx.append(' AND ');
296
+ }
297
+ this.compareFieldOperator(ctx, entity, key, op, value[op], opts);
298
+ });
299
+ if (operators.length > 1) {
300
+ ctx.append(')');
301
+ }
302
+ }
303
+ compareFieldOperator(ctx, entity, key, op, val, opts = {}) {
304
+ switch (op) {
305
+ case '$eq':
306
+ this.getComparisonKey(ctx, entity, key, opts);
307
+ if (val === null) {
308
+ ctx.append(' IS NULL');
309
+ }
310
+ else {
311
+ ctx.append(' = ');
312
+ ctx.addValue(val);
313
+ }
314
+ break;
315
+ case '$ne':
316
+ this.getComparisonKey(ctx, entity, key, opts);
317
+ if (val === null) {
318
+ ctx.append(' IS NOT NULL');
319
+ }
320
+ else {
321
+ ctx.append(' <> ');
322
+ ctx.addValue(val);
323
+ }
324
+ break;
325
+ case '$not':
326
+ ctx.append('NOT (');
327
+ this.compare(ctx, entity, key, val, opts);
328
+ ctx.append(')');
329
+ break;
330
+ case '$gt':
331
+ this.getComparisonKey(ctx, entity, key, opts);
332
+ ctx.append(' > ');
333
+ ctx.addValue(val);
334
+ break;
335
+ case '$gte':
336
+ this.getComparisonKey(ctx, entity, key, opts);
337
+ ctx.append(' >= ');
338
+ ctx.addValue(val);
339
+ break;
340
+ case '$lt':
341
+ this.getComparisonKey(ctx, entity, key, opts);
342
+ ctx.append(' < ');
343
+ ctx.addValue(val);
344
+ break;
345
+ case '$lte':
346
+ this.getComparisonKey(ctx, entity, key, opts);
347
+ ctx.append(' <= ');
348
+ ctx.addValue(val);
349
+ break;
350
+ case '$startsWith':
351
+ this.getComparisonKey(ctx, entity, key, opts);
352
+ ctx.append(' LIKE ');
353
+ ctx.addValue(`${val}%`);
354
+ break;
355
+ case '$istartsWith':
356
+ this.getComparisonKey(ctx, entity, key, opts);
357
+ ctx.append(' LIKE ');
358
+ ctx.addValue(`${val.toLowerCase()}%`);
359
+ break;
360
+ case '$endsWith':
361
+ this.getComparisonKey(ctx, entity, key, opts);
362
+ ctx.append(' LIKE ');
363
+ ctx.addValue(`%${val}`);
364
+ break;
365
+ case '$iendsWith':
366
+ this.getComparisonKey(ctx, entity, key, opts);
367
+ ctx.append(' LIKE ');
368
+ ctx.addValue(`%${val.toLowerCase()}`);
369
+ break;
370
+ case '$includes':
371
+ this.getComparisonKey(ctx, entity, key, opts);
372
+ ctx.append(' LIKE ');
373
+ ctx.addValue(`%${val}%`);
374
+ break;
375
+ case '$iincludes':
376
+ this.getComparisonKey(ctx, entity, key, opts);
377
+ ctx.append(' LIKE ');
378
+ ctx.addValue(`%${val.toLowerCase()}%`);
379
+ break;
380
+ case '$ilike':
381
+ this.getComparisonKey(ctx, entity, key, opts);
382
+ ctx.append(' LIKE ');
383
+ ctx.addValue(val.toLowerCase());
384
+ break;
385
+ case '$like':
386
+ this.getComparisonKey(ctx, entity, key, opts);
387
+ ctx.append(' LIKE ');
388
+ ctx.addValue(val);
389
+ break;
390
+ case '$in':
391
+ this.getComparisonKey(ctx, entity, key, opts);
392
+ if (Array.isArray(val) && val.length > 0) {
393
+ ctx.append(' IN (');
394
+ this.addValues(ctx, val);
395
+ ctx.append(')');
396
+ }
397
+ else {
398
+ ctx.append(' IN (NULL)');
399
+ }
400
+ break;
401
+ case '$nin':
402
+ this.getComparisonKey(ctx, entity, key, opts);
403
+ if (Array.isArray(val) && val.length > 0) {
404
+ ctx.append(' NOT IN (');
405
+ this.addValues(ctx, val);
406
+ ctx.append(')');
407
+ }
408
+ else {
409
+ ctx.append(' NOT IN (NULL)');
410
+ }
411
+ break;
412
+ case '$regex':
413
+ this.getComparisonKey(ctx, entity, key, opts);
414
+ ctx.append(' REGEXP ');
415
+ ctx.addValue(val);
416
+ break;
417
+ default:
418
+ throw TypeError(`unknown operator: ${op}`);
419
+ }
420
+ }
421
+ addValues(ctx, vals) {
422
+ vals.forEach((val, index) => {
423
+ if (index > 0) {
424
+ ctx.append(', ');
425
+ }
426
+ ctx.addValue(val);
427
+ });
428
+ }
429
+ getComparisonKey(ctx, entity, key, { prefix } = {}) {
430
+ const meta = getMeta(entity);
431
+ const escapedPrefix = this.escapeId(prefix, true, true);
432
+ const field = meta.fields[key];
433
+ if (field?.virtual) {
434
+ this.getRawValue(ctx, {
435
+ value: field.virtual,
436
+ prefix,
437
+ escapedPrefix,
438
+ });
439
+ return;
440
+ }
441
+ const columnName = this.resolveColumnName(key, field);
442
+ ctx.append(escapedPrefix + this.escapeId(columnName));
443
+ }
444
+ sort(ctx, entity, sort, { prefix }) {
445
+ const sortMap = buildSortMap(sort);
446
+ if (!hasKeys(sortMap)) {
447
+ return;
448
+ }
449
+ const meta = getMeta(entity);
450
+ const flattenedSort = flatObject(sortMap, prefix);
451
+ const directionMap = { 1: '', asc: '', '-1': ' DESC', desc: ' DESC' };
452
+ ctx.append(' ORDER BY ');
453
+ Object.entries(flattenedSort).forEach(([key, sort], index) => {
454
+ if (index > 0) {
455
+ ctx.append(', ');
456
+ }
457
+ const field = meta.fields[key];
458
+ const name = this.resolveColumnName(key, field);
459
+ const direction = directionMap[sort];
460
+ ctx.append(this.escapeId(name) + direction);
461
+ });
462
+ }
463
+ pager(ctx, opts) {
464
+ if (opts.$limit) {
465
+ ctx.append(` LIMIT ${Number(opts.$limit)}`);
466
+ }
467
+ if (opts.$skip !== undefined) {
468
+ ctx.append(` OFFSET ${Number(opts.$skip)}`);
469
+ }
470
+ }
471
+ count(ctx, entity, q, opts) {
472
+ const search = { ...q };
473
+ delete search.$sort;
474
+ this.select(ctx, entity, [raw('COUNT(*)', 'count')], undefined);
475
+ this.search(ctx, entity, search, opts);
476
+ }
477
+ find(ctx, entity, q = {}, opts) {
478
+ this.select(ctx, entity, q.$select, opts);
479
+ this.search(ctx, entity, q, opts);
480
+ }
481
+ insert(ctx, entity, payload, opts) {
482
+ const meta = getMeta(entity);
483
+ const payloads = fillOnFields(meta, payload, 'onInsert');
484
+ const keys = filterFieldKeys(meta, payloads[0], 'onInsert');
485
+ const columns = keys.map((key) => {
486
+ const field = meta.fields[key];
487
+ return this.escapeId(this.resolveColumnName(key, field));
488
+ });
489
+ const tableName = this.resolveTableName(entity, meta);
490
+ ctx.append(`INSERT INTO ${this.escapeId(tableName)} (${columns.join(', ')}) VALUES (`);
491
+ payloads.forEach((it, recordIndex) => {
492
+ if (recordIndex > 0) {
493
+ ctx.append('), (');
494
+ }
495
+ keys.forEach((key, keyIndex) => {
496
+ if (keyIndex > 0) {
497
+ ctx.append(', ');
498
+ }
499
+ const field = meta.fields[key];
500
+ this.formatPersistableValue(ctx, field, it[key]);
501
+ });
502
+ });
503
+ ctx.append(')');
504
+ }
505
+ update(ctx, entity, q, payload, opts) {
506
+ const meta = getMeta(entity);
507
+ const [filledPayload] = fillOnFields(meta, payload, 'onUpdate');
508
+ const keys = filterFieldKeys(meta, filledPayload, 'onUpdate');
509
+ const tableName = this.resolveTableName(entity, meta);
510
+ ctx.append(`UPDATE ${this.escapeId(tableName)} SET `);
511
+ keys.forEach((key, index) => {
512
+ if (index > 0) {
513
+ ctx.append(', ');
514
+ }
515
+ const field = meta.fields[key];
516
+ const columnName = this.resolveColumnName(key, field);
517
+ ctx.append(`${this.escapeId(columnName)} = `);
518
+ this.formatPersistableValue(ctx, field, filledPayload[key]);
519
+ });
520
+ this.search(ctx, entity, q, opts);
521
+ }
522
+ upsert(ctx, entity, conflictPaths, payload) {
523
+ const meta = getMeta(entity);
524
+ const update = this.getUpsertUpdateAssignments(ctx, meta, conflictPaths, payload, (name) => `VALUES(${name})`);
525
+ if (update) {
526
+ this.insert(ctx, entity, payload);
527
+ ctx.append(` ON DUPLICATE KEY UPDATE ${update}`);
528
+ }
529
+ else {
530
+ const insertCtx = this.createContext();
531
+ this.insert(insertCtx, entity, payload);
532
+ ctx.append(insertCtx.sql.replace(/^INSERT/, 'INSERT IGNORE'));
533
+ insertCtx.values.forEach((val) => {
534
+ ctx.pushValue(val);
535
+ });
536
+ }
537
+ }
538
+ getUpsertUpdateAssignments(ctx, meta, conflictPaths, payload, callback) {
539
+ const [filledPayload] = fillOnFields(meta, payload, 'onUpdate');
540
+ const fields = filterFieldKeys(meta, filledPayload, 'onUpdate');
541
+ return fields
542
+ .filter((col) => !conflictPaths[col])
543
+ .map((col) => {
544
+ const field = meta.fields[col];
545
+ const columnName = this.resolveColumnName(col, field);
546
+ if (callback) {
547
+ return `${this.escapeId(columnName)} = ${callback(this.escapeId(columnName))}`;
548
+ }
549
+ const valCtx = this.createContext();
550
+ this.formatPersistableValue(valCtx, field, filledPayload[col]);
551
+ valCtx.values.forEach((val) => {
552
+ ctx.pushValue(val);
553
+ });
554
+ return `${this.escapeId(columnName)} = ${valCtx.sql}`;
555
+ })
556
+ .join(', ');
557
+ }
558
+ getUpsertConflictPathsStr(meta, conflictPaths) {
559
+ return getKeys(conflictPaths)
560
+ .map((key) => {
561
+ const field = meta.fields[key];
562
+ const columnName = this.resolveColumnName(key, field);
563
+ return this.escapeId(columnName);
564
+ })
565
+ .join(', ');
566
+ }
567
+ delete(ctx, entity, q, opts = {}) {
568
+ const meta = getMeta(entity);
569
+ const tableName = this.resolveTableName(entity, meta);
570
+ if (opts.softDelete || opts.softDelete === undefined) {
571
+ if (meta.softDelete) {
572
+ const field = meta.fields[meta.softDelete];
573
+ const value = getFieldCallbackValue(field.onDelete);
574
+ const columnName = this.resolveColumnName(meta.softDelete, field);
575
+ ctx.append(`UPDATE ${this.escapeId(tableName)} SET ${this.escapeId(columnName)} = `);
576
+ ctx.addValue(value);
577
+ this.search(ctx, entity, q, opts);
578
+ return;
579
+ }
580
+ if (opts.softDelete) {
581
+ throw TypeError(`'${tableName}' has not enabled 'softDelete'`);
582
+ }
583
+ }
584
+ ctx.append(`DELETE FROM ${this.escapeId(tableName)}`);
585
+ this.search(ctx, entity, q, opts);
586
+ }
587
+ escapeId(val, forbidQualified, addDot) {
588
+ return escapeSqlId(val, this.escapeIdChar, forbidQualified, addDot);
589
+ }
590
+ getPersistables(ctx, meta, payload, callbackKey) {
591
+ const payloads = fillOnFields(meta, payload, callbackKey);
592
+ return payloads.map((it) => this.getPersistable(ctx, meta, it, callbackKey));
593
+ }
594
+ getPersistable(ctx, meta, payload, callbackKey) {
595
+ const filledPayload = fillOnFields(meta, payload, callbackKey)[0];
596
+ const keys = filterFieldKeys(meta, filledPayload, callbackKey);
597
+ return keys.reduce((acc, key) => {
598
+ const field = meta.fields[key];
599
+ const valCtx = this.createContext();
600
+ this.formatPersistableValue(valCtx, field, filledPayload[key]);
601
+ valCtx.values.forEach((val) => {
602
+ ctx.pushValue(val);
603
+ });
604
+ acc[key] = valCtx.sql;
605
+ return acc;
606
+ }, {});
607
+ }
608
+ formatPersistableValue(ctx, field, value) {
609
+ if (value instanceof QueryRaw) {
610
+ this.getRawValue(ctx, { value });
611
+ return;
612
+ }
613
+ if (field?.type === 'json' || field?.type === 'jsonb') {
614
+ ctx.addValue(value ? JSON.stringify(value) : null);
615
+ return;
616
+ }
617
+ if (field?.type === 'vector' && Array.isArray(value)) {
618
+ ctx.addValue(`[${value.join(',')}]`);
619
+ return;
620
+ }
621
+ ctx.addValue(value);
622
+ }
623
+ getRawValue(ctx, opts) {
624
+ const { value, prefix = '', escapedPrefix, autoPrefixAlias } = opts;
625
+ if (typeof value.value === 'function') {
626
+ const res = value.value({
627
+ ...opts,
628
+ ctx,
629
+ dialect: this,
630
+ prefix,
631
+ escapedPrefix: escapedPrefix ?? this.escapeId(prefix, true, true),
632
+ });
633
+ if (typeof res === 'string' || (typeof res === 'number' && !Number.isNaN(res))) {
634
+ ctx.append(String(res));
635
+ }
636
+ }
637
+ else {
638
+ ctx.append(prefix + String(value.value));
639
+ }
640
+ const alias = value.alias;
641
+ if (alias) {
642
+ const fullAlias = autoPrefixAlias ? prefix + alias : alias;
643
+ // Replace dots with underscores for alias to avoid syntax errors
644
+ const safeAlias = fullAlias.replace(/\./g, '_');
645
+ const escapedFullAlias = this.escapeId(safeAlias, true);
646
+ ctx.append(' ' + escapedFullAlias);
647
+ }
648
+ }
649
+ }
650
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWJzdHJhY3RTcWxEaWFsZWN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RpYWxlY3QvYWJzdHJhY3RTcWxEaWFsZWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUM3QyxPQUFPLEVBWUwsUUFBUSxHQWdCVCxNQUFNLGtCQUFrQixDQUFDO0FBRTFCLE9BQU8sRUFDTCxZQUFZLEVBQ1osbUJBQW1CLEVBRW5CLFdBQVcsRUFDWCxZQUFZLEVBQ1osZUFBZSxFQUNmLGtCQUFrQixFQUNsQixVQUFVLEVBQ1YscUJBQXFCLEVBQ3JCLFlBQVksRUFDWixPQUFPLEVBQ1AsT0FBTyxFQUNQLG9CQUFvQixFQUNwQixHQUFHLEdBQ0osTUFBTSxrQkFBa0IsQ0FBQztBQUUxQixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDdkQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBRXBELE1BQU0sT0FBZ0Isa0JBQW1CLFNBQVEsZUFBZTtJQUduRDtJQUNBO0lBQ0E7SUFDQTtJQUxYLFlBQ0UsY0FBK0IsRUFDdEIsZUFBMEIsR0FBRyxFQUM3QiwwQkFBa0MsbUJBQW1CLEVBQ3JELDJCQUFtQyxRQUFRLEVBQzNDLDZCQUFxQyxVQUFVO1FBRXhELEtBQUssQ0FBQyxjQUFjLENBQUMsQ0FBQztRQUxiLGlCQUFZLEdBQVosWUFBWSxDQUFpQjtRQUM3Qiw0QkFBdUIsR0FBdkIsdUJBQXVCLENBQThCO1FBQ3JELDZCQUF3QixHQUF4Qix3QkFBd0IsQ0FBbUI7UUFDM0MsK0JBQTBCLEdBQTFCLDBCQUEwQixDQUFxQjtJQUcxRCxDQUFDO0lBRUQsYUFBYTtRQUNYLE9BQU8sSUFBSSxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVELFFBQVEsQ0FBQyxNQUFpQixFQUFFLEtBQWM7UUFDeEMsTUFBTSxDQUFD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QUksQ0FBQyxDQUFDO1lBQ3hELEdBQUcsQ0FBQyxNQUFNLENBQUMsR0FBRyxHQUFHLGdCQUFnQixDQUFDLENBQUM7UUFDckMsQ0FBQztJQUNILENBQUM7Q0FHRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGdldE1ldGEgfSBmcm9tICcuLi9lbnRpdHkvaW5kZXguanMnO1xuaW1wb3J0IHtcbiAgdHlwZSBFbnRpdHlNZXRhLFxuICB0eXBlIEZpZWxkS2V5LFxuICB0eXBlIEZpZWxkT3B0aW9ucyxcbiAgdHlwZSBOYW1pbmdTdHJhdGVneSxcbiAgdHlwZSBRdWVyeSxcbiAgdHlwZSBRdWVyeUNvbXBhcmlzb25PcHRpb25zLFxuICB0eXBlIFF1ZXJ5Q29uZmxpY3RQYXRocyxcbiAgdHlwZSBRdWVyeUNvbnRleHQsXG4gIHR5cGUgUXVlcnlEaWFsZWN0LFxuICB0eXBlIFF1ZXJ5T3B0aW9ucyxcbiAgdHlwZSBRdWVyeVBhZ2VyLFxuICBRdWVyeVJhdyxcbiAgdHlwZSBRdWVyeVJhd0ZuT3B0aW9ucyxcbiAgdHlwZSBRdWVyeVNlYXJjaCxcbiAgdHlwZSBRdWVyeVNlbGVjdCxcbiAgdHlwZSBRdWVyeVNlbGVjdEFycmF5LFxuICB0eXBlIFF1ZXJ5U2VsZWN0T3B0aW9ucyxcbiAgdHlwZSBRdWVyeVNvcnQsXG4gIHR5cGUgUXVlcnlTb3J0RGlyZWN0aW9uLFxuICB0eXBlIFF1ZXJ5VGV4dFNlYXJjaE9wdGlvbnMsXG4gIHR5cGUgUXVlcnlXaGVyZSxcbiAgdHlwZSBRdWVyeVdoZXJlQXJyYXksXG4gIHR5cGUgUXVlcnlXaGVyZUZpZWxkT3BlcmF0b3JNYXAsXG4gIHR5cGUgUXVlcnlXaGVyZU1hcCxcbiAgdHlwZSBRdWVyeVdoZXJlT3B0aW9ucyxcbiAgdHlwZSBTcWxRdWVyeURpYWxlY3QsXG4gIHR5cGUgVHlwZSxcbn0gZnJvbSAnLi4vdHlwZS9pbmRleC5qcyc7XG5cbmltcG9ydCB7XG4gIGJ1aWxkU29ydE1hcCxcbiAgYnVsZFF1ZXJ5V2hlcmVBc01hcCxcbiAgdHlwZSBDYWxsYmFja0tleSxcbiAgZXNjYXBlU3FsSWQsXG4gIGZpbGxPbkZpZWxkcyxcbiAgZmlsdGVyRmllbGRLZXlzLFxuICBmaWx0ZXJSZWxhdGlvbktleXMsXG4gIGZsYXRPYmplY3QsXG4gIGdldEZpZWxkQ2FsbGJhY2tWYWx1ZSxcbiAgZ2V0RmllbGRLZXlzLFxuICBnZXRLZXlzLFxuICBoYXNLZXlzLFxuICBpc1NlbGVjdGluZ1JlbGF0aW9ucyxcbiAgcmF3LFxufSBmcm9tICcuLi91dGlsL2luZGV4LmpzJztcblxuaW1wb3J0IHsgQWJzdHJhY3REaWFsZWN0IH0gZnJvbSAnLi9hYnN0cmFjdERpYWxlY3QuanMnO1xuaW1wb3J0IHsgU3FsUXVlcnlDb250ZXh0IH0gZnJvbSAnLi9xdWVyeUNvbnRleHQuanMnO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQWJzdHJhY3RTcWxEaWFsZWN0IGV4dGVuZHMgQWJzdHJhY3REaWFsZWN0IGltcGxlbWVudHMgUXVlcnlEaWFsZWN0LCBTcWxRdWVyeURpYWxlY3Qge1xuICBjb25zdHJ1Y3RvcihcbiAgICBuYW1pbmdTdHJhdGVneT86IE5hbWluZ1N0cmF0ZWd5LFxuICAgIHJlYWRvbmx5IGVzY2FwZUlkQ2hhcjogJ2AnIHwgJ1wiJyA9ICdgJyxcbiAgICByZWFkb25seSBiZWdpblRyYW5zYWN0aW9uQ29tbWFuZDogc3RyaW5nID0gJ1NUQVJUIFRSQU5TQUNUSU9OJyxcbiAgICByZWFkb25seSBjb21taXRUcmFuc2FjdGlvbkNvbW1hbmQ6IHN0cmluZyA9ICdDT01NSVQnLFxuICAgIHJlYWRvbmx5IHJvbGxiYWNrVHJhbnNhY3Rpb25Db21tYW5kOiBzdHJpbmcgPSAnUk9MTEJBQ0snLFxuICApIHtcbiAgICBzdXBlcihuYW1pbmdTdHJhdGVneSk7XG4gIH1cblxuICBjcmVhdGVDb250ZXh0KCk6IFF1ZXJ5Q29udGV4dCB7XG4gICAgcmV0dXJuIG5ldyBTcWxRdWVyeUNvbnRleHQodGhpcyk7XG4gIH1cblxuICBhZGRWYWx1ZSh2YWx1ZXM6IHVua25vd25bXSwgdmFsdWU6IHVua25vd24pOiBzdHJpbmcge1xuICAgIHZhbHVlcy5wdXNoKHZhbHVlID8/IG51bGwpO1xuICAgIHJldHVybiB0aGlzLnBsYWNlaG9sZGVyKHZhbHVlcy5sZW5ndGgpO1xuICB9XG5cbiAgcGxhY2Vob2xkZXIoX2luZGV4OiBudW1iZXIpOiBzdHJpbmcge1xuICAgIHJldHVybiAnPyc7XG4gIH1cblxuICByZXR1cm5pbmdJZDxFPihlbnRpdHk6IFR5cGU8RT4pOiBzdHJpbmcge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG4gICAgY29uc3QgaWROYW1lID0gdGhpcy5yZXNvbHZlQ29sdW1uTmFtZShtZXRhLmlkLCBtZXRhLmZpZWxkc1ttZXRhLmlkXSk7XG4gICAgcmV0dXJuIGBSRVRVUk5JTkcgJHt0aGlzLmVzY2FwZUlkKGlkTmFtZSl9ICR7dGhpcy5lc2NhcGVJZCgnaWQnKX1gO1xuICB9XG5cbiAgc2VhcmNoPEU+KGN0eDogUXVlcnlDb250ZXh0LCBlbnRpdHk6IFR5cGU8RT4sIHE6IFF1ZXJ5PEU+ID0ge30sIG9wdHM6IFF1ZXJ5T3B0aW9ucyA9IHt9KTogdm9pZCB7XG4gICAgY29uc3QgbWV0YSA9IGdldE1ldGEoZW50aXR5KTtcbiAgICBjb25zdCB0YWJsZU5hbWUgPSB0aGlzLnJlc29sdmVUYWJsZU5hbWUoZW50aXR5LCBtZXRhKTtcbiAgICBjb25zdCBwcmVmaXggPSAob3B0cy5wcmVmaXggPz8gKG9wdHMuYXV0b1ByZWZpeCB8fCBpc1NlbGVjdGluZ1JlbGF0aW9ucyhtZXRhLCBxLiRzZWxlY3QpKSkgPyB0YWJsZU5hbWUgOiB1bmRlZmluZWQ7XG4gICAgb3B0cyA9IHsgLi4ub3B0cywgcHJlZml4IH07XG4gICAgdGhpcy53aGVyZTxFPihjdHgsIGVudGl0eSwgcS4kd2hlcmUsIG9wdHMpO1xuICAgIHRoaXMuc29ydDxFPihjdHgsIGVudGl0eSwgcS4kc29ydCwgb3B0cyk7XG4gICAgdGhpcy5wYWdlcihjdHgsIHEpO1xuICB9XG5cbiAgc2VsZWN0RmllbGRzPEU+KGN0eDogUXVlcnlDb250ZXh0LCBlbnRpdHk6IFR5cGU8RT4sIHNlbGVjdDogUXVlcnlTZWxlY3Q8RT4sIG9wdHM6IFF1ZXJ5U2VsZWN0T3B0aW9ucyA9IHt9KTogdm9pZCB7XG4gICAgY29uc3QgbWV0YSA9IGdldE1ldGEoZW50aXR5KTtcbiAgICBjb25zdCBwcmVmaXggPSBvcHRzLnByZWZpeCA/IG9wdHMucHJlZml4ICsgJy4nIDogJyc7XG4gICAgY29uc3QgZXNjYXBlZFByZWZpeCA9IHRoaXMuZXNjYXBlSWQob3B0cy5wcmVmaXgsIHRydWUsIHRydWUpO1xuXG4gICAgbGV0IHNlbGVjdEFycjogUXVlcnlTZWxlY3RBcnJheTxFPjtcblxuICAgIGlmIChzZWxlY3QpIHtcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KHNlbGVjdCkpIHtcbiAgICAgICAgc2VsZWN0QXJyID0gc2VsZWN0O1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29uc3Qgc2VsZWN0UG9zaXRpdmUgPSBnZXRLZXlzKHNlbGVjdCkuZmlsdGVyKChpdCkgPT4gc2VsZWN0W2l0XSkgYXMgRmllbGRLZXk8RT5bXTtcbiAgICAgICAgc2VsZWN0QXJyID0gc2VsZWN0UG9zaXRpdmUubGVuZ3RoXG4gICAgICAgICAgPyBzZWxlY3RQb3NpdGl2ZVxuICAgICAgICAgIDogKGdldEZpZWxkS2V5cyhtZXRhLmZpZWxkcykuZmlsdGVyKChpdCkgPT4gIShpdCBpbiBzZWxlY3QpKSBhcyBGaWVsZEtleTxFPltdKTtcbiAgICAgIH1cbiAgICAgIHNlbGVjdEFyciA9IHNlbGVjdEFyci5maWx0ZXIoKGl0KSA9PiBpdCBpbnN0YW5jZW9mIFF1ZXJ5UmF3IHx8IGl0IGluIG1ldGEuZmllbGRzKTtcbiAgICAgIGlmIChvcHRzLnByZWZpeCAmJiAhc2VsZWN0QXJyLmluY2x1ZGVzKG1ldGEuaWQpKSB7XG4gICAgICAgIHNlbGVjdEFyciA9IFttZXRhLmlkLCAuLi5zZWxlY3RBcnJdO1xuICAgICAgfVxuICAgIH0gZWxzZSB7XG4gICAgICBzZWxlY3RBcnIgPSBnZXRGaWVsZEtleXMobWV0YS5maWVsZHMpIGFzIEZpZWxkS2V5PEU+W107XG4gICAgfVxuXG4gICAgaWYgKCFzZWxlY3RBcnIubGVuZ3RoKSB7XG4gICAgICBjdHguYXBwZW5kKGVzY2FwZWRQcmVmaXggKyAnKicpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHNlbGVjdEFyci5mb3JFYWNoKChrZXksIGluZGV4KSA9PiB7XG4gICAgICBpZiAoaW5kZXggPiAwKSBjdHguYXBwZW5kKCcsICcpO1xuICAgICAgaWYgKGtleSBpbnN0YW5jZW9mIFF1ZXJ5UmF3KSB7XG4gICAgICAgIHRoaXMuZ2V0UmF3VmFsdWUoY3R4LCB7XG4gICAgICAgICAgdmFsdWU6IGtleSxcbiAgICAgICAgICBwcmVmaXg6IG9wdHMucHJlZml4LFxuICAgICAgICAgIGVzY2FwZWRQcmVmaXgsXG4gICAgICAgICAgYXV0b1ByZWZpeEFsaWFzOiBvcHRzLmF1dG9QcmVmaXhBbGlhcyxcbiAgICAgICAgfSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCBmaWVsZCA9IG1ldGEuZmllbGRzW2tleSBhcyBGaWVsZEtleTxFPl07XG4gICAgICAgIGNvbnN0IGNvbHVtbk5hbWUgPSB0aGlzLnJlc29sdmVDb2x1bW5OYW1lKGtleSwgZmllbGQpO1xuICAgICAgICBpZiAoZmllbGQudmlydHVhbCkge1xuICAgICAgICAgIHRoaXMuZ2V0UmF3VmFsdWUoY3R4LCB7XG4gICAgICAgICAgICB2YWx1ZTogcmF3KGZpZWxkLnZpcnR1YWwudmFsdWUsIGtleSBhcyBzdHJpbmcpLFxuICAgICAgICAgICAgcHJlZml4OiBvcHRzLnByZWZpeCxcbiAgICAgICAgICAgIGVzY2FwZWRQcmVmaXgsXG4gICAgICAgICAgICBhdXRvUHJlZml4QWxpYXM6IG9wdHMuYXV0b1ByZWZpeEFsaWFzLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGN0eC5hcHBlbmQoZXNjYXBlZFByZWZpeCArIHRoaXMuZXNjYXBlSWQoY29sdW1uTmFtZSkpO1xuICAgICAgICB9XG4gICAgICAgIGlmICghZmllbGQudmlydHVhbCAmJiAoY29sdW1uTmFtZSAhPT0ga2V5IHx8IG9wdHMuYXV0b1ByZWZpeEFsaWFzKSkge1xuICAgICAgICAgIGNvbnN0IGFsaWFzU3RyID0gKHByZWZpeCArIGtleSkgYXMgc3RyaW5nO1xuICAgICAgICAgIC8vIFJlcGxhY2UgZG90cyB3aXRoIHVuZGVyc2NvcmVzIGZvciBhbGlhcyB0byBhdm9pZCBzeW50YXggZXJyb3JzXG4gICAgICAgICAgY29uc3Qgc2FmZUFsaWFzID0gYWxpYXNTdHIucmVwbGFjZSgvXFwuL2csICdfJyk7XG4gICAgICAgICAgY3R4LmFwcGVuZCgnICcgKyB0aGlzLmVzY2FwZUlkKHNhZmVBbGlhcywgdHJ1ZSkpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBzZWxlY3Q8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGVudGl0eTogVHlwZTxFPiwgc2VsZWN0OiBRdWVyeVNlbGVjdDxFPiwgb3B0czogUXVlcnlPcHRpb25zID0ge30pOiB2b2lkIHtcbiAgICBjb25zdCBtZXRhID0gZ2V0TWV0YShlbnRpdHkpO1xuICAgIGNvbnN0IHRhYmxlTmFtZSA9IHRoaXMucmVzb2x2ZVRhYmxlTmFtZShlbnRpdHksIG1ldGEpO1xuICAgIGNvbnN0IHByZWZpeCA9IChvcHRzLnByZWZpeCA/PyAob3B0cy5hdXRvUHJlZml4IHx8IGlzU2VsZWN0aW5nUmVsYXRpb25zKG1ldGEsIHNlbGVjdCkpKSA/IHRhYmxlTmFtZSA6IHVuZGVmaW5lZDtcblxuICAgIGN0eC5hcHBlbmQoJ1NFTEVDVCAnKTtcbiAgICB0aGlzLnNlbGVjdEZpZWxkcyhjdHgsIGVudGl0eSwgc2VsZWN0LCB7IHByZWZpeCB9KTtcbiAgICAvLyBBZGQgcmVsYXRlZCBmaWVsZHMgQkVGT1JFIEZST00gY2xhdXNlXG4gICAgdGhpcy5zZWxlY3RSZWxhdGlvbkZpZWxkcyhjdHgsIGVudGl0eSwgc2VsZWN0LCB7IHByZWZpeCB9KTtcbiAgICBjdHguYXBwZW5kKGAgRlJPTSAke3RoaXMuZXNjYXBlSWQodGFibGVOYW1lKX1gKTtcbiAgICAvLyBBZGQgSk9JTnMgQUZURVIgRlJPTSBjbGF1c2VcbiAgICB0aGlzLnNlbGVjdFJlbGF0aW9uSm9pbnMoY3R4LCBlbnRpdHksIHNlbGVjdCwgeyBwcmVmaXggfSk7XG4gIH1cblxuICBwcm90ZWN0ZWQgc2VsZWN0UmVsYXRpb25GaWVsZHM8RT4oXG4gICAgY3R4OiBRdWVyeUNvbnRleHQsXG4gICAgZW50aXR5OiBUeXBlPEU+LFxuICAgIHNlbGVjdDogUXVlcnlTZWxlY3Q8RT4sXG4gICAgb3B0czogeyBwcmVmaXg/OiBzdHJpbmcgfSA9IHt9LFxuICApOiB2b2lkIHtcbiAgICBpZiAoQXJyYXkuaXNBcnJheShzZWxlY3QpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgbWV0YSA9IGdldE1ldGEoZW50aXR5KTtcbiAgICBjb25zdCB0YWJsZU5hbWUgPSB0aGlzLnJlc29sdmVUYWJsZU5hbWUoZW50aXR5LCBtZXRhKTtcbiAgICBjb25zdCByZWxLZXlzID0gZmlsdGVyUmVsYXRpb25LZXlzKG1ldGEsIHNlbGVjdCk7XG4gICAgY29uc3QgaXNTZWxlY3RBcnJheSA9IEFycmF5LmlzQXJyYXkoc2VsZWN0KTtcbiAgICBjb25zdCBwcmVmaXggPSBvcHRzLnByZWZpeDtcblxuICAgIGZvciAoY29uc3QgcmVsS2V5IG9mIHJlbEtleXMpIHtcbiAgICAgIGNvbnN0IHJlbE9wdHMgPSBtZXRhLnJlbGF0aW9uc1tyZWxLZXldO1xuXG4gICAgICBpZiAocmVsT3B0cy5jYXJkaW5hbGl0eSA9PT0gJzFtJyB8fCByZWxPcHRzLmNhcmRpbmFsaXR5ID09PSAnbW0nKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBpc0ZpcnN0TGV2ZWwgPSBwcmVmaXggPT09IHRhYmxlTmFtZTtcbiAgICAgIGNvbnN0IGpvaW5SZWxBbGlhcyA9IGlzRmlyc3RMZXZlbCA/IHJlbEtleSA6IHByZWZpeCA/IHByZWZpeCArICcuJyArIHJlbEtleSA6IHJlbEtleTtcbiAgICAgIGNvbnN0IHJlbEVudGl0eSA9IHJlbE9wdHMuZW50aXR5KCk7XG4gICAgICBjb25zdCByZWxTZWxlY3QgPSBzZWxlY3RbcmVsS2V5IGFzIHN0cmluZ107XG4gICAgICBjb25zdCByZWxRdWVyeSA9IGlzU2VsZWN0QXJyYXkgPyB7fSA6IEFycmF5LmlzQXJyYXkocmVsU2VsZWN0KSA/IHsgJHNlbGVjdDogcmVsU2VsZWN0IH0gOiByZWxTZWxlY3Q7XG5cbiAgICAgIGN0eC5hcHBlbmQoJywgJyk7XG4gICAgICB0aGlzLnNlbGVjdEZpZWxkcyhjdHgsIHJlbEVudGl0eSwgcmVsUXVlcnkuJHNlbGVjdCwge1xuICAgICAgICBwcmVmaXg6IGpvaW5SZWxBbGlhcyxcbiAgICAgICAgYXV0b1ByZWZpeEFsaWFzOiB0cnVlLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIFJlY3Vyc2l2ZWx5IGFkZCBuZXN0ZWQgcmVsYXRpb24gZmllbGRzXG4gICAgICB0aGlzLnNlbGVjdFJlbGF0aW9uRmllbGRzKGN0eCwgcmVsRW50aXR5LCByZWxRdWVyeS4kc2VsZWN0LCB7XG4gICAgICAgIHByZWZpeDogam9pblJlbEFsaWFzLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIHNlbGVjdFJlbGF0aW9uSm9pbnM8RT4oXG4gICAgY3R4OiBRdWVyeUNvbnRleHQsXG4gICAgZW50aXR5OiBUeXBlPEU+LFxuICAgIHNlbGVjdDogUXVlcnlTZWxlY3Q8RT4sXG4gICAgb3B0czogeyBwcmVmaXg/OiBzdHJpbmcgfSA9IHt9LFxuICApOiB2b2lkIHtcbiAgICBpZiAoQXJyYXkuaXNBcnJheShzZWxlY3QpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgbWV0YSA9IGdldE1ldGEoZW50aXR5KTtcbiAgICBjb25zdCB0YWJsZU5hbWUgPSB0aGlzLnJlc29sdmVUYWJsZU5hbWUoZW50aXR5LCBtZXRhKTtcbiAgICBjb25zdCByZWxLZXlzID0gZmlsdGVyUmVsYXRpb25LZXlzKG1ldGEsIHNlbGVjdCk7XG4gICAgY29uc3QgaXNTZWxlY3RBcnJheSA9IEFycmF5LmlzQXJyYXkoc2VsZWN0KTtcbiAgICBjb25zdCBwcmVmaXggPSBvcHRzLnByZWZpeDtcblxuICAgIGZvciAoY29uc3QgcmVsS2V5IG9mIHJlbEtleXMpIHtcbiAgICAgIGNvbnN0IHJlbE9wdHMgPSBtZXRhLnJlbGF0aW9uc1tyZWxLZXldO1xuXG4gICAgICBpZiAocmVsT3B0cy5jYXJkaW5hbGl0eSA9PT0gJzFtJyB8fCByZWxPcHRzLmNhcmRpbmFsaXR5ID09PSAnbW0nKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBpc0ZpcnN0TGV2ZWwgPSBwcmVmaXggPT09IHRhYmxlTmFtZTtcbiAgICAgIGNvbnN0IGpvaW5SZWxBbGlhcyA9IGlzRmlyc3RMZXZlbCA/IChyZWxLZXkgYXMgc3RyaW5nKSA6IHByZWZpeCA/IHByZWZpeCArICcuJyArIHJlbEtleSA6IChyZWxLZXkgYXMgc3RyaW5nKTtcbiAgICAgIGNvbnN0IHJlbEVudGl0eSA9IHJlbE9wdHMuZW50aXR5KCk7XG4gICAgICBjb25zdCByZWxTZWxlY3QgPSBzZWxlY3RbcmVsS2V5IGFzIHN0cmluZ107XG4gICAgICBjb25zdCByZWxRdWVyeSA9IGlzU2VsZWN0QXJyYXkgPyB7fSA6IEFycmF5LmlzQXJyYXkocmVsU2VsZWN0KSA/IHsgJHNlbGVjdDogcmVsU2VsZWN0IH0gOiByZWxTZWxlY3Q7XG5cbiAgICAgIGNvbnN0IHJlbE1ldGEgPSBnZXRNZXRhKHJlbEVudGl0eSk7XG4gICAgICBjb25zdCByZWxUYWJsZU5hbWUgPSB0aGlzLnJlc29sdmVUYWJsZU5hbWUocmVsRW50aXR5LCByZWxNZXRhKTtcbiAgICAgIGNvbnN0IHJlbEVudGl0eU5hbWUgPSB0aGlzLmVzY2FwZUlkKHJlbFRhYmxlTmFtZSk7XG4gICAgICBjb25zdCByZWxQYXRoID0gcHJlZml4ID8gdGhpcy5lc2NhcGVJZChwcmVmaXgsIHRydWUpIDogdGhpcy5lc2NhcGVJZCh0YWJsZU5hbWUpO1xuICAgICAgY29uc3Qgam9pblR5cGUgPSByZWxRdWVyeS4kcmVxdWlyZWQgPyAnSU5ORVInIDogJ0xFRlQnO1xuICAgICAgY29uc3Qgam9pbkFsaWFzID0gdGhpcy5lc2NhcGVJZChqb2luUmVsQWxpYXMsIHRydWUpO1xuXG4gICAgICBjdHguYXBwZW5kKGAgJHtqb2luVHlwZX0gSk9JTiAke3JlbEVudGl0eU5hbWV9ICR7am9pbkFsaWFzfSBPTiBgKTtcbiAgICAgIGN0eC5hcHBlbmQoXG4gICAgICAgIHJlbE9wdHMucmVmZXJlbmNlc1xuICAgICAgICAgIC5tYXAoKGl0KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBmb3JlaWduQ29sdW1uTmFtZSA9IHRoaXMucmVzb2x2ZUNvbHVtbk5hbWUoaXQuZm9yZWlnbiwgcmVsTWV0YS5maWVsZHNbaXQuZm9yZWlnbl0pO1xuICAgICAgICAgICAgY29uc3QgbG9jYWxDb2x1bW5OYW1lID0gdGhpcy5yZXNvbHZlQ29sdW1uTmFtZShpdC5sb2NhbCwgbWV0YS5maWVsZHNbaXQubG9jYWxdKTtcbiAgICAgICAgICAgIHJldHVybiBgJHtqb2luQWxpYXN9LiR7dGhpcy5lc2NhcGVJZChmb3JlaWduQ29sdW1uTmFtZSl9ID0gJHtyZWxQYXRofS4ke3RoaXMuZXNjYXBlSWQobG9jYWxDb2x1bW5OYW1lKX1gO1xuICAgICAgICAgIH0pXG4gICAgICAgICAgLmpvaW4oJyBBTkQgJyksXG4gICAgICApO1xuXG4gICAgICBpZiAocmVsUXVlcnkuJHdoZXJlKSB7XG4gICAgICAgIGN0eC5hcHBlbmQoJyBBTkQgJyk7XG4gICAgICAgIHRoaXMud2hlcmUoY3R4LCByZWxFbnRpdHksIHJlbFF1ZXJ5LiR3aGVyZSwgeyBwcmVmaXg6IGpvaW5SZWxBbGlhcywgY2xhdXNlOiBmYWxzZSB9KTtcbiAgICAgIH1cblxuICAgICAgLy8gUmVjdXJzaXZlbHkgYWRkIG5lc3RlZCByZWxhdGlvbiBKT0lOc1xuICAgICAgdGhpcy5zZWxlY3RSZWxhdGlvbkpvaW5zKGN0eCwgcmVsRW50aXR5LCByZWxRdWVyeS4kc2VsZWN0LCB7XG4gICAgICAgIHByZWZpeDogam9pblJlbEFsaWFzLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgd2hlcmU8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGVudGl0eTogVHlwZTxFPiwgd2hlcmU6IFF1ZXJ5V2hlcmU8RT4gPSB7fSwgb3B0czogUXVlcnlXaGVyZU9wdGlvbnMgPSB7fSk6IHZvaWQge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG4gICAgY29uc3QgeyB1c2VQcmVjZWRlbmNlLCBjbGF1c2UgPSAnV0hFUkUnLCBzb2Z0RGVsZXRlIH0gPSBvcHRzO1xuXG4gICAgd2hlcmUgPSBidWxkUXVlcnlXaGVyZUFzTWFwKG1ldGEsIHdoZXJlKTtcblxuICAgIGlmIChtZXRhLnNvZnREZWxldGUgJiYgKHNvZnREZWxldGUgfHwgc29mdERlbGV0ZSA9PT0gdW5kZWZpbmVkKSAmJiAhd2hlcmVbbWV0YS5zb2Z0RGVsZXRlIGFzIHN0cmluZ10pIHtcbiAgICAgIHdoZXJlW21ldGEuc29mdERlbGV0ZSBhcyBzdHJpbmddID0gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBlbnRyaWVzID0gT2JqZWN0LmVudHJpZXMod2hlcmUpO1xuXG4gICAgaWYgKCFlbnRyaWVzLmxlbmd0aCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChjbGF1c2UpIHtcbiAgICAgIGN0eC5hcHBlbmQoYCAke2NsYXVzZX0gYCk7XG4gICAgfVxuXG4gICAgaWYgKHVzZVByZWNlZGVuY2UpIHtcbiAgICAgIGN0eC5hcHBlbmQoJygnKTtcbiAgICB9XG5cbiAgICBlbnRyaWVzLmZvckVhY2goKFtrZXksIHZhbF0sIGluZGV4KSA9PiB7XG4gICAgICBpZiAoaW5kZXggPiAwKSB7XG4gICAgICAgIGN0eC5hcHBlbmQoJyBBTkQgJyk7XG4gICAgICB9XG4gICAgICB0aGlzLmNvbXBhcmUoY3R4LCBlbnRpdHksIGtleSBhcyBrZXlvZiBRdWVyeVdoZXJlTWFwPEU+LCB2YWwgYXMgYW55LCB7XG4gICAgICAgIC4uLm9wdHMsXG4gICAgICAgIHVzZVByZWNlZGVuY2U6IGVudHJpZXMubGVuZ3RoID4gMSxcbiAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAgaWYgKHVzZVByZWNlZGVuY2UpIHtcbiAgICAgIGN0eC5hcHBlbmQoJyknKTtcbiAgICB9XG4gIH1cblxuICBjb21wYXJlPEUsIEsgZXh0ZW5kcyBrZXlvZiBRdWVyeVdoZXJlTWFwPEU+PihcbiAgICBjdHg6IFF1ZXJ5Q29udGV4dCxcbiAgICBlbnRpdHk6IFR5cGU8RT4sXG4gICAga2V5OiBLLFxuICAgIHZhbDogUXVlcnlXaGVyZU1hcDxFPltLXSxcbiAgICBvcHRzOiBRdWVyeUNvbXBhcmlzb25PcHRpb25zID0ge30sXG4gICk6IHZvaWQge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG5cbiAgICBpZiAodmFsIGluc3RhbmNlb2YgUXVlcnlSYXcpIHtcbiAgICAgIGlmIChrZXkgPT09ICckZXhpc3RzJyB8fCBrZXkgPT09ICckbmV4aXN0cycpIHtcbiAgICAgICAgY3R4LmFwcGVuZChrZXkgPT09ICckZXhpc3RzJyA/ICdFWElTVFMgKCcgOiAnTk9UIEVYSVNUUyAoJyk7XG4gICAgICAgIGNvbnN0IHRhYmxlTmFtZSA9IHRoaXMucmVzb2x2ZVRhYmxlTmFtZShlbnRpdHksIG1ldGEpO1xuICAgICAgICB0aGlzLmdldFJhd1ZhbHVlKGN0eCwge1xuICAgICAgICAgIHZhbHVlOiB2YWwsXG4gICAgICAgICAgcHJlZml4OiB0YWJsZU5hbWUsXG4gICAgICAgICAgZXNjYXBlZFByZWZpeDogdGhpcy5lc2NhcGVJZCh0YWJsZU5hbWUsIGZhbHNlLCB0cnVlKSxcbiAgICAgICAgfSk7XG4gICAgICAgIGN0eC5hcHBlbmQoJyknKTtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXkgYXMgRmllbGRLZXk8RT4sIG9wdHMpO1xuICAgICAgY3R4LmFwcGVuZCgnID0gJyk7XG4gICAgICB0aGlzLmdldFJhd1ZhbHVlKGN0eCwgeyB2YWx1ZTogdmFsIH0pO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChrZXkgPT09ICckdGV4dCcpIHtcbiAgICAgIGNvbnN0IHNlYXJjaCA9IHZhbCBhcyBRdWVyeVRleHRTZWFyY2hPcHRpb25zPEU+O1xuICAgICAgY29uc3QgZmllbGRzID0gc2VhcmNoLiRmaWVsZHMubWFwKChmS2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGZpZWxkID0gbWV0YS5maWVsZHNbZktleV07XG4gICAgICAgIGNvbnN0IGNvbHVtbk5hbWUgPSB0aGlzLnJlc29sdmVDb2x1bW5OYW1lKGZLZXksIGZpZWxkKTtcbiAgICAgICAgcmV0dXJuIHRoaXMuZXNjYXBlSWQoY29sdW1uTmFtZSk7XG4gICAgICB9KTtcbiAgICAgIGN0eC5hcHBlbmQoYE1BVENIKCR7ZmllbGRzLmpvaW4oJywgJyl9KSBBR0FJTlNUKGApO1xuICAgICAgY3R4LmFkZFZhbHVlKHNlYXJjaC4kdmFsdWUpO1xuICAgICAgY3R4LmFwcGVuZCgnKScpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChrZXkgPT09ICckYW5kJyB8fCBrZXkgPT09ICckb3InIHx8IGtleSA9PT0gJyRub3QnIHx8IGtleSA9PT0gJyRub3InKSB7XG4gICAgICBjb25zdCBuZWdhdGVPcGVyYXRvck1hcCA9IHtcbiAgICAgICAgJG5vdDogJyRhbmQnLFxuICAgICAgICAkbm9yOiAnJG9yJyxcbiAgICAgIH0gYXMgY29uc3Q7XG5cbiAgICAgIGNvbnN0IG9wOiAnJGFuZCcgfCAnJG9yJyA9IG5lZ2F0ZU9wZXJhdG9yTWFwW2tleSBhcyBzdHJpbmddID8/IGtleTtcbiAgICAgIGNvbnN0IG5lZ2F0ZSA9IGtleSBpbiBuZWdhdGVPcGVyYXRvck1hcCA/ICdOT1QnIDogJyc7XG5cbiAgICAgIGNvbnN0IHZhbEFyciA9IHZhbCBhcyBRdWVyeVdoZXJlQXJyYXk8RT47XG4gICAgICBjb25zdCBoYXNNYW55SXRlbXMgPSB2YWxBcnIubGVuZ3RoID4gMTtcblxuICAgICAgaWYgKChvcHRzLnVzZVByZWNlZGVuY2UgfHwgbmVnYXRlKSAmJiBoYXNNYW55SXRlbXMpIHtcbiAgICAgICAgY3R4LmFwcGVuZCgobmVnYXRlID8gbmVnYXRlICsgJyAnIDogJycpICsgJygnKTtcbiAgICAgIH0gZWxzZSBpZiAobmVnYXRlKSB7XG4gICAgICAgIGN0eC5hcHBlbmQobmVnYXRlICsgJyAnKTtcbiAgICAgIH1cblxuICAgICAgdmFsQXJyLmZvckVhY2goKHdoZXJlRW50cnksIGluZGV4KSA9PiB7XG4gICAgICAgIGlmIChpbmRleCA+IDApIHtcbiAgICAgICAgICBjdHguYXBwZW5kKG9wID09PSAnJG9yJyA/ICcgT1IgJyA6ICcgQU5EICcpO1xuICAgICAgICB9XG4gICAgICAgIGlmICh3aGVyZUVudHJ5IGluc3RhbmNlb2YgUXVlcnlSYXcpIHtcbiAgICAgICAgICB0aGlzLmdldFJhd1ZhbHVlKGN0eCwge1xuICAgICAgICAgICAgdmFsdWU6IHdoZXJlRW50cnksXG4gICAgICAgICAgICBwcmVmaXg6IG9wdHMucHJlZml4LFxuICAgICAgICAgICAgZXNjYXBlZFByZWZpeDogdGhpcy5lc2NhcGVJZChvcHRzLnByZWZpeCwgdHJ1ZSwgdHJ1ZSksXG4gICAgICAgICAgfSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy53aGVyZShjdHgsIGVudGl0eSwgd2hlcmVFbnRyeSwge1xuICAgICAgICAgICAgcHJlZml4OiBvcHRzLnByZWZpeCxcbiAgICAgICAgICAgIHVzZVByZWNlZGVuY2U6IGhhc01hbnlJdGVtcyAmJiAhQXJyYXkuaXNBcnJheSh3aGVyZUVudHJ5KSAmJiBnZXRLZXlzKHdoZXJlRW50cnkpLmxlbmd0aCA+IDEsXG4gICAgICAgICAgICBjbGF1c2U6IGZhbHNlLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgaWYgKChvcHRzLnVzZVByZWNlZGVuY2UgfHwgbmVnYXRlKSAmJiBoYXNNYW55SXRlbXMpIHtcbiAgICAgICAgY3R4LmFwcGVuZCgnKScpO1xuICAgICAgfVxuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHZhbHVlID0gQXJyYXkuaXNBcnJheSh2YWwpID8geyAkaW46IHZhbCB9IDogdHlwZW9mIHZhbCA9PT0gJ29iamVjdCcgJiYgdmFsICE9PSBudWxsID8gdmFsIDogeyAkZXE6IHZhbCB9O1xuICAgIGNvbnN0IG9wZXJhdG9ycyA9IGdldEtleXModmFsdWUpIGFzIChrZXlvZiBRdWVyeVdoZXJlRmllbGRPcGVyYXRvck1hcDxFPilbXTtcblxuICAgIGlmIChvcGVyYXRvcnMubGVuZ3RoID4gMSkge1xuICAgICAgY3R4LmFwcGVuZCgnKCcpO1xuICAgIH1cblxuICAgIG9wZXJhdG9ycy5mb3JFYWNoKChvcCwgaW5kZXgpID0+IHtcbiAgICAgIGlmIChpbmRleCA+IDApIHtcbiAgICAgICAgY3R4LmFwcGVuZCgnIEFORCAnKTtcbiAgICAgIH1cbiAgICAgIHRoaXMuY29tcGFyZUZpZWxkT3BlcmF0b3IoY3R4LCBlbnRpdHksIGtleSBhcyBGaWVsZEtleTxFPiwgb3AsIHZhbHVlW29wXSwgb3B0cyk7XG4gICAgfSk7XG5cbiAgICBpZiAob3BlcmF0b3JzLmxlbmd0aCA+IDEpIHtcbiAgICAgIGN0eC5hcHBlbmQoJyknKTtcbiAgICB9XG4gIH1cblxuICBjb21wYXJlRmllbGRPcGVyYXRvcjxFLCBLIGV4dGVuZHMga2V5b2YgUXVlcnlXaGVyZUZpZWxkT3BlcmF0b3JNYXA8RT4+KFxuICAgIGN0eDogUXVlcnlDb250ZXh0LFxuICAgIGVudGl0eTogVHlwZTxFPixcbiAgICBrZXk6IEZpZWxkS2V5PEU+LFxuICAgIG9wOiBLLFxuICAgIHZhbDogUXVlcnlXaGVyZUZpZWxkT3BlcmF0b3JNYXA8RT5bS10sXG4gICAgb3B0czogUXVlcnlPcHRpb25zID0ge30sXG4gICk6IHZvaWQge1xuICAgIHN3aXRjaCAob3ApIHtcbiAgICAgIGNhc2UgJyRlcSc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgaWYgKHZhbCA9PT0gbnVsbCkge1xuICAgICAgICAgIGN0eC5hcHBlbmQoJyBJUyBOVUxMJyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY3R4LmFwcGVuZCgnID0gJyk7XG4gICAgICAgICAgY3R4LmFkZFZhbHVlKHZhbCk7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICckbmUnOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGlmICh2YWwgPT09IG51bGwpIHtcbiAgICAgICAgICBjdHguYXBwZW5kKCcgSVMgTk9UIE5VTEwnKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjdHguYXBwZW5kKCcgPD4gJyk7XG4gICAgICAgICAgY3R4LmFkZFZhbHVlKHZhbCk7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICckbm90JzpcbiAgICAgICAgY3R4LmFwcGVuZCgnTk9UICgnKTtcbiAgICAgICAgdGhpcy5jb21wYXJlKGN0eCwgZW50aXR5LCBrZXkgYXMgYW55LCB2YWwgYXMgYW55LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnKScpO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRndCc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnID4gJyk7XG4gICAgICAgIGN0eC5hZGRWYWx1ZSh2YWwpO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRndGUnOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGN0eC5hcHBlbmQoJyA+PSAnKTtcbiAgICAgICAgY3R4LmFkZFZhbHVlKHZhbCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGx0JzpcbiAgICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXksIG9wdHMpO1xuICAgICAgICBjdHguYXBwZW5kKCcgPCAnKTtcbiAgICAgICAgY3R4LmFkZFZhbHVlKHZhbCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGx0ZSc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnIDw9ICcpO1xuICAgICAgICBjdHguYWRkVmFsdWUodmFsKTtcbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICckc3RhcnRzV2l0aCc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnIExJS0UgJyk7XG4gICAgICAgIGN0eC5hZGRWYWx1ZShgJHt2YWx9JWApO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRpc3RhcnRzV2l0aCc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnIExJS0UgJyk7XG4gICAgICAgIGN0eC5hZGRWYWx1ZShgJHsodmFsIGFzIHN0cmluZykudG9Mb3dlckNhc2UoKX0lYCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGVuZHNXaXRoJzpcbiAgICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXksIG9wdHMpO1xuICAgICAgICBjdHguYXBwZW5kKCcgTElLRSAnKTtcbiAgICAgICAgY3R4LmFkZFZhbHVlKGAlJHt2YWx9YCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGllbmRzV2l0aCc6XG4gICAgICAgIHRoaXMuZ2V0Q29tcGFyaXNvbktleShjdHgsIGVudGl0eSwga2V5LCBvcHRzKTtcbiAgICAgICAgY3R4LmFwcGVuZCgnIExJS0UgJyk7XG4gICAgICAgIGN0eC5hZGRWYWx1ZShgJSR7KHZhbCBhcyBzdHJpbmcpLnRvTG93ZXJDYXNlKCl9YCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGluY2x1ZGVzJzpcbiAgICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXksIG9wdHMpO1xuICAgICAgICBjdHguYXBwZW5kKCcgTElLRSAnKTtcbiAgICAgICAgY3R4LmFkZFZhbHVlKGAlJHt2YWx9JWApO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRpaW5jbHVkZXMnOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGN0eC5hcHBlbmQoJyBMSUtFICcpO1xuICAgICAgICBjdHguYWRkVmFsdWUoYCUkeyh2YWwgYXMgc3RyaW5nKS50b0xvd2VyQ2FzZSgpfSVgKTtcbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICckaWxpa2UnOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGN0eC5hcHBlbmQoJyBMSUtFICcpO1xuICAgICAgICBjdHguYWRkVmFsdWUoKHZhbCBhcyBzdHJpbmcpLnRvTG93ZXJDYXNlKCkpO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRsaWtlJzpcbiAgICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXksIG9wdHMpO1xuICAgICAgICBjdHguYXBwZW5kKCcgTElLRSAnKTtcbiAgICAgICAgY3R4LmFkZFZhbHVlKHZhbCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnJGluJzpcbiAgICAgICAgdGhpcy5nZXRDb21wYXJpc29uS2V5KGN0eCwgZW50aXR5LCBrZXksIG9wdHMpO1xuICAgICAgICBpZiAoQXJyYXkuaXNBcnJheSh2YWwpICYmIHZhbC5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY3R4LmFwcGVuZCgnIElOICgnKTtcbiAgICAgICAgICB0aGlzLmFkZFZhbHVlcyhjdHgsIHZhbCBhcyBhbnlbXSk7XG4gICAgICAgICAgY3R4LmFwcGVuZCgnKScpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGN0eC5hcHBlbmQoJyBJTiAoTlVMTCknKTtcbiAgICAgICAgfVxuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJyRuaW4nOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGlmIChBcnJheS5pc0FycmF5KHZhbCkgJiYgdmFsLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICBjdHguYXBwZW5kKCcgTk9UIElOICgnKTtcbiAgICAgICAgICB0aGlzLmFkZFZhbHVlcyhjdHgsIHZhbCBhcyBhbnlbXSk7XG4gICAgICAgICAgY3R4LmFwcGVuZCgnKScpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGN0eC5hcHBlbmQoJyBOT1QgSU4gKE5VTEwpJyk7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICckcmVnZXgnOlxuICAgICAgICB0aGlzLmdldENvbXBhcmlzb25LZXkoY3R4LCBlbnRpdHksIGtleSwgb3B0cyk7XG4gICAgICAgIGN0eC5hcHBlbmQoJyBSRUdFWFAgJyk7XG4gICAgICAgIGN0eC5hZGRWYWx1ZSh2YWwpO1xuICAgICAgICBicmVhaztcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHRocm93IFR5cGVFcnJvcihgdW5rbm93biBvcGVyYXRvcjogJHtvcH1gKTtcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgYWRkVmFsdWVzKGN0eDogUXVlcnlDb250ZXh0LCB2YWxzOiB1bmtub3duW10pOiB2b2lkIHtcbiAgICB2YWxzLmZvckVhY2goKHZhbCwgaW5kZXgpID0+IHtcbiAgICAgIGlmIChpbmRleCA+IDApIHtcbiAgICAgICAgY3R4LmFwcGVuZCgnLCAnKTtcbiAgICAgIH1cbiAgICAgIGN0eC5hZGRWYWx1ZSh2YWwpO1xuICAgIH0pO1xuICB9XG5cbiAgZ2V0Q29tcGFyaXNvbktleTxFPihjdHg6IFF1ZXJ5Q29udGV4dCwgZW50aXR5OiBUeXBlPEU+LCBrZXk6IEZpZWxkS2V5PEU+LCB7IHByZWZpeCB9OiBRdWVyeU9wdGlvbnMgPSB7fSk6IHZvaWQge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG4gICAgY29uc3QgZXNjYXBlZFByZWZpeCA9IHRoaXMuZXNjYXBlSWQocHJlZml4LCB0cnVlLCB0cnVlKTtcbiAgICBjb25zdCBmaWVsZCA9IG1ldGEuZmllbGRzW2tleV07XG5cbiAgICBpZiAoZmllbGQ/LnZpcnR1YWwpIHtcbiAgICAgIHRoaXMuZ2V0UmF3VmFsdWUoY3R4LCB7XG4gICAgICAgIHZhbHVlOiBmaWVsZC52aXJ0dWFsLFxuICAgICAgICBwcmVmaXgsXG4gICAgICAgIGVzY2FwZWRQcmVmaXgsXG4gICAgICB9KTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBjb2x1bW5OYW1lID0gdGhpcy5yZXNvbHZlQ29sdW1uTmFtZShrZXksIGZpZWxkKTtcbiAgICBjdHguYXBwZW5kKGVzY2FwZWRQcmVmaXggKyB0aGlzLmVzY2FwZUlkKGNvbHVtbk5hbWUpKTtcbiAgfVxuXG4gIHNvcnQ8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGVudGl0eTogVHlwZTxFPiwgc29ydDogUXVlcnlTb3J0PEU+LCB7IHByZWZpeCB9OiBRdWVyeU9wdGlvbnMpOiB2b2lkIHtcbiAgICBjb25zdCBzb3J0TWFwID0gYnVpbGRTb3J0TWFwKHNvcnQpO1xuICAgIGlmICghaGFzS2V5cyhzb3J0TWFwKSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCBtZXRhID0gZ2V0TWV0YShlbnRpdHkpO1xuICAgIGNvbnN0IGZsYXR0ZW5lZFNvcnQgPSBmbGF0T2JqZWN0KHNvcnRNYXAsIHByZWZpeCk7XG4gICAgY29uc3QgZGlyZWN0aW9uTWFwID0geyAxOiAnJywgYXNjOiAnJywgJy0xJzogJyBERVNDJywgZGVzYzogJyBERVNDJyB9IGFzIGNvbnN0O1xuXG4gICAgY3R4LmFwcGVuZCgnIE9SREVSIEJZICcpO1xuXG4gICAgT2JqZWN0LmVudHJpZXMoZmxhdHRlbmVkU29ydCkuZm9yRWFjaCgoW2tleSwgc29ydF0sIGluZGV4KSA9PiB7XG4gICAgICBpZiAoaW5kZXggPiAwKSB7XG4gICAgICAgIGN0eC5hcHBlbmQoJywgJyk7XG4gICAgICB9XG4gICAgICBjb25zdCBmaWVsZCA9IG1ldGEuZmllbGRzW2tleV07XG4gICAgICBjb25zdCBuYW1lID0gdGhpcy5yZXNvbHZlQ29sdW1uTmFtZShrZXksIGZpZWxkKTtcbiAgICAgIGNvbnN0IGRpcmVjdGlvbiA9IGRpcmVjdGlvbk1hcFtzb3J0IGFzIFF1ZXJ5U29ydERpcmVjdGlvbl07XG4gICAgICBjdHguYXBwZW5kKHRoaXMuZXNjYXBlSWQobmFtZSkgKyBkaXJlY3Rpb24pO1xuICAgIH0pO1xuICB9XG5cbiAgcGFnZXIoY3R4OiBRdWVyeUNvbnRleHQsIG9wdHM6IFF1ZXJ5UGFnZXIpOiB2b2lkIHtcbiAgICBpZiAob3B0cy4kbGltaXQpIHtcbiAgICAgIGN0eC5hcHBlbmQoYCBMSU1JVCAke051bWJlcihvcHRzLiRsaW1pdCl9YCk7XG4gICAgfVxuICAgIGlmIChvcHRzLiRza2lwICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIGN0eC5hcHBlbmQoYCBPRkZTRVQgJHtOdW1iZXIob3B0cy4kc2tpcCl9YCk7XG4gICAgfVxuICB9XG5cbiAgY291bnQ8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGVudGl0eTogVHlwZTxFPiwgcTogUXVlcnlTZWFyY2g8RT4sIG9wdHM/OiBRdWVyeU9wdGlvbnMpOiB2b2lkIHtcbiAgICBjb25zdCBzZWFyY2g6IFF1ZXJ5PEU+ID0geyAuLi5xIH07XG4gICAgZGVsZXRlIHNlYXJjaC4kc29ydDtcbiAgICB0aGlzLnNlbGVjdDxFPihjdHgsIGVudGl0eSwgW3JhdygnQ09VTlQoKiknLCAnY291bnQnKV0sIHVuZGVmaW5lZCk7XG4gICAgdGhpcy5zZWFyY2goY3R4LCBlbnRpdHksIHNlYXJjaCwgb3B0cyk7XG4gIH1cblxuICBmaW5kPEU+KGN0eDogUXVlcnlDb250ZXh0LCBlbnRpdHk6IFR5cGU8RT4sIHE6IFF1ZXJ5PEU+ID0ge30sIG9wdHM/OiBRdWVyeU9wdGlvbnMpOiB2b2lkIHtcbiAgICB0aGlzLnNlbGVjdChjdHgsIGVudGl0eSwgcS4kc2VsZWN0LCBvcHRzKTtcbiAgICB0aGlzLnNlYXJjaChjdHgsIGVudGl0eSwgcSwgb3B0cyk7XG4gIH1cblxuICBpbnNlcnQ8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGVudGl0eTogVHlwZTxFPiwgcGF5bG9hZDogRSB8IEVbXSwgb3B0cz86IFF1ZXJ5T3B0aW9ucyk6IHZvaWQge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG4gICAgY29uc3QgcGF5bG9hZHMgPSBmaWxsT25GaWVsZHMobWV0YSwgcGF5bG9hZCwgJ29uSW5zZXJ0Jyk7XG4gICAgY29uc3Qga2V5cyA9IGZpbHRlckZpZWxkS2V5cyhtZXRhLCBwYXlsb2Fkc1swXSwgJ29uSW5zZXJ0Jyk7XG5cbiAgICBjb25zdCBjb2x1bW5zID0ga2V5cy5tYXAoKGtleSkgPT4ge1xuICAgICAgY29uc3QgZmllbGQgPSBtZXRhLmZpZWxkc1trZXldO1xuICAgICAgcmV0dXJuIHRoaXMuZXNjYXBlSWQodGhpcy5yZXNvbHZlQ29sdW1uTmFtZShrZXksIGZpZWxkKSk7XG4gICAgfSk7XG4gICAgY29uc3QgdGFibGVOYW1lID0gdGhpcy5yZXNvbHZlVGFibGVOYW1lKGVudGl0eSwgbWV0YSk7XG4gICAgY3R4LmFwcGVuZChgSU5TRVJUIElOVE8gJHt0aGlzLmVzY2FwZUlkKHRhYmxlTmFtZSl9ICgke2NvbHVtbnMuam9pbignLCAnKX0pIFZBTFVFUyAoYCk7XG5cbiAgICBwYXlsb2Fkcy5mb3JFYWNoKChpdCwgcmVjb3JkSW5kZXgpID0+IHtcbiAgICAgIGlmIChyZWNvcmRJbmRleCA+IDApIHtcbiAgICAgICAgY3R4LmFwcGVuZCgnKSwgKCcpO1xuICAgICAgfVxuICAgICAga2V5cy5mb3JFYWNoKChrZXksIGtleUluZGV4KSA9PiB7XG4gICAgICAgIGlmIChrZXlJbmRleCA+IDApIHtcbiAgICAgICAgICBjdHguYXBwZW5kKCcsICcpO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IGZpZWxkID0gbWV0YS5maWVsZHNba2V5XTtcbiAgICAgICAgdGhpcy5mb3JtYXRQZXJzaXN0YWJsZVZhbHVlKGN0eCwgZmllbGQsIGl0W2tleV0pO1xuICAgICAgfSk7XG4gICAgfSk7XG4gICAgY3R4LmFwcGVuZCgnKScpO1xuICB9XG5cbiAgdXBkYXRlPEU+KGN0eDogUXVlcnlDb250ZXh0LCBlbnRpdHk6IFR5cGU8RT4sIHE6IFF1ZXJ5U2VhcmNoPEU+LCBwYXlsb2FkOiBFLCBvcHRzPzogUXVlcnlPcHRpb25zKTogdm9pZCB7XG4gICAgY29uc3QgbWV0YSA9IGdldE1ldGEoZW50aXR5KTtcbiAgICBjb25zdCBbZmlsbGVkUGF5bG9hZF0gPSBmaWxsT25GaWVsZHMobWV0YSwgcGF5bG9hZCwgJ29uVXBkYXRlJyk7XG4gICAgY29uc3Qga2V5cyA9IGZpbHRlckZpZWxkS2V5cyhtZXRhLCBmaWxsZWRQYXlsb2FkLCAnb25VcGRhdGUnKTtcblxuICAgIGNvbnN0IHRhYmxlTmFtZSA9IHRoaXMucmVzb2x2ZVRhYmxlTmFtZShlbnRpdHksIG1ldGEpO1xuICAgIGN0eC5hcHBlbmQoYFVQREFURSAke3RoaXMuZXNjYXBlSWQodGFibGVOYW1lKX0gU0VUIGApO1xuICAgIGtleXMuZm9yRWFjaCgoa2V5LCBpbmRleCkgPT4ge1xuICAgICAgaWYgKGluZGV4ID4gMCkge1xuICAgICAgICBjdHguYXBwZW5kKCcsICcpO1xuICAgICAgfVxuICAgICAgY29uc3QgZmllbGQgPSBtZXRhLmZpZWxkc1trZXldO1xuICAgICAgY29uc3QgY29sdW1uTmFtZSA9IHRoaXMucmVzb2x2ZUNvbHVtbk5hbWUoa2V5LCBmaWVsZCk7XG4gICAgICBjdHguYXBwZW5kKGAke3RoaXMuZXNjYXBlSWQoY29sdW1uTmFtZSl9ID0gYCk7XG4gICAgICB0aGlzLmZvcm1hdFBlcnNpc3RhYmxlVmFsdWUoY3R4LCBmaWVsZCwgZmlsbGVkUGF5bG9hZFtrZXldKTtcbiAgICB9KTtcblxuICAgIHRoaXMuc2VhcmNoKGN0eCwgZW50aXR5LCBxLCBvcHRzKTtcbiAgfVxuXG4gIHVwc2VydDxFPihjdHg6IFF1ZXJ5Q29udGV4dCwgZW50aXR5OiBUeXBlPEU+LCBjb25mbGljdFBhdGhzOiBRdWVyeUNvbmZsaWN0UGF0aHM8RT4sIHBheWxvYWQ6IEUpOiB2b2lkIHtcbiAgICBjb25zdCBtZXRhID0gZ2V0TWV0YShlbnRpdHkpO1xuICAgIGNvbnN0IHVwZGF0ZSA9IHRoaXMuZ2V0VXBzZXJ0VXBkYXRlQXNzaWdubWVudHMoY3R4LCBtZXRhLCBjb25mbGljdFBhdGhzLCBwYXlsb2FkLCAobmFtZSkgPT4gYFZBTFVFUygke25hbWV9KWApO1xuXG4gICAgaWYgKHVwZGF0ZSkge1xuICAgICAgdGhpcy5pbnNlcnQoY3R4LCBlbnRpdHksIHBheWxvYWQpO1xuICAgICAgY3R4LmFwcGVuZChgIE9OIERVUExJQ0FURSBLRVkgVVBEQVRFICR7dXBkYXRlfWApO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBpbnNlcnRDdHggPSB0aGlzLmNyZWF0ZUNvbnRleHQoKTtcbiAgICAgIHRoaXMuaW5zZXJ0KGluc2VydEN0eCwgZW50aXR5LCBwYXlsb2FkKTtcbiAgICAgIGN0eC5hcHBlbmQoaW5zZXJ0Q3R4LnNxbC5yZXBsYWNlKC9eSU5TRVJULywgJ0lOU0VSVCBJR05PUkUnKSk7XG4gICAgICBpbnNlcnRDdHgudmFsdWVzLmZvckVhY2goKHZhbCkgPT4ge1xuICAgICAgICBjdHgucHVzaFZhbHVlKHZhbCk7XG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgZ2V0VXBzZXJ0VXBkYXRlQXNzaWdubWVudHM8RT4oXG4gICAgY3R4OiBRdWVyeUNvbnRleHQsXG4gICAgbWV0YTogRW50aXR5TWV0YTxFPixcbiAgICBjb25mbGljdFBhdGhzOiBRdWVyeUNvbmZsaWN0UGF0aHM8RT4sXG4gICAgcGF5bG9hZDogRSxcbiAgICBjYWxsYmFjaz86IChjb2x1bW5OYW1lOiBzdHJpbmcpID0+IHN0cmluZyxcbiAgKTogc3RyaW5nIHtcbiAgICBjb25zdCBbZmlsbGVkUGF5bG9hZF0gPSBmaWxsT25GaWVsZHMobWV0YSwgcGF5bG9hZCwgJ29uVXBkYXRlJyk7XG4gICAgY29uc3QgZmllbGRzID0gZmlsdGVyRmllbGRLZXlzKG1ldGEsIGZpbGxlZFBheWxvYWQsICdvblVwZGF0ZScpO1xuICAgIHJldHVybiBmaWVsZHNcbiAgICAgIC5maWx0ZXIoKGNvbCkgPT4gIWNvbmZsaWN0UGF0aHNbY29sXSlcbiAgICAgIC5tYXAoKGNvbCkgPT4ge1xuICAgICAgICBjb25zdCBmaWVsZCA9IG1ldGEuZmllbGRzW2NvbF07XG4gICAgICAgIGNvbnN0IGNvbHVtbk5hbWUgPSB0aGlzLnJlc29sdmVDb2x1bW5OYW1lKGNvbCwgZmllbGQpO1xuICAgICAgICBpZiAoY2FsbGJhY2spIHtcbiAgICAgICAgICByZXR1cm4gYCR7dGhpcy5lc2NhcGVJZChjb2x1bW5OYW1lKX0gPSAke2NhbGxiYWNrKHRoaXMuZXNjYXBlSWQoY29sdW1uTmFtZSkpfWA7XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgdmFsQ3R4ID0gdGhpcy5jcmVhdGVDb250ZXh0KCk7XG4gICAgICAgIHRoaXMuZm9ybWF0UGVyc2lzdGFibGVWYWx1ZSh2YWxDdHgsIGZpZWxkLCBmaWxsZWRQYXlsb2FkW2NvbF0pO1xuICAgICAgICB2YWxDdHgudmFsdWVzLmZvckVhY2goKHZhbCkgPT4ge1xuICAgICAgICAgIGN0eC5wdXNoVmFsdWUodmFsKTtcbiAgICAgICAgfSk7XG4gICAgICAgIHJldHVybiBgJHt0aGlzLmVzY2FwZUlkKGNvbHVtbk5hbWUpfSA9ICR7dmFsQ3R4LnNxbH1gO1xuICAgICAgfSlcbiAgICAgIC5qb2luKCcsICcpO1xuICB9XG5cbiAgcHJvdGVjdGVkIGdldFVwc2VydENvbmZsaWN0UGF0aHNTdHI8RT4obWV0YTogRW50aXR5TWV0YTxFPiwgY29uZmxpY3RQYXRoczogUXVlcnlDb25mbGljdFBhdGhzPEU+KTogc3RyaW5nIHtcbiAgICByZXR1cm4gZ2V0S2V5cyhjb25mbGljdFBhdGhzKVxuICAgICAgLm1hcCgoa2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGZpZWxkID0gbWV0YS5maWVsZHNba2V5XTtcbiAgICAgICAgY29uc3QgY29sdW1uTmFtZSA9IHRoaXMucmVzb2x2ZUNvbHVtbk5hbWUoa2V5LCBmaWVsZCk7XG4gICAgICAgIHJldHVybiB0aGlzLmVzY2FwZUlkKGNvbHVtbk5hbWUpO1xuICAgICAgfSlcbiAgICAgIC5qb2luKCcsICcpO1xuICB9XG5cbiAgZGVsZXRlPEU+KGN0eDogUXVlcnlDb250ZXh0LCBlbnRpdHk6IFR5cGU8RT4sIHE6IFF1ZXJ5U2VhcmNoPEU+LCBvcHRzOiBRdWVyeU9wdGlvbnMgPSB7fSk6IHZvaWQge1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNZXRhKGVudGl0eSk7XG4gICAgY29uc3QgdGFibGVOYW1lID0gdGhpcy5yZXNvbHZlVGFibGVOYW1lKGVudGl0eSwgbWV0YSk7XG5cbiAgICBpZiAob3B0cy5zb2Z0RGVsZXRlIHx8IG9wdHMuc29mdERlbGV0ZSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICBpZiAobWV0YS5zb2Z0RGVsZXRlKSB7XG4gICAgICAgIGNvbnN0IGZpZWxkID0gbWV0YS5maWVsZHNbbWV0YS5zb2Z0RGVsZXRlXTtcbiAgICAgICAgY29uc3QgdmFsdWUgPSBnZXRGaWVsZENhbGxiYWNrVmFsdWUoZmllbGQub25EZWxldGUpO1xuICAgICAgICBjb25zdCBjb2x1bW5OYW1lID0gdGhpcy5yZXNvbHZlQ29sdW1uTmFtZShtZXRhLnNvZnREZWxldGUsIGZpZWxkKTtcbiAgICAgICAgY3R4LmFwcGVuZChgVVBEQVRFICR7dGhpcy5lc2NhcGVJZCh0YWJsZU5hbWUpfSBTRVQgJHt0aGlzLmVzY2FwZUlkKGNvbHVtbk5hbWUpfSA9IGApO1xuICAgICAgICBjdHguYWRkVmFsdWUodmFsdWUpO1xuICAgICAgICB0aGlzLnNlYXJjaChjdHgsIGVudGl0eSwgcSwgb3B0cyk7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIGlmIChvcHRzLnNvZnREZWxldGUpIHtcbiAgICAgICAgdGhyb3cgVHlwZUVycm9yKGAnJHt0YWJsZU5hbWV9JyBoYXMgbm90IGVuYWJsZWQgJ3NvZnREZWxldGUnYCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgY3R4LmFwcGVuZChgREVMRVRFIEZST00gJHt0aGlzLmVzY2FwZUlkKHRhYmxlTmFtZSl9YCk7XG4gICAgdGhpcy5zZWFyY2goY3R4LCBlbnRpdHksIHEsIG9wdHMpO1xuICB9XG5cbiAgZXNjYXBlSWQodmFsOiBzdHJpbmcsIGZvcmJpZFF1YWxpZmllZD86IGJvb2xlYW4sIGFkZERvdD86IGJvb2xlYW4pOiBzdHJpbmcge1xuICAgIHJldHVybiBlc2NhcGVTcWxJZCh2YWwsIHRoaXMuZXNjYXBlSWRDaGFyLCBmb3JiaWRRdWFsaWZpZWQsIGFkZERvdCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgZ2V0UGVyc2lzdGFibGVzPEU+KFxuICAgIGN0eDogUXVlcnlDb250ZXh0LFxuICAgIG1ldGE6IEVudGl0eU1ldGE8RT4sXG4gICAgcGF5bG9hZDogRSB8IEVbXSxcbiAgICBjYWxsYmFja0tleTogQ2FsbGJhY2tLZXksXG4gICk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W10ge1xuICAgIGNvbnN0IHBheWxvYWRzID0gZmlsbE9uRmllbGRzKG1ldGEsIHBheWxvYWQsIGNhbGxiYWNrS2V5KTtcbiAgICByZXR1cm4gcGF5bG9hZHMubWFwKChpdCkgPT4gdGhpcy5nZXRQZXJzaXN0YWJsZShjdHgsIG1ldGEsIGl0LCBjYWxsYmFja0tleSkpO1xuICB9XG5cbiAgcHJvdGVjdGVkIGdldFBlcnNpc3RhYmxlPEU+KFxuICAgIGN0eDogUXVlcnlDb250ZXh0LFxuICAgIG1ldGE6IEVudGl0eU1ldGE8RT4sXG4gICAgcGF5bG9hZDogRSxcbiAgICBjYWxsYmFja0tleTogQ2FsbGJhY2tLZXksXG4gICk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+IHtcbiAgICBjb25zdCBmaWxsZWRQYXlsb2FkID0gZmlsbE9uRmllbGRzKG1ldGEsIHBheWxvYWQsIGNhbGxiYWNrS2V5KVswXTtcbiAgICBjb25zdCBrZXlzID0gZmlsdGVyRmllbGRLZXlzKG1ldGEsIGZpbGxlZFBheWxvYWQsIGNhbGxiYWNrS2V5KTtcbiAgICByZXR1cm4ga2V5cy5yZWR1Y2UoXG4gICAgICAoYWNjLCBrZXkpID0+IHtcbiAgICAgICAgY29uc3QgZmllbGQgPSBtZXRhLmZpZWxkc1trZXldO1xuICAgICAgICBjb25zdCB2YWxDdHggPSB0aGlzLmNyZWF0ZUNvbnRleHQoKTtcbiAgICAgICAgdGhpcy5mb3JtYXRQZXJzaXN0YWJsZVZhbHVlKHZhbEN0eCwgZmllbGQsIGZpbGxlZFBheWxvYWRba2V5XSk7XG4gICAgICAgIHZhbEN0eC52YWx1ZXMuZm9yRWFjaCgodmFsKSA9PiB7XG4gICAgICAgICAgY3R4LnB1c2hWYWx1ZSh2YWwpO1xuICAgICAgICB9KTtcbiAgICAgICAgYWNjW2tleSBhcyBzdHJpbmddID0gdmFsQ3R4LnNxbDtcbiAgICAgICAgcmV0dXJuIGFjYztcbiAgICAgIH0sXG4gICAgICB7fSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbiAgICApO1xuICB9XG5cbiAgcHJvdGVjdGVkIGZvcm1hdFBlcnNpc3RhYmxlVmFsdWU8RT4oY3R4OiBRdWVyeUNvbnRleHQsIGZpZWxkOiBGaWVsZE9wdGlvbnMsIHZhbHVlOiB1bmtub3duKTogdm9pZCB7XG4gICAgaWYgKHZhbHVlIGluc3RhbmNlb2YgUXVlcnlSYXcpIHtcbiAgICAgIHRoaXMuZ2V0UmF3VmFsdWUoY3R4LCB7IHZhbHVlIH0pO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBpZiAoZmllbGQ/LnR5cGUgPT09ICdqc29uJyB8fCBmaWVsZD8udHlwZSA9PT0gJ2pzb25iJykge1xuICAgICAgY3R4LmFkZFZhbHVlKHZhbHVlID8gSlNPTi5zdHJpbmdpZnkodmFsdWUpIDogbnVsbCk7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmIChmaWVsZD8udHlwZSA9PT0gJ3ZlY3RvcicgJiYgQXJyYXkuaXNBcnJheSh2YWx1ZSkpIHtcbiAgICAgIGN0eC5hZGRWYWx1ZShgWyR7dmFsdWUuam9pbignLCcpfV1gKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgY3R4LmFkZFZhbHVlKHZhbHVlKTtcbiAgfVxuXG4gIGdldFJhd1ZhbHVlKGN0eDogUXVlcnlDb250ZXh0LCBvcHRzOiBRdWVyeVJhd0ZuT3B0aW9ucyAmIHsgdmFsdWU6IFF1ZXJ5UmF3OyBhdXRvUHJlZml4QWxpYXM/OiBib29sZWFuIH0pIHtcbiAgICBjb25zdCB7IHZhbHVlLCBwcmVmaXggPSAnJywgZXNjYXBlZFByZWZpeCwgYXV0b1ByZWZpeEFsaWFzIH0gPSBvcHRzO1xuICAgIGlmICh0eXBlb2YgdmFsdWUudmFsdWUgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgIGNvbnN0IHJlcyA9IHZhbHVlLnZhbHVlKHtcbiAgICAgICAgLi4ub3B0cyxcbiAgICAgICAgY3R4LFxuICAgICAgICBkaWFsZWN0OiB0aGlzLFxuICAgICAgICBwcmVmaXgsXG4gICAgICAgIGVzY2FwZWRQcmVmaXg6IGVzY2FwZWRQcmVmaXggPz8gdGhpcy5lc2NhcGVJZChwcmVmaXgsIHRydWUsIHRydWUpLFxuICAgICAgfSk7XG4gICAgICBpZiAodHlwZW9mIHJlcyA9PT0gJ3N0cmluZycgfHwgKHR5cGVvZiByZXMgPT09ICdudW1iZXInICYmICFOdW1iZXIuaXNOYU4ocmVzKSkpIHtcbiAgICAgICAgY3R4LmFwcGVuZChTdHJpbmcocmVzKSk7XG4gICAgICB9XG4gICAgfSBlbHNlIHtcbiAgICAgIGN0eC5hcHBlbmQocHJlZml4ICsgU3RyaW5nKHZhbHVlLnZhbHVlKSk7XG4gICAgfVxuICAgIGNvbnN0IGFsaWFzID0gdmFsdWUuYWxpYXM7XG4gICAgaWYgKGFsaWFzKSB7XG4gICAgICBjb25zdCBmdWxsQWxpYXMgPSBhdXRvUHJlZml4QWxpYXMgPyBwcmVmaXggKyBhbGlhcyA6IGFsaWFzO1xuICAgICAgLy8gUmVwbGFjZSBkb3RzIHdpdGggdW5kZXJzY29yZXMgZm9yIGFsaWFzIHRvIGF2b2lkIHN5bnRheCBlcnJvcnNcbiAgICAgIGNvbnN0IHNhZmVBbGlhcyA9IGZ1bGxBbGlhcy5yZXBsYWNlKC9cXC4vZywgJ18nKTtcbiAgICAgIGNvbnN0IGVzY2FwZWRGdWxsQWxpYXMgPSB0aGlzLmVzY2FwZUlkKHNhZmVBbGlhcywgdHJ1ZSk7XG4gICAgICBjdHguYXBwZW5kKCcgJyArIGVzY2FwZWRGdWxsQWxpYXMpO1xuICAgIH1cbiAgfVxuXG4gIGFic3RyYWN0IGVzY2FwZSh2YWx1ZTogdW5rbm93bik6IHN0cmluZztcbn1cbiJdfQ==