@sumaris-net/ngx-components 18.23.62 → 21.0.0-rc1

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 (799) hide show
  1. package/README.md +2 -0
  2. package/doc/build.md +3 -5
  3. package/fesm2022/sumaris-net.ngx-components.mjs +2383 -2150
  4. package/fesm2022/sumaris-net.ngx-components.mjs.map +1 -1
  5. package/package.json +15 -15
  6. package/src/assets/environments/environment-test.json +2 -2
  7. package/src/assets/i18n/en-US.json +4 -1
  8. package/src/assets/i18n/en.json +4 -1
  9. package/src/assets/i18n/fr.json +10 -6
  10. package/src/assets/manifest.json +1 -1
  11. package/src/theme/_icons.scss +8 -12
  12. package/src/theme/_ionic.globals.scss +8 -1
  13. package/src/theme/_material.globals.scss +4 -1
  14. package/src/theme/_material.scss +16 -13
  15. package/src/theme/_mixins.scss +7 -1
  16. package/src/theme/_ngx-components.forms.scss +87 -84
  17. package/src/theme/_ngx-components.globals.scss +2 -0
  18. package/src/theme/_ngx-components.scss +19 -4
  19. package/src/theme/_ngx-components.table.scss +31 -26
  20. package/src/theme/_ngx-components.tabs.scss +16 -14
  21. package/src/theme/_responsive.scss +2 -0
  22. package/src/theme/_roboto.scss +3 -0
  23. package/src/theme/_theme.scss +11 -14
  24. package/src/theme/_theme.variables.scss +6 -1
  25. package/src/theme/ionic/ionic.functions.color.scss +13 -5
  26. package/src/theme/ionic/ionic.globals.scss +4 -4
  27. package/src/theme/ionic/ionic.theme.default.scss +2 -0
  28. package/types/sumaris-net.ngx-components.d.ts +14132 -0
  29. package/esm2022/public_api.mjs +0 -403
  30. package/esm2022/src/app/admin/admin-routing.module.mjs +0 -34
  31. package/esm2022/src/app/admin/admin.module.mjs +0 -32
  32. package/esm2022/src/app/admin/users/person.filter.mjs +0 -157
  33. package/esm2022/src/app/admin/users/person.service.mjs +0 -226
  34. package/esm2022/src/app/admin/users/person.validator.mjs +0 -78
  35. package/esm2022/src/app/admin/users/users-select.modal.mjs +0 -167
  36. package/esm2022/src/app/admin/users/users.mjs +0 -502
  37. package/esm2022/src/app/admin/users/users.module.mjs +0 -59
  38. package/esm2022/src/app/admin/users/users.utils.mjs +0 -29
  39. package/esm2022/src/app/core/about/about.modal.mjs +0 -119
  40. package/esm2022/src/app/core/about/about.module.mjs +0 -22
  41. package/esm2022/src/app/core/account/account.module.mjs +0 -75
  42. package/esm2022/src/app/core/account/account.page.mjs +0 -426
  43. package/esm2022/src/app/core/account/new-token.form.mjs +0 -109
  44. package/esm2022/src/app/core/account/new-token.modal.mjs +0 -66
  45. package/esm2022/src/app/core/account/password/change-password.form.mjs +0 -63
  46. package/esm2022/src/app/core/account/password/change-password.module.mjs +0 -22
  47. package/esm2022/src/app/core/account/password/change-password.page.mjs +0 -150
  48. package/esm2022/src/app/core/account/token.table.mjs +0 -160
  49. package/esm2022/src/app/core/auth/auth.form.mjs +0 -159
  50. package/esm2022/src/app/core/auth/auth.modal.mjs +0 -76
  51. package/esm2022/src/app/core/auth/auth.module.mjs +0 -56
  52. package/esm2022/src/app/core/auth/reset-password.modal.mjs +0 -89
  53. package/esm2022/src/app/core/core.module.mjs +0 -157
  54. package/esm2022/src/app/core/core.testing.module.mjs +0 -128
  55. package/esm2022/src/app/core/form/array/form-array.mjs +0 -328
  56. package/esm2022/src/app/core/form/array/testing/form-array-test.module.mjs +0 -23
  57. package/esm2022/src/app/core/form/array/testing/form-array.test.mjs +0 -97
  58. package/esm2022/src/app/core/form/buttons/form-buttons-bar.component.mjs +0 -111
  59. package/esm2022/src/app/core/form/buttons/form-buttons-bar.module.mjs +0 -31
  60. package/esm2022/src/app/core/form/entity/editor.class.mjs +0 -268
  61. package/esm2022/src/app/core/form/entity/entity-editor-modal.class.mjs +0 -420
  62. package/esm2022/src/app/core/form/entity/entity-editor.class.mjs +0 -748
  63. package/esm2022/src/app/core/form/entity/entity-metadata.component.mjs +0 -25
  64. package/esm2022/src/app/core/form/entity/entity.module.mjs +0 -33
  65. package/esm2022/src/app/core/form/entity/tab-editor.class.mjs +0 -190
  66. package/esm2022/src/app/core/form/form-container.class.mjs +0 -347
  67. package/esm2022/src/app/core/form/form.class.mjs +0 -342
  68. package/esm2022/src/app/core/form/form.module.mjs +0 -77
  69. package/esm2022/src/app/core/form/form.utils.mjs +0 -238
  70. package/esm2022/src/app/core/form/list/list.form.mjs +0 -243
  71. package/esm2022/src/app/core/form/list/list.module.mjs +0 -31
  72. package/esm2022/src/app/core/form/properties/properties-file.service.mjs +0 -133
  73. package/esm2022/src/app/core/form/properties/properties.form.mjs +0 -397
  74. package/esm2022/src/app/core/form/properties/properties.module.mjs +0 -85
  75. package/esm2022/src/app/core/form/properties/properties.table.mjs +0 -267
  76. package/esm2022/src/app/core/form/properties/properties.utils.mjs +0 -304
  77. package/esm2022/src/app/core/form/properties/property.validator.mjs +0 -70
  78. package/esm2022/src/app/core/form/properties/testing/properties-form.test.mjs +0 -103
  79. package/esm2022/src/app/core/form/properties/testing/properties-form.testing.module.mjs +0 -23
  80. package/esm2022/src/app/core/form/text/testing/text-form.testing.mjs +0 -42
  81. package/esm2022/src/app/core/form/text/testing/text-form.testing.module.mjs +0 -25
  82. package/esm2022/src/app/core/form/text/text-form.component.mjs +0 -205
  83. package/esm2022/src/app/core/form/text/text-form.module.mjs +0 -31
  84. package/esm2022/src/app/core/form/text-popover/testing/text-popover.testing.mjs +0 -73
  85. package/esm2022/src/app/core/form/text-popover/testing/text-popover.testing.module.mjs +0 -22
  86. package/esm2022/src/app/core/form/text-popover/text-popover.component.mjs +0 -164
  87. package/esm2022/src/app/core/form/text-popover/text-popover.module.mjs +0 -32
  88. package/esm2022/src/app/core/form/username/username.form.mjs +0 -55
  89. package/esm2022/src/app/core/form/username/username.module.mjs +0 -20
  90. package/esm2022/src/app/core/graphql/graphql.module.mjs +0 -14
  91. package/esm2022/src/app/core/graphql/graphql.service.mjs +0 -781
  92. package/esm2022/src/app/core/graphql/graphql.utils.mjs +0 -119
  93. package/esm2022/src/app/core/home/home.mjs +0 -472
  94. package/esm2022/src/app/core/home/home.module.mjs +0 -54
  95. package/esm2022/src/app/core/icon/icon.component.mjs +0 -40
  96. package/esm2022/src/app/core/icon/icon.module.mjs +0 -20
  97. package/esm2022/src/app/core/install/install-upgrade-card.component.mjs +0 -394
  98. package/esm2022/src/app/core/install/install-upgrade-card.module.mjs +0 -31
  99. package/esm2022/src/app/core/menu/menu.component.mjs +0 -273
  100. package/esm2022/src/app/core/menu/menu.model.mjs +0 -327
  101. package/esm2022/src/app/core/menu/menu.module.mjs +0 -47
  102. package/esm2022/src/app/core/menu/menu.service.mjs +0 -666
  103. package/esm2022/src/app/core/menu/sub-menu-tab.directive.mjs +0 -113
  104. package/esm2022/src/app/core/menu/testing/menu-other.testing.mjs +0 -51
  105. package/esm2022/src/app/core/menu/testing/menu.testing.mjs +0 -165
  106. package/esm2022/src/app/core/menu/testing/menu.testing.module.mjs +0 -60
  107. package/esm2022/src/app/core/offline/update-offline-mode-card.component.mjs +0 -28
  108. package/esm2022/src/app/core/offline/update-offline-mode-card.module.mjs +0 -31
  109. package/esm2022/src/app/core/peer/select-peer.modal.mjs +0 -350
  110. package/esm2022/src/app/core/peer/select-peer.module.mjs +0 -36
  111. package/esm2022/src/app/core/register/register-confirm.page.mjs +0 -90
  112. package/esm2022/src/app/core/register/register.form.mjs +0 -197
  113. package/esm2022/src/app/core/register/register.modal.mjs +0 -55
  114. package/esm2022/src/app/core/register/register.module.mjs +0 -23
  115. package/esm2022/src/app/core/services/account.service.mjs +0 -1403
  116. package/esm2022/src/app/core/services/auth-guard.service.mjs +0 -74
  117. package/esm2022/src/app/core/services/base-entity-service.class.mjs +0 -499
  118. package/esm2022/src/app/core/services/base-graphql-service.class.mjs +0 -224
  119. package/esm2022/src/app/core/services/base58.mjs +0 -83
  120. package/esm2022/src/app/core/services/config/core.config.mjs +0 -293
  121. package/esm2022/src/app/core/services/config.service.mjs +0 -384
  122. package/esm2022/src/app/core/services/crypto.service.mjs +0 -107
  123. package/esm2022/src/app/core/services/errors.mjs +0 -54
  124. package/esm2022/src/app/core/services/local-settings.service.mjs +0 -540
  125. package/esm2022/src/app/core/services/model/account.model.mjs +0 -192
  126. package/esm2022/src/app/core/services/model/config.model.mjs +0 -111
  127. package/esm2022/src/app/core/services/model/department.model.mjs +0 -35
  128. package/esm2022/src/app/core/services/model/entity.decorators.mjs +0 -97
  129. package/esm2022/src/app/core/services/model/entity.model.mjs +0 -347
  130. package/esm2022/src/app/core/services/model/filter.model.mjs +0 -169
  131. package/esm2022/src/app/core/services/model/history.model.mjs +0 -2
  132. package/esm2022/src/app/core/services/model/model.enum.mjs +0 -11
  133. package/esm2022/src/app/core/services/model/node-feature.model.mjs +0 -31
  134. package/esm2022/src/app/core/services/model/peer.model.mjs +0 -114
  135. package/esm2022/src/app/core/services/model/person.model.mjs +0 -118
  136. package/esm2022/src/app/core/services/model/referential.model.mjs +0 -207
  137. package/esm2022/src/app/core/services/model/settings.model.mjs +0 -6
  138. package/esm2022/src/app/core/services/model/token.model.mjs +0 -57
  139. package/esm2022/src/app/core/services/model/tree-item-entity.model.mjs +0 -234
  140. package/esm2022/src/app/core/services/network.service.mjs +0 -759
  141. package/esm2022/src/app/core/services/network.types.mjs +0 -2
  142. package/esm2022/src/app/core/services/network.utils.mjs +0 -35
  143. package/esm2022/src/app/core/services/pipes/account.pipes.mjs +0 -38
  144. package/esm2022/src/app/core/services/pipes/department-to-string.pipe.mjs +0 -23
  145. package/esm2022/src/app/core/services/pipes/person-to-string.pipe.mjs +0 -23
  146. package/esm2022/src/app/core/services/pipes/pipes.module.mjs +0 -58
  147. package/esm2022/src/app/core/services/pipes/referential-to-string.pipe.mjs +0 -36
  148. package/esm2022/src/app/core/services/pipes/usage-mode.pipes.mjs +0 -37
  149. package/esm2022/src/app/core/services/platform.service.mjs +0 -780
  150. package/esm2022/src/app/core/services/storage/entities-storage.service.mjs +0 -427
  151. package/esm2022/src/app/core/services/storage/entity-store.class.mjs +0 -450
  152. package/esm2022/src/app/core/services/testing/referential-filter.model.mjs +0 -25
  153. package/esm2022/src/app/core/services/testing/referential.validator.mjs +0 -46
  154. package/esm2022/src/app/core/services/validator/account.validator.mjs +0 -46
  155. package/esm2022/src/app/core/services/validator/base.validator.class.mjs +0 -49
  156. package/esm2022/src/app/core/services/validator/local-settings.validator.mjs +0 -57
  157. package/esm2022/src/app/core/services/validator/user-settings.validator.mjs +0 -27
  158. package/esm2022/src/app/core/services/validator/user-token.validator.mjs +0 -55
  159. package/esm2022/src/app/core/settings/settings.module.mjs +0 -23
  160. package/esm2022/src/app/core/settings/settings.page.mjs +0 -406
  161. package/esm2022/src/app/core/table/async-table.class.mjs +0 -1882
  162. package/esm2022/src/app/core/table/column/actions-column.component.mjs +0 -137
  163. package/esm2022/src/app/core/table/column/nav-actions-column.component.mjs +0 -120
  164. package/esm2022/src/app/core/table/column/row-field.component.mjs +0 -116
  165. package/esm2022/src/app/core/table/entities-async-table-datasource.class.mjs +0 -415
  166. package/esm2022/src/app/core/table/entities-table-datasource.class.mjs +0 -407
  167. package/esm2022/src/app/core/table/memory-table.class.mjs +0 -57
  168. package/esm2022/src/app/core/table/table-select-columns.component.mjs +0 -46
  169. package/esm2022/src/app/core/table/table.class.mjs +0 -1918
  170. package/esm2022/src/app/core/table/table.model.mjs +0 -21
  171. package/esm2022/src/app/core/table/table.module.mjs +0 -51
  172. package/esm2022/src/app/core/table/table.pipes.mjs +0 -76
  173. package/esm2022/src/app/core/table/table.utils.mjs +0 -34
  174. package/esm2022/src/app/core/table/testing/nested-table.testing.mjs +0 -326
  175. package/esm2022/src/app/core/table/testing/table-validator.service.mjs +0 -32
  176. package/esm2022/src/app/core/table/testing/table.testing.mjs +0 -342
  177. package/esm2022/src/app/core/table/testing/table.testing.module.mjs +0 -47
  178. package/esm2022/src/app/core/table/testing/table2-validator.service.mjs +0 -25
  179. package/esm2022/src/app/core/table/testing/table2.testing.mjs +0 -322
  180. package/esm2022/src/app/shared/alerts.mjs +0 -243
  181. package/esm2022/src/app/shared/audio/audio.mjs +0 -255
  182. package/esm2022/src/app/shared/audio/audio.testing.mjs +0 -32
  183. package/esm2022/src/app/shared/audio/audio.testing.module.mjs +0 -30
  184. package/esm2022/src/app/shared/base64.utils.mjs +0 -59
  185. package/esm2022/src/app/shared/capacitor/keyboard.mjs +0 -42
  186. package/esm2022/src/app/shared/capacitor/plugins.mjs +0 -10
  187. package/esm2022/src/app/shared/constants.mjs +0 -18
  188. package/esm2022/src/app/shared/dates.mjs +0 -212
  189. package/esm2022/src/app/shared/debug/debug-service.class.mjs +0 -3
  190. package/esm2022/src/app/shared/debug/debug.component.mjs +0 -44
  191. package/esm2022/src/app/shared/debug/debug.module.mjs +0 -23
  192. package/esm2022/src/app/shared/directives/autofocus.directive.mjs +0 -91
  193. package/esm2022/src/app/shared/directives/autoresize.directive.mjs +0 -75
  194. package/esm2022/src/app/shared/directives/autotitle.directive.mjs +0 -30
  195. package/esm2022/src/app/shared/directives/autotooltip.directive.mjs +0 -35
  196. package/esm2022/src/app/shared/directives/cell-selection/cell-identifier.directive.mjs +0 -55
  197. package/esm2022/src/app/shared/directives/cell-selection/cell-selection.directive.mjs +0 -206
  198. package/esm2022/src/app/shared/directives/cell-selection/cell-selection.service.mjs +0 -204
  199. package/esm2022/src/app/shared/directives/directives.module.mjs +0 -53
  200. package/esm2022/src/app/shared/directives/drag-and-drop.directive.mjs +0 -52
  201. package/esm2022/src/app/shared/directives/ng-var.directive.mjs +0 -33
  202. package/esm2022/src/app/shared/directives/resizable/resizable.component.mjs +0 -62
  203. package/esm2022/src/app/shared/directives/resizable/resizable.directive.mjs +0 -96
  204. package/esm2022/src/app/shared/directives/resizable/resizable.module.mjs +0 -17
  205. package/esm2022/src/app/shared/directives/throttled-click.directive.mjs +0 -43
  206. package/esm2022/src/app/shared/events.mjs +0 -64
  207. package/esm2022/src/app/shared/file/csv.utils.mjs +0 -97
  208. package/esm2022/src/app/shared/file/file.service.mjs +0 -118
  209. package/esm2022/src/app/shared/file/file.utils.mjs +0 -117
  210. package/esm2022/src/app/shared/file/images.utils.mjs +0 -186
  211. package/esm2022/src/app/shared/file/json.utils.mjs +0 -58
  212. package/esm2022/src/app/shared/file/uri.utils.mjs +0 -29
  213. package/esm2022/src/app/shared/file/url.utils.mjs +0 -213
  214. package/esm2022/src/app/shared/focusable.mjs +0 -6
  215. package/esm2022/src/app/shared/form/field.component.mjs +0 -388
  216. package/esm2022/src/app/shared/form/field.model.mjs +0 -100
  217. package/esm2022/src/app/shared/form/loading-spinner.mjs +0 -35
  218. package/esm2022/src/app/shared/forms.mjs +0 -593
  219. package/esm2022/src/app/shared/functions.mjs +0 -825
  220. package/esm2022/src/app/shared/geolocation/geolocation.utils.mjs +0 -94
  221. package/esm2022/src/app/shared/gesture/gesture-config.mjs +0 -46
  222. package/esm2022/src/app/shared/gesture/hammer.utils.mjs +0 -2
  223. package/esm2022/src/app/shared/graph/colors.utils.mjs +0 -85
  224. package/esm2022/src/app/shared/graph/graph-colors.mjs +0 -214
  225. package/esm2022/src/app/shared/guard/component-dirty.guard.mjs +0 -68
  226. package/esm2022/src/app/shared/hotkeys/dialog/hotkeys-dialog.component.mjs +0 -23
  227. package/esm2022/src/app/shared/hotkeys/hotkeys.service.mjs +0 -147
  228. package/esm2022/src/app/shared/hotkeys/shared-hotkeys.module.mjs +0 -24
  229. package/esm2022/src/app/shared/html.utils.mjs +0 -35
  230. package/esm2022/src/app/shared/http/http.utils.mjs +0 -56
  231. package/esm2022/src/app/shared/image/gallery/image-gallery.component.mjs +0 -452
  232. package/esm2022/src/app/shared/image/gallery/image-gallery.module.mjs +0 -62
  233. package/esm2022/src/app/shared/image/gallery/testing/gallegry.model.testing.mjs +0 -26
  234. package/esm2022/src/app/shared/image/gallery/testing/gallery.service.testing.mjs +0 -92
  235. package/esm2022/src/app/shared/image/gallery/testing/gallery.testing.mjs +0 -37
  236. package/esm2022/src/app/shared/image/gallery/testing/gallery.testing.module.mjs +0 -59
  237. package/esm2022/src/app/shared/image/image.model.mjs +0 -2
  238. package/esm2022/src/app/shared/image/image.module.mjs +0 -24
  239. package/esm2022/src/app/shared/image/image.service.mjs +0 -117
  240. package/esm2022/src/app/shared/inputs.mjs +0 -281
  241. package/esm2022/src/app/shared/interceptors/progess.interceptor.mjs +0 -27
  242. package/esm2022/src/app/shared/logging/log-level.model.mjs +0 -48
  243. package/esm2022/src/app/shared/logging/logger.model.mjs +0 -47
  244. package/esm2022/src/app/shared/logging/logging-service.class.mjs +0 -125
  245. package/esm2022/src/app/shared/logging/logging-service.config.mjs +0 -2
  246. package/esm2022/src/app/shared/logging/logging-service.module.mjs +0 -30
  247. package/esm2022/src/app/shared/markdown/markdown.component.mjs +0 -321
  248. package/esm2022/src/app/shared/markdown/markdown.directive.mjs +0 -34
  249. package/esm2022/src/app/shared/markdown/markdown.modal.mjs +0 -99
  250. package/esm2022/src/app/shared/markdown/markdown.module.mjs +0 -87
  251. package/esm2022/src/app/shared/markdown/markdown.service.mjs +0 -204
  252. package/esm2022/src/app/shared/markdown/markdown.utils.mjs +0 -87
  253. package/esm2022/src/app/shared/markdown/testing/markdown.test.mjs +0 -55
  254. package/esm2022/src/app/shared/markdown/testing/markdown.testing.module.mjs +0 -30
  255. package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.config.mjs +0 -46
  256. package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.mjs +0 -1356
  257. package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.module.mjs +0 -81
  258. package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.utils.mjs +0 -18
  259. package/esm2022/src/app/shared/material/autocomplete/testing/autocomplete.test.mjs +0 -242
  260. package/esm2022/src/app/shared/material/badge/badge.directive.mjs +0 -251
  261. package/esm2022/src/app/shared/material/badge/badge.module.mjs +0 -20
  262. package/esm2022/src/app/shared/material/badge/badge.test.mjs +0 -42
  263. package/esm2022/src/app/shared/material/boolean/boolean.module.mjs +0 -70
  264. package/esm2022/src/app/shared/material/boolean/material.boolean.mjs +0 -374
  265. package/esm2022/src/app/shared/material/boolean/testing/boolean.test.page.mjs +0 -83
  266. package/esm2022/src/app/shared/material/chips/chips.module.mjs +0 -74
  267. package/esm2022/src/app/shared/material/chips/material.chips.mjs +0 -843
  268. package/esm2022/src/app/shared/material/chips/testing/chips.test.mjs +0 -151
  269. package/esm2022/src/app/shared/material/datetime/datetime.module.mjs +0 -79
  270. package/esm2022/src/app/shared/material/datetime/material.date.mjs +0 -473
  271. package/esm2022/src/app/shared/material/datetime/material.dateshort.mjs +0 -457
  272. package/esm2022/src/app/shared/material/datetime/material.datetime.mjs +0 -850
  273. package/esm2022/src/app/shared/material/datetime/testing/mat-date-time.test.mjs +0 -140
  274. package/esm2022/src/app/shared/material/datetime/testing/mat-date.test.mjs +0 -127
  275. package/esm2022/src/app/shared/material/datetime/testing/mat-dateshort.test.mjs +0 -118
  276. package/esm2022/src/app/shared/material/duration/duration.module.mjs +0 -61
  277. package/esm2022/src/app/shared/material/duration/duration.utils.mjs +0 -16
  278. package/esm2022/src/app/shared/material/duration/material.duration.mjs +0 -283
  279. package/esm2022/src/app/shared/material/duration/testing/mat-duration.test.mjs +0 -85
  280. package/esm2022/src/app/shared/material/latlong/latlong.utils.mjs +0 -319
  281. package/esm2022/src/app/shared/material/latlong/material.latlong-input.mjs +0 -471
  282. package/esm2022/src/app/shared/material/latlong/material.latlong.mjs +0 -231
  283. package/esm2022/src/app/shared/material/latlong/material.latlong.module.mjs +0 -68
  284. package/esm2022/src/app/shared/material/latlong/testing/latlong.test.mjs +0 -138
  285. package/esm2022/src/app/shared/material/material.animations.mjs +0 -200
  286. package/esm2022/src/app/shared/material/material.config.mjs +0 -3
  287. package/esm2022/src/app/shared/material/material.module.mjs +0 -224
  288. package/esm2022/src/app/shared/material/material.testing.module.mjs +0 -234
  289. package/esm2022/src/app/shared/material/paginator/material.paginator-i18n.mjs +0 -51
  290. package/esm2022/src/app/shared/material/stepper/material.stepper-i18n.mjs +0 -23
  291. package/esm2022/src/app/shared/material/swipe/material.swipe.mjs +0 -379
  292. package/esm2022/src/app/shared/material/swipe/swipe.module.mjs +0 -52
  293. package/esm2022/src/app/shared/material/swipe/testing/swipe.test.mjs +0 -67
  294. package/esm2022/src/app/shared/material/test/test-component.mjs +0 -264
  295. package/esm2022/src/app/shared/material/testing/common.test.mjs +0 -111
  296. package/esm2022/src/app/shared/modules.mjs +0 -12
  297. package/esm2022/src/app/shared/named-filter/named-filter-selector.component.mjs +0 -306
  298. package/esm2022/src/app/shared/named-filter/named-filter.model.mjs +0 -49
  299. package/esm2022/src/app/shared/named-filter/named-filter.module.mjs +0 -24
  300. package/esm2022/src/app/shared/named-filter/named-filter.service.mjs +0 -69
  301. package/esm2022/src/app/shared/named-filter/testing/named-filter-selector.testing.mjs +0 -74
  302. package/esm2022/src/app/shared/named-filter/testing/named-filter.testing.module.mjs +0 -28
  303. package/esm2022/src/app/shared/observables.mjs +0 -104
  304. package/esm2022/src/app/shared/pipes/arrays.pipe.mjs +0 -251
  305. package/esm2022/src/app/shared/pipes/badge.pipes.mjs +0 -25
  306. package/esm2022/src/app/shared/pipes/colors.pipe.mjs +0 -29
  307. package/esm2022/src/app/shared/pipes/date-diff-duration.pipe.mjs +0 -50
  308. package/esm2022/src/app/shared/pipes/date-format.pipe.mjs +0 -72
  309. package/esm2022/src/app/shared/pipes/date-from-now.pipe.mjs +0 -28
  310. package/esm2022/src/app/shared/pipes/date-from.pipe.mjs +0 -29
  311. package/esm2022/src/app/shared/pipes/dates.pipe.mjs +0 -18
  312. package/esm2022/src/app/shared/pipes/display-with.pipe.mjs +0 -23
  313. package/esm2022/src/app/shared/pipes/duration.pipe.mjs +0 -34
  314. package/esm2022/src/app/shared/pipes/file-size.pipe.mjs +0 -53
  315. package/esm2022/src/app/shared/pipes/form.pipes.mjs +0 -219
  316. package/esm2022/src/app/shared/pipes/highlight.pipe.mjs +0 -50
  317. package/esm2022/src/app/shared/pipes/html.pipes.mjs +0 -102
  318. package/esm2022/src/app/shared/pipes/latlong-format.pipe.mjs +0 -42
  319. package/esm2022/src/app/shared/pipes/maps.pipe.mjs +0 -65
  320. package/esm2022/src/app/shared/pipes/maskito.pipe.mjs +0 -67
  321. package/esm2022/src/app/shared/pipes/math.pipes.mjs +0 -77
  322. package/esm2022/src/app/shared/pipes/ng-init.pipe.mjs +0 -19
  323. package/esm2022/src/app/shared/pipes/number-format.pipe.mjs +0 -17
  324. package/esm2022/src/app/shared/pipes/observable.pipes.mjs +0 -63
  325. package/esm2022/src/app/shared/pipes/pipes.module.mjs +0 -304
  326. package/esm2022/src/app/shared/pipes/property.pipes.mjs +0 -148
  327. package/esm2022/src/app/shared/pipes/selection.pipes.mjs +0 -177
  328. package/esm2022/src/app/shared/pipes/string.pipes.mjs +0 -183
  329. package/esm2022/src/app/shared/pipes/translate-context.pipe.mjs +0 -44
  330. package/esm2022/src/app/shared/pipes/types.pipes.mjs +0 -74
  331. package/esm2022/src/app/shared/pipes/url.pipes.mjs +0 -35
  332. package/esm2022/src/app/shared/platforms.mjs +0 -81
  333. package/esm2022/src/app/shared/print/print.service.mjs +0 -305
  334. package/esm2022/src/app/shared/regexps.mjs +0 -103
  335. package/esm2022/src/app/shared/rx-state/rx-state.decorators.mjs +0 -193
  336. package/esm2022/src/app/shared/rx-state/rx-state.module.mjs +0 -19
  337. package/esm2022/src/app/shared/rx-state/rx-state.types.mjs +0 -2
  338. package/esm2022/src/app/shared/services/entity-service.class.mjs +0 -44
  339. package/esm2022/src/app/shared/services/job.utils.mjs +0 -92
  340. package/esm2022/src/app/shared/services/memory-entity-service.class.mjs +0 -362
  341. package/esm2022/src/app/shared/services/progress-bar.service.mjs +0 -31
  342. package/esm2022/src/app/shared/services/startable-observable-service.class.mjs +0 -144
  343. package/esm2022/src/app/shared/services/startable-service.class.mjs +0 -118
  344. package/esm2022/src/app/shared/services/translate-context.service.mjs +0 -107
  345. package/esm2022/src/app/shared/services/validator-service.class.mjs +0 -2
  346. package/esm2022/src/app/shared/services.mjs +0 -85
  347. package/esm2022/src/app/shared/shared-routing.module.mjs +0 -72
  348. package/esm2022/src/app/shared/shared.module.mjs +0 -229
  349. package/esm2022/src/app/shared/shared.testing.module.mjs +0 -108
  350. package/esm2022/src/app/shared/storage/storage-explorer.component.mjs +0 -281
  351. package/esm2022/src/app/shared/storage/storage-explorer.module.mjs +0 -42
  352. package/esm2022/src/app/shared/storage/storage-explorer.testing-routing.module.mjs +0 -27
  353. package/esm2022/src/app/shared/storage/storage-explorer.testing.module.mjs +0 -25
  354. package/esm2022/src/app/shared/storage/storage.service.mjs +0 -134
  355. package/esm2022/src/app/shared/storage/storage.utils.mjs +0 -13
  356. package/esm2022/src/app/shared/testing/maskito.test.mjs +0 -41
  357. package/esm2022/src/app/shared/testing/observable.test.mjs +0 -94
  358. package/esm2022/src/app/shared/testing/tests.page.mjs +0 -37
  359. package/esm2022/src/app/shared/toast/toast.testing.mjs +0 -54
  360. package/esm2022/src/app/shared/toast/toast.testing.module.mjs +0 -31
  361. package/esm2022/src/app/shared/toast/toasts.mjs +0 -134
  362. package/esm2022/src/app/shared/toolbar/modal-toolbar.mjs +0 -62
  363. package/esm2022/src/app/shared/toolbar/toolbar.mjs +0 -272
  364. package/esm2022/src/app/shared/toolbar/toolbar.module.mjs +0 -26
  365. package/esm2022/src/app/shared/types.mjs +0 -2
  366. package/esm2022/src/app/shared/upload-file/testing/upload-file.testing.mjs +0 -59
  367. package/esm2022/src/app/shared/upload-file/testing/upload-file.testing.module.mjs +0 -30
  368. package/esm2022/src/app/shared/upload-file/upload-file-popover.component.mjs +0 -108
  369. package/esm2022/src/app/shared/upload-file/upload-file.component.mjs +0 -216
  370. package/esm2022/src/app/shared/upload-file/upload-file.model.mjs +0 -26
  371. package/esm2022/src/app/shared/validator/form-error-adapter.class.mjs +0 -122
  372. package/esm2022/src/app/shared/validator/validators.mjs +0 -658
  373. package/esm2022/src/app/shared/version/versions.mjs +0 -81
  374. package/esm2022/src/app/social/config/social.config.mjs +0 -9
  375. package/esm2022/src/app/social/feed/feed.component.mjs +0 -317
  376. package/esm2022/src/app/social/feed/feed.directive.mjs +0 -151
  377. package/esm2022/src/app/social/feed/feed.model.mjs +0 -60
  378. package/esm2022/src/app/social/feed/feed.module.mjs +0 -24
  379. package/esm2022/src/app/social/feed/feed.page.mjs +0 -67
  380. package/esm2022/src/app/social/feed/feed.service.mjs +0 -222
  381. package/esm2022/src/app/social/feed/testing/feed.testing.mjs +0 -66
  382. package/esm2022/src/app/social/feed/testing/feed.testing.module.mjs +0 -23
  383. package/esm2022/src/app/social/job/job.module.mjs +0 -23
  384. package/esm2022/src/app/social/job/progression/job-progression.component.mjs +0 -38
  385. package/esm2022/src/app/social/job/progression/job-progression.icon.mjs +0 -241
  386. package/esm2022/src/app/social/job/progression/job-progression.list.mjs +0 -50
  387. package/esm2022/src/app/social/job/progression/job-progression.model.mjs +0 -29
  388. package/esm2022/src/app/social/job/progression/job-progression.service.mjs +0 -91
  389. package/esm2022/src/app/social/job/testing/job-progression.testing.mjs +0 -38
  390. package/esm2022/src/app/social/job/testing/job-progression.testing.service.mjs +0 -34
  391. package/esm2022/src/app/social/job/testing/job.testing.module.mjs +0 -22
  392. package/esm2022/src/app/social/message/message.form.mjs +0 -149
  393. package/esm2022/src/app/social/message/message.modal.mjs +0 -116
  394. package/esm2022/src/app/social/message/message.model.mjs +0 -86
  395. package/esm2022/src/app/social/message/message.module.mjs +0 -25
  396. package/esm2022/src/app/social/message/message.service.mjs +0 -198
  397. package/esm2022/src/app/social/social.errors.mjs +0 -11
  398. package/esm2022/src/app/social/social.module.mjs +0 -22
  399. package/esm2022/src/app/social/social.testing.module.mjs +0 -72
  400. package/esm2022/src/app/social/user-event/notification/user-event-notification.icon.mjs +0 -191
  401. package/esm2022/src/app/social/user-event/notification/user-event-notification.list.mjs +0 -280
  402. package/esm2022/src/app/social/user-event/notification/user-event-notification.modal.mjs +0 -84
  403. package/esm2022/src/app/social/user-event/testing/user-event.testing.mjs +0 -273
  404. package/esm2022/src/app/social/user-event/testing/user-event.testing.model.mjs +0 -129
  405. package/esm2022/src/app/social/user-event/testing/user-event.testing.module.mjs +0 -22
  406. package/esm2022/src/app/social/user-event/testing/user-event.testing.service.mjs +0 -128
  407. package/esm2022/src/app/social/user-event/user-event.model.mjs +0 -2
  408. package/esm2022/src/app/social/user-event/user-event.module.mjs +0 -24
  409. package/esm2022/src/app/social/user-event/user-event.service.mjs +0 -512
  410. package/esm2022/src/environments/environment.class.mjs +0 -69
  411. package/esm2022/src/environments/environment.loader.mjs +0 -77
  412. package/esm2022/src/environments/environment.mjs +0 -133
  413. package/esm2022/sumaris-net.ngx-components.mjs +0 -5
  414. package/index.d.ts +0 -5
  415. package/public_api.d.ts +0 -364
  416. package/src/app/admin/admin-routing.module.d.ts +0 -9
  417. package/src/app/admin/admin.module.d.ts +0 -10
  418. package/src/app/admin/users/person.filter.d.ts +0 -37
  419. package/src/app/admin/users/person.service.d.ts +0 -61
  420. package/src/app/admin/users/person.validator.d.ts +0 -23
  421. package/src/app/admin/users/users-select.modal.d.ts +0 -73
  422. package/src/app/admin/users/users.d.ts +0 -89
  423. package/src/app/admin/users/users.module.d.ts +0 -15
  424. package/src/app/admin/users/users.utils.d.ts +0 -6
  425. package/src/app/core/about/about.modal.d.ts +0 -44
  426. package/src/app/core/about/about.module.d.ts +0 -11
  427. package/src/app/core/account/account.module.d.ts +0 -18
  428. package/src/app/core/account/account.page.d.ts +0 -93
  429. package/src/app/core/account/new-token.form.d.ts +0 -30
  430. package/src/app/core/account/new-token.modal.d.ts +0 -29
  431. package/src/app/core/account/password/change-password.form.d.ts +0 -19
  432. package/src/app/core/account/password/change-password.module.d.ts +0 -11
  433. package/src/app/core/account/password/change-password.page.d.ts +0 -36
  434. package/src/app/core/account/token.table.d.ts +0 -28
  435. package/src/app/core/auth/auth.form.d.ts +0 -38
  436. package/src/app/core/auth/auth.modal.d.ts +0 -24
  437. package/src/app/core/auth/auth.module.d.ts +0 -13
  438. package/src/app/core/auth/reset-password.modal.d.ts +0 -29
  439. package/src/app/core/core.module.d.ts +0 -28
  440. package/src/app/core/core.testing.module.d.ts +0 -16
  441. package/src/app/core/form/array/form-array.d.ts +0 -147
  442. package/src/app/core/form/array/testing/form-array-test.module.d.ts +0 -12
  443. package/src/app/core/form/array/testing/form-array.test.d.ts +0 -27
  444. package/src/app/core/form/buttons/form-buttons-bar.component.d.ts +0 -38
  445. package/src/app/core/form/buttons/form-buttons-bar.module.d.ts +0 -9
  446. package/src/app/core/form/entity/editor.class.d.ts +0 -85
  447. package/src/app/core/form/entity/entity-editor-modal.class.d.ts +0 -126
  448. package/src/app/core/form/entity/entity-editor.class.d.ts +0 -211
  449. package/src/app/core/form/entity/entity-metadata.component.d.ts +0 -13
  450. package/src/app/core/form/entity/entity.module.d.ts +0 -11
  451. package/src/app/core/form/entity/tab-editor.class.d.ts +0 -66
  452. package/src/app/core/form/form-container.class.d.ts +0 -131
  453. package/src/app/core/form/form.class.d.ts +0 -148
  454. package/src/app/core/form/form.module.d.ts +0 -15
  455. package/src/app/core/form/form.utils.d.ts +0 -301
  456. package/src/app/core/form/list/list.form.d.ts +0 -83
  457. package/src/app/core/form/list/list.module.d.ts +0 -9
  458. package/src/app/core/form/properties/properties-file.service.d.ts +0 -26
  459. package/src/app/core/form/properties/properties.form.d.ts +0 -109
  460. package/src/app/core/form/properties/properties.module.d.ts +0 -18
  461. package/src/app/core/form/properties/properties.table.d.ts +0 -73
  462. package/src/app/core/form/properties/properties.utils.d.ts +0 -59
  463. package/src/app/core/form/properties/property.validator.d.ts +0 -27
  464. package/src/app/core/form/properties/testing/properties-form.test.d.ts +0 -22
  465. package/src/app/core/form/properties/testing/properties-form.testing.module.d.ts +0 -12
  466. package/src/app/core/form/text/testing/text-form.testing.d.ts +0 -19
  467. package/src/app/core/form/text/testing/text-form.testing.module.d.ts +0 -14
  468. package/src/app/core/form/text/text-form.component.d.ts +0 -66
  469. package/src/app/core/form/text/text-form.module.d.ts +0 -9
  470. package/src/app/core/form/text-popover/testing/text-popover.testing.d.ts +0 -16
  471. package/src/app/core/form/text-popover/testing/text-popover.testing.module.d.ts +0 -11
  472. package/src/app/core/form/text-popover/text-popover.component.d.ts +0 -81
  473. package/src/app/core/form/text-popover/text-popover.module.d.ts +0 -10
  474. package/src/app/core/form/username/username.form.d.ts +0 -19
  475. package/src/app/core/form/username/username.module.d.ts +0 -9
  476. package/src/app/core/graphql/graphql.module.d.ts +0 -7
  477. package/src/app/core/graphql/graphql.service.d.ts +0 -138
  478. package/src/app/core/graphql/graphql.utils.d.ts +0 -47
  479. package/src/app/core/home/home.d.ts +0 -135
  480. package/src/app/core/home/home.module.d.ts +0 -16
  481. package/src/app/core/icon/icon.component.d.ts +0 -13
  482. package/src/app/core/icon/icon.module.d.ts +0 -9
  483. package/src/app/core/install/install-upgrade-card.component.d.ts +0 -63
  484. package/src/app/core/install/install-upgrade-card.module.d.ts +0 -9
  485. package/src/app/core/menu/menu.component.d.ts +0 -69
  486. package/src/app/core/menu/menu.model.d.ts +0 -161
  487. package/src/app/core/menu/menu.module.d.ts +0 -12
  488. package/src/app/core/menu/menu.service.d.ts +0 -93
  489. package/src/app/core/menu/sub-menu-tab.directive.d.ts +0 -29
  490. package/src/app/core/menu/testing/menu-other.testing.d.ts +0 -15
  491. package/src/app/core/menu/testing/menu.testing.d.ts +0 -53
  492. package/src/app/core/menu/testing/menu.testing.module.d.ts +0 -12
  493. package/src/app/core/offline/update-offline-mode-card.component.d.ts +0 -11
  494. package/src/app/core/offline/update-offline-mode-card.module.d.ts +0 -9
  495. package/src/app/core/peer/select-peer.modal.d.ts +0 -108
  496. package/src/app/core/peer/select-peer.module.d.ts +0 -12
  497. package/src/app/core/register/register-confirm.page.d.ts +0 -24
  498. package/src/app/core/register/register.form.d.ts +0 -49
  499. package/src/app/core/register/register.modal.d.ts +0 -14
  500. package/src/app/core/register/register.module.d.ts +0 -12
  501. package/src/app/core/services/account.service.d.ts +0 -236
  502. package/src/app/core/services/auth-guard.service.d.ts +0 -17
  503. package/src/app/core/services/base-entity-service.class.d.ts +0 -134
  504. package/src/app/core/services/base-graphql-service.class.d.ts +0 -77
  505. package/src/app/core/services/base58.d.ts +0 -7
  506. package/src/app/core/services/config/core.config.d.ts +0 -52
  507. package/src/app/core/services/config.service.d.ts +0 -63
  508. package/src/app/core/services/crypto.service.d.ts +0 -44
  509. package/src/app/core/services/errors.d.ts +0 -65
  510. package/src/app/core/services/local-settings.service.d.ts +0 -130
  511. package/src/app/core/services/model/account.model.d.ts +0 -66
  512. package/src/app/core/services/model/config.model.d.ts +0 -37
  513. package/src/app/core/services/model/department.model.d.ts +0 -11
  514. package/src/app/core/services/model/entity.decorators.d.ts +0 -17
  515. package/src/app/core/services/model/entity.model.d.ts +0 -133
  516. package/src/app/core/services/model/filter.model.d.ts +0 -37
  517. package/src/app/core/services/model/history.model.d.ts +0 -20
  518. package/src/app/core/services/model/model.enum.d.ts +0 -7
  519. package/src/app/core/services/model/node-feature.model.d.ts +0 -14
  520. package/src/app/core/services/model/peer.model.d.ts +0 -35
  521. package/src/app/core/services/model/person.model.d.ts +0 -39
  522. package/src/app/core/services/model/referential.model.d.ts +0 -85
  523. package/src/app/core/services/model/settings.model.d.ts +0 -29
  524. package/src/app/core/services/model/token.model.d.ts +0 -22
  525. package/src/app/core/services/model/tree-item-entity.model.d.ts +0 -62
  526. package/src/app/core/services/network.service.d.ts +0 -184
  527. package/src/app/core/services/network.types.d.ts +0 -4
  528. package/src/app/core/services/network.utils.d.ts +0 -39
  529. package/src/app/core/services/pipes/account.pipes.d.ts +0 -15
  530. package/src/app/core/services/pipes/department-to-string.pipe.d.ts +0 -9
  531. package/src/app/core/services/pipes/person-to-string.pipe.d.ts +0 -11
  532. package/src/app/core/services/pipes/pipes.module.d.ts +0 -11
  533. package/src/app/core/services/pipes/referential-to-string.pipe.d.ts +0 -23
  534. package/src/app/core/services/pipes/usage-mode.pipes.d.ts +0 -15
  535. package/src/app/core/services/platform.service.d.ts +0 -126
  536. package/src/app/core/services/storage/entities-storage.service.d.ts +0 -108
  537. package/src/app/core/services/storage/entity-store.class.d.ts +0 -113
  538. package/src/app/core/services/testing/referential-filter.model.d.ts +0 -9
  539. package/src/app/core/services/testing/referential.validator.d.ts +0 -22
  540. package/src/app/core/services/validator/account.validator.d.ts +0 -24
  541. package/src/app/core/services/validator/base.validator.class.d.ts +0 -21
  542. package/src/app/core/services/validator/local-settings.validator.d.ts +0 -18
  543. package/src/app/core/services/validator/user-settings.validator.d.ts +0 -10
  544. package/src/app/core/services/validator/user-token.validator.d.ts +0 -13
  545. package/src/app/core/settings/settings.module.d.ts +0 -12
  546. package/src/app/core/settings/settings.page.d.ts +0 -89
  547. package/src/app/core/table/async-table.class.d.ts +0 -443
  548. package/src/app/core/table/column/actions-column.component.d.ts +0 -65
  549. package/src/app/core/table/column/nav-actions-column.component.d.ts +0 -47
  550. package/src/app/core/table/column/row-field.component.d.ts +0 -41
  551. package/src/app/core/table/entities-async-table-datasource.class.d.ts +0 -103
  552. package/src/app/core/table/entities-table-datasource.class.d.ts +0 -102
  553. package/src/app/core/table/memory-table.class.d.ts +0 -25
  554. package/src/app/core/table/table-select-columns.component.d.ts +0 -22
  555. package/src/app/core/table/table.class.d.ts +0 -427
  556. package/src/app/core/table/table.model.d.ts +0 -31
  557. package/src/app/core/table/table.module.d.ts +0 -14
  558. package/src/app/core/table/table.pipes.d.ts +0 -30
  559. package/src/app/core/table/table.utils.d.ts +0 -13
  560. package/src/app/core/table/testing/nested-table.testing.d.ts +0 -54
  561. package/src/app/core/table/testing/table-validator.service.d.ts +0 -17
  562. package/src/app/core/table/testing/table.testing.d.ts +0 -55
  563. package/src/app/core/table/testing/table.testing.module.d.ts +0 -17
  564. package/src/app/core/table/testing/table2-validator.service.d.ts +0 -15
  565. package/src/app/core/table/testing/table2.testing.d.ts +0 -53
  566. package/src/app/shared/alerts.d.ts +0 -55
  567. package/src/app/shared/audio/audio.d.ts +0 -45
  568. package/src/app/shared/audio/audio.testing.d.ts +0 -13
  569. package/src/app/shared/audio/audio.testing.module.d.ts +0 -11
  570. package/src/app/shared/base64.utils.d.ts +0 -7
  571. package/src/app/shared/capacitor/keyboard.d.ts +0 -16
  572. package/src/app/shared/capacitor/plugins.d.ts +0 -8
  573. package/src/app/shared/constants.d.ts +0 -15
  574. package/src/app/shared/dates.d.ts +0 -85
  575. package/src/app/shared/debug/debug-service.class.d.ts +0 -5
  576. package/src/app/shared/debug/debug.component.d.ts +0 -14
  577. package/src/app/shared/debug/debug.module.d.ts +0 -12
  578. package/src/app/shared/directives/autofocus.directive.d.ts +0 -18
  579. package/src/app/shared/directives/autoresize.directive.d.ts +0 -20
  580. package/src/app/shared/directives/autotitle.directive.d.ts +0 -11
  581. package/src/app/shared/directives/autotooltip.directive.d.ts +0 -14
  582. package/src/app/shared/directives/cell-selection/cell-identifier.directive.d.ts +0 -17
  583. package/src/app/shared/directives/cell-selection/cell-selection.directive.d.ts +0 -29
  584. package/src/app/shared/directives/cell-selection/cell-selection.service.d.ts +0 -67
  585. package/src/app/shared/directives/directives.module.d.ts +0 -15
  586. package/src/app/shared/directives/drag-and-drop.directive.d.ts +0 -17
  587. package/src/app/shared/directives/ng-var.directive.d.ts +0 -12
  588. package/src/app/shared/directives/resizable/resizable.component.d.ts +0 -21
  589. package/src/app/shared/directives/resizable/resizable.directive.d.ts +0 -24
  590. package/src/app/shared/directives/resizable/resizable.module.d.ts +0 -8
  591. package/src/app/shared/directives/throttled-click.directive.d.ts +0 -16
  592. package/src/app/shared/events.d.ts +0 -29
  593. package/src/app/shared/file/csv.utils.d.ts +0 -30
  594. package/src/app/shared/file/file.service.d.ts +0 -23
  595. package/src/app/shared/file/file.utils.d.ts +0 -20
  596. package/src/app/shared/file/images.utils.d.ts +0 -36
  597. package/src/app/shared/file/json.utils.d.ts +0 -19
  598. package/src/app/shared/file/uri.utils.d.ts +0 -8
  599. package/src/app/shared/file/url.utils.d.ts +0 -108
  600. package/src/app/shared/focusable.d.ts +0 -4
  601. package/src/app/shared/form/field.component.d.ts +0 -83
  602. package/src/app/shared/form/field.model.d.ts +0 -88
  603. package/src/app/shared/form/loading-spinner.d.ts +0 -8
  604. package/src/app/shared/forms.d.ts +0 -170
  605. package/src/app/shared/functions.d.ts +0 -296
  606. package/src/app/shared/geolocation/geolocation.utils.d.ts +0 -25
  607. package/src/app/shared/gesture/gesture-config.d.ts +0 -15
  608. package/src/app/shared/gesture/hammer.utils.d.ts +0 -16
  609. package/src/app/shared/graph/colors.utils.d.ts +0 -34
  610. package/src/app/shared/graph/graph-colors.d.ts +0 -63
  611. package/src/app/shared/guard/component-dirty.guard.d.ts +0 -14
  612. package/src/app/shared/hotkeys/dialog/hotkeys-dialog.component.d.ts +0 -8
  613. package/src/app/shared/hotkeys/hotkeys.service.d.ts +0 -48
  614. package/src/app/shared/hotkeys/shared-hotkeys.module.d.ts +0 -11
  615. package/src/app/shared/html.utils.d.ts +0 -17
  616. package/src/app/shared/http/http.utils.d.ts +0 -28
  617. package/src/app/shared/image/gallery/image-gallery.component.d.ts +0 -139
  618. package/src/app/shared/image/gallery/image-gallery.module.d.ts +0 -15
  619. package/src/app/shared/image/gallery/testing/gallegry.model.testing.d.ts +0 -12
  620. package/src/app/shared/image/gallery/testing/gallery.service.testing.d.ts +0 -33
  621. package/src/app/shared/image/gallery/testing/gallery.testing.d.ts +0 -16
  622. package/src/app/shared/image/gallery/testing/gallery.testing.module.d.ts +0 -13
  623. package/src/app/shared/image/image.model.d.ts +0 -7
  624. package/src/app/shared/image/image.module.d.ts +0 -8
  625. package/src/app/shared/image/image.service.d.ts +0 -37
  626. package/src/app/shared/inputs.d.ts +0 -37
  627. package/src/app/shared/interceptors/progess.interceptor.d.ts +0 -8
  628. package/src/app/shared/logging/log-level.model.d.ts +0 -38
  629. package/src/app/shared/logging/logger.model.d.ts +0 -68
  630. package/src/app/shared/logging/logging-service.class.d.ts +0 -39
  631. package/src/app/shared/logging/logging-service.config.d.ts +0 -25
  632. package/src/app/shared/logging/logging-service.module.d.ts +0 -9
  633. package/src/app/shared/markdown/markdown.component.d.ts +0 -86
  634. package/src/app/shared/markdown/markdown.directive.d.ts +0 -13
  635. package/src/app/shared/markdown/markdown.modal.d.ts +0 -49
  636. package/src/app/shared/markdown/markdown.module.d.ts +0 -17
  637. package/src/app/shared/markdown/markdown.service.d.ts +0 -37
  638. package/src/app/shared/markdown/markdown.utils.d.ts +0 -26
  639. package/src/app/shared/markdown/testing/markdown.test.d.ts +0 -18
  640. package/src/app/shared/markdown/testing/markdown.testing.module.d.ts +0 -12
  641. package/src/app/shared/material/autocomplete/material.autocomplete.config.d.ts +0 -74
  642. package/src/app/shared/material/autocomplete/material.autocomplete.d.ts +0 -224
  643. package/src/app/shared/material/autocomplete/material.autocomplete.module.d.ts +0 -23
  644. package/src/app/shared/material/autocomplete/material.autocomplete.utils.d.ts +0 -5
  645. package/src/app/shared/material/autocomplete/testing/autocomplete.test.d.ts +0 -48
  646. package/src/app/shared/material/badge/badge.directive.d.ts +0 -43
  647. package/src/app/shared/material/badge/badge.module.d.ts +0 -10
  648. package/src/app/shared/material/badge/badge.test.d.ts +0 -21
  649. package/src/app/shared/material/boolean/boolean.module.d.ts +0 -21
  650. package/src/app/shared/material/boolean/material.boolean.d.ts +0 -94
  651. package/src/app/shared/material/boolean/testing/boolean.test.page.d.ts +0 -31
  652. package/src/app/shared/material/chips/chips.module.d.ts +0 -22
  653. package/src/app/shared/material/chips/material.chips.d.ts +0 -167
  654. package/src/app/shared/material/chips/testing/chips.test.d.ts +0 -32
  655. package/src/app/shared/material/datetime/datetime.module.d.ts +0 -25
  656. package/src/app/shared/material/datetime/material.date.d.ts +0 -90
  657. package/src/app/shared/material/datetime/material.dateshort.d.ts +0 -89
  658. package/src/app/shared/material/datetime/material.datetime.d.ts +0 -136
  659. package/src/app/shared/material/datetime/testing/mat-date-time.test.d.ts +0 -38
  660. package/src/app/shared/material/datetime/testing/mat-date.test.d.ts +0 -32
  661. package/src/app/shared/material/datetime/testing/mat-dateshort.test.d.ts +0 -31
  662. package/src/app/shared/material/duration/duration.module.d.ts +0 -18
  663. package/src/app/shared/material/duration/duration.utils.d.ts +0 -2
  664. package/src/app/shared/material/duration/material.duration.d.ts +0 -64
  665. package/src/app/shared/material/duration/testing/mat-duration.test.d.ts +0 -23
  666. package/src/app/shared/material/latlong/latlong.utils.d.ts +0 -59
  667. package/src/app/shared/material/latlong/material.latlong-input.d.ts +0 -93
  668. package/src/app/shared/material/latlong/material.latlong.d.ts +0 -90
  669. package/src/app/shared/material/latlong/material.latlong.module.d.ts +0 -21
  670. package/src/app/shared/material/latlong/testing/latlong.test.d.ts +0 -30
  671. package/src/app/shared/material/material.animations.d.ts +0 -8
  672. package/src/app/shared/material/material.config.d.ts +0 -3
  673. package/src/app/shared/material/material.module.d.ts +0 -44
  674. package/src/app/shared/material/material.testing.module.d.ts +0 -31
  675. package/src/app/shared/material/paginator/material.paginator-i18n.d.ts +0 -12
  676. package/src/app/shared/material/stepper/material.stepper-i18n.d.ts +0 -10
  677. package/src/app/shared/material/swipe/material.swipe.d.ts +0 -89
  678. package/src/app/shared/material/swipe/swipe.module.d.ts +0 -16
  679. package/src/app/shared/material/swipe/testing/swipe.test.d.ts +0 -22
  680. package/src/app/shared/material/test/test-component.d.ts +0 -75
  681. package/src/app/shared/material/testing/common.test.d.ts +0 -25
  682. package/src/app/shared/modules.d.ts +0 -9
  683. package/src/app/shared/named-filter/named-filter-selector.component.d.ts +0 -63
  684. package/src/app/shared/named-filter/named-filter.model.d.ts +0 -32
  685. package/src/app/shared/named-filter/named-filter.module.d.ts +0 -13
  686. package/src/app/shared/named-filter/named-filter.service.d.ts +0 -32
  687. package/src/app/shared/named-filter/testing/named-filter-selector.testing.d.ts +0 -34
  688. package/src/app/shared/named-filter/testing/named-filter.testing.module.d.ts +0 -10
  689. package/src/app/shared/observables.d.ts +0 -32
  690. package/src/app/shared/pipes/arrays.pipe.d.ts +0 -93
  691. package/src/app/shared/pipes/badge.pipes.d.ts +0 -7
  692. package/src/app/shared/pipes/colors.pipe.d.ts +0 -9
  693. package/src/app/shared/pipes/date-diff-duration.pipe.d.ts +0 -25
  694. package/src/app/shared/pipes/date-format.pipe.d.ts +0 -35
  695. package/src/app/shared/pipes/date-from-now.pipe.d.ts +0 -14
  696. package/src/app/shared/pipes/date-from.pipe.d.ts +0 -14
  697. package/src/app/shared/pipes/dates.pipe.d.ts +0 -10
  698. package/src/app/shared/pipes/display-with.pipe.d.ts +0 -8
  699. package/src/app/shared/pipes/duration.pipe.d.ts +0 -23
  700. package/src/app/shared/pipes/file-size.pipe.d.ts +0 -30
  701. package/src/app/shared/pipes/form.pipes.d.ts +0 -69
  702. package/src/app/shared/pipes/highlight.pipe.d.ts +0 -11
  703. package/src/app/shared/pipes/html.pipes.d.ts +0 -50
  704. package/src/app/shared/pipes/latlong-format.pipe.d.ts +0 -20
  705. package/src/app/shared/pipes/maps.pipe.d.ts +0 -23
  706. package/src/app/shared/pipes/maskito.pipe.d.ts +0 -14
  707. package/src/app/shared/pipes/math.pipes.d.ts +0 -26
  708. package/src/app/shared/pipes/ng-init.pipe.d.ts +0 -8
  709. package/src/app/shared/pipes/number-format.pipe.d.ts +0 -9
  710. package/src/app/shared/pipes/observable.pipes.d.ts +0 -25
  711. package/src/app/shared/pipes/pipes.module.d.ts +0 -36
  712. package/src/app/shared/pipes/property.pipes.d.ts +0 -46
  713. package/src/app/shared/pipes/selection.pipes.d.ts +0 -62
  714. package/src/app/shared/pipes/string.pipes.d.ts +0 -67
  715. package/src/app/shared/pipes/translate-context.pipe.d.ts +0 -17
  716. package/src/app/shared/pipes/types.pipes.d.ts +0 -30
  717. package/src/app/shared/pipes/url.pipes.d.ts +0 -27
  718. package/src/app/shared/platforms.d.ts +0 -16
  719. package/src/app/shared/print/print.service.d.ts +0 -94
  720. package/src/app/shared/regexps.d.ts +0 -32
  721. package/src/app/shared/rx-state/rx-state.decorators.d.ts +0 -33
  722. package/src/app/shared/rx-state/rx-state.module.d.ts +0 -10
  723. package/src/app/shared/rx-state/rx-state.types.d.ts +0 -5
  724. package/src/app/shared/services/entity-service.class.d.ts +0 -117
  725. package/src/app/shared/services/job.utils.d.ts +0 -20
  726. package/src/app/shared/services/memory-entity-service.class.d.ts +0 -87
  727. package/src/app/shared/services/progress-bar.service.d.ts +0 -20
  728. package/src/app/shared/services/startable-observable-service.class.d.ts +0 -51
  729. package/src/app/shared/services/startable-service.class.d.ts +0 -38
  730. package/src/app/shared/services/translate-context.service.d.ts +0 -29
  731. package/src/app/shared/services/validator-service.class.d.ts +0 -11
  732. package/src/app/shared/services.d.ts +0 -17
  733. package/src/app/shared/shared-routing.module.d.ts +0 -23
  734. package/src/app/shared/shared.module.d.ts +0 -33
  735. package/src/app/shared/shared.testing.module.d.ts +0 -20
  736. package/src/app/shared/storage/storage-explorer.component.d.ts +0 -60
  737. package/src/app/shared/storage/storage-explorer.module.d.ts +0 -13
  738. package/src/app/shared/storage/storage-explorer.testing-routing.module.d.ts +0 -9
  739. package/src/app/shared/storage/storage-explorer.testing.module.d.ts +0 -10
  740. package/src/app/shared/storage/storage.service.d.ts +0 -22
  741. package/src/app/shared/storage/storage.utils.d.ts +0 -19
  742. package/src/app/shared/testing/maskito.test.d.ts +0 -10
  743. package/src/app/shared/testing/observable.test.d.ts +0 -25
  744. package/src/app/shared/testing/tests.page.d.ts +0 -20
  745. package/src/app/shared/toast/toast.testing.d.ts +0 -15
  746. package/src/app/shared/toast/toast.testing.module.d.ts +0 -12
  747. package/src/app/shared/toast/toasts.d.ts +0 -16
  748. package/src/app/shared/toolbar/modal-toolbar.d.ts +0 -21
  749. package/src/app/shared/toolbar/toolbar.d.ts +0 -70
  750. package/src/app/shared/toolbar/toolbar.module.d.ts +0 -16
  751. package/src/app/shared/types.d.ts +0 -45
  752. package/src/app/shared/upload-file/testing/upload-file.testing.d.ts +0 -21
  753. package/src/app/shared/upload-file/testing/upload-file.testing.module.d.ts +0 -11
  754. package/src/app/shared/upload-file/upload-file-popover.component.d.ts +0 -41
  755. package/src/app/shared/upload-file/upload-file.component.d.ts +0 -50
  756. package/src/app/shared/upload-file/upload-file.model.d.ts +0 -30
  757. package/src/app/shared/validator/form-error-adapter.class.d.ts +0 -33
  758. package/src/app/shared/validator/validators.d.ts +0 -145
  759. package/src/app/shared/version/versions.d.ts +0 -26
  760. package/src/app/social/config/social.config.d.ts +0 -4
  761. package/src/app/social/feed/feed.component.d.ts +0 -91
  762. package/src/app/social/feed/feed.directive.d.ts +0 -32
  763. package/src/app/social/feed/feed.model.d.ts +0 -69
  764. package/src/app/social/feed/feed.module.d.ts +0 -13
  765. package/src/app/social/feed/feed.page.d.ts +0 -24
  766. package/src/app/social/feed/feed.service.d.ts +0 -85
  767. package/src/app/social/feed/testing/feed.testing.d.ts +0 -23
  768. package/src/app/social/feed/testing/feed.testing.module.d.ts +0 -12
  769. package/src/app/social/job/job.module.d.ts +0 -12
  770. package/src/app/social/job/progression/job-progression.component.d.ts +0 -24
  771. package/src/app/social/job/progression/job-progression.icon.d.ts +0 -63
  772. package/src/app/social/job/progression/job-progression.list.d.ts +0 -20
  773. package/src/app/social/job/progression/job-progression.model.d.ts +0 -10
  774. package/src/app/social/job/progression/job-progression.service.d.ts +0 -33
  775. package/src/app/social/job/testing/job-progression.testing.d.ts +0 -13
  776. package/src/app/social/job/testing/job-progression.testing.service.d.ts +0 -9
  777. package/src/app/social/job/testing/job.testing.module.d.ts +0 -11
  778. package/src/app/social/message/message.form.d.ts +0 -36
  779. package/src/app/social/message/message.modal.d.ts +0 -47
  780. package/src/app/social/message/message.model.d.ts +0 -36
  781. package/src/app/social/message/message.module.d.ts +0 -14
  782. package/src/app/social/message/message.service.d.ts +0 -44
  783. package/src/app/social/social.errors.d.ts +0 -10
  784. package/src/app/social/social.module.d.ts +0 -20
  785. package/src/app/social/social.testing.module.d.ts +0 -14
  786. package/src/app/social/user-event/notification/user-event-notification.icon.d.ts +0 -47
  787. package/src/app/social/user-event/notification/user-event-notification.list.d.ts +0 -77
  788. package/src/app/social/user-event/notification/user-event-notification.modal.d.ts +0 -37
  789. package/src/app/social/user-event/testing/user-event.testing.d.ts +0 -44
  790. package/src/app/social/user-event/testing/user-event.testing.model.d.ts +0 -42
  791. package/src/app/social/user-event/testing/user-event.testing.module.d.ts +0 -11
  792. package/src/app/social/user-event/testing/user-event.testing.service.d.ts +0 -41
  793. package/src/app/social/user-event/user-event.model.d.ts +0 -44
  794. package/src/app/social/user-event/user-event.module.d.ts +0 -14
  795. package/src/app/social/user-event/user-event.service.d.ts +0 -124
  796. package/src/environments/environment.class.d.ts +0 -86
  797. package/src/environments/environment.d.ts +0 -2
  798. package/src/environments/environment.loader.d.ts +0 -24
  799. package/src/theme/_globals.scss +0 -5
