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