sonamu 0.8.26 → 0.9.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 (684) hide show
  1. package/bin/cli.js +60 -13
  2. package/dist/_virtual/rolldown_runtime.js +39 -0
  3. package/dist/ai/agents/agent.d.ts +3 -3
  4. package/dist/ai/agents/agent.d.ts.map +1 -1
  5. package/dist/ai/agents/agent.js +76 -73
  6. package/dist/ai/agents/index.js +3 -3
  7. package/dist/ai/agents/types.d.ts +3 -3
  8. package/dist/ai/agents/types.d.ts.map +1 -1
  9. package/dist/ai/agents/types.js +1 -3
  10. package/dist/ai/index.js +3 -2
  11. package/dist/ai/providers/rtzr/api.js +25 -25
  12. package/dist/ai/providers/rtzr/error.js +25 -26
  13. package/dist/ai/providers/rtzr/index.js +5 -5
  14. package/dist/ai/providers/rtzr/model.d.ts +1 -1
  15. package/dist/ai/providers/rtzr/model.d.ts.map +1 -1
  16. package/dist/ai/providers/rtzr/model.js +117 -133
  17. package/dist/ai/providers/rtzr/options.d.ts.map +1 -1
  18. package/dist/ai/providers/rtzr/options.js +35 -41
  19. package/dist/ai/providers/rtzr/provider.d.ts +1 -1
  20. package/dist/ai/providers/rtzr/provider.d.ts.map +1 -1
  21. package/dist/ai/providers/rtzr/provider.js +53 -51
  22. package/dist/ai/providers/rtzr/utils.d.ts.map +1 -1
  23. package/dist/ai/providers/rtzr/utils.js +84 -84
  24. package/dist/api/base-frame.d.ts +2 -2
  25. package/dist/api/base-frame.d.ts.map +1 -1
  26. package/dist/api/base-frame.js +29 -19
  27. package/dist/api/caster.d.ts +1 -1
  28. package/dist/api/caster.d.ts.map +1 -1
  29. package/dist/api/caster.js +51 -61
  30. package/dist/api/code-converters.d.ts +4 -3
  31. package/dist/api/code-converters.d.ts.map +1 -1
  32. package/dist/api/code-converters.js +226 -249
  33. package/dist/api/config.d.ts +17 -17
  34. package/dist/api/config.d.ts.map +1 -1
  35. package/dist/api/config.js +37 -30
  36. package/dist/api/context.d.ts +10 -10
  37. package/dist/api/context.d.ts.map +1 -1
  38. package/dist/api/context.js +8 -2
  39. package/dist/api/decorators.d.ts +8 -8
  40. package/dist/api/decorators.d.ts.map +1 -1
  41. package/dist/api/decorators.js +245 -268
  42. package/dist/api/index.js +39 -7
  43. package/dist/api/secret.js +22 -15
  44. package/dist/api/sonamu.d.ts +15 -15
  45. package/dist/api/sonamu.d.ts.map +1 -1
  46. package/dist/api/sonamu.js +1012 -1131
  47. package/dist/api/validator.js +88 -79
  48. package/dist/auth/auth-generator.d.ts.map +1 -1
  49. package/dist/auth/auth-generator.js +203 -200
  50. package/dist/auth/better-auth-entities.d.ts +2 -2
  51. package/dist/auth/better-auth-entities.d.ts.map +1 -1
  52. package/dist/auth/better-auth-entities.js +369 -429
  53. package/dist/auth/index.js +21 -6
  54. package/dist/auth/knex-adapter.d.ts +2 -2
  55. package/dist/auth/knex-adapter.d.ts.map +1 -1
  56. package/dist/auth/knex-adapter.js +153 -157
  57. package/dist/auth/plugins/entity-definitions/admin.d.ts +1 -1
  58. package/dist/auth/plugins/entity-definitions/admin.d.ts.map +1 -1
  59. package/dist/auth/plugins/entity-definitions/admin.js +58 -56
  60. package/dist/auth/plugins/entity-definitions/anonymous.d.ts +1 -1
  61. package/dist/auth/plugins/entity-definitions/anonymous.d.ts.map +1 -1
  62. package/dist/auth/plugins/entity-definitions/anonymous.js +20 -20
  63. package/dist/auth/plugins/entity-definitions/api-key.d.ts +1 -1
  64. package/dist/auth/plugins/entity-definitions/api-key.d.ts.map +1 -1
  65. package/dist/auth/plugins/entity-definitions/api-key.js +185 -196
  66. package/dist/auth/plugins/entity-definitions/index.d.ts +1 -1
  67. package/dist/auth/plugins/entity-definitions/index.d.ts.map +1 -1
  68. package/dist/auth/plugins/entity-definitions/index.js +26 -29
  69. package/dist/auth/plugins/entity-definitions/jwt.d.ts +1 -1
  70. package/dist/auth/plugins/entity-definitions/jwt.d.ts.map +1 -1
  71. package/dist/auth/plugins/entity-definitions/jwt.js +62 -64
  72. package/dist/auth/plugins/entity-definitions/organization.d.ts +1 -1
  73. package/dist/auth/plugins/entity-definitions/organization.d.ts.map +1 -1
  74. package/dist/auth/plugins/entity-definitions/organization.js +362 -421
  75. package/dist/auth/plugins/entity-definitions/passkey.d.ts +1 -1
  76. package/dist/auth/plugins/entity-definitions/passkey.d.ts.map +1 -1
  77. package/dist/auth/plugins/entity-definitions/passkey.js +115 -126
  78. package/dist/auth/plugins/entity-definitions/phone-number.d.ts +1 -1
  79. package/dist/auth/plugins/entity-definitions/phone-number.d.ts.map +1 -1
  80. package/dist/auth/plugins/entity-definitions/phone-number.js +31 -40
  81. package/dist/auth/plugins/entity-definitions/sso.d.ts +1 -1
  82. package/dist/auth/plugins/entity-definitions/sso.d.ts.map +1 -1
  83. package/dist/auth/plugins/entity-definitions/sso.js +94 -107
  84. package/dist/auth/plugins/entity-definitions/two-factor.d.ts +1 -1
  85. package/dist/auth/plugins/entity-definitions/two-factor.d.ts.map +1 -1
  86. package/dist/auth/plugins/entity-definitions/two-factor.js +78 -92
  87. package/dist/auth/plugins/entity-definitions/types.d.ts +1 -1
  88. package/dist/auth/plugins/entity-definitions/types.d.ts.map +1 -1
  89. package/dist/auth/plugins/entity-definitions/types.js +1 -10
  90. package/dist/auth/plugins/entity-definitions/username.d.ts +1 -1
  91. package/dist/auth/plugins/entity-definitions/username.d.ts.map +1 -1
  92. package/dist/auth/plugins/entity-definitions/username.js +31 -40
  93. package/dist/auth/plugins/index.js +12 -3
  94. package/dist/auth/plugins/wrappers/admin.d.ts +2 -2
  95. package/dist/auth/plugins/wrappers/admin.d.ts.map +1 -1
  96. package/dist/auth/plugins/wrappers/admin.js +28 -29
  97. package/dist/auth/plugins/wrappers/anonymous.d.ts +2 -1
  98. package/dist/auth/plugins/wrappers/anonymous.d.ts.map +1 -1
  99. package/dist/auth/plugins/wrappers/anonymous.js +23 -22
  100. package/dist/auth/plugins/wrappers/api-key.d.ts +2 -1
  101. package/dist/auth/plugins/wrappers/api-key.d.ts.map +1 -1
  102. package/dist/auth/plugins/wrappers/api-key.js +39 -34
  103. package/dist/auth/plugins/wrappers/index.js +11 -11
  104. package/dist/auth/plugins/wrappers/jwt.d.ts +2 -1
  105. package/dist/auth/plugins/wrappers/jwt.d.ts.map +1 -1
  106. package/dist/auth/plugins/wrappers/jwt.js +31 -26
  107. package/dist/auth/plugins/wrappers/organization.d.ts +2 -1
  108. package/dist/auth/plugins/wrappers/organization.d.ts.map +1 -1
  109. package/dist/auth/plugins/wrappers/organization.js +65 -62
  110. package/dist/auth/plugins/wrappers/passkey.d.ts +2 -1
  111. package/dist/auth/plugins/wrappers/passkey.d.ts.map +1 -1
  112. package/dist/auth/plugins/wrappers/passkey.js +33 -28
  113. package/dist/auth/plugins/wrappers/phone-number.d.ts.map +1 -1
  114. package/dist/auth/plugins/wrappers/phone-number.js +26 -23
  115. package/dist/auth/plugins/wrappers/sso.d.ts.map +1 -1
  116. package/dist/auth/plugins/wrappers/sso.js +37 -31
  117. package/dist/auth/plugins/wrappers/two-factor.d.ts.map +1 -1
  118. package/dist/auth/plugins/wrappers/two-factor.js +31 -28
  119. package/dist/auth/plugins/wrappers/username.d.ts.map +1 -1
  120. package/dist/auth/plugins/wrappers/username.js +23 -23
  121. package/dist/bin/build-config.js +31 -31
  122. package/dist/bin/cli.js +1063 -1204
  123. package/dist/bin/fixture.d.ts.map +1 -1
  124. package/dist/bin/fixture.js +266 -259
  125. package/dist/bin/hmr-hook-register.d.ts.map +1 -1
  126. package/dist/bin/hmr-hook-register.js +19 -18
  127. package/dist/bin/test-command.d.ts.map +1 -1
  128. package/dist/bin/test-command.js +180 -177
  129. package/dist/bin/ts-loader-register.js +13 -6
  130. package/dist/bin/ts-loader-registration.d.ts.map +1 -1
  131. package/dist/bin/ts-loader-registration.js +28 -38
  132. package/dist/cache/cache-manager.d.ts +1 -1
  133. package/dist/cache/cache-manager.d.ts.map +1 -1
  134. package/dist/cache/cache-manager.js +20 -15
  135. package/dist/cache/decorator.d.ts +1 -1
  136. package/dist/cache/decorator.d.ts.map +1 -1
  137. package/dist/cache/decorator.js +84 -76
  138. package/dist/cache/drivers.js +21 -34
  139. package/dist/cache/index.js +10 -7
  140. package/dist/cache/types.d.ts +2 -2
  141. package/dist/cache/types.d.ts.map +1 -1
  142. package/dist/cache/types.js +1 -6
  143. package/dist/cache-control/cache-control.d.ts +2 -2
  144. package/dist/cache-control/cache-control.d.ts.map +1 -1
  145. package/dist/cache-control/cache-control.js +106 -122
  146. package/dist/cache-control/types.d.ts +2 -2
  147. package/dist/cache-control/types.d.ts.map +1 -1
  148. package/dist/cache-control/types.js +1 -19
  149. package/dist/compress/compress.d.ts +1 -1
  150. package/dist/compress/compress.d.ts.map +1 -1
  151. package/dist/compress/compress.js +58 -56
  152. package/dist/compress/index.js +7 -2
  153. package/dist/compress/types.js +1 -11
  154. package/dist/cone/cone-generator.d.ts +1 -1
  155. package/dist/cone/cone-generator.d.ts.map +1 -1
  156. package/dist/cone/cone-generator.js +216 -219
  157. package/dist/database/_batch_update.d.ts +1 -1
  158. package/dist/database/_batch_update.d.ts.map +1 -1
  159. package/dist/database/_batch_update.js +107 -102
  160. package/dist/database/base-model.d.ts +8 -9
  161. package/dist/database/base-model.d.ts.map +1 -1
  162. package/dist/database/base-model.js +371 -392
  163. package/dist/database/base-model.types.d.ts +5 -5
  164. package/dist/database/base-model.types.d.ts.map +1 -1
  165. package/dist/database/base-model.types.js +1 -20
  166. package/dist/database/db.d.ts +5 -2
  167. package/dist/database/db.d.ts.map +1 -1
  168. package/dist/database/db.js +185 -171
  169. package/dist/database/knex.d.ts +1 -1
  170. package/dist/database/knex.d.ts.map +1 -1
  171. package/dist/database/knex.js +48 -42
  172. package/dist/database/puri-subset.types.d.ts +6 -7
  173. package/dist/database/puri-subset.types.d.ts.map +1 -1
  174. package/dist/database/puri-subset.types.js +1 -16
  175. package/dist/database/puri-wrapper.d.ts +6 -6
  176. package/dist/database/puri-wrapper.d.ts.map +1 -1
  177. package/dist/database/puri-wrapper.js +99 -101
  178. package/dist/database/puri.d.ts +4 -5
  179. package/dist/database/puri.d.ts.map +1 -1
  180. package/dist/database/puri.js +1021 -1227
  181. package/dist/database/puri.types.d.ts +6 -6
  182. package/dist/database/puri.types.d.ts.map +1 -1
  183. package/dist/database/puri.types.js +15 -6
  184. package/dist/database/transaction-context.d.ts +2 -2
  185. package/dist/database/transaction-context.d.ts.map +1 -1
  186. package/dist/database/transaction-context.js +22 -13
  187. package/dist/database/upsert-builder.d.ts +3 -3
  188. package/dist/database/upsert-builder.d.ts.map +1 -1
  189. package/dist/database/upsert-builder.js +405 -465
  190. package/dist/dict/en.js +72 -74
  191. package/dist/dict/index.js +13 -13
  192. package/dist/dict/ko.js +72 -74
  193. package/dist/dict/rc-keys.js +150 -168
  194. package/dist/dict/sd.d.ts +3 -1
  195. package/dist/dict/sd.d.ts.map +1 -1
  196. package/dist/dict/sd.js +54 -40
  197. package/dist/dict/sonamu-dictionary.d.ts +1 -1
  198. package/dist/dict/sonamu-dictionary.d.ts.map +1 -1
  199. package/dist/dict/sonamu-dictionary.js +887 -955
  200. package/dist/dict/types.js +1 -7
  201. package/dist/dict/utils.js +26 -24
  202. package/dist/entity/entity-manager.d.ts +9 -9
  203. package/dist/entity/entity-manager.d.ts.map +1 -1
  204. package/dist/entity/entity-manager.js +226 -223
  205. package/dist/entity/entity-template-cone.d.ts +1 -1
  206. package/dist/entity/entity-template-cone.d.ts.map +1 -1
  207. package/dist/entity/entity-template-cone.js +152 -151
  208. package/dist/entity/entity.d.ts.map +1 -1
  209. package/dist/entity/entity.js +952 -1089
  210. package/dist/exceptions/error-handler.d.ts +1 -1
  211. package/dist/exceptions/error-handler.d.ts.map +1 -1
  212. package/dist/exceptions/error-handler.js +32 -27
  213. package/dist/exceptions/so-exceptions.d.ts +1 -1
  214. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  215. package/dist/exceptions/so-exceptions.js +61 -68
  216. package/dist/filter/index.js +9 -3
  217. package/dist/filter/types.js +92 -88
  218. package/dist/filter/utils.d.ts +1 -1
  219. package/dist/filter/utils.d.ts.map +1 -1
  220. package/dist/filter/utils.js +147 -161
  221. package/dist/index.js +87 -40
  222. package/dist/logger/category.d.ts.map +1 -1
  223. package/dist/logger/category.js +30 -29
  224. package/dist/logger/configure.d.ts.map +1 -1
  225. package/dist/logger/configure.js +83 -107
  226. package/dist/migration/code-generation.d.ts +2 -2
  227. package/dist/migration/code-generation.d.ts.map +1 -1
  228. package/dist/migration/code-generation.js +1385 -1578
  229. package/dist/migration/migration-set.d.ts +1 -1
  230. package/dist/migration/migration-set.d.ts.map +1 -1
  231. package/dist/migration/migration-set.js +177 -227
  232. package/dist/migration/migrator.d.ts +4 -3
  233. package/dist/migration/migrator.d.ts.map +1 -1
  234. package/dist/migration/migrator.js +340 -345
  235. package/dist/migration/postgresql-schema-reader.d.ts +2 -2
  236. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  237. package/dist/migration/postgresql-schema-reader.js +506 -564
  238. package/dist/migration/slack-confirm.d.ts +2 -2
  239. package/dist/migration/slack-confirm.d.ts.map +1 -1
  240. package/dist/migration/slack-confirm.js +205 -193
  241. package/dist/migration/types.d.ts +2 -2
  242. package/dist/migration/types.d.ts.map +1 -1
  243. package/dist/migration/types.js +1 -3
  244. package/dist/naite/messaging-types.d.ts +1 -0
  245. package/dist/naite/messaging-types.d.ts.map +1 -1
  246. package/dist/naite/messaging-types.js +1 -7
  247. package/dist/naite/naite-reporter.d.ts +2 -2
  248. package/dist/naite/naite-reporter.d.ts.map +1 -1
  249. package/dist/naite/naite-reporter.js +127 -120
  250. package/dist/naite/naite.d.ts +3 -2
  251. package/dist/naite/naite.d.ts.map +1 -1
  252. package/dist/naite/naite.js +266 -300
  253. package/dist/ssr/index.d.ts +2 -2
  254. package/dist/ssr/index.d.ts.map +1 -1
  255. package/dist/ssr/index.js +13 -3
  256. package/dist/ssr/registry.d.ts +1 -1
  257. package/dist/ssr/registry.d.ts.map +1 -1
  258. package/dist/ssr/registry.js +45 -37
  259. package/dist/ssr/renderer.d.ts +4 -4
  260. package/dist/ssr/renderer.d.ts.map +1 -1
  261. package/dist/ssr/renderer.js +84 -91
  262. package/dist/ssr/types.d.ts +2 -2
  263. package/dist/ssr/types.d.ts.map +1 -1
  264. package/dist/ssr/types.js +1 -3
  265. package/dist/storage/base-file.js +54 -41
  266. package/dist/storage/buffered-file.d.ts +2 -2
  267. package/dist/storage/buffered-file.d.ts.map +1 -1
  268. package/dist/storage/buffered-file.js +51 -44
  269. package/dist/storage/drivers.d.ts +2 -2
  270. package/dist/storage/drivers.d.ts.map +1 -1
  271. package/dist/storage/drivers.js +12 -7
  272. package/dist/storage/index.js +14 -7
  273. package/dist/storage/s3-driver.d.ts +2 -2
  274. package/dist/storage/s3-driver.d.ts.map +1 -1
  275. package/dist/storage/s3-driver.js +52 -48
  276. package/dist/storage/storage-manager.d.ts +2 -2
  277. package/dist/storage/storage-manager.d.ts.map +1 -1
  278. package/dist/storage/storage-manager.js +33 -25
  279. package/dist/storage/types.d.ts +2 -2
  280. package/dist/storage/types.d.ts.map +1 -1
  281. package/dist/storage/types.js +1 -5
  282. package/dist/storage/uploaded-file.d.ts +1 -1
  283. package/dist/storage/uploaded-file.d.ts.map +1 -1
  284. package/dist/storage/uploaded-file.js +45 -35
  285. package/dist/stream/index.js +7 -2
  286. package/dist/stream/sse.d.ts +2 -2
  287. package/dist/stream/sse.d.ts.map +1 -1
  288. package/dist/stream/sse.js +72 -67
  289. package/dist/syncer/api-parser.d.ts +1 -1
  290. package/dist/syncer/api-parser.d.ts.map +1 -1
  291. package/dist/syncer/api-parser.js +224 -245
  292. package/dist/syncer/checksum.d.ts +1 -1
  293. package/dist/syncer/checksum.d.ts.map +1 -1
  294. package/dist/syncer/checksum.js +86 -72
  295. package/dist/syncer/code-generator.d.ts +2 -2
  296. package/dist/syncer/code-generator.d.ts.map +1 -1
  297. package/dist/syncer/code-generator.js +154 -160
  298. package/dist/syncer/entity-operations.d.ts +1 -1
  299. package/dist/syncer/entity-operations.d.ts.map +1 -1
  300. package/dist/syncer/entity-operations.js +63 -54
  301. package/dist/syncer/file-patterns.d.ts +1 -1
  302. package/dist/syncer/file-patterns.d.ts.map +1 -1
  303. package/dist/syncer/file-patterns.js +38 -38
  304. package/dist/syncer/index.js +19 -8
  305. package/dist/syncer/module-loader.d.ts +5 -5
  306. package/dist/syncer/module-loader.d.ts.map +1 -1
  307. package/dist/syncer/module-loader.js +83 -78
  308. package/dist/syncer/syncer-actions.d.ts +2 -2
  309. package/dist/syncer/syncer-actions.d.ts.map +1 -1
  310. package/dist/syncer/syncer-actions.js +76 -91
  311. package/dist/syncer/syncer.d.ts +7 -6
  312. package/dist/syncer/syncer.d.ts.map +1 -1
  313. package/dist/syncer/syncer.js +426 -492
  314. package/dist/tasks/decorator.d.ts +3 -3
  315. package/dist/tasks/decorator.d.ts.map +1 -1
  316. package/dist/tasks/decorator.js +32 -28
  317. package/dist/tasks/step-wrapper.d.ts +1 -1
  318. package/dist/tasks/step-wrapper.d.ts.map +1 -1
  319. package/dist/tasks/step-wrapper.js +42 -41
  320. package/dist/tasks/workflow-manager.d.ts +2 -2
  321. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  322. package/dist/tasks/workflow-manager.js +192 -221
  323. package/dist/template/entity-converter.d.ts +1 -1
  324. package/dist/template/entity-converter.d.ts.map +1 -1
  325. package/dist/template/entity-converter.js +103 -103
  326. package/dist/template/helpers.d.ts.map +1 -1
  327. package/dist/template/helpers.js +163 -163
  328. package/dist/template/implementations/entity.template.d.ts +1 -1
  329. package/dist/template/implementations/entity.template.d.ts.map +1 -1
  330. package/dist/template/implementations/entity.template.js +76 -85
  331. package/dist/template/implementations/entry-server.template.d.ts +1 -1
  332. package/dist/template/implementations/entry-server.template.d.ts.map +1 -1
  333. package/dist/template/implementations/entry-server.template.js +32 -27
  334. package/dist/template/implementations/generated.template.d.ts +1 -1
  335. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  336. package/dist/template/implementations/generated.template.js +254 -275
  337. package/dist/template/implementations/generated_http.template.d.ts +2 -2
  338. package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
  339. package/dist/template/implementations/generated_http.template.js +114 -133
  340. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  341. package/dist/template/implementations/generated_sso.template.js +249 -275
  342. package/dist/template/implementations/init_types.template.d.ts +1 -1
  343. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  344. package/dist/template/implementations/init_types.template.js +40 -34
  345. package/dist/template/implementations/model.template.d.ts +1 -1
  346. package/dist/template/implementations/model.template.d.ts.map +1 -1
  347. package/dist/template/implementations/model.template.js +56 -53
  348. package/dist/template/implementations/model_test.template.d.ts +1 -1
  349. package/dist/template/implementations/model_test.template.d.ts.map +1 -1
  350. package/dist/template/implementations/model_test.template.js +32 -24
  351. package/dist/template/implementations/queries.template.d.ts +1 -1
  352. package/dist/template/implementations/queries.template.d.ts.map +1 -1
  353. package/dist/template/implementations/queries.template.js +84 -89
  354. package/dist/template/implementations/sd.template.d.ts +1 -1
  355. package/dist/template/implementations/sd.template.d.ts.map +1 -1
  356. package/dist/template/implementations/sd.template.js +137 -144
  357. package/dist/template/implementations/services.template.d.ts +1 -1
  358. package/dist/template/implementations/services.template.d.ts.map +1 -1
  359. package/dist/template/implementations/services.template.js +164 -189
  360. package/dist/template/implementations/view_form.template.d.ts +1 -1
  361. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  362. package/dist/template/implementations/view_form.template.js +258 -285
  363. package/dist/template/implementations/view_id_all_select.template.d.ts +1 -1
  364. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
  365. package/dist/template/implementations/view_id_all_select.template.js +31 -25
  366. package/dist/template/implementations/view_list.template.d.ts +1 -1
  367. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  368. package/dist/template/implementations/view_list.template.js +304 -355
  369. package/dist/template/implementations/view_search_input.template.d.ts +1 -1
  370. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  371. package/dist/template/implementations/view_search_input.template.js +31 -27
  372. package/dist/template/index.js +21 -7
  373. package/dist/template/template-manager.d.ts +1 -1
  374. package/dist/template/template-manager.d.ts.map +1 -1
  375. package/dist/template/template-manager.js +132 -123
  376. package/dist/template/template-types.js +8 -6
  377. package/dist/template/template.d.ts +2 -2
  378. package/dist/template/template.d.ts.map +1 -1
  379. package/dist/template/template.js +73 -68
  380. package/dist/template/zod-converter.d.ts.map +1 -1
  381. package/dist/template/zod-converter.js +603 -657
  382. package/dist/testing/_relation-graph.d.ts +1 -1
  383. package/dist/testing/_relation-graph.d.ts.map +1 -1
  384. package/dist/testing/_relation-graph.js +93 -88
  385. package/dist/testing/bootstrap.d.ts +22 -13
  386. package/dist/testing/bootstrap.d.ts.map +1 -1
  387. package/dist/testing/bootstrap.js +114 -114
  388. package/dist/testing/data-explorer.d.ts +3 -3
  389. package/dist/testing/data-explorer.d.ts.map +1 -1
  390. package/dist/testing/data-explorer.js +237 -265
  391. package/dist/testing/dev-test-routes.d.ts +2 -2
  392. package/dist/testing/dev-test-routes.d.ts.map +1 -1
  393. package/dist/testing/dev-test-routes.js +258 -249
  394. package/dist/testing/dev-vitest-manager.d.ts +1 -1
  395. package/dist/testing/dev-vitest-manager.d.ts.map +1 -1
  396. package/dist/testing/dev-vitest-manager.js +514 -539
  397. package/dist/testing/faker-mappings.js +422 -420
  398. package/dist/testing/fixture-generator.d.ts +3 -3
  399. package/dist/testing/fixture-generator.d.ts.map +1 -1
  400. package/dist/testing/fixture-generator.js +1216 -1346
  401. package/dist/testing/fixture-loader.js +26 -25
  402. package/dist/testing/fixture-manager.d.ts +3 -3
  403. package/dist/testing/fixture-manager.d.ts.map +1 -1
  404. package/dist/testing/fixture-manager.js +706 -776
  405. package/dist/testing/global-setup.js +53 -49
  406. package/dist/testing/index.js +19 -11
  407. package/dist/testing/naite-vitest-reporter.js +18 -13
  408. package/dist/testing/parallel-db-manager.d.ts +1 -1
  409. package/dist/testing/parallel-db-manager.d.ts.map +1 -1
  410. package/dist/testing/parallel-db-manager.js +63 -78
  411. package/dist/testing/vitest-helpers.d.ts +1 -1
  412. package/dist/testing/vitest-helpers.d.ts.map +1 -1
  413. package/dist/testing/vitest-helpers.js +37 -33
  414. package/dist/types/types.d.ts +28 -28
  415. package/dist/types/types.d.ts.map +1 -1
  416. package/dist/types/types.js +764 -890
  417. package/dist/ui/ai-api.d.ts +1 -1
  418. package/dist/ui/ai-api.d.ts.map +1 -1
  419. package/dist/ui/ai-api.js +52 -42
  420. package/dist/ui/ai-client.d.ts +1 -2
  421. package/dist/ui/ai-client.d.ts.map +1 -1
  422. package/dist/ui/ai-client.js +353 -388
  423. package/dist/ui/api.d.ts +1 -1
  424. package/dist/ui/api.d.ts.map +1 -1
  425. package/dist/ui/api.js +903 -1145
  426. package/dist/ui/cdd-service.d.ts +1 -1
  427. package/dist/ui/cdd-service.d.ts.map +1 -1
  428. package/dist/ui/cdd-service.js +406 -407
  429. package/dist/ui/cdd-types.js +1 -3
  430. package/dist/ui-web/assets/index-C-Zz-wYg.css +1 -0
  431. package/dist/ui-web/assets/index-DejDON8K.js +238 -0
  432. package/dist/ui-web/index.html +3 -3
  433. package/dist/utils/async-utils.js +57 -45
  434. package/dist/utils/console-util.d.ts.map +1 -1
  435. package/dist/utils/console-util.js +104 -87
  436. package/dist/utils/controller.js +26 -19
  437. package/dist/utils/esm-utils.js +49 -38
  438. package/dist/utils/formatter.d.ts +1 -2
  439. package/dist/utils/formatter.d.ts.map +1 -1
  440. package/dist/utils/formatter.js +89 -115
  441. package/dist/utils/fs-utils.d.ts.map +1 -1
  442. package/dist/utils/fs-utils.js +68 -65
  443. package/dist/utils/lodash-able.js +11 -4
  444. package/dist/utils/model.d.ts +1 -1
  445. package/dist/utils/model.d.ts.map +1 -1
  446. package/dist/utils/model.js +21 -19
  447. package/dist/utils/object-utils.js +148 -186
  448. package/dist/utils/path-utils.js +67 -57
  449. package/dist/utils/process-utils.d.ts.map +1 -1
  450. package/dist/utils/process-utils.js +37 -31
  451. package/dist/utils/sql-parser.d.ts +1 -1
  452. package/dist/utils/sql-parser.d.ts.map +1 -1
  453. package/dist/utils/sql-parser.js +40 -40
  454. package/dist/utils/type-utils.js +44 -43
  455. package/dist/utils/utils.d.ts +2 -3
  456. package/dist/utils/utils.d.ts.map +1 -1
  457. package/dist/utils/utils.js +81 -93
  458. package/dist/utils/zod-error.d.ts +1 -1
  459. package/dist/utils/zod-error.d.ts.map +1 -1
  460. package/dist/utils/zod-error.js +24 -17
  461. package/dist/vector/chunking.d.ts +1 -1
  462. package/dist/vector/chunking.d.ts.map +1 -1
  463. package/dist/vector/chunking.js +100 -94
  464. package/dist/vector/config.d.ts +1 -1
  465. package/dist/vector/config.d.ts.map +1 -1
  466. package/dist/vector/config.js +76 -78
  467. package/dist/vector/embedding.d.ts +1 -1
  468. package/dist/vector/embedding.d.ts.map +1 -1
  469. package/dist/vector/embedding.js +128 -125
  470. package/dist/vector/index.js +5 -5
  471. package/dist/vector/types.js +1 -5
  472. package/package.json +31 -36
  473. package/src/ai/agents/agent.ts +12 -5
  474. package/src/ai/agents/types.ts +5 -5
  475. package/src/ai/providers/rtzr/model.ts +8 -10
  476. package/src/ai/providers/rtzr/options.ts +2 -1
  477. package/src/ai/providers/rtzr/provider.ts +5 -3
  478. package/src/ai/providers/rtzr/utils.ts +2 -7
  479. package/src/api/__tests__/config.test.ts +15 -8
  480. package/src/api/base-frame.ts +5 -3
  481. package/src/api/caster.ts +7 -6
  482. package/src/api/code-converters.ts +23 -26
  483. package/src/api/config.ts +23 -17
  484. package/src/api/context.ts +18 -11
  485. package/src/api/decorators.ts +17 -18
  486. package/src/api/sonamu.ts +44 -49
  487. package/src/auth/auth-generator.ts +4 -6
  488. package/src/auth/better-auth-entities.ts +3 -2
  489. package/src/auth/knex-adapter.ts +6 -5
  490. package/src/auth/plugins/entity-definitions/admin.ts +1 -1
  491. package/src/auth/plugins/entity-definitions/anonymous.ts +1 -1
  492. package/src/auth/plugins/entity-definitions/api-key.ts +1 -1
  493. package/src/auth/plugins/entity-definitions/index.ts +1 -1
  494. package/src/auth/plugins/entity-definitions/jwt.ts +1 -1
  495. package/src/auth/plugins/entity-definitions/organization.ts +1 -1
  496. package/src/auth/plugins/entity-definitions/passkey.ts +1 -1
  497. package/src/auth/plugins/entity-definitions/phone-number.ts +1 -1
  498. package/src/auth/plugins/entity-definitions/sso.ts +1 -1
  499. package/src/auth/plugins/entity-definitions/two-factor.ts +1 -1
  500. package/src/auth/plugins/entity-definitions/types.ts +1 -1
  501. package/src/auth/plugins/entity-definitions/username.ts +1 -1
  502. package/src/auth/plugins/wrappers/admin.ts +3 -1
  503. package/src/auth/plugins/wrappers/anonymous.ts +3 -1
  504. package/src/auth/plugins/wrappers/api-key.ts +3 -1
  505. package/src/auth/plugins/wrappers/jwt.ts +3 -1
  506. package/src/auth/plugins/wrappers/organization.ts +3 -1
  507. package/src/auth/plugins/wrappers/passkey.ts +3 -1
  508. package/src/auth/plugins/wrappers/phone-number.ts +3 -1
  509. package/src/auth/plugins/wrappers/sso.ts +2 -1
  510. package/src/auth/plugins/wrappers/two-factor.ts +3 -1
  511. package/src/auth/plugins/wrappers/username.ts +3 -1
  512. package/src/bin/__tests__/ts-loader-register.test.ts +7 -12
  513. package/src/bin/build-config.ts +3 -3
  514. package/src/bin/cli.ts +27 -25
  515. package/src/bin/fixture.ts +4 -2
  516. package/src/bin/hmr-hook-register.ts +1 -0
  517. package/src/bin/test-command.ts +4 -2
  518. package/src/bin/ts-loader-registration.ts +6 -22
  519. package/src/cache/cache-manager.ts +2 -1
  520. package/src/cache/decorator.ts +2 -2
  521. package/src/cache/types.ts +3 -3
  522. package/src/cache-control/cache-control.ts +3 -2
  523. package/src/cache-control/types.ts +2 -2
  524. package/src/compress/compress.ts +1 -1
  525. package/src/cone/cone-generator.ts +5 -3
  526. package/src/database/_batch_update.ts +1 -1
  527. package/src/database/base-model.ts +20 -14
  528. package/src/database/base-model.types.ts +12 -11
  529. package/src/database/db.ts +56 -21
  530. package/src/database/knex.ts +2 -2
  531. package/src/database/puri-subset.test-d.ts +33 -32
  532. package/src/database/puri-subset.types.ts +6 -7
  533. package/src/database/puri-wrapper.ts +29 -26
  534. package/src/database/puri.ts +36 -34
  535. package/src/database/puri.types.test-d.ts +6 -5
  536. package/src/database/puri.types.ts +9 -12
  537. package/src/database/transaction-context.ts +2 -2
  538. package/src/database/upsert-builder.ts +17 -10
  539. package/src/dict/sd.ts +17 -4
  540. package/src/dict/sonamu-dictionary.ts +23 -17
  541. package/src/entity/entity-manager.ts +9 -7
  542. package/src/entity/entity-template-cone.ts +10 -3
  543. package/src/entity/entity.ts +20 -16
  544. package/src/exceptions/error-handler.ts +2 -1
  545. package/src/exceptions/so-exceptions.ts +1 -1
  546. package/src/filter/utils.ts +3 -2
  547. package/src/logger/category.ts +1 -0
  548. package/src/logger/configure.ts +5 -5
  549. package/src/migration/__tests__/code-generation.search-text.test.ts +2 -3
  550. package/src/migration/code-generation.ts +26 -25
  551. package/src/migration/migration-set.ts +16 -18
  552. package/src/migration/migrator.ts +38 -33
  553. package/src/migration/postgresql-schema-reader.ts +12 -12
  554. package/src/migration/slack-confirm.ts +5 -4
  555. package/src/migration/types.ts +2 -2
  556. package/src/naite/messaging-types.ts +1 -1
  557. package/src/naite/naite-reporter.ts +5 -3
  558. package/src/naite/naite.ts +12 -7
  559. package/src/shared/app.shared.ts.txt +2 -2
  560. package/src/shared/web.shared.ts.txt +2 -2
  561. package/src/skills/AGENTS.md +19 -18
  562. package/src/skills/commands/sonamu-skills.md +9 -9
  563. package/src/skills/sonamu/SKILL.md +111 -104
  564. package/src/skills/sonamu/ai-agents.md +27 -26
  565. package/src/skills/sonamu/api.md +81 -69
  566. package/src/skills/sonamu/auth-migration.md +13 -27
  567. package/src/skills/sonamu/auth-plugins.md +41 -31
  568. package/src/skills/sonamu/auth.md +30 -24
  569. package/src/skills/sonamu/cdd.md +26 -17
  570. package/src/skills/sonamu/cone.md +50 -50
  571. package/src/skills/sonamu/config.md +74 -51
  572. package/src/skills/sonamu/create-sonamu.md +31 -19
  573. package/src/skills/sonamu/database.md +43 -26
  574. package/src/skills/sonamu/entity-basic.md +61 -61
  575. package/src/skills/sonamu/entity-relations.md +84 -80
  576. package/src/skills/sonamu/entity-validation-checklist.md +19 -15
  577. package/src/skills/sonamu/fixture-cli.md +52 -30
  578. package/src/skills/sonamu/framework-change.md +9 -7
  579. package/src/skills/sonamu/frontend.md +64 -82
  580. package/src/skills/sonamu/i18n.md +45 -37
  581. package/src/skills/sonamu/migration.md +54 -31
  582. package/src/skills/sonamu/model.md +98 -66
  583. package/src/skills/sonamu/naite.md +34 -32
  584. package/src/skills/sonamu/project-init.md +28 -8
  585. package/src/skills/sonamu/puri.md +82 -91
  586. package/src/skills/sonamu/scaffolding.md +44 -32
  587. package/src/skills/sonamu/skill-contribution.md +50 -45
  588. package/src/skills/sonamu/subset.md +13 -13
  589. package/src/skills/sonamu/tasks.md +73 -58
  590. package/src/skills/sonamu/testing-devrunner.md +56 -36
  591. package/src/skills/sonamu/testing.md +23 -58
  592. package/src/skills/sonamu/upsert.md +32 -31
  593. package/src/skills/sonamu/vector.md +37 -36
  594. package/src/ssr/index.ts +2 -12
  595. package/src/ssr/registry.ts +1 -1
  596. package/src/ssr/renderer.ts +7 -5
  597. package/src/ssr/types.ts +2 -2
  598. package/src/storage/buffered-file.ts +4 -2
  599. package/src/storage/drivers.ts +3 -2
  600. package/src/storage/s3-driver.ts +7 -4
  601. package/src/storage/storage-manager.ts +3 -2
  602. package/src/storage/types.ts +3 -2
  603. package/src/storage/uploaded-file.ts +1 -1
  604. package/src/stream/sse.ts +2 -2
  605. package/src/syncer/api-parser.ts +8 -5
  606. package/src/syncer/checksum.ts +9 -5
  607. package/src/syncer/code-generator.ts +16 -8
  608. package/src/syncer/entity-operations.ts +5 -3
  609. package/src/syncer/file-patterns.ts +2 -1
  610. package/src/syncer/module-loader.ts +9 -6
  611. package/src/syncer/syncer-actions.ts +5 -3
  612. package/src/syncer/syncer.ts +18 -24
  613. package/src/tasks/decorator.ts +10 -8
  614. package/src/tasks/step-wrapper.ts +1 -1
  615. package/src/tasks/workflow-manager.ts +18 -15
  616. package/src/template/__tests__/generated.template.search-text.test.ts +1 -0
  617. package/src/template/entity-converter.ts +4 -2
  618. package/src/template/generated.template.test-d.ts +2 -1
  619. package/src/template/helpers.ts +5 -2
  620. package/src/template/implementations/entity.template.ts +9 -8
  621. package/src/template/implementations/entry-server.template.ts +1 -1
  622. package/src/template/implementations/generated.template.ts +21 -29
  623. package/src/template/implementations/generated_http.template.ts +9 -6
  624. package/src/template/implementations/generated_sso.template.ts +6 -4
  625. package/src/template/implementations/init_types.template.ts +3 -2
  626. package/src/template/implementations/model.template.ts +4 -2
  627. package/src/template/implementations/model_test.template.ts +3 -2
  628. package/src/template/implementations/queries.template.ts +6 -14
  629. package/src/template/implementations/sd.template.ts +4 -2
  630. package/src/template/implementations/services.template.ts +7 -11
  631. package/src/template/implementations/view_form.template.ts +5 -3
  632. package/src/template/implementations/view_id_all_select.template.ts +3 -2
  633. package/src/template/implementations/view_list.template.ts +7 -5
  634. package/src/template/implementations/view_search_input.template.ts +3 -2
  635. package/src/template/template-manager.ts +4 -3
  636. package/src/template/template.ts +4 -3
  637. package/src/template/zod-converter.ts +10 -7
  638. package/src/testing/__tests__/dev-test-routes.test.ts +3 -2
  639. package/src/testing/__tests__/dev-vitest-manager.test.ts +1 -0
  640. package/src/testing/_relation-graph.ts +2 -2
  641. package/src/testing/bootstrap.ts +55 -27
  642. package/src/testing/data-explorer.ts +5 -4
  643. package/src/testing/dev-test-routes.ts +8 -5
  644. package/src/testing/dev-vitest-manager.ts +13 -12
  645. package/src/testing/fixture-generator.ts +11 -17
  646. package/src/testing/fixture-manager.ts +21 -17
  647. package/src/testing/parallel-db-manager.ts +2 -1
  648. package/src/testing/vitest-helpers.ts +2 -1
  649. package/src/types/__tests__/entity-json-schema-search-text.test.ts +1 -0
  650. package/src/types/types.ts +8 -8
  651. package/src/typings/knex.d.ts +4 -4
  652. package/src/ui/ai-api.ts +5 -3
  653. package/src/ui/ai-client.ts +6 -5
  654. package/src/ui/api.ts +25 -23
  655. package/src/ui/cdd-service.ts +12 -11
  656. package/src/utils/console-util.ts +3 -1
  657. package/src/utils/formatter.ts +94 -102
  658. package/src/utils/fs-utils.ts +2 -1
  659. package/src/utils/model.ts +2 -2
  660. package/src/utils/object-utils.ts +3 -3
  661. package/src/utils/process-utils.ts +2 -1
  662. package/src/utils/sql-parser.ts +10 -1
  663. package/src/utils/type-utils.ts +3 -3
  664. package/src/utils/utils.ts +9 -7
  665. package/src/utils/zod-error.ts +1 -1
  666. package/src/vector/chunking.ts +1 -1
  667. package/src/vector/config.ts +1 -1
  668. package/src/vector/embedding.ts +11 -9
  669. package/tsdown.api.config.ts +50 -0
  670. package/.swcrc.project-default +0 -18
  671. package/dist/api/__tests__/config.test.js +0 -189
  672. package/dist/bin/__tests__/test-command.test.js +0 -112
  673. package/dist/bin/__tests__/ts-loader-register.test.js +0 -45
  674. package/dist/database/puri-subset.test-d.js +0 -89
  675. package/dist/database/puri.types.test-d.js +0 -129
  676. package/dist/migration/__tests__/code-generation.search-text.test.js +0 -435
  677. package/dist/template/__tests__/generated.template.search-text.test.js +0 -99
  678. package/dist/template/generated.template.test-d.js +0 -24
  679. package/dist/testing/__tests__/dev-test-routes.test.js +0 -144
  680. package/dist/testing/__tests__/dev-vitest-manager.test.js +0 -152
  681. package/dist/types/__tests__/entity-json-schema-search-text.test.js +0 -256
  682. package/dist/typings/knex.d.js +0 -3
  683. package/dist/ui-web/assets/index-CKo0Z2Iu.css +0 -1
  684. package/dist/ui-web/assets/index-DK-2aacv.js +0 -257
