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,469 +1,409 @@
|
|
|
1
|
+
import { __esmMin } from "../_virtual/rolldown_runtime.js";
|
|
2
|
+
import { assertDefined, init_utils, nonNullable } from "../utils/utils.js";
|
|
3
|
+
import { EntityManager, init_entity_manager } from "../entity/entity-manager.js";
|
|
4
|
+
import { Naite, init_naite } from "../naite/naite.js";
|
|
5
|
+
import { batchUpdate, init__batch_update } from "./_batch_update.js";
|
|
1
6
|
import { getLogger } from "@logtape/logtape";
|
|
7
|
+
import { cluster, isArray, unique } from "radashi";
|
|
2
8
|
import { randomUUID } from "crypto";
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
import { batchUpdate } from "./_batch_update.js";
|
|
8
|
-
const logger = getLogger([
|
|
9
|
-
"sonamu",
|
|
10
|
-
"internal",
|
|
11
|
-
"upsert-builder"
|
|
12
|
-
]);
|
|
13
|
-
export function isRefField(field) {
|
|
14
|
-
return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
|
|
15
|
-
}
|
|
16
|
-
export class UpsertBuilder {
|
|
17
|
-
tables;
|
|
18
|
-
constructor(){
|
|
19
|
-
this.tables = new Map();
|
|
20
|
-
}
|
|
21
|
-
getTable(tableName) {
|
|
22
|
-
const table = this.tables.get(tableName);
|
|
23
|
-
if (table) {
|
|
24
|
-
return table;
|
|
25
|
-
}
|
|
26
|
-
const tableSpec = (()=>{
|
|
27
|
-
try {
|
|
28
|
-
return EntityManager.getTableSpec(tableName);
|
|
29
|
-
} catch {
|
|
30
|
-
return null;
|
|
31
|
-
}
|
|
32
|
-
})();
|
|
33
|
-
const tableData = {
|
|
34
|
-
references: new Set(),
|
|
35
|
-
rows: [],
|
|
36
|
-
uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
|
|
37
|
-
uniquesMap: new Map(),
|
|
38
|
-
jsonColumns: tableSpec?.jsonColumns ?? []
|
|
39
|
-
};
|
|
40
|
-
this.tables.set(tableName, tableData);
|
|
41
|
-
return tableData;
|
|
42
|
-
}
|
|
43
|
-
hasTable(tableName) {
|
|
44
|
-
return this.tables.has(tableName);
|
|
45
|
-
}
|
|
46
|
-
register(tableName, row) {
|
|
47
|
-
const table = this.getTable(tableName);
|
|
48
|
-
// 해당 테이블의 unique 인덱스를 순회하며 키 생성
|
|
49
|
-
const uniqueKeys = table.uniqueIndexes.map((unqIndex)=>{
|
|
50
|
-
const uniqueKeyArray = unqIndex.columns.map((unqCol)=>{
|
|
51
|
-
const val = row[unqCol.name];
|
|
52
|
-
if (isRefField(val)) {
|
|
53
|
-
return val.uuid;
|
|
54
|
-
} else {
|
|
55
|
-
return row[unqCol.name] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입
|
|
56
|
-
}
|
|
57
|
-
});
|
|
58
|
-
// 값이 모두 null인 경우 키 생성 패스
|
|
59
|
-
if (uniqueKeyArray.length === 0) {
|
|
60
|
-
return null;
|
|
61
|
-
}
|
|
62
|
-
return uniqueKeyArray.join("---delimiter--");
|
|
63
|
-
}).filter(nonNullable);
|
|
64
|
-
// uuid 생성 로직
|
|
65
|
-
const { uuid, isReused } = (()=>{
|
|
66
|
-
// 키를 순회하여 이미 존재하는 키가 있는지 확인
|
|
67
|
-
if (uniqueKeys.length > 0) {
|
|
68
|
-
for (const uniqueKey of uniqueKeys){
|
|
69
|
-
if (table.uniquesMap.has(uniqueKey)) {
|
|
70
|
-
return {
|
|
71
|
-
uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
|
|
72
|
-
isReused: true
|
|
73
|
-
};
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
// 찾을 수 없는 경우 생성
|
|
78
|
-
return {
|
|
79
|
-
uuid: randomUUID(),
|
|
80
|
-
isReused: false
|
|
81
|
-
};
|
|
82
|
-
})();
|
|
83
|
-
// 모든 유니크키에 대해 유니크맵에 uuid 저장
|
|
84
|
-
if (uniqueKeys.length > 0) {
|
|
85
|
-
for (const uniqueKey of uniqueKeys){
|
|
86
|
-
table.uniquesMap.set(uniqueKey, uuid);
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
// 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가
|
|
90
|
-
// 이 정보를 나중에 치환할 때 사용
|
|
91
|
-
row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue])=>{
|
|
92
|
-
if (isRefField(rowValue)) {
|
|
93
|
-
rowValue.use ??= "id";
|
|
94
|
-
table.references.add(`${rowValue.of}.${rowValue.use}`);
|
|
95
|
-
return [
|
|
96
|
-
rowKey,
|
|
97
|
-
rowValue
|
|
98
|
-
];
|
|
99
|
-
} else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {
|
|
100
|
-
// JSON 컬럼인 경우 JSON.stringify 처리 (Knex는 JSON 타입을 지원하지 않음)
|
|
101
|
-
return [
|
|
102
|
-
rowKey,
|
|
103
|
-
JSON.stringify(rowValue)
|
|
104
|
-
];
|
|
105
|
-
} else if (isArray(rowValue)) {
|
|
106
|
-
// 배열은 그대로 저장
|
|
107
|
-
return [
|
|
108
|
-
rowKey,
|
|
109
|
-
rowValue
|
|
110
|
-
];
|
|
111
|
-
} else {
|
|
112
|
-
return [
|
|
113
|
-
rowKey,
|
|
114
|
-
rowValue
|
|
115
|
-
];
|
|
116
|
-
}
|
|
117
|
-
}));
|
|
118
|
-
table.rows.push({
|
|
119
|
-
uuid,
|
|
120
|
-
...row
|
|
121
|
-
});
|
|
122
|
-
const result = {
|
|
123
|
-
of: tableName,
|
|
124
|
-
uuid: row.uuid ?? uuid
|
|
125
|
-
};
|
|
126
|
-
Naite.t("puri:ub-register", {
|
|
127
|
-
tableName,
|
|
128
|
-
uuid: result.uuid,
|
|
129
|
-
isUuidReused: isReused,
|
|
130
|
-
row
|
|
131
|
-
});
|
|
132
|
-
return result;
|
|
133
|
-
}
|
|
134
|
-
async upsert(wdb, tableName, options) {
|
|
135
|
-
return this.upsertOrInsert(wdb, tableName, "upsert", options);
|
|
136
|
-
}
|
|
137
|
-
async insertOnly(wdb, tableName, options) {
|
|
138
|
-
return this.upsertOrInsert(wdb, tableName, "insert", options);
|
|
139
|
-
}
|
|
140
|
-
async upsertOrInsert(wdb, tableName, mode, options) {
|
|
141
|
-
if (this.hasTable(tableName) === false) {
|
|
142
|
-
return [];
|
|
143
|
-
}
|
|
144
|
-
const table = this.tables.get(tableName);
|
|
145
|
-
if (table === undefined) {
|
|
146
|
-
throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);
|
|
147
|
-
} else if (table.rows.length === 0) {
|
|
148
|
-
throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);
|
|
149
|
-
}
|
|
150
|
-
if (table.rows.some((row)=>Object.entries(row).some(([, value])=>isRefField(value) && value.of !== tableName))) {
|
|
151
|
-
throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);
|
|
152
|
-
}
|
|
153
|
-
// 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출
|
|
154
|
-
const { references, refTables } = Array.from(this.tables).reduce((r, [, table])=>{
|
|
155
|
-
const reference = Array.from(table.references.values()).find((ref)=>ref.includes(`${tableName}.`));
|
|
156
|
-
if (reference) {
|
|
157
|
-
r.references.push(reference);
|
|
158
|
-
r.refTables.push(table);
|
|
159
|
-
}
|
|
160
|
-
return r;
|
|
161
|
-
}, {
|
|
162
|
-
references: [],
|
|
163
|
-
refTables: []
|
|
164
|
-
});
|
|
165
|
-
const extractFields = unique(references).map((reference)=>reference.split(".")[1]).filter((field)=>field !== undefined);
|
|
166
|
-
// 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)
|
|
167
|
-
const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
|
|
168
|
-
if (hasCircular) {
|
|
169
|
-
throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
|
|
170
|
-
}
|
|
171
|
-
const uuidMap = new Map();
|
|
172
|
-
const allIds = [];
|
|
173
|
-
// 레벨별로 순차 처리
|
|
174
|
-
for(let levelIdx = 0; levelIdx < levels.length; levelIdx++){
|
|
175
|
-
const levelRows = levels[levelIdx];
|
|
176
|
-
logger.debug("Processing Query Level: {current} / {total}", {
|
|
177
|
-
current: levelIdx + 1,
|
|
178
|
-
total: levels.length
|
|
179
|
-
});
|
|
180
|
-
// 이전 레벨에서 얻은 ID로 자기 참조 해결
|
|
181
|
-
const resolvedRows = levelRows.map((row)=>{
|
|
182
|
-
const resolved = {
|
|
183
|
-
...row
|
|
184
|
-
};
|
|
185
|
-
for (const [key, value] of Object.entries(row)){
|
|
186
|
-
if (isRefField(value) && value.of === tableName) {
|
|
187
|
-
const parent = uuidMap.get(value.uuid);
|
|
188
|
-
if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
|
|
189
|
-
resolved[key] = parent[value.use ?? "id"];
|
|
190
|
-
Naite.t("puri:ub-ref-resolved", {
|
|
191
|
-
tableName,
|
|
192
|
-
field: key,
|
|
193
|
-
from: {
|
|
194
|
-
of: value.of,
|
|
195
|
-
uuid: value.uuid,
|
|
196
|
-
use: value.use ?? "id"
|
|
197
|
-
},
|
|
198
|
-
to: resolved[key]
|
|
199
|
-
});
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
return resolved;
|
|
203
|
-
});
|
|
204
|
-
// 현재 레벨 upsert
|
|
205
|
-
const chunkSize = options?.chunkSize;
|
|
206
|
-
const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [
|
|
207
|
-
resolvedRows
|
|
208
|
-
];
|
|
209
|
-
const selectFields = unique([
|
|
210
|
-
"id",
|
|
211
|
-
...extractFields
|
|
212
|
-
]);
|
|
213
|
-
for(let index = 0; index < levelChunks.length; index++){
|
|
214
|
-
const dataChunk = levelChunks[index];
|
|
215
|
-
if (dataChunk.length === 0) continue;
|
|
216
|
-
logger.debug("Processing Chunk: {current} / {total}", {
|
|
217
|
-
current: index + 1,
|
|
218
|
-
total: levelChunks.length
|
|
219
|
-
});
|
|
220
|
-
// uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거
|
|
221
|
-
const originalUuids = dataChunk.map((r)=>r.uuid);
|
|
222
|
-
const dataForDb = dataChunk.map(({ uuid, ...rest })=>rest);
|
|
223
|
-
let resultRows;
|
|
224
|
-
if (mode === "insert") {
|
|
225
|
-
// INSERT 모드 - RETURNING 사용
|
|
226
|
-
resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
|
|
227
|
-
} else {
|
|
228
|
-
// UPSERT 모드 - id 없는 row들의 id를 사전 조회로 채우기
|
|
229
|
-
const rowsWithoutId = dataForDb.filter((row)=>!row.id);
|
|
230
|
-
if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {
|
|
231
|
-
// 모든 uniqueIndexes로 기존 레코드 조회
|
|
232
|
-
for (const uniqueIndex of table.uniqueIndexes){
|
|
233
|
-
const columns = uniqueIndex.columns.map((c)=>c.name);
|
|
234
|
-
// 조회할 조건들 추출 (각 row의 unique 컬럼 값들)
|
|
235
|
-
const conditions = [];
|
|
236
|
-
for (const row of rowsWithoutId){
|
|
237
|
-
const values = columns.map((col)=>row[col]);
|
|
238
|
-
// null이 포함된 조건은 제외 (PostgreSQL UNIQUE는 NULL 무시)
|
|
239
|
-
if (!values.some((v)=>v == null)) {
|
|
240
|
-
conditions.push(values);
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
if (conditions.length === 0) continue;
|
|
244
|
-
// 배치 SELECT
|
|
245
|
-
const existingRows = await wdb(tableName).whereIn(columns, conditions).select("id", ...columns);
|
|
246
|
-
// Map 생성: unique 컬럼 조합 → id
|
|
247
|
-
const existingMap = new Map();
|
|
248
|
-
for (const existing of existingRows){
|
|
249
|
-
const key = columns.map((col)=>String(existing[col] ?? "")).join("---delimiter---");
|
|
250
|
-
const id = existing.id;
|
|
251
|
-
if (typeof id === "number" || typeof id === "string") {
|
|
252
|
-
existingMap.set(key, id);
|
|
253
|
-
}
|
|
254
|
-
}
|
|
255
|
-
// id 없는 row들에 매칭되는 id 채우기
|
|
256
|
-
for (const row of rowsWithoutId){
|
|
257
|
-
if (row.id) continue; // 이미 다른 uniqueIndex에서 채워진 경우 스킵
|
|
258
|
-
const key = columns.map((col)=>String(row[col] ?? "")).join("---delimiter---");
|
|
259
|
-
const existingId = existingMap.get(key);
|
|
260
|
-
if (existingId) {
|
|
261
|
-
row.id = existingId;
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
// onConflict는 id만 사용 (모든 uniqueIndexes는 이미 사전 조회로 처리됨)
|
|
267
|
-
const conflictColumns = [
|
|
268
|
-
"id"
|
|
269
|
-
];
|
|
270
|
-
const allColumns = Object.keys(dataForDb[0]);
|
|
271
|
-
let updateColumns = allColumns.filter((c)=>c !== "id");
|
|
272
|
-
// inherit 옵션 처리 - inherit 컬럼은 update 대상에서 제외
|
|
273
|
-
if (options?.inherit?.length) {
|
|
274
|
-
const inheritColumns = options.inherit;
|
|
275
|
-
const excludedFromUpdate = updateColumns.filter((c)=>inheritColumns.includes(c));
|
|
276
|
-
updateColumns = updateColumns.filter((c)=>!inheritColumns.includes(c));
|
|
277
|
-
// 실제로 제외된 컬럼 로깅
|
|
278
|
-
if (excludedFromUpdate.length) {
|
|
279
|
-
Naite.t("puri:ub-inherit", {
|
|
280
|
-
tableName,
|
|
281
|
-
inheritColumns,
|
|
282
|
-
excludedFromUpdate
|
|
283
|
-
});
|
|
284
|
-
}
|
|
285
|
-
}
|
|
286
|
-
// updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장
|
|
287
|
-
const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;
|
|
288
|
-
resultRows = await wdb.insert(dataForDb).into(tableName).onConflict(conflictColumns).merge(mergeColumns).returning(selectFields);
|
|
289
|
-
}
|
|
290
|
-
if (originalUuids.length !== resultRows.length) {
|
|
291
|
-
throw new Error(`${tableName}: register/returning 불일치`);
|
|
292
|
-
}
|
|
293
|
-
for(let i = 0; i < resultRows.length; i++){
|
|
294
|
-
uuidMap.set(originalUuids[i], resultRows[i]);
|
|
295
|
-
allIds.push(resultRows[i].id);
|
|
296
|
-
}
|
|
297
|
-
}
|
|
298
|
-
}
|
|
299
|
-
// 해당 테이블 참조를 실제 밸류로 변경
|
|
300
|
-
for (const table of refTables){
|
|
301
|
-
table.rows = table.rows.map((row)=>{
|
|
302
|
-
for (const key of Object.keys(row)){
|
|
303
|
-
const prop = row[key];
|
|
304
|
-
if (isRefField(prop) && prop.of === tableName) {
|
|
305
|
-
const parent = uuidMap.get(prop.uuid);
|
|
306
|
-
if (!parent) {
|
|
307
|
-
console.error(prop);
|
|
308
|
-
throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
|
|
309
|
-
}
|
|
310
|
-
const resolvedValue = parent[prop.use ?? "id"];
|
|
311
|
-
row[key] = resolvedValue;
|
|
312
|
-
Naite.t("puri:ub-ref-resolved", {
|
|
313
|
-
tableName,
|
|
314
|
-
field: key,
|
|
315
|
-
from: {
|
|
316
|
-
of: prop.of,
|
|
317
|
-
uuid: prop.uuid,
|
|
318
|
-
use: prop.use ?? "id"
|
|
319
|
-
},
|
|
320
|
-
to: resolvedValue
|
|
321
|
-
});
|
|
322
|
-
}
|
|
323
|
-
}
|
|
324
|
-
return row;
|
|
325
|
-
});
|
|
326
|
-
}
|
|
327
|
-
if (options?.cleanOrphans) {
|
|
328
|
-
const cleanOrphans = options.cleanOrphans;
|
|
329
|
-
const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [
|
|
330
|
-
cleanOrphans
|
|
331
|
-
];
|
|
332
|
-
// 현재 register된 레코드들의 FK 값들 추출
|
|
333
|
-
const fkConditions = fkColumns.map((fkCol)=>{
|
|
334
|
-
const fkValues = [
|
|
335
|
-
...new Set(table.rows.map((row)=>row[fkCol]).filter((v)=>v != null))
|
|
336
|
-
];
|
|
337
|
-
return {
|
|
338
|
-
column: fkCol,
|
|
339
|
-
values: fkValues
|
|
340
|
-
};
|
|
341
|
-
});
|
|
342
|
-
// 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행
|
|
343
|
-
if (fkConditions.every((fc)=>fc.values.length > 0)) {
|
|
344
|
-
let deleteQuery = wdb(tableName);
|
|
345
|
-
// 각 FK 컬럼에 대한 WHERE IN 조건 추가
|
|
346
|
-
for (const { column, values } of fkConditions){
|
|
347
|
-
deleteQuery = deleteQuery.whereIn(column, values);
|
|
348
|
-
}
|
|
349
|
-
// 방금 upsert한 ID는 제외
|
|
350
|
-
deleteQuery = deleteQuery.whereNotIn("id", allIds);
|
|
351
|
-
const deletedCount = await deleteQuery.delete();
|
|
352
|
-
Naite.t("puri:ub-clean-orphans", {
|
|
353
|
-
tableName,
|
|
354
|
-
cleanOrphans: fkColumns,
|
|
355
|
-
deletedCount
|
|
356
|
-
});
|
|
357
|
-
}
|
|
358
|
-
}
|
|
359
|
-
// 해당 테이블의 데이터 초기화
|
|
360
|
-
table.rows = [];
|
|
361
|
-
table.references.clear();
|
|
362
|
-
table.uniquesMap.clear();
|
|
363
|
-
Naite.t("puri:ub-upserted", {
|
|
364
|
-
tableName,
|
|
365
|
-
mode,
|
|
366
|
-
rowCount: allIds.length,
|
|
367
|
-
returnedIds: allIds
|
|
368
|
-
});
|
|
369
|
-
return allIds;
|
|
370
|
-
}
|
|
371
|
-
async updateBatch(wdb, tableName, options) {
|
|
372
|
-
options = {
|
|
373
|
-
...options,
|
|
374
|
-
chunkSize: options?.chunkSize ?? 500,
|
|
375
|
-
where: options?.where ?? "id"
|
|
376
|
-
};
|
|
377
|
-
if (this.hasTable(tableName) === false) {
|
|
378
|
-
return;
|
|
379
|
-
}
|
|
380
|
-
const table = this.tables.get(tableName);
|
|
381
|
-
if (!table) {
|
|
382
|
-
throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
|
|
383
|
-
} else if (table.rows.length === 0) {
|
|
384
|
-
return;
|
|
385
|
-
}
|
|
386
|
-
const whereColumns = Array.isArray(options.where) ? options.where : [
|
|
387
|
-
options.where ?? "id"
|
|
388
|
-
];
|
|
389
|
-
const rows = table.rows.map((_row)=>{
|
|
390
|
-
const { uuid: _, ...row } = _row; // uuid 제외
|
|
391
|
-
return row;
|
|
392
|
-
});
|
|
393
|
-
await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
|
|
394
|
-
Naite.t("puri:ub-batch-updated", {
|
|
395
|
-
tableName,
|
|
396
|
-
rowCount: rows.length,
|
|
397
|
-
whereColumns
|
|
398
|
-
});
|
|
399
|
-
// updateBatch 완료 후 처리된 데이터 제거
|
|
400
|
-
table.rows = [];
|
|
401
|
-
table.references.clear();
|
|
402
|
-
table.uniquesMap.clear();
|
|
403
|
-
}
|
|
404
|
-
// ============================================================================
|
|
405
|
-
// Private Helper Methods
|
|
406
|
-
// ============================================================================
|
|
407
|
-
/**
|
|
408
|
-
* rows를 의존성 순서에 따라 레벨별로 그룹화
|
|
409
|
-
* - 자기 참조 없는 경우 : 모든 rows가 Level 0
|
|
410
|
-
* - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
|
|
411
|
-
*/ buildInsertLevels(rows, tableName) {
|
|
412
|
-
// 1. 자기 참조가 없으면 한 레벨로 처리
|
|
413
|
-
const hasSelfRef = rows.flatMap((row)=>Object.values(row)).some((value)=>isRefField(value) && value.of === tableName);
|
|
414
|
-
if (!hasSelfRef) return {
|
|
415
|
-
levels: [
|
|
416
|
-
rows
|
|
417
|
-
],
|
|
418
|
-
hasCircular: false
|
|
419
|
-
};
|
|
420
|
-
// 2. uuid → row 매핑 (중복 uuid 방지)
|
|
421
|
-
const rowByUuid = new Map();
|
|
422
|
-
for (const row of rows){
|
|
423
|
-
const uuid = row.uuid;
|
|
424
|
-
if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
|
|
425
|
-
rowByUuid.set(uuid, row);
|
|
426
|
-
}
|
|
427
|
-
let pending = Array.from(rowByUuid.values());
|
|
428
|
-
const levels = [];
|
|
429
|
-
const inserted = new Set();
|
|
430
|
-
// 3. 레벨별 분류
|
|
431
|
-
while(pending.length > 0){
|
|
432
|
-
const currentLevel = [];
|
|
433
|
-
const nextPending = [];
|
|
434
|
-
for (const row of pending){
|
|
435
|
-
// 이 row가 참조하는 자기 참조들
|
|
436
|
-
const selfRefs = Object.values(row).filter((value)=>isRefField(value) && value.of === tableName);
|
|
437
|
-
// 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함
|
|
438
|
-
const canInsert = selfRefs.every((ref)=>{
|
|
439
|
-
if (!rowByUuid.has(ref.uuid)) {
|
|
440
|
-
throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
|
|
441
|
-
}
|
|
442
|
-
return inserted.has(ref.uuid);
|
|
443
|
-
});
|
|
444
|
-
if (canInsert) {
|
|
445
|
-
currentLevel.push(row);
|
|
446
|
-
} else {
|
|
447
|
-
nextPending.push(row);
|
|
448
|
-
}
|
|
449
|
-
}
|
|
450
|
-
// 순환 참조 감지
|
|
451
|
-
if (currentLevel.length === 0) return {
|
|
452
|
-
levels: [],
|
|
453
|
-
hasCircular: true
|
|
454
|
-
};
|
|
455
|
-
// 레벨 확정 + inserted 갱신
|
|
456
|
-
levels.push(currentLevel);
|
|
457
|
-
for (const row of currentLevel){
|
|
458
|
-
inserted.add(row.uuid);
|
|
459
|
-
}
|
|
460
|
-
pending = nextPending;
|
|
461
|
-
}
|
|
462
|
-
return {
|
|
463
|
-
levels,
|
|
464
|
-
hasCircular: false
|
|
465
|
-
};
|
|
466
|
-
}
|
|
9
|
+
|
|
10
|
+
//#region src/database/upsert-builder.ts
|
|
11
|
+
function isRefField(field) {
|
|
12
|
+
return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
|
|
467
13
|
}
|
|
14
|
+
var logger, UpsertBuilder;
|
|
15
|
+
var init_upsert_builder = __esmMin((() => {
|
|
16
|
+
init_entity_manager();
|
|
17
|
+
init_naite();
|
|
18
|
+
init_utils();
|
|
19
|
+
init__batch_update();
|
|
20
|
+
logger = getLogger([
|
|
21
|
+
"sonamu",
|
|
22
|
+
"internal",
|
|
23
|
+
"upsert-builder"
|
|
24
|
+
]);
|
|
25
|
+
UpsertBuilder = class {
|
|
26
|
+
tables;
|
|
27
|
+
constructor() {
|
|
28
|
+
this.tables = new Map();
|
|
29
|
+
}
|
|
30
|
+
getTable(tableName) {
|
|
31
|
+
const table = this.tables.get(tableName);
|
|
32
|
+
if (table) {
|
|
33
|
+
return table;
|
|
34
|
+
}
|
|
35
|
+
const tableSpec = (() => {
|
|
36
|
+
try {
|
|
37
|
+
return EntityManager.getTableSpec(tableName);
|
|
38
|
+
} catch {
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
})();
|
|
42
|
+
const tableData = {
|
|
43
|
+
references: new Set(),
|
|
44
|
+
rows: [],
|
|
45
|
+
uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
|
|
46
|
+
uniquesMap: new Map(),
|
|
47
|
+
jsonColumns: tableSpec?.jsonColumns ?? []
|
|
48
|
+
};
|
|
49
|
+
this.tables.set(tableName, tableData);
|
|
50
|
+
return tableData;
|
|
51
|
+
}
|
|
52
|
+
hasTable(tableName) {
|
|
53
|
+
return this.tables.has(tableName);
|
|
54
|
+
}
|
|
55
|
+
register(tableName, row) {
|
|
56
|
+
const table = this.getTable(tableName);
|
|
57
|
+
const uniqueKeys = table.uniqueIndexes.map((unqIndex) => {
|
|
58
|
+
const uniqueKeyArray = unqIndex.columns.map((unqCol) => {
|
|
59
|
+
const val = row[unqCol.name];
|
|
60
|
+
if (isRefField(val)) {
|
|
61
|
+
return val.uuid;
|
|
62
|
+
} else {
|
|
63
|
+
return row[unqCol.name] ?? randomUUID();
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
if (uniqueKeyArray.length === 0) {
|
|
67
|
+
return null;
|
|
68
|
+
}
|
|
69
|
+
return uniqueKeyArray.join("---delimiter--");
|
|
70
|
+
}).filter(nonNullable);
|
|
71
|
+
const { uuid, isReused } = (() => {
|
|
72
|
+
if (uniqueKeys.length > 0) {
|
|
73
|
+
for (const uniqueKey of uniqueKeys) {
|
|
74
|
+
if (table.uniquesMap.has(uniqueKey)) {
|
|
75
|
+
return {
|
|
76
|
+
uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
|
|
77
|
+
isReused: true
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
return {
|
|
83
|
+
uuid: randomUUID(),
|
|
84
|
+
isReused: false
|
|
85
|
+
};
|
|
86
|
+
})();
|
|
87
|
+
if (uniqueKeys.length > 0) {
|
|
88
|
+
for (const uniqueKey of uniqueKeys) {
|
|
89
|
+
table.uniquesMap.set(uniqueKey, uuid);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue]) => {
|
|
93
|
+
if (isRefField(rowValue)) {
|
|
94
|
+
rowValue.use ??= "id";
|
|
95
|
+
table.references.add(`${rowValue.of}.${rowValue.use}`);
|
|
96
|
+
return [rowKey, rowValue];
|
|
97
|
+
} else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {
|
|
98
|
+
return [rowKey, JSON.stringify(rowValue)];
|
|
99
|
+
} else if (isArray(rowValue)) {
|
|
100
|
+
return [rowKey, rowValue];
|
|
101
|
+
} else {
|
|
102
|
+
return [rowKey, rowValue];
|
|
103
|
+
}
|
|
104
|
+
}));
|
|
105
|
+
table.rows.push({
|
|
106
|
+
uuid,
|
|
107
|
+
...row
|
|
108
|
+
});
|
|
109
|
+
const result = {
|
|
110
|
+
of: tableName,
|
|
111
|
+
uuid: row.uuid ?? uuid
|
|
112
|
+
};
|
|
113
|
+
Naite.t("puri:ub-register", {
|
|
114
|
+
tableName,
|
|
115
|
+
uuid: result.uuid,
|
|
116
|
+
isUuidReused: isReused,
|
|
117
|
+
row
|
|
118
|
+
});
|
|
119
|
+
return result;
|
|
120
|
+
}
|
|
121
|
+
async upsert(wdb, tableName, options) {
|
|
122
|
+
return this.upsertOrInsert(wdb, tableName, "upsert", options);
|
|
123
|
+
}
|
|
124
|
+
async insertOnly(wdb, tableName, options) {
|
|
125
|
+
return this.upsertOrInsert(wdb, tableName, "insert", options);
|
|
126
|
+
}
|
|
127
|
+
async upsertOrInsert(wdb, tableName, mode, options) {
|
|
128
|
+
if (!this.hasTable(tableName)) {
|
|
129
|
+
return [];
|
|
130
|
+
}
|
|
131
|
+
const table = this.tables.get(tableName);
|
|
132
|
+
if (table === undefined) {
|
|
133
|
+
throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);
|
|
134
|
+
} else if (table.rows.length === 0) {
|
|
135
|
+
throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);
|
|
136
|
+
}
|
|
137
|
+
if (table.rows.some((row) => Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName))) {
|
|
138
|
+
throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);
|
|
139
|
+
}
|
|
140
|
+
const { references, refTables } = Array.from(this.tables).reduce((r, [, table$1]) => {
|
|
141
|
+
const reference = Array.from(table$1.references.values()).find((ref) => ref.includes(`${tableName}.`));
|
|
142
|
+
if (reference) {
|
|
143
|
+
r.references.push(reference);
|
|
144
|
+
r.refTables.push(table$1);
|
|
145
|
+
}
|
|
146
|
+
return r;
|
|
147
|
+
}, {
|
|
148
|
+
references: [],
|
|
149
|
+
refTables: []
|
|
150
|
+
});
|
|
151
|
+
const extractFields = unique(references).map((reference) => reference.split(".")[1]).filter((field) => field !== undefined);
|
|
152
|
+
const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
|
|
153
|
+
if (hasCircular) {
|
|
154
|
+
throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
|
|
155
|
+
}
|
|
156
|
+
const uuidMap = new Map();
|
|
157
|
+
const allIds = [];
|
|
158
|
+
for (let levelIdx = 0; levelIdx < levels.length; levelIdx++) {
|
|
159
|
+
const levelRows = levels[levelIdx];
|
|
160
|
+
logger.debug("Processing Query Level: {current} / {total}", {
|
|
161
|
+
current: levelIdx + 1,
|
|
162
|
+
total: levels.length
|
|
163
|
+
});
|
|
164
|
+
const resolvedRows = levelRows.map((row) => {
|
|
165
|
+
const resolved = { ...row };
|
|
166
|
+
for (const [key, value] of Object.entries(row)) {
|
|
167
|
+
if (isRefField(value) && value.of === tableName) {
|
|
168
|
+
const parent = uuidMap.get(value.uuid);
|
|
169
|
+
if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
|
|
170
|
+
resolved[key] = parent[value.use ?? "id"];
|
|
171
|
+
Naite.t("puri:ub-ref-resolved", {
|
|
172
|
+
tableName,
|
|
173
|
+
field: key,
|
|
174
|
+
from: {
|
|
175
|
+
of: value.of,
|
|
176
|
+
uuid: value.uuid,
|
|
177
|
+
use: value.use ?? "id"
|
|
178
|
+
},
|
|
179
|
+
to: resolved[key]
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
return resolved;
|
|
184
|
+
});
|
|
185
|
+
const chunkSize = options?.chunkSize;
|
|
186
|
+
const levelChunks = chunkSize ? cluster(resolvedRows, chunkSize) : [resolvedRows];
|
|
187
|
+
const selectFields = unique(["id", ...extractFields]);
|
|
188
|
+
for (let index = 0; index < levelChunks.length; index++) {
|
|
189
|
+
const dataChunk = levelChunks[index];
|
|
190
|
+
if (dataChunk.length === 0) continue;
|
|
191
|
+
logger.debug("Processing Chunk: {current} / {total}", {
|
|
192
|
+
current: index + 1,
|
|
193
|
+
total: levelChunks.length
|
|
194
|
+
});
|
|
195
|
+
const originalUuids = dataChunk.map((r) => r.uuid);
|
|
196
|
+
const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);
|
|
197
|
+
let resultRows;
|
|
198
|
+
if (mode === "insert") {
|
|
199
|
+
resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
|
|
200
|
+
} else {
|
|
201
|
+
const rowsWithoutId = dataForDb.filter((row) => !row.id);
|
|
202
|
+
if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {
|
|
203
|
+
for (const uniqueIndex of table.uniqueIndexes) {
|
|
204
|
+
const columns = uniqueIndex.columns.map((c) => c.name);
|
|
205
|
+
const conditions = [];
|
|
206
|
+
for (const row of rowsWithoutId) {
|
|
207
|
+
const values = columns.map((col) => row[col]);
|
|
208
|
+
if (!values.some((v) => v == null)) {
|
|
209
|
+
conditions.push(values);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
if (conditions.length === 0) continue;
|
|
213
|
+
const existingRows = await wdb(tableName).whereIn(columns, conditions).select("id", ...columns);
|
|
214
|
+
const existingMap = new Map();
|
|
215
|
+
for (const existing of existingRows) {
|
|
216
|
+
const key = columns.map((col) => String(existing[col] ?? "")).join("---delimiter---");
|
|
217
|
+
const id = existing.id;
|
|
218
|
+
if (typeof id === "number" || typeof id === "string") {
|
|
219
|
+
existingMap.set(key, id);
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
for (const row of rowsWithoutId) {
|
|
223
|
+
if (row.id) continue;
|
|
224
|
+
const key = columns.map((col) => String(row[col] ?? "")).join("---delimiter---");
|
|
225
|
+
const existingId = existingMap.get(key);
|
|
226
|
+
if (existingId) {
|
|
227
|
+
row.id = existingId;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
const conflictColumns = ["id"];
|
|
233
|
+
const allColumns = Object.keys(dataForDb[0]);
|
|
234
|
+
let updateColumns = allColumns.filter((c) => c !== "id");
|
|
235
|
+
if (options?.inherit?.length) {
|
|
236
|
+
const inheritColumns = options.inherit;
|
|
237
|
+
const excludedFromUpdate = updateColumns.filter((c) => inheritColumns.includes(c));
|
|
238
|
+
updateColumns = updateColumns.filter((c) => !inheritColumns.includes(c));
|
|
239
|
+
if (excludedFromUpdate.length) {
|
|
240
|
+
Naite.t("puri:ub-inherit", {
|
|
241
|
+
tableName,
|
|
242
|
+
inheritColumns,
|
|
243
|
+
excludedFromUpdate
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;
|
|
248
|
+
resultRows = await wdb.insert(dataForDb).into(tableName).onConflict(conflictColumns).merge(mergeColumns).returning(selectFields);
|
|
249
|
+
}
|
|
250
|
+
if (originalUuids.length !== resultRows.length) {
|
|
251
|
+
throw new Error(`${tableName}: register/returning 불일치`);
|
|
252
|
+
}
|
|
253
|
+
for (let i = 0; i < resultRows.length; i++) {
|
|
254
|
+
uuidMap.set(originalUuids[i], resultRows[i]);
|
|
255
|
+
allIds.push(resultRows[i].id);
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
for (const table$1 of refTables) {
|
|
260
|
+
table$1.rows = table$1.rows.map((row) => {
|
|
261
|
+
for (const key of Object.keys(row)) {
|
|
262
|
+
const prop = row[key];
|
|
263
|
+
if (isRefField(prop) && prop.of === tableName) {
|
|
264
|
+
const parent = uuidMap.get(prop.uuid);
|
|
265
|
+
if (!parent) {
|
|
266
|
+
console.error(prop);
|
|
267
|
+
throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
|
|
268
|
+
}
|
|
269
|
+
const resolvedValue = parent[prop.use ?? "id"];
|
|
270
|
+
row[key] = resolvedValue;
|
|
271
|
+
Naite.t("puri:ub-ref-resolved", {
|
|
272
|
+
tableName,
|
|
273
|
+
field: key,
|
|
274
|
+
from: {
|
|
275
|
+
of: prop.of,
|
|
276
|
+
uuid: prop.uuid,
|
|
277
|
+
use: prop.use ?? "id"
|
|
278
|
+
},
|
|
279
|
+
to: resolvedValue
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
return row;
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
if (options?.cleanOrphans) {
|
|
287
|
+
const cleanOrphans = options.cleanOrphans;
|
|
288
|
+
const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [cleanOrphans];
|
|
289
|
+
const fkConditions = fkColumns.map((fkCol) => {
|
|
290
|
+
const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter((v) => v != null))];
|
|
291
|
+
return {
|
|
292
|
+
column: fkCol,
|
|
293
|
+
values: fkValues
|
|
294
|
+
};
|
|
295
|
+
});
|
|
296
|
+
if (fkConditions.every((fc) => fc.values.length > 0)) {
|
|
297
|
+
let deleteQuery = wdb(tableName);
|
|
298
|
+
for (const { column, values } of fkConditions) {
|
|
299
|
+
deleteQuery = deleteQuery.whereIn(column, values);
|
|
300
|
+
}
|
|
301
|
+
deleteQuery = deleteQuery.whereNotIn("id", allIds);
|
|
302
|
+
const deletedCount = await deleteQuery.delete();
|
|
303
|
+
Naite.t("puri:ub-clean-orphans", {
|
|
304
|
+
tableName,
|
|
305
|
+
cleanOrphans: fkColumns,
|
|
306
|
+
deletedCount
|
|
307
|
+
});
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
table.rows = [];
|
|
311
|
+
table.references.clear();
|
|
312
|
+
table.uniquesMap.clear();
|
|
313
|
+
Naite.t("puri:ub-upserted", {
|
|
314
|
+
tableName,
|
|
315
|
+
mode,
|
|
316
|
+
rowCount: allIds.length,
|
|
317
|
+
returnedIds: allIds
|
|
318
|
+
});
|
|
319
|
+
return allIds;
|
|
320
|
+
}
|
|
321
|
+
async updateBatch(wdb, tableName, options) {
|
|
322
|
+
options = {
|
|
323
|
+
...options,
|
|
324
|
+
chunkSize: options?.chunkSize ?? 500,
|
|
325
|
+
where: options?.where ?? "id"
|
|
326
|
+
};
|
|
327
|
+
if (!this.hasTable(tableName)) {
|
|
328
|
+
return;
|
|
329
|
+
}
|
|
330
|
+
const table = this.tables.get(tableName);
|
|
331
|
+
if (!table) {
|
|
332
|
+
throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
|
|
333
|
+
} else if (table.rows.length === 0) {
|
|
334
|
+
return;
|
|
335
|
+
}
|
|
336
|
+
const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? "id"];
|
|
337
|
+
const rows = table.rows.map((_row) => {
|
|
338
|
+
const { uuid: _, ...row } = _row;
|
|
339
|
+
return row;
|
|
340
|
+
});
|
|
341
|
+
await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
|
|
342
|
+
Naite.t("puri:ub-batch-updated", {
|
|
343
|
+
tableName,
|
|
344
|
+
rowCount: rows.length,
|
|
345
|
+
whereColumns
|
|
346
|
+
});
|
|
347
|
+
table.rows = [];
|
|
348
|
+
table.references.clear();
|
|
349
|
+
table.uniquesMap.clear();
|
|
350
|
+
}
|
|
351
|
+
/**
|
|
352
|
+
* rows를 의존성 순서에 따라 레벨별로 그룹화
|
|
353
|
+
* - 자기 참조 없는 경우 : 모든 rows가 Level 0
|
|
354
|
+
* - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
|
|
355
|
+
*/
|
|
356
|
+
buildInsertLevels(rows, tableName) {
|
|
357
|
+
const hasSelfRef = rows.flatMap((row) => Object.values(row)).some((value) => isRefField(value) && value.of === tableName);
|
|
358
|
+
if (!hasSelfRef) return {
|
|
359
|
+
levels: [rows],
|
|
360
|
+
hasCircular: false
|
|
361
|
+
};
|
|
362
|
+
const rowByUuid = new Map();
|
|
363
|
+
for (const row of rows) {
|
|
364
|
+
const uuid = row.uuid;
|
|
365
|
+
if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
|
|
366
|
+
rowByUuid.set(uuid, row);
|
|
367
|
+
}
|
|
368
|
+
let pending = Array.from(rowByUuid.values());
|
|
369
|
+
const levels = [];
|
|
370
|
+
const inserted = new Set();
|
|
371
|
+
while (pending.length > 0) {
|
|
372
|
+
const currentLevel = [];
|
|
373
|
+
const nextPending = [];
|
|
374
|
+
for (const row of pending) {
|
|
375
|
+
const selfRefs = Object.values(row).filter((value) => isRefField(value) && value.of === tableName);
|
|
376
|
+
const canInsert = selfRefs.every((ref) => {
|
|
377
|
+
if (!rowByUuid.has(ref.uuid)) {
|
|
378
|
+
throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
|
|
379
|
+
}
|
|
380
|
+
return inserted.has(ref.uuid);
|
|
381
|
+
});
|
|
382
|
+
if (canInsert) {
|
|
383
|
+
currentLevel.push(row);
|
|
384
|
+
} else {
|
|
385
|
+
nextPending.push(row);
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
if (currentLevel.length === 0) return {
|
|
389
|
+
levels: [],
|
|
390
|
+
hasCircular: true
|
|
391
|
+
};
|
|
392
|
+
levels.push(currentLevel);
|
|
393
|
+
for (const row of currentLevel) {
|
|
394
|
+
inserted.add(row.uuid);
|
|
395
|
+
}
|
|
396
|
+
pending = nextPending;
|
|
397
|
+
}
|
|
398
|
+
return {
|
|
399
|
+
levels,
|
|
400
|
+
hasCircular: false
|
|
401
|
+
};
|
|
402
|
+
}
|
|
403
|
+
};
|
|
404
|
+
}));
|
|
468
405
|
|
|
469
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhYmFzZS91cHNlcnQtYnVpbGRlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBnZXRMb2dnZXIgfSBmcm9tIFwiQGxvZ3RhcGUvbG9ndGFwZVwiO1xuaW1wb3J0IHsgcmFuZG9tVVVJRCB9IGZyb20gXCJjcnlwdG9cIjtcbmltcG9ydCB0eXBlIHsgS25leCB9IGZyb20gXCJrbmV4XCI7XG5pbXBvcnQgeyBpc0FycmF5LCB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgRW50aXR5TWFuYWdlciB9IGZyb20gXCIuLi9lbnRpdHkvZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB7IE5haXRlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQgdHlwZSB7IERhdGFiYXNlRm9yZWlnbktleXMsIERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBFbnRpdHlJbmRleCB9IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgYXNzZXJ0RGVmaW5lZCwgY2h1bmssIG5vbk51bGxhYmxlIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgeyBiYXRjaFVwZGF0ZSwgdHlwZSBSb3dXaXRoSWQgfSBmcm9tIFwiLi9fYmF0Y2hfdXBkYXRlXCI7XG5pbXBvcnQgdHlwZSB7IENvbHVtbktleXMsIEZvcmVpZ25LZXlDb2x1bW5zLCBJZFR5cGUsIFRhYmxlTmFtZSB9IGZyb20gXCIuL3B1cmkudHlwZXNcIjtcblxuY29uc3QgbG9nZ2VyID0gZ2V0TG9nZ2VyKFtcInNvbmFtdVwiLCBcImludGVybmFsXCIsIFwidXBzZXJ0LWJ1aWxkZXJcIl0pO1xuXG4vKipcbiAqIEZLIO2DgOyehSDstpTroaDsnYQg7JyE7ZW0IERhdGFiYXNlRm9yZWlnbktleXMgZXhwb3J0XG4gKiAobW9kdWxlIGF1Z21lbnRhdGlvbiDsnpDrj5kg66Gc65OcIOuztOyepSlcbiAqL1xuZXhwb3J0IHR5cGUgeyBEYXRhYmFzZUZvcmVpZ25LZXlzIH07XG5cbnR5cGUgSW5oZXJpdGFibGVDb2x1bW5zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID1cbiAgVFRhYmxlIGV4dGVuZHMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQgPyBDb2x1bW5LZXlzPERhdGFiYXNlU2NoZW1hRXh0ZW5kW1RUYWJsZV0+IDogbmV2ZXI7XG5cbi8vIO2FjOydtOu4lCDrjbDsnbTthLAg7YOA7J6FXG50eXBlIFRhYmxlRGF0YSA9IHtcbiAgcmVmZXJlbmNlczogU2V0PHN0cmluZz47XG4gIHJvd3M6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W107XG4gIHVuaXF1ZUluZGV4ZXM6IEVudGl0eUluZGV4W107XG4gIHVuaXF1ZXNNYXA6IE1hcDxzdHJpbmcsIHN0cmluZz47XG4gIGpzb25Db2x1bW5zOiBzdHJpbmdbXTtcbn07XG5cbi8vIOywuOyhsCDtlYTrk5wg7YOA7J6FXG5leHBvcnQgdHlwZSBVQlJlZiA9IHtcbiAgdXVpZDogc3RyaW5nO1xuICBvZjogc3RyaW5nO1xuICB1c2U/OiBzdHJpbmc7XG59O1xuXG4vLyB1cHNlcnQg7Ji17IWYXG5leHBvcnQgdHlwZSBVcHNlcnRPcHRpb25zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG4gIGNsZWFuT3JwaGFucz86IEZvcmVpZ25LZXlDb2x1bW5zPFRUYWJsZT4gfCBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W107XG4gIGluaGVyaXQ/OiBJbmhlcml0YWJsZUNvbHVtbnM8VFRhYmxlPltdO1xufTtcblxuLy8gaW5zZXJ0T25seSDsmLXshZhcbmV4cG9ydCB0eXBlIEluc2VydE9ubHlPcHRpb25zID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gaXNSZWZGaWVsZChmaWVsZDogdW5rbm93bik6IGZpZWxkIGlzIFVCUmVmIHtcbiAgcmV0dXJuIChcbiAgICBmaWVsZCAhPT0gdW5kZWZpbmVkICYmXG4gICAgZmllbGQgIT09IG51bGwgJiZcbiAgICAoZmllbGQgYXMgVUJSZWYpPy5vZiAhPT0gdW5kZWZpbmVkICYmXG4gICAgKGZpZWxkIGFzIFVCUmVmKT8udXVpZCAhPT0gdW5kZWZpbmVkXG4gICk7XG59XG5cbmV4cG9ydCBjbGFzcyBVcHNlcnRCdWlsZGVyIHtcbiAgdGFibGVzOiBNYXA8c3RyaW5nLCBUYWJsZURhdGE+O1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnRhYmxlcyA9IG5ldyBNYXAoKTtcbiAgfVxuXG4gIGdldFRhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogVGFibGVEYXRhIHtcbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICh0YWJsZSkge1xuICAgICAgcmV0dXJuIHRhYmxlO1xuICAgIH1cblxuICAgIGNvbnN0IHRhYmxlU3BlYyA9ICgoKSA9PiB7XG4gICAgICB0cnkge1xuICAgICAgICByZXR1cm4gRW50aXR5TWFuYWdlci5nZXRUYWJsZVNwZWModGFibGVOYW1lKTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgIH1cbiAgICB9KSgpO1xuXG4gICAgY29uc3QgdGFibGVEYXRhID0ge1xuICAgICAgcmVmZXJlbmNlczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICByb3dzOiBbXSxcbiAgICAgIHVuaXF1ZUluZGV4ZXM6IHRhYmxlU3BlYz8udW5pcXVlSW5kZXhlcyA/PyBbXSxcbiAgICAgIHVuaXF1ZXNNYXA6IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+KCksXG4gICAgICBqc29uQ29sdW1uczogdGFibGVTcGVjPy5qc29uQ29sdW1ucyA/PyBbXSxcbiAgICB9O1xuICAgIHRoaXMudGFibGVzLnNldCh0YWJsZU5hbWUsIHRhYmxlRGF0YSk7XG4gICAgcmV0dXJuIHRhYmxlRGF0YTtcbiAgfVxuXG4gIGhhc1RhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMudGFibGVzLmhhcyh0YWJsZU5hbWUpO1xuICB9XG5cbiAgcmVnaXN0ZXI8VCBleHRlbmRzIHN0cmluZz4oXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgcm93OiB7XG4gICAgICBba2V5IGluIFRdPzogVUJSZWYgfCBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgYmlnaW50IHwgbnVsbCB8IG9iamVjdCB8IHVua25vd247XG4gICAgfSxcbiAgKTogVUJSZWYge1xuICAgIGNvbnN0IHRhYmxlID0gdGhpcy5nZXRUYWJsZSh0YWJsZU5hbWUpO1xuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCB1bmlxdWUg7J24642x7Iqk66W8IOyInO2ajO2VmOupsCDtgqQg7IOd7ISxXG4gICAgY29uc3QgdW5pcXVlS2V5cyA9IHRhYmxlLnVuaXF1ZUluZGV4ZXNcbiAgICAgIC5tYXAoKHVucUluZGV4KSA9PiB7XG4gICAgICAgIGNvbnN0IHVuaXF1ZUtleUFycmF5ID0gdW5xSW5kZXguY29sdW1ucy5tYXAoKHVucUNvbCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHZhbCA9IHJvd1t1bnFDb2wubmFtZSBhcyBrZXlvZiB0eXBlb2Ygcm93XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZCh2YWwpKSB7XG4gICAgICAgICAgICByZXR1cm4gdmFsLnV1aWQ7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJldHVybiByb3dbdW5xQ29sLm5hbWUgYXMga2V5b2YgdHlwZW9mIHJvd10gPz8gcmFuZG9tVVVJRCgpOyAvLyBudWxsYWJsZeyduCDqsr3smrAgdXVpZOuhnCDrnpzrjaTqsJIg7IK97J6FXG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICAvLyDqsJLsnbQg66qo65GQIG51bGzsnbgg6rK97JqwIO2CpCDsg53shLEg7Yyo7IqkXG4gICAgICAgIGlmICh1bmlxdWVLZXlBcnJheS5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdW5pcXVlS2V5QXJyYXkuam9pbihcIi0tLWRlbGltaXRlci0tXCIpO1xuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgLy8gdXVpZCDsg53shLEg66Gc7KeBXG4gICAgY29uc3QgeyB1dWlkLCBpc1JldXNlZCB9ID0gKCgpID0+IHtcbiAgICAgIC8vIO2CpOulvCDsiJztmoztlZjsl6wg7J2066+4IOyhtOyerO2VmOuKlCDtgqTqsIAg7J6I64qU7KeAIO2ZleyduFxuICAgICAgaWYgKHVuaXF1ZUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgICAgaWYgKHRhYmxlLnVuaXF1ZXNNYXAuaGFzKHVuaXF1ZUtleSkpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIHV1aWQ6IGFzc2VydERlZmluZWQodGFibGUudW5pcXVlc01hcC5nZXQodW5pcXVlS2V5KSwgXCJVbmlxdWUga2V5IG5vdCBmb3VuZFwiKSxcbiAgICAgICAgICAgICAgaXNSZXVzZWQ6IHRydWUsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyDssL7snYQg7IiYIOyXhuuKlCDqsr3smrAg7IOd7ISxXG4gICAgICByZXR1cm4geyB1dWlkOiByYW5kb21VVUlEKCksIGlzUmV1c2VkOiBmYWxzZSB9O1xuICAgIH0pKCk7XG5cbiAgICAvLyDrqqjrk6Ag7Jyg64uI7YGs7YKk7JeQIOuMgO2VtCDsnKDri4jtgazrp7Xsl5AgdXVpZCDsoIDsnqVcbiAgICBpZiAodW5pcXVlS2V5cy5sZW5ndGggPiAwKSB7XG4gICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgIHRhYmxlLnVuaXF1ZXNNYXAuc2V0KHVuaXF1ZUtleSwgdXVpZCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7J20IO2FjOydtOu4lOyXkCDsgqzsmqnrkJwgUmVmRmllbGTrpbwg7Iic7ZqM7ZWY7JesLCDtmITsnqwg7YWM7J2067iUIOygleuztOyXkCDslrTrlqQg7ZWE65Oc66W8IOywuOyhsO2VmOuKlOyngCDstpTqsIBcbiAgICAvLyDsnbQg7KCV67O066W8IOuCmOykkeyXkCDsuZjtmZjtlaAg65WMIOyCrOyaqVxuICAgIHJvdyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHJvdykubWFwKChbcm93S2V5LCByb3dWYWx1ZV0pID0+IHtcbiAgICAgICAgaWYgKGlzUmVmRmllbGQocm93VmFsdWUpKSB7XG4gICAgICAgICAgcm93VmFsdWUudXNlID8/PSBcImlkXCI7XG4gICAgICAgICAgdGFibGUucmVmZXJlbmNlcy5hZGQoYCR7cm93VmFsdWUub2Z9LiR7cm93VmFsdWUudXNlfWApO1xuICAgICAgICAgIHJldHVybiBbcm93S2V5LCByb3dWYWx1ZV07XG4gICAgICAgIH0gZWxzZSBpZiAodGFibGUuanNvbkNvbHVtbnMuaW5jbHVkZXMocm93S2V5KSAmJiByb3dWYWx1ZSAhPT0gbnVsbCkge1xuICAgICAgICAgIC8vIEpTT04g7Lus65+87J24IOqyveyasCBKU09OLnN0cmluZ2lmeSDsspjrpqwgKEtuZXjripQgSlNPTiDtg4DsnoXsnYQg7KeA7JuQ7ZWY7KeAIOyViuydjClcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgSlNPTi5zdHJpbmdpZnkocm93VmFsdWUpXTtcbiAgICAgICAgfSBlbHNlIGlmIChpc0FycmF5KHJvd1ZhbHVlKSkge1xuICAgICAgICAgIC8vIOuwsOyXtOydgCDqt7jrjIDroZwg7KCA7J6lXG4gICAgICAgICAgcmV0dXJuIFtyb3dLZXksIHJvd1ZhbHVlXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgcm93VmFsdWVdO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApIGFzIHsgW2tleSBpbiBUXT86IHVua25vd24gfTtcblxuICAgIHRhYmxlLnJvd3MucHVzaCh7XG4gICAgICB1dWlkLFxuICAgICAgLi4ucm93LFxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzdWx0OiBVQlJlZiA9IHtcbiAgICAgIG9mOiB0YWJsZU5hbWUsXG4gICAgICB1dWlkOiAocm93IGFzIHsgdXVpZD86IHN0cmluZyB9KS51dWlkID8/IHV1aWQsXG4gICAgfTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLXJlZ2lzdGVyXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHV1aWQ6IHJlc3VsdC51dWlkLFxuICAgICAgaXNVdWlkUmV1c2VkOiBpc1JldXNlZCxcbiAgICAgIHJvdyxcbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyB1cHNlcnQ8VFRhYmxlIGV4dGVuZHMgVGFibGVOYW1lPERhdGFiYXNlU2NoZW1hRXh0ZW5kPj4oXG4gICAgd2RiOiBLbmV4LFxuICAgIHRhYmxlTmFtZTogVFRhYmxlLFxuICAgIG9wdGlvbnM/OiBVcHNlcnRPcHRpb25zPFRUYWJsZT4sXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJ1cHNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyBpbnNlcnRPbmx5PFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+KFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IFRUYWJsZSxcbiAgICBvcHRpb25zPzogSW5zZXJ0T25seU9wdGlvbnMsXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJpbnNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyB1cHNlcnRPckluc2VydDxUVGFibGUgZXh0ZW5kcyBUYWJsZU5hbWU8RGF0YWJhc2VTY2hlbWFFeHRlbmQ+PihcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBUVGFibGUsXG4gICAgbW9kZTogXCJ1cHNlcnRcIiB8IFwiaW5zZXJ0XCIsXG4gICAgb3B0aW9ucz86IFVwc2VydE9wdGlvbnM8VFRhYmxlPixcbiAgKTogUHJvbWlzZTxJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXT4ge1xuICAgIGlmICh0aGlzLmhhc1RhYmxlKHRhYmxlTmFtZSkgPT09IGZhbHNlKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgdGFibGUgPSB0aGlzLnRhYmxlcy5nZXQodGFibGVOYW1lKTtcbiAgICBpZiAodGFibGUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBzZXJ0IOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgJHt0YWJsZU5hbWV97JeQIHVwc2VydCDtlaAg642w7J207YSw6rCAIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICBpZiAoXG4gICAgICB0YWJsZS5yb3dzLnNvbWUoKHJvdykgPT5cbiAgICAgICAgT2JqZWN0LmVudHJpZXMocm93KS5zb21lKChbLCB2YWx1ZV0pID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mICE9PSB0YWJsZU5hbWUpLFxuICAgICAgKVxuICAgICkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX0g7ZW06rKw65CY7KeAIOyViuydgCDssLjsobDqsIAg7J6I7Iq164uI64ukLmApO1xuICAgIH1cblxuICAgIC8vIOyghOyytCDthYzsnbTruJQg7Iic7ZqM7ZWY7JesIO2YhOyerCDthYzsnbTruJQg7LC47KGw7ZWY64qUIOuqqOuToCDthYzsnbTruJQg7LaU7LacXG4gICAgY29uc3QgeyByZWZlcmVuY2VzLCByZWZUYWJsZXMgfSA9IEFycmF5LmZyb20odGhpcy50YWJsZXMpLnJlZHVjZShcbiAgICAgIChyLCBbLCB0YWJsZV0pID0+IHtcbiAgICAgICAgY29uc3QgcmVmZXJlbmNlID0gQXJyYXkuZnJvbSh0YWJsZS5yZWZlcmVuY2VzLnZhbHVlcygpKS5maW5kKChyZWYpID0+XG4gICAgICAgICAgcmVmLmluY2x1ZGVzKGAke3RhYmxlTmFtZX0uYCksXG4gICAgICAgICk7XG4gICAgICAgIGlmIChyZWZlcmVuY2UpIHtcbiAgICAgICAgICByLnJlZmVyZW5jZXMucHVzaChyZWZlcmVuY2UpO1xuICAgICAgICAgIHIucmVmVGFibGVzLnB1c2godGFibGUpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9LFxuICAgICAge1xuICAgICAgICByZWZlcmVuY2VzOiBbXSBhcyBzdHJpbmdbXSxcbiAgICAgICAgcmVmVGFibGVzOiBbXSBhcyBUYWJsZURhdGFbXSxcbiAgICAgIH0sXG4gICAgKTtcbiAgICBjb25zdCBleHRyYWN0RmllbGRzID0gdW5pcXVlKHJlZmVyZW5jZXMpXG4gICAgICAubWFwKChyZWZlcmVuY2UpID0+IHJlZmVyZW5jZS5zcGxpdChcIi5cIilbMV0pXG4gICAgICAuZmlsdGVyKChmaWVsZCk6IGZpZWxkIGlzIHN0cmluZyA9PiBmaWVsZCAhPT0gdW5kZWZpbmVkKTtcblxuICAgIC8vIOydmOyhtOyEsSDsiJzshJzsl5Ag65Sw6528IOugiOuyqOuzhCDqt7jro7ntmZQgKOyekOq4sCDssLjsobDqsIAg7JeG7Jy866m0IExldmVsIDAg7ZWY64KYKVxuICAgIGNvbnN0IHsgbGV2ZWxzLCBoYXNDaXJjdWxhciB9ID0gdGhpcy5idWlsZEluc2VydExldmVscyh0YWJsZS5yb3dzLCB0YWJsZU5hbWUpO1xuXG4gICAgaWYgKGhhc0NpcmN1bGFyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfeyXkCDsiJztmZgg7J6Q6riwIOywuOyhsOqwgCDsnojsirXri4jri6QuYCk7XG4gICAgfVxuXG4gICAgY29uc3QgdXVpZE1hcCA9IG5ldyBNYXA8c3RyaW5nLCB1bmtub3duPigpO1xuICAgIGNvbnN0IGFsbElkczogKG51bWJlciB8IHN0cmluZylbXSA9IFtdO1xuXG4gICAgLy8g66CI67Ko67OE66GcIOyInOywqCDsspjrpqxcbiAgICBmb3IgKGxldCBsZXZlbElkeCA9IDA7IGxldmVsSWR4IDwgbGV2ZWxzLmxlbmd0aDsgbGV2ZWxJZHgrKykge1xuICAgICAgY29uc3QgbGV2ZWxSb3dzID0gbGV2ZWxzW2xldmVsSWR4XTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhcIlByb2Nlc3NpbmcgUXVlcnkgTGV2ZWw6IHtjdXJyZW50fSAvIHt0b3RhbH1cIiwge1xuICAgICAgICBjdXJyZW50OiBsZXZlbElkeCArIDEsXG4gICAgICAgIHRvdGFsOiBsZXZlbHMubGVuZ3RoLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIOydtOyghCDroIjrsqjsl5DshJwg7Ja77J2AIElE66GcIOyekOq4sCDssLjsobAg7ZW06rKwXG4gICAgICBjb25zdCByZXNvbHZlZFJvd3MgPSBsZXZlbFJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgY29uc3QgcmVzb2x2ZWQgPSB7IC4uLnJvdyB9O1xuICAgICAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhyb3cpKSB7XG4gICAgICAgICAgaWYgKGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHZhbHVlLnV1aWQpO1xuXG4gICAgICAgICAgICBpZiAoIXBhcmVudCkgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHt2YWx1ZS51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcblxuICAgICAgICAgICAgcmVzb2x2ZWRba2V5XSA9IChwYXJlbnQgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW3ZhbHVlLnVzZSA/PyBcImlkXCJdO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHZhbHVlLm9mLCB1dWlkOiB2YWx1ZS51dWlkLCB1c2U6IHZhbHVlLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkW2tleV0sXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJlc29sdmVkO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIO2YhOyerCDroIjrsqggdXBzZXJ0XG4gICAgICBjb25zdCBjaHVua1NpemUgPSBvcHRpb25zPy5jaHVua1NpemU7XG4gICAgICBjb25zdCBsZXZlbENodW5rcyA9IGNodW5rU2l6ZSA/IGNodW5rKHJlc29sdmVkUm93cywgY2h1bmtTaXplKSA6IFtyZXNvbHZlZFJvd3NdO1xuICAgICAgY29uc3Qgc2VsZWN0RmllbGRzID0gdW5pcXVlKFtcImlkXCIsIC4uLmV4dHJhY3RGaWVsZHNdKTtcblxuICAgICAgZm9yIChsZXQgaW5kZXggPSAwOyBpbmRleCA8IGxldmVsQ2h1bmtzLmxlbmd0aDsgaW5kZXgrKykge1xuICAgICAgICBjb25zdCBkYXRhQ2h1bmsgPSBsZXZlbENodW5rc1tpbmRleF07XG4gICAgICAgIGlmIChkYXRhQ2h1bmsubGVuZ3RoID09PSAwKSBjb250aW51ZTtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiUHJvY2Vzc2luZyBDaHVuazoge2N1cnJlbnR9IC8ge3RvdGFsfVwiLCB7XG4gICAgICAgICAgY3VycmVudDogaW5kZXggKyAxLFxuICAgICAgICAgIHRvdGFsOiBsZXZlbENodW5rcy5sZW5ndGgsXG4gICAgICAgIH0pO1xuXG4gICAgICAgIC8vIHV1aWTrpbwg67OE64+E66GcIOuztOq0gO2VmOqzoCwgRELsl5Ag7KCA7J6l7ZWgIOuNsOydtO2EsOyXkOyEnCDsoJzqsbBcbiAgICAgICAgY29uc3Qgb3JpZ2luYWxVdWlkcyA9IGRhdGFDaHVuay5tYXAoKHIpID0+IHIudXVpZCBhcyBzdHJpbmcpO1xuICAgICAgICBjb25zdCBkYXRhRm9yRGIgPSBkYXRhQ2h1bmsubWFwKCh7IHV1aWQsIC4uLnJlc3QgfSkgPT4gcmVzdCk7XG5cbiAgICAgICAgbGV0IHJlc3VsdFJvd3M6IHsgaWQ6IG51bWJlciB8IHN0cmluZzsgW2tleTogc3RyaW5nXTogdW5rbm93biB9W107XG5cbiAgICAgICAgaWYgKG1vZGUgPT09IFwiaW5zZXJ0XCIpIHtcbiAgICAgICAgICAvLyBJTlNFUlQg66qo65OcIC0gUkVUVVJOSU5HIOyCrOyaqVxuICAgICAgICAgIHJlc3VsdFJvd3MgPSBhd2FpdCB3ZGIuaW5zZXJ0KGRhdGFGb3JEYikuaW50byh0YWJsZU5hbWUpLnJldHVybmluZyhzZWxlY3RGaWVsZHMpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIFVQU0VSVCDrqqjrk5wgLSBpZCDsl4bripQgcm9365Ok7J2YIGlk66W8IOyCrOyghCDsobDtmozroZwg7LGE7Jqw6riwXG4gICAgICAgICAgY29uc3Qgcm93c1dpdGhvdXRJZCA9IGRhdGFGb3JEYi5maWx0ZXIoKHJvdykgPT4gIXJvdy5pZCk7XG5cbiAgICAgICAgICBpZiAocm93c1dpdGhvdXRJZC5sZW5ndGggPiAwICYmIHRhYmxlLnVuaXF1ZUluZGV4ZXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgLy8g66qo65OgIHVuaXF1ZUluZGV4ZXProZwg6riw7KG0IOugiOy9lOuTnCDsobDtmoxcbiAgICAgICAgICAgIGZvciAoY29uc3QgdW5pcXVlSW5kZXggb2YgdGFibGUudW5pcXVlSW5kZXhlcykge1xuICAgICAgICAgICAgICBjb25zdCBjb2x1bW5zID0gdW5pcXVlSW5kZXguY29sdW1ucy5tYXAoKGMpID0+IGMubmFtZSk7XG5cbiAgICAgICAgICAgICAgLy8g7KGw7ZqM7ZWgIOyhsOqxtOuTpCDstpTstpwgKOqwgSByb3fsnZggdW5pcXVlIOy7rOufvCDqsJLrk6QpXG4gICAgICAgICAgICAgIGNvbnN0IGNvbmRpdGlvbnM6IHVua25vd25bXVtdID0gW107XG4gICAgICAgICAgICAgIGZvciAoY29uc3Qgcm93IG9mIHJvd3NXaXRob3V0SWQpIHtcbiAgICAgICAgICAgICAgICBjb25zdCB2YWx1ZXMgPSBjb2x1bW5zLm1hcCgoY29sKSA9PiByb3dbY29sXSk7XG4gICAgICAgICAgICAgICAgLy8gbnVsbOydtCDtj6ztlajrkJwg7KGw6rG07J2AIOygnOyZuCAoUG9zdGdyZVNRTCBVTklRVUXripQgTlVMTCDrrLTsi5wpXG4gICAgICAgICAgICAgICAgaWYgKCF2YWx1ZXMuc29tZSgodikgPT4gdiA9PSBudWxsKSkge1xuICAgICAgICAgICAgICAgICAgY29uZGl0aW9ucy5wdXNoKHZhbHVlcyk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgaWYgKGNvbmRpdGlvbnMubGVuZ3RoID09PSAwKSBjb250aW51ZTtcblxuICAgICAgICAgICAgICAvLyDrsLDsuZggU0VMRUNUXG4gICAgICAgICAgICAgIGNvbnN0IGV4aXN0aW5nUm93cyA9IChhd2FpdCB3ZGIodGFibGVOYW1lKVxuICAgICAgICAgICAgICAgIC53aGVyZUluKGNvbHVtbnMsIGNvbmRpdGlvbnMgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdKVxuICAgICAgICAgICAgICAgIC5zZWxlY3QoXCJpZFwiLCAuLi5jb2x1bW5zKSkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXTtcblxuICAgICAgICAgICAgICAvLyBNYXAg7IOd7ISxOiB1bmlxdWUg7Lus65+8IOyhsO2VqSDihpIgaWRcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdNYXAgPSBuZXcgTWFwPHN0cmluZywgbnVtYmVyIHwgc3RyaW5nPigpO1xuICAgICAgICAgICAgICBmb3IgKGNvbnN0IGV4aXN0aW5nIG9mIGV4aXN0aW5nUm93cykge1xuICAgICAgICAgICAgICAgIGNvbnN0IGtleSA9IGNvbHVtbnNcbiAgICAgICAgICAgICAgICAgIC5tYXAoKGNvbCkgPT4gU3RyaW5nKGV4aXN0aW5nW2NvbF0gPz8gXCJcIikpXG4gICAgICAgICAgICAgICAgICAuam9pbihcIi0tLWRlbGltaXRlci0tLVwiKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpZCA9IGV4aXN0aW5nLmlkO1xuICAgICAgICAgICAgICAgIGlmICh0eXBlb2YgaWQgPT09IFwibnVtYmVyXCIgfHwgdHlwZW9mIGlkID09PSBcInN0cmluZ1wiKSB7XG4gICAgICAgICAgICAgICAgICBleGlzdGluZ01hcC5zZXQoa2V5LCBpZCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgLy8gaWQg7JeG64qUIHJvd+uTpOyXkCDrp6Tsua3rkJjripQgaWQg7LGE7Jqw6riwXG4gICAgICAgICAgICAgIGZvciAoY29uc3Qgcm93IG9mIHJvd3NXaXRob3V0SWQpIHtcbiAgICAgICAgICAgICAgICBpZiAocm93LmlkKSBjb250aW51ZTsgLy8g7J2066+4IOuLpOuluCB1bmlxdWVJbmRleOyXkOyEnCDssYTsm4zsp4Qg6rK97JqwIOyKpO2CtVxuXG4gICAgICAgICAgICAgICAgY29uc3Qga2V5ID0gY29sdW1ucy5tYXAoKGNvbCkgPT4gU3RyaW5nKHJvd1tjb2xdID8/IFwiXCIpKS5qb2luKFwiLS0tZGVsaW1pdGVyLS0tXCIpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGV4aXN0aW5nSWQgPSBleGlzdGluZ01hcC5nZXQoa2V5KTtcblxuICAgICAgICAgICAgICAgIGlmIChleGlzdGluZ0lkKSB7XG4gICAgICAgICAgICAgICAgICByb3cuaWQgPSBleGlzdGluZ0lkO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIG9uQ29uZmxpY3TripQgaWTrp4wg7IKs7JqpICjrqqjrk6AgdW5pcXVlSW5kZXhlc+uKlCDsnbTrr7gg7IKs7KCEIOyhsO2ajOuhnCDsspjrpqzrkKgpXG4gICAgICAgICAgY29uc3QgY29uZmxpY3RDb2x1bW5zID0gW1wiaWRcIl07XG5cbiAgICAgICAgICBjb25zdCBhbGxDb2x1bW5zID0gT2JqZWN0LmtleXMoZGF0YUZvckRiWzBdKTtcbiAgICAgICAgICBsZXQgdXBkYXRlQ29sdW1ucyA9IGFsbENvbHVtbnMuZmlsdGVyKChjKSA9PiBjICE9PSBcImlkXCIpO1xuXG4gICAgICAgICAgLy8gaW5oZXJpdCDsmLXshZgg7LKY66asIC0gaW5oZXJpdCDsu6zrn7zsnYAgdXBkYXRlIOuMgOyDgeyXkOyEnCDsoJzsmbhcbiAgICAgICAgICBpZiAob3B0aW9ucz8uaW5oZXJpdD8ubGVuZ3RoKSB7XG4gICAgICAgICAgICBjb25zdCBpbmhlcml0Q29sdW1ucyA9IG9wdGlvbnMuaW5oZXJpdCBhcyBzdHJpbmdbXTtcblxuICAgICAgICAgICAgY29uc3QgZXhjbHVkZWRGcm9tVXBkYXRlID0gdXBkYXRlQ29sdW1ucy5maWx0ZXIoKGMpID0+IGluaGVyaXRDb2x1bW5zLmluY2x1ZGVzKGMpKTtcbiAgICAgICAgICAgIHVwZGF0ZUNvbHVtbnMgPSB1cGRhdGVDb2x1bW5zLmZpbHRlcigoYykgPT4gIWluaGVyaXRDb2x1bW5zLmluY2x1ZGVzKGMpKTtcblxuICAgICAgICAgICAgLy8g7Iuk7KCc66GcIOygnOyZuOuQnCDsu6zrn7wg66Gc6rmFXG4gICAgICAgICAgICBpZiAoZXhjbHVkZWRGcm9tVXBkYXRlLmxlbmd0aCkge1xuICAgICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1pbmhlcml0XCIsIHtcbiAgICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgICAgaW5oZXJpdENvbHVtbnMsXG4gICAgICAgICAgICAgICAgZXhjbHVkZWRGcm9tVXBkYXRlLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyB1cGRhdGVDb2x1bW5z6rCAIOu5hOyWtOyeiOyWtOuPhCBtZXJnZSgp66W8IOyCrOyaqe2VmOyXrCDrqqjrk6Ag7ZaJ7J20IFJFVFVSTklOR+uQmOuPhOuhnSDrs7TsnqVcbiAgICAgICAgICBjb25zdCBtZXJnZUNvbHVtbnMgPSB1cGRhdGVDb2x1bW5zLmxlbmd0aCA/IHVwZGF0ZUNvbHVtbnMgOiBjb25mbGljdENvbHVtbnM7XG5cbiAgICAgICAgICByZXN1bHRSb3dzID0gYXdhaXQgd2RiXG4gICAgICAgICAgICAuaW5zZXJ0KGRhdGFGb3JEYilcbiAgICAgICAgICAgIC5pbnRvKHRhYmxlTmFtZSlcbiAgICAgICAgICAgIC5vbkNvbmZsaWN0KGNvbmZsaWN0Q29sdW1ucylcbiAgICAgICAgICAgIC5tZXJnZShtZXJnZUNvbHVtbnMpXG4gICAgICAgICAgICAucmV0dXJuaW5nKHNlbGVjdEZpZWxkcyk7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAob3JpZ2luYWxVdWlkcy5sZW5ndGggIT09IHJlc3VsdFJvd3MubGVuZ3RoKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX06IHJlZ2lzdGVyL3JldHVybmluZyDrtojsnbzsuZhgKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgcmVzdWx0Um93cy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgIHV1aWRNYXAuc2V0KG9yaWdpbmFsVXVpZHNbaV0sIHJlc3VsdFJvd3NbaV0pO1xuICAgICAgICAgIGFsbElkcy5wdXNoKHJlc3VsdFJvd3NbaV0uaWQpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lCDssLjsobDrpbwg7Iuk7KCcIOuwuOulmOuhnCDrs4Dqsr1cbiAgICBmb3IgKGNvbnN0IHRhYmxlIG9mIHJlZlRhYmxlcykge1xuICAgICAgdGFibGUucm93cyA9IHRhYmxlLnJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMocm93KSkge1xuICAgICAgICAgIGNvbnN0IHByb3AgPSByb3dba2V5XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZChwcm9wKSAmJiBwcm9wLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHByb3AudXVpZCk7XG4gICAgICAgICAgICBpZiAoIXBhcmVudCkge1xuICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKHByb3ApO1xuICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYOyhtOyerO2VmOyngCDslYrripQgdXVpZCAke3Byb3AudXVpZH0gLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCByZXNvbHZlZFZhbHVlID0gKHBhcmVudCBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbcHJvcC51c2UgPz8gXCJpZFwiXTtcbiAgICAgICAgICAgIHJvd1trZXldID0gcmVzb2x2ZWRWYWx1ZTtcblxuICAgICAgICAgICAgTmFpdGUudChcInB1cmk6dWItcmVmLXJlc29sdmVkXCIsIHtcbiAgICAgICAgICAgICAgdGFibGVOYW1lLFxuICAgICAgICAgICAgICBmaWVsZDoga2V5LFxuICAgICAgICAgICAgICBmcm9tOiB7IG9mOiBwcm9wLm9mLCB1dWlkOiBwcm9wLnV1aWQsIHVzZTogcHJvcC51c2UgPz8gXCJpZFwiIH0sXG4gICAgICAgICAgICAgIHRvOiByZXNvbHZlZFZhbHVlLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiByb3c7XG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBpZiAob3B0aW9ucz8uY2xlYW5PcnBoYW5zKSB7XG4gICAgICBjb25zdCBjbGVhbk9ycGhhbnMgPSBvcHRpb25zLmNsZWFuT3JwaGFucztcbiAgICAgIGNvbnN0IGZrQ29sdW1ucyA9IGlzQXJyYXkoY2xlYW5PcnBoYW5zKVxuICAgICAgICA/IChjbGVhbk9ycGhhbnMgYXMgRm9yZWlnbktleUNvbHVtbnM8VFRhYmxlPltdKVxuICAgICAgICA6IFtjbGVhbk9ycGhhbnMgYXMgRm9yZWlnbktleUNvbHVtbnM8VFRhYmxlPl07XG5cbiAgICAgIC8vIO2YhOyerCByZWdpc3RlcuuQnCDroIjsvZTrk5zrk6TsnZggRksg6rCS65OkIOy2lOy2nFxuICAgICAgY29uc3QgZmtDb25kaXRpb25zID0gZmtDb2x1bW5zLm1hcCgoZmtDb2wpID0+IHtcbiAgICAgICAgY29uc3QgZmtWYWx1ZXMgPSBbLi4ubmV3IFNldCh0YWJsZS5yb3dzLm1hcCgocm93KSA9PiByb3dbZmtDb2xdKS5maWx0ZXIoKHYpID0+IHYgIT0gbnVsbCkpXTtcbiAgICAgICAgcmV0dXJuIHsgY29sdW1uOiBma0NvbCwgdmFsdWVzOiBma1ZhbHVlcyB9O1xuICAgICAgfSk7XG5cbiAgICAgIC8vIOuqqOuToCBGSyDsu6zrn7zsl5Ag6rCS7J20IOyeiOuKlCDqsr3smrDsl5Drp4wg7IKt7KCcIOyLpO2WiVxuICAgICAgaWYgKGZrQ29uZGl0aW9ucy5ldmVyeSgoZmMpID0+IGZjLnZhbHVlcy5sZW5ndGggPiAwKSkge1xuICAgICAgICBsZXQgZGVsZXRlUXVlcnkgPSB3ZGIodGFibGVOYW1lKTtcblxuICAgICAgICAvLyDqsIEgRksg7Lus65+87JeQIOuMgO2VnCBXSEVSRSBJTiDsobDqsbQg7LaU6rCAXG4gICAgICAgIGZvciAoY29uc3QgeyBjb2x1bW4sIHZhbHVlcyB9IG9mIGZrQ29uZGl0aW9ucykge1xuICAgICAgICAgIGRlbGV0ZVF1ZXJ5ID0gZGVsZXRlUXVlcnkud2hlcmVJbihjb2x1bW4sIHZhbHVlcyk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyDrsKnquIggdXBzZXJ07ZWcIElE64qUIOygnOyZuFxuICAgICAgICBkZWxldGVRdWVyeSA9IGRlbGV0ZVF1ZXJ5LndoZXJlTm90SW4oXCJpZFwiLCBhbGxJZHMpO1xuXG4gICAgICAgIGNvbnN0IGRlbGV0ZWRDb3VudCA9IGF3YWl0IGRlbGV0ZVF1ZXJ5LmRlbGV0ZSgpO1xuXG4gICAgICAgIE5haXRlLnQoXCJwdXJpOnViLWNsZWFuLW9ycGhhbnNcIiwge1xuICAgICAgICAgIHRhYmxlTmFtZSxcbiAgICAgICAgICBjbGVhbk9ycGhhbnM6IGZrQ29sdW1ucyxcbiAgICAgICAgICBkZWxldGVkQ291bnQsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIO2VtOuLuSDthYzsnbTruJTsnZgg642w7J207YSwIOy0iOq4sO2ZlFxuICAgIHRhYmxlLnJvd3MgPSBbXTtcbiAgICB0YWJsZS5yZWZlcmVuY2VzLmNsZWFyKCk7XG4gICAgdGFibGUudW5pcXVlc01hcC5jbGVhcigpO1xuXG4gICAgTmFpdGUudChcInB1cmk6dWItdXBzZXJ0ZWRcIiwge1xuICAgICAgdGFibGVOYW1lLFxuICAgICAgbW9kZSxcbiAgICAgIHJvd0NvdW50OiBhbGxJZHMubGVuZ3RoLFxuICAgICAgcmV0dXJuZWRJZHM6IGFsbElkcyxcbiAgICB9KTtcblxuICAgIHJldHVybiBhbGxJZHMgYXMgSWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W107XG4gIH1cblxuICBhc3luYyB1cGRhdGVCYXRjaChcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgb3B0aW9ucz86IHtcbiAgICAgIGNodW5rU2l6ZT86IG51bWJlcjtcbiAgICAgIHdoZXJlPzogc3RyaW5nIHwgc3RyaW5nW107XG4gICAgfSxcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgb3B0aW9ucyA9IHtcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgICBjaHVua1NpemU6IG9wdGlvbnM/LmNodW5rU2l6ZSA/PyA1MDAsXG4gICAgICB3aGVyZTogb3B0aW9ucz8ud2hlcmUgPz8gXCJpZFwiLFxuICAgIH07XG5cbiAgICBpZiAodGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpID09PSBmYWxzZSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICghdGFibGUpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihg65Ox66Gd65CY7KeAIOyViuydgCDthYzsnbTruJQgJHt0YWJsZU5hbWV97JeQIHVwZGF0ZUJhdGNoIOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCB3aGVyZUNvbHVtbnMgPSBBcnJheS5pc0FycmF5KG9wdGlvbnMud2hlcmUpID8gb3B0aW9ucy53aGVyZSA6IFtvcHRpb25zLndoZXJlID8/IFwiaWRcIl07XG4gICAgY29uc3Qgcm93cyA9IHRhYmxlLnJvd3MubWFwKChfcm93KSA9PiB7XG4gICAgICBjb25zdCB7IHV1aWQ6IF8sIC4uLnJvdyB9ID0gX3JvdzsgLy8gdXVpZCDsoJzsmbhcbiAgICAgIHJldHVybiByb3cgYXMgUm93V2l0aElkPHN0cmluZz47XG4gICAgfSk7XG5cbiAgICBhd2FpdCBiYXRjaFVwZGF0ZSh3ZGIsIHRhYmxlTmFtZSwgd2hlcmVDb2x1bW5zLCByb3dzLCBvcHRpb25zLmNodW5rU2l6ZSk7XG5cbiAgICBOYWl0ZS50KFwicHVyaTp1Yi1iYXRjaC11cGRhdGVkXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHJvd0NvdW50OiByb3dzLmxlbmd0aCxcbiAgICAgIHdoZXJlQ29sdW1ucyxcbiAgICB9KTtcblxuICAgIC8vIHVwZGF0ZUJhdGNoIOyZhOujjCDtm4Qg7LKY66as65CcIOuNsOydtO2EsCDsoJzqsbBcbiAgICB0YWJsZS5yb3dzID0gW107XG4gICAgdGFibGUucmVmZXJlbmNlcy5jbGVhcigpO1xuICAgIHRhYmxlLnVuaXF1ZXNNYXAuY2xlYXIoKTtcbiAgfVxuXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbiAgLy8gUHJpdmF0ZSBIZWxwZXIgTWV0aG9kc1xuICAvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbiAgLyoqXG4gICAqIHJvd3Prpbwg7J2Y7KG07ISxIOyInOyEnOyXkCDrlLDrnbwg66CI67Ko67OE66GcIOq3uOujue2ZlFxuICAgKiAtIOyekOq4sCDssLjsobAg7JeG64qUIOqyveyasCA6IOuqqOuToCByb3dz6rCAIExldmVsIDBcbiAgICogLSDsnpDquLAg7LC47KGwIOyeiOuKlCDqsr3smrAgOiDsnpDquLAg7LC47KGwIOq0gOqzhOulvCDsnITsg4Eg7KCV66Cs7ZWY7JesIOugiOuyqOuzhOuhnCDqt7jro7ntmZRcbiAgICovXG4gIHByaXZhdGUgYnVpbGRJbnNlcnRMZXZlbHMoXG4gICAgcm93czogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSxcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgKTogeyBsZXZlbHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W11bXTsgaGFzQ2lyY3VsYXI6IGJvb2xlYW4gfSB7XG4gICAgLy8gMS4g7J6Q6riwIOywuOyhsOqwgCDsl4bsnLzrqbQg7ZWcIOugiOuyqOuhnCDsspjrpqxcbiAgICBjb25zdCBoYXNTZWxmUmVmID0gcm93c1xuICAgICAgLmZsYXRNYXAoKHJvdykgPT4gT2JqZWN0LnZhbHVlcyhyb3cpKVxuICAgICAgLnNvbWUoKHZhbHVlKSA9PiBpc1JlZkZpZWxkKHZhbHVlKSAmJiB2YWx1ZS5vZiA9PT0gdGFibGVOYW1lKTtcbiAgICBpZiAoIWhhc1NlbGZSZWYpIHJldHVybiB7IGxldmVsczogW3Jvd3NdLCBoYXNDaXJjdWxhcjogZmFsc2UgfTtcblxuICAgIC8vIDIuIHV1aWQg4oaSIHJvdyDrp6TtlZEgKOykkeuztSB1dWlkIOuwqeyngClcbiAgICBjb25zdCByb3dCeVV1aWQgPSBuZXcgTWFwPHN0cmluZywgUmVjb3JkPHN0cmluZywgdW5rbm93bj4+KCk7XG4gICAgZm9yIChjb25zdCByb3cgb2Ygcm93cykge1xuICAgICAgY29uc3QgdXVpZCA9IHJvdy51dWlkIGFzIHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICAgIGlmICghdXVpZCkgdGhyb3cgbmV3IEVycm9yKGBidWlsZEluc2VydExldmVsczogdXVpZOqwgCDsl4bripQgcm93IC0tIGluICR7dGFibGVOYW1lfWApO1xuICAgICAgcm93QnlVdWlkLnNldCh1dWlkLCByb3cpO1xuICAgIH1cblxuICAgIGxldCBwZW5kaW5nID0gQXJyYXkuZnJvbShyb3dCeVV1aWQudmFsdWVzKCkpO1xuICAgIGNvbnN0IGxldmVsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdID0gW107XG4gICAgY29uc3QgaW5zZXJ0ZWQgPSBuZXcgU2V0PHN0cmluZz4oKTtcblxuICAgIC8vIDMuIOugiOuyqOuzhCDrtoTrpZhcbiAgICB3aGlsZSAocGVuZGluZy5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBjdXJyZW50TGV2ZWw6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W10gPSBbXTtcbiAgICAgIGNvbnN0IG5leHRQZW5kaW5nOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdID0gW107XG5cbiAgICAgIGZvciAoY29uc3Qgcm93IG9mIHBlbmRpbmcpIHtcbiAgICAgICAgLy8g7J20IHJvd+qwgCDssLjsobDtlZjripQg7J6Q6riwIOywuOyhsOuTpFxuICAgICAgICBjb25zdCBzZWxmUmVmcyA9IE9iamVjdC52YWx1ZXMocm93KS5maWx0ZXIoXG4gICAgICAgICAgKHZhbHVlKSA9PiBpc1JlZkZpZWxkKHZhbHVlKSAmJiB2YWx1ZS5vZiA9PT0gdGFibGVOYW1lLFxuICAgICAgICApIGFzIFVCUmVmW107XG5cbiAgICAgICAgLy8g7LC47KGw7ZWY64qUIOuqqOuToCB1dWlk6rCAIOydtOuvuCBpbnNlcnRlZOyXkCDsnojslrTslbwg7J2067KIIOugiOuyqOyXkCDtj6ztlahcbiAgICAgICAgY29uc3QgY2FuSW5zZXJ0ID0gc2VsZlJlZnMuZXZlcnkoKHJlZikgPT4ge1xuICAgICAgICAgIGlmICghcm93QnlVdWlkLmhhcyhyZWYudXVpZCkpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihg7KG07J6s7ZWY7KeAIOyViuuKlCB1dWlkICR7cmVmLnV1aWR9IC0tIGluICR7dGFibGVOYW1lfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gaW5zZXJ0ZWQuaGFzKHJlZi51dWlkKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKGNhbkluc2VydCkge1xuICAgICAgICAgIGN1cnJlbnRMZXZlbC5wdXNoKHJvdyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgbmV4dFBlbmRpbmcucHVzaChyb3cpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIOyInO2ZmCDssLjsobAg6rCQ7KeAXG4gICAgICBpZiAoY3VycmVudExldmVsLmxlbmd0aCA9PT0gMCkgcmV0dXJuIHsgbGV2ZWxzOiBbXSwgaGFzQ2lyY3VsYXI6IHRydWUgfTtcblxuICAgICAgLy8g66CI67KoIO2ZleyglSArIGluc2VydGVkIOqwseyLoFxuICAgICAgbGV2ZWxzLnB1c2goY3VycmVudExldmVsKTtcbiAgICAgIGZvciAoY29uc3Qgcm93IG9mIGN1cnJlbnRMZXZlbCkge1xuICAgICAgICBpbnNlcnRlZC5hZGQocm93LnV1aWQgYXMgc3RyaW5nKTtcbiAgICAgIH1cblxuICAgICAgcGVuZGluZyA9IG5leHRQZW5kaW5nO1xuICAgIH1cblxuICAgIHJldHVybiB7IGxldmVscywgaGFzQ2lyY3VsYXI6IGZhbHNlIH07XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJnZXRMb2dnZXIiLCJyYW5kb21VVUlEIiwiaXNBcnJheSIsInVuaXF1ZSIsIkVudGl0eU1hbmFnZXIiLCJOYWl0ZSIsImFzc2VydERlZmluZWQiLCJjaHVuayIsIm5vbk51bGxhYmxlIiwiYmF0Y2hVcGRhdGUiLCJsb2dnZXIiLCJpc1JlZkZpZWxkIiwiZmllbGQiLCJ1bmRlZmluZWQiLCJvZiIsInV1aWQiLCJVcHNlcnRCdWlsZGVyIiwidGFibGVzIiwiTWFwIiwiZ2V0VGFibGUiLCJ0YWJsZU5hbWUiLCJ0YWJsZSIsImdldCIsInRhYmxlU3BlYyIsImdldFRhYmxlU3BlYyIsInRhYmxlRGF0YSIsInJlZmVyZW5jZXMiLCJTZXQiLCJyb3dzIiwidW5pcXVlSW5kZXhlcyIsInVuaXF1ZXNNYXAiLCJqc29uQ29sdW1ucyIsInNldCIsImhhc1RhYmxlIiwiaGFzIiwicmVnaXN0ZXIiLCJyb3ciLCJ1bmlxdWVLZXlzIiwibWFwIiwidW5xSW5kZXgiLCJ1bmlxdWVLZXlBcnJheSIsImNvbHVtbnMiLCJ1bnFDb2wiLCJ2YWwiLCJuYW1lIiwibGVuZ3RoIiwiam9pbiIsImZpbHRlciIsImlzUmV1c2VkIiwidW5pcXVlS2V5IiwiT2JqZWN0IiwiZnJvbUVudHJpZXMiLCJlbnRyaWVzIiwicm93S2V5Iiwicm93VmFsdWUiLCJ1c2UiLCJhZGQiLCJpbmNsdWRlcyIsIkpTT04iLCJzdHJpbmdpZnkiLCJwdXNoIiwicmVzdWx0IiwidCIsImlzVXVpZFJldXNlZCIsInVwc2VydCIsIndkYiIsIm9wdGlvbnMiLCJ1cHNlcnRPckluc2VydCIsImluc2VydE9ubHkiLCJtb2RlIiwiRXJyb3IiLCJzb21lIiwidmFsdWUiLCJyZWZUYWJsZXMiLCJBcnJheSIsImZyb20iLCJyZWR1Y2UiLCJyIiwicmVmZXJlbmNlIiwidmFsdWVzIiwiZmluZCIsInJlZiIsImV4dHJhY3RGaWVsZHMiLCJzcGxpdCIsImxldmVscyIsImhhc0NpcmN1bGFyIiwiYnVpbGRJbnNlcnRMZXZlbHMiLCJ1dWlkTWFwIiwiYWxsSWRzIiwibGV2ZWxJZHgiLCJsZXZlbFJvd3MiLCJkZWJ1ZyIsImN1cnJlbnQiLCJ0b3RhbCIsInJlc29sdmVkUm93cyIsInJlc29sdmVkIiwia2V5IiwicGFyZW50IiwidG8iLCJjaHVua1NpemUiLCJsZXZlbENodW5rcyIsInNlbGVjdEZpZWxkcyIsImluZGV4IiwiZGF0YUNodW5rIiwib3JpZ2luYWxVdWlkcyIsImRhdGFGb3JEYiIsInJlc3QiLCJyZXN1bHRSb3dzIiwiaW5zZXJ0IiwiaW50byIsInJldHVybmluZyIsInJvd3NXaXRob3V0SWQiLCJpZCIsInVuaXF1ZUluZGV4IiwiYyIsImNvbmRpdGlvbnMiLCJjb2wiLCJ2IiwiZXhpc3RpbmdSb3dzIiwid2hlcmVJbiIsInNlbGVjdCIsImV4aXN0aW5nTWFwIiwiZXhpc3RpbmciLCJTdHJpbmciLCJleGlzdGluZ0lkIiwiY29uZmxpY3RDb2x1bW5zIiwiYWxsQ29sdW1ucyIsImtleXMiLCJ1cGRhdGVDb2x1bW5zIiwiaW5oZXJpdCIsImluaGVyaXRDb2x1bW5zIiwiZXhjbHVkZWRGcm9tVXBkYXRlIiwibWVyZ2VDb2x1bW5zIiwib25Db25mbGljdCIsIm1lcmdlIiwiaSIsInByb3AiLCJjb25zb2xlIiwiZXJyb3IiLCJyZXNvbHZlZFZhbHVlIiwiY2xlYW5PcnBoYW5zIiwiZmtDb2x1bW5zIiwiZmtDb25kaXRpb25zIiwiZmtDb2wiLCJma1ZhbHVlcyIsImNvbHVtbiIsImV2ZXJ5IiwiZmMiLCJkZWxldGVRdWVyeSIsIndoZXJlTm90SW4iLCJkZWxldGVkQ291bnQiLCJkZWxldGUiLCJjbGVhciIsInJvd0NvdW50IiwicmV0dXJuZWRJZHMiLCJ1cGRhdGVCYXRjaCIsIndoZXJlIiwid2hlcmVDb2x1bW5zIiwiX3JvdyIsIl8iLCJoYXNTZWxmUmVmIiwiZmxhdE1hcCIsInJvd0J5VXVpZCIsInBlbmRpbmciLCJpbnNlcnRlZCIsImN1cnJlbnRMZXZlbCIsIm5leHRQZW5kaW5nIiwic2VsZlJlZnMiLCJjYW5JbnNlcnQiXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFNBQVMsUUFBUSxtQkFBbUI7QUFDN0MsU0FBU0MsVUFBVSxRQUFRLFNBQVM7QUFFcEMsU0FBU0MsT0FBTyxFQUFFQyxNQUFNLFFBQVEsVUFBVTtBQUMxQyxTQUFTQyxhQUFhLFFBQVEsOEJBQTJCO0FBQ3pELFNBQVNDLEtBQUssUUFBUSxvQkFBaUI7QUFFdkMsU0FBU0MsYUFBYSxFQUFFQyxLQUFLLEVBQUVDLFdBQVcsUUFBUSxvQkFBaUI7QUFDbkUsU0FBU0MsV0FBVyxRQUF3QixxQkFBa0I7QUFHOUQsTUFBTUMsU0FBU1YsVUFBVTtJQUFDO0lBQVU7SUFBWTtDQUFpQjtBQXVDakUsT0FBTyxTQUFTVyxXQUFXQyxLQUFjO0lBQ3ZDLE9BQ0VBLFVBQVVDLGFBQ1ZELFVBQVUsUUFDVixBQUFDQSxPQUFpQkUsT0FBT0QsYUFDekIsQUFBQ0QsT0FBaUJHLFNBQVNGO0FBRS9CO0FBRUEsT0FBTyxNQUFNRztJQUNYQyxPQUErQjtJQUMvQixhQUFjO1FBQ1osSUFBSSxDQUFDQSxNQUFNLEdBQUcsSUFBSUM7SUFDcEI7SUFFQUMsU0FBU0MsU0FBaUIsRUFBYTtRQUNyQyxNQUFNQyxRQUFRLElBQUksQ0FBQ0osTUFBTSxDQUFDSyxHQUFHLENBQUNGO1FBQzlCLElBQUlDLE9BQU87WUFDVCxPQUFPQTtRQUNUO1FBRUEsTUFBTUUsWUFBWSxBQUFDLENBQUE7WUFDakIsSUFBSTtnQkFDRixPQUFPbkIsY0FBY29CLFlBQVksQ0FBQ0o7WUFDcEMsRUFBRSxPQUFNO2dCQUNOLE9BQU87WUFDVDtRQUNGLENBQUE7UUFFQSxNQUFNSyxZQUFZO1lBQ2hCQyxZQUFZLElBQUlDO1lBQ2hCQyxNQUFNLEVBQUU7WUFDUkMsZUFBZU4sV0FBV00saUJBQWlCLEVBQUU7WUFDN0NDLFlBQVksSUFBSVo7WUFDaEJhLGFBQWFSLFdB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
|
|
406
|
+
//#endregion
|
|
407
|
+
init_upsert_builder();
|
|
408
|
+
export { UpsertBuilder, init_upsert_builder, isRefField };
|
|
409
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBzZXJ0LWJ1aWxkZXIuanMiLCJuYW1lcyI6WyJyZXN1bHQ6IFVCUmVmIiwidGFibGUiLCJhbGxJZHM6IChudW1iZXIgfCBzdHJpbmcpW10iLCJyZXN1bHRSb3dzOiB7IGlkOiBudW1iZXIgfCBzdHJpbmc7IFtrZXk6IHN0cmluZ106IHVua25vd24gfVtdIiwiY29uZGl0aW9uczogdW5rbm93bltdW10iLCJsZXZlbHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W11bXSIsImN1cnJlbnRMZXZlbDogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSIsIm5leHRQZW5kaW5nOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RhdGFiYXNlL3Vwc2VydC1idWlsZGVyLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJhbmRvbVVVSUQgfSBmcm9tIFwiY3J5cHRvXCI7XG5cbmltcG9ydCB7IGdldExvZ2dlciB9IGZyb20gXCJAbG9ndGFwZS9sb2d0YXBlXCI7XG5pbXBvcnQgeyB0eXBlIEtuZXggfSBmcm9tIFwia25leFwiO1xuaW1wb3J0IHsgY2x1c3RlciwgaXNBcnJheSwgdW5pcXVlIH0gZnJvbSBcInJhZGFzaGlcIjtcblxuaW1wb3J0IHsgRW50aXR5TWFuYWdlciB9IGZyb20gXCIuLi9lbnRpdHkvZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB7IE5haXRlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQge1xuICB0eXBlIERhdGFiYXNlRm9yZWlnbktleXMsXG4gIHR5cGUgRGF0YWJhc2VTY2hlbWFFeHRlbmQsXG4gIHR5cGUgRW50aXR5SW5kZXgsXG59IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgYXNzZXJ0RGVmaW5lZCwgbm9uTnVsbGFibGUgfSBmcm9tIFwiLi4vdXRpbHMvdXRpbHNcIjtcbmltcG9ydCB7IGJhdGNoVXBkYXRlIH0gZnJvbSBcIi4vX2JhdGNoX3VwZGF0ZVwiO1xuaW1wb3J0IHsgdHlwZSBSb3dXaXRoSWQgfSBmcm9tIFwiLi9fYmF0Y2hfdXBkYXRlXCI7XG5pbXBvcnQgeyB0eXBlIENvbHVtbktleXMsIHR5cGUgRm9yZWlnbktleUNvbHVtbnMsIHR5cGUgSWRUeXBlLCB0eXBlIFRhYmxlTmFtZSB9IGZyb20gXCIuL3B1cmkudHlwZXNcIjtcblxuY29uc3QgbG9nZ2VyID0gZ2V0TG9nZ2VyKFtcInNvbmFtdVwiLCBcImludGVybmFsXCIsIFwidXBzZXJ0LWJ1aWxkZXJcIl0pO1xuXG4vKipcbiAqIEZLIO2DgOyehSDstpTroaDsnYQg7JyE7ZW0IERhdGFiYXNlRm9yZWlnbktleXMgZXhwb3J0XG4gKiAobW9kdWxlIGF1Z21lbnRhdGlvbiDsnpDrj5kg66Gc65OcIOuztOyepSlcbiAqL1xuZXhwb3J0IHR5cGUgeyBEYXRhYmFzZUZvcmVpZ25LZXlzIH07XG5cbnR5cGUgSW5oZXJpdGFibGVDb2x1bW5zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID1cbiAgVFRhYmxlIGV4dGVuZHMga2V5b2YgRGF0YWJhc2VTY2hlbWFFeHRlbmQgPyBDb2x1bW5LZXlzPERhdGFiYXNlU2NoZW1hRXh0ZW5kW1RUYWJsZV0+IDogbmV2ZXI7XG5cbi8vIO2FjOydtOu4lCDrjbDsnbTthLAg7YOA7J6FXG50eXBlIFRhYmxlRGF0YSA9IHtcbiAgcmVmZXJlbmNlczogU2V0PHN0cmluZz47XG4gIHJvd3M6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W107XG4gIHVuaXF1ZUluZGV4ZXM6IEVudGl0eUluZGV4W107XG4gIHVuaXF1ZXNNYXA6IE1hcDxzdHJpbmcsIHN0cmluZz47XG4gIGpzb25Db2x1bW5zOiBzdHJpbmdbXTtcbn07XG5cbi8vIOywuOyhsCDtlYTrk5wg7YOA7J6FXG5leHBvcnQgdHlwZSBVQlJlZiA9IHtcbiAgdXVpZDogc3RyaW5nO1xuICBvZjogc3RyaW5nO1xuICB1c2U/OiBzdHJpbmc7XG59O1xuXG4vLyB1cHNlcnQg7Ji17IWYXG5leHBvcnQgdHlwZSBVcHNlcnRPcHRpb25zPFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+ID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG4gIGNsZWFuT3JwaGFucz86IEZvcmVpZ25LZXlDb2x1bW5zPFRUYWJsZT4gfCBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W107XG4gIGluaGVyaXQ/OiBJbmhlcml0YWJsZUNvbHVtbnM8VFRhYmxlPltdO1xufTtcblxuLy8gaW5zZXJ0T25seSDsmLXshZhcbmV4cG9ydCB0eXBlIEluc2VydE9ubHlPcHRpb25zID0ge1xuICBjaHVua1NpemU/OiBudW1iZXI7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gaXNSZWZGaWVsZChmaWVsZDogdW5rbm93bik6IGZpZWxkIGlzIFVCUmVmIHtcbiAgcmV0dXJuIChcbiAgICBmaWVsZCAhPT0gdW5kZWZpbmVkICYmXG4gICAgZmllbGQgIT09IG51bGwgJiZcbiAgICAoZmllbGQgYXMgVUJSZWYpPy5vZiAhPT0gdW5kZWZpbmVkICYmXG4gICAgKGZpZWxkIGFzIFVCUmVmKT8udXVpZCAhPT0gdW5kZWZpbmVkXG4gICk7XG59XG5cbmV4cG9ydCBjbGFzcyBVcHNlcnRCdWlsZGVyIHtcbiAgdGFibGVzOiBNYXA8c3RyaW5nLCBUYWJsZURhdGE+O1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnRhYmxlcyA9IG5ldyBNYXAoKTtcbiAgfVxuXG4gIGdldFRhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogVGFibGVEYXRhIHtcbiAgICBjb25zdCB0YWJsZSA9IHRoaXMudGFibGVzLmdldCh0YWJsZU5hbWUpO1xuICAgIGlmICh0YWJsZSkge1xuICAgICAgcmV0dXJuIHRhYmxlO1xuICAgIH1cblxuICAgIGNvbnN0IHRhYmxlU3BlYyA9ICgoKSA9PiB7XG4gICAgICB0cnkge1xuICAgICAgICByZXR1cm4gRW50aXR5TWFuYWdlci5nZXRUYWJsZVNwZWModGFibGVOYW1lKTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgIH1cbiAgICB9KSgpO1xuXG4gICAgY29uc3QgdGFibGVEYXRhID0ge1xuICAgICAgcmVmZXJlbmNlczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICByb3dzOiBbXSxcbiAgICAgIHVuaXF1ZUluZGV4ZXM6IHRhYmxlU3BlYz8udW5pcXVlSW5kZXhlcyA/PyBbXSxcbiAgICAgIHVuaXF1ZXNNYXA6IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+KCksXG4gICAgICBqc29uQ29sdW1uczogdGFibGVTcGVjPy5qc29uQ29sdW1ucyA/PyBbXSxcbiAgICB9O1xuICAgIHRoaXMudGFibGVzLnNldCh0YWJsZU5hbWUsIHRhYmxlRGF0YSk7XG4gICAgcmV0dXJuIHRhYmxlRGF0YTtcbiAgfVxuXG4gIGhhc1RhYmxlKHRhYmxlTmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMudGFibGVzLmhhcyh0YWJsZU5hbWUpO1xuICB9XG5cbiAgcmVnaXN0ZXI8VCBleHRlbmRzIHN0cmluZz4oXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICAgcm93OiB7XG4gICAgICBba2V5IGluIFRdPzogVUJSZWYgfCBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgYmlnaW50IHwgbnVsbCB8IG9iamVjdCB8IHVua25vd247XG4gICAgfSxcbiAgKTogVUJSZWYge1xuICAgIGNvbnN0IHRhYmxlID0gdGhpcy5nZXRUYWJsZSh0YWJsZU5hbWUpO1xuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCB1bmlxdWUg7J24642x7Iqk66W8IOyInO2ajO2VmOupsCDtgqQg7IOd7ISxXG4gICAgY29uc3QgdW5pcXVlS2V5cyA9IHRhYmxlLnVuaXF1ZUluZGV4ZXNcbiAgICAgIC5tYXAoKHVucUluZGV4KSA9PiB7XG4gICAgICAgIGNvbnN0IHVuaXF1ZUtleUFycmF5ID0gdW5xSW5kZXguY29sdW1ucy5tYXAoKHVucUNvbCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHZhbCA9IHJvd1t1bnFDb2wubmFtZSBhcyBrZXlvZiB0eXBlb2Ygcm93XTtcbiAgICAgICAgICBpZiAoaXNSZWZGaWVsZCh2YWwpKSB7XG4gICAgICAgICAgICByZXR1cm4gdmFsLnV1aWQ7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJldHVybiByb3dbdW5xQ29sLm5hbWUgYXMga2V5b2YgdHlwZW9mIHJvd10gPz8gcmFuZG9tVVVJRCgpOyAvLyBudWxsYWJsZeyduCDqsr3smrAgdXVpZOuhnCDrnpzrjaTqsJIg7IK97J6FXG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICAvLyDqsJLsnbQg66qo65GQIG51bGzsnbgg6rK97JqwIO2CpCDsg53shLEg7Yyo7IqkXG4gICAgICAgIGlmICh1bmlxdWVLZXlBcnJheS5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdW5pcXVlS2V5QXJyYXkuam9pbihcIi0tLWRlbGltaXRlci0tXCIpO1xuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgLy8gdXVpZCDsg53shLEg66Gc7KeBXG4gICAgY29uc3QgeyB1dWlkLCBpc1JldXNlZCB9ID0gKCgpID0+IHtcbiAgICAgIC8vIO2CpOulvCDsiJztmoztlZjsl6wg7J2066+4IOyhtOyerO2VmOuKlCDtgqTqsIAg7J6I64qU7KeAIO2ZleyduFxuICAgICAgaWYgKHVuaXF1ZUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgICAgaWYgKHRhYmxlLnVuaXF1ZXNNYXAuaGFzKHVuaXF1ZUtleSkpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIHV1aWQ6IGFzc2VydERlZmluZWQodGFibGUudW5pcXVlc01hcC5nZXQodW5pcXVlS2V5KSwgXCJVbmlxdWUga2V5IG5vdCBmb3VuZFwiKSxcbiAgICAgICAgICAgICAgaXNSZXVzZWQ6IHRydWUsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyDssL7snYQg7IiYIOyXhuuKlCDqsr3smrAg7IOd7ISxXG4gICAgICByZXR1cm4geyB1dWlkOiByYW5kb21VVUlEKCksIGlzUmV1c2VkOiBmYWxzZSB9O1xuICAgIH0pKCk7XG5cbiAgICAvLyDrqqjrk6Ag7Jyg64uI7YGs7YKk7JeQIOuMgO2VtCDsnKDri4jtgazrp7Xsl5AgdXVpZCDsoIDsnqVcbiAgICBpZiAodW5pcXVlS2V5cy5sZW5ndGggPiAwKSB7XG4gICAgICBmb3IgKGNvbnN0IHVuaXF1ZUtleSBvZiB1bmlxdWVLZXlzKSB7XG4gICAgICAgIHRhYmxlLnVuaXF1ZXNNYXAuc2V0KHVuaXF1ZUtleSwgdXVpZCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7J20IO2FjOydtOu4lOyXkCDsgqzsmqnrkJwgUmVmRmllbGTrpbwg7Iic7ZqM7ZWY7JesLCDtmITsnqwg7YWM7J2067iUIOygleuztOyXkCDslrTrlqQg7ZWE65Oc66W8IOywuOyhsO2VmOuKlOyngCDstpTqsIBcbiAgICAvLyDsnbQg7KCV67O066W8IOuCmOykkeyXkCDsuZjtmZjtlaAg65WMIOyCrOyaqVxuICAgIHJvdyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHJvdykubWFwKChbcm93S2V5LCByb3dWYWx1ZV0pID0+IHtcbiAgICAgICAgaWYgKGlzUmVmRmllbGQocm93VmFsdWUpKSB7XG4gICAgICAgICAgcm93VmFsdWUudXNlID8/PSBcImlkXCI7XG4gICAgICAgICAgdGFibGUucmVmZXJlbmNlcy5hZGQoYCR7cm93VmFsdWUub2Z9LiR7cm93VmFsdWUudXNlfWApO1xuICAgICAgICAgIHJldHVybiBbcm93S2V5LCByb3dWYWx1ZV07XG4gICAgICAgIH0gZWxzZSBpZiAodGFibGUuanNvbkNvbHVtbnMuaW5jbHVkZXMocm93S2V5KSAmJiByb3dWYWx1ZSAhPT0gbnVsbCkge1xuICAgICAgICAgIC8vIEpTT04g7Lus65+87J24IOqyveyasCBKU09OLnN0cmluZ2lmeSDsspjrpqwgKEtuZXjripQgSlNPTiDtg4DsnoXsnYQg7KeA7JuQ7ZWY7KeAIOyViuydjClcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgSlNPTi5zdHJpbmdpZnkocm93VmFsdWUpXTtcbiAgICAgICAgfSBlbHNlIGlmIChpc0FycmF5KHJvd1ZhbHVlKSkge1xuICAgICAgICAgIC8vIOuwsOyXtOydgCDqt7jrjIDroZwg7KCA7J6lXG4gICAgICAgICAgcmV0dXJuIFtyb3dLZXksIHJvd1ZhbHVlXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gW3Jvd0tleSwgcm93VmFsdWVdO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApIGFzIHsgW2tleSBpbiBUXT86IHVua25vd24gfTtcblxuICAgIHRhYmxlLnJvd3MucHVzaCh7XG4gICAgICB1dWlkLFxuICAgICAgLi4ucm93LFxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzdWx0OiBVQlJlZiA9IHtcbiAgICAgIG9mOiB0YWJsZU5hbWUsXG4gICAgICB1dWlkOiAocm93IGFzIHsgdXVpZD86IHN0cmluZyB9KS51dWlkID8/IHV1aWQsXG4gICAgfTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLXJlZ2lzdGVyXCIsIHtcbiAgICAgIHRhYmxlTmFtZSxcbiAgICAgIHV1aWQ6IHJlc3VsdC51dWlkLFxuICAgICAgaXNVdWlkUmV1c2VkOiBpc1JldXNlZCxcbiAgICAgIHJvdyxcbiAgICB9KTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyB1cHNlcnQ8VFRhYmxlIGV4dGVuZHMgVGFibGVOYW1lPERhdGFiYXNlU2NoZW1hRXh0ZW5kPj4oXG4gICAgd2RiOiBLbmV4LFxuICAgIHRhYmxlTmFtZTogVFRhYmxlLFxuICAgIG9wdGlvbnM/OiBVcHNlcnRPcHRpb25zPFRUYWJsZT4sXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJ1cHNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyBpbnNlcnRPbmx5PFRUYWJsZSBleHRlbmRzIFRhYmxlTmFtZTxEYXRhYmFzZVNjaGVtYUV4dGVuZD4+KFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IFRUYWJsZSxcbiAgICBvcHRpb25zPzogSW5zZXJ0T25seU9wdGlvbnMsXG4gICk6IFByb21pc2U8SWRUeXBlPERhdGFiYXNlU2NoZW1hRXh0ZW5kLCBUVGFibGU+W10+IHtcbiAgICByZXR1cm4gdGhpcy51cHNlcnRPckluc2VydCh3ZGIsIHRhYmxlTmFtZSwgXCJpbnNlcnRcIiwgb3B0aW9ucyk7XG4gIH1cblxuICBhc3luYyB1cHNlcnRPckluc2VydDxUVGFibGUgZXh0ZW5kcyBUYWJsZU5hbWU8RGF0YWJhc2VTY2hlbWFFeHRlbmQ+PihcbiAgICB3ZGI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBUVGFibGUsXG4gICAgbW9kZTogXCJ1cHNlcnRcIiB8IFwiaW5zZXJ0XCIsXG4gICAgb3B0aW9ucz86IFVwc2VydE9wdGlvbnM8VFRhYmxlPixcbiAgKTogUHJvbWlzZTxJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXT4ge1xuICAgIGlmICghdGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3QgdGFibGUgPSB0aGlzLnRhYmxlcy5nZXQodGFibGVOYW1lKTtcbiAgICBpZiAodGFibGUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBzZXJ0IOyalOyyrWApO1xuICAgIH0gZWxzZSBpZiAodGFibGUucm93cy5sZW5ndGggPT09IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgJHt0YWJsZU5hbWV97JeQIHVwc2VydCDtlaAg642w7J207YSw6rCAIOyXhuyKteuLiOuLpC5gKTtcbiAgICB9XG5cbiAgICBpZiAoXG4gICAgICB0YWJsZS5yb3dzLnNvbWUoKHJvdykgPT5cbiAgICAgICAgT2JqZWN0LmVudHJpZXMocm93KS5zb21lKChbLCB2YWx1ZV0pID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mICE9PSB0YWJsZU5hbWUpLFxuICAgICAgKVxuICAgICkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGAke3RhYmxlTmFtZX0g7ZW06rKw65CY7KeAIOyViuydgCDssLjsobDqsIAg7J6I7Iq164uI64ukLmApO1xuICAgIH1cblxuICAgIC8vIOyghOyytCDthYzsnbTruJQg7Iic7ZqM7ZWY7JesIO2YhOyerCDthYzsnbTruJQg7LC47KGw7ZWY64qUIOuqqOuToCDthYzsnbTruJQg7LaU7LacXG4gICAgY29uc3QgeyByZWZlcmVuY2VzLCByZWZUYWJsZXMgfSA9IEFycmF5LmZyb20odGhpcy50YWJsZXMpLnJlZHVjZShcbiAgICAgIChyLCBbLCB0YWJsZV0pID0+IHtcbiAgICAgICAgY29uc3QgcmVmZXJlbmNlID0gQXJyYXkuZnJvbSh0YWJsZS5yZWZlcmVuY2VzLnZhbHVlcygpKS5maW5kKChyZWYpID0+XG4gICAgICAgICAgcmVmLmluY2x1ZGVzKGAke3RhYmxlTmFtZX0uYCksXG4gICAgICAgICk7XG4gICAgICAgIGlmIChyZWZlcmVuY2UpIHtcbiAgICAgICAgICByLnJlZmVyZW5jZXMucHVzaChyZWZlcmVuY2UpO1xuICAgICAgICAgIHIucmVmVGFibGVzLnB1c2godGFibGUpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9LFxuICAgICAge1xuICAgICAgICByZWZlcmVuY2VzOiBbXSBhcyBzdHJpbmdbXSxcbiAgICAgICAgcmVmVGFibGVzOiBbXSBhcyBUYWJsZURhdGFbXSxcbiAgICAgIH0sXG4gICAgKTtcbiAgICBjb25zdCBleHRyYWN0RmllbGRzID0gdW5pcXVlKHJlZmVyZW5jZXMpXG4gICAgICAubWFwKChyZWZlcmVuY2UpID0+IHJlZmVyZW5jZS5zcGxpdChcIi5cIilbMV0pXG4gICAgICAuZmlsdGVyKChmaWVsZCk6IGZpZWxkIGlzIHN0cmluZyA9PiBmaWVsZCAhPT0gdW5kZWZpbmVkKTtcblxuICAgIC8vIOydmOyhtOyEsSDsiJzshJzsl5Ag65Sw6528IOugiOuyqOuzhCDqt7jro7ntmZQgKOyekOq4sCDssLjsobDqsIAg7JeG7Jy866m0IExldmVsIDAg7ZWY64KYKVxuICAgIGNvbnN0IHsgbGV2ZWxzLCBoYXNDaXJjdWxhciB9ID0gdGhpcy5idWlsZEluc2VydExldmVscyh0YWJsZS5yb3dzLCB0YWJsZU5hbWUpO1xuXG4gICAgaWYgKGhhc0NpcmN1bGFyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfeyXkCDsiJztmZgg7J6Q6riwIOywuOyhsOqwgCDsnojsirXri4jri6QuYCk7XG4gICAgfVxuXG4gICAgY29uc3QgdXVpZE1hcCA9IG5ldyBNYXA8c3RyaW5nLCB1bmtub3duPigpO1xuICAgIGNvbnN0IGFsbElkczogKG51bWJlciB8IHN0cmluZylbXSA9IFtdO1xuXG4gICAgLy8g66CI67Ko67OE66GcIOyInOywqCDsspjrpqxcbiAgICBmb3IgKGxldCBsZXZlbElkeCA9IDA7IGxldmVsSWR4IDwgbGV2ZWxzLmxlbmd0aDsgbGV2ZWxJZHgrKykge1xuICAgICAgY29uc3QgbGV2ZWxSb3dzID0gbGV2ZWxzW2xldmVsSWR4XTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhcIlByb2Nlc3NpbmcgUXVlcnkgTGV2ZWw6IHtjdXJyZW50fSAvIHt0b3RhbH1cIiwge1xuICAgICAgICBjdXJyZW50OiBsZXZlbElkeCArIDEsXG4gICAgICAgIHRvdGFsOiBsZXZlbHMubGVuZ3RoLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIOydtOyghCDroIjrsqjsl5DshJwg7Ja77J2AIElE66GcIOyekOq4sCDssLjsobAg7ZW06rKwXG4gICAgICBjb25zdCByZXNvbHZlZFJvd3MgPSBsZXZlbFJvd3MubWFwKChyb3cpID0+IHtcbiAgICAgICAgY29uc3QgcmVzb2x2ZWQgPSB7IC4uLnJvdyB9O1xuICAgICAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhyb3cpKSB7XG4gICAgICAgICAgaWYgKGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpIHtcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IHV1aWRNYXAuZ2V0KHZhbHVlLnV1aWQpO1xuXG4gICAgICAgICAgICBpZiAoIXBhcmVudCkgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHt2YWx1ZS51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcblxuICAgICAgICAgICAgcmVzb2x2ZWRba2V5XSA9IChwYXJlbnQgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW3ZhbHVlLnVzZSA/PyBcImlkXCJdO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHZhbHVlLm9mLCB1dWlkOiB2YWx1ZS51dWlkLCB1c2U6IHZhbHVlLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkW2tleV0sXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJlc29sdmVkO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIO2YhOyerCDroIjrsqggdXBzZXJ0XG4gICAgICBjb25zdCBjaHVua1NpemUgPSBvcHRpb25zPy5jaHVua1NpemU7XG4gICAgICBjb25zdCBsZXZlbENodW5rcyA9IGNodW5rU2l6ZSA/IGNsdXN0ZXIocmVzb2x2ZWRSb3dzLCBjaHVua1NpemUpIDogW3Jlc29sdmVkUm93c107XG4gICAgICBjb25zdCBzZWxlY3RGaWVsZHMgPSB1bmlxdWUoW1wiaWRcIiwgLi4uZXh0cmFjdEZpZWxkc10pO1xuXG4gICAgICBmb3IgKGxldCBpbmRleCA9IDA7IGluZGV4IDwgbGV2ZWxDaHVua3MubGVuZ3RoOyBpbmRleCsrKSB7XG4gICAgICAgIGNvbnN0IGRhdGFDaHVuayA9IGxldmVsQ2h1bmtzW2luZGV4XTtcbiAgICAgICAgaWYgKGRhdGFDaHVuay5sZW5ndGggPT09IDApIGNvbnRpbnVlO1xuICAgICAgICBsb2dnZXIuZGVidWcoXCJQcm9jZXNzaW5nIENodW5rOiB7Y3VycmVudH0gLyB7dG90YWx9XCIsIHtcbiAgICAgICAgICBjdXJyZW50OiBpbmRleCArIDEsXG4gICAgICAgICAgdG90YWw6IGxldmVsQ2h1bmtzLmxlbmd0aCxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gdXVpZOulvCDrs4Trj4TroZwg67O06rSA7ZWY6rOgLCBEQuyXkCDsoIDsnqXtlaAg642w7J207YSw7JeQ7IScIOygnOqxsFxuICAgICAgICBjb25zdCBvcmlnaW5hbFV1aWRzID0gZGF0YUNodW5rLm1hcCgocikgPT4gci51dWlkIGFzIHN0cmluZyk7XG4gICAgICAgIGNvbnN0IGRhdGFGb3JEYiA9IGRhdGFDaHVuay5tYXAoKHsgdXVpZCwgLi4ucmVzdCB9KSA9PiByZXN0KTtcblxuICAgICAgICBsZXQgcmVzdWx0Um93czogeyBpZDogbnVtYmVyIHwgc3RyaW5nOyBba2V5OiBzdHJpbmddOiB1bmtub3duIH1bXTtcblxuICAgICAgICBpZiAobW9kZSA9PT0gXCJpbnNlcnRcIikge1xuICAgICAgICAgIC8vIElOU0VSVCDrqqjrk5wgLSBSRVRVUk5JTkcg7IKs7JqpXG4gICAgICAgICAgcmVzdWx0Um93cyA9IGF3YWl0IHdkYi5pbnNlcnQoZGF0YUZvckRiKS5pbnRvKHRhYmxlTmFtZSkucmV0dXJuaW5nKHNlbGVjdEZpZWxkcyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gVVBTRVJUIOuqqOuTnCAtIGlkIOyXhuuKlCByb3frk6TsnZggaWTrpbwg7IKs7KCEIOyhsO2ajOuhnCDssYTsmrDquLBcbiAgICAgICAgICBjb25zdCByb3dzV2l0aG91dElkID0gZGF0YUZvckRiLmZpbHRlcigocm93KSA9PiAhcm93LmlkKTtcblxuICAgICAgICAgIGlmIChyb3dzV2l0aG91dElkLmxlbmd0aCA+IDAgJiYgdGFibGUudW5pcXVlSW5kZXhlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAvLyDrqqjrk6AgdW5pcXVlSW5kZXhlc+uhnCDquLDsobQg66CI7L2U65OcIOyhsO2ajFxuICAgICAgICAgICAgZm9yIChjb25zdCB1bmlxdWVJbmRleCBvZiB0YWJsZS51bmlxdWVJbmRleGVzKSB7XG4gICAgICAgICAgICAgIGNvbnN0IGNvbHVtbnMgPSB1bmlxdWVJbmRleC5jb2x1bW5zLm1hcCgoYykgPT4gYy5uYW1lKTtcblxuICAgICAgICAgICAgICAvLyDsobDtmoztlaAg7KGw6rG065OkIOy2lOy2nCAo6rCBIHJvd+ydmCB1bmlxdWUg7Lus65+8IOqwkuuTpClcbiAgICAgICAgICAgICAgY29uc3QgY29uZGl0aW9uczogdW5rbm93bltdW10gPSBbXTtcbiAgICAgICAgICAgICAgZm9yIChjb25zdCByb3cgb2Ygcm93c1dpdGhvdXRJZCkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHZhbHVlcyA9IGNvbHVtbnMubWFwKChjb2wpID0+IHJvd1tjb2xdKTtcbiAgICAgICAgICAgICAgICAvLyBudWxs7J20IO2PrO2VqOuQnCDsobDqsbTsnYAg7KCc7Jm4IChQb3N0Z3JlU1FMIFVOSVFVReuKlCBOVUxMIOustOyLnClcbiAgICAgICAgICAgICAgICBpZiAoIXZhbHVlcy5zb21lKCh2KSA9PiB2ID09IG51bGwpKSB7XG4gICAgICAgICAgICAgICAgICBjb25kaXRpb25zLnB1c2godmFsdWVzKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICBpZiAoY29uZGl0aW9ucy5sZW5ndGggPT09IDApIGNvbnRpbnVlO1xuXG4gICAgICAgICAgICAgIC8vIOuwsOy5mCBTRUxFQ1RcbiAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdSb3dzID0gKGF3YWl0IHdkYih0YWJsZU5hbWUpXG4gICAgICAgICAgICAgICAgLndoZXJlSW4oY29sdW1ucywgY29uZGl0aW9ucyBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdW10pXG4gICAgICAgICAgICAgICAgLnNlbGVjdChcImlkXCIsIC4uLmNvbHVtbnMpKSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdO1xuXG4gICAgICAgICAgICAgIC8vIE1hcCDsg53shLE6IHVuaXF1ZSDsu6zrn7wg7KGw7ZWpIOKGkiBpZFxuICAgICAgICAgICAgICBjb25zdCBleGlzdGluZ01hcCA9IG5ldyBNYXA8c3RyaW5nLCBudW1iZXIgfCBzdHJpbmc+KCk7XG4gICAgICAgICAgICAgIGZvciAoY29uc3QgZXhpc3Rpbmcgb2YgZXhpc3RpbmdSb3dzKSB7XG4gICAgICAgICAgICAgICAgY29uc3Qga2V5ID0gY29sdW1uc1xuICAgICAgICAgICAgICAgICAgLm1hcCgoY29sKSA9PiBTdHJpbmcoZXhpc3RpbmdbY29sXSA/PyBcIlwiKSlcbiAgICAgICAgICAgICAgICAgIC5qb2luKFwiLS0tZGVsaW1pdGVyLS0tXCIpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGlkID0gZXhpc3RpbmcuaWQ7XG4gICAgICAgICAgICAgICAgaWYgKHR5cGVvZiBpZCA9PT0gXCJudW1iZXJcIiB8fCB0eXBlb2YgaWQgPT09IFwic3RyaW5nXCIpIHtcbiAgICAgICAgICAgICAgICAgIGV4aXN0aW5nTWFwLnNldChrZXksIGlkKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAvLyBpZCDsl4bripQgcm9365Ok7JeQIOunpOy5reuQmOuKlCBpZCDssYTsmrDquLBcbiAgICAgICAgICAgICAgZm9yIChjb25zdCByb3cgb2Ygcm93c1dpdGhvdXRJZCkge1xuICAgICAgICAgICAgICAgIGlmIChyb3cuaWQpIGNvbnRpbnVlOyAvLyDsnbTrr7gg64uk66W4IHVuaXF1ZUluZGV47JeQ7IScIOyxhOybjOynhCDqsr3smrAg7Iqk7YK1XG5cbiAgICAgICAgICAgICAgICBjb25zdCBrZXkgPSBjb2x1bW5zLm1hcCgoY29sKSA9PiBTdHJpbmcocm93W2NvbF0gPz8gXCJcIikpLmpvaW4oXCItLS1kZWxpbWl0ZXItLS1cIik7XG4gICAgICAgICAgICAgICAgY29uc3QgZXhpc3RpbmdJZCA9IGV4aXN0aW5nTWFwLmdldChrZXkpO1xuXG4gICAgICAgICAgICAgICAgaWYgKGV4aXN0aW5nSWQpIHtcbiAgICAgICAgICAgICAgICAgIHJvdy5pZCA9IGV4aXN0aW5nSWQ7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gb25Db25mbGljdOuKlCBpZOunjCDsgqzsmqkgKOuqqOuToCB1bmlxdWVJbmRleGVz64qUIOydtOuvuCDsgqzsoIQg7KGw7ZqM66GcIOyymOumrOuQqClcbiAgICAgICAgICBjb25zdCBjb25mbGljdENvbHVtbnMgPSBbXCJpZFwiXTtcblxuICAgICAgICAgIGNvbnN0IGFsbENvbHVtbnMgPSBPYmplY3Qua2V5cyhkYXRhRm9yRGJbMF0pO1xuICAgICAgICAgIGxldCB1cGRhdGVDb2x1bW5zID0gYWxsQ29sdW1ucy5maWx0ZXIoKGMpID0+IGMgIT09IFwiaWRcIik7XG5cbiAgICAgICAgICAvLyBpbmhlcml0IOyYteyFmCDsspjrpqwgLSBpbmhlcml0IOy7rOufvOydgCB1cGRhdGUg64yA7IOB7JeQ7IScIOygnOyZuFxuICAgICAgICAgIGlmIChvcHRpb25zPy5pbmhlcml0Py5sZW5ndGgpIHtcbiAgICAgICAgICAgIGNvbnN0IGluaGVyaXRDb2x1bW5zID0gb3B0aW9ucy5pbmhlcml0IGFzIHN0cmluZ1tdO1xuXG4gICAgICAgICAgICBjb25zdCBleGNsdWRlZEZyb21VcGRhdGUgPSB1cGRhdGVDb2x1bW5zLmZpbHRlcigoYykgPT4gaW5oZXJpdENvbHVtbnMuaW5jbHVkZXMoYykpO1xuICAgICAgICAgICAgdXBkYXRlQ29sdW1ucyA9IHVwZGF0ZUNvbHVtbnMuZmlsdGVyKChjKSA9PiAhaW5oZXJpdENvbHVtbnMuaW5jbHVkZXMoYykpO1xuXG4gICAgICAgICAgICAvLyDsi6TsoJzroZwg7KCc7Jm465CcIOy7rOufvCDroZzquYVcbiAgICAgICAgICAgIGlmIChleGNsdWRlZEZyb21VcGRhdGUubGVuZ3RoKSB7XG4gICAgICAgICAgICAgIE5haXRlLnQoXCJwdXJpOnViLWluaGVyaXRcIiwge1xuICAgICAgICAgICAgICAgIHRhYmxlTmFtZSxcbiAgICAgICAgICAgICAgICBpbmhlcml0Q29sdW1ucyxcbiAgICAgICAgICAgICAgICBleGNsdWRlZEZyb21VcGRhdGUsXG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIHVwZGF0ZUNvbHVtbnPqsIAg67mE7Ja07J6I7Ja064+EIG1lcmdlKCnrpbwg7IKs7Jqp7ZWY7JesIOuqqOuToCDtlonsnbQgUkVUVVJOSU5H65CY64+E66GdIOuztOyepVxuICAgICAgICAgIGNvbnN0IG1lcmdlQ29sdW1ucyA9IHVwZGF0ZUNvbHVtbnMubGVuZ3RoID8gdXBkYXRlQ29sdW1ucyA6IGNvbmZsaWN0Q29sdW1ucztcblxuICAgICAgICAgIHJlc3VsdFJvd3MgPSBhd2FpdCB3ZGJcbiAgICAgICAgICAgIC5pbnNlcnQoZGF0YUZvckRiKVxuICAgICAgICAgICAgLmludG8odGFibGVOYW1lKVxuICAgICAgICAgICAgLm9uQ29uZmxpY3QoY29uZmxpY3RDb2x1bW5zKVxuICAgICAgICAgICAgLm1lcmdlKG1lcmdlQ29sdW1ucylcbiAgICAgICAgICAgIC5yZXR1cm5pbmcoc2VsZWN0RmllbGRzKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChvcmlnaW5hbFV1aWRzLmxlbmd0aCAhPT0gcmVzdWx0Um93cy5sZW5ndGgpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7dGFibGVOYW1lfTogcmVnaXN0ZXIvcmV0dXJuaW5nIOu2iOydvOy5mGApO1xuICAgICAgICB9XG5cbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCByZXN1bHRSb3dzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgdXVpZE1hcC5zZXQob3JpZ2luYWxVdWlkc1tpXSwgcmVzdWx0Um93c1tpXSk7XG4gICAgICAgICAgYWxsSWRzLnB1c2gocmVzdWx0Um93c1tpXS5pZCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyDtlbTri7kg7YWM7J2067iUIOywuOyhsOulvCDsi6TsoJwg67C466WY66GcIOuzgOqyvVxuICAgIGZvciAoY29uc3QgdGFibGUgb2YgcmVmVGFibGVzKSB7XG4gICAgICB0YWJsZS5yb3dzID0gdGFibGUucm93cy5tYXAoKHJvdykgPT4ge1xuICAgICAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyhyb3cpKSB7XG4gICAgICAgICAgY29uc3QgcHJvcCA9IHJvd1trZXldO1xuICAgICAgICAgIGlmIChpc1JlZkZpZWxkKHByb3ApICYmIHByb3Aub2YgPT09IHRhYmxlTmFtZSkge1xuICAgICAgICAgICAgY29uc3QgcGFyZW50ID0gdXVpZE1hcC5nZXQocHJvcC51dWlkKTtcbiAgICAgICAgICAgIGlmICghcGFyZW50KSB7XG4gICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IocHJvcCk7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihg7KG07J6s7ZWY7KeAIOyViuuKlCB1dWlkICR7cHJvcC51dWlkfSAtLSBpbiAke3RhYmxlTmFtZX1gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnN0IHJlc29sdmVkVmFsdWUgPSAocGFyZW50IGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+KVtwcm9wLnVzZSA/PyBcImlkXCJdO1xuICAgICAgICAgICAgcm93W2tleV0gPSByZXNvbHZlZFZhbHVlO1xuXG4gICAgICAgICAgICBOYWl0ZS50KFwicHVyaTp1Yi1yZWYtcmVzb2x2ZWRcIiwge1xuICAgICAgICAgICAgICB0YWJsZU5hbWUsXG4gICAgICAgICAgICAgIGZpZWxkOiBrZXksXG4gICAgICAgICAgICAgIGZyb206IHsgb2Y6IHByb3Aub2YsIHV1aWQ6IHByb3AudXVpZCwgdXNlOiBwcm9wLnVzZSA/PyBcImlkXCIgfSxcbiAgICAgICAgICAgICAgdG86IHJlc29sdmVkVmFsdWUsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJvdztcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIGlmIChvcHRpb25zPy5jbGVhbk9ycGhhbnMpIHtcbiAgICAgIGNvbnN0IGNsZWFuT3JwaGFucyA9IG9wdGlvbnMuY2xlYW5PcnBoYW5zO1xuICAgICAgY29uc3QgZmtDb2x1bW5zID0gaXNBcnJheShjbGVhbk9ycGhhbnMpXG4gICAgICAgID8gKGNsZWFuT3JwaGFucyBhcyBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+W10pXG4gICAgICAgIDogW2NsZWFuT3JwaGFucyBhcyBGb3JlaWduS2V5Q29sdW1uczxUVGFibGU+XTtcblxuICAgICAgLy8g7ZiE7J6sIHJlZ2lzdGVy65CcIOugiOy9lOuTnOuTpOydmCBGSyDqsJLrk6Qg7LaU7LacXG4gICAgICBjb25zdCBma0NvbmRpdGlvbnMgPSBma0NvbHVtbnMubWFwKChma0NvbCkgPT4ge1xuICAgICAgICBjb25zdCBma1ZhbHVlcyA9IFsuLi5uZXcgU2V0KHRhYmxlLnJvd3MubWFwKChyb3cpID0+IHJvd1tma0NvbF0pLmZpbHRlcigodikgPT4gdiAhPSBudWxsKSldO1xuICAgICAgICByZXR1cm4geyBjb2x1bW46IGZrQ29sLCB2YWx1ZXM6IGZrVmFsdWVzIH07XG4gICAgICB9KTtcblxuICAgICAgLy8g66qo65OgIEZLIOy7rOufvOyXkCDqsJLsnbQg7J6I64qUIOqyveyasOyXkOunjCDsgq3soJwg7Iuk7ZaJXG4gICAgICBpZiAoZmtDb25kaXRpb25zLmV2ZXJ5KChmYykgPT4gZmMudmFsdWVzLmxlbmd0aCA+IDApKSB7XG4gICAgICAgIGxldCBkZWxldGVRdWVyeSA9IHdkYih0YWJsZU5hbWUpO1xuXG4gICAgICAgIC8vIOqwgSBGSyDsu6zrn7zsl5Ag64yA7ZWcIFdIRVJFIElOIOyhsOqxtCDstpTqsIBcbiAgICAgICAgZm9yIChjb25zdCB7IGNvbHVtbiwgdmFsdWVzIH0gb2YgZmtDb25kaXRpb25zKSB7XG4gICAgICAgICAgZGVsZXRlUXVlcnkgPSBkZWxldGVRdWVyeS53aGVyZUluKGNvbHVtbiwgdmFsdWVzKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIOuwqeq4iCB1cHNlcnTtlZwgSUTripQg7KCc7Jm4XG4gICAgICAgIGRlbGV0ZVF1ZXJ5ID0gZGVsZXRlUXVlcnkud2hlcmVOb3RJbihcImlkXCIsIGFsbElkcyk7XG5cbiAgICAgICAgY29uc3QgZGVsZXRlZENvdW50ID0gYXdhaXQgZGVsZXRlUXVlcnkuZGVsZXRlKCk7XG5cbiAgICAgICAgTmFpdGUudChcInB1cmk6dWItY2xlYW4tb3JwaGFuc1wiLCB7XG4gICAgICAgICAgdGFibGVOYW1lLFxuICAgICAgICAgIGNsZWFuT3JwaGFuczogZmtDb2x1bW5zLFxuICAgICAgICAgIGRlbGV0ZWRDb3VudCxcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZW064u5IO2FjOydtOu4lOydmCDrjbDsnbTthLAg7LSI6riw7ZmUXG4gICAgdGFibGUucm93cyA9IFtdO1xuICAgIHRhYmxlLnJlZmVyZW5jZXMuY2xlYXIoKTtcbiAgICB0YWJsZS51bmlxdWVzTWFwLmNsZWFyKCk7XG5cbiAgICBOYWl0ZS50KFwicHVyaTp1Yi11cHNlcnRlZFwiLCB7XG4gICAgICB0YWJsZU5hbWUsXG4gICAgICBtb2RlLFxuICAgICAgcm93Q291bnQ6IGFsbElkcy5sZW5ndGgsXG4gICAgICByZXR1cm5lZElkczogYWxsSWRzLFxuICAgIH0pO1xuXG4gICAgcmV0dXJuIGFsbElkcyBhcyBJZFR5cGU8RGF0YWJhc2VTY2hlbWFFeHRlbmQsIFRUYWJsZT5bXTtcbiAgfVxuXG4gIGFzeW5jIHVwZGF0ZUJhdGNoKFxuICAgIHdkYjogS25leCxcbiAgICB0YWJsZU5hbWU6IHN0cmluZyxcbiAgICBvcHRpb25zPzoge1xuICAgICAgY2h1bmtTaXplPzogbnVtYmVyO1xuICAgICAgd2hlcmU/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgICB9LFxuICApOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBvcHRpb25zID0ge1xuICAgICAgLi4ub3B0aW9ucyxcbiAgICAgIGNodW5rU2l6ZTogb3B0aW9ucz8uY2h1bmtTaXplID8/IDUwMCxcbiAgICAgIHdoZXJlOiBvcHRpb25zPy53aGVyZSA/PyBcImlkXCIsXG4gICAgfTtcblxuICAgIGlmICghdGhpcy5oYXNUYWJsZSh0YWJsZU5hbWUpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGNvbnN0IHRhYmxlID0gdGhpcy50YWJsZXMuZ2V0KHRhYmxlTmFtZSk7XG4gICAgaWYgKCF0YWJsZSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGDrk7HroZ3rkJjsp4Ag7JWK7J2AIO2FjOydtOu4lCAke3RhYmxlTmFtZX3sl5AgdXBkYXRlQmF0Y2gg7JqU7LKtYCk7XG4gICAgfSBlbHNlIGlmICh0YWJsZS5yb3dzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHdoZXJlQ29sdW1ucyA9IEFycmF5LmlzQXJyYXkob3B0aW9ucy53aGVyZSkgPyBvcHRpb25zLndoZXJlIDogW29wdGlvbnMud2hlcmUgPz8gXCJpZFwiXTtcbiAgICBjb25zdCByb3dzID0gdGFibGUucm93cy5tYXAoKF9yb3cpID0+IHtcbiAgICAgIGNvbnN0IHsgdXVpZDogXywgLi4ucm93IH0gPSBfcm93OyAvLyB1dWlkIOygnOyZuFxuICAgICAgcmV0dXJuIHJvdyBhcyBSb3dXaXRoSWQ8c3RyaW5nPjtcbiAgICB9KTtcblxuICAgIGF3YWl0IGJhdGNoVXBkYXRlKHdkYiwgdGFibGVOYW1lLCB3aGVyZUNvbHVtbnMsIHJvd3MsIG9wdGlvbnMuY2h1bmtTaXplKTtcblxuICAgIE5haXRlLnQoXCJwdXJpOnViLWJhdGNoLXVwZGF0ZWRcIiwge1xuICAgICAgdGFibGVOYW1lLFxuICAgICAgcm93Q291bnQ6IHJvd3MubGVuZ3RoLFxuICAgICAgd2hlcmVDb2x1bW5zLFxuICAgIH0pO1xuXG4gICAgLy8gdXBkYXRlQmF0Y2gg7JmE66OMIO2bhCDsspjrpqzrkJwg642w7J207YSwIOygnOqxsFxuICAgIHRhYmxlLnJvd3MgPSBbXTtcbiAgICB0YWJsZS5yZWZlcmVuY2VzLmNsZWFyKCk7XG4gICAgdGFibGUudW5pcXVlc01hcC5jbGVhcigpO1xuICB9XG5cbiAgLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICAvLyBQcml2YXRlIEhlbHBlciBNZXRob2RzXG4gIC8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuICAvKipcbiAgICogcm93c+ulvCDsnZjsobTshLEg7Iic7ISc7JeQIOuUsOudvCDroIjrsqjrs4TroZwg6re466O57ZmUXG4gICAqIC0g7J6Q6riwIOywuOyhsCDsl4bripQg6rK97JqwIDog66qo65OgIHJvd3PqsIAgTGV2ZWwgMFxuICAgKiAtIOyekOq4sCDssLjsobAg7J6I64qUIOqyveyasCA6IOyekOq4sCDssLjsobAg6rSA6rOE66W8IOychOyDgSDsoJXroKztlZjsl6wg66CI67Ko67OE66GcIOq3uOujue2ZlFxuICAgKi9cbiAgcHJpdmF0ZSBidWlsZEluc2VydExldmVscyhcbiAgICByb3dzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdLFxuICAgIHRhYmxlTmFtZTogc3RyaW5nLFxuICApOiB7IGxldmVsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXVtdOyBoYXNDaXJjdWxhcjogYm9vbGVhbiB9IHtcbiAgICAvLyAxLiDsnpDquLAg7LC47KGw6rCAIOyXhuycvOuptCDtlZwg66CI67Ko66GcIOyymOumrFxuICAgIGNvbnN0IGhhc1NlbGZSZWYgPSByb3dzXG4gICAgICAuZmxhdE1hcCgocm93KSA9PiBPYmplY3QudmFsdWVzKHJvdykpXG4gICAgICAuc29tZSgodmFsdWUpID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUpO1xuICAgIGlmICghaGFzU2VsZlJlZikgcmV0dXJuIHsgbGV2ZWxzOiBbcm93c10sIGhhc0NpcmN1bGFyOiBmYWxzZSB9O1xuXG4gICAgLy8gMi4gdXVpZCDihpIgcm93IOunpO2VkSAo7KSR67O1IHV1aWQg67Cp7KeAKVxuICAgIGNvbnN0IHJvd0J5VXVpZCA9IG5ldyBNYXA8c3RyaW5nLCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4oKTtcbiAgICBmb3IgKGNvbnN0IHJvdyBvZiByb3dzKSB7XG4gICAgICBjb25zdCB1dWlkID0gcm93LnV1aWQgYXMgc3RyaW5nIHwgdW5kZWZpbmVkO1xuICAgICAgaWYgKCF1dWlkKSB0aHJvdyBuZXcgRXJyb3IoYGJ1aWxkSW5zZXJ0TGV2ZWxzOiB1dWlk6rCAIOyXhuuKlCByb3cgLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICByb3dCeVV1aWQuc2V0KHV1aWQsIHJvdyk7XG4gICAgfVxuXG4gICAgbGV0IHBlbmRpbmcgPSBBcnJheS5mcm9tKHJvd0J5VXVpZC52YWx1ZXMoKSk7XG4gICAgY29uc3QgbGV2ZWxzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPltdW10gPSBbXTtcbiAgICBjb25zdCBpbnNlcnRlZCA9IG5ldyBTZXQ8c3RyaW5nPigpO1xuXG4gICAgLy8gMy4g66CI67Ko67OEIOu2hOulmFxuICAgIHdoaWxlIChwZW5kaW5nLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRMZXZlbDogUmVjb3JkPHN0cmluZywgdW5rbm93bj5bXSA9IFtdO1xuICAgICAgY29uc3QgbmV4dFBlbmRpbmc6IFJlY29yZDxzdHJpbmcsIHVua25vd24+W10gPSBbXTtcblxuICAgICAgZm9yIChjb25zdCByb3cgb2YgcGVuZGluZykge1xuICAgICAgICAvLyDsnbQgcm936rCAIOywuOyhsO2VmOuKlCDsnpDquLAg7LC47KGw65OkXG4gICAgICAgIGNvbnN0IHNlbGZSZWZzID0gT2JqZWN0LnZhbHVlcyhyb3cpLmZpbHRlcihcbiAgICAgICAgICAodmFsdWUpID0+IGlzUmVmRmllbGQodmFsdWUpICYmIHZhbHVlLm9mID09PSB0YWJsZU5hbWUsXG4gICAgICAgICkgYXMgVUJSZWZbXTtcblxuICAgICAgICAvLyDssLjsobDtlZjripQg66qo65OgIHV1aWTqsIAg7J2066+4IGluc2VydGVk7JeQIOyeiOyWtOyVvCDsnbTrsogg66CI67Ko7JeQIO2PrO2VqFxuICAgICAgICBjb25zdCBjYW5JbnNlcnQgPSBzZWxmUmVmcy5ldmVyeSgocmVmKSA9PiB7XG4gICAgICAgICAgaWYgKCFyb3dCeVV1aWQuaGFzKHJlZi51dWlkKSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHV1aWQgJHtyZWYudXVpZH0gLS0gaW4gJHt0YWJsZU5hbWV9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiBpbnNlcnRlZC5oYXMocmVmLnV1aWQpO1xuICAgICAgICB9KTtcblxuICAgICAgICBpZiAoY2FuSW5zZXJ0KSB7XG4gICAgICAgICAgY3VycmVudExldmVsLnB1c2gocm93KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBuZXh0UGVuZGluZy5wdXNoKHJvdyk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8g7Iic7ZmYIOywuOyhsCDqsJDsp4BcbiAgICAgIGlmIChjdXJyZW50TGV2ZWwubGVuZ3RoID09PSAwKSByZXR1cm4geyBsZXZlbHM6IFtdLCBoYXNDaXJjdWxhcjogdHJ1ZSB9O1xuXG4gICAgICAvLyDroIjrsqgg7ZmV7KCVICsgaW5zZXJ0ZWQg6rCx7IugXG4gICAgICBsZXZlbHMucHVzaChjdXJyZW50TGV2ZWwpO1xuICAgICAgZm9yIChjb25zdCByb3cgb2YgY3VycmVudExldmVsKSB7XG4gICAgICAgIGluc2VydGVkLmFkZChyb3cudXVpZCBhcyBzdHJpbmcpO1xuICAgICAgfVxuXG4gICAgICBwZW5kaW5nID0gbmV4dFBlbmRpbmc7XG4gICAgfVxuXG4gICAgcmV0dXJuIHsgbGV2ZWxzLCBoYXNDaXJjdWxhcjogZmFsc2UgfTtcbiAgfVxufVxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7O0FBeURBLFNBQWdCLFdBQVcsT0FBZ0M7QUFDekQsUUFDRSxVQUFVLGFBQ1YsVUFBVSxRQUNULE9BQWlCLE9BQU8sYUFDeEIsT0FBaUIsU0FBUzs7OztzQkF4RDBCO2FBQ2xCO2FBTXFCO3FCQUNkO0NBSXhDLFNBQVMsVUFBVTtFQUFDO0VBQVU7RUFBWTtFQUFpQixDQUFDO0NBZ0RyRCxnQkFBYixNQUEyQjtFQUN6QjtFQUNBLGNBQWM7QUFDWixRQUFLLFNBQVMsSUFBSSxLQUFLOztFQUd6QixTQUFTLFdBQThCO0dBQ3JDLE1BQU0sUUFBUSxLQUFLLE9BQU8sSUFBSSxVQUFVO0FBQ3hDLE9BQUksT0FBTztBQUNULFdBQU87O0dBR1QsTUFBTSxtQkFBbUI7QUFDdkIsUUFBSTtBQUNGLFlBQU8sY0FBYyxhQUFhLFVBQVU7WUFDdEM7QUFDTixZQUFPOztPQUVQO0dBRUosTUFBTSxZQUFZO0lBQ2hCLFlBQVksSUFBSSxLQUFh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
|