@dollhousemcp/mcp-server 1.4.4 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (667) hide show
  1. package/CHANGELOG.md +31 -0
  2. package/README.md +68 -7
  3. package/dist/auth/GitHubAuthManager.d.ts +93 -0
  4. package/dist/auth/GitHubAuthManager.d.ts.map +1 -0
  5. package/dist/auth/GitHubAuthManager.js +465 -0
  6. package/dist/generated/version.d.ts +2 -2
  7. package/dist/generated/version.js +3 -3
  8. package/dist/index.d.ts +23 -0
  9. package/dist/index.d.ts.map +1 -1
  10. package/dist/index.js +244 -5
  11. package/dist/security/tokenManager.d.ts +34 -0
  12. package/dist/security/tokenManager.d.ts.map +1 -1
  13. package/dist/security/tokenManager.js +199 -1
  14. package/dist/server/ServerSetup.d.ts.map +1 -1
  15. package/dist/server/ServerSetup.js +4 -1
  16. package/dist/server/tools/AuthTools.d.ts +10 -0
  17. package/dist/server/tools/AuthTools.d.ts.map +1 -0
  18. package/dist/server/tools/AuthTools.js +41 -0
  19. package/dist/server/types.d.ts +3 -0
  20. package/dist/server/types.d.ts.map +1 -1
  21. package/dist/server/types.js +1 -1
  22. package/package.json +1 -1
  23. package/dist/collection/MarketplaceBrowser.d.ts +0 -24
  24. package/dist/collection/MarketplaceBrowser.d.ts.map +0 -1
  25. package/dist/collection/MarketplaceBrowser.js +0 -115
  26. package/dist/collection/MarketplaceSearch.d.ts +0 -18
  27. package/dist/collection/MarketplaceSearch.d.ts.map +0 -1
  28. package/dist/collection/MarketplaceSearch.js +0 -48
  29. package/dist/collection/PersonaInstaller.d.ts +0 -26
  30. package/dist/collection/PersonaInstaller.d.ts.map +0 -1
  31. package/dist/collection/PersonaInstaller.js +0 -103
  32. package/dist/elements/ensembles/Ensemble.d.ts +0 -144
  33. package/dist/elements/ensembles/Ensemble.d.ts.map +0 -1
  34. package/dist/elements/ensembles/Ensemble.js +0 -860
  35. package/dist/elements/ensembles/EnsembleManager.d.ts +0 -85
  36. package/dist/elements/ensembles/EnsembleManager.d.ts.map +0 -1
  37. package/dist/elements/ensembles/EnsembleManager.js +0 -378
  38. package/dist/elements/ensembles/constants.d.ts +0 -73
  39. package/dist/elements/ensembles/constants.d.ts.map +0 -1
  40. package/dist/elements/ensembles/constants.js +0 -92
  41. package/dist/elements/ensembles/index.d.ts +0 -8
  42. package/dist/elements/ensembles/index.d.ts.map +0 -1
  43. package/dist/elements/ensembles/index.js +0 -8
  44. package/dist/elements/ensembles/types.d.ts +0 -92
  45. package/dist/elements/ensembles/types.d.ts.map +0 -1
  46. package/dist/elements/ensembles/types.js +0 -8
  47. package/dist/elements/memories/Memory.d.ts +0 -110
  48. package/dist/elements/memories/Memory.d.ts.map +0 -1
  49. package/dist/elements/memories/Memory.js +0 -470
  50. package/dist/elements/memories/MemoryManager.d.ts +0 -86
  51. package/dist/elements/memories/MemoryManager.d.ts.map +0 -1
  52. package/dist/elements/memories/MemoryManager.js +0 -435
  53. package/dist/elements/memories/constants.d.ts +0 -42
  54. package/dist/elements/memories/constants.d.ts.map +0 -1
  55. package/dist/elements/memories/constants.js +0 -49
  56. package/dist/elements/memories/index.d.ts +0 -6
  57. package/dist/elements/memories/index.d.ts.map +0 -1
  58. package/dist/elements/memories/index.js +0 -6
  59. package/dist/marketplace/GitHubClient.d.ts +0 -22
  60. package/dist/marketplace/GitHubClient.d.ts.map +0 -1
  61. package/dist/marketplace/GitHubClient.js +0 -112
  62. package/dist/marketplace/MarketplaceBrowser.d.ts +0 -24
  63. package/dist/marketplace/MarketplaceBrowser.d.ts.map +0 -1
  64. package/dist/marketplace/MarketplaceBrowser.js +0 -115
  65. package/dist/marketplace/MarketplaceSearch.d.ts +0 -18
  66. package/dist/marketplace/MarketplaceSearch.d.ts.map +0 -1
  67. package/dist/marketplace/MarketplaceSearch.js +0 -48
  68. package/dist/marketplace/PersonaDetails.d.ts +0 -22
  69. package/dist/marketplace/PersonaDetails.d.ts.map +0 -1
  70. package/dist/marketplace/PersonaDetails.js +0 -71
  71. package/dist/marketplace/PersonaInstaller.d.ts +0 -25
  72. package/dist/marketplace/PersonaInstaller.d.ts.map +0 -1
  73. package/dist/marketplace/PersonaInstaller.js +0 -100
  74. package/dist/marketplace/PersonaSubmitter.d.ts +0 -19
  75. package/dist/marketplace/PersonaSubmitter.d.ts.map +0 -1
  76. package/dist/marketplace/PersonaSubmitter.js +0 -57
  77. package/dist/marketplace/index.d.ts +0 -10
  78. package/dist/marketplace/index.d.ts.map +0 -1
  79. package/dist/marketplace/index.js +0 -10
  80. package/dist/server/tools/MarketplaceTools.d.ts +0 -10
  81. package/dist/server/tools/MarketplaceTools.d.ts.map +0 -1
  82. package/dist/server/tools/MarketplaceTools.js +0 -96
  83. package/dist/src/cache/APICache.d.ts +0 -23
  84. package/dist/src/cache/APICache.d.ts.map +0 -1
  85. package/dist/src/cache/APICache.js +0 -42
  86. package/dist/src/cache/index.d.ts +0 -5
  87. package/dist/src/cache/index.d.ts.map +0 -1
  88. package/dist/src/cache/index.js +0 -5
  89. package/dist/src/config/constants.d.ts +0 -25
  90. package/dist/src/config/constants.d.ts.map +0 -1
  91. package/dist/src/config/constants.js +0 -30
  92. package/dist/src/config/index.d.ts +0 -6
  93. package/dist/src/config/index.d.ts.map +0 -1
  94. package/dist/src/config/index.js +0 -6
  95. package/dist/src/config/indicator-config.d.ts +0 -107
  96. package/dist/src/config/indicator-config.d.ts.map +0 -1
  97. package/dist/src/config/indicator-config.js +0 -158
  98. package/dist/src/constants/defaultPersonas.d.ts +0 -10
  99. package/dist/src/constants/defaultPersonas.d.ts.map +0 -1
  100. package/dist/src/constants/defaultPersonas.js +0 -18
  101. package/dist/src/constants/limits.d.ts +0 -10
  102. package/dist/src/constants/limits.d.ts.map +0 -1
  103. package/dist/src/constants/limits.js +0 -13
  104. package/dist/src/errors/SecurityError.d.ts +0 -29
  105. package/dist/src/errors/SecurityError.d.ts.map +0 -1
  106. package/dist/src/errors/SecurityError.js +0 -47
  107. package/dist/src/errors/index.d.ts +0 -2
  108. package/dist/src/errors/index.d.ts.map +0 -1
  109. package/dist/src/errors/index.js +0 -2
  110. package/dist/src/index.barrel.d.ts +0 -21
  111. package/dist/src/index.barrel.d.ts.map +0 -1
  112. package/dist/src/index.barrel.js +0 -31
  113. package/dist/src/index.d.ts +0 -220
  114. package/dist/src/index.d.ts.map +0 -1
  115. package/dist/src/index.js +0 -1559
  116. package/dist/src/marketplace/GitHubClient.d.ts +0 -22
  117. package/dist/src/marketplace/GitHubClient.d.ts.map +0 -1
  118. package/dist/src/marketplace/GitHubClient.js +0 -112
  119. package/dist/src/marketplace/MarketplaceBrowser.d.ts +0 -21
  120. package/dist/src/marketplace/MarketplaceBrowser.d.ts.map +0 -1
  121. package/dist/src/marketplace/MarketplaceBrowser.js +0 -45
  122. package/dist/src/marketplace/MarketplaceSearch.d.ts +0 -18
  123. package/dist/src/marketplace/MarketplaceSearch.d.ts.map +0 -1
  124. package/dist/src/marketplace/MarketplaceSearch.js +0 -36
  125. package/dist/src/marketplace/PersonaDetails.d.ts +0 -22
  126. package/dist/src/marketplace/PersonaDetails.d.ts.map +0 -1
  127. package/dist/src/marketplace/PersonaDetails.js +0 -71
  128. package/dist/src/marketplace/PersonaInstaller.d.ts +0 -25
  129. package/dist/src/marketplace/PersonaInstaller.d.ts.map +0 -1
  130. package/dist/src/marketplace/PersonaInstaller.js +0 -100
  131. package/dist/src/marketplace/PersonaSubmitter.d.ts +0 -19
  132. package/dist/src/marketplace/PersonaSubmitter.d.ts.map +0 -1
  133. package/dist/src/marketplace/PersonaSubmitter.js +0 -57
  134. package/dist/src/marketplace/index.d.ts +0 -10
  135. package/dist/src/marketplace/index.d.ts.map +0 -1
  136. package/dist/src/marketplace/index.js +0 -10
  137. package/dist/src/persona/PersonaLoader.d.ts +0 -33
  138. package/dist/src/persona/PersonaLoader.d.ts.map +0 -1
  139. package/dist/src/persona/PersonaLoader.js +0 -139
  140. package/dist/src/persona/PersonaManager.d.ts +0 -112
  141. package/dist/src/persona/PersonaManager.d.ts.map +0 -1
  142. package/dist/src/persona/PersonaManager.js +0 -341
  143. package/dist/src/persona/PersonaValidator.d.ts +0 -33
  144. package/dist/src/persona/PersonaValidator.d.ts.map +0 -1
  145. package/dist/src/persona/PersonaValidator.js +0 -157
  146. package/dist/src/persona/export-import/PersonaExporter.d.ts +0 -43
  147. package/dist/src/persona/export-import/PersonaExporter.d.ts.map +0 -1
  148. package/dist/src/persona/export-import/PersonaExporter.js +0 -99
  149. package/dist/src/persona/export-import/PersonaImporter.d.ts +0 -65
  150. package/dist/src/persona/export-import/PersonaImporter.d.ts.map +0 -1
  151. package/dist/src/persona/export-import/PersonaImporter.js +0 -313
  152. package/dist/src/persona/export-import/PersonaSharer.d.ts +0 -60
  153. package/dist/src/persona/export-import/PersonaSharer.d.ts.map +0 -1
  154. package/dist/src/persona/export-import/PersonaSharer.js +0 -363
  155. package/dist/src/persona/export-import/index.d.ts +0 -10
  156. package/dist/src/persona/export-import/index.d.ts.map +0 -1
  157. package/dist/src/persona/export-import/index.js +0 -7
  158. package/dist/src/persona/index.d.ts +0 -7
  159. package/dist/src/persona/index.d.ts.map +0 -1
  160. package/dist/src/persona/index.js +0 -7
  161. package/dist/src/security/InputValidator.d.ts +0 -69
  162. package/dist/src/security/InputValidator.d.ts.map +0 -1
  163. package/dist/src/security/InputValidator.js +0 -381
  164. package/dist/src/security/commandValidator.d.ts +0 -7
  165. package/dist/src/security/commandValidator.d.ts.map +0 -1
  166. package/dist/src/security/commandValidator.js +0 -77
  167. package/dist/src/security/constants.d.ts +0 -21
  168. package/dist/src/security/constants.d.ts.map +0 -1
  169. package/dist/src/security/constants.js +0 -23
  170. package/dist/src/security/contentValidator.d.ts +0 -47
  171. package/dist/src/security/contentValidator.d.ts.map +0 -1
  172. package/dist/src/security/contentValidator.js +0 -188
  173. package/dist/src/security/fileLockManager.d.ts +0 -70
  174. package/dist/src/security/fileLockManager.d.ts.map +0 -1
  175. package/dist/src/security/fileLockManager.js +0 -187
  176. package/dist/src/security/index.d.ts +0 -12
  177. package/dist/src/security/index.d.ts.map +0 -1
  178. package/dist/src/security/index.js +0 -14
  179. package/dist/src/security/pathValidator.d.ts +0 -9
  180. package/dist/src/security/pathValidator.d.ts.map +0 -1
  181. package/dist/src/security/pathValidator.js +0 -97
  182. package/dist/src/security/secureYamlParser.d.ts +0 -46
  183. package/dist/src/security/secureYamlParser.d.ts.map +0 -1
  184. package/dist/src/security/secureYamlParser.js +0 -203
  185. package/dist/src/security/securityMonitor.d.ts +0 -58
  186. package/dist/src/security/securityMonitor.d.ts.map +0 -1
  187. package/dist/src/security/securityMonitor.js +0 -108
  188. package/dist/src/security/tokenManager.d.ts +0 -59
  189. package/dist/src/security/tokenManager.d.ts.map +0 -1
  190. package/dist/src/security/tokenManager.js +0 -216
  191. package/dist/src/security/yamlValidator.d.ts +0 -20
  192. package/dist/src/security/yamlValidator.d.ts.map +0 -1
  193. package/dist/src/security/yamlValidator.js +0 -138
  194. package/dist/src/server/ServerSetup.d.ts +0 -31
  195. package/dist/src/server/ServerSetup.d.ts.map +0 -1
  196. package/dist/src/server/ServerSetup.js +0 -79
  197. package/dist/src/server/index.d.ts +0 -7
  198. package/dist/src/server/index.d.ts.map +0 -1
  199. package/dist/src/server/index.js +0 -7
  200. package/dist/src/server/tools/ConfigTools.d.ts +0 -10
  201. package/dist/src/server/tools/ConfigTools.d.ts.map +0 -1
  202. package/dist/src/server/tools/ConfigTools.js +0 -63
  203. package/dist/src/server/tools/MarketplaceTools.d.ts +0 -10
  204. package/dist/src/server/tools/MarketplaceTools.d.ts.map +0 -1
  205. package/dist/src/server/tools/MarketplaceTools.js +0 -92
  206. package/dist/src/server/tools/PersonaTools.d.ts +0 -10
  207. package/dist/src/server/tools/PersonaTools.d.ts.map +0 -1
  208. package/dist/src/server/tools/PersonaTools.js +0 -257
  209. package/dist/src/server/tools/ToolRegistry.d.ts +0 -37
  210. package/dist/src/server/tools/ToolRegistry.d.ts.map +0 -1
  211. package/dist/src/server/tools/ToolRegistry.js +0 -40
  212. package/dist/src/server/tools/UpdateTools.d.ts +0 -10
  213. package/dist/src/server/tools/UpdateTools.d.ts.map +0 -1
  214. package/dist/src/server/tools/UpdateTools.js +0 -64
  215. package/dist/src/server/tools/UserTools.d.ts +0 -10
  216. package/dist/src/server/tools/UserTools.d.ts.map +0 -1
  217. package/dist/src/server/tools/UserTools.js +0 -51
  218. package/dist/src/server/tools/index.d.ts +0 -10
  219. package/dist/src/server/tools/index.d.ts.map +0 -1
  220. package/dist/src/server/tools/index.js +0 -10
  221. package/dist/src/server/types.d.ts +0 -34
  222. package/dist/src/server/types.d.ts.map +0 -1
  223. package/dist/src/server/types.js +0 -5
  224. package/dist/src/tools/debug.d.ts +0 -20
  225. package/dist/src/tools/debug.d.ts.map +0 -1
  226. package/dist/src/tools/debug.js +0 -37
  227. package/dist/src/types/cache.d.ts +0 -8
  228. package/dist/src/types/cache.d.ts.map +0 -1
  229. package/dist/src/types/cache.js +0 -5
  230. package/dist/src/types/index.d.ts +0 -8
  231. package/dist/src/types/index.d.ts.map +0 -1
  232. package/dist/src/types/index.js +0 -8
  233. package/dist/src/types/marketplace.d.ts +0 -23
  234. package/dist/src/types/marketplace.d.ts.map +0 -1
  235. package/dist/src/types/marketplace.js +0 -5
  236. package/dist/src/types/mcp.d.ts +0 -161
  237. package/dist/src/types/mcp.d.ts.map +0 -1
  238. package/dist/src/types/mcp.js +0 -75
  239. package/dist/src/types/persona.d.ts +0 -30
  240. package/dist/src/types/persona.d.ts.map +0 -1
  241. package/dist/src/types/persona.js +0 -5
  242. package/dist/src/update/BackupManager.d.ts +0 -46
  243. package/dist/src/update/BackupManager.d.ts.map +0 -1
  244. package/dist/src/update/BackupManager.js +0 -261
  245. package/dist/src/update/DependencyChecker.d.ts +0 -41
  246. package/dist/src/update/DependencyChecker.d.ts.map +0 -1
  247. package/dist/src/update/DependencyChecker.js +0 -132
  248. package/dist/src/update/RateLimiter.d.ts +0 -80
  249. package/dist/src/update/RateLimiter.d.ts.map +0 -1
  250. package/dist/src/update/RateLimiter.js +0 -172
  251. package/dist/src/update/SignatureVerifier.d.ts +0 -71
  252. package/dist/src/update/SignatureVerifier.d.ts.map +0 -1
  253. package/dist/src/update/SignatureVerifier.js +0 -214
  254. package/dist/src/update/UpdateChecker.d.ts +0 -127
  255. package/dist/src/update/UpdateChecker.d.ts.map +0 -1
  256. package/dist/src/update/UpdateChecker.js +0 -460
  257. package/dist/src/update/UpdateManager.d.ts +0 -41
  258. package/dist/src/update/UpdateManager.d.ts.map +0 -1
  259. package/dist/src/update/UpdateManager.js +0 -260
  260. package/dist/src/update/VersionManager.d.ts +0 -31
  261. package/dist/src/update/VersionManager.d.ts.map +0 -1
  262. package/dist/src/update/VersionManager.js +0 -134
  263. package/dist/src/update/index.d.ts +0 -9
  264. package/dist/src/update/index.d.ts.map +0 -1
  265. package/dist/src/update/index.js +0 -9
  266. package/dist/src/utils/filesystem.d.ts +0 -32
  267. package/dist/src/utils/filesystem.d.ts.map +0 -1
  268. package/dist/src/utils/filesystem.js +0 -73
  269. package/dist/src/utils/git.d.ts +0 -32
  270. package/dist/src/utils/git.d.ts.map +0 -1
  271. package/dist/src/utils/git.js +0 -65
  272. package/dist/src/utils/index.d.ts +0 -7
  273. package/dist/src/utils/index.d.ts.map +0 -1
  274. package/dist/src/utils/index.js +0 -7
  275. package/dist/src/utils/logger.d.ts +0 -45
  276. package/dist/src/utils/logger.d.ts.map +0 -1
  277. package/dist/src/utils/logger.js +0 -91
  278. package/dist/src/utils/version.d.ts +0 -25
  279. package/dist/src/utils/version.d.ts.map +0 -1
  280. package/dist/src/utils/version.js +0 -97
  281. package/dist/test/src/cache/APICache.d.ts +0 -23
  282. package/dist/test/src/cache/APICache.d.ts.map +0 -1
  283. package/dist/test/src/cache/APICache.js +0 -42
  284. package/dist/test/src/cache/index.d.ts +0 -5
  285. package/dist/test/src/cache/index.d.ts.map +0 -1
  286. package/dist/test/src/cache/index.js +0 -5
  287. package/dist/test/src/collection/CollectionBrowser.d.ts +0 -24
  288. package/dist/test/src/collection/CollectionBrowser.d.ts.map +0 -1
  289. package/dist/test/src/collection/CollectionBrowser.js +0 -115
  290. package/dist/test/src/collection/CollectionSearch.d.ts +0 -18
  291. package/dist/test/src/collection/CollectionSearch.d.ts.map +0 -1
  292. package/dist/test/src/collection/CollectionSearch.js +0 -48
  293. package/dist/test/src/collection/GitHubClient.d.ts +0 -22
  294. package/dist/test/src/collection/GitHubClient.d.ts.map +0 -1
  295. package/dist/test/src/collection/GitHubClient.js +0 -114
  296. package/dist/test/src/collection/PersonaDetails.d.ts +0 -22
  297. package/dist/test/src/collection/PersonaDetails.d.ts.map +0 -1
  298. package/dist/test/src/collection/PersonaDetails.js +0 -71
  299. package/dist/test/src/collection/PersonaInstaller.d.ts +0 -26
  300. package/dist/test/src/collection/PersonaInstaller.d.ts.map +0 -1
  301. package/dist/test/src/collection/PersonaInstaller.js +0 -103
  302. package/dist/test/src/collection/PersonaSubmitter.d.ts +0 -19
  303. package/dist/test/src/collection/PersonaSubmitter.d.ts.map +0 -1
  304. package/dist/test/src/collection/PersonaSubmitter.js +0 -57
  305. package/dist/test/src/collection/index.d.ts +0 -10
  306. package/dist/test/src/collection/index.d.ts.map +0 -1
  307. package/dist/test/src/collection/index.js +0 -10
  308. package/dist/test/src/config/constants.d.ts +0 -25
  309. package/dist/test/src/config/constants.d.ts.map +0 -1
  310. package/dist/test/src/config/constants.js +0 -30
  311. package/dist/test/src/config/index.d.ts +0 -6
  312. package/dist/test/src/config/index.d.ts.map +0 -1
  313. package/dist/test/src/config/index.js +0 -6
  314. package/dist/test/src/config/indicator-config.d.ts +0 -107
  315. package/dist/test/src/config/indicator-config.d.ts.map +0 -1
  316. package/dist/test/src/config/indicator-config.js +0 -158
  317. package/dist/test/src/constants/defaultPersonas.d.ts +0 -10
  318. package/dist/test/src/constants/defaultPersonas.d.ts.map +0 -1
  319. package/dist/test/src/constants/defaultPersonas.js +0 -18
  320. package/dist/test/src/constants/limits.d.ts +0 -10
  321. package/dist/test/src/constants/limits.d.ts.map +0 -1
  322. package/dist/test/src/constants/limits.js +0 -13
  323. package/dist/test/src/elements/BaseElement.d.ts +0 -81
  324. package/dist/test/src/elements/BaseElement.d.ts.map +0 -1
  325. package/dist/test/src/elements/BaseElement.js +0 -381
  326. package/dist/test/src/elements/FeedbackProcessor.d.ts +0 -57
  327. package/dist/test/src/elements/FeedbackProcessor.d.ts.map +0 -1
  328. package/dist/test/src/elements/FeedbackProcessor.js +0 -418
  329. package/dist/test/src/elements/agents/Agent.d.ts +0 -145
  330. package/dist/test/src/elements/agents/Agent.d.ts.map +0 -1
  331. package/dist/test/src/elements/agents/Agent.js +0 -848
  332. package/dist/test/src/elements/agents/AgentManager.d.ts +0 -125
  333. package/dist/test/src/elements/agents/AgentManager.d.ts.map +0 -1
  334. package/dist/test/src/elements/agents/AgentManager.js +0 -608
  335. package/dist/test/src/elements/agents/constants.d.ts +0 -42
  336. package/dist/test/src/elements/agents/constants.d.ts.map +0 -1
  337. package/dist/test/src/elements/agents/constants.js +0 -45
  338. package/dist/test/src/elements/agents/goalTemplates.d.ts +0 -44
  339. package/dist/test/src/elements/agents/goalTemplates.d.ts.map +0 -1
  340. package/dist/test/src/elements/agents/goalTemplates.js +0 -297
  341. package/dist/test/src/elements/agents/index.d.ts +0 -8
  342. package/dist/test/src/elements/agents/index.d.ts.map +0 -1
  343. package/dist/test/src/elements/agents/index.js +0 -8
  344. package/dist/test/src/elements/agents/ruleEngineConfig.d.ts +0 -76
  345. package/dist/test/src/elements/agents/ruleEngineConfig.d.ts.map +0 -1
  346. package/dist/test/src/elements/agents/ruleEngineConfig.js +0 -143
  347. package/dist/test/src/elements/agents/types.d.ts +0 -97
  348. package/dist/test/src/elements/agents/types.d.ts.map +0 -1
  349. package/dist/test/src/elements/agents/types.js +0 -5
  350. package/dist/test/src/elements/index.d.ts +0 -6
  351. package/dist/test/src/elements/index.d.ts.map +0 -1
  352. package/dist/test/src/elements/index.js +0 -6
  353. package/dist/test/src/elements/memories/Memory.d.ts +0 -110
  354. package/dist/test/src/elements/memories/Memory.d.ts.map +0 -1
  355. package/dist/test/src/elements/memories/Memory.js +0 -470
  356. package/dist/test/src/elements/memories/MemoryManager.d.ts +0 -86
  357. package/dist/test/src/elements/memories/MemoryManager.d.ts.map +0 -1
  358. package/dist/test/src/elements/memories/MemoryManager.js +0 -435
  359. package/dist/test/src/elements/memories/constants.d.ts +0 -42
  360. package/dist/test/src/elements/memories/constants.d.ts.map +0 -1
  361. package/dist/test/src/elements/memories/constants.js +0 -49
  362. package/dist/test/src/elements/memories/index.d.ts +0 -6
  363. package/dist/test/src/elements/memories/index.d.ts.map +0 -1
  364. package/dist/test/src/elements/memories/index.js +0 -6
  365. package/dist/test/src/elements/skills/Skill.d.ts +0 -109
  366. package/dist/test/src/elements/skills/Skill.d.ts.map +0 -1
  367. package/dist/test/src/elements/skills/Skill.js +0 -381
  368. package/dist/test/src/elements/templates/Template.d.ts +0 -138
  369. package/dist/test/src/elements/templates/Template.d.ts.map +0 -1
  370. package/dist/test/src/elements/templates/Template.js +0 -673
  371. package/dist/test/src/elements/templates/TemplateManager.d.ts +0 -104
  372. package/dist/test/src/elements/templates/TemplateManager.d.ts.map +0 -1
  373. package/dist/test/src/elements/templates/TemplateManager.js +0 -496
  374. package/dist/test/src/elements/templates/index.d.ts +0 -6
  375. package/dist/test/src/elements/templates/index.d.ts.map +0 -1
  376. package/dist/test/src/elements/templates/index.js +0 -6
  377. package/dist/test/src/errors/SecurityError.d.ts +0 -29
  378. package/dist/test/src/errors/SecurityError.d.ts.map +0 -1
  379. package/dist/test/src/errors/SecurityError.js +0 -47
  380. package/dist/test/src/errors/index.d.ts +0 -2
  381. package/dist/test/src/errors/index.d.ts.map +0 -1
  382. package/dist/test/src/errors/index.js +0 -2
  383. package/dist/test/src/index.barrel.d.ts +0 -21
  384. package/dist/test/src/index.barrel.d.ts.map +0 -1
  385. package/dist/test/src/index.barrel.js +0 -31
  386. package/dist/test/src/index.d.ts +0 -223
  387. package/dist/test/src/index.d.ts.map +0 -1
  388. package/dist/test/src/index.js +0 -1594
  389. package/dist/test/src/marketplace/GitHubClient.d.ts +0 -22
  390. package/dist/test/src/marketplace/GitHubClient.d.ts.map +0 -1
  391. package/dist/test/src/marketplace/GitHubClient.js +0 -112
  392. package/dist/test/src/marketplace/MarketplaceBrowser.d.ts +0 -21
  393. package/dist/test/src/marketplace/MarketplaceBrowser.d.ts.map +0 -1
  394. package/dist/test/src/marketplace/MarketplaceBrowser.js +0 -45
  395. package/dist/test/src/marketplace/MarketplaceSearch.d.ts +0 -18
  396. package/dist/test/src/marketplace/MarketplaceSearch.d.ts.map +0 -1
  397. package/dist/test/src/marketplace/MarketplaceSearch.js +0 -36
  398. package/dist/test/src/marketplace/PersonaDetails.d.ts +0 -22
  399. package/dist/test/src/marketplace/PersonaDetails.d.ts.map +0 -1
  400. package/dist/test/src/marketplace/PersonaDetails.js +0 -71
  401. package/dist/test/src/marketplace/PersonaInstaller.d.ts +0 -25
  402. package/dist/test/src/marketplace/PersonaInstaller.d.ts.map +0 -1
  403. package/dist/test/src/marketplace/PersonaInstaller.js +0 -100
  404. package/dist/test/src/marketplace/PersonaSubmitter.d.ts +0 -19
  405. package/dist/test/src/marketplace/PersonaSubmitter.d.ts.map +0 -1
  406. package/dist/test/src/marketplace/PersonaSubmitter.js +0 -57
  407. package/dist/test/src/marketplace/index.d.ts +0 -10
  408. package/dist/test/src/marketplace/index.d.ts.map +0 -1
  409. package/dist/test/src/marketplace/index.js +0 -10
  410. package/dist/test/src/persona/PersonaElement.d.ts +0 -64
  411. package/dist/test/src/persona/PersonaElement.d.ts.map +0 -1
  412. package/dist/test/src/persona/PersonaElement.js +0 -223
  413. package/dist/test/src/persona/PersonaElementManager.d.ts +0 -97
  414. package/dist/test/src/persona/PersonaElementManager.d.ts.map +0 -1
  415. package/dist/test/src/persona/PersonaElementManager.js +0 -342
  416. package/dist/test/src/persona/PersonaLoader.d.ts +0 -34
  417. package/dist/test/src/persona/PersonaLoader.d.ts.map +0 -1
  418. package/dist/test/src/persona/PersonaLoader.js +0 -145
  419. package/dist/test/src/persona/PersonaManager.d.ts +0 -112
  420. package/dist/test/src/persona/PersonaManager.d.ts.map +0 -1
  421. package/dist/test/src/persona/PersonaManager.js +0 -341
  422. package/dist/test/src/persona/PersonaValidator.d.ts +0 -33
  423. package/dist/test/src/persona/PersonaValidator.d.ts.map +0 -1
  424. package/dist/test/src/persona/PersonaValidator.js +0 -157
  425. package/dist/test/src/persona/export-import/PersonaExporter.d.ts +0 -43
  426. package/dist/test/src/persona/export-import/PersonaExporter.d.ts.map +0 -1
  427. package/dist/test/src/persona/export-import/PersonaExporter.js +0 -99
  428. package/dist/test/src/persona/export-import/PersonaImporter.d.ts +0 -65
  429. package/dist/test/src/persona/export-import/PersonaImporter.d.ts.map +0 -1
  430. package/dist/test/src/persona/export-import/PersonaImporter.js +0 -315
  431. package/dist/test/src/persona/export-import/PersonaSharer.d.ts +0 -60
  432. package/dist/test/src/persona/export-import/PersonaSharer.d.ts.map +0 -1
  433. package/dist/test/src/persona/export-import/PersonaSharer.js +0 -502
  434. package/dist/test/src/persona/export-import/index.d.ts +0 -10
  435. package/dist/test/src/persona/export-import/index.d.ts.map +0 -1
  436. package/dist/test/src/persona/export-import/index.js +0 -7
  437. package/dist/test/src/persona/index.d.ts +0 -7
  438. package/dist/test/src/persona/index.d.ts.map +0 -1
  439. package/dist/test/src/persona/index.js +0 -7
  440. package/dist/test/src/portfolio/MigrationManager.d.ts +0 -44
  441. package/dist/test/src/portfolio/MigrationManager.d.ts.map +0 -1
  442. package/dist/test/src/portfolio/MigrationManager.js +0 -163
  443. package/dist/test/src/portfolio/PortfolioManager.d.ts +0 -54
  444. package/dist/test/src/portfolio/PortfolioManager.d.ts.map +0 -1
  445. package/dist/test/src/portfolio/PortfolioManager.js +0 -224
  446. package/dist/test/src/portfolio/types.d.ts +0 -18
  447. package/dist/test/src/portfolio/types.d.ts.map +0 -1
  448. package/dist/test/src/portfolio/types.js +0 -13
  449. package/dist/test/src/security/InputValidator.d.ts +0 -80
  450. package/dist/test/src/security/InputValidator.d.ts.map +0 -1
  451. package/dist/test/src/security/InputValidator.js +0 -436
  452. package/dist/test/src/security/audit/SecurityAuditor.d.ts +0 -44
  453. package/dist/test/src/security/audit/SecurityAuditor.d.ts.map +0 -1
  454. package/dist/test/src/security/audit/SecurityAuditor.js +0 -274
  455. package/dist/test/src/security/audit/config/suppressions.d.ts +0 -34
  456. package/dist/test/src/security/audit/config/suppressions.d.ts.map +0 -1
  457. package/dist/test/src/security/audit/config/suppressions.js +0 -575
  458. package/dist/test/src/security/audit/index.d.ts +0 -14
  459. package/dist/test/src/security/audit/index.d.ts.map +0 -1
  460. package/dist/test/src/security/audit/index.js +0 -15
  461. package/dist/test/src/security/audit/reporters/ConsoleReporter.d.ts +0 -46
  462. package/dist/test/src/security/audit/reporters/ConsoleReporter.d.ts.map +0 -1
  463. package/dist/test/src/security/audit/reporters/ConsoleReporter.js +0 -174
  464. package/dist/test/src/security/audit/reporters/JsonReporter.d.ts +0 -13
  465. package/dist/test/src/security/audit/reporters/JsonReporter.d.ts.map +0 -1
  466. package/dist/test/src/security/audit/reporters/JsonReporter.js +0 -25
  467. package/dist/test/src/security/audit/reporters/MarkdownReporter.d.ts +0 -13
  468. package/dist/test/src/security/audit/reporters/MarkdownReporter.d.ts.map +0 -1
  469. package/dist/test/src/security/audit/reporters/MarkdownReporter.js +0 -79
  470. package/dist/test/src/security/audit/rules/SecurityRules.d.ts +0 -20
  471. package/dist/test/src/security/audit/rules/SecurityRules.d.ts.map +0 -1
  472. package/dist/test/src/security/audit/rules/SecurityRules.js +0 -244
  473. package/dist/test/src/security/audit/scanners/CodeScanner.d.ts +0 -47
  474. package/dist/test/src/security/audit/scanners/CodeScanner.d.ts.map +0 -1
  475. package/dist/test/src/security/audit/scanners/CodeScanner.js +0 -174
  476. package/dist/test/src/security/audit/scanners/ConfigurationScanner.d.ts +0 -13
  477. package/dist/test/src/security/audit/scanners/ConfigurationScanner.d.ts.map +0 -1
  478. package/dist/test/src/security/audit/scanners/ConfigurationScanner.js +0 -22
  479. package/dist/test/src/security/audit/scanners/DependencyScanner.d.ts +0 -13
  480. package/dist/test/src/security/audit/scanners/DependencyScanner.d.ts.map +0 -1
  481. package/dist/test/src/security/audit/scanners/DependencyScanner.js +0 -22
  482. package/dist/test/src/security/audit/types.d.ts +0 -94
  483. package/dist/test/src/security/audit/types.d.ts.map +0 -1
  484. package/dist/test/src/security/audit/types.js +0 -6
  485. package/dist/test/src/security/commandValidator.d.ts +0 -7
  486. package/dist/test/src/security/commandValidator.d.ts.map +0 -1
  487. package/dist/test/src/security/commandValidator.js +0 -78
  488. package/dist/test/src/security/constants.d.ts +0 -24
  489. package/dist/test/src/security/constants.d.ts.map +0 -1
  490. package/dist/test/src/security/constants.js +0 -26
  491. package/dist/test/src/security/contentValidator.d.ts +0 -47
  492. package/dist/test/src/security/contentValidator.d.ts.map +0 -1
  493. package/dist/test/src/security/contentValidator.js +0 -301
  494. package/dist/test/src/security/errors.d.ts +0 -14
  495. package/dist/test/src/security/errors.d.ts.map +0 -1
  496. package/dist/test/src/security/errors.js +0 -28
  497. package/dist/test/src/security/fileLockManager.d.ts +0 -70
  498. package/dist/test/src/security/fileLockManager.d.ts.map +0 -1
  499. package/dist/test/src/security/fileLockManager.js +0 -187
  500. package/dist/test/src/security/index.d.ts +0 -12
  501. package/dist/test/src/security/index.d.ts.map +0 -1
  502. package/dist/test/src/security/index.js +0 -14
  503. package/dist/test/src/security/pathValidator.d.ts +0 -9
  504. package/dist/test/src/security/pathValidator.d.ts.map +0 -1
  505. package/dist/test/src/security/pathValidator.js +0 -98
  506. package/dist/test/src/security/regexValidator.d.ts +0 -59
  507. package/dist/test/src/security/regexValidator.d.ts.map +0 -1
  508. package/dist/test/src/security/regexValidator.js +0 -214
  509. package/dist/test/src/security/secureYamlParser.d.ts +0 -46
  510. package/dist/test/src/security/secureYamlParser.d.ts.map +0 -1
  511. package/dist/test/src/security/secureYamlParser.js +0 -203
  512. package/dist/test/src/security/securityMonitor.d.ts +0 -58
  513. package/dist/test/src/security/securityMonitor.d.ts.map +0 -1
  514. package/dist/test/src/security/securityMonitor.js +0 -108
  515. package/dist/test/src/security/tokenManager.d.ts +0 -85
  516. package/dist/test/src/security/tokenManager.d.ts.map +0 -1
  517. package/dist/test/src/security/tokenManager.js +0 -286
  518. package/dist/test/src/security/validators/unicodeValidator.d.ts +0 -97
  519. package/dist/test/src/security/validators/unicodeValidator.d.ts.map +0 -1
  520. package/dist/test/src/security/validators/unicodeValidator.js +0 -312
  521. package/dist/test/src/security/yamlValidator.d.ts +0 -21
  522. package/dist/test/src/security/yamlValidator.d.ts.map +0 -1
  523. package/dist/test/src/security/yamlValidator.js +0 -164
  524. package/dist/test/src/server/ServerSetup.d.ts +0 -35
  525. package/dist/test/src/server/ServerSetup.d.ts.map +0 -1
  526. package/dist/test/src/server/ServerSetup.js +0 -116
  527. package/dist/test/src/server/index.d.ts +0 -7
  528. package/dist/test/src/server/index.d.ts.map +0 -1
  529. package/dist/test/src/server/index.js +0 -7
  530. package/dist/test/src/server/startup.d.ts +0 -31
  531. package/dist/test/src/server/startup.d.ts.map +0 -1
  532. package/dist/test/src/server/startup.js +0 -67
  533. package/dist/test/src/server/tools/CollectionTools.d.ts +0 -10
  534. package/dist/test/src/server/tools/CollectionTools.d.ts.map +0 -1
  535. package/dist/test/src/server/tools/CollectionTools.js +0 -96
  536. package/dist/test/src/server/tools/ConfigTools.d.ts +0 -10
  537. package/dist/test/src/server/tools/ConfigTools.d.ts.map +0 -1
  538. package/dist/test/src/server/tools/ConfigTools.js +0 -63
  539. package/dist/test/src/server/tools/MarketplaceTools.d.ts +0 -10
  540. package/dist/test/src/server/tools/MarketplaceTools.d.ts.map +0 -1
  541. package/dist/test/src/server/tools/MarketplaceTools.js +0 -92
  542. package/dist/test/src/server/tools/PersonaTools.d.ts +0 -10
  543. package/dist/test/src/server/tools/PersonaTools.d.ts.map +0 -1
  544. package/dist/test/src/server/tools/PersonaTools.js +0 -257
  545. package/dist/test/src/server/tools/ToolRegistry.d.ts +0 -37
  546. package/dist/test/src/server/tools/ToolRegistry.d.ts.map +0 -1
  547. package/dist/test/src/server/tools/ToolRegistry.js +0 -40
  548. package/dist/test/src/server/tools/UpdateTools.d.ts +0 -10
  549. package/dist/test/src/server/tools/UpdateTools.d.ts.map +0 -1
  550. package/dist/test/src/server/tools/UpdateTools.js +0 -64
  551. package/dist/test/src/server/tools/UserTools.d.ts +0 -10
  552. package/dist/test/src/server/tools/UserTools.d.ts.map +0 -1
  553. package/dist/test/src/server/tools/UserTools.js +0 -51
  554. package/dist/test/src/server/tools/index.d.ts +0 -10
  555. package/dist/test/src/server/tools/index.d.ts.map +0 -1
  556. package/dist/test/src/server/tools/index.js +0 -10
  557. package/dist/test/src/server/types.d.ts +0 -34
  558. package/dist/test/src/server/types.d.ts.map +0 -1
  559. package/dist/test/src/server/types.js +0 -5
  560. package/dist/test/src/tools/debug.d.ts +0 -20
  561. package/dist/test/src/tools/debug.d.ts.map +0 -1
  562. package/dist/test/src/tools/debug.js +0 -37
  563. package/dist/test/src/types/cache.d.ts +0 -8
  564. package/dist/test/src/types/cache.d.ts.map +0 -1
  565. package/dist/test/src/types/cache.js +0 -5
  566. package/dist/test/src/types/collection.d.ts +0 -23
  567. package/dist/test/src/types/collection.d.ts.map +0 -1
  568. package/dist/test/src/types/collection.js +0 -5
  569. package/dist/test/src/types/elements/IElement.d.ts +0 -123
  570. package/dist/test/src/types/elements/IElement.d.ts.map +0 -1
  571. package/dist/test/src/types/elements/IElement.js +0 -30
  572. package/dist/test/src/types/elements/IElementManager.d.ts +0 -65
  573. package/dist/test/src/types/elements/IElementManager.d.ts.map +0 -1
  574. package/dist/test/src/types/elements/IElementManager.js +0 -6
  575. package/dist/test/src/types/elements/IRatingManager.d.ts +0 -109
  576. package/dist/test/src/types/elements/IRatingManager.d.ts.map +0 -1
  577. package/dist/test/src/types/elements/IRatingManager.js +0 -6
  578. package/dist/test/src/types/elements/IReferenceResolver.d.ts +0 -52
  579. package/dist/test/src/types/elements/IReferenceResolver.d.ts.map +0 -1
  580. package/dist/test/src/types/elements/IReferenceResolver.js +0 -6
  581. package/dist/test/src/types/elements/RatingBreakdowns.d.ts +0 -49
  582. package/dist/test/src/types/elements/RatingBreakdowns.d.ts.map +0 -1
  583. package/dist/test/src/types/elements/RatingBreakdowns.js +0 -6
  584. package/dist/test/src/types/elements/index.d.ts +0 -9
  585. package/dist/test/src/types/elements/index.d.ts.map +0 -1
  586. package/dist/test/src/types/elements/index.js +0 -11
  587. package/dist/test/src/types/index.d.ts +0 -9
  588. package/dist/test/src/types/index.d.ts.map +0 -1
  589. package/dist/test/src/types/index.js +0 -9
  590. package/dist/test/src/types/marketplace.d.ts +0 -23
  591. package/dist/test/src/types/marketplace.d.ts.map +0 -1
  592. package/dist/test/src/types/marketplace.js +0 -5
  593. package/dist/test/src/types/mcp.d.ts +0 -84
  594. package/dist/test/src/types/mcp.d.ts.map +0 -1
  595. package/dist/test/src/types/mcp.js +0 -80
  596. package/dist/test/src/types/persona.d.ts +0 -30
  597. package/dist/test/src/types/persona.d.ts.map +0 -1
  598. package/dist/test/src/types/persona.js +0 -5
  599. package/dist/test/src/update/BackupManager.d.ts +0 -46
  600. package/dist/test/src/update/BackupManager.d.ts.map +0 -1
  601. package/dist/test/src/update/BackupManager.js +0 -261
  602. package/dist/test/src/update/DependencyChecker.d.ts +0 -41
  603. package/dist/test/src/update/DependencyChecker.d.ts.map +0 -1
  604. package/dist/test/src/update/DependencyChecker.js +0 -132
  605. package/dist/test/src/update/RateLimiter.d.ts +0 -80
  606. package/dist/test/src/update/RateLimiter.d.ts.map +0 -1
  607. package/dist/test/src/update/RateLimiter.js +0 -172
  608. package/dist/test/src/update/SignatureVerifier.d.ts +0 -71
  609. package/dist/test/src/update/SignatureVerifier.d.ts.map +0 -1
  610. package/dist/test/src/update/SignatureVerifier.js +0 -214
  611. package/dist/test/src/update/UpdateChecker.d.ts +0 -127
  612. package/dist/test/src/update/UpdateChecker.d.ts.map +0 -1
  613. package/dist/test/src/update/UpdateChecker.js +0 -469
  614. package/dist/test/src/update/UpdateManager.d.ts +0 -41
  615. package/dist/test/src/update/UpdateManager.d.ts.map +0 -1
  616. package/dist/test/src/update/UpdateManager.js +0 -260
  617. package/dist/test/src/update/VersionManager.d.ts +0 -31
  618. package/dist/test/src/update/VersionManager.d.ts.map +0 -1
  619. package/dist/test/src/update/VersionManager.js +0 -134
  620. package/dist/test/src/update/index.d.ts +0 -9
  621. package/dist/test/src/update/index.d.ts.map +0 -1
  622. package/dist/test/src/update/index.js +0 -9
  623. package/dist/test/src/utils/filesystem.d.ts +0 -29
  624. package/dist/test/src/utils/filesystem.d.ts.map +0 -1
  625. package/dist/test/src/utils/filesystem.js +0 -94
  626. package/dist/test/src/utils/git.d.ts +0 -32
  627. package/dist/test/src/utils/git.d.ts.map +0 -1
  628. package/dist/test/src/utils/git.js +0 -65
  629. package/dist/test/src/utils/index.d.ts +0 -7
  630. package/dist/test/src/utils/index.d.ts.map +0 -1
  631. package/dist/test/src/utils/index.js +0 -7
  632. package/dist/test/src/utils/logger.d.ts +0 -45
  633. package/dist/test/src/utils/logger.d.ts.map +0 -1
  634. package/dist/test/src/utils/logger.js +0 -91
  635. package/dist/test/src/utils/version.d.ts +0 -25
  636. package/dist/test/src/utils/version.d.ts.map +0 -1
  637. package/dist/test/src/utils/version.js +0 -97
  638. package/dist/test/test/__tests__/integration/helpers/file-utils.d.ts +0 -33
  639. package/dist/test/test/__tests__/integration/helpers/file-utils.d.ts.map +0 -1
  640. package/dist/test/test/__tests__/integration/helpers/file-utils.js +0 -83
  641. package/dist/test/test/__tests__/integration/helpers/test-fixtures.d.ts +0 -26
  642. package/dist/test/test/__tests__/integration/helpers/test-fixtures.d.ts.map +0 -1
  643. package/dist/test/test/__tests__/integration/helpers/test-fixtures.js +0 -95
  644. package/dist/test/test/__tests__/integration/helpers/test-server.d.ts +0 -26
  645. package/dist/test/test/__tests__/integration/helpers/test-server.d.ts.map +0 -1
  646. package/dist/test/test/__tests__/integration/helpers/test-server.js +0 -41
  647. package/dist/test/test/__tests__/integration/setup.d.ts +0 -8
  648. package/dist/test/test/__tests__/integration/setup.d.ts.map +0 -1
  649. package/dist/test/test/__tests__/integration/setup.js +0 -31
  650. package/dist/test/test/__tests__/integration/teardown.d.ts +0 -5
  651. package/dist/test/test/__tests__/integration/teardown.d.ts.map +0 -1
  652. package/dist/test/test/__tests__/integration/teardown.js +0 -23
  653. package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.d.ts +0 -34
  654. package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.d.ts.map +0 -1
  655. package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.js +0 -224
  656. package/dist/test/test/__tests__/security/framework/SecurityTestFramework.d.ts +0 -89
  657. package/dist/test/test/__tests__/security/framework/SecurityTestFramework.d.ts.map +0 -1
  658. package/dist/test/test/__tests__/security/framework/SecurityTestFramework.js +0 -543
  659. package/dist/test/test/__tests__/security/index.d.ts +0 -46
  660. package/dist/test/test/__tests__/security/index.d.ts.map +0 -1
  661. package/dist/test/test/__tests__/security/index.js +0 -98
  662. package/dist/test/test/__tests__/security/setup.d.ts +0 -3
  663. package/dist/test/test/__tests__/security/setup.d.ts.map +0 -1
  664. package/dist/test/test/__tests__/security/setup.js +0 -23
  665. package/dist/types/marketplace.d.ts +0 -23
  666. package/dist/types/marketplace.d.ts.map +0 -1
  667. package/dist/types/marketplace.js +0 -5