@@ -1,1403 +0,0 @@
1
- import { Inject, Injectable, InjectionToken, Optional } from '@angular/core';
2
- import { CryptoService } from './crypto.service';
3
- import { Account, AccountUtils, UserSettings } from './model/account.model';
4
- import { PersonUtils } from './model/person.model';
5
- import { BehaviorSubject, from, Subject, Subscription } from 'rxjs';
6
- import { gql } from '@apollo/client/core';
7
- import { isEmptyArray, isNil, removeDuplicatesFromArray, sleep, toNumber } from '../../shared/functions';
8
- import { BaseGraphqlService } from './base-graphql-service.class';
9
- import { ErrorCodes, ServerErrorCodes } from './errors';
10
- import { MINIFY_ENTITY_FOR_POD, ReferentialUtils } from './model/referential.model';
11
- import { StatusIds } from './model/model.enum';
12
- import { Base58 } from './base58';
13
- import { ENVIRONMENT } from '../../../environments/environment.class';
14
- import { fromDateISOString } from '../../shared/dates';
15
- import { firstNotNilPromise } from '../../shared/observables';
16
- import { debounceTime, filter, map, startWith, switchMap } from 'rxjs/operators';
17
- import { Toasts } from '../../shared/toast/toasts';
18
- import { UserToken } from './model/token.model';
19
- import { AppPropertiesUtils } from '../form/properties/properties.utils';
20
- import * as i0 from "@angular/core";
21
- import * as i1 from "./network.service";
22
- import * as i2 from "../graphql/graphql.service";
23
- import * as i3 from "./local-settings.service";
24
- import * as i4 from "@ionic/storage-angular";
25
- import * as i5 from "../../shared/file/file.service";
26
- import * as i6 from "@ngx-translate/core";
27
- import * as i7 from "@ionic/angular";
28
- import * as i8 from "../../../environments/environment.class";
29
- const TOKEN_STORAGE_KEY = 'token';
30
- const PUBKEY_STORAGE_KEY = 'pubkey';
31
- const SECKEY_STORAGE_KEY = 'seckey';
32
- const ACCOUNT_STORAGE_KEY = 'account';
33
- const DEFAULT_AVATAR_IMAGE = PersonUtils.DEFAULT_AVATAR_IMAGE;
34
- export const APP_USER_SETTINGS_OPTIONS = new InjectionToken('UserSettingsOptions');
35
- export const APP_USER_TOKEN_SCOPES = new InjectionToken('UserTokenScopes');
36
- // Allow apps to inject/override the account service implementation
37
- export const APP_ACCOUNT_SERVICE = new InjectionToken('AccountService');
38
- export const APP_ACCOUNT_SERVICE_OPTIONS = new InjectionToken('AccountServiceOptions');
39
- /* ------------------------------------
40
- * GraphQL queries
41
- * ------------------------------------*/
42
- const Fragments = {
43
- account: gql `
44
- fragment AccountFragment on AccountVO {
45
- id
46
- firstName
47
- lastName
48
- email
49
- pubkey
50
- avatar
51
- statusId
52
- updateDate
53
- creationDate
54
- profiles
55
- settings {
56
- ...UserSettingsFragment
57
- }
58
- department {
59
- id
60
- label
61
- name
62
- __typename
63
- }
64
- __typename
65
- }
66
- `,
67
- settings: gql `
68
- fragment UserSettingsFragment on UserSettingsVO {
69
- id
70
- locale
71
- latLongFormat
72
- content
73
- nonce
74
- updateDate
75
- __typename
76
- }
77
- `,
78
- token: gql `
79
- fragment UserTokenFragment on UserTokenVO {
80
- id
81
- pubkey
82
- name
83
- flags
84
- expirationDate
85
- lastUsedDate
86
- creationDate
87
- updateDate
88
- __typename
89
- }
90
- `,
91
- };
92
- // Account queries
93
- const Queries = {
94
- load: gql `
95
- query Account {
96
- data: account {
97
- ...AccountFragment
98
- }
99
- }
100
- ${Fragments.account}
101
- ${Fragments.settings}
102
- `,
103
- loadWithTokens: gql `
104
- query AccountWithToken {
105
- data: account {
106
- ...AccountFragment
107
- tokens {
108
- ...UserTokenFragment
109
- }
110
- }
111
- }
112
- ${Fragments.account}
113
- ${Fragments.settings}
114
- ${Fragments.token}
115
- `,
116
- };
117
- // Check email query
118
- const IsEmailExistsQuery = gql `
119
- query IsEmailExists($email: String, $hash: String) {
120
- isEmailExists(email: $email, hash: $hash)
121
- }
122
- `;
123
- // Account mutations
124
- const Mutations = {
125
- save: gql `
126
- mutation SaveAccount($data: AccountVOInput) {
127
- data: saveAccount(account: $data) {
128
- ...AccountFragment
129
- }
130
- }
131
- ${Fragments.account}
132
- ${Fragments.settings}
133
- `,
134
- saveWithTokens: gql `
135
- mutation SaveAccountWithTokens($data: AccountVOInput) {
136
- data: saveAccount(account: $data) {
137
- ...AccountFragment
138
- tokens {
139
- ...UserTokenFragment
140
- }
141
- }
142
- }
143
- ${Fragments.account}
144
- ${Fragments.settings}
145
- ${Fragments.token}
146
- `,
147
- create: gql `
148
- mutation CreateAccount($data: AccountVOInput) {
149
- data: createAccount(account: $data) {
150
- ...AccountFragment
151
- }
152
- }
153
- ${Fragments.account}
154
- ${Fragments.settings}
155
- `,
156
- saveSettings: gql `
157
- mutation SaveSettings($data: UserSettingsVOInput) {
158
- data: saveSettings(settings: $data) {
159
- ...UserSettingsFragment
160
- }
161
- }
162
- ${Fragments.settings}
163
- `,
164
- };
165
- // Sent confirmation email
166
- const SendConfirmEmailQuery = gql `
167
- query sendAccountConfirmationEmail($email: String, $locale: String) {
168
- data: sendAccountConfirmationEmail(email: $email, locale: $locale)
169
- }
170
- `;
171
- // Confirm account email
172
- const ConfirmEmailQuery = gql `
173
- query confirmAccountEmail($email: String, $code: String) {
174
- data: confirmAccountEmail(email: $email, code: $code)
175
- }
176
- `;
177
- // Send email to reset password
178
- const SendResetPasswordEmailQuery = gql `
179
- query SendResetPasswordEmail($username: String!, $locale: String!) {
180
- sendResetPasswordEmail(username: $username, locale: $locale)
181
- }
182
- `;
183
- const ResetAccountPubkeyQuery = gql `
184
- query ResetPubkey($username: String!, $token: String!) {
185
- data: resetAccountPubkey(username: $username, token: $token)
186
- }
187
- `;
188
- const UpdatePubKeyQuery = gql `
189
- query UpdateAccountPubkey($pubkey: String!) {
190
- data: updateAccountPubkey(pubkey: $pubkey)
191
- }
192
- `;
193
- // Authentication query
194
- const AuthQuery = gql `
195
- query Auth($token: String) {
196
- authenticate(token: $token)
197
- }
198
- `;
199
- // New auth challenge query
200
- const AuthChallengeQuery = gql `
201
- query AuthChallenge {
202
- authChallenge {
203
- challenge
204
- pubkey
205
- signature
206
- }
207
- }
208
- `;
209
- const AccountSubscriptions = {
210
- listenChanges: gql `
211
- subscription updateAccount($interval: Int) {
212
- data: updateAccount(interval: $interval) {
213
- id
214
- updateDate
215
- }
216
- }
217
- `,
218
- };
219
- export class AccountService extends BaseGraphqlService {
220
- network;
221
- graphql;
222
- settings;
223
- storage;
224
- file;
225
- translate;
226
- toastController;
227
- environment;
228
- // GraphQL operations (can be overridden using APP_ACCOUNT_SERVICE_OPTIONS)
229
- queries;
230
- mutations;
231
- subscriptions;
232
- onLogin = new Subject();
233
- onWillLogout = new Subject();
234
- onLogout = new Subject();
235
- onChange = new Subject();
236
- onAuthTokenChange = new BehaviorSubject(undefined);
237
- onAuthBasicChange = new BehaviorSubject(undefined);
238
- _listenChangesSubscription = null;
239
- _enableListenChanges;
240
- _listenIntervalInSeconds;
241
- _cache = {
242
- loaded: false,
243
- keypair: null,
244
- authToken: null,
245
- authBasic: null,
246
- pubkey: null,
247
- mainProfile: null,
248
- person: null,
249
- department: null,
250
- };
251
- _optionDefs;
252
- _remoteLocalSettingsKeys;
253
- _$additionalFields = new BehaviorSubject([]);
254
- _tokenType$ = new BehaviorSubject(undefined);
255
- _apiTokenEnabled = true;
256
- get account() {
257
- return this._cache.loaded ? this._data : undefined;
258
- }
259
- get account$() {
260
- return this.onChange.asObservable().pipe(startWith(this.account));
261
- }
262
- get person() {
263
- if (this._cache.loaded && !this._cache.person) {
264
- this._cache.person = this._cache.loaded ? this._data.asPerson() : undefined;
265
- }
266
- return this._cache.person;
267
- }
268
- get person$() {
269
- return this.onChange.asObservable().pipe(map(() => this.person), startWith(this.person));
270
- }
271
- get department() {
272
- if (this._cache.loaded && !this._cache.department) {
273
- this._cache.department = this._cache.loaded ? this._data.asPerson().department : undefined;
274
- }
275
- return this._cache.department;
276
- }
277
- get tokenType() {
278
- return this._tokenType$.value;
279
- }
280
- set tokenType(value) {
281
- if (this._tokenType$.value !== value) {
282
- console.info('[account] Using authentication token type: ' + value);
283
- this._tokenType$.next(value);
284
- // Reset values
285
- this._cache.authToken = undefined;
286
- this.onAuthTokenChange.next(undefined);
287
- this._cache.authBasic = undefined;
288
- this.onAuthBasicChange.next(undefined);
289
- }
290
- }
291
- get apiTokenEnabled() {
292
- return this._apiTokenEnabled;
293
- }
294
- set apiTokenEnabled(value) {
295
- this._apiTokenEnabled = value;
296
- }
297
- get remoteLocalSettingsKeys() {
298
- return this._remoteLocalSettingsKeys;
299
- }
300
- constructor(network, graphql, settings, storage, file, translate, toastController, environment, options, accountOptions) {
301
- super(graphql, environment);
302
- this.network = network;
303
- this.graphql = graphql;
304
- this.settings = settings;
305
- this.storage = storage;
306
- this.file = file;
307
- this.translate = translate;
308
- this.toastController = toastController;
309
- this.environment = environment;
310
- // Initialize GraphQL operations (allow overrides)
311
- this.queries = { ...Queries, ...(accountOptions?.queries || {}) };
312
- this.mutations = { ...Mutations, ...(accountOptions?.mutations || {}) };
313
- this.subscriptions = { ...AccountSubscriptions, ...(accountOptions?.subscriptions || {}) };
314
- this._enableListenChanges = !environment.account || environment.account.enableListenChanges !== false; // True by default
315
- this._listenIntervalInSeconds = toNumber(environment.account && environment.account.listenIntervalInSeconds, 0); // no timer by default
316
- this._debug = !environment.production;
317
- if (this._debug)
318
- console.debug('[account-service] Creating service');
319
- this._optionDefs = Object.values(options?.options || {});
320
- this._remoteLocalSettingsKeys = options?.remoteLocalSettingsKeys || [];
321
- this.resetData();
322
- // Send auth token to the graphql layer, when changed
323
- this.onAuthTokenChange.subscribe((token) => this.network.setAuthToken(token));
324
- this.onAuthBasicChange.subscribe((basic) => this.network.setAuthBasic(basic));
325
- // Force network to wait account service, after getting connection to the peer
326
- this.network.on('beforeTryOnlineFinish', async (online) => {
327
- // If online, wait a full restart, because it can force offline mode
328
- if (online && (!this.started || this.isLogin())) {
329
- console.debug('[account] Force networkService.tryOnline() to wait, that graphql and account service is restarted...');
330
- await sleep(500); // wait graphql service to be restarted
331
- if (!this.started)
332
- await this.ready();
333
- }
334
- });
335
- }
336
- async ngOnStart() {
337
- await Promise.all([
338
- this.settings.ready(),
339
- // Wait token type to be set
340
- firstNotNilPromise(this._tokenType$, { stop: this.stopSubject }),
341
- ]);
342
- // Listen graphql start (or restart)
343
- this.registerSubscription(this.graphql.stopSubject.subscribe(() => {
344
- if (this.started && this.isLogin()) {
345
- console.debug('[account] Restarting, to retry to authenticate...');
346
- this.restart();
347
- }
348
- }));
349
- await this.restoreLocally();
350
- await this.listenSettings();
351
- return this._data;
352
- }
353
- async ngOnStop() {
354
- const hadAuthToken = this._cache.authToken && true;
355
- const hadAuthBasic = this._cache.authBasic && true;
356
- const hadAuth = hadAuthToken || hadAuthBasic;
357
- this.resetData();
358
- if (hadAuth) {
359
- this.onLogout.next();
360
- this.onChange.next(undefined);
361
- if (hadAuthToken)
362
- this.onAuthTokenChange.next(undefined);
363
- if (hadAuthBasic)
364
- this.onAuthBasicChange.next(undefined);
365
- }
366
- }
367
- isLogin() {
368
- return !!(this._cache.pubkey && this._cache.loaded);
369
- }
370
- isAuth() {
371
- return !!(this._cache.pubkey && this._cache.keypair && this._cache.keypair.secretKey);
372
- }
373
- hasMinProfile(userProfile) {
374
- // should be login, and status ENABLE or TEMPORARY
375
- if (!this._data || !this._data.pubkey || (this._data.statusId !== StatusIds.ENABLE && this._data.statusId !== StatusIds.TEMPORARY)) {
376
- return false;
377
- }
378
- return PersonUtils.hasUpperOrEqualsProfile(this._data.profiles, userProfile);
379
- }
380
- hasExactProfile(label) {
381
- return AccountUtils.hasExactProfile(this._data, label);
382
- }
383
- hasProfileAndIsEnable(userProfile) {
384
- return AccountUtils.hasProfileAndIsEnable(this._data, userProfile);
385
- }
386
- isAdmin() {
387
- return this.hasProfileAndIsEnable('ADMIN');
388
- }
389
- isSupervisor() {
390
- return this.hasProfileAndIsEnable('SUPERVISOR');
391
- }
392
- isUser() {
393
- return this.hasProfileAndIsEnable('USER');
394
- }
395
- /**
396
- * @deprecated
397
- * @param mode
398
- */
399
- isUsageMode(mode) {
400
- return this.settings.isUsageMode(mode);
401
- }
402
- isOnlyGuest() {
403
- // Should be login, and status ENABLE or TEMPORARY
404
- if (!this._data || !this._data.pubkey || (this._data.statusId !== StatusIds.ENABLE && this._data.statusId !== StatusIds.TEMPORARY))
405
- return false;
406
- // Profile less then user
407
- return !PersonUtils.hasUpperOrEqualsProfile(this._data.profiles, 'USER');
408
- }
409
- /**
410
- *
411
- * @param recorderDepartment
412
- * @deprecated use ProgramRefService.canUserWriteEntity() instead
413
- */
414
- canUserWriteDataForDepartment(recorderDepartment) {
415
- if (ReferentialUtils.isEmpty(recorderDepartment)) {
416
- return this.isAdmin();
417
- }
418
- // Should be login, and status ENABLE
419
- if (!this._data || !this._data.pubkey || this._data.statusId !== StatusIds.ENABLE)
420
- return false;
421
- if (!this._data.department || !this._data.department.id) {
422
- console.warn('User account has no department ! Unable to check write right against recorderDepartment');
423
- return false;
424
- }
425
- // Same recorder department: OK, user can write
426
- if (this._data.department.id === recorderDepartment.id)
427
- return true;
428
- // Cannot write
429
- return false;
430
- }
431
- async register(data) {
432
- if (this.isLogin()) {
433
- throw new Error('User already login. Please logout before register.');
434
- }
435
- if (!data.username || !data.password)
436
- throw new Error('Missing required username or password');
437
- if (this._debug)
438
- console.debug('[account] Register new user account...', data.account);
439
- this._cache.loaded = false;
440
- const now = Date.now();
441
- try {
442
- const keypair = await CryptoService.scryptKeypair(data.username, data.password);
443
- data.account.pubkey = Base58.encode(keypair.publicKey);
444
- // Default values
445
- data.account.settings = data.account.settings || new UserSettings();
446
- data.account.settings.merge(this.settings.settings, this._remoteLocalSettingsKeys, true);
447
- data.account.department.id = data.account.department.id || this.environment.defaultDepartmentId;
448
- this._cache.keypair = keypair;
449
- const account = await this.saveRemotely(data.account, keypair);
450
- // Default values
451
- account.avatar = account.avatar || this.environment.baseUrl + PersonUtils.DEFAULT_AVATAR_IMAGE;
452
- this._cache.mainProfile = PersonUtils.getMainProfile(account.profiles);
453
- this._data = account;
454
- this._cache.pubkey = account.pubkey;
455
- // Try to auth on pod
456
- await this.authenticate(data);
457
- this._cache.loaded = true;
458
- await this.saveLocally();
459
- console.debug(`[account] Account successfully registered in ${Date.now() - now}ms`);
460
- // Emit events
461
- this.onLogin.next(this._data);
462
- this.onChange.next(this._data);
463
- // Listen remote changes
464
- if (this._enableListenChanges)
465
- this.startListenRemoteChanges();
466
- return this._data;
467
- }
468
- catch (error) {
469
- console.error((error && error.message) || error);
470
- this.resetData();
471
- throw error;
472
- }
473
- }
474
- async authenticate(data) {
475
- // Wait the auth token, then continue
476
- const tokenType = await firstNotNilPromise(this._tokenType$);
477
- // Basic auth
478
- if (tokenType === 'basic' || tokenType === 'basic-and-token') {
479
- // Generate the authBasic, if used
480
- if (!this._cache.authBasic) {
481
- // Skip if token already provided
482
- if (!(this._cache.authToken && tokenType === 'basic-and-token')) {
483
- if (!data || !data.username || !data.password)
484
- throw new Error('Missing username and password');
485
- this._cache.authBasic = CryptoService.encodeBase64(`${data.username}:${data.password}`);
486
- }
487
- }
488
- this.onAuthBasicChange.next(this._cache.authBasic);
489
- }
490
- // Generate the authToken, if used
491
- if (tokenType === 'token' || tokenType === 'basic-and-token') {
492
- try {
493
- this._cache.authToken = await this.authenticateAndGetToken(this._cache.authToken);
494
- }
495
- catch (error) {
496
- // Never authenticate, or not ready for offline mode => exit
497
- console.error(error);
498
- this.resetData();
499
- throw error;
500
- }
501
- }
502
- // Forget authBasic, to switch to authToken
503
- if (tokenType === 'basic-and-token') {
504
- this._cache.authBasic = undefined;
505
- this.onAuthBasicChange.next(undefined);
506
- }
507
- }
508
- async login(data) {
509
- if (!data || !data.username || !data.password)
510
- throw new Error('Missing required username or password');
511
- console.debug('[account] Login...');
512
- let keypair;
513
- try {
514
- keypair = await CryptoService.scryptKeypair(data.username, data.password);
515
- }
516
- catch (error) {
517
- console.error(error);
518
- this.resetData();
519
- throw { code: ErrorCodes.UNKNOWN_ERROR, message: 'ERROR.SCRYPT_ERROR' };
520
- }
521
- // Store pubkey+keypair
522
- this._cache.pubkey = Base58.encode(keypair.publicKey);
523
- this._cache.keypair = keypair;
524
- // Try to load previous token
525
- let previousToken = await this.storage.get(TOKEN_STORAGE_KEY);
526
- previousToken = (previousToken && previousToken.startsWith(this._cache.pubkey) && previousToken) || null;
527
- // Offline mode
528
- const offline = this.settings.hasOfflineFeature() && (this.network.offline || data.offline === true);
529
- if (offline) {
530
- this._cache.authToken = previousToken;
531
- // Make sure network if set as offline
532
- this.network.setForceOffline(true, { showToast: false });
533
- console.info(`[account] Login [OK] {pubkey: ${this._cache.pubkey.substr(0, 8)}}, {offline: true}`);
534
- }
535
- // Online mode: try to auth on pod
536
- else {
537
- try {
538
- await this.authenticate(data);
539
- }
540
- catch (error) {
541
- // Never authenticate, or not ready for offline mode => exit
542
- console.error(error);
543
- this.resetData();
544
- throw error;
545
- }
546
- }
547
- // Load account data
548
- try {
549
- await this.loadData({ offline, fetchPolicy: 'network-only' });
550
- }
551
- catch (err) {
552
- // If account not found, check if email is valid
553
- if (err && +err.code === ErrorCodes.LOAD_ACCOUNT_ERROR) {
554
- // Check email exists
555
- if (data.username.indexOf('@') !== -1) {
556
- let isEmailExists;
557
- try {
558
- isEmailExists = await this.isEmailExists(data.username);
559
- }
560
- catch (otherError) {
561
- throw err; // resend the first error
562
- }
563
- // Email not exists (no account)
564
- if (!isEmailExists) {
565
- throw { code: ErrorCodes.UNKNOWN_ACCOUNT_EMAIL, message: 'ERROR.UNKNOWN_ACCOUNT_EMAIL' };
566
- }
567
- }
568
- // Email exists, so error = 'bad password'
569
- throw { code: ErrorCodes.BAD_PASSWORD, message: 'ERROR.BAD_PASSWORD' };
570
- }
571
- throw err; // resend the first error
572
- }
573
- try {
574
- // Store to local storage
575
- await this.saveLocally();
576
- }
577
- catch (error) {
578
- console.error(error);
579
- this.resetData();
580
- throw error;
581
- }
582
- // Emit event to observers
583
- this.onLogin.next(this._data);
584
- this.onChange.next(this._data);
585
- if (this._enableListenChanges)
586
- this.startListenRemoteChanges();
587
- return this._data;
588
- }
589
- async reload(opts) {
590
- if (!this._cache.pubkey)
591
- throw new Error('User not logged');
592
- if (this.network.offline)
593
- throw new Error('Cannot check account in offline mode');
594
- const now = Date.now();
595
- console.debug(`[account] Reloading account...`);
596
- const wasLogin = this.isLogin();
597
- try {
598
- await this.loadData();
599
- await this.saveLocally();
600
- console.debug(`[account] Reloading account [OK] in ${Date.now() - now}ms`);
601
- // Emit login event to subscribers
602
- this.onLogin.next(this._data);
603
- this.onChange.next(this._data);
604
- // Display toast (without await, because not need to wait toast close event)
605
- if (!opts || opts.showToast !== false) {
606
- this.showToast({ message: 'ACCOUNT.INFO.RELOADED', type: 'info' });
607
- }
608
- return this._data;
609
- }
610
- catch (error) {
611
- // Cannot reload but was login: force to logout
612
- if (wasLogin && !this.isLogin()) {
613
- console.error(`[account] Reloading account failed. Will force logout...`, error);
614
- await this.logout();
615
- }
616
- else {
617
- throw error;
618
- }
619
- }
620
- }
621
- /**
622
- * Create or update an user account, to the remote storage
623
- *
624
- * @param account
625
- */
626
- async save(account) {
627
- if (!this._cache.pubkey)
628
- return Promise.reject('User not logged');
629
- if (this._cache.pubkey !== account.pubkey)
630
- return Promise.reject('Not user account');
631
- account = await this.saveRemotely(account, this._cache.keypair);
632
- // Set defaults
633
- account.avatar = account.avatar || this.environment.baseUrl + DEFAULT_AVATAR_IMAGE;
634
- this._cache.mainProfile = PersonUtils.getMainProfile(account.profiles);
635
- this._data = account;
636
- // Update cache
637
- this._cache.person = account.asPerson();
638
- this._cache.department = this._cache.person.department;
639
- this._cache.loaded = true;
640
- // Save locally (in storage)
641
- await this.saveLocally();
642
- // Send event
643
- this.onLogin.next(this._data);
644
- this.onChange.next(this._data);
645
- return this._data;
646
- }
647
- async logout() {
648
- const hadAuthToken = this._cache.authToken && true;
649
- const hadAuthBasic = this._cache.authBasic && true;
650
- const pubkey = this._cache && this._cache.pubkey;
651
- // Notify observers
652
- this.onWillLogout.next(this.account);
653
- this.resetData();
654
- if (!this.settings.hasOfflineFeature()) {
655
- // Remove all data from the local storage
656
- await Promise.all([
657
- this.storage.remove(PUBKEY_STORAGE_KEY),
658
- this.storage.remove(TOKEN_STORAGE_KEY),
659
- this.storage.remove(ACCOUNT_STORAGE_KEY),
660
- (pubkey && this.storage.remove(ACCOUNT_STORAGE_KEY + '#' + pubkey)) || Promise.resolve(),
661
- this.storage.remove(SECKEY_STORAGE_KEY),
662
- ]);
663
- }
664
- // Offline features enable: need to keep some data
665
- else {
666
- // Always remove only secret key
667
- // But keep:
668
- // - account by pubkey
669
- // - auth token
670
- await Promise.all([this.storage.remove(PUBKEY_STORAGE_KEY), this.storage.remove(ACCOUNT_STORAGE_KEY), this.storage.remove(SECKEY_STORAGE_KEY)]);
671
- }
672
- // Clean page history, in local settings
673
- await this.settings.clearPageHistory();
674
- // Notify observers
675
- this.onLogout.next();
676
- if (hadAuthToken)
677
- this.onAuthTokenChange.next(undefined);
678
- if (hadAuthBasic)
679
- this.onAuthBasicChange.next(undefined);
680
- this.onChange.next(undefined);
681
- }
682
- /**
683
- * Load a account
684
- *
685
- * @param opts
686
- */
687
- async load(opts) {
688
- const now = this._debug && Date.now();
689
- if (this._debug)
690
- console.debug(`[account] Loading account...`);
691
- let json;
692
- // Load locally
693
- const offline = (this.network.offline && (!opts || (opts.fetchPolicy !== 'network-only' && opts.fetchPolicy !== 'no-cache'))) ||
694
- (opts && opts.offline === true);
695
- if (offline) {
696
- json = await this.storage.get(ACCOUNT_STORAGE_KEY);
697
- json = (json && typeof json === 'string' && JSON.parse(json)) || json;
698
- json = (json && this._cache.pubkey && json.pubkey === this._cache.pubkey && json) || null;
699
- if (!json && this._cache.pubkey) {
700
- json = await this.storage.get(ACCOUNT_STORAGE_KEY + '#' + this._cache.pubkey);
701
- json = (json && typeof json === 'string' && JSON.parse(json)) || json;
702
- }
703
- }
704
- // Load remotely
705
- else {
706
- const query = opts?.query || (this.apiTokenEnabled ? this.queries.loadWithTokens : this.queries.load);
707
- const { data } = await this.graphql.query({
708
- query,
709
- error: { code: ErrorCodes.LOAD_ACCOUNT_ERROR, message: 'ERROR.LOAD_ACCOUNT_ERROR' },
710
- fetchPolicy: (opts && opts.fetchPolicy) || 'no-cache' || undefined,
711
- });
712
- json = data;
713
- }
714
- if (json) {
715
- const account = Account.fromObject(json);
716
- // Clean remote
717
- if (account.settings && this._remoteLocalSettingsKeys) {
718
- account.settings.merge(account.settings.asLocalSettings(), this._remoteLocalSettingsKeys);
719
- }
720
- if (this._debug)
721
- console.debug(`[account] Account loaded in ${Date.now() - now}ms`, account);
722
- return account;
723
- }
724
- else {
725
- console.warn(`[account] Account not found !`);
726
- return undefined;
727
- }
728
- }
729
- async generateToken(flags) {
730
- const now = this._debug && Date.now();
731
- if (this._debug)
732
- console.debug(`[account] Generate token...`);
733
- const authChallenge = await this.getAuthChallenge();
734
- // Add Flags to challenge
735
- const challenge = `${authChallenge.challenge}:${flags}`;
736
- const signature = await CryptoService.sign(challenge, this._cache.keypair);
737
- const newToken = `${this._cache.pubkey}:${challenge}|${signature}`;
738
- if (newToken) {
739
- if (this._debug)
740
- console.debug(`[account] Token generated in ${Date.now() - now}ms`);
741
- }
742
- else {
743
- console.warn(`[account] Token generation failed !`);
744
- }
745
- return newToken;
746
- }
747
- /**
748
- * Check if email is available for new account registration.
749
- * Throw an error if not available
750
- *
751
- * @param email
752
- */
753
- async checkEmailAvailable(email) {
754
- const isEmailExists = await this.isEmailExists(email);
755
- if (isEmailExists) {
756
- throw { code: ErrorCodes.EMAIL_ALREADY_REGISTERED, message: 'ERROR.EMAIL_ALREADY_REGISTERED' };
757
- }
758
- }
759
- /**
760
- * Check if email is exists in server.
761
- *
762
- * @param email
763
- */
764
- async isEmailExists(email) {
765
- if (this._debug)
766
- console.debug('[account] Checking if {' + email + '} exists...');
767
- const data = await this.graphql.query({
768
- query: IsEmailExistsQuery,
769
- variables: {
770
- email,
771
- hash: undefined,
772
- },
773
- });
774
- if (this._debug)
775
- console.debug('[account] Email exist: ' + (data && data.isEmailExists));
776
- return data && data.isEmailExists;
777
- }
778
- async sendConfirmationEmail(email, locale) {
779
- locale = locale || this.settings.locale;
780
- console.debug('[account] Sending confirmation email to {' + email + '} with locale {' + locale + '}...');
781
- const { data } = await this.graphql.query({
782
- query: SendConfirmEmailQuery,
783
- variables: {
784
- email,
785
- locale,
786
- },
787
- error: {
788
- code: ErrorCodes.SENT_CONFIRMATION_EMAIL_FAILED,
789
- message: 'ERROR.SENT_ACCOUNT_CONFIRMATION_EMAIL_FAILED',
790
- },
791
- });
792
- return data;
793
- }
794
- async confirmEmail(email, code) {
795
- console.debug('[account] Sending confirm request for email {' + email + '} with code {' + code + '}...');
796
- const { data } = await this.graphql.query({
797
- query: ConfirmEmailQuery,
798
- variables: {
799
- email,
800
- code,
801
- },
802
- error: {
803
- code: ErrorCodes.CONFIRM_EMAIL_FAILED,
804
- message: 'ERROR.CONFIRM_ACCOUNT_EMAIL_FAILED',
805
- },
806
- });
807
- return data;
808
- }
809
- async sendResetPasswordEmail(username, locale) {
810
- locale = locale || this.settings.locale;
811
- console.debug(`[account] Sending change password email to {${username}} with locale {${locale}}...`);
812
- const { data: isOk } = await this.graphql.query({
813
- query: SendResetPasswordEmailQuery,
814
- variables: {
815
- username,
816
- locale,
817
- },
818
- error: {
819
- code: ErrorCodes.AUTH_SEND_RESET_PASSWORD_EMAIL_ERROR,
820
- message: 'ACCOUNT.ERROR.SEND_RESET_PASSWORD_EMAIL_ERROR',
821
- },
822
- fetchPolicy: 'no-cache',
823
- });
824
- return isOk;
825
- }
826
- async updatePubkey(data) {
827
- if (!data.username || !data.password)
828
- throw new Error('Missing required username or password');
829
- if (!this.isLogin())
830
- throw new Error('Should be login');
831
- const keypair = await CryptoService.scryptKeypair(data.username, data.password);
832
- const pubkey = Base58.encode(keypair.publicKey);
833
- console.debug(`[account] Updating account pubkey to ${pubkey} ...`);
834
- const { data: res } = await this.graphql.query({
835
- query: UpdatePubKeyQuery,
836
- variables: { pubkey },
837
- error: {
838
- code: ErrorCodes.AUTH_UPDATE_PUBKEY_ERROR,
839
- message: 'ACCOUNT.ERROR.UPDATE_PUBKEY_ERROR',
840
- },
841
- fetchPolicy: 'no-cache',
842
- });
843
- console.debug(`[account] Updating account pubkey response`, res);
844
- if (res) {
845
- await this.logout();
846
- await this.login(data);
847
- }
848
- return res;
849
- }
850
- async resetPubkey(data, token) {
851
- if (this.isLogin()) {
852
- await this.logout();
853
- }
854
- // Parse challenge
855
- const challenge = token.split('|', 2)[0].split(':', 2)[1];
856
- const keypair = await CryptoService.scryptKeypair(data.username, data.password);
857
- const pubkey = Base58.encode(keypair.publicKey);
858
- const signature = await CryptoService.sign(challenge, keypair);
859
- const newToken = `${pubkey}:${challenge}|${signature}`;
860
- console.debug(`[account] Resetting account pubkey, using token: `, newToken);
861
- const { data: res } = await this.graphql.query({
862
- query: ResetAccountPubkeyQuery,
863
- variables: { token: newToken, username: data.username },
864
- error: {
865
- code: ErrorCodes.AUTH_RESET_PUBKEY_ERROR,
866
- message: 'ACCOUNT.ERROR.RESET_PUBKEY_ERROR',
867
- },
868
- fetchPolicy: 'no-cache',
869
- });
870
- // Login
871
- if (res) {
872
- await this.login(data);
873
- }
874
- return res;
875
- }
876
- listenChanges(opts) {
877
- if (this._enableListenChanges)
878
- return new Subscription(); // Already started: skip
879
- return this.startListenRemoteChanges(opts);
880
- }
881
- get additionalFields() {
882
- return this._$additionalFields.getValue();
883
- }
884
- /**
885
- * @deprecated Use additionalFields$ instead
886
- */
887
- get $additionalFields() {
888
- return this._$additionalFields.asObservable();
889
- }
890
- get additionalFields$() {
891
- return this._$additionalFields.asObservable();
892
- }
893
- getAdditionalField(key) {
894
- return this._$additionalFields.getValue().find((f) => f.key === key);
895
- }
896
- registerAdditionalField(...fields) {
897
- const oldFields = this._$additionalFields.getValue() || [];
898
- const newFields = fields?.reduce((res, field) => {
899
- if (res.some((f) => f.key === field.key)) {
900
- throw new Error(`Additional account field {key: ${field.key}} already define.`);
901
- }
902
- //if (this._debug)
903
- console.debug(`[account] Register additional account's field {key: ${field.key}}`);
904
- return res.concat(field);
905
- }, oldFields);
906
- this._$additionalFields.next(newFields);
907
- }
908
- get optionDefs() {
909
- return this._optionDefs;
910
- }
911
- registerOption(def) {
912
- if (this._optionDefs.findIndex((f) => f.key === def.key) !== -1) {
913
- throw new Error(`Additional option {${def.key}} already define.`);
914
- }
915
- if (this._debug)
916
- console.debug(`[account] Adding additional option {${def.key}}`, def);
917
- this._optionDefs.push(def);
918
- }
919
- registerOptions(defs) {
920
- (defs || []).forEach((def) => this.registerOption(def));
921
- }
922
- /* -- protected method -- */
923
- resetData() {
924
- this.stopListenRemoteChanges();
925
- this._cache.loaded = false;
926
- this._cache.keypair = null;
927
- this._cache.authToken = null;
928
- this._cache.authBasic = null;
929
- this._cache.pubkey = null;
930
- this._cache.mainProfile = null;
931
- this._cache.person = null;
932
- this._cache.department = null;
933
- this._data = new Account();
934
- }
935
- async loadData(opts) {
936
- if (!this._cache.pubkey)
937
- throw new Error('User not logged');
938
- this._cache.loaded = false;
939
- console.debug('[account-service] Loading account data...');
940
- try {
941
- const account = (await this.load(opts)) || new Account();
942
- // Set defaults
943
- account.avatar = account.avatar || this.environment.baseUrl + PersonUtils.DEFAULT_AVATAR_IMAGE;
944
- account.settings = account.settings || new UserSettings();
945
- account.settings.locale = account.settings.locale || this.settings.locale;
946
- account.settings.latLongFormat = account.settings.latLongFormat || this.settings.latLongFormat || 'DDMM';
947
- account.settings.content = account.settings.content || {};
948
- // Read main profile
949
- this._cache.mainProfile = PersonUtils.getMainProfile(account.profiles);
950
- // Update, instead of replace it
951
- if (this._data) {
952
- this._data.fromObject(account);
953
- }
954
- else {
955
- this._data = account;
956
- }
957
- this._cache.loaded = true;
958
- // Apply settings, found in remote account
959
- if (account.settings && this.settings.settings.accountInheritance) {
960
- console.debug('[account-service] Copying account settings into local settings...');
961
- const localSettings = this.settings.settings;
962
- const accountSettings = account.settings.asLocalSettings();
963
- const definitions = removeDuplicatesFromArray([...this.optionDefs, ...this.settings.optionDefs], 'key');
964
- const mergedProperties = AppPropertiesUtils.sanitize({
965
- ...localSettings.properties,
966
- ...accountSettings.properties,
967
- }, { definitions });
968
- // Merge local/account settings
969
- const settings = {
970
- ...localSettings,
971
- ...accountSettings,
972
- properties: mergedProperties,
973
- };
974
- await this.settings.apply(settings);
975
- }
976
- return this._data;
977
- }
978
- catch (error) {
979
- this.resetData();
980
- if (error.code && error.message)
981
- throw error;
982
- console.error(error);
983
- throw {
984
- code: ErrorCodes.LOAD_ACCOUNT_ERROR,
985
- message: 'ERROR.LOAD_ACCOUNT_ERROR',
986
- };
987
- }
988
- }
989
- async listenSettings() {
990
- // When settings changed: save it remotely
991
- this.registerSubscription(this.settings.onChange
992
- .pipe(debounceTime(2000), filter(() => this.isLogin() && this.network.online))
993
- .subscribe((settings) => this.saveLocalSettingsRemotely(settings)));
994
- }
995
- async restoreLocally() {
996
- // Restore from storage
997
- const values = await Promise.all([
998
- this.storage.get(PUBKEY_STORAGE_KEY),
999
- this.storage.get(TOKEN_STORAGE_KEY),
1000
- this.storage.get(SECKEY_STORAGE_KEY),
1001
- ]);
1002
- let pubkey = values[0];
1003
- let token = values[1];
1004
- const seckey = values[2];
1005
- // DEV only - auth by token
1006
- if (!this.environment.production && this.environment.defaultAuthValues?.token) {
1007
- token = token || this.environment.defaultAuthValues.token;
1008
- pubkey = pubkey || token?.split(':', 2)[0];
1009
- }
1010
- // Quit if no pubkey (not logged)
1011
- if (!pubkey)
1012
- return;
1013
- // Quit if could not auth on pod
1014
- const canRemoteAuth = token || seckey || false;
1015
- if (!canRemoteAuth)
1016
- return;
1017
- if (this._debug)
1018
- console.debug(`[account] Account restoration...`);
1019
- this._cache.authToken = token;
1020
- this._cache.pubkey = pubkey;
1021
- this._cache.keypair =
1022
- (seckey && {
1023
- publicKey: Base58.decode(pubkey),
1024
- secretKey: Base58.decode(seckey),
1025
- }) ||
1026
- null;
1027
- // Online mode: try to connect to pod
1028
- if (this.network.online) {
1029
- try {
1030
- await this.authenticate();
1031
- if (!this._cache.authToken && !this._cache.authBasic)
1032
- throw new Error('Authentication failed');
1033
- }
1034
- catch (error) {
1035
- // Offline feature are enable: continue in offline mode
1036
- if (this.settings.hasOfflineFeature()) {
1037
- console.warn('[account] Unable to authenticate on pod: forcing offline mode');
1038
- this.network.setForceOffline(true, { showToast: false });
1039
- // Continue
1040
- }
1041
- // No offline features enable (=offline mode not allowed)
1042
- else {
1043
- console.error(error);
1044
- this.logout();
1045
- return;
1046
- }
1047
- }
1048
- }
1049
- // Get the account, from pubkey
1050
- let jsonAccount = await this.storage.get(`${ACCOUNT_STORAGE_KEY}#${pubkey}`);
1051
- if (!jsonAccount) {
1052
- // Try using the old storage key
1053
- const accountStr = await this.storage.get(ACCOUNT_STORAGE_KEY);
1054
- jsonAccount = accountStr && ((typeof accountStr === 'string' && JSON.parse(jsonAccount)) || accountStr);
1055
- }
1056
- // Invalid account: do not use it
1057
- if (!jsonAccount || jsonAccount.pubkey !== pubkey) {
1058
- // DEV only: create a fake account with given username
1059
- if (!this.environment.production && this.environment.defaultAuthValues.username) {
1060
- const username = this.environment.defaultAuthValues.username;
1061
- jsonAccount = { pubkey, username, firstName: username, lastName: username };
1062
- }
1063
- else {
1064
- return;
1065
- }
1066
- }
1067
- // Transform to entity
1068
- const account = Account.fromObject(jsonAccount);
1069
- // Update data
1070
- this._data = account;
1071
- this._cache.mainProfile = PersonUtils.getMainProfile(account.profiles);
1072
- this._cache.loaded = true;
1073
- // Emit event
1074
- this.onLogin.next(this._data);
1075
- this.onChange.next(this._data);
1076
- if (this._enableListenChanges)
1077
- this.startListenRemoteChanges();
1078
- if (this._debug)
1079
- console.debug(`[account] Account restoration [OK] {pubkey: ${pubkey.substr(0, 8)}}, {profile: ${this._cache.mainProfile}}`);
1080
- return account;
1081
- }
1082
- /**
1083
- * Save account into the local storage
1084
- */
1085
- async saveLocally() {
1086
- if (!this._cache.pubkey)
1087
- throw new Error('User not logged');
1088
- if (this._debug)
1089
- console.debug(`[account] Saving account {${this._cache.pubkey.substring(0, 6)}} in local storage...`);
1090
- // Convert account to json
1091
- const json = this._data.asObject({ keepTypename: true });
1092
- const seckey = (this._cache.keypair && this._cache.keypair.secretKey && Base58.encode(this._cache.keypair.secretKey)) || null;
1093
- // Convert avatar URL to dataUrl (e.g. 'data:image/png:<base64 content>')
1094
- const hasAvatarUrl = json.avatar &&
1095
- !json.avatar.endsWith(PersonUtils.DEFAULT_AVATAR_IMAGE) &&
1096
- (json.avatar.startsWith('http://') || json.avatar.startsWith('https://'));
1097
- if (hasAvatarUrl && this.network.online) {
1098
- await this.file
1099
- .getImage(json.avatar, {
1100
- thumbnail: true,
1101
- outputType: 'dataUrl',
1102
- })
1103
- .then((dataUrl) => {
1104
- if (dataUrl && this._debug)
1105
- console.debug('[account] Image fetched: ', dataUrl.substring(0, 50));
1106
- // TODO: make sure to display Base64 image in the menu top header
1107
- //jsonAccount.avatar = dataUrl;
1108
- })
1109
- .catch((err) => {
1110
- console.error(`[account] Error while fetching image: ${json.avatar}: ${err}`);
1111
- });
1112
- }
1113
- try {
1114
- await Promise.all([
1115
- this.storage.set(PUBKEY_STORAGE_KEY, this._cache.pubkey),
1116
- this.storage.set(TOKEN_STORAGE_KEY, this._cache.authToken),
1117
- this.storage.set(`${ACCOUNT_STORAGE_KEY}#${this._cache.pubkey}`, json),
1118
- // Secret key (optional)
1119
- (seckey && this.storage.set(SECKEY_STORAGE_KEY, seckey)) || this.storage.remove(SECKEY_STORAGE_KEY),
1120
- // Remove old storage key
1121
- this.storage.remove(ACCOUNT_STORAGE_KEY),
1122
- ]);
1123
- if (this._debug)
1124
- console.debug('[account] Account saved in local storage');
1125
- }
1126
- catch (err) {
1127
- console.error('[account] Error while saving account locally: ' + ((err && err.message) || err), err);
1128
- }
1129
- }
1130
- async saveLocalSettingsRemotely(source) {
1131
- if (!source || !this.isLogin() || source.accountInheritance === false)
1132
- return; // Skip
1133
- const account = this.account;
1134
- // Merge local settings into account's settings
1135
- const settings = UserSettings.fromObject(account.settings) || new UserSettings();
1136
- const changed = settings.merge(source, this._remoteLocalSettingsKeys);
1137
- if (!changed)
1138
- return; // Skip if unchanged
1139
- // Save remotely
1140
- this.account.settings = settings;
1141
- await this.saveSettingsRemotely(settings);
1142
- }
1143
- /**
1144
- * Create or update an user account
1145
- *
1146
- * @param account
1147
- * @param keyPair
1148
- */
1149
- async saveRemotely(account, keyPair) {
1150
- account.pubkey = account.pubkey || (keyPair && Base58.encode(keyPair.publicKey));
1151
- if (!account.pubkey)
1152
- throw new Error('Missing account pubkey');
1153
- const now = this._debug && Date.now();
1154
- if (this._debug)
1155
- console.debug(`[account] Saving account remotely...`);
1156
- const isNew = isNil(account.id);
1157
- // If this is an update: get existing account's updateDate, to avoid 'version error' when saving
1158
- if (!isNew) {
1159
- const existingAccount = await this.load({ fetchPolicy: 'network-only' });
1160
- if (!existingAccount || !existingAccount.updateDate) {
1161
- throw { code: ErrorCodes.ACCOUNT_NOT_EXISTS, message: 'ERROR.ACCOUNT_NOT_EXISTS' };
1162
- }
1163
- this.copyIdAndUpdateDate(existingAccount, account);
1164
- }
1165
- // Merging settings - FIXME - Ludo - A revoir avec la MR sur le user settings
1166
- //account.settings.content['pages'] = this.settings.settings.pages;
1167
- //account.settings.content['pageHistory'] = this.settings.settings.pageHistory;
1168
- // Convert to json
1169
- const json = account.asObject(MINIFY_ENTITY_FOR_POD);
1170
- // User not allow to change his profiles
1171
- delete json.profiles;
1172
- const mutation = isNew ? this.mutations.create : this._apiTokenEnabled ? this.mutations.saveWithTokens : this.mutations.save;
1173
- // Execute mutation
1174
- const { data } = await this.graphql.mutate({
1175
- mutation,
1176
- variables: { data: json },
1177
- error: {
1178
- code: ErrorCodes.SAVE_ACCOUNT_ERROR,
1179
- message: 'ERROR.SAVE_ACCOUNT_ERROR',
1180
- },
1181
- });
1182
- // Copy update properties
1183
- this.copyIdAndUpdateDate(data, account);
1184
- if (this._debug)
1185
- console.debug(`[account] Account remotely saved in ${Date.now() - now}ms`);
1186
- return account;
1187
- }
1188
- /**
1189
- * Create or update user settings
1190
- *
1191
- * @param settings
1192
- */
1193
- async saveSettingsRemotely(settings) {
1194
- const now = this._debug && Date.now();
1195
- if (this._debug)
1196
- console.debug(`[account] Saving settings remotely...`);
1197
- const json = settings.asObject(MINIFY_ENTITY_FOR_POD);
1198
- // Execute mutation
1199
- const { data } = await this.graphql.mutate({
1200
- mutation: this.mutations.saveSettings,
1201
- variables: { data: json },
1202
- error: {
1203
- code: ErrorCodes.SAVE_SETTINGS_ERROR,
1204
- message: 'ERROR.SAVE_SETTINGS_ERROR',
1205
- },
1206
- });
1207
- // Copy update properties
1208
- this.copyIdAndUpdateDateSettings(data, settings);
1209
- if (this._debug)
1210
- console.debug(`[account] Settings remotely saved in ${Date.now() - now}ms`);
1211
- // Save into account
1212
- if (this.account)
1213
- this.account.settings = settings;
1214
- }
1215
- async authenticateAndGetToken(token, counter) {
1216
- if (!this._cache.pubkey)
1217
- throw new Error('User not logged');
1218
- if (!counter)
1219
- console.info('[account] Authentication on pod...');
1220
- if (counter > 4) {
1221
- if (this._debug)
1222
- console.debug(`[account] Failed to authentication on pod (after ${counter} attempts)`);
1223
- throw { code: ErrorCodes.AUTH_SERVER_ERROR, message: 'ERROR.AUTH_SERVER_ERROR' };
1224
- }
1225
- // Check if valid
1226
- if (token) {
1227
- const data = await this.graphql.query({
1228
- query: AuthQuery,
1229
- variables: {
1230
- token,
1231
- },
1232
- error: {
1233
- code: ErrorCodes.UNAUTHORIZED,
1234
- message: 'ERROR.UNAUTHORIZED',
1235
- },
1236
- fetchPolicy: 'no-cache',
1237
- });
1238
- // Token is accepted by the server
1239
- if (data && data.authenticate) {
1240
- // Store the token
1241
- this.onAuthTokenChange.next(token);
1242
- console.info(`[account] Authentication on pod [OK] {pubkey: '${this._cache.pubkey.substr(0, 8)}'}`);
1243
- return token; // return the token
1244
- }
1245
- // Continue (will retry with another challenge)
1246
- }
1247
- // Generate a new token
1248
- const authChallenge = await this.getAuthChallenge();
1249
- // TODO: check server pubkey as a valid certificate
1250
- // Do the challenge
1251
- const signature = await CryptoService.sign(authChallenge.challenge, this._cache.keypair);
1252
- const newToken = `${this._cache.pubkey}:${authChallenge.challenge}|${signature}`;
1253
- // iterate with the new token
1254
- return await this.authenticateAndGetToken(newToken, (counter || 1) + 1 /* increment */);
1255
- }
1256
- async getAuthChallenge() {
1257
- const challengeError = {
1258
- code: ErrorCodes.AUTH_CHALLENGE_ERROR,
1259
- message: 'ERROR.AUTH_CHALLENGE_ERROR',
1260
- };
1261
- const { authChallenge } = await this.graphql.query({
1262
- query: AuthChallengeQuery,
1263
- variables: {},
1264
- error: challengeError,
1265
- fetchPolicy: 'network-only',
1266
- });
1267
- // Check challenge
1268
- if (!authChallenge)
1269
- throw challengeError; // Should never occur
1270
- // Check server signature
1271
- const signatureOK = await CryptoService.verify(authChallenge.challenge, authChallenge.signature, authChallenge.pubkey);
1272
- if (!signatureOK) {
1273
- console.warn('FIXME: Bad peer signature on auth challenge !', authChallenge);
1274
- }
1275
- return authChallenge;
1276
- }
1277
- startListenRemoteChanges(opts) {
1278
- if (this._listenChangesSubscription)
1279
- this.stopListenRemoteChanges(); // Stop previous subscription
1280
- if (this.network.offline)
1281
- return new Subscription(); // Offline: skip
1282
- this._listenChangesSubscription = this.watchChanges(opts).subscribe({
1283
- next: (data) => {
1284
- console.debug(`[account] Remote update detected at ${data.updateDate}`);
1285
- this.reload();
1286
- },
1287
- error: (err) => {
1288
- if (err && +err.code === ServerErrorCodes.NOT_FOUND) {
1289
- console.info('[account] [WS] Account not exists anymore: force user to logout...', err);
1290
- this.logout();
1291
- }
1292
- else if (err && +err.code === ServerErrorCodes.UNAUTHORIZED) {
1293
- console.info('[account] [WS] Account not authorized: force user to logout...', err);
1294
- this.logout();
1295
- }
1296
- else {
1297
- console.warn('[account] [WS] Received error:', err);
1298
- }
1299
- },
1300
- });
1301
- this._listenChangesSubscription.add(() => {
1302
- console.debug(`[account] Stop watching remote changes`);
1303
- this._listenChangesSubscription = null;
1304
- });
1305
- return this._listenChangesSubscription;
1306
- }
1307
- stopListenRemoteChanges() {
1308
- this._listenChangesSubscription?.unsubscribe();
1309
- }
1310
- watchChanges(opts) {
1311
- if (!this.started) {
1312
- // Wait service ready, then loop
1313
- return from(this.ready()).pipe(switchMap(() => this.watchChanges(opts)));
1314
- }
1315
- if (!this._cache.pubkey)
1316
- throw new Error('Not logged in');
1317
- if (this.network.offline)
1318
- throw new Error('Cannot watch account changes: network is offline.');
1319
- const variables = {
1320
- interval: toNumber(opts && opts.intervalInSeconds, this._listenIntervalInSeconds),
1321
- };
1322
- console.debug(`[account] Watching remote changes, every ${variables.interval}s`);
1323
- return this.graphql
1324
- .subscribe({
1325
- query: this.subscriptions.listenChanges,
1326
- variables,
1327
- error: {
1328
- code: ErrorCodes.SUBSCRIBE_ACCOUNT_ERROR,
1329
- message: 'ACCOUNT.ERROR.SUBSCRIBE_ERROR',
1330
- },
1331
- })
1332
- .pipe(map(({ data }) => data),
1333
- // Keep only if newer
1334
- filter((data) => {
1335
- const currentUpdateDate = fromDateISOString(this._data?.updateDate);
1336
- const remoteUpdateDate = fromDateISOString(data?.updateDate);
1337
- return remoteUpdateDate && (!currentUpdateDate || !remoteUpdateDate.isSame(currentUpdateDate, 'millisecond'));
1338
- }));
1339
- }
1340
- copyIdAndUpdateDate(source, target) {
1341
- if (!source)
1342
- return;
1343
- target.id = source.id || target.id;
1344
- target.updateDate = source.updateDate || target.updateDate;
1345
- // Update settings
1346
- this.copyIdAndUpdateDateSettings(source.settings, target.settings);
1347
- // Update tokens
1348
- if (this._apiTokenEnabled) {
1349
- this.copyIdAndUpdateDateToken(source.tokens, target.tokens);
1350
- }
1351
- }
1352
- copyIdAndUpdateDateSettings(source, target) {
1353
- if (!source)
1354
- return;
1355
- target.id = source.id || target.id;
1356
- target.updateDate = source.updateDate || target.updateDate;
1357
- }
1358
- copyIdAndUpdateDateToken(sources, targets) {
1359
- if (isEmptyArray(sources))
1360
- return;
1361
- sources.forEach((source) => {
1362
- const target = targets.find((value) => UserToken.equals(value, source));
1363
- if (target) {
1364
- target.id = source.id || target.id;
1365
- target.creationDate = source.creationDate || target.creationDate;
1366
- target.updateDate = source.updateDate || target.updateDate;
1367
- target.token = undefined;
1368
- }
1369
- });
1370
- }
1371
- showToast(opts) {
1372
- return Toasts.show(this.toastController, this.translate, opts);
1373
- }
1374
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AccountService, deps: [{ token: i1.NetworkService }, { token: i2.GraphqlService }, { token: i3.LocalSettingsService }, { token: i4.Storage }, { token: i5.FileService }, { token: i6.TranslateService, optional: true }, { token: i7.ToastController, optional: true }, { token: ENVIRONMENT }, { token: APP_USER_SETTINGS_OPTIONS, optional: true }, { token: APP_ACCOUNT_SERVICE_OPTIONS, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
1375
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AccountService, providedIn: 'root' });
1376
- }
1377
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AccountService, decorators: [{
1378
- type: Injectable,
1379
- args: [{ providedIn: 'root', deps: [ENVIRONMENT] }]
1380
- }], ctorParameters: () => [{ type: i1.NetworkService }, { type: i2.GraphqlService }, { type: i3.LocalSettingsService }, { type: i4.Storage }, { type: i5.FileService }, { type: i6.TranslateService, decorators: [{
1381
- type: Optional
1382
- }] }, { type: i7.ToastController, decorators: [{
1383
- type: Optional
1384
- }] }, { type: i8.Environment, decorators: [{
1385
- type: Inject,
1386
- args: [ENVIRONMENT]
1387
- }] }, { type: undefined, decorators: [{
1388
- type: Optional
1389
- }, {
1390
- type: Inject,
1391
- args: [APP_USER_SETTINGS_OPTIONS]
1392
- }] }, { type: undefined, decorators: [{
1393
- type: Optional
1394
- }, {
1395
- type: Inject,
1396
- args: [APP_ACCOUNT_SERVICE_OPTIONS]
1397
- }] }] });
1398
- export function provideAccountService() {
1399
- return {
1400
- provide: APP_ACCOUNT_SERVICE, useClass: AccountService, deps: [ENVIRONMENT]
1401
- };
1402
- }
1403
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNjb3VudC5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9jb3JlL3NlcnZpY2VzL2FjY291bnQuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLFVBQVUsRUFBRSxjQUFjLEVBQUUsUUFBUSxFQUFZLE1BQU0sZUFBZSxDQUFDO0FBQ3ZGLE9BQU8sRUFBRSxhQUFhLEVBQVcsTUFBTSxrQkFBa0IsQ0FBQztBQUUxRCxPQUFPLEVBQUUsT0FBTyxFQUFFLFlBQVksRUFBRSxZQUFZLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUM1RSxPQUFPLEVBQVUsV0FBVyxFQUFvQixNQUFNLHNCQUFzQixDQUFDO0FBRTdFLE9BQU8sRUFBRSxlQUFlLEVBQUUsSUFBSSxFQUFjLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDaEYsT0FBTyxFQUFlLEdBQUcsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBR3ZELE9BQU8sRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLHlCQUF5QixFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN6RyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUNsRSxPQUFPLEVBQUUsVUFBVSxFQUFFLGdCQUFnQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBT3hELE9BQU8sRUFBRSxxQkFBcUIsRUFBZSxnQkFBZ0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ2pHLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUMvQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQ2xDLE9BQU8sRUFBZSxXQUFXLEVBQUUsTUFBTSx5Q0FBeUMsQ0FBQztBQUNuRixPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztB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LE1BQU0sRUFBRSxJQUFJLEVBQUUsQ0FBQyxXQUFXLENBQUMsRUFBRTs7MEJBK0ZsRCxRQUFROzswQkFDUixRQUFROzswQkFDUixNQUFNOzJCQUFDLFdBQVc7OzBCQUNsQixRQUFROzswQkFBSSxNQUFNOzJCQUFDLHlCQUF5Qjs7MEJBQzVDLFFBQVE7OzBCQUFJLE1BQU07MkJBQUMsMkJBQTJCOztBQXVzQ25ELE1BQU0sVUFBVSxxQkFBcUI7SUFDbkMsT0FBaUI7UUFDZixPQUFPLEVBQUUsbUJBQW1CLEVBQUUsUUFBUSxFQUFFLGNBQWMsRUFBRSxJQUFJLEVBQUUsQ0FBQyxXQUFXLENBQUM7S0FDNUUsQ0FBQTtBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3QsIEluamVjdGFibGUsIEluamVjdGlvblRva2VuLCBPcHRpb25hbCwgUHJvdmlkZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENyeXB0b1NlcnZpY2UsIEtleXBhaXIgfSBmcm9tICcuL2NyeXB0by5zZXJ2aWNlJztcbmltcG9ydCB7IERlcGFydG1lbnQgfSBmcm9tICcuL21vZGVsL2RlcGFydG1lbnQubW9kZWwnO1xuaW1wb3J0IHsgQWNjb3VudCwgQWNjb3VudFV0aWxzLCBVc2VyU2V0dGluZ3MgfSBmcm9tICcuL21vZGVsL2FjY291bnQubW9kZWwnO1xuaW1wb3J0IHsgUGVyc29uLCBQZXJzb25VdGlscywgVXNlclByb2ZpbGVMYWJlbCB9IGZyb20gJy4vbW9kZWwvcGVyc29uLm1vZGVsJztcbmltcG9ydCB7IExvY2FsU2V0dGluZ3MsIFVzYWdlTW9kZSB9IGZyb20gJy4vbW9kZWwvc2V0dGluZ3MubW9kZWwnO1xuaW1wb3J0IHsgQmVoYXZpb3JTdWJqZWN0LCBmcm9tLCBPYnNlcnZhYmxlLCBTdWJqZWN0LCBTdWJzY3JpcHRpb24gfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IEZldGNoUG9saWN5LCBncWwgfSBmcm9tICdAYXBvbGxvL2NsaWVudC9jb3JlJztcbmltcG9ydCB7IFN0b3JhZ2UgfSBmcm9tICdAaW9uaWMvc3RvcmFnZS1hbmd1bGFyJztcblxuaW1wb3J0IHsgaXNFbXB0eUFycmF5LCBpc05pbCwgcmVtb3ZlRHVwbGljYXRlc0Zyb21BcnJheSwgc2xlZXAsIHRvTnVtYmVyIH0gZnJvbSAnLi4vLi4vc2hhcmVkL2Z1bmN0aW9ucyc7XG5pbXBvcnQgeyBCYXNlR3JhcGhxbFNlcnZpY2UgfSBmcm9tICcuL2Jhc2UtZ3JhcGhxbC1zZXJ2aWNlLmNsYXNzJztcbmltcG9ydCB7IEVycm9yQ29kZXMsIFNlcnZlckVycm9yQ29kZXMgfSBmcm9tICcuL2Vycm9ycyc7XG5pbXBvcnQgeyBHcmFwaHFsU2VydmljZSB9IGZyb20gJy4uL2dyYXBocWwvZ3JhcGhxbC5zZXJ2aWNlJztcbmltcG9ydCB7IExvY2FsU2V0dGluZ3NTZXJ2aWNlIH0gZnJvbSAnLi9sb2NhbC1zZXR0aW5ncy5zZXJ2aWNlJztcbmltcG9ydCB7IEZvcm1GaWVsZERlZmluaXRpb24sIEZvcm1GaWVsZERlZmluaXRpb25NYXAgfSBmcm9tICcuLi8uLi9zaGFyZWQvZm9ybS9maWVsZC5tb2RlbCc7XG5pbXBvcnQgeyBOZXR3b3JrU2VydmljZSB9IGZyb20gJy4vbmV0d29yay5zZXJ2aWNlJztcbmltcG9ydCB7IEF1dGhUb2tlblR5cGUgfSBmcm9tICcuL25ldHdvcmsudHlwZXMnO1xuaW1wb3J0IHsgRmlsZVNlcnZpY2UgfSBmcm9tICcuLi8uLi9zaGFyZWQvZmlsZS9maWxlLnNlcnZpY2UnO1xuaW1wb3J0IHsgTUlOSUZZX0VOVElUWV9GT1JfUE9ELCBSZWZlcmVudGlhbCwgUmVmZXJlbnRpYWxVdGlscyB9IGZyb20gJy4vbW9kZWwvcmVmZXJlbnRpYWwubW9kZWwnO1xuaW1wb3J0IHsgU3RhdHVzSWRzIH0gZnJvbSAnLi9tb2RlbC9tb2RlbC5lbnVtJztcbmltcG9ydCB7IEJhc2U1OCB9IGZyb20gJy4vYmFzZTU4JztcbmltcG9ydCB7IEVudmlyb25tZW50LCBFTlZJUk9OTUVOVCB9IGZyb20gJy4uLy4uLy4uL2Vudmlyb25tZW50cy9lbnZpcm9ubWVudC5jbGFzcyc7XG5pbXBvcnQgeyBmcm9tRGF0ZUlTT1N0cmluZyB9IGZyb20gJy4uLy4uL3NoYXJlZC9kYXRlcyc7XG5pbXBvcnQgeyBmaXJzdE5vdE5pbFByb21pc2UgfSBmcm9tICcuLi8uLi9zaGFyZWQvb2JzZXJ2YWJsZXMnO1xuaW1wb3J0IHsgZGVib3VuY2VUaW1lLCBmaWx0ZXIsIG1hcCwgc3RhcnRXaXRoLCBzd2l0Y2hNYXAgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQgeyBCYXNlRW50aXR5R3JhcGhxbE11dGF0aW9ucywgQmFzZUVudGl0eUdyYXBocWxRdWVyaWVzLCBCYXNlRW50aXR5R3JhcGhxbFN1YnNjcmlwdGlvbnMgfSBmcm9tICcuL2Jhc2UtZW50aXR5LXNlcnZpY2UuY2xhc3MnO1xuaW1wb3J0IHsgTW9tZW50IH0gZnJvbSAnbW9tZW50JztcbmltcG9ydCB7IFNob3dUb2FzdE9wdGlvbnMsIFRvYXN0cyB9IGZyb20gJy4uLy4uL3NoYXJlZC90b2FzdC90b2FzdHMnO1xuaW1wb3J0IHsgT3ZlcmxheUV2ZW50RGV0YWlsIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuaW1wb3J0IHsgVG9hc3RDb250cm9sbGVyIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXInO1xuaW1wb3J0IHsgVHJhbnNsYXRlU2VydmljZSB9IGZyb20gJ0BuZ3gtdHJhbnNsYXRlL2NvcmUnO1xuaW1wb3J0IHsgVG9rZW5TY29wZSwgVXNlclRva2VuIH0gZnJvbSAnLi9tb2RlbC90b2tlbi5tb2RlbCc7XG5pbXBvcnQgeyBBcHBQcm9wZXJ0aWVzVXRpbHMgfSBmcm9tICcuLi9mb3JtL3Byb3BlcnRpZXMvcHJvcGVydGllcy51dGlscyc7XG5pbXBvcnQgeyBBUFBfTE9DQUxfU1RPUkFHRV9UWVBFX1BPTElDSUVTIH0gZnJvbSAnLi9zdG9yYWdlL2VudGl0aWVzLXN0b3JhZ2Uuc2VydmljZSc7XG5cbmV4cG9ydCBkZWNsYXJlIGludGVyZmFjZSBBY2NvdW50RGV0YWlscyB7XG4gIGxvYWRlZDogYm9vbGVhbjtcbiAga2V5cGFpcjogS2V5cGFpcjtcbiAgYXV0aFRva2VuOiBzdHJpbmc7XG4gIGF1dGhCYXNpYz86IHN0cmluZztcbiAgcHVia2V5OiBzdHJpbmc7XG4gIHBlcnNvbjogUGVyc29uO1xuICBkZXBhcnRtZW50OiBEZXBhcnRtZW50O1xuICBtYWluUHJvZmlsZTogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEF1dGhEYXRhIHtcbiAgdXNlcm5hbWU6IHN0cmluZztcbiAgcGFzc3dvcmQ6IHN0cmluZztcbiAgb2ZmbGluZT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVnaXN0ZXJEYXRhIGV4dGVuZHMgQXV0aERhdGEge1xuICBhY2NvdW50OiBBY2NvdW50O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENoYW5nZVBhc3N3b3JkRGF0YSB7XG4gIHBhc3N3b3JkOiBzdHJpbmc7XG4gIGNvbmZpcm1QYXNzd29yZDogc3RyaW5nO1xuICBlbWFpbDogc3RyaW5nO1xufVxuXG5jb25zdCBUT0tFTl9TVE9SQUdFX0tFWSA9ICd0b2tlbic7XG5jb25zdCBQVUJLRVlfU1RPUkFHRV9LRVkgPSAncHVia2V5JztcbmNvbnN0IFNFQ0tFWV9TVE9SQUdFX0tFWSA9ICdzZWNrZXknO1xuY29uc3QgQUNDT1VOVF9TVE9SQUdFX0tFWSA9ICdhY2NvdW50JztcblxuY29uc3QgREVGQVVMVF9BVkFUQVJfSU1BR0UgPSBQZXJzb25VdGlscy5ERUZBVUxUX0FWQVRBUl9JTUFHRTtcblxuZXhwb3J0IGludGVyZmFjZSBVc2VyU2V0dGluZ3NPcHRpb25zIHtcbiAgb3B0aW9uczogRm9ybUZpZWxkRGVmaW5pdGlvbk1hcDtcbiAgcmVtb3RlTG9jYWxTZXR0aW5nc0tleXM/OiAoa2V5b2YgTG9jYWxTZXR0aW5ncyB8IHN0cmluZylbXTtcbn1cblxuZXhwb3J0IGNvbnN0IEFQUF9VU0VSX1NFVFRJTkdTX09QVElPTlMgPSBuZXcgSW5qZWN0aW9uVG9rZW48VXNlclNldHRpbmdzT3B0aW9ucz4oJ1VzZXJTZXR0aW5nc09wdGlvbnMnKTtcbmV4cG9ydCBjb25zdCBBUFBfVVNFUl9UT0tFTl9TQ09QRVMgPSBuZXcgSW5qZWN0aW9uVG9rZW48VG9rZW5TY29wZVtdPignVXNlclRva2VuU2NvcGVzJyk7XG4vLyBBbGxvdyBhcHBzIHRvIGluamVjdC9vdmVycmlkZSB0aGUgYWNjb3VudCBzZXJ2aWNlIGltcGxlbWVudGF0aW9uXG5leHBvcnQgY29uc3QgQVBQX0FDQ09VTlRfU0VSVklDRSA9IG5ldyBJbmplY3Rpb25Ub2tlbjxBY2NvdW50U2VydmljZT4oJ0FjY291bnRTZXJ2aWNlJyk7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQWNjb3VudFNlcnZpY2VHcmFwaHFsUXVlcmllcyBleHRlbmRzIEJhc2VFbnRpdHlHcmFwaHFsUXVlcmllcyB7XG4gIGxvYWRXaXRoVG9rZW5zPzogYW55O1xufVxuZXhwb3J0IGludGVyZmFjZSBBY2NvdW50U2VydmljZUdyYXBocWxNdXRhdGlvbnMgZXh0ZW5kcyBCYXNlRW50aXR5R3JhcGhxbE11dGF0aW9ucyB7XG4gIHNhdmVXaXRoVG9rZW5zPzogYW55O1xuICBzYXZlU2V0dGluZ3M/OiBhbnk7XG4gIGNyZWF0ZT86IGFueTtcbn1cbmV4cG9ydCBpbnRlcmZhY2UgQWNjb3VudFNlcnZpY2VHcmFwaHFsU3Vic2NyaXB0aW9ucyBleHRlbmRzIEJhc2VFbnRpdHlHcmFwaHFsU3Vic2NyaXB0aW9ucyB7fVxuXG5leHBvcnQgaW50ZXJmYWNlIEFjY291bnRTZXJ2aWNlT3B0aW9ucyB7XG4gIHF1ZXJpZXM/OiBQYXJ0aWFsPEFjY291bnRTZXJ2aWNlR3JhcGhxbFF1ZXJpZXM+O1xuICBtdXRhdGlvbnM/OiBQYXJ0aWFsPEFjY291bnRTZXJ2aWNlR3JhcGhxbE11dGF0aW9ucz47XG4gIHN1YnNjcmlwdGlvbnM/OiBQYXJ0aWFsPEFjY291bnRTZXJ2aWNlR3JhcGhxbFN1YnNjcmlwdGlvbnM+O1xufVxuZXhwb3J0IGNvbnN0IEFQUF9BQ0NPVU5UX1NFUlZJQ0VfT1BUSU9OUyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxBY2NvdW50U2VydmljZU9wdGlvbnM+KCdBY2NvdW50U2VydmljZU9wdGlvbnMnKTtcblxuLyogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBHcmFwaFFMIHF1ZXJpZXNcbiAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovXG5jb25zdCBGcmFnbWVudHMgPSB7XG4gIGFjY291bnQ6IGdxbGBcbiAgICBmcmFnbWVudCBBY2NvdW50RnJhZ21lbnQgb24gQWNjb3VudFZPIHtcbiAgICAgIGlkXG4gICAgICBmaXJzdE5hbWVcbiAgICAgIGxhc3ROYW1lXG4gICAgICBlbWFpbFxuICAgICAgcHVia2V5XG4gICAgICBhdmF0YXJcbiAgICAgIHN0YXR1c0lkXG4gICAgICB1cGRhdGVEYXRlXG4gICAgICBjcmVhdGlvbkRhdGVcbiAgICAgIHByb2ZpbGVzXG4gICAgICBzZXR0aW5ncyB7XG4gICAgICAgIC4uLlVzZXJTZXR0aW5nc0ZyYWdtZW50XG4gICAgICB9XG4gICAgICBkZXBhcnRtZW50IHtcbiAgICAgICAgaWRcbiAgICAgICAgbGFiZWxcbiAgICAgICAgbmFtZVxuICAgICAgICBfX3R5cGVuYW1lXG4gICAgICB9XG4gICAgICBfX3R5cGVuYW1lXG4gICAgfVxuICBgLFxuXG4gIHNldHRpbmdzOiBncWxgXG4gICAgZnJhZ21lbnQgVXNlclNldHRpbmdzRnJhZ21lbnQgb24gVXNlclNldHRpbmdzVk8ge1xuICAgICAgaWRcbiAgICAgIGxvY2FsZVxuICAgICAgbGF0TG9uZ0Zvcm1hdFxuICAgICAgY29udGVudFxuICAgICAgbm9uY2VcbiAgICAgIHVwZGF0ZURhdGVcbiAgICAgIF9fdHlwZW5hbWVcbiAgICB9XG4gIGAsXG5cbiAgdG9rZW46IGdxbGBcbiAgICBmcmFnbWVudCBVc2VyVG9rZW5GcmFnbWVudCBvbiBVc2VyVG9rZW5WTyB7XG4gICAgICBpZFxuICAgICAgcHVia2V5XG4gICAgICBuYW1lXG4gICAgICBmbGFnc1xuICAgICAgZXhwaXJhdGlvbkRhdGVcbiAgICAgIGxhc3RVc2VkRGF0ZVxuICAgICAgY3JlYXRpb25EYXRlXG4gICAgICB1cGRhdGVEYXRlXG4gICAgICBfX3R5cGVuYW1lXG4gICAgfVxuICBgLFxufTtcblxuLy8gQWNjb3VudCBxdWVyaWVzXG5jb25zdCBRdWVyaWVzOiBQYXJ0aWFsPEJhc2VFbnRpdHlHcmFwaHFsUXVlcmllcz4gJiB7IGxvYWRXaXRoVG9rZW5zOiBhbnkgfSA9IHtcbiAgbG9hZDogZ3FsYFxuICAgIHF1ZXJ5IEFjY291bnQge1xuICAgICAgZGF0YTogYWNjb3VudCB7XG4gICAgICAgIC4uLkFjY291bnRGcmFnbWVudFxuICAgICAgfVxuICAgIH1cbiAgICAke0ZyYWdtZW50cy5hY2NvdW50fVxuICAgICR7RnJhZ21lbnRzLnNldHRpbmdzfVxuICBgLFxuXG4gIGxvYWRXaXRoVG9rZW5zOiBncWxgXG4gICAgcXVlcnkgQWNjb3VudFdpdGhUb2tlbiB7XG4gICAgICBkYXRhOiBhY2NvdW50IHtcbiAgICAgICAgLi4uQWNjb3VudEZyYWdtZW50XG4gICAgICAgIHRva2VucyB7XG4gICAgICAgICAgLi4uVXNlclRva2VuRnJhZ21lbnRcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICAke0ZyYWdtZW50cy5hY2NvdW50fVxuICAgICR7RnJhZ21lbnRzLnNldHRpbmdzfVxuICAgICR7RnJhZ21lbnRzLnRva2VufVxuICBgLFxufTtcblxuLy8gQ2hlY2sgZW1haWwgcXVlcnlcbmNvbnN0IElzRW1haWxFeGlzdHNRdWVyeTogYW55ID0gZ3FsYFxuICBxdWVyeSBJc0VtYWlsRXhpc3RzKCRlbWFpbDogU3RyaW5nLCAkaGFzaDogU3RyaW5nKSB7XG4gICAgaXNFbWFpbEV4aXN0cyhlbWFpbDogJGVtYWlsLCBoYXNoOiAkaGFzaClcbiAgfVxuYDtcblxuZXhwb3J0IGRlY2xhcmUgaW50ZXJmYWNlIElzRW1haWxFeGlzdHNWYXJpYWJsZXMge1xuICBlbWFpbDogc3RyaW5nO1xuICBoYXNoOiBzdHJpbmc7XG59XG5cbi8vIEFjY291bnQgbXV0YXRpb25zXG5jb25zdCBNdXRhdGlvbnM6IFBhcnRpYWw8QmFzZUVudGl0eUdyYXBocWxNdXRhdGlvbnM+ICYgeyBzYXZlV2l0aFRva2VuczogYW55OyBzYXZlU2V0dGluZ3M6IGFueTsgY3JlYXRlOiBhbnkgfSA9IHtcbiAgc2F2ZTogZ3FsYFxuICAgIG11dGF0aW9uIFNhdmVBY2NvdW50KCRkYXRhOiBBY2NvdW50Vk9JbnB1dCkge1xuICAgICAgZGF0YTogc2F2ZUFjY291bnQoYWNjb3VudDogJGRhdGEpIHtcbiAgICAgICAgLi4uQWNjb3VudEZyYWdtZW50XG4gICAgICB9XG4gICAgfVxuICAgICR7RnJhZ21lbnRzLmFjY291bnR9XG4gICAgJHtGcmFnbWVudHMuc2V0dGluZ3N9XG4gIGAsXG5cbiAgc2F2ZVdpdGhUb2tlbnM6IGdxbGBcbiAgICBtdXRhdGlvbiBTYXZlQWNjb3VudFdpdGhUb2tlbnMoJGRhdGE6IEFjY291bnRWT0lucHV0KSB7XG4gICAgICBkYXRhOiBzYXZlQWNjb3VudChhY2NvdW50OiAkZGF0YSkge1xuICAgICAgICAuLi5BY2NvdW50RnJhZ21lbnRcbiAgICAgICAgdG9rZW5zIHtcbiAgICAgICAgICAuLi5Vc2VyVG9rZW5GcmFnbWVudFxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICAgICR7RnJhZ21lbnRzLmFjY291bnR9XG4gICAgJHtGcmFnbWVudHMuc2V0dGluZ3N9XG4gICAgJHtGcmFnbWVudHMudG9rZW59XG4gIGAsXG5cbiAgY3JlYXRlOiBncWxgXG4gICAgbXV0YXRpb24gQ3JlYXRlQWNjb3VudCgkZGF0YTogQWNjb3VudFZPSW5wdXQpIHtcbiAgICAgIGRhdGE6IGNyZWF0ZUFjY291bnQoYWNjb3VudDogJGRhdGEpIHtcbiAgICAgICAgLi4uQWNjb3VudEZyYWdtZW50XG4gICAgICB9XG4gICAgfVxuICAgICR7RnJhZ21lbnRzLmFjY291bnR9XG4gICAgJHtGcmFnbWVudHMuc2V0dGluZ3N9XG4gIGAsXG5cbiAgc2F2ZVNldHRpbmdzOiBncWxgXG4gICAgbXV0YXRpb24gU2F2ZVNldHRpbmdzKCRkYXRhOiBVc2VyU2V0dGluZ3NWT0lucHV0KSB7XG4gICAgICBkYXRhOiBzYXZlU2V0dGluZ3Moc2V0dGluZ3M6ICRkYXRhKSB7XG4gICAgICAgIC4uLlVzZXJTZXR0aW5nc0ZyYWdtZW50XG4gICAgICB9XG4gICAgfVxuICAgICR7RnJhZ21lbnRzLnNldHRpbmdzfVxuICBgLFxufTtcblxuLy8gU2VudCBjb25maXJtYXRpb24gZW1haWxcbmNvbnN0IFNlbmRDb25maXJtRW1haWxRdWVyeTogYW55ID0gZ3FsYFxuICBxdWVyeSBzZW5kQWNjb3VudENvbmZpcm1hdGlvbkVtYWlsKCRlbWFpbDogU3RyaW5nLCAkbG9jYWxlOiBTdHJpbmcpIHtcbiAgICBkYXRhOiBzZW5kQWNjb3VudENvbmZpcm1hdGlvbkVtYWlsKGVtYWlsOiAkZW1haWwsIGxvY2FsZTogJGxvY2FsZSlcbiAgfVxuYDtcblxuLy8gQ29uZmlybSBhY2NvdW50IGVtYWlsXG5jb25zdCBDb25maXJtRW1haWxRdWVyeTogYW55ID0gZ3FsYFxuICBxdWVyeSBjb25maXJtQWNjb3VudEVtYWlsKCRlbWFpbDogU3RyaW5nLCAkY29kZTogU3RyaW5nKSB7XG4gICAgZGF0YTogY29uZmlybUFjY291bnRFbWFpbChlbWFpbDogJGVtYWlsLCBjb2RlOiAkY29kZSlcbiAgfVxuYDtcblxuLy8gU2VuZCBlbWFpbCB0byByZXNldCBwYXNzd29yZFxuY29uc3QgU2VuZFJlc2V0UGFzc3dvcmRFbWFpbFF1ZXJ5OiBhbnkgPSBncWxgXG4gIHF1ZXJ5IFNlbmRSZXNldFBhc3N3b3JkRW1haWwoJHVzZXJuYW1lOiBTdHJpbmchLCAkbG9jYWxlOiBTdHJpbmchKSB7XG4gICAgc2VuZFJlc2V0UGFzc3dvcmRFbWFpbCh1c2VybmFtZTogJHVzZXJuYW1lLCBsb2NhbGU6ICRsb2NhbGUpXG4gIH1cbmA7XG5cbmNvbnN0IFJlc2V0QWNjb3VudFB1YmtleVF1ZXJ5OiBhbnkgPSBncWxgXG4gIHF1ZXJ5IFJlc2V0UHVia2V5KCR1c2VybmFtZTogU3RyaW5nISwgJHRva2VuOiBTdHJpbmchKSB7XG4gICAgZGF0YTogcmVzZXRBY2NvdW50UHVia2V5KHVzZXJuYW1lOiAkdXNlcm5hbWUsIHRva2VuOiAkdG9rZW4pXG4gIH1cbmA7XG5cbmNvbnN0IFVwZGF0ZVB1YktleVF1ZXJ5OiBhbnkgPSBncWxgXG4gIHF1ZXJ5IFVwZGF0ZUFjY291bnRQdWJrZXkoJHB1YmtleTogU3RyaW5nISkge1xuICAgIGRhdGE6IHVwZGF0ZUFjY291bnRQdWJrZXkocHVia2V5OiAkcHVia2V5KVxuICB9XG5gO1xuXG4vLyBBdXRoZW50aWNhdGlvbiAgcXVlcnlcbmNvbnN0IEF1dGhRdWVyeTogYW55ID0gZ3FsYFxuICBxdWVyeSBBdXRoKCR0b2tlbjogU3RyaW5nKSB7XG4gICAgYXV0aGVudGljYXRlKHRva2VuOiAkdG9rZW4pXG4gIH1cbmA7XG5cbi8vIE5ldyBhdXRoIGNoYWxsZW5nZSBxdWVyeVxuY29uc3QgQXV0aENoYWxsZW5nZVF1ZXJ5OiBhbnkgPSBncWxgXG4gIHF1ZXJ5IEF1dGhDaGFsbGVuZ2Uge1xuICAgIGF1dGhDaGFsbGVuZ2Uge1xuICAgICAgY2hhbGxlbmdlXG4gICAgICBwdWJrZXlcbiAgICAgIHNpZ25hdHVyZVxuICAgIH1cbiAgfVxuYDtcblxuaW50ZXJmYWNlIElBdXRoQ2hhbGxlbmdlIHtcbiAgcHVia2V5OiBzdHJpbmc7XG4gIGNoYWxsZW5nZTogc3RyaW5nO1xuICBzaWduYXR1cmU6IHN0cmluZztcbn1cblxuY29uc3QgQWNjb3VudFN1YnNjcmlwdGlvbnM6IEJhc2VFbnRpdHlHcmFwaHFsU3Vic2NyaXB0aW9ucyA9IHtcbiAgbGlzdGVuQ2hhbmdlczogZ3FsYFxuICAgIHN1YnNjcmlwdGlvbiB1cGRhdGVBY2NvdW50KCRpbnRlcnZhbDogSW50KSB7XG4gICAgICBkYXRhOiB1cGRhdGVBY2NvdW50KGludGVydmFsOiAkaW50ZXJ2YWwpIHtcbiAgICAgICAgaWRcbiAgICAgICAgdXBkYXRlRGF0ZVxuICAgICAgfVxuICAgIH1cbiAgYCxcbn07XG5cbmV4cG9ydCBpbnRlcmZhY2UgQWNjb3VudFdhdGNoT3B0aW9ucyB7XG4gIGludGVydmFsSW5TZWNvbmRzPzogbnVtYmVyO1xufVxuXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46ICdyb290JywgZGVwczogW0VOVklST05NRU5UXSB9KVxuZXhwb3J0IGNsYXNzIEFjY291bnRTZXJ2aWNlIGV4dGVuZHMgQmFzZUdyYXBocWxTZXJ2aWNlPEFjY291bnQsIGFueSwgbnVtYmVyLCBBY2NvdW50PiB7XG4gIC8vIEdyYXBoUUwgb3BlcmF0aW9ucyAoY2FuIGJlIG92ZXJyaWRkZW4gdXNpbmcgQVBQX0FDQ09VTlRfU0VSVklDRV9PUFRJT05TKVxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgcXVlcmllczogQWNjb3VudFNlcnZpY2VHcmFwaHFsUXVlcmllcztcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IG11dGF0aW9uczogQWNjb3VudFNlcnZpY2VHcmFwaHFsTXV0YXRpb25zO1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgc3Vic2NyaXB0aW9uczogQWNjb3VudFNlcnZpY2VHcmFwaHFsU3Vic2NyaXB0aW9ucztcbiAgb25Mb2dpbiA9IG5ldyBTdWJqZWN0PEFjY291bnQ+KCk7XG4gIG9uV2lsbExvZ291dCA9IG5ldyBTdWJqZWN0PEFjY291bnQ+KCk7XG4gIG9uTG9nb3V0ID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcbiAgb25DaGFuZ2UgPSBuZXcgU3ViamVjdDxBY2NvdW50PigpO1xuICBvbkF1dGhUb2tlbkNoYW5nZSA9IG5ldyBCZWhhdmlvclN1YmplY3Q8c3RyaW5nIHwgdW5kZWZpbmVkPih1bmRlZmluZWQpO1xuICBvbkF1dGhCYXNpY0NoYW5nZSA9IG5ldyBCZWhhdmlvclN1YmplY3Q8c3RyaW5nIHwgdW5kZWZpbmVkPih1bmRlZmluZWQpO1xuXG4gIHByaXZhdGUgX2xpc3RlbkNoYW5nZXNTdWJzY3JpcHRpb246IFN1YnNjcmlwdGlvbiA9IG51bGw7XG4gIHByaXZhdGUgcmVhZG9ubHkgX2VuYWJsZUxpc3RlbkNoYW5nZXM6IGJvb2xlYW47XG4gIHByaXZhdGUgcmVhZG9ubHkgX2xpc3RlbkludGVydmFsSW5TZWNvbmRzOiBudW1iZXI7XG4gIHByaXZhdGUgX2NhY2hlOiBBY2NvdW50RGV0YWlscyA9IHtcbiAgICBsb2FkZWQ6IGZhbHNlLFxuICAgIGtleXBhaXI6IG51bGwsXG4gICAgYXV0aFRva2VuOiBudWxsLFxuICAgIGF1dGhCYXNpYzogbnVsbCxcbiAgICBwdWJrZXk6IG51bGwsXG4gICAgbWFpblByb2ZpbGU6IG51bGwsXG4gICAgcGVyc29uOiBudWxsLFxuICAgIGRlcGFydG1lbnQ6IG51bGwsXG4gIH07XG4gIHByaXZhdGUgcmVhZG9ubHkgX29wdGlvbkRlZnM6IEZvcm1GaWVsZERlZmluaXRpb25bXTtcbiAgcHJpdmF0ZSByZWFkb25seSBfcmVtb3RlTG9jYWxTZXR0aW5nc0tleXM6IChrZXlvZiBMb2NhbFNldHRpbmdzIHwgc3RyaW5nKVtdO1xuICBwcml2YXRlIF8kYWRkaXRpb25hbEZpZWxkcyA9IG5ldyBCZWhhdmlvclN1YmplY3Q8Rm9ybUZpZWxkRGVmaW5pdGlvbltdPihbXSk7XG4gIHByaXZhdGUgX3Rva2VuVHlwZSQgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PEF1dGhUb2tlblR5cGU+KHVuZGVmaW5lZCk7XG4gIHByaXZhdGUgX2FwaVRva2VuRW5hYmxlZCA9IHRydWU7XG5cbiAgZ2V0IGFjY291bnQoKTogQWNjb3VudCB7XG4gICAgcmV0dXJuIHRoaXMuX2NhY2hlLmxvYWRlZCA/IHRoaXMuX2RhdGEgOiB1bmRlZmluZWQ7XG4gIH1cblxuICBnZXQgYWNjb3VudCQoKTogT2JzZXJ2YWJsZTxBY2NvdW50PiB7XG4gICAgcmV0dXJuIHRoaXMub25DaGFuZ2UuYXNPYnNlcnZhYmxlKCkucGlwZShzdGFydFdpdGgodGhpcy5hY2NvdW50KSk7XG4gIH1cblxuICBnZXQgcGVyc29uKCk6IFBlcnNvbiB7XG4gICAgaWYgKHRoaXMuX2NhY2hlLmxvYWRlZCAmJiAhdGhpcy5fY2FjaGUucGVyc29uKSB7XG4gICAgICB0aGlzLl9jYWNoZS5wZXJzb24gPSB0aGlzLl9jYWNoZS5sb2FkZWQgPyB0aGlzLl9kYXRhLmFzUGVyc29uKCkgOiB1bmRlZmluZWQ7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9jYWNoZS5wZXJzb247XG4gIH1cblxuICBnZXQgcGVyc29uJCgpOiBPYnNlcnZhYmxlPFBlcnNvbj4ge1xuICAgIHJldHVybiB0aGlzLm9uQ2hhbmdlLmFzT2JzZXJ2YWJsZSgpLnBpcGUoXG4gICAgICBtYXAoKCkgPT4gdGhpcy5wZXJzb24pLFxuICAgICAgc3RhcnRXaXRoKHRoaXMucGVyc29uKVxuICAgICk7XG4gIH1cblxuICBnZXQgZGVwYXJ0bWVudCgpOiBEZXBhcnRtZW50IHtcbiAgICBpZiAodGhpcy5fY2FjaGUubG9hZGVkICYmICF0aGlzLl9jYWNoZS5kZXBhcnRtZW50KSB7XG4gICAgICB0aGlzLl9jYWNoZS5kZXBhcnRtZW50ID0gdGhpcy5fY2FjaGUubG9hZGVkID8gdGhpcy5fZGF0YS5hc1BlcnNvbigpLmRlcGFydG1lbnQgOiB1bmRlZmluZWQ7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9jYWNoZS5kZXBhcnRtZW50O1xuICB9XG5cbiAgZ2V0IHRva2VuVHlwZSgpOiBBdXRoVG9rZW5UeXBlIHtcbiAgICByZXR1cm4gdGhpcy5fdG9rZW5UeXBlJC52YWx1ZTtcbiAgfVxuXG4gIHNldCB0b2tlblR5cGUodmFsdWU6IEF1dGhUb2tlblR5cGUpIHtcbiAgICBpZiAodGhpcy5fdG9rZW5UeXBlJC52YWx1ZSAhPT0gdmFsdWUpIHtcbiAgICAgIGNvbnNvbGUuaW5mbygnW2FjY291bnRdIFVzaW5nIGF1dGhlbnRpY2F0aW9uIHRva2VuIHR5cGU6ICcgKyB2YWx1ZSk7XG4gICAgICB0aGlzLl90b2tlblR5cGUkLm5leHQodmFsdWUpO1xuICAgICAgLy8gUmVzZXQgdmFsdWVzXG4gICAgICB0aGlzLl9jYWNoZS5hdXRoVG9rZW4gPSB1bmRlZmluZWQ7XG4gICAgICB0aGlzLm9uQXV0aFRva2VuQ2hhbmdlLm5leHQodW5kZWZpbmVkKTtcbiAgICAgIHRoaXMuX2NhY2hlLmF1dGhCYXNpYyA9IHVuZGVmaW5lZDtcbiAgICAgIHRoaXMub25BdXRoQmFzaWNDaGFuZ2UubmV4dCh1bmRlZmluZWQpO1xuICAgIH1cbiAgfVxuXG4gIGdldCBhcGlUb2tlbkVuYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuX2FwaVRva2VuRW5hYmxlZDtcbiAgfVxuXG4gIHNldCBhcGlUb2tlbkVuYWJsZWQodmFsdWU6IGJvb2xlYW4pIHtcbiAgICB0aGlzLl9hcGlUb2tlbkVuYWJsZWQgPSB2YWx1ZTtcbiAgfVxuXG4gIGdldCByZW1vdGVMb2NhbFNldHRpbmdzS2V5cygpIHtcbiAgICByZXR1cm4gdGhpcy5fcmVtb3RlTG9jYWxTZXR0aW5nc0tleXM7XG4gIH1cblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcm90ZWN0ZWQgbmV0d29yazogTmV0d29ya1NlcnZpY2UsXG4gICAgcHJvdGVjdGVkIGdyYXBocWw6IEdyYXBocWxTZXJ2aWNlLFxuICAgIHByb3RlY3RlZCBzZXR0aW5nczogTG9jYWxTZXR0aW5nc1NlcnZpY2UsXG4gICAgcHJvdGVjdGVkIHN0b3JhZ2U6IFN0b3JhZ2UsXG4gICAgcHJvdGVjdGVkIGZpbGU6IEZpbGVTZXJ2aWNlLFxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgdHJhbnNsYXRlOiBUcmFuc2xhdGVTZXJ2aWNlLFxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgdG9hc3RDb250cm9sbGVyOiBUb2FzdENvbnRyb2xsZXIsXG4gICAgQEluamVjdChFTlZJUk9OTUVOVCkgcHJvdGVjdGVkIGVudmlyb25tZW50OiBFbnZpcm9ubWVudCxcbiAgICBAT3B0aW9uYWwoKSBASW5qZWN0KEFQUF9VU0VSX1NFVFRJTkdTX09QVElPTlMpIG9wdGlvbnM6IFVzZXJTZXR0aW5nc09wdGlvbnMsXG4gICAgQE9wdGlvbmFsKCkgQEluamVjdChBUFBfQUNDT1VOVF9TRVJWSUNFX09QVElPTlMpIGFjY291bnRPcHRpb25zPzogQWNjb3VudFNlcnZpY2VPcHRpb25zXG4gICkge1xuICAgIHN1cGVyKGdyYXBocWwsIGVudmlyb25tZW50KTtcbiAgICAvLyBJbml0aWFsaXplIEdyYXBoUUwgb3BlcmF0aW9ucyAoYWxsb3cgb3ZlcnJpZGVzKVxuICAgIHRoaXMucXVlcmllcyA9IHsgLi4uKFF1ZXJpZXMgYXMgQWNjb3VudFNlcnZpY2VHcmFwaHFsUXVlcmllcyksIC4uLihhY2NvdW50T3B0aW9ucz8ucXVlcmllcyB8fCB7fSkgfSBhcyBBY2NvdW50U2VydmljZUdyYXBocWxRdWVyaWVzO1xuICAgIHRoaXMubXV0YXRpb25zID0geyAuLi4oTXV0YXRpb25zIGFzIEFjY291bnRTZXJ2aWNlR3JhcGhxbE11dGF0aW9ucyksIC4uLihhY2NvdW50T3B0aW9ucz8ubXV0YXRpb25zIHx8IHt9KSB9IGFzIEFjY291bnRTZXJ2aWNlR3JhcGhxbE11dGF0aW9ucztcbiAgICB0aGlzLnN1YnNjcmlwdGlvbnMgPSB7IC4uLihBY2NvdW50U3Vic2NyaXB0aW9ucyBhcyBBY2NvdW50U2VydmljZUdyYXBocWxTdWJzY3JpcHRpb25zKSwgLi4uKGFjY291bnRPcHRpb25zPy5zdWJzY3JpcHRpb25zIHx8IHt9KSB9IGFzIEFjY291bnRTZXJ2aWNlR3JhcGhxbFN1YnNjcmlwdGlvbnM7XG4gICAgdGhpcy5fZW5hYmxlTGlzdGVuQ2hhbmdlcyA9ICFlbnZpcm9ubWVudC5hY2NvdW50IHx8IGVudmlyb25tZW50LmFjY291bnQuZW5hYmxlTGlzdGVuQ2hhbmdlcyAhPT0gZmFsc2U7IC8vIFRydWUgYnkgZGVmYXVsdFxuICAgIHRoaXMuX2xpc3RlbkludGVydmFsSW5TZWNvbmRzID0gdG9OdW1iZXIoZW52aXJvbm1lbnQuYWNjb3VudCAmJiBlbnZpcm9ubWVudC5hY2NvdW50Lmxpc3RlbkludGVydmFsSW5TZWNvbmRzLCAwKTsgLy8gbm8gdGltZXIgYnkgZGVmYXVsdFxuXG4gICAgdGhpcy5fZGVidWcgPSAhZW52aXJvbm1lbnQucHJvZHVjdGlvbjtcbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoJ1thY2NvdW50LXNlcnZpY2VdIENyZWF0aW5nIHNlcnZpY2UnKTtcbiAgICB0aGlzLl9vcHRpb25EZWZzID0gT2JqZWN0LnZhbHVlcyhvcHRpb25zPy5vcHRpb25zIHx8IHt9KTtcbiAgICB0aGlzLl9yZW1vdGVMb2NhbFNldHRpbmdzS2V5cyA9IG9wdGlvbnM/LnJlbW90ZUxvY2FsU2V0dGluZ3NLZXlzIHx8IFtdO1xuXG4gICAgdGhpcy5yZXNldERhdGEoKTtcblxuICAgIC8vIFNlbmQgYXV0aCB0b2tlbiB0byB0aGUgZ3JhcGhxbCBsYXllciwgd2hlbiBjaGFuZ2VkXG4gICAgdGhpcy5vbkF1dGhUb2tlbkNoYW5nZS5zdWJzY3JpYmUoKHRva2VuKSA9PiB0aGlzLm5ldHdvcmsuc2V0QXV0aFRva2VuKHRva2VuKSk7XG4gICAgdGhpcy5vbkF1dGhCYXNpY0NoYW5nZS5zdWJzY3JpYmUoKGJhc2ljKSA9PiB0aGlzLm5ldHdvcmsuc2V0QXV0aEJhc2ljKGJhc2ljKSk7XG5cbiAgICAvLyBGb3JjZSBuZXR3b3JrIHRvIHdhaXQgYWNjb3VudCBzZXJ2aWNlLCBhZnRlciBnZXR0aW5nIGNvbm5lY3Rpb24gdG8gdGhlIHBlZXJcbiAgICB0aGlzLm5ldHdvcmsub24oJ2JlZm9yZVRyeU9ubGluZUZpbmlzaCcsIGFzeW5jIChvbmxpbmUpID0+IHtcbiAgICAgIC8vIElmIG9ubGluZSwgd2FpdCBhIGZ1bGwgcmVzdGFydCwgYmVjYXVzZSBpdCBjYW4gZm9yY2Ugb2ZmbGluZSBtb2RlXG4gICAgICBpZiAob25saW5lICYmICghdGhpcy5zdGFydGVkIHx8IHRoaXMuaXNMb2dpbigpKSkge1xuICAgICAgICBjb25zb2xlLmRlYnVnKCdbYWNjb3VudF0gRm9yY2UgbmV0d29ya1NlcnZpY2UudHJ5T25saW5lKCkgdG8gd2FpdCwgdGhhdCBncmFwaHFsIGFuZCBhY2NvdW50IHNlcnZpY2UgaXMgcmVzdGFydGVkLi4uJyk7XG4gICAgICAgIGF3YWl0IHNsZWVwKDUwMCk7IC8vIHdhaXQgZ3JhcGhxbCBzZXJ2aWNlIHRvIGJlIHJlc3RhcnRlZFxuICAgICAgICBpZiAoIXRoaXMuc3RhcnRlZCkgYXdhaXQgdGhpcy5yZWFkeSgpO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgYXN5bmMgbmdPblN0YXJ0KCk6IFByb21pc2U8QWNjb3VudD4ge1xuICAgIGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgIHRoaXMuc2V0dGluZ3MucmVhZHkoKSxcbiAgICAgIC8vIFdhaXQgdG9rZW4gdHlwZSB0byBiZSBzZXRcbiAgICAgIGZpcnN0Tm90TmlsUHJvbWlzZSh0aGlzLl90b2tlblR5cGUkLCB7IHN0b3A6IHRoaXMuc3RvcFN1YmplY3QgfSksXG4gICAgXSk7XG5cbiAgICAvLyBMaXN0ZW4gZ3JhcGhxbCBzdGFydCAob3IgcmVzdGFydClcbiAgICB0aGlzLnJlZ2lzdGVyU3Vic2NyaXB0aW9uKFxuICAgICAgdGhpcy5ncmFwaHFsLnN0b3BTdWJqZWN0LnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgIGlmICh0aGlzLnN0YXJ0ZWQgJiYgdGhpcy5pc0xvZ2luKCkpIHtcbiAgICAgICAgICBjb25zb2xlLmRlYnVnKCdbYWNjb3VudF0gUmVzdGFydGluZywgdG8gcmV0cnkgdG8gYXV0aGVudGljYXRlLi4uJyk7XG4gICAgICAgICAgdGhpcy5yZXN0YXJ0KCk7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgKTtcblxuICAgIGF3YWl0IHRoaXMucmVzdG9yZUxvY2FsbHkoKTtcbiAgICBhd2FpdCB0aGlzLmxpc3RlblNldHRpbmdzKCk7XG5cbiAgICByZXR1cm4gdGhpcy5fZGF0YTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyBuZ09uU3RvcCgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBoYWRBdXRoVG9rZW4gPSB0aGlzLl9jYWNoZS5hdXRoVG9rZW4gJiYgdHJ1ZTtcbiAgICBjb25zdCBoYWRBdXRoQmFzaWMgPSB0aGlzLl9jYWNoZS5hdXRoQmFzaWMgJiYgdHJ1ZTtcbiAgICBjb25zdCBoYWRBdXRoID0gaGFkQXV0aFRva2VuIHx8IGhhZEF1dGhCYXNpYztcbiAgICB0aGlzLnJlc2V0RGF0YSgpO1xuICAgIGlmIChoYWRBdXRoKSB7XG4gICAgICB0aGlzLm9uTG9nb3V0Lm5leHQoKTtcbiAgICAgIHRoaXMub25DaGFuZ2UubmV4dCh1bmRlZmluZWQpO1xuICAgICAgaWYgKGhhZEF1dGhUb2tlbikgdGhpcy5vbkF1dGhUb2tlbkNoYW5nZS5uZXh0KHVuZGVmaW5lZCk7XG4gICAgICBpZiAoaGFkQXV0aEJhc2ljKSB0aGlzLm9uQXV0aEJhc2ljQ2hhbmdlLm5leHQodW5kZWZpbmVkKTtcbiAgICB9XG4gIH1cblxuICBpc0xvZ2luKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiAhISh0aGlzLl9jYWNoZS5wdWJrZXkgJiYgdGhpcy5fY2FjaGUubG9hZGVkKTtcbiAgfVxuXG4gIGlzQXV0aCgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gISEodGhpcy5fY2FjaGUucHVia2V5ICYmIHRoaXMuX2NhY2hlLmtleXBhaXIgJiYgdGhpcy5fY2FjaGUua2V5cGFpci5zZWNyZXRLZXkpO1xuICB9XG5cbiAgaGFzTWluUHJvZmlsZSh1c2VyUHJvZmlsZTogVXNlclByb2ZpbGVMYWJlbCk6IGJvb2xlYW4ge1xuICAgIC8vIHNob3VsZCBiZSBsb2dpbiwgYW5kIHN0YXR1cyBFTkFCTEUgb3IgVEVNUE9SQVJZXG4gICAgaWYgKCF0aGlzLl9kYXRhIHx8ICF0aGlzLl9kYXRhLnB1YmtleSB8fCAodGhpcy5fZGF0YS5zdGF0dXNJZCAhPT0gU3RhdHVzSWRzLkVOQUJMRSAmJiB0aGlzLl9kYXRhLnN0YXR1c0lkICE9PSBTdGF0dXNJZHMuVEVNUE9SQVJZKSkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gUGVyc29uVXRpbHMuaGFzVXBwZXJPckVxdWFsc1Byb2ZpbGUodGhpcy5fZGF0YS5wcm9maWxlcywgdXNlclByb2ZpbGUpO1xuICB9XG5cbiAgaGFzRXhhY3RQcm9maWxlKGxhYmVsOiBVc2VyUHJvZmlsZUxhYmVsKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIEFjY291bnRVdGlscy5oYXNFeGFjdFByb2ZpbGUodGhpcy5fZGF0YSwgbGFiZWwpO1xuICB9XG5cbiAgaGFzUHJvZmlsZUFuZElzRW5hYmxlKHVzZXJQcm9maWxlOiBVc2VyUHJvZmlsZUxhYmVsKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIEFjY291bnRVdGlscy5oYXNQcm9maWxlQW5kSXNFbmFibGUodGhpcy5fZGF0YSwgdXNlclByb2ZpbGUpO1xuICB9XG5cbiAgaXNBZG1pbigpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5oYXNQcm9maWxlQW5kSXNFbmFibGUoJ0FETUlOJyk7XG4gIH1cblxuICBpc1N1cGVydmlzb3IoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuaGFzUHJvZmlsZUFuZElzRW5hYmxlKCdTVVBFUlZJU09SJyk7XG4gIH1cblxuICBpc1VzZXIoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuaGFzUHJvZmlsZUFuZElzRW5hYmxlKCdVU0VSJyk7XG4gIH1cblxuICAvKipcbiAgICogQGRlcHJlY2F0ZWRcbiAgICogQHBhcmFtIG1vZGVcbiAgICovXG4gIGlzVXNhZ2VNb2RlKG1vZGU6IFVzYWdlTW9kZSk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnNldHRpbmdzLmlzVXNhZ2VNb2RlKG1vZGUpO1xuICB9XG5cbiAgaXNPbmx5R3Vlc3QoKTogYm9vbGVhbiB7XG4gICAgLy8gU2hvdWxkIGJlIGxvZ2luLCBhbmQgc3RhdHVzIEVOQUJMRSBvciBURU1QT1JBUllcbiAgICBpZiAoIXRoaXMuX2RhdGEgfHwgIXRoaXMuX2RhdGEucHVia2V5IHx8ICh0aGlzLl9kYXRhLnN0YXR1c0lkICE9PSBTdGF0dXNJZHMuRU5BQkxFICYmIHRoaXMuX2RhdGEuc3RhdHVzSWQgIT09IFN0YXR1c0lkcy5URU1QT1JBUlkpKSByZXR1cm4gZmFsc2U7XG4gICAgLy8gUHJvZmlsZSBsZXNzIHRoZW4gdXNlclxuICAgIHJldHVybiAhUGVyc29uVXRpbHMuaGFzVXBwZXJPckVxdWFsc1Byb2ZpbGUodGhpcy5fZGF0YS5wcm9maWxlcywgJ1VTRVInKTtcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0gcmVjb3JkZXJEZXBhcnRtZW50XG4gICAqIEBkZXByZWNhdGVkIHVzZSBQcm9ncmFtUmVmU2VydmljZS5jYW5Vc2VyV3JpdGVFbnRpdHkoKSBpbnN0ZWFkXG4gICAqL1xuICBjYW5Vc2VyV3JpdGVEYXRhRm9yRGVwYXJ0bWVudChyZWNvcmRlckRlcGFydG1lbnQ6IFJlZmVyZW50aWFsIHwgYW55KTogYm9vbGVhbiB7XG4gICAgaWYgKFJlZmVyZW50aWFsVXRpbHMuaXNFbXB0eShyZWNvcmRlckRlcGFydG1lbnQpKSB7XG4gICAgICByZXR1cm4gdGhpcy5pc0FkbWluKCk7XG4gICAgfVxuXG4gICAgLy8gU2hvdWxkIGJlIGxvZ2luLCBhbmQgc3RhdHVzIEVOQUJMRVxuICAgIGlmICghdGhpcy5fZGF0YSB8fCAhdGhpcy5fZGF0YS5wdWJrZXkgfHwgdGhpcy5fZGF0YS5zdGF0dXNJZCAhPT0gU3RhdHVzSWRzLkVOQUJMRSkgcmV0dXJuIGZhbHNlO1xuXG4gICAgaWYgKCF0aGlzLl9kYXRhLmRlcGFydG1lbnQgfHwgIXRoaXMuX2RhdGEuZGVwYXJ0bWVudC5pZCkge1xuICAgICAgY29uc29sZS53YXJuKCdVc2VyIGFjY291bnQgaGFzIG5vIGRlcGFydG1lbnQgISBVbmFibGUgdG8gY2hlY2sgd3JpdGUgcmlnaHQgYWdhaW5zdCByZWNvcmRlckRlcGFydG1lbnQnKTtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICAvLyBTYW1lIHJlY29yZGVyIGRlcGFydG1lbnQ6IE9LLCB1c2VyIGNhbiB3cml0ZVxuICAgIGlmICh0aGlzLl9kYXRhLmRlcGFydG1lbnQuaWQgPT09IHJlY29yZGVyRGVwYXJ0bWVudC5pZCkgcmV0dXJuIHRydWU7XG5cbiAgICAvLyBDYW5ub3Qgd3JpdGVcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICBhc3luYyByZWdpc3RlcihkYXRhOiBSZWdpc3RlckRhdGEpOiBQcm9taXNlPEFjY291bnQ+IHtcbiAgICBpZiAodGhpcy5pc0xvZ2luKCkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignVXNlciBhbHJlYWR5IGxvZ2luLiBQbGVhc2UgbG9nb3V0IGJlZm9yZSByZWdpc3Rlci4nKTtcbiAgICB9XG4gICAgaWYgKCFkYXRhLnVzZXJuYW1lIHx8ICFkYXRhLnBhc3N3b3JkKSB0aHJvdyBuZXcgRXJyb3IoJ01pc3NpbmcgcmVxdWlyZWQgdXNlcm5hbWUgb3IgcGFzc3dvcmQnKTtcblxuICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZygnW2FjY291bnRdIFJlZ2lzdGVyIG5ldyB1c2VyIGFjY291bnQuLi4nLCBkYXRhLmFjY291bnQpO1xuICAgIHRoaXMuX2NhY2hlLmxvYWRlZCA9IGZhbHNlO1xuICAgIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3Qga2V5cGFpciA9IGF3YWl0IENyeXB0b1NlcnZpY2Uuc2NyeXB0S2V5cGFpcihkYXRhLnVzZXJuYW1lLCBkYXRhLnBhc3N3b3JkKTtcbiAgICAgIGRhdGEuYWNjb3VudC5wdWJrZXkgPSBCYXNlNTguZW5jb2RlKGtleXBhaXIucHVibGljS2V5KTtcblxuICAgICAgLy8gRGVmYXVsdCB2YWx1ZXNcbiAgICAgIGRhdGEuYWNjb3VudC5zZXR0aW5ncyA9IGRhdGEuYWNjb3VudC5zZXR0aW5ncyB8fCBuZXcgVXNlclNldHRpbmdzKCk7XG4gICAgICBkYXRhLmFjY291bnQuc2V0dGluZ3MubWVyZ2UodGhpcy5zZXR0aW5ncy5zZXR0aW5ncywgdGhpcy5fcmVtb3RlTG9jYWxTZXR0aW5nc0tleXMsIHRydWUpO1xuICAgICAgZGF0YS5hY2NvdW50LmRlcGFydG1lbnQuaWQgPSBkYXRhLmFjY291bnQuZGVwYXJ0bWVudC5pZCB8fCB0aGlzLmVudmlyb25tZW50LmRlZmF1bHREZXBhcnRtZW50SWQ7XG5cbiAgICAgIHRoaXMuX2NhY2hlLmtleXBhaXIgPSBrZXlwYWlyO1xuICAgICAgY29uc3QgYWNjb3VudCA9IGF3YWl0IHRoaXMuc2F2ZVJlbW90ZWx5KGRhdGEuYWNjb3VudCwga2V5cGFpcik7XG5cbiAgICAgIC8vIERlZmF1bHQgdmFsdWVzXG4gICAgICBhY2NvdW50LmF2YXRhciA9IGFjY291bnQuYXZhdGFyIHx8IHRoaXMuZW52aXJvbm1lbnQuYmFzZVVybCArIFBlcnNvblV0aWxzLkRFRkFVTFRfQVZBVEFSX0lNQUdFO1xuICAgICAgdGhpcy5fY2FjaGUubWFpblByb2ZpbGUgPSBQZXJzb25VdGlscy5nZXRNYWluUHJvZmlsZShhY2NvdW50LnByb2ZpbGVzKTtcblxuICAgICAgdGhpcy5fZGF0YSA9IGFjY291bnQ7XG4gICAgICB0aGlzLl9jYWNoZS5wdWJrZXkgPSBhY2NvdW50LnB1YmtleTtcblxuICAgICAgLy8gVHJ5IHRvIGF1dGggb24gcG9kXG4gICAgICBhd2FpdCB0aGlzLmF1dGhlbnRpY2F0ZShkYXRhKTtcblxuICAgICAgdGhpcy5fY2FjaGUubG9hZGVkID0gdHJ1ZTtcblxuICAgICAgYXdhaXQgdGhpcy5zYXZlTG9jYWxseSgpO1xuXG4gICAgICBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gQWNjb3VudCBzdWNjZXNzZnVsbHkgcmVnaXN0ZXJlZCBpbiAke0RhdGUubm93KCkgLSBub3d9bXNgKTtcblxuICAgICAgLy8gRW1pdCBldmVudHNcbiAgICAgIHRoaXMub25Mb2dpbi5uZXh0KHRoaXMuX2RhdGEpO1xuICAgICAgdGhpcy5vbkNoYW5nZS5uZXh0KHRoaXMuX2RhdGEpO1xuXG4gICAgICAvLyBMaXN0ZW4gcmVtb3RlIGNoYW5nZXNcbiAgICAgIGlmICh0aGlzLl9lbmFibGVMaXN0ZW5DaGFuZ2VzKSB0aGlzLnN0YXJ0TGlzdGVuUmVtb3RlQ2hhbmdlcygpO1xuXG4gICAgICByZXR1cm4gdGhpcy5fZGF0YTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgY29uc29sZS5lcnJvcigoZXJyb3IgJiYgZXJyb3IubWVzc2FnZSkgfHwgZXJyb3IpO1xuICAgICAgdGhpcy5yZXNldERhdGEoKTtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbiAgfVxuXG4gIGFzeW5jIGF1dGhlbnRpY2F0ZShkYXRhPzogQXV0aERhdGEpIHtcbiAgICAvLyBXYWl0IHRoZSBhdXRoIHRva2VuLCB0aGVuIGNvbnRpbnVlXG4gICAgY29uc3QgdG9rZW5UeXBlID0gYXdhaXQgZmlyc3ROb3ROaWxQcm9taXNlKHRoaXMuX3Rva2VuVHlwZSQpO1xuXG4gICAgLy8gQmFzaWMgYXV0aFxuICAgIGlmICh0b2tlblR5cGUgPT09ICdiYXNpYycgfHwgdG9rZW5UeXBlID09PSAnYmFzaWMtYW5kLXRva2VuJykge1xuICAgICAgLy8gR2VuZXJhdGUgdGhlIGF1dGhCYXNpYywgaWYgdXNlZFxuICAgICAgaWYgKCF0aGlzLl9jYWNoZS5hdXRoQmFzaWMpIHtcbiAgICAgICAgLy8gU2tpcCBpZiB0b2tlbiBhbHJlYWR5IHByb3ZpZGVkXG4gICAgICAgIGlmICghKHRoaXMuX2NhY2hlLmF1dGhUb2tlbiAmJiB0b2tlblR5cGUgPT09ICdiYXNpYy1hbmQtdG9rZW4nKSkge1xuICAgICAgICAgIGlmICghZGF0YSB8fCAhZGF0YS51c2VybmFtZSB8fCAhZGF0YS5wYXNzd29yZCkgdGhyb3cgbmV3IEVycm9yKCdNaXNzaW5nIHVzZXJuYW1lIGFuZCBwYXNzd29yZCcpO1xuICAgICAgICAgIHRoaXMuX2NhY2hlLmF1dGhCYXNpYyA9IENyeXB0b1NlcnZpY2UuZW5jb2RlQmFzZTY0KGAke2RhdGEudXNlcm5hbWV9OiR7ZGF0YS5wYXNzd29yZH1gKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgdGhpcy5vbkF1dGhCYXNpY0NoYW5nZS5uZXh0KHRoaXMuX2NhY2hlLmF1dGhCYXNpYyk7XG4gICAgfVxuXG4gICAgLy8gR2VuZXJhdGUgdGhlIGF1dGhUb2tlbiwgaWYgdXNlZFxuICAgIGlmICh0b2tlblR5cGUgPT09ICd0b2tlbicgfHwgdG9rZW5UeXBlID09PSAnYmFzaWMtYW5kLXRva2VuJykge1xuICAgICAgdHJ5IHtcbiAgICAgICAgdGhpcy5fY2FjaGUuYXV0aFRva2VuID0gYXdhaXQgdGhpcy5hdXRoZW50aWNhdGVBbmRHZXRUb2tlbih0aGlzLl9jYWNoZS5hdXRoVG9rZW4pO1xuICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgLy8gTmV2ZXIgYXV0aGVudGljYXRlLCBvciBub3QgcmVhZHkgZm9yIG9mZmxpbmUgbW9kZSA9PiBleGl0XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgICB0aGlzLnJlc2V0RGF0YSgpO1xuICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBGb3JnZXQgYXV0aEJhc2ljLCB0byBzd2l0Y2ggdG8gYXV0aFRva2VuXG4gICAgaWYgKHRva2VuVHlwZSA9PT0gJ2Jhc2ljLWFuZC10b2tlbicpIHtcbiAgICAgIHRoaXMuX2NhY2hlLmF1dGhCYXNpYyA9IHVuZGVmaW5lZDtcbiAgICAgIHRoaXMub25BdXRoQmFzaWNDaGFuZ2UubmV4dCh1bmRlZmluZWQpO1xuICAgIH1cbiAgfVxuXG4gIGFzeW5jIGxvZ2luKGRhdGE6IEF1dGhEYXRhKTogUHJvbWlzZTxBY2NvdW50PiB7XG4gICAgaWYgKCFkYXRhIHx8ICFkYXRhLnVzZXJuYW1lIHx8ICFkYXRhLnBhc3N3b3JkKSB0aHJvdyBuZXcgRXJyb3IoJ01pc3NpbmcgcmVxdWlyZWQgdXNlcm5hbWUgb3IgcGFzc3dvcmQnKTtcblxuICAgIGNvbnNvbGUuZGVidWcoJ1thY2NvdW50XSBMb2dpbi4uLicpO1xuXG4gICAgbGV0IGtleXBhaXI7XG4gICAgdHJ5IHtcbiAgICAgIGtleXBhaXIgPSBhd2FpdCBDcnlwdG9TZXJ2aWNlLnNjcnlwdEtleXBhaXIoZGF0YS51c2VybmFtZSwgZGF0YS5wYXNzd29yZCk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgdGhpcy5yZXNldERhdGEoKTtcbiAgICAgIHRocm93IHsgY29kZTogRXJyb3JDb2Rlcy5VTktOT1dOX0VSUk9SLCBtZXNzYWdlOiAnRVJST1IuU0NSWVBUX0VSUk9SJyB9O1xuICAgIH1cblxuICAgIC8vIFN0b3JlIHB1YmtleStrZXlwYWlyXG4gICAgdGhpcy5fY2FjaGUucHVia2V5ID0gQmFzZTU4LmVuY29kZShrZXlwYWlyLnB1YmxpY0tleSk7XG4gICAgdGhpcy5fY2FjaGUua2V5cGFpciA9IGtleXBhaXI7XG5cbiAgICAvLyBUcnkgdG8gbG9hZCBwcmV2aW91cyB0b2tlblxuICAgIGxldCBwcmV2aW91c1Rva2VuOiBzdHJpbmcgPSBhd2FpdCB0aGlzLnN0b3JhZ2UuZ2V0KFRPS0VOX1NUT1JBR0VfS0VZKTtcbiAgICBwcmV2aW91c1Rva2VuID0gKHByZXZpb3VzVG9rZW4gJiYgcHJldmlvdXNUb2tlbi5zdGFydHNXaXRoKHRoaXMuX2NhY2hlLnB1YmtleSkgJiYgcHJldmlvdXNUb2tlbikgfHwgbnVsbDtcblxuICAgIC8vIE9mZmxpbmUgbW9kZVxuICAgIGNvbnN0IG9mZmxpbmUgPSB0aGlzLnNldHRpbmdzLmhhc09mZmxpbmVGZWF0dXJlKCkgJiYgKHRoaXMubmV0d29yay5vZmZsaW5lIHx8IGRhdGEub2ZmbGluZSA9PT0gdHJ1ZSk7XG4gICAgaWYgKG9mZmxpbmUpIHtcbiAgICAgIHRoaXMuX2NhY2hlLmF1dGhUb2tlbiA9IHByZXZpb3VzVG9rZW47XG5cbiAgICAgIC8vIE1ha2Ugc3VyZSBuZXR3b3JrIGlmIHNldCBhcyBvZmZsaW5lXG4gICAgICB0aGlzLm5ldHdvcmsuc2V0Rm9yY2VPZmZsaW5lKHRydWUsIHsgc2hvd1RvYXN0OiBmYWxzZSB9KTtcbiAgICAgIGNvbnNvbGUuaW5mbyhgW2FjY291bnRdIExvZ2luIFtPS10ge3B1YmtleTogJHt0aGlzLl9jYWNoZS5wdWJrZXkuc3Vic3RyKDAsIDgpfX0sIHtvZmZsaW5lOiB0cnVlfWApO1xuICAgIH1cblxuICAgIC8vIE9ubGluZSBtb2RlOiB0cnkgdG8gYXV0aCBvbiBwb2RcbiAgICBlbHNlIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IHRoaXMuYXV0aGVudGljYXRlKGRhdGEpO1xuICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgLy8gTmV2ZXIgYXV0aGVudGljYXRlLCBvciBub3QgcmVhZHkgZm9yIG9mZmxpbmUgbW9kZSA9PiBleGl0XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgICB0aGlzLnJlc2V0RGF0YSgpO1xuICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBMb2FkIGFjY291bnQgZGF0YVxuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLmxvYWREYXRhKHsgb2ZmbGluZSwgZmV0Y2hQb2xpY3k6ICduZXR3b3JrLW9ubHknIH0pO1xuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgLy8gSWYgYWNjb3VudCBub3QgZm91bmQsIGNoZWNrIGlmIGVtYWlsIGlzIHZhbGlkXG4gICAgICBpZiAoZXJyICYmICtlcnIuY29kZSA9PT0gRXJyb3JDb2Rlcy5MT0FEX0FDQ09VTlRfRVJST1IpIHtcbiAgICAgICAgLy8gQ2hlY2sgZW1haWwgZXhpc3RzXG4gICAgICAgIGlmIChkYXRhLnVzZXJuYW1lLmluZGV4T2YoJ0AnKSAhPT0gLTEpIHtcbiAgICAgICAgICBsZXQgaXNFbWFpbEV4aXN0cztcbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgaXNFbWFpbEV4aXN0cyA9IGF3YWl0IHRoaXMuaXNFbWFpbEV4aXN0cyhkYXRhLnVzZXJuYW1lKTtcbiAgICAgICAgICB9IGNhdGNoIChvdGhlckVycm9yKSB7XG4gICAgICAgICAgICB0aHJvdyBlcnI7IC8vIHJlc2VuZCB0aGUgZmlyc3QgZXJyb3JcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBFbWFpbCBub3QgZXhpc3RzIChubyBhY2NvdW50KVxuICAgICAgICAgIGlmICghaXNFbWFpbEV4aXN0cykge1xuICAgICAgICAgICAgdGhyb3cgeyBjb2RlOiBFcnJvckNvZGVzLlVOS05PV05fQUNDT1VOVF9FTUFJTCwgbWVzc2FnZTogJ0VSUk9SLlVOS05PV05fQUNDT1VOVF9FTUFJTCcgfTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBFbWFpbCBleGlzdHMsIHNvIGVycm9yID0gJ2JhZCBwYXNzd29yZCdcbiAgICAgICAgdGhyb3cgeyBjb2RlOiBFcnJvckNvZGVzLkJBRF9QQVNTV09SRCwgbWVzc2FnZTogJ0VSUk9SLkJBRF9QQVNTV09SRCcgfTtcbiAgICAgIH1cblxuICAgICAgdGhyb3cgZXJyOyAvLyByZXNlbmQgdGhlIGZpcnN0IGVycm9yXG4gICAgfVxuXG4gICAgdHJ5IHtcbiAgICAgIC8vIFN0b3JlIHRvIGxvY2FsIHN0b3JhZ2VcbiAgICAgIGF3YWl0IHRoaXMuc2F2ZUxvY2FsbHkoKTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgY29uc29sZS5lcnJvcihlcnJvcik7XG4gICAgICB0aGlzLnJlc2V0RGF0YSgpO1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfVxuXG4gICAgLy8gRW1pdCBldmVudCB0byBvYnNlcnZlcnNcbiAgICB0aGlzLm9uTG9naW4ubmV4dCh0aGlzLl9kYXRhKTtcbiAgICB0aGlzLm9uQ2hhbmdlLm5leHQodGhpcy5fZGF0YSk7XG5cbiAgICBpZiAodGhpcy5fZW5hYmxlTGlzdGVuQ2hhbmdlcykgdGhpcy5zdGFydExpc3RlblJlbW90ZUNoYW5nZXMoKTtcblxuICAgIHJldHVybiB0aGlzLl9kYXRhO1xuICB9XG5cbiAgYXN5bmMgcmVsb2FkKG9wdHM/OiB7IHNob3dUb2FzdD86IGJvb2xlYW4gfSk6IFByb21pc2U8QWNjb3VudD4ge1xuICAgIGlmICghdGhpcy5fY2FjaGUucHVia2V5KSB0aHJvdyBuZXcgRXJyb3IoJ1VzZXIgbm90IGxvZ2dlZCcpO1xuICAgIGlmICh0aGlzLm5ldHdvcmsub2ZmbGluZSkgdGhyb3cgbmV3IEVycm9yKCdDYW5ub3QgY2hlY2sgYWNjb3VudCBpbiBvZmZsaW5lIG1vZGUnKTtcblxuICAgIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG4gICAgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFJlbG9hZGluZyBhY2NvdW50Li4uYCk7XG4gICAgY29uc3Qgd2FzTG9naW4gPSB0aGlzLmlzTG9naW4oKTtcblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLmxvYWREYXRhKCk7XG4gICAgICBhd2FpdCB0aGlzLnNhdmVMb2NhbGx5KCk7XG5cbiAgICAgIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBSZWxvYWRpbmcgYWNjb3VudCBbT0tdIGluICR7RGF0ZS5ub3coKSAtIG5vd31tc2ApO1xuXG4gICAgICAvLyBFbWl0IGxvZ2luIGV2ZW50IHRvIHN1YnNjcmliZXJzXG4gICAgICB0aGlzLm9uTG9naW4ubmV4dCh0aGlzLl9kYXRhKTtcbiAgICAgIHRoaXMub25DaGFuZ2UubmV4dCh0aGlzLl9kYXRhKTtcblxuICAgICAgLy8gRGlzcGxheSB0b2FzdCAod2l0aG91dCBhd2FpdCwgYmVjYXVzZSBub3QgbmVlZCB0byB3YWl0IHRvYXN0IGNsb3NlIGV2ZW50KVxuICAgICAgaWYgKCFvcHRzIHx8IG9wdHMuc2hvd1RvYXN0ICE9PSBmYWxzZSkge1xuICAgICAgICB0aGlzLnNob3dUb2FzdCh7IG1lc3NhZ2U6ICdBQ0NPVU5ULklORk8uUkVMT0FERUQnLCB0eXBlOiAnaW5mbycgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB0aGlzLl9kYXRhO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAvLyBDYW5ub3QgcmVsb2FkIGJ1dCB3YXMgbG9naW46IGZvcmNlIHRvIGxvZ291dFxuICAgICAgaWYgKHdhc0xvZ2luICYmICF0aGlzLmlzTG9naW4oKSkge1xuICAgICAgICBjb25zb2xlLmVycm9yKGBbYWNjb3VudF0gUmVsb2FkaW5nIGFjY291bnQgZmFpbGVkLiBXaWxsIGZvcmNlIGxvZ291dC4uLmAsIGVycm9yKTtcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dvdXQoKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRocm93IGVycm9yO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGUgb3IgdXBkYXRlIGFuIHVzZXIgYWNjb3VudCwgdG8gdGhlIHJlbW90ZSBzdG9yYWdlXG4gICAqXG4gICAqIEBwYXJhbSBhY2NvdW50XG4gICAqL1xuICBhc3luYyBzYXZlKGFjY291bnQ6IEFjY291bnQpOiBQcm9taXNlPEFjY291bnQ+IHtcbiAgICBpZiAoIXRoaXMuX2NhY2hlLnB1YmtleSkgcmV0dXJuIFByb21pc2UucmVqZWN0KCdVc2VyIG5vdCBsb2dnZWQnKTtcbiAgICBpZiAodGhpcy5fY2FjaGUucHVia2V5ICE9PSBhY2NvdW50LnB1YmtleSkgcmV0dXJuIFByb21pc2UucmVqZWN0KCdOb3QgdXNlciBhY2NvdW50Jyk7XG5cbiAgICBhY2NvdW50ID0gYXdhaXQgdGhpcy5zYXZlUmVtb3RlbHkoYWNjb3VudCwgdGhpcy5fY2FjaGUua2V5cGFpcik7XG5cbiAgICAvLyBTZXQgZGVmYXVsdHNcbiAgICBhY2NvdW50LmF2YXRhciA9IGFjY291bnQuYXZhdGFyIHx8IHRoaXMuZW52aXJvbm1lbnQuYmFzZVVybCArIERFRkFVTFRfQVZBVEFSX0lNQUdFO1xuICAgIHRoaXMuX2NhY2hlLm1haW5Qcm9maWxlID0gUGVyc29uVXRpbHMuZ2V0TWFpblByb2ZpbGUoYWNjb3VudC5wcm9maWxlcyk7XG5cbiAgICB0aGlzLl9kYXRhID0gYWNjb3VudDtcblxuICAgIC8vIFVwZGF0ZSBjYWNoZVxuICAgIHRoaXMuX2NhY2hlLnBlcnNvbiA9IGFjY291bnQuYXNQZXJzb24oKTtcbiAgICB0aGlzLl9jYWNoZS5kZXBhcnRtZW50ID0gdGhpcy5fY2FjaGUucGVyc29uLmRlcGFydG1lbnQ7XG5cbiAgICB0aGlzLl9jYWNoZS5sb2FkZWQgPSB0cnVlO1xuXG4gICAgLy8gU2F2ZSBsb2NhbGx5IChpbiBzdG9yYWdlKVxuICAgIGF3YWl0IHRoaXMuc2F2ZUxvY2FsbHkoKTtcblxuICAgIC8vIFNlbmQgZXZlbnRcbiAgICB0aGlzLm9uTG9naW4ubmV4dCh0aGlzLl9kYXRhKTtcbiAgICB0aGlzLm9uQ2hhbmdlLm5leHQodGhpcy5fZGF0YSk7XG5cbiAgICByZXR1cm4gdGhpcy5fZGF0YTtcbiAgfVxuXG4gIGFzeW5jIGxvZ291dCgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBoYWRBdXRoVG9rZW4gPSB0aGlzLl9jYWNoZS5hdXRoVG9rZW4gJiYgdHJ1ZTtcbiAgICBjb25zdCBoYWRBdXRoQmFzaWMgPSB0aGlzLl9jYWNoZS5hdXRoQmFzaWMgJiYgdHJ1ZTtcbiAgICBjb25zdCBwdWJrZXkgPSB0aGlzLl9jYWNoZSAmJiB0aGlzLl9jYWNoZS5wdWJrZXk7XG5cbiAgICAvLyBOb3RpZnkgb2JzZXJ2ZXJzXG4gICAgdGhpcy5vbldpbGxMb2dvdXQubmV4dCh0aGlzLmFjY291bnQpO1xuXG4gICAgdGhpcy5yZXNldERhdGEoKTtcblxuICAgIGlmICghdGhpcy5zZXR0aW5ncy5oYXNPZmZsaW5lRmVhdHVyZSgpKSB7XG4gICAgICAvLyBSZW1vdmUgYWxsIGRhdGEgZnJvbSB0aGUgbG9jYWwgc3RvcmFnZVxuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoW1xuICAgICAgICB0aGlzLnN0b3JhZ2UucmVtb3ZlKFBVQktFWV9TVE9SQUdFX0tFWSksXG4gICAgICAgIHRoaXMuc3RvcmFnZS5yZW1vdmUoVE9LRU5fU1RPUkFHRV9LRVkpLFxuICAgICAgICB0aGlzLnN0b3JhZ2UucmVtb3ZlKEFDQ09VTlRfU1RPUkFHRV9LRVkpLFxuICAgICAgICAocHVia2V5ICYmIHRoaXMuc3RvcmFnZS5yZW1vdmUoQUNDT1VOVF9TVE9SQUdFX0tFWSArICcjJyArIHB1YmtleSkpIHx8IFByb21pc2UucmVzb2x2ZSgpLFxuICAgICAgICB0aGlzLnN0b3JhZ2UucmVtb3ZlKFNFQ0tFWV9TVE9SQUdFX0tFWSksXG4gICAgICBdKTtcbiAgICB9XG5cbiAgICAvLyBPZmZsaW5lIGZlYXR1cmVzIGVuYWJsZTogbmVlZCB0byBrZWVwIHNvbWUgZGF0YVxuICAgIGVsc2Uge1xuICAgICAgLy8gQWx3YXlzIHJlbW92ZSBvbmx5IHNlY3JldCBrZXlcbiAgICAgIC8vIEJ1dCBrZWVwOlxuICAgICAgLy8gLSBhY2NvdW50IGJ5IHB1YmtleVxuICAgICAgLy8gLSBhdXRoIHRva2VuXG4gICAgICBhd2FpdCBQcm9taXNlLmFsbChbdGhpcy5zdG9yYWdlLnJlbW92ZShQVUJLRVlfU1RPUkFHRV9LRVkpLCB0aGlzLnN0b3JhZ2UucmVtb3ZlKEFDQ09VTlRfU1RPUkFHRV9LRVkpLCB0aGlzLnN0b3JhZ2UucmVtb3ZlKFNFQ0tFWV9TVE9SQUdFX0tFWSldKTtcbiAgICB9XG5cbiAgICAvLyBDbGVhbiBwYWdlIGhpc3RvcnksIGluIGxvY2FsIHNldHRpbmdzXG4gICAgYXdhaXQgdGhpcy5zZXR0aW5ncy5jbGVhclBhZ2VIaXN0b3J5KCk7XG5cbiAgICAvLyBOb3RpZnkgb2JzZXJ2ZXJzXG4gICAgdGhpcy5vbkxvZ291dC5uZXh0KCk7XG4gICAgaWYgKGhhZEF1dGhUb2tlbikgdGhpcy5vbkF1dGhUb2tlbkNoYW5nZS5uZXh0KHVuZGVmaW5lZCk7XG4gICAgaWYgKGhhZEF1dGhCYXNpYykgdGhpcy5vbkF1dGhCYXNpY0NoYW5nZS5uZXh0KHVuZGVmaW5lZCk7XG4gICAgdGhpcy5vbkNoYW5nZS5uZXh0KHVuZGVmaW5lZCk7XG4gIH1cblxuICAvKipcbiAgICogTG9hZCBhIGFjY291bnRcbiAgICpcbiAgICogQHBhcmFtIG9wdHNcbiAgICovXG4gIGFzeW5jIGxvYWQob3B0cz86IHsgb2ZmbGluZT86IGJvb2xlYW47IGZldGNoUG9saWN5PzogRmV0Y2hQb2xpY3k7IHF1ZXJ5PzogYW55IH0pOiBQcm9taXNlPEFjY291bnQgfCB1bmRlZmluZWQ+IHtcbiAgICBjb25zdCBub3cgPSB0aGlzLl9kZWJ1ZyAmJiBEYXRlLm5vdygpO1xuICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIExvYWRpbmcgYWNjb3VudC4uLmApO1xuICAgIGxldCBqc29uOiBhbnk7XG5cbiAgICAvLyBMb2FkIGxvY2FsbHlcbiAgICBjb25zdCBvZmZsaW5lID1cbiAgICAgICh0aGlzLm5ldHdvcmsub2ZmbGluZSAmJiAoIW9wdHMgfHwgKG9wdHMuZmV0Y2hQb2xpY3kgIT09ICduZXR3b3JrLW9ubHknICYmIG9wdHMuZmV0Y2hQb2xpY3kgIT09ICduby1jYWNoZScpKSkgfHxcbiAgICAgIChvcHRzICYmIG9wdHMub2ZmbGluZSA9PT0gdHJ1ZSk7XG4gICAgaWYgKG9mZmxpbmUpIHtcbiAgICAgIGpzb24gPSBhd2FpdCB0aGlzLnN0b3JhZ2UuZ2V0KEFDQ09VTlRfU1RPUkFHRV9LRVkpO1xuICAgICAganNvbiA9IChqc29uICYmIHR5cGVvZiBqc29uID09PSAnc3RyaW5nJyAmJiBKU09OLnBhcnNlKGpzb24pKSB8fCBqc29uO1xuICAgICAganNvbiA9IChqc29uICYmIHRoaXMuX2NhY2hlLnB1YmtleSAmJiBqc29uLnB1YmtleSA9PT0gdGhpcy5fY2FjaGUucHVia2V5ICYmIGpzb24pIHx8IG51bGw7XG4gICAgICBpZiAoIWpzb24gJiYgdGhpcy5fY2FjaGUucHVia2V5KSB7XG4gICAgICAgIGpzb24gPSBhd2FpdCB0aGlzLnN0b3JhZ2UuZ2V0KEFDQ09VTlRfU1RPUkFHRV9LRVkgKyAnIycgKyB0aGlzLl9jYWNoZS5wdWJrZXkpO1xuICAgICAgICBqc29uID0gKGpzb24gJiYgdHlwZW9mIGpzb24gPT09ICdzdHJpbmcnICYmIEpTT04ucGFyc2UoanNvbikpIHx8IGpzb247XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gTG9hZCByZW1vdGVseVxuICAgIGVsc2Uge1xuICAgICAgY29uc3QgcXVlcnkgPSBvcHRzPy5xdWVyeSB8fCAodGhpcy5hcGlUb2tlbkVuYWJsZWQgPyB0aGlzLnF1ZXJpZXMubG9hZFdpdGhUb2tlbnMgOiB0aGlzLnF1ZXJpZXMubG9hZCk7XG4gICAgICBjb25zdCB7IGRhdGEgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5xdWVyeTx7IGRhdGE6IGFueSB9Pih7XG4gICAgICAgIHF1ZXJ5LFxuICAgICAgICBlcnJvcjogeyBjb2RlOiBFcnJvckNvZGVzLkxPQURfQUNDT1VOVF9FUlJPUiwgbWVzc2FnZTogJ0VSUk9SLkxPQURfQUNDT1VOVF9FUlJPUicgfSxcbiAgICAgICAgZmV0Y2hQb2xpY3k6IChvcHRzICYmIG9wdHMuZmV0Y2hQb2xpY3kpIHx8ICduby1jYWNoZScgfHwgdW5kZWZpbmVkLFxuICAgICAgfSk7XG4gICAgICBqc29uID0gZGF0YTtcbiAgICB9XG5cbiAgICBpZiAoanNvbikge1xuICAgICAgY29uc3QgYWNjb3VudCA9IEFjY291bnQuZnJvbU9iamVjdChqc29uKTtcblxuICAgICAgLy8gQ2xlYW4gcmVtb3RlXG4gICAgICBpZiAoYWNjb3VudC5zZXR0aW5ncyAmJiB0aGlzLl9yZW1vdGVMb2NhbFNldHRpbmdzS2V5cykge1xuICAgICAgICBhY2NvdW50LnNldHRpbmdzLm1lcmdlKGFjY291bnQuc2V0dGluZ3MuYXNMb2NhbFNldHRpbmdzKCksIHRoaXMuX3JlbW90ZUxvY2FsU2V0dGluZ3NLZXlzKTtcbiAgICAgIH1cblxuICAgICAgaWYgKHRoaXMuX2RlYnVnKSBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gQWNjb3VudCBsb2FkZWQgaW4gJHtEYXRlLm5vdygpIC0gbm93fW1zYCwgYWNjb3VudCk7XG4gICAgICByZXR1cm4gYWNjb3VudDtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc29sZS53YXJuKGBbYWNjb3VudF0gQWNjb3VudCBub3QgZm91bmQgIWApO1xuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICB9XG4gIH1cblxuICBhc3luYyBnZW5lcmF0ZVRva2VuKGZsYWdzOiBudW1iZXIpOiBQcm9taXNlPHN0cmluZyB8IHVuZGVmaW5lZD4ge1xuICAgIGNvbnN0IG5vdyA9IHRoaXMuX2RlYnVnICYmIERhdGUubm93KCk7XG4gICAgaWYgKHRoaXMuX2RlYnVnKSBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gR2VuZXJhdGUgdG9rZW4uLi5gKTtcblxuICAgIGNvbnN0IGF1dGhDaGFsbGVuZ2UgPSBhd2FpdCB0aGlzLmdldEF1dGhDaGFsbGVuZ2UoKTtcbiAgICAvLyBBZGQgRmxhZ3MgdG8gY2hhbGxlbmdlXG4gICAgY29uc3QgY2hhbGxlbmdlID0gYCR7YXV0aENoYWxsZW5nZS5jaGFsbGVuZ2V9OiR7ZmxhZ3N9YDtcbiAgICBjb25zdCBzaWduYXR1cmUgPSBhd2FpdCBDcnlwdG9TZXJ2aWNlLnNpZ24oY2hhbGxlbmdlLCB0aGlzLl9jYWNoZS5rZXlwYWlyKTtcbiAgICBjb25zdCBuZXdUb2tlbiA9IGAke3RoaXMuX2NhY2hlLnB1YmtleX06JHtjaGFsbGVuZ2V9fCR7c2lnbmF0dXJlfWA7XG5cbiAgICBpZiAobmV3VG9rZW4pIHtcbiAgICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFRva2VuIGdlbmVyYXRlZCBpbiAke0RhdGUubm93KCkgLSBub3d9bXNgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc29sZS53YXJuKGBbYWNjb3VudF0gVG9rZW4gZ2VuZXJhdGlvbiBmYWlsZWQgIWApO1xuICAgIH1cblxuICAgIHJldHVybiBuZXdUb2tlbjtcbiAgfVxuXG4gIC8qKlxuICAgKiBDaGVjayBpZiBlbWFpbCBpcyBhdmFpbGFibGUgZm9yIG5ldyBhY2NvdW50IHJlZ2lzdHJhdGlvbi5cbiAgICogVGhyb3cgYW4gZXJyb3IgaWYgbm90IGF2YWlsYWJsZVxuICAgKlxuICAgKiBAcGFyYW0gZW1haWxcbiAgICovXG4gIGFzeW5jIGNoZWNrRW1haWxBdmFpbGFibGUoZW1haWw6IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGlzRW1haWxFeGlzdHMgPSBhd2FpdCB0aGlzLmlzRW1haWxFeGlzdHMoZW1haWwpO1xuICAgIGlmIChpc0VtYWlsRXhpc3RzKSB7XG4gICAgICB0aHJvdyB7IGNvZGU6IEVycm9yQ29kZXMuRU1BSUxfQUxSRUFEWV9SRUdJU1RFUkVELCBtZXNzYWdlOiAnRVJST1IuRU1BSUxfQUxSRUFEWV9SRUdJU1RFUkVEJyB9O1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDaGVjayBpZiBlbWFpbCBpcyBleGlzdHMgaW4gc2VydmVyLlxuICAgKlxuICAgKiBAcGFyYW0gZW1haWxcbiAgICovXG4gIGFzeW5jIGlzRW1haWxFeGlzdHMoZW1haWw6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZygnW2FjY291bnRdIENoZWNraW5nIGlmIHsnICsgZW1haWwgKyAnfSBleGlzdHMuLi4nKTtcblxuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCB0aGlzLmdyYXBocWwucXVlcnk8eyBpc0VtYWlsRXhpc3RzOiBib29sZWFuIH0sIElzRW1haWxFeGlzdHNWYXJpYWJsZXM+KHtcbiAgICAgIHF1ZXJ5OiBJc0VtYWlsRXhpc3RzUXVlcnksXG4gICAgICB2YXJpYWJsZXM6IHtcbiAgICAgICAgZW1haWwsXG4gICAgICAgIGhhc2g6IHVuZGVmaW5lZCxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoJ1thY2NvdW50XSBFbWFpbCBleGlzdDogJyArIChkYXRhICYmIGRhdGEuaXNFbWFpbEV4aXN0cykpO1xuXG4gICAgcmV0dXJuIGRhdGEgJiYgZGF0YS5pc0VtYWlsRXhpc3RzO1xuICB9XG5cbiAgYXN5bmMgc2VuZENvbmZpcm1hdGlvbkVtYWlsKGVtYWlsOiBzdHJpbmcsIGxvY2FsZT86IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGxvY2FsZSA9IGxvY2FsZSB8fCB0aGlzLnNldHRpbmdzLmxvY2FsZTtcbiAgICBjb25zb2xlLmRlYnVnKCdbYWNjb3VudF0gU2VuZGluZyBjb25maXJtYXRpb24gZW1haWwgdG8geycgKyBlbWFpbCArICd9IHdpdGggbG9jYWxlIHsnICsgbG9jYWxlICsgJ30uLi4nKTtcblxuICAgIGNvbnN0IHsgZGF0YSB9ID0gYXdhaXQgdGhpcy5ncmFwaHFsLnF1ZXJ5PHsgZGF0YTogYm9vbGVhbiB9Pih7XG4gICAgICBxdWVyeTogU2VuZENvbmZpcm1FbWFpbFF1ZXJ5LFxuICAgICAgdmFyaWFibGVzOiB7XG4gICAgICAgIGVtYWlsLFxuICAgICAgICBsb2NhbGUsXG4gICAgICB9LFxuICAgICAgZXJyb3I6IHtcbiAgICAgICAgY29kZTogRXJyb3JDb2Rlcy5TRU5UX0NPTkZJUk1BVElPTl9FTUFJTF9GQUlMRUQsXG4gICAgICAgIG1lc3NhZ2U6ICdFUlJPUi5TRU5UX0FDQ09VTlRfQ09ORklSTUFUSU9OX0VNQUlMX0ZBSUxFRCcsXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgcmV0dXJuIGRhdGE7XG4gIH1cblxuICBhc3luYyBjb25maXJtRW1haWwoZW1haWw6IHN0cmluZywgY29kZTogc3RyaW5nKTogUHJvbWlzZTxib29sZWFuPiB7XG4gICAgY29uc29sZS5kZWJ1ZygnW2FjY291bnRdIFNlbmRpbmcgY29uZmlybSByZXF1ZXN0IGZvciBlbWFpbCB7JyArIGVtYWlsICsgJ30gd2l0aCBjb2RlIHsnICsgY29kZSArICd9Li4uJyk7XG5cbiAgICBjb25zdCB7IGRhdGEgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5xdWVyeTx7IGRhdGE6IGJvb2xlYW4gfT4oe1xuICAgICAgcXVlcnk6IENvbmZpcm1FbWFpbFF1ZXJ5LFxuICAgICAgdmFyaWFibGVzOiB7XG4gICAgICAgIGVtYWlsLFxuICAgICAgICBjb2RlLFxuICAgICAgfSxcbiAgICAgIGVycm9yOiB7XG4gICAgICAgIGNvZGU6IEVycm9yQ29kZXMuQ09ORklSTV9FTUFJTF9GQUlMRUQsXG4gICAgICAgIG1lc3NhZ2U6ICdFUlJPUi5DT05GSVJNX0FDQ09VTlRfRU1BSUxfRkFJTEVEJyxcbiAgICAgIH0sXG4gICAgfSk7XG4gICAgcmV0dXJuIGRhdGE7XG4gIH1cblxuICBhc3luYyBzZW5kUmVzZXRQYXNzd29yZEVtYWlsKHVzZXJuYW1lOiBzdHJpbmcsIGxvY2FsZT86IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGxvY2FsZSA9IGxvY2FsZSB8fCB0aGlzLnNldHRpbmdzLmxvY2FsZTtcblxuICAgIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBTZW5kaW5nIGNoYW5nZSBwYXNzd29yZCBlbWFpbCB0byB7JHt1c2VybmFtZX19IHdpdGggbG9jYWxlIHske2xvY2FsZX19Li4uYCk7XG5cbiAgICBjb25zdCB7IGRhdGE6IGlzT2sgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5xdWVyeTx7IGRhdGE6IGJvb2xlYW4gfT4oe1xuICAgICAgcXVlcnk6IFNlbmRSZXNldFBhc3N3b3JkRW1haWxRdWVyeSxcbiAgICAgIHZhcmlhYmxlczoge1xuICAgICAgICB1c2VybmFtZSxcbiAgICAgICAgbG9jYWxlLFxuICAgICAgfSxcbiAgICAgIGVycm9yOiB7XG4gICAgICAgIGNvZGU6IEVycm9yQ29kZXMuQVVUSF9TRU5EX1JFU0VUX1BBU1NXT1JEX0VNQUlMX0VSUk9SLFxuICAgICAgICBtZXNzYWdlOiAnQUNDT1VOVC5FUlJPUi5TRU5EX1JFU0VUX1BBU1NXT1JEX0VNQUlMX0VSUk9SJyxcbiAgICAgIH0sXG4gICAgICBmZXRjaFBvbGljeTogJ25vLWNhY2hlJyxcbiAgICB9KTtcbiAgICByZXR1cm4gaXNPaztcbiAgfVxuXG4gIGFzeW5jIHVwZGF0ZVB1YmtleShkYXRhOiBBdXRoRGF0YSk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGlmICghZGF0YS51c2VybmFtZSB8fCAhZGF0YS5wYXNzd29yZCkgdGhyb3cgbmV3IEVycm9yKCdNaXNzaW5nIHJlcXVpcmVkIHVzZXJuYW1lIG9yIHBhc3N3b3JkJyk7XG4gICAgaWYgKCF0aGlzLmlzTG9naW4oKSkgdGhyb3cgbmV3IEVycm9yKCdTaG91bGQgYmUgbG9naW4nKTtcblxuICAgIGNvbnN0IGtleXBhaXIgPSBhd2FpdCBDcnlwdG9TZXJ2aWNlLnNjcnlwdEtleXBhaXIoZGF0YS51c2VybmFtZSwgZGF0YS5wYXNzd29yZCk7XG4gICAgY29uc3QgcHVia2V5ID0gQmFzZTU4LmVuY29kZShrZXlwYWlyLnB1YmxpY0tleSk7XG5cbiAgICBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gVXBkYXRpbmcgYWNjb3VudCBwdWJrZXkgdG8gJHtwdWJrZXl9IC4uLmApO1xuXG4gICAgY29uc3QgeyBkYXRhOiByZXMgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5xdWVyeTx7IGRhdGE6IGJvb2xlYW4gfT4oe1xuICAgICAgcXVlcnk6IFVwZGF0ZVB1YktleVF1ZXJ5LFxuICAgICAgdmFyaWFibGVzOiB7IHB1YmtleSB9LFxuICAgICAgZXJyb3I6IHtcbiAgICAgICAgY29kZTogRXJyb3JDb2Rlcy5BVVRIX1VQREFURV9QVUJLRVlfRVJST1IsXG4gICAgICAgIG1lc3NhZ2U6ICdBQ0NPVU5ULkVSUk9SLlVQREFURV9QVUJLRVlfRVJST1InLFxuICAgICAgfSxcbiAgICAgIGZldGNoUG9saWN5OiAnbm8tY2FjaGUnLFxuICAgIH0pO1xuXG4gICAgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFVwZGF0aW5nIGFjY291bnQgcHVia2V5IHJlc3BvbnNlYCwgcmVzKTtcblxuICAgIGlmIChyZXMpIHtcbiAgICAgIGF3YWl0IHRoaXMubG9nb3V0KCk7XG4gICAgICBhd2FpdCB0aGlzLmxvZ2luKGRhdGEpO1xuICAgIH1cblxuICAgIHJldHVybiByZXM7XG4gIH1cblxuICBhc3luYyByZXNldFB1YmtleShkYXRhOiBBdXRoRGF0YSwgdG9rZW46IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGlmICh0aGlzLmlzTG9naW4oKSkge1xuICAgICAgYXdhaXQgdGhpcy5sb2dvdXQoKTtcbiAgICB9XG5cbiAgICAvLyBQYXJzZSBjaGFsbGVuZ2VcbiAgICBjb25zdCBjaGFsbGVuZ2UgPSB0b2tlbi5zcGxpdCgnfCcsIDIpWzBdLnNwbGl0KCc6JywgMilbMV07XG4gICAgY29uc3Qga2V5cGFpciA9IGF3YWl0IENyeXB0b1NlcnZpY2Uuc2NyeXB0S2V5cGFpcihkYXRhLnVzZXJuYW1lLCBkYXRhLnBhc3N3b3JkKTtcbiAgICBjb25zdCBwdWJrZXkgPSBCYXNlNTguZW5jb2RlKGtleXBhaXIucHVibGljS2V5KTtcbiAgICBjb25zdCBzaWduYXR1cmUgPSBhd2FpdCBDcnlwdG9TZXJ2aWNlLnNpZ24oY2hhbGxlbmdlLCBrZXlwYWlyKTtcbiAgICBjb25zdCBuZXdUb2tlbiA9IGAke3B1YmtleX06JHtjaGFsbGVuZ2V9fCR7c2lnbmF0dXJlfWA7XG5cbiAgICBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gUmVzZXR0aW5nIGFjY291bnQgcHVia2V5LCB1c2luZyB0b2tlbjogYCwgbmV3VG9rZW4pO1xuXG4gICAgY29uc3QgeyBkYXRhOiByZXMgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5xdWVyeTx7IGRhdGE6IGJvb2xlYW4gfT4oe1xuICAgICAgcXVlcnk6IFJlc2V0QWNjb3VudFB1YmtleVF1ZXJ5LFxuICAgICAgdmFyaWFibGVzOiB7IHRva2VuOiBuZXdUb2tlbiwgdXNlcm5hbWU6IGRhdGEudXNlcm5hbWUgfSxcbiAgICAgIGVycm9yOiB7XG4gICAgICAgIGNvZGU6IEVycm9yQ29kZXMuQVVUSF9SRVNFVF9QVUJLRVlfRVJST1IsXG4gICAgICAgIG1lc3NhZ2U6ICdBQ0NPVU5ULkVSUk9SLlJFU0VUX1BVQktFWV9FUlJPUicsXG4gICAgICB9LFxuICAgICAgZmV0Y2hQb2xpY3k6ICduby1jYWNoZScsXG4gICAgfSk7XG5cbiAgICAvLyBMb2dpblxuICAgIGlmIChyZXMpIHtcbiAgICAgIGF3YWl0IHRoaXMubG9naW4oZGF0YSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHJlcztcbiAgfVxuXG4gIGxpc3RlbkNoYW5nZXMob3B0cz86IEFjY291bnRXYXRjaE9wdGlvbnMpOiBTdWJzY3JpcHRpb24ge1xuICAgIGlmICh0aGlzLl9lbmFibGVMaXN0ZW5DaGFuZ2VzKSByZXR1cm4gbmV3IFN1YnNjcmlwdGlvbigpOyAvLyBBbHJlYWR5IHN0YXJ0ZWQ6IHNraXBcblxuICAgIHJldHVybiB0aGlzLnN0YXJ0TGlzdGVuUmVtb3RlQ2hhbmdlcyhvcHRzKTtcbiAgfVxuXG4gIGdldCBhZGRpdGlvbmFsRmllbGRzKCk6IEZvcm1GaWVsZERlZmluaXRpb25bXSB7XG4gICAgcmV0dXJuIHRoaXMuXyRhZGRpdGlvbmFsRmllbGRzLmdldFZhbHVlKCk7XG4gIH1cblxuICAvKipcbiAgICogQGRlcHJlY2F0ZWQgVXNlIGFkZGl0aW9uYWxGaWVsZHMkIGluc3RlYWRcbiAgICovXG4gIGdldCAkYWRkaXRpb25hbEZpZWxkcygpOiBPYnNlcnZhYmxlPEZvcm1GaWVsZERlZmluaXRpb25bXT4ge1xuICAgIHJldHVybiB0aGlzLl8kYWRkaXRpb25hbEZpZWxkcy5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIGdldCBhZGRpdGlvbmFsRmllbGRzJCgpOiBPYnNlcnZhYmxlPEZvcm1GaWVsZERlZmluaXRpb25bXT4ge1xuICAgIHJldHVybiB0aGlzLl8kYWRkaXRpb25hbEZpZWxkcy5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIGdldEFkZGl0aW9uYWxGaWVsZChrZXk6IHN0cmluZyk6IEZvcm1GaWVsZERlZmluaXRpb24gfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLl8kYWRkaXRpb25hbEZpZWxkcy5nZXRWYWx1ZSgpLmZpbmQoKGYpID0+IGYua2V5ID09PSBrZXkpO1xuICB9XG5cbiAgcmVnaXN0ZXJBZGRpdGlvbmFsRmllbGQoLi4uZmllbGRzOiBGb3JtRmllbGREZWZpbml0aW9uW10pIHtcbiAgICBjb25zdCBvbGRGaWVsZHMgPSB0aGlzLl8kYWRkaXRpb25hbEZpZWxkcy5nZXRWYWx1ZSgpIHx8IFtdO1xuICAgIGNvbnN0IG5ld0ZpZWxkcyA9IGZpZWxkcz8ucmVkdWNlKChyZXMsIGZpZWxkKSA9PiB7XG4gICAgICBpZiAocmVzLnNvbWUoKGYpID0+IGYua2V5ID09PSBmaWVsZC5rZXkpKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgQWRkaXRpb25hbCBhY2NvdW50IGZpZWxkIHtrZXk6ICR7ZmllbGQua2V5fX0gYWxyZWFkeSBkZWZpbmUuYCk7XG4gICAgICB9XG4gICAgICAvL2lmICh0aGlzLl9kZWJ1ZylcbiAgICAgIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBSZWdpc3RlciBhZGRpdGlvbmFsIGFjY291bnQncyBmaWVsZCB7a2V5OiAke2ZpZWxkLmtleX19YCk7XG4gICAgICByZXR1cm4gcmVzLmNvbmNhdChmaWVsZCk7XG4gICAgfSwgb2xkRmllbGRzKTtcblxuICAgIHRoaXMuXyRhZGRpdGlvbmFsRmllbGRzLm5leHQobmV3RmllbGRzKTtcbiAgfVxuXG4gIGdldCBvcHRpb25EZWZzKCk6IEZvcm1GaWVsZERlZmluaXRpb25bXSB7XG4gICAgcmV0dXJuIHRoaXMuX29wdGlvbkRlZnM7XG4gIH1cblxuICByZWdpc3Rlck9wdGlvbihkZWY6IEZvcm1GaWVsZERlZmluaXRpb24pIHtcbiAgICBpZiAodGhpcy5fb3B0aW9uRGVmcy5maW5kSW5kZXgoKGYpID0+IGYua2V5ID09PSBkZWYua2V5KSAhPT0gLTEpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgQWRkaXRpb25hbCBvcHRpb24geyR7ZGVmLmtleX19IGFscmVhZHkgZGVmaW5lLmApO1xuICAgIH1cbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBBZGRpbmcgYWRkaXRpb25hbCBvcHRpb24geyR7ZGVmLmtleX19YCwgZGVmKTtcbiAgICB0aGlzLl9vcHRpb25EZWZzLnB1c2goZGVmKTtcbiAgfVxuXG4gIHJlZ2lzdGVyT3B0aW9ucyhkZWZzOiBGb3JtRmllbGREZWZpbml0aW9uW10pIHtcbiAgICAoZGVmcyB8fCBbXSkuZm9yRWFjaCgoZGVmKSA9PiB0aGlzLnJlZ2lzdGVyT3B0aW9uKGRlZikpO1xuICB9XG5cbiAgLyogLS0gcHJvdGVjdGVkIG1ldGhvZCAtLSAqL1xuXG4gIHByb3RlY3RlZCByZXNldERhdGEoKSB7XG4gICAgdGhpcy5zdG9wTGlzdGVuUmVtb3RlQ2hhbmdlcygpO1xuICAgIHRoaXMuX2NhY2hlLmxvYWRlZCA9IGZhbHNlO1xuICAgIHRoaXMuX2NhY2hlLmtleXBhaXIgPSBudWxsO1xuICAgIHRoaXMuX2NhY2hlLmF1dGhUb2tlbiA9IG51bGw7XG4gICAgdGhpcy5fY2FjaGUuYXV0aEJhc2ljID0gbnVsbDtcbiAgICB0aGlzLl9jYWNoZS5wdWJrZXkgPSBudWxsO1xuICAgIHRoaXMuX2NhY2hlLm1haW5Qcm9maWxlID0gbnVsbDtcbiAgICB0aGlzLl9jYWNoZS5wZXJzb24gPSBudWxsO1xuICAgIHRoaXMuX2NhY2hlLmRlcGFydG1lbnQgPSBudWxsO1xuICAgIHRoaXMuX2RhdGEgPSBuZXcgQWNjb3VudCgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIGFzeW5jIGxvYWREYXRhKG9wdHM/OiB7IG9mZmxpbmU/OiBib29sZWFuOyBmZXRjaFBvbGljeT86IEZldGNoUG9saWN5IH0pOiBQcm9taXNlPEFjY291bnQ+IHtcbiAgICBpZiAoIXRoaXMuX2NhY2hlLnB1YmtleSkgdGhyb3cgbmV3IEVycm9yKCdVc2VyIG5vdCBsb2dnZWQnKTtcblxuICAgIHRoaXMuX2NhY2hlLmxvYWRlZCA9IGZhbHNlO1xuXG4gICAgY29uc29sZS5kZWJ1ZygnW2FjY291bnQtc2VydmljZV0gTG9hZGluZyBhY2NvdW50IGRhdGEuLi4nKTtcbiAgICB0cnkge1xuICAgICAgY29uc3QgYWNjb3VudCA9IChhd2FpdCB0aGlzLmxvYWQob3B0cykpIHx8IG5ldyBBY2NvdW50KCk7XG5cbiAgICAgIC8vIFNldCBkZWZhdWx0c1xuICAgICAgYWNjb3VudC5hdmF0YXIgPSBhY2NvdW50LmF2YXRhciB8fCB0aGlzLmVudmlyb25tZW50LmJhc2VVcmwgKyBQZXJzb25VdGlscy5ERUZBVUxUX0FWQVRBUl9JTUFHRTtcbiAgICAgIGFjY291bnQuc2V0dGluZ3MgPSBhY2NvdW50LnNldHRpbmdzIHx8IG5ldyBVc2VyU2V0dGluZ3MoKTtcbiAgICAgIGFjY291bnQuc2V0dGluZ3MubG9jYWxlID0gYWNjb3VudC5zZXR0aW5ncy5sb2NhbGUgfHwgdGhpcy5zZXR0aW5ncy5sb2NhbGU7XG4gICAgICBhY2NvdW50LnNldHRpbmdzLmxhdExvbmdGb3JtYXQgPSBhY2NvdW50LnNldHRpbmdzLmxhdExvbmdGb3JtYXQgfHwgdGhpcy5zZXR0aW5ncy5sYXRMb25nRm9ybWF0IHx8ICdERE1NJztcbiAgICAgIGFjY291bnQuc2V0dGluZ3MuY29udGVudCA9IGFjY291bnQuc2V0dGluZ3MuY29udGVudCB8fCB7fTtcblxuICAgICAgLy8gUmVhZCBtYWluIHByb2ZpbGVcbiAgICAgIHRoaXMuX2NhY2hlLm1haW5Qcm9maWxlID0gUGVyc29uVXRpbHMuZ2V0TWFpblByb2ZpbGUoYWNjb3VudC5wcm9maWxlcyk7XG5cbiAgICAgIC8vIFVwZGF0ZSwgaW5zdGVhZCBvZiByZXBsYWNlIGl0XG4gICAgICBpZiAodGhpcy5fZGF0YSkge1xuICAgICAgICB0aGlzLl9kYXRhLmZyb21PYmplY3QoYWNjb3VudCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aGlzLl9kYXRhID0gYWNjb3VudDtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2NhY2hlLmxvYWRlZCA9IHRydWU7XG5cbiAgICAgIC8vIEFwcGx5IHNldHRpbmdzLCBmb3VuZCBpbiByZW1vdGUgYWNjb3VudFxuICAgICAgaWYgKGFjY291bnQuc2V0dGluZ3MgJiYgdGhpcy5zZXR0aW5ncy5zZXR0aW5ncy5hY2NvdW50SW5oZXJpdGFuY2UpIHtcbiAgICAgICAgY29uc29sZS5kZWJ1ZygnW2FjY291bnQtc2VydmljZV0gQ29weWluZyBhY2NvdW50IHNldHRpbmdzIGludG8gbG9jYWwgc2V0dGluZ3MuLi4nKTtcbiAgICAgICAgY29uc3QgbG9jYWxTZXR0aW5ncyA9IHRoaXMuc2V0dGluZ3Muc2V0dGluZ3M7XG4gICAgICAgIGNvbnN0IGFjY291bnRTZXR0aW5ncyA9IGFjY291bnQuc2V0dGluZ3MuYXNMb2NhbFNldHRpbmdzKCk7XG4gICAgICAgIGNvbnN0IGRlZmluaXRpb25zID0gcmVtb3ZlRHVwbGljYXRlc0Zyb21BcnJheShbLi4udGhpcy5vcHRpb25EZWZzLCAuLi50aGlzLnNldHRpbmdzLm9wdGlvbkRlZnNdLCAna2V5Jyk7XG4gICAgICAgIGNvbnN0IG1lcmdlZFByb3BlcnRpZXMgPSBBcHBQcm9wZXJ0aWVzVXRpbHMuc2FuaXRpemUoXG4gICAgICAgICAge1xuICAgICAgICAgICAgLi4ubG9jYWxTZXR0aW5ncy5wcm9wZXJ0aWVzLFxuICAgICAgICAgICAgLi4uYWNjb3VudFNldHRpbmdzLnByb3BlcnRpZXMsXG4gICAgICAgICAgfSxcbiAgICAgICAgICB7IGRlZmluaXRpb25zIH1cbiAgICAgICAgKTtcblxuICAgICAgICAvLyBNZXJnZSBsb2NhbC9hY2NvdW50IHNldHRpbmdzXG4gICAgICAgIGNvbnN0IHNldHRpbmdzID0ge1xuICAgICAgICAgIC4uLmxvY2FsU2V0dGluZ3MsXG4gICAgICAgICAgLi4uYWNjb3VudFNldHRpbmdzLFxuICAgICAgICAgIHByb3BlcnRpZXM6IG1lcmdlZFByb3BlcnRpZXMsXG4gICAgICAgIH07XG5cbiAgICAgICAgYXdhaXQgdGhpcy5zZXR0aW5ncy5hcHBseShzZXR0aW5ncyk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB0aGlzLl9kYXRhO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICB0aGlzLnJlc2V0RGF0YSgpO1xuICAgICAgaWYgKGVycm9yLmNvZGUgJiYgZXJyb3IubWVzc2FnZSkgdGhyb3cgZXJyb3I7XG5cbiAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgdGhyb3cge1xuICAgICAgICBjb2RlOiBFcnJvckNvZGVzLkxPQURfQUNDT1VOVF9FUlJPUixcbiAgICAgICAgbWVzc2FnZTogJ0VSUk9SLkxPQURfQUNDT1VOVF9FUlJPUicsXG4gICAgICB9O1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyBsaXN0ZW5TZXR0aW5ncygpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBXaGVuIHNldHRpbmdzIGNoYW5nZWQ6IHNhdmUgaXQgcmVtb3RlbHlcbiAgICB0aGlzLnJlZ2lzdGVyU3Vic2NyaXB0aW9uKFxuICAgICAgdGhpcy5zZXR0aW5ncy5vbkNoYW5nZVxuICAgICAgICAucGlwZShcbiAgICAgICAgICBkZWJvdW5jZVRpbWUoMjAwMCksXG4gICAgICAgICAgZmlsdGVyKCgpID0+IHRoaXMuaXNMb2dpbigpICYmIHRoaXMubmV0d29yay5vbmxpbmUpXG4gICAgICAgIClcbiAgICAgICAgLnN1YnNjcmliZSgoc2V0dGluZ3MpID0+IHRoaXMuc2F2ZUxvY2FsU2V0dGluZ3NSZW1vdGVseShzZXR0aW5ncykpXG4gICAgKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyByZXN0b3JlTG9jYWxseSgpOiBQcm9taXNlPEFjY291bnQgfCB1bmRlZmluZWQ+IHtcbiAgICAvLyBSZXN0b3JlIGZyb20gc3RvcmFnZVxuICAgIGNvbnN0IHZhbHVlcyA9IGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgIHRoaXMuc3RvcmFnZS5nZXQoUFVCS0VZX1NUT1JBR0VfS0VZKSxcbiAgICAgIHRoaXMuc3RvcmFnZS5nZXQoVE9LRU5fU1RPUkFHRV9LRVkpLFxuICAgICAgdGhpcy5zdG9yYWdlLmdldChTRUNLRVlfU1RPUkFHRV9LRVkpLFxuICAgIF0pO1xuICAgIGxldCBwdWJrZXkgPSB2YWx1ZXNbMF07XG4gICAgbGV0IHRva2VuID0gdmFsdWVzWzFdO1xuICAgIGNvbnN0IHNlY2tleSA9IHZhbHVlc1syXTtcblxuICAgIC8vIERFViBvbmx5IC0gYXV0aCBieSB0b2tlblxuICAgIGlmICghdGhpcy5lbnZpcm9ubWVudC5wcm9kdWN0aW9uICYmIHRoaXMuZW52aXJvbm1lbnQuZGVmYXVsdEF1dGhWYWx1ZXM/LnRva2VuKSB7XG4gICAgICB0b2tlbiA9IHRva2VuIHx8IHRoaXMuZW52aXJvbm1lbnQuZGVmYXVsdEF1dGhWYWx1ZXMudG9rZW47XG4gICAgICBwdWJrZXkgPSBwdWJrZXkgfHwgdG9rZW4/LnNwbGl0KCc6JywgMilbMF07XG4gICAgfVxuXG4gICAgLy8gUXVpdCBpZiBubyBwdWJrZXkgKG5vdCBsb2dnZWQpXG4gICAgaWYgKCFwdWJrZXkpIHJldHVybjtcblxuICAgIC8vIFF1aXQgaWYgY291bGQgbm90IGF1dGggb24gcG9kXG4gICAgY29uc3QgY2FuUmVtb3RlQXV0aCA9IHRva2VuIHx8IHNlY2tleSB8fCBmYWxzZTtcbiAgICBpZiAoIWNhblJlbW90ZUF1dGgpIHJldHVybjtcblxuICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIEFjY291bnQgcmVzdG9yYXRpb24uLi5gKTtcblxuICAgIHRoaXMuX2NhY2hlLmF1dGhUb2tlbiA9IHRva2VuO1xuICAgIHRoaXMuX2NhY2hlLnB1YmtleSA9IHB1YmtleTtcbiAgICB0aGlzLl9jYWNoZS5rZXlwYWlyID1cbiAgICAgIChzZWNrZXkgJiYge1xuICAgICAgICBwdWJsaWNLZXk6IEJhc2U1OC5kZWNvZGUocHVia2V5KSxcbiAgICAgICAgc2VjcmV0S2V5OiBCYXNlNTguZGVjb2RlKHNlY2tleSksXG4gICAgICB9KSB8fFxuICAgICAgbnVsbDtcblxuICAgIC8vIE9ubGluZSBtb2RlOiB0cnkgdG8gY29ubmVjdCB0byBwb2RcbiAgICBpZiAodGhpcy5uZXR3b3JrLm9ubGluZSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgdGhpcy5hdXRoZW50aWNhdGUoKTtcbiAgICAgICAgaWYgKCF0aGlzLl9jYWNoZS5hdXRoVG9rZW4gJiYgIXRoaXMuX2NhY2hlLmF1dGhCYXNpYykgdGhyb3cgbmV3IEVycm9yKCdBdXRoZW50aWNhdGlvbiBmYWlsZWQnKTtcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIC8vIE9mZmxpbmUgZmVhdHVyZSBhcmUgZW5hYmxlOiBjb250aW51ZSBpbiBvZmZsaW5lIG1vZGVcbiAgICAgICAgaWYgKHRoaXMuc2V0dGluZ3MuaGFzT2ZmbGluZUZlYXR1cmUoKSkge1xuICAgICAgICAgIGNvbnNvbGUud2FybignW2FjY291bnRdIFVuYWJsZSB0byBhdXRoZW50aWNhdGUgb24gcG9kOiBmb3JjaW5nIG9mZmxpbmUgbW9kZScpO1xuICAgICAgICAgIHRoaXMubmV0d29yay5zZXRGb3JjZU9mZmxpbmUodHJ1ZSwgeyBzaG93VG9hc3Q6IGZhbHNlIH0pO1xuICAgICAgICAgIC8vIENvbnRpbnVlXG4gICAgICAgIH1cbiAgICAgICAgLy8gTm8gb2ZmbGluZSBmZWF0dXJlcyBlbmFibGUgKD1vZmZsaW5lIG1vZGUgbm90IGFsbG93ZWQpXG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgICAgIHRoaXMubG9nb3V0KCk7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gR2V0IHRoZSBhY2NvdW50LCBmcm9tIHB1YmtleVxuICAgIGxldCBqc29uQWNjb3VudCA9IGF3YWl0IHRoaXMuc3RvcmFnZS5nZXQoYCR7QUNDT1VOVF9TVE9SQUdFX0tFWX0jJHtwdWJrZXl9YCk7XG4gICAgaWYgKCFqc29uQWNjb3VudCkge1xuICAgICAgLy8gVHJ5IHVzaW5nIHRoZSBvbGQgc3RvcmFnZSBrZXlcbiAgICAgIGNvbnN0IGFjY291bnRTdHIgPSBhd2FpdCB0aGlzLnN0b3JhZ2UuZ2V0KEFDQ09VTlRfU1RPUkFHRV9LRVkpO1xuICAgICAganNvbkFjY291bnQgPSBhY2NvdW50U3RyICYmICgodHlwZW9mIGFjY291bnRTdHIgPT09ICdzdHJpbmcnICYmIEpTT04ucGFyc2UoanNvbkFjY291bnQpKSB8fCBhY2NvdW50U3RyKTtcbiAgICB9XG5cbiAgICAvLyBJbnZhbGlkIGFjY291bnQ6IGRvIG5vdCB1c2UgaXRcbiAgICBpZiAoIWpzb25BY2NvdW50IHx8IGpzb25BY2NvdW50LnB1YmtleSAhPT0gcHVia2V5KSB7XG4gICAgICAvLyBERVYgb25seTogY3JlYXRlIGEgZmFrZSBhY2NvdW50IHdpdGggZ2l2ZW4gdXNlcm5hbWVcbiAgICAgIGlmICghdGhpcy5lbnZpcm9ubWVudC5wcm9kdWN0aW9uICYmIHRoaXMuZW52aXJvbm1lbnQuZGVmYXVsdEF1dGhWYWx1ZXMudXNlcm5hbWUpIHtcbiAgICAgICAgY29uc3QgdXNlcm5hbWUgPSB0aGlzLmVudmlyb25tZW50LmRlZmF1bHRBdXRoVmFsdWVzLnVzZXJuYW1lO1xuICAgICAgICBqc29uQWNjb3VudCA9IHsgcHVia2V5LCB1c2VybmFtZSwgZmlyc3ROYW1lOiB1c2VybmFtZSwgbGFzdE5hbWU6IHVzZXJuYW1lIH07XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gVHJhbnNmb3JtIHRvIGVudGl0eVxuICAgIGNvbnN0IGFjY291bnQgPSBBY2NvdW50LmZyb21PYmplY3QoanNvbkFjY291bnQpO1xuXG4gICAgLy8gVXBkYXRlIGRhdGFcbiAgICB0aGlzLl9kYXRhID0gYWNjb3VudDtcbiAgICB0aGlzLl9jYWNoZS5tYWluUHJvZmlsZSA9IFBlcnNvblV0aWxzLmdldE1haW5Qcm9maWxlKGFjY291bnQucHJvZmlsZXMpO1xuICAgIHRoaXMuX2NhY2hlLmxvYWRlZCA9IHRydWU7XG5cbiAgICAvLyBFbWl0IGV2ZW50XG4gICAgdGhpcy5vbkxvZ2luLm5leHQodGhpcy5fZGF0YSk7XG4gICAgdGhpcy5vbkNoYW5nZS5uZXh0KHRoaXMuX2RhdGEpO1xuXG4gICAgaWYgKHRoaXMuX2VuYWJsZUxpc3RlbkNoYW5nZXMpIHRoaXMuc3RhcnRMaXN0ZW5SZW1vdGVDaGFuZ2VzKCk7XG5cbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBBY2NvdW50IHJlc3RvcmF0aW9uIFtPS10ge3B1YmtleTogJHtwdWJrZXkuc3Vic3RyKDAsIDgpfX0sIHtwcm9maWxlOiAke3RoaXMuX2NhY2hlLm1haW5Qcm9maWxlfX1gKTtcblxuICAgIHJldHVybiBhY2NvdW50O1xuICB9XG5cbiAgLyoqXG4gICAqIFNhdmUgYWNjb3VudCBpbnRvIHRoZSBsb2NhbCBzdG9yYWdlXG4gICAqL1xuICBwcm90ZWN0ZWQgYXN5bmMgc2F2ZUxvY2FsbHkoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYgKCF0aGlzLl9jYWNoZS5wdWJrZXkpIHRocm93IG5ldyBFcnJvcignVXNlciBub3QgbG9nZ2VkJyk7XG5cbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBTYXZpbmcgYWNjb3VudCB7JHt0aGlzLl9jYWNoZS5wdWJrZXkuc3Vic3RyaW5nKDAsIDYpfX0gaW4gbG9jYWwgc3RvcmFnZS4uLmApO1xuXG4gICAgLy8gQ29udmVydCBhY2NvdW50IHRvIGpzb25cbiAgICBjb25zdCBqc29uID0gdGhpcy5fZGF0YS5hc09iamVjdCh7IGtlZXBUeXBlbmFtZTogdHJ1ZSB9KTtcbiAgICBjb25zdCBzZWNrZXkgPSAodGhpcy5fY2FjaGUua2V5cGFpciAmJiB0aGlzLl9jYWNoZS5rZXlwYWlyLnNlY3JldEtleSAmJiBCYXNlNTguZW5jb2RlKHRoaXMuX2NhY2hlLmtleXBhaXIuc2VjcmV0S2V5KSkgfHwgbnVsbDtcblxuICAgIC8vIENvbnZlcnQgYXZhdGFyIFVSTCB0byBkYXRhVXJsIChlLmcuICdkYXRhOmltYWdlL3BuZzo8YmFzZTY0IGNvbnRlbnQ+JylcbiAgICBjb25zdCBoYXNBdmF0YXJVcmwgPVxuICAgICAganNvbi5hdmF0YXIgJiZcbiAgICAgICFqc29uLmF2YXRhci5lbmRzV2l0aChQZXJzb25VdGlscy5ERUZBVUxUX0FWQVRBUl9JTUFHRSkgJiZcbiAgICAgIChqc29uLmF2YXRhci5zdGFydHNXaXRoKCdodHRwOi8vJykgfHwganNvbi5hdmF0YXIuc3RhcnRzV2l0aCgnaHR0cHM6Ly8nKSk7XG4gICAgaWYgKGhhc0F2YXRhclVybCAmJiB0aGlzLm5ldHdvcmsub25saW5lKSB7XG4gICAgICBhd2FpdCB0aGlzLmZpbGVcbiAgICAgICAgLmdldEltYWdlKGpzb24uYXZhdGFyLCB7XG4gICAgICAgICAgdGh1bWJuYWlsOiB0cnVlLFxuICAgICAgICAgIG91dHB1dFR5cGU6ICdkYXRhVXJsJyxcbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKGRhdGFVcmwpID0+IHtcbiAgICAgICAgICBpZiAoZGF0YVVybCAmJiB0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZygnW2FjY291bnRdIEltYWdlIGZldGNoZWQ6ICcsIGRhdGFVcmwuc3Vic3RyaW5nKDAsIDUwKSk7XG5cbiAgICAgICAgICAvLyBUT0RPOiBtYWtlIHN1cmUgdG8gZGlzcGxheSBCYXNlNjQgaW1hZ2UgaW4gdGhlIG1lbnUgdG9wIGhlYWRlclxuICAgICAgICAgIC8vanNvbkFjY291bnQuYXZhdGFyID0gZGF0YVVybDtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnIpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmVycm9yKGBbYWNjb3VudF0gRXJyb3Igd2hpbGUgZmV0Y2hpbmcgaW1hZ2U6ICR7anNvbi5hdmF0YXJ9OiAke2Vycn1gKTtcbiAgICAgICAgfSk7XG4gICAgfVxuXG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgICAgdGhpcy5zdG9yYWdlLnNldChQVUJLRVlfU1RPUkFHRV9LRVksIHRoaXMuX2NhY2hlLnB1YmtleSksXG4gICAgICAgIHRoaXMuc3RvcmFnZS5zZXQoVE9LRU5fU1RPUkFHRV9LRVksIHRoaXMuX2NhY2hlLmF1dGhUb2tlbiksXG4gICAgICAgIHRoaXMuc3RvcmFnZS5zZXQoYCR7QUNDT1VOVF9TVE9SQUdFX0tFWX0jJHt0aGlzLl9jYWNoZS5wdWJrZXl9YCwganNvbiksXG4gICAgICAgIC8vIFNlY3JldCBrZXkgKG9wdGlvbmFsKVxuICAgICAgICAoc2Vja2V5ICYmIHRoaXMuc3RvcmFnZS5zZXQoU0VDS0VZX1NUT1JBR0VfS0VZLCBzZWNrZXkpKSB8fCB0aGlzLnN0b3JhZ2UucmVtb3ZlKFNFQ0tFWV9TVE9SQUdFX0tFWSksXG4gICAgICAgIC8vIFJlbW92ZSBvbGQgc3RvcmFnZSBrZXlcbiAgICAgICAgdGhpcy5zdG9yYWdlLnJlbW92ZShBQ0NPVU5UX1NUT1JBR0VfS0VZKSxcbiAgICAgIF0pO1xuXG4gICAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoJ1thY2NvdW50XSBBY2NvdW50IHNhdmVkIGluIGxvY2FsIHN0b3JhZ2UnKTtcbiAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoJ1thY2NvdW50XSBFcnJvciB3aGlsZSBzYXZpbmcgYWNjb3VudCBsb2NhbGx5OiAnICsgKChlcnIgJiYgZXJyLm1lc3NhZ2UpIHx8IGVyciksIGVycik7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgc2F2ZUxvY2FsU2V0dGluZ3NSZW1vdGVseShzb3VyY2U6IExvY2FsU2V0dGluZ3MpIHtcbiAgICBpZiAoIXNvdXJjZSB8fCAhdGhpcy5pc0xvZ2luKCkgfHwgc291cmNlLmFjY291bnRJbmhlcml0YW5jZSA9PT0gZmFsc2UpIHJldHVybjsgLy8gU2tpcFxuICAgIGNvbnN0IGFjY291bnQgPSB0aGlzLmFjY291bnQ7XG5cbiAgICAvLyBNZXJnZSBsb2NhbCBzZXR0aW5ncyBpbnRvIGFjY291bnQncyBzZXR0aW5nc1xuICAgIGNvbnN0IHNldHRpbmdzID0gVXNlclNldHRpbmdzLmZyb21PYmplY3QoYWNjb3VudC5zZXR0aW5ncykgfHwgbmV3IFVzZXJTZXR0aW5ncygpO1xuICAgIGNvbnN0IGNoYW5nZWQgPSBzZXR0aW5ncy5tZXJnZShzb3VyY2UsIHRoaXMuX3JlbW90ZUxvY2FsU2V0dGluZ3NLZXlzKTtcblxuICAgIGlmICghY2hhbmdlZCkgcmV0dXJuOyAvLyBTa2lwIGlmIHVuY2hhbmdlZFxuXG4gICAgLy8gU2F2ZSByZW1vdGVseVxuICAgIHRoaXMuYWNjb3VudC5zZXR0aW5ncyA9IHNldHRpbmdzO1xuICAgIGF3YWl0IHRoaXMuc2F2ZVNldHRpbmdzUmVtb3RlbHkoc2V0dGluZ3MpO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBvciB1cGRhdGUgYW4gdXNlciBhY2NvdW50XG4gICAqXG4gICAqIEBwYXJhbSBhY2NvdW50XG4gICAqIEBwYXJhbSBrZXlQYWlyXG4gICAqL1xuICBwcm90ZWN0ZWQgYXN5bmMgc2F2ZVJlbW90ZWx5KGFjY291bnQ6IEFjY291bnQsIGtleVBhaXI/OiBLZXlwYWlyKTogUHJvbWlzZTxBY2NvdW50PiB7XG4gICAgYWNjb3VudC5wdWJrZXkgPSBhY2NvdW50LnB1YmtleSB8fCAoa2V5UGFpciAmJiBCYXNlNTguZW5jb2RlKGtleVBhaXIucHVibGljS2V5KSk7XG4gICAgaWYgKCFhY2NvdW50LnB1YmtleSkgdGhyb3cgbmV3IEVycm9yKCdNaXNzaW5nIGFjY291bnQgcHVia2V5Jyk7XG5cbiAgICBjb25zdCBub3cgPSB0aGlzLl9kZWJ1ZyAmJiBEYXRlLm5vdygpO1xuICAgIGlmICh0aGlzLl9kZWJ1ZykgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFNhdmluZyBhY2NvdW50IHJlbW90ZWx5Li4uYCk7XG5cbiAgICBjb25zdCBpc05ldyA9IGlzTmlsKGFjY291bnQuaWQpO1xuXG4gICAgLy8gSWYgdGhpcyBpcyBhbiB1cGRhdGU6IGdldCBleGlzdGluZyBhY2NvdW50J3MgdXBkYXRlRGF0ZSwgdG8gYXZvaWQgJ3ZlcnNpb24gZXJyb3InIHdoZW4gc2F2aW5nXG4gICAgaWYgKCFpc05ldykge1xuICAgICAgY29uc3QgZXhpc3RpbmdBY2NvdW50ID0gYXdhaXQgdGhpcy5sb2FkKHsgZmV0Y2hQb2xpY3k6ICduZXR3b3JrLW9ubHknIH0pO1xuICAgICAgaWYgKCFleGlzdGluZ0FjY291bnQgfHwgIWV4aXN0aW5nQWNjb3VudC51cGRhdGVEYXRlKSB7XG4gICAgICAgIHRocm93IHsgY29kZTogRXJyb3JDb2Rlcy5BQ0NPVU5UX05PVF9FWElTVFMsIG1lc3NhZ2U6ICdFUlJPUi5BQ0NPVU5UX05PVF9FWElTVFMnIH07XG4gICAgICB9XG4gICAgICB0aGlzLmNvcHlJZEFuZFVwZGF0ZURhdGUoZXhpc3RpbmdBY2NvdW50LCBhY2NvdW50KTtcbiAgICB9XG5cbiAgICAvLyBNZXJnaW5nIHNldHRpbmdzIC0gRklYTUUgLSBMdWRvIC0gQSByZXZvaXIgYXZlYyBsYSBNUiBzdXIgbGUgdXNlciBzZXR0aW5nc1xuICAgIC8vYWNjb3VudC5zZXR0aW5ncy5jb250ZW50WydwYWdlcyddID0gdGhpcy5zZXR0aW5ncy5zZXR0aW5ncy5wYWdlcztcbiAgICAvL2FjY291bnQuc2V0dGluZ3MuY29udGVudFsncGFnZUhpc3RvcnknXSA9IHRoaXMuc2V0dGluZ3Muc2V0dGluZ3MucGFnZUhpc3Rvcnk7XG5cbiAgICAvLyBDb252ZXJ0IHRvIGpzb25cbiAgICBjb25zdCBqc29uID0gYWNjb3VudC5hc09iamVjdChNSU5JRllfRU5USVRZX0ZPUl9QT0QpO1xuXG4gICAgLy8gVXNlciBub3QgYWxsb3cgdG8gY2hhbmdlIGhpcyBwcm9maWxlc1xuICAgIGRlbGV0ZSBqc29uLnByb2ZpbGVzO1xuXG4gICAgY29uc3QgbXV0YXRpb24gPSBpc05ldyA/IHRoaXMubXV0YXRpb25zLmNyZWF0ZSA6IHRoaXMuX2FwaVRva2VuRW5hYmxlZCA/IHRoaXMubXV0YXRpb25zLnNhdmVXaXRoVG9rZW5zIDogdGhpcy5tdXRhdGlvbnMuc2F2ZTtcblxuICAgIC8vIEV4ZWN1dGUgbXV0YXRpb25cbiAgICBjb25zdCB7IGRhdGEgfSA9IGF3YWl0IHRoaXMuZ3JhcGhxbC5tdXRhdGU8eyBkYXRhOiBBY2NvdW50IH0+KHtcbiAgICAgIG11dGF0aW9uLFxuICAgICAgdmFyaWFibGVzOiB7IGRhdGE6IGpzb24gfSxcbiAgICAgIGVycm9yOiB7XG4gICAgICAgIGNvZGU6IEVycm9yQ29kZXMuU0FWRV9BQ0NPVU5UX0VSUk9SLFxuICAgICAgICBtZXNzYWdlOiAnRVJST1IuU0FWRV9BQ0NPVU5UX0VSUk9SJyxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyBDb3B5IHVwZGF0ZSBwcm9wZXJ0aWVzXG4gICAgdGhpcy5jb3B5SWRBbmRVcGRhdGVEYXRlKGRhdGEsIGFjY291bnQpO1xuXG4gICAgaWYgKHRoaXMuX2RlYnVnKSBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gQWNjb3VudCByZW1vdGVseSBzYXZlZCBpbiAke0RhdGUubm93KCkgLSBub3d9bXNgKTtcblxuICAgIHJldHVybiBhY2NvdW50O1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBvciB1cGRhdGUgdXNlciBzZXR0aW5nc1xuICAgKlxuICAgKiBAcGFyYW0gc2V0dGluZ3NcbiAgICovXG4gIHByb3RlY3RlZCBhc3luYyBzYXZlU2V0dGluZ3NSZW1vdGVseShzZXR0aW5nczogVXNlclNldHRpbmdzKSB7XG4gICAgY29uc3Qgbm93ID0gdGhpcy5fZGVidWcgJiYgRGF0ZS5ub3coKTtcbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBTYXZpbmcgc2V0dGluZ3MgcmVtb3RlbHkuLi5gKTtcblxuICAgIGNvbnN0IGpzb24gPSBzZXR0aW5ncy5hc09iamVjdChNSU5JRllfRU5USVRZX0ZPUl9QT0QpO1xuXG4gICAgLy8gRXhlY3V0ZSBtdXRhdGlvblxuICAgIGNvbnN0IHsgZGF0YSB9ID0gYXdhaXQgdGhpcy5ncmFwaHFsLm11dGF0ZTx7IGRhdGE6IFVzZXJTZXR0aW5ncyB9Pih7XG4gICAgICBtdXRhdGlvbjogdGhpcy5tdXRhdGlvbnMuc2F2ZVNldHRpbmdzLFxuICAgICAgdmFyaWFibGVzOiB7IGRhdGE6IGpzb24gfSxcbiAgICAgIGVycm9yOiB7XG4gICAgICAgIGNvZGU6IEVycm9yQ29kZXMuU0FWRV9TRVRUSU5HU19FUlJPUixcbiAgICAgICAgbWVzc2FnZTogJ0VSUk9SLlNBVkVfU0VUVElOR1NfRVJST1InLFxuICAgICAgfSxcbiAgICB9KTtcblxuICAgIC8vIENvcHkgdXBkYXRlIHByb3BlcnRpZXNcbiAgICB0aGlzLmNvcHlJZEFuZFVwZGF0ZURhdGVTZXR0aW5ncyhkYXRhLCBzZXR0aW5ncyk7XG5cbiAgICBpZiAodGhpcy5fZGVidWcpIGNvbnNvbGUuZGVidWcoYFthY2NvdW50XSBTZXR0aW5ncyByZW1vdGVseSBzYXZlZCBpbiAke0RhdGUubm93KCkgLSBub3d9bXNgKTtcblxuICAgIC8vIFNhdmUgaW50byBhY2NvdW50XG4gICAgaWYgKHRoaXMuYWNjb3VudCkgdGhpcy5hY2NvdW50LnNldHRpbmdzID0gc2V0dGluZ3M7XG4gIH1cblxuICBwcm90ZWN0ZWQgYXN5bmMgYXV0aGVudGljYXRlQW5kR2V0VG9rZW4odG9rZW4/OiBzdHJpbmcsIGNvdW50ZXI/OiBudW1iZXIpOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIGlmICghdGhpcy5fY2FjaGUucHVia2V5KSB0aHJvdyBuZXcgRXJyb3IoJ1VzZXIgbm90IGxvZ2dlZCcpO1xuXG4gICAgaWYgKCFjb3VudGVyKSBjb25zb2xlLmluZm8oJ1thY2NvdW50XSBBdXRoZW50aWNhdGlvbiBvbiBwb2QuLi4nKTtcblxuICAgIGlmIChjb3VudGVyID4gNCkge1xuICAgICAgaWYgKHRoaXMuX2RlYnVnKSBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gRmFpbGVkIHRvIGF1dGhlbnRpY2F0aW9uIG9uIHBvZCAoYWZ0ZXIgJHtjb3VudGVyfSBhdHRlbXB0cylgKTtcbiAgICAgIHRocm93IHsgY29kZTogRXJyb3JDb2Rlcy5BVVRIX1NFUlZFUl9FUlJPUiwgbWVzc2FnZTogJ0VSUk9SLkFVVEhfU0VSVkVSX0VSUk9SJyB9O1xuICAgIH1cblxuICAgIC8vIENoZWNrIGlmIHZhbGlkXG4gICAgaWYgKHRva2VuKSB7XG4gICAgICBjb25zdCBkYXRhID0gYXdhaXQgdGhpcy5ncmFwaHFsLnF1ZXJ5PHsgYXV0aGVudGljYXRlOiBib29sZWFuIH0sIHsgdG9rZW46IHN0cmluZyB9Pih7XG4gICAgICAgIHF1ZXJ5OiBBdXRoUXVlcnksXG4gICAgICAgIHZhcmlhYmxlczoge1xuICAgICAgICAgIHRva2VuLFxuICAgICAgICB9LFxuICAgICAgICBlcnJvcjoge1xuICAgICAgICAgIGNvZGU6IEVycm9yQ29kZXMuVU5BVVRIT1JJWkVELFxuICAgICAgICAgIG1lc3NhZ2U6ICdFUlJPUi5VTkFVVEhPUklaRUQnLFxuICAgICAgICB9LFxuICAgICAgICBmZXRjaFBvbGljeTogJ25vLWNhY2hlJyxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBUb2tlbiBpcyBhY2NlcHRlZCBieSB0aGUgc2VydmVyXG4gICAgICBpZiAoZGF0YSAmJiBkYXRhLmF1dGhlbnRpY2F0ZSkge1xuICAgICAgICAvLyBTdG9yZSB0aGUgdG9rZW5cbiAgICAgICAgdGhpcy5vbkF1dGhUb2tlbkNoYW5nZS5uZXh0KHRva2VuKTtcblxuICAgICAgICBjb25zb2xlLmluZm8oYFthY2NvdW50XSBBdXRoZW50aWNhdGlvbiBvbiBwb2QgW09LXSB7cHVia2V5OiAnJHt0aGlzLl9jYWNoZS5wdWJrZXkuc3Vic3RyKDAsIDgpfSd9YCk7XG4gICAgICAgIHJldHVybiB0b2tlbjsgLy8gcmV0dXJuIHRoZSB0b2tlblxuICAgICAgfVxuXG4gICAgICAvLyBDb250aW51ZSAod2lsbCByZXRyeSB3aXRoIGFub3RoZXIgY2hhbGxlbmdlKVxuICAgIH1cblxuICAgIC8vIEdlbmVyYXRlIGEgbmV3IHRva2VuXG4gICAgY29uc3QgYXV0aENoYWxsZW5nZSA9IGF3YWl0IHRoaXMuZ2V0QXV0aENoYWxsZW5nZSgpO1xuICAgIC8vIFRPRE86IGNoZWNrIHNlcnZlciBwdWJrZXkgYXMgYSB2YWxpZCBjZXJ0aWZpY2F0ZVxuXG4gICAgLy8gRG8gdGhlIGNoYWxsZW5nZVxuICAgIGNvbnN0IHNpZ25hdHVyZSA9IGF3YWl0IENyeXB0b1NlcnZpY2Uuc2lnbihhdXRoQ2hhbGxlbmdlLmNoYWxsZW5nZSwgdGhpcy5fY2FjaGUua2V5cGFpcik7XG4gICAgY29uc3QgbmV3VG9rZW4gPSBgJHt0aGlzLl9jYWNoZS5wdWJrZXl9OiR7YXV0aENoYWxsZW5nZS5jaGFsbGVuZ2V9fCR7c2lnbmF0dXJlfWA7XG5cbiAgICAvLyBpdGVyYXRlIHdpdGggdGhlIG5ldyB0b2tlblxuICAgIHJldHVybiBhd2FpdCB0aGlzLmF1dGhlbnRpY2F0ZUFuZEdldFRva2VuKG5ld1Rva2VuLCAoY291bnRlciB8fCAxKSArIDEgLyogaW5jcmVtZW50ICovKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyBnZXRBdXRoQ2hhbGxlbmdlKCk6IFByb21pc2U8SUF1dGhDaGFsbGVuZ2U+IHtcbiAgICBjb25zdCBjaGFsbGVuZ2VFcnJvciA9IHtcbiAgICAgIGNvZGU6IEVycm9yQ29kZXMuQVVUSF9DSEFMTEVOR0VfRVJST1IsXG4gICAgICBtZXNzYWdlOiAnRVJST1IuQVVUSF9DSEFMTEVOR0VfRVJST1InLFxuICAgIH07XG4gICAgY29uc3QgeyBhdXRoQ2hhbGxlbmdlIH0gPSBhd2FpdCB0aGlzLmdyYXBocWwucXVlcnk8e1xuICAgICAgYXV0aENoYWxsZW5nZTogSUF1dGhDaGFsbGVuZ2U7XG4gICAgfT4oe1xuICAgICAgcXVlcnk6IEF1dGhDaGFsbGVuZ2VRdWVyeSxcbiAgICAgIHZhcmlhYmxlczoge30sXG4gICAgICBlcnJvcjogY2hhbGxlbmdlRXJyb3IsXG4gICAgICBmZXRjaFBvbGljeTogJ25ldHdvcmstb25seScsXG4gICAgfSk7XG5cbiAgICAvLyBDaGVjayBjaGFsbGVuZ2VcbiAgICBpZiAoIWF1dGhDaGFsbGVuZ2UpIHRocm93IGNoYWxsZW5nZUVycm9yOyAvLyBTaG91bGQgbmV2ZXIgb2NjdXJcblxuICAgIC8vIENoZWNrIHNlcnZlciBzaWduYXR1cmVcbiAgICBjb25zdCBzaWduYXR1cmVPSyA9IGF3YWl0IENyeXB0b1NlcnZpY2UudmVyaWZ5KGF1dGhDaGFsbGVuZ2UuY2hhbGxlbmdlLCBhdXRoQ2hhbGxlbmdlLnNpZ25hdHVyZSwgYXV0aENoYWxsZW5nZS5wdWJrZXkpO1xuICAgIGlmICghc2lnbmF0dXJlT0spIHtcbiAgICAgIGNvbnNvbGUud2FybignRklYTUU6IEJhZCBwZWVyIHNpZ25hdHVyZSBvbiBhdXRoIGNoYWxsZW5nZSAhJywgYXV0aENoYWxsZW5nZSk7XG4gICAgfVxuICAgIHJldHVybiBhdXRoQ2hhbGxlbmdlO1xuICB9XG5cbiAgcHJvdGVjdGVkIHN0YXJ0TGlzdGVuUmVtb3RlQ2hhbmdlcyhvcHRzPzogQWNjb3VudFdhdGNoT3B0aW9ucyk6IFN1YnNjcmlwdGlvbiB7XG4gICAgaWYgKHRoaXMuX2xpc3RlbkNoYW5nZXNTdWJzY3JpcHRpb24pIHRoaXMuc3RvcExpc3RlblJlbW90ZUNoYW5nZXMoKTsgLy8gU3RvcCBwcmV2aW91cyBzdWJzY3JpcHRpb25cbiAgICBpZiAodGhpcy5uZXR3b3JrLm9mZmxpbmUpIHJldHVybiBuZXcgU3Vic2NyaXB0aW9uKCk7IC8vIE9mZmxpbmU6IHNraXBcblxuICAgIHRoaXMuX2xpc3RlbkNoYW5nZXNTdWJzY3JpcHRpb24gPSB0aGlzLndhdGNoQ2hhbmdlcyhvcHRzKS5zdWJzY3JpYmUoe1xuICAgICAgbmV4dDogKGRhdGEpID0+IHtcbiAgICAgICAgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFJlbW90ZSB1cGRhdGUgZGV0ZWN0ZWQgYXQgJHtkYXRhLnVwZGF0ZURhdGV9YCk7XG4gICAgICAgIHRoaXMucmVsb2FkKCk7XG4gICAgICB9LFxuICAgICAgZXJyb3I6IChlcnIpID0+IHtcbiAgICAgICAgaWYgKGVyciAmJiArZXJyLmNvZGUgPT09IFNlcnZlckVycm9yQ29kZXMuTk9UX0ZPVU5EKSB7XG4gICAgICAgICAgY29uc29sZS5pbmZvKCdbYWNjb3VudF0gW1dTXSBBY2NvdW50IG5vdCBleGlzdHMgYW55bW9yZTogZm9yY2UgdXNlciB0byBsb2dvdXQuLi4nLCBlcnIpO1xuICAgICAgICAgIHRoaXMubG9nb3V0KCk7XG4gICAgICAgIH0gZWxzZSBpZiAoZXJyICYmICtlcnIuY29kZSA9PT0gU2VydmVyRXJyb3JDb2Rlcy5VTkFVVEhPUklaRUQpIHtcbiAgICAgICAgICBjb25zb2xlLmluZm8oJ1thY2NvdW50XSBbV1NdIEFjY291bnQgbm90IGF1dGhvcml6ZWQ6IGZvcmNlIHVzZXIgdG8gbG9nb3V0Li4uJywgZXJyKTtcbiAgICAgICAgICB0aGlzLmxvZ291dCgpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnNvbGUud2FybignW2FjY291bnRdIFtXU10gUmVjZWl2ZWQgZXJyb3I6JywgZXJyKTtcbiAgICAgICAgfVxuICAgICAgfSxcbiAgICB9KTtcbiAgICB0aGlzLl9saXN0ZW5DaGFuZ2VzU3Vic2NyaXB0aW9uLmFkZCgoKSA9PiB7XG4gICAgICBjb25zb2xlLmRlYnVnKGBbYWNjb3VudF0gU3RvcCB3YXRjaGluZyByZW1vdGUgY2hhbmdlc2ApO1xuICAgICAgdGhpcy5fbGlzdGVuQ2hhbmdlc1N1YnNjcmlwdGlvbiA9IG51bGw7XG4gICAgfSk7XG4gICAgcmV0dXJuIHRoaXMuX2xpc3RlbkNoYW5nZXNTdWJzY3JpcHRpb247XG4gIH1cblxuICBwcm90ZWN0ZWQgc3RvcExpc3RlblJlbW90ZUNoYW5nZXMoKSB7XG4gICAgdGhpcy5fbGlzdGVuQ2hhbmdlc1N1YnNjcmlwdGlvbj8udW5zdWJzY3JpYmUoKTtcbiAgfVxuXG4gIHByb3RlY3RlZCB3YXRjaENoYW5nZXMob3B0cz86IEFjY291bnRXYXRjaE9wdGlvbnMpOiBPYnNlcnZhYmxlPEFjY291bnQ+IHtcbiAgICBpZiAoIXRoaXMuc3RhcnRlZCkge1xuICAgICAgLy8gV2FpdCBzZXJ2aWNlIHJlYWR5LCB0aGVuIGxvb3BcbiAgICAgIHJldHVybiBmcm9tKHRoaXMucmVhZHkoKSkucGlwZShzd2l0Y2hNYXAoKCkgPT4gdGhpcy53YXRjaENoYW5nZXMob3B0cykpKTtcbiAgICB9XG5cbiAgICBpZiAoIXRoaXMuX2NhY2hlLnB1YmtleSkgdGhyb3cgbmV3IEVycm9yKCdOb3QgbG9nZ2VkIGluJyk7XG4gICAgaWYgKHRoaXMubmV0d29yay5vZmZsaW5lKSB0aHJvdyBuZXcgRXJyb3IoJ0Nhbm5vdCB3YXRjaCBhY2NvdW50IGNoYW5nZXM6IG5ldHdvcmsgaXMgb2ZmbGluZS4nKTtcblxuICAgIGNvbnN0IHZhcmlhYmxlcyA9IHtcbiAgICAgIGludGVydmFsOiB0b051bWJlcihvcHRzICYmIG9wdHMuaW50ZXJ2YWxJblNlY29uZHMsIHRoaXMuX2xpc3RlbkludGVydmFsSW5TZWNvbmRzKSxcbiAgICB9O1xuXG4gICAgY29uc29sZS5kZWJ1ZyhgW2FjY291bnRdIFdhdGNoaW5nIHJlbW90ZSBjaGFuZ2VzLCBldmVyeSAke3ZhcmlhYmxlcy5pbnRlcnZhbH1zYCk7XG5cbiAgICByZXR1cm4gdGhpcy5ncmFwaHFsXG4gICAgICAuc3Vic2NyaWJlPHsgZGF0YTogYW55IH0+KHtcbiAgICAgICAgcXVlcnk6IHRoaXMuc3Vic2NyaXB0aW9ucy5saXN0ZW5DaGFuZ2VzLFxuICAgICAgICB2YXJpYWJsZXMsXG4gICAgICAgIGVycm9yOiB7XG4gICAgICAgICAgY29kZTogRXJyb3JDb2Rlcy5TVUJTQ1JJQkVfQUNDT1VOVF9FUlJPUixcbiAgICAgICAgICBtZXNzYWdlOiAnQUNDT1VOVC5FUlJPUi5TVUJTQ1JJQkVfRVJST1InLFxuICAgICAgICB9LFxuICAgICAgfSlcbiAgICAgIC5waXBlKFxuICAgICAgICBtYXAoKHsgZGF0YSB9KSA9PiBkYXRhKSxcbiAgICAgICAgLy8gS2VlcCBvbmx5IGlmIG5ld2VyXG4gICAgICAgIGZpbHRlcigoZGF0YSkgPT4ge1xuICAgICAgICAgIGNvbnN0IGN1cnJlbnRVcGRhdGVEYXRlOiBNb21lbnQgPSBmcm9tRGF0ZUlTT1N0cmluZyh0aGlzLl9kYXRhPy51cGRhdGVEYXRlKTtcbiAgICAgICAgICBjb25zdCByZW1vdGVVcGRhdGVEYXRlOiBNb21lbnQgPSBmcm9tRGF0ZUlTT1N0cmluZyhkYXRhPy51cGRhdGVEYXRlKTtcbiAgICAgICAgICByZXR1cm4gcmVtb3RlVXBkYXRlRGF0ZSAmJiAoIWN1cnJlbnRVcGRhdGVEYXRlIHx8ICFyZW1vdGVVcGRhdGVEYXRlLmlzU2FtZShjdXJyZW50VXBkYXRlRGF0ZSwgJ21pbGxpc2Vjb25kJykpO1xuICAgICAgICB9KVxuICAgICAgKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBjb3B5SWRBbmRVcGRhdGVEYXRlKHNvdXJjZTogQWNjb3VudCB8IHVuZGVmaW5lZCwgdGFyZ2V0OiBBY2NvdW50KSB7XG4gICAgaWYgKCFzb3VyY2UpIHJldHVybjtcblxuICAgIHRhcmdldC5pZCA9IHNvdXJjZS5pZCB8fCB0YXJnZXQuaWQ7XG4gICAgdGFyZ2V0LnVwZGF0ZURhdGUgPSBzb3VyY2UudXBkYXRlRGF0ZSB8fCB0YXJnZXQudXBkYXRlRGF0ZTtcblxuICAgIC8vIFVwZGF0ZSBzZXR0aW5nc1xuICAgIHRoaXMuY29weUlkQW5kVXBkYXRlRGF0ZVNldHRpbmdzKHNvdXJjZS5zZXR0aW5ncywgdGFyZ2V0LnNldHRpbmdzKTtcblxuICAgIC8vIFVwZGF0ZSB0b2tlbnNcbiAgICBpZiAodGhpcy5fYXBpVG9rZW5FbmFibGVkKSB7XG4gICAgICB0aGlzLmNvcHlJZEFuZFVwZGF0ZURhdGVUb2tlbihzb3VyY2UudG9rZW5zLCB0YXJnZXQudG9rZW5zKTtcbiAgICB9XG4gIH1cblxuICBwcm90ZWN0ZWQgY29weUlkQW5kVXBkYXRlRGF0ZVNldHRpbmdzKHNvdXJjZTogVXNlclNldHRpbmdzIHwgdW5kZWZpbmVkLCB0YXJnZXQ6IFVzZXJTZXR0aW5ncykge1xuICAgIGlmICghc291cmNlKSByZXR1cm47XG5cbiAgICB0YXJnZXQuaWQgPSBzb3VyY2UuaWQgfHwgdGFyZ2V0LmlkO1xuICAgIHRhcmdldC51cGRhdGVEYXRlID0gc291cmNlLnVwZGF0ZURhdGUgfHwgdGFyZ2V0LnVwZGF0ZURhdGU7XG4gIH1cblxuICBwcm90ZWN0ZWQgY29weUlkQW5kVXBkYXRlRGF0ZVRva2VuKHNvdXJjZXM6IFVzZXJUb2tlbltdIHwgdW5kZWZpbmVkLCB0YXJnZXRzOiBVc2VyVG9rZW5bXSkge1xuICAgIGlmIChpc0VtcHR5QXJyYXkoc291cmNlcykpIHJldHVybjtcblxuICAgIHNvdXJjZXMuZm9yRWFjaCgoc291cmNlKSA9PiB7XG4gICAgICBjb25zdCB0YXJnZXQgPSB0YXJnZXRzLmZpbmQoKHZhbHVlKSA9PiBVc2VyVG9rZW4uZXF1YWxzKHZhbHVlLCBzb3VyY2UpKTtcbiAgICAgIGlmICh0YXJnZXQpIHtcbiAgICAgICAgdGFyZ2V0LmlkID0gc291cmNlLmlkIHx8IHRhcmdldC5pZDtcbiAgICAgICAgdGFyZ2V0LmNyZWF0aW9uRGF0ZSA9IHNvdXJjZS5jcmVhdGlvbkRhdGUgfHwgdGFyZ2V0LmNyZWF0aW9uRGF0ZTtcbiAgICAgICAgdGFyZ2V0LnVwZGF0ZURhdGUgPSBzb3VyY2UudXBkYXRlRGF0ZSB8fCB0YXJnZXQudXBkYXRlRGF0ZTtcbiAgICAgICAgdGFyZ2V0LnRva2VuID0gdW5kZWZpbmVkO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgcHJvdGVjdGVkIHNob3dUb2FzdDxUID0gYW55PihvcHRzOiBTaG93VG9hc3RPcHRpb25zKTogUHJvbWlzZTxPdmVybGF5RXZlbnREZXRhaWw8VD4+IHtcbiAgICByZXR1cm4gVG9hc3RzLnNob3codGhpcy50b2FzdENvbnRyb2xsZXIsIHRoaXMudHJhbnNsYXRlLCBvcHRzKTtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZUFjY291bnRTZXJ2aWNlKCkge1xuICByZXR1cm4gPFByb3ZpZGVyPntcbiAgICBwcm92aWRlOiBBUFBfQUNDT1VOVF9TRVJWSUNFLCB1c2VDbGFzczogQWNjb3VudFNlcnZpY2UsIGRlcHM6IFtFTlZJUk9OTUVOVF1cbiAgfVxufVxuIl19