sonamu 0.5.7 → 0.7.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 (529) hide show
  1. package/.swcrc.project-default +18 -0
  2. package/bin/cli.js +24 -0
  3. package/dist/ai/agents/agent.d.ts +11 -0
  4. package/dist/ai/agents/agent.d.ts.map +1 -0
  5. package/dist/ai/agents/agent.js +65 -0
  6. package/dist/ai/agents/index.d.ts +3 -0
  7. package/dist/ai/agents/index.d.ts.map +1 -0
  8. package/dist/ai/agents/index.js +4 -0
  9. package/dist/ai/agents/types.d.ts +43 -0
  10. package/dist/ai/agents/types.d.ts.map +1 -0
  11. package/dist/ai/agents/types.js +3 -0
  12. package/dist/ai/index.d.ts +2 -0
  13. package/dist/ai/index.d.ts.map +1 -0
  14. package/dist/ai/index.js +3 -0
  15. package/dist/ai/providers/rtzr/api.d.ts +22 -0
  16. package/dist/ai/providers/rtzr/api.d.ts.map +1 -0
  17. package/dist/ai/providers/rtzr/api.js +28 -0
  18. package/dist/ai/providers/rtzr/error.d.ts +18 -0
  19. package/dist/ai/providers/rtzr/error.d.ts.map +1 -0
  20. package/dist/ai/providers/rtzr/error.js +29 -0
  21. package/dist/ai/providers/rtzr/index.d.ts +5 -0
  22. package/dist/ai/providers/rtzr/index.d.ts.map +1 -0
  23. package/dist/ai/providers/rtzr/index.js +6 -0
  24. package/dist/ai/providers/rtzr/model.d.ts +52 -0
  25. package/dist/ai/providers/rtzr/model.d.ts.map +1 -0
  26. package/dist/ai/providers/rtzr/model.js +137 -0
  27. package/dist/ai/providers/rtzr/options.d.ts +7 -0
  28. package/dist/ai/providers/rtzr/options.d.ts.map +1 -0
  29. package/dist/ai/providers/rtzr/options.js +47 -0
  30. package/dist/ai/providers/rtzr/provider.d.ts +18 -0
  31. package/dist/ai/providers/rtzr/provider.d.ts.map +1 -0
  32. package/dist/ai/providers/rtzr/provider.js +54 -0
  33. package/dist/ai/providers/rtzr/utils.d.ts +19 -0
  34. package/dist/ai/providers/rtzr/utils.d.ts.map +1 -0
  35. package/dist/ai/providers/rtzr/utils.js +88 -0
  36. package/dist/api/base-frame.d.ts +2 -2
  37. package/dist/api/base-frame.d.ts.map +1 -1
  38. package/dist/api/base-frame.js +13 -2
  39. package/dist/api/caster.d.ts.map +1 -1
  40. package/dist/api/caster.js +71 -2
  41. package/dist/api/code-converters.d.ts +58 -14
  42. package/dist/api/code-converters.d.ts.map +1 -1
  43. package/dist/api/code-converters.js +258 -2
  44. package/dist/api/config.d.ts +90 -0
  45. package/dist/api/config.d.ts.map +1 -0
  46. package/dist/api/config.js +25 -0
  47. package/dist/api/context.d.ts +4 -2
  48. package/dist/api/context.d.ts.map +1 -1
  49. package/dist/api/context.js +3 -2
  50. package/dist/api/decorators.d.ts +20 -6
  51. package/dist/api/decorators.d.ts.map +1 -1
  52. package/dist/api/decorators.js +235 -2
  53. package/dist/api/index.d.ts +2 -2
  54. package/dist/api/index.d.ts.map +1 -1
  55. package/dist/api/index.js +9 -2
  56. package/dist/api/sonamu.d.ts +10 -24
  57. package/dist/api/sonamu.d.ts.map +1 -1
  58. package/dist/api/sonamu.js +514 -2
  59. package/dist/api/validator.d.ts +6 -0
  60. package/dist/api/validator.d.ts.map +1 -0
  61. package/dist/api/validator.js +81 -0
  62. package/dist/bin/build-config.d.ts +6 -1
  63. package/dist/bin/build-config.d.ts.map +1 -1
  64. package/dist/bin/build-config.js +15 -2
  65. package/dist/bin/cli.js +519 -2
  66. package/dist/bin/hot-hook-register.d.ts +11 -0
  67. package/dist/bin/hot-hook-register.d.ts.map +1 -0
  68. package/dist/bin/hot-hook-register.js +21 -0
  69. package/dist/bin/loader-register.d.ts +2 -0
  70. package/dist/bin/loader-register.d.ts.map +1 -0
  71. package/dist/bin/loader-register.js +34 -0
  72. package/dist/database/_batch_update.d.ts +5 -3
  73. package/dist/database/_batch_update.d.ts.map +1 -1
  74. package/dist/database/_batch_update.js +95 -2
  75. package/dist/database/base-model.d.ts +96 -10
  76. package/dist/database/base-model.d.ts.map +1 -1
  77. package/dist/database/base-model.js +390 -2
  78. package/dist/database/base-model.types.d.ts +93 -0
  79. package/dist/database/base-model.types.d.ts.map +1 -0
  80. package/dist/database/base-model.types.js +10 -0
  81. package/dist/database/code-generator.d.ts +1 -1
  82. package/dist/database/code-generator.d.ts.map +1 -1
  83. package/dist/database/code-generator.js +54 -2
  84. package/dist/database/db.d.ts +6 -21
  85. package/dist/database/db.d.ts.map +1 -1
  86. package/dist/database/db.js +129 -2
  87. package/dist/database/puri-subset.test-d.js +81 -0
  88. package/dist/database/puri-subset.types.d.ts +123 -0
  89. package/dist/database/puri-subset.types.d.ts.map +1 -0
  90. package/dist/database/puri-subset.types.js +16 -0
  91. package/dist/database/puri-wrapper.d.ts +13 -11
  92. package/dist/database/puri-wrapper.d.ts.map +1 -1
  93. package/dist/database/puri-wrapper.js +109 -2
  94. package/dist/database/puri.d.ts +41 -23
  95. package/dist/database/puri.d.ts.map +1 -1
  96. package/dist/database/puri.js +601 -2
  97. package/dist/database/puri.types.d.ts +25 -6
  98. package/dist/database/puri.types.d.ts.map +1 -1
  99. package/dist/database/puri.types.js +6 -2
  100. package/dist/database/transaction-context.d.ts +1 -1
  101. package/dist/database/transaction-context.d.ts.map +1 -1
  102. package/dist/database/transaction-context.js +14 -2
  103. package/dist/database/upsert-builder.d.ts +9 -3
  104. package/dist/database/upsert-builder.d.ts.map +1 -1
  105. package/dist/database/upsert-builder.js +365 -2
  106. package/dist/entity/entity-manager.d.ts +167 -2
  107. package/dist/entity/entity-manager.d.ts.map +1 -1
  108. package/dist/entity/entity-manager.js +130 -2
  109. package/dist/entity/entity.d.ts +5 -3
  110. package/dist/entity/entity.d.ts.map +1 -1
  111. package/dist/entity/entity.js +750 -2
  112. package/dist/exceptions/error-handler.d.ts +1 -1
  113. package/dist/exceptions/error-handler.d.ts.map +1 -1
  114. package/dist/exceptions/error-handler.js +29 -2
  115. package/dist/exceptions/so-exceptions.d.ts +1 -1
  116. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  117. package/dist/exceptions/so-exceptions.js +85 -2
  118. package/dist/file-storage/driver.d.ts +1 -1
  119. package/dist/file-storage/driver.d.ts.map +1 -1
  120. package/dist/file-storage/driver.js +79 -2
  121. package/dist/file-storage/file-storage.js +75 -2
  122. package/dist/index.d.ts +18 -9
  123. package/dist/index.d.ts.map +1 -1
  124. package/dist/index.js +34 -2
  125. package/dist/migration/code-generation.d.ts +1 -1
  126. package/dist/migration/code-generation.d.ts.map +1 -1
  127. package/dist/migration/code-generation.js +614 -2
  128. package/dist/migration/migration-set.d.ts +2 -10
  129. package/dist/migration/migration-set.d.ts.map +1 -1
  130. package/dist/migration/migration-set.js +213 -2
  131. package/dist/migration/migrator.d.ts +24 -82
  132. package/dist/migration/migrator.d.ts.map +1 -1
  133. package/dist/migration/migrator.js +330 -2
  134. package/dist/migration/postgresql-schema-reader.d.ts +51 -0
  135. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -0
  136. package/dist/migration/postgresql-schema-reader.js +245 -0
  137. package/dist/migration/types.d.ts +6 -38
  138. package/dist/migration/types.d.ts.map +1 -1
  139. package/dist/migration/types.js +3 -2
  140. package/dist/naite/messaging-types.d.ts +43 -0
  141. package/dist/naite/messaging-types.d.ts.map +1 -0
  142. package/dist/naite/messaging-types.js +7 -0
  143. package/dist/naite/naite-reporter.d.ts +41 -0
  144. package/dist/naite/naite-reporter.d.ts.map +1 -0
  145. package/dist/naite/naite-reporter.js +102 -0
  146. package/dist/naite/naite.d.ts +95 -0
  147. package/dist/naite/naite.d.ts.map +1 -0
  148. package/dist/naite/naite.js +316 -0
  149. package/dist/stream/index.js +3 -2
  150. package/dist/stream/sse.d.ts +2 -2
  151. package/dist/stream/sse.d.ts.map +1 -1
  152. package/dist/stream/sse.js +38 -2
  153. package/dist/syncer/api-parser.d.ts +10 -0
  154. package/dist/syncer/api-parser.d.ts.map +1 -0
  155. package/dist/syncer/api-parser.js +240 -0
  156. package/dist/syncer/checksum.d.ts +21 -0
  157. package/dist/syncer/checksum.d.ts.map +1 -0
  158. package/dist/syncer/checksum.js +98 -0
  159. package/dist/syncer/code-generator.d.ts +20 -0
  160. package/dist/syncer/code-generator.d.ts.map +1 -0
  161. package/dist/syncer/code-generator.js +161 -0
  162. package/dist/syncer/entity-operations.d.ts +17 -0
  163. package/dist/syncer/entity-operations.d.ts.map +1 -0
  164. package/dist/syncer/entity-operations.js +59 -0
  165. package/dist/syncer/file-patterns.d.ts +29 -0
  166. package/dist/syncer/file-patterns.d.ts.map +1 -0
  167. package/dist/syncer/file-patterns.js +38 -0
  168. package/dist/syncer/index.d.ts +6 -0
  169. package/dist/syncer/index.d.ts.map +1 -1
  170. package/dist/syncer/index.js +9 -2
  171. package/dist/syncer/module-loader.d.ts +35 -0
  172. package/dist/syncer/module-loader.d.ts.map +1 -0
  173. package/dist/syncer/module-loader.js +87 -0
  174. package/dist/syncer/syncer.d.ts +98 -106
  175. package/dist/syncer/syncer.d.ts.map +1 -1
  176. package/dist/syncer/syncer.js +422 -2
  177. package/dist/template/entity-converter.d.ts +14 -0
  178. package/dist/template/entity-converter.d.ts.map +1 -0
  179. package/dist/template/entity-converter.js +108 -0
  180. package/dist/template/helpers.d.ts +23 -0
  181. package/dist/template/helpers.d.ts.map +1 -0
  182. package/dist/template/helpers.js +64 -0
  183. package/dist/{templates → template/implementations}/entity.template.d.ts +3 -3
  184. package/dist/template/implementations/entity.template.d.ts.map +1 -0
  185. package/dist/template/implementations/entity.template.js +86 -0
  186. package/dist/{templates → template/implementations}/generated.template.d.ts +3 -4
  187. package/dist/template/implementations/generated.template.d.ts.map +1 -0
  188. package/dist/template/implementations/generated.template.js +249 -0
  189. package/dist/{templates → template/implementations}/generated_http.template.d.ts +3 -4
  190. package/dist/template/implementations/generated_http.template.d.ts.map +1 -0
  191. package/dist/template/implementations/generated_http.template.js +131 -0
  192. package/dist/{templates → template/implementations}/generated_sso.template.d.ts +4 -5
  193. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -0
  194. package/dist/template/implementations/generated_sso.template.js +134 -0
  195. package/dist/{templates → template/implementations}/init_types.template.d.ts +3 -3
  196. package/dist/template/implementations/init_types.template.d.ts.map +1 -0
  197. package/dist/template/implementations/init_types.template.js +38 -0
  198. package/dist/template/implementations/model.template.d.ts +17 -0
  199. package/dist/template/implementations/model.template.d.ts.map +1 -0
  200. package/dist/template/implementations/model.template.js +181 -0
  201. package/dist/{templates → template/implementations}/model_test.template.d.ts +3 -3
  202. package/dist/template/implementations/model_test.template.d.ts.map +1 -0
  203. package/dist/template/implementations/model_test.template.js +35 -0
  204. package/dist/{templates → template/implementations}/service.template.d.ts +6 -6
  205. package/dist/template/implementations/service.template.d.ts.map +1 -0
  206. package/dist/template/implementations/service.template.js +201 -0
  207. package/dist/{templates → template/implementations}/view_enums_buttonset.template.d.ts +3 -3
  208. package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -0
  209. package/dist/template/implementations/view_enums_buttonset.template.js +31 -0
  210. package/dist/{templates → template/implementations}/view_enums_dropdown.template.d.ts +3 -4
  211. package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -0
  212. package/dist/template/implementations/view_enums_dropdown.template.js +50 -0
  213. package/dist/{templates → template/implementations}/view_enums_select.template.d.ts +3 -3
  214. package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -0
  215. package/dist/template/implementations/view_enums_select.template.js +55 -0
  216. package/dist/{templates → template/implementations}/view_form.template.d.ts +5 -5
  217. package/dist/template/implementations/view_form.template.d.ts.map +1 -0
  218. package/dist/template/implementations/view_form.template.js +337 -0
  219. package/dist/{templates → template/implementations}/view_id_all_select.template.d.ts +3 -3
  220. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -0
  221. package/dist/template/implementations/view_id_all_select.template.js +31 -0
  222. package/dist/{templates → template/implementations}/view_id_async_select.template.d.ts +3 -3
  223. package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -0
  224. package/dist/template/implementations/view_id_async_select.template.js +105 -0
  225. package/dist/{templates → template/implementations}/view_list.template.d.ts +5 -13
  226. package/dist/template/implementations/view_list.template.d.ts.map +1 -0
  227. package/dist/template/implementations/view_list.template.js +475 -0
  228. package/dist/template/implementations/view_list_columns.template.d.ts +17 -0
  229. package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -0
  230. package/dist/template/implementations/view_list_columns.template.js +49 -0
  231. package/dist/{templates → template/implementations}/view_search_input.template.d.ts +3 -3
  232. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -0
  233. package/dist/template/implementations/view_search_input.template.js +64 -0
  234. package/dist/template/index.d.ts +7 -0
  235. package/dist/template/index.d.ts.map +1 -0
  236. package/dist/template/index.js +8 -0
  237. package/dist/template/template-manager.d.ts +56 -0
  238. package/dist/template/template-manager.d.ts.map +1 -0
  239. package/dist/template/template-manager.js +125 -0
  240. package/dist/template/template-types.d.ts +16 -0
  241. package/dist/template/template-types.d.ts.map +1 -0
  242. package/dist/template/template-types.js +7 -0
  243. package/dist/template/template.d.ts +49 -0
  244. package/dist/template/template.d.ts.map +1 -0
  245. package/dist/template/template.js +60 -0
  246. package/dist/template/zod-converter.d.ts +51 -0
  247. package/dist/template/zod-converter.d.ts.map +1 -0
  248. package/dist/template/zod-converter.js +449 -0
  249. package/dist/testing/_relation-graph.d.ts +1 -1
  250. package/dist/testing/_relation-graph.d.ts.map +1 -1
  251. package/dist/testing/_relation-graph.js +89 -2
  252. package/dist/testing/fixture-manager.d.ts +42 -11
  253. package/dist/testing/fixture-manager.d.ts.map +1 -1
  254. package/dist/testing/fixture-manager.js +623 -2
  255. package/dist/types/types.d.ts +747 -143
  256. package/dist/types/types.d.ts.map +1 -1
  257. package/dist/types/types.js +546 -2
  258. package/dist/typings/knex.d.js +3 -2
  259. package/dist/utils/async-utils.d.ts +7 -0
  260. package/dist/utils/async-utils.d.ts.map +1 -1
  261. package/dist/utils/async-utils.js +57 -2
  262. package/dist/utils/console-util.d.ts +2 -0
  263. package/dist/utils/console-util.d.ts.map +1 -0
  264. package/dist/utils/console-util.js +6 -0
  265. package/dist/utils/controller.d.ts +1 -0
  266. package/dist/utils/controller.d.ts.map +1 -1
  267. package/dist/utils/controller.js +29 -2
  268. package/dist/utils/esm-utils.d.ts +39 -0
  269. package/dist/utils/esm-utils.d.ts.map +1 -0
  270. package/dist/utils/esm-utils.js +49 -0
  271. package/dist/utils/formatter.d.ts +3 -0
  272. package/dist/utils/formatter.d.ts.map +1 -0
  273. package/dist/utils/formatter.js +110 -0
  274. package/dist/utils/fs-utils.d.ts +1 -1
  275. package/dist/utils/fs-utils.d.ts.map +1 -1
  276. package/dist/utils/fs-utils.js +17 -2
  277. package/dist/utils/lodash-able.d.ts.map +1 -1
  278. package/dist/utils/lodash-able.js +6 -2
  279. package/dist/utils/model.js +22 -2
  280. package/dist/utils/object-utils.d.ts +44 -0
  281. package/dist/utils/object-utils.d.ts.map +1 -0
  282. package/dist/utils/object-utils.js +191 -0
  283. package/dist/utils/path-utils.d.ts +89 -0
  284. package/dist/utils/path-utils.d.ts.map +1 -0
  285. package/dist/utils/path-utils.js +60 -0
  286. package/dist/utils/process-utils.d.ts +13 -0
  287. package/dist/utils/process-utils.d.ts.map +1 -0
  288. package/dist/utils/process-utils.js +36 -0
  289. package/dist/utils/sql-parser.d.ts +5 -1
  290. package/dist/utils/sql-parser.d.ts.map +1 -1
  291. package/dist/utils/sql-parser.js +46 -2
  292. package/dist/utils/type-utils.d.ts +23 -0
  293. package/dist/utils/type-utils.d.ts.map +1 -0
  294. package/dist/utils/type-utils.js +45 -0
  295. package/dist/utils/utils.d.ts +10 -7
  296. package/dist/utils/utils.d.ts.map +1 -1
  297. package/dist/utils/utils.js +72 -2
  298. package/dist/utils/zod-error.d.ts +1 -1
  299. package/dist/utils/zod-error.d.ts.map +1 -1
  300. package/dist/utils/zod-error.js +19 -2
  301. package/package.json +65 -27
  302. package/src/ai/agents/agent.ts +87 -0
  303. package/src/ai/agents/index.ts +2 -0
  304. package/src/ai/agents/types.ts +47 -0
  305. package/src/ai/index.ts +1 -0
  306. package/src/ai/providers/rtzr/api.ts +37 -0
  307. package/src/ai/providers/rtzr/error.ts +34 -0
  308. package/src/ai/providers/rtzr/index.ts +4 -0
  309. package/src/ai/providers/rtzr/model.ts +201 -0
  310. package/src/ai/providers/rtzr/options.ts +49 -0
  311. package/src/ai/providers/rtzr/provider.ts +91 -0
  312. package/src/ai/providers/rtzr/utils.ts +127 -0
  313. package/src/api/base-frame.ts +4 -2
  314. package/src/api/caster.ts +17 -23
  315. package/src/api/code-converters.ts +178 -535
  316. package/src/api/config.ts +125 -0
  317. package/src/api/context.ts +7 -17
  318. package/src/api/decorators.ts +176 -46
  319. package/src/api/index.ts +2 -2
  320. package/src/api/sonamu.ts +190 -167
  321. package/src/api/validator.ts +83 -0
  322. package/src/bin/build-config.ts +8 -1
  323. package/src/bin/cli.ts +258 -124
  324. package/src/bin/hot-hook-register.ts +22 -0
  325. package/src/bin/loader-register.ts +38 -0
  326. package/src/database/_batch_update.ts +46 -31
  327. package/src/database/base-model.ts +390 -182
  328. package/src/database/base-model.types.ts +155 -0
  329. package/src/database/code-generator.ts +13 -32
  330. package/src/database/db.ts +40 -96
  331. package/src/database/puri-subset.test-d.ts +471 -0
  332. package/src/database/puri-subset.types.ts +195 -0
  333. package/src/database/puri-wrapper.ts +58 -67
  334. package/src/database/puri.ts +229 -148
  335. package/src/database/puri.types.ts +76 -30
  336. package/src/database/transaction-context.ts +1 -1
  337. package/src/database/upsert-builder.ts +262 -132
  338. package/src/entity/entity-manager.ts +48 -36
  339. package/src/entity/entity.ts +330 -248
  340. package/src/exceptions/error-handler.ts +3 -3
  341. package/src/exceptions/so-exceptions.ts +11 -11
  342. package/src/file-storage/driver.ts +5 -5
  343. package/src/file-storage/file-storage.ts +2 -2
  344. package/src/index.ts +18 -10
  345. package/src/migration/code-generation.ts +185 -172
  346. package/src/migration/migration-set.ts +80 -293
  347. package/src/migration/migrator.ts +199 -571
  348. package/src/migration/mysql-schema-reader.ts.txt +272 -0
  349. package/src/migration/postgresql-schema-reader.ts +310 -0
  350. package/src/migration/types.ts +6 -39
  351. package/src/naite/messaging-types.ts +51 -0
  352. package/src/naite/naite-reporter.ts +128 -0
  353. package/src/naite/naite.ts +415 -0
  354. package/src/shared/web.shared.ts.txt +20 -24
  355. package/src/stream/sse.ts +5 -5
  356. package/src/syncer/api-parser.ts +282 -0
  357. package/src/syncer/checksum.ts +140 -0
  358. package/src/syncer/code-generator.ts +198 -0
  359. package/src/syncer/entity-operations.ts +65 -0
  360. package/src/syncer/file-patterns.ts +56 -0
  361. package/src/syncer/index.ts +6 -0
  362. package/src/syncer/module-loader.ts +128 -0
  363. package/src/syncer/syncer.ts +389 -1453
  364. package/src/template/entity-converter.ts +114 -0
  365. package/src/template/helpers.ts +81 -0
  366. package/src/{templates → template/implementations}/entity.template.ts +7 -7
  367. package/src/{templates → template/implementations}/generated.template.ts +101 -101
  368. package/src/{templates → template/implementations}/generated_http.template.ts +27 -57
  369. package/src/template/implementations/generated_sso.template.ts +151 -0
  370. package/src/{templates → template/implementations}/init_types.template.ts +5 -7
  371. package/src/{templates → template/implementations}/model.template.ts +52 -43
  372. package/src/{templates → template/implementations}/model_test.template.ts +5 -5
  373. package/src/{templates → template/implementations}/service.template.ts +66 -82
  374. package/src/{templates → template/implementations}/view_enums_buttonset.template.ts +3 -3
  375. package/src/{templates → template/implementations}/view_enums_dropdown.template.ts +4 -20
  376. package/src/{templates → template/implementations}/view_enums_select.template.ts +4 -4
  377. package/src/{templates → template/implementations}/view_form.template.ts +40 -83
  378. package/src/{templates → template/implementations}/view_id_all_select.template.ts +3 -3
  379. package/src/{templates → template/implementations}/view_id_async_select.template.ts +10 -24
  380. package/src/{templates → template/implementations}/view_list.template.ts +60 -152
  381. package/src/{templates → template/implementations}/view_list_columns.template.ts +5 -11
  382. package/src/{templates → template/implementations}/view_search_input.template.ts +3 -3
  383. package/src/template/index.ts +6 -0
  384. package/src/template/template-manager.ts +166 -0
  385. package/src/template/template-types.ts +16 -0
  386. package/src/template/template.ts +105 -0
  387. package/src/template/zod-converter.ts +525 -0
  388. package/src/testing/_relation-graph.ts +18 -11
  389. package/src/testing/fixture-manager.ts +472 -359
  390. package/src/types/types.ts +553 -308
  391. package/src/typings/knex.d.ts +7 -9
  392. package/src/utils/async-utils.ts +23 -10
  393. package/src/utils/console-util.ts +4 -0
  394. package/src/utils/controller.ts +3 -0
  395. package/src/utils/esm-utils.ts +59 -0
  396. package/src/utils/formatter.ts +109 -0
  397. package/src/utils/fs-utils.ts +1 -1
  398. package/src/utils/lodash-able.ts +1 -4
  399. package/src/utils/object-utils.ts +217 -0
  400. package/src/utils/path-utils.ts +99 -0
  401. package/src/utils/process-utils.ts +46 -0
  402. package/src/utils/sql-parser.ts +23 -5
  403. package/src/utils/type-utils.ts +83 -0
  404. package/src/utils/utils.ts +66 -43
  405. package/src/utils/zod-error.ts +3 -4
  406. package/dist/api/base-frame.js.map +0 -1
  407. package/dist/api/caster.js.map +0 -1
  408. package/dist/api/code-converters.js.map +0 -1
  409. package/dist/api/context.js.map +0 -1
  410. package/dist/api/decorators.js.map +0 -1
  411. package/dist/api/index.js.map +0 -1
  412. package/dist/api/sonamu.js.map +0 -1
  413. package/dist/bin/build-config.js.map +0 -1
  414. package/dist/bin/cli-wrapper.d.ts +0 -3
  415. package/dist/bin/cli-wrapper.d.ts.map +0 -1
  416. package/dist/bin/cli-wrapper.js +0 -3
  417. package/dist/bin/cli-wrapper.js.map +0 -1
  418. package/dist/bin/cli.js.map +0 -1
  419. package/dist/database/_batch_update.js.map +0 -1
  420. package/dist/database/base-model.js.map +0 -1
  421. package/dist/database/code-generator.js.map +0 -1
  422. package/dist/database/db.js.map +0 -1
  423. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts +0 -2
  424. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts.map +0 -1
  425. package/dist/database/knex-plugins/knex-on-duplicate-update.js +0 -2
  426. package/dist/database/knex-plugins/knex-on-duplicate-update.js.map +0 -1
  427. package/dist/database/puri-wrapper.js.map +0 -1
  428. package/dist/database/puri.js.map +0 -1
  429. package/dist/database/puri.types.js.map +0 -1
  430. package/dist/database/transaction-context.js.map +0 -1
  431. package/dist/database/upsert-builder.js.map +0 -1
  432. package/dist/entity/entity-manager.js.map +0 -1
  433. package/dist/entity/entity-utils.d.ts +0 -61
  434. package/dist/entity/entity-utils.d.ts.map +0 -1
  435. package/dist/entity/entity-utils.js +0 -2
  436. package/dist/entity/entity-utils.js.map +0 -1
  437. package/dist/entity/entity.js.map +0 -1
  438. package/dist/exceptions/error-handler.js.map +0 -1
  439. package/dist/exceptions/so-exceptions.js.map +0 -1
  440. package/dist/file-storage/driver.js.map +0 -1
  441. package/dist/file-storage/file-storage.js.map +0 -1
  442. package/dist/index.js.map +0 -1
  443. package/dist/migration/code-generation.js.map +0 -1
  444. package/dist/migration/migration-set.js.map +0 -1
  445. package/dist/migration/migrator.js.map +0 -1
  446. package/dist/migration/types.js.map +0 -1
  447. package/dist/stream/index.js.map +0 -1
  448. package/dist/stream/sse.js.map +0 -1
  449. package/dist/syncer/index.js.map +0 -1
  450. package/dist/syncer/syncer.js.map +0 -1
  451. package/dist/templates/base-template.d.ts +0 -13
  452. package/dist/templates/base-template.d.ts.map +0 -1
  453. package/dist/templates/base-template.js +0 -2
  454. package/dist/templates/base-template.js.map +0 -1
  455. package/dist/templates/entity.template.d.ts.map +0 -1
  456. package/dist/templates/entity.template.js +0 -2
  457. package/dist/templates/entity.template.js.map +0 -1
  458. package/dist/templates/generated.template.d.ts.map +0 -1
  459. package/dist/templates/generated.template.js +0 -2
  460. package/dist/templates/generated.template.js.map +0 -1
  461. package/dist/templates/generated_http.template.d.ts.map +0 -1
  462. package/dist/templates/generated_http.template.js +0 -2
  463. package/dist/templates/generated_http.template.js.map +0 -1
  464. package/dist/templates/generated_sso.template.d.ts.map +0 -1
  465. package/dist/templates/generated_sso.template.js +0 -2
  466. package/dist/templates/generated_sso.template.js.map +0 -1
  467. package/dist/templates/index.d.ts +0 -2
  468. package/dist/templates/index.d.ts.map +0 -1
  469. package/dist/templates/index.js +0 -2
  470. package/dist/templates/index.js.map +0 -1
  471. package/dist/templates/init_types.template.d.ts.map +0 -1
  472. package/dist/templates/init_types.template.js +0 -2
  473. package/dist/templates/init_types.template.js.map +0 -1
  474. package/dist/templates/model.template.d.ts +0 -17
  475. package/dist/templates/model.template.d.ts.map +0 -1
  476. package/dist/templates/model.template.js +0 -2
  477. package/dist/templates/model.template.js.map +0 -1
  478. package/dist/templates/model_test.template.d.ts.map +0 -1
  479. package/dist/templates/model_test.template.js +0 -2
  480. package/dist/templates/model_test.template.js.map +0 -1
  481. package/dist/templates/service.template.d.ts.map +0 -1
  482. package/dist/templates/service.template.js +0 -2
  483. package/dist/templates/service.template.js.map +0 -1
  484. package/dist/templates/view_enums_buttonset.template.d.ts.map +0 -1
  485. package/dist/templates/view_enums_buttonset.template.js +0 -2
  486. package/dist/templates/view_enums_buttonset.template.js.map +0 -1
  487. package/dist/templates/view_enums_dropdown.template.d.ts.map +0 -1
  488. package/dist/templates/view_enums_dropdown.template.js +0 -2
  489. package/dist/templates/view_enums_dropdown.template.js.map +0 -1
  490. package/dist/templates/view_enums_select.template.d.ts.map +0 -1
  491. package/dist/templates/view_enums_select.template.js +0 -2
  492. package/dist/templates/view_enums_select.template.js.map +0 -1
  493. package/dist/templates/view_form.template.d.ts.map +0 -1
  494. package/dist/templates/view_form.template.js +0 -2
  495. package/dist/templates/view_form.template.js.map +0 -1
  496. package/dist/templates/view_id_all_select.template.d.ts.map +0 -1
  497. package/dist/templates/view_id_all_select.template.js +0 -2
  498. package/dist/templates/view_id_all_select.template.js.map +0 -1
  499. package/dist/templates/view_id_async_select.template.d.ts.map +0 -1
  500. package/dist/templates/view_id_async_select.template.js +0 -2
  501. package/dist/templates/view_id_async_select.template.js.map +0 -1
  502. package/dist/templates/view_list.template.d.ts.map +0 -1
  503. package/dist/templates/view_list.template.js +0 -2
  504. package/dist/templates/view_list.template.js.map +0 -1
  505. package/dist/templates/view_list_columns.template.d.ts +0 -17
  506. package/dist/templates/view_list_columns.template.d.ts.map +0 -1
  507. package/dist/templates/view_list_columns.template.js +0 -2
  508. package/dist/templates/view_list_columns.template.js.map +0 -1
  509. package/dist/templates/view_search_input.template.d.ts.map +0 -1
  510. package/dist/templates/view_search_input.template.js +0 -2
  511. package/dist/templates/view_search_input.template.js.map +0 -1
  512. package/dist/testing/_relation-graph.js.map +0 -1
  513. package/dist/testing/fixture-manager.js.map +0 -1
  514. package/dist/types/types.js.map +0 -1
  515. package/dist/typings/knex.d.js.map +0 -1
  516. package/dist/utils/async-utils.js.map +0 -1
  517. package/dist/utils/controller.js.map +0 -1
  518. package/dist/utils/fs-utils.js.map +0 -1
  519. package/dist/utils/lodash-able.js.map +0 -1
  520. package/dist/utils/model.js.map +0 -1
  521. package/dist/utils/sql-parser.js.map +0 -1
  522. package/dist/utils/utils.js.map +0 -1
  523. package/dist/utils/zod-error.js.map +0 -1
  524. package/src/bin/cli-wrapper.ts +0 -75
  525. package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
  526. package/src/entity/entity-utils.ts +0 -291
  527. package/src/templates/base-template.ts +0 -19
  528. package/src/templates/generated_sso.template.ts +0 -138
  529. package/src/templates/index.ts +0 -1
