@nlabs/reaktor 0.9.0 → 0.10.1

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 (400) hide show
  1. package/.env +1 -0
  2. package/.env.example +1 -0
  3. package/DATABASE_I18N_GUIDE.md +434 -0
  4. package/README.md +9 -0
  5. package/TEST_UTILITIES_GUIDE.md +360 -0
  6. package/coverage/index.html +61 -46
  7. package/dist/actions/apps.js +242 -0
  8. package/dist/actions/connections.js +90 -0
  9. package/dist/actions/conversations.js +350 -0
  10. package/dist/actions/dynamodb.js +150 -0
  11. package/dist/actions/email.js +152 -0
  12. package/dist/actions/files.js +283 -0
  13. package/dist/actions/groups.js +292 -0
  14. package/dist/actions/images.js +735 -0
  15. package/dist/actions/index.js +66 -0
  16. package/dist/actions/ios.js +164 -0
  17. package/dist/actions/locations.js +122 -0
  18. package/dist/actions/messages.js +208 -0
  19. package/dist/actions/notifications.js +59 -0
  20. package/dist/actions/payments.js +497 -0
  21. package/dist/actions/personas.js +110 -0
  22. package/dist/actions/posts.js +595 -0
  23. package/dist/actions/reactions.js +322 -0
  24. package/dist/actions/s3.js +133 -0
  25. package/dist/actions/search.js +90 -0
  26. package/dist/actions/sms.js +108 -0
  27. package/dist/actions/statistics.js +62 -0
  28. package/dist/actions/subscription.js +220 -0
  29. package/dist/actions/tags.js +292 -0
  30. package/dist/actions/users.js +784 -0
  31. package/dist/actions/websockets.js +174 -0
  32. package/dist/adapters/arangoAdapter.js +46 -0
  33. package/dist/adapters/fileAdapter.js +76 -0
  34. package/dist/adapters/imageAdapter.js +40 -0
  35. package/dist/adapters/messageAdapter.js +49 -0
  36. package/dist/adapters/postAdapter.js +70 -0
  37. package/dist/adapters/reaktorAdapter.js +44 -0
  38. package/dist/adapters/tagAdapter.js +50 -0
  39. package/dist/adapters/userAdapter.js +115 -0
  40. package/dist/config.js +125 -0
  41. package/dist/index.js +66 -0
  42. package/dist/lambdas/actions/websockets.js +132 -0
  43. package/dist/lambdas/authorizer.js +67 -0
  44. package/dist/lambdas/connection.js +91 -0
  45. package/dist/lambdas/utils/message.js +42 -0
  46. package/dist/lambdas/utils/websocket.js +105 -0
  47. package/dist/mocks/conversation.js +35 -0
  48. package/dist/mocks/file.js +38 -0
  49. package/dist/mocks/group.js +47 -0
  50. package/dist/mocks/image.js +44 -0
  51. package/dist/mocks/nlabs.png +0 -0
  52. package/dist/mocks/post.js +55 -0
  53. package/dist/mocks/tag.js +37 -0
  54. package/dist/mocks/user.js +88 -0
  55. package/dist/mutations/index.js +26 -0
  56. package/dist/mutations/locations.js +44 -0
  57. package/dist/mutations/messages.js +86 -0
  58. package/dist/mutations/personas.js +100 -0
  59. package/dist/mutations/posts.js +53 -0
  60. package/dist/mutations/reactions.js +51 -0
  61. package/dist/mutations/statistics.js +39 -0
  62. package/dist/mutations/subscriptions.js +56 -0
  63. package/dist/mutations/tags.js +120 -0
  64. package/dist/mutations/users.js +116 -0
  65. package/dist/objectTypes/app.js +173 -0
  66. package/dist/objectTypes/bankAccount.js +76 -0
  67. package/dist/objectTypes/connection.js +48 -0
  68. package/dist/objectTypes/conversation.js +77 -0
  69. package/dist/objectTypes/creditCard.js +86 -0
  70. package/dist/objectTypes/document.js +46 -0
  71. package/dist/objectTypes/error.js +46 -0
  72. package/dist/objectTypes/external.js +74 -0
  73. package/dist/objectTypes/file.js +100 -0
  74. package/dist/objectTypes/filter.js +43 -0
  75. package/dist/objectTypes/group.js +123 -0
  76. package/dist/objectTypes/iapSubscription.js +40 -0
  77. package/dist/objectTypes/image.js +129 -0
  78. package/dist/objectTypes/index.js +68 -0
  79. package/dist/objectTypes/location.js +109 -0
  80. package/dist/objectTypes/message.js +96 -0
  81. package/dist/objectTypes/passcode.js +42 -0
  82. package/dist/objectTypes/persona.js +87 -0
  83. package/dist/objectTypes/plan.js +95 -0
  84. package/dist/objectTypes/post.js +125 -0
  85. package/dist/objectTypes/reaction.js +61 -0
  86. package/dist/objectTypes/relation.js +49 -0
  87. package/dist/objectTypes/search.js +72 -0
  88. package/dist/objectTypes/statistics.js +39 -0
  89. package/dist/objectTypes/subscription.js +117 -0
  90. package/dist/objectTypes/tag.js +65 -0
  91. package/dist/objectTypes/user.js +144 -0
  92. package/dist/queries/index.js +33 -0
  93. package/dist/queries/locations.js +45 -0
  94. package/dist/queries/messages.js +52 -0
  95. package/dist/queries/posts.js +154 -0
  96. package/dist/queries/reactions.js +56 -0
  97. package/dist/queries/statistics.js +39 -0
  98. package/dist/queries/subscriptions.js +44 -0
  99. package/dist/queries/tags.js +75 -0
  100. package/dist/queries/users.js +64 -0
  101. package/dist/templates/email/layout.js +302 -0
  102. package/dist/templates/email/passwordForgot.js +38 -0
  103. package/dist/templates/email/passwordRecovery.js +35 -0
  104. package/dist/templates/email/verifyEmail.js +38 -0
  105. package/dist/templates/email/welcome.js +38 -0
  106. package/dist/templates/sms/passwordForgot.js +24 -0
  107. package/dist/templates/sms/passwordRecovery.js +24 -0
  108. package/dist/templates/sms/verifyEmail.js +24 -0
  109. package/dist/templates/sms/verifyPhone.js +24 -0
  110. package/dist/templates/sms/welcome.js +24 -0
  111. package/dist/types/apps.js +32 -0
  112. package/{lib → dist}/types/arangodb.js +1 -1
  113. package/{lib → dist}/types/auth.js +1 -1
  114. package/{lib → dist}/types/connections.js +1 -1
  115. package/dist/types/conversations.js +16 -0
  116. package/{lib → dist}/types/email.js +1 -1
  117. package/dist/types/files.js +16 -0
  118. package/dist/types/google.js +16 -0
  119. package/{lib → dist}/types/groups.js +1 -1
  120. package/dist/types/images.js +16 -0
  121. package/dist/types/index.js +60 -0
  122. package/{lib → dist}/types/locations.js +1 -1
  123. package/{lib → dist}/types/messages.js +1 -1
  124. package/{lib → dist}/types/notifications.js +1 -1
  125. package/dist/types/payments.js +16 -0
  126. package/dist/types/personas.js +16 -0
  127. package/dist/types/posts.js +16 -0
  128. package/{lib → dist}/types/tags.js +1 -1
  129. package/dist/types/users.js +16 -0
  130. package/dist/types/websockets.js +16 -0
  131. package/dist/utils/adapterUtils.js +45 -0
  132. package/dist/utils/analyticsUtils.js +72 -0
  133. package/dist/utils/arangodbUtils.js +165 -0
  134. package/dist/utils/auth.js +57 -0
  135. package/dist/utils/index.js +30 -0
  136. package/{lib → dist}/utils/session.js +10 -7
  137. package/index.js +1 -1
  138. package/jest.config.js +17 -0
  139. package/jest.setup.js +36 -0
  140. package/lex.config.cjs +13 -0
  141. package/lib/actions/apps.js +17 -249
  142. package/lib/actions/connections.js +7 -0
  143. package/lib/actions/content.js +17 -0
  144. package/lib/actions/conversations.js +19 -325
  145. package/lib/actions/dynamodb.js +2 -150
  146. package/lib/actions/email.js +2 -152
  147. package/lib/actions/files.js +5 -287
  148. package/lib/actions/groups.js +23 -263
  149. package/lib/actions/images.js +31 -646
  150. package/lib/actions/index.js +2 -62
  151. package/lib/actions/ios.js +9 -162
  152. package/lib/actions/locations.js +7 -110
  153. package/lib/actions/messages.js +21 -193
  154. package/lib/actions/notifications.js +2 -59
  155. package/lib/actions/payments.js +11 -461
  156. package/lib/actions/posts.js +77 -515
  157. package/lib/actions/profiles.js +8 -0
  158. package/lib/actions/reactions.js +25 -312
  159. package/lib/actions/s3.js +2 -133
  160. package/lib/actions/search.js +5 -90
  161. package/lib/actions/sms.js +2 -106
  162. package/lib/actions/statistics.js +6 -60
  163. package/lib/actions/subscriptions.js +12 -0
  164. package/lib/actions/tags.js +19 -262
  165. package/lib/actions/users.js +67 -537
  166. package/lib/actions/websockets.js +16 -158
  167. package/lib/adapters/arangoAdapter.js +2 -46
  168. package/lib/adapters/contentAdapter.js +2 -0
  169. package/lib/adapters/fileAdapter.js +2 -76
  170. package/lib/adapters/imageAdapter.js +2 -0
  171. package/lib/adapters/messageAdapter.js +2 -0
  172. package/lib/adapters/postAdapter.js +2 -70
  173. package/lib/adapters/reaktorAdapter.js +2 -44
  174. package/lib/adapters/tagAdapter.js +2 -50
  175. package/lib/adapters/userAdapter.js +2 -110
  176. package/lib/config.js +2 -125
  177. package/lib/handlers/graphqlHandler.js +2 -0
  178. package/lib/index.js +2 -30
  179. package/lib/lambdas/actions/websockets.js +14 -112
  180. package/lib/lambdas/authorizer.js +2 -67
  181. package/lib/lambdas/connection.js +2 -92
  182. package/lib/lambdas/utils/message.js +2 -42
  183. package/lib/lambdas/utils/websocket.js +2 -105
  184. package/lib/mocks/conversation.js +2 -35
  185. package/lib/mocks/file.js +2 -38
  186. package/lib/mocks/group.js +2 -47
  187. package/lib/mocks/image.js +2 -43
  188. package/lib/mocks/post.js +2 -55
  189. package/lib/mocks/tag.js +2 -37
  190. package/lib/mocks/user.js +2 -88
  191. package/lib/mutations/index.js +2 -0
  192. package/lib/mutations/locations.integration.js +2 -0
  193. package/lib/mutations/locations.js +2 -0
  194. package/lib/mutations/messages.integration.js +2 -0
  195. package/lib/mutations/messages.js +2 -0
  196. package/lib/mutations/posts.integration.js +2 -0
  197. package/lib/mutations/posts.js +2 -0
  198. package/lib/mutations/profiles.integration.js +2 -0
  199. package/lib/mutations/profiles.js +2 -0
  200. package/lib/mutations/reactions.integration.js +2 -0
  201. package/lib/mutations/reactions.js +2 -0
  202. package/lib/mutations/statistics.integration.js +2 -0
  203. package/lib/mutations/statistics.js +2 -0
  204. package/lib/mutations/subscriptions.integration.js +2 -0
  205. package/lib/mutations/subscriptions.js +2 -0
  206. package/lib/mutations/tags.integration.js +2 -0
  207. package/lib/mutations/tags.js +2 -0
  208. package/lib/mutations/users.integration.js +2 -0
  209. package/lib/mutations/users.js +2 -0
  210. package/lib/objectTypes/app.js +2 -0
  211. package/lib/objectTypes/bankAccount.js +2 -0
  212. package/lib/objectTypes/connection.js +2 -0
  213. package/lib/objectTypes/conversation.js +2 -0
  214. package/lib/objectTypes/creditCard.js +2 -0
  215. package/lib/objectTypes/document.js +2 -0
  216. package/lib/objectTypes/error.js +2 -0
  217. package/lib/objectTypes/external.js +2 -0
  218. package/lib/objectTypes/file.js +2 -0
  219. package/lib/objectTypes/filter.js +2 -0
  220. package/lib/objectTypes/group.js +2 -0
  221. package/lib/objectTypes/iapSubscription.js +2 -0
  222. package/lib/objectTypes/image.js +2 -0
  223. package/lib/objectTypes/index.js +2 -0
  224. package/lib/objectTypes/location.js +2 -0
  225. package/lib/objectTypes/message.js +2 -0
  226. package/lib/objectTypes/passcode.js +2 -0
  227. package/lib/objectTypes/plan.js +2 -0
  228. package/lib/objectTypes/post.js +2 -0
  229. package/lib/objectTypes/profile.js +2 -0
  230. package/lib/objectTypes/reaction.js +2 -0
  231. package/lib/objectTypes/relation.js +2 -0
  232. package/lib/objectTypes/search.js +2 -0
  233. package/lib/objectTypes/statistics.js +2 -0
  234. package/lib/objectTypes/subscription.js +2 -0
  235. package/lib/objectTypes/tag.js +2 -0
  236. package/lib/objectTypes/user.js +2 -0
  237. package/lib/queries/index.js +2 -0
  238. package/lib/queries/locations.integration.js +2 -0
  239. package/lib/queries/locations.js +2 -0
  240. package/lib/queries/messages.integration.js +2 -0
  241. package/lib/queries/messages.js +2 -0
  242. package/lib/queries/posts.integration.js +2 -0
  243. package/lib/queries/posts.js +2 -0
  244. package/lib/queries/reactions.integration.js +2 -0
  245. package/lib/queries/reactions.js +2 -0
  246. package/lib/queries/statistics.js +2 -0
  247. package/lib/queries/subscriptions.js +2 -0
  248. package/lib/queries/tags.integration.js +2 -0
  249. package/lib/queries/tags.js +2 -0
  250. package/lib/queries/users.integration.js +2 -0
  251. package/lib/queries/users.js +2 -0
  252. package/lib/templates/email/layout.js +3 -25
  253. package/lib/templates/email/passwordForgot.js +3 -25
  254. package/lib/templates/email/passwordRecovery.js +3 -25
  255. package/lib/templates/email/verifyEmail.js +3 -25
  256. package/lib/templates/email/welcome.js +3 -25
  257. package/lib/templates/sms/passwordForgot.js +2 -24
  258. package/lib/templates/sms/passwordRecovery.js +2 -24
  259. package/lib/templates/sms/verifyEmail.js +2 -24
  260. package/lib/templates/sms/verifyPhone.js +2 -24
  261. package/lib/templates/sms/welcome.js +2 -24
  262. package/lib/types/apps.types.js +2 -0
  263. package/lib/types/arangodb.types.js +1 -0
  264. package/lib/types/auth.types.js +1 -0
  265. package/lib/types/connections.types.js +1 -0
  266. package/lib/types/content.types.js +1 -0
  267. package/lib/types/conversations.types.js +1 -0
  268. package/lib/types/email.types.js +1 -0
  269. package/lib/types/error.types.js +2 -0
  270. package/lib/types/files.types.js +1 -0
  271. package/lib/types/google.types.js +1 -0
  272. package/lib/types/groups.types.js +1 -0
  273. package/lib/types/images.types.js +1 -0
  274. package/lib/types/index.js +2 -56
  275. package/lib/types/locations.types.js +1 -0
  276. package/lib/types/messages.types.js +1 -0
  277. package/lib/types/notifications.types.js +1 -0
  278. package/lib/types/payments.types.js +1 -0
  279. package/lib/types/posts.types.js +1 -0
  280. package/lib/types/profiles.types.js +1 -0
  281. package/lib/types/statistics.types.js +1 -0
  282. package/lib/types/tags.types.js +1 -0
  283. package/lib/types/users.types.js +1 -0
  284. package/lib/types/websockets.types.js +1 -0
  285. package/lib/utils/adapterUtils.js +2 -45
  286. package/lib/utils/analyticsUtils.js +2 -72
  287. package/lib/utils/arangodbUtils.js +5 -142
  288. package/lib/utils/authUtils.js +2 -0
  289. package/lib/utils/contextUtils.js +2 -0
  290. package/lib/utils/dbI18n.example.js +6 -0
  291. package/lib/utils/dbI18n.js +2 -0
  292. package/lib/utils/googleTranslate.js +2 -0
  293. package/lib/utils/graphqlUtils.js +2 -0
  294. package/lib/utils/index.js +2 -30
  295. package/lib/utils/languageDetection.js +2 -0
  296. package/lib/utils/localeUtils.example.js +2 -0
  297. package/lib/utils/localeUtils.js +2 -0
  298. package/lib/utils/middlewareUtils.js +2 -0
  299. package/lib/utils/sessionUtils.js +2 -0
  300. package/lib/utils/stripeUtils.js +2 -0
  301. package/lib/utils/templateUtils.js +2 -0
  302. package/lib/utils/testUtils.js +2 -0
  303. package/lib/utils/translationQueue.example.js +2 -0
  304. package/lib/utils/translationQueue.js +2 -0
  305. package/package.json +41 -37
  306. package/.eslintrc +0 -10
  307. package/.prettierrc.js +0 -4
  308. package/lib/actions/apps.d.ts +0 -25
  309. package/lib/actions/conversations.d.ts +0 -14
  310. package/lib/actions/dynamodb.d.ts +0 -8
  311. package/lib/actions/email.d.ts +0 -5
  312. package/lib/actions/files.d.ts +0 -19
  313. package/lib/actions/groups.d.ts +0 -14
  314. package/lib/actions/images.d.ts +0 -26
  315. package/lib/actions/index.d.ts +0 -21
  316. package/lib/actions/ios.d.ts +0 -7
  317. package/lib/actions/locations.d.ts +0 -5
  318. package/lib/actions/messages.d.ts +0 -13
  319. package/lib/actions/notifications.d.ts +0 -5
  320. package/lib/actions/payments.d.ts +0 -10
  321. package/lib/actions/posts.d.ts +0 -19
  322. package/lib/actions/reactions.d.ts +0 -30
  323. package/lib/actions/s3.d.ts +0 -7
  324. package/lib/actions/search.d.ts +0 -3
  325. package/lib/actions/sms.d.ts +0 -3
  326. package/lib/actions/statistics.d.ts +0 -3
  327. package/lib/actions/subscription.d.ts +0 -7
  328. package/lib/actions/subscription.js +0 -218
  329. package/lib/actions/tags.d.ts +0 -29
  330. package/lib/actions/users.d.ts +0 -47
  331. package/lib/actions/websockets.d.ts +0 -19
  332. package/lib/adapters/arangoAdapter.d.ts +0 -2
  333. package/lib/adapters/fileAdapter.d.ts +0 -3
  334. package/lib/adapters/postAdapter.d.ts +0 -2
  335. package/lib/adapters/reaktorAdapter.d.ts +0 -6
  336. package/lib/adapters/tagAdapter.d.ts +0 -2
  337. package/lib/adapters/userAdapter.d.ts +0 -2
  338. package/lib/config.d.ts +0 -20
  339. package/lib/index.d.ts +0 -5
  340. package/lib/lambdas/actions/websockets.d.ts +0 -6
  341. package/lib/lambdas/authorizer.d.ts +0 -20
  342. package/lib/lambdas/connection.d.ts +0 -12
  343. package/lib/lambdas/utils/message.d.ts +0 -1
  344. package/lib/lambdas/utils/websocket.d.ts +0 -7
  345. package/lib/mocks/conversation.d.ts +0 -8
  346. package/lib/mocks/file.d.ts +0 -11
  347. package/lib/mocks/group.d.ts +0 -17
  348. package/lib/mocks/image.d.ts +0 -3
  349. package/lib/mocks/post.d.ts +0 -38
  350. package/lib/mocks/tag.d.ts +0 -2
  351. package/lib/mocks/user.d.ts +0 -4
  352. package/lib/templates/email/layout.d.ts +0 -2
  353. package/lib/templates/email/passwordForgot.d.ts +0 -2
  354. package/lib/templates/email/passwordRecovery.d.ts +0 -2
  355. package/lib/templates/email/verifyEmail.d.ts +0 -2
  356. package/lib/templates/email/welcome.d.ts +0 -2
  357. package/lib/templates/sms/passwordForgot.d.ts +0 -2
  358. package/lib/templates/sms/passwordRecovery.d.ts +0 -2
  359. package/lib/templates/sms/verifyEmail.d.ts +0 -2
  360. package/lib/templates/sms/verifyPhone.d.ts +0 -2
  361. package/lib/templates/sms/welcome.d.ts +0 -2
  362. package/lib/types/apps.d.ts +0 -46
  363. package/lib/types/apps.js +0 -32
  364. package/lib/types/arangodb.d.ts +0 -30
  365. package/lib/types/auth.d.ts +0 -7
  366. package/lib/types/connections.d.ts +0 -7
  367. package/lib/types/conversations.d.ts +0 -29
  368. package/lib/types/conversations.js +0 -16
  369. package/lib/types/email.d.ts +0 -13
  370. package/lib/types/error.d.ts +0 -20
  371. package/lib/types/files.d.ts +0 -26
  372. package/lib/types/files.js +0 -16
  373. package/lib/types/google.d.ts +0 -29
  374. package/lib/types/google.js +0 -16
  375. package/lib/types/groups.d.ts +0 -21
  376. package/lib/types/images.d.ts +0 -51
  377. package/lib/types/images.js +0 -16
  378. package/lib/types/index.d.ts +0 -18
  379. package/lib/types/locations.d.ts +0 -20
  380. package/lib/types/messages.d.ts +0 -16
  381. package/lib/types/notifications.d.ts +0 -17
  382. package/lib/types/payments.d.ts +0 -112
  383. package/lib/types/payments.js +0 -16
  384. package/lib/types/posts.d.ts +0 -31
  385. package/lib/types/posts.js +0 -16
  386. package/lib/types/statistics.d.ts +0 -3
  387. package/lib/types/tags.d.ts +0 -10
  388. package/lib/types/users.d.ts +0 -76
  389. package/lib/types/users.js +0 -16
  390. package/lib/types/websocket.d.ts +0 -13
  391. package/lib/types/websocket.js +0 -16
  392. package/lib/utils/adapterUtils.d.ts +0 -1
  393. package/lib/utils/analyticsUtils.d.ts +0 -21
  394. package/lib/utils/arangodbUtils.d.ts +0 -65
  395. package/lib/utils/auth.d.ts +0 -20
  396. package/lib/utils/auth.js +0 -50
  397. package/lib/utils/index.d.ts +0 -5
  398. package/lib/utils/session.d.ts +0 -16
  399. /package/{lib → dist}/types/error.js +0 -0
  400. /package/{lib → dist}/types/statistics.js +0 -0