@@ -1,673 +0,0 @@
1
- /**
2
- * Template element class implementing IElement interface.
3
- * Represents reusable content structures with variable substitution and dynamic content.
4
- *
5
- * SECURITY FIXES IMPLEMENTED (Following PR #319 patterns):
6
- * 1. CRITICAL: Template injection prevention - no eval() or Function() constructor
7
- * 2. CRITICAL: Path traversal prevention for template includes
8
- * 3. HIGH: Input validation and sanitization for all template variables
9
- * 4. MEDIUM: Memory limits to prevent resource exhaustion (100KB templates, 100 variables)
10
- * 5. MEDIUM: Audit logging for all security operations via SecurityMonitor
11
- * 6. MEDIUM: Unicode normalization to prevent homograph attacks
12
- */
13
- import { BaseElement } from '../BaseElement.js';
14
- import { ElementType } from '../../portfolio/types.js';
15
- import { logger } from '../../utils/logger.js';
16
- import { sanitizeInput } from '../../security/InputValidator.js';
17
- import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
18
- import { SecurityMonitor } from '../../security/securityMonitor.js';
19
- import * as path from 'path';
20
- export class Template extends BaseElement {
21
- content;
22
- compiledTemplate;
23
- // SECURITY FIX #4: Memory management constants
24
- // Prevents unbounded template size and variable count that could exhaust memory
25
- MAX_TEMPLATE_SIZE = 100 * 1024; // 100KB max template size
26
- MAX_VARIABLE_COUNT = 100; // Max variables per template
27
- MAX_INCLUDE_DEPTH = 5; // Prevent infinite include loops
28
- MAX_STRING_LENGTH = 10000; // Max length for string variables
29
- constructor(metadata, content = '') {
30
- // SECURITY FIX #3 & #6: Validate and sanitize ALL metadata fields
31
- // Unicode normalization prevents homograph attacks
32
- // Input sanitization prevents XSS and injection attacks
33
- const sanitizedMetadata = {
34
- ...metadata,
35
- name: metadata.name ? sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) : undefined,
36
- description: metadata.description ? sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) : undefined,
37
- category: metadata.category ? sanitizeInput(UnicodeValidator.normalize(metadata.category).normalizedContent, 50) : undefined,
38
- output_format: metadata.output_format ? sanitizeInput(metadata.output_format, 20) : undefined
39
- };
40
- super(ElementType.TEMPLATE, sanitizedMetadata);
41
- // SECURITY FIX #4: Enforce template size limit
42
- if (content.length > this.MAX_TEMPLATE_SIZE) {
43
- SecurityMonitor.logSecurityEvent({
44
- type: 'CONTENT_SIZE_EXCEEDED',
45
- severity: 'HIGH',
46
- source: 'Template.constructor',
47
- details: `Template size ${content.length} exceeds maximum ${this.MAX_TEMPLATE_SIZE}`
48
- });
49
- throw new Error(`Template content exceeds maximum size of ${this.MAX_TEMPLATE_SIZE} bytes`);
50
- }
51
- // SECURITY FIX #3: Sanitize template content
52
- // Note: We preserve the template syntax but normalize Unicode
53
- this.content = UnicodeValidator.normalize(content).normalizedContent;
54
- // Ensure template-specific metadata
55
- this.metadata = {
56
- ...this.metadata,
57
- category: sanitizedMetadata.category || 'general',
58
- output_format: sanitizedMetadata.output_format || 'markdown',
59
- variables: metadata.variables || [],
60
- includes: metadata.includes || [],
61
- tags: metadata.tags || [],
62
- usage_count: metadata.usage_count || 0,
63
- examples: metadata.examples || []
64
- };
65
- // SECURITY FIX #3 & #4: Validate variables
66
- if (this.metadata.variables) {
67
- if (this.metadata.variables.length > this.MAX_VARIABLE_COUNT) {
68
- throw new Error(`Variable count ${this.metadata.variables.length} exceeds maximum ${this.MAX_VARIABLE_COUNT}`);
69
- }
70
- this.metadata.variables = this.metadata.variables.map(variable => ({
71
- ...variable,
72
- name: sanitizeInput(UnicodeValidator.normalize(variable.name).normalizedContent, 50),
73
- description: variable.description ? sanitizeInput(UnicodeValidator.normalize(variable.description).normalizedContent, 200) : undefined,
74
- validation: variable.validation ? sanitizeInput(variable.validation, 200) : undefined
75
- }));
76
- }
77
- // SECURITY FIX #2: Validate include paths
78
- if (this.metadata.includes) {
79
- this.metadata.includes = this.metadata.includes.map(includePath => {
80
- const sanitizedPath = sanitizeInput(includePath, 200);
81
- // Prevent path traversal attacks
82
- if (!this.isValidIncludePath(sanitizedPath)) {
83
- SecurityMonitor.logSecurityEvent({
84
- type: 'PATH_TRAVERSAL_ATTEMPT',
85
- severity: 'CRITICAL',
86
- source: 'Template.constructor',
87
- details: `Invalid include path: ${sanitizedPath}`
88
- });
89
- throw new Error(`Invalid include path: ${sanitizedPath}`);
90
- }
91
- return sanitizedPath;
92
- });
93
- }
94
- }
95
- /**
96
- * Validate include paths to prevent directory traversal
97
- * SECURITY FIX #2: Prevents accessing files outside template directory
98
- */
99
- isValidIncludePath(includePath) {
100
- // Normalize the path
101
- const normalized = path.normalize(includePath);
102
- // Check for path traversal patterns
103
- if (normalized.includes('..') || normalized.includes('~') || path.isAbsolute(normalized)) {
104
- return false;
105
- }
106
- // Only allow alphanumeric, dash, underscore, forward slash, backslash (for Windows), and .md extension
107
- // Note: We test against the original path to preserve cross-platform compatibility
108
- const validPathPattern = /^[a-zA-Z0-9\-_\/\\]+\.md$/;
109
- return validPathPattern.test(includePath);
110
- }
111
- /**
112
- * Compile the template for efficient rendering
113
- * SECURITY FIX #1: Safe template compilation without eval() or Function()
114
- * Uses regex-based token replacement instead of dynamic code execution
115
- */
116
- compile() {
117
- if (this.compiledTemplate) {
118
- return this.compiledTemplate;
119
- }
120
- // Extract all variable tokens from the template
121
- const variablePattern = /\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*(?:\.[a-zA-Z_][a-zA-Z0-9_]*)*)\s*\}\}/g;
122
- const tokens = [];
123
- let match;
124
- while ((match = variablePattern.exec(this.content)) !== null) {
125
- tokens.push({
126
- token: match[0],
127
- variable: match[1],
128
- position: match.index
129
- });
130
- }
131
- this.compiledTemplate = {
132
- content: this.content,
133
- tokens,
134
- variables: this.metadata.variables || []
135
- };
136
- return this.compiledTemplate;
137
- }
138
- /**
139
- * Render the template with provided variables
140
- * SECURITY FIX #1: Safe rendering without code execution
141
- * SECURITY FIX #3: All variables are validated and sanitized
142
- * TYPE SAFETY: Strong typing for variables with runtime validation
143
- */
144
- async render(variables = {}, includeDepth = 0) {
145
- // SECURITY FIX #4: Prevent infinite include loops
146
- if (includeDepth > this.MAX_INCLUDE_DEPTH) {
147
- SecurityMonitor.logSecurityEvent({
148
- type: 'INCLUDE_DEPTH_EXCEEDED',
149
- severity: 'HIGH',
150
- source: 'Template.render',
151
- details: `Include depth ${includeDepth} exceeds maximum ${this.MAX_INCLUDE_DEPTH}`
152
- });
153
- throw new Error('Maximum template include depth exceeded');
154
- }
155
- // Compile the template
156
- const compiled = this.compile();
157
- // Validate and sanitize all provided variables
158
- const sanitizedVariables = await this.validateAndSanitizeVariables(variables);
159
- // Start with the template content
160
- let rendered = compiled.content;
161
- // Replace tokens in reverse order to maintain positions
162
- const sortedTokens = [...compiled.tokens].sort((a, b) => b.position - a.position);
163
- for (const token of sortedTokens) {
164
- const value = this.resolveVariable(token.variable, sanitizedVariables);
165
- const stringValue = this.formatValue(value);
166
- // Replace the token with the sanitized value
167
- rendered = rendered.substring(0, token.position) +
168
- stringValue +
169
- rendered.substring(token.position + token.token.length);
170
- }
171
- // Process includes if any
172
- if (this.metadata.includes && this.metadata.includes.length > 0) {
173
- rendered = await this.processIncludes(rendered, sanitizedVariables, includeDepth);
174
- }
175
- // Update usage statistics
176
- // NOTE: These updates are not atomic and may have race conditions under concurrent access
177
- // This is acceptable for usage statistics which don't require perfect accuracy
178
- // For production systems requiring atomic counters, consider using a database or atomic operations
179
- this.metadata.usage_count = (this.metadata.usage_count || 0) + 1;
180
- this.metadata.last_used = new Date().toISOString();
181
- // SECURITY FIX #5: Log template usage for audit trail
182
- SecurityMonitor.logSecurityEvent({
183
- type: 'TEMPLATE_RENDERED',
184
- severity: 'LOW',
185
- source: 'Template.render',
186
- details: `Template ${this.metadata.name} rendered with ${Object.keys(sanitizedVariables).length} variables`
187
- });
188
- return rendered;
189
- }
190
- /**
191
- * Validate and sanitize variables according to their definitions
192
- * SECURITY FIX #3: Comprehensive validation of all input variables
193
- * TYPE SAFETY: Improved type safety for variable validation
194
- */
195
- async validateAndSanitizeVariables(variables) {
196
- const sanitized = {};
197
- // Check required variables
198
- for (const varDef of this.metadata.variables || []) {
199
- if (varDef.required && !(varDef.name in variables)) {
200
- if (varDef.default !== undefined) {
201
- sanitized[varDef.name] = varDef.default;
202
- }
203
- else {
204
- throw new Error(`Required variable '${varDef.name}' not provided`);
205
- }
206
- }
207
- }
208
- // Validate and sanitize provided variables
209
- for (const [name, value] of Object.entries(variables)) {
210
- const varDef = this.metadata.variables?.find(v => v.name === name);
211
- if (!varDef) {
212
- // Skip unknown variables (they won't be used anyway)
213
- logger.warn(`Unknown variable '${name}' provided to template`);
214
- continue;
215
- }
216
- // Type validation and sanitization
217
- const sanitizedValue = await this.sanitizeVariableValue(value, varDef);
218
- sanitized[name] = sanitizedValue;
219
- }
220
- // Apply defaults for missing optional variables
221
- for (const varDef of this.metadata.variables || []) {
222
- if (!varDef.required && !(varDef.name in sanitized) && varDef.default !== undefined) {
223
- sanitized[varDef.name] = varDef.default;
224
- }
225
- }
226
- return sanitized;
227
- }
228
- /**
229
- * Sanitize a single variable value according to its definition
230
- * SECURITY FIX #3 & #6: Type-specific validation and Unicode normalization
231
- */
232
- async sanitizeVariableValue(value, varDef) {
233
- switch (varDef.type) {
234
- case 'string':
235
- // SECURITY FIX #6: Unicode normalization
236
- const normalized = UnicodeValidator.normalize(String(value));
237
- let stringValue = sanitizeInput(normalized.normalizedContent, this.MAX_STRING_LENGTH);
238
- // Apply regex validation if specified
239
- if (varDef.validation) {
240
- // SECURITY FIX: Validate regex complexity to prevent ReDoS attacks
241
- // Previously: User-provided regex executed without limits
242
- // Now: Check for dangerous patterns and limit execution time
243
- try {
244
- // Check for dangerous regex patterns
245
- if (this.isDangerousRegex(varDef.validation)) {
246
- throw new Error(`Variable '${varDef.name}' has potentially dangerous validation pattern`);
247
- }
248
- const regex = new RegExp(varDef.validation);
249
- // Use a simple timeout mechanism - in production, consider using a worker thread
250
- const startTime = Date.now();
251
- const result = regex.test(stringValue);
252
- const duration = Date.now() - startTime;
253
- // If regex takes too long, it might be malicious
254
- if (duration > 100) { // 100ms threshold
255
- SecurityMonitor.logSecurityEvent({
256
- type: 'CONTENT_INJECTION_ATTEMPT',
257
- severity: 'HIGH',
258
- source: 'Template.sanitizeVariableValue',
259
- details: `Regex validation took ${duration}ms for variable '${varDef.name}', possible ReDoS`
260
- });
261
- throw new Error(`Variable '${varDef.name}' validation pattern is too complex`);
262
- }
263
- if (!result) {
264
- throw new Error(`Variable '${varDef.name}' does not match validation pattern`);
265
- }
266
- }
267
- catch (e) {
268
- if (e instanceof SyntaxError) {
269
- throw new Error(`Variable '${varDef.name}' has invalid validation pattern`);
270
- }
271
- throw e;
272
- }
273
- }
274
- // Check enum options if specified
275
- if (varDef.options && !varDef.options.includes(stringValue)) {
276
- throw new Error(`Variable '${varDef.name}' must be one of: ${varDef.options.join(', ')}`);
277
- }
278
- return stringValue;
279
- case 'number':
280
- const num = Number(value);
281
- if (isNaN(num)) {
282
- throw new Error(`Variable '${varDef.name}' must be a number`);
283
- }
284
- return num;
285
- case 'boolean':
286
- return Boolean(value);
287
- case 'date':
288
- const date = new Date(value);
289
- if (isNaN(date.getTime())) {
290
- throw new Error(`Variable '${varDef.name}' must be a valid date`);
291
- }
292
- return date;
293
- case 'array':
294
- if (!Array.isArray(value)) {
295
- throw new Error(`Variable '${varDef.name}' must be an array`);
296
- }
297
- // SECURITY FIX: Limit array size to prevent memory exhaustion attacks
298
- // Previously: No limit on array size could lead to DoS
299
- // Now: Enforces reasonable size limit with logging
300
- const MAX_ARRAY_SIZE = 1000;
301
- if (value.length > MAX_ARRAY_SIZE) {
302
- SecurityMonitor.logSecurityEvent({
303
- type: 'CONTENT_SIZE_EXCEEDED',
304
- severity: 'MEDIUM',
305
- source: 'Template.sanitizeVariableValue',
306
- details: `Array variable '${varDef.name}' has ${value.length} items, limiting to ${MAX_ARRAY_SIZE}`
307
- });
308
- value = value.slice(0, MAX_ARRAY_SIZE);
309
- }
310
- // Sanitize string elements in arrays
311
- return value.map((item) => typeof item === 'string' ? sanitizeInput(item, 1000) : item);
312
- case 'object':
313
- if (typeof value !== 'object' || value === null) {
314
- throw new Error(`Variable '${varDef.name}' must be an object`);
315
- }
316
- // Deep sanitize string values in objects
317
- return this.sanitizeObject(value);
318
- default:
319
- return value;
320
- }
321
- }
322
- /**
323
- * Recursively sanitize string values in objects
324
- * SECURITY FIX: Truncate deep nesting instead of throwing to prevent DoS
325
- */
326
- sanitizeObject(obj, depth = 0) {
327
- // SECURITY FIX: Return safe default instead of throwing to prevent DoS attacks
328
- // Previously: Threw error on deep nesting which could be exploited
329
- // Now: Returns string representation for excessively nested objects
330
- if (depth > 10) {
331
- SecurityMonitor.logSecurityEvent({
332
- type: 'CONTENT_SIZE_EXCEEDED',
333
- severity: 'MEDIUM',
334
- source: 'Template.sanitizeObject',
335
- details: 'Object nesting depth exceeded, truncating to string representation'
336
- });
337
- return '[Object too deeply nested]';
338
- }
339
- if (typeof obj !== 'object' || obj === null) {
340
- return obj;
341
- }
342
- if (Array.isArray(obj)) {
343
- // SECURITY FIX: Limit array size to prevent memory exhaustion
344
- const MAX_ARRAY_SIZE = 1000;
345
- if (obj.length > MAX_ARRAY_SIZE) {
346
- SecurityMonitor.logSecurityEvent({
347
- type: 'CONTENT_SIZE_EXCEEDED',
348
- severity: 'MEDIUM',
349
- source: 'Template.sanitizeObject',
350
- details: `Array size ${obj.length} exceeds maximum ${MAX_ARRAY_SIZE}, truncating`
351
- });
352
- obj = obj.slice(0, MAX_ARRAY_SIZE);
353
- }
354
- return obj.map((item) => this.sanitizeObject(item, depth + 1));
355
- }
356
- const sanitized = {};
357
- // SECURITY FIX: Limit number of object properties to prevent memory exhaustion
358
- const MAX_OBJECT_KEYS = 100;
359
- const entries = Object.entries(obj);
360
- if (entries.length > MAX_OBJECT_KEYS) {
361
- SecurityMonitor.logSecurityEvent({
362
- type: 'CONTENT_SIZE_EXCEEDED',
363
- severity: 'MEDIUM',
364
- source: 'Template.sanitizeObject',
365
- details: `Object has ${entries.length} keys, limiting to ${MAX_OBJECT_KEYS}`
366
- });
367
- }
368
- for (let i = 0; i < Math.min(entries.length, MAX_OBJECT_KEYS); i++) {
369
- const [key, value] = entries[i];
370
- const sanitizedKey = sanitizeInput(key, 50);
371
- if (typeof value === 'string') {
372
- sanitized[sanitizedKey] = sanitizeInput(value, 1000);
373
- }
374
- else if (typeof value === 'object') {
375
- sanitized[sanitizedKey] = this.sanitizeObject(value, depth + 1);
376
- }
377
- else {
378
- sanitized[sanitizedKey] = value;
379
- }
380
- }
381
- return sanitized;
382
- }
383
- /**
384
- * Resolve nested variable paths (e.g., "user.name")
385
- * TYPE SAFETY: Improved type safety for variable resolution
386
- */
387
- resolveVariable(path, variables) {
388
- const parts = path.split('.');
389
- let value = variables;
390
- for (const part of parts) {
391
- if (value && typeof value === 'object' && value !== null && part in value) {
392
- // Type assertion with runtime check - safe because we verified the property exists
393
- value = value[part];
394
- }
395
- else {
396
- return undefined;
397
- }
398
- }
399
- return value;
400
- }
401
- /**
402
- * Check if a regex pattern is potentially dangerous (ReDoS)
403
- * SECURITY FIX: Detect patterns that could cause exponential backtracking
404
- */
405
- isDangerousRegex(pattern) {
406
- // Check for nested quantifiers which can cause exponential backtracking
407
- const dangerousPatterns = [
408
- /(\+|\*){2,}/, // Multiple quantifiers
409
- /\([^)]*\+\)[+*]/, // Quantified groups with quantifiers inside
410
- /\[[^\]]*\+\][+*]/, // Quantified character classes with quantifiers
411
- /(\\[dws])\1{2,}/, // Repeated character classes
412
- /\(\?\<[!=][^)]+\)/, // Complex lookbehinds
413
- ];
414
- for (const dangerous of dangerousPatterns) {
415
- if (dangerous.test(pattern)) {
416
- return true;
417
- }
418
- }
419
- // Check for excessive backtracking potential
420
- // Count groups and quantifiers
421
- const groups = (pattern.match(/\(/g) || []).length;
422
- const quantifiers = (pattern.match(/[+*?{]/g) || []).length;
423
- // If there are many groups and quantifiers, it's potentially dangerous
424
- if (groups > 5 && quantifiers > 5) {
425
- return true;
426
- }
427
- return false;
428
- }
429
- /**
430
- * Format a value for template output
431
- */
432
- formatValue(value) {
433
- if (value === undefined || value === null) {
434
- return '';
435
- }
436
- if (value instanceof Date) {
437
- return value.toISOString();
438
- }
439
- if (Array.isArray(value)) {
440
- return value.join(', ');
441
- }
442
- if (typeof value === 'object') {
443
- return JSON.stringify(value, null, 2);
444
- }
445
- return String(value);
446
- }
447
- /**
448
- * Process template includes
449
- * SECURITY FIX #2: Safe include processing with path validation
450
- *
451
- * TODO: Implement actual include processing functionality
452
- * This is currently a placeholder that validates the security model
453
- * but does not actually load and render included templates.
454
- *
455
- * Future implementation should:
456
- * 1. Load templates from validated paths
457
- * 2. Render them with current variables
458
- * 3. Replace include markers in content
459
- * 4. Respect includeDepth limit
460
- */
461
- async processIncludes(content, variables, includeDepth) {
462
- // TODO: Implement actual template include processing
463
- // Current implementation only validates the security model
464
- if (!this.metadata.includes || this.metadata.includes.length === 0) {
465
- return content;
466
- }
467
- // Log security event for audit trail
468
- SecurityMonitor.logSecurityEvent({
469
- type: 'TEMPLATE_INCLUDE',
470
- severity: 'LOW',
471
- source: 'Template.processIncludes',
472
- details: `Processing ${this.metadata.includes.length} includes at depth ${includeDepth}`
473
- });
474
- // TODO: Future implementation would:
475
- // for (const includePath of this.metadata.includes) {
476
- // const template = await this.loadIncludedTemplate(includePath);
477
- // const rendered = await template.render(variables, includeDepth + 1);
478
- // content = content.replace(`{{include:${includePath}}}`, rendered);
479
- // }
480
- return content;
481
- }
482
- /**
483
- * Template-specific validation
484
- */
485
- validate() {
486
- const result = super.validate();
487
- // Initialize arrays if not present
488
- if (!result.errors)
489
- result.errors = [];
490
- if (!result.warnings)
491
- result.warnings = [];
492
- // Content validation
493
- if (!this.content || this.content.trim().length === 0) {
494
- result.errors.push({
495
- field: 'content',
496
- message: 'Template content cannot be empty',
497
- code: 'EMPTY_CONTENT'
498
- });
499
- }
500
- // Check for unmatched tokens
501
- const openTokens = (this.content.match(/\{\{/g) || []).length;
502
- const closeTokens = (this.content.match(/\}\}/g) || []).length;
503
- if (openTokens !== closeTokens) {
504
- result.errors.push({
505
- field: 'content',
506
- message: 'Template has unmatched variable tokens',
507
- code: 'UNMATCHED_TOKENS'
508
- });
509
- }
510
- // Validate output format
511
- const validFormats = ['markdown', 'html', 'json', 'yaml', 'text', 'xml'];
512
- if (this.metadata.output_format && !validFormats.includes(this.metadata.output_format)) {
513
- result.warnings.push({
514
- field: 'output_format',
515
- message: `Unknown output format '${this.metadata.output_format}'. Common formats: ${validFormats.join(', ')}`,
516
- severity: 'low'
517
- });
518
- }
519
- // Validate variables
520
- if (this.metadata.variables) {
521
- const variableNames = new Set();
522
- this.metadata.variables.forEach((variable, index) => {
523
- // Check for duplicate names
524
- if (variableNames.has(variable.name)) {
525
- result.errors.push({
526
- field: `variables[${index}].name`,
527
- message: `Duplicate variable name '${variable.name}'`,
528
- code: 'DUPLICATE_VARIABLE'
529
- });
530
- }
531
- variableNames.add(variable.name);
532
- // Validate regex patterns
533
- if (variable.validation) {
534
- try {
535
- new RegExp(variable.validation);
536
- }
537
- catch (e) {
538
- result.errors.push({
539
- field: `variables[${index}].validation`,
540
- message: `Invalid regex pattern: ${e}`,
541
- code: 'INVALID_REGEX'
542
- });
543
- }
544
- }
545
- });
546
- }
547
- // Check if all tokens have corresponding variable definitions
548
- const compiled = this.compile();
549
- const definedVars = new Set(this.metadata.variables?.map(v => v.name) || []);
550
- const usedVars = new Set(compiled.tokens.map(t => t.variable.split('.')[0]));
551
- usedVars.forEach(varName => {
552
- if (!definedVars.has(varName)) {
553
- result.warnings.push({
554
- field: 'variables',
555
- message: `Template uses undefined variable '${varName}'`,
556
- severity: 'medium'
557
- });
558
- }
559
- });
560
- // Warnings for best practices
561
- if (!this.metadata.tags || this.metadata.tags.length === 0) {
562
- result.warnings.push({
563
- field: 'tags',
564
- message: 'Consider adding tags for better searchability',
565
- severity: 'low'
566
- });
567
- }
568
- if (!this.metadata.examples || this.metadata.examples.length === 0) {
569
- result.warnings.push({
570
- field: 'examples',
571
- message: 'Adding examples improves template usability',
572
- severity: 'medium'
573
- });
574
- }
575
- // Update the valid flag based on final errors
576
- result.valid = (result.errors?.length || 0) === 0;
577
- return result;
578
- }
579
- /**
580
- * Serialize template to JSON format
581
- */
582
- serialize() {
583
- const data = {
584
- id: this.id,
585
- type: this.type,
586
- version: this.version,
587
- metadata: this.metadata,
588
- content: this.content,
589
- references: this.references,
590
- extensions: this.extensions,
591
- ratings: this.ratings
592
- };
593
- return JSON.stringify(data, null, 2);
594
- }
595
- /**
596
- * Deserialize template from JSON format
597
- */
598
- deserialize(data) {
599
- try {
600
- const parsed = JSON.parse(data);
601
- // Update metadata
602
- this.metadata = { ...this.metadata, ...parsed.metadata };
603
- // Update content
604
- this.content = parsed.content || '';
605
- // Update other properties
606
- this.references = parsed.references || [];
607
- this.extensions = parsed.extensions || {};
608
- this.ratings = parsed.ratings || this.ratings;
609
- // Update ID and version if provided
610
- if (parsed.id)
611
- this.id = parsed.id;
612
- if (parsed.version)
613
- this.version = parsed.version;
614
- // Clear compiled template cache
615
- this.compiledTemplate = undefined;
616
- this._isDirty = true;
617
- logger.debug(`Deserialized template: ${this.metadata.name}`);
618
- }
619
- catch (error) {
620
- logger.error(`Failed to deserialize template: ${error}`);
621
- throw new Error(`Deserialization failed: ${error}`);
622
- }
623
- }
624
- /**
625
- * Get a preview of the template with sample data
626
- */
627
- async preview() {
628
- const sampleVars = {};
629
- // Generate sample data for each variable
630
- for (const varDef of this.metadata.variables || []) {
631
- switch (varDef.type) {
632
- case 'string':
633
- sampleVars[varDef.name] = varDef.default || `[${varDef.name}]`;
634
- break;
635
- case 'number':
636
- sampleVars[varDef.name] = varDef.default || 42;
637
- break;
638
- case 'boolean':
639
- sampleVars[varDef.name] = varDef.default || true;
640
- break;
641
- case 'date':
642
- sampleVars[varDef.name] = varDef.default || new Date();
643
- break;
644
- case 'array':
645
- sampleVars[varDef.name] = varDef.default || ['item1', 'item2'];
646
- break;
647
- case 'object':
648
- sampleVars[varDef.name] = varDef.default || { key: 'value' };
649
- break;
650
- }
651
- }
652
- return this.render(sampleVars);
653
- }
654
- /**
655
- * Template activation lifecycle
656
- */
657
- async activate() {
658
- logger.info(`Activating template: ${this.metadata.name} (${this.id})`);
659
- // Compile the template to check for errors
660
- this.compile();
661
- await super.activate?.();
662
- }
663
- /**
664
- * Template deactivation lifecycle
665
- */
666
- async deactivate() {
667
- logger.info(`Deactivating template: ${this.metadata.name} (${this.id})`);
668
- // Clear compiled template cache
669
- this.compiledTemplate = undefined;
670
- await super.deactivate?.();
671
- }
672
- }
673
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVGVtcGxhdGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvZWxlbWVudHMvdGVtcGxhdGVzL1RlbXBsYXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7OztHQVdHO0FBRUgsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBRWhELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUN2RCxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGFBQWEsRUFBZ0IsTUFBTSxrQ0FBa0MsQ0FBQztBQUMvRSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSwrQ0FBK0MsQ0FBQztBQUNqRixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sbUNBQW1DLENBQUM7QUFDcEUsT0FBTyxLQUFLLElBQUksTUFBTSxNQUFNLENBQUM7QUFnQzdCLE1BQU0sT0FBTyxRQUFTLFNBQVEsV0FBVztJQUVoQyxPQUFPLENBQVM7SUFDZixnQkFBZ0IsQ0FBb0I7SUFFNUMsK0NBQStDO0lBQy9DLGdGQUFnRjtJQUMvRCxpQkFBaUIsR0FBRyxHQUFHLEdBQUcsSUFBSSxDQUFDLENBQUMsMEJBQTBCO0lBQzFELGtCQUFrQixHQUFHLEdBQUcsQ0FBQyxDQUFPLDZCQUE2QjtJQUM3RCxpQkFBaUIsR0FBRyxDQUFDLENBQUMsQ0FBVSxpQ0FBaUM7SUFDakUsaUJBQWlCLEdBQUcsS0FBSyxDQUFDLENBQU0sa0NBQWtDO0lBRW5GLFlBQVksUUFBbUMsRUFBRSxVQUFrQixFQUFFO1FBQ25FLGtFQUFrRTtRQUNsRSxtREFBbUQ7UUFDbkQsd0RBQXdEO1FBQ3hELE1BQU0saUJBQWlCLEdBQUc7WUFDeEIsR0FBRyxRQUFRO1lBQ1gsSUFBSSxFQUFFLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLGFBQWEsQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLENB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O29CQUMvRCxNQUFNO2dCQUNSLEtBQUssUUFBUTtvQkFDWCxVQUFVLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLE1BQU0sQ0FBQyxPQUFPLElBQUksRUFBRSxHQUFHLEVBQUUsT0FBTyxFQUFFLENBQUM7b0JBQzdELE1BQU07WUFDVixDQUFDO1FBQ0gsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBRUQ7O09BRUc7SUFDYSxLQUFLLENBQUMsUUFBUTtRQUM1QixNQUFNLENBQUMsSUFBSSxDQUFDLHdCQUF3QixJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksS0FBSyxJQUFJLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUV2RSwyQ0FBMkM7UUFDM0MsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBRWYsTUFBTSxLQUFLLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FBQztJQUMzQixDQUFDO0lBRUQ7O09BRUc7SUFDYSxLQUFLLENBQUMsVUFBVTtRQUM5QixNQUFNLENBQUMsSUFBSSxDQUFDLDBCQUEwQixJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksS0FBSyxJQUFJLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUV6RSxnQ0FBZ0M7UUFDaEMsSUFBSSxDQUFDLGdCQUFnQixHQUFHLFNBQVMsQ0FBQztRQUVsQyxNQUFNLEtBQUssQ0FBQyxVQUFVLEVBQUUsRUFBRSxDQUFDO0lBQzdCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVGVtcGxhdGUgZWxlbWVudCBjbGFzcyBpbXBsZW1lbnRpbmcgSUVsZW1lbnQgaW50ZXJmYWNlLlxuICogUmVwcmVzZW50cyByZXVzYWJsZSBjb250ZW50IHN0cnVjdHVyZXMgd2l0aCB2YXJpYWJsZSBzdWJzdGl0dXRpb24gYW5kIGR5bmFtaWMgY29udGVudC5cbiAqIFxuICogU0VDVVJJVFkgRklYRVMgSU1QTEVNRU5URUQgKEZvbGxvd2luZyBQUiAjMzE5IHBhdHRlcm5zKTpcbiAqIDEuIENSSVRJQ0FMOiBUZW1wbGF0ZSBpbmplY3Rpb24gcHJldmVudGlvbiAtIG5vIGV2YWwoKSBvciBGdW5jdGlvbigpIGNvbnN0cnVjdG9yXG4gKiAyLiBDUklUSUNBTDogUGF0aCB0cmF2ZXJzYWwgcHJldmVudGlvbiBmb3IgdGVtcGxhdGUgaW5jbHVkZXNcbiAqIDMuIEhJR0g6IElucHV0IHZhbGlkYXRpb24gYW5kIHNhbml0aXphdGlvbiBmb3IgYWxsIHRlbXBsYXRlIHZhcmlhYmxlc1xuICogNC4gTUVESVVNOiBNZW1vcnkgbGltaXRzIHRvIHByZXZlbnQgcmVzb3VyY2UgZXhoYXVzdGlvbiAoMTAwS0IgdGVtcGxhdGVzLCAxMDAgdmFyaWFibGVzKVxuICogNS4gTUVESVVNOiBBdWRpdCBsb2dnaW5nIGZvciBhbGwgc2VjdXJpdHkgb3BlcmF0aW9ucyB2aWEgU2VjdXJpdHlNb25pdG9yXG4gKiA2LiBNRURJVU06IFVuaWNvZGUgbm9ybWFsaXphdGlvbiB0byBwcmV2ZW50IGhvbW9ncmFwaCBhdHRhY2tzXG4gKi9cblxuaW1wb3J0IHsgQmFzZUVsZW1lbnQgfSBmcm9tICcuLi9CYXNlRWxlbWVudC5qcyc7XG5pbXBvcnQgeyBJRWxlbWVudCwgSUVsZW1lbnRNZXRhZGF0YSwgRWxlbWVudFZhbGlkYXRpb25SZXN1bHQgfSBmcm9tICcuLi8uLi90eXBlcy9lbGVtZW50cy9pbmRleC5qcyc7XG5pbXBvcnQgeyBFbGVtZW50VHlwZSB9IGZyb20gJy4uLy4uL3BvcnRmb2xpby90eXBlcy5qcyc7XG5pbXBvcnQgeyBsb2dnZXIgfSBmcm9tICcuLi8uLi91dGlscy9sb2dnZXIuanMnO1xuaW1wb3J0IHsgc2FuaXRpemVJbnB1dCwgdmFsaWRhdGVQYXRoIH0gZnJvbSAnLi4vLi4vc2VjdXJpdHkvSW5wdXRWYWxpZGF0b3IuanMnO1xuaW1wb3J0IHsgVW5pY29kZVZhbGlkYXRvciB9IGZyb20gJy4uLy4uL3NlY3VyaXR5L3ZhbGlkYXRvcnMvdW5pY29kZVZhbGlkYXRvci5qcyc7XG5pbXBvcnQgeyBTZWN1cml0eU1vbml0b3IgfSBmcm9tICcuLi8uLi9zZWN1cml0eS9zZWN1cml0eU1vbml0b3IuanMnO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcblxuLy8gRXh0ZW5kIElFbGVtZW50TWV0YWRhdGEgd2l0aCB0ZW1wbGF0ZS1zcGVjaWZpYyBmaWVsZHNcbmV4cG9ydCBpbnRlcmZhY2UgVGVtcGxhdGVNZXRhZGF0YSBleHRlbmRzIElFbGVtZW50TWV0YWRhdGEge1xuICBjYXRlZ29yeT86IHN0cmluZzsgICAgICAgICAgICAgIC8vIFRlbXBsYXRlIGNhdGVnb3J5IChkb2N1bWVudHMsIGVtYWlscywgY29kZSwgZXRjLilcbiAgb3V0cHV0X2Zvcm1hdD86IHN0cmluZzsgICAgICAgICAvLyBPdXRwdXQgZm9ybWF0IChtYXJrZG93biwgaHRtbCwganNvbiwgeWFtbCwgZXRjLilcbiAgdmFyaWFibGVzPzogVGVtcGxhdGVWYXJpYWJsZVtdOyAvLyBWYXJpYWJsZSBkZWZpbml0aW9uc1xuICBpbmNsdWRlcz86IHN0cmluZ1tdOyAgICAgICAgICAgIC8vIE90aGVyIHRlbXBsYXRlcyB0byBpbmNsdWRlXG4gIHRhZ3M/OiBzdHJpbmdbXTsgICAgICAgICAgICAgICAvLyBTZWFyY2hhYmxlIHRhZ3NcbiAgdXNhZ2VfY291bnQ/OiBudW1iZXI7ICAgICAgICAgIC8vIFRyYWNrIHBvcHVsYXJpdHlcbiAgbGFzdF91c2VkPzogc3RyaW5nOyAgICAgICAgICAgIC8vIElTTyBkYXRlIHN0cmluZ1xuICBleGFtcGxlcz86IFRlbXBsYXRlRXhhbXBsZVtdOyAgIC8vIFVzYWdlIGV4YW1wbGVzXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGVtcGxhdGVWYXJpYWJsZSB7XG4gIG5hbWU6IHN0cmluZzsgICAgICAgICAgICAgICAgICAgLy8gVmFyaWFibGUgbmFtZSAoZS5nLiwgXCJwcm9qZWN0X25hbWVcIilcbiAgdHlwZTogJ3N0cmluZycgfCAnbnVtYmVyJyB8ICdib29sZWFuJyB8ICdkYXRlJyB8ICdhcnJheScgfCAnb2JqZWN0JztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7ICAgICAgICAgICAvLyBIZWxwIHRleHQgZm9yIHRoZSB2YXJpYWJsZVxuICByZXF1aXJlZD86IGJvb2xlYW47ICAgICAgICAgICAgIC8vIElzIHRoaXMgdmFyaWFibGUgcmVxdWlyZWQ/XG4gIGRlZmF1bHQ/OiB1bmtub3duOyAgICAgICAgICAgICAgLy8gRGVmYXVsdCB2YWx1ZSBpZiBub3QgcHJvdmlkZWQgKHR5cGUtc2FmZSlcbiAgdmFsaWRhdGlvbj86IHN0cmluZzsgICAgICAgICAgICAvLyBSZWdleCBwYXR0ZXJuIGZvciB2YWxpZGF0aW9uIChzdHJpbmcgdHlwZSBvbmx5KVxuICBvcHRpb25zPzogc3RyaW5nW107ICAgICAgICAgICAgIC8vIEZvciBlbnVtLWxpa2Ugc3RyaW5nc1xuICBmb3JtYXQ/OiBzdHJpbmc7ICAgICAgICAgICAgICAgIC8vIERhdGUgZm9ybWF0IHN0cmluZyAoZGF0ZSB0eXBlIG9ubHkpXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGVtcGxhdGVFeGFtcGxlIHtcbiAgdGl0bGU6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHZhcmlhYmxlczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIG91dHB1dD86IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIFRlbXBsYXRlIGV4dGVuZHMgQmFzZUVsZW1lbnQgaW1wbGVtZW50cyBJRWxlbWVudCB7XG4gIHB1YmxpYyBkZWNsYXJlIG1ldGFkYXRhOiBUZW1wbGF0ZU1ldGFkYXRhO1xuICBwdWJsaWMgY29udGVudDogc3RyaW5nO1xuICBwcml2YXRlIGNvbXBpbGVkVGVtcGxhdGU/OiBDb21waWxlZFRlbXBsYXRlO1xuICBcbiAgLy8gU0VDVVJJVFkgRklYICM0OiBNZW1vcnkgbWFuYWdlbWVudCBjb25zdGFudHNcbiAgLy8gUHJldmVudHMgdW5ib3VuZGVkIHRlbXBsYXRlIHNpemUgYW5kIHZhcmlhYmxlIGNvdW50IHRoYXQgY291bGQgZXhoYXVzdCBtZW1vcnlcbiAgcHJpdmF0ZSByZWFkb25seSBNQVhfVEVNUExBVEVfU0laRSA9IDEwMCAqIDEwMjQ7IC8vIDEwMEtCIG1heCB0ZW1wbGF0ZSBzaXplXG4gIHByaXZhdGUgcmVhZG9ubHkgTUFYX1ZBUklBQkxFX0NPVU5UID0gMTAwOyAgICAgICAvLyBNYXggdmFyaWFibGVzIHBlciB0ZW1wbGF0ZVxuICBwcml2YXRlIHJlYWRvbmx5IE1BWF9JTkNMVURFX0RFUFRIID0gNTsgICAgICAgICAgLy8gUHJldmVudCBpbmZpbml0ZSBpbmNsdWRlIGxvb3BzXG4gIHByaXZhdGUgcmVhZG9ubHkgTUFYX1NUUklOR19MRU5HVEggPSAxMDAwMDsgICAgICAvLyBNYXggbGVuZ3RoIGZvciBzdHJpbmcgdmFyaWFibGVzXG5cbiAgY29uc3RydWN0b3IobWV0YWRhdGE6IFBhcnRpYWw8VGVtcGxhdGVNZXRhZGF0YT4sIGNvbnRlbnQ6IHN0cmluZyA9ICcnKSB7XG4gICAgLy8gU0VDVVJJVFkgRklYICMzICYgIzY6IFZhbGlkYXRlIGFuZCBzYW5pdGl6ZSBBTEwgbWV0YWRhdGEgZmllbGRzXG4gICAgLy8gVW5pY29kZSBub3JtYWxpemF0aW9uIHByZXZlbnRzIGhvbW9ncmFwaCBhdHRhY2tzXG4gICAgLy8gSW5wdXQgc2FuaXRpemF0aW9uIHByZXZlbnRzIFhTUyBhbmQgaW5qZWN0aW9uIGF0dGFja3NcbiAgICBjb25zdCBzYW5pdGl6ZWRNZXRhZGF0YSA9IHtcbiAgICAgIC4uLm1ldGFkYXRhLFxuICAgICAgbmFtZTogbWV0YWRhdGEubmFtZSA/IHNhbml0aXplSW5wdXQoVW5pY29kZVZhbGlkYXRvci5ub3JtYWxpemUobWV0YWRhdGEubmFtZSkubm9ybWFsaXplZENvbnRlbnQsIDEwMCkgOiB1bmRlZmluZWQsXG4gICAgICBkZXNjcmlwdGlvbjogbWV0YWRhdGEuZGVzY3JpcHRpb24gPyBzYW5pdGl6ZUlucHV0KFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKG1ldGFkYXRhLmRlc2NyaXB0aW9uKS5ub3JtYWxpemVkQ29udGVudCwgNTAwKSA6IHVuZGVmaW5lZCxcbiAgICAgIGNhdGVnb3J5OiBtZXRhZGF0YS5jYXRlZ29yeSA/IHNhbml0aXplSW5wdXQoVW5pY29kZVZhbGlkYXRvci5ub3JtYWxpemUobWV0YWRhdGEuY2F0ZWdvcnkpLm5vcm1hbGl6ZWRDb250ZW50LCA1MCkgOiB1bmRlZmluZWQsXG4gICAgICBvdXRwdXRfZm9ybWF0OiBtZXRhZGF0YS5vdXRwdXRfZm9ybWF0ID8gc2FuaXRpemVJbnB1dChtZXRhZGF0YS5vdXRwdXRfZm9ybWF0LCAyMCkgOiB1bmRlZmluZWRcbiAgICB9O1xuICAgIFxuICAgIHN1cGVyKEVsZW1lbnRUeXBlLlRFTVBMQVRFLCBzYW5pdGl6ZWRNZXRhZGF0YSk7XG4gICAgXG4gICAgLy8gU0VDVVJJVFkgRklYICM0OiBFbmZvcmNlIHRlbXBsYXRlIHNpemUgbGltaXRcbiAgICBpZiAoY29udGVudC5sZW5ndGggPiB0aGlzLk1BWF9URU1QTEFURV9TSVpFKSB7XG4gICAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICAgIHR5cGU6ICdDT05URU5UX1NJWkVfRVhDRUVERUQnLFxuICAgICAgICBzZXZlcml0eTogJ0hJR0gnLFxuICAgICAgICBzb3VyY2U6ICdUZW1wbGF0ZS5jb25zdHJ1Y3RvcicsXG4gICAgICAgIGRldGFpbHM6IGBUZW1wbGF0ZSBzaXplICR7Y29udGVudC5sZW5ndGh9IGV4Y2VlZHMgbWF4aW11bSAke3RoaXMuTUFYX1RFTVBMQVRFX1NJWkV9YFxuICAgICAgfSk7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFRlbXBsYXRlIGNvbnRlbnQgZXhjZWVkcyBtYXhpbXVtIHNpemUgb2YgJHt0aGlzLk1BWF9URU1QTEFURV9TSVpFfSBieXRlc2ApO1xuICAgIH1cbiAgICBcbiAgICAvLyBTRUNVUklUWSBGSVggIzM6IFNhbml0aXplIHRlbXBsYXRlIGNvbnRlbnRcbiAgICAvLyBOb3RlOiBXZSBwcmVzZXJ2ZSB0aGUgdGVtcGxhdGUgc3ludGF4IGJ1dCBub3JtYWxpemUgVW5pY29kZVxuICAgIHRoaXMuY29udGVudCA9IFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKGNvbnRlbnQpLm5vcm1hbGl6ZWRDb250ZW50O1xuICAgIFxuICAgIC8vIEVuc3VyZSB0ZW1wbGF0ZS1zcGVjaWZpYyBtZXRhZGF0YVxuICAgIHRoaXMubWV0YWRhdGEgPSB7XG4gICAgICAuLi50aGlzLm1ldGFkYXRhLFxuICAgICAgY2F0ZWdvcnk6IHNhbml0aXplZE1ldGFkYXRhLmNhdGVnb3J5IHx8ICdnZW5lcmFsJyxcbiAgICAgIG91dHB1dF9mb3JtYXQ6IHNhbml0aXplZE1ldGFkYXRhLm91dHB1dF9mb3JtYXQgfHwgJ21hcmtkb3duJyxcbiAgICAgIHZhcmlhYmxlczogbWV0YWRhdGEudmFyaWFibGVzIHx8IFtdLFxuICAgICAgaW5jbHVkZXM6IG1ldGFkYXRhLmluY2x1ZGVzIHx8IFtdLFxuICAgICAgdGFnczogbWV0YWRhdGEudGFncyB8fCBbXSxcbiAgICAgIHVzYWdlX2NvdW50OiBtZXRhZGF0YS51c2FnZV9jb3VudCB8fCAwLFxuICAgICAgZXhhbXBsZXM6IG1ldGFkYXRhLmV4YW1wbGVzIHx8IFtdXG4gICAgfTtcblxuICAgIC8vIFNFQ1VSSVRZIEZJWCAjMyAmICM0OiBWYWxpZGF0ZSB2YXJpYWJsZXNcbiAgICBpZiAodGhpcy5tZXRhZGF0YS52YXJpYWJsZXMpIHtcbiAgICAgIGlmICh0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcy5sZW5ndGggPiB0aGlzLk1BWF9WQVJJQUJMRV9DT1VOVCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFZhcmlhYmxlIGNvdW50ICR7dGhpcy5tZXRhZGF0YS52YXJpYWJsZXMubGVuZ3RofSBleGNlZWRzIG1heGltdW0gJHt0aGlzLk1BWF9WQVJJQUJMRV9DT1VOVH1gKTtcbiAgICAgIH1cbiAgICAgIFxuICAgICAgdGhpcy5tZXRhZGF0YS52YXJpYWJsZXMgPSB0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcy5tYXAodmFyaWFibGUgPT4gKHtcbiAgICAgICAgLi4udmFyaWFibGUsXG4gICAgICAgIG5hbWU6IHNhbml0aXplSW5wdXQoVW5pY29kZVZhbGlkYXRvci5ub3JtYWxpemUodmFyaWFibGUubmFtZSkubm9ybWFsaXplZENvbnRlbnQsIDUwKSxcbiAgICAgICAgZGVzY3JpcHRpb246IHZhcmlhYmxlLmRlc2NyaXB0aW9uID8gc2FuaXRpemVJbnB1dChVbmljb2RlVmFsaWRhdG9yLm5vcm1hbGl6ZSh2YXJpYWJsZS5kZXNjcmlwdGlvbikubm9ybWFsaXplZENvbnRlbnQsIDIwMCkgOiB1bmRlZmluZWQsXG4gICAgICAgIHZhbGlkYXRpb246IHZhcmlhYmxlLnZhbGlkYXRpb24gPyBzYW5pdGl6ZUlucHV0KHZhcmlhYmxlLnZhbGlkYXRpb24sIDIwMCkgOiB1bmRlZmluZWRcbiAgICAgIH0pKTtcbiAgICB9XG5cbiAgICAvLyBTRUNVUklUWSBGSVggIzI6IFZhbGlkYXRlIGluY2x1ZGUgcGF0aHNcbiAgICBpZiAodGhpcy5tZXRhZGF0YS5pbmNsdWRlcykge1xuICAgICAgdGhpcy5tZXRhZGF0YS5pbmNsdWRlcyA9IHRoaXMubWV0YWRhdGEuaW5jbHVkZXMubWFwKGluY2x1ZGVQYXRoID0+IHtcbiAgICAgICAgY29uc3Qgc2FuaXRpemVkUGF0aCA9IHNhbml0aXplSW5wdXQoaW5jbHVkZVBhdGgsIDIwMCk7XG4gICAgICAgIC8vIFByZXZlbnQgcGF0aCB0cmF2ZXJzYWwgYXR0YWNrc1xuICAgICAgICBpZiAoIXRoaXMuaXNWYWxpZEluY2x1ZGVQYXRoKHNhbml0aXplZFBhdGgpKSB7XG4gICAgICAgICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgICAgICAgdHlwZTogJ1BBVEhfVFJBVkVSU0FMX0FUVEVNUFQnLFxuICAgICAgICAgICAgc2V2ZXJpdHk6ICdDUklUSUNBTCcsXG4gICAgICAgICAgICBzb3VyY2U6ICdUZW1wbGF0ZS5jb25zdHJ1Y3RvcicsXG4gICAgICAgICAgICBkZXRhaWxzOiBgSW52YWxpZCBpbmNsdWRlIHBhdGg6ICR7c2FuaXRpemVkUGF0aH1gXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbnZhbGlkIGluY2x1ZGUgcGF0aDogJHtzYW5pdGl6ZWRQYXRofWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBzYW5pdGl6ZWRQYXRoO1xuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFZhbGlkYXRlIGluY2x1ZGUgcGF0aHMgdG8gcHJldmVudCBkaXJlY3RvcnkgdHJhdmVyc2FsXG4gICAqIFNFQ1VSSVRZIEZJWCAjMjogUHJldmVudHMgYWNjZXNzaW5nIGZpbGVzIG91dHNpZGUgdGVtcGxhdGUgZGlyZWN0b3J5XG4gICAqL1xuICBwcml2YXRlIGlzVmFsaWRJbmNsdWRlUGF0aChpbmNsdWRlUGF0aDogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgLy8gTm9ybWFsaXplIHRoZSBwYXRoXG4gICAgY29uc3Qgbm9ybWFsaXplZCA9IHBhdGgubm9ybWFsaXplKGluY2x1ZGVQYXRoKTtcbiAgICBcbiAgICAvLyBDaGVjayBmb3IgcGF0aCB0cmF2ZXJzYWwgcGF0dGVybnNcbiAgICBpZiAobm9ybWFsaXplZC5pbmNsdWRlcygnLi4nKSB8fCBub3JtYWxpemVkLmluY2x1ZGVzKCd+JykgfHwgcGF0aC5pc0Fic29sdXRlKG5vcm1hbGl6ZWQpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIFxuICAgIC8vIE9ubHkgYWxsb3cgYWxwaGFudW1lcmljLCBkYXNoLCB1bmRlcnNjb3JlLCBmb3J3YXJkIHNsYXNoLCBiYWNrc2xhc2ggKGZvciBXaW5kb3dzKSwgYW5kIC5tZCBleHRlbnNpb25cbiAgICAvLyBOb3RlOiBXZSB0ZXN0IGFnYWluc3QgdGhlIG9yaWdpbmFsIHBhdGggdG8gcHJlc2VydmUgY3Jvc3MtcGxhdGZvcm0gY29tcGF0aWJpbGl0eVxuICAgIGNvbnN0IHZhbGlkUGF0aFBhdHRlcm4gPSAvXlthLXpBLVowLTlcXC1fXFwvXFxcXF0rXFwubWQkLztcbiAgICByZXR1cm4gdmFsaWRQYXRoUGF0dGVybi50ZXN0KGluY2x1ZGVQYXRoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDb21waWxlIHRoZSB0ZW1wbGF0ZSBmb3IgZWZmaWNpZW50IHJlbmRlcmluZ1xuICAgKiBTRUNVUklUWSBGSVggIzE6IFNhZmUgdGVtcGxhdGUgY29tcGlsYXRpb24gd2l0aG91dCBldmFsKCkgb3IgRnVuY3Rpb24oKVxuICAgKiBVc2VzIHJlZ2V4LWJhc2VkIHRva2VuIHJlcGxhY2VtZW50IGluc3RlYWQgb2YgZHluYW1pYyBjb2RlIGV4ZWN1dGlvblxuICAgKi9cbiAgcHJpdmF0ZSBjb21waWxlKCk6IENvbXBpbGVkVGVtcGxhdGUge1xuICAgIGlmICh0aGlzLmNvbXBpbGVkVGVtcGxhdGUpIHtcbiAgICAgIHJldHVybiB0aGlzLmNvbXBpbGVkVGVtcGxhdGU7XG4gICAgfVxuXG4gICAgLy8gRXh0cmFjdCBhbGwgdmFyaWFibGUgdG9rZW5zIGZyb20gdGhlIHRlbXBsYXRlXG4gICAgY29uc3QgdmFyaWFibGVQYXR0ZXJuID0gL1xce1xce1xccyooW2EtekEtWl9dW2EtekEtWjAtOV9dKig/OlxcLlthLXpBLVpfXVthLXpBLVowLTlfXSopKilcXHMqXFx9XFx9L2c7XG4gICAgY29uc3QgdG9rZW5zOiBUZW1wbGF0ZVRva2VuW10gPSBbXTtcbiAgICBsZXQgbWF0Y2g7XG4gICAgXG4gICAgd2hpbGUgKChtYXRjaCA9IHZhcmlhYmxlUGF0dGVybi5leGVjKHRoaXMuY29udGVudCkpICE9PSBudWxsKSB7XG4gICAgICB0b2tlbnMucHVzaCh7XG4gICAgICAgIHRva2VuOiBtYXRjaFswXSxcbiAgICAgICAgdmFyaWFibGU6IG1hdGNoWzFdLFxuICAgICAgICBwb3NpdGlvbjogbWF0Y2guaW5kZXhcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIHRoaXMuY29tcGlsZWRUZW1wbGF0ZSA9IHtcbiAgICAgIGNvbnRlbnQ6IHRoaXMuY29udGVudCxcbiAgICAgIHRva2VucyxcbiAgICAgIHZhcmlhYmxlczogdGhpcy5tZXRhZGF0YS52YXJpYWJsZXMgfHwgW11cbiAgICB9O1xuXG4gICAgcmV0dXJuIHRoaXMuY29tcGlsZWRUZW1wbGF0ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW5kZXIgdGhlIHRlbXBsYXRlIHdpdGggcHJvdmlkZWQgdmFyaWFibGVzXG4gICAqIFNFQ1VSSVRZIEZJWCAjMTogU2FmZSByZW5kZXJpbmcgd2l0aG91dCBjb2RlIGV4ZWN1dGlvblxuICAgKiBTRUNVUklUWSBGSVggIzM6IEFsbCB2YXJpYWJsZXMgYXJlIHZhbGlkYXRlZCBhbmQgc2FuaXRpemVkXG4gICAqIFRZUEUgU0FGRVRZOiBTdHJvbmcgdHlwaW5nIGZvciB2YXJpYWJsZXMgd2l0aCBydW50aW1lIHZhbGlkYXRpb25cbiAgICovXG4gIGFzeW5jIHJlbmRlcjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4+KFxuICAgIHZhcmlhYmxlczogVCA9IHt9IGFzIFQsIFxuICAgIGluY2x1ZGVEZXB0aDogbnVtYmVyID0gMFxuICApOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIC8vIFNFQ1VSSVRZIEZJWCAjNDogUHJldmVudCBpbmZpbml0ZSBpbmNsdWRlIGxvb3BzXG4gICAgaWYgKGluY2x1ZGVEZXB0aCA+IHRoaXMuTUFYX0lOQ0xVREVfREVQVEgpIHtcbiAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgdHlwZTogJ0lOQ0xVREVfREVQVEhfRVhDRUVERUQnLFxuICAgICAgICBzZXZlcml0eTogJ0hJR0gnLFxuICAgICAgICBzb3VyY2U6ICdUZW1wbGF0ZS5yZW5kZXInLFxuICAgICAgICBkZXRhaWxzOiBgSW5jbHVkZSBkZXB0aCAke2luY2x1ZGVEZXB0aH0gZXhjZWVkcyBtYXhpbXVtICR7dGhpcy5NQVhfSU5DTFVERV9ERVBUSH1gXG4gICAgICB9KTtcbiAgICAgIHRocm93IG5ldyBFcnJvcignTWF4aW11bSB0ZW1wbGF0ZSBpbmNsdWRlIGRlcHRoIGV4Y2VlZGVkJyk7XG4gICAgfVxuXG4gICAgLy8gQ29tcGlsZSB0aGUgdGVtcGxhdGVcbiAgICBjb25zdCBjb21waWxlZCA9IHRoaXMuY29tcGlsZSgpO1xuICAgIFxuICAgIC8vIFZhbGlkYXRlIGFuZCBzYW5pdGl6ZSBhbGwgcHJvdmlkZWQgdmFyaWFibGVzXG4gICAgY29uc3Qgc2FuaXRpemVkVmFyaWFibGVzID0gYXdhaXQgdGhpcy52YWxpZGF0ZUFuZFNhbml0aXplVmFyaWFibGVzKHZhcmlhYmxlcyk7XG4gICAgXG4gICAgLy8gU3RhcnQgd2l0aCB0aGUgdGVtcGxhdGUgY29udGVudFxuICAgIGxldCByZW5kZXJlZCA9IGNvbXBpbGVkLmNvbnRlbnQ7XG4gICAgXG4gICAgLy8gUmVwbGFjZSB0b2tlbnMgaW4gcmV2ZXJzZSBvcmRlciB0byBtYWludGFpbiBwb3NpdGlvbnNcbiAgICBjb25zdCBzb3J0ZWRUb2tlbnMgPSBbLi4uY29tcGlsZWQudG9rZW5zXS5zb3J0KChhLCBiKSA9PiBiLnBvc2l0aW9uIC0gYS5wb3NpdGlvbik7XG4gICAgXG4gICAgZm9yIChjb25zdCB0b2tlbiBvZiBzb3J0ZWRUb2tlbnMpIHtcbiAgICAgIGNvbnN0IHZhbHVlID0gdGhpcy5yZXNvbHZlVmFyaWFibGUodG9rZW4udmFyaWFibGUsIHNhbml0aXplZFZhcmlhYmxlcyk7XG4gICAgICBjb25zdCBzdHJpbmdWYWx1ZSA9IHRoaXMuZm9ybWF0VmFsdWUodmFsdWUpO1xuICAgICAgXG4gICAgICAvLyBSZXBsYWNlIHRoZSB0b2tlbiB3aXRoIHRoZSBzYW5pdGl6ZWQgdmFsdWVcbiAgICAgIHJlbmRlcmVkID0gcmVuZGVyZWQuc3Vic3RyaW5nKDAsIHRva2VuLnBvc2l0aW9uKSArIFxuICAgICAgICAgICAgICAgICBzdHJpbmdWYWx1ZSArIFxuICAgICAgICAgICAgICAgICByZW5kZXJlZC5zdWJzdHJpbmcodG9rZW4ucG9zaXRpb24gKyB0b2tlbi50b2tlbi5sZW5ndGgpO1xuICAgIH1cbiAgICBcbiAgICAvLyBQcm9jZXNzIGluY2x1ZGVzIGlmIGFueVxuICAgIGlmICh0aGlzLm1ldGFkYXRhLmluY2x1ZGVzICYmIHRoaXMubWV0YWRhdGEuaW5jbHVkZXMubGVuZ3RoID4gMCkge1xuICAgICAgcmVuZGVyZWQgPSBhd2FpdCB0aGlzLnByb2Nlc3NJbmNsdWRlcyhyZW5kZXJlZCwgc2FuaXRpemVkVmFyaWFibGVzLCBpbmNsdWRlRGVwdGgpO1xuICAgIH1cbiAgICBcbiAgICAvLyBVcGRhdGUgdXNhZ2Ugc3RhdGlzdGljc1xuICAgIC8vIE5PVEU6IFRoZXNlIHVwZGF0ZXMgYXJlIG5vdCBhdG9taWMgYW5kIG1heSBoYXZlIHJhY2UgY29uZGl0aW9ucyB1bmRlciBjb25jdXJyZW50IGFjY2Vzc1xuICAgIC8vIFRoaXMgaXMgYWNjZXB0YWJsZSBmb3IgdXNhZ2Ugc3RhdGlzdGljcyB3aGljaCBkb24ndCByZXF1aXJlIHBlcmZlY3QgYWNjdXJhY3lcbiAgICAvLyBGb3IgcHJvZHVjdGlvbiBzeXN0ZW1zIHJlcXVpcmluZyBhdG9taWMgY291bnRlcnMsIGNvbnNpZGVyIHVzaW5nIGEgZGF0YWJhc2Ugb3IgYXRvbWljIG9wZXJhdGlvbnNcbiAgICB0aGlzLm1ldGFkYXRhLnVzYWdlX2NvdW50ID0gKHRoaXMubWV0YWRhdGEudXNhZ2VfY291bnQgfHwgMCkgKyAxO1xuICAgIHRoaXMubWV0YWRhdGEubGFzdF91c2VkID0gbmV3IERhdGUoKS50b0lTT1N0cmluZygpO1xuICAgIFxuICAgIC8vIFNFQ1VSSVRZIEZJWCAjNTogTG9nIHRlbXBsYXRlIHVzYWdlIGZvciBhdWRpdCB0cmFpbFxuICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgIHR5cGU6ICdURU1QTEFURV9SRU5ERVJFRCcsXG4gICAgICBzZXZlcml0eTogJ0xPVycsXG4gICAgICBzb3VyY2U6ICdUZW1wbGF0ZS5yZW5kZXInLFxuICAgICAgZGV0YWlsczogYFRlbXBsYXRlICR7dGhpcy5tZXRhZGF0YS5uYW1lfSByZW5kZXJlZCB3aXRoICR7T2JqZWN0LmtleXMoc2FuaXRpemVkVmFyaWFibGVzKS5sZW5ndGh9IHZhcmlhYmxlc2BcbiAgICB9KTtcbiAgICBcbiAgICByZXR1cm4gcmVuZGVyZWQ7XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGUgYW5kIHNhbml0aXplIHZhcmlhYmxlcyBhY2NvcmRpbmcgdG8gdGhlaXIgZGVmaW5pdGlvbnNcbiAgICogU0VDVVJJVFkgRklYICMzOiBDb21wcmVoZW5zaXZlIHZhbGlkYXRpb24gb2YgYWxsIGlucHV0IHZhcmlhYmxlc1xuICAgKiBUWVBFIFNBRkVUWTogSW1wcm92ZWQgdHlwZSBzYWZldHkgZm9yIHZhcmlhYmxlIHZhbGlkYXRpb25cbiAgICovXG4gIHByaXZhdGUgYXN5bmMgdmFsaWRhdGVBbmRTYW5pdGl6ZVZhcmlhYmxlcyhcbiAgICB2YXJpYWJsZXM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+XG4gICk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgdW5rbm93bj4+IHtcbiAgICBjb25zdCBzYW5pdGl6ZWQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gICAgXG4gICAgLy8gQ2hlY2sgcmVxdWlyZWQgdmFyaWFibGVzXG4gICAgZm9yIChjb25zdCB2YXJEZWYgb2YgdGhpcy5tZXRhZGF0YS52YXJpYWJsZXMgfHwgW10pIHtcbiAgICAgIGlmICh2YXJEZWYucmVxdWlyZWQgJiYgISh2YXJEZWYubmFtZSBpbiB2YXJpYWJsZXMpKSB7XG4gICAgICAgIGlmICh2YXJEZWYuZGVmYXVsdCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgc2FuaXRpemVkW3ZhckRlZi5uYW1lXSA9IHZhckRlZi5kZWZhdWx0O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgUmVxdWlyZWQgdmFyaWFibGUgJyR7dmFyRGVmLm5hbWV9JyBub3QgcHJvdmlkZWRgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICAvLyBWYWxpZGF0ZSBhbmQgc2FuaXRpemUgcHJvdmlkZWQgdmFyaWFibGVzXG4gICAgZm9yIChjb25zdCBbbmFtZSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKHZhcmlhYmxlcykpIHtcbiAgICAgIGNvbnN0IHZhckRlZiA9IHRoaXMubWV0YWRhdGEudmFyaWFibGVzPy5maW5kKHYgPT4gdi5uYW1lID09PSBuYW1lKTtcbiAgICAgIFxuICAgICAgaWYgKCF2YXJEZWYpIHtcbiAgICAgICAgLy8gU2tpcCB1bmtub3duIHZhcmlhYmxlcyAodGhleSB3b24ndCBiZSB1c2VkIGFueXdheSlcbiAgICAgICAgbG9nZ2VyLndhcm4oYFVua25vd24gdmFyaWFibGUgJyR7bmFtZX0nIHByb3ZpZGVkIHRvIHRlbXBsYXRlYCk7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuICAgICAgXG4gICAgICAvLyBUeXBlIHZhbGlkYXRpb24gYW5kIHNhbml0aXphdGlvblxuICAgICAgY29uc3Qgc2FuaXRpemVkVmFsdWUgPSBhd2FpdCB0aGlzLnNhbml0aXplVmFyaWFibGVWYWx1ZSh2YWx1ZSwgdmFyRGVmKTtcbiAgICAgIHNhbml0aXplZFtuYW1lXSA9IHNhbml0aXplZFZhbHVlO1xuICAgIH1cbiAgICBcbiAgICAvLyBBcHBseSBkZWZhdWx0cyBmb3IgbWlzc2luZyBvcHRpb25hbCB2YXJpYWJsZXNcbiAgICBmb3IgKGNvbnN0IHZhckRlZiBvZiB0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcyB8fCBbXSkge1xuICAgICAgaWYgKCF2YXJEZWYucmVxdWlyZWQgJiYgISh2YXJEZWYubmFtZSBpbiBzYW5pdGl6ZWQpICYmIHZhckRlZi5kZWZhdWx0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgc2FuaXRpemVkW3ZhckRlZi5uYW1lXSA9IHZhckRlZi5kZWZhdWx0O1xuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICByZXR1cm4gc2FuaXRpemVkO1xuICB9XG5cbiAgLyoqXG4gICAqIFNhbml0aXplIGEgc2luZ2xlIHZhcmlhYmxlIHZhbHVlIGFjY29yZGluZyB0byBpdHMgZGVmaW5pdGlvblxuICAgKiBTRUNVUklUWSBGSVggIzMgJiAjNjogVHlwZS1zcGVjaWZpYyB2YWxpZGF0aW9uIGFuZCBVbmljb2RlIG5vcm1hbGl6YXRpb25cbiAgICovXG4gIHByaXZhdGUgYXN5bmMgc2FuaXRpemVWYXJpYWJsZVZhbHVlKHZhbHVlOiBhbnksIHZhckRlZjogVGVtcGxhdGVWYXJpYWJsZSk6IFByb21pc2U8YW55PiB7XG4gICAgc3dpdGNoICh2YXJEZWYudHlwZSkge1xuICAgICAgY2FzZSAnc3RyaW5nJzpcbiAgICAgICAgLy8gU0VDVVJJVFkgRklYICM2OiBVbmljb2RlIG5vcm1hbGl6YXRpb25cbiAgICAgICAgY29uc3Qgbm9ybWFsaXplZCA9IFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKFN0cmluZyh2YWx1ZSkpO1xuICAgICAgICBsZXQgc3RyaW5nVmFsdWUgPSBzYW5pdGl6ZUlucHV0KG5vcm1hbGl6ZWQubm9ybWFsaXplZENvbnRlbnQsIHRoaXMuTUFYX1NUUklOR19MRU5HVEgpO1xuICAgICAgICBcbiAgICAgICAgLy8gQXBwbHkgcmVnZXggdmFsaWRhdGlvbiBpZiBzcGVjaWZpZWRcbiAgICAgICAgaWYgKHZhckRlZi52YWxpZGF0aW9uKSB7XG4gICAgICAgICAgLy8gU0VDVVJJVFkgRklYOiBWYWxpZGF0ZSByZWdleCBjb21wbGV4aXR5IHRvIHByZXZlbnQgUmVEb1MgYXR0YWNrc1xuICAgICAgICAgIC8vIFByZXZpb3VzbHk6IFVzZXItcHJvdmlkZWQgcmVnZXggZXhlY3V0ZWQgd2l0aG91dCBsaW1pdHNcbiAgICAgICAgICAvLyBOb3c6IENoZWNrIGZvciBkYW5nZXJvdXMgcGF0dGVybnMgYW5kIGxpbWl0IGV4ZWN1dGlvbiB0aW1lXG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIENoZWNrIGZvciBkYW5nZXJvdXMgcmVnZXggcGF0dGVybnNcbiAgICAgICAgICAgIGlmICh0aGlzLmlzRGFuZ2Vyb3VzUmVnZXgodmFyRGVmLnZhbGlkYXRpb24pKSB7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgVmFyaWFibGUgJyR7dmFyRGVmLm5hbWV9JyBoYXMgcG90ZW50aWFsbHkgZGFuZ2Vyb3VzIHZhbGlkYXRpb24gcGF0dGVybmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgXG4gICAgICAgICAgICBjb25zdCByZWdleCA9IG5ldyBSZWdFeHAodmFyRGVmLnZhbGlkYXRpb24pO1xuICAgICAgICAgICAgLy8gVXNlIGEgc2ltcGxlIHRpbWVvdXQgbWVjaGFuaXNtIC0gaW4gcHJvZHVjdGlvbiwgY29uc2lkZXIgdXNpbmcgYSB3b3JrZXIgdGhyZWFkXG4gICAgICAgICAgICBjb25zdCBzdGFydFRpbWUgPSBEYXRlLm5vdygpO1xuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gcmVnZXgudGVzdChzdHJpbmdWYWx1ZSk7XG4gICAgICAgICAgICBjb25zdCBkdXJhdGlvbiA9IERhdGUubm93KCkgLSBzdGFydFRpbWU7XG4gICAgICAgICAgICBcbiAgICAgICAgICAgIC8vIElmIHJlZ2V4IHRha2VzIHRvbyBsb25nLCBpdCBtaWdodCBiZSBtYWxpY2lvdXNcbiAgICAgICAgICAgIGlmIChkdXJhdGlvbiA+IDEwMCkgeyAvLyAxMDBtcyB0aHJlc2hvbGRcbiAgICAgICAgICAgICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgICAgICAgICAgIHR5cGU6ICdDT05URU5UX0lOSkVDVElPTl9BVFRFTVBUJyxcbiAgICAgICAgICAgICAgICBzZXZlcml0eTogJ0hJR0gnLFxuICAgICAgICAgICAgICAgIHNvdXJjZTogJ1RlbXBsYXRlLnNhbml0aXplVmFyaWFibGVWYWx1ZScsXG4gICAgICAgICAgICAgICAgZGV0YWlsczogYFJlZ2V4IHZhbGlkYXRpb24gdG9vayAke2R1cmF0aW9ufW1zIGZvciB2YXJpYWJsZSAnJHt2YXJEZWYubmFtZX0nLCBwb3NzaWJsZSBSZURvU2BcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgVmFyaWFibGUgJyR7dmFyRGVmLm5hbWV9JyB2YWxpZGF0aW9uIHBhdHRlcm4gaXMgdG9vIGNvbXBsZXhgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIFxuICAgICAgICAgICAgaWYgKCFyZXN1bHQpIHtcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBWYXJpYWJsZSAnJHt2YXJEZWYubmFtZX0nIGRvZXMgbm90IG1hdGNoIHZhbGlkYXRpb24gcGF0dGVybmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgICAgIGlmIChlIGluc3RhbmNlb2YgU3ludGF4RXJyb3IpIHtcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBWYXJpYWJsZSAnJHt2YXJEZWYubmFtZX0nIGhhcyBpbnZhbGlkIHZhbGlkYXRpb24gcGF0dGVybmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgdGhyb3cgZTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIC8vIENoZWNrIGVudW0gb3B0aW9ucyBpZiBzcGVjaWZpZWRcbiAgICAgICAgaWYgKHZhckRlZi5vcHRpb25zICYmICF2YXJEZWYub3B0aW9ucy5pbmNsdWRlcyhzdHJpbmdWYWx1ZSkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFZhcmlhYmxlICcke3ZhckRlZi5uYW1lfScgbXVzdCBiZSBvbmUgb2Y6ICR7dmFyRGVmLm9wdGlvbnMuam9pbignLCAnKX1gKTtcbiAgICAgICAgfVxuICAgICAgICBcbiAgICAgICAgcmV0dXJuIHN0cmluZ1ZhbHVlO1xuICAgICAgICBcbiAgICAgIGNhc2UgJ251bWJlcic6XG4gICAgICAgIGNvbnN0IG51bSA9IE51bWJlcih2YWx1ZSk7XG4gICAgICAgIGlmIChpc05hTihudW0pKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBWYXJpYWJsZSAnJHt2YXJEZWYubmFtZX0nIG11c3QgYmUgYSBudW1iZXJgKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gbnVtO1xuICAgICAgICBcbiAgICAgIGNhc2UgJ2Jvb2xlYW4nOlxuICAgICAgICByZXR1cm4gQm9vbGVhbih2YWx1ZSk7XG4gICAgICAgIFxuICAgICAgY2FzZSAnZGF0ZSc6XG4gICAgICAgIGNvbnN0IGRhdGUgPSBuZXcgRGF0ZSh2YWx1ZSk7XG4gICAgICAgIGlmIChpc05hTihkYXRlLmdldFRpbWUoKSkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFZhcmlhYmxlICcke3ZhckRlZi5uYW1lfScgbXVzdCBiZSBhIHZhbGlkIGRhdGVgKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gZGF0ZTtcbiAgICAgICAgXG4gICAgICBjYXNlICdhcnJheSc6XG4gICAgICAgIGlmICghQXJyYXkuaXNBcnJheSh2YWx1ZSkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFZhcmlhYmxlICcke3ZhckRlZi5uYW1lfScgbXVzdCBiZSBhbiBhcnJheWApO1xuICAgICAgICB9XG4gICAgICAgIC8vIFNFQ1VSSVRZIEZJWDogTGltaXQgYXJyYXkgc2l6ZSB0byBwcmV2ZW50IG1lbW9yeSBleGhhdXN0aW9uIGF0dGFja3NcbiAgICAgICAgLy8gUHJldmlvdXNseTogTm8gbGltaXQgb24gYXJyYXkgc2l6ZSBjb3VsZCBsZWFkIHRvIERvU1xuICAgICAgICAvLyBOb3c6IEVuZm9yY2VzIHJlYXNvbmFibGUgc2l6ZSBsaW1pdCB3aXRoIGxvZ2dpbmdcbiAgICAgICAgY29uc3QgTUFYX0FSUkFZX1NJWkUgPSAxMDAwO1xuICAgICAgICBpZiAodmFsdWUubGVuZ3RoID4gTUFYX0FSUkFZX1NJWkUpIHtcbiAgICAgICAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICAgICAgICB0eXBlOiAnQ09OVEVOVF9TSVpFX0VYQ0VFREVEJyxcbiAgICAgICAgICAgIHNldmVyaXR5OiAnTUVESVVNJyxcbiAgICAgICAgICAgIHNvdXJjZTogJ1RlbXBsYXRlLnNhbml0aXplVmFyaWFibGVWYWx1ZScsXG4gICAgICAgICAgICBkZXRhaWxzOiBgQXJyYXkgdmFyaWFibGUgJyR7dmFyRGVmLm5hbWV9JyBoYXMgJHt2YWx1ZS5sZW5ndGh9IGl0ZW1zLCBsaW1pdGluZyB0byAke01BWF9BUlJBWV9TSVpFfWBcbiAgICAgICAgICB9KTtcbiAgICAgICAgICB2YWx1ZSA9IHZhbHVlLnNsaWNlKDAsIE1BWF9BUlJBWV9TSVpFKTtcbiAgICAgICAgfVxuICAgICAgICAvLyBTYW5pdGl6ZSBzdHJpbmcgZWxlbWVudHMgaW4gYXJyYXlzXG4gICAgICAgIHJldHVybiB2YWx1ZS5tYXAoKGl0ZW06IGFueSkgPT4gXG4gICAgICAgICAgdHlwZW9mIGl0ZW0gPT09ICdzdHJpbmcnID8gc2FuaXRpemVJbnB1dChpdGVtLCAxMDAwKSA6IGl0ZW1cbiAgICAgICAgKTtcbiAgICAgICAgXG4gICAgICBjYXNlICdvYmplY3QnOlxuICAgICAgICBpZiAodHlwZW9mIHZhbHVlICE9PSAnb2JqZWN0JyB8fCB2YWx1ZSA9PT0gbnVsbCkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgVmFyaWFibGUgJyR7dmFyRGVmLm5hbWV9JyBtdXN0IGJlIGFuIG9iamVjdGApO1xuICAgICAgICB9XG4gICAgICAgIC8vIERlZXAgc2FuaXRpemUgc3RyaW5nIHZhbHVlcyBpbiBvYmplY3RzXG4gICAgICAgIHJldHVybiB0aGlzLnNhbml0aXplT2JqZWN0KHZhbHVlKTtcbiAgICAgICAgXG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdmFsdWU7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJlY3Vyc2l2ZWx5IHNhbml0aXplIHN0cmluZyB2YWx1ZXMgaW4gb2JqZWN0c1xuICAgKiBTRUNVUklUWSBGSVg6IFRydW5jYXRlIGRlZXAgbmVzdGluZyBpbnN0ZWFkIG9mIHRocm93aW5nIHRvIHByZXZlbnQgRG9TXG4gICAqL1xuICBwcml2YXRlIHNhbml0aXplT2JqZWN0KG9iajogYW55LCBkZXB0aDogbnVtYmVyID0gMCk6IGFueSB7XG4gICAgLy8gU0VDVVJJVFkgRklYOiBSZXR1cm4gc2FmZSBkZWZhdWx0IGluc3RlYWQgb2YgdGhyb3dpbmcgdG8gcHJldmVudCBEb1MgYXR0YWNrc1xuICAgIC8vIFByZXZpb3VzbHk6IFRocmV3IGVycm9yIG9uIGRlZXAgbmVzdGluZyB3aGljaCBjb3VsZCBiZSBleHBsb2l0ZWRcbiAgICAvLyBOb3c6IFJldHVybnMgc3RyaW5nIHJlcHJlc2VudGF0aW9uIGZvciBleGNlc3NpdmVseSBuZXN0ZWQgb2JqZWN0c1xuICAgIGlmIChkZXB0aCA+IDEwKSB7XG4gICAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICAgIHR5cGU6ICdDT05URU5UX1NJWkVfRVhDRUVERUQnLFxuICAgICAgICBzZXZlcml0eTogJ01FRElVTScsXG4gICAgICAgIHNvdXJjZTogJ1RlbXBsYXRlLnNhbml0aXplT2JqZWN0JyxcbiAgICAgICAgZGV0YWlsczogJ09iamVjdCBuZXN0aW5nIGRlcHRoIGV4Y2VlZGVkLCB0cnVuY2F0aW5nIHRvIHN0cmluZyByZXByZXNlbnRhdGlvbidcbiAgICAgIH0pO1xuICAgICAgcmV0dXJuICdbT2JqZWN0IHRvbyBkZWVwbHkgbmVzdGVkXSc7XG4gICAgfVxuICAgIFxuICAgIGlmICh0eXBlb2Ygb2JqICE9PSAnb2JqZWN0JyB8fCBvYmogPT09IG51bGwpIHtcbiAgICAgIHJldHVybiBvYmo7XG4gICAgfVxuICAgIFxuICAgIGlmIChBcnJheS5pc0FycmF5KG9iaikpIHtcbiAgICAgIC8vIFNFQ1VSSVRZIEZJWDogTGltaXQgYXJyYXkgc2l6ZSB0byBwcmV2ZW50IG1lbW9yeSBleGhhdXN0aW9uXG4gICAgICBjb25zdCBNQVhfQVJSQVlfU0laRSA9IDEwMDA7XG4gICAgICBpZiAob2JqLmxlbmd0aCA+IE1BWF9BUlJBWV9TSVpFKSB7XG4gICAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgICB0eXBlOiAnQ09OVEVOVF9TSVpFX0VYQ0VFREVEJyxcbiAgICAgICAgICBzZXZlcml0eTogJ01FRElVTScsXG4gICAgICAgICAgc291cmNlOiAnVGVtcGxhdGUuc2FuaXRpemVPYmplY3QnLFxuICAgICAgICAgIGRldGFpbHM6IGBBcnJheSBzaXplICR7b2JqLmxlbmd0aH0gZXhjZWVkcyBtYXhpbXVtICR7TUFYX0FSUkFZX1NJWkV9LCB0cnVuY2F0aW5nYFxuICAgICAgICB9KTtcbiAgICAgICAgb2JqID0gb2JqLnNsaWNlKDAsIE1BWF9BUlJBWV9TSVpFKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBvYmoubWFwKChpdGVtOiBhbnkpID0+IHRoaXMuc2FuaXRpemVPYmplY3QoaXRlbSwgZGVwdGggKyAxKSk7XG4gICAgfVxuICAgIFxuICAgIGNvbnN0IHNhbml0aXplZDogUmVjb3JkPHN0cmluZywgYW55PiA9IHt9O1xuICAgIC8vIFNFQ1VSSVRZIEZJWDogTGltaXQgbnVtYmVyIG9mIG9iamVjdCBwcm9wZXJ0aWVzIHRvIHByZXZlbnQgbWVtb3J5IGV4aGF1c3Rpb25cbiAgICBjb25zdCBNQVhfT0JKRUNUX0tFWVMgPSAxMDA7XG4gICAgY29uc3QgZW50cmllcyA9IE9iamVjdC5lbnRyaWVzKG9iaik7XG4gICAgaWYgKGVudHJpZXMubGVuZ3RoID4gTUFYX09CSkVDVF9LRVlTKSB7XG4gICAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICAgIHR5cGU6ICdDT05URU5UX1NJWkVfRVhDRUVERUQnLFxuICAgICAgICBzZXZlcml0eTogJ01FRElVTScsXG4gICAgICAgIHNvdXJjZTogJ1RlbXBsYXRlLnNhbml0aXplT2JqZWN0JyxcbiAgICAgICAgZGV0YWlsczogYE9iamVjdCBoYXMgJHtlbnRyaWVzLmxlbmd0aH0ga2V5cywgbGltaXRpbmcgdG8gJHtNQVhfT0JKRUNUX0tFWVN9YFxuICAgICAgfSk7XG4gICAgfVxuICAgIFxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgTWF0aC5taW4oZW50cmllcy5sZW5ndGgsIE1BWF9PQkpFQ1RfS0VZUyk7IGkrKykge1xuICAgICAgY29uc3QgW2tleSwgdmFsdWVdID0gZW50cmllc1tpXTtcbiAgICAgIGNvbnN0IHNhbml0aXplZEtleSA9IHNhbml0aXplSW5wdXQoa2V5LCA1MCk7XG4gICAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJykge1xuICAgICAgICBzYW5pdGl6ZWRbc2FuaXRpemVkS2V5XSA9IHNhbml0aXplSW5wdXQodmFsdWUsIDEwMDApO1xuICAgICAgfSBlbHNlIGlmICh0eXBlb2YgdmFsdWUgPT09ICdvYmplY3QnKSB7XG4gICAgICAgIHNhbml0aXplZFtzYW5pdGl6ZWRLZXldID0gdGhpcy5zYW5pdGl6ZU9iamVjdCh2YWx1ZSwgZGVwdGggKyAxKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNhbml0aXplZFtzYW5pdGl6ZWRLZXldID0gdmFsdWU7XG4gICAgICB9XG4gICAgfVxuICAgIFxuICAgIHJldHVybiBzYW5pdGl6ZWQ7XG4gIH1cblxuICAvKipcbiAgICogUmVzb2x2ZSBuZXN0ZWQgdmFyaWFibGUgcGF0aHMgKGUuZy4sIFwidXNlci5uYW1lXCIpXG4gICAqIFRZUEUgU0FGRVRZOiBJbXByb3ZlZCB0eXBlIHNhZmV0eSBmb3IgdmFyaWFibGUgcmVzb2x1dGlvblxuICAgKi9cbiAgcHJpdmF0ZSByZXNvbHZlVmFyaWFibGUocGF0aDogc3RyaW5nLCB2YXJpYWJsZXM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogdW5rbm93biB7XG4gICAgY29uc3QgcGFydHMgPSBwYXRoLnNwbGl0KCcuJyk7XG4gICAgbGV0IHZhbHVlOiB1bmtub3duID0gdmFyaWFibGVzO1xuICAgIFxuICAgIGZvciAoY29uc3QgcGFydCBvZiBwYXJ0cykge1xuICAgICAgaWYgKHZhbHVlICYmIHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcgJiYgdmFsdWUgIT09IG51bGwgJiYgcGFydCBpbiB2YWx1ZSkge1xuICAgICAgICAvLyBUeXBlIGFzc2VydGlvbiB3aXRoIHJ1bnRpbWUgY2hlY2sgLSBzYWZlIGJlY2F1c2Ugd2UgdmVyaWZpZWQgdGhlIHByb3BlcnR5IGV4aXN0c1xuICAgICAgICB2YWx1ZSA9ICh2YWx1ZSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbcGFydF07XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXR1cm4gdW5kZWZpbmVkO1xuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICByZXR1cm4gdmFsdWU7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgYSByZWdleCBwYXR0ZXJuIGlzIHBvdGVudGlhbGx5IGRhbmdlcm91cyAoUmVEb1MpXG4gICAqIFNFQ1VSSVRZIEZJWDogRGV0ZWN0IHBhdHRlcm5zIHRoYXQgY291bGQgY2F1c2UgZXhwb25lbnRpYWwgYmFja3RyYWNraW5nXG4gICAqL1xuICBwcml2YXRlIGlzRGFuZ2Vyb3VzUmVnZXgocGF0dGVybjogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgLy8gQ2hlY2sgZm9yIG5lc3RlZCBxdWFudGlmaWVycyB3aGljaCBjYW4gY2F1c2UgZXhwb25lbnRpYWwgYmFja3RyYWNraW5nXG4gICAgY29uc3QgZGFuZ2Vyb3VzUGF0dGVybnMgPSBbXG4gICAgICAvKFxcK3xcXCopezIsfS8sICAgICAgICAgICAvLyBNdWx0aXBsZSBxdWFudGlmaWVyc1xuICAgICAgL1xcKFteKV0qXFwrXFwpWysqXS8sICAgICAgIC8vIFF1YW50aWZpZWQgZ3JvdXBzIHdpdGggcXVhbnRpZmllcnMgaW5zaWRlXG4gICAgICAvXFxbW15cXF1dKlxcK1xcXVsrKl0vLCAgICAgIC8vIFF1YW50aWZpZWQgY2hhcmFjdGVyIGNsYXNzZXMgd2l0aCBxdWFudGlmaWVyc1xuICAgICAgLyhcXFxcW2R3c10pXFwxezIsfS8sICAgICAgIC8vIFJlcGVhdGVkIGNoYXJhY3RlciBjbGFzc2VzXG4gICAgICAvXFwoXFw/XFw8WyE9XVteKV0rXFwpLywgICAgIC8vIENvbXBsZXggbG9va2JlaGluZHNcbiAgICBdO1xuICAgIFxuICAgIGZvciAoY29uc3QgZGFuZ2Vyb3VzIG9mIGRhbmdlcm91c1BhdHRlcm5zKSB7XG4gICAgICBpZiAoZGFuZ2Vyb3VzLnRlc3QocGF0dGVybikpIHtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICB9XG4gICAgfVxuICAgIFxuICAgIC8vIENoZWNrIGZvciBleGNlc3NpdmUgYmFja3RyYWNraW5nIHBvdGVudGlhbFxuICAgIC8vIENvdW50IGdyb3VwcyBhbmQgcXVhbnRpZmllcnNcbiAgICBjb25zdCBncm91cHMgPSAocGF0dGVybi5tYXRjaCgvXFwoL2cpIHx8IFtdKS5sZW5ndGg7XG4gICAgY29uc3QgcXVhbnRpZmllcnMgPSAocGF0dGVybi5tYXRjaCgvWysqP3tdL2cpIHx8IFtdKS5sZW5ndGg7XG4gICAgXG4gICAgLy8gSWYgdGhlcmUgYXJlIG1hbnkgZ3JvdXBzIGFuZCBxdWFudGlmaWVycywgaXQncyBwb3RlbnRpYWxseSBkYW5nZXJvdXNcbiAgICBpZiAoZ3JvdXBzID4gNSAmJiBxdWFudGlmaWVycyA+IDUpIHtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgICBcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICAvKipcbiAgICogRm9ybWF0IGEgdmFsdWUgZm9yIHRlbXBsYXRlIG91dHB1dFxuICAgKi9cbiAgcHJpdmF0ZSBmb3JtYXRWYWx1ZSh2YWx1ZTogYW55KTogc3RyaW5nIHtcbiAgICBpZiAodmFsdWUgPT09IHVuZGVmaW5lZCB8fCB2YWx1ZSA9PT0gbnVsbCkge1xuICAgICAgcmV0dXJuICcnO1xuICAgIH1cbiAgICBcbiAgICBpZiAodmFsdWUgaW5zdGFuY2VvZiBEYXRlKSB7XG4gICAgICByZXR1cm4gdmFsdWUudG9JU09TdHJpbmcoKTtcbiAgICB9XG4gICAgXG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICByZXR1cm4gdmFsdWUuam9pbignLCAnKTtcbiAgICB9XG4gICAgXG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcpIHtcbiAgICAgIHJldHVybiBKU09OLnN0cmluZ2lmeSh2YWx1ZSwgbnVsbCwgMik7XG4gICAgfVxuICAgIFxuICAgIHJldHVybiBTdHJpbmcodmFsdWUpO1xuICB9XG5cbiAgLyoqXG4gICAqIFByb2Nlc3MgdGVtcGxhdGUgaW5jbHVkZXNcbiAgICogU0VDVVJJVFkgRklYICMyOiBTYWZlIGluY2x1ZGUgcHJvY2Vzc2luZyB3aXRoIHBhdGggdmFsaWRhdGlvblxuICAgKiBcbiAgICogVE9ETzogSW1wbGVtZW50IGFjdHVhbCBpbmNsdWRlIHByb2Nlc3NpbmcgZnVuY3Rpb25hbGl0eVxuICAgKiBUaGlzIGlzIGN1cnJlbnRseSBhIHBsYWNlaG9sZGVyIHRoYXQgdmFsaWRhdGVzIHRoZSBzZWN1cml0eSBtb2RlbFxuICAgKiBidXQgZG9lcyBub3QgYWN0dWFsbHkgbG9hZCBhbmQgcmVuZGVyIGluY2x1ZGVkIHRlbXBsYXRlcy5cbiAgICogXG4gICAqIEZ1dHVyZSBpbXBsZW1lbnRhdGlvbiBzaG91bGQ6XG4gICAqIDEuIExvYWQgdGVtcGxhdGVzIGZyb20gdmFsaWRhdGVkIHBhdGhzXG4gICAqIDIuIFJlbmRlciB0aGVtIHdpdGggY3VycmVudCB2YXJpYWJsZXNcbiAgICogMy4gUmVwbGFjZSBpbmNsdWRlIG1hcmtlcnMgaW4gY29udGVudFxuICAgKiA0LiBSZXNwZWN0IGluY2x1ZGVEZXB0aCBsaW1pdFxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyBwcm9jZXNzSW5jbHVkZXMoXG4gICAgY29udGVudDogc3RyaW5nLCBcbiAgICB2YXJpYWJsZXM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LCBcbiAgICBpbmNsdWRlRGVwdGg6IG51bWJlclxuICApOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIC8vIFRPRE86IEltcGxlbWVudCBhY3R1YWwgdGVtcGxhdGUgaW5jbHVkZSBwcm9jZXNzaW5nXG4gICAgLy8gQ3VycmVudCBpbXBsZW1lbnRhdGlvbiBvbmx5IHZhbGlkYXRlcyB0aGUgc2VjdXJpdHkgbW9kZWxcbiAgICBcbiAgICBpZiAoIXRoaXMubWV0YWRhdGEuaW5jbHVkZXMgfHwgdGhpcy5tZXRhZGF0YS5pbmNsdWRlcy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBjb250ZW50O1xuICAgIH1cbiAgICBcbiAgICAvLyBMb2cgc2VjdXJpdHkgZXZlbnQgZm9yIGF1ZGl0IHRyYWlsXG4gICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgdHlwZTogJ1RFTVBMQVRFX0lOQ0xVREUnLFxuICAgICAgc2V2ZXJpdHk6ICdMT1cnLFxuICAgICAgc291cmNlOiAnVGVtcGxhdGUucHJvY2Vzc0luY2x1ZGVzJyxcbiAgICAgIGRldGFpbHM6IGBQcm9jZXNzaW5nICR7dGhpcy5tZXRhZGF0YS5pbmNsdWRlcy5sZW5ndGh9IGluY2x1ZGVzIGF0IGRlcHRoICR7aW5jbHVkZURlcHRofWBcbiAgICB9KTtcbiAgICBcbiAgICAvLyBUT0RPOiBGdXR1cmUgaW1wbGVtZW50YXRpb24gd291bGQ6XG4gICAgLy8gZm9yIChjb25zdCBpbmNsdWRlUGF0aCBvZiB0aGlzLm1ldGFkYXRhLmluY2x1ZGVzKSB7XG4gICAgLy8gICBjb25zdCB0ZW1wbGF0ZSA9IGF3YWl0IHRoaXMubG9hZEluY2x1ZGVkVGVtcGxhdGUoaW5jbHVkZVBhdGgpO1xuICAgIC8vICAgY29uc3QgcmVuZGVyZWQgPSBhd2FpdCB0ZW1wbGF0ZS5yZW5kZXIodmFyaWFibGVzLCBpbmNsdWRlRGVwdGggKyAxKTtcbiAgICAvLyAgIGNvbnRlbnQgPSBjb250ZW50LnJlcGxhY2UoYHt7aW5jbHVkZToke2luY2x1ZGVQYXRofX19YCwgcmVuZGVyZWQpO1xuICAgIC8vIH1cbiAgICBcbiAgICByZXR1cm4gY29udGVudDtcbiAgfVxuXG4gIC8qKlxuICAgKiBUZW1wbGF0ZS1zcGVjaWZpYyB2YWxpZGF0aW9uXG4gICAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgdmFsaWRhdGUoKTogRWxlbWVudFZhbGlkYXRpb25SZXN1bHQge1xuICAgIGNvbnN0IHJlc3VsdCA9IHN1cGVyLnZhbGlkYXRlKCk7XG4gICAgXG4gICAgLy8gSW5pdGlhbGl6ZSBhcnJheXMgaWYgbm90IHByZXNlbnRcbiAgICBpZiAoIXJlc3VsdC5lcnJvcnMpIHJlc3VsdC5lcnJvcnMgPSBbXTtcbiAgICBpZiAoIXJlc3VsdC53YXJuaW5ncykgcmVzdWx0Lndhcm5pbmdzID0gW107XG4gICAgXG4gICAgLy8gQ29udGVudCB2YWxpZGF0aW9uXG4gICAgaWYgKCF0aGlzLmNvbnRlbnQgfHwgdGhpcy5jb250ZW50LnRyaW0oKS5sZW5ndGggPT09IDApIHtcbiAgICAgIHJlc3VsdC5lcnJvcnMucHVzaCh7XG4gICAgICAgIGZpZWxkOiAnY29udGVudCcsXG4gICAgICAgIG1lc3NhZ2U6ICdUZW1wbGF0ZSBjb250ZW50IGNhbm5vdCBiZSBlbXB0eScsXG4gICAgICAgIGNvZGU6ICdFTVBUWV9DT05URU5UJ1xuICAgICAgfSk7XG4gICAgfVxuICAgIFxuICAgIC8vIENoZWNrIGZvciB1bm1hdGNoZWQgdG9rZW5zXG4gICAgY29uc3Qgb3BlblRva2VucyA9ICh0aGlzLmNvbnRlbnQubWF0Y2goL1xce1xcey9nKSB8fCBbXSkubGVuZ3RoO1xuICAgIGNvbnN0IGNsb3NlVG9rZW5zID0gKHRoaXMuY29udGVudC5tYXRjaCgvXFx9XFx9L2cpIHx8IFtdKS5sZW5ndGg7XG4gICAgaWYgKG9wZW5Ub2tlbnMgIT09IGNsb3NlVG9rZW5zKSB7XG4gICAgICByZXN1bHQuZXJyb3JzLnB1c2goe1xuICAgICAgICBmaWVsZDogJ2NvbnRlbnQnLFxuICAgICAgICBtZXNzYWdlOiAnVGVtcGxhdGUgaGFzIHVubWF0Y2hlZCB2YXJpYWJsZSB0b2tlbnMnLFxuICAgICAgICBjb2RlOiAnVU5NQVRDSEVEX1RPS0VOUydcbiAgICAgIH0pO1xuICAgIH1cbiAgICBcbiAgICAvLyBWYWxpZGF0ZSBvdXRwdXQgZm9ybWF0XG4gICAgY29uc3QgdmFsaWRGb3JtYXRzID0gWydtYXJrZG93bicsICdodG1sJywgJ2pzb24nLCAneWFtbCcsICd0ZXh0JywgJ3htbCddO1xuICAgIGlmICh0aGlzLm1ldGFkYXRhLm91dHB1dF9mb3JtYXQgJiYgIXZhbGlkRm9ybWF0cy5pbmNsdWRlcyh0aGlzLm1ldGFkYXRhLm91dHB1dF9mb3JtYXQpKSB7XG4gICAgICByZXN1bHQud2FybmluZ3MucHVzaCh7XG4gICAgICAgIGZpZWxkOiAnb3V0cHV0X2Zvcm1hdCcsXG4gICAgICAgIG1lc3NhZ2U6IGBVbmtub3duIG91dHB1dCBmb3JtYXQgJyR7dGhpcy5tZXRhZGF0YS5vdXRwdXRfZm9ybWF0fScuIENvbW1vbiBmb3JtYXRzOiAke3ZhbGlkRm9ybWF0cy5qb2luKCcsICcpfWAsXG4gICAgICAgIHNldmVyaXR5OiAnbG93J1xuICAgICAgfSk7XG4gICAgfVxuICAgIFxuICAgIC8vIFZhbGlkYXRlIHZhcmlhYmxlc1xuICAgIGlmICh0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcykge1xuICAgICAgY29uc3QgdmFyaWFibGVOYW1lcyA9IG5ldyBTZXQ8c3RyaW5nPigpO1xuICAgICAgXG4gICAgICB0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcy5mb3JFYWNoKCh2YXJpYWJsZSwgaW5kZXgpID0+IHtcbiAgICAgICAgLy8gQ2hlY2sgZm9yIGR1cGxpY2F0ZSBuYW1lc1xuICAgICAgICBpZiAodmFyaWFibGVOYW1lcy5oYXModmFyaWFibGUubmFtZSkpIHtcbiAgICAgICAgICByZXN1bHQuZXJyb3JzIS5wdXNoKHtcbiAgICAgICAgICAgIGZpZWxkOiBgdmFyaWFibGVzWyR7aW5kZXh9XS5uYW1lYCxcbiAgICAgICAgICAgIG1lc3NhZ2U6IGBEdXBsaWNhdGUgdmFyaWFibGUgbmFtZSAnJHt2YXJpYWJsZS5uYW1lfSdgLFxuICAgICAgICAgICAgY29kZTogJ0RVUExJQ0FURV9WQVJJQUJMRSdcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgICB2YXJpYWJsZU5hbWVzLmFkZCh2YXJpYWJsZS5uYW1lKTtcbiAgICAgICAgXG4gICAgICAgIC8vIFZhbGlkYXRlIHJlZ2V4IHBhdHRlcm5zXG4gICAgICAgIGlmICh2YXJpYWJsZS52YWxpZGF0aW9uKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIG5ldyBSZWdFeHAodmFyaWFibGUudmFsaWRhdGlvbik7XG4gICAgICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICAgICAgcmVzdWx0LmVycm9ycyEucHVzaCh7XG4gICAgICAgICAgICAgIGZpZWxkOiBgdmFyaWFibGVzWyR7aW5kZXh9XS52YWxpZGF0aW9uYCxcbiAgICAgICAgICAgICAgbWVzc2FnZTogYEludmFsaWQgcmVnZXggcGF0dGVybjogJHtlfWAsXG4gICAgICAgICAgICAgIGNvZGU6ICdJTlZBTElEX1JFR0VYJ1xuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9XG4gICAgXG4gICAgLy8gQ2hlY2sgaWYgYWxsIHRva2VucyBoYXZlIGNvcnJlc3BvbmRpbmcgdmFyaWFibGUgZGVmaW5pdGlvbnNcbiAgICBjb25zdCBjb21waWxlZCA9IHRoaXMuY29tcGlsZSgpO1xuICAgIGNvbnN0IGRlZmluZWRWYXJzID0gbmV3IFNldCh0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcz8ubWFwKHYgPT4gdi5uYW1lKSB8fCBbXSk7XG4gICAgY29uc3QgdXNlZFZhcnMgPSBuZXcgU2V0KGNvbXBpbGVkLnRva2Vucy5tYXAodCA9PiB0LnZhcmlhYmxlLnNwbGl0KCcuJylbMF0pKTtcbiAgICBcbiAgICB1c2VkVmFycy5mb3JFYWNoKHZhck5hbWUgPT4ge1xuICAgICAgaWYgKCFkZWZpbmVkVmFycy5oYXModmFyTmFtZSkpIHtcbiAgICAgICAgcmVzdWx0Lndhcm5pbmdzIS5wdXNoKHtcbiAgICAgICAgICBmaWVsZDogJ3ZhcmlhYmxlcycsXG4gICAgICAgICAgbWVzc2FnZTogYFRlbXBsYXRlIHVzZXMgdW5kZWZpbmVkIHZhcmlhYmxlICcke3Zhck5hbWV9J2AsXG4gICAgICAgICAgc2V2ZXJpdHk6ICdtZWRpdW0nXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH0pO1xuICAgIFxuICAgIC8vIFdhcm5pbmdzIGZvciBiZXN0IHByYWN0aWNlc1xuICAgIGlmICghdGhpcy5tZXRhZGF0YS50YWdzIHx8IHRoaXMubWV0YWRhdGEudGFncy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJlc3VsdC53YXJuaW5ncyEucHVzaCh7XG4gICAgICAgIGZpZWxkOiAndGFncycsXG4gICAgICAgIG1lc3NhZ2U6ICdDb25zaWRlciBhZGRpbmcgdGFncyBmb3IgYmV0dGVyIHNlYXJjaGFiaWxpdHknLFxuICAgICAgICBzZXZlcml0eTogJ2xvdydcbiAgICAgIH0pO1xuICAgIH1cbiAgICBcbiAgICBpZiAoIXRoaXMubWV0YWRhdGEuZXhhbXBsZXMgfHwgdGhpcy5tZXRhZGF0YS5leGFtcGxlcy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJlc3VsdC53YXJuaW5ncyEucHVzaCh7XG4gICAgICAgIGZpZWxkOiAnZXhhbXBsZXMnLFxuICAgICAgICBtZXNzYWdlOiAnQWRkaW5nIGV4YW1wbGVzIGltcHJvdmVzIHRlbXBsYXRlIHVzYWJpbGl0eScsXG4gICAgICAgIHNldmVyaXR5OiAnbWVkaXVtJ1xuICAgICAgfSk7XG4gICAgfVxuICAgIFxuICAgIC8vIFVwZGF0ZSB0aGUgdmFsaWQgZmxhZyBiYXNlZCBvbiBmaW5hbCBlcnJvcnNcbiAgICByZXN1bHQudmFsaWQgPSAocmVzdWx0LmVycm9ycz8ubGVuZ3RoIHx8IDApID09PSAwO1xuICAgIFxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICAvKipcbiAgICogU2VyaWFsaXplIHRlbXBsYXRlIHRvIEpTT04gZm9ybWF0XG4gICAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgc2VyaWFsaXplKCk6IHN0cmluZyB7XG4gICAgY29uc3QgZGF0YSA9IHtcbiAgICAgIGlkOiB0aGlzLmlkLFxuICAgICAgdHlwZTogdGhpcy50eXBlLFxuICAgICAgdmVyc2lvbjogdGhpcy52ZXJzaW9uLFxuICAgICAgbWV0YWRhdGE6IHRoaXMubWV0YWRhdGEsXG4gICAgICBjb250ZW50OiB0aGlzLmNvbnRlbnQsXG4gICAgICByZWZlcmVuY2VzOiB0aGlzLnJlZmVyZW5jZXMsXG4gICAgICBleHRlbnNpb25zOiB0aGlzLmV4dGVuc2lvbnMsXG4gICAgICByYXRpbmdzOiB0aGlzLnJhdGluZ3NcbiAgICB9O1xuXG4gICAgcmV0dXJuIEpTT04uc3RyaW5naWZ5KGRhdGEsIG51bGwsIDIpO1xuICB9XG5cbiAgLyoqXG4gICAqIERlc2VyaWFsaXplIHRlbXBsYXRlIGZyb20gSlNPTiBmb3JtYXRcbiAgICovXG4gIHB1YmxpYyBvdmVycmlkZSBkZXNlcmlhbGl6ZShkYXRhOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZShkYXRhKTtcbiAgICAgIFxuICAgICAgLy8gVXBkYXRlIG1ldGFkYXRhXG4gICAgICB0aGlzLm1ldGFkYXRhID0geyAuLi50aGlzLm1ldGFkYXRhLCAuLi5wYXJzZWQubWV0YWRhdGEgfTtcbiAgICAgIFxuICAgICAgLy8gVXBkYXRlIGNvbnRlbnRcbiAgICAgIHRoaXMuY29udGVudCA9IHBhcnNlZC5jb250ZW50IHx8ICcnO1xuICAgICAgXG4gICAgICAvLyBVcGRhdGUgb3RoZXIgcHJvcGVydGllc1xuICAgICAgdGhpcy5yZWZlcmVuY2VzID0gcGFyc2VkLnJlZmVyZW5jZXMgfHwgW107XG4gICAgICB0aGlzLmV4dGVuc2lvbnMgPSBwYXJzZWQuZXh0ZW5zaW9ucyB8fCB7fTtcbiAgICAgIHRoaXMucmF0aW5ncyA9IHBhcnNlZC5yYXRpbmdzIHx8IHRoaXMucmF0aW5ncztcbiAgICAgIFxuICAgICAgLy8gVXBkYXRlIElEIGFuZCB2ZXJzaW9uIGlmIHByb3ZpZGVkXG4gICAgICBpZiAocGFyc2VkLmlkKSB0aGlzLmlkID0gcGFyc2VkLmlkO1xuICAgICAgaWYgKHBhcnNlZC52ZXJzaW9uKSB0aGlzLnZlcnNpb24gPSBwYXJzZWQudmVyc2lvbjtcbiAgICAgIFxuICAgICAgLy8gQ2xlYXIgY29tcGlsZWQgdGVtcGxhdGUgY2FjaGVcbiAgICAgIHRoaXMuY29tcGlsZWRUZW1wbGF0ZSA9IHVuZGVmaW5lZDtcbiAgICAgIFxuICAgICAgdGhpcy5faXNEaXJ0eSA9IHRydWU7XG4gICAgICBsb2dnZXIuZGVidWcoYERlc2VyaWFsaXplZCB0ZW1wbGF0ZTogJHt0aGlzLm1ldGFkYXRhLm5hbWV9YCk7XG4gICAgICBcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgbG9nZ2VyLmVycm9yKGBGYWlsZWQgdG8gZGVzZXJpYWxpemUgdGVtcGxhdGU6ICR7ZXJyb3J9YCk7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYERlc2VyaWFsaXphdGlvbiBmYWlsZWQ6ICR7ZXJyb3J9YCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEdldCBhIHByZXZpZXcgb2YgdGhlIHRlbXBsYXRlIHdpdGggc2FtcGxlIGRhdGFcbiAgICovXG4gIGFzeW5jIHByZXZpZXcoKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICBjb25zdCBzYW1wbGVWYXJzOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0ge307XG4gICAgXG4gICAgLy8gR2VuZXJhdGUgc2FtcGxlIGRhdGEgZm9yIGVhY2ggdmFyaWFibGVcbiAgICBmb3IgKGNvbnN0IHZhckRlZiBvZiB0aGlzLm1ldGFkYXRhLnZhcmlhYmxlcyB8fCBbXSkge1xuICAgICAgc3dpdGNoICh2YXJEZWYudHlwZSkge1xuICAgICAgICBjYXNlICdzdHJpbmcnOlxuICAgICAgICAgIHNhbXBsZVZhcnNbdmFyRGVmLm5hbWVdID0gdmFyRGVmLmRlZmF1bHQgfHwgYFske3ZhckRlZi5uYW1lfV1gO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlICdudW1iZXInOlxuICAgICAgICAgIHNhbXBsZVZhcnNbdmFyRGVmLm5hbWVdID0gdmFyRGVmLmRlZmF1bHQgfHwgNDI7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgJ2Jvb2xlYW4nOlxuICAgICAgICAgIHNhbXBsZVZhcnNbdmFyRGVmLm5hbWVdID0gdmFyRGVmLmRlZmF1bHQgfHwgdHJ1ZTtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSAnZGF0ZSc6XG4gICAgICAgICAgc2FtcGxlVmFyc1t2YXJEZWYubmFtZV0gPSB2YXJEZWYuZGVmYXVsdCB8fCBuZXcgRGF0ZSgpO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlICdhcnJheSc6XG4gICAgICAgICAgc2FtcGxlVmFyc1t2YXJEZWYubmFtZV0gPSB2YXJEZWYuZGVmYXVsdCB8fCBbJ2l0ZW0xJywgJ2l0ZW0yJ107XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgJ29iamVjdCc6XG4gICAgICAgICAgc2FtcGxlVmFyc1t2YXJEZWYubmFtZV0gPSB2YXJEZWYuZGVmYXVsdCB8fCB7IGtleTogJ3ZhbHVlJyB9O1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICByZXR1cm4gdGhpcy5yZW5kZXIoc2FtcGxlVmFycyk7XG4gIH1cblxuICAvKipcbiAgICogVGVtcGxhdGUgYWN0aXZhdGlvbiBsaWZlY3ljbGVcbiAgICovXG4gIHB1YmxpYyBvdmVycmlkZSBhc3luYyBhY3RpdmF0ZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBsb2dnZXIuaW5mbyhgQWN0aXZhdGluZyB0ZW1wbGF0ZTogJHt0aGlzLm1ldGFkYXRhLm5hbWV9ICgke3RoaXMuaWR9KWApO1xuICAgIFxuICAgIC8vIENvbXBpbGUgdGhlIHRlbXBsYXRlIHRvIGNoZWNrIGZvciBlcnJvcnNcbiAgICB0aGlzLmNvbXBpbGUoKTtcbiAgICBcbiAgICBhd2FpdCBzdXBlci5hY3RpdmF0ZT8uKCk7XG4gIH1cblxuICAvKipcbiAgICogVGVtcGxhdGUgZGVhY3RpdmF0aW9uIGxpZmVjeWNsZVxuICAgKi9cbiAgcHVibGljIG92ZXJyaWRlIGFzeW5jIGRlYWN0aXZhdGUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgbG9nZ2VyLmluZm8oYERlYWN0aXZhdGluZyB0ZW1wbGF0ZTogJHt0aGlzLm1ldGFkYXRhLm5hbWV9ICgke3RoaXMuaWR9KWApO1xuICAgIFxuICAgIC8vIENsZWFyIGNvbXBpbGVkIHRlbXBsYXRlIGNhY2hlXG4gICAgdGhpcy5jb21waWxlZFRlbXBsYXRlID0gdW5kZWZpbmVkO1xuICAgIFxuICAgIGF3YWl0IHN1cGVyLmRlYWN0aXZhdGU/LigpO1xuICB9XG59XG5cbi8vIEludGVybmFsIHR5cGVzIGZvciB0ZW1wbGF0ZSBjb21waWxhdGlvblxuaW50ZXJmYWNlIENvbXBpbGVkVGVtcGxhdGUge1xuICBjb250ZW50OiBzdHJpbmc7XG4gIHRva2VuczogVGVtcGxhdGVUb2tlbltdO1xuICB2YXJpYWJsZXM6IFRlbXBsYXRlVmFyaWFibGVbXTtcbn1cblxuaW50ZXJmYWNlIFRlbXBsYXRlVG9rZW4ge1xuICB0b2tlbjogc3RyaW5nOyAgICAgIC8vIFRoZSBmdWxsIHRva2VuIGluY2x1ZGluZyBicmFjZXMgKGUuZy4sIFwie3sgbmFtZSB9fVwiKVxuICB2YXJpYWJsZTogc3RyaW5nOyAgIC8vIFRoZSB2YXJpYWJsZSBwYXRoIChlLmcuLCBcInVzZXIubmFtZVwiKVxuICBwb3NpdGlvbjogbnVtYmVyOyAgIC8vIFBvc2l0aW9uIGluIHRoZSB0ZW1wbGF0ZSBzdHJpbmdcbn0iXX0=