@@ -1,469 +1,409 @@
1
+ import { __esmMin } from "../_virtual/rolldown_runtime.js";
2
+ import { assertDefined, init_utils, nonNullable } from "../utils/utils.js";
3
+ import { EntityManager, init_entity_manager } from "../entity/entity-manager.js";
4
+ import { Naite, init_naite } from "../naite/naite.js";
5
+ import { batchUpdate, init__batch_update } from "./_batch_update.js";
1
6
  import { getLogger } from "@logtape/logtape";
7
+ import { cluster, isArray, unique } from "radashi";
2
8
  import { randomUUID } from "crypto";
3
- import { isArray, unique } from "radashi";
4
- import { EntityManager } from "../entity/entity-manager.js";
5
- import { Naite } from "../naite/naite.js";
6
- import { assertDefined, chunk, nonNullable } from "../utils/utils.js";
7
- import { batchUpdate } from "./_batch_update.js";
8
- const logger = getLogger([
9
- "sonamu",
10
- "internal",
11
- "upsert-builder"
12
- ]);
13
- export function isRefField(field) {
14
- return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
15
- }
16
- export class UpsertBuilder {
17
- tables;
18
- constructor(){
19
- this.tables = new Map();
20
- }
21
- getTable(tableName) {
22
- const table = this.tables.get(tableName);
23
- if (table) {
24
- return table;
25
- }
26
- const tableSpec = (()=>{
27
- try {
28
- return EntityManager.getTableSpec(tableName);
29
- } catch {
30
- return null;
31
- }
32
- })();
33
- const tableData = {
34
- references: new Set(),
35
- rows: [],
36
- uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
37
- uniquesMap: new Map(),
38
- jsonColumns: tableSpec?.jsonColumns ?? []
39
- };
40
- this.tables.set(tableName, tableData);
41
- return tableData;
42
- }
43
- hasTable(tableName) {
44
- return this.tables.has(tableName);
45
- }
46
- register(tableName, row) {
47
- const table = this.getTable(tableName);
48
- // 해당 테이블의 unique 인덱스를 순회하며 키 생성
49
- const uniqueKeys = table.uniqueIndexes.map((unqIndex)=>{
50
- const uniqueKeyArray = unqIndex.columns.map((unqCol)=>{
51
- const val = row[unqCol.name];
52
- if (isRefField(val)) {
53
- return val.uuid;
54
- } else {
55
- return row[unqCol.name] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입
56
- }
57
- });
58
- // 값이 모두 null인 경우 키 생성 패스
59
- if (uniqueKeyArray.length === 0) {
60
- return null;
61
- }
62
- return uniqueKeyArray.join("---delimiter--");
63
- }).filter(nonNullable);
64
- // uuid 생성 로직
65
- const { uuid, isReused } = (()=>{
66
- // 키를 순회하여 이미 존재하는 키가 있는지 확인
67
- if (uniqueKeys.length > 0) {
68
- for (const uniqueKey of uniqueKeys){
69
- if (table.uniquesMap.has(uniqueKey)) {
70
- return {
71
- uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
72
- isReused: true
73
- };
74
- }
75
- }
76
- }
77
- // 찾을 수 없는 경우 생성
78
- return {
79
- uuid: randomUUID(),
80
- isReused: false
81
- };
82
- })();
83
- // 모든 유니크키에 대해 유니크맵에 uuid 저장
84
- if (uniqueKeys.length > 0) {
85
- for (const uniqueKey of uniqueKeys){
86
- table.uniquesMap.set(uniqueKey, uuid);
87
- }
88
- }
89
- // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가
90
- // 이 정보를 나중에 치환할 때 사용
91
- row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue])=>{
92
- if (isRefField(rowValue)) {
93
- rowValue.use ??= "id";
94
- table.references.add(`${rowValue.of}.${rowValue.use}`);
95
- return [
96
- rowKey,
97
- rowValue
98
- ];
99
- } else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {
100
- // JSON 컬럼인 경우 JSON.stringify 처리 (Knex는 JSON 타입을 지원하지 않음)
101
- return [
102
- rowKey,
103
- JSON.stringify(rowValue)
104
- ];
105
- } else if (isArray(rowValue)) {
106
- // 배열은 그대로 저장
107
- return [
108
- rowKey,
109
- rowValue
110
- ];
111
- } else {
112
- return [
113
- rowKey,
114
- rowValue
115
- ];
116
- }
117
- }));
118
- table.rows.push({
119
- uuid,
120
- ...row
121
- });
122
- const result = {
123
- of: tableName,
124
- uuid: row.uuid ?? uuid
125
- };
126
- Naite.t("puri:ub-register", {
127
- tableName,
128
- uuid: result.uuid,
129
- isUuidReused: isReused,
130
- row
131
- });
132
- return result;
133
- }
134
- async upsert(wdb, tableName, options) {
135
- return this.upsertOrInsert(wdb, tableName, "upsert", options);
136
- }
137
- async insertOnly(wdb, tableName, options) {
138
- return this.upsertOrInsert(wdb, tableName, "insert", options);
139
- }
140
- async upsertOrInsert(wdb, tableName, mode, options) {
141
- if (this.hasTable(tableName) === false) {
142
- return [];
143
- }
144
- const table = this.tables.get(tableName);
145
- if (table === undefined) {
146
- throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);
147
- } else if (table.rows.length === 0) {
148
- throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);
149
- }
150
- if (table.rows.some((row)=>Object.entries(row).some(([, value])=>isRefField(value) && value.of !== tableName))) {
151
- throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);
152
- }
153
- // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출
154
- const { references, refTables } = Array.from(this.tables).reduce((r, [, table])=>{
155
- const reference = Array.from(table.references.values()).find((ref)=>ref.includes(`${tableName}.`));
156
- if (reference) {
157
- r.references.push(reference);
158
- r.refTables.push(table);
159
- }
160
- return r;
161
- }, {
162
- references: [],
163
- refTables: []
164
- });
165
- const extractFields = unique(references).map((reference)=>reference.split(".")[1]).filter((field)=>field !== undefined);
166
- // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)
167
- const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
168
- if (hasCircular) {
169
- throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
170
- }
171
- const uuidMap = new Map();
172
- const allIds = [];
173
- // 레벨별로 순차 처리
174
- for(let levelIdx = 0; levelIdx < levels.length; levelIdx++){
175
- const levelRows = levels[levelIdx];
176
- logger.debug("Processing Query Level: {current} / {total}", {
177
- current: levelIdx + 1,
178
- total: levels.length
179
- });
180
- // 이전 레벨에서 얻은 ID로 자기 참조 해결
181
- const resolvedRows = levelRows.map((row)=>{
182
- const resolved = {
183
- ...row
184
- };
185
- for (const [key, value] of Object.entries(row)){
186
- if (isRefField(value) && value.of === tableName) {
187
- const parent = uuidMap.get(value.uuid);
188
- if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
189
- resolved[key] = parent[value.use ?? "id"];
190
- Naite.t("puri:ub-ref-resolved", {
191
- tableName,
192
- field: key,
193
- from: {
194
- of: value.of,
195
- uuid: value.uuid,
196
- use: value.use ?? "id"
197
- },
198
- to: resolved[key]
199
- });
200
- }
201
- }
202
- return resolved;
203
- });
204
- // 현재 레벨 upsert
205
- const chunkSize = options?.chunkSize;
206
- const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [
207
- resolvedRows
208
- ];
209
- const selectFields = unique([
210
- "id",
211
- ...extractFields
212
- ]);
213
- for(let index = 0; index < levelChunks.length; index++){
214
- const dataChunk = levelChunks[index];
215
- if (dataChunk.length === 0) continue;
216
- logger.debug("Processing Chunk: {current} / {total}", {
217
- current: index + 1,
218
- total: levelChunks.length
219
- });
220
- // uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거
221
- const originalUuids = dataChunk.map((r)=>r.uuid);
222
- const dataForDb = dataChunk.map(({ uuid, ...rest })=>rest);
223
- let resultRows;
224
- if (mode === "insert") {
225
- // INSERT 모드 - RETURNING 사용
226
- resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
227
- } else {
228
- // UPSERT 모드 - id 없는 row들의 id를 사전 조회로 채우기
229
- const rowsWithoutId = dataForDb.filter((row)=>!row.id);
230
- if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {
231
- // 모든 uniqueIndexes로 기존 레코드 조회
232
- for (const uniqueIndex of table.uniqueIndexes){
233
- const columns = uniqueIndex.columns.map((c)=>c.name);
234
- // 조회할 조건들 추출 (각 row의 unique 컬럼 값들)
235
- const conditions = [];
236
- for (const row of rowsWithoutId){
237
- const values = columns.map((col)=>row[col]);
238
- // null이 포함된 조건은 제외 (PostgreSQL UNIQUE는 NULL 무시)
239
- if (!values.some((v)=>v == null)) {
240
- conditions.push(values);
241
- }
242
- }
243
- if (conditions.length === 0) continue;
244
- // 배치 SELECT
245
- const existingRows = await wdb(tableName).whereIn(columns, conditions).select("id", ...columns);
246
- // Map 생성: unique 컬럼 조합 → id
247
- const existingMap = new Map();
248
- for (const existing of existingRows){
249
- const key = columns.map((col)=>String(existing[col] ?? "")).join("---delimiter---");
250
- const id = existing.id;
251
- if (typeof id === "number" || typeof id === "string") {
252
- existingMap.set(key, id);
253
- }
254
- }
255
- // id 없는 row들에 매칭되는 id 채우기
256
- for (const row of rowsWithoutId){
257
- if (row.id) continue; // 이미 다른 uniqueIndex에서 채워진 경우 스킵
258
- const key = columns.map((col)=>String(row[col] ?? "")).join("---delimiter---");
259
- const existingId = existingMap.get(key);
260
- if (existingId) {
261
- row.id = existingId;
262
- }
263
- }
264
- }
265
- }
266
- // onConflict는 id만 사용 (모든 uniqueIndexes는 이미 사전 조회로 처리됨)
267
- const conflictColumns = [
268
- "id"
269
- ];
270
- const allColumns = Object.keys(dataForDb[0]);
271
- let updateColumns = allColumns.filter((c)=>c !== "id");
272
- // inherit 옵션 처리 - inherit 컬럼은 update 대상에서 제외
273
- if (options?.inherit?.length) {
274
- const inheritColumns = options.inherit;
275
- const excludedFromUpdate = updateColumns.filter((c)=>inheritColumns.includes(c));
276
- updateColumns = updateColumns.filter((c)=>!inheritColumns.includes(c));
277
- // 실제로 제외된 컬럼 로깅
278
- if (excludedFromUpdate.length) {
279
- Naite.t("puri:ub-inherit", {
280
- tableName,
281
- inheritColumns,
282
- excludedFromUpdate
283
- });
284
- }
285
- }
286
- // updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장
287
- const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;
288
- resultRows = await wdb.insert(dataForDb).into(tableName).onConflict(conflictColumns).merge(mergeColumns).returning(selectFields);
289
- }
290
- if (originalUuids.length !== resultRows.length) {
291
- throw new Error(`${tableName}: register/returning 불일치`);
292
- }
293
- for(let i = 0; i < resultRows.length; i++){
294
- uuidMap.set(originalUuids[i], resultRows[i]);
295
- allIds.push(resultRows[i].id);
296
- }
297
- }
298
- }
299
- // 해당 테이블 참조를 실제 밸류로 변경
300
- for (const table of refTables){
301
- table.rows = table.rows.map((row)=>{
302
- for (const key of Object.keys(row)){
303
- const prop = row[key];
304
- if (isRefField(prop) && prop.of === tableName) {
305
- const parent = uuidMap.get(prop.uuid);
306
- if (!parent) {
307
- console.error(prop);
308
- throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
309
- }
310
- const resolvedValue = parent[prop.use ?? "id"];
311
- row[key] = resolvedValue;
312
- Naite.t("puri:ub-ref-resolved", {
313
- tableName,
314
- field: key,
315
- from: {
316
- of: prop.of,
317
- uuid: prop.uuid,
318
- use: prop.use ?? "id"
319
- },
320
- to: resolvedValue
321
- });
322
- }
323
- }
324
- return row;
325
- });
326
- }
327
- if (options?.cleanOrphans) {
328
- const cleanOrphans = options.cleanOrphans;
329
- const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [
330
- cleanOrphans
331
- ];
332
- // 현재 register된 레코드들의 FK 값들 추출
333
- const fkConditions = fkColumns.map((fkCol)=>{
334
- const fkValues = [
335
- ...new Set(table.rows.map((row)=>row[fkCol]).filter((v)=>v != null))
336
- ];
337
- return {
338
- column: fkCol,
339
- values: fkValues
340
- };
341
- });
342
- // 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행
343
- if (fkConditions.every((fc)=>fc.values.length > 0)) {
344
- let deleteQuery = wdb(tableName);
345
- // 각 FK 컬럼에 대한 WHERE IN 조건 추가
346
- for (const { column, values } of fkConditions){
347
- deleteQuery = deleteQuery.whereIn(column, values);
348
- }
349
- // 방금 upsert한 ID는 제외
350
- deleteQuery = deleteQuery.whereNotIn("id", allIds);
351
- const deletedCount = await deleteQuery.delete();
352
- Naite.t("puri:ub-clean-orphans", {
353
- tableName,
354
- cleanOrphans: fkColumns,
355
- deletedCount
356
- });
357
- }
358
- }
359
- // 해당 테이블의 데이터 초기화
360
- table.rows = [];
361
- table.references.clear();
362
- table.uniquesMap.clear();
363
- Naite.t("puri:ub-upserted", {
364
- tableName,
365
- mode,
366
- rowCount: allIds.length,
367
- returnedIds: allIds
368
- });
369
- return allIds;
370
- }
371
- async updateBatch(wdb, tableName, options) {
372
- options = {
373
- ...options,
374
- chunkSize: options?.chunkSize ?? 500,
375
- where: options?.where ?? "id"
376
- };
377
- if (this.hasTable(tableName) === false) {
378
- return;
379
- }
380
- const table = this.tables.get(tableName);
381
- if (!table) {
382
- throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
383
- } else if (table.rows.length === 0) {
384
- return;
385
- }
386
- const whereColumns = Array.isArray(options.where) ? options.where : [
387
- options.where ?? "id"
388
- ];
389
- const rows = table.rows.map((_row)=>{
390
- const { uuid: _, ...row } = _row; // uuid 제외
391
- return row;
392
- });
393
- await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
394
- Naite.t("puri:ub-batch-updated", {
395
- tableName,
396
- rowCount: rows.length,
397
- whereColumns
398
- });
399
- // updateBatch 완료 후 처리된 데이터 제거
400
- table.rows = [];
401
- table.references.clear();
402
- table.uniquesMap.clear();
403
- }
404
- // ============================================================================
405
- // Private Helper Methods
406
- // ============================================================================
407
- /**
408
- * rows를 의존성 순서에 따라 레벨별로 그룹화
409
- * - 자기 참조 없는 경우 : 모든 rows가 Level 0
410
- * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
411
- */ buildInsertLevels(rows, tableName) {
412
- // 1. 자기 참조가 없으면 한 레벨로 처리
413
- const hasSelfRef = rows.flatMap((row)=>Object.values(row)).some((value)=>isRefField(value) && value.of === tableName);
414
- if (!hasSelfRef) return {
415
- levels: [
416
- rows
417
- ],
418
- hasCircular: false
419
- };
420
- // 2. uuid → row 매핑 (중복 uuid 방지)
421
- const rowByUuid = new Map();
422
- for (const row of rows){
423
- const uuid = row.uuid;
424
- if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
425
- rowByUuid.set(uuid, row);
426
- }
427
- let pending = Array.from(rowByUuid.values());
428
- const levels = [];
429
- const inserted = new Set();
430
- // 3. 레벨별 분류
431
- while(pending.length > 0){
432
- const currentLevel = [];
433
- const nextPending = [];
434
- for (const row of pending){
435
- // 이 row가 참조하는 자기 참조들
436
- const selfRefs = Object.values(row).filter((value)=>isRefField(value) && value.of === tableName);
437
- // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함
438
- const canInsert = selfRefs.every((ref)=>{
439
- if (!rowByUuid.has(ref.uuid)) {
440
- throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
441
- }
442
- return inserted.has(ref.uuid);
443
- });
444
- if (canInsert) {
445
- currentLevel.push(row);
446
- } else {
447
- nextPending.push(row);
448
- }
449
- }
450
- // 순환 참조 감지
451
- if (currentLevel.length === 0) return {
452
- levels: [],
453
- hasCircular: true
454
- };
455
- // 레벨 확정 + inserted 갱신
456
- levels.push(currentLevel);
457
- for (const row of currentLevel){
458
- inserted.add(row.uuid);
459
- }
460
- pending = nextPending;
461
- }
462
- return {
463
- levels,
464
- hasCircular: false
465
- };
466
- }
9
+
10
+ //#region src/database/upsert-builder.ts
11
+ function isRefField(field) {
12
+ return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
467
13
  }