@@ -1,194 +1,37 @@
1
- var __create = Object.create;
2
- var __defProp = Object.defineProperty;
3
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
- var __getOwnPropNames = Object.getOwnPropertyNames;
5
- var __getProtoOf = Object.getPrototypeOf;
6
- var __hasOwnProp = Object.prototype.hasOwnProperty;
7
- var __export = (target, all) => {
8
- for (var name in all)
9
- __defProp(target, name, { get: all[name], enumerable: true });
10
- };
11
- var __copyProps = (to, from, except, desc) => {
12
- if (from && typeof from === "object" || typeof from === "function") {
13
- for (let key of __getOwnPropNames(from))
14
- if (!__hasOwnProp.call(to, key) && key !== except)
15
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
16
- }
17
- return to;
18
- };
19
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
- // If the importer is in node compatibility mode or this is not an ESM
21
- // file that has been converted to a CommonJS file using a Babel-
22
- // compatible transform (i.e. "__esModule" has not been set), then set
23
- // "default" to the CommonJS "module.exports" for node compatibility.
24
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
25
- mod
26
- ));
27
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
- var images_exports = {};
29
- __export(images_exports, {
30
- addImage: () => addImage,
31
- addImageEdge: () => addImageEdge,
32
- deleteImage: () => deleteImage,
33
- getAppImageUrl: () => getAppImageUrl,
34
- getImage: () => getImage,
35
- getImageCountByItem: () => getImageCountByItem,
36
- getImageOptional: () => getImageOptional,
37
- getImageUrl: () => getImageUrl,
38
- getImagesByGroup: () => getImagesByGroup,
39
- getImagesByItem: () => getImagesByItem,
40
- getImagesByReactions: () => getImagesByReactions,
41
- getImagesByUser: () => getImagesByUser,
42
- getPathUserImages: () => getPathUserImages,
43
- parseImageOptions: () => parseImageOptions,
44
- resizeSaveImage: () => resizeSaveImage,
45
- updateImage: () => updateImage
46
- });
47
- module.exports = __toCommonJS(images_exports);
48
- var import_rip_hunter = require("@nlabs/rip-hunter");
49
- var import_utils = require("@nlabs/utils");
50
- var import_arangojs = require("arangojs");
51
- var import_gm = __toESM(require("gm"));
52
- var import_luxon = require("luxon");
53
- var import_groups = require("./groups");
54
- var import_s3 = require("./s3");
55
- var import_config = require("../config");
56
- var import_analyticsUtils = require("../utils/analyticsUtils");
57
- var import_error = require("../types/error");
58
- var import_arangodbUtils = require("../utils/arangodbUtils");
59
- const eventCategory = "images";
60
- import_gm.default.subClass({ imageMagick: "7+" });
61
- const parseImageOptions = (options = {}) => {
62
- const {
63
- from = 0,
64
- to = 30,
65
- type = "default"
66
- } = options;
67
- return {
68
- limit: (0, import_arangodbUtils.getLimit)(from, to),
69
- type: (0, import_utils.parseChar)(type, 32)
70
- };
71
- };
72
- const getImageOptional = (fields = []) => fields.reduce((selects, field) => {
73
- if (field.includes("Count")) {
74
- return (0, import_arangodbUtils.selectReactionCountByType)("images", "i", field, selects);
75
- }
76
- switch (field) {
77
- case "reactions": {
78
- selects.queries.push(`LET reactions = (
1
+ import{get as J}from"@nlabs/rip-hunter";import{createHash as z,lowerCaseKeys as Z,parseArangoId as K,parseChar as j,parseId as q,parseNum as X}from"@nlabs/utils";import{aql as L}from"arangojs";import w from"gm";import{DateTime as Y}from"luxon";import{getGroupDetails as ee,isGrouped as te}from"./groups";import{s3DeleteList as re,s3GetSignedUrl as ae,s3Put as G}from"./s3";import{Config as O}from"../config";import{ErrorTypes as E}from"../types/error.types";import{logError as R,logException as se}from"../utils/analyticsUtils";import{getDocId as ne,getLimit as _,selectReactionCountByType as oe,useDb as N}from"../utils/arangodbUtils";const f="images";w.subClass({imageMagick:"7+"});const H=(e={})=>{const{from:t=0,to:s=30,type:r="default"}=e;return{limit:_(t,s),type:j(r,32)}},W=(e=[])=>e.reduce((t,s)=>{if(s.includes("Count"))return oe("images","i",s,t);switch(s){case"reactions":return t.queries.push(`LET reactions = (
79
2
  FOR image, r IN INBOUND i._id reactions
80
3
  COLLECT reactionName = r.value INTO reactionItems
81
4
  RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
82
- )`);
83
- selects.objects.push("reactions:reactions");
84
- return selects;
85
- }
86
- case "tags": {
87
- selects.queries.push(`LET tags = (
5
+ )`),t.objects.push("reactions:reactions"),t;case"tags":return t.queries.push(`LET tags = (
88
6
  FOR t, pl IN INBOUND i._id isTagged
89
7
  RETURN t
90
- )`);
91
- selects.objects.push("tags:tags");
92
- return selects;
93
- }
94
- case "users": {
95
- selects.queries.push(`LET users = FIRST(
8
+ )`),t.objects.push("tags:tags"),t;case"users":return t.queries.push(`LET users = FIRST(
96
9
  FOR u IN users
97
10
  FILTER i.userId == u._key
98
11
  LIMIT 1
99
12
  RETURN u
100
- )`);
101
- selects.objects.push("users:users");
102
- return selects;
103
- }
104
- default: {
105
- return selects;
106
- }
107
- }
108
- }, { objects: [], queries: [] });
109
- const getImagesByUser = (context, userId, from = 0, to = 30) => {
110
- const action = "getImagesByUser";
111
- const { database } = context;
112
- const formatUserId = (0, import_utils.parseId)(userId);
113
- const limit = (0, import_arangodbUtils.getLimit)(from, to);
114
- const aqlQry = `FOR i IN images
115
- FILTER i.userId == "${formatUserId}"
13
+ )`),t.objects.push("users:users"),t;default:return t}},{objects:[],queries:[]}),Te=(e,t,s=0,r=30)=>{const o="getImagesByUser",{databaseName:n}=e,a=q(t),u=_(s,r),g=`FOR i IN images
14
+ FILTER i.userId == "${a}"
116
15
  LET user = FIRST(
117
16
  FOR u IN users
118
17
  FILTER u._key == i.userId
119
18
  LIMIT 1
120
19
  RETURN u
121
20
  )
122
- ${limit.aql}
21
+ ${u.aql}
123
22
  SORT i.added
124
- RETURN MERGE(i, {user:user})`;
125
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
126
- action,
127
- category: eventCategory,
128
- value: import_error.ErrorTypes.DATABASE_ERROR,
129
- params: { from, to, userId }
130
- }, error, context));
131
- };
132
- const getImageCountByItem = (context, itemId) => {
133
- const action = "getImageCountByItem";
134
- const { database } = context;
135
- const formatItemId = (0, import_utils.parseArangoId)(itemId);
136
- const aqlQry = import_arangojs.aql`FOR i IN hasImage
137
- FILTER i._to == ${formatItemId}
138
- RETURN {count: COUNT(i)}`;
139
- return database.query(aqlQry).then((cursor) => cursor.next()).then(({ count }) => count).catch((error) => (0, import_analyticsUtils.logError)({
140
- action,
141
- category: eventCategory,
142
- value: import_error.ErrorTypes.DATABASE_ERROR,
143
- params: { itemId }
144
- }, error, context));
145
- };
146
- const getImagesByItem = async (context, itemId, options = {}) => {
147
- const action = "getImagesByItem";
148
- const { database, fields = [] } = context;
149
- const formatItemId = (0, import_utils.parseArangoId)(itemId);
150
- const { limit } = parseImageOptions(options);
151
- const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
152
- const aqlQry = `FOR i, l IN 1..1 OUTBOUND "${formatItemId}" hasImage
23
+ RETURN MERGE(i, {user:user})`;return N(n).query(g).then(i=>i.all()).catch(i=>(R({action:o,category:f,params:{from:s,to:r,userId:t},value:E.DATABASE_ERROR},i,e),[]))},he=(e,t)=>{const s="getImageCountByItem",{databaseName:r}=e,o=K(t),n=L`FOR i IN hasImage
24
+ FILTER i._to == ${o}
25
+ RETURN {count: COUNT(i)}`;return N(r).query(n).then(a=>a.next()).then(({count:a})=>a).catch(a=>R({action:s,category:f,params:{itemId:t},value:E.DATABASE_ERROR},a,e))},be=async(e,t,s={})=>{const r="getImagesByItem",{databaseName:o,fields:n=[]}=e,a=K(t),{limit:u}=H(s),{objects:g,queries:i}=W(n),p=`FOR i, l IN 1..1 OUTBOUND "${a}" hasImage
153
26
  FILTER NOT IS_NULL(i)
154
- ${selectQueries.join("\n")}
27
+ ${i.join(`
28
+ `)}
155
29
  SORT i.added
156
- ${limit.aql}
157
- RETURN MERGE(i, {${selectObjects.join(", ")}})`;
158
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
159
- action,
160
- category: eventCategory,
161
- value: import_error.ErrorTypes.DATABASE_ERROR,
162
- params: { itemId, options }
163
- }, error, context));
164
- };
165
- const getImagesByGroup = (context, params) => {
166
- const action = "getImagesByGroup";
167
- const { database, session: { userId: sessionId } } = context;
168
- const { filters, groupId, from, to } = params;
169
- const formatGroupId = (0, import_utils.parseId)(groupId);
170
- const limit = (0, import_arangodbUtils.getLimit)(from, to);
171
- filters.map((filter) => {
172
- const { conditional, name, value } = filter;
173
- let formatCond = conditional;
174
- if (conditional !== ">=" && conditional !== "<=" && conditional !== ">" && conditional !== "<") {
175
- formatCond = "==";
176
- }
177
- switch (name) {
178
- case "added":
179
- return `p.added ${formatCond} ${(0, import_utils.parseNum)(value)}`;
180
- default:
181
- return "";
182
- }
183
- });
184
- return (0, import_groups.getGroupDetails)(context, formatGroupId).then((group) => {
185
- if (group.privacy === "public") {
186
- filters.push(`p.groupId == "${formatGroupId}"`);
187
- const filterStr = filters.join(" && ");
188
- const aqlQry = `FOR i IN
30
+ ${u.aql}
31
+ RETURN MERGE(i, {${g.join(", ")}})`;return N(o).query(p).then(y=>y.all()).catch(y=>(R({action:r,category:f,params:{itemId:t,options:s},value:E.DATABASE_ERROR},y,e),[]))},Ae=(e,t)=>{const s="getImagesByGroup",{databaseName:r,session:{userId:o}}=e,{filters:n,groupId:a,from:u,to:g}=t,i=q(a),p=_(u,g);return n.map(y=>{const{conditional:I,name:m,value:c}=y;let l=I;switch(I!==">="&&I!=="<="&&I!==">"&&I!=="<"&&(l="=="),m){case"added":return`p.added ${l} ${X(c)}`;default:return""}}),ee(e,i).then(y=>{if(y.privacy==="public"){n.push({conditional:"==",name:"groupId",value:i});const m=`FOR i IN
189
32
  FLATTEN(
190
33
  FOR p IN posts
191
- FILTER ${filterStr}
34
+ FILTER ${n.join(" && ")}
192
35
  LET images = (
193
36
  FOR i, e IN INBOUND p._id hasImage
194
37
  RETURN i
@@ -197,485 +40,27 @@ const getImagesByGroup = (context, params) => {
197
40
  RETURN images
198
41
  )
199
42
  SORT i.added DESC
200
- ${limit.aql}
201
- RETURN i`;
202
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
203
- action,
204
- category: eventCategory,
205
- value: import_error.ErrorTypes.DATABASE_ERROR,
206
- params
207
- }, error, context));
208
- }
209
- return (0, import_groups.isGrouped)(database, sessionId, groupId).then((grouped) => {
210
- if (grouped.isValid) {
211
- filters.push(`p.groupId == "${grouped.groupId}"`);
212
- const filterList = filters.join(" && ");
213
- const aqlQry = `FOR p IN post
214
- FILTER ${filterList}
43
+ ${p.aql}
44
+ RETURN i`;return N(r).query(m).then(c=>c.all()).catch(c=>(R({action:s,category:f,params:{filters:n,from:u,groupId:a,to:g},value:E.DATABASE_ERROR},c,e),[]))}return te(e,o,a).then(I=>{if(I.isValid){n.push({conditional:"==",name:"groupId",value:I.groupId});const c=`FOR p IN post
45
+ FILTER ${n.join(" && ")}
215
46
  FOR f IN p.files
216
47
  FILTER f.type == "image/jpeg" || f.type == "image/png"
217
- ${limit.aql}
48
+ ${p.aql}
218
49
  SORT p.added DESC
219
- RETURN f`;
220
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
221
- action,
222
- category: eventCategory,
223
- value: import_error.ErrorTypes.DATABASE_ERROR,
224
- params
225
- }, error, context));
226
- }
227
- return [];
228
- });
229
- });
230
- };
231
- const getImagesByReactions = (context, reactions = [], options) => {
232
- const action = "getUsersByImage";
233
- const { database, fields = [], session: { userId: sessionId } } = context;
234
- const { limit } = parseImageOptions(options);
235
- const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
236
- const formatSessionId = `users/${sessionId}`;
237
- const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32));
238
- const filterBy = [
239
- 'r.type == "images"',
240
- `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
241
- ];
242
- const aqlQry = `FOR i, r IN OUTBOUND "${formatSessionId}" hasReaction
50
+ RETURN f`;return N(r).query(c).then(l=>l.all()).catch(l=>(R({action:s,category:f,params:{filters:n,from:u,groupId:a,to:g},value:E.DATABASE_ERROR},l,e),[]))}return[]})})},Ne=(e,t=[],s)=>{const r="getUsersByImage",{databaseName:o,fields:n=[],session:{userId:a}}=e,{limit:u}=H(s),{objects:g,queries:i}=W(n),p=`users/${a}`,y=t.map(c=>j(c,32)),I=['r.type == "images"',`POSITION(${JSON.stringify(y)}, LOWER(r.name))`],m=`FOR i, r IN OUTBOUND "${p}" hasReaction
243
51
  OPTIONS {vertexCollections: "images"}
244
- ${selectQueries.join("\n")}
245
- FILTER ${filterBy.join(" && ")}
246
- ${limit.aql}
247
- RETURN MERGE(i, {${selectObjects.join(", ")}})`;
248
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
249
- action,
250
- category: eventCategory,
251
- value: import_error.ErrorTypes.DATABASE_ERROR,
252
- params: { reactions, options }
253
- }, error, context).then(() => []));
254
- };
255
- const getImage = (context, imageId) => {
256
- const action = "getItem";
257
- const { database } = context;
258
- const formatImageId = (0, import_utils.parseId)(imageId);
259
- const aqlQry = import_arangojs.aql`FOR i IN images
260
- FILTER i._key==${formatImageId}
52
+ ${i.join(`
53
+ `)}
54
+ FILTER ${I.join(" && ")}
55
+ ${u.aql}
56
+ RETURN MERGE(i, {${g.join(", ")}})`;return N(o).query(m).then(c=>c.all()).catch(c=>(R({action:r,category:f,params:{options:s,reactions:t},value:E.DATABASE_ERROR},c,e),[]))},Oe=(e,t)=>{const s="getItem",{databaseName:r}=e,o=q(t),n=L`FOR i IN images
57
+ FILTER i._key==${o}
261
58
  LIMIT 1
262
- RETURN i`;
263
- return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
264
- action,
265
- category: eventCategory,
266
- value: import_error.ErrorTypes.DATABASE_ERROR,
267
- params: { imageId }
268
- }, error, context));
269
- };
270
- const getPathUserImages = (userId, imageId, type, dir = "images") => {
271
- let filename = imageId;
272
- switch (type) {
273
- case "image/png":
274
- filename = `${imageId}.png`;
275
- break;
276
- default:
277
- filename = `${imageId}.jpg`;
278
- break;
279
- }
280
- return `users/${userId}/${dir}/${filename}`;
281
- };
282
- const getAppImageUrl = (data) => {
283
- const { imageId, directory = "images", imageType = "profile", isThumb, type, typeId } = data;
284
- const host = import_config.Config.get("environment") === "prod" ? `https://box.${import_config.Config.get("app.url")}` : `https://s3.amazonaws.com/dev.${import_config.Config.get("app.url")}`;
285
- const suffix = isThumb ? "-th" : "";
286
- if (imageId) {
287
- switch (type) {
288
- case "apps":
289
- return `${host}/${type}/${directory}/${imageId}${suffix}.jpg`;
290
- case "users":
291
- return `${host}/${type}/${typeId}/${directory}/${imageId}${suffix}.jpg`;
292
- }
293
- if (imageType === "profile") {
294
- return `${host}/defaults/${type}_bk${suffix}.jpg`;
295
- }
296
- return `${host}/defaults/${type}_wh${suffix}.jpg`;
297
- }
298
- return "";
299
- };
300
- const getImageUrl = async (data) => {
301
- const { bucket, imageId, isThumb = false, type, typeId } = data;
302
- const imgDir = isThumb ? "thumbs" : "images";
303
- if (imageId) {
304
- const imageKey = `${type}/${typeId}/${imgDir}/${imageId}.jpg`;
305
- return await (0, import_s3.s3GetSignedUrl)({ Bucket: bucket, Key: imageKey, Expires: 900 });
306
- }
307
- return "";
308
- };
309
- const resizeSaveImage = async (context, imageId, buffer, fileType = "image/jpeg", s3Options) => {
310
- const action = "resizeSaveImage";
311
- const { session: { userId: sessionId } } = context;
312
- const imgW = import_config.Config.get("image.imgSize");
313
- const imgQ = import_config.Config.get("image.imgQuality");
314
- const format = fileType.split("/")[1];
315
- const imageOptimizedBuffer = await new Promise((resolve, reject) => {
316
- (0, import_gm.default)(buffer, "img").setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, ">").stream((streamError, stdout) => {
317
- if (streamError) {
318
- reject(
319
- (0, import_analyticsUtils.logError)({
320
- action,
321
- category: eventCategory,
322
- params: { imageId, fileType },
323
- value: import_error.ErrorTypes.IMAGE_SAVE
324
- }, streamError, context)
325
- );
326
- return;
327
- }
328
- let imageBuffer = Buffer.from("");
329
- stdout.on("data", (data) => {
330
- imageBuffer = Buffer.concat([imageBuffer, data]);
331
- });
332
- stdout.on("end", () => resolve(imageBuffer));
333
- });
334
- });
335
- try {
336
- const imageObj = {
337
- ACL: "authenticated-read",
338
- Body: imageOptimizedBuffer,
339
- Bucket: null,
340
- ContentType: fileType,
341
- ...s3Options || {},
342
- Key: getPathUserImages(sessionId, imageId, fileType, "images")
343
- };
344
- await (0, import_s3.s3Put)(imageObj);
345
- } catch (error) {
346
- return (0, import_analyticsUtils.logError)({
347
- action,
348
- category: eventCategory,
349
- params: { imageId, fileType },
350
- value: import_error.ErrorTypes.IMAGE_SAVE
351
- }, error, context);
352
- }
353
- const imageIdentity = await new Promise((resolve) => {
354
- (0, import_gm.default)(imageOptimizedBuffer, "img").identify({ bufferStream: true }, (error, imageData = {}) => {
355
- if (error) {
356
- return (0, import_analyticsUtils.logError)({
357
- action,
358
- category: eventCategory,
359
- value: import_error.ErrorTypes.IMAGE_SAVE
360
- }, error, context);
361
- }
362
- const formatValues = (0, import_utils.lowerCaseKeys)(imageData);
363
- const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatValues;
364
- const stats = formatValues["channel statistics"];
365
- let color;
366
- if (stats) {
367
- let { red, green, blue, mean } = stats;
368
- if (red) {
369
- mean = red["standard deviation"] || red.mean;
370
- red = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
371
- } else {
372
- red = 0;
373
- }
374
- if (green) {
375
- mean = green["standard deviation"] || green.mean;
376
- green = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
377
- } else {
378
- green = 0;
379
- }
380
- if (blue) {
381
- mean = blue["standard deviation"] || blue.mean;
382
- blue = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
383
- } else {
384
- blue = 0;
385
- }
386
- const rgb = blue | green << 8 | red << 16;
387
- const rgbColor = rgb.toString(16);
388
- color = rgbColor === "0" ? "000000" : rgbColor;
389
- }
390
- const {
391
- "JPEG-Quality": quality,
392
- Orientation: orientationData,
393
- Resolution: resolutionData,
394
- size
395
- } = imageData;
396
- let resolution;
397
- if (resolutionData) {
398
- const [resolutionNumber, resolutionUnit] = resolutionData.split(" ");
399
- const resolutionValue = resolutionNumber.split("x")[0];
400
- const resolutionUnitValue = resolutionUnit === "pixels/inch" ? "ppi" : "";
401
- resolution = `${resolutionValue} ${resolutionUnitValue}`;
402
- }
403
- return resolve({
404
- color,
405
- fileSize: imageOptimizedBuffer.length,
406
- height: size?.height || 0,
407
- make: cameraMake,
408
- model: cameraModel,
409
- orientation: orientationData === "Unknown" ? void 0 : orientationData?.toLowerCase(),
410
- quality: quality ? +quality : void 0,
411
- resolution,
412
- taken: taken ? import_luxon.DateTime.fromMillis(taken).millisecond : void 0,
413
- width: size?.width || 0
414
- });
415
- });
416
- });
417
- const thumbOptimizedBuffer = await new Promise((resolve, reject) => {
418
- const thmW = import_config.Config.get("image.thmSize");
419
- const thmQ = import_config.Config.get("image.thmQuality");
420
- (0, import_gm.default)(imageOptimizedBuffer, "img").setFormat(format).gravity("Center").resize(thmW, thmW, "^").extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout) => {
421
- if (streamError) {
422
- (0, import_analyticsUtils.logError)({
423
- action,
424
- category: eventCategory,
425
- params: { imageId, fileType },
426
- value: import_error.ErrorTypes.IMAGE_SAVE
427
- }, streamError, context);
428
- return;
429
- }
430
- let thumbBuffer = Buffer.from("");
431
- thumbStdout.on("data", (data) => {
432
- thumbBuffer = Buffer.concat([thumbBuffer, data]);
433
- });
434
- thumbStdout.on("end", () => resolve(thumbBuffer));
435
- });
436
- });
437
- try {
438
- const thumbObj = {
439
- ACL: "authenticated-read",
440
- Body: thumbOptimizedBuffer,
441
- Bucket: null,
442
- ContentType: fileType,
443
- ...s3Options || {},
444
- Key: getPathUserImages(sessionId, imageId, fileType, "thumbs")
445
- };
446
- await (0, import_s3.s3Put)(thumbObj);
447
- } catch (error) {
448
- return (0, import_analyticsUtils.logError)({
449
- action,
450
- category: eventCategory,
451
- value: import_error.ErrorTypes.IMAGE_SAVE
452
- }, error, context);
453
- }
454
- return imageIdentity;
455
- };
456
- const addImage = (context, image, s3Options) => {
457
- const action = "addImage";
458
- const { database, session: { userId: sessionId } } = context;
459
- const { imageId, description, buffer, fileType } = image;
460
- const now = Date.now();
461
- return resizeSaveImage(context, imageId, buffer, fileType, s3Options).then((resizedImage) => {
462
- const insert = {
463
- ...resizedImage,
464
- _key: imageId,
465
- added: now,
466
- description,
467
- fileType,
468
- modified: now,
469
- userId: sessionId
470
- };
471
- const aqlQry = import_arangojs.aql`INSERT ${insert} IN images RETURN NEW`;
472
- return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
473
- action,
474
- category: eventCategory,
475
- params: { image, s3Options },
476
- value: import_error.ErrorTypes.DATABASE_ERROR
477
- }, error, context));
478
- }).catch((error) => (0, import_analyticsUtils.logError)({
479
- action,
480
- category: eventCategory,
481
- value: import_error.ErrorTypes.IMAGE_RESIZE
482
- }, error, context));
483
- };
484
- const addImageEdge = async (context, imageEdge) => {
485
- const action = "addImageEdge";
486
- const { database, session: { userId: sessionId } } = context;
487
- const { imageId, itemId, itemType } = imageEdge;
488
- const now = Date.now();
489
- const edgeCollection = database.collection("hasImage");
490
- const edgeId = (0, import_utils.createHash)(`hasImage-${imageId}-${itemId}-${sessionId}`);
491
- const formatItemType = (0, import_utils.parseChar)(itemType).toLowerCase();
492
- const formatItemId = (0, import_utils.parseId)(itemId);
493
- const itemDocId = (0, import_arangodbUtils.getDocId)(formatItemType, { _key: formatItemId });
494
- const formatImageId = (0, import_utils.parseId)(imageId);
495
- const fileDocId = `images/${formatImageId}`;
496
- const edge = {
497
- _from: itemDocId,
498
- _key: edgeId,
499
- _to: fileDocId,
500
- added: now,
501
- type: itemType
502
- };
503
- if (itemDocId) {
504
- return edgeCollection.save(edge, { returnNew: true }).then((fileEdge) => edgeCollection.document(fileEdge)).catch((error) => (0, import_analyticsUtils.logError)({
505
- action,
506
- category: eventCategory,
507
- params: {
508
- edge,
509
- fileDocId,
510
- imageId,
511
- itemDocId,
512
- itemId,
513
- itemType
514
- },
515
- value: import_error.ErrorTypes.DATABASE_ERROR
516
- }, error, context));
517
- }
518
- return null;
519
- };
520
- const updateImage = async (context, image, s3Options) => {
521
- const action = "updateImage";
522
- const { database, session: { userId: sessionId } } = context;
523
- const {
524
- base64 = "",
525
- buffer: imageBuffer,
526
- description = "",
527
- imageId,
528
- itemId,
529
- itemType,
530
- fileType,
531
- url = ""
532
- } = image;
533
- const now = Date.now();
534
- const formatImageId = imageId || (0, import_utils.createHash)(`image-${sessionId}`);
535
- const formatItemId = (0, import_utils.parseId)(itemId);
536
- const formatItemType = (0, import_utils.parseChar)(itemType, 16).toLowerCase();
537
- const customParams = {
538
- description,
539
- imageId: formatImageId,
540
- userId: sessionId
541
- };
542
- if (base64 || imageBuffer) {
543
- let buffer = imageBuffer;
544
- if (base64) {
545
- const formatBase64 = base64.substr(base64.indexOf(",") + 1);
546
- buffer = Buffer.from(formatBase64, "base64");
547
- }
548
- let updatedFileType = fileType;
549
- if (!fileType) {
550
- const { fileTypeFromBuffer } = await import("file-type");
551
- const { mime } = await fileTypeFromBuffer(buffer);
552
- updatedFileType = mime;
553
- }
554
- const imgParams = {
555
- ...customParams,
556
- buffer,
557
- fileType: updatedFileType
558
- };
559
- return addImage(context, imgParams, s3Options).then((image2) => {
560
- if (formatItemId && formatItemType) {
561
- const imageEdge = {
562
- imageId: formatImageId,
563
- itemId: formatItemId,
564
- itemType: formatItemType
565
- };
566
- return addImageEdge(context, imageEdge).then(() => image2);
567
- }
568
- return image2;
569
- }).catch((error) => (0, import_analyticsUtils.logError)({
570
- action,
571
- category: eventCategory,
572
- value: import_error.ErrorTypes.IMAGE_SAVE
573
- }, error, context));
574
- } else if (url !== "") {
575
- let contentType;
576
- return (0, import_rip_hunter.get)(url).then((res) => {
577
- if (res.status !== 200) {
578
- return (0, import_analyticsUtils.logException)({
579
- action,
580
- category: eventCategory,
581
- value: import_error.ErrorTypes.IMAGE_REQUEST
582
- }, context);
583
- }
584
- contentType = res.headers.get("content-type");
585
- return res;
586
- }).then((res) => res.buffer()).then((buffer) => {
587
- const imgParams = {
588
- ...customParams,
589
- buffer,
590
- fileType: contentType
591
- };
592
- return addImage(context, imgParams, s3Options).then((image2) => {
593
- if (formatItemId && formatItemType) {
594
- const imageEdge = { imageId: formatImageId, itemId: formatItemId, itemType: formatItemType };
595
- return addImageEdge(context, imageEdge).then(() => image2);
596
- }
597
- return image2;
598
- });
599
- }).catch((error) => (0, import_analyticsUtils.logError)({
600
- action,
601
- category: eventCategory,
602
- value: "fetch_error"
603
- }, error, context));
604
- } else if (imageId !== "") {
605
- const update = {
606
- description,
607
- modified: now
608
- };
609
- const aqlQry = import_arangojs.aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;
610
- return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
611
- action,
612
- category: eventCategory,
613
- params: {
614
- aqlQry,
615
- formatImageId,
616
- update
617
- },
618
- value: import_error.ErrorTypes.DATABASE_ERROR
619
- }, error, context));
620
- }
621
- return null;
622
- };
623
- const deleteImage = async (context, imageId) => {
624
- const action = "delete";
625
- const { database, session: { userId: sessionId } } = context;
626
- const formatImageId = (0, import_utils.parseId)(imageId);
627
- try {
628
- const removeEdgeQuery = import_arangojs.aql`FOR hi IN hasImage
629
- FILTER hi._from == ${formatImageId}
59
+ RETURN i`;return N(r).query(n).then(a=>a.next()).catch(a=>R({action:s,category:f,params:{imageId:t},value:E.DATABASE_ERROR},a,e))},M=(e,t,s,r="images")=>{let o=t;switch(s){case"image/png":o=`${t}.png`;break;default:o=`${t}.jpg`;break}return`users/${e}/${r}/${o}`},$e=({bucket:e,directory:t="images",imageId:s,imageType:r="jpg",isThumb:o=!1,type:n,typeId:a})=>{if(!n)return"";if(!s)return r==="profile"?`${O.get("image.defaultUrl")}/users_bk.jpg`:`${O.get("image.defaultUrl")}/users_wh.jpg`;const u=e||O.get("image.bucket")||"dev.reaktor.io",g=o?`${s}-th.${r}`:`${s}.${r==="profile"||r==="other"?"jpg":r}`,i=O.get("image.url")||`https://s3.amazonaws.com/${u}`;return n==="apps"?`${i}/${n}/${t}/${g}`:`${i}/${n}/${a}/${t}/${g}`},Be=async({bucket:e,imageDir:t="images",imageId:s,imageType:r="jpg",isThumb:o=!1,type:n,typeId:a,urlType:u="public"})=>{if(!s)return"";o&&(u="signed");const i=`${n}/${a}/${t}/${s}${o?"-th":""}.${r}`;switch(u){case"signed":try{const p=O.get("image.bucket")||"dev.reaktor.io";return await ae({Bucket:e||p,Expires:900,Key:i})}catch(p){throw p}case"public":return`https://box.${O.get("app.url")}/${i}`;case"dev":return`https://s3.amazonaws.com/dev.${O.get("app.url")}/${i}`;default:return""}},ie=async(e,t,s,r="image/jpeg",o)=>{const n="resizeSaveImage",{session:{userId:a}}=e,u=O.get("image.imgSize"),g=O.get("image.imgQuality"),i=r.split("/")[1],p=await new Promise((m,c)=>{w(s,"img").setFormat(i).quality(g).autoOrient().resize(u,u,">").stream((l,b)=>{if(l){c(R({action:n,category:f,params:{fileType:r,imageId:t},value:E.IMAGE_SAVE},l,e));return}let T=Buffer.from("");b.on("data",C=>{T=Buffer.concat([T,C])}),b.on("end",()=>m(T))})});try{const m={ACL:"authenticated-read",Body:p,Bucket:null,ContentType:r,...o||{},Key:M(a,t,r,"images")};await G(m)}catch(m){return R({action:n,category:f,params:{fileType:r,imageId:t},value:E.IMAGE_SAVE},m,e)}const y=await new Promise(m=>{w(p,"img").identify({bufferStream:!0},(c,l={})=>{if(c)return R({action:n,category:f,value:E.IMAGE_SAVE},c,e);const b=Z(l),{make:T,model:C,datetimeoriginal:h}=b,d=b["channel statistics"];let $;if(d){let{red:D,green:U,blue:S,mean:B}=d;D?(B=D["standard deviation"]||D.mean,D=B?+B.split(" ")[0].substring(0,3):0):D=0,U?(B=U["standard deviation"]||U.mean,U=B?+B.split(" ")[0].substring(0,3):0):U=0,S?(B=S["standard deviation"]||S.mean,S=B?+B.split(" ")[0].substring(0,3):0):S=0;const Q=(S|U<<8|D<<16).toString(16);$=Q==="0"?"000000":Q}const{"JPEG-Quality":A,Orientation:v,Resolution:F,size:k}=l;let P;if(F){const[D,U]=F.split(" ");P=`${D.split("x")[0]} ${U==="pixels/inch"?"ppi":""}`}return m({color:$,fileSize:p.length,fileType:i,height:k?.height||0,imageId:t,make:T,model:C,orientation:v==="Unknown"?void 0:v?.toLowerCase(),quality:A?+A:void 0,resolution:P,taken:h?Y.fromMillis(h).millisecond:void 0,width:k?.width||0})})}),I=await new Promise((m,c)=>{const l=O.get("image.thmSize"),b=O.get("image.thmQuality");w(p,"img").setFormat(i).gravity("Center").resize(l,l,"^").extent(l,l).quality(b).stream((T,C)=>{if(T){R({action:n,category:f,params:{fileType:r,imageId:t},value:E.IMAGE_SAVE},T,e);return}let h=Buffer.from("");C.on("data",d=>{h=Buffer.concat([h,d])}),C.on("end",()=>m(h))})});try{const m={ACL:"authenticated-read",Body:I,Bucket:null,ContentType:r,...o||{},Key:M(a,t,r,"thumbs")};await G(m)}catch(m){return R({action:n,category:f,value:E.IMAGE_SAVE},m,e)}return y},x=(e,t,s)=>{const r="addImage",{databaseName:o,session:{userId:n}}=e,{imageId:a,description:u,buffer:g,fileType:i}=t,p=Date.now();return ie(e,a,g,i,s).then(y=>{const I={...y,_key:a,added:p,description:u,fileType:i,modified:p,userId:n},m=L`INSERT ${I} IN images RETURN NEW`;return N(o).query(m).then(c=>c.next()).catch(c=>R({action:r,category:f,params:{image:t,s3Options:s},value:E.DATABASE_ERROR},c,e))}).catch(y=>R({action:r,category:f,value:E.IMAGE_RESIZE},y,e))},V=async(e,t)=>{const s="addImageEdge",{databaseName:r,session:{userId:o}}=e,{imageId:n,itemId:a,itemType:u}=t,g=Date.now(),i=N(r).collection("hasImage"),p=z(`hasImage-${n}-${a}-${o}`),y=j(u).toLowerCase(),I=q(a),m=ne(y,{_key:I}),l=`images/${q(n)}`,b={_from:m,_key:p,_to:l,added:g,itemType:u};return m?N(r).collection("hasImage").save(b,{returnNew:!0}).then(T=>N(r).collection("hasImage").document(T)).catch(T=>R({action:s,category:f,params:{edge:b,fileDocId:l,imageId:n,itemDocId:m,itemId:a,itemType:u},value:E.DATABASE_ERROR},T,e)):null},Ce=async(e,t,s)=>{const r="updateImage",{databaseName:o,session:{userId:n}}=e,{base64:a="",buffer:u,description:g="",imageId:i,itemId:p,itemType:y,fileType:I,url:m=""}=t,c=Date.now(),l=i||z(`image-${n}`),b=q(p),T=j(y,16).toLowerCase(),C={description:g,imageId:l,userId:n};if(a||u){let h=u;if(a){const A=a.substr(a.indexOf(",")+1);h=Buffer.from(A,"base64")}let d=I;if(!I){const{fileTypeFromBuffer:A}=await import("file-type"),{mime:v}=await A(h);d=v}const $={...C,buffer:h,fileType:d};return x(e,$,s).then(A=>b&&T?V(e,{imageId:l,itemId:b,itemType:T}).then(()=>A):A).catch(A=>R({action:r,category:f,value:E.IMAGE_SAVE},A,e))}else if(m!==""){let h;return J(m).then(d=>d.status!==200?se({action:r,category:f,value:E.IMAGE_REQUEST},e):(h=d.headers.get("content-type"),d)).then(d=>d.buffer()).then(d=>{const $={...C,buffer:d,fileType:h};return x(e,$,s).then(A=>b&&T?V(e,{imageId:l,itemId:b,itemType:T}).then(()=>A):A)}).catch(d=>R({action:r,category:f,value:"fetch_error"},d,e))}else if(i!==""){const h={description:g,modified:c},d=L`UPDATE {_key: ${l}} WITH ${h} IN images RETURN NEW`;return N(o).query(d).then($=>$.next()).catch($=>R({action:r,category:f,params:{aqlQry:d,formatImageId:l,update:h},value:E.DATABASE_ERROR},$,e))}return null},De=async(e,t)=>{const s="delete",{databaseName:r,session:{userId:o}}=e,n=q(t);try{const a=L`FOR hi IN hasImage
60
+ FILTER hi._from == ${n}
630
61
  REMOVE hi IN hasImage
631
- RETURN OLD`;
632
- await database.query(removeEdgeQuery);
633
- const aqlQuery = import_arangojs.aql`FOR i IN images
634
- FILTER i._key == ${formatImageId} && i.userId == ${sessionId}
62
+ RETURN OLD`;await N(r).query(a);const u=L`FOR i IN images
63
+ FILTER i._key == ${n} && i.userId == ${o}
635
64
  REMOVE i IN images
636
- RETURN OLD`;
637
- const image = await database.query(aqlQuery).then((cursor) => cursor.next());
638
- if (image) {
639
- const { _key: imageKey } = image;
640
- const params = {
641
- Bucket: null,
642
- Delete: {
643
- Objects: [
644
- { Key: `users/${sessionId}/images/${imageKey}.jpg` },
645
- { Key: `users/${sessionId}/thumbs/${imageKey}.jpg` }
646
- ],
647
- Quiet: true
648
- }
649
- };
650
- return (0, import_s3.s3DeleteList)(params).then(() => image);
651
- }
652
- return null;
653
- } catch (error) {
654
- return (0, import_analyticsUtils.logError)({
655
- action,
656
- category: eventCategory,
657
- params: { imageId },
658
- value: import_error.ErrorTypes.DATABASE_ERROR
659
- }, error, context);
660
- }
661
- };
662
- // Annotate the CommonJS export names for ESM import in node:
663
- 0 && (module.exports = {
664
- addImage,
665
- addImageEdge,
666
- deleteImage,
667
- getAppImageUrl,
668
- getImage,
669
- getImageCountByItem,
670
- getImageOptional,
671
- getImageUrl,
672
- getImagesByGroup,
673
- getImagesByItem,
674
- getImagesByReactions,
675
- getImagesByUser,
676
- getPathUserImages,
677
- parseImageOptions,
678
- resizeSaveImage,
679
- updateImage
680
- });
681
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW1hZ2VzLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtEZWxldGVPYmplY3RzQ29tbWFuZElucHV0LCBQdXRPYmplY3RDb21tYW5kSW5wdXR9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1zMyc7XG5pbXBvcnQge2dldCBhcyBodHRwR2V0fSBmcm9tICdAbmxhYnMvcmlwLWh1bnRlcic7XG5pbXBvcnQge2NyZWF0ZUhhc2gsIGxvd2VyQ2FzZUtleXMsIHBhcnNlQXJhbmdvSWQsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW19IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbH0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvYXFsJztcbmltcG9ydCB7RWRnZUNvbGxlY3Rpb259IGZyb20gJ2FyYW5nb2pzL2NvbGxlY3Rpb24nO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvY3Vyc29yJztcbmltcG9ydCB7TWltZVR5cGV9IGZyb20gJ2ZpbGUtdHlwZSc7XG5pbXBvcnQgZ20gZnJvbSAnZ20nO1xuaW1wb3J0IHtEYXRlVGltZX0gZnJvbSAnbHV4b24nO1xuXG5pbXBvcnQge2dldEdyb3VwRGV0YWlscywgaXNHcm91cGVkfSBmcm9tICcuL2dyb3Vwcyc7XG5pbXBvcnQge3MzRGVsZXRlTGlzdCwgczNHZXRTaWduZWRVcmwsIHMzUHV0fSBmcm9tICcuL3MzJztcbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHR5cGUge1xuICBBcGlDb250ZXh0LFxuICBBcmFuZ29EYkNvbGxlY3Rpb24sXG4gIEFyYW5nb0RiTGltaXQsXG4gIEdyb3VwVHlwZSxcbiAgR3JvdXBVc2VyLFxuICBJbWFnZUVkZ2VUeXBlLFxuICBJbWFnZUlkZW50aWZ5VHlwZSxcbiAgSW1hZ2VPcHRpb25zLFxuICBJbWFnZVR5cGUsXG4gIEltYWdlVXJsRGF0YSxcbiAgUXVlcnlGaWx0ZXJcbn0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtsb2dFcnJvciwgbG9nRXhjZXB0aW9ufSBmcm9tICcuLi91dGlscy9hbmFseXRpY3NVdGlscyc7XG5pbXBvcnQge0Vycm9yVHlwZXN9IGZyb20gJy4uL3R5cGVzL2Vycm9yJztcbmltcG9ydCB7Z2V0RG9jSWQsIGdldExpbWl0LCBzZWxlY3RSZWFjdGlvbkNvdW50QnlUeXBlfSBmcm9tICcuLi91dGlscy9hcmFuZ29kYlV0aWxzJztcblxuY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ2ltYWdlcyc7XG5nbS5zdWJDbGFzcyh7aW1hZ2VNYWdpY2s6ICc3Kyd9KTtcblxuZXhwb3J0IGNvbnN0IHBhcnNlSW1hZ2VPcHRpb25zID0gKG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICB0byA9IDMwLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIHtcbiAgICBsaW1pdDogZ2V0TGltaXQoZnJvbSwgdG8pIGFzIEFyYW5nb0RiTGltaXQsXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlT3B0aW9uYWwgPSAoZmllbGRzOiBzdHJpbmdbXSA9IFtdKSA9PlxuICBmaWVsZHMucmVkdWNlKChzZWxlY3RzOiBhbnksIGZpZWxkOiBzdHJpbmcpID0+IHtcbiAgICBpZihmaWVsZC5pbmNsdWRlcygnQ291bnQnKSkge1xuICAgICAgcmV0dXJuIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGUoJ2ltYWdlcycsICdpJywgZmllbGQsIHNlbGVjdHMpO1xuICAgIH1cblxuICAgIHN3aXRjaChmaWVsZCkge1xuICAgICAgY2FzZSAncmVhY3Rpb25zJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHJlYWN0aW9ucyA9IChcbiAgICAgICAgICBGT1IgaW1hZ2UsIHIgSU4gSU5CT1VORCBpLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICBDT0xMRUNUIHJlYWN0aW9uTmFtZSA9IHIudmFsdWUgSU5UTyByZWFjdGlvbkl0ZW1zXG4gICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgncmVhY3Rpb25zOnJlYWN0aW9ucycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3RhZ3MnOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgdGFncyA9IChcbiAgICAgICAgICBGT1IgdCwgcGwgSU4gSU5CT1VORCBpLl9pZCBpc1RhZ2dlZFxuICAgICAgICAgIFJFVFVSTiB0XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3RhZ3M6dGFncycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3VzZXJzJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHVzZXJzID0gRklSU1QoXG4gICAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgICBGSUxURVIgaS51c2VySWQgPT0gdS5fa2V5XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiB1XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3VzZXJzOnVzZXJzJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDoge1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICB9XG4gIH0sIHtvYmplY3RzOiBbXSwgcXVlcmllczogW119KTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5VXNlciA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgdXNlcklkOiBzdHJpbmcsXG4gIGZyb206IG51bWJlciA9IDAsXG4gIHRvOiBudW1iZXIgPSAzMFxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VXNlcklkOiBzdHJpbmcgPSBwYXJzZUlkKHVzZXJJZCk7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EYkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSBJTiBpbWFnZXNcbiAgICAgIEZJTFRFUiBpLnVzZXJJZCA9PSBcIiR7Zm9ybWF0VXNlcklkfVwiXG4gICAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICBGSUxURVIgdS5fa2V5ID09IGkudXNlcklkXG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgICR7bGltaXQuYXFsfVxuICAgICAgU09SVCBpLmFkZGVkXG4gICAgICBSRVRVUk4gTUVSR0UoaSwge3VzZXI6dXNlcn0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtmcm9tLCB0bywgdXNlcklkfVxuICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VDb3VudEJ5SXRlbSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpdGVtSWQ6IHN0cmluZyk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlQ291bnRCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUFyYW5nb0lkKGl0ZW1JZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGkgSU4gaGFzSW1hZ2VcbiAgICAgIEZJTFRFUiBpLl90byA9PSAke2Zvcm1hdEl0ZW1JZH1cbiAgICAgIFJFVFVSTiB7Y291bnQ6IENPVU5UKGkpfWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoe2NvdW50fSkgPT4gY291bnQpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtpdGVtSWR9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZXNCeUl0ZW0gPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGl0ZW1JZDogc3RyaW5nLFxuICBvcHRpb25zOiBJbWFnZU9wdGlvbnMgPSB7fVxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkcyA9IFtdfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VBcmFuZ29JZChpdGVtSWQpO1xuICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldEltYWdlT3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGksIGwgSU4gMS4uMSBPVVRCT1VORCBcIiR7Zm9ybWF0SXRlbUlkfVwiIGhhc0ltYWdlXG4gICAgRklMVEVSIE5PVCBJU19OVUxMKGkpXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIFNPUlQgaS5hZGRlZFxuICAgICR7bGltaXQuYXFsfVxuICAgIFJFVFVSTiBNRVJHRShpLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUixcbiAgICAgIHBhcmFtczoge2l0ZW1JZCwgb3B0aW9uc31cbiAgICB9LCBlcnJvciwgY29udGV4dCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5R3JvdXAgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHBhcmFtczoge2ZpbHRlcnM6IGFueVtdLCBncm91cElkOiBzdHJpbmcsIGZyb206IG51bWJlciwgdG86IG51bWJlcn1cbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0SW1hZ2VzQnlHcm91cCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7ZmlsdGVycywgZ3JvdXBJZCwgZnJvbSwgdG99ID0gcGFyYW1zO1xuICBjb25zdCBmb3JtYXRHcm91cElkOiBzdHJpbmcgPSBwYXJzZUlkKGdyb3VwSWQpO1xuICBjb25zdCBsaW1pdDogQXJhbmdvRGJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcblxuICBmaWx0ZXJzXG4gICAgLm1hcCgoZmlsdGVyOiBRdWVyeUZpbHRlcikgPT4ge1xuICAgICAgY29uc3Qge2NvbmRpdGlvbmFsLCBuYW1lLCB2YWx1ZX0gPSBmaWx0ZXI7XG4gICAgICBsZXQgZm9ybWF0Q29uZDogc3RyaW5nID0gY29uZGl0aW9uYWw7XG5cbiAgICAgIGlmKGNvbmRpdGlvbmFsICE9PSAnPj0nICYmIGNvbmRpdGlvbmFsICE9PSAnPD0nICYmIGNvbmRpdGlvbmFsICE9PSAnPicgJiYgY29uZGl0aW9uYWwgIT09ICc8Jykge1xuICAgICAgICBmb3JtYXRDb25kID0gJz09JztcbiAgICAgIH1cblxuICAgICAgc3dpdGNoKG5hbWUpIHtcbiAgICAgICAgY2FzZSAnYWRkZWQnOlxuICAgICAgICAgIHJldHVybiBgcC5hZGRlZCAke2Zvcm1hdENvbmR9ICR7cGFyc2VOdW0odmFsdWUpfWA7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgcmV0dXJuICcnO1xuICAgICAgfVxuICAgIH0pO1xuXG4gIHJldHVybiBnZXRHcm91cERldGFpbHMoY29udGV4dCwgZm9ybWF0R3JvdXBJZClcbiAgICAudGhlbigoZ3JvdXA6IEdyb3VwVHlwZSkgPT4ge1xuICAgICAgaWYoZ3JvdXAucHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2Zvcm1hdEdyb3VwSWR9XCJgKTtcbiAgICAgICAgY29uc3QgZmlsdGVyU3RyID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBpIElOXG4gICAgICAgICAgRkxBVFRFTihcbiAgICAgICAgICAgIEZPUiBwIElOIHBvc3RzXG4gICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJTdHJ9XG4gICAgICAgICAgICBMRVQgaW1hZ2VzID0gKFxuICAgICAgICAgICAgICBGT1IgaSwgZSBJTiBJTkJPVU5EIHAuX2lkIGhhc0ltYWdlXG4gICAgICAgICAgICAgIFJFVFVSTiBpXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBTT1JUIHAuYWRkZWQgREVTQ1xuICAgICAgICAgICAgUkVUVVJOIGltYWdlc1xuICAgICAgICAgIClcbiAgICAgICAgICBTT1JUIGkuYWRkZWQgREVTQ1xuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBpYDtcblxuICAgICAgICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICAgICAgICBwYXJhbXNcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGlzR3JvdXBlZChkYXRhYmFzZSwgc2Vzc2lvbklkLCBncm91cElkKVxuICAgICAgICAudGhlbigoZ3JvdXBlZDogR3JvdXBVc2VyKSA9PiB7XG4gICAgICAgICAgaWYoZ3JvdXBlZC5pc1ZhbGlkKSB7XG4gICAgICAgICAgICBmaWx0ZXJzLnB1c2goYHAuZ3JvdXBJZCA9PSBcIiR7Z3JvdXBlZC5ncm91cElkfVwiYCk7XG4gICAgICAgICAgICBjb25zdCBmaWx0ZXJMaXN0OiBzdHJpbmcgPSBmaWx0ZXJzLmpvaW4oJyAmJiAnKTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RcbiAgICAgICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJMaXN0fVxuICAgICAgICAgICAgICAgIEZPUiBmIElOIHAuZmlsZXNcbiAgICAgICAgICAgICAgICBGSUxURVIgZi50eXBlID09IFwiaW1hZ2UvanBlZ1wiIHx8IGYudHlwZSA9PSBcImltYWdlL3BuZ1wiXG4gICAgICAgICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgICAgICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICAgICAgICAgICAgICBSRVRVUk4gZmA7XG5cbiAgICAgICAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SLFxuICAgICAgICAgICAgICAgIHBhcmFtc1xuICAgICAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5UmVhY3Rpb25zID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICByZWFjdGlvbnM6IHN0cmluZ1tdID0gW10sXG4gIG9wdGlvbnM/OiBJbWFnZU9wdGlvbnNcbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0VXNlcnNCeUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMgPSBbXSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldEltYWdlT3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgZm9ybWF0U2Vzc2lvbklkOiBzdHJpbmcgPSBgdXNlcnMvJHtzZXNzaW9uSWR9YDtcbiAgY29uc3QgZm9ybWF0UmVhY3Rpb25zOiBzdHJpbmdbXSA9IHJlYWN0aW9ucy5tYXAoKHJlYWN0aW9uTmFtZSkgPT4gcGFyc2VDaGFyKHJlYWN0aW9uTmFtZSwgMzIpKTtcbiAgY29uc3QgZmlsdGVyQnk6IHN0cmluZ1tdID0gW1xuICAgICdyLnR5cGUgPT0gXCJpbWFnZXNcIicsXG4gICAgYFBPU0lUSU9OKCR7SlNPTi5zdHJpbmdpZnkoZm9ybWF0UmVhY3Rpb25zKX0sIExPV0VSKHIubmFtZSkpYF07XG5cbiAgLy8gR2V0IGRhdGEgZnJvbSBkYXRhYmFzZVxuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSwgciBJTiBPVVRCT1VORCBcIiR7Zm9ybWF0U2Vzc2lvbklkfVwiIGhhc1JlYWN0aW9uXG4gICAgT1BUSU9OUyB7dmVydGV4Q29sbGVjdGlvbnM6IFwiaW1hZ2VzXCJ9XG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiAke2ZpbHRlckJ5LmpvaW4oJyAmJiAnKX1cbiAgICAke2xpbWl0LmFxbH1cbiAgICBSRVRVUk4gTUVSR0UoaSwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtyZWFjdGlvbnMsIG9wdGlvbnN9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gW10pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEltYWdlSWQ6IHN0cmluZyA9IHBhcnNlSWQoaW1hZ2VJZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGkgSU4gaW1hZ2VzXG4gICAgRklMVEVSIGkuX2tleT09JHtmb3JtYXRJbWFnZUlkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gaWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUixcbiAgICAgIHBhcmFtczoge2ltYWdlSWR9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQYXRoVXNlckltYWdlcyA9ICh1c2VySWQ6IHN0cmluZywgaW1hZ2VJZDogc3RyaW5nLCB0eXBlOiBzdHJpbmcsIGRpcjogc3RyaW5nID0gJ2ltYWdlcycpOiBzdHJpbmcgPT4ge1xuICBsZXQgZmlsZW5hbWU6IHN0cmluZyA9IGltYWdlSWQ7XG5cbiAgc3dpdGNoKHR5cGUpIHtcbiAgICBjYXNlICdpbWFnZS9wbmcnOlxuICAgICAgZmlsZW5hbWUgPSBgJHtpbWFnZUlkfS5wbmdgO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGZpbGVuYW1lID0gYCR7aW1hZ2VJZH0uanBnYDtcbiAgICAgIGJyZWFrO1xuICB9XG5cbiAgcmV0dXJuIGB1c2Vycy8ke3VzZXJJZH0vJHtkaXJ9LyR7ZmlsZW5hbWV9YDtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRBcHBJbWFnZVVybCA9IChkYXRhOiBJbWFnZVVybERhdGEpOiBzdHJpbmcgPT4ge1xuICBjb25zdCB7aW1hZ2VJZCwgZGlyZWN0b3J5ID0gJ2ltYWdlcycsIGltYWdlVHlwZSA9ICdwcm9maWxlJywgaXNUaHVtYiwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGhvc3Q6IHN0cmluZyA9IENvbmZpZy5nZXQoJ2Vudmlyb25tZW50JykgPT09ICdwcm9kJ1xuICAgID8gYGh0dHBzOi8vYm94LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWBcbiAgICA6IGBodHRwczovL3MzLmFtYXpvbmF3cy5jb20vZGV2LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWA7XG4gIGNvbnN0IHN1ZmZpeDogc3RyaW5nID0gaXNUaHVtYiA/ICctdGgnIDogJyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIHN3aXRjaCh0eXBlKSB7XG4gICAgICBjYXNlICdhcHBzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC9hcHAvaW1hZ2VzLzEyMy5qcGdcbiAgICAgICAgcmV0dXJuIGAke2hvc3R9LyR7dHlwZX0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICAgIGNhc2UgJ3VzZXJzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC91c2Vycy9kZW1vVXNlci9pbWFnZXMvMTIzLmpwZ1xuICAgICAgICByZXR1cm4gYCR7aG9zdH0vJHt0eXBlfS8ke3R5cGVJZH0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICBpZihpbWFnZVR5cGUgPT09ICdwcm9maWxlJykge1xuICAgICAgcmV0dXJuIGAke2hvc3R9L2RlZmF1bHRzLyR7dHlwZX1fYmske3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICByZXR1cm4gYCR7aG9zdH0vZGVmYXVsdHMvJHt0eXBlfV93aCR7c3VmZml4fS5qcGdgO1xuICB9XG5cbiAgcmV0dXJuICcnO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlVXJsID0gYXN5bmMgKGRhdGE6IEltYWdlVXJsRGF0YSk6IFByb21pc2U8c3RyaW5nPiA9PiB7XG4gIGNvbnN0IHtidWNrZXQsIGltYWdlSWQsIGlzVGh1bWIgPSBmYWxzZSwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGltZ0Rpcjogc3RyaW5nID0gaXNUaHVtYiA/ICd0aHVtYnMnIDogJ2ltYWdlcyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIGNvbnN0IGltYWdlS2V5OiBzdHJpbmcgPSBgJHt0eXBlfS8ke3R5cGVJZH0vJHtpbWdEaXJ9LyR7aW1hZ2VJZH0uanBnYDtcbiAgICByZXR1cm4gYXdhaXQgczNHZXRTaWduZWRVcmwoe0J1Y2tldDogYnVja2V0LCBLZXk6IGltYWdlS2V5LCBFeHBpcmVzOiA5MDB9KTtcbiAgfVxuXG4gIHJldHVybiAnJztcbn07XG5cbmV4cG9ydCBjb25zdCByZXNpemVTYXZlSW1hZ2UgPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGltYWdlSWQ6IHN0cmluZyxcbiAgYnVmZmVyOiBCdWZmZXIsXG4gIGZpbGVUeXBlOiBzdHJpbmcgPSAnaW1hZ2UvanBlZycsIHMzT3B0aW9ucz86IFB1dE9iamVjdENvbW1hbmRJbnB1dFxuKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAncmVzaXplU2F2ZUltYWdlJztcbiAgY29uc3Qge3Nlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3QgaW1nVzogbnVtYmVyID0gQ29uZmlnLmdldCgnaW1hZ2UuaW1nU2l6ZScpO1xuICBjb25zdCBpbWdROiBudW1iZXIgPSBDb25maWcuZ2V0KCdpbWFnZS5pbWdRdWFsaXR5Jyk7XG4gIGNvbnN0IGZvcm1hdDogc3RyaW5nID0gKGZpbGVUeXBlLnNwbGl0KCcvJykpWzFdO1xuXG4gIGNvbnN0IGltYWdlT3B0aW1pemVkQnVmZmVyOiBCdWZmZXIgPSBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgZ20oYnVmZmVyLCAnaW1nJylcbiAgICAgIC5zZXRGb3JtYXQoZm9ybWF0KVxuICAgICAgLnF1YWxpdHkoaW1nUSlcbiAgICAgIC5hdXRvT3JpZW50KClcbiAgICAgIC5yZXNpemUoaW1nVywgaW1nVywgJz4nKVxuICAgICAgLnN0cmVhbSgoc3RyZWFtRXJyb3I6IEVycm9yLCBzdGRvdXQpOiB2b2lkID0+IHtcbiAgICAgICAgaWYoc3RyZWFtRXJyb3IpIHtcbiAgICAgICAgICByZWplY3QoXG4gICAgICAgICAgICBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIHBhcmFtczoge2ltYWdlSWQsIGZpbGVUeXBlfSxcbiAgICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgICAgICAgfSwgc3RyZWFtRXJyb3IsIGNvbnRleHQpXG4gICAgICAgICAgKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBsZXQgaW1hZ2VCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICBzdGRvdXQub24oJ2RhdGEnLCAoZGF0YSkgPT4ge1xuICAgICAgICAgIGltYWdlQnVmZmVyID0gQnVmZmVyLmNvbmNhdChbaW1hZ2VCdWZmZXIsIGRhdGFdKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgc3Rkb3V0Lm9uKCdlbmQnLCAoKSA9PiByZXNvbHZlKGltYWdlQnVmZmVyKSk7XG4gICAgICB9KTtcbiAgfSk7XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBpbWFnZU9iajogUHV0T2JqZWN0Q29tbWFuZElucHV0ID0ge1xuICAgICAgQUNMOiAnYXV0aGVudGljYXRlZC1yZWFkJyxcbiAgICAgIEJvZHk6IGltYWdlT3B0aW1pemVkQnVmZmVyLFxuICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgQ29udGVudFR5cGU6IGZpbGVUeXBlLFxuICAgICAgLi4uKHMzT3B0aW9ucyB8fCB7fSksXG4gICAgICBLZXk6IGdldFBhdGhVc2VySW1hZ2VzKHNlc3Npb25JZCwgaW1hZ2VJZCwgZmlsZVR5cGUsICdpbWFnZXMnKVxuICAgIH07XG4gICAgYXdhaXQgczNQdXQoaW1hZ2VPYmopO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgcmV0dXJuIGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgcGFyYW1zOiB7aW1hZ2VJZCwgZmlsZVR5cGV9LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgIH0sIGVycm9yLCBjb250ZXh0KTtcbiAgfVxuXG4gIGNvbnN0IGltYWdlSWRlbnRpdHkgPSBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4ge1xuICAgIGdtKGltYWdlT3B0aW1pemVkQnVmZmVyLCAnaW1nJylcbiAgICAgIC5pZGVudGlmeSh7YnVmZmVyU3RyZWFtOiB0cnVlfSwgKGVycm9yOiBFcnJvciwgaW1hZ2VEYXRhOiBJbWFnZUlkZW50aWZ5VHlwZSA9IHt9KTogYW55ID0+IHtcbiAgICAgICAgaWYoZXJyb3IpIHtcbiAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTUFHRV9TQVZFXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgZm9ybWF0VmFsdWVzID0gbG93ZXJDYXNlS2V5cyhpbWFnZURhdGEpO1xuICAgICAgICBjb25zdCB7bWFrZTogY2FtZXJhTWFrZSwgbW9kZWw6IGNhbWVyYU1vZGVsLCBkYXRldGltZW9yaWdpbmFsOiB0YWtlbn06IEltYWdlSWRlbnRpZnlUeXBlID0gZm9ybWF0VmFsdWVzO1xuICAgICAgICBjb25zdCBzdGF0cyA9IGZvcm1hdFZhbHVlc1snY2hhbm5lbCBzdGF0aXN0aWNzJ107XG4gICAgICAgIGxldCBjb2xvcjogc3RyaW5nO1xuXG4gICAgICAgIGlmKHN0YXRzKSB7XG4gICAgICAgICAgbGV0IHtyZWQsIGdyZWVuLCBibHVlLCBtZWFufSA9IHN0YXRzO1xuXG4gICAgICAgICAgaWYocmVkKSB7XG4gICAgICAgICAgICBtZWFuID0gcmVkWydzdGFuZGFyZCBkZXZpYXRpb24nXSB8fCByZWQubWVhbjtcbiAgICAgICAgICAgIHJlZCA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZWQgPSAwO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGlmKGdyZWVuKSB7XG4gICAgICAgICAgICBtZWFuID0gZ3JlZW5bJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGdyZWVuLm1lYW47XG4gICAgICAgICAgICBncmVlbiA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBncmVlbiA9IDA7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYoYmx1ZSkge1xuICAgICAgICAgICAgbWVhbiA9IGJsdWVbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGJsdWUubWVhbjtcbiAgICAgICAgICAgIGJsdWUgPSBtZWFuID8gKygobWVhbi5zcGxpdCgnICcpWzBdKS5zdWJzdHJpbmcoMCwgMykpIDogMDtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgYmx1ZSA9IDA7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgcmdiID0gYmx1ZSB8IChncmVlbiA8PCA4KSB8IChyZWQgPDwgMTYpO1xuICAgICAgICAgIGNvbnN0IHJnYkNvbG9yID0gcmdiLnRvU3RyaW5nKDE2KTtcbiAgICAgICAgICBjb2xvciA9IHJnYkNvbG9yID09PSAnMCcgPyAnMDAwMDAwJyA6IHJnYkNvbG9yO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3Qge1xuICAgICAgICAgICdKUEVHLVF1YWxpdHknOiBxdWFsaXR5LFxuICAgICAgICAgIE9yaWVudGF0aW9uOiBvcmllbnRhdGlvbkRhdGEsXG4gICAgICAgICAgUmVzb2x1dGlvbjogcmVzb2x1dGlvbkRhdGEsXG4gICAgICAgICAgc2l6ZVxuICAgICAgICB9ID0gaW1hZ2VEYXRhO1xuICAgICAgICBsZXQgcmVzb2x1dGlvbjtcblxuICAgICAgICBpZihyZXNvbHV0aW9uRGF0YSkge1xuICAgICAgICAgIGNvbnN0IFtyZXNvbHV0aW9uTnVtYmVyLCByZXNvbHV0aW9uVW5pdF0gPSByZXNvbHV0aW9uRGF0YS5zcGxpdCgnICcpO1xuICAgICAgICAgIGNvbnN0IHJlc29sdXRpb25WYWx1ZSA9IHJlc29sdXRpb25OdW1iZXIuc3BsaXQoJ3gnKVswXTtcbiAgICAgICAgICBjb25zdCByZXNvbHV0aW9uVW5pdFZhbHVlID0gcmVzb2x1dGlvblVuaXQgPT09ICdwaXhlbHMvaW5jaCcgPyAncHBpJyA6ICcnO1xuICAgICAgICAgIHJlc29sdXRpb24gPSBgJHtyZXNvbHV0aW9uVmFsdWV9ICR7cmVzb2x1dGlvblVuaXRWYWx1ZX1gO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHJlc29sdmUoe1xuICAgICAgICAgIGNvbG9yLFxuICAgICAgICAgIGZpbGVTaXplOiBpbWFnZU9wdGltaXplZEJ1ZmZlci5sZW5ndGgsXG4gICAgICAgICAgaGVpZ2h0OiBzaXplPy5oZWlnaHQgfHwgMCxcbiAgICAgICAgICBtYWtlOiBjYW1lcmFNYWtlLFxuICAgICAgICAgIG1vZGVsOiBjYW1lcmFNb2RlbCxcbiAgICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb25EYXRhID09PSAnVW5rbm93bicgPyB1bmRlZmluZWQgOiBvcmllbnRhdGlvbkRhdGE/LnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgcXVhbGl0eTogcXVhbGl0eSA/ICtxdWFsaXR5IDogdW5kZWZpbmVkLFxuICAgICAgICAgIHJlc29sdXRpb24sXG4gICAgICAgICAgdGFrZW46IHRha2VuID8gRGF0ZVRpbWUuZnJvbU1pbGxpcyh0YWtlbikubWlsbGlzZWNvbmQgOiB1bmRlZmluZWQsXG4gICAgICAgICAgd2lkdGg6IHNpemU/LndpZHRoIHx8IDBcbiAgICAgICAgfSk7XG4gICAgICB9KTtcbiAgfSk7XG5cbiAgY29uc3QgdGh1bWJPcHRpbWl6ZWRCdWZmZXI6IEJ1ZmZlciA9IGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICBjb25zdCB0aG1XID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtU2l6ZScpO1xuICAgIGNvbnN0IHRobVEgPSBDb25maWcuZ2V0KCdpbWFnZS50aG1RdWFsaXR5Jyk7XG5cbiAgICBnbShpbWFnZU9wdGltaXplZEJ1ZmZlciwgJ2ltZycpXG4gICAgICAuc2V0Rm9ybWF0KGZvcm1hdClcbiAgICAgIC5ncmF2aXR5KCdDZW50ZXInKVxuICAgICAgLnJlc2l6ZSh0aG1XLCB0aG1XLCAnXicpXG4gICAgICAuZXh0ZW50KHRobVcsIHRobVcpXG4gICAgICAucXVhbGl0eSh0aG1RKVxuICAgICAgLnN0cmVhbSgoc3RyZWFtRXJyb3I6IEVycm9yLCB0aHVtYlN0ZG91dCk6IHZvaWQgPT4ge1xuICAgICAgICBpZihzdHJlYW1FcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgcGFyYW1zOiB7aW1hZ2VJZCwgZmlsZVR5cGV9LFxuICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgICAgIH0sIHN0cmVhbUVycm9yLCBjb250ZXh0KTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBsZXQgdGh1bWJCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgdGh1bWJCdWZmZXIgPSBCdWZmZXIuY29uY2F0KFt0aHVtYkJ1ZmZlciwgZGF0YV0pO1xuICAgICAgICB9KTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZW5kJywgKCkgPT4gcmVzb2x2ZSh0aHVtYkJ1ZmZlcikpO1xuICAgICAgfSk7XG4gIH0pO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgdGh1bWJPYmo6IFB1dE9iamVjdENvbW1hbmRJbnB1dCA9IHtcbiAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICBCb2R5OiB0aHVtYk9wdGltaXplZEJ1ZmZlcixcbiAgICAgIEJ1Y2tldDogbnVsbCxcbiAgICAgIENvbnRlbnRUeXBlOiBmaWxlVHlwZSxcbiAgICAgIC4uLihzM09wdGlvbnMgfHwge30pLFxuICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzZXNzaW9uSWQsIGltYWdlSWQsIGZpbGVUeXBlLCAndGh1bWJzJylcbiAgICB9O1xuICAgIGF3YWl0IHMzUHV0KHRodW1iT2JqKTtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gIH1cblxuICByZXR1cm4gaW1hZ2VJZGVudGl0eTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRJbWFnZSA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgaW1hZ2U6IEltYWdlVHlwZSxcbiAgczNPcHRpb25zPzogUHV0T2JqZWN0Q29tbWFuZElucHV0XG4pOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7aW1hZ2VJZCwgZGVzY3JpcHRpb24sIGJ1ZmZlciwgZmlsZVR5cGV9ID0gaW1hZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICByZXR1cm4gcmVzaXplU2F2ZUltYWdlKGNvbnRleHQsIGltYWdlSWQsIGJ1ZmZlciwgZmlsZVR5cGUsIHMzT3B0aW9ucylcbiAgICAudGhlbigocmVzaXplZEltYWdlOiBhbnkpID0+IHtcbiAgICAgIGNvbnN0IGluc2VydDogSW1hZ2VUeXBlID0ge1xuICAgICAgICAuLi5yZXNpemVkSW1hZ2UsXG4gICAgICAgIF9rZXk6IGltYWdlSWQsXG4gICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBmaWxlVHlwZSxcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICBwYXJhbXM6IHtpbWFnZSwgczNPcHRpb25zfSxcbiAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfUkVTSVpFXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRJbWFnZUVkZ2UgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgaW1hZ2VFZGdlOiBJbWFnZUVkZ2VUeXBlKTogUHJvbWlzZTxvYmplY3Q+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkSW1hZ2VFZGdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtpbWFnZUlkLCBpdGVtSWQsIGl0ZW1UeXBlfSA9IGltYWdlRWRnZTtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBlZGdlQ29sbGVjdGlvbjogRWRnZUNvbGxlY3Rpb24gPSBkYXRhYmFzZS5jb2xsZWN0aW9uKCdoYXNJbWFnZScpO1xuICBjb25zdCBlZGdlSWQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYGhhc0ltYWdlLSR7aW1hZ2VJZH0tJHtpdGVtSWR9LSR7c2Vzc2lvbklkfWApO1xuICBjb25zdCBmb3JtYXRJdGVtVHlwZTogQXJhbmdvRGJDb2xsZWN0aW9uID0gcGFyc2VDaGFyKGl0ZW1UeXBlKS50b0xvd2VyQ2FzZSgpIGFzIEFyYW5nb0RiQ29sbGVjdGlvbjtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGl0ZW1Eb2NJZDogc3RyaW5nID0gZ2V0RG9jSWQoZm9ybWF0SXRlbVR5cGUsIHtfa2V5OiBmb3JtYXRJdGVtSWR9KTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gcGFyc2VJZChpbWFnZUlkKTtcbiAgY29uc3QgZmlsZURvY0lkOiBzdHJpbmcgPSBgaW1hZ2VzLyR7Zm9ybWF0SW1hZ2VJZH1gO1xuXG4gIGNvbnN0IGVkZ2U6IGFueSA9IHtcbiAgICBfZnJvbTogaXRlbURvY0lkLFxuICAgIF9rZXk6IGVkZ2VJZCxcbiAgICBfdG86IGZpbGVEb2NJZCxcbiAgICBhZGRlZDogbm93LFxuICAgIHR5cGU6IGl0ZW1UeXBlXG4gIH07XG5cbiAgaWYoaXRlbURvY0lkKSB7XG4gICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLnNhdmUoZWRnZSwge3JldHVybk5ldzogdHJ1ZX0pXG4gICAgICAudGhlbigoZmlsZUVkZ2UpID0+IGVkZ2VDb2xsZWN0aW9uLmRvY3VtZW50KGZpbGVFZGdlKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGVkZ2UsXG4gICAgICAgICAgZmlsZURvY0lkLFxuICAgICAgICAgIGltYWdlSWQsXG4gICAgICAgICAgaXRlbURvY0lkLFxuICAgICAgICAgIGl0ZW1JZCxcbiAgICAgICAgICBpdGVtVHlwZVxuICAgICAgICB9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZUltYWdlID0gYXN5bmMgKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBpbWFnZTogSW1hZ2VUeXBlLFxuICBzM09wdGlvbnM/OiBQdXRPYmplY3RDb21tYW5kSW5wdXRcbik6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3VwZGF0ZUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtcbiAgICBiYXNlNjQgPSAnJyxcbiAgICBidWZmZXI6IGltYWdlQnVmZmVyLFxuICAgIGRlc2NyaXB0aW9uID0gJycsXG4gICAgaW1hZ2VJZCxcbiAgICBpdGVtSWQsXG4gICAgaXRlbVR5cGUsXG4gICAgZmlsZVR5cGUsXG4gICAgdXJsID0gJydcbiAgfSA9IGltYWdlO1xuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gaW1hZ2VJZCB8fCBjcmVhdGVIYXNoKGBpbWFnZS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBBcmFuZ29EYkNvbGxlY3Rpb24gPSBwYXJzZUNoYXIoaXRlbVR5cGUsIDE2KS50b0xvd2VyQ2FzZSgpIGFzIEFyYW5nb0RiQ29sbGVjdGlvbjtcbiAgY29uc3QgY3VzdG9tUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgZGVzY3JpcHRpb24sXG4gICAgaW1hZ2VJZDogZm9ybWF0SW1hZ2VJZCxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuXG4gIGlmKGJhc2U2NCB8fCBpbWFnZUJ1ZmZlcikge1xuICAgIGxldCBidWZmZXI6IEJ1ZmZlciA9IGltYWdlQnVmZmVyO1xuXG4gICAgaWYoYmFzZTY0KSB7XG4gICAgICBjb25zdCBmb3JtYXRCYXNlNjQ6IHN0cmluZyA9IGJhc2U2NC5zdWJzdHIoYmFzZTY0LmluZGV4T2YoJywnKSArIDEpO1xuICAgICAgYnVmZmVyID0gQnVmZmVyLmZyb20oZm9ybWF0QmFzZTY0LCAnYmFzZTY0Jyk7XG4gICAgfVxuXG4gICAgbGV0IHVwZGF0ZWRGaWxlVHlwZSA9IGZpbGVUeXBlO1xuXG4gICAgaWYoIWZpbGVUeXBlKSB7XG4gICAgICBjb25zdCB7ZmlsZVR5cGVGcm9tQnVmZmVyfSA9IGF3YWl0IGltcG9ydCgnZmlsZS10eXBlJyk7XG4gICAgICBjb25zdCB7bWltZX0gPSBhd2FpdCBmaWxlVHlwZUZyb21CdWZmZXIoYnVmZmVyKTtcbiAgICAgIHVwZGF0ZWRGaWxlVHlwZSA9IG1pbWUgYXMgYW55O1xuICAgIH1cblxuICAgIGNvbnN0IGltZ1BhcmFtczogSW1hZ2VUeXBlID0ge1xuICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgYnVmZmVyLFxuICAgICAgZmlsZVR5cGU6IHVwZGF0ZWRGaWxlVHlwZVxuICAgIH07XG5cbiAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpXG4gICAgICAudGhlbigoaW1hZ2U6IEltYWdlVHlwZSkgPT4ge1xuICAgICAgICBpZihmb3JtYXRJdGVtSWQgJiYgZm9ybWF0SXRlbVR5cGUpIHtcbiAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7XG4gICAgICAgICAgICBpbWFnZUlkOiBmb3JtYXRJbWFnZUlkLFxuICAgICAgICAgICAgaXRlbUlkOiBmb3JtYXRJdGVtSWQsXG4gICAgICAgICAgICBpdGVtVHlwZTogZm9ybWF0SXRlbVR5cGVcbiAgICAgICAgICB9O1xuICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBpbWFnZTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG4gIH0gZWxzZSBpZih1cmwgIT09ICcnKSB7XG4gICAgbGV0IGNvbnRlbnRUeXBlOiBNaW1lVHlwZTtcblxuICAgIHJldHVybiBodHRwR2V0KHVybClcbiAgICAgIC50aGVuKChyZXM6IFJlc3BvbnNlKSA9PiB7XG4gICAgICAgIGlmKHJlcy5zdGF0dXMgIT09IDIwMCkge1xuICAgICAgICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTUFHRV9SRVFVRVNUXG4gICAgICAgICAgfSwgY29udGV4dCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb250ZW50VHlwZSA9IHJlcy5oZWFkZXJzLmdldCgnY29udGVudC10eXBlJykgYXMgTWltZVR5cGU7XG5cbiAgICAgICAgcmV0dXJuIHJlcztcbiAgICAgIH0pXG4gICAgICAudGhlbigocmVzKSA9PiByZXMuYnVmZmVyKCkpXG4gICAgICAudGhlbigoYnVmZmVyOiBCdWZmZXIpID0+IHtcbiAgICAgICAgY29uc3QgaW1nUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgICAgIGJ1ZmZlcixcbiAgICAgICAgICBmaWxlVHlwZTogY29udGVudFR5cGVcbiAgICAgICAgfTtcblxuICAgICAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpXG4gICAgICAgICAgLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUpID0+IHtcbiAgICAgICAgICAgIGlmKGZvcm1hdEl0ZW1JZCAmJiBmb3JtYXRJdGVtVHlwZSkge1xuICAgICAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7aW1hZ2VJZDogZm9ybWF0SW1hZ2VJZCwgaXRlbUlkOiBmb3JtYXRJdGVtSWQsIGl0ZW1UeXBlOiBmb3JtYXRJdGVtVHlwZX07XG4gICAgICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgcmV0dXJuIGltYWdlO1xuICAgICAgICAgIH0pO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHZhbHVlOiAnZmV0Y2hfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICB9IGVsc2UgaWYoaW1hZ2VJZCAhPT0gJycpIHtcbiAgICBjb25zdCB1cGRhdGU6IGFueSA9IHtcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgbW9kaWZpZWQ6IG5vd1xuICAgIH07XG4gICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUge19rZXk6ICR7Zm9ybWF0SW1hZ2VJZH19IFdJVEggJHt1cGRhdGV9IElOIGltYWdlcyBSRVRVUk4gTkVXYDtcblxuICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGFxbFFyeSxcbiAgICAgICAgICBmb3JtYXRJbWFnZUlkLFxuICAgICAgICAgIHVwZGF0ZVxuICAgICAgICB9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUltYWdlID0gYXN5bmMgKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltYWdlSWQ6IHN0cmluZyk6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2RlbGV0ZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRJbWFnZUlkID0gcGFyc2VJZChpbWFnZUlkKTtcblxuICB0cnkge1xuICAgIGNvbnN0IHJlbW92ZUVkZ2VRdWVyeSA9IGFxbGBGT1IgaGkgSU4gaGFzSW1hZ2VcbiAgICBGSUxURVIgaGkuX2Zyb20gPT0gJHtmb3JtYXRJbWFnZUlkfVxuICAgIFJFTU9WRSBoaSBJTiBoYXNJbWFnZVxuICAgIFJFVFVSTiBPTERgO1xuXG4gICAgYXdhaXQgZGF0YWJhc2UucXVlcnkocmVtb3ZlRWRnZVF1ZXJ5KTtcblxuICAgIGNvbnN0IGFxbFF1ZXJ5ID0gYXFsYEZPUiBpIElOIGltYWdlc1xuICAgIEZJTFRFUiBpLl9rZXkgPT0gJHtmb3JtYXRJbWFnZUlkfSAmJiBpLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICBSRU1PVkUgaSBJTiBpbWFnZXNcbiAgICBSRVRVUk4gT0xEYDtcblxuICAgIGNvbnN0IGltYWdlID0gYXdhaXQgZGF0YWJhc2UucXVlcnkoYXFsUXVlcnkpLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpO1xuXG4gICAgaWYoaW1hZ2UpIHtcbiAgICAgIGNvbnN0IHtfa2V5OiBpbWFnZUtleX0gPSBpbWFnZTtcbiAgICAgIGNvbnN0IHBhcmFtczogRGVsZXRlT2JqZWN0c0NvbW1hbmRJbnB1dCA9IHtcbiAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICBEZWxldGU6IHtcbiAgICAgICAgICBPYmplY3RzOiBbXG4gICAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L2ltYWdlcy8ke2ltYWdlS2V5fS5qcGdgfSxcbiAgICAgICAgICAgIHtLZXk6IGB1c2Vycy8ke3Nlc3Npb25JZH0vdGh1bWJzLyR7aW1hZ2VLZXl9LmpwZ2B9XG4gICAgICAgICAgXSxcbiAgICAgICAgICBRdWlldDogdHJ1ZVxuICAgICAgICB9XG4gICAgICB9O1xuXG4gICAgICByZXR1cm4gczNEZWxldGVMaXN0KHBhcmFtcykudGhlbigoKSA9PiBpbWFnZSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIG51bGw7XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBwYXJhbXM6IHtpbWFnZUlkfSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuICB9XG59O1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FB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
65
+ RETURN OLD`,g=await N(r).query(u).then(i=>i.next());if(g){const{_key:i}=g,p={Bucket:null,Delete:{Objects:[{Key:`users/${o}/images/${i}.jpg`},{Key:`users/${o}/thumbs/${i}.jpg`}],Quiet:!0}};return re(p).then(()=>g)}return null}catch(a){return R({action:s,category:f,params:{imageId:t},value:E.DATABASE_ERROR},a,e)}};export{x as addImage,V as addImageEdge,De as deleteImage,$e as getAppImageUrl,Oe as getImage,he as getImageCountByItem,W as getImageOptional,Be as getImageUrl,Ae as getImagesByGroup,be as getImagesByItem,Ne as getImagesByReactions,Te as getImagesByUser,M as getPathUserImages,H as parseImageOptions,ie as resizeSaveImage,Ce as updateImage};
66
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW1hZ2VzLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtEZWxldGVPYmplY3RzQ29tbWFuZElucHV0LCBQdXRPYmplY3RDb21tYW5kSW5wdXR9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1zMyc7XG5pbXBvcnQge2dldCBhcyBodHRwR2V0fSBmcm9tICdAbmxhYnMvcmlwLWh1bnRlcic7XG5pbXBvcnQge2NyZWF0ZUhhc2gsIGxvd2VyQ2FzZUtleXMsIHBhcnNlQXJhbmdvSWQsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW19IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbH0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvYXFsJztcbmltcG9ydCBnbSBmcm9tICdnbSc7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5cbmltcG9ydCB7Z2V0R3JvdXBEZXRhaWxzLCBpc0dyb3VwZWR9IGZyb20gJy4vZ3JvdXBzJztcbmltcG9ydCB7czNEZWxldGVMaXN0LCBzM0dldFNpZ25lZFVybCwgczNQdXR9IGZyb20gJy4vczMnO1xuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge0Vycm9yVHlwZXN9IGZyb20gJy4uL3R5cGVzL2Vycm9yLnR5cGVzJztcbmltcG9ydCB7bG9nRXJyb3IsIGxvZ0V4Y2VwdGlvbn0gZnJvbSAnLi4vdXRpbHMvYW5hbHl0aWNzVXRpbHMnO1xuaW1wb3J0IHtnZXREb2NJZCwgZ2V0TGltaXQsIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGUsIHVzZURifSBmcm9tICcuLi91dGlscy9hcmFuZ29kYlV0aWxzJztcblxuaW1wb3J0IHR5cGUge1xuICBBcGlDb250ZXh0LFxuICBBcmFuZ29EYkNvbGxlY3Rpb24sXG4gIEFyYW5nb0RiTGltaXQsXG4gIEdyb3VwVHlwZSxcbiAgR3JvdXBVc2VyLFxuICBJbWFnZUVkZ2VUeXBlLFxuICBJbWFnZUlkZW50aWZ5VHlwZSxcbiAgSW1hZ2VPcHRpb25zLFxuICBJbWFnZVR5cGUsXG4gIEltYWdlVXJsRGF0YSxcbiAgUXVlcnlGaWx0ZXJcbn0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHR5cGUge0VkZ2VDb2xsZWN0aW9ufSBmcm9tICdhcmFuZ29qcy9jb2xsZWN0aW9ucyc7XG5cbi8vIERlZmluZSBpbnRlcmZhY2UgZm9yIEhUVFAgcmVzcG9uc2Ugd2l0aCBidWZmZXIgbWV0aG9kXG5pbnRlcmZhY2UgSHR0cFJlc3BvbnNlIGV4dGVuZHMgUmVzcG9uc2Uge1xuICBidWZmZXIoKTogUHJvbWlzZTxCdWZmZXI+O1xufVxuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnaW1hZ2VzJztcbmdtLnN1YkNsYXNzKHtpbWFnZU1hZ2ljazogJzcrJ30pO1xuXG5leHBvcnQgY29uc3QgcGFyc2VJbWFnZU9wdGlvbnMgPSAob3B0aW9uczogSW1hZ2VPcHRpb25zID0ge30pID0+IHtcbiAgY29uc3Qge1xuICAgIGZyb20gPSAwLFxuICAgIHRvID0gMzAsXG4gICAgdHlwZSA9ICdkZWZhdWx0J1xuICB9ID0gb3B0aW9ucztcblxuICByZXR1cm4ge1xuICAgIGxpbWl0OiBnZXRMaW1pdChmcm9tLCB0bykgYXMgQXJhbmdvRGJMaW1pdCxcbiAgICB0eXBlOiBwYXJzZUNoYXIodHlwZSwgMzIpXG4gIH07XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VPcHRpb25hbCA9IChmaWVsZHM6IHN0cmluZ1tdID0gW10pID0+XG4gIGZpZWxkcy5yZWR1Y2UoKHNlbGVjdHMsIGZpZWxkOiBzdHJpbmcpID0+IHtcbiAgICBpZihmaWVsZC5pbmNsdWRlcygnQ291bnQnKSkge1xuICAgICAgcmV0dXJuIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGUoJ2ltYWdlcycsICdpJywgZmllbGQsIHNlbGVjdHMpO1xuICAgIH1cblxuICAgIHN3aXRjaChmaWVsZCkge1xuICAgICAgY2FzZSAncmVhY3Rpb25zJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHJlYWN0aW9ucyA9IChcbiAgICAgICAgICBGT1IgaW1hZ2UsIHIgSU4gSU5CT1VORCBpLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICBDT0xMRUNUIHJlYWN0aW9uTmFtZSA9IHIudmFsdWUgSU5UTyByZWFjdGlvbkl0ZW1zXG4gICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgncmVhY3Rpb25zOnJlYWN0aW9ucycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3RhZ3MnOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgdGFncyA9IChcbiAgICAgICAgICBGT1IgdCwgcGwgSU4gSU5CT1VORCBpLl9pZCBpc1RhZ2dlZFxuICAgICAgICAgIFJFVFVSTiB0XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3RhZ3M6dGFncycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3VzZXJzJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHVzZXJzID0gRklSU1QoXG4gICAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgICBGSUxURVIgaS51c2VySWQgPT0gdS5fa2V5XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiB1XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3VzZXJzOnVzZXJzJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDoge1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICB9XG4gIH0sIHtvYmplY3RzOiBbXSwgcXVlcmllczogW119KTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5VXNlciA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgdXNlcklkOiBzdHJpbmcsXG4gIGZyb206IG51bWJlciA9IDAsXG4gIHRvOiBudW1iZXIgPSAzMFxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdFVzZXJJZDogc3RyaW5nID0gcGFyc2VJZCh1c2VySWQpO1xuICBjb25zdCBsaW1pdDogQXJhbmdvRGJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGkgSU4gaW1hZ2VzXG4gICAgICBGSUxURVIgaS51c2VySWQgPT0gXCIke2Zvcm1hdFVzZXJJZH1cIlxuICAgICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgRklMVEVSIHUuX2tleSA9PSBpLnVzZXJJZFxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiB1XG4gICAgICApXG4gICAgICAke2xpbWl0LmFxbH1cbiAgICAgIFNPUlQgaS5hZGRlZFxuICAgICAgUkVUVVJOIE1FUkdFKGksIHt1c2VyOnVzZXJ9KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5hbGwoKSBhcyB1bmtub3duIGFzIEltYWdlVHlwZVtdKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge2Zyb20sIHRvLCB1c2VySWR9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuXG4gICAgICByZXR1cm4gW107XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VDb3VudEJ5SXRlbSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpdGVtSWQ6IHN0cmluZyk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlQ291bnRCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VBcmFuZ29JZChpdGVtSWQpO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBpIElOIGhhc0ltYWdlXG4gICAgICBGSUxURVIgaS5fdG8gPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgICBSRVRVUk4ge2NvdW50OiBDT1VOVChpKX1gO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh7Y291bnR9KSA9PiBjb3VudClcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBwYXJhbXM6IHtpdGVtSWR9LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5SXRlbSA9IGFzeW5jIChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgaXRlbUlkOiBzdHJpbmcsXG4gIG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9XG4pOiBQcm9taXNlPEltYWdlVHlwZVtdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlc0J5SXRlbSc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIGZpZWxkcyA9IFtdfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VBcmFuZ29JZChpdGVtSWQpO1xuICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldEltYWdlT3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGksIGwgSU4gMS4uMSBPVVRCT1VORCBcIiR7Zm9ybWF0SXRlbUlkfVwiIGhhc0ltYWdlXG4gICAgRklMVEVSIE5PVCBJU19OVUxMKGkpXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIFNPUlQgaS5hZGRlZFxuICAgICR7bGltaXQuYXFsfVxuICAgIFJFVFVSTiBNRVJHRShpLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkgYXMgdW5rbm93biBhcyBJbWFnZVR5cGVbXSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICBwYXJhbXM6IHtpdGVtSWQsIG9wdGlvbnN9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuICAgICAgcmV0dXJuIFtdO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5R3JvdXAgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHBhcmFtczoge2ZpbHRlcnM6IFF1ZXJ5RmlsdGVyW10sIGdyb3VwSWQ6IHN0cmluZywgZnJvbTogbnVtYmVyLCB0bzogbnVtYmVyfVxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeUdyb3VwJztcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7ZmlsdGVycywgZ3JvdXBJZCwgZnJvbSwgdG99ID0gcGFyYW1zO1xuICBjb25zdCBmb3JtYXRHcm91cElkOiBzdHJpbmcgPSBwYXJzZUlkKGdyb3VwSWQpO1xuICBjb25zdCBsaW1pdDogQXJhbmdvRGJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcblxuICBmaWx0ZXJzXG4gICAgLm1hcCgoZmlsdGVyOiBRdWVyeUZpbHRlcikgPT4ge1xuICAgICAgY29uc3Qge2NvbmRpdGlvbmFsLCBuYW1lLCB2YWx1ZX0gPSBmaWx0ZXI7XG4gICAgICBsZXQgZm9ybWF0Q29uZDogc3RyaW5nID0gY29uZGl0aW9uYWw7XG5cbiAgICAgIGlmKGNvbmRpdGlvbmFsICE9PSAnPj0nICYmIGNvbmRpdGlvbmFsICE9PSAnPD0nICYmIGNvbmRpdGlvbmFsICE9PSAnPicgJiYgY29uZGl0aW9uYWwgIT09ICc8Jykge1xuICAgICAgICBmb3JtYXRDb25kID0gJz09JztcbiAgICAgIH1cblxuICAgICAgc3dpdGNoKG5hbWUpIHtcbiAgICAgICAgY2FzZSAnYWRkZWQnOlxuICAgICAgICAgIHJldHVybiBgcC5hZGRlZCAke2Zvcm1hdENvbmR9ICR7cGFyc2VOdW0odmFsdWUpfWA7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgcmV0dXJuICcnO1xuICAgICAgfVxuICAgIH0pO1xuXG4gIHJldHVybiBnZXRHcm91cERldGFpbHMoY29udGV4dCwgZm9ybWF0R3JvdXBJZClcbiAgICAudGhlbigoZ3JvdXA6IEdyb3VwVHlwZSkgPT4ge1xuICAgICAgaWYoZ3JvdXAucHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgZmlsdGVycy5wdXNoKHtjb25kaXRpb25hbDogJz09JywgbmFtZTogJ2dyb3VwSWQnLCB2YWx1ZTogZm9ybWF0R3JvdXBJZH0pO1xuICAgICAgICBjb25zdCBmaWx0ZXJTdHIgPSBmaWx0ZXJzLmpvaW4oJyAmJiAnKTtcbiAgICAgICAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGkgSU5cbiAgICAgICAgICBGTEFUVEVOKFxuICAgICAgICAgICAgRk9SIHAgSU4gcG9zdHNcbiAgICAgICAgICAgIEZJTFRFUiAke2ZpbHRlclN0cn1cbiAgICAgICAgICAgIExFVCBpbWFnZXMgPSAoXG4gICAgICAgICAgICAgIEZPUiBpLCBlIElOIElOQk9VTkQgcC5faWQgaGFzSW1hZ2VcbiAgICAgICAgICAgICAgUkVUVVJOIGlcbiAgICAgICAgICAgIClcbiAgICAgICAgICAgIFNPUlQgcC5hZGRlZCBERVNDXG4gICAgICAgICAgICBSRVRVUk4gaW1hZ2VzXG4gICAgICAgICAgKVxuICAgICAgICAgIFNPUlQgaS5hZGRlZCBERVNDXG4gICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgUkVUVVJOIGlgO1xuXG4gICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkgYXMgdW5rbm93biBhcyBJbWFnZVR5cGVbXSlcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICBwYXJhbXM6IHtmaWx0ZXJzLCBmcm9tLCBncm91cElkLCB0b30sXG4gICAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gICAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgICAgfSk7XG4gICAgICB9XG4gICAgICByZXR1cm4gaXNHcm91cGVkKGNvbnRleHQsIHNlc3Npb25JZCwgZ3JvdXBJZClcbiAgICAgICAgLnRoZW4oKGdyb3VwZWQ6IEdyb3VwVXNlcikgPT4ge1xuICAgICAgICAgIGlmKGdyb3VwZWQuaXNWYWxpZCkge1xuICAgICAgICAgICAgZmlsdGVycy5wdXNoKHtjb25kaXRpb25hbDogJz09JywgbmFtZTogJ2dyb3VwSWQnLCB2YWx1ZTogZ3JvdXBlZC5ncm91cElkfSk7XG4gICAgICAgICAgICBjb25zdCBmaWx0ZXJMaXN0OiBzdHJpbmcgPSBmaWx0ZXJzLmpvaW4oJyAmJiAnKTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RcbiAgICAgICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJMaXN0fVxuICAgICAgICAgICAgICAgIEZPUiBmIElOIHAuZmlsZXNcbiAgICAgICAgICAgICAgICBGSUxURVIgZi50eXBlID09IFwiaW1hZ2UvanBlZ1wiIHx8IGYudHlwZSA9PSBcImltYWdlL3BuZ1wiXG4gICAgICAgICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgICAgICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICAgICAgICAgICAgICBSRVRVUk4gZmA7XG5cbiAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLmFsbCgpIGFzIHVua25vd24gYXMgSW1hZ2VUeXBlW10pXG4gICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgICBwYXJhbXM6IHtmaWx0ZXJzLCBmcm9tLCBncm91cElkLCB0b30sXG4gICAgICAgICAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KTtcbiAgICAgICAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5UmVhY3Rpb25zID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICByZWFjdGlvbnM6IHN0cmluZ1tdID0gW10sXG4gIG9wdGlvbnM/OiBJbWFnZU9wdGlvbnNcbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0VXNlcnNCeUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgZmllbGRzID0gW10sIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0fSA9IHBhcnNlSW1hZ2VPcHRpb25zKG9wdGlvbnMpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRJbWFnZU9wdGlvbmFsKGZpZWxkcyk7XG4gIGNvbnN0IGZvcm1hdFNlc3Npb25JZDogc3RyaW5nID0gYHVzZXJzLyR7c2Vzc2lvbklkfWA7XG4gIGNvbnN0IGZvcm1hdFJlYWN0aW9uczogc3RyaW5nW10gPSByZWFjdGlvbnMubWFwKChyZWFjdGlvbk5hbWUpID0+IHBhcnNlQ2hhcihyZWFjdGlvbk5hbWUsIDMyKSk7XG4gIGNvbnN0IGZpbHRlckJ5OiBzdHJpbmdbXSA9IFtcbiAgICAnci50eXBlID09IFwiaW1hZ2VzXCInLFxuICAgIGBQT1NJVElPTigke0pTT04uc3RyaW5naWZ5KGZvcm1hdFJlYWN0aW9ucyl9LCBMT1dFUihyLm5hbWUpKWBdO1xuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGksIHIgSU4gT1VUQk9VTkQgXCIke2Zvcm1hdFNlc3Npb25JZH1cIiBoYXNSZWFjdGlvblxuICAgIE9QVElPTlMge3ZlcnRleENvbGxlY3Rpb25zOiBcImltYWdlc1wifVxuICAgICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgICBGSUxURVIgJHtmaWx0ZXJCeS5qb2luKCcgJiYgJyl9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgUkVUVVJOIE1FUkdFKGksIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5hbGwoKSBhcyB1bmtub3duIGFzIEltYWdlVHlwZVtdKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge29wdGlvbnMsIHJlYWN0aW9uc30sXG4gICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gICAgICByZXR1cm4gW107XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaW1hZ2VJZDogc3RyaW5nKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0SXRlbSc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWV9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gcGFyc2VJZChpbWFnZUlkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5PT0ke2Zvcm1hdEltYWdlSWR9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBpYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBwYXJhbXM6IHtpbWFnZUlkfSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQYXRoVXNlckltYWdlcyA9ICh1c2VySWQ6IHN0cmluZywgaW1hZ2VJZDogc3RyaW5nLCBpbWFnZVR5cGU6IHN0cmluZywgZGlyOiBzdHJpbmcgPSAnaW1hZ2VzJyk6IHN0cmluZyA9PiB7XG4gIGxldCBmaWxlbmFtZTogc3RyaW5nID0gaW1hZ2VJZDtcblxuICBzd2l0Y2goaW1hZ2VUeXBlKSB7XG4gICAgY2FzZSAnaW1hZ2UvcG5nJzpcbiAgICAgIGZpbGVuYW1lID0gYCR7aW1hZ2VJZH0ucG5nYDtcbiAgICAgIGJyZWFrO1xuICAgIGRlZmF1bHQ6XG4gICAgICBmaWxlbmFtZSA9IGAke2ltYWdlSWR9LmpwZ2A7XG4gICAgICBicmVhaztcbiAgfVxuXG4gIHJldHVybiBgdXNlcnMvJHt1c2VySWR9LyR7ZGlyfS8ke2ZpbGVuYW1lfWA7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0QXBwSW1hZ2VVcmwgPSAoe1xuICBidWNrZXQsXG4gIGRpcmVjdG9yeSA9ICdpbWFnZXMnLFxuICBpbWFnZUlkLFxuICBpbWFnZVR5cGUgPSAnanBnJyxcbiAgaXNUaHVtYiA9IGZhbHNlLFxuICB0eXBlLFxuICB0eXBlSWRcbn06IEltYWdlVXJsRGF0YSk6IHN0cmluZyA9PiB7XG4gIGlmKCF0eXBlKSB7XG4gICAgcmV0dXJuICcnO1xuICB9XG5cbiAgaWYoIWltYWdlSWQpIHtcbiAgICBpZihpbWFnZVR5cGUgPT09ICdwcm9maWxlJykge1xuICAgICAgcmV0dXJuIGAke0NvbmZpZy5nZXQoJ2ltYWdlLmRlZmF1bHRVcmwnKX0vdXNlcnNfYmsuanBnYDtcbiAgICB9XG4gICAgcmV0dXJuIGAke0NvbmZpZy5nZXQoJ2ltYWdlLmRlZmF1bHRVcmwnKX0vdXNlcnNfd2guanBnYDtcbiAgfVxuXG4gIGNvbnN0IGJ1Y2tldE5hbWUgPSBidWNrZXQgfHwgQ29uZmlnLmdldCgnaW1hZ2UuYnVja2V0JykgfHwgJ2Rldi5yZWFrdG9yLmlvJztcbiAgY29uc3QgaW1hZ2VOYW1lID0gaXNUaHVtYiA/IGAke2ltYWdlSWR9LXRoLiR7aW1hZ2VUeXBlfWAgOiBgJHtpbWFnZUlkfS4ke2ltYWdlVHlwZSA9PT0gJ3Byb2ZpbGUnIHx8IGltYWdlVHlwZSA9PT0gJ290aGVyJyA/ICdqcGcnIDogaW1hZ2VUeXBlfWA7XG4gIGNvbnN0IGltYWdlVXJsID0gQ29uZmlnLmdldCgnaW1hZ2UudXJsJykgfHwgYGh0dHBzOi8vczMuYW1hem9uYXdzLmNvbS8ke2J1Y2tldE5hbWV9YDtcblxuICBpZih0eXBlID09PSAnYXBwcycpIHtcbiAgICByZXR1cm4gYCR7aW1hZ2VVcmx9LyR7dHlwZX0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VOYW1lfWA7XG4gIH1cblxuICByZXR1cm4gYCR7aW1hZ2VVcmx9LyR7dHlwZX0vJHt0eXBlSWR9LyR7ZGlyZWN0b3J5fS8ke2ltYWdlTmFtZX1gO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlVXJsID0gYXN5bmMgKHtcbiAgYnVja2V0LFxuICBpbWFnZURpciA9ICdpbWFnZXMnLFxuICBpbWFnZUlkLFxuICBpbWFnZVR5cGUgPSAnanBnJyxcbiAgaXNUaHVtYiA9IGZhbHNlLFxuICB0eXBlLFxuICB0eXBlSWQsXG4gIHVybFR5cGUgPSAncHVibGljJ1xufTogSW1hZ2VVcmxEYXRhKTogUHJvbWlzZTxzdHJpbmc+ID0+IHtcbiAgaWYoIWltYWdlSWQpIHtcbiAgICByZXR1cm4gJyc7XG4gIH1cblxuICAvLyBBbHdheXMgdXNlIHNpZ25lZCBVUkxzIGZvciB0aHVtYm5haWxzXG4gIGlmKGlzVGh1bWIpIHtcbiAgICB1cmxUeXBlID0gJ3NpZ25lZCc7XG4gIH1cblxuICBjb25zdCB0aHVtYlN1ZmZpeCA9IGlzVGh1bWIgPyAnLXRoJyA6ICcnO1xuICBjb25zdCBpbWFnZUtleTogc3RyaW5nID0gYCR7dHlwZX0vJHt0eXBlSWR9LyR7aW1hZ2VEaXJ9LyR7aW1hZ2VJZH0ke3RodW1iU3VmZml4fS4ke2ltYWdlVHlwZX1gO1xuXG4gIHN3aXRjaCh1cmxUeXBlKSB7XG4gICAgY2FzZSAnc2lnbmVkJzoge1xuICAgICAgdHJ5IHtcbiAgICAgICAgY29uc3QgZGVmYXVsdEJ1Y2tldCA9IENvbmZpZy5nZXQoJ2ltYWdlLmJ1Y2tldCcpIHx8ICdkZXYucmVha3Rvci5pbyc7XG4gICAgICAgIHJldHVybiBhd2FpdCBzM0dldFNpZ25lZFVybCh7QnVja2V0OiBidWNrZXQgfHwgZGVmYXVsdEJ1Y2tldCwgRXhwaXJlczogOTAwLCBLZXk6IGltYWdlS2V5fSk7XG4gICAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICAgIHRocm93IGVycm9yO1xuICAgICAgfVxuICAgIH1cbiAgICBjYXNlICdwdWJsaWMnOiB7XG4gICAgICByZXR1cm4gYGh0dHBzOi8vYm94LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfS8ke2ltYWdlS2V5fWA7XG4gICAgfVxuICAgIGNhc2UgJ2Rldic6IHtcbiAgICAgIHJldHVybiBgaHR0cHM6Ly9zMy5hbWF6b25hd3MuY29tL2Rldi4ke0NvbmZpZy5nZXQoJ2FwcC51cmwnKX0vJHtpbWFnZUtleX1gO1xuICAgIH1cbiAgICBkZWZhdWx0OiB7XG4gICAgICByZXR1cm4gJyc7XG4gICAgfVxuICB9XG59O1xuXG5leHBvcnQgY29uc3QgcmVzaXplU2F2ZUltYWdlID0gYXN5bmMgKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBpbWFnZUlkOiBzdHJpbmcsXG4gIGJ1ZmZlcjogQnVmZmVyLFxuICBmaWxlVHlwZTogc3RyaW5nID0gJ2ltYWdlL2pwZWcnLCBzM09wdGlvbnM/OiBQdXRPYmplY3RDb21tYW5kSW5wdXRcbik6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3Jlc2l6ZVNhdmVJbWFnZSc7XG4gIGNvbnN0IHtzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGltZ1c6IG51bWJlciA9IENvbmZpZy5nZXQoJ2ltYWdlLmltZ1NpemUnKTtcbiAgY29uc3QgaW1nUTogbnVtYmVyID0gQ29uZmlnLmdldCgnaW1hZ2UuaW1nUXVhbGl0eScpO1xuICBjb25zdCBmb3JtYXQ6IHN0cmluZyA9IChmaWxlVHlwZS5zcGxpdCgnLycpKVsxXTtcblxuICBjb25zdCBpbWFnZU9wdGltaXplZEJ1ZmZlcjogQnVmZmVyID0gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIGdtKGJ1ZmZlciwgJ2ltZycpXG4gICAgICAuc2V0Rm9ybWF0KGZvcm1hdClcbiAgICAgIC5xdWFsaXR5KGltZ1EpXG4gICAgICAuYXV0b09yaWVudCgpXG4gICAgICAucmVzaXplKGltZ1csIGltZ1csICc+JylcbiAgICAgIC5zdHJlYW0oKHN0cmVhbUVycm9yOiBFcnJvciwgc3Rkb3V0KTogdm9pZCA9PiB7XG4gICAgICAgIGlmKHN0cmVhbUVycm9yKSB7XG4gICAgICAgICAgcmVqZWN0KFxuICAgICAgICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICBwYXJhbXM6IHtmaWxlVHlwZSwgaW1hZ2VJZH0sXG4gICAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICAgICAgICAgIH0sIHN0cmVhbUVycm9yLCBjb250ZXh0KVxuICAgICAgICAgICk7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgbGV0IGltYWdlQnVmZmVyOiBCdWZmZXIgPSBCdWZmZXIuZnJvbSgnJyk7XG5cbiAgICAgICAgc3Rkb3V0Lm9uKCdkYXRhJywgKGRhdGEpID0+IHtcbiAgICAgICAgICBpbWFnZUJ1ZmZlciA9IEJ1ZmZlci5jb25jYXQoW2ltYWdlQnVmZmVyLCBkYXRhXSk7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIHN0ZG91dC5vbignZW5kJywgKCkgPT4gcmVzb2x2ZShpbWFnZUJ1ZmZlcikpO1xuICAgICAgfSk7XG4gIH0pO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgaW1hZ2VPYmo6IFB1dE9iamVjdENvbW1hbmRJbnB1dCA9IHtcbiAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICBCb2R5OiBpbWFnZU9wdGltaXplZEJ1ZmZlcixcbiAgICAgIEJ1Y2tldDogbnVsbCxcbiAgICAgIENvbnRlbnRUeXBlOiBmaWxlVHlwZSxcbiAgICAgIC4uLihzM09wdGlvbnMgfHwge30pLFxuICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzZXNzaW9uSWQsIGltYWdlSWQsIGZpbGVUeXBlLCAnaW1hZ2VzJylcbiAgICB9O1xuICAgIGF3YWl0IHMzUHV0KGltYWdlT2JqKTtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHBhcmFtczoge2ZpbGVUeXBlLCBpbWFnZUlkfSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gIH1cblxuICBjb25zdCBpbWFnZUlkZW50aXR5ID0gYXdhaXQgbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICBnbShpbWFnZU9wdGltaXplZEJ1ZmZlciwgJ2ltZycpXG4gICAgICAuaWRlbnRpZnkoe2J1ZmZlclN0cmVhbTogdHJ1ZX0sIChlcnJvcjogRXJyb3IsIGltYWdlRGF0YTogSW1hZ2VJZGVudGlmeVR5cGUgPSB7fSkgPT4ge1xuICAgICAgICBpZihlcnJvcikge1xuICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBmb3JtYXRWYWx1ZXMgPSBsb3dlckNhc2VLZXlzKGltYWdlRGF0YSk7XG4gICAgICAgIGNvbnN0IHttYWtlOiBjYW1lcmFNYWtlLCBtb2RlbDogY2FtZXJhTW9kZWwsIGRhdGV0aW1lb3JpZ2luYWw6IHRha2VufTogSW1hZ2VJZGVudGlmeVR5cGUgPSBmb3JtYXRWYWx1ZXM7XG4gICAgICAgIGNvbnN0IHN0YXRzID0gZm9ybWF0VmFsdWVzWydjaGFubmVsIHN0YXRpc3RpY3MnXTtcbiAgICAgICAgbGV0IGNvbG9yOiBzdHJpbmc7XG5cbiAgICAgICAgaWYoc3RhdHMpIHtcbiAgICAgICAgICBsZXQge3JlZCwgZ3JlZW4sIGJsdWUsIG1lYW59ID0gc3RhdHM7XG5cbiAgICAgICAgICBpZihyZWQpIHtcbiAgICAgICAgICAgIG1lYW4gPSByZWRbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IHJlZC5tZWFuO1xuICAgICAgICAgICAgcmVkID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlZCA9IDA7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYoZ3JlZW4pIHtcbiAgICAgICAgICAgIG1lYW4gPSBncmVlblsnc3RhbmRhcmQgZGV2aWF0aW9uJ10gfHwgZ3JlZW4ubWVhbjtcbiAgICAgICAgICAgIGdyZWVuID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGdyZWVuID0gMDtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBpZihibHVlKSB7XG4gICAgICAgICAgICBtZWFuID0gYmx1ZVsnc3RhbmRhcmQgZGV2aWF0aW9uJ10gfHwgYmx1ZS5tZWFuO1xuICAgICAgICAgICAgYmx1ZSA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBibHVlID0gMDtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCByZ2IgPSBibHVlIHwgKGdyZWVuIDw8IDgpIHwgKHJlZCA8PCAxNik7XG4gICAgICAgICAgY29uc3QgcmdiQ29sb3IgPSByZ2IudG9TdHJpbmcoMTYpO1xuICAgICAgICAgIGNvbG9yID0gcmdiQ29sb3IgPT09ICcwJyA/ICcwMDAwMDAnIDogcmdiQ29sb3I7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCB7XG4gICAgICAgICAgJ0pQRUctUXVhbGl0eSc6IHF1YWxpdHksXG4gICAgICAgICAgT3JpZW50YXRpb246IG9yaWVudGF0aW9uRGF0YSxcbiAgICAgICAgICBSZXNvbHV0aW9uOiByZXNvbHV0aW9uRGF0YSxcbiAgICAgICAgICBzaXplXG4gICAgICAgIH0gPSBpbWFnZURhdGE7XG4gICAgICAgIGxldCByZXNvbHV0aW9uO1xuXG4gICAgICAgIGlmKHJlc29sdXRpb25EYXRhKSB7XG4gICAgICAgICAgY29uc3QgW3Jlc29sdXRpb25OdW1iZXIsIHJlc29sdXRpb25Vbml0XSA9IHJlc29sdXRpb25EYXRhLnNwbGl0KCcgJyk7XG4gICAgICAgICAgY29uc3QgcmVzb2x1dGlvblZhbHVlID0gcmVzb2x1dGlvbk51bWJlci5zcGxpdCgneCcpWzBdO1xuICAgICAgICAgIGNvbnN0IHJlc29sdXRpb25Vbml0VmFsdWUgPSByZXNvbHV0aW9uVW5pdCA9PT0gJ3BpeGVscy9pbmNoJyA/ICdwcGknIDogJyc7XG4gICAgICAgICAgcmVzb2x1dGlvbiA9IGAke3Jlc29sdXRpb25WYWx1ZX0gJHtyZXNvbHV0aW9uVW5pdFZhbHVlfWA7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gcmVzb2x2ZSh7XG4gICAgICAgICAgY29sb3IsXG4gICAgICAgICAgZmlsZVNpemU6IGltYWdlT3B0aW1pemVkQnVmZmVyLmxlbmd0aCxcbiAgICAgICAgICBmaWxlVHlwZTogZm9ybWF0LFxuICAgICAgICAgIGhlaWdodDogc2l6ZT8uaGVpZ2h0IHx8IDAsXG4gICAgICAgICAgaW1hZ2VJZCxcbiAgICAgICAgICBtYWtlOiBjYW1lcmFNYWtlLFxuICAgICAgICAgIG1vZGVsOiBjYW1lcmFNb2RlbCxcbiAgICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb25EYXRhID09PSAnVW5rbm93bicgPyB1bmRlZmluZWQgOiBvcmllbnRhdGlvbkRhdGE/LnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgcXVhbGl0eTogcXVhbGl0eSA/ICtxdWFsaXR5IDogdW5kZWZpbmVkLFxuICAgICAgICAgIHJlc29sdXRpb24sXG4gICAgICAgICAgdGFrZW46IHRha2VuID8gRGF0ZVRpbWUuZnJvbU1pbGxpcyh0YWtlbikubWlsbGlzZWNvbmQgOiB1bmRlZmluZWQsXG4gICAgICAgICAgd2lkdGg6IHNpemU/LndpZHRoIHx8IDBcbiAgICAgICAgfSk7XG4gICAgICB9KTtcbiAgfSk7XG5cbiAgY29uc3QgdGh1bWJPcHRpbWl6ZWRCdWZmZXI6IEJ1ZmZlciA9IGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICBjb25zdCB0aG1XID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtU2l6ZScpO1xuICAgIGNvbnN0IHRobVEgPSBDb25maWcuZ2V0KCdpbWFnZS50aG1RdWFsaXR5Jyk7XG5cbiAgICBnbShpbWFnZU9wdGltaXplZEJ1ZmZlciwgJ2ltZycpXG4gICAgICAuc2V0Rm9ybWF0KGZvcm1hdClcbiAgICAgIC5ncmF2aXR5KCdDZW50ZXInKVxuICAgICAgLnJlc2l6ZSh0aG1XLCB0aG1XLCAnXicpXG4gICAgICAuZXh0ZW50KHRobVcsIHRobVcpXG4gICAgICAucXVhbGl0eSh0aG1RKVxuICAgICAgLnN0cmVhbSgoc3RyZWFtRXJyb3I6IEVycm9yLCB0aHVtYlN0ZG91dCk6IHZvaWQgPT4ge1xuICAgICAgICBpZihzdHJlYW1FcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgcGFyYW1zOiB7ZmlsZVR5cGUsIGltYWdlSWR9LFxuICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgICAgIH0sIHN0cmVhbUVycm9yLCBjb250ZXh0KTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBsZXQgdGh1bWJCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgdGh1bWJCdWZmZXIgPSBCdWZmZXIuY29uY2F0KFt0aHVtYkJ1ZmZlciwgZGF0YV0pO1xuICAgICAgICB9KTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZW5kJywgKCkgPT4gcmVzb2x2ZSh0aHVtYkJ1ZmZlcikpO1xuICAgICAgfSk7XG4gIH0pO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgdGh1bWJPYmo6IFB1dE9iamVjdENvbW1hbmRJbnB1dCA9IHtcbiAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICBCb2R5OiB0aHVtYk9wdGltaXplZEJ1ZmZlcixcbiAgICAgIEJ1Y2tldDogbnVsbCxcbiAgICAgIENvbnRlbnRUeXBlOiBmaWxlVHlwZSxcbiAgICAgIC4uLihzM09wdGlvbnMgfHwge30pLFxuICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzZXNzaW9uSWQsIGltYWdlSWQsIGZpbGVUeXBlLCAndGh1bWJzJylcbiAgICB9O1xuICAgIGF3YWl0IHMzUHV0KHRodW1iT2JqKTtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gIH1cblxuICByZXR1cm4gaW1hZ2VJZGVudGl0eTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRJbWFnZSA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgaW1hZ2U6IEltYWdlVHlwZSxcbiAgczNPcHRpb25zPzogUHV0T2JqZWN0Q29tbWFuZElucHV0XG4pOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2ltYWdlSWQsIGRlc2NyaXB0aW9uLCBidWZmZXIsIGZpbGVUeXBlfSA9IGltYWdlO1xuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG5cbiAgcmV0dXJuIHJlc2l6ZVNhdmVJbWFnZShjb250ZXh0LCBpbWFnZUlkLCBidWZmZXIsIGZpbGVUeXBlLCBzM09wdGlvbnMpXG4gICAgLnRoZW4oKHJlc2l6ZWRJbWFnZTogSW1hZ2VUeXBlKSA9PiB7XG4gICAgICBjb25zdCBpbnNlcnQ6IEltYWdlVHlwZSA9IHtcbiAgICAgICAgLi4ucmVzaXplZEltYWdlLFxuICAgICAgICBfa2V5OiBpbWFnZUlkLFxuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgZmlsZVR5cGUsXG4gICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gaW1hZ2VzIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIHBhcmFtczoge2ltYWdlLCBzM09wdGlvbnN9LFxuICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTUFHRV9SRVNJWkVcbiAgICB9LCBlcnJvciwgY29udGV4dCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEltYWdlRWRnZSA9IGFzeW5jIChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUpOiBQcm9taXNlPG9iamVjdD4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZUVkZ2UnO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtpbWFnZUlkLCBpdGVtSWQsIGl0ZW1UeXBlfSA9IGltYWdlRWRnZTtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBlZGdlQ29sbGVjdGlvbjogRWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZU5hbWUpLmNvbGxlY3Rpb24oJ2hhc0ltYWdlJyk7XG4gIGNvbnN0IGVkZ2VJZDogc3RyaW5nID0gY3JlYXRlSGFzaChgaGFzSW1hZ2UtJHtpbWFnZUlkfS0ke2l0ZW1JZH0tJHtzZXNzaW9uSWR9YCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBBcmFuZ29EYkNvbGxlY3Rpb24gPSBwYXJzZUNoYXIoaXRlbVR5cGUpLnRvTG93ZXJDYXNlKCkgYXMgQXJhbmdvRGJDb2xsZWN0aW9uO1xuICBjb25zdCBmb3JtYXRJdGVtSWQ6IHN0cmluZyA9IHBhcnNlSWQoaXRlbUlkKTtcbiAgY29uc3QgaXRlbURvY0lkOiBzdHJpbmcgPSBnZXREb2NJZChmb3JtYXRJdGVtVHlwZSwge19rZXk6IGZvcm1hdEl0ZW1JZH0pO1xuICBjb25zdCBmb3JtYXRJbWFnZUlkOiBzdHJpbmcgPSBwYXJzZUlkKGltYWdlSWQpO1xuICBjb25zdCBmaWxlRG9jSWQ6IHN0cmluZyA9IGBpbWFnZXMvJHtmb3JtYXRJbWFnZUlkfWA7XG5cbiAgY29uc3QgZWRnZTogSW1hZ2VFZGdlVHlwZSA9IHtcbiAgICBfZnJvbTogaXRlbURvY0lkLFxuICAgIF9rZXk6IGVkZ2VJZCxcbiAgICBfdG86IGZpbGVEb2NJZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGl0ZW1UeXBlXG4gIH07XG5cbiAgaWYoaXRlbURvY0lkKSB7XG4gICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkuY29sbGVjdGlvbignaGFzSW1hZ2UnKS5zYXZlKGVkZ2UsIHtyZXR1cm5OZXc6IHRydWV9KVxuICAgICAgLnRoZW4oKGZpbGVFZGdlKSA9PiB1c2VEYihkYXRhYmFzZU5hbWUpLmNvbGxlY3Rpb24oJ2hhc0ltYWdlJykuZG9jdW1lbnQoZmlsZUVkZ2UpKVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgcGFyYW1zOiB7XG4gICAgICAgICAgZWRnZSxcbiAgICAgICAgICBmaWxlRG9jSWQsXG4gICAgICAgICAgaW1hZ2VJZCxcbiAgICAgICAgICBpdGVtRG9jSWQsXG4gICAgICAgICAgaXRlbUlkLFxuICAgICAgICAgIGl0ZW1UeXBlXG4gICAgICAgIH0sXG4gICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICB9XG5cbiAgcmV0dXJuIG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlSW1hZ2UgPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGltYWdlOiBJbWFnZVR5cGUsXG4gIHMzT3B0aW9ucz86IFB1dE9iamVjdENvbW1hbmRJbnB1dFxuKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAndXBkYXRlSW1hZ2UnO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtcbiAgICBiYXNlNjQgPSAnJyxcbiAgICBidWZmZXI6IGltYWdlQnVmZmVyLFxuICAgIGRlc2NyaXB0aW9uID0gJycsXG4gICAgaW1hZ2VJZCxcbiAgICBpdGVtSWQsXG4gICAgaXRlbVR5cGUsXG4gICAgZmlsZVR5cGUsXG4gICAgdXJsID0gJydcbiAgfSA9IGltYWdlO1xuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gaW1hZ2VJZCB8fCBjcmVhdGVIYXNoKGBpbWFnZS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBBcmFuZ29EYkNvbGxlY3Rpb24gPSBwYXJzZUNoYXIoaXRlbVR5cGUsIDE2KS50b0xvd2VyQ2FzZSgpIGFzIEFyYW5nb0RiQ29sbGVjdGlvbjtcbiAgY29uc3QgY3VzdG9tUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgZGVzY3JpcHRpb24sXG4gICAgaW1hZ2VJZDogZm9ybWF0SW1hZ2VJZCxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuXG4gIGlmKGJhc2U2NCB8fCBpbWFnZUJ1ZmZlcikge1xuICAgIGxldCBidWZmZXI6IEJ1ZmZlciA9IGltYWdlQnVmZmVyO1xuXG4gICAgaWYoYmFzZTY0KSB7XG4gICAgICBjb25zdCBmb3JtYXRCYXNlNjQ6IHN0cmluZyA9IGJhc2U2NC5zdWJzdHIoYmFzZTY0LmluZGV4T2YoJywnKSArIDEpO1xuICAgICAgYnVmZmVyID0gQnVmZmVyLmZyb20oZm9ybWF0QmFzZTY0LCAnYmFzZTY0Jyk7XG4gICAgfVxuXG4gICAgbGV0IHVwZGF0ZWRGaWxlVHlwZSA9IGZpbGVUeXBlO1xuXG4gICAgaWYoIWZpbGVUeXBlKSB7XG4gICAgICBjb25zdCB7ZmlsZVR5cGVGcm9tQnVmZmVyfSA9IGF3YWl0IGltcG9ydCgnZmlsZS10eXBlJyk7XG4gICAgICBjb25zdCB7bWltZX0gPSBhd2FpdCBmaWxlVHlwZUZyb21CdWZmZXIoYnVmZmVyKTtcbiAgICAgIHVwZGF0ZWRGaWxlVHlwZSA9IG1pbWU7XG4gICAgfVxuXG4gICAgY29uc3QgaW1nUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgICAuLi5jdXN0b21QYXJhbXMsXG4gICAgICBidWZmZXIsXG4gICAgICBmaWxlVHlwZTogdXBkYXRlZEZpbGVUeXBlXG4gICAgfTtcblxuICAgIHJldHVybiBhZGRJbWFnZShjb250ZXh0LCBpbWdQYXJhbXMsIHMzT3B0aW9ucylcbiAgICAgIC50aGVuKChpbWFnZTogSW1hZ2VUeXBlKSA9PiB7XG4gICAgICAgIGlmKGZvcm1hdEl0ZW1JZCAmJiBmb3JtYXRJdGVtVHlwZSkge1xuICAgICAgICAgIGNvbnN0IGltYWdlRWRnZTogSW1hZ2VFZGdlVHlwZSA9IHtcbiAgICAgICAgICAgIGltYWdlSWQ6IGZvcm1hdEltYWdlSWQsXG4gICAgICAgICAgICBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCxcbiAgICAgICAgICAgIGl0ZW1UeXBlOiBmb3JtYXRJdGVtVHlwZVxuICAgICAgICAgIH07XG4gICAgICAgICAgcmV0dXJuIGFkZEltYWdlRWRnZShjb250ZXh0LCBpbWFnZUVkZ2UpLnRoZW4oKCkgPT4gaW1hZ2UpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIGltYWdlO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbiAgfSBlbHNlIGlmKHVybCAhPT0gJycpIHtcbiAgICBsZXQgY29udGVudFR5cGU7XG5cbiAgICByZXR1cm4gaHR0cEdldCh1cmwpXG4gICAgICAudGhlbigocmVzOiBIdHRwUmVzcG9uc2UpID0+IHtcbiAgICAgICAgaWYocmVzLnN0YXR1cyAhPT0gMjAwKSB7XG4gICAgICAgICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1JFUVVFU1RcbiAgICAgICAgICB9LCBjb250ZXh0KTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnRlbnRUeXBlID0gcmVzLmhlYWRlcnMuZ2V0KCdjb250ZW50LXR5cGUnKTtcblxuICAgICAgICByZXR1cm4gcmVzO1xuICAgICAgfSlcbiAgICAgIC50aGVuKChyZXMpID0+IChyZXMgYXMgSHR0cFJlc3BvbnNlKS5idWZmZXIoKSlcbiAgICAgIC50aGVuKChidWZmZXI6IEJ1ZmZlcikgPT4ge1xuICAgICAgICBjb25zdCBpbWdQYXJhbXM6IEltYWdlVHlwZSA9IHtcbiAgICAgICAgICAuLi5jdXN0b21QYXJhbXMsXG4gICAgICAgICAgYnVmZmVyLFxuICAgICAgICAgIGZpbGVUeXBlOiBjb250ZW50VHlwZVxuICAgICAgICB9O1xuXG4gICAgICAgIHJldHVybiBhZGRJbWFnZShjb250ZXh0LCBpbWdQYXJhbXMsIHMzT3B0aW9ucylcbiAgICAgICAgICAudGhlbigoaW1hZ2U6IEltYWdlVHlwZSkgPT4ge1xuICAgICAgICAgICAgaWYoZm9ybWF0SXRlbUlkICYmIGZvcm1hdEl0ZW1UeXBlKSB7XG4gICAgICAgICAgICAgIGNvbnN0IGltYWdlRWRnZTogSW1hZ2VFZGdlVHlwZSA9IHtpbWFnZUlkOiBmb3JtYXRJbWFnZUlkLCBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCwgaXRlbVR5cGU6IGZvcm1hdEl0ZW1UeXBlfTtcbiAgICAgICAgICAgICAgcmV0dXJuIGFkZEltYWdlRWRnZShjb250ZXh0LCBpbWFnZUVkZ2UpLnRoZW4oKCkgPT4gaW1hZ2UpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICByZXR1cm4gaW1hZ2U7XG4gICAgICAgICAgfSk7XG4gICAgICB9KVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgdmFsdWU6ICdmZXRjaF9lcnJvcidcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG4gIH0gZWxzZSBpZihpbWFnZUlkICE9PSAnJykge1xuICAgIGNvbnN0IHVwZGF0ZSA9IHtcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgbW9kaWZpZWQ6IG5vd1xuICAgIH07XG4gICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUge19rZXk6ICR7Zm9ybWF0SW1hZ2VJZH19IFdJVEggJHt1cGRhdGV9IElOIGltYWdlcyBSRVRVUk4gTkVXYDtcblxuICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICBwYXJhbXM6IHtcbiAgICAgICAgICBhcWxRcnksXG4gICAgICAgICAgZm9ybWF0SW1hZ2VJZCxcbiAgICAgICAgICB1cGRhdGVcbiAgICAgICAgfSxcbiAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG4gIH1cblxuICByZXR1cm4gbnVsbDtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVJbWFnZSA9IGFzeW5jIChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdkZWxldGUnO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEltYWdlSWQgPSBwYXJzZUlkKGltYWdlSWQpO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgcmVtb3ZlRWRnZVF1ZXJ5ID0gYXFsYEZPUiBoaSBJTiBoYXNJbWFnZVxuICAgIEZJTFRFUiBoaS5fZnJvbSA9PSAke2Zvcm1hdEltYWdlSWR9XG4gICAgUkVNT1ZFIGhpIElOIGhhc0ltYWdlXG4gICAgUkVUVVJOIE9MRGA7XG5cbiAgICBhd2FpdCB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KHJlbW92ZUVkZ2VRdWVyeSk7XG5cbiAgICBjb25zdCBhcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5ID09ICR7Zm9ybWF0SW1hZ2VJZH0gJiYgaS51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgUkVNT1ZFIGkgSU4gaW1hZ2VzXG4gICAgUkVUVVJOIE9MRGA7XG5cbiAgICBjb25zdCBpbWFnZSA9IGF3YWl0IHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXVlcnkpLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSk7XG5cbiAgICBpZihpbWFnZSkge1xuICAgICAgY29uc3Qge19rZXk6IGltYWdlS2V5fSA9IGltYWdlO1xuICAgICAgY29uc3QgcGFyYW1zOiBEZWxldGVPYmplY3RzQ29tbWFuZElucHV0ID0ge1xuICAgICAgICBCdWNrZXQ6IG51bGwsXG4gICAgICAgIERlbGV0ZToge1xuICAgICAgICAgIE9iamVjdHM6IFtcbiAgICAgICAgICAgIHtLZXk6IGB1c2Vycy8ke3Nlc3Npb25JZH0vaW1hZ2VzLyR7aW1hZ2VLZXl9LmpwZ2B9LFxuICAgICAgICAgICAge0tleTogYHVzZXJzLyR7c2Vzc2lvbklkfS90aHVtYnMvJHtpbWFnZUtleX0uanBnYH1cbiAgICAgICAgICBdLFxuICAgICAgICAgIFF1aWV0OiB0cnVlXG4gICAgICAgIH1cbiAgICAgIH07XG5cbiAgICAgIHJldHVybiBzM0RlbGV0ZUxpc3QocGFyYW1zKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICB9XG5cbiAgICByZXR1cm4gbnVsbDtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHBhcmFtczoge2ltYWdlSWR9LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gIH1cbn07XG4iXSwKICAibWFwcGluZ3MiOiAiQUFLQSxPQUFRLE9BQU9BLE1BQWMsb0JBQzdCLE9BQVEsY0FBQUMsRUFBWSxpQkFBQUMsRUFBZSxpQkFBQUMsRUFBZSxhQUFBQyxFQUFXLFdBQUFDLEVBQVMsWUFBQUMsTUFBZSxlQUNyRixPQUFRLE9BQUFDLE1BQVUsV0FFbEIsT0FBT0MsTUFBUSxLQUNmLE9BQVEsWUFBQUMsTUFBZSxRQUV2QixPQUFRLG1CQUFBQyxHQUFpQixhQUFBQyxPQUFnQixXQUN6QyxPQUFRLGdCQUFBQyxHQUFjLGtCQUFBQyxHQUFnQixTQUFBQyxNQUFZLE9BQ2xELE9BQVEsVUFBQUMsTUFBYSxZQUNyQixPQUFRLGNBQUFDLE1BQWlCLHVCQUN6QixPQUFRLFlBQUFDLEVBQVUsZ0JBQUFDLE9BQW1CLDBCQUNyQyxPQUFRLFlBQUFDLEdBQVUsWUFBQUMsRUFBVSw2QkFBQUMsR0FBMkIsU0FBQUMsTUFBWSx5QkFzQm5FLE1BQU1DLEVBQXdCLFNBQzlCZixFQUFHLFNBQVMsQ0FBQyxZQUFhLElBQUksQ0FBQyxFQUV4QixNQUFNZ0IsRUFBb0IsQ0FBQ0MsRUFBd0IsQ0FBQyxJQUFNLENBQy9ELEtBQU0sQ0FDSixLQUFBQyxFQUFPLEVBQ1AsR0FBQUMsRUFBSyxHQUNMLEtBQUFDLEVBQU8sU0FD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LENBQVMsV0FBV21DLENBQVEsTUFBTSxFQUNqRCxDQUFDLElBQUssU0FBU25DLENBQVMsV0FBV21DLENBQVEsTUFBTSxDQUNuRCxFQUNBLE1BQU8sRUFDVCxDQUNGLEVBRUEsT0FBTzNFLEdBQWF1QyxDQUFNLEVBQUUsS0FBSyxJQUFNZ0YsQ0FBSyxDQUM5QyxDQUVBLE9BQU8sSUFDVCxPQUFRekYsRUFBTyxDQUNiLE9BQU96QixFQUFTLENBQ2QsT0FBQW1CLEVBQ0EsU0FBVWIsRUFDVixPQUFRLENBQUMsUUFBQStDLENBQU8sRUFDaEIsTUFBT3RELEVBQVcsY0FDcEIsRUFBRzBCLEVBQU9SLENBQU8sQ0FDbkIsQ0FDRiIsCiAgIm5hbWVzIjogWyJodHRwR2V0IiwgImNyZWF0ZUhhc2giLCAibG93ZXJDYXNlS2V5cyIsICJwYXJzZUFyYW5nb0lkIiwgInBhcnNlQ2hhciIsICJwYXJzZUlkIiwgInBhcnNlTnVtIiwgImFxbCIsICJnbSIsICJEYXRlVGltZSIsICJnZXRHcm91cERldGFpbHMiLCAiaXNHcm91cGVkIiwgInMzRGVsZXRlTGlzdCIsICJzM0dldFNpZ25lZFVybCIsICJzM1B1dCIsICJDb25maWciLCAiRXJyb3JUeXBlcyIsICJsb2dFcnJvciIsICJsb2dFeGNlcHRpb24iLCAiZ2V0RG9jSWQiLCAiZ2V0TGltaXQiLCAic2VsZWN0UmVhY3Rpb25Db3VudEJ5VHlwZSIsICJ1c2VEYiIsICJldmVudENhdGVnb3J5IiwgInBhcnNlSW1hZ2VPcHRpb25zIiwgIm9wdGlvbnMiLCAiZnJvbSIsICJ0byIsICJ0eXBlIiwgImdldEltYWdlT3B0aW9uYWwiLCAiZmllbGRzIiwgInNlbGVjdHMiLCAiZmllbGQiLCAiZ2V0SW1hZ2VzQnlVc2VyIiwgImNvbnRleHQiLCAidXNlcklkIiwgImFjdGlvbiIsICJkYXRhYmFzZU5hbWUiLCAiZm9ybWF0VXNlcklkIiwgImxpbWl0IiwgImFxbFFyeSIsICJjdXJzb3IiLCAiZXJyb3IiLCAiZ2V0SW1hZ2VDb3VudEJ5SXRlbSIsICJpdGVtSWQiLCAiZm9ybWF0SXRlbUlkIiwgImNvdW50IiwgImdldEltYWdlc0J5SXRlbSIsICJzZWxlY3RPYmplY3RzIiwgInNlbGVjdFF1ZXJpZXMiLCAiZ2V0SW1hZ2VzQnlHcm91cCIsICJwYXJhbXMiLCAic2Vzc2lvbklkIiwgImZpbHRlcnMiLCAiZ3JvdXBJZCIsICJmb3JtYXRHcm91cElkIiwgImZpbHRlciIsICJjb25kaXRpb25hbCIsICJuYW1lIiwgInZhbHVlIiwgImZvcm1hdENvbmQiLCAiZ3JvdXAiLCAiZ3JvdXBlZCIsICJnZXRJbWFnZXNCeVJlYWN0aW9ucyIsICJyZWFjdGlvbnMiLCAiZm9ybWF0U2Vzc2lvbklkIiwgImZvcm1hdFJlYWN0aW9ucyIsICJyZWFjdGlvbk5hbWUiLCAiZmlsdGVyQnkiLCAiZ2V0SW1hZ2UiLCAiaW1hZ2VJZCIsICJmb3JtYXRJbWFnZUlkIiwgImdldFBhdGhVc2VySW1hZ2VzIiwgImltYWdlVHlwZSIsICJkaXIiLCAiZmlsZW5hbWUiLCAiZ2V0QXBwSW1hZ2VVcmwiLCAiYnVja2V0IiwgImRpcmVjdG9yeSIsICJpc1RodW1iIiwgInR5cGVJZCIsICJidWNrZXROYW1lIiwgImltYWdlTmFtZSIsICJpbWFnZVVybCIsICJnZXRJbWFnZVVybCIsICJpbWFnZURpciIsICJ1cmxUeXBlIiwgImltYWdlS2V5IiwgImRlZmF1bHRCdWNrZXQiLCAicmVzaXplU2F2ZUltYWdlIiwgImJ1ZmZlciIsICJmaWxlVHlwZSIsICJzM09wdGlvbnMiLCAiaW1nVyIsICJpbWdRIiwgImZvcm1hdCIsICJpbWFnZU9wdGltaXplZEJ1ZmZlciIsICJyZXNvbHZlIiwgInJlamVjdCIsICJzdHJlYW1FcnJvciIsICJzdGRvdXQiLCAiaW1hZ2VCdWZmZXIiLCAiZGF0YSIsICJpbWFnZU9iaiIsICJpbWFnZUlkZW50aXR5IiwgImltYWdlRGF0YSIsICJmb3JtYXRWYWx1ZXMiLCAiY2FtZXJhTWFrZSIsICJjYW1lcmFNb2RlbCIsICJ0YWtlbiIsICJzdGF0cyIsICJjb2xvciIsICJyZWQiLCAiZ3JlZW4iLCAiYmx1ZSIsICJtZWFuIiwgInJnYkNvbG9yIiwgInF1YWxpdHkiLCAib3JpZW50YXRpb25EYXRhIiwgInJlc29sdXRpb25EYXRhIiwgInNpemUiLCAicmVzb2x1dGlvbiIsICJyZXNvbHV0aW9uTnVtYmVyIiwgInJlc29sdXRpb25Vbml0IiwgInRodW1iT3B0aW1pemVkQnVmZmVyIiwgInRobVciLCAidGhtUSIsICJ0aHVtYlN0ZG91dCIsICJ0aHVtYkJ1ZmZlciIsICJ0aHVtYk9iaiIsICJhZGRJbWFnZSIsICJpbWFnZSIsICJkZXNjcmlwdGlvbiIsICJub3ciLCAicmVzaXplZEltYWdlIiwgImluc2VydCIsICJhZGRJbWFnZUVkZ2UiLCAiaW1hZ2VFZGdlIiwgIml0ZW1UeXBlIiwgImVkZ2VDb2xsZWN0aW9uIiwgImVkZ2VJZCIsICJmb3JtYXRJdGVtVHlwZSIsICJpdGVtRG9jSWQiLCAiZmlsZURvY0lkIiwgImVkZ2UiLCAiZmlsZUVkZ2UiLCAidXBkYXRlSW1hZ2UiLCAiYmFzZTY0IiwgInVybCIsICJjdXN0b21QYXJhbXMiLCAiZm9ybWF0QmFzZTY0IiwgInVwZGF0ZWRGaWxlVHlwZSIsICJmaWxlVHlwZUZyb21CdWZmZXIiLCAibWltZSIsICJpbWdQYXJhbXMiLCAiY29udGVudFR5cGUiLCAicmVzIiwgInVwZGF0ZSIsICJkZWxldGVJbWFnZSIsICJyZW1vdmVFZGdlUXVlcnkiLCAiYXFsUXVlcnkiXQp9Cg==