@decaf-ts/for-nest 0.11.1 → 0.11.2-refactor.2

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 (370) hide show
  1. package/README.md +1 -1
  2. package/dist/for-nest.cjs +2654 -2
  3. package/dist/for-nest.js +2756 -2
  4. package/lib/cjs/auth/AuthInterceptor.cjs +90 -0
  5. package/lib/cjs/auth/DecafAuthHandler.cjs +37 -0
  6. package/lib/cjs/auth/DecafAuthModule.cjs +56 -0
  7. package/lib/cjs/auth/constants.cjs +10 -0
  8. package/lib/cjs/auth/decorators.cjs +28 -0
  9. package/lib/cjs/auth/index.cjs +23 -0
  10. package/lib/cjs/bin/cli.cjs +2 -2
  11. package/lib/cjs/cli-module.cjs +64 -7
  12. package/lib/cjs/constants.cjs +5 -6
  13. package/lib/cjs/controllers.cjs +13 -109
  14. package/lib/cjs/core-module.cjs +8 -8
  15. package/lib/cjs/decaf-model/DecafModelModule.cjs +23 -4
  16. package/lib/cjs/decaf-model/FromModelController.cjs +751 -789
  17. package/lib/cjs/decaf-model/decorators/ApiOperationFromModel.cjs +53 -16
  18. package/lib/cjs/decaf-model/decorators/ApiParamsFromModel.cjs +2 -2
  19. package/lib/cjs/decaf-model/decorators/DecafBody.cjs +2 -2
  20. package/lib/cjs/decaf-model/decorators/DecafParams.cjs +2 -2
  21. package/lib/cjs/decaf-model/decorators/controller-config.cjs +29 -0
  22. package/lib/cjs/decaf-model/decorators/decorators.cjs +5 -45
  23. package/lib/cjs/decaf-model/decorators/expose.cjs +13 -0
  24. package/lib/cjs/decaf-model/decorators/index.cjs +3 -2
  25. package/lib/cjs/decaf-model/decorators/types.cjs +2 -2
  26. package/lib/cjs/decaf-model/decorators/utils.cjs +2 -2
  27. package/lib/cjs/decaf-model/index.cjs +28 -19
  28. package/lib/cjs/decaf-model/types.cjs +2 -2
  29. package/lib/cjs/decaf-model/utils.cjs +2 -2
  30. package/lib/cjs/decoration.cjs +2 -2
  31. package/lib/cjs/decorators.cjs +85 -0
  32. package/lib/cjs/events-module/DecafStreamModule.cjs +2 -2
  33. package/lib/cjs/events-module/EventsController.cjs +2 -2
  34. package/lib/cjs/events-module/constant.cjs +2 -2
  35. package/lib/cjs/events-module/index.cjs +2 -2
  36. package/lib/cjs/events-module/utils.cjs +2 -2
  37. package/lib/cjs/factory/NestBootstraper.cjs +2 -2
  38. package/lib/cjs/factory/errors/cors.cjs +2 -2
  39. package/lib/cjs/factory/errors/index.cjs +2 -2
  40. package/lib/cjs/factory/errors/throttling.cjs +2 -2
  41. package/lib/cjs/factory/exceptions/DecafErrorFilter.cjs +2 -2
  42. package/lib/cjs/factory/exceptions/decorators.cjs +2 -2
  43. package/lib/cjs/factory/exceptions/index.cjs +2 -2
  44. package/lib/cjs/factory/index.cjs +2 -2
  45. package/lib/cjs/factory/openapi/DtoBuilder.cjs +37 -6
  46. package/lib/cjs/factory/openapi/SwaggerBuilder.cjs +2 -2
  47. package/lib/cjs/factory/openapi/SwaggerCustomUI.cjs +2 -2
  48. package/lib/cjs/factory/openapi/constants.cjs +2 -2
  49. package/lib/cjs/factory/openapi/index.cjs +2 -2
  50. package/lib/cjs/index.cjs +8 -7
  51. package/lib/cjs/interceptors/DecafRequestHandlerInterceptor.cjs +25 -21
  52. package/lib/cjs/interceptors/index.cjs +3 -4
  53. package/lib/cjs/module.cjs +5 -2
  54. package/lib/cjs/overrides/Adapter.cjs +2 -2
  55. package/lib/cjs/overrides/ModelBuilderExtensions.cjs +2 -2
  56. package/lib/cjs/overrides/constants.cjs +2 -2
  57. package/lib/cjs/overrides/decoration.cjs +2 -2
  58. package/lib/cjs/overrides/helpers.cjs +2 -2
  59. package/lib/cjs/overrides/index.cjs +3 -2
  60. package/lib/cjs/overrides/logging.cjs +5 -0
  61. package/lib/cjs/overrides/overrides.cjs +2 -2
  62. package/lib/cjs/request/DecafAuthHandler.cjs +5 -39
  63. package/lib/cjs/request/DecafHandlerExecutor.cjs +2 -2
  64. package/lib/cjs/request/DecafRequestContext.cjs +11 -20
  65. package/lib/cjs/request/DecafResponseInterceptor.cjs +3 -3
  66. package/lib/cjs/request/index.cjs +2 -2
  67. package/lib/cjs/swagger-types.cjs +2 -2
  68. package/lib/cjs/types.cjs +2 -2
  69. package/lib/cjs/utils.cjs +2 -2
  70. package/lib/cjs/webhooks/DecafWebhookModule.cjs +11 -12
  71. package/lib/cjs/webhooks/controllers.cjs +10 -10
  72. package/lib/cjs/webhooks/index.cjs +2 -15
  73. package/lib/cjs/webhooks/types.cjs +2 -2
  74. package/lib/esm/auth/AuthInterceptor.js +81 -0
  75. package/lib/esm/auth/DecafAuthHandler.js +31 -0
  76. package/lib/esm/auth/DecafAuthModule.js +48 -0
  77. package/lib/esm/auth/constants.js +6 -0
  78. package/lib/esm/auth/decorators.js +22 -0
  79. package/lib/esm/auth/index.js +6 -0
  80. package/lib/esm/bin/cli.js +1 -1
  81. package/lib/esm/cli-module.js +63 -6
  82. package/lib/esm/constants.js +3 -4
  83. package/lib/esm/controllers.js +13 -109
  84. package/lib/esm/core-module.js +7 -7
  85. package/lib/esm/decaf-model/DecafModelModule.js +22 -3
  86. package/lib/esm/decaf-model/FromModelController.js +753 -791
  87. package/lib/esm/decaf-model/decorators/ApiOperationFromModel.js +53 -15
  88. package/lib/esm/decaf-model/decorators/ApiParamsFromModel.js +1 -1
  89. package/lib/esm/decaf-model/decorators/DecafBody.js +1 -1
  90. package/lib/esm/decaf-model/decorators/DecafParams.js +1 -1
  91. package/lib/esm/decaf-model/decorators/controller-config.js +25 -0
  92. package/lib/esm/decaf-model/decorators/decorators.js +2 -43
  93. package/lib/esm/decaf-model/decorators/expose.js +9 -0
  94. package/lib/esm/decaf-model/decorators/index.js +2 -1
  95. package/lib/esm/decaf-model/decorators/types.js +1 -1
  96. package/lib/esm/decaf-model/decorators/utils.js +1 -1
  97. package/lib/esm/decaf-model/index.js +11 -4
  98. package/lib/esm/decaf-model/types.js +1 -1
  99. package/lib/esm/decaf-model/utils.js +1 -1
  100. package/lib/esm/decoration.js +1 -1
  101. package/lib/esm/decorators.js +79 -0
  102. package/lib/esm/events-module/DecafStreamModule.js +1 -1
  103. package/lib/esm/events-module/EventsController.js +1 -1
  104. package/lib/esm/events-module/constant.js +1 -1
  105. package/lib/esm/events-module/index.js +1 -1
  106. package/lib/esm/events-module/utils.js +1 -1
  107. package/lib/esm/factory/NestBootstraper.js +1 -1
  108. package/lib/esm/factory/errors/cors.js +1 -1
  109. package/lib/esm/factory/errors/index.js +1 -1
  110. package/lib/esm/factory/errors/throttling.js +1 -1
  111. package/lib/esm/factory/exceptions/DecafErrorFilter.js +1 -1
  112. package/lib/esm/factory/exceptions/decorators.js +1 -1
  113. package/lib/esm/factory/exceptions/index.js +1 -1
  114. package/lib/esm/factory/index.js +1 -1
  115. package/lib/esm/factory/openapi/DtoBuilder.js +36 -5
  116. package/lib/esm/factory/openapi/SwaggerBuilder.js +1 -1
  117. package/lib/esm/factory/openapi/SwaggerCustomUI.js +1 -1
  118. package/lib/esm/factory/openapi/constants.js +1 -1
  119. package/lib/esm/factory/openapi/index.js +1 -1
  120. package/lib/esm/index.js +7 -6
  121. package/lib/esm/interceptors/DecafRequestHandlerInterceptor.js +25 -21
  122. package/lib/esm/interceptors/index.js +2 -3
  123. package/lib/esm/module.js +4 -1
  124. package/lib/esm/overrides/Adapter.js +1 -1
  125. package/lib/esm/overrides/ModelBuilderExtensions.js +1 -1
  126. package/lib/esm/overrides/constants.js +1 -1
  127. package/lib/esm/overrides/decoration.js +1 -1
  128. package/lib/esm/overrides/helpers.js +1 -1
  129. package/lib/esm/overrides/index.js +2 -1
  130. package/lib/esm/overrides/logging.js +2 -0
  131. package/lib/esm/overrides/overrides.js +1 -1
  132. package/lib/esm/request/DecafAuthHandler.js +2 -36
  133. package/lib/esm/request/DecafHandlerExecutor.js +1 -1
  134. package/lib/esm/request/DecafRequestContext.js +10 -19
  135. package/lib/esm/request/DecafResponseInterceptor.js +2 -2
  136. package/lib/esm/request/index.js +1 -1
  137. package/lib/esm/swagger-types.js +1 -1
  138. package/lib/esm/types.js +1 -1
  139. package/lib/esm/utils.js +1 -1
  140. package/lib/esm/webhooks/DecafWebhookModule.js +5 -6
  141. package/lib/esm/webhooks/controllers.js +2 -2
  142. package/lib/esm/webhooks/index.js +1 -2
  143. package/lib/esm/webhooks/types.js +1 -1
  144. package/lib/types/{interceptors → auth}/AuthInterceptor.d.cts +4 -1
  145. package/lib/types/{interceptors → auth}/AuthInterceptor.d.mts +4 -1
  146. package/lib/types/auth/DecafAuthHandler.d.cts +22 -0
  147. package/lib/types/auth/DecafAuthHandler.d.mts +22 -0
  148. package/lib/types/auth/DecafAuthModule.d.cts +9 -0
  149. package/lib/types/auth/DecafAuthModule.d.mts +9 -0
  150. package/lib/types/auth/constants.d.cts +5 -0
  151. package/lib/types/auth/constants.d.mts +5 -0
  152. package/lib/types/auth/decorators.d.cts +4 -0
  153. package/lib/types/auth/decorators.d.mts +4 -0
  154. package/lib/types/auth/index.d.cts +6 -0
  155. package/lib/types/auth/index.d.mts +6 -0
  156. package/lib/types/cli-module.d.cts +1 -4
  157. package/lib/types/cli-module.d.mts +1 -4
  158. package/lib/types/constants.d.cts +2 -3
  159. package/lib/types/constants.d.mts +2 -3
  160. package/lib/types/controllers.d.cts +7 -12
  161. package/lib/types/controllers.d.mts +7 -12
  162. package/lib/types/decaf-model/DecafModelModule.d.cts +1 -0
  163. package/lib/types/decaf-model/DecafModelModule.d.mts +1 -0
  164. package/lib/types/decaf-model/FromModelController.d.cts +40 -52
  165. package/lib/types/decaf-model/FromModelController.d.mts +40 -52
  166. package/lib/types/decaf-model/decorators/ApiOperationFromModel.d.cts +0 -2
  167. package/lib/types/decaf-model/decorators/ApiOperationFromModel.d.mts +0 -2
  168. package/lib/types/decaf-model/decorators/controller-config.d.cts +18 -0
  169. package/lib/types/decaf-model/decorators/controller-config.d.mts +18 -0
  170. package/lib/types/decaf-model/decorators/decorators.d.cts +1 -29
  171. package/lib/types/decaf-model/decorators/decorators.d.mts +1 -29
  172. package/lib/types/decaf-model/decorators/expose.d.cts +1 -0
  173. package/lib/types/decaf-model/decorators/expose.d.mts +1 -0
  174. package/lib/types/decaf-model/decorators/index.d.cts +1 -0
  175. package/lib/types/decaf-model/decorators/index.d.mts +1 -0
  176. package/lib/types/decaf-model/decorators/types.d.cts +8 -21
  177. package/lib/types/decaf-model/decorators/types.d.mts +8 -21
  178. package/lib/types/decaf-model/index.d.cts +11 -3
  179. package/lib/types/decaf-model/index.d.mts +11 -3
  180. package/lib/types/decaf-model/utils.d.cts +2 -1
  181. package/lib/types/decaf-model/utils.d.mts +2 -1
  182. package/lib/types/decorators.d.cts +34 -0
  183. package/lib/types/decorators.d.mts +34 -0
  184. package/lib/types/factory/openapi/DtoBuilder.d.cts +7 -3
  185. package/lib/types/factory/openapi/DtoBuilder.d.mts +7 -3
  186. package/lib/types/index.d.cts +5 -5
  187. package/lib/types/index.d.mts +5 -5
  188. package/lib/types/interceptors/DecafRequestHandlerInterceptor.d.cts +1 -2
  189. package/lib/types/interceptors/DecafRequestHandlerInterceptor.d.mts +1 -2
  190. package/lib/types/interceptors/index.d.cts +1 -2
  191. package/lib/types/interceptors/index.d.mts +1 -2
  192. package/lib/types/overrides/Adapter.d.cts +1 -1
  193. package/lib/types/overrides/Adapter.d.mts +1 -1
  194. package/lib/types/overrides/index.d.cts +1 -0
  195. package/lib/types/overrides/index.d.mts +1 -0
  196. package/lib/types/overrides/logging.d.cts +1 -0
  197. package/lib/types/overrides/logging.d.mts +1 -0
  198. package/lib/types/request/DecafAuthHandler.d.cts +1 -10
  199. package/lib/types/request/DecafAuthHandler.d.mts +1 -10
  200. package/lib/types/request/DecafRequestContext.d.cts +4 -5
  201. package/lib/types/request/DecafRequestContext.d.mts +4 -5
  202. package/lib/types/types.d.cts +12 -24
  203. package/lib/types/types.d.mts +12 -24
  204. package/lib/types/webhooks/controllers.d.cts +1 -1
  205. package/lib/types/webhooks/controllers.d.mts +1 -1
  206. package/lib/types/webhooks/index.d.cts +0 -1
  207. package/lib/types/webhooks/index.d.mts +0 -1
  208. package/package.json +5 -13
  209. package/dist/for-nest.cjs.map +0 -1
  210. package/dist/for-nest.js.map +0 -1
  211. package/lib/cjs/bin/cli.cjs.map +0 -1
  212. package/lib/cjs/cli-module.cjs.map +0 -1
  213. package/lib/cjs/constants.cjs.map +0 -1
  214. package/lib/cjs/controllers.cjs.map +0 -1
  215. package/lib/cjs/core-module.cjs.map +0 -1
  216. package/lib/cjs/decaf-model/DecafModelModule.cjs.map +0 -1
  217. package/lib/cjs/decaf-model/FromModelController.cjs.map +0 -1
  218. package/lib/cjs/decaf-model/decorators/ApiOperationFromModel.cjs.map +0 -1
  219. package/lib/cjs/decaf-model/decorators/ApiParamsFromModel.cjs.map +0 -1
  220. package/lib/cjs/decaf-model/decorators/DecafBody.cjs.map +0 -1
  221. package/lib/cjs/decaf-model/decorators/DecafParams.cjs.map +0 -1
  222. package/lib/cjs/decaf-model/decorators/decorators.cjs.map +0 -1
  223. package/lib/cjs/decaf-model/decorators/index.cjs.map +0 -1
  224. package/lib/cjs/decaf-model/decorators/types.cjs.map +0 -1
  225. package/lib/cjs/decaf-model/decorators/utils.cjs.map +0 -1
  226. package/lib/cjs/decaf-model/index.cjs.map +0 -1
  227. package/lib/cjs/decaf-model/types.cjs.map +0 -1
  228. package/lib/cjs/decaf-model/utils.cjs.map +0 -1
  229. package/lib/cjs/decoration.cjs.map +0 -1
  230. package/lib/cjs/events-module/DecafStreamModule.cjs.map +0 -1
  231. package/lib/cjs/events-module/EventsController.cjs.map +0 -1
  232. package/lib/cjs/events-module/constant.cjs.map +0 -1
  233. package/lib/cjs/events-module/index.cjs.map +0 -1
  234. package/lib/cjs/events-module/utils.cjs.map +0 -1
  235. package/lib/cjs/factory/NestBootstraper.cjs.map +0 -1
  236. package/lib/cjs/factory/errors/cors.cjs.map +0 -1
  237. package/lib/cjs/factory/errors/index.cjs.map +0 -1
  238. package/lib/cjs/factory/errors/throttling.cjs.map +0 -1
  239. package/lib/cjs/factory/exceptions/DecafErrorFilter.cjs.map +0 -1
  240. package/lib/cjs/factory/exceptions/decorators.cjs.map +0 -1
  241. package/lib/cjs/factory/exceptions/index.cjs.map +0 -1
  242. package/lib/cjs/factory/index.cjs.map +0 -1
  243. package/lib/cjs/factory/openapi/DtoBuilder.cjs.map +0 -1
  244. package/lib/cjs/factory/openapi/SwaggerBuilder.cjs.map +0 -1
  245. package/lib/cjs/factory/openapi/SwaggerCustomUI.cjs.map +0 -1
  246. package/lib/cjs/factory/openapi/constants.cjs.map +0 -1
  247. package/lib/cjs/factory/openapi/index.cjs.map +0 -1
  248. package/lib/cjs/index.cjs.map +0 -1
  249. package/lib/cjs/interceptors/AuthInterceptor.cjs +0 -52
  250. package/lib/cjs/interceptors/AuthInterceptor.cjs.map +0 -1
  251. package/lib/cjs/interceptors/DecafRequestHandlerInterceptor.cjs.map +0 -1
  252. package/lib/cjs/interceptors/context.cjs +0 -18
  253. package/lib/cjs/interceptors/context.cjs.map +0 -1
  254. package/lib/cjs/interceptors/index.cjs.map +0 -1
  255. package/lib/cjs/migrations/index.cjs +0 -35
  256. package/lib/cjs/migrations/index.cjs.map +0 -1
  257. package/lib/cjs/migrations/migration-module.cjs +0 -60
  258. package/lib/cjs/migrations/migration-module.cjs.map +0 -1
  259. package/lib/cjs/migrations/migration-options.cjs +0 -4
  260. package/lib/cjs/migrations/migration-options.cjs.map +0 -1
  261. package/lib/cjs/module.cjs.map +0 -1
  262. package/lib/cjs/overrides/Adapter.cjs.map +0 -1
  263. package/lib/cjs/overrides/ModelBuilderExtensions.cjs.map +0 -1
  264. package/lib/cjs/overrides/constants.cjs.map +0 -1
  265. package/lib/cjs/overrides/decoration.cjs.map +0 -1
  266. package/lib/cjs/overrides/helpers.cjs.map +0 -1
  267. package/lib/cjs/overrides/index.cjs.map +0 -1
  268. package/lib/cjs/overrides/overrides.cjs.map +0 -1
  269. package/lib/cjs/ram/RamRequestTransformer.cjs +0 -26
  270. package/lib/cjs/ram/RamRequestTransformer.cjs.map +0 -1
  271. package/lib/cjs/ram/index.cjs +0 -19
  272. package/lib/cjs/ram/index.cjs.map +0 -1
  273. package/lib/cjs/request/DecafAuthHandler.cjs.map +0 -1
  274. package/lib/cjs/request/DecafHandlerExecutor.cjs.map +0 -1
  275. package/lib/cjs/request/DecafRequestContext.cjs.map +0 -1
  276. package/lib/cjs/request/DecafResponseInterceptor.cjs.map +0 -1
  277. package/lib/cjs/request/index.cjs.map +0 -1
  278. package/lib/cjs/swagger-types.cjs.map +0 -1
  279. package/lib/cjs/types.cjs.map +0 -1
  280. package/lib/cjs/utils.cjs.map +0 -1
  281. package/lib/cjs/webhooks/DecafWebhookModule.cjs.map +0 -1
  282. package/lib/cjs/webhooks/controllers.cjs.map +0 -1
  283. package/lib/cjs/webhooks/index.cjs.map +0 -1
  284. package/lib/cjs/webhooks/types.cjs.map +0 -1
  285. package/lib/esm/bin/cli.js.map +0 -1
  286. package/lib/esm/cli-module.js.map +0 -1
  287. package/lib/esm/constants.js.map +0 -1
  288. package/lib/esm/controllers.js.map +0 -1
  289. package/lib/esm/core-module.js.map +0 -1
  290. package/lib/esm/decaf-model/DecafModelModule.js.map +0 -1
  291. package/lib/esm/decaf-model/FromModelController.js.map +0 -1
  292. package/lib/esm/decaf-model/decorators/ApiOperationFromModel.js.map +0 -1
  293. package/lib/esm/decaf-model/decorators/ApiParamsFromModel.js.map +0 -1
  294. package/lib/esm/decaf-model/decorators/DecafBody.js.map +0 -1
  295. package/lib/esm/decaf-model/decorators/DecafParams.js.map +0 -1
  296. package/lib/esm/decaf-model/decorators/decorators.js.map +0 -1
  297. package/lib/esm/decaf-model/decorators/index.js.map +0 -1
  298. package/lib/esm/decaf-model/decorators/types.js.map +0 -1
  299. package/lib/esm/decaf-model/decorators/utils.js.map +0 -1
  300. package/lib/esm/decaf-model/index.js.map +0 -1
  301. package/lib/esm/decaf-model/types.js.map +0 -1
  302. package/lib/esm/decaf-model/utils.js.map +0 -1
  303. package/lib/esm/decoration.js.map +0 -1
  304. package/lib/esm/events-module/DecafStreamModule.js.map +0 -1
  305. package/lib/esm/events-module/EventsController.js.map +0 -1
  306. package/lib/esm/events-module/constant.js.map +0 -1
  307. package/lib/esm/events-module/index.js.map +0 -1
  308. package/lib/esm/events-module/utils.js.map +0 -1
  309. package/lib/esm/factory/NestBootstraper.js.map +0 -1
  310. package/lib/esm/factory/errors/cors.js.map +0 -1
  311. package/lib/esm/factory/errors/index.js.map +0 -1
  312. package/lib/esm/factory/errors/throttling.js.map +0 -1
  313. package/lib/esm/factory/exceptions/DecafErrorFilter.js.map +0 -1
  314. package/lib/esm/factory/exceptions/decorators.js.map +0 -1
  315. package/lib/esm/factory/exceptions/index.js.map +0 -1
  316. package/lib/esm/factory/index.js.map +0 -1
  317. package/lib/esm/factory/openapi/DtoBuilder.js.map +0 -1
  318. package/lib/esm/factory/openapi/SwaggerBuilder.js.map +0 -1
  319. package/lib/esm/factory/openapi/SwaggerCustomUI.js.map +0 -1
  320. package/lib/esm/factory/openapi/constants.js.map +0 -1
  321. package/lib/esm/factory/openapi/index.js.map +0 -1
  322. package/lib/esm/index.js.map +0 -1
  323. package/lib/esm/interceptors/AuthInterceptor.js +0 -43
  324. package/lib/esm/interceptors/AuthInterceptor.js.map +0 -1
  325. package/lib/esm/interceptors/DecafRequestHandlerInterceptor.js.map +0 -1
  326. package/lib/esm/interceptors/context.js +0 -12
  327. package/lib/esm/interceptors/context.js.map +0 -1
  328. package/lib/esm/interceptors/index.js.map +0 -1
  329. package/lib/esm/migrations/index.js +0 -18
  330. package/lib/esm/migrations/index.js.map +0 -1
  331. package/lib/esm/migrations/migration-module.js +0 -52
  332. package/lib/esm/migrations/migration-module.js.map +0 -1
  333. package/lib/esm/migrations/migration-options.js +0 -7
  334. package/lib/esm/migrations/migration-options.js.map +0 -1
  335. package/lib/esm/module.js.map +0 -1
  336. package/lib/esm/overrides/Adapter.js.map +0 -1
  337. package/lib/esm/overrides/ModelBuilderExtensions.js.map +0 -1
  338. package/lib/esm/overrides/constants.js.map +0 -1
  339. package/lib/esm/overrides/decoration.js.map +0 -1
  340. package/lib/esm/overrides/helpers.js.map +0 -1
  341. package/lib/esm/overrides/index.js.map +0 -1
  342. package/lib/esm/overrides/overrides.js.map +0 -1
  343. package/lib/esm/ram/RamRequestTransformer.js +0 -21
  344. package/lib/esm/ram/RamRequestTransformer.js.map +0 -1
  345. package/lib/esm/ram/index.js +0 -2
  346. package/lib/esm/ram/index.js.map +0 -1
  347. package/lib/esm/request/DecafAuthHandler.js.map +0 -1
  348. package/lib/esm/request/DecafHandlerExecutor.js.map +0 -1
  349. package/lib/esm/request/DecafRequestContext.js.map +0 -1
  350. package/lib/esm/request/DecafResponseInterceptor.js.map +0 -1
  351. package/lib/esm/request/index.js.map +0 -1
  352. package/lib/esm/swagger-types.js.map +0 -1
  353. package/lib/esm/types.js.map +0 -1
  354. package/lib/esm/utils.js.map +0 -1
  355. package/lib/esm/webhooks/DecafWebhookModule.js.map +0 -1
  356. package/lib/esm/webhooks/controllers.js.map +0 -1
  357. package/lib/esm/webhooks/index.js.map +0 -1
  358. package/lib/esm/webhooks/types.js.map +0 -1
  359. package/lib/types/interceptors/context.d.cts +0 -6
  360. package/lib/types/interceptors/context.d.mts +0 -6
  361. package/lib/types/migrations/index.d.cts +0 -17
  362. package/lib/types/migrations/index.d.mts +0 -17
  363. package/lib/types/migrations/migration-module.d.cts +0 -15
  364. package/lib/types/migrations/migration-module.d.mts +0 -15
  365. package/lib/types/migrations/migration-options.d.cts +0 -11
  366. package/lib/types/migrations/migration-options.d.mts +0 -11
  367. package/lib/types/ram/RamRequestTransformer.d.cts +0 -5
  368. package/lib/types/ram/RamRequestTransformer.d.mts +0 -5
  369. package/lib/types/ram/index.d.cts +0 -1
  370. package/lib/types/ram/index.d.mts +0 -1
package/dist/for-nest.cjs CHANGED
@@ -1,2 +1,2654 @@
1
- var e,t;e=this,t=function(e,t,r,o,a,i,s,n,c,p,d,l,u,h,m,f,y,g,_){"use strict";function A(e){var t=Object.create(null);return e&&Object.keys(e).forEach(r=>{if("default"!==r){var o=Object.getOwnPropertyDescriptor(e,r);Object.defineProperty(t,r,o.get?o:{enumerable:!0,get:()=>e[r]})}}),t.default=e,Object.freeze(t)}var b=A(m);const P="swagger",E={API_OPERATION:P+"/apiOperation",API_RESPONSE:P+"/apiResponse",API_PRODUCES:P+"/apiProduces",API_CONSUMES:P+"/apiConsumes",API_TAGS:P+"/apiUseTags",API_CALLBACKS:P+"/apiCallbacks",API_PARAMETERS:P+"/apiParameters",API_HEADERS:P+"/apiHeaders",API_MODEL_PROPERTIES:P+"/apiModelProperties",API_MODEL_PROPERTIES_ARRAY:P+"/apiModelPropertiesArray",API_SECURITY:P+"/apiSecurity",API_EXCLUDE_ENDPOINT:P+"/apiExcludeEndpoint",API_EXCLUDE_CONTROLLER:P+"/apiExcludeController",API_EXTRA_MODELS:P+"/apiExtraModels",API_EXTENSION:P+"/apiExtension",API_SCHEMA:P+"/apiSchema",API_DEFAULT_GETTER:P+"/apiDefaultGetter",API_LINK:P+"/apiLink"},v=e=>void 0===e;function O(e){if("function"==typeof e)return O(e());if(Array.isArray(e))return e;if("object"!=typeof e)return[];const t=Object.values(e).filter(e=>"number"==typeof e).map(e=>e.toString());return Object.keys(e).filter(e=>!t.includes(e)).map(t=>e[t])}function R(e){return e.filter(s.isString).length>0?"string":"number"}function w(e,t,r=!0){return(o,a)=>{const i=Reflect.getMetadata(E.API_MODEL_PROPERTIES_ARRAY,o)||[],n=":"+a;i.includes(n)||Reflect.defineMetadata(E.API_MODEL_PROPERTIES_ARRAY,[...i,":"+a],o);const c=Reflect.getMetadata(e,o,a);if(c){const i=s.pickBy(t,s.negate(v)),n=r?{...c,...i}:{...i,...c};Reflect.defineMetadata(e,n,o,a)}else{const r=o?.constructor?._OPENAPI_METADATA_FACTORY?.()[a]?.type??Reflect.getMetadata("design:type",o,a);Reflect.defineMetadata(e,{type:r,required:!1,...s.pickBy(t,s.negate(v))},o,a)}}}function S(e={}){return D(e)}function D(e={},t=!0){const[r,o]=((e,t)=>{if(!e)return[e,t];if(t)return[e,t];const r=s.isArray(e);return[r?e[0]:e,r]})(e.type,e.isArray);if((a=e={...e,type:r,isArray:o}).isArray&&"enum"in a&&void 0!==a.enum){e.type="array";const t=O(e.enum);e.items={type:R(t),enum:t},delete e.enum}else if("enum"in e&&void 0!==e.enum){const t=O(e.enum);e.enum=t,e.type=R(t)}var a;return Array.isArray(e.type)&&(e.type="array",e.items={type:"array",items:{type:e.type[0]}}),w(E.API_MODEL_PROPERTIES,e,t)}function C(e,r){const{handler:o,args:a}=t.Metadata.get(e,c.OperationKeys.REFLECT+c.OperationKeys.BLOCK)||{};return!!o&&(o(...a,r)??!1)}function T(e,r,a){const i={GET:[c.OperationKeys.READ,o.Get],POST:[c.OperationKeys.CREATE,o.Post],PUT:[c.OperationKeys.UPDATE,o.Put],PATCH:[c.OperationKeys.UPDATE,o.Patch],DELETE:[c.OperationKeys.DELETE,o.Delete]},[s,p]=i[r];return C(e,s)?t.apply(n.ApiExcludeEndpoint()):t.apply(p(a))}function x(e=[]){const t=e.map(e=>n.ApiParam({name:e.name,description:e.description??"Path parameter: "+e.name,required:e.required??!0,type:e.type??String}));return o.applyDecorators(...t)}const M=o.createParamDecorator((e,t)=>{const r=t.switchToHttp().getRequest().body,a=t.getClass(),i=a.class;if(!i)throw new o.InternalServerErrorException(`ModelConstructor not found on controller ${a.name}. Ensure the controller was created via FromModelController.`);return r?Array.isArray(r)?r.map(e=>new i(e)):new i(r):r}),k=o.createParamDecorator((e,t)=>{const r=t.switchToHttp().getRequest(),o=r?.params??{},a=e??Object.keys(o),i=a.map(e=>o[e]);return{raw:o,valuesInOrder:i,keysInOrder:a}});function I(e=[]){const t=e.map(e=>e.name);return k(t)}const $=o.createParamDecorator((e,t)=>{const r={...t.switchToHttp().getRequest().query??{}};if(void 0!==r.limit){const e=Number(r.limit);Number.isNaN(e)||(r.limit=e)}if(void 0!==r.offset){const e=Number(r.offset);Number.isNaN(e)||(r.offset=e)}if(void 0!==r.bookmark){const e=Number(r.bookmark);r.bookmark=Number.isNaN(e)?r.bookmark:e}return r});function q(e){const t={GET:o.Get,POST:o.Post,PUT:o.Put,PATCH:o.Patch,DELETE:o.Delete}[e];if(!t)throw Error(`Unsupported HTTP verb "${e}". No NestJS decorator mapping was found.`);return t}const F="DecafModuleOptions",N="DecafAdapter",j="DecafRoute",L=Symbol("DecafHandlers"),B=Symbol("DecafAdapterForOptions"),K=Symbol("AUTH_HANDLER"),U="auth:meta",W=Symbol("decaf:context");function H(t){const r=t?"string"==typeof t?t:t.name:void 0,a=[n.ApiBearerAuth(),o.UseInterceptors(e.AuthInterceptor)];return r&&a.push(o.SetMetadata(U,r)),o.applyDecorators(...a)}e.AuthInterceptor=class{constructor(e,t){this.reflector=e,this.authHandler=t}async intercept(e,t){const r=l.Logging.for(this).for(this.intercept),o=this.reflector.get(U,e.getHandler())??this.reflector.get(U,e.getClass());return r.verbose("Intercepted request"+(o?" for "+o:"")),this.authHandler?await this.authHandler.authorize(e,o):r.debug("No auth handler/model"),t.handle()}},e.AuthInterceptor=p.__decorate([o.Injectable(),p.__param(1,o.Optional()),p.__param(1,o.Inject(K)),p.__metadata("design:paramtypes",[d.Reflector,Object])],e.AuthInterceptor);class G{parseRequest(e){const t=e.headers.authorization?.split(" ")[1];return t}async authorize(e,r){const o=e.switchToHttp().getRequest(),s=this.parseRequest(o);if(!s)throw new i.AuthorizationError("Unauthenticated");if(!t.Metadata.get(a.Model.get(r),i.PersistenceKeys.AUTH_ROLE).includes(s))throw new i.AuthorizationError("Missing role: "+s)}}var z;e.DecafRequestContext=class{constructor(e){this.req=e,this.uuid=i.UUID.instance.generate()}get request(){return this.req}put(e){let t;try{t=this.ctx.get("overrides")}catch(e){t={}}this._ctx=this.ctx.accumulate({overrides:Object.assign(t,e)})}applyCtx(e){if(this._ctx)throw new c.InternalError("Trying to overwrite context");this._ctx=e}get ctx(){if(!this._ctx)throw new c.InternalError("Context not initialized for request");return this._ctx}},e.DecafRequestContext=p.__decorate([o.Injectable({scope:o.Scope.REQUEST}),p.__param(0,o.Inject(d.REQUEST)),p.__metadata("design:paramtypes",[Request])],e.DecafRequestContext),e.DecafHandlerExecutor=z=class{constructor(e,t){this.handlers=e,this.context=t}async exec(e,t){l.Logging.for(z.name).for(this.exec).debug(`CONTEXT ${this.context.uuid} running ${this.handlers.length} handlers for request ${e.method} ${e.url}`);for(const r of this.handlers)await r.handle(this.context,e,t)}},e.DecafHandlerExecutor=z=p.__decorate([o.Injectable({scope:o.Scope.REQUEST}),p.__param(0,o.Inject(L)),p.__metadata("design:paramtypes",[Array,e.DecafRequestContext])],e.DecafHandlerExecutor),e.DecafResponseInterceptor=class{constructor(e){this.ctx=e}intercept(e,t){let r=e.switchToHttp().getResponse();return t.handle().pipe(u.tap(e=>{r=this.ctx.ctx.toResponse(r)}))}},e.DecafResponseInterceptor=p.__decorate([o.Injectable(),p.__metadata("design:paramtypes",[e.DecafRequestContext])],e.DecafResponseInterceptor);const Q=e=>({name:e,description:e+" parameter for the query",required:!0,type:String});function V(e,t,r,a=!1){const s=q(r),c=(e=>e.split("/").filter(e=>e.startsWith(":")).map(e=>e.slice(1)))(t).map(Q),p=[];return"GET"===r&&a&&p.push(n.ApiQuery({name:"direction",required:!1,enum:i.OrderDirection,description:"the sort order when applicable"}),n.ApiQuery({name:"limit",required:!1,description:"limit or page size when applicable"}),n.ApiQuery({name:"offset",required:!1,description:"offset or bookmark when applicable"})),{method:[s(t),...c.map(n.ApiParam),...p,n.ApiOperation({summary:`Retrieve records using according to "${e}".`}),n.ApiOkResponse({description:"Result successfully retrieved."}),n.ApiNoContentResponse({description:"No content returned by the method."})],params:[I(c),o.Query()]}}function Y(e,t,r,o){const a=e?.prototype??e;o.method.forEach(e=>e(a,t,r)),o.params?.forEach((e,r)=>e(a,t,r))}function J(e,t,...r){if(e instanceof i.ModelService)return"function"==typeof e[t]?e[t](...r):e.statement(t,...r);if("function"==typeof e[t])return e[t](...r);throw Error(`Persistence method "${t}" not found on ${e?.constructor?.name}`)}function X(e){return async function(t,r){const o=this.log.for(e);try{o.debug(`Invoking persistence method "${e}" given parameters: ${JSON.stringify(t.valuesInOrder)}`);const{direction:a,limit:i,offset:s}=r;return await J(this.persistence(),e,...t.valuesInOrder,a,i,s)}catch(t){throw o.error(`Custom query "${e}" failed`,t),t}}}function Z(e,t,r){return Object.defineProperty(e.prototype||e,t,{value:r,writable:!1}),Object.getOwnPropertyDescriptor(e.prototype||e,t)}class ee extends i.Service{constructor(e,t){super(),this.clientContext=e}logCtx(e,t,r=!1,o){const a=this.clientContext.ctx;let s;e=e.filter(e=>void 0!==e),o&&(o.headers||o.ip)&&(s=o,o={});const n=i.ContextualLoggedClass.logCtx.call(this,t,o||{},r,...e,a);return this.bindLoggerToRequest(n,s)}bindLoggerToRequest(e,t){const r=this;function o(e,t){return e.log=((e,t)=>{const o=(e=>{if(!e)return;const t=e.headers,r=re(t?.["x-forwarded-for"])??re(t?.["x-real-ip"])??re(t?.["X-Forwarded-For"])??re(t?.["X-Real-IP"]);if(r)return r;if("string"==typeof e.ip&&e.ip.length)return e.ip;const o=e.socket||e.connection;return o&&"string"==typeof o.remoteAddress&&o.remoteAddress.length?o.remoteAddress:void 0})(t??r.clientContext.request);if(!o)return e;const a={ip:o};return e.for(a)})(e.log,t),e}return(e=>"object"==typeof e&&null!==e&&"function"==typeof e.then)(e)?e.then(e=>o(e,t)):o(e,t)}}class te extends ee{persistence(e){if(!this._persistence)try{this._persistence=i.Service.get(this.class)}catch(e){try{this._persistence=i.ModelService.getService(this.class)}catch(e){this._persistence=i.Repository.forModel(this.class)}}const t=this.clientContext.request[B]||{};return e&&this.clientContext.put(t),e?this._persistence instanceof i.Repository?this._persistence.override(t):this._persistence.for(t):this._persistence}constructor(e,t){super(e,t)}logCtx(e,t,r=!1,o){const a=this.clientContext.ctx;let s;o&&(o.headers||o.ip)&&(s=o,o={});try{o=a.get("overrides")}catch(e){}const n=this.persistence(a);let c,p;return n instanceof i.ModelService?c=n.repo._adapter:n instanceof i.Repository?c=n._adapter:n.context&&(c=n),r?c.logCtx(e,t,!0,o).then(e=>this.bindLoggerToRequest(e,s)):(p=c.logCtx(e,t,!1,o),this.bindLoggerToRequest(p,s))}}function re(e){if(e)return(Array.isArray(e)?e[0]:e).split(",").map(e=>e.trim()).filter(Boolean)[0]}const oe=new Map;function ae(e,r){if(!i.TransactionOperationKeys.includes(e))return r;const o=(e=>(oe.has(e)||oe.set(e,new WeakMap),oe.get(e)))(e),s=o.get(r);if(s)return s;const n=[c.OperationKeys.UPDATE,c.BulkCrudOperationKeys.UPDATE_ALL].includes(e);class p{}o.set(r,p),Object.defineProperty(p,"name",{value:`${l.toPascalCase(r.name)}${l.toPascalCase(e)}DTO`});const d=(()=>{try{return a.Model.pk(r)}catch{return}})(),u=d?a.Model.pkProps(r):void 0,h=d?Reflect.getMetadata("design:type",r.prototype,d):void 0,m=h===Number||h===BigInt,f=!!u?.generated||!!d&&(a.Model.generatedBySequence(r,d)||ne(r,d)||m),y=Array.from(new Set(t.Metadata.properties(r)||[])),g=new Set(a.Model.relations(r)||[]),_=[];for(const e of y)e&&(g.has(e)||e===d&&!n&&f||e!==d&&ne(r,e)||_.push(e));for(const e of _){const t=ce(r,e),o=!!t?.[a.ValidationKeys.REQUIRED],i=pe(r,e)??Reflect.getMetadata("design:type",r.prototype,e),s={required:o};i&&(s.type=i),S(s)(p.prototype,e);const n=Reflect.getMetadata("design:type",r.prototype,e)??i;void 0!==n&&Reflect.defineMetadata("design:type",n,p.prototype,e),Object.defineProperty(p.prototype,e,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}for(const o of g){const i=t.Metadata.relations(r,o);if(!i)throw new c.InternalError(`Metadata for relation ${o} not found`);let s=i.class;if("function"!=typeof s||s.name||(s=s()),!s||"function"!=typeof s)throw new c.InternalError(`Type for relation ${o} not found`);if(!a.Model.get(s.name))continue;const d=t.Metadata.validationFor(r,o),l=!!d?.[a.ValidationKeys.LIST],u=!!d?.[a.ValidationKeys.REQUIRED],h=ae(e,s);n?se(p,o,s,h,l,u):ie(p,o,h,l,u)}return p}function ie(e,t,r,o,a){S({type:r,required:a,isArray:o})(e.prototype,t),Reflect.defineMetadata("design:type",o?Array:r,e.prototype,t),Object.defineProperty(e.prototype,t,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}function se(e,t,r,o,i,s){const n=Reflect.getMetadata(E.API_EXTRA_MODELS,e)||[];n.includes(o)||Reflect.defineMetadata(E.API_EXTRA_MODELS,[...n,o],e);const c="#/components/schemas/"+o.name,p=(e=>{try{const t=a.Model.pkProps(e),r=t?.type;return r===Number||r===BigInt?"integer":"string"}catch{return"string"}})(r),d=[{$ref:c},{type:p}];S(i?{type:"array",required:s,oneOf:d}:{type:Object,required:s,oneOf:d})(e.prototype,t),Reflect.defineMetadata("design:type",i?Array:Object,e.prototype,t),Object.defineProperty(e.prototype,t,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}function ne(e,t){let r=e;for(;r&&r!==Object&&r!==Function;){if(a.Model.generated(r,t))return!0;r=Object.getPrototypeOf(r)}return!1}function ce(e,r){let o=e;for(;o&&o!==Object&&o!==Function;){const e=t.Metadata.validationFor(o,r);if(e)return e;o=Object.getPrototypeOf(o)}}function pe(e,r){let o=e;for(;o&&o!==Object&&o!==Function;){const e=t.Metadata.type(o,r);if(e)return e;o=Object.getPrototypeOf(o)}}i.Adapter.transformerFor=(e=>{const r="string"==typeof e?e:e.alias;return t.Metadata.innerGet(Symbol.for("transformers"),r)}).bind(i.Adapter),i.Adapter.flavoursToTransform=(()=>{const e=t.Metadata.innerGet(Symbol.for("transformers"));if(e)return Object.keys(e)}).bind(i.Adapter),i.Context.prototype.toResponse=function(e){const t=this.pending();return t&&e.header("x-pending-task",JSON.stringify(t)),e};const de=a.ModelBuilder.prototype;if(de.decorateClass||(de.decorateClass=function(e){return this._classDecorators||(this._classDecorators=[]),this._classDecorators.push(e),this}),de.Auth=function(e){return this.decorateClass(H(e))},!de.__hasClassDecoratorSupport){const e=de.build;de.build=function(){let t=e.call(this);const r=this._classDecorators;if(r?.length)for(const e of r){const r=e(t);r&&(t=r)}return t},de.__hasClassDecoratorSupport=!0}class le{static{this.log=l.Logging.for(le.name)}static getPersistence(e){try{return i.Service.get(e)}catch(t){try{return i.ModelService.getService(e)}catch(t){return i.Repository.forModel(e)}}}static createQueryRoutesFromRepository(e,r=i.PersistenceKeys.QUERY){const o=le.log.for(le.createQueryRoutesFromRepository),a=e instanceof i.ModelService?e.repo:e,s=a.class,n=t.Metadata.get(a.constructor,t.Metadata.key(i.PersistenceKeys.QUERY))??{},c=t.Metadata.get(e.constructor,t.Metadata.key(j))??{};class p extends te{get class(){throw Error("Method not implemented.")}constructor(e,t){super(e,t)}}for(const[e,t]of Object.entries(c)){const r=[t.path.replace(/^\/+|\/+$/g,"")].filter(e=>e&&e.trim()).join("/"),a=X(e);if(!a){const t=`Invalid or missing handler for model ${s.name} on decorated method ${e}`;throw o.error(t),Error(t)}const i=Z(p,e,a);i&&Y(p,e,i,V(e,r,t.httpMethod))}for(const[e,t]of Object.entries(n)){const o=[r,e,...(t.fields??[]).map(e=>":"+e)].filter(e=>e&&e.trim()).join("/"),a=Z(p,e,X(e));a&&Y(p,e,a,V(e,o,"GET",!0))}return p}static create(t){const r=le.log.for(le.create),s=a.Model.tableName(t),d=l.toKebabCase(s),u=t.name,h=le.getPersistence(t),{description:m,getPK:f,apiProperties:y,path:g}=le.getRouteParametersFromModel(t);r.debug("Creating controller for model: "+u);const _=le.createQueryRoutesFromRepository(h);let A=class extends _{static get class(){return t}get class(){return t}constructor(e){super(e),this.pk=a.Model.pk(t),r.info(`Registering dynamic controller for model: ${this.class.name} route: /${d}`)}async listBy(e,t){const{ctx:r}=(await this.logCtx([],i.PreparedStatementKeys.LIST_BY,!0)).for(this.listBy);return this.persistence(r).listBy(e,t.direction,r)}async paginateBy(e,t){const{ctx:r}=(await this.logCtx([],i.PreparedStatementKeys.PAGE_BY,!0)).for(this.paginateBy);return this.persistence(r).paginateBy(e,t.direction,t,r)}async find(e,t){const{ctx:r}=(await this.logCtx([],i.PreparedStatementKeys.FIND,!0)).for(this.find),o=t.direction??i.OrderDirection.ASC;return J(this.persistence(r),this.find.name,e,o,r)}async page(e,t){const{ctx:r}=(await this.logCtx([],i.PreparedStatementKeys.PAGE,!0)).for(this.page),{direction:o=i.OrderDirection.ASC,limit:a,offset:s,bookmark:n}=t,c={offset:s??1,limit:a??10,bookmark:n};return J(this.persistence(r),this.page.name,e,o,c,r)}async findOneBy(e,t){const{ctx:r}=(await this.logCtx([],i.PreparedStatementKeys.FIND_ONE_BY,!0)).for(this.findOneBy);return this.persistence(r).findOneBy(e,t,r)}async findBy(e,t,r){const{ctx:o}=(await this.logCtx([],i.PreparedStatementKeys.FIND_BY,!0)).for(this.findBy);return this.persistence(o).for(o.toOverrides()).findBy(e,t,o)}async statement(e,t,r){const{ctx:o}=(await this.logCtx([],i.PersistenceKeys.STATEMENT,!0)).for(this.statement),{direction:a,offset:s,limit:n,bookmark:c}=r,p=(t=t.map(e=>("string"==typeof e?parseInt(e):e)||e)).length>1?t[1]:void 0,d=a??p;switch(d&&t.length>1&&(t[1]=d),e){case i.PreparedStatementKeys.FIND:case i.PreparedStatementKeys.FIND_BY:break;case i.PreparedStatementKeys.LIST_BY:t.push(a);break;case i.PreparedStatementKeys.PAGE:case i.PreparedStatementKeys.PAGE_BY:t=[t[0],d,{limit:n,offset:s,bookmark:c}];case i.PreparedStatementKeys.FIND_ONE_BY:case i.PreparedStatementKeys.COUNT_OF:case i.PreparedStatementKeys.MAX_OF:case i.PreparedStatementKeys.MIN_OF:case i.PreparedStatementKeys.AVG_OF:case i.PreparedStatementKeys.SUM_OF:case i.PreparedStatementKeys.DISTINCT_OF:case i.PreparedStatementKeys.GROUP_OF:}return this.persistence(o).statement(e,...t,o)}async createAll(e,r){const{ctx:o,log:a}=(await this.logCtx([],c.BulkCrudOperationKeys.CREATE_ALL,!0)).for(this.createAll);let i;a.verbose("creating new "+u);try{i=await this.persistence(o).createAll(e.map(e=>new t(e)),o)}catch(e){throw a.error("Failed to create new "+u,e),e}return a.info(`created new ${u} with id ${i[this.pk]}`),o.toResponse(r),i}async create(e,t){const{ctx:r,log:o}=(await this.logCtx([],c.OperationKeys.CREATE,!0)).for(this.create);let a;o.verbose("creating new "+u);try{const t=this.persistence(r);a=await t.create(e,r)}catch(e){throw o.error("Failed to create new "+u,e),e}return o.info(`created new ${u} with id ${a[this.pk]}`),r.toResponse(t),a}async readAll(e){const{ctx:t,log:r}=(await this.logCtx([],c.BulkCrudOperationKeys.READ_ALL,!0)).for(this.readAll),o=Array.isArray(e)?e:[e];let a;try{r.debug(`reading ${o} ${u}: ${o}`);const e=this.persistence(t);a=await e.readAll(o,t)}catch(e){throw r.error(`Failed to ${u} with id ${o}`,e),e}return r.info(`read ${a.length} ${u}`),a}async read(e){const{ctx:t,log:r}=(await this.logCtx([],c.OperationKeys.READ,!0)).for(this.read),o=f(...e.valuesInOrder);if(void 0===o)throw new c.ValidationError(`No ${this.pk} provided`);let a;try{r.debug(`reading ${u} with ${this.pk} ${o}`);const e=this.persistence(t);a=await e.read(o,t)}catch(e){throw r.error(`Failed to read ${u} with id ${o}`,e),e}return r.info(`read ${u} with id ${a[this.pk]}`),a}async updateAll(e,t){const{ctx:r,log:o}=(await this.logCtx([],c.BulkCrudOperationKeys.UPDATE_ALL,!0)).for(this.updateAll);let a;try{o.info(`updating ${e.length} ${u}`),a=await this.persistence(r).updateAll(e,r)}catch(e){throw o.error(e),e}return r.toResponse(t),a}async update(e,r,o){const{ctx:a,log:i}=(await this.logCtx([],c.OperationKeys.UPDATE,!0)).for(this.update),s=f(...e.valuesInOrder);if(void 0===s)throw new c.ValidationError(`No ${this.pk} provided`);let n;try{i.info(`updating ${u} with ${this.pk} ${s}`);const e=JSON.parse(JSON.stringify(r)),o=this.persistence(a);n=await o.update(new t({...e,[this.pk]:s}),a)}catch(e){throw i.error(e),e}return a.toResponse(o),n}async deleteAll(e,t){const{ctx:r,log:o}=(await this.logCtx([],c.BulkCrudOperationKeys.DELETE_ALL,!0)).for(this.deleteAll),a=Array.isArray(e)?e:[e];let i;try{o.debug(`deleting ${a.length} ${u}: ${a}`),i=await this.persistence(r).deleteAll(a,r)}catch(e){throw o.error(`Failed to delete ${u} with id ${a}`,e),e}return o.info(`deleted ${i.length} ${u}`),r.toResponse(t),i}async delete(e,t){const{ctx:r,log:o}=(await this.logCtx([],c.OperationKeys.DELETE,!0)).for(this.delete),a=f(...e.valuesInOrder);if(void 0===a)throw new c.ValidationError(`No ${this.pk} provided`);let i;try{o.debug(`deleting ${u} with ${this.pk} ${a}`),i=await this.persistence(r).delete(a,r)}catch(e){throw o.error(`Failed to delete ${u} with id ${a}`,e),e}return o.info(`deleted ${u} with id ${a}`),r.toResponse(t),i}};return p.__decorate([T(t,"GET","listBy/:key"),n.ApiOperation({summary:`Retrieve ${u} records by query.`}),n.ApiParam({name:"key",description:"the model key to sort by"}),n.ApiQuery({name:"direction",required:!0,enum:i.OrderDirection}),n.ApiOkResponse({description:u+" listed successfully."}),p.__param(0,o.Param("key")),p.__param(1,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"listBy",null),p.__decorate([T(t,"GET","paginateBy/:key/:page"),n.ApiOperation({summary:`Retrieve ${u} records by query.`}),n.ApiParam({name:"key",description:"the model key to sort by"}),n.ApiParam({name:"page",description:"the page to retrieve or the bookmark"}),n.ApiQuery({name:"direction",required:!0,enum:i.OrderDirection,description:"the sort order"}),n.ApiQuery({name:"limit",required:!0,description:"the page size"}),n.ApiQuery({name:"offset",description:"the bookmark when necessary"}),n.ApiOkResponse({description:u+" listed paginated."}),p.__param(0,o.Param("key")),p.__param(1,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"paginateBy",null),p.__decorate([T(t,"GET","find/:value"),n.ApiOperation({summary:`Find ${u} records using the default query attributes.`}),n.ApiParam({name:"value",description:"The string to match against the default query attributes"}),n.ApiQuery({name:"direction",required:!1,enum:i.OrderDirection,description:"the sort order for the matching results"}),n.ApiOkResponse({description:u+" records matching the provided prefix."}),p.__param(0,o.Param("value")),p.__param(1,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"find",null),p.__decorate([T(t,"GET","page/:value"),n.ApiOperation({summary:`Page ${u} records using the default query attributes.`}),n.ApiParam({name:"value",description:"The string to match against the default query attributes"}),n.ApiQuery({name:"direction",required:!1,enum:i.OrderDirection,description:"the sort order for the paged results"}),n.ApiQuery({name:"limit",required:!1,description:"page size"}),n.ApiQuery({name:"offset",required:!1,description:"page number"}),n.ApiQuery({name:"bookmark",required:!1,description:"bookmark for cursor pagination"}),n.ApiOkResponse({description:u+" records paged by the provided prefix."}),p.__param(0,o.Param("value")),p.__param(1,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"page",null),p.__decorate([T(t,"GET","findOneBy/:key"),n.ApiOperation({summary:`Retrieve ${u} records by query.`}),n.ApiParam({name:"key",description:"the model key to sort by"}),n.ApiOkResponse({description:u+" listed found."}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,o.Param("key")),p.__param(1,o.Param("value")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"findOneBy",null),p.__decorate([T(t,"GET","findBy/:key/:value"),n.ApiOperation({summary:`Retrieve ${u} records by query.`}),n.ApiParam({name:"key",description:"the model key to compare"}),n.ApiParam({name:"value",description:"the value to match"}),n.ApiQuery({name:"direction",required:!0,enum:i.OrderDirection,description:"the sort order when applicable"}),n.ApiOkResponse({description:u+" listed found."}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,o.Param("key")),p.__param(1,o.Param("value")),p.__param(2,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Object,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"findBy",null),p.__decorate([T(t,"GET","statement/:method/*args"),n.ApiOperation({summary:`Executes a prepared statement on ${u}.`}),n.ApiParam({name:"method",description:"the prepared statement to execute"}),n.ApiParam({name:"args",description:"concatenated list of arguments the prepared statement can accept"}),n.ApiQuery({name:"direction",required:!0,enum:i.OrderDirection,description:"the sort order when applicable"}),n.ApiQuery({name:"limit",required:!0,description:"limit or page size when applicable"}),n.ApiQuery({name:"offset",required:!0,description:"offset or bookmark when applicable"}),n.ApiOkResponse({description:u+" listed found."}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,o.Param("method")),p.__param(1,o.Param("args")),p.__param(2,$()),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String,Array,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"statement",null),p.__decorate([T(t,"POST","bulk"),n.ApiOperation({summary:`Create a new ${u}.`}),n.ApiBody({description:"Payload for "+u,schema:{type:"array",items:{$ref:n.getSchemaPath(t)}}}),n.ApiCreatedResponse({description:u+" created successfully.",schema:{type:"array",items:{$ref:n.getSchemaPath(t)}}}),n.ApiBadRequestResponse({description:"Payload validation failed."}),n.ApiUnprocessableEntityResponse({description:"Repository rejected the provided payload."}),p.__param(0,M()),p.__param(1,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Array,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"createAll",null),p.__decorate([T(t,"POST"),n.ApiOperation({summary:`Create a new ${u}.`}),n.ApiBody({description:"Payload for "+u,type:ae(c.OperationKeys.CREATE,t)}),n.ApiCreatedResponse({description:u+" created successfully.",schema:{$ref:n.getSchemaPath(t)}}),n.ApiBadRequestResponse({description:"Payload validation failed."}),n.ApiUnprocessableEntityResponse({description:"Repository rejected the provided payload."}),p.__param(0,M()),p.__param(1,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Object,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"create",null),p.__decorate([T(t,"GET","bulk"),n.ApiOperation({summary:`Retrieve a ${u} record by id.`}),n.ApiQuery({name:"ids",required:!0,type:"array"}),n.ApiOkResponse({description:u+" retrieved successfully.",schema:{type:"array",items:{$ref:n.getSchemaPath(t)}}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,o.Query("ids")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Array]),p.__metadata("design:returntype",Promise)],A.prototype,"readAll",null),p.__decorate([T(t,"GET",g),x(y),n.ApiOperation({summary:`Retrieve a ${u} record by id.`}),n.ApiOkResponse({description:u+" retrieved successfully.",schema:{$ref:n.getSchemaPath(t)}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,I(y)),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Object]),p.__metadata("design:returntype",Promise)],A.prototype,"read",null),p.__decorate([T(t,"PUT","bulk"),x(y),n.ApiOperation({summary:`Replace an existing ${u} record with a new payload.`}),n.ApiBody({description:"Payload for replace a existing record of "+u,schema:{type:"array",$ref:n.getSchemaPath(ae(c.OperationKeys.UPDATE,t))}}),n.ApiOkResponse({description:u+" updated successfully.",schema:{type:"array",items:{$ref:n.getSchemaPath(t)}}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),n.ApiBadRequestResponse({description:"Payload validation failed."}),p.__param(0,M()),p.__param(1,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Array,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"updateAll",null),p.__decorate([T(t,"PUT",g),x(y),n.ApiOperation({summary:`Replace an existing ${u} record with a new payload.`}),n.ApiBody({description:"Payload for replace a existing record of "+u,type:ae(c.OperationKeys.UPDATE,t)}),n.ApiOkResponse({description:u+" updated successfully.",schema:{$ref:n.getSchemaPath(t)}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),n.ApiBadRequestResponse({description:"Payload validation failed."}),p.__param(0,I(y)),p.__param(1,M()),p.__param(2,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Object,Object,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"update",null),p.__decorate([T(t,"DELETE","bulk"),x(y),n.ApiOperation({summary:`Retrieve a ${u} record by id.`}),n.ApiQuery({name:"ids",required:!0,type:"array"}),n.ApiOkResponse({description:u+" deleted successfully.",schema:{type:"array",items:{$ref:n.getSchemaPath(t)}}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,o.Query("ids")),p.__param(1,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Array,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"deleteAll",null),p.__decorate([T(t,"DELETE",g),x(y),n.ApiOperation({summary:`Delete a ${u} record by id.`}),n.ApiOkResponse({description:u+" deleted successfully.",schema:{$ref:n.getSchemaPath(t)}}),n.ApiNotFoundResponse({description:`No ${u} record matches the provided identifier.`}),p.__param(0,I(y)),p.__param(1,o.Response({passthrough:!0})),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[Object,Object]),p.__metadata("design:returntype",Promise)],A.prototype,"delete",null),A=p.__decorate([o.Controller(d),n.ApiTags(u),n.ApiExtraModels(t),H(t),p.__metadata("design:paramtypes",[e.DecafRequestContext])],A),A}static getRouteParametersFromModel(e){const r=a.Model.pk(e),o=t.Metadata.get(e,t.Metadata.key(c.DBKeys.COMPOSED,r)),i=o?.args??[],s=Array.isArray(i)&&i.length>0?Array.from(new Set([...i])):Array.from(new Set([r])),n=t.Metadata.description(e);return{path:":"+s.join("/:"),description:n,apiProperties:s.map(r=>({name:r,description:t.Metadata.description(e,r),required:!0,type:String})),getPK:(...e)=>o?.separator?e.join(o.separator):e.join("")}}}function ue(e){var t;let r=class{static{t=this}static{this.log=l.Logging.for(t.name).for(e)}static createModelServices(e){return e.map(e=>({provide:e.name+"Service",useFactory:()=>i.ModelService.forModel(e)}))}static forRoot(e,r={}){const o=this.log.for(this.forRoot);o.info("Generating controllers for flavour...");const a=i.Adapter.models(e);let s=[];r.autoServices&&(o.info("Auto-services enabled. Initializing service generation."),s=this.createModelServices(a),o.info(`Auto-services completed. ${s.length} services initialized.`));const n=a.map(le.create);return o.info(`Generated ${n.length} controllers`),{module:t,controllers:n,providers:[...s]}}};return r=t=p.__decorate([o.Module({})],r),Object.assign(r,"name",{value:"DecafModule"+e}),r}t.Decoration.for(r.InjectablesKeys.INJECTABLE).extend({decorator:(e,t)=>(t=t||("object"==typeof e?Object.assign(e,r.DefaultInjectablesConfig):r.DefaultInjectablesConfig),o.Injectable({scope:t.singleton?o.Scope.DEFAULT:o.Scope.REQUEST,durable:!t.singleton||void 0}))}).apply(),t.Decoration.for(r.InjectablesKeys.INJECT).extend({decorator:(e,t)=>(t,r,a)=>o.Inject(e||t)(t,r,a)}).apply(),t.Decoration.for(a.ValidationKeys.REQUIRED).extend(S({required:!0})).apply(),t.Decoration.for(a.ValidationKeys.MAX).extend({decorator:e=>S({maximum:e,required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.MIN).extend({decorator:e=>S({minimum:e,required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.MAX_LENGTH).extend({decorator:e=>S({maxLength:e,required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.MIN_LENGTH).extend({decorator:e=>S({minLength:e,required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.TYPE).extend({decorator:e=>(t,r)=>("function"!=typeof(e=Array.isArray(e)?e[0]:e)||e.name||(e=e()),S({type:e,required:!1})(t,r))}).apply(),t.Decoration.for(a.ValidationKeys.DATE).extend({decorator:e=>S({type:String,format:"date-time",required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.ENUM).extend({decorator:e=>S({enum:Array.isArray(e)?e:Object.values(e),required:!1})}).apply(),t.Decoration.for(a.ValidationKeys.PATTERN).extend({decorator:e=>S({pattern:"string"==typeof e?e:e.source,required:!1})}).apply(),t.Decoration.for(i.PersistenceKeys.COLUMN).extend({decorator:e=>S({name:e,required:!1})}).apply(),t.Decoration.for(t.DecorationKeys.DESCRIPTION).extend({decorator:e=>S({description:e,required:!1})}).apply(),t.Decoration.for(i.PersistenceKeys.AUTH).extend({decorator:H}).apply();class he extends i.ForbiddenError{constructor(e){super(e,he.name)}}class me extends c.BaseError{constructor(e){super(me.name,e,429)}}e.DecafExceptionFilter=class{catch(e,t){const r=t.switchToHttp(),a=r.getResponse(),s=r.getRequest(),n="production"===l.LoggedEnvironment.env;let p;e instanceof o.NotFoundException||e instanceof i.UnsupportedError?p=(e=new o.NotAcceptableException(e.message)).getStatus():e instanceof c.BaseError||(e=429===e.status?new me(e.message):new c.InternalError(e.message)),a.status(e.code||p).json({status:e.code||p,error:n?e.name:e.message,timestamp:(new Date).toISOString(),path:s.url,method:s.method})}},e.DecafExceptionFilter=p.__decorate([o.Catch()],e.DecafExceptionFilter);const fe={path:"docs",auth:{type:"http",scheme:"bearer",bearerFormat:"JWT",name:"Authorization",description:"Enter JWT token",in:"header"},topbarBgColor:"#000000"};class ye{constructor(e){this.options={...e}}customCSS(){let e="";return this.options.topbarIconPath&&(e+=`.topbar-wrapper { content: url('data:image/png;base64,${this.b64(this.options.topbarIconPath)}'); width: 200px; height: auto; }\n`),e+`\n .topbar-wrapper svg { visibility: hidden; }\n .swagger-ui .topbar { background-color: ${this.options.topbarBgColor||fe.topbarBgColor}; }\n `}getCustomOptions(){const e={};return this.options.faviconPath&&(e.customfavIcon=this.b64(this.options.faviconPath,!0)),{customSiteTitle:this.options.title,...e,customCss:this.customCSS(),swaggerOptions:{persistAuthorization:this.options.persistAuthorization}}}b64(e,t=!1){const r=b.join(this.options.assetsPath||"",e),o=h.readFileSync(r,{encoding:"base64"});return t?"data:image/png;base64,"+o:o}}class ge{constructor(e,t){this.app=e,this.options=t}createDocument(){const e=this.options.path?this.options.description+""+`<br><br><a href="${this.options.openApiJsonPath}">OpenAPI JSON Spec</a> | `+`<a href="${this.options.openApiYamlPath}">OpenAPI YAML Spec</a>`:this.options.description,t=(new n.DocumentBuilder).setTitle(this.options.title).setDescription(e).setVersion(this.options.version||"0.0.1").addBearerAuth(this.options.auth||fe.auth).build();return n.SwaggerModule.createDocument(this.app,t,{extraModels:this.options.extraModels||[]})}registerOpenApiRoute(e,t,r){if(!e)return;const o=this.app.getHttpAdapter();e=e.startsWith("/")?e:"/"+e,o.get(e,(e,a)=>{o.reply(a,r(),200,{"Content-Type":t})})}setupSwagger(){const e=this.createDocument(),t=new ye({title:this.options.title,path:this.options.path||fe.path,persistAuthorization:this.options.persistAuthorization??!0,assetsPath:this.options.assetsPath,faviconPath:this.options.faviconFilePath,topbarIconPath:this.options.topbarIconFilePath,topbarBgColor:this.options.topbarBgColor});n.SwaggerModule.setup(this.options.path||fe.path,this.app,e,{...t.getCustomOptions(),jsonDocumentUrl:this.options.openApiJsonPath?""+this.options.openApiJsonPath:void 0,yamlDocumentUrl:this.options.openApiYamlPath?""+this.options.openApiYamlPath:void 0}),this.registerOpenApiRoute(this.options.openApiJsonPath,"application/json",()=>e),this.registerOpenApiRoute(this.options.openApiYamlPath,"application/x-yaml",()=>f.stringify(e))}}var _e,Ae;function be(e){return r=>(t.Metadata.set("transformers",e,r),"function"==typeof r?t.metadata("transformers",e)(r):r)}e.DecafRequestHandlerInterceptor=_e=class{constructor(e,t){this.requestContext=e,this.executor=t}async contextualize(e){const t={headers:e.headers,overrides:{}},r=i.Adapter.flavoursToTransform();if(r)for(const o of r)try{const r=i.Adapter.transformerFor(o),a=await r.from(e);Object.assign(t.overrides,a)}catch(e){throw new c.InternalError("Failed to contextualize request: "+e)}return(new i.Context).accumulate(Object.assign({},i.DefaultAdapterFlags,{logger:l.Logging.get(),timestamp:new Date},t))}async intercept(e,t){const r=e.switchToHttp().getRequest(),o=e.switchToHttp().getResponse(),a=l.Logging.for(_e).for(this.intercept);a.debug(`CONTEXT ${this.requestContext.uuid} - request: ${r.method} ${r.url}`);const i=await this.contextualize(r);return a.debug(`CONTEXT ${this.requestContext.uuid} contextualized - request: ${r.method} ${r.url}`),this.requestContext.applyCtx(i),a.debug(`CONTEXT ${this.requestContext.uuid} applied - request: ${r.method} ${r.url}`),await this.executor.exec(r,o),a.debug(`CONTEXT ${this.requestContext.uuid} executors finished - request: ${r.method} ${r.url}`),t.handle()}},e.DecafRequestHandlerInterceptor=_e=p.__decorate([o.Injectable({scope:o.Scope.REQUEST}),p.__metadata("design:paramtypes",[e.DecafRequestContext,e.DecafHandlerExecutor])],e.DecafRequestHandlerInterceptor),e.DecafCoreModule=class{static{Ae=this}static get persistence(){if(!this._persistence)throw new c.InternalError("Persistence service not initialized");return this._persistence}static get log(){return this._logger||(this._logger=l.Logging.for(Ae)),this._logger}constructor(e,t){this.options=e,this.moduleRef=t}static forRoot(t){const r=this.log.for(this.forRoot);return{module:Ae,providers:[{provide:F,useValue:t},{provide:N,useValue:this.persistence?.client},{provide:L,useFactory:()=>t.handlers?.map(e=>(r.info("Registered request handler: "+e.name),new e))??[]},e.AuthInterceptor,{provide:d.APP_INTERCEPTOR,useExisting:e.AuthInterceptor},e.DecafRequestContext,e.DecafHandlerExecutor,{provide:d.APP_INTERCEPTOR,useClass:e.DecafRequestHandlerInterceptor}],exports:[F,N,L,e.DecafRequestContext,e.DecafHandlerExecutor]}}static async bootPersistence(e){const t=this.log.for(this.bootPersistence);if(!this._persistence){const r=e.conf.map(([e,t,...r])=>r.pop()?[e,t,...r]:[e,t]);this._persistence=new i.PersistenceService,await this._persistence.boot(r);const o=this._persistence.client;for(let t=0;t<o.length;t++){const r=e.conf[t];let a=r.slice(2,r.length).pop();if(!a||!a.from){const e=i.Adapter.transformerFor(o[t].flavour);if(!e)throw new c.InternalError(`No transformer found for flavour ${o[t].flavour}. you should either @requestToContextTransformer or provide a transformer in the config`);try{a=e.from?e:new e}catch(e){throw new c.InternalError(`Failed to boot transformer for ${o[t].flavour}: ${e}`)}}be(o[t].flavour)(a)}if(t.info("persistence layer created successfully!"),e.initialization)try{await e.initialization()}catch(e){throw new c.InternalError("Failed to initialize application: "+e)}}return this.persistence.client}async onApplicationShutdown(){const e=Ae.log.for(this.onApplicationShutdown),t=this.moduleRef.get(N);for(const r of t)try{r&&(e.info("Shutting down "+r.toString()),await r.shutdown())}catch(t){e.error("Failed to shutdown application",t)}try{await i.Service.shutdown()}catch(t){e.error("Failed to shutdown services",t)}}},e.DecafCoreModule=Ae=p.__decorate([o.Global(),o.Module({}),p.__param(0,o.Inject(F)),p.__metadata("design:paramtypes",[Object,d.ModuleRef])],e.DecafCoreModule);const Pe=Symbol("LISTENING_ADAPTERS_FLAVOURS");var Ee,ve,Oe,Re;e.EventsController=Ee=class extends ee{constructor(e,t){super(e,Ee.name),this.adapters=t.map(e=>i.Adapter.get(e))}listen(){const e=l.Logging.for(Ee.name),t=new u.Observable(t=>{const r=("B-"+Math.random().toString(36).slice(2,8)).toUpperCase();e.info(`Creating SSE observer: ${r} for client ${this.clientContext.uuid}`);const o=new class{constructor(){this.observerId=r}refresh(...r){return e.debug(`SSE observer ${this.observerId} received refresh event`),Promise.resolve().then(()=>{const o=(e=>{const[t,r,o,a]=e;return[t?.name??t,r,o,Array.isArray(a)?a.map(e=>{try{return"function"==typeof e.serialize?e.serialize():"string"==typeof e?e:JSON.stringify(e)}catch(e){return}}):a&&"function"==typeof a.serialize?a.serialize():"string"==typeof a?a:JSON.stringify(a)]})(r);t.next({type:"message",data:o}),e.debug(`SSE observer ${this.observerId} event pushed to client`)})}};e.verbose(`Registering observer ${r} across ${this.adapters.length} adapter(s)`);for(const t of this.adapters){const a=t?.constructor?.name??"UnknownAdapter";try{e.debug(`Registering observer ${r} in adapter ${a}`),t.observe(o)}catch(t){e.debug(`Failed to register observer ${r} in adapter ${a}: ${t?.message||t}`),e.error(t)}}return()=>{e.debug("Cleaning up SSE observer "+r);for(const t of this.adapters){const a=t?.constructor?.name??"UnknownAdapter";try{e.debug(`Unregistering observer ${r} from adapter ${a}`),t.unObserve(o)}catch(t){e.debug(`Failed during cleanup of observer ${r} in adapter ${a}: ${t?.message||t}`),e.error(t)}}}}),r=u.interval(15e3).pipe(y.tap(()=>{e.debug("Sending heartbeat")}),y.map(()=>({type:"heartbeat",data:{ts:(new Date).toISOString()}})));return u.merge(t,r)}listenForModel(e){const t=l.Logging.for(Ee.name);return new u.Observable(e=>{const r=new class{refresh(...t){return Promise.resolve().then(()=>{e.next({data:t})})}};try{for(const e of this.adapters)e.observe(r)}catch(t){e.error("Failed to observe event: "+(t.message||t))}return()=>{try{for(const e of this.adapters)e.unObserve(r)}catch(e){t.error(e)}}})}},p.__decorate([o.Sse(),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[]),p.__metadata("design:returntype",u.Observable)],e.EventsController.prototype,"listen",null),p.__decorate([o.Sse("/:model"),p.__param(0,o.Query("model")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String]),p.__metadata("design:returntype",u.Observable)],e.EventsController.prototype,"listenForModel",null),e.EventsController=Ee=p.__decorate([o.Controller(),p.__param(1,o.Inject(Pe)),p.__metadata("design:paramtypes",[e.DecafRequestContext,Array])],e.EventsController),e.DecafStreamModule=ve=class{static forFlavours(t,r="events"){return{module:ve,controllers:[e.EventsController],imports:[d.RouterModule.register([{path:r.replace(/^\//,""),module:ve}])],providers:[e.DecafRequestContext,{provide:Pe,useValue:t??[]}]}}},e.DecafStreamModule=ve=p.__decorate([o.Module({})],e.DecafStreamModule),e.DecafModule=Oe=class{static async forRootAsync(t){const{autoControllers:r,autoServices:o}=t,a=(await e.DecafCoreModule.bootPersistence(t)).map(e=>e.flavour),i=[e.DecafCoreModule.forRoot(t)];return r&&a.forEach(e=>{i.push(ue(e).forRoot(e,{autoServices:o}))}),t.observerOptions?.enableObserverEvents&&i.push(e.DecafStreamModule.forFlavours(t.observerOptions.observerFlavours||a)),{module:Oe,imports:i}}},e.DecafModule=Oe=p.__decorate([o.Module({})],e.DecafModule);const we=l.Logging.for("DecafMigrationModule");var Se,De,Ce;e.DecafMigrationModule=Re=class{static forRoot(){return{module:Re}}static async migrate(e,t){const r=we.for(this.migrate);if(!t||0===t.length)throw new c.InternalError("No adapters provided. Make sure DecafCoreModule is configured and adapters are available.");return r.info("Running migrations with config: "+JSON.stringify(e)),g.MigrationService.migrateAdapters(t,e||{})}},e.DecafMigrationModule=Re=p.__decorate([o.Global(),o.Module({providers:[{provide:"MIGRATION_ADAPTERS",useFactory:e=>e,inject:[N]}],exports:["MIGRATION_ADAPTERS"]})],e.DecafMigrationModule),e.WebhookSubscriptionActionsController=Se=class extends ee{constructor(e){super(e,Se.name)}async deactivate(e){const{ctx:t}=(await this.logCtx([],"deactivate",!0)).for(this.deactivate),r=i.Repository.forModel(_.WebhookSubscription),o=await r.read(e,t);return o.active=!1,r.update(o,t)}async reactivate(e){const{ctx:t}=(await this.logCtx([],"reactivate",!0)).for(this.reactivate),r=i.Repository.forModel(_.WebhookSubscription),o=await r.read(e,t);return o.active=!0,r.update(o,t)}},p.__decorate([o.Post(":id/deactivate"),p.__param(0,o.Param("id")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String]),p.__metadata("design:returntype",Promise)],e.WebhookSubscriptionActionsController.prototype,"deactivate",null),p.__decorate([o.Post(":id/reactivate"),p.__param(0,o.Param("id")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String]),p.__metadata("design:returntype",Promise)],e.WebhookSubscriptionActionsController.prototype,"reactivate",null),e.WebhookSubscriptionActionsController=Se=p.__decorate([o.Controller("webhook-subscriptions"),p.__metadata("design:paramtypes",[e.DecafRequestContext])],e.WebhookSubscriptionActionsController),e.WebhookEventActionsController=De=class extends ee{constructor(e){super(e,De.name)}async replay(e){const{ctx:t}=(await this.logCtx([],"replay",!0)).for(this.replay),r=i.Repository.forModel(_.WebhookEventRecord),o=i.Repository.forModel(_.WebhookDelivery);try{let a;try{a=await r.read(e,t)}catch(o){const i=await r.select().where(r.attr("id").eq(e)).limit(1).execute(t);if(!i.length)throw o;a=i[0]}let s=[];try{s=await _.collectPagedResults(()=>o.select().where(o.attr("eventId").eq(a.id)).orderBy("createdAt",i.OrderDirection.ASC).thenBy("id",i.OrderDirection.ASC).paginate(250,t),250,t)}catch{try{s=await o.select().where(o.attr("eventId").eq(a.id)).execute(t)}catch{s=[]}}const n=new Date;for(const e of s)e.status=_.WebhookStatus.PENDING,e.attempts=0,e.nextAttemptAt=n,e.lastAttemptAt=null,e.errorMessage=void 0,e.responseStatus=void 0,e.responseBody=void 0;if(a.status=_.WebhookStatus.PENDING,a.deliveriesSucceeded=0,a.deliveriesFailed=0,a.nextAttemptAt=n,a.updatedAt=n,s.length)try{await o.updateAll(s,t.override({applyUpdateValidation:!1}))}catch{}return r.update(a,t)}catch(e){throw e}}},p.__decorate([o.Post(":id/replay"),p.__param(0,o.Param("id")),p.__metadata("design:type",Function),p.__metadata("design:paramtypes",[String]),p.__metadata("design:returntype",Promise)],e.WebhookEventActionsController.prototype,"replay",null),e.WebhookEventActionsController=De=p.__decorate([o.Controller("webhook-events"),p.__metadata("design:paramtypes",[e.DecafRequestContext])],e.WebhookEventActionsController),e.DecafWebhookModule=class{static{Ce=this}static{this._logger=l.Logging.for(Ce)}static get log(){return this._logger}static async bootPersistence(e){const r=this.log.for(this.bootPersistence);if(!this._persistence){const o=e.conf.map(([e,t,...r])=>r.pop()?[e,t,...r]:[e,t]);this._persistence=new i.PersistenceService,await this._persistence.boot(o);const a=this._persistence.client;for(let r=0;r<a.length;r++){const o=i.Adapter._cache||(i.Adapter._cache={}),s=[a[r].flavour,a[r].alias,"webhook_deliveries","webhook_events","webhook_subscriptions"].filter(e=>!!e);for(const e of s)o[e]=a[r];const n=e.conf[r];let p=n.slice(2,n.length).pop();if(!p||!p.from){const e=i.Adapter.transformerFor(a[r].flavour);if(!e)throw new c.InternalError(`No transformer found for flavour ${a[r].flavour}.`);try{p=e.from?e:new e}catch(e){throw new c.InternalError(`Failed to boot transformer for ${a[r].flavour}: ${e}`)}}be(a[r].flavour)(p),t.uses(a[r].flavour)(_.WebhookSubscription),t.uses(a[r].flavour)(_.WebhookEventRecord),t.uses(a[r].flavour)(_.WebhookDelivery)}if(r.info("persistence layer created successfully!"),e.initialization)try{await e.initialization()}catch(e){throw new c.InternalError("Failed to initialize webhook module: "+e)}}return this._persistence.client}static async forRoot(e){return this.forRootAsync(e)}static async forRootAsync(t){await this.bootPersistence(t);const r=[le.create(_.WebhookSubscription),le.create(_.WebhookEventRecord),le.create(_.WebhookDelivery),e.WebhookSubscriptionActionsController,e.WebhookEventActionsController];return{module:Ce,controllers:r,imports:[d.RouterModule.register([{path:(t.webhookApiPath||"webhooks").replace(/^\//,""),module:Ce}])],providers:[e.DecafRequestContext,e.DecafHandlerExecutor,{provide:L,useFactory:()=>t.handlers?.map(e=>new e)??[]},{provide:d.APP_INTERCEPTOR,useClass:e.DecafRequestHandlerInterceptor}],exports:[e.DecafRequestContext,e.DecafHandlerExecutor]}}},e.DecafWebhookModule=Ce=p.__decorate([o.Module({})],e.DecafWebhookModule);const Te=e.DecafWebhookModule,xe="##VERSION##",Me="##PACKAGE##";t.Metadata.allowReregistration(!0),t.Metadata.registerLibrary(Me,xe),t.Metadata.allowReregistration(!1),Object.defineProperty(e,"WebhookDelivery",{enumerable:!0,get:()=>_.WebhookDelivery}),Object.defineProperty(e,"WebhookDeliveryService",{enumerable:!0,get:()=>_.WebhookDeliveryService}),Object.defineProperty(e,"WebhookEventRecord",{enumerable:!0,get:()=>_.WebhookEventRecord}),Object.defineProperty(e,"WebhookPublisherService",{enumerable:!0,get:()=>_.WebhookPublisherService}),Object.defineProperty(e,"WebhookSignatureMiddleware",{enumerable:!0,get:()=>_.WebhookSignatureMiddleware}),Object.defineProperty(e,"WebhookStatus",{enumerable:!0,get:()=>_.WebhookStatus}),Object.defineProperty(e,"WebhookSubscription",{enumerable:!0,get:()=>_.WebhookSubscription}),Object.defineProperty(e,"WebhookSubscriptionService",{enumerable:!0,get:()=>_.WebhookSubscriptionService}),Object.defineProperty(e,"computeNextAttempt",{enumerable:!0,get:()=>_.computeNextAttempt}),Object.defineProperty(e,"signWebhookPayload",{enumerable:!0,get:()=>_.signWebhookPayload}),Object.defineProperty(e,"verifyWebhookSignature",{enumerable:!0,get:()=>_.verifyWebhookSignature}),e.AUTH_HANDLER=K,e.AUTH_META_KEY=U,e.ApiOperationFromModel=T,e.ApiParamsFromModel=x,e.ApiProperty=S,e.Auth=H,e.BulkApiOperationFromModel=(e,r,a)=>{const i={GET:[c.BulkCrudOperationKeys.READ_ALL,o.Get],POST:[c.BulkCrudOperationKeys.CREATE_ALL,o.Post],PUT:[c.BulkCrudOperationKeys.UPDATE_ALL,o.Put],PATCH:[c.BulkCrudOperationKeys.UPDATE_ALL,o.Patch],DELETE:[c.BulkCrudOperationKeys.DELETE_ALL,o.Delete]},[s,p]=i[r];return C(e,s)?t.apply(n.ApiExcludeEndpoint()):t.apply(p(a))},e.COMMIT="356fbe1",e.CorsError=he,e.DECAF_ADAPTER_ID=N,e.DECAF_ADAPTER_OPTIONS=B,e.DECAF_CONTEXT_KEY=W,e.DECAF_HANDLERS=L,e.DECAF_MODULE_OPTIONS=F,e.DECAF_ROUTE=j,e.DECAF_TASK_SERVICE_ID="DecafTaskService",e.DECORATORS=E,e.DECORATORS_PREFIX=P,e.DecafAuthHandler=G,e.DecafBody=M,e.DecafController=ee,e.DecafModelController=te,e.DecafParams=I,e.DecafQuery=$,e.DecafRoleAuthHandler=class extends G{constructor(){super()}async authorize(e,r){const o=e.switchToHttp().getRequest(),s=this.parseRequest(o);if(!s)throw new i.AuthorizationError("Unauthenticated");const n=t.Metadata.get(a.Model.get(r),i.PersistenceKeys.AUTH_ROLE);if(!n.includes(s))throw new i.AuthorizationError("Missing role: "+s);return n}},e.DecafWebhooksModule=Te,e.DtoFor=ae,e.FULL_VERSION="0.11.0-356fbe1",e.FromModelController=le,e.HttpVerbToDecorator=q,e.NestBootstraper=class{static get logger(){return this._logger||(this._logger=new o.Logger("NestBootstrap")),this._logger}static initialize(e){return this.app=e,this}static enableLogger(e){return this._logger=e||new o.Logger("NestBootstrap"),this.app.useLogger(this._logger),this}static enableCors(e=[],t=["GET","POST","PUT","DELETE"]){const r="*"===e?"*":e.map(e=>e.trim().toLowerCase()),o={origin:(e,t)=>e?"*"===r||Array.isArray(r)&&r.includes(e.toLowerCase())?t(null,!0):void t(new he(`Origin ${e} not allowed`)):t(null,!0),credentials:!0,methods:t.join(",")};return this.app.enableCors(o),this}static useHelmet(e){try{const t=require("helmet");this.app.use(t(e)),this.logger.log("Helmet middleware enabled successfully.")}catch(e){this.logger.warn("Helmet not installed. Skipping middleware.")}return this}static setupSwagger(e){return new ge(this.app,{title:e.title,description:e.description,version:e.version,path:e.path||"api",persistAuthorization:e.persistAuthorization??!0,assetsPath:e.assetsPath,faviconFilePath:e.faviconPath,topbarIconFilePath:e.topbarIconPath,topbarBgColor:e.topbarBgColor,openApiJsonPath:e.openApiJsonPath,openApiYamlPath:e.openApiYamlPath}).setupSwagger(),this}static useGlobalPipes(...e){return e.length>0&&this.app.useGlobalPipes(...e),this}static useGlobalFilters(...t){return this.app.useGlobalFilters(...t.length>0?t:[new e.DecafExceptionFilter]),this}static useGlobalInterceptors(...e){return e.length>0&&this.app.useGlobalInterceptors(...e),this}static async start(e=Number(process.env.PORT)||3e3,t=void 0,r=!0){this.app.listen(e,t).then(async()=>{if(r){const e=await this.app.getUrl();this.logger.log("\ud83d\ude80 Application is running at: "+e)}})}},e.PACKAGE_NAME=Me,e.RequestToContextTransformer=class{},e.SwaggerBuilder=ge,e.UseDecafFilter=()=>o.UseFilters(new e.DecafExceptionFilter),e.UseDecafHeaders=()=>o.UseInterceptors(e.DecafResponseInterceptor),e.VERSION=xe,e.createApiPropertyDecorator=D,e.createPropertyDecorator=w,e.getEnumType=R,e.getEnumValues=O,e.getModuleFor=ue,e.isOperationBlocked=C,e.isUndefined=v,e.repoForModel=e=>{const t=a.Model.get(e);if(!t)throw new c.InternalError("Failed to find repository for "+e);return i.Repository.forModel(t)},e.requestToContextTransformer=be,e.runMigrations=async()=>{},e.runWebhooksMigrations=async()=>{}},"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("@decaf-ts/decoration"),require("@decaf-ts/injectable-decorators"),require("@nestjs/common"),require("@decaf-ts/decorator-validation"),require("@decaf-ts/core"),require("lodash"),require("@nestjs/swagger"),require("@decaf-ts/db-decorators"),require("tslib"),require("@nestjs/core"),require("@decaf-ts/logging"),require("rxjs"),require("fs"),require("path"),require("yaml"),require("rxjs/operators"),require("@decaf-ts/core/migrations"),require("@decaf-ts/for-http/server")):"function"==typeof define&&define.amd?define(["exports","@decaf-ts/decoration","@decaf-ts/injectable-decorators","@nestjs/common","@decaf-ts/decorator-validation","@decaf-ts/core","lodash","@nestjs/swagger","@decaf-ts/db-decorators","tslib","@nestjs/core","@decaf-ts/logging","rxjs","fs","path","yaml","rxjs/operators","@decaf-ts/core/migrations","@decaf-ts/for-http/server"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self)["for-nest"]={},e.decafTsDecoration,e.decafTsInjectableDecorators,e.nestjsCommon,e.decafTsDecoratorValidation,e.decafTsCore,e.lodash,e.nestjsSwagger,e.decafTsDbDecorators,e.tslib,e.nestjsCore,e.decafTsLogging,e.rxjs,e.fs,e.path,e.yaml,e.operators,e.migrations,e.server);
2
- //# sourceMappingURL=for-nest.cjs.map
1
+ (function(global, factory) {
2
+ typeof exports === "object" && typeof module !== "undefined" ? factory(exports, require("@decaf-ts/decoration"), require("@decaf-ts/injectable-decorators"), require("@nestjs/common"), require("@decaf-ts/decorator-validation"), require("@decaf-ts/core"), require("lodash"), require("@nestjs/swagger"), require("@decaf-ts/db-decorators"), require("tslib"), require("@nestjs/core"), require("@decaf-ts/logging"), require("@decaf-ts/for-http/server"), require("rxjs"), require("fs"), require("path"), require("yaml"), require("rxjs/operators")) : typeof define === "function" && define.amd ? define([ "exports", "@decaf-ts/decoration", "@decaf-ts/injectable-decorators", "@nestjs/common", "@decaf-ts/decorator-validation", "@decaf-ts/core", "lodash", "@nestjs/swagger", "@decaf-ts/db-decorators", "tslib", "@nestjs/core", "@decaf-ts/logging", "@decaf-ts/for-http/server", "rxjs", "fs", "path", "yaml", "rxjs/operators" ], factory) : (global = typeof globalThis !== "undefined" ? globalThis : global || self,
3
+ factory(global["for-nest"] = {}, global.decafTsDecoration, global.decafTsInjectableDecorators, global.nestjsCommon, global.decafTsDecoratorValidation, global.decafTsCore, global.lodash, global.nestjsSwagger, global.decafTsDbDecorators, global.tslib, global.nestjsCore, global.decafTsLogging, global.server, global.rxjs, global.fs, global.path, global.yaml, global.operators));
4
+ })(this, function(exports, decoration, injectableDecorators, common, decoratorValidation, core, lodash, swagger, dbDecorators, tslib, core$1, logging, server, rxjs, fs, path, YAML, operators) {
5
+ "use strict";
6
+ function _interopNamespaceDefault(e) {
7
+ var n = Object.create(null);
8
+ if (e) {
9
+ Object.keys(e).forEach(function(k) {
10
+ if (k !== "default") {
11
+ var d = Object.getOwnPropertyDescriptor(e, k);
12
+ Object.defineProperty(n, k, d.get ? d : {
13
+ enumerable: true,
14
+ get: function() {
15
+ return e[k];
16
+ }
17
+ });
18
+ }
19
+ });
20
+ }
21
+ n.default = e;
22
+ return Object.freeze(n);
23
+ }
24
+ var path__namespace = _interopNamespaceDefault(path);
25
+ const DECORATORS_PREFIX = "swagger";
26
+ const DECORATORS = {
27
+ API_OPERATION: `${DECORATORS_PREFIX}/apiOperation`,
28
+ API_RESPONSE: `${DECORATORS_PREFIX}/apiResponse`,
29
+ API_PRODUCES: `${DECORATORS_PREFIX}/apiProduces`,
30
+ API_CONSUMES: `${DECORATORS_PREFIX}/apiConsumes`,
31
+ API_TAGS: `${DECORATORS_PREFIX}/apiUseTags`,
32
+ API_CALLBACKS: `${DECORATORS_PREFIX}/apiCallbacks`,
33
+ API_PARAMETERS: `${DECORATORS_PREFIX}/apiParameters`,
34
+ API_HEADERS: `${DECORATORS_PREFIX}/apiHeaders`,
35
+ API_MODEL_PROPERTIES: `${DECORATORS_PREFIX}/apiModelProperties`,
36
+ API_MODEL_PROPERTIES_ARRAY: `${DECORATORS_PREFIX}/apiModelPropertiesArray`,
37
+ API_SECURITY: `${DECORATORS_PREFIX}/apiSecurity`,
38
+ API_EXCLUDE_ENDPOINT: `${DECORATORS_PREFIX}/apiExcludeEndpoint`,
39
+ API_EXCLUDE_CONTROLLER: `${DECORATORS_PREFIX}/apiExcludeController`,
40
+ API_EXTRA_MODELS: `${DECORATORS_PREFIX}/apiExtraModels`,
41
+ API_EXTENSION: `${DECORATORS_PREFIX}/apiExtension`,
42
+ API_SCHEMA: `${DECORATORS_PREFIX}/apiSchema`,
43
+ API_DEFAULT_GETTER: `${DECORATORS_PREFIX}/apiDefaultGetter`,
44
+ API_LINK: `${DECORATORS_PREFIX}/apiLink`
45
+ };
46
+ const isConstructor = val => val === "constructor";
47
+ const METADATA_FACTORY_NAME = "_OPENAPI_METADATA_FACTORY";
48
+ const METHOD_METADATA = "method";
49
+ function createMethodDecorator(metakey, metadata, {overrideExisting: overrideExisting} = {
50
+ overrideExisting: true
51
+ }) {
52
+ return (target, key, descriptor) => {
53
+ if (typeof metadata === "object") {
54
+ const prevValue = Reflect.getMetadata(metakey, descriptor.value);
55
+ if (prevValue && !overrideExisting) {
56
+ return descriptor;
57
+ }
58
+ Reflect.defineMetadata(metakey, {
59
+ ...prevValue,
60
+ ...metadata
61
+ }, descriptor.value);
62
+ return descriptor;
63
+ }
64
+ Reflect.defineMetadata(metakey, metadata, descriptor.value);
65
+ return descriptor;
66
+ };
67
+ }
68
+ function createClassDecorator(metakey, metadata = []) {
69
+ return target => {
70
+ const prevValue = Reflect.getMetadata(metakey, target) || [];
71
+ Reflect.defineMetadata(metakey, [ ...prevValue, ...metadata ], target);
72
+ return target;
73
+ };
74
+ }
75
+ function createPropertyDecorator$1(metakey, metadata, overrideExisting = true) {
76
+ return (target, propertyKey) => {
77
+ const properties = Reflect.getMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, target) || [];
78
+ const key = `:${propertyKey}`;
79
+ if (!properties.includes(key)) {
80
+ Reflect.defineMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, [ ...properties, `:${propertyKey}` ], target);
81
+ }
82
+ const existingMetadata = Reflect.getMetadata(metakey, target, propertyKey);
83
+ if (existingMetadata) {
84
+ const newMetadata = lodash.pickBy(metadata, lodash.negate(lodash.isUndefined));
85
+ const metadataToSave = overrideExisting ? {
86
+ ...existingMetadata,
87
+ ...newMetadata
88
+ } : {
89
+ ...newMetadata,
90
+ ...existingMetadata
91
+ };
92
+ Reflect.defineMetadata(metakey, metadataToSave, target, propertyKey);
93
+ } else {
94
+ const type = target?.constructor?.[METADATA_FACTORY_NAME]?.()[propertyKey]?.type ?? Reflect.getMetadata("design:type", target, propertyKey);
95
+ Reflect.defineMetadata(metakey, {
96
+ type: type,
97
+ ...lodash.pickBy(metadata, lodash.negate(lodash.isUndefined))
98
+ }, target, propertyKey);
99
+ }
100
+ };
101
+ }
102
+ function createMixedDecorator(metakey, metadata) {
103
+ return (target, key, descriptor) => {
104
+ if (descriptor) {
105
+ let metadatas;
106
+ if (Array.isArray(metadata)) {
107
+ const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || [];
108
+ metadatas = [ ...previousMetadata, ...metadata ];
109
+ } else {
110
+ const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || {};
111
+ metadatas = {
112
+ ...previousMetadata,
113
+ ...metadata
114
+ };
115
+ }
116
+ Reflect.defineMetadata(metakey, metadatas, descriptor.value);
117
+ return descriptor;
118
+ }
119
+ let metadatas;
120
+ if (Array.isArray(metadata)) {
121
+ const previousMetadata = Reflect.getMetadata(metakey, target) || [];
122
+ metadatas = [ ...previousMetadata, ...metadata ];
123
+ } else {
124
+ const previousMetadata = Reflect.getMetadata(metakey, target) || {};
125
+ metadatas = Object.assign(Object.assign({}, previousMetadata), metadata);
126
+ }
127
+ Reflect.defineMetadata(metakey, metadatas, target);
128
+ return target;
129
+ };
130
+ }
131
+ function createParamDecorator(metadata, initial) {
132
+ return (target, key, descriptor) => {
133
+ const paramOptions = {
134
+ ...initial,
135
+ ...lodash.pickBy(metadata, lodash.negate(lodash.isUndefined))
136
+ };
137
+ if (descriptor) {
138
+ const parameters = Reflect.getMetadata(DECORATORS.API_PARAMETERS, descriptor.value) || [];
139
+ Reflect.defineMetadata(DECORATORS.API_PARAMETERS, [ ...parameters, paramOptions ], descriptor.value);
140
+ return descriptor;
141
+ }
142
+ if (typeof target === "object") {
143
+ return target;
144
+ }
145
+ const propertyKeys = Object.getOwnPropertyNames(target.prototype);
146
+ for (const propertyKey of propertyKeys) {
147
+ if (isConstructor(propertyKey)) {
148
+ continue;
149
+ }
150
+ const methodDescriptor = Object.getOwnPropertyDescriptor(target.prototype, propertyKey);
151
+ if (!methodDescriptor) {
152
+ continue;
153
+ }
154
+ const isApiMethod = Reflect.hasMetadata(METHOD_METADATA, methodDescriptor.value);
155
+ if (!isApiMethod) {
156
+ continue;
157
+ }
158
+ const parameters = Reflect.getMetadata(DECORATORS.API_PARAMETERS, methodDescriptor.value) || [];
159
+ Reflect.defineMetadata(DECORATORS.API_PARAMETERS, [ ...parameters, paramOptions ], methodDescriptor.value);
160
+ }
161
+ };
162
+ }
163
+ function getTypeIsArrayTuple(input, isArrayFlag) {
164
+ if (!input) {
165
+ return [ input, isArrayFlag ];
166
+ }
167
+ if (isArrayFlag) {
168
+ return [ input, isArrayFlag ];
169
+ }
170
+ const isInputArray = lodash.isArray(input);
171
+ const type = isInputArray ? input[0] : input;
172
+ return [ type, isInputArray ];
173
+ }
174
+ const isUndefined = obj => typeof obj === "undefined";
175
+ function getEnumValues(enumType) {
176
+ if (typeof enumType === "function") {
177
+ return getEnumValues(enumType());
178
+ }
179
+ if (Array.isArray(enumType)) {
180
+ return enumType;
181
+ }
182
+ if (typeof enumType !== "object") {
183
+ return [];
184
+ }
185
+ const numericValues = Object.values(enumType).filter(value => typeof value === "number").map(value => value.toString());
186
+ return Object.keys(enumType).filter(key => !numericValues.includes(key)).map(key => enumType[key]);
187
+ }
188
+ function getEnumType(values) {
189
+ const hasString = values.filter(lodash.isString).length > 0;
190
+ return hasString ? "string" : "number";
191
+ }
192
+ function createPropertyDecorator(metakey, metadata, overrideExisting = true) {
193
+ return (target, propertyKey) => {
194
+ const properties = Reflect.getMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, target) || [];
195
+ const key = `:${propertyKey}`;
196
+ if (!properties.includes(key)) {
197
+ Reflect.defineMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, [ ...properties, `:${propertyKey}` ], target);
198
+ }
199
+ const existingMetadata = Reflect.getMetadata(metakey, target, propertyKey);
200
+ if (existingMetadata) {
201
+ const newMetadata = lodash.pickBy(metadata, lodash.negate(isUndefined));
202
+ const metadataToSave = overrideExisting ? {
203
+ ...existingMetadata,
204
+ ...newMetadata
205
+ } : {
206
+ ...newMetadata,
207
+ ...existingMetadata
208
+ };
209
+ Reflect.defineMetadata(metakey, metadataToSave, target, propertyKey);
210
+ } else {
211
+ const type = target?.constructor?.[METADATA_FACTORY_NAME]?.()[propertyKey]?.type ?? Reflect.getMetadata("design:type", target, propertyKey);
212
+ Reflect.defineMetadata(metakey, {
213
+ type: type,
214
+ required: false,
215
+ ...lodash.pickBy(metadata, lodash.negate(isUndefined))
216
+ }, target, propertyKey);
217
+ }
218
+ };
219
+ }
220
+ const isEnumArray = opts => opts.isArray && "enum" in opts && opts.enum !== undefined;
221
+ function ApiProperty(options = {}) {
222
+ return createApiPropertyDecorator(options);
223
+ }
224
+ function createApiPropertyDecorator(options = {}, overrideExisting = true) {
225
+ const [type, isArray] = getTypeIsArrayTuple(options.type, options.isArray);
226
+ options = {
227
+ ...options,
228
+ type: type,
229
+ isArray: isArray
230
+ };
231
+ if (isEnumArray(options)) {
232
+ options.type = "array";
233
+ const enumValues = getEnumValues(options.enum);
234
+ options.items = {
235
+ type: getEnumType(enumValues),
236
+ enum: enumValues
237
+ };
238
+ delete options.enum;
239
+ } else if ("enum" in options && options.enum !== undefined) {
240
+ const enumValues = getEnumValues(options.enum);
241
+ options.enum = enumValues;
242
+ options.type = getEnumType(enumValues);
243
+ }
244
+ if (Array.isArray(options.type)) {
245
+ options.type = "array";
246
+ options.items = {
247
+ type: "array",
248
+ items: {
249
+ type: options.type[0]
250
+ }
251
+ };
252
+ }
253
+ return createPropertyDecorator(DECORATORS.API_MODEL_PROPERTIES, options, overrideExisting);
254
+ }
255
+ function ApiOperationFromModel(ModelConstructor, verb, path) {
256
+ const httpToCrud = {
257
+ GET: [ dbDecorators.OperationKeys.READ, common.Get ],
258
+ POST: [ dbDecorators.OperationKeys.CREATE, common.Post ],
259
+ PUT: [ dbDecorators.OperationKeys.UPDATE, common.Put ],
260
+ PATCH: [ dbDecorators.OperationKeys.UPDATE, common.Patch ],
261
+ DELETE: [ dbDecorators.OperationKeys.DELETE, common.Delete ]
262
+ };
263
+ const [crudOp, HttpMethodDecorator] = httpToCrud[verb];
264
+ const target = resolveBlockTarget(verb, path);
265
+ return target ? core.isOperationBlocked(ModelConstructor, target.kind, target.value) ? decoration.apply(swagger.ApiExcludeEndpoint()) : decoration.apply(HttpMethodDecorator(path)) : core.isOperationBlocked(ModelConstructor, crudOp) ? decoration.apply(swagger.ApiExcludeEndpoint()) : decoration.apply(HttpMethodDecorator(path));
266
+ }
267
+ function BulkApiOperationFromModel(ModelConstructor, verb, path) {
268
+ const httpToCrud = {
269
+ GET: [ dbDecorators.BulkCrudOperationKeys.READ_ALL, common.Get ],
270
+ POST: [ dbDecorators.BulkCrudOperationKeys.CREATE_ALL, common.Post ],
271
+ PUT: [ dbDecorators.BulkCrudOperationKeys.UPDATE_ALL, common.Put ],
272
+ PATCH: [ dbDecorators.BulkCrudOperationKeys.UPDATE_ALL, common.Patch ],
273
+ DELETE: [ dbDecorators.BulkCrudOperationKeys.DELETE_ALL, common.Delete ]
274
+ };
275
+ const [crudOp, HttpMethodDecorator] = httpToCrud[verb];
276
+ const target = path ? resolveBlockTarget(verb, path) : undefined;
277
+ return target ? core.isOperationBlocked(ModelConstructor, target.kind, target.value) ? decoration.apply(swagger.ApiExcludeEndpoint()) : decoration.apply(HttpMethodDecorator(path)) : core.isOperationBlocked(ModelConstructor, "bulk", crudOp) ? decoration.apply(swagger.ApiExcludeEndpoint()) : decoration.apply(HttpMethodDecorator(path));
278
+ }
279
+ function resolveBlockTarget(verb, path) {
280
+ if (!path) return undefined;
281
+ const normalized = path.replace(/^\/+|\/+$/g, "");
282
+ const statementTargets = {
283
+ "listBy/:key": core.PreparedStatementKeys.LIST_BY,
284
+ "paginateBy/:key/:page": core.PreparedStatementKeys.PAGE_BY,
285
+ "find/:value": core.PreparedStatementKeys.FIND,
286
+ "page/:value": core.PreparedStatementKeys.PAGE,
287
+ "findOneBy/:key/:value": core.PreparedStatementKeys.FIND_ONE_BY,
288
+ "findBy/:key/:value": core.PreparedStatementKeys.FIND_BY,
289
+ "statement/:method/*args": "statement",
290
+ "countOf/:field": core.PreparedStatementKeys.COUNT_OF,
291
+ "maxOf/:field": core.PreparedStatementKeys.MAX_OF,
292
+ "minOf/:field": core.PreparedStatementKeys.MIN_OF,
293
+ "avgOf/:field": core.PreparedStatementKeys.AVG_OF,
294
+ "sumOf/:field": core.PreparedStatementKeys.SUM_OF,
295
+ "distinctOf/:field": core.PreparedStatementKeys.DISTINCT_OF,
296
+ "groupOf/:field": core.PreparedStatementKeys.GROUP_OF
297
+ };
298
+ if (normalized.startsWith("query/")) {
299
+ return {
300
+ kind: "query",
301
+ value: normalized.replace(/^query\//, "")
302
+ };
303
+ }
304
+ const statementValue = statementTargets[normalized];
305
+ if (statementValue) {
306
+ return {
307
+ kind: "statement",
308
+ value: statementValue
309
+ };
310
+ }
311
+ if (verb === "GET" && normalized === "") {
312
+ return {
313
+ kind: "crud",
314
+ value: dbDecorators.OperationKeys.READ
315
+ };
316
+ }
317
+ return undefined;
318
+ }
319
+ function ApiParamsFromModel(props = []) {
320
+ const decorators = props.map(p => swagger.ApiParam({
321
+ name: p.name,
322
+ description: p.description ?? `Path parameter: ${p.name}`,
323
+ required: p.required ?? true,
324
+ type: p.type ?? String
325
+ }));
326
+ return common.applyDecorators(...decorators);
327
+ }
328
+ const DecafBody = common.createParamDecorator((data, ctx) => {
329
+ const request = ctx.switchToHttp().getRequest();
330
+ const body = request.body;
331
+ const controller = ctx.getClass();
332
+ const ModelConstr = controller.class;
333
+ if (!ModelConstr) {
334
+ throw new common.InternalServerErrorException(`ModelConstructor not found on controller ${controller.name}. Ensure the controller was created via FromModelController.`);
335
+ }
336
+ if (!body) return body;
337
+ if (Array.isArray(body)) {
338
+ return body.map(item => new ModelConstr(item));
339
+ }
340
+ return new ModelConstr(body);
341
+ });
342
+ const OrderedParams = common.createParamDecorator((order, ctx) => {
343
+ const req = ctx.switchToHttp().getRequest();
344
+ const original = req?.params ?? {};
345
+ const orderList = order ?? Object.keys(original);
346
+ const ordered = orderList.map(k => original[k]);
347
+ return {
348
+ raw: original,
349
+ valuesInOrder: ordered,
350
+ keysInOrder: orderList
351
+ };
352
+ });
353
+ function DecafParams(props = []) {
354
+ const order = props.map(p => p.name);
355
+ return OrderedParams(order);
356
+ }
357
+ const DecafQuery = common.createParamDecorator((_, ctx) => {
358
+ const req = ctx.switchToHttp().getRequest();
359
+ const query = req.query ?? {};
360
+ const parsed = {
361
+ ...query
362
+ };
363
+ if (parsed.limit !== undefined) {
364
+ const n = Number(parsed.limit);
365
+ if (!Number.isNaN(n)) parsed.limit = n;
366
+ }
367
+ if (parsed.offset !== undefined) {
368
+ const n = Number(parsed.offset);
369
+ if (!Number.isNaN(n)) parsed.offset = n;
370
+ }
371
+ if (parsed.bookmark !== undefined) {
372
+ const n = Number(parsed.bookmark);
373
+ parsed.bookmark = Number.isNaN(n) ? parsed.bookmark : n;
374
+ }
375
+ return parsed;
376
+ });
377
+ const DECAF_MODULE_OPTIONS = "DecafModuleOptions";
378
+ const DECAF_ADAPTER_ID = "DecafAdapter";
379
+ const DECAF_TASK_SERVICE_ID = "DecafTaskService";
380
+ const DECAF_ROUTE = "DecafRoute";
381
+ const DECAF_HANDLERS = Symbol("DecafHandlers");
382
+ const DECAF_EXPOSE = "DecafExpose";
383
+ const DECAF_CONTROLLER_CONFIG = "DecafControllerConfig";
384
+ const DECAF_CONTEXT_KEY = Symbol("decaf:context");
385
+ function expose(...flavours) {
386
+ return function expose(target) {
387
+ decoration.Metadata.set(target, DECAF_EXPOSE, flavours.length ? flavours : true);
388
+ return target;
389
+ };
390
+ }
391
+ function controllerConfig(config) {
392
+ return function controllerConfigDecorator(target) {
393
+ decoration.Metadata.set(target, DECAF_CONTROLLER_CONFIG, config);
394
+ return target;
395
+ };
396
+ }
397
+ const AUTH_HANDLER = Symbol("AUTH_HANDLER");
398
+ const AUTH_META_KEY = "auth:meta";
399
+ const IS_PUBLIC_KEY = "isPublic";
400
+ const REQUIRED_ROLES_KEY = "requiredRoles";
401
+ const SKIP_MODEL_ROLES_KEY = "skipModelRoles";
402
+ exports.DecafRequestContext = class DecafRequestContext extends server.RequestContext {
403
+ constructor(req) {
404
+ super({
405
+ headersOf: request => request?.headers || undefined
406
+ }, req);
407
+ this.req = req;
408
+ this.uuid = core.UUID.instance.generate();
409
+ this.request = req;
410
+ }
411
+ put(record) {
412
+ let overrides;
413
+ try {
414
+ overrides = this.get("overrides");
415
+ } catch (e) {
416
+ overrides = {};
417
+ }
418
+ this.accumulate({
419
+ overrides: Object.assign(overrides, record)
420
+ });
421
+ }
422
+ };
423
+ exports.DecafRequestContext = tslib.__decorate([ common.Injectable({
424
+ scope: common.Scope.REQUEST
425
+ }), tslib.__param(0, common.Inject(core$1.REQUEST)), tslib.__metadata("design:paramtypes", [ Object ]) ], exports.DecafRequestContext);
426
+ exports.AuthInterceptor = class AuthInterceptor {
427
+ constructor(reflector, requestContext, authHandler) {
428
+ this.reflector = reflector;
429
+ this.requestContext = requestContext;
430
+ this.authHandler = authHandler;
431
+ }
432
+ async intercept(ctx, next) {
433
+ const log = logging.Logging.for(this).for(this.intercept);
434
+ const isPublic = this.reflector.getAllAndOverride(IS_PUBLIC_KEY, [ ctx.getHandler(), ctx.getClass() ]);
435
+ const modelName = this.reflector.get(AUTH_META_KEY, ctx.getHandler()) ?? this.reflector.get(AUTH_META_KEY, ctx.getClass());
436
+ const requiredRoles = this.reflector.getAllAndOverride(REQUIRED_ROLES_KEY, [ ctx.getHandler(), ctx.getClass() ]);
437
+ const skipModelRoles = this.reflector.getAllAndOverride(SKIP_MODEL_ROLES_KEY, [ ctx.getHandler(), ctx.getClass() ]);
438
+ const effectiveModel = skipModelRoles ? undefined : modelName;
439
+ log.verbose(`Intercepted request${modelName ? ` for ${modelName}` : ""}`);
440
+ if (isPublic) {
441
+ log.debug(`Public route — skipping auth`);
442
+ } else if (this.authHandler) {
443
+ await this.authHandler.authorize(ctx, effectiveModel, requiredRoles, this.requestContext);
444
+ } else {
445
+ log.debug(`No auth handler registered`);
446
+ }
447
+ await this.applyTransformers();
448
+ const user = this.requestContext.getOrUndefined("user");
449
+ const organization = this.requestContext.getOrUndefined("organization");
450
+ if (user || organization) {
451
+ const currentLog = this.requestContext.get("logger");
452
+ const childLog = currentLog.for({
453
+ user: user,
454
+ organization: organization
455
+ });
456
+ this.requestContext.accumulate({
457
+ logger: childLog
458
+ });
459
+ }
460
+ return next.handle();
461
+ }
462
+ async applyTransformers() {
463
+ const flavours = core.Adapter.flavoursToTransform();
464
+ if (!flavours) return;
465
+ for (const flavour of flavours) {
466
+ const transformer = core.Adapter.transformerFor(flavour);
467
+ if (!transformer) continue;
468
+ const from = await transformer.from(this.requestContext);
469
+ if (from) this.requestContext.accumulate(from);
470
+ }
471
+ }
472
+ };
473
+ exports.AuthInterceptor = tslib.__decorate([ common.Injectable({
474
+ scope: common.Scope.REQUEST
475
+ }), tslib.__param(2, common.Optional()), tslib.__param(2, common.Inject(AUTH_HANDLER)), tslib.__metadata("design:paramtypes", [ core$1.Reflector, exports.DecafRequestContext, Object ]) ], exports.AuthInterceptor);
476
+ function Auth(model) {
477
+ const resource = model ? typeof model === "string" ? model : model.name : undefined;
478
+ const decs = [ swagger.ApiBearerAuth(), common.UseInterceptors(exports.AuthInterceptor) ];
479
+ if (resource) decs.push(common.SetMetadata(AUTH_META_KEY, resource));
480
+ return common.applyDecorators(...decs);
481
+ }
482
+ function Public() {
483
+ return common.applyDecorators(common.SetMetadata(IS_PUBLIC_KEY, true));
484
+ }
485
+ function RequireRoles(...roles) {
486
+ return common.applyDecorators(swagger.ApiSecurity("bearer"), common.SetMetadata(REQUIRED_ROLES_KEY, roles), common.UseInterceptors(exports.AuthInterceptor));
487
+ }
488
+ function HttpVerbToDecorator(verb) {
489
+ const httpToCrud = {
490
+ GET: common.Get,
491
+ POST: common.Post,
492
+ PUT: common.Put,
493
+ PATCH: common.Patch,
494
+ DELETE: common.Delete
495
+ };
496
+ const decorator = httpToCrud[verb];
497
+ if (!decorator) {
498
+ throw new Error(`Unsupported HTTP verb "${verb}". No NestJS decorator mapping was found.`);
499
+ }
500
+ return decorator;
501
+ }
502
+ const extractPathParams = routePath => routePath.split("/").filter(p => p.startsWith(":")).map(p => p.slice(1));
503
+ const apiParamSpec = name => ({
504
+ name: name,
505
+ description: `${name} parameter for the query`,
506
+ required: true,
507
+ type: String
508
+ });
509
+ function getApiDecorators(methodName, routePath, httpVerb, includeQueryParams = false) {
510
+ const NestHttpRouteDec = HttpVerbToDecorator(httpVerb);
511
+ const apiPathParams = extractPathParams(routePath).map(apiParamSpec);
512
+ const swaggerQueryParams = [];
513
+ if (httpVerb === "GET" && includeQueryParams) {
514
+ swaggerQueryParams.push(swagger.ApiQuery({
515
+ name: "direction",
516
+ required: false,
517
+ enum: core.OrderDirection,
518
+ description: "the sort order when applicable"
519
+ }), swagger.ApiQuery({
520
+ name: "limit",
521
+ required: false,
522
+ description: "limit or page size when applicable"
523
+ }), swagger.ApiQuery({
524
+ name: "offset",
525
+ required: false,
526
+ description: "offset or bookmark when applicable"
527
+ }));
528
+ }
529
+ return {
530
+ method: [ NestHttpRouteDec(routePath), ...apiPathParams.map(swagger.ApiParam), ...swaggerQueryParams, swagger.ApiOperation({
531
+ summary: `Retrieve records using according to "${methodName}".`
532
+ }), swagger.ApiOkResponse({
533
+ description: `Result successfully retrieved.`
534
+ }), swagger.ApiNoContentResponse({
535
+ description: `No content returned by the method.`
536
+ }) ],
537
+ params: [ DecafParams(apiPathParams), common.Query() ]
538
+ };
539
+ }
540
+ function applyApiDecorators(target, methodName, descriptor, decorators) {
541
+ const proto = target?.prototype ?? target;
542
+ decorators.method.forEach(d => d(proto, methodName, descriptor));
543
+ decorators.params?.forEach((d, index) => d(proto, methodName, index));
544
+ }
545
+ function resolvePersistenceMethod(persistence, methodName, ...args) {
546
+ if (persistence instanceof core.ModelService) {
547
+ return typeof persistence[methodName] === "function" ? persistence[methodName](...args) : persistence.statement(methodName, ...args);
548
+ }
549
+ if (typeof persistence[methodName] === "function") return persistence[methodName](...args);
550
+ throw new Error(`Persistence method "${methodName}" not found on ${persistence?.constructor?.name}`);
551
+ }
552
+ function createRouteHandler(methodName) {
553
+ return async function(pathParams, queryParams) {
554
+ const log = this.log.for(methodName);
555
+ try {
556
+ log.debug(`Invoking persistence method "${methodName}" given parameters: ${JSON.stringify(pathParams.valuesInOrder)}`);
557
+ const {direction: direction, limit: limit, offset: offset} = queryParams;
558
+ return await resolvePersistenceMethod(this.persistence(), methodName, ...pathParams.valuesInOrder, direction, limit, offset);
559
+ } catch (e) {
560
+ log.error(`Custom query "${methodName}" failed`, e);
561
+ throw e;
562
+ }
563
+ };
564
+ }
565
+ function defineRouteMethod(ControllerClass, methodName, handler) {
566
+ Object.defineProperty(ControllerClass.prototype || ControllerClass, methodName, {
567
+ value: handler,
568
+ writable: false
569
+ });
570
+ return Object.getOwnPropertyDescriptor(ControllerClass.prototype || ControllerClass, methodName);
571
+ }
572
+ class DecafAuthHandler extends server.AuthHandler {
573
+ parseRequest(req) {
574
+ const userRole = req.headers.authorization?.split(" ")[1];
575
+ return userRole;
576
+ }
577
+ extractFromAuth(ctx) {
578
+ const req = ctx.switchToHttp().getRequest();
579
+ const userRole = this.parseRequest(req);
580
+ if (!userRole) throw new core.AuthorizationError("Unauthenticated");
581
+ return {
582
+ user: userRole,
583
+ roles: [ userRole ]
584
+ };
585
+ }
586
+ }
587
+ class DecafRoleAuthHandler extends DecafAuthHandler {}
588
+ core.Adapter.transformerFor = function toContextFlags(adapter) {
589
+ const alias = typeof adapter === "string" ? adapter : adapter.alias;
590
+ return decoration.Metadata["innerGet"](Symbol.for("transformers"), alias);
591
+ }.bind(core.Adapter);
592
+ core.Adapter.flavoursToTransform = function requestTransformers() {
593
+ const meta = decoration.Metadata["innerGet"](Symbol.for("transformers"));
594
+ if (!meta) return undefined;
595
+ return Object.keys(meta);
596
+ }.bind(core.Adapter);
597
+ core.Context.prototype.toResponse = function toResponse(res) {
598
+ const pending = this.pending();
599
+ if (pending) res.header("x-pending-task", JSON.stringify(pending));
600
+ return res;
601
+ };
602
+ const prototype = decoratorValidation.ModelBuilder.prototype;
603
+ if (!prototype.decorateClass) {
604
+ prototype.decorateClass = function(decorator) {
605
+ if (!this._classDecorators) {
606
+ this._classDecorators = [];
607
+ }
608
+ this._classDecorators.push(decorator);
609
+ return this;
610
+ };
611
+ }
612
+ prototype.Auth = function(model) {
613
+ return this.decorateClass(Auth(model));
614
+ };
615
+ if (!prototype.__hasClassDecoratorSupport) {
616
+ const originalBuild = prototype.build;
617
+ prototype.build = function() {
618
+ let result = originalBuild.call(this);
619
+ const decorators = this._classDecorators;
620
+ if (decorators?.length) {
621
+ for (const decorator of decorators) {
622
+ const decorated = decorator(result);
623
+ if (decorated) {
624
+ result = decorated;
625
+ }
626
+ }
627
+ }
628
+ return result;
629
+ };
630
+ prototype.__hasClassDecoratorSupport = true;
631
+ }
632
+ var DecafRequestHandlerInterceptor_1;
633
+ exports.DecafRequestHandlerInterceptor = DecafRequestHandlerInterceptor_1 = class DecafRequestHandlerInterceptor {
634
+ constructor(requestContext, executor) {
635
+ this.requestContext = requestContext;
636
+ this.executor = executor;
637
+ }
638
+ async contextualize(req) {
639
+ const headers = req.headers;
640
+ const flags = {
641
+ headers: headers,
642
+ overrides: {}
643
+ };
644
+ const ip = extractIp(req);
645
+ const logger = logging.Logging.get().for({
646
+ ip: ip
647
+ });
648
+ this.requestContext.accumulate(Object.assign({}, core.DefaultAdapterFlags, {
649
+ logger: logger,
650
+ timestamp: new Date,
651
+ operation: `${req.method} ${req.url}`
652
+ }, flags));
653
+ }
654
+ async intercept(context, next) {
655
+ const req = context.switchToHttp().getRequest();
656
+ const res = context.switchToHttp().getResponse();
657
+ const log = logging.Logging.for(DecafRequestHandlerInterceptor_1).for(this.intercept);
658
+ log.debug(`CONTEXT ${this.requestContext.uuid} - request: ${req.method} ${req.url}`);
659
+ await this.contextualize(req);
660
+ log.debug(`CONTEXT ${this.requestContext.uuid} contextualized - request: ${req.method} ${req.url}`);
661
+ await this.executor.exec(req, res);
662
+ log.debug(`CONTEXT ${this.requestContext.uuid} executors finished - request: ${req.method} ${req.url}`);
663
+ return next.handle();
664
+ }
665
+ };
666
+ exports.DecafRequestHandlerInterceptor = DecafRequestHandlerInterceptor_1 = tslib.__decorate([ common.Injectable({
667
+ scope: common.Scope.REQUEST
668
+ }), tslib.__metadata("design:paramtypes", [ exports.DecafRequestContext, exports.DecafHandlerExecutor ]) ], exports.DecafRequestHandlerInterceptor);
669
+ function extractIp(req) {
670
+ const headers = req.headers;
671
+ function parseIpHeader(value) {
672
+ if (!value) return undefined;
673
+ const candidate = Array.isArray(value) ? value[0] : value;
674
+ return candidate.split(",").map(segment => segment.trim()).filter(Boolean)[0];
675
+ }
676
+ return parseIpHeader(headers?.["x-forwarded-for"]) ?? parseIpHeader(headers?.["x-real-ip"]) ?? parseIpHeader(headers?.["X-Forwarded-For"]) ?? parseIpHeader(headers?.["X-Real-IP"]) ?? req.ip;
677
+ }
678
+ var DecafAuthModule_1;
679
+ exports.DecafAuthModule = DecafAuthModule_1 = class DecafAuthModule {
680
+ static forRoot(options = {}) {
681
+ const providers = [ exports.AuthInterceptor, exports.DecafRequestHandlerInterceptor, {
682
+ provide: core$1.APP_INTERCEPTOR,
683
+ useClass: exports.DecafRequestHandlerInterceptor
684
+ } ];
685
+ if (options.handler) {
686
+ providers.push(options.handler);
687
+ providers.push({
688
+ provide: AUTH_HANDLER,
689
+ useClass: options.handler
690
+ });
691
+ }
692
+ if (options.global) {
693
+ providers.push({
694
+ provide: core$1.APP_INTERCEPTOR,
695
+ useExisting: exports.AuthInterceptor
696
+ });
697
+ }
698
+ return {
699
+ module: DecafAuthModule_1,
700
+ global: options.global ?? false,
701
+ providers: providers,
702
+ exports: [ exports.AuthInterceptor, AUTH_HANDLER ]
703
+ };
704
+ }
705
+ };
706
+ exports.DecafAuthModule = DecafAuthModule_1 = tslib.__decorate([ common.Module({}) ], exports.DecafAuthModule);
707
+ var DecafHandlerExecutor_1;
708
+ exports.DecafHandlerExecutor = DecafHandlerExecutor_1 = class DecafHandlerExecutor {
709
+ constructor(handlers, context) {
710
+ this.handlers = handlers;
711
+ this.context = context;
712
+ }
713
+ async exec(req, res) {
714
+ const log = logging.Logging.for(DecafHandlerExecutor_1.name).for(this.exec);
715
+ log.debug(`CONTEXT ${this.context.uuid} running ${this.handlers.length} handlers for request ${req.method} ${req.url}`);
716
+ for (const handler of this.handlers) {
717
+ await handler.handle(this.context, req, res);
718
+ }
719
+ }
720
+ };
721
+ exports.DecafHandlerExecutor = DecafHandlerExecutor_1 = tslib.__decorate([ common.Injectable({
722
+ scope: common.Scope.REQUEST
723
+ }), tslib.__param(0, common.Inject(DECAF_HANDLERS)), tslib.__metadata("design:paramtypes", [ Array, exports.DecafRequestContext ]) ], exports.DecafHandlerExecutor);
724
+ exports.DecafResponseInterceptor = class DecafResponseInterceptor {
725
+ constructor(ctx) {
726
+ this.ctx = ctx;
727
+ }
728
+ intercept(context, next) {
729
+ let response = context.switchToHttp().getResponse();
730
+ return next.handle().pipe(rxjs.tap(data => {
731
+ response = this.ctx.toResponse(response);
732
+ }));
733
+ }
734
+ };
735
+ exports.DecafResponseInterceptor = tslib.__decorate([ common.Injectable(), tslib.__metadata("design:paramtypes", [ exports.DecafRequestContext ]) ], exports.DecafResponseInterceptor);
736
+ class DecafController extends server.DecafController {
737
+ constructor(clientContext, _name) {
738
+ super(clientContext);
739
+ this.clientContext = clientContext;
740
+ }
741
+ }
742
+ class DecafModelController extends DecafController {
743
+ constructor(clientContext, _name) {
744
+ super(clientContext, _name);
745
+ this.clientContext = clientContext;
746
+ }
747
+ persistence(ctx) {
748
+ if (!this._persistence) try {
749
+ this._persistence = core.Service.get(this.class);
750
+ } catch (e) {
751
+ try {
752
+ this._persistence = core.ModelService.getService(this.class);
753
+ } catch (e) {
754
+ this._persistence = core.Repository.forModel(this.class);
755
+ }
756
+ }
757
+ const overrides = this.clientContext.toOverrides();
758
+ return ctx ? this._persistence instanceof core.Repository ? this._persistence.override(overrides) : this._persistence.for(overrides) : this._persistence;
759
+ }
760
+ }
761
+ const dtoCache = new Map;
762
+ function DtoFor(op, model, stack = new Set) {
763
+ if (!core.TransactionOperationKeys.includes(op)) {
764
+ return model;
765
+ }
766
+ const cache = getDtoCache(op);
767
+ const cached = cache.get(model);
768
+ if (cached) return cached;
769
+ const isUpdateOp = [ dbDecorators.OperationKeys.UPDATE, dbDecorators.BulkCrudOperationKeys.UPDATE_ALL ].includes(op);
770
+ class DynamicDTO {}
771
+ cache.set(model, DynamicDTO);
772
+ Object.defineProperty(DynamicDTO, "name", {
773
+ value: `${logging.toPascalCase(model.name)}${logging.toPascalCase(op)}DTO`
774
+ });
775
+ const pkProp = (() => {
776
+ try {
777
+ return decoratorValidation.Model.pk(model);
778
+ } catch {
779
+ return undefined;
780
+ }
781
+ })();
782
+ const pkPropsMetadata = pkProp ? decoratorValidation.Model.pkProps(model) : undefined;
783
+ const pkDesignType = pkProp ? Reflect.getMetadata("design:type", model.prototype, pkProp) : undefined;
784
+ const pkTypeIsNumeric = pkDesignType === Number || pkDesignType === BigInt;
785
+ const pkIsGenerated = !!pkPropsMetadata?.generated || (pkProp ? decoratorValidation.Model.generatedBySequence(model, pkProp) || isPropertyGeneratedAcrossInheritance(model, pkProp) || pkTypeIsNumeric : false);
786
+ const allProps = Array.from(new Set(decoration.Metadata.properties(model) || []));
787
+ const relations = new Set(decoratorValidation.Model.relations(model) || []);
788
+ const scalarProps = [];
789
+ for (const prop of allProps) {
790
+ if (!prop) continue;
791
+ if (relations.has(prop)) continue;
792
+ if (prop === pkProp && !isUpdateOp && pkIsGenerated) continue;
793
+ if (prop !== pkProp && isPropertyGeneratedAcrossInheritance(model, prop)) continue;
794
+ scalarProps.push(prop);
795
+ }
796
+ for (const prop of scalarProps) {
797
+ const validation = getValidationAcrossInheritance(model, prop);
798
+ const isRequired = !!validation?.[decoratorValidation.ValidationKeys.REQUIRED];
799
+ const typeHint = getTypeAcrossInheritance(model, prop) ?? Reflect.getMetadata("design:type", model.prototype, prop);
800
+ const apiOptions = {
801
+ required: isRequired
802
+ };
803
+ if (typeHint) apiOptions.type = typeHint;
804
+ ApiProperty(apiOptions)(DynamicDTO.prototype, prop);
805
+ const designType = Reflect.getMetadata("design:type", model.prototype, prop) ?? typeHint;
806
+ if (typeof designType !== "undefined") {
807
+ Reflect.defineMetadata("design:type", designType, DynamicDTO.prototype, prop);
808
+ }
809
+ Object.defineProperty(DynamicDTO.prototype, prop, {
810
+ value: undefined,
811
+ writable: true,
812
+ enumerable: true,
813
+ configurable: true
814
+ });
815
+ }
816
+ for (const relation of relations) {
817
+ const relationMeta = decoration.Metadata.relations(model, relation);
818
+ if (!relationMeta) {
819
+ throw new dbDecorators.InternalError(`Metadata for relation ${relation} not found`);
820
+ }
821
+ let relationType = relationMeta.class;
822
+ if (typeof relationType === "function" && !relationType.name) {
823
+ relationType = relationType();
824
+ }
825
+ if (!relationType || typeof relationType !== "function") {
826
+ throw new dbDecorators.InternalError(`Type for relation ${relation} not found`);
827
+ }
828
+ if (!decoratorValidation.Model.get(relationType.name)) {
829
+ continue;
830
+ }
831
+ const meta = decoration.Metadata.validationFor(model, relation);
832
+ const isArray = !!meta?.[decoratorValidation.ValidationKeys.LIST];
833
+ const isRequired = !!meta?.[decoratorValidation.ValidationKeys.REQUIRED];
834
+ const isCircular = stack.has(relationType);
835
+ if (isCircular) {
836
+ const pkTypeName = getPkOpenApiType(relationType);
837
+ addRelationPkRef(DynamicDTO, relation, pkTypeName, isArray, isRequired);
838
+ continue;
839
+ }
840
+ const relationDto = DtoFor(op, relationType, new Set(stack).add(model));
841
+ if (isUpdateOp) {
842
+ addRelationUpdate(DynamicDTO, relation, relationType, relationDto, isArray, isRequired);
843
+ } else {
844
+ addRelation(DynamicDTO, relation, relationDto, isArray, isRequired);
845
+ }
846
+ }
847
+ return DynamicDTO;
848
+ }
849
+ function addRelation(DtoClass, relation, relationDto, isArray, isRequired) {
850
+ const apiOptions = {
851
+ type: relationDto,
852
+ required: isRequired,
853
+ isArray: isArray
854
+ };
855
+ ApiProperty(apiOptions)(DtoClass.prototype, relation);
856
+ Reflect.defineMetadata("design:type", isArray ? Array : relationDto, DtoClass.prototype, relation);
857
+ Object.defineProperty(DtoClass.prototype, relation, {
858
+ value: undefined,
859
+ writable: true,
860
+ enumerable: true,
861
+ configurable: true
862
+ });
863
+ }
864
+ function addRelationPkRef(DtoClass, relation, pkTypeName, isArray, isRequired) {
865
+ const apiOptions = isArray ? {
866
+ type: "array",
867
+ items: {
868
+ type: pkTypeName
869
+ },
870
+ required: isRequired
871
+ } : {
872
+ type: pkTypeName === "integer" ? Number : String,
873
+ required: isRequired
874
+ };
875
+ ApiProperty(apiOptions)(DtoClass.prototype, relation);
876
+ Reflect.defineMetadata("design:type", isArray ? Array : pkTypeName === "integer" ? Number : String, DtoClass.prototype, relation);
877
+ Object.defineProperty(DtoClass.prototype, relation, {
878
+ value: undefined,
879
+ writable: true,
880
+ enumerable: true,
881
+ configurable: true
882
+ });
883
+ }
884
+ function addRelationUpdate(DtoClass, relation, relationType, relationDto, isArray, isRequired) {
885
+ const extraModels = Reflect.getMetadata(DECORATORS.API_EXTRA_MODELS, DtoClass) || [];
886
+ if (!extraModels.includes(relationDto)) {
887
+ Reflect.defineMetadata(DECORATORS.API_EXTRA_MODELS, [ ...extraModels, relationDto ], DtoClass);
888
+ }
889
+ const dtoRef = `#/components/schemas/${relationDto.name}`;
890
+ const pkTypeName = getPkOpenApiType(relationType);
891
+ const oneOfItems = [ {
892
+ $ref: dtoRef
893
+ }, {
894
+ type: pkTypeName
895
+ } ];
896
+ const apiOptions = isArray ? {
897
+ type: "array",
898
+ required: isRequired,
899
+ oneOf: oneOfItems
900
+ } : {
901
+ type: Object,
902
+ required: isRequired,
903
+ oneOf: oneOfItems
904
+ };
905
+ ApiProperty(apiOptions)(DtoClass.prototype, relation);
906
+ Reflect.defineMetadata("design:type", isArray ? Array : Object, DtoClass.prototype, relation);
907
+ Object.defineProperty(DtoClass.prototype, relation, {
908
+ value: undefined,
909
+ writable: true,
910
+ enumerable: true,
911
+ configurable: true
912
+ });
913
+ }
914
+ function getPkOpenApiType(relationType) {
915
+ try {
916
+ const pkPropsMetadata = decoratorValidation.Model.pkProps(relationType);
917
+ const pkType = pkPropsMetadata?.type;
918
+ if (pkType === Number || pkType === BigInt) return "integer";
919
+ return "string";
920
+ } catch {
921
+ return "string";
922
+ }
923
+ }
924
+ function isPropertyGeneratedAcrossInheritance(model, prop) {
925
+ let current = model;
926
+ while (current && current !== Object && current !== Function) {
927
+ if (decoratorValidation.Model.generated(current, prop)) return true;
928
+ current = Object.getPrototypeOf(current);
929
+ }
930
+ return false;
931
+ }
932
+ function getValidationAcrossInheritance(model, prop) {
933
+ let current = model;
934
+ while (current && current !== Object && current !== Function) {
935
+ const validation = decoration.Metadata.validationFor(current, prop);
936
+ if (validation) return validation;
937
+ current = Object.getPrototypeOf(current);
938
+ }
939
+ return undefined;
940
+ }
941
+ function getTypeAcrossInheritance(model, prop) {
942
+ let current = model;
943
+ while (current && current !== Object && current !== Function) {
944
+ const type = decoration.Metadata.type(current, prop);
945
+ if (type) return type;
946
+ current = Object.getPrototypeOf(current);
947
+ }
948
+ return undefined;
949
+ }
950
+ function getDtoCache(op) {
951
+ if (!dtoCache.has(op)) {
952
+ dtoCache.set(op, new WeakMap);
953
+ }
954
+ return dtoCache.get(op);
955
+ }
956
+ var DynamicModelController_1;
957
+ class FromModelController {
958
+ static {
959
+ this.log = logging.Logging.for(FromModelController.name);
960
+ }
961
+ static getPersistence(ModelClazz) {
962
+ try {
963
+ return core.Service.get(ModelClazz);
964
+ } catch (e) {
965
+ try {
966
+ return core.ModelService.getService(ModelClazz);
967
+ } catch (e2) {
968
+ return core.Repository.forModel(ModelClazz);
969
+ }
970
+ }
971
+ }
972
+ static createQueryRoutesFromRepository(persistence, prefix = core.PersistenceKeys.QUERY) {
973
+ const repo = persistence instanceof core.ModelService ? persistence.repo : persistence;
974
+ const ModelConstr = repo.class;
975
+ const queryMethods = decoration.Metadata.get(repo.constructor, decoration.Metadata.key(core.PersistenceKeys.QUERY)) ?? {};
976
+ const routeMethods = decoration.Metadata.get(persistence.constructor, decoration.Metadata.key(DECAF_ROUTE)) ?? {};
977
+ class QueryController extends DecafModelController {
978
+ get class() {
979
+ throw new Error("Method not implemented.");
980
+ }
981
+ constructor(clientContext, name) {
982
+ super(clientContext, name);
983
+ }
984
+ }
985
+ for (const [methodName, params] of Object.entries(routeMethods)) {
986
+ const routePath = [ params.path.replace(/^\/+|\/+$/g, "") ].filter(segment => segment && segment.trim()).join("/");
987
+ const handler = FromModelController.createComplexQueryHandler(methodName);
988
+ FromModelController.defineMethod(QueryController, methodName, handler);
989
+ const httpDecorator = HttpVerbToDecorator(params.httpMethod)(routePath || undefined);
990
+ const decorators = FromModelController.getQueryDecorators(methodName, routePath, params.httpMethod);
991
+ FromModelController.applyDecorators(QueryController, methodName, [ httpDecorator, ...decorators ]);
992
+ }
993
+ for (const [methodName, objValues] of Object.entries(queryMethods)) {
994
+ const fields = objValues.fields ?? [];
995
+ const routePath = [ prefix, methodName, ...fields.map(f => `:${f}`) ].filter(segment => segment && segment.trim()).join("/");
996
+ const handler = FromModelController.createComplexQueryHandler(methodName);
997
+ FromModelController.defineMethod(QueryController, methodName, handler);
998
+ const httpDecorator = HttpVerbToDecorator("GET")(routePath || undefined);
999
+ const decorators = FromModelController.getQueryDecorators(methodName, routePath, "GET", true);
1000
+ FromModelController.applyDecorators(QueryController, methodName, [ httpDecorator, ...decorators ]);
1001
+ }
1002
+ return QueryController;
1003
+ }
1004
+ static create(ModelConstr, moduleConfigOverrides, globalDefaults) {
1005
+ const log = FromModelController.log.for(FromModelController.create);
1006
+ const tableName = decoratorValidation.Model.tableName(ModelConstr);
1007
+ const routePath = logging.toKebabCase(tableName);
1008
+ const modelClazzName = ModelConstr.name;
1009
+ const persistence = FromModelController.getPersistence(ModelConstr);
1010
+ const factoryPersistence = persistence instanceof core.ModelService ? persistence.repo : persistence;
1011
+ const decoratorConfig = decoration.Metadata.get(ModelConstr, decoration.Metadata.key(DECAF_CONTROLLER_CONFIG));
1012
+ const moduleOverride = moduleConfigOverrides?.[ModelConstr.name];
1013
+ const mergedConfig = {
1014
+ ...globalDefaults || {},
1015
+ ...decoratorConfig || {},
1016
+ ...moduleOverride || {}
1017
+ };
1018
+ const FactoryController = server.ModelControllerFactory.create(ModelConstr, factoryPersistence, mergedConfig);
1019
+ const factoryRoutes = FactoryController.__routes__;
1020
+ const {getPK: getPK, apiProperties: apiProperties, path: pkPath} = FromModelController.getRouteParametersFromModel(ModelConstr);
1021
+ log.debug(`Creating controller for model: ${modelClazzName} with ${factoryRoutes?.length ?? 0} factory routes`);
1022
+ const authConfig = mergedConfig.auth;
1023
+ function applyClassAuth(target) {
1024
+ if (authConfig?.public) {
1025
+ Public()(target);
1026
+ } else if (authConfig?.roles?.length) {
1027
+ RequireRoles(...authConfig.roles)(target);
1028
+ } else {
1029
+ Auth(ModelConstr)(target);
1030
+ }
1031
+ if (authConfig?.skipModelRoles) {
1032
+ common.SetMetadata(SKIP_MODEL_ROLES_KEY, true)(target);
1033
+ }
1034
+ }
1035
+ let DynamicModelController = DynamicModelController_1 = class DynamicModelController extends DecafModelController {
1036
+ static get class() {
1037
+ return ModelConstr;
1038
+ }
1039
+ get class() {
1040
+ return ModelConstr;
1041
+ }
1042
+ constructor(clientContext) {
1043
+ super(clientContext, DynamicModelController_1.name);
1044
+ this.pk = decoratorValidation.Model.pk(ModelConstr);
1045
+ log.info(`Registering dynamic controller for model: ${this.class.name} route: /${routePath}`);
1046
+ }
1047
+ };
1048
+ DynamicModelController = DynamicModelController_1 = tslib.__decorate([ common.Controller(routePath), swagger.ApiTags(modelClazzName), swagger.ApiExtraModels(ModelConstr), tslib.__metadata("design:paramtypes", [ exports.DecafRequestContext ]) ], DynamicModelController);
1049
+ applyClassAuth(DynamicModelController);
1050
+ if (factoryRoutes) {
1051
+ const sortedRoutes = [ ...factoryRoutes ].sort((a, b) => {
1052
+ const aSegments = a.path.split("/").filter(Boolean);
1053
+ const bSegments = b.path.split("/").filter(Boolean);
1054
+ const aParamCount = aSegments.filter(s => s.startsWith(":")).length;
1055
+ const bParamCount = bSegments.filter(s => s.startsWith(":")).length;
1056
+ const aLiteralCount = aSegments.length - aParamCount;
1057
+ const bLiteralCount = bSegments.length - bParamCount;
1058
+ if (aLiteralCount !== bLiteralCount) return bLiteralCount - aLiteralCount;
1059
+ if (aParamCount !== bParamCount) return aParamCount - bParamCount;
1060
+ return 0;
1061
+ });
1062
+ for (const route of sortedRoutes) {
1063
+ const registration = FromModelController.matchRoute(route, pkPath, apiProperties, getPK, ModelConstr, modelClazzName, factoryPersistence);
1064
+ if (!registration) continue;
1065
+ const {methodName: methodName, handler: handler, decorators: decorators, paramDecorators: paramDecorators} = registration;
1066
+ const descriptor = FromModelController.defineMethod(DynamicModelController, methodName, handler);
1067
+ if (descriptor) {
1068
+ const httpDecorator = HttpVerbToDecorator(route.method)(route.path.replace(/^\/+|\/+$/g, "") || undefined);
1069
+ FromModelController.applyDecorators(DynamicModelController, methodName, [ httpDecorator, ...decorators ], paramDecorators);
1070
+ }
1071
+ }
1072
+ }
1073
+ return DynamicModelController;
1074
+ }
1075
+ static getRouteParametersFromModel(ModelClazz) {
1076
+ const pk = decoratorValidation.Model.pk(ModelClazz);
1077
+ const composed = decoration.Metadata.get(ModelClazz, decoration.Metadata.key(dbDecorators.DBKeys.COMPOSED, pk));
1078
+ const composedKeys = composed?.args ?? [];
1079
+ const uniqueKeys = Array.isArray(composedKeys) && composedKeys.length > 0 ? Array.from(new Set([ ...composedKeys ])) : Array.from(new Set([ pk ]));
1080
+ const description = decoration.Metadata.description(ModelClazz) ?? "";
1081
+ const path = `:${uniqueKeys.join("/:")}`;
1082
+ const apiProperties = uniqueKeys.map(key => ({
1083
+ name: key,
1084
+ description: decoration.Metadata.description(ModelClazz, key),
1085
+ required: true,
1086
+ type: String
1087
+ }));
1088
+ return {
1089
+ path: path,
1090
+ description: description,
1091
+ apiProperties: apiProperties,
1092
+ getPK: (...params) => composed?.separator ? params.join(composed.separator) : params.join("")
1093
+ };
1094
+ }
1095
+ static defineMethod(target, methodName, handler) {
1096
+ Object.defineProperty(target.prototype || target, methodName, {
1097
+ value: handler,
1098
+ writable: false,
1099
+ configurable: true,
1100
+ enumerable: false
1101
+ });
1102
+ return Object.getOwnPropertyDescriptor(target.prototype || target, methodName);
1103
+ }
1104
+ static applyDecorators(target, methodName, methodDecorators, paramDecorators = []) {
1105
+ const proto = target?.prototype ?? target;
1106
+ const descriptor = Object.getOwnPropertyDescriptor(proto, methodName);
1107
+ methodDecorators.forEach(d => d(proto, methodName, descriptor));
1108
+ paramDecorators.forEach(({decorator: decorator, index: index}) => decorator(proto, methodName, index));
1109
+ }
1110
+ static matchRoute(route, pkPath, apiProperties, getPK, ModelConstr, modelClazzName, persistence) {
1111
+ const {method: method, path: path} = route;
1112
+ const normalizedPath = path.replace(/^\/+|\/+$/g, "");
1113
+ if (method === "POST" && normalizedPath === "") {
1114
+ return FromModelController.createRegistration("create", FromModelController.createCreateHandler(ModelConstr, modelClazzName), FromModelController.createCreateDecorators(ModelConstr, modelClazzName), [ {
1115
+ decorator: DecafBody(),
1116
+ index: 0
1117
+ }, {
1118
+ decorator: common.Response({
1119
+ passthrough: true
1120
+ }),
1121
+ index: 1
1122
+ } ]);
1123
+ }
1124
+ if (method === "POST" && normalizedPath === "bulk") {
1125
+ return FromModelController.createRegistration("createAll", FromModelController.createBulkCreateHandler(ModelConstr, modelClazzName), FromModelController.bulkCreateDecorators(ModelConstr, modelClazzName), [ {
1126
+ decorator: DecafBody(),
1127
+ index: 0
1128
+ }, {
1129
+ decorator: common.Response({
1130
+ passthrough: true
1131
+ }),
1132
+ index: 1
1133
+ } ]);
1134
+ }
1135
+ if (method === "GET" && normalizedPath === "bulk") {
1136
+ return FromModelController.createRegistration("readAll", FromModelController.createBulkReadHandler(modelClazzName), FromModelController.bulkReadDecorators(ModelConstr, modelClazzName), [ {
1137
+ decorator: common.Query("ids"),
1138
+ index: 0
1139
+ } ]);
1140
+ }
1141
+ if (method === "PUT" && normalizedPath === "bulk") {
1142
+ return FromModelController.createRegistration("updateAll", FromModelController.createBulkUpdateHandler(modelClazzName), FromModelController.bulkUpdateDecorators(ModelConstr, modelClazzName, apiProperties), [ {
1143
+ decorator: DecafBody(),
1144
+ index: 0
1145
+ }, {
1146
+ decorator: common.Response({
1147
+ passthrough: true
1148
+ }),
1149
+ index: 1
1150
+ } ]);
1151
+ }
1152
+ if (method === "DELETE" && normalizedPath === "bulk") {
1153
+ return FromModelController.createRegistration("deleteAll", FromModelController.createBulkDeleteHandler(modelClazzName), FromModelController.bulkDeleteDecorators(ModelConstr, modelClazzName, apiProperties), [ {
1154
+ decorator: common.Query("ids"),
1155
+ index: 0
1156
+ }, {
1157
+ decorator: common.Response({
1158
+ passthrough: true
1159
+ }),
1160
+ index: 1
1161
+ } ]);
1162
+ }
1163
+ if (method === "GET" && normalizedPath === pkPath) {
1164
+ return FromModelController.createRegistration("read", FromModelController.createReadHandler(getPK, modelClazzName), FromModelController.readDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
1165
+ decorator: DecafParams(apiProperties),
1166
+ index: 0
1167
+ } ]);
1168
+ }
1169
+ if (method === "PUT" && normalizedPath === pkPath) {
1170
+ return FromModelController.createRegistration("update", FromModelController.createUpdateHandler(getPK, ModelConstr, modelClazzName), FromModelController.updateDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
1171
+ decorator: DecafParams(apiProperties),
1172
+ index: 0
1173
+ }, {
1174
+ decorator: DecafBody(),
1175
+ index: 1
1176
+ }, {
1177
+ decorator: common.Response({
1178
+ passthrough: true
1179
+ }),
1180
+ index: 2
1181
+ } ]);
1182
+ }
1183
+ if (method === "DELETE" && normalizedPath === pkPath) {
1184
+ return FromModelController.createRegistration("delete", FromModelController.createDeleteHandler(getPK, modelClazzName), FromModelController.deleteDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
1185
+ decorator: DecafParams(apiProperties),
1186
+ index: 0
1187
+ }, {
1188
+ decorator: common.Response({
1189
+ passthrough: true
1190
+ }),
1191
+ index: 1
1192
+ } ]);
1193
+ }
1194
+ const fallbackSegments = normalizedPath.split("/").filter(Boolean);
1195
+ const isAllParams = fallbackSegments.length > 0 && fallbackSegments.every(s => s.startsWith(":"));
1196
+ if (isAllParams && normalizedPath !== pkPath) {
1197
+ const fallbackApiProps = fallbackSegments.map(s => s.slice(1)).map(name => ({
1198
+ name: name,
1199
+ description: `${name} parameter`,
1200
+ required: true,
1201
+ type: String
1202
+ }));
1203
+ const suffix = fallbackSegments.map(s => s.slice(1)).join("And");
1204
+ if (method === "GET") {
1205
+ return FromModelController.createRegistration(`readBy${suffix}`, FromModelController.createReadHandler(getPK, modelClazzName), FromModelController.readDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
1206
+ decorator: DecafParams(fallbackApiProps),
1207
+ index: 0
1208
+ } ]);
1209
+ }
1210
+ if (method === "PUT") {
1211
+ return FromModelController.createRegistration(`updateBy${suffix}`, FromModelController.createUpdateHandler(getPK, ModelConstr, modelClazzName), FromModelController.updateDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
1212
+ decorator: DecafParams(fallbackApiProps),
1213
+ index: 0
1214
+ }, {
1215
+ decorator: DecafBody(),
1216
+ index: 1
1217
+ }, {
1218
+ decorator: common.Response({
1219
+ passthrough: true
1220
+ }),
1221
+ index: 2
1222
+ } ]);
1223
+ }
1224
+ if (method === "DELETE") {
1225
+ return FromModelController.createRegistration(`deleteBy${suffix}`, FromModelController.createDeleteHandler(getPK, modelClazzName), FromModelController.deleteDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
1226
+ decorator: DecafParams(fallbackApiProps),
1227
+ index: 0
1228
+ }, {
1229
+ decorator: common.Response({
1230
+ passthrough: true
1231
+ }),
1232
+ index: 1
1233
+ } ]);
1234
+ }
1235
+ }
1236
+ if (method === "GET" && normalizedPath === "statement/:method/*args") {
1237
+ return FromModelController.createRegistration("statement", FromModelController.createStatementHandler(modelClazzName), FromModelController.statementDecorators(ModelConstr, modelClazzName), [ {
1238
+ decorator: common.Param("method"),
1239
+ index: 0
1240
+ }, {
1241
+ decorator: common.Param("args"),
1242
+ index: 1
1243
+ }, {
1244
+ decorator: DecafQuery(),
1245
+ index: 2
1246
+ } ]);
1247
+ }
1248
+ const statementRoutes = {
1249
+ "listBy/:key": core.PreparedStatementKeys.LIST_BY,
1250
+ "paginateBy/:key/:page": core.PreparedStatementKeys.PAGE_BY,
1251
+ "find/:value": core.PreparedStatementKeys.FIND,
1252
+ "page/:value": core.PreparedStatementKeys.PAGE,
1253
+ "findOneBy/:key/:value": core.PreparedStatementKeys.FIND_ONE_BY,
1254
+ "findBy/:key/:value": core.PreparedStatementKeys.FIND_BY,
1255
+ "countOf/:field": core.PreparedStatementKeys.COUNT_OF,
1256
+ "maxOf/:field": core.PreparedStatementKeys.MAX_OF,
1257
+ "minOf/:field": core.PreparedStatementKeys.MIN_OF,
1258
+ "avgOf/:field": core.PreparedStatementKeys.AVG_OF,
1259
+ "sumOf/:field": core.PreparedStatementKeys.SUM_OF,
1260
+ "distinctOf/:field": core.PreparedStatementKeys.DISTINCT_OF,
1261
+ "groupOf/:field": core.PreparedStatementKeys.GROUP_OF
1262
+ };
1263
+ const statementKey = statementRoutes[normalizedPath];
1264
+ if (statementKey && method === "GET") {
1265
+ return FromModelController.createRegistration(FromModelController.statementMethodName(normalizedPath), FromModelController.createStatementShortcutHandler(statementKey, modelClazzName), FromModelController.statementShortcutDecorators(ModelConstr, modelClazzName, normalizedPath, statementKey), FromModelController.statementShortcutParams(normalizedPath));
1266
+ }
1267
+ if (method === "GET" && normalizedPath.startsWith("query/")) {
1268
+ const queryMethod = normalizedPath.replace(/^query\//, "").split("/")[0];
1269
+ return FromModelController.createRegistration(queryMethod, FromModelController.createComplexQueryHandler(queryMethod), FromModelController.getQueryDecorators(queryMethod, normalizedPath, "GET", true), FromModelController.complexQueryParams(normalizedPath));
1270
+ }
1271
+ const pathSegments = normalizedPath.split("/").filter(Boolean);
1272
+ const knownPrefixes = new Set([ "listBy", "findBy", "findByPaginate", "findOneBy", "paginateBy", "find", "page", "countOf", "maxOf", "minOf", "avgOf", "sumOf", "distinctOf", "groupOf", "statement", "bulk", "query" ]);
1273
+ if (pathSegments.length > 0 && !normalizedPath.startsWith("query/") && !knownPrefixes.has(pathSegments[0])) {
1274
+ const routeMetadata = decoration.Metadata.get(persistence?.constructor, decoration.Metadata.key(DECAF_ROUTE)) ?? {};
1275
+ const matchedEntry = Object.entries(routeMetadata).find(([, info]) => info && typeof info === "object" && info.path?.replace(/^\/+|\/+$/g, "") === normalizedPath);
1276
+ const actualMethodName = matchedEntry?.[0] || pathSegments[0];
1277
+ const paramSegments = pathSegments.filter(s => s.startsWith(":"));
1278
+ const apiPathParams = paramSegments.map(s => s.slice(1)).map(name => ({
1279
+ name: name,
1280
+ description: `${name} parameter for the query`,
1281
+ required: true,
1282
+ type: String
1283
+ }));
1284
+ return FromModelController.createRegistration(actualMethodName, FromModelController.createCustomRouteHandler(actualMethodName), [ ...apiPathParams.map(p => swagger.ApiParam(p)), swagger.ApiOperation({
1285
+ summary: `Retrieve records using "${actualMethodName}".`
1286
+ }), swagger.ApiOkResponse({
1287
+ description: "Result successfully retrieved."
1288
+ }), swagger.ApiNoContentResponse({
1289
+ description: "No content returned by the method."
1290
+ }) ], FromModelController.complexQueryParams(normalizedPath));
1291
+ }
1292
+ return undefined;
1293
+ }
1294
+ static createRegistration(methodName, handler, decorators, paramDecorators) {
1295
+ return {
1296
+ methodName: methodName,
1297
+ handler: handler,
1298
+ decorators: decorators,
1299
+ paramDecorators: paramDecorators
1300
+ };
1301
+ }
1302
+ static statementMethodName(path) {
1303
+ const firstSegment = path.split("/")[0];
1304
+ return firstSegment;
1305
+ }
1306
+ static statementShortcutParams(path) {
1307
+ const segments = path.split("/").filter(s => s.startsWith(":"));
1308
+ const params = [];
1309
+ segments.forEach((seg, i) => {
1310
+ const name = seg.replace(":", "");
1311
+ params.push({
1312
+ decorator: common.Param(name),
1313
+ index: i
1314
+ });
1315
+ });
1316
+ if (path.startsWith("listBy/") || path.startsWith("paginateBy/") || path.startsWith("find/") || path.startsWith("page/")) {
1317
+ params.push({
1318
+ decorator: DecafQuery(),
1319
+ index: segments.length
1320
+ });
1321
+ }
1322
+ return params;
1323
+ }
1324
+ static complexQueryParams(path) {
1325
+ const segments = path.split("/").filter(s => s.startsWith(":"));
1326
+ const params = [];
1327
+ segments.forEach((seg, i) => {
1328
+ const name = seg.replace(":", "");
1329
+ params.push({
1330
+ decorator: common.Param(name),
1331
+ index: i
1332
+ });
1333
+ });
1334
+ if (path.startsWith("query/")) {
1335
+ params.push({
1336
+ decorator: DecafQuery(),
1337
+ index: segments.length
1338
+ });
1339
+ }
1340
+ return params;
1341
+ }
1342
+ static createCreateHandler(ModelConstr, modelClazzName) {
1343
+ return async function create(data, resp) {
1344
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.OperationKeys.CREATE, true)).for(create);
1345
+ log.verbose(`creating new ${modelClazzName}`);
1346
+ let created;
1347
+ try {
1348
+ created = await this.persistence(ctx).create(data, ctx);
1349
+ } catch (e) {
1350
+ log.error(`Failed to create new ${modelClazzName}`, e);
1351
+ throw e;
1352
+ }
1353
+ log.info(`created new ${modelClazzName} with id ${created[this.pk]}`);
1354
+ if (resp) ctx.toResponse(resp);
1355
+ return created;
1356
+ };
1357
+ }
1358
+ static createBulkCreateHandler(ModelConstr, modelClazzName) {
1359
+ return async function createAll(data, resp) {
1360
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.BulkCrudOperationKeys.CREATE_ALL, true)).for(createAll);
1361
+ log.verbose(`creating new ${modelClazzName}`);
1362
+ let created;
1363
+ try {
1364
+ created = await this.persistence(ctx).createAll(data.map(d => new ModelConstr(d)), ctx);
1365
+ } catch (e) {
1366
+ log.error(`Failed to create new ${modelClazzName}`, e);
1367
+ throw e;
1368
+ }
1369
+ log.info(`created new ${modelClazzName} with id ${created[this.pk]}`);
1370
+ if (resp) ctx.toResponse(resp);
1371
+ return created;
1372
+ };
1373
+ }
1374
+ static createBulkReadHandler(modelClazzName) {
1375
+ return async function readAll(ids) {
1376
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.BulkCrudOperationKeys.READ_ALL, true)).for(readAll);
1377
+ const normalizedIds = Array.isArray(ids) ? ids : [ ids ];
1378
+ let read;
1379
+ try {
1380
+ log.debug(`reading ${normalizedIds} ${modelClazzName}`);
1381
+ read = await this.persistence(ctx).readAll(normalizedIds, ctx);
1382
+ } catch (e) {
1383
+ log.error(`Failed to read ${modelClazzName}`, e);
1384
+ throw e;
1385
+ }
1386
+ log.info(`read ${read.length} ${modelClazzName}`);
1387
+ return read;
1388
+ };
1389
+ }
1390
+ static createBulkUpdateHandler(modelClazzName) {
1391
+ return async function updateAll(body, resp) {
1392
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.BulkCrudOperationKeys.UPDATE_ALL, true)).for(updateAll);
1393
+ let updated;
1394
+ try {
1395
+ log.info(`updating ${body.length} ${modelClazzName}`);
1396
+ updated = await this.persistence(ctx).updateAll(body, ctx);
1397
+ } catch (e) {
1398
+ log.error(e);
1399
+ throw e;
1400
+ }
1401
+ if (resp) ctx.toResponse(resp);
1402
+ return updated;
1403
+ };
1404
+ }
1405
+ static createBulkDeleteHandler(modelClazzName) {
1406
+ return async function deleteAll(ids, resp) {
1407
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.BulkCrudOperationKeys.DELETE_ALL, true)).for(deleteAll);
1408
+ const normalizedIds = Array.isArray(ids) ? ids : [ ids ];
1409
+ let read;
1410
+ try {
1411
+ log.debug(`deleting ${normalizedIds.length} ${modelClazzName}`);
1412
+ read = await this.persistence(ctx).deleteAll(normalizedIds, ctx);
1413
+ } catch (e) {
1414
+ log.error(`Failed to delete ${modelClazzName}`, e);
1415
+ throw e;
1416
+ }
1417
+ log.info(`deleted ${read.length} ${modelClazzName}`);
1418
+ if (resp) ctx.toResponse(resp);
1419
+ return read;
1420
+ };
1421
+ }
1422
+ static createReadHandler(getPK, modelClazzName) {
1423
+ return async function read(routeParams) {
1424
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.OperationKeys.READ, true)).for(read);
1425
+ const id = getPK(...routeParams.valuesInOrder);
1426
+ if (typeof id === "undefined") throw new dbDecorators.ValidationError(`No ${this.pk} provided`);
1427
+ let readResult;
1428
+ try {
1429
+ log.debug(`reading ${modelClazzName} with ${this.pk} ${id}`);
1430
+ readResult = await this.persistence(ctx).read(id, ctx);
1431
+ } catch (e) {
1432
+ log.error(`Failed to read ${modelClazzName} with id ${id}`, e);
1433
+ throw e;
1434
+ }
1435
+ log.info(`read ${modelClazzName} with id ${readResult[this.pk]}`);
1436
+ return readResult;
1437
+ };
1438
+ }
1439
+ static createUpdateHandler(getPK, ModelConstr, modelClazzName) {
1440
+ return async function update(routeParams, body, resp) {
1441
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.OperationKeys.UPDATE, true)).for(update);
1442
+ const id = getPK(...routeParams.valuesInOrder);
1443
+ if (typeof id === "undefined") throw new dbDecorators.ValidationError(`No ${this.pk} provided`);
1444
+ let updated;
1445
+ try {
1446
+ log.info(`updating ${modelClazzName} with ${this.pk} ${id}`);
1447
+ const payload = JSON.parse(JSON.stringify(body));
1448
+ updated = await this.persistence(ctx).update(new ModelConstr({
1449
+ ...payload,
1450
+ [this.pk]: id
1451
+ }), ctx);
1452
+ } catch (e) {
1453
+ log.error(e);
1454
+ throw e;
1455
+ }
1456
+ if (resp) ctx.toResponse(resp);
1457
+ return updated;
1458
+ };
1459
+ }
1460
+ static createDeleteHandler(getPK, modelClazzName) {
1461
+ return async function remove(routeParams, resp) {
1462
+ const {ctx: ctx, log: log} = (await this.logCtx([], dbDecorators.OperationKeys.DELETE, true)).for(remove);
1463
+ const id = getPK(...routeParams.valuesInOrder);
1464
+ if (typeof id === "undefined") throw new dbDecorators.ValidationError(`No ${this.pk} provided`);
1465
+ let del;
1466
+ try {
1467
+ log.debug(`deleting ${modelClazzName} with ${this.pk} ${id}`);
1468
+ del = await this.persistence(ctx).delete(id, ctx);
1469
+ } catch (e) {
1470
+ log.error(`Failed to delete ${modelClazzName} with id ${id}`, e);
1471
+ throw e;
1472
+ }
1473
+ log.info(`deleted ${modelClazzName} with id ${id}`);
1474
+ if (resp) ctx.toResponse(resp);
1475
+ return del;
1476
+ };
1477
+ }
1478
+ static createStatementHandler(modelClazzName) {
1479
+ return async function statement(name, args, details) {
1480
+ const {ctx: ctx} = (await this.logCtx([], core.PersistenceKeys.STATEMENT, true)).for(statement);
1481
+ const {direction: direction, offset: offset, limit: limit, bookmark: bookmark} = details;
1482
+ args = args.map(a => (typeof a === "string" ? parseInt(a) : a) || a);
1483
+ const pathDirection = args.length > 1 ? args[1] : undefined;
1484
+ const resolvedDirection = direction ?? pathDirection;
1485
+ if (resolvedDirection && args.length > 1) args[1] = resolvedDirection;
1486
+ switch (name) {
1487
+ case core.PreparedStatementKeys.FIND:
1488
+ case core.PreparedStatementKeys.FIND_BY:
1489
+ break;
1490
+
1491
+ case core.PreparedStatementKeys.LIST_BY:
1492
+ args.push(direction);
1493
+ break;
1494
+
1495
+ case core.PreparedStatementKeys.PAGE:
1496
+ case core.PreparedStatementKeys.PAGE_BY:
1497
+ args = [ args[0], resolvedDirection, {
1498
+ limit: limit,
1499
+ offset: offset,
1500
+ bookmark: bookmark
1501
+ } ];
1502
+ break;
1503
+
1504
+ case core.PreparedStatementKeys.FIND_ONE_BY:
1505
+ break;
1506
+
1507
+ case core.PreparedStatementKeys.COUNT_OF:
1508
+ case core.PreparedStatementKeys.MAX_OF:
1509
+ case core.PreparedStatementKeys.MIN_OF:
1510
+ case core.PreparedStatementKeys.AVG_OF:
1511
+ case core.PreparedStatementKeys.SUM_OF:
1512
+ case core.PreparedStatementKeys.DISTINCT_OF:
1513
+ case core.PreparedStatementKeys.GROUP_OF:
1514
+ break;
1515
+ }
1516
+ return this.persistence(ctx).statement(name, ...args, ctx);
1517
+ };
1518
+ }
1519
+ static createStatementShortcutHandler(statementKey, modelClazzName) {
1520
+ return async function statementShortcut(...args) {
1521
+ const {ctx: ctx} = (await this.logCtx([], statementKey, true)).for(statementShortcut);
1522
+ switch (statementKey) {
1523
+ case core.PreparedStatementKeys.LIST_BY:
1524
+ {
1525
+ const [key, details] = args;
1526
+ return this.persistence(ctx).listBy(key, details?.direction, ctx);
1527
+ }
1528
+
1529
+ case core.PreparedStatementKeys.PAGE_BY:
1530
+ {
1531
+ const [key, page, details] = args;
1532
+ return this.persistence(ctx).paginateBy(key, details?.direction, {
1533
+ limit: details?.limit,
1534
+ offset: details?.offset,
1535
+ page: page
1536
+ }, ctx);
1537
+ }
1538
+
1539
+ case core.PreparedStatementKeys.FIND:
1540
+ {
1541
+ const [value, details] = args;
1542
+ const direction = details?.direction ?? core.OrderDirection.ASC;
1543
+ const persistence = this.persistence(ctx);
1544
+ if (typeof persistence.find === "function") return persistence.find(value, direction, ctx);
1545
+ return persistence.statement(core.PreparedStatementKeys.FIND, value, direction, ctx);
1546
+ }
1547
+
1548
+ case core.PreparedStatementKeys.PAGE:
1549
+ {
1550
+ const [value, details] = args;
1551
+ const ref = {
1552
+ offset: details?.offset ?? 1,
1553
+ limit: details?.limit ?? 10,
1554
+ bookmark: details?.bookmark
1555
+ };
1556
+ const persistence = this.persistence(ctx);
1557
+ const direction = details?.direction ?? core.OrderDirection.ASC;
1558
+ if (typeof persistence.page === "function") return persistence.page(value, direction, ref, ctx);
1559
+ return persistence.statement(core.PreparedStatementKeys.PAGE, value, direction, ref, ctx);
1560
+ }
1561
+
1562
+ case core.PreparedStatementKeys.FIND_ONE_BY:
1563
+ {
1564
+ const [key, value] = args;
1565
+ return this.persistence(ctx).findOneBy(key, value, ctx);
1566
+ }
1567
+
1568
+ case core.PreparedStatementKeys.FIND_BY:
1569
+ {
1570
+ const [key, value] = args;
1571
+ return this.persistence(ctx).for(ctx.toOverrides()).findBy(key, value, ctx);
1572
+ }
1573
+
1574
+ default:
1575
+ if (statementKey === core.PreparedStatementKeys.COUNT_OF || statementKey === core.PreparedStatementKeys.MAX_OF || statementKey === core.PreparedStatementKeys.MIN_OF || statementKey === core.PreparedStatementKeys.AVG_OF || statementKey === core.PreparedStatementKeys.SUM_OF || statementKey === core.PreparedStatementKeys.DISTINCT_OF || statementKey === core.PreparedStatementKeys.GROUP_OF) {
1576
+ const [field] = args;
1577
+ return this.persistence(ctx).statement(statementKey, field, ctx);
1578
+ }
1579
+ throw new Error(`Unknown statement: ${statementKey}`);
1580
+ }
1581
+ };
1582
+ }
1583
+ static createComplexQueryHandler(methodName) {
1584
+ return async function complexQuery(...args) {
1585
+ const log = this.log?.for?.(complexQuery);
1586
+ try {
1587
+ if (log) log.debug(`Invoking custom query "${methodName}"`);
1588
+ const {ctx: ctx} = (await this.logCtx([], methodName, true)).for(complexQuery);
1589
+ const persistence = this.persistence(ctx);
1590
+ const spreadArgs = FromModelController.extractQueryArgs(args);
1591
+ if (typeof persistence[methodName] === "function") {
1592
+ return persistence[methodName](...spreadArgs, ctx);
1593
+ }
1594
+ if (typeof persistence.statement === "function") {
1595
+ return persistence.statement(methodName, ...spreadArgs, ctx);
1596
+ }
1597
+ throw new Error(`Persistence method "${methodName}" not found on ${persistence?.constructor?.name}`);
1598
+ } catch (e) {
1599
+ if (log) log.error(`Custom query "${methodName}" failed`, e);
1600
+ throw e;
1601
+ }
1602
+ };
1603
+ }
1604
+ static createCustomRouteHandler(methodName) {
1605
+ return async function customRoute(...args) {
1606
+ const log = this.log?.for?.(customRoute);
1607
+ const {ctx: ctx} = (await this.logCtx([], methodName, true)).for(customRoute);
1608
+ const persistence = this.persistence(ctx);
1609
+ const spreadArgs = FromModelController.extractQueryArgs(args);
1610
+ if (typeof persistence[methodName] === "function") {
1611
+ return persistence[methodName](...spreadArgs, ctx);
1612
+ }
1613
+ if (persistence?.repo && typeof persistence.repo[methodName] === "function") {
1614
+ return persistence.repo[methodName](...spreadArgs, ctx);
1615
+ }
1616
+ if (typeof persistence.statement === "function") {
1617
+ return persistence.statement(methodName, ...spreadArgs, ctx);
1618
+ }
1619
+ throw new Error(`Method "${methodName}" not found on ${persistence?.constructor?.name} or its repo`);
1620
+ };
1621
+ }
1622
+ static extractQueryArgs(args) {
1623
+ if (args.length === 0) return args;
1624
+ const last = args[args.length - 1];
1625
+ if (last && typeof last === "object" && !Array.isArray(last)) {
1626
+ const queryObj = args.pop();
1627
+ const hasDirection = queryObj.direction !== undefined;
1628
+ const hasLimit = queryObj.limit !== undefined;
1629
+ const hasOffset = queryObj.offset !== undefined;
1630
+ if (!hasDirection && !hasLimit && !hasOffset) return args;
1631
+ const extras = [];
1632
+ if (hasDirection) extras.push(queryObj.direction); else if (hasLimit || hasOffset) extras.push(undefined);
1633
+ if (hasLimit) extras.push(queryObj.limit);
1634
+ if (hasOffset) extras.push(queryObj.offset);
1635
+ return [ ...args, ...extras ];
1636
+ }
1637
+ return args;
1638
+ }
1639
+ static createCreateDecorators(ModelConstr, modelClazzName) {
1640
+ return [ ApiOperationFromModel(ModelConstr, "POST"), swagger.ApiOperation({
1641
+ summary: `Create a new ${modelClazzName}.`
1642
+ }), swagger.ApiBody({
1643
+ description: `Payload for ${modelClazzName}`,
1644
+ type: DtoFor(dbDecorators.OperationKeys.CREATE, ModelConstr)
1645
+ }), swagger.ApiCreatedResponse({
1646
+ description: `${modelClazzName} created successfully.`,
1647
+ schema: {
1648
+ $ref: swagger.getSchemaPath(ModelConstr)
1649
+ }
1650
+ }), swagger.ApiBadRequestResponse({
1651
+ description: "Payload validation failed."
1652
+ }), swagger.ApiUnprocessableEntityResponse({
1653
+ description: "Repository rejected the provided payload."
1654
+ }) ];
1655
+ }
1656
+ static bulkCreateDecorators(ModelConstr, modelClazzName) {
1657
+ return [ ApiOperationFromModel(ModelConstr, "POST", "bulk"), swagger.ApiOperation({
1658
+ summary: `Create a new ${modelClazzName}.`
1659
+ }), swagger.ApiBody({
1660
+ description: `Payload for ${modelClazzName}`,
1661
+ schema: {
1662
+ type: "array",
1663
+ items: {
1664
+ $ref: swagger.getSchemaPath(ModelConstr)
1665
+ }
1666
+ }
1667
+ }), swagger.ApiCreatedResponse({
1668
+ description: `${modelClazzName} created successfully.`,
1669
+ schema: {
1670
+ type: "array",
1671
+ items: {
1672
+ $ref: swagger.getSchemaPath(ModelConstr)
1673
+ }
1674
+ }
1675
+ }), swagger.ApiBadRequestResponse({
1676
+ description: "Payload validation failed."
1677
+ }), swagger.ApiUnprocessableEntityResponse({
1678
+ description: "Repository rejected the provided payload."
1679
+ }) ];
1680
+ }
1681
+ static bulkReadDecorators(ModelConstr, modelClazzName) {
1682
+ return [ ApiOperationFromModel(ModelConstr, "GET", "bulk"), swagger.ApiOperation({
1683
+ summary: `Retrieve ${modelClazzName} records by ids.`
1684
+ }), swagger.ApiQuery({
1685
+ name: "ids",
1686
+ required: true,
1687
+ type: "array"
1688
+ }), swagger.ApiOkResponse({
1689
+ description: `${modelClazzName} retrieved successfully.`,
1690
+ schema: {
1691
+ type: "array",
1692
+ items: {
1693
+ $ref: swagger.getSchemaPath(ModelConstr)
1694
+ }
1695
+ }
1696
+ }), swagger.ApiNotFoundResponse({
1697
+ description: `No ${modelClazzName} record matches the provided identifier.`
1698
+ }) ];
1699
+ }
1700
+ static bulkUpdateDecorators(ModelConstr, modelClazzName, apiProperties) {
1701
+ return [ ApiOperationFromModel(ModelConstr, "PUT", "bulk"), ApiParamsFromModel(apiProperties), swagger.ApiOperation({
1702
+ summary: `Replace existing ${modelClazzName} records with new payloads.`
1703
+ }), swagger.ApiBody({
1704
+ description: `Payload for replacing existing records of ${modelClazzName}`,
1705
+ schema: {
1706
+ type: "array",
1707
+ $ref: swagger.getSchemaPath(DtoFor(dbDecorators.OperationKeys.UPDATE, ModelConstr))
1708
+ }
1709
+ }), swagger.ApiOkResponse({
1710
+ description: `${modelClazzName} updated successfully.`,
1711
+ schema: {
1712
+ type: "array",
1713
+ items: {
1714
+ $ref: swagger.getSchemaPath(ModelConstr)
1715
+ }
1716
+ }
1717
+ }), swagger.ApiNotFoundResponse({
1718
+ description: `No ${modelClazzName} record matches the provided identifier.`
1719
+ }), swagger.ApiBadRequestResponse({
1720
+ description: "Payload validation failed."
1721
+ }) ];
1722
+ }
1723
+ static bulkDeleteDecorators(ModelConstr, modelClazzName, apiProperties) {
1724
+ return [ ApiOperationFromModel(ModelConstr, "DELETE", "bulk"), ApiParamsFromModel(apiProperties), swagger.ApiOperation({
1725
+ summary: `Delete ${modelClazzName} records by ids.`
1726
+ }), swagger.ApiQuery({
1727
+ name: "ids",
1728
+ required: true,
1729
+ type: "array"
1730
+ }), swagger.ApiOkResponse({
1731
+ description: `${modelClazzName} deleted successfully.`,
1732
+ schema: {
1733
+ type: "array",
1734
+ items: {
1735
+ $ref: swagger.getSchemaPath(ModelConstr)
1736
+ }
1737
+ }
1738
+ }), swagger.ApiNotFoundResponse({
1739
+ description: `No ${modelClazzName} record matches the provided identifier.`
1740
+ }) ];
1741
+ }
1742
+ static readDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
1743
+ return [ ApiOperationFromModel(ModelConstr, "GET", pkPath), ApiParamsFromModel(apiProperties), swagger.ApiOperation({
1744
+ summary: `Retrieve a ${modelClazzName} record by id.`
1745
+ }), swagger.ApiOkResponse({
1746
+ description: `${modelClazzName} retrieved successfully.`,
1747
+ schema: {
1748
+ $ref: swagger.getSchemaPath(ModelConstr)
1749
+ }
1750
+ }), swagger.ApiNotFoundResponse({
1751
+ description: `No ${modelClazzName} record matches the provided identifier.`
1752
+ }) ];
1753
+ }
1754
+ static updateDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
1755
+ return [ ApiOperationFromModel(ModelConstr, "PUT", pkPath), ApiParamsFromModel(apiProperties), swagger.ApiOperation({
1756
+ summary: `Replace an existing ${modelClazzName} record with a new payload.`
1757
+ }), swagger.ApiBody({
1758
+ description: `Payload for replacing an existing record of ${modelClazzName}`,
1759
+ type: DtoFor(dbDecorators.OperationKeys.UPDATE, ModelConstr)
1760
+ }), swagger.ApiOkResponse({
1761
+ description: `${modelClazzName} updated successfully.`,
1762
+ schema: {
1763
+ $ref: swagger.getSchemaPath(ModelConstr)
1764
+ }
1765
+ }), swagger.ApiNotFoundResponse({
1766
+ description: `No ${modelClazzName} record matches the provided identifier.`
1767
+ }), swagger.ApiBadRequestResponse({
1768
+ description: "Payload validation failed."
1769
+ }) ];
1770
+ }
1771
+ static deleteDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
1772
+ return [ ApiOperationFromModel(ModelConstr, "DELETE", pkPath), ApiParamsFromModel(apiProperties), swagger.ApiOperation({
1773
+ summary: `Delete a ${modelClazzName} record by id.`
1774
+ }), swagger.ApiOkResponse({
1775
+ description: `${modelClazzName} deleted successfully.`,
1776
+ schema: {
1777
+ $ref: swagger.getSchemaPath(ModelConstr)
1778
+ }
1779
+ }), swagger.ApiNotFoundResponse({
1780
+ description: `No ${modelClazzName} record matches the provided identifier.`
1781
+ }) ];
1782
+ }
1783
+ static statementDecorators(ModelConstr, modelClazzName) {
1784
+ return [ ApiOperationFromModel(ModelConstr, "GET", "statement/:method/*args"), swagger.ApiOperation({
1785
+ summary: `Executes a prepared statement on ${modelClazzName}.`
1786
+ }), swagger.ApiParam({
1787
+ name: "method",
1788
+ description: "the prepared statement to execute"
1789
+ }), swagger.ApiParam({
1790
+ name: "args",
1791
+ description: "concatenated list of arguments the prepared statement can accept"
1792
+ }), swagger.ApiQuery({
1793
+ name: "direction",
1794
+ required: true,
1795
+ enum: core.OrderDirection,
1796
+ description: "the sort order when applicable"
1797
+ }), swagger.ApiQuery({
1798
+ name: "limit",
1799
+ required: true,
1800
+ description: "limit or page size when applicable"
1801
+ }), swagger.ApiQuery({
1802
+ name: "offset",
1803
+ required: true,
1804
+ description: "offset or bookmark when applicable"
1805
+ }), swagger.ApiOkResponse({
1806
+ description: `${modelClazzName} listed found.`
1807
+ }), swagger.ApiNotFoundResponse({
1808
+ description: `No ${modelClazzName} record matches the provided identifier.`
1809
+ }) ];
1810
+ }
1811
+ static statementShortcutDecorators(ModelConstr, modelClazzName, path, statementKey) {
1812
+ const base = [ ApiOperationFromModel(ModelConstr, "GET", path), swagger.ApiOperation({
1813
+ summary: `Retrieve ${modelClazzName} records.`
1814
+ }), swagger.ApiOkResponse({
1815
+ description: `${modelClazzName} retrieved successfully.`
1816
+ }) ];
1817
+ const segments = path.split("/").filter(s => s.startsWith(":"));
1818
+ segments.forEach(seg => {
1819
+ const name = seg.replace(":", "");
1820
+ base.push(swagger.ApiParam({
1821
+ name: name,
1822
+ description: `The ${name} parameter`
1823
+ }));
1824
+ });
1825
+ if (path.startsWith("listBy/") || path.startsWith("paginateBy/") || path.startsWith("find/") || path.startsWith("page/")) {
1826
+ base.push(swagger.ApiQuery({
1827
+ name: "direction",
1828
+ required: true,
1829
+ enum: core.OrderDirection,
1830
+ description: "the sort order"
1831
+ }));
1832
+ }
1833
+ if (path.startsWith("paginateBy/") || path.startsWith("page/")) {
1834
+ base.push(swagger.ApiQuery({
1835
+ name: "limit",
1836
+ required: false,
1837
+ description: "page size"
1838
+ }), swagger.ApiQuery({
1839
+ name: "offset",
1840
+ required: false,
1841
+ description: "page number"
1842
+ }), swagger.ApiQuery({
1843
+ name: "bookmark",
1844
+ required: false,
1845
+ description: "bookmark for cursor pagination"
1846
+ }));
1847
+ }
1848
+ if (path.startsWith("findOneBy/") || path.startsWith("findBy/")) {
1849
+ base.push(swagger.ApiNotFoundResponse({
1850
+ description: `No ${modelClazzName} record matches the provided identifier.`
1851
+ }));
1852
+ }
1853
+ if (statementKey === core.PreparedStatementKeys.COUNT_OF || statementKey === core.PreparedStatementKeys.AVG_OF || statementKey === core.PreparedStatementKeys.SUM_OF) {
1854
+ base.push(swagger.ApiOkResponse({
1855
+ description: `Result for ${modelClazzName}.`,
1856
+ type: Number
1857
+ }));
1858
+ }
1859
+ if (statementKey === core.PreparedStatementKeys.DISTINCT_OF) {
1860
+ base.push(swagger.ApiOkResponse({
1861
+ description: `Distinct values for ${modelClazzName}.`,
1862
+ type: [ String ]
1863
+ }));
1864
+ }
1865
+ return base;
1866
+ }
1867
+ static getQueryDecorators(methodName, routePath, httpVerb, includeQueryParams = false) {
1868
+ const extractPathParams = p => p.split("/").filter(s => s.startsWith(":")).map(s => s.slice(1));
1869
+ const apiPathParams = extractPathParams(routePath).map(name => ({
1870
+ name: name,
1871
+ description: `${name} parameter for the query`,
1872
+ required: true,
1873
+ type: String
1874
+ }));
1875
+ const decorators = [ ...apiPathParams.map(p => swagger.ApiParam(p)), swagger.ApiOperation({
1876
+ summary: `Retrieve records using "${methodName}".`
1877
+ }), swagger.ApiOkResponse({
1878
+ description: "Result successfully retrieved."
1879
+ }), swagger.ApiNoContentResponse({
1880
+ description: "No content returned by the method."
1881
+ }) ];
1882
+ if (httpVerb === "GET" && includeQueryParams) {
1883
+ decorators.push(swagger.ApiQuery({
1884
+ name: "direction",
1885
+ required: false,
1886
+ enum: core.OrderDirection,
1887
+ description: "the sort order when applicable"
1888
+ }), swagger.ApiQuery({
1889
+ name: "limit",
1890
+ required: false,
1891
+ description: "limit or page size"
1892
+ }), swagger.ApiQuery({
1893
+ name: "offset",
1894
+ required: false,
1895
+ description: "offset or bookmark"
1896
+ }));
1897
+ }
1898
+ return decorators;
1899
+ }
1900
+ }
1901
+ function getModuleFor(flavour) {
1902
+ var DecafModelModule_1;
1903
+ let DecafModelModule = class DecafModelModule {
1904
+ static {
1905
+ DecafModelModule_1 = this;
1906
+ }
1907
+ static {
1908
+ this.log = logging.Logging.for(DecafModelModule_1.name).for(flavour);
1909
+ }
1910
+ static createModelServices(models) {
1911
+ return models.map(model => ({
1912
+ provide: `${model.name}Service`,
1913
+ useFactory: () => core.ModelService.forModel(model)
1914
+ }));
1915
+ }
1916
+ static isExposed(model, exposure) {
1917
+ const override = exposure?.[model.name];
1918
+ const value = typeof override !== "undefined" ? override : decoration.Metadata.get(model, decoration.Metadata.key(DECAF_EXPOSE));
1919
+ if (typeof value === "undefined") return true;
1920
+ if (value === true) return true;
1921
+ if (Array.isArray(value)) return value.includes(flavour);
1922
+ return false;
1923
+ }
1924
+ static forRoot(flavour, options = {}) {
1925
+ const log = this.log.for(this.forRoot);
1926
+ log.info(`Generating controllers for flavour...`);
1927
+ const trackedModels = core.Adapter.models(flavour).filter(model => this.isExposed(model, options.controllerExposure));
1928
+ let modelServices = [];
1929
+ if (options.autoServices) {
1930
+ log.info("Auto-services enabled. Initializing service generation.");
1931
+ modelServices = this.createModelServices(trackedModels);
1932
+ log.info(`Auto-services completed. ${modelServices.length} services initialized.`);
1933
+ }
1934
+ const globalDefaults = {};
1935
+ if (options.aggregations === false) {
1936
+ globalDefaults.allowGroupingQueries = false;
1937
+ }
1938
+ const controllers = trackedModels.map(model => FromModelController.create(model, options.controllerConfig, globalDefaults));
1939
+ log.info(`Generated ${controllers.length} controllers`);
1940
+ return {
1941
+ module: DecafModelModule_1,
1942
+ controllers: controllers,
1943
+ providers: [ ...modelServices ]
1944
+ };
1945
+ }
1946
+ };
1947
+ DecafModelModule = DecafModelModule_1 = tslib.__decorate([ common.Module({}) ], DecafModelModule);
1948
+ Object.assign(DecafModelModule, "name", {
1949
+ value: `DecafModule${flavour}`
1950
+ });
1951
+ return DecafModelModule;
1952
+ }
1953
+ decoration.Decoration.for(injectableDecorators.InjectablesKeys.INJECTABLE).extend({
1954
+ decorator: function nestInjectable(category, cfg) {
1955
+ cfg = cfg || (typeof category === "object" ? Object.assign(category, injectableDecorators.DefaultInjectablesConfig) : injectableDecorators.DefaultInjectablesConfig);
1956
+ return common.Injectable({
1957
+ scope: cfg.singleton ? common.Scope.DEFAULT : common.Scope.REQUEST,
1958
+ durable: cfg.singleton ? undefined : true
1959
+ });
1960
+ }
1961
+ }).apply();
1962
+ decoration.Decoration.for(injectableDecorators.InjectablesKeys.INJECT).extend({
1963
+ decorator: function nestInject(category, cfg) {
1964
+ return function innerNestInject(target, propertyKey, descriptor) {
1965
+ return common.Inject(category || target)(target, propertyKey, descriptor);
1966
+ };
1967
+ }
1968
+ }).apply();
1969
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.REQUIRED).extend(ApiProperty({
1970
+ required: true
1971
+ })).apply();
1972
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MAX).extend({
1973
+ decorator: function maxDec(max) {
1974
+ return ApiProperty({
1975
+ maximum: max,
1976
+ required: false
1977
+ });
1978
+ }
1979
+ }).apply();
1980
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MIN).extend({
1981
+ decorator: function minDec(min) {
1982
+ return ApiProperty({
1983
+ minimum: min,
1984
+ required: false
1985
+ });
1986
+ }
1987
+ }).apply();
1988
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MAX_LENGTH).extend({
1989
+ decorator: function maxLengthDec(max) {
1990
+ return ApiProperty({
1991
+ maxLength: max,
1992
+ required: false
1993
+ });
1994
+ }
1995
+ }).apply();
1996
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.MIN_LENGTH).extend({
1997
+ decorator: function minLengthDec(min) {
1998
+ return ApiProperty({
1999
+ minLength: min,
2000
+ required: false
2001
+ });
2002
+ }
2003
+ }).apply();
2004
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.TYPE).extend({
2005
+ decorator: function typeDec(type) {
2006
+ return (target, prop) => {
2007
+ type = Array.isArray(type) ? type[0] : type;
2008
+ if (typeof type === "function" && !type.name) type = type();
2009
+ return ApiProperty({
2010
+ type: type,
2011
+ required: false
2012
+ })(target, prop);
2013
+ };
2014
+ }
2015
+ }).apply();
2016
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.DATE).extend({
2017
+ decorator: function dateDec(format) {
2018
+ return ApiProperty({
2019
+ type: String,
2020
+ format: "date-time",
2021
+ required: false
2022
+ });
2023
+ }
2024
+ }).apply();
2025
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.ENUM).extend({
2026
+ decorator: function optionDec(options) {
2027
+ const opts = Array.isArray(options) ? options : Object.values(options);
2028
+ return ApiProperty({
2029
+ enum: opts,
2030
+ required: false
2031
+ });
2032
+ }
2033
+ }).apply();
2034
+ decoration.Decoration.for(decoratorValidation.ValidationKeys.PATTERN).extend({
2035
+ decorator: function patternDec(pat) {
2036
+ return ApiProperty({
2037
+ pattern: typeof pat === "string" ? pat : pat.source,
2038
+ required: false
2039
+ });
2040
+ }
2041
+ }).apply();
2042
+ decoration.Decoration.for(core.PersistenceKeys.COLUMN).extend({
2043
+ decorator: function columnDec(name) {
2044
+ return ApiProperty({
2045
+ name: name,
2046
+ required: false
2047
+ });
2048
+ }
2049
+ }).apply();
2050
+ decoration.Decoration.for(decoration.DecorationKeys.DESCRIPTION).extend({
2051
+ decorator: function descriptionDec(description) {
2052
+ return ApiProperty({
2053
+ description: description,
2054
+ required: false
2055
+ });
2056
+ }
2057
+ }).apply();
2058
+ decoration.Decoration.for(core.PersistenceKeys.AUTH).extend({
2059
+ decorator: Auth
2060
+ }).apply();
2061
+ class CorsError extends core.ForbiddenError {
2062
+ constructor(msg) {
2063
+ super(msg, CorsError.name);
2064
+ }
2065
+ }
2066
+ class ToManyRequestsError extends dbDecorators.BaseError {
2067
+ constructor(msg) {
2068
+ super(ToManyRequestsError.name, msg, 429);
2069
+ }
2070
+ }
2071
+ exports.DecafExceptionFilter = class DecafExceptionFilter {
2072
+ catch(exception, host) {
2073
+ const ctx = host.switchToHttp();
2074
+ const response = ctx.getResponse();
2075
+ const request = ctx.getRequest();
2076
+ const isProduction = logging.LoggedEnvironment.env === "production";
2077
+ let statusCode;
2078
+ if (exception instanceof common.NotFoundException || exception instanceof core.UnsupportedError) {
2079
+ exception = new common.NotAcceptableException(exception.message);
2080
+ statusCode = exception.getStatus();
2081
+ } else if (!(exception instanceof dbDecorators.BaseError)) {
2082
+ if (exception.status === 429) {
2083
+ exception = new ToManyRequestsError(exception.message);
2084
+ } else {
2085
+ exception = new dbDecorators.InternalError(exception.message);
2086
+ }
2087
+ }
2088
+ response.status(exception.code || statusCode).json({
2089
+ status: exception.code || statusCode,
2090
+ error: isProduction ? exception.name : exception.message,
2091
+ timestamp: (new Date).toISOString(),
2092
+ path: request.url,
2093
+ method: request.method
2094
+ });
2095
+ }
2096
+ };
2097
+ exports.DecafExceptionFilter = tslib.__decorate([ common.Catch() ], exports.DecafExceptionFilter);
2098
+ function UseDecafFilter() {
2099
+ return common.UseFilters(new exports.DecafExceptionFilter);
2100
+ }
2101
+ function UseDecafHeaders() {
2102
+ return common.UseInterceptors(exports.DecafResponseInterceptor);
2103
+ }
2104
+ const SWAGGER_UI_CONSTANTS = {
2105
+ title: "Swagger | OpenAPI Specification (OAS)",
2106
+ description: "Standardized format for describing RESTful APIs",
2107
+ version: "0.0.1",
2108
+ path: "docs",
2109
+ faviconFilePath: "",
2110
+ topbarIconFilePath: "",
2111
+ auth: {
2112
+ type: "http",
2113
+ scheme: "bearer",
2114
+ bearerFormat: "JWT",
2115
+ name: "Authorization",
2116
+ description: "Enter JWT token",
2117
+ in: "header"
2118
+ },
2119
+ persistAuthorization: true,
2120
+ topbarBgColor: "#000000"
2121
+ };
2122
+ class SwaggerCustomUI {
2123
+ constructor(options) {
2124
+ this.options = {
2125
+ ...options
2126
+ };
2127
+ }
2128
+ customCSS() {
2129
+ let css = "";
2130
+ if (this.options.topbarIconPath) {
2131
+ const img = this.b64(this.options.topbarIconPath);
2132
+ css += `.topbar-wrapper { content: url('data:image/png;base64,${img}'); width: 200px; height: auto; }\n`;
2133
+ }
2134
+ return css + `\n .topbar-wrapper svg { visibility: hidden; }\n .swagger-ui .topbar { background-color: ${this.options.topbarBgColor || SWAGGER_UI_CONSTANTS.topbarBgColor}; }\n `;
2135
+ }
2136
+ getCustomOptions() {
2137
+ const favicon = {};
2138
+ if (this.options.faviconPath) {
2139
+ favicon["customfavIcon"] = this.b64(this.options.faviconPath, true);
2140
+ }
2141
+ return {
2142
+ customSiteTitle: this.options.title,
2143
+ ...favicon,
2144
+ customCss: this.customCSS(),
2145
+ swaggerOptions: {
2146
+ persistAuthorization: this.options.persistAuthorization
2147
+ }
2148
+ };
2149
+ }
2150
+ b64(file, img = false) {
2151
+ const filePath = path__namespace.join(this.options.assetsPath || "", file);
2152
+ const b64 = fs.readFileSync(filePath, {
2153
+ encoding: "base64"
2154
+ });
2155
+ return img ? "data:image/png;base64," + b64 : b64;
2156
+ }
2157
+ }
2158
+ class SwaggerBuilder {
2159
+ constructor(app, options) {
2160
+ this.app = app;
2161
+ this.options = options;
2162
+ }
2163
+ createDocument() {
2164
+ const description = this.options.path ? this.options.description + "" + `<br><br><a href="${this.options.openApiJsonPath}">OpenAPI JSON Spec</a> | ` + `<a href="${this.options.openApiYamlPath}">OpenAPI YAML Spec</a>` : this.options.description;
2165
+ const config = (new swagger.DocumentBuilder).setTitle(this.options.title).setDescription(description).setVersion(this.options.version || "0.0.1").addBearerAuth(this.options.auth || SWAGGER_UI_CONSTANTS.auth).build();
2166
+ return swagger.SwaggerModule.createDocument(this.app, config, {
2167
+ extraModels: this.options.extraModels || []
2168
+ });
2169
+ }
2170
+ registerOpenApiRoute(path, contentType, bodyFactory) {
2171
+ if (!path) return;
2172
+ const httpAdapter = this.app.getHttpAdapter();
2173
+ path = path.startsWith("/") ? path : `/${path}`;
2174
+ httpAdapter.get(path, (_req, res) => {
2175
+ httpAdapter.reply(res, bodyFactory(), 200, {
2176
+ "Content-Type": contentType
2177
+ });
2178
+ });
2179
+ }
2180
+ setupSwagger() {
2181
+ const document = this.createDocument();
2182
+ const swaggerUI = new SwaggerCustomUI({
2183
+ title: this.options.title,
2184
+ path: this.options.path || SWAGGER_UI_CONSTANTS.path,
2185
+ persistAuthorization: this.options.persistAuthorization ?? true,
2186
+ assetsPath: this.options.assetsPath,
2187
+ faviconPath: this.options.faviconFilePath,
2188
+ topbarIconPath: this.options.topbarIconFilePath,
2189
+ topbarBgColor: this.options.topbarBgColor
2190
+ });
2191
+ swagger.SwaggerModule.setup(this.options.path || SWAGGER_UI_CONSTANTS.path, this.app, document, {
2192
+ ...swaggerUI.getCustomOptions(),
2193
+ jsonDocumentUrl: this.options.openApiJsonPath ? `${this.options.openApiJsonPath}` : undefined,
2194
+ yamlDocumentUrl: this.options.openApiYamlPath ? `${this.options.openApiYamlPath}` : undefined
2195
+ });
2196
+ this.registerOpenApiRoute(this.options.openApiJsonPath, "application/json", () => document);
2197
+ this.registerOpenApiRoute(this.options.openApiYamlPath, "application/x-yaml", () => YAML.stringify(document));
2198
+ }
2199
+ }
2200
+ class NestBootstraper {
2201
+ static get logger() {
2202
+ if (!this._logger) {
2203
+ this._logger = new common.Logger("NestBootstrap");
2204
+ }
2205
+ return this._logger;
2206
+ }
2207
+ static initialize(app) {
2208
+ this.app = app;
2209
+ return this;
2210
+ }
2211
+ static enableLogger(customLogger) {
2212
+ this._logger = customLogger || new common.Logger("NestBootstrap");
2213
+ this.app.useLogger(this._logger);
2214
+ return this;
2215
+ }
2216
+ static enableCors(origins = [], allowMethods = [ "GET", "POST", "PUT", "DELETE" ]) {
2217
+ const allowedOrigins = origins === "*" ? "*" : origins.map(o => o.trim().toLowerCase());
2218
+ const corsOptions = {
2219
+ origin: (origin, callback) => {
2220
+ if (!origin) return callback(null, true);
2221
+ if (allowedOrigins === "*" || Array.isArray(allowedOrigins) && allowedOrigins.includes(origin.toLowerCase())) {
2222
+ return callback(null, true);
2223
+ }
2224
+ callback(new CorsError(`Origin ${origin} not allowed`));
2225
+ },
2226
+ credentials: true,
2227
+ methods: allowMethods.join(",")
2228
+ };
2229
+ this.app.enableCors(corsOptions);
2230
+ return this;
2231
+ }
2232
+ static useHelmet(options) {
2233
+ try {
2234
+ const helmet = require("helmet");
2235
+ this.app.use(helmet(options));
2236
+ this.logger.log("Helmet middleware enabled successfully.");
2237
+ } catch (e) {
2238
+ this.logger.warn("Helmet not installed. Skipping middleware.");
2239
+ }
2240
+ return this;
2241
+ }
2242
+ static setupSwagger(options) {
2243
+ const swagger = new SwaggerBuilder(this.app, {
2244
+ title: options.title,
2245
+ description: options.description,
2246
+ version: options.version,
2247
+ path: options.path || "api",
2248
+ persistAuthorization: options.persistAuthorization ?? true,
2249
+ assetsPath: options.assetsPath,
2250
+ faviconFilePath: options.faviconPath,
2251
+ topbarIconFilePath: options.topbarIconPath,
2252
+ topbarBgColor: options.topbarBgColor,
2253
+ openApiJsonPath: options.openApiJsonPath,
2254
+ openApiYamlPath: options.openApiYamlPath
2255
+ });
2256
+ swagger.setupSwagger();
2257
+ return this;
2258
+ }
2259
+ static useGlobalPipes(...pipes) {
2260
+ if (pipes.length > 0) this.app.useGlobalPipes(...pipes);
2261
+ return this;
2262
+ }
2263
+ static useGlobalFilters(...filters) {
2264
+ this.app.useGlobalFilters(...filters.length > 0 ? filters : [ new exports.DecafExceptionFilter ]);
2265
+ return this;
2266
+ }
2267
+ static useGlobalInterceptors(...interceptors) {
2268
+ if (interceptors.length > 0) this.app.useGlobalInterceptors(...interceptors);
2269
+ return this;
2270
+ }
2271
+ static async start(port = Number(process.env.PORT) || 3e3, host = undefined, log = true) {
2272
+ this.app.listen(port, host).then(async () => {
2273
+ if (log) {
2274
+ const url = await this.app.getUrl();
2275
+ this.logger.log(`🚀 Application is running at: ${url}`);
2276
+ }
2277
+ });
2278
+ }
2279
+ }
2280
+ const pendingProviders = new Map;
2281
+ function getRegisteredDecafProviders() {
2282
+ return Array.from(pendingProviders.values());
2283
+ }
2284
+ function registerProvider(token, useFactory) {
2285
+ if (pendingProviders.has(token)) return;
2286
+ pendingProviders.set(token, {
2287
+ provide: token,
2288
+ useFactory: useFactory
2289
+ });
2290
+ }
2291
+ function isModelConstructor(value) {
2292
+ return typeof value === "function" && value.prototype instanceof decoratorValidation.Model;
2293
+ }
2294
+ function modelServiceToken(model) {
2295
+ return `${model.name}Service`;
2296
+ }
2297
+ function modelRepositoryToken(model, flavour) {
2298
+ return flavour ? `${model.name}Repository@${flavour}` : `${model.name}Repository`;
2299
+ }
2300
+ function Service(key) {
2301
+ return (target, propertyKey, parameterIndex) => {
2302
+ let resolved = key;
2303
+ if (typeof resolved === "undefined") {
2304
+ const paramTypes = Reflect.getMetadata("design:paramtypes", target) || [];
2305
+ resolved = paramTypes[parameterIndex];
2306
+ }
2307
+ if (typeof resolved === "undefined") {
2308
+ throw new Error(`@Service() could not determine an injection type for parameter ${parameterIndex} of ${target.name}. Provide an explicit argument, e.g. @Service(SomeService).`);
2309
+ }
2310
+ const asModel = typeof resolved !== "string" && isModelConstructor(resolved);
2311
+ const token = asModel ? modelServiceToken(resolved) : core.injectableServiceKey(resolved);
2312
+ registerProvider(token, () => {
2313
+ if (asModel) return core.ModelService.forModel(resolved);
2314
+ return typeof resolved === "string" ? core.Service.get(resolved) : core.Service.get(resolved);
2315
+ });
2316
+ return common.Inject(token)(target, propertyKey, parameterIndex);
2317
+ };
2318
+ }
2319
+ function Repository(model, flavour) {
2320
+ return (target, propertyKey, parameterIndex) => {
2321
+ const token = modelRepositoryToken(model, flavour);
2322
+ registerProvider(token, () => core.Repository.forModel(model, flavour));
2323
+ return common.Inject(token)(target, propertyKey, parameterIndex);
2324
+ };
2325
+ }
2326
+ var DecafCoreModule_1;
2327
+ exports.DecafCoreModule = class DecafCoreModule {
2328
+ static {
2329
+ DecafCoreModule_1 = this;
2330
+ }
2331
+ static get persistence() {
2332
+ if (!this._persistence) throw new dbDecorators.InternalError("Persistence service not initialized");
2333
+ return this._persistence;
2334
+ }
2335
+ static get log() {
2336
+ if (!this._logger) this._logger = logging.Logging.for(DecafCoreModule_1);
2337
+ return this._logger;
2338
+ }
2339
+ constructor(options, moduleRef) {
2340
+ this.options = options;
2341
+ this.moduleRef = moduleRef;
2342
+ }
2343
+ static forRoot(options) {
2344
+ const log = this.log.for(this.forRoot);
2345
+ const decafProviders = getRegisteredDecafProviders();
2346
+ return {
2347
+ module: DecafCoreModule_1,
2348
+ providers: [ {
2349
+ provide: DECAF_MODULE_OPTIONS,
2350
+ useValue: options
2351
+ }, {
2352
+ provide: DECAF_ADAPTER_ID,
2353
+ useValue: this.persistence?.client
2354
+ }, {
2355
+ provide: DECAF_HANDLERS,
2356
+ useFactory: () => options.handlers?.map(H => {
2357
+ log.info(`Registered request handler: ${H.name}`);
2358
+ return new H;
2359
+ }) ?? []
2360
+ }, exports.DecafRequestContext, exports.DecafHandlerExecutor, {
2361
+ provide: core$1.APP_INTERCEPTOR,
2362
+ useClass: exports.DecafRequestHandlerInterceptor
2363
+ }, ...decafProviders ],
2364
+ exports: [ DECAF_MODULE_OPTIONS, DECAF_ADAPTER_ID, DECAF_HANDLERS, exports.DecafRequestContext, exports.DecafHandlerExecutor, ...decafProviders.map(provider => provider.provide) ]
2365
+ };
2366
+ }
2367
+ static async bootPersistence(options) {
2368
+ const log = this.log.for(this.bootPersistence);
2369
+ if (!this._persistence) {
2370
+ const trimmed = options.conf.map(([contr, cfg, ...args]) => {
2371
+ const possible = args.pop();
2372
+ if (!possible) return [ contr, cfg ];
2373
+ return [ contr, cfg, ...args ];
2374
+ });
2375
+ this._persistence = new core.PersistenceService;
2376
+ await this._persistence.boot(trimmed);
2377
+ const clients = this._persistence.client;
2378
+ for (let i = 0; i < clients.length; i++) {
2379
+ const c = options.conf[i];
2380
+ const possibleTransf = c.slice(2, c.length);
2381
+ let transformer = possibleTransf.pop();
2382
+ if (!transformer || !transformer.from) {
2383
+ const contr = core.Adapter.transformerFor(clients[i].flavour);
2384
+ if (!contr) throw new dbDecorators.InternalError(`No transformer found for flavour ${clients[i].flavour}. you should either @requestToContextTransformer or provide a transformer in the config`);
2385
+ try {
2386
+ transformer = contr.from ? contr : new contr;
2387
+ } catch (e) {
2388
+ throw new dbDecorators.InternalError(`Failed to boot transformer for ${clients[i].flavour}: ${e}`);
2389
+ }
2390
+ }
2391
+ server.requestToContextTransformer(clients[i].flavour)(transformer);
2392
+ }
2393
+ log.info("persistence layer created successfully!");
2394
+ if (options.initialization) {
2395
+ try {
2396
+ await options.initialization();
2397
+ } catch (e) {
2398
+ throw new dbDecorators.InternalError(`Failed to initialize application: ${e}`);
2399
+ }
2400
+ }
2401
+ }
2402
+ return this.persistence.client;
2403
+ }
2404
+ async onApplicationShutdown() {
2405
+ const log = DecafCoreModule_1.log.for(this.onApplicationShutdown);
2406
+ const adapters = this.moduleRef.get(DECAF_ADAPTER_ID);
2407
+ for (const adapter of adapters) try {
2408
+ if (adapter) {
2409
+ log.info(`Shutting down ${adapter.toString()}`);
2410
+ await adapter.shutdown();
2411
+ }
2412
+ } catch (e) {
2413
+ log.error(`Failed to shutdown application`, e);
2414
+ }
2415
+ try {
2416
+ await core.Service.shutdown();
2417
+ } catch (e) {
2418
+ log.error(`Failed to shutdown services`, e);
2419
+ }
2420
+ }
2421
+ };
2422
+ exports.DecafCoreModule = DecafCoreModule_1 = tslib.__decorate([ common.Global(), common.Module({}), tslib.__param(0, common.Inject(DECAF_MODULE_OPTIONS)), tslib.__metadata("design:paramtypes", [ Object, core$1.ModuleRef ]) ], exports.DecafCoreModule);
2423
+ const LISTENING_ADAPTERS_FLAVOURS = Symbol("LISTENING_ADAPTERS_FLAVOURS");
2424
+ function normalizeEventResponse(args) {
2425
+ const [modelConstr, operation, id, payload] = args;
2426
+ const modelName = modelConstr?.name ?? modelConstr;
2427
+ const serializedPayload = Array.isArray(payload) ? payload.map(e => {
2428
+ try {
2429
+ if (typeof e.serialize === "function") return e.serialize();
2430
+ console.warn(`Payload item for ${modelName} does not have serialize method and is an ${typeof e}, attempting to stringify directly. Item: ${e}`);
2431
+ return typeof e === "string" ? e : JSON.stringify(e);
2432
+ } catch (err) {
2433
+ console.warn(`Failed to serialize payload for ${modelName}: ${err}`);
2434
+ return undefined;
2435
+ }
2436
+ }) : payload && typeof payload.serialize === "function" ? payload.serialize() : typeof payload === "string" ? payload : JSON.stringify(payload);
2437
+ console.debug(`Normalized event response for model ${modelName}, operation ${operation}, id ${id}:`, serializedPayload);
2438
+ return [ modelName, operation, id, serializedPayload ];
2439
+ }
2440
+ var EventsController_1;
2441
+ exports.EventsController = EventsController_1 = class EventsController extends DecafController {
2442
+ constructor(clientContext, flavours) {
2443
+ super(clientContext, EventsController_1.name);
2444
+ this.adapters = flavours.map(flavour => core.Adapter.get(flavour));
2445
+ }
2446
+ listen() {
2447
+ const logger = logging.Logging.for(EventsController_1.name);
2448
+ const events$ = new rxjs.Observable(observer => {
2449
+ const observerId = `B-${Math.random().toString(36).slice(2, 8)}`.toUpperCase();
2450
+ logger.info(`Creating SSE observer: ${observerId} for client ${this.clientContext.uuid}`);
2451
+ const cb = new class {
2452
+ constructor() {
2453
+ this.observerId = observerId;
2454
+ }
2455
+ refresh(...args) {
2456
+ logger.debug(`SSE observer ${this.observerId} received refresh event`);
2457
+ return Promise.resolve().then(() => {
2458
+ const data = normalizeEventResponse(args);
2459
+ observer.next({
2460
+ type: "message",
2461
+ data: data
2462
+ });
2463
+ logger.debug(`SSE observer ${this.observerId} event pushed to client`);
2464
+ });
2465
+ }
2466
+ };
2467
+ logger.verbose(`Registering observer ${observerId} across ${this.adapters.length} adapter(s)`);
2468
+ for (const adapter of this.adapters) {
2469
+ const adapterName = adapter?.constructor?.name ?? "UnknownAdapter";
2470
+ try {
2471
+ logger.debug(`Registering observer ${observerId} in adapter ${adapterName}`);
2472
+ adapter.observe(cb);
2473
+ } catch (e) {
2474
+ logger.debug(`Failed to register observer ${observerId} in adapter ${adapterName}: ${e?.message || e}`);
2475
+ logger.error(e);
2476
+ }
2477
+ }
2478
+ return () => {
2479
+ logger.debug(`Cleaning up SSE observer ${observerId}`);
2480
+ for (const adapter of this.adapters) {
2481
+ const adapterName = adapter?.constructor?.name ?? "UnknownAdapter";
2482
+ try {
2483
+ logger.debug(`Unregistering observer ${observerId} from adapter ${adapterName}`);
2484
+ adapter.unObserve(cb);
2485
+ } catch (e) {
2486
+ logger.debug(`Failed during cleanup of observer ${observerId} in adapter ${adapterName}: ${e?.message || e}`);
2487
+ logger.error(e);
2488
+ }
2489
+ }
2490
+ };
2491
+ });
2492
+ const HEARTBEAT_INTERVAL_MS = 15e3;
2493
+ const heartbeat$ = rxjs.interval(HEARTBEAT_INTERVAL_MS).pipe(operators.tap(() => {
2494
+ logger.debug("Sending heartbeat");
2495
+ }), operators.map(() => ({
2496
+ type: "heartbeat",
2497
+ data: {
2498
+ ts: (new Date).toISOString()
2499
+ }
2500
+ })));
2501
+ return rxjs.merge(events$, heartbeat$);
2502
+ }
2503
+ listenForModel(model) {
2504
+ const logger = logging.Logging.for(EventsController_1.name);
2505
+ return new rxjs.Observable(observer => {
2506
+ const cb = new class {
2507
+ refresh(...args) {
2508
+ return Promise.resolve().then(() => {
2509
+ observer.next({
2510
+ data: args
2511
+ });
2512
+ });
2513
+ }
2514
+ };
2515
+ try {
2516
+ for (const adapter of this.adapters) {
2517
+ adapter.observe(cb);
2518
+ }
2519
+ } catch (e) {
2520
+ observer.error(`Failed to observe event: ${e.message || e}`);
2521
+ }
2522
+ return () => {
2523
+ try {
2524
+ for (const adapter of this.adapters) {
2525
+ adapter.unObserve(cb);
2526
+ }
2527
+ } catch (e) {
2528
+ logger.error(e);
2529
+ }
2530
+ };
2531
+ });
2532
+ }
2533
+ };
2534
+ tslib.__decorate([ common.Sse(), tslib.__metadata("design:type", Function), tslib.__metadata("design:paramtypes", []), tslib.__metadata("design:returntype", rxjs.Observable) ], exports.EventsController.prototype, "listen", null);
2535
+ tslib.__decorate([ common.Sse("/:model"), tslib.__param(0, common.Query("model")), tslib.__metadata("design:type", Function), tslib.__metadata("design:paramtypes", [ String ]), tslib.__metadata("design:returntype", rxjs.Observable) ], exports.EventsController.prototype, "listenForModel", null);
2536
+ exports.EventsController = EventsController_1 = tslib.__decorate([ common.Controller(), tslib.__param(1, common.Inject(LISTENING_ADAPTERS_FLAVOURS)), tslib.__metadata("design:paramtypes", [ exports.DecafRequestContext, Array ]) ], exports.EventsController);
2537
+ var DecafStreamModule_1;
2538
+ exports.DecafStreamModule = DecafStreamModule_1 = class DecafStreamModule {
2539
+ static forFlavours(flavours, path = "events") {
2540
+ return {
2541
+ module: DecafStreamModule_1,
2542
+ controllers: [ exports.EventsController ],
2543
+ imports: [ core$1.RouterModule.register([ {
2544
+ path: path.replace(/^\//, ""),
2545
+ module: DecafStreamModule_1
2546
+ } ]) ],
2547
+ providers: [ exports.DecafRequestContext, {
2548
+ provide: LISTENING_ADAPTERS_FLAVOURS,
2549
+ useValue: flavours ?? []
2550
+ } ]
2551
+ };
2552
+ }
2553
+ };
2554
+ exports.DecafStreamModule = DecafStreamModule_1 = tslib.__decorate([ common.Module({}) ], exports.DecafStreamModule);
2555
+ var DecafModule_1;
2556
+ exports.DecafModule = DecafModule_1 = class DecafModule {
2557
+ static async forRootAsync(options) {
2558
+ const {autoControllers: autoControllers, autoServices: autoServices} = options;
2559
+ const adapters = await exports.DecafCoreModule.bootPersistence(options);
2560
+ const flavours = adapters.map(adapter => adapter.flavour);
2561
+ const imports = [ exports.DecafCoreModule.forRoot(options) ];
2562
+ if (autoControllers) {
2563
+ flavours.forEach(flavour => {
2564
+ imports.push(getModuleFor(flavour).forRoot(flavour, {
2565
+ autoServices: autoServices,
2566
+ controllerExposure: options.controllerExposure,
2567
+ controllerConfig: options.controllerConfig,
2568
+ aggregations: options.aggregations
2569
+ }));
2570
+ });
2571
+ }
2572
+ if (options.observerOptions?.enableObserverEvents) {
2573
+ imports.push(exports.DecafStreamModule.forFlavours(options.observerOptions.observerFlavours || flavours));
2574
+ }
2575
+ return {
2576
+ module: DecafModule_1,
2577
+ imports: imports
2578
+ };
2579
+ }
2580
+ };
2581
+ exports.DecafModule = DecafModule_1 = tslib.__decorate([ common.Module({}) ], exports.DecafModule);
2582
+ async function runMigrations() {}
2583
+ function repoForModel(model) {
2584
+ const m = decoratorValidation.Model.get(model);
2585
+ if (!m) throw new dbDecorators.InternalError(`Failed to find repository for ${model}`);
2586
+ const repo = core.Repository.forModel(m);
2587
+ return repo;
2588
+ }
2589
+ const VERSION = "0.11.2-refactor.2";
2590
+ const COMMIT = "85f1cec";
2591
+ const FULL_VERSION = "0.11.2-refactor.2-85f1cec";
2592
+ const PACKAGE_NAME = "@decaf-ts/for-nest";
2593
+ decoration.Metadata.allowReregistration(true);
2594
+ decoration.Metadata.registerLibrary(PACKAGE_NAME, VERSION);
2595
+ decoration.Metadata.allowReregistration(false);
2596
+ exports.AUTH_HANDLER = AUTH_HANDLER;
2597
+ exports.AUTH_META_KEY = AUTH_META_KEY;
2598
+ exports.ApiOperationFromModel = ApiOperationFromModel;
2599
+ exports.ApiParamsFromModel = ApiParamsFromModel;
2600
+ exports.ApiProperty = ApiProperty;
2601
+ exports.Auth = Auth;
2602
+ exports.BulkApiOperationFromModel = BulkApiOperationFromModel;
2603
+ exports.COMMIT = COMMIT;
2604
+ exports.CorsError = CorsError;
2605
+ exports.DECAF_ADAPTER_ID = DECAF_ADAPTER_ID;
2606
+ exports.DECAF_CONTEXT_KEY = DECAF_CONTEXT_KEY;
2607
+ exports.DECAF_CONTROLLER_CONFIG = DECAF_CONTROLLER_CONFIG;
2608
+ exports.DECAF_EXPOSE = DECAF_EXPOSE;
2609
+ exports.DECAF_HANDLERS = DECAF_HANDLERS;
2610
+ exports.DECAF_MODULE_OPTIONS = DECAF_MODULE_OPTIONS;
2611
+ exports.DECAF_ROUTE = DECAF_ROUTE;
2612
+ exports.DECAF_TASK_SERVICE_ID = DECAF_TASK_SERVICE_ID;
2613
+ exports.DECORATORS = DECORATORS;
2614
+ exports.DECORATORS_PREFIX = DECORATORS_PREFIX;
2615
+ exports.DecafAuthHandler = DecafAuthHandler;
2616
+ exports.DecafBody = DecafBody;
2617
+ exports.DecafController = DecafController;
2618
+ exports.DecafModelController = DecafModelController;
2619
+ exports.DecafParams = DecafParams;
2620
+ exports.DecafRoleAuthHandler = DecafRoleAuthHandler;
2621
+ exports.DtoFor = DtoFor;
2622
+ exports.FULL_VERSION = FULL_VERSION;
2623
+ exports.FromModelController = FromModelController;
2624
+ exports.IS_PUBLIC_KEY = IS_PUBLIC_KEY;
2625
+ exports.NestBootstraper = NestBootstraper;
2626
+ exports.PACKAGE_NAME = PACKAGE_NAME;
2627
+ exports.Public = Public;
2628
+ exports.REQUIRED_ROLES_KEY = REQUIRED_ROLES_KEY;
2629
+ exports.Repository = Repository;
2630
+ exports.RequireRoles = RequireRoles;
2631
+ exports.SKIP_MODEL_ROLES_KEY = SKIP_MODEL_ROLES_KEY;
2632
+ exports.Service = Service;
2633
+ exports.SwaggerBuilder = SwaggerBuilder;
2634
+ exports.UseDecafFilter = UseDecafFilter;
2635
+ exports.UseDecafHeaders = UseDecafHeaders;
2636
+ exports.VERSION = VERSION;
2637
+ exports.applyApiDecorators = applyApiDecorators;
2638
+ exports.controllerConfig = controllerConfig;
2639
+ exports.createApiPropertyDecorator = createApiPropertyDecorator;
2640
+ exports.createPropertyDecorator = createPropertyDecorator;
2641
+ exports.createRouteHandler = createRouteHandler;
2642
+ exports.defineRouteMethod = defineRouteMethod;
2643
+ exports.expose = expose;
2644
+ exports.getApiDecorators = getApiDecorators;
2645
+ exports.getEnumType = getEnumType;
2646
+ exports.getEnumValues = getEnumValues;
2647
+ exports.getModuleFor = getModuleFor;
2648
+ exports.getRegisteredDecafProviders = getRegisteredDecafProviders;
2649
+ exports.isUndefined = isUndefined;
2650
+ exports.repoForModel = repoForModel;
2651
+ exports.resolvePersistenceMethod = resolvePersistenceMethod;
2652
+ exports.runMigrations = runMigrations;
2653
+ });
2654
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLW5lc3QuY2pzIiwic291cmNlcyI6WyIuLi9zcmMvb3ZlcnJpZGVzL2NvbnN0YW50cy50cyIsIi4uL3NyYy9vdmVycmlkZXMvaGVscGVycy50cyIsIi4uL3NyYy9vdmVycmlkZXMvZGVjb3JhdGlvbi50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL0FwaU9wZXJhdGlvbkZyb21Nb2RlbC50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL0FwaVBhcmFtc0Zyb21Nb2RlbC50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL0RlY2FmQm9keS50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL0RlY2FmUGFyYW1zLnRzIiwiLi4vc3JjL2NvbnN0YW50cy50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL2V4cG9zZS50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL2NvbnRyb2xsZXItY29uZmlnLnRzIiwiLi4vc3JjL2F1dGgvY29uc3RhbnRzLnRzIiwiLi4vc3JjL3JlcXVlc3QvRGVjYWZSZXF1ZXN0Q29udGV4dC50cyIsIi4uL3NyYy9hdXRoL0F1dGhJbnRlcmNlcHRvci50cyIsIi4uL3NyYy9hdXRoL2RlY29yYXRvcnMudHMiLCIuLi9zcmMvZGVjYWYtbW9kZWwvZGVjb3JhdG9ycy91dGlscy50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC91dGlscy50cyIsIi4uL3NyYy9hdXRoL0RlY2FmQXV0aEhhbmRsZXIudHMiLCIuLi9zcmMvb3ZlcnJpZGVzL292ZXJyaWRlcy50cyIsIi4uL3NyYy9vdmVycmlkZXMvTW9kZWxCdWlsZGVyRXh0ZW5zaW9ucy50cyIsIi4uL3NyYy9pbnRlcmNlcHRvcnMvRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yLnRzIiwiLi4vc3JjL2F1dGgvRGVjYWZBdXRoTW9kdWxlLnRzIiwiLi4vc3JjL3JlcXVlc3QvRGVjYWZIYW5kbGVyRXhlY3V0b3IudHMiLCIuLi9zcmMvcmVxdWVzdC9EZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IudHMiLCIuLi9zcmMvY29udHJvbGxlcnMudHMiLCIuLi9zcmMvZmFjdG9yeS9vcGVuYXBpL0R0b0J1aWxkZXIudHMiLCIuLi9zcmMvZGVjYWYtbW9kZWwvRnJvbU1vZGVsQ29udHJvbGxlci50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9EZWNhZk1vZGVsTW9kdWxlLnRzIiwiLi4vc3JjL2RlY29yYXRpb24udHMiLCIuLi9zcmMvZmFjdG9yeS9lcnJvcnMvY29ycy50cyIsIi4uL3NyYy9mYWN0b3J5L2Vycm9ycy90aHJvdHRsaW5nLnRzIiwiLi4vc3JjL2ZhY3RvcnkvZXhjZXB0aW9ucy9EZWNhZkVycm9yRmlsdGVyLnRzIiwiLi4vc3JjL2ZhY3RvcnkvZXhjZXB0aW9ucy9kZWNvcmF0b3JzLnRzIiwiLi4vc3JjL2ZhY3Rvcnkvb3BlbmFwaS9jb25zdGFudHMudHMiLCIuLi9zcmMvZmFjdG9yeS9vcGVuYXBpL1N3YWdnZXJDdXN0b21VSS50cyIsIi4uL3NyYy9mYWN0b3J5L29wZW5hcGkvU3dhZ2dlckJ1aWxkZXIudHMiLCIuLi9zcmMvZmFjdG9yeS9OZXN0Qm9vdHN0cmFwZXIudHMiLCIuLi9zcmMvZGVjb3JhdG9ycy50cyIsIi4uL3NyYy9jb3JlLW1vZHVsZS50cyIsIi4uL3NyYy9ldmVudHMtbW9kdWxlL2NvbnN0YW50LnRzIiwiLi4vc3JjL2V2ZW50cy1tb2R1bGUvdXRpbHMudHMiLCIuLi9zcmMvZXZlbnRzLW1vZHVsZS9FdmVudHNDb250cm9sbGVyLnRzIiwiLi4vc3JjL2V2ZW50cy1tb2R1bGUvRGVjYWZTdHJlYW1Nb2R1bGUudHMiLCIuLi9zcmMvbW9kdWxlLnRzIiwiLi4vc3JjL3V0aWxzLnRzIiwiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBERUNPUkFUT1JTX1BSRUZJWCA9IFwic3dhZ2dlclwiO1xuZXhwb3J0IGNvbnN0IERFQ09SQVRPUlMgPSB7XG4gIEFQSV9PUEVSQVRJT046IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlPcGVyYXRpb25gLFxuICBBUElfUkVTUE9OU0U6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlSZXNwb25zZWAsXG4gIEFQSV9QUk9EVUNFUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaVByb2R1Y2VzYCxcbiAgQVBJX0NPTlNVTUVTOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpQ29uc3VtZXNgLFxuICBBUElfVEFHUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaVVzZVRhZ3NgLFxuICBBUElfQ0FMTEJBQ0tTOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpQ2FsbGJhY2tzYCxcbiAgQVBJX1BBUkFNRVRFUlM6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlQYXJhbWV0ZXJzYCxcbiAgQVBJX0hFQURFUlM6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlIZWFkZXJzYCxcbiAgQVBJX01PREVMX1BST1BFUlRJRVM6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlNb2RlbFByb3BlcnRpZXNgLFxuICBBUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWTogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaU1vZGVsUHJvcGVydGllc0FycmF5YCxcbiAgQVBJX1NFQ1VSSVRZOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpU2VjdXJpdHlgLFxuICBBUElfRVhDTFVERV9FTkRQT0lOVDogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaUV4Y2x1ZGVFbmRwb2ludGAsXG4gIEFQSV9FWENMVURFX0NPTlRST0xMRVI6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlFeGNsdWRlQ29udHJvbGxlcmAsXG4gIEFQSV9FWFRSQV9NT0RFTFM6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlFeHRyYU1vZGVsc2AsXG4gIEFQSV9FWFRFTlNJT046IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlFeHRlbnNpb25gLFxuICBBUElfU0NIRU1BOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpU2NoZW1hYCxcbiAgQVBJX0RFRkFVTFRfR0VUVEVSOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpRGVmYXVsdEdldHRlcmAsXG4gIEFQSV9MSU5LOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpTGlua2AsXG59O1xuIiwiaW1wb3J0IHsgaXNBcnJheSwgaXNVbmRlZmluZWQsIG5lZ2F0ZSwgcGlja0J5IH0gZnJvbSBcImxvZGFzaFwiO1xuaW1wb3J0IHsgREVDT1JBVE9SUyB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuXG5leHBvcnQgY29uc3QgaXNDb25zdHJ1Y3RvciA9ICh2YWw6IGFueSk6IGJvb2xlYW4gPT4gdmFsID09PSBcImNvbnN0cnVjdG9yXCI7XG5leHBvcnQgY29uc3QgTUVUQURBVEFfRkFDVE9SWV9OQU1FID0gXCJfT1BFTkFQSV9NRVRBREFUQV9GQUNUT1JZXCI7XG5cbmV4cG9ydCBjb25zdCBNRVRIT0RfTUVUQURBVEEgPSBcIm1ldGhvZFwiO1xuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlTWV0aG9kRGVjb3JhdG9yPFQgPSBhbnk+KFxuICBtZXRha2V5OiBzdHJpbmcsXG4gIG1ldGFkYXRhOiBULFxuICB7IG92ZXJyaWRlRXhpc3RpbmcgfSA9IHsgb3ZlcnJpZGVFeGlzdGluZzogdHJ1ZSB9XG4pOiBNZXRob2REZWNvcmF0b3Ige1xuICByZXR1cm4gKFxuICAgIHRhcmdldDogb2JqZWN0LFxuICAgIGtleTogc3RyaW5nIHwgc3ltYm9sLFxuICAgIGRlc2NyaXB0b3I6IFByb3BlcnR5RGVzY3JpcHRvclxuICApID0+IHtcbiAgICBpZiAodHlwZW9mIG1ldGFkYXRhID09PSBcIm9iamVjdFwiKSB7XG4gICAgICBjb25zdCBwcmV2VmFsdWUgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIGRlc2NyaXB0b3IudmFsdWUpO1xuICAgICAgaWYgKHByZXZWYWx1ZSAmJiAhb3ZlcnJpZGVFeGlzdGluZykge1xuICAgICAgICByZXR1cm4gZGVzY3JpcHRvcjtcbiAgICAgIH1cbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICAgIG1ldGFrZXksXG4gICAgICAgIHsgLi4ucHJldlZhbHVlLCAuLi5tZXRhZGF0YSB9LFxuICAgICAgICBkZXNjcmlwdG9yLnZhbHVlXG4gICAgICApO1xuICAgICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gICAgfVxuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgbWV0YWRhdGEsIGRlc2NyaXB0b3IudmFsdWUpO1xuICAgIHJldHVybiBkZXNjcmlwdG9yO1xuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlQ2xhc3NEZWNvcmF0b3I8VCBleHRlbmRzIEFycmF5PGFueT4gPSBhbnk+KFxuICBtZXRha2V5OiBzdHJpbmcsXG4gIG1ldGFkYXRhOiBUID0gW10gYXMgYW55XG4pOiBDbGFzc0RlY29yYXRvciB7XG4gIHJldHVybiAodGFyZ2V0KSA9PiB7XG4gICAgY29uc3QgcHJldlZhbHVlID0gUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCB0YXJnZXQpIHx8IFtdO1xuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgWy4uLnByZXZWYWx1ZSwgLi4ubWV0YWRhdGFdLCB0YXJnZXQpO1xuICAgIHJldHVybiB0YXJnZXQ7XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQcm9wZXJ0eURlY29yYXRvcjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55PiA9IGFueT4oXG4gIG1ldGFrZXk6IHN0cmluZyxcbiAgbWV0YWRhdGE6IFQsXG4gIG92ZXJyaWRlRXhpc3RpbmcgPSB0cnVlXG4pOiBQcm9wZXJ0eURlY29yYXRvciB7XG4gIHJldHVybiAoKHRhcmdldDogb2JqZWN0LCBwcm9wZXJ0eUtleTogc3RyaW5nKSA9PiB7XG4gICAgY29uc3QgcHJvcGVydGllcyA9XG4gICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKERFQ09SQVRPUlMuQVBJX01PREVMX1BST1BFUlRJRVNfQVJSQVksIHRhcmdldCkgfHwgW107XG5cbiAgICBjb25zdCBrZXkgPSBgOiR7cHJvcGVydHlLZXl9YDtcbiAgICBpZiAoIXByb3BlcnRpZXMuaW5jbHVkZXMoa2V5KSkge1xuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgREVDT1JBVE9SUy5BUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWSxcbiAgICAgICAgWy4uLnByb3BlcnRpZXMsIGA6JHtwcm9wZXJ0eUtleX1gXSxcbiAgICAgICAgdGFyZ2V0XG4gICAgICApO1xuICAgIH1cbiAgICBjb25zdCBleGlzdGluZ01ldGFkYXRhID0gUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCB0YXJnZXQsIHByb3BlcnR5S2V5KTtcbiAgICBpZiAoZXhpc3RpbmdNZXRhZGF0YSkge1xuICAgICAgY29uc3QgbmV3TWV0YWRhdGEgPSBwaWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpO1xuICAgICAgY29uc3QgbWV0YWRhdGFUb1NhdmUgPSBvdmVycmlkZUV4aXN0aW5nXG4gICAgICAgID8ge1xuICAgICAgICAgICAgLi4uZXhpc3RpbmdNZXRhZGF0YSxcbiAgICAgICAgICAgIC4uLm5ld01ldGFkYXRhLFxuICAgICAgICAgIH1cbiAgICAgICAgOiB7XG4gICAgICAgICAgICAuLi5uZXdNZXRhZGF0YSxcbiAgICAgICAgICAgIC4uLmV4aXN0aW5nTWV0YWRhdGEsXG4gICAgICAgICAgfTtcblxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBtZXRhZGF0YVRvU2F2ZSwgdGFyZ2V0LCBwcm9wZXJ0eUtleSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHR5cGUgPVxuICAgICAgICAodGFyZ2V0Py5jb25zdHJ1Y3RvciBhcyBhbnkpPy5bTUVUQURBVEFfRkFDVE9SWV9OQU1FIGFzIGFueV0/LigpW1xuICAgICAgICAgIHByb3BlcnR5S2V5XG4gICAgICAgIF0/LnR5cGUgPz8gUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuXG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBtZXRha2V5LFxuICAgICAgICB7XG4gICAgICAgICAgdHlwZSxcbiAgICAgICAgICAuLi5waWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpLFxuICAgICAgICB9LFxuICAgICAgICB0YXJnZXQsXG4gICAgICAgIHByb3BlcnR5S2V5XG4gICAgICApO1xuICAgIH1cbiAgfSkgYXMgUHJvcGVydHlEZWNvcmF0b3I7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVNaXhlZERlY29yYXRvcjxUID0gYW55PihcbiAgbWV0YWtleTogc3RyaW5nLFxuICBtZXRhZGF0YTogVFxuKTogTWV0aG9kRGVjb3JhdG9yICYgQ2xhc3NEZWNvcmF0b3Ige1xuICByZXR1cm4gKFxuICAgIHRhcmdldDogb2JqZWN0LFxuICAgIGtleT86IHN0cmluZyB8IHN5bWJvbCxcbiAgICBkZXNjcmlwdG9yPzogVHlwZWRQcm9wZXJ0eURlc2NyaXB0b3I8YW55PlxuICApOiBhbnkgPT4ge1xuICAgIGlmIChkZXNjcmlwdG9yKSB7XG4gICAgICBsZXQgbWV0YWRhdGFzOiBhbnk7XG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShtZXRhZGF0YSkpIHtcbiAgICAgICAgY29uc3QgcHJldmlvdXNNZXRhZGF0YSA9XG4gICAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCBkZXNjcmlwdG9yLnZhbHVlKSB8fCBbXTtcbiAgICAgICAgbWV0YWRhdGFzID0gWy4uLnByZXZpb3VzTWV0YWRhdGEsIC4uLm1ldGFkYXRhXTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IHByZXZpb3VzTWV0YWRhdGEgPVxuICAgICAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgZGVzY3JpcHRvci52YWx1ZSkgfHwge307XG4gICAgICAgIG1ldGFkYXRhcyA9IHsgLi4ucHJldmlvdXNNZXRhZGF0YSwgLi4ubWV0YWRhdGEgfTtcbiAgICAgIH1cbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgbWV0YWRhdGFzLCBkZXNjcmlwdG9yLnZhbHVlKTtcbiAgICAgIHJldHVybiBkZXNjcmlwdG9yO1xuICAgIH1cblxuICAgIGxldCBtZXRhZGF0YXM6IGFueTtcbiAgICBpZiAoQXJyYXkuaXNBcnJheShtZXRhZGF0YSkpIHtcbiAgICAgIGNvbnN0IHByZXZpb3VzTWV0YWRhdGEgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCkgfHwgW107XG4gICAgICBtZXRhZGF0YXMgPSBbLi4ucHJldmlvdXNNZXRhZGF0YSwgLi4ubWV0YWRhdGFdO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBwcmV2aW91c01ldGFkYXRhID0gUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCB0YXJnZXQpIHx8IHt9O1xuICAgICAgbWV0YWRhdGFzID0gT2JqZWN0LmFzc2lnbihPYmplY3QuYXNzaWduKHt9LCBwcmV2aW91c01ldGFkYXRhKSwgbWV0YWRhdGEpO1xuICAgIH1cbiAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKG1ldGFrZXksIG1ldGFkYXRhcywgdGFyZ2V0KTtcbiAgICByZXR1cm4gdGFyZ2V0O1xuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlUGFyYW1EZWNvcmF0b3I8VCBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4gPSBhbnk+KFxuICBtZXRhZGF0YTogVCxcbiAgaW5pdGlhbDogUGFydGlhbDxUPlxuKTogTWV0aG9kRGVjb3JhdG9yICYgQ2xhc3NEZWNvcmF0b3Ige1xuICByZXR1cm4gKFxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGVcbiAgICB0YXJnZXQ6IG9iamVjdCB8IEZ1bmN0aW9uLFxuICAgIGtleT86IHN0cmluZyB8IHN5bWJvbCxcbiAgICBkZXNjcmlwdG9yPzogVHlwZWRQcm9wZXJ0eURlc2NyaXB0b3I8YW55PlxuICApOiBhbnkgPT4ge1xuICAgIGNvbnN0IHBhcmFtT3B0aW9ucyA9IHtcbiAgICAgIC4uLmluaXRpYWwsXG4gICAgICAuLi5waWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpLFxuICAgIH07XG5cbiAgICBpZiAoZGVzY3JpcHRvcikge1xuICAgICAgY29uc3QgcGFyYW1ldGVycyA9XG4gICAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoREVDT1JBVE9SUy5BUElfUEFSQU1FVEVSUywgZGVzY3JpcHRvci52YWx1ZSkgfHwgW107XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBERUNPUkFUT1JTLkFQSV9QQVJBTUVURVJTLFxuICAgICAgICBbLi4ucGFyYW1ldGVycywgcGFyYW1PcHRpb25zXSxcbiAgICAgICAgZGVzY3JpcHRvci52YWx1ZVxuICAgICAgKTtcbiAgICAgIHJldHVybiBkZXNjcmlwdG9yO1xuICAgIH1cblxuICAgIGlmICh0eXBlb2YgdGFyZ2V0ID09PSBcIm9iamVjdFwiKSB7XG4gICAgICByZXR1cm4gdGFyZ2V0O1xuICAgIH1cblxuICAgIGNvbnN0IHByb3BlcnR5S2V5cyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzKHRhcmdldC5wcm90b3R5cGUpO1xuXG4gICAgZm9yIChjb25zdCBwcm9wZXJ0eUtleSBvZiBwcm9wZXJ0eUtleXMpIHtcbiAgICAgIGlmIChpc0NvbnN0cnVjdG9yKHByb3BlcnR5S2V5KSkge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgbWV0aG9kRGVzY3JpcHRvciA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoXG4gICAgICAgIHRhcmdldC5wcm90b3R5cGUsXG4gICAgICAgIHByb3BlcnR5S2V5XG4gICAgICApO1xuXG4gICAgICBpZiAoIW1ldGhvZERlc2NyaXB0b3IpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGlzQXBpTWV0aG9kID0gUmVmbGVjdC5oYXNNZXRhZGF0YShcbiAgICAgICAgTUVUSE9EX01FVEFEQVRBLFxuICAgICAgICBtZXRob2REZXNjcmlwdG9yLnZhbHVlXG4gICAgICApO1xuXG4gICAgICBpZiAoIWlzQXBpTWV0aG9kKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBwYXJhbWV0ZXJzID1cbiAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcbiAgICAgICAgICBERUNPUkFUT1JTLkFQSV9QQVJBTUVURVJTLFxuICAgICAgICAgIG1ldGhvZERlc2NyaXB0b3IudmFsdWVcbiAgICAgICAgKSB8fCBbXTtcbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICAgIERFQ09SQVRPUlMuQVBJX1BBUkFNRVRFUlMsXG4gICAgICAgIFsuLi5wYXJhbWV0ZXJzLCBwYXJhbU9wdGlvbnNdLFxuICAgICAgICBtZXRob2REZXNjcmlwdG9yLnZhbHVlXG4gICAgICApO1xuICAgIH1cbiAgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldFR5cGVJc0FycmF5VHVwbGUoXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGVcbiAgaW5wdXQ6IEZ1bmN0aW9uIHwgW0Z1bmN0aW9uXSB8IHVuZGVmaW5lZCB8IHN0cmluZyB8IFJlY29yZDxzdHJpbmcsIGFueT4sXG4gIGlzQXJyYXlGbGFnOiBib29sZWFuXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGVcbik6IFtGdW5jdGlvbiB8IHVuZGVmaW5lZCwgYm9vbGVhbl0ge1xuICBpZiAoIWlucHV0KSB7XG4gICAgcmV0dXJuIFtpbnB1dCBhcyB1bmRlZmluZWQsIGlzQXJyYXlGbGFnXTtcbiAgfVxuICBpZiAoaXNBcnJheUZsYWcpIHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1mdW5jdGlvbi10eXBlXG4gICAgcmV0dXJuIFtpbnB1dCBhcyBGdW5jdGlvbiwgaXNBcnJheUZsYWddO1xuICB9XG4gIGNvbnN0IGlzSW5wdXRBcnJheSA9IGlzQXJyYXkoaW5wdXQpO1xuICBjb25zdCB0eXBlID0gaXNJbnB1dEFycmF5ID8gaW5wdXRbMF0gOiBpbnB1dDtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtZnVuY3Rpb24tdHlwZVxuICByZXR1cm4gW3R5cGUgYXMgRnVuY3Rpb24sIGlzSW5wdXRBcnJheV07XG59XG4iLCJpbXBvcnQgeyBUeXBlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBpc1N0cmluZywgbmVnYXRlLCBwaWNrQnkgfSBmcm9tIFwibG9kYXNoXCI7XG5pbXBvcnQgeyBERUNPUkFUT1JTIH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBnZXRUeXBlSXNBcnJheVR1cGxlLCBNRVRBREFUQV9GQUNUT1JZX05BTUUgfSBmcm9tIFwiLi9oZWxwZXJzXCI7XG5pbXBvcnQgdHlwZSB7XG4gIEVudW1BbGxvd2VkVHlwZXMsXG4gIEVudW1TY2hlbWFBdHRyaWJ1dGVzLFxuICBTY2hlbWFPYmplY3RNZXRhZGF0YSxcbiAgU3dhZ2dlckVudW1UeXBlLFxufSBmcm9tIFwiLi4vc3dhZ2dlci10eXBlc1wiO1xuXG5leHBvcnQgY29uc3QgaXNVbmRlZmluZWQgPSAob2JqOiBhbnkpOiBvYmogaXMgdW5kZWZpbmVkID0+XG4gIHR5cGVvZiBvYmogPT09IFwidW5kZWZpbmVkXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRFbnVtVmFsdWVzKFxuICBlbnVtVHlwZTogU3dhZ2dlckVudW1UeXBlIHwgKCgpID0+IFN3YWdnZXJFbnVtVHlwZSlcbik6IHN0cmluZ1tdIHwgbnVtYmVyW10ge1xuICBpZiAodHlwZW9mIGVudW1UeXBlID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICByZXR1cm4gZ2V0RW51bVZhbHVlcyhlbnVtVHlwZSgpKTtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KGVudW1UeXBlKSkge1xuICAgIHJldHVybiBlbnVtVHlwZSBhcyBzdHJpbmdbXTtcbiAgfVxuICBpZiAodHlwZW9mIGVudW1UeXBlICE9PSBcIm9iamVjdFwiKSB7XG4gICAgcmV0dXJuIFtdO1xuICB9XG4gIC8vIEVudW1zIHdpdGggbnVtZXJpYyB2YWx1ZXNcbiAgLy8gICBlbnVtIFNpemUge1xuICAvLyAgICAgU01BTEwgPSAxLFxuICAvLyAgICAgQklHID0gMlxuICAvLyAgIH1cbiAgLy8gYXJlIHRyYW5zcGlsZWQgdG8gaW5jbHVkZSBhIHJldmVyc2UgbWFwcGluZ1xuICAvLyAgIGNvbnN0IFNpemUgPSB7XG4gIC8vICAgICBcIjFcIjogXCJTTUFMTFwiLFxuICAvLyAgICAgXCIyXCI6IFwiQklHXCIsXG4gIC8vICAgICBcIlNNQUxMXCI6IDEsXG4gIC8vICAgICBcIkJJR1wiOiAyLFxuICAvLyAgIH1cbiAgY29uc3QgbnVtZXJpY1ZhbHVlcyA9IE9iamVjdC52YWx1ZXMoZW51bVR5cGUpXG4gICAgLmZpbHRlcigodmFsdWUpID0+IHR5cGVvZiB2YWx1ZSA9PT0gXCJudW1iZXJcIilcbiAgICAubWFwKCh2YWx1ZTogYW55KSA9PiB2YWx1ZS50b1N0cmluZygpKTtcblxuICByZXR1cm4gT2JqZWN0LmtleXMoZW51bVR5cGUpXG4gICAgLmZpbHRlcigoa2V5KSA9PiAhbnVtZXJpY1ZhbHVlcy5pbmNsdWRlcyhrZXkpKVxuICAgIC5tYXAoKGtleSkgPT4gZW51bVR5cGVba2V5IGFzIGFueV0pO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0RW51bVR5cGUodmFsdWVzOiAoc3RyaW5nIHwgbnVtYmVyKVtdKTogXCJzdHJpbmdcIiB8IFwibnVtYmVyXCIge1xuICBjb25zdCBoYXNTdHJpbmcgPSB2YWx1ZXMuZmlsdGVyKGlzU3RyaW5nKS5sZW5ndGggPiAwO1xuICByZXR1cm4gaGFzU3RyaW5nID8gXCJzdHJpbmdcIiA6IFwibnVtYmVyXCI7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQcm9wZXJ0eURlY29yYXRvcjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55PiA9IGFueT4oXG4gIG1ldGFrZXk6IHN0cmluZyxcbiAgbWV0YWRhdGE6IFQsXG4gIG92ZXJyaWRlRXhpc3RpbmcgPSB0cnVlXG4pOiBQcm9wZXJ0eURlY29yYXRvciB7XG4gIHJldHVybiAoKHRhcmdldDogb2JqZWN0LCBwcm9wZXJ0eUtleTogc3RyaW5nKSA9PiB7XG4gICAgY29uc3QgcHJvcGVydGllcyA9XG4gICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKERFQ09SQVRPUlMuQVBJX01PREVMX1BST1BFUlRJRVNfQVJSQVksIHRhcmdldCkgfHwgW107XG5cbiAgICBjb25zdCBrZXkgPSBgOiR7cHJvcGVydHlLZXl9YDtcbiAgICBpZiAoIXByb3BlcnRpZXMuaW5jbHVkZXMoa2V5KSkge1xuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgREVDT1JBVE9SUy5BUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWSxcbiAgICAgICAgWy4uLnByb3BlcnRpZXMsIGA6JHtwcm9wZXJ0eUtleX1gXSxcbiAgICAgICAgdGFyZ2V0XG4gICAgICApO1xuICAgIH1cbiAgICBjb25zdCBleGlzdGluZ01ldGFkYXRhID0gUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCB0YXJnZXQsIHByb3BlcnR5S2V5KTtcbiAgICBpZiAoZXhpc3RpbmdNZXRhZGF0YSkge1xuICAgICAgY29uc3QgbmV3TWV0YWRhdGEgPSBwaWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpO1xuICAgICAgY29uc3QgbWV0YWRhdGFUb1NhdmUgPSBvdmVycmlkZUV4aXN0aW5nXG4gICAgICAgID8ge1xuICAgICAgICAgICAgLi4uZXhpc3RpbmdNZXRhZGF0YSxcbiAgICAgICAgICAgIC4uLm5ld01ldGFkYXRhLFxuICAgICAgICAgIH1cbiAgICAgICAgOiB7XG4gICAgICAgICAgICAuLi5uZXdNZXRhZGF0YSxcbiAgICAgICAgICAgIC4uLmV4aXN0aW5nTWV0YWRhdGEsXG4gICAgICAgICAgfTtcblxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBtZXRhZGF0YVRvU2F2ZSwgdGFyZ2V0LCBwcm9wZXJ0eUtleSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHR5cGUgPVxuICAgICAgICAvLyBAdHMtZXhwZWN0LWVycm9yIG5lc3QganMgY29kZVxuICAgICAgICB0YXJnZXQ/LmNvbnN0cnVjdG9yPy5bTUVUQURBVEFfRkFDVE9SWV9OQU1FXT8uKClbcHJvcGVydHlLZXldPy50eXBlID8/XG4gICAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoXCJkZXNpZ246dHlwZVwiLCB0YXJnZXQsIHByb3BlcnR5S2V5KTtcblxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgbWV0YWtleSxcbiAgICAgICAge1xuICAgICAgICAgIHR5cGUsXG4gICAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLCAvLyBEZWZhdWx0IHRvIG9wdGlvbmFsIHVubGVzcyBAcmVxdWlyZWQoKSBpcyB1c2VkXG4gICAgICAgICAgLi4ucGlja0J5KG1ldGFkYXRhLCBuZWdhdGUoaXNVbmRlZmluZWQpKSxcbiAgICAgICAgfSxcbiAgICAgICAgdGFyZ2V0LFxuICAgICAgICBwcm9wZXJ0eUtleVxuICAgICAgKTtcbiAgICB9XG4gIH0pIGFzIFByb3BlcnR5RGVjb3JhdG9yO1xufVxuXG5leHBvcnQgdHlwZSBBcGlQcm9wZXJ0eUNvbW1vbk9wdGlvbnMgPSBTY2hlbWFPYmplY3RNZXRhZGF0YSAmIHtcbiAgXCJ4LWVudW1OYW1lc1wiPzogc3RyaW5nW107XG4gIC8qKlxuICAgKiBMYXp5IGZ1bmN0aW9uIHJldHVybmluZyB0aGUgdHlwZSBmb3Igd2hpY2ggdGhlIGRlY29yYXRlZCBwcm9wZXJ0eVxuICAgKiBjYW4gYmUgdXNlZCBhcyBhbiBpZFxuICAgKlxuICAgKiBVc2UgdG9nZXRoZXIgd2l0aCBAQXBpRGVmYXVsdEdldHRlciBvbiB0aGUgZ2V0dGVyIHJvdXRlIG9mIHRoZSB0eXBlXG4gICAqIHRvIGdlbmVyYXRlIE9wZW5BUEkgbGluayBvYmplY3RzXG4gICAqXG4gICAqIEBzZWUgW1N3YWdnZXIgbGluayBvYmplY3RzXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2xpbmtzLylcbiAgICovXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGVcbiAgbGluaz86ICgpID0+IFR5cGU8dW5rbm93bj4gfCBGdW5jdGlvbjtcbn07XG5cbmV4cG9ydCB0eXBlIEFwaVByb3BlcnR5T3B0aW9ucyA9XG4gIHwgQXBpUHJvcGVydHlDb21tb25PcHRpb25zXG4gIHwgKEFwaVByb3BlcnR5Q29tbW9uT3B0aW9ucyAmIHtcbiAgICAgIGVudW1OYW1lOiBzdHJpbmc7XG4gICAgICBlbnVtU2NoZW1hPzogRW51bVNjaGVtYUF0dHJpYnV0ZXM7XG4gICAgfSk7XG5cbmNvbnN0IGlzRW51bUFycmF5ID0gKFxuICBvcHRzOiBBcGlQcm9wZXJ0eU9wdGlvbnNcbik6IG9wdHMgaXMge1xuICBpc0FycmF5OiB0cnVlO1xuICBlbnVtOiBFbnVtQWxsb3dlZFR5cGVzO1xuICB0eXBlOiBhbnk7XG4gIGl0ZW1zOiBhbnk7XG59ID0+IChvcHRzLmlzQXJyYXkgJiYgXCJlbnVtXCIgaW4gb3B0cyAmJiBvcHRzLmVudW0gIT09IHVuZGVmaW5lZCkgYXMgYm9vbGVhbjtcblxuLyoqXG4gKiBAcHVibGljQXBpXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBBcGlQcm9wZXJ0eShcbiAgb3B0aW9uczogQXBpUHJvcGVydHlPcHRpb25zID0ge31cbik6IFByb3BlcnR5RGVjb3JhdG9yIHtcbiAgcmV0dXJuIGNyZWF0ZUFwaVByb3BlcnR5RGVjb3JhdG9yKG9wdGlvbnMpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlQXBpUHJvcGVydHlEZWNvcmF0b3IoXG4gIG9wdGlvbnM6IEFwaVByb3BlcnR5T3B0aW9ucyA9IHt9LFxuICBvdmVycmlkZUV4aXN0aW5nID0gdHJ1ZVxuKTogUHJvcGVydHlEZWNvcmF0b3Ige1xuICBjb25zdCBbdHlwZSwgaXNBcnJheV0gPSBnZXRUeXBlSXNBcnJheVR1cGxlKFxuICAgIG9wdGlvbnMudHlwZSxcbiAgICBvcHRpb25zLmlzQXJyYXkgYXMgYW55XG4gICk7XG4gIG9wdGlvbnMgPSB7XG4gICAgLi4ub3B0aW9ucyxcbiAgICB0eXBlLFxuICAgIGlzQXJyYXksXG4gIH0gYXMgQXBpUHJvcGVydHlPcHRpb25zO1xuXG4gIGlmIChpc0VudW1BcnJheShvcHRpb25zKSkge1xuICAgIG9wdGlvbnMudHlwZSA9IFwiYXJyYXlcIjtcblxuICAgIGNvbnN0IGVudW1WYWx1ZXMgPSBnZXRFbnVtVmFsdWVzKG9wdGlvbnMuZW51bSk7XG4gICAgb3B0aW9ucy5pdGVtcyA9IHtcbiAgICAgIHR5cGU6IGdldEVudW1UeXBlKGVudW1WYWx1ZXMpLFxuICAgICAgZW51bTogZW51bVZhbHVlcyxcbiAgICB9O1xuICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgbmVzdCBqcyBjb2RlXG4gICAgZGVsZXRlIG9wdGlvbnMuZW51bTtcbiAgfSBlbHNlIGlmIChcImVudW1cIiBpbiBvcHRpb25zICYmIG9wdGlvbnMuZW51bSAhPT0gdW5kZWZpbmVkKSB7XG4gICAgY29uc3QgZW51bVZhbHVlcyA9IGdldEVudW1WYWx1ZXMob3B0aW9ucy5lbnVtKTtcblxuICAgIG9wdGlvbnMuZW51bSA9IGVudW1WYWx1ZXM7XG4gICAgb3B0aW9ucy50eXBlID0gZ2V0RW51bVR5cGUoZW51bVZhbHVlcyk7XG4gIH1cblxuICBpZiAoQXJyYXkuaXNBcnJheShvcHRpb25zLnR5cGUpKSB7XG4gICAgb3B0aW9ucy50eXBlID0gXCJhcnJheVwiO1xuICAgIG9wdGlvbnMuaXRlbXMgPSB7XG4gICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICBpdGVtczoge1xuICAgICAgICB0eXBlOiBvcHRpb25zLnR5cGVbMF0sXG4gICAgICB9LFxuICAgIH07XG4gIH1cblxuICByZXR1cm4gY3JlYXRlUHJvcGVydHlEZWNvcmF0b3IoXG4gICAgREVDT1JBVE9SUy5BUElfTU9ERUxfUFJPUEVSVElFUyxcbiAgICBvcHRpb25zLFxuICAgIG92ZXJyaWRlRXhpc3RpbmdcbiAgKTtcbn1cbiIsImltcG9ydCB7IGFwcGx5LCBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgQXBpRXhjbHVkZUVuZHBvaW50IH0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlclwiO1xuaW1wb3J0IHtcbiAgQnVsa0NydWRPcGVyYXRpb25LZXlzLFxuICBPcGVyYXRpb25LZXlzLFxuICBDcnVkT3BlcmF0aW9ucyxcbn0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBNb2RlbENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgRGVsZXRlLCBHZXQsIFBhdGNoLCBQb3N0LCBQdXQgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcblxuaW1wb3J0IHsgaXNPcGVyYXRpb25CbG9ja2VkIGFzIGNvcmVJc09wZXJhdGlvbkJsb2NrZWQgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IFByZXBhcmVkU3RhdGVtZW50S2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgSHR0cFZlcmJzIH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRGV0ZXJtaW5lcyBpZiBhIGdpdmVuIENSVUQgb3BlcmF0aW9uIGlzIGJsb2NrZWQgZm9yIGEgc3BlY2lmaWMgbW9kZWwgY29uc3RydWN0b3IuXG4gKiBAc3VtbWFyeSBSZXRyaWV2ZXMgdGhlIG9wZXJhdGlvbi1ibG9ja2luZyBoYW5kbGVyIG1ldGFkYXRhIHN0b3JlZCBvbiB0aGUgcHJvdmlkZWQgbW9kZWwgY29uc3RydWN0b3IgKHVuZGVyIGBPcGVyYXRpb25LZXlzLlJFRkxFQ1QgKyBPcGVyYXRpb25LZXlzLkJMT0NLYCksIGV4ZWN1dGVzIGl0IChpZiBwcmVzZW50KSB3aXRoIGl0cyBwZXJzaXN0ZWQgYXJndW1lbnRzIHBsdXMgdGhlIHJlcXVlc3RlZCBvcGVyYXRpb24sIGFuZCByZXR1cm5zIHdoZXRoZXIgdGhlIG9wZXJhdGlvbiBpcyBibG9ja2VkLiBJZiBubyBoYW5kbGVyIGV4aXN0cywgdGhlIG9wZXJhdGlvbiBpcyBjb25zaWRlcmVkIGFsbG93ZWQgKHJldHVybnMgYGZhbHNlYCkuXG4gKiBAcGFyYW0ge01vZGVsQ29uc3RydWN0b3I8YW55Pn0gTW9kZWxDb25zdHJ1Y3RvciAtIFRoZSB0YXJnZXQgbW9kZWwgY29uc3RydWN0b3Igd2hvc2UgbWV0YWRhdGEgbWF5IGluY2x1ZGUgYSBibG9ja2luZyBoYW5kbGVyLlxuICogQHBhcmFtIHtDcnVkT3BlcmF0aW9uc30gb3AgLSBUaGUgQ1JVRCBvcGVyYXRpb24gdG8gZXZhbHVhdGUgKGUuZy4sIGBPcGVyYXRpb25LZXlzLkNSRUFURWAsIGBPcGVyYXRpb25LZXlzLlJFQURgLCBgT3BlcmF0aW9uS2V5cy5VUERBVEVgLCBgT3BlcmF0aW9uS2V5cy5ERUxFVEVgKS5cbiAqIEByZXR1cm4ge2Jvb2xlYW59IGB0cnVlYCB3aGVuIHRoZSBvcGVyYXRpb24gaXMgZXhwbGljaXRseSBibG9ja2VkIGJ5IHRoZSBtb2RlbCdzIGhhbmRsZXI7IG90aGVyd2lzZSBgZmFsc2VgLlxuICogQGZ1bmN0aW9uIGlzT3BlcmF0aW9uQmxvY2tlZFxuICovXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDb25kaXRpb25hbGx5IGFwcGxpZXMgYW4gSFRUUCBtZXRob2QgZGVjb3JhdG9yIGZvciBhIGdpdmVuIG1vZGVsIGFuZCB2ZXJiLCBoaWRpbmcgdGhlIGVuZHBvaW50IGluIFN3YWdnZXIgKGFuZCBub3QgcmVnaXN0ZXJpbmcgdGhlIHJvdXRlKSB3aGVuIHRoZSBtb2RlbCBibG9ja3MgdGhhdCBDUlVEIG9wZXJhdGlvbi5cbiAqIEBzdW1tYXJ5IE1hcHMgYW4gSFRUUCB2ZXJiIHRvIGl0cyBjb3JyZXNwb25kaW5nIGBDcnVkT3BlcmF0aW9uc2Aga2V5IGFuZCBOZXN0IEhUVFAgZGVjb3JhdG9yIChgQEdldGAsIGBAUG9zdGAsIGV0Yy4pLiBJdCBjaGVja3MgYGlzT3BlcmF0aW9uQmxvY2tlZChNb2RlbENvbnN0cnVjdG9yLCBjcnVkT3ApYCBhbmQsIGlmIGJsb2NrZWQsIGFwcGxpZXMgb25seSBgQEFwaUV4Y2x1ZGVFbmRwb2ludCgpYCAoU3dhZ2dlci1oaWRkZW4sIG5vIE5lc3Qgcm91dGUpLiBJZiBwZXJtaXR0ZWQsIGl0IGFwcGxpZXMgdGhlIGFwcHJvcHJpYXRlIEhUVFAgZGVjb3JhdG9yIHdpdGggdGhlIG9wdGlvbmFsIGBwYXRoYC5cbiAqIEBwYXJhbSB7TW9kZWxDb25zdHJ1Y3Rvcjxhbnk+fSBNb2RlbENvbnN0cnVjdG9yIC0gVGhlIG1vZGVsIGNvbnN0cnVjdG9yIHVzZWQgdG8gcmVzb2x2ZSBvcGVyYXRpb24tYmxvY2tpbmcgcnVsZXMuXG4gKiBAcGFyYW0ge0h0dHBWZXJic30gdmVyYiAtIFRoZSBIVFRQIHZlcmIgdG8gbWFwIChlLmcuLCBgXCJHRVRcIiB8IFwiUE9TVFwiIHwgXCJQVVRcIiB8IFwiUEFUQ0hcIiB8IFwiREVMRVRFXCJgKS5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbcGF0aF0gLSBPcHRpb25hbCByb3V0ZSBwYXRoIHBhc3NlZCB0aHJvdWdoIHRvIHRoZSBjb3JyZXNwb25kaW5nIE5lc3QgSFRUUCBtZXRob2QgZGVjb3JhdG9yLlxuICogQHJldHVybiB7TWV0aG9kRGVjb3JhdG9yfSBBIG1ldGhvZCBkZWNvcmF0b3IgdGhhdCBlaXRoZXIgZXhjbHVkZXMgdGhlIGVuZHBvaW50IGZyb20gU3dhZ2dlciAoYW5kIHJvdXRlIHJlZ2lzdHJhdGlvbikgb3IgYXBwbGllcyB0aGUgY29ycmVjdCBIVFRQIGRlY29yYXRvci5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChcbiAgTW9kZWxDb25zdHJ1Y3RvcjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICB2ZXJiOiBIdHRwVmVyYnMsXG4gIHBhdGg/OiBzdHJpbmdcbik6IE1ldGhvZERlY29yYXRvciB7XG4gIGNvbnN0IGh0dHBUb0NydWQ6IFJlY29yZDxcbiAgICBIdHRwVmVyYnMsXG4gICAgW0NydWRPcGVyYXRpb25zLCAocGF0aD86IHN0cmluZykgPT4gTWV0aG9kRGVjb3JhdG9yXVxuICA+ID0ge1xuICAgIEdFVDogW09wZXJhdGlvbktleXMuUkVBRCwgR2V0XSxcbiAgICBQT1NUOiBbT3BlcmF0aW9uS2V5cy5DUkVBVEUsIFBvc3RdLFxuICAgIFBVVDogW09wZXJhdGlvbktleXMuVVBEQVRFLCBQdXRdLFxuICAgIFBBVENIOiBbT3BlcmF0aW9uS2V5cy5VUERBVEUsIFBhdGNoXSxcbiAgICBERUxFVEU6IFtPcGVyYXRpb25LZXlzLkRFTEVURSwgRGVsZXRlXSxcbiAgfTtcblxuICBjb25zdCBbY3J1ZE9wLCBIdHRwTWV0aG9kRGVjb3JhdG9yXSA9IGh0dHBUb0NydWRbdmVyYl07XG4gIGNvbnN0IHRhcmdldCA9IHJlc29sdmVCbG9ja1RhcmdldCh2ZXJiLCBwYXRoKTtcbiAgcmV0dXJuIHRhcmdldFxuICAgID8gY29yZUlzT3BlcmF0aW9uQmxvY2tlZChNb2RlbENvbnN0cnVjdG9yLCB0YXJnZXQua2luZCBhcyBhbnksIHRhcmdldC52YWx1ZSlcbiAgICAgID8gYXBwbHkoQXBpRXhjbHVkZUVuZHBvaW50KCkpXG4gICAgICA6IGFwcGx5KEh0dHBNZXRob2REZWNvcmF0b3IocGF0aCkpXG4gICAgOiBjb3JlSXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIGNydWRPcClcbiAgICA/IGFwcGx5KEFwaUV4Y2x1ZGVFbmRwb2ludCgpKVxuICAgIDogYXBwbHkoSHR0cE1ldGhvZERlY29yYXRvcihwYXRoKSk7XG59XG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDb25kaXRpb25hbGx5IGFwcGxpZXMgYW4gSFRUUCBtZXRob2QgZGVjb3JhdG9yIGZvciBhIGdpdmVuIG1vZGVsIGFuZCB2ZXJiLCBoaWRpbmcgdGhlIGVuZHBvaW50IGluIFN3YWdnZXIgKGFuZCBub3QgcmVnaXN0ZXJpbmcgdGhlIHJvdXRlKSB3aGVuIHRoZSBtb2RlbCBibG9ja3MgdGhhdCBDUlVEIG9wZXJhdGlvbi5cbiAqIEBzdW1tYXJ5IE1hcHMgYW4gSFRUUCB2ZXJiIHRvIGl0cyBjb3JyZXNwb25kaW5nIGBDcnVkT3BlcmF0aW9uc2Aga2V5IGFuZCBOZXN0IEhUVFAgZGVjb3JhdG9yIChgQEdldGAsIGBAUG9zdGAsIGV0Yy4pLiBJdCBjaGVja3MgYGlzT3BlcmF0aW9uQmxvY2tlZChNb2RlbENvbnN0cnVjdG9yLCBjcnVkT3ApYCBhbmQsIGlmIGJsb2NrZWQsIGFwcGxpZXMgb25seSBgQEFwaUV4Y2x1ZGVFbmRwb2ludCgpYCAoU3dhZ2dlci1oaWRkZW4sIG5vIE5lc3Qgcm91dGUpLiBJZiBwZXJtaXR0ZWQsIGl0IGFwcGxpZXMgdGhlIGFwcHJvcHJpYXRlIEhUVFAgZGVjb3JhdG9yIHdpdGggdGhlIG9wdGlvbmFsIGBwYXRoYC5cbiAqIEBwYXJhbSB7TW9kZWxDb25zdHJ1Y3Rvcjxhbnk+fSBNb2RlbENvbnN0cnVjdG9yIC0gVGhlIG1vZGVsIGNvbnN0cnVjdG9yIHVzZWQgdG8gcmVzb2x2ZSBvcGVyYXRpb24tYmxvY2tpbmcgcnVsZXMuXG4gKiBAcGFyYW0ge0h0dHBWZXJic30gdmVyYiAtIFRoZSBIVFRQIHZlcmIgdG8gbWFwIChlLmcuLCBgXCJHRVRcIiB8IFwiUE9TVFwiIHwgXCJQVVRcIiB8IFwiUEFUQ0hcIiB8IFwiREVMRVRFXCJgKS5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbcGF0aF0gLSBPcHRpb25hbCByb3V0ZSBwYXRoIHBhc3NlZCB0aHJvdWdoIHRvIHRoZSBjb3JyZXNwb25kaW5nIE5lc3QgSFRUUCBtZXRob2QgZGVjb3JhdG9yLlxuICogQHJldHVybiB7TWV0aG9kRGVjb3JhdG9yfSBBIG1ldGhvZCBkZWNvcmF0b3IgdGhhdCBlaXRoZXIgZXhjbHVkZXMgdGhlIGVuZHBvaW50IGZyb20gU3dhZ2dlciAoYW5kIHJvdXRlIHJlZ2lzdHJhdGlvbikgb3IgYXBwbGllcyB0aGUgY29ycmVjdCBIVFRQIGRlY29yYXRvci5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEJ1bGtBcGlPcGVyYXRpb25Gcm9tTW9kZWwoXG4gIE1vZGVsQ29uc3RydWN0b3I6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgdmVyYjogSHR0cFZlcmJzLFxuICBwYXRoPzogc3RyaW5nXG4pOiBNZXRob2REZWNvcmF0b3Ige1xuICBjb25zdCBodHRwVG9DcnVkOiBSZWNvcmQ8XG4gICAgSHR0cFZlcmJzLFxuICAgIFtCdWxrQ3J1ZE9wZXJhdGlvbktleXMsIChwYXRoPzogc3RyaW5nKSA9PiBNZXRob2REZWNvcmF0b3JdXG4gID4gPSB7XG4gICAgR0VUOiBbQnVsa0NydWRPcGVyYXRpb25LZXlzLlJFQURfQUxMLCBHZXRdLFxuICAgIFBPU1Q6IFtCdWxrQ3J1ZE9wZXJhdGlvbktleXMuQ1JFQVRFX0FMTCwgUG9zdF0sXG4gICAgUFVUOiBbQnVsa0NydWRPcGVyYXRpb25LZXlzLlVQREFURV9BTEwsIFB1dF0sXG4gICAgUEFUQ0g6IFtCdWxrQ3J1ZE9wZXJhdGlvbktleXMuVVBEQVRFX0FMTCwgUGF0Y2hdLFxuICAgIERFTEVURTogW0J1bGtDcnVkT3BlcmF0aW9uS2V5cy5ERUxFVEVfQUxMLCBEZWxldGVdLFxuICB9O1xuXG4gIGNvbnN0IFtjcnVkT3AsIEh0dHBNZXRob2REZWNvcmF0b3JdID0gaHR0cFRvQ3J1ZFt2ZXJiXTtcbiAgY29uc3QgdGFyZ2V0ID0gcGF0aCA/IHJlc29sdmVCbG9ja1RhcmdldCh2ZXJiLCBwYXRoKSA6IHVuZGVmaW5lZDtcbiAgcmV0dXJuIHRhcmdldFxuICAgID8gY29yZUlzT3BlcmF0aW9uQmxvY2tlZChNb2RlbENvbnN0cnVjdG9yLCB0YXJnZXQua2luZCBhcyBhbnksIHRhcmdldC52YWx1ZSlcbiAgICAgID8gYXBwbHkoQXBpRXhjbHVkZUVuZHBvaW50KCkpXG4gICAgICA6IGFwcGx5KEh0dHBNZXRob2REZWNvcmF0b3IocGF0aCkpXG4gICAgOiBjb3JlSXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIFwiYnVsa1wiIGFzIGFueSwgY3J1ZE9wIGFzIGFueSlcbiAgICA/IGFwcGx5KEFwaUV4Y2x1ZGVFbmRwb2ludCgpKVxuICAgIDogYXBwbHkoSHR0cE1ldGhvZERlY29yYXRvcihwYXRoKSk7XG59XG5mdW5jdGlvbiByZXNvbHZlQmxvY2tUYXJnZXQoXG4gIHZlcmI6IEh0dHBWZXJicyxcbiAgcGF0aD86IHN0cmluZ1xuKTogeyBraW5kOiBcImNydWRcIiB8IFwic3RhdGVtZW50XCIgfCBcInF1ZXJ5XCIgfCBcImJ1bGtcIjsgdmFsdWU6IHN0cmluZyB9IHwgdW5kZWZpbmVkIHtcbiAgaWYgKCFwYXRoKSByZXR1cm4gdW5kZWZpbmVkO1xuXG4gIGNvbnN0IG5vcm1hbGl6ZWQgPSBwYXRoLnJlcGxhY2UoL15cXC8rfFxcLyskL2csIFwiXCIpO1xuICBjb25zdCBzdGF0ZW1lbnRUYXJnZXRzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICAgIFwibGlzdEJ5LzprZXlcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkxJU1RfQlksXG4gICAgXCJwYWdpbmF0ZUJ5LzprZXkvOnBhZ2VcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0VfQlksXG4gICAgXCJmaW5kLzp2YWx1ZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORCxcbiAgICBcInBhZ2UvOnZhbHVlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5QQUdFLFxuICAgIFwiZmluZE9uZUJ5LzprZXkvOnZhbHVlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5EX09ORV9CWSxcbiAgICBcImZpbmRCeS86a2V5Lzp2YWx1ZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9CWSxcbiAgICBcInN0YXRlbWVudC86bWV0aG9kLyphcmdzXCI6IFwic3RhdGVtZW50XCIsXG4gICAgXCJjb3VudE9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuQ09VTlRfT0YsXG4gICAgXCJtYXhPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLk1BWF9PRixcbiAgICBcIm1pbk9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuTUlOX09GLFxuICAgIFwiYXZnT2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5BVkdfT0YsXG4gICAgXCJzdW1PZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlNVTV9PRixcbiAgICBcImRpc3RpbmN0T2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5ESVNUSU5DVF9PRixcbiAgICBcImdyb3VwT2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5HUk9VUF9PRixcbiAgfTtcblxuICBpZiAobm9ybWFsaXplZC5zdGFydHNXaXRoKFwicXVlcnkvXCIpKSB7XG4gICAgcmV0dXJuIHsga2luZDogXCJxdWVyeVwiLCB2YWx1ZTogbm9ybWFsaXplZC5yZXBsYWNlKC9ecXVlcnlcXC8vLCBcIlwiKSB9O1xuICB9XG5cbiAgY29uc3Qgc3RhdGVtZW50VmFsdWUgPSBzdGF0ZW1lbnRUYXJnZXRzW25vcm1hbGl6ZWRdO1xuICBpZiAoc3RhdGVtZW50VmFsdWUpIHtcbiAgICByZXR1cm4geyBraW5kOiBcInN0YXRlbWVudFwiLCB2YWx1ZTogc3RhdGVtZW50VmFsdWUgfTtcbiAgfVxuXG4gIGlmICh2ZXJiID09PSBcIkdFVFwiICYmIG5vcm1hbGl6ZWQgPT09IFwiXCIpIHtcbiAgICByZXR1cm4geyBraW5kOiBcImNydWRcIiwgdmFsdWU6IE9wZXJhdGlvbktleXMuUkVBRCB9O1xuICB9XG5cbiAgcmV0dXJuIHVuZGVmaW5lZDtcbn1cbiIsImltcG9ydCB7IGFwcGx5RGVjb3JhdG9ycyB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgQXBpUGFyYW0gYXMgQXBpUGFyYW1EZWMgfSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyXCI7XG5pbXBvcnQgdHlwZSB7IERlY2FmQXBpUHJvcGVydHkgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBBcHBsaWVzIGEgc2V0IG9mIFN3YWdnZXIgYEBBcGlQYXJhbWAgZGVjb3JhdG9ycyBnZW5lcmF0ZWQgZnJvbSBhIHR5cGVkIHNwZWNpZmljYXRpb24gYXJyYXkuXG4gKiBAc3VtbWFyeSBUcmFuc2Zvcm1zIGVhY2ggZW50cnkgb2YgdGhlIHByb3ZpZGVkIGBBcGlQYXJhbVtdYCBpbnRvIGEgY29ycmVzcG9uZGluZyBgQEFwaVBhcmFtKClgIGRlY29yYXRvciAoZGVmYXVsdGluZyBgZGVzY3JpcHRpb25gLCBgcmVxdWlyZWRgLCBhbmQgYHR5cGVgIHdoZW4gb21pdHRlZCkgYW5kIGNvbXBvc2VzIHRoZW0gaW50byBhIHNpbmdsZSBkZWNvcmF0b3IgdmlhIE5lc3QncyBgYXBwbHlEZWNvcmF0b3JzYC4gVXNlZnVsIGZvciBzeW5jaHJvbml6aW5nIHJ1bnRpbWUgcm91dGUgcGFyYW1zIHdpdGggT3BlbkFQSSBkb2N1bWVudGF0aW9uIGluIGEgY29uY2lzZSwgZGVjbGFyYXRpdmUgd2F5LlxuICogQHBhcmFtIHtBcGlQYXJhbVtdfSBbcHJvcHM9W11dIEFycmF5IGRlc2NyaWJpbmcgcGF0aCBwYXJhbWV0ZXJzIHRvIGJlIGRvY3VtZW50ZWQuXG4gKiBAcGFyYW0ge3N0cmluZ30gcHJvcHNbXS5uYW1lIFRoZSBwYXJhbWV0ZXIncyBuYW1lIGFzIGl0IGFwcGVhcnMgaW4gdGhlIHJvdXRlIHRlbXBsYXRlIChlLmcuLCBgOmlkYCkuXG4gKiBAcGFyYW0ge3N0cmluZ30gW3Byb3BzW10uZGVzY3JpcHRpb25dIEh1bWFuLXJlYWRhYmxlIGV4cGxhbmF0aW9uIG9mIHRoZSBwYXJhbWV0ZXI7IGRlZmF1bHRzIHRvIGBcIlBhdGggcGFyYW1ldGVyOiA8bmFtZT5cImAuXG4gKiBAcGFyYW0ge2Jvb2xlYW59IFtwcm9wc1tdLnJlcXVpcmVkPXRydWVdIFdoZXRoZXIgdGhlIHBhcmFtZXRlciBpcyByZXF1aXJlZDsgZGVmYXVsdHMgdG8gYHRydWVgLlxuICogQHBhcmFtIHtDb25zdHJ1Y3Rvcjxhbnk+fSBbcHJvcHNbXS50eXBlPVN0cmluZ10gQ29uc3RydWN0b3IvdHlwZSB1c2VkIGJ5IFN3YWdnZXIgdG8gaW5mZXIgc2NoZW1hIChlLmcuLCBgU3RyaW5nYCwgYE51bWJlcmAsIGBVVUlEYCwgY3VzdG9tIGNsYXNzKS5cbiAqIEByZXR1cm4ge2FueX0gQSBjb21wb3NlZCBkZWNvcmF0b3IgYXBwbHlpbmcgYWxsIGdlbmVyYXRlZCBgQEFwaVBhcmFtYCBkZWNvcmF0b3JzIHRvIHRoZSB0YXJnZXQgbWV0aG9kIG9yIGNvbnRyb2xsZXIuXG4gKiBAZnVuY3Rpb24gQXBpUGFyYW1zRnJvbU1vZGVsXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBBcGlQYXJhbXNGcm9tTW9kZWwoXG4gIHByb3BzOiBEZWNhZkFwaVByb3BlcnR5W10gPSBbXVxuKTogTWV0aG9kRGVjb3JhdG9yICYgQ2xhc3NEZWNvcmF0b3Ige1xuICBjb25zdCBkZWNvcmF0b3JzID0gcHJvcHMubWFwKChwKSA9PlxuICAgIEFwaVBhcmFtRGVjKHtcbiAgICAgIG5hbWU6IHAubmFtZSxcbiAgICAgIGRlc2NyaXB0aW9uOiBwLmRlc2NyaXB0aW9uID8/IGBQYXRoIHBhcmFtZXRlcjogJHtwLm5hbWV9YCxcbiAgICAgIHJlcXVpcmVkOiBwLnJlcXVpcmVkID8/IHRydWUsXG4gICAgICAvLyBzY2hlbWE6IHsgdHlwZTogJ3N0cmluZycgfSxcbiAgICAgIHR5cGU6IHAudHlwZSA/PyBTdHJpbmcsXG4gICAgfSlcbiAgKTtcbiAgcmV0dXJuIGFwcGx5RGVjb3JhdG9ycyguLi5kZWNvcmF0b3JzKTtcbn1cbiIsImltcG9ydCB7XG4gIGNyZWF0ZVBhcmFtRGVjb3JhdG9yLFxuICBFeGVjdXRpb25Db250ZXh0LFxuICBJbnRlcm5hbFNlcnZlckVycm9yRXhjZXB0aW9uLFxufSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQ3VzdG9tIGRlY29yYXRvciB0aGF0IGV4dHJhY3RzIHRoZSByZXF1ZXN0IGJvZHkgYW5kIGluc3RhbnRpYXRlcyBpdCB1c2luZyB0aGUgTW9kZWwgY29uc3RydWN0b3JcbiAqIGZvdW5kIG9uIHRoZSBjb250cm9sbGVyLiBIYW5kbGVzIGJvdGggc2luZ2xlIG9iamVjdHMgYW5kIGFycmF5cy5cbiAqL1xuZXhwb3J0IGNvbnN0IERlY2FmQm9keSA9IGNyZWF0ZVBhcmFtRGVjb3JhdG9yKFxuICAoZGF0YTogdW5rbm93biwgY3R4OiBFeGVjdXRpb25Db250ZXh0KSA9PiB7XG4gICAgY29uc3QgcmVxdWVzdCA9IGN0eC5zd2l0Y2hUb0h0dHAoKS5nZXRSZXF1ZXN0KCk7XG4gICAgY29uc3QgYm9keSA9IHJlcXVlc3QuYm9keTtcblxuICAgIC8vIFRoZSBEeW5hbWljTW9kZWxDb250cm9sbGVyIHdpbGwgaGF2ZSAnTW9kZWxDb25zdHInIGF2YWlsYWJsZVxuICAgIGNvbnN0IGNvbnRyb2xsZXIgPSBjdHguZ2V0Q2xhc3MoKTtcbiAgICAvLyBXZSBhY2Nlc3MgdGhlIHN0YXRpYyBwcm9wZXJ0eSB3ZSB3aWxsIGFkZCB0byB0aGUgRHluYW1pY01vZGVsQ29udHJvbGxlclxuICAgIGNvbnN0IE1vZGVsQ29uc3RyID0gKGNvbnRyb2xsZXIgYXMgYW55KS5jbGFzcztcblxuICAgIGlmICghTW9kZWxDb25zdHIpIHtcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbFNlcnZlckVycm9yRXhjZXB0aW9uKFxuICAgICAgICBgTW9kZWxDb25zdHJ1Y3RvciBub3QgZm91bmQgb24gY29udHJvbGxlciAke2NvbnRyb2xsZXIubmFtZX0uIEVuc3VyZSB0aGUgY29udHJvbGxlciB3YXMgY3JlYXRlZCB2aWEgRnJvbU1vZGVsQ29udHJvbGxlci5gXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmICghYm9keSkgcmV0dXJuIGJvZHk7XG5cbiAgICBpZiAoQXJyYXkuaXNBcnJheShib2R5KSkge1xuICAgICAgcmV0dXJuIGJvZHkubWFwKChpdGVtKSA9PiBuZXcgTW9kZWxDb25zdHIoaXRlbSkpO1xuICAgIH1cblxuICAgIHJldHVybiBuZXcgTW9kZWxDb25zdHIoYm9keSk7XG4gIH1cbik7XG4iLCJpbXBvcnQgeyBjcmVhdGVQYXJhbURlY29yYXRvciwgRXhlY3V0aW9uQ29udGV4dCB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRGVjYWZQYXJhbVByb3BzLCBEZWNhZkFwaVByb3BlcnR5IH0gZnJvbSBcIi4vdHlwZXNcIjtcbi8qKlxuICogQGRlc2NyaXB0aW9uIENyZWF0ZXMgYSBjdXN0b20gTmVzdEpTIHBhcmFtZXRlciBkZWNvcmF0b3IgdGhhdCBleHRyYWN0cyBhbmQgcmV0dXJucyByb3V0ZSBwYXJhbWV0ZXJzIG9yZGVyZWQgYnkgYSBzcGVjaWZpYyBrZXkgc2VxdWVuY2UuXG4gKiBAc3VtbWFyeSBUaGUgYE9yZGVyZWRQYXJhbXNgIGRlY29yYXRvciByZWFkcyB0aGUgaW5jb21pbmcgSFRUUCByZXF1ZXN0J3MgYHBhcmFtc2Agb2JqZWN0LCBvcHRpb25hbGx5IG9yZGVycyBpdCBhY2NvcmRpbmcgdG8gYSBwcm92aWRlZCBsaXN0IG9mIHBhcmFtZXRlciBuYW1lcywgYW5kIHJldHVybnMgYW4gb2JqZWN0IGNvbnRhaW5pbmcgdGhlIG9yaWdpbmFsIHBhcmFtcywgdGhlIG9yZGVyZWQgdmFsdWVzLCBhbmQgdGhlIGFwcGxpZWQgb3JkZXIgbGlzdC4gVGhpcyBlbnN1cmVzIGRldGVybWluaXN0aWMgcGFyYW1ldGVyIGFjY2VzcyBmb3Igcm91dGVzIHdpdGggbXVsdGlwbGUgcGF0aCBwYXJhbWV0ZXJzLlxuICogQHRlbXBsYXRlIHtzdHJpbmdbXX0gVE9yZGVyIC0gUmVwcmVzZW50cyB0aGUgZXhwZWN0ZWQgc2VxdWVuY2Ugb2YgcGFyYW1ldGVyIG5hbWVzLlxuICogQHBhcmFtIHtUT3JkZXIgfCB1bmRlZmluZWR9IG9yZGVyIEFuIG9wdGlvbmFsIGFycmF5IHNwZWNpZnlpbmcgdGhlIHBhcmFtZXRlciBuYW1lcycgZGVzaXJlZCBvcmRlci4gSWYgbm90IHByb3ZpZGVkLCB0aGUgZGVjb3JhdG9yIHdpbGwgcHJlc2VydmUgdGhlIG9yZGVyIG9mIHRoZSBrZXlzIGFzIHRoZXkgYXBwZWFyIGluIHRoZSBvcmlnaW5hbCBgcmVxLnBhcmFtc2Agb2JqZWN0LlxuICogQHBhcmFtIHtFeGVjdXRpb25Db250ZXh0fSBjdHggVGhlIE5lc3RKUyBleGVjdXRpb24gY29udGV4dCwgdXNlZCB0byByZXRyaWV2ZSB0aGUgY3VycmVudCBIVFRQIHJlcXVlc3Qgb2JqZWN0LlxuICogQHJldHVybiB7RGVjYWZQYXJhbVByb3BzfSBBbiBvYmplY3QgY29udGFpbmluZzpcbiAqIC0gYG9yaWdpbmFsYDogdGhlIHJhdyBwYXJhbWV0ZXIgbWFwIGZyb20gYHJlcS5wYXJhbXNgO1xuICogLSBgb3JkZXJlZGA6IGFuIGFycmF5IG9mIHBhcmFtZXRlciB2YWx1ZXMgaW4gdGhlIHJlcXVlc3RlZCBvcmRlcjtcbiAqIC0gYG9yZGVyYDogdGhlIGVmZmVjdGl2ZSBvcmRlciBsaXN0IHVzZWQuXG4gKiBAZnVuY3Rpb24gT3JkZXJlZFBhcmFtc1xuICovXG5jb25zdCBPcmRlcmVkUGFyYW1zID0gY3JlYXRlUGFyYW1EZWNvcmF0b3IoXG4gIChvcmRlcjogc3RyaW5nW10gfCB1bmRlZmluZWQsIGN0eDogRXhlY3V0aW9uQ29udGV4dCk6IERlY2FmUGFyYW1Qcm9wcyA9PiB7XG4gICAgY29uc3QgcmVxID0gY3R4LnN3aXRjaFRvSHR0cCgpLmdldFJlcXVlc3QoKTtcbiAgICBjb25zdCBvcmlnaW5hbCA9IChyZXE/LnBhcmFtcyA/PyB7fSkgYXMgUmVjb3JkPHN0cmluZywgYW55PjtcbiAgICBjb25zdCBvcmRlckxpc3QgPSBvcmRlciA/PyBPYmplY3Qua2V5cyhvcmlnaW5hbCk7XG4gICAgY29uc3Qgb3JkZXJlZCA9IG9yZGVyTGlzdC5tYXAoKGspID0+IG9yaWdpbmFsW2tdKTtcbiAgICByZXR1cm4geyByYXc6IG9yaWdpbmFsLCB2YWx1ZXNJbk9yZGVyOiBvcmRlcmVkLCBrZXlzSW5PcmRlcjogb3JkZXJMaXN0IH07XG4gIH1cbik7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEEgaGlnaGVyLWxldmVsIGRlY29yYXRvciBmYWN0b3J5IHRoYXQgbGV2ZXJhZ2VzIGBPcmRlcmVkUGFyYW1zYCB0byBleHRyYWN0IHJvdXRlIHBhcmFtZXRlcnMgaW4gYSBzcGVjaWZpYyBvcmRlciBkZXJpdmVkIGZyb20gYSBsaXN0IG9mIGBBcGlQYXJhbWAgZGVmaW5pdGlvbnMuXG4gKiBAc3VtbWFyeSBgRGVjYWZQYXJhbXNgIGNvbXB1dGVzIHRoZSBvcmRlciBvZiByb3V0ZSBwYXJhbWV0ZXJzIGJhc2VkIG9uIHRoZSBwcm92aWRlZCBgQXBpUGFyYW1bXWAgc3BlY2lmaWNhdGlvbiAodXNpbmcgZWFjaCBlbGVtZW504oCZcyBgbmFtZWApLCB0aGVuIGFwcGxpZXMgYE9yZGVyZWRQYXJhbXMob3JkZXIpYCBhcyBhIHBhcmFtZXRlciBkZWNvcmF0b3IuIFRoaXMgZW5hYmxlcyBwYXJhbWV0ZXItbGV2ZWwgYmluZGluZyB0aGF0IHJlbWFpbnMgY29uc2lzdGVudCB3aXRoIHRoZSBkb2N1bWVudGVkIEFQSSBwYXJhbWV0ZXIgbWV0YWRhdGEuXG4gKiBAcGFyYW0ge0FwaVBhcmFtW119IFtwcm9wcz1bXV0gQXJyYXkgb2YgYEFwaVBhcmFtYCBkZWZpbml0aW9ucyB3aG9zZSBgbmFtZWAgZmllbGRzIGRldGVybWluZSB0aGUgcGFyYW1ldGVyIGV4dHJhY3Rpb24gb3JkZXIuXG4gKiBAcmV0dXJuIHtQYXJhbWV0ZXJEZWNvcmF0b3J9IEEgTmVzdEpTIHBhcmFtZXRlciBkZWNvcmF0b3IgdGhhdCBpbmplY3RzIGFuIG9yZGVyZWQgbGlzdCBvZiBwYXJhbWV0ZXJzIGFuZCBtZXRhZGF0YSBpbnRvIHRoZSBjb250cm9sbGVyIG1ldGhvZCBhcmd1bWVudC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIERlY2FmUGFyYW1zKFxuICBwcm9wczogRGVjYWZBcGlQcm9wZXJ0eVtdID0gW11cbik6IFBhcmFtZXRlckRlY29yYXRvciB7XG4gIGNvbnN0IG9yZGVyID0gcHJvcHMubWFwKChwKSA9PiBwLm5hbWUpO1xuICByZXR1cm4gT3JkZXJlZFBhcmFtcyhvcmRlcik7XG59XG5cbmV4cG9ydCBjb25zdCBEZWNhZlF1ZXJ5ID0gY3JlYXRlUGFyYW1EZWNvcmF0b3IoXG4gIChfOiB1bmtub3duLCBjdHg6IEV4ZWN1dGlvbkNvbnRleHQpID0+IHtcbiAgICBjb25zdCByZXEgPSBjdHguc3dpdGNoVG9IdHRwKCkuZ2V0UmVxdWVzdCgpO1xuICAgIGNvbnN0IHF1ZXJ5ID0gcmVxLnF1ZXJ5ID8/IHt9O1xuXG4gICAgY29uc3QgcGFyc2VkOiBhbnkgPSB7IC4uLnF1ZXJ5IH07XG5cbiAgICAvLyBQYXJzZSBsaW1pdCAmIG9mZnNldFxuICAgIGlmIChwYXJzZWQubGltaXQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgY29uc3QgbiA9IE51bWJlcihwYXJzZWQubGltaXQpO1xuICAgICAgaWYgKCFOdW1iZXIuaXNOYU4obikpIHBhcnNlZC5saW1pdCA9IG47XG4gICAgfVxuXG4gICAgaWYgKHBhcnNlZC5vZmZzZXQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgY29uc3QgbiA9IE51bWJlcihwYXJzZWQub2Zmc2V0KTtcbiAgICAgIGlmICghTnVtYmVyLmlzTmFOKG4pKSBwYXJzZWQub2Zmc2V0ID0gbjtcbiAgICB9XG5cbiAgICAvLyBQYXJzZSBib29rbWFyayBvbmx5IGlmIG51bWVyaWNcbiAgICBpZiAocGFyc2VkLmJvb2ttYXJrICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIGNvbnN0IG4gPSBOdW1iZXIocGFyc2VkLmJvb2ttYXJrKTtcbiAgICAgIHBhcnNlZC5ib29rbWFyayA9IE51bWJlci5pc05hTihuKSA/IHBhcnNlZC5ib29rbWFyayA6IG47XG4gICAgfVxuXG4gICAgcmV0dXJuIHBhcnNlZDtcbiAgfVxuKTtcbiIsImltcG9ydCB7IEFkYXB0ZXJGbGFncywgQ29udGV4dCB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5cbmV4cG9ydCBjb25zdCBERUNBRl9NT0RVTEVfT1BUSU9OUyA9IFwiRGVjYWZNb2R1bGVPcHRpb25zXCI7XG5leHBvcnQgY29uc3QgREVDQUZfQURBUFRFUl9JRCA9IFwiRGVjYWZBZGFwdGVyXCI7XG5leHBvcnQgY29uc3QgREVDQUZfVEFTS19TRVJWSUNFX0lEID0gXCJEZWNhZlRhc2tTZXJ2aWNlXCI7XG5cbmV4cG9ydCBjb25zdCBERUNBRl9ST1VURSA9IFwiRGVjYWZSb3V0ZVwiO1xuZXhwb3J0IGNvbnN0IERFQ0FGX0hBTkRMRVJTID0gU3ltYm9sKFwiRGVjYWZIYW5kbGVyc1wiKTtcbmV4cG9ydCBjb25zdCBERUNBRl9FWFBPU0UgPSBcIkRlY2FmRXhwb3NlXCI7XG5leHBvcnQgY29uc3QgREVDQUZfQ09OVFJPTExFUl9DT05GSUcgPSBcIkRlY2FmQ29udHJvbGxlckNvbmZpZ1wiO1xuXG5leHBvcnQgY29uc3QgREVDQUZfQ09OVEVYVF9LRVkgPSBTeW1ib2woXCJkZWNhZjpjb250ZXh0XCIpO1xuXG5leHBvcnQgdHlwZSBEZWNhZlNlcnZlckZsYWdzPExPRyBleHRlbmRzIExvZ2dlciA9IExvZ2dlcj4gPVxuICBBZGFwdGVyRmxhZ3M8TE9HPiAmIHtcbiAgICBoZWFkZXJzOiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICAgIG92ZXJyaWRlczogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgfTtcblxuZXhwb3J0IHR5cGUgRGVjYWZTZXJ2ZXJDdHggPSBDb250ZXh0PERlY2FmU2VydmVyRmxhZ3M+O1xuIiwiaW1wb3J0IHsgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcblxuaW1wb3J0IHsgREVDQUZfRVhQT1NFIH0gZnJvbSBcIi4uLy4uL2NvbnN0YW50c1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gZXhwb3NlKC4uLmZsYXZvdXJzOiBzdHJpbmdbXSkge1xuICByZXR1cm4gZnVuY3Rpb24gZXhwb3NlKHRhcmdldDogYW55KSB7XG4gICAgTWV0YWRhdGEuc2V0KFxuICAgICAgdGFyZ2V0LFxuICAgICAgREVDQUZfRVhQT1NFLFxuICAgICAgKGZsYXZvdXJzLmxlbmd0aCA/IGZsYXZvdXJzIDogdHJ1ZSkgYXMgYW55XG4gICAgKTtcbiAgICByZXR1cm4gdGFyZ2V0O1xuICB9O1xufVxuIiwiaW1wb3J0IHsgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB0eXBlIHsgTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZyB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5cbmltcG9ydCB7IERFQ0FGX0NPTlRST0xMRVJfQ09ORklHIH0gZnJvbSBcIi4uLy4uL2NvbnN0YW50c1wiO1xuXG4vKipcbiAqIENsYXNzIGRlY29yYXRvciB0aGF0IGF0dGFjaGVzIGEge0BsaW5rIE1vZGVsQ29udHJvbGxlckZhY3RvcnlDb25maWd9IHRvIGEgTW9kZWwsXG4gKiBzbyBgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGUoKWAgY2FuIHBhc3MgaXQgdG8gYE1vZGVsQ29udHJvbGxlckZhY3RvcnkuY3JlYXRlKClgLlxuICpcbiAqIFRoZSBwZXItbW9kZWwgY29uZmlnIGNhbiBiZSBvdmVycmlkZGVuIGJ5IHRoZSBtb2R1bGUtbGV2ZWwgYGNvbnRyb2xsZXJDb25maWdgXG4gKiBvcHRpb24gaW4gYERlY2FmTW9kdWxlT3B0aW9uc2AuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHRzXG4gKiBAY29udHJvbGxlckNvbmZpZyh7IGFsbG93R3JvdXBpbmdRdWVyaWVzOiB0cnVlLCBhbGxvd0J1bGtTdGF0ZW1lbnQ6IHsgY3JlYXRlOiB0cnVlLCByZWFkOiB0cnVlLCB1cGRhdGU6IGZhbHNlLCBkZWxldGU6IHRydWUgfSB9KVxuICogQG1vZGVsKClcbiAqIGNsYXNzIE9yZGVyIGV4dGVuZHMgTW9kZWw8T3JkZXI+IHsgLi4uIH1cbiAqIGBgYFxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBGYWN0b3J5IGNvbmZpZ3VyYXRpb24ga25vYnMgKGFsbG93U3RhdGVtZW50bGVzc1F1ZXJ5LCBhbGxvd0dyb3VwaW5nUXVlcmllcywgYWxsb3dCdWxrU3RhdGVtZW50KS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNvbnRyb2xsZXJDb25maWcoY29uZmlnOiBNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnKSB7XG4gIHJldHVybiBmdW5jdGlvbiBjb250cm9sbGVyQ29uZmlnRGVjb3JhdG9yKHRhcmdldDogYW55KSB7XG4gICAgTWV0YWRhdGEuc2V0KHRhcmdldCwgREVDQUZfQ09OVFJPTExFUl9DT05GSUcsIGNvbmZpZyBhcyBhbnkpO1xuICAgIHJldHVybiB0YXJnZXQ7XG4gIH07XG59XG4iLCJleHBvcnQgY29uc3QgQVVUSF9IQU5ETEVSID0gU3ltYm9sKFwiQVVUSF9IQU5ETEVSXCIpO1xuZXhwb3J0IGNvbnN0IEFVVEhfTUVUQV9LRVkgPSBcImF1dGg6bWV0YVwiO1xuZXhwb3J0IGNvbnN0IElTX1BVQkxJQ19LRVkgPSBcImlzUHVibGljXCI7XG5leHBvcnQgY29uc3QgUkVRVUlSRURfUk9MRVNfS0VZID0gXCJyZXF1aXJlZFJvbGVzXCI7XG5leHBvcnQgY29uc3QgU0tJUF9NT0RFTF9ST0xFU19LRVkgPSBcInNraXBNb2RlbFJvbGVzXCI7XG4iLCJpbXBvcnQgeyBJbmplY3QsIEluamVjdGFibGUsIFNjb3BlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBSRVFVRVNUIH0gZnJvbSBcIkBuZXN0anMvY29yZVwiO1xuaW1wb3J0IHsgVVVJRCB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuaW1wb3J0IHsgdHlwZSBSZXF1ZXN0IH0gZnJvbSBcImV4cHJlc3NcIjtcblxuaW1wb3J0IHsgRGVjYWZTZXJ2ZXJDdHggfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5cbkBJbmplY3RhYmxlKHsgc2NvcGU6IFNjb3BlLlJFUVVFU1QgfSlcbmV4cG9ydCBjbGFzcyBEZWNhZlJlcXVlc3RDb250ZXh0PEMgZXh0ZW5kcyBEZWNhZlNlcnZlckN0eCA9IERlY2FmU2VydmVyQ3R4PiBleHRlbmRzIFJlcXVlc3RDb250ZXh0PFJlcXVlc3Q+IHtcbiAgdXVpZCA9IFVVSUQuaW5zdGFuY2UuZ2VuZXJhdGUoKTtcbiAgb3ZlcnJpZGUgcmVhZG9ubHkgcmVxdWVzdDogUmVxdWVzdDtcblxuICBjb25zdHJ1Y3RvcihASW5qZWN0KFJFUVVFU1QpIHByaXZhdGUgcmVhZG9ubHkgcmVxOiBSZXF1ZXN0KSB7XG4gICAgc3VwZXIoXG4gICAgICB7XG4gICAgICAgIGhlYWRlcnNPZjogKHJlcXVlc3Q6IFJlcXVlc3QpID0+IChyZXF1ZXN0IGFzIGFueSk/LmhlYWRlcnMgfHwgdW5kZWZpbmVkLFxuICAgICAgfSBhcyBhbnksXG4gICAgICByZXFcbiAgICApO1xuICAgIHRoaXMucmVxdWVzdCA9IHJlcTtcbiAgfVxuXG4gIHB1dChyZWNvcmQ6IFJlY29yZDxhbnksIGFueT4pIHtcbiAgICBsZXQgb3ZlcnJpZGVzOiBhbnk7XG4gICAgdHJ5IHtcbiAgICAgIG92ZXJyaWRlcyA9IHRoaXMuZ2V0KFwib3ZlcnJpZGVzXCIpO1xuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgIG92ZXJyaWRlcyA9IHt9O1xuICAgIH1cblxuICAgIHRoaXMuYWNjdW11bGF0ZSh7XG4gICAgICBvdmVycmlkZXM6IE9iamVjdC5hc3NpZ24ob3ZlcnJpZGVzLCByZWNvcmQpLFxuICAgIH0pO1xuICB9XG59XG4iLCJpbXBvcnQge1xuICBJbmplY3RhYmxlLFxuICBOZXN0SW50ZXJjZXB0b3IsXG4gIEV4ZWN1dGlvbkNvbnRleHQsXG4gIENhbGxIYW5kbGVyLFxuICBJbmplY3QsXG4gIE9wdGlvbmFsLFxuICBTY29wZSxcbn0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBSZWZsZWN0b3IgfSBmcm9tIFwiQG5lc3Rqcy9jb3JlXCI7XG5pbXBvcnQgeyBPYnNlcnZhYmxlIH0gZnJvbSBcInJ4anNcIjtcblxuaW1wb3J0IHsgQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IExvZ2dpbmcgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7IEFkYXB0ZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lciB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5cbmltcG9ydCB7IEFVVEhfSEFORExFUiwgQVVUSF9NRVRBX0tFWSwgSVNfUFVCTElDX0tFWSwgUkVRVUlSRURfUk9MRVNfS0VZLCBTS0lQX01PREVMX1JPTEVTX0tFWSB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHR5cGUgeyBBdXRoSGFuZGxlciB9IGZyb20gXCIuLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZSZXF1ZXN0Q29udGV4dCB9IGZyb20gXCIuLi9yZXF1ZXN0L0RlY2FmUmVxdWVzdENvbnRleHRcIjtcblxuQEluamVjdGFibGUoeyBzY29wZTogU2NvcGUuUkVRVUVTVCB9KVxuZXhwb3J0IGNsYXNzIEF1dGhJbnRlcmNlcHRvciBpbXBsZW1lbnRzIE5lc3RJbnRlcmNlcHRvciB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgcmVmbGVjdG9yOiBSZWZsZWN0b3IsXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IHJlcXVlc3RDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIEBPcHRpb25hbCgpIEBJbmplY3QoQVVUSF9IQU5ETEVSKSBwcml2YXRlIHJlYWRvbmx5IGF1dGhIYW5kbGVyPzogQXV0aEhhbmRsZXJcbiAgKSB7fVxuXG4gIGFzeW5jIGludGVyY2VwdChcbiAgICBjdHg6IEV4ZWN1dGlvbkNvbnRleHQsXG4gICAgbmV4dDogQ2FsbEhhbmRsZXJcbiAgKTogUHJvbWlzZTxPYnNlcnZhYmxlPGFueT4+IHtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcih0aGlzIGFzIGFueSkuZm9yKHRoaXMuaW50ZXJjZXB0KTtcblxuICAgIGNvbnN0IGlzUHVibGljID0gdGhpcy5yZWZsZWN0b3IuZ2V0QWxsQW5kT3ZlcnJpZGU8Ym9vbGVhbj4oSVNfUFVCTElDX0tFWSwgW1xuICAgICAgY3R4LmdldEhhbmRsZXIoKSxcbiAgICAgIGN0eC5nZXRDbGFzcygpLFxuICAgIF0pO1xuXG4gICAgY29uc3QgbW9kZWxOYW1lID1cbiAgICAgIHRoaXMucmVmbGVjdG9yLmdldDxzdHJpbmcgfCBDb25zdHJ1Y3Rvcj4oXG4gICAgICAgIEFVVEhfTUVUQV9LRVksXG4gICAgICAgIGN0eC5nZXRIYW5kbGVyKClcbiAgICAgICkgPz9cbiAgICAgIHRoaXMucmVmbGVjdG9yLmdldDxzdHJpbmcgfCBDb25zdHJ1Y3Rvcj4oQVVUSF9NRVRBX0tFWSwgY3R4LmdldENsYXNzKCkpO1xuXG4gICAgY29uc3QgcmVxdWlyZWRSb2xlcyA9IHRoaXMucmVmbGVjdG9yLmdldEFsbEFuZE92ZXJyaWRlPHN0cmluZ1tdPihcbiAgICAgIFJFUVVJUkVEX1JPTEVTX0tFWSxcbiAgICAgIFtjdHguZ2V0SGFuZGxlcigpLCBjdHguZ2V0Q2xhc3MoKV1cbiAgICApO1xuXG4gICAgY29uc3Qgc2tpcE1vZGVsUm9sZXMgPSB0aGlzLnJlZmxlY3Rvci5nZXRBbGxBbmRPdmVycmlkZTxib29sZWFuPihcbiAgICAgIFNLSVBfTU9ERUxfUk9MRVNfS0VZLFxuICAgICAgW2N0eC5nZXRIYW5kbGVyKCksIGN0eC5nZXRDbGFzcygpXVxuICAgICk7XG5cbiAgICBjb25zdCBlZmZlY3RpdmVNb2RlbCA9IHNraXBNb2RlbFJvbGVzID8gdW5kZWZpbmVkIDogbW9kZWxOYW1lO1xuXG4gICAgbG9nLnZlcmJvc2UoYEludGVyY2VwdGVkIHJlcXVlc3Qke21vZGVsTmFtZSA/IGAgZm9yICR7bW9kZWxOYW1lfWAgOiBcIlwifWApO1xuXG4gICAgaWYgKGlzUHVibGljKSB7XG4gICAgICBsb2cuZGVidWcoYFB1YmxpYyByb3V0ZSDigJQgc2tpcHBpbmcgYXV0aGApO1xuICAgIH0gZWxzZSBpZiAodGhpcy5hdXRoSGFuZGxlcikge1xuICAgICAgYXdhaXQgdGhpcy5hdXRoSGFuZGxlci5hdXRob3JpemUoXG4gICAgICAgIGN0eCxcbiAgICAgICAgZWZmZWN0aXZlTW9kZWwgYXMgc3RyaW5nIHwgQ29uc3RydWN0b3IsXG4gICAgICAgIHJlcXVpcmVkUm9sZXMsXG4gICAgICAgIHRoaXMucmVxdWVzdENvbnRleHRcbiAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGxvZy5kZWJ1ZyhgTm8gYXV0aCBoYW5kbGVyIHJlZ2lzdGVyZWRgKTtcbiAgICB9XG5cbiAgICAvLyBUcmFuc2Zvcm1lcnMgcnVuIEFGVEVSIGF1dGggc28gdGhleSBjYW4gcmVhZCBhdXRoLXBvcHVsYXRlZCBmaWVsZHNcbiAgICAvLyAoZS5nLiBgdXNlcmApIGZyb20gdGhlIGNvbnRleHQgYW5kIG1hcCB0aGVtIHRvIGFkYXB0ZXItc3BlY2lmaWMga2V5c1xuICAgIC8vIChlLmcuIGBVVUlEYCBmb3IgUmFtQWRhcHRlcidzIEBjcmVhdGVkQnkvQHVwZGF0ZWRCeSBoYW5kbGVycykuXG4gICAgYXdhaXQgdGhpcy5hcHBseVRyYW5zZm9ybWVycygpO1xuXG4gICAgY29uc3QgdXNlciA9IHRoaXMucmVxdWVzdENvbnRleHQuZ2V0T3JVbmRlZmluZWQoXCJ1c2VyXCIgYXMgYW55KTtcbiAgICBjb25zdCBvcmdhbml6YXRpb24gPSB0aGlzLnJlcXVlc3RDb250ZXh0LmdldE9yVW5kZWZpbmVkKFwib3JnYW5pemF0aW9uXCIgYXMgYW55KTtcbiAgICBpZiAodXNlciB8fCBvcmdhbml6YXRpb24pIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRMb2cgPSB0aGlzLnJlcXVlc3RDb250ZXh0LmdldChcImxvZ2dlclwiIGFzIGFueSk7XG4gICAgICBjb25zdCBjaGlsZExvZyA9IGN1cnJlbnRMb2cuZm9yKHsgdXNlciwgb3JnYW5pemF0aW9uIH0pO1xuICAgICAgdGhpcy5yZXF1ZXN0Q29udGV4dC5hY2N1bXVsYXRlKHsgbG9nZ2VyOiBjaGlsZExvZyB9IGFzIGFueSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIG5leHQuaGFuZGxlKCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgYXN5bmMgYXBwbHlUcmFuc2Zvcm1lcnMoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgZmxhdm91cnMgPSBBZGFwdGVyLmZsYXZvdXJzVG9UcmFuc2Zvcm0oKTtcbiAgICBpZiAoIWZsYXZvdXJzKSByZXR1cm47XG5cbiAgICBmb3IgKGNvbnN0IGZsYXZvdXIgb2YgZmxhdm91cnMpIHtcbiAgICAgIGNvbnN0IHRyYW5zZm9ybWVyID0gQWRhcHRlci50cmFuc2Zvcm1lckZvcihcbiAgICAgICAgZmxhdm91clxuICAgICAgKSBhcyBSZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXI8YW55PjtcbiAgICAgIGlmICghdHJhbnNmb3JtZXIpIGNvbnRpbnVlO1xuICAgICAgY29uc3QgZnJvbSA9IGF3YWl0IHRyYW5zZm9ybWVyLmZyb20odGhpcy5yZXF1ZXN0Q29udGV4dCk7XG4gICAgICBpZiAoZnJvbSkgdGhpcy5yZXF1ZXN0Q29udGV4dC5hY2N1bXVsYXRlKGZyb20pO1xuICAgIH1cbiAgfVxufVxuIiwiaW1wb3J0IHsgYXBwbHlEZWNvcmF0b3JzLCBTZXRNZXRhZGF0YSwgVXNlSW50ZXJjZXB0b3JzIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBcGlCZWFyZXJBdXRoLCBBcGlTZWN1cml0eSB9IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5cbmltcG9ydCB7IEFVVEhfTUVUQV9LRVksIElTX1BVQkxJQ19LRVksIFJFUVVJUkVEX1JPTEVTX0tFWSB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgQXV0aEludGVyY2VwdG9yIH0gZnJvbSBcIi4vQXV0aEludGVyY2VwdG9yXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBBdXRoKG1vZGVsPzogc3RyaW5nIHwgQ29uc3RydWN0b3IpIHtcbiAgY29uc3QgcmVzb3VyY2UgPSBtb2RlbFxuICAgID8gdHlwZW9mIG1vZGVsID09PSBcInN0cmluZ1wiXG4gICAgICA/IG1vZGVsXG4gICAgICA6IG1vZGVsLm5hbWVcbiAgICA6IHVuZGVmaW5lZDtcbiAgY29uc3QgZGVjcyA9IFtBcGlCZWFyZXJBdXRoKCksIFVzZUludGVyY2VwdG9ycyhBdXRoSW50ZXJjZXB0b3IpXTtcbiAgaWYgKHJlc291cmNlKSBkZWNzLnB1c2goU2V0TWV0YWRhdGEoQVVUSF9NRVRBX0tFWSwgcmVzb3VyY2UpKTtcbiAgcmV0dXJuIGFwcGx5RGVjb3JhdG9ycyguLi5kZWNzKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIFB1YmxpYygpIHtcbiAgcmV0dXJuIGFwcGx5RGVjb3JhdG9ycyhTZXRNZXRhZGF0YShJU19QVUJMSUNfS0VZLCB0cnVlKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBSZXF1aXJlUm9sZXMoLi4ucm9sZXM6IHN0cmluZ1tdKSB7XG4gIHJldHVybiBhcHBseURlY29yYXRvcnMoXG4gICAgQXBpU2VjdXJpdHkoXCJiZWFyZXJcIiksXG4gICAgU2V0TWV0YWRhdGEoUkVRVUlSRURfUk9MRVNfS0VZLCByb2xlcyksXG4gICAgVXNlSW50ZXJjZXB0b3JzKEF1dGhJbnRlcmNlcHRvcilcbiAgKTtcbn1cbiIsImltcG9ydCB7IERlbGV0ZSwgR2V0LCBQYXRjaCwgUG9zdCwgUHV0IH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIE1hcHMgYW4gSFRUUCB2ZXJiIHRvIGl0cyBjb3JyZXNwb25kaW5nIE5lc3RKUyBtZXRob2QgZGVjb3JhdG9yLlxuICpcbiAqIEBwYXJhbSB2ZXJiIC0gSFRUUCB2ZXJiIHRvIGJlIGNvbnZlcnRlZCAoZS5nLiBHRVQsIFBPU1QsIFBVVCwgUEFUQ0gsIERFTEVURSkuXG4gKiBAcmV0dXJucyBBIE5lc3RKUyBtZXRob2QgZGVjb3JhdG9yIG1hdGNoaW5nIHRoZSBwcm92aWRlZCBIVFRQIHZlcmIuXG4gKlxuICogQHRocm93cyB7RXJyb3J9IElmIHRoZSBwcm92aWRlZCBIVFRQIHZlcmIgaXMgbm90IHN1cHBvcnRlZCBvciBub3QgbWFwcGVkXG4gKiB0byBhIE5lc3RKUyBkZWNvcmF0b3IuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBIdHRwVmVyYlRvRGVjb3JhdG9yKFxuICB2ZXJiOiBIdHRwVmVyYnNcbik6IChwYXRoPzogc3RyaW5nKSA9PiBNZXRob2REZWNvcmF0b3Ige1xuICBjb25zdCBodHRwVG9DcnVkOiBSZWNvcmQ8SHR0cFZlcmJzLCAocGF0aD86IHN0cmluZykgPT4gTWV0aG9kRGVjb3JhdG9yPiA9IHtcbiAgICBHRVQ6IEdldCxcbiAgICBQT1NUOiBQb3N0LFxuICAgIFBVVDogUHV0LFxuICAgIFBBVENIOiBQYXRjaCxcbiAgICBERUxFVEU6IERlbGV0ZSxcbiAgfTtcblxuICBjb25zdCBkZWNvcmF0b3IgPSBodHRwVG9DcnVkW3ZlcmJdO1xuXG4gIGlmICghZGVjb3JhdG9yKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgYFVuc3VwcG9ydGVkIEhUVFAgdmVyYiBcIiR7dmVyYn1cIi4gTm8gTmVzdEpTIGRlY29yYXRvciBtYXBwaW5nIHdhcyBmb3VuZC5gXG4gICAgKTtcbiAgfVxuXG4gIHJldHVybiBkZWNvcmF0b3I7XG59XG4iLCJpbXBvcnQgeyBRdWVyeSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyB0eXBlIERlY29yYXRvckJ1bmRsZSB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQge1xuICBBcGlOb0NvbnRlbnRSZXNwb25zZSxcbiAgQXBpT2tSZXNwb25zZSxcbiAgQXBpT3BlcmF0aW9uLFxuICBBcGlQYXJhbSxcbiAgQXBpUXVlcnksXG59IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB7XG4gIERlY2FmQXBpUHJvcGVydHksXG4gIHR5cGUgRGVjYWZQYXJhbVByb3BzLFxuICBEZWNhZlBhcmFtcyxcbn0gZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHR5cGUgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi9kZWNvcmF0b3JzL3R5cGVzXCI7XG5pbXBvcnQge1xuICBEaXJlY3Rpb25MaW1pdE9mZnNldCxcbiAgTW9kZWxTZXJ2aWNlLFxuICBPcmRlckRpcmVjdGlvbixcbiAgUmVwbyxcbn0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IEh0dHBWZXJiVG9EZWNvcmF0b3IgfSBmcm9tIFwiLi9kZWNvcmF0b3JzL3V0aWxzXCI7XG5pbXBvcnQgeyBEZWNhZk1vZGVsQ29udHJvbGxlciB9IGZyb20gXCIuLi9jb250cm9sbGVyc1wiO1xuXG5jb25zdCBleHRyYWN0UGF0aFBhcmFtcyA9IChyb3V0ZVBhdGg6IHN0cmluZyk6IHN0cmluZ1tdID0+IHtcbiAgcmV0dXJuIHJvdXRlUGF0aFxuICAgIC5zcGxpdChcIi9cIilcbiAgICAuZmlsdGVyKChwKSA9PiBwLnN0YXJ0c1dpdGgoXCI6XCIpKVxuICAgIC5tYXAoKHApID0+IHAuc2xpY2UoMSkpO1xufTtcblxuY29uc3QgYXBpUGFyYW1TcGVjID0gKG5hbWU6IHN0cmluZyk6IERlY2FmQXBpUHJvcGVydHkgPT4gKHtcbiAgbmFtZSxcbiAgZGVzY3JpcHRpb246IGAke25hbWV9IHBhcmFtZXRlciBmb3IgdGhlIHF1ZXJ5YCxcbiAgcmVxdWlyZWQ6IHRydWUsXG4gIHR5cGU6IFN0cmluZyxcbn0pO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0QXBpRGVjb3JhdG9ycyhcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICByb3V0ZVBhdGg6IHN0cmluZyxcbiAgaHR0cFZlcmI6IEh0dHBWZXJicyxcbiAgaW5jbHVkZVF1ZXJ5UGFyYW1zOiBib29sZWFuID0gZmFsc2Vcbik6IERlY29yYXRvckJ1bmRsZSB7XG4gIGNvbnN0IE5lc3RIdHRwUm91dGVEZWMgPSBIdHRwVmVyYlRvRGVjb3JhdG9yKGh0dHBWZXJiKTtcbiAgY29uc3QgYXBpUGF0aFBhcmFtcyA9IGV4dHJhY3RQYXRoUGFyYW1zKHJvdXRlUGF0aCkubWFwKGFwaVBhcmFtU3BlYyk7XG5cbiAgY29uc3Qgc3dhZ2dlclF1ZXJ5UGFyYW1zID0gW107XG4gIGlmIChodHRwVmVyYiA9PT0gXCJHRVRcIiAmJiBpbmNsdWRlUXVlcnlQYXJhbXMpIHtcbiAgICBzd2FnZ2VyUXVlcnlQYXJhbXMucHVzaChcbiAgICAgIEFwaVF1ZXJ5KHtcbiAgICAgICAgbmFtZTogXCJkaXJlY3Rpb25cIixcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgICBlbnVtOiBPcmRlckRpcmVjdGlvbixcbiAgICAgICAgZGVzY3JpcHRpb246IFwidGhlIHNvcnQgb3JkZXIgd2hlbiBhcHBsaWNhYmxlXCIsXG4gICAgICB9KSxcbiAgICAgIEFwaVF1ZXJ5KHtcbiAgICAgICAgbmFtZTogXCJsaW1pdFwiLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcImxpbWl0IG9yIHBhZ2Ugc2l6ZSB3aGVuIGFwcGxpY2FibGVcIixcbiAgICAgIH0pLFxuICAgICAgQXBpUXVlcnkoe1xuICAgICAgICBuYW1lOiBcIm9mZnNldFwiLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcIm9mZnNldCBvciBib29rbWFyayB3aGVuIGFwcGxpY2FibGVcIixcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbWV0aG9kOiBbXG4gICAgICBOZXN0SHR0cFJvdXRlRGVjKHJvdXRlUGF0aCksXG4gICAgICAuLi5hcGlQYXRoUGFyYW1zLm1hcChBcGlQYXJhbSksXG4gICAgICAuLi5zd2FnZ2VyUXVlcnlQYXJhbXMsXG4gICAgICBBcGlPcGVyYXRpb24oe1xuICAgICAgICBzdW1tYXJ5OiBgUmV0cmlldmUgcmVjb3JkcyB1c2luZyBhY2NvcmRpbmcgdG8gXCIke21ldGhvZE5hbWV9XCIuYCxcbiAgICAgIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgUmVzdWx0IHN1Y2Nlc3NmdWxseSByZXRyaWV2ZWQuYCxcbiAgICAgIH0pLFxuICAgICAgQXBpTm9Db250ZW50UmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vIGNvbnRlbnQgcmV0dXJuZWQgYnkgdGhlIG1ldGhvZC5gLFxuICAgICAgfSksXG4gICAgXSxcbiAgICBwYXJhbXM6IFtEZWNhZlBhcmFtcyhhcGlQYXRoUGFyYW1zKSwgUXVlcnkoKV0sIC8vICwgRGVjYWZCb2R5KCldLFxuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gYXBwbHlBcGlEZWNvcmF0b3JzKFxuICB0YXJnZXQ6IGFueSxcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICBkZXNjcmlwdG9yOiBQcm9wZXJ0eURlc2NyaXB0b3IsXG4gIGRlY29yYXRvcnM6IERlY29yYXRvckJ1bmRsZVxuKSB7XG4gIGNvbnN0IHByb3RvID0gdGFyZ2V0Py5wcm90b3R5cGUgPz8gdGFyZ2V0O1xuICBkZWNvcmF0b3JzLm1ldGhvZC5mb3JFYWNoKChkKSA9PiBkKHByb3RvLCBtZXRob2ROYW1lLCBkZXNjcmlwdG9yKSk7XG4gIGRlY29yYXRvcnMucGFyYW1zPy5mb3JFYWNoKChkLCBpbmRleCkgPT4gZChwcm90bywgbWV0aG9kTmFtZSwgaW5kZXgpKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVQZXJzaXN0ZW5jZU1ldGhvZDxUIGV4dGVuZHMgTW9kZWw8Ym9vbGVhbj4+KFxuICBwZXJzaXN0ZW5jZTogUmVwbzxUPiB8IE1vZGVsU2VydmljZTxUPixcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICAuLi5hcmdzOiBhbnlbXVxuKTogYW55IHtcbiAgaWYgKHBlcnNpc3RlbmNlIGluc3RhbmNlb2YgTW9kZWxTZXJ2aWNlKSB7XG4gICAgcmV0dXJuIHR5cGVvZiAocGVyc2lzdGVuY2UgYXMgYW55KVttZXRob2ROYW1lXSA9PT0gXCJmdW5jdGlvblwiXG4gICAgICA/IChwZXJzaXN0ZW5jZSBhcyBhbnkpW21ldGhvZE5hbWVdKC4uLmFyZ3MpXG4gICAgICA6IHBlcnNpc3RlbmNlLnN0YXRlbWVudChtZXRob2ROYW1lLCAuLi5hcmdzKTtcbiAgfVxuXG4gIGlmICh0eXBlb2YgKHBlcnNpc3RlbmNlIGFzIGFueSlbbWV0aG9kTmFtZV0gPT09IFwiZnVuY3Rpb25cIilcbiAgICByZXR1cm4gKHBlcnNpc3RlbmNlIGFzIGFueSlbbWV0aG9kTmFtZV0oLi4uYXJncyk7XG5cbiAgdGhyb3cgbmV3IEVycm9yKFxuICAgIGBQZXJzaXN0ZW5jZSBtZXRob2QgXCIke21ldGhvZE5hbWV9XCIgbm90IGZvdW5kIG9uICR7cGVyc2lzdGVuY2U/LmNvbnN0cnVjdG9yPy5uYW1lfWBcbiAgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVJvdXRlSGFuZGxlcjxUPihtZXRob2ROYW1lOiBzdHJpbmcpIHtcbiAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIChcbiAgICB0aGlzOiBEZWNhZk1vZGVsQ29udHJvbGxlcjxhbnk+LFxuICAgIHBhdGhQYXJhbXM6IERlY2FmUGFyYW1Qcm9wcyxcbiAgICBxdWVyeVBhcmFtczogRGlyZWN0aW9uTGltaXRPZmZzZXRcbiAgKTogUHJvbWlzZTxUPiB7XG4gICAgY29uc3QgbG9nOiBMb2dnZXIgPSB0aGlzLmxvZy5mb3IobWV0aG9kTmFtZSk7XG5cbiAgICB0cnkge1xuICAgICAgbG9nLmRlYnVnKFxuICAgICAgICBgSW52b2tpbmcgcGVyc2lzdGVuY2UgbWV0aG9kIFwiJHttZXRob2ROYW1lfVwiIGdpdmVuIHBhcmFtZXRlcnM6ICR7SlNPTi5zdHJpbmdpZnkocGF0aFBhcmFtcy52YWx1ZXNJbk9yZGVyKX1gXG4gICAgICApO1xuICAgICAgY29uc3QgeyBkaXJlY3Rpb24sIGxpbWl0LCBvZmZzZXQgfSA9IHF1ZXJ5UGFyYW1zO1xuICAgICAgcmV0dXJuIGF3YWl0IHJlc29sdmVQZXJzaXN0ZW5jZU1ldGhvZChcbiAgICAgICAgdGhpcy5wZXJzaXN0ZW5jZSgpLFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICAuLi5wYXRoUGFyYW1zLnZhbHVlc0luT3JkZXIsXG4gICAgICAgIGRpcmVjdGlvbixcbiAgICAgICAgbGltaXQsXG4gICAgICAgIG9mZnNldFxuICAgICAgKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIGxvZy5lcnJvcihgQ3VzdG9tIHF1ZXJ5IFwiJHttZXRob2ROYW1lfVwiIGZhaWxlZGAsIGUpO1xuICAgICAgdGhyb3cgZTtcbiAgICB9XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBkZWZpbmVSb3V0ZU1ldGhvZChcbiAgQ29udHJvbGxlckNsYXNzOiBuZXcgKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnksXG4gIG1ldGhvZE5hbWU6IHN0cmluZyxcbiAgaGFuZGxlcjogKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnlcbik6IFByb3BlcnR5RGVzY3JpcHRvciB8IHVuZGVmaW5lZCB7XG4gIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShcbiAgICBDb250cm9sbGVyQ2xhc3MucHJvdG90eXBlIHx8IENvbnRyb2xsZXJDbGFzcyxcbiAgICBtZXRob2ROYW1lLFxuICAgIHtcbiAgICAgIHZhbHVlOiBoYW5kbGVyLFxuICAgICAgd3JpdGFibGU6IGZhbHNlLFxuICAgIH1cbiAgKTtcblxuICByZXR1cm4gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihcbiAgICBDb250cm9sbGVyQ2xhc3MucHJvdG90eXBlIHx8IENvbnRyb2xsZXJDbGFzcyxcbiAgICBtZXRob2ROYW1lXG4gICk7XG59XG4iLCJpbXBvcnQgeyBFeGVjdXRpb25Db250ZXh0IH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBdXRob3JpemF0aW9uRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IEF1dGhIYW5kbGVyLCBBdXRoRGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5pbXBvcnQgeyBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4uL3JlcXVlc3QvRGVjYWZSZXF1ZXN0Q29udGV4dFwiO1xuXG4vKipcbiAqIFNpbXBsZSBhdXRoIGhhbmRsZXIgdGhhdCByZWFkcyBhIHJvbGUgc3RyaW5nIGZyb20gdGhlIGBBdXRob3JpemF0aW9uOiBCZWFyZXIgPHJvbGU+YCBoZWFkZXIuXG4gKlxuICogT25seSBvdmVycmlkZXMge0BsaW5rIEF1dGhIYW5kbGVyLmV4dHJhY3RGcm9tQXV0aH0gdG8gcmV0dXJuIHRoZSBiZWFyZXIgdG9rZW4gYXNcbiAqIGJvdGggdGhlIHVzZXIgaWRlbnRpZmllciBhbmQgdGhlIHNpbmdsZSByb2xlLiBUaGUgYmFzZSBjbGFzcyBgYmluZFRvQ29udGV4dGBcbiAqIChgY3R4LmFjY3VtdWxhdGUoZGF0YSlgKSBpcyBzdWZmaWNpZW50IOKAlCB0aGUgdHJhbnNmb3JtZXIvYWRhcHRlciBpcyByZXNwb25zaWJsZVxuICogZm9yIG1hcHBpbmcgY29udGV4dCBmaWVsZHMgdG8gYWRhcHRlci1zcGVjaWZpYyBrZXlzIGxpa2UgYFVVSURgLlxuICovXG5leHBvcnQgY2xhc3MgRGVjYWZBdXRoSGFuZGxlciBleHRlbmRzIEF1dGhIYW5kbGVyPFxuICBFeGVjdXRpb25Db250ZXh0LFxuICBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICBBdXRoRGF0YVxuPiB7XG4gIHByb3RlY3RlZCBwYXJzZVJlcXVlc3QocmVxOiBhbnkpOiBzdHJpbmcge1xuICAgIGNvbnN0IHVzZXJSb2xlID0gcmVxLmhlYWRlcnMuYXV0aG9yaXphdGlvbj8uc3BsaXQoXCIgXCIpWzFdIGFzIHN0cmluZztcbiAgICByZXR1cm4gdXNlclJvbGU7XG4gIH1cblxuICBwcm90ZWN0ZWQgZXh0cmFjdEZyb21BdXRoKGN0eDogRXhlY3V0aW9uQ29udGV4dCk6IEF1dGhEYXRhIHtcbiAgICBjb25zdCByZXEgPSBjdHguc3dpdGNoVG9IdHRwKCkuZ2V0UmVxdWVzdCgpO1xuICAgIGNvbnN0IHVzZXJSb2xlID0gdGhpcy5wYXJzZVJlcXVlc3QocmVxKTtcbiAgICBpZiAoIXVzZXJSb2xlKSB0aHJvdyBuZXcgQXV0aG9yaXphdGlvbkVycm9yKFwiVW5hdXRoZW50aWNhdGVkXCIpO1xuICAgIHJldHVybiB7IHVzZXI6IHVzZXJSb2xlLCByb2xlczogW3VzZXJSb2xlXSB9O1xuICB9XG59XG5cbi8qKlxuICogQWxpYXMgZm9yIHtAbGluayBEZWNhZkF1dGhIYW5kbGVyfSBrZXB0IGZvciBiYWNrd2FyZCBjb21wYXRpYmlsaXR5LlxuICogVGhlIFwicm9sZVwiIHZhcmlhbnQgcHJldmlvdXNseSByZXR1cm5lZCByb2xlcyBmcm9tIGBhdXRob3JpemVgOyB3aXRoIHRoZSBiYXNlXG4gKiBjbGFzcyBub3cgb3JjaGVzdHJhdGluZyByb2xlIGNoZWNrcyBpbnRlcm5hbGx5LCB0aGUgdHdvIGNsYXNzZXMgYXJlIGVxdWl2YWxlbnQuXG4gKi9cbmV4cG9ydCBjbGFzcyBEZWNhZlJvbGVBdXRoSGFuZGxlciBleHRlbmRzIERlY2FmQXV0aEhhbmRsZXIge31cbiIsImltcG9ydCB7IENvbnN0cnVjdG9yLCBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgUmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyIH0gZnJvbSBcIkBkZWNhZi10cy9mb3ItaHR0cC9zZXJ2ZXJcIjtcbmltcG9ydCB7IEFkYXB0ZXIsIENvbnRleHRPZiwgQ29udGV4dCB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuXG4oQWRhcHRlciBhcyBhbnkpLnRyYW5zZm9ybWVyRm9yID0gZnVuY3Rpb24gdG9Db250ZXh0RmxhZ3M8XG4gIEEgZXh0ZW5kcyBBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueT4sXG4+KGFkYXB0ZXI6IEEgfCBzdHJpbmcpOiBDb25zdHJ1Y3RvcjxSZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXI8Q29udGV4dE9mPEE+Pj4ge1xuICBjb25zdCBhbGlhcyA9XG4gICAgdHlwZW9mIGFkYXB0ZXIgPT09IFwic3RyaW5nXCIgPyBhZGFwdGVyIDogKGFkYXB0ZXIuYWxpYXMgYXMgc3RyaW5nKTtcbiAgcmV0dXJuIE1ldGFkYXRhW1wiaW5uZXJHZXRcIl0oU3ltYm9sLmZvcihcInRyYW5zZm9ybWVyc1wiKSwgYWxpYXMpO1xufS5iaW5kKEFkYXB0ZXIpO1xuXG4oQWRhcHRlciBhcyBhbnkpLmZsYXZvdXJzVG9UcmFuc2Zvcm0gPSBmdW5jdGlvbiByZXF1ZXN0VHJhbnNmb3JtZXJzKCk6XG4gIHwgc3RyaW5nW11cbiAgfCB1bmRlZmluZWQge1xuICBjb25zdCBtZXRhID0gTWV0YWRhdGFbXCJpbm5lckdldFwiXShTeW1ib2wuZm9yKFwidHJhbnNmb3JtZXJzXCIpKTtcbiAgaWYgKCFtZXRhKSByZXR1cm4gdW5kZWZpbmVkO1xuICByZXR1cm4gT2JqZWN0LmtleXMobWV0YSk7XG59LmJpbmQoQWRhcHRlcik7XG5cbihDb250ZXh0IGFzIGFueSkucHJvdG90eXBlLnRvUmVzcG9uc2UgPSBmdW5jdGlvbiB0b1Jlc3BvbnNlPFxuICBSRVMgZXh0ZW5kcyB7IGhlYWRlcjogYW55IH0sXG4+KHRoaXM6IENvbnRleHQsIHJlczogUkVTKTogUkVTIHtcbiAgY29uc3QgcGVuZGluZyA9IHRoaXMucGVuZGluZygpO1xuICBpZiAocGVuZGluZykgcmVzLmhlYWRlcihcIngtcGVuZGluZy10YXNrXCIsIEpTT04uc3RyaW5naWZ5KHBlbmRpbmcpKTtcbiAgcmV0dXJuIHJlcztcbn07XG4iLCJpbXBvcnQgXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBNb2RlbEJ1aWxkZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBBdXRoIH0gZnJvbSBcIi4uL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvZGVjb3JhdG9yc1wiO1xuXG5kZWNsYXJlIG1vZHVsZSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiIHtcbiAgZXhwb3J0IGludGVyZmFjZSBNb2RlbEJ1aWxkZXI8TT4ge1xuICAgIEF1dGgobW9kZWw6IHN0cmluZyB8IENvbnN0cnVjdG9yKTogTW9kZWxCdWlsZGVyPE0+O1xuICAgIGRlY29yYXRlQ2xhc3MoZGVjb3JhdG9yOiBDbGFzc0RlY29yYXRvcik6IE1vZGVsQnVpbGRlcjxNPjtcbiAgfVxufVxuXG5jb25zdCBwcm90b3R5cGUgPSBNb2RlbEJ1aWxkZXIucHJvdG90eXBlIGFzIE1vZGVsQnVpbGRlcjxhbnk+ICYge1xuICBBdXRoOiAobW9kZWw6IHN0cmluZyB8IENvbnN0cnVjdG9yKSA9PiBNb2RlbEJ1aWxkZXI8YW55Pjtcbn07XG5cbmlmICghcHJvdG90eXBlLmRlY29yYXRlQ2xhc3MpIHtcbiAgcHJvdG90eXBlLmRlY29yYXRlQ2xhc3MgPSBmdW5jdGlvbiAoZGVjb3JhdG9yOiBDbGFzc0RlY29yYXRvcikge1xuICAgIGlmICghKHRoaXMgYXMgYW55KS5fY2xhc3NEZWNvcmF0b3JzKSB7XG4gICAgICAodGhpcyBhcyBhbnkpLl9jbGFzc0RlY29yYXRvcnMgPSBbXTtcbiAgICB9XG4gICAgKHRoaXMgYXMgYW55KS5fY2xhc3NEZWNvcmF0b3JzLnB1c2goZGVjb3JhdG9yKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfTtcbn1cblxucHJvdG90eXBlLkF1dGggPSBmdW5jdGlvbiAobW9kZWw6IHN0cmluZyB8IENvbnN0cnVjdG9yKSB7XG4gIHJldHVybiB0aGlzLmRlY29yYXRlQ2xhc3MoQXV0aChtb2RlbCkpO1xufTtcblxuaWYgKCEocHJvdG90eXBlIGFzIGFueSkuX19oYXNDbGFzc0RlY29yYXRvclN1cHBvcnQpIHtcbiAgY29uc3Qgb3JpZ2luYWxCdWlsZCA9IHByb3RvdHlwZS5idWlsZDtcbiAgcHJvdG90eXBlLmJ1aWxkID0gZnVuY3Rpb24gKCkge1xuICAgIGxldCByZXN1bHQgPSBvcmlnaW5hbEJ1aWxkLmNhbGwodGhpcyk7XG4gICAgY29uc3QgZGVjb3JhdG9ycyA9ICh0aGlzIGFzIGFueSkuX2NsYXNzRGVjb3JhdG9ycztcbiAgICBpZiAoZGVjb3JhdG9ycz8ubGVuZ3RoKSB7XG4gICAgICBmb3IgKGNvbnN0IGRlY29yYXRvciBvZiBkZWNvcmF0b3JzKSB7XG4gICAgICAgIGNvbnN0IGRlY29yYXRlZCA9IGRlY29yYXRvcihyZXN1bHQgYXMgYW55KSBhcyB0eXBlb2YgcmVzdWx0IHwgdm9pZDtcbiAgICAgICAgaWYgKGRlY29yYXRlZCkge1xuICAgICAgICAgIHJlc3VsdCA9IGRlY29yYXRlZDtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gcmVzdWx0O1xuICB9O1xuICAocHJvdG90eXBlIGFzIGFueSkuX19oYXNDbGFzc0RlY29yYXRvclN1cHBvcnQgPSB0cnVlO1xufVxuIiwiaW1wb3J0IHtcbiAgQ2FsbEhhbmRsZXIsXG4gIEV4ZWN1dGlvbkNvbnRleHQsXG4gIEluamVjdGFibGUsXG4gIE5lc3RJbnRlcmNlcHRvcixcbiAgU2NvcGUsXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRGVjYWZIYW5kbGVyRXhlY3V0b3IsIERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi4vcmVxdWVzdFwiO1xuaW1wb3J0IHsgRGVmYXVsdEFkYXB0ZXJGbGFncyB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgRGVjYWZTZXJ2ZXJGbGFncyB9IGZyb20gXCIuLi9jb25zdGFudHNcIjtcbmltcG9ydCBcIi4uL292ZXJyaWRlc1wiO1xuaW1wb3J0IHsgTG9nZ2luZyB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvblxuICogSW50ZXJjZXB0b3IgcmVzcG9uc2libGUgZm9yIGV4ZWN1dGluZyBhbGwgcmVnaXN0ZXJlZCBEZWNhZiByZXF1ZXN0IGhhbmRsZXJzXG4gKiBiZWZvcmUgdGhlIGNvbnRyb2xsZXIgbWV0aG9kIGlzIGludm9rZWQuXG4gKlxuICogQHN1bW1hcnlcbiAqIFRoZSB7QGxpbmsgRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yfSBpbnRlZ3JhdGVzIHRoZSBEZWNhZiByZXF1ZXN0LWhhbmRsaW5nIHBpcGVsaW5lXG4gKiBpbnRvIE5lc3RKUycgaW50ZXJjZXB0b3IgbWVjaGFuaXNtLiBCZWZvcmUgcGFzc2luZyBleGVjdXRpb24gdG8gdGhlIG5leHQgaGFuZGxlciBpbiB0aGVcbiAqIE5lc3RKUyBjaGFpbiwgaXQgZGVsZWdhdGVzIHJlcXVlc3QgcHJvY2Vzc2luZyB0byB0aGUge0BsaW5rIERlY2FmSGFuZGxlckV4ZWN1dG9yfSwgd2hpY2hcbiAqIHNlcXVlbnRpYWxseSBydW5zIGFsbCByZWdpc3RlcmVkIHtAbGluayBEZWNhZlJlcXVlc3RIYW5kbGVyfSBpbnN0YW5jZXMuIFRoaXMgYWxsb3dzXG4gKiBiZWhhdmlvcnMgc3VjaCBhcyBhdXRoZW50aWNhdGlvbiwgbG9nZ2luZywgdGVuYW50IHJlc29sdXRpb24sIG9yIG1ldGFkYXRhIGVucmljaG1lbnRcbiAqIHRvIG9jY3VyIHByaW9yIHRvIGNvbnRyb2xsZXIgZXhlY3V0aW9uLlxuICpcbiAqIEBjbGFzcyBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3JcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIC8vIEFwcGx5IGdsb2JhbGx5OlxuICogYXBwLnVzZUdsb2JhbEludGVyY2VwdG9ycyhuZXcgRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yKGV4ZWN1dG9yKSk7XG4gKlxuICogLy8gT3IgaW4gYSBtb2R1bGU6XG4gKiBATW9kdWxlKHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgRGVjYWZIYW5kbGVyRXhlY3V0b3IsXG4gKiAgICAge1xuICogICAgICAgcHJvdmlkZTogQVBQX0lOVEVSQ0VQVE9SLFxuICogICAgICAgdXNlQ2xhc3M6IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvcixcbiAqICAgICB9LFxuICogICBdLFxuICogfSlcbiAqIGV4cG9ydCBjbGFzcyBBcHBNb2R1bGUge31cbiAqIGBgYFxuICpcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgICBwYXJ0aWNpcGFudCBDbGllbnRcbiAqICAgICBwYXJ0aWNpcGFudCBJbnRlcmNlcHRvclxuICogICAgIHBhcnRpY2lwYW50IEV4ZWN1dG9yXG4gKiAgICAgcGFydGljaXBhbnQgQ29udHJvbGxlclxuICpcbiAqICAgICBDbGllbnQtPj5JbnRlcmNlcHRvcjogSFRUUCBSZXF1ZXN0XG4gKiAgICAgSW50ZXJjZXB0b3ItPj5FeGVjdXRvcjogZXhlYyhyZXF1ZXN0KVxuICogICAgIEV4ZWN1dG9yLS0+PkludGVyY2VwdG9yOiBoYW5kbGVycyBjb21wbGV0ZWRcbiAqICAgICBJbnRlcmNlcHRvci0+PkNvbnRyb2xsZXI6IG5leHQuaGFuZGxlKClcbiAqICAgICBDb250cm9sbGVyLS0+PkNsaWVudDogUmVzcG9uc2VcbiAqL1xuQEluamVjdGFibGUoeyBzY29wZTogU2NvcGUuUkVRVUVTVCB9KVxuZXhwb3J0IGNsYXNzIERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvciBpbXBsZW1lbnRzIE5lc3RJbnRlcmNlcHRvciB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHByb3RlY3RlZCByZWFkb25seSByZXF1ZXN0Q29udGV4dDogRGVjYWZSZXF1ZXN0Q29udGV4dCxcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgZXhlY3V0b3I6IERlY2FmSGFuZGxlckV4ZWN1dG9yXG4gICkge31cblxuICBwcm90ZWN0ZWQgYXN5bmMgY29udGV4dHVhbGl6ZShyZXE6IGFueSk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGhlYWRlcnMgPSByZXEuaGVhZGVycztcbiAgICBjb25zdCBmbGFnczogRGVjYWZTZXJ2ZXJGbGFncyA9IHtcbiAgICAgIGhlYWRlcnM6IGhlYWRlcnMsXG4gICAgICBvdmVycmlkZXM6IHt9LFxuICAgIH0gYXMgYW55O1xuXG4gICAgY29uc3QgaXAgPSBleHRyYWN0SXAocmVxKTtcbiAgICBjb25zdCBsb2dnZXIgPSBMb2dnaW5nLmdldCgpLmZvcih7IGlwIH0pO1xuXG4gICAgdGhpcy5yZXF1ZXN0Q29udGV4dC5hY2N1bXVsYXRlKFxuICAgICAgT2JqZWN0LmFzc2lnbihcbiAgICAgICAge30sXG4gICAgICAgIERlZmF1bHRBZGFwdGVyRmxhZ3MsXG4gICAgICAgIHtcbiAgICAgICAgICBsb2dnZXIsXG4gICAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLFxuICAgICAgICAgIG9wZXJhdGlvbjogYCR7cmVxLm1ldGhvZH0gJHtyZXEudXJsfWAsXG4gICAgICAgIH0sXG4gICAgICAgIGZsYWdzXG4gICAgICApXG4gICAgKTtcbiAgfVxuXG4gIGFzeW5jIGludGVyY2VwdChjb250ZXh0OiBFeGVjdXRpb25Db250ZXh0LCBuZXh0OiBDYWxsSGFuZGxlcikge1xuICAgIGNvbnN0IHJlcSA9IGNvbnRleHQuc3dpdGNoVG9IdHRwKCkuZ2V0UmVxdWVzdCgpO1xuICAgIGNvbnN0IHJlcyA9IGNvbnRleHQuc3dpdGNoVG9IdHRwKCkuZ2V0UmVzcG9uc2UoKTtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IpLmZvcih0aGlzLmludGVyY2VwdCk7XG4gICAgbG9nLmRlYnVnKFxuICAgICAgYENPTlRFWFQgJHt0aGlzLnJlcXVlc3RDb250ZXh0LnV1aWR9IC0gcmVxdWVzdDogJHtyZXEubWV0aG9kfSAke3JlcS51cmx9YFxuICAgICk7XG4gICAgYXdhaXQgdGhpcy5jb250ZXh0dWFsaXplKHJlcSk7XG4gICAgbG9nLmRlYnVnKFxuICAgICAgYENPTlRFWFQgJHt0aGlzLnJlcXVlc3RDb250ZXh0LnV1aWR9IGNvbnRleHR1YWxpemVkIC0gcmVxdWVzdDogJHtyZXEubWV0aG9kfSAke3JlcS51cmx9YFxuICAgICk7XG5cbiAgICBhd2FpdCB0aGlzLmV4ZWN1dG9yLmV4ZWMocmVxLCByZXMpO1xuICAgIGxvZy5kZWJ1ZyhcbiAgICAgIGBDT05URVhUICR7dGhpcy5yZXF1ZXN0Q29udGV4dC51dWlkfSBleGVjdXRvcnMgZmluaXNoZWQgLSByZXF1ZXN0OiAke3JlcS5tZXRob2R9ICR7cmVxLnVybH1gXG4gICAgKTtcbiAgICByZXR1cm4gbmV4dC5oYW5kbGUoKTtcbiAgfVxufVxuXG5mdW5jdGlvbiBleHRyYWN0SXAocmVxOiBhbnkpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICBjb25zdCBoZWFkZXJzID0gcmVxLmhlYWRlcnM7XG4gIGZ1bmN0aW9uIHBhcnNlSXBIZWFkZXIodmFsdWU/OiBzdHJpbmcgfCBzdHJpbmdbXSk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gICAgaWYgKCF2YWx1ZSkgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICBjb25zdCBjYW5kaWRhdGUgPSBBcnJheS5pc0FycmF5KHZhbHVlKSA/IHZhbHVlWzBdIDogdmFsdWU7XG4gICAgcmV0dXJuIGNhbmRpZGF0ZVxuICAgICAgLnNwbGl0KFwiLFwiKVxuICAgICAgLm1hcCgoc2VnbWVudCkgPT4gc2VnbWVudC50cmltKCkpXG4gICAgICAuZmlsdGVyKEJvb2xlYW4pWzBdO1xuICB9XG4gIHJldHVybiAoXG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJ4LWZvcndhcmRlZC1mb3JcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJ4LXJlYWwtaXBcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJYLUZvcndhcmRlZC1Gb3JcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJYLVJlYWwtSVBcIl0pID8/XG4gICAgcmVxLmlwXG4gICk7XG59XG4iLCJpbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBNb2R1bGUsIFR5cGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IEFQUF9JTlRFUkNFUFRPUiB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcblxuaW1wb3J0IHsgQXV0aEludGVyY2VwdG9yIH0gZnJvbSBcIi4vQXV0aEludGVyY2VwdG9yXCI7XG5pbXBvcnQgeyBBVVRIX0hBTkRMRVIgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvciB9IGZyb20gXCIuLi9pbnRlcmNlcHRvcnMvRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yXCI7XG5pbXBvcnQgeyBBdXRoSGFuZGxlciB9IGZyb20gXCIuLi90eXBlc1wiO1xuXG5leHBvcnQgdHlwZSBEZWNhZkF1dGhNb2R1bGVPcHRpb25zID0ge1xuICBnbG9iYWw/OiBib29sZWFuO1xuICBoYW5kbGVyPzogVHlwZTxBdXRoSGFuZGxlcj47XG59O1xuXG5ATW9kdWxlKHt9KVxuZXhwb3J0IGNsYXNzIERlY2FmQXV0aE1vZHVsZSB7XG4gIHN0YXRpYyBmb3JSb290KFxuICAgIG9wdGlvbnM6IERlY2FmQXV0aE1vZHVsZU9wdGlvbnMgPSB7fVxuICApOiBEeW5hbWljTW9kdWxlIHtcbiAgICBjb25zdCBwcm92aWRlcnM6IER5bmFtaWNNb2R1bGVbXCJwcm92aWRlcnNcIl0gPSBbXG4gICAgICBBdXRoSW50ZXJjZXB0b3IsXG4gICAgICBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IsXG4gICAgICB7XG4gICAgICAgIHByb3ZpZGU6IEFQUF9JTlRFUkNFUFRPUixcbiAgICAgICAgdXNlQ2xhc3M6IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvcixcbiAgICAgIH0sXG4gICAgXTtcblxuICAgIGlmIChvcHRpb25zLmhhbmRsZXIpIHtcbiAgICAgIHByb3ZpZGVycy5wdXNoKG9wdGlvbnMuaGFuZGxlcik7XG4gICAgICBwcm92aWRlcnMucHVzaCh7XG4gICAgICAgIHByb3ZpZGU6IEFVVEhfSEFORExFUixcbiAgICAgICAgdXNlQ2xhc3M6IG9wdGlvbnMuaGFuZGxlcixcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIGlmIChvcHRpb25zLmdsb2JhbCkge1xuICAgICAgKHByb3ZpZGVycyBhcyBhbnlbXSkucHVzaCh7XG4gICAgICAgIHByb3ZpZGU6IEFQUF9JTlRFUkNFUFRPUixcbiAgICAgICAgdXNlRXhpc3Rpbmc6IEF1dGhJbnRlcmNlcHRvcixcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IERlY2FmQXV0aE1vZHVsZSxcbiAgICAgIGdsb2JhbDogb3B0aW9ucy5nbG9iYWwgPz8gZmFsc2UsXG4gICAgICBwcm92aWRlcnMsXG4gICAgICBleHBvcnRzOiBbQXV0aEludGVyY2VwdG9yLCBBVVRIX0hBTkRMRVJdLFxuICAgIH07XG4gIH1cbn1cbiIsImltcG9ydCB7IEluamVjdCwgSW5qZWN0YWJsZSwgU2NvcGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5pbXBvcnQgeyB0eXBlIERlY2FmUmVxdWVzdEhhbmRsZXIgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB7IERFQ0FGX0hBTkRMRVJTIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgTG9nZ2luZyB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvblxuICogRXhlY3V0ZXMgYWxsIHJlZ2lzdGVyZWQge0BsaW5rIERlY2FmUmVxdWVzdEhhbmRsZXJ9IGluc3RhbmNlcyBmb3IgdGhlIGN1cnJlbnQgcmVxdWVzdCxcbiAqIHByb3ZpZGluZyB0aGVtIHdpdGggYSBzaGFyZWQge0BsaW5rIERlY2FmUmVxdWVzdENvbnRleHR9LlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUge0BsaW5rIERlY2FmSGFuZGxlckV4ZWN1dG9yfSBjbGFzcyBpcyByZXNwb25zaWJsZSBmb3Igb3JjaGVzdHJhdGluZyBhbmQgZXhlY3V0aW5nXG4gKiBhIHNlcXVlbmNlIG9mIHJlcXVlc3QgaGFuZGxlcnMuIEVhY2ggaGFuZGxlciByZWNlaXZlcyB0aGUgc2FtZSByZXF1ZXN0LXNjb3BlZCBjb250ZXh0LFxuICogYWxsb3dpbmcgY29vcmRpbmF0ZWQgcHJvY2Vzc2luZyBzdWNoIGFzIGF1dGhlbnRpY2F0aW9uLCBtZXRhZGF0YSBleHRyYWN0aW9uLCBhdWRpdGluZyxcbiAqIGFuZCBjdXN0b20gcGlwZWxpbmUgYmVoYXZpb3IuIEhhbmRsZXJzIGFyZSBpbmplY3RlZCB2aWEgdGhlIHtAbGluayBERUNBRl9IQU5ETEVSU30gdG9rZW4sXG4gKiBlbnN1cmluZyBleHRlbnNpYmlsaXR5IGFuZCBsb29zZSBjb3VwbGluZy5cbiAqXG4gKiBAY2xhc3MgRGVjYWZIYW5kbGVyRXhlY3V0b3JcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIC8vIEV4YW1wbGUgaGFuZGxlcjpcbiAqIGNsYXNzIEF1dGhIYW5kbGVyIGltcGxlbWVudHMgRGVjYWZSZXF1ZXN0SGFuZGxlciB7XG4gKiAgIGFzeW5jIGhhbmRsZShjb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LCByZXE6IFJlcXVlc3QpIHtcbiAqICAgICBjb25zdCB0b2tlbiA9IHJlcS5oZWFkZXJzW1wiYXV0aG9yaXphdGlvblwiXTtcbiAqICAgICBjb25zdCByZXN1bHQgPSBNeVNlcnZpY2UuZG9Tb21ldGhpbmcodG9rZW4pO1xuICogICAgIGNvbnRleHQuc2V0KFwibXkta2V5XCIsIHJlc3VsdCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBFeGVjdXRvciB1c2FnZSBpbiBhIHJlcXVlc3Q6XG4gKiBhd2FpdCBleGVjdXRvci5leGVjKHJlcXVlc3QpO1xuICogLy8gQWxsIGhhbmRsZXJzIHdpbGwgcnVuIGluIHNlcXVlbmNlXG4gKiBgYGBcbiAqXG4gKiBAbWVybWFpZFxuICogc2VxdWVuY2VEaWFncmFtXG4gKiAgICAgcGFydGljaXBhbnQgQ2xpZW50XG4gKiAgICAgcGFydGljaXBhbnQgRXhlY3V0b3JcbiAqICAgICBwYXJ0aWNpcGFudCBIYW5kbGVyQVxuICogICAgIHBhcnRpY2lwYW50IEhhbmRsZXJCXG4gKlxuICogICAgIENsaWVudC0+PkV4ZWN1dG9yOiBleGVjKHJlcSlcbiAqICAgICBFeGVjdXRvci0+PkhhbmRsZXJBOiBoYW5kbGUoY29udGV4dCwgcmVxKVxuICogICAgIEhhbmRsZXJBLS0+PkV4ZWN1dG9yOiBjb21wbGV0ZWRcbiAqICAgICBFeGVjdXRvci0+PkhhbmRsZXJCOiBoYW5kbGUoY29udGV4dCwgcmVxKVxuICogICAgIEhhbmRsZXJCLS0+PkV4ZWN1dG9yOiBjb21wbGV0ZWRcbiAqICAgICBFeGVjdXRvci0tPj5DbGllbnQ6IHByb2Nlc3NpbmcgZmluaXNoZWRcbiAqL1xuQEluamVjdGFibGUoeyBzY29wZTogU2NvcGUuUkVRVUVTVCB9KVxuZXhwb3J0IGNsYXNzIERlY2FmSGFuZGxlckV4ZWN1dG9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgQEluamVjdChERUNBRl9IQU5ETEVSUykgcHJpdmF0ZSByZWFkb25seSBoYW5kbGVyczogRGVjYWZSZXF1ZXN0SGFuZGxlcltdLFxuICAgIHByaXZhdGUgcmVhZG9ubHkgY29udGV4dDogRGVjYWZSZXF1ZXN0Q29udGV4dFxuICApIHt9XG5cbiAgYXN5bmMgZXhlYyhyZXE6IFJlcXVlc3QsIHJlczogUmVzcG9uc2UpIHtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZkhhbmRsZXJFeGVjdXRvci5uYW1lKS5mb3IodGhpcy5leGVjKTtcbiAgICBsb2cuZGVidWcoXG4gICAgICBgQ09OVEVYVCAke3RoaXMuY29udGV4dC51dWlkfSBydW5uaW5nICR7dGhpcy5oYW5kbGVycy5sZW5ndGh9IGhhbmRsZXJzIGZvciByZXF1ZXN0ICR7cmVxLm1ldGhvZH0gJHtyZXEudXJsfWBcbiAgICApO1xuICAgIGZvciAoY29uc3QgaGFuZGxlciBvZiB0aGlzLmhhbmRsZXJzKSB7XG4gICAgICBhd2FpdCBoYW5kbGVyLmhhbmRsZSh0aGlzLmNvbnRleHQsIHJlcSwgcmVzKTtcbiAgICB9XG4gIH1cbn1cbiIsImltcG9ydCB7XG4gIEluamVjdGFibGUsXG4gIE5lc3RJbnRlcmNlcHRvcixcbiAgRXhlY3V0aW9uQ29udGV4dCxcbiAgQ2FsbEhhbmRsZXIsXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgdGFwIH0gZnJvbSBcInJ4anNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBEZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IgaW1wbGVtZW50cyBOZXN0SW50ZXJjZXB0b3Ige1xuICBjb25zdHJ1Y3Rvcihwcm90ZWN0ZWQgY3R4OiBEZWNhZlJlcXVlc3RDb250ZXh0KSB7fVxuXG4gIGludGVyY2VwdChjb250ZXh0OiBFeGVjdXRpb25Db250ZXh0LCBuZXh0OiBDYWxsSGFuZGxlcik6IE9ic2VydmFibGU8YW55PiB7XG4gICAgbGV0IHJlc3BvbnNlID0gY29udGV4dC5zd2l0Y2hUb0h0dHAoKS5nZXRSZXNwb25zZSgpO1xuXG4gICAgcmV0dXJuIG5leHQuaGFuZGxlKCkucGlwZShcbiAgICAgIHRhcCgoZGF0YSkgPT4ge1xuICAgICAgICByZXNwb25zZSA9IHRoaXMuY3R4LnRvUmVzcG9uc2UocmVzcG9uc2UpO1xuICAgICAgfSlcbiAgICApO1xuICB9XG59XG4iLCJpbXBvcnQgeyB0eXBlIFJlcXVlc3QgfSBmcm9tIFwiZXhwcmVzc1wiO1xuaW1wb3J0IHsgRGVjYWZDb250cm9sbGVyIGFzIEh0dHBEZWNhZkNvbnRyb2xsZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuaW1wb3J0IHtcbiAgQ29udGV4dCxcbiAgTW9kZWxTZXJ2aWNlLFxuICBSZXBvLFxuICBSZXBvc2l0b3J5LFxuICBTZXJ2aWNlLFxufSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IE1vZGVsLCB0eXBlIE1vZGVsQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5cbmltcG9ydCB7IERlY2FmU2VydmVyQ3R4IH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4vcmVxdWVzdC9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBEZWNhZkNvbnRyb2xsZXI8XG4gIENPTlRFWFQgZXh0ZW5kcyBEZWNhZlNlcnZlckN0eCA9IERlY2FmU2VydmVyQ3R4LFxuPiBleHRlbmRzIEh0dHBEZWNhZkNvbnRyb2xsZXI8UmVxdWVzdCwgYW55LCBEZWNhZlJlcXVlc3RDb250ZXh0PiB7XG4gIHByb3RlY3RlZCBjb25zdHJ1Y3RvcihcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgY2xpZW50Q29udGV4dDogRGVjYWZSZXF1ZXN0Q29udGV4dCxcbiAgICBfbmFtZTogc3RyaW5nXG4gICkge1xuICAgIHN1cGVyKGNsaWVudENvbnRleHQpO1xuICB9XG59XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBEZWNhZk1vZGVsQ29udHJvbGxlcjxcbiAgTSBleHRlbmRzIE1vZGVsPGJvb2xlYW4+LFxuICBDIGV4dGVuZHMgRGVjYWZTZXJ2ZXJDdHggPSBEZWNhZlNlcnZlckN0eCxcbj4gZXh0ZW5kcyBEZWNhZkNvbnRyb2xsZXI8Qz4ge1xuICBwcml2YXRlIF9wZXJzaXN0ZW5jZT86IFJlcG88TT4gfCBNb2RlbFNlcnZpY2U8TT47XG5cbiAgcHJvdGVjdGVkIGNvbnN0cnVjdG9yKFxuICAgIHByb3RlY3RlZCBvdmVycmlkZSByZWFkb25seSBjbGllbnRDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIF9uYW1lOiBzdHJpbmdcbiAgKSB7XG4gICAgc3VwZXIoY2xpZW50Q29udGV4dCwgX25hbWUpO1xuICB9XG5cbiAgYWJzdHJhY3QgZ2V0IGNsYXNzKCk6IE1vZGVsQ29uc3RydWN0b3I8TT47XG5cbiAgcGVyc2lzdGVuY2UoY3R4PzogQ29udGV4dDxhbnk+KTogUmVwbzxNPiB8IE1vZGVsU2VydmljZTxNPiB7XG4gICAgaWYgKCF0aGlzLl9wZXJzaXN0ZW5jZSlcbiAgICAgIHRyeSB7XG4gICAgICAgIHRoaXMuX3BlcnNpc3RlbmNlID0gU2VydmljZS5nZXQ8TW9kZWxTZXJ2aWNlPE0+Pih0aGlzLmNsYXNzKTtcbiAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIHRoaXMuX3BlcnNpc3RlbmNlID0gTW9kZWxTZXJ2aWNlLmdldFNlcnZpY2UoXG4gICAgICAgICAgICB0aGlzLmNsYXNzXG4gICAgICAgICAgKSBhcyBNb2RlbFNlcnZpY2U8TT47XG4gICAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgICAgdGhpcy5fcGVyc2lzdGVuY2UgPSBSZXBvc2l0b3J5LmZvck1vZGVsKHRoaXMuY2xhc3MpIGFzIFJlcG88TT47XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgIGNvbnN0IG92ZXJyaWRlcyA9IHRoaXMuY2xpZW50Q29udGV4dC50b092ZXJyaWRlcygpO1xuXG4gICAgcmV0dXJuIGN0eFxuICAgICAgPyB0aGlzLl9wZXJzaXN0ZW5jZSBpbnN0YW5jZW9mIFJlcG9zaXRvcnlcbiAgICAgICAgPyB0aGlzLl9wZXJzaXN0ZW5jZS5vdmVycmlkZShvdmVycmlkZXMpXG4gICAgICAgIDogdGhpcy5fcGVyc2lzdGVuY2UuZm9yKG92ZXJyaWRlcylcbiAgICAgIDogdGhpcy5fcGVyc2lzdGVuY2U7XG4gIH1cbn1cbiIsImltcG9ydCB7XG4gIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cyxcbiAgSW50ZXJuYWxFcnJvcixcbiAgT3BlcmF0aW9uS2V5cyxcbn0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciwgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IEFwaVByb3BlcnR5IH0gZnJvbSBcIi4uLy4uL292ZXJyaWRlcy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBNb2RlbCwgVmFsaWRhdGlvbktleXMgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBUcmFuc2FjdGlvbk9wZXJhdGlvbktleXMgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IHRvUGFzY2FsQ2FzZSB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuaW1wb3J0IHsgREVDT1JBVE9SUyB9IGZyb20gXCIuLi8uLi9vdmVycmlkZXMvY29uc3RhbnRzXCI7XG5cbmNvbnN0IGR0b0NhY2hlID0gbmV3IE1hcDxcbiAgT3BlcmF0aW9uS2V5cyxcbiAgV2Vha01hcDxDb25zdHJ1Y3Rvcjxhbnk+LCBDb25zdHJ1Y3Rvcjxhbnk+PlxuPigpO1xuXG4vKipcbiAqIEJ1aWxkcyBhIE5lc3QvU3dhZ2dlciBEVE8gY2xhc3MgZm9yIHRoZSBnaXZlbiBtb2RlbCBhbmQgQ1JVRCBvcGVyYXRpb24uXG4gKlxuICogUnVsZXM6XG4gKiAgLSBPbmx5IENSRUFURSBhbmQgVVBEQVRFIChhbmQgdGhlaXIgYnVsayB2YXJpYW50cykgcHJvZHVjZSBhIERUTztcbiAqICAgIGFsbCBvdGhlciBvcGVyYXRpb25zIHJldHVybiB0aGUgb3JpZ2luYWwgbW9kZWwgY2xhc3MgdW5jaGFuZ2VkLlxuICogIC0gQGdlbmVyYXRlZCgpIHByb3BlcnRpZXMgKGNyZWF0ZWRBdCwgdXBkYXRlZEF0LCBjcmVhdGVkQnksIHVwZGF0ZWRCeSxcbiAqICAgIHV1aWQsIHZlcnNpb24sIEBjb21wb3NlZCBwa3MsIOKApikgYXJlICoqbmV2ZXIqKiBleHBvc2VkIGluIGFueSBEVE8uXG4gKiAgLSBUaGUgQHBrKCkgcHJvcGVydHk6XG4gKiAgICAgIOKAoiBVUERBVEUg4oCTIGFsd2F5cyBpbmNsdWRlZC5cbiAqICAgICAg4oCiIENSRUFURSAg4oCTIGluY2x1ZGVkIG9ubHkgd2hlbiB0aGUgcGsgaXMgTk9UIGF1dG8tZ2VuZXJhdGVkXG4gKiAgICAgICAgICAgICAgICAgIChjaGVja2VkIHZpYSBNb2RlbC5wa1Byb3BzKCkuZ2VuZXJhdGVkIEFORCBNb2RlbC5nZW5lcmF0ZWQoKSkuXG4gKiAgLSBSZWxhdGlvbiBwcm9wZXJ0aWVzIChAb25lVG9PbmUsIEBvbmVUb01hbnksIOKApik6XG4gKiAgICAgIOKAoiBDUkVBVEUgIOKAkyBuZXN0ZWQgYXMgRHRvRm9yKENSRUFURSwgUmVsYXRlZE1vZGVsKS4gV2hlbiB0aGUgcmVsYXRpb25cbiAqICAgICAgICAgICAgICAgICAgY3JlYXRlcyBhIGNpcmN1bGFyIHJlZmVyZW5jZSAobW9kZWwgQSDihpIgbW9kZWwgQiDihpIgbW9kZWwgQSksXG4gKiAgICAgICAgICAgICAgICAgIHRoZSBiYWNrLXJlZmVyZW5jZSB1c2VzIHRoZSByZWxhdGVkIG1vZGVsJ3MgUEsgdHlwZVxuICogICAgICAgICAgICAgICAgICAoc3RyaW5nL2ludGVnZXIpIGluc3RlYWQgb2YgdGhlIGZ1bGwgRFRPIHRvIGJyZWFrIHRoZSBjeWNsZS5cbiAqICAgICAg4oCiIFVQREFURSAg4oCTIHVuaW9uIG9mIER0b0ZvcihVUERBVEUsIFJlbGF0ZWRNb2RlbCkgKipvcioqIHRoZVxuICogICAgICAgICAgICAgICAgICByZWxhdGVkIG1vZGVsJ3MgcHJpbWFyeS1rZXkgdHlwZSAoc3RyaW5nIC8gaW50ZWdlciksXG4gKiAgICAgICAgICAgICAgICAgIGV4cHJlc3NlZCBhcyBhIFN3YWdnZXIgb25lT2YuIENpcmN1bGFyIGJhY2stcmVmZXJlbmNlc1xuICogICAgICAgICAgICAgICAgICBvbWl0IHRoZSBEVE8gJHJlZiBhbmQgb25seSBpbmNsdWRlIHRoZSBQSyB0eXBlLlxuICpcbiAqIE1ldGFkYXRhLnByb3BlcnRpZXMoKSBub3cgcmV0dXJucyBBTEwgcHJvcGVydGllcyBhY3Jvc3MgdGhlIHByb3RvdHlwZSBjaGFpbixcbiAqIHNvIERUTyBpbmhlcml0YW5jZSBpcyBubyBsb25nZXIgbmVlZGVkOyBldmVyeSBEVE8gaXMgYSBmbGF0IGNsYXNzLlxuICovXG5leHBvcnQgZnVuY3Rpb24gRHRvRm9yPE0gZXh0ZW5kcyBNb2RlbD4oXG4gIG9wOiBPcGVyYXRpb25LZXlzLFxuICBtb2RlbDogQ29uc3RydWN0b3I8TT4sXG4gIHN0YWNrOiBTZXQ8Q29uc3RydWN0b3I8YW55Pj4gPSBuZXcgU2V0KClcbik6IENvbnN0cnVjdG9yPGFueT4ge1xuICBpZiAoIVRyYW5zYWN0aW9uT3BlcmF0aW9uS2V5cy5pbmNsdWRlcyhvcCkpIHtcbiAgICByZXR1cm4gbW9kZWw7XG4gIH1cblxuICBjb25zdCBjYWNoZSA9IGdldER0b0NhY2hlKG9wKTtcbiAgY29uc3QgY2FjaGVkID0gY2FjaGUuZ2V0KG1vZGVsKTtcbiAgaWYgKGNhY2hlZCkgcmV0dXJuIGNhY2hlZDtcblxuICBjb25zdCBpc1VwZGF0ZU9wID0gW1xuICAgIE9wZXJhdGlvbktleXMuVVBEQVRFLFxuICAgIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cy5VUERBVEVfQUxMLFxuICBdLmluY2x1ZGVzKG9wIGFzIGFueSk7XG5cbiAgY2xhc3MgRHluYW1pY0RUTyB7fVxuICBjYWNoZS5zZXQobW9kZWwsIER5bmFtaWNEVE8pO1xuXG4gIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShEeW5hbWljRFRPLCBcIm5hbWVcIiwge1xuICAgIHZhbHVlOiBgJHt0b1Bhc2NhbENhc2UobW9kZWwubmFtZSl9JHt0b1Bhc2NhbENhc2Uob3ApfURUT2AsXG4gIH0pO1xuXG4gIGNvbnN0IHBrUHJvcCA9ICgoKSA9PiB7XG4gICAgdHJ5IHtcbiAgICAgIHJldHVybiBNb2RlbC5wayhtb2RlbCk7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm4gdW5kZWZpbmVkO1xuICAgIH1cbiAgfSkoKTtcblxuICBjb25zdCBwa1Byb3BzTWV0YWRhdGEgPSBwa1Byb3AgPyBNb2RlbC5wa1Byb3BzKG1vZGVsIGFzIGFueSkgOiB1bmRlZmluZWQ7XG4gIGNvbnN0IHBrRGVzaWduVHlwZSA9IHBrUHJvcFxuICAgID8gUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIG1vZGVsLnByb3RvdHlwZSwgcGtQcm9wIGFzIHN0cmluZylcbiAgICA6IHVuZGVmaW5lZDtcbiAgY29uc3QgcGtUeXBlSXNOdW1lcmljID1cbiAgICBwa0Rlc2lnblR5cGUgPT09IE51bWJlciB8fCBwa0Rlc2lnblR5cGUgPT09IEJpZ0ludDtcbiAgY29uc3QgcGtJc0dlbmVyYXRlZCA9XG4gICAgISFwa1Byb3BzTWV0YWRhdGE/LmdlbmVyYXRlZCB8fFxuICAgIChwa1Byb3BcbiAgICAgID8gTW9kZWwuZ2VuZXJhdGVkQnlTZXF1ZW5jZShtb2RlbCBhcyBhbnksIHBrUHJvcCBhcyBhbnkpIHx8XG4gICAgICAgIGlzUHJvcGVydHlHZW5lcmF0ZWRBY3Jvc3NJbmhlcml0YW5jZShtb2RlbCwgcGtQcm9wIGFzIHN0cmluZykgfHxcbiAgICAgICAgcGtUeXBlSXNOdW1lcmljXG4gICAgICA6IGZhbHNlKTtcblxuICBjb25zdCBhbGxQcm9wcyA9IEFycmF5LmZyb20obmV3IFNldChNZXRhZGF0YS5wcm9wZXJ0aWVzKG1vZGVsKSB8fCBbXSkpO1xuICBjb25zdCByZWxhdGlvbnMgPSBuZXcgU2V0PHN0cmluZz4oKE1vZGVsLnJlbGF0aW9ucyhtb2RlbCkgYXMgc3RyaW5nW10pIHx8IFtdKTtcbiAgY29uc3Qgc2NhbGFyUHJvcHM6IHN0cmluZ1tdID0gW107XG5cbiAgZm9yIChjb25zdCBwcm9wIG9mIGFsbFByb3BzKSB7XG4gICAgaWYgKCFwcm9wKSBjb250aW51ZTtcbiAgICBpZiAocmVsYXRpb25zLmhhcyhwcm9wKSkgY29udGludWU7XG5cbiAgICBpZiAocHJvcCA9PT0gcGtQcm9wICYmICFpc1VwZGF0ZU9wICYmIHBrSXNHZW5lcmF0ZWQpIGNvbnRpbnVlO1xuICAgIGlmIChwcm9wICE9PSBwa1Byb3AgJiYgaXNQcm9wZXJ0eUdlbmVyYXRlZEFjcm9zc0luaGVyaXRhbmNlKG1vZGVsLCBwcm9wKSlcbiAgICAgIGNvbnRpbnVlO1xuXG4gICAgc2NhbGFyUHJvcHMucHVzaChwcm9wKTtcbiAgfVxuXG4gIGZvciAoY29uc3QgcHJvcCBvZiBzY2FsYXJQcm9wcykge1xuICAgIGNvbnN0IHZhbGlkYXRpb24gPSBnZXRWYWxpZGF0aW9uQWNyb3NzSW5oZXJpdGFuY2UobW9kZWwsIHByb3ApO1xuICAgIGNvbnN0IGlzUmVxdWlyZWQgPSAhIXZhbGlkYXRpb24/LltWYWxpZGF0aW9uS2V5cy5SRVFVSVJFRF07XG4gICAgY29uc3QgdHlwZUhpbnQgPVxuICAgICAgZ2V0VHlwZUFjcm9zc0luaGVyaXRhbmNlKG1vZGVsLCBwcm9wKSA/P1xuICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIG1vZGVsLnByb3RvdHlwZSwgcHJvcCk7XG5cbiAgICBjb25zdCBhcGlPcHRpb25zOiBQYXJhbWV0ZXJzPHR5cGVvZiBBcGlQcm9wZXJ0eT5bMF0gPSB7XG4gICAgICByZXF1aXJlZDogaXNSZXF1aXJlZCxcbiAgICB9O1xuICAgIGlmICh0eXBlSGludCkgYXBpT3B0aW9ucy50eXBlID0gdHlwZUhpbnQ7XG5cbiAgICBBcGlQcm9wZXJ0eShhcGlPcHRpb25zKShEeW5hbWljRFRPLnByb3RvdHlwZSwgcHJvcCk7XG5cbiAgICBjb25zdCBkZXNpZ25UeXBlID1cbiAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoXCJkZXNpZ246dHlwZVwiLCBtb2RlbC5wcm90b3R5cGUsIHByb3ApID8/IHR5cGVIaW50O1xuICAgIGlmICh0eXBlb2YgZGVzaWduVHlwZSAhPT0gXCJ1bmRlZmluZWRcIikge1xuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgXCJkZXNpZ246dHlwZVwiLFxuICAgICAgICBkZXNpZ25UeXBlLFxuICAgICAgICBEeW5hbWljRFRPLnByb3RvdHlwZSxcbiAgICAgICAgcHJvcFxuICAgICAgKTtcbiAgICB9XG5cbiAgICBPYmplY3QuZGVmaW5lUHJvcGVydHkoRHluYW1pY0RUTy5wcm90b3R5cGUsIHByb3AsIHtcbiAgICAgIHZhbHVlOiB1bmRlZmluZWQsXG4gICAgICB3cml0YWJsZTogdHJ1ZSxcbiAgICAgIGVudW1lcmFibGU6IHRydWUsXG4gICAgICBjb25maWd1cmFibGU6IHRydWUsXG4gICAgfSk7XG4gIH1cblxuICBmb3IgKGNvbnN0IHJlbGF0aW9uIG9mIHJlbGF0aW9ucykge1xuICAgIGNvbnN0IHJlbGF0aW9uTWV0YSA9IE1ldGFkYXRhLnJlbGF0aW9ucyhtb2RlbCwgcmVsYXRpb24gYXMgYW55KTtcbiAgICBpZiAoIXJlbGF0aW9uTWV0YSkge1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoYE1ldGFkYXRhIGZvciByZWxhdGlvbiAke3JlbGF0aW9ufSBub3QgZm91bmRgKTtcbiAgICB9XG5cbiAgICBsZXQgcmVsYXRpb25UeXBlOiBDb25zdHJ1Y3Rvcjxhbnk+IHwgdW5kZWZpbmVkID1cbiAgICAgIHJlbGF0aW9uTWV0YS5jbGFzcyBhcyBDb25zdHJ1Y3Rvcjxhbnk+O1xuICAgIGlmICh0eXBlb2YgcmVsYXRpb25UeXBlID09PSBcImZ1bmN0aW9uXCIgJiYgIXJlbGF0aW9uVHlwZS5uYW1lKSB7XG4gICAgICByZWxhdGlvblR5cGUgPSAocmVsYXRpb25UeXBlIGFzIGFueSkoKTtcbiAgICB9XG4gICAgaWYgKCFyZWxhdGlvblR5cGUgfHwgdHlwZW9mIHJlbGF0aW9uVHlwZSAhPT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihgVHlwZSBmb3IgcmVsYXRpb24gJHtyZWxhdGlvbn0gbm90IGZvdW5kYCk7XG4gICAgfVxuICAgIGlmICghTW9kZWwuZ2V0KHJlbGF0aW9uVHlwZS5uYW1lKSkge1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgY29uc3QgbWV0YSA9IE1ldGFkYXRhLnZhbGlkYXRpb25Gb3IobW9kZWwsIHJlbGF0aW9uIGFzIGFueSk7XG4gICAgY29uc3QgaXNBcnJheSA9ICEhKG1ldGEgYXMgYW55KT8uW1ZhbGlkYXRpb25LZXlzLkxJU1RdO1xuICAgIGNvbnN0IGlzUmVxdWlyZWQgPSAhIShtZXRhIGFzIGFueSk/LltWYWxpZGF0aW9uS2V5cy5SRVFVSVJFRF07XG5cbiAgICAvLyBEZXRlY3QgY2lyY3VsYXIgcmVmZXJlbmNlOiBpZiB0aGUgcmVsYXRlZCBtb2RlbCBpcyBhbHJlYWR5IGluIHRoZVxuICAgIC8vIHJlY3Vyc2lvbiBzdGFjaywgdXNlIHRoZSBQSyB0eXBlIGluc3RlYWQgb2YgdGhlIGZ1bGwgRFRPIHRvIGJyZWFrXG4gICAgLy8gdGhlIGN5Y2xlLiBUaGlzIHByZXZlbnRzIEBuZXN0anMvc3dhZ2dlcidzIFNjaGVtYU9iamVjdEZhY3RvcnkgZnJvbVxuICAgIC8vIGluZmluaXRlbHkgcmVjdXJzaW5nIHdoZW4gZXhwbG9yaW5nIHRoZSBzY2hlbWFzLlxuICAgIGNvbnN0IGlzQ2lyY3VsYXIgPSBzdGFjay5oYXMocmVsYXRpb25UeXBlKTtcblxuICAgIGlmIChpc0NpcmN1bGFyKSB7XG4gICAgICBjb25zdCBwa1R5cGVOYW1lID0gZ2V0UGtPcGVuQXBpVHlwZShyZWxhdGlvblR5cGUpO1xuICAgICAgYWRkUmVsYXRpb25Qa1JlZihEeW5hbWljRFRPLCByZWxhdGlvbiwgcGtUeXBlTmFtZSwgaXNBcnJheSwgaXNSZXF1aXJlZCk7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBjb25zdCByZWxhdGlvbkR0byA9IER0b0ZvcihvcCwgcmVsYXRpb25UeXBlLCBuZXcgU2V0KHN0YWNrKS5hZGQobW9kZWwpKTtcblxuICAgIGlmIChpc1VwZGF0ZU9wKSB7XG4gICAgICBhZGRSZWxhdGlvblVwZGF0ZShcbiAgICAgICAgRHluYW1pY0RUTyxcbiAgICAgICAgcmVsYXRpb24sXG4gICAgICAgIHJlbGF0aW9uVHlwZSxcbiAgICAgICAgcmVsYXRpb25EdG8sXG4gICAgICAgIGlzQXJyYXksXG4gICAgICAgIGlzUmVxdWlyZWRcbiAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGFkZFJlbGF0aW9uKER5bmFtaWNEVE8sIHJlbGF0aW9uLCByZWxhdGlvbkR0bywgaXNBcnJheSwgaXNSZXF1aXJlZCk7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIER5bmFtaWNEVE87XG59XG5cbmZ1bmN0aW9uIGFkZFJlbGF0aW9uKFxuICBEdG9DbGFzczogYW55LFxuICByZWxhdGlvbjogc3RyaW5nLFxuICByZWxhdGlvbkR0bzogYW55LFxuICBpc0FycmF5OiBib29sZWFuLFxuICBpc1JlcXVpcmVkOiBib29sZWFuXG4pOiB2b2lkIHtcbiAgY29uc3QgYXBpT3B0aW9uczogUGFyYW1ldGVyczx0eXBlb2YgQXBpUHJvcGVydHk+WzBdID0ge1xuICAgIHR5cGU6IHJlbGF0aW9uRHRvLFxuICAgIHJlcXVpcmVkOiBpc1JlcXVpcmVkLFxuICAgIGlzQXJyYXksXG4gIH07XG4gIEFwaVByb3BlcnR5KGFwaU9wdGlvbnMpKER0b0NsYXNzLnByb3RvdHlwZSwgcmVsYXRpb24pO1xuICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgIFwiZGVzaWduOnR5cGVcIixcbiAgICBpc0FycmF5ID8gQXJyYXkgOiByZWxhdGlvbkR0byxcbiAgICBEdG9DbGFzcy5wcm90b3R5cGUsXG4gICAgcmVsYXRpb25cbiAgKTtcbiAgT2JqZWN0LmRlZmluZVByb3BlcnR5KER0b0NsYXNzLnByb3RvdHlwZSwgcmVsYXRpb24sIHtcbiAgICB2YWx1ZTogdW5kZWZpbmVkLFxuICAgIHdyaXRhYmxlOiB0cnVlLFxuICAgIGVudW1lcmFibGU6IHRydWUsXG4gICAgY29uZmlndXJhYmxlOiB0cnVlLFxuICB9KTtcbn1cblxuLyoqXG4gKiBBZGRzIGEgcmVsYXRpb24gcHJvcGVydHkgdGhhdCB1c2VzIHRoZSBQSyB0eXBlIChzdHJpbmcvaW50ZWdlcikgaW5zdGVhZCBvZlxuICogYSBmdWxsIERUTyBjbGFzcy4gVXNlZCB0byBicmVhayBjaXJjdWxhciByZWZlcmVuY2VzIChlLmcuIEN1c3RvbWVyIOKGkiBPcmRlciDihpIgQ3VzdG9tZXIpLlxuICovXG5mdW5jdGlvbiBhZGRSZWxhdGlvblBrUmVmKFxuICBEdG9DbGFzczogYW55LFxuICByZWxhdGlvbjogc3RyaW5nLFxuICBwa1R5cGVOYW1lOiBzdHJpbmcsXG4gIGlzQXJyYXk6IGJvb2xlYW4sXG4gIGlzUmVxdWlyZWQ6IGJvb2xlYW5cbik6IHZvaWQge1xuICBjb25zdCBhcGlPcHRpb25zOiBQYXJhbWV0ZXJzPHR5cGVvZiBBcGlQcm9wZXJ0eT5bMF0gPSBpc0FycmF5XG4gICAgPyAoeyB0eXBlOiBcImFycmF5XCIsIGl0ZW1zOiB7IHR5cGU6IHBrVHlwZU5hbWUgfSwgcmVxdWlyZWQ6IGlzUmVxdWlyZWQgfSBhcyBhbnkpXG4gICAgOiAoeyB0eXBlOiBwa1R5cGVOYW1lID09PSBcImludGVnZXJcIiA/IE51bWJlciA6IFN0cmluZywgcmVxdWlyZWQ6IGlzUmVxdWlyZWQgfSBhcyBhbnkpO1xuICBBcGlQcm9wZXJ0eShhcGlPcHRpb25zKShEdG9DbGFzcy5wcm90b3R5cGUsIHJlbGF0aW9uKTtcbiAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICBcImRlc2lnbjp0eXBlXCIsXG4gICAgaXNBcnJheSA/IEFycmF5IDogKHBrVHlwZU5hbWUgPT09IFwiaW50ZWdlclwiID8gTnVtYmVyIDogU3RyaW5nKSxcbiAgICBEdG9DbGFzcy5wcm90b3R5cGUsXG4gICAgcmVsYXRpb25cbiAgKTtcbiAgT2JqZWN0LmRlZmluZVByb3BlcnR5KER0b0NsYXNzLnByb3RvdHlwZSwgcmVsYXRpb24sIHtcbiAgICB2YWx1ZTogdW5kZWZpbmVkLFxuICAgIHdyaXRhYmxlOiB0cnVlLFxuICAgIGVudW1lcmFibGU6IHRydWUsXG4gICAgY29uZmlndXJhYmxlOiB0cnVlLFxuICB9KTtcbn1cblxuZnVuY3Rpb24gYWRkUmVsYXRpb25VcGRhdGUoXG4gIER0b0NsYXNzOiBhbnksXG4gIHJlbGF0aW9uOiBzdHJpbmcsXG4gIHJlbGF0aW9uVHlwZTogQ29uc3RydWN0b3I8YW55PixcbiAgcmVsYXRpb25EdG86IGFueSxcbiAgaXNBcnJheTogYm9vbGVhbixcbiAgaXNSZXF1aXJlZDogYm9vbGVhblxuKTogdm9pZCB7XG4gIGNvbnN0IGV4dHJhTW9kZWxzID1cbiAgICBSZWZsZWN0LmdldE1ldGFkYXRhKERFQ09SQVRPUlMuQVBJX0VYVFJBX01PREVMUywgRHRvQ2xhc3MpIHx8IFtdO1xuICBpZiAoIWV4dHJhTW9kZWxzLmluY2x1ZGVzKHJlbGF0aW9uRHRvKSkge1xuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICBERUNPUkFUT1JTLkFQSV9FWFRSQV9NT0RFTFMsXG4gICAgICBbLi4uZXh0cmFNb2RlbHMsIHJlbGF0aW9uRHRvXSxcbiAgICAgIER0b0NsYXNzXG4gICAgKTtcbiAgfVxuXG4gIGNvbnN0IGR0b1JlZiA9IGAjL2NvbXBvbmVudHMvc2NoZW1hcy8ke3JlbGF0aW9uRHRvLm5hbWV9YDtcbiAgY29uc3QgcGtUeXBlTmFtZSA9IGdldFBrT3BlbkFwaVR5cGUocmVsYXRpb25UeXBlKTtcbiAgY29uc3Qgb25lT2ZJdGVtcyA9IFt7ICRyZWY6IGR0b1JlZiB9LCB7IHR5cGU6IHBrVHlwZU5hbWUgfV07XG5cbiAgY29uc3QgYXBpT3B0aW9uczogUGFyYW1ldGVyczx0eXBlb2YgQXBpUHJvcGVydHk+WzBdID0gaXNBcnJheVxuICAgID8gKHsgdHlwZTogXCJhcnJheVwiLCByZXF1aXJlZDogaXNSZXF1aXJlZCwgb25lT2Y6IG9uZU9mSXRlbXMgfSBhcyBhbnkpXG4gICAgOiAoeyB0eXBlOiBPYmplY3QsIHJlcXVpcmVkOiBpc1JlcXVpcmVkLCBvbmVPZjogb25lT2ZJdGVtcyB9IGFzIGFueSk7XG5cbiAgQXBpUHJvcGVydHkoYXBpT3B0aW9ucykoRHRvQ2xhc3MucHJvdG90eXBlLCByZWxhdGlvbik7XG4gIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgXCJkZXNpZ246dHlwZVwiLFxuICAgIGlzQXJyYXkgPyBBcnJheSA6IE9iamVjdCxcbiAgICBEdG9DbGFzcy5wcm90b3R5cGUsXG4gICAgcmVsYXRpb25cbiAgKTtcbiAgT2JqZWN0LmRlZmluZVByb3BlcnR5KER0b0NsYXNzLnByb3RvdHlwZSwgcmVsYXRpb24sIHtcbiAgICB2YWx1ZTogdW5kZWZpbmVkLFxuICAgIHdyaXRhYmxlOiB0cnVlLFxuICAgIGVudW1lcmFibGU6IHRydWUsXG4gICAgY29uZmlndXJhYmxlOiB0cnVlLFxuICB9KTtcbn1cblxuZnVuY3Rpb24gZ2V0UGtPcGVuQXBpVHlwZShyZWxhdGlvblR5cGU6IENvbnN0cnVjdG9yPGFueT4pOiBzdHJpbmcge1xuICB0cnkge1xuICAgIGNvbnN0IHBrUHJvcHNNZXRhZGF0YSA9IE1vZGVsLnBrUHJvcHMocmVsYXRpb25UeXBlIGFzIGFueSk7XG4gICAgY29uc3QgcGtUeXBlID0gcGtQcm9wc01ldGFkYXRhPy50eXBlO1xuICAgIGlmIChwa1R5cGUgPT09IE51bWJlciB8fCBwa1R5cGUgPT09IEJpZ0ludCkgcmV0dXJuIFwiaW50ZWdlclwiO1xuICAgIHJldHVybiBcInN0cmluZ1wiO1xuICB9IGNhdGNoIHtcbiAgICByZXR1cm4gXCJzdHJpbmdcIjtcbiAgfVxufVxuXG5mdW5jdGlvbiBpc1Byb3BlcnR5R2VuZXJhdGVkQWNyb3NzSW5oZXJpdGFuY2UoXG4gIG1vZGVsOiBDb25zdHJ1Y3Rvcjxhbnk+LFxuICBwcm9wOiBzdHJpbmdcbik6IGJvb2xlYW4ge1xuICBsZXQgY3VycmVudDogYW55ID0gbW9kZWw7XG4gIHdoaWxlIChjdXJyZW50ICYmIGN1cnJlbnQgIT09IE9iamVjdCAmJiBjdXJyZW50ICE9PSBGdW5jdGlvbikge1xuICAgIGlmIChNb2RlbC5nZW5lcmF0ZWQoY3VycmVudCwgcHJvcCBhcyBhbnkpKSByZXR1cm4gdHJ1ZTtcbiAgICBjdXJyZW50ID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKGN1cnJlbnQpO1xuICB9XG4gIHJldHVybiBmYWxzZTtcbn1cblxuZnVuY3Rpb24gZ2V0VmFsaWRhdGlvbkFjcm9zc0luaGVyaXRhbmNlKFxuICBtb2RlbDogQ29uc3RydWN0b3I8YW55PixcbiAgcHJvcDogc3RyaW5nXG4pOiBSZWNvcmQ8c3RyaW5nLCBhbnk+IHwgdW5kZWZpbmVkIHtcbiAgbGV0IGN1cnJlbnQ6IGFueSA9IG1vZGVsO1xuICB3aGlsZSAoY3VycmVudCAmJiBjdXJyZW50ICE9PSBPYmplY3QgJiYgY3VycmVudCAhPT0gRnVuY3Rpb24pIHtcbiAgICBjb25zdCB2YWxpZGF0aW9uID0gTWV0YWRhdGEudmFsaWRhdGlvbkZvcihjdXJyZW50LCBwcm9wIGFzIGFueSk7XG4gICAgaWYgKHZhbGlkYXRpb24pIHJldHVybiB2YWxpZGF0aW9uIGFzIGFueTtcbiAgICBjdXJyZW50ID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKGN1cnJlbnQpO1xuICB9XG4gIHJldHVybiB1bmRlZmluZWQ7XG59XG5cbmZ1bmN0aW9uIGdldFR5cGVBY3Jvc3NJbmhlcml0YW5jZShtb2RlbDogQ29uc3RydWN0b3I8YW55PiwgcHJvcDogc3RyaW5nKTogYW55IHtcbiAgbGV0IGN1cnJlbnQ6IGFueSA9IG1vZGVsO1xuICB3aGlsZSAoY3VycmVudCAmJiBjdXJyZW50ICE9PSBPYmplY3QgJiYgY3VycmVudCAhPT0gRnVuY3Rpb24pIHtcbiAgICBjb25zdCB0eXBlID0gTWV0YWRhdGEudHlwZShjdXJyZW50LCBwcm9wKTtcbiAgICBpZiAodHlwZSkgcmV0dXJuIHR5cGU7XG4gICAgY3VycmVudCA9IE9iamVjdC5nZXRQcm90b3R5cGVPZihjdXJyZW50KTtcbiAgfVxuICByZXR1cm4gdW5kZWZpbmVkO1xufVxuXG5mdW5jdGlvbiBnZXREdG9DYWNoZShcbiAgb3A6IE9wZXJhdGlvbktleXNcbik6IFdlYWtNYXA8Q29uc3RydWN0b3I8YW55PiwgQ29uc3RydWN0b3I8YW55Pj4ge1xuICBpZiAoIWR0b0NhY2hlLmhhcyhvcCkpIHtcbiAgICBkdG9DYWNoZS5zZXQob3AsIG5ldyBXZWFrTWFwKCkpO1xuICB9XG4gIHJldHVybiBkdG9DYWNoZS5nZXQob3ApITtcbn1cbiIsImltcG9ydCB7IENvbnRyb2xsZXIsIFBhcmFtLCBRdWVyeSwgUmVzcG9uc2UsIFNldE1ldGFkYXRhIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQge1xuICBBcGlCYWRSZXF1ZXN0UmVzcG9uc2UsXG4gIEFwaUJvZHksXG4gIEFwaUNyZWF0ZWRSZXNwb25zZSxcbiAgQXBpRXh0cmFNb2RlbHMsXG4gIEFwaU5vQ29udGVudFJlc3BvbnNlLFxuICBBcGlOb3RGb3VuZFJlc3BvbnNlLFxuICBBcGlPa1Jlc3BvbnNlLFxuICBBcGlPcGVyYXRpb24sXG4gIEFwaVBhcmFtLFxuICBBcGlRdWVyeSxcbiAgQXBpVGFncyxcbiAgQXBpVW5wcm9jZXNzYWJsZUVudGl0eVJlc3BvbnNlLFxuICBnZXRTY2hlbWFQYXRoLFxufSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyXCI7XG5pbXBvcnQge1xuICB0eXBlIERpcmVjdGlvbkxpbWl0T2Zmc2V0LFxuICBNb2RlbFNlcnZpY2UsXG4gIE9yZGVyRGlyZWN0aW9uLFxuICBQZXJzaXN0ZW5jZUtleXMsXG4gIFByZXBhcmVkU3RhdGVtZW50S2V5cyxcbiAgdHlwZSBSZXBvLFxuICBSZXBvc2l0b3J5LFxuICBTZXJ2aWNlLFxufSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IE1vZGVsLCBNb2RlbENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgTG9nZ2luZywgdG9LZWJhYkNhc2UgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7XG4gIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cyxcbiAgREJLZXlzLFxuICBPcGVyYXRpb25LZXlzLFxuICBWYWxpZGF0aW9uRXJyb3IsXG59IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IsIE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQge1xuICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwsXG4gIEFwaVBhcmFtc0Zyb21Nb2RlbCxcbiAgdHlwZSBEZWNhZkFwaVByb3BlcnR5LFxuICBEZWNhZkJvZHksXG4gIERlY2FmUGFyYW1zLFxuICBEZWNhZlF1ZXJ5LFxufSBmcm9tIFwiLi9kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBIdHRwVmVyYlRvRGVjb3JhdG9yIH0gZnJvbSBcIi4vZGVjb3JhdG9ycy91dGlsc1wiO1xuaW1wb3J0IHR5cGUgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi9kZWNvcmF0b3JzL3R5cGVzXCI7XG5pbXBvcnQgeyBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4uL3JlcXVlc3RcIjtcbmltcG9ydCB7IERFQ0FGX0NPTlRST0xMRVJfQ09ORklHLCBERUNBRl9ST1VURSB9IGZyb20gXCIuLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IFNLSVBfTU9ERUxfUk9MRVNfS0VZIH0gZnJvbSBcIi4uL2F1dGgvY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBBdXRoLCBQdWJsaWMsIFJlcXVpcmVSb2xlcyB9IGZyb20gXCIuLi9hdXRoL2RlY29yYXRvcnNcIjtcbmltcG9ydCB7IENvbnRyb2xsZXJDb25zdHJ1Y3RvciB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQgeyBEZWNhZk1vZGVsQ29udHJvbGxlciB9IGZyb20gXCIuLi9jb250cm9sbGVyc1wiO1xuaW1wb3J0IHsgRHRvRm9yIH0gZnJvbSBcIi4uL2ZhY3Rvcnkvb3BlbmFwaS9EdG9CdWlsZGVyXCI7XG5pbXBvcnQgXCIuLi9vdmVycmlkZXNcIjtcbmltcG9ydCB7XG4gIE1vZGVsQ29udHJvbGxlckZhY3RvcnksXG4gIHR5cGUgTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZyxcbiAgdHlwZSBBdXRoQ29uZmlnLFxuICB0eXBlIFNlcnZlclJvdXRlLFxufSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuXG5leHBvcnQgY2xhc3MgRnJvbU1vZGVsQ29udHJvbGxlciB7XG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IGxvZyA9IExvZ2dpbmcuZm9yKEZyb21Nb2RlbENvbnRyb2xsZXIubmFtZSk7XG5cbiAgc3RhdGljIGdldFBlcnNpc3RlbmNlPFQgZXh0ZW5kcyBNb2RlbDxib29sZWFuPj4oXG4gICAgTW9kZWxDbGF6ejogTW9kZWxDb25zdHJ1Y3RvcjxUPlxuICApOiBSZXBvPFQ+IHwgTW9kZWxTZXJ2aWNlPFQ+IHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIFNlcnZpY2UuZ2V0PE1vZGVsU2VydmljZTxUPj4oTW9kZWxDbGF6eik7XG4gICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgdHJ5IHtcbiAgICAgICAgcmV0dXJuIE1vZGVsU2VydmljZS5nZXRTZXJ2aWNlKE1vZGVsQ2xhenopIGFzIE1vZGVsU2VydmljZTxUPjtcbiAgICAgIH0gY2F0Y2ggKGUyOiB1bmtub3duKSB7XG4gICAgICAgIHJldHVybiBSZXBvc2l0b3J5LmZvck1vZGVsKE1vZGVsQ2xhenopIGFzIFJlcG88VD47XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgc3RhdGljIGNyZWF0ZVF1ZXJ5Um91dGVzRnJvbVJlcG9zaXRvcnk8VCBleHRlbmRzIE1vZGVsPGJvb2xlYW4+PihcbiAgICBwZXJzaXN0ZW5jZTogUmVwbzxUPiB8IE1vZGVsU2VydmljZTxUPixcbiAgICBwcmVmaXg6IHN0cmluZyA9IFBlcnNpc3RlbmNlS2V5cy5RVUVSWVxuICApOiBDb250cm9sbGVyQ29uc3RydWN0b3I8VD4ge1xuICAgIGNvbnN0IHJlcG86IFJlcG88VD4gPVxuICAgICAgcGVyc2lzdGVuY2UgaW5zdGFuY2VvZiBNb2RlbFNlcnZpY2UgPyBwZXJzaXN0ZW5jZS5yZXBvIDogcGVyc2lzdGVuY2U7XG4gICAgY29uc3QgTW9kZWxDb25zdHI6IENvbnN0cnVjdG9yID0gcmVwby5jbGFzcztcbiAgICBjb25zdCBxdWVyeU1ldGhvZHM6IFJlY29yZDxzdHJpbmcsIHsgZmllbGRzPzogc3RyaW5nW10gfCB1bmRlZmluZWQgfT4gPVxuICAgICAgTWV0YWRhdGEuZ2V0KFxuICAgICAgICByZXBvLmNvbnN0cnVjdG9yIGFzIENvbnN0cnVjdG9yLFxuICAgICAgICBNZXRhZGF0YS5rZXkoUGVyc2lzdGVuY2VLZXlzLlFVRVJZKVxuICAgICAgKSA/PyB7fTtcblxuICAgIGNvbnN0IHJvdXRlTWV0aG9kczogUmVjb3JkPHN0cmluZywgYW55PiA9XG4gICAgICBNZXRhZGF0YS5nZXQoXG4gICAgICAgIHBlcnNpc3RlbmNlLmNvbnN0cnVjdG9yIGFzIENvbnN0cnVjdG9yLFxuICAgICAgICBNZXRhZGF0YS5rZXkoREVDQUZfUk9VVEUpXG4gICAgICApID8/IHt9O1xuXG4gICAgY2xhc3MgUXVlcnlDb250cm9sbGVyIGV4dGVuZHMgRGVjYWZNb2RlbENvbnRyb2xsZXI8VD4ge1xuICAgICAgb3ZlcnJpZGUgZ2V0IGNsYXNzKCk6IE1vZGVsQ29uc3RydWN0b3I8VD4ge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJNZXRob2Qgbm90IGltcGxlbWVudGVkLlwiKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0cnVjdG9yKGNsaWVudENvbnRleHQ6IERlY2FmUmVxdWVzdENvbnRleHQsIG5hbWU6IHN0cmluZykge1xuICAgICAgICBzdXBlcihjbGllbnRDb250ZXh0LCBuYW1lKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBmb3IgKGNvbnN0IFttZXRob2ROYW1lLCBwYXJhbXNdIG9mIE9iamVjdC5lbnRyaWVzKHJvdXRlTWV0aG9kcykpIHtcbiAgICAgIGNvbnN0IHJvdXRlUGF0aCA9IFtwYXJhbXMucGF0aC5yZXBsYWNlKC9eXFwvK3xcXC8rJC9nLCBcIlwiKV1cbiAgICAgICAgLmZpbHRlcigoc2VnbWVudDogc3RyaW5nKSA9PiBzZWdtZW50ICYmIHNlZ21lbnQudHJpbSgpKVxuICAgICAgICAuam9pbihcIi9cIik7XG5cbiAgICAgIGNvbnN0IGhhbmRsZXIgPSBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUNvbXBsZXhRdWVyeUhhbmRsZXIoXG4gICAgICAgIG1ldGhvZE5hbWVcbiAgICAgICkgYXMgYW55O1xuICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5kZWZpbmVNZXRob2QoXG4gICAgICAgIFF1ZXJ5Q29udHJvbGxlcixcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgaGFuZGxlclxuICAgICAgKTtcblxuICAgICAgY29uc3QgaHR0cERlY29yYXRvciA9IEh0dHBWZXJiVG9EZWNvcmF0b3IocGFyYW1zLmh0dHBNZXRob2QgYXMgSHR0cFZlcmJzKShyb3V0ZVBhdGggfHwgdW5kZWZpbmVkKTtcbiAgICAgIGNvbnN0IGRlY29yYXRvcnMgPSBGcm9tTW9kZWxDb250cm9sbGVyLmdldFF1ZXJ5RGVjb3JhdG9ycyhcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgcm91dGVQYXRoLFxuICAgICAgICBwYXJhbXMuaHR0cE1ldGhvZFxuICAgICAgKTtcbiAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuYXBwbHlEZWNvcmF0b3JzKFxuICAgICAgICBRdWVyeUNvbnRyb2xsZXIsXG4gICAgICAgIG1ldGhvZE5hbWUsXG4gICAgICAgIFtodHRwRGVjb3JhdG9yLCAuLi5kZWNvcmF0b3JzXVxuICAgICAgKTtcbiAgICB9XG5cbiAgICBmb3IgKGNvbnN0IFttZXRob2ROYW1lLCBvYmpWYWx1ZXNdIG9mIE9iamVjdC5lbnRyaWVzKHF1ZXJ5TWV0aG9kcykpIHtcbiAgICAgIGNvbnN0IGZpZWxkcyA9IG9ialZhbHVlcy5maWVsZHMgPz8gW107XG4gICAgICBjb25zdCByb3V0ZVBhdGggPSBbcHJlZml4LCBtZXRob2ROYW1lLCAuLi5maWVsZHMubWFwKChmKSA9PiBgOiR7Zn1gKV1cbiAgICAgICAgLmZpbHRlcigoc2VnbWVudCkgPT4gc2VnbWVudCAmJiBzZWdtZW50LnRyaW0oKSlcbiAgICAgICAgLmpvaW4oXCIvXCIpO1xuXG4gICAgICBjb25zdCBoYW5kbGVyID0gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVDb21wbGV4UXVlcnlIYW5kbGVyKFxuICAgICAgICBtZXRob2ROYW1lXG4gICAgICApIGFzIGFueTtcbiAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuZGVmaW5lTWV0aG9kKFxuICAgICAgICBRdWVyeUNvbnRyb2xsZXIsXG4gICAgICAgIG1ldGhvZE5hbWUsXG4gICAgICAgIGhhbmRsZXJcbiAgICAgICk7XG5cbiAgICAgIGNvbnN0IGh0dHBEZWNvcmF0b3IgPSBIdHRwVmVyYlRvRGVjb3JhdG9yKFwiR0VUXCIgYXMgSHR0cFZlcmJzKShyb3V0ZVBhdGggfHwgdW5kZWZpbmVkKTtcbiAgICAgIGNvbnN0IGRlY29yYXRvcnMgPSBGcm9tTW9kZWxDb250cm9sbGVyLmdldFF1ZXJ5RGVjb3JhdG9ycyhcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgcm91dGVQYXRoLFxuICAgICAgICBcIkdFVFwiLFxuICAgICAgICB0cnVlXG4gICAgICApO1xuICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5hcHBseURlY29yYXRvcnMoXG4gICAgICAgIFF1ZXJ5Q29udHJvbGxlcixcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgW2h0dHBEZWNvcmF0b3IsIC4uLmRlY29yYXRvcnNdXG4gICAgICApO1xuICAgIH1cblxuICAgIHJldHVybiBRdWVyeUNvbnRyb2xsZXI7XG4gIH1cblxuICBzdGF0aWMgY3JlYXRlPFQgZXh0ZW5kcyBNb2RlbDxhbnk+PihcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3RvcjxUPixcbiAgICBtb2R1bGVDb25maWdPdmVycmlkZXM/OiBSZWNvcmQ8c3RyaW5nLCBNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnPixcbiAgICBnbG9iYWxEZWZhdWx0cz86IFBhcnRpYWw8TW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZz5cbiAgKTogQ29udHJvbGxlckNvbnN0cnVjdG9yPFQ+IHtcbiAgICBjb25zdCBsb2cgPSBGcm9tTW9kZWxDb250cm9sbGVyLmxvZy5mb3IoRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGUpO1xuICAgIGNvbnN0IHRhYmxlTmFtZSA9IE1vZGVsLnRhYmxlTmFtZShNb2RlbENvbnN0cik7XG4gICAgY29uc3Qgcm91dGVQYXRoID0gdG9LZWJhYkNhc2UodGFibGVOYW1lKTtcbiAgICBjb25zdCBtb2RlbENsYXp6TmFtZSA9IE1vZGVsQ29uc3RyLm5hbWU7XG4gICAgY29uc3QgcGVyc2lzdGVuY2UgPSBGcm9tTW9kZWxDb250cm9sbGVyLmdldFBlcnNpc3RlbmNlKE1vZGVsQ29uc3RyKTtcblxuICAgIC8vIFdoZW4gcGVyc2lzdGVuY2UgaXMgYSBNb2RlbFNlcnZpY2UsIHRoZSBAcXVlcnkvQHJvdXRlIG1ldGFkYXRhIGxpdmVzIG9uXG4gICAgLy8gdGhlIHVuZGVybHlpbmcgcmVwb3NpdG9yeSBjbGFzcyAoY3VzdG9tIHJlcG8pLCBub3Qgb24gTW9kZWxTZXJ2aWNlIGl0c2VsZi5cbiAgICAvLyBQYXNzIHRoZSByZXBvIHRvIHRoZSBmYWN0b3J5IHNvIGFkZENvbXBsZXhRdWVyaWVzKCkgY2FuIGRpc2NvdmVyIHRoZW0uXG4gICAgY29uc3QgZmFjdG9yeVBlcnNpc3RlbmNlID1cbiAgICAgIHBlcnNpc3RlbmNlIGluc3RhbmNlb2YgTW9kZWxTZXJ2aWNlID8gcGVyc2lzdGVuY2UucmVwbyA6IHBlcnNpc3RlbmNlO1xuXG4gICAgY29uc3QgZGVjb3JhdG9yQ29uZmlnID0gTWV0YWRhdGEuZ2V0KFxuICAgICAgTW9kZWxDb25zdHIsXG4gICAgICBNZXRhZGF0YS5rZXkoREVDQUZfQ09OVFJPTExFUl9DT05GSUcpXG4gICAgKSBhcyBNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnIHwgdW5kZWZpbmVkO1xuICAgIGNvbnN0IG1vZHVsZU92ZXJyaWRlID0gbW9kdWxlQ29uZmlnT3ZlcnJpZGVzPy5bTW9kZWxDb25zdHIubmFtZV07XG4gICAgY29uc3QgbWVyZ2VkQ29uZmlnOiBNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnID0ge1xuICAgICAgLi4uKGdsb2JhbERlZmF1bHRzIHx8IHt9KSxcbiAgICAgIC4uLihkZWNvcmF0b3JDb25maWcgfHwge30pLFxuICAgICAgLi4uKG1vZHVsZU92ZXJyaWRlIHx8IHt9KSxcbiAgICB9O1xuXG4gICAgY29uc3QgRmFjdG9yeUNvbnRyb2xsZXIgPSBNb2RlbENvbnRyb2xsZXJGYWN0b3J5LmNyZWF0ZTxUPihcbiAgICAgIE1vZGVsQ29uc3RyLFxuICAgICAgZmFjdG9yeVBlcnNpc3RlbmNlLFxuICAgICAgbWVyZ2VkQ29uZmlnXG4gICAgKTtcbiAgICBjb25zdCBmYWN0b3J5Um91dGVzID0gKEZhY3RvcnlDb250cm9sbGVyIGFzIGFueSkuX19yb3V0ZXNfXyBhc1xuICAgICAgfCBTZXJ2ZXJSb3V0ZVtdXG4gICAgICB8IHVuZGVmaW5lZDtcblxuICAgIGNvbnN0IHsgZ2V0UEssIGFwaVByb3BlcnRpZXMsIHBhdGg6IHBrUGF0aCB9ID1cbiAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuZ2V0Um91dGVQYXJhbWV0ZXJzRnJvbU1vZGVsKE1vZGVsQ29uc3RyKTtcblxuICAgIGxvZy5kZWJ1ZyhcbiAgICAgIGBDcmVhdGluZyBjb250cm9sbGVyIGZvciBtb2RlbDogJHttb2RlbENsYXp6TmFtZX0gd2l0aCAke2ZhY3RvcnlSb3V0ZXM/Lmxlbmd0aCA/PyAwfSBmYWN0b3J5IHJvdXRlc2BcbiAgICApO1xuXG4gICAgY29uc3QgYXV0aENvbmZpZzogQXV0aENvbmZpZyB8IHVuZGVmaW5lZCA9IG1lcmdlZENvbmZpZy5hdXRoO1xuXG4gICAgZnVuY3Rpb24gYXBwbHlDbGFzc0F1dGgodGFyZ2V0OiBhbnkpIHtcbiAgICAgIGlmIChhdXRoQ29uZmlnPy5wdWJsaWMpIHtcbiAgICAgICAgUHVibGljKCkodGFyZ2V0KTtcbiAgICAgIH0gZWxzZSBpZiAoYXV0aENvbmZpZz8ucm9sZXM/Lmxlbmd0aCkge1xuICAgICAgICBSZXF1aXJlUm9sZXMoLi4uYXV0aENvbmZpZy5yb2xlcykodGFyZ2V0KTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIEF1dGgoTW9kZWxDb25zdHIpKHRhcmdldCk7XG4gICAgICB9XG4gICAgICBpZiAoYXV0aENvbmZpZz8uc2tpcE1vZGVsUm9sZXMpIHtcbiAgICAgICAgU2V0TWV0YWRhdGEoU0tJUF9NT0RFTF9ST0xFU19LRVksIHRydWUpKHRhcmdldCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgQENvbnRyb2xsZXIocm91dGVQYXRoKVxuICAgIEBBcGlUYWdzKG1vZGVsQ2xhenpOYW1lKVxuICAgIEBBcGlFeHRyYU1vZGVscyhNb2RlbENvbnN0cilcbiAgICBjbGFzcyBEeW5hbWljTW9kZWxDb250cm9sbGVyIGV4dGVuZHMgRGVjYWZNb2RlbENvbnRyb2xsZXI8VD4ge1xuICAgICAgcHJpdmF0ZSByZWFkb25seSBwazogc3RyaW5nID0gTW9kZWwucGsoTW9kZWxDb25zdHIpIGFzIHN0cmluZztcblxuICAgICAgcHJvdGVjdGVkIHN0YXRpYyBnZXQgY2xhc3MoKSB7XG4gICAgICAgIHJldHVybiBNb2RlbENvbnN0cjtcbiAgICAgIH1cblxuICAgICAgb3ZlcnJpZGUgZ2V0IGNsYXNzKCk6IE1vZGVsQ29uc3RydWN0b3I8VD4ge1xuICAgICAgICByZXR1cm4gTW9kZWxDb25zdHI7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0cnVjdG9yKGNsaWVudENvbnRleHQ6IERlY2FmUmVxdWVzdENvbnRleHQpIHtcbiAgICAgICAgc3VwZXIoY2xpZW50Q29udGV4dCwgRHluYW1pY01vZGVsQ29udHJvbGxlci5uYW1lKTtcbiAgICAgICAgbG9nLmluZm8oXG4gICAgICAgICAgYFJlZ2lzdGVyaW5nIGR5bmFtaWMgY29udHJvbGxlciBmb3IgbW9kZWw6ICR7dGhpcy5jbGFzcy5uYW1lfSByb3V0ZTogLyR7cm91dGVQYXRofWBcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBhcHBseUNsYXNzQXV0aChEeW5hbWljTW9kZWxDb250cm9sbGVyKTtcblxuICAgIGlmIChmYWN0b3J5Um91dGVzKSB7XG4gICAgICBjb25zdCBzb3J0ZWRSb3V0ZXMgPSBbLi4uZmFjdG9yeVJvdXRlc10uc29ydCgoYSwgYikgPT4ge1xuICAgICAgICBjb25zdCBhU2VnbWVudHMgPSBhLnBhdGguc3BsaXQoXCIvXCIpLmZpbHRlcihCb29sZWFuKTtcbiAgICAgICAgY29uc3QgYlNlZ21lbnRzID0gYi5wYXRoLnNwbGl0KFwiL1wiKS5maWx0ZXIoQm9vbGVhbik7XG4gICAgICAgIGNvbnN0IGFQYXJhbUNvdW50ID0gYVNlZ21lbnRzLmZpbHRlcigocykgPT4gcy5zdGFydHNXaXRoKFwiOlwiKSkubGVuZ3RoO1xuICAgICAgICBjb25zdCBiUGFyYW1Db3VudCA9IGJTZWdtZW50cy5maWx0ZXIoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpLmxlbmd0aDtcbiAgICAgICAgY29uc3QgYUxpdGVyYWxDb3VudCA9IGFTZWdtZW50cy5sZW5ndGggLSBhUGFyYW1Db3VudDtcbiAgICAgICAgY29uc3QgYkxpdGVyYWxDb3VudCA9IGJTZWdtZW50cy5sZW5ndGggLSBiUGFyYW1Db3VudDtcbiAgICAgICAgaWYgKGFMaXRlcmFsQ291bnQgIT09IGJMaXRlcmFsQ291bnQpXG4gICAgICAgICAgcmV0dXJuIGJMaXRlcmFsQ291bnQgLSBhTGl0ZXJhbENvdW50O1xuICAgICAgICBpZiAoYVBhcmFtQ291bnQgIT09IGJQYXJhbUNvdW50KVxuICAgICAgICAgIHJldHVybiBhUGFyYW1Db3VudCAtIGJQYXJhbUNvdW50O1xuICAgICAgICByZXR1cm4gMDtcbiAgICAgIH0pO1xuICAgICAgZm9yIChjb25zdCByb3V0ZSBvZiBzb3J0ZWRSb3V0ZXMpIHtcbiAgICAgICAgY29uc3QgcmVnaXN0cmF0aW9uID0gRnJvbU1vZGVsQ29udHJvbGxlci5tYXRjaFJvdXRlKFxuICAgICAgICAgIHJvdXRlLFxuICAgICAgICAgIHBrUGF0aCxcbiAgICAgICAgICBhcGlQcm9wZXJ0aWVzLFxuICAgICAgICAgIGdldFBLLFxuICAgICAgICAgIE1vZGVsQ29uc3RyLFxuICAgICAgICAgIG1vZGVsQ2xhenpOYW1lLFxuICAgICAgICAgIGZhY3RvcnlQZXJzaXN0ZW5jZVxuICAgICAgICApO1xuICAgICAgICBpZiAoIXJlZ2lzdHJhdGlvbikgY29udGludWU7XG5cbiAgICAgICAgY29uc3QgeyBtZXRob2ROYW1lLCBoYW5kbGVyLCBkZWNvcmF0b3JzLCBwYXJhbURlY29yYXRvcnMgfSA9XG4gICAgICAgICAgcmVnaXN0cmF0aW9uO1xuXG4gICAgICAgIGNvbnN0IGRlc2NyaXB0b3IgPSBGcm9tTW9kZWxDb250cm9sbGVyLmRlZmluZU1ldGhvZChcbiAgICAgICAgICBEeW5hbWljTW9kZWxDb250cm9sbGVyLFxuICAgICAgICAgIG1ldGhvZE5hbWUsXG4gICAgICAgICAgaGFuZGxlclxuICAgICAgICApO1xuXG4gICAgICAgIGlmIChkZXNjcmlwdG9yKSB7XG4gICAgICAgICAgY29uc3QgaHR0cERlY29yYXRvciA9IEh0dHBWZXJiVG9EZWNvcmF0b3Iocm91dGUubWV0aG9kIGFzIEh0dHBWZXJicykoXG4gICAgICAgICAgICByb3V0ZS5wYXRoLnJlcGxhY2UoL15cXC8rfFxcLyskL2csIFwiXCIpIHx8IHVuZGVmaW5lZFxuICAgICAgICAgICk7XG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5hcHBseURlY29yYXRvcnMoXG4gICAgICAgICAgICBEeW5hbWljTW9kZWxDb250cm9sbGVyLFxuICAgICAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgICAgIFtodHRwRGVjb3JhdG9yLCAuLi5kZWNvcmF0b3JzXSxcbiAgICAgICAgICAgIHBhcmFtRGVjb3JhdG9yc1xuICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gRHluYW1pY01vZGVsQ29udHJvbGxlciBhcyBhbnk7XG4gIH1cblxuICBzdGF0aWMgZ2V0Um91dGVQYXJhbWV0ZXJzRnJvbU1vZGVsPFQgZXh0ZW5kcyBNb2RlbDxhbnk+PihcbiAgICBNb2RlbENsYXp6OiBNb2RlbENvbnN0cnVjdG9yPFQ+XG4gICk6IHtcbiAgICBwYXRoOiBzdHJpbmc7XG4gICAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgICBhcGlQcm9wZXJ0aWVzOiBEZWNhZkFwaVByb3BlcnR5W107XG4gICAgZ2V0UEs6ICguLi5wYXJhbXM6IEFycmF5PHN0cmluZyB8IG51bWJlcj4pID0+IHN0cmluZztcbiAgfSB7XG4gICAgY29uc3QgcGsgPSBNb2RlbC5wayhNb2RlbENsYXp6KSBhcyBrZXlvZiBNb2RlbDxhbnk+O1xuICAgIGNvbnN0IGNvbXBvc2VkID0gTWV0YWRhdGEuZ2V0KFxuICAgICAgTW9kZWxDbGF6eixcbiAgICAgIE1ldGFkYXRhLmtleShEQktleXMuQ09NUE9TRUQsIHBrKVxuICAgICk7XG4gICAgY29uc3QgY29tcG9zZWRLZXlzID0gY29tcG9zZWQ/LmFyZ3MgPz8gW107XG5cbiAgICBjb25zdCB1bmlxdWVLZXlzID1cbiAgICAgIEFycmF5LmlzQXJyYXkoY29tcG9zZWRLZXlzKSAmJiBjb21wb3NlZEtleXMubGVuZ3RoID4gMFxuICAgICAgICA/IEFycmF5LmZyb20obmV3IFNldChbLi4uY29tcG9zZWRLZXlzXSkpXG4gICAgICAgIDogQXJyYXkuZnJvbShuZXcgU2V0KFtwa10pKTtcblxuICAgIGNvbnN0IGRlc2NyaXB0aW9uID0gTWV0YWRhdGEuZGVzY3JpcHRpb24oTW9kZWxDbGF6eikgPz8gXCJcIjtcbiAgICBjb25zdCBwYXRoID0gYDoke3VuaXF1ZUtleXMuam9pbihcIi86XCIpfWA7XG4gICAgY29uc3QgYXBpUHJvcGVydGllczogRGVjYWZBcGlQcm9wZXJ0eVtdID0gdW5pcXVlS2V5cy5tYXAoKGtleSkgPT4ge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbmFtZToga2V5LFxuICAgICAgICBkZXNjcmlwdGlvbjogTWV0YWRhdGEuZGVzY3JpcHRpb24oTW9kZWxDbGF6eiwga2V5KSxcbiAgICAgICAgcmVxdWlyZWQ6IHRydWUsXG4gICAgICAgIHR5cGU6IFN0cmluZyxcbiAgICAgIH07XG4gICAgfSk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgcGF0aCxcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgYXBpUHJvcGVydGllcyxcbiAgICAgIGdldFBLOiAoLi4ucGFyYW1zOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PlxuICAgICAgICBjb21wb3NlZD8uc2VwYXJhdG9yID8gcGFyYW1zLmpvaW4oY29tcG9zZWQuc2VwYXJhdG9yKSA6IHBhcmFtcy5qb2luKFwiXCIpLFxuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBkZWZpbmVNZXRob2QoXG4gICAgdGFyZ2V0OiBhbnksXG4gICAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICAgIGhhbmRsZXI6ICguLi5hcmdzOiBhbnlbXSkgPT4gYW55XG4gICk6IFByb3BlcnR5RGVzY3JpcHRvciB8IHVuZGVmaW5lZCB7XG4gICAgT2JqZWN0LmRlZmluZVByb3BlcnR5KFxuICAgICAgdGFyZ2V0LnByb3RvdHlwZSB8fCB0YXJnZXQsXG4gICAgICBtZXRob2ROYW1lLFxuICAgICAge1xuICAgICAgICB2YWx1ZTogaGFuZGxlcixcbiAgICAgICAgd3JpdGFibGU6IGZhbHNlLFxuICAgICAgICBjb25maWd1cmFibGU6IHRydWUsXG4gICAgICAgIGVudW1lcmFibGU6IGZhbHNlLFxuICAgICAgfVxuICAgICk7XG5cbiAgICByZXR1cm4gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihcbiAgICAgIHRhcmdldC5wcm90b3R5cGUgfHwgdGFyZ2V0LFxuICAgICAgbWV0aG9kTmFtZVxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBhcHBseURlY29yYXRvcnMoXG4gICAgdGFyZ2V0OiBhbnksXG4gICAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICAgIG1ldGhvZERlY29yYXRvcnM6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4sXG4gICAgcGFyYW1EZWNvcmF0b3JzOiBBcnJheTx7IGRlY29yYXRvcjogUGFyYW1ldGVyRGVjb3JhdG9yOyBpbmRleDogbnVtYmVyIH0+ID0gW11cbiAgKSB7XG4gICAgY29uc3QgcHJvdG8gPSB0YXJnZXQ/LnByb3RvdHlwZSA/PyB0YXJnZXQ7XG4gICAgY29uc3QgZGVzY3JpcHRvciA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IocHJvdG8sIG1ldGhvZE5hbWUpO1xuICAgIG1ldGhvZERlY29yYXRvcnMuZm9yRWFjaCgoZCkgPT4gZChwcm90bywgbWV0aG9kTmFtZSwgZGVzY3JpcHRvcikpO1xuICAgIHBhcmFtRGVjb3JhdG9ycy5mb3JFYWNoKCh7IGRlY29yYXRvciwgaW5kZXggfSkgPT5cbiAgICAgIGRlY29yYXRvcihwcm90bywgbWV0aG9kTmFtZSwgaW5kZXgpXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIG1hdGNoUm91dGUoXG4gICAgcm91dGU6IFNlcnZlclJvdXRlLFxuICAgIHBrUGF0aDogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXSxcbiAgICBnZXRQSzogKC4uLnA6IEFycmF5PHN0cmluZyB8IG51bWJlcj4pID0+IHN0cmluZyxcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcsXG4gICAgcGVyc2lzdGVuY2U/OiBhbnlcbiAgKToge1xuICAgIG1ldGhvZE5hbWU6IHN0cmluZztcbiAgICBoYW5kbGVyOiAoLi4uYXJnczogYW55W10pID0+IGFueTtcbiAgICBkZWNvcmF0b3JzOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+O1xuICAgIHBhcmFtRGVjb3JhdG9yczogQXJyYXk8eyBkZWNvcmF0b3I6IFBhcmFtZXRlckRlY29yYXRvcjsgaW5kZXg6IG51bWJlciB9PjtcbiAgfSB8IHVuZGVmaW5lZCB7XG4gICAgY29uc3QgeyBtZXRob2QsIHBhdGggfSA9IHJvdXRlO1xuICAgIGNvbnN0IG5vcm1hbGl6ZWRQYXRoID0gcGF0aC5yZXBsYWNlKC9eXFwvK3xcXC8rJC9nLCBcIlwiKTtcblxuICAgIGlmIChtZXRob2QgPT09IFwiUE9TVFwiICYmIG5vcm1hbGl6ZWRQYXRoID09PSBcIlwiKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwiY3JlYXRlXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQ3JlYXRlSGFuZGxlcihNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUNyZWF0ZURlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBEZWNhZkJvZHkoKSBhcyBhbnksIGluZGV4OiAwIH0sIHsgZGVjb3JhdG9yOiBSZXNwb25zZSh7IHBhc3N0aHJvdWdoOiB0cnVlIH0pIGFzIGFueSwgaW5kZXg6IDEgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJQT1NUXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IFwiYnVsa1wiKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwiY3JlYXRlQWxsXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQnVsa0NyZWF0ZUhhbmRsZXIoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5idWxrQ3JlYXRlRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBbeyBkZWNvcmF0b3I6IERlY2FmQm9keSgpIGFzIGFueSwgaW5kZXg6IDAgfSwgeyBkZWNvcmF0b3I6IFJlc3BvbnNlKHsgcGFzc3Rocm91Z2g6IHRydWUgfSkgYXMgYW55LCBpbmRleDogMSB9XVxuICAgICAgKTtcbiAgICB9XG5cbiAgICBpZiAobWV0aG9kID09PSBcIkdFVFwiICYmIG5vcm1hbGl6ZWRQYXRoID09PSBcImJ1bGtcIikge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcInJlYWRBbGxcIixcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVCdWxrUmVhZEhhbmRsZXIobW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmJ1bGtSZWFkRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBbeyBkZWNvcmF0b3I6IFF1ZXJ5KFwiaWRzXCIpIGFzIGFueSwgaW5kZXg6IDAgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJQVVRcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gXCJidWxrXCIpIHtcbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgXCJ1cGRhdGVBbGxcIixcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVCdWxrVXBkYXRlSGFuZGxlcihtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuYnVsa1VwZGF0ZURlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lLCBhcGlQcm9wZXJ0aWVzKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBEZWNhZkJvZHkoKSBhcyBhbnksIGluZGV4OiAwIH0sIHsgZGVjb3JhdG9yOiBSZXNwb25zZSh7IHBhc3N0aHJvdWdoOiB0cnVlIH0pIGFzIGFueSwgaW5kZXg6IDEgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJERUxFVEVcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gXCJidWxrXCIpIHtcbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgXCJkZWxldGVBbGxcIixcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVCdWxrRGVsZXRlSGFuZGxlcihtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuYnVsa0RlbGV0ZURlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lLCBhcGlQcm9wZXJ0aWVzKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBRdWVyeShcImlkc1wiKSBhcyBhbnksIGluZGV4OiAwIH0sIHsgZGVjb3JhdG9yOiBSZXNwb25zZSh7IHBhc3N0aHJvdWdoOiB0cnVlIH0pIGFzIGFueSwgaW5kZXg6IDEgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJHRVRcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gcGtQYXRoKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwicmVhZFwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlYWRIYW5kbGVyKGdldFBLLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIucmVhZERlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lLCBhcGlQcm9wZXJ0aWVzLCBwa1BhdGgpLFxuICAgICAgICBbeyBkZWNvcmF0b3I6IERlY2FmUGFyYW1zKGFwaVByb3BlcnRpZXMpIGFzIGFueSwgaW5kZXg6IDAgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJQVVRcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gcGtQYXRoKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwidXBkYXRlXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlVXBkYXRlSGFuZGxlcihnZXRQSywgTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci51cGRhdGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgYXBpUHJvcGVydGllcywgcGtQYXRoKSxcbiAgICAgICAgW1xuICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZlBhcmFtcyhhcGlQcm9wZXJ0aWVzKSBhcyBhbnksIGluZGV4OiAwIH0sXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IERlY2FmQm9keSgpIGFzIGFueSwgaW5kZXg6IDEgfSxcbiAgICAgICAgICB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAyIH0sXG4gICAgICAgIF1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJERUxFVEVcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gcGtQYXRoKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwiZGVsZXRlXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlRGVsZXRlSGFuZGxlcihnZXRQSywgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmRlbGV0ZURlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lLCBhcGlQcm9wZXJ0aWVzLCBwa1BhdGgpLFxuICAgICAgICBbXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IERlY2FmUGFyYW1zKGFwaVByb3BlcnRpZXMpIGFzIGFueSwgaW5kZXg6IDAgfSxcbiAgICAgICAgICB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAxIH0sXG4gICAgICAgIF1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgLy8gQ29tcG9zZWQgUEsgZmFsbGJhY2sgcm91dGVzIChmaWx0ZXJFbXB0eSkg4oCUIHBhdGggZGlmZmVycyBmcm9tIHBrUGF0aFxuICAgIGNvbnN0IGZhbGxiYWNrU2VnbWVudHMgPSBub3JtYWxpemVkUGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKEJvb2xlYW4pO1xuICAgIGNvbnN0IGlzQWxsUGFyYW1zID0gZmFsbGJhY2tTZWdtZW50cy5sZW5ndGggPiAwICYmIGZhbGxiYWNrU2VnbWVudHMuZXZlcnkoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpO1xuICAgIGlmIChpc0FsbFBhcmFtcyAmJiBub3JtYWxpemVkUGF0aCAhPT0gcGtQYXRoKSB7XG4gICAgICBjb25zdCBmYWxsYmFja0FwaVByb3BzID0gZmFsbGJhY2tTZWdtZW50cy5tYXAoKHMpID0+IHMuc2xpY2UoMSkpLm1hcCgobmFtZSkgPT4gKHtcbiAgICAgICAgbmFtZSxcbiAgICAgICAgZGVzY3JpcHRpb246IGAke25hbWV9IHBhcmFtZXRlcmAsXG4gICAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgICAgICB0eXBlOiBTdHJpbmcsXG4gICAgICB9KSk7XG4gICAgICBjb25zdCBzdWZmaXggPSBmYWxsYmFja1NlZ21lbnRzLm1hcCgocykgPT4gcy5zbGljZSgxKSkuam9pbihcIkFuZFwiKTtcbiAgICAgIGlmIChtZXRob2QgPT09IFwiR0VUXCIpIHtcbiAgICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICAgIGByZWFkQnkke3N1ZmZpeH1gLFxuICAgICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVhZEhhbmRsZXIoZ2V0UEssIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLnJlYWREZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgZmFsbGJhY2tBcGlQcm9wcywgbm9ybWFsaXplZFBhdGgpLFxuICAgICAgICAgIFt7IGRlY29yYXRvcjogRGVjYWZQYXJhbXMoZmFsbGJhY2tBcGlQcm9wcykgYXMgYW55LCBpbmRleDogMCB9XVxuICAgICAgICApO1xuICAgICAgfVxuICAgICAgaWYgKG1ldGhvZCA9PT0gXCJQVVRcIikge1xuICAgICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgICAgYHVwZGF0ZUJ5JHtzdWZmaXh9YCxcbiAgICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVVwZGF0ZUhhbmRsZXIoZ2V0UEssIE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci51cGRhdGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgZmFsbGJhY2tBcGlQcm9wcywgbm9ybWFsaXplZFBhdGgpLFxuICAgICAgICAgIFtcbiAgICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZlBhcmFtcyhmYWxsYmFja0FwaVByb3BzKSBhcyBhbnksIGluZGV4OiAwIH0sXG4gICAgICAgICAgICB7IGRlY29yYXRvcjogRGVjYWZCb2R5KCkgYXMgYW55LCBpbmRleDogMSB9LFxuICAgICAgICAgICAgeyBkZWNvcmF0b3I6IFJlc3BvbnNlKHsgcGFzc3Rocm91Z2g6IHRydWUgfSkgYXMgYW55LCBpbmRleDogMiB9LFxuICAgICAgICAgIF1cbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICAgIGlmIChtZXRob2QgPT09IFwiREVMRVRFXCIpIHtcbiAgICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICAgIGBkZWxldGVCeSR7c3VmZml4fWAsXG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVEZWxldGVIYW5kbGVyKGdldFBLLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5kZWxldGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgZmFsbGJhY2tBcGlQcm9wcywgbm9ybWFsaXplZFBhdGgpLFxuICAgICAgICAgIFtcbiAgICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZlBhcmFtcyhmYWxsYmFja0FwaVByb3BzKSBhcyBhbnksIGluZGV4OiAwIH0sXG4gICAgICAgICAgICB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAxIH0sXG4gICAgICAgICAgXVxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiR0VUXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IFwic3RhdGVtZW50LzptZXRob2QvKmFyZ3NcIikge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcInN0YXRlbWVudFwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVN0YXRlbWVudEhhbmRsZXIobW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLnN0YXRlbWVudERlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgW1xuICAgICAgICAgIHsgZGVjb3JhdG9yOiBQYXJhbShcIm1ldGhvZFwiKSBhcyBhbnksIGluZGV4OiAwIH0sXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IFBhcmFtKFwiYXJnc1wiKSBhcyBhbnksIGluZGV4OiAxIH0sXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IERlY2FmUXVlcnkoKSBhcyBhbnksIGluZGV4OiAyIH0sXG4gICAgICAgIF1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgY29uc3Qgc3RhdGVtZW50Um91dGVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICAgICAgXCJsaXN0QnkvOmtleVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuTElTVF9CWSxcbiAgICAgIFwicGFnaW5hdGVCeS86a2V5LzpwYWdlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5QQUdFX0JZLFxuICAgICAgXCJmaW5kLzp2YWx1ZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORCxcbiAgICAgIFwicGFnZS86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0UsXG4gICAgICBcImZpbmRPbmVCeS86a2V5Lzp2YWx1ZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9PTkVfQlksXG4gICAgICBcImZpbmRCeS86a2V5Lzp2YWx1ZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9CWSxcbiAgICAgIFwiY291bnRPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkNPVU5UX09GLFxuICAgICAgXCJtYXhPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLk1BWF9PRixcbiAgICAgIFwibWluT2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5NSU5fT0YsXG4gICAgICBcImF2Z09mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuQVZHX09GLFxuICAgICAgXCJzdW1PZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlNVTV9PRixcbiAgICAgIFwiZGlzdGluY3RPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkRJU1RJTkNUX09GLFxuICAgICAgXCJncm91cE9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuR1JPVVBfT0YsXG4gICAgfTtcblxuICAgIGNvbnN0IHN0YXRlbWVudEtleSA9IHN0YXRlbWVudFJvdXRlc1tub3JtYWxpemVkUGF0aF07XG4gICAgaWYgKHN0YXRlbWVudEtleSAmJiBtZXRob2QgPT09IFwiR0VUXCIpIHtcbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5zdGF0ZW1lbnRNZXRob2ROYW1lKG5vcm1hbGl6ZWRQYXRoKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVTdGF0ZW1lbnRTaG9ydGN1dEhhbmRsZXIoc3RhdGVtZW50S2V5LCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuc3RhdGVtZW50U2hvcnRjdXREZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgbm9ybWFsaXplZFBhdGgsIHN0YXRlbWVudEtleSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuc3RhdGVtZW50U2hvcnRjdXRQYXJhbXMobm9ybWFsaXplZFBhdGgpXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiR0VUXCIgJiYgbm9ybWFsaXplZFBhdGguc3RhcnRzV2l0aChcInF1ZXJ5L1wiKSkge1xuICAgICAgY29uc3QgcXVlcnlNZXRob2QgPSBub3JtYWxpemVkUGF0aC5yZXBsYWNlKC9ecXVlcnlcXC8vLCBcIlwiKS5zcGxpdChcIi9cIilbMF07XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIHF1ZXJ5TWV0aG9kLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUNvbXBsZXhRdWVyeUhhbmRsZXIocXVlcnlNZXRob2QpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmdldFF1ZXJ5RGVjb3JhdG9ycyhxdWVyeU1ldGhvZCwgbm9ybWFsaXplZFBhdGgsIFwiR0VUXCIsIHRydWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNvbXBsZXhRdWVyeVBhcmFtcyhub3JtYWxpemVkUGF0aClcbiAgICAgICk7XG4gICAgfVxuXG4gICAgLy8gRmFsbGJhY2sgZm9yIGN1c3RvbSBAcm91dGUoKSBwYXRocyAoZS5nLiBcIm1ldGFkYXRhL2Zvci1wcm9kdWN0Lzpwcm9kdWN0Q29kZVwiKVxuICAgIGNvbnN0IHBhdGhTZWdtZW50cyA9IG5vcm1hbGl6ZWRQYXRoLnNwbGl0KFwiL1wiKS5maWx0ZXIoQm9vbGVhbik7XG4gICAgY29uc3Qga25vd25QcmVmaXhlcyA9IG5ldyBTZXQ8c3RyaW5nPihbXG4gICAgICBcImxpc3RCeVwiLCBcImZpbmRCeVwiLCBcImZpbmRCeVBhZ2luYXRlXCIsIFwiZmluZE9uZUJ5XCIsIFwicGFnaW5hdGVCeVwiLFxuICAgICAgXCJmaW5kXCIsIFwicGFnZVwiLCBcImNvdW50T2ZcIiwgXCJtYXhPZlwiLCBcIm1pbk9mXCIsIFwiYXZnT2ZcIiwgXCJzdW1PZlwiLFxuICAgICAgXCJkaXN0aW5jdE9mXCIsIFwiZ3JvdXBPZlwiLCBcInN0YXRlbWVudFwiLCBcImJ1bGtcIiwgXCJxdWVyeVwiLFxuICAgIF0pO1xuICAgIGlmIChcbiAgICAgIHBhdGhTZWdtZW50cy5sZW5ndGggPiAwICYmXG4gICAgICAhbm9ybWFsaXplZFBhdGguc3RhcnRzV2l0aChcInF1ZXJ5L1wiKSAmJlxuICAgICAgIWtub3duUHJlZml4ZXMuaGFzKHBhdGhTZWdtZW50c1swXSlcbiAgICApIHtcbiAgICAgIC8vIExvb2sgdXAgdGhlIGFjdHVhbCBtZXRob2QgbmFtZSBmcm9tIEByb3V0ZSBtZXRhZGF0YSBieSBtYXRjaGluZyB0aGUgcGF0aC5cbiAgICAgIC8vIFRoZSBAcm91dGUgZGVjb3JhdG9yIHN0b3JlcyB7IHBhdGgsIGh0dHBNZXRob2QsIGhhbmRsZXIgfSBrZXllZCBieSBtZXRob2QgbmFtZVxuICAgICAgLy8gb24gdGhlIHJlcG9zaXRvcnkgY29uc3RydWN0b3IuXG4gICAgICBjb25zdCByb3V0ZU1ldGFkYXRhOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID1cbiAgICAgICAgTWV0YWRhdGEuZ2V0KFxuICAgICAgICAgIChwZXJzaXN0ZW5jZSBhcyBhbnkpPy5jb25zdHJ1Y3RvcixcbiAgICAgICAgICBNZXRhZGF0YS5rZXkoREVDQUZfUk9VVEUpXG4gICAgICAgICkgPz8ge307XG4gICAgICBjb25zdCBtYXRjaGVkRW50cnkgPSBPYmplY3QuZW50cmllcyhyb3V0ZU1ldGFkYXRhKS5maW5kKFxuICAgICAgICAoWywgaW5mb10pID0+XG4gICAgICAgICAgaW5mbyAmJlxuICAgICAgICAgIHR5cGVvZiBpbmZvID09PSBcIm9iamVjdFwiICYmXG4gICAgICAgICAgaW5mby5wYXRoPy5yZXBsYWNlKC9eXFwvK3xcXC8rJC9nLCBcIlwiKSA9PT0gbm9ybWFsaXplZFBhdGhcbiAgICAgICk7XG4gICAgICBjb25zdCBhY3R1YWxNZXRob2ROYW1lID0gbWF0Y2hlZEVudHJ5Py5bMF0gfHwgcGF0aFNlZ21lbnRzWzBdO1xuXG4gICAgICBjb25zdCBwYXJhbVNlZ21lbnRzID0gcGF0aFNlZ21lbnRzLmZpbHRlcigocykgPT4gcy5zdGFydHNXaXRoKFwiOlwiKSk7XG4gICAgICBjb25zdCBhcGlQYXRoUGFyYW1zID0gcGFyYW1TZWdtZW50cy5tYXAoKHMpID0+IHMuc2xpY2UoMSkpLm1hcCgobmFtZSkgPT4gKHtcbiAgICAgICAgbmFtZSxcbiAgICAgICAgZGVzY3JpcHRpb246IGAke25hbWV9IHBhcmFtZXRlciBmb3IgdGhlIHF1ZXJ5YCxcbiAgICAgICAgcmVxdWlyZWQ6IHRydWUsXG4gICAgICAgIHR5cGU6IFN0cmluZyxcbiAgICAgIH0pKTtcblxuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBhY3R1YWxNZXRob2ROYW1lLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUN1c3RvbVJvdXRlSGFuZGxlcihhY3R1YWxNZXRob2ROYW1lKSxcbiAgICAgICAgW1xuICAgICAgICAgIC4uLmFwaVBhdGhQYXJhbXMubWFwKChwKSA9PiBBcGlQYXJhbShwKSksXG4gICAgICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYFJldHJpZXZlIHJlY29yZHMgdXNpbmcgXCIke2FjdHVhbE1ldGhvZE5hbWV9XCIuYCB9KSxcbiAgICAgICAgICBBcGlPa1Jlc3BvbnNlKHsgZGVzY3JpcHRpb246IFwiUmVzdWx0IHN1Y2Nlc3NmdWxseSByZXRyaWV2ZWQuXCIgfSksXG4gICAgICAgICAgQXBpTm9Db250ZW50UmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogXCJObyBjb250ZW50IHJldHVybmVkIGJ5IHRoZSBtZXRob2QuXCIgfSksXG4gICAgICAgIF0sXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY29tcGxleFF1ZXJ5UGFyYW1zKG5vcm1hbGl6ZWRQYXRoKVxuICAgICAgKTtcbiAgICB9XG5cbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgIG1ldGhvZE5hbWU6IHN0cmluZyxcbiAgICBoYW5kbGVyOiAoLi4uYXJnczogYW55W10pID0+IGFueSxcbiAgICBkZWNvcmF0b3JzOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+LFxuICAgIHBhcmFtRGVjb3JhdG9yczogQXJyYXk8eyBkZWNvcmF0b3I6IFBhcmFtZXRlckRlY29yYXRvcjsgaW5kZXg6IG51bWJlciB9PlxuICApIHtcbiAgICByZXR1cm4geyBtZXRob2ROYW1lLCBoYW5kbGVyLCBkZWNvcmF0b3JzLCBwYXJhbURlY29yYXRvcnMgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHN0YXRlbWVudE1ldGhvZE5hbWUocGF0aDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCBmaXJzdFNlZ21lbnQgPSBwYXRoLnNwbGl0KFwiL1wiKVswXTtcbiAgICByZXR1cm4gZmlyc3RTZWdtZW50O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgc3RhdGVtZW50U2hvcnRjdXRQYXJhbXMoXG4gICAgcGF0aDogc3RyaW5nXG4gICk6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT4ge1xuICAgIGNvbnN0IHNlZ21lbnRzID0gcGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKTtcbiAgICBjb25zdCBwYXJhbXM6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT4gPSBbXTtcbiAgICBzZWdtZW50cy5mb3JFYWNoKChzZWcsIGkpID0+IHtcbiAgICAgIGNvbnN0IG5hbWUgPSBzZWcucmVwbGFjZShcIjpcIiwgXCJcIik7XG4gICAgICBwYXJhbXMucHVzaCh7IGRlY29yYXRvcjogUGFyYW0obmFtZSkgYXMgYW55LCBpbmRleDogaSB9KTtcbiAgICB9KTtcbiAgICBpZiAoXG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJsaXN0QnkvXCIpIHx8XG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJwYWdpbmF0ZUJ5L1wiKSB8fFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwiZmluZC9cIikgfHxcbiAgICAgIHBhdGguc3RhcnRzV2l0aChcInBhZ2UvXCIpXG4gICAgKSB7XG4gICAgICBwYXJhbXMucHVzaCh7IGRlY29yYXRvcjogRGVjYWZRdWVyeSgpIGFzIGFueSwgaW5kZXg6IHNlZ21lbnRzLmxlbmd0aCB9KTtcbiAgICB9XG4gICAgcmV0dXJuIHBhcmFtcztcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNvbXBsZXhRdWVyeVBhcmFtcyhcbiAgICBwYXRoOiBzdHJpbmdcbiAgKTogQXJyYXk8eyBkZWNvcmF0b3I6IFBhcmFtZXRlckRlY29yYXRvcjsgaW5kZXg6IG51bWJlciB9PiB7XG4gICAgY29uc3Qgc2VnbWVudHMgPSBwYXRoLnNwbGl0KFwiL1wiKS5maWx0ZXIoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpO1xuICAgIGNvbnN0IHBhcmFtczogQXJyYXk8eyBkZWNvcmF0b3I6IFBhcmFtZXRlckRlY29yYXRvcjsgaW5kZXg6IG51bWJlciB9PiA9IFtdO1xuICAgIHNlZ21lbnRzLmZvckVhY2goKHNlZywgaSkgPT4ge1xuICAgICAgY29uc3QgbmFtZSA9IHNlZy5yZXBsYWNlKFwiOlwiLCBcIlwiKTtcbiAgICAgIHBhcmFtcy5wdXNoKHsgZGVjb3JhdG9yOiBQYXJhbShuYW1lKSBhcyBhbnksIGluZGV4OiBpIH0pO1xuICAgIH0pO1xuICAgIGlmIChwYXRoLnN0YXJ0c1dpdGgoXCJxdWVyeS9cIikpIHtcbiAgICAgIHBhcmFtcy5wdXNoKHsgZGVjb3JhdG9yOiBEZWNhZlF1ZXJ5KCkgYXMgYW55LCBpbmRleDogc2VnbWVudHMubGVuZ3RoIH0pO1xuICAgIH1cbiAgICByZXR1cm4gcGFyYW1zO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQ3JlYXRlSGFuZGxlcihcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIGNyZWF0ZShcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIGRhdGE6IGFueSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPE1vZGVsPGFueT4+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIE9wZXJhdGlvbktleXMuQ1JFQVRFLCB0cnVlKVxuICAgICAgKS5mb3IoY3JlYXRlKTtcbiAgICAgIGxvZy52ZXJib3NlKGBjcmVhdGluZyBuZXcgJHttb2RlbENsYXp6TmFtZX1gKTtcbiAgICAgIGxldCBjcmVhdGVkOiBNb2RlbDtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNyZWF0ZWQgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuY3JlYXRlKGRhdGEsIGN0eCk7XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIGNyZWF0ZSBuZXcgJHttb2RlbENsYXp6TmFtZX1gLCBlIGFzIEVycm9yKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKGBjcmVhdGVkIG5ldyAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7KGNyZWF0ZWQgYXMgYW55KVt0aGlzLnBrXX1gKTtcbiAgICAgIGlmIChyZXNwKSBjdHgudG9SZXNwb25zZShyZXNwKTtcbiAgICAgIHJldHVybiBjcmVhdGVkO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVCdWxrQ3JlYXRlSGFuZGxlcihcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIGNyZWF0ZUFsbChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIGRhdGE6IGFueVtdLFxuICAgICAgcmVzcD86IGFueVxuICAgICk6IFByb21pc2U8TW9kZWxbXT4ge1xuICAgICAgY29uc3QgeyBjdHgsIGxvZyB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgQnVsa0NydWRPcGVyYXRpb25LZXlzLkNSRUFURV9BTEwsIHRydWUpXG4gICAgICApLmZvcihjcmVhdGVBbGwpO1xuICAgICAgbG9nLnZlcmJvc2UoYGNyZWF0aW5nIG5ldyAke21vZGVsQ2xhenpOYW1lfWApO1xuICAgICAgbGV0IGNyZWF0ZWQ6IGFueVtdO1xuICAgICAgdHJ5IHtcbiAgICAgICAgY3JlYXRlZCA9IGF3YWl0IHRoaXMucGVyc2lzdGVuY2UoY3R4KS5jcmVhdGVBbGwoXG4gICAgICAgICAgZGF0YS5tYXAoKGQpID0+IG5ldyBNb2RlbENvbnN0cihkKSksXG4gICAgICAgICAgY3R4XG4gICAgICAgICk7XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIGNyZWF0ZSBuZXcgJHttb2RlbENsYXp6TmFtZX1gLCBlIGFzIEVycm9yKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKGBjcmVhdGVkIG5ldyAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7KGNyZWF0ZWQgYXMgYW55KVt0aGlzLnBrXX1gKTtcbiAgICAgIGlmIChyZXNwKSBjdHgudG9SZXNwb25zZShyZXNwKTtcbiAgICAgIHJldHVybiBjcmVhdGVkO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVCdWxrUmVhZEhhbmRsZXIobW9kZWxDbGF6ek5hbWU6IHN0cmluZykge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiByZWFkQWxsKFxuICAgICAgdGhpczogYW55LFxuICAgICAgaWRzOiBzdHJpbmdbXVxuICAgICk6IFByb21pc2U8TW9kZWxbXT4ge1xuICAgICAgY29uc3QgeyBjdHgsIGxvZyB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgQnVsa0NydWRPcGVyYXRpb25LZXlzLlJFQURfQUxMLCB0cnVlKVxuICAgICAgKS5mb3IocmVhZEFsbCk7XG4gICAgICBjb25zdCBub3JtYWxpemVkSWRzID0gQXJyYXkuaXNBcnJheShpZHMpID8gaWRzIDogW2lkc107XG4gICAgICBsZXQgcmVhZDogTW9kZWxbXTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGxvZy5kZWJ1ZyhgcmVhZGluZyAke25vcm1hbGl6ZWRJZHN9ICR7bW9kZWxDbGF6ek5hbWV9YCk7XG4gICAgICAgIHJlYWQgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkucmVhZEFsbChub3JtYWxpemVkSWRzIGFzIGFueSwgY3R4KTtcbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgbG9nLmVycm9yKGBGYWlsZWQgdG8gcmVhZCAke21vZGVsQ2xhenpOYW1lfWAsIGUgYXMgRXJyb3IpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgICAgbG9nLmluZm8oYHJlYWQgJHtyZWFkLmxlbmd0aH0gJHttb2RlbENsYXp6TmFtZX1gKTtcbiAgICAgIHJldHVybiByZWFkO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVCdWxrVXBkYXRlSGFuZGxlcihtb2RlbENsYXp6TmFtZTogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIHVwZGF0ZUFsbChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIGJvZHk6IGFueVtdLFxuICAgICAgcmVzcD86IGFueVxuICAgICk6IFByb21pc2U8YW55W10+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cy5VUERBVEVfQUxMLCB0cnVlKVxuICAgICAgKS5mb3IodXBkYXRlQWxsKTtcbiAgICAgIGxldCB1cGRhdGVkOiBhbnlbXTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGxvZy5pbmZvKGB1cGRhdGluZyAke2JvZHkubGVuZ3RofSAke21vZGVsQ2xhenpOYW1lfWApO1xuICAgICAgICB1cGRhdGVkID0gYXdhaXQgdGhpcy5wZXJzaXN0ZW5jZShjdHgpLnVwZGF0ZUFsbChib2R5LCBjdHgpO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gdXBkYXRlZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQnVsa0RlbGV0ZUhhbmRsZXIobW9kZWxDbGF6ek5hbWU6IHN0cmluZykge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiBkZWxldGVBbGwoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICBpZHM6IHN0cmluZ1tdLFxuICAgICAgcmVzcD86IGFueVxuICAgICk6IFByb21pc2U8TW9kZWxbXT4ge1xuICAgICAgY29uc3QgeyBjdHgsIGxvZyB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgQnVsa0NydWRPcGVyYXRpb25LZXlzLkRFTEVURV9BTEwsIHRydWUpXG4gICAgICApLmZvcihkZWxldGVBbGwpO1xuICAgICAgY29uc3Qgbm9ybWFsaXplZElkcyA9IEFycmF5LmlzQXJyYXkoaWRzKSA/IGlkcyA6IFtpZHNdO1xuICAgICAgbGV0IHJlYWQ6IE1vZGVsW107XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuZGVidWcoYGRlbGV0aW5nICR7bm9ybWFsaXplZElkcy5sZW5ndGh9ICR7bW9kZWxDbGF6ek5hbWV9YCk7XG4gICAgICAgIHJlYWQgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuZGVsZXRlQWxsKG5vcm1hbGl6ZWRJZHMsIGN0eCk7XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIGRlbGV0ZSAke21vZGVsQ2xhenpOYW1lfWAsIGUgYXMgRXJyb3IpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgICAgbG9nLmluZm8oYGRlbGV0ZWQgJHtyZWFkLmxlbmd0aH0gJHttb2RlbENsYXp6TmFtZX1gKTtcbiAgICAgIGlmIChyZXNwKSBjdHgudG9SZXNwb25zZShyZXNwKTtcbiAgICAgIHJldHVybiByZWFkO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVSZWFkSGFuZGxlcihcbiAgICBnZXRQSzogKC4uLnA6IEFycmF5PHN0cmluZyB8IG51bWJlcj4pID0+IHN0cmluZyxcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiByZWFkKFxuICAgICAgdGhpczogYW55LFxuICAgICAgcm91dGVQYXJhbXM6IGFueVxuICAgICk6IFByb21pc2U8TW9kZWw+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIE9wZXJhdGlvbktleXMuUkVBRCwgdHJ1ZSlcbiAgICAgICkuZm9yKHJlYWQpO1xuICAgICAgY29uc3QgaWQgPSBnZXRQSyguLi5yb3V0ZVBhcmFtcy52YWx1ZXNJbk9yZGVyKTtcbiAgICAgIGlmICh0eXBlb2YgaWQgPT09IFwidW5kZWZpbmVkXCIpXG4gICAgICAgIHRocm93IG5ldyBWYWxpZGF0aW9uRXJyb3IoYE5vICR7dGhpcy5wa30gcHJvdmlkZWRgKTtcbiAgICAgIGxldCByZWFkUmVzdWx0OiBNb2RlbDtcbiAgICAgIHRyeSB7XG4gICAgICAgIGxvZy5kZWJ1ZyhgcmVhZGluZyAke21vZGVsQ2xhenpOYW1lfSB3aXRoICR7dGhpcy5wa30gJHtpZH1gKTtcbiAgICAgICAgcmVhZFJlc3VsdCA9IGF3YWl0IHRoaXMucGVyc2lzdGVuY2UoY3R4KS5yZWFkKGlkLCBjdHgpO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byByZWFkICR7bW9kZWxDbGF6ek5hbWV9IHdpdGggaWQgJHtpZH1gLCBlIGFzIEVycm9yKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKGByZWFkICR7bW9kZWxDbGF6ek5hbWV9IHdpdGggaWQgJHsocmVhZFJlc3VsdCBhcyBhbnkpW3RoaXMucGtdfWApO1xuICAgICAgcmV0dXJuIHJlYWRSZXN1bHQ7XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZVVwZGF0ZUhhbmRsZXIoXG4gICAgZ2V0UEs6ICguLi5wOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PiBzdHJpbmcsXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiB1cGRhdGUoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICByb3V0ZVBhcmFtczogYW55LFxuICAgICAgYm9keTogYW55LFxuICAgICAgcmVzcD86IGFueVxuICAgICk6IFByb21pc2U8YW55PiB7XG4gICAgICBjb25zdCB7IGN0eCwgbG9nIH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBPcGVyYXRpb25LZXlzLlVQREFURSwgdHJ1ZSlcbiAgICAgICkuZm9yKHVwZGF0ZSk7XG4gICAgICBjb25zdCBpZCA9IGdldFBLKC4uLnJvdXRlUGFyYW1zLnZhbHVlc0luT3JkZXIpO1xuICAgICAgaWYgKHR5cGVvZiBpZCA9PT0gXCJ1bmRlZmluZWRcIilcbiAgICAgICAgdGhyb3cgbmV3IFZhbGlkYXRpb25FcnJvcihgTm8gJHt0aGlzLnBrfSBwcm92aWRlZGApO1xuICAgICAgbGV0IHVwZGF0ZWQ6IGFueTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGxvZy5pbmZvKGB1cGRhdGluZyAke21vZGVsQ2xhenpOYW1lfSB3aXRoICR7dGhpcy5wa30gJHtpZH1gKTtcbiAgICAgICAgY29uc3QgcGF5bG9hZCA9IEpTT04ucGFyc2UoSlNPTi5zdHJpbmdpZnkoYm9keSkpO1xuICAgICAgICB1cGRhdGVkID0gYXdhaXQgdGhpcy5wZXJzaXN0ZW5jZShjdHgpLnVwZGF0ZShcbiAgICAgICAgICBuZXcgTW9kZWxDb25zdHIoeyAuLi5wYXlsb2FkLCBbdGhpcy5wa106IGlkIH0pLFxuICAgICAgICAgIGN0eFxuICAgICAgICApO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gdXBkYXRlZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlRGVsZXRlSGFuZGxlcihcbiAgICBnZXRQSzogKC4uLnA6IEFycmF5PHN0cmluZyB8IG51bWJlcj4pID0+IHN0cmluZyxcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiByZW1vdmUoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICByb3V0ZVBhcmFtczogYW55LFxuICAgICAgcmVzcD86IGFueVxuICAgICk6IFByb21pc2U8TW9kZWw+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIE9wZXJhdGlvbktleXMuREVMRVRFLCB0cnVlKVxuICAgICAgKS5mb3IocmVtb3ZlKTtcbiAgICAgIGNvbnN0IGlkID0gZ2V0UEsoLi4ucm91dGVQYXJhbXMudmFsdWVzSW5PcmRlcik7XG4gICAgICBpZiAodHlwZW9mIGlkID09PSBcInVuZGVmaW5lZFwiKVxuICAgICAgICB0aHJvdyBuZXcgVmFsaWRhdGlvbkVycm9yKGBObyAke3RoaXMucGt9IHByb3ZpZGVkYCk7XG4gICAgICBsZXQgZGVsOiBNb2RlbDtcbiAgICAgIHRyeSB7XG4gICAgICAgIGxvZy5kZWJ1ZyhgZGVsZXRpbmcgJHttb2RlbENsYXp6TmFtZX0gd2l0aCAke3RoaXMucGt9ICR7aWR9YCk7XG4gICAgICAgIGRlbCA9IGF3YWl0IHRoaXMucGVyc2lzdGVuY2UoY3R4KS5kZWxldGUoaWQsIGN0eCk7XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIGRlbGV0ZSAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7aWR9YCwgZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBsb2cuaW5mbyhgZGVsZXRlZCAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7aWR9YCk7XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gZGVsO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVTdGF0ZW1lbnRIYW5kbGVyKG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gc3RhdGVtZW50KFxuICAgICAgdGhpczogYW55LFxuICAgICAgbmFtZTogc3RyaW5nLFxuICAgICAgYXJnczogKHN0cmluZyB8IG51bWJlcilbXSxcbiAgICAgIGRldGFpbHM6IERpcmVjdGlvbkxpbWl0T2Zmc2V0XG4gICAgKTogUHJvbWlzZTxhbnk+IHtcbiAgICAgIGNvbnN0IHsgY3R4IH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBQZXJzaXN0ZW5jZUtleXMuU1RBVEVNRU5ULCB0cnVlKVxuICAgICAgKS5mb3Ioc3RhdGVtZW50KTtcbiAgICAgIGNvbnN0IHsgZGlyZWN0aW9uLCBvZmZzZXQsIGxpbWl0LCBib29rbWFyayB9ID0gZGV0YWlscztcbiAgICAgIGFyZ3MgPSBhcmdzLm1hcChcbiAgICAgICAgKGEpID0+ICh0eXBlb2YgYSA9PT0gXCJzdHJpbmdcIiA/IHBhcnNlSW50KGEgYXMgc3RyaW5nKSA6IGEpIHx8IGFcbiAgICAgICkgYXMgYW55W107XG4gICAgICBjb25zdCBwYXRoRGlyZWN0aW9uID0gYXJncy5sZW5ndGggPiAxID8gYXJnc1sxXSA6IHVuZGVmaW5lZDtcbiAgICAgIGNvbnN0IHJlc29sdmVkRGlyZWN0aW9uID0gKGRpcmVjdGlvbiA/PyBwYXRoRGlyZWN0aW9uKSBhc1xuICAgICAgICB8IHN0cmluZ1xuICAgICAgICB8IHVuZGVmaW5lZDtcbiAgICAgIGlmIChyZXNvbHZlZERpcmVjdGlvbiAmJiBhcmdzLmxlbmd0aCA+IDEpIGFyZ3NbMV0gPSByZXNvbHZlZERpcmVjdGlvbjtcbiAgICAgIHN3aXRjaCAobmFtZSkge1xuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5EOlxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5EX0JZOlxuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5MSVNUX0JZOlxuICAgICAgICAgIGFyZ3MucHVzaChkaXJlY3Rpb24gYXMgc3RyaW5nKTtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRTpcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRV9CWTpcbiAgICAgICAgICBhcmdzID0gW1xuICAgICAgICAgICAgYXJnc1swXSxcbiAgICAgICAgICAgIHJlc29sdmVkRGlyZWN0aW9uIGFzIGFueSxcbiAgICAgICAgICAgIHsgbGltaXQsIG9mZnNldCwgYm9va21hcmsgfSxcbiAgICAgICAgICBdO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5EX09ORV9CWTpcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuQ09VTlRfT0Y6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLk1BWF9PRjpcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuTUlOX09GOlxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5BVkdfT0Y6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlNVTV9PRjpcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRElTVElOQ1RfT0Y6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkdST1VQX09GOlxuICAgICAgICAgIGJyZWFrO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UoY3R4KS5zdGF0ZW1lbnQobmFtZSwgLi4uYXJncywgY3R4KTtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlU3RhdGVtZW50U2hvcnRjdXRIYW5kbGVyKFxuICAgIHN0YXRlbWVudEtleTogc3RyaW5nLFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIHN0YXRlbWVudFNob3J0Y3V0KFxuICAgICAgdGhpczogYW55LFxuICAgICAgLi4uYXJnczogYW55W11cbiAgICApOiBQcm9taXNlPGFueT4ge1xuICAgICAgY29uc3QgeyBjdHggfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIHN0YXRlbWVudEtleSwgdHJ1ZSlcbiAgICAgICkuZm9yKHN0YXRlbWVudFNob3J0Y3V0KTtcblxuICAgICAgc3dpdGNoIChzdGF0ZW1lbnRLZXkpIHtcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuTElTVF9CWToge1xuICAgICAgICAgIGNvbnN0IFtrZXksIGRldGFpbHNdID0gYXJncztcbiAgICAgICAgICByZXR1cm4gdGhpcy5wZXJzaXN0ZW5jZShjdHgpLmxpc3RCeShcbiAgICAgICAgICAgIGtleSxcbiAgICAgICAgICAgIChkZXRhaWxzIGFzIGFueSk/LmRpcmVjdGlvbiBhcyBPcmRlckRpcmVjdGlvbixcbiAgICAgICAgICAgIGN0eFxuICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRV9CWToge1xuICAgICAgICAgIGNvbnN0IFtrZXksIHBhZ2UsIGRldGFpbHNdID0gYXJncztcbiAgICAgICAgICByZXR1cm4gdGhpcy5wZXJzaXN0ZW5jZShjdHgpLnBhZ2luYXRlQnkoXG4gICAgICAgICAgICBrZXksXG4gICAgICAgICAgICAoZGV0YWlscyBhcyBhbnkpPy5kaXJlY3Rpb24gYXMgT3JkZXJEaXJlY3Rpb24sXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIGxpbWl0OiAoZGV0YWlscyBhcyBhbnkpPy5saW1pdCxcbiAgICAgICAgICAgICAgb2Zmc2V0OiAoZGV0YWlscyBhcyBhbnkpPy5vZmZzZXQsXG4gICAgICAgICAgICAgIHBhZ2UsXG4gICAgICAgICAgICB9IGFzIGFueSxcbiAgICAgICAgICAgIGN0eFxuICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORDoge1xuICAgICAgICAgIGNvbnN0IFt2YWx1ZSwgZGV0YWlsc10gPSBhcmdzO1xuICAgICAgICAgIGNvbnN0IGRpcmVjdGlvbiA9XG4gICAgICAgICAgICAoZGV0YWlscyBhcyBhbnkpPy5kaXJlY3Rpb24gPz8gT3JkZXJEaXJlY3Rpb24uQVNDO1xuICAgICAgICAgIGNvbnN0IHBlcnNpc3RlbmNlID0gdGhpcy5wZXJzaXN0ZW5jZShjdHgpO1xuICAgICAgICAgIGlmICh0eXBlb2YgcGVyc2lzdGVuY2UuZmluZCA9PT0gXCJmdW5jdGlvblwiKVxuICAgICAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlLmZpbmQodmFsdWUsIGRpcmVjdGlvbiwgY3R4KTtcbiAgICAgICAgICByZXR1cm4gcGVyc2lzdGVuY2Uuc3RhdGVtZW50KFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5ELCB2YWx1ZSwgZGlyZWN0aW9uLCBjdHgpO1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0U6IHtcbiAgICAgICAgICBjb25zdCBbdmFsdWUsIGRldGFpbHNdID0gYXJncztcbiAgICAgICAgICBjb25zdCByZWYgPSB7XG4gICAgICAgICAgICBvZmZzZXQ6IChkZXRhaWxzIGFzIGFueSk/Lm9mZnNldCA/PyAxLFxuICAgICAgICAgICAgbGltaXQ6IChkZXRhaWxzIGFzIGFueSk/LmxpbWl0ID8/IDEwLFxuICAgICAgICAgICAgYm9va21hcms6IChkZXRhaWxzIGFzIGFueSk/LmJvb2ttYXJrLFxuICAgICAgICAgIH07XG4gICAgICAgICAgY29uc3QgcGVyc2lzdGVuY2UgPSB0aGlzLnBlcnNpc3RlbmNlKGN0eCk7XG4gICAgICAgICAgY29uc3QgZGlyZWN0aW9uID0gKGRldGFpbHMgYXMgYW55KT8uZGlyZWN0aW9uID8/IE9yZGVyRGlyZWN0aW9uLkFTQztcbiAgICAgICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlLnBhZ2UgPT09IFwiZnVuY3Rpb25cIilcbiAgICAgICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZS5wYWdlKHZhbHVlLCBkaXJlY3Rpb24sIHJlZiwgY3R4KTtcbiAgICAgICAgICByZXR1cm4gcGVyc2lzdGVuY2Uuc3RhdGVtZW50KFByZXBhcmVkU3RhdGVtZW50S2V5cy5QQUdFLCB2YWx1ZSwgZGlyZWN0aW9uLCByZWYsIGN0eCk7XG4gICAgICAgIH1cbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9PTkVfQlk6IHtcbiAgICAgICAgICBjb25zdCBba2V5LCB2YWx1ZV0gPSBhcmdzO1xuICAgICAgICAgIHJldHVybiB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuZmluZE9uZUJ5KGtleSwgdmFsdWUsIGN0eCk7XG4gICAgICAgIH1cbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9CWToge1xuICAgICAgICAgIGNvbnN0IFtrZXksIHZhbHVlXSA9IGFyZ3M7XG4gICAgICAgICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UoY3R4KVxuICAgICAgICAgICAgLmZvcihjdHgudG9PdmVycmlkZXMoKSlcbiAgICAgICAgICAgIC5maW5kQnkoa2V5LCB2YWx1ZSwgY3R4KTtcbiAgICAgICAgfVxuICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkNPVU5UX09GIHx8XG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5NQVhfT0YgfHxcbiAgICAgICAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLk1JTl9PRiB8fFxuICAgICAgICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuQVZHX09GIHx8XG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5TVU1fT0YgfHxcbiAgICAgICAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkRJU1RJTkNUX09GIHx8XG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5HUk9VUF9PRlxuICAgICAgICAgICkge1xuICAgICAgICAgICAgY29uc3QgW2ZpZWxkXSA9IGFyZ3M7XG4gICAgICAgICAgICByZXR1cm4gdGhpcy5wZXJzaXN0ZW5jZShjdHgpLnN0YXRlbWVudChzdGF0ZW1lbnRLZXksIGZpZWxkLCBjdHgpO1xuICAgICAgICAgIH1cbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFVua25vd24gc3RhdGVtZW50OiAke3N0YXRlbWVudEtleX1gKTtcbiAgICAgIH1cbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQ29tcGxleFF1ZXJ5SGFuZGxlcihtZXRob2ROYW1lOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gY29tcGxleFF1ZXJ5KFxuICAgICAgdGhpczogYW55LFxuICAgICAgLi4uYXJnczogYW55W11cbiAgICApOiBQcm9taXNlPGFueT4ge1xuICAgICAgY29uc3QgbG9nOiBhbnkgPSB0aGlzLmxvZz8uZm9yPy4oY29tcGxleFF1ZXJ5KTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGlmIChsb2cpIGxvZy5kZWJ1ZyhgSW52b2tpbmcgY3VzdG9tIHF1ZXJ5IFwiJHttZXRob2ROYW1lfVwiYCk7XG4gICAgICAgIGNvbnN0IHsgY3R4IH0gPSAoXG4gICAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIG1ldGhvZE5hbWUsIHRydWUpXG4gICAgICAgICkuZm9yKGNvbXBsZXhRdWVyeSk7XG4gICAgICAgIGNvbnN0IHBlcnNpc3RlbmNlID0gdGhpcy5wZXJzaXN0ZW5jZShjdHgpO1xuICAgICAgICBjb25zdCBzcHJlYWRBcmdzID0gRnJvbU1vZGVsQ29udHJvbGxlci5leHRyYWN0UXVlcnlBcmdzKGFyZ3MpO1xuICAgICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlW21ldGhvZE5hbWVdID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgICByZXR1cm4gcGVyc2lzdGVuY2VbbWV0aG9kTmFtZV0oLi4uc3ByZWFkQXJncywgY3R4KTtcbiAgICAgICAgfVxuICAgICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlLnN0YXRlbWVudCA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlLnN0YXRlbWVudChtZXRob2ROYW1lLCAuLi5zcHJlYWRBcmdzLCBjdHgpO1xuICAgICAgICB9XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICBgUGVyc2lzdGVuY2UgbWV0aG9kIFwiJHttZXRob2ROYW1lfVwiIG5vdCBmb3VuZCBvbiAke3BlcnNpc3RlbmNlPy5jb25zdHJ1Y3Rvcj8ubmFtZX1gXG4gICAgICAgICk7XG4gICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgaWYgKGxvZykgbG9nLmVycm9yKGBDdXN0b20gcXVlcnkgXCIke21ldGhvZE5hbWV9XCIgZmFpbGVkYCwgZSk7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZUN1c3RvbVJvdXRlSGFuZGxlcihtZXRob2ROYW1lOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gY3VzdG9tUm91dGUoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICAuLi5hcmdzOiBhbnlbXVxuICAgICk6IFByb21pc2U8YW55PiB7XG4gICAgICBjb25zdCBsb2c6IGFueSA9IHRoaXMubG9nPy5mb3I/LihjdXN0b21Sb3V0ZSk7XG4gICAgICBjb25zdCB7IGN0eCB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgbWV0aG9kTmFtZSwgdHJ1ZSlcbiAgICAgICkuZm9yKGN1c3RvbVJvdXRlKTtcbiAgICAgIGNvbnN0IHBlcnNpc3RlbmNlID0gdGhpcy5wZXJzaXN0ZW5jZShjdHgpO1xuICAgICAgY29uc3Qgc3ByZWFkQXJncyA9IEZyb21Nb2RlbENvbnRyb2xsZXIuZXh0cmFjdFF1ZXJ5QXJncyhhcmdzKTtcblxuICAgICAgLy8gVHJ5IHRoZSBwZXJzaXN0ZW5jZSBkaXJlY3RseSAod29ya3Mgd2hlbiBpdCdzIGEgY3VzdG9tIFJlcG9zaXRvcnkpXG4gICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlW21ldGhvZE5hbWVdID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlW21ldGhvZE5hbWVdKC4uLnNwcmVhZEFyZ3MsIGN0eCk7XG4gICAgICB9XG4gICAgICAvLyBXaGVuIHBlcnNpc3RlbmNlIGlzIGEgTW9kZWxTZXJ2aWNlLCB0aGUgbWV0aG9kIGxpdmVzIG9uIHRoZSB1bmRlcmx5aW5nIHJlcG9cbiAgICAgIGlmIChwZXJzaXN0ZW5jZT8ucmVwbyAmJiB0eXBlb2YgcGVyc2lzdGVuY2UucmVwb1ttZXRob2ROYW1lXSA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZS5yZXBvW21ldGhvZE5hbWVdKC4uLnNwcmVhZEFyZ3MsIGN0eCk7XG4gICAgICB9XG4gICAgICAvLyBGYWxsIGJhY2sgdG8gc3RhdGVtZW50IGdhdGV3YXlcbiAgICAgIGlmICh0eXBlb2YgcGVyc2lzdGVuY2Uuc3RhdGVtZW50ID09PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlLnN0YXRlbWVudChtZXRob2ROYW1lLCAuLi5zcHJlYWRBcmdzLCBjdHgpO1xuICAgICAgfVxuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgTWV0aG9kIFwiJHttZXRob2ROYW1lfVwiIG5vdCBmb3VuZCBvbiAke3BlcnNpc3RlbmNlPy5jb25zdHJ1Y3Rvcj8ubmFtZX0gb3IgaXRzIHJlcG9gXG4gICAgICApO1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBleHRyYWN0UXVlcnlBcmdzKGFyZ3M6IGFueVtdKTogYW55W10ge1xuICAgIGlmIChhcmdzLmxlbmd0aCA9PT0gMCkgcmV0dXJuIGFyZ3M7XG4gICAgY29uc3QgbGFzdCA9IGFyZ3NbYXJncy5sZW5ndGggLSAxXTtcbiAgICBpZiAoXG4gICAgICBsYXN0ICYmXG4gICAgICB0eXBlb2YgbGFzdCA9PT0gXCJvYmplY3RcIiAmJlxuICAgICAgIUFycmF5LmlzQXJyYXkobGFzdClcbiAgICApIHtcbiAgICAgIGNvbnN0IHF1ZXJ5T2JqID0gYXJncy5wb3AoKTtcbiAgICAgIGNvbnN0IGhhc0RpcmVjdGlvbiA9IHF1ZXJ5T2JqLmRpcmVjdGlvbiAhPT0gdW5kZWZpbmVkO1xuICAgICAgY29uc3QgaGFzTGltaXQgPSBxdWVyeU9iai5saW1pdCAhPT0gdW5kZWZpbmVkO1xuICAgICAgY29uc3QgaGFzT2Zmc2V0ID0gcXVlcnlPYmoub2Zmc2V0ICE9PSB1bmRlZmluZWQ7XG4gICAgICBpZiAoIWhhc0RpcmVjdGlvbiAmJiAhaGFzTGltaXQgJiYgIWhhc09mZnNldCkgcmV0dXJuIGFyZ3M7XG4gICAgICBjb25zdCBleHRyYXM6IGFueVtdID0gW107XG4gICAgICBpZiAoaGFzRGlyZWN0aW9uKSBleHRyYXMucHVzaChxdWVyeU9iai5kaXJlY3Rpb24pO1xuICAgICAgZWxzZSBpZiAoaGFzTGltaXQgfHwgaGFzT2Zmc2V0KSBleHRyYXMucHVzaCh1bmRlZmluZWQpO1xuICAgICAgaWYgKGhhc0xpbWl0KSBleHRyYXMucHVzaChxdWVyeU9iai5saW1pdCk7XG4gICAgICBpZiAoaGFzT2Zmc2V0KSBleHRyYXMucHVzaChxdWVyeU9iai5vZmZzZXQpO1xuICAgICAgcmV0dXJuIFsuLi5hcmdzLCAuLi5leHRyYXNdO1xuICAgIH1cbiAgICByZXR1cm4gYXJncztcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZUNyZWF0ZURlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICk6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4ge1xuICAgIHJldHVybiBbXG4gICAgICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwoTW9kZWxDb25zdHIsIFwiUE9TVFwiKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBDcmVhdGUgYSBuZXcgJHttb2RlbENsYXp6TmFtZX0uYCB9KSxcbiAgICAgIEFwaUJvZHkoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYFBheWxvYWQgZm9yICR7bW9kZWxDbGF6ek5hbWV9YCxcbiAgICAgICAgdHlwZTogRHRvRm9yKE9wZXJhdGlvbktleXMuQ1JFQVRFLCBNb2RlbENvbnN0ciksXG4gICAgICB9KSxcbiAgICAgIEFwaUNyZWF0ZWRSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gY3JlYXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaUJhZFJlcXVlc3RSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlBheWxvYWQgdmFsaWRhdGlvbiBmYWlsZWQuXCIgfSksXG4gICAgICBBcGlVbnByb2Nlc3NhYmxlRW50aXR5UmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogXCJSZXBvc2l0b3J5IHJlamVjdGVkIHRoZSBwcm92aWRlZCBwYXlsb2FkLlwiLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGJ1bGtDcmVhdGVEZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIlBPU1RcIiwgXCJidWxrXCIpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYENyZWF0ZSBhIG5ldyAke21vZGVsQ2xhenpOYW1lfS5gIH0pLFxuICAgICAgQXBpQm9keSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgUGF5bG9hZCBmb3IgJHttb2RlbENsYXp6TmFtZX1gLFxuICAgICAgICBzY2hlbWE6IHtcbiAgICAgICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICAgICAgaXRlbXM6IHsgJHJlZjogZ2V0U2NoZW1hUGF0aChNb2RlbENvbnN0cikgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pLFxuICAgICAgQXBpQ3JlYXRlZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSBjcmVhdGVkIHN1Y2Nlc3NmdWxseS5gLFxuICAgICAgICBzY2hlbWE6IHtcbiAgICAgICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICAgICAgaXRlbXM6IHsgJHJlZjogZ2V0U2NoZW1hUGF0aChNb2RlbENvbnN0cikgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pLFxuICAgICAgQXBpQmFkUmVxdWVzdFJlc3BvbnNlKHsgZGVzY3JpcHRpb246IFwiUGF5bG9hZCB2YWxpZGF0aW9uIGZhaWxlZC5cIiB9KSxcbiAgICAgIEFwaVVucHJvY2Vzc2FibGVFbnRpdHlSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBcIlJlcG9zaXRvcnkgcmVqZWN0ZWQgdGhlIHByb3ZpZGVkIHBheWxvYWQuXCIsXG4gICAgICB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgYnVsa1JlYWREZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIkdFVFwiLCBcImJ1bGtcIiksXG4gICAgICBBcGlPcGVyYXRpb24oeyBzdW1tYXJ5OiBgUmV0cmlldmUgJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkcyBieSBpZHMuYCB9KSxcbiAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJpZHNcIiwgcmVxdWlyZWQ6IHRydWUsIHR5cGU6IFwiYXJyYXlcIiB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IHJldHJpZXZlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgIGl0ZW1zOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICAgIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgYnVsa1VwZGF0ZURlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXVxuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIlBVVFwiLCBcImJ1bGtcIiksXG4gICAgICBBcGlQYXJhbXNGcm9tTW9kZWwoYXBpUHJvcGVydGllcyksXG4gICAgICBBcGlPcGVyYXRpb24oe1xuICAgICAgICBzdW1tYXJ5OiBgUmVwbGFjZSBleGlzdGluZyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmRzIHdpdGggbmV3IHBheWxvYWRzLmAsXG4gICAgICB9KSxcbiAgICAgIEFwaUJvZHkoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYFBheWxvYWQgZm9yIHJlcGxhY2luZyBleGlzdGluZyByZWNvcmRzIG9mICR7bW9kZWxDbGF6ek5hbWV9YCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgICRyZWY6IGdldFNjaGVtYVBhdGgoRHRvRm9yKE9wZXJhdGlvbktleXMuVVBEQVRFLCBNb2RlbENvbnN0cikpLFxuICAgICAgICB9LFxuICAgICAgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSB1cGRhdGVkIHN1Y2Nlc3NmdWxseS5gLFxuICAgICAgICBzY2hlbWE6IHtcbiAgICAgICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICAgICAgaXRlbXM6IHsgJHJlZjogZ2V0U2NoZW1hUGF0aChNb2RlbENvbnN0cikgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pLFxuICAgICAgQXBpTm90Rm91bmRSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgTm8gJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIG1hdGNoZXMgdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuYCxcbiAgICAgIH0pLFxuICAgICAgQXBpQmFkUmVxdWVzdFJlc3BvbnNlKHsgZGVzY3JpcHRpb246IFwiUGF5bG9hZCB2YWxpZGF0aW9uIGZhaWxlZC5cIiB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgYnVsa0RlbGV0ZURlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXVxuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIkRFTEVURVwiLCBcImJ1bGtcIiksXG4gICAgICBBcGlQYXJhbXNGcm9tTW9kZWwoYXBpUHJvcGVydGllcyksXG4gICAgICBBcGlPcGVyYXRpb24oeyBzdW1tYXJ5OiBgRGVsZXRlICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZHMgYnkgaWRzLmAgfSksXG4gICAgICBBcGlRdWVyeSh7IG5hbWU6IFwiaWRzXCIsIHJlcXVpcmVkOiB0cnVlLCB0eXBlOiBcImFycmF5XCIgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSBkZWxldGVkIHN1Y2Nlc3NmdWxseS5gLFxuICAgICAgICBzY2hlbWE6IHtcbiAgICAgICAgICB0eXBlOiBcImFycmF5XCIsXG4gICAgICAgICAgaXRlbXM6IHsgJHJlZjogZ2V0U2NoZW1hUGF0aChNb2RlbENvbnN0cikgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pLFxuICAgICAgQXBpTm90Rm91bmRSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgTm8gJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIG1hdGNoZXMgdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuYCxcbiAgICAgIH0pLFxuICAgIF07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyByZWFkRGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcsXG4gICAgYXBpUHJvcGVydGllczogRGVjYWZBcGlQcm9wZXJ0eVtdLFxuICAgIHBrUGF0aDogc3RyaW5nXG4gICk6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4ge1xuICAgIHJldHVybiBbXG4gICAgICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwoTW9kZWxDb25zdHIsIFwiR0VUXCIsIHBrUGF0aCksXG4gICAgICBBcGlQYXJhbXNGcm9tTW9kZWwoYXBpUHJvcGVydGllcyksXG4gICAgICBBcGlPcGVyYXRpb24oeyBzdW1tYXJ5OiBgUmV0cmlldmUgYSAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgYnkgaWQuYCB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IHJldHJpZXZlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgdXBkYXRlRGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcsXG4gICAgYXBpUHJvcGVydGllczogRGVjYWZBcGlQcm9wZXJ0eVtdLFxuICAgIHBrUGF0aDogc3RyaW5nXG4gICk6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4ge1xuICAgIHJldHVybiBbXG4gICAgICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwoTW9kZWxDb25zdHIsIFwiUFVUXCIsIHBrUGF0aCksXG4gICAgICBBcGlQYXJhbXNGcm9tTW9kZWwoYXBpUHJvcGVydGllcyksXG4gICAgICBBcGlPcGVyYXRpb24oe1xuICAgICAgICBzdW1tYXJ5OiBgUmVwbGFjZSBhbiBleGlzdGluZyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgd2l0aCBhIG5ldyBwYXlsb2FkLmAsXG4gICAgICB9KSxcbiAgICAgIEFwaUJvZHkoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYFBheWxvYWQgZm9yIHJlcGxhY2luZyBhbiBleGlzdGluZyByZWNvcmQgb2YgJHttb2RlbENsYXp6TmFtZX1gLFxuICAgICAgICB0eXBlOiBEdG9Gb3IoT3BlcmF0aW9uS2V5cy5VUERBVEUsIE1vZGVsQ29uc3RyKSxcbiAgICAgIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gdXBkYXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICAgIEFwaUJhZFJlcXVlc3RSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlBheWxvYWQgdmFsaWRhdGlvbiBmYWlsZWQuXCIgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGRlbGV0ZURlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXSxcbiAgICBwa1BhdGg6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIkRFTEVURVwiLCBwa1BhdGgpLFxuICAgICAgQXBpUGFyYW1zRnJvbU1vZGVsKGFwaVByb3BlcnRpZXMpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYERlbGV0ZSBhICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBieSBpZC5gIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gZGVsZXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgc3RhdGVtZW50RGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJHRVRcIiwgXCJzdGF0ZW1lbnQvOm1ldGhvZC8qYXJnc1wiKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7XG4gICAgICAgIHN1bW1hcnk6IGBFeGVjdXRlcyBhIHByZXBhcmVkIHN0YXRlbWVudCBvbiAke21vZGVsQ2xhenpOYW1lfS5gLFxuICAgICAgfSksXG4gICAgICBBcGlQYXJhbSh7IG5hbWU6IFwibWV0aG9kXCIsIGRlc2NyaXB0aW9uOiBcInRoZSBwcmVwYXJlZCBzdGF0ZW1lbnQgdG8gZXhlY3V0ZVwiIH0pLFxuICAgICAgQXBpUGFyYW0oe1xuICAgICAgICBuYW1lOiBcImFyZ3NcIixcbiAgICAgICAgZGVzY3JpcHRpb246IFwiY29uY2F0ZW5hdGVkIGxpc3Qgb2YgYXJndW1lbnRzIHRoZSBwcmVwYXJlZCBzdGF0ZW1lbnQgY2FuIGFjY2VwdFwiLFxuICAgICAgfSksXG4gICAgICBBcGlRdWVyeSh7XG4gICAgICAgIG5hbWU6IFwiZGlyZWN0aW9uXCIsXG4gICAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgICAgICBlbnVtOiBPcmRlckRpcmVjdGlvbixcbiAgICAgICAgZGVzY3JpcHRpb246IFwidGhlIHNvcnQgb3JkZXIgd2hlbiBhcHBsaWNhYmxlXCIsXG4gICAgICB9KSxcbiAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJsaW1pdFwiLCByZXF1aXJlZDogdHJ1ZSwgZGVzY3JpcHRpb246IFwibGltaXQgb3IgcGFnZSBzaXplIHdoZW4gYXBwbGljYWJsZVwiIH0pLFxuICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcIm9mZnNldFwiLCByZXF1aXJlZDogdHJ1ZSwgZGVzY3JpcHRpb246IFwib2Zmc2V0IG9yIGJvb2ttYXJrIHdoZW4gYXBwbGljYWJsZVwiIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gbGlzdGVkIGZvdW5kLmAgfSksXG4gICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBObyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5gLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHN0YXRlbWVudFNob3J0Y3V0RGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcsXG4gICAgcGF0aDogc3RyaW5nLFxuICAgIHN0YXRlbWVudEtleTogc3RyaW5nXG4gICk6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4ge1xuICAgIGNvbnN0IGJhc2U6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4gPSBbXG4gICAgICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwoTW9kZWxDb25zdHIsIFwiR0VUXCIsIHBhdGgpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYFJldHJpZXZlICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZHMuYCB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IHJldHJpZXZlZCBzdWNjZXNzZnVsbHkuYCB9KSxcbiAgICBdO1xuXG4gICAgY29uc3Qgc2VnbWVudHMgPSBwYXRoLnNwbGl0KFwiL1wiKS5maWx0ZXIoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpO1xuICAgIHNlZ21lbnRzLmZvckVhY2goKHNlZykgPT4ge1xuICAgICAgY29uc3QgbmFtZSA9IHNlZy5yZXBsYWNlKFwiOlwiLCBcIlwiKTtcbiAgICAgIGJhc2UucHVzaChBcGlQYXJhbSh7IG5hbWUsIGRlc2NyaXB0aW9uOiBgVGhlICR7bmFtZX0gcGFyYW1ldGVyYCB9KSk7XG4gICAgfSk7XG5cbiAgICBpZiAoXG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJsaXN0QnkvXCIpIHx8XG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJwYWdpbmF0ZUJ5L1wiKSB8fFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwiZmluZC9cIikgfHxcbiAgICAgIHBhdGguc3RhcnRzV2l0aChcInBhZ2UvXCIpXG4gICAgKSB7XG4gICAgICBiYXNlLnB1c2goXG4gICAgICAgIEFwaVF1ZXJ5KHtcbiAgICAgICAgICBuYW1lOiBcImRpcmVjdGlvblwiLFxuICAgICAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgICAgICAgIGVudW06IE9yZGVyRGlyZWN0aW9uLFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiBcInRoZSBzb3J0IG9yZGVyXCIsXG4gICAgICAgIH0pXG4gICAgICApO1xuICAgIH1cbiAgICBpZiAocGF0aC5zdGFydHNXaXRoKFwicGFnaW5hdGVCeS9cIikgfHwgcGF0aC5zdGFydHNXaXRoKFwicGFnZS9cIikpIHtcbiAgICAgIGJhc2UucHVzaChcbiAgICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcImxpbWl0XCIsIHJlcXVpcmVkOiBmYWxzZSwgZGVzY3JpcHRpb246IFwicGFnZSBzaXplXCIgfSksXG4gICAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJvZmZzZXRcIiwgcmVxdWlyZWQ6IGZhbHNlLCBkZXNjcmlwdGlvbjogXCJwYWdlIG51bWJlclwiIH0pLFxuICAgICAgICBBcGlRdWVyeSh7IG5hbWU6IFwiYm9va21hcmtcIiwgcmVxdWlyZWQ6IGZhbHNlLCBkZXNjcmlwdGlvbjogXCJib29rbWFyayBmb3IgY3Vyc29yIHBhZ2luYXRpb25cIiB9KVxuICAgICAgKTtcbiAgICB9XG5cbiAgICBpZiAocGF0aC5zdGFydHNXaXRoKFwiZmluZE9uZUJ5L1wiKSB8fCBwYXRoLnN0YXJ0c1dpdGgoXCJmaW5kQnkvXCIpKSB7XG4gICAgICBiYXNlLnB1c2goXG4gICAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICAgIGRlc2NyaXB0aW9uOiBgTm8gJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIG1hdGNoZXMgdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuYCxcbiAgICAgICAgfSlcbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKFxuICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuQ09VTlRfT0YgfHxcbiAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkFWR19PRiB8fFxuICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuU1VNX09GXG4gICAgKSB7XG4gICAgICBiYXNlLnB1c2goQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBgUmVzdWx0IGZvciAke21vZGVsQ2xhenpOYW1lfS5gLCB0eXBlOiBOdW1iZXIgfSkpO1xuICAgIH1cbiAgICBpZiAoc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRElTVElOQ1RfT0YpIHtcbiAgICAgIGJhc2UucHVzaChBcGlPa1Jlc3BvbnNlKHsgZGVzY3JpcHRpb246IGBEaXN0aW5jdCB2YWx1ZXMgZm9yICR7bW9kZWxDbGF6ek5hbWV9LmAsIHR5cGU6IFtTdHJpbmddIH0pKTtcbiAgICB9XG5cbiAgICByZXR1cm4gYmFzZTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGdldFF1ZXJ5RGVjb3JhdG9ycyhcbiAgICBtZXRob2ROYW1lOiBzdHJpbmcsXG4gICAgcm91dGVQYXRoOiBzdHJpbmcsXG4gICAgaHR0cFZlcmI6IHN0cmluZyxcbiAgICBpbmNsdWRlUXVlcnlQYXJhbXM6IGJvb2xlYW4gPSBmYWxzZVxuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICBjb25zdCBleHRyYWN0UGF0aFBhcmFtcyA9IChwOiBzdHJpbmcpOiBzdHJpbmdbXSA9PlxuICAgICAgcC5zcGxpdChcIi9cIikuZmlsdGVyKChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKS5tYXAoKHMpID0+IHMuc2xpY2UoMSkpO1xuXG4gICAgY29uc3QgYXBpUGF0aFBhcmFtcyA9IGV4dHJhY3RQYXRoUGFyYW1zKHJvdXRlUGF0aCkubWFwKChuYW1lKSA9PiAoe1xuICAgICAgbmFtZSxcbiAgICAgIGRlc2NyaXB0aW9uOiBgJHtuYW1lfSBwYXJhbWV0ZXIgZm9yIHRoZSBxdWVyeWAsXG4gICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICAgIHR5cGU6IFN0cmluZyxcbiAgICB9KSk7XG5cbiAgICBjb25zdCBkZWNvcmF0b3JzOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+ID0gW1xuICAgICAgLi4uYXBpUGF0aFBhcmFtcy5tYXAoKHApID0+IEFwaVBhcmFtKHApKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBSZXRyaWV2ZSByZWNvcmRzIHVzaW5nIFwiJHttZXRob2ROYW1lfVwiLmAgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHsgZGVzY3JpcHRpb246IFwiUmVzdWx0IHN1Y2Nlc3NmdWxseSByZXRyaWV2ZWQuXCIgfSksXG4gICAgICBBcGlOb0NvbnRlbnRSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIk5vIGNvbnRlbnQgcmV0dXJuZWQgYnkgdGhlIG1ldGhvZC5cIiB9KSxcbiAgICBdO1xuXG4gICAgaWYgKGh0dHBWZXJiID09PSBcIkdFVFwiICYmIGluY2x1ZGVRdWVyeVBhcmFtcykge1xuICAgICAgZGVjb3JhdG9ycy5wdXNoKFxuICAgICAgICBBcGlRdWVyeSh7XG4gICAgICAgICAgbmFtZTogXCJkaXJlY3Rpb25cIixcbiAgICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgICAgZW51bTogT3JkZXJEaXJlY3Rpb24sXG4gICAgICAgICAgZGVzY3JpcHRpb246IFwidGhlIHNvcnQgb3JkZXIgd2hlbiBhcHBsaWNhYmxlXCIsXG4gICAgICAgIH0pLFxuICAgICAgICBBcGlRdWVyeSh7IG5hbWU6IFwibGltaXRcIiwgcmVxdWlyZWQ6IGZhbHNlLCBkZXNjcmlwdGlvbjogXCJsaW1pdCBvciBwYWdlIHNpemVcIiB9KSxcbiAgICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcIm9mZnNldFwiLCByZXF1aXJlZDogZmFsc2UsIGRlc2NyaXB0aW9uOiBcIm9mZnNldCBvciBib29rbWFya1wiIH0pXG4gICAgICApO1xuICAgIH1cblxuICAgIHJldHVybiBkZWNvcmF0b3JzO1xuICB9XG59XG4iLCJpbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBNb2R1bGUsIFByb3ZpZGVyIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBZGFwdGVyLCBNb2RlbFNlcnZpY2UgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IExvZ2dpbmcgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7IEZyb21Nb2RlbENvbnRyb2xsZXIgfSBmcm9tIFwiLi9Gcm9tTW9kZWxDb250cm9sbGVyXCI7XG5pbXBvcnQgeyBEZWNhZk1vZHVsZU9wdGlvbnMgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB7IE1vZGVsLCBNb2RlbENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IERFQ0FGX0VYUE9TRSB9IGZyb20gXCIuLi9jb25zdGFudHNcIjtcbmltcG9ydCB0eXBlIHsgTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZyB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRNb2R1bGVGb3IoZmxhdm91cjogc3RyaW5nKSB7XG4gIEBNb2R1bGUoe30pXG4gIGNsYXNzIERlY2FmTW9kZWxNb2R1bGUge1xuICAgIHN0YXRpYyByZWFkb25seSBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZk1vZGVsTW9kdWxlLm5hbWUpLmZvcihmbGF2b3VyKTtcblxuICAgIHN0YXRpYyBjcmVhdGVNb2RlbFNlcnZpY2VzPFQgZXh0ZW5kcyBNb2RlbDxib29sZWFuPj4oXG4gICAgICBtb2RlbHM6IE1vZGVsQ29uc3RydWN0b3I8VD5bXVxuICAgICk6IFByb3ZpZGVyW10ge1xuICAgICAgcmV0dXJuIG1vZGVscy5tYXAoKG1vZGVsKSA9PiAoe1xuICAgICAgICBwcm92aWRlOiBgJHttb2RlbC5uYW1lfVNlcnZpY2VgLFxuICAgICAgICB1c2VGYWN0b3J5OiAoKSA9PiBNb2RlbFNlcnZpY2UuZm9yTW9kZWwobW9kZWwgYXMgYW55KSxcbiAgICAgIH0pKTtcbiAgICB9XG5cbiAgICBzdGF0aWMgaXNFeHBvc2VkKFxuICAgICAgbW9kZWw6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICAgIGV4cG9zdXJlPzogUmVjb3JkPHN0cmluZywgYm9vbGVhbiB8IHN0cmluZ1tdPlxuICAgICk6IGJvb2xlYW4ge1xuICAgICAgY29uc3Qgb3ZlcnJpZGUgPSBleHBvc3VyZT8uW21vZGVsLm5hbWVdO1xuICAgICAgY29uc3QgdmFsdWUgPVxuICAgICAgICB0eXBlb2Ygb3ZlcnJpZGUgIT09IFwidW5kZWZpbmVkXCJcbiAgICAgICAgICA/IG92ZXJyaWRlXG4gICAgICAgICAgOiBNZXRhZGF0YS5nZXQobW9kZWwsIE1ldGFkYXRhLmtleShERUNBRl9FWFBPU0UpKTtcblxuICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJ1bmRlZmluZWRcIikgcmV0dXJuIHRydWU7XG4gICAgICBpZiAodmFsdWUgPT09IHRydWUpIHJldHVybiB0cnVlO1xuICAgICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSByZXR1cm4gdmFsdWUuaW5jbHVkZXMoZmxhdm91cik7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgc3RhdGljIGZvclJvb3QoXG4gICAgICBmbGF2b3VyOiBzdHJpbmcsXG4gICAgICBvcHRpb25zOiBQYXJ0aWFsPERlY2FmTW9kdWxlT3B0aW9ucz4gPSB7fVxuICAgICk6IER5bmFtaWNNb2R1bGUge1xuICAgICAgY29uc3QgbG9nID0gdGhpcy5sb2cuZm9yKHRoaXMuZm9yUm9vdCk7XG4gICAgICBsb2cuaW5mbyhgR2VuZXJhdGluZyBjb250cm9sbGVycyBmb3IgZmxhdm91ci4uLmApO1xuXG4gICAgICBjb25zdCB0cmFja2VkTW9kZWxzID0gQWRhcHRlci5tb2RlbHMoZmxhdm91cikuZmlsdGVyKChtb2RlbCkgPT5cbiAgICAgICAgdGhpcy5pc0V4cG9zZWQobW9kZWwsIG9wdGlvbnMuY29udHJvbGxlckV4cG9zdXJlKVxuICAgICAgKTtcblxuICAgICAgbGV0IG1vZGVsU2VydmljZXM6IFByb3ZpZGVyW10gPSBbXTtcbiAgICAgIGlmIChvcHRpb25zLmF1dG9TZXJ2aWNlcykge1xuICAgICAgICBsb2cuaW5mbyhcIkF1dG8tc2VydmljZXMgZW5hYmxlZC4gSW5pdGlhbGl6aW5nIHNlcnZpY2UgZ2VuZXJhdGlvbi5cIik7XG4gICAgICAgIG1vZGVsU2VydmljZXMgPSB0aGlzLmNyZWF0ZU1vZGVsU2VydmljZXModHJhY2tlZE1vZGVscyk7XG4gICAgICAgIGxvZy5pbmZvKFxuICAgICAgICAgIGBBdXRvLXNlcnZpY2VzIGNvbXBsZXRlZC4gJHttb2RlbFNlcnZpY2VzLmxlbmd0aH0gc2VydmljZXMgaW5pdGlhbGl6ZWQuYFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBnbG9iYWxEZWZhdWx0czogUGFydGlhbDxNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnPiA9IHt9O1xuICAgICAgaWYgKG9wdGlvbnMuYWdncmVnYXRpb25zID09PSBmYWxzZSkge1xuICAgICAgICBnbG9iYWxEZWZhdWx0cy5hbGxvd0dyb3VwaW5nUXVlcmllcyA9IGZhbHNlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBjb250cm9sbGVycyA9IHRyYWNrZWRNb2RlbHMubWFwKChtb2RlbCkgPT5cbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGUobW9kZWwsIG9wdGlvbnMuY29udHJvbGxlckNvbmZpZywgZ2xvYmFsRGVmYXVsdHMpXG4gICAgICApO1xuICAgICAgbG9nLmluZm8oYEdlbmVyYXRlZCAke2NvbnRyb2xsZXJzLmxlbmd0aH0gY29udHJvbGxlcnNgKTtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbW9kdWxlOiBEZWNhZk1vZGVsTW9kdWxlLFxuICAgICAgICBjb250cm9sbGVycyxcbiAgICAgICAgcHJvdmlkZXJzOiBbXG4gICAgICAgICAgLi4ubW9kZWxTZXJ2aWNlcyxcbiAgICAgICAgXSxcbiAgICAgIH07XG4gICAgfVxuICB9XG4gIE9iamVjdC5hc3NpZ24oRGVjYWZNb2RlbE1vZHVsZSwgXCJuYW1lXCIsIHtcbiAgICB2YWx1ZTogYERlY2FmTW9kdWxlJHtmbGF2b3VyfWAsXG4gIH0pO1xuICByZXR1cm4gRGVjYWZNb2RlbE1vZHVsZTtcbn1cbiIsImltcG9ydCB7XG4gIERlZmF1bHRJbmplY3RhYmxlc0NvbmZpZyxcbiAgSW5qZWN0YWJsZUNvbmZpZyxcbiAgSW5qZWN0YWJsZXNLZXlzLFxuICBJbmplY3RPcHRpb25zLFxufSBmcm9tIFwiQGRlY2FmLXRzL2luamVjdGFibGUtZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgSW5qZWN0LCBJbmplY3RhYmxlLCBTY29wZSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IsIERlY29yYXRpb24sIERlY29yYXRpb25LZXlzIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBWYWxpZGF0aW9uS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IFBlcnNpc3RlbmNlS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgQXBpUHJvcGVydHkgfSBmcm9tIFwiLi9vdmVycmlkZXMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgQXV0aCB9IGZyb20gXCIuL2RlY2FmLW1vZGVsXCI7XG5cbkRlY29yYXRpb24uZm9yKEluamVjdGFibGVzS2V5cy5JTkpFQ1RBQkxFKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG5lc3RJbmplY3RhYmxlKFxuICAgICAgY2F0ZWdvcnk6IHN0cmluZyB8IENvbnN0cnVjdG9yLFxuICAgICAgY2ZnPzogSW5qZWN0YWJsZUNvbmZpZ1xuICAgICkge1xuICAgICAgY2ZnID1cbiAgICAgICAgY2ZnIHx8XG4gICAgICAgICh0eXBlb2YgY2F0ZWdvcnkgPT09IFwib2JqZWN0XCJcbiAgICAgICAgICA/IE9iamVjdC5hc3NpZ24oXG4gICAgICAgICAgICAgIGNhdGVnb3J5IGFzIFJlY29yZDxhbnksIGFueT4sXG4gICAgICAgICAgICAgIERlZmF1bHRJbmplY3RhYmxlc0NvbmZpZ1xuICAgICAgICAgICAgKVxuICAgICAgICAgIDogRGVmYXVsdEluamVjdGFibGVzQ29uZmlnKTtcbiAgICAgIHJldHVybiBJbmplY3RhYmxlKHtcbiAgICAgICAgc2NvcGU6IGNmZy5zaW5nbGV0b24gPyBTY29wZS5ERUZBVUxUIDogU2NvcGUuUkVRVUVTVCxcbiAgICAgICAgZHVyYWJsZTogY2ZnLnNpbmdsZXRvbiA/IHVuZGVmaW5lZCA6IHRydWUsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoSW5qZWN0YWJsZXNLZXlzLklOSkVDVClcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBuZXN0SW5qZWN0KFxuICAgICAgY2F0ZWdvcnk6IHN5bWJvbCB8IHN0cmluZyB8IENvbnN0cnVjdG9yLFxuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgICAgY2ZnOiBJbmplY3RPcHRpb25zXG4gICAgKSB7XG4gICAgICByZXR1cm4gZnVuY3Rpb24gaW5uZXJOZXN0SW5qZWN0KFxuICAgICAgICB0YXJnZXQ6IG9iamVjdCxcbiAgICAgICAgcHJvcGVydHlLZXk/OiBhbnksXG4gICAgICAgIGRlc2NyaXB0b3I/OiBhbnlcbiAgICAgICkge1xuICAgICAgICByZXR1cm4gSW5qZWN0KGNhdGVnb3J5IHx8ICh0YXJnZXQgYXMgQ29uc3RydWN0b3IpKShcbiAgICAgICAgICB0YXJnZXQsXG4gICAgICAgICAgcHJvcGVydHlLZXksXG4gICAgICAgICAgZGVzY3JpcHRvclxuICAgICAgICApO1xuICAgICAgfTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuUkVRVUlSRUQpXG4gIC5leHRlbmQoXG4gICAgQXBpUHJvcGVydHkoe1xuICAgICAgcmVxdWlyZWQ6IHRydWUsXG4gICAgfSlcbiAgKVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuTUFYKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG1heERlYyhtYXg6IG51bWJlcikge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHsgbWF4aW11bTogbWF4LCByZXF1aXJlZDogZmFsc2UgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLk1JTilcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBtaW5EZWMobWluOiBudW1iZXIpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IG1pbmltdW06IG1pbiwgcmVxdWlyZWQ6IGZhbHNlIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5NQVhfTEVOR1RIKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG1heExlbmd0aERlYyhtYXg6IG51bWJlcikge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHsgbWF4TGVuZ3RoOiBtYXgsIHJlcXVpcmVkOiBmYWxzZSB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuTUlOX0xFTkdUSClcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBtaW5MZW5ndGhEZWMobWluOiBudW1iZXIpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IG1pbkxlbmd0aDogbWluLCByZXF1aXJlZDogZmFsc2UgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLlRZUEUpXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gdHlwZURlYyhcbiAgICAgIHR5cGU6XG4gICAgICAgIHwgKENvbnN0cnVjdG9yIHwgKCgpID0+IENvbnN0cnVjdG9yKSlbXVxuICAgICAgICB8IENvbnN0cnVjdG9yXG4gICAgICAgIHwgKCgpID0+IENvbnN0cnVjdG9yKVxuICAgICkge1xuICAgICAgcmV0dXJuICh0YXJnZXQ6IG9iamVjdCwgcHJvcDogYW55KSA9PiB7XG4gICAgICAgIHR5cGUgPSBBcnJheS5pc0FycmF5KHR5cGUpID8gdHlwZVswXSA6IHR5cGU7XG4gICAgICAgIGlmICh0eXBlb2YgdHlwZSA9PT0gXCJmdW5jdGlvblwiICYmICF0eXBlLm5hbWUpXG4gICAgICAgICAgdHlwZSA9ICh0eXBlIGFzICgpID0+IENvbnN0cnVjdG9yKSgpO1xuICAgICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICAgIHR5cGU6IHR5cGUgYXMgYW55LFxuICAgICAgICAgIHJlcXVpcmVkOiBmYWxzZSxcbiAgICAgICAgfSkodGFyZ2V0LCBwcm9wKTtcbiAgICAgIH07XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG4vL1xuLy8gRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuTElTVClcbi8vICAgLmV4dGVuZCh7XG4vLyAgICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBsaXN0RGVjKFxuLy8gICAgICAgY2xheno6XG4vLyAgICAgICAgIHwgQ29uc3RydWN0b3Jcbi8vICAgICAgICAgfCAoKCkgPT4gQ29uc3RydWN0b3IpXG4vLyAgICAgICAgIHwgKENvbnN0cnVjdG9yIHwgKCgpID0+IENvbnN0cnVjdG9yKSlbXSxcbi8vICAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbi8vICAgICAgIGNvbGxlY3Rpb246IFwiQXJyYXlcIiB8IFwiU2V0XCIgPSBcIkFycmF5XCJcbi8vICAgICApIHtcbi8vICAgICAgIHJldHVybiAodGFyZ2V0OiBvYmplY3QsIHByb3A6IGFueSkgPT4ge1xuLy8gICAgICAgICBjbGF6eiA9IEFycmF5LmlzQXJyYXkoY2xhenopID8gY2xhenpbMF0gOiBjbGF6ejtcbi8vICAgICAgICAgaWYgKHR5cGVvZiBjbGF6eiA9PT0gXCJmdW5jdGlvblwiICYmICFjbGF6ei5uYW1lKVxuLy8gICAgICAgICAgIGNsYXp6ID0gKGNsYXp6IGFzICgpID0+IENvbnN0cnVjdG9yKSgpO1xuLy8gICAgICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyB0eXBlOiBbY2xhenogYXMgYW55XSB9KSh0YXJnZXQsIHByb3ApO1xuLy8gICAgICAgfTtcbi8vICAgICB9LFxuLy8gICB9KVxuLy8gICAuYXBwbHkoKTtcbi8vIC8vXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5EQVRFKVxuICAuZXh0ZW5kKHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBkYXRlRGVjKGZvcm1hdDogc3RyaW5nKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICB0eXBlOiBTdHJpbmcsXG4gICAgICAgIGZvcm1hdDogXCJkYXRlLXRpbWVcIixcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgICAvLyBleGFtcGxlOiBwYXJzZURhdGUoZm9ybWF0LCBuZXcgRGF0ZSgpKSxcbiAgICAgIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5FTlVNKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG9wdGlvbkRlYyhvcHRpb25zOiBzdHJpbmdbXSB8IFJlY29yZDxzdHJpbmcsIGFueT4pIHtcbiAgICAgIGNvbnN0IG9wdHMgPSBBcnJheS5pc0FycmF5KG9wdGlvbnMpID8gb3B0aW9ucyA6IE9iamVjdC52YWx1ZXMob3B0aW9ucyk7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICBlbnVtOiBvcHRzLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuUEFUVEVSTilcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBwYXR0ZXJuRGVjKHBhdDogUmVnRXhwIHwgc3RyaW5nKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICBwYXR0ZXJuOiB0eXBlb2YgcGF0ID09PSBcInN0cmluZ1wiID8gcGF0IDogcGF0LnNvdXJjZSxcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFBlcnNpc3RlbmNlS2V5cy5DT0xVTU4pXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gY29sdW1uRGVjKG5hbWU6IHN0cmluZykge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHtcbiAgICAgICAgbmFtZTogbmFtZSxcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKERlY29yYXRpb25LZXlzLkRFU0NSSVBUSU9OKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGRlc2NyaXB0aW9uRGVjKGRlc2NyaXB0aW9uOiBzdHJpbmcpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBkZXNjcmlwdGlvbixcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFBlcnNpc3RlbmNlS2V5cy5BVVRIKS5leHRlbmQoeyBkZWNvcmF0b3I6IEF1dGggfSkuYXBwbHkoKTtcbiIsImltcG9ydCB7IEZvcmJpZGRlbkVycm9yIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5cbmV4cG9ydCBjbGFzcyBDb3JzRXJyb3IgZXh0ZW5kcyBGb3JiaWRkZW5FcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihtc2csIENvcnNFcnJvci5uYW1lKTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgQmFzZUVycm9yIH0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5cbmV4cG9ydCBjbGFzcyBUb01hbnlSZXF1ZXN0c0Vycm9yIGV4dGVuZHMgQmFzZUVycm9yIHtcbiAgY29uc3RydWN0b3IobXNnOiBzdHJpbmcgfCBFcnJvciApIHtcbiAgICBzdXBlcihUb01hbnlSZXF1ZXN0c0Vycm9yLm5hbWUsIG1zZyAsIDQyOSk7XG4gIH1cbn0iLCJpbXBvcnQge1xuICBBcmd1bWVudHNIb3N0LFxuICBDYXRjaCxcbiAgRXhjZXB0aW9uRmlsdGVyLFxuICBOb3RBY2NlcHRhYmxlRXhjZXB0aW9uLFxuICBOb3RGb3VuZEV4Y2VwdGlvbixcbn0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBCYXNlRXJyb3IsIEludGVybmFsRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IExvZ2dlZEVudmlyb25tZW50IH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyBVbnN1cHBvcnRlZEVycm9yIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBUb01hbnlSZXF1ZXN0c0Vycm9yIH0gZnJvbSBcIi4uL2Vycm9ycy90aHJvdHRsaW5nXCI7XG5cbkBDYXRjaCgpXG5leHBvcnQgY2xhc3MgRGVjYWZFeGNlcHRpb25GaWx0ZXIgaW1wbGVtZW50cyBFeGNlcHRpb25GaWx0ZXIge1xuICBjYXRjaChleGNlcHRpb246IEVycm9yLCBob3N0OiBBcmd1bWVudHNIb3N0KSB7XG4gICAgY29uc3QgY3R4ID0gaG9zdC5zd2l0Y2hUb0h0dHAoKTtcbiAgICBjb25zdCByZXNwb25zZSA9IGN0eC5nZXRSZXNwb25zZSgpO1xuICAgIGNvbnN0IHJlcXVlc3QgPSBjdHguZ2V0UmVxdWVzdCgpO1xuXG4gICAgY29uc3QgaXNQcm9kdWN0aW9uID0gTG9nZ2VkRW52aXJvbm1lbnQuZW52ID09PSBcInByb2R1Y3Rpb25cIjtcbiAgICBsZXQgc3RhdHVzQ29kZTtcblxuICAgIGlmIChcbiAgICAgIGV4Y2VwdGlvbiBpbnN0YW5jZW9mIE5vdEZvdW5kRXhjZXB0aW9uIHx8XG4gICAgICBleGNlcHRpb24gaW5zdGFuY2VvZiBVbnN1cHBvcnRlZEVycm9yXG4gICAgKSB7XG4gICAgICBleGNlcHRpb24gPSBuZXcgTm90QWNjZXB0YWJsZUV4Y2VwdGlvbihleGNlcHRpb24ubWVzc2FnZSk7XG4gICAgICBzdGF0dXNDb2RlID0gKGV4Y2VwdGlvbiBhcyBOb3RBY2NlcHRhYmxlRXhjZXB0aW9uKS5nZXRTdGF0dXMoKTtcbiAgICB9IGVsc2UgaWYgKCEoZXhjZXB0aW9uIGluc3RhbmNlb2YgQmFzZUVycm9yKSkge1xuICAgICAgaWYoKGV4Y2VwdGlvbiBhcyBhbnkpLnN0YXR1cyA9PT0gNDI5KXtcbiAgICAgICAgZXhjZXB0aW9uID0gbmV3IFRvTWFueVJlcXVlc3RzRXJyb3IoZXhjZXB0aW9uLm1lc3NhZ2UpO1xuICAgICAgfWVsc2V7XG4gICAgICAgIGV4Y2VwdGlvbiA9IG5ldyBJbnRlcm5hbEVycm9yKGV4Y2VwdGlvbi5tZXNzYWdlKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXNwb25zZS5zdGF0dXMoKGV4Y2VwdGlvbiBhcyBCYXNlRXJyb3IpLmNvZGUgfHwgc3RhdHVzQ29kZSkuanNvbih7XG4gICAgICBzdGF0dXM6IChleGNlcHRpb24gYXMgQmFzZUVycm9yKS5jb2RlIHx8IHN0YXR1c0NvZGUsXG4gICAgICBlcnJvcjogaXNQcm9kdWN0aW9uID8gZXhjZXB0aW9uLm5hbWUgOiBleGNlcHRpb24ubWVzc2FnZSxcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgcGF0aDogcmVxdWVzdC51cmwsXG4gICAgICBtZXRob2Q6IHJlcXVlc3QubWV0aG9kLFxuICAgIH0pO1xuICB9XG59XG4iLCJpbXBvcnQgeyBVc2VGaWx0ZXJzLCBVc2VJbnRlcmNlcHRvcnMgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmRXhjZXB0aW9uRmlsdGVyIH0gZnJvbSBcIi4vRGVjYWZFcnJvckZpbHRlclwiO1xuaW1wb3J0IHsgRGVjYWZSZXNwb25zZUludGVyY2VwdG9yIH0gZnJvbSBcIi4uLy4uL3JlcXVlc3QvaW5kZXhcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIFVzZURlY2FmRmlsdGVyKCkge1xuICByZXR1cm4gVXNlRmlsdGVycyhuZXcgRGVjYWZFeGNlcHRpb25GaWx0ZXIoKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBVc2VEZWNhZkhlYWRlcnMoKSB7XG4gIHJldHVybiBVc2VJbnRlcmNlcHRvcnMoRGVjYWZSZXNwb25zZUludGVyY2VwdG9yKTtcbn1cbiIsImltcG9ydCB7IENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBTZWN1cml0eVNjaGVtZU9iamVjdCB9IGZyb20gXCIuLi8uLi9zd2FnZ2VyLXR5cGVzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3dhZ2dlck9wdGlvbnMge1xuICB0aXRsZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICB2ZXJzaW9uPzogc3RyaW5nO1xuICBhc3NldHNQYXRoPzogc3RyaW5nO1xuICBmYXZpY29uRmlsZVBhdGg/OiBzdHJpbmc7XG4gIHRvcGJhckljb25GaWxlUGF0aD86IHN0cmluZztcbiAgcGVyc2lzdEF1dGhvcml6YXRpb24/OiBib29sZWFuO1xuICBwYXRoOiBzdHJpbmc7XG4gIGF1dGg/OiBTZWN1cml0eVNjaGVtZU9iamVjdDtcbiAgdG9wYmFyQmdDb2xvcj86IHN0cmluZztcbiAgZXh0cmFNb2RlbHM/OiBDb25zdHJ1Y3RvcltdO1xuXG4gIC8qKlxuICAgKiBQYXRoIHRvIGV4cG9zZSB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGluIEpTT04gZm9ybWF0LlxuICAgKiBFeGFtcGxlOiAvYXBpL29wZW5hcGkuanNvblxuICAgKi9cbiAgb3BlbkFwaUpzb25QYXRoPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQYXRoIHRvIGV4cG9zZSB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGluIFlBTUwgZm9ybWF0LlxuICAgKiBSZXF1aXJlcyAneWFtbCcgcGFja2FnZSBhcyBwZWVyIGRlcGVuZGVuY3kuXG4gICAqIEV4YW1wbGU6IC9hcGkvb3BlbmFwaS55YW1sXG4gICAqL1xuICBvcGVuQXBpWWFtbFBhdGg/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBjb25zdCBTV0FHR0VSX1VJX0NPTlNUQU5UUzogU3dhZ2dlck9wdGlvbnMgPSB7XG4gIHRpdGxlOiBcIlN3YWdnZXIgfCBPcGVuQVBJIFNwZWNpZmljYXRpb24gKE9BUylcIixcbiAgZGVzY3JpcHRpb246IFwiU3RhbmRhcmRpemVkIGZvcm1hdCBmb3IgZGVzY3JpYmluZyBSRVNUZnVsIEFQSXNcIixcbiAgdmVyc2lvbjogXCIwLjAuMVwiLFxuICBwYXRoOiBcImRvY3NcIixcbiAgZmF2aWNvbkZpbGVQYXRoOiBcIlwiLFxuICB0b3BiYXJJY29uRmlsZVBhdGg6IFwiXCIsXG4gIGF1dGg6IHtcbiAgICB0eXBlOiBcImh0dHBcIixcbiAgICBzY2hlbWU6IFwiYmVhcmVyXCIsXG4gICAgYmVhcmVyRm9ybWF0OiBcIkpXVFwiLFxuICAgIG5hbWU6IFwiQXV0aG9yaXphdGlvblwiLFxuICAgIGRlc2NyaXB0aW9uOiBcIkVudGVyIEpXVCB0b2tlblwiLFxuICAgIGluOiBcImhlYWRlclwiLFxuICB9LFxuICBwZXJzaXN0QXV0aG9yaXphdGlvbjogdHJ1ZSxcbiAgdG9wYmFyQmdDb2xvcjogXCIjMDAwMDAwXCIsXG59O1xuIiwiaW1wb3J0IHsgU1dBR0dFUl9VSV9DT05TVEFOVFMgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IHJlYWRGaWxlU3luYyB9IGZyb20gXCJmc1wiO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tIFwicGF0aFwiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFN3YWdnZXJVSU9wdGlvbnMge1xuICB0aXRsZT86IHN0cmluZztcbiAgcGF0aD86IHN0cmluZztcbiAgcGVyc2lzdEF1dGhvcml6YXRpb246IGJvb2xlYW47XG4gIGFzc2V0c1BhdGg/OiBzdHJpbmc7XG4gIGZhdmljb25QYXRoPzogc3RyaW5nO1xuICB0b3BiYXJJY29uUGF0aD86IHN0cmluZztcbiAgdG9wYmFyQmdDb2xvcj86IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIFN3YWdnZXJDdXN0b21VSSB7XG4gIHJlYWRvbmx5IG9wdGlvbnM6IFN3YWdnZXJVSU9wdGlvbnM7XG4gIC8vIHByaXZhdGUgcmVhZG9ubHkgYXNzZXRzUGF0aDogc3RyaW5nID0gcGF0aC5qb2luKFxuICAvLyAgIF9fZGlybmFtZSxcbiAgLy8gICBcIi4uXCIsXG4gIC8vICAgXCIuLlwiLFxuICAvLyAgIFwid29ya2RvY3NcIixcbiAgLy8gICBcImFzc2V0c1wiXG4gIC8vICk7XG5cbiAgY29uc3RydWN0b3Iob3B0aW9uczogU3dhZ2dlclVJT3B0aW9ucykge1xuICAgIHRoaXMub3B0aW9ucyA9IHtcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgY3VzdG9tQ1NTKCkge1xuICAgIGxldCBjc3MgPSBcIlwiO1xuICAgIGlmICh0aGlzLm9wdGlvbnMudG9wYmFySWNvblBhdGgpIHtcbiAgICAgIGNvbnN0IGltZyA9IHRoaXMuYjY0KHRoaXMub3B0aW9ucy50b3BiYXJJY29uUGF0aCk7XG4gICAgICBjc3MgKz0gYC50b3BiYXItd3JhcHBlciB7IGNvbnRlbnQ6IHVybCgnZGF0YTppbWFnZS9wbmc7YmFzZTY0LCR7aW1nfScpOyB3aWR0aDogMjAwcHg7IGhlaWdodDogYXV0bzsgfVxcbmA7XG4gICAgfVxuICAgIHJldHVybiAoXG4gICAgICBjc3MgK1xuICAgICAgYFxuICAgICAgICAudG9wYmFyLXdyYXBwZXIgc3ZnIHsgdmlzaWJpbGl0eTogaGlkZGVuOyB9XG4gICAgICAgIC5zd2FnZ2VyLXVpIC50b3BiYXIgeyBiYWNrZ3JvdW5kLWNvbG9yOiAke3RoaXMub3B0aW9ucy50b3BiYXJCZ0NvbG9yIHx8IFNXQUdHRVJfVUlfQ09OU1RBTlRTLnRvcGJhckJnQ29sb3J9OyB9XG4gICAgICBgXG4gICAgKTtcbiAgfVxuXG4gIGdldEN1c3RvbU9wdGlvbnMoKSB7XG4gICAgY29uc3QgZmF2aWNvbjogUmVjb3JkPHN0cmluZywgYW55PiA9IHt9O1xuICAgIGlmICh0aGlzLm9wdGlvbnMuZmF2aWNvblBhdGgpIHtcbiAgICAgIGZhdmljb25bXCJjdXN0b21mYXZJY29uXCJdID0gdGhpcy5iNjQodGhpcy5vcHRpb25zLmZhdmljb25QYXRoLCB0cnVlKTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgY3VzdG9tU2l0ZVRpdGxlOiB0aGlzLm9wdGlvbnMudGl0bGUsXG4gICAgICAuLi5mYXZpY29uLFxuICAgICAgY3VzdG9tQ3NzOiB0aGlzLmN1c3RvbUNTUygpLFxuICAgICAgc3dhZ2dlck9wdGlvbnM6IHtcbiAgICAgICAgcGVyc2lzdEF1dGhvcml6YXRpb246IHRoaXMub3B0aW9ucy5wZXJzaXN0QXV0aG9yaXphdGlvbixcbiAgICAgIH0sXG4gICAgfTtcbiAgfVxuXG4gIGI2NChmaWxlOiBzdHJpbmcsIGltZzogYm9vbGVhbiA9IGZhbHNlKSB7XG4gICAgY29uc3QgZmlsZVBhdGggPSBwYXRoLmpvaW4odGhpcy5vcHRpb25zLmFzc2V0c1BhdGggfHwgXCJcIiwgZmlsZSk7XG4gICAgY29uc3QgYjY0ID0gcmVhZEZpbGVTeW5jKGZpbGVQYXRoLCB7IGVuY29kaW5nOiBcImJhc2U2NFwiIH0pO1xuICAgIHJldHVybiBpbWcgPyBcImRhdGE6aW1hZ2UvcG5nO2Jhc2U2NCxcIiArIGI2NCA6IGI2NDtcbiAgfVxufVxuIiwiaW1wb3J0IHsgSU5lc3RBcHBsaWNhdGlvbiB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRG9jdW1lbnRCdWlsZGVyLCBTd2FnZ2VyTW9kdWxlIH0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlclwiO1xuaW1wb3J0IHsgU3dhZ2dlckN1c3RvbVVJIH0gZnJvbSBcIi4vU3dhZ2dlckN1c3RvbVVJXCI7XG5pbXBvcnQgeyBTV0FHR0VSX1VJX0NPTlNUQU5UUywgU3dhZ2dlck9wdGlvbnMgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCBZQU1MIGZyb20gXCJ5YW1sXCI7XG5cbmV4cG9ydCBjbGFzcyBTd2FnZ2VyQnVpbGRlciB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgYXBwOiBJTmVzdEFwcGxpY2F0aW9uLFxuICAgIHByaXZhdGUgcmVhZG9ubHkgb3B0aW9uczogU3dhZ2dlck9wdGlvbnNcbiAgKSB7fVxuXG4gIHByaXZhdGUgY3JlYXRlRG9jdW1lbnQoKSB7XG4gICAgY29uc3QgZGVzY3JpcHRpb24gPSB0aGlzLm9wdGlvbnMucGF0aFxuICAgICAgPyB0aGlzLm9wdGlvbnMuZGVzY3JpcHRpb24gK1xuICAgICAgICBcIlwiICtcbiAgICAgICAgYDxicj48YnI+PGEgaHJlZj1cIiR7dGhpcy5vcHRpb25zLm9wZW5BcGlKc29uUGF0aH1cIj5PcGVuQVBJIEpTT04gU3BlYzwvYT4gfCBgICtcbiAgICAgICAgYDxhIGhyZWY9XCIke3RoaXMub3B0aW9ucy5vcGVuQXBpWWFtbFBhdGh9XCI+T3BlbkFQSSBZQU1MIFNwZWM8L2E+YFxuICAgICAgOiB0aGlzLm9wdGlvbnMuZGVzY3JpcHRpb247XG5cbiAgICBjb25zdCBjb25maWcgPSBuZXcgRG9jdW1lbnRCdWlsZGVyKClcbiAgICAgIC5zZXRUaXRsZSh0aGlzLm9wdGlvbnMudGl0bGUpXG4gICAgICAuc2V0RGVzY3JpcHRpb24oZGVzY3JpcHRpb24pXG4gICAgICAuc2V0VmVyc2lvbih0aGlzLm9wdGlvbnMudmVyc2lvbiB8fCBcIjAuMC4xXCIpXG4gICAgICAuYWRkQmVhcmVyQXV0aCh0aGlzLm9wdGlvbnMuYXV0aCB8fCBTV0FHR0VSX1VJX0NPTlNUQU5UUy5hdXRoKVxuICAgICAgLmJ1aWxkKCk7XG5cbiAgICByZXR1cm4gU3dhZ2dlck1vZHVsZS5jcmVhdGVEb2N1bWVudCh0aGlzLmFwcCwgY29uZmlnLCB7XG4gICAgICBleHRyYU1vZGVsczogdGhpcy5vcHRpb25zLmV4dHJhTW9kZWxzIHx8IFtdLFxuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSByZWdpc3Rlck9wZW5BcGlSb3V0ZShcbiAgICBwYXRoOiBzdHJpbmcgfCB1bmRlZmluZWQsXG4gICAgY29udGVudFR5cGU6IFwiYXBwbGljYXRpb24vanNvblwiIHwgXCJhcHBsaWNhdGlvbi94LXlhbWxcIixcbiAgICBib2R5RmFjdG9yeTogKCkgPT4gdm9pZFxuICApOiB2b2lkIHtcbiAgICBpZiAoIXBhdGgpIHJldHVybjtcblxuICAgIGNvbnN0IGh0dHBBZGFwdGVyID0gdGhpcy5hcHAuZ2V0SHR0cEFkYXB0ZXIoKSBhcyBhbnk7XG4gICAgcGF0aCA9IHBhdGguc3RhcnRzV2l0aChcIi9cIikgPyBwYXRoIDogYC8ke3BhdGh9YDtcbiAgICBodHRwQWRhcHRlci5nZXQocGF0aCwgKF9yZXE6IGFueSwgcmVzOiBhbnkpID0+IHtcbiAgICAgIGh0dHBBZGFwdGVyLnJlcGx5KHJlcywgYm9keUZhY3RvcnkoKSwgMjAwLCB7XG4gICAgICAgIFwiQ29udGVudC1UeXBlXCI6IGNvbnRlbnRUeXBlLFxuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICBwdWJsaWMgc2V0dXBTd2FnZ2VyKCkge1xuICAgIGNvbnN0IGRvY3VtZW50ID0gdGhpcy5jcmVhdGVEb2N1bWVudCgpO1xuICAgIGNvbnN0IHN3YWdnZXJVSSA9IG5ldyBTd2FnZ2VyQ3VzdG9tVUkoe1xuICAgICAgdGl0bGU6IHRoaXMub3B0aW9ucy50aXRsZSxcbiAgICAgIHBhdGg6IHRoaXMub3B0aW9ucy5wYXRoIHx8IFNXQUdHRVJfVUlfQ09OU1RBTlRTLnBhdGgsXG4gICAgICBwZXJzaXN0QXV0aG9yaXphdGlvbjogdGhpcy5vcHRpb25zLnBlcnNpc3RBdXRob3JpemF0aW9uID8/IHRydWUsXG4gICAgICBhc3NldHNQYXRoOiB0aGlzLm9wdGlvbnMuYXNzZXRzUGF0aCxcbiAgICAgIGZhdmljb25QYXRoOiB0aGlzLm9wdGlvbnMuZmF2aWNvbkZpbGVQYXRoLFxuICAgICAgdG9wYmFySWNvblBhdGg6IHRoaXMub3B0aW9ucy50b3BiYXJJY29uRmlsZVBhdGgsXG4gICAgICB0b3BiYXJCZ0NvbG9yOiB0aGlzLm9wdGlvbnMudG9wYmFyQmdDb2xvcixcbiAgICB9KTtcbiAgICBTd2FnZ2VyTW9kdWxlLnNldHVwKFxuICAgICAgdGhpcy5vcHRpb25zLnBhdGggfHwgU1dBR0dFUl9VSV9DT05TVEFOVFMucGF0aCxcbiAgICAgIHRoaXMuYXBwLFxuICAgICAgZG9jdW1lbnQsXG4gICAgICB7XG4gICAgICAgIC4uLnN3YWdnZXJVSS5nZXRDdXN0b21PcHRpb25zKCksXG4gICAgICAgIGpzb25Eb2N1bWVudFVybDogdGhpcy5vcHRpb25zLm9wZW5BcGlKc29uUGF0aFxuICAgICAgICAgID8gYCR7dGhpcy5vcHRpb25zLm9wZW5BcGlKc29uUGF0aH1gXG4gICAgICAgICAgOiB1bmRlZmluZWQsXG4gICAgICAgIHlhbWxEb2N1bWVudFVybDogdGhpcy5vcHRpb25zLm9wZW5BcGlZYW1sUGF0aFxuICAgICAgICAgID8gYCR7dGhpcy5vcHRpb25zLm9wZW5BcGlZYW1sUGF0aH1gXG4gICAgICAgICAgOiB1bmRlZmluZWQsXG4gICAgICB9XG4gICAgKTtcblxuICAgIHRoaXMucmVnaXN0ZXJPcGVuQXBpUm91dGUoXG4gICAgICB0aGlzLm9wdGlvbnMub3BlbkFwaUpzb25QYXRoLFxuICAgICAgXCJhcHBsaWNhdGlvbi9qc29uXCIsXG4gICAgICAoKSA9PiBkb2N1bWVudFxuICAgICk7XG5cbiAgICB0aGlzLnJlZ2lzdGVyT3BlbkFwaVJvdXRlKFxuICAgICAgdGhpcy5vcHRpb25zLm9wZW5BcGlZYW1sUGF0aCxcbiAgICAgIFwiYXBwbGljYXRpb24veC15YW1sXCIsXG4gICAgICAoKSA9PiBZQU1MLnN0cmluZ2lmeShkb2N1bWVudClcbiAgICApO1xuICB9XG5cbiAgLy8gcHJpdmF0ZSBnZXRWZXJzaW9uKCkge1xuICAvLyAgIC8vIGNvbnN0IHBhY2thZ2VKc29uID0gcGF0aC5qb2luKF9fZGlybmFtZSwgXCIuLlwiLCBcIi4uXCIsIFwicGFja2FnZS5qc29uXCIpO1xuICAvLyAgIC8vIGNvbnN0IHt2ZXJzaW9ufSA9IHJlcXVpcmUocGFja2FnZUpzb24pO1xuICAvLyAgIHJldHVybiBcInZlcnNpb25cIjtcbiAgLy8gfVxufVxuIiwiaW1wb3J0IHtcbiAgSU5lc3RBcHBsaWNhdGlvbixcbiAgTG9nZ2VyLFxuICBOZXN0SW50ZXJjZXB0b3IsXG4gIFBpcGVUcmFuc2Zvcm0sXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRGVjYWZFeGNlcHRpb25GaWx0ZXIgfSBmcm9tIFwiLi9leGNlcHRpb25zXCI7XG5pbXBvcnQgeyBTd2FnZ2VyQnVpbGRlciB9IGZyb20gXCIuL29wZW5hcGlcIjtcbmltcG9ydCB7IENvcnNPcHRpb25zIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uL2ludGVyZmFjZXMvZXh0ZXJuYWwvY29ycy1vcHRpb25zLmludGVyZmFjZVwiO1xuaW1wb3J0IHsgQ29yc0Vycm9yIH0gZnJvbSBcIi4vZXJyb3JzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBEZWZpbmVzIGFsbCBjdXN0b21pemFibGUgcGFyYW1ldGVycyBmb3IgU3dhZ2dlciBzZXR1cC5cbiAqXG4gKiBAc3VtbWFyeVxuICogVGhpcyBpbnRlcmZhY2UgYWxsb3dzIGRldmVsb3BlcnMgdG8gY3VzdG9taXplIGhvdyBTd2FnZ2VyIFVJIGlzIGNvbmZpZ3VyZWRcbiAqIHdpdGhpbiB0aGUgTmVzdEpTIGFwcGxpY2F0aW9uLiBJdCBpbmNsdWRlcyBwYXJhbWV0ZXJzIGZvciB0aXRsZXMsIHBhdGhzLFxuICogY29sb3Igc2NoZW1lcywgYW5kIGFzc2V0IHBhdGhzIHRvIHRhaWxvciB0aGUgQVBJIGRvY3VtZW50YXRpb24gZXhwZXJpZW5jZS5cbiAqXG4gKiBAcGFyYW0ge3N0cmluZ30gdGl0bGUgLSBUaXRsZSBkaXNwbGF5ZWQgaW4gU3dhZ2dlciBVSS5cbiAqIEBwYXJhbSB7c3RyaW5nfSBkZXNjcmlwdGlvbiAtIERlc2NyaXB0aW9uIHNob3duIGJlbG93IHRoZSB0aXRsZS5cbiAqIEBwYXJhbSB7c3RyaW5nfSB2ZXJzaW9uIC0gQVBJIHZlcnNpb24gZGlzcGxheWVkIGluIHRoZSBkb2N1bWVudGF0aW9uLlxuICogQHBhcmFtIHtzdHJpbmd9IFtwYXRoXSAtIE9wdGlvbmFsIHBhdGggd2hlcmUgU3dhZ2dlciB3aWxsIGJlIGF2YWlsYWJsZS5cbiAqIEBwYXJhbSB7Ym9vbGVhbn0gW3BlcnNpc3RBdXRob3JpemF0aW9uXSAtIFdoZXRoZXIgYXV0aG9yaXphdGlvbiB0b2tlbnMgcGVyc2lzdCBhY3Jvc3MgcmVsb2Fkcy5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbYXNzZXRzUGF0aF0gLSBQYXRoIHRvIGN1c3RvbSBhc3NldHMgZm9yIFN3YWdnZXIgVUkuXG4gKiBAcGFyYW0ge3N0cmluZ30gW3RvcGJhckJnQ29sb3JdIC0gQ3VzdG9tIGJhY2tncm91bmQgY29sb3IgZm9yIHRoZSBTd2FnZ2VyIHRvcCBiYXIuXG4gKiBAcGFyYW0ge3N0cmluZ30gW3RvcGJhckljb25QYXRoXSAtIFBhdGggdG8gYSBjdXN0b20gaWNvbiBkaXNwbGF5ZWQgaW4gdGhlIHRvcCBiYXIuXG4gKiBAcGFyYW0ge3N0cmluZ30gW2Zhdmljb25QYXRoXSAtIFBhdGggdG8gYSBjdXN0b20gZmF2aWNvbi5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTd2FnZ2VyU2V0dXBPcHRpb25zIHtcbiAgdGl0bGU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgdmVyc2lvbjogc3RyaW5nO1xuICBwYXRoPzogc3RyaW5nO1xuICBwZXJzaXN0QXV0aG9yaXphdGlvbj86IGJvb2xlYW47XG4gIGFzc2V0c1BhdGg/OiBzdHJpbmc7XG4gIHRvcGJhckJnQ29sb3I/OiBzdHJpbmc7XG4gIHRvcGJhckljb25QYXRoPzogc3RyaW5nO1xuICBmYXZpY29uUGF0aD86IHN0cmluZztcbiAgLyoqXG4gICAqIFBhdGggdG8gZXhwb3NlIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gSlNPTiBmb3JtYXQuXG4gICAqIEV4YW1wbGU6IC9hcGkvb3BlbmFwaS5qc29uXG4gICAqL1xuICBvcGVuQXBpSnNvblBhdGg/OiBzdHJpbmc7XG4gIC8qKlxuICAgKiBQYXRoIHRvIGV4cG9zZSB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGluIFlBTUwgZm9ybWF0LlxuICAgKiBSZXF1aXJlcyAneWFtbCcgcGFja2FnZSBhcyBwZWVyIGRlcGVuZGVuY3kuXG4gICAqIEV4YW1wbGU6IC9hcGkvb3BlbmFwaS55YW1sXG4gICAqL1xuICBvcGVuQXBpWWFtbFBhdGg/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBBIGZsdWVudCwgc3RhdGljIGJvb3RzdHJhcCBjbGFzcyBmb3IgaW5pdGlhbGl6aW5nIGFuZCBjb25maWd1cmluZyBhIE5lc3RKUyBhcHBsaWNhdGlvbi5cbiAqXG4gKiBAc3VtbWFyeVxuICogVGhlIGBOZXN0Qm9vdHN0cmFwZXJgIGNsYXNzIHByb3ZpZGVzIGEgY2hhaW5hYmxlIEFQSSBmb3IgY29uZmlndXJpbmdcbiAqIGEgTmVzdEpTIGFwcGxpY2F0aW9uIGluc3RhbmNlLiBJdCBpbmNsdWRlcyBidWlsdC1pbiBtZXRob2RzIGZvciBlbmFibGluZ1xuICogQ09SUywgSGVsbWV0IHNlY3VyaXR5LCBTd2FnZ2VyIGRvY3VtZW50YXRpb24sIGdsb2JhbCBwaXBlcywgZmlsdGVycyxcbiAqIGludGVyY2VwdG9ycywgYW5kIHN0YXJ0aW5nIHRoZSBzZXJ2ZXIuXG4gKlxuICogVGhpcyBjbGFzcyBwcm9tb3RlcyBjb25zaXN0ZW5jeSBhbmQgcmVkdWNlcyByZXBldGl0aXZlIHNldHVwIGNvZGVcbiAqIGFjcm9zcyBtdWx0aXBsZSBOZXN0SlMgcHJvamVjdHMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHRzXG4gKiBpbXBvcnQgeyBOZXN0RmFjdG9yeSB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcbiAqIGltcG9ydCB7IEFwcE1vZHVsZSB9IGZyb20gXCIuL2FwcC5tb2R1bGVcIjtcbiAqIGltcG9ydCB7IE15TG9nZ2VyIH0gZnJvbSBcIi4vTXlMb2dnZXJcIjtcbiAqIGltcG9ydCB7IE5lc3RCb290c3RyYXBlciB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLW5lc3RcIjtcbiAqXG4gKiBhc3luYyBmdW5jdGlvbiBib290c3RyYXAoKSB7XG4gKiAgIGNvbnN0IGFwcCA9IGF3YWl0IE5lc3RGYWN0b3J5LmNyZWF0ZShBcHBNb2R1bGUpO1xuICpcbiAqICAgYXdhaXQgTmVzdEJvb3RzdHJhcGVyXG4gKiAgICAgLmluaXRpYWxpemUoYXBwKVxuICogICAgIC5lbmFibGVMb2dnZXIobmV3IE15TG9nZ2VyKCkpXG4gKiAgICAgLmVuYWJsZUNvcnMoW1wiaHR0cDovL2xvY2FsaG9zdDo0MjAwXCJdKVxuICogICAgIC51c2VIZWxtZXQoKVxuICogICAgIC5zZXR1cFN3YWdnZXIoe1xuICogICAgICAgdGl0bGU6IFwiT3BlbkFQSSBieSBUcmFkZU1hcmvihKJcIixcbiAqICAgICAgIGRlc2NyaXB0aW9uOiBcIlRyYWRlTWFya+KEoiBBUEkgZG9jdW1lbnRhdGlvblwiLFxuICogICAgICAgdmVyc2lvbjogXCIxLjAuMFwiLFxuICogICAgICAgcGF0aDogXCJhcGlcIixcbiAqICAgICAgIHBlcnNpc3RBdXRob3JpemF0aW9uOiB0cnVlLFxuICogICAgICAgdG9wYmFyQmdDb2xvcjogXCIjMkMzRTUwXCIsXG4gKiAgICAgICB0b3BiYXJJY29uUGF0aDogXCIvYXNzZXRzL2xvZ28uc3ZnXCIsXG4gKiAgICAgICBmYXZpY29uUGF0aDogXCIvYXNzZXRzL2Zhdmljb24uaWNvXCJcbiAqICAgICB9KVxuICogICAgIC51c2VHbG9iYWxGaWx0ZXJzKClcbiAqICAgICAudXNlR2xvYmFsUGlwZXMoLi4uKVxuICogICAgIC51c2VHbG9iYWxJbnRlcmNlcHRvcnMoLi4uKVxuICogICAgIC5zdGFydCgzMDAwKTtcbiAqIH1cbiAqXG4gKiBib290c3RyYXAoKTtcbiAqIGBgYFxuICogQGNsYXNzXG4gKi9cbmV4cG9ydCBjbGFzcyBOZXN0Qm9vdHN0cmFwZXIge1xuICBwcml2YXRlIHN0YXRpYyBhcHA6IElOZXN0QXBwbGljYXRpb247XG4gIHByaXZhdGUgc3RhdGljIF9sb2dnZXI6IExvZ2dlcjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIFJldHVybnMgdGhlIGN1cnJlbnQgbG9nZ2VyIGluc3RhbmNlLCBjcmVhdGluZyBhIGRlZmF1bHQgb25lIGlmIG5vdCBzZXQuXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIEVuc3VyZXMgdGhhdCBhIHZhbGlkIGBMb2dnZXJgIGluc3RhbmNlIGlzIGFsd2F5cyBhdmFpbGFibGVcbiAgICogZm9yIGxvZ2dpbmcgYm9vdHN0cmFwLXJlbGF0ZWQgbWVzc2FnZXMuXG4gICAqXG4gICAqIEByZXR1cm4ge0xvZ2dlcn0gVGhlIGFjdGl2ZSBsb2dnZXIgaW5zdGFuY2UuXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBnZXQgbG9nZ2VyKCk6IExvZ2dlciB7XG4gICAgaWYgKCF0aGlzLl9sb2dnZXIpIHtcbiAgICAgIC8vIGZhbGxiYWNrXG4gICAgICB0aGlzLl9sb2dnZXIgPSBuZXcgTG9nZ2VyKFwiTmVzdEJvb3RzdHJhcFwiKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2xvZ2dlcjtcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogSW5pdGlhbGl6ZXMgdGhlIGJvb3RzdHJhcHBlciB3aXRoIGEgZ2l2ZW4gTmVzdEpTIGFwcGxpY2F0aW9uLlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBCaW5kcyB0aGUgcHJvdmlkZWQgTmVzdEpTIGFwcCBpbnN0YW5jZSB0byB0aGUgYm9vdHN0cmFwcGVyLCBlbmFibGluZ1xuICAgKiBjaGFpbmVkIGNvbmZpZ3VyYXRpb24gbWV0aG9kcy5cbiAgICpcbiAgICogQHBhcmFtIHtJTmVzdEFwcGxpY2F0aW9ufSBhcHAgLSBUaGUgTmVzdEpTIGFwcGxpY2F0aW9uIGluc3RhbmNlIHRvIGluaXRpYWxpemUuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nIGNvbmZpZ3VyYXRpb24gbWV0aG9kcy5cbiAgICovXG4gIHN0YXRpYyBpbml0aWFsaXplKGFwcDogSU5lc3RBcHBsaWNhdGlvbikge1xuICAgIHRoaXMuYXBwID0gYXBwO1xuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBFbmFibGVzIG9yIHJlcGxhY2VzIHRoZSBnbG9iYWwgbG9nZ2VyIGZvciB0aGUgTmVzdEpTIGFwcGxpY2F0aW9uLlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBJZiBhIGN1c3RvbSBsb2dnZXIgaXMgcHJvdmlkZWQsIGl0IHJlcGxhY2VzIHRoZSBkZWZhdWx0IGxvZ2dlci4gT3RoZXJ3aXNlLFxuICAgKiBhIG5ldyBsb2dnZXIgbmFtZWQgYFwiTmVzdEJvb3RzdHJhcFwiYCBpcyB1c2VkLiBUaGlzIGxvZ2dlciBpcyBhbHNvIHJlZ2lzdGVyZWRcbiAgICogd2l0aCB0aGUgTmVzdEpTIGFwcGxpY2F0aW9uLlxuICAgKlxuICAgKiBAcGFyYW0ge0xvZ2dlcn0gW2N1c3RvbUxvZ2dlcl0gLSBPcHRpb25hbCBjdXN0b20gbG9nZ2VyIGluc3RhbmNlLlxuICAgKiBAcmV0dXJuIHtOZXN0Qm9vdHN0cmFwZXJ9IFJldHVybnMgdGhlIGNsYXNzIGZvciBjaGFpbmluZy5cbiAgICovXG4gIHN0YXRpYyBlbmFibGVMb2dnZXIoY3VzdG9tTG9nZ2VyPzogTG9nZ2VyKSB7XG4gICAgdGhpcy5fbG9nZ2VyID0gY3VzdG9tTG9nZ2VyIHx8IG5ldyBMb2dnZXIoXCJOZXN0Qm9vdHN0cmFwXCIpO1xuICAgIHRoaXMuYXBwLnVzZUxvZ2dlcih0aGlzLl9sb2dnZXIpO1xuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBFbmFibGVzIENyb3NzLU9yaWdpbiBSZXNvdXJjZSBTaGFyaW5nIChDT1JTKSBmb3IgdGhlIGFwcGxpY2F0aW9uLlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBBbGxvd3MgZGVmaW5pbmcgZWl0aGVyIGEgd2lsZGNhcmQgb3JpZ2luIChgXCIqXCJgKSBvciBhIGxpc3Qgb2YgYWxsb3dlZCBvcmlnaW5zLlxuICAgKiBBdXRvbWF0aWNhbGx5IGFjY2VwdHMgbG9jYWwgZGV2ZWxvcG1lbnQgcmVxdWVzdHMgYW5kIHRob3NlIHdpdGhvdXQgb3JpZ2luIGhlYWRlcnMuXG4gICAqIFRocm93cyBhIGBDb3JzRXJyb3JgIGZvciB1bmF1dGhvcml6ZWQgb3JpZ2lucy5cbiAgICpcbiAgICogQHBhcmFtIHsnKicgfCBzdHJpbmdbXX0gW29yaWdpbnM9W11dIC0gTGlzdCBvZiBhbGxvd2VkIG9yaWdpbnMgb3IgYFwiKlwiYCB0byBhbGxvdyBhbGwuXG4gICAqIEBwYXJhbSB7c3RyaW5nW119IFthbGxvd01ldGhvZHM9WydHRVQnLCAnUE9TVCcsICdQVVQnLCAnREVMRVRFJ11dIC0gQWxsb3dlZCBIVFRQIG1ldGhvZHMuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nIGNvbmZpZ3VyYXRpb24uXG4gICAqXG4gICAqL1xuICBzdGF0aWMgZW5hYmxlQ29ycyhcbiAgICBvcmlnaW5zOiBcIipcIiB8IHN0cmluZ1tdID0gW10sXG4gICAgYWxsb3dNZXRob2RzOiBzdHJpbmdbXSA9IFtcIkdFVFwiLCBcIlBPU1RcIiwgXCJQVVRcIiwgXCJERUxFVEVcIl1cbiAgKSB7XG4gICAgY29uc3QgYWxsb3dlZE9yaWdpbnMgPVxuICAgICAgb3JpZ2lucyA9PT0gXCIqXCIgPyBcIipcIiA6IG9yaWdpbnMubWFwKChvKSA9PiBvLnRyaW0oKS50b0xvd2VyQ2FzZSgpKTtcblxuICAgIGNvbnN0IGNvcnNPcHRpb25zOiBDb3JzT3B0aW9ucyA9IHtcbiAgICAgIG9yaWdpbjogKG9yaWdpbiwgY2FsbGJhY2spID0+IHtcbiAgICAgICAgLy8gQWxsb3cgcmVxdWVzdCB3aXRob3V0IG9yaWdpbi4uLlxuICAgICAgICBpZiAoIW9yaWdpbikgcmV0dXJuIGNhbGxiYWNrKG51bGwsIHRydWUpO1xuXG4gICAgICAgIGlmIChcbiAgICAgICAgICBhbGxvd2VkT3JpZ2lucyA9PT0gXCIqXCIgfHxcbiAgICAgICAgICAoQXJyYXkuaXNBcnJheShhbGxvd2VkT3JpZ2lucykgJiZcbiAgICAgICAgICAgIGFsbG93ZWRPcmlnaW5zLmluY2x1ZGVzKG9yaWdpbi50b0xvd2VyQ2FzZSgpKSlcbiAgICAgICAgKSB7XG4gICAgICAgICAgcmV0dXJuIGNhbGxiYWNrKG51bGwsIHRydWUpO1xuICAgICAgICB9XG5cbiAgICAgICAgY2FsbGJhY2sobmV3IENvcnNFcnJvcihgT3JpZ2luICR7b3JpZ2lufSBub3QgYWxsb3dlZGApKTtcbiAgICAgIH0sXG4gICAgICBjcmVkZW50aWFsczogdHJ1ZSxcbiAgICAgIG1ldGhvZHM6IGFsbG93TWV0aG9kcy5qb2luKFwiLFwiKSxcbiAgICB9O1xuXG4gICAgdGhpcy5hcHAuZW5hYmxlQ29ycyhjb3JzT3B0aW9ucyk7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIEFwcGxpZXMgdGhlIEhlbG1ldCBtaWRkbGV3YXJlIGZvciBlbmhhbmNlZCBzZWN1cml0eS5cbiAgICpcbiAgICogQHN1bW1hcnlcbiAgICogRHluYW1pY2FsbHkgbG9hZHMgdGhlIGBoZWxtZXRgIHBhY2thZ2UgaWYgYXZhaWxhYmxlIGFuZCByZWdpc3RlcnMgaXRcbiAgICogYXMgbWlkZGxld2FyZSB0byBpbXByb3ZlIEhUVFAgaGVhZGVyIHNlY3VyaXR5LiBJZiBub3QgaW5zdGFsbGVkLCBsb2dzIGEgd2FybmluZ1xuICAgKiBhbmQgY29udGludWVzIGV4ZWN1dGlvbiB3aXRob3V0IHRocm93aW5nIGVycm9ycy5cbiAgICpcbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBbb3B0aW9uc10gLSBPcHRpb25hbCBjb25maWd1cmF0aW9uIHBhc3NlZCB0byBIZWxtZXQuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBzdGF0aWMgdXNlSGVsbWV0KG9wdGlvbnM/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KSB7XG4gICAgdHJ5IHtcbiAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzXG4gICAgICBjb25zdCBoZWxtZXQgPSByZXF1aXJlKFwiaGVsbWV0XCIpOyAvLyBEeW5hbWljIGltcG9ydCB0byBhdm9pZCBoYXJkIGRlcGVuZGVuY3lcbiAgICAgIHRoaXMuYXBwLnVzZShoZWxtZXQob3B0aW9ucykpO1xuICAgICAgdGhpcy5sb2dnZXIubG9nKFwiSGVsbWV0IG1pZGRsZXdhcmUgZW5hYmxlZCBzdWNjZXNzZnVsbHkuXCIpO1xuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgdGhpcy5sb2dnZXIud2FybihcIkhlbG1ldCBub3QgaW5zdGFsbGVkLiBTa2lwcGluZyBtaWRkbGV3YXJlLlwiKTtcbiAgICB9XG5cbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogQ29uZmlndXJlcyBhbmQgaW5pdGlhbGl6ZXMgU3dhZ2dlciBVSSBmb3IgQVBJIGRvY3VtZW50YXRpb24uXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIFVzZXMgdGhlIGBTd2FnZ2VyQnVpbGRlcmAgdXRpbGl0eSB0byBjb25maWd1cmUgQVBJIGRvY3VtZW50YXRpb25cbiAgICogd2l0aCBkZXRhaWxlZCBjdXN0b21pemF0aW9uIGZvciB0aXRsZSwgdmVyc2lvbiwgcGF0aHMsIGFuZCBjb2xvcnMuXG4gICAqIFN3YWdnZXIgaXMgYXV0b21hdGljYWxseSBleHBvc2VkIGF0IHRoZSBjb25maWd1cmVkIHBhdGguXG4gICAqXG4gICAqIEBwYXJhbSB7U3dhZ2dlclNldHVwT3B0aW9uc30gb3B0aW9ucyAtIFN3YWdnZXIgY29uZmlndXJhdGlvbiBvcHRpb25zLlxuICAgKiBAcmV0dXJuIHtOZXN0Qm9vdHN0cmFwZXJ9IFJldHVybnMgdGhlIGNsYXNzIGZvciBjaGFpbmluZyBjb25maWd1cmF0aW9uLlxuICAgKi9cbiAgc3RhdGljIHNldHVwU3dhZ2dlcihvcHRpb25zOiBTd2FnZ2VyU2V0dXBPcHRpb25zKSB7XG4gICAgY29uc3Qgc3dhZ2dlciA9IG5ldyBTd2FnZ2VyQnVpbGRlcih0aGlzLmFwcCwge1xuICAgICAgdGl0bGU6IG9wdGlvbnMudGl0bGUsXG4gICAgICBkZXNjcmlwdGlvbjogb3B0aW9ucy5kZXNjcmlwdGlvbixcbiAgICAgIHZlcnNpb246IG9wdGlvbnMudmVyc2lvbixcbiAgICAgIHBhdGg6IG9wdGlvbnMucGF0aCB8fCBcImFwaVwiLFxuICAgICAgcGVyc2lzdEF1dGhvcml6YXRpb246IG9wdGlvbnMucGVyc2lzdEF1dGhvcml6YXRpb24gPz8gdHJ1ZSxcbiAgICAgIGFzc2V0c1BhdGg6IG9wdGlvbnMuYXNzZXRzUGF0aCxcbiAgICAgIGZhdmljb25GaWxlUGF0aDogb3B0aW9ucy5mYXZpY29uUGF0aCxcbiAgICAgIHRvcGJhckljb25GaWxlUGF0aDogb3B0aW9ucy50b3BiYXJJY29uUGF0aCxcbiAgICAgIHRvcGJhckJnQ29sb3I6IG9wdGlvbnMudG9wYmFyQmdDb2xvcixcbiAgICAgIG9wZW5BcGlKc29uUGF0aDogb3B0aW9ucy5vcGVuQXBpSnNvblBhdGgsXG4gICAgICBvcGVuQXBpWWFtbFBhdGg6IG9wdGlvbnMub3BlbkFwaVlhbWxQYXRoLFxuICAgIH0pO1xuICAgIHN3YWdnZXIuc2V0dXBTd2FnZ2VyKCk7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIFJlZ2lzdGVycyBvbmUgb3IgbW9yZSBnbG9iYWwgdmFsaWRhdGlvbiBwaXBlcy5cbiAgICpcbiAgICogQHN1bW1hcnlcbiAgICogRW5hYmxlcyByZXF1ZXN0IHBheWxvYWQgdmFsaWRhdGlvbiBhbmQgdHJhbnNmb3JtYXRpb24gZ2xvYmFsbHkgYWNyb3NzXG4gICAqIHRoZSBlbnRpcmUgTmVzdEpTIGFwcGxpY2F0aW9uLiBNdWx0aXBsZSBwaXBlcyBjYW4gYmUgY2hhaW5lZCB0b2dldGhlclxuICAgKiBmb3IgbW9kdWxhciBpbnB1dCB2YWxpZGF0aW9uLlxuICAgKlxuICAgKiBAcGFyYW0gey4uLlBpcGVUcmFuc2Zvcm1bXX0gcGlwZXMgLSBQaXBlIGluc3RhbmNlcyB0byByZWdpc3RlciBnbG9iYWxseS5cbiAgICogQHJldHVybiB7TmVzdEJvb3RzdHJhcGVyfSBSZXR1cm5zIHRoZSBjbGFzcyBmb3IgY2hhaW5pbmcuXG4gICAqL1xuICBzdGF0aWMgdXNlR2xvYmFsUGlwZXMoLi4ucGlwZXM6IFBpcGVUcmFuc2Zvcm1bXSkge1xuICAgIGlmIChwaXBlcy5sZW5ndGggPiAwKSB0aGlzLmFwcC51c2VHbG9iYWxQaXBlcyguLi5waXBlcyk7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIFJlZ2lzdGVycyBvbmUgb3IgbW9yZSBnbG9iYWwgZXhjZXB0aW9uIGZpbHRlcnMuXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIElmIG5vIGZpbHRlcnMgYXJlIHByb3ZpZGVkLCBpdCBhdXRvbWF0aWNhbGx5IHJlZ2lzdGVycyBhIGRlZmF1bHRcbiAgICogc2V0IG9mIHN0YW5kYXJkIGV4Y2VwdGlvbiBmaWx0ZXJzIGZvciBjb21tb24gZXJyb3IgdHlwZXMgbGlrZVxuICAgKiBgSHR0cEV4Y2VwdGlvbmAsIGBWYWxpZGF0aW9uRXhjZXB0aW9uYCwgYENvbmZsaWN0RXhjZXB0aW9uYCwgYW5kIG90aGVycy5cbiAgICpcbiAgICogQHBhcmFtIHsuLi5FeGNlcHRpb25GaWx0ZXJbXX0gZmlsdGVycyAtIE9wdGlvbmFsIGZpbHRlcnMgdG8gYXBwbHkgZ2xvYmFsbHkuXG4gICAqL1xuICBzdGF0aWMgdXNlR2xvYmFsRmlsdGVycyguLi5maWx0ZXJzOiBhbnlbXSkge1xuICAgIHRoaXMuYXBwLnVzZUdsb2JhbEZpbHRlcnMoXG4gICAgICAuLi4oZmlsdGVycy5sZW5ndGggPiAwID8gZmlsdGVycyA6IFtuZXcgRGVjYWZFeGNlcHRpb25GaWx0ZXIoKV0pXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogUmVnaXN0ZXJzIGdsb2JhbCBpbnRlcmNlcHRvcnMgZm9yIHJlcXVlc3QgYW5kIHJlc3BvbnNlIHRyYW5zZm9ybWF0aW9uLlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBJbnRlcmNlcHRvcnMgYWxsb3cgYWR2YW5jZWQgcmVxdWVzdC9yZXNwb25zZSBtYW5pcHVsYXRpb24gc3VjaCBhc1xuICAgKiBzZXJpYWxpemF0aW9uLCBsb2dnaW5nLCBvciB0cmFuc2Zvcm1hdGlvbi4gTXVsdGlwbGUgaW50ZXJjZXB0b3JzXG4gICAqIGNhbiBiZSBhZGRlZCBmb3IgbW9kdWxhciBjb25maWd1cmF0aW9uLlxuICAgKlxuICAgKiBAcGFyYW0gey4uLk5lc3RJbnRlcmNlcHRvcltdfSBpbnRlcmNlcHRvcnMgLSBJbnRlcmNlcHRvciBpbnN0YW5jZXMgdG8gcmVnaXN0ZXIuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBzdGF0aWMgdXNlR2xvYmFsSW50ZXJjZXB0b3JzKC4uLmludGVyY2VwdG9yczogTmVzdEludGVyY2VwdG9yW10pIHtcbiAgICBpZiAoaW50ZXJjZXB0b3JzLmxlbmd0aCA+IDApXG4gICAgICB0aGlzLmFwcC51c2VHbG9iYWxJbnRlcmNlcHRvcnMoLi4uaW50ZXJjZXB0b3JzKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogU3RhcnRzIHRoZSBOZXN0SlMgYXBwbGljYXRpb24gYW5kIGJpbmRzIGl0IHRvIHRoZSBnaXZlbiBwb3J0IGFuZCBob3N0LlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBMaXN0ZW5zIG9uIHRoZSBzcGVjaWZpZWQgcG9ydCBhbmQgb3B0aW9uYWxseSBhIGhvc3QuIE9uY2Ugc3RhcnRlZCxcbiAgICogbG9ncyB0aGUgYXBwbGljYXRpb24gVVJMIGZvciBlYXN5IGFjY2Vzcy4gVGhlIHN0YXJ0dXAgcHJvY2VzcyByZXNvbHZlc1xuICAgKiBvbmNlIHRoZSBhcHBsaWNhdGlvbiBpcyBzdWNjZXNzZnVsbHkgcnVubmluZy5cbiAgICpcbiAgICogQHBhcmFtIHtudW1iZXJ9IFtwb3J0PTMwMDBdIC0gUG9ydCBudW1iZXIgdG8gbGlzdGVuIG9uLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gW2hvc3RdIC0gT3B0aW9uYWwgaG9zdCBvciBJUCBhZGRyZXNzIHRvIGJpbmQgdG8uXG4gICAqIEBwYXJhbSB7Ym9vbGVhbn0gW2xvZz10cnVlXSAtIFdoZXRoZXIgdG8gbG9nIHRoZSBhcHBsaWNhdGlvbiBVUkwgdXBvbiBzdGFydHVwLlxuICAgKiBAcmV0dXJuIHtQcm9taXNlPHZvaWQ+fSBSZXNvbHZlcyBvbmNlIHRoZSBhcHBsaWNhdGlvbiBzdGFydHMgc3VjY2Vzc2Z1bGx5LlxuICAgKi9cbiAgc3RhdGljIGFzeW5jIHN0YXJ0KFxuICAgIHBvcnQ6IG51bWJlciA9IE51bWJlcihwcm9jZXNzLmVudi5QT1JUKSB8fCAzMDAwLFxuICAgIGhvc3Q6IHN0cmluZyB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZCxcbiAgICBsb2c6IGJvb2xlYW4gPSB0cnVlXG4gICkge1xuICAgIHRoaXMuYXBwLmxpc3Rlbihwb3J0LCBob3N0IGFzIGFueSkudGhlbihhc3luYyAoKSA9PiB7XG4gICAgICBpZiAobG9nKSB7XG4gICAgICAgIGNvbnN0IHVybCA9IGF3YWl0IHRoaXMuYXBwLmdldFVybCgpO1xuICAgICAgICB0aGlzLmxvZ2dlci5sb2coYPCfmoAgQXBwbGljYXRpb24gaXMgcnVubmluZyBhdDogJHt1cmx9YCk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cbn1cbiIsImltcG9ydCB7IEluamVjdCB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7XG4gIGluamVjdGFibGVTZXJ2aWNlS2V5LFxuICBNb2RlbFNlcnZpY2UsXG4gIFJlcG9zaXRvcnkgYXMgQ29yZVJlcG9zaXRvcnksXG4gIFNlcnZpY2UgYXMgQ29yZVNlcnZpY2UsXG59IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5cbmV4cG9ydCB0eXBlIERlY2FmRmFjdG9yeVByb3ZpZGVyID0ge1xuICBwcm92aWRlOiBzdHJpbmc7XG4gIHVzZUZhY3Rvcnk6ICgpID0+IHVua25vd247XG59O1xuXG5jb25zdCBwZW5kaW5nUHJvdmlkZXJzID0gbmV3IE1hcDxzdHJpbmcsIERlY2FmRmFjdG9yeVByb3ZpZGVyPigpO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBEcmFpbnMgdGhlIHByb3ZpZGVycyByZWdpc3RlcmVkIGJ5IHtAbGluayBTZXJ2aWNlfS97QGxpbmsgUmVwb3NpdG9yeX0gdXNhZ2VzLlxuICogQHN1bW1hcnkgQ29uc3VtZWQgYnkge0BsaW5rIERlY2FmQ29yZU1vZHVsZS5mb3JSb290fSB0byB3aXJlIHRoZSB0b2tlbnMgdGhhdCB0aG9zZSBwYXJhbWV0ZXJcbiAqIGRlY29yYXRvcnMgcmVmZXJlbmNlIGJhY2sgaW50byBOZXN0J3MgREkgY29udGFpbmVyLCBzaW5jZSBhIHBhcmFtZXRlciBkZWNvcmF0b3IgYWxvbmUgY2Fubm90XG4gKiBzdXBwbHkgYSBwcm92aWRlci5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFJlZ2lzdGVyZWREZWNhZlByb3ZpZGVycygpOiBEZWNhZkZhY3RvcnlQcm92aWRlcltdIHtcbiAgcmV0dXJuIEFycmF5LmZyb20ocGVuZGluZ1Byb3ZpZGVycy52YWx1ZXMoKSk7XG59XG5cbmZ1bmN0aW9uIHJlZ2lzdGVyUHJvdmlkZXIodG9rZW46IHN0cmluZywgdXNlRmFjdG9yeTogKCkgPT4gdW5rbm93bik6IHZvaWQge1xuICBpZiAocGVuZGluZ1Byb3ZpZGVycy5oYXModG9rZW4pKSByZXR1cm47XG4gIHBlbmRpbmdQcm92aWRlcnMuc2V0KHRva2VuLCB7IHByb3ZpZGU6IHRva2VuLCB1c2VGYWN0b3J5IH0pO1xufVxuXG5mdW5jdGlvbiBpc01vZGVsQ29uc3RydWN0b3IodmFsdWU6IHVua25vd24pOiB2YWx1ZSBpcyBDb25zdHJ1Y3Rvcjxhbnk+IHtcbiAgcmV0dXJuIHR5cGVvZiB2YWx1ZSA9PT0gXCJmdW5jdGlvblwiICYmIHZhbHVlLnByb3RvdHlwZSBpbnN0YW5jZW9mIE1vZGVsO1xufVxuXG5mdW5jdGlvbiBtb2RlbFNlcnZpY2VUb2tlbihtb2RlbDogQ29uc3RydWN0b3I8YW55Pik6IHN0cmluZyB7XG4gIHJldHVybiBgJHttb2RlbC5uYW1lfVNlcnZpY2VgO1xufVxuXG5mdW5jdGlvbiBtb2RlbFJlcG9zaXRvcnlUb2tlbihcbiAgbW9kZWw6IENvbnN0cnVjdG9yPGFueT4sXG4gIGZsYXZvdXI/OiBzdHJpbmdcbik6IHN0cmluZyB7XG4gIHJldHVybiBmbGF2b3VyID8gYCR7bW9kZWwubmFtZX1SZXBvc2l0b3J5QCR7Zmxhdm91cn1gIDogYCR7bW9kZWwubmFtZX1SZXBvc2l0b3J5YDtcbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQ29uc3RydWN0b3IgcGFyYW1ldGVyIGRlY29yYXRvciB0aGF0IGluamVjdHMgYSBkZWNhZiBzZXJ2aWNlLlxuICogQHN1bW1hcnkgVHJhbnNsYXRlcyBkZWNhZidzIHNlcnZpY2UgcmVzb2x1dGlvbiBBUElzIGludG8gYSBOZXN0IGBASW5qZWN0KClgIGNhbGw6XG4gKiAtIGBAU2VydmljZShTb21lTW9kZWwpYCBpbmplY3RzIHRoZSBgTW9kZWxTZXJ2aWNlPFNvbWVNb2RlbD5gIHNpbmdsZXRvbiBmb3IgdGhhdCBtb2RlbFxuICogICAodmlhIHtAbGluayBNb2RlbFNlcnZpY2UuZm9yTW9kZWx9KS5cbiAqIC0gYEBTZXJ2aWNlKFNvbWVTZXJ2aWNlQ2xhc3MpYCBvciBgQFNlcnZpY2UoXCJhbGlhc1wiKWAgaW5qZWN0cyB0aGUgbWF0Y2hpbmcgYEBzZXJ2aWNlKClgLWRlY29yYXRlZFxuICogICBkZWNhZiB7QGxpbmsgQ29yZVNlcnZpY2V9ICh2aWEge0BsaW5rIENvcmVTZXJ2aWNlLmdldH0pLlxuICogLSBgQFNlcnZpY2UoKWAgaW5mZXJzIHRoZSBwYXJhbWV0ZXIncyB0eXBlIGZyb20gZW1pdHRlZCBjb25zdHJ1Y3RvciBtZXRhZGF0YSBhbmQgcmVzb2x2ZXMgaXQgdGhlXG4gKiAgIHNhbWUgd2F5LCBwcm92aWRlZCB0aGUgcGFyYW1ldGVyIGlzIHR5cGVkIGFzIGEgY29uY3JldGUgY2xhc3MgKGdlbmVyaWNzIGxpa2UgYE1vZGVsU2VydmljZTxYPmBcbiAqICAgYXJlIGVyYXNlZCBhdCBydW50aW1lLCBzbyB0aGUgbW9kZWwtc2VydmljZSBmb3JtIGFsd2F5cyByZXF1aXJlcyBhbiBleHBsaWNpdCBhcmd1bWVudCkuXG4gKiBAcGFyYW0ga2V5IHRoZSBtb2RlbCBjbGFzcywgc2VydmljZSBjbGFzcywgb3IgYWxpYXMgdG8gcmVzb2x2ZS4gT21pdCB0byBpbmZlciBmcm9tIHRoZSBwYXJhbWV0ZXIgdHlwZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFNlcnZpY2Uoa2V5Pzogc3RyaW5nIHwgQ29uc3RydWN0b3I8YW55Pik6IFBhcmFtZXRlckRlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgdGFyZ2V0OiBvYmplY3QsXG4gICAgcHJvcGVydHlLZXk6IHN0cmluZyB8IHN5bWJvbCB8IHVuZGVmaW5lZCxcbiAgICBwYXJhbWV0ZXJJbmRleDogbnVtYmVyXG4gICk6IHZvaWQgPT4ge1xuICAgIGxldCByZXNvbHZlZDogc3RyaW5nIHwgQ29uc3RydWN0b3I8YW55PiB8IHVuZGVmaW5lZCA9IGtleTtcbiAgICBpZiAodHlwZW9mIHJlc29sdmVkID09PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgICBjb25zdCBwYXJhbVR5cGVzOiBhbnlbXSA9XG4gICAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoXCJkZXNpZ246cGFyYW10eXBlc1wiLCB0YXJnZXQpIHx8IFtdO1xuICAgICAgcmVzb2x2ZWQgPSBwYXJhbVR5cGVzW3BhcmFtZXRlckluZGV4XTtcbiAgICB9XG4gICAgaWYgKHR5cGVvZiByZXNvbHZlZCA9PT0gXCJ1bmRlZmluZWRcIikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgQFNlcnZpY2UoKSBjb3VsZCBub3QgZGV0ZXJtaW5lIGFuIGluamVjdGlvbiB0eXBlIGZvciBwYXJhbWV0ZXIgJHtwYXJhbWV0ZXJJbmRleH0gb2YgJHsodGFyZ2V0IGFzIENvbnN0cnVjdG9yPGFueT4pLm5hbWV9LiBQcm92aWRlIGFuIGV4cGxpY2l0IGFyZ3VtZW50LCBlLmcuIEBTZXJ2aWNlKFNvbWVTZXJ2aWNlKS5gXG4gICAgICApO1xuICAgIH1cblxuICAgIGNvbnN0IGFzTW9kZWwgPSB0eXBlb2YgcmVzb2x2ZWQgIT09IFwic3RyaW5nXCIgJiYgaXNNb2RlbENvbnN0cnVjdG9yKHJlc29sdmVkKTtcbiAgICBjb25zdCB0b2tlbiA9IGFzTW9kZWxcbiAgICAgID8gbW9kZWxTZXJ2aWNlVG9rZW4ocmVzb2x2ZWQgYXMgQ29uc3RydWN0b3I8YW55PilcbiAgICAgIDogaW5qZWN0YWJsZVNlcnZpY2VLZXkocmVzb2x2ZWQgYXMgc3RyaW5nIHwgQ29uc3RydWN0b3I8YW55Pik7XG5cbiAgICByZWdpc3RlclByb3ZpZGVyKHRva2VuLCAoKSA9PiB7XG4gICAgICBpZiAoYXNNb2RlbCkgcmV0dXJuIE1vZGVsU2VydmljZS5mb3JNb2RlbChyZXNvbHZlZCBhcyBDb25zdHJ1Y3Rvcjxhbnk+KTtcbiAgICAgIHJldHVybiB0eXBlb2YgcmVzb2x2ZWQgPT09IFwic3RyaW5nXCJcbiAgICAgICAgPyBDb3JlU2VydmljZS5nZXQocmVzb2x2ZWQpXG4gICAgICAgIDogQ29yZVNlcnZpY2UuZ2V0KHJlc29sdmVkIGFzIENvbnN0cnVjdG9yPGFueT4pO1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIEluamVjdCh0b2tlbikodGFyZ2V0LCBwcm9wZXJ0eUtleSwgcGFyYW1ldGVySW5kZXgpO1xuICB9O1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDb25zdHJ1Y3RvciBwYXJhbWV0ZXIgZGVjb3JhdG9yIHRoYXQgaW5qZWN0cyBhIGRlY2FmIHtAbGluayBDb3JlUmVwb3NpdG9yeX0gZm9yIGBtb2RlbGAuXG4gKiBAc3VtbWFyeSBVbmxpa2Uge0BsaW5rIFNlcnZpY2V9LCBgbW9kZWxgIGlzIGFsd2F5cyB0cmVhdGVkIGFzIGEgTW9kZWwgY2xhc3M6IHRoZSBzYW1lIGNsYXNzIHBhc3NlZFxuICogdG8gYEBTZXJ2aWNlKG1vZGVsKWAgYW5kIGBAUmVwb3NpdG9yeShtb2RlbClgIHJlc29sdmVzIHRvIHR3byBkaWZmZXJlbnQgaW5qZWN0ZWQgb2JqZWN0cyAoYVxuICogYE1vZGVsU2VydmljZTxNPmAgdnMgYSBgUmVwb3NpdG9yeTxNPmAgcmVzcGVjdGl2ZWx5KSwgdmlhIHtAbGluayBDb3JlUmVwb3NpdG9yeS5mb3JNb2RlbH0uXG4gKiBAcGFyYW0gbW9kZWwgdGhlIG1vZGVsIGNsYXNzIHRvIHJlc29sdmUgYSByZXBvc2l0b3J5IGZvci5cbiAqIEBwYXJhbSBmbGF2b3VyIG9wdGlvbmFsIGFkYXB0ZXIgZmxhdm91ci9hbGlhcywgZm9yd2FyZGVkIHRvIHtAbGluayBDb3JlUmVwb3NpdG9yeS5mb3JNb2RlbH0uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBSZXBvc2l0b3J5KFxuICBtb2RlbDogQ29uc3RydWN0b3I8YW55PixcbiAgZmxhdm91cj86IHN0cmluZ1xuKTogUGFyYW1ldGVyRGVjb3JhdG9yIHtcbiAgcmV0dXJuIChcbiAgICB0YXJnZXQ6IG9iamVjdCxcbiAgICBwcm9wZXJ0eUtleTogc3RyaW5nIHwgc3ltYm9sIHwgdW5kZWZpbmVkLFxuICAgIHBhcmFtZXRlckluZGV4OiBudW1iZXJcbiAgKTogdm9pZCA9PiB7XG4gICAgY29uc3QgdG9rZW4gPSBtb2RlbFJlcG9zaXRvcnlUb2tlbihtb2RlbCwgZmxhdm91cik7XG4gICAgcmVnaXN0ZXJQcm92aWRlcih0b2tlbiwgKCkgPT4gQ29yZVJlcG9zaXRvcnkuZm9yTW9kZWwobW9kZWwsIGZsYXZvdXIpKTtcbiAgICByZXR1cm4gSW5qZWN0KHRva2VuKSh0YXJnZXQsIHByb3BlcnR5S2V5LCBwYXJhbWV0ZXJJbmRleCk7XG4gIH07XG59XG4iLCJpbXBvcnQge1xuICBEeW5hbWljTW9kdWxlLFxuICBHbG9iYWwsXG4gIEluamVjdCxcbiAgTW9kdWxlLFxuICBPbkFwcGxpY2F0aW9uU2h1dGRvd24sXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgQVBQX0lOVEVSQ0VQVE9SLCBNb2R1bGVSZWYgfSBmcm9tIFwiQG5lc3Rqcy9jb3JlXCI7XG5pbXBvcnQgdHlwZSB7IERlY2FmTW9kdWxlT3B0aW9ucyB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQge1xuICBERUNBRl9BREFQVEVSX0lELFxuICBERUNBRl9IQU5ETEVSUyxcbiAgREVDQUZfTU9EVUxFX09QVElPTlMsXG59IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHtcbiAgRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yLFxufSBmcm9tIFwiLi9pbnRlcmNlcHRvcnNcIjtcbmltcG9ydCB7IERlY2FmSGFuZGxlckV4ZWN1dG9yLCBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4vcmVxdWVzdFwiO1xuaW1wb3J0IHsgQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IEFkYXB0ZXIsIFBlcnNpc3RlbmNlU2VydmljZSwgU2VydmljZSB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgZ2V0UmVnaXN0ZXJlZERlY2FmUHJvdmlkZXJzIH0gZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgSW50ZXJuYWxFcnJvciB9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgTG9nZ2VyLCBMb2dnaW5nIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQge1xuICBSZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXIsXG4gIHJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcixcbn0gZnJvbSBcIkBkZWNhZi10cy9mb3ItaHR0cC9zZXJ2ZXJcIjtcblxuQEdsb2JhbCgpXG5ATW9kdWxlKHt9KVxuZXhwb3J0IGNsYXNzIERlY2FmQ29yZU1vZHVsZTxDT05GLCBBREFQVEVSIGV4dGVuZHMgQWRhcHRlcjxDT05GLCBhbnksIGFueSwgYW55Pj5cbiAgaW1wbGVtZW50cyBPbkFwcGxpY2F0aW9uU2h1dGRvd25cbntcbiAgcHJpdmF0ZSBzdGF0aWMgX2xvZ2dlcjogTG9nZ2VyO1xuXG4gIHByaXZhdGUgc3RhdGljIF9wZXJzaXN0ZW5jZT86IFBlcnNpc3RlbmNlU2VydmljZTxBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueT4+O1xuXG4gIHByb3RlY3RlZCBzdGF0aWMgZ2V0IHBlcnNpc3RlbmNlKCk6IFBlcnNpc3RlbmNlU2VydmljZTxBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueT4+IHtcbiAgICBpZiAoIXRoaXMuX3BlcnNpc3RlbmNlKVxuICAgICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoXCJQZXJzaXN0ZW5jZSBzZXJ2aWNlIG5vdCBpbml0aWFsaXplZFwiKTtcbiAgICByZXR1cm4gdGhpcy5fcGVyc2lzdGVuY2U7XG4gIH1cblxuICBwcm90ZWN0ZWQgc3RhdGljIGdldCBsb2coKTogTG9nZ2VyIHtcbiAgICBpZiAoIXRoaXMuX2xvZ2dlcikgdGhpcy5fbG9nZ2VyID0gTG9nZ2luZy5mb3IoRGVjYWZDb3JlTW9kdWxlKTtcbiAgICByZXR1cm4gdGhpcy5fbG9nZ2VyO1xuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgQEluamVjdChERUNBRl9NT0RVTEVfT1BUSU9OUylcbiAgICBwcml2YXRlIHJlYWRvbmx5IG9wdGlvbnM6IERlY2FmTW9kdWxlT3B0aW9uczxDT05GLCBBREFQVEVSPixcbiAgICBwcml2YXRlIHJlYWRvbmx5IG1vZHVsZVJlZjogTW9kdWxlUmVmXG4gICkge31cblxuICBzdGF0aWMgZm9yUm9vdChvcHRpb25zOiBEZWNhZk1vZHVsZU9wdGlvbnMpOiBEeW5hbWljTW9kdWxlIHtcbiAgICBjb25zdCBsb2cgPSB0aGlzLmxvZy5mb3IodGhpcy5mb3JSb290KTtcbiAgICBjb25zdCBkZWNhZlByb3ZpZGVycyA9IGdldFJlZ2lzdGVyZWREZWNhZlByb3ZpZGVycygpO1xuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IERlY2FmQ29yZU1vZHVsZSxcbiAgICAgIHByb3ZpZGVyczogW1xuICAgICAgICB7IHByb3ZpZGU6IERFQ0FGX01PRFVMRV9PUFRJT05TLCB1c2VWYWx1ZTogb3B0aW9ucyB9LFxuICAgICAgICB7IHByb3ZpZGU6IERFQ0FGX0FEQVBURVJfSUQsIHVzZVZhbHVlOiB0aGlzLnBlcnNpc3RlbmNlPy5jbGllbnQgfSxcbiAgICAgICAge1xuICAgICAgICAgIHByb3ZpZGU6IERFQ0FGX0hBTkRMRVJTLFxuICAgICAgICAgIHVzZUZhY3Rvcnk6ICgpID0+XG4gICAgICAgICAgICBvcHRpb25zLmhhbmRsZXJzPy5tYXAoKEgpID0+IHtcbiAgICAgICAgICAgICAgbG9nLmluZm8oYFJlZ2lzdGVyZWQgcmVxdWVzdCBoYW5kbGVyOiAke0gubmFtZX1gKTtcbiAgICAgICAgICAgICAgcmV0dXJuIG5ldyBIKCk7XG4gICAgICAgICAgICB9KSA/PyBbXSxcbiAgICAgICAgfSxcbiAgICAgICAgRGVjYWZSZXF1ZXN0Q29udGV4dCxcbiAgICAgICAgRGVjYWZIYW5kbGVyRXhlY3V0b3IsXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBBUFBfSU5URVJDRVBUT1IsXG4gICAgICAgICAgdXNlQ2xhc3M6IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvcixcbiAgICAgICAgfSxcbiAgICAgICAgLi4uZGVjYWZQcm92aWRlcnMsXG4gICAgICBdLFxuICAgICAgZXhwb3J0czogW1xuICAgICAgICBERUNBRl9NT0RVTEVfT1BUSU9OUyxcbiAgICAgICAgREVDQUZfQURBUFRFUl9JRCxcbiAgICAgICAgREVDQUZfSEFORExFUlMsXG4gICAgICAgIERlY2FmUmVxdWVzdENvbnRleHQsXG4gICAgICAgIERlY2FmSGFuZGxlckV4ZWN1dG9yLFxuICAgICAgICAuLi5kZWNhZlByb3ZpZGVycy5tYXAoKHByb3ZpZGVyKSA9PiBwcm92aWRlci5wcm92aWRlKSxcbiAgICAgIF0sXG4gICAgfTtcbiAgfVxuXG4gIHN0YXRpYyBhc3luYyBib290UGVyc2lzdGVuY2UoXG4gICAgb3B0aW9uczogRGVjYWZNb2R1bGVPcHRpb25zXG4gICk6IFByb21pc2U8QWRhcHRlcjxhbnksIGFueSwgYW55LCBhbnk+W10+IHtcbiAgICBjb25zdCBsb2cgPSB0aGlzLmxvZy5mb3IodGhpcy5ib290UGVyc2lzdGVuY2UpO1xuXG4gICAgaWYgKCF0aGlzLl9wZXJzaXN0ZW5jZSkge1xuICAgICAgY29uc3QgdHJpbW1lZCA9IG9wdGlvbnMuY29uZi5tYXAoKFtjb250ciwgY2ZnLCAuLi5hcmdzXSkgPT4ge1xuICAgICAgICBjb25zdCBwb3NzaWJsZSA9IGFyZ3MucG9wKCk7XG4gICAgICAgIGlmICghcG9zc2libGUpIHJldHVybiBbY29udHIsIGNmZ107XG4gICAgICAgIHJldHVybiBbY29udHIsIGNmZywgLi4uYXJnc107XG4gICAgICB9KTtcbiAgICAgIHRoaXMuX3BlcnNpc3RlbmNlID0gbmV3IFBlcnNpc3RlbmNlU2VydmljZSgpO1xuICAgICAgYXdhaXQgdGhpcy5fcGVyc2lzdGVuY2UuYm9vdCh0cmltbWVkKTtcbiAgICAgIGNvbnN0IGNsaWVudHMgPSB0aGlzLl9wZXJzaXN0ZW5jZS5jbGllbnQ7XG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGNsaWVudHMubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgY29uc3QgYyA9IG9wdGlvbnMuY29uZltpXTtcbiAgICAgICAgY29uc3QgcG9zc2libGVUcmFuc2YgPSBjLnNsaWNlKDIsIGMubGVuZ3RoKTtcbiAgICAgICAgbGV0IHRyYW5zZm9ybWVyID0gcG9zc2libGVUcmFuc2YucG9wKCk7XG4gICAgICAgIGlmIChcbiAgICAgICAgICAhdHJhbnNmb3JtZXIgfHxcbiAgICAgICAgICAhKHRyYW5zZm9ybWVyIGFzIFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcjxhbnk+KS5mcm9tXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnN0IGNvbnRyID0gQWRhcHRlci50cmFuc2Zvcm1lckZvcihjbGllbnRzW2ldLmZsYXZvdXIpO1xuICAgICAgICAgIGlmICghY29udHIpXG4gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgICAgICAgYE5vIHRyYW5zZm9ybWVyIGZvdW5kIGZvciBmbGF2b3VyICR7Y2xpZW50c1tpXS5mbGF2b3VyfS4geW91IHNob3VsZCBlaXRoZXIgQHJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lciBvciBwcm92aWRlIGEgdHJhbnNmb3JtZXIgaW4gdGhlIGNvbmZpZ2BcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIHRyYW5zZm9ybWVyID0gKGNvbnRyIGFzIFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcjxhbnk+KS5mcm9tXG4gICAgICAgICAgICAgID8gY29udHJcbiAgICAgICAgICAgICAgOiBuZXcgKGNvbnRyIGFzIENvbnN0cnVjdG9yPFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcjxhbnk+PikoKTtcbiAgICAgICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcbiAgICAgICAgICAgICAgYEZhaWxlZCB0byBib290IHRyYW5zZm9ybWVyIGZvciAke2NsaWVudHNbaV0uZmxhdm91cn06ICR7ZX1gXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXIoY2xpZW50c1tpXS5mbGF2b3VyKSh0cmFuc2Zvcm1lcik7XG4gICAgICB9XG4gICAgICBsb2cuaW5mbyhcInBlcnNpc3RlbmNlIGxheWVyIGNyZWF0ZWQgc3VjY2Vzc2Z1bGx5IVwiKTtcblxuICAgICAgaWYgKG9wdGlvbnMuaW5pdGlhbGl6YXRpb24pIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBhd2FpdCBvcHRpb25zLmluaXRpYWxpemF0aW9uKCk7XG4gICAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihgRmFpbGVkIHRvIGluaXRpYWxpemUgYXBwbGljYXRpb246ICR7ZX1gKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB0aGlzLnBlcnNpc3RlbmNlLmNsaWVudDtcbiAgfVxuXG4gIGFzeW5jIG9uQXBwbGljYXRpb25TaHV0ZG93bigpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBsb2cgPSBEZWNhZkNvcmVNb2R1bGUubG9nLmZvcih0aGlzLm9uQXBwbGljYXRpb25TaHV0ZG93bik7XG4gICAgY29uc3QgYWRhcHRlcnM6IEFkYXB0ZXI8YW55LCBhbnksIGFueSwgYW55PltdID1cbiAgICAgIHRoaXMubW9kdWxlUmVmLmdldDxhbnk+KERFQ0FGX0FEQVBURVJfSUQpO1xuICAgIGZvciAoY29uc3QgYWRhcHRlciBvZiBhZGFwdGVycylcbiAgICAgIHRyeSB7XG4gICAgICAgIGlmIChhZGFwdGVyKSB7XG4gICAgICAgICAgbG9nLmluZm8oYFNodXR0aW5nIGRvd24gJHthZGFwdGVyLnRvU3RyaW5nKCl9YCk7XG4gICAgICAgICAgYXdhaXQgYWRhcHRlci5zaHV0ZG93bigpO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIHNodXRkb3duIGFwcGxpY2F0aW9uYCwgZSBhcyBFcnJvcik7XG4gICAgICB9XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IFNlcnZpY2Uuc2h1dGRvd24oKTtcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBzaHV0ZG93biBzZXJ2aWNlc2AsIGUgYXMgRXJyb3IpO1xuICAgIH1cbiAgfVxufVxuIiwiZXhwb3J0IGNvbnN0IExJU1RFTklOR19BREFQVEVSU19GTEFWT1VSUyA9IFN5bWJvbChcbiAgXCJMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlNcIlxuKTtcbiIsImltcG9ydCB7IE1vZGVsIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuXG5leHBvcnQgZnVuY3Rpb24gbm9ybWFsaXplRXZlbnRSZXNwb25zZShhcmdzOiBhbnlbXSk6IHVua25vd25bXSB7XG4gIGNvbnN0IFttb2RlbENvbnN0ciwgb3BlcmF0aW9uLCBpZCwgcGF5bG9hZF0gPSBhcmdzO1xuXG4gIGNvbnN0IG1vZGVsTmFtZSA9IG1vZGVsQ29uc3RyPy5uYW1lID8/IG1vZGVsQ29uc3RyO1xuXG4gIGNvbnN0IHNlcmlhbGl6ZWRQYXlsb2FkID0gQXJyYXkuaXNBcnJheShwYXlsb2FkKVxuICAgID8gcGF5bG9hZC5tYXAoKGUpID0+IHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBpZiAodHlwZW9mIGUuc2VyaWFsaXplID09PSBcImZ1bmN0aW9uXCIpIHJldHVybiBlLnNlcmlhbGl6ZSgpO1xuXG4gICAgICAgICAgY29uc29sZS53YXJuKFxuICAgICAgICAgICAgYFBheWxvYWQgaXRlbSBmb3IgJHttb2RlbE5hbWV9IGRvZXMgbm90IGhhdmUgc2VyaWFsaXplIG1ldGhvZCBhbmQgaXMgYW4gJHt0eXBlb2YgZX0sIGF0dGVtcHRpbmcgdG8gc3RyaW5naWZ5IGRpcmVjdGx5LiBJdGVtOiAke2V9YFxuICAgICAgICAgICk7XG4gICAgICAgICAgcmV0dXJuIHR5cGVvZiBlID09PSBcInN0cmluZ1wiID8gZSA6IEpTT04uc3RyaW5naWZ5KGUpO1xuICAgICAgICB9IGNhdGNoIChlcnI6IHVua25vd24pIHtcbiAgICAgICAgICBjb25zb2xlLndhcm4oYEZhaWxlZCB0byBzZXJpYWxpemUgcGF5bG9hZCBmb3IgJHttb2RlbE5hbWV9OiAke2Vycn1gKTtcbiAgICAgICAgICByZXR1cm4gdW5kZWZpbmVkO1xuICAgICAgICB9XG4gICAgICB9KVxuICAgIDogcGF5bG9hZCAmJiB0eXBlb2YgcGF5bG9hZC5zZXJpYWxpemUgPT09IFwiZnVuY3Rpb25cIlxuICAgICAgPyBwYXlsb2FkLnNlcmlhbGl6ZSgpXG4gICAgICA6IHR5cGVvZiBwYXlsb2FkID09PSBcInN0cmluZ1wiXG4gICAgICAgID8gcGF5bG9hZFxuICAgICAgICA6IEpTT04uc3RyaW5naWZ5KHBheWxvYWQpO1xuXG4gIGNvbnNvbGUuZGVidWcoXG4gICAgYE5vcm1hbGl6ZWQgZXZlbnQgcmVzcG9uc2UgZm9yIG1vZGVsICR7bW9kZWxOYW1lfSwgb3BlcmF0aW9uICR7b3BlcmF0aW9ufSwgaWQgJHtpZH06YCxcbiAgICBzZXJpYWxpemVkUGF5bG9hZFxuICApO1xuXG4gIHJldHVybiBbbW9kZWxOYW1lLCBvcGVyYXRpb24sIGlkLCBzZXJpYWxpemVkUGF5bG9hZF07XG59XG4iLCJpbXBvcnQgeyBEZWNhZkNvbnRyb2xsZXIgfSBmcm9tIFwiLi4vY29udHJvbGxlcnNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi4vcmVxdWVzdFwiO1xuaW1wb3J0IHsgQWRhcHRlciwgT2JzZXJ2ZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IENvbnRyb2xsZXIsIEluamVjdCwgTWVzc2FnZUV2ZW50LCBRdWVyeSwgU3NlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBpbnRlcnZhbCwgbWVyZ2UsIE9ic2VydmFibGUgfSBmcm9tIFwicnhqc1wiO1xuaW1wb3J0IHsgTG9nZ2luZyB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuaW1wb3J0IHsgTElTVEVOSU5HX0FEQVBURVJTX0ZMQVZPVVJTIH0gZnJvbSBcIi4vY29uc3RhbnRcIjtcbmltcG9ydCB7IERlY2FmU2VydmVyQ3R4IH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgbm9ybWFsaXplRXZlbnRSZXNwb25zZSB9IGZyb20gXCIuL3V0aWxzXCI7XG5pbXBvcnQgeyBtYXAsIHRhcCB9IGZyb20gXCJyeGpzL29wZXJhdG9yc1wiO1xuXG5AQ29udHJvbGxlcigpXG5leHBvcnQgY2xhc3MgRXZlbnRzQ29udHJvbGxlciBleHRlbmRzIERlY2FmQ29udHJvbGxlcjxEZWNhZlNlcnZlckN0eD4ge1xuICBwcml2YXRlIHJlYWRvbmx5IGFkYXB0ZXJzOiBBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueT5bXTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBjbGllbnRDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIEBJbmplY3QoTElTVEVOSU5HX0FEQVBURVJTX0ZMQVZPVVJTKSBmbGF2b3Vyczogc3RyaW5nW11cbiAgKSB7XG4gICAgc3VwZXIoY2xpZW50Q29udGV4dCwgRXZlbnRzQ29udHJvbGxlci5uYW1lKTtcbiAgICB0aGlzLmFkYXB0ZXJzID0gZmxhdm91cnMubWFwKChmbGF2b3VyKSA9PiAoQWRhcHRlciBhcyBhbnkpLmdldChmbGF2b3VyKSk7IC8vIGNoYW5nZSB0byBBZGFwdGVyLmNhY2hlKFwiXCIpXG4gIH1cblxuICBAU3NlKClcbiAgbGlzdGVuKCk6IE9ic2VydmFibGU8TWVzc2FnZUV2ZW50PiB7XG4gICAgY29uc3QgbG9nZ2VyID0gTG9nZ2luZy5mb3IoRXZlbnRzQ29udHJvbGxlci5uYW1lKTtcblxuICAgIGNvbnN0IGV2ZW50cyQgPSBuZXcgT2JzZXJ2YWJsZTxNZXNzYWdlRXZlbnQ+KChvYnNlcnZlcikgPT4ge1xuICAgICAgY29uc3Qgb2JzZXJ2ZXJJZCA9XG4gICAgICAgIGBCLSR7TWF0aC5yYW5kb20oKS50b1N0cmluZygzNikuc2xpY2UoMiwgOCl9YC50b1VwcGVyQ2FzZSgpO1xuXG4gICAgICBsb2dnZXIuaW5mbyhcbiAgICAgICAgYENyZWF0aW5nIFNTRSBvYnNlcnZlcjogJHtvYnNlcnZlcklkfSBmb3IgY2xpZW50ICR7dGhpcy5jbGllbnRDb250ZXh0LnV1aWR9YFxuICAgICAgKTtcbiAgICAgIGNvbnN0IGNiID0gbmV3IChjbGFzcyBpbXBsZW1lbnRzIE9ic2VydmVyIHtcbiAgICAgICAgb2JzZXJ2ZXJJZCA9IG9ic2VydmVySWQ7XG4gICAgICAgIHJlZnJlc2goLi4uYXJnczogYW55W10pOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAgICAgICBsb2dnZXIuZGVidWcoXG4gICAgICAgICAgICBgU1NFIG9ic2VydmVyICR7dGhpcy5vYnNlcnZlcklkfSByZWNlaXZlZCByZWZyZXNoIGV2ZW50YFxuICAgICAgICAgICk7XG4gICAgICAgICAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgY29uc3QgZGF0YSA9IG5vcm1hbGl6ZUV2ZW50UmVzcG9uc2UoYXJncyk7XG4gICAgICAgICAgICBvYnNlcnZlci5uZXh0KHsgdHlwZTogXCJtZXNzYWdlXCIsIGRhdGEgfSk7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoXG4gICAgICAgICAgICAgIGBTU0Ugb2JzZXJ2ZXIgJHt0aGlzLm9ic2VydmVySWR9IGV2ZW50IHB1c2hlZCB0byBjbGllbnRgXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KSgpO1xuXG4gICAgICBsb2dnZXIudmVyYm9zZShcbiAgICAgICAgYFJlZ2lzdGVyaW5nIG9ic2VydmVyICR7b2JzZXJ2ZXJJZH0gYWNyb3NzICR7dGhpcy5hZGFwdGVycy5sZW5ndGh9IGFkYXB0ZXIocylgXG4gICAgICApO1xuICAgICAgZm9yIChjb25zdCBhZGFwdGVyIG9mIHRoaXMuYWRhcHRlcnMpIHtcbiAgICAgICAgY29uc3QgYWRhcHRlck5hbWUgPSBhZGFwdGVyPy5jb25zdHJ1Y3Rvcj8ubmFtZSA/PyBcIlVua25vd25BZGFwdGVyXCI7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgbG9nZ2VyLmRlYnVnKFxuICAgICAgICAgICAgYFJlZ2lzdGVyaW5nIG9ic2VydmVyICR7b2JzZXJ2ZXJJZH0gaW4gYWRhcHRlciAke2FkYXB0ZXJOYW1lfWBcbiAgICAgICAgICApO1xuICAgICAgICAgIGFkYXB0ZXIub2JzZXJ2ZShjYik7XG4gICAgICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhcbiAgICAgICAgICAgIGBGYWlsZWQgdG8gcmVnaXN0ZXIgb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfSBpbiBhZGFwdGVyICR7YWRhcHRlck5hbWV9OiAke2U/Lm1lc3NhZ2UgfHwgZX1gXG4gICAgICAgICAgKTtcbiAgICAgICAgICBsb2dnZXIuZXJyb3IoZSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgcmV0dXJuICgpID0+IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBDbGVhbmluZyB1cCBTU0Ugb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfWApO1xuXG4gICAgICAgIGZvciAoY29uc3QgYWRhcHRlciBvZiB0aGlzLmFkYXB0ZXJzKSB7XG4gICAgICAgICAgY29uc3QgYWRhcHRlck5hbWUgPSBhZGFwdGVyPy5jb25zdHJ1Y3Rvcj8ubmFtZSA/PyBcIlVua25vd25BZGFwdGVyXCI7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhcbiAgICAgICAgICAgICAgYFVucmVnaXN0ZXJpbmcgb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfSBmcm9tIGFkYXB0ZXIgJHthZGFwdGVyTmFtZX1gXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgYWRhcHRlci51bk9ic2VydmUoY2IpO1xuICAgICAgICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKFxuICAgICAgICAgICAgICBgRmFpbGVkIGR1cmluZyBjbGVhbnVwIG9mIG9ic2VydmVyICR7b2JzZXJ2ZXJJZH0gaW4gYWRhcHRlciAke2FkYXB0ZXJOYW1lfTogJHtlPy5tZXNzYWdlIHx8IGV9YFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihlKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH07XG4gICAgfSk7XG5cbiAgICBjb25zdCBIRUFSVEJFQVRfSU5URVJWQUxfTVMgPSAxNTAwMDtcbiAgICBjb25zdCBoZWFydGJlYXQkID0gaW50ZXJ2YWwoSEVBUlRCRUFUX0lOVEVSVkFMX01TKS5waXBlKFxuICAgICAgdGFwKCgpID0+IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiU2VuZGluZyBoZWFydGJlYXRcIik7XG4gICAgICB9KSxcbiAgICAgIG1hcChcbiAgICAgICAgKCk6IE1lc3NhZ2VFdmVudCA9PiAoe1xuICAgICAgICAgIHR5cGU6IFwiaGVhcnRiZWF0XCIsXG4gICAgICAgICAgZGF0YToge1xuICAgICAgICAgICAgdHM6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICAgICAgICB9LFxuICAgICAgICB9KVxuICAgICAgKVxuICAgICk7XG5cbiAgICByZXR1cm4gbWVyZ2UoZXZlbnRzJCwgaGVhcnRiZWF0JCk7XG4gIH1cblxuICBAU3NlKFwiLzptb2RlbFwiKVxuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gIGxpc3RlbkZvck1vZGVsKEBRdWVyeShcIm1vZGVsXCIpIG1vZGVsOiBzdHJpbmcpOiBPYnNlcnZhYmxlPE1lc3NhZ2VFdmVudD4ge1xuICAgIGNvbnN0IGxvZ2dlciA9IExvZ2dpbmcuZm9yKEV2ZW50c0NvbnRyb2xsZXIubmFtZSk7XG5cbiAgICByZXR1cm4gbmV3IE9ic2VydmFibGU8TWVzc2FnZUV2ZW50Pigob2JzZXJ2ZXIpID0+IHtcbiAgICAgIGNvbnN0IGNiID0gbmV3IChjbGFzcyBpbXBsZW1lbnRzIE9ic2VydmVyIHtcbiAgICAgICAgcmVmcmVzaCguLi5hcmdzOiBhbnlbXSk6IFByb21pc2U8dm9pZD4ge1xuICAgICAgICAgIHJldHVybiBQcm9taXNlLnJlc29sdmUoKS50aGVuKCgpID0+IHtcbiAgICAgICAgICAgIG9ic2VydmVyLm5leHQoeyBkYXRhOiBhcmdzIH0gYXMgYW55KTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfSkoKTtcblxuICAgICAgdHJ5IHtcbiAgICAgICAgZm9yIChjb25zdCBhZGFwdGVyIG9mIHRoaXMuYWRhcHRlcnMpIHtcbiAgICAgICAgICBhZGFwdGVyLm9ic2VydmUoY2IpO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgb2JzZXJ2ZXIuZXJyb3IoYEZhaWxlZCB0byBvYnNlcnZlIGV2ZW50OiAke2UubWVzc2FnZSB8fCBlfWApO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGZvciAoY29uc3QgYWRhcHRlciBvZiB0aGlzLmFkYXB0ZXJzKSB7XG4gICAgICAgICAgICBhZGFwdGVyLnVuT2JzZXJ2ZShjYik7XG4gICAgICAgICAgfVxuICAgICAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgICAgICBsb2dnZXIuZXJyb3IoZSk7XG4gICAgICAgIH1cbiAgICAgIH07XG4gICAgfSk7XG4gIH1cbn1cbiIsImltcG9ydCB7IER5bmFtaWNNb2R1bGUsIE1vZHVsZSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRXZlbnRzQ29udHJvbGxlciB9IGZyb20gXCIuL0V2ZW50c0NvbnRyb2xsZXJcIjtcbmltcG9ydCB7IFJvdXRlck1vZHVsZSB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcbmltcG9ydCB7IExJU1RFTklOR19BREFQVEVSU19GTEFWT1VSUyB9IGZyb20gXCIuL2NvbnN0YW50XCI7XG5pbXBvcnQgeyBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4uL3JlcXVlc3RcIjtcblxuQE1vZHVsZSh7fSlcbmV4cG9ydCBjbGFzcyBEZWNhZlN0cmVhbU1vZHVsZSB7XG4gIHN0YXRpYyBmb3JGbGF2b3VycyhcbiAgICBmbGF2b3Vyczogc3RyaW5nW10sXG4gICAgcGF0aDogc3RyaW5nID0gXCJldmVudHNcIlxuICApOiBEeW5hbWljTW9kdWxlIHtcbiAgICByZXR1cm4ge1xuICAgICAgbW9kdWxlOiBEZWNhZlN0cmVhbU1vZHVsZSxcbiAgICAgIGNvbnRyb2xsZXJzOiBbRXZlbnRzQ29udHJvbGxlcl0sXG4gICAgICBpbXBvcnRzOiBbXG4gICAgICAgIFJvdXRlck1vZHVsZS5yZWdpc3RlcihbXG4gICAgICAgICAge1xuICAgICAgICAgICAgcGF0aDogcGF0aC5yZXBsYWNlKC9eXFwvLywgXCJcIiksXG4gICAgICAgICAgICBtb2R1bGU6IERlY2FmU3RyZWFtTW9kdWxlLFxuICAgICAgICAgIH0sXG4gICAgICAgIF0pLFxuICAgICAgXSxcbiAgICAgIHByb3ZpZGVyczogW1xuICAgICAgICBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgICAgICB7XG4gICAgICAgICAgcHJvdmlkZTogTElTVEVOSU5HX0FEQVBURVJTX0ZMQVZPVVJTLFxuICAgICAgICAgIHVzZVZhbHVlOiBmbGF2b3VycyA/PyBbXSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgRHluYW1pY01vZHVsZSwgRm9yd2FyZFJlZmVyZW5jZSwgTW9kdWxlLCBUeXBlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBEZWNhZk1vZHVsZU9wdGlvbnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZDb3JlTW9kdWxlIH0gZnJvbSBcIi4vY29yZS1tb2R1bGVcIjtcbmltcG9ydCB7IEFkYXB0ZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IGdldE1vZHVsZUZvciB9IGZyb20gXCIuL2RlY2FmLW1vZGVsL2luZGV4XCI7XG5pbXBvcnQgeyBEZWNhZlN0cmVhbU1vZHVsZSB9IGZyb20gXCIuL2V2ZW50cy1tb2R1bGVcIjtcblxuLyoqXG4gKiBAcHVibGljQXBpXG4gKi9cbkBNb2R1bGUoe30pXG5leHBvcnQgY2xhc3MgRGVjYWZNb2R1bGUge1xuICBzdGF0aWMgYXN5bmMgZm9yUm9vdEFzeW5jKFxuICAgIG9wdGlvbnM6IERlY2FmTW9kdWxlT3B0aW9uc1xuICApOiBQcm9taXNlPER5bmFtaWNNb2R1bGU+IHtcbiAgICBjb25zdCB7IGF1dG9Db250cm9sbGVycywgYXV0b1NlcnZpY2VzIH0gPSBvcHRpb25zO1xuXG4gICAgY29uc3QgYWRhcHRlcnM6IEFkYXB0ZXI8YW55LCBhbnksIGFueSwgYW55PltdID1cbiAgICAgIGF3YWl0IERlY2FmQ29yZU1vZHVsZS5ib290UGVyc2lzdGVuY2Uob3B0aW9ucyk7XG4gICAgY29uc3QgZmxhdm91cnMgPSBhZGFwdGVycy5tYXAoKGFkYXB0ZXIpID0+IGFkYXB0ZXIuZmxhdm91cik7XG5cbiAgICBjb25zdCBpbXBvcnRzOlxuICAgICAgfCAoXG4gICAgICAgICAgfCBEeW5hbWljTW9kdWxlXG4gICAgICAgICAgfCBUeXBlPGFueT5cbiAgICAgICAgICB8IFByb21pc2U8RHluYW1pY01vZHVsZT5cbiAgICAgICAgICB8IEZvcndhcmRSZWZlcmVuY2U8YW55PlxuICAgICAgICApW11cbiAgICAgIHwgdW5kZWZpbmVkID0gW0RlY2FmQ29yZU1vZHVsZS5mb3JSb290KG9wdGlvbnMpXTtcblxuICAgIGlmIChhdXRvQ29udHJvbGxlcnMpIHtcbiAgICAgICAgZmxhdm91cnMuZm9yRWFjaCgoZmxhdm91cikgPT4ge1xuICAgICAgICAgIGltcG9ydHMucHVzaChcbiAgICAgICAgICAgIGdldE1vZHVsZUZvcihmbGF2b3VyKS5mb3JSb290KGZsYXZvdXIsIHtcbiAgICAgICAgICAgICAgYXV0b1NlcnZpY2VzLFxuICAgICAgICAgICAgICBjb250cm9sbGVyRXhwb3N1cmU6IG9wdGlvbnMuY29udHJvbGxlckV4cG9zdXJlLFxuICAgICAgICAgICAgICBjb250cm9sbGVyQ29uZmlnOiBvcHRpb25zLmNvbnRyb2xsZXJDb25maWcsXG4gICAgICAgICAgICAgIGFnZ3JlZ2F0aW9uczogb3B0aW9ucy5hZ2dyZWdhdGlvbnMsXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICk7XG4gICAgICAgIH0pO1xuICAgICAgfVxuXG4gICAgaWYgKG9wdGlvbnMub2JzZXJ2ZXJPcHRpb25zPy5lbmFibGVPYnNlcnZlckV2ZW50cykge1xuICAgICAgaW1wb3J0cy5wdXNoKFxuICAgICAgICBEZWNhZlN0cmVhbU1vZHVsZS5mb3JGbGF2b3VycyhcbiAgICAgICAgICBvcHRpb25zLm9ic2VydmVyT3B0aW9ucy5vYnNlcnZlckZsYXZvdXJzIHx8IGZsYXZvdXJzXG4gICAgICAgIClcbiAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIG1vZHVsZTogRGVjYWZNb2R1bGUsXG4gICAgICBpbXBvcnRzOiBpbXBvcnRzLFxuICAgIH07XG4gIH1cbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHJ1bk1pZ3JhdGlvbnMoKTogUHJvbWlzZTx2b2lkPiB7fVxuIiwiaW1wb3J0IHtNb2RlbH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHtJbnRlcm5hbEVycm9yfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7UmVwb3NpdG9yeX0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiByZXBvRm9yTW9kZWwobW9kZWw6IHN0cmluZykge1xuXHRjb25zdCBtID0gTW9kZWwuZ2V0KG1vZGVsKTtcblx0aWYgKCFtKVxuXHRcdHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGBGYWlsZWQgdG8gZmluZCByZXBvc2l0b3J5IGZvciAke21vZGVsfWApXG5cdGNvbnN0IHJlcG8gPSBSZXBvc2l0b3J5LmZvck1vZGVsKG0pO1xuXHRyZXR1cm4gcmVwbztcbn0iLCIvKipcbiAqIEBtb2R1bGUgZm9yLW5lc3RcbiAqIEBkZXNjcmlwdGlvbiBUaGlzIG1vZHVsZSBzZXJ2ZXMgYXMgdGhlIG1haW4gZW50cnkgcG9pbnQgZm9yIHRoZSB0cy13b3Jrc3BhY2UgbGlicmFyeS4gSXQgYWdncmVnYXRlcyBhbmQgZXhwb3J0c1xuICogZnVuY3Rpb25hbGl0eSBmcm9tIHZhcmlvdXMgc3VibW9kdWxlcyBhbmQgdXRpbGl0aWVzIHdpdGhpbiB0aGUgcHJvamVjdC5cbiAqXG4gKiBUaGUgbW9kdWxlIGluY2x1ZGVzOlxuICogMS4gVXRpbGl0eSBmdW5jdGlvbnMgYW5kIHR5cGVzIGZyb20gdGhlIFwiLi91dGlsc1wiIGRpcmVjdG9yeTpcbiAqICAgIC0gVGhlc2UgbGlrZWx5IGNvbnRhaW4gaGVscGVyIGZ1bmN0aW9ucywgY29tbW9uIHR5cGVzLCBhbmQgc2hhcmVkIGZ1bmN0aW9uYWxpdHkgdXNlZCB0aHJvdWdob3V0IHRoZSBwcm9qZWN0LlxuICogICAgLSBNYXkgaW5jbHVkZSBvcGVyYXRpb25zIGZvciBkYXRhIG1hbmlwdWxhdGlvbiwgdHlwZSBjaGVja2luZywgb3Igb3RoZXIgZ2VuZXJhbC1wdXJwb3NlIHV0aWxpdGllcy5cbiAqXG4gKiAyLiBBIG5hbWVzcGFjZSBhbmQgcmVsYXRlZCB0eXBlcyBmcm9tIHRoZSBcIi4vbmFtZXNwYWNlXCIgZGlyZWN0b3J5OlxuICogICAgLSBUaGlzIGNvdWxkIGNvbnRhaW4gZG9tYWluLXNwZWNpZmljIGNvZGUgb3IgYSBjb2xsZWN0aW9uIG9mIHJlbGF0ZWQgZnVuY3Rpb25hbGl0eS5cbiAqICAgIC0gTWlnaHQgaW5jbHVkZSBpbnRlcmZhY2VzLCB0eXBlcywgb3IgY2xhc3NlcyB0aGF0IHJlcHJlc2VudCBjb3JlIGNvbmNlcHRzIGluIHRoZSBsaWJyYXJ5LlxuICpcbiAqIDMuIEEgVkVSU0lPTiBjb25zdGFudDpcbiAqICAgIC0gUmVwcmVzZW50cyB0aGUgY3VycmVudCB2ZXJzaW9uIG9mIHRoZSBtb2R1bGUuXG4gKiAgICAtIFVzZWZ1bCBmb3IgdmVyc2lvbiBjaGVja2luZyBhbmQgY29tcGF0aWJpbGl0eSBwdXJwb3Nlcy5cbiAqXG4gKiBUaGlzIHN0cnVjdHVyZSBwcm92aWRlcyBhIGNsZWFuIGFuZCBvcmdhbml6ZWQgZXhwb3J0IG9mIHRoZSBtb2R1bGUncyBmdW5jdGlvbmFsaXR5LCBhbGxvd2luZyBjb25zdW1lcnNcbiAqIHRvIGVhc2lseSBpbXBvcnQgYW5kIHVzZSBzcGVjaWZpYyBwYXJ0cyBvZiB0aGUgbGlicmFyeSBhcyBuZWVkZWQuXG4gKi9cblxuaW1wb3J0IHsgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCBcIi4vZGVjb3JhdGlvblwiO1xuXG5leHBvcnQgKiBmcm9tIFwiLi9kZWNvcmF0aW9uXCI7IC8vIG9uIHRvcCBvbiBwdXJwb3NlXG5leHBvcnQgKiBmcm9tIFwiLi9kZWNhZi1tb2RlbFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXV0aFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZmFjdG9yeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vaW50ZXJjZXB0b3JzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9vdmVycmlkZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3JlcXVlc3RcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2NvbnN0YW50c1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29udHJvbGxlcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL21vZHVsZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29yZS1tb2R1bGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3R5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kZWNvcmF0b3JzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi91dGlsc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZXZlbnRzLW1vZHVsZVwiO1xuLy8gZXhwb3J0ICogZnJvbSBcIi4vd2ViaG9va3NcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgdGhlIHRzLXdvcmtzcGFjZSBtb2R1bGUuXG4gKiBUaGUgYWN0dWFsIHZlcnNpb24gbnVtYmVyIGlzIHJlcGxhY2VkIGR1cmluZyB0aGUgYnVpbGQgcHJvY2Vzcy5cbiAqIEBjb25zdGFudFxuICogQHR5cGUge3N0cmluZ31cbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFJlcHJlc2VudHMgdGhlIGN1cnJlbnQgY29tbWl0IGhhc2ggb2YgdGhlIG1vZHVsZSBidWlsZC5cbiAqIEBzdW1tYXJ5IFN0b3JlcyB0aGUgY3VycmVudCBnaXQgY29tbWl0IGhhc2ggZm9yIHRoZSBwYWNrYWdlLiBUaGUgYnVpbGQgcmVwbGFjZXNcbiAqIHRoZSBwbGFjZWhvbGRlciB3aXRoIHRoZSBhY3R1YWwgY29tbWl0IGhhc2ggYXQgcHVibGlzaCB0aW1lLlxuICogQGNvbnN0IENPTU1JVFxuICovXG5leHBvcnQgY29uc3QgQ09NTUlUID0gXCIjI0NPTU1JVCMjXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFJlcHJlc2VudHMgdGhlIGZ1bGwgdmVyc2lvbiBzdHJpbmcgb2YgdGhlIG1vZHVsZS5cbiAqIEBzdW1tYXJ5IFN0b3JlcyB0aGUgc2VtdmVyIHZlcnNpb24gYW5kIGNvbW1pdCBoYXNoIGZvciB0aGUgcGFja2FnZS5cbiAqIFRoZSBidWlsZCByZXBsYWNlcyB0aGUgcGxhY2Vob2xkZXIgd2l0aCB0aGUgYWN0dWFsIGA8dmVyc2lvbj4tPGNvbW1pdD5gIHZhbHVlIGF0IHB1Ymxpc2ggdGltZS5cbiAqIEBjb25zdCBGVUxMX1ZFUlNJT05cbiAqL1xuZXhwb3J0IGNvbnN0IEZVTExfVkVSU0lPTiA9IFwiIyNGVUxMX1ZFUlNJT04jI1wiO1xuXG5leHBvcnQgY29uc3QgUEFDS0FHRV9OQU1FID0gXCIjI1BBQ0tBR0UjI1wiO1xuXG5NZXRhZGF0YS5hbGxvd1JlcmVnaXN0cmF0aW9uKHRydWUpO1xuTWV0YWRhdGEucmVnaXN0ZXJMaWJyYXJ5KFBBQ0tBR0VfTkFNRSwgVkVSU0lPTik7XG5NZXRhZGF0YS5hbGxvd1JlcmVnaXN0cmF0aW9uKGZhbHNlKTtcbiJdLCJuYW1lcyI6WyJERUNPUkFUT1JTX1BSRUZJWCIsIkRFQ09SQVRPUlMiLCJBUElfT1BFUkFUSU9OIiwiQVBJX1JFU1BPTlNFIiwiQVBJX1BST0RVQ0VTIiwiQVBJX0NPTlNVTUVTIiwiQVBJX1RBR1MiLCJBUElfQ0FMTEJBQ0tTIiwiQVBJX1BBUkFNRVRFUlMiLCJBUElfSEVBREVSUyIsIkFQSV9NT0RFTF9QUk9QRVJUSUVTIiwiQVBJX01PREVMX1BST1BFUlRJRVNfQVJSQVkiLCJBUElfU0VDVVJJVFkiLCJBUElfRVhDTFVERV9FTkRQT0lOVCIsIkFQSV9FWENMVURFX0NPTlRST0xMRVIiLCJBUElfRVhUUkFfTU9ERUxTIiwiQVBJX0VYVEVOU0lPTiIsIkFQSV9TQ0hFTUEiLCJBUElfREVGQVVMVF9HRVRURVIiLCJBUElfTElOSyIsImlzQ29uc3RydWN0b3IiLCJ2YWwiLCJNRVRBREFUQV9GQUNUT1JZX05BTUUiLCJNRVRIT0RfTUVUQURBVEEiLCJjcmVhdGVNZXRob2REZWNvcmF0b3IiLCJtZXRha2V5IiwibWV0YWRhdGEiLCJvdmVycmlkZUV4aXN0aW5nIiwidGFyZ2V0Iiwia2V5IiwiZGVzY3JpcHRvciIsInByZXZWYWx1ZSIsIlJlZmxlY3QiLCJnZXRNZXRhZGF0YSIsInZhbHVlIiwiZGVmaW5lTWV0YWRhdGEiLCJjcmVhdGVDbGFzc0RlY29yYXRvciIsImNyZWF0ZVByb3BlcnR5RGVjb3JhdG9yIiwicHJvcGVydHlLZXkiLCJwcm9wZXJ0aWVzIiwiaW5jbHVkZXMiLCJleGlzdGluZ01ldGFkYXRhIiwibmV3TWV0YWRhdGEiLCJwaWNrQnkiLCJuZWdhdGUiLCJpc1VuZGVmaW5lZCIsIm1ldGFkYXRhVG9TYXZlIiwidHlwZSIsImNvbnN0cnVjdG9yIiwiY3JlYXRlTWl4ZWREZWNvcmF0b3IiLCJtZXRhZGF0YXMiLCJBcnJheSIsImlzQXJyYXkiLCJwcmV2aW91c01ldGFkYXRhIiwiT2JqZWN0IiwiYXNzaWduIiwiY3JlYXRlUGFyYW1EZWNvcmF0b3IiLCJpbml0aWFsIiwicGFyYW1PcHRpb25zIiwicGFyYW1ldGVycyIsInByb3BlcnR5S2V5cyIsImdldE93blByb3BlcnR5TmFtZXMiLCJwcm90b3R5cGUiLCJtZXRob2REZXNjcmlwdG9yIiwiZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yIiwiaXNBcGlNZXRob2QiLCJoYXNNZXRhZGF0YSIsImdldFR5cGVJc0FycmF5VHVwbGUiLCJpbnB1dCIsImlzQXJyYXlGbGFnIiwiaXNJbnB1dEFycmF5Iiwib2JqIiwiZ2V0RW51bVZhbHVlcyIsImVudW1UeXBlIiwibnVtZXJpY1ZhbHVlcyIsInZhbHVlcyIsImZpbHRlciIsIm1hcCIsInRvU3RyaW5nIiwia2V5cyIsImdldEVudW1UeXBlIiwiaGFzU3RyaW5nIiwiaXNTdHJpbmciLCJsZW5ndGgiLCJyZXF1aXJlZCIsImlzRW51bUFycmF5Iiwib3B0cyIsImVudW0iLCJ1bmRlZmluZWQiLCJBcGlQcm9wZXJ0eSIsIm9wdGlvbnMiLCJjcmVhdGVBcGlQcm9wZXJ0eURlY29yYXRvciIsImVudW1WYWx1ZXMiLCJpdGVtcyIsIkFwaU9wZXJhdGlvbkZyb21Nb2RlbCIsIk1vZGVsQ29uc3RydWN0b3IiLCJ2ZXJiIiwicGF0aCIsImh0dHBUb0NydWQiLCJHRVQiLCJPcGVyYXRpb25LZXlzIiwiUkVBRCIsIkdldCIsIlBPU1QiLCJDUkVBVEUiLCJQb3N0IiwiUFVUIiwiVVBEQVRFIiwiUHV0IiwiUEFUQ0giLCJQYXRjaCIsIkRFTEVURSIsIkRlbGV0ZSIsImNydWRPcCIsIkh0dHBNZXRob2REZWNvcmF0b3IiLCJyZXNvbHZlQmxvY2tUYXJnZXQiLCJjb3JlSXNPcGVyYXRpb25CbG9ja2VkIiwia2luZCIsImFwcGx5IiwiQXBpRXhjbHVkZUVuZHBvaW50IiwiQnVsa0FwaU9wZXJhdGlvbkZyb21Nb2RlbCIsIkJ1bGtDcnVkT3BlcmF0aW9uS2V5cyIsIlJFQURfQUxMIiwiQ1JFQVRFX0FMTCIsIlVQREFURV9BTEwiLCJERUxFVEVfQUxMIiwibm9ybWFsaXplZCIsInJlcGxhY2UiLCJzdGF0ZW1lbnRUYXJnZXRzIiwiUHJlcGFyZWRTdGF0ZW1lbnRLZXlzIiwiTElTVF9CWSIsIlBBR0VfQlkiLCJGSU5EIiwiUEFHRSIsIkZJTkRfT05FX0JZIiwiRklORF9CWSIsIkNPVU5UX09GIiwiTUFYX09GIiwiTUlOX09GIiwiQVZHX09GIiwiU1VNX09GIiwiRElTVElOQ1RfT0YiLCJHUk9VUF9PRiIsInN0YXJ0c1dpdGgiLCJzdGF0ZW1lbnRWYWx1ZSIsIkFwaVBhcmFtc0Zyb21Nb2RlbCIsInByb3BzIiwiZGVjb3JhdG9ycyIsInAiLCJBcGlQYXJhbURlYyIsIm5hbWUiLCJkZXNjcmlwdGlvbiIsIlN0cmluZyIsImFwcGx5RGVjb3JhdG9ycyIsIkRlY2FmQm9keSIsImRhdGEiLCJjdHgiLCJyZXF1ZXN0Iiwic3dpdGNoVG9IdHRwIiwiZ2V0UmVxdWVzdCIsImJvZHkiLCJjb250cm9sbGVyIiwiZ2V0Q2xhc3MiLCJNb2RlbENvbnN0ciIsImNsYXNzIiwiSW50ZXJuYWxTZXJ2ZXJFcnJvckV4Y2VwdGlvbiIsIml0ZW0iLCJPcmRlcmVkUGFyYW1zIiwib3JkZXIiLCJyZXEiLCJvcmlnaW5hbCIsInBhcmFtcyIsIm9yZGVyTGlzdCIsIm9yZGVyZWQiLCJrIiwicmF3IiwidmFsdWVzSW5PcmRlciIsImtleXNJbk9yZGVyIiwiRGVjYWZQYXJhbXMiLCJEZWNhZlF1ZXJ5IiwiXyIsInF1ZXJ5IiwicGFyc2VkIiwibGltaXQiLCJuIiwiTnVtYmVyIiwiaXNOYU4iLCJvZmZzZXQiLCJib29rbWFyayIsIkRFQ0FGX01PRFVMRV9PUFRJT05TIiwiREVDQUZfQURBUFRFUl9JRCIsIkRFQ0FGX1RBU0tfU0VSVklDRV9JRCIsIkRFQ0FGX1JPVVRFIiwiREVDQUZfSEFORExFUlMiLCJTeW1ib2wiLCJERUNBRl9FWFBPU0UiLCJERUNBRl9DT05UUk9MTEVSX0NPTkZJRyIsIkRFQ0FGX0NPTlRFWFRfS0VZIiwiZXhwb3NlIiwiZmxhdm91cnMiLCJNZXRhZGF0YSIsInNldCIsImNvbnRyb2xsZXJDb25maWciLCJjb25maWciLCJjb250cm9sbGVyQ29uZmlnRGVjb3JhdG9yIiwiQVVUSF9IQU5ETEVSIiwiQVVUSF9NRVRBX0tFWSIsIklTX1BVQkxJQ19LRVkiLCJSRVFVSVJFRF9ST0xFU19LRVkiLCJTS0lQX01PREVMX1JPTEVTX0tFWSIsIkRlY2FmUmVxdWVzdENvbnRleHQiLCJSZXF1ZXN0Q29udGV4dCIsInN1cGVyIiwiaGVhZGVyc09mIiwiaGVhZGVycyIsInRoaXMiLCJ1dWlkIiwiVVVJRCIsImluc3RhbmNlIiwiZ2VuZXJhdGUiLCJwdXQiLCJyZWNvcmQiLCJvdmVycmlkZXMiLCJnZXQiLCJlIiwiYWNjdW11bGF0ZSIsIl9fZGVjb3JhdGUiLCJJbmplY3RhYmxlIiwic2NvcGUiLCJTY29wZSIsIlJFUVVFU1QiLCJfX3BhcmFtIiwiSW5qZWN0IiwiQXV0aEludGVyY2VwdG9yIiwicmVmbGVjdG9yIiwicmVxdWVzdENvbnRleHQiLCJhdXRoSGFuZGxlciIsImludGVyY2VwdCIsIm5leHQiLCJsb2ciLCJMb2dnaW5nIiwiZm9yIiwiaXNQdWJsaWMiLCJnZXRBbGxBbmRPdmVycmlkZSIsImdldEhhbmRsZXIiLCJtb2RlbE5hbWUiLCJyZXF1aXJlZFJvbGVzIiwic2tpcE1vZGVsUm9sZXMiLCJlZmZlY3RpdmVNb2RlbCIsInZlcmJvc2UiLCJkZWJ1ZyIsImF1dGhvcml6ZSIsImFwcGx5VHJhbnNmb3JtZXJzIiwidXNlciIsImdldE9yVW5kZWZpbmVkIiwib3JnYW5pemF0aW9uIiwiY3VycmVudExvZyIsImNoaWxkTG9nIiwibG9nZ2VyIiwiaGFuZGxlIiwiQWRhcHRlciIsImZsYXZvdXJzVG9UcmFuc2Zvcm0iLCJmbGF2b3VyIiwidHJhbnNmb3JtZXIiLCJ0cmFuc2Zvcm1lckZvciIsImZyb20iLCJPcHRpb25hbCIsIlJlZmxlY3RvciIsIkF1dGgiLCJtb2RlbCIsInJlc291cmNlIiwiZGVjcyIsIkFwaUJlYXJlckF1dGgiLCJVc2VJbnRlcmNlcHRvcnMiLCJwdXNoIiwiU2V0TWV0YWRhdGEiLCJQdWJsaWMiLCJSZXF1aXJlUm9sZXMiLCJyb2xlcyIsIkFwaVNlY3VyaXR5IiwiSHR0cFZlcmJUb0RlY29yYXRvciIsImRlY29yYXRvciIsIkVycm9yIiwiZXh0cmFjdFBhdGhQYXJhbXMiLCJyb3V0ZVBhdGgiLCJzcGxpdCIsInNsaWNlIiwiYXBpUGFyYW1TcGVjIiwiZ2V0QXBpRGVjb3JhdG9ycyIsIm1ldGhvZE5hbWUiLCJodHRwVmVyYiIsImluY2x1ZGVRdWVyeVBhcmFtcyIsIk5lc3RIdHRwUm91dGVEZWMiLCJhcGlQYXRoUGFyYW1zIiwic3dhZ2dlclF1ZXJ5UGFyYW1zIiwiQXBpUXVlcnkiLCJPcmRlckRpcmVjdGlvbiIsIm1ldGhvZCIsIkFwaVBhcmFtIiwiQXBpT3BlcmF0aW9uIiwic3VtbWFyeSIsIkFwaU9rUmVzcG9uc2UiLCJBcGlOb0NvbnRlbnRSZXNwb25zZSIsIlF1ZXJ5IiwiYXBwbHlBcGlEZWNvcmF0b3JzIiwicHJvdG8iLCJmb3JFYWNoIiwiZCIsImluZGV4IiwicmVzb2x2ZVBlcnNpc3RlbmNlTWV0aG9kIiwicGVyc2lzdGVuY2UiLCJhcmdzIiwiTW9kZWxTZXJ2aWNlIiwic3RhdGVtZW50IiwiY3JlYXRlUm91dGVIYW5kbGVyIiwiYXN5bmMiLCJwYXRoUGFyYW1zIiwicXVlcnlQYXJhbXMiLCJKU09OIiwic3RyaW5naWZ5IiwiZGlyZWN0aW9uIiwiZXJyb3IiLCJkZWZpbmVSb3V0ZU1ldGhvZCIsIkNvbnRyb2xsZXJDbGFzcyIsImhhbmRsZXIiLCJkZWZpbmVQcm9wZXJ0eSIsIndyaXRhYmxlIiwiRGVjYWZBdXRoSGFuZGxlciIsIkF1dGhIYW5kbGVyIiwicGFyc2VSZXF1ZXN0IiwidXNlclJvbGUiLCJhdXRob3JpemF0aW9uIiwiZXh0cmFjdEZyb21BdXRoIiwiQXV0aG9yaXphdGlvbkVycm9yIiwiRGVjYWZSb2xlQXV0aEhhbmRsZXIiLCJ0b0NvbnRleHRGbGFncyIsImFkYXB0ZXIiLCJhbGlhcyIsImJpbmQiLCJyZXF1ZXN0VHJhbnNmb3JtZXJzIiwibWV0YSIsIkNvbnRleHQiLCJ0b1Jlc3BvbnNlIiwicmVzIiwicGVuZGluZyIsImhlYWRlciIsIk1vZGVsQnVpbGRlciIsImRlY29yYXRlQ2xhc3MiLCJfY2xhc3NEZWNvcmF0b3JzIiwiX19oYXNDbGFzc0RlY29yYXRvclN1cHBvcnQiLCJvcmlnaW5hbEJ1aWxkIiwiYnVpbGQiLCJyZXN1bHQiLCJjYWxsIiwiZGVjb3JhdGVkIiwiRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yIiwiRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yXzEiLCJleGVjdXRvciIsImNvbnRleHR1YWxpemUiLCJmbGFncyIsImlwIiwiZXh0cmFjdElwIiwiRGVmYXVsdEFkYXB0ZXJGbGFncyIsInRpbWVzdGFtcCIsIkRhdGUiLCJvcGVyYXRpb24iLCJ1cmwiLCJjb250ZXh0IiwiZ2V0UmVzcG9uc2UiLCJleGVjIiwiRGVjYWZIYW5kbGVyRXhlY3V0b3IiLCJwYXJzZUlwSGVhZGVyIiwiY2FuZGlkYXRlIiwic2VnbWVudCIsInRyaW0iLCJCb29sZWFuIiwiRGVjYWZBdXRoTW9kdWxlIiwiRGVjYWZBdXRoTW9kdWxlXzEiLCJmb3JSb290IiwicHJvdmlkZXJzIiwicHJvdmlkZSIsIkFQUF9JTlRFUkNFUFRPUiIsInVzZUNsYXNzIiwiZ2xvYmFsIiwidXNlRXhpc3RpbmciLCJtb2R1bGUiLCJleHBvcnRzIiwiTW9kdWxlIiwiRGVjYWZIYW5kbGVyRXhlY3V0b3JfMSIsImhhbmRsZXJzIiwiRGVjYWZSZXNwb25zZUludGVyY2VwdG9yIiwicmVzcG9uc2UiLCJwaXBlIiwidGFwIiwiRGVjYWZDb250cm9sbGVyIiwiSHR0cERlY2FmQ29udHJvbGxlciIsImNsaWVudENvbnRleHQiLCJfbmFtZSIsIkRlY2FmTW9kZWxDb250cm9sbGVyIiwiX3BlcnNpc3RlbmNlIiwiU2VydmljZSIsImdldFNlcnZpY2UiLCJSZXBvc2l0b3J5IiwiZm9yTW9kZWwiLCJ0b092ZXJyaWRlcyIsIm92ZXJyaWRlIiwiZHRvQ2FjaGUiLCJNYXAiLCJEdG9Gb3IiLCJvcCIsInN0YWNrIiwiU2V0IiwiVHJhbnNhY3Rpb25PcGVyYXRpb25LZXlzIiwiY2FjaGUiLCJnZXREdG9DYWNoZSIsImNhY2hlZCIsImlzVXBkYXRlT3AiLCJEeW5hbWljRFRPIiwidG9QYXNjYWxDYXNlIiwicGtQcm9wIiwiTW9kZWwiLCJwayIsInBrUHJvcHNNZXRhZGF0YSIsInBrUHJvcHMiLCJwa0Rlc2lnblR5cGUiLCJwa1R5cGVJc051bWVyaWMiLCJCaWdJbnQiLCJwa0lzR2VuZXJhdGVkIiwiZ2VuZXJhdGVkIiwiZ2VuZXJhdGVkQnlTZXF1ZW5jZSIsImlzUHJvcGVydHlHZW5lcmF0ZWRBY3Jvc3NJbmhlcml0YW5jZSIsImFsbFByb3BzIiwicmVsYXRpb25zIiwic2NhbGFyUHJvcHMiLCJwcm9wIiwiaGFzIiwidmFsaWRhdGlvbiIsImdldFZhbGlkYXRpb25BY3Jvc3NJbmhlcml0YW5jZSIsImlzUmVxdWlyZWQiLCJWYWxpZGF0aW9uS2V5cyIsIlJFUVVJUkVEIiwidHlwZUhpbnQiLCJnZXRUeXBlQWNyb3NzSW5oZXJpdGFuY2UiLCJhcGlPcHRpb25zIiwiZGVzaWduVHlwZSIsImVudW1lcmFibGUiLCJjb25maWd1cmFibGUiLCJyZWxhdGlvbiIsInJlbGF0aW9uTWV0YSIsIkludGVybmFsRXJyb3IiLCJyZWxhdGlvblR5cGUiLCJ2YWxpZGF0aW9uRm9yIiwiTElTVCIsImlzQ2lyY3VsYXIiLCJwa1R5cGVOYW1lIiwiZ2V0UGtPcGVuQXBpVHlwZSIsImFkZFJlbGF0aW9uUGtSZWYiLCJyZWxhdGlvbkR0byIsImFkZCIsImFkZFJlbGF0aW9uVXBkYXRlIiwiYWRkUmVsYXRpb24iLCJEdG9DbGFzcyIsImV4dHJhTW9kZWxzIiwiZHRvUmVmIiwib25lT2ZJdGVtcyIsIiRyZWYiLCJvbmVPZiIsInBrVHlwZSIsImN1cnJlbnQiLCJGdW5jdGlvbiIsImdldFByb3RvdHlwZU9mIiwiV2Vha01hcCIsIkZyb21Nb2RlbENvbnRyb2xsZXIiLCJnZXRQZXJzaXN0ZW5jZSIsIk1vZGVsQ2xhenoiLCJlMiIsImNyZWF0ZVF1ZXJ5Um91dGVzRnJvbVJlcG9zaXRvcnkiLCJwcmVmaXgiLCJQZXJzaXN0ZW5jZUtleXMiLCJRVUVSWSIsInJlcG8iLCJxdWVyeU1ldGhvZHMiLCJyb3V0ZU1ldGhvZHMiLCJRdWVyeUNvbnRyb2xsZXIiLCJlbnRyaWVzIiwiam9pbiIsImNyZWF0ZUNvbXBsZXhRdWVyeUhhbmRsZXIiLCJkZWZpbmVNZXRob2QiLCJodHRwRGVjb3JhdG9yIiwiaHR0cE1ldGhvZCIsImdldFF1ZXJ5RGVjb3JhdG9ycyIsIm9ialZhbHVlcyIsImZpZWxkcyIsImYiLCJjcmVhdGUiLCJtb2R1bGVDb25maWdPdmVycmlkZXMiLCJnbG9iYWxEZWZhdWx0cyIsInRhYmxlTmFtZSIsInRvS2ViYWJDYXNlIiwibW9kZWxDbGF6ek5hbWUiLCJmYWN0b3J5UGVyc2lzdGVuY2UiLCJkZWNvcmF0b3JDb25maWciLCJtb2R1bGVPdmVycmlkZSIsIm1lcmdlZENvbmZpZyIsIkZhY3RvcnlDb250cm9sbGVyIiwiTW9kZWxDb250cm9sbGVyRmFjdG9yeSIsImZhY3RvcnlSb3V0ZXMiLCJfX3JvdXRlc19fIiwiZ2V0UEsiLCJhcGlQcm9wZXJ0aWVzIiwicGtQYXRoIiwiZ2V0Um91dGVQYXJhbWV0ZXJzRnJvbU1vZGVsIiwiYXV0aENvbmZpZyIsImF1dGgiLCJhcHBseUNsYXNzQXV0aCIsInB1YmxpYyIsIkR5bmFtaWNNb2RlbENvbnRyb2xsZXIiLCJEeW5hbWljTW9kZWxDb250cm9sbGVyXzEiLCJpbmZvIiwiQ29udHJvbGxlciIsIkFwaVRhZ3MiLCJBcGlFeHRyYU1vZGVscyIsInNvcnRlZFJvdXRlcyIsInNvcnQiLCJhIiwiYiIsImFTZWdtZW50cyIsImJTZWdtZW50cyIsImFQYXJhbUNvdW50IiwicyIsImJQYXJhbUNvdW50IiwiYUxpdGVyYWxDb3VudCIsImJMaXRlcmFsQ291bnQiLCJyb3V0ZSIsInJlZ2lzdHJhdGlvbiIsIm1hdGNoUm91dGUiLCJwYXJhbURlY29yYXRvcnMiLCJjb21wb3NlZCIsIkRCS2V5cyIsIkNPTVBPU0VEIiwiY29tcG9zZWRLZXlzIiwidW5pcXVlS2V5cyIsInNlcGFyYXRvciIsIm1ldGhvZERlY29yYXRvcnMiLCJub3JtYWxpemVkUGF0aCIsImNyZWF0ZVJlZ2lzdHJhdGlvbiIsImNyZWF0ZUNyZWF0ZUhhbmRsZXIiLCJjcmVhdGVDcmVhdGVEZWNvcmF0b3JzIiwiUmVzcG9uc2UiLCJwYXNzdGhyb3VnaCIsImNyZWF0ZUJ1bGtDcmVhdGVIYW5kbGVyIiwiYnVsa0NyZWF0ZURlY29yYXRvcnMiLCJjcmVhdGVCdWxrUmVhZEhhbmRsZXIiLCJidWxrUmVhZERlY29yYXRvcnMiLCJjcmVhdGVCdWxrVXBkYXRlSGFuZGxlciIsImJ1bGtVcGRhdGVEZWNvcmF0b3JzIiwiY3JlYXRlQnVsa0RlbGV0ZUhhbmRsZXIiLCJidWxrRGVsZXRlRGVjb3JhdG9ycyIsImNyZWF0ZVJlYWRIYW5kbGVyIiwicmVhZERlY29yYXRvcnMiLCJjcmVhdGVVcGRhdGVIYW5kbGVyIiwidXBkYXRlRGVjb3JhdG9ycyIsImNyZWF0ZURlbGV0ZUhhbmRsZXIiLCJkZWxldGVEZWNvcmF0b3JzIiwiZmFsbGJhY2tTZWdtZW50cyIsImlzQWxsUGFyYW1zIiwiZXZlcnkiLCJmYWxsYmFja0FwaVByb3BzIiwic3VmZml4IiwiY3JlYXRlU3RhdGVtZW50SGFuZGxlciIsInN0YXRlbWVudERlY29yYXRvcnMiLCJQYXJhbSIsInN0YXRlbWVudFJvdXRlcyIsInN0YXRlbWVudEtleSIsInN0YXRlbWVudE1ldGhvZE5hbWUiLCJjcmVhdGVTdGF0ZW1lbnRTaG9ydGN1dEhhbmRsZXIiLCJzdGF0ZW1lbnRTaG9ydGN1dERlY29yYXRvcnMiLCJzdGF0ZW1lbnRTaG9ydGN1dFBhcmFtcyIsInF1ZXJ5TWV0aG9kIiwiY29tcGxleFF1ZXJ5UGFyYW1zIiwicGF0aFNlZ21lbnRzIiwia25vd25QcmVmaXhlcyIsInJvdXRlTWV0YWRhdGEiLCJtYXRjaGVkRW50cnkiLCJmaW5kIiwiYWN0dWFsTWV0aG9kTmFtZSIsInBhcmFtU2VnbWVudHMiLCJjcmVhdGVDdXN0b21Sb3V0ZUhhbmRsZXIiLCJmaXJzdFNlZ21lbnQiLCJzZWdtZW50cyIsInNlZyIsImkiLCJyZXNwIiwibG9nQ3R4IiwiY3JlYXRlZCIsImNyZWF0ZUFsbCIsInJlYWRBbGwiLCJpZHMiLCJub3JtYWxpemVkSWRzIiwicmVhZCIsInVwZGF0ZUFsbCIsInVwZGF0ZWQiLCJkZWxldGVBbGwiLCJyb3V0ZVBhcmFtcyIsImlkIiwiVmFsaWRhdGlvbkVycm9yIiwicmVhZFJlc3VsdCIsInVwZGF0ZSIsInBheWxvYWQiLCJwYXJzZSIsInJlbW92ZSIsImRlbCIsImRlbGV0ZSIsImRldGFpbHMiLCJTVEFURU1FTlQiLCJwYXJzZUludCIsInBhdGhEaXJlY3Rpb24iLCJyZXNvbHZlZERpcmVjdGlvbiIsInN0YXRlbWVudFNob3J0Y3V0IiwibGlzdEJ5IiwicGFnZSIsInBhZ2luYXRlQnkiLCJBU0MiLCJyZWYiLCJmaW5kT25lQnkiLCJmaW5kQnkiLCJmaWVsZCIsImNvbXBsZXhRdWVyeSIsInNwcmVhZEFyZ3MiLCJleHRyYWN0UXVlcnlBcmdzIiwiY3VzdG9tUm91dGUiLCJsYXN0IiwicXVlcnlPYmoiLCJwb3AiLCJoYXNEaXJlY3Rpb24iLCJoYXNMaW1pdCIsImhhc09mZnNldCIsImV4dHJhcyIsIkFwaUJvZHkiLCJBcGlDcmVhdGVkUmVzcG9uc2UiLCJzY2hlbWEiLCJnZXRTY2hlbWFQYXRoIiwiQXBpQmFkUmVxdWVzdFJlc3BvbnNlIiwiQXBpVW5wcm9jZXNzYWJsZUVudGl0eVJlc3BvbnNlIiwiQXBpTm90Rm91bmRSZXNwb25zZSIsImJhc2UiLCJnZXRNb2R1bGVGb3IiLCJEZWNhZk1vZGVsTW9kdWxlIiwiRGVjYWZNb2RlbE1vZHVsZV8xIiwiY3JlYXRlTW9kZWxTZXJ2aWNlcyIsIm1vZGVscyIsInVzZUZhY3RvcnkiLCJpc0V4cG9zZWQiLCJleHBvc3VyZSIsInRyYWNrZWRNb2RlbHMiLCJjb250cm9sbGVyRXhwb3N1cmUiLCJtb2RlbFNlcnZpY2VzIiwiYXV0b1NlcnZpY2VzIiwiYWdncmVnYXRpb25zIiwiYWxsb3dHcm91cGluZ1F1ZXJpZXMiLCJjb250cm9sbGVycyIsIkRlY29yYXRpb24iLCJJbmplY3RhYmxlc0tleXMiLCJJTkpFQ1RBQkxFIiwiZXh0ZW5kIiwibmVzdEluamVjdGFibGUiLCJjYXRlZ29yeSIsImNmZyIsIkRlZmF1bHRJbmplY3RhYmxlc0NvbmZpZyIsInNpbmdsZXRvbiIsIkRFRkFVTFQiLCJkdXJhYmxlIiwiSU5KRUNUIiwibmVzdEluamVjdCIsImlubmVyTmVzdEluamVjdCIsIk1BWCIsIm1heERlYyIsIm1heCIsIm1heGltdW0iLCJNSU4iLCJtaW5EZWMiLCJtaW4iLCJtaW5pbXVtIiwiTUFYX0xFTkdUSCIsIm1heExlbmd0aERlYyIsIm1heExlbmd0aCIsIk1JTl9MRU5HVEgiLCJtaW5MZW5ndGhEZWMiLCJtaW5MZW5ndGgiLCJUWVBFIiwidHlwZURlYyIsIkRBVEUiLCJkYXRlRGVjIiwiZm9ybWF0IiwiRU5VTSIsIm9wdGlvbkRlYyIsIlBBVFRFUk4iLCJwYXR0ZXJuRGVjIiwicGF0IiwicGF0dGVybiIsInNvdXJjZSIsIkNPTFVNTiIsImNvbHVtbkRlYyIsIkRlY29yYXRpb25LZXlzIiwiREVTQ1JJUFRJT04iLCJkZXNjcmlwdGlvbkRlYyIsIkFVVEgiLCJDb3JzRXJyb3IiLCJGb3JiaWRkZW5FcnJvciIsIm1zZyIsIlRvTWFueVJlcXVlc3RzRXJyb3IiLCJCYXNlRXJyb3IiLCJEZWNhZkV4Y2VwdGlvbkZpbHRlciIsImV4Y2VwdGlvbiIsImhvc3QiLCJpc1Byb2R1Y3Rpb24iLCJMb2dnZWRFbnZpcm9ubWVudCIsImVudiIsInN0YXR1c0NvZGUiLCJOb3RGb3VuZEV4Y2VwdGlvbiIsIlVuc3VwcG9ydGVkRXJyb3IiLCJOb3RBY2NlcHRhYmxlRXhjZXB0aW9uIiwibWVzc2FnZSIsImdldFN0YXR1cyIsInN0YXR1cyIsImNvZGUiLCJqc29uIiwidG9JU09TdHJpbmciLCJDYXRjaCIsIlVzZURlY2FmRmlsdGVyIiwiVXNlRmlsdGVycyIsIlVzZURlY2FmSGVhZGVycyIsIlNXQUdHRVJfVUlfQ09OU1RBTlRTIiwidGl0bGUiLCJ2ZXJzaW9uIiwiZmF2aWNvbkZpbGVQYXRoIiwidG9wYmFySWNvbkZpbGVQYXRoIiwic2NoZW1lIiwiYmVhcmVyRm9ybWF0IiwiaW4iLCJwZXJzaXN0QXV0aG9yaXphdGlvbiIsInRvcGJhckJnQ29sb3IiLCJTd2FnZ2VyQ3VzdG9tVUkiLCJjdXN0b21DU1MiLCJjc3MiLCJ0b3BiYXJJY29uUGF0aCIsImltZyIsImI2NCIsImdldEN1c3RvbU9wdGlvbnMiLCJmYXZpY29uIiwiZmF2aWNvblBhdGgiLCJjdXN0b21TaXRlVGl0bGUiLCJjdXN0b21Dc3MiLCJzd2FnZ2VyT3B0aW9ucyIsImZpbGUiLCJmaWxlUGF0aCIsImFzc2V0c1BhdGgiLCJyZWFkRmlsZVN5bmMiLCJlbmNvZGluZyIsIlN3YWdnZXJCdWlsZGVyIiwiYXBwIiwiY3JlYXRlRG9jdW1lbnQiLCJvcGVuQXBpSnNvblBhdGgiLCJvcGVuQXBpWWFtbFBhdGgiLCJEb2N1bWVudEJ1aWxkZXIiLCJzZXRUaXRsZSIsInNldERlc2NyaXB0aW9uIiwic2V0VmVyc2lvbiIsImFkZEJlYXJlckF1dGgiLCJTd2FnZ2VyTW9kdWxlIiwicmVnaXN0ZXJPcGVuQXBpUm91dGUiLCJjb250ZW50VHlwZSIsImJvZHlGYWN0b3J5IiwiaHR0cEFkYXB0ZXIiLCJnZXRIdHRwQWRhcHRlciIsIl9yZXEiLCJyZXBseSIsInNldHVwU3dhZ2dlciIsImRvY3VtZW50Iiwic3dhZ2dlclVJIiwic2V0dXAiLCJqc29uRG9jdW1lbnRVcmwiLCJ5YW1sRG9jdW1lbnRVcmwiLCJZQU1MIiwiTmVzdEJvb3RzdHJhcGVyIiwiX2xvZ2dlciIsIkxvZ2dlciIsImluaXRpYWxpemUiLCJlbmFibGVMb2dnZXIiLCJjdXN0b21Mb2dnZXIiLCJ1c2VMb2dnZXIiLCJlbmFibGVDb3JzIiwib3JpZ2lucyIsImFsbG93TWV0aG9kcyIsImFsbG93ZWRPcmlnaW5zIiwibyIsInRvTG93ZXJDYXNlIiwiY29yc09wdGlvbnMiLCJvcmlnaW4iLCJjYWxsYmFjayIsImNyZWRlbnRpYWxzIiwibWV0aG9kcyIsInVzZUhlbG1ldCIsImhlbG1ldCIsInJlcXVpcmUiLCJ1c2UiLCJ3YXJuIiwic3dhZ2dlciIsInVzZUdsb2JhbFBpcGVzIiwicGlwZXMiLCJ1c2VHbG9iYWxGaWx0ZXJzIiwiZmlsdGVycyIsInVzZUdsb2JhbEludGVyY2VwdG9ycyIsImludGVyY2VwdG9ycyIsInN0YXJ0IiwicG9ydCIsInByb2Nlc3MiLCJQT1JUIiwibGlzdGVuIiwidGhlbiIsImdldFVybCIsInBlbmRpbmdQcm92aWRlcnMiLCJnZXRSZWdpc3RlcmVkRGVjYWZQcm92aWRlcnMiLCJyZWdpc3RlclByb3ZpZGVyIiwidG9rZW4iLCJpc01vZGVsQ29uc3RydWN0b3IiLCJtb2RlbFNlcnZpY2VUb2tlbiIsIm1vZGVsUmVwb3NpdG9yeVRva2VuIiwicGFyYW1ldGVySW5kZXgiLCJyZXNvbHZlZCIsInBhcmFtVHlwZXMiLCJhc01vZGVsIiwiaW5qZWN0YWJsZVNlcnZpY2VLZXkiLCJDb3JlU2VydmljZSIsIkNvcmVSZXBvc2l0b3J5IiwiRGVjYWZDb3JlTW9kdWxlIiwiRGVjYWZDb3JlTW9kdWxlXzEiLCJtb2R1bGVSZWYiLCJkZWNhZlByb3ZpZGVycyIsInVzZVZhbHVlIiwiY2xpZW50IiwiSCIsInByb3ZpZGVyIiwiYm9vdFBlcnNpc3RlbmNlIiwidHJpbW1lZCIsImNvbmYiLCJjb250ciIsInBvc3NpYmxlIiwiUGVyc2lzdGVuY2VTZXJ2aWNlIiwiYm9vdCIsImNsaWVudHMiLCJjIiwicG9zc2libGVUcmFuc2YiLCJyZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXIiLCJpbml0aWFsaXphdGlvbiIsIm9uQXBwbGljYXRpb25TaHV0ZG93biIsImFkYXB0ZXJzIiwic2h1dGRvd24iLCJHbG9iYWwiLCJNb2R1bGVSZWYiLCJMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlMiLCJub3JtYWxpemVFdmVudFJlc3BvbnNlIiwibW9kZWxDb25zdHIiLCJzZXJpYWxpemVkUGF5bG9hZCIsInNlcmlhbGl6ZSIsImNvbnNvbGUiLCJlcnIiLCJFdmVudHNDb250cm9sbGVyIiwiRXZlbnRzQ29udHJvbGxlcl8xIiwiZXZlbnRzJCIsIk9ic2VydmFibGUiLCJvYnNlcnZlciIsIm9ic2VydmVySWQiLCJNYXRoIiwicmFuZG9tIiwidG9VcHBlckNhc2UiLCJjYiIsInJlZnJlc2giLCJQcm9taXNlIiwicmVzb2x2ZSIsImFkYXB0ZXJOYW1lIiwib2JzZXJ2ZSIsInVuT2JzZXJ2ZSIsIkhFQVJUQkVBVF9JTlRFUlZBTF9NUyIsImhlYXJ0YmVhdCQiLCJpbnRlcnZhbCIsInRzIiwibWVyZ2UiLCJsaXN0ZW5Gb3JNb2RlbCIsIlNzZSIsIkRlY2FmU3RyZWFtTW9kdWxlIiwiRGVjYWZTdHJlYW1Nb2R1bGVfMSIsImZvckZsYXZvdXJzIiwiaW1wb3J0cyIsIlJvdXRlck1vZHVsZSIsInJlZ2lzdGVyIiwiRGVjYWZNb2R1bGUiLCJEZWNhZk1vZHVsZV8xIiwiZm9yUm9vdEFzeW5jIiwiYXV0b0NvbnRyb2xsZXJzIiwib2JzZXJ2ZXJPcHRpb25zIiwiZW5hYmxlT2JzZXJ2ZXJFdmVudHMiLCJvYnNlcnZlckZsYXZvdXJzIiwicnVuTWlncmF0aW9ucyIsInJlcG9Gb3JNb2RlbCIsIm0iLCJWRVJTSU9OIiwiQ09NTUlUIiwiRlVMTF9WRVJTSU9OIiwiUEFDS0FHRV9OQU1FIiwiYWxsb3dSZXJlZ2lzdHJhdGlvbiIsInJlZ2lzdGVyTGlicmFyeSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0lBQU8sTUFBTUEsb0JBQW9CO0lBQzFCLE1BQU1DLGFBQWE7UUFDeEJDLGVBQWUsR0FBR0Y7UUFDbEJHLGNBQWMsR0FBR0g7UUFDakJJLGNBQWMs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