@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.
- package/README.md +2 -0
- package/doc/build.md +3 -5
- package/fesm2022/sumaris-net.ngx-components.mjs +2383 -2150
- package/fesm2022/sumaris-net.ngx-components.mjs.map +1 -1
- package/package.json +15 -15
- package/src/assets/environments/environment-test.json +2 -2
- package/src/assets/i18n/en-US.json +4 -1
- package/src/assets/i18n/en.json +4 -1
- package/src/assets/i18n/fr.json +10 -6
- package/src/assets/manifest.json +1 -1
- package/src/theme/_icons.scss +8 -12
- package/src/theme/_ionic.globals.scss +8 -1
- package/src/theme/_material.globals.scss +4 -1
- package/src/theme/_material.scss +16 -13
- package/src/theme/_mixins.scss +7 -1
- package/src/theme/_ngx-components.forms.scss +87 -84
- package/src/theme/_ngx-components.globals.scss +2 -0
- package/src/theme/_ngx-components.scss +19 -4
- package/src/theme/_ngx-components.table.scss +31 -26
- package/src/theme/_ngx-components.tabs.scss +16 -14
- package/src/theme/_responsive.scss +2 -0
- package/src/theme/_roboto.scss +3 -0
- package/src/theme/_theme.scss +11 -14
- package/src/theme/_theme.variables.scss +6 -1
- package/src/theme/ionic/ionic.functions.color.scss +13 -5
- package/src/theme/ionic/ionic.globals.scss +4 -4
- package/src/theme/ionic/ionic.theme.default.scss +2 -0
- package/types/sumaris-net.ngx-components.d.ts +14132 -0
- package/esm2022/public_api.mjs +0 -403
- package/esm2022/src/app/admin/admin-routing.module.mjs +0 -34
- package/esm2022/src/app/admin/admin.module.mjs +0 -32
- package/esm2022/src/app/admin/users/person.filter.mjs +0 -157
- package/esm2022/src/app/admin/users/person.service.mjs +0 -226
- package/esm2022/src/app/admin/users/person.validator.mjs +0 -78
- package/esm2022/src/app/admin/users/users-select.modal.mjs +0 -167
- package/esm2022/src/app/admin/users/users.mjs +0 -502
- package/esm2022/src/app/admin/users/users.module.mjs +0 -59
- package/esm2022/src/app/admin/users/users.utils.mjs +0 -29
- package/esm2022/src/app/core/about/about.modal.mjs +0 -119
- package/esm2022/src/app/core/about/about.module.mjs +0 -22
- package/esm2022/src/app/core/account/account.module.mjs +0 -75
- package/esm2022/src/app/core/account/account.page.mjs +0 -426
- package/esm2022/src/app/core/account/new-token.form.mjs +0 -109
- package/esm2022/src/app/core/account/new-token.modal.mjs +0 -66
- package/esm2022/src/app/core/account/password/change-password.form.mjs +0 -63
- package/esm2022/src/app/core/account/password/change-password.module.mjs +0 -22
- package/esm2022/src/app/core/account/password/change-password.page.mjs +0 -150
- package/esm2022/src/app/core/account/token.table.mjs +0 -160
- package/esm2022/src/app/core/auth/auth.form.mjs +0 -159
- package/esm2022/src/app/core/auth/auth.modal.mjs +0 -76
- package/esm2022/src/app/core/auth/auth.module.mjs +0 -56
- package/esm2022/src/app/core/auth/reset-password.modal.mjs +0 -89
- package/esm2022/src/app/core/core.module.mjs +0 -157
- package/esm2022/src/app/core/core.testing.module.mjs +0 -128
- package/esm2022/src/app/core/form/array/form-array.mjs +0 -328
- package/esm2022/src/app/core/form/array/testing/form-array-test.module.mjs +0 -23
- package/esm2022/src/app/core/form/array/testing/form-array.test.mjs +0 -97
- package/esm2022/src/app/core/form/buttons/form-buttons-bar.component.mjs +0 -111
- package/esm2022/src/app/core/form/buttons/form-buttons-bar.module.mjs +0 -31
- package/esm2022/src/app/core/form/entity/editor.class.mjs +0 -268
- package/esm2022/src/app/core/form/entity/entity-editor-modal.class.mjs +0 -420
- package/esm2022/src/app/core/form/entity/entity-editor.class.mjs +0 -748
- package/esm2022/src/app/core/form/entity/entity-metadata.component.mjs +0 -25
- package/esm2022/src/app/core/form/entity/entity.module.mjs +0 -33
- package/esm2022/src/app/core/form/entity/tab-editor.class.mjs +0 -190
- package/esm2022/src/app/core/form/form-container.class.mjs +0 -347
- package/esm2022/src/app/core/form/form.class.mjs +0 -342
- package/esm2022/src/app/core/form/form.module.mjs +0 -77
- package/esm2022/src/app/core/form/form.utils.mjs +0 -238
- package/esm2022/src/app/core/form/list/list.form.mjs +0 -243
- package/esm2022/src/app/core/form/list/list.module.mjs +0 -31
- package/esm2022/src/app/core/form/properties/properties-file.service.mjs +0 -133
- package/esm2022/src/app/core/form/properties/properties.form.mjs +0 -397
- package/esm2022/src/app/core/form/properties/properties.module.mjs +0 -85
- package/esm2022/src/app/core/form/properties/properties.table.mjs +0 -267
- package/esm2022/src/app/core/form/properties/properties.utils.mjs +0 -304
- package/esm2022/src/app/core/form/properties/property.validator.mjs +0 -70
- package/esm2022/src/app/core/form/properties/testing/properties-form.test.mjs +0 -103
- package/esm2022/src/app/core/form/properties/testing/properties-form.testing.module.mjs +0 -23
- package/esm2022/src/app/core/form/text/testing/text-form.testing.mjs +0 -42
- package/esm2022/src/app/core/form/text/testing/text-form.testing.module.mjs +0 -25
- package/esm2022/src/app/core/form/text/text-form.component.mjs +0 -205
- package/esm2022/src/app/core/form/text/text-form.module.mjs +0 -31
- package/esm2022/src/app/core/form/text-popover/testing/text-popover.testing.mjs +0 -73
- package/esm2022/src/app/core/form/text-popover/testing/text-popover.testing.module.mjs +0 -22
- package/esm2022/src/app/core/form/text-popover/text-popover.component.mjs +0 -164
- package/esm2022/src/app/core/form/text-popover/text-popover.module.mjs +0 -32
- package/esm2022/src/app/core/form/username/username.form.mjs +0 -55
- package/esm2022/src/app/core/form/username/username.module.mjs +0 -20
- package/esm2022/src/app/core/graphql/graphql.module.mjs +0 -14
- package/esm2022/src/app/core/graphql/graphql.service.mjs +0 -781
- package/esm2022/src/app/core/graphql/graphql.utils.mjs +0 -119
- package/esm2022/src/app/core/home/home.mjs +0 -472
- package/esm2022/src/app/core/home/home.module.mjs +0 -54
- package/esm2022/src/app/core/icon/icon.component.mjs +0 -40
- package/esm2022/src/app/core/icon/icon.module.mjs +0 -20
- package/esm2022/src/app/core/install/install-upgrade-card.component.mjs +0 -394
- package/esm2022/src/app/core/install/install-upgrade-card.module.mjs +0 -31
- package/esm2022/src/app/core/menu/menu.component.mjs +0 -273
- package/esm2022/src/app/core/menu/menu.model.mjs +0 -327
- package/esm2022/src/app/core/menu/menu.module.mjs +0 -47
- package/esm2022/src/app/core/menu/menu.service.mjs +0 -666
- package/esm2022/src/app/core/menu/sub-menu-tab.directive.mjs +0 -113
- package/esm2022/src/app/core/menu/testing/menu-other.testing.mjs +0 -51
- package/esm2022/src/app/core/menu/testing/menu.testing.mjs +0 -165
- package/esm2022/src/app/core/menu/testing/menu.testing.module.mjs +0 -60
- package/esm2022/src/app/core/offline/update-offline-mode-card.component.mjs +0 -28
- package/esm2022/src/app/core/offline/update-offline-mode-card.module.mjs +0 -31
- package/esm2022/src/app/core/peer/select-peer.modal.mjs +0 -350
- package/esm2022/src/app/core/peer/select-peer.module.mjs +0 -36
- package/esm2022/src/app/core/register/register-confirm.page.mjs +0 -90
- package/esm2022/src/app/core/register/register.form.mjs +0 -197
- package/esm2022/src/app/core/register/register.modal.mjs +0 -55
- package/esm2022/src/app/core/register/register.module.mjs +0 -23
- package/esm2022/src/app/core/services/account.service.mjs +0 -1403
- package/esm2022/src/app/core/services/auth-guard.service.mjs +0 -74
- package/esm2022/src/app/core/services/base-entity-service.class.mjs +0 -499
- package/esm2022/src/app/core/services/base-graphql-service.class.mjs +0 -224
- package/esm2022/src/app/core/services/base58.mjs +0 -83
- package/esm2022/src/app/core/services/config/core.config.mjs +0 -293
- package/esm2022/src/app/core/services/config.service.mjs +0 -384
- package/esm2022/src/app/core/services/crypto.service.mjs +0 -107
- package/esm2022/src/app/core/services/errors.mjs +0 -54
- package/esm2022/src/app/core/services/local-settings.service.mjs +0 -540
- package/esm2022/src/app/core/services/model/account.model.mjs +0 -192
- package/esm2022/src/app/core/services/model/config.model.mjs +0 -111
- package/esm2022/src/app/core/services/model/department.model.mjs +0 -35
- package/esm2022/src/app/core/services/model/entity.decorators.mjs +0 -97
- package/esm2022/src/app/core/services/model/entity.model.mjs +0 -347
- package/esm2022/src/app/core/services/model/filter.model.mjs +0 -169
- package/esm2022/src/app/core/services/model/history.model.mjs +0 -2
- package/esm2022/src/app/core/services/model/model.enum.mjs +0 -11
- package/esm2022/src/app/core/services/model/node-feature.model.mjs +0 -31
- package/esm2022/src/app/core/services/model/peer.model.mjs +0 -114
- package/esm2022/src/app/core/services/model/person.model.mjs +0 -118
- package/esm2022/src/app/core/services/model/referential.model.mjs +0 -207
- package/esm2022/src/app/core/services/model/settings.model.mjs +0 -6
- package/esm2022/src/app/core/services/model/token.model.mjs +0 -57
- package/esm2022/src/app/core/services/model/tree-item-entity.model.mjs +0 -234
- package/esm2022/src/app/core/services/network.service.mjs +0 -759
- package/esm2022/src/app/core/services/network.types.mjs +0 -2
- package/esm2022/src/app/core/services/network.utils.mjs +0 -35
- package/esm2022/src/app/core/services/pipes/account.pipes.mjs +0 -38
- package/esm2022/src/app/core/services/pipes/department-to-string.pipe.mjs +0 -23
- package/esm2022/src/app/core/services/pipes/person-to-string.pipe.mjs +0 -23
- package/esm2022/src/app/core/services/pipes/pipes.module.mjs +0 -58
- package/esm2022/src/app/core/services/pipes/referential-to-string.pipe.mjs +0 -36
- package/esm2022/src/app/core/services/pipes/usage-mode.pipes.mjs +0 -37
- package/esm2022/src/app/core/services/platform.service.mjs +0 -780
- package/esm2022/src/app/core/services/storage/entities-storage.service.mjs +0 -427
- package/esm2022/src/app/core/services/storage/entity-store.class.mjs +0 -450
- package/esm2022/src/app/core/services/testing/referential-filter.model.mjs +0 -25
- package/esm2022/src/app/core/services/testing/referential.validator.mjs +0 -46
- package/esm2022/src/app/core/services/validator/account.validator.mjs +0 -46
- package/esm2022/src/app/core/services/validator/base.validator.class.mjs +0 -49
- package/esm2022/src/app/core/services/validator/local-settings.validator.mjs +0 -57
- package/esm2022/src/app/core/services/validator/user-settings.validator.mjs +0 -27
- package/esm2022/src/app/core/services/validator/user-token.validator.mjs +0 -55
- package/esm2022/src/app/core/settings/settings.module.mjs +0 -23
- package/esm2022/src/app/core/settings/settings.page.mjs +0 -406
- package/esm2022/src/app/core/table/async-table.class.mjs +0 -1882
- package/esm2022/src/app/core/table/column/actions-column.component.mjs +0 -137
- package/esm2022/src/app/core/table/column/nav-actions-column.component.mjs +0 -120
- package/esm2022/src/app/core/table/column/row-field.component.mjs +0 -116
- package/esm2022/src/app/core/table/entities-async-table-datasource.class.mjs +0 -415
- package/esm2022/src/app/core/table/entities-table-datasource.class.mjs +0 -407
- package/esm2022/src/app/core/table/memory-table.class.mjs +0 -57
- package/esm2022/src/app/core/table/table-select-columns.component.mjs +0 -46
- package/esm2022/src/app/core/table/table.class.mjs +0 -1918
- package/esm2022/src/app/core/table/table.model.mjs +0 -21
- package/esm2022/src/app/core/table/table.module.mjs +0 -51
- package/esm2022/src/app/core/table/table.pipes.mjs +0 -76
- package/esm2022/src/app/core/table/table.utils.mjs +0 -34
- package/esm2022/src/app/core/table/testing/nested-table.testing.mjs +0 -326
- package/esm2022/src/app/core/table/testing/table-validator.service.mjs +0 -32
- package/esm2022/src/app/core/table/testing/table.testing.mjs +0 -342
- package/esm2022/src/app/core/table/testing/table.testing.module.mjs +0 -47
- package/esm2022/src/app/core/table/testing/table2-validator.service.mjs +0 -25
- package/esm2022/src/app/core/table/testing/table2.testing.mjs +0 -322
- package/esm2022/src/app/shared/alerts.mjs +0 -243
- package/esm2022/src/app/shared/audio/audio.mjs +0 -255
- package/esm2022/src/app/shared/audio/audio.testing.mjs +0 -32
- package/esm2022/src/app/shared/audio/audio.testing.module.mjs +0 -30
- package/esm2022/src/app/shared/base64.utils.mjs +0 -59
- package/esm2022/src/app/shared/capacitor/keyboard.mjs +0 -42
- package/esm2022/src/app/shared/capacitor/plugins.mjs +0 -10
- package/esm2022/src/app/shared/constants.mjs +0 -18
- package/esm2022/src/app/shared/dates.mjs +0 -212
- package/esm2022/src/app/shared/debug/debug-service.class.mjs +0 -3
- package/esm2022/src/app/shared/debug/debug.component.mjs +0 -44
- package/esm2022/src/app/shared/debug/debug.module.mjs +0 -23
- package/esm2022/src/app/shared/directives/autofocus.directive.mjs +0 -91
- package/esm2022/src/app/shared/directives/autoresize.directive.mjs +0 -75
- package/esm2022/src/app/shared/directives/autotitle.directive.mjs +0 -30
- package/esm2022/src/app/shared/directives/autotooltip.directive.mjs +0 -35
- package/esm2022/src/app/shared/directives/cell-selection/cell-identifier.directive.mjs +0 -55
- package/esm2022/src/app/shared/directives/cell-selection/cell-selection.directive.mjs +0 -206
- package/esm2022/src/app/shared/directives/cell-selection/cell-selection.service.mjs +0 -204
- package/esm2022/src/app/shared/directives/directives.module.mjs +0 -53
- package/esm2022/src/app/shared/directives/drag-and-drop.directive.mjs +0 -52
- package/esm2022/src/app/shared/directives/ng-var.directive.mjs +0 -33
- package/esm2022/src/app/shared/directives/resizable/resizable.component.mjs +0 -62
- package/esm2022/src/app/shared/directives/resizable/resizable.directive.mjs +0 -96
- package/esm2022/src/app/shared/directives/resizable/resizable.module.mjs +0 -17
- package/esm2022/src/app/shared/directives/throttled-click.directive.mjs +0 -43
- package/esm2022/src/app/shared/events.mjs +0 -64
- package/esm2022/src/app/shared/file/csv.utils.mjs +0 -97
- package/esm2022/src/app/shared/file/file.service.mjs +0 -118
- package/esm2022/src/app/shared/file/file.utils.mjs +0 -117
- package/esm2022/src/app/shared/file/images.utils.mjs +0 -186
- package/esm2022/src/app/shared/file/json.utils.mjs +0 -58
- package/esm2022/src/app/shared/file/uri.utils.mjs +0 -29
- package/esm2022/src/app/shared/file/url.utils.mjs +0 -213
- package/esm2022/src/app/shared/focusable.mjs +0 -6
- package/esm2022/src/app/shared/form/field.component.mjs +0 -388
- package/esm2022/src/app/shared/form/field.model.mjs +0 -100
- package/esm2022/src/app/shared/form/loading-spinner.mjs +0 -35
- package/esm2022/src/app/shared/forms.mjs +0 -593
- package/esm2022/src/app/shared/functions.mjs +0 -825
- package/esm2022/src/app/shared/geolocation/geolocation.utils.mjs +0 -94
- package/esm2022/src/app/shared/gesture/gesture-config.mjs +0 -46
- package/esm2022/src/app/shared/gesture/hammer.utils.mjs +0 -2
- package/esm2022/src/app/shared/graph/colors.utils.mjs +0 -85
- package/esm2022/src/app/shared/graph/graph-colors.mjs +0 -214
- package/esm2022/src/app/shared/guard/component-dirty.guard.mjs +0 -68
- package/esm2022/src/app/shared/hotkeys/dialog/hotkeys-dialog.component.mjs +0 -23
- package/esm2022/src/app/shared/hotkeys/hotkeys.service.mjs +0 -147
- package/esm2022/src/app/shared/hotkeys/shared-hotkeys.module.mjs +0 -24
- package/esm2022/src/app/shared/html.utils.mjs +0 -35
- package/esm2022/src/app/shared/http/http.utils.mjs +0 -56
- package/esm2022/src/app/shared/image/gallery/image-gallery.component.mjs +0 -452
- package/esm2022/src/app/shared/image/gallery/image-gallery.module.mjs +0 -62
- package/esm2022/src/app/shared/image/gallery/testing/gallegry.model.testing.mjs +0 -26
- package/esm2022/src/app/shared/image/gallery/testing/gallery.service.testing.mjs +0 -92
- package/esm2022/src/app/shared/image/gallery/testing/gallery.testing.mjs +0 -37
- package/esm2022/src/app/shared/image/gallery/testing/gallery.testing.module.mjs +0 -59
- package/esm2022/src/app/shared/image/image.model.mjs +0 -2
- package/esm2022/src/app/shared/image/image.module.mjs +0 -24
- package/esm2022/src/app/shared/image/image.service.mjs +0 -117
- package/esm2022/src/app/shared/inputs.mjs +0 -281
- package/esm2022/src/app/shared/interceptors/progess.interceptor.mjs +0 -27
- package/esm2022/src/app/shared/logging/log-level.model.mjs +0 -48
- package/esm2022/src/app/shared/logging/logger.model.mjs +0 -47
- package/esm2022/src/app/shared/logging/logging-service.class.mjs +0 -125
- package/esm2022/src/app/shared/logging/logging-service.config.mjs +0 -2
- package/esm2022/src/app/shared/logging/logging-service.module.mjs +0 -30
- package/esm2022/src/app/shared/markdown/markdown.component.mjs +0 -321
- package/esm2022/src/app/shared/markdown/markdown.directive.mjs +0 -34
- package/esm2022/src/app/shared/markdown/markdown.modal.mjs +0 -99
- package/esm2022/src/app/shared/markdown/markdown.module.mjs +0 -87
- package/esm2022/src/app/shared/markdown/markdown.service.mjs +0 -204
- package/esm2022/src/app/shared/markdown/markdown.utils.mjs +0 -87
- package/esm2022/src/app/shared/markdown/testing/markdown.test.mjs +0 -55
- package/esm2022/src/app/shared/markdown/testing/markdown.testing.module.mjs +0 -30
- package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.config.mjs +0 -46
- package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.mjs +0 -1356
- package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.module.mjs +0 -81
- package/esm2022/src/app/shared/material/autocomplete/material.autocomplete.utils.mjs +0 -18
- package/esm2022/src/app/shared/material/autocomplete/testing/autocomplete.test.mjs +0 -242
- package/esm2022/src/app/shared/material/badge/badge.directive.mjs +0 -251
- package/esm2022/src/app/shared/material/badge/badge.module.mjs +0 -20
- package/esm2022/src/app/shared/material/badge/badge.test.mjs +0 -42
- package/esm2022/src/app/shared/material/boolean/boolean.module.mjs +0 -70
- package/esm2022/src/app/shared/material/boolean/material.boolean.mjs +0 -374
- package/esm2022/src/app/shared/material/boolean/testing/boolean.test.page.mjs +0 -83
- package/esm2022/src/app/shared/material/chips/chips.module.mjs +0 -74
- package/esm2022/src/app/shared/material/chips/material.chips.mjs +0 -843
- package/esm2022/src/app/shared/material/chips/testing/chips.test.mjs +0 -151
- package/esm2022/src/app/shared/material/datetime/datetime.module.mjs +0 -79
- package/esm2022/src/app/shared/material/datetime/material.date.mjs +0 -473
- package/esm2022/src/app/shared/material/datetime/material.dateshort.mjs +0 -457
- package/esm2022/src/app/shared/material/datetime/material.datetime.mjs +0 -850
- package/esm2022/src/app/shared/material/datetime/testing/mat-date-time.test.mjs +0 -140
- package/esm2022/src/app/shared/material/datetime/testing/mat-date.test.mjs +0 -127
- package/esm2022/src/app/shared/material/datetime/testing/mat-dateshort.test.mjs +0 -118
- package/esm2022/src/app/shared/material/duration/duration.module.mjs +0 -61
- package/esm2022/src/app/shared/material/duration/duration.utils.mjs +0 -16
- package/esm2022/src/app/shared/material/duration/material.duration.mjs +0 -283
- package/esm2022/src/app/shared/material/duration/testing/mat-duration.test.mjs +0 -85
- package/esm2022/src/app/shared/material/latlong/latlong.utils.mjs +0 -319
- package/esm2022/src/app/shared/material/latlong/material.latlong-input.mjs +0 -471
- package/esm2022/src/app/shared/material/latlong/material.latlong.mjs +0 -231
- package/esm2022/src/app/shared/material/latlong/material.latlong.module.mjs +0 -68
- package/esm2022/src/app/shared/material/latlong/testing/latlong.test.mjs +0 -138
- package/esm2022/src/app/shared/material/material.animations.mjs +0 -200
- package/esm2022/src/app/shared/material/material.config.mjs +0 -3
- package/esm2022/src/app/shared/material/material.module.mjs +0 -224
- package/esm2022/src/app/shared/material/material.testing.module.mjs +0 -234
- package/esm2022/src/app/shared/material/paginator/material.paginator-i18n.mjs +0 -51
- package/esm2022/src/app/shared/material/stepper/material.stepper-i18n.mjs +0 -23
- package/esm2022/src/app/shared/material/swipe/material.swipe.mjs +0 -379
- package/esm2022/src/app/shared/material/swipe/swipe.module.mjs +0 -52
- package/esm2022/src/app/shared/material/swipe/testing/swipe.test.mjs +0 -67
- package/esm2022/src/app/shared/material/test/test-component.mjs +0 -264
- package/esm2022/src/app/shared/material/testing/common.test.mjs +0 -111
- package/esm2022/src/app/shared/modules.mjs +0 -12
- package/esm2022/src/app/shared/named-filter/named-filter-selector.component.mjs +0 -306
- package/esm2022/src/app/shared/named-filter/named-filter.model.mjs +0 -49
- package/esm2022/src/app/shared/named-filter/named-filter.module.mjs +0 -24
- package/esm2022/src/app/shared/named-filter/named-filter.service.mjs +0 -69
- package/esm2022/src/app/shared/named-filter/testing/named-filter-selector.testing.mjs +0 -74
- package/esm2022/src/app/shared/named-filter/testing/named-filter.testing.module.mjs +0 -28
- package/esm2022/src/app/shared/observables.mjs +0 -104
- package/esm2022/src/app/shared/pipes/arrays.pipe.mjs +0 -251
- package/esm2022/src/app/shared/pipes/badge.pipes.mjs +0 -25
- package/esm2022/src/app/shared/pipes/colors.pipe.mjs +0 -29
- package/esm2022/src/app/shared/pipes/date-diff-duration.pipe.mjs +0 -50
- package/esm2022/src/app/shared/pipes/date-format.pipe.mjs +0 -72
- package/esm2022/src/app/shared/pipes/date-from-now.pipe.mjs +0 -28
- package/esm2022/src/app/shared/pipes/date-from.pipe.mjs +0 -29
- package/esm2022/src/app/shared/pipes/dates.pipe.mjs +0 -18
- package/esm2022/src/app/shared/pipes/display-with.pipe.mjs +0 -23
- package/esm2022/src/app/shared/pipes/duration.pipe.mjs +0 -34
- package/esm2022/src/app/shared/pipes/file-size.pipe.mjs +0 -53
- package/esm2022/src/app/shared/pipes/form.pipes.mjs +0 -219
- package/esm2022/src/app/shared/pipes/highlight.pipe.mjs +0 -50
- package/esm2022/src/app/shared/pipes/html.pipes.mjs +0 -102
- package/esm2022/src/app/shared/pipes/latlong-format.pipe.mjs +0 -42
- package/esm2022/src/app/shared/pipes/maps.pipe.mjs +0 -65
- package/esm2022/src/app/shared/pipes/maskito.pipe.mjs +0 -67
- package/esm2022/src/app/shared/pipes/math.pipes.mjs +0 -77
- package/esm2022/src/app/shared/pipes/ng-init.pipe.mjs +0 -19
- package/esm2022/src/app/shared/pipes/number-format.pipe.mjs +0 -17
- package/esm2022/src/app/shared/pipes/observable.pipes.mjs +0 -63
- package/esm2022/src/app/shared/pipes/pipes.module.mjs +0 -304
- package/esm2022/src/app/shared/pipes/property.pipes.mjs +0 -148
- package/esm2022/src/app/shared/pipes/selection.pipes.mjs +0 -177
- package/esm2022/src/app/shared/pipes/string.pipes.mjs +0 -183
- package/esm2022/src/app/shared/pipes/translate-context.pipe.mjs +0 -44
- package/esm2022/src/app/shared/pipes/types.pipes.mjs +0 -74
- package/esm2022/src/app/shared/pipes/url.pipes.mjs +0 -35
- package/esm2022/src/app/shared/platforms.mjs +0 -81
- package/esm2022/src/app/shared/print/print.service.mjs +0 -305
- package/esm2022/src/app/shared/regexps.mjs +0 -103
- package/esm2022/src/app/shared/rx-state/rx-state.decorators.mjs +0 -193
- package/esm2022/src/app/shared/rx-state/rx-state.module.mjs +0 -19
- package/esm2022/src/app/shared/rx-state/rx-state.types.mjs +0 -2
- package/esm2022/src/app/shared/services/entity-service.class.mjs +0 -44
- package/esm2022/src/app/shared/services/job.utils.mjs +0 -92
- package/esm2022/src/app/shared/services/memory-entity-service.class.mjs +0 -362
- package/esm2022/src/app/shared/services/progress-bar.service.mjs +0 -31
- package/esm2022/src/app/shared/services/startable-observable-service.class.mjs +0 -144
- package/esm2022/src/app/shared/services/startable-service.class.mjs +0 -118
- package/esm2022/src/app/shared/services/translate-context.service.mjs +0 -107
- package/esm2022/src/app/shared/services/validator-service.class.mjs +0 -2
- package/esm2022/src/app/shared/services.mjs +0 -85
- package/esm2022/src/app/shared/shared-routing.module.mjs +0 -72
- package/esm2022/src/app/shared/shared.module.mjs +0 -229
- package/esm2022/src/app/shared/shared.testing.module.mjs +0 -108
- package/esm2022/src/app/shared/storage/storage-explorer.component.mjs +0 -281
- package/esm2022/src/app/shared/storage/storage-explorer.module.mjs +0 -42
- package/esm2022/src/app/shared/storage/storage-explorer.testing-routing.module.mjs +0 -27
- package/esm2022/src/app/shared/storage/storage-explorer.testing.module.mjs +0 -25
- package/esm2022/src/app/shared/storage/storage.service.mjs +0 -134
- package/esm2022/src/app/shared/storage/storage.utils.mjs +0 -13
- package/esm2022/src/app/shared/testing/maskito.test.mjs +0 -41
- package/esm2022/src/app/shared/testing/observable.test.mjs +0 -94
- package/esm2022/src/app/shared/testing/tests.page.mjs +0 -37
- package/esm2022/src/app/shared/toast/toast.testing.mjs +0 -54
- package/esm2022/src/app/shared/toast/toast.testing.module.mjs +0 -31
- package/esm2022/src/app/shared/toast/toasts.mjs +0 -134
- package/esm2022/src/app/shared/toolbar/modal-toolbar.mjs +0 -62
- package/esm2022/src/app/shared/toolbar/toolbar.mjs +0 -272
- package/esm2022/src/app/shared/toolbar/toolbar.module.mjs +0 -26
- package/esm2022/src/app/shared/types.mjs +0 -2
- package/esm2022/src/app/shared/upload-file/testing/upload-file.testing.mjs +0 -59
- package/esm2022/src/app/shared/upload-file/testing/upload-file.testing.module.mjs +0 -30
- package/esm2022/src/app/shared/upload-file/upload-file-popover.component.mjs +0 -108
- package/esm2022/src/app/shared/upload-file/upload-file.component.mjs +0 -216
- package/esm2022/src/app/shared/upload-file/upload-file.model.mjs +0 -26
- package/esm2022/src/app/shared/validator/form-error-adapter.class.mjs +0 -122
- package/esm2022/src/app/shared/validator/validators.mjs +0 -658
- package/esm2022/src/app/shared/version/versions.mjs +0 -81
- package/esm2022/src/app/social/config/social.config.mjs +0 -9
- package/esm2022/src/app/social/feed/feed.component.mjs +0 -317
- package/esm2022/src/app/social/feed/feed.directive.mjs +0 -151
- package/esm2022/src/app/social/feed/feed.model.mjs +0 -60
- package/esm2022/src/app/social/feed/feed.module.mjs +0 -24
- package/esm2022/src/app/social/feed/feed.page.mjs +0 -67
- package/esm2022/src/app/social/feed/feed.service.mjs +0 -222
- package/esm2022/src/app/social/feed/testing/feed.testing.mjs +0 -66
- package/esm2022/src/app/social/feed/testing/feed.testing.module.mjs +0 -23
- package/esm2022/src/app/social/job/job.module.mjs +0 -23
- package/esm2022/src/app/social/job/progression/job-progression.component.mjs +0 -38
- package/esm2022/src/app/social/job/progression/job-progression.icon.mjs +0 -241
- package/esm2022/src/app/social/job/progression/job-progression.list.mjs +0 -50
- package/esm2022/src/app/social/job/progression/job-progression.model.mjs +0 -29
- package/esm2022/src/app/social/job/progression/job-progression.service.mjs +0 -91
- package/esm2022/src/app/social/job/testing/job-progression.testing.mjs +0 -38
- package/esm2022/src/app/social/job/testing/job-progression.testing.service.mjs +0 -34
- package/esm2022/src/app/social/job/testing/job.testing.module.mjs +0 -22
- package/esm2022/src/app/social/message/message.form.mjs +0 -149
- package/esm2022/src/app/social/message/message.modal.mjs +0 -116
- package/esm2022/src/app/social/message/message.model.mjs +0 -86
- package/esm2022/src/app/social/message/message.module.mjs +0 -25
- package/esm2022/src/app/social/message/message.service.mjs +0 -198
- package/esm2022/src/app/social/social.errors.mjs +0 -11
- package/esm2022/src/app/social/social.module.mjs +0 -22
- package/esm2022/src/app/social/social.testing.module.mjs +0 -72
- package/esm2022/src/app/social/user-event/notification/user-event-notification.icon.mjs +0 -191
- package/esm2022/src/app/social/user-event/notification/user-event-notification.list.mjs +0 -280
- package/esm2022/src/app/social/user-event/notification/user-event-notification.modal.mjs +0 -84
- package/esm2022/src/app/social/user-event/testing/user-event.testing.mjs +0 -273
- package/esm2022/src/app/social/user-event/testing/user-event.testing.model.mjs +0 -129
- package/esm2022/src/app/social/user-event/testing/user-event.testing.module.mjs +0 -22
- package/esm2022/src/app/social/user-event/testing/user-event.testing.service.mjs +0 -128
- package/esm2022/src/app/social/user-event/user-event.model.mjs +0 -2
- package/esm2022/src/app/social/user-event/user-event.module.mjs +0 -24
- package/esm2022/src/app/social/user-event/user-event.service.mjs +0 -512
- package/esm2022/src/environments/environment.class.mjs +0 -69
- package/esm2022/src/environments/environment.loader.mjs +0 -77
- package/esm2022/src/environments/environment.mjs +0 -133
- package/esm2022/sumaris-net.ngx-components.mjs +0 -5
- package/index.d.ts +0 -5
- package/public_api.d.ts +0 -364
- package/src/app/admin/admin-routing.module.d.ts +0 -9
- package/src/app/admin/admin.module.d.ts +0 -10
- package/src/app/admin/users/person.filter.d.ts +0 -37
- package/src/app/admin/users/person.service.d.ts +0 -61
- package/src/app/admin/users/person.validator.d.ts +0 -23
- package/src/app/admin/users/users-select.modal.d.ts +0 -73
- package/src/app/admin/users/users.d.ts +0 -89
- package/src/app/admin/users/users.module.d.ts +0 -15
- package/src/app/admin/users/users.utils.d.ts +0 -6
- package/src/app/core/about/about.modal.d.ts +0 -44
- package/src/app/core/about/about.module.d.ts +0 -11
- package/src/app/core/account/account.module.d.ts +0 -18
- package/src/app/core/account/account.page.d.ts +0 -93
- package/src/app/core/account/new-token.form.d.ts +0 -30
- package/src/app/core/account/new-token.modal.d.ts +0 -29
- package/src/app/core/account/password/change-password.form.d.ts +0 -19
- package/src/app/core/account/password/change-password.module.d.ts +0 -11
- package/src/app/core/account/password/change-password.page.d.ts +0 -36
- package/src/app/core/account/token.table.d.ts +0 -28
- package/src/app/core/auth/auth.form.d.ts +0 -38
- package/src/app/core/auth/auth.modal.d.ts +0 -24
- package/src/app/core/auth/auth.module.d.ts +0 -13
- package/src/app/core/auth/reset-password.modal.d.ts +0 -29
- package/src/app/core/core.module.d.ts +0 -28
- package/src/app/core/core.testing.module.d.ts +0 -16
- package/src/app/core/form/array/form-array.d.ts +0 -147
- package/src/app/core/form/array/testing/form-array-test.module.d.ts +0 -12
- package/src/app/core/form/array/testing/form-array.test.d.ts +0 -27
- package/src/app/core/form/buttons/form-buttons-bar.component.d.ts +0 -38
- package/src/app/core/form/buttons/form-buttons-bar.module.d.ts +0 -9
- package/src/app/core/form/entity/editor.class.d.ts +0 -85
- package/src/app/core/form/entity/entity-editor-modal.class.d.ts +0 -126
- package/src/app/core/form/entity/entity-editor.class.d.ts +0 -211
- package/src/app/core/form/entity/entity-metadata.component.d.ts +0 -13
- package/src/app/core/form/entity/entity.module.d.ts +0 -11
- package/src/app/core/form/entity/tab-editor.class.d.ts +0 -66
- package/src/app/core/form/form-container.class.d.ts +0 -131
- package/src/app/core/form/form.class.d.ts +0 -148
- package/src/app/core/form/form.module.d.ts +0 -15
- package/src/app/core/form/form.utils.d.ts +0 -301
- package/src/app/core/form/list/list.form.d.ts +0 -83
- package/src/app/core/form/list/list.module.d.ts +0 -9
- package/src/app/core/form/properties/properties-file.service.d.ts +0 -26
- package/src/app/core/form/properties/properties.form.d.ts +0 -109
- package/src/app/core/form/properties/properties.module.d.ts +0 -18
- package/src/app/core/form/properties/properties.table.d.ts +0 -73
- package/src/app/core/form/properties/properties.utils.d.ts +0 -59
- package/src/app/core/form/properties/property.validator.d.ts +0 -27
- package/src/app/core/form/properties/testing/properties-form.test.d.ts +0 -22
- package/src/app/core/form/properties/testing/properties-form.testing.module.d.ts +0 -12
- package/src/app/core/form/text/testing/text-form.testing.d.ts +0 -19
- package/src/app/core/form/text/testing/text-form.testing.module.d.ts +0 -14
- package/src/app/core/form/text/text-form.component.d.ts +0 -66
- package/src/app/core/form/text/text-form.module.d.ts +0 -9
- package/src/app/core/form/text-popover/testing/text-popover.testing.d.ts +0 -16
- package/src/app/core/form/text-popover/testing/text-popover.testing.module.d.ts +0 -11
- package/src/app/core/form/text-popover/text-popover.component.d.ts +0 -81
- package/src/app/core/form/text-popover/text-popover.module.d.ts +0 -10
- package/src/app/core/form/username/username.form.d.ts +0 -19
- package/src/app/core/form/username/username.module.d.ts +0 -9
- package/src/app/core/graphql/graphql.module.d.ts +0 -7
- package/src/app/core/graphql/graphql.service.d.ts +0 -138
- package/src/app/core/graphql/graphql.utils.d.ts +0 -47
- package/src/app/core/home/home.d.ts +0 -135
- package/src/app/core/home/home.module.d.ts +0 -16
- package/src/app/core/icon/icon.component.d.ts +0 -13
- package/src/app/core/icon/icon.module.d.ts +0 -9
- package/src/app/core/install/install-upgrade-card.component.d.ts +0 -63
- package/src/app/core/install/install-upgrade-card.module.d.ts +0 -9
- package/src/app/core/menu/menu.component.d.ts +0 -69
- package/src/app/core/menu/menu.model.d.ts +0 -161
- package/src/app/core/menu/menu.module.d.ts +0 -12
- package/src/app/core/menu/menu.service.d.ts +0 -93
- package/src/app/core/menu/sub-menu-tab.directive.d.ts +0 -29
- package/src/app/core/menu/testing/menu-other.testing.d.ts +0 -15
- package/src/app/core/menu/testing/menu.testing.d.ts +0 -53
- package/src/app/core/menu/testing/menu.testing.module.d.ts +0 -12
- package/src/app/core/offline/update-offline-mode-card.component.d.ts +0 -11
- package/src/app/core/offline/update-offline-mode-card.module.d.ts +0 -9
- package/src/app/core/peer/select-peer.modal.d.ts +0 -108
- package/src/app/core/peer/select-peer.module.d.ts +0 -12
- package/src/app/core/register/register-confirm.page.d.ts +0 -24
- package/src/app/core/register/register.form.d.ts +0 -49
- package/src/app/core/register/register.modal.d.ts +0 -14
- package/src/app/core/register/register.module.d.ts +0 -12
- package/src/app/core/services/account.service.d.ts +0 -236
- package/src/app/core/services/auth-guard.service.d.ts +0 -17
- package/src/app/core/services/base-entity-service.class.d.ts +0 -134
- package/src/app/core/services/base-graphql-service.class.d.ts +0 -77
- package/src/app/core/services/base58.d.ts +0 -7
- package/src/app/core/services/config/core.config.d.ts +0 -52
- package/src/app/core/services/config.service.d.ts +0 -63
- package/src/app/core/services/crypto.service.d.ts +0 -44
- package/src/app/core/services/errors.d.ts +0 -65
- package/src/app/core/services/local-settings.service.d.ts +0 -130
- package/src/app/core/services/model/account.model.d.ts +0 -66
- package/src/app/core/services/model/config.model.d.ts +0 -37
- package/src/app/core/services/model/department.model.d.ts +0 -11
- package/src/app/core/services/model/entity.decorators.d.ts +0 -17
- package/src/app/core/services/model/entity.model.d.ts +0 -133
- package/src/app/core/services/model/filter.model.d.ts +0 -37
- package/src/app/core/services/model/history.model.d.ts +0 -20
- package/src/app/core/services/model/model.enum.d.ts +0 -7
- package/src/app/core/services/model/node-feature.model.d.ts +0 -14
- package/src/app/core/services/model/peer.model.d.ts +0 -35
- package/src/app/core/services/model/person.model.d.ts +0 -39
- package/src/app/core/services/model/referential.model.d.ts +0 -85
- package/src/app/core/services/model/settings.model.d.ts +0 -29
- package/src/app/core/services/model/token.model.d.ts +0 -22
- package/src/app/core/services/model/tree-item-entity.model.d.ts +0 -62
- package/src/app/core/services/network.service.d.ts +0 -184
- package/src/app/core/services/network.types.d.ts +0 -4
- package/src/app/core/services/network.utils.d.ts +0 -39
- package/src/app/core/services/pipes/account.pipes.d.ts +0 -15
- package/src/app/core/services/pipes/department-to-string.pipe.d.ts +0 -9
- package/src/app/core/services/pipes/person-to-string.pipe.d.ts +0 -11
- package/src/app/core/services/pipes/pipes.module.d.ts +0 -11
- package/src/app/core/services/pipes/referential-to-string.pipe.d.ts +0 -23
- package/src/app/core/services/pipes/usage-mode.pipes.d.ts +0 -15
- package/src/app/core/services/platform.service.d.ts +0 -126
- package/src/app/core/services/storage/entities-storage.service.d.ts +0 -108
- package/src/app/core/services/storage/entity-store.class.d.ts +0 -113
- package/src/app/core/services/testing/referential-filter.model.d.ts +0 -9
- package/src/app/core/services/testing/referential.validator.d.ts +0 -22
- package/src/app/core/services/validator/account.validator.d.ts +0 -24
- package/src/app/core/services/validator/base.validator.class.d.ts +0 -21
- package/src/app/core/services/validator/local-settings.validator.d.ts +0 -18
- package/src/app/core/services/validator/user-settings.validator.d.ts +0 -10
- package/src/app/core/services/validator/user-token.validator.d.ts +0 -13
- package/src/app/core/settings/settings.module.d.ts +0 -12
- package/src/app/core/settings/settings.page.d.ts +0 -89
- package/src/app/core/table/async-table.class.d.ts +0 -443
- package/src/app/core/table/column/actions-column.component.d.ts +0 -65
- package/src/app/core/table/column/nav-actions-column.component.d.ts +0 -47
- package/src/app/core/table/column/row-field.component.d.ts +0 -41
- package/src/app/core/table/entities-async-table-datasource.class.d.ts +0 -103
- package/src/app/core/table/entities-table-datasource.class.d.ts +0 -102
- package/src/app/core/table/memory-table.class.d.ts +0 -25
- package/src/app/core/table/table-select-columns.component.d.ts +0 -22
- package/src/app/core/table/table.class.d.ts +0 -427
- package/src/app/core/table/table.model.d.ts +0 -31
- package/src/app/core/table/table.module.d.ts +0 -14
- package/src/app/core/table/table.pipes.d.ts +0 -30
- package/src/app/core/table/table.utils.d.ts +0 -13
- package/src/app/core/table/testing/nested-table.testing.d.ts +0 -54
- package/src/app/core/table/testing/table-validator.service.d.ts +0 -17
- package/src/app/core/table/testing/table.testing.d.ts +0 -55
- package/src/app/core/table/testing/table.testing.module.d.ts +0 -17
- package/src/app/core/table/testing/table2-validator.service.d.ts +0 -15
- package/src/app/core/table/testing/table2.testing.d.ts +0 -53
- package/src/app/shared/alerts.d.ts +0 -55
- package/src/app/shared/audio/audio.d.ts +0 -45
- package/src/app/shared/audio/audio.testing.d.ts +0 -13
- package/src/app/shared/audio/audio.testing.module.d.ts +0 -11
- package/src/app/shared/base64.utils.d.ts +0 -7
- package/src/app/shared/capacitor/keyboard.d.ts +0 -16
- package/src/app/shared/capacitor/plugins.d.ts +0 -8
- package/src/app/shared/constants.d.ts +0 -15
- package/src/app/shared/dates.d.ts +0 -85
- package/src/app/shared/debug/debug-service.class.d.ts +0 -5
- package/src/app/shared/debug/debug.component.d.ts +0 -14
- package/src/app/shared/debug/debug.module.d.ts +0 -12
- package/src/app/shared/directives/autofocus.directive.d.ts +0 -18
- package/src/app/shared/directives/autoresize.directive.d.ts +0 -20
- package/src/app/shared/directives/autotitle.directive.d.ts +0 -11
- package/src/app/shared/directives/autotooltip.directive.d.ts +0 -14
- package/src/app/shared/directives/cell-selection/cell-identifier.directive.d.ts +0 -17
- package/src/app/shared/directives/cell-selection/cell-selection.directive.d.ts +0 -29
- package/src/app/shared/directives/cell-selection/cell-selection.service.d.ts +0 -67
- package/src/app/shared/directives/directives.module.d.ts +0 -15
- package/src/app/shared/directives/drag-and-drop.directive.d.ts +0 -17
- package/src/app/shared/directives/ng-var.directive.d.ts +0 -12
- package/src/app/shared/directives/resizable/resizable.component.d.ts +0 -21
- package/src/app/shared/directives/resizable/resizable.directive.d.ts +0 -24
- package/src/app/shared/directives/resizable/resizable.module.d.ts +0 -8
- package/src/app/shared/directives/throttled-click.directive.d.ts +0 -16
- package/src/app/shared/events.d.ts +0 -29
- package/src/app/shared/file/csv.utils.d.ts +0 -30
- package/src/app/shared/file/file.service.d.ts +0 -23
- package/src/app/shared/file/file.utils.d.ts +0 -20
- package/src/app/shared/file/images.utils.d.ts +0 -36
- package/src/app/shared/file/json.utils.d.ts +0 -19
- package/src/app/shared/file/uri.utils.d.ts +0 -8
- package/src/app/shared/file/url.utils.d.ts +0 -108
- package/src/app/shared/focusable.d.ts +0 -4
- package/src/app/shared/form/field.component.d.ts +0 -83
- package/src/app/shared/form/field.model.d.ts +0 -88
- package/src/app/shared/form/loading-spinner.d.ts +0 -8
- package/src/app/shared/forms.d.ts +0 -170
- package/src/app/shared/functions.d.ts +0 -296
- package/src/app/shared/geolocation/geolocation.utils.d.ts +0 -25
- package/src/app/shared/gesture/gesture-config.d.ts +0 -15
- package/src/app/shared/gesture/hammer.utils.d.ts +0 -16
- package/src/app/shared/graph/colors.utils.d.ts +0 -34
- package/src/app/shared/graph/graph-colors.d.ts +0 -63
- package/src/app/shared/guard/component-dirty.guard.d.ts +0 -14
- package/src/app/shared/hotkeys/dialog/hotkeys-dialog.component.d.ts +0 -8
- package/src/app/shared/hotkeys/hotkeys.service.d.ts +0 -48
- package/src/app/shared/hotkeys/shared-hotkeys.module.d.ts +0 -11
- package/src/app/shared/html.utils.d.ts +0 -17
- package/src/app/shared/http/http.utils.d.ts +0 -28
- package/src/app/shared/image/gallery/image-gallery.component.d.ts +0 -139
- package/src/app/shared/image/gallery/image-gallery.module.d.ts +0 -15
- package/src/app/shared/image/gallery/testing/gallegry.model.testing.d.ts +0 -12
- package/src/app/shared/image/gallery/testing/gallery.service.testing.d.ts +0 -33
- package/src/app/shared/image/gallery/testing/gallery.testing.d.ts +0 -16
- package/src/app/shared/image/gallery/testing/gallery.testing.module.d.ts +0 -13
- package/src/app/shared/image/image.model.d.ts +0 -7
- package/src/app/shared/image/image.module.d.ts +0 -8
- package/src/app/shared/image/image.service.d.ts +0 -37
- package/src/app/shared/inputs.d.ts +0 -37
- package/src/app/shared/interceptors/progess.interceptor.d.ts +0 -8
- package/src/app/shared/logging/log-level.model.d.ts +0 -38
- package/src/app/shared/logging/logger.model.d.ts +0 -68
- package/src/app/shared/logging/logging-service.class.d.ts +0 -39
- package/src/app/shared/logging/logging-service.config.d.ts +0 -25
- package/src/app/shared/logging/logging-service.module.d.ts +0 -9
- package/src/app/shared/markdown/markdown.component.d.ts +0 -86
- package/src/app/shared/markdown/markdown.directive.d.ts +0 -13
- package/src/app/shared/markdown/markdown.modal.d.ts +0 -49
- package/src/app/shared/markdown/markdown.module.d.ts +0 -17
- package/src/app/shared/markdown/markdown.service.d.ts +0 -37
- package/src/app/shared/markdown/markdown.utils.d.ts +0 -26
- package/src/app/shared/markdown/testing/markdown.test.d.ts +0 -18
- package/src/app/shared/markdown/testing/markdown.testing.module.d.ts +0 -12
- package/src/app/shared/material/autocomplete/material.autocomplete.config.d.ts +0 -74
- package/src/app/shared/material/autocomplete/material.autocomplete.d.ts +0 -224
- package/src/app/shared/material/autocomplete/material.autocomplete.module.d.ts +0 -23
- package/src/app/shared/material/autocomplete/material.autocomplete.utils.d.ts +0 -5
- package/src/app/shared/material/autocomplete/testing/autocomplete.test.d.ts +0 -48
- package/src/app/shared/material/badge/badge.directive.d.ts +0 -43
- package/src/app/shared/material/badge/badge.module.d.ts +0 -10
- package/src/app/shared/material/badge/badge.test.d.ts +0 -21
- package/src/app/shared/material/boolean/boolean.module.d.ts +0 -21
- package/src/app/shared/material/boolean/material.boolean.d.ts +0 -94
- package/src/app/shared/material/boolean/testing/boolean.test.page.d.ts +0 -31
- package/src/app/shared/material/chips/chips.module.d.ts +0 -22
- package/src/app/shared/material/chips/material.chips.d.ts +0 -167
- package/src/app/shared/material/chips/testing/chips.test.d.ts +0 -32
- package/src/app/shared/material/datetime/datetime.module.d.ts +0 -25
- package/src/app/shared/material/datetime/material.date.d.ts +0 -90
- package/src/app/shared/material/datetime/material.dateshort.d.ts +0 -89
- package/src/app/shared/material/datetime/material.datetime.d.ts +0 -136
- package/src/app/shared/material/datetime/testing/mat-date-time.test.d.ts +0 -38
- package/src/app/shared/material/datetime/testing/mat-date.test.d.ts +0 -32
- package/src/app/shared/material/datetime/testing/mat-dateshort.test.d.ts +0 -31
- package/src/app/shared/material/duration/duration.module.d.ts +0 -18
- package/src/app/shared/material/duration/duration.utils.d.ts +0 -2
- package/src/app/shared/material/duration/material.duration.d.ts +0 -64
- package/src/app/shared/material/duration/testing/mat-duration.test.d.ts +0 -23
- package/src/app/shared/material/latlong/latlong.utils.d.ts +0 -59
- package/src/app/shared/material/latlong/material.latlong-input.d.ts +0 -93
- package/src/app/shared/material/latlong/material.latlong.d.ts +0 -90
- package/src/app/shared/material/latlong/material.latlong.module.d.ts +0 -21
- package/src/app/shared/material/latlong/testing/latlong.test.d.ts +0 -30
- package/src/app/shared/material/material.animations.d.ts +0 -8
- package/src/app/shared/material/material.config.d.ts +0 -3
- package/src/app/shared/material/material.module.d.ts +0 -44
- package/src/app/shared/material/material.testing.module.d.ts +0 -31
- package/src/app/shared/material/paginator/material.paginator-i18n.d.ts +0 -12
- package/src/app/shared/material/stepper/material.stepper-i18n.d.ts +0 -10
- package/src/app/shared/material/swipe/material.swipe.d.ts +0 -89
- package/src/app/shared/material/swipe/swipe.module.d.ts +0 -16
- package/src/app/shared/material/swipe/testing/swipe.test.d.ts +0 -22
- package/src/app/shared/material/test/test-component.d.ts +0 -75
- package/src/app/shared/material/testing/common.test.d.ts +0 -25
- package/src/app/shared/modules.d.ts +0 -9
- package/src/app/shared/named-filter/named-filter-selector.component.d.ts +0 -63
- package/src/app/shared/named-filter/named-filter.model.d.ts +0 -32
- package/src/app/shared/named-filter/named-filter.module.d.ts +0 -13
- package/src/app/shared/named-filter/named-filter.service.d.ts +0 -32
- package/src/app/shared/named-filter/testing/named-filter-selector.testing.d.ts +0 -34
- package/src/app/shared/named-filter/testing/named-filter.testing.module.d.ts +0 -10
- package/src/app/shared/observables.d.ts +0 -32
- package/src/app/shared/pipes/arrays.pipe.d.ts +0 -93
- package/src/app/shared/pipes/badge.pipes.d.ts +0 -7
- package/src/app/shared/pipes/colors.pipe.d.ts +0 -9
- package/src/app/shared/pipes/date-diff-duration.pipe.d.ts +0 -25
- package/src/app/shared/pipes/date-format.pipe.d.ts +0 -35
- package/src/app/shared/pipes/date-from-now.pipe.d.ts +0 -14
- package/src/app/shared/pipes/date-from.pipe.d.ts +0 -14
- package/src/app/shared/pipes/dates.pipe.d.ts +0 -10
- package/src/app/shared/pipes/display-with.pipe.d.ts +0 -8
- package/src/app/shared/pipes/duration.pipe.d.ts +0 -23
- package/src/app/shared/pipes/file-size.pipe.d.ts +0 -30
- package/src/app/shared/pipes/form.pipes.d.ts +0 -69
- package/src/app/shared/pipes/highlight.pipe.d.ts +0 -11
- package/src/app/shared/pipes/html.pipes.d.ts +0 -50
- package/src/app/shared/pipes/latlong-format.pipe.d.ts +0 -20
- package/src/app/shared/pipes/maps.pipe.d.ts +0 -23
- package/src/app/shared/pipes/maskito.pipe.d.ts +0 -14
- package/src/app/shared/pipes/math.pipes.d.ts +0 -26
- package/src/app/shared/pipes/ng-init.pipe.d.ts +0 -8
- package/src/app/shared/pipes/number-format.pipe.d.ts +0 -9
- package/src/app/shared/pipes/observable.pipes.d.ts +0 -25
- package/src/app/shared/pipes/pipes.module.d.ts +0 -36
- package/src/app/shared/pipes/property.pipes.d.ts +0 -46
- package/src/app/shared/pipes/selection.pipes.d.ts +0 -62
- package/src/app/shared/pipes/string.pipes.d.ts +0 -67
- package/src/app/shared/pipes/translate-context.pipe.d.ts +0 -17
- package/src/app/shared/pipes/types.pipes.d.ts +0 -30
- package/src/app/shared/pipes/url.pipes.d.ts +0 -27
- package/src/app/shared/platforms.d.ts +0 -16
- package/src/app/shared/print/print.service.d.ts +0 -94
- package/src/app/shared/regexps.d.ts +0 -32
- package/src/app/shared/rx-state/rx-state.decorators.d.ts +0 -33
- package/src/app/shared/rx-state/rx-state.module.d.ts +0 -10
- package/src/app/shared/rx-state/rx-state.types.d.ts +0 -5
- package/src/app/shared/services/entity-service.class.d.ts +0 -117
- package/src/app/shared/services/job.utils.d.ts +0 -20
- package/src/app/shared/services/memory-entity-service.class.d.ts +0 -87
- package/src/app/shared/services/progress-bar.service.d.ts +0 -20
- package/src/app/shared/services/startable-observable-service.class.d.ts +0 -51
- package/src/app/shared/services/startable-service.class.d.ts +0 -38
- package/src/app/shared/services/translate-context.service.d.ts +0 -29
- package/src/app/shared/services/validator-service.class.d.ts +0 -11
- package/src/app/shared/services.d.ts +0 -17
- package/src/app/shared/shared-routing.module.d.ts +0 -23
- package/src/app/shared/shared.module.d.ts +0 -33
- package/src/app/shared/shared.testing.module.d.ts +0 -20
- package/src/app/shared/storage/storage-explorer.component.d.ts +0 -60
- package/src/app/shared/storage/storage-explorer.module.d.ts +0 -13
- package/src/app/shared/storage/storage-explorer.testing-routing.module.d.ts +0 -9
- package/src/app/shared/storage/storage-explorer.testing.module.d.ts +0 -10
- package/src/app/shared/storage/storage.service.d.ts +0 -22
- package/src/app/shared/storage/storage.utils.d.ts +0 -19
- package/src/app/shared/testing/maskito.test.d.ts +0 -10
- package/src/app/shared/testing/observable.test.d.ts +0 -25
- package/src/app/shared/testing/tests.page.d.ts +0 -20
- package/src/app/shared/toast/toast.testing.d.ts +0 -15
- package/src/app/shared/toast/toast.testing.module.d.ts +0 -12
- package/src/app/shared/toast/toasts.d.ts +0 -16
- package/src/app/shared/toolbar/modal-toolbar.d.ts +0 -21
- package/src/app/shared/toolbar/toolbar.d.ts +0 -70
- package/src/app/shared/toolbar/toolbar.module.d.ts +0 -16
- package/src/app/shared/types.d.ts +0 -45
- package/src/app/shared/upload-file/testing/upload-file.testing.d.ts +0 -21
- package/src/app/shared/upload-file/testing/upload-file.testing.module.d.ts +0 -11
- package/src/app/shared/upload-file/upload-file-popover.component.d.ts +0 -41
- package/src/app/shared/upload-file/upload-file.component.d.ts +0 -50
- package/src/app/shared/upload-file/upload-file.model.d.ts +0 -30
- package/src/app/shared/validator/form-error-adapter.class.d.ts +0 -33
- package/src/app/shared/validator/validators.d.ts +0 -145
- package/src/app/shared/version/versions.d.ts +0 -26
- package/src/app/social/config/social.config.d.ts +0 -4
- package/src/app/social/feed/feed.component.d.ts +0 -91
- package/src/app/social/feed/feed.directive.d.ts +0 -32
- package/src/app/social/feed/feed.model.d.ts +0 -69
- package/src/app/social/feed/feed.module.d.ts +0 -13
- package/src/app/social/feed/feed.page.d.ts +0 -24
- package/src/app/social/feed/feed.service.d.ts +0 -85
- package/src/app/social/feed/testing/feed.testing.d.ts +0 -23
- package/src/app/social/feed/testing/feed.testing.module.d.ts +0 -12
- package/src/app/social/job/job.module.d.ts +0 -12
- package/src/app/social/job/progression/job-progression.component.d.ts +0 -24
- package/src/app/social/job/progression/job-progression.icon.d.ts +0 -63
- package/src/app/social/job/progression/job-progression.list.d.ts +0 -20
- package/src/app/social/job/progression/job-progression.model.d.ts +0 -10
- package/src/app/social/job/progression/job-progression.service.d.ts +0 -33
- package/src/app/social/job/testing/job-progression.testing.d.ts +0 -13
- package/src/app/social/job/testing/job-progression.testing.service.d.ts +0 -9
- package/src/app/social/job/testing/job.testing.module.d.ts +0 -11
- package/src/app/social/message/message.form.d.ts +0 -36
- package/src/app/social/message/message.modal.d.ts +0 -47
- package/src/app/social/message/message.model.d.ts +0 -36
- package/src/app/social/message/message.module.d.ts +0 -14
- package/src/app/social/message/message.service.d.ts +0 -44
- package/src/app/social/social.errors.d.ts +0 -10
- package/src/app/social/social.module.d.ts +0 -20
- package/src/app/social/social.testing.module.d.ts +0 -14
- package/src/app/social/user-event/notification/user-event-notification.icon.d.ts +0 -47
- package/src/app/social/user-event/notification/user-event-notification.list.d.ts +0 -77
- package/src/app/social/user-event/notification/user-event-notification.modal.d.ts +0 -37
- package/src/app/social/user-event/testing/user-event.testing.d.ts +0 -44
- package/src/app/social/user-event/testing/user-event.testing.model.d.ts +0 -42
- package/src/app/social/user-event/testing/user-event.testing.module.d.ts +0 -11
- package/src/app/social/user-event/testing/user-event.testing.service.d.ts +0 -41
- package/src/app/social/user-event/user-event.model.d.ts +0 -44
- package/src/app/social/user-event/user-event.module.d.ts +0 -14
- package/src/app/social/user-event/user-event.service.d.ts +0 -124
- package/src/environments/environment.class.d.ts +0 -86
- package/src/environments/environment.d.ts +0 -2
- package/src/environments/environment.loader.d.ts +0 -24
- 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
|