sonamu 0.8.26 → 0.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (684) hide show
  1. package/bin/cli.js +60 -13
  2. package/dist/_virtual/rolldown_runtime.js +39 -0
  3. package/dist/ai/agents/agent.d.ts +3 -3
  4. package/dist/ai/agents/agent.d.ts.map +1 -1
  5. package/dist/ai/agents/agent.js +76 -73
  6. package/dist/ai/agents/index.js +3 -3
  7. package/dist/ai/agents/types.d.ts +3 -3
  8. package/dist/ai/agents/types.d.ts.map +1 -1
  9. package/dist/ai/agents/types.js +1 -3
  10. package/dist/ai/index.js +3 -2
  11. package/dist/ai/providers/rtzr/api.js +25 -25
  12. package/dist/ai/providers/rtzr/error.js +25 -26
  13. package/dist/ai/providers/rtzr/index.js +5 -5
  14. package/dist/ai/providers/rtzr/model.d.ts +1 -1
  15. package/dist/ai/providers/rtzr/model.d.ts.map +1 -1
  16. package/dist/ai/providers/rtzr/model.js +117 -133
  17. package/dist/ai/providers/rtzr/options.d.ts.map +1 -1
  18. package/dist/ai/providers/rtzr/options.js +35 -41
  19. package/dist/ai/providers/rtzr/provider.d.ts +1 -1
  20. package/dist/ai/providers/rtzr/provider.d.ts.map +1 -1
  21. package/dist/ai/providers/rtzr/provider.js +53 -51
  22. package/dist/ai/providers/rtzr/utils.d.ts.map +1 -1
  23. package/dist/ai/providers/rtzr/utils.js +84 -84
  24. package/dist/api/base-frame.d.ts +2 -2
  25. package/dist/api/base-frame.d.ts.map +1 -1
  26. package/dist/api/base-frame.js +29 -19
  27. package/dist/api/caster.d.ts +1 -1
  28. package/dist/api/caster.d.ts.map +1 -1
  29. package/dist/api/caster.js +51 -61
  30. package/dist/api/code-converters.d.ts +4 -3
  31. package/dist/api/code-converters.d.ts.map +1 -1
  32. package/dist/api/code-converters.js +226 -249
  33. package/dist/api/config.d.ts +17 -17
  34. package/dist/api/config.d.ts.map +1 -1
  35. package/dist/api/config.js +37 -30
  36. package/dist/api/context.d.ts +10 -10
  37. package/dist/api/context.d.ts.map +1 -1
  38. package/dist/api/context.js +8 -2
  39. package/dist/api/decorators.d.ts +8 -8
  40. package/dist/api/decorators.d.ts.map +1 -1
  41. package/dist/api/decorators.js +245 -268
  42. package/dist/api/index.js +39 -7
  43. package/dist/api/secret.js +22 -15
  44. package/dist/api/sonamu.d.ts +15 -15
  45. package/dist/api/sonamu.d.ts.map +1 -1
  46. package/dist/api/sonamu.js +1012 -1131
  47. package/dist/api/validator.js +88 -79
  48. package/dist/auth/auth-generator.d.ts.map +1 -1
  49. package/dist/auth/auth-generator.js +203 -200
  50. package/dist/auth/better-auth-entities.d.ts +2 -2
  51. package/dist/auth/better-auth-entities.d.ts.map +1 -1
  52. package/dist/auth/better-auth-entities.js +369 -429
  53. package/dist/auth/index.js +21 -6
  54. package/dist/auth/knex-adapter.d.ts +2 -2
  55. package/dist/auth/knex-adapter.d.ts.map +1 -1
  56. package/dist/auth/knex-adapter.js +153 -157
  57. package/dist/auth/plugins/entity-definitions/admin.d.ts +1 -1
  58. package/dist/auth/plugins/entity-definitions/admin.d.ts.map +1 -1
  59. package/dist/auth/plugins/entity-definitions/admin.js +58 -56
  60. package/dist/auth/plugins/entity-definitions/anonymous.d.ts +1 -1
  61. package/dist/auth/plugins/entity-definitions/anonymous.d.ts.map +1 -1
  62. package/dist/auth/plugins/entity-definitions/anonymous.js +20 -20
  63. package/dist/auth/plugins/entity-definitions/api-key.d.ts +1 -1
  64. package/dist/auth/plugins/entity-definitions/api-key.d.ts.map +1 -1
  65. package/dist/auth/plugins/entity-definitions/api-key.js +185 -196
  66. package/dist/auth/plugins/entity-definitions/index.d.ts +1 -1
  67. package/dist/auth/plugins/entity-definitions/index.d.ts.map +1 -1
  68. package/dist/auth/plugins/entity-definitions/index.js +26 -29
  69. package/dist/auth/plugins/entity-definitions/jwt.d.ts +1 -1
  70. package/dist/auth/plugins/entity-definitions/jwt.d.ts.map +1 -1
  71. package/dist/auth/plugins/entity-definitions/jwt.js +62 -64
  72. package/dist/auth/plugins/entity-definitions/organization.d.ts +1 -1
  73. package/dist/auth/plugins/entity-definitions/organization.d.ts.map +1 -1
  74. package/dist/auth/plugins/entity-definitions/organization.js +362 -421
  75. package/dist/auth/plugins/entity-definitions/passkey.d.ts +1 -1
  76. package/dist/auth/plugins/entity-definitions/passkey.d.ts.map +1 -1
  77. package/dist/auth/plugins/entity-definitions/passkey.js +115 -126
  78. package/dist/auth/plugins/entity-definitions/phone-number.d.ts +1 -1
  79. package/dist/auth/plugins/entity-definitions/phone-number.d.ts.map +1 -1
  80. package/dist/auth/plugins/entity-definitions/phone-number.js +31 -40
  81. package/dist/auth/plugins/entity-definitions/sso.d.ts +1 -1
  82. package/dist/auth/plugins/entity-definitions/sso.d.ts.map +1 -1
  83. package/dist/auth/plugins/entity-definitions/sso.js +94 -107
  84. package/dist/auth/plugins/entity-definitions/two-factor.d.ts +1 -1
  85. package/dist/auth/plugins/entity-definitions/two-factor.d.ts.map +1 -1
  86. package/dist/auth/plugins/entity-definitions/two-factor.js +78 -92
  87. package/dist/auth/plugins/entity-definitions/types.d.ts +1 -1
  88. package/dist/auth/plugins/entity-definitions/types.d.ts.map +1 -1
  89. package/dist/auth/plugins/entity-definitions/types.js +1 -10
  90. package/dist/auth/plugins/entity-definitions/username.d.ts +1 -1
  91. package/dist/auth/plugins/entity-definitions/username.d.ts.map +1 -1
  92. package/dist/auth/plugins/entity-definitions/username.js +31 -40
  93. package/dist/auth/plugins/index.js +12 -3
  94. package/dist/auth/plugins/wrappers/admin.d.ts +2 -2
  95. package/dist/auth/plugins/wrappers/admin.d.ts.map +1 -1
  96. package/dist/auth/plugins/wrappers/admin.js +28 -29
  97. package/dist/auth/plugins/wrappers/anonymous.d.ts +2 -1
  98. package/dist/auth/plugins/wrappers/anonymous.d.ts.map +1 -1
  99. package/dist/auth/plugins/wrappers/anonymous.js +23 -22
  100. package/dist/auth/plugins/wrappers/api-key.d.ts +2 -1
  101. package/dist/auth/plugins/wrappers/api-key.d.ts.map +1 -1
  102. package/dist/auth/plugins/wrappers/api-key.js +39 -34
  103. package/dist/auth/plugins/wrappers/index.js +11 -11
  104. package/dist/auth/plugins/wrappers/jwt.d.ts +2 -1
  105. package/dist/auth/plugins/wrappers/jwt.d.ts.map +1 -1
  106. package/dist/auth/plugins/wrappers/jwt.js +31 -26
  107. package/dist/auth/plugins/wrappers/organization.d.ts +2 -1
  108. package/dist/auth/plugins/wrappers/organization.d.ts.map +1 -1
  109. package/dist/auth/plugins/wrappers/organization.js +65 -62
  110. package/dist/auth/plugins/wrappers/passkey.d.ts +2 -1
  111. package/dist/auth/plugins/wrappers/passkey.d.ts.map +1 -1
  112. package/dist/auth/plugins/wrappers/passkey.js +33 -28
  113. package/dist/auth/plugins/wrappers/phone-number.d.ts.map +1 -1
  114. package/dist/auth/plugins/wrappers/phone-number.js +26 -23
  115. package/dist/auth/plugins/wrappers/sso.d.ts.map +1 -1
  116. package/dist/auth/plugins/wrappers/sso.js +37 -31
  117. package/dist/auth/plugins/wrappers/two-factor.d.ts.map +1 -1
  118. package/dist/auth/plugins/wrappers/two-factor.js +31 -28
  119. package/dist/auth/plugins/wrappers/username.d.ts.map +1 -1
  120. package/dist/auth/plugins/wrappers/username.js +23 -23
  121. package/dist/bin/build-config.js +31 -31
  122. package/dist/bin/cli.js +1063 -1204
  123. package/dist/bin/fixture.d.ts.map +1 -1
  124. package/dist/bin/fixture.js +266 -259
  125. package/dist/bin/hmr-hook-register.d.ts.map +1 -1
  126. package/dist/bin/hmr-hook-register.js +19 -18
  127. package/dist/bin/test-command.d.ts.map +1 -1
  128. package/dist/bin/test-command.js +180 -177
  129. package/dist/bin/ts-loader-register.js +13 -6
  130. package/dist/bin/ts-loader-registration.d.ts.map +1 -1
  131. package/dist/bin/ts-loader-registration.js +28 -38
  132. package/dist/cache/cache-manager.d.ts +1 -1
  133. package/dist/cache/cache-manager.d.ts.map +1 -1
  134. package/dist/cache/cache-manager.js +20 -15
  135. package/dist/cache/decorator.d.ts +1 -1
  136. package/dist/cache/decorator.d.ts.map +1 -1
  137. package/dist/cache/decorator.js +84 -76
  138. package/dist/cache/drivers.js +21 -34
  139. package/dist/cache/index.js +10 -7
  140. package/dist/cache/types.d.ts +2 -2
  141. package/dist/cache/types.d.ts.map +1 -1
  142. package/dist/cache/types.js +1 -6
  143. package/dist/cache-control/cache-control.d.ts +2 -2
  144. package/dist/cache-control/cache-control.d.ts.map +1 -1
  145. package/dist/cache-control/cache-control.js +106 -122
  146. package/dist/cache-control/types.d.ts +2 -2
  147. package/dist/cache-control/types.d.ts.map +1 -1
  148. package/dist/cache-control/types.js +1 -19
  149. package/dist/compress/compress.d.ts +1 -1
  150. package/dist/compress/compress.d.ts.map +1 -1
  151. package/dist/compress/compress.js +58 -56
  152. package/dist/compress/index.js +7 -2
  153. package/dist/compress/types.js +1 -11
  154. package/dist/cone/cone-generator.d.ts +1 -1
  155. package/dist/cone/cone-generator.d.ts.map +1 -1
  156. package/dist/cone/cone-generator.js +216 -219
  157. package/dist/database/_batch_update.d.ts +1 -1
  158. package/dist/database/_batch_update.d.ts.map +1 -1
  159. package/dist/database/_batch_update.js +107 -102
  160. package/dist/database/base-model.d.ts +8 -9
  161. package/dist/database/base-model.d.ts.map +1 -1
  162. package/dist/database/base-model.js +371 -392
  163. package/dist/database/base-model.types.d.ts +5 -5
  164. package/dist/database/base-model.types.d.ts.map +1 -1
  165. package/dist/database/base-model.types.js +1 -20
  166. package/dist/database/db.d.ts +5 -2
  167. package/dist/database/db.d.ts.map +1 -1
  168. package/dist/database/db.js +185 -171
  169. package/dist/database/knex.d.ts +1 -1
  170. package/dist/database/knex.d.ts.map +1 -1
  171. package/dist/database/knex.js +48 -42
  172. package/dist/database/puri-subset.types.d.ts +6 -7
  173. package/dist/database/puri-subset.types.d.ts.map +1 -1
  174. package/dist/database/puri-subset.types.js +1 -16
  175. package/dist/database/puri-wrapper.d.ts +6 -6
  176. package/dist/database/puri-wrapper.d.ts.map +1 -1
  177. package/dist/database/puri-wrapper.js +99 -101
  178. package/dist/database/puri.d.ts +4 -5
  179. package/dist/database/puri.d.ts.map +1 -1
  180. package/dist/database/puri.js +1021 -1227
  181. package/dist/database/puri.types.d.ts +6 -6
  182. package/dist/database/puri.types.d.ts.map +1 -1
  183. package/dist/database/puri.types.js +15 -6
  184. package/dist/database/transaction-context.d.ts +2 -2
  185. package/dist/database/transaction-context.d.ts.map +1 -1
  186. package/dist/database/transaction-context.js +22 -13
  187. package/dist/database/upsert-builder.d.ts +3 -3
  188. package/dist/database/upsert-builder.d.ts.map +1 -1
  189. package/dist/database/upsert-builder.js +405 -465
  190. package/dist/dict/en.js +72 -74
  191. package/dist/dict/index.js +13 -13
  192. package/dist/dict/ko.js +72 -74
  193. package/dist/dict/rc-keys.js +150 -168
  194. package/dist/dict/sd.d.ts +3 -1
  195. package/dist/dict/sd.d.ts.map +1 -1
  196. package/dist/dict/sd.js +54 -40
  197. package/dist/dict/sonamu-dictionary.d.ts +1 -1
  198. package/dist/dict/sonamu-dictionary.d.ts.map +1 -1
  199. package/dist/dict/sonamu-dictionary.js +887 -955
  200. package/dist/dict/types.js +1 -7
  201. package/dist/dict/utils.js +26 -24
  202. package/dist/entity/entity-manager.d.ts +9 -9
  203. package/dist/entity/entity-manager.d.ts.map +1 -1
  204. package/dist/entity/entity-manager.js +226 -223
  205. package/dist/entity/entity-template-cone.d.ts +1 -1
  206. package/dist/entity/entity-template-cone.d.ts.map +1 -1
  207. package/dist/entity/entity-template-cone.js +152 -151
  208. package/dist/entity/entity.d.ts.map +1 -1
  209. package/dist/entity/entity.js +952 -1089
  210. package/dist/exceptions/error-handler.d.ts +1 -1
  211. package/dist/exceptions/error-handler.d.ts.map +1 -1
  212. package/dist/exceptions/error-handler.js +32 -27
  213. package/dist/exceptions/so-exceptions.d.ts +1 -1
  214. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  215. package/dist/exceptions/so-exceptions.js +61 -68
  216. package/dist/filter/index.js +9 -3
  217. package/dist/filter/types.js +92 -88
  218. package/dist/filter/utils.d.ts +1 -1
  219. package/dist/filter/utils.d.ts.map +1 -1
  220. package/dist/filter/utils.js +147 -161
  221. package/dist/index.js +87 -40
  222. package/dist/logger/category.d.ts.map +1 -1
  223. package/dist/logger/category.js +30 -29
  224. package/dist/logger/configure.d.ts.map +1 -1
  225. package/dist/logger/configure.js +83 -107
  226. package/dist/migration/code-generation.d.ts +2 -2
  227. package/dist/migration/code-generation.d.ts.map +1 -1
  228. package/dist/migration/code-generation.js +1385 -1578
  229. package/dist/migration/migration-set.d.ts +1 -1
  230. package/dist/migration/migration-set.d.ts.map +1 -1
  231. package/dist/migration/migration-set.js +177 -227
  232. package/dist/migration/migrator.d.ts +4 -3
  233. package/dist/migration/migrator.d.ts.map +1 -1
  234. package/dist/migration/migrator.js +340 -345
  235. package/dist/migration/postgresql-schema-reader.d.ts +2 -2
  236. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  237. package/dist/migration/postgresql-schema-reader.js +506 -564
  238. package/dist/migration/slack-confirm.d.ts +2 -2
  239. package/dist/migration/slack-confirm.d.ts.map +1 -1
  240. package/dist/migration/slack-confirm.js +205 -193
  241. package/dist/migration/types.d.ts +2 -2
  242. package/dist/migration/types.d.ts.map +1 -1
  243. package/dist/migration/types.js +1 -3
  244. package/dist/naite/messaging-types.d.ts +1 -0
  245. package/dist/naite/messaging-types.d.ts.map +1 -1
  246. package/dist/naite/messaging-types.js +1 -7
  247. package/dist/naite/naite-reporter.d.ts +2 -2
  248. package/dist/naite/naite-reporter.d.ts.map +1 -1
  249. package/dist/naite/naite-reporter.js +127 -120
  250. package/dist/naite/naite.d.ts +3 -2
  251. package/dist/naite/naite.d.ts.map +1 -1
  252. package/dist/naite/naite.js +266 -300
  253. package/dist/ssr/index.d.ts +2 -2
  254. package/dist/ssr/index.d.ts.map +1 -1
  255. package/dist/ssr/index.js +13 -3
  256. package/dist/ssr/registry.d.ts +1 -1
  257. package/dist/ssr/registry.d.ts.map +1 -1
  258. package/dist/ssr/registry.js +45 -37
  259. package/dist/ssr/renderer.d.ts +4 -4
  260. package/dist/ssr/renderer.d.ts.map +1 -1
  261. package/dist/ssr/renderer.js +84 -91
  262. package/dist/ssr/types.d.ts +2 -2
  263. package/dist/ssr/types.d.ts.map +1 -1
  264. package/dist/ssr/types.js +1 -3
  265. package/dist/storage/base-file.js +54 -41
  266. package/dist/storage/buffered-file.d.ts +2 -2
  267. package/dist/storage/buffered-file.d.ts.map +1 -1
  268. package/dist/storage/buffered-file.js +51 -44
  269. package/dist/storage/drivers.d.ts +2 -2
  270. package/dist/storage/drivers.d.ts.map +1 -1
  271. package/dist/storage/drivers.js +12 -7
  272. package/dist/storage/index.js +14 -7
  273. package/dist/storage/s3-driver.d.ts +2 -2
  274. package/dist/storage/s3-driver.d.ts.map +1 -1
  275. package/dist/storage/s3-driver.js +52 -48
  276. package/dist/storage/storage-manager.d.ts +2 -2
  277. package/dist/storage/storage-manager.d.ts.map +1 -1
  278. package/dist/storage/storage-manager.js +33 -25
  279. package/dist/storage/types.d.ts +2 -2
  280. package/dist/storage/types.d.ts.map +1 -1
  281. package/dist/storage/types.js +1 -5
  282. package/dist/storage/uploaded-file.d.ts +1 -1
  283. package/dist/storage/uploaded-file.d.ts.map +1 -1
  284. package/dist/storage/uploaded-file.js +45 -35
  285. package/dist/stream/index.js +7 -2
  286. package/dist/stream/sse.d.ts +2 -2
  287. package/dist/stream/sse.d.ts.map +1 -1
  288. package/dist/stream/sse.js +72 -67
  289. package/dist/syncer/api-parser.d.ts +1 -1
  290. package/dist/syncer/api-parser.d.ts.map +1 -1
  291. package/dist/syncer/api-parser.js +224 -245
  292. package/dist/syncer/checksum.d.ts +1 -1
  293. package/dist/syncer/checksum.d.ts.map +1 -1
  294. package/dist/syncer/checksum.js +86 -72
  295. package/dist/syncer/code-generator.d.ts +2 -2
  296. package/dist/syncer/code-generator.d.ts.map +1 -1
  297. package/dist/syncer/code-generator.js +154 -160
  298. package/dist/syncer/entity-operations.d.ts +1 -1
  299. package/dist/syncer/entity-operations.d.ts.map +1 -1
  300. package/dist/syncer/entity-operations.js +63 -54
  301. package/dist/syncer/file-patterns.d.ts +1 -1
  302. package/dist/syncer/file-patterns.d.ts.map +1 -1
  303. package/dist/syncer/file-patterns.js +38 -38
  304. package/dist/syncer/index.js +19 -8
  305. package/dist/syncer/module-loader.d.ts +5 -5
  306. package/dist/syncer/module-loader.d.ts.map +1 -1
  307. package/dist/syncer/module-loader.js +83 -78
  308. package/dist/syncer/syncer-actions.d.ts +2 -2
  309. package/dist/syncer/syncer-actions.d.ts.map +1 -1
  310. package/dist/syncer/syncer-actions.js +76 -91
  311. package/dist/syncer/syncer.d.ts +7 -6
  312. package/dist/syncer/syncer.d.ts.map +1 -1
  313. package/dist/syncer/syncer.js +426 -492
  314. package/dist/tasks/decorator.d.ts +3 -3
  315. package/dist/tasks/decorator.d.ts.map +1 -1
  316. package/dist/tasks/decorator.js +32 -28
  317. package/dist/tasks/step-wrapper.d.ts +1 -1
  318. package/dist/tasks/step-wrapper.d.ts.map +1 -1
  319. package/dist/tasks/step-wrapper.js +42 -41
  320. package/dist/tasks/workflow-manager.d.ts +2 -2
  321. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  322. package/dist/tasks/workflow-manager.js +192 -221
  323. package/dist/template/entity-converter.d.ts +1 -1
  324. package/dist/template/entity-converter.d.ts.map +1 -1
  325. package/dist/template/entity-converter.js +103 -103
  326. package/dist/template/helpers.d.ts.map +1 -1
  327. package/dist/template/helpers.js +163 -163
  328. package/dist/template/implementations/entity.template.d.ts +1 -1
  329. package/dist/template/implementations/entity.template.d.ts.map +1 -1
  330. package/dist/template/implementations/entity.template.js +76 -85
  331. package/dist/template/implementations/entry-server.template.d.ts +1 -1
  332. package/dist/template/implementations/entry-server.template.d.ts.map +1 -1
  333. package/dist/template/implementations/entry-server.template.js +32 -27
  334. package/dist/template/implementations/generated.template.d.ts +1 -1
  335. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  336. package/dist/template/implementations/generated.template.js +254 -275
  337. package/dist/template/implementations/generated_http.template.d.ts +2 -2
  338. package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
  339. package/dist/template/implementations/generated_http.template.js +114 -133
  340. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  341. package/dist/template/implementations/generated_sso.template.js +249 -275
  342. package/dist/template/implementations/init_types.template.d.ts +1 -1
  343. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  344. package/dist/template/implementations/init_types.template.js +40 -34
  345. package/dist/template/implementations/model.template.d.ts +1 -1
  346. package/dist/template/implementations/model.template.d.ts.map +1 -1
  347. package/dist/template/implementations/model.template.js +56 -53
  348. package/dist/template/implementations/model_test.template.d.ts +1 -1
  349. package/dist/template/implementations/model_test.template.d.ts.map +1 -1
  350. package/dist/template/implementations/model_test.template.js +32 -24
  351. package/dist/template/implementations/queries.template.d.ts +1 -1
  352. package/dist/template/implementations/queries.template.d.ts.map +1 -1
  353. package/dist/template/implementations/queries.template.js +84 -89
  354. package/dist/template/implementations/sd.template.d.ts +1 -1
  355. package/dist/template/implementations/sd.template.d.ts.map +1 -1
  356. package/dist/template/implementations/sd.template.js +137 -144
  357. package/dist/template/implementations/services.template.d.ts +1 -1
  358. package/dist/template/implementations/services.template.d.ts.map +1 -1
  359. package/dist/template/implementations/services.template.js +164 -189
  360. package/dist/template/implementations/view_form.template.d.ts +1 -1
  361. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  362. package/dist/template/implementations/view_form.template.js +258 -285
  363. package/dist/template/implementations/view_id_all_select.template.d.ts +1 -1
  364. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
  365. package/dist/template/implementations/view_id_all_select.template.js +31 -25
  366. package/dist/template/implementations/view_list.template.d.ts +1 -1
  367. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  368. package/dist/template/implementations/view_list.template.js +304 -355
  369. package/dist/template/implementations/view_search_input.template.d.ts +1 -1
  370. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  371. package/dist/template/implementations/view_search_input.template.js +31 -27
  372. package/dist/template/index.js +21 -7
  373. package/dist/template/template-manager.d.ts +1 -1
  374. package/dist/template/template-manager.d.ts.map +1 -1
  375. package/dist/template/template-manager.js +132 -123
  376. package/dist/template/template-types.js +8 -6
  377. package/dist/template/template.d.ts +2 -2
  378. package/dist/template/template.d.ts.map +1 -1
  379. package/dist/template/template.js +73 -68
  380. package/dist/template/zod-converter.d.ts.map +1 -1
  381. package/dist/template/zod-converter.js +603 -657
  382. package/dist/testing/_relation-graph.d.ts +1 -1
  383. package/dist/testing/_relation-graph.d.ts.map +1 -1
  384. package/dist/testing/_relation-graph.js +93 -88
  385. package/dist/testing/bootstrap.d.ts +22 -13
  386. package/dist/testing/bootstrap.d.ts.map +1 -1
  387. package/dist/testing/bootstrap.js +114 -114
  388. package/dist/testing/data-explorer.d.ts +3 -3
  389. package/dist/testing/data-explorer.d.ts.map +1 -1
  390. package/dist/testing/data-explorer.js +237 -265
  391. package/dist/testing/dev-test-routes.d.ts +2 -2
  392. package/dist/testing/dev-test-routes.d.ts.map +1 -1
  393. package/dist/testing/dev-test-routes.js +258 -249
  394. package/dist/testing/dev-vitest-manager.d.ts +1 -1
  395. package/dist/testing/dev-vitest-manager.d.ts.map +1 -1
  396. package/dist/testing/dev-vitest-manager.js +514 -539
  397. package/dist/testing/faker-mappings.js +422 -420
  398. package/dist/testing/fixture-generator.d.ts +3 -3
  399. package/dist/testing/fixture-generator.d.ts.map +1 -1
  400. package/dist/testing/fixture-generator.js +1216 -1346
  401. package/dist/testing/fixture-loader.js +26 -25
  402. package/dist/testing/fixture-manager.d.ts +3 -3
  403. package/dist/testing/fixture-manager.d.ts.map +1 -1
  404. package/dist/testing/fixture-manager.js +706 -776
  405. package/dist/testing/global-setup.js +53 -49
  406. package/dist/testing/index.js +19 -11
  407. package/dist/testing/naite-vitest-reporter.js +18 -13
  408. package/dist/testing/parallel-db-manager.d.ts +1 -1
  409. package/dist/testing/parallel-db-manager.d.ts.map +1 -1
  410. package/dist/testing/parallel-db-manager.js +63 -78
  411. package/dist/testing/vitest-helpers.d.ts +1 -1
  412. package/dist/testing/vitest-helpers.d.ts.map +1 -1
  413. package/dist/testing/vitest-helpers.js +37 -33
  414. package/dist/types/types.d.ts +28 -28
  415. package/dist/types/types.d.ts.map +1 -1
  416. package/dist/types/types.js +764 -890
  417. package/dist/ui/ai-api.d.ts +1 -1
  418. package/dist/ui/ai-api.d.ts.map +1 -1
  419. package/dist/ui/ai-api.js +52 -42
  420. package/dist/ui/ai-client.d.ts +1 -2
  421. package/dist/ui/ai-client.d.ts.map +1 -1
  422. package/dist/ui/ai-client.js +353 -388
  423. package/dist/ui/api.d.ts +1 -1
  424. package/dist/ui/api.d.ts.map +1 -1
  425. package/dist/ui/api.js +903 -1145
  426. package/dist/ui/cdd-service.d.ts +1 -1
  427. package/dist/ui/cdd-service.d.ts.map +1 -1
  428. package/dist/ui/cdd-service.js +406 -407
  429. package/dist/ui/cdd-types.js +1 -3
  430. package/dist/ui-web/assets/index-C-Zz-wYg.css +1 -0
  431. package/dist/ui-web/assets/index-DejDON8K.js +238 -0
  432. package/dist/ui-web/index.html +3 -3
  433. package/dist/utils/async-utils.js +57 -45
  434. package/dist/utils/console-util.d.ts.map +1 -1
  435. package/dist/utils/console-util.js +104 -87
  436. package/dist/utils/controller.js +26 -19
  437. package/dist/utils/esm-utils.js +49 -38
  438. package/dist/utils/formatter.d.ts +1 -2
  439. package/dist/utils/formatter.d.ts.map +1 -1
  440. package/dist/utils/formatter.js +89 -115
  441. package/dist/utils/fs-utils.d.ts.map +1 -1
  442. package/dist/utils/fs-utils.js +68 -65
  443. package/dist/utils/lodash-able.js +11 -4
  444. package/dist/utils/model.d.ts +1 -1
  445. package/dist/utils/model.d.ts.map +1 -1
  446. package/dist/utils/model.js +21 -19
  447. package/dist/utils/object-utils.js +148 -186
  448. package/dist/utils/path-utils.js +67 -57
  449. package/dist/utils/process-utils.d.ts.map +1 -1
  450. package/dist/utils/process-utils.js +37 -31
  451. package/dist/utils/sql-parser.d.ts +1 -1
  452. package/dist/utils/sql-parser.d.ts.map +1 -1
  453. package/dist/utils/sql-parser.js +40 -40
  454. package/dist/utils/type-utils.js +44 -43
  455. package/dist/utils/utils.d.ts +2 -3
  456. package/dist/utils/utils.d.ts.map +1 -1
  457. package/dist/utils/utils.js +81 -93
  458. package/dist/utils/zod-error.d.ts +1 -1
  459. package/dist/utils/zod-error.d.ts.map +1 -1
  460. package/dist/utils/zod-error.js +24 -17
  461. package/dist/vector/chunking.d.ts +1 -1
  462. package/dist/vector/chunking.d.ts.map +1 -1
  463. package/dist/vector/chunking.js +100 -94
  464. package/dist/vector/config.d.ts +1 -1
  465. package/dist/vector/config.d.ts.map +1 -1
  466. package/dist/vector/config.js +76 -78
  467. package/dist/vector/embedding.d.ts +1 -1
  468. package/dist/vector/embedding.d.ts.map +1 -1
  469. package/dist/vector/embedding.js +128 -125
  470. package/dist/vector/index.js +5 -5
  471. package/dist/vector/types.js +1 -5
  472. package/package.json +31 -36
  473. package/src/ai/agents/agent.ts +12 -5
  474. package/src/ai/agents/types.ts +5 -5
  475. package/src/ai/providers/rtzr/model.ts +8 -10
  476. package/src/ai/providers/rtzr/options.ts +2 -1
  477. package/src/ai/providers/rtzr/provider.ts +5 -3
  478. package/src/ai/providers/rtzr/utils.ts +2 -7
  479. package/src/api/__tests__/config.test.ts +15 -8
  480. package/src/api/base-frame.ts +5 -3
  481. package/src/api/caster.ts +7 -6
  482. package/src/api/code-converters.ts +23 -26
  483. package/src/api/config.ts +23 -17
  484. package/src/api/context.ts +18 -11
  485. package/src/api/decorators.ts +17 -18
  486. package/src/api/sonamu.ts +44 -49
  487. package/src/auth/auth-generator.ts +4 -6
  488. package/src/auth/better-auth-entities.ts +3 -2
  489. package/src/auth/knex-adapter.ts +6 -5
  490. package/src/auth/plugins/entity-definitions/admin.ts +1 -1
  491. package/src/auth/plugins/entity-definitions/anonymous.ts +1 -1
  492. package/src/auth/plugins/entity-definitions/api-key.ts +1 -1
  493. package/src/auth/plugins/entity-definitions/index.ts +1 -1
  494. package/src/auth/plugins/entity-definitions/jwt.ts +1 -1
  495. package/src/auth/plugins/entity-definitions/organization.ts +1 -1
  496. package/src/auth/plugins/entity-definitions/passkey.ts +1 -1
  497. package/src/auth/plugins/entity-definitions/phone-number.ts +1 -1
  498. package/src/auth/plugins/entity-definitions/sso.ts +1 -1
  499. package/src/auth/plugins/entity-definitions/two-factor.ts +1 -1
  500. package/src/auth/plugins/entity-definitions/types.ts +1 -1
  501. package/src/auth/plugins/entity-definitions/username.ts +1 -1
  502. package/src/auth/plugins/wrappers/admin.ts +3 -1
  503. package/src/auth/plugins/wrappers/anonymous.ts +3 -1
  504. package/src/auth/plugins/wrappers/api-key.ts +3 -1
  505. package/src/auth/plugins/wrappers/jwt.ts +3 -1
  506. package/src/auth/plugins/wrappers/organization.ts +3 -1
  507. package/src/auth/plugins/wrappers/passkey.ts +3 -1
  508. package/src/auth/plugins/wrappers/phone-number.ts +3 -1
  509. package/src/auth/plugins/wrappers/sso.ts +2 -1
  510. package/src/auth/plugins/wrappers/two-factor.ts +3 -1
  511. package/src/auth/plugins/wrappers/username.ts +3 -1
  512. package/src/bin/__tests__/ts-loader-register.test.ts +7 -12
  513. package/src/bin/build-config.ts +3 -3
  514. package/src/bin/cli.ts +27 -25
  515. package/src/bin/fixture.ts +4 -2
  516. package/src/bin/hmr-hook-register.ts +1 -0
  517. package/src/bin/test-command.ts +4 -2
  518. package/src/bin/ts-loader-registration.ts +6 -22
  519. package/src/cache/cache-manager.ts +2 -1
  520. package/src/cache/decorator.ts +2 -2
  521. package/src/cache/types.ts +3 -3
  522. package/src/cache-control/cache-control.ts +3 -2
  523. package/src/cache-control/types.ts +2 -2
  524. package/src/compress/compress.ts +1 -1
  525. package/src/cone/cone-generator.ts +5 -3
  526. package/src/database/_batch_update.ts +1 -1
  527. package/src/database/base-model.ts +20 -14
  528. package/src/database/base-model.types.ts +12 -11
  529. package/src/database/db.ts +56 -21
  530. package/src/database/knex.ts +2 -2
  531. package/src/database/puri-subset.test-d.ts +33 -32
  532. package/src/database/puri-subset.types.ts +6 -7
  533. package/src/database/puri-wrapper.ts +29 -26
  534. package/src/database/puri.ts +36 -34
  535. package/src/database/puri.types.test-d.ts +6 -5
  536. package/src/database/puri.types.ts +9 -12
  537. package/src/database/transaction-context.ts +2 -2
  538. package/src/database/upsert-builder.ts +17 -10
  539. package/src/dict/sd.ts +17 -4
  540. package/src/dict/sonamu-dictionary.ts +23 -17
  541. package/src/entity/entity-manager.ts +9 -7
  542. package/src/entity/entity-template-cone.ts +10 -3
  543. package/src/entity/entity.ts +20 -16
  544. package/src/exceptions/error-handler.ts +2 -1
  545. package/src/exceptions/so-exceptions.ts +1 -1
  546. package/src/filter/utils.ts +3 -2
  547. package/src/logger/category.ts +1 -0
  548. package/src/logger/configure.ts +5 -5
  549. package/src/migration/__tests__/code-generation.search-text.test.ts +2 -3
  550. package/src/migration/code-generation.ts +26 -25
  551. package/src/migration/migration-set.ts +16 -18
  552. package/src/migration/migrator.ts +38 -33
  553. package/src/migration/postgresql-schema-reader.ts +12 -12
  554. package/src/migration/slack-confirm.ts +5 -4
  555. package/src/migration/types.ts +2 -2
  556. package/src/naite/messaging-types.ts +1 -1
  557. package/src/naite/naite-reporter.ts +5 -3
  558. package/src/naite/naite.ts +12 -7
  559. package/src/shared/app.shared.ts.txt +2 -2
  560. package/src/shared/web.shared.ts.txt +2 -2
  561. package/src/skills/AGENTS.md +19 -18
  562. package/src/skills/commands/sonamu-skills.md +9 -9
  563. package/src/skills/sonamu/SKILL.md +111 -104
  564. package/src/skills/sonamu/ai-agents.md +27 -26
  565. package/src/skills/sonamu/api.md +81 -69
  566. package/src/skills/sonamu/auth-migration.md +13 -27
  567. package/src/skills/sonamu/auth-plugins.md +41 -31
  568. package/src/skills/sonamu/auth.md +30 -24
  569. package/src/skills/sonamu/cdd.md +26 -17
  570. package/src/skills/sonamu/cone.md +50 -50
  571. package/src/skills/sonamu/config.md +74 -51
  572. package/src/skills/sonamu/create-sonamu.md +31 -19
  573. package/src/skills/sonamu/database.md +43 -26
  574. package/src/skills/sonamu/entity-basic.md +61 -61
  575. package/src/skills/sonamu/entity-relations.md +84 -80
  576. package/src/skills/sonamu/entity-validation-checklist.md +19 -15
  577. package/src/skills/sonamu/fixture-cli.md +52 -30
  578. package/src/skills/sonamu/framework-change.md +9 -7
  579. package/src/skills/sonamu/frontend.md +64 -82
  580. package/src/skills/sonamu/i18n.md +45 -37
  581. package/src/skills/sonamu/migration.md +54 -31
  582. package/src/skills/sonamu/model.md +98 -66
  583. package/src/skills/sonamu/naite.md +34 -32
  584. package/src/skills/sonamu/project-init.md +28 -8
  585. package/src/skills/sonamu/puri.md +82 -91
  586. package/src/skills/sonamu/scaffolding.md +44 -32
  587. package/src/skills/sonamu/skill-contribution.md +50 -45
  588. package/src/skills/sonamu/subset.md +13 -13
  589. package/src/skills/sonamu/tasks.md +73 -58
  590. package/src/skills/sonamu/testing-devrunner.md +56 -36
  591. package/src/skills/sonamu/testing.md +23 -58
  592. package/src/skills/sonamu/upsert.md +32 -31
  593. package/src/skills/sonamu/vector.md +37 -36
  594. package/src/ssr/index.ts +2 -12
  595. package/src/ssr/registry.ts +1 -1
  596. package/src/ssr/renderer.ts +7 -5
  597. package/src/ssr/types.ts +2 -2
  598. package/src/storage/buffered-file.ts +4 -2
  599. package/src/storage/drivers.ts +3 -2
  600. package/src/storage/s3-driver.ts +7 -4
  601. package/src/storage/storage-manager.ts +3 -2
  602. package/src/storage/types.ts +3 -2
  603. package/src/storage/uploaded-file.ts +1 -1
  604. package/src/stream/sse.ts +2 -2
  605. package/src/syncer/api-parser.ts +8 -5
  606. package/src/syncer/checksum.ts +9 -5
  607. package/src/syncer/code-generator.ts +16 -8
  608. package/src/syncer/entity-operations.ts +5 -3
  609. package/src/syncer/file-patterns.ts +2 -1
  610. package/src/syncer/module-loader.ts +9 -6
  611. package/src/syncer/syncer-actions.ts +5 -3
  612. package/src/syncer/syncer.ts +18 -24
  613. package/src/tasks/decorator.ts +10 -8
  614. package/src/tasks/step-wrapper.ts +1 -1
  615. package/src/tasks/workflow-manager.ts +18 -15
  616. package/src/template/__tests__/generated.template.search-text.test.ts +1 -0
  617. package/src/template/entity-converter.ts +4 -2
  618. package/src/template/generated.template.test-d.ts +2 -1
  619. package/src/template/helpers.ts +5 -2
  620. package/src/template/implementations/entity.template.ts +9 -8
  621. package/src/template/implementations/entry-server.template.ts +1 -1
  622. package/src/template/implementations/generated.template.ts +21 -29
  623. package/src/template/implementations/generated_http.template.ts +9 -6
  624. package/src/template/implementations/generated_sso.template.ts +6 -4
  625. package/src/template/implementations/init_types.template.ts +3 -2
  626. package/src/template/implementations/model.template.ts +4 -2
  627. package/src/template/implementations/model_test.template.ts +3 -2
  628. package/src/template/implementations/queries.template.ts +6 -14
  629. package/src/template/implementations/sd.template.ts +4 -2
  630. package/src/template/implementations/services.template.ts +7 -11
  631. package/src/template/implementations/view_form.template.ts +5 -3
  632. package/src/template/implementations/view_id_all_select.template.ts +3 -2
  633. package/src/template/implementations/view_list.template.ts +7 -5
  634. package/src/template/implementations/view_search_input.template.ts +3 -2
  635. package/src/template/template-manager.ts +4 -3
  636. package/src/template/template.ts +4 -3
  637. package/src/template/zod-converter.ts +10 -7
  638. package/src/testing/__tests__/dev-test-routes.test.ts +3 -2
  639. package/src/testing/__tests__/dev-vitest-manager.test.ts +1 -0
  640. package/src/testing/_relation-graph.ts +2 -2
  641. package/src/testing/bootstrap.ts +55 -27
  642. package/src/testing/data-explorer.ts +5 -4
  643. package/src/testing/dev-test-routes.ts +8 -5
  644. package/src/testing/dev-vitest-manager.ts +13 -12
  645. package/src/testing/fixture-generator.ts +11 -17
  646. package/src/testing/fixture-manager.ts +21 -17
  647. package/src/testing/parallel-db-manager.ts +2 -1
  648. package/src/testing/vitest-helpers.ts +2 -1
  649. package/src/types/__tests__/entity-json-schema-search-text.test.ts +1 -0
  650. package/src/types/types.ts +8 -8
  651. package/src/typings/knex.d.ts +4 -4
  652. package/src/ui/ai-api.ts +5 -3
  653. package/src/ui/ai-client.ts +6 -5
  654. package/src/ui/api.ts +25 -23
  655. package/src/ui/cdd-service.ts +12 -11
  656. package/src/utils/console-util.ts +3 -1
  657. package/src/utils/formatter.ts +94 -102
  658. package/src/utils/fs-utils.ts +2 -1
  659. package/src/utils/model.ts +2 -2
  660. package/src/utils/object-utils.ts +3 -3
  661. package/src/utils/process-utils.ts +2 -1
  662. package/src/utils/sql-parser.ts +10 -1
  663. package/src/utils/type-utils.ts +3 -3
  664. package/src/utils/utils.ts +9 -7
  665. package/src/utils/zod-error.ts +1 -1
  666. package/src/vector/chunking.ts +1 -1
  667. package/src/vector/config.ts +1 -1
  668. package/src/vector/embedding.ts +11 -9
  669. package/tsdown.api.config.ts +50 -0
  670. package/.swcrc.project-default +0 -18
  671. package/dist/api/__tests__/config.test.js +0 -189
  672. package/dist/bin/__tests__/test-command.test.js +0 -112
  673. package/dist/bin/__tests__/ts-loader-register.test.js +0 -45
  674. package/dist/database/puri-subset.test-d.js +0 -89
  675. package/dist/database/puri.types.test-d.js +0 -129
  676. package/dist/migration/__tests__/code-generation.search-text.test.js +0 -435
  677. package/dist/template/__tests__/generated.template.search-text.test.js +0 -99
  678. package/dist/template/generated.template.test-d.js +0 -24
  679. package/dist/testing/__tests__/dev-test-routes.test.js +0 -144
  680. package/dist/testing/__tests__/dev-vitest-manager.test.js +0 -152
  681. package/dist/types/__tests__/entity-json-schema-search-text.test.js +0 -256
  682. package/dist/typings/knex.d.js +0 -3
  683. package/dist/ui-web/assets/index-CKo0Z2Iu.css +0 -1
  684. package/dist/ui-web/assets/index-DK-2aacv.js +0 -257
@@ -1,791 +1,721 @@
1
+ import { __esmMin } from "../_virtual/rolldown_runtime.js";
2
+ import { createKnexInstance, init_knex } from "../database/knex.js";
3
+ import { init_controller, isTest } from "../utils/controller.js";
4
+ import { Sonamu, init_sonamu } from "../api/sonamu.js";
5
+ import { init_types, isBelongsToOneRelationProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
6
+ import { EntityManager, init_entity_manager } from "../entity/entity-manager.js";
7
+ import { UpsertBuilder, init_upsert_builder } from "../database/upsert-builder.js";
8
+ import { BaseModel, init_base_model } from "../database/base-model.js";
9
+ import { RelationGraph, init__relation_graph } from "./_relation-graph.js";
10
+ import inflection from "inflection";
11
+ import { unique } from "radashi";
1
12
  import assert from "assert";
2
13
  import chalk from "chalk";
3
- import { execSync } from "child_process";
4
14
  import { readFileSync, writeFileSync } from "fs";
5
- import inflection from "inflection";
6
- import { unique } from "radashi";
15
+ import { execSync } from "child_process";
7
16
  import { inspect } from "util";
8
- import { Sonamu } from "../api/index.js";
9
- import { BaseModel } from "../database/base-model.js";
10
- import { createKnexInstance } from "../database/knex.js";
11
- import { UpsertBuilder } from "../database/upsert-builder.js";
12
- import { EntityManager } from "../entity/entity-manager.js";
13
- import { isBelongsToOneRelationProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
14
- import { isTest } from "../utils/controller.js";
15
- import { RelationGraph } from "./_relation-graph.js";
16
- export class FixtureManagerClass {
17
- _tdb = null;
18
- set tdb(tdb) {
19
- this._tdb = tdb;
20
- }
21
- get tdb() {
22
- if (this._tdb === null) {
23
- throw new Error("FixtureManager has not been initialized");
24
- }
25
- return this._tdb;
26
- }
27
- _fdb = null;
28
- set fdb(fdb) {
29
- this._fdb = fdb;
30
- }
31
- get fdb() {
32
- if (this._fdb === null) {
33
- throw new Error("FixtureManager has not been initialized");
34
- }
35
- return this._fdb;
36
- }
37
- cachedTableNames = null;
38
- relationGraph = new RelationGraph();
39
- // UpsertBuilder 기반 import를 위한 상태
40
- builder = new UpsertBuilder();
41
- fixtureRefMap = new Map();
42
- skippedFixtures = new Map();
43
- init() {
44
- if (this._tdb !== null) {
45
- return;
46
- }
47
- if (Sonamu.dbConfig.test && Sonamu.dbConfig.production_master) {
48
- const tConn = Sonamu.dbConfig.test.connection;
49
- const pConn = Sonamu.dbConfig.production_master.connection;
50
- if (`${tConn.host ?? "localhost"}:${tConn.port ?? 5432}/${tConn.database}` === `${pConn.host ?? "localhost"}:${pConn.port ?? 5432}/${pConn.database}`) {
51
- throw new Error(`테스트DB와 프로덕션DB에 동일한 데이터베이스가 사용되었습니다.`);
52
- }
53
- }
54
- this.tdb = createKnexInstance(Sonamu.dbConfig.test);
55
- this.fdb = createKnexInstance(Sonamu.dbConfig.fixture);
56
- }
57
- /**
58
- 원격 fixture DB를 로컬 test DB로 복사합니다.
59
- pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.
60
- */ async sync() {
61
- const fixtureConn = Sonamu.dbConfig.fixture.connection;
62
- const testConn = Sonamu.dbConfig.test.connection;
63
- // 1. 로컬 test DB 연결 종료 및 재생성
64
- const testPgEnv = {
65
- PGPASSWORD: testConn.password || ""
66
- };
67
- execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "
17
+
18
+ //#region src/testing/fixture-manager.ts
19
+ var FixtureManagerClass, FixtureManager;
20
+ var init_fixture_manager = __esmMin((() => {
21
+ init_sonamu();
22
+ init_base_model();
23
+ init_knex();
24
+ init_upsert_builder();
25
+ init_entity_manager();
26
+ init_types();
27
+ init_controller();
28
+ init__relation_graph();
29
+ FixtureManagerClass = class {
30
+ _tdb = null;
31
+ set tdb(tdb) {
32
+ this._tdb = tdb;
33
+ }
34
+ get tdb() {
35
+ if (this._tdb === null) {
36
+ throw new Error("FixtureManager has not been initialized");
37
+ }
38
+ return this._tdb;
39
+ }
40
+ _fdb = null;
41
+ set fdb(fdb) {
42
+ this._fdb = fdb;
43
+ }
44
+ get fdb() {
45
+ if (this._fdb === null) {
46
+ throw new Error("FixtureManager has not been initialized");
47
+ }
48
+ return this._fdb;
49
+ }
50
+ cachedTableNames = null;
51
+ relationGraph = new RelationGraph();
52
+ builder = new UpsertBuilder();
53
+ fixtureRefMap = new Map();
54
+ skippedFixtures = new Map();
55
+ init() {
56
+ if (this._tdb !== null) {
57
+ return;
58
+ }
59
+ if (Sonamu.dbConfig.test && Sonamu.dbConfig.production_master) {
60
+ const tConn = Sonamu.dbConfig.test.connection;
61
+ const pConn = Sonamu.dbConfig.production_master.connection;
62
+ if (`${tConn.host ?? "localhost"}:${tConn.port ?? 5432}/${tConn.database}` === `${pConn.host ?? "localhost"}:${pConn.port ?? 5432}/${pConn.database}`) {
63
+ throw new Error(`테스트DB와 프로덕션DB에 동일한 데이터베이스가 사용되었습니다.`);
64
+ }
65
+ }
66
+ this.tdb = createKnexInstance(Sonamu.dbConfig.test);
67
+ this.fdb = createKnexInstance(Sonamu.dbConfig.fixture);
68
+ }
69
+ /**
70
+ 원격 fixture DB를 로컬 test DB로 복사합니다.
71
+ pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.
72
+ */
73
+ async sync() {
74
+ const fixtureConn = Sonamu.dbConfig.fixture.connection;
75
+ const testConn = Sonamu.dbConfig.test.connection;
76
+ const testPgEnv = { PGPASSWORD: testConn.password || "" };
77
+ execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "
68
78
  SELECT pg_terminate_backend(pg_stat_activity.pid)
69
79
  FROM pg_stat_activity
70
80
  WHERE datname = '${testConn.database}'
71
81
  AND pid <> pg_backend_pid();
72
82
  "`, {
73
- stdio: "inherit",
74
- env: {
75
- ...process.env,
76
- ...testPgEnv
77
- }
78
- });
79
- execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "DROP DATABASE IF EXISTS \\"${testConn.database}\\""`, {
80
- stdio: "inherit",
81
- env: {
82
- ...process.env,
83
- ...testPgEnv
84
- }
85
- });
86
- execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "CREATE DATABASE \\"${testConn.database}\\""`, {
87
- stdio: "inherit",
88
- env: {
89
- ...process.env,
90
- ...testPgEnv
91
- }
92
- });
93
- // 2. 원격 fixture DB 로컬 test DB로 복사 (pg_dump | pg_restore)
94
- const fixturePgEnv = {
95
- PGPASSWORD: fixtureConn.password || ""
96
- };
97
- const dumpCmd = `pg_dump -h ${fixtureConn.host} -p ${fixtureConn.port ?? 5432} -U ${fixtureConn.user} -d ${fixtureConn.database} -Fc`;
98
- const restoreCmd = `pg_restore -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d ${testConn.database} --no-owner --no-acl`;
99
- execSync(`${dumpCmd} | PGPASSWORD="${testConn.password || ""}" ${restoreCmd}`, {
100
- stdio: "inherit",
101
- env: {
102
- ...process.env,
103
- ...fixturePgEnv
104
- },
105
- shell: "/bin/bash"
106
- });
107
- // 3. 시퀀스 리셋 (데이터 복사 후 시퀀스를 MAX(id)로 정렬)
108
- await this.resetSequences(Sonamu.dbConfig.test);
109
- }
110
- /**
111
- * 모든 테이블의 시퀀스를 현재 MAX(id)로 리셋합니다.
112
- * fixture sync 후 시퀀스가 실제 데이터와 맞지 않는 문제를 해결합니다.
113
- */ async resetSequences(dbConfig) {
114
- const testDb = createKnexInstance(dbConfig);
115
- const entities = EntityManager.getAllEntities();
116
- try {
117
- for (const entity of entities){
118
- const tableName = entity.table || entity.id.toLowerCase();
119
- // id 필드의 타입을 확인합니다
120
- const idProp = entity.props.find((p)=>p.name === "id");
121
- const idType = idProp?.type;
122
- // integer/bigInteger이거나, string이지만 fixtureStrategy=sequence인 경우에만 리셋합니다
123
- const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
124
- if (!usesSequence) {
125
- !isTest() && console.log(`Skipping sequence reset for ${tableName} (id type: ${idType || "unknown"})`);
126
- continue;
127
- }
128
- // PostgreSQL 시퀀스를 현재 테이블의 MAX(id)로 리셋합니다.
129
- // string 타입의 경우 숫자 캐스팅이 필요합니다.
130
- const maxExpr = idType === "string" ? "MAX(id::bigint)" : "MAX(id)";
131
- await testDb.raw(`
83
+ stdio: "inherit",
84
+ env: {
85
+ ...process.env,
86
+ ...testPgEnv
87
+ }
88
+ });
89
+ execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "DROP DATABASE IF EXISTS \\"${testConn.database}\\""`, {
90
+ stdio: "inherit",
91
+ env: {
92
+ ...process.env,
93
+ ...testPgEnv
94
+ }
95
+ });
96
+ execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "CREATE DATABASE \\"${testConn.database}\\""`, {
97
+ stdio: "inherit",
98
+ env: {
99
+ ...process.env,
100
+ ...testPgEnv
101
+ }
102
+ });
103
+ const fixturePgEnv = { PGPASSWORD: fixtureConn.password || "" };
104
+ const dumpCmd = `pg_dump -h ${fixtureConn.host} -p ${fixtureConn.port ?? 5432} -U ${fixtureConn.user} -d ${fixtureConn.database} -Fc`;
105
+ const restoreCmd = `pg_restore -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d ${testConn.database} --no-owner --no-acl`;
106
+ execSync(`${dumpCmd} | PGPASSWORD="${testConn.password || ""}" ${restoreCmd}`, {
107
+ stdio: "inherit",
108
+ env: {
109
+ ...process.env,
110
+ ...fixturePgEnv
111
+ },
112
+ shell: "/bin/bash"
113
+ });
114
+ await this.resetSequences(Sonamu.dbConfig.test);
115
+ }
116
+ /**
117
+ * 모든 테이블의 시퀀스를 현재 MAX(id)로 리셋합니다.
118
+ * fixture sync 후 시퀀스가 실제 데이터와 맞지 않는 문제를 해결합니다.
119
+ */
120
+ async resetSequences(dbConfig) {
121
+ const testDb = createKnexInstance(dbConfig);
122
+ const entities = EntityManager.getAllEntities();
123
+ try {
124
+ for (const entity of entities) {
125
+ const tableName = entity.table || entity.id.toLowerCase();
126
+ const idProp = entity.props.find((p) => p.name === "id");
127
+ const idType = idProp?.type;
128
+ const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
129
+ if (!usesSequence) {
130
+ !isTest() && console.log(`Skipping sequence reset for ${tableName} (id type: ${idType || "unknown"})`);
131
+ continue;
132
+ }
133
+ const maxExpr = idType === "string" ? "MAX(id::bigint)" : "MAX(id)";
134
+ await testDb.raw(`
132
135
  SELECT setval(
133
136
  pg_get_serial_sequence('public.${tableName}', 'id'),
134
137
  COALESCE((SELECT ${maxExpr} FROM ${tableName}), 1)
135
138
  )
136
139
  `);
137
- }
138
- } finally{
139
- await testDb.destroy();
140
- }
141
- }
142
- visitedRecords = new Set();
143
- async importFixture(entityId, ids) {
144
- // 방문 기록 초기화 (새로운 import 작업 시작)
145
- this.visitedRecords.clear();
146
- const queries = unique((await Promise.all(ids.map(async (id)=>{
147
- return await this.getImportQueries(entityId, "id", id);
148
- }))).flat());
149
- const wdb = BaseModel.getDB("w");
150
- for (const query of queries){
151
- await wdb.raw(query);
152
- }
153
- }
154
- async getImportQueries(entityId, field, id) {
155
- const recordKey = `${entityId}#${field}#${id}`;
156
- // 순환 참조 방지: 이미 방문한 레코드는 스킵
157
- if (this.visitedRecords.has(recordKey)) {
158
- return [];
159
- }
160
- this.visitedRecords.add(recordKey);
161
- const entity = EntityManager.get(entityId);
162
- const wdb = BaseModel.getDB("w");
163
- // 여기서 실DB의 row 가져옴
164
- const [row] = await wdb(entity.table).where(field, id).limit(1);
165
- if (row === undefined) {
166
- throw new Error(`${entityId}#${id} row를 찾을 수 없습니다.`);
167
- }
168
- // 픽스쳐DB, 실DB
169
- const fixtureDatabase = Sonamu.dbConfig.fixture.connection.database;
170
- const realDatabase = Sonamu.dbConfig.production_master.connection.database;
171
- const selfQuery = `INSERT IGNORE INTO \`${fixtureDatabase}\`.\`${entity.table}\` (SELECT * FROM \`${realDatabase}\`.\`${entity.table}\` WHERE \`id\` = ${id})`;
172
- const args = Object.entries(entity.relations).filter(([, relation])=>isBelongsToOneRelationProp(relation) || isOneToOneRelationProp(relation) && relation.customJoinClause === undefined).map(([, relation])=>{
173
- /*
174
- BelongsToOne인 경우
175
- Category / 'id' / row[category_id] 호출
176
- OneToOne에 joinColumn === true 인 경우
177
- Profile / 'id' / row[profile_id] 호출
178
- OneToOne에 joinColumn === false 인 경우
179
- Profile / 'profile_id' / row['id'] 호출
180
- */ let field;
181
- let id;
182
- if (isOneToOneRelationProp(relation) && !relation.hasJoinColumn) {
183
- const relatedEntity = EntityManager.get(relation.with);
184
- const relatedIdColumnName = relatedEntity.props.find((p)=>isRelationProp(p) && p.with === entity.id)?.name;
185
- if (!relatedIdColumnName) {
186
- throw new Error(`${relatedEntity.id}의 ${entity.id} 관계 프롭을 찾을 수 없습니다.`);
187
- }
188
- field = `${relatedIdColumnName}_id`;
189
- id = row.id;
190
- } else {
191
- field = "id";
192
- id = row[`${relation.name}_id`];
193
- }
194
- return {
195
- entityId: relation.with,
196
- field,
197
- id
198
- };
199
- }).filter((arg)=>arg.id !== null);
200
- const relQueries = await Promise.all(args.map(async (args)=>{
201
- return this.getImportQueries(args.entityId, args.field, args.id);
202
- }));
203
- return [
204
- ...unique(relQueries.reverse().flat()),
205
- selfQuery
206
- ];
207
- }
208
- async destroy() {
209
- if (this._tdb) {
210
- await this._tdb.destroy();
211
- this._tdb = null;
212
- }
213
- if (this._fdb) {
214
- await this._fdb.destroy();
215
- this._fdb = null;
216
- }
217
- await BaseModel.destroy();
218
- }
219
- async getFixtures(sourceDBName, targetDBName, searchOptions, duplicateCheck) {
220
- const sourceDB = createKnexInstance(Sonamu.dbConfig[sourceDBName]);
221
- const targetDB = createKnexInstance(Sonamu.dbConfig[targetDBName]);
222
- try {
223
- const { entityId, field, value, searchType } = searchOptions;
224
- const entity = EntityManager.get(entityId);
225
- const column = entity.props.find((prop)=>prop.name === field)?.type === "relation" ? `${field}_id` : field;
226
- let query = sourceDB(entity.table);
227
- if (searchType === "equals") {
228
- query = query.where(column, value);
229
- } else if (searchType === "like") {
230
- query = query.where(column, "like", `%${value}%`);
231
- }
232
- const rows = await query;
233
- if (rows.length === 0) {
234
- throw new Error("No records found");
235
- }
236
- const fixtures = [];
237
- for (const row of rows){
238
- const initialRecordsLength = fixtures.length;
239
- const newRecords = await this.createFixtureRecord(entity, row, {
240
- _db: sourceDB
241
- });
242
- fixtures.push(...newRecords);
243
- const currentFixtureRecord = fixtures.find((r)=>r.fixtureId === `${entityId}#${row.id}`);
244
- if (currentFixtureRecord) {
245
- // 현재 fixture로부터 생성된 fetchedRecords 설정
246
- currentFixtureRecord.fetchedRecords = fixtures.filter((r)=>r.fixtureId !== currentFixtureRecord.fixtureId).slice(initialRecordsLength).map((r)=>r.fixtureId);
247
- }
248
- }
249
- for await (const fixture of fixtures){
250
- const entity = EntityManager.get(fixture.entityId);
251
- // 사용자 지정 컬럼 기준 중복 확인 → target
252
- const customColumns = duplicateCheck?.columns?.[fixture.entityId];
253
- if (customColumns && customColumns.length > 0) {
254
- const customDuplicateRow = await this.checkDuplicateByColumns(targetDB, entity, fixture, customColumns);
255
- if (customDuplicateRow) {
256
- const [record] = await this.createFixtureRecord(entity, customDuplicateRow, {
257
- singleRecord: true,
258
- _db: targetDB
259
- });
260
- fixture.target = record;
261
- }
262
- }
263
- // Unique index 기준 중복 확인 → fixture.unique
264
- const uniqueRow = await this.checkUniqueViolation(targetDB, entity, fixture);
265
- if (uniqueRow) {
266
- const [record] = await this.createFixtureRecord(entity, uniqueRow, {
267
- singleRecord: true,
268
- _db: targetDB
269
- });
270
- fixture.unique = record;
271
- }
272
- }
273
- return unique(fixtures, (f)=>f.fixtureId);
274
- } finally{
275
- await Promise.allSettled([
276
- targetDB.destroy(),
277
- sourceDB.destroy()
278
- ]);
279
- }
280
- }
281
- async createFixtureRecord(entity, row, options) {
282
- const records = [];
283
- const visitedEntities = new Set();
284
- const create = async (entity, row)=>{
285
- const fixtureId = `${entity.id}#${row.id}`;
286
- if (visitedEntities.has(fixtureId)) {
287
- return;
288
- }
289
- visitedEntities.add(fixtureId);
290
- const record = {
291
- fixtureId,
292
- entityId: entity.id,
293
- id: row.id,
294
- columns: {},
295
- fetchedRecords: [],
296
- belongsRecords: []
297
- };
298
- for (const prop of entity.props){
299
- if (isVirtualProp(prop)) {
300
- continue;
301
- }
302
- record.columns[prop.name] = {
303
- prop: prop,
304
- value: row[prop.name]
305
- };
306
- const db = options?._db ?? BaseModel.getDB("w");
307
- if (isManyToManyRelationProp(prop)) {
308
- const relatedEntity = EntityManager.get(prop.with);
309
- const throughTable = prop.joinTable;
310
- const fromColumn = `${inflection.singularize(entity.table)}_id`;
311
- const toColumn = `${inflection.singularize(relatedEntity.table)}_id`;
312
- const relatedIds = await db(throughTable).where(fromColumn, row.id).pluck(toColumn);
313
- record.columns[prop.name].value = relatedIds;
314
- } else if (isHasManyRelationProp(prop)) {
315
- const relatedEntity = EntityManager.get(prop.with);
316
- const relatedIds = await db(relatedEntity.table).where(prop.joinColumn, row.id).pluck("id");
317
- record.columns[prop.name].value = relatedIds;
318
- } else if (isOneToOneRelationProp(prop) && !prop.hasJoinColumn) {
319
- // 역방향 OneToOne: FK를 가진 관련 엔티티를 찾습니다
320
- // 예시: User OneToOne Employee (Employee가 user_id FK를 가짐)
321
- const relatedEntity = EntityManager.get(prop.with);
322
- const relatedProp = relatedEntity.props.find((p)=>isRelationProp(p) && p.with === entity.id);
323
- if (relatedProp && isRelationProp(relatedProp)) {
324
- // 관련 엔티티에서 FK 컬럼으로 쿼리합니다 (id가 아님)
325
- const fkColumn = `${relatedProp.name}_id`;
326
- const relatedRow = await db(relatedEntity.table).where(fkColumn, row.id).first();
327
- record.columns[prop.name].value = relatedRow?.id;
328
- }
329
- } else if (isRelationProp(prop)) {
330
- const relatedId = row[`${prop.name}_id`];
331
- record.columns[prop.name].value = relatedId;
332
- if (relatedId) {
333
- record.belongsRecords.push(`${prop.with}#${relatedId}`);
334
- }
335
- if (!options?.singleRecord && relatedId) {
336
- const relatedEntity = EntityManager.get(prop.with);
337
- const relatedRow = await db(relatedEntity.table).where("id", relatedId).first();
338
- if (relatedRow) {
339
- await create(relatedEntity, relatedRow);
340
- }
341
- }
342
- }
343
- }
344
- records.push(record);
345
- };
346
- await create(entity, row);
347
- return records;
348
- }
349
- /**
350
- * 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)
351
- * 2. 테이블별 레벨별로 UpsertBuilder에 등록 및 upsert 실행
352
- * 3. 순서 기반 uuid→id 매핑 (UpsertBuilder가 uuid를 DB에 저장하지 않으므로)
353
- *
354
- * UpsertBuilder는 self-reference가 있으면 buildInsertLevels() 재정렬하여
355
- * 등록 순서와 반환 순서가 달라질 수 있습니다. 이를 방지하기 위해
356
- * FixtureManager가 레벨별로 나눠서 처리하여 각 upsert 호출에서는
357
- * self-reference가 없도록 합니다.
358
- */ async insertFixtures(dbName, _fixtures) {
359
- const fixtures = unique(_fixtures, (f)=>f.fixtureId);
360
- // 초기화
361
- this.builder = new UpsertBuilder();
362
- this.fixtureRefMap = new Map();
363
- this.skippedFixtures = new Map();
364
- // 병렬 테스트 모드에서는 worker별 DB에 저장
365
- const dbConfig = process.env.SONAMU_WORKER_DB === "true" && process.env.VITEST_POOL_ID ? (()=>{
366
- const workerId = parseInt(process.env.VITEST_POOL_ID ?? "1", 10);
367
- const baseConfig = Sonamu.dbConfig[dbName];
368
- const connection = baseConfig.connection;
369
- return {
370
- ...baseConfig,
371
- connection: {
372
- ...connection,
373
- database: `${connection.database}_${workerId}`
374
- },
375
- pool: {
376
- min: 1,
377
- max: 1
378
- }
379
- };
380
- })() : Sonamu.dbConfig[dbName];
381
- const db = createKnexInstance(dbConfig);
382
- const results = [];
383
- try {
384
- // 1. RelationGraph로 fixture 단위 삽입 순서 계산
385
- this.relationGraph.buildGraph(fixtures);
386
- const insertionOrder = this.relationGraph.getInsertionOrder();
387
- // 2. 스킵할 fixture 먼저 처리 (override 체크)
388
- for (const fixtureId of insertionOrder){
389
- const fixture = fixtures.find((f)=>f.fixtureId === fixtureId);
390
- if (!fixture) continue;
391
- const hasTarget = !!fixture.target;
392
- const hasUnique = !!fixture.unique;
393
- const hasDuplicate = hasTarget || hasUnique;
394
- // 중복이 있고 override=false인 경우: 스킵
395
- if (hasDuplicate && !fixture.override) {
396
- // 기존 레코드 ID 저장 (unique 우선, 없으면 target)
397
- const existingId = fixture.unique?.id ?? fixture.target?.id;
398
- assert(existingId);
399
- this.skippedFixtures.set(fixtureId, {
400
- entityId: fixture.entityId,
401
- existingId
402
- });
403
- !isTest() && console.log(chalk.yellow(`Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`));
404
- }
405
- }
406
- // 3. 테이블별 fixture 그룹화 (insertionOrder 순서 기반)
407
- const fixturesByTable = new Map();
408
- const tableOrder = [];
409
- for (const fixtureId of insertionOrder){
410
- // 스킵된 fixture 제외
411
- if (this.skippedFixtures.has(fixtureId)) continue;
412
- const fixture = fixtures.find((f)=>f.fixtureId === fixtureId);
413
- if (!fixture) continue;
414
- const entity = EntityManager.get(fixture.entityId);
415
- const tableName = entity.table;
416
- if (!fixturesByTable.has(tableName)) {
417
- fixturesByTable.set(tableName, []);
418
- tableOrder.push(tableName);
419
- }
420
- fixturesByTable.get(tableName)?.push(fixture);
421
- }
422
- await db.transaction(async (trx)=>{
423
- const insertedIdsByTable = new Map();
424
- // 4. 테이블별 레벨별 처리
425
- for (const tableName of tableOrder){
426
- const tableFixtures = fixturesByTable.get(tableName) ?? [];
427
- const levels = this.groupFixturesByLevel(tableFixtures);
428
- for (const levelFixtures of levels){
429
- // 해당 레벨의 fixture들 register
430
- for (const fixture of levelFixtures){
431
- this.registerFixture(fixture, insertedIdsByTable);
432
- !isTest() && console.log(chalk.blue(`Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override)` : ""}`));
433
- }
434
- // upsert 실행 uuid 목록 저장
435
- const table = this.builder.getTable(tableName);
436
- const uuids = table.rows.map((row)=>row.uuid);
437
- !isTest() && console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows (level ${levels.indexOf(levelFixtures) + 1}/${levels.length})`));
438
- const ids = await this.builder.upsert(trx, tableName);
439
- // 순서 기반 uuid -> id 매핑
440
- // self-reference가 없으므로 등록 순서 = 반환 순서 보장
441
- if (uuids.length > 0 && uuids.length === ids.length) {
442
- const existingMap = insertedIdsByTable.get(tableName) ?? new Map();
443
- for(let i = 0; i < uuids.length; i++){
444
- existingMap.set(uuids[i], ids[i]);
445
- }
446
- insertedIdsByTable.set(tableName, existingMap);
447
- } else if (uuids.length !== ids.length) {
448
- console.warn(chalk.yellow(`Warning: uuid count (${uuids.length}) != id count (${ids.length}) for ${tableName}`));
449
- }
450
- }
451
- }
452
- // 5. ManyToMany 관계 처리
453
- await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);
454
- // 6. PostgreSQL 시퀀스 리셋
455
- // Fixture 삽입 각 테이블의 ID 시퀀스를 최대 ID 값으로 업데이트합니다.
456
- // 이렇게 하지 않으면 다음 INSERT 시 ID가 2000번대로 생성될 수 있습니다.
457
- !isTest() && console.log(chalk.blue("Resetting sequences..."));
458
- for (const tableName of tableOrder){
459
- try {
460
- // Entity를 찾아서 id 타입 확인
461
- const entity = EntityManager.getAllEntities().find((e)=>e.table === tableName || e.id.toLowerCase() === tableName);
462
- if (entity) {
463
- const idProp = entity.props.find((p)=>p.name === "id");
464
- const idType = idProp?.type;
465
- // integer/bigInteger이거나, string이지만 fixtureStrategy=sequence인 경우에만 리셋합니다
466
- const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
467
- if (!usesSequence) {
468
- !isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName} (id type: ${idType || "unknown"})`));
469
- continue;
470
- }
471
- }
472
- // 테이블의 최대 ID 조회 (string 타입은 숫자 캐스팅 필요)
473
- const entity2 = EntityManager.getAllEntities().find((e)=>e.table === tableName || e.id.toLowerCase() === tableName);
474
- const idType2 = entity2?.props.find((p)=>p.name === "id")?.type;
475
- const maxIdResult = idType2 === "string" ? await trx.raw(`SELECT MAX(id::bigint) as max_id FROM "${tableName}"`).then((r)=>r.rows[0]) : await trx(tableName).max("id as max_id").first();
476
- const maxId = maxIdResult?.max_id;
477
- if (maxId !== null && maxId !== undefined) {
478
- // 시퀀스명을 pg_get_serial_sequence로 안전하게 조회
479
- await trx.raw(`SELECT setval(pg_get_serial_sequence(?, 'id'), ?)`, [
480
- tableName,
481
- maxId
482
- ]);
483
- !isTest() && console.log(chalk.green(`Reset sequence for ${tableName}: ${maxId}`));
484
- }
485
- } catch (_err) {
486
- // 시퀀스가 없는 테이블(join table 등)은 무시
487
- !isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName}`));
488
- }
489
- }
490
- // 7. 결과 수집
491
- for (const fixture of fixtures){
492
- const entity = EntityManager.get(fixture.entityId);
493
- // 스킵된 fixture는 기존 레코드 정보로 결과 추가
494
- const skipped = this.skippedFixtures.get(fixture.fixtureId);
495
- if (skipped) {
496
- results.push({
497
- entityId: fixture.entityId,
498
- data: await trx(entity.table).where("id", skipped.existingId).first()
499
- });
500
- continue;
501
- }
502
- const ref = this.fixtureRefMap.get(fixture.fixtureId);
503
- if (ref) {
504
- const uuidToId = insertedIdsByTable.get(entity.table);
505
- const insertedId = uuidToId?.get(ref.uuid);
506
- if (insertedId !== undefined) {
507
- results.push({
508
- entityId: fixture.entityId,
509
- data: await trx(entity.table).where("id", insertedId).first()
510
- });
511
- !isTest() && console.log(chalk.green(`Inserted into ${entity.table}: #${fixture.id} -> #${insertedId}`));
512
- }
513
- }
514
- }
515
- });
516
- } finally{
517
- await db.destroy();
518
- }
519
- return unique(results, (r)=>`${r.entityId}#${r.data.id}`);
520
- }
521
- /**
522
- * FixtureRecord를 UpsertBuilder에 등록
523
- * @param insertedIdsByTable 이미 upsert된 테이블의 uuid→id 매핑 (레벨별 처리 시 사용)
524
- */ registerFixture(fixture, insertedIdsByTable) {
525
- const entity = EntityManager.get(fixture.entityId);
526
- const row = {};
527
- // Override 모드 판단: target 또는 unique가 있고 override=true인 경우
528
- const existingRecord = fixture.target ?? fixture.unique;
529
- const isOverrideMode = fixture.override && existingRecord;
530
- for (const [propName, column] of Object.entries(fixture.columns)){
531
- const prop = column.prop;
532
- if (isVirtualProp(prop)) {
533
- continue;
534
- }
535
- // Generated column은 INSERT에서 제외 (DB가 자동 생성)
536
- if ("generated" in prop && prop.generated) {
537
- continue;
538
- }
539
- // id 처리
540
- if (propName === "id") {
541
- const idProp = entity.props.find((p)=>p.name === "id");
542
- // parentId 엔티티의 id는 부모 엔티티의 FK이므로 시퀀스 미사용 (명시적으로 포함)
543
- const usesSequence = !entity.parentId && (idProp?.type === "integer" || idProp?.type === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence");
544
- if (isOverrideMode && existingRecord) {
545
- // Override: 기존 레코드의 값 사용 → UPDATE
546
- row[propName] = existingRecord.columns[propName]?.value;
547
- } else if (!usesSequence) {
548
- // string PK 또는 parentId 엔티티 FK: 생성된 id 값을 INSERT에 포함
549
- row[propName] = column.value;
550
- }
551
- continue;
552
- }
553
- if (isRelationProp(prop)) {
554
- if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) && prop.hasJoinColumn) {
555
- const relatedId = column.value;
556
- if (relatedId !== null && relatedId !== undefined) {
557
- const relatedFixtureId = `${prop.with}#${relatedId}`;
558
- // 먼저 skip된 fixture인지 확인
559
- const skippedExistingId = this.skippedFixtures.get(relatedFixtureId)?.existingId;
560
- if (skippedExistingId !== undefined) {
561
- // skip된 fixture → target DB의 기존 레코드 id 사용
562
- row[`${propName}_id`] = skippedExistingId;
563
- } else {
564
- const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
565
- if (relatedRef) {
566
- // 이미 upsert된 같은 테이블 fixture 확인
567
- const relatedEntity = EntityManager.get(prop.with);
568
- const relatedInsertedIds = insertedIdsByTable?.get(relatedEntity.table);
569
- const actualId = relatedInsertedIds?.get(relatedRef.uuid);
570
- if (actualId !== undefined) {
571
- // 이미 upsert됨 → 실제 ID 사용
572
- row[`${propName}_id`] = actualId;
573
- } else {
574
- // 아직 upsert 안됨 → UBRef 사용
575
- row[`${propName}_id`] = relatedRef;
576
- }
577
- } else {
578
- // fixtures에 포함되지 않은 레코드 → ID 그대로 사용
579
- row[`${propName}_id`] = relatedId;
580
- }
581
- }
582
- } else {
583
- row[`${propName}_id`] = null;
584
- }
585
- }
586
- // HasMany, ManyToMany는 별도 처리
587
- } else {
588
- // 일반 컬럼
589
- row[propName] = this.convertColumnValue(prop, column.value);
590
- }
591
- }
592
- !isTest() && console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));
593
- const ref = this.builder.register(entity.table, row);
594
- this.fixtureRefMap.set(fixture.fixtureId, ref);
595
- return ref;
596
- }
597
- /**
598
- * 컬럼 값 변환
599
- */ convertColumnValue(prop, value) {
600
- if (value === null || value === undefined) {
601
- return null;
602
- }
603
- switch(prop.type){
604
- case "json":
605
- // UpsertBuilder.register에서 JSON.stringify 처리하므로 object 그대로 전달
606
- return value;
607
- case "date":
608
- if (typeof value === "string" || typeof value === "number") {
609
- return new Date(value);
610
- }
611
- return value;
612
- default:
613
- return value;
614
- }
615
- }
616
- async processManyToManyRelations(trx, fixtures, insertedIdsByTable) {
617
- for (const fixture of fixtures){
618
- const entity = EntityManager.get(fixture.entityId);
619
- const sourceRef = this.fixtureRefMap.get(fixture.fixtureId);
620
- if (!sourceRef) continue;
621
- const sourceUuidToId = insertedIdsByTable.get(entity.table);
622
- const sourceId = sourceUuidToId?.get(sourceRef.uuid);
623
- if (sourceId === undefined) continue;
624
- for (const [, column] of Object.entries(fixture.columns)){
625
- const prop = column.prop;
626
- if (isManyToManyRelationProp(prop) && Array.isArray(column.value)) {
627
- // 선택되지 않은 ManyToMany 관계는 저장하지 않음
628
- const targetTable = EntityManager.get(prop.with);
629
- if (this.builder.hasTable(targetTable.table) === false) continue;
630
- const relatedIds = column.value;
631
- if (relatedIds.length === 0) continue;
632
- const joinTable = prop.joinTable;
633
- const relatedEntity = EntityManager.get(prop.with);
634
- const sourceColumn = `${inflection.singularize(entity.table)}_id`;
635
- const targetColumn = `${inflection.singularize(relatedEntity.table)}_id`;
636
- for (const relatedId of relatedIds){
637
- const relatedFixtureId = `${prop.with}#${relatedId}`;
638
- const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
639
- let targetId;
640
- if (relatedRef) {
641
- const relatedUuidToId = insertedIdsByTable.get(relatedEntity.table);
642
- const resolvedId = relatedUuidToId?.get(relatedRef.uuid);
643
- if (resolvedId === undefined) {
644
- console.warn(`Related fixture ${relatedFixtureId} not found in insertedIds, skipping`);
645
- continue;
646
- }
647
- targetId = resolvedId;
648
- } else {
649
- targetId = relatedId;
650
- }
651
- // JoinTable에 삽입
652
- const [found] = await trx(joinTable).where({
653
- [sourceColumn]: sourceId,
654
- [targetColumn]: targetId
655
- }).limit(1);
656
- if (!found) {
657
- await trx(joinTable).insert({
658
- [sourceColumn]: sourceId,
659
- [targetColumn]: targetId
660
- });
661
- !isTest() && console.log(chalk.green(`Inserted into ${joinTable}: ${entity.table}(${sourceId}) - ${relatedEntity.table}(${targetId})`));
662
- }
663
- }
664
- }
665
- }
666
- }
667
- }
668
- /**
669
- * 같은 테이블 내 fixture들을 self-reference 레벨별로 분할
670
- * - self-reference가 없는 fixture들: Level 0
671
- * - Level 0을 참조하는 fixture들: Level 1
672
- * - 반복
673
- *
674
- * UpsertBuilder가 self-reference가 있으면 buildInsertLevels()로 재정렬하여
675
- * 등록 순서와 반환 순서가 달라질 수 있습니다.
676
- * 이를 방지하기 위해 FixtureManager가 레벨별로 나눠서 처리합니다.
677
- */ groupFixturesByLevel(fixtures) {
678
- if (fixtures.length === 0) {
679
- return [];
680
- }
681
- const entity = EntityManager.get(fixtures[0].entityId);
682
- // self-reference relation prop 찾기
683
- const selfRefProps = entity.props.filter((p)=>isRelationProp(p) && (isBelongsToOneRelationProp(p) || isOneToOneRelationProp(p) && p.hasJoinColumn) && p.with === entity.id);
684
- if (selfRefProps.length === 0) {
685
- // self-reference 없음 → 단일 레벨
686
- return [
687
- fixtures
688
- ];
689
- }
690
- // 레벨별 분할 (topological sort)
691
- const levels = [];
692
- const remaining = new Set(fixtures.map((f)=>f.fixtureId));
693
- const processed = new Set();
694
- while(remaining.size > 0){
695
- const currentLevel = [];
696
- for (const fixture of fixtures){
697
- if (!remaining.has(fixture.fixtureId)) continue;
698
- // self-reference가 모두 이미 처리됐거나 null인 경우
699
- const canProcess = selfRefProps.every((prop)=>{
700
- const refId = fixture.columns[prop.name]?.value;
701
- if (refId === null || refId === undefined) return true;
702
- const refFixtureId = `${prop.with}#${refId}`;
703
- // 이미 처리됐거나, 현재 fixtures에 포함되지 않은 경우 (외부 참조)
704
- return processed.has(refFixtureId) || !remaining.has(refFixtureId);
705
- });
706
- if (canProcess) {
707
- currentLevel.push(fixture);
708
- }
709
- }
710
- if (currentLevel.length === 0) {
711
- const remainingIds = Array.from(remaining).join(", ");
712
- throw new Error(`Circular self-reference detected in ${entity.table}. Remaining fixtures: ${remainingIds}`);
713
- }
714
- for (const fixture of currentLevel){
715
- remaining.delete(fixture.fixtureId);
716
- processed.add(fixture.fixtureId);
717
- }
718
- levels.push(currentLevel);
719
- }
720
- return levels;
721
- }
722
- async checkUniqueViolation(db, entity, fixture) {
723
- const _uniqueIndexes = entity.indexes?.filter((i)=>i.type === "unique") ?? [];
724
- const uniqueIndexes = _uniqueIndexes.filter((index)=>index.columns.every((column)=>!column.name.startsWith(`${entity.table}__`)));
725
- if (uniqueIndexes.length === 0) {
726
- return null;
727
- }
728
- let uniqueQuery = db(entity.table);
729
- let hasCondition = false;
730
- for (const index of uniqueIndexes){
731
- // 컬럼 중 하나라도 null이면 유니크 제약을 위반하지 않기 때문에 해당 인덱스는 무시
732
- const containsNull = index.columns.some((column)=>{
733
- const field = column.name.replace(/_id$/, "");
734
- return fixture.columns[field]?.value === null;
735
- });
736
- if (containsNull) {
737
- continue;
738
- }
739
- uniqueQuery = uniqueQuery.orWhere((qb)=>{
740
- for (const column of index.columns){
741
- const field = column.name.replace(/_id$/, "");
742
- if (Array.isArray(fixture.columns[field]?.value)) {
743
- qb.whereIn(column.name, fixture.columns[field].value);
744
- } else {
745
- qb.andWhere(column.name, fixture.columns[field]?.value);
746
- }
747
- }
748
- });
749
- hasCondition = true;
750
- }
751
- if (!hasCondition) {
752
- return null;
753
- }
754
- const [uniqueFound] = await uniqueQuery;
755
- return uniqueFound;
756
- }
757
- async checkDuplicateByColumns(db, entity, fixture, columns) {
758
- if (columns.length === 0) {
759
- return null;
760
- }
761
- const whereClause = {};
762
- for (const column of columns){
763
- // relation 필드인 경우 _id 붙이기
764
- const prop = entity.props.find((p)=>p.name === column);
765
- const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;
766
- const value = fixture.columns[column]?.value;
767
- // null 값이 포함된 경우 중복 확인 스킵
768
- if (value === null || value === undefined) {
769
- return null;
770
- }
771
- whereClause[dbColumn] = value;
772
- }
773
- const [found] = await db(entity.table).where(whereClause).limit(1);
774
- return found;
775
- }
776
- async addFixtureLoader(code) {
777
- const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;
778
- const content = readFileSync(path).toString();
779
- const fixtureLoaderStart = content.indexOf("const fixtureLoader = {");
780
- const fixtureLoaderEnd = content.indexOf("};", fixtureLoaderStart);
781
- if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {
782
- const newContent = `${content.slice(0, fixtureLoaderEnd)} ${code}\n${content.slice(fixtureLoaderEnd)}`;
783
- writeFileSync(path, newContent);
784
- } else {
785
- throw new Error("Failed to find fixtureLoader in fixture.ts");
786
- }
787
- }
788
- }
789
- export const FixtureManager = new FixtureManagerClass();
140
+ }
141
+ } finally {
142
+ await testDb.destroy();
143
+ }
144
+ }
145
+ visitedRecords = new Set();
146
+ async importFixture(entityId, ids) {
147
+ this.visitedRecords.clear();
148
+ const queries = unique((await Promise.all(ids.map(async (id) => {
149
+ return await this.getImportQueries(entityId, "id", id);
150
+ }))).flat());
151
+ const wdb = BaseModel.getDB("w");
152
+ for (const query of queries) {
153
+ await wdb.raw(query);
154
+ }
155
+ }
156
+ async getImportQueries(entityId, field, id) {
157
+ const recordKey = `${entityId}#${field}#${id}`;
158
+ if (this.visitedRecords.has(recordKey)) {
159
+ return [];
160
+ }
161
+ this.visitedRecords.add(recordKey);
162
+ const entity = EntityManager.get(entityId);
163
+ const wdb = BaseModel.getDB("w");
164
+ const [row] = await wdb(entity.table).where(field, id).limit(1);
165
+ if (row === undefined) {
166
+ throw new Error(`${entityId}#${id} row 찾을 수 없습니다.`);
167
+ }
168
+ const fixtureDatabase = Sonamu.dbConfig.fixture.connection.database;
169
+ const realDatabase = Sonamu.dbConfig.production_master.connection.database;
170
+ const selfQuery = `INSERT IGNORE INTO \`${fixtureDatabase}\`.\`${entity.table}\` (SELECT * FROM \`${realDatabase}\`.\`${entity.table}\` WHERE \`id\` = ${id})`;
171
+ const args = Object.entries(entity.relations).filter(([, relation]) => isBelongsToOneRelationProp(relation) || isOneToOneRelationProp(relation) && relation.customJoinClause === undefined).map(([, relation]) => {
172
+ let field$1;
173
+ let id$1;
174
+ if (isOneToOneRelationProp(relation) && !relation.hasJoinColumn) {
175
+ const relatedEntity = EntityManager.get(relation.with);
176
+ const relatedIdColumnName = relatedEntity.props.find((p) => isRelationProp(p) && p.with === entity.id)?.name;
177
+ if (!relatedIdColumnName) {
178
+ throw new Error(`${relatedEntity.id}의 ${entity.id} 관계 프롭을 찾을 수 없습니다.`);
179
+ }
180
+ field$1 = `${relatedIdColumnName}_id`;
181
+ id$1 = row.id;
182
+ } else {
183
+ field$1 = "id";
184
+ id$1 = row[`${relation.name}_id`];
185
+ }
186
+ return {
187
+ entityId: relation.with,
188
+ field: field$1,
189
+ id: id$1
190
+ };
191
+ }).filter((arg) => arg.id !== null);
192
+ const relQueries = await Promise.all(args.map(async (args$1) => {
193
+ return this.getImportQueries(args$1.entityId, args$1.field, args$1.id);
194
+ }));
195
+ return [...unique(relQueries.toReversed().flat()), selfQuery];
196
+ }
197
+ async destroy() {
198
+ if (this._tdb) {
199
+ await this._tdb.destroy();
200
+ this._tdb = null;
201
+ }
202
+ if (this._fdb) {
203
+ await this._fdb.destroy();
204
+ this._fdb = null;
205
+ }
206
+ await BaseModel.destroy();
207
+ }
208
+ async getFixtures(sourceDBName, targetDBName, searchOptions, duplicateCheck) {
209
+ const sourceDB = createKnexInstance(Sonamu.dbConfig[sourceDBName]);
210
+ const targetDB = createKnexInstance(Sonamu.dbConfig[targetDBName]);
211
+ try {
212
+ const { entityId, field, value, searchType } = searchOptions;
213
+ const entity = EntityManager.get(entityId);
214
+ const column = entity.props.find((prop) => prop.name === field)?.type === "relation" ? `${field}_id` : field;
215
+ let query = sourceDB(entity.table);
216
+ if (searchType === "equals") {
217
+ query = query.where(column, value);
218
+ } else if (searchType === "like") {
219
+ query = query.where(column, "like", `%${value}%`);
220
+ }
221
+ const rows = await query;
222
+ if (rows.length === 0) {
223
+ throw new Error("No records found");
224
+ }
225
+ const fixtures = [];
226
+ for (const row of rows) {
227
+ const initialRecordsLength = fixtures.length;
228
+ const newRecords = await this.createFixtureRecord(entity, row, { _db: sourceDB });
229
+ fixtures.push(...newRecords);
230
+ const currentFixtureRecord = fixtures.find((r) => r.fixtureId === `${entityId}#${row.id}`);
231
+ if (currentFixtureRecord) {
232
+ currentFixtureRecord.fetchedRecords = fixtures.filter((r) => r.fixtureId !== currentFixtureRecord.fixtureId).slice(initialRecordsLength).map((r) => r.fixtureId);
233
+ }
234
+ }
235
+ for await (const fixture of fixtures) {
236
+ const entity$1 = EntityManager.get(fixture.entityId);
237
+ const customColumns = duplicateCheck?.columns?.[fixture.entityId];
238
+ if (customColumns && customColumns.length > 0) {
239
+ const customDuplicateRow = await this.checkDuplicateByColumns(targetDB, entity$1, fixture, customColumns);
240
+ if (customDuplicateRow) {
241
+ const [record] = await this.createFixtureRecord(entity$1, customDuplicateRow, {
242
+ singleRecord: true,
243
+ _db: targetDB
244
+ });
245
+ fixture.target = record;
246
+ }
247
+ }
248
+ const uniqueRow = await this.checkUniqueViolation(targetDB, entity$1, fixture);
249
+ if (uniqueRow) {
250
+ const [record] = await this.createFixtureRecord(entity$1, uniqueRow, {
251
+ singleRecord: true,
252
+ _db: targetDB
253
+ });
254
+ fixture.unique = record;
255
+ }
256
+ }
257
+ return unique(fixtures, (f) => f.fixtureId);
258
+ } finally {
259
+ await Promise.allSettled([targetDB.destroy(), sourceDB.destroy()]);
260
+ }
261
+ }
262
+ async createFixtureRecord(entity, row, options) {
263
+ const records = [];
264
+ const visitedEntities = new Set();
265
+ const create = async (entity$1, row$1) => {
266
+ const fixtureId = `${entity$1.id}#${row$1.id}`;
267
+ if (visitedEntities.has(fixtureId)) {
268
+ return;
269
+ }
270
+ visitedEntities.add(fixtureId);
271
+ const record = {
272
+ fixtureId,
273
+ entityId: entity$1.id,
274
+ id: row$1.id,
275
+ columns: {},
276
+ fetchedRecords: [],
277
+ belongsRecords: []
278
+ };
279
+ for (const prop of entity$1.props) {
280
+ if (isVirtualProp(prop)) {
281
+ continue;
282
+ }
283
+ record.columns[prop.name] = {
284
+ prop,
285
+ value: row$1[prop.name]
286
+ };
287
+ const db = options?._db ?? BaseModel.getDB("w");
288
+ if (isManyToManyRelationProp(prop)) {
289
+ const relatedEntity = EntityManager.get(prop.with);
290
+ const throughTable = prop.joinTable;
291
+ const fromColumn = `${inflection.singularize(entity$1.table)}_id`;
292
+ const toColumn = `${inflection.singularize(relatedEntity.table)}_id`;
293
+ const relatedIds = await db(throughTable).where(fromColumn, row$1.id).pluck(toColumn);
294
+ record.columns[prop.name].value = relatedIds;
295
+ } else if (isHasManyRelationProp(prop)) {
296
+ const relatedEntity = EntityManager.get(prop.with);
297
+ const relatedIds = await db(relatedEntity.table).where(prop.joinColumn, row$1.id).pluck("id");
298
+ record.columns[prop.name].value = relatedIds;
299
+ } else if (isOneToOneRelationProp(prop) && !prop.hasJoinColumn) {
300
+ const relatedEntity = EntityManager.get(prop.with);
301
+ const relatedProp = relatedEntity.props.find((p) => isRelationProp(p) && p.with === entity$1.id);
302
+ if (relatedProp && isRelationProp(relatedProp)) {
303
+ const fkColumn = `${relatedProp.name}_id`;
304
+ const relatedRow = await db(relatedEntity.table).where(fkColumn, row$1.id).first();
305
+ record.columns[prop.name].value = relatedRow?.id;
306
+ }
307
+ } else if (isRelationProp(prop)) {
308
+ const relatedId = row$1[`${prop.name}_id`];
309
+ record.columns[prop.name].value = relatedId;
310
+ if (relatedId) {
311
+ record.belongsRecords.push(`${prop.with}#${relatedId}`);
312
+ }
313
+ if (!options?.singleRecord && relatedId) {
314
+ const relatedEntity = EntityManager.get(prop.with);
315
+ const relatedRow = await db(relatedEntity.table).where("id", relatedId).first();
316
+ if (relatedRow) {
317
+ await create(relatedEntity, relatedRow);
318
+ }
319
+ }
320
+ }
321
+ }
322
+ records.push(record);
323
+ };
324
+ await create(entity, row);
325
+ return records;
326
+ }
327
+ /**
328
+ * 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)
329
+ * 2. 테이블별 레벨별로 UpsertBuilder에 등록 및 upsert 실행
330
+ * 3. 순서 기반 uuid→id 매핑 (UpsertBuilder가 uuid를 DB에 저장하지 않으므로)
331
+ *
332
+ * UpsertBuilder는 self-reference가 있으면 buildInsertLevels() 재정렬하여
333
+ * 등록 순서와 반환 순서가 달라질 수 있습니다. 이를 방지하기 위해
334
+ * FixtureManager가 레벨별로 나눠서 처리하여 각 upsert 호출에서는
335
+ * self-reference가 없도록 합니다.
336
+ */
337
+ async insertFixtures(dbName, _fixtures) {
338
+ const fixtures = unique(_fixtures, (f) => f.fixtureId);
339
+ this.builder = new UpsertBuilder();
340
+ this.fixtureRefMap = new Map();
341
+ this.skippedFixtures = new Map();
342
+ const dbConfig = process.env.SONAMU_WORKER_DB === "true" && process.env.VITEST_POOL_ID ? (() => {
343
+ const workerId = parseInt(process.env.VITEST_POOL_ID ?? "1", 10);
344
+ const baseConfig = Sonamu.dbConfig[dbName];
345
+ const connection = baseConfig.connection;
346
+ return {
347
+ ...baseConfig,
348
+ connection: {
349
+ ...connection,
350
+ database: `${connection.database}_${workerId}`
351
+ },
352
+ pool: {
353
+ min: 1,
354
+ max: 1
355
+ }
356
+ };
357
+ })() : Sonamu.dbConfig[dbName];
358
+ const db = createKnexInstance(dbConfig);
359
+ const results = [];
360
+ try {
361
+ this.relationGraph.buildGraph(fixtures);
362
+ const insertionOrder = this.relationGraph.getInsertionOrder();
363
+ for (const fixtureId of insertionOrder) {
364
+ const fixture = fixtures.find((f) => f.fixtureId === fixtureId);
365
+ if (!fixture) continue;
366
+ const hasTarget = !!fixture.target;
367
+ const hasUnique = !!fixture.unique;
368
+ const hasDuplicate = hasTarget || hasUnique;
369
+ if (hasDuplicate && !fixture.override) {
370
+ const existingId = fixture.unique?.id ?? fixture.target?.id;
371
+ assert(existingId);
372
+ this.skippedFixtures.set(fixtureId, {
373
+ entityId: fixture.entityId,
374
+ existingId
375
+ });
376
+ !isTest() && console.log(chalk.yellow(`Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`));
377
+ }
378
+ }
379
+ const fixturesByTable = new Map();
380
+ const tableOrder = [];
381
+ for (const fixtureId of insertionOrder) {
382
+ if (this.skippedFixtures.has(fixtureId)) continue;
383
+ const fixture = fixtures.find((f) => f.fixtureId === fixtureId);
384
+ if (!fixture) continue;
385
+ const entity = EntityManager.get(fixture.entityId);
386
+ const tableName = entity.table;
387
+ if (!fixturesByTable.has(tableName)) {
388
+ fixturesByTable.set(tableName, []);
389
+ tableOrder.push(tableName);
390
+ }
391
+ fixturesByTable.get(tableName)?.push(fixture);
392
+ }
393
+ await db.transaction(async (trx) => {
394
+ const insertedIdsByTable = new Map();
395
+ for (const tableName of tableOrder) {
396
+ const tableFixtures = fixturesByTable.get(tableName) ?? [];
397
+ const levels = this.groupFixturesByLevel(tableFixtures);
398
+ for (const levelFixtures of levels) {
399
+ for (const fixture of levelFixtures) {
400
+ this.registerFixture(fixture, insertedIdsByTable);
401
+ !isTest() && console.log(chalk.blue(`Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override)` : ""}`));
402
+ }
403
+ const table = this.builder.getTable(tableName);
404
+ const uuids = table.rows.map((row) => row.uuid);
405
+ !isTest() && console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows (level ${levels.indexOf(levelFixtures) + 1}/${levels.length})`));
406
+ const ids = await this.builder.upsert(trx, tableName);
407
+ if (uuids.length > 0 && uuids.length === ids.length) {
408
+ const existingMap = insertedIdsByTable.get(tableName) ?? new Map();
409
+ for (let i = 0; i < uuids.length; i++) {
410
+ existingMap.set(uuids[i], ids[i]);
411
+ }
412
+ insertedIdsByTable.set(tableName, existingMap);
413
+ } else if (uuids.length !== ids.length) {
414
+ console.warn(chalk.yellow(`Warning: uuid count (${uuids.length}) != id count (${ids.length}) for ${tableName}`));
415
+ }
416
+ }
417
+ }
418
+ await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);
419
+ !isTest() && console.log(chalk.blue("Resetting sequences..."));
420
+ for (const tableName of tableOrder) {
421
+ try {
422
+ const entity = EntityManager.getAllEntities().find((e) => e.table === tableName || e.id.toLowerCase() === tableName);
423
+ if (entity) {
424
+ const idProp = entity.props.find((p) => p.name === "id");
425
+ const idType = idProp?.type;
426
+ const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
427
+ if (!usesSequence) {
428
+ !isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName} (id type: ${idType || "unknown"})`));
429
+ continue;
430
+ }
431
+ }
432
+ const entity2 = EntityManager.getAllEntities().find((e) => e.table === tableName || e.id.toLowerCase() === tableName);
433
+ const idType2 = entity2?.props.find((p) => p.name === "id")?.type;
434
+ const maxIdResult = idType2 === "string" ? await trx.raw(`SELECT MAX(id::bigint) as max_id FROM "${tableName}"`).then((r) => r.rows[0]) : await trx(tableName).max("id as max_id").first();
435
+ const maxId = maxIdResult?.max_id;
436
+ if (maxId !== null && maxId !== undefined) {
437
+ await trx.raw(`SELECT setval(pg_get_serial_sequence(?, 'id'), ?)`, [tableName, maxId]);
438
+ !isTest() && console.log(chalk.green(`Reset sequence for ${tableName}: ${maxId}`));
439
+ }
440
+ } catch (_err) {
441
+ !isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName}`));
442
+ }
443
+ }
444
+ for (const fixture of fixtures) {
445
+ const entity = EntityManager.get(fixture.entityId);
446
+ const skipped = this.skippedFixtures.get(fixture.fixtureId);
447
+ if (skipped) {
448
+ results.push({
449
+ entityId: fixture.entityId,
450
+ data: await trx(entity.table).where("id", skipped.existingId).first()
451
+ });
452
+ continue;
453
+ }
454
+ const ref = this.fixtureRefMap.get(fixture.fixtureId);
455
+ if (ref) {
456
+ const uuidToId = insertedIdsByTable.get(entity.table);
457
+ const insertedId = uuidToId?.get(ref.uuid);
458
+ if (insertedId !== undefined) {
459
+ results.push({
460
+ entityId: fixture.entityId,
461
+ data: await trx(entity.table).where("id", insertedId).first()
462
+ });
463
+ !isTest() && console.log(chalk.green(`Inserted into ${entity.table}: #${fixture.id} -> #${insertedId}`));
464
+ }
465
+ }
466
+ }
467
+ });
468
+ } finally {
469
+ await db.destroy();
470
+ }
471
+ return unique(results, (r) => `${r.entityId}#${r.data.id}`);
472
+ }
473
+ /**
474
+ * FixtureRecord를 UpsertBuilder에 등록
475
+ * @param insertedIdsByTable 이미 upsert된 테이블의 uuid→id 매핑 (레벨별 처리 사용)
476
+ */
477
+ registerFixture(fixture, insertedIdsByTable) {
478
+ const entity = EntityManager.get(fixture.entityId);
479
+ const row = {};
480
+ const existingRecord = fixture.target ?? fixture.unique;
481
+ const isOverrideMode = fixture.override && existingRecord;
482
+ for (const [propName, column] of Object.entries(fixture.columns)) {
483
+ const prop = column.prop;
484
+ if (isVirtualProp(prop)) {
485
+ continue;
486
+ }
487
+ if ("generated" in prop && prop.generated) {
488
+ continue;
489
+ }
490
+ if (propName === "id") {
491
+ const idProp = entity.props.find((p) => p.name === "id");
492
+ const usesSequence = !entity.parentId && (idProp?.type === "integer" || idProp?.type === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence");
493
+ if (isOverrideMode && existingRecord) {
494
+ row[propName] = existingRecord.columns[propName]?.value;
495
+ } else if (!usesSequence) {
496
+ row[propName] = column.value;
497
+ }
498
+ continue;
499
+ }
500
+ if (isRelationProp(prop)) {
501
+ if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) && prop.hasJoinColumn) {
502
+ const relatedId = column.value;
503
+ if (relatedId !== null && relatedId !== undefined) {
504
+ const relatedFixtureId = `${prop.with}#${relatedId}`;
505
+ const skippedExistingId = this.skippedFixtures.get(relatedFixtureId)?.existingId;
506
+ if (skippedExistingId !== undefined) {
507
+ row[`${propName}_id`] = skippedExistingId;
508
+ } else {
509
+ const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
510
+ if (relatedRef) {
511
+ const relatedEntity = EntityManager.get(prop.with);
512
+ const relatedInsertedIds = insertedIdsByTable?.get(relatedEntity.table);
513
+ const actualId = relatedInsertedIds?.get(relatedRef.uuid);
514
+ if (actualId !== undefined) {
515
+ row[`${propName}_id`] = actualId;
516
+ } else {
517
+ row[`${propName}_id`] = relatedRef;
518
+ }
519
+ } else {
520
+ row[`${propName}_id`] = relatedId;
521
+ }
522
+ }
523
+ } else {
524
+ row[`${propName}_id`] = null;
525
+ }
526
+ }
527
+ } else {
528
+ row[propName] = this.convertColumnValue(prop, column.value);
529
+ }
530
+ }
531
+ !isTest() && console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));
532
+ const ref = this.builder.register(entity.table, row);
533
+ this.fixtureRefMap.set(fixture.fixtureId, ref);
534
+ return ref;
535
+ }
536
+ /**
537
+ * 컬럼 값 변환
538
+ */
539
+ convertColumnValue(prop, value) {
540
+ if (value === null || value === undefined) {
541
+ return null;
542
+ }
543
+ switch (prop.type) {
544
+ case "json": return value;
545
+ case "date":
546
+ if (typeof value === "string" || typeof value === "number") {
547
+ return new Date(value);
548
+ }
549
+ return value;
550
+ default: return value;
551
+ }
552
+ }
553
+ async processManyToManyRelations(trx, fixtures, insertedIdsByTable) {
554
+ for (const fixture of fixtures) {
555
+ const entity = EntityManager.get(fixture.entityId);
556
+ const sourceRef = this.fixtureRefMap.get(fixture.fixtureId);
557
+ if (!sourceRef) continue;
558
+ const sourceUuidToId = insertedIdsByTable.get(entity.table);
559
+ const sourceId = sourceUuidToId?.get(sourceRef.uuid);
560
+ if (sourceId === undefined) continue;
561
+ for (const [, column] of Object.entries(fixture.columns)) {
562
+ const prop = column.prop;
563
+ if (isManyToManyRelationProp(prop) && Array.isArray(column.value)) {
564
+ const targetTable = EntityManager.get(prop.with);
565
+ if (!this.builder.hasTable(targetTable.table)) continue;
566
+ const relatedIds = column.value;
567
+ if (relatedIds.length === 0) continue;
568
+ const joinTable = prop.joinTable;
569
+ const relatedEntity = EntityManager.get(prop.with);
570
+ const sourceColumn = `${inflection.singularize(entity.table)}_id`;
571
+ const targetColumn = `${inflection.singularize(relatedEntity.table)}_id`;
572
+ for (const relatedId of relatedIds) {
573
+ const relatedFixtureId = `${prop.with}#${relatedId}`;
574
+ const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
575
+ let targetId;
576
+ if (relatedRef) {
577
+ const relatedUuidToId = insertedIdsByTable.get(relatedEntity.table);
578
+ const resolvedId = relatedUuidToId?.get(relatedRef.uuid);
579
+ if (resolvedId === undefined) {
580
+ console.warn(`Related fixture ${relatedFixtureId} not found in insertedIds, skipping`);
581
+ continue;
582
+ }
583
+ targetId = resolvedId;
584
+ } else {
585
+ targetId = relatedId;
586
+ }
587
+ const [found] = await trx(joinTable).where({
588
+ [sourceColumn]: sourceId,
589
+ [targetColumn]: targetId
590
+ }).limit(1);
591
+ if (!found) {
592
+ await trx(joinTable).insert({
593
+ [sourceColumn]: sourceId,
594
+ [targetColumn]: targetId
595
+ });
596
+ !isTest() && console.log(chalk.green(`Inserted into ${joinTable}: ${entity.table}(${sourceId}) - ${relatedEntity.table}(${targetId})`));
597
+ }
598
+ }
599
+ }
600
+ }
601
+ }
602
+ }
603
+ /**
604
+ * 같은 테이블 내 fixture들을 self-reference 레벨별로 분할
605
+ * - self-reference가 없는 fixture들: Level 0
606
+ * - Level 0을 참조하는 fixture들: Level 1
607
+ * - 반복
608
+ *
609
+ * UpsertBuilder가 self-reference가 있으면 buildInsertLevels()로 재정렬하여
610
+ * 등록 순서와 반환 순서가 달라질 수 있습니다.
611
+ * 이를 방지하기 위해 FixtureManager가 레벨별로 나눠서 처리합니다.
612
+ */
613
+ groupFixturesByLevel(fixtures) {
614
+ if (fixtures.length === 0) {
615
+ return [];
616
+ }
617
+ const entity = EntityManager.get(fixtures[0].entityId);
618
+ const selfRefProps = entity.props.filter((p) => isRelationProp(p) && (isBelongsToOneRelationProp(p) || isOneToOneRelationProp(p) && p.hasJoinColumn) && p.with === entity.id);
619
+ if (selfRefProps.length === 0) {
620
+ return [fixtures];
621
+ }
622
+ const levels = [];
623
+ const remaining = new Set(fixtures.map((f) => f.fixtureId));
624
+ const processed = new Set();
625
+ while (remaining.size > 0) {
626
+ const currentLevel = [];
627
+ for (const fixture of fixtures) {
628
+ if (!remaining.has(fixture.fixtureId)) continue;
629
+ const canProcess = selfRefProps.every((prop) => {
630
+ const refId = fixture.columns[prop.name]?.value;
631
+ if (refId === null || refId === undefined) return true;
632
+ const refFixtureId = `${prop.with}#${refId}`;
633
+ return processed.has(refFixtureId) || !remaining.has(refFixtureId);
634
+ });
635
+ if (canProcess) {
636
+ currentLevel.push(fixture);
637
+ }
638
+ }
639
+ if (currentLevel.length === 0) {
640
+ const remainingIds = Array.from(remaining).join(", ");
641
+ throw new Error(`Circular self-reference detected in ${entity.table}. Remaining fixtures: ${remainingIds}`);
642
+ }
643
+ for (const fixture of currentLevel) {
644
+ remaining.delete(fixture.fixtureId);
645
+ processed.add(fixture.fixtureId);
646
+ }
647
+ levels.push(currentLevel);
648
+ }
649
+ return levels;
650
+ }
651
+ async checkUniqueViolation(db, entity, fixture) {
652
+ const _uniqueIndexes = entity.indexes?.filter((i) => i.type === "unique") ?? [];
653
+ const uniqueIndexes = _uniqueIndexes.filter((index) => index.columns.every((column) => !column.name.startsWith(`${entity.table}__`)));
654
+ if (uniqueIndexes.length === 0) {
655
+ return null;
656
+ }
657
+ let uniqueQuery = db(entity.table);
658
+ let hasCondition = false;
659
+ for (const index of uniqueIndexes) {
660
+ const containsNull = index.columns.some((column) => {
661
+ const field = column.name.replace(/_id$/, "");
662
+ return fixture.columns[field]?.value === null;
663
+ });
664
+ if (containsNull) {
665
+ continue;
666
+ }
667
+ uniqueQuery = uniqueQuery.orWhere((qb) => {
668
+ for (const column of index.columns) {
669
+ const field = column.name.replace(/_id$/, "");
670
+ if (Array.isArray(fixture.columns[field]?.value)) {
671
+ qb.whereIn(column.name, fixture.columns[field].value);
672
+ } else {
673
+ qb.andWhere(column.name, fixture.columns[field]?.value);
674
+ }
675
+ }
676
+ });
677
+ hasCondition = true;
678
+ }
679
+ if (!hasCondition) {
680
+ return null;
681
+ }
682
+ const [uniqueFound] = await uniqueQuery;
683
+ return uniqueFound;
684
+ }
685
+ async checkDuplicateByColumns(db, entity, fixture, columns) {
686
+ if (columns.length === 0) {
687
+ return null;
688
+ }
689
+ const whereClause = {};
690
+ for (const column of columns) {
691
+ const prop = entity.props.find((p) => p.name === column);
692
+ const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;
693
+ const value = fixture.columns[column]?.value;
694
+ if (value === null || value === undefined) {
695
+ return null;
696
+ }
697
+ whereClause[dbColumn] = value;
698
+ }
699
+ const [found] = await db(entity.table).where(whereClause).limit(1);
700
+ return found;
701
+ }
702
+ async addFixtureLoader(code) {
703
+ const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;
704
+ const content = readFileSync(path).toString();
705
+ const fixtureLoaderStart = content.indexOf("const fixtureLoader = {");
706
+ const fixtureLoaderEnd = content.indexOf("};", fixtureLoaderStart);
707
+ if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {
708
+ const newContent = `${content.slice(0, fixtureLoaderEnd)} ${code}\n${content.slice(fixtureLoaderEnd)}`;
709
+ writeFileSync(path, newContent);
710
+ } else {
711
+ throw new Error("Failed to find fixtureLoader in fixture.ts");
712
+ }
713
+ }
714
+ };
715
+ FixtureManager = new FixtureManagerClass();
716
+ }));
790
717
 
791
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90ZXN0aW5nL2ZpeHR1cmUtbWFuYWdlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgYXNzZXJ0IGZyb20gXCJhc3NlcnRcIjtcbmltcG9ydCBjaGFsayBmcm9tIFwiY2hhbGtcIjtcbmltcG9ydCB7IGV4ZWNTeW5jIH0gZnJvbSBcImNoaWxkX3Byb2Nlc3NcIjtcbmltcG9ydCB7IHJlYWRGaWxlU3luYywgd3JpdGVGaWxlU3luYyB9IGZyb20gXCJmc1wiO1xuaW1wb3J0IGluZmxlY3Rpb24gZnJvbSBcImluZmxlY3Rpb25cIjtcbmltcG9ydCB0eXBlIHsgS25leCB9IGZyb20gXCJrbmV4XCI7XG5pbXBvcnQgeyB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgaW5zcGVjdCB9IGZyb20gXCJ1dGlsXCI7XG5pbXBvcnQgeyBTb25hbXUgfSBmcm9tIFwiLi4vYXBpXCI7XG5pbXBvcnQgeyBCYXNlTW9kZWwgfSBmcm9tIFwiLi4vZGF0YWJhc2UvYmFzZS1tb2RlbFwiO1xuaW1wb3J0IHR5cGUgeyBTb25hbXVEQkNvbmZpZyB9IGZyb20gXCIuLi9kYXRhYmFzZS9kYlwiO1xuaW1wb3J0IHsgY3JlYXRlS25leEluc3RhbmNlIH0gZnJvbSBcIi4uL2RhdGFiYXNlL2tuZXhcIjtcbmltcG9ydCB7IHR5cGUgVUJSZWYsIFVwc2VydEJ1aWxkZXIgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB0eXBlIHsgRW50aXR5IH0gZnJvbSBcIi4uL2VudGl0eS9lbnRpdHlcIjtcbmltcG9ydCB7IEVudGl0eU1hbmFnZXIgfSBmcm9tIFwiLi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5pbXBvcnQge1xuICB0eXBlIEJlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCxcbiAgdHlwZSBEYXRhYmFzZVNjaGVtYUV4dGVuZCxcbiAgdHlwZSBFbnRpdHlQcm9wLFxuICB0eXBlIEZpeHR1cmVJbXBvcnRSZXN1bHQsXG4gIHR5cGUgRml4dHVyZVJlY29yZCxcbiAgdHlwZSBGaXh0dXJlU2VhcmNoT3B0aW9ucyxcbiAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AsXG4gIGlzSGFzTWFueVJlbGF0aW9uUHJvcCxcbiAgaXNNYW55VG9NYW55UmVsYXRpb25Qcm9wLFxuICBpc09uZVRvT25lUmVsYXRpb25Qcm9wLFxuICBpc1JlbGF0aW9uUHJvcCxcbiAgaXNWaXJ0dWFsUHJvcCxcbiAgdHlwZSBNYW55VG9NYW55UmVsYXRpb25Qcm9wLFxuICB0eXBlIE9uZVRvT25lUmVsYXRpb25Qcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7IGlzVGVzdCB9IGZyb20gXCIuLi91dGlscy9jb250cm9sbGVyXCI7XG5pbXBvcnQgeyBSZWxhdGlvbkdyYXBoIH0gZnJvbSBcIi4vX3JlbGF0aW9uLWdyYXBoXCI7XG5cbi8qKiDsgqzsmqnsnpAg7KeA7KCVIOykkeuztSDtmZXsnbgg7Lus65+8IChlbnRpdHlJZOuzhOuhnCDsp4DsoJUpICovXG5leHBvcnQgaW50ZXJmYWNlIER1cGxpY2F0ZUNoZWNrT3B0aW9ucyB7XG4gIGNvbHVtbnM/OiB7XG4gICAgW2VudGl0eUlkOiBzdHJpbmddOiBzdHJpbmdbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGNsYXNzIEZpeHR1cmVNYW5hZ2VyQ2xhc3Mge1xuICBwcml2YXRlIF90ZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IHRkYih0ZGI6IEtuZXgpIHtcbiAgICB0aGlzLl90ZGIgPSB0ZGI7XG4gIH1cbiAgZ2V0IHRkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fdGRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl90ZGI7XG4gIH1cblxuICBwcml2YXRlIF9mZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IGZkYihmZGI6IEtuZXgpIHtcbiAgICB0aGlzLl9mZGIgPSBmZGI7XG4gIH1cbiAgZ2V0IGZkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fZmRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9mZGI7XG4gIH1cbiAgY2FjaGVkVGFibGVOYW1lczogc3RyaW5nW10gfCBudWxsID0gbnVsbDtcblxuICBwcml2YXRlIHJlbGF0aW9uR3JhcGggPSBuZXcgUmVsYXRpb25HcmFwaCgpO1xuXG4gIC8vIFVwc2VydEJ1aWxkZXIg6riw67CYIGltcG9ydOulvCDsnITtlZwg7IOB7YOcXG4gIHByaXZhdGUgYnVpbGRlcjogVXBzZXJ0QnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gIHByaXZhdGUgZml4dHVyZVJlZk1hcDogTWFwPHN0cmluZywgVUJSZWY+ID0gbmV3IE1hcCgpO1xuICBwcml2YXRlIHNraXBwZWRGaXh0dXJlczogTWFwPHN0cmluZywgeyBlbnRpdHlJZDogc3RyaW5nOyBleGlzdGluZ0lkOiBudW1iZXIgfCBzdHJpbmcgfT4gPVxuICAgIG5ldyBNYXAoKTtcblxuICBpbml0KCkge1xuICAgIGlmICh0aGlzLl90ZGIgIT09IG51bGwpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKFNvbmFtdS5kYkNvbmZpZy50ZXN0ICYmIFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlcikge1xuICAgICAgY29uc3QgdENvbm4gPSBTb25hbXUuZGJDb25maWcudGVzdC5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBjb25zdCBwQ29ubiA9IFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlci5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBpZiAoXG4gICAgICAgIGAke3RDb25uLmhvc3QgPz8gXCJsb2NhbGhvc3RcIn06JHt0Q29ubi5wb3J0ID8/IDU0MzJ9LyR7dENvbm4uZGF0YWJhc2V9YCA9PT1cbiAgICAgICAgYCR7cENvbm4uaG9zdCA/PyBcImxvY2FsaG9zdFwifToke3BDb25uLnBvcnQgPz8gNTQzMn0vJHtwQ29ubi5kYXRhYmFzZX1gXG4gICAgICApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDthYzsiqTtirhEQuyZgCDtlITroZzrjZXshZhEQuyXkCDrj5nsnbztlZwg642w7J207YSw67Kg7J207Iqk6rCAIOyCrOyaqeuQmOyXiOyKteuLiOuLpC5gKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICB0aGlzLnRkYiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWcudGVzdCk7XG4gICAgdGhpcy5mZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUpO1xuICB9XG5cbiAgLyoqXG4gICAg7JuQ6rKpIGZpeHR1cmUgRELrpbwg66Gc7LusIHRlc3QgRELroZwg67O17IKs7ZWp64uI64ukLlxuICAgIHBnX2R1bXDroZwg7JuQ6rKpIERC66W8IOuNpO2UhO2VmOqzoCwgcGdfcmVzdG9yZeuhnCDroZzsu6zsl5Ag67O17JuQ7ZWp64uI64ukLlxuICAqL1xuICBhc3luYyBzeW5jKCkge1xuICAgIGNvbnN0IGZpeHR1cmVDb25uID0gU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LlBnQ29ubmVjdGlvbkNvbmZpZztcbiAgICBjb25zdCB0ZXN0Q29ubiA9IFNvbmFtdS5kYkNvbmZpZy50ZXN0LmNvbm5lY3Rpb24gYXMgS25leC5QZ0Nvbm5lY3Rpb25Db25maWc7XG5cbiAgICAvLyAxLiDroZzsu6wgdGVzdCBEQiDsl7DqsrAg7KKF66OMIOuwjyDsnqzsg53shLFcbiAgICBjb25zdCB0ZXN0UGdFbnYgPSB7IFBHUEFTU1dPUkQ6IHRlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCIgfTtcbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJcbiAgICAgICAgU0VMRUNUIHBnX3Rlcm1pbmF0ZV9iYWNrZW5kKHBnX3N0YXRfYWN0aXZpdHkucGlkKVxuICAgICAgICBGUk9NIHBnX3N0YXRfYWN0aXZpdHlcbiAgICAgICAgV0hFUkUgZGF0bmFtZSA9ICcke3Rlc3RDb25uLmRhdGFiYXNlfSdcbiAgICAgICAgICBBTkQgcGlkIDw+IHBnX2JhY2tlbmRfcGlkKCk7XG4gICAgICBcImAsXG4gICAgICB7IHN0ZGlvOiBcImluaGVyaXRcIiwgZW52OiB7IC4uLnByb2Nlc3MuZW52LCAuLi50ZXN0UGdFbnYgfSBhcyBOb2RlSlMuUHJvY2Vzc0VudiB9LFxuICAgICk7XG5cbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJEUk9QIERBVEFCQVNFIElGIEVYSVNUUyBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIGV4ZWNTeW5jKFxuICAgICAgYHBzcWwgLWggJHt0ZXN0Q29ubi5ob3N0fSAtcCAke3Rlc3RDb25uLnBvcnQgPz8gNTQzMn0gLVUgJHt0ZXN0Q29ubi51c2VyfSAtZCBwb3N0Z3JlcyAtYyBcIkNSRUFURSBEQVRBQkFTRSBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIC8vIDIuIOybkOqyqSBmaXh0dXJlIERCIOKGkiDroZzsu6wgdGVzdCBEQuuhnCDrs7XsgqwgKHBnX2R1bXAgfCBwZ19yZXN0b3JlKVxuICAgIGNvbnN0IGZpeHR1cmVQZ0VudiA9IHsgUEdQQVNTV09SRDogZml4dHVyZUNvbm4ucGFzc3dvcmQgfHwgXCJcIiB9O1xuICAgIGNvbnN0IGR1bXBDbWQgPSBgcGdfZHVtcCAtaCAke2ZpeHR1cmVDb25uLmhvc3R9IC1wICR7Zml4dHVyZUNvbm4ucG9ydCA/PyA1NDMyfSAtVSAke2ZpeHR1cmVDb25uLnVzZXJ9IC1kICR7Zml4dHVyZUNvbm4uZGF0YWJhc2V9IC1GY2A7XG4gICAgY29uc3QgcmVzdG9yZUNtZCA9IGBwZ19yZXN0b3JlIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgJHt0ZXN0Q29ubi5kYXRhYmFzZX0gLS1uby1vd25lciAtLW5vLWFjbGA7XG5cbiAgICBleGVjU3luYyhgJHtkdW1wQ21kfSB8IFBHUEFTU1dPUkQ9XCIke3Rlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCJ9XCIgJHtyZXN0b3JlQ21kfWAsIHtcbiAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICAgIGVudjogeyAuLi5wcm9jZXNzLmVudiwgLi4uZml4dHVyZVBnRW52IH0gYXMgTm9kZUpTLlByb2Nlc3NFbnYsXG4gICAgICBzaGVsbDogXCIvYmluL2Jhc2hcIixcbiAgICB9KTtcblxuICAgIC8vIDMuIOyLnO2AgOyKpCDrpqzshYsgKOuNsOydtO2EsCDrs7Xsgqwg7ZuEIOyLnO2AgOyKpOulvCBNQVgoaWQp66GcIOygleugrClcbiAgICBhd2FpdCB0aGlzLnJlc2V0U2VxdWVuY2VzKFNvbmFtdS5kYkNvbmZpZy50ZXN0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiDrqqjrk6Ag7YWM7J2067iU7J2YIOyLnO2AgOyKpOulvCDtmITsnqwgTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAqIGZpeHR1cmUgc3luYyDtm4Qg7Iuc7YCA7Iqk6rCAIOyLpOygnCDrjbDsnbTthLDsmYAg66ee7KeAIOyViuuKlCDrrLjsoJzrpbwg7ZW06rKw7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyByZXNldFNlcXVlbmNlcyhkYkNvbmZpZzogU29uYW11REJDb25maWdbXCJ0ZXN0XCJdKSB7XG4gICAgY29uc3QgdGVzdERiID0gY3JlYXRlS25leEluc3RhbmNlKGRiQ29uZmlnKTtcbiAgICBjb25zdCBlbnRpdGllcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKTtcblxuICAgIHRyeSB7XG4gICAgICBmb3IgKGNvbnN0IGVudGl0eSBvZiBlbnRpdGllcykge1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGUgfHwgZW50aXR5LmlkLnRvTG93ZXJDYXNlKCk7XG5cbiAgICAgICAgLy8gaWQg7ZWE65Oc7J2YIO2DgOyeheydhCDtmZXsnbjtlanri4jri6RcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgY29uc3QgaWRUeXBlID0gaWRQcm9wPy50eXBlO1xuXG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlcuydtOqxsOuCmCwgc3RyaW5n7J207KeA66eMIGZpeHR1cmVTdHJhdGVneT1zZXF1ZW5jZeyduCDqsr3smrDsl5Drp4wg66as7IWL7ZWp64uI64ukXG4gICAgICAgIGNvbnN0IHVzZXNTZXF1ZW5jZSA9XG4gICAgICAgICAgaWRUeXBlID09PSBcImludGVnZXJcIiB8fFxuICAgICAgICAgIGlkVHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICBpZFByb3A/LmNvbmU/LmZpeHR1cmVTdHJhdGVneSA9PT0gXCJzZXF1ZW5jZVwiO1xuXG4gICAgICAgIGlmICghdXNlc1NlcXVlbmNlKSB7XG4gICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgYFNraXBwaW5nIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX0gKGlkIHR5cGU6ICR7aWRUeXBlIHx8IFwidW5rbm93blwifSlgLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFBvc3RncmVTUUwg7Iuc7YCA7Iqk66W8IO2YhOyerCDthYzsnbTruJTsnZggTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAgICAgIC8vIHN0cmluZyDtg4DsnoXsnZgg6rK97JqwIOyIq+yekCDsupDsiqTtjIXsnbQg7ZWE7JqU7ZWp64uI64ukLlxuICAgICAgICBjb25zdCBtYXhFeHByID0gaWRUeXBlID09PSBcInN0cmluZ1wiID8gXCJNQVgoaWQ6OmJpZ2ludClcIiA6IFwiTUFYKGlkKVwiO1xuICAgICAgICBhd2FpdCB0ZXN0RGIucmF3KGBcbiAgICAgICAgICBTRUxFQ1Qgc2V0dmFsKFxuICAgICAgICAgICAgcGdfZ2V0X3NlcmlhbF9zZXF1ZW5jZSgncHVibGljLiR7dGFibGVOYW1lfScsICdpZCcpLFxuICAgICAgICAgICAgQ09BTEVTQ0UoKFNFTEVDVCAke21heEV4cHJ9IEZST00gJHt0YWJsZU5hbWV9KSwgMSlcbiAgICAgICAgICApXG4gICAgICAgIGApO1xuICAgICAgfVxuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCB0ZXN0RGIuZGVzdHJveSgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgdmlzaXRlZFJlY29yZHMgPSBuZXcgU2V0PHN0cmluZz4oKTtcbiAgYXN5bmMgaW1wb3J0Rml4dHVyZShlbnRpdHlJZDogc3RyaW5nLCBpZHM6IG51bWJlcltdKSB7XG4gICAgLy8g67Cp66y4IOq4sOuhnSDstIjquLDtmZQgKOyDiOuhnOyatCBpbXBvcnQg7J6R7JeFIOyLnOyekSlcbiAgICB0aGlzLnZpc2l0ZWRSZWNvcmRzLmNsZWFyKCk7XG5cbiAgICBjb25zdCBxdWVyaWVzID0gdW5pcXVlKFxuICAgICAgKFxuICAgICAgICBhd2FpdCBQcm9taXNlLmFsbChcbiAgICAgICAgICBpZHMubWFwKGFzeW5jIChpZCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZCwgXCJpZFwiLCBpZCk7XG4gICAgICAgICAgfSksXG4gICAgICAgIClcbiAgICAgICkuZmxhdCgpLFxuICAgICk7XG5cbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuICAgIGZvciAoY29uc3QgcXVlcnkgb2YgcXVlcmllcykge1xuICAgICAgYXdhaXQgd2RiLnJhdyhxdWVyeSk7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZDogc3RyaW5nLCBmaWVsZDogc3RyaW5nLCBpZDogbnVtYmVyKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHJlY29yZEtleSA9IGAke2VudGl0eUlkfSMke2ZpZWxkfSMke2lkfWA7XG5cbiAgICAvLyDsiJztmZgg7LC47KGwIOuwqeyngDog7J2066+4IOuwqeusuO2VnCDroIjsvZTrk5zripQg7Iqk7YK1XG4gICAgaWYgKHRoaXMudmlzaXRlZFJlY29yZHMuaGFzKHJlY29yZEtleSkpIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG4gICAgdGhpcy52aXNpdGVkUmVjb3Jkcy5hZGQocmVjb3JkS2V5KTtcblxuICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKTtcbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuXG4gICAgLy8g7Jes6riw7IScIOyLpERC7J2YIHJvdyDqsIDsoLjsmLRcbiAgICBjb25zdCBbcm93XSA9IGF3YWl0IHdkYihlbnRpdHkudGFibGUpLndoZXJlKGZpZWxkLCBpZCkubGltaXQoMSk7XG4gICAgaWYgKHJvdyA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5SWR9IyR7aWR9IHJvd+ulvCDssL7snYQg7IiYIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICAvLyDtlL3siqTss5BEQiwg7IukREJcbiAgICBjb25zdCBmaXh0dXJlRGF0YWJhc2UgPSAoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpLmRhdGFiYXNlO1xuICAgIGNvbnN0IHJlYWxEYXRhYmFzZSA9IChTb25hbXUuZGJDb25maWcucHJvZHVjdGlvbl9tYXN0ZXIuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpXG4gICAgICAuZGF0YWJhc2U7XG5cbiAgICBjb25zdCBzZWxmUXVlcnkgPSBgSU5TRVJUIElHTk9SRSBJTlRPIFxcYCR7Zml4dHVyZURhdGFiYXNlfVxcYC5cXGAke2VudGl0eS50YWJsZX1cXGAgKFNFTEVDVCAqIEZST00gXFxgJHtyZWFsRGF0YWJhc2V9XFxgLlxcYCR7ZW50aXR5LnRhYmxlfVxcYCBXSEVSRSBcXGBpZFxcYCA9ICR7aWR9KWA7XG5cbiAgICBjb25zdCBhcmdzID0gT2JqZWN0LmVudHJpZXMoZW50aXR5LnJlbGF0aW9ucylcbiAgICAgIC5maWx0ZXIoXG4gICAgICAgIChbLCByZWxhdGlvbl0pID0+XG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8XG4gICAgICAgICAgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pICYmIHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UgPT09IHVuZGVmaW5lZCksXG4gICAgICApXG4gICAgICAubWFwKChbLCByZWxhdGlvbl0pID0+IHtcbiAgICAgICAgLypcbiAgICAgICAgQmVsb25nc1RvT25l7J24IOqyveyasFxuICAgICAgICAgIENhdGVnb3J5IC8gJ2lkJyAvIHJvd1tjYXRlZ29yeV9pZF0g7Zi47LacXG4gICAgICAgIE9uZVRvT25l7JeQIGpvaW5Db2x1bW4gPT09IHRydWUg7J24IOqyveyasFxuICAgICAgICAgIFByb2ZpbGUgLyAnaWQnIC8gcm93W3Byb2ZpbGVfaWRdIO2YuOy2nFxuICAgICAgICBPbmVUb09uZeyXkCBqb2luQ29sdW1uID09PSBmYWxzZSDsnbgg6rK97JqwXG4gICAgICAgICAgUHJvZmlsZSAvICdwcm9maWxlX2lkJyAvIHJvd1snaWQnXSDtmLjstpxcbiAgICAgICAgKi9cbiAgICAgICAgbGV0IGZpZWxkOiBzdHJpbmc7XG4gICAgICAgIGxldCBpZDogbnVtYmVyO1xuICAgICAgICBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChyZWxhdGlvbikgJiYgIXJlbGF0aW9uLmhhc0pvaW5Db2x1bW4pIHtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG4gICAgICAgICAgY29uc3QgcmVsYXRlZElkQ29sdW1uTmFtZSA9IHJlbGF0ZWRFbnRpdHkucHJvcHMuZmluZChcbiAgICAgICAgICAgIChwKSA9PiBpc1JlbGF0aW9uUHJvcChwKSAmJiBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICAgICAgICApPy5uYW1lO1xuICAgICAgICAgIGlmICghcmVsYXRlZElkQ29sdW1uTmFtZSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3JlbGF0ZWRFbnRpdHkuaWR97J2YICR7ZW50aXR5LmlkfSDqtIDqs4Qg7ZSE66Gt7J2EIOywvuydhCDsiJgg7JeG7Iq164uI64ukLmApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBmaWVsZCA9IGAke3JlbGF0ZWRJZENvbHVtbk5hbWV9X2lkYDtcbiAgICAgICAgICBpZCA9IHJvdy5pZDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBmaWVsZCA9IFwiaWRcIjtcbiAgICAgICAgICBpZCA9IHJvd1tgJHtyZWxhdGlvbi5uYW1lfV9pZGBdO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZW50aXR5SWQ6IHJlbGF0aW9uLndpdGgsXG4gICAgICAgICAgZmllbGQsXG4gICAgICAgICAgaWQsXG4gICAgICAgIH07XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoYXJnKSA9PiBhcmcuaWQgIT09IG51bGwpO1xuXG4gICAgY29uc3QgcmVsUXVlcmllcyA9IGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgYXJncy5tYXAoYXN5bmMgKGFyZ3MpID0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhhcmdzLmVudGl0eUlkLCBhcmdzLmZpZWxkLCBhcmdzLmlkKTtcbiAgICAgIH0pLFxuICAgICk7XG5cbiAgICByZXR1cm4gWy4uLnVuaXF1ZShyZWxRdWVyaWVzLnJldmVyc2UoKS5mbGF0KCkpLCBzZWxmUXVlcnldO1xuICB9XG5cbiAgYXN5bmMgZGVzdHJveSgpIHtcbiAgICBpZiAodGhpcy5fdGRiKSB7XG4gICAgICBhd2FpdCB0aGlzLl90ZGIuZGVzdHJveSgpO1xuICAgICAgdGhpcy5fdGRiID0gbnVsbDtcbiAgICB9XG4gICAgaWYgKHRoaXMuX2ZkYikge1xuICAgICAgYXdhaXQgdGhpcy5fZmRiLmRlc3Ryb3koKTtcbiAgICAgIHRoaXMuX2ZkYiA9IG51bGw7XG4gICAgfVxuICAgIGF3YWl0IEJhc2VNb2RlbC5kZXN0cm95KCk7XG4gIH1cblxuICBhc3luYyBnZXRGaXh0dXJlcyhcbiAgICBzb3VyY2VEQk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIHRhcmdldERCTmFtZToga2V5b2YgU29uYW11REJDb25maWcsXG4gICAgc2VhcmNoT3B0aW9uczogRml4dHVyZVNlYXJjaE9wdGlvbnMsXG4gICAgZHVwbGljYXRlQ2hlY2s/OiBEdXBsaWNhdGVDaGVja09wdGlvbnMsXG4gICkge1xuICAgIGNvbnN0IHNvdXJjZURCID0gY3JlYXRlS25leEluc3RhbmNlKFNvbmFtdS5kYkNvbmZpZ1tzb3VyY2VEQk5hbWVdKTtcbiAgICBjb25zdCB0YXJnZXREQiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWdbdGFyZ2V0REJOYW1lXSk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgeyBlbnRpdHlJZCwgZmllbGQsIHZhbHVlLCBzZWFyY2hUeXBlIH0gPSBzZWFyY2hPcHRpb25zO1xuXG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCk7XG4gICAgICBjb25zdCBjb2x1bW4gPVxuICAgICAgICBlbnRpdHkucHJvcHMuZmluZCgocHJvcCkgPT4gcHJvcC5uYW1lID09PSBmaWVsZCk/LnR5cGUgPT09IFwicmVsYXRpb25cIlxuICAgICAgICAgID8gYCR7ZmllbGR9X2lkYFxuICAgICAgICAgIDogZmllbGQ7XG5cbiAgICAgIGxldCBxdWVyeSA9IHNvdXJjZURCKGVudGl0eS50YWJsZSk7XG4gICAgICBpZiAoc2VhcmNoVHlwZSA9PT0gXCJlcXVhbHNcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgdmFsdWUpO1xuICAgICAgfSBlbHNlIGlmIChzZWFyY2hUeXBlID09PSBcImxpa2VcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgXCJsaWtlXCIsIGAlJHt2YWx1ZX0lYCk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHJvd3MgPSBhd2FpdCBxdWVyeTtcbiAgICAgIGlmIChyb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJObyByZWNvcmRzIGZvdW5kXCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmaXh0dXJlczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICAgIGNvbnN0IGluaXRpYWxSZWNvcmRzTGVuZ3RoID0gZml4dHVyZXMubGVuZ3RoO1xuICAgICAgICBjb25zdCBuZXdSZWNvcmRzID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgcm93LCB7XG4gICAgICAgICAgX2RiOiBzb3VyY2VEQixcbiAgICAgICAgfSk7XG4gICAgICAgIGZpeHR1cmVzLnB1c2goLi4ubmV3UmVjb3Jkcyk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRGaXh0dXJlUmVjb3JkID0gZml4dHVyZXMuZmluZCgocikgPT4gci5maXh0dXJlSWQgPT09IGAke2VudGl0eUlkfSMke3Jvdy5pZH1gKTtcblxuICAgICAgICBpZiAoY3VycmVudEZpeHR1cmVSZWNvcmQpIHtcbiAgICAgICAgICAvLyDtmITsnqwgZml4dHVyZeuhnOu2gO2EsCDsg53shLHrkJwgZmV0Y2hlZFJlY29yZHMg7ISk7KCVXG4gICAgICAgICAgY3VycmVudEZpeHR1cmVSZWNvcmQuZmV0Y2hlZFJlY29yZHMgPSBmaXh0dXJlc1xuICAgICAgICAgICAgLmZpbHRlcigocikgPT4gci5maXh0dXJlSWQgIT09IGN1cnJlbnRGaXh0dXJlUmVjb3JkLmZpeHR1cmVJZClcbiAgICAgICAgICAgIC5zbGljZShpbml0aWFsUmVjb3Jkc0xlbmd0aClcbiAgICAgICAgICAgIC5tYXAoKHIpID0+IHIuZml4dHVyZUlkKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZS5lbnRpdHlJZCk7XG5cbiAgICAgICAgLy8g7IKs7Jqp7J6QIOyngOyglSDsu6zrn7wg6riw7KSAIOykkeuztSDtmZXsnbgg4oaSIHRhcmdldFxuICAgICAgICBjb25zdCBjdXN0b21Db2x1bW5zID0gZHVwbGljYXRlQ2hlY2s/LmNvbHVtbnM/LltmaXh0dXJlLmVudGl0eUlkXTtcbiAgICAgICAgaWYgKGN1c3RvbUNvbHVtbnMgJiYgY3VzdG9tQ29sdW1ucy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgY3VzdG9tRHVwbGljYXRlUm93ID0gYXdhaXQgdGhpcy5jaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICAgICAgICAgIHRhcmdldERCLFxuICAgICAgICAgICAgZW50aXR5LFxuICAgICAgICAgICAgZml4dHVyZSxcbiAgICAgICAgICAgIGN1c3RvbUNvbHVtbnMsXG4gICAgICAgICAgKTtcbiAgICAgICAgICBpZiAoY3VzdG9tRHVwbGljYXRlUm93KSB7XG4gICAgICAgICAgICBjb25zdCBbcmVjb3JkXSA9IGF3YWl0IHRoaXMuY3JlYXRlRml4dHVyZVJlY29yZChlbnRpdHksIGN1c3RvbUR1cGxpY2F0ZVJvdywge1xuICAgICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICAgIF9kYjogdGFyZ2V0REIsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGZpeHR1cmUudGFyZ2V0ID0gcmVjb3JkO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVuaXF1ZSBpbmRleCDquLDspIAg7KSR67O1IO2ZleyduCDihpIgZml4dHVyZS51bmlxdWVcbiAgICAgICAgY29uc3QgdW5pcXVlUm93ID0gYXdhaXQgdGhpcy5jaGVja1VuaXF1ZVZpb2xhdGlvbih0YXJnZXREQiwgZW50aXR5LCBmaXh0dXJlKTtcbiAgICAgICAgaWYgKHVuaXF1ZVJvdykge1xuICAgICAgICAgIGNvbnN0IFtyZWNvcmRdID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgdW5pcXVlUm93LCB7XG4gICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICBfZGI6IHRhcmdldERCLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGZpeHR1cmUudW5pcXVlID0gcmVjb3JkO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB1bmlxdWUoZml4dHVyZXMsIChmKSA9PiBmLmZpeHR1cmVJZCk7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZChbdGFyZ2V0REIuZGVzdHJveSgpLCBzb3VyY2VEQi5kZXN0cm95KCldKTtcbiAgICB9XG4gIH1cblxuICBhc3luYyBjcmVhdGVGaXh0dXJlUmVjb3JkKFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIHJvdzoge1xuICAgICAgaWQ6IG51bWJlciB8IHN0cmluZztcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsO1xuICAgIH0sXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIHNpbmdsZVJlY29yZD86IGJvb2xlYW47XG4gICAgICBfZGI/OiBLbmV4O1xuICAgIH0sXG4gICk6IFByb21pc2U8Rml4dHVyZVJlY29yZFtdPiB7XG4gICAgY29uc3QgcmVjb3JkczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgY29uc3QgdmlzaXRlZEVudGl0aWVzID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICBjb25zdCBjcmVhdGUgPSBhc3luYyAoXG4gICAgICBlbnRpdHk6IEVudGl0eSxcbiAgICAgIHJvdzoge1xuICAgICAgICBpZDogbnVtYmVyIHwgc3RyaW5nO1xuICAgICAgICBba2V5OiBzdHJpbmddOiBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgbnVsbDtcbiAgICAgIH0sXG4gICAgKSA9PiB7XG4gICAgICBjb25zdCBmaXh0dXJlSWQgPSBgJHtlbnRpdHkuaWR9IyR7cm93LmlkfWA7XG4gICAgICBpZiAodmlzaXRlZEVudGl0aWVzLmhhcyhmaXh0dXJlSWQpKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIHZpc2l0ZWRFbnRpdGllcy5hZGQoZml4dHVyZUlkKTtcblxuICAgICAgY29uc3QgcmVjb3JkOiBGaXh0dXJlUmVjb3JkID0ge1xuICAgICAgICBmaXh0dXJlSWQsXG4gICAgICAgIGVudGl0eUlkOiBlbnRpdHkuaWQsXG4gICAgICAgIGlkOiByb3cuaWQsXG4gICAgICAgIGNvbHVtbnM6IHt9LFxuICAgICAgICBmZXRjaGVkUmVjb3JkczogW10sXG4gICAgICAgIGJlbG9uZ3NSZWNvcmRzOiBbXSxcbiAgICAgIH07XG5cbiAgICAgIGZvciAoY29uc3QgcHJvcCBvZiBlbnRpdHkucHJvcHMpIHtcbiAgICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0gPSB7XG4gICAgICAgICAgcHJvcDogcHJvcCxcbiAgICAgICAgICB2YWx1ZTogcm93W3Byb3AubmFtZV0sXG4gICAgICAgIH07XG5cbiAgICAgICAgY29uc3QgZGIgPSBvcHRpb25zPy5fZGIgPz8gQmFzZU1vZGVsLmdldERCKFwid1wiKTtcbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHRocm91Z2hUYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IGZyb21Db2x1bW4gPSBgJHtpbmZsZWN0aW9uLnNpbmd1bGFyaXplKGVudGl0eS50YWJsZSl9X2lkYDtcbiAgICAgICAgICBjb25zdCB0b0NvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYih0aHJvdWdoVGFibGUpLndoZXJlKGZyb21Db2x1bW4sIHJvdy5pZCkucGx1Y2sodG9Db2x1bW4pO1xuICAgICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0udmFsdWUgPSByZWxhdGVkSWRzO1xuICAgICAgICB9IGVsc2UgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKVxuICAgICAgICAgICAgLndoZXJlKHByb3Auam9pbkNvbHVtbiwgcm93LmlkKVxuICAgICAgICAgICAgLnBsdWNrKFwiaWRcIik7XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZHM7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiAhcHJvcC5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgLy8g7Jet67Cp7ZalIE9uZVRvT25lOiBGS+ulvCDqsIDsp4Qg6rSA66CoIOyXlO2LsO2LsOulvCDssL7sirXri4jri6RcbiAgICAgICAgICAvLyDsmIjsi5w6IFVzZXIgT25lVG9PbmUgRW1wbG95ZWUgKEVtcGxveWVl6rCAIHVzZXJfaWQgRkvrpbwg6rCA7KeQKVxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRQcm9wID0gcmVsYXRlZEVudGl0eS5wcm9wcy5maW5kKFxuICAgICAgICAgICAgKHApID0+IGlzUmVsYXRpb25Qcm9wKHApICYmIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICAgICAgICk7XG4gICAgICAgICAgaWYgKHJlbGF0ZWRQcm9wICYmIGlzUmVsYXRpb25Qcm9wKHJlbGF0ZWRQcm9wKSkge1xuICAgICAgICAgICAgLy8g6rSA66CoIOyXlO2LsO2LsOyXkOyEnCBGSyDsu6zrn7zsnLzroZwg7L+866as7ZWp64uI64ukIChpZOqwgCDslYTri5gpXG4gICAgICAgICAgICBjb25zdCBma0NvbHVtbiA9IGAke3JlbGF0ZWRQcm9wLm5hbWV9X2lkYDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSb3cgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKS53aGVyZShma0NvbHVtbiwgcm93LmlkKS5maXJzdCgpO1xuICAgICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRSb3c/LmlkO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IHJvd1tgJHtwcm9wLm5hbWV9X2lkYF07XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICBpZiAocmVsYXRlZElkKSB7XG4gICAgICAgICAgICByZWNvcmQuYmVsb25nc1JlY29yZHMucHVzaChgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIW9wdGlvbnM/LnNpbmdsZVJlY29yZCAmJiByZWxhdGVkSWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJvdyA9IGF3YWl0IGRiKHJlbGF0ZWRFbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgcmVsYXRlZElkKS5maXJzdCgpO1xuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSb3cpIHtcbiAgICAgICAgICAgICAgYXdhaXQgY3JlYXRlKHJlbGF0ZWRFbnRpdHksIHJlbGF0ZWRSb3cpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZWNvcmRzLnB1c2gocmVjb3JkKTtcbiAgICB9O1xuXG4gICAgYXdhaXQgY3JlYXRlKGVudGl0eSwgcm93KTtcblxuICAgIHJldHVybiByZWNvcmRzO1xuICB9XG5cbiAgLyoqXG4gICAqIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrAgKHNlbGYtcmVmZXJlbmNlIO2PrO2VqClcbiAgICogMi4g7YWM7J2067iU67OEIOugiOuyqOuzhOuhnCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnSDrsI8gdXBzZXJ0IOyLpO2WiVxuICAgKiAzLiDsiJzshJwg6riw67CYIHV1aWTihpJpZCDrp6TtlZEgKFVwc2VydEJ1aWxkZXLqsIAgdXVpZOulvCBEQuyXkCDsoIDsnqXtlZjsp4Ag7JWK7Jy866+A66GcKVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy64qUIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0XG4gICAqIEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWY7JesIOqwgSB1cHNlcnQg7Zi47Lac7JeQ7ISc64qUXG4gICAqIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuPhOuhnSDtlanri4jri6QuXG4gICAqL1xuICBhc3luYyBpbnNlcnRGaXh0dXJlcyhcbiAgICBkYk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIF9maXh0dXJlczogRml4dHVyZVJlY29yZFtdLFxuICApOiBQcm9taXNlPEZpeHR1cmVJbXBvcnRSZXN1bHRbXT4ge1xuICAgIGNvbnN0IGZpeHR1cmVzID0gdW5pcXVlKF9maXh0dXJlcywgKGYpID0+IGYuZml4dHVyZUlkKTtcblxuICAgIC8vIOy0iOq4sO2ZlFxuICAgIHRoaXMuYnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gICAgdGhpcy5maXh0dXJlUmVmTWFwID0gbmV3IE1hcCgpO1xuICAgIHRoaXMuc2tpcHBlZEZpeHR1cmVzID0gbmV3IE1hcCgpO1xuXG4gICAgLy8g67OR66CsIO2FjOyKpO2KuCDrqqjrk5zsl5DshJzripQgd29ya2Vy67OEIERC7JeQIOyggOyepVxuICAgIGNvbnN0IGRiQ29uZmlnID1cbiAgICAgIHByb2Nlc3MuZW52LlNPTkFNVV9XT1JLRVJfREIgPT09IFwidHJ1ZVwiICYmIHByb2Nlc3MuZW52LlZJVEVTVF9QT09MX0lEXG4gICAgICAgID8gKCgpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHdvcmtlcklkID0gcGFyc2VJbnQocHJvY2Vzcy5lbnYuVklURVNUX1BPT0xfSUQgPz8gXCIxXCIsIDEwKTtcbiAgICAgICAgICAgIGNvbnN0IGJhc2VDb25maWcgPSBTb25hbXUuZGJDb25maWdbZGJOYW1lXTtcbiAgICAgICAgICAgIGNvbnN0IGNvbm5lY3Rpb24gPSBiYXNlQ29uZmlnLmNvbm5lY3Rpb24gYXMgeyBkYXRhYmFzZTogc3RyaW5nIH07XG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAuLi5iYXNlQ29uZmlnLFxuICAgICAgICAgICAgICBjb25uZWN0aW9uOiB7IC4uLmNvbm5lY3Rpb24sIGRhdGFiYXNlOiBgJHtjb25uZWN0aW9uLmRhdGFiYXNlfV8ke3dvcmtlcklkfWAgfSxcbiAgICAgICAgICAgICAgcG9vbDogeyBtaW46IDEsIG1heDogMSB9LFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9KSgpXG4gICAgICAgIDogU29uYW11LmRiQ29uZmlnW2RiTmFtZV07XG4gICAgY29uc3QgZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoZGJDb25maWcpO1xuICAgIGNvbnN0IHJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSA9IFtdO1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrBcbiAgICAgIHRoaXMucmVsYXRpb25HcmFwaC5idWlsZEdyYXBoKGZpeHR1cmVzKTtcbiAgICAgIGNvbnN0IGluc2VydGlvbk9yZGVyID0gdGhpcy5yZWxhdGlvbkdyYXBoLmdldEluc2VydGlvbk9yZGVyKCk7XG5cbiAgICAgIC8vIDIuIOyKpO2Cte2VoCBmaXh0dXJlIOuovOyggCDsspjrpqwgKG92ZXJyaWRlIOyytO2BrClcbiAgICAgIGZvciAoY29uc3QgZml4dHVyZUlkIG9mIGluc2VydGlvbk9yZGVyKSB7XG4gICAgICAgIGNvbnN0IGZpeHR1cmUgPSBmaXh0dXJlcy5maW5kKChmKSA9PiBmLmZpeHR1cmVJZCA9PT0gZml4dHVyZUlkKTtcbiAgICAgICAgaWYgKCFmaXh0dXJlKSBjb250aW51ZTtcblxuICAgICAgICBjb25zdCBoYXNUYXJnZXQgPSAhIWZpeHR1cmUudGFyZ2V0O1xuICAgICAgICBjb25zdCBoYXNVbmlxdWUgPSAhIWZpeHR1cmUudW5pcXVlO1xuICAgICAgICBjb25zdCBoYXNEdXBsaWNhdGUgPSBoYXNUYXJnZXQgfHwgaGFzVW5pcXVlO1xuXG4gICAgICAgIC8vIOykkeuzteydtCDsnojqs6Agb3ZlcnJpZGU9ZmFsc2Xsnbgg6rK97JqwOiDsiqTtgrVcbiAgICAgICAgaWYgKGhhc0R1cGxpY2F0ZSAmJiAhZml4dHVyZS5vdmVycmlkZSkge1xuICAgICAgICAgIC8vIOq4sOyhtCDroIjsvZTrk5wgSUQg7KCA7J6lICh1bmlxdWUg7Jqw7ISgLCDsl4bsnLzrqbQgdGFyZ2V0KVxuICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBmaXh0dXJlLnVuaXF1ZT8uaWQgPz8gZml4dHVyZS50YXJnZXQ/LmlkO1xuICAgICAgICAgIGFzc2VydChleGlzdGluZ0lkKTtcbiAgICAgICAgICB0aGlzLnNraXBwZWRGaXh0dXJlcy5zZXQoZml4dHVyZUlkLCB7XG4gICAgICAgICAgICBlbnRpdHlJZDogZml4dHVyZS5lbnRpdHlJZCxcbiAgICAgICAgICAgIGV4aXN0aW5nSWQsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICBjaGFsay55ZWxsb3coXG4gICAgICAgICAgICAgICAgYFNraXBwZWQgJHtmaXh0dXJlLmVudGl0eUlkfSMke2ZpeHR1cmUuaWR9IChleGlzdGluZzogIyR7ZXhpc3RpbmdJZH0sIG92ZXJyaWRlOiBmYWxzZSlgLFxuICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyAzLiDthYzsnbTruJTrs4QgZml4dHVyZSDqt7jro7ntmZQgKGluc2VydGlvbk9yZGVyIOyInOyEnCDquLDrsJgpXG4gICAgICBjb25zdCBmaXh0dXJlc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgRml4dHVyZVJlY29yZFtdPigpO1xuICAgICAgY29uc3QgdGFibGVPcmRlcjogc3RyaW5nW10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlSWQgb2YgaW5zZXJ0aW9uT3JkZXIpIHtcbiAgICAgICAgLy8g7Iqk7YK165CcIGZpeHR1cmUg7KCc7Jm4XG4gICAgICAgIGlmICh0aGlzLnNraXBwZWRGaXh0dXJlcy5oYXMoZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgY29uc3QgZml4dHVyZSA9IGZpeHR1cmVzLmZpbmQoKGYpID0+IGYuZml4dHVyZUlkID09PSBmaXh0dXJlSWQpO1xuICAgICAgICBpZiAoIWZpeHR1cmUpIGNvbnRpbnVlO1xuXG4gICAgICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGZpeHR1cmUuZW50aXR5SWQpO1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGU7XG5cbiAgICAgICAgaWYgKCFmaXh0dXJlc0J5VGFibGUuaGFzKHRhYmxlTmFtZSkpIHtcbiAgICAgICAgICBmaXh0dXJlc0J5VGFibGUuc2V0KHRhYmxlTmFtZSwgW10pO1xuICAgICAgICAgIHRhYmxlT3JkZXIucHVzaCh0YWJsZU5hbWUpO1xuICAgICAgICB9XG4gICAgICAgIGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKT8ucHVzaChmaXh0dXJlKTtcbiAgICAgIH1cblxuICAgICAgYXdhaXQgZGIudHJhbnNhY3Rpb24oYXN5bmMgKHRyeCkgPT4ge1xuICAgICAgICBjb25zdCBpbnNlcnRlZElkc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4oKTtcblxuICAgICAgICAvLyA0LiDthYzsnbTruJTrs4Qg66CI67Ko67OEIOyymOumrFxuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgY29uc3QgdGFibGVGaXh0dXJlcyA9IGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBbXTtcbiAgICAgICAgICBjb25zdCBsZXZlbHMgPSB0aGlzLmdyb3VwRml4dHVyZXNCeUxldmVsKHRhYmxlRml4dHVyZXMpO1xuXG4gICAgICAgICAgZm9yIChjb25zdCBsZXZlbEZpeHR1cmVzIG9mIGxldmVscykge1xuICAgICAgICAgICAgLy8g7ZW064u5IOugiOuyqOydmCBmaXh0dXJl65OkIHJlZ2lzdGVyXG4gICAgICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgbGV2ZWxGaXh0dXJlcykge1xuICAgICAgICAgICAgICB0aGlzLnJlZ2lzdGVyRml4dHVyZShmaXh0dXJlLCBpbnNlcnRlZElkc0J5VGFibGUpO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICAgIGBSZWdpc3RlcmVkICR7Zml4dHVyZS5lbnRpdHlJZH0jJHtmaXh0dXJlLmlkfSR7Zml4dHVyZS5vdmVycmlkZSA/IGAgKG92ZXJyaWRlKWAgOiBcIlwifWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIHVwc2VydCDsi6Ttlokg7KCEIHV1aWQg66qp66GdIOyggOyepVxuICAgICAgICAgICAgY29uc3QgdGFibGUgPSB0aGlzLmJ1aWxkZXIuZ2V0VGFibGUodGFibGVOYW1lKTtcbiAgICAgICAgICAgIGNvbnN0IHV1aWRzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4gcm93LnV1aWQgYXMgc3RyaW5nKTtcblxuICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICBgVXBzZXJ0aW5nICR7dGFibGVOYW1lfSB3aXRoICR7dXVpZHMubGVuZ3RofSByb3dzIChsZXZlbCAke2xldmVscy5pbmRleE9mKGxldmVsRml4dHVyZXMpICsgMX0vJHtsZXZlbHMubGVuZ3RofSlgLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICBjb25zdCBpZHMgPSAoYXdhaXQgdGhpcy5idWlsZGVyLnVwc2VydChcbiAgICAgICAgICAgICAgdHJ4LFxuICAgICAgICAgICAgICB0YWJsZU5hbWUgYXMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gICAgICAgICAgICApKSBhcyAobnVtYmVyIHwgc3RyaW5nKVtdO1xuXG4gICAgICAgICAgICAvLyDsiJzshJwg6riw67CYIHV1aWQgLT4gaWQg66ek7ZWRXG4gICAgICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDsl4bsnLzrr4DroZwg65Ox66GdIOyInOyEnCA9IOuwmO2ZmCDsiJzshJwg67O07J6lXG4gICAgICAgICAgICBpZiAodXVpZHMubGVuZ3RoID4gMCAmJiB1dWlkcy5sZW5ndGggPT09IGlkcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPVxuICAgICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHV1aWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICAgICAgZXhpc3RpbmdNYXAuc2V0KHV1aWRzW2ldLCBpZHNbaV0pO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5zZXQodGFibGVOYW1lLCBleGlzdGluZ01hcCk7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKHV1aWRzLmxlbmd0aCAhPT0gaWRzLmxlbmd0aCkge1xuICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgY2hhbGsueWVsbG93KFxuICAgICAgICAgICAgICAgICAgYFdhcm5pbmc6IHV1aWQgY291bnQgKCR7dXVpZHMubGVuZ3RofSkgIT0gaWQgY291bnQgKCR7aWRzLmxlbmd0aH0pIGZvciAke3RhYmxlTmFtZX1gLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNS4gTWFueVRvTWFueSDqtIDqs4Qg7LKY66asXG4gICAgICAgIGF3YWl0IHRoaXMucHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnModHJ4LCBmaXh0dXJlcywgaW5zZXJ0ZWRJZHNCeVRhYmxlKTtcblxuICAgICAgICAvLyA2LiBQb3N0Z3JlU1FMIOyLnO2AgOyKpCDrpqzshYtcbiAgICAgICAgLy8gRml4dHVyZSDsgr3snoUg7ZuEIOqwgSDthYzsnbTruJTsnZggSUQg7Iuc7YCA7Iqk66W8IOy1nOuMgCBJRCDqsJLsnLzroZwg7JeF642w7J207Yq47ZWp64uI64ukLlxuICAgICAgICAvLyDsnbTroIfqsowg7ZWY7KeAIOyViuycvOuptCDri6TsnYwgSU5TRVJUIOyLnCBJROqwgCAyMDAw67KI64yA66GcIOyDneyEseuQoCDsiJgg7J6I7Iq164uI64ukLlxuICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJlc2V0dGluZyBzZXF1ZW5jZXMuLi5cIikpO1xuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIEVudGl0eeulvCDssL7slYTshJwgaWQg7YOA7J6FIO2ZleyduFxuICAgICAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXRBbGxFbnRpdGllcygpLmZpbmQoXG4gICAgICAgICAgICAgIChlKSA9PiBlLnRhYmxlID09PSB0YWJsZU5hbWUgfHwgZS5pZC50b0xvd2VyQ2FzZSgpID09PSB0YWJsZU5hbWUsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICBpZiAoZW50aXR5KSB7XG4gICAgICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik7XG4gICAgICAgICAgICAgIGNvbnN0IGlkVHlwZSA9IGlkUHJvcD8udHlwZTtcblxuICAgICAgICAgICAgICAvLyBpbnRlZ2VyL2JpZ0ludGVnZXLsnbTqsbDrgpgsIHN0cmluZ+ydtOyngOunjCBmaXh0dXJlU3RyYXRlZ3k9c2VxdWVuY2Xsnbgg6rK97Jqw7JeQ66eMIOumrOyFi+2VqeuLiOuLpFxuICAgICAgICAgICAgICBjb25zdCB1c2VzU2VxdWVuY2UgPVxuICAgICAgICAgICAgICAgIGlkVHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgICAgICBpZFR5cGUgPT09IFwiYmlnSW50ZWdlclwiIHx8XG4gICAgICAgICAgICAgICAgaWRQcm9wPy5jb25lPy5maXh0dXJlU3RyYXRlZ3kgPT09IFwic2VxdWVuY2VcIjtcblxuICAgICAgICAgICAgICBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgICAgICAgICFpc1Rlc3QoKSAmJlxuICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICAgIGNoYWxrLmdyYXkoXG4gICAgICAgICAgICAgICAgICAgICAgYFNraXBwZWQgc2VxdWVuY2UgcmVzZXQgZm9yICR7dGFibGVOYW1lfSAoaWQgdHlwZTogJHtpZFR5cGUgfHwgXCJ1bmtub3duXCJ9KWAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIO2FjOydtOu4lOydmCDstZzrjIAgSUQg7KGw7ZqMIChzdHJpbmcg7YOA7J6F7J2AIOyIq+yekCDsupDsiqTtjIUg7ZWE7JqUKVxuICAgICAgICAgICAgY29uc3QgZW50aXR5MiA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKS5maW5kKFxuICAgICAgICAgICAgICAoZSkgPT4gZS50YWJsZSA9PT0gdGFibGVOYW1lIHx8IGUuaWQudG9Mb3dlckNhc2UoKSA9PT0gdGFibGVOYW1lLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIGNvbnN0IGlkVHlwZTIgPSBlbnRpdHkyPy5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik/LnR5cGU7XG4gICAgICAgICAgICBjb25zdCBtYXhJZFJlc3VsdCA9XG4gICAgICAgICAgICAgIGlkVHlwZTIgPT09IFwic3RyaW5nXCJcbiAgICAgICAgICAgICAgICA/IGF3YWl0IHRyeFxuICAgICAgICAgICAgICAgICAgICAucmF3KGBTRUxFQ1QgTUFYKGlkOjpiaWdpbnQpIGFzIG1heF9pZCBGUk9NIFwiJHt0YWJsZU5hbWV9XCJgKVxuICAgICAgICAgICAgICAgICAgICAudGhlbigocikgPT4gci5yb3dzWzBdKVxuICAgICAgICAgICAgICAgIDogYXdhaXQgdHJ4KHRhYmxlTmFtZSkubWF4KFwiaWQgYXMgbWF4X2lkXCIpLmZpcnN0KCk7XG4gICAgICAgICAgICBjb25zdCBtYXhJZCA9IG1heElkUmVzdWx0Py5tYXhfaWQ7XG5cbiAgICAgICAgICAgIGlmIChtYXhJZCAhPT0gbnVsbCAmJiBtYXhJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIOyLnO2AgOyKpOuqheydhCBwZ19nZXRfc2VyaWFsX3NlcXVlbmNl66GcIOyViOyghO2VmOqyjCDsobDtmoxcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4LnJhdyhgU0VMRUNUIHNldHZhbChwZ19nZXRfc2VyaWFsX3NlcXVlbmNlKD8sICdpZCcpLCA/KWAsIFtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgbWF4SWQsXG4gICAgICAgICAgICAgIF0pO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuZ3JlZW4oYFJlc2V0IHNlcXVlbmNlIGZvciAke3RhYmxlTmFtZX06ICR7bWF4SWR9YCkpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKF9lcnIpIHtcbiAgICAgICAgICAgIC8vIOyLnO2AgOyKpOqwgCDsl4bripQg7YWM7J2067iUKGpvaW4gdGFibGUg65OxKeydgCDrrLTsi5xcbiAgICAgICAgICAgICFpc1Rlc3QoKSAmJiBjb25zb2xlLmxvZyhjaGFsay5ncmF5KGBTa2lwcGVkIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX1gKSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNy4g6rKw6rO8IOyImOynkVxuICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcblxuICAgICAgICAgIC8vIOyKpO2CteuQnCBmaXh0dXJl64qUIOq4sOyhtCDroIjsvZTrk5wg7KCV67O066GcIOqysOqzvCDstpTqsIBcbiAgICAgICAgICBjb25zdCBza2lwcGVkID0gdGhpcy5za2lwcGVkRml4dHVyZXMuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAoc2tpcHBlZCkge1xuICAgICAgICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgICAgICAgZW50aXR5SWQ6IGZpeHR1cmUuZW50aXR5SWQsXG4gICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgc2tpcHBlZC5leGlzdGluZ0lkKS5maXJzdCgpLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCByZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAocmVmKSB7XG4gICAgICAgICAgICBjb25zdCB1dWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQoZW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgIGNvbnN0IGluc2VydGVkSWQgPSB1dWlkVG9JZD8uZ2V0KHJlZi51dWlkKTtcblxuICAgICAgICAgICAgaWYgKGluc2VydGVkSWQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICByZXN1bHRzLnB1c2goe1xuICAgICAgICAgICAgICAgIGVudGl0eUlkOiBmaXh0dXJlLmVudGl0eUlkLFxuICAgICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgaW5zZXJ0ZWRJZCkuZmlyc3QoKSxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihgSW5zZXJ0ZWQgaW50byAke2VudGl0eS50YWJsZX06ICMke2ZpeHR1cmUuaWR9IC0+ICMke2luc2VydGVkSWR9YCksXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCBkYi5kZXN0cm95KCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuaXF1ZShyZXN1bHRzLCAocikgPT4gYCR7ci5lbnRpdHlJZH0jJHtyLmRhdGEuaWR9YCk7XG4gIH1cblxuICAvKipcbiAgICogRml4dHVyZVJlY29yZOulvCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnVxuICAgKiBAcGFyYW0gaW5zZXJ0ZWRJZHNCeVRhYmxlIOydtOuvuCB1cHNlcnTrkJwg7YWM7J2067iU7J2YIHV1aWTihpJpZCDrp6TtlZEgKOugiOuyqOuzhCDsspjrpqwg7IucIOyCrOyaqSlcbiAgICovXG4gIHByaXZhdGUgcmVnaXN0ZXJGaXh0dXJlKFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlPzogTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4sXG4gICk6IFVCUmVmIHtcbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICBjb25zdCByb3c6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICAvLyBPdmVycmlkZSDrqqjrk5wg7YyQ64uoOiB0YXJnZXQg65iQ64qUIHVuaXF1ZeqwgCDsnojqs6Agb3ZlcnJpZGU9dHJ1ZeyduCDqsr3smrBcbiAgICBjb25zdCBleGlzdGluZ1JlY29yZCA9IGZpeHR1cmUudGFyZ2V0ID8/IGZpeHR1cmUudW5pcXVlO1xuICAgIGNvbnN0IGlzT3ZlcnJpZGVNb2RlID0gZml4dHVyZS5vdmVycmlkZSAmJiBleGlzdGluZ1JlY29yZDtcblxuICAgIGZvciAoY29uc3QgW3Byb3BOYW1lLCBjb2x1bW5dIG9mIE9iamVjdC5lbnRyaWVzKGZpeHR1cmUuY29sdW1ucykpIHtcbiAgICAgIGNvbnN0IHByb3AgPSBjb2x1bW4ucHJvcDtcblxuICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIEdlbmVyYXRlZCBjb2x1bW7snYAgSU5TRVJU7JeQ7IScIOygnOyZuCAoRELqsIAg7J6Q64+ZIOyDneyEsSlcbiAgICAgIGlmIChcImdlbmVyYXRlZFwiIGluIHByb3AgJiYgcHJvcC5nZW5lcmF0ZWQpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIGlkIOyymOumrFxuICAgICAgaWYgKHByb3BOYW1lID09PSBcImlkXCIpIHtcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgLy8gcGFyZW50SWQg7JeU7Yuw7Yuw7J2YIGlk64qUIOu2gOuqqCDsl5Tti7Dti7DsnZggRkvsnbTrr4DroZwg7Iuc7YCA7IqkIOuvuOyCrOyaqSAo66qF7Iuc7KCB7Jy866GcIO2PrO2VqClcbiAgICAgICAgY29uc3QgdXNlc1NlcXVlbmNlID1cbiAgICAgICAgICAhZW50aXR5LnBhcmVudElkICYmXG4gICAgICAgICAgKGlkUHJvcD8udHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8udHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8uY29uZT8uZml4dHVyZVN0cmF0ZWd5ID09PSBcInNlcXVlbmNlXCIpO1xuXG4gICAgICAgIGlmIChpc092ZXJyaWRlTW9kZSAmJiBleGlzdGluZ1JlY29yZCkge1xuICAgICAgICAgIC8vIE92ZXJyaWRlOiDquLDsobQg66CI7L2U65Oc7J2YIOqwkiDsgqzsmqkg4oaSIFVQREFURVxuICAgICAgICAgIHJvd1twcm9wTmFtZV0gPSBleGlzdGluZ1JlY29yZC5jb2x1bW5zW3Byb3BOYW1lXT8udmFsdWU7XG4gICAgICAgIH0gZWxzZSBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgIC8vIHN0cmluZyBQSyDrmJDripQgcGFyZW50SWQg7JeU7Yuw7YuwIEZLOiDsg53shLHrkJwgaWQg6rCS7J2EIElOU0VSVOyXkCDtj6ztlahcbiAgICAgICAgICByb3dbcHJvcE5hbWVdID0gY29sdW1uLnZhbHVlO1xuICAgICAgICB9XG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlciBQSzogREIg7Iuc7YCA7Iqk7JeQIOunoeq5gCAo6rCSIOygnOyZuClcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHxcbiAgICAgICAgICAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiBwcm9wLmhhc0pvaW5Db2x1bW4pXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IGNvbHVtbi52YWx1ZSBhcyBudW1iZXIgfCBudWxsO1xuICAgICAgICAgIGlmIChyZWxhdGVkSWQgIT09IG51bGwgJiYgcmVsYXRlZElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG5cbiAgICAgICAgICAgIC8vIOuovOyggCBza2lw65CcIGZpeHR1cmXsnbjsp4Ag7ZmV7J24XG4gICAgICAgICAgICBjb25zdCBza2lwcGVkRXhpc3RpbmdJZCA9IHRoaXMuc2tpcHBlZEZpeHR1cmVzLmdldChyZWxhdGVkRml4dHVyZUlkKT8uZXhpc3RpbmdJZDtcbiAgICAgICAgICAgIGlmIChza2lwcGVkRXhpc3RpbmdJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIHNraXDrkJwgZml4dHVyZSDihpIgdGFyZ2V0IERC7J2YIOq4sOyhtCDroIjsvZTrk5wgaWQg7IKs7JqpXG4gICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHNraXBwZWRFeGlzdGluZ0lkO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQocmVsYXRlZEZpeHR1cmVJZCk7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgICAgLy8g7J2066+4IHVwc2VydOuQnCDqsJnsnYAg7YWM7J2067iUIGZpeHR1cmUg7ZmV7J24XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEluc2VydGVkSWRzID0gaW5zZXJ0ZWRJZHNCeVRhYmxlPy5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgICAgY29uc3QgYWN0dWFsSWQgPSByZWxhdGVkSW5zZXJ0ZWRJZHM/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGFjdHVhbElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIC8vIOydtOuvuCB1cHNlcnTrkKgg4oaSIOyLpOygnCBJRCDsgqzsmqlcbiAgICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IGFjdHVhbElkO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAvLyDslYTsp4EgdXBzZXJ0IOyViOuQqCDihpIgVUJSZWYg7IKs7JqpXG4gICAgICAgICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSByZWxhdGVkUmVmO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAvLyBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOugiOy9lOuTnCDihpIgSUQg6re464yA66GcIOyCrOyaqVxuICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAvLyBIYXNNYW55LCBNYW55VG9NYW5564qUIOuzhOuPhCDsspjrpqxcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIOydvOuwmCDsu6zrn7xcbiAgICAgICAgcm93W3Byb3BOYW1lXSA9IHRoaXMuY29udmVydENvbHVtblZhbHVlKHByb3AgYXMgRW50aXR5UHJvcCwgY29sdW1uLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAhaXNUZXN0KCkgJiZcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoYFJlZ2lzdGVyaW5nICR7ZW50aXR5LnRhYmxlfSAtICR7aW5zcGVjdChyb3csIGZhbHNlLCBudWxsLCB0cnVlKX1gKSk7XG4gICAgY29uc3QgcmVmID0gdGhpcy5idWlsZGVyLnJlZ2lzdGVyKGVudGl0eS50YWJsZSwgcm93KTtcbiAgICB0aGlzLmZpeHR1cmVSZWZNYXAuc2V0KGZpeHR1cmUuZml4dHVyZUlkLCByZWYpO1xuXG4gICAgcmV0dXJuIHJlZjtcbiAgfVxuXG4gIC8qKlxuICAgKiDsu6zrn7wg6rCSIOuzgO2ZmFxuICAgKi9cbiAgcHJpdmF0ZSBjb252ZXJ0Q29sdW1uVmFsdWUocHJvcDogRW50aXR5UHJvcCwgdmFsdWU6IHVua25vd24pOiB1bmtub3duIHtcbiAgICBpZiAodmFsdWUgPT09IG51bGwgfHwgdmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgc3dpdGNoIChwcm9wLnR5cGUpIHtcbiAgICAgIGNhc2UgXCJqc29uXCI6XG4gICAgICAgIC8vIFVwc2VydEJ1aWxkZXIucmVnaXN0ZXLsl5DshJwgSlNPTi5zdHJpbmdpZnkg7LKY66as7ZWY66+A66GcIG9iamVjdCDqt7jrjIDroZwg7KCE64usXG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgY2FzZSBcImRhdGVcIjpcbiAgICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlb2YgdmFsdWUgPT09IFwibnVtYmVyXCIpIHtcbiAgICAgICAgICByZXR1cm4gbmV3IERhdGUodmFsdWUpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgcHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnMoXG4gICAgdHJ4OiBLbmV4LlRyYW5zYWN0aW9uLFxuICAgIGZpeHR1cmVzOiBGaXh0dXJlUmVjb3JkW10sXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlOiBNYXA8c3RyaW5nLCBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+PixcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGZpeHR1cmVzKSB7XG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICAgIGNvbnN0IHNvdXJjZVJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQoZml4dHVyZS5maXh0dXJlSWQpO1xuXG4gICAgICBpZiAoIXNvdXJjZVJlZikgY29udGludWU7XG5cbiAgICAgIGNvbnN0IHNvdXJjZVV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChlbnRpdHkudGFibGUpO1xuICAgICAgY29uc3Qgc291cmNlSWQgPSBzb3VyY2VVdWlkVG9JZD8uZ2V0KHNvdXJjZVJlZi51dWlkKTtcblxuICAgICAgaWYgKHNvdXJjZUlkID09PSB1bmRlZmluZWQpIGNvbnRpbnVlO1xuXG4gICAgICBmb3IgKGNvbnN0IFssIGNvbHVtbl0gb2YgT2JqZWN0LmVudHJpZXMoZml4dHVyZS5jb2x1bW5zKSkge1xuICAgICAgICBjb25zdCBwcm9wID0gY29sdW1uLnByb3A7XG5cbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSAmJiBBcnJheS5pc0FycmF5KGNvbHVtbi52YWx1ZSkpIHtcbiAgICAgICAgICAvLyDshKDtg53rkJjsp4Ag7JWK7J2AIE1hbnlUb01hbnkg6rSA6rOE64qUIOyggOyepe2VmOyngCDslYrsnYxcbiAgICAgICAgICBjb25zdCB0YXJnZXRUYWJsZSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgaWYgKHRoaXMuYnVpbGRlci5oYXNUYWJsZSh0YXJnZXRUYWJsZS50YWJsZSkgPT09IGZhbHNlKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBjb2x1bW4udmFsdWUgYXMgbnVtYmVyW107XG4gICAgICAgICAgaWYgKHJlbGF0ZWRJZHMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IGpvaW5UYWJsZSA9IChwcm9wIGFzIE1hbnlUb01hbnlSZWxhdGlvblByb3ApLmpvaW5UYWJsZTtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcblxuICAgICAgICAgIGNvbnN0IHNvdXJjZUNvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUoZW50aXR5LnRhYmxlKX1faWRgO1xuICAgICAgICAgIGNvbnN0IHRhcmdldENvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGZvciAoY29uc3QgcmVsYXRlZElkIG9mIHJlbGF0ZWRJZHMpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG4gICAgICAgICAgICBjb25zdCByZWxhdGVkUmVmID0gdGhpcy5maXh0dXJlUmVmTWFwLmdldChyZWxhdGVkRml4dHVyZUlkKTtcblxuICAgICAgICAgICAgbGV0IHRhcmdldElkOiBudW1iZXIgfCBzdHJpbmc7XG5cbiAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRVdWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgIGNvbnN0IHJlc29sdmVkSWQgPSByZWxhdGVkVXVpZFRvSWQ/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgIGlmIChyZXNvbHZlZElkID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgICBgUmVsYXRlZCBmaXh0dXJlICR7cmVsYXRlZEZpeHR1cmVJZH0gbm90IGZvdW5kIGluIGluc2VydGVkSWRzLCBza2lwcGluZ2AsXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB0YXJnZXRJZCA9IHJlc29sdmVkSWQ7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICB0YXJnZXRJZCA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gSm9pblRhYmxl7JeQIOyCveyehVxuICAgICAgICAgICAgY29uc3QgW2ZvdW5kXSA9IGF3YWl0IHRyeChqb2luVGFibGUpXG4gICAgICAgICAgICAgIC53aGVyZSh7XG4gICAgICAgICAgICAgICAgW3NvdXJjZUNvbHVtbl06IHNvdXJjZUlkLFxuICAgICAgICAgICAgICAgIFt0YXJnZXRDb2x1bW5dOiB0YXJnZXRJZCxcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgLmxpbWl0KDEpO1xuXG4gICAgICAgICAgICBpZiAoIWZvdW5kKSB7XG4gICAgICAgICAgICAgIGF3YWl0IHRyeChqb2luVGFibGUpLmluc2VydCh7XG4gICAgICAgICAgICAgICAgW3NvdXJjZUNvbHVtbl06IHNvdXJjZUlkLFxuICAgICAgICAgICAgICAgIFt0YXJnZXRDb2x1bW5dOiB0YXJnZXRJZCxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihcbiAgICAgICAgICAgICAgICAgICAgYEluc2VydGVkIGludG8gJHtqb2luVGFibGV9OiAke2VudGl0eS50YWJsZX0oJHtzb3VyY2VJZH0pIC0gJHtyZWxhdGVkRW50aXR5LnRhYmxlfSgke3RhcmdldElkfSlgLFxuICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiDqsJnsnYAg7YWM7J2067iUIOuCtCBmaXh0dXJl65Ok7J2EIHNlbGYtcmVmZXJlbmNlIOugiOuyqOuzhOuhnCDrtoTtlaBcbiAgICogLSBzZWxmLXJlZmVyZW5jZeqwgCDsl4bripQgZml4dHVyZeuTpDogTGV2ZWwgMFxuICAgKiAtIExldmVsIDDsnYQg7LC47KGw7ZWY64qUIGZpeHR1cmXrk6Q6IExldmVsIDFcbiAgICogLSDrsJjrs7VcbiAgICpcbiAgICogVXBzZXJ0QnVpbGRlcuqwgCBzZWxmLXJlZmVyZW5jZeqwgCDsnojsnLzrqbQgYnVpbGRJbnNlcnRMZXZlbHMoKeuhnCDsnqzsoJXroKztlZjsl6xcbiAgICog65Ox66GdIOyInOyEnOyZgCDrsJjtmZgg7Iic7ISc6rCAIOuLrOudvOyniCDsiJgg7J6I7Iq164uI64ukLlxuICAgKiDsnbTrpbwg67Cp7KeA7ZWY6riwIOychO2VtCBGaXh0dXJlTWFuYWdlcuqwgCDroIjrsqjrs4TroZwg64KY64ig7IScIOyymOumrO2VqeuLiOuLpC5cbiAgICovXG4gIHByaXZhdGUgZ3JvdXBGaXh0dXJlc0J5TGV2ZWwoZml4dHVyZXM6IEZpeHR1cmVSZWNvcmRbXSk6IEZpeHR1cmVSZWNvcmRbXVtdIHtcbiAgICBpZiAoZml4dHVyZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZXNbMF0uZW50aXR5SWQpO1xuXG4gICAgLy8gc2VsZi1yZWZlcmVuY2UgcmVsYXRpb24gcHJvcCDssL7quLBcbiAgICBjb25zdCBzZWxmUmVmUHJvcHMgPSBlbnRpdHkucHJvcHMuZmlsdGVyKFxuICAgICAgKHApOiBwIGlzIEJlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCB8IE9uZVRvT25lUmVsYXRpb25Qcm9wID0+XG4gICAgICAgIGlzUmVsYXRpb25Qcm9wKHApICYmXG4gICAgICAgIChpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcChwKSB8fCAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwKSAmJiBwLmhhc0pvaW5Db2x1bW4pKSAmJlxuICAgICAgICBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICApO1xuXG4gICAgaWYgKHNlbGZSZWZQcm9wcy5sZW5ndGggPT09IDApIHtcbiAgICAgIC8vIHNlbGYtcmVmZXJlbmNlIOyXhuydjCDihpIg64uo7J28IOugiOuyqFxuICAgICAgcmV0dXJuIFtmaXh0dXJlc107XG4gICAgfVxuXG4gICAgLy8g66CI67Ko67OEIOu2hO2VoCAodG9wb2xvZ2ljYWwgc29ydClcbiAgICBjb25zdCBsZXZlbHM6IEZpeHR1cmVSZWNvcmRbXVtdID0gW107XG4gICAgY29uc3QgcmVtYWluaW5nID0gbmV3IFNldChmaXh0dXJlcy5tYXAoKGYpID0+IGYuZml4dHVyZUlkKSk7XG4gICAgY29uc3QgcHJvY2Vzc2VkID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICB3aGlsZSAocmVtYWluaW5nLnNpemUgPiAwKSB7XG4gICAgICBjb25zdCBjdXJyZW50TGV2ZWw6IEZpeHR1cmVSZWNvcmRbXSA9IFtdO1xuXG4gICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgaWYgKCFyZW1haW5pbmcuaGFzKGZpeHR1cmUuZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgLy8gc2VsZi1yZWZlcmVuY2XqsIAg66qo65GQIOydtOuvuCDsspjrpqzrkJDqsbDrgpggbnVsbOyduCDqsr3smrBcbiAgICAgICAgY29uc3QgY2FuUHJvY2VzcyA9IHNlbGZSZWZQcm9wcy5ldmVyeSgocHJvcCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHJlZklkID0gZml4dHVyZS5jb2x1bW5zW3Byb3AubmFtZV0/LnZhbHVlIGFzIG51bWJlciB8IG51bGw7XG4gICAgICAgICAgaWYgKHJlZklkID09PSBudWxsIHx8IHJlZklkID09PSB1bmRlZmluZWQpIHJldHVybiB0cnVlO1xuICAgICAgICAgIGNvbnN0IHJlZkZpeHR1cmVJZCA9IGAke3Byb3Aud2l0aH0jJHtyZWZJZH1gO1xuICAgICAgICAgIC8vIOydtOuvuCDsspjrpqzrkJDqsbDrgpgsIO2YhOyerCBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOqyveyasCAo7Jm467aAIOywuOyhsClcbiAgICAgICAgICByZXR1cm4gcHJvY2Vzc2VkLmhhcyhyZWZGaXh0dXJlSWQpIHx8ICFyZW1haW5pbmcuaGFzKHJlZkZpeHR1cmVJZCk7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGlmIChjYW5Qcm9jZXNzKSB7XG4gICAgICAgICAgY3VycmVudExldmVsLnB1c2goZml4dHVyZSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKGN1cnJlbnRMZXZlbC5sZW5ndGggPT09IDApIHtcbiAgICAgICAgY29uc3QgcmVtYWluaW5nSWRzID0gQXJyYXkuZnJvbShyZW1haW5pbmcpLmpvaW4oXCIsIFwiKTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgIGBDaXJjdWxhciBzZWxmLXJlZmVyZW5jZSBkZXRlY3RlZCBpbiAke2VudGl0eS50YWJsZX0uIFJlbWFpbmluZyBmaXh0dXJlczogJHtyZW1haW5pbmdJZHN9YCxcbiAgICAgICAgKTtcbiAgICAgIH1cblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGN1cnJlbnRMZXZlbCkge1xuICAgICAgICByZW1haW5pbmcuZGVsZXRlKGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgcHJvY2Vzc2VkLmFkZChmaXh0dXJlLmZpeHR1cmVJZCk7XG4gICAgICB9XG5cbiAgICAgIGxldmVscy5wdXNoKGN1cnJlbnRMZXZlbCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGxldmVscztcbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgY2hlY2tVbmlxdWVWaW9sYXRpb24oZGI6IEtuZXgsIGVudGl0eTogRW50aXR5LCBmaXh0dXJlOiBGaXh0dXJlUmVjb3JkKSB7XG4gICAgY29uc3QgX3VuaXF1ZUluZGV4ZXMgPSBlbnRpdHkuaW5kZXhlcz8uZmlsdGVyKChpKSA9PiBpLnR5cGUgPT09IFwidW5pcXVlXCIpID8/IFtdO1xuXG4gICAgY29uc3QgdW5pcXVlSW5kZXhlcyA9IF91bmlxdWVJbmRleGVzLmZpbHRlcigoaW5kZXgpID0+XG4gICAgICBpbmRleC5jb2x1bW5zLmV2ZXJ5KChjb2x1bW4pID0+ICFjb2x1bW4ubmFtZS5zdGFydHNXaXRoKGAke2VudGl0eS50YWJsZX1fX2ApKSxcbiAgICApO1xuICAgIGlmICh1bmlxdWVJbmRleGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgbGV0IHVuaXF1ZVF1ZXJ5ID0gZGIoZW50aXR5LnRhYmxlKTtcbiAgICBsZXQgaGFzQ29uZGl0aW9uID0gZmFsc2U7XG5cbiAgICBmb3IgKGNvbnN0IGluZGV4IG9mIHVuaXF1ZUluZGV4ZXMpIHtcbiAgICAgIC8vIOy7rOufvCDspJEg7ZWY64KY652864+EIG51bGzsnbTrqbQg7Jyg64uI7YGsIOygnOyVveydhCDsnITrsJjtlZjsp4Ag7JWK6riwIOuVjOusuOyXkCDtlbTri7kg7J24642x7Iqk64qUIOustOyLnFxuICAgICAgY29uc3QgY29udGFpbnNOdWxsID0gaW5kZXguY29sdW1ucy5zb21lKChjb2x1bW4pID0+IHtcbiAgICAgICAgY29uc3QgZmllbGQgPSBjb2x1bW4ubmFtZS5yZXBsYWNlKC9faWQkLywgXCJcIik7XG4gICAgICAgIHJldHVybiBmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSA9PT0gbnVsbDtcbiAgICAgIH0pO1xuICAgICAgaWYgKGNvbnRhaW5zTnVsbCkge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgdW5pcXVlUXVlcnkgPSB1bmlxdWVRdWVyeS5vcldoZXJlKChxYikgPT4ge1xuICAgICAgICBmb3IgKGNvbnN0IGNvbHVtbiBvZiBpbmRleC5jb2x1bW5zKSB7XG4gICAgICAgICAgY29uc3QgZmllbGQgPSBjb2x1bW4ubmFtZS5yZXBsYWNlKC9faWQkLywgXCJcIik7XG5cbiAgICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSkpIHtcbiAgICAgICAgICAgIHFiLndoZXJlSW4oY29sdW1uLm5hbWUsIGZpeHR1cmUuY29sdW1uc1tmaWVsZF0udmFsdWUpO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBxYi5hbmRXaGVyZShjb2x1bW4ubmFtZSwgZml4dHVyZS5jb2x1bW5zW2ZpZWxkXT8udmFsdWUpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgICBoYXNDb25kaXRpb24gPSB0cnVlO1xuICAgIH1cblxuICAgIGlmICghaGFzQ29uZGl0aW9uKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBbdW5pcXVlRm91bmRdID0gYXdhaXQgdW5pcXVlUXVlcnk7XG4gICAgcmV0dXJuIHVuaXF1ZUZvdW5kO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBjaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICBkYjogS25leCxcbiAgICBlbnRpdHk6IEVudGl0eSxcbiAgICBmaXh0dXJlOiBGaXh0dXJlUmVjb3JkLFxuICAgIGNvbHVtbnM6IHN0cmluZ1tdLFxuICApIHtcbiAgICBpZiAoY29sdW1ucy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGNvbnN0IHdoZXJlQ2xhdXNlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuXG4gICAgZm9yIChjb25zdCBjb2x1bW4gb2YgY29sdW1ucykge1xuICAgICAgLy8gcmVsYXRpb24g7ZWE65Oc7J24IOqyveyasCBfaWQg67aZ7J206riwXG4gICAgICBjb25zdCBwcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gY29sdW1uKTtcbiAgICAgIGNvbnN0IGRiQ29sdW1uID0gcHJvcCAmJiBpc1JlbGF0aW9uUHJvcChwcm9wKSA/IGAke2NvbHVtbn1faWRgIDogY29sdW1uO1xuICAgICAgY29uc3QgdmFsdWUgPSBmaXh0dXJlLmNvbHVtbnNbY29sdW1uXT8udmFsdWU7XG5cbiAgICAgIC8vIG51bGwg6rCS7J20IO2PrO2VqOuQnCDqsr3smrAg7KSR67O1IO2ZleyduCDsiqTtgrVcbiAgICAgIGlmICh2YWx1ZSA9PT0gbnVsbCB8fCB2YWx1ZSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHJldHVybiBudWxsO1xuICAgICAgfVxuXG4gICAgICB3aGVyZUNsYXVzZVtkYkNvbHVtbl0gPSB2YWx1ZTtcbiAgICB9XG5cbiAgICBjb25zdCBbZm91bmRdID0gYXdhaXQgZGIoZW50aXR5LnRhYmxlKS53aGVyZSh3aGVyZUNsYXVzZSkubGltaXQoMSk7XG4gICAgcmV0dXJuIGZvdW5kO1xuICB9XG5cbiAgYXN5bmMgYWRkRml4dHVyZUxvYWRlcihjb2RlOiBzdHJpbmcpIHtcbiAgICBjb25zdCBwYXRoID0gYCR7U29uYW11LmFwaVJvb3RQYXRofS9zcmMvdGVzdGluZy9maXh0dXJlLnRzYDtcbiAgICBjb25zdCBjb250ZW50ID0gcmVhZEZpbGVTeW5jKHBhdGgpLnRvU3RyaW5nKCk7XG5cbiAgICBjb25zdCBmaXh0dXJlTG9hZGVyU3RhcnQgPSBjb250ZW50LmluZGV4T2YoXCJjb25zdCBmaXh0dXJlTG9hZGVyID0ge1wiKTtcbiAgICBjb25zdCBmaXh0dXJlTG9hZGVyRW5kID0gY29udGVudC5pbmRleE9mKFwifTtcIiwgZml4dHVyZUxvYWRlclN0YXJ0KTtcblxuICAgIGlmIChmaXh0dXJlTG9hZGVyU3RhcnQgIT09IC0xICYmIGZpeHR1cmVMb2FkZXJFbmQgIT09IC0xKSB7XG4gICAgICBjb25zdCBuZXdDb250ZW50ID0gYCR7Y29udGVudC5zbGljZSgwLCBmaXh0dXJlTG9hZGVyRW5kKX0gICR7Y29kZX1cXG4ke2NvbnRlbnQuc2xpY2UoZml4dHVyZUxvYWRlckVuZCl9YDtcblxuICAgICAgd3JpdGVGaWxlU3luYyhwYXRoLCBuZXdDb250ZW50KTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFwiRmFpbGVkIHRvIGZpbmQgZml4dHVyZUxvYWRlciBpbiBmaXh0dXJlLnRzXCIpO1xuICAgIH1cbiAgfVxufVxuXG5leHBvcnQgY29uc3QgRml4dHVyZU1hbmFnZXIgPSBuZXcgRml4dHVyZU1hbmFnZXJDbGFzcygpO1xuIl0sIm5hbWVzIjpbImFzc2VydCIsImNoYWxrIiwiZXhlY1N5bmMiLCJyZWFkRmlsZVN5bmMiLCJ3cml0ZUZpbGVTeW5jIiwiaW5mbGVjdGlvbiIsInVuaXF1ZSIsImluc3BlY3QiLCJTb25hbXUiLCJCYXNlTW9kZWwiLCJjcmVhdGVLbmV4SW5zdGFuY2UiLCJVcHNlcnRCdWlsZGVyIiwiRW50aXR5TWFuYWdlciIsImlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIiwiaXNIYXNNYW55UmVsYXRpb25Qcm9wIiwiaXNNYW55VG9NYW55UmVsYXRpb25Qcm9wIiwiaXNPbmVUb09uZVJlbGF0aW9uUHJvcCIsImlzUmVsYXRpb25Qcm9wIiwiaXNWaXJ0dWFsUHJvcCIsImlzVGVzdCIsIlJlbGF0aW9uR3JhcGgiLCJGaXh0dXJlTWFuYWdlckNsYXNzIiwiX3RkYiIsInRkYiIsIkVycm9yIiwiX2ZkYiIsImZkYiIsImNhY2hlZFRhYmxlTmFtZXMiLCJyZWxhdGlvbkdyYXBoIiwiYnVpbGRlciIsImZpeHR1cmVSZWZNYXAiLCJNYXAiLCJza2lwcGVkRml4dHVyZXMiLCJpbml0IiwiZGJDb25maWciLCJ0ZXN0IiwicHJvZHVjdGlvbl9tYXN0ZXIiLCJ0Q29ubiIsImNvbm5lY3Rpb24iLCJwQ29ubiIsImhvc3QiLCJwb3J0IiwiZGF0YWJhc2UiLCJmaXh0dXJlIiwic3luYyIsImZpeHR1cmVDb25uIiwidGVzdENvbm4iLCJ0ZXN0UGdFbnYiLCJQR1BBU1NXT1JEIiwicGFzc3dvcmQiLCJ1c2VyIiwic3RkaW8iLCJlbnYiLCJwcm9jZXNzIiwiZml4dHVyZVBnRW52IiwiZHVtcENtZCIsInJlc3RvcmVDbWQiLCJzaGVsbCIsInJlc2V0U2VxdWVuY2VzIiwidGVzdERiIiwiZW50aXRpZXMiLCJnZXRBbGxFbnRpdGllcyIsImVudGl0eSIsInRhYmxlTmFtZSIsInRhYmxlIiwiaWQiLCJ0b0xvd2VyQ2FzZSIsImlkUHJvcCIsInByb3BzIiwiZmluZCIsInAiLCJuYW1lIiwiaWRUeXBlIiwidHlwZSIsInVzZXNTZXF1ZW5jZSIsImNvbmUiLCJmaXh0dXJlU3RyYXRlZ3kiLCJjb25zb2xlIiwibG9nIiwibWF4RXhwciIsInJhdyIsImRlc3Ryb3kiLCJ2aXNpdGVkUmVjb3JkcyIsIlNldCIsImltcG9ydEZpeHR1cmUiLCJlbnRpdHlJZCIsImlkcyIsImNsZWFyIiwicXVlcmllcyIsIlByb21pc2UiLCJhbGwiLCJtYXAiLCJnZXRJbXBvcnRRdWVyaWVzIiwiZmxhdCIsIndkYiIsImdldERCIiwicXVlcnkiLCJmaWVsZCIsInJlY29yZEtleSIsImhhcyIsImFkZCIsImdldCIsInJvdyIsIndoZXJlIiwibGltaXQiLCJ1bmRlZmluZWQiLCJmaXh0dXJlRGF0YWJhc2UiLCJyZWFsRGF0YWJhc2UiLCJzZWxmUXVlcnkiLCJhcmdzIiwiT2JqZWN0IiwiZW50cmllcyIsInJlbGF0aW9ucyIsImZpbHRlciIsInJlbGF0aW9uIiwiY3VzdG9tSm9pbkNsYXVzZSIsImhhc0pvaW5Db2x1bW4iLCJyZWxhdGVkRW50aXR5Iiwid2l0aCIsInJlbGF0ZWRJZENvbHVtbk5hbWUiLCJhcmciLCJyZWxRdWVyaWVzIiwicmV2ZXJzZSIsImdldEZpeHR1cmVzIiwic291cmNlREJOYW1lIiwidGFyZ2V0REJOYW1lIiwic2VhcmNoT3B0aW9ucyIsImR1cGxpY2F0ZUNoZWNrIiwic291cmNlREIiLCJ0YXJnZXREQiIsInZhbHVlIiwic2VhcmNoVHlwZSIsImNvbHVtbiIsInByb3AiLCJyb3dzIiwibGVuZ3RoIiwiZml4dHVyZXMiLCJpbml0aWFsUmVjb3Jkc0xlbmd0aCIsIm5ld1JlY29yZHMiLCJjcmVhdGVGaXh0dXJlUmVjb3JkIiwiX2RiIiwicHVzaCIsImN1cnJlbnRGaXh0dXJlUmVjb3JkIiwiciIsImZpeHR1cmVJZCIsImZldGNoZWRSZWNvcmRzIiwic2xpY2UiLCJjdXN0b21Db2x1bW5zIiwiY29sdW1ucyIsImN1c3RvbUR1cGxpY2F0ZVJvdyIsImNoZWNrRHVwbGljYXRlQnlDb2x1bW5zIiwicmVjb3JkIiwic2luZ2xlUmVjb3JkIiwidGFyZ2V0IiwidW5pcXVlUm93IiwiY2hlY2tVbmlxdWVWaW9sYXRpb24iLCJmIiwiYWxsU2V0dGxlZCIsIm9wdGlvbnMiLCJyZWNvcmRzIiwidmlzaXRlZEVudGl0aWVzIiwiY3JlYXRlIiwiYmVsb25nc1JlY29yZHMiLCJkYiIsInRocm91Z2hUYWJsZSIsImpvaW5UYWJsZSIsImZyb21Db2x1bW4iLCJzaW5ndWxhcml6ZSIsInRvQ29sdW1uIiwicmVsYXRlZElkcyIsInBsdWNrIiwiam9pbkNvbHVtbiIsInJlbGF0ZWRQcm9wIiwiZmtDb2x1bW4iLCJyZWxhdGVkUm93IiwiZmlyc3QiLCJyZWxhdGVkSWQiLCJpbnNlcnRGaXh0dXJlcyIsImRiTmFtZSIsIl9maXh0dXJlcyIsIlNPTkFNVV9XT1JLRVJfREIiLCJWSVRFU1RfUE9PTF9JRCIsIndvcmtlcklkIiwicGFyc2VJbnQiLCJiYXNlQ29uZmlnIiwicG9vbCIsIm1pbiIsIm1heCIsInJlc3VsdHMiLCJidWlsZEdyYXBoIiwiaW5zZXJ0aW9uT3JkZXIiLCJnZXRJbnNlcnRpb25PcmRlciIsImhhc1RhcmdldCIsImhhc1VuaXF1ZSIsImhhc0R1cGxpY2F0ZSIsIm92ZXJyaWRlIiwiZXhpc3RpbmdJZCIsInNldCIsInllbGxvdyIsImZpeHR1cmVzQnlUYWJsZSIsInRhYmxlT3JkZXIiLCJ0cmFuc2FjdGlvbiIsInRyeCIsImluc2VydGVkSWRzQnlUYWJsZSIsInRhYmxlRml4dHVyZXMiLCJsZXZlbHMiLCJncm91cEZpeHR1cmVzQnlMZXZlbCIsImxldmVsRml4dHVyZXMiLCJyZWdpc3RlckZpeHR1cmUiLCJibHVlIiwiZ2V0VGFibGUiLCJ1dWlkcyIsInV1aWQiLCJpbmRleE9mIiwidXBzZXJ0IiwiZXhpc3RpbmdNYXAiLCJpIiwid2FybiIsInByb2Nlc3NNYW55VG9NYW55UmVsYXRpb25zIiwiZSIsImdyYXkiLCJlbnRpdHkyIiwiaWRUeXBlMiIsIm1heElkUmVzdWx0IiwidGhlbiIsIm1heElkIiwibWF4X2lkIiwiZ3JlZW4iLCJfZXJyIiwic2tpcHBlZCIsImRhdGEiLCJyZWYiLCJ1dWlkVG9JZCIsImluc2VydGVkSWQiLCJleGlzdGluZ1JlY29yZCIsImlzT3ZlcnJpZGVNb2RlIiwicHJvcE5hbWUiLCJnZW5lcmF0ZWQiLCJwYXJlbnRJZCIsInJlbGF0ZWRGaXh0dXJlSWQiLCJza2lwcGVkRXhpc3RpbmdJZCIsInJlbGF0ZWRSZWYiLCJyZWxhdGVkSW5zZXJ0ZWRJZHMiLCJhY3R1YWxJZCIsImNvbnZlcnRDb2x1bW5WYWx1ZSIsInJlZ2lzdGVyIiwiRGF0ZSIsInNvdXJjZVJlZiIsInNvdXJjZVV1aWRUb0lkIiwic291cmNlSWQiLCJBcnJheSIsImlzQXJyYXkiLCJ0YXJnZXRUYWJsZSIsImhhc1RhYmxlIiwic291cmNlQ29sdW1uIiwidGFyZ2V0Q29sdW1uIiwidGFyZ2V0SWQiLCJyZWxhdGVkVXVpZFRvSWQiLCJyZXNvbHZlZElkIiwiZm91bmQiLCJpbnNlcnQiLCJzZWxmUmVmUHJvcHMiLCJyZW1haW5pbmciLCJwcm9jZXNzZWQiLCJzaXplIiwiY3VycmVudExldmVsIiwiY2FuUHJvY2VzcyIsImV2ZXJ5IiwicmVmSWQiLCJyZWZGaXh0dXJlSWQiLCJyZW1haW5pbmdJZHMiLCJmcm9tIiwiam9pbiIsImRlbGV0ZSIsIl91bmlxdWVJbmRleGVzIiwiaW5kZXhlcyIsInVuaXF1ZUluZGV4ZXMiLCJpbmRleCIsInN0YXJ0c1dpdGgiLCJ1bmlxdWVRdWVyeSIsImhhc0NvbmRpdGlvbiIsImNvbnRhaW5zTnVsbCIsInNvbWUiLCJyZXBsYWNlIiwib3JXaGVyZSIsInFiIiwid2hlcmVJbiIsImFuZFdoZXJlIiwidW5pcXVlRm91bmQiLCJ3aGVyZUNsYXVzZSIsImRiQ29sdW1uIiwiYWRkRml4dHVyZUxvYWRlciIsImNvZGUiLCJwYXRoIiwiYXBpUm9vdFBhdGgiLCJjb250ZW50IiwidG9TdHJpbmciLCJmaXh0dXJlTG9hZGVyU3RhcnQiLCJmaXh0dXJlTG9hZGVyRW5kIiwibmV3Q29udGVudCIsIkZpeHR1cmVNYW5hZ2VyIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxZQUFZLFNBQVM7QUFDNUIsT0FBT0MsV0FBVyxRQUFRO0FBQzFCLFNBQVNDLFFBQVEsUUFBUSxnQkFBZ0I7QUFDekMsU0FBU0MsWUFBWSxFQUFFQyxhQUFhLFFBQVEs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
718
+ //#endregion
719
+ init_fixture_manager();
720
+ export { FixtureManager, FixtureManagerClass, init_fixture_manager };
721
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4dHVyZS1tYW5hZ2VyLmpzIiwibmFtZXMiOlsiZmllbGQ6IHN0cmluZyIsImlkOiBudW1iZXIiLCJhcmdzIiwiZml4dHVyZXM6IEZpeHR1cmVSZWNvcmRbXSIsImVudGl0eSIsInJlY29yZHM6IEZpeHR1cmVSZWNvcmRbXSIsInJvdyIsInJlY29yZDogRml4dHVyZVJlY29yZCIsInJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSIsInRhYmxlT3JkZXI6IHN0cmluZ1tdIiwicm93OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiIsInRhcmdldElkOiBudW1iZXIgfCBzdHJpbmciLCJsZXZlbHM6IEZpeHR1cmVSZWNvcmRbXVtdIiwiY3VycmVudExldmVsOiBGaXh0dXJlUmVjb3JkW10iLCJ3aGVyZUNsYXVzZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4iXSwic291cmNlcyI6WyIuLi8uLi9zcmMvdGVzdGluZy9maXh0dXJlLW1hbmFnZXIudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFzc2VydCBmcm9tIFwiYXNzZXJ0XCI7XG5pbXBvcnQgeyBleGVjU3luYyB9IGZyb20gXCJjaGlsZF9wcm9jZXNzXCI7XG5pbXBvcnQgeyByZWFkRmlsZVN5bmMsIHdyaXRlRmlsZVN5bmMgfSBmcm9tIFwiZnNcIjtcbmltcG9ydCB7IGluc3BlY3QgfSBmcm9tIFwidXRpbFwiO1xuXG5pbXBvcnQgY2hhbGsgZnJvbSBcImNoYWxrXCI7XG5pbXBvcnQgaW5mbGVjdGlvbiBmcm9tIFwiaW5mbGVjdGlvblwiO1xuaW1wb3J0IHsgdHlwZSBLbmV4IH0gZnJvbSBcImtuZXhcIjtcbmltcG9ydCB7IHVuaXF1ZSB9IGZyb20gXCJyYWRhc2hpXCI7XG5cbmltcG9ydCB7IFNvbmFtdSB9IGZyb20gXCIuLi9hcGkvc29uYW11XCI7XG5pbXBvcnQgeyBCYXNlTW9kZWwgfSBmcm9tIFwiLi4vZGF0YWJhc2UvYmFzZS1tb2RlbFwiO1xuaW1wb3J0IHsgdHlwZSBTb25hbXVEQkNvbmZpZyB9IGZyb20gXCIuLi9kYXRhYmFzZS9kYlwiO1xuaW1wb3J0IHsgY3JlYXRlS25leEluc3RhbmNlIH0gZnJvbSBcIi4uL2RhdGFiYXNlL2tuZXhcIjtcbmltcG9ydCB7IFVwc2VydEJ1aWxkZXIgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB7IHR5cGUgVUJSZWYgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB7IHR5cGUgRW50aXR5IH0gZnJvbSBcIi4uL2VudGl0eS9lbnRpdHlcIjtcbmltcG9ydCB7IEVudGl0eU1hbmFnZXIgfSBmcm9tIFwiLi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5pbXBvcnQge1xuICBpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNIYXNNYW55UmVsYXRpb25Qcm9wLFxuICBpc01hbnlUb01hbnlSZWxhdGlvblByb3AsXG4gIGlzT25lVG9PbmVSZWxhdGlvblByb3AsXG4gIGlzUmVsYXRpb25Qcm9wLFxuICBpc1ZpcnR1YWxQcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7XG4gIHR5cGUgQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wLFxuICB0eXBlIERhdGFiYXNlU2NoZW1hRXh0ZW5kLFxuICB0eXBlIEVudGl0eVByb3AsXG4gIHR5cGUgRml4dHVyZUltcG9ydFJlc3VsdCxcbiAgdHlwZSBGaXh0dXJlUmVjb3JkLFxuICB0eXBlIEZpeHR1cmVTZWFyY2hPcHRpb25zLFxuICB0eXBlIE9uZVRvT25lUmVsYXRpb25Qcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7IGlzVGVzdCB9IGZyb20gXCIuLi91dGlscy9jb250cm9sbGVyXCI7XG5pbXBvcnQgeyBSZWxhdGlvbkdyYXBoIH0gZnJvbSBcIi4vX3JlbGF0aW9uLWdyYXBoXCI7XG5cbi8qKiDsgqzsmqnsnpAg7KeA7KCVIOykkeuztSDtmZXsnbgg7Lus65+8IChlbnRpdHlJZOuzhOuhnCDsp4DsoJUpICovXG5leHBvcnQgaW50ZXJmYWNlIER1cGxpY2F0ZUNoZWNrT3B0aW9ucyB7XG4gIGNvbHVtbnM/OiB7XG4gICAgW2VudGl0eUlkOiBzdHJpbmddOiBzdHJpbmdbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGNsYXNzIEZpeHR1cmVNYW5hZ2VyQ2xhc3Mge1xuICBwcml2YXRlIF90ZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IHRkYih0ZGI6IEtuZXgpIHtcbiAgICB0aGlzLl90ZGIgPSB0ZGI7XG4gIH1cbiAgZ2V0IHRkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fdGRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl90ZGI7XG4gIH1cblxuICBwcml2YXRlIF9mZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IGZkYihmZGI6IEtuZXgpIHtcbiAgICB0aGlzLl9mZGIgPSBmZGI7XG4gIH1cbiAgZ2V0IGZkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fZmRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9mZGI7XG4gIH1cbiAgY2FjaGVkVGFibGVOYW1lczogc3RyaW5nW10gfCBudWxsID0gbnVsbDtcblxuICBwcml2YXRlIHJlbGF0aW9uR3JhcGggPSBuZXcgUmVsYXRpb25HcmFwaCgpO1xuXG4gIC8vIFVwc2VydEJ1aWxkZXIg6riw67CYIGltcG9ydOulvCDsnITtlZwg7IOB7YOcXG4gIHByaXZhdGUgYnVpbGRlcjogVXBzZXJ0QnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gIHByaXZhdGUgZml4dHVyZVJlZk1hcDogTWFwPHN0cmluZywgVUJSZWY+ID0gbmV3IE1hcCgpO1xuICBwcml2YXRlIHNraXBwZWRGaXh0dXJlczogTWFwPHN0cmluZywgeyBlbnRpdHlJZDogc3RyaW5nOyBleGlzdGluZ0lkOiBudW1iZXIgfCBzdHJpbmcgfT4gPVxuICAgIG5ldyBNYXAoKTtcblxuICBpbml0KCkge1xuICAgIGlmICh0aGlzLl90ZGIgIT09IG51bGwpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKFNvbmFtdS5kYkNvbmZpZy50ZXN0ICYmIFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlcikge1xuICAgICAgY29uc3QgdENvbm4gPSBTb25hbXUuZGJDb25maWcudGVzdC5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBjb25zdCBwQ29ubiA9IFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlci5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBpZiAoXG4gICAgICAgIGAke3RDb25uLmhvc3QgPz8gXCJsb2NhbGhvc3RcIn06JHt0Q29ubi5wb3J0ID8/IDU0MzJ9LyR7dENvbm4uZGF0YWJhc2V9YCA9PT1cbiAgICAgICAgYCR7cENvbm4uaG9zdCA/PyBcImxvY2FsaG9zdFwifToke3BDb25uLnBvcnQgPz8gNTQzMn0vJHtwQ29ubi5kYXRhYmFzZX1gXG4gICAgICApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDthYzsiqTtirhEQuyZgCDtlITroZzrjZXshZhEQuyXkCDrj5nsnbztlZwg642w7J207YSw67Kg7J207Iqk6rCAIOyCrOyaqeuQmOyXiOyKteuLiOuLpC5gKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICB0aGlzLnRkYiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWcudGVzdCk7XG4gICAgdGhpcy5mZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUpO1xuICB9XG5cbiAgLyoqXG4gICAg7JuQ6rKpIGZpeHR1cmUgRELrpbwg66Gc7LusIHRlc3QgRELroZwg67O17IKs7ZWp64uI64ukLlxuICAgIHBnX2R1bXDroZwg7JuQ6rKpIERC66W8IOuNpO2UhO2VmOqzoCwgcGdfcmVzdG9yZeuhnCDroZzsu6zsl5Ag67O17JuQ7ZWp64uI64ukLlxuICAqL1xuICBhc3luYyBzeW5jKCkge1xuICAgIGNvbnN0IGZpeHR1cmVDb25uID0gU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LlBnQ29ubmVjdGlvbkNvbmZpZztcbiAgICBjb25zdCB0ZXN0Q29ubiA9IFNvbmFtdS5kYkNvbmZpZy50ZXN0LmNvbm5lY3Rpb24gYXMgS25leC5QZ0Nvbm5lY3Rpb25Db25maWc7XG5cbiAgICAvLyAxLiDroZzsu6wgdGVzdCBEQiDsl7DqsrAg7KKF66OMIOuwjyDsnqzsg53shLFcbiAgICBjb25zdCB0ZXN0UGdFbnYgPSB7IFBHUEFTU1dPUkQ6IHRlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCIgfTtcbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJcbiAgICAgICAgU0VMRUNUIHBnX3Rlcm1pbmF0ZV9iYWNrZW5kKHBnX3N0YXRfYWN0aXZpdHkucGlkKVxuICAgICAgICBGUk9NIHBnX3N0YXRfYWN0aXZpdHlcbiAgICAgICAgV0hFUkUgZGF0bmFtZSA9ICcke3Rlc3RDb25uLmRhdGFiYXNlfSdcbiAgICAgICAgICBBTkQgcGlkIDw+IHBnX2JhY2tlbmRfcGlkKCk7XG4gICAgICBcImAsXG4gICAgICB7IHN0ZGlvOiBcImluaGVyaXRcIiwgZW52OiB7IC4uLnByb2Nlc3MuZW52LCAuLi50ZXN0UGdFbnYgfSBhcyBOb2RlSlMuUHJvY2Vzc0VudiB9LFxuICAgICk7XG5cbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJEUk9QIERBVEFCQVNFIElGIEVYSVNUUyBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIGV4ZWNTeW5jKFxuICAgICAgYHBzcWwgLWggJHt0ZXN0Q29ubi5ob3N0fSAtcCAke3Rlc3RDb25uLnBvcnQgPz8gNTQzMn0gLVUgJHt0ZXN0Q29ubi51c2VyfSAtZCBwb3N0Z3JlcyAtYyBcIkNSRUFURSBEQVRBQkFTRSBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIC8vIDIuIOybkOqyqSBmaXh0dXJlIERCIOKGkiDroZzsu6wgdGVzdCBEQuuhnCDrs7XsgqwgKHBnX2R1bXAgfCBwZ19yZXN0b3JlKVxuICAgIGNvbnN0IGZpeHR1cmVQZ0VudiA9IHsgUEdQQVNTV09SRDogZml4dHVyZUNvbm4ucGFzc3dvcmQgfHwgXCJcIiB9O1xuICAgIGNvbnN0IGR1bXBDbWQgPSBgcGdfZHVtcCAtaCAke2ZpeHR1cmVDb25uLmhvc3R9IC1wICR7Zml4dHVyZUNvbm4ucG9ydCA/PyA1NDMyfSAtVSAke2ZpeHR1cmVDb25uLnVzZXJ9IC1kICR7Zml4dHVyZUNvbm4uZGF0YWJhc2V9IC1GY2A7XG4gICAgY29uc3QgcmVzdG9yZUNtZCA9IGBwZ19yZXN0b3JlIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgJHt0ZXN0Q29ubi5kYXRhYmFzZX0gLS1uby1vd25lciAtLW5vLWFjbGA7XG5cbiAgICBleGVjU3luYyhgJHtkdW1wQ21kfSB8IFBHUEFTU1dPUkQ9XCIke3Rlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCJ9XCIgJHtyZXN0b3JlQ21kfWAsIHtcbiAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICAgIGVudjogeyAuLi5wcm9jZXNzLmVudiwgLi4uZml4dHVyZVBnRW52IH0gYXMgTm9kZUpTLlByb2Nlc3NFbnYsXG4gICAgICBzaGVsbDogXCIvYmluL2Jhc2hcIixcbiAgICB9KTtcblxuICAgIC8vIDMuIOyLnO2AgOyKpCDrpqzshYsgKOuNsOydtO2EsCDrs7Xsgqwg7ZuEIOyLnO2AgOyKpOulvCBNQVgoaWQp66GcIOygleugrClcbiAgICBhd2FpdCB0aGlzLnJlc2V0U2VxdWVuY2VzKFNvbmFtdS5kYkNvbmZpZy50ZXN0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiDrqqjrk6Ag7YWM7J2067iU7J2YIOyLnO2AgOyKpOulvCDtmITsnqwgTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAqIGZpeHR1cmUgc3luYyDtm4Qg7Iuc7YCA7Iqk6rCAIOyLpOygnCDrjbDsnbTthLDsmYAg66ee7KeAIOyViuuKlCDrrLjsoJzrpbwg7ZW06rKw7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyByZXNldFNlcXVlbmNlcyhkYkNvbmZpZzogU29uYW11REJDb25maWdbXCJ0ZXN0XCJdKSB7XG4gICAgY29uc3QgdGVzdERiID0gY3JlYXRlS25leEluc3RhbmNlKGRiQ29uZmlnKTtcbiAgICBjb25zdCBlbnRpdGllcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKTtcblxuICAgIHRyeSB7XG4gICAgICBmb3IgKGNvbnN0IGVudGl0eSBvZiBlbnRpdGllcykge1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGUgfHwgZW50aXR5LmlkLnRvTG93ZXJDYXNlKCk7XG5cbiAgICAgICAgLy8gaWQg7ZWE65Oc7J2YIO2DgOyeheydhCDtmZXsnbjtlanri4jri6RcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgY29uc3QgaWRUeXBlID0gaWRQcm9wPy50eXBlO1xuXG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlcuydtOqxsOuCmCwgc3RyaW5n7J207KeA66eMIGZpeHR1cmVTdHJhdGVneT1zZXF1ZW5jZeyduCDqsr3smrDsl5Drp4wg66as7IWL7ZWp64uI64ukXG4gICAgICAgIGNvbnN0IHVzZXNTZXF1ZW5jZSA9XG4gICAgICAgICAgaWRUeXBlID09PSBcImludGVnZXJcIiB8fFxuICAgICAgICAgIGlkVHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICBpZFByb3A/LmNvbmU/LmZpeHR1cmVTdHJhdGVneSA9PT0gXCJzZXF1ZW5jZVwiO1xuXG4gICAgICAgIGlmICghdXNlc1NlcXVlbmNlKSB7XG4gICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgYFNraXBwaW5nIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX0gKGlkIHR5cGU6ICR7aWRUeXBlIHx8IFwidW5rbm93blwifSlgLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFBvc3RncmVTUUwg7Iuc7YCA7Iqk66W8IO2YhOyerCDthYzsnbTruJTsnZggTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAgICAgIC8vIHN0cmluZyDtg4DsnoXsnZgg6rK97JqwIOyIq+yekCDsupDsiqTtjIXsnbQg7ZWE7JqU7ZWp64uI64ukLlxuICAgICAgICBjb25zdCBtYXhFeHByID0gaWRUeXBlID09PSBcInN0cmluZ1wiID8gXCJNQVgoaWQ6OmJpZ2ludClcIiA6IFwiTUFYKGlkKVwiO1xuICAgICAgICBhd2FpdCB0ZXN0RGIucmF3KGBcbiAgICAgICAgICBTRUxFQ1Qgc2V0dmFsKFxuICAgICAgICAgICAgcGdfZ2V0X3NlcmlhbF9zZXF1ZW5jZSgncHVibGljLiR7dGFibGVOYW1lfScsICdpZCcpLFxuICAgICAgICAgICAgQ09BTEVTQ0UoKFNFTEVDVCAke21heEV4cHJ9IEZST00gJHt0YWJsZU5hbWV9KSwgMSlcbiAgICAgICAgICApXG4gICAgICAgIGApO1xuICAgICAgfVxuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCB0ZXN0RGIuZGVzdHJveSgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgdmlzaXRlZFJlY29yZHMgPSBuZXcgU2V0PHN0cmluZz4oKTtcbiAgYXN5bmMgaW1wb3J0Rml4dHVyZShlbnRpdHlJZDogc3RyaW5nLCBpZHM6IG51bWJlcltdKSB7XG4gICAgLy8g67Cp66y4IOq4sOuhnSDstIjquLDtmZQgKOyDiOuhnOyatCBpbXBvcnQg7J6R7JeFIOyLnOyekSlcbiAgICB0aGlzLnZpc2l0ZWRSZWNvcmRzLmNsZWFyKCk7XG5cbiAgICBjb25zdCBxdWVyaWVzID0gdW5pcXVlKFxuICAgICAgKFxuICAgICAgICBhd2FpdCBQcm9taXNlLmFsbChcbiAgICAgICAgICBpZHMubWFwKGFzeW5jIChpZCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZCwgXCJpZFwiLCBpZCk7XG4gICAgICAgICAgfSksXG4gICAgICAgIClcbiAgICAgICkuZmxhdCgpLFxuICAgICk7XG5cbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuICAgIGZvciAoY29uc3QgcXVlcnkgb2YgcXVlcmllcykge1xuICAgICAgYXdhaXQgd2RiLnJhdyhxdWVyeSk7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZDogc3RyaW5nLCBmaWVsZDogc3RyaW5nLCBpZDogbnVtYmVyKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHJlY29yZEtleSA9IGAke2VudGl0eUlkfSMke2ZpZWxkfSMke2lkfWA7XG5cbiAgICAvLyDsiJztmZgg7LC47KGwIOuwqeyngDog7J2066+4IOuwqeusuO2VnCDroIjsvZTrk5zripQg7Iqk7YK1XG4gICAgaWYgKHRoaXMudmlzaXRlZFJlY29yZHMuaGFzKHJlY29yZEtleSkpIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG4gICAgdGhpcy52aXNpdGVkUmVjb3Jkcy5hZGQocmVjb3JkS2V5KTtcblxuICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKTtcbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuXG4gICAgLy8g7Jes6riw7IScIOyLpERC7J2YIHJvdyDqsIDsoLjsmLRcbiAgICBjb25zdCBbcm93XSA9IGF3YWl0IHdkYihlbnRpdHkudGFibGUpLndoZXJlKGZpZWxkLCBpZCkubGltaXQoMSk7XG4gICAgaWYgKHJvdyA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5SWR9IyR7aWR9IHJvd+ulvCDssL7snYQg7IiYIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICAvLyDtlL3siqTss5BEQiwg7IukREJcbiAgICBjb25zdCBmaXh0dXJlRGF0YWJhc2UgPSAoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpLmRhdGFiYXNlO1xuICAgIGNvbnN0IHJlYWxEYXRhYmFzZSA9IChTb25hbXUuZGJDb25maWcucHJvZHVjdGlvbl9tYXN0ZXIuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpXG4gICAgICAuZGF0YWJhc2U7XG5cbiAgICBjb25zdCBzZWxmUXVlcnkgPSBgSU5TRVJUIElHTk9SRSBJTlRPIFxcYCR7Zml4dHVyZURhdGFiYXNlfVxcYC5cXGAke2VudGl0eS50YWJsZX1cXGAgKFNFTEVDVCAqIEZST00gXFxgJHtyZWFsRGF0YWJhc2V9XFxgLlxcYCR7ZW50aXR5LnRhYmxlfVxcYCBXSEVSRSBcXGBpZFxcYCA9ICR7aWR9KWA7XG5cbiAgICBjb25zdCBhcmdzID0gT2JqZWN0LmVudHJpZXMoZW50aXR5LnJlbGF0aW9ucylcbiAgICAgIC5maWx0ZXIoXG4gICAgICAgIChbLCByZWxhdGlvbl0pID0+XG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8XG4gICAgICAgICAgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pICYmIHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UgPT09IHVuZGVmaW5lZCksXG4gICAgICApXG4gICAgICAubWFwKChbLCByZWxhdGlvbl0pID0+IHtcbiAgICAgICAgLypcbiAgICAgICAgQmVsb25nc1RvT25l7J24IOqyveyasFxuICAgICAgICAgIENhdGVnb3J5IC8gJ2lkJyAvIHJvd1tjYXRlZ29yeV9pZF0g7Zi47LacXG4gICAgICAgIE9uZVRvT25l7JeQIGpvaW5Db2x1bW4gPT09IHRydWUg7J24IOqyveyasFxuICAgICAgICAgIFByb2ZpbGUgLyAnaWQnIC8gcm93W3Byb2ZpbGVfaWRdIO2YuOy2nFxuICAgICAgICBPbmVUb09uZeyXkCBqb2luQ29sdW1uID09PSBmYWxzZSDsnbgg6rK97JqwXG4gICAgICAgICAgUHJvZmlsZSAvICdwcm9maWxlX2lkJyAvIHJvd1snaWQnXSDtmLjstpxcbiAgICAgICAgKi9cbiAgICAgICAgbGV0IGZpZWxkOiBzdHJpbmc7XG4gICAgICAgIGxldCBpZDogbnVtYmVyO1xuICAgICAgICBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChyZWxhdGlvbikgJiYgIXJlbGF0aW9uLmhhc0pvaW5Db2x1bW4pIHtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG4gICAgICAgICAgY29uc3QgcmVsYXRlZElkQ29sdW1uTmFtZSA9IHJlbGF0ZWRFbnRpdHkucHJvcHMuZmluZChcbiAgICAgICAgICAgIChwKSA9PiBpc1JlbGF0aW9uUHJvcChwKSAmJiBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICAgICAgICApPy5uYW1lO1xuICAgICAgICAgIGlmICghcmVsYXRlZElkQ29sdW1uTmFtZSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3JlbGF0ZWRFbnRpdHkuaWR97J2YICR7ZW50aXR5LmlkfSDqtIDqs4Qg7ZSE66Gt7J2EIOywvuydhCDsiJgg7JeG7Iq164uI64ukLmApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBmaWVsZCA9IGAke3JlbGF0ZWRJZENvbHVtbk5hbWV9X2lkYDtcbiAgICAgICAgICBpZCA9IHJvdy5pZDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBmaWVsZCA9IFwiaWRcIjtcbiAgICAgICAgICBpZCA9IHJvd1tgJHtyZWxhdGlvbi5uYW1lfV9pZGBdO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZW50aXR5SWQ6IHJlbGF0aW9uLndpdGgsXG4gICAgICAgICAgZmllbGQsXG4gICAgICAgICAgaWQsXG4gICAgICAgIH07XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoYXJnKSA9PiBhcmcuaWQgIT09IG51bGwpO1xuXG4gICAgY29uc3QgcmVsUXVlcmllcyA9IGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgYXJncy5tYXAoYXN5bmMgKGFyZ3MpID0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhhcmdzLmVudGl0eUlkLCBhcmdzLmZpZWxkLCBhcmdzLmlkKTtcbiAgICAgIH0pLFxuICAgICk7XG5cbiAgICByZXR1cm4gWy4uLnVuaXF1ZShyZWxRdWVyaWVzLnRvUmV2ZXJzZWQoKS5mbGF0KCkpLCBzZWxmUXVlcnldO1xuICB9XG5cbiAgYXN5bmMgZGVzdHJveSgpIHtcbiAgICBpZiAodGhpcy5fdGRiKSB7XG4gICAgICBhd2FpdCB0aGlzLl90ZGIuZGVzdHJveSgpO1xuICAgICAgdGhpcy5fdGRiID0gbnVsbDtcbiAgICB9XG4gICAgaWYgKHRoaXMuX2ZkYikge1xuICAgICAgYXdhaXQgdGhpcy5fZmRiLmRlc3Ryb3koKTtcbiAgICAgIHRoaXMuX2ZkYiA9IG51bGw7XG4gICAgfVxuICAgIGF3YWl0IEJhc2VNb2RlbC5kZXN0cm95KCk7XG4gIH1cblxuICBhc3luYyBnZXRGaXh0dXJlcyhcbiAgICBzb3VyY2VEQk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIHRhcmdldERCTmFtZToga2V5b2YgU29uYW11REJDb25maWcsXG4gICAgc2VhcmNoT3B0aW9uczogRml4dHVyZVNlYXJjaE9wdGlvbnMsXG4gICAgZHVwbGljYXRlQ2hlY2s/OiBEdXBsaWNhdGVDaGVja09wdGlvbnMsXG4gICkge1xuICAgIGNvbnN0IHNvdXJjZURCID0gY3JlYXRlS25leEluc3RhbmNlKFNvbmFtdS5kYkNvbmZpZ1tzb3VyY2VEQk5hbWVdKTtcbiAgICBjb25zdCB0YXJnZXREQiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWdbdGFyZ2V0REJOYW1lXSk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgeyBlbnRpdHlJZCwgZmllbGQsIHZhbHVlLCBzZWFyY2hUeXBlIH0gPSBzZWFyY2hPcHRpb25zO1xuXG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCk7XG4gICAgICBjb25zdCBjb2x1bW4gPVxuICAgICAgICBlbnRpdHkucHJvcHMuZmluZCgocHJvcCkgPT4gcHJvcC5uYW1lID09PSBmaWVsZCk/LnR5cGUgPT09IFwicmVsYXRpb25cIlxuICAgICAgICAgID8gYCR7ZmllbGR9X2lkYFxuICAgICAgICAgIDogZmllbGQ7XG5cbiAgICAgIGxldCBxdWVyeSA9IHNvdXJjZURCKGVudGl0eS50YWJsZSk7XG4gICAgICBpZiAoc2VhcmNoVHlwZSA9PT0gXCJlcXVhbHNcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgdmFsdWUpO1xuICAgICAgfSBlbHNlIGlmIChzZWFyY2hUeXBlID09PSBcImxpa2VcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgXCJsaWtlXCIsIGAlJHt2YWx1ZX0lYCk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHJvd3MgPSBhd2FpdCBxdWVyeTtcbiAgICAgIGlmIChyb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJObyByZWNvcmRzIGZvdW5kXCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmaXh0dXJlczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICAgIGNvbnN0IGluaXRpYWxSZWNvcmRzTGVuZ3RoID0gZml4dHVyZXMubGVuZ3RoO1xuICAgICAgICBjb25zdCBuZXdSZWNvcmRzID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgcm93LCB7XG4gICAgICAgICAgX2RiOiBzb3VyY2VEQixcbiAgICAgICAgfSk7XG4gICAgICAgIGZpeHR1cmVzLnB1c2goLi4ubmV3UmVjb3Jkcyk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRGaXh0dXJlUmVjb3JkID0gZml4dHVyZXMuZmluZCgocikgPT4gci5maXh0dXJlSWQgPT09IGAke2VudGl0eUlkfSMke3Jvdy5pZH1gKTtcblxuICAgICAgICBpZiAoY3VycmVudEZpeHR1cmVSZWNvcmQpIHtcbiAgICAgICAgICAvLyDtmITsnqwgZml4dHVyZeuhnOu2gO2EsCDsg53shLHrkJwgZmV0Y2hlZFJlY29yZHMg7ISk7KCVXG4gICAgICAgICAgY3VycmVudEZpeHR1cmVSZWNvcmQuZmV0Y2hlZFJlY29yZHMgPSBmaXh0dXJlc1xuICAgICAgICAgICAgLmZpbHRlcigocikgPT4gci5maXh0dXJlSWQgIT09IGN1cnJlbnRGaXh0dXJlUmVjb3JkLmZpeHR1cmVJZClcbiAgICAgICAgICAgIC5zbGljZShpbml0aWFsUmVjb3Jkc0xlbmd0aClcbiAgICAgICAgICAgIC5tYXAoKHIpID0+IHIuZml4dHVyZUlkKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZS5lbnRpdHlJZCk7XG5cbiAgICAgICAgLy8g7IKs7Jqp7J6QIOyngOyglSDsu6zrn7wg6riw7KSAIOykkeuztSDtmZXsnbgg4oaSIHRhcmdldFxuICAgICAgICBjb25zdCBjdXN0b21Db2x1bW5zID0gZHVwbGljYXRlQ2hlY2s/LmNvbHVtbnM/LltmaXh0dXJlLmVudGl0eUlkXTtcbiAgICAgICAgaWYgKGN1c3RvbUNvbHVtbnMgJiYgY3VzdG9tQ29sdW1ucy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgY3VzdG9tRHVwbGljYXRlUm93ID0gYXdhaXQgdGhpcy5jaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICAgICAgICAgIHRhcmdldERCLFxuICAgICAgICAgICAgZW50aXR5LFxuICAgICAgICAgICAgZml4dHVyZSxcbiAgICAgICAgICAgIGN1c3RvbUNvbHVtbnMsXG4gICAgICAgICAgKTtcbiAgICAgICAgICBpZiAoY3VzdG9tRHVwbGljYXRlUm93KSB7XG4gICAgICAgICAgICBjb25zdCBbcmVjb3JkXSA9IGF3YWl0IHRoaXMuY3JlYXRlRml4dHVyZVJlY29yZChlbnRpdHksIGN1c3RvbUR1cGxpY2F0ZVJvdywge1xuICAgICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICAgIF9kYjogdGFyZ2V0REIsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGZpeHR1cmUudGFyZ2V0ID0gcmVjb3JkO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVuaXF1ZSBpbmRleCDquLDspIAg7KSR67O1IO2ZleyduCDihpIgZml4dHVyZS51bmlxdWVcbiAgICAgICAgY29uc3QgdW5pcXVlUm93ID0gYXdhaXQgdGhpcy5jaGVja1VuaXF1ZVZpb2xhdGlvbih0YXJnZXREQiwgZW50aXR5LCBmaXh0dXJlKTtcbiAgICAgICAgaWYgKHVuaXF1ZVJvdykge1xuICAgICAgICAgIGNvbnN0IFtyZWNvcmRdID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgdW5pcXVlUm93LCB7XG4gICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICBfZGI6IHRhcmdldERCLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGZpeHR1cmUudW5pcXVlID0gcmVjb3JkO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB1bmlxdWUoZml4dHVyZXMsIChmKSA9PiBmLmZpeHR1cmVJZCk7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZChbdGFyZ2V0REIuZGVzdHJveSgpLCBzb3VyY2VEQi5kZXN0cm95KCldKTtcbiAgICB9XG4gIH1cblxuICBhc3luYyBjcmVhdGVGaXh0dXJlUmVjb3JkKFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIHJvdzoge1xuICAgICAgaWQ6IG51bWJlciB8IHN0cmluZztcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsO1xuICAgIH0sXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIHNpbmdsZVJlY29yZD86IGJvb2xlYW47XG4gICAgICBfZGI/OiBLbmV4O1xuICAgIH0sXG4gICk6IFByb21pc2U8Rml4dHVyZVJlY29yZFtdPiB7XG4gICAgY29uc3QgcmVjb3JkczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgY29uc3QgdmlzaXRlZEVudGl0aWVzID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICBjb25zdCBjcmVhdGUgPSBhc3luYyAoXG4gICAgICBlbnRpdHk6IEVudGl0eSxcbiAgICAgIHJvdzoge1xuICAgICAgICBpZDogbnVtYmVyIHwgc3RyaW5nO1xuICAgICAgICBba2V5OiBzdHJpbmddOiBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgbnVsbDtcbiAgICAgIH0sXG4gICAgKSA9PiB7XG4gICAgICBjb25zdCBmaXh0dXJlSWQgPSBgJHtlbnRpdHkuaWR9IyR7cm93LmlkfWA7XG4gICAgICBpZiAodmlzaXRlZEVudGl0aWVzLmhhcyhmaXh0dXJlSWQpKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIHZpc2l0ZWRFbnRpdGllcy5hZGQoZml4dHVyZUlkKTtcblxuICAgICAgY29uc3QgcmVjb3JkOiBGaXh0dXJlUmVjb3JkID0ge1xuICAgICAgICBmaXh0dXJlSWQsXG4gICAgICAgIGVudGl0eUlkOiBlbnRpdHkuaWQsXG4gICAgICAgIGlkOiByb3cuaWQsXG4gICAgICAgIGNvbHVtbnM6IHt9LFxuICAgICAgICBmZXRjaGVkUmVjb3JkczogW10sXG4gICAgICAgIGJlbG9uZ3NSZWNvcmRzOiBbXSxcbiAgICAgIH07XG5cbiAgICAgIGZvciAoY29uc3QgcHJvcCBvZiBlbnRpdHkucHJvcHMpIHtcbiAgICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0gPSB7XG4gICAgICAgICAgcHJvcDogcHJvcCxcbiAgICAgICAgICB2YWx1ZTogcm93W3Byb3AubmFtZV0sXG4gICAgICAgIH07XG5cbiAgICAgICAgY29uc3QgZGIgPSBvcHRpb25zPy5fZGIgPz8gQmFzZU1vZGVsLmdldERCKFwid1wiKTtcbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHRocm91Z2hUYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IGZyb21Db2x1bW4gPSBgJHtpbmZsZWN0aW9uLnNpbmd1bGFyaXplKGVudGl0eS50YWJsZSl9X2lkYDtcbiAgICAgICAgICBjb25zdCB0b0NvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYih0aHJvdWdoVGFibGUpLndoZXJlKGZyb21Db2x1bW4sIHJvdy5pZCkucGx1Y2sodG9Db2x1bW4pO1xuICAgICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0udmFsdWUgPSByZWxhdGVkSWRzO1xuICAgICAgICB9IGVsc2UgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKVxuICAgICAgICAgICAgLndoZXJlKHByb3Auam9pbkNvbHVtbiwgcm93LmlkKVxuICAgICAgICAgICAgLnBsdWNrKFwiaWRcIik7XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZHM7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiAhcHJvcC5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgLy8g7Jet67Cp7ZalIE9uZVRvT25lOiBGS+ulvCDqsIDsp4Qg6rSA66CoIOyXlO2LsO2LsOulvCDssL7sirXri4jri6RcbiAgICAgICAgICAvLyDsmIjsi5w6IFVzZXIgT25lVG9PbmUgRW1wbG95ZWUgKEVtcGxveWVl6rCAIHVzZXJfaWQgRkvrpbwg6rCA7KeQKVxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRQcm9wID0gcmVsYXRlZEVudGl0eS5wcm9wcy5maW5kKFxuICAgICAgICAgICAgKHApID0+IGlzUmVsYXRpb25Qcm9wKHApICYmIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICAgICAgICk7XG4gICAgICAgICAgaWYgKHJlbGF0ZWRQcm9wICYmIGlzUmVsYXRpb25Qcm9wKHJlbGF0ZWRQcm9wKSkge1xuICAgICAgICAgICAgLy8g6rSA66CoIOyXlO2LsO2LsOyXkOyEnCBGSyDsu6zrn7zsnLzroZwg7L+866as7ZWp64uI64ukIChpZOqwgCDslYTri5gpXG4gICAgICAgICAgICBjb25zdCBma0NvbHVtbiA9IGAke3JlbGF0ZWRQcm9wLm5hbWV9X2lkYDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSb3cgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKS53aGVyZShma0NvbHVtbiwgcm93LmlkKS5maXJzdCgpO1xuICAgICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRSb3c/LmlkO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IHJvd1tgJHtwcm9wLm5hbWV9X2lkYF07XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICBpZiAocmVsYXRlZElkKSB7XG4gICAgICAgICAgICByZWNvcmQuYmVsb25nc1JlY29yZHMucHVzaChgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIW9wdGlvbnM/LnNpbmdsZVJlY29yZCAmJiByZWxhdGVkSWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJvdyA9IGF3YWl0IGRiKHJlbGF0ZWRFbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgcmVsYXRlZElkKS5maXJzdCgpO1xuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSb3cpIHtcbiAgICAgICAgICAgICAgYXdhaXQgY3JlYXRlKHJlbGF0ZWRFbnRpdHksIHJlbGF0ZWRSb3cpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZWNvcmRzLnB1c2gocmVjb3JkKTtcbiAgICB9O1xuXG4gICAgYXdhaXQgY3JlYXRlKGVudGl0eSwgcm93KTtcblxuICAgIHJldHVybiByZWNvcmRzO1xuICB9XG5cbiAgLyoqXG4gICAqIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrAgKHNlbGYtcmVmZXJlbmNlIO2PrO2VqClcbiAgICogMi4g7YWM7J2067iU67OEIOugiOuyqOuzhOuhnCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnSDrsI8gdXBzZXJ0IOyLpO2WiVxuICAgKiAzLiDsiJzshJwg6riw67CYIHV1aWTihpJpZCDrp6TtlZEgKFVwc2VydEJ1aWxkZXLqsIAgdXVpZOulvCBEQuyXkCDsoIDsnqXtlZjsp4Ag7JWK7Jy866+A66GcKVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy64qUIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0XG4gICAqIEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWY7JesIOqwgSB1cHNlcnQg7Zi47Lac7JeQ7ISc64qUXG4gICAqIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuPhOuhnSDtlanri4jri6QuXG4gICAqL1xuICBhc3luYyBpbnNlcnRGaXh0dXJlcyhcbiAgICBkYk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIF9maXh0dXJlczogRml4dHVyZVJlY29yZFtdLFxuICApOiBQcm9taXNlPEZpeHR1cmVJbXBvcnRSZXN1bHRbXT4ge1xuICAgIGNvbnN0IGZpeHR1cmVzID0gdW5pcXVlKF9maXh0dXJlcywgKGYpID0+IGYuZml4dHVyZUlkKTtcblxuICAgIC8vIOy0iOq4sO2ZlFxuICAgIHRoaXMuYnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gICAgdGhpcy5maXh0dXJlUmVmTWFwID0gbmV3IE1hcCgpO1xuICAgIHRoaXMuc2tpcHBlZEZpeHR1cmVzID0gbmV3IE1hcCgpO1xuXG4gICAgLy8g67OR66CsIO2FjOyKpO2KuCDrqqjrk5zsl5DshJzripQgd29ya2Vy67OEIERC7JeQIOyggOyepVxuICAgIGNvbnN0IGRiQ29uZmlnID1cbiAgICAgIHByb2Nlc3MuZW52LlNPTkFNVV9XT1JLRVJfREIgPT09IFwidHJ1ZVwiICYmIHByb2Nlc3MuZW52LlZJVEVTVF9QT09MX0lEXG4gICAgICAgID8gKCgpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHdvcmtlcklkID0gcGFyc2VJbnQocHJvY2Vzcy5lbnYuVklURVNUX1BPT0xfSUQgPz8gXCIxXCIsIDEwKTtcbiAgICAgICAgICAgIGNvbnN0IGJhc2VDb25maWcgPSBTb25hbXUuZGJDb25maWdbZGJOYW1lXTtcbiAgICAgICAgICAgIGNvbnN0IGNvbm5lY3Rpb24gPSBiYXNlQ29uZmlnLmNvbm5lY3Rpb24gYXMgeyBkYXRhYmFzZTogc3RyaW5nIH07XG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAuLi5iYXNlQ29uZmlnLFxuICAgICAgICAgICAgICBjb25uZWN0aW9uOiB7IC4uLmNvbm5lY3Rpb24sIGRhdGFiYXNlOiBgJHtjb25uZWN0aW9uLmRhdGFiYXNlfV8ke3dvcmtlcklkfWAgfSxcbiAgICAgICAgICAgICAgcG9vbDogeyBtaW46IDEsIG1heDogMSB9LFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9KSgpXG4gICAgICAgIDogU29uYW11LmRiQ29uZmlnW2RiTmFtZV07XG4gICAgY29uc3QgZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoZGJDb25maWcpO1xuICAgIGNvbnN0IHJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSA9IFtdO1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrBcbiAgICAgIHRoaXMucmVsYXRpb25HcmFwaC5idWlsZEdyYXBoKGZpeHR1cmVzKTtcbiAgICAgIGNvbnN0IGluc2VydGlvbk9yZGVyID0gdGhpcy5yZWxhdGlvbkdyYXBoLmdldEluc2VydGlvbk9yZGVyKCk7XG5cbiAgICAgIC8vIDIuIOyKpO2Cte2VoCBmaXh0dXJlIOuovOyggCDsspjrpqwgKG92ZXJyaWRlIOyytO2BrClcbiAgICAgIGZvciAoY29uc3QgZml4dHVyZUlkIG9mIGluc2VydGlvbk9yZGVyKSB7XG4gICAgICAgIGNvbnN0IGZpeHR1cmUgPSBmaXh0dXJlcy5maW5kKChmKSA9PiBmLmZpeHR1cmVJZCA9PT0gZml4dHVyZUlkKTtcbiAgICAgICAgaWYgKCFmaXh0dXJlKSBjb250aW51ZTtcblxuICAgICAgICBjb25zdCBoYXNUYXJnZXQgPSAhIWZpeHR1cmUudGFyZ2V0O1xuICAgICAgICBjb25zdCBoYXNVbmlxdWUgPSAhIWZpeHR1cmUudW5pcXVlO1xuICAgICAgICBjb25zdCBoYXNEdXBsaWNhdGUgPSBoYXNUYXJnZXQgfHwgaGFzVW5pcXVlO1xuXG4gICAgICAgIC8vIOykkeuzteydtCDsnojqs6Agb3ZlcnJpZGU9ZmFsc2Xsnbgg6rK97JqwOiDsiqTtgrVcbiAgICAgICAgaWYgKGhhc0R1cGxpY2F0ZSAmJiAhZml4dHVyZS5vdmVycmlkZSkge1xuICAgICAgICAgIC8vIOq4sOyhtCDroIjsvZTrk5wgSUQg7KCA7J6lICh1bmlxdWUg7Jqw7ISgLCDsl4bsnLzrqbQgdGFyZ2V0KVxuICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBmaXh0dXJlLnVuaXF1ZT8uaWQgPz8gZml4dHVyZS50YXJnZXQ/LmlkO1xuICAgICAgICAgIGFzc2VydChleGlzdGluZ0lkKTtcbiAgICAgICAgICB0aGlzLnNraXBwZWRGaXh0dXJlcy5zZXQoZml4dHVyZUlkLCB7XG4gICAgICAgICAgICBlbnRpdHlJZDogZml4dHVyZS5lbnRpdHlJZCxcbiAgICAgICAgICAgIGV4aXN0aW5nSWQsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICBjaGFsay55ZWxsb3coXG4gICAgICAgICAgICAgICAgYFNraXBwZWQgJHtmaXh0dXJlLmVudGl0eUlkfSMke2ZpeHR1cmUuaWR9IChleGlzdGluZzogIyR7ZXhpc3RpbmdJZH0sIG92ZXJyaWRlOiBmYWxzZSlgLFxuICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyAzLiDthYzsnbTruJTrs4QgZml4dHVyZSDqt7jro7ntmZQgKGluc2VydGlvbk9yZGVyIOyInOyEnCDquLDrsJgpXG4gICAgICBjb25zdCBmaXh0dXJlc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgRml4dHVyZVJlY29yZFtdPigpO1xuICAgICAgY29uc3QgdGFibGVPcmRlcjogc3RyaW5nW10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlSWQgb2YgaW5zZXJ0aW9uT3JkZXIpIHtcbiAgICAgICAgLy8g7Iqk7YK165CcIGZpeHR1cmUg7KCc7Jm4XG4gICAgICAgIGlmICh0aGlzLnNraXBwZWRGaXh0dXJlcy5oYXMoZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgY29uc3QgZml4dHVyZSA9IGZpeHR1cmVzLmZpbmQoKGYpID0+IGYuZml4dHVyZUlkID09PSBmaXh0dXJlSWQpO1xuICAgICAgICBpZiAoIWZpeHR1cmUpIGNvbnRpbnVlO1xuXG4gICAgICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGZpeHR1cmUuZW50aXR5SWQpO1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGU7XG5cbiAgICAgICAgaWYgKCFmaXh0dXJlc0J5VGFibGUuaGFzKHRhYmxlTmFtZSkpIHtcbiAgICAgICAgICBmaXh0dXJlc0J5VGFibGUuc2V0KHRhYmxlTmFtZSwgW10pO1xuICAgICAgICAgIHRhYmxlT3JkZXIucHVzaCh0YWJsZU5hbWUpO1xuICAgICAgICB9XG4gICAgICAgIGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKT8ucHVzaChmaXh0dXJlKTtcbiAgICAgIH1cblxuICAgICAgYXdhaXQgZGIudHJhbnNhY3Rpb24oYXN5bmMgKHRyeCkgPT4ge1xuICAgICAgICBjb25zdCBpbnNlcnRlZElkc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4oKTtcblxuICAgICAgICAvLyA0LiDthYzsnbTruJTrs4Qg66CI67Ko67OEIOyymOumrFxuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgY29uc3QgdGFibGVGaXh0dXJlcyA9IGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBbXTtcbiAgICAgICAgICBjb25zdCBsZXZlbHMgPSB0aGlzLmdyb3VwRml4dHVyZXNCeUxldmVsKHRhYmxlRml4dHVyZXMpO1xuXG4gICAgICAgICAgZm9yIChjb25zdCBsZXZlbEZpeHR1cmVzIG9mIGxldmVscykge1xuICAgICAgICAgICAgLy8g7ZW064u5IOugiOuyqOydmCBmaXh0dXJl65OkIHJlZ2lzdGVyXG4gICAgICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgbGV2ZWxGaXh0dXJlcykge1xuICAgICAgICAgICAgICB0aGlzLnJlZ2lzdGVyRml4dHVyZShmaXh0dXJlLCBpbnNlcnRlZElkc0J5VGFibGUpO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICAgIGBSZWdpc3RlcmVkICR7Zml4dHVyZS5lbnRpdHlJZH0jJHtmaXh0dXJlLmlkfSR7Zml4dHVyZS5vdmVycmlkZSA/IGAgKG92ZXJyaWRlKWAgOiBcIlwifWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIHVwc2VydCDsi6Ttlokg7KCEIHV1aWQg66qp66GdIOyggOyepVxuICAgICAgICAgICAgY29uc3QgdGFibGUgPSB0aGlzLmJ1aWxkZXIuZ2V0VGFibGUodGFibGVOYW1lKTtcbiAgICAgICAgICAgIGNvbnN0IHV1aWRzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4gcm93LnV1aWQgYXMgc3RyaW5nKTtcblxuICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICBgVXBzZXJ0aW5nICR7dGFibGVOYW1lfSB3aXRoICR7dXVpZHMubGVuZ3RofSByb3dzIChsZXZlbCAke2xldmVscy5pbmRleE9mKGxldmVsRml4dHVyZXMpICsgMX0vJHtsZXZlbHMubGVuZ3RofSlgLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICBjb25zdCBpZHMgPSAoYXdhaXQgdGhpcy5idWlsZGVyLnVwc2VydChcbiAgICAgICAgICAgICAgdHJ4LFxuICAgICAgICAgICAgICB0YWJsZU5hbWUgYXMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gICAgICAgICAgICApKSBhcyAobnVtYmVyIHwgc3RyaW5nKVtdO1xuXG4gICAgICAgICAgICAvLyDsiJzshJwg6riw67CYIHV1aWQgLT4gaWQg66ek7ZWRXG4gICAgICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDsl4bsnLzrr4DroZwg65Ox66GdIOyInOyEnCA9IOuwmO2ZmCDsiJzshJwg67O07J6lXG4gICAgICAgICAgICBpZiAodXVpZHMubGVuZ3RoID4gMCAmJiB1dWlkcy5sZW5ndGggPT09IGlkcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPVxuICAgICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHV1aWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICAgICAgZXhpc3RpbmdNYXAuc2V0KHV1aWRzW2ldLCBpZHNbaV0pO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5zZXQodGFibGVOYW1lLCBleGlzdGluZ01hcCk7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKHV1aWRzLmxlbmd0aCAhPT0gaWRzLmxlbmd0aCkge1xuICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgY2hhbGsueWVsbG93KFxuICAgICAgICAgICAgICAgICAgYFdhcm5pbmc6IHV1aWQgY291bnQgKCR7dXVpZHMubGVuZ3RofSkgIT0gaWQgY291bnQgKCR7aWRzLmxlbmd0aH0pIGZvciAke3RhYmxlTmFtZX1gLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNS4gTWFueVRvTWFueSDqtIDqs4Qg7LKY66asXG4gICAgICAgIGF3YWl0IHRoaXMucHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnModHJ4LCBmaXh0dXJlcywgaW5zZXJ0ZWRJZHNCeVRhYmxlKTtcblxuICAgICAgICAvLyA2LiBQb3N0Z3JlU1FMIOyLnO2AgOyKpCDrpqzshYtcbiAgICAgICAgLy8gRml4dHVyZSDsgr3snoUg7ZuEIOqwgSDthYzsnbTruJTsnZggSUQg7Iuc7YCA7Iqk66W8IOy1nOuMgCBJRCDqsJLsnLzroZwg7JeF642w7J207Yq47ZWp64uI64ukLlxuICAgICAgICAvLyDsnbTroIfqsowg7ZWY7KeAIOyViuycvOuptCDri6TsnYwgSU5TRVJUIOyLnCBJROqwgCAyMDAw67KI64yA66GcIOyDneyEseuQoCDsiJgg7J6I7Iq164uI64ukLlxuICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJlc2V0dGluZyBzZXF1ZW5jZXMuLi5cIikpO1xuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIEVudGl0eeulvCDssL7slYTshJwgaWQg7YOA7J6FIO2ZleyduFxuICAgICAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXRBbGxFbnRpdGllcygpLmZpbmQoXG4gICAgICAgICAgICAgIChlKSA9PiBlLnRhYmxlID09PSB0YWJsZU5hbWUgfHwgZS5pZC50b0xvd2VyQ2FzZSgpID09PSB0YWJsZU5hbWUsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICBpZiAoZW50aXR5KSB7XG4gICAgICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik7XG4gICAgICAgICAgICAgIGNvbnN0IGlkVHlwZSA9IGlkUHJvcD8udHlwZTtcblxuICAgICAgICAgICAgICAvLyBpbnRlZ2VyL2JpZ0ludGVnZXLsnbTqsbDrgpgsIHN0cmluZ+ydtOyngOunjCBmaXh0dXJlU3RyYXRlZ3k9c2VxdWVuY2Xsnbgg6rK97Jqw7JeQ66eMIOumrOyFi+2VqeuLiOuLpFxuICAgICAgICAgICAgICBjb25zdCB1c2VzU2VxdWVuY2UgPVxuICAgICAgICAgICAgICAgIGlkVHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgICAgICBpZFR5cGUgPT09IFwiYmlnSW50ZWdlclwiIHx8XG4gICAgICAgICAgICAgICAgaWRQcm9wPy5jb25lPy5maXh0dXJlU3RyYXRlZ3kgPT09IFwic2VxdWVuY2VcIjtcblxuICAgICAgICAgICAgICBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgICAgICAgICFpc1Rlc3QoKSAmJlxuICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICAgIGNoYWxrLmdyYXkoXG4gICAgICAgICAgICAgICAgICAgICAgYFNraXBwZWQgc2VxdWVuY2UgcmVzZXQgZm9yICR7dGFibGVOYW1lfSAoaWQgdHlwZTogJHtpZFR5cGUgfHwgXCJ1bmtub3duXCJ9KWAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIO2FjOydtOu4lOydmCDstZzrjIAgSUQg7KGw7ZqMIChzdHJpbmcg7YOA7J6F7J2AIOyIq+yekCDsupDsiqTtjIUg7ZWE7JqUKVxuICAgICAgICAgICAgY29uc3QgZW50aXR5MiA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKS5maW5kKFxuICAgICAgICAgICAgICAoZSkgPT4gZS50YWJsZSA9PT0gdGFibGVOYW1lIHx8IGUuaWQudG9Mb3dlckNhc2UoKSA9PT0gdGFibGVOYW1lLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIGNvbnN0IGlkVHlwZTIgPSBlbnRpdHkyPy5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik/LnR5cGU7XG4gICAgICAgICAgICBjb25zdCBtYXhJZFJlc3VsdCA9XG4gICAgICAgICAgICAgIGlkVHlwZTIgPT09IFwic3RyaW5nXCJcbiAgICAgICAgICAgICAgICA/IGF3YWl0IHRyeFxuICAgICAgICAgICAgICAgICAgICAucmF3KGBTRUxFQ1QgTUFYKGlkOjpiaWdpbnQpIGFzIG1heF9pZCBGUk9NIFwiJHt0YWJsZU5hbWV9XCJgKVxuICAgICAgICAgICAgICAgICAgICAudGhlbigocikgPT4gci5yb3dzWzBdKVxuICAgICAgICAgICAgICAgIDogYXdhaXQgdHJ4KHRhYmxlTmFtZSkubWF4KFwiaWQgYXMgbWF4X2lkXCIpLmZpcnN0KCk7XG4gICAgICAgICAgICBjb25zdCBtYXhJZCA9IG1heElkUmVzdWx0Py5tYXhfaWQ7XG5cbiAgICAgICAgICAgIGlmIChtYXhJZCAhPT0gbnVsbCAmJiBtYXhJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIOyLnO2AgOyKpOuqheydhCBwZ19nZXRfc2VyaWFsX3NlcXVlbmNl66GcIOyViOyghO2VmOqyjCDsobDtmoxcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4LnJhdyhgU0VMRUNUIHNldHZhbChwZ19nZXRfc2VyaWFsX3NlcXVlbmNlKD8sICdpZCcpLCA/KWAsIFtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgbWF4SWQsXG4gICAgICAgICAgICAgIF0pO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuZ3JlZW4oYFJlc2V0IHNlcXVlbmNlIGZvciAke3RhYmxlTmFtZX06ICR7bWF4SWR9YCkpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKF9lcnIpIHtcbiAgICAgICAgICAgIC8vIOyLnO2AgOyKpOqwgCDsl4bripQg7YWM7J2067iUKGpvaW4gdGFibGUg65OxKeydgCDrrLTsi5xcbiAgICAgICAgICAgICFpc1Rlc3QoKSAmJiBjb25zb2xlLmxvZyhjaGFsay5ncmF5KGBTa2lwcGVkIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX1gKSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNy4g6rKw6rO8IOyImOynkVxuICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcblxuICAgICAgICAgIC8vIOyKpO2CteuQnCBmaXh0dXJl64qUIOq4sOyhtCDroIjsvZTrk5wg7KCV67O066GcIOqysOqzvCDstpTqsIBcbiAgICAgICAgICBjb25zdCBza2lwcGVkID0gdGhpcy5za2lwcGVkRml4dHVyZXMuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAoc2tpcHBlZCkge1xuICAgICAgICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgICAgICAgZW50aXR5SWQ6IGZpeHR1cmUuZW50aXR5SWQsXG4gICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgc2tpcHBlZC5leGlzdGluZ0lkKS5maXJzdCgpLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCByZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAocmVmKSB7XG4gICAgICAgICAgICBjb25zdCB1dWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQoZW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgIGNvbnN0IGluc2VydGVkSWQgPSB1dWlkVG9JZD8uZ2V0KHJlZi51dWlkKTtcblxuICAgICAgICAgICAgaWYgKGluc2VydGVkSWQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICByZXN1bHRzLnB1c2goe1xuICAgICAgICAgICAgICAgIGVudGl0eUlkOiBmaXh0dXJlLmVudGl0eUlkLFxuICAgICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgaW5zZXJ0ZWRJZCkuZmlyc3QoKSxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihgSW5zZXJ0ZWQgaW50byAke2VudGl0eS50YWJsZX06ICMke2ZpeHR1cmUuaWR9IC0+ICMke2luc2VydGVkSWR9YCksXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCBkYi5kZXN0cm95KCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuaXF1ZShyZXN1bHRzLCAocikgPT4gYCR7ci5lbnRpdHlJZH0jJHtyLmRhdGEuaWR9YCk7XG4gIH1cblxuICAvKipcbiAgICogRml4dHVyZVJlY29yZOulvCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnVxuICAgKiBAcGFyYW0gaW5zZXJ0ZWRJZHNCeVRhYmxlIOydtOuvuCB1cHNlcnTrkJwg7YWM7J2067iU7J2YIHV1aWTihpJpZCDrp6TtlZEgKOugiOuyqOuzhCDsspjrpqwg7IucIOyCrOyaqSlcbiAgICovXG4gIHByaXZhdGUgcmVnaXN0ZXJGaXh0dXJlKFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlPzogTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4sXG4gICk6IFVCUmVmIHtcbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICBjb25zdCByb3c6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICAvLyBPdmVycmlkZSDrqqjrk5wg7YyQ64uoOiB0YXJnZXQg65iQ64qUIHVuaXF1ZeqwgCDsnojqs6Agb3ZlcnJpZGU9dHJ1ZeyduCDqsr3smrBcbiAgICBjb25zdCBleGlzdGluZ1JlY29yZCA9IGZpeHR1cmUudGFyZ2V0ID8/IGZpeHR1cmUudW5pcXVlO1xuICAgIGNvbnN0IGlzT3ZlcnJpZGVNb2RlID0gZml4dHVyZS5vdmVycmlkZSAmJiBleGlzdGluZ1JlY29yZDtcblxuICAgIGZvciAoY29uc3QgW3Byb3BOYW1lLCBjb2x1bW5dIG9mIE9iamVjdC5lbnRyaWVzKGZpeHR1cmUuY29sdW1ucykpIHtcbiAgICAgIGNvbnN0IHByb3AgPSBjb2x1bW4ucHJvcDtcblxuICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIEdlbmVyYXRlZCBjb2x1bW7snYAgSU5TRVJU7JeQ7IScIOygnOyZuCAoRELqsIAg7J6Q64+ZIOyDneyEsSlcbiAgICAgIGlmIChcImdlbmVyYXRlZFwiIGluIHByb3AgJiYgcHJvcC5nZW5lcmF0ZWQpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIGlkIOyymOumrFxuICAgICAgaWYgKHByb3BOYW1lID09PSBcImlkXCIpIHtcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgLy8gcGFyZW50SWQg7JeU7Yuw7Yuw7J2YIGlk64qUIOu2gOuqqCDsl5Tti7Dti7DsnZggRkvsnbTrr4DroZwg7Iuc7YCA7IqkIOuvuOyCrOyaqSAo66qF7Iuc7KCB7Jy866GcIO2PrO2VqClcbiAgICAgICAgY29uc3QgdXNlc1NlcXVlbmNlID1cbiAgICAgICAgICAhZW50aXR5LnBhcmVudElkICYmXG4gICAgICAgICAgKGlkUHJvcD8udHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8udHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8uY29uZT8uZml4dHVyZVN0cmF0ZWd5ID09PSBcInNlcXVlbmNlXCIpO1xuXG4gICAgICAgIGlmIChpc092ZXJyaWRlTW9kZSAmJiBleGlzdGluZ1JlY29yZCkge1xuICAgICAgICAgIC8vIE92ZXJyaWRlOiDquLDsobQg66CI7L2U65Oc7J2YIOqwkiDsgqzsmqkg4oaSIFVQREFURVxuICAgICAgICAgIHJvd1twcm9wTmFtZV0gPSBleGlzdGluZ1JlY29yZC5jb2x1bW5zW3Byb3BOYW1lXT8udmFsdWU7XG4gICAgICAgIH0gZWxzZSBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgIC8vIHN0cmluZyBQSyDrmJDripQgcGFyZW50SWQg7JeU7Yuw7YuwIEZLOiDsg53shLHrkJwgaWQg6rCS7J2EIElOU0VSVOyXkCDtj6ztlahcbiAgICAgICAgICByb3dbcHJvcE5hbWVdID0gY29sdW1uLnZhbHVlO1xuICAgICAgICB9XG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlciBQSzogREIg7Iuc7YCA7Iqk7JeQIOunoeq5gCAo6rCSIOygnOyZuClcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHxcbiAgICAgICAgICAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiBwcm9wLmhhc0pvaW5Db2x1bW4pXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IGNvbHVtbi52YWx1ZSBhcyBudW1iZXIgfCBudWxsO1xuICAgICAgICAgIGlmIChyZWxhdGVkSWQgIT09IG51bGwgJiYgcmVsYXRlZElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG5cbiAgICAgICAgICAgIC8vIOuovOyggCBza2lw65CcIGZpeHR1cmXsnbjsp4Ag7ZmV7J24XG4gICAgICAgICAgICBjb25zdCBza2lwcGVkRXhpc3RpbmdJZCA9IHRoaXMuc2tpcHBlZEZpeHR1cmVzLmdldChyZWxhdGVkRml4dHVyZUlkKT8uZXhpc3RpbmdJZDtcbiAgICAgICAgICAgIGlmIChza2lwcGVkRXhpc3RpbmdJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIHNraXDrkJwgZml4dHVyZSDihpIgdGFyZ2V0IERC7J2YIOq4sOyhtCDroIjsvZTrk5wgaWQg7IKs7JqpXG4gICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHNraXBwZWRFeGlzdGluZ0lkO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQocmVsYXRlZEZpeHR1cmVJZCk7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgICAgLy8g7J2066+4IHVwc2VydOuQnCDqsJnsnYAg7YWM7J2067iUIGZpeHR1cmUg7ZmV7J24XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEluc2VydGVkSWRzID0gaW5zZXJ0ZWRJZHNCeVRhYmxlPy5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgICAgY29uc3QgYWN0dWFsSWQgPSByZWxhdGVkSW5zZXJ0ZWRJZHM/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGFjdHVhbElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIC8vIOydtOuvuCB1cHNlcnTrkKgg4oaSIOyLpOygnCBJRCDsgqzsmqlcbiAgICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IGFjdHVhbElkO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAvLyDslYTsp4EgdXBzZXJ0IOyViOuQqCDihpIgVUJSZWYg7IKs7JqpXG4gICAgICAgICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSByZWxhdGVkUmVmO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAvLyBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOugiOy9lOuTnCDihpIgSUQg6re464yA66GcIOyCrOyaqVxuICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAvLyBIYXNNYW55LCBNYW55VG9NYW5564qUIOuzhOuPhCDsspjrpqxcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIOydvOuwmCDsu6zrn7xcbiAgICAgICAgcm93W3Byb3BOYW1lXSA9IHRoaXMuY29udmVydENvbHVtblZhbHVlKHByb3AgYXMgRW50aXR5UHJvcCwgY29sdW1uLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAhaXNUZXN0KCkgJiZcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoYFJlZ2lzdGVyaW5nICR7ZW50aXR5LnRhYmxlfSAtICR7aW5zcGVjdChyb3csIGZhbHNlLCBudWxsLCB0cnVlKX1gKSk7XG4gICAgY29uc3QgcmVmID0gdGhpcy5idWlsZGVyLnJlZ2lzdGVyKGVudGl0eS50YWJsZSwgcm93KTtcbiAgICB0aGlzLmZpeHR1cmVSZWZNYXAuc2V0KGZpeHR1cmUuZml4dHVyZUlkLCByZWYpO1xuXG4gICAgcmV0dXJuIHJlZjtcbiAgfVxuXG4gIC8qKlxuICAgKiDsu6zrn7wg6rCSIOuzgO2ZmFxuICAgKi9cbiAgcHJpdmF0ZSBjb252ZXJ0Q29sdW1uVmFsdWUocHJvcDogRW50aXR5UHJvcCwgdmFsdWU6IHVua25vd24pOiB1bmtub3duIHtcbiAgICBpZiAodmFsdWUgPT09IG51bGwgfHwgdmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgc3dpdGNoIChwcm9wLnR5cGUpIHtcbiAgICAgIGNhc2UgXCJqc29uXCI6XG4gICAgICAgIC8vIFVwc2VydEJ1aWxkZXIucmVnaXN0ZXLsl5DshJwgSlNPTi5zdHJpbmdpZnkg7LKY66as7ZWY66+A66GcIG9iamVjdCDqt7jrjIDroZwg7KCE64usXG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgY2FzZSBcImRhdGVcIjpcbiAgICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlb2YgdmFsdWUgPT09IFwibnVtYmVyXCIpIHtcbiAgICAgICAgICByZXR1cm4gbmV3IERhdGUodmFsdWUpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgcHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnMoXG4gICAgdHJ4OiBLbmV4LlRyYW5zYWN0aW9uLFxuICAgIGZpeHR1cmVzOiBGaXh0dXJlUmVjb3JkW10sXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlOiBNYXA8c3RyaW5nLCBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+PixcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGZpeHR1cmVzKSB7XG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICAgIGNvbnN0IHNvdXJjZVJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQoZml4dHVyZS5maXh0dXJlSWQpO1xuXG4gICAgICBpZiAoIXNvdXJjZVJlZikgY29udGludWU7XG5cbiAgICAgIGNvbnN0IHNvdXJjZVV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChlbnRpdHkudGFibGUpO1xuICAgICAgY29uc3Qgc291cmNlSWQgPSBzb3VyY2VVdWlkVG9JZD8uZ2V0KHNvdXJjZVJlZi51dWlkKTtcblxuICAgICAgaWYgKHNvdXJjZUlkID09PSB1bmRlZmluZWQpIGNvbnRpbnVlO1xuXG4gICAgICBmb3IgKGNvbnN0IFssIGNvbHVtbl0gb2YgT2JqZWN0LmVudHJpZXMoZml4dHVyZS5jb2x1bW5zKSkge1xuICAgICAgICBjb25zdCBwcm9wID0gY29sdW1uLnByb3A7XG5cbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSAmJiBBcnJheS5pc0FycmF5KGNvbHVtbi52YWx1ZSkpIHtcbiAgICAgICAgICAvLyDshKDtg53rkJjsp4Ag7JWK7J2AIE1hbnlUb01hbnkg6rSA6rOE64qUIOyggOyepe2VmOyngCDslYrsnYxcbiAgICAgICAgICBjb25zdCB0YXJnZXRUYWJsZSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgaWYgKCF0aGlzLmJ1aWxkZXIuaGFzVGFibGUodGFyZ2V0VGFibGUudGFibGUpKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBjb2x1bW4udmFsdWUgYXMgbnVtYmVyW107XG4gICAgICAgICAgaWYgKHJlbGF0ZWRJZHMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IGpvaW5UYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuXG4gICAgICAgICAgY29uc3Qgc291cmNlQ29sdW1uID0gYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShlbnRpdHkudGFibGUpfV9pZGA7XG4gICAgICAgICAgY29uc3QgdGFyZ2V0Q29sdW1uID0gYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShyZWxhdGVkRW50aXR5LnRhYmxlKX1faWRgO1xuXG4gICAgICAgICAgZm9yIChjb25zdCByZWxhdGVkSWQgb2YgcmVsYXRlZElkcykge1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZEZpeHR1cmVJZCA9IGAke3Byb3Aud2l0aH0jJHtyZWxhdGVkSWR9YDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KHJlbGF0ZWRGaXh0dXJlSWQpO1xuXG4gICAgICAgICAgICBsZXQgdGFyZ2V0SWQ6IG51bWJlciB8IHN0cmluZztcblxuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSZWYpIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChyZWxhdGVkRW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgICAgY29uc3QgcmVzb2x2ZWRJZCA9IHJlbGF0ZWRVdWlkVG9JZD8uZ2V0KHJlbGF0ZWRSZWYudXVpZCk7XG5cbiAgICAgICAgICAgICAgaWYgKHJlc29sdmVkSWQgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgICAgICAgICAgIGBSZWxhdGVkIGZpeHR1cmUgJHtyZWxhdGVkRml4dHVyZUlkfSBub3QgZm91bmQgaW4gaW5zZXJ0ZWRJZHMsIHNraXBwaW5nYCxcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHRhcmdldElkID0gcmVzb2x2ZWRJZDtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHRhcmdldElkID0gcmVsYXRlZElkO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAvLyBKb2luVGFibGXsl5Ag7IK97J6FXG4gICAgICAgICAgICBjb25zdCBbZm91bmRdID0gYXdhaXQgdHJ4KGpvaW5UYWJsZSlcbiAgICAgICAgICAgICAgLndoZXJlKHtcbiAgICAgICAgICAgICAgICBbc291cmNlQ29sdW1uXTogc291cmNlSWQsXG4gICAgICAgICAgICAgICAgW3RhcmdldENvbHVtbl06IHRhcmdldElkLFxuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAubGltaXQoMSk7XG5cbiAgICAgICAgICAgIGlmICghZm91bmQpIHtcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4KGpvaW5UYWJsZSkuaW5zZXJ0KHtcbiAgICAgICAgICAgICAgICBbc291cmNlQ29sdW1uXTogc291cmNlSWQsXG4gICAgICAgICAgICAgICAgW3RhcmdldENvbHVtbl06IHRhcmdldElkLFxuICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmdyZWVuKFxuICAgICAgICAgICAgICAgICAgICBgSW5zZXJ0ZWQgaW50byAke2pvaW5UYWJsZX06ICR7ZW50aXR5LnRhYmxlfSgke3NvdXJjZUlkfSkgLSAke3JlbGF0ZWRFbnRpdHkudGFibGV9KCR7dGFyZ2V0SWR9KWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIOqwmeydgCDthYzsnbTruJQg64K0IGZpeHR1cmXrk6TsnYQgc2VsZi1yZWZlcmVuY2Ug66CI67Ko67OE66GcIOu2hO2VoFxuICAgKiAtIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuKlCBmaXh0dXJl65OkOiBMZXZlbCAwXG4gICAqIC0gTGV2ZWwgMOydhCDssLjsobDtlZjripQgZml4dHVyZeuTpDogTGV2ZWwgMVxuICAgKiAtIOuwmOuztVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy6rCAIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuXG4gICAqIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0IEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBncm91cEZpeHR1cmVzQnlMZXZlbChmaXh0dXJlczogRml4dHVyZVJlY29yZFtdKTogRml4dHVyZVJlY29yZFtdW10ge1xuICAgIGlmIChmaXh0dXJlcy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG5cbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlc1swXS5lbnRpdHlJZCk7XG5cbiAgICAvLyBzZWxmLXJlZmVyZW5jZSByZWxhdGlvbiBwcm9wIOywvuq4sFxuICAgIGNvbnN0IHNlbGZSZWZQcm9wcyA9IGVudGl0eS5wcm9wcy5maWx0ZXIoXG4gICAgICAocCk6IHAgaXMgQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIHwgT25lVG9PbmVSZWxhdGlvblByb3AgPT5cbiAgICAgICAgaXNSZWxhdGlvblByb3AocCkgJiZcbiAgICAgICAgKGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHApIHx8IChpc09uZVRvT25lUmVsYXRpb25Qcm9wKHApICYmIHAuaGFzSm9pbkNvbHVtbikpICYmXG4gICAgICAgIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICk7XG5cbiAgICBpZiAoc2VsZlJlZlByb3BzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgLy8gc2VsZi1yZWZlcmVuY2Ug7JeG7J2MIOKGkiDri6jsnbwg66CI67KoXG4gICAgICByZXR1cm4gW2ZpeHR1cmVzXTtcbiAgICB9XG5cbiAgICAvLyDroIjrsqjrs4Qg67aE7ZWgICh0b3BvbG9naWNhbCBzb3J0KVxuICAgIGNvbnN0IGxldmVsczogRml4dHVyZVJlY29yZFtdW10gPSBbXTtcbiAgICBjb25zdCByZW1haW5pbmcgPSBuZXcgU2V0KGZpeHR1cmVzLm1hcCgoZikgPT4gZi5maXh0dXJlSWQpKTtcbiAgICBjb25zdCBwcm9jZXNzZWQgPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgIHdoaWxlIChyZW1haW5pbmcuc2l6ZSA+IDApIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRMZXZlbDogRml4dHVyZVJlY29yZFtdID0gW107XG5cbiAgICAgIGZvciAoY29uc3QgZml4dHVyZSBvZiBmaXh0dXJlcykge1xuICAgICAgICBpZiAoIXJlbWFpbmluZy5oYXMoZml4dHVyZS5maXh0dXJlSWQpKSBjb250aW51ZTtcblxuICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDrqqjrkZAg7J2066+4IOyymOumrOuQkOqxsOuCmCBudWxs7J24IOqyveyasFxuICAgICAgICBjb25zdCBjYW5Qcm9jZXNzID0gc2VsZlJlZlByb3BzLmV2ZXJ5KChwcm9wKSA9PiB7XG4gICAgICAgICAgY29uc3QgcmVmSWQgPSBmaXh0dXJlLmNvbHVtbnNbcHJvcC5uYW1lXT8udmFsdWUgYXMgbnVtYmVyIHwgbnVsbDtcbiAgICAgICAgICBpZiAocmVmSWQgPT09IG51bGwgfHwgcmVmSWQgPT09IHVuZGVmaW5lZCkgcmV0dXJuIHRydWU7XG4gICAgICAgICAgY29uc3QgcmVmRml4dHVyZUlkID0gYCR7cHJvcC53aXRofSMke3JlZklkfWA7XG4gICAgICAgICAgLy8g7J2066+4IOyymOumrOuQkOqxsOuCmCwg7ZiE7J6sIGZpeHR1cmVz7JeQIO2PrO2VqOuQmOyngCDslYrsnYAg6rK97JqwICjsmbjrtoAg7LC47KGwKVxuICAgICAgICAgIHJldHVybiBwcm9jZXNzZWQuaGFzKHJlZkZpeHR1cmVJZCkgfHwgIXJlbWFpbmluZy5oYXMocmVmRml4dHVyZUlkKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKGNhblByb2Nlc3MpIHtcbiAgICAgICAgICBjdXJyZW50TGV2ZWwucHVzaChmaXh0dXJlKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAoY3VycmVudExldmVsLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICBjb25zdCByZW1haW5pbmdJZHMgPSBBcnJheS5mcm9tKHJlbWFpbmluZykuam9pbihcIiwgXCIpO1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgYENpcmN1bGFyIHNlbGYtcmVmZXJlbmNlIGRldGVjdGVkIGluICR7ZW50aXR5LnRhYmxlfS4gUmVtYWluaW5nIGZpeHR1cmVzOiAke3JlbWFpbmluZ0lkc31gLFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgY3VycmVudExldmVsKSB7XG4gICAgICAgIHJlbWFpbmluZy5kZWxldGUoZml4dHVyZS5maXh0dXJlSWQpO1xuICAgICAgICBwcm9jZXNzZWQuYWRkKGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgIH1cblxuICAgICAgbGV2ZWxzLnB1c2goY3VycmVudExldmVsKTtcbiAgICB9XG5cbiAgICByZXR1cm4gbGV2ZWxzO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBjaGVja1VuaXF1ZVZpb2xhdGlvbihkYjogS25leCwgZW50aXR5OiBFbnRpdHksIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQpIHtcbiAgICBjb25zdCBfdW5pcXVlSW5kZXhlcyA9IGVudGl0eS5pbmRleGVzPy5maWx0ZXIoKGkpID0+IGkudHlwZSA9PT0gXCJ1bmlxdWVcIikgPz8gW107XG5cbiAgICBjb25zdCB1bmlxdWVJbmRleGVzID0gX3VuaXF1ZUluZGV4ZXMuZmlsdGVyKChpbmRleCkgPT5cbiAgICAgIGluZGV4LmNvbHVtbnMuZXZlcnkoKGNvbHVtbikgPT4gIWNvbHVtbi5uYW1lLnN0YXJ0c1dpdGgoYCR7ZW50aXR5LnRhYmxlfV9fYCkpLFxuICAgICk7XG4gICAgaWYgKHVuaXF1ZUluZGV4ZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBsZXQgdW5pcXVlUXVlcnkgPSBkYihlbnRpdHkudGFibGUpO1xuICAgIGxldCBoYXNDb25kaXRpb24gPSBmYWxzZTtcblxuICAgIGZvciAoY29uc3QgaW5kZXggb2YgdW5pcXVlSW5kZXhlcykge1xuICAgICAgLy8g7Lus65+8IOykkSDtlZjrgpjrnbzrj4QgbnVsbOydtOuptCDsnKDri4jtgawg7KCc7JW97J2EIOychOuwmO2VmOyngCDslYrquLAg65WM66y47JeQIO2VtOuLuSDsnbjrjbHsiqTripQg66y07IucXG4gICAgICBjb25zdCBjb250YWluc051bGwgPSBpbmRleC5jb2x1bW5zLnNvbWUoKGNvbHVtbikgPT4ge1xuICAgICAgICBjb25zdCBmaWVsZCA9IGNvbHVtbi5uYW1lLnJlcGxhY2UoL19pZCQvLCBcIlwiKTtcbiAgICAgICAgcmV0dXJuIGZpeHR1cmUuY29sdW1uc1tmaWVsZF0/LnZhbHVlID09PSBudWxsO1xuICAgICAgfSk7XG4gICAgICBpZiAoY29udGFpbnNOdWxsKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICB1bmlxdWVRdWVyeSA9IHVuaXF1ZVF1ZXJ5Lm9yV2hlcmUoKHFiKSA9PiB7XG4gICAgICAgIGZvciAoY29uc3QgY29sdW1uIG9mIGluZGV4LmNvbHVtbnMpIHtcbiAgICAgICAgICBjb25zdCBmaWVsZCA9IGNvbHVtbi5uYW1lLnJlcGxhY2UoL19pZCQvLCBcIlwiKTtcblxuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KGZpeHR1cmUuY29sdW1uc1tmaWVsZF0/LnZhbHVlKSkge1xuICAgICAgICAgICAgcWIud2hlcmVJbihjb2x1bW4ubmFtZSwgZml4dHVyZS5jb2x1bW5zW2ZpZWxkXS52YWx1ZSk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHFiLmFuZFdoZXJlKGNvbHVtbi5uYW1lLCBmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KTtcbiAgICAgIGhhc0NvbmRpdGlvbiA9IHRydWU7XG4gICAgfVxuXG4gICAgaWYgKCFoYXNDb25kaXRpb24pIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGNvbnN0IFt1bmlxdWVGb3VuZF0gPSBhd2FpdCB1bmlxdWVRdWVyeTtcbiAgICByZXR1cm4gdW5pcXVlRm91bmQ7XG4gIH1cblxuICBwcml2YXRlIGFzeW5jIGNoZWNrRHVwbGljYXRlQnlDb2x1bW5zKFxuICAgIGRiOiBLbmV4LFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgY29sdW1uczogc3RyaW5nW10sXG4gICkge1xuICAgIGlmIChjb2x1bW5zLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgY29uc3Qgd2hlcmVDbGF1c2U6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICBmb3IgKGNvbnN0IGNvbHVtbiBvZiBjb2x1bW5zKSB7XG4gICAgICAvLyByZWxhdGlvbiDtlYTrk5zsnbgg6rK97JqwIF9pZCDrtpnsnbTquLBcbiAgICAgIGNvbnN0IHByb3AgPSBlbnRpdHkucHJvcHMuZmluZCgocCkgPT4gcC5uYW1lID09PSBjb2x1bW4pO1xuICAgICAgY29uc3QgZGJDb2x1bW4gPSBwcm9wICYmIGlzUmVsYXRpb25Qcm9wKHByb3ApID8gYCR7Y29sdW1ufV9pZGAgOiBjb2x1bW47XG4gICAgICBjb25zdCB2YWx1ZSA9IGZpeHR1cmUuY29sdW1uc1tjb2x1bW5dPy52YWx1ZTtcblxuICAgICAgLy8gbnVsbCDqsJLsnbQg7Y+s7ZWo65CcIOqyveyasCDspJHrs7Ug7ZmV7J24IOyKpO2CtVxuICAgICAgaWYgKHZhbHVlID09PSBudWxsIHx8IHZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG5cbiAgICAgIHdoZXJlQ2xhdXNlW2RiQ29sdW1uXSA9IHZhbHVlO1xuICAgIH1cblxuICAgIGNvbnN0IFtmb3VuZF0gPSBhd2FpdCBkYihlbnRpdHkudGFibGUpLndoZXJlKHdoZXJlQ2xhdXNlKS5saW1pdCgxKTtcbiAgICByZXR1cm4gZm91bmQ7XG4gIH1cblxuICBhc3luYyBhZGRGaXh0dXJlTG9hZGVyKGNvZGU6IHN0cmluZykge1xuICAgIGNvbnN0IHBhdGggPSBgJHtTb25hbXUuYXBpUm9vdFBhdGh9L3NyYy90ZXN0aW5nL2ZpeHR1cmUudHNgO1xuICAgIGNvbnN0IGNvbnRlbnQgPSByZWFkRmlsZVN5bmMocGF0aCkudG9TdHJpbmcoKTtcblxuICAgIGNvbnN0IGZpeHR1cmVMb2FkZXJTdGFydCA9IGNvbnRlbnQuaW5kZXhPZihcImNvbnN0IGZpeHR1cmVMb2FkZXIgPSB7XCIpO1xuICAgIGNvbnN0IGZpeHR1cmVMb2FkZXJFbmQgPSBjb250ZW50LmluZGV4T2YoXCJ9O1wiLCBmaXh0dXJlTG9hZGVyU3RhcnQpO1xuXG4gICAgaWYgKGZpeHR1cmVMb2FkZXJTdGFydCAhPT0gLTEgJiYgZml4dHVyZUxvYWRlckVuZCAhPT0gLTEpIHtcbiAgICAgIGNvbnN0IG5ld0NvbnRlbnQgPSBgJHtjb250ZW50LnNsaWNlKDAsIGZpeHR1cmVMb2FkZXJFbmQpfSAgJHtjb2RlfVxcbiR7Y29udGVudC5zbGljZShmaXh0dXJlTG9hZGVyRW5kKX1gO1xuXG4gICAgICB3cml0ZUZpbGVTeW5jKHBhdGgsIG5ld0NvbnRlbnQpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGYWlsZWQgdG8gZmluZCBmaXh0dXJlTG9hZGVyIGluIGZpeHR1cmUudHNcIik7XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBjb25zdCBGaXh0dXJlTWFuYWdlciA9IG5ldyBGaXh0dXJlTWFuYWdlckNsYXNzKCk7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O2NBVXVDO2tCQUNZO1lBRUc7c0JBQ0s7c0JBR0Y7YUFRakM7a0JBVXFCO3VCQUNLO0NBU3JDLHNCQUFiLE1BQWlDO0VBQy9CLEFBQVEsT0FBb0I7RUFDNUIsSUFBSSxJQUFJLEtBQVc7QUFDakIsUUFBSyxPQUFPOztFQUVkLElBQUksTUFBWTtBQUNkLE9BQUksS0FBSyxTQUFTLE1BQU07QUFDdEIsVUFBTSxJQUFJLE1BQU0sMENBQTBDOztBQUU1RCxVQUFPLEtBQUs7O0VBR2QsQUFBUSxPQUFvQjtFQUM1QixJQUFJLElBQUksS0FBVztBQUNqQixRQUFLLE9BQU87O0VBRWQsSUFBSSxNQUFZO0FBQ2QsT0FBSSxLQUFLLFNBQVMsTUFBTTtBQUN0QixVQUFNLElBQUksTUFBTSwwQ0FBMEM7O0FBRTVELFVBQU8sS0FBSzs7RUFFZCxtQkFBb0M7RUFFcEMsQUFBUSxnQkFBZ0IsSUFBSSxlQUFlO0VBRzNDLEFBQVEsVUFBeUIsSUFBSSxlQUFlO0VBQ3BELEFBQVEsZ0JBQW9DLElBQUksS0FBSztFQUNyRCxBQUFRLGtCQUNOLElBQUksS0FBSztFQUVYLE9BQU87QUFDTCxPQUFJLEtBQUssU0FBUyxNQUFNO0FBQ3RCOztBQUVGLE9BQUksT0FBTyxTQUFTLFFBQVEsT0FBTyxTQUFTLG1CQUFtQjtJQUM3RCxNQUFNLFFBQVEsT0FBTyxTQUFTLEtBQUs7SUFHbkMsTUFBTSxRQUFRLE9BQU8sU0FBUyxrQkFBa0I7QUFHaEQsUUFDRSxHQUFHLE1BQU0sUUFBUSxZQUFZLEdBQUcsTUFBTSxRQUFRLEtBQUssR0FBRyxNQUFNLGVBQzVELEdBQUcsTUFBTSxRQUFRLFlBQVksR0FBRyxNQUFNLFFBQVEsS0FBSyxHQUFHLE1BQU0sWUFDNUQ7QUFDQSxXQUFNLElBQUksTUFBTSxzQ0FBc0M7OztBQUkxRCxRQUFLLE1BQU0sbUJBQW1CLE9BQU8sU0FBUyxLQUFLO0FBQ25ELFFBQUssTUFBTSxtQkFBbUIsT0FBTyxTQUFTLFFBQVE7Ozs7OztFQU94RCxNQUFNLE9BQU87R0FDWCxNQUFNLGNBQWMsT0FBTyxTQUFTLFFBQVE7R0FDNUMsTUFBTSxXQUFXLE9BQU8sU0FBUyxLQUFLO0dBR3RDLE1BQU0sWUFBWSxFQUFFLFlBQVksU0FBUyxZQUFZLElBQUk7QUFDekQsWUFDRSxXQUFXLFNBQVMsS0FBSyxNQUFNLFNBQVMsUUFBUSxLQUFLLE1BQU0sU0FBUyxLQUFLOzs7MkJBR3BELFNBQVMsU0FBUzs7VUFHdkM7SUFBRSxPQUFPO0lBQVcsS0FBSztLQUFFLEdBQUcsUUFBUTtLQUFLLEdBQUc7S0FBVztJQUF1QixDQUNqRjtBQUVELFlBQ0UsV0FBVyxTQUFTLEtBQUssTUFBTSxTQUFTLFFBQVEsS0FBSyxNQUFNLFNBQVMsS0FBSyw4Q0FBOEMsU0FBUyxTQUFTLE9BQ3pJO0lBQUUsT0FBTztJQUFXLEtBQUs7S0FBRSxHQUFHLFFBQVE7S0FBSyxHQUFHO0tBQVc7SUFBdUIsQ0FDakY7QUFFRCxZQUNFLFdBQVcsU0FBUyxLQUFLLE1BQU0sU0FBUyxRQUFRLEtBQUssTUFBTSxTQUFTLEtBQUssc0NBQXNDLFNBQVMsU0FBUyxPQUNqSTtJQUFFLE9BQU87SUFBVyxLQUFLO0tBQUUsR0FBRyxRQUFRO0tBQUssR0FBRztLQUFXO0lBQXVCLENBQ2pGO0dBR0QsTUFBTSxlQUFlLEVBQUUsWUFBWSxZQUFZLFlBQVksSUFBSTtHQUMvRCxNQUFNLFVBQVUsY0FBYyxZQUFZLEtBQUssTUFBTSxZQUFZLFFBQVEsS0FBSyxNQUFNLFlBQVksS0FBSyxNQUFNLFlBQVksU0FBUztHQUNoSSxNQUFNLGFBQWEsaUJBQWlCLFNBQVMsS0FBSyxNQUFNLFNBQVMsUUFBUSxLQUFLLE1BQU0sU0FBUyxLQUFLLE1BQU0sU0FBUyxTQUFTO0FBRTFILFlBQVMsR0FBRyxRQUFRLGlCQUFpQixTQUFTLFlBQVksR0FBRyxJQUFJLGNBQWM7SUFDN0UsT0FBTztJQUNQLEtBQUs7S0FBRSxHQUFHLFFBQVE7S0FBSyxHQUFHO0tBQWM7SUFDeEMsT0FBTztJQUNSLENBQUM7QUFHRixTQUFNLEtBQUssZUFBZSxPQUFPLFNBQVMsS0FBSzs7Ozs7O0VBT2pELE1BQWMsZUFBZSxVQUFrQztHQUM3RCxNQUFNLFNBQVMsbUJBQW1CLFNBQVM7R0FDM0MsTUFBTSxXQUFXLGNBQWMsZ0JBQWdCO0FBRS9DLE9BQUk7QUFDRixTQUFLLE1BQU0sVUFBVSxVQUFVO0tBQzdCLE1BQU0sWUFBWSxPQUFPLFNBQVMsT0FBTyxHQUFHLGFBQWE7S0FHekQsTUFBTSxTQUFTLE9BQU8sTUFBTSxNQUFNLE1BQU0sRUFBRSxTQUFTLEtBQUs7S0FDeEQsTUFBTSxTQUFTLFFBQVE7S0FHdkIsTUFBTSxlQUNKLFdBQVcsYUFDWCxXQUFXLGdCQUNYLFFBQVEsTUFBTSxvQkFBb0I7QUFFcEMsU0FBSSxDQUFDLGNBQWM7QUFDakIsT0FBQyxRQUFRLElBQ1AsUUFBUSxJQUNOLCtCQUErQixVQUFVLGFBQWEsVUFBVSxVQUFVLEdBQzNFO0FBQ0g7O0tBS0YsTUFBTSxVQUFVLFdBQVcsV0FBVyxvQkFBb0I7QUFDMUQsV0FBTSxPQUFPLElBQUk7OzZDQUVvQixVQUFVOytCQUN4QixRQUFRLFFBQVEsVUFBVTs7VUFFL0M7O2FBRUk7QUFDUixVQUFNLE9BQU8sU0FBUzs7O0VBSTFCLEFBQVEsaUJBQWlCLElBQUksS0FBYTtFQUMxQyxNQUFNLGNBQWMsVUFBa0IsS0FBZTtBQUVuRCxRQUFLLGVBQWUsT0FBTztHQUUzQixNQUFNLFVBQVUsUUFFWixNQUFNLFFBQVEsSUFDWixJQUFJLElBQUksT0FBTyxPQUFPO0FBQ3BCLFdBQU8sTUFBTSxLQUFLLGlCQUFpQixVQUFVLE1BQU0sR0FBRztLQUN0RCxDQUNILEVBQ0QsTUFBTSxDQUNUO0dBRUQsTUFBTSxNQUFNLFVBQVUsTUFBTSxJQUFJO0FBQ2hDLFFBQUssTUFBTSxTQUFTLFNBQVM7QUFDM0IsVUFBTSxJQUFJLElBQUksTUFBTTs7O0VBSXhCLE1BQU0saUJBQWlCLFVBQWtCLE9BQWUsSUFBK0I7R0FDckYsTUFBTSxZQUFZLEdBQUcsU0FBUyxHQUFHLE1BQU0sR0FBRztBQUcxQyxPQUFJLEtBQUssZUFBZSxJQUFJLFVBQVUsRUFBRTtBQUN0QyxXQUFPLEVBQUU7O0FBRVgsUUFBSyxlQUFlLElBQUksVUFBVTtHQUVsQyxNQUFNLFNBQVMsY0FBYyxJQUFJLFNBQVM7R0FDMUMsTUFBTSxNQUFNLFVBQVUsTUFBTSxJQUFJO0dBR2hDLE1BQU0sQ0FBQyxPQUFPLE1BQU0sSUFBSSxPQUFPLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxDQUFDLE1BQU0sRUFBRTtBQUMvRCxPQUFJLFFBQVEsV0FBVztBQUNyQixVQUFNLElBQUksTUFBTSxHQUFHLFNBQVMsR0FBRyxHQUFHLGtCQUFrQjs7R0FJdEQsTUFBTSxrQkFBbUIsT0FBTyxTQUFTLFFBQVEsV0FBcUM7R0FDdEYsTUFBTSxlQUFnQixPQUFPLFNBQVMsa0JBQWtCLFdBQ3JEO0dBRUgsTUFBTSxZQUFZLHdCQUF3QixnQkFBZ0IsT0FBTyxPQUFPLE1BQU0sc0JBQXNCLGFBQWEsT0FBTyxPQUFPLE1BQU0sb0JBQW9CLEdBQUc7R0FFNUosTUFBTSxPQUFPLE9BQU8sUUFBUSxPQUFPLFVBQVUsQ0FDMUMsUUFDRSxHQUFHLGNBQ0YsMkJBQTJCLFNBQVMsSUFDbkMsdUJBQXVCLFNBQVMsSUFBSSxTQUFTLHFCQUFxQixVQUN0RSxDQUNBLEtBQUssR0FBRyxjQUFjO0lBU3JCLElBQUlBO0lBQ0osSUFBSUM7QUFDSixRQUFJLHVCQUF1QixTQUFTLElBQUksQ0FBQyxTQUFTLGVBQWU7S0FDL0QsTUFBTSxnQkFBZ0IsY0FBYyxJQUFJLFNBQVMsS0FBSztLQUN0RCxNQUFNLHNCQUFzQixjQUFjLE1BQU0sTUFDN0MsTUFBTSxlQUFlLEVBQUUsSUFBSSxFQUFFLFNBQVMsT0FBTyxHQUMvQyxFQUFFO0FBQ0gsU0FBSSxDQUFDLHFCQUFxQjtBQUN4QixZQUFNLElBQUksTUFBTSxHQUFHLGNBQWMsR0FBRyxJQUFJLE9BQU8sR0FBRyxvQkFBb0I7O0FBRXhFLGVBQVEsR0FBRyxvQkFBb0I7QUFDL0IsWUFBSyxJQUFJO1dBQ0o7QUFDTCxlQUFRO0FBQ1IsWUFBSyxJQUFJLEdBQUcsU0FBUyxLQUFLOztBQUU1QixXQUFPO0tBQ0wsVUFBVSxTQUFTO0tBQ25CO0tBQ0E7S0FDRDtLQUNELENBQ0QsUUFBUSxRQUFRLElBQUksT0FBTyxLQUFLO0dBRW5DLE1BQU0sYUFBYSxNQUFNLFFBQVEsSUFDL0IsS0FBSyxJQUFJLE9BQU8sV0FBUztBQUN2QixXQUFPLEtBQUssaUJBQWlCQyxPQUFLLFVBQVVBLE9BQUssT0FBT0EsT0FBSyxHQUFHO0tBQ2hFLENBQ0g7QUFFRCxVQUFPLENBQUMsR0FBRyxPQUFPLFdBQVcsWUFBWSxDQUFDLE1BQU0sQ0FBQyxFQUFFLFVBQVU7O0VBRy9ELE1BQU0sVUFBVTtBQUNkLE9BQUksS0FBSyxNQUFNO0FBQ2IsVUFBTSxLQUFLLEtBQUssU0FBUztBQUN6QixTQUFLLE9BQU87O0FBRWQsT0FBSSxLQUFLLE1BQU07QUFDYixVQUFNLEtBQUssS0FBSyxTQUFTO0FBQ3pCLFNBQUssT0FBTzs7QUFFZCxTQUFNLFVBQVUsU0FBUzs7RUFHM0IsTUFBTSxZQUNKLGNBQ0EsY0FDQSxlQUNBLGdCQUNBO0dBQ0EsTUFBTSxXQUFXLG1CQUFtQixPQUFPLFNBQVMsY0FBYztHQUNsRSxNQUFNLFdBQVcsbUJBQW1CLE9BQU8sU0FBUyxjQUFjO0FBRWxFLE9BQUk7SUFDRixNQUFNLEVBQUUsVUFBVSxPQUFPLE9BQU8sZUFBZTtJQUUvQyxNQUFNLFNBQVMsY0FBYyxJQUFJLFNBQVM7SUFDMUMsTUFBTSxTQUNKLE9BQU8sTUFBTSxNQUFNLFNBQVMsS0FBSyxTQUFTLE1BQU0sRUFBRSxTQUFTLGFBQ3ZELEdBQUcsTUFBTSxPQUNUO0lBRU4sSUFBSSxRQUFRLFNBQVMsT0FBTyxNQUFNO0FBQ2xDLFFBQUksZUFBZSxVQUFVO0FBQzNCLGFBQVEsTUFBTSxNQUFNLFFBQVEsTUFBTTtlQUN6QixlQUFlLFFBQVE7QUFDaEMsYUFBUSxNQUFNLE1BQU0sUUFBUSxRQUFRLElBQUksTUFBTSxHQUFHOztJQUduRCxNQUFNLE9BQU8sTUFBTTtBQUNuQixRQUFJLEtBQUssV0FBVyxHQUFHO0FBQ3JCLFdBQU0sSUFBSSxNQUFNLG1CQUFtQjs7SUFHckMsTUFBTUMsV0FBNEIsRUFBRTtBQUNwQyxTQUFLLE1BQU0sT0FBTyxNQUFNO0tBQ3RCLE1BQU0sdUJBQXVCLFNBQVM7S0FDdEMsTUFBTSxhQUFhLE1BQU0sS0FBSyxvQkFBb0IsUUFBUSxLQUFLLEVBQzdELEtBQUssVUFDTixDQUFDO0FBQ0YsY0FBUyxLQUFLLEdBQUcsV0FBVztLQUM1QixNQUFNLHVCQUF1QixTQUFTLE1BQU0sTUFBTSxFQUFFLGNBQWMsR0FBRyxTQUFTLEdBQUcsSUFBSSxLQUFLO0FBRTFGLFNBQUksc0JBQXNCO0FBRXhCLDJCQUFxQixpQkFBaUIsU0FDbkMsUUFBUSxNQUFNLEVBQUUsY0FBYyxxQkFBcUIsVUFBVSxDQUM3RCxNQUFNLHFCQUFxQixDQUMzQixLQUFLLE1BQU0sRUFBRSxVQUFVOzs7QUFJOUIsZUFBVyxNQUFNLFdBQVcsVUFBVTtLQUNwQyxNQUFNQyxXQUFTLGNBQWMsSUFBSSxRQUFRLFNBQVM7S0FHbEQsTUFBTSxnQkFBZ0IsZ0JBQWdCLFVBQVUsUUFBUTtBQUN4RCxTQUFJLGlCQUFpQixjQUFjLFNBQVMsR0FBRztNQUM3QyxNQUFNLHFCQUFxQixNQUFNLEtBQUssd0JBQ3BDLFVBQ0FBLFVBQ0EsU0FDQSxjQUNEO0FBQ0QsVUFBSSxvQkFBb0I7T0FDdEIsTUFBTSxDQUFDLFVBQVUsTUFBTSxLQUFLLG9CQUFvQkEsVUFBUSxvQkFBb0I7UUFDMUUsY0FBYztRQUNkLEtBQUs7UUFDTixDQUFDO0FBQ0YsZUFBUSxTQUFTOzs7S0FLckIsTUFBTSxZQUFZLE1BQU0sS0FBSyxxQkFBcUIsVUFBVUEsVUFBUSxRQUFRO0FBQzVFLFNBQUksV0FBVztNQUNiLE1BQU0sQ0FBQyxVQUFVLE1BQU0sS0FBSyxvQkFBb0JBLFVBQVEsV0FBVztPQUNqRSxjQUFjO09BQ2QsS0FBSztPQUNOLENBQUM7QUFDRixjQUFRLFNBQVM7OztBQUlyQixXQUFPLE9BQU8sV0FBVyxNQUFNLEVBQUUsVUFBVTthQUNuQztBQUNSLFVBQU0sUUFBUSxXQUFXLENBQUMsU0FBUyxTQUFTLEVBQUUsU0FBUyxTQUFTLENBQUMsQ0FBQzs7O0VBSXRFLE1BQU0sb0JBQ0osUUFDQSxLQUlBLFNBSTBCO0dBQzFCLE1BQU1DLFVBQTJCLEVBQUU7R0FDbkMsTUFBTSxrQkFBa0IsSUFBSSxLQUFhO0dBRXpDLE1BQU0sU0FBUyxPQUNiLFVBQ0EsVUFJRztJQUNILE1BQU0sWUFBWSxHQUFHRCxTQUFPLEdBQUcsR0FBR0UsTUFBSTtBQUN0QyxRQUFJLGdCQUFnQixJQUFJLFVBQVUsRUFBRTtBQUNsQzs7QUFFRixvQkFBZ0IsSUFBSSxVQUFVO0lBRTlCLE1BQU1DLFNBQXdCO0tBQzVCO0tBQ0EsVUFBVUgsU0FBTztLQUNqQixJQUFJRSxNQUFJO0tBQ1IsU0FBUyxFQUFFO0tBQ1gsZ0JBQWdCLEVBQUU7S0FDbEIsZ0JBQWdCLEVBQUU7S0FDbkI7QUFFRCxTQUFLLE1BQU0sUUFBUUYsU0FBTyxPQUFPO0FBQy9CLFNBQUksY0FBYyxLQUFLLEVBQUU7QUFDdkI7O0FBR0YsWUFBTyxRQUFRLEtBQUssUUFBUTtNQUNwQjtNQUNOLE9BQU9FLE1BQUksS0FBSztNQUNqQjtLQUVELE1BQU0sS0FBSyxTQUFTLE9BQU8sVUFBVSxNQUFNLElBQUk7QUFDL0MsU0FBSSx5QkFBeUIsS0FBSyxFQUFFO01BQ2xDLE1BQU0sZ0JBQWdCLGNBQWMsSUFBSSxLQUFLLEtBQUs7TUFDbEQsTUFBTSxlQUFlLEtBQUs7TUFDMUIsTUFBTSxhQUFhLEdBQUcsV0FBVyxZQUFZRixTQUFPLE1BQU0sQ0FBQztNQUMzRCxNQUFNLFdBQVcsR0FBRyxXQUFXLFlBQVksY0FBYyxNQUFNLENBQUM7TUFFaEUsTUFBTSxhQUFhLE1BQU0sR0FBRyxhQUFhLENBQUMsTUFBTSxZQUFZRSxNQUFJLEdBQUcsQ0FBQyxNQUFNLFNBQVM7QUFDbkYsYUFBTyxRQUFRLEtBQUssTUFBTSxRQUFRO2dCQUN6QixzQkFBc0IsS0FBSyxFQUFFO01BQ3RDLE1BQU0sZ0JBQWdCLGNBQWMsSUFBSSxLQUFLLEtBQUs7TUFDbEQsTUFBTSxhQUFhLE1BQU0sR0FBRyxjQUFjLE1BQU0sQ0FDN0MsTUFBTSxLQUFLLFlBQVlBLE1BQUksR0FBRyxDQUM5QixNQUFNLEtBQUs7QUFDZCxhQUFPLFFBQVEsS0FBSyxNQUFNLFFBQVE7Z0JBQ3pCLHVCQUF1QixLQUFLLElBQUksQ0FBQyxLQUFLLGVBQWU7TUFHOUQsTUFBTSxnQkFBZ0IsY0FBYyxJQUFJLEtBQUssS0FBSztNQUNsRCxNQUFNLGNBQWMsY0FBYyxNQUFNLE1BQ3JDLE1BQU0sZUFBZSxFQUFFLElBQUksRUFBRSxTQUFTRixTQUFPLEdBQy9DO0FBQ0QsVUFBSSxlQUFlLGVBQWUsWUFBWSxFQUFFO09BRTlDLE1BQU0sV0FBVyxHQUFHLFlBQVksS0FBSztPQUNyQyxNQUFNLGFBQWEsTUFBTSxHQUFHLGNBQWMsTUFBTSxDQUFDLE1BQU0sVUFBVUUsTUFBSSxHQUFHLENBQUMsT0FBTztBQUNoRixjQUFPLFFBQVEsS0FBSyxNQUFNLFFBQVEsWUFBWTs7Z0JBRXZDLGVBQWUsS0FBSyxFQUFFO01BQy9CLE1BQU0sWUFBWUEsTUFBSSxHQUFHLEtBQUssS0FBSztBQUNuQyxhQUFPLFFBQVEsS0FBSyxNQUFNLFFBQVE7QUFDbEMsVUFBSSxXQUFXO0FBQ2IsY0FBTyxlQUFlLEtBQUssR0FBRyxLQUFLLEtBQUssR0FBRyxZQUFZOztBQUV6RCxVQUFJLENBQUMsU0FBUyxnQkFBZ0IsV0FBVztPQUN2QyxNQUFNLGdCQUFnQixjQUFjLElBQUksS0FBSyxLQUFLO09BQ2xELE1BQU0sYUFBYSxNQUFNLEdBQUcsY0FBYyxNQUFNLENBQUMsTUFBTSxNQUFNLFVBQVUsQ0FBQyxPQUFPO0FBQy9FLFdBQUksWUFBWTtBQUNkLGNBQU0sT0FBTyxlQUFlLFdBQVc7Ozs7O0FBTS9DLFlBQVEsS0FBSyxPQUFPOztBQUd0QixTQUFNLE9BQU8sUUFBUSxJQUFJO0FBRXpCLFVBQU87Ozs7Ozs7Ozs7OztFQWFULE1BQU0sZUFDSixRQUNBLFdBQ2dDO0dBQ2hDLE1BQU0sV0FBVyxPQUFPLFlBQVksTUFBTSxFQUFFLFVBQVU7QUFHdEQsUUFBSyxVQUFVLElBQUksZUFBZTtBQUNsQyxRQUFLLGdCQUFnQixJQUFJLEtBQUs7QUFDOUIsUUFBSyxrQkFBa0IsSUFBSSxLQUFLO0dBR2hDLE1BQU0sV0FDSixRQUFRLElBQUkscUJBQXFCLFVBQVUsUUFBUSxJQUFJLHdCQUM1QztJQUNMLE1BQU0sV0FBVyxTQUFTLFFBQVEsSUFBSSxrQkFBa0IsS0FBSyxHQUFHO0lBQ2hFLE1BQU0sYUFBYSxPQUFPLFNBQVM7SUFDbkMsTUFBTSxhQUFhLFdBQVc7QUFDOUIsV0FBTztLQUNMLEdBQUc7S0FDSCxZQUFZO01BQUUsR0FBRztNQUFZLFVBQVUsR0FBRyxXQUFXLFNBQVMsR0FBRztNQUFZO0tBQzdFLE1BQU07TUFBRSxLQUFLO01BQUcsS0FBSztNQUFHO0tBQ3pCO09BQ0MsR0FDSixPQUFPLFNBQVM7R0FDdEIsTUFBTSxLQUFLLG1CQUFtQixTQUFTO0dBQ3ZDLE1BQU1FLFVBQWlDLEVBQUU7QUFFekMsT0FBSTtBQUVGLFNBQUssY0FBYyxXQUFXLFNBQVM7SUFDdkMsTUFBTSxpQkFBaUIsS0FBSyxjQUFjLG1CQUFtQjtBQUc3RCxTQUFLLE1BQU0sYUFBYSxnQkFBZ0I7S0FDdEMsTUFBTSxVQUFVLFNBQVMsTUFBTSxNQUFNLEVBQUUsY0FBYyxVQUFVO0FBQy9ELFNBQUksQ0FBQyxRQUFTO0tBRWQsTUFBTSxZQUFZLENBQUMsQ0FBQyxRQUFRO0tBQzVCLE1BQU0sWUFBWSxDQUFDLENBQUMsUUFBUTtLQUM1QixNQUFNLGVBQWUsYUFBYTtBQUdsQyxTQUFJLGdCQUFnQixDQUFDLFFBQVEsVUFBVTtNQUVyQyxNQUFNLGFBQWEsUUFBUSxRQUFRLE1BQU0sUUFBUSxRQUFRO0FBQ3pELGFBQU8sV0FBVztBQUNsQixXQUFLLGdCQUFnQixJQUFJLFdBQVc7T0FDbEMsVUFBVSxRQUFRO09BQ2xCO09BQ0QsQ0FBQztBQUVGLE9BQUMsUUFBUSxJQUNQLFFBQVEsSUFDTixNQUFNLE9BQ0osV0FBVyxRQUFRLFNBQVMsR0FBRyxRQUFRLEdBQUcsZUFBZSxXQUFXLG9CQUNyRSxDQUNGOzs7SUFLUCxNQUFNLGtCQUFrQixJQUFJLEtBQThCO0lBQzFELE1BQU1DLGFBQXVCLEVBQUU7QUFFL0IsU0FBSyxNQUFNLGFBQWEsZ0JBQWdCO0FBRXRDLFNBQUksS0FBSyxnQkFBZ0IsSUFBSSxVQUFVLENBQUU7S0FFekMsTUFBTSxVQUFVLFNBQVMsTUFBTSxNQUFNLEVBQUUsY0FBYyxVQUFVO0FBQy9ELFNBQUksQ0FBQyxRQUFTO0tBRWQsTUFBTSxTQUFTLGNBQWMsSUFBSSxRQUFRLFNBQVM7S0FDbEQsTUFBTSxZQUFZLE9BQU87QUFFekIsU0FBSSxDQUFDLGdCQUFnQixJQUFJLFVBQVUsRUFBRTtBQUNuQyxzQkFBZ0IsSUFBSSxXQUFXLEVBQUUsQ0FBQztBQUNsQyxpQkFBVyxLQUFLLFVBQVU7O0FBRTVCLHFCQUFnQixJQUFJLFVBQVUsRUFBRSxLQUFLLFFBQVE7O0FBRy9DLFVBQU0sR0FBRyxZQUFZLE9BQU8sUUFBUTtLQUNsQyxNQUFNLHFCQUFxQixJQUFJLEtBQTJDO0FBRzFFLFVBQUssTUFBTSxhQUFhLFlBQVk7TUFDbEMsTUFBTSxnQkFBZ0IsZ0JBQWdCLElBQUksVUFBVSxJQUFJLEVBQUU7TUFDMUQsTUFBTSxTQUFTLEtBQUsscUJBQXFCLGNBQWM7QUFFdkQsV0FBSyxNQUFNLGlCQUFpQixRQUFRO0FBRWxDLFlBQUssTUFBTSxXQUFXLGVBQWU7QUFDbkMsYUFBSyxnQkFBZ0IsU0FBUyxtQkFBbUI7QUFDakQsU0FBQyxRQUFRLElBQ1AsUUFBUSxJQUNOLE1BQU0sS0FDSixjQUFjLFFBQVEsU0FBUyxHQUFHLFFBQVEsS0FBSyxRQUFRLFdBQVcsZ0JBQWdCLEtBQ25GLENBQ0Y7O09BSUwsTUFBTSxRQUFRLEtBQUssUUFBUSxTQUFTLFVBQVU7T0FDOUMsTUFBTSxRQUFRLE1BQU0sS0FBSyxLQUFLLFFBQVEsSUFBSSxLQUFlO0FBRXpELFFBQUMsUUFBUSxJQUNQLFFBQVEsSUFDTixNQUFNLEtBQ0osYUFBYSxVQUFVLFFBQVEsTUFBTSxPQUFPLGVBQWUsT0FBTyxRQUFRLGNBQWMsR0FBRyxFQUFFLEdBQUcsT0FBTyxPQUFPLEdBQy9HLENBQ0Y7T0FDSCxNQUFNLE1BQU8sTUFBTSxLQUFLLFFBQVEsT0FDOUIsS0FDQSxVQUNEO0FBSUQsV0FBSSxNQUFNLFNBQVMsS0FBSyxNQUFNLFdBQVcsSUFBSSxRQUFRO1FBQ25ELE1BQU0sY0FDSixtQkFBbUIsSUFBSSxVQUFVLElBQUksSUFBSSxLQUE4QjtBQUN6RSxhQUFLLElBQUksSUFBSSxHQUFHLElBQUksTUFBTSxRQUFRLEtBQUs7QUFDckMscUJBQVksSUFBSSxNQUFNLElBQUksSUFBSSxHQUFHOztBQUVuQywyQkFBbUIsSUFBSSxXQUFXLFlBQVk7a0JBQ3JDLE1BQU0sV0FBVyxJQUFJLFFBQVE7QUFDdEMsZ0JBQVEsS0FDTixNQUFNLE9BQ0osd0JBQXdCLE1BQU0sT0FBTyxpQkFBaUIsSUFBSSxPQUFPLFFBQVEsWUFDMUUsQ0FDRjs7OztBQU1QLFdBQU0sS0FBSywyQkFBMkIsS0FBSyxVQUFVLG1CQUFtQjtBQUt4RSxNQUFDLFFBQVEsSUFBSSxRQUFRLElBQUksTUFBTSxLQUFLLHlCQUF5QixDQUFDO0FBQzlELFVBQUssTUFBTSxhQUFhLFlBQVk7QUFDbEMsVUFBSTtPQUVGLE1BQU0sU0FBUyxjQUFjLGdCQUFnQixDQUFDLE1BQzNDLE1BQU0sRUFBRSxVQUFVLGFBQWEsRUFBRSxHQUFHLGFBQWEsS0FBSyxVQUN4RDtBQUVELFdBQUksUUFBUTtRQUNWLE1BQU0sU0FBUyxPQUFPLE1BQU0sTUFBTSxNQUFNLEVBQUUsU0FBUyxLQUFLO1FBQ3hELE1BQU0sU0FBUyxRQUFRO1FBR3ZCLE1BQU0sZUFDSixXQUFXLGFBQ1gsV0FBVyxnQkFDWCxRQUFRLE1BQU0sb0JBQW9CO0FBRXBDLFlBQUksQ0FBQyxjQUFjO0FBQ2pCLFVBQUMsUUFBUSxJQUNQLFFBQVEsSUFDTixNQUFNLEtBQ0osOEJBQThCLFVBQVUsYUFBYSxVQUFVLFVBQVUsR0FDMUUsQ0FDRjtBQUNIOzs7T0FLSixNQUFNLFVBQVUsY0FBYyxnQkFBZ0IsQ0FBQyxNQUM1QyxNQUFNLEVBQUUsVUFBVSxhQUFhLEVBQUUsR0FBRyxhQUFhLEtBQUssVUFDeEQ7T0FDRCxNQUFNLFVBQVUsU0FBUyxNQUFNLE1BQU0sTUFBTSxFQUFFLFNBQVMsS0FBSyxFQUFFO09BQzdELE1BQU0sY0FDSixZQUFZLFdBQ1IsTUFBTSxJQUNILElBQUksMENBQTBDLFVBQVUsR0FBRyxDQUMzRCxNQUFNLE1BQU0sRUFBRSxLQUFLLEdBQUcsR0FDekIsTUFBTSxJQUFJLFVBQVUsQ0FBQyxJQUFJLGVBQWUsQ0FBQyxPQUFPO09BQ3RELE1BQU0sUUFBUSxhQUFhO0FBRTNCLFdBQUksVUFBVSxRQUFRLFVBQVUsV0FBVztBQUV6QyxjQUFNLElBQUksSUFBSSxxREFBcUQsQ0FDakUsV0FDQSxNQUNELENBQUM7QUFDRixTQUFDLFFBQVEsSUFBSSxRQUFRLElBQUksTUFBTSxNQUFNLHNCQUFzQixVQUFVLElBQUksUUFBUSxDQUFDOztlQUU3RSxNQUFNO0FBRWIsUUFBQyxRQUFRLElBQUksUUFBUSxJQUFJLE1BQU0sS0FBSyw4QkFBOEIsWUFBWSxDQUFDOzs7QUFLbkYsVUFBSyxNQUFNLFdBQVcsVUFBVTtNQUM5QixNQUFNLFNBQVMsY0FBYyxJQUFJLFFBQVEsU0FBUztNQUdsRCxNQUFNLFVBQVUsS0FBSyxnQkFBZ0IsSUFBSSxRQUFRLFVBQVU7QUFDM0QsVUFBSSxTQUFTO0FBQ1gsZUFBUSxLQUFLO1FBQ1gsVUFBVSxRQUFRO1FBQ2xCLE1BQU0sTUFBTSxJQUFJLE9BQU8sTUFBTSxDQUFDLE1BQU0sTUFBTSxRQUFRLFdBQVcsQ0FBQyxPQUFPO1FBQ3RFLENBQUM7QUFDRjs7TUFHRixNQUFNLE1BQU0sS0FBSyxjQUFjLElBQUksUUFBUSxVQUFVO0FBQ3JELFVBQUksS0FBSztPQUNQLE1BQU0sV0FBVyxtQkFBbUIsSUFBSSxPQUFPLE1BQU07T0FDckQsTUFBTSxhQUFhLFVBQVUsSUFBSSxJQUFJLEtBQUs7QUFFMUMsV0FBSSxlQUFlLFdBQVc7QUFDNUIsZ0JBQVEsS0FBSztTQUNYLFVBQVUsUUFBUTtTQUNsQixNQUFNLE1BQU0sSUFBSSxPQUFPLE1BQU0sQ0FBQyxNQUFNLE1BQU0sV0FBVyxDQUFDLE9BQU87U0FDOUQsQ0FBQztBQUVGLFNBQUMsUUFBUSxJQUNQLFFBQVEsSUFDTixNQUFNLE1BQU0saUJBQWlCLE9BQU8sTUFBTSxLQUFLLFFBQVEsR0FBRyxPQUFPLGFBQWEsQ0FDL0U7Ozs7TUFJVDthQUNNO0FBQ1IsVUFBTSxHQUFHLFNBQVM7O0FBR3BCLFVBQU8sT0FBTyxVQUFVLE1BQU0sR0FBRyxFQUFFLFNBQVMsR0FBRyxFQUFFLEtBQUssS0FBSzs7Ozs7O0VBTzdELEFBQVEsZ0JBQ04sU0FDQSxvQkFDTztHQUNQLE1BQU0sU0FBUyxjQUFjLElBQUksUUFBUSxTQUFTO0dBQ2xELE1BQU1DLE1BQStCLEVBQUU7R0FHdkMsTUFBTSxpQkFBaUIsUUFBUSxVQUFVLFFBQVE7R0FDakQsTUFBTSxpQkFBaUIsUUFBUSxZQUFZO0FBRTNDLFFBQUssTUFBTSxDQUFDLFVBQVUsV0FBVyxPQUFPLFFBQVEsUUFBUSxRQUFRLEVBQUU7SUFDaEUsTUFBTSxPQUFPLE9BQU87QUFFcEIsUUFBSSxjQUFjLEtBQUssRUFBRTtBQUN2Qjs7QUFJRixRQUFJLGVBQWUsUUFBUSxLQUFLLFdBQVc7QUFDekM7O0FBSUYsUUFBSSxhQUFhLE1BQU07S0FDckIsTUFBTSxTQUFTLE9BQU8sTUFBTSxNQUFNLE1BQU0sRUFBRSxTQUFTLEtBQUs7S0FFeEQsTUFBTSxlQUNKLENBQUMsT0FBTyxhQUNQLFFBQVEsU0FBUyxhQUNoQixRQUFRLFNBQVMsZ0JBQ2pCLFFBQVEsTUFBTSxvQkFBb0I7QUFFdEMsU0FBSSxrQkFBa0IsZ0JBQWdCO0FBRXBDLFVBQUksWUFBWSxlQUFlLFFBQVEsV0FBVztnQkFDekMsQ0FBQyxjQUFjO0FBRXhCLFVBQUksWUFBWSxPQUFPOztBQUd6Qjs7QUFHRixRQUFJLGVBQWUsS0FBSyxFQUFFO0FBQ3hCLFNBQ0UsMkJBQTJCLEtBQUssSUFDL0IsdUJBQXVCLEtBQUssSUFBSSxLQUFLLGVBQ3RDO01BQ0EsTUFBTSxZQUFZLE9BQU87QUFDekIsVUFBSSxjQUFjLFFBQVEsY0FBYyxXQUFXO09BQ2pELE1BQU0sbUJBQW1CLEdBQUcsS0FBSyxLQUFLLEdBQUc7T0FHekMsTUFBTSxvQkFBb0IsS0FBSyxnQkFBZ0IsSUFBSSxpQkFBaUIsRUFBRTtBQUN0RSxXQUFJLHNCQUFzQixXQUFXO0FBRW5DLFlBQUksR0FBRyxTQUFTLFFBQVE7Y0FDbkI7UUFDTCxNQUFNLGFBQWEsS0FBSyxjQUFjLElBQUksaUJBQWlCO0FBQzNELFlBQUksWUFBWTtTQUVkLE1BQU0sZ0JBQWdCLGNBQWMsSUFBSSxLQUFLLEtBQUs7U0FDbEQsTUFBTSxxQkFBcUIsb0JBQW9CLElBQUksY0FBYyxNQUFNO1NBQ3ZFLE1BQU0sV0FBVyxvQkFBb0IsSUFBSSxXQUFXLEtBQUs7QUFFekQsYUFBSSxhQUFhLFdBQVc7QUFFMUIsY0FBSSxHQUFHLFNBQVMsUUFBUTtnQkFDbkI7QUFFTCxjQUFJLEdBQUcsU0FBUyxRQUFROztlQUVyQjtBQUVMLGFBQUksR0FBRyxTQUFTLFFBQVE7OzthQUd2QjtBQUNMLFdBQUksR0FBRyxTQUFTLFFBQVE7OztXQUl2QjtBQUVMLFNBQUksWUFBWSxLQUFLLG1CQUFtQixNQUFvQixPQUFPLE1BQU07OztBQUk3RSxJQUFDLFFBQVEsSUFDUCxRQUFRLElBQUksTUFBTSxLQUFLLGVBQWUsT0FBTyxNQUFNLEtBQUssUUFBUSxLQUFLLE9BQU8sTUFBTSxLQUFLLEdBQUcsQ0FBQztHQUM3RixNQUFNLE1BQU0sS0FBSyxRQUFRLFNBQVMsT0FBTyxPQUFPLElBQUk7QUFDcEQsUUFBSyxjQUFjLElBQUksUUFBUSxXQUFXLElBQUk7QUFFOUMsVUFBTzs7Ozs7RUFNVCxBQUFRLG1CQUFtQixNQUFrQixPQUF5QjtBQUNwRSxPQUFJLFVBQVUsUUFBUSxVQUFVLFdBQVc7QUFDekMsV0FBTzs7QUFHVCxXQUFRLEtBQUssTUFBYjtJQUNFLEtBQUssT0FFSCxRQUFPO0lBRVQsS0FBSztBQUNILFNBQUksT0FBTyxVQUFVLFlBQVksT0FBTyxVQUFVLFVBQVU7QUFDMUQsYUFBTyxJQUFJLEtBQUssTUFBTTs7QUFFeEIsWUFBTztJQUVULFFBQ0UsUUFBTzs7O0VBSWIsTUFBYywyQkFDWixLQUNBLFVBQ0Esb0JBQ2U7QUFDZixRQUFLLE1BQU0sV0FBVyxVQUFVO0lBQzlCLE1BQU0sU0FBUyxjQUFjLElBQUksUUFBUSxTQUFTO0lBQ2xELE1BQU0sWUFBWSxLQUFLLGNBQWMsSUFBSSxRQUFRLFVBQVU7QUFFM0QsUUFBSSxDQUFDLFVBQVc7SUFFaEIsTUFBTSxpQkFBaUIsbUJBQW1CLElBQUksT0FBTyxNQUFNO0lBQzNELE1BQU0sV0FBVyxnQkFBZ0IsSUFBSSxVQUFVLEtBQUs7QUFFcEQsUUFBSSxhQUFhLFVBQVc7QUFFNUIsU0FBSyxNQUFNLEdBQUcsV0FBVyxPQUFPLFFBQVEsUUFBUSxRQUFRLEVBQUU7S0FDeEQsTUFBTSxPQUFPLE9BQU87QUFFcEIsU0FBSSx5QkFBeUIsS0FBSyxJQUFJLE1BQU0sUUFBUSxPQUFPLE1BQU0sRUFBRTtNQUVqRSxNQUFNLGNBQWMsY0FBYyxJQUFJLEtBQUssS0FBSztBQUNoRCxVQUFJLENBQUMsS0FBSyxRQUFRLFNBQVMsWUFBWSxNQUFNLENBQUU7TUFFL0MsTUFBTSxhQUFhLE9BQU87QUFDMUIsVUFBSSxXQUFXLFdBQVcsRUFBRztNQUU3QixNQUFNLFlBQVksS0FBSztNQUN2QixNQUFNLGdCQUFnQixjQUFjLElBQUksS0FBSyxLQUFLO01BRWxELE1BQU0sZUFBZSxHQUFHLFdBQVcsWUFBWSxPQUFPLE1BQU0sQ0FBQztNQUM3RCxNQUFNLGVBQWUsR0FBRyxXQUFXLFlBQVksY0FBYyxNQUFNLENBQUM7QUFFcEUsV0FBSyxNQUFNLGFBQWEsWUFBWTtPQUNsQyxNQUFNLG1CQUFtQixHQUFHLEtBQUssS0FBSyxHQUFHO09BQ3pDLE1BQU0sYUFBYSxLQUFLLGNBQWMsSUFBSSxpQkFBaUI7T0FFM0QsSUFBSUM7QUFFSixXQUFJLFlBQVk7UUFDZCxNQUFNLGtCQUFrQixtQkFBbUIsSUFBSSxjQUFjLE1BQU07UUFDbkUsTUFBTSxhQUFhLGlCQUFpQixJQUFJLFdBQVcsS0FBSztBQUV4RCxZQUFJLGVBQWUsV0FBVztBQUM1QixpQkFBUSxLQUNOLG1CQUFtQixpQkFBaUIscUNBQ3JDO0FBQ0Q7O0FBRUYsbUJBQVc7Y0FDTjtBQUNMLG1CQUFXOztPQUliLE1BQU0sQ0FBQyxTQUFTLE1BQU0sSUFBSSxVQUFVLENBQ2pDLE1BQU07U0FDSixlQUFlO1NBQ2YsZUFBZTtRQUNqQixDQUFDLENBQ0QsTUFBTSxFQUFFO0FBRVgsV0FBSSxDQUFDLE9BQU87QUFDVixjQUFNLElBQUksVUFBVSxDQUFDLE9BQU87VUFDekIsZUFBZTtVQUNmLGVBQWU7U0FDakIsQ0FBQztBQUVGLFNBQUMsUUFBUSxJQUNQLFFBQVEsSUFDTixNQUFNLE1BQ0osaUJBQWlCLFVBQVUsSUFBSSxPQUFPLE1BQU0sR0FBRyxTQUFTLE1BQU0sY0FBYyxNQUFNLEdBQUcsU0FBUyxHQUMvRixDQUNGOzs7Ozs7Ozs7Ozs7Ozs7OztFQWtCZixBQUFRLHFCQUFxQixVQUE4QztBQUN6RSxPQUFJLFNBQVMsV0FBVyxHQUFHO0FBQ3pCLFdBQU8sRUFBRTs7R0FHWCxNQUFNLFNBQVMsY0FBYyxJQUFJLFNBQVMsR0FBRyxTQUFTO0dBR3RELE1BQU0sZUFBZSxPQUFPLE1BQU0sUUFDL0IsTUFDQyxlQUFlLEVBQUUsS0FDaEIsMkJBQTJCLEVBQUUsSUFBSyx1QkFBdUIsRUFBRSxJQUFJLEVBQUUsa0JBQ2xFLEVBQUUsU0FBUyxPQUFPLEdBQ3JCO0FBRUQsT0FBSSxhQUFhLFdBQVcsR0FBRztBQUU3QixXQUFPLENBQUMsU0FBUzs7R0FJbkIsTUFBTUMsU0FBNEIsRUFBRTtHQUNwQyxNQUFNLFlBQVksSUFBSSxJQUFJLFNBQVMsS0FBSyxNQUFNLEVBQUUsVUFBVSxDQUFDO0dBQzNELE1BQU0sWUFBWSxJQUFJLEtBQWE7QUFFbkMsVUFBTyxVQUFVLE9BQU8sR0FBRztJQUN6QixNQUFNQyxlQUFnQyxFQUFFO0FBRXhDLFNBQUssTUFBTSxXQUFXLFVBQVU7QUFDOUIsU0FBSSxDQUFDLFVBQVUsSUFBSSxRQUFRLFVBQVUsQ0FBRTtLQUd2QyxNQUFNLGFBQWEsYUFBYSxPQUFPLFNBQVM7TUFDOUMsTUFBTSxRQUFRLFFBQVEsUUFBUSxLQUFLLE9BQU87QUFDMUMsVUFBSSxVQUFVLFFBQVEsVUFBVSxVQUFXLFFBQU87TUFDbEQsTUFBTSxlQUFlLEdBQUcsS0FBSyxLQUFLLEdBQUc7QUFFckMsYUFBTyxVQUFVLElBQUksYUFBYSxJQUFJLENBQUMsVUFBVSxJQUFJLGFBQWE7T0FDbEU7QUFFRixTQUFJLFlBQVk7QUFDZCxtQkFBYSxLQUFLLFFBQVE7OztBQUk5QixRQUFJLGFBQWEsV0FBVyxHQUFHO0tBQzdCLE1BQU0sZUFBZSxNQUFNLEtBQUssVUFBVSxDQUFDLEtBQUssS0FBSztBQUNyRCxXQUFNLElBQUksTUFDUix1Q0FBdUMsT0FBTyxNQUFNLHdCQUF3QixlQUM3RTs7QUFHSCxTQUFLLE1BQU0sV0FBVyxjQUFjO0FBQ2xDLGVBQVUsT0FBTyxRQUFRLFVBQVU7QUFDbkMsZUFBVSxJQUFJLFFBQVEsVUFBVTs7QUFHbEMsV0FBTyxLQUFLLGFBQWE7O0FBRzNCLFVBQU87O0VBR1QsTUFBYyxxQkFBcUIsSUFBVSxRQUFnQixTQUF3QjtHQUNuRixNQUFNLGlCQUFpQixPQUFPLFNBQVMsUUFBUSxNQUFNLEVBQUUsU0FBUyxTQUFTLElBQUksRUFBRTtHQUUvRSxNQUFNLGdCQUFnQixlQUFlLFFBQVEsVUFDM0MsTUFBTSxRQUFRLE9BQU8sV0FBVyxDQUFDLE9BQU8sS0FBSyxXQUFXLEdBQUcsT0FBTyxNQUFNLElBQUksQ0FBQyxDQUM5RTtBQUNELE9BQUksY0FBYyxXQUFXLEdBQUc7QUFDOUIsV0FBTzs7R0FHVCxJQUFJLGNBQWMsR0FBRyxPQUFPLE1BQU07R0FDbEMsSUFBSSxlQUFlO0FBRW5CLFFBQUssTUFBTSxTQUFTLGVBQWU7SUFFakMsTUFBTSxlQUFlLE1BQU0sUUFBUSxNQUFNLFdBQVc7S0FDbEQsTUFBTSxRQUFRLE9BQU8sS0FBSyxRQUFRLFFBQVEsR0FBRztBQUM3QyxZQUFPLFFBQVEsUUFBUSxRQUFRLFVBQVU7TUFDekM7QUFDRixRQUFJLGNBQWM7QUFDaEI7O0FBR0Ysa0JBQWMsWUFBWSxTQUFTLE9BQU87QUFDeEMsVUFBSyxNQUFNLFVBQVUsTUFBTSxTQUFTO01BQ2xDLE1BQU0sUUFBUSxPQUFPLEtBQUssUUFBUSxRQUFRLEdBQUc7QUFFN0MsVUFBSSxNQUFNLFFBQVEsUUFBUSxRQUFRLFFBQVEsTUFBTSxFQUFFO0FBQ2hELFVBQUcsUUFBUSxPQUFPLE1BQU0sUUFBUSxRQUFRLE9BQU8sTUFBTTthQUNoRDtBQUNMLFVBQUcsU0FBUyxPQUFPLE1BQU0sUUFBUSxRQUFRLFFBQVEsTUFBTTs7O01BRzNEO0FBQ0YsbUJBQWU7O0FBR2pCLE9BQUksQ0FBQyxjQUFjO0FBQ2pCLFdBQU87O0dBR1QsTUFBTSxDQUFDLGVBQWUsTUFBTTtBQUM1QixVQUFPOztFQUdULE1BQWMsd0JBQ1osSUFDQSxRQUNBLFNBQ0EsU0FDQTtBQUNBLE9BQUksUUFBUSxXQUFXLEdBQUc7QUFDeEIsV0FBTzs7R0FHVCxNQUFNQyxjQUF1QyxFQUFFO0FBRS9DLFFBQUssTUFBTSxVQUFVLFNBQVM7SUFFNUIsTUFBTSxPQUFPLE9BQU8sTUFBTSxNQUFNLE1BQU0sRUFBRSxTQUFTLE9BQU87SUFDeEQsTUFBTSxXQUFXLFFBQVEsZUFBZSxLQUFLLEdBQUcsR0FBRyxPQUFPLE9BQU87SUFDakUsTUFBTSxRQUFRLFFBQVEsUUFBUSxTQUFTO0FBR3ZDLFFBQUksVUFBVSxRQUFRLFVBQVUsV0FBVztBQUN6QyxZQUFPOztBQUdULGdCQUFZLFlBQVk7O0dBRzFCLE1BQU0sQ0FBQyxTQUFTLE1BQU0sR0FBRyxPQUFPLE1BQU0sQ0FBQyxNQUFNLFlBQVksQ0FBQyxNQUFNLEVBQUU7QUFDbEUsVUFBTzs7RUFHVCxNQUFNLGlCQUFpQixNQUFjO0dBQ25DLE1BQU0sT0FBTyxHQUFHLE9BQU8sWUFBWTtHQUNuQyxNQUFNLFVBQVUsYUFBYSxLQUFLLENBQUMsVUFBVTtHQUU3QyxNQUFNLHFCQUFxQixRQUFRLFFBQVEsMEJBQTBCO0dBQ3JFLE1BQU0sbUJBQW1CLFFBQVEsUUFBUSxNQUFNLG1CQUFtQjtBQUVsRSxPQUFJLHVCQUF1QixDQUFDLEtBQUsscUJBQXFCLENBQUMsR0FBRztJQUN4RCxNQUFNLGFBQWEsR0FBRyxRQUFRLE1BQU0sR0FBRyxpQkFBaUIsQ0FBQyxJQUFJLEtBQUssSUFBSSxRQUFRLE1BQU0saUJBQWlCO0FBRXJHLGtCQUFjLE1BQU0sV0FBVztVQUMxQjtBQUNMLFVBQU0sSUFBSSxNQUFNLDZDQUE2Qzs7OztDQUt0RCxpQkFBaUIsSUFBSSxxQkFBcUIifQ==