14
+ var logger, UpsertBuilder;
15
+ var init_upsert_builder = __esmMin((() => {
16
+ init_entity_manager();
17
+ init_naite();
18
+ init_utils();
19
+ init__batch_update();
20
+ logger = getLogger([
21
+ "sonamu",
22
+ "internal",
23
+ "upsert-builder"
24
+ ]);
25
+ UpsertBuilder = class {
26
+ tables;
27
+ constructor() {
28
+ this.tables = new Map();
29
+ }
30
+ getTable(tableName) {
31
+ const table = this.tables.get(tableName);
32
+ if (table) {
33
+ return table;
34
+ }
35
+ const tableSpec = (() => {
36
+ try {
37
+ return EntityManager.getTableSpec(tableName);
38
+ } catch {
39
+ return null;
40
+ }
41
+ })();
42
+ const tableData = {
43
+ references: new Set(),
44
+ rows: [],
45
+ uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
46
+ uniquesMap: new Map(),
47
+ jsonColumns: tableSpec?.jsonColumns ?? []
48
+ };
49
+ this.tables.set(tableName, tableData);
50
+ return tableData;
51
+ }
52
+ hasTable(tableName) {
53
+ return this.tables.has(tableName);
54
+ }
55
+ register(tableName, row) {
56
+ const table = this.getTable(tableName);
57
+ const uniqueKeys = table.uniqueIndexes.map((unqIndex) => {
58
+ const uniqueKeyArray = unqIndex.columns.map((unqCol) => {
59
+ const val = row[unqCol.name];
60
+ if (isRefField(val)) {
61
+ return val.uuid;
62
+ } else {
63
+ return row[unqCol.name] ?? randomUUID();
64
+ }
65
+ });
66
+ if (uniqueKeyArray.length === 0) {
67
+ return null;
68
+ }
69
+ return uniqueKeyArray.join("---delimiter--");
70
+ }).filter(nonNullable);
71
+ const { uuid, isReused } = (() => {
72
+ if (uniqueKeys.length > 0) {
73
+ for (const uniqueKey of uniqueKeys) {
74
+ if (table.uniquesMap.has(uniqueKey)) {
75
+ return {
76
+ uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
77
+ isReused: true
78
+ };
79
+ }
80
+ }
81
+ }
82
+ return {
83
+ uuid: randomUUID(),
84
+ isReused: false
85
+ };
86
+ })();
87
+ if (uniqueKeys.length > 0) {
88
+ for (const uniqueKey of uniqueKeys) {
89
+ table.uniquesMap.set(uniqueKey, uuid);
90
+ }
91
+ }
92
+ row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue]) => {
93
+ if (isRefField(rowValue)) {
94
+ rowValue.use ??= "id";
95
+ table.references.add(`${rowValue.of}.${rowValue.use}`);
96
+ return [rowKey, rowValue];
97
+ } else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {
98
+ return [rowKey, JSON.stringify(rowValue)];
99
+ } else if (isArray(rowValue)) {
100
+ return [rowKey, rowValue];
101
+ } else {
102
+ return [rowKey, rowValue];
103
+ }
104
+ }));
105
+ table.rows.push({
106
+ uuid,
107
+ ...row
108
+ });
109
+ const result = {
110
+ of: tableName,
111
+ uuid: row.uuid ?? uuid
112
+ };
113
+ Naite.t("puri:ub-register", {
114
+ tableName,
115
+ uuid: result.uuid,
116
+ isUuidReused: isReused,
117
+ row
118
+ });
119
+ return result;
120
+ }
121
+ async upsert(wdb, tableName, options) {
122
+ return this.upsertOrInsert(wdb, tableName, "upsert", options);
123
+ }
124
+ async insertOnly(wdb, tableName, options) {
125
+ return this.upsertOrInsert(wdb, tableName, "insert", options);
126
+ }
127
+ async upsertOrInsert(wdb, tableName, mode, options) {
128
+ if (!this.hasTable(tableName)) {
129
+ return [];
130
+ }
131
+ const table = this.tables.get(tableName);
132
+ if (table === undefined) {
133
+ throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);
134
+ } else if (table.rows.length === 0) {
135
+ throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);
136
+ }
137
+ if (table.rows.some((row) => Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName))) {
138
+ throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);
139
+ }
140
+ const { references, refTables } = Array.from(this.tables).reduce((r, [, table$1]) => {
141
+ const reference = Array.from(table$1.references.values()).find((ref) => ref.includes(`${tableName}.`));
142
+ if (reference) {
143
+ r.references.push(reference);
144
+ r.refTables.push(table$1);
145
+ }
146
+ return r;
147
+ }, {
148
+ references: [],
149
+ refTables: []
150
+ });
151
+ const extractFields = unique(references).map((reference) => reference.split(".")[1]).filter((field) => field !== undefined);
152
+ const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
153
+ if (hasCircular) {
154
+ throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
155
+ }
156
+ const uuidMap = new Map();
157
+ const allIds = [];
158
+ for (let levelIdx = 0; levelIdx < levels.length; levelIdx++) {
159
+ const levelRows = levels[levelIdx];
160
+ logger.debug("Processing Query Level: {current} / {total}", {
161
+ current: levelIdx + 1,
162
+ total: levels.length
163
+ });
164
+ const resolvedRows = levelRows.map((row) => {
165
+ const resolved = { ...row };
166
+ for (const [key, value] of Object.entries(row)) {
167
+ if (isRefField(value) && value.of === tableName) {
168
+ const parent = uuidMap.get(value.uuid);
169
+ if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
170
+ resolved[key] = parent[value.use ?? "id"];
171
+ Naite.t("puri:ub-ref-resolved", {
172
+ tableName,
173
+ field: key,
174
+ from: {
175
+ of: value.of,
176
+ uuid: value.uuid,
177
+ use: value.use ?? "id"
178
+ },
179
+ to: resolved[key]
180
+ });
181
+ }
182
+ }
183
+ return resolved;
184
+ });
185
+ const chunkSize = options?.chunkSize;
186
+ const levelChunks = chunkSize ? cluster(resolvedRows, chunkSize) : [resolvedRows];
187
+ const selectFields = unique(["id", ...extractFields]);
188
+ for (let index = 0; index < levelChunks.length; index++) {
189
+ const dataChunk = levelChunks[index];
190
+ if (dataChunk.length === 0) continue;
191
+ logger.debug("Processing Chunk: {current} / {total}", {
192
+ current: index + 1,
193
+ total: levelChunks.length
194
+ });
195
+ const originalUuids = dataChunk.map((r) => r.uuid);
196
+ const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);
197
+ let resultRows;
198
+ if (mode === "insert") {
199
+ resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
200
+ } else {
201
+ const rowsWithoutId = dataForDb.filter((row) => !row.id);
202
+ if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {
203
+ for (const uniqueIndex of table.uniqueIndexes) {
204
+ const columns = uniqueIndex.columns.map((c) => c.name);
205
+ const conditions = [];
206
+ for (const row of rowsWithoutId) {
207
+ const values = columns.map((col) => row[col]);
208
+ if (!values.some((v) => v == null)) {
209
+ conditions.push(values);
210
+ }
211
+ }
212
+ if (conditions.length === 0) continue;
213
+ const existingRows = await wdb(tableName).whereIn(columns, conditions).select("id", ...columns);
214
+ const existingMap = new Map();
215
+ for (const existing of existingRows) {
216
+ const key = columns.map((col) => String(existing[col] ?? "")).join("---delimiter---");
217
+ const id = existing.id;
218
+ if (typeof id === "number" || typeof id === "string") {
219
+ existingMap.set(key, id);
220
+ }
221
+ }
222
+ for (const row of rowsWithoutId) {
223
+ if (row.id) continue;
224
+ const key = columns.map((col) => String(row[col] ?? "")).join("---delimiter---");
225
+ const existingId = existingMap.get(key);
226
+ if (existingId) {
227
+ row.id = existingId;
228
+ }
229
+ }
230
+ }
231
+ }
232
+ const conflictColumns = ["id"];
233
+ const allColumns = Object.keys(dataForDb[0]);
234
+ let updateColumns = allColumns.filter((c) => c !== "id");
235
+ if (options?.inherit?.length) {
236
+ const inheritColumns = options.inherit;
237
+ const excludedFromUpdate = updateColumns.filter((c) => inheritColumns.includes(c));
238
+ updateColumns = updateColumns.filter((c) => !inheritColumns.includes(c));
239
+ if (excludedFromUpdate.length) {
240
+ Naite.t("puri:ub-inherit", {
241
+ tableName,
242
+ inheritColumns,
243
+ excludedFromUpdate
244
+ });
245
+ }
246
+ }
247
+ const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;
248
+ resultRows = await wdb.insert(dataForDb).into(tableName).onConflict(conflictColumns).merge(mergeColumns).returning(selectFields);
249
+ }
250
+ if (originalUuids.length !== resultRows.length) {
251
+ throw new Error(`${tableName}: register/returning 불일치`);
252
+ }
253
+ for (let i = 0; i < resultRows.length; i++) {
254
+ uuidMap.set(originalUuids[i], resultRows[i]);
255
+ allIds.push(resultRows[i].id);
256
+ }
257
+ }
258
+ }
259
+ for (const table$1 of refTables) {
260
+ table$1.rows = table$1.rows.map((row) => {
261
+ for (const key of Object.keys(row)) {
262
+ const prop = row[key];
263
+ if (isRefField(prop) && prop.of === tableName) {
264
+ const parent = uuidMap.get(prop.uuid);
265
+ if (!parent) {
266
+ console.error(prop);
267
+ throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
268
+ }
269
+ const resolvedValue = parent[prop.use ?? "id"];
270
+ row[key] = resolvedValue;
271
+ Naite.t("puri:ub-ref-resolved", {
272
+ tableName,
273
+ field: key,
274
+ from: {
275
+ of: prop.of,
276
+ uuid: prop.uuid,
277
+ use: prop.use ?? "id"
278
+ },
279
+ to: resolvedValue
280
+ });
281
+ }
282
+ }
283
+ return row;
284
+ });
285
+ }
286
+ if (options?.cleanOrphans) {
287
+ const cleanOrphans = options.cleanOrphans;
288
+ const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [cleanOrphans];
289
+ const fkConditions = fkColumns.map((fkCol) => {
290
+ const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter((v) => v != null))];
291
+ return {
292
+ column: fkCol,
293
+ values: fkValues
294
+ };
295
+ });
296
+ if (fkConditions.every((fc) => fc.values.length > 0)) {
297
+ let deleteQuery = wdb(tableName);
298
+ for (const { column, values } of fkConditions) {
299
+ deleteQuery = deleteQuery.whereIn(column, values);
300
+ }
301
+ deleteQuery = deleteQuery.whereNotIn("id", allIds);
302
+ const deletedCount = await deleteQuery.delete();
303
+ Naite.t("puri:ub-clean-orphans", {
304
+ tableName,
305
+ cleanOrphans: fkColumns,
306
+ deletedCount
307
+ });
308
+ }
309
+ }
310
+ table.rows = [];
311
+ table.references.clear();
312
+ table.uniquesMap.clear();
313
+ Naite.t("puri:ub-upserted", {
314
+ tableName,
315
+ mode,
316
+ rowCount: allIds.length,
317
+ returnedIds: allIds
318
+ });
319
+ return allIds;
320
+ }
321
+ async updateBatch(wdb, tableName, options) {
322
+ options = {
323
+ ...options,
324
+ chunkSize: options?.chunkSize ?? 500,
325
+ where: options?.where ?? "id"
326
+ };
327
+ if (!this.hasTable(tableName)) {
328
+ return;
329
+ }
330
+ const table = this.tables.get(tableName);
331
+ if (!table) {
332
+ throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
333
+ } else if (table.rows.length === 0) {
334
+ return;
335
+ }
336
+ const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? "id"];
337
+ const rows = table.rows.map((_row) => {
338
+ const { uuid: _, ...row } = _row;
339
+ return row;
340
+ });
341
+ await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
342
+ Naite.t("puri:ub-batch-updated", {
343
+ tableName,
344
+ rowCount: rows.length,
345
+ whereColumns
346
+ });
347
+ table.rows = [];
348
+ table.references.clear();
349
+ table.uniquesMap.clear();
350
+ }
351
+ /**
352
+ * rows를 의존성 순서에 따라 레벨별로 그룹화
353
+ * - 자기 참조 없는 경우 : 모든 rows가 Level 0
354
+ * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
355
+ */
356
+ buildInsertLevels(rows, tableName) {
357
+ const hasSelfRef = rows.flatMap((row) => Object.values(row)).some((value) => isRefField(value) && value.of === tableName);
358
+ if (!hasSelfRef) return {
359
+ levels: [rows],
360
+ hasCircular: false
361
+ };
362
+ const rowByUuid = new Map();
363
+ for (const row of rows) {
364
+ const uuid = row.uuid;
365
+ if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
366
+ rowByUuid.set(uuid, row);
367
+ }
368
+ let pending = Array.from(rowByUuid.values());
369
+ const levels = [];
370
+ const inserted = new Set();
371
+ while (pending.length > 0) {
372
+ const currentLevel = [];
373
+ const nextPending = [];
374
+ for (const row of pending) {
375
+ const selfRefs = Object.values(row).filter((value) => isRefField(value) && value.of === tableName);
376
+ const canInsert = selfRefs.every((ref) => {
377
+ if (!rowByUuid.has(ref.uuid)) {
378
+ throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
379
+ }
380
+ return inserted.has(ref.uuid);
381
+ });
382
+ if (canInsert) {
383
+ currentLevel.push(row);
384
+ } else {
385
+ nextPending.push(row);
386
+ }
387
+ }
388
+ if (currentLevel.length === 0) return {
389
+ levels: [],
390
+ hasCircular: true
391
+ };
392
+ levels.push(currentLevel);
393
+ for (const row of currentLevel) {
394
+ inserted.add(row.uuid);
395
+ }
396
+ pending = nextPending;
397
+ }
398
+ return {
399
+ levels,
400
+ hasCircular: false
401
+ };
402
+ }
403
+ };
404
+ }));
468
405
 