@@ -1,2 +1,750 @@
1
- "use strict";Object.defineProperty(exports,"__esModule",{value:true});Object.defineProperty(exports,"Entity",{enumerable:true,get:function(){return Entity}});var _lodash=/*#__PURE__*/_interop_require_default(require("lodash"));var _entitymanager=require("./entity-manager");var _types=require("../types/types");var _inflection=/*#__PURE__*/_interop_require_default(require("inflection"));var _path=/*#__PURE__*/_interop_require_default(require("path"));var _promises=require("fs/promises");var _zod=require("zod");var _sonamu=require("../api/sonamu");var _prettier=/*#__PURE__*/_interop_require_default(require("prettier"));var _utils=require("../utils/utils");var _fsutils=require("../utils/fs-utils");function _array_like_to_array(arr,len){if(len==null||len>arr.length)len=arr.length;for(var i=0,arr2=new Array(len);i<len;i++)arr2[i]=arr[i];return arr2}function _array_with_holes(arr){if(Array.isArray(arr))return arr}function _array_without_holes(arr){if(Array.isArray(arr))return _array_like_to_array(arr)}function asyncGeneratorStep(gen,resolve,reject,_next,_throw,key,arg){try{var info=gen[key](arg);var value=info.value}catch(error){reject(error);return}if(info.done){resolve(value)}else{Promise.resolve(value).then(_next,_throw)}}function _async_to_generator(fn){return function(){var self=this,args=arguments;return new Promise(function(resolve,reject){var gen=fn.apply(self,args);function _next(value){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"next",value)}function _throw(err){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"throw",err)}_next(undefined)})}}function _class_call_check(instance,Constructor){if(!(instance instanceof Constructor)){throw new TypeError("Cannot call a class as a function")}}function _defineProperties(target,props){for(var i=0;i<props.length;i++){var descriptor=props[i];descriptor.enumerable=descriptor.enumerable||false;descriptor.configurable=true;if("value"in descriptor)descriptor.writable=true;Object.defineProperty(target,descriptor.key,descriptor)}}function _create_class(Constructor,protoProps,staticProps){if(protoProps)_defineProperties(Constructor.prototype,protoProps);if(staticProps)_defineProperties(Constructor,staticProps);return Constructor}function _define_property(obj,key,value){if(key in obj){Object.defineProperty(obj,key,{value:value,enumerable:true,configurable:true,writable:true})}else{obj[key]=value}return obj}function _interop_require_default(obj){return obj&&obj.__esModule?obj:{default:obj}}function _getRequireWildcardCache(nodeInterop){if(typeof WeakMap!=="function")return null;var cacheBabelInterop=new WeakMap;var cacheNodeInterop=new WeakMap;return(_getRequireWildcardCache=function(nodeInterop){return nodeInterop?cacheNodeInterop:cacheBabelInterop})(nodeInterop)}function _interop_require_wildcard(obj,nodeInterop){if(!nodeInterop&&obj&&obj.__esModule){return obj}if(obj===null||typeof obj!=="object"&&typeof obj!=="function"){return{default:obj}}var cache=_getRequireWildcardCache(nodeInterop);if(cache&&cache.has(obj)){return cache.get(obj)}var newObj={__proto__:null};var hasPropertyDescriptor=Object.defineProperty&&Object.getOwnPropertyDescriptor;for(var key in obj){if(key!=="default"&&Object.prototype.hasOwnProperty.call(obj,key)){var desc=hasPropertyDescriptor?Object.getOwnPropertyDescriptor(obj,key):null;if(desc&&(desc.get||desc.set)){Object.defineProperty(newObj,key,desc)}else{newObj[key]=obj[key]}}}newObj.default=obj;if(cache){cache.set(obj,newObj)}return newObj}function _iterable_to_array(iter){if(typeof Symbol!=="undefined"&&iter[Symbol.iterator]!=null||iter["@@iterator"]!=null)return Array.from(iter)}function _iterable_to_array_limit(arr,i){var _i=arr==null?null:typeof Symbol!=="undefined"&&arr[Symbol.iterator]||arr["@@iterator"];if(_i==null)return;var _arr=[];var _n=true;var _d=false;var _s,_e;try{for(_i=_i.call(arr);!(_n=(_s=_i.next()).done);_n=true){_arr.push(_s.value);if(i&&_arr.length===i)break}}catch(err){_d=true;_e=err}finally{try{if(!_n&&_i["return"]!=null)_i["return"]()}finally{if(_d)throw _e}}return _arr}function _non_iterable_rest(){throw new TypeError("Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function _non_iterable_spread(){throw new TypeError("Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function _object_spread(target){for(var i=1;i<arguments.length;i++){var source=arguments[i]!=null?arguments[i]:{};var ownKeys=Object.keys(source);if(typeof Object.getOwnPropertySymbols==="function"){ownKeys=ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function(sym){return Object.getOwnPropertyDescriptor(source,sym).enumerable}))}ownKeys.forEach(function(key){_define_property(target,key,source[key])})}return target}function ownKeys(object,enumerableOnly){var keys=Object.keys(object);if(Object.getOwnPropertySymbols){var symbols=Object.getOwnPropertySymbols(object);if(enumerableOnly){symbols=symbols.filter(function(sym){return Object.getOwnPropertyDescriptor(object,sym).enumerable})}keys.push.apply(keys,symbols)}return keys}function _object_spread_props(target,source){source=source!=null?source:{};if(Object.getOwnPropertyDescriptors){Object.defineProperties(target,Object.getOwnPropertyDescriptors(source))}else{ownKeys(Object(source)).forEach(function(key){Object.defineProperty(target,key,Object.getOwnPropertyDescriptor(source,key))})}return target}function _sliced_to_array(arr,i){return _array_with_holes(arr)||_iterable_to_array_limit(arr,i)||_unsupported_iterable_to_array(arr,i)||_non_iterable_rest()}function _to_array(arr){return _array_with_holes(arr)||_iterable_to_array(arr)||_unsupported_iterable_to_array(arr)||_non_iterable_rest()}function _to_consumable_array(arr){return _array_without_holes(arr)||_iterable_to_array(arr)||_unsupported_iterable_to_array(arr)||_non_iterable_spread()}function _unsupported_iterable_to_array(o,minLen){if(!o)return;if(typeof o==="string")return _array_like_to_array(o,minLen);var n=Object.prototype.toString.call(o).slice(8,-1);if(n==="Object"&&o.constructor)n=o.constructor.name;if(n==="Map"||n==="Set")return Array.from(n);if(n==="Arguments"||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return _array_like_to_array(o,minLen)}function _ts_generator(thisArg,body){var f,y,t,_={label:0,sent:function(){if(t[0]&1)throw t[1];return t[1]},trys:[],ops:[]},g=Object.create((typeof Iterator==="function"?Iterator:Object).prototype);return g.next=verb(0),g["throw"]=verb(1),g["return"]=verb(2),typeof Symbol==="function"&&(g[Symbol.iterator]=function(){return this}),g;function verb(n){return function(v){return step([n,v])}}function step(op){if(f)throw new TypeError("Generator is already executing.");while(g&&(g=0,op[0]&&(_=0)),_)try{if(f=1,y&&(t=op[0]&2?y["return"]:op[0]?y["throw"]||((t=y["return"])&&t.call(y),0):y.next)&&!(t=t.call(y,op[1])).done)return t;if(y=0,t)op=[op[0]&2,t.value];switch(op[0]){case 0:case 1:t=op;break;case 4:_.label++;return{value:op[1],done:false};case 5:_.label++;y=op[1];op=[0];continue;case 7:op=_.ops.pop();_.trys.pop();continue;default:if(!(t=_.trys,t=t.length>0&&t[t.length-1])&&(op[0]===6||op[0]===2)){_=0;continue}if(op[0]===3&&(!t||op[1]>t[0]&&op[1]<t[3])){_.label=op[1];break}if(op[0]===6&&_.label<t[1]){_.label=t[1];t=op;break}if(t&&_.label<t[2]){_.label=t[2];_.ops.push(op);break}if(t[2])_.ops.pop();_.trys.pop();continue}op=body.call(thisArg,_)}catch(e){op=[6,e];y=0}finally{f=t=0}if(op[0]&5)throw op[1];return{value:op[0]?op[1]:void 0,done:true}}}var Entity=/*#__PURE__*/function(){"use strict";function Entity(param){var id=param.id,parentId=param.parentId,table=param.table,title=param.title,props=param.props,indexes=param.indexes,subsets=param.subsets,enums=param.enums;_class_call_check(this,Entity);_define_property(this,"id",void 0);_define_property(this,"parentId",void 0);_define_property(this,"table",void 0);_define_property(this,"title",void 0);_define_property(this,"names",void 0);_define_property(this,"props",void 0);_define_property(this,"propsDict",void 0);_define_property(this,"relations",void 0);_define_property(this,"indexes",void 0);_define_property(this,"subsets",void 0);_define_property(this,"types",{});_define_property(this,"enums",{});_define_property(this,"enumLabels",{});this.id=id;this.parentId=parentId;this.title=title!==null&&title!==void 0?title:this.id;this.table=table!==null&&table!==void 0?table:_inflection.default.underscore(_inflection.default.pluralize(id));if(props){this.props=props.map(function(prop){if((0,_types.isEnumProp)(prop)){if(prop.id.includes("$Model")){prop.id=prop.id.replace("$Model",id)}}return prop});this.propsDict=props.reduce(function(result,prop){return _object_spread_props(_object_spread({},result),_define_property({},prop.name,prop))},{});this.relations=props.filter(function(prop){return(0,_types.isRelationProp)(prop)}).reduce(function(result,prop){return _object_spread_props(_object_spread({},result),_define_property({},prop.name,prop))},{})}else{this.props=[];this.propsDict={};this.relations={}}this.indexes=indexes!==null&&indexes!==void 0?indexes:[];this.subsets=subsets!==null&&subsets!==void 0?subsets:{};this.enumLabels=enums!==null&&enums!==void 0?enums:{};this.enums=Object.fromEntries(Object.entries(this.enumLabels).map(function(param){var _param=_sliced_to_array(param,2),key=_param[0],enumLabel=_param[1];return[key,_zod.z.enum(Object.keys(enumLabel))]}));this.names={parentFs:_inflection.default.dasherize(_inflection.default.underscore(parentId!==null&&parentId!==void 0?parentId:id)).toLowerCase(),fs:_inflection.default.dasherize(_inflection.default.underscore(id)).toLowerCase(),module:id}}_create_class(Entity,[{key:"getSubsetQuery",value:function getSubsetQuery(subsetKey){var subset=this.subsets[subsetKey];var result=this.resolveSubsetQuery("",subset);return result}},{key:"resolveSubsetQuery",value:function resolveSubsetQuery(prefix,fields){var _this=this;var isAlreadyOuterJoined=arguments.length>2&&arguments[2]!==void 0?arguments[2]:false;prefix=prefix.replace(/\./g,"__");var subsetGroup=_lodash.default.groupBy(fields,function(field){if(field.includes(".")){var _field_split=_sliced_to_array(field.split("."),1),rel=_field_split[0];return rel}else{return""}});var result=Object.keys(subsetGroup).reduce(function(r,groupKey){var _$fields=subsetGroup[groupKey];if(groupKey===""){var realFields=_$fields.filter(function(field){return!(0,_types.isVirtualProp)(_this.propsDict[field])});var virtualFields=_$fields.filter(function(field){return(0,_types.isVirtualProp)(_this.propsDict[field])});if(prefix===""){r.select=r.select.concat(realFields.map(function(field){return"".concat(_this.table,".").concat(field)}));r.virtual=r.virtual.concat(virtualFields)}else{r.select=r.select.concat(realFields.map(function(field){return"".concat(prefix,".").concat(field," as ").concat(prefix,"__").concat(field)}))}return r}var relation=_this.relations[groupKey];if(relation===undefined){throw new Error("존재하지 않는 relation 참조 ".concat(groupKey))}var relEntity=_entitymanager.EntityManager.get(relation.with);if((0,_types.isOneToOneRelationProp)(relation)||(0,_types.isBelongsToOneRelationProp)(relation)){var relFields=_$fields.map(function(field){return field.split(".").slice(1).join(".")});if(relFields.length===1&&relFields[0]==="id"){if(prefix===""){r.select=r.select.concat("".concat(_this.table,".").concat(groupKey,"_id"))}else{r.select=r.select.concat("".concat(prefix,".").concat(groupKey,"_id as ").concat(prefix,"__").concat(groupKey,"_id"))}return r}var innerOrOuter=function(){if(isAlreadyOuterJoined){return"outer"}if((0,_types.isOneToOneRelationProp)(relation)){var _relation_nullable;if(relation.hasJoinColumn===true&&((_relation_nullable=relation.nullable)!==null&&_relation_nullable!==void 0?_relation_nullable:false)===false){return"inner"}else{return"outer"}}else{if(relation.nullable){return"outer"}else{return"inner"}}}();var relSubsetQuery=relEntity.resolveSubsetQuery("".concat(prefix!==""?prefix+".":"").concat(groupKey),relFields,innerOrOuter==="outer");r.select=r.select.concat(relSubsetQuery.select);r.virtual=r.virtual.concat(relSubsetQuery.virtual);var joinAs=prefix===""?groupKey:prefix+"__"+groupKey;var fromTable=prefix===""?_this.table:prefix;var joinClause;if(relation.customJoinClause){joinClause={custom:relation.customJoinClause}}else{var from,to;if((0,_types.isOneToOneRelationProp)(relation)){if(relation.hasJoinColumn){from="".concat(fromTable,".").concat(relation.name,"_id");to="".concat(joinAs,".id")}else{from="".concat(fromTable,".id");to="".concat(joinAs,".").concat(_inflection.default.underscore(_this.names.fs.replace(/\-/g,"_")),"_id")}}else{from="".concat(fromTable,".").concat(relation.name,"_id");to="".concat(joinAs,".id")}joinClause={from:from,to:to}}r.joins.push(_object_spread({as:joinAs,join:innerOrOuter,table:relEntity.table},joinClause));if(relSubsetQuery.loaders.length>0){var convertedLoaders=relSubsetQuery.loaders.map(function(loader){var newAs=[groupKey,loader.as].join("__");return{as:newAs,table:loader.table,manyJoin:loader.manyJoin,oneJoins:loader.oneJoins,select:loader.select,loaders:loader.loaders}});r.loaders=_to_consumable_array(r.loaders).concat(_to_consumable_array(convertedLoaders))}r.joins=r.joins.concat(relSubsetQuery.joins)}else if((0,_types.isHasManyRelationProp)(relation)||(0,_types.isManyToManyRelationProp)(relation)){var relFields1=_$fields.map(function(field){return field.split(".").slice(1).join(".")});var relSubsetQuery1=relEntity.resolveSubsetQuery("",relFields1);var manyJoin;if((0,_types.isHasManyRelationProp)(relation)){var _relation_fromColumn;var fromCol=(_relation_fromColumn=relation===null||relation===void 0?void 0:relation.fromColumn)!==null&&_relation_fromColumn!==void 0?_relation_fromColumn:"id";manyJoin={fromTable:_this.table,fromCol:fromCol,idField:prefix===""?"".concat(fromCol):"".concat(prefix,"__").concat(fromCol),toTable:relEntity.table,toCol:relation.joinColumn}}else if((0,_types.isManyToManyRelationProp)(relation)){manyJoin={fromTable:_this.table,fromCol:"id",idField:prefix===""?"id":"".concat(prefix,"__id"),through:{table:relation.joinTable,fromCol:"".concat(_inflection.default.singularize(_this.table),"_id"),toCol:"".concat(_inflection.default.singularize(relEntity.table),"_id")},toTable:relEntity.table,toCol:"id"}}else{throw new Error}r.loaders.push({as:groupKey,table:relEntity.table,manyJoin:manyJoin,oneJoins:relSubsetQuery1.joins,select:relSubsetQuery1.select,loaders:relSubsetQuery1.loaders})}return r},{select:[],virtual:[],joins:[],loaders:[]});return result}},{key:"fieldExprsToPropNodes",value:function fieldExprsToPropNodes(fieldExprs){var _this=this;var entity=arguments.length>1&&arguments[1]!==void 0?arguments[1]:this;var groups=fieldExprs.reduce(function(result,fieldExpr){var key,value,elseExpr;if(fieldExpr.includes(".")){var ref;ref=_to_array(fieldExpr.split(".")),key=ref[0],elseExpr=ref.slice(1),ref;value=elseExpr.join(".")}else{key="";value=fieldExpr}var _result_key;result[key]=((_result_key=result[key])!==null&&_result_key!==void 0?_result_key:[]).concat(value);return result},{});return Object.keys(groups).map(function(key){var group=groups[key];if(key===""){return group.map(function(propName){if(propName==="uuid"){return{nodeType:"plain",prop:{type:"string",name:"uuid",length:128},children:[]}}var prop=entity.props.find(function(p){return p.name===propName});if(prop===undefined){console.log({propName:propName,groups:groups});throw new Error("".concat(entity.id," -- 잘못된 FieldExpr ").concat(propName))}return{nodeType:"plain",prop:prop,children:[]}})}var prop=entity.propsDict[key];if(!(0,_types.isRelationProp)(prop)){throw new Error("잘못된 FieldExpr ".concat(key,".").concat(group[0]))}var relEntity=_entitymanager.EntityManager.get(prop.with);if((0,_types.isBelongsToOneRelationProp)(prop)||(0,_types.isOneToOneRelationProp)(prop)){if(group.length==1&&(group[0]==="id"||group[0]=="id?")){var idProp=relEntity.propsDict.id;return{nodeType:"plain",prop:_object_spread_props(_object_spread({},idProp),{name:key+"_id",nullable:prop.nullable}),children:[]}}}var children=_this.fieldExprsToPropNodes(group,relEntity);var nodeType=(0,_types.isBelongsToOneRelationProp)(prop)||(0,_types.isOneToOneRelationProp)(prop)?"object":"array";return{prop:prop,children:children,nodeType:nodeType}}).flat()}},{key:"getFieldExprs",value:function getFieldExprs(){var _this=this;var prefix=arguments.length>0&&arguments[0]!==void 0?arguments[0]:"",maxDepth=arguments.length>1&&arguments[1]!==void 0?arguments[1]:3,froms=arguments.length>2&&arguments[2]!==void 0?arguments[2]:[];return this.props.map(function(prop){var propName=[prefix,prop.name].filter(function(v){return v!==""}).join(".");if(propName===prefix){return null}if((0,_types.isRelationProp)(prop)){if(maxDepth<0){return null}if(froms.includes(prop.with)){return null}var relMd=_entitymanager.EntityManager.get(prop.with);return relMd.getFieldExprs(propName,maxDepth-1,_to_consumable_array(froms).concat([_this.id]))}return propName}).flat().filter(function(f){return f!==null})}},{key:"getTableColumns",value:function getTableColumns(){return this.props.map(function(prop){if(prop.type==="relation"){if(prop.relationType==="BelongsToOne"||prop.relationType==="OneToOne"&&prop.hasJoinColumn===true){return{name:"".concat(prop.name,"_id"),type:"int_unsigned"}}else{return null}}return{name:prop.name,type:prop.type}}).filter(_utils.nonNullable)}},{key:"registerModulePaths",value:function registerModulePaths(){return _async_to_generator(function(){var _this,basePath,typesModulePath,typesFileDistPath,importPath,t;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;basePath="".concat(this.names.parentFs);_entitymanager.EntityManager.setModulePath("".concat(this.id,"BaseSchema"),"sonamu.generated");if(Object.keys(this.subsets).length>0){_entitymanager.EntityManager.setModulePath("".concat(this.id,"SubsetKey"),"sonamu.generated");_entitymanager.EntityManager.setModulePath("".concat(this.id,"SubsetMapping"),"sonamu.generated");Object.keys(this.subsets).map(function(subsetKey){_entitymanager.EntityManager.setModulePath("".concat(_this.id,"Subset").concat(subsetKey.toUpperCase()),"sonamu.generated")})}Object.keys(this.enumLabels).map(function(enumId){_entitymanager.EntityManager.setModulePath(enumId,"sonamu.generated")});typesModulePath="".concat(basePath,"/").concat(this.names.parentFs,".types");typesFileDistPath=_path.default.join(_sonamu.Sonamu.apiRootPath,"dist/application/".concat(typesModulePath,".js"));return[4,(0,_fsutils.exists)(typesFileDistPath)];case 1:if(!_state.sent())return[3,3];importPath=_path.default.relative(__dirname,typesFileDistPath);return[4,Promise.resolve(importPath).then(function(p){return /*#__PURE__*/_interop_require_wildcard(require(p))})];case 2:t=_state.sent();this.types=Object.keys(t).reduce(function(result,key){_entitymanager.EntityManager.setModulePath(key,typesModulePath);return _object_spread_props(_object_spread({},result),_define_property({},key,t[key]))},{});_state.label=3;case 3:return[2]}})}).call(this)}},{key:"registerTableSpecs",value:function registerTableSpecs(){var uniqueIndexes=this.indexes.filter(function(idx){return idx.type==="unique"});_entitymanager.EntityManager.setTableSpec({name:this.table,uniqueIndexes:uniqueIndexes})}},{key:"toJson",value:function toJson(){return{id:this.id,parentId:this.parentId,table:this.table,title:this.title,props:this.props,indexes:this.indexes,subsets:this.subsets,enums:this.enumLabels}}},{key:"save",value:function save(){return _async_to_generator(function(){var _this,subsetRows,jsonPath,json,_tmp;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;subsetRows=this.getSubsetRows();this.subsets=Object.fromEntries(Object.entries(this.subsets).map(function(param){var _param=_sliced_to_array(param,1),subsetKey=_param[0];return[subsetKey,_this.subsetRowsToSubsetFields(subsetRows,subsetKey)]}));jsonPath=_path.default.join(_sonamu.Sonamu.apiRootPath,"src/application/".concat(this.names.parentFs,"/").concat(this.names.fs,".entity.json"));json=this.toJson();_tmp=[jsonPath];return[4,_prettier.default.format(JSON.stringify(json),{parser:"json"})];case 1:return[4,_promises.writeFile.apply(void 0,_tmp.concat([_state.sent()]))];case 2:_state.sent();return[4,_entitymanager.EntityManager.register(json)];case 3:_state.sent();return[2]}})}).call(this)}},{key:"getSubsetRows",value:function getSubsetRows(_subsets){var prefixes=arguments.length>1&&arguments[1]!==void 0?arguments[1]:[];if(prefixes.length>10){return[]}var subsets=_subsets!==null&&_subsets!==void 0?_subsets:this.subsets;var subsetKeys=Object.keys(subsets);var allFields=_lodash.default.uniq(subsetKeys.map(function(key){return subsets[key]}).flat());return this.props.map(function(prop){if(prop.type==="relation"&&allFields.find(function(f){return f.startsWith(_to_consumable_array(prefixes).concat([prop.name]).join(".")+".")})){var relEntity=_entitymanager.EntityManager.get(prop.with);var children=relEntity.getSubsetRows(subsets,_to_consumable_array(prefixes).concat(["".concat(prop.name)]));return{field:prop.name,children:children,relationEntity:prop.with,prefixes:prefixes,isOpen:children.length>0,has:Object.fromEntries(subsetKeys.map(function(subsetKey){return[subsetKey,children.every(function(child){return child.has[subsetKey]===true})]}))}}return{field:prop.name,children:[],relationEntity:prop.type==="relation"?prop.with:undefined,prefixes:prefixes,has:Object.fromEntries(subsetKeys.map(function(subsetKey){var subsetFields=subsets[subsetKey];var has=subsetFields.some(function(f){var field=_to_consumable_array(prefixes).concat([prop.name]).join(".");return f===field||f.startsWith(field+".")});return[subsetKey,has]}))}})}},{key:"subsetRowsToSubsetFields",value:function subsetRowsToSubsetFields(subsetRows,subsetKey){var _this=this;return subsetRows.map(function(subsetRow){if(subsetRow.children.length>0){return _this.subsetRowsToSubsetFields(subsetRow.children,subsetKey)}else if(subsetRow.has[subsetKey]){return subsetRow.prefixes.concat(subsetRow.field).join(".")}else{return null}}).filter(_utils.nonNullable).flat()}},{key:"createProp",value:function createProp(prop,at){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:if(!at){this.props.push(prop)}else{this.props.splice(at,0,prop)}return[4,this.save()];case 1:_state.sent();return[2]}})}).call(this)}},{key:"analyzeSubsetField",value:function analyzeSubsetField(subsetField){var _loop=function(i){var propName=arr[i];result.push({entityId:entityId,propName:propName});var prop=_entitymanager.EntityManager.get(entityId).props.find(function(p){return p.name===propName});if(!prop){throw new Error("".concat(entityId,"의 잘못된 서브셋키 ").concat(subsetField))}if((0,_types.isRelationProp)(prop)){entityId=prop.with}};var arr=subsetField.split(".");var entityId=this.id;var result=[];for(var i=0;i<arr.length;i++)_loop(i);return result}},{key:"modifyProp",value:function modifyProp(newProp,at){return _async_to_generator(function(){var _this,oldName,entities,allEntityIds,_iteratorNormalCompletion,_didIteratorError,_iteratorError,_loop,_iterator,_step;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;oldName=this.props[at].name;entities=[this];if(oldName!==newProp.name){allEntityIds=_entitymanager.EntityManager.getAllIds();_iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{_loop=function(){var relEntityId=_step.value;var relEntity=_entitymanager.EntityManager.get(relEntityId);var relEntitySubsetKeys=Object.keys(relEntity.subsets);var _iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{for(var _iterator=relEntitySubsetKeys[Symbol.iterator](),_step1;!(_iteratorNormalCompletion=(_step1=_iterator.next()).done);_iteratorNormalCompletion=true){var subsetKey=_step1.value;var subset=relEntity.subsets[subsetKey];var modifiedSubsetFields=subset.map(function(subsetField){var analyzed=relEntity.analyzeSubsetField(subsetField);var modified=analyzed.map(function(a){return a.propName===oldName&&a.entityId===_this.id?_object_spread_props(_object_spread({},a),{propName:newProp.name}):a});return modified.map(function(a){return a.propName}).join(".")});if(subset.join(",")!==modifiedSubsetFields.join(",")){relEntity.subsets[subsetKey]=modifiedSubsetFields;entities.push(relEntity)}}}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}};for(_iterator=allEntityIds[Symbol.iterator]();!(_iteratorNormalCompletion=(_step=_iterator.next()).done);_iteratorNormalCompletion=true)_loop()}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}}this.props[at]=newProp;return[4,Promise.all(entities.map(function(entity){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,entity.save()]})})()}))];case 1:_state.sent();return[2]}})}).call(this)}},{key:"delProp",value:function delProp(at){return _async_to_generator(function(){var _this,oldName,entities,allEntityIds,_iteratorNormalCompletion,_didIteratorError,_iteratorError,_loop,_iterator,_step;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;oldName=this.props[at].name;entities=[this];allEntityIds=_entitymanager.EntityManager.getAllIds();_iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{_loop=function(){var relEntityId=_step.value;var relEntity=_entitymanager.EntityManager.get(relEntityId);var relEntitySubsetKeys=Object.keys(relEntity.subsets);var _iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{for(var _iterator=relEntitySubsetKeys[Symbol.iterator](),_step1;!(_iteratorNormalCompletion=(_step1=_iterator.next()).done);_iteratorNormalCompletion=true){var subsetKey=_step1.value;var subset=relEntity.subsets[subsetKey];var modifiedSubsetFields=subset.map(function(subsetField){var analyzed=relEntity.analyzeSubsetField(subsetField);if(analyzed.find(function(a){return a.propName===oldName&&a.entityId===_this.id})){return null}else{return subsetField}}).filter(_utils.nonNullable);if(subset.join(",")!==modifiedSubsetFields.join(",")){relEntity.subsets[subsetKey]=modifiedSubsetFields;entities.push(relEntity)}}}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}};for(_iterator=allEntityIds[Symbol.iterator]();!(_iteratorNormalCompletion=(_step=_iterator.next()).done);_iteratorNormalCompletion=true)_loop()}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}_entitymanager.EntityManager.get(this.id).indexes.map(function(index){index.columns=index.columns.filter(function(col){return col!==oldName})});this.props.splice(at,1);return[4,Promise.all(entities.map(function(entity){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,entity.save()]})})()}))];case 1:_state.sent();return[2]}})}).call(this)}},{key:"getEntityIdFromSubsetField",value:function getEntityIdFromSubsetField(subsetField){var _this=this;if(subsetField.includes(".")===false){return this.id}var arr=subsetField.split(".").slice(0,-1);var lastEntityId=arr.reduce(function(entityId,field){var relProp=_entitymanager.EntityManager.get(entityId).props.find(function(p){return p.name===field});if(!relProp||relProp.type!=="relation"){console.debug({arr:arr,thisId:_this.id,entityId:entityId,field:field});throw new Error("잘못된 서브셋키 ".concat(subsetField))}return relProp.with},this.id);return lastEntityId}},{key:"moveProp",value:function moveProp(at,to){return _async_to_generator(function(){var prop,newProps;return _ts_generator(this,function(_state){switch(_state.label){case 0:prop=this.props[at];newProps=_to_consumable_array(this.props);newProps.splice(to,0,prop);newProps.splice(at<to?at:at+1,1);this.props=newProps;return[4,this.save()];case 1:_state.sent();return[2]}})}).call(this)}}]);return Entity}();
2
- //# sourceMappingURL=entity.js.map
1
+ import assert from "assert";
2
+ import { writeFile } from "node:fs/promises";
3
+ import inflection from "inflection";
4
+ import path from "path";
5
+ import { group, unique } from "radashi";
6
+ import { z } from "zod";
7
+ import { Sonamu } from "../api/sonamu.js";
8
+ import { isBelongsToOneRelationProp, isEnumProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
9
+ import { importMembers } from "../utils/esm-utils.js";
10
+ import { formatCode } from "../utils/formatter.js";
11
+ import { exists } from "../utils/fs-utils.js";
12
+ import { runtimePath } from "../utils/path-utils.js";
13
+ import { assertDefined, nonNullable } from "../utils/utils.js";
14
+ import { EntityManager } from "./entity-manager.js";
15
+ export class Entity {
16
+ id;
17
+ parentId;
18
+ table;
19
+ title;
20
+ names;
21
+ props;
22
+ propsDict;
23
+ relations;
24
+ indexes;
25
+ subsets;
26
+ types = {};
27
+ enums = {};
28
+ enumLabels = {};
29
+ constructor({ id, parentId, table, title, props, indexes, subsets, enums }){
30
+ // id
31
+ this.id = id;
32
+ this.parentId = parentId;
33
+ this.title = title ?? this.id;
34
+ this.table = table ?? inflection.underscore(inflection.pluralize(id));
35
+ // props
36
+ if (props) {
37
+ this.props = props.map((prop)=>{
38
+ if (isEnumProp(prop)) {
39
+ if (prop.id.includes("$Model")) {
40
+ prop.id = prop.id.replace("$Model", id);
41
+ }
42
+ }
43
+ return prop;
44
+ });
45
+ this.propsDict = Object.fromEntries(props.map((prop)=>{
46
+ return [
47
+ prop.name,
48
+ prop
49
+ ];
50
+ }));
51
+ // relations
52
+ this.relations = Object.fromEntries(props.filter((prop)=>isRelationProp(prop)).map((prop)=>[
53
+ prop.name,
54
+ prop
55
+ ]));
56
+ } else {
57
+ this.props = [];
58
+ this.propsDict = {};
59
+ this.relations = {};
60
+ }
61
+ // indexes
62
+ this.indexes = indexes ?? [];
63
+ // subsets
64
+ this.subsets = subsets ?? {};
65
+ // enums
66
+ this.enumLabels = enums ?? {};
67
+ this.enums = Object.fromEntries(Object.entries(this.enumLabels).map(([key, enumLabel])=>{
68
+ return [
69
+ key,
70
+ z.enum(Object.keys(enumLabel))
71
+ ];
72
+ }));
73
+ // names
74
+ this.names = {
75
+ parentFs: inflection.dasherize(inflection.underscore(parentId ?? id)).toLowerCase(),
76
+ fs: inflection.dasherize(inflection.underscore(id)).toLowerCase(),
77
+ module: id
78
+ };
79
+ }
80
+ /*
81
+ subset을 Puri 코드로 변환
82
+ */ getPuriSubsetQuery(subsetKey) {
83
+ const subset = this.subsets[subsetKey];
84
+ const subsetQuery = this.resolveSubsetQuery("", subset);
85
+ const lines = [];
86
+ // from
87
+ lines.push(`return qbWrapper`);
88
+ lines.push(`.from("${this.table}")`);
89
+ // join
90
+ for (const join of subsetQuery.joins){
91
+ const joinMethod = join.join === "inner" ? "join" : "leftJoin";
92
+ if ("custom" in join) {
93
+ // custom join clause는 raw 사용
94
+ lines.push(`.${joinMethod}({ ${join.as}: "${join.table}" }, qbWrapper.knex.raw(\`${join.custom}\`))`);
95
+ } else {
96
+ lines.push(`.${joinMethod}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
97
+ }
98
+ }
99
+ // select
100
+ const selectObj = {};
101
+ for (const selectItem of subsetQuery.select){
102
+ // "users.id" 또는 "users.id as user__id" 형태
103
+ const match = selectItem.match(/^(.+?)(?: as (.+))?$/);
104
+ if (match) {
105
+ const [, column, alias] = match;
106
+ const key = alias ?? assertDefined(column.split(".").pop());
107
+ selectObj[key] = `"${column.trim()}"`;
108
+ }
109
+ }
110
+ lines.push(`.select({`);
111
+ Object.entries(selectObj).forEach(([key, value])=>{
112
+ lines.push(`${key}: ${value},`);
113
+ });
114
+ lines.push(`});`);
115
+ return lines.join("\n");
116
+ }
117
+ getPuriLoaderQuery(subsetKey) {
118
+ const subset = this.subsets[subsetKey];
119
+ const { loaders } = this.resolveSubsetQuery("", subset);
120
+ const lines = [
121
+ `[`
122
+ ];
123
+ const parseSelect = (select, table)=>{
124
+ const tablePrefix = `${table}.`;
125
+ if (select.startsWith(tablePrefix)) {
126
+ return `${select.replace(tablePrefix, "")}: "${select}"`;
127
+ }
128
+ if (select.includes(" as ")) {
129
+ const [column, alias] = select.split(" as ");
130
+ return `${alias}: "${column}"`;
131
+ }
132
+ return `${select}: "${select}"`;
133
+ };
134
+ // 재귀적으로 loader 생성하는 헬퍼 함수
135
+ const generateLoaderCode = (loaders)=>{
136
+ const loaderLines = [];
137
+ for (const loader of loaders){
138
+ const { toTable, toCol, through } = loader.manyJoin;
139
+ loaderLines.push("{", `as: "${loader.as}",`, `refId: "${loader.manyJoin.idField}",`, `qb: (qbWrapper: PuriWrapper<DatabaseSchemaExtend>, fromIds: number[]) => {`);
140
+ if (through === undefined) {
141
+ // HasMany
142
+ loaderLines.push(//
143
+ "return qbWrapper", `.from("${toTable}")`);
144
+ loader.oneJoins.forEach((join)=>{
145
+ const joinType = join.join === "inner" ? "join" : "leftJoin";
146
+ if ("custom" in join) {
147
+ // FIXME: 검증 필요
148
+ loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, (j) => {`, `j.on(Puri.rawString("${join.custom}"));`, "})");
149
+ } else {
150
+ loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
151
+ }
152
+ });
153
+ loaderLines.push(`.whereIn("${toTable}.${toCol}", fromIds)`, `.select({`, `${loader.select.map((select)=>parseSelect(select, toTable)).join(",")},`, `refId: "${toTable}.${toCol}",`, `});`);
154
+ } else {
155
+ // ManyToMany
156
+ loaderLines.push("return qbWrapper", `.from("${through.table}")`, `.join("${toTable}", "${through.table}.${through.toCol}", "${toTable}.${toCol}")`);
157
+ loader.oneJoins.forEach((join)=>{
158
+ const joinType = join.join === "inner" ? "join" : "leftJoin";
159
+ if ("custom" in join) {
160
+ // FIXME: 검증 필요
161
+ loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, (j) => {`, `j.on(Puri.rawString("${join.custom}"));`, "})");
162
+ } else {
163
+ loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
164
+ }
165
+ });
166
+ loaderLines.push(`.whereIn("${through.table}.${through.fromCol}", fromIds)`, `.select({`, `${loader.select.map((select)=>parseSelect(select, toTable)).join(",")},`, `refId: "${through.table}.${through.fromCol}",`, `});`);
167
+ }
168
+ loaderLines.push(`},`);
169
+ // 중첩 loaders 처리
170
+ if (loader.loaders && loader.loaders.length > 0) {
171
+ loaderLines.push("loaders: [", ...generateLoaderCode(loader.loaders), "],");
172
+ }
173
+ loaderLines.push("},");
174
+ }
175
+ return loaderLines;
176
+ };
177
+ lines.push(...generateLoaderCode(loaders));
178
+ lines.push(`]`);
179
+ return lines.join("\n");
180
+ }
181
+ /*
182
+ subset SELECT/JOIN/LOADER 결과 리턴
183
+ */ getSubsetQuery(subsetKey) {
184
+ const subset = this.subsets[subsetKey];
185
+ const result = this.resolveSubsetQuery("", subset);
186
+ return result;
187
+ }
188
+ /*
189
+ */ resolveSubsetQuery(prefix, fields, isAlreadyOuterJoined = false) {
190
+ // prefix 치환 (prefix는 ToOneRelation이 복수로 붙은 경우 모두 __로 변경됨)
191
+ prefix = prefix.replace(/\./g, "__");
192
+ // 서브셋을 1뎁스만 분리하여 그룹핑
193
+ const subsetGroup = group(fields, (field)=>{
194
+ if (field.includes(".")) {
195
+ const [rel] = field.split(".");
196
+ return rel;
197
+ } else {
198
+ return "";
199
+ }
200
+ });
201
+ const result = Object.keys(subsetGroup).reduce((r, groupKey)=>{
202
+ const fields = subsetGroup[groupKey];
203
+ assert(fields !== undefined, "fields is undefined");
204
+ // 현재 테이블 필드셋은 select, virtual에 추가하고 리턴
205
+ if (groupKey === "") {
206
+ const realFields = fields.filter((field)=>!isVirtualProp(this.propsDict[field]));
207
+ const virtualFields = fields.filter((field)=>isVirtualProp(this.propsDict[field]));
208
+ if (prefix === "") {
209
+ // 현재 테이블인 경우
210
+ r.select = r.select.concat(realFields.map((field)=>`${this.table}.${field}`));
211
+ r.virtual = r.virtual.concat(virtualFields);
212
+ } else {
213
+ // 넘어온 테이블인 경우
214
+ r.select = r.select.concat(realFields.map((field)=>`${prefix}.${field} as ${prefix}__${field}`));
215
+ }
216
+ return r;
217
+ }
218
+ const relation = this.relations[groupKey];
219
+ if (relation === undefined) {
220
+ throw new Error(`존재하지 않는 relation 참조 ${groupKey}`);
221
+ }
222
+ const relEntity = EntityManager.get(relation.with);
223
+ if (isOneToOneRelationProp(relation) || isBelongsToOneRelationProp(relation)) {
224
+ // -One Relation: JOIN 으로 처리
225
+ const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
226
+ // -One Relation에서 id 필드만 참조하는 경우 릴레이션 넘기지 않고 리턴
227
+ if (relFields.length === 1 && relFields[0] === "id") {
228
+ if (prefix === "") {
229
+ r.select = r.select.concat(`${this.table}.${groupKey}_id`);
230
+ } else {
231
+ r.select = r.select.concat(`${prefix}.${groupKey}_id as ${prefix}__${groupKey}_id`);
232
+ }
233
+ return r;
234
+ }
235
+ // innerOrOuter
236
+ const innerOrOuter = (()=>{
237
+ if (isAlreadyOuterJoined) {
238
+ return "outer";
239
+ }
240
+ if (isOneToOneRelationProp(relation)) {
241
+ if (relation.hasJoinColumn === true && (relation.nullable ?? false) === false) {
242
+ return "inner";
243
+ } else {
244
+ return "outer";
245
+ }
246
+ } else {
247
+ if (relation.nullable) {
248
+ return "outer";
249
+ } else {
250
+ return "inner";
251
+ }
252
+ }
253
+ })();
254
+ const relSubsetQuery = relEntity.resolveSubsetQuery(`${prefix !== "" ? `${prefix}.` : ""}${groupKey}`, relFields, innerOrOuter === "outer");
255
+ r.select = r.select.concat(relSubsetQuery.select);
256
+ r.virtual = r.virtual.concat(relSubsetQuery.virtual);
257
+ const joinAs = prefix === "" ? groupKey : `${prefix}__${groupKey}`;
258
+ const fromTable = prefix === "" ? this.table : prefix;
259
+ let joinClause;
260
+ if (relation.customJoinClause) {
261
+ joinClause = {
262
+ custom: relation.customJoinClause
263
+ };
264
+ } else {
265
+ let from, to;
266
+ if (isOneToOneRelationProp(relation)) {
267
+ if (relation.hasJoinColumn) {
268
+ from = `${fromTable}.${relation.name}_id`;
269
+ to = `${joinAs}.id`;
270
+ } else {
271
+ from = `${fromTable}.id`;
272
+ to = `${joinAs}.${inflection.underscore(this.names.fs.replace(/-/g, "_"))}_id`;
273
+ }
274
+ } else {
275
+ from = `${fromTable}.${relation.name}_id`;
276
+ to = `${joinAs}.id`;
277
+ }
278
+ joinClause = {
279
+ from,
280
+ to
281
+ };
282
+ }
283
+ r.joins.push({
284
+ as: joinAs,
285
+ join: innerOrOuter,
286
+ table: relEntity.table,
287
+ ...joinClause
288
+ });
289
+ // BelongsToOne 밑에 HasMany가 붙은 경우
290
+ if (relSubsetQuery.loaders.length > 0) {
291
+ const convertedLoaders = relSubsetQuery.loaders.map((loader)=>{
292
+ const newAs = [
293
+ groupKey,
294
+ loader.as
295
+ ].join("__");
296
+ return {
297
+ as: newAs,
298
+ table: loader.table,
299
+ manyJoin: loader.manyJoin,
300
+ oneJoins: loader.oneJoins,
301
+ select: loader.select,
302
+ loaders: loader.loaders
303
+ };
304
+ });
305
+ r.loaders = [
306
+ ...r.loaders,
307
+ ...convertedLoaders
308
+ ];
309
+ }
310
+ r.joins = r.joins.concat(relSubsetQuery.joins);
311
+ } else if (isHasManyRelationProp(relation) || isManyToManyRelationProp(relation)) {
312
+ // -Many Relation: Loader 로 처리
313
+ const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
314
+ const relSubsetQuery = relEntity.resolveSubsetQuery("", relFields);
315
+ let manyJoin;
316
+ if (isHasManyRelationProp(relation)) {
317
+ const fromCol = relation?.fromColumn ?? "id";
318
+ manyJoin = {
319
+ fromTable: this.table,
320
+ fromCol,
321
+ idField: prefix === "" ? `${fromCol}` : `${prefix}__${fromCol}`,
322
+ toTable: relEntity.table,
323
+ toCol: relation.joinColumn
324
+ };
325
+ } else if (isManyToManyRelationProp(relation)) {
326
+ manyJoin = {
327
+ fromTable: this.table,
328
+ fromCol: "id",
329
+ idField: prefix === "" ? `id` : `${prefix}__id`,
330
+ through: {
331
+ table: relation.joinTable,
332
+ fromCol: `${inflection.singularize(this.table)}_id`,
333
+ toCol: `${inflection.singularize(relEntity.table)}_id`
334
+ },
335
+ toTable: relEntity.table,
336
+ toCol: "id"
337
+ };
338
+ } else {
339
+ throw new Error();
340
+ }
341
+ r.loaders.push({
342
+ as: groupKey,
343
+ table: relEntity.table,
344
+ manyJoin,
345
+ oneJoins: relSubsetQuery.joins,
346
+ select: relSubsetQuery.select,
347
+ loaders: relSubsetQuery.loaders
348
+ });
349
+ }
350
+ return r;
351
+ }, {
352
+ select: [],
353
+ virtual: [],
354
+ joins: [],
355
+ loaders: []
356
+ });
357
+ return result;
358
+ }
359
+ /*
360
+ FieldExpr[] 을 EntityPropNode[] 로 변환
361
+ */ fieldExprsToPropNodes(fieldExprs, entity = this) {
362
+ const groups = fieldExprs.reduce((result, fieldExpr)=>{
363
+ let key, value, elseExpr;
364
+ if (fieldExpr.includes(".")) {
365
+ [key, ...elseExpr] = fieldExpr.split(".");
366
+ value = elseExpr.join(".");
367
+ } else {
368
+ key = "";
369
+ value = fieldExpr;
370
+ }
371
+ result[key] = (result[key] ?? []).concat(value);
372
+ return result;
373
+ }, {});
374
+ return Object.keys(groups).flatMap((key)=>{
375
+ const group = groups[key];
376
+ // 일반 prop 처리
377
+ if (key === "") {
378
+ return group.map((propName)=>{
379
+ // uuid 개별 처리
380
+ if (propName === "uuid") {
381
+ return {
382
+ nodeType: "plain",
383
+ prop: {
384
+ type: "string",
385
+ name: "uuid",
386
+ length: 128
387
+ },
388
+ children: []
389
+ };
390
+ }
391
+ const prop = entity.props.find((p)=>p.name === propName);
392
+ if (prop === undefined) {
393
+ console.log({
394
+ propName,
395
+ groups
396
+ });
397
+ throw new Error(`${entity.id} -- 잘못된 FieldExpr ${propName}`);
398
+ }
399
+ return {
400
+ nodeType: "plain",
401
+ prop,
402
+ children: []
403
+ };
404
+ });
405
+ }
406
+ // relation prop 처리
407
+ const prop = entity.propsDict[key];
408
+ if (!isRelationProp(prop)) {
409
+ throw new Error(`잘못된 FieldExpr ${key}.${group[0]}`);
410
+ }
411
+ const relEntity = EntityManager.get(prop.with);
412
+ // relation -One 에 id 필드 하나인 경우
413
+ if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)) {
414
+ if (group.length === 1 && (group[0] === "id" || group[0] === "id?")) {
415
+ // id 하나만 있는지 체크해서, 하나만 있으면 상위 prop으로 id를 리턴
416
+ const idProp = relEntity.propsDict.id;
417
+ return {
418
+ nodeType: "plain",
419
+ prop: {
420
+ ...idProp,
421
+ name: `${key}_id`,
422
+ nullable: prop.nullable
423
+ },
424
+ children: []
425
+ };
426
+ }
427
+ }
428
+ // -One 그외의 경우 object로 리턴
429
+ // -Many의 경우 array로 리턴
430
+ // Recursive 로 뎁스 처리
431
+ const children = this.fieldExprsToPropNodes(group, relEntity);
432
+ const nodeType = isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) ? "object" : "array";
433
+ return {
434
+ prop,
435
+ children,
436
+ nodeType
437
+ };
438
+ });
439
+ }
440
+ getFieldExprs(prefix = "", maxDepth = 3, froms = []) {
441
+ return this.props.flatMap((prop)=>{
442
+ const propName = [
443
+ prefix,
444
+ prop.name
445
+ ].filter((v)=>v !== "").join(".");
446
+ if (propName === prefix) {
447
+ return null;
448
+ }
449
+ if (isRelationProp(prop)) {
450
+ if (maxDepth < 0) {
451
+ return null;
452
+ }
453
+ if (froms.includes(prop.with)) {
454
+ // 역방향 relation인 경우 제외
455
+ return null;
456
+ }
457
+ // 정방향 relation인 경우 recursive 콜
458
+ const relMd = EntityManager.get(prop.with);
459
+ return relMd.getFieldExprs(propName, maxDepth - 1, [
460
+ ...froms,
461
+ this.id
462
+ ]);
463
+ }
464
+ return propName;
465
+ }).filter((f)=>f !== null);
466
+ }
467
+ getTableColumns() {
468
+ return this.props.map((prop)=>{
469
+ if (prop.type === "relation") {
470
+ if (prop.relationType === "BelongsToOne" || prop.relationType === "OneToOne" && prop.hasJoinColumn === true) {
471
+ return {
472
+ name: `${prop.name}_id`,
473
+ type: "int_unsigned"
474
+ };
475
+ } else {
476
+ return null;
477
+ }
478
+ }
479
+ return {
480
+ name: prop.name,
481
+ type: prop.type
482
+ };
483
+ }).filter(nonNullable);
484
+ }
485
+ async registerModulePaths() {
486
+ const basePath = `${this.names.parentFs}`;
487
+ // base-scheme
488
+ EntityManager.setModulePath(`${this.id}BaseSchema`, `sonamu.generated`);
489
+ // subset
490
+ if (Object.keys(this.subsets).length > 0) {
491
+ EntityManager.setModulePath(`${this.id}SubsetKey`, `sonamu.generated`);
492
+ EntityManager.setModulePath(`${this.id}SubsetMapping`, `sonamu.generated`);
493
+ for (const subsetKey of Object.keys(this.subsets)){
494
+ EntityManager.setModulePath(`${this.id}Subset${subsetKey.toUpperCase()}`, `sonamu.generated`);
495
+ }
496
+ }
497
+ // enums
498
+ for (const enumId of Object.keys(this.enumLabels)){
499
+ EntityManager.setModulePath(enumId, `sonamu.generated`);
500
+ }
501
+ // types
502
+ const typesModulePath = `${basePath}/${this.names.parentFs}.types`;
503
+ const typesFilePath = path.join(Sonamu.apiRootPath, runtimePath(`dist/application/${typesModulePath}.js`));
504
+ if (await exists(typesFilePath)) {
505
+ const importedMembers = await importMembers(typesFilePath);
506
+ this.types = Object.fromEntries(importedMembers.map(({ name, value })=>{
507
+ EntityManager.setModulePath(name, typesModulePath);
508
+ return [
509
+ name,
510
+ value
511
+ ];
512
+ }));
513
+ }
514
+ }
515
+ registerTableSpecs() {
516
+ const uniqueIndexes = this.indexes.filter((idx)=>idx.type === "unique");
517
+ EntityManager.setTableSpec({
518
+ name: this.table,
519
+ uniqueIndexes
520
+ });
521
+ }
522
+ toJson() {
523
+ return {
524
+ id: this.id,
525
+ parentId: this.parentId,
526
+ table: this.table,
527
+ title: this.title,
528
+ props: this.props,
529
+ indexes: this.indexes,
530
+ subsets: this.subsets,
531
+ enums: this.enumLabels
532
+ };
533
+ }
534
+ async save() {
535
+ // sort: subsets
536
+ const subsetRows = this.getSubsetRows();
537
+ this.subsets = Object.fromEntries(Object.entries(this.subsets).map(([subsetKey])=>{
538
+ return [
539
+ subsetKey,
540
+ this.subsetRowsToSubsetFields(subsetRows, subsetKey)
541
+ ];
542
+ }));
543
+ // save
544
+ const jsonPath = path.join(Sonamu.apiRootPath, `src/application/${this.names.parentFs}/${this.names.fs}.entity.json`);
545
+ const json = this.toJson();
546
+ await writeFile(jsonPath, formatCode(JSON.stringify(json), "json", jsonPath));
547
+ // reload
548
+ await EntityManager.register(json);
549
+ }
550
+ getSubsetRows(_subsets, prefixes = []) {
551
+ if (prefixes.length > 10) {
552
+ return [];
553
+ }
554
+ const subsets = _subsets ?? this.subsets;
555
+ const subsetKeys = Object.keys(subsets);
556
+ const allFields = unique(subsetKeys.flatMap((key)=>subsets[key]));
557
+ return this.props.map((prop)=>{
558
+ if (prop.type === "relation" && allFields.find((f)=>f.startsWith(`${[
559
+ ...prefixes,
560
+ prop.name
561
+ ].join(".")}.`))) {
562
+ const relEntity = EntityManager.get(prop.with);
563
+ const children = relEntity.getSubsetRows(subsets, [
564
+ ...prefixes,
565
+ `${prop.name}`
566
+ ]);
567
+ return {
568
+ field: prop.name,
569
+ children,
570
+ relationEntity: prop.with,
571
+ prefixes,
572
+ isOpen: children.length > 0,
573
+ has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
574
+ return [
575
+ subsetKey,
576
+ children.every((child)=>child.has[subsetKey] === true)
577
+ ];
578
+ }))
579
+ };
580
+ }
581
+ return {
582
+ field: prop.name,
583
+ children: [],
584
+ relationEntity: prop.type === "relation" ? prop.with : undefined,
585
+ prefixes,
586
+ has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
587
+ const subsetFields = subsets[subsetKey];
588
+ const has = subsetFields.some((f)=>{
589
+ const field = [
590
+ ...prefixes,
591
+ prop.name
592
+ ].join(".");
593
+ return f === field || f.startsWith(`${field}.`);
594
+ });
595
+ return [
596
+ subsetKey,
597
+ has
598
+ ];
599
+ }))
600
+ };
601
+ });
602
+ }
603
+ subsetRowsToSubsetFields(subsetRows, subsetKey) {
604
+ return subsetRows.map((subsetRow)=>{
605
+ if (subsetRow.children.length > 0) {
606
+ return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey);
607
+ } else if (subsetRow.has[subsetKey]) {
608
+ return subsetRow.prefixes.concat(subsetRow.field).join(".");
609
+ } else {
610
+ return null;
611
+ }
612
+ }).filter(nonNullable).flat();
613
+ }
614
+ async createProp(prop, at) {
615
+ if (!at) {
616
+ this.props.push(prop);
617
+ } else {
618
+ this.props.splice(at, 0, prop);
619
+ }
620
+ await this.save();
621
+ }
622
+ analyzeSubsetField(subsetField) {
623
+ const arr = subsetField.split(".");
624
+ let entityId = this.id;
625
+ const result = [];
626
+ for(let i = 0; i < arr.length; i++){
627
+ const propName = arr[i];
628
+ result.push({
629
+ entityId,
630
+ propName
631
+ });
632
+ const prop = EntityManager.get(entityId).props.find((p)=>p.name === propName);
633
+ if (!prop) {
634
+ throw new Error(`${entityId}의 잘못된 서브셋키 ${subsetField}`);
635
+ }
636
+ if (isRelationProp(prop)) {
637
+ entityId = prop.with;
638
+ }
639
+ }
640
+ return result;
641
+ }
642
+ async modifyProp(newProp, at) {
643
+ // 이전 프롭 이름 저장
644
+ const oldName = this.props[at].name;
645
+ // 저장할 엔티티
646
+ const entities = [
647
+ this
648
+ ];
649
+ // 이름이 바뀐 경우
650
+ if (oldName !== newProp.name) {
651
+ // 전체 엔티티에서 현재 수정된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 수정
652
+ const allEntityIds = EntityManager.getAllIds();
653
+ for (const relEntityId of allEntityIds){
654
+ const relEntity = EntityManager.get(relEntityId);
655
+ const relEntitySubsetKeys = Object.keys(relEntity.subsets);
656
+ for (const subsetKey of relEntitySubsetKeys){
657
+ const subset = relEntity.subsets[subsetKey];
658
+ // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 수정 처리
659
+ const modifiedSubsetFields = subset.map((subsetField)=>{
660
+ const analyzed = relEntity.analyzeSubsetField(subsetField);
661
+ const modified = analyzed.map((a)=>a.propName === oldName && a.entityId === this.id ? {
662
+ ...a,
663
+ propName: newProp.name
664
+ } : a);
665
+ // 분석한 필드를 다시 서브셋 필드로 복구
666
+ return modified.map((a)=>a.propName).join(".");
667
+ });
668
+ if (subset.join(",") !== modifiedSubsetFields.join(",")) {
669
+ relEntity.subsets[subsetKey] = modifiedSubsetFields;
670
+ entities.push(relEntity);
671
+ }
672
+ }
673
+ }
674
+ }
675
+ // 프롭 수정
676
+ this.props[at] = newProp;
677
+ await Promise.all(entities.map(async (entity)=>entity.save()));
678
+ }
679
+ async delProp(at) {
680
+ // 이전 프롭 이름 저장
681
+ const oldName = this.props[at].name;
682
+ // 저장할 엔티티
683
+ const entities = [
684
+ this
685
+ ];
686
+ // 전체 엔티티에서 현재 삭제된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 제외
687
+ const allEntityIds = EntityManager.getAllIds();
688
+ for (const relEntityId of allEntityIds){
689
+ const relEntity = EntityManager.get(relEntityId);
690
+ const relEntitySubsetKeys = Object.keys(relEntity.subsets);
691
+ for (const subsetKey of relEntitySubsetKeys){
692
+ const subset = relEntity.subsets[subsetKey];
693
+ // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 이후의 필드를 제외
694
+ const modifiedSubsetFields = subset.map((subsetField)=>{
695
+ const analyzed = relEntity.analyzeSubsetField(subsetField);
696
+ if (analyzed.find((a)=>a.propName === oldName && a.entityId === this.id)) {
697
+ return null;
698
+ } else {
699
+ return subsetField;
700
+ }
701
+ }).filter(nonNullable);
702
+ if (subset.join(",") !== modifiedSubsetFields.join(",")) {
703
+ relEntity.subsets[subsetKey] = modifiedSubsetFields;
704
+ entities.push(relEntity);
705
+ }
706
+ }
707
+ }
708
+ // 현재 엔티티의 인덱스에서 제외
709
+ for (const index of EntityManager.get(this.id).indexes){
710
+ index.columns = index.columns.filter((col)=>col !== oldName);
711
+ }
712
+ // 프롭 삭제
713
+ this.props.splice(at, 1);
714
+ await Promise.all(entities.map(async (entity)=>entity.save()));
715
+ }
716
+ getEntityIdFromSubsetField(subsetField) {
717
+ if (subsetField.includes(".") === false) {
718
+ return this.id;
719
+ }
720
+ // 서브셋 필드의 마지막은 프롭이므로 제외
721
+ const arr = subsetField.split(".").slice(0, -1);
722
+ // 서브셋 필드를 내려가면서 마지막으로 relation된 엔티티를 찾음
723
+ const lastEntityId = arr.reduce((entityId, field)=>{
724
+ const relProp = EntityManager.get(entityId).props.find((p)=>p.name === field);
725
+ if (!relProp || relProp.type !== "relation") {
726
+ console.debug({
727
+ arr,
728
+ thisId: this.id,
729
+ entityId,
730
+ field
731
+ });
732
+ throw new Error(`잘못된 서브셋키 ${subsetField}`);
733
+ }
734
+ return relProp.with;
735
+ }, this.id);
736
+ return lastEntityId;
737
+ }
738
+ async moveProp(at, to) {
739
+ const prop = this.props[at];
740
+ const newProps = [
741
+ ...this.props
742
+ ];
743
+ newProps.splice(to, 0, prop);
744
+ newProps.splice(at < to ? at : at + 1, 1);
745
+ this.props = newProps;
746
+ await this.save();
747
+ }
748
+ }
749
+
750
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9lbnRpdHkvZW50aXR5LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBhc3NlcnQgZnJvbSBcImFzc2VydFwiO1xuaW1wb3J0IHsgd3JpdGVGaWxlIH0gZnJvbSBcImZzL3Byb21pc2VzXCI7XG5pbXBvcnQgaW5mbGVjdGlvbiBmcm9tIFwiaW5mbGVjdGlvblwiO1xuaW1wb3J0IHBhdGggZnJvbSBcInBhdGhcIjtcbmltcG9ydCB7IGdyb3VwLCB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgeiB9IGZyb20gXCJ6b2RcIjtcbmltcG9ydCB7IFNvbmFtdSB9IGZyb20gXCIuLi9hcGkvc29uYW11XCI7XG5pbXBvcnQge1xuICB0eXBlIEVudGl0eUluZGV4LFxuICB0eXBlIEVudGl0eUpzb24sXG4gIHR5cGUgRW50aXR5UHJvcCxcbiAgdHlwZSBFbnRpdHlQcm9wTm9kZSxcbiAgdHlwZSBFbnRpdHlTdWJzZXRSb3csXG4gIGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wLFxuICBpc0VudW1Qcm9wLFxuICBpc0hhc01hbnlSZWxhdGlvblByb3AsXG4gIGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcCxcbiAgaXNPbmVUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNSZWxhdGlvblByb3AsXG4gIGlzVmlydHVhbFByb3AsXG4gIHR5cGUgUmVsYXRpb25Qcm9wLFxuICB0eXBlIFN0cmluZ1Byb3AsXG4gIHR5cGUgU3Vic2V0UXVlcnksXG59IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgaW1wb3J0TWVtYmVycyB9IGZyb20gXCIuLi91dGlscy9lc20tdXRpbHNcIjtcbmltcG9ydCB7IGZvcm1hdENvZGUgfSBmcm9tIFwiLi4vdXRpbHMvZm9ybWF0dGVyXCI7XG5pbXBvcnQgeyBleGlzdHMgfSBmcm9tIFwiLi4vdXRpbHMvZnMtdXRpbHNcIjtcbmltcG9ydCB7IHJ1bnRpbWVQYXRoIH0gZnJvbSBcIi4uL3V0aWxzL3BhdGgtdXRpbHNcIjtcbmltcG9ydCB7IGFzc2VydERlZmluZWQsIG5vbk51bGxhYmxlIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgeyBFbnRpdHlNYW5hZ2VyIH0gZnJvbSBcIi4vZW50aXR5LW1hbmFnZXJcIjtcblxuZXhwb3J0IGNsYXNzIEVudGl0eSB7XG4gIGlkOiBzdHJpbmc7XG4gIHBhcmVudElkPzogc3RyaW5nO1xuICB0YWJsZTogc3RyaW5nO1xuICB0aXRsZTogc3RyaW5nO1xuICBuYW1lczoge1xuICAgIHBhcmVudEZzOiBzdHJpbmc7XG4gICAgZnM6IHN0cmluZztcbiAgICBtb2R1bGU6IHN0cmluZztcbiAgfTtcbiAgcHJvcHM6IEVudGl0eVByb3BbXTtcbiAgcHJvcHNEaWN0OiB7XG4gICAgW2tleTogc3RyaW5nXTogRW50aXR5UHJvcDtcbiAgfTtcbiAgcmVsYXRpb25zOiB7XG4gICAgW2tleTogc3RyaW5nXTogUmVsYXRpb25Qcm9wO1xuICB9O1xuICBpbmRleGVzOiBFbnRpdHlJbmRleFtdO1xuICBzdWJzZXRzOiB7XG4gICAgW2tleTogc3RyaW5nXTogc3RyaW5nW107XG4gIH07XG4gIHR5cGVzOiB7XG4gICAgW25hbWU6IHN0cmluZ106IHouWm9kVHlwZUFueTtcbiAgfSA9IHt9O1xuICBlbnVtczoge1xuICAgIFtlbnVtSWQ6IHN0cmluZ106IHouWm9kRW51bTxSZWFkb25seTxSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+Pj47XG4gIH0gPSB7fTtcbiAgZW51bUxhYmVsczoge1xuICAgIFtlbnVtSWQ6IHN0cmluZ106IHtcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZztcbiAgICB9O1xuICB9ID0ge307XG5cbiAgY29uc3RydWN0b3IoeyBpZCwgcGFyZW50SWQsIHRhYmxlLCB0aXRsZSwgcHJvcHMsIGluZGV4ZXMsIHN1YnNldHMsIGVudW1zIH06IEVudGl0eUpzb24pIHtcbiAgICAvLyBpZFxuICAgIHRoaXMuaWQgPSBpZDtcbiAgICB0aGlzLnBhcmVudElkID0gcGFyZW50SWQ7XG4gICAgdGhpcy50aXRsZSA9IHRpdGxlID8/IHRoaXMuaWQ7XG4gICAgdGhpcy50YWJsZSA9IHRhYmxlID8/IGluZmxlY3Rpb24udW5kZXJzY29yZShpbmZsZWN0aW9uLnBsdXJhbGl6ZShpZCkpO1xuXG4gICAgLy8gcHJvcHNcbiAgICBpZiAocHJvcHMpIHtcbiAgICAgIHRoaXMucHJvcHMgPSBwcm9wcy5tYXAoKHByb3ApID0+IHtcbiAgICAgICAgaWYgKGlzRW51bVByb3AocHJvcCkpIHtcbiAgICAgICAgICBpZiAocHJvcC5pZC5pbmNsdWRlcyhcIiRNb2RlbFwiKSkge1xuICAgICAgICAgICAgcHJvcC5pZCA9IHByb3AuaWQucmVwbGFjZShcIiRNb2RlbFwiLCBpZCk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBwcm9wO1xuICAgICAgfSk7XG4gICAgICB0aGlzLnByb3BzRGljdCA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgICAgcHJvcHMubWFwKChwcm9wKSA9PiB7XG4gICAgICAgICAgcmV0dXJuIFtwcm9wLm5hbWUsIHByb3BdO1xuICAgICAgICB9KSxcbiAgICAgICk7XG5cbiAgICAgIC8vIHJlbGF0aW9uc1xuICAgICAgdGhpcy5yZWxhdGlvbnMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICAgIHByb3BzLmZpbHRlcigocHJvcCkgPT4gaXNSZWxhdGlvblByb3AocHJvcCkpLm1hcCgocHJvcCkgPT4gW3Byb3AubmFtZSwgcHJvcF0pLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5wcm9wcyA9IFtdO1xuICAgICAgdGhpcy5wcm9wc0RpY3QgPSB7fTtcbiAgICAgIHRoaXMucmVsYXRpb25zID0ge307XG4gICAgfVxuXG4gICAgLy8gaW5kZXhlc1xuICAgIHRoaXMuaW5kZXhlcyA9IGluZGV4ZXMgPz8gW107XG5cbiAgICAvLyBzdWJzZXRzXG4gICAgdGhpcy5zdWJzZXRzID0gc3Vic2V0cyA/PyB7fTtcblxuICAgIC8vIGVudW1zXG4gICAgdGhpcy5lbnVtTGFiZWxzID0gZW51bXMgPz8ge307XG4gICAgdGhpcy5lbnVtcyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHRoaXMuZW51bUxhYmVscykubWFwKChba2V5LCBlbnVtTGFiZWxdKSA9PiB7XG4gICAgICAgIHJldHVybiBba2V5LCB6LmVudW0oT2JqZWN0LmtleXMoZW51bUxhYmVsKSBhcyB1bmtub3duIGFzIHJlYWRvbmx5IFtzdHJpbmcsIC4uLnN0cmluZ1tdXSldO1xuICAgICAgfSksXG4gICAgKTtcblxuICAgIC8vIG5hbWVzXG4gICAgdGhpcy5uYW1lcyA9IHtcbiAgICAgIHBhcmVudEZzOiBpbmZsZWN0aW9uLmRhc2hlcml6ZShpbmZsZWN0aW9uLnVuZGVyc2NvcmUocGFyZW50SWQgPz8gaWQpKS50b0xvd2VyQ2FzZSgpLFxuICAgICAgZnM6IGluZmxlY3Rpb24uZGFzaGVyaXplKGluZmxlY3Rpb24udW5kZXJzY29yZShpZCkpLnRvTG93ZXJDYXNlKCksXG4gICAgICBtb2R1bGU6IGlkLFxuICAgIH07XG4gIH1cblxuICAvKlxuICAgIHN1YnNldOydhCBQdXJpIOy9lOuTnOuhnCDrs4DtmZhcbiAgKi9cbiAgZ2V0UHVyaVN1YnNldFF1ZXJ5KHN1YnNldEtleTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCBzdWJzZXQgPSB0aGlzLnN1YnNldHNbc3Vic2V0S2V5XTtcbiAgICBjb25zdCBzdWJzZXRRdWVyeSA9IHRoaXMucmVzb2x2ZVN1YnNldFF1ZXJ5KFwiXCIsIHN1YnNldCk7XG5cbiAgICBjb25zdCBsaW5lczogc3RyaW5nW10gPSBbXTtcblxuICAgIC8vIGZyb21cbiAgICBsaW5lcy5wdXNoKGByZXR1cm4gcWJXcmFwcGVyYCk7XG4gICAgbGluZXMucHVzaChgLmZyb20oXCIke3RoaXMudGFibGV9XCIpYCk7XG5cbiAgICAvLyBqb2luXG4gICAgZm9yIChjb25zdCBqb2luIG9mIHN1YnNldFF1ZXJ5LmpvaW5zKSB7XG4gICAgICBjb25zdCBqb2luTWV0aG9kID0gam9pbi5qb2luID09PSBcImlubmVyXCIgPyBcImpvaW5cIiA6IFwibGVmdEpvaW5cIjtcblxuICAgICAgaWYgKFwiY3VzdG9tXCIgaW4gam9pbikge1xuICAgICAgICAvLyBjdXN0b20gam9pbiBjbGF1c2XripQgcmF3IOyCrOyaqVxuICAgICAgICBsaW5lcy5wdXNoKFxuICAgICAgICAgIGAuJHtqb2luTWV0aG9kfSh7ICR7am9pbi5hc306IFwiJHtqb2luLnRhYmxlfVwiIH0sIHFiV3JhcHBlci5rbmV4LnJhdyhcXGAke2pvaW4uY3VzdG9tfVxcYCkpYCxcbiAgICAgICAgKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxpbmVzLnB1c2goYC4ke2pvaW5NZXRob2R9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgXCIke2pvaW4uZnJvbX1cIiwgXCIke2pvaW4udG99XCIpYCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gc2VsZWN0XG4gICAgY29uc3Qgc2VsZWN0T2JqOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge307XG4gICAgZm9yIChjb25zdCBzZWxlY3RJdGVtIG9mIHN1YnNldFF1ZXJ5LnNlbGVjdCkge1xuICAgICAgLy8gXCJ1c2Vycy5pZFwiIOuYkOuKlCBcInVzZXJzLmlkIGFzIHVzZXJfX2lkXCIg7ZiV7YOcXG4gICAgICBjb25zdCBtYXRjaCA9IHNlbGVjdEl0ZW0ubWF0Y2goL14oLis/KSg/OiBhcyAoLispKT8kLyk7XG4gICAgICBpZiAobWF0Y2gpIHtcbiAgICAgICAgY29uc3QgWywgY29sdW1uLCBhbGlhc10gPSBtYXRjaDtcbiAgICAgICAgY29uc3Qga2V5ID0gYWxpYXMgPz8gYXNzZXJ0RGVmaW5lZChjb2x1bW4uc3BsaXQoXCIuXCIpLnBvcCgpKTtcbiAgICAgICAgc2VsZWN0T2JqW2tleV0gPSBgXCIke2NvbHVtbi50cmltKCl9XCJgO1xuICAgICAgfVxuICAgIH1cblxuICAgIGxpbmVzLnB1c2goYC5zZWxlY3Qoe2ApO1xuICAgIE9iamVjdC5lbnRyaWVzKHNlbGVjdE9iaikuZm9yRWFjaCgoW2tleSwgdmFsdWVdKSA9PiB7XG4gICAgICBsaW5lcy5wdXNoKGAke2tleX06ICR7dmFsdWV9LGApO1xuICAgIH0pO1xuICAgIGxpbmVzLnB1c2goYH0pO2ApO1xuXG4gICAgcmV0dXJuIGxpbmVzLmpvaW4oXCJcXG5cIik7XG4gIH1cblxuICBnZXRQdXJpTG9hZGVyUXVlcnkoc3Vic2V0S2V5OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHN1YnNldCA9IHRoaXMuc3Vic2V0c1tzdWJzZXRLZXldO1xuICAgIGNvbnN0IHsgbG9hZGVycyB9ID0gdGhpcy5yZXNvbHZlU3Vic2V0UXVlcnkoXCJcIiwgc3Vic2V0KTtcblxuICAgIGNvbnN0IGxpbmVzOiBzdHJpbmdbXSA9IFtgW2BdO1xuXG4gICAgY29uc3QgcGFyc2VTZWxlY3QgPSAoc2VsZWN0OiBzdHJpbmcsIHRhYmxlOiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IHRhYmxlUHJlZml4ID0gYCR7dGFibGV9LmA7XG4gICAgICBpZiAoc2VsZWN0LnN0YXJ0c1dpdGgodGFibGVQcmVmaXgpKSB7XG4gICAgICAgIHJldHVybiBgJHtzZWxlY3QucmVwbGFjZSh0YWJsZVByZWZpeCwgXCJcIil9OiBcIiR7c2VsZWN0fVwiYDtcbiAgICAgIH1cblxuICAgICAgaWYgKHNlbGVjdC5pbmNsdWRlcyhcIiBhcyBcIikpIHtcbiAgICAgICAgY29uc3QgW2NvbHVtbiwgYWxpYXNdID0gc2VsZWN0LnNwbGl0KFwiIGFzIFwiKTtcbiAgICAgICAgcmV0dXJuIGAke2FsaWFzfTogXCIke2NvbHVtbn1cImA7XG4gICAgICB9XG4gICAgICByZXR1cm4gYCR7c2VsZWN0fTogXCIke3NlbGVjdH1cImA7XG4gICAgfTtcblxuICAgIC8vIOyerOq3gOyggeycvOuhnCBsb2FkZXIg7IOd7ISx7ZWY64qUIO2XrO2NvCDtlajsiJhcbiAgICBjb25zdCBnZW5lcmF0ZUxvYWRlckNvZGUgPSAobG9hZGVyczogU3Vic2V0UXVlcnlbXCJsb2FkZXJzXCJdKTogc3RyaW5nW10gPT4ge1xuICAgICAgY29uc3QgbG9hZGVyTGluZXM6IHN0cmluZ1tdID0gW107XG5cbiAgICAgIGZvciAoY29uc3QgbG9hZGVyIG9mIGxvYWRlcnMpIHtcbiAgICAgICAgY29uc3QgeyB0b1RhYmxlLCB0b0NvbCwgdGhyb3VnaCB9ID0gbG9hZGVyLm1hbnlKb2luO1xuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgIFwie1wiLFxuICAgICAgICAgIGBhczogXCIke2xvYWRlci5hc31cIixgLFxuICAgICAgICAgIGByZWZJZDogXCIke2xvYWRlci5tYW55Sm9pbi5pZEZpZWxkfVwiLGAsXG4gICAgICAgICAgYHFiOiAocWJXcmFwcGVyOiBQdXJpV3JhcHBlcjxEYXRhYmFzZVNjaGVtYUV4dGVuZD4sIGZyb21JZHM6IG51bWJlcltdKSA9PiB7YCxcbiAgICAgICAgKTtcblxuICAgICAgICBpZiAodGhyb3VnaCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgLy8gSGFzTWFueVxuICAgICAgICAgIGxvYWRlckxpbmVzLnB1c2goXG4gICAgICAgICAgICAvL1xuICAgICAgICAgICAgXCJyZXR1cm4gcWJXcmFwcGVyXCIsXG4gICAgICAgICAgICBgLmZyb20oXCIke3RvVGFibGV9XCIpYCxcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgbG9hZGVyLm9uZUpvaW5zLmZvckVhY2goKGpvaW46IFN1YnNldFF1ZXJ5W1wiam9pbnNcIl1bbnVtYmVyXSkgPT4ge1xuICAgICAgICAgICAgY29uc3Qgam9pblR5cGUgPSBqb2luLmpvaW4gPT09IFwiaW5uZXJcIiA/IFwiam9pblwiIDogXCJsZWZ0Sm9pblwiO1xuICAgICAgICAgICAgaWYgKFwiY3VzdG9tXCIgaW4gam9pbikge1xuICAgICAgICAgICAgICAvLyBGSVhNRTog6rKA7KadIO2VhOyalFxuICAgICAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgICAgIGAuJHtqb2luVHlwZX0oeyAke2pvaW4uYXN9OiBcIiR7am9pbi50YWJsZX1cIiB9LCAoaikgPT4ge2AsXG4gICAgICAgICAgICAgICAgYGoub24oUHVyaS5yYXdTdHJpbmcoXCIke2pvaW4uY3VzdG9tfVwiKSk7YCxcbiAgICAgICAgICAgICAgICBcIn0pXCIsXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgICAgIGAuJHtqb2luVHlwZX0oeyAke2pvaW4uYXN9OiBcIiR7am9pbi50YWJsZX1cIiB9LCBcIiR7am9pbi5mcm9tfVwiLCBcIiR7am9pbi50b31cIilgLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgIGAud2hlcmVJbihcIiR7dG9UYWJsZX0uJHt0b0NvbH1cIiwgZnJvbUlkcylgLFxuICAgICAgICAgICAgYC5zZWxlY3Qoe2AsXG4gICAgICAgICAgICBgJHtsb2FkZXIuc2VsZWN0Lm1hcCgoc2VsZWN0OiBzdHJpbmcpID0+IHBhcnNlU2VsZWN0KHNlbGVjdCwgdG9UYWJsZSkpLmpvaW4oXCIsXCIpfSxgLFxuICAgICAgICAgICAgYHJlZklkOiBcIiR7dG9UYWJsZX0uJHt0b0NvbH1cIixgLFxuICAgICAgICAgICAgYH0pO2AsXG4gICAgICAgICAgKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBNYW55VG9NYW55XG4gICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgIFwicmV0dXJuIHFiV3JhcHBlclwiLFxuICAgICAgICAgICAgYC5mcm9tKFwiJHt0aHJvdWdoLnRhYmxlfVwiKWAsXG4gICAgICAgICAgICBgLmpvaW4oXCIke3RvVGFibGV9XCIsIFwiJHt0aHJvdWdoLnRhYmxlfS4ke3Rocm91Z2gudG9Db2x9XCIsIFwiJHt0b1RhYmxlfS4ke3RvQ29sfVwiKWAsXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIGxvYWRlci5vbmVKb2lucy5mb3JFYWNoKChqb2luOiBTdWJzZXRRdWVyeVtcImpvaW5zXCJdW251bWJlcl0pID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGpvaW5UeXBlID0gam9pbi5qb2luID09PSBcImlubmVyXCIgPyBcImpvaW5cIiA6IFwibGVmdEpvaW5cIjtcbiAgICAgICAgICAgIGlmIChcImN1c3RvbVwiIGluIGpvaW4pIHtcbiAgICAgICAgICAgICAgLy8gRklYTUU6IOqygOymnSDtlYTsmpRcbiAgICAgICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgICAgICBgLiR7am9pblR5cGV9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgKGopID0+IHtgLFxuICAgICAgICAgICAgICAgIGBqLm9uKFB1cmkucmF3U3RyaW5nKFwiJHtqb2luLmN1c3RvbX1cIikpO2AsXG4gICAgICAgICAgICAgICAgXCJ9KVwiLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgICAgICBgLiR7am9pblR5cGV9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgXCIke2pvaW4uZnJvbX1cIiwgXCIke2pvaW4udG99XCIpYCxcbiAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KTtcbiAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgYC53aGVyZUluKFwiJHt0aHJvdWdoLnRhYmxlfS4ke3Rocm91Z2guZnJvbUNvbH1cIiwgZnJvbUlkcylgLFxuICAgICAgICAgICAgYC5zZWxlY3Qoe2AsXG4gICAgICAgICAgICBgJHtsb2FkZXIuc2VsZWN0Lm1hcCgoc2VsZWN0OiBzdHJpbmcpID0+IHBhcnNlU2VsZWN0KHNlbGVjdCwgdG9UYWJsZSkpLmpvaW4oXCIsXCIpfSxgLFxuICAgICAgICAgICAgYHJlZklkOiBcIiR7dGhyb3VnaC50YWJsZX0uJHt0aHJvdWdoLmZyb21Db2x9XCIsYCxcbiAgICAgICAgICAgIGB9KTtgLFxuICAgICAgICAgICk7XG4gICAgICAgIH1cblxuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKGB9LGApO1xuXG4gICAgICAgIC8vIOykkeyyqSBsb2FkZXJzIOyymOumrFxuICAgICAgICBpZiAobG9hZGVyLmxvYWRlcnMgJiYgbG9hZGVyLmxvYWRlcnMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGxvYWRlckxpbmVzLnB1c2goXCJsb2FkZXJzOiBbXCIsIC4uLmdlbmVyYXRlTG9hZGVyQ29kZShsb2FkZXIubG9hZGVycyksIFwiXSxcIik7XG4gICAgICAgIH1cblxuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFwifSxcIik7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBsb2FkZXJMaW5lcztcbiAgICB9O1xuXG4gICAgbGluZXMucHVzaCguLi5nZW5lcmF0ZUxvYWRlckNvZGUobG9hZGVycykpO1xuICAgIGxpbmVzLnB1c2goYF1gKTtcblxuICAgIHJldHVybiBsaW5lcy5qb2luKFwiXFxuXCIpO1xuICB9XG5cbiAgLypcbiAgICBzdWJzZXQgU0VMRUNUL0pPSU4vTE9BREVSIOqysOqzvCDrpqzthLRcbiAgKi9cbiAgZ2V0U3Vic2V0UXVlcnkoc3Vic2V0S2V5OiBzdHJpbmcpOiBTdWJzZXRRdWVyeSB7XG4gICAgY29uc3Qgc3Vic2V0ID0gdGhpcy5zdWJzZXRzW3N1YnNldEtleV07XG5cbiAgICBjb25zdCByZXN1bHQ6IFN1YnNldFF1ZXJ5ID0gdGhpcy5yZXNvbHZlU3Vic2V0UXVlcnkoXCJcIiwgc3Vic2V0KTtcbiAgICByZXR1cm4gcmVzdWx0O1xuICB9XG5cbiAgLypcbiAgICovXG4gIHJlc29sdmVTdWJzZXRRdWVyeShcbiAgICBwcmVmaXg6IHN0cmluZyxcbiAgICBmaWVsZHM6IHN0cmluZ1tdLFxuICAgIGlzQWxyZWFkeU91dGVySm9pbmVkOiBib29sZWFuID0gZmFsc2UsXG4gICk6IFN1YnNldFF1ZXJ5IHtcbiAgICAvLyBwcmVmaXgg7LmY7ZmYIChwcmVmaXjripQgVG9PbmVSZWxhdGlvbuydtCDrs7XsiJjroZwg67aZ7J2AIOqyveyasCDrqqjrkZAgX1/roZwg67OA6rK965CoKVxuICAgIHByZWZpeCA9IHByZWZpeC5yZXBsYWNlKC9cXC4vZywgXCJfX1wiKTtcblxuICAgIC8vIOyEnOu4jOyFi+ydhCAx646B7Iqk66eMIOu2hOumrO2VmOyXrCDqt7jro7ntlZFcbiAgICBjb25zdCBzdWJzZXRHcm91cCA9IGdyb3VwKGZpZWxkcywgKGZpZWxkKSA9PiB7XG4gICAgICBpZiAoZmllbGQuaW5jbHVkZXMoXCIuXCIpKSB7XG4gICAgICAgIGNvbnN0IFtyZWxdID0gZmllbGQuc3BsaXQoXCIuXCIpO1xuICAgICAgICByZXR1cm4gcmVsO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIFwiXCI7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBjb25zdCByZXN1bHQgPSBPYmplY3Qua2V5cyhzdWJzZXRHcm91cCkucmVkdWNlKFxuICAgICAgKHIsIGdyb3VwS2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGZpZWxkcyA9IHN1YnNldEdyb3VwW2dyb3VwS2V5XTtcbiAgICAgICAgYXNzZXJ0KGZpZWxkcyAhPT0gdW5kZWZpbmVkLCBcImZpZWxkcyBpcyB1bmRlZmluZWRcIik7XG5cbiAgICAgICAgLy8g7ZiE7J6sIO2FjOydtOu4lCDtlYTrk5zshYvsnYAgc2VsZWN0LCB2aXJ0dWFs7JeQIOy2lOqwgO2VmOqzoCDrpqzthLRcbiAgICAgICAgaWYgKGdyb3VwS2V5ID09PSBcIlwiKSB7XG4gICAgICAgICAgY29uc3QgcmVhbEZpZWxkcyA9IGZpZWxkcy5maWx0ZXIoKGZpZWxkKSA9PiAhaXNWaXJ0dWFsUHJvcCh0aGlzLnByb3BzRGljdFtmaWVsZF0pKTtcbiAgICAgICAgICBjb25zdCB2aXJ0dWFsRmllbGRzID0gZmllbGRzLmZpbHRlcigoZmllbGQpID0+IGlzVmlydHVhbFByb3AodGhpcy5wcm9wc0RpY3RbZmllbGRdKSk7XG5cbiAgICAgICAgICBpZiAocHJlZml4ID09PSBcIlwiKSB7XG4gICAgICAgICAgICAvLyDtmITsnqwg7YWM7J2067iU7J24IOqyveyasFxuICAgICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQocmVhbEZpZWxkcy5tYXAoKGZpZWxkKSA9PiBgJHt0aGlzLnRhYmxlfS4ke2ZpZWxkfWApKTtcbiAgICAgICAgICAgIHIudmlydHVhbCA9IHIudmlydHVhbC5jb25jYXQodmlydHVhbEZpZWxkcyk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIOuEmOyWtOyYqCDthYzsnbTruJTsnbgg6rK97JqwXG4gICAgICAgICAgICByLnNlbGVjdCA9IHIuc2VsZWN0LmNvbmNhdChcbiAgICAgICAgICAgICAgcmVhbEZpZWxkcy5tYXAoKGZpZWxkKSA9PiBgJHtwcmVmaXh9LiR7ZmllbGR9IGFzICR7cHJlZml4fV9fJHtmaWVsZH1gKSxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIHI7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCByZWxhdGlvbiA9IHRoaXMucmVsYXRpb25zW2dyb3VwS2V5XTtcbiAgICAgICAgaWYgKHJlbGF0aW9uID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYOyhtOyerO2VmOyngCDslYrripQgcmVsYXRpb24g7LC47KGwICR7Z3JvdXBLZXl9YCk7XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG5cbiAgICAgICAgaWYgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8IGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgIC8vIC1PbmUgUmVsYXRpb246IEpPSU4g7Jy866GcIOyymOumrFxuICAgICAgICAgIGNvbnN0IHJlbEZpZWxkcyA9IGZpZWxkcy5tYXAoKGZpZWxkKSA9PiBmaWVsZC5zcGxpdChcIi5cIikuc2xpY2UoMSkuam9pbihcIi5cIikpO1xuXG4gICAgICAgICAgLy8gLU9uZSBSZWxhdGlvbuyXkOyEnCBpZCDtlYTrk5zrp4wg7LC47KGw7ZWY64qUIOqyveyasCDrprTroIjsnbTshZgg64SY6riw7KeAIOyViuqzoCDrpqzthLRcbiAgICAgICAgICBpZiAocmVsRmllbGRzLmxlbmd0aCA9PT0gMSAmJiByZWxGaWVsZHNbMF0gPT09IFwiaWRcIikge1xuICAgICAgICAgICAgaWYgKHByZWZpeCA9PT0gXCJcIikge1xuICAgICAgICAgICAgICByLnNlbGVjdCA9IHIuc2VsZWN0LmNvbmNhdChgJHt0aGlzLnRhYmxlfS4ke2dyb3VwS2V5fV9pZGApO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQoYCR7cHJlZml4fS4ke2dyb3VwS2V5fV9pZCBhcyAke3ByZWZpeH1fXyR7Z3JvdXBLZXl9X2lkYCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gcjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBpbm5lck9yT3V0ZXJcbiAgICAgICAgICBjb25zdCBpbm5lck9yT3V0ZXIgPSAoKCkgPT4ge1xuICAgICAgICAgICAgaWYgKGlzQWxyZWFkeU91dGVySm9pbmVkKSB7XG4gICAgICAgICAgICAgIHJldHVybiBcIm91dGVyXCI7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChpc09uZVRvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgICAgICBpZiAocmVsYXRpb24uaGFzSm9pbkNvbHVtbiA9PT0gdHJ1ZSAmJiAocmVsYXRpb24ubnVsbGFibGUgPz8gZmFsc2UpID09PSBmYWxzZSkge1xuICAgICAgICAgICAgICAgIHJldHVybiBcImlubmVyXCI7XG4gICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgaWYgKHJlbGF0aW9uLm51bGxhYmxlKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICByZXR1cm4gXCJpbm5lclwiO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSkoKTtcbiAgICAgICAgICBjb25zdCByZWxTdWJzZXRRdWVyeSA9IHJlbEVudGl0eS5yZXNvbHZlU3Vic2V0UXVlcnkoXG4gICAgICAgICAgICBgJHtwcmVmaXggIT09IFwiXCIgPyBgJHtwcmVmaXh9LmAgOiBcIlwifSR7Z3JvdXBLZXl9YCxcbiAgICAgICAgICAgIHJlbEZpZWxkcyxcbiAgICAgICAgICAgIGlubmVyT3JPdXRlciA9PT0gXCJvdXRlclwiLFxuICAgICAgICAgICk7XG4gICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQocmVsU3Vic2V0UXVlcnkuc2VsZWN0KTtcbiAgICAgICAgICByLnZpcnR1YWwgPSByLnZpcnR1YWwuY29uY2F0KHJlbFN1YnNldFF1ZXJ5LnZpcnR1YWwpO1xuXG4gICAgICAgICAgY29uc3Qgam9pbkFzID0gcHJlZml4ID09PSBcIlwiID8gZ3JvdXBLZXkgOiBgJHtwcmVmaXh9X18ke2dyb3VwS2V5fWA7XG4gICAgICAgICAgY29uc3QgZnJvbVRhYmxlID0gcHJlZml4ID09PSBcIlwiID8gdGhpcy50YWJsZSA6IHByZWZpeDtcblxuICAgICAgICAgIGxldCBqb2luQ2xhdXNlOlxuICAgICAgICAgICAgfCB7XG4gICAgICAgICAgICAgICAgZnJvbTogc3RyaW5nO1xuICAgICAgICAgICAgICAgIHRvOiBzdHJpbmc7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHwge1xuICAgICAgICAgICAgICAgIGN1c3RvbTogc3RyaW5nO1xuICAgICAgICAgICAgICB9O1xuICAgICAgICAgIGlmIChyZWxhdGlvbi5jdXN0b21Kb2luQ2xhdXNlKSB7XG4gICAgICAgICAgICBqb2luQ2xhdXNlID0ge1xuICAgICAgICAgICAgICBjdXN0b206IHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBsZXQgZnJvbTogc3RyaW5nLCB0bzogc3RyaW5nO1xuICAgICAgICAgICAgaWYgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGlvbi5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgICAgICAgZnJvbSA9IGAke2Zyb21UYWJsZX0uJHtyZWxhdGlvbi5uYW1lfV9pZGA7XG4gICAgICAgICAgICAgICAgdG8gPSBgJHtqb2luQXN9LmlkYDtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBmcm9tID0gYCR7ZnJvbVRhYmxlfS5pZGA7XG4gICAgICAgICAgICAgICAgdG8gPSBgJHtqb2luQXN9LiR7aW5mbGVjdGlvbi51bmRlcnNjb3JlKHRoaXMubmFtZXMuZnMucmVwbGFjZSgvLS9nLCBcIl9cIikpfV9pZGA7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIGZyb20gPSBgJHtmcm9tVGFibGV9LiR7cmVsYXRpb24ubmFtZX1faWRgO1xuICAgICAgICAgICAgICB0byA9IGAke2pvaW5Bc30uaWRgO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgam9pbkNsYXVzZSA9IHtcbiAgICAgICAgICAgICAgZnJvbSxcbiAgICAgICAgICAgICAgdG8sXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMucHVzaCh7XG4gICAgICAgICAgICBhczogam9pbkFzLFxuICAgICAgICAgICAgam9pbjogaW5uZXJPck91dGVyLFxuICAgICAgICAgICAgdGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgIC4uLmpvaW5DbGF1c2UsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAvLyBCZWxvbmdzVG9PbmUg67CR7JeQIEhhc01hbnnqsIAg67aZ7J2AIOqyveyasFxuICAgICAgICAgIGlmIChyZWxTdWJzZXRRdWVyeS5sb2FkZXJzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIGNvbnN0IGNvbnZlcnRlZExvYWRlcnMgPSByZWxTdWJzZXRRdWVyeS5sb2FkZXJzLm1hcCgobG9hZGVyKSA9PiB7XG4gICAgICAgICAgICAgIGNvbnN0IG5ld0FzID0gW2dyb3VwS2V5LCBsb2FkZXIuYXNdLmpvaW4oXCJfX1wiKTtcbiAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBhczogbmV3QXMsXG4gICAgICAgICAgICAgICAgdGFibGU6IGxvYWRlci50YWJsZSxcbiAgICAgICAgICAgICAgICBtYW55Sm9pbjogbG9hZGVyLm1hbnlKb2luLFxuICAgICAgICAgICAgICAgIG9uZUpvaW5zOiBsb2FkZXIub25lSm9pbnMsXG4gICAgICAgICAgICAgICAgc2VsZWN0OiBsb2FkZXIuc2VsZWN0LFxuICAgICAgICAgICAgICAgIGxvYWRlcnM6IGxvYWRlci5sb2FkZXJzLFxuICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIHIubG9hZGVycyA9IFsuLi5yLmxvYWRlcnMsIC4uLmNvbnZlcnRlZExvYWRlcnNdO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMgPSByLmpvaW5zLmNvbmNhdChyZWxTdWJzZXRRdWVyeS5qb2lucyk7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNIYXNNYW55UmVsYXRpb25Qcm9wKHJlbGF0aW9uKSB8fCBpc01hbnlUb01hbnlSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgLy8gLU1hbnkgUmVsYXRpb246IExvYWRlciDroZwg7LKY66asXG4gICAgICAgICAgY29uc3QgcmVsRmllbGRzID0gZmllbGRzLm1hcCgoZmllbGQpID0+IGZpZWxkLnNwbGl0KFwiLlwiKS5zbGljZSgxKS5qb2luKFwiLlwiKSk7XG4gICAgICAgICAgY29uc3QgcmVsU3Vic2V0UXVlcnkgPSByZWxFbnRpdHkucmVzb2x2ZVN1YnNldFF1ZXJ5KFwiXCIsIHJlbEZpZWxkcyk7XG5cbiAgICAgICAgICBsZXQgbWFueUpvaW46IFN1YnNldFF1ZXJ5W1wibG9hZGVyc1wiXVtudW1iZXJdW1wibWFueUpvaW5cIl07XG4gICAgICAgICAgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChyZWxhdGlvbikpIHtcbiAgICAgICAgICAgIGNvbnN0IGZyb21Db2wgPSByZWxhdGlvbj8uZnJvbUNvbHVtbiA/PyBcImlkXCI7XG4gICAgICAgICAgICBtYW55Sm9pbiA9IHtcbiAgICAgICAgICAgICAgZnJvbVRhYmxlOiB0aGlzLnRhYmxlLFxuICAgICAgICAgICAgICBmcm9tQ29sLFxuICAgICAgICAgICAgICBpZEZpZWxkOiBwcmVmaXggPT09IFwiXCIgPyBgJHtmcm9tQ29sfWAgOiBgJHtwcmVmaXh9X18ke2Zyb21Db2x9YCxcbiAgICAgICAgICAgICAgdG9UYWJsZTogcmVsRW50aXR5LnRhYmxlLFxuICAgICAgICAgICAgICB0b0NvbDogcmVsYXRpb24uam9pbkNvbHVtbixcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgfSBlbHNlIGlmIChpc01hbnlUb01hbnlSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgICBtYW55Sm9pbiA9IHtcbiAgICAgICAgICAgICAgZnJvbVRhYmxlOiB0aGlzLnRhYmxlLFxuICAgICAgICAgICAgICBmcm9tQ29sOiBcImlkXCIsXG4gICAgICAgICAgICAgIGlkRmllbGQ6IHByZWZpeCA9PT0gXCJcIiA/IGBpZGAgOiBgJHtwcmVmaXh9X19pZGAsXG4gICAgICAgICAgICAgIHRocm91Z2g6IHtcbiAgICAgICAgICAgICAgICB0YWJsZTogcmVsYXRpb24uam9pblRhYmxlLFxuICAgICAgICAgICAgICAgIGZyb21Db2w6IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUodGhpcy50YWJsZSl9X2lkYCxcbiAgICAgICAgICAgICAgICB0b0NvbDogYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShyZWxFbnRpdHkudGFibGUpfV9pZGAsXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIHRvVGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgICAgdG9Db2w6IFwiaWRcIixcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcigpO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIubG9hZGVycy5wdXNoKHtcbiAgICAgICAgICAgIGFzOiBncm91cEtleSxcbiAgICAgICAgICAgIHRhYmxlOiByZWxFbnRpdHkudGFibGUsXG4gICAgICAgICAgICBtYW55Sm9pbixcbiAgICAgICAgICAgIG9uZUpvaW5zOiByZWxTdWJzZXRRdWVyeS5qb2lucyxcbiAgICAgICAgICAgIHNlbGVjdDogcmVsU3Vic2V0UXVlcnkuc2VsZWN0LFxuICAgICAgICAgICAgbG9hZGVyczogcmVsU3Vic2V0UXVlcnkubG9hZGVycyxcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiByO1xuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgc2VsZWN0OiBbXSxcbiAgICAgICAgdmlydHVhbDogW10sXG4gICAgICAgIGpvaW5zOiBbXSxcbiAgICAgICAgbG9hZGVyczogW10sXG4gICAgICB9IGFzIFN1YnNldFF1ZXJ5LFxuICAgICk7XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qXG4gICAgRmllbGRFeHByW10g7J2EIEVudGl0eVByb3BOb2RlW10g66GcIOuzgO2ZmFxuICAqL1xuICBmaWVsZEV4cHJzVG9Qcm9wTm9kZXMoZmllbGRFeHByczogc3RyaW5nW10sIGVudGl0eTogRW50aXR5ID0gdGhpcyk6IEVudGl0eVByb3BOb2RlW10ge1xuICAgIGNvbnN0IGdyb3VwcyA9IGZpZWxkRXhwcnMucmVkdWNlKFxuICAgICAgKHJlc3VsdCwgZmllbGRFeHByKSA9PiB7XG4gICAgICAgIGxldCBrZXk6IHN0cmluZywgdmFsdWU6IHN0cmluZywgZWxzZUV4cHI6IHN0cmluZ1tdO1xuICAgICAgICBpZiAoZmllbGRFeHByLmluY2x1ZGVzKFwiLlwiKSkge1xuICAgICAgICAgIFtrZXksIC4uLmVsc2VFeHByXSA9IGZpZWxkRXhwci5zcGxpdChcIi5cIik7XG4gICAgICAgICAgdmFsdWUgPSBlbHNlRXhwci5qb2luKFwiLlwiKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBrZXkgPSBcIlwiO1xuICAgICAgICAgIHZhbHVlID0gZmllbGRFeHByO1xuICAgICAgICB9XG4gICAgICAgIHJlc3VsdFtrZXldID0gKHJlc3VsdFtrZXldID8/IFtdKS5jb25jYXQodmFsdWUpO1xuXG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9LFxuICAgICAge30gYXMge1xuICAgICAgICBbazogc3RyaW5nXTogc3RyaW5nW107XG4gICAgICB9LFxuICAgICk7XG5cbiAgICByZXR1cm4gT2JqZWN0LmtleXMoZ3JvdXBzKS5mbGF0TWFwKChrZXkpID0+IHtcbiAgICAgIGNvbnN0IGdyb3VwID0gZ3JvdXBzW2tleV07XG5cbiAgICAgIC8vIOydvOuwmCBwcm9wIOyymOumrFxuICAgICAgaWYgKGtleSA9PT0gXCJcIikge1xuICAgICAgICByZXR1cm4gZ3JvdXAubWFwKChwcm9wTmFtZSkgPT4ge1xuICAgICAgICAgIC8vIHV1aWQg6rCc67OEIOyymOumrFxuICAgICAgICAgIGlmIChwcm9wTmFtZSA9PT0gXCJ1dWlkXCIpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICAgIHByb3A6IHtcbiAgICAgICAgICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICAgICAgICAgIG5hbWU6IFwidXVpZFwiLFxuICAgICAgICAgICAgICAgIGxlbmd0aDogMTI4LFxuICAgICAgICAgICAgICB9IGFzIFN0cmluZ1Byb3AsXG4gICAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICAgIH0gYXMgRW50aXR5UHJvcE5vZGU7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgcHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IHByb3BOYW1lKTtcbiAgICAgICAgICBpZiAocHJvcCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyh7IHByb3BOYW1lLCBncm91cHMgfSk7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5LmlkfSAtLSDsnpjrqrvrkJwgRmllbGRFeHByICR7cHJvcE5hbWV9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBub2RlVHlwZTogXCJwbGFpblwiIGFzIGNvbnN0LFxuICAgICAgICAgICAgcHJvcCxcbiAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICB9O1xuICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgLy8gcmVsYXRpb24gcHJvcCDsspjrpqxcbiAgICAgIGNvbnN0IHByb3AgPSBlbnRpdHkucHJvcHNEaWN0W2tleV07XG4gICAgICBpZiAoIWlzUmVsYXRpb25Qcm9wKHByb3ApKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihg7J6Y66q765CcIEZpZWxkRXhwciAke2tleX0uJHtncm91cFswXX1gKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IHJlbEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG5cbiAgICAgIC8vIHJlbGF0aW9uIC1PbmUg7JeQIGlkIO2VhOuTnCDtlZjrgpjsnbgg6rK97JqwXG4gICAgICBpZiAoaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHwgaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoZ3JvdXAubGVuZ3RoID09PSAxICYmIChncm91cFswXSA9PT0gXCJpZFwiIHx8IGdyb3VwWzBdID09PSBcImlkP1wiKSkge1xuICAgICAgICAgIC8vIGlkIO2VmOuCmOunjCDsnojripTsp4Ag7LK07YGs7ZW07IScLCDtlZjrgpjrp4wg7J6I7Jy866m0IOyDgeychCBwcm9w7Jy866GcIGlk66W8IOumrO2EtFxuICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IHJlbEVudGl0eS5wcm9wc0RpY3QuaWQ7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICBwcm9wOiB7XG4gICAgICAgICAgICAgIC4uLmlkUHJvcCxcbiAgICAgICAgICAgICAgbmFtZTogYCR7a2V5fV9pZGAsXG4gICAgICAgICAgICAgIG51bGxhYmxlOiBwcm9wLm51bGxhYmxlLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIC1PbmUg6re47Jm47J2YIOqyveyasCBvYmplY3TroZwg66as7YS0XG4gICAgICAvLyAtTWFueeydmCDqsr3smrAgYXJyYXnroZwg66as7YS0XG4gICAgICAvLyBSZWN1cnNpdmUg66GcIOuOgeyKpCDsspjrpqxcbiAgICAgIGNvbnN0IGNoaWxkcmVuID0gdGhpcy5maWVsZEV4cHJzVG9Qcm9wTm9kZXMoZ3JvdXAsIHJlbEVudGl0eSk7XG4gICAgICBjb25zdCBub2RlVHlwZSA9XG4gICAgICAgIGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHByb3ApIHx8IGlzT25lVG9PbmVSZWxhdGlvblByb3AocHJvcClcbiAgICAgICAgICA/IChcIm9iamVjdFwiIGFzIGNvbnN0KVxuICAgICAgICAgIDogKFwiYXJyYXlcIiBhcyBjb25zdCk7XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIHByb3AsXG4gICAgICAgIGNoaWxkcmVuLFxuICAgICAgICBub2RlVHlwZSxcbiAgICAgIH07XG4gICAgfSk7XG4gIH1cblxuICBnZXRGaWVsZEV4cHJzKHByZWZpeCA9IFwiXCIsIG1heERlcHRoOiBudW1iZXIgPSAzLCBmcm9tczogc3RyaW5nW10gPSBbXSk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gdGhpcy5wcm9wc1xuICAgICAgLmZsYXRNYXAoKHByb3ApID0+IHtcbiAgICAgICAgY29uc3QgcHJvcE5hbWUgPSBbcHJlZml4LCBwcm9wLm5hbWVdLmZpbHRlcigodikgPT4gdiAhPT0gXCJcIikuam9pbihcIi5cIik7XG4gICAgICAgIGlmIChwcm9wTmFtZSA9PT0gcHJlZml4KSB7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKGlzUmVsYXRpb25Qcm9wKHByb3ApKSB7XG4gICAgICAgICAgaWYgKG1heERlcHRoIDwgMCkge1xuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChmcm9tcy5pbmNsdWRlcyhwcm9wLndpdGgpKSB7XG4gICAgICAgICAgICAvLyDsl63rsKntlqUgcmVsYXRpb27snbgg6rK97JqwIOygnOyZuFxuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIOygleuwqe2WpSByZWxhdGlvbuyduCDqsr3smrAgcmVjdXJzaXZlIOy9nFxuICAgICAgICAgIGNvbnN0IHJlbE1kID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcbiAgICAgICAgICByZXR1cm4gcmVsTWQuZ2V0RmllbGRFeHBycyhwcm9wTmFtZSwgbWF4RGVwdGggLSAxLCBbLi4uZnJvbXMsIHRoaXMuaWRdKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcHJvcE5hbWU7XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoZikgPT4gZiAhPT0gbnVsbCkgYXMgc3RyaW5nW107XG4gIH1cblxuICBnZXRUYWJsZUNvbHVtbnMoKTogeyBuYW1lOiBzdHJpbmc7IHR5cGU6IHN0cmluZyB9W10ge1xuICAgIHJldHVybiB0aGlzLnByb3BzXG4gICAgICAubWFwKChwcm9wKSA9PiB7XG4gICAgICAgIGlmIChwcm9wLnR5cGUgPT09IFwicmVsYXRpb25cIikge1xuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIHByb3AucmVsYXRpb25UeXBlID09PSBcIkJlbG9uZ3NUb09uZVwiIHx8XG4gICAgICAgICAgICAocHJvcC5yZWxhdGlvblR5cGUgPT09IFwiT25lVG9PbmVcIiAmJiBwcm9wLmhhc0pvaW5Db2x1bW4gPT09IHRydWUpXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICByZXR1cm4geyBuYW1lOiBgJHtwcm9wLm5hbWV9X2lkYCwgdHlwZTogXCJpbnRfdW5zaWduZWRcIiB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHsgbmFtZTogcHJvcC5uYW1lLCB0eXBlOiBwcm9wLnR5cGUgfTtcbiAgICAgIH0pXG4gICAgICAuZmlsdGVyKG5vbk51bGxhYmxlKTtcbiAgfVxuXG4gIGFzeW5jIHJlZ2lzdGVyTW9kdWxlUGF0aHMoKSB7XG4gICAgY29uc3QgYmFzZVBhdGggPSBgJHt0aGlzLm5hbWVzLnBhcmVudEZzfWA7XG5cbiAgICAvLyBiYXNlLXNjaGVtZVxuICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChgJHt0aGlzLmlkfUJhc2VTY2hlbWFgLCBgc29uYW11LmdlbmVyYXRlZGApO1xuXG4gICAgLy8gc3Vic2V0XG4gICAgaWYgKE9iamVjdC5rZXlzKHRoaXMuc3Vic2V0cykubGVuZ3RoID4gMCkge1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGAke3RoaXMuaWR9U3Vic2V0S2V5YCwgYHNvbmFtdS5nZW5lcmF0ZWRgKTtcbiAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChgJHt0aGlzLmlkfVN1YnNldE1hcHBpbmdgLCBgc29uYW11LmdlbmVyYXRlZGApO1xuICAgICAgZm9yIChjb25zdCBzdWJzZXRLZXkgb2YgT2JqZWN0LmtleXModGhpcy5zdWJzZXRzKSkge1xuICAgICAgICBFbnRpdHlNYW5hZ2VyLnNldE1vZHVsZVBhdGgoXG4gICAgICAgICAgYCR7dGhpcy5pZH1TdWJzZXQke3N1YnNldEtleS50b1VwcGVyQ2FzZSgpfWAsXG4gICAgICAgICAgYHNvbmFtdS5nZW5lcmF0ZWRgLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIGVudW1zXG4gICAgZm9yIChjb25zdCBlbnVtSWQgb2YgT2JqZWN0LmtleXModGhpcy5lbnVtTGFiZWxzKSkge1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGVudW1JZCwgYHNvbmFtdS5nZW5lcmF0ZWRgKTtcbiAgICB9XG5cbiAgICAvLyB0eXBlc1xuICAgIGNvbnN0IHR5cGVzTW9kdWxlUGF0aCA9IGAke2Jhc2VQYXRofS8ke3RoaXMubmFtZXMucGFyZW50RnN9LnR5cGVzYDtcbiAgICBjb25zdCB0eXBlc0ZpbGVQYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgcnVudGltZVBhdGgoYGRpc3QvYXBwbGljYXRpb24vJHt0eXBlc01vZHVsZVBhdGh9LmpzYCksXG4gICAgKTtcblxuICAgIGlmIChhd2FpdCBleGlzdHModHlwZXNGaWxlUGF0aCkpIHtcbiAgICAgIGNvbnN0IGltcG9ydGVkTWVtYmVycyA9IGF3YWl0IGltcG9ydE1lbWJlcnM8ei5ab2RUeXBlQW55Pih0eXBlc0ZpbGVQYXRoKTtcbiAgICAgIHRoaXMudHlwZXMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICAgIGltcG9ydGVkTWVtYmVycy5tYXAoKHsgbmFtZSwgdmFsdWUgfSkgPT4ge1xuICAgICAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChuYW1lLCB0eXBlc01vZHVsZVBhdGgpO1xuICAgICAgICAgIHJldHVybiBbbmFtZSwgdmFsdWVdO1xuICAgICAgICB9KSxcbiAgICAgICkgYXMgeyBbbmFtZTogc3RyaW5nXTogei5ab2RUeXBlQW55IH07XG4gICAgfVxuICB9XG5cbiAgcmVnaXN0ZXJUYWJsZVNwZWNzKCk6IHZvaWQge1xuICAgIGNvbnN0IHVuaXF1ZUluZGV4ZXMgPSB0aGlzLmluZGV4ZXMuZmlsdGVyKChpZHgpID0+IGlkeC50eXBlID09PSBcInVuaXF1ZVwiKTtcblxuICAgIEVudGl0eU1hbmFnZXIuc2V0VGFibGVTcGVjKHtcbiAgICAgIG5hbWU6IHRoaXMudGFibGUsXG4gICAgICB1bmlxdWVJbmRleGVzLFxuICAgIH0pO1xuICB9XG5cbiAgdG9Kc29uKCk6IEVudGl0eUpzb24ge1xuICAgIHJldHVybiB7XG4gICAgICBpZDogdGhpcy5pZCxcbiAgICAgIHBhcmVudElkOiB0aGlzLnBhcmVudElkLFxuICAgICAgdGFibGU6IHRoaXMudGFibGUsXG4gICAgICB0aXRsZTogdGhpcy50aXRsZSxcbiAgICAgIHByb3BzOiB0aGlzLnByb3BzLFxuICAgICAgaW5kZXhlczogdGhpcy5pbmRleGVzLFxuICAgICAgc3Vic2V0czogdGhpcy5zdWJzZXRzLFxuICAgICAgZW51bXM6IHRoaXMuZW51bUxhYmVscyxcbiAgICB9O1xuICB9XG5cbiAgYXN5bmMgc2F2ZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBzb3J0OiBzdWJzZXRzXG4gICAgY29uc3Qgc3Vic2V0Um93cyA9IHRoaXMuZ2V0U3Vic2V0Um93cygpO1xuICAgIHRoaXMuc3Vic2V0cyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHRoaXMuc3Vic2V0cykubWFwKChbc3Vic2V0S2V5XSkgPT4ge1xuICAgICAgICByZXR1cm4gW3N1YnNldEtleSwgdGhpcy5zdWJzZXRSb3dzVG9TdWJzZXRGaWVsZHMoc3Vic2V0Um93cywgc3Vic2V0S2V5KV07XG4gICAgICB9KSxcbiAgICApO1xuXG4gICAgLy8gc2F2ZVxuICAgIGNvbnN0IGpzb25QYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgYHNyYy9hcHBsaWNhdGlvbi8ke3RoaXMubmFtZXMucGFyZW50RnN9LyR7dGhpcy5uYW1lcy5mc30uZW50aXR5Lmpzb25gLFxuICAgICk7XG4gICAgY29uc3QganNvbiA9IHRoaXMudG9Kc29uKCk7XG4gICAgYXdhaXQgd3JpdGVGaWxlKGpzb25QYXRoLCBmb3JtYXRDb2RlKEpTT04uc3RyaW5naWZ5KGpzb24pLCBcImpzb25cIiwganNvblBhdGgpKTtcblxuICAgIC8vIHJlbG9hZFxuICAgIGF3YWl0IEVudGl0eU1hbmFnZXIucmVnaXN0ZXIoanNvbik7XG4gIH1cblxuICBnZXRTdWJzZXRSb3dzKFxuICAgIF9zdWJzZXRzPzogeyBba2V5OiBzdHJpbmddOiBzdHJpbmdbXSB9LFxuICAgIHByZWZpeGVzOiBzdHJpbmdbXSA9IFtdLFxuICApOiBFbnRpdHlTdWJzZXRSb3dbXSB7XG4gICAgaWYgKHByZWZpeGVzLmxlbmd0aCA+IDEwKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3Qgc3Vic2V0cyA9IF9zdWJzZXRzID8/IHRoaXMuc3Vic2V0cztcbiAgICBjb25zdCBzdWJzZXRLZXlzID0gT2JqZWN0LmtleXMoc3Vic2V0cyk7XG4gICAgY29uc3QgYWxsRmllbGRzID0gdW5pcXVlKHN1YnNldEtleXMuZmxhdE1hcCgoa2V5KSA9PiBzdWJzZXRzW2tleV0pKTtcblxuICAgIHJldHVybiB0aGlzLnByb3BzLm1hcCgocHJvcCkgPT4ge1xuICAgICAgaWYgKFxuICAgICAgICBwcm9wLnR5cGUgPT09IFwicmVsYXRpb25cIiAmJlxuICAgICAgICBhbGxGaWVsZHMuZmluZCgoZikgPT4gZi5zdGFydHNXaXRoKGAke1suLi5wcmVmaXhlcywgcHJvcC5uYW1lXS5qb2luKFwiLlwiKX0uYCkpXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcbiAgICAgICAgY29uc3QgY2hpbGRyZW4gPSByZWxFbnRpdHkuZ2V0U3Vic2V0Um93cyhzdWJzZXRzLCBbLi4ucHJlZml4ZXMsIGAke3Byb3AubmFtZX1gXSk7XG5cbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBmaWVsZDogcHJvcC5uYW1lLFxuICAgICAgICAgIGNoaWxkcmVuLFxuICAgICAgICAgIHJlbGF0aW9uRW50aXR5OiBwcm9wLndpdGgsXG4gICAgICAgICAgcHJlZml4ZXMsXG4gICAgICAgICAgaXNPcGVuOiBjaGlsZHJlbi5sZW5ndGggPiAwLFxuICAgICAgICAgIGhhczogT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgICAgICAgICAgc3Vic2V0S2V5cy5tYXAoKHN1YnNldEtleSkgPT4ge1xuICAgICAgICAgICAgICByZXR1cm4gW3N1YnNldEtleSwgY2hpbGRyZW4uZXZlcnkoKGNoaWxkKSA9PiBjaGlsZC5oYXNbc3Vic2V0S2V5XSA9PT0gdHJ1ZSldO1xuICAgICAgICAgICAgfSksXG4gICAgICAgICAgKSxcbiAgICAgICAgfTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgZmllbGQ6IHByb3AubmFtZSxcbiAgICAgICAgY2hpbGRyZW46IFtdLFxuICAgICAgICByZWxhdGlvbkVudGl0eTogcHJvcC50eXBlID09PSBcInJlbGF0aW9uXCIgPyBwcm9wLndpdGggOiB1bmRlZmluZWQsXG4gICAgICAgIHByZWZpeGVzLFxuICAgICAgICBoYXM6IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgICAgICBzdWJzZXRLZXlzLm1hcCgoc3Vic2V0S2V5KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBzdWJzZXRGaWVsZHMgPSBzdWJzZXRzW3N1YnNldEtleV07XG4gICAgICAgICAgICBjb25zdCBoYXMgPSBzdWJzZXRGaWVsZHMuc29tZSgoZikgPT4ge1xuICAgICAgICAgICAgICBjb25zdCBmaWVsZCA9IFsuLi5wcmVmaXhlcywgcHJvcC5uYW1lXS5qb2luKFwiLlwiKTtcbiAgICAgICAgICAgICAgcmV0dXJuIGYgPT09IGZpZWxkIHx8IGYuc3RhcnRzV2l0aChgJHtmaWVsZH0uYCk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHJldHVybiBbc3Vic2V0S2V5LCBoYXNdO1xuICAgICAgICAgIH0pLFxuICAgICAgICApLFxuICAgICAgfTtcbiAgICB9KTtcbiAgfVxuXG4gIHN1YnNldFJvd3NUb1N1YnNldEZpZWxkcyhzdWJzZXRSb3dzOiBFbnRpdHlTdWJzZXRSb3dbXSwgc3Vic2V0S2V5OiBzdHJpbmcpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIHN1YnNldFJvd3NcbiAgICAgIC5tYXAoKHN1YnNldFJvdykgPT4ge1xuICAgICAgICBpZiAoc3Vic2V0Um93LmNoaWxkcmVuLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5zdWJzZXRSb3dzVG9TdWJzZXRGaWVsZHMoc3Vic2V0Um93LmNoaWxkcmVuLCBzdWJzZXRLZXkpO1xuICAgICAgICB9IGVsc2UgaWYgKHN1YnNldFJvdy5oYXNbc3Vic2V0S2V5XSkge1xuICAgICAgICAgIHJldHVybiBzdWJzZXRSb3cucHJlZml4ZXMuY29uY2F0KHN1YnNldFJvdy5maWVsZCkuam9pbihcIi5cIik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAuZmlsdGVyKG5vbk51bGxhYmxlKVxuICAgICAgLmZsYXQoKTtcbiAgfVxuXG4gIGFzeW5jIGNyZWF0ZVByb3AocHJvcDogRW50aXR5UHJvcCwgYXQ/OiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZiAoIWF0KSB7XG4gICAgICB0aGlzLnByb3BzLnB1c2gocHJvcCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMucHJvcHMuc3BsaWNlKGF0LCAwLCBwcm9wKTtcbiAgICB9XG4gICAgYXdhaXQgdGhpcy5zYXZlKCk7XG4gIH1cblxuICBhbmFseXplU3Vic2V0RmllbGQoc3Vic2V0RmllbGQ6IHN0cmluZyk6IHtcbiAgICBlbnRpdHlJZDogc3RyaW5nO1xuICAgIHByb3BOYW1lOiBzdHJpbmc7XG4gIH1bXSB7XG4gICAgY29uc3QgYXJyID0gc3Vic2V0RmllbGQuc3BsaXQoXCIuXCIpO1xuXG4gICAgbGV0IGVudGl0eUlkID0gdGhpcy5pZDtcbiAgICBjb25zdCByZXN1bHQ6IHtcbiAgICAgIGVudGl0eUlkOiBzdHJpbmc7XG4gICAgICBwcm9wTmFtZTogc3RyaW5nO1xuICAgIH1bXSA9IFtdO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYXJyLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBwcm9wTmFtZSA9IGFycltpXTtcbiAgICAgIHJlc3VsdC5wdXNoKHtcbiAgICAgICAgZW50aXR5SWQsXG4gICAgICAgIHByb3BOYW1lLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHByb3AgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCkucHJvcHMuZmluZCgocCkgPT4gcC5uYW1lID09PSBwcm9wTmFtZSk7XG4gICAgICBpZiAoIXByb3ApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke2VudGl0eUlkfeydmCDsnpjrqrvrkJwg7ISc67iM7IWL7YKkICR7c3Vic2V0RmllbGR9YCk7XG4gICAgICB9XG4gICAgICBpZiAoaXNSZWxhdGlvblByb3AocHJvcCkpIHtcbiAgICAgICAgZW50aXR5SWQgPSBwcm9wLndpdGg7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyBtb2RpZnlQcm9wKG5ld1Byb3A6IEVudGl0eVByb3AsIGF0OiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyDsnbTsoIQg7ZSE66GtIOydtOumhCDsoIDsnqVcbiAgICBjb25zdCBvbGROYW1lID0gdGhpcy5wcm9wc1thdF0ubmFtZTtcblxuICAgIC8vIOyggOyepe2VoCDsl5Tti7Dti7BcbiAgICBjb25zdCBlbnRpdGllczogRW50aXR5W10gPSBbdGhpc107XG5cbiAgICAvLyDsnbTrpoTsnbQg67CU64CQIOqyveyasFxuICAgIGlmIChvbGROYW1lICE9PSBuZXdQcm9wLm5hbWUpIHtcbiAgICAgIC8vIOyghOyytCDsl5Tti7Dti7Dsl5DshJwg7ZiE7J6sIOyImOygleuQnCDtlITroa3snYQg7LC47KGw7ZWY6rOgIOyeiOuKlCDrqqjrk6Ag7ISc67iM7IWL7ZWE65OcIOywvuyVhOyEnCDsiJjsoJVcbiAgICAgIGNvbnN0IGFsbEVudGl0eUlkcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsSWRzKCk7XG4gICAgICBmb3IgKGNvbnN0IHJlbEVudGl0eUlkIG9mIGFsbEVudGl0eUlkcykge1xuICAgICAgICBjb25zdCByZWxFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChyZWxFbnRpdHlJZCk7XG4gICAgICAgIGNvbnN0IHJlbEVudGl0eVN1YnNldEtleXMgPSBPYmplY3Qua2V5cyhyZWxFbnRpdHkuc3Vic2V0cyk7XG4gICAgICAgIGZvciAoY29uc3Qgc3Vic2V0S2V5IG9mIHJlbEVudGl0eVN1YnNldEtleXMpIHtcbiAgICAgICAgICBjb25zdCBzdWJzZXQgPSByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldO1xuXG4gICAgICAgICAgLy8g7ISc67iM7IWLIO2VhOuTnOulvCDsiJztmoztlZjrqbAsIOyXlO2LsO2LsC3tlITroa0g64uo7JyE66GcIOu2hOyEne2VnCDtm4Qg7ZiE7J6sIOyXlO2LsO2LsC3tlITroa3qs7wg7J287LmY7ZWY64qUIOqyveyasCDsiJjsoJUg7LKY66asXG4gICAgICAgICAgY29uc3QgbW9kaWZpZWRTdWJzZXRGaWVsZHMgPSBzdWJzZXQubWFwKChzdWJzZXRGaWVsZCkgPT4ge1xuICAgICAgICAgICAgY29uc3QgYW5hbHl6ZWQgPSByZWxFbnRpdHkuYW5hbHl6ZVN1YnNldEZpZWxkKHN1YnNldEZpZWxkKTtcbiAgICAgICAgICAgIGNvbnN0IG1vZGlmaWVkID0gYW5hbHl6ZWQubWFwKChhKSA9PlxuICAgICAgICAgICAgICBhLnByb3BOYW1lID09PSBvbGROYW1lICYmIGEuZW50aXR5SWQgPT09IHRoaXMuaWRcbiAgICAgICAgICAgICAgICA/IHtcbiAgICAgICAgICAgICAgICAgICAgLi4uYSxcbiAgICAgICAgICAgICAgICAgICAgcHJvcE5hbWU6IG5ld1Byb3AubmFtZSxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICA6IGEsXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgLy8g67aE7ISd7ZWcIO2VhOuTnOulvCDri6Tsi5wg7ISc67iM7IWLIO2VhOuTnOuhnCDrs7XqtaxcbiAgICAgICAgICAgIHJldHVybiBtb2RpZmllZC5tYXAoKGEpID0+IGEucHJvcE5hbWUpLmpvaW4oXCIuXCIpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaWYgKHN1YnNldC5qb2luKFwiLFwiKSAhPT0gbW9kaWZpZWRTdWJzZXRGaWVsZHMuam9pbihcIixcIikpIHtcbiAgICAgICAgICAgIHJlbEVudGl0eS5zdWJzZXRzW3N1YnNldEtleV0gPSBtb2RpZmllZFN1YnNldEZpZWxkcztcbiAgICAgICAgICAgIGVudGl0aWVzLnB1c2gocmVsRW50aXR5KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyDtlITroa0g7IiY7KCVXG4gICAgdGhpcy5wcm9wc1thdF0gPSBuZXdQcm9wO1xuXG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoZW50aXRpZXMubWFwKGFzeW5jIChlbnRpdHkpID0+IGVudGl0eS5zYXZlKCkpKTtcbiAgfVxuXG4gIGFzeW5jIGRlbFByb3AoYXQ6IG51bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIOydtOyghCDtlITroa0g7J2066aEIOyggOyepVxuICAgIGNvbnN0IG9sZE5hbWUgPSB0aGlzLnByb3BzW2F0XS5uYW1lO1xuXG4gICAgLy8g7KCA7J6l7ZWgIOyXlO2LsO2LsFxuICAgIGNvbnN0IGVudGl0aWVzOiBFbnRpdHlbXSA9IFt0aGlzXTtcblxuICAgIC8vIOyghOyytCDsl5Tti7Dti7Dsl5DshJwg7ZiE7J6sIOyCreygnOuQnCDtlITroa3snYQg7LC47KGw7ZWY6rOgIOyeiOuKlCDrqqjrk6Ag7ISc67iM7IWL7ZWE65OcIOywvuyVhOyEnCDsoJzsmbhcbiAgICBjb25zdCBhbGxFbnRpdHlJZHMgPSBFbnRpdHlNYW5hZ2VyLmdldEFsbElkcygpO1xuICAgIGZvciAoY29uc3QgcmVsRW50aXR5SWQgb2YgYWxsRW50aXR5SWRzKSB7XG4gICAgICBjb25zdCByZWxFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChyZWxFbnRpdHlJZCk7XG4gICAgICBjb25zdCByZWxFbnRpdHlTdWJzZXRLZXlzID0gT2JqZWN0LmtleXMocmVsRW50aXR5LnN1YnNldHMpO1xuICAgICAgZm9yIChjb25zdCBzdWJzZXRLZXkgb2YgcmVsRW50aXR5U3Vic2V0S2V5cykge1xuICAgICAgICBjb25zdCBzdWJzZXQgPSByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldO1xuICAgICAgICAvLyDshJzruIzshYsg7ZWE65Oc66W8IOyInO2ajO2VmOupsCwg7JeU7Yuw7YuwLe2UhOuhrSDri6jsnITroZwg67aE7ISd7ZWcIO2bhCDtmITsnqwg7JeU7Yuw7YuwLe2UhOuhreqzvCDsnbzsuZjtlZjripQg6rK97JqwIOydtO2bhOydmCDtlYTrk5zrpbwg7KCc7Jm4XG4gICAgICAgIGNvbnN0IG1vZGlmaWVkU3Vic2V0RmllbGRzID0gc3Vic2V0XG4gICAgICAgICAgLm1hcCgoc3Vic2V0RmllbGQpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGFuYWx5emVkID0gcmVsRW50aXR5LmFuYWx5emVTdWJzZXRGaWVsZChzdWJzZXRGaWVsZCk7XG4gICAgICAgICAgICBpZiAoYW5hbHl6ZWQuZmluZCgoYSkgPT4gYS5wcm9wTmFtZSA9PT0gb2xkTmFtZSAmJiBhLmVudGl0eUlkID09PSB0aGlzLmlkKSkge1xuICAgICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHJldHVybiBzdWJzZXRGaWVsZDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KVxuICAgICAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgICAgIGlmIChzdWJzZXQuam9pbihcIixcIikgIT09IG1vZGlmaWVkU3Vic2V0RmllbGRzLmpvaW4oXCIsXCIpKSB7XG4gICAgICAgICAgcmVsRW50aXR5LnN1YnNldHNbc3Vic2V0S2V5XSA9IG1vZGlmaWVkU3Vic2V0RmllbGRzO1xuICAgICAgICAgIGVudGl0aWVzLnB1c2gocmVsRW50aXR5KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIO2YhOyerCDsl5Tti7Dti7DsnZgg7J24642x7Iqk7JeQ7IScIOygnOyZuFxuICAgIGZvciAoY29uc3QgaW5kZXggb2YgRW50aXR5TWFuYWdlci5nZXQodGhpcy5pZCkuaW5kZXhlcykge1xuICAgICAgaW5kZXguY29sdW1ucyA9IGluZGV4LmNvbHVtbnMuZmlsdGVyKChjb2wpID0+IGNvbCAhPT0gb2xkTmFtZSk7XG4gICAgfVxuXG4gICAgLy8g7ZSE66GtIOyCreygnFxuICAgIHRoaXMucHJvcHMuc3BsaWNlKGF0LCAxKTtcblxuICAgIGF3YWl0IFByb21pc2UuYWxsKGVudGl0aWVzLm1hcChhc3luYyAoZW50aXR5KSA9PiBlbnRpdHkuc2F2ZSgpKSk7XG4gIH1cblxuICBnZXRFbnRpdHlJZEZyb21TdWJzZXRGaWVsZChzdWJzZXRGaWVsZDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBpZiAoc3Vic2V0RmllbGQuaW5jbHVkZXMoXCIuXCIpID09PSBmYWxzZSkge1xuICAgICAgcmV0dXJuIHRoaXMuaWQ7XG4gICAgfVxuXG4gICAgLy8g7ISc67iM7IWLIO2VhOuTnOydmCDrp4jsp4Drp4nsnYAg7ZSE66Gt7J2066+A66GcIOygnOyZuFxuICAgIGNvbnN0IGFyciA9IHN1YnNldEZpZWxkLnNwbGl0KFwiLlwiKS5zbGljZSgwLCAtMSk7XG5cbiAgICAvLyDshJzruIzshYsg7ZWE65Oc66W8IOuCtOugpOqwgOuptOyEnCDrp4jsp4Drp4nsnLzroZwgcmVsYXRpb27rkJwg7JeU7Yuw7Yuw66W8IOywvuydjFxuICAgIGNvbnN0IGxhc3RFbnRpdHlJZCA9IGFyci5yZWR1Y2UoKGVudGl0eUlkLCBmaWVsZCkgPT4ge1xuICAgICAgY29uc3QgcmVsUHJvcCA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IGZpZWxkKTtcbiAgICAgIGlmICghcmVsUHJvcCB8fCByZWxQcm9wLnR5cGUgIT09IFwicmVsYXRpb25cIikge1xuICAgICAgICBjb25zb2xlLmRlYnVnKHsgYXJyLCB0aGlzSWQ6IHRoaXMuaWQsIGVudGl0eUlkLCBmaWVsZCB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsnpjrqrvrkJwg7ISc67iM7IWL7YKkICR7c3Vic2V0RmllbGR9YCk7XG4gICAgICB9XG4gICAgICByZXR1cm4gcmVsUHJvcC53aXRoO1xuICAgIH0sIHRoaXMuaWQpO1xuICAgIHJldHVybiBsYXN0RW50aXR5SWQ7XG4gIH1cblxuICBhc3luYyBtb3ZlUHJvcChhdDogbnVtYmVyLCB0bzogbnVtYmVyKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgcHJvcCA9IHRoaXMucHJvcHNbYXRdO1xuICAgIGNvbnN0IG5ld1Byb3BzID0gWy4uLnRoaXMucHJvcHNdO1xuICAgIG5ld1Byb3BzLnNwbGljZSh0bywgMCwgcHJvcCk7XG4gICAgbmV3UHJvcHMuc3BsaWNlKGF0IDwgdG8gPyBhdCA6IGF0ICsgMSwgMSk7XG4gICAgdGhpcy5wcm9wcyA9IG5ld1Byb3BzO1xuXG4gICAgYXdhaXQgdGhpcy5zYXZlKCk7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJhc3NlcnQiLCJ3cml0ZUZpbGUiLCJpbmZsZWN0aW9uIiwicGF0aCIsImdyb3VwIiwidW5pcXVlIiwieiIsIlNvbmFtdSIsImlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIiwiaXNFbnVtUHJvcCIsImlzSGFzTWFueVJlbGF0aW9uUHJvcCIsImlzTWFueVRvTWFueVJlbGF0aW9uUHJvcCIsImlzT25lVG9PbmVSZWxhdGlvblByb3AiLCJpc1JlbGF0aW9uUHJvcCIsImlzVmlydHVhbFByb3AiLCJpbXBvcnRNZW1iZXJzIiwiZm9ybWF0Q29kZSIsImV4aXN0cyIsInJ1bnRpbWVQYXRoIiwiYXNzZXJ0RGVmaW5lZCIsIm5vbk51bGxhYmxlIiwiRW50aXR5TWFuYWdlciIsIkVudGl0eSIsImlkIiwicGFyZW50SWQiLCJ0YWJsZSIsInRpdGxlIiwibmFtZXMiLCJwcm9wcyIsInByb3BzRGljdCIsInJlbGF0aW9ucyIsImluZGV4ZXMiLCJzdWJzZXRzIiwidHlwZXMiLCJlbnVtcyIsImVudW1MYWJlbHMiLCJ1bmRlcnNjb3JlIiwicGx1cmFsaXplIiwibWFwIiwicHJvcCIsImluY2x1ZGVzIiwicmVwbGFjZSIsIk9iamVjdCIsImZyb21FbnRyaWVzIiwibmFtZSIsImZpbHRlciIsImVudHJpZXMiLCJrZXkiLCJlbnVtTGFiZWwiLCJlbnVtIiwia2V5cyIsInBhcmVudEZzIiwiZGFzaGVyaXplIiwidG9Mb3dlckNhc2UiLCJmcyIsIm1vZHVsZSIsImdldFB1cmlTdWJzZXRRdWVyeSIsInN1YnNldEtleSIsInN1YnNldCIsInN1YnNldFF1ZXJ5IiwicmVzb2x2ZVN1YnNldFF1ZXJ5IiwibGluZXMiLCJwdXNoIiwiam9pbiIsImpvaW5zIiwiam9pbk1ldGhvZCIsImFzIiwiY3VzdG9tIiwiZnJvbSIsInRvIiwic2VsZWN0T2JqIiwic2VsZWN0SXRlbSIsInNlbGVjdCIsIm1hdGNoIiwiY29sdW1uIiwiYWxpYXMiLCJzcGxpdCIsInBvcCIsInRyaW0iLCJmb3JFYWNoIiwidmFsdWUiLCJnZXRQdXJpTG9hZGVyUXVlcnkiLCJsb2FkZXJzIiwicGFyc2VTZWxlY3QiLCJ0YWJsZVByZWZpeCIsInN0YXJ0c1dpdGgiLCJnZW5lcmF0ZUxvYWRlckNvZGUiLCJsb2FkZXJMaW5lcyIsImxvYWRlciIsInRvVGFibGUiLCJ0b0NvbCIsInRocm91Z2giLCJtYW55Sm9pbiIsImlkRmllbGQiLCJ1bmRlZmluZWQiLCJvbmVKb2lucyIsImpvaW5UeXBlIiwiZnJvbUNvbCIsImxlbmd0aCIsImdldFN1YnNldFF1ZXJ5IiwicmVzdWx0IiwicHJlZml4IiwiZmllbGRzIiwiaXNBbHJlYWR5T3V0ZXJKb2luZWQiLCJzdWJzZXRHcm91cCIsImZpZWxkIiwicmVsIiwicmVkdWNlIiwiciIsImdyb3VwS2V5IiwicmVhbEZpZWxkcyIsInZpcnR1YWxGaWVsZHMiLCJjb25jYXQiLCJ2aXJ0dWFsIiwicmVsYXRpb24iLCJFcnJvciIsInJlbEVudGl0eSIsImdldCIsIndpdGgiLCJyZWxGaWVsZHMiLCJzbGljZSIsImlubmVyT3JPdXRlciIsImhhc0pvaW5Db2x1bW4iLCJudWxsYWJsZSIsInJlbFN1YnNldFF1ZXJ5Iiwiam9pbkFzIiwiZnJvbVRhYmxlIiwiam9pbkNsYXVzZSIsImN1c3RvbUpvaW5DbGF1c2UiLCJjb252ZXJ0ZWRMb2FkZXJzIiwibmV3QXMiLCJmcm9tQ29sdW1uIiwiam9pbkNvbHVtbiIsImpvaW5UYWJsZSIsInNpbmd1bGFyaXplIiwiZmllbGRFeHByc1RvUHJvcE5vZGVzIiwiZmllbGRFeHBycyIsImVudGl0eSIsImdyb3VwcyIsImZpZWxkRXhwciIsImVsc2VFeHByIiwiZmxhdE1hcCIsInByb3BOYW1lIiwibm9kZVR5cGUiLCJ0eXBlIiwiY2hpbGRyZW4iLCJmaW5kIiwicCIsImNvbnNvbGUiLCJsb2ciLCJpZFByb3AiLCJnZXRGaWVsZEV4cHJzIiwibWF4RGVwdGgiLCJmcm9tcyIsInYiLCJyZWxNZCIsImYiLCJnZXRUYWJsZUNvbHVtbnMiLCJyZWxhdGlvblR5cGUiLCJyZWdpc3Rlck1vZHVsZVBhdGhzIiwiYmFzZVBhdGgiLCJzZXRNb2R1bGVQYXRoIiwidG9VcHBlckNhc2UiLCJlbnVtSWQiLCJ0eXBlc01vZHVsZVBhdGgiLCJ0eXBlc0ZpbGVQYXRoIiwiYXBpUm9vdFBhdGgiLCJpbXBvcnRlZE1lbWJlcnMiLCJyZWdpc3RlclRhYmxlU3BlY3MiLCJ1bmlxdWVJbmRleGVzIiwiaWR4Iiwic2V0VGFibGVTcGVjIiwidG9Kc29uIiwic2F2ZSIsInN1YnNldFJvd3MiLCJnZXRTdWJzZXRSb3dzIiwic3Vic2V0Um93c1RvU3Vic2V0RmllbGRzIiwianNvblBhdGgiLCJqc29uIiwiSlNPTiIsInN0cmluZ2lmeSIsInJlZ2lzdGVyIiwiX3N1YnNldHMiLCJwcmVmaXhlcyIsInN1YnNldEtleXMiLCJhbGxGaWVsZHMiLCJyZWxhdGlvbkVudGl0eSIsImlzT3BlbiIsImhhcyIsImV2ZXJ5IiwiY2hpbGQiLCJzdWJzZXRGaWVsZHMiLCJzb21lIiwic3Vic2V0Um93IiwiZmxhdCIsImNyZWF0ZVByb3AiLCJhdCIsInNwbGljZSIsImFuYWx5emVTdWJzZXRGaWVsZCIsInN1YnNldEZpZWxkIiwiYXJyIiwiZW50aXR5SWQiLCJpIiwibW9kaWZ5UHJvcCIsIm5ld1Byb3AiLCJvbGROYW1lIiwiZW50aXRpZXMiLCJhbGxFbnRpdHlJZHMiLCJnZXRBbGxJZHMiLCJyZWxFbnRpdHlJZCIsInJlbEVudGl0eVN1YnNldEtleXMiLCJtb2RpZmllZFN1YnNldEZpZWxkcyIsImFuYWx5emVkIiwibW9kaWZpZWQiLCJhIiwiUHJvbWlzZSIsImFsbCIsImRlbFByb3AiLCJpbmRleCIsImNvbHVtbnMiLCJjb2wiLCJnZXRFbnRpdHlJZEZyb21TdWJzZXRGaWVsZCIsImxhc3RFbnRpdHlJZCIsInJlbFByb3AiLCJkZWJ1ZyIsInRoaXNJZCIsIm1vdmVQcm9wIiwibmV3UHJvcHMiXSwibWFwcGluZ3MiOiJBQUFBLE9BQU9BLFlBQVksU0FBUztBQUM1QixTQUFTQyxTQUFTLFFBQVEsbUJBQWM7QUFDeEMsT0FBT0MsZ0JBQWdCLGFBQWE7QUFDcEMsT0FBT0MsVUFBVSxPQUFPO0FBQ3hCLFNBQVNDLEtBQUssRUFBRUMsTUFBTSxRQUFRLFVBQVU7QUFDeEMsU0FBU0MsQ0FBQyxRQUFRLE1BQU07QUFDeEIsU0FBU0MsTUFBTSxRQUFRLG1CQUFnQjtBQUN2QyxTQU1FQywwQkFBMEIsRUFDMUJDLFVBQVUsRUFDVkMscUJBQXFCLEVBQ3JCQyx3QkFBd0IsRUFDeEJDLHNCQUFzQixFQUN0QkMsY0FBYyxFQUNkQyxhQUFhLFFBSVIsb0JBQWlCO0FBQ3hCLFNBQVNDLGFBQWEsUUFBUSx3QkFBcUI7QUFDbkQsU0FBU0MsVUFBVSxRQUFRLHdCQUFxQjtBQUNoRCxTQUFTQyxNQUFNLFFBQVEsdUJBQW9CO0FBQzNDLFNBQVNDLFdBQVcsUUFBUSx5QkFBc0I7QUFDbEQsU0FBU0MsYUFBYSxFQUFFQyxXQUFXLFFBQVEsb0JBQWlCO0FBQzVELFNBQVNDLGFBQWEsUUFBUSxzQkFBbUI7QUFFakQsT0FBTyxNQUFNQztJQUNYQyxHQUFXO0lBQ1hDLFNBQWtCO0lBQ2xCQyxNQUFjO0lBQ2RDLE1BQWM7SUFDZEMsTUFJRTtJQUNGQyxNQUFvQjtJQUNwQkMsVUFFRTtJQUNGQyxVQUVFO0lBQ0ZDLFFBQXVCO0lBQ3ZCQyxRQUVFO0lBQ0ZDLFFBRUksQ0FBQyxFQUFFO0lBQ1BDLFFBRUksQ0FBQyxFQUFFO0lBQ1BDLGFBSUksQ0FBQyxFQUFFO0lBRVAsWUFBWSxFQUFFWixFQUFFLEVBQUVDLFFBQVEsRUFBRUMsS0FBSyxFQUFFQyxLQUFLLEVBQUVFLEtBQUssRUFBRUcsT0FBTyxFQUFFQyxP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