sonamu 0.8.26 → 0.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/cli.js +60 -13
- package/dist/_virtual/rolldown_runtime.js +39 -0
- package/dist/ai/agents/agent.d.ts +3 -3
- package/dist/ai/agents/agent.d.ts.map +1 -1
- package/dist/ai/agents/agent.js +76 -73
- package/dist/ai/agents/index.js +3 -3
- package/dist/ai/agents/types.d.ts +3 -3
- package/dist/ai/agents/types.d.ts.map +1 -1
- package/dist/ai/agents/types.js +1 -3
- package/dist/ai/index.js +3 -2
- package/dist/ai/providers/rtzr/api.js +25 -25
- package/dist/ai/providers/rtzr/error.js +25 -26
- package/dist/ai/providers/rtzr/index.js +5 -5
- package/dist/ai/providers/rtzr/model.d.ts +1 -1
- package/dist/ai/providers/rtzr/model.d.ts.map +1 -1
- package/dist/ai/providers/rtzr/model.js +117 -133
- package/dist/ai/providers/rtzr/options.d.ts.map +1 -1
- package/dist/ai/providers/rtzr/options.js +35 -41
- package/dist/ai/providers/rtzr/provider.d.ts +1 -1
- package/dist/ai/providers/rtzr/provider.d.ts.map +1 -1
- package/dist/ai/providers/rtzr/provider.js +53 -51
- package/dist/ai/providers/rtzr/utils.d.ts.map +1 -1
- package/dist/ai/providers/rtzr/utils.js +84 -84
- package/dist/api/base-frame.d.ts +2 -2
- package/dist/api/base-frame.d.ts.map +1 -1
- package/dist/api/base-frame.js +29 -19
- package/dist/api/caster.d.ts +1 -1
- package/dist/api/caster.d.ts.map +1 -1
- package/dist/api/caster.js +51 -61
- package/dist/api/code-converters.d.ts +4 -3
- package/dist/api/code-converters.d.ts.map +1 -1
- package/dist/api/code-converters.js +226 -249
- package/dist/api/config.d.ts +17 -17
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +37 -30
- package/dist/api/context.d.ts +10 -10
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +8 -2
- package/dist/api/decorators.d.ts +8 -8
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +245 -268
- package/dist/api/index.js +39 -7
- package/dist/api/secret.js +22 -15
- package/dist/api/sonamu.d.ts +15 -15
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +1012 -1131
- package/dist/api/validator.js +88 -79
- package/dist/auth/auth-generator.d.ts.map +1 -1
- package/dist/auth/auth-generator.js +203 -200
- package/dist/auth/better-auth-entities.d.ts +2 -2
- package/dist/auth/better-auth-entities.d.ts.map +1 -1
- package/dist/auth/better-auth-entities.js +369 -429
- package/dist/auth/index.js +21 -6
- package/dist/auth/knex-adapter.d.ts +2 -2
- package/dist/auth/knex-adapter.d.ts.map +1 -1
- package/dist/auth/knex-adapter.js +153 -157
- package/dist/auth/plugins/entity-definitions/admin.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/admin.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/admin.js +58 -56
- package/dist/auth/plugins/entity-definitions/anonymous.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/anonymous.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/anonymous.js +20 -20
- package/dist/auth/plugins/entity-definitions/api-key.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/api-key.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/api-key.js +185 -196
- package/dist/auth/plugins/entity-definitions/index.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/index.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/index.js +26 -29
- package/dist/auth/plugins/entity-definitions/jwt.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/jwt.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/jwt.js +62 -64
- package/dist/auth/plugins/entity-definitions/organization.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/organization.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/organization.js +362 -421
- package/dist/auth/plugins/entity-definitions/passkey.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/passkey.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/passkey.js +115 -126
- package/dist/auth/plugins/entity-definitions/phone-number.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/phone-number.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/phone-number.js +31 -40
- package/dist/auth/plugins/entity-definitions/sso.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/sso.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/sso.js +94 -107
- package/dist/auth/plugins/entity-definitions/two-factor.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/two-factor.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/two-factor.js +78 -92
- package/dist/auth/plugins/entity-definitions/types.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/types.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/types.js +1 -10
- package/dist/auth/plugins/entity-definitions/username.d.ts +1 -1
- package/dist/auth/plugins/entity-definitions/username.d.ts.map +1 -1
- package/dist/auth/plugins/entity-definitions/username.js +31 -40
- package/dist/auth/plugins/index.js +12 -3
- package/dist/auth/plugins/wrappers/admin.d.ts +2 -2
- package/dist/auth/plugins/wrappers/admin.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/admin.js +28 -29
- package/dist/auth/plugins/wrappers/anonymous.d.ts +2 -1
- package/dist/auth/plugins/wrappers/anonymous.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/anonymous.js +23 -22
- package/dist/auth/plugins/wrappers/api-key.d.ts +2 -1
- package/dist/auth/plugins/wrappers/api-key.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/api-key.js +39 -34
- package/dist/auth/plugins/wrappers/index.js +11 -11
- package/dist/auth/plugins/wrappers/jwt.d.ts +2 -1
- package/dist/auth/plugins/wrappers/jwt.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/jwt.js +31 -26
- package/dist/auth/plugins/wrappers/organization.d.ts +2 -1
- package/dist/auth/plugins/wrappers/organization.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/organization.js +65 -62
- package/dist/auth/plugins/wrappers/passkey.d.ts +2 -1
- package/dist/auth/plugins/wrappers/passkey.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/passkey.js +33 -28
- package/dist/auth/plugins/wrappers/phone-number.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/phone-number.js +26 -23
- package/dist/auth/plugins/wrappers/sso.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/sso.js +37 -31
- package/dist/auth/plugins/wrappers/two-factor.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/two-factor.js +31 -28
- package/dist/auth/plugins/wrappers/username.d.ts.map +1 -1
- package/dist/auth/plugins/wrappers/username.js +23 -23
- package/dist/bin/build-config.js +31 -31
- package/dist/bin/cli.js +1063 -1204
- package/dist/bin/fixture.d.ts.map +1 -1
- package/dist/bin/fixture.js +266 -259
- package/dist/bin/hmr-hook-register.d.ts.map +1 -1
- package/dist/bin/hmr-hook-register.js +19 -18
- package/dist/bin/test-command.d.ts.map +1 -1
- package/dist/bin/test-command.js +180 -177
- package/dist/bin/ts-loader-register.js +13 -6
- package/dist/bin/ts-loader-registration.d.ts.map +1 -1
- package/dist/bin/ts-loader-registration.js +28 -38
- package/dist/cache/cache-manager.d.ts +1 -1
- package/dist/cache/cache-manager.d.ts.map +1 -1
- package/dist/cache/cache-manager.js +20 -15
- package/dist/cache/decorator.d.ts +1 -1
- package/dist/cache/decorator.d.ts.map +1 -1
- package/dist/cache/decorator.js +84 -76
- package/dist/cache/drivers.js +21 -34
- package/dist/cache/index.js +10 -7
- package/dist/cache/types.d.ts +2 -2
- package/dist/cache/types.d.ts.map +1 -1
- package/dist/cache/types.js +1 -6
- package/dist/cache-control/cache-control.d.ts +2 -2
- package/dist/cache-control/cache-control.d.ts.map +1 -1
- package/dist/cache-control/cache-control.js +106 -122
- package/dist/cache-control/types.d.ts +2 -2
- package/dist/cache-control/types.d.ts.map +1 -1
- package/dist/cache-control/types.js +1 -19
- package/dist/compress/compress.d.ts +1 -1
- package/dist/compress/compress.d.ts.map +1 -1
- package/dist/compress/compress.js +58 -56
- package/dist/compress/index.js +7 -2
- package/dist/compress/types.js +1 -11
- package/dist/cone/cone-generator.d.ts +1 -1
- package/dist/cone/cone-generator.d.ts.map +1 -1
- package/dist/cone/cone-generator.js +216 -219
- package/dist/database/_batch_update.d.ts +1 -1
- package/dist/database/_batch_update.d.ts.map +1 -1
- package/dist/database/_batch_update.js +107 -102
- package/dist/database/base-model.d.ts +8 -9
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +371 -392
- package/dist/database/base-model.types.d.ts +5 -5
- package/dist/database/base-model.types.d.ts.map +1 -1
- package/dist/database/base-model.types.js +1 -20
- package/dist/database/db.d.ts +5 -2
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +185 -171
- package/dist/database/knex.d.ts +1 -1
- package/dist/database/knex.d.ts.map +1 -1
- package/dist/database/knex.js +48 -42
- package/dist/database/puri-subset.types.d.ts +6 -7
- package/dist/database/puri-subset.types.d.ts.map +1 -1
- package/dist/database/puri-subset.types.js +1 -16
- package/dist/database/puri-wrapper.d.ts +6 -6
- package/dist/database/puri-wrapper.d.ts.map +1 -1
- package/dist/database/puri-wrapper.js +99 -101
- package/dist/database/puri.d.ts +4 -5
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +1021 -1227
- package/dist/database/puri.types.d.ts +6 -6
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +15 -6
- package/dist/database/transaction-context.d.ts +2 -2
- package/dist/database/transaction-context.d.ts.map +1 -1
- package/dist/database/transaction-context.js +22 -13
- package/dist/database/upsert-builder.d.ts +3 -3
- package/dist/database/upsert-builder.d.ts.map +1 -1
- package/dist/database/upsert-builder.js +405 -465
- package/dist/dict/en.js +72 -74
- package/dist/dict/index.js +13 -13
- package/dist/dict/ko.js +72 -74
- package/dist/dict/rc-keys.js +150 -168
- package/dist/dict/sd.d.ts +3 -1
- package/dist/dict/sd.d.ts.map +1 -1
- package/dist/dict/sd.js +54 -40
- package/dist/dict/sonamu-dictionary.d.ts +1 -1
- package/dist/dict/sonamu-dictionary.d.ts.map +1 -1
- package/dist/dict/sonamu-dictionary.js +887 -955
- package/dist/dict/types.js +1 -7
- package/dist/dict/utils.js +26 -24
- package/dist/entity/entity-manager.d.ts +9 -9
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +226 -223
- package/dist/entity/entity-template-cone.d.ts +1 -1
- package/dist/entity/entity-template-cone.d.ts.map +1 -1
- package/dist/entity/entity-template-cone.js +152 -151
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +952 -1089
- package/dist/exceptions/error-handler.d.ts +1 -1
- package/dist/exceptions/error-handler.d.ts.map +1 -1
- package/dist/exceptions/error-handler.js +32 -27
- package/dist/exceptions/so-exceptions.d.ts +1 -1
- package/dist/exceptions/so-exceptions.d.ts.map +1 -1
- package/dist/exceptions/so-exceptions.js +61 -68
- package/dist/filter/index.js +9 -3
- package/dist/filter/types.js +92 -88
- package/dist/filter/utils.d.ts +1 -1
- package/dist/filter/utils.d.ts.map +1 -1
- package/dist/filter/utils.js +147 -161
- package/dist/index.js +87 -40
- package/dist/logger/category.d.ts.map +1 -1
- package/dist/logger/category.js +30 -29
- package/dist/logger/configure.d.ts.map +1 -1
- package/dist/logger/configure.js +83 -107
- package/dist/migration/code-generation.d.ts +2 -2
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +1385 -1578
- package/dist/migration/migration-set.d.ts +1 -1
- package/dist/migration/migration-set.d.ts.map +1 -1
- package/dist/migration/migration-set.js +177 -227
- package/dist/migration/migrator.d.ts +4 -3
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +340 -345
- package/dist/migration/postgresql-schema-reader.d.ts +2 -2
- package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
- package/dist/migration/postgresql-schema-reader.js +506 -564
- package/dist/migration/slack-confirm.d.ts +2 -2
- package/dist/migration/slack-confirm.d.ts.map +1 -1
- package/dist/migration/slack-confirm.js +205 -193
- package/dist/migration/types.d.ts +2 -2
- package/dist/migration/types.d.ts.map +1 -1
- package/dist/migration/types.js +1 -3
- package/dist/naite/messaging-types.d.ts +1 -0
- package/dist/naite/messaging-types.d.ts.map +1 -1
- package/dist/naite/messaging-types.js +1 -7
- package/dist/naite/naite-reporter.d.ts +2 -2
- package/dist/naite/naite-reporter.d.ts.map +1 -1
- package/dist/naite/naite-reporter.js +127 -120
- package/dist/naite/naite.d.ts +3 -2
- package/dist/naite/naite.d.ts.map +1 -1
- package/dist/naite/naite.js +266 -300
- package/dist/ssr/index.d.ts +2 -2
- package/dist/ssr/index.d.ts.map +1 -1
- package/dist/ssr/index.js +13 -3
- package/dist/ssr/registry.d.ts +1 -1
- package/dist/ssr/registry.d.ts.map +1 -1
- package/dist/ssr/registry.js +45 -37
- package/dist/ssr/renderer.d.ts +4 -4
- package/dist/ssr/renderer.d.ts.map +1 -1
- package/dist/ssr/renderer.js +84 -91
- package/dist/ssr/types.d.ts +2 -2
- package/dist/ssr/types.d.ts.map +1 -1
- package/dist/ssr/types.js +1 -3
- package/dist/storage/base-file.js +54 -41
- package/dist/storage/buffered-file.d.ts +2 -2
- package/dist/storage/buffered-file.d.ts.map +1 -1
- package/dist/storage/buffered-file.js +51 -44
- package/dist/storage/drivers.d.ts +2 -2
- package/dist/storage/drivers.d.ts.map +1 -1
- package/dist/storage/drivers.js +12 -7
- package/dist/storage/index.js +14 -7
- package/dist/storage/s3-driver.d.ts +2 -2
- package/dist/storage/s3-driver.d.ts.map +1 -1
- package/dist/storage/s3-driver.js +52 -48
- package/dist/storage/storage-manager.d.ts +2 -2
- package/dist/storage/storage-manager.d.ts.map +1 -1
- package/dist/storage/storage-manager.js +33 -25
- package/dist/storage/types.d.ts +2 -2
- package/dist/storage/types.d.ts.map +1 -1
- package/dist/storage/types.js +1 -5
- package/dist/storage/uploaded-file.d.ts +1 -1
- package/dist/storage/uploaded-file.d.ts.map +1 -1
- package/dist/storage/uploaded-file.js +45 -35
- package/dist/stream/index.js +7 -2
- package/dist/stream/sse.d.ts +2 -2
- package/dist/stream/sse.d.ts.map +1 -1
- package/dist/stream/sse.js +72 -67
- package/dist/syncer/api-parser.d.ts +1 -1
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +224 -245
- package/dist/syncer/checksum.d.ts +1 -1
- package/dist/syncer/checksum.d.ts.map +1 -1
- package/dist/syncer/checksum.js +86 -72
- package/dist/syncer/code-generator.d.ts +2 -2
- package/dist/syncer/code-generator.d.ts.map +1 -1
- package/dist/syncer/code-generator.js +154 -160
- package/dist/syncer/entity-operations.d.ts +1 -1
- package/dist/syncer/entity-operations.d.ts.map +1 -1
- package/dist/syncer/entity-operations.js +63 -54
- package/dist/syncer/file-patterns.d.ts +1 -1
- package/dist/syncer/file-patterns.d.ts.map +1 -1
- package/dist/syncer/file-patterns.js +38 -38
- package/dist/syncer/index.js +19 -8
- package/dist/syncer/module-loader.d.ts +5 -5
- package/dist/syncer/module-loader.d.ts.map +1 -1
- package/dist/syncer/module-loader.js +83 -78
- package/dist/syncer/syncer-actions.d.ts +2 -2
- package/dist/syncer/syncer-actions.d.ts.map +1 -1
- package/dist/syncer/syncer-actions.js +76 -91
- package/dist/syncer/syncer.d.ts +7 -6
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +426 -492
- package/dist/tasks/decorator.d.ts +3 -3
- package/dist/tasks/decorator.d.ts.map +1 -1
- package/dist/tasks/decorator.js +32 -28
- package/dist/tasks/step-wrapper.d.ts +1 -1
- package/dist/tasks/step-wrapper.d.ts.map +1 -1
- package/dist/tasks/step-wrapper.js +42 -41
- package/dist/tasks/workflow-manager.d.ts +2 -2
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +192 -221
- package/dist/template/entity-converter.d.ts +1 -1
- package/dist/template/entity-converter.d.ts.map +1 -1
- package/dist/template/entity-converter.js +103 -103
- package/dist/template/helpers.d.ts.map +1 -1
- package/dist/template/helpers.js +163 -163
- package/dist/template/implementations/entity.template.d.ts +1 -1
- package/dist/template/implementations/entity.template.d.ts.map +1 -1
- package/dist/template/implementations/entity.template.js +76 -85
- package/dist/template/implementations/entry-server.template.d.ts +1 -1
- package/dist/template/implementations/entry-server.template.d.ts.map +1 -1
- package/dist/template/implementations/entry-server.template.js +32 -27
- package/dist/template/implementations/generated.template.d.ts +1 -1
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +254 -275
- package/dist/template/implementations/generated_http.template.d.ts +2 -2
- package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_http.template.js +114 -133
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_sso.template.js +249 -275
- package/dist/template/implementations/init_types.template.d.ts +1 -1
- package/dist/template/implementations/init_types.template.d.ts.map +1 -1
- package/dist/template/implementations/init_types.template.js +40 -34
- package/dist/template/implementations/model.template.d.ts +1 -1
- package/dist/template/implementations/model.template.d.ts.map +1 -1
- package/dist/template/implementations/model.template.js +56 -53
- package/dist/template/implementations/model_test.template.d.ts +1 -1
- package/dist/template/implementations/model_test.template.d.ts.map +1 -1
- package/dist/template/implementations/model_test.template.js +32 -24
- package/dist/template/implementations/queries.template.d.ts +1 -1
- package/dist/template/implementations/queries.template.d.ts.map +1 -1
- package/dist/template/implementations/queries.template.js +84 -89
- package/dist/template/implementations/sd.template.d.ts +1 -1
- package/dist/template/implementations/sd.template.d.ts.map +1 -1
- package/dist/template/implementations/sd.template.js +137 -144
- package/dist/template/implementations/services.template.d.ts +1 -1
- package/dist/template/implementations/services.template.d.ts.map +1 -1
- package/dist/template/implementations/services.template.js +164 -189
- package/dist/template/implementations/view_form.template.d.ts +1 -1
- package/dist/template/implementations/view_form.template.d.ts.map +1 -1
- package/dist/template/implementations/view_form.template.js +258 -285
- package/dist/template/implementations/view_id_all_select.template.d.ts +1 -1
- package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
- package/dist/template/implementations/view_id_all_select.template.js +31 -25
- package/dist/template/implementations/view_list.template.d.ts +1 -1
- package/dist/template/implementations/view_list.template.d.ts.map +1 -1
- package/dist/template/implementations/view_list.template.js +304 -355
- package/dist/template/implementations/view_search_input.template.d.ts +1 -1
- package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
- package/dist/template/implementations/view_search_input.template.js +31 -27
- package/dist/template/index.js +21 -7
- package/dist/template/template-manager.d.ts +1 -1
- package/dist/template/template-manager.d.ts.map +1 -1
- package/dist/template/template-manager.js +132 -123
- package/dist/template/template-types.js +8 -6
- package/dist/template/template.d.ts +2 -2
- package/dist/template/template.d.ts.map +1 -1
- package/dist/template/template.js +73 -68
- package/dist/template/zod-converter.d.ts.map +1 -1
- package/dist/template/zod-converter.js +603 -657
- package/dist/testing/_relation-graph.d.ts +1 -1
- package/dist/testing/_relation-graph.d.ts.map +1 -1
- package/dist/testing/_relation-graph.js +93 -88
- package/dist/testing/bootstrap.d.ts +22 -13
- package/dist/testing/bootstrap.d.ts.map +1 -1
- package/dist/testing/bootstrap.js +114 -114
- package/dist/testing/data-explorer.d.ts +3 -3
- package/dist/testing/data-explorer.d.ts.map +1 -1
- package/dist/testing/data-explorer.js +237 -265
- package/dist/testing/dev-test-routes.d.ts +2 -2
- package/dist/testing/dev-test-routes.d.ts.map +1 -1
- package/dist/testing/dev-test-routes.js +258 -249
- package/dist/testing/dev-vitest-manager.d.ts +1 -1
- package/dist/testing/dev-vitest-manager.d.ts.map +1 -1
- package/dist/testing/dev-vitest-manager.js +514 -539
- package/dist/testing/faker-mappings.js +422 -420
- package/dist/testing/fixture-generator.d.ts +3 -3
- package/dist/testing/fixture-generator.d.ts.map +1 -1
- package/dist/testing/fixture-generator.js +1216 -1346
- package/dist/testing/fixture-loader.js +26 -25
- package/dist/testing/fixture-manager.d.ts +3 -3
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +706 -776
- package/dist/testing/global-setup.js +53 -49
- package/dist/testing/index.js +19 -11
- package/dist/testing/naite-vitest-reporter.js +18 -13
- package/dist/testing/parallel-db-manager.d.ts +1 -1
- package/dist/testing/parallel-db-manager.d.ts.map +1 -1
- package/dist/testing/parallel-db-manager.js +63 -78
- package/dist/testing/vitest-helpers.d.ts +1 -1
- package/dist/testing/vitest-helpers.d.ts.map +1 -1
- package/dist/testing/vitest-helpers.js +37 -33
- package/dist/types/types.d.ts +28 -28
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +764 -890
- package/dist/ui/ai-api.d.ts +1 -1
- package/dist/ui/ai-api.d.ts.map +1 -1
- package/dist/ui/ai-api.js +52 -42
- package/dist/ui/ai-client.d.ts +1 -2
- package/dist/ui/ai-client.d.ts.map +1 -1
- package/dist/ui/ai-client.js +353 -388
- package/dist/ui/api.d.ts +1 -1
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +903 -1145
- package/dist/ui/cdd-service.d.ts +1 -1
- package/dist/ui/cdd-service.d.ts.map +1 -1
- package/dist/ui/cdd-service.js +406 -407
- package/dist/ui/cdd-types.js +1 -3
- package/dist/ui-web/assets/index-C-Zz-wYg.css +1 -0
- package/dist/ui-web/assets/index-DejDON8K.js +238 -0
- package/dist/ui-web/index.html +3 -3
- package/dist/utils/async-utils.js +57 -45
- package/dist/utils/console-util.d.ts.map +1 -1
- package/dist/utils/console-util.js +104 -87
- package/dist/utils/controller.js +26 -19
- package/dist/utils/esm-utils.js +49 -38
- package/dist/utils/formatter.d.ts +1 -2
- package/dist/utils/formatter.d.ts.map +1 -1
- package/dist/utils/formatter.js +89 -115
- package/dist/utils/fs-utils.d.ts.map +1 -1
- package/dist/utils/fs-utils.js +68 -65
- package/dist/utils/lodash-able.js +11 -4
- package/dist/utils/model.d.ts +1 -1
- package/dist/utils/model.d.ts.map +1 -1
- package/dist/utils/model.js +21 -19
- package/dist/utils/object-utils.js +148 -186
- package/dist/utils/path-utils.js +67 -57
- package/dist/utils/process-utils.d.ts.map +1 -1
- package/dist/utils/process-utils.js +37 -31
- package/dist/utils/sql-parser.d.ts +1 -1
- package/dist/utils/sql-parser.d.ts.map +1 -1
- package/dist/utils/sql-parser.js +40 -40
- package/dist/utils/type-utils.js +44 -43
- package/dist/utils/utils.d.ts +2 -3
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +81 -93
- package/dist/utils/zod-error.d.ts +1 -1
- package/dist/utils/zod-error.d.ts.map +1 -1
- package/dist/utils/zod-error.js +24 -17
- package/dist/vector/chunking.d.ts +1 -1
- package/dist/vector/chunking.d.ts.map +1 -1
- package/dist/vector/chunking.js +100 -94
- package/dist/vector/config.d.ts +1 -1
- package/dist/vector/config.d.ts.map +1 -1
- package/dist/vector/config.js +76 -78
- package/dist/vector/embedding.d.ts +1 -1
- package/dist/vector/embedding.d.ts.map +1 -1
- package/dist/vector/embedding.js +128 -125
- package/dist/vector/index.js +5 -5
- package/dist/vector/types.js +1 -5
- package/package.json +31 -36
- package/src/ai/agents/agent.ts +12 -5
- package/src/ai/agents/types.ts +5 -5
- package/src/ai/providers/rtzr/model.ts +8 -10
- package/src/ai/providers/rtzr/options.ts +2 -1
- package/src/ai/providers/rtzr/provider.ts +5 -3
- package/src/ai/providers/rtzr/utils.ts +2 -7
- package/src/api/__tests__/config.test.ts +15 -8
- package/src/api/base-frame.ts +5 -3
- package/src/api/caster.ts +7 -6
- package/src/api/code-converters.ts +23 -26
- package/src/api/config.ts +23 -17
- package/src/api/context.ts +18 -11
- package/src/api/decorators.ts +17 -18
- package/src/api/sonamu.ts +44 -49
- package/src/auth/auth-generator.ts +4 -6
- package/src/auth/better-auth-entities.ts +3 -2
- package/src/auth/knex-adapter.ts +6 -5
- package/src/auth/plugins/entity-definitions/admin.ts +1 -1
- package/src/auth/plugins/entity-definitions/anonymous.ts +1 -1
- package/src/auth/plugins/entity-definitions/api-key.ts +1 -1
- package/src/auth/plugins/entity-definitions/index.ts +1 -1
- package/src/auth/plugins/entity-definitions/jwt.ts +1 -1
- package/src/auth/plugins/entity-definitions/organization.ts +1 -1
- package/src/auth/plugins/entity-definitions/passkey.ts +1 -1
- package/src/auth/plugins/entity-definitions/phone-number.ts +1 -1
- package/src/auth/plugins/entity-definitions/sso.ts +1 -1
- package/src/auth/plugins/entity-definitions/two-factor.ts +1 -1
- package/src/auth/plugins/entity-definitions/types.ts +1 -1
- package/src/auth/plugins/entity-definitions/username.ts +1 -1
- package/src/auth/plugins/wrappers/admin.ts +3 -1
- package/src/auth/plugins/wrappers/anonymous.ts +3 -1
- package/src/auth/plugins/wrappers/api-key.ts +3 -1
- package/src/auth/plugins/wrappers/jwt.ts +3 -1
- package/src/auth/plugins/wrappers/organization.ts +3 -1
- package/src/auth/plugins/wrappers/passkey.ts +3 -1
- package/src/auth/plugins/wrappers/phone-number.ts +3 -1
- package/src/auth/plugins/wrappers/sso.ts +2 -1
- package/src/auth/plugins/wrappers/two-factor.ts +3 -1
- package/src/auth/plugins/wrappers/username.ts +3 -1
- package/src/bin/__tests__/ts-loader-register.test.ts +7 -12
- package/src/bin/build-config.ts +3 -3
- package/src/bin/cli.ts +27 -25
- package/src/bin/fixture.ts +4 -2
- package/src/bin/hmr-hook-register.ts +1 -0
- package/src/bin/test-command.ts +4 -2
- package/src/bin/ts-loader-registration.ts +6 -22
- package/src/cache/cache-manager.ts +2 -1
- package/src/cache/decorator.ts +2 -2
- package/src/cache/types.ts +3 -3
- package/src/cache-control/cache-control.ts +3 -2
- package/src/cache-control/types.ts +2 -2
- package/src/compress/compress.ts +1 -1
- package/src/cone/cone-generator.ts +5 -3
- package/src/database/_batch_update.ts +1 -1
- package/src/database/base-model.ts +20 -14
- package/src/database/base-model.types.ts +12 -11
- package/src/database/db.ts +56 -21
- package/src/database/knex.ts +2 -2
- package/src/database/puri-subset.test-d.ts +33 -32
- package/src/database/puri-subset.types.ts +6 -7
- package/src/database/puri-wrapper.ts +29 -26
- package/src/database/puri.ts +36 -34
- package/src/database/puri.types.test-d.ts +6 -5
- package/src/database/puri.types.ts +9 -12
- package/src/database/transaction-context.ts +2 -2
- package/src/database/upsert-builder.ts +17 -10
- package/src/dict/sd.ts +17 -4
- package/src/dict/sonamu-dictionary.ts +23 -17
- package/src/entity/entity-manager.ts +9 -7
- package/src/entity/entity-template-cone.ts +10 -3
- package/src/entity/entity.ts +20 -16
- package/src/exceptions/error-handler.ts +2 -1
- package/src/exceptions/so-exceptions.ts +1 -1
- package/src/filter/utils.ts +3 -2
- package/src/logger/category.ts +1 -0
- package/src/logger/configure.ts +5 -5
- package/src/migration/__tests__/code-generation.search-text.test.ts +2 -3
- package/src/migration/code-generation.ts +26 -25
- package/src/migration/migration-set.ts +16 -18
- package/src/migration/migrator.ts +38 -33
- package/src/migration/postgresql-schema-reader.ts +12 -12
- package/src/migration/slack-confirm.ts +5 -4
- package/src/migration/types.ts +2 -2
- package/src/naite/messaging-types.ts +1 -1
- package/src/naite/naite-reporter.ts +5 -3
- package/src/naite/naite.ts +12 -7
- package/src/shared/app.shared.ts.txt +2 -2
- package/src/shared/web.shared.ts.txt +2 -2
- package/src/skills/AGENTS.md +19 -18
- package/src/skills/commands/sonamu-skills.md +9 -9
- package/src/skills/sonamu/SKILL.md +111 -104
- package/src/skills/sonamu/ai-agents.md +27 -26
- package/src/skills/sonamu/api.md +81 -69
- package/src/skills/sonamu/auth-migration.md +13 -27
- package/src/skills/sonamu/auth-plugins.md +41 -31
- package/src/skills/sonamu/auth.md +30 -24
- package/src/skills/sonamu/cdd.md +26 -17
- package/src/skills/sonamu/cone.md +50 -50
- package/src/skills/sonamu/config.md +74 -51
- package/src/skills/sonamu/create-sonamu.md +31 -19
- package/src/skills/sonamu/database.md +43 -26
- package/src/skills/sonamu/entity-basic.md +61 -61
- package/src/skills/sonamu/entity-relations.md +84 -80
- package/src/skills/sonamu/entity-validation-checklist.md +19 -15
- package/src/skills/sonamu/fixture-cli.md +52 -30
- package/src/skills/sonamu/framework-change.md +9 -7
- package/src/skills/sonamu/frontend.md +64 -82
- package/src/skills/sonamu/i18n.md +45 -37
- package/src/skills/sonamu/migration.md +54 -31
- package/src/skills/sonamu/model.md +98 -66
- package/src/skills/sonamu/naite.md +34 -32
- package/src/skills/sonamu/project-init.md +28 -8
- package/src/skills/sonamu/puri.md +82 -91
- package/src/skills/sonamu/scaffolding.md +44 -32
- package/src/skills/sonamu/skill-contribution.md +50 -45
- package/src/skills/sonamu/subset.md +13 -13
- package/src/skills/sonamu/tasks.md +73 -58
- package/src/skills/sonamu/testing-devrunner.md +56 -36
- package/src/skills/sonamu/testing.md +23 -58
- package/src/skills/sonamu/upsert.md +32 -31
- package/src/skills/sonamu/vector.md +37 -36
- package/src/ssr/index.ts +2 -12
- package/src/ssr/registry.ts +1 -1
- package/src/ssr/renderer.ts +7 -5
- package/src/ssr/types.ts +2 -2
- package/src/storage/buffered-file.ts +4 -2
- package/src/storage/drivers.ts +3 -2
- package/src/storage/s3-driver.ts +7 -4
- package/src/storage/storage-manager.ts +3 -2
- package/src/storage/types.ts +3 -2
- package/src/storage/uploaded-file.ts +1 -1
- package/src/stream/sse.ts +2 -2
- package/src/syncer/api-parser.ts +8 -5
- package/src/syncer/checksum.ts +9 -5
- package/src/syncer/code-generator.ts +16 -8
- package/src/syncer/entity-operations.ts +5 -3
- package/src/syncer/file-patterns.ts +2 -1
- package/src/syncer/module-loader.ts +9 -6
- package/src/syncer/syncer-actions.ts +5 -3
- package/src/syncer/syncer.ts +18 -24
- package/src/tasks/decorator.ts +10 -8
- package/src/tasks/step-wrapper.ts +1 -1
- package/src/tasks/workflow-manager.ts +18 -15
- package/src/template/__tests__/generated.template.search-text.test.ts +1 -0
- package/src/template/entity-converter.ts +4 -2
- package/src/template/generated.template.test-d.ts +2 -1
- package/src/template/helpers.ts +5 -2
- package/src/template/implementations/entity.template.ts +9 -8
- package/src/template/implementations/entry-server.template.ts +1 -1
- package/src/template/implementations/generated.template.ts +21 -29
- package/src/template/implementations/generated_http.template.ts +9 -6
- package/src/template/implementations/generated_sso.template.ts +6 -4
- package/src/template/implementations/init_types.template.ts +3 -2
- package/src/template/implementations/model.template.ts +4 -2
- package/src/template/implementations/model_test.template.ts +3 -2
- package/src/template/implementations/queries.template.ts +6 -14
- package/src/template/implementations/sd.template.ts +4 -2
- package/src/template/implementations/services.template.ts +7 -11
- package/src/template/implementations/view_form.template.ts +5 -3
- package/src/template/implementations/view_id_all_select.template.ts +3 -2
- package/src/template/implementations/view_list.template.ts +7 -5
- package/src/template/implementations/view_search_input.template.ts +3 -2
- package/src/template/template-manager.ts +4 -3
- package/src/template/template.ts +4 -3
- package/src/template/zod-converter.ts +10 -7
- package/src/testing/__tests__/dev-test-routes.test.ts +3 -2
- package/src/testing/__tests__/dev-vitest-manager.test.ts +1 -0
- package/src/testing/_relation-graph.ts +2 -2
- package/src/testing/bootstrap.ts +55 -27
- package/src/testing/data-explorer.ts +5 -4
- package/src/testing/dev-test-routes.ts +8 -5
- package/src/testing/dev-vitest-manager.ts +13 -12
- package/src/testing/fixture-generator.ts +11 -17
- package/src/testing/fixture-manager.ts +21 -17
- package/src/testing/parallel-db-manager.ts +2 -1
- package/src/testing/vitest-helpers.ts +2 -1
- package/src/types/__tests__/entity-json-schema-search-text.test.ts +1 -0
- package/src/types/types.ts +8 -8
- package/src/typings/knex.d.ts +4 -4
- package/src/ui/ai-api.ts +5 -3
- package/src/ui/ai-client.ts +6 -5
- package/src/ui/api.ts +25 -23
- package/src/ui/cdd-service.ts +12 -11
- package/src/utils/console-util.ts +3 -1
- package/src/utils/formatter.ts +94 -102
- package/src/utils/fs-utils.ts +2 -1
- package/src/utils/model.ts +2 -2
- package/src/utils/object-utils.ts +3 -3
- package/src/utils/process-utils.ts +2 -1
- package/src/utils/sql-parser.ts +10 -1
- package/src/utils/type-utils.ts +3 -3
- package/src/utils/utils.ts +9 -7
- package/src/utils/zod-error.ts +1 -1
- package/src/vector/chunking.ts +1 -1
- package/src/vector/config.ts +1 -1
- package/src/vector/embedding.ts +11 -9
- package/tsdown.api.config.ts +50 -0
- package/.swcrc.project-default +0 -18
- package/dist/api/__tests__/config.test.js +0 -189
- package/dist/bin/__tests__/test-command.test.js +0 -112
- package/dist/bin/__tests__/ts-loader-register.test.js +0 -45
- package/dist/database/puri-subset.test-d.js +0 -89
- package/dist/database/puri.types.test-d.js +0 -129
- package/dist/migration/__tests__/code-generation.search-text.test.js +0 -435
- package/dist/template/__tests__/generated.template.search-text.test.js +0 -99
- package/dist/template/generated.template.test-d.js +0 -24
- package/dist/testing/__tests__/dev-test-routes.test.js +0 -144
- package/dist/testing/__tests__/dev-vitest-manager.test.js +0 -152
- package/dist/types/__tests__/entity-json-schema-search-text.test.js +0 -256
- package/dist/typings/knex.d.js +0 -3
- package/dist/ui-web/assets/index-CKo0Z2Iu.css +0 -1
- package/dist/ui-web/assets/index-DK-2aacv.js +0 -257
|
@@ -1,791 +1,721 @@
|
|
|
1
|
+
import { __esmMin } from "../_virtual/rolldown_runtime.js";
|
|
2
|
+
import { createKnexInstance, init_knex } from "../database/knex.js";
|
|
3
|
+
import { init_controller, isTest } from "../utils/controller.js";
|
|
4
|
+
import { Sonamu, init_sonamu } from "../api/sonamu.js";
|
|
5
|
+
import { init_types, isBelongsToOneRelationProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
|
|
6
|
+
import { EntityManager, init_entity_manager } from "../entity/entity-manager.js";
|
|
7
|
+
import { UpsertBuilder, init_upsert_builder } from "../database/upsert-builder.js";
|
|
8
|
+
import { BaseModel, init_base_model } from "../database/base-model.js";
|
|
9
|
+
import { RelationGraph, init__relation_graph } from "./_relation-graph.js";
|
|
10
|
+
import inflection from "inflection";
|
|
11
|
+
import { unique } from "radashi";
|
|
1
12
|
import assert from "assert";
|
|
2
13
|
import chalk from "chalk";
|
|
3
|
-
import { execSync } from "child_process";
|
|
4
14
|
import { readFileSync, writeFileSync } from "fs";
|
|
5
|
-
import
|
|
6
|
-
import { unique } from "radashi";
|
|
15
|
+
import { execSync } from "child_process";
|
|
7
16
|
import { inspect } from "util";
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
17
|
+
|
|
18
|
+
//#region src/testing/fixture-manager.ts
|
|
19
|
+
var FixtureManagerClass, FixtureManager;
|
|
20
|
+
var init_fixture_manager = __esmMin((() => {
|
|
21
|
+
init_sonamu();
|
|
22
|
+
init_base_model();
|
|
23
|
+
init_knex();
|
|
24
|
+
init_upsert_builder();
|
|
25
|
+
init_entity_manager();
|
|
26
|
+
init_types();
|
|
27
|
+
init_controller();
|
|
28
|
+
init__relation_graph();
|
|
29
|
+
FixtureManagerClass = class {
|
|
30
|
+
_tdb = null;
|
|
31
|
+
set tdb(tdb) {
|
|
32
|
+
this._tdb = tdb;
|
|
33
|
+
}
|
|
34
|
+
get tdb() {
|
|
35
|
+
if (this._tdb === null) {
|
|
36
|
+
throw new Error("FixtureManager has not been initialized");
|
|
37
|
+
}
|
|
38
|
+
return this._tdb;
|
|
39
|
+
}
|
|
40
|
+
_fdb = null;
|
|
41
|
+
set fdb(fdb) {
|
|
42
|
+
this._fdb = fdb;
|
|
43
|
+
}
|
|
44
|
+
get fdb() {
|
|
45
|
+
if (this._fdb === null) {
|
|
46
|
+
throw new Error("FixtureManager has not been initialized");
|
|
47
|
+
}
|
|
48
|
+
return this._fdb;
|
|
49
|
+
}
|
|
50
|
+
cachedTableNames = null;
|
|
51
|
+
relationGraph = new RelationGraph();
|
|
52
|
+
builder = new UpsertBuilder();
|
|
53
|
+
fixtureRefMap = new Map();
|
|
54
|
+
skippedFixtures = new Map();
|
|
55
|
+
init() {
|
|
56
|
+
if (this._tdb !== null) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
if (Sonamu.dbConfig.test && Sonamu.dbConfig.production_master) {
|
|
60
|
+
const tConn = Sonamu.dbConfig.test.connection;
|
|
61
|
+
const pConn = Sonamu.dbConfig.production_master.connection;
|
|
62
|
+
if (`${tConn.host ?? "localhost"}:${tConn.port ?? 5432}/${tConn.database}` === `${pConn.host ?? "localhost"}:${pConn.port ?? 5432}/${pConn.database}`) {
|
|
63
|
+
throw new Error(`테스트DB와 프로덕션DB에 동일한 데이터베이스가 사용되었습니다.`);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
this.tdb = createKnexInstance(Sonamu.dbConfig.test);
|
|
67
|
+
this.fdb = createKnexInstance(Sonamu.dbConfig.fixture);
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
원격 fixture DB를 로컬 test DB로 복사합니다.
|
|
71
|
+
pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.
|
|
72
|
+
*/
|
|
73
|
+
async sync() {
|
|
74
|
+
const fixtureConn = Sonamu.dbConfig.fixture.connection;
|
|
75
|
+
const testConn = Sonamu.dbConfig.test.connection;
|
|
76
|
+
const testPgEnv = { PGPASSWORD: testConn.password || "" };
|
|
77
|
+
execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "
|
|
68
78
|
SELECT pg_terminate_backend(pg_stat_activity.pid)
|
|
69
79
|
FROM pg_stat_activity
|
|
70
80
|
WHERE datname = '${testConn.database}'
|
|
71
81
|
AND pid <> pg_backend_pid();
|
|
72
82
|
"`, {
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
!isTest() && console.log(`Skipping sequence reset for ${tableName} (id type: ${idType || "unknown"})`);
|
|
126
|
-
continue;
|
|
127
|
-
}
|
|
128
|
-
// PostgreSQL 시퀀스를 현재 테이블의 MAX(id)로 리셋합니다.
|
|
129
|
-
// string 타입의 경우 숫자 캐스팅이 필요합니다.
|
|
130
|
-
const maxExpr = idType === "string" ? "MAX(id::bigint)" : "MAX(id)";
|
|
131
|
-
await testDb.raw(`
|
|
83
|
+
stdio: "inherit",
|
|
84
|
+
env: {
|
|
85
|
+
...process.env,
|
|
86
|
+
...testPgEnv
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "DROP DATABASE IF EXISTS \\"${testConn.database}\\""`, {
|
|
90
|
+
stdio: "inherit",
|
|
91
|
+
env: {
|
|
92
|
+
...process.env,
|
|
93
|
+
...testPgEnv
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
execSync(`psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c "CREATE DATABASE \\"${testConn.database}\\""`, {
|
|
97
|
+
stdio: "inherit",
|
|
98
|
+
env: {
|
|
99
|
+
...process.env,
|
|
100
|
+
...testPgEnv
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
const fixturePgEnv = { PGPASSWORD: fixtureConn.password || "" };
|
|
104
|
+
const dumpCmd = `pg_dump -h ${fixtureConn.host} -p ${fixtureConn.port ?? 5432} -U ${fixtureConn.user} -d ${fixtureConn.database} -Fc`;
|
|
105
|
+
const restoreCmd = `pg_restore -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d ${testConn.database} --no-owner --no-acl`;
|
|
106
|
+
execSync(`${dumpCmd} | PGPASSWORD="${testConn.password || ""}" ${restoreCmd}`, {
|
|
107
|
+
stdio: "inherit",
|
|
108
|
+
env: {
|
|
109
|
+
...process.env,
|
|
110
|
+
...fixturePgEnv
|
|
111
|
+
},
|
|
112
|
+
shell: "/bin/bash"
|
|
113
|
+
});
|
|
114
|
+
await this.resetSequences(Sonamu.dbConfig.test);
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* 모든 테이블의 시퀀스를 현재 MAX(id)로 리셋합니다.
|
|
118
|
+
* fixture sync 후 시퀀스가 실제 데이터와 맞지 않는 문제를 해결합니다.
|
|
119
|
+
*/
|
|
120
|
+
async resetSequences(dbConfig) {
|
|
121
|
+
const testDb = createKnexInstance(dbConfig);
|
|
122
|
+
const entities = EntityManager.getAllEntities();
|
|
123
|
+
try {
|
|
124
|
+
for (const entity of entities) {
|
|
125
|
+
const tableName = entity.table || entity.id.toLowerCase();
|
|
126
|
+
const idProp = entity.props.find((p) => p.name === "id");
|
|
127
|
+
const idType = idProp?.type;
|
|
128
|
+
const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
|
|
129
|
+
if (!usesSequence) {
|
|
130
|
+
!isTest() && console.log(`Skipping sequence reset for ${tableName} (id type: ${idType || "unknown"})`);
|
|
131
|
+
continue;
|
|
132
|
+
}
|
|
133
|
+
const maxExpr = idType === "string" ? "MAX(id::bigint)" : "MAX(id)";
|
|
134
|
+
await testDb.raw(`
|
|
132
135
|
SELECT setval(
|
|
133
136
|
pg_get_serial_sequence('public.${tableName}', 'id'),
|
|
134
137
|
COALESCE((SELECT ${maxExpr} FROM ${tableName}), 1)
|
|
135
138
|
)
|
|
136
139
|
`);
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
670
|
-
|
|
671
|
-
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
684
|
-
|
|
685
|
-
|
|
686
|
-
|
|
687
|
-
|
|
688
|
-
|
|
689
|
-
|
|
690
|
-
|
|
691
|
-
|
|
692
|
-
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
710
|
-
|
|
711
|
-
|
|
712
|
-
|
|
713
|
-
|
|
714
|
-
for (const fixture of currentLevel){
|
|
715
|
-
remaining.delete(fixture.fixtureId);
|
|
716
|
-
processed.add(fixture.fixtureId);
|
|
717
|
-
}
|
|
718
|
-
levels.push(currentLevel);
|
|
719
|
-
}
|
|
720
|
-
return levels;
|
|
721
|
-
}
|
|
722
|
-
async checkUniqueViolation(db, entity, fixture) {
|
|
723
|
-
const _uniqueIndexes = entity.indexes?.filter((i)=>i.type === "unique") ?? [];
|
|
724
|
-
const uniqueIndexes = _uniqueIndexes.filter((index)=>index.columns.every((column)=>!column.name.startsWith(`${entity.table}__`)));
|
|
725
|
-
if (uniqueIndexes.length === 0) {
|
|
726
|
-
return null;
|
|
727
|
-
}
|
|
728
|
-
let uniqueQuery = db(entity.table);
|
|
729
|
-
let hasCondition = false;
|
|
730
|
-
for (const index of uniqueIndexes){
|
|
731
|
-
// 컬럼 중 하나라도 null이면 유니크 제약을 위반하지 않기 때문에 해당 인덱스는 무시
|
|
732
|
-
const containsNull = index.columns.some((column)=>{
|
|
733
|
-
const field = column.name.replace(/_id$/, "");
|
|
734
|
-
return fixture.columns[field]?.value === null;
|
|
735
|
-
});
|
|
736
|
-
if (containsNull) {
|
|
737
|
-
continue;
|
|
738
|
-
}
|
|
739
|
-
uniqueQuery = uniqueQuery.orWhere((qb)=>{
|
|
740
|
-
for (const column of index.columns){
|
|
741
|
-
const field = column.name.replace(/_id$/, "");
|
|
742
|
-
if (Array.isArray(fixture.columns[field]?.value)) {
|
|
743
|
-
qb.whereIn(column.name, fixture.columns[field].value);
|
|
744
|
-
} else {
|
|
745
|
-
qb.andWhere(column.name, fixture.columns[field]?.value);
|
|
746
|
-
}
|
|
747
|
-
}
|
|
748
|
-
});
|
|
749
|
-
hasCondition = true;
|
|
750
|
-
}
|
|
751
|
-
if (!hasCondition) {
|
|
752
|
-
return null;
|
|
753
|
-
}
|
|
754
|
-
const [uniqueFound] = await uniqueQuery;
|
|
755
|
-
return uniqueFound;
|
|
756
|
-
}
|
|
757
|
-
async checkDuplicateByColumns(db, entity, fixture, columns) {
|
|
758
|
-
if (columns.length === 0) {
|
|
759
|
-
return null;
|
|
760
|
-
}
|
|
761
|
-
const whereClause = {};
|
|
762
|
-
for (const column of columns){
|
|
763
|
-
// relation 필드인 경우 _id 붙이기
|
|
764
|
-
const prop = entity.props.find((p)=>p.name === column);
|
|
765
|
-
const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;
|
|
766
|
-
const value = fixture.columns[column]?.value;
|
|
767
|
-
// null 값이 포함된 경우 중복 확인 스킵
|
|
768
|
-
if (value === null || value === undefined) {
|
|
769
|
-
return null;
|
|
770
|
-
}
|
|
771
|
-
whereClause[dbColumn] = value;
|
|
772
|
-
}
|
|
773
|
-
const [found] = await db(entity.table).where(whereClause).limit(1);
|
|
774
|
-
return found;
|
|
775
|
-
}
|
|
776
|
-
async addFixtureLoader(code) {
|
|
777
|
-
const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;
|
|
778
|
-
const content = readFileSync(path).toString();
|
|
779
|
-
const fixtureLoaderStart = content.indexOf("const fixtureLoader = {");
|
|
780
|
-
const fixtureLoaderEnd = content.indexOf("};", fixtureLoaderStart);
|
|
781
|
-
if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {
|
|
782
|
-
const newContent = `${content.slice(0, fixtureLoaderEnd)} ${code}\n${content.slice(fixtureLoaderEnd)}`;
|
|
783
|
-
writeFileSync(path, newContent);
|
|
784
|
-
} else {
|
|
785
|
-
throw new Error("Failed to find fixtureLoader in fixture.ts");
|
|
786
|
-
}
|
|
787
|
-
}
|
|
788
|
-
}
|
|
789
|
-
export const FixtureManager = new FixtureManagerClass();
|
|
140
|
+
}
|
|
141
|
+
} finally {
|
|
142
|
+
await testDb.destroy();
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
visitedRecords = new Set();
|
|
146
|
+
async importFixture(entityId, ids) {
|
|
147
|
+
this.visitedRecords.clear();
|
|
148
|
+
const queries = unique((await Promise.all(ids.map(async (id) => {
|
|
149
|
+
return await this.getImportQueries(entityId, "id", id);
|
|
150
|
+
}))).flat());
|
|
151
|
+
const wdb = BaseModel.getDB("w");
|
|
152
|
+
for (const query of queries) {
|
|
153
|
+
await wdb.raw(query);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
async getImportQueries(entityId, field, id) {
|
|
157
|
+
const recordKey = `${entityId}#${field}#${id}`;
|
|
158
|
+
if (this.visitedRecords.has(recordKey)) {
|
|
159
|
+
return [];
|
|
160
|
+
}
|
|
161
|
+
this.visitedRecords.add(recordKey);
|
|
162
|
+
const entity = EntityManager.get(entityId);
|
|
163
|
+
const wdb = BaseModel.getDB("w");
|
|
164
|
+
const [row] = await wdb(entity.table).where(field, id).limit(1);
|
|
165
|
+
if (row === undefined) {
|
|
166
|
+
throw new Error(`${entityId}#${id} row를 찾을 수 없습니다.`);
|
|
167
|
+
}
|
|
168
|
+
const fixtureDatabase = Sonamu.dbConfig.fixture.connection.database;
|
|
169
|
+
const realDatabase = Sonamu.dbConfig.production_master.connection.database;
|
|
170
|
+
const selfQuery = `INSERT IGNORE INTO \`${fixtureDatabase}\`.\`${entity.table}\` (SELECT * FROM \`${realDatabase}\`.\`${entity.table}\` WHERE \`id\` = ${id})`;
|
|
171
|
+
const args = Object.entries(entity.relations).filter(([, relation]) => isBelongsToOneRelationProp(relation) || isOneToOneRelationProp(relation) && relation.customJoinClause === undefined).map(([, relation]) => {
|
|
172
|
+
let field$1;
|
|
173
|
+
let id$1;
|
|
174
|
+
if (isOneToOneRelationProp(relation) && !relation.hasJoinColumn) {
|
|
175
|
+
const relatedEntity = EntityManager.get(relation.with);
|
|
176
|
+
const relatedIdColumnName = relatedEntity.props.find((p) => isRelationProp(p) && p.with === entity.id)?.name;
|
|
177
|
+
if (!relatedIdColumnName) {
|
|
178
|
+
throw new Error(`${relatedEntity.id}의 ${entity.id} 관계 프롭을 찾을 수 없습니다.`);
|
|
179
|
+
}
|
|
180
|
+
field$1 = `${relatedIdColumnName}_id`;
|
|
181
|
+
id$1 = row.id;
|
|
182
|
+
} else {
|
|
183
|
+
field$1 = "id";
|
|
184
|
+
id$1 = row[`${relation.name}_id`];
|
|
185
|
+
}
|
|
186
|
+
return {
|
|
187
|
+
entityId: relation.with,
|
|
188
|
+
field: field$1,
|
|
189
|
+
id: id$1
|
|
190
|
+
};
|
|
191
|
+
}).filter((arg) => arg.id !== null);
|
|
192
|
+
const relQueries = await Promise.all(args.map(async (args$1) => {
|
|
193
|
+
return this.getImportQueries(args$1.entityId, args$1.field, args$1.id);
|
|
194
|
+
}));
|
|
195
|
+
return [...unique(relQueries.toReversed().flat()), selfQuery];
|
|
196
|
+
}
|
|
197
|
+
async destroy() {
|
|
198
|
+
if (this._tdb) {
|
|
199
|
+
await this._tdb.destroy();
|
|
200
|
+
this._tdb = null;
|
|
201
|
+
}
|
|
202
|
+
if (this._fdb) {
|
|
203
|
+
await this._fdb.destroy();
|
|
204
|
+
this._fdb = null;
|
|
205
|
+
}
|
|
206
|
+
await BaseModel.destroy();
|
|
207
|
+
}
|
|
208
|
+
async getFixtures(sourceDBName, targetDBName, searchOptions, duplicateCheck) {
|
|
209
|
+
const sourceDB = createKnexInstance(Sonamu.dbConfig[sourceDBName]);
|
|
210
|
+
const targetDB = createKnexInstance(Sonamu.dbConfig[targetDBName]);
|
|
211
|
+
try {
|
|
212
|
+
const { entityId, field, value, searchType } = searchOptions;
|
|
213
|
+
const entity = EntityManager.get(entityId);
|
|
214
|
+
const column = entity.props.find((prop) => prop.name === field)?.type === "relation" ? `${field}_id` : field;
|
|
215
|
+
let query = sourceDB(entity.table);
|
|
216
|
+
if (searchType === "equals") {
|
|
217
|
+
query = query.where(column, value);
|
|
218
|
+
} else if (searchType === "like") {
|
|
219
|
+
query = query.where(column, "like", `%${value}%`);
|
|
220
|
+
}
|
|
221
|
+
const rows = await query;
|
|
222
|
+
if (rows.length === 0) {
|
|
223
|
+
throw new Error("No records found");
|
|
224
|
+
}
|
|
225
|
+
const fixtures = [];
|
|
226
|
+
for (const row of rows) {
|
|
227
|
+
const initialRecordsLength = fixtures.length;
|
|
228
|
+
const newRecords = await this.createFixtureRecord(entity, row, { _db: sourceDB });
|
|
229
|
+
fixtures.push(...newRecords);
|
|
230
|
+
const currentFixtureRecord = fixtures.find((r) => r.fixtureId === `${entityId}#${row.id}`);
|
|
231
|
+
if (currentFixtureRecord) {
|
|
232
|
+
currentFixtureRecord.fetchedRecords = fixtures.filter((r) => r.fixtureId !== currentFixtureRecord.fixtureId).slice(initialRecordsLength).map((r) => r.fixtureId);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
for await (const fixture of fixtures) {
|
|
236
|
+
const entity$1 = EntityManager.get(fixture.entityId);
|
|
237
|
+
const customColumns = duplicateCheck?.columns?.[fixture.entityId];
|
|
238
|
+
if (customColumns && customColumns.length > 0) {
|
|
239
|
+
const customDuplicateRow = await this.checkDuplicateByColumns(targetDB, entity$1, fixture, customColumns);
|
|
240
|
+
if (customDuplicateRow) {
|
|
241
|
+
const [record] = await this.createFixtureRecord(entity$1, customDuplicateRow, {
|
|
242
|
+
singleRecord: true,
|
|
243
|
+
_db: targetDB
|
|
244
|
+
});
|
|
245
|
+
fixture.target = record;
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
const uniqueRow = await this.checkUniqueViolation(targetDB, entity$1, fixture);
|
|
249
|
+
if (uniqueRow) {
|
|
250
|
+
const [record] = await this.createFixtureRecord(entity$1, uniqueRow, {
|
|
251
|
+
singleRecord: true,
|
|
252
|
+
_db: targetDB
|
|
253
|
+
});
|
|
254
|
+
fixture.unique = record;
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
return unique(fixtures, (f) => f.fixtureId);
|
|
258
|
+
} finally {
|
|
259
|
+
await Promise.allSettled([targetDB.destroy(), sourceDB.destroy()]);
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
async createFixtureRecord(entity, row, options) {
|
|
263
|
+
const records = [];
|
|
264
|
+
const visitedEntities = new Set();
|
|
265
|
+
const create = async (entity$1, row$1) => {
|
|
266
|
+
const fixtureId = `${entity$1.id}#${row$1.id}`;
|
|
267
|
+
if (visitedEntities.has(fixtureId)) {
|
|
268
|
+
return;
|
|
269
|
+
}
|
|
270
|
+
visitedEntities.add(fixtureId);
|
|
271
|
+
const record = {
|
|
272
|
+
fixtureId,
|
|
273
|
+
entityId: entity$1.id,
|
|
274
|
+
id: row$1.id,
|
|
275
|
+
columns: {},
|
|
276
|
+
fetchedRecords: [],
|
|
277
|
+
belongsRecords: []
|
|
278
|
+
};
|
|
279
|
+
for (const prop of entity$1.props) {
|
|
280
|
+
if (isVirtualProp(prop)) {
|
|
281
|
+
continue;
|
|
282
|
+
}
|
|
283
|
+
record.columns[prop.name] = {
|
|
284
|
+
prop,
|
|
285
|
+
value: row$1[prop.name]
|
|
286
|
+
};
|
|
287
|
+
const db = options?._db ?? BaseModel.getDB("w");
|
|
288
|
+
if (isManyToManyRelationProp(prop)) {
|
|
289
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
290
|
+
const throughTable = prop.joinTable;
|
|
291
|
+
const fromColumn = `${inflection.singularize(entity$1.table)}_id`;
|
|
292
|
+
const toColumn = `${inflection.singularize(relatedEntity.table)}_id`;
|
|
293
|
+
const relatedIds = await db(throughTable).where(fromColumn, row$1.id).pluck(toColumn);
|
|
294
|
+
record.columns[prop.name].value = relatedIds;
|
|
295
|
+
} else if (isHasManyRelationProp(prop)) {
|
|
296
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
297
|
+
const relatedIds = await db(relatedEntity.table).where(prop.joinColumn, row$1.id).pluck("id");
|
|
298
|
+
record.columns[prop.name].value = relatedIds;
|
|
299
|
+
} else if (isOneToOneRelationProp(prop) && !prop.hasJoinColumn) {
|
|
300
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
301
|
+
const relatedProp = relatedEntity.props.find((p) => isRelationProp(p) && p.with === entity$1.id);
|
|
302
|
+
if (relatedProp && isRelationProp(relatedProp)) {
|
|
303
|
+
const fkColumn = `${relatedProp.name}_id`;
|
|
304
|
+
const relatedRow = await db(relatedEntity.table).where(fkColumn, row$1.id).first();
|
|
305
|
+
record.columns[prop.name].value = relatedRow?.id;
|
|
306
|
+
}
|
|
307
|
+
} else if (isRelationProp(prop)) {
|
|
308
|
+
const relatedId = row$1[`${prop.name}_id`];
|
|
309
|
+
record.columns[prop.name].value = relatedId;
|
|
310
|
+
if (relatedId) {
|
|
311
|
+
record.belongsRecords.push(`${prop.with}#${relatedId}`);
|
|
312
|
+
}
|
|
313
|
+
if (!options?.singleRecord && relatedId) {
|
|
314
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
315
|
+
const relatedRow = await db(relatedEntity.table).where("id", relatedId).first();
|
|
316
|
+
if (relatedRow) {
|
|
317
|
+
await create(relatedEntity, relatedRow);
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
records.push(record);
|
|
323
|
+
};
|
|
324
|
+
await create(entity, row);
|
|
325
|
+
return records;
|
|
326
|
+
}
|
|
327
|
+
/**
|
|
328
|
+
* 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)
|
|
329
|
+
* 2. 테이블별 레벨별로 UpsertBuilder에 등록 및 upsert 실행
|
|
330
|
+
* 3. 순서 기반 uuid→id 매핑 (UpsertBuilder가 uuid를 DB에 저장하지 않으므로)
|
|
331
|
+
*
|
|
332
|
+
* UpsertBuilder는 self-reference가 있으면 buildInsertLevels()로 재정렬하여
|
|
333
|
+
* 등록 순서와 반환 순서가 달라질 수 있습니다. 이를 방지하기 위해
|
|
334
|
+
* FixtureManager가 레벨별로 나눠서 처리하여 각 upsert 호출에서는
|
|
335
|
+
* self-reference가 없도록 합니다.
|
|
336
|
+
*/
|
|
337
|
+
async insertFixtures(dbName, _fixtures) {
|
|
338
|
+
const fixtures = unique(_fixtures, (f) => f.fixtureId);
|
|
339
|
+
this.builder = new UpsertBuilder();
|
|
340
|
+
this.fixtureRefMap = new Map();
|
|
341
|
+
this.skippedFixtures = new Map();
|
|
342
|
+
const dbConfig = process.env.SONAMU_WORKER_DB === "true" && process.env.VITEST_POOL_ID ? (() => {
|
|
343
|
+
const workerId = parseInt(process.env.VITEST_POOL_ID ?? "1", 10);
|
|
344
|
+
const baseConfig = Sonamu.dbConfig[dbName];
|
|
345
|
+
const connection = baseConfig.connection;
|
|
346
|
+
return {
|
|
347
|
+
...baseConfig,
|
|
348
|
+
connection: {
|
|
349
|
+
...connection,
|
|
350
|
+
database: `${connection.database}_${workerId}`
|
|
351
|
+
},
|
|
352
|
+
pool: {
|
|
353
|
+
min: 1,
|
|
354
|
+
max: 1
|
|
355
|
+
}
|
|
356
|
+
};
|
|
357
|
+
})() : Sonamu.dbConfig[dbName];
|
|
358
|
+
const db = createKnexInstance(dbConfig);
|
|
359
|
+
const results = [];
|
|
360
|
+
try {
|
|
361
|
+
this.relationGraph.buildGraph(fixtures);
|
|
362
|
+
const insertionOrder = this.relationGraph.getInsertionOrder();
|
|
363
|
+
for (const fixtureId of insertionOrder) {
|
|
364
|
+
const fixture = fixtures.find((f) => f.fixtureId === fixtureId);
|
|
365
|
+
if (!fixture) continue;
|
|
366
|
+
const hasTarget = !!fixture.target;
|
|
367
|
+
const hasUnique = !!fixture.unique;
|
|
368
|
+
const hasDuplicate = hasTarget || hasUnique;
|
|
369
|
+
if (hasDuplicate && !fixture.override) {
|
|
370
|
+
const existingId = fixture.unique?.id ?? fixture.target?.id;
|
|
371
|
+
assert(existingId);
|
|
372
|
+
this.skippedFixtures.set(fixtureId, {
|
|
373
|
+
entityId: fixture.entityId,
|
|
374
|
+
existingId
|
|
375
|
+
});
|
|
376
|
+
!isTest() && console.log(chalk.yellow(`Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`));
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
const fixturesByTable = new Map();
|
|
380
|
+
const tableOrder = [];
|
|
381
|
+
for (const fixtureId of insertionOrder) {
|
|
382
|
+
if (this.skippedFixtures.has(fixtureId)) continue;
|
|
383
|
+
const fixture = fixtures.find((f) => f.fixtureId === fixtureId);
|
|
384
|
+
if (!fixture) continue;
|
|
385
|
+
const entity = EntityManager.get(fixture.entityId);
|
|
386
|
+
const tableName = entity.table;
|
|
387
|
+
if (!fixturesByTable.has(tableName)) {
|
|
388
|
+
fixturesByTable.set(tableName, []);
|
|
389
|
+
tableOrder.push(tableName);
|
|
390
|
+
}
|
|
391
|
+
fixturesByTable.get(tableName)?.push(fixture);
|
|
392
|
+
}
|
|
393
|
+
await db.transaction(async (trx) => {
|
|
394
|
+
const insertedIdsByTable = new Map();
|
|
395
|
+
for (const tableName of tableOrder) {
|
|
396
|
+
const tableFixtures = fixturesByTable.get(tableName) ?? [];
|
|
397
|
+
const levels = this.groupFixturesByLevel(tableFixtures);
|
|
398
|
+
for (const levelFixtures of levels) {
|
|
399
|
+
for (const fixture of levelFixtures) {
|
|
400
|
+
this.registerFixture(fixture, insertedIdsByTable);
|
|
401
|
+
!isTest() && console.log(chalk.blue(`Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override)` : ""}`));
|
|
402
|
+
}
|
|
403
|
+
const table = this.builder.getTable(tableName);
|
|
404
|
+
const uuids = table.rows.map((row) => row.uuid);
|
|
405
|
+
!isTest() && console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows (level ${levels.indexOf(levelFixtures) + 1}/${levels.length})`));
|
|
406
|
+
const ids = await this.builder.upsert(trx, tableName);
|
|
407
|
+
if (uuids.length > 0 && uuids.length === ids.length) {
|
|
408
|
+
const existingMap = insertedIdsByTable.get(tableName) ?? new Map();
|
|
409
|
+
for (let i = 0; i < uuids.length; i++) {
|
|
410
|
+
existingMap.set(uuids[i], ids[i]);
|
|
411
|
+
}
|
|
412
|
+
insertedIdsByTable.set(tableName, existingMap);
|
|
413
|
+
} else if (uuids.length !== ids.length) {
|
|
414
|
+
console.warn(chalk.yellow(`Warning: uuid count (${uuids.length}) != id count (${ids.length}) for ${tableName}`));
|
|
415
|
+
}
|
|
416
|
+
}
|
|
417
|
+
}
|
|
418
|
+
await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);
|
|
419
|
+
!isTest() && console.log(chalk.blue("Resetting sequences..."));
|
|
420
|
+
for (const tableName of tableOrder) {
|
|
421
|
+
try {
|
|
422
|
+
const entity = EntityManager.getAllEntities().find((e) => e.table === tableName || e.id.toLowerCase() === tableName);
|
|
423
|
+
if (entity) {
|
|
424
|
+
const idProp = entity.props.find((p) => p.name === "id");
|
|
425
|
+
const idType = idProp?.type;
|
|
426
|
+
const usesSequence = idType === "integer" || idType === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence";
|
|
427
|
+
if (!usesSequence) {
|
|
428
|
+
!isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName} (id type: ${idType || "unknown"})`));
|
|
429
|
+
continue;
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
const entity2 = EntityManager.getAllEntities().find((e) => e.table === tableName || e.id.toLowerCase() === tableName);
|
|
433
|
+
const idType2 = entity2?.props.find((p) => p.name === "id")?.type;
|
|
434
|
+
const maxIdResult = idType2 === "string" ? await trx.raw(`SELECT MAX(id::bigint) as max_id FROM "${tableName}"`).then((r) => r.rows[0]) : await trx(tableName).max("id as max_id").first();
|
|
435
|
+
const maxId = maxIdResult?.max_id;
|
|
436
|
+
if (maxId !== null && maxId !== undefined) {
|
|
437
|
+
await trx.raw(`SELECT setval(pg_get_serial_sequence(?, 'id'), ?)`, [tableName, maxId]);
|
|
438
|
+
!isTest() && console.log(chalk.green(`Reset sequence for ${tableName}: ${maxId}`));
|
|
439
|
+
}
|
|
440
|
+
} catch (_err) {
|
|
441
|
+
!isTest() && console.log(chalk.gray(`Skipped sequence reset for ${tableName}`));
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
for (const fixture of fixtures) {
|
|
445
|
+
const entity = EntityManager.get(fixture.entityId);
|
|
446
|
+
const skipped = this.skippedFixtures.get(fixture.fixtureId);
|
|
447
|
+
if (skipped) {
|
|
448
|
+
results.push({
|
|
449
|
+
entityId: fixture.entityId,
|
|
450
|
+
data: await trx(entity.table).where("id", skipped.existingId).first()
|
|
451
|
+
});
|
|
452
|
+
continue;
|
|
453
|
+
}
|
|
454
|
+
const ref = this.fixtureRefMap.get(fixture.fixtureId);
|
|
455
|
+
if (ref) {
|
|
456
|
+
const uuidToId = insertedIdsByTable.get(entity.table);
|
|
457
|
+
const insertedId = uuidToId?.get(ref.uuid);
|
|
458
|
+
if (insertedId !== undefined) {
|
|
459
|
+
results.push({
|
|
460
|
+
entityId: fixture.entityId,
|
|
461
|
+
data: await trx(entity.table).where("id", insertedId).first()
|
|
462
|
+
});
|
|
463
|
+
!isTest() && console.log(chalk.green(`Inserted into ${entity.table}: #${fixture.id} -> #${insertedId}`));
|
|
464
|
+
}
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
});
|
|
468
|
+
} finally {
|
|
469
|
+
await db.destroy();
|
|
470
|
+
}
|
|
471
|
+
return unique(results, (r) => `${r.entityId}#${r.data.id}`);
|
|
472
|
+
}
|
|
473
|
+
/**
|
|
474
|
+
* FixtureRecord를 UpsertBuilder에 등록
|
|
475
|
+
* @param insertedIdsByTable 이미 upsert된 테이블의 uuid→id 매핑 (레벨별 처리 시 사용)
|
|
476
|
+
*/
|
|
477
|
+
registerFixture(fixture, insertedIdsByTable) {
|
|
478
|
+
const entity = EntityManager.get(fixture.entityId);
|
|
479
|
+
const row = {};
|
|
480
|
+
const existingRecord = fixture.target ?? fixture.unique;
|
|
481
|
+
const isOverrideMode = fixture.override && existingRecord;
|
|
482
|
+
for (const [propName, column] of Object.entries(fixture.columns)) {
|
|
483
|
+
const prop = column.prop;
|
|
484
|
+
if (isVirtualProp(prop)) {
|
|
485
|
+
continue;
|
|
486
|
+
}
|
|
487
|
+
if ("generated" in prop && prop.generated) {
|
|
488
|
+
continue;
|
|
489
|
+
}
|
|
490
|
+
if (propName === "id") {
|
|
491
|
+
const idProp = entity.props.find((p) => p.name === "id");
|
|
492
|
+
const usesSequence = !entity.parentId && (idProp?.type === "integer" || idProp?.type === "bigInteger" || idProp?.cone?.fixtureStrategy === "sequence");
|
|
493
|
+
if (isOverrideMode && existingRecord) {
|
|
494
|
+
row[propName] = existingRecord.columns[propName]?.value;
|
|
495
|
+
} else if (!usesSequence) {
|
|
496
|
+
row[propName] = column.value;
|
|
497
|
+
}
|
|
498
|
+
continue;
|
|
499
|
+
}
|
|
500
|
+
if (isRelationProp(prop)) {
|
|
501
|
+
if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) && prop.hasJoinColumn) {
|
|
502
|
+
const relatedId = column.value;
|
|
503
|
+
if (relatedId !== null && relatedId !== undefined) {
|
|
504
|
+
const relatedFixtureId = `${prop.with}#${relatedId}`;
|
|
505
|
+
const skippedExistingId = this.skippedFixtures.get(relatedFixtureId)?.existingId;
|
|
506
|
+
if (skippedExistingId !== undefined) {
|
|
507
|
+
row[`${propName}_id`] = skippedExistingId;
|
|
508
|
+
} else {
|
|
509
|
+
const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
|
|
510
|
+
if (relatedRef) {
|
|
511
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
512
|
+
const relatedInsertedIds = insertedIdsByTable?.get(relatedEntity.table);
|
|
513
|
+
const actualId = relatedInsertedIds?.get(relatedRef.uuid);
|
|
514
|
+
if (actualId !== undefined) {
|
|
515
|
+
row[`${propName}_id`] = actualId;
|
|
516
|
+
} else {
|
|
517
|
+
row[`${propName}_id`] = relatedRef;
|
|
518
|
+
}
|
|
519
|
+
} else {
|
|
520
|
+
row[`${propName}_id`] = relatedId;
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
} else {
|
|
524
|
+
row[`${propName}_id`] = null;
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
} else {
|
|
528
|
+
row[propName] = this.convertColumnValue(prop, column.value);
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
!isTest() && console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));
|
|
532
|
+
const ref = this.builder.register(entity.table, row);
|
|
533
|
+
this.fixtureRefMap.set(fixture.fixtureId, ref);
|
|
534
|
+
return ref;
|
|
535
|
+
}
|
|
536
|
+
/**
|
|
537
|
+
* 컬럼 값 변환
|
|
538
|
+
*/
|
|
539
|
+
convertColumnValue(prop, value) {
|
|
540
|
+
if (value === null || value === undefined) {
|
|
541
|
+
return null;
|
|
542
|
+
}
|
|
543
|
+
switch (prop.type) {
|
|
544
|
+
case "json": return value;
|
|
545
|
+
case "date":
|
|
546
|
+
if (typeof value === "string" || typeof value === "number") {
|
|
547
|
+
return new Date(value);
|
|
548
|
+
}
|
|
549
|
+
return value;
|
|
550
|
+
default: return value;
|
|
551
|
+
}
|
|
552
|
+
}
|
|
553
|
+
async processManyToManyRelations(trx, fixtures, insertedIdsByTable) {
|
|
554
|
+
for (const fixture of fixtures) {
|
|
555
|
+
const entity = EntityManager.get(fixture.entityId);
|
|
556
|
+
const sourceRef = this.fixtureRefMap.get(fixture.fixtureId);
|
|
557
|
+
if (!sourceRef) continue;
|
|
558
|
+
const sourceUuidToId = insertedIdsByTable.get(entity.table);
|
|
559
|
+
const sourceId = sourceUuidToId?.get(sourceRef.uuid);
|
|
560
|
+
if (sourceId === undefined) continue;
|
|
561
|
+
for (const [, column] of Object.entries(fixture.columns)) {
|
|
562
|
+
const prop = column.prop;
|
|
563
|
+
if (isManyToManyRelationProp(prop) && Array.isArray(column.value)) {
|
|
564
|
+
const targetTable = EntityManager.get(prop.with);
|
|
565
|
+
if (!this.builder.hasTable(targetTable.table)) continue;
|
|
566
|
+
const relatedIds = column.value;
|
|
567
|
+
if (relatedIds.length === 0) continue;
|
|
568
|
+
const joinTable = prop.joinTable;
|
|
569
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
570
|
+
const sourceColumn = `${inflection.singularize(entity.table)}_id`;
|
|
571
|
+
const targetColumn = `${inflection.singularize(relatedEntity.table)}_id`;
|
|
572
|
+
for (const relatedId of relatedIds) {
|
|
573
|
+
const relatedFixtureId = `${prop.with}#${relatedId}`;
|
|
574
|
+
const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
|
|
575
|
+
let targetId;
|
|
576
|
+
if (relatedRef) {
|
|
577
|
+
const relatedUuidToId = insertedIdsByTable.get(relatedEntity.table);
|
|
578
|
+
const resolvedId = relatedUuidToId?.get(relatedRef.uuid);
|
|
579
|
+
if (resolvedId === undefined) {
|
|
580
|
+
console.warn(`Related fixture ${relatedFixtureId} not found in insertedIds, skipping`);
|
|
581
|
+
continue;
|
|
582
|
+
}
|
|
583
|
+
targetId = resolvedId;
|
|
584
|
+
} else {
|
|
585
|
+
targetId = relatedId;
|
|
586
|
+
}
|
|
587
|
+
const [found] = await trx(joinTable).where({
|
|
588
|
+
[sourceColumn]: sourceId,
|
|
589
|
+
[targetColumn]: targetId
|
|
590
|
+
}).limit(1);
|
|
591
|
+
if (!found) {
|
|
592
|
+
await trx(joinTable).insert({
|
|
593
|
+
[sourceColumn]: sourceId,
|
|
594
|
+
[targetColumn]: targetId
|
|
595
|
+
});
|
|
596
|
+
!isTest() && console.log(chalk.green(`Inserted into ${joinTable}: ${entity.table}(${sourceId}) - ${relatedEntity.table}(${targetId})`));
|
|
597
|
+
}
|
|
598
|
+
}
|
|
599
|
+
}
|
|
600
|
+
}
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
/**
|
|
604
|
+
* 같은 테이블 내 fixture들을 self-reference 레벨별로 분할
|
|
605
|
+
* - self-reference가 없는 fixture들: Level 0
|
|
606
|
+
* - Level 0을 참조하는 fixture들: Level 1
|
|
607
|
+
* - 반복
|
|
608
|
+
*
|
|
609
|
+
* UpsertBuilder가 self-reference가 있으면 buildInsertLevels()로 재정렬하여
|
|
610
|
+
* 등록 순서와 반환 순서가 달라질 수 있습니다.
|
|
611
|
+
* 이를 방지하기 위해 FixtureManager가 레벨별로 나눠서 처리합니다.
|
|
612
|
+
*/
|
|
613
|
+
groupFixturesByLevel(fixtures) {
|
|
614
|
+
if (fixtures.length === 0) {
|
|
615
|
+
return [];
|
|
616
|
+
}
|
|
617
|
+
const entity = EntityManager.get(fixtures[0].entityId);
|
|
618
|
+
const selfRefProps = entity.props.filter((p) => isRelationProp(p) && (isBelongsToOneRelationProp(p) || isOneToOneRelationProp(p) && p.hasJoinColumn) && p.with === entity.id);
|
|
619
|
+
if (selfRefProps.length === 0) {
|
|
620
|
+
return [fixtures];
|
|
621
|
+
}
|
|
622
|
+
const levels = [];
|
|
623
|
+
const remaining = new Set(fixtures.map((f) => f.fixtureId));
|
|
624
|
+
const processed = new Set();
|
|
625
|
+
while (remaining.size > 0) {
|
|
626
|
+
const currentLevel = [];
|
|
627
|
+
for (const fixture of fixtures) {
|
|
628
|
+
if (!remaining.has(fixture.fixtureId)) continue;
|
|
629
|
+
const canProcess = selfRefProps.every((prop) => {
|
|
630
|
+
const refId = fixture.columns[prop.name]?.value;
|
|
631
|
+
if (refId === null || refId === undefined) return true;
|
|
632
|
+
const refFixtureId = `${prop.with}#${refId}`;
|
|
633
|
+
return processed.has(refFixtureId) || !remaining.has(refFixtureId);
|
|
634
|
+
});
|
|
635
|
+
if (canProcess) {
|
|
636
|
+
currentLevel.push(fixture);
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
if (currentLevel.length === 0) {
|
|
640
|
+
const remainingIds = Array.from(remaining).join(", ");
|
|
641
|
+
throw new Error(`Circular self-reference detected in ${entity.table}. Remaining fixtures: ${remainingIds}`);
|
|
642
|
+
}
|
|
643
|
+
for (const fixture of currentLevel) {
|
|
644
|
+
remaining.delete(fixture.fixtureId);
|
|
645
|
+
processed.add(fixture.fixtureId);
|
|
646
|
+
}
|
|
647
|
+
levels.push(currentLevel);
|
|
648
|
+
}
|
|
649
|
+
return levels;
|
|
650
|
+
}
|
|
651
|
+
async checkUniqueViolation(db, entity, fixture) {
|
|
652
|
+
const _uniqueIndexes = entity.indexes?.filter((i) => i.type === "unique") ?? [];
|
|
653
|
+
const uniqueIndexes = _uniqueIndexes.filter((index) => index.columns.every((column) => !column.name.startsWith(`${entity.table}__`)));
|
|
654
|
+
if (uniqueIndexes.length === 0) {
|
|
655
|
+
return null;
|
|
656
|
+
}
|
|
657
|
+
let uniqueQuery = db(entity.table);
|
|
658
|
+
let hasCondition = false;
|
|
659
|
+
for (const index of uniqueIndexes) {
|
|
660
|
+
const containsNull = index.columns.some((column) => {
|
|
661
|
+
const field = column.name.replace(/_id$/, "");
|
|
662
|
+
return fixture.columns[field]?.value === null;
|
|
663
|
+
});
|
|
664
|
+
if (containsNull) {
|
|
665
|
+
continue;
|
|
666
|
+
}
|
|
667
|
+
uniqueQuery = uniqueQuery.orWhere((qb) => {
|
|
668
|
+
for (const column of index.columns) {
|
|
669
|
+
const field = column.name.replace(/_id$/, "");
|
|
670
|
+
if (Array.isArray(fixture.columns[field]?.value)) {
|
|
671
|
+
qb.whereIn(column.name, fixture.columns[field].value);
|
|
672
|
+
} else {
|
|
673
|
+
qb.andWhere(column.name, fixture.columns[field]?.value);
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
});
|
|
677
|
+
hasCondition = true;
|
|
678
|
+
}
|
|
679
|
+
if (!hasCondition) {
|
|
680
|
+
return null;
|
|
681
|
+
}
|
|
682
|
+
const [uniqueFound] = await uniqueQuery;
|
|
683
|
+
return uniqueFound;
|
|
684
|
+
}
|
|
685
|
+
async checkDuplicateByColumns(db, entity, fixture, columns) {
|
|
686
|
+
if (columns.length === 0) {
|
|
687
|
+
return null;
|
|
688
|
+
}
|
|
689
|
+
const whereClause = {};
|
|
690
|
+
for (const column of columns) {
|
|
691
|
+
const prop = entity.props.find((p) => p.name === column);
|
|
692
|
+
const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;
|
|
693
|
+
const value = fixture.columns[column]?.value;
|
|
694
|
+
if (value === null || value === undefined) {
|
|
695
|
+
return null;
|
|
696
|
+
}
|
|
697
|
+
whereClause[dbColumn] = value;
|
|
698
|
+
}
|
|
699
|
+
const [found] = await db(entity.table).where(whereClause).limit(1);
|
|
700
|
+
return found;
|
|
701
|
+
}
|
|
702
|
+
async addFixtureLoader(code) {
|
|
703
|
+
const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;
|
|
704
|
+
const content = readFileSync(path).toString();
|
|
705
|
+
const fixtureLoaderStart = content.indexOf("const fixtureLoader = {");
|
|
706
|
+
const fixtureLoaderEnd = content.indexOf("};", fixtureLoaderStart);
|
|
707
|
+
if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {
|
|
708
|
+
const newContent = `${content.slice(0, fixtureLoaderEnd)} ${code}\n${content.slice(fixtureLoaderEnd)}`;
|
|
709
|
+
writeFileSync(path, newContent);
|
|
710
|
+
} else {
|
|
711
|
+
throw new Error("Failed to find fixtureLoader in fixture.ts");
|
|
712
|
+
}
|
|
713
|
+
}
|
|
714
|
+
};
|
|
715
|
+
FixtureManager = new FixtureManagerClass();
|
|
716
|
+
}));
|
|
790
717
|
|
|
791
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90ZXN0aW5nL2ZpeHR1cmUtbWFuYWdlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgYXNzZXJ0IGZyb20gXCJhc3NlcnRcIjtcbmltcG9ydCBjaGFsayBmcm9tIFwiY2hhbGtcIjtcbmltcG9ydCB7IGV4ZWNTeW5jIH0gZnJvbSBcImNoaWxkX3Byb2Nlc3NcIjtcbmltcG9ydCB7IHJlYWRGaWxlU3luYywgd3JpdGVGaWxlU3luYyB9IGZyb20gXCJmc1wiO1xuaW1wb3J0IGluZmxlY3Rpb24gZnJvbSBcImluZmxlY3Rpb25cIjtcbmltcG9ydCB0eXBlIHsgS25leCB9IGZyb20gXCJrbmV4XCI7XG5pbXBvcnQgeyB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgaW5zcGVjdCB9IGZyb20gXCJ1dGlsXCI7XG5pbXBvcnQgeyBTb25hbXUgfSBmcm9tIFwiLi4vYXBpXCI7XG5pbXBvcnQgeyBCYXNlTW9kZWwgfSBmcm9tIFwiLi4vZGF0YWJhc2UvYmFzZS1tb2RlbFwiO1xuaW1wb3J0IHR5cGUgeyBTb25hbXVEQkNvbmZpZyB9IGZyb20gXCIuLi9kYXRhYmFzZS9kYlwiO1xuaW1wb3J0IHsgY3JlYXRlS25leEluc3RhbmNlIH0gZnJvbSBcIi4uL2RhdGFiYXNlL2tuZXhcIjtcbmltcG9ydCB7IHR5cGUgVUJSZWYsIFVwc2VydEJ1aWxkZXIgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB0eXBlIHsgRW50aXR5IH0gZnJvbSBcIi4uL2VudGl0eS9lbnRpdHlcIjtcbmltcG9ydCB7IEVudGl0eU1hbmFnZXIgfSBmcm9tIFwiLi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5pbXBvcnQge1xuICB0eXBlIEJlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCxcbiAgdHlwZSBEYXRhYmFzZVNjaGVtYUV4dGVuZCxcbiAgdHlwZSBFbnRpdHlQcm9wLFxuICB0eXBlIEZpeHR1cmVJbXBvcnRSZXN1bHQsXG4gIHR5cGUgRml4dHVyZVJlY29yZCxcbiAgdHlwZSBGaXh0dXJlU2VhcmNoT3B0aW9ucyxcbiAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AsXG4gIGlzSGFzTWFueVJlbGF0aW9uUHJvcCxcbiAgaXNNYW55VG9NYW55UmVsYXRpb25Qcm9wLFxuICBpc09uZVRvT25lUmVsYXRpb25Qcm9wLFxuICBpc1JlbGF0aW9uUHJvcCxcbiAgaXNWaXJ0dWFsUHJvcCxcbiAgdHlwZSBNYW55VG9NYW55UmVsYXRpb25Qcm9wLFxuICB0eXBlIE9uZVRvT25lUmVsYXRpb25Qcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7IGlzVGVzdCB9IGZyb20gXCIuLi91dGlscy9jb250cm9sbGVyXCI7XG5pbXBvcnQgeyBSZWxhdGlvbkdyYXBoIH0gZnJvbSBcIi4vX3JlbGF0aW9uLWdyYXBoXCI7XG5cbi8qKiDsgqzsmqnsnpAg7KeA7KCVIOykkeuztSDtmZXsnbgg7Lus65+8IChlbnRpdHlJZOuzhOuhnCDsp4DsoJUpICovXG5leHBvcnQgaW50ZXJmYWNlIER1cGxpY2F0ZUNoZWNrT3B0aW9ucyB7XG4gIGNvbHVtbnM/OiB7XG4gICAgW2VudGl0eUlkOiBzdHJpbmddOiBzdHJpbmdbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGNsYXNzIEZpeHR1cmVNYW5hZ2VyQ2xhc3Mge1xuICBwcml2YXRlIF90ZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IHRkYih0ZGI6IEtuZXgpIHtcbiAgICB0aGlzLl90ZGIgPSB0ZGI7XG4gIH1cbiAgZ2V0IHRkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fdGRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl90ZGI7XG4gIH1cblxuICBwcml2YXRlIF9mZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IGZkYihmZGI6IEtuZXgpIHtcbiAgICB0aGlzLl9mZGIgPSBmZGI7XG4gIH1cbiAgZ2V0IGZkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fZmRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9mZGI7XG4gIH1cbiAgY2FjaGVkVGFibGVOYW1lczogc3RyaW5nW10gfCBudWxsID0gbnVsbDtcblxuICBwcml2YXRlIHJlbGF0aW9uR3JhcGggPSBuZXcgUmVsYXRpb25HcmFwaCgpO1xuXG4gIC8vIFVwc2VydEJ1aWxkZXIg6riw67CYIGltcG9ydOulvCDsnITtlZwg7IOB7YOcXG4gIHByaXZhdGUgYnVpbGRlcjogVXBzZXJ0QnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gIHByaXZhdGUgZml4dHVyZVJlZk1hcDogTWFwPHN0cmluZywgVUJSZWY+ID0gbmV3IE1hcCgpO1xuICBwcml2YXRlIHNraXBwZWRGaXh0dXJlczogTWFwPHN0cmluZywgeyBlbnRpdHlJZDogc3RyaW5nOyBleGlzdGluZ0lkOiBudW1iZXIgfCBzdHJpbmcgfT4gPVxuICAgIG5ldyBNYXAoKTtcblxuICBpbml0KCkge1xuICAgIGlmICh0aGlzLl90ZGIgIT09IG51bGwpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKFNvbmFtdS5kYkNvbmZpZy50ZXN0ICYmIFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlcikge1xuICAgICAgY29uc3QgdENvbm4gPSBTb25hbXUuZGJDb25maWcudGVzdC5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBjb25zdCBwQ29ubiA9IFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlci5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBpZiAoXG4gICAgICAgIGAke3RDb25uLmhvc3QgPz8gXCJsb2NhbGhvc3RcIn06JHt0Q29ubi5wb3J0ID8/IDU0MzJ9LyR7dENvbm4uZGF0YWJhc2V9YCA9PT1cbiAgICAgICAgYCR7cENvbm4uaG9zdCA/PyBcImxvY2FsaG9zdFwifToke3BDb25uLnBvcnQgPz8gNTQzMn0vJHtwQ29ubi5kYXRhYmFzZX1gXG4gICAgICApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDthYzsiqTtirhEQuyZgCDtlITroZzrjZXshZhEQuyXkCDrj5nsnbztlZwg642w7J207YSw67Kg7J207Iqk6rCAIOyCrOyaqeuQmOyXiOyKteuLiOuLpC5gKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICB0aGlzLnRkYiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWcudGVzdCk7XG4gICAgdGhpcy5mZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUpO1xuICB9XG5cbiAgLyoqXG4gICAg7JuQ6rKpIGZpeHR1cmUgRELrpbwg66Gc7LusIHRlc3QgRELroZwg67O17IKs7ZWp64uI64ukLlxuICAgIHBnX2R1bXDroZwg7JuQ6rKpIERC66W8IOuNpO2UhO2VmOqzoCwgcGdfcmVzdG9yZeuhnCDroZzsu6zsl5Ag67O17JuQ7ZWp64uI64ukLlxuICAqL1xuICBhc3luYyBzeW5jKCkge1xuICAgIGNvbnN0IGZpeHR1cmVDb25uID0gU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LlBnQ29ubmVjdGlvbkNvbmZpZztcbiAgICBjb25zdCB0ZXN0Q29ubiA9IFNvbmFtdS5kYkNvbmZpZy50ZXN0LmNvbm5lY3Rpb24gYXMgS25leC5QZ0Nvbm5lY3Rpb25Db25maWc7XG5cbiAgICAvLyAxLiDroZzsu6wgdGVzdCBEQiDsl7DqsrAg7KKF66OMIOuwjyDsnqzsg53shLFcbiAgICBjb25zdCB0ZXN0UGdFbnYgPSB7IFBHUEFTU1dPUkQ6IHRlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCIgfTtcbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJcbiAgICAgICAgU0VMRUNUIHBnX3Rlcm1pbmF0ZV9iYWNrZW5kKHBnX3N0YXRfYWN0aXZpdHkucGlkKVxuICAgICAgICBGUk9NIHBnX3N0YXRfYWN0aXZpdHlcbiAgICAgICAgV0hFUkUgZGF0bmFtZSA9ICcke3Rlc3RDb25uLmRhdGFiYXNlfSdcbiAgICAgICAgICBBTkQgcGlkIDw+IHBnX2JhY2tlbmRfcGlkKCk7XG4gICAgICBcImAsXG4gICAgICB7IHN0ZGlvOiBcImluaGVyaXRcIiwgZW52OiB7IC4uLnByb2Nlc3MuZW52LCAuLi50ZXN0UGdFbnYgfSBhcyBOb2RlSlMuUHJvY2Vzc0VudiB9LFxuICAgICk7XG5cbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJEUk9QIERBVEFCQVNFIElGIEVYSVNUUyBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIGV4ZWNTeW5jKFxuICAgICAgYHBzcWwgLWggJHt0ZXN0Q29ubi5ob3N0fSAtcCAke3Rlc3RDb25uLnBvcnQgPz8gNTQzMn0gLVUgJHt0ZXN0Q29ubi51c2VyfSAtZCBwb3N0Z3JlcyAtYyBcIkNSRUFURSBEQVRBQkFTRSBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIC8vIDIuIOybkOqyqSBmaXh0dXJlIERCIOKGkiDroZzsu6wgdGVzdCBEQuuhnCDrs7XsgqwgKHBnX2R1bXAgfCBwZ19yZXN0b3JlKVxuICAgIGNvbnN0IGZpeHR1cmVQZ0VudiA9IHsgUEdQQVNTV09SRDogZml4dHVyZUNvbm4ucGFzc3dvcmQgfHwgXCJcIiB9O1xuICAgIGNvbnN0IGR1bXBDbWQgPSBgcGdfZHVtcCAtaCAke2ZpeHR1cmVDb25uLmhvc3R9IC1wICR7Zml4dHVyZUNvbm4ucG9ydCA/PyA1NDMyfSAtVSAke2ZpeHR1cmVDb25uLnVzZXJ9IC1kICR7Zml4dHVyZUNvbm4uZGF0YWJhc2V9IC1GY2A7XG4gICAgY29uc3QgcmVzdG9yZUNtZCA9IGBwZ19yZXN0b3JlIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgJHt0ZXN0Q29ubi5kYXRhYmFzZX0gLS1uby1vd25lciAtLW5vLWFjbGA7XG5cbiAgICBleGVjU3luYyhgJHtkdW1wQ21kfSB8IFBHUEFTU1dPUkQ9XCIke3Rlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCJ9XCIgJHtyZXN0b3JlQ21kfWAsIHtcbiAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICAgIGVudjogeyAuLi5wcm9jZXNzLmVudiwgLi4uZml4dHVyZVBnRW52IH0gYXMgTm9kZUpTLlByb2Nlc3NFbnYsXG4gICAgICBzaGVsbDogXCIvYmluL2Jhc2hcIixcbiAgICB9KTtcblxuICAgIC8vIDMuIOyLnO2AgOyKpCDrpqzshYsgKOuNsOydtO2EsCDrs7Xsgqwg7ZuEIOyLnO2AgOyKpOulvCBNQVgoaWQp66GcIOygleugrClcbiAgICBhd2FpdCB0aGlzLnJlc2V0U2VxdWVuY2VzKFNvbmFtdS5kYkNvbmZpZy50ZXN0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiDrqqjrk6Ag7YWM7J2067iU7J2YIOyLnO2AgOyKpOulvCDtmITsnqwgTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAqIGZpeHR1cmUgc3luYyDtm4Qg7Iuc7YCA7Iqk6rCAIOyLpOygnCDrjbDsnbTthLDsmYAg66ee7KeAIOyViuuKlCDrrLjsoJzrpbwg7ZW06rKw7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyByZXNldFNlcXVlbmNlcyhkYkNvbmZpZzogU29uYW11REJDb25maWdbXCJ0ZXN0XCJdKSB7XG4gICAgY29uc3QgdGVzdERiID0gY3JlYXRlS25leEluc3RhbmNlKGRiQ29uZmlnKTtcbiAgICBjb25zdCBlbnRpdGllcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKTtcblxuICAgIHRyeSB7XG4gICAgICBmb3IgKGNvbnN0IGVudGl0eSBvZiBlbnRpdGllcykge1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGUgfHwgZW50aXR5LmlkLnRvTG93ZXJDYXNlKCk7XG5cbiAgICAgICAgLy8gaWQg7ZWE65Oc7J2YIO2DgOyeheydhCDtmZXsnbjtlanri4jri6RcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgY29uc3QgaWRUeXBlID0gaWRQcm9wPy50eXBlO1xuXG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlcuydtOqxsOuCmCwgc3RyaW5n7J207KeA66eMIGZpeHR1cmVTdHJhdGVneT1zZXF1ZW5jZeyduCDqsr3smrDsl5Drp4wg66as7IWL7ZWp64uI64ukXG4gICAgICAgIGNvbnN0IHVzZXNTZXF1ZW5jZSA9XG4gICAgICAgICAgaWRUeXBlID09PSBcImludGVnZXJcIiB8fFxuICAgICAgICAgIGlkVHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICBpZFByb3A/LmNvbmU/LmZpeHR1cmVTdHJhdGVneSA9PT0gXCJzZXF1ZW5jZVwiO1xuXG4gICAgICAgIGlmICghdXNlc1NlcXVlbmNlKSB7XG4gICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgYFNraXBwaW5nIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX0gKGlkIHR5cGU6ICR7aWRUeXBlIHx8IFwidW5rbm93blwifSlgLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFBvc3RncmVTUUwg7Iuc7YCA7Iqk66W8IO2YhOyerCDthYzsnbTruJTsnZggTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAgICAgIC8vIHN0cmluZyDtg4DsnoXsnZgg6rK97JqwIOyIq+yekCDsupDsiqTtjIXsnbQg7ZWE7JqU7ZWp64uI64ukLlxuICAgICAgICBjb25zdCBtYXhFeHByID0gaWRUeXBlID09PSBcInN0cmluZ1wiID8gXCJNQVgoaWQ6OmJpZ2ludClcIiA6IFwiTUFYKGlkKVwiO1xuICAgICAgICBhd2FpdCB0ZXN0RGIucmF3KGBcbiAgICAgICAgICBTRUxFQ1Qgc2V0dmFsKFxuICAgICAgICAgICAgcGdfZ2V0X3NlcmlhbF9zZXF1ZW5jZSgncHVibGljLiR7dGFibGVOYW1lfScsICdpZCcpLFxuICAgICAgICAgICAgQ09BTEVTQ0UoKFNFTEVDVCAke21heEV4cHJ9IEZST00gJHt0YWJsZU5hbWV9KSwgMSlcbiAgICAgICAgICApXG4gICAgICAgIGApO1xuICAgICAgfVxuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCB0ZXN0RGIuZGVzdHJveSgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgdmlzaXRlZFJlY29yZHMgPSBuZXcgU2V0PHN0cmluZz4oKTtcbiAgYXN5bmMgaW1wb3J0Rml4dHVyZShlbnRpdHlJZDogc3RyaW5nLCBpZHM6IG51bWJlcltdKSB7XG4gICAgLy8g67Cp66y4IOq4sOuhnSDstIjquLDtmZQgKOyDiOuhnOyatCBpbXBvcnQg7J6R7JeFIOyLnOyekSlcbiAgICB0aGlzLnZpc2l0ZWRSZWNvcmRzLmNsZWFyKCk7XG5cbiAgICBjb25zdCBxdWVyaWVzID0gdW5pcXVlKFxuICAgICAgKFxuICAgICAgICBhd2FpdCBQcm9taXNlLmFsbChcbiAgICAgICAgICBpZHMubWFwKGFzeW5jIChpZCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZCwgXCJpZFwiLCBpZCk7XG4gICAgICAgICAgfSksXG4gICAgICAgIClcbiAgICAgICkuZmxhdCgpLFxuICAgICk7XG5cbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuICAgIGZvciAoY29uc3QgcXVlcnkgb2YgcXVlcmllcykge1xuICAgICAgYXdhaXQgd2RiLnJhdyhxdWVyeSk7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZDogc3RyaW5nLCBmaWVsZDogc3RyaW5nLCBpZDogbnVtYmVyKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHJlY29yZEtleSA9IGAke2VudGl0eUlkfSMke2ZpZWxkfSMke2lkfWA7XG5cbiAgICAvLyDsiJztmZgg7LC47KGwIOuwqeyngDog7J2066+4IOuwqeusuO2VnCDroIjsvZTrk5zripQg7Iqk7YK1XG4gICAgaWYgKHRoaXMudmlzaXRlZFJlY29yZHMuaGFzKHJlY29yZEtleSkpIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG4gICAgdGhpcy52aXNpdGVkUmVjb3Jkcy5hZGQocmVjb3JkS2V5KTtcblxuICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKTtcbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuXG4gICAgLy8g7Jes6riw7IScIOyLpERC7J2YIHJvdyDqsIDsoLjsmLRcbiAgICBjb25zdCBbcm93XSA9IGF3YWl0IHdkYihlbnRpdHkudGFibGUpLndoZXJlKGZpZWxkLCBpZCkubGltaXQoMSk7XG4gICAgaWYgKHJvdyA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5SWR9IyR7aWR9IHJvd+ulvCDssL7snYQg7IiYIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICAvLyDtlL3siqTss5BEQiwg7IukREJcbiAgICBjb25zdCBmaXh0dXJlRGF0YWJhc2UgPSAoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpLmRhdGFiYXNlO1xuICAgIGNvbnN0IHJlYWxEYXRhYmFzZSA9IChTb25hbXUuZGJDb25maWcucHJvZHVjdGlvbl9tYXN0ZXIuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpXG4gICAgICAuZGF0YWJhc2U7XG5cbiAgICBjb25zdCBzZWxmUXVlcnkgPSBgSU5TRVJUIElHTk9SRSBJTlRPIFxcYCR7Zml4dHVyZURhdGFiYXNlfVxcYC5cXGAke2VudGl0eS50YWJsZX1cXGAgKFNFTEVDVCAqIEZST00gXFxgJHtyZWFsRGF0YWJhc2V9XFxgLlxcYCR7ZW50aXR5LnRhYmxlfVxcYCBXSEVSRSBcXGBpZFxcYCA9ICR7aWR9KWA7XG5cbiAgICBjb25zdCBhcmdzID0gT2JqZWN0LmVudHJpZXMoZW50aXR5LnJlbGF0aW9ucylcbiAgICAgIC5maWx0ZXIoXG4gICAgICAgIChbLCByZWxhdGlvbl0pID0+XG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8XG4gICAgICAgICAgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pICYmIHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UgPT09IHVuZGVmaW5lZCksXG4gICAgICApXG4gICAgICAubWFwKChbLCByZWxhdGlvbl0pID0+IHtcbiAgICAgICAgLypcbiAgICAgICAgQmVsb25nc1RvT25l7J24IOqyveyasFxuICAgICAgICAgIENhdGVnb3J5IC8gJ2lkJyAvIHJvd1tjYXRlZ29yeV9pZF0g7Zi47LacXG4gICAgICAgIE9uZVRvT25l7JeQIGpvaW5Db2x1bW4gPT09IHRydWUg7J24IOqyveyasFxuICAgICAgICAgIFByb2ZpbGUgLyAnaWQnIC8gcm93W3Byb2ZpbGVfaWRdIO2YuOy2nFxuICAgICAgICBPbmVUb09uZeyXkCBqb2luQ29sdW1uID09PSBmYWxzZSDsnbgg6rK97JqwXG4gICAgICAgICAgUHJvZmlsZSAvICdwcm9maWxlX2lkJyAvIHJvd1snaWQnXSDtmLjstpxcbiAgICAgICAgKi9cbiAgICAgICAgbGV0IGZpZWxkOiBzdHJpbmc7XG4gICAgICAgIGxldCBpZDogbnVtYmVyO1xuICAgICAgICBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChyZWxhdGlvbikgJiYgIXJlbGF0aW9uLmhhc0pvaW5Db2x1bW4pIHtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG4gICAgICAgICAgY29uc3QgcmVsYXRlZElkQ29sdW1uTmFtZSA9IHJlbGF0ZWRFbnRpdHkucHJvcHMuZmluZChcbiAgICAgICAgICAgIChwKSA9PiBpc1JlbGF0aW9uUHJvcChwKSAmJiBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICAgICAgICApPy5uYW1lO1xuICAgICAgICAgIGlmICghcmVsYXRlZElkQ29sdW1uTmFtZSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3JlbGF0ZWRFbnRpdHkuaWR97J2YICR7ZW50aXR5LmlkfSDqtIDqs4Qg7ZSE66Gt7J2EIOywvuydhCDsiJgg7JeG7Iq164uI64ukLmApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBmaWVsZCA9IGAke3JlbGF0ZWRJZENvbHVtbk5hbWV9X2lkYDtcbiAgICAgICAgICBpZCA9IHJvdy5pZDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBmaWVsZCA9IFwiaWRcIjtcbiAgICAgICAgICBpZCA9IHJvd1tgJHtyZWxhdGlvbi5uYW1lfV9pZGBdO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZW50aXR5SWQ6IHJlbGF0aW9uLndpdGgsXG4gICAgICAgICAgZmllbGQsXG4gICAgICAgICAgaWQsXG4gICAgICAgIH07XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoYXJnKSA9PiBhcmcuaWQgIT09IG51bGwpO1xuXG4gICAgY29uc3QgcmVsUXVlcmllcyA9IGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgYXJncy5tYXAoYXN5bmMgKGFyZ3MpID0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhhcmdzLmVudGl0eUlkLCBhcmdzLmZpZWxkLCBhcmdzLmlkKTtcbiAgICAgIH0pLFxuICAgICk7XG5cbiAgICByZXR1cm4gWy4uLnVuaXF1ZShyZWxRdWVyaWVzLnJldmVyc2UoKS5mbGF0KCkpLCBzZWxmUXVlcnldO1xuICB9XG5cbiAgYXN5bmMgZGVzdHJveSgpIHtcbiAgICBpZiAodGhpcy5fdGRiKSB7XG4gICAgICBhd2FpdCB0aGlzLl90ZGIuZGVzdHJveSgpO1xuICAgICAgdGhpcy5fdGRiID0gbnVsbDtcbiAgICB9XG4gICAgaWYgKHRoaXMuX2ZkYikge1xuICAgICAgYXdhaXQgdGhpcy5fZmRiLmRlc3Ryb3koKTtcbiAgICAgIHRoaXMuX2ZkYiA9IG51bGw7XG4gICAgfVxuICAgIGF3YWl0IEJhc2VNb2RlbC5kZXN0cm95KCk7XG4gIH1cblxuICBhc3luYyBnZXRGaXh0dXJlcyhcbiAgICBzb3VyY2VEQk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIHRhcmdldERCTmFtZToga2V5b2YgU29uYW11REJDb25maWcsXG4gICAgc2VhcmNoT3B0aW9uczogRml4dHVyZVNlYXJjaE9wdGlvbnMsXG4gICAgZHVwbGljYXRlQ2hlY2s/OiBEdXBsaWNhdGVDaGVja09wdGlvbnMsXG4gICkge1xuICAgIGNvbnN0IHNvdXJjZURCID0gY3JlYXRlS25leEluc3RhbmNlKFNvbmFtdS5kYkNvbmZpZ1tzb3VyY2VEQk5hbWVdKTtcbiAgICBjb25zdCB0YXJnZXREQiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWdbdGFyZ2V0REJOYW1lXSk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgeyBlbnRpdHlJZCwgZmllbGQsIHZhbHVlLCBzZWFyY2hUeXBlIH0gPSBzZWFyY2hPcHRpb25zO1xuXG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCk7XG4gICAgICBjb25zdCBjb2x1bW4gPVxuICAgICAgICBlbnRpdHkucHJvcHMuZmluZCgocHJvcCkgPT4gcHJvcC5uYW1lID09PSBmaWVsZCk/LnR5cGUgPT09IFwicmVsYXRpb25cIlxuICAgICAgICAgID8gYCR7ZmllbGR9X2lkYFxuICAgICAgICAgIDogZmllbGQ7XG5cbiAgICAgIGxldCBxdWVyeSA9IHNvdXJjZURCKGVudGl0eS50YWJsZSk7XG4gICAgICBpZiAoc2VhcmNoVHlwZSA9PT0gXCJlcXVhbHNcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgdmFsdWUpO1xuICAgICAgfSBlbHNlIGlmIChzZWFyY2hUeXBlID09PSBcImxpa2VcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgXCJsaWtlXCIsIGAlJHt2YWx1ZX0lYCk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHJvd3MgPSBhd2FpdCBxdWVyeTtcbiAgICAgIGlmIChyb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJObyByZWNvcmRzIGZvdW5kXCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmaXh0dXJlczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICAgIGNvbnN0IGluaXRpYWxSZWNvcmRzTGVuZ3RoID0gZml4dHVyZXMubGVuZ3RoO1xuICAgICAgICBjb25zdCBuZXdSZWNvcmRzID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgcm93LCB7XG4gICAgICAgICAgX2RiOiBzb3VyY2VEQixcbiAgICAgICAgfSk7XG4gICAgICAgIGZpeHR1cmVzLnB1c2goLi4ubmV3UmVjb3Jkcyk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRGaXh0dXJlUmVjb3JkID0gZml4dHVyZXMuZmluZCgocikgPT4gci5maXh0dXJlSWQgPT09IGAke2VudGl0eUlkfSMke3Jvdy5pZH1gKTtcblxuICAgICAgICBpZiAoY3VycmVudEZpeHR1cmVSZWNvcmQpIHtcbiAgICAgICAgICAvLyDtmITsnqwgZml4dHVyZeuhnOu2gO2EsCDsg53shLHrkJwgZmV0Y2hlZFJlY29yZHMg7ISk7KCVXG4gICAgICAgICAgY3VycmVudEZpeHR1cmVSZWNvcmQuZmV0Y2hlZFJlY29yZHMgPSBmaXh0dXJlc1xuICAgICAgICAgICAgLmZpbHRlcigocikgPT4gci5maXh0dXJlSWQgIT09IGN1cnJlbnRGaXh0dXJlUmVjb3JkLmZpeHR1cmVJZClcbiAgICAgICAgICAgIC5zbGljZShpbml0aWFsUmVjb3Jkc0xlbmd0aClcbiAgICAgICAgICAgIC5tYXAoKHIpID0+IHIuZml4dHVyZUlkKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZS5lbnRpdHlJZCk7XG5cbiAgICAgICAgLy8g7IKs7Jqp7J6QIOyngOyglSDsu6zrn7wg6riw7KSAIOykkeuztSDtmZXsnbgg4oaSIHRhcmdldFxuICAgICAgICBjb25zdCBjdXN0b21Db2x1bW5zID0gZHVwbGljYXRlQ2hlY2s/LmNvbHVtbnM/LltmaXh0dXJlLmVudGl0eUlkXTtcbiAgICAgICAgaWYgKGN1c3RvbUNvbHVtbnMgJiYgY3VzdG9tQ29sdW1ucy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgY3VzdG9tRHVwbGljYXRlUm93ID0gYXdhaXQgdGhpcy5jaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICAgICAgICAgIHRhcmdldERCLFxuICAgICAgICAgICAgZW50aXR5LFxuICAgICAgICAgICAgZml4dHVyZSxcbiAgICAgICAgICAgIGN1c3RvbUNvbHVtbnMsXG4gICAgICAgICAgKTtcbiAgICAgICAgICBpZiAoY3VzdG9tRHVwbGljYXRlUm93KSB7XG4gICAgICAgICAgICBjb25zdCBbcmVjb3JkXSA9IGF3YWl0IHRoaXMuY3JlYXRlRml4dHVyZVJlY29yZChlbnRpdHksIGN1c3RvbUR1cGxpY2F0ZVJvdywge1xuICAgICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICAgIF9kYjogdGFyZ2V0REIsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGZpeHR1cmUudGFyZ2V0ID0gcmVjb3JkO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVuaXF1ZSBpbmRleCDquLDspIAg7KSR67O1IO2ZleyduCDihpIgZml4dHVyZS51bmlxdWVcbiAgICAgICAgY29uc3QgdW5pcXVlUm93ID0gYXdhaXQgdGhpcy5jaGVja1VuaXF1ZVZpb2xhdGlvbih0YXJnZXREQiwgZW50aXR5LCBmaXh0dXJlKTtcbiAgICAgICAgaWYgKHVuaXF1ZVJvdykge1xuICAgICAgICAgIGNvbnN0IFtyZWNvcmRdID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgdW5pcXVlUm93LCB7XG4gICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICBfZGI6IHRhcmdldERCLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGZpeHR1cmUudW5pcXVlID0gcmVjb3JkO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB1bmlxdWUoZml4dHVyZXMsIChmKSA9PiBmLmZpeHR1cmVJZCk7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZChbdGFyZ2V0REIuZGVzdHJveSgpLCBzb3VyY2VEQi5kZXN0cm95KCldKTtcbiAgICB9XG4gIH1cblxuICBhc3luYyBjcmVhdGVGaXh0dXJlUmVjb3JkKFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIHJvdzoge1xuICAgICAgaWQ6IG51bWJlciB8IHN0cmluZztcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsO1xuICAgIH0sXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIHNpbmdsZVJlY29yZD86IGJvb2xlYW47XG4gICAgICBfZGI/OiBLbmV4O1xuICAgIH0sXG4gICk6IFByb21pc2U8Rml4dHVyZVJlY29yZFtdPiB7XG4gICAgY29uc3QgcmVjb3JkczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgY29uc3QgdmlzaXRlZEVudGl0aWVzID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICBjb25zdCBjcmVhdGUgPSBhc3luYyAoXG4gICAgICBlbnRpdHk6IEVudGl0eSxcbiAgICAgIHJvdzoge1xuICAgICAgICBpZDogbnVtYmVyIHwgc3RyaW5nO1xuICAgICAgICBba2V5OiBzdHJpbmddOiBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgbnVsbDtcbiAgICAgIH0sXG4gICAgKSA9PiB7XG4gICAgICBjb25zdCBmaXh0dXJlSWQgPSBgJHtlbnRpdHkuaWR9IyR7cm93LmlkfWA7XG4gICAgICBpZiAodmlzaXRlZEVudGl0aWVzLmhhcyhmaXh0dXJlSWQpKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIHZpc2l0ZWRFbnRpdGllcy5hZGQoZml4dHVyZUlkKTtcblxuICAgICAgY29uc3QgcmVjb3JkOiBGaXh0dXJlUmVjb3JkID0ge1xuICAgICAgICBmaXh0dXJlSWQsXG4gICAgICAgIGVudGl0eUlkOiBlbnRpdHkuaWQsXG4gICAgICAgIGlkOiByb3cuaWQsXG4gICAgICAgIGNvbHVtbnM6IHt9LFxuICAgICAgICBmZXRjaGVkUmVjb3JkczogW10sXG4gICAgICAgIGJlbG9uZ3NSZWNvcmRzOiBbXSxcbiAgICAgIH07XG5cbiAgICAgIGZvciAoY29uc3QgcHJvcCBvZiBlbnRpdHkucHJvcHMpIHtcbiAgICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0gPSB7XG4gICAgICAgICAgcHJvcDogcHJvcCxcbiAgICAgICAgICB2YWx1ZTogcm93W3Byb3AubmFtZV0sXG4gICAgICAgIH07XG5cbiAgICAgICAgY29uc3QgZGIgPSBvcHRpb25zPy5fZGIgPz8gQmFzZU1vZGVsLmdldERCKFwid1wiKTtcbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHRocm91Z2hUYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IGZyb21Db2x1bW4gPSBgJHtpbmZsZWN0aW9uLnNpbmd1bGFyaXplKGVudGl0eS50YWJsZSl9X2lkYDtcbiAgICAgICAgICBjb25zdCB0b0NvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYih0aHJvdWdoVGFibGUpLndoZXJlKGZyb21Db2x1bW4sIHJvdy5pZCkucGx1Y2sodG9Db2x1bW4pO1xuICAgICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0udmFsdWUgPSByZWxhdGVkSWRzO1xuICAgICAgICB9IGVsc2UgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKVxuICAgICAgICAgICAgLndoZXJlKHByb3Auam9pbkNvbHVtbiwgcm93LmlkKVxuICAgICAgICAgICAgLnBsdWNrKFwiaWRcIik7XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZHM7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiAhcHJvcC5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgLy8g7Jet67Cp7ZalIE9uZVRvT25lOiBGS+ulvCDqsIDsp4Qg6rSA66CoIOyXlO2LsO2LsOulvCDssL7sirXri4jri6RcbiAgICAgICAgICAvLyDsmIjsi5w6IFVzZXIgT25lVG9PbmUgRW1wbG95ZWUgKEVtcGxveWVl6rCAIHVzZXJfaWQgRkvrpbwg6rCA7KeQKVxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRQcm9wID0gcmVsYXRlZEVudGl0eS5wcm9wcy5maW5kKFxuICAgICAgICAgICAgKHApID0+IGlzUmVsYXRpb25Qcm9wKHApICYmIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICAgICAgICk7XG4gICAgICAgICAgaWYgKHJlbGF0ZWRQcm9wICYmIGlzUmVsYXRpb25Qcm9wKHJlbGF0ZWRQcm9wKSkge1xuICAgICAgICAgICAgLy8g6rSA66CoIOyXlO2LsO2LsOyXkOyEnCBGSyDsu6zrn7zsnLzroZwg7L+866as7ZWp64uI64ukIChpZOqwgCDslYTri5gpXG4gICAgICAgICAgICBjb25zdCBma0NvbHVtbiA9IGAke3JlbGF0ZWRQcm9wLm5hbWV9X2lkYDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSb3cgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKS53aGVyZShma0NvbHVtbiwgcm93LmlkKS5maXJzdCgpO1xuICAgICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRSb3c/LmlkO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IHJvd1tgJHtwcm9wLm5hbWV9X2lkYF07XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICBpZiAocmVsYXRlZElkKSB7XG4gICAgICAgICAgICByZWNvcmQuYmVsb25nc1JlY29yZHMucHVzaChgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIW9wdGlvbnM/LnNpbmdsZVJlY29yZCAmJiByZWxhdGVkSWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJvdyA9IGF3YWl0IGRiKHJlbGF0ZWRFbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgcmVsYXRlZElkKS5maXJzdCgpO1xuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSb3cpIHtcbiAgICAgICAgICAgICAgYXdhaXQgY3JlYXRlKHJlbGF0ZWRFbnRpdHksIHJlbGF0ZWRSb3cpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZWNvcmRzLnB1c2gocmVjb3JkKTtcbiAgICB9O1xuXG4gICAgYXdhaXQgY3JlYXRlKGVudGl0eSwgcm93KTtcblxuICAgIHJldHVybiByZWNvcmRzO1xuICB9XG5cbiAgLyoqXG4gICAqIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrAgKHNlbGYtcmVmZXJlbmNlIO2PrO2VqClcbiAgICogMi4g7YWM7J2067iU67OEIOugiOuyqOuzhOuhnCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnSDrsI8gdXBzZXJ0IOyLpO2WiVxuICAgKiAzLiDsiJzshJwg6riw67CYIHV1aWTihpJpZCDrp6TtlZEgKFVwc2VydEJ1aWxkZXLqsIAgdXVpZOulvCBEQuyXkCDsoIDsnqXtlZjsp4Ag7JWK7Jy866+A66GcKVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy64qUIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0XG4gICAqIEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWY7JesIOqwgSB1cHNlcnQg7Zi47Lac7JeQ7ISc64qUXG4gICAqIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuPhOuhnSDtlanri4jri6QuXG4gICAqL1xuICBhc3luYyBpbnNlcnRGaXh0dXJlcyhcbiAgICBkYk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIF9maXh0dXJlczogRml4dHVyZVJlY29yZFtdLFxuICApOiBQcm9taXNlPEZpeHR1cmVJbXBvcnRSZXN1bHRbXT4ge1xuICAgIGNvbnN0IGZpeHR1cmVzID0gdW5pcXVlKF9maXh0dXJlcywgKGYpID0+IGYuZml4dHVyZUlkKTtcblxuICAgIC8vIOy0iOq4sO2ZlFxuICAgIHRoaXMuYnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gICAgdGhpcy5maXh0dXJlUmVmTWFwID0gbmV3IE1hcCgpO1xuICAgIHRoaXMuc2tpcHBlZEZpeHR1cmVzID0gbmV3IE1hcCgpO1xuXG4gICAgLy8g67OR66CsIO2FjOyKpO2KuCDrqqjrk5zsl5DshJzripQgd29ya2Vy67OEIERC7JeQIOyggOyepVxuICAgIGNvbnN0IGRiQ29uZmlnID1cbiAgICAgIHByb2Nlc3MuZW52LlNPTkFNVV9XT1JLRVJfREIgPT09IFwidHJ1ZVwiICYmIHByb2Nlc3MuZW52LlZJVEVTVF9QT09MX0lEXG4gICAgICAgID8gKCgpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHdvcmtlcklkID0gcGFyc2VJbnQocHJvY2Vzcy5lbnYuVklURVNUX1BPT0xfSUQgPz8gXCIxXCIsIDEwKTtcbiAgICAgICAgICAgIGNvbnN0IGJhc2VDb25maWcgPSBTb25hbXUuZGJDb25maWdbZGJOYW1lXTtcbiAgICAgICAgICAgIGNvbnN0IGNvbm5lY3Rpb24gPSBiYXNlQ29uZmlnLmNvbm5lY3Rpb24gYXMgeyBkYXRhYmFzZTogc3RyaW5nIH07XG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAuLi5iYXNlQ29uZmlnLFxuICAgICAgICAgICAgICBjb25uZWN0aW9uOiB7IC4uLmNvbm5lY3Rpb24sIGRhdGFiYXNlOiBgJHtjb25uZWN0aW9uLmRhdGFiYXNlfV8ke3dvcmtlcklkfWAgfSxcbiAgICAgICAgICAgICAgcG9vbDogeyBtaW46IDEsIG1heDogMSB9LFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9KSgpXG4gICAgICAgIDogU29uYW11LmRiQ29uZmlnW2RiTmFtZV07XG4gICAgY29uc3QgZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoZGJDb25maWcpO1xuICAgIGNvbnN0IHJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSA9IFtdO1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrBcbiAgICAgIHRoaXMucmVsYXRpb25HcmFwaC5idWlsZEdyYXBoKGZpeHR1cmVzKTtcbiAgICAgIGNvbnN0IGluc2VydGlvbk9yZGVyID0gdGhpcy5yZWxhdGlvbkdyYXBoLmdldEluc2VydGlvbk9yZGVyKCk7XG5cbiAgICAgIC8vIDIuIOyKpO2Cte2VoCBmaXh0dXJlIOuovOyggCDsspjrpqwgKG92ZXJyaWRlIOyytO2BrClcbiAgICAgIGZvciAoY29uc3QgZml4dHVyZUlkIG9mIGluc2VydGlvbk9yZGVyKSB7XG4gICAgICAgIGNvbnN0IGZpeHR1cmUgPSBmaXh0dXJlcy5maW5kKChmKSA9PiBmLmZpeHR1cmVJZCA9PT0gZml4dHVyZUlkKTtcbiAgICAgICAgaWYgKCFmaXh0dXJlKSBjb250aW51ZTtcblxuICAgICAgICBjb25zdCBoYXNUYXJnZXQgPSAhIWZpeHR1cmUudGFyZ2V0O1xuICAgICAgICBjb25zdCBoYXNVbmlxdWUgPSAhIWZpeHR1cmUudW5pcXVlO1xuICAgICAgICBjb25zdCBoYXNEdXBsaWNhdGUgPSBoYXNUYXJnZXQgfHwgaGFzVW5pcXVlO1xuXG4gICAgICAgIC8vIOykkeuzteydtCDsnojqs6Agb3ZlcnJpZGU9ZmFsc2Xsnbgg6rK97JqwOiDsiqTtgrVcbiAgICAgICAgaWYgKGhhc0R1cGxpY2F0ZSAmJiAhZml4dHVyZS5vdmVycmlkZSkge1xuICAgICAgICAgIC8vIOq4sOyhtCDroIjsvZTrk5wgSUQg7KCA7J6lICh1bmlxdWUg7Jqw7ISgLCDsl4bsnLzrqbQgdGFyZ2V0KVxuICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBmaXh0dXJlLnVuaXF1ZT8uaWQgPz8gZml4dHVyZS50YXJnZXQ/LmlkO1xuICAgICAgICAgIGFzc2VydChleGlzdGluZ0lkKTtcbiAgICAgICAgICB0aGlzLnNraXBwZWRGaXh0dXJlcy5zZXQoZml4dHVyZUlkLCB7XG4gICAgICAgICAgICBlbnRpdHlJZDogZml4dHVyZS5lbnRpdHlJZCxcbiAgICAgICAgICAgIGV4aXN0aW5nSWQsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICBjaGFsay55ZWxsb3coXG4gICAgICAgICAgICAgICAgYFNraXBwZWQgJHtmaXh0dXJlLmVudGl0eUlkfSMke2ZpeHR1cmUuaWR9IChleGlzdGluZzogIyR7ZXhpc3RpbmdJZH0sIG92ZXJyaWRlOiBmYWxzZSlgLFxuICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyAzLiDthYzsnbTruJTrs4QgZml4dHVyZSDqt7jro7ntmZQgKGluc2VydGlvbk9yZGVyIOyInOyEnCDquLDrsJgpXG4gICAgICBjb25zdCBmaXh0dXJlc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgRml4dHVyZVJlY29yZFtdPigpO1xuICAgICAgY29uc3QgdGFibGVPcmRlcjogc3RyaW5nW10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlSWQgb2YgaW5zZXJ0aW9uT3JkZXIpIHtcbiAgICAgICAgLy8g7Iqk7YK165CcIGZpeHR1cmUg7KCc7Jm4XG4gICAgICAgIGlmICh0aGlzLnNraXBwZWRGaXh0dXJlcy5oYXMoZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgY29uc3QgZml4dHVyZSA9IGZpeHR1cmVzLmZpbmQoKGYpID0+IGYuZml4dHVyZUlkID09PSBmaXh0dXJlSWQpO1xuICAgICAgICBpZiAoIWZpeHR1cmUpIGNvbnRpbnVlO1xuXG4gICAgICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGZpeHR1cmUuZW50aXR5SWQpO1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGU7XG5cbiAgICAgICAgaWYgKCFmaXh0dXJlc0J5VGFibGUuaGFzKHRhYmxlTmFtZSkpIHtcbiAgICAgICAgICBmaXh0dXJlc0J5VGFibGUuc2V0KHRhYmxlTmFtZSwgW10pO1xuICAgICAgICAgIHRhYmxlT3JkZXIucHVzaCh0YWJsZU5hbWUpO1xuICAgICAgICB9XG4gICAgICAgIGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKT8ucHVzaChmaXh0dXJlKTtcbiAgICAgIH1cblxuICAgICAgYXdhaXQgZGIudHJhbnNhY3Rpb24oYXN5bmMgKHRyeCkgPT4ge1xuICAgICAgICBjb25zdCBpbnNlcnRlZElkc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4oKTtcblxuICAgICAgICAvLyA0LiDthYzsnbTruJTrs4Qg66CI67Ko67OEIOyymOumrFxuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgY29uc3QgdGFibGVGaXh0dXJlcyA9IGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBbXTtcbiAgICAgICAgICBjb25zdCBsZXZlbHMgPSB0aGlzLmdyb3VwRml4dHVyZXNCeUxldmVsKHRhYmxlRml4dHVyZXMpO1xuXG4gICAgICAgICAgZm9yIChjb25zdCBsZXZlbEZpeHR1cmVzIG9mIGxldmVscykge1xuICAgICAgICAgICAgLy8g7ZW064u5IOugiOuyqOydmCBmaXh0dXJl65OkIHJlZ2lzdGVyXG4gICAgICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgbGV2ZWxGaXh0dXJlcykge1xuICAgICAgICAgICAgICB0aGlzLnJlZ2lzdGVyRml4dHVyZShmaXh0dXJlLCBpbnNlcnRlZElkc0J5VGFibGUpO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICAgIGBSZWdpc3RlcmVkICR7Zml4dHVyZS5lbnRpdHlJZH0jJHtmaXh0dXJlLmlkfSR7Zml4dHVyZS5vdmVycmlkZSA/IGAgKG92ZXJyaWRlKWAgOiBcIlwifWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIHVwc2VydCDsi6Ttlokg7KCEIHV1aWQg66qp66GdIOyggOyepVxuICAgICAgICAgICAgY29uc3QgdGFibGUgPSB0aGlzLmJ1aWxkZXIuZ2V0VGFibGUodGFibGVOYW1lKTtcbiAgICAgICAgICAgIGNvbnN0IHV1aWRzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4gcm93LnV1aWQgYXMgc3RyaW5nKTtcblxuICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICBgVXBzZXJ0aW5nICR7dGFibGVOYW1lfSB3aXRoICR7dXVpZHMubGVuZ3RofSByb3dzIChsZXZlbCAke2xldmVscy5pbmRleE9mKGxldmVsRml4dHVyZXMpICsgMX0vJHtsZXZlbHMubGVuZ3RofSlgLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICBjb25zdCBpZHMgPSAoYXdhaXQgdGhpcy5idWlsZGVyLnVwc2VydChcbiAgICAgICAgICAgICAgdHJ4LFxuICAgICAgICAgICAgICB0YWJsZU5hbWUgYXMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gICAgICAgICAgICApKSBhcyAobnVtYmVyIHwgc3RyaW5nKVtdO1xuXG4gICAgICAgICAgICAvLyDsiJzshJwg6riw67CYIHV1aWQgLT4gaWQg66ek7ZWRXG4gICAgICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDsl4bsnLzrr4DroZwg65Ox66GdIOyInOyEnCA9IOuwmO2ZmCDsiJzshJwg67O07J6lXG4gICAgICAgICAgICBpZiAodXVpZHMubGVuZ3RoID4gMCAmJiB1dWlkcy5sZW5ndGggPT09IGlkcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPVxuICAgICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHV1aWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICAgICAgZXhpc3RpbmdNYXAuc2V0KHV1aWRzW2ldLCBpZHNbaV0pO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5zZXQodGFibGVOYW1lLCBleGlzdGluZ01hcCk7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKHV1aWRzLmxlbmd0aCAhPT0gaWRzLmxlbmd0aCkge1xuICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgY2hhbGsueWVsbG93KFxuICAgICAgICAgICAgICAgICAgYFdhcm5pbmc6IHV1aWQgY291bnQgKCR7dXVpZHMubGVuZ3RofSkgIT0gaWQgY291bnQgKCR7aWRzLmxlbmd0aH0pIGZvciAke3RhYmxlTmFtZX1gLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNS4gTWFueVRvTWFueSDqtIDqs4Qg7LKY66asXG4gICAgICAgIGF3YWl0IHRoaXMucHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnModHJ4LCBmaXh0dXJlcywgaW5zZXJ0ZWRJZHNCeVRhYmxlKTtcblxuICAgICAgICAvLyA2LiBQb3N0Z3JlU1FMIOyLnO2AgOyKpCDrpqzshYtcbiAgICAgICAgLy8gRml4dHVyZSDsgr3snoUg7ZuEIOqwgSDthYzsnbTruJTsnZggSUQg7Iuc7YCA7Iqk66W8IOy1nOuMgCBJRCDqsJLsnLzroZwg7JeF642w7J207Yq47ZWp64uI64ukLlxuICAgICAgICAvLyDsnbTroIfqsowg7ZWY7KeAIOyViuycvOuptCDri6TsnYwgSU5TRVJUIOyLnCBJROqwgCAyMDAw67KI64yA66GcIOyDneyEseuQoCDsiJgg7J6I7Iq164uI64ukLlxuICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJlc2V0dGluZyBzZXF1ZW5jZXMuLi5cIikpO1xuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIEVudGl0eeulvCDssL7slYTshJwgaWQg7YOA7J6FIO2ZleyduFxuICAgICAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXRBbGxFbnRpdGllcygpLmZpbmQoXG4gICAgICAgICAgICAgIChlKSA9PiBlLnRhYmxlID09PSB0YWJsZU5hbWUgfHwgZS5pZC50b0xvd2VyQ2FzZSgpID09PSB0YWJsZU5hbWUsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICBpZiAoZW50aXR5KSB7XG4gICAgICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik7XG4gICAgICAgICAgICAgIGNvbnN0IGlkVHlwZSA9IGlkUHJvcD8udHlwZTtcblxuICAgICAgICAgICAgICAvLyBpbnRlZ2VyL2JpZ0ludGVnZXLsnbTqsbDrgpgsIHN0cmluZ+ydtOyngOunjCBmaXh0dXJlU3RyYXRlZ3k9c2VxdWVuY2Xsnbgg6rK97Jqw7JeQ66eMIOumrOyFi+2VqeuLiOuLpFxuICAgICAgICAgICAgICBjb25zdCB1c2VzU2VxdWVuY2UgPVxuICAgICAgICAgICAgICAgIGlkVHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgICAgICBpZFR5cGUgPT09IFwiYmlnSW50ZWdlclwiIHx8XG4gICAgICAgICAgICAgICAgaWRQcm9wPy5jb25lPy5maXh0dXJlU3RyYXRlZ3kgPT09IFwic2VxdWVuY2VcIjtcblxuICAgICAgICAgICAgICBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgICAgICAgICFpc1Rlc3QoKSAmJlxuICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICAgIGNoYWxrLmdyYXkoXG4gICAgICAgICAgICAgICAgICAgICAgYFNraXBwZWQgc2VxdWVuY2UgcmVzZXQgZm9yICR7dGFibGVOYW1lfSAoaWQgdHlwZTogJHtpZFR5cGUgfHwgXCJ1bmtub3duXCJ9KWAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIO2FjOydtOu4lOydmCDstZzrjIAgSUQg7KGw7ZqMIChzdHJpbmcg7YOA7J6F7J2AIOyIq+yekCDsupDsiqTtjIUg7ZWE7JqUKVxuICAgICAgICAgICAgY29uc3QgZW50aXR5MiA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKS5maW5kKFxuICAgICAgICAgICAgICAoZSkgPT4gZS50YWJsZSA9PT0gdGFibGVOYW1lIHx8IGUuaWQudG9Mb3dlckNhc2UoKSA9PT0gdGFibGVOYW1lLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIGNvbnN0IGlkVHlwZTIgPSBlbnRpdHkyPy5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik/LnR5cGU7XG4gICAgICAgICAgICBjb25zdCBtYXhJZFJlc3VsdCA9XG4gICAgICAgICAgICAgIGlkVHlwZTIgPT09IFwic3RyaW5nXCJcbiAgICAgICAgICAgICAgICA/IGF3YWl0IHRyeFxuICAgICAgICAgICAgICAgICAgICAucmF3KGBTRUxFQ1QgTUFYKGlkOjpiaWdpbnQpIGFzIG1heF9pZCBGUk9NIFwiJHt0YWJsZU5hbWV9XCJgKVxuICAgICAgICAgICAgICAgICAgICAudGhlbigocikgPT4gci5yb3dzWzBdKVxuICAgICAgICAgICAgICAgIDogYXdhaXQgdHJ4KHRhYmxlTmFtZSkubWF4KFwiaWQgYXMgbWF4X2lkXCIpLmZpcnN0KCk7XG4gICAgICAgICAgICBjb25zdCBtYXhJZCA9IG1heElkUmVzdWx0Py5tYXhfaWQ7XG5cbiAgICAgICAgICAgIGlmIChtYXhJZCAhPT0gbnVsbCAmJiBtYXhJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIOyLnO2AgOyKpOuqheydhCBwZ19nZXRfc2VyaWFsX3NlcXVlbmNl66GcIOyViOyghO2VmOqyjCDsobDtmoxcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4LnJhdyhgU0VMRUNUIHNldHZhbChwZ19nZXRfc2VyaWFsX3NlcXVlbmNlKD8sICdpZCcpLCA/KWAsIFtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgbWF4SWQsXG4gICAgICAgICAgICAgIF0pO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuZ3JlZW4oYFJlc2V0IHNlcXVlbmNlIGZvciAke3RhYmxlTmFtZX06ICR7bWF4SWR9YCkpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKF9lcnIpIHtcbiAgICAgICAgICAgIC8vIOyLnO2AgOyKpOqwgCDsl4bripQg7YWM7J2067iUKGpvaW4gdGFibGUg65OxKeydgCDrrLTsi5xcbiAgICAgICAgICAgICFpc1Rlc3QoKSAmJiBjb25zb2xlLmxvZyhjaGFsay5ncmF5KGBTa2lwcGVkIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX1gKSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNy4g6rKw6rO8IOyImOynkVxuICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcblxuICAgICAgICAgIC8vIOyKpO2CteuQnCBmaXh0dXJl64qUIOq4sOyhtCDroIjsvZTrk5wg7KCV67O066GcIOqysOqzvCDstpTqsIBcbiAgICAgICAgICBjb25zdCBza2lwcGVkID0gdGhpcy5za2lwcGVkRml4dHVyZXMuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAoc2tpcHBlZCkge1xuICAgICAgICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgICAgICAgZW50aXR5SWQ6IGZpeHR1cmUuZW50aXR5SWQsXG4gICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgc2tpcHBlZC5leGlzdGluZ0lkKS5maXJzdCgpLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCByZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAocmVmKSB7XG4gICAgICAgICAgICBjb25zdCB1dWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQoZW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgIGNvbnN0IGluc2VydGVkSWQgPSB1dWlkVG9JZD8uZ2V0KHJlZi51dWlkKTtcblxuICAgICAgICAgICAgaWYgKGluc2VydGVkSWQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICByZXN1bHRzLnB1c2goe1xuICAgICAgICAgICAgICAgIGVudGl0eUlkOiBmaXh0dXJlLmVudGl0eUlkLFxuICAgICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgaW5zZXJ0ZWRJZCkuZmlyc3QoKSxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihgSW5zZXJ0ZWQgaW50byAke2VudGl0eS50YWJsZX06ICMke2ZpeHR1cmUuaWR9IC0+ICMke2luc2VydGVkSWR9YCksXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCBkYi5kZXN0cm95KCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuaXF1ZShyZXN1bHRzLCAocikgPT4gYCR7ci5lbnRpdHlJZH0jJHtyLmRhdGEuaWR9YCk7XG4gIH1cblxuICAvKipcbiAgICogRml4dHVyZVJlY29yZOulvCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnVxuICAgKiBAcGFyYW0gaW5zZXJ0ZWRJZHNCeVRhYmxlIOydtOuvuCB1cHNlcnTrkJwg7YWM7J2067iU7J2YIHV1aWTihpJpZCDrp6TtlZEgKOugiOuyqOuzhCDsspjrpqwg7IucIOyCrOyaqSlcbiAgICovXG4gIHByaXZhdGUgcmVnaXN0ZXJGaXh0dXJlKFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlPzogTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4sXG4gICk6IFVCUmVmIHtcbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICBjb25zdCByb3c6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICAvLyBPdmVycmlkZSDrqqjrk5wg7YyQ64uoOiB0YXJnZXQg65iQ64qUIHVuaXF1ZeqwgCDsnojqs6Agb3ZlcnJpZGU9dHJ1ZeyduCDqsr3smrBcbiAgICBjb25zdCBleGlzdGluZ1JlY29yZCA9IGZpeHR1cmUudGFyZ2V0ID8/IGZpeHR1cmUudW5pcXVlO1xuICAgIGNvbnN0IGlzT3ZlcnJpZGVNb2RlID0gZml4dHVyZS5vdmVycmlkZSAmJiBleGlzdGluZ1JlY29yZDtcblxuICAgIGZvciAoY29uc3QgW3Byb3BOYW1lLCBjb2x1bW5dIG9mIE9iamVjdC5lbnRyaWVzKGZpeHR1cmUuY29sdW1ucykpIHtcbiAgICAgIGNvbnN0IHByb3AgPSBjb2x1bW4ucHJvcDtcblxuICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIEdlbmVyYXRlZCBjb2x1bW7snYAgSU5TRVJU7JeQ7IScIOygnOyZuCAoRELqsIAg7J6Q64+ZIOyDneyEsSlcbiAgICAgIGlmIChcImdlbmVyYXRlZFwiIGluIHByb3AgJiYgcHJvcC5nZW5lcmF0ZWQpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIGlkIOyymOumrFxuICAgICAgaWYgKHByb3BOYW1lID09PSBcImlkXCIpIHtcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgLy8gcGFyZW50SWQg7JeU7Yuw7Yuw7J2YIGlk64qUIOu2gOuqqCDsl5Tti7Dti7DsnZggRkvsnbTrr4DroZwg7Iuc7YCA7IqkIOuvuOyCrOyaqSAo66qF7Iuc7KCB7Jy866GcIO2PrO2VqClcbiAgICAgICAgY29uc3QgdXNlc1NlcXVlbmNlID1cbiAgICAgICAgICAhZW50aXR5LnBhcmVudElkICYmXG4gICAgICAgICAgKGlkUHJvcD8udHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8udHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8uY29uZT8uZml4dHVyZVN0cmF0ZWd5ID09PSBcInNlcXVlbmNlXCIpO1xuXG4gICAgICAgIGlmIChpc092ZXJyaWRlTW9kZSAmJiBleGlzdGluZ1JlY29yZCkge1xuICAgICAgICAgIC8vIE92ZXJyaWRlOiDquLDsobQg66CI7L2U65Oc7J2YIOqwkiDsgqzsmqkg4oaSIFVQREFURVxuICAgICAgICAgIHJvd1twcm9wTmFtZV0gPSBleGlzdGluZ1JlY29yZC5jb2x1bW5zW3Byb3BOYW1lXT8udmFsdWU7XG4gICAgICAgIH0gZWxzZSBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgIC8vIHN0cmluZyBQSyDrmJDripQgcGFyZW50SWQg7JeU7Yuw7YuwIEZLOiDsg53shLHrkJwgaWQg6rCS7J2EIElOU0VSVOyXkCDtj6ztlahcbiAgICAgICAgICByb3dbcHJvcE5hbWVdID0gY29sdW1uLnZhbHVlO1xuICAgICAgICB9XG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlciBQSzogREIg7Iuc7YCA7Iqk7JeQIOunoeq5gCAo6rCSIOygnOyZuClcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHxcbiAgICAgICAgICAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiBwcm9wLmhhc0pvaW5Db2x1bW4pXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IGNvbHVtbi52YWx1ZSBhcyBudW1iZXIgfCBudWxsO1xuICAgICAgICAgIGlmIChyZWxhdGVkSWQgIT09IG51bGwgJiYgcmVsYXRlZElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG5cbiAgICAgICAgICAgIC8vIOuovOyggCBza2lw65CcIGZpeHR1cmXsnbjsp4Ag7ZmV7J24XG4gICAgICAgICAgICBjb25zdCBza2lwcGVkRXhpc3RpbmdJZCA9IHRoaXMuc2tpcHBlZEZpeHR1cmVzLmdldChyZWxhdGVkRml4dHVyZUlkKT8uZXhpc3RpbmdJZDtcbiAgICAgICAgICAgIGlmIChza2lwcGVkRXhpc3RpbmdJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIHNraXDrkJwgZml4dHVyZSDihpIgdGFyZ2V0IERC7J2YIOq4sOyhtCDroIjsvZTrk5wgaWQg7IKs7JqpXG4gICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHNraXBwZWRFeGlzdGluZ0lkO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQocmVsYXRlZEZpeHR1cmVJZCk7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgICAgLy8g7J2066+4IHVwc2VydOuQnCDqsJnsnYAg7YWM7J2067iUIGZpeHR1cmUg7ZmV7J24XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEluc2VydGVkSWRzID0gaW5zZXJ0ZWRJZHNCeVRhYmxlPy5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgICAgY29uc3QgYWN0dWFsSWQgPSByZWxhdGVkSW5zZXJ0ZWRJZHM/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGFjdHVhbElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIC8vIOydtOuvuCB1cHNlcnTrkKgg4oaSIOyLpOygnCBJRCDsgqzsmqlcbiAgICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IGFjdHVhbElkO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAvLyDslYTsp4EgdXBzZXJ0IOyViOuQqCDihpIgVUJSZWYg7IKs7JqpXG4gICAgICAgICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSByZWxhdGVkUmVmO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAvLyBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOugiOy9lOuTnCDihpIgSUQg6re464yA66GcIOyCrOyaqVxuICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAvLyBIYXNNYW55LCBNYW55VG9NYW5564qUIOuzhOuPhCDsspjrpqxcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIOydvOuwmCDsu6zrn7xcbiAgICAgICAgcm93W3Byb3BOYW1lXSA9IHRoaXMuY29udmVydENvbHVtblZhbHVlKHByb3AgYXMgRW50aXR5UHJvcCwgY29sdW1uLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAhaXNUZXN0KCkgJiZcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoYFJlZ2lzdGVyaW5nICR7ZW50aXR5LnRhYmxlfSAtICR7aW5zcGVjdChyb3csIGZhbHNlLCBudWxsLCB0cnVlKX1gKSk7XG4gICAgY29uc3QgcmVmID0gdGhpcy5idWlsZGVyLnJlZ2lzdGVyKGVudGl0eS50YWJsZSwgcm93KTtcbiAgICB0aGlzLmZpeHR1cmVSZWZNYXAuc2V0KGZpeHR1cmUuZml4dHVyZUlkLCByZWYpO1xuXG4gICAgcmV0dXJuIHJlZjtcbiAgfVxuXG4gIC8qKlxuICAgKiDsu6zrn7wg6rCSIOuzgO2ZmFxuICAgKi9cbiAgcHJpdmF0ZSBjb252ZXJ0Q29sdW1uVmFsdWUocHJvcDogRW50aXR5UHJvcCwgdmFsdWU6IHVua25vd24pOiB1bmtub3duIHtcbiAgICBpZiAodmFsdWUgPT09IG51bGwgfHwgdmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgc3dpdGNoIChwcm9wLnR5cGUpIHtcbiAgICAgIGNhc2UgXCJqc29uXCI6XG4gICAgICAgIC8vIFVwc2VydEJ1aWxkZXIucmVnaXN0ZXLsl5DshJwgSlNPTi5zdHJpbmdpZnkg7LKY66as7ZWY66+A66GcIG9iamVjdCDqt7jrjIDroZwg7KCE64usXG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgY2FzZSBcImRhdGVcIjpcbiAgICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlb2YgdmFsdWUgPT09IFwibnVtYmVyXCIpIHtcbiAgICAgICAgICByZXR1cm4gbmV3IERhdGUodmFsdWUpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgcHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnMoXG4gICAgdHJ4OiBLbmV4LlRyYW5zYWN0aW9uLFxuICAgIGZpeHR1cmVzOiBGaXh0dXJlUmVjb3JkW10sXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlOiBNYXA8c3RyaW5nLCBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+PixcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGZpeHR1cmVzKSB7XG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICAgIGNvbnN0IHNvdXJjZVJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQoZml4dHVyZS5maXh0dXJlSWQpO1xuXG4gICAgICBpZiAoIXNvdXJjZVJlZikgY29udGludWU7XG5cbiAgICAgIGNvbnN0IHNvdXJjZVV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChlbnRpdHkudGFibGUpO1xuICAgICAgY29uc3Qgc291cmNlSWQgPSBzb3VyY2VVdWlkVG9JZD8uZ2V0KHNvdXJjZVJlZi51dWlkKTtcblxuICAgICAgaWYgKHNvdXJjZUlkID09PSB1bmRlZmluZWQpIGNvbnRpbnVlO1xuXG4gICAgICBmb3IgKGNvbnN0IFssIGNvbHVtbl0gb2YgT2JqZWN0LmVudHJpZXMoZml4dHVyZS5jb2x1bW5zKSkge1xuICAgICAgICBjb25zdCBwcm9wID0gY29sdW1uLnByb3A7XG5cbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSAmJiBBcnJheS5pc0FycmF5KGNvbHVtbi52YWx1ZSkpIHtcbiAgICAgICAgICAvLyDshKDtg53rkJjsp4Ag7JWK7J2AIE1hbnlUb01hbnkg6rSA6rOE64qUIOyggOyepe2VmOyngCDslYrsnYxcbiAgICAgICAgICBjb25zdCB0YXJnZXRUYWJsZSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgaWYgKHRoaXMuYnVpbGRlci5oYXNUYWJsZSh0YXJnZXRUYWJsZS50YWJsZSkgPT09IGZhbHNlKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBjb2x1bW4udmFsdWUgYXMgbnVtYmVyW107XG4gICAgICAgICAgaWYgKHJlbGF0ZWRJZHMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IGpvaW5UYWJsZSA9IChwcm9wIGFzIE1hbnlUb01hbnlSZWxhdGlvblByb3ApLmpvaW5UYWJsZTtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcblxuICAgICAgICAgIGNvbnN0IHNvdXJjZUNvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUoZW50aXR5LnRhYmxlKX1faWRgO1xuICAgICAgICAgIGNvbnN0IHRhcmdldENvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGZvciAoY29uc3QgcmVsYXRlZElkIG9mIHJlbGF0ZWRJZHMpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG4gICAgICAgICAgICBjb25zdCByZWxhdGVkUmVmID0gdGhpcy5maXh0dXJlUmVmTWFwLmdldChyZWxhdGVkRml4dHVyZUlkKTtcblxuICAgICAgICAgICAgbGV0IHRhcmdldElkOiBudW1iZXIgfCBzdHJpbmc7XG5cbiAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRVdWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgIGNvbnN0IHJlc29sdmVkSWQgPSByZWxhdGVkVXVpZFRvSWQ/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgIGlmIChyZXNvbHZlZElkID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgICBgUmVsYXRlZCBmaXh0dXJlICR7cmVsYXRlZEZpeHR1cmVJZH0gbm90IGZvdW5kIGluIGluc2VydGVkSWRzLCBza2lwcGluZ2AsXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB0YXJnZXRJZCA9IHJlc29sdmVkSWQ7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICB0YXJnZXRJZCA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gSm9pblRhYmxl7JeQIOyCveyehVxuICAgICAgICAgICAgY29uc3QgW2ZvdW5kXSA9IGF3YWl0IHRyeChqb2luVGFibGUpXG4gICAgICAgICAgICAgIC53aGVyZSh7XG4gICAgICAgICAgICAgICAgW3NvdXJjZUNvbHVtbl06IHNvdXJjZUlkLFxuICAgICAgICAgICAgICAgIFt0YXJnZXRDb2x1bW5dOiB0YXJnZXRJZCxcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgLmxpbWl0KDEpO1xuXG4gICAgICAgICAgICBpZiAoIWZvdW5kKSB7XG4gICAgICAgICAgICAgIGF3YWl0IHRyeChqb2luVGFibGUpLmluc2VydCh7XG4gICAgICAgICAgICAgICAgW3NvdXJjZUNvbHVtbl06IHNvdXJjZUlkLFxuICAgICAgICAgICAgICAgIFt0YXJnZXRDb2x1bW5dOiB0YXJnZXRJZCxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihcbiAgICAgICAgICAgICAgICAgICAgYEluc2VydGVkIGludG8gJHtqb2luVGFibGV9OiAke2VudGl0eS50YWJsZX0oJHtzb3VyY2VJZH0pIC0gJHtyZWxhdGVkRW50aXR5LnRhYmxlfSgke3RhcmdldElkfSlgLFxuICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiDqsJnsnYAg7YWM7J2067iUIOuCtCBmaXh0dXJl65Ok7J2EIHNlbGYtcmVmZXJlbmNlIOugiOuyqOuzhOuhnCDrtoTtlaBcbiAgICogLSBzZWxmLXJlZmVyZW5jZeqwgCDsl4bripQgZml4dHVyZeuTpDogTGV2ZWwgMFxuICAgKiAtIExldmVsIDDsnYQg7LC47KGw7ZWY64qUIGZpeHR1cmXrk6Q6IExldmVsIDFcbiAgICogLSDrsJjrs7VcbiAgICpcbiAgICogVXBzZXJ0QnVpbGRlcuqwgCBzZWxmLXJlZmVyZW5jZeqwgCDsnojsnLzrqbQgYnVpbGRJbnNlcnRMZXZlbHMoKeuhnCDsnqzsoJXroKztlZjsl6xcbiAgICog65Ox66GdIOyInOyEnOyZgCDrsJjtmZgg7Iic7ISc6rCAIOuLrOudvOyniCDsiJgg7J6I7Iq164uI64ukLlxuICAgKiDsnbTrpbwg67Cp7KeA7ZWY6riwIOychO2VtCBGaXh0dXJlTWFuYWdlcuqwgCDroIjrsqjrs4TroZwg64KY64ig7IScIOyymOumrO2VqeuLiOuLpC5cbiAgICovXG4gIHByaXZhdGUgZ3JvdXBGaXh0dXJlc0J5TGV2ZWwoZml4dHVyZXM6IEZpeHR1cmVSZWNvcmRbXSk6IEZpeHR1cmVSZWNvcmRbXVtdIHtcbiAgICBpZiAoZml4dHVyZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZXNbMF0uZW50aXR5SWQpO1xuXG4gICAgLy8gc2VsZi1yZWZlcmVuY2UgcmVsYXRpb24gcHJvcCDssL7quLBcbiAgICBjb25zdCBzZWxmUmVmUHJvcHMgPSBlbnRpdHkucHJvcHMuZmlsdGVyKFxuICAgICAgKHApOiBwIGlzIEJlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCB8IE9uZVRvT25lUmVsYXRpb25Qcm9wID0+XG4gICAgICAgIGlzUmVsYXRpb25Qcm9wKHApICYmXG4gICAgICAgIChpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcChwKSB8fCAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwKSAmJiBwLmhhc0pvaW5Db2x1bW4pKSAmJlxuICAgICAgICBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICApO1xuXG4gICAgaWYgKHNlbGZSZWZQcm9wcy5sZW5ndGggPT09IDApIHtcbiAgICAgIC8vIHNlbGYtcmVmZXJlbmNlIOyXhuydjCDihpIg64uo7J28IOugiOuyqFxuICAgICAgcmV0dXJuIFtmaXh0dXJlc107XG4gICAgfVxuXG4gICAgLy8g66CI67Ko67OEIOu2hO2VoCAodG9wb2xvZ2ljYWwgc29ydClcbiAgICBjb25zdCBsZXZlbHM6IEZpeHR1cmVSZWNvcmRbXVtdID0gW107XG4gICAgY29uc3QgcmVtYWluaW5nID0gbmV3IFNldChmaXh0dXJlcy5tYXAoKGYpID0+IGYuZml4dHVyZUlkKSk7XG4gICAgY29uc3QgcHJvY2Vzc2VkID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICB3aGlsZSAocmVtYWluaW5nLnNpemUgPiAwKSB7XG4gICAgICBjb25zdCBjdXJyZW50TGV2ZWw6IEZpeHR1cmVSZWNvcmRbXSA9IFtdO1xuXG4gICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgaWYgKCFyZW1haW5pbmcuaGFzKGZpeHR1cmUuZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgLy8gc2VsZi1yZWZlcmVuY2XqsIAg66qo65GQIOydtOuvuCDsspjrpqzrkJDqsbDrgpggbnVsbOyduCDqsr3smrBcbiAgICAgICAgY29uc3QgY2FuUHJvY2VzcyA9IHNlbGZSZWZQcm9wcy5ldmVyeSgocHJvcCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHJlZklkID0gZml4dHVyZS5jb2x1bW5zW3Byb3AubmFtZV0/LnZhbHVlIGFzIG51bWJlciB8IG51bGw7XG4gICAgICAgICAgaWYgKHJlZklkID09PSBudWxsIHx8IHJlZklkID09PSB1bmRlZmluZWQpIHJldHVybiB0cnVlO1xuICAgICAgICAgIGNvbnN0IHJlZkZpeHR1cmVJZCA9IGAke3Byb3Aud2l0aH0jJHtyZWZJZH1gO1xuICAgICAgICAgIC8vIOydtOuvuCDsspjrpqzrkJDqsbDrgpgsIO2YhOyerCBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOqyveyasCAo7Jm467aAIOywuOyhsClcbiAgICAgICAgICByZXR1cm4gcHJvY2Vzc2VkLmhhcyhyZWZGaXh0dXJlSWQpIHx8ICFyZW1haW5pbmcuaGFzKHJlZkZpeHR1cmVJZCk7XG4gICAgICAgIH0pO1xuXG4gICAgICAgIGlmIChjYW5Qcm9jZXNzKSB7XG4gICAgICAgICAgY3VycmVudExldmVsLnB1c2goZml4dHVyZSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKGN1cnJlbnRMZXZlbC5sZW5ndGggPT09IDApIHtcbiAgICAgICAgY29uc3QgcmVtYWluaW5nSWRzID0gQXJyYXkuZnJvbShyZW1haW5pbmcpLmpvaW4oXCIsIFwiKTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgIGBDaXJjdWxhciBzZWxmLXJlZmVyZW5jZSBkZXRlY3RlZCBpbiAke2VudGl0eS50YWJsZX0uIFJlbWFpbmluZyBmaXh0dXJlczogJHtyZW1haW5pbmdJZHN9YCxcbiAgICAgICAgKTtcbiAgICAgIH1cblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGN1cnJlbnRMZXZlbCkge1xuICAgICAgICByZW1haW5pbmcuZGVsZXRlKGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgcHJvY2Vzc2VkLmFkZChmaXh0dXJlLmZpeHR1cmVJZCk7XG4gICAgICB9XG5cbiAgICAgIGxldmVscy5wdXNoKGN1cnJlbnRMZXZlbCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGxldmVscztcbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgY2hlY2tVbmlxdWVWaW9sYXRpb24oZGI6IEtuZXgsIGVudGl0eTogRW50aXR5LCBmaXh0dXJlOiBGaXh0dXJlUmVjb3JkKSB7XG4gICAgY29uc3QgX3VuaXF1ZUluZGV4ZXMgPSBlbnRpdHkuaW5kZXhlcz8uZmlsdGVyKChpKSA9PiBpLnR5cGUgPT09IFwidW5pcXVlXCIpID8/IFtdO1xuXG4gICAgY29uc3QgdW5pcXVlSW5kZXhlcyA9IF91bmlxdWVJbmRleGVzLmZpbHRlcigoaW5kZXgpID0+XG4gICAgICBpbmRleC5jb2x1bW5zLmV2ZXJ5KChjb2x1bW4pID0+ICFjb2x1bW4ubmFtZS5zdGFydHNXaXRoKGAke2VudGl0eS50YWJsZX1fX2ApKSxcbiAgICApO1xuICAgIGlmICh1bmlxdWVJbmRleGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgbGV0IHVuaXF1ZVF1ZXJ5ID0gZGIoZW50aXR5LnRhYmxlKTtcbiAgICBsZXQgaGFzQ29uZGl0aW9uID0gZmFsc2U7XG5cbiAgICBmb3IgKGNvbnN0IGluZGV4IG9mIHVuaXF1ZUluZGV4ZXMpIHtcbiAgICAgIC8vIOy7rOufvCDspJEg7ZWY64KY652864+EIG51bGzsnbTrqbQg7Jyg64uI7YGsIOygnOyVveydhCDsnITrsJjtlZjsp4Ag7JWK6riwIOuVjOusuOyXkCDtlbTri7kg7J24642x7Iqk64qUIOustOyLnFxuICAgICAgY29uc3QgY29udGFpbnNOdWxsID0gaW5kZXguY29sdW1ucy5zb21lKChjb2x1bW4pID0+IHtcbiAgICAgICAgY29uc3QgZmllbGQgPSBjb2x1bW4ubmFtZS5yZXBsYWNlKC9faWQkLywgXCJcIik7XG4gICAgICAgIHJldHVybiBmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSA9PT0gbnVsbDtcbiAgICAgIH0pO1xuICAgICAgaWYgKGNvbnRhaW5zTnVsbCkge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgdW5pcXVlUXVlcnkgPSB1bmlxdWVRdWVyeS5vcldoZXJlKChxYikgPT4ge1xuICAgICAgICBmb3IgKGNvbnN0IGNvbHVtbiBvZiBpbmRleC5jb2x1bW5zKSB7XG4gICAgICAgICAgY29uc3QgZmllbGQgPSBjb2x1bW4ubmFtZS5yZXBsYWNlKC9faWQkLywgXCJcIik7XG5cbiAgICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSkpIHtcbiAgICAgICAgICAgIHFiLndoZXJlSW4oY29sdW1uLm5hbWUsIGZpeHR1cmUuY29sdW1uc1tmaWVsZF0udmFsdWUpO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBxYi5hbmRXaGVyZShjb2x1bW4ubmFtZSwgZml4dHVyZS5jb2x1bW5zW2ZpZWxkXT8udmFsdWUpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgICBoYXNDb25kaXRpb24gPSB0cnVlO1xuICAgIH1cblxuICAgIGlmICghaGFzQ29uZGl0aW9uKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBbdW5pcXVlRm91bmRdID0gYXdhaXQgdW5pcXVlUXVlcnk7XG4gICAgcmV0dXJuIHVuaXF1ZUZvdW5kO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBjaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICBkYjogS25leCxcbiAgICBlbnRpdHk6IEVudGl0eSxcbiAgICBmaXh0dXJlOiBGaXh0dXJlUmVjb3JkLFxuICAgIGNvbHVtbnM6IHN0cmluZ1tdLFxuICApIHtcbiAgICBpZiAoY29sdW1ucy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGNvbnN0IHdoZXJlQ2xhdXNlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuXG4gICAgZm9yIChjb25zdCBjb2x1bW4gb2YgY29sdW1ucykge1xuICAgICAgLy8gcmVsYXRpb24g7ZWE65Oc7J24IOqyveyasCBfaWQg67aZ7J206riwXG4gICAgICBjb25zdCBwcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gY29sdW1uKTtcbiAgICAgIGNvbnN0IGRiQ29sdW1uID0gcHJvcCAmJiBpc1JlbGF0aW9uUHJvcChwcm9wKSA/IGAke2NvbHVtbn1faWRgIDogY29sdW1uO1xuICAgICAgY29uc3QgdmFsdWUgPSBmaXh0dXJlLmNvbHVtbnNbY29sdW1uXT8udmFsdWU7XG5cbiAgICAgIC8vIG51bGwg6rCS7J20IO2PrO2VqOuQnCDqsr3smrAg7KSR67O1IO2ZleyduCDsiqTtgrVcbiAgICAgIGlmICh2YWx1ZSA9PT0gbnVsbCB8fCB2YWx1ZSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHJldHVybiBudWxsO1xuICAgICAgfVxuXG4gICAgICB3aGVyZUNsYXVzZVtkYkNvbHVtbl0gPSB2YWx1ZTtcbiAgICB9XG5cbiAgICBjb25zdCBbZm91bmRdID0gYXdhaXQgZGIoZW50aXR5LnRhYmxlKS53aGVyZSh3aGVyZUNsYXVzZSkubGltaXQoMSk7XG4gICAgcmV0dXJuIGZvdW5kO1xuICB9XG5cbiAgYXN5bmMgYWRkRml4dHVyZUxvYWRlcihjb2RlOiBzdHJpbmcpIHtcbiAgICBjb25zdCBwYXRoID0gYCR7U29uYW11LmFwaVJvb3RQYXRofS9zcmMvdGVzdGluZy9maXh0dXJlLnRzYDtcbiAgICBjb25zdCBjb250ZW50ID0gcmVhZEZpbGVTeW5jKHBhdGgpLnRvU3RyaW5nKCk7XG5cbiAgICBjb25zdCBmaXh0dXJlTG9hZGVyU3RhcnQgPSBjb250ZW50LmluZGV4T2YoXCJjb25zdCBmaXh0dXJlTG9hZGVyID0ge1wiKTtcbiAgICBjb25zdCBmaXh0dXJlTG9hZGVyRW5kID0gY29udGVudC5pbmRleE9mKFwifTtcIiwgZml4dHVyZUxvYWRlclN0YXJ0KTtcblxuICAgIGlmIChmaXh0dXJlTG9hZGVyU3RhcnQgIT09IC0xICYmIGZpeHR1cmVMb2FkZXJFbmQgIT09IC0xKSB7XG4gICAgICBjb25zdCBuZXdDb250ZW50ID0gYCR7Y29udGVudC5zbGljZSgwLCBmaXh0dXJlTG9hZGVyRW5kKX0gICR7Y29kZX1cXG4ke2NvbnRlbnQuc2xpY2UoZml4dHVyZUxvYWRlckVuZCl9YDtcblxuICAgICAgd3JpdGVGaWxlU3luYyhwYXRoLCBuZXdDb250ZW50KTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFwiRmFpbGVkIHRvIGZpbmQgZml4dHVyZUxvYWRlciBpbiBmaXh0dXJlLnRzXCIpO1xuICAgIH1cbiAgfVxufVxuXG5leHBvcnQgY29uc3QgRml4dHVyZU1hbmFnZXIgPSBuZXcgRml4dHVyZU1hbmFnZXJDbGFzcygpO1xuIl0sIm5hbWVzIjpbImFzc2VydCIsImNoYWxrIiwiZXhlY1N5bmMiLCJyZWFkRmlsZVN5bmMiLCJ3cml0ZUZpbGVTeW5jIiwiaW5mbGVjdGlvbiIsInVuaXF1ZSIsImluc3BlY3QiLCJTb25hbXUiLCJCYXNlTW9kZWwiLCJjcmVhdGVLbmV4SW5zdGFuY2UiLCJVcHNlcnRCdWlsZGVyIiwiRW50aXR5TWFuYWdlciIsImlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIiwiaXNIYXNNYW55UmVsYXRpb25Qcm9wIiwiaXNNYW55VG9NYW55UmVsYXRpb25Qcm9wIiwiaXNPbmVUb09uZVJlbGF0aW9uUHJvcCIsImlzUmVsYXRpb25Qcm9wIiwiaXNWaXJ0dWFsUHJvcCIsImlzVGVzdCIsIlJlbGF0aW9uR3JhcGgiLCJGaXh0dXJlTWFuYWdlckNsYXNzIiwiX3RkYiIsInRkYiIsIkVycm9yIiwiX2ZkYiIsImZkYiIsImNhY2hlZFRhYmxlTmFtZXMiLCJyZWxhdGlvbkdyYXBoIiwiYnVpbGRlciIsImZpeHR1cmVSZWZNYXAiLCJNYXAiLCJza2lwcGVkRml4dHVyZXMiLCJpbml0IiwiZGJDb25maWciLCJ0ZXN0IiwicHJvZHVjdGlvbl9tYXN0ZXIiLCJ0Q29ubiIsImNvbm5lY3Rpb24iLCJwQ29ubiIsImhvc3QiLCJwb3J0IiwiZGF0YWJhc2UiLCJmaXh0dXJlIiwic3luYyIsImZpeHR1cmVDb25uIiwidGVzdENvbm4iLCJ0ZXN0UGdFbnYiLCJQR1BBU1NXT1JEIiwicGFzc3dvcmQiLCJ1c2VyIiwic3RkaW8iLCJlbnYiLCJwcm9jZXNzIiwiZml4dHVyZVBnRW52IiwiZHVtcENtZCIsInJlc3RvcmVDbWQiLCJzaGVsbCIsInJlc2V0U2VxdWVuY2VzIiwidGVzdERiIiwiZW50aXRpZXMiLCJnZXRBbGxFbnRpdGllcyIsImVudGl0eSIsInRhYmxlTmFtZSIsInRhYmxlIiwiaWQiLCJ0b0xvd2VyQ2FzZSIsImlkUHJvcCIsInByb3BzIiwiZmluZCIsInAiLCJuYW1lIiwiaWRUeXBlIiwidHlwZSIsInVzZXNTZXF1ZW5jZSIsImNvbmUiLCJmaXh0dXJlU3RyYXRlZ3kiLCJjb25zb2xlIiwibG9nIiwibWF4RXhwciIsInJhdyIsImRlc3Ryb3kiLCJ2aXNpdGVkUmVjb3JkcyIsIlNldCIsImltcG9ydEZpeHR1cmUiLCJlbnRpdHlJZCIsImlkcyIsImNsZWFyIiwicXVlcmllcyIsIlByb21pc2UiLCJhbGwiLCJtYXAiLCJnZXRJbXBvcnRRdWVyaWVzIiwiZmxhdCIsIndkYiIsImdldERCIiwicXVlcnkiLCJmaWVsZCIsInJlY29yZEtleSIsImhhcyIsImFkZCIsImdldCIsInJvdyIsIndoZXJlIiwibGltaXQiLCJ1bmRlZmluZWQiLCJmaXh0dXJlRGF0YWJhc2UiLCJyZWFsRGF0YWJhc2UiLCJzZWxmUXVlcnkiLCJhcmdzIiwiT2JqZWN0IiwiZW50cmllcyIsInJlbGF0aW9ucyIsImZpbHRlciIsInJlbGF0aW9uIiwiY3VzdG9tSm9pbkNsYXVzZSIsImhhc0pvaW5Db2x1bW4iLCJyZWxhdGVkRW50aXR5Iiwid2l0aCIsInJlbGF0ZWRJZENvbHVtbk5hbWUiLCJhcmciLCJyZWxRdWVyaWVzIiwicmV2ZXJzZSIsImdldEZpeHR1cmVzIiwic291cmNlREJOYW1lIiwidGFyZ2V0REJOYW1lIiwic2VhcmNoT3B0aW9ucyIsImR1cGxpY2F0ZUNoZWNrIiwic291cmNlREIiLCJ0YXJnZXREQiIsInZhbHVlIiwic2VhcmNoVHlwZSIsImNvbHVtbiIsInByb3AiLCJyb3dzIiwibGVuZ3RoIiwiZml4dHVyZXMiLCJpbml0aWFsUmVjb3Jkc0xlbmd0aCIsIm5ld1JlY29yZHMiLCJjcmVhdGVGaXh0dXJlUmVjb3JkIiwiX2RiIiwicHVzaCIsImN1cnJlbnRGaXh0dXJlUmVjb3JkIiwiciIsImZpeHR1cmVJZCIsImZldGNoZWRSZWNvcmRzIiwic2xpY2UiLCJjdXN0b21Db2x1bW5zIiwiY29sdW1ucyIsImN1c3RvbUR1cGxpY2F0ZVJvdyIsImNoZWNrRHVwbGljYXRlQnlDb2x1bW5zIiwicmVjb3JkIiwic2luZ2xlUmVjb3JkIiwidGFyZ2V0IiwidW5pcXVlUm93IiwiY2hlY2tVbmlxdWVWaW9sYXRpb24iLCJmIiwiYWxsU2V0dGxlZCIsIm9wdGlvbnMiLCJyZWNvcmRzIiwidmlzaXRlZEVudGl0aWVzIiwiY3JlYXRlIiwiYmVsb25nc1JlY29yZHMiLCJkYiIsInRocm91Z2hUYWJsZSIsImpvaW5UYWJsZSIsImZyb21Db2x1bW4iLCJzaW5ndWxhcml6ZSIsInRvQ29sdW1uIiwicmVsYXRlZElkcyIsInBsdWNrIiwiam9pbkNvbHVtbiIsInJlbGF0ZWRQcm9wIiwiZmtDb2x1bW4iLCJyZWxhdGVkUm93IiwiZmlyc3QiLCJyZWxhdGVkSWQiLCJpbnNlcnRGaXh0dXJlcyIsImRiTmFtZSIsIl9maXh0dXJlcyIsIlNPTkFNVV9XT1JLRVJfREIiLCJWSVRFU1RfUE9PTF9JRCIsIndvcmtlcklkIiwicGFyc2VJbnQiLCJiYXNlQ29uZmlnIiwicG9vbCIsIm1pbiIsIm1heCIsInJlc3VsdHMiLCJidWlsZEdyYXBoIiwiaW5zZXJ0aW9uT3JkZXIiLCJnZXRJbnNlcnRpb25PcmRlciIsImhhc1RhcmdldCIsImhhc1VuaXF1ZSIsImhhc0R1cGxpY2F0ZSIsIm92ZXJyaWRlIiwiZXhpc3RpbmdJZCIsInNldCIsInllbGxvdyIsImZpeHR1cmVzQnlUYWJsZSIsInRhYmxlT3JkZXIiLCJ0cmFuc2FjdGlvbiIsInRyeCIsImluc2VydGVkSWRzQnlUYWJsZSIsInRhYmxlRml4dHVyZXMiLCJsZXZlbHMiLCJncm91cEZpeHR1cmVzQnlMZXZlbCIsImxldmVsRml4dHVyZXMiLCJyZWdpc3RlckZpeHR1cmUiLCJibHVlIiwiZ2V0VGFibGUiLCJ1dWlkcyIsInV1aWQiLCJpbmRleE9mIiwidXBzZXJ0IiwiZXhpc3RpbmdNYXAiLCJpIiwid2FybiIsInByb2Nlc3NNYW55VG9NYW55UmVsYXRpb25zIiwiZSIsImdyYXkiLCJlbnRpdHkyIiwiaWRUeXBlMiIsIm1heElkUmVzdWx0IiwidGhlbiIsIm1heElkIiwibWF4X2lkIiwiZ3JlZW4iLCJfZXJyIiwic2tpcHBlZCIsImRhdGEiLCJyZWYiLCJ1dWlkVG9JZCIsImluc2VydGVkSWQiLCJleGlzdGluZ1JlY29yZCIsImlzT3ZlcnJpZGVNb2RlIiwicHJvcE5hbWUiLCJnZW5lcmF0ZWQiLCJwYXJlbnRJZCIsInJlbGF0ZWRGaXh0dXJlSWQiLCJza2lwcGVkRXhpc3RpbmdJZCIsInJlbGF0ZWRSZWYiLCJyZWxhdGVkSW5zZXJ0ZWRJZHMiLCJhY3R1YWxJZCIsImNvbnZlcnRDb2x1bW5WYWx1ZSIsInJlZ2lzdGVyIiwiRGF0ZSIsInNvdXJjZVJlZiIsInNvdXJjZVV1aWRUb0lkIiwic291cmNlSWQiLCJBcnJheSIsImlzQXJyYXkiLCJ0YXJnZXRUYWJsZSIsImhhc1RhYmxlIiwic291cmNlQ29sdW1uIiwidGFyZ2V0Q29sdW1uIiwidGFyZ2V0SWQiLCJyZWxhdGVkVXVpZFRvSWQiLCJyZXNvbHZlZElkIiwiZm91bmQiLCJpbnNlcnQiLCJzZWxmUmVmUHJvcHMiLCJyZW1haW5pbmciLCJwcm9jZXNzZWQiLCJzaXplIiwiY3VycmVudExldmVsIiwiY2FuUHJvY2VzcyIsImV2ZXJ5IiwicmVmSWQiLCJyZWZGaXh0dXJlSWQiLCJyZW1haW5pbmdJZHMiLCJmcm9tIiwiam9pbiIsImRlbGV0ZSIsIl91bmlxdWVJbmRleGVzIiwiaW5kZXhlcyIsInVuaXF1ZUluZGV4ZXMiLCJpbmRleCIsInN0YXJ0c1dpdGgiLCJ1bmlxdWVRdWVyeSIsImhhc0NvbmRpdGlvbiIsImNvbnRhaW5zTnVsbCIsInNvbWUiLCJyZXBsYWNlIiwib3JXaGVyZSIsInFiIiwid2hlcmVJbiIsImFuZFdoZXJlIiwidW5pcXVlRm91bmQiLCJ3aGVyZUNsYXVzZSIsImRiQ29sdW1uIiwiYWRkRml4dHVyZUxvYWRlciIsImNvZGUiLCJwYXRoIiwiYXBpUm9vdFBhdGgiLCJjb250ZW50IiwidG9TdHJpbmciLCJmaXh0dXJlTG9hZGVyU3RhcnQiLCJmaXh0dXJlTG9hZGVyRW5kIiwibmV3Q29udGVudCIsIkZpeHR1cmVNYW5hZ2VyIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxZQUFZLFNBQVM7QUFDNUIsT0FBT0MsV0FBVyxRQUFRO0FBQzFCLFNBQVNDLFFBQVEsUUFBUSxnQkFBZ0I7QUFDekMsU0FBU0MsWUFBWSxFQUFFQyxhQUFhLFFBQVEs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
|
|
718
|
+
//#endregion
|
|
719
|
+
init_fixture_manager();
|
|
720
|
+
export { FixtureManager, FixtureManagerClass, init_fixture_manager };
|
|
721
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4dHVyZS1tYW5hZ2VyLmpzIiwibmFtZXMiOlsiZmllbGQ6IHN0cmluZyIsImlkOiBudW1iZXIiLCJhcmdzIiwiZml4dHVyZXM6IEZpeHR1cmVSZWNvcmRbXSIsImVudGl0eSIsInJlY29yZHM6IEZpeHR1cmVSZWNvcmRbXSIsInJvdyIsInJlY29yZDogRml4dHVyZVJlY29yZCIsInJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSIsInRhYmxlT3JkZXI6IHN0cmluZ1tdIiwicm93OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiIsInRhcmdldElkOiBudW1iZXIgfCBzdHJpbmciLCJsZXZlbHM6IEZpeHR1cmVSZWNvcmRbXVtdIiwiY3VycmVudExldmVsOiBGaXh0dXJlUmVjb3JkW10iLCJ3aGVyZUNsYXVzZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4iXSwic291cmNlcyI6WyIuLi8uLi9zcmMvdGVzdGluZy9maXh0dXJlLW1hbmFnZXIudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFzc2VydCBmcm9tIFwiYXNzZXJ0XCI7XG5pbXBvcnQgeyBleGVjU3luYyB9IGZyb20gXCJjaGlsZF9wcm9jZXNzXCI7XG5pbXBvcnQgeyByZWFkRmlsZVN5bmMsIHdyaXRlRmlsZVN5bmMgfSBmcm9tIFwiZnNcIjtcbmltcG9ydCB7IGluc3BlY3QgfSBmcm9tIFwidXRpbFwiO1xuXG5pbXBvcnQgY2hhbGsgZnJvbSBcImNoYWxrXCI7XG5pbXBvcnQgaW5mbGVjdGlvbiBmcm9tIFwiaW5mbGVjdGlvblwiO1xuaW1wb3J0IHsgdHlwZSBLbmV4IH0gZnJvbSBcImtuZXhcIjtcbmltcG9ydCB7IHVuaXF1ZSB9IGZyb20gXCJyYWRhc2hpXCI7XG5cbmltcG9ydCB7IFNvbmFtdSB9IGZyb20gXCIuLi9hcGkvc29uYW11XCI7XG5pbXBvcnQgeyBCYXNlTW9kZWwgfSBmcm9tIFwiLi4vZGF0YWJhc2UvYmFzZS1tb2RlbFwiO1xuaW1wb3J0IHsgdHlwZSBTb25hbXVEQkNvbmZpZyB9IGZyb20gXCIuLi9kYXRhYmFzZS9kYlwiO1xuaW1wb3J0IHsgY3JlYXRlS25leEluc3RhbmNlIH0gZnJvbSBcIi4uL2RhdGFiYXNlL2tuZXhcIjtcbmltcG9ydCB7IFVwc2VydEJ1aWxkZXIgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB7IHR5cGUgVUJSZWYgfSBmcm9tIFwiLi4vZGF0YWJhc2UvdXBzZXJ0LWJ1aWxkZXJcIjtcbmltcG9ydCB7IHR5cGUgRW50aXR5IH0gZnJvbSBcIi4uL2VudGl0eS9lbnRpdHlcIjtcbmltcG9ydCB7IEVudGl0eU1hbmFnZXIgfSBmcm9tIFwiLi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5pbXBvcnQge1xuICBpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNIYXNNYW55UmVsYXRpb25Qcm9wLFxuICBpc01hbnlUb01hbnlSZWxhdGlvblByb3AsXG4gIGlzT25lVG9PbmVSZWxhdGlvblByb3AsXG4gIGlzUmVsYXRpb25Qcm9wLFxuICBpc1ZpcnR1YWxQcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7XG4gIHR5cGUgQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wLFxuICB0eXBlIERhdGFiYXNlU2NoZW1hRXh0ZW5kLFxuICB0eXBlIEVudGl0eVByb3AsXG4gIHR5cGUgRml4dHVyZUltcG9ydFJlc3VsdCxcbiAgdHlwZSBGaXh0dXJlUmVjb3JkLFxuICB0eXBlIEZpeHR1cmVTZWFyY2hPcHRpb25zLFxuICB0eXBlIE9uZVRvT25lUmVsYXRpb25Qcm9wLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCB7IGlzVGVzdCB9IGZyb20gXCIuLi91dGlscy9jb250cm9sbGVyXCI7XG5pbXBvcnQgeyBSZWxhdGlvbkdyYXBoIH0gZnJvbSBcIi4vX3JlbGF0aW9uLWdyYXBoXCI7XG5cbi8qKiDsgqzsmqnsnpAg7KeA7KCVIOykkeuztSDtmZXsnbgg7Lus65+8IChlbnRpdHlJZOuzhOuhnCDsp4DsoJUpICovXG5leHBvcnQgaW50ZXJmYWNlIER1cGxpY2F0ZUNoZWNrT3B0aW9ucyB7XG4gIGNvbHVtbnM/OiB7XG4gICAgW2VudGl0eUlkOiBzdHJpbmddOiBzdHJpbmdbXTtcbiAgfTtcbn1cblxuZXhwb3J0IGNsYXNzIEZpeHR1cmVNYW5hZ2VyQ2xhc3Mge1xuICBwcml2YXRlIF90ZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IHRkYih0ZGI6IEtuZXgpIHtcbiAgICB0aGlzLl90ZGIgPSB0ZGI7XG4gIH1cbiAgZ2V0IHRkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fdGRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl90ZGI7XG4gIH1cblxuICBwcml2YXRlIF9mZGI6IEtuZXggfCBudWxsID0gbnVsbDtcbiAgc2V0IGZkYihmZGI6IEtuZXgpIHtcbiAgICB0aGlzLl9mZGIgPSBmZGI7XG4gIH1cbiAgZ2V0IGZkYigpOiBLbmV4IHtcbiAgICBpZiAodGhpcy5fZmRiID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGaXh0dXJlTWFuYWdlciBoYXMgbm90IGJlZW4gaW5pdGlhbGl6ZWRcIik7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9mZGI7XG4gIH1cbiAgY2FjaGVkVGFibGVOYW1lczogc3RyaW5nW10gfCBudWxsID0gbnVsbDtcblxuICBwcml2YXRlIHJlbGF0aW9uR3JhcGggPSBuZXcgUmVsYXRpb25HcmFwaCgpO1xuXG4gIC8vIFVwc2VydEJ1aWxkZXIg6riw67CYIGltcG9ydOulvCDsnITtlZwg7IOB7YOcXG4gIHByaXZhdGUgYnVpbGRlcjogVXBzZXJ0QnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gIHByaXZhdGUgZml4dHVyZVJlZk1hcDogTWFwPHN0cmluZywgVUJSZWY+ID0gbmV3IE1hcCgpO1xuICBwcml2YXRlIHNraXBwZWRGaXh0dXJlczogTWFwPHN0cmluZywgeyBlbnRpdHlJZDogc3RyaW5nOyBleGlzdGluZ0lkOiBudW1iZXIgfCBzdHJpbmcgfT4gPVxuICAgIG5ldyBNYXAoKTtcblxuICBpbml0KCkge1xuICAgIGlmICh0aGlzLl90ZGIgIT09IG51bGwpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKFNvbmFtdS5kYkNvbmZpZy50ZXN0ICYmIFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlcikge1xuICAgICAgY29uc3QgdENvbm4gPSBTb25hbXUuZGJDb25maWcudGVzdC5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBjb25zdCBwQ29ubiA9IFNvbmFtdS5kYkNvbmZpZy5wcm9kdWN0aW9uX21hc3Rlci5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyAmIHtcbiAgICAgICAgcG9ydD86IG51bWJlcjtcbiAgICAgIH07XG4gICAgICBpZiAoXG4gICAgICAgIGAke3RDb25uLmhvc3QgPz8gXCJsb2NhbGhvc3RcIn06JHt0Q29ubi5wb3J0ID8/IDU0MzJ9LyR7dENvbm4uZGF0YWJhc2V9YCA9PT1cbiAgICAgICAgYCR7cENvbm4uaG9zdCA/PyBcImxvY2FsaG9zdFwifToke3BDb25uLnBvcnQgPz8gNTQzMn0vJHtwQ29ubi5kYXRhYmFzZX1gXG4gICAgICApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDthYzsiqTtirhEQuyZgCDtlITroZzrjZXshZhEQuyXkCDrj5nsnbztlZwg642w7J207YSw67Kg7J207Iqk6rCAIOyCrOyaqeuQmOyXiOyKteuLiOuLpC5gKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICB0aGlzLnRkYiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWcudGVzdCk7XG4gICAgdGhpcy5mZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUpO1xuICB9XG5cbiAgLyoqXG4gICAg7JuQ6rKpIGZpeHR1cmUgRELrpbwg66Gc7LusIHRlc3QgRELroZwg67O17IKs7ZWp64uI64ukLlxuICAgIHBnX2R1bXDroZwg7JuQ6rKpIERC66W8IOuNpO2UhO2VmOqzoCwgcGdfcmVzdG9yZeuhnCDroZzsu6zsl5Ag67O17JuQ7ZWp64uI64ukLlxuICAqL1xuICBhc3luYyBzeW5jKCkge1xuICAgIGNvbnN0IGZpeHR1cmVDb25uID0gU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LlBnQ29ubmVjdGlvbkNvbmZpZztcbiAgICBjb25zdCB0ZXN0Q29ubiA9IFNvbmFtdS5kYkNvbmZpZy50ZXN0LmNvbm5lY3Rpb24gYXMgS25leC5QZ0Nvbm5lY3Rpb25Db25maWc7XG5cbiAgICAvLyAxLiDroZzsu6wgdGVzdCBEQiDsl7DqsrAg7KKF66OMIOuwjyDsnqzsg53shLFcbiAgICBjb25zdCB0ZXN0UGdFbnYgPSB7IFBHUEFTU1dPUkQ6IHRlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCIgfTtcbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJcbiAgICAgICAgU0VMRUNUIHBnX3Rlcm1pbmF0ZV9iYWNrZW5kKHBnX3N0YXRfYWN0aXZpdHkucGlkKVxuICAgICAgICBGUk9NIHBnX3N0YXRfYWN0aXZpdHlcbiAgICAgICAgV0hFUkUgZGF0bmFtZSA9ICcke3Rlc3RDb25uLmRhdGFiYXNlfSdcbiAgICAgICAgICBBTkQgcGlkIDw+IHBnX2JhY2tlbmRfcGlkKCk7XG4gICAgICBcImAsXG4gICAgICB7IHN0ZGlvOiBcImluaGVyaXRcIiwgZW52OiB7IC4uLnByb2Nlc3MuZW52LCAuLi50ZXN0UGdFbnYgfSBhcyBOb2RlSlMuUHJvY2Vzc0VudiB9LFxuICAgICk7XG5cbiAgICBleGVjU3luYyhcbiAgICAgIGBwc3FsIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgcG9zdGdyZXMgLWMgXCJEUk9QIERBVEFCQVNFIElGIEVYSVNUUyBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIGV4ZWNTeW5jKFxuICAgICAgYHBzcWwgLWggJHt0ZXN0Q29ubi5ob3N0fSAtcCAke3Rlc3RDb25uLnBvcnQgPz8gNTQzMn0gLVUgJHt0ZXN0Q29ubi51c2VyfSAtZCBwb3N0Z3JlcyAtYyBcIkNSRUFURSBEQVRBQkFTRSBcXFxcXCIke3Rlc3RDb25uLmRhdGFiYXNlfVxcXFxcIlwiYCxcbiAgICAgIHsgc3RkaW86IFwiaW5oZXJpdFwiLCBlbnY6IHsgLi4ucHJvY2Vzcy5lbnYsIC4uLnRlc3RQZ0VudiB9IGFzIE5vZGVKUy5Qcm9jZXNzRW52IH0sXG4gICAgKTtcblxuICAgIC8vIDIuIOybkOqyqSBmaXh0dXJlIERCIOKGkiDroZzsu6wgdGVzdCBEQuuhnCDrs7XsgqwgKHBnX2R1bXAgfCBwZ19yZXN0b3JlKVxuICAgIGNvbnN0IGZpeHR1cmVQZ0VudiA9IHsgUEdQQVNTV09SRDogZml4dHVyZUNvbm4ucGFzc3dvcmQgfHwgXCJcIiB9O1xuICAgIGNvbnN0IGR1bXBDbWQgPSBgcGdfZHVtcCAtaCAke2ZpeHR1cmVDb25uLmhvc3R9IC1wICR7Zml4dHVyZUNvbm4ucG9ydCA/PyA1NDMyfSAtVSAke2ZpeHR1cmVDb25uLnVzZXJ9IC1kICR7Zml4dHVyZUNvbm4uZGF0YWJhc2V9IC1GY2A7XG4gICAgY29uc3QgcmVzdG9yZUNtZCA9IGBwZ19yZXN0b3JlIC1oICR7dGVzdENvbm4uaG9zdH0gLXAgJHt0ZXN0Q29ubi5wb3J0ID8/IDU0MzJ9IC1VICR7dGVzdENvbm4udXNlcn0gLWQgJHt0ZXN0Q29ubi5kYXRhYmFzZX0gLS1uby1vd25lciAtLW5vLWFjbGA7XG5cbiAgICBleGVjU3luYyhgJHtkdW1wQ21kfSB8IFBHUEFTU1dPUkQ9XCIke3Rlc3RDb25uLnBhc3N3b3JkIHx8IFwiXCJ9XCIgJHtyZXN0b3JlQ21kfWAsIHtcbiAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICAgIGVudjogeyAuLi5wcm9jZXNzLmVudiwgLi4uZml4dHVyZVBnRW52IH0gYXMgTm9kZUpTLlByb2Nlc3NFbnYsXG4gICAgICBzaGVsbDogXCIvYmluL2Jhc2hcIixcbiAgICB9KTtcblxuICAgIC8vIDMuIOyLnO2AgOyKpCDrpqzshYsgKOuNsOydtO2EsCDrs7Xsgqwg7ZuEIOyLnO2AgOyKpOulvCBNQVgoaWQp66GcIOygleugrClcbiAgICBhd2FpdCB0aGlzLnJlc2V0U2VxdWVuY2VzKFNvbmFtdS5kYkNvbmZpZy50ZXN0KTtcbiAgfVxuXG4gIC8qKlxuICAgKiDrqqjrk6Ag7YWM7J2067iU7J2YIOyLnO2AgOyKpOulvCDtmITsnqwgTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAqIGZpeHR1cmUgc3luYyDtm4Qg7Iuc7YCA7Iqk6rCAIOyLpOygnCDrjbDsnbTthLDsmYAg66ee7KeAIOyViuuKlCDrrLjsoJzrpbwg7ZW06rKw7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyByZXNldFNlcXVlbmNlcyhkYkNvbmZpZzogU29uYW11REJDb25maWdbXCJ0ZXN0XCJdKSB7XG4gICAgY29uc3QgdGVzdERiID0gY3JlYXRlS25leEluc3RhbmNlKGRiQ29uZmlnKTtcbiAgICBjb25zdCBlbnRpdGllcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKTtcblxuICAgIHRyeSB7XG4gICAgICBmb3IgKGNvbnN0IGVudGl0eSBvZiBlbnRpdGllcykge1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGUgfHwgZW50aXR5LmlkLnRvTG93ZXJDYXNlKCk7XG5cbiAgICAgICAgLy8gaWQg7ZWE65Oc7J2YIO2DgOyeheydhCDtmZXsnbjtlanri4jri6RcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgY29uc3QgaWRUeXBlID0gaWRQcm9wPy50eXBlO1xuXG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlcuydtOqxsOuCmCwgc3RyaW5n7J207KeA66eMIGZpeHR1cmVTdHJhdGVneT1zZXF1ZW5jZeyduCDqsr3smrDsl5Drp4wg66as7IWL7ZWp64uI64ukXG4gICAgICAgIGNvbnN0IHVzZXNTZXF1ZW5jZSA9XG4gICAgICAgICAgaWRUeXBlID09PSBcImludGVnZXJcIiB8fFxuICAgICAgICAgIGlkVHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICBpZFByb3A/LmNvbmU/LmZpeHR1cmVTdHJhdGVneSA9PT0gXCJzZXF1ZW5jZVwiO1xuXG4gICAgICAgIGlmICghdXNlc1NlcXVlbmNlKSB7XG4gICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgYFNraXBwaW5nIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX0gKGlkIHR5cGU6ICR7aWRUeXBlIHx8IFwidW5rbm93blwifSlgLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFBvc3RncmVTUUwg7Iuc7YCA7Iqk66W8IO2YhOyerCDthYzsnbTruJTsnZggTUFYKGlkKeuhnCDrpqzshYvtlanri4jri6QuXG4gICAgICAgIC8vIHN0cmluZyDtg4DsnoXsnZgg6rK97JqwIOyIq+yekCDsupDsiqTtjIXsnbQg7ZWE7JqU7ZWp64uI64ukLlxuICAgICAgICBjb25zdCBtYXhFeHByID0gaWRUeXBlID09PSBcInN0cmluZ1wiID8gXCJNQVgoaWQ6OmJpZ2ludClcIiA6IFwiTUFYKGlkKVwiO1xuICAgICAgICBhd2FpdCB0ZXN0RGIucmF3KGBcbiAgICAgICAgICBTRUxFQ1Qgc2V0dmFsKFxuICAgICAgICAgICAgcGdfZ2V0X3NlcmlhbF9zZXF1ZW5jZSgncHVibGljLiR7dGFibGVOYW1lfScsICdpZCcpLFxuICAgICAgICAgICAgQ09BTEVTQ0UoKFNFTEVDVCAke21heEV4cHJ9IEZST00gJHt0YWJsZU5hbWV9KSwgMSlcbiAgICAgICAgICApXG4gICAgICAgIGApO1xuICAgICAgfVxuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCB0ZXN0RGIuZGVzdHJveSgpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgdmlzaXRlZFJlY29yZHMgPSBuZXcgU2V0PHN0cmluZz4oKTtcbiAgYXN5bmMgaW1wb3J0Rml4dHVyZShlbnRpdHlJZDogc3RyaW5nLCBpZHM6IG51bWJlcltdKSB7XG4gICAgLy8g67Cp66y4IOq4sOuhnSDstIjquLDtmZQgKOyDiOuhnOyatCBpbXBvcnQg7J6R7JeFIOyLnOyekSlcbiAgICB0aGlzLnZpc2l0ZWRSZWNvcmRzLmNsZWFyKCk7XG5cbiAgICBjb25zdCBxdWVyaWVzID0gdW5pcXVlKFxuICAgICAgKFxuICAgICAgICBhd2FpdCBQcm9taXNlLmFsbChcbiAgICAgICAgICBpZHMubWFwKGFzeW5jIChpZCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZCwgXCJpZFwiLCBpZCk7XG4gICAgICAgICAgfSksXG4gICAgICAgIClcbiAgICAgICkuZmxhdCgpLFxuICAgICk7XG5cbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuICAgIGZvciAoY29uc3QgcXVlcnkgb2YgcXVlcmllcykge1xuICAgICAgYXdhaXQgd2RiLnJhdyhxdWVyeSk7XG4gICAgfVxuICB9XG5cbiAgYXN5bmMgZ2V0SW1wb3J0UXVlcmllcyhlbnRpdHlJZDogc3RyaW5nLCBmaWVsZDogc3RyaW5nLCBpZDogbnVtYmVyKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHJlY29yZEtleSA9IGAke2VudGl0eUlkfSMke2ZpZWxkfSMke2lkfWA7XG5cbiAgICAvLyDsiJztmZgg7LC47KGwIOuwqeyngDog7J2066+4IOuwqeusuO2VnCDroIjsvZTrk5zripQg7Iqk7YK1XG4gICAgaWYgKHRoaXMudmlzaXRlZFJlY29yZHMuaGFzKHJlY29yZEtleSkpIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG4gICAgdGhpcy52aXNpdGVkUmVjb3Jkcy5hZGQocmVjb3JkS2V5KTtcblxuICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKTtcbiAgICBjb25zdCB3ZGIgPSBCYXNlTW9kZWwuZ2V0REIoXCJ3XCIpO1xuXG4gICAgLy8g7Jes6riw7IScIOyLpERC7J2YIHJvdyDqsIDsoLjsmLRcbiAgICBjb25zdCBbcm93XSA9IGF3YWl0IHdkYihlbnRpdHkudGFibGUpLndoZXJlKGZpZWxkLCBpZCkubGltaXQoMSk7XG4gICAgaWYgKHJvdyA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5SWR9IyR7aWR9IHJvd+ulvCDssL7snYQg7IiYIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICAvLyDtlL3siqTss5BEQiwg7IukREJcbiAgICBjb25zdCBmaXh0dXJlRGF0YWJhc2UgPSAoU29uYW11LmRiQ29uZmlnLmZpeHR1cmUuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpLmRhdGFiYXNlO1xuICAgIGNvbnN0IHJlYWxEYXRhYmFzZSA9IChTb25hbXUuZGJDb25maWcucHJvZHVjdGlvbl9tYXN0ZXIuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWcpXG4gICAgICAuZGF0YWJhc2U7XG5cbiAgICBjb25zdCBzZWxmUXVlcnkgPSBgSU5TRVJUIElHTk9SRSBJTlRPIFxcYCR7Zml4dHVyZURhdGFiYXNlfVxcYC5cXGAke2VudGl0eS50YWJsZX1cXGAgKFNFTEVDVCAqIEZST00gXFxgJHtyZWFsRGF0YWJhc2V9XFxgLlxcYCR7ZW50aXR5LnRhYmxlfVxcYCBXSEVSRSBcXGBpZFxcYCA9ICR7aWR9KWA7XG5cbiAgICBjb25zdCBhcmdzID0gT2JqZWN0LmVudHJpZXMoZW50aXR5LnJlbGF0aW9ucylcbiAgICAgIC5maWx0ZXIoXG4gICAgICAgIChbLCByZWxhdGlvbl0pID0+XG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8XG4gICAgICAgICAgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pICYmIHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UgPT09IHVuZGVmaW5lZCksXG4gICAgICApXG4gICAgICAubWFwKChbLCByZWxhdGlvbl0pID0+IHtcbiAgICAgICAgLypcbiAgICAgICAgQmVsb25nc1RvT25l7J24IOqyveyasFxuICAgICAgICAgIENhdGVnb3J5IC8gJ2lkJyAvIHJvd1tjYXRlZ29yeV9pZF0g7Zi47LacXG4gICAgICAgIE9uZVRvT25l7JeQIGpvaW5Db2x1bW4gPT09IHRydWUg7J24IOqyveyasFxuICAgICAgICAgIFByb2ZpbGUgLyAnaWQnIC8gcm93W3Byb2ZpbGVfaWRdIO2YuOy2nFxuICAgICAgICBPbmVUb09uZeyXkCBqb2luQ29sdW1uID09PSBmYWxzZSDsnbgg6rK97JqwXG4gICAgICAgICAgUHJvZmlsZSAvICdwcm9maWxlX2lkJyAvIHJvd1snaWQnXSDtmLjstpxcbiAgICAgICAgKi9cbiAgICAgICAgbGV0IGZpZWxkOiBzdHJpbmc7XG4gICAgICAgIGxldCBpZDogbnVtYmVyO1xuICAgICAgICBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChyZWxhdGlvbikgJiYgIXJlbGF0aW9uLmhhc0pvaW5Db2x1bW4pIHtcbiAgICAgICAgICBjb25zdCByZWxhdGVkRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG4gICAgICAgICAgY29uc3QgcmVsYXRlZElkQ29sdW1uTmFtZSA9IHJlbGF0ZWRFbnRpdHkucHJvcHMuZmluZChcbiAgICAgICAgICAgIChwKSA9PiBpc1JlbGF0aW9uUHJvcChwKSAmJiBwLndpdGggPT09IGVudGl0eS5pZCxcbiAgICAgICAgICApPy5uYW1lO1xuICAgICAgICAgIGlmICghcmVsYXRlZElkQ29sdW1uTmFtZSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3JlbGF0ZWRFbnRpdHkuaWR97J2YICR7ZW50aXR5LmlkfSDqtIDqs4Qg7ZSE66Gt7J2EIOywvuydhCDsiJgg7JeG7Iq164uI64ukLmApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBmaWVsZCA9IGAke3JlbGF0ZWRJZENvbHVtbk5hbWV9X2lkYDtcbiAgICAgICAgICBpZCA9IHJvdy5pZDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBmaWVsZCA9IFwiaWRcIjtcbiAgICAgICAgICBpZCA9IHJvd1tgJHtyZWxhdGlvbi5uYW1lfV9pZGBdO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZW50aXR5SWQ6IHJlbGF0aW9uLndpdGgsXG4gICAgICAgICAgZmllbGQsXG4gICAgICAgICAgaWQsXG4gICAgICAgIH07XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoYXJnKSA9PiBhcmcuaWQgIT09IG51bGwpO1xuXG4gICAgY29uc3QgcmVsUXVlcmllcyA9IGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgYXJncy5tYXAoYXN5bmMgKGFyZ3MpID0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZ2V0SW1wb3J0UXVlcmllcyhhcmdzLmVudGl0eUlkLCBhcmdzLmZpZWxkLCBhcmdzLmlkKTtcbiAgICAgIH0pLFxuICAgICk7XG5cbiAgICByZXR1cm4gWy4uLnVuaXF1ZShyZWxRdWVyaWVzLnRvUmV2ZXJzZWQoKS5mbGF0KCkpLCBzZWxmUXVlcnldO1xuICB9XG5cbiAgYXN5bmMgZGVzdHJveSgpIHtcbiAgICBpZiAodGhpcy5fdGRiKSB7XG4gICAgICBhd2FpdCB0aGlzLl90ZGIuZGVzdHJveSgpO1xuICAgICAgdGhpcy5fdGRiID0gbnVsbDtcbiAgICB9XG4gICAgaWYgKHRoaXMuX2ZkYikge1xuICAgICAgYXdhaXQgdGhpcy5fZmRiLmRlc3Ryb3koKTtcbiAgICAgIHRoaXMuX2ZkYiA9IG51bGw7XG4gICAgfVxuICAgIGF3YWl0IEJhc2VNb2RlbC5kZXN0cm95KCk7XG4gIH1cblxuICBhc3luYyBnZXRGaXh0dXJlcyhcbiAgICBzb3VyY2VEQk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIHRhcmdldERCTmFtZToga2V5b2YgU29uYW11REJDb25maWcsXG4gICAgc2VhcmNoT3B0aW9uczogRml4dHVyZVNlYXJjaE9wdGlvbnMsXG4gICAgZHVwbGljYXRlQ2hlY2s/OiBEdXBsaWNhdGVDaGVja09wdGlvbnMsXG4gICkge1xuICAgIGNvbnN0IHNvdXJjZURCID0gY3JlYXRlS25leEluc3RhbmNlKFNvbmFtdS5kYkNvbmZpZ1tzb3VyY2VEQk5hbWVdKTtcbiAgICBjb25zdCB0YXJnZXREQiA9IGNyZWF0ZUtuZXhJbnN0YW5jZShTb25hbXUuZGJDb25maWdbdGFyZ2V0REJOYW1lXSk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgeyBlbnRpdHlJZCwgZmllbGQsIHZhbHVlLCBzZWFyY2hUeXBlIH0gPSBzZWFyY2hPcHRpb25zO1xuXG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCk7XG4gICAgICBjb25zdCBjb2x1bW4gPVxuICAgICAgICBlbnRpdHkucHJvcHMuZmluZCgocHJvcCkgPT4gcHJvcC5uYW1lID09PSBmaWVsZCk/LnR5cGUgPT09IFwicmVsYXRpb25cIlxuICAgICAgICAgID8gYCR7ZmllbGR9X2lkYFxuICAgICAgICAgIDogZmllbGQ7XG5cbiAgICAgIGxldCBxdWVyeSA9IHNvdXJjZURCKGVudGl0eS50YWJsZSk7XG4gICAgICBpZiAoc2VhcmNoVHlwZSA9PT0gXCJlcXVhbHNcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgdmFsdWUpO1xuICAgICAgfSBlbHNlIGlmIChzZWFyY2hUeXBlID09PSBcImxpa2VcIikge1xuICAgICAgICBxdWVyeSA9IHF1ZXJ5LndoZXJlKGNvbHVtbiwgXCJsaWtlXCIsIGAlJHt2YWx1ZX0lYCk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHJvd3MgPSBhd2FpdCBxdWVyeTtcbiAgICAgIGlmIChyb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJObyByZWNvcmRzIGZvdW5kXCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmaXh0dXJlczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICAgIGNvbnN0IGluaXRpYWxSZWNvcmRzTGVuZ3RoID0gZml4dHVyZXMubGVuZ3RoO1xuICAgICAgICBjb25zdCBuZXdSZWNvcmRzID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgcm93LCB7XG4gICAgICAgICAgX2RiOiBzb3VyY2VEQixcbiAgICAgICAgfSk7XG4gICAgICAgIGZpeHR1cmVzLnB1c2goLi4ubmV3UmVjb3Jkcyk7XG4gICAgICAgIGNvbnN0IGN1cnJlbnRGaXh0dXJlUmVjb3JkID0gZml4dHVyZXMuZmluZCgocikgPT4gci5maXh0dXJlSWQgPT09IGAke2VudGl0eUlkfSMke3Jvdy5pZH1gKTtcblxuICAgICAgICBpZiAoY3VycmVudEZpeHR1cmVSZWNvcmQpIHtcbiAgICAgICAgICAvLyDtmITsnqwgZml4dHVyZeuhnOu2gO2EsCDsg53shLHrkJwgZmV0Y2hlZFJlY29yZHMg7ISk7KCVXG4gICAgICAgICAgY3VycmVudEZpeHR1cmVSZWNvcmQuZmV0Y2hlZFJlY29yZHMgPSBmaXh0dXJlc1xuICAgICAgICAgICAgLmZpbHRlcigocikgPT4gci5maXh0dXJlSWQgIT09IGN1cnJlbnRGaXh0dXJlUmVjb3JkLmZpeHR1cmVJZClcbiAgICAgICAgICAgIC5zbGljZShpbml0aWFsUmVjb3Jkc0xlbmd0aClcbiAgICAgICAgICAgIC5tYXAoKHIpID0+IHIuZml4dHVyZUlkKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBmb3IgYXdhaXQgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQoZml4dHVyZS5lbnRpdHlJZCk7XG5cbiAgICAgICAgLy8g7IKs7Jqp7J6QIOyngOyglSDsu6zrn7wg6riw7KSAIOykkeuztSDtmZXsnbgg4oaSIHRhcmdldFxuICAgICAgICBjb25zdCBjdXN0b21Db2x1bW5zID0gZHVwbGljYXRlQ2hlY2s/LmNvbHVtbnM/LltmaXh0dXJlLmVudGl0eUlkXTtcbiAgICAgICAgaWYgKGN1c3RvbUNvbHVtbnMgJiYgY3VzdG9tQ29sdW1ucy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgY3VzdG9tRHVwbGljYXRlUm93ID0gYXdhaXQgdGhpcy5jaGVja0R1cGxpY2F0ZUJ5Q29sdW1ucyhcbiAgICAgICAgICAgIHRhcmdldERCLFxuICAgICAgICAgICAgZW50aXR5LFxuICAgICAgICAgICAgZml4dHVyZSxcbiAgICAgICAgICAgIGN1c3RvbUNvbHVtbnMsXG4gICAgICAgICAgKTtcbiAgICAgICAgICBpZiAoY3VzdG9tRHVwbGljYXRlUm93KSB7XG4gICAgICAgICAgICBjb25zdCBbcmVjb3JkXSA9IGF3YWl0IHRoaXMuY3JlYXRlRml4dHVyZVJlY29yZChlbnRpdHksIGN1c3RvbUR1cGxpY2F0ZVJvdywge1xuICAgICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICAgIF9kYjogdGFyZ2V0REIsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGZpeHR1cmUudGFyZ2V0ID0gcmVjb3JkO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVuaXF1ZSBpbmRleCDquLDspIAg7KSR67O1IO2ZleyduCDihpIgZml4dHVyZS51bmlxdWVcbiAgICAgICAgY29uc3QgdW5pcXVlUm93ID0gYXdhaXQgdGhpcy5jaGVja1VuaXF1ZVZpb2xhdGlvbih0YXJnZXREQiwgZW50aXR5LCBmaXh0dXJlKTtcbiAgICAgICAgaWYgKHVuaXF1ZVJvdykge1xuICAgICAgICAgIGNvbnN0IFtyZWNvcmRdID0gYXdhaXQgdGhpcy5jcmVhdGVGaXh0dXJlUmVjb3JkKGVudGl0eSwgdW5pcXVlUm93LCB7XG4gICAgICAgICAgICBzaW5nbGVSZWNvcmQ6IHRydWUsXG4gICAgICAgICAgICBfZGI6IHRhcmdldERCLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGZpeHR1cmUudW5pcXVlID0gcmVjb3JkO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB1bmlxdWUoZml4dHVyZXMsIChmKSA9PiBmLmZpeHR1cmVJZCk7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZChbdGFyZ2V0REIuZGVzdHJveSgpLCBzb3VyY2VEQi5kZXN0cm95KCldKTtcbiAgICB9XG4gIH1cblxuICBhc3luYyBjcmVhdGVGaXh0dXJlUmVjb3JkKFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIHJvdzoge1xuICAgICAgaWQ6IG51bWJlciB8IHN0cmluZztcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsO1xuICAgIH0sXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIHNpbmdsZVJlY29yZD86IGJvb2xlYW47XG4gICAgICBfZGI/OiBLbmV4O1xuICAgIH0sXG4gICk6IFByb21pc2U8Rml4dHVyZVJlY29yZFtdPiB7XG4gICAgY29uc3QgcmVjb3JkczogRml4dHVyZVJlY29yZFtdID0gW107XG4gICAgY29uc3QgdmlzaXRlZEVudGl0aWVzID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgICBjb25zdCBjcmVhdGUgPSBhc3luYyAoXG4gICAgICBlbnRpdHk6IEVudGl0eSxcbiAgICAgIHJvdzoge1xuICAgICAgICBpZDogbnVtYmVyIHwgc3RyaW5nO1xuICAgICAgICBba2V5OiBzdHJpbmddOiBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgbnVsbDtcbiAgICAgIH0sXG4gICAgKSA9PiB7XG4gICAgICBjb25zdCBmaXh0dXJlSWQgPSBgJHtlbnRpdHkuaWR9IyR7cm93LmlkfWA7XG4gICAgICBpZiAodmlzaXRlZEVudGl0aWVzLmhhcyhmaXh0dXJlSWQpKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIHZpc2l0ZWRFbnRpdGllcy5hZGQoZml4dHVyZUlkKTtcblxuICAgICAgY29uc3QgcmVjb3JkOiBGaXh0dXJlUmVjb3JkID0ge1xuICAgICAgICBmaXh0dXJlSWQsXG4gICAgICAgIGVudGl0eUlkOiBlbnRpdHkuaWQsXG4gICAgICAgIGlkOiByb3cuaWQsXG4gICAgICAgIGNvbHVtbnM6IHt9LFxuICAgICAgICBmZXRjaGVkUmVjb3JkczogW10sXG4gICAgICAgIGJlbG9uZ3NSZWNvcmRzOiBbXSxcbiAgICAgIH07XG5cbiAgICAgIGZvciAoY29uc3QgcHJvcCBvZiBlbnRpdHkucHJvcHMpIHtcbiAgICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0gPSB7XG4gICAgICAgICAgcHJvcDogcHJvcCxcbiAgICAgICAgICB2YWx1ZTogcm93W3Byb3AubmFtZV0sXG4gICAgICAgIH07XG5cbiAgICAgICAgY29uc3QgZGIgPSBvcHRpb25zPy5fZGIgPz8gQmFzZU1vZGVsLmdldERCKFwid1wiKTtcbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHRocm91Z2hUYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IGZyb21Db2x1bW4gPSBgJHtpbmZsZWN0aW9uLnNpbmd1bGFyaXplKGVudGl0eS50YWJsZSl9X2lkYDtcbiAgICAgICAgICBjb25zdCB0b0NvbHVtbiA9IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsYXRlZEVudGl0eS50YWJsZSl9X2lkYDtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYih0aHJvdWdoVGFibGUpLndoZXJlKGZyb21Db2x1bW4sIHJvdy5pZCkucGx1Y2sodG9Db2x1bW4pO1xuICAgICAgICAgIHJlY29yZC5jb2x1bW5zW3Byb3AubmFtZV0udmFsdWUgPSByZWxhdGVkSWRzO1xuICAgICAgICB9IGVsc2UgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKVxuICAgICAgICAgICAgLndoZXJlKHByb3Auam9pbkNvbHVtbiwgcm93LmlkKVxuICAgICAgICAgICAgLnBsdWNrKFwiaWRcIik7XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZHM7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiAhcHJvcC5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgLy8g7Jet67Cp7ZalIE9uZVRvT25lOiBGS+ulvCDqsIDsp4Qg6rSA66CoIOyXlO2LsO2LsOulvCDssL7sirXri4jri6RcbiAgICAgICAgICAvLyDsmIjsi5w6IFVzZXIgT25lVG9PbmUgRW1wbG95ZWUgKEVtcGxveWVl6rCAIHVzZXJfaWQgRkvrpbwg6rCA7KeQKVxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRQcm9wID0gcmVsYXRlZEVudGl0eS5wcm9wcy5maW5kKFxuICAgICAgICAgICAgKHApID0+IGlzUmVsYXRpb25Qcm9wKHApICYmIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICAgICAgICk7XG4gICAgICAgICAgaWYgKHJlbGF0ZWRQcm9wICYmIGlzUmVsYXRpb25Qcm9wKHJlbGF0ZWRQcm9wKSkge1xuICAgICAgICAgICAgLy8g6rSA66CoIOyXlO2LsO2LsOyXkOyEnCBGSyDsu6zrn7zsnLzroZwg7L+866as7ZWp64uI64ukIChpZOqwgCDslYTri5gpXG4gICAgICAgICAgICBjb25zdCBma0NvbHVtbiA9IGAke3JlbGF0ZWRQcm9wLm5hbWV9X2lkYDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSb3cgPSBhd2FpdCBkYihyZWxhdGVkRW50aXR5LnRhYmxlKS53aGVyZShma0NvbHVtbiwgcm93LmlkKS5maXJzdCgpO1xuICAgICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRSb3c/LmlkO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IHJvd1tgJHtwcm9wLm5hbWV9X2lkYF07XG4gICAgICAgICAgcmVjb3JkLmNvbHVtbnNbcHJvcC5uYW1lXS52YWx1ZSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICBpZiAocmVsYXRlZElkKSB7XG4gICAgICAgICAgICByZWNvcmQuYmVsb25nc1JlY29yZHMucHVzaChgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIW9wdGlvbnM/LnNpbmdsZVJlY29yZCAmJiByZWxhdGVkSWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJvdyA9IGF3YWl0IGRiKHJlbGF0ZWRFbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgcmVsYXRlZElkKS5maXJzdCgpO1xuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSb3cpIHtcbiAgICAgICAgICAgICAgYXdhaXQgY3JlYXRlKHJlbGF0ZWRFbnRpdHksIHJlbGF0ZWRSb3cpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZWNvcmRzLnB1c2gocmVjb3JkKTtcbiAgICB9O1xuXG4gICAgYXdhaXQgY3JlYXRlKGVudGl0eSwgcm93KTtcblxuICAgIHJldHVybiByZWNvcmRzO1xuICB9XG5cbiAgLyoqXG4gICAqIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrAgKHNlbGYtcmVmZXJlbmNlIO2PrO2VqClcbiAgICogMi4g7YWM7J2067iU67OEIOugiOuyqOuzhOuhnCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnSDrsI8gdXBzZXJ0IOyLpO2WiVxuICAgKiAzLiDsiJzshJwg6riw67CYIHV1aWTihpJpZCDrp6TtlZEgKFVwc2VydEJ1aWxkZXLqsIAgdXVpZOulvCBEQuyXkCDsoIDsnqXtlZjsp4Ag7JWK7Jy866+A66GcKVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy64qUIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0XG4gICAqIEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWY7JesIOqwgSB1cHNlcnQg7Zi47Lac7JeQ7ISc64qUXG4gICAqIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuPhOuhnSDtlanri4jri6QuXG4gICAqL1xuICBhc3luYyBpbnNlcnRGaXh0dXJlcyhcbiAgICBkYk5hbWU6IGtleW9mIFNvbmFtdURCQ29uZmlnLFxuICAgIF9maXh0dXJlczogRml4dHVyZVJlY29yZFtdLFxuICApOiBQcm9taXNlPEZpeHR1cmVJbXBvcnRSZXN1bHRbXT4ge1xuICAgIGNvbnN0IGZpeHR1cmVzID0gdW5pcXVlKF9maXh0dXJlcywgKGYpID0+IGYuZml4dHVyZUlkKTtcblxuICAgIC8vIOy0iOq4sO2ZlFxuICAgIHRoaXMuYnVpbGRlciA9IG5ldyBVcHNlcnRCdWlsZGVyKCk7XG4gICAgdGhpcy5maXh0dXJlUmVmTWFwID0gbmV3IE1hcCgpO1xuICAgIHRoaXMuc2tpcHBlZEZpeHR1cmVzID0gbmV3IE1hcCgpO1xuXG4gICAgLy8g67OR66CsIO2FjOyKpO2KuCDrqqjrk5zsl5DshJzripQgd29ya2Vy67OEIERC7JeQIOyggOyepVxuICAgIGNvbnN0IGRiQ29uZmlnID1cbiAgICAgIHByb2Nlc3MuZW52LlNPTkFNVV9XT1JLRVJfREIgPT09IFwidHJ1ZVwiICYmIHByb2Nlc3MuZW52LlZJVEVTVF9QT09MX0lEXG4gICAgICAgID8gKCgpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHdvcmtlcklkID0gcGFyc2VJbnQocHJvY2Vzcy5lbnYuVklURVNUX1BPT0xfSUQgPz8gXCIxXCIsIDEwKTtcbiAgICAgICAgICAgIGNvbnN0IGJhc2VDb25maWcgPSBTb25hbXUuZGJDb25maWdbZGJOYW1lXTtcbiAgICAgICAgICAgIGNvbnN0IGNvbm5lY3Rpb24gPSBiYXNlQ29uZmlnLmNvbm5lY3Rpb24gYXMgeyBkYXRhYmFzZTogc3RyaW5nIH07XG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAuLi5iYXNlQ29uZmlnLFxuICAgICAgICAgICAgICBjb25uZWN0aW9uOiB7IC4uLmNvbm5lY3Rpb24sIGRhdGFiYXNlOiBgJHtjb25uZWN0aW9uLmRhdGFiYXNlfV8ke3dvcmtlcklkfWAgfSxcbiAgICAgICAgICAgICAgcG9vbDogeyBtaW46IDEsIG1heDogMSB9LFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9KSgpXG4gICAgICAgIDogU29uYW11LmRiQ29uZmlnW2RiTmFtZV07XG4gICAgY29uc3QgZGIgPSBjcmVhdGVLbmV4SW5zdGFuY2UoZGJDb25maWcpO1xuICAgIGNvbnN0IHJlc3VsdHM6IEZpeHR1cmVJbXBvcnRSZXN1bHRbXSA9IFtdO1xuXG4gICAgdHJ5IHtcbiAgICAgIC8vIDEuIFJlbGF0aW9uR3JhcGjroZwgZml4dHVyZSDri6jsnIQg7IK97J6FIOyInOyEnCDqs4TsgrBcbiAgICAgIHRoaXMucmVsYXRpb25HcmFwaC5idWlsZEdyYXBoKGZpeHR1cmVzKTtcbiAgICAgIGNvbnN0IGluc2VydGlvbk9yZGVyID0gdGhpcy5yZWxhdGlvbkdyYXBoLmdldEluc2VydGlvbk9yZGVyKCk7XG5cbiAgICAgIC8vIDIuIOyKpO2Cte2VoCBmaXh0dXJlIOuovOyggCDsspjrpqwgKG92ZXJyaWRlIOyytO2BrClcbiAgICAgIGZvciAoY29uc3QgZml4dHVyZUlkIG9mIGluc2VydGlvbk9yZGVyKSB7XG4gICAgICAgIGNvbnN0IGZpeHR1cmUgPSBmaXh0dXJlcy5maW5kKChmKSA9PiBmLmZpeHR1cmVJZCA9PT0gZml4dHVyZUlkKTtcbiAgICAgICAgaWYgKCFmaXh0dXJlKSBjb250aW51ZTtcblxuICAgICAgICBjb25zdCBoYXNUYXJnZXQgPSAhIWZpeHR1cmUudGFyZ2V0O1xuICAgICAgICBjb25zdCBoYXNVbmlxdWUgPSAhIWZpeHR1cmUudW5pcXVlO1xuICAgICAgICBjb25zdCBoYXNEdXBsaWNhdGUgPSBoYXNUYXJnZXQgfHwgaGFzVW5pcXVlO1xuXG4gICAgICAgIC8vIOykkeuzteydtCDsnojqs6Agb3ZlcnJpZGU9ZmFsc2Xsnbgg6rK97JqwOiDsiqTtgrVcbiAgICAgICAgaWYgKGhhc0R1cGxpY2F0ZSAmJiAhZml4dHVyZS5vdmVycmlkZSkge1xuICAgICAgICAgIC8vIOq4sOyhtCDroIjsvZTrk5wgSUQg7KCA7J6lICh1bmlxdWUg7Jqw7ISgLCDsl4bsnLzrqbQgdGFyZ2V0KVxuICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBmaXh0dXJlLnVuaXF1ZT8uaWQgPz8gZml4dHVyZS50YXJnZXQ/LmlkO1xuICAgICAgICAgIGFzc2VydChleGlzdGluZ0lkKTtcbiAgICAgICAgICB0aGlzLnNraXBwZWRGaXh0dXJlcy5zZXQoZml4dHVyZUlkLCB7XG4gICAgICAgICAgICBlbnRpdHlJZDogZml4dHVyZS5lbnRpdHlJZCxcbiAgICAgICAgICAgIGV4aXN0aW5nSWQsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICBjaGFsay55ZWxsb3coXG4gICAgICAgICAgICAgICAgYFNraXBwZWQgJHtmaXh0dXJlLmVudGl0eUlkfSMke2ZpeHR1cmUuaWR9IChleGlzdGluZzogIyR7ZXhpc3RpbmdJZH0sIG92ZXJyaWRlOiBmYWxzZSlgLFxuICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyAzLiDthYzsnbTruJTrs4QgZml4dHVyZSDqt7jro7ntmZQgKGluc2VydGlvbk9yZGVyIOyInOyEnCDquLDrsJgpXG4gICAgICBjb25zdCBmaXh0dXJlc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgRml4dHVyZVJlY29yZFtdPigpO1xuICAgICAgY29uc3QgdGFibGVPcmRlcjogc3RyaW5nW10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCBmaXh0dXJlSWQgb2YgaW5zZXJ0aW9uT3JkZXIpIHtcbiAgICAgICAgLy8g7Iqk7YK165CcIGZpeHR1cmUg7KCc7Jm4XG4gICAgICAgIGlmICh0aGlzLnNraXBwZWRGaXh0dXJlcy5oYXMoZml4dHVyZUlkKSkgY29udGludWU7XG5cbiAgICAgICAgY29uc3QgZml4dHVyZSA9IGZpeHR1cmVzLmZpbmQoKGYpID0+IGYuZml4dHVyZUlkID09PSBmaXh0dXJlSWQpO1xuICAgICAgICBpZiAoIWZpeHR1cmUpIGNvbnRpbnVlO1xuXG4gICAgICAgIGNvbnN0IGVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KGZpeHR1cmUuZW50aXR5SWQpO1xuICAgICAgICBjb25zdCB0YWJsZU5hbWUgPSBlbnRpdHkudGFibGU7XG5cbiAgICAgICAgaWYgKCFmaXh0dXJlc0J5VGFibGUuaGFzKHRhYmxlTmFtZSkpIHtcbiAgICAgICAgICBmaXh0dXJlc0J5VGFibGUuc2V0KHRhYmxlTmFtZSwgW10pO1xuICAgICAgICAgIHRhYmxlT3JkZXIucHVzaCh0YWJsZU5hbWUpO1xuICAgICAgICB9XG4gICAgICAgIGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKT8ucHVzaChmaXh0dXJlKTtcbiAgICAgIH1cblxuICAgICAgYXdhaXQgZGIudHJhbnNhY3Rpb24oYXN5bmMgKHRyeCkgPT4ge1xuICAgICAgICBjb25zdCBpbnNlcnRlZElkc0J5VGFibGUgPSBuZXcgTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4oKTtcblxuICAgICAgICAvLyA0LiDthYzsnbTruJTrs4Qg66CI67Ko67OEIOyymOumrFxuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgY29uc3QgdGFibGVGaXh0dXJlcyA9IGZpeHR1cmVzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBbXTtcbiAgICAgICAgICBjb25zdCBsZXZlbHMgPSB0aGlzLmdyb3VwRml4dHVyZXNCeUxldmVsKHRhYmxlRml4dHVyZXMpO1xuXG4gICAgICAgICAgZm9yIChjb25zdCBsZXZlbEZpeHR1cmVzIG9mIGxldmVscykge1xuICAgICAgICAgICAgLy8g7ZW064u5IOugiOuyqOydmCBmaXh0dXJl65OkIHJlZ2lzdGVyXG4gICAgICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgbGV2ZWxGaXh0dXJlcykge1xuICAgICAgICAgICAgICB0aGlzLnJlZ2lzdGVyRml4dHVyZShmaXh0dXJlLCBpbnNlcnRlZElkc0J5VGFibGUpO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICAgIGBSZWdpc3RlcmVkICR7Zml4dHVyZS5lbnRpdHlJZH0jJHtmaXh0dXJlLmlkfSR7Zml4dHVyZS5vdmVycmlkZSA/IGAgKG92ZXJyaWRlKWAgOiBcIlwifWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIHVwc2VydCDsi6Ttlokg7KCEIHV1aWQg66qp66GdIOyggOyepVxuICAgICAgICAgICAgY29uc3QgdGFibGUgPSB0aGlzLmJ1aWxkZXIuZ2V0VGFibGUodGFibGVOYW1lKTtcbiAgICAgICAgICAgIGNvbnN0IHV1aWRzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4gcm93LnV1aWQgYXMgc3RyaW5nKTtcblxuICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgICAgICAgIGNoYWxrLmJsdWUoXG4gICAgICAgICAgICAgICAgICBgVXBzZXJ0aW5nICR7dGFibGVOYW1lfSB3aXRoICR7dXVpZHMubGVuZ3RofSByb3dzIChsZXZlbCAke2xldmVscy5pbmRleE9mKGxldmVsRml4dHVyZXMpICsgMX0vJHtsZXZlbHMubGVuZ3RofSlgLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICBjb25zdCBpZHMgPSAoYXdhaXQgdGhpcy5idWlsZGVyLnVwc2VydChcbiAgICAgICAgICAgICAgdHJ4LFxuICAgICAgICAgICAgICB0YWJsZU5hbWUgYXMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gICAgICAgICAgICApKSBhcyAobnVtYmVyIHwgc3RyaW5nKVtdO1xuXG4gICAgICAgICAgICAvLyDsiJzshJwg6riw67CYIHV1aWQgLT4gaWQg66ek7ZWRXG4gICAgICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDsl4bsnLzrr4DroZwg65Ox66GdIOyInOyEnCA9IOuwmO2ZmCDsiJzshJwg67O07J6lXG4gICAgICAgICAgICBpZiAodXVpZHMubGVuZ3RoID4gMCAmJiB1dWlkcy5sZW5ndGggPT09IGlkcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPVxuICAgICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5nZXQodGFibGVOYW1lKSA/PyBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHV1aWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICAgICAgZXhpc3RpbmdNYXAuc2V0KHV1aWRzW2ldLCBpZHNbaV0pO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIGluc2VydGVkSWRzQnlUYWJsZS5zZXQodGFibGVOYW1lLCBleGlzdGluZ01hcCk7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKHV1aWRzLmxlbmd0aCAhPT0gaWRzLmxlbmd0aCkge1xuICAgICAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICAgICAgY2hhbGsueWVsbG93KFxuICAgICAgICAgICAgICAgICAgYFdhcm5pbmc6IHV1aWQgY291bnQgKCR7dXVpZHMubGVuZ3RofSkgIT0gaWQgY291bnQgKCR7aWRzLmxlbmd0aH0pIGZvciAke3RhYmxlTmFtZX1gLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNS4gTWFueVRvTWFueSDqtIDqs4Qg7LKY66asXG4gICAgICAgIGF3YWl0IHRoaXMucHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnModHJ4LCBmaXh0dXJlcywgaW5zZXJ0ZWRJZHNCeVRhYmxlKTtcblxuICAgICAgICAvLyA2LiBQb3N0Z3JlU1FMIOyLnO2AgOyKpCDrpqzshYtcbiAgICAgICAgLy8gRml4dHVyZSDsgr3snoUg7ZuEIOqwgSDthYzsnbTruJTsnZggSUQg7Iuc7YCA7Iqk66W8IOy1nOuMgCBJRCDqsJLsnLzroZwg7JeF642w7J207Yq47ZWp64uI64ukLlxuICAgICAgICAvLyDsnbTroIfqsowg7ZWY7KeAIOyViuycvOuptCDri6TsnYwgSU5TRVJUIOyLnCBJROqwgCAyMDAw67KI64yA66GcIOyDneyEseuQoCDsiJgg7J6I7Iq164uI64ukLlxuICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJlc2V0dGluZyBzZXF1ZW5jZXMuLi5cIikpO1xuICAgICAgICBmb3IgKGNvbnN0IHRhYmxlTmFtZSBvZiB0YWJsZU9yZGVyKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIEVudGl0eeulvCDssL7slYTshJwgaWQg7YOA7J6FIO2ZleyduFxuICAgICAgICAgICAgY29uc3QgZW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXRBbGxFbnRpdGllcygpLmZpbmQoXG4gICAgICAgICAgICAgIChlKSA9PiBlLnRhYmxlID09PSB0YWJsZU5hbWUgfHwgZS5pZC50b0xvd2VyQ2FzZSgpID09PSB0YWJsZU5hbWUsXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICBpZiAoZW50aXR5KSB7XG4gICAgICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik7XG4gICAgICAgICAgICAgIGNvbnN0IGlkVHlwZSA9IGlkUHJvcD8udHlwZTtcblxuICAgICAgICAgICAgICAvLyBpbnRlZ2VyL2JpZ0ludGVnZXLsnbTqsbDrgpgsIHN0cmluZ+ydtOyngOunjCBmaXh0dXJlU3RyYXRlZ3k9c2VxdWVuY2Xsnbgg6rK97Jqw7JeQ66eMIOumrOyFi+2VqeuLiOuLpFxuICAgICAgICAgICAgICBjb25zdCB1c2VzU2VxdWVuY2UgPVxuICAgICAgICAgICAgICAgIGlkVHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgICAgICBpZFR5cGUgPT09IFwiYmlnSW50ZWdlclwiIHx8XG4gICAgICAgICAgICAgICAgaWRQcm9wPy5jb25lPy5maXh0dXJlU3RyYXRlZ3kgPT09IFwic2VxdWVuY2VcIjtcblxuICAgICAgICAgICAgICBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgICAgICAgICFpc1Rlc3QoKSAmJlxuICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICAgIGNoYWxrLmdyYXkoXG4gICAgICAgICAgICAgICAgICAgICAgYFNraXBwZWQgc2VxdWVuY2UgcmVzZXQgZm9yICR7dGFibGVOYW1lfSAoaWQgdHlwZTogJHtpZFR5cGUgfHwgXCJ1bmtub3duXCJ9KWAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIO2FjOydtOu4lOydmCDstZzrjIAgSUQg7KGw7ZqMIChzdHJpbmcg7YOA7J6F7J2AIOyIq+yekCDsupDsiqTtjIUg7ZWE7JqUKVxuICAgICAgICAgICAgY29uc3QgZW50aXR5MiA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsRW50aXRpZXMoKS5maW5kKFxuICAgICAgICAgICAgICAoZSkgPT4gZS50YWJsZSA9PT0gdGFibGVOYW1lIHx8IGUuaWQudG9Mb3dlckNhc2UoKSA9PT0gdGFibGVOYW1lLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIGNvbnN0IGlkVHlwZTIgPSBlbnRpdHkyPy5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IFwiaWRcIik/LnR5cGU7XG4gICAgICAgICAgICBjb25zdCBtYXhJZFJlc3VsdCA9XG4gICAgICAgICAgICAgIGlkVHlwZTIgPT09IFwic3RyaW5nXCJcbiAgICAgICAgICAgICAgICA/IGF3YWl0IHRyeFxuICAgICAgICAgICAgICAgICAgICAucmF3KGBTRUxFQ1QgTUFYKGlkOjpiaWdpbnQpIGFzIG1heF9pZCBGUk9NIFwiJHt0YWJsZU5hbWV9XCJgKVxuICAgICAgICAgICAgICAgICAgICAudGhlbigocikgPT4gci5yb3dzWzBdKVxuICAgICAgICAgICAgICAgIDogYXdhaXQgdHJ4KHRhYmxlTmFtZSkubWF4KFwiaWQgYXMgbWF4X2lkXCIpLmZpcnN0KCk7XG4gICAgICAgICAgICBjb25zdCBtYXhJZCA9IG1heElkUmVzdWx0Py5tYXhfaWQ7XG5cbiAgICAgICAgICAgIGlmIChtYXhJZCAhPT0gbnVsbCAmJiBtYXhJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIOyLnO2AgOyKpOuqheydhCBwZ19nZXRfc2VyaWFsX3NlcXVlbmNl66GcIOyViOyghO2VmOqyjCDsobDtmoxcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4LnJhdyhgU0VMRUNUIHNldHZhbChwZ19nZXRfc2VyaWFsX3NlcXVlbmNlKD8sICdpZCcpLCA/KWAsIFtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgbWF4SWQsXG4gICAgICAgICAgICAgIF0pO1xuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiYgY29uc29sZS5sb2coY2hhbGsuZ3JlZW4oYFJlc2V0IHNlcXVlbmNlIGZvciAke3RhYmxlTmFtZX06ICR7bWF4SWR9YCkpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKF9lcnIpIHtcbiAgICAgICAgICAgIC8vIOyLnO2AgOyKpOqwgCDsl4bripQg7YWM7J2067iUKGpvaW4gdGFibGUg65OxKeydgCDrrLTsi5xcbiAgICAgICAgICAgICFpc1Rlc3QoKSAmJiBjb25zb2xlLmxvZyhjaGFsay5ncmF5KGBTa2lwcGVkIHNlcXVlbmNlIHJlc2V0IGZvciAke3RhYmxlTmFtZX1gKSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gNy4g6rKw6rO8IOyImOynkVxuICAgICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgZml4dHVyZXMpIHtcbiAgICAgICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcblxuICAgICAgICAgIC8vIOyKpO2CteuQnCBmaXh0dXJl64qUIOq4sOyhtCDroIjsvZTrk5wg7KCV67O066GcIOqysOqzvCDstpTqsIBcbiAgICAgICAgICBjb25zdCBza2lwcGVkID0gdGhpcy5za2lwcGVkRml4dHVyZXMuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAoc2tpcHBlZCkge1xuICAgICAgICAgICAgcmVzdWx0cy5wdXNoKHtcbiAgICAgICAgICAgICAgZW50aXR5SWQ6IGZpeHR1cmUuZW50aXR5SWQsXG4gICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgc2tpcHBlZC5leGlzdGluZ0lkKS5maXJzdCgpLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCByZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgICAgICBpZiAocmVmKSB7XG4gICAgICAgICAgICBjb25zdCB1dWlkVG9JZCA9IGluc2VydGVkSWRzQnlUYWJsZS5nZXQoZW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgIGNvbnN0IGluc2VydGVkSWQgPSB1dWlkVG9JZD8uZ2V0KHJlZi51dWlkKTtcblxuICAgICAgICAgICAgaWYgKGluc2VydGVkSWQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICByZXN1bHRzLnB1c2goe1xuICAgICAgICAgICAgICAgIGVudGl0eUlkOiBmaXh0dXJlLmVudGl0eUlkLFxuICAgICAgICAgICAgICAgIGRhdGE6IGF3YWl0IHRyeChlbnRpdHkudGFibGUpLndoZXJlKFwiaWRcIiwgaW5zZXJ0ZWRJZCkuZmlyc3QoKSxcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgIWlzVGVzdCgpICYmXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICAgICAgICBjaGFsay5ncmVlbihgSW5zZXJ0ZWQgaW50byAke2VudGl0eS50YWJsZX06ICMke2ZpeHR1cmUuaWR9IC0+ICMke2luc2VydGVkSWR9YCksXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0gZmluYWxseSB7XG4gICAgICBhd2FpdCBkYi5kZXN0cm95KCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuaXF1ZShyZXN1bHRzLCAocikgPT4gYCR7ci5lbnRpdHlJZH0jJHtyLmRhdGEuaWR9YCk7XG4gIH1cblxuICAvKipcbiAgICogRml4dHVyZVJlY29yZOulvCBVcHNlcnRCdWlsZGVy7JeQIOuTseuhnVxuICAgKiBAcGFyYW0gaW5zZXJ0ZWRJZHNCeVRhYmxlIOydtOuvuCB1cHNlcnTrkJwg7YWM7J2067iU7J2YIHV1aWTihpJpZCDrp6TtlZEgKOugiOuyqOuzhCDsspjrpqwg7IucIOyCrOyaqSlcbiAgICovXG4gIHByaXZhdGUgcmVnaXN0ZXJGaXh0dXJlKFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlPzogTWFwPHN0cmluZywgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPj4sXG4gICk6IFVCUmVmIHtcbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICBjb25zdCByb3c6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICAvLyBPdmVycmlkZSDrqqjrk5wg7YyQ64uoOiB0YXJnZXQg65iQ64qUIHVuaXF1ZeqwgCDsnojqs6Agb3ZlcnJpZGU9dHJ1ZeyduCDqsr3smrBcbiAgICBjb25zdCBleGlzdGluZ1JlY29yZCA9IGZpeHR1cmUudGFyZ2V0ID8/IGZpeHR1cmUudW5pcXVlO1xuICAgIGNvbnN0IGlzT3ZlcnJpZGVNb2RlID0gZml4dHVyZS5vdmVycmlkZSAmJiBleGlzdGluZ1JlY29yZDtcblxuICAgIGZvciAoY29uc3QgW3Byb3BOYW1lLCBjb2x1bW5dIG9mIE9iamVjdC5lbnRyaWVzKGZpeHR1cmUuY29sdW1ucykpIHtcbiAgICAgIGNvbnN0IHByb3AgPSBjb2x1bW4ucHJvcDtcblxuICAgICAgaWYgKGlzVmlydHVhbFByb3AocHJvcCkpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIEdlbmVyYXRlZCBjb2x1bW7snYAgSU5TRVJU7JeQ7IScIOygnOyZuCAoRELqsIAg7J6Q64+ZIOyDneyEsSlcbiAgICAgIGlmIChcImdlbmVyYXRlZFwiIGluIHByb3AgJiYgcHJvcC5nZW5lcmF0ZWQpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIC8vIGlkIOyymOumrFxuICAgICAgaWYgKHByb3BOYW1lID09PSBcImlkXCIpIHtcbiAgICAgICAgY29uc3QgaWRQcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gXCJpZFwiKTtcbiAgICAgICAgLy8gcGFyZW50SWQg7JeU7Yuw7Yuw7J2YIGlk64qUIOu2gOuqqCDsl5Tti7Dti7DsnZggRkvsnbTrr4DroZwg7Iuc7YCA7IqkIOuvuOyCrOyaqSAo66qF7Iuc7KCB7Jy866GcIO2PrO2VqClcbiAgICAgICAgY29uc3QgdXNlc1NlcXVlbmNlID1cbiAgICAgICAgICAhZW50aXR5LnBhcmVudElkICYmXG4gICAgICAgICAgKGlkUHJvcD8udHlwZSA9PT0gXCJpbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8udHlwZSA9PT0gXCJiaWdJbnRlZ2VyXCIgfHxcbiAgICAgICAgICAgIGlkUHJvcD8uY29uZT8uZml4dHVyZVN0cmF0ZWd5ID09PSBcInNlcXVlbmNlXCIpO1xuXG4gICAgICAgIGlmIChpc092ZXJyaWRlTW9kZSAmJiBleGlzdGluZ1JlY29yZCkge1xuICAgICAgICAgIC8vIE92ZXJyaWRlOiDquLDsobQg66CI7L2U65Oc7J2YIOqwkiDsgqzsmqkg4oaSIFVQREFURVxuICAgICAgICAgIHJvd1twcm9wTmFtZV0gPSBleGlzdGluZ1JlY29yZC5jb2x1bW5zW3Byb3BOYW1lXT8udmFsdWU7XG4gICAgICAgIH0gZWxzZSBpZiAoIXVzZXNTZXF1ZW5jZSkge1xuICAgICAgICAgIC8vIHN0cmluZyBQSyDrmJDripQgcGFyZW50SWQg7JeU7Yuw7YuwIEZLOiDsg53shLHrkJwgaWQg6rCS7J2EIElOU0VSVOyXkCDtj6ztlahcbiAgICAgICAgICByb3dbcHJvcE5hbWVdID0gY29sdW1uLnZhbHVlO1xuICAgICAgICB9XG4gICAgICAgIC8vIGludGVnZXIvYmlnSW50ZWdlciBQSzogREIg7Iuc7YCA7Iqk7JeQIOunoeq5gCAo6rCSIOygnOyZuClcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHxcbiAgICAgICAgICAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSAmJiBwcm9wLmhhc0pvaW5Db2x1bW4pXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZCA9IGNvbHVtbi52YWx1ZSBhcyBudW1iZXIgfCBudWxsO1xuICAgICAgICAgIGlmIChyZWxhdGVkSWQgIT09IG51bGwgJiYgcmVsYXRlZElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRGaXh0dXJlSWQgPSBgJHtwcm9wLndpdGh9IyR7cmVsYXRlZElkfWA7XG5cbiAgICAgICAgICAgIC8vIOuovOyggCBza2lw65CcIGZpeHR1cmXsnbjsp4Ag7ZmV7J24XG4gICAgICAgICAgICBjb25zdCBza2lwcGVkRXhpc3RpbmdJZCA9IHRoaXMuc2tpcHBlZEZpeHR1cmVzLmdldChyZWxhdGVkRml4dHVyZUlkKT8uZXhpc3RpbmdJZDtcbiAgICAgICAgICAgIGlmIChza2lwcGVkRXhpc3RpbmdJZCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICAgIC8vIHNraXDrkJwgZml4dHVyZSDihpIgdGFyZ2V0IERC7J2YIOq4sOyhtCDroIjsvZTrk5wgaWQg7IKs7JqpXG4gICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHNraXBwZWRFeGlzdGluZ0lkO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQocmVsYXRlZEZpeHR1cmVJZCk7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGVkUmVmKSB7XG4gICAgICAgICAgICAgICAgLy8g7J2066+4IHVwc2VydOuQnCDqsJnsnYAg7YWM7J2067iUIGZpeHR1cmUg7ZmV7J24XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZEluc2VydGVkSWRzID0gaW5zZXJ0ZWRJZHNCeVRhYmxlPy5nZXQocmVsYXRlZEVudGl0eS50YWJsZSk7XG4gICAgICAgICAgICAgICAgY29uc3QgYWN0dWFsSWQgPSByZWxhdGVkSW5zZXJ0ZWRJZHM/LmdldChyZWxhdGVkUmVmLnV1aWQpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGFjdHVhbElkICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIC8vIOydtOuvuCB1cHNlcnTrkKgg4oaSIOyLpOygnCBJRCDsgqzsmqlcbiAgICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IGFjdHVhbElkO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAvLyDslYTsp4EgdXBzZXJ0IOyViOuQqCDihpIgVUJSZWYg7IKs7JqpXG4gICAgICAgICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSByZWxhdGVkUmVmO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAvLyBmaXh0dXJlc+yXkCDtj6ztlajrkJjsp4Ag7JWK7J2AIOugiOy9lOuTnCDihpIgSUQg6re464yA66GcIOyCrOyaqVxuICAgICAgICAgICAgICAgIHJvd1tgJHtwcm9wTmFtZX1faWRgXSA9IHJlbGF0ZWRJZDtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByb3dbYCR7cHJvcE5hbWV9X2lkYF0gPSBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAvLyBIYXNNYW55LCBNYW55VG9NYW5564qUIOuzhOuPhCDsspjrpqxcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIOydvOuwmCDsu6zrn7xcbiAgICAgICAgcm93W3Byb3BOYW1lXSA9IHRoaXMuY29udmVydENvbHVtblZhbHVlKHByb3AgYXMgRW50aXR5UHJvcCwgY29sdW1uLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAhaXNUZXN0KCkgJiZcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoYFJlZ2lzdGVyaW5nICR7ZW50aXR5LnRhYmxlfSAtICR7aW5zcGVjdChyb3csIGZhbHNlLCBudWxsLCB0cnVlKX1gKSk7XG4gICAgY29uc3QgcmVmID0gdGhpcy5idWlsZGVyLnJlZ2lzdGVyKGVudGl0eS50YWJsZSwgcm93KTtcbiAgICB0aGlzLmZpeHR1cmVSZWZNYXAuc2V0KGZpeHR1cmUuZml4dHVyZUlkLCByZWYpO1xuXG4gICAgcmV0dXJuIHJlZjtcbiAgfVxuXG4gIC8qKlxuICAgKiDsu6zrn7wg6rCSIOuzgO2ZmFxuICAgKi9cbiAgcHJpdmF0ZSBjb252ZXJ0Q29sdW1uVmFsdWUocHJvcDogRW50aXR5UHJvcCwgdmFsdWU6IHVua25vd24pOiB1bmtub3duIHtcbiAgICBpZiAodmFsdWUgPT09IG51bGwgfHwgdmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgc3dpdGNoIChwcm9wLnR5cGUpIHtcbiAgICAgIGNhc2UgXCJqc29uXCI6XG4gICAgICAgIC8vIFVwc2VydEJ1aWxkZXIucmVnaXN0ZXLsl5DshJwgSlNPTi5zdHJpbmdpZnkg7LKY66as7ZWY66+A66GcIG9iamVjdCDqt7jrjIDroZwg7KCE64usXG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgY2FzZSBcImRhdGVcIjpcbiAgICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlb2YgdmFsdWUgPT09IFwibnVtYmVyXCIpIHtcbiAgICAgICAgICByZXR1cm4gbmV3IERhdGUodmFsdWUpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB2YWx1ZTtcblxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgcHJvY2Vzc01hbnlUb01hbnlSZWxhdGlvbnMoXG4gICAgdHJ4OiBLbmV4LlRyYW5zYWN0aW9uLFxuICAgIGZpeHR1cmVzOiBGaXh0dXJlUmVjb3JkW10sXG4gICAgaW5zZXJ0ZWRJZHNCeVRhYmxlOiBNYXA8c3RyaW5nLCBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+PixcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgZm9yIChjb25zdCBmaXh0dXJlIG9mIGZpeHR1cmVzKSB7XG4gICAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlLmVudGl0eUlkKTtcbiAgICAgIGNvbnN0IHNvdXJjZVJlZiA9IHRoaXMuZml4dHVyZVJlZk1hcC5nZXQoZml4dHVyZS5maXh0dXJlSWQpO1xuXG4gICAgICBpZiAoIXNvdXJjZVJlZikgY29udGludWU7XG5cbiAgICAgIGNvbnN0IHNvdXJjZVV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChlbnRpdHkudGFibGUpO1xuICAgICAgY29uc3Qgc291cmNlSWQgPSBzb3VyY2VVdWlkVG9JZD8uZ2V0KHNvdXJjZVJlZi51dWlkKTtcblxuICAgICAgaWYgKHNvdXJjZUlkID09PSB1bmRlZmluZWQpIGNvbnRpbnVlO1xuXG4gICAgICBmb3IgKGNvbnN0IFssIGNvbHVtbl0gb2YgT2JqZWN0LmVudHJpZXMoZml4dHVyZS5jb2x1bW5zKSkge1xuICAgICAgICBjb25zdCBwcm9wID0gY29sdW1uLnByb3A7XG5cbiAgICAgICAgaWYgKGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChwcm9wKSAmJiBBcnJheS5pc0FycmF5KGNvbHVtbi52YWx1ZSkpIHtcbiAgICAgICAgICAvLyDshKDtg53rkJjsp4Ag7JWK7J2AIE1hbnlUb01hbnkg6rSA6rOE64qUIOyggOyepe2VmOyngCDslYrsnYxcbiAgICAgICAgICBjb25zdCB0YXJnZXRUYWJsZSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgaWYgKCF0aGlzLmJ1aWxkZXIuaGFzVGFibGUodGFyZ2V0VGFibGUudGFibGUpKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRJZHMgPSBjb2x1bW4udmFsdWUgYXMgbnVtYmVyW107XG4gICAgICAgICAgaWYgKHJlbGF0ZWRJZHMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgIGNvbnN0IGpvaW5UYWJsZSA9IHByb3Auam9pblRhYmxlO1xuICAgICAgICAgIGNvbnN0IHJlbGF0ZWRFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChwcm9wLndpdGgpO1xuXG4gICAgICAgICAgY29uc3Qgc291cmNlQ29sdW1uID0gYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShlbnRpdHkudGFibGUpfV9pZGA7XG4gICAgICAgICAgY29uc3QgdGFyZ2V0Q29sdW1uID0gYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShyZWxhdGVkRW50aXR5LnRhYmxlKX1faWRgO1xuXG4gICAgICAgICAgZm9yIChjb25zdCByZWxhdGVkSWQgb2YgcmVsYXRlZElkcykge1xuICAgICAgICAgICAgY29uc3QgcmVsYXRlZEZpeHR1cmVJZCA9IGAke3Byb3Aud2l0aH0jJHtyZWxhdGVkSWR9YDtcbiAgICAgICAgICAgIGNvbnN0IHJlbGF0ZWRSZWYgPSB0aGlzLmZpeHR1cmVSZWZNYXAuZ2V0KHJlbGF0ZWRGaXh0dXJlSWQpO1xuXG4gICAgICAgICAgICBsZXQgdGFyZ2V0SWQ6IG51bWJlciB8IHN0cmluZztcblxuICAgICAgICAgICAgaWYgKHJlbGF0ZWRSZWYpIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVsYXRlZFV1aWRUb0lkID0gaW5zZXJ0ZWRJZHNCeVRhYmxlLmdldChyZWxhdGVkRW50aXR5LnRhYmxlKTtcbiAgICAgICAgICAgICAgY29uc3QgcmVzb2x2ZWRJZCA9IHJlbGF0ZWRVdWlkVG9JZD8uZ2V0KHJlbGF0ZWRSZWYudXVpZCk7XG5cbiAgICAgICAgICAgICAgaWYgKHJlc29sdmVkSWQgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICAgIGNvbnNvbGUud2FybihcbiAgICAgICAgICAgICAgICAgIGBSZWxhdGVkIGZpeHR1cmUgJHtyZWxhdGVkRml4dHVyZUlkfSBub3QgZm91bmQgaW4gaW5zZXJ0ZWRJZHMsIHNraXBwaW5nYCxcbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHRhcmdldElkID0gcmVzb2x2ZWRJZDtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHRhcmdldElkID0gcmVsYXRlZElkO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAvLyBKb2luVGFibGXsl5Ag7IK97J6FXG4gICAgICAgICAgICBjb25zdCBbZm91bmRdID0gYXdhaXQgdHJ4KGpvaW5UYWJsZSlcbiAgICAgICAgICAgICAgLndoZXJlKHtcbiAgICAgICAgICAgICAgICBbc291cmNlQ29sdW1uXTogc291cmNlSWQsXG4gICAgICAgICAgICAgICAgW3RhcmdldENvbHVtbl06IHRhcmdldElkLFxuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAubGltaXQoMSk7XG5cbiAgICAgICAgICAgIGlmICghZm91bmQpIHtcbiAgICAgICAgICAgICAgYXdhaXQgdHJ4KGpvaW5UYWJsZSkuaW5zZXJ0KHtcbiAgICAgICAgICAgICAgICBbc291cmNlQ29sdW1uXTogc291cmNlSWQsXG4gICAgICAgICAgICAgICAgW3RhcmdldENvbHVtbl06IHRhcmdldElkLFxuICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICAhaXNUZXN0KCkgJiZcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgICAgIGNoYWxrLmdyZWVuKFxuICAgICAgICAgICAgICAgICAgICBgSW5zZXJ0ZWQgaW50byAke2pvaW5UYWJsZX06ICR7ZW50aXR5LnRhYmxlfSgke3NvdXJjZUlkfSkgLSAke3JlbGF0ZWRFbnRpdHkudGFibGV9KCR7dGFyZ2V0SWR9KWAsXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIOqwmeydgCDthYzsnbTruJQg64K0IGZpeHR1cmXrk6TsnYQgc2VsZi1yZWZlcmVuY2Ug66CI67Ko67OE66GcIOu2hO2VoFxuICAgKiAtIHNlbGYtcmVmZXJlbmNl6rCAIOyXhuuKlCBmaXh0dXJl65OkOiBMZXZlbCAwXG4gICAqIC0gTGV2ZWwgMOydhCDssLjsobDtlZjripQgZml4dHVyZeuTpDogTGV2ZWwgMVxuICAgKiAtIOuwmOuztVxuICAgKlxuICAgKiBVcHNlcnRCdWlsZGVy6rCAIHNlbGYtcmVmZXJlbmNl6rCAIOyeiOycvOuptCBidWlsZEluc2VydExldmVscygp66GcIOyerOygleugrO2VmOyXrFxuICAgKiDrk7HroZ0g7Iic7ISc7JmAIOuwmO2ZmCDsiJzshJzqsIAg64us65287KeIIOyImCDsnojsirXri4jri6QuXG4gICAqIOydtOulvCDrsKnsp4DtlZjquLAg7JyE7ZW0IEZpeHR1cmVNYW5hZ2Vy6rCAIOugiOuyqOuzhOuhnCDrgpjriKDshJwg7LKY66as7ZWp64uI64ukLlxuICAgKi9cbiAgcHJpdmF0ZSBncm91cEZpeHR1cmVzQnlMZXZlbChmaXh0dXJlczogRml4dHVyZVJlY29yZFtdKTogRml4dHVyZVJlY29yZFtdW10ge1xuICAgIGlmIChmaXh0dXJlcy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG5cbiAgICBjb25zdCBlbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChmaXh0dXJlc1swXS5lbnRpdHlJZCk7XG5cbiAgICAvLyBzZWxmLXJlZmVyZW5jZSByZWxhdGlvbiBwcm9wIOywvuq4sFxuICAgIGNvbnN0IHNlbGZSZWZQcm9wcyA9IGVudGl0eS5wcm9wcy5maWx0ZXIoXG4gICAgICAocCk6IHAgaXMgQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIHwgT25lVG9PbmVSZWxhdGlvblByb3AgPT5cbiAgICAgICAgaXNSZWxhdGlvblByb3AocCkgJiZcbiAgICAgICAgKGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHApIHx8IChpc09uZVRvT25lUmVsYXRpb25Qcm9wKHApICYmIHAuaGFzSm9pbkNvbHVtbikpICYmXG4gICAgICAgIHAud2l0aCA9PT0gZW50aXR5LmlkLFxuICAgICk7XG5cbiAgICBpZiAoc2VsZlJlZlByb3BzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgLy8gc2VsZi1yZWZlcmVuY2Ug7JeG7J2MIOKGkiDri6jsnbwg66CI67KoXG4gICAgICByZXR1cm4gW2ZpeHR1cmVzXTtcbiAgICB9XG5cbiAgICAvLyDroIjrsqjrs4Qg67aE7ZWgICh0b3BvbG9naWNhbCBzb3J0KVxuICAgIGNvbnN0IGxldmVsczogRml4dHVyZVJlY29yZFtdW10gPSBbXTtcbiAgICBjb25zdCByZW1haW5pbmcgPSBuZXcgU2V0KGZpeHR1cmVzLm1hcCgoZikgPT4gZi5maXh0dXJlSWQpKTtcbiAgICBjb25zdCBwcm9jZXNzZWQgPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgIHdoaWxlIChyZW1haW5pbmcuc2l6ZSA+IDApIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRMZXZlbDogRml4dHVyZVJlY29yZFtdID0gW107XG5cbiAgICAgIGZvciAoY29uc3QgZml4dHVyZSBvZiBmaXh0dXJlcykge1xuICAgICAgICBpZiAoIXJlbWFpbmluZy5oYXMoZml4dHVyZS5maXh0dXJlSWQpKSBjb250aW51ZTtcblxuICAgICAgICAvLyBzZWxmLXJlZmVyZW5jZeqwgCDrqqjrkZAg7J2066+4IOyymOumrOuQkOqxsOuCmCBudWxs7J24IOqyveyasFxuICAgICAgICBjb25zdCBjYW5Qcm9jZXNzID0gc2VsZlJlZlByb3BzLmV2ZXJ5KChwcm9wKSA9PiB7XG4gICAgICAgICAgY29uc3QgcmVmSWQgPSBmaXh0dXJlLmNvbHVtbnNbcHJvcC5uYW1lXT8udmFsdWUgYXMgbnVtYmVyIHwgbnVsbDtcbiAgICAgICAgICBpZiAocmVmSWQgPT09IG51bGwgfHwgcmVmSWQgPT09IHVuZGVmaW5lZCkgcmV0dXJuIHRydWU7XG4gICAgICAgICAgY29uc3QgcmVmRml4dHVyZUlkID0gYCR7cHJvcC53aXRofSMke3JlZklkfWA7XG4gICAgICAgICAgLy8g7J2066+4IOyymOumrOuQkOqxsOuCmCwg7ZiE7J6sIGZpeHR1cmVz7JeQIO2PrO2VqOuQmOyngCDslYrsnYAg6rK97JqwICjsmbjrtoAg7LC47KGwKVxuICAgICAgICAgIHJldHVybiBwcm9jZXNzZWQuaGFzKHJlZkZpeHR1cmVJZCkgfHwgIXJlbWFpbmluZy5oYXMocmVmRml4dHVyZUlkKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKGNhblByb2Nlc3MpIHtcbiAgICAgICAgICBjdXJyZW50TGV2ZWwucHVzaChmaXh0dXJlKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAoY3VycmVudExldmVsLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICBjb25zdCByZW1haW5pbmdJZHMgPSBBcnJheS5mcm9tKHJlbWFpbmluZykuam9pbihcIiwgXCIpO1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgYENpcmN1bGFyIHNlbGYtcmVmZXJlbmNlIGRldGVjdGVkIGluICR7ZW50aXR5LnRhYmxlfS4gUmVtYWluaW5nIGZpeHR1cmVzOiAke3JlbWFpbmluZ0lkc31gLFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBmb3IgKGNvbnN0IGZpeHR1cmUgb2YgY3VycmVudExldmVsKSB7XG4gICAgICAgIHJlbWFpbmluZy5kZWxldGUoZml4dHVyZS5maXh0dXJlSWQpO1xuICAgICAgICBwcm9jZXNzZWQuYWRkKGZpeHR1cmUuZml4dHVyZUlkKTtcbiAgICAgIH1cblxuICAgICAgbGV2ZWxzLnB1c2goY3VycmVudExldmVsKTtcbiAgICB9XG5cbiAgICByZXR1cm4gbGV2ZWxzO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBjaGVja1VuaXF1ZVZpb2xhdGlvbihkYjogS25leCwgZW50aXR5OiBFbnRpdHksIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQpIHtcbiAgICBjb25zdCBfdW5pcXVlSW5kZXhlcyA9IGVudGl0eS5pbmRleGVzPy5maWx0ZXIoKGkpID0+IGkudHlwZSA9PT0gXCJ1bmlxdWVcIikgPz8gW107XG5cbiAgICBjb25zdCB1bmlxdWVJbmRleGVzID0gX3VuaXF1ZUluZGV4ZXMuZmlsdGVyKChpbmRleCkgPT5cbiAgICAgIGluZGV4LmNvbHVtbnMuZXZlcnkoKGNvbHVtbikgPT4gIWNvbHVtbi5uYW1lLnN0YXJ0c1dpdGgoYCR7ZW50aXR5LnRhYmxlfV9fYCkpLFxuICAgICk7XG4gICAgaWYgKHVuaXF1ZUluZGV4ZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBsZXQgdW5pcXVlUXVlcnkgPSBkYihlbnRpdHkudGFibGUpO1xuICAgIGxldCBoYXNDb25kaXRpb24gPSBmYWxzZTtcblxuICAgIGZvciAoY29uc3QgaW5kZXggb2YgdW5pcXVlSW5kZXhlcykge1xuICAgICAgLy8g7Lus65+8IOykkSDtlZjrgpjrnbzrj4QgbnVsbOydtOuptCDsnKDri4jtgawg7KCc7JW97J2EIOychOuwmO2VmOyngCDslYrquLAg65WM66y47JeQIO2VtOuLuSDsnbjrjbHsiqTripQg66y07IucXG4gICAgICBjb25zdCBjb250YWluc051bGwgPSBpbmRleC5jb2x1bW5zLnNvbWUoKGNvbHVtbikgPT4ge1xuICAgICAgICBjb25zdCBmaWVsZCA9IGNvbHVtbi5uYW1lLnJlcGxhY2UoL19pZCQvLCBcIlwiKTtcbiAgICAgICAgcmV0dXJuIGZpeHR1cmUuY29sdW1uc1tmaWVsZF0/LnZhbHVlID09PSBudWxsO1xuICAgICAgfSk7XG4gICAgICBpZiAoY29udGFpbnNOdWxsKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICB1bmlxdWVRdWVyeSA9IHVuaXF1ZVF1ZXJ5Lm9yV2hlcmUoKHFiKSA9PiB7XG4gICAgICAgIGZvciAoY29uc3QgY29sdW1uIG9mIGluZGV4LmNvbHVtbnMpIHtcbiAgICAgICAgICBjb25zdCBmaWVsZCA9IGNvbHVtbi5uYW1lLnJlcGxhY2UoL19pZCQvLCBcIlwiKTtcblxuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KGZpeHR1cmUuY29sdW1uc1tmaWVsZF0/LnZhbHVlKSkge1xuICAgICAgICAgICAgcWIud2hlcmVJbihjb2x1bW4ubmFtZSwgZml4dHVyZS5jb2x1bW5zW2ZpZWxkXS52YWx1ZSk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHFiLmFuZFdoZXJlKGNvbHVtbi5uYW1lLCBmaXh0dXJlLmNvbHVtbnNbZmllbGRdPy52YWx1ZSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KTtcbiAgICAgIGhhc0NvbmRpdGlvbiA9IHRydWU7XG4gICAgfVxuXG4gICAgaWYgKCFoYXNDb25kaXRpb24pIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGNvbnN0IFt1bmlxdWVGb3VuZF0gPSBhd2FpdCB1bmlxdWVRdWVyeTtcbiAgICByZXR1cm4gdW5pcXVlRm91bmQ7XG4gIH1cblxuICBwcml2YXRlIGFzeW5jIGNoZWNrRHVwbGljYXRlQnlDb2x1bW5zKFxuICAgIGRiOiBLbmV4LFxuICAgIGVudGl0eTogRW50aXR5LFxuICAgIGZpeHR1cmU6IEZpeHR1cmVSZWNvcmQsXG4gICAgY29sdW1uczogc3RyaW5nW10sXG4gICkge1xuICAgIGlmIChjb2x1bW5zLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgY29uc3Qgd2hlcmVDbGF1c2U6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG5cbiAgICBmb3IgKGNvbnN0IGNvbHVtbiBvZiBjb2x1bW5zKSB7XG4gICAgICAvLyByZWxhdGlvbiDtlYTrk5zsnbgg6rK97JqwIF9pZCDrtpnsnbTquLBcbiAgICAgIGNvbnN0IHByb3AgPSBlbnRpdHkucHJvcHMuZmluZCgocCkgPT4gcC5uYW1lID09PSBjb2x1bW4pO1xuICAgICAgY29uc3QgZGJDb2x1bW4gPSBwcm9wICYmIGlzUmVsYXRpb25Qcm9wKHByb3ApID8gYCR7Y29sdW1ufV9pZGAgOiBjb2x1bW47XG4gICAgICBjb25zdCB2YWx1ZSA9IGZpeHR1cmUuY29sdW1uc1tjb2x1bW5dPy52YWx1ZTtcblxuICAgICAgLy8gbnVsbCDqsJLsnbQg7Y+s7ZWo65CcIOqyveyasCDspJHrs7Ug7ZmV7J24IOyKpO2CtVxuICAgICAgaWYgKHZhbHVlID09PSBudWxsIHx8IHZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG5cbiAgICAgIHdoZXJlQ2xhdXNlW2RiQ29sdW1uXSA9IHZhbHVlO1xuICAgIH1cblxuICAgIGNvbnN0IFtmb3VuZF0gPSBhd2FpdCBkYihlbnRpdHkudGFibGUpLndoZXJlKHdoZXJlQ2xhdXNlKS5saW1pdCgxKTtcbiAgICByZXR1cm4gZm91bmQ7XG4gIH1cblxuICBhc3luYyBhZGRGaXh0dXJlTG9hZGVyKGNvZGU6IHN0cmluZykge1xuICAgIGNvbnN0IHBhdGggPSBgJHtTb25hbXUuYXBpUm9vdFBhdGh9L3NyYy90ZXN0aW5nL2ZpeHR1cmUudHNgO1xuICAgIGNvbnN0IGNvbnRlbnQgPSByZWFkRmlsZVN5bmMocGF0aCkudG9TdHJpbmcoKTtcblxuICAgIGNvbnN0IGZpeHR1cmVMb2FkZXJTdGFydCA9IGNvbnRlbnQuaW5kZXhPZihcImNvbnN0IGZpeHR1cmVMb2FkZXIgPSB7XCIpO1xuICAgIGNvbnN0IGZpeHR1cmVMb2FkZXJFbmQgPSBjb250ZW50LmluZGV4T2YoXCJ9O1wiLCBmaXh0dXJlTG9hZGVyU3RhcnQpO1xuXG4gICAgaWYgKGZpeHR1cmVMb2FkZXJTdGFydCAhPT0gLTEgJiYgZml4dHVyZUxvYWRlckVuZCAhPT0gLTEpIHtcbiAgICAgIGNvbnN0IG5ld0NvbnRlbnQgPSBgJHtjb250ZW50LnNsaWNlKDAsIGZpeHR1cmVMb2FkZXJFbmQpfSAgJHtjb2RlfVxcbiR7Y29udGVudC5zbGljZShmaXh0dXJlTG9hZGVyRW5kKX1gO1xuXG4gICAgICB3cml0ZUZpbGVTeW5jKHBhdGgsIG5ld0NvbnRlbnQpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGYWlsZWQgdG8gZmluZCBmaXh0dXJlTG9hZGVyIGluIGZpeHR1cmUudHNcIik7XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBjb25zdCBGaXh0dXJlTWFuYWdlciA9IG5ldyBGaXh0dXJlTWFuYWdlckNsYXNzKCk7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O2NBVXVDO2tCQUNZO1lBRUc7c0JBQ0s7c0JBR0Y7YUFRakM7a0JBVXFCO3VCQUNLO0NBU3JDLHNCQUFiLE1BQWlDO0VBQy9CLEFBQVEsT0FBb0I7RUFDNUIsSUFBSSxJQUFJLEtBQVc7QUFDakIsUUFBSyxPQUFPOztFQUVkLElBQUksTUFBWTtBQUNkLE9BQUksS0FBSyxTQUFTLE1BQU07QUFDdEIsVUFBTSxJQUFJLE1BQU0sMENBQTBDOztBQUU1RCxVQUFPLEtBQUs7O0VBR2QsQUFBUSxPQUFvQjtFQUM1QixJQUFJLElBQUksS0FBVztBQUNqQixRQUFLLE9BQU87O0VBRWQsSUFBSSxNQUFZO0FBQ2QsT0FBSSxLQUFLLFNBQVMsTUFBTTtBQUN0QixVQUFNLElBQUksTUFBTSwwQ0FBMEM7O0FBRTVELFVBQU8sS0FBSzs7RUFFZCxtQkFBb0M7RUFFcEMsQUFBUSxnQkFBZ0IsSUFBSSxlQUFlO0VBRzNDLEFBQVEsVUFBeUIsSUFBSSxlQUFlO0VBQ3BELEFBQVEsZ0JBQW9DLElBQUksS0FBSztFQUNyRCxBQUFRLGtCQUNOLElBQUksS0FBSztFQUVYLE9BQU87QUFDTCxPQUFJLEtBQUssU0FBUyxNQUFNO0FBQ3RCOztBQUVGLE9BQUksT0FBTyxTQUFTLFFBQVEsT0FBTyxTQUFTLG1CQUFt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
|