469
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhYmFzZS91cHNlcnQtYnVpbGRlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBnZXRMb2dnZXIgfSBmcm9tIFwiQGxvZ3RhcGUvbG9ndGFwZVwiO1xuaW1wb3J0IHsgcmFuZG9tVVVJRCB9IGZyb20gXCJjcnlwdG9cIjtcbmltcG9ydCB0eXBlIHsgS25leCB9IGZyb20gXCJrbmV4XCI7XG5pbXBvcnQgeyBpc0FycmF5LCB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgRW50aXR5TWFuYWdlciB9IGZyb20gXCIuLi9lbnRpdHkvZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB7IE5haXRlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQgdHlwZSB7IERhdGFiYXNlRm9yZWlnbktleXMsIERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBFbnRpdHlJbmRleCB9IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgYXNzZXJ0RGVmaW5lZCwgY2h1bmssIG5vbk51bGxhYmxlIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgeyBiYXRjaFVwZGF0ZSwgdHlwZSBSb3dXaXRoSWQgfSBmcm9tIFwiLi9fYmF0Y2hfdXBkYXRlXCI7XG5pbXBvcnQgdHlwZSB7IENvbHVtbktleXMsIEZvcmVpZ25LZXlDb2x1bW5zLCBJZFR5cGUsIFRhYmxlTmFtZSB9IGZyb20gXCIuL3B1cmkudHlwZXNcIjtcblxuY29uc3QgbG9nZ2VyID0gZ2V0TG9nZ2VyKFtcInNvbmFtdVwiLCBcImludGVybmFsXCIsIFwidXBzZXJ0LWJ1aWxkZXJcIl0pO1xuXG4vKipcbiAqIEZLIO2DgOyehSDstpTroaDsnYQg7JyE7ZW0IERhdGFiYXNlRm9yZWlnbktleXMgZXhwb3J0XG4gKiAobW9kdWxlIGF1Z21lbnRhdGlvbiDsnpDrj5kg66Gc65OcIOuztOyepSlcbiAqL1xuZXhwb3J0IHR5cGUgeyBEYXRhYmFzZUZvcmVpZ25LZXlzIH07XG5cbnR5cGUgSW5oZXJpdGFibGVDb2x1bW5zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID1cbiAgVFRhYmxlIGV4dGVuZHMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQgPyBDb2x1bW5LZXlzPERhdGFiYXNlU2NoZW1hRXh0ZW5kW1RUYWJsZV0+IDogbmV2ZXI7XG5cbi8vIO2FjOydtOu4lCDrjbDsnbTthLAg7YOA7J6FXG50eXBlIFRhYmxlRGF0YSA9IHtcbiAgcmVmZXJlbmNlczogU2V0PHN0cmluZz47XG4gIHJvd3M6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W107XG4gIHVuaXF1ZUluZGV4ZXM6IEVudGl0eUluZGV4W107XG4gIHVuaXF1ZXNNYXA6IE1hcDxzdHJpbmcsIHN0cmluZz47XG4gIGpzb25Db2x1bW5zOiBzdHJpbmdbXTtcbn07XG5cbi8vIOywuOyhsCDtlYTrk5wg7YOA7J6FXG5leHBvcnQgdHlwZSBVQlJlZiA9IHtcbiAgdXVpZDogc3RyaW5nO1xuICBvZjogc3RyaW5nO1xuICB1c2U/OiBzdHJpbmc7XG59O1xuXG4vLyB1cHNlcnQg7Ji17IWYXG5leHBvcnQgdHlwZSBVcHNlcnRPcHRpb25zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG4gIGNsZWFuT3JwaGFucz86IEZvcmVpZ25LZXlDb2x1bW5zPFRUYWJsZT4gfCBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W107XG4gIGluaGVyaXQ/OiBJbmhlcml0YWJsZUNvbHVtbnM8VFRhYmxlPltdO1xufTtcblxuLy8gaW5zZXJ0T25seSDsmLXshZhcbmV4cG9ydCB0eXBlIEluc2VydE9ubHlPcHRpb25zID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gaXNSZWZGaWVsZChmaWVsZDogdW5rbm93bik6IGZpZWxkIGlzIFVCUmVmIHtcbiAgcmV0dXJuIChcbiAgICBmaWVsZCAhPT0gdW5kZWZpbmVkICYmXG4gICAgZmllbGQgIT09IG51bGwgJiZcbiAgICAoZmllbGQgYXMgVUJSZWYpPy5vZiAhPT0gdW5kZWZpbmVkICYmXG4gICAgKGZpZWxkIGFzIFVCUmVmKT8udXVpZCAhPT0gdW5kZWZpbmVkXG4gICk7XG59XG5cbmV4cG9ydCBjbGFzcyBVcHNlcnRCdWlsZGVyIHtcbiAgdGFibGVzOiBNYXA8c3RyaW5nLCBUYWJsZURhdGE+O1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnRhYmxlcyA9IG5ldyBNYXAoKTtcbiAgfVxuXG4gIGdldFRhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogVGFibGVEYXRhIHtcbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICh0YWJsZSkge1xuICAgICAgcmV0dXJuIHRhYmxlO1xuICAgIH1cblxuICAgIGNvbnN0IHRhYmxlU3BlYyA9ICgoKSA9PiB7XG4gICAgICB0cnkge1xuICAgICAgICByZXR1cm4gRW50aXR5TWFuYWdlci5nZXRUYWJsZVNwZWModGFibGVOYW1lKTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgIH1cbiAgICB9KSgpO1xuXG4gICAgY29uc3QgdGFibGVEYXRhID0ge1xuICAgICAgcmVmZXJlbmNlczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICByb3dzOiBbXSxcbiAgICAgIHVuaXF1ZUluZGV4ZXM6IHRhYmxlU3BlYz8udW5pcXVlSW5kZXhlcyA/PyBbXSxcbiAgICAgIHVuaXF1ZXNNYXA6IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+KCksXG4gICAgICBqc29uQ29sdW1uczogdGFibGVTcGVjPy5qc29uQ29sdW1ucyA/PyBbXSxcbiAgICB9O1xuICAgIHRoaXMudGFibGVzLnNldCh0YWJsZU5hbWUsIHRhYmxlRGF0YSk7XG4gICAgcmV0dXJuIHRhYmxlRGF0YTtcbiAgfVxuXG4gIGhhc1RhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMudGFibGVzLmhhcyh0YWJsZU5hbWUpO1xuICB9XG5cbiAgcmVnaXN0ZXI8VCBleHRlbmRzIHN0cmluZz4oXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgcm93OiB7XG4gICAgICBba2V5IGluIFRdPzogVUJSZWYgfCBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgYmlnaW50IHwgbnVsbCB8IG9iamVjdCB8IHVua25vd247XG4gICAgfSxcbiAgKTogVUJSZWYge1xuICAgIGNvbnN0IHRhYmxlID0gdGhpcy5nZXRUYWJsZSh0YWJsZU5hbWUpO1xuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCB1bmlxdWUg7J24642x7Iqk66W8IOyInO2ajO2VmOupsCDtgqQg7IOd7ISxXG4gICAgY29uc3QgdW5pcXVlS2V5cyA9IHRhYmxlLnVuaXF1ZUluZGV4ZXNcbiAgICAgIC5tYXAoKHVucUluZGV4KSA9PiB7XG4gICAgICAgIGNvbnN0IHVuaXF1ZUtleUFycmF5ID0gdW5xSW5kZXguY29sdW1ucy5tYXAoKHVucUNvbCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHZhbCA9IHJvd1t1bnFDb2wubmFtZSBhcyBrZXlvZiB0eXBlb2Ygcm93XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZCh2YWwpKSB7XG4gICAgICAgICAgICByZXR1cm4gdmFsLnV1aWQ7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJldHVybiByb3dbdW5xQ29sLm5hbWUgYXMga2V5b2YgdHlwZW9mIHJvd10gPz8gcmFuZG9tVVVJRCgpOyAvLyBudWxsYWJsZeyduCDqsr3smrAgdXVpZOuhnCDrnpzrjaTqsJIg7IK97J6FXG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICAvLyDqsJLsnbQg66qo65GQIG51bGzsnbgg6rK97JqwIO2CpCDsg53shLEg7Yyo7IqkXG4gICAgICAgIGlmICh1bmlxdWVLZXlBcnJheS5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdW5pcXVlS2V5QXJyYXkuam9pbihcIi0tLWRlbGltaXRlci0tXCIpO1xuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgLy8gdXVpZCDsg53shLEg66Gc7KeBXG4gICAgY29uc3QgeyB1dWlkLCBpc1JldXNlZCB9ID0gKCgpID0+IHtcbiAgICAgIC8vIO2CpOulvCDsiJztmoztlZjsl6wg7J2066+4IOyhtOyerO2VmOuKlCDtgqTqsIAg7J6I64qU7KeAIO2ZleyduFxuICAgICAgaWYgKHVuaXF1ZUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgICAgaWYgKHRhYmxlLnVuaXF1ZXNNYXAuaGFzKHVuaXF1ZUtleSkpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIHV1aWQ6IGFzc2VydERlZmluZWQodGFibGUudW5pcXVlc01hcC5nZXQodW5pcXVlS2V5KSwgXCJVbmlxdWUga2V5IG5vdCBmb3VuZFwiKSxcbiAgICAgICAgICAgICAgaXNSZXVzZWQ6IHRydWUsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyDssL7snYQg7IiYIOyXhuuKlCDqsr3smrAg7IOd7ISxXG4gICAgICByZXR1cm4geyB1dWlkOiByYW5kb21VVUlEKCksIGlzUmV1c2VkOiBmYWxzZSB9O1xuICAgIH0pKCk7XG5cbiAgICAvLyDrqqjrk6Ag7Jyg64uI7YGs7YKk7JeQIOuMgO2VtCDsnKDri4jtgazrp7Xsl5AgdXVpZCDsoIDsnqVcbiAgICBpZiAodW5pcXVlS2V5cy5sZW5ndGggPiAwKSB7XG4gICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgIHRhYmxlLnVuaXF1ZXNNYXAuc2V0KHVuaXF1ZUtleSwgdXVpZCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7J20IO2FjOydtOu4lOyXkCDsgqzsmqnrkJwgUmVmRmllbGTrpbwg7Iic7ZqM7ZWY7JesLCDtmITsnqwg7YWM7J2067iUIOygleuztOyXkCDslrTrlqQg7ZWE65Oc66W8IOywuOyhsO2VmOuKlOyngCDstpTqsIBcbiAgICAvLyDsnbQg7KCV67O066W8IOuCmOykkeyXkCDsuZjtmZjtlaAg65WMIOyCrOyaqVxuICAgIHJvdyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHJvdykubWFwKChbcm93S2V5LCByb3dWYWx1ZV0pID0+IHtcbiAgICAgICAgaWYgKGlzUmVmRmllbGQocm93VmFsdWUpKSB7XG4gICAgICAgICAgcm93VmFsdWUudXNlID8/PSBcImlkXCI7XG4gICAgICAgICAgdGFibGUucmVmZXJlbmNlcy5hZGQoYCR7cm93VmFsdWUub2Z9LiR7cm93VmFsdWUudXNlfWApO1xuICAgICAgICAgIHJldHVybiBbcm93S2V5LCByb3dWYWx1ZV07XG4gICAgICAgIH0gZWxzZSBpZiAodGFibGUuanNvbkNvbHVtbnMuaW5jbHVkZXMocm93S2V5KSAmJiByb3dWYWx1ZSAhPT0gbnVsbCkge1xuICAgICAgICAgIC8vIEpTT04g7Lus65+87J24IOqyveyasCBKU09OLnN0cmluZ2lmeSDsspjrpqwgKEtuZXjripQgSlNPTiDtg4DsnoXsnYQg7KeA7JuQ7ZWY7KeAIOyViuydjClcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgSlNPTi5zdHJpbmdpZnkocm93VmFsdWUpXTtcbiAgICAgICAgfSBlbHNlIGlmIChpc0FycmF5KHJvd1ZhbHVlKSkge1xuICAgICAgICAgIC8vIOuwsOyXtOydgCDqt7jrjIDroZwg7KCA7J6lXG4gICAgICAgICAgcmV0dXJuIFtyb3dLZXksIHJvd1ZhbHVlXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgcm93VmFsdWVdO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApIGFzIHsgW2tleSBpbiBUXT86IHVua25vd24gfTtcblxuICAgIHRhYmxlLnJvd3MucHVzaCh7XG4gICAgICB1dWlkLFxuICAgICAgLi4ucm93LFxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzdWx0OiBVQlJlZiA9IHtcbiAgICAgIG9mOiB0YWJsZU5hbWUsXG4gICAgICB1dWlkOiAocm93IGFzIHsgdXVpZD86IHN0cmluZyB9KS51dWlkID8/IHV1aWQsXG4gICAgfTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLXJlZ2lzdGVyXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHV1aWQ6IHJlc3VsdC51dWlkLFxuICAgICAgaXNVdWlkUmV1c2VkOiBpc1JldXNlZCxcbiAgICAgIHJvdyxcbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyB1cHNlcnQ8VFRhYmxlIGV4dGVuZHMgVGFibGVOYW1lPERhdGFiYXNlU2NoZW1hRXh0ZW5kPj4oXG4gICAgd2RiOiBLbmV4LFxuICAgIHRhYmxlTmFtZTogVFRhYmxlLFxuICAgIG9wdGlvbnM/OiBVcHNlcnRPcHRpb25zPFRUYWJsZT4sXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJ1cHNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyBpbnNlcnRPbmx5PFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+KFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IFRUYWJsZSxcbiAgICBvcHRpb25zPzogSW5zZXJ0T25seU9wdGlvbnMsXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJpbnNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyB1cHNlcnRPckluc2VydDxUVGFibGUgZXh0ZW5kcyBUYWJsZU5hbWU8RGF0YWJhc2VTY2hlbWFFeHRlbmQ+PihcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBUVGFibGUsXG4gICAgbW9kZTogXCJ1cHNlcnRcIiB8IFwiaW5zZXJ0XCIsXG4gICAgb3B0aW9ucz86IFVwc2VydE9wdGlvbnM8VFRhYmxlPixcbiAgKTogUHJvbWlzZTxJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXT4ge1xuICAgIGlmICh0aGlzLmhhc1RhYmxlKHRhYmxlTmFtZSkgPT09IGZhbHNlKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgdGFibGUgPSB0aGlzLnRhYmxlcy5nZXQodGFibGVOYW1lKTtcbiAgICBpZiAodGFibGUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBzZXJ0IOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgJHt0YWJsZU5hbWV97JeQIHVwc2VydCDtlaAg642w7J207YSw6rCAIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICBpZiAoXG4gICAgICB0YWJsZS5yb3dzLnNvbWUoKHJvdykgPT5cbiAgICAgICAgT2JqZWN0LmVudHJpZXMocm93KS5zb21lKChbLCB2YWx1ZV0pID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mICE9PSB0YWJsZU5hbWUpLFxuICAgICAgKVxuICAgICkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX0g7ZW06rKw65CY7KeAIOyViuydgCDssLjsobDqsIAg7J6I7Iq164uI64ukLmApO1xuICAgIH1cblxuICAgIC8vIOyghOyytCDthYzsnbTruJQg7Iic7ZqM7ZWY7JesIO2YhOyerCDthYzsnbTruJQg7LC47KGw7ZWY64qUIOuqqOuToCDthYzsnbTruJQg7LaU7LacXG4gICAgY29uc3QgeyByZWZlcmVuY2VzLCByZWZUYWJsZXMgfSA9IEFycmF5LmZyb20odGhpcy50YWJsZXMpLnJlZHVjZShcbiAgICAgIChyLCBbLCB0YWJsZV0pID0+IHtcbiAgICAgICAgY29uc3QgcmVmZXJlbmNlID0gQXJyYXkuZnJvbSh0YWJsZS5yZWZlcmVuY2VzLnZhbHVlcygpKS5maW5kKChyZWYpID0+XG4gICAgICAgICAgcmVmLmluY2x1ZGVzKGAke3RhYmxlTmFtZX0uYCksXG4gICAgICAgICk7XG4gICAgICAgIGlmIChyZWZlcmVuY2UpIHtcbiAgICAgICAgICByLnJlZmVyZW5jZXMucHVzaChyZWZlcmVuY2UpO1xuICAgICAgICAgIHIucmVmVGFibGVzLnB1c2godGFibGUpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9LFxuICAgICAge1xuICAgICAgICByZWZlcmVuY2VzOiBbXSBhcyBzdHJpbmdbXSxcbiAgICAgICAgcmVmVGFibGVzOiBbXSBhcyBUYWJsZURhdGFbXSxcbiAgICAgIH0sXG4gICAgKTtcbiAgICBjb25zdCBleHRyYWN0RmllbGRzID0gdW5pcXVlKHJlZmVyZW5jZXMpXG4gICAgICAubWFwKChyZWZlcmVuY2UpID0+IHJlZmVyZW5jZS5zcGxpdChcIi5cIilbMV0pXG4gICAgICAuZmlsdGVyKChmaWVsZCk6IGZpZWxkIGlzIHN0cmluZyA9PiBmaWVsZCAhPT0gdW5kZWZpbmVkKTtcblxuICAgIC8vIOydmOyhtOyEsSDsiJzshJzsl5Ag65Sw6528IOugiOuyqOuzhCDqt7jro7ntmZQgKOyekOq4sCDssLjsobDqsIAg7JeG7Jy866m0IExldmVsIDAg7ZWY64KYKVxuICAgIGNvbnN0IHsgbGV2ZWxzLCBoYXNDaXJjdWxhciB9ID0gdGhpcy5idWlsZEluc2VydExldmVscyh0YWJsZS5yb3dzLCB0YWJsZU5hbWUpO1xuXG4gICAgaWYgKGhhc0NpcmN1bGFyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfeyXkCDsiJztmZgg7J6Q6riwIOywuOyhsOqwgCDsnojsirXri4jri6QuYCk7XG4gICAgfVxuXG4gICAgY29uc3QgdXVpZE1hcCA9IG5ldyBNYXA8c3RyaW5nLCB1bmtub3duPigpO1xuICAgIGNvbnN0IGFsbElkczogKG51bWJlciB8IHN0cmluZylbXSA9IFtdO1xuXG4gICAgLy8g66CI67Ko67OE66GcIOyInOywqCDsspjrpqxcbiAgICBmb3IgKGxldCBsZXZlbElkeCA9IDA7IGxldmVsSWR4IDwgbGV2ZWxzLmxlbmd0aDsgbGV2ZWxJZHgrKykge1xuICAgICAgY29uc3QgbGV2ZWxSb3dzID0gbGV2ZWxzW2xldmVsSWR4XTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhcIlByb2Nlc3NpbmcgUXVlcnkgTGV2ZWw6IHtjdXJyZW50fSAvIHt0b3RhbH1cIiwge1xuICAgICAgICBjdXJyZW50OiBsZXZlbElkeCArIDEsXG4gICAgICAgIHRvdGFsOiBsZXZlbHMubGVuZ3RoLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIOydtOyghCDroIjrsqjsl5DshJwg7Ja77J2AIElE66GcIOyekOq4sCDssLjsobAg7ZW06rKwXG4gICAgICBjb25zdCByZXNvbHZlZFJvd3MgPSBsZXZlbFJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgY29uc3QgcmVzb2x2ZWQgPSB7IC4uLnJvdyB9O1xuICAgICAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhyb3cpKSB7XG4gICAgICAgICAgaWYgKGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHZhbHVlLnV1aWQpO1xuXG4gICAgICAgICAgICBpZiAoIXBhcmVudCkgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHt2YWx1ZS51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcblxuICAgICAgICAgICAgcmVzb2x2ZWRba2V5XSA9IChwYXJlbnQgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW3ZhbHVlLnVzZSA/PyBcImlkXCJdO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHZhbHVlLm9mLCB1dWlkOiB2YWx1ZS51dWlkLCB1c2U6IHZhbHVlLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkW2tleV0sXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJlc29sdmVkO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIO2YhOyerCDroIjrsqggdXBzZXJ0XG4gICAgICBjb25zdCBjaHVua1NpemUgPSBvcHRpb25zPy5jaHVua1NpemU7XG4gICAgICBjb25zdCBsZXZlbENodW5rcyA9IGNodW5rU2l6ZSA/IGNodW5rKHJlc29sdmVkUm93cywgY2h1bmtTaXplKSA6IFtyZXNvbHZlZFJvd3NdO1xuICAgICAgY29uc3Qgc2VsZWN0RmllbGRzID0gdW5pcXVlKFtcImlkXCIsIC4uLmV4dHJhY3RGaWVsZHNdKTtcblxuICAgICAgZm9yIChsZXQgaW5kZXggPSAwOyBpbmRleCA8IGxldmVsQ2h1bmtzLmxlbmd0aDsgaW5kZXgrKykge1xuICAgICAgICBjb25zdCBkYXRhQ2h1bmsgPSBsZXZlbENodW5rc1tpbmRleF07XG4gICAgICAgIGlmIChkYXRhQ2h1bmsubGVuZ3RoID09PSAwKSBjb250aW51ZTtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiUHJvY2Vzc2luZyBDaHVuazoge2N1cnJlbnR9IC8ge3RvdGFsfVwiLCB7XG4gICAgICAgICAgY3VycmVudDogaW5kZXggKyAxLFxuICAgICAgICAgIHRvdGFsOiBsZXZlbENodW5rcy5sZW5ndGgsXG4gICAgICAgIH0pO1xuXG4gICAgICAgIC8vIHV1aWTrpbwg67OE64+E66GcIOuztOq0gO2VmOqzoCwgRELsl5Ag7KCA7J6l7ZWgIOuNsOydtO2EsOyXkOyEnCDsoJzqsbBcbiAgICAgICAgY29uc3Qgb3JpZ2luYWxVdWlkcyA9IGRhdGFDaHVuay5tYXAoKHIpID0+IHIudXVpZCBhcyBzdHJpbmcpO1xuICAgICAgICBjb25zdCBkYXRhRm9yRGIgPSBkYXRhQ2h1bmsubWFwKCh7IHV1aWQsIC4uLnJlc3QgfSkgPT4gcmVzdCk7XG5cbiAgICAgICAgbGV0IHJlc3VsdFJvd3M6IHsgaWQ6IG51bWJlciB8IHN0cmluZzsgW2tleTogc3RyaW5nXTogdW5rbm93biB9W107XG5cbiAgICAgICAgaWYgKG1vZGUgPT09IFwiaW5zZXJ0XCIpIHtcbiAgICAgICAgICAvLyBJTlNFUlQg66qo65OcIC0gUkVUVVJOSU5HIOyCrOyaqVxuICAgICAgICAgIHJlc3VsdFJvd3MgPSBhd2FpdCB3ZGIuaW5zZXJ0KGRhdGFGb3JEYikuaW50byh0YWJsZU5hbWUpLnJldHVybmluZyhzZWxlY3RGaWVsZHMpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIFVQU0VSVCDrqqjrk5wgLSBpZCDsl4bripQgcm9365Ok7J2YIGlk66W8IOyCrOyghCDsobDtmozroZwg7LGE7Jqw6riwXG4gICAgICAgICAgY29uc3Qgcm93c1dpdGhvdXRJZCA9IGRhdGFGb3JEYi5maWx0ZXIoKHJvdykgPT4gIXJvdy5pZCk7XG5cbiAgICAgICAgICBpZiAocm93c1dpdGhvdXRJZC5sZW5ndGggPiAwICYmIHRhYmxlLnVuaXF1ZUluZGV4ZXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgLy8g66qo65OgIHVuaXF1ZUluZGV4ZXProZwg6riw7KG0IOugiOy9lOuTnCDsobDtmoxcbiAgICAgICAgICAgIGZvciAoY29uc3QgdW5pcXVlSW5kZXggb2YgdGFibGUudW5pcXVlSW5kZXhlcykge1xuICAgICAgICAgICAgICBjb25zdCBjb2x1bW5zID0gdW5pcXVlSW5kZXguY29sdW1ucy5tYXAoKGMpID0+IGMubmFtZSk7XG5cbiAgICAgICAgICAgICAgLy8g7KGw7ZqM7ZWgIOyhsOqxtOuTpCDstpTstpwgKOqwgSByb3fsnZggdW5pcXVlIOy7rOufvCDqsJLrk6QpXG4gICAgICAgICAgICAgIGNvbnN0IGNvbmRpdGlvbnM6IHVua25vd25bXVtdID0gW107XG4gICAgICAgICAgICAgIGZvciAoY29uc3Qgcm93IG9mIHJvd3NXaXRob3V0SWQpIHtcbiAgICAgICAgICAgICAgICBjb25zdCB2YWx1ZXMgPSBjb2x1bW5zLm1hcCgoY29sKSA9PiByb3dbY29sXSk7XG4gICAgICAgICAgICAgICAgLy8gbnVsbOydtCDtj6ztlajrkJwg7KGw6rG07J2AIOygnOyZuCAoUG9zdGdyZVNRTCBVTklRVUXripQgTlVMTCDrrLTsi5wpXG4gICAgICAgICAgICAgICAgaWYgKCF2YWx1ZXMuc29tZSgodikgPT4gdiA9PSBudWxsKSkge1xuICAgICAgICAgICAgICAgICAgY29uZGl0aW9ucy5wdXNoKHZhbHVlcyk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgaWYgKGNvbmRpdGlvbnMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgICAgICAvLyDrsLDsuZggU0VMRUNUXG4gICAgICAgICAgICAgIGNvbnN0IGV4aXN0aW5nUm93cyA9IChhd2FpdCB3ZGIodGFibGVOYW1lKVxuICAgICAgICAgICAgICAgIC53aGVyZUluKGNvbHVtbnMsIGNvbmRpdGlvbnMgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdKVxuICAgICAgICAgICAgICAgIC5zZWxlY3QoXCJpZFwiLCAuLi5jb2x1bW5zKSkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXTtcblxuICAgICAgICAgICAgICAvLyBNYXAg7IOd7ISxOiB1bmlxdWUg7Lus65+8IOyhsO2VqSDihpIgaWRcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPSBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGNvbnN0IGV4aXN0aW5nIG9mIGV4aXN0aW5nUm93cykge1xuICAgICAgICAgICAgICAgIGNvbnN0IGtleSA9IGNvbHVtbnNcbiAgICAgICAgICAgICAgICAgIC5tYXAoKGNvbCkgPT4gU3RyaW5nKGV4aXN0aW5nW2NvbF0gPz8gXCJcIikpXG4gICAgICAgICAgICAgICAgICAuam9pbihcIi0tLWRlbGltaXRlci0tLVwiKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpZCA9IGV4aXN0aW5nLmlkO1xuICAgICAgICAgICAgICAgIGlmICh0eXBlb2YgaWQgPT09IFwibnVtYmVyXCIgfHwgdHlwZW9mIGlkID09PSBcInN0cmluZ1wiKSB7XG4gICAgICAgICAgICAgICAgICBleGlzdGluZ01hcC5zZXQoa2V5LCBpZCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgLy8gaWQg7JeG64qUIHJvd+uTpOyXkCDrp6Tsua3rkJjripQgaWQg7LGE7Jqw6riwXG4gICAgICAgICAgICAgIGZvciAoY29uc3Qgcm93IG9mIHJvd3NXaXRob3V0SWQpIHtcbiAgICAgICAgICAgICAgICBpZiAocm93LmlkKSBjb250aW51ZTsgLy8g7J2066+4IOuLpOuluCB1bmlxdWVJbmRleOyXkOyEnCDssYTsm4zsp4Qg6rK97JqwIOyKpO2CtVxuXG4gICAgICAgICAgICAgICAgY29uc3Qga2V5ID0gY29sdW1ucy5tYXAoKGNvbCkgPT4gU3RyaW5nKHJvd1tjb2xdID8/IFwiXCIpKS5qb2luKFwiLS0tZGVsaW1pdGVyLS0tXCIpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBleGlzdGluZ01hcC5nZXQoa2V5KTtcblxuICAgICAgICAgICAgICAgIGlmIChleGlzdGluZ0lkKSB7XG4gICAgICAgICAgICAgICAgICByb3cuaWQgPSBleGlzdGluZ0lkO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIG9uQ29uZmxpY3TripQgaWTrp4wg7IKs7JqpICjrqqjrk6AgdW5pcXVlSW5kZXhlc+uKlCDsnbTrr7gg7IKs7KCEIOyhsO2ajOuhnCDsspjrpqzrkKgpXG4gICAgICAgICAgY29uc3QgY29uZmxpY3RDb2x1bW5zID0gW1wiaWRcIl07XG5cbiAgICAgICAgICBjb25zdCBhbGxDb2x1bW5zID0gT2JqZWN0LmtleXMoZGF0YUZvckRiWzBdKTtcbiAgICAgICAgICBsZXQgdXBkYXRlQ29sdW1ucyA9IGFsbENvbHVtbnMuZmlsdGVyKChjKSA9PiBjICE9PSBcImlkXCIpO1xuXG4gICAgICAgICAgLy8gaW5oZXJpdCDsmLXshZgg7LKY66asIC0gaW5oZXJpdCDsu6zrn7zsnYAgdXBkYXRlIOuMgOyDgeyXkOyEnCDsoJzsmbhcbiAgICAgICAgICBpZiAob3B0aW9ucz8uaW5oZXJpdD8ubGVuZ3RoKSB7XG4gICAgICAgICAgICBjb25zdCBpbmhlcml0Q29sdW1ucyA9IG9wdGlvbnMuaW5oZXJpdCBhcyBzdHJpbmdbXTtcblxuICAgICAgICAgICAgY29uc3QgZXhjbHVkZWRGcm9tVXBkYXRlID0gdXBkYXRlQ29sdW1ucy5maWx0ZXIoKGMpID0+IGluaGVyaXRDb2x1bW5zLmluY2x1ZGVzKGMpKTtcbiAgICAgICAgICAgIHVwZGF0ZUNvbHVtbnMgPSB1cGRhdGVDb2x1bW5zLmZpbHRlcigoYykgPT4gIWluaGVyaXRDb2x1bW5zLmluY2x1ZGVzKGMpKTtcblxuICAgICAgICAgICAgLy8g7Iuk7KCc66GcIOygnOyZuOuQnCDsu6zrn7wg66Gc6rmFXG4gICAgICAgICAgICBpZiAoZXhjbHVkZWRGcm9tVXBkYXRlLmxlbmd0aCkge1xuICAgICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1pbmhlcml0XCIsIHtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgaW5oZXJpdENvbHVtbnMsXG4gICAgICAgICAgICAgICAgZXhjbHVkZWRGcm9tVXBkYXRlLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyB1cGRhdGVDb2x1bW5z6rCAIOu5hOyWtOyeiOyWtOuPhCBtZXJnZSgp66W8IOyCrOyaqe2VmOyXrCDrqqjrk6Ag7ZaJ7J20IFJFVFVSTklOR+uQmOuPhOuhnSDrs7TsnqVcbiAgICAgICAgICBjb25zdCBtZXJnZUNvbHVtbnMgPSB1cGRhdGVDb2x1bW5zLmxlbmd0aCA/IHVwZGF0ZUNvbHVtbnMgOiBjb25mbGljdENvbHVtbnM7XG5cbiAgICAgICAgICByZXN1bHRSb3dzID0gYXdhaXQgd2RiXG4gICAgICAgICAgICAuaW5zZXJ0KGRhdGFGb3JEYilcbiAgICAgICAgICAgIC5pbnRvKHRhYmxlTmFtZSlcbiAgICAgICAgICAgIC5vbkNvbmZsaWN0KGNvbmZsaWN0Q29sdW1ucylcbiAgICAgICAgICAgIC5tZXJnZShtZXJnZUNvbHVtbnMpXG4gICAgICAgICAgICAucmV0dXJuaW5nKHNlbGVjdEZpZWxkcyk7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAob3JpZ2luYWxVdWlkcy5sZW5ndGggIT09IHJlc3VsdFJvd3MubGVuZ3RoKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX06IHJlZ2lzdGVyL3JldHVybmluZyDrtojsnbzsuZhgKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgcmVzdWx0Um93cy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgIHV1aWRNYXAuc2V0KG9yaWdpbmFsVXVpZHNbaV0sIHJlc3VsdFJvd3NbaV0pO1xuICAgICAgICAgIGFsbElkcy5wdXNoKHJlc3VsdFJvd3NbaV0uaWQpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lCDssLjsobDrpbwg7Iuk7KCcIOuwuOulmOuhnCDrs4Dqsr1cbiAgICBmb3IgKGNvbnN0IHRhYmxlIG9mIHJlZlRhYmxlcykge1xuICAgICAgdGFibGUucm93cyA9IHRhYmxlLnJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMocm93KSkge1xuICAgICAgICAgIGNvbnN0IHByb3AgPSByb3dba2V5XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZChwcm9wKSAmJiBwcm9wLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHByb3AudXVpZCk7XG4gICAgICAgICAgICBpZiAoIXBhcmVudCkge1xuICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKHByb3ApO1xuICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYOyhtOyerO2VmOyngCDslYrripQgdXVpZCAke3Byb3AudXVpZH0gLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCByZXNvbHZlZFZhbHVlID0gKHBhcmVudCBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbcHJvcC51c2UgPz8gXCJpZFwiXTtcbiAgICAgICAgICAgIHJvd1trZXldID0gcmVzb2x2ZWRWYWx1ZTtcblxuICAgICAgICAgICAgTmFpdGUudChcInB1cmk6dWItcmVmLXJlc29sdmVkXCIsIHtcbiAgICAgICAgICAgICAgdGFibGVOYW1lLFxuICAgICAgICAgICAgICBmaWVsZDoga2V5LFxuICAgICAgICAgICAgICBmcm9tOiB7IG9mOiBwcm9wLm9mLCB1dWlkOiBwcm9wLnV1aWQsIHVzZTogcHJvcC51c2UgPz8gXCJpZFwiIH0sXG4gICAgICAgICAgICAgIHRvOiByZXNvbHZlZFZhbHVlLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiByb3c7XG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBpZiAob3B0aW9ucz8uY2xlYW5PcnBoYW5zKSB7XG4gICAgICBjb25zdCBjbGVhbk9ycGhhbnMgPSBvcHRpb25zLmNsZWFuT3JwaGFucztcbiAgICAgIGNvbnN0IGZrQ29sdW1ucyA9IGlzQXJyYXkoY2xlYW5PcnBoYW5zKVxuICAgICAgICA/IChjbGVhbk9ycGhhbnMgYXMgRm9yZWlnbktleUNvbHVtbnM8VFRhYmxlPltdKVxuICAgICAgICA6IFtjbGVhbk9ycGhhbnMgYXMgRm9yZWlnbktleUNvbHVtbnM8VFRhYmxlPl07XG5cbiAgICAgIC8vIO2YhOyerCByZWdpc3RlcuuQnCDroIjsvZTrk5zrk6TsnZggRksg6rCS65OkIOy2lOy2nFxuICAgICAgY29uc3QgZmtDb25kaXRpb25zID0gZmtDb2x1bW5zLm1hcCgoZmtDb2wpID0+IHtcbiAgICAgICAgY29uc3QgZmtWYWx1ZXMgPSBbLi4ubmV3IFNldCh0YWJsZS5yb3dzLm1hcCgocm93KSA9PiByb3dbZmtDb2xdKS5maWx0ZXIoKHYpID0+IHYgIT0gbnVsbCkpXTtcbiAgICAgICAgcmV0dXJuIHsgY29sdW1uOiBma0NvbCwgdmFsdWVzOiBma1ZhbHVlcyB9O1xuICAgICAgfSk7XG5cbiAgICAgIC8vIOuqqOuToCBGSyDsu6zrn7zsl5Ag6rCS7J20IOyeiOuKlCDqsr3smrDsl5Drp4wg7IKt7KCcIOyLpO2WiVxuICAgICAgaWYgKGZrQ29uZGl0aW9ucy5ldmVyeSgoZmMpID0+IGZjLnZhbHVlcy5sZW5ndGggPiAwKSkge1xuICAgICAgICBsZXQgZGVsZXRlUXVlcnkgPSB3ZGIodGFibGVOYW1lKTtcblxuICAgICAgICAvLyDqsIEgRksg7Lus65+87JeQIOuMgO2VnCBXSEVSRSBJTiDsobDqsbQg7LaU6rCAXG4gICAgICAgIGZvciAoY29uc3QgeyBjb2x1bW4sIHZhbHVlcyB9IG9mIGZrQ29uZGl0aW9ucykge1xuICAgICAgICAgIGRlbGV0ZVF1ZXJ5ID0gZGVsZXRlUXVlcnkud2hlcmVJbihjb2x1bW4sIHZhbHVlcyk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyDrsKnquIggdXBzZXJ07ZWcIElE64qUIOygnOyZuFxuICAgICAgICBkZWxldGVRdWVyeSA9IGRlbGV0ZVF1ZXJ5LndoZXJlTm90SW4oXCJpZFwiLCBhbGxJZHMpO1xuXG4gICAgICAgIGNvbnN0IGRlbGV0ZWRDb3VudCA9IGF3YWl0IGRlbGV0ZVF1ZXJ5LmRlbGV0ZSgpO1xuXG4gICAgICAgIE5haXRlLnQoXCJwdXJpOnViLWNsZWFuLW9ycGhhbnNcIiwge1xuICAgICAgICAgIHRhYmxlTmFtZSxcbiAgICAgICAgICBjbGVhbk9ycGhhbnM6IGZrQ29sdW1ucyxcbiAgICAgICAgICBkZWxldGVkQ291bnQsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIO2VtOuLuSDthYzsnbTruJTsnZgg642w7J207YSwIOy0iOq4sO2ZlFxuICAgIHRhYmxlLnJvd3MgPSBbXTtcbiAgICB0YWJsZS5yZWZlcmVuY2VzLmNsZWFyKCk7XG4gICAgdGFibGUudW5pcXVlc01hcC5jbGVhcigpO1xuXG4gICAgTmFpdGUudChcInB1cmk6dWItdXBzZXJ0ZWRcIiwge1xuICAgICAgdGFibGVOYW1lLFxuICAgICAgbW9kZSxcbiAgICAgIHJvd0NvdW50OiBhbGxJZHMubGVuZ3RoLFxuICAgICAgcmV0dXJuZWRJZHM6IGFsbElkcyxcbiAgICB9KTtcblxuICAgIHJldHVybiBhbGxJZHMgYXMgSWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W107XG4gIH1cblxuICBhc3luYyB1cGRhdGVCYXRjaChcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIGNodW5rU2l6ZT86IG51bWJlcjtcbiAgICAgIHdoZXJlPzogc3RyaW5nIHwgc3RyaW5nW107XG4gICAgfSxcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgb3B0aW9ucyA9IHtcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgICBjaHVua1NpemU6IG9wdGlvbnM/LmNodW5rU2l6ZSA/PyA1MDAsXG4gICAgICB3aGVyZTogb3B0aW9ucz8ud2hlcmUgPz8gXCJpZFwiLFxuICAgIH07XG5cbiAgICBpZiAodGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpID09PSBmYWxzZSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICghdGFibGUpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihg65Ox66Gd65CY7KeAIOyViuydgCDthYzsnbTruJQgJHt0YWJsZU5hbWV97JeQIHVwZGF0ZUJhdGNoIOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCB3aGVyZUNvbHVtbnMgPSBBcnJheS5pc0FycmF5KG9wdGlvbnMud2hlcmUpID8gb3B0aW9ucy53aGVyZSA6IFtvcHRpb25zLndoZXJlID8/IFwiaWRcIl07XG4gICAgY29uc3Qgcm93cyA9IHRhYmxlLnJvd3MubWFwKChfcm93KSA9PiB7XG4gICAgICBjb25zdCB7IHV1aWQ6IF8sIC4uLnJvdyB9ID0gX3JvdzsgLy8gdXVpZCDsoJzsmbhcbiAgICAgIHJldHVybiByb3cgYXMgUm93V2l0aElkPHN0cmluZz47XG4gICAgfSk7XG5cbiAgICBhd2FpdCBiYXRjaFVwZGF0ZSh3ZGIsIHRhYmxlTmFtZSwgd2hlcmVDb2x1bW5zLCByb3dzLCBvcHRpb25zLmNodW5rU2l6ZSk7XG5cbiAgICBOYWl0ZS50KFwicHVyaTp1Yi1iYXRjaC11cGRhdGVkXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHJvd0NvdW50OiByb3dzLmxlbmd0aCxcbiAgICAgIHdoZXJlQ29sdW1ucyxcbiAgICB9KTtcblxuICAgIC8vIHVwZGF0ZUJhdGNoIOyZhOujjCDtm4Qg7LKY66as65CcIOuNsOydtO2EsCDsoJzqsbBcbiAgICB0YWJsZS5yb3dzID0gW107XG4gICAgdGFibGUucmVmZXJlbmNlcy5jbGVhcigpO1xuICAgIHRhYmxlLnVuaXF1ZXNNYXAuY2xlYXIoKTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gUHJpdmF0ZSBIZWxwZXIgTWV0aG9kc1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqXG4gICAqIHJvd3Prpbwg7J2Y7KG07ISxIOyInOyEnOyXkCDrlLDrnbwg66CI67Ko67OE66GcIOq3uOujue2ZlFxuICAgKiAtIOyekOq4sCDssLjsobAg7JeG64qUIOqyveyasCA6IOuqqOuToCByb3dz6rCAIExldmVsIDBcbiAgICogLSDsnpDquLAg7LC47KGwIOyeiOuKlCDqsr3smrAgOiDsnpDquLAg7LC47KGwIOq0gOqzhOulvCDsnITsg4Eg7KCV66Cs7ZWY7JesIOugiOuyqOuzhOuhnCDqt7jro7ntmZRcbiAgICovXG4gIHByaXZhdGUgYnVpbGRJbnNlcnRMZXZlbHMoXG4gICAgcm93czogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSxcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgKTogeyBsZXZlbHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W11bXTsgaGFzQ2lyY3VsYXI6IGJvb2xlYW4gfSB7XG4gICAgLy8gMS4g7J6Q6riwIOywuOyhsOqwgCDsl4bsnLzrqbQg7ZWcIOugiOuyqOuhnCDsspjrpqxcbiAgICBjb25zdCBoYXNTZWxmUmVmID0gcm93c1xuICAgICAgLmZsYXRNYXAoKHJvdykgPT4gT2JqZWN0LnZhbHVlcyhyb3cpKVxuICAgICAgLnNvbWUoKHZhbHVlKSA9PiBpc1JlZkZpZWxkKHZhbHVlKSAmJiB2YWx1ZS5vZiA9PT0gdGFibGVOYW1lKTtcbiAgICBpZiAoIWhhc1NlbGZSZWYpIHJldHVybiB7IGxldmVsczogW3Jvd3NdLCBoYXNDaXJjdWxhcjogZmFsc2UgfTtcblxuICAgIC8vIDIuIHV1aWQg4oaSIHJvdyDrp6TtlZEgKOykkeuztSB1dWlkIOuwqeyngClcbiAgICBjb25zdCByb3dCeVV1aWQgPSBuZXcgTWFwPHN0cmluZywgUmVjb3JkPHN0cmluZywgdW5rbm93bj4+KCk7XG4gICAgZm9yIChjb25zdCByb3cgb2Ygcm93cykge1xuICAgICAgY29uc3QgdXVpZCA9IHJvdy51dWlkIGFzIHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICAgIGlmICghdXVpZCkgdGhyb3cgbmV3IEVycm9yKGBidWlsZEluc2VydExldmVsczogdXVpZOqwgCDsl4bripQgcm93IC0tIGluICR7dGFibGVOYW1lfWApO1xuICAgICAgcm93QnlVdWlkLnNldCh1dWlkLCByb3cpO1xuICAgIH1cblxuICAgIGxldCBwZW5kaW5nID0gQXJyYXkuZnJvbShyb3dCeVV1aWQudmFsdWVzKCkpO1xuICAgIGNvbnN0IGxldmVsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdID0gW107XG4gICAgY29uc3QgaW5zZXJ0ZWQgPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgIC8vIDMuIOugiOuyqOuzhCDrtoTrpZhcbiAgICB3aGlsZSAocGVuZGluZy5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBjdXJyZW50TGV2ZWw6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W10gPSBbXTtcbiAgICAgIGNvbnN0IG5leHRQZW5kaW5nOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdID0gW107XG5cbiAgICAgIGZvciAoY29uc3Qgcm93IG9mIHBlbmRpbmcpIHtcbiAgICAgICAgLy8g7J20IHJvd+qwgCDssLjsobDtlZjripQg7J6Q6riwIOywuOyhsOuTpFxuICAgICAgICBjb25zdCBzZWxmUmVmcyA9IE9iamVjdC52YWx1ZXMocm93KS5maWx0ZXIoXG4gICAgICAgICAgKHZhbHVlKSA9PiBpc1JlZkZpZWxkKHZhbHVlKSAmJiB2YWx1ZS5vZiA9PT0gdGFibGVOYW1lLFxuICAgICAgICApIGFzIFVCUmVmW107XG5cbiAgICAgICAgLy8g7LC47KGw7ZWY64qUIOuqqOuToCB1dWlk6rCAIOydtOuvuCBpbnNlcnRlZOyXkCDsnojslrTslbwg7J2067KIIOugiOuyqOyXkCDtj6ztlahcbiAgICAgICAgY29uc3QgY2FuSW5zZXJ0ID0gc2VsZlJlZnMuZXZlcnkoKHJlZikgPT4ge1xuICAgICAgICAgIGlmICghcm93QnlVdWlkLmhhcyhyZWYudXVpZCkpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihg7KG07J6s7ZWY7KeAIOyViuuKlCB1dWlkICR7cmVmLnV1aWR9IC0tIGluICR7dGFibGVOYW1lfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gaW5zZXJ0ZWQuaGFzKHJlZi51dWlkKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKGNhbkluc2VydCkge1xuICAgICAgICAgIGN1cnJlbnRMZXZlbC5wdXNoKHJvdyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgbmV4dFBlbmRpbmcucHVzaChyb3cpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIOyInO2ZmCDssLjsobAg6rCQ7KeAXG4gICAgICBpZiAoY3VycmVudExldmVsLmxlbmd0aCA9PT0gMCkgcmV0dXJuIHsgbGV2ZWxzOiBbXSwgaGFzQ2lyY3VsYXI6IHRydWUgfTtcblxuICAgICAgLy8g66CI67KoIO2ZleyglSArIGluc2VydGVkIOqwseyLoFxuICAgICAgbGV2ZWxzLnB1c2goY3VycmVudExldmVsKTtcbiAgICAgIGZvciAoY29uc3Qgcm93IG9mIGN1cnJlbnRMZXZlbCkge1xuICAgICAgICBpbnNlcnRlZC5hZGQocm93LnV1aWQgYXMgc3RyaW5nKTtcbiAgICAgIH1cblxuICAgICAgcGVuZGluZyA9IG5leHRQZW5kaW5nO1xuICAgIH1cblxuICAgIHJldHVybiB7IGxldmVscywgaGFzQ2lyY3VsYXI6IGZhbHNlIH07XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJnZXRMb2dnZXIiLCJyYW5kb21VVUlEIiwiaXNBcnJheSIsInVuaXF1ZSIsIkVudGl0eU1hbmFnZXIiLCJOYWl0ZSIsImFzc2VydERlZmluZWQiLCJjaHVuayIsIm5vbk51bGxhYmxlIiwiYmF0Y2hVcGRhdGUiLCJsb2dnZXIiLCJpc1JlZkZpZWxkIiwiZmllbGQiLCJ1bmRlZmluZWQiLCJvZiIsInV1aWQiLCJVcHNlcnRCdWlsZGVyIiwidGFibGVzIiwiTWFwIiwiZ2V0VGFibGUiLCJ0YWJsZU5hbWUiLCJ0YWJsZSIsImdldCIsInRhYmxlU3BlYyIsImdldFRhYmxlU3BlYyIsInRhYmxlRGF0YSIsInJlZmVyZW5jZXMiLCJTZXQiLCJyb3dzIiwidW5pcXVlSW5kZXhlcyIsInVuaXF1ZXNNYXAiLCJqc29uQ29sdW1ucyIsInNldCIsImhhc1RhYmxlIiwiaGFzIiwicmVnaXN0ZXIiLCJyb3ciLCJ1bmlxdWVLZXlzIiwibWFwIiwidW5xSW5kZXgiLCJ1bmlxdWVLZXlBcnJheSIsImNvbHVtbnMiLCJ1bnFDb2wiLCJ2YWwiLCJuYW1lIiwibGVuZ3RoIiwiam9pbiIsImZpbHRlciIsImlzUmV1c2VkIiwidW5pcXVlS2V5IiwiT2JqZWN0IiwiZnJvbUVudHJpZXMiLCJlbnRyaWVzIiwicm93S2V5Iiwicm93VmFsdWUiLCJ1c2UiLCJhZGQiLCJpbmNsdWRlcyIsIkpTT04iLCJzdHJpbmdpZnkiLCJwdXNoIiwicmVzdWx0IiwidCIsImlzVXVpZFJldXNlZCIsInVwc2VydCIsIndkYiIsIm9wdGlvbnMiLCJ1cHNlcnRPckluc2VydCIsImluc2VydE9ubHkiLCJtb2RlIiwiRXJyb3IiLCJzb21lIiwidmFsdWUiLCJyZWZUYWJsZXMiLCJBcnJheSIsImZyb20iLCJyZWR1Y2UiLCJyIiwicmVmZXJlbmNlIiwidmFsdWVzIiwiZmluZCIsInJlZiIsImV4dHJhY3RGaWVsZHMiLCJzcGxpdCIsImxldmVscyIsImhhc0NpcmN1bGFyIiwiYnVpbGRJbnNlcnRMZXZlbHMiLCJ1dWlkTWFwIiwiYWxsSWRzIiwibGV2ZWxJZHgiLCJsZXZlbFJvd3MiLCJkZWJ1ZyIsImN1cnJlbnQiLCJ0b3RhbCIsInJlc29sdmVkUm93cyIsInJlc29sdmVkIiwia2V5IiwicGFyZW50IiwidG8iLCJjaHVua1NpemUiLCJsZXZlbENodW5rcyIsInNlbGVjdEZpZWxkcyIsImluZGV4IiwiZGF0YUNodW5rIiwib3JpZ2luYWxVdWlkcyIsImRhdGFGb3JEYiIsInJlc3QiLCJyZXN1bHRSb3dzIiwiaW5zZXJ0IiwiaW50byIsInJldHVybmluZyIsInJvd3NXaXRob3V0SWQiLCJpZCIsInVuaXF1ZUluZGV4IiwiYyIsImNvbmRpdGlvbnMiLCJjb2wiLCJ2IiwiZXhpc3RpbmdSb3dzIiwid2hlcmVJbiIsInNlbGVjdCIsImV4aXN0aW5nTWFwIiwiZXhpc3RpbmciLCJTdHJpbmciLCJleGlzdGluZ0lkIiwiY29uZmxpY3RDb2x1bW5zIiwiYWxsQ29sdW1ucyIsImtleXMiLCJ1cGRhdGVDb2x1bW5zIiwiaW5oZXJpdCIsImluaGVyaXRDb2x1bW5zIiwiZXhjbHVkZWRGcm9tVXBkYXRlIiwibWVyZ2VDb2x1bW5zIiwib25Db25mbGljdCIsIm1lcmdlIiwiaSIsInByb3AiLCJjb25zb2xlIiwiZXJyb3IiLCJyZXNvbHZlZFZhbHVlIiwiY2xlYW5PcnBoYW5zIiwiZmtDb2x1bW5zIiwiZmtDb25kaXRpb25zIiwiZmtDb2wiLCJma1ZhbHVlcyIsImNvbHVtbiIsImV2ZXJ5IiwiZmMiLCJkZWxldGVRdWVyeSIsIndoZXJlTm90SW4iLCJkZWxldGVkQ291bnQiLCJkZWxldGUiLCJjbGVhciIsInJvd0NvdW50IiwicmV0dXJuZWRJZHMiLCJ1cGRhdGVCYXRjaCIsIndoZXJlIiwid2hlcmVDb2x1bW5zIiwiX3JvdyIsIl8iLCJoYXNTZWxmUmVmIiwiZmxhdE1hcCIsInJvd0J5VXVpZCIsInBlbmRpbmciLCJpbnNlcnRlZCIsImN1cnJlbnRMZXZlbCIsIm5leHRQZW5kaW5nIiwic2VsZlJlZnMiLCJjYW5JbnNlcnQiXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFNBQVMsUUFBUSxtQkFBbUI7QUFDN0MsU0FBU0MsVUFBVSxRQUFRLFNBQVM7QUFFcEMsU0FBU0MsT0FBTyxFQUFFQyxNQUFNLFFBQVEsVUFBVTtBQUMxQyxTQUFTQyxhQUFhLFFBQVEsOEJBQTJCO0FBQ3pELFNBQVNDLEtBQUssUUFBUSxvQkFBaUI7QUFFdkMsU0FBU0MsYUFBYSxFQUFFQyxLQUFLLEVBQUVDLFdBQVcsUUFBUSxvQkFBaUI7QUFDbkUsU0FBU0MsV0FBVyxRQUF3QixxQkFBa0I7QUFHOUQsTUFBTUMsU0FBU1YsVUFBVTtJQUFDO0lBQVU7SUFBWTtDQUFpQjtBQXVDakUsT0FBTyxTQUFTVyxXQUFXQyxLQUFjO0lBQ3ZDLE9BQ0VBLFVBQVVDLGFBQ1ZELFVBQVUsUUFDVixBQUFDQSxPQUFpQkUsT0FBT0QsYUFDekIsQUFBQ0QsT0FBaUJHLFNBQVNGO0FBRS9CO0FBRUEsT0FBTyxNQUFNRztJQUNYQyxPQUErQjtJQUMvQixhQUFjO1FBQ1osSUFBSSxDQUFDQSxNQUFNLEdBQUcsSUFBSUM7SUFDcEI7SUFFQUMsU0FBU0MsU0FBaUIsRUFBYTtRQUNyQyxNQUFNQyxRQUFRLElBQUksQ0FBQ0osTUFBTSxDQUFDSyxHQUFHLENBQUNGO1FBQzlCLElBQUlDLE9BQU87WUFDVCxPQUFPQTtRQUNUO1FBRUEsTUFBTUUsWUFBWSxBQUFDLENBQUE7WUFDakIsSUFBSTtnQkFDRixPQUFPbkIsY0FBY29CLFlBQVksQ0FBQ0o7WUFDcEMsRUFBRSxPQUFNO2dCQUNOLE9BQU87WUFDVDtRQUNGLENBQUE7UUFFQSxNQUFNSyxZQUFZO1lBQ2hCQyxZQUFZLElBQUlDO1lBQ2hCQyxNQUFNLEVBQUU7WUFDUkMsZUFBZU4sV0FBV00saUJBQWlCLEVBQUU7WUFDN0NDLFlBQVksSUFBSVo7WUFDaEJhLGFBQWFSLFdB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
406
+ //#endregion
407
+ init_upsert_builder();
408
+ export { UpsertBuilder, init_upsert_builder, isRefField };
409
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBzZXJ0LWJ1aWxkZXIuanMiLCJuYW1lcyI6WyJyZXN1bHQ6IFVCUmVmIiwidGFibGUiLCJhbGxJZHM6IChudW1iZXIgfCBzdHJpbmcpW10iLCJyZXN1bHRSb3dzOiB7IGlkOiBudW1iZXIgfCBzdHJpbmc7IFtrZXk6IHN0cmluZ106IHVua25vd24gfVtdIiwiY29uZGl0aW9uczogdW5rbm93bltdW10iLCJsZXZlbHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W11bXSIsImN1cnJlbnRMZXZlbDogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSIsIm5leHRQZW5kaW5nOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RhdGFiYXNlL3Vwc2VydC1idWlsZGVyLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJhbmRvbVVVSUQgfSBmcm9tIFwiY3J5cHRvXCI7XG5cbmltcG9ydCB7IGdldExvZ2dlciB9IGZyb20gXCJAbG9ndGFwZS9sb2d0YXBlXCI7XG5pbXBvcnQgeyB0eXBlIEtuZXggfSBmcm9tIFwia25leFwiO1xuaW1wb3J0IHsgY2x1c3RlciwgaXNBcnJheSwgdW5pcXVlIH0gZnJvbSBcInJhZGFzaGlcIjtcblxuaW1wb3J0IHsgRW50aXR5TWFuYWdlciB9IGZyb20gXCIuLi9lbnRpdHkvZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB7IE5haXRlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQge1xuICB0eXBlIERhdGFiYXNlRm9yZWlnbktleXMsXG4gIHR5cGUgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gIHR5cGUgRW50aXR5SW5kZXgsXG59IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgYXNzZXJ0RGVmaW5lZCwgbm9uTnVsbGFibGUgfSBmcm9tIFwiLi4vdXRpbHMvdXRpbHNcIjtcbmltcG9ydCB7IGJhdGNoVXBkYXRlIH0gZnJvbSBcIi4vX2JhdGNoX3VwZGF0ZVwiO1xuaW1wb3J0IHsgdHlwZSBSb3dXaXRoSWQgfSBmcm9tIFwiLi9fYmF0Y2hfdXBkYXRlXCI7XG5pbXBvcnQgeyB0eXBlIENvbHVtbktleXMsIHR5cGUgRm9yZWlnbktleUNvbHVtbnMsIHR5cGUgSWRUeXBlLCB0eXBlIFRhYmxlTmFtZSB9IGZyb20gXCIuL3B1cmkudHlwZXNcIjtcblxuY29uc3QgbG9nZ2VyID0gZ2V0TG9nZ2VyKFtcInNvbmFtdVwiLCBcImludGVybmFsXCIsIFwidXBzZXJ0LWJ1aWxkZXJcIl0pO1xuXG4vKipcbiAqIEZLIO2DgOyehSDstpTroaDsnYQg7JyE7ZW0IERhdGFiYXNlRm9yZWlnbktleXMgZXhwb3J0XG4gKiAobW9kdWxlIGF1Z21lbnRhdGlvbiDsnpDrj5kg66Gc65OcIOuztOyepSlcbiAqL1xuZXhwb3J0IHR5cGUgeyBEYXRhYmFzZUZvcmVpZ25LZXlzIH07XG5cbnR5cGUgSW5oZXJpdGFibGVDb2x1bW5zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID1cbiAgVFRhYmxlIGV4dGVuZHMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQgPyBDb2x1bW5LZXlzPERhdGFiYXNlU2NoZW1hRXh0ZW5kW1RUYWJsZV0+IDogbmV2ZXI7XG5cbi8vIO2FjOydtOu4lCDrjbDsnbTthLAg7YOA7J6FXG50eXBlIFRhYmxlRGF0YSA9IHtcbiAgcmVmZXJlbmNlczogU2V0PHN0cmluZz47XG4gIHJvd3M6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W107XG4gIHVuaXF1ZUluZGV4ZXM6IEVudGl0eUluZGV4W107XG4gIHVuaXF1ZXNNYXA6IE1hcDxzdHJpbmcsIHN0cmluZz47XG4gIGpzb25Db2x1bW5zOiBzdHJpbmdbXTtcbn07XG5cbi8vIOywuOyhsCDtlYTrk5wg7YOA7J6FXG5leHBvcnQgdHlwZSBVQlJlZiA9IHtcbiAgdXVpZDogc3RyaW5nO1xuICBvZjogc3RyaW5nO1xuICB1c2U/OiBzdHJpbmc7XG59O1xuXG4vLyB1cHNlcnQg7Ji17IWYXG5leHBvcnQgdHlwZSBVcHNlcnRPcHRpb25zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG4gIGNsZWFuT3JwaGFucz86IEZvcmVpZ25LZXlDb2x1bW5zPFRUYWJsZT4gfCBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W107XG4gIGluaGVyaXQ/OiBJbmhlcml0YWJsZUNvbHVtbnM8VFRhYmxlPltdO1xufTtcblxuLy8gaW5zZXJ0T25seSDsmLXshZhcbmV4cG9ydCB0eXBlIEluc2VydE9ubHlPcHRpb25zID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gaXNSZWZGaWVsZChmaWVsZDogdW5rbm93bik6IGZpZWxkIGlzIFVCUmVmIHtcbiAgcmV0dXJuIChcbiAgICBmaWVsZCAhPT0gdW5kZWZpbmVkICYmXG4gICAgZmllbGQgIT09IG51bGwgJiZcbiAgICAoZmllbGQgYXMgVUJSZWYpPy5vZiAhPT0gdW5kZWZpbmVkICYmXG4gICAgKGZpZWxkIGFzIFVCUmVmKT8udXVpZCAhPT0gdW5kZWZpbmVkXG4gICk7XG59XG5cbmV4cG9ydCBjbGFzcyBVcHNlcnRCdWlsZGVyIHtcbiAgdGFibGVzOiBNYXA8c3RyaW5nLCBUYWJsZURhdGE+O1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnRhYmxlcyA9IG5ldyBNYXAoKTtcbiAgfVxuXG4gIGdldFRhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogVGFibGVEYXRhIHtcbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICh0YWJsZSkge1xuICAgICAgcmV0dXJuIHRhYmxlO1xuICAgIH1cblxuICAgIGNvbnN0IHRhYmxlU3BlYyA9ICgoKSA9PiB7XG4gICAgICB0cnkge1xuICAgICAgICByZXR1cm4gRW50aXR5TWFuYWdlci5nZXRUYWJsZVNwZWModGFibGVOYW1lKTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgIH1cbiAgICB9KSgpO1xuXG4gICAgY29uc3QgdGFibGVEYXRhID0ge1xuICAgICAgcmVmZXJlbmNlczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICByb3dzOiBbXSxcbiAgICAgIHVuaXF1ZUluZGV4ZXM6IHRhYmxlU3BlYz8udW5pcXVlSW5kZXhlcyA/PyBbXSxcbiAgICAgIHVuaXF1ZXNNYXA6IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+KCksXG4gICAgICBqc29uQ29sdW1uczogdGFibGVTcGVjPy5qc29uQ29sdW1ucyA/PyBbXSxcbiAgICB9O1xuICAgIHRoaXMudGFibGVzLnNldCh0YWJsZU5hbWUsIHRhYmxlRGF0YSk7XG4gICAgcmV0dXJuIHRhYmxlRGF0YTtcbiAgfVxuXG4gIGhhc1RhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMudGFibGVzLmhhcyh0YWJsZU5hbWUpO1xuICB9XG5cbiAgcmVnaXN0ZXI8VCBleHRlbmRzIHN0cmluZz4oXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgcm93OiB7XG4gICAgICBba2V5IGluIFRdPzogVUJSZWYgfCBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgYmlnaW50IHwgbnVsbCB8IG9iamVjdCB8IHVua25vd247XG4gICAgfSxcbiAgKTogVUJSZWYge1xuICAgIGNvbnN0IHRhYmxlID0gdGhpcy5nZXRUYWJsZSh0YWJsZU5hbWUpO1xuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCB1bmlxdWUg7J24642x7Iqk66W8IOyInO2ajO2VmOupsCDtgqQg7IOd7ISxXG4gICAgY29uc3QgdW5pcXVlS2V5cyA9IHRhYmxlLnVuaXF1ZUluZGV4ZXNcbiAgICAgIC5tYXAoKHVucUluZGV4KSA9PiB7XG4gICAgICAgIGNvbnN0IHVuaXF1ZUtleUFycmF5ID0gdW5xSW5kZXguY29sdW1ucy5tYXAoKHVucUNvbCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHZhbCA9IHJvd1t1bnFDb2wubmFtZSBhcyBrZXlvZiB0eXBlb2Ygcm93XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZCh2YWwpKSB7XG4gICAgICAgICAgICByZXR1cm4gdmFsLnV1aWQ7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJldHVybiByb3dbdW5xQ29sLm5hbWUgYXMga2V5b2YgdHlwZW9mIHJvd10gPz8gcmFuZG9tVVVJRCgpOyAvLyBudWxsYWJsZeyduCDqsr3smrAgdXVpZOuhnCDrnpzrjaTqsJIg7IK97J6FXG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICAvLyDqsJLsnbQg66qo65GQIG51bGzsnbgg6rK97JqwIO2CpCDsg53shLEg7Yyo7IqkXG4gICAgICAgIGlmICh1bmlxdWVLZXlBcnJheS5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdW5pcXVlS2V5QXJyYXkuam9pbihcIi0tLWRlbGltaXRlci0tXCIpO1xuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgLy8gdXVpZCDsg53shLEg66Gc7KeBXG4gICAgY29uc3QgeyB1dWlkLCBpc1JldXNlZCB9ID0gKCgpID0+IHtcbiAgICAgIC8vIO2CpOulvCDsiJztmoztlZjsl6wg7J2066+4IOyhtOyerO2VmOuKlCDtgqTqsIAg7J6I64qU7KeAIO2ZleyduFxuICAgICAgaWYgKHVuaXF1ZUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgICAgaWYgKHRhYmxlLnVuaXF1ZXNNYXAuaGFzKHVuaXF1ZUtleSkpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIHV1aWQ6IGFzc2VydERlZmluZWQodGFibGUudW5pcXVlc01hcC5nZXQodW5pcXVlS2V5KSwgXCJVbmlxdWUga2V5IG5vdCBmb3VuZFwiKSxcbiAgICAgICAgICAgICAgaXNSZXVzZWQ6IHRydWUsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyDssL7snYQg7IiYIOyXhuuKlCDqsr3smrAg7IOd7ISxXG4gICAgICByZXR1cm4geyB1dWlkOiByYW5kb21VVUlEKCksIGlzUmV1c2VkOiBmYWxzZSB9O1xuICAgIH0pKCk7XG5cbiAgICAvLyDrqqjrk6Ag7Jyg64uI7YGs7YKk7JeQIOuMgO2VtCDsnKDri4jtgazrp7Xsl5AgdXVpZCDsoIDsnqVcbiAgICBpZiAodW5pcXVlS2V5cy5sZW5ndGggPiAwKSB7XG4gICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgIHRhYmxlLnVuaXF1ZXNNYXAuc2V0KHVuaXF1ZUtleSwgdXVpZCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7J20IO2FjOydtOu4lOyXkCDsgqzsmqnrkJwgUmVmRmllbGTrpbwg7Iic7ZqM7ZWY7JesLCDtmITsnqwg7YWM7J2067iUIOygleuztOyXkCDslrTrlqQg7ZWE65Oc66W8IOywuOyhsO2VmOuKlOyngCDstpTqsIBcbiAgICAvLyDsnbQg7KCV67O066W8IOuCmOykkeyXkCDsuZjtmZjtlaAg65WMIOyCrOyaqVxuICAgIHJvdyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHJvdykubWFwKChbcm93S2V5LCByb3dWYWx1ZV0pID0+IHtcbiAgICAgICAgaWYgKGlzUmVmRmllbGQocm93VmFsdWUpKSB7XG4gICAgICAgICAgcm93VmFsdWUudXNlID8/PSBcImlkXCI7XG4gICAgICAgICAgdGFibGUucmVmZXJlbmNlcy5hZGQoYCR7cm93VmFsdWUub2Z9LiR7cm93VmFsdWUudXNlfWApO1xuICAgICAgICAgIHJldHVybiBbcm93S2V5LCByb3dWYWx1ZV07XG4gICAgICAgIH0gZWxzZSBpZiAodGFibGUuanNvbkNvbHVtbnMuaW5jbHVkZXMocm93S2V5KSAmJiByb3dWYWx1ZSAhPT0gbnVsbCkge1xuICAgICAgICAgIC8vIEpTT04g7Lus65+87J24IOqyveyasCBKU09OLnN0cmluZ2lmeSDsspjrpqwgKEtuZXjripQgSlNPTiDtg4DsnoXsnYQg7KeA7JuQ7ZWY7KeAIOyViuydjClcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgSlNPTi5zdHJpbmdpZnkocm93VmFsdWUpXTtcbiAgICAgICAgfSBlbHNlIGlmIChpc0FycmF5KHJvd1ZhbHVlKSkge1xuICAgICAgICAgIC8vIOuwsOyXtOydgCDqt7jrjIDroZwg7KCA7J6lXG4gICAgICAgICAgcmV0dXJuIFtyb3dLZXksIHJvd1ZhbHVlXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgcm93VmFsdWVdO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApIGFzIHsgW2tleSBpbiBUXT86IHVua25vd24gfTtcblxuICAgIHRhYmxlLnJvd3MucHVzaCh7XG4gICAgICB1dWlkLFxuICAgICAgLi4ucm93LFxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzdWx0OiBVQlJlZiA9IHtcbiAgICAgIG9mOiB0YWJsZU5hbWUsXG4gICAgICB1dWlkOiAocm93IGFzIHsgdXVpZD86IHN0cmluZyB9KS51dWlkID8/IHV1aWQsXG4gICAgfTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLXJlZ2lzdGVyXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHV1aWQ6IHJlc3VsdC51dWlkLFxuICAgICAgaXNVdWlkUmV1c2VkOiBpc1JldXNlZCxcbiAgICAgIHJvdyxcbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyB1cHNlcnQ8VFRhYmxlIGV4dGVuZHMgVGFibGVOYW1lPERhdGFiYXNlU2NoZW1hRXh0ZW5kPj4oXG4gICAgd2RiOiBLbmV4LFxuICAgIHRhYmxlTmFtZTogVFRhYmxlLFxuICAgIG9wdGlvbnM/OiBVcHNlcnRPcHRpb25zPFRUYWJsZT4sXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJ1cHNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyBpbnNlcnRPbmx5PFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+KFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IFRUYWJsZSxcbiAgICBvcHRpb25zPzogSW5zZXJ0T25seU9wdGlvbnMsXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJpbnNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyB1cHNlcnRPckluc2VydDxUVGFibGUgZXh0ZW5kcyBUYWJsZU5hbWU8RGF0YWJhc2VTY2hlbWFFeHRlbmQ+PihcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBUVGFibGUsXG4gICAgbW9kZTogXCJ1cHNlcnRcIiB8IFwiaW5zZXJ0XCIsXG4gICAgb3B0aW9ucz86IFVwc2VydE9wdGlvbnM8VFRhYmxlPixcbiAgKTogUHJvbWlzZTxJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXT4ge1xuICAgIGlmICghdGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgdGFibGUgPSB0aGlzLnRhYmxlcy5nZXQodGFibGVOYW1lKTtcbiAgICBpZiAodGFibGUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBzZXJ0IOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgJHt0YWJsZU5hbWV97JeQIHVwc2VydCDtlaAg642w7J207YSw6rCAIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICBpZiAoXG4gICAgICB0YWJsZS5yb3dzLnNvbWUoKHJvdykgPT5cbiAgICAgICAgT2JqZWN0LmVudHJpZXMocm93KS5zb21lKChbLCB2YWx1ZV0pID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mICE9PSB0YWJsZU5hbWUpLFxuICAgICAgKVxuICAgICkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX0g7ZW06rKw65CY7KeAIOyViuydgCDssLjsobDqsIAg7J6I7Iq164uI64ukLmApO1xuICAgIH1cblxuICAgIC8vIOyghOyytCDthYzsnbTruJQg7Iic7ZqM7ZWY7JesIO2YhOyerCDthYzsnbTruJQg7LC47KGw7ZWY64qUIOuqqOuToCDthYzsnbTruJQg7LaU7LacXG4gICAgY29uc3QgeyByZWZlcmVuY2VzLCByZWZUYWJsZXMgfSA9IEFycmF5LmZyb20odGhpcy50YWJsZXMpLnJlZHVjZShcbiAgICAgIChyLCBbLCB0YWJsZV0pID0+IHtcbiAgICAgICAgY29uc3QgcmVmZXJlbmNlID0gQXJyYXkuZnJvbSh0YWJsZS5yZWZlcmVuY2VzLnZhbHVlcygpKS5maW5kKChyZWYpID0+XG4gICAgICAgICAgcmVmLmluY2x1ZGVzKGAke3RhYmxlTmFtZX0uYCksXG4gICAgICAgICk7XG4gICAgICAgIGlmIChyZWZlcmVuY2UpIHtcbiAgICAgICAgICByLnJlZmVyZW5jZXMucHVzaChyZWZlcmVuY2UpO1xuICAgICAgICAgIHIucmVmVGFibGVzLnB1c2godGFibGUpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9LFxuICAgICAge1xuICAgICAgICByZWZlcmVuY2VzOiBbXSBhcyBzdHJpbmdbXSxcbiAgICAgICAgcmVmVGFibGVzOiBbXSBhcyBUYWJsZURhdGFbXSxcbiAgICAgIH0sXG4gICAgKTtcbiAgICBjb25zdCBleHRyYWN0RmllbGRzID0gdW5pcXVlKHJlZmVyZW5jZXMpXG4gICAgICAubWFwKChyZWZlcmVuY2UpID0+IHJlZmVyZW5jZS5zcGxpdChcIi5cIilbMV0pXG4gICAgICAuZmlsdGVyKChmaWVsZCk6IGZpZWxkIGlzIHN0cmluZyA9PiBmaWVsZCAhPT0gdW5kZWZpbmVkKTtcblxuICAgIC8vIOydmOyhtOyEsSDsiJzshJzsl5Ag65Sw6528IOugiOuyqOuzhCDqt7jro7ntmZQgKOyekOq4sCDssLjsobDqsIAg7JeG7Jy866m0IExldmVsIDAg7ZWY64KYKVxuICAgIGNvbnN0IHsgbGV2ZWxzLCBoYXNDaXJjdWxhciB9ID0gdGhpcy5idWlsZEluc2VydExldmVscyh0YWJsZS5yb3dzLCB0YWJsZU5hbWUpO1xuXG4gICAgaWYgKGhhc0NpcmN1bGFyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfeyXkCDsiJztmZgg7J6Q6riwIOywuOyhsOqwgCDsnojsirXri4jri6QuYCk7XG4gICAgfVxuXG4gICAgY29uc3QgdXVpZE1hcCA9IG5ldyBNYXA8c3RyaW5nLCB1bmtub3duPigpO1xuICAgIGNvbnN0IGFsbElkczogKG51bWJlciB8IHN0cmluZylbXSA9IFtdO1xuXG4gICAgLy8g66CI67Ko67OE66GcIOyInOywqCDsspjrpqxcbiAgICBmb3IgKGxldCBsZXZlbElkeCA9IDA7IGxldmVsSWR4IDwgbGV2ZWxzLmxlbmd0aDsgbGV2ZWxJZHgrKykge1xuICAgICAgY29uc3QgbGV2ZWxSb3dzID0gbGV2ZWxzW2xldmVsSWR4XTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhcIlByb2Nlc3NpbmcgUXVlcnkgTGV2ZWw6IHtjdXJyZW50fSAvIHt0b3RhbH1cIiwge1xuICAgICAgICBjdXJyZW50OiBsZXZlbElkeCArIDEsXG4gICAgICAgIHRvdGFsOiBsZXZlbHMubGVuZ3RoLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIOydtOyghCDroIjrsqjsl5DshJwg7Ja77J2AIElE66GcIOyekOq4sCDssLjsobAg7ZW06rKwXG4gICAgICBjb25zdCByZXNvbHZlZFJvd3MgPSBsZXZlbFJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgY29uc3QgcmVzb2x2ZWQgPSB7IC4uLnJvdyB9O1xuICAgICAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhyb3cpKSB7XG4gICAgICAgICAgaWYgKGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHZhbHVlLnV1aWQpO1xuXG4gICAgICAgICAgICBpZiAoIXBhcmVudCkgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHt2YWx1ZS51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcblxuICAgICAgICAgICAgcmVzb2x2ZWRba2V5XSA9IChwYXJlbnQgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW3ZhbHVlLnVzZSA/PyBcImlkXCJdO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHZhbHVlLm9mLCB1dWlkOiB2YWx1ZS51dWlkLCB1c2U6IHZhbHVlLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkW2tleV0sXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJlc29sdmVkO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIO2YhOyerCDroIjrsqggdXBzZXJ0XG4gICAgICBjb25zdCBjaHVua1NpemUgPSBvcHRpb25zPy5jaHVua1NpemU7XG4gICAgICBjb25zdCBsZXZlbENodW5rcyA9IGNodW5rU2l6ZSA/IGNsdXN0ZXIocmVzb2x2ZWRSb3dzLCBjaHVua1NpemUpIDogW3Jlc29sdmVkUm93c107XG4gICAgICBjb25zdCBzZWxlY3RGaWVsZHMgPSB1bmlxdWUoW1wiaWRcIiwgLi4uZXh0cmFjdEZpZWxkc10pO1xuXG4gICAgICBmb3IgKGxldCBpbmRleCA9IDA7IGluZGV4IDwgbGV2ZWxDaHVua3MubGVuZ3RoOyBpbmRleCsrKSB7XG4gICAgICAgIGNvbnN0IGRhdGFDaHVuayA9IGxldmVsQ2h1bmtzW2luZGV4XTtcbiAgICAgICAgaWYgKGRhdGFDaHVuay5sZW5ndGggPT09IDApIGNvbnRpbnVlO1xuICAgICAgICBsb2dnZXIuZGVidWcoXCJQcm9jZXNzaW5nIENodW5rOiB7Y3VycmVudH0gLyB7dG90YWx9XCIsIHtcbiAgICAgICAgICBjdXJyZW50OiBpbmRleCArIDEsXG4gICAgICAgICAgdG90YWw6IGxldmVsQ2h1bmtzLmxlbmd0aCxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gdXVpZOulvCDrs4Trj4TroZwg67O06rSA7ZWY6rOgLCBEQuyXkCDsoIDsnqXtlaAg642w7J207YSw7JeQ7IScIOygnOqxsFxuICAgICAgICBjb25zdCBvcmlnaW5hbFV1aWRzID0gZGF0YUNodW5rLm1hcCgocikgPT4gci51dWlkIGFzIHN0cmluZyk7XG4gICAgICAgIGNvbnN0IGRhdGFGb3JEYiA9IGRhdGFDaHVuay5tYXAoKHsgdXVpZCwgLi4ucmVzdCB9KSA9PiByZXN0KTtcblxuICAgICAgICBsZXQgcmVzdWx0Um93czogeyBpZDogbnVtYmVyIHwgc3RyaW5nOyBba2V5OiBzdHJpbmddOiB1bmtub3duIH1bXTtcblxuICAgICAgICBpZiAobW9kZSA9PT0gXCJpbnNlcnRcIikge1xuICAgICAgICAgIC8vIElOU0VSVCDrqqjrk5wgLSBSRVRVUk5JTkcg7IKs7JqpXG4gICAgICAgICAgcmVzdWx0Um93cyA9IGF3YWl0IHdkYi5pbnNlcnQoZGF0YUZvckRiKS5pbnRvKHRhYmxlTmFtZSkucmV0dXJuaW5nKHNlbGVjdEZpZWxkcyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gVVBTRVJUIOuqqOuTnCAtIGlkIOyXhuuKlCByb3frk6TsnZggaWTrpbwg7IKs7KCEIOyhsO2ajOuhnCDssYTsmrDquLBcbiAgICAgICAgICBjb25zdCByb3dzV2l0aG91dElkID0gZGF0YUZvckRiLmZpbHRlcigocm93KSA9PiAhcm93LmlkKTtcblxuICAgICAgICAgIGlmIChyb3dzV2l0aG91dElkLmxlbmd0aCA+IDAgJiYgdGFibGUudW5pcXVlSW5kZXhlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAvLyDrqqjrk6AgdW5pcXVlSW5kZXhlc+uhnCDquLDsobQg66CI7L2U65OcIOyhsO2ajFxuICAgICAgICAgICAgZm9yIChjb25zdCB1bmlxdWVJbmRleCBvZiB0YWJsZS51bmlxdWVJbmRleGVzKSB7XG4gICAgICAgICAgICAgIGNvbnN0IGNvbHVtbnMgPSB1bmlxdWVJbmRleC5jb2x1bW5zLm1hcCgoYykgPT4gYy5uYW1lKTtcblxuICAgICAgICAgICAgICAvLyDsobDtmoztlaAg7KGw6rG065OkIOy2lOy2nCAo6rCBIHJvd+ydmCB1bmlxdWUg7Lus65+8IOqwkuuTpClcbiAgICAgICAgICAgICAgY29uc3QgY29uZGl0aW9uczogdW5rbm93bltdW10gPSBbXTtcbiAgICAgICAgICAgICAgZm9yIChjb25zdCByb3cgb2Ygcm93c1dpdGhvdXRJZCkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHZhbHVlcyA9IGNvbHVtbnMubWFwKChjb2wpID0+IHJvd1tjb2xdKTtcbiAgICAgICAgICAgICAgICAvLyBudWxs7J20IO2PrO2VqOuQnCDsobDqsbTsnYAg7KCc7Jm4IChQb3N0Z3JlU1FMIFVOSVFVReuKlCBOVUxMIOustOyLnClcbiAgICAgICAgICAgICAgICBpZiAoIXZhbHVlcy5zb21lKCh2KSA9PiB2ID09IG51bGwpKSB7XG4gICAgICAgICAgICAgICAgICBjb25kaXRpb25zLnB1c2godmFsdWVzKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICBpZiAoY29uZGl0aW9ucy5sZW5ndGggPT09IDApIGNvbnRpbnVlO1xuXG4gICAgICAgICAgICAgIC8vIOuwsOy5mCBTRUxFQ1RcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdSb3dzID0gKGF3YWl0IHdkYih0YWJsZU5hbWUpXG4gICAgICAgICAgICAgICAgLndoZXJlSW4oY29sdW1ucywgY29uZGl0aW9ucyBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdW10pXG4gICAgICAgICAgICAgICAgLnNlbGVjdChcImlkXCIsIC4uLmNvbHVtbnMpKSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdO1xuXG4gICAgICAgICAgICAgIC8vIE1hcCDsg53shLE6IHVuaXF1ZSDsu6zrn7wg7KGw7ZWpIOKGkiBpZFxuICAgICAgICAgICAgICBjb25zdCBleGlzdGluZ01hcCA9IG5ldyBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+KCk7XG4gICAgICAgICAgICAgIGZvciAoY29uc3QgZXhpc3Rpbmcgb2YgZXhpc3RpbmdSb3dzKSB7XG4gICAgICAgICAgICAgICAgY29uc3Qga2V5ID0gY29sdW1uc1xuICAgICAgICAgICAgICAgICAgLm1hcCgoY29sKSA9PiBTdHJpbmcoZXhpc3RpbmdbY29sXSA/PyBcIlwiKSlcbiAgICAgICAgICAgICAgICAgIC5qb2luKFwiLS0tZGVsaW1pdGVyLS0tXCIpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGlkID0gZXhpc3RpbmcuaWQ7XG4gICAgICAgICAgICAgICAgaWYgKHR5cGVvZiBpZCA9PT0gXCJudW1iZXJcIiB8fCB0eXBlb2YgaWQgPT09IFwic3RyaW5nXCIpIHtcbiAgICAgICAgICAgICAgICAgIGV4aXN0aW5nTWFwLnNldChrZXksIGlkKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAvLyBpZCDsl4bripQgcm9365Ok7JeQIOunpOy5reuQmOuKlCBpZCDssYTsmrDquLBcbiAgICAgICAgICAgICAgZm9yIChjb25zdCByb3cgb2Ygcm93c1dpdGhvdXRJZCkge1xuICAgICAgICAgICAgICAgIGlmIChyb3cuaWQpIGNvbnRpbnVlOyAvLyDsnbTrr7gg64uk66W4IHVuaXF1ZUluZGV47JeQ7IScIOyxhOybjOynhCDqsr3smrAg7Iqk7YK1XG5cbiAgICAgICAgICAgICAgICBjb25zdCBrZXkgPSBjb2x1bW5zLm1hcCgoY29sKSA9PiBTdHJpbmcocm93W2NvbF0gPz8gXCJcIikpLmpvaW4oXCItLS1kZWxpbWl0ZXItLS1cIik7XG4gICAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdJZCA9IGV4aXN0aW5nTWFwLmdldChrZXkpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGV4aXN0aW5nSWQpIHtcbiAgICAgICAgICAgICAgICAgIHJvdy5pZCA9IGV4aXN0aW5nSWQ7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gb25Db25mbGljdOuKlCBpZOunjCDsgqzsmqkgKOuqqOuToCB1bmlxdWVJbmRleGVz64qUIOydtOuvuCDsgqzsoIQg7KGw7ZqM66GcIOyymOumrOuQqClcbiAgICAgICAgICBjb25zdCBjb25mbGljdENvbHVtbnMgPSBbXCJpZFwiXTtcblxuICAgICAgICAgIGNvbnN0IGFsbENvbHVtbnMgPSBPYmplY3Qua2V5cyhkYXRhRm9yRGJbMF0pO1xuICAgICAgICAgIGxldCB1cGRhdGVDb2x1bW5zID0gYWxsQ29sdW1ucy5maWx0ZXIoKGMpID0+IGMgIT09IFwiaWRcIik7XG5cbiAgICAgICAgICAvLyBpbmhlcml0IOyYteyFmCDsspjrpqwgLSBpbmhlcml0IOy7rOufvOydgCB1cGRhdGUg64yA7IOB7JeQ7IScIOygnOyZuFxuICAgICAgICAgIGlmIChvcHRpb25zPy5pbmhlcml0Py5sZW5ndGgpIHtcbiAgICAgICAgICAgIGNvbnN0IGluaGVyaXRDb2x1bW5zID0gb3B0aW9ucy5pbmhlcml0IGFzIHN0cmluZ1tdO1xuXG4gICAgICAgICAgICBjb25zdCBleGNsdWRlZEZyb21VcGRhdGUgPSB1cGRhdGVDb2x1bW5zLmZpbHRlcigoYykgPT4gaW5oZXJpdENvbHVtbnMuaW5jbHVkZXMoYykpO1xuICAgICAgICAgICAgdXBkYXRlQ29sdW1ucyA9IHVwZGF0ZUNvbHVtbnMuZmlsdGVyKChjKSA9PiAhaW5oZXJpdENvbHVtbnMuaW5jbHVkZXMoYykpO1xuXG4gICAgICAgICAgICAvLyDsi6TsoJzroZwg7KCc7Jm465CcIOy7rOufvCDroZzquYVcbiAgICAgICAgICAgIGlmIChleGNsdWRlZEZyb21VcGRhdGUubGVuZ3RoKSB7XG4gICAgICAgICAgICAgIE5haXRlLnQoXCJwdXJpOnViLWluaGVyaXRcIiwge1xuICAgICAgICAgICAgICAgIHRhYmxlTmFtZSxcbiAgICAgICAgICAgICAgICBpbmhlcml0Q29sdW1ucyxcbiAgICAgICAgICAgICAgICBleGNsdWRlZEZyb21VcGRhdGUsXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIHVwZGF0ZUNvbHVtbnPqsIAg67mE7Ja07J6I7Ja064+EIG1lcmdlKCnrpbwg7IKs7Jqp7ZWY7JesIOuqqOuToCDtlonsnbQgUkVUVVJOSU5H65CY64+E66GdIOuztOyepVxuICAgICAgICAgIGNvbnN0IG1lcmdlQ29sdW1ucyA9IHVwZGF0ZUNvbHVtbnMubGVuZ3RoID8gdXBkYXRlQ29sdW1ucyA6IGNvbmZsaWN0Q29sdW1ucztcblxuICAgICAgICAgIHJlc3VsdFJvd3MgPSBhd2FpdCB3ZGJcbiAgICAgICAgICAgIC5pbnNlcnQoZGF0YUZvckRiKVxuICAgICAgICAgICAgLmludG8odGFibGVOYW1lKVxuICAgICAgICAgICAgLm9uQ29uZmxpY3QoY29uZmxpY3RDb2x1bW5zKVxuICAgICAgICAgICAgLm1lcmdlKG1lcmdlQ29sdW1ucylcbiAgICAgICAgICAgIC5yZXR1cm5pbmcoc2VsZWN0RmllbGRzKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChvcmlnaW5hbFV1aWRzLmxlbmd0aCAhPT0gcmVzdWx0Um93cy5sZW5ndGgpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfTogcmVnaXN0ZXIvcmV0dXJuaW5nIOu2iOydvOy5mGApO1xuICAgICAgICB9XG5cbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCByZXN1bHRSb3dzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgdXVpZE1hcC5zZXQob3JpZ2luYWxVdWlkc1tpXSwgcmVzdWx0Um93c1tpXSk7XG4gICAgICAgICAgYWxsSWRzLnB1c2gocmVzdWx0Um93c1tpXS5pZCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyDtlbTri7kg7YWM7J2067iUIOywuOyhsOulvCDsi6TsoJwg67C466WY66GcIOuzgOqyvVxuICAgIGZvciAoY29uc3QgdGFibGUgb2YgcmVmVGFibGVzKSB7XG4gICAgICB0YWJsZS5yb3dzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4ge1xuICAgICAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyhyb3cpKSB7XG4gICAgICAgICAgY29uc3QgcHJvcCA9IHJvd1trZXldO1xuICAgICAgICAgIGlmIChpc1JlZkZpZWxkKHByb3ApICYmIHByb3Aub2YgPT09IHRhYmxlTmFtZSkge1xuICAgICAgICAgICAgY29uc3QgcGFyZW50ID0gdXVpZE1hcC5nZXQocHJvcC51dWlkKTtcbiAgICAgICAgICAgIGlmICghcGFyZW50KSB7XG4gICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IocHJvcCk7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihg7KG07J6s7ZWY7KeAIOyViuuKlCB1dWlkICR7cHJvcC51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnN0IHJlc29sdmVkVmFsdWUgPSAocGFyZW50IGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+KVtwcm9wLnVzZSA/PyBcImlkXCJdO1xuICAgICAgICAgICAgcm93W2tleV0gPSByZXNvbHZlZFZhbHVlO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHByb3Aub2YsIHV1aWQ6IHByb3AudXVpZCwgdXNlOiBwcm9wLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkVmFsdWUsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJvdztcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIGlmIChvcHRpb25zPy5jbGVhbk9ycGhhbnMpIHtcbiAgICAgIGNvbnN0IGNsZWFuT3JwaGFucyA9IG9wdGlvbnMuY2xlYW5PcnBoYW5zO1xuICAgICAgY29uc3QgZmtDb2x1bW5zID0gaXNBcnJheShjbGVhbk9ycGhhbnMpXG4gICAgICAgID8gKGNsZWFuT3JwaGFucyBhcyBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W10pXG4gICAgICAgIDogW2NsZWFuT3JwaGFucyBhcyBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+XTtcblxuICAgICAgLy8g7ZiE7J6sIHJlZ2lzdGVy65CcIOugiOy9lOuTnOuTpOydmCBGSyDqsJLrk6Qg7LaU7LacXG4gICAgICBjb25zdCBma0NvbmRpdGlvbnMgPSBma0NvbHVtbnMubWFwKChma0NvbCkgPT4ge1xuICAgICAgICBjb25zdCBma1ZhbHVlcyA9IFsuLi5uZXcgU2V0KHRhYmxlLnJvd3MubWFwKChyb3cpID0+IHJvd1tma0NvbF0pLmZpbHRlcigodikgPT4gdiAhPSBudWxsKSldO1xuICAgICAgICByZXR1cm4geyBjb2x1bW46IGZrQ29sLCB2YWx1ZXM6IGZrVmFsdWVzIH07XG4gICAgICB9KTtcblxuICAgICAgLy8g66qo65OgIEZLIOy7rOufvOyXkCDqsJLsnbQg7J6I64qUIOqyveyasOyXkOunjCDsgq3soJwg7Iuk7ZaJXG4gICAgICBpZiAoZmtDb25kaXRpb25zLmV2ZXJ5KChmYykgPT4gZmMudmFsdWVzLmxlbmd0aCA+IDApKSB7XG4gICAgICAgIGxldCBkZWxldGVRdWVyeSA9IHdkYih0YWJsZU5hbWUpO1xuXG4gICAgICAgIC8vIOqwgSBGSyDsu6zrn7zsl5Ag64yA7ZWcIFdIRVJFIElOIOyhsOqxtCDstpTqsIBcbiAgICAgICAgZm9yIChjb25zdCB7IGNvbHVtbiwgdmFsdWVzIH0gb2YgZmtDb25kaXRpb25zKSB7XG4gICAgICAgICAgZGVsZXRlUXVlcnkgPSBkZWxldGVRdWVyeS53aGVyZUluKGNvbHVtbiwgdmFsdWVzKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIOuwqeq4iCB1cHNlcnTtlZwgSUTripQg7KCc7Jm4XG4gICAgICAgIGRlbGV0ZVF1ZXJ5ID0gZGVsZXRlUXVlcnkud2hlcmVOb3RJbihcImlkXCIsIGFsbElkcyk7XG5cbiAgICAgICAgY29uc3QgZGVsZXRlZENvdW50ID0gYXdhaXQgZGVsZXRlUXVlcnkuZGVsZXRlKCk7XG5cbiAgICAgICAgTmFpdGUudChcInB1cmk6dWItY2xlYW4tb3JwaGFuc1wiLCB7XG4gICAgICAgICAgdGFibGVOYW1lLFxuICAgICAgICAgIGNsZWFuT3JwaGFuczogZmtDb2x1bW5zLFxuICAgICAgICAgIGRlbGV0ZWRDb3VudCxcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCDrjbDsnbTthLAg7LSI6riw7ZmUXG4gICAgdGFibGUucm93cyA9IFtdO1xuICAgIHRhYmxlLnJlZmVyZW5jZXMuY2xlYXIoKTtcbiAgICB0YWJsZS51bmlxdWVzTWFwLmNsZWFyKCk7XG5cbiAgICBOYWl0ZS50KFwicHVyaTp1Yi11cHNlcnRlZFwiLCB7XG4gICAgICB0YWJsZU5hbWUsXG4gICAgICBtb2RlLFxuICAgICAgcm93Q291bnQ6IGFsbElkcy5sZW5ndGgsXG4gICAgICByZXR1cm5lZElkczogYWxsSWRzLFxuICAgIH0pO1xuXG4gICAgcmV0dXJuIGFsbElkcyBhcyBJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXTtcbiAgfVxuXG4gIGFzeW5jIHVwZGF0ZUJhdGNoKFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBvcHRpb25zPzoge1xuICAgICAgY2h1bmtTaXplPzogbnVtYmVyO1xuICAgICAgd2hlcmU/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgICB9LFxuICApOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBvcHRpb25zID0ge1xuICAgICAgLi4ub3B0aW9ucyxcbiAgICAgIGNodW5rU2l6ZTogb3B0aW9ucz8uY2h1bmtTaXplID8/IDUwMCxcbiAgICAgIHdoZXJlOiBvcHRpb25zPy53aGVyZSA/PyBcImlkXCIsXG4gICAgfTtcblxuICAgIGlmICghdGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGNvbnN0IHRhYmxlID0gdGhpcy50YWJsZXMuZ2V0KHRhYmxlTmFtZSk7XG4gICAgaWYgKCF0YWJsZSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDrk7HroZ3rkJjsp4Ag7JWK7J2AIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBkYXRlQmF0Y2gg7JqU7LKtYCk7XG4gICAgfSBlbHNlIGlmICh0YWJsZS5yb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHdoZXJlQ29sdW1ucyA9IEFycmF5LmlzQXJyYXkob3B0aW9ucy53aGVyZSkgPyBvcHRpb25zLndoZXJlIDogW29wdGlvbnMud2hlcmUgPz8gXCJpZFwiXTtcbiAgICBjb25zdCByb3dzID0gdGFibGUucm93cy5tYXAoKF9yb3cpID0+IHtcbiAgICAgIGNvbnN0IHsgdXVpZDogXywgLi4ucm93IH0gPSBfcm93OyAvLyB1dWlkIOygnOyZuFxuICAgICAgcmV0dXJuIHJvdyBhcyBSb3dXaXRoSWQ8c3RyaW5nPjtcbiAgICB9KTtcblxuICAgIGF3YWl0IGJhdGNoVXBkYXRlKHdkYiwgdGFibGVOYW1lLCB3aGVyZUNvbHVtbnMsIHJvd3MsIG9wdGlvbnMuY2h1bmtTaXplKTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLWJhdGNoLXVwZGF0ZWRcIiwge1xuICAgICAgdGFibGVOYW1lLFxuICAgICAgcm93Q291bnQ6IHJvd3MubGVuZ3RoLFxuICAgICAgd2hlcmVDb2x1bW5zLFxuICAgIH0pO1xuXG4gICAgLy8gdXBkYXRlQmF0Y2gg7JmE66OMIO2bhCDsspjrpqzrkJwg642w7J207YSwIOygnOqxsFxuICAgIHRhYmxlLnJvd3MgPSBbXTtcbiAgICB0YWJsZS5yZWZlcmVuY2VzLmNsZWFyKCk7XG4gICAgdGFibGUudW5pcXVlc01hcC5jbGVhcigpO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBQcml2YXRlIEhlbHBlciBNZXRob2RzXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICAvKipcbiAgICogcm93c+ulvCDsnZjsobTshLEg7Iic7ISc7JeQIOuUsOudvCDroIjrsqjrs4TroZwg6re466O57ZmUXG4gICAqIC0g7J6Q6riwIOywuOyhsCDsl4bripQg6rK97JqwIDog66qo65OgIHJvd3PqsIAgTGV2ZWwgMFxuICAgKiAtIOyekOq4sCDssLjsobAg7J6I64qUIOqyveyasCA6IOyekOq4sCDssLjsobAg6rSA6rOE66W8IOychOyDgSDsoJXroKztlZjsl6wg66CI67Ko67OE66GcIOq3uOujue2ZlFxuICAgKi9cbiAgcHJpdmF0ZSBidWlsZEluc2VydExldmVscyhcbiAgICByb3dzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdLFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICApOiB7IGxldmVsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdOyBoYXNDaXJjdWxhcjogYm9vbGVhbiB9IHtcbiAgICAvLyAxLiDsnpDquLAg7LC47KGw6rCAIOyXhuycvOuptCDtlZwg66CI67Ko66GcIOyymOumrFxuICAgIGNvbnN0IGhhc1NlbGZSZWYgPSByb3dzXG4gICAgICAuZmxhdE1hcCgocm93KSA9PiBPYmplY3QudmFsdWVzKHJvdykpXG4gICAgICAuc29tZSgodmFsdWUpID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpO1xuICAgIGlmICghaGFzU2VsZlJlZikgcmV0dXJuIHsgbGV2ZWxzOiBbcm93c10sIGhhc0NpcmN1bGFyOiBmYWxzZSB9O1xuXG4gICAgLy8gMi4gdXVpZCDihpIgcm93IOunpO2VkSAo7KSR67O1IHV1aWQg67Cp7KeAKVxuICAgIGNvbnN0IHJvd0J5VXVpZCA9IG5ldyBNYXA8c3RyaW5nLCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4oKTtcbiAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICBjb25zdCB1dWlkID0gcm93LnV1aWQgYXMgc3RyaW5nIHwgdW5kZWZpbmVkO1xuICAgICAgaWYgKCF1dWlkKSB0aHJvdyBuZXcgRXJyb3IoYGJ1aWxkSW5zZXJ0TGV2ZWxzOiB1dWlk6rCAIOyXhuuKlCByb3cgLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICByb3dCeVV1aWQuc2V0KHV1aWQsIHJvdyk7XG4gICAgfVxuXG4gICAgbGV0IHBlbmRpbmcgPSBBcnJheS5mcm9tKHJvd0J5VXVpZC52YWx1ZXMoKSk7XG4gICAgY29uc3QgbGV2ZWxzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdW10gPSBbXTtcbiAgICBjb25zdCBpbnNlcnRlZCA9IG5ldyBTZXQ8c3RyaW5nPigpO1xuXG4gICAgLy8gMy4g66CI67Ko67OEIOu2hOulmFxuICAgIHdoaWxlIChwZW5kaW5nLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRMZXZlbDogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSA9IFtdO1xuICAgICAgY29uc3QgbmV4dFBlbmRpbmc6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCByb3cgb2YgcGVuZGluZykge1xuICAgICAgICAvLyDsnbQgcm936rCAIOywuOyhsO2VmOuKlCDsnpDquLAg7LC47KGw65OkXG4gICAgICAgIGNvbnN0IHNlbGZSZWZzID0gT2JqZWN0LnZhbHVlcyhyb3cpLmZpbHRlcihcbiAgICAgICAgICAodmFsdWUpID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUsXG4gICAgICAgICkgYXMgVUJSZWZbXTtcblxuICAgICAgICAvLyDssLjsobDtlZjripQg66qo65OgIHV1aWTqsIAg7J2066+4IGluc2VydGVk7JeQIOyeiOyWtOyVvCDsnbTrsogg66CI67Ko7JeQIO2PrO2VqFxuICAgICAgICBjb25zdCBjYW5JbnNlcnQgPSBzZWxmUmVmcy5ldmVyeSgocmVmKSA9PiB7XG4gICAgICAgICAgaWYgKCFyb3dCeVV1aWQuaGFzKHJlZi51dWlkKSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHtyZWYudXVpZH0gLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiBpbnNlcnRlZC5oYXMocmVmLnV1aWQpO1xuICAgICAgICB9KTtcblxuICAgICAgICBpZiAoY2FuSW5zZXJ0KSB7XG4gICAgICAgICAgY3VycmVudExldmVsLnB1c2gocm93KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBuZXh0UGVuZGluZy5wdXNoKHJvdyk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8g7Iic7ZmYIOywuOyhsCDqsJDsp4BcbiAgICAgIGlmIChjdXJyZW50TGV2ZWwubGVuZ3RoID09PSAwKSByZXR1cm4geyBsZXZlbHM6IFtdLCBoYXNDaXJjdWxhcjogdHJ1ZSB9O1xuXG4gICAgICAvLyDroIjrsqgg7ZmV7KCVICsgaW5zZXJ0ZWQg6rCx7IugXG4gICAgICBsZXZlbHMucHVzaChjdXJyZW50TGV2ZWwpO1xuICAgICAgZm9yIChjb25zdCByb3cgb2YgY3VycmVudExldmVsKSB7XG4gICAgICAgIGluc2VydGVkLmFkZChyb3cudXVpZCBhcyBzdHJpbmcpO1xuICAgICAgfVxuXG4gICAgICBwZW5kaW5nID0gbmV4dFBlbmRpbmc7XG4gICAgfVxuXG4gICAgcmV0dXJuIHsgbGV2ZWxzLCBoYXNDaXJjdWxhcjogZmFsc2UgfTtcbiAgfVxufVxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7O0FBeURBLFNBQWdCLFdBQVcsT0FBZ0M7QUFDekQsUUFDRSxVQUFVLGFBQ1YsVUFBVSxRQUNULE9BQWlCLE9BQU8sYUFDeEIsT0FBaUIsU0FBUzs7OztzQkF4RDBCO2FBQ2xCO2FBTXFCO3FCQUNkO0NBSXhDLFNBQVMsVUFBVTtFQUFDO0VBQVU7RUFBWTtFQUFpQixDQUFDO0NBZ0RyRCxnQkFBYixNQUEyQjtFQUN6QjtFQUNBLGNBQWM7QUFDWixRQUFLLFNBQVMsSUFBSSxLQUFLOztFQUd6QixTQUFTLFdBQThCO0dBQ3JDLE1BQU0sUUFBUSxLQUFLLE9BQU8sSUFBSSxVQUFVO0FBQ3hDLE9BQUksT0FBTztBQUNULFdBQU87O0dBR1QsTUFBTSxtQkFBbUI7QUFDdkIsUUFBSTtBQUNGLFlBQU8sY0FBYyxhQUFhLFVBQVU7WUFDdEM7QUFDTixZQUFPOztPQUVQO0dBRUosTUFBTSxZQUFZO0lBQ2hCLFlBQVksSUFBSSxLQUFh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