velocious 1.0.430 → 1.0.431
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/velocious.js +48 -0
- package/build/bin/velocious.js +39 -34
- package/build/index.js +1 -2
- package/build/src/application.js +214 -187
- package/build/src/authorization/ability.d.ts +24 -23
- package/build/src/authorization/ability.d.ts.map +1 -1
- package/build/src/authorization/ability.js +300 -252
- package/build/src/authorization/base-resource.d.ts +20 -26
- package/build/src/authorization/base-resource.d.ts.map +1 -1
- package/build/src/authorization/base-resource.js +136 -118
- package/build/src/background-jobs/client.js +47 -43
- package/build/src/background-jobs/cron-expression.js +166 -127
- package/build/src/background-jobs/forked-runner-child.js +47 -37
- package/build/src/background-jobs/job-record.js +10 -8
- package/build/src/background-jobs/job-registry.js +84 -72
- package/build/src/background-jobs/job-runner.js +81 -74
- package/build/src/background-jobs/job.js +72 -62
- package/build/src/background-jobs/json-socket.js +70 -65
- package/build/src/background-jobs/main.js +900 -841
- package/build/src/background-jobs/normalize-error.js +11 -12
- package/build/src/background-jobs/scheduler.js +247 -205
- package/build/src/background-jobs/socket-request.js +65 -60
- package/build/src/background-jobs/status-reporter.js +96 -86
- package/build/src/background-jobs/store.js +980 -862
- package/build/src/background-jobs/types.js +3 -2
- package/build/src/background-jobs/web/authorization.js +50 -38
- package/build/src/background-jobs/web/controller.js +268 -232
- package/build/src/background-jobs/web/index.js +40 -36
- package/build/src/background-jobs/web/path-matcher.js +48 -45
- package/build/src/background-jobs/web/registry.js +14 -9
- package/build/src/background-jobs/worker.js +639 -585
- package/build/src/beacon/client.js +293 -264
- package/build/src/beacon/in-process-broker.js +25 -20
- package/build/src/beacon/in-process-client.js +116 -104
- package/build/src/beacon/server.js +126 -110
- package/build/src/beacon/types.js +8 -2
- package/build/src/cli/base-command.js +57 -49
- package/build/src/cli/browser-cli.js +42 -37
- package/build/src/cli/commands/background-jobs-main.js +5 -5
- package/build/src/cli/commands/background-jobs-runner.js +5 -5
- package/build/src/cli/commands/background-jobs-worker.js +5 -5
- package/build/src/cli/commands/beacon.js +5 -5
- package/build/src/cli/commands/console.js +10 -10
- package/build/src/cli/commands/db/base-command.js +76 -71
- package/build/src/cli/commands/db/create.js +61 -53
- package/build/src/cli/commands/db/drop.js +71 -62
- package/build/src/cli/commands/db/migrate.js +15 -13
- package/build/src/cli/commands/db/reset.js +19 -16
- package/build/src/cli/commands/db/rollback.js +13 -12
- package/build/src/cli/commands/db/schema/dump.js +9 -9
- package/build/src/cli/commands/db/schema/load.js +9 -9
- package/build/src/cli/commands/db/seed.js +9 -9
- package/build/src/cli/commands/db/tenants/check.js +35 -32
- package/build/src/cli/commands/db/tenants/create.js +29 -26
- package/build/src/cli/commands/db/tenants/migrate.js +44 -40
- package/build/src/cli/commands/destroy/migration.js +5 -5
- package/build/src/cli/commands/generate/base-models.js +5 -5
- package/build/src/cli/commands/generate/frontend-models.js +9 -9
- package/build/src/cli/commands/generate/migration.js +5 -5
- package/build/src/cli/commands/generate/model.js +5 -5
- package/build/src/cli/commands/init.js +9 -7
- package/build/src/cli/commands/routes.js +6 -6
- package/build/src/cli/commands/run-script.js +9 -9
- package/build/src/cli/commands/runner.js +9 -9
- package/build/src/cli/commands/server.js +6 -6
- package/build/src/cli/commands/test.js +7 -6
- package/build/src/cli/index.js +141 -127
- package/build/src/cli/tenant-database-command-helper.js +185 -154
- package/build/src/cli/use-browser-cli.js +20 -15
- package/build/src/configuration-resolver.js +54 -47
- package/build/src/configuration-types.d.ts +21 -2
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +60 -3
- package/build/src/configuration.js +2547 -2240
- package/build/src/controller.js +407 -363
- package/build/src/current-configuration.js +12 -9
- package/build/src/current.js +75 -70
- package/build/src/database/annotations-async-hooks.js +22 -16
- package/build/src/database/annotations.js +18 -12
- package/build/src/database/drivers/base-column.js +179 -155
- package/build/src/database/drivers/base-columns-index.js +78 -69
- package/build/src/database/drivers/base-foreign-key.js +101 -89
- package/build/src/database/drivers/base-table.js +149 -124
- package/build/src/database/drivers/base.js +1489 -1306
- package/build/src/database/drivers/mssql/column.js +50 -39
- package/build/src/database/drivers/mssql/columns-index.js +3 -2
- package/build/src/database/drivers/mssql/connect-connection.js +9 -11
- package/build/src/database/drivers/mssql/foreign-key.js +9 -8
- package/build/src/database/drivers/mssql/index.js +587 -507
- package/build/src/database/drivers/mssql/options.js +75 -68
- package/build/src/database/drivers/mssql/query-parser.js +3 -2
- package/build/src/database/drivers/mssql/sql/alter-table.js +2 -2
- package/build/src/database/drivers/mssql/sql/create-database.js +31 -24
- package/build/src/database/drivers/mssql/sql/create-index.js +2 -2
- package/build/src/database/drivers/mssql/sql/create-table.js +2 -2
- package/build/src/database/drivers/mssql/sql/delete.js +16 -14
- package/build/src/database/drivers/mssql/sql/drop-database.js +31 -24
- package/build/src/database/drivers/mssql/sql/drop-table.js +2 -2
- package/build/src/database/drivers/mssql/sql/insert.js +2 -2
- package/build/src/database/drivers/mssql/sql/update.js +28 -24
- package/build/src/database/drivers/mssql/sql/upsert.js +20 -18
- package/build/src/database/drivers/mssql/structure-sql.js +114 -102
- package/build/src/database/drivers/mssql/table.js +96 -81
- package/build/src/database/drivers/mysql/column.js +92 -75
- package/build/src/database/drivers/mysql/columns-index.js +19 -16
- package/build/src/database/drivers/mysql/foreign-key.js +9 -8
- package/build/src/database/drivers/mysql/index.js +457 -396
- package/build/src/database/drivers/mysql/options.js +30 -26
- package/build/src/database/drivers/mysql/query-parser.js +3 -2
- package/build/src/database/drivers/mysql/query.js +29 -26
- package/build/src/database/drivers/mysql/sql/alter-table.js +3 -2
- package/build/src/database/drivers/mysql/sql/create-database.js +28 -23
- package/build/src/database/drivers/mysql/sql/create-index.js +3 -2
- package/build/src/database/drivers/mysql/sql/create-table.js +3 -2
- package/build/src/database/drivers/mysql/sql/delete.js +17 -14
- package/build/src/database/drivers/mysql/sql/drop-database.js +3 -2
- package/build/src/database/drivers/mysql/sql/drop-table.js +3 -2
- package/build/src/database/drivers/mysql/sql/insert.js +3 -2
- package/build/src/database/drivers/mysql/sql/update.js +29 -24
- package/build/src/database/drivers/mysql/sql/upsert.js +10 -8
- package/build/src/database/drivers/mysql/structure-sql.js +88 -79
- package/build/src/database/drivers/mysql/table.js +98 -83
- package/build/src/database/drivers/pgsql/column.js +72 -56
- package/build/src/database/drivers/pgsql/columns-index.js +3 -2
- package/build/src/database/drivers/pgsql/foreign-key.js +9 -8
- package/build/src/database/drivers/pgsql/index.js +438 -377
- package/build/src/database/drivers/pgsql/options.js +28 -25
- package/build/src/database/drivers/pgsql/query-parser.js +3 -2
- package/build/src/database/drivers/pgsql/sql/alter-table.js +3 -2
- package/build/src/database/drivers/pgsql/sql/create-database.js +23 -19
- package/build/src/database/drivers/pgsql/sql/create-index.js +3 -2
- package/build/src/database/drivers/pgsql/sql/create-table.js +3 -2
- package/build/src/database/drivers/pgsql/sql/delete.js +17 -14
- package/build/src/database/drivers/pgsql/sql/drop-database.js +3 -2
- package/build/src/database/drivers/pgsql/sql/drop-table.js +3 -2
- package/build/src/database/drivers/pgsql/sql/insert.js +3 -2
- package/build/src/database/drivers/pgsql/sql/update.js +29 -24
- package/build/src/database/drivers/pgsql/sql/upsert.js +11 -9
- package/build/src/database/drivers/pgsql/structure-sql.js +120 -108
- package/build/src/database/drivers/pgsql/table.js +77 -60
- package/build/src/database/drivers/sqlite/base.js +478 -405
- package/build/src/database/drivers/sqlite/column.js +69 -54
- package/build/src/database/drivers/sqlite/columns-index.js +27 -22
- package/build/src/database/drivers/sqlite/connection-sql-js.js +42 -35
- package/build/src/database/drivers/sqlite/foreign-key.js +21 -18
- package/build/src/database/drivers/sqlite/index.js +373 -330
- package/build/src/database/drivers/sqlite/index.native.js +64 -55
- package/build/src/database/drivers/sqlite/index.web.js +87 -69
- package/build/src/database/drivers/sqlite/options.js +28 -25
- package/build/src/database/drivers/sqlite/query-parser.js +3 -2
- package/build/src/database/drivers/sqlite/query.js +24 -21
- package/build/src/database/drivers/sqlite/query.native.js +25 -20
- package/build/src/database/drivers/sqlite/query.web.js +37 -30
- package/build/src/database/drivers/sqlite/sql/alter-table.js +179 -159
- package/build/src/database/drivers/sqlite/sql/create-index.js +3 -2
- package/build/src/database/drivers/sqlite/sql/create-table.js +3 -2
- package/build/src/database/drivers/sqlite/sql/delete.js +22 -17
- package/build/src/database/drivers/sqlite/sql/drop-table.js +3 -2
- package/build/src/database/drivers/sqlite/sql/insert.js +3 -2
- package/build/src/database/drivers/sqlite/sql/update.js +29 -24
- package/build/src/database/drivers/sqlite/sql/upsert.js +11 -9
- package/build/src/database/drivers/sqlite/structure-sql.js +52 -49
- package/build/src/database/drivers/sqlite/table-rebuilder.js +75 -62
- package/build/src/database/drivers/sqlite/table.js +125 -102
- package/build/src/database/drivers/structure-sql/utils.js +17 -14
- package/build/src/database/handler.js +10 -9
- package/build/src/database/initializer-from-require-context.js +87 -76
- package/build/src/database/migration/index.js +395 -332
- package/build/src/database/migrator/files-finder.js +50 -40
- package/build/src/database/migrator/types.js +30 -2
- package/build/src/database/migrator.js +526 -454
- package/build/src/database/pool/async-tracked-multi-connection.js +1147 -997
- package/build/src/database/pool/base-methods-forward.js +43 -40
- package/build/src/database/pool/base.js +343 -298
- package/build/src/database/pool/single-multi-use.js +110 -93
- package/build/src/database/query/alter-table-base.js +99 -84
- package/build/src/database/query/base.js +46 -39
- package/build/src/database/query/create-database-base.js +30 -25
- package/build/src/database/query/create-index-base.js +94 -75
- package/build/src/database/query/create-table-base.js +193 -151
- package/build/src/database/query/delete-base.js +16 -14
- package/build/src/database/query/drop-database-base.js +28 -23
- package/build/src/database/query/drop-table-base.js +53 -42
- package/build/src/database/query/from-base.js +33 -30
- package/build/src/database/query/from-plain.js +13 -11
- package/build/src/database/query/from-table.js +15 -13
- package/build/src/database/query/index.js +472 -410
- package/build/src/database/query/insert-base.js +164 -143
- package/build/src/database/query/join-base.js +40 -35
- package/build/src/database/query/join-object.js +153 -128
- package/build/src/database/query/join-plain.js +15 -13
- package/build/src/database/query/join-tracker.js +90 -76
- package/build/src/database/query/model-class-query.js +1370 -1134
- package/build/src/database/query/order-base.js +30 -27
- package/build/src/database/query/order-column.js +53 -44
- package/build/src/database/query/order-plain.js +24 -20
- package/build/src/database/query/preloader/belongs-to.js +258 -210
- package/build/src/database/query/preloader/ensure-model-class-initialized.js +9 -8
- package/build/src/database/query/preloader/has-many.js +301 -240
- package/build/src/database/query/preloader/has-one.js +117 -91
- package/build/src/database/query/preloader/selection.js +129 -117
- package/build/src/database/query/preloader.js +185 -160
- package/build/src/database/query/query-data.js +201 -157
- package/build/src/database/query/select-base.js +27 -25
- package/build/src/database/query/select-plain.js +15 -13
- package/build/src/database/query/select-table-and-column.js +25 -21
- package/build/src/database/query/update-base.js +38 -35
- package/build/src/database/query/upsert-base.js +100 -93
- package/build/src/database/query/where-base.js +35 -32
- package/build/src/database/query/where-combinator.d.ts.map +1 -1
- package/build/src/database/query/where-combinator.js +28 -26
- package/build/src/database/query/where-hash.js +68 -61
- package/build/src/database/query/where-model-class-hash.js +469 -414
- package/build/src/database/query/where-not.js +20 -18
- package/build/src/database/query/where-plain.js +17 -15
- package/build/src/database/query/with-count.js +159 -125
- package/build/src/database/query-parser/base-query-parser.js +37 -32
- package/build/src/database/query-parser/from-parser.js +45 -36
- package/build/src/database/query-parser/group-parser.js +50 -42
- package/build/src/database/query-parser/joins-parser.js +33 -28
- package/build/src/database/query-parser/limit-parser.js +70 -67
- package/build/src/database/query-parser/options.js +82 -75
- package/build/src/database/query-parser/order-parser.js +40 -36
- package/build/src/database/query-parser/select-parser.js +60 -49
- package/build/src/database/query-parser/where-parser.js +41 -36
- package/build/src/database/record/acts-as-list.js +273 -235
- package/build/src/database/record/attachments/download.js +45 -44
- package/build/src/database/record/attachments/handle.js +161 -141
- package/build/src/database/record/attachments/normalize-input.js +138 -128
- package/build/src/database/record/attachments/storage-drivers/filesystem.js +91 -77
- package/build/src/database/record/attachments/storage-drivers/native.js +121 -112
- package/build/src/database/record/attachments/storage-drivers/s3.js +208 -177
- package/build/src/database/record/attachments/store.d.ts +1 -1
- package/build/src/database/record/attachments/store.d.ts.map +1 -1
- package/build/src/database/record/attachments/store.js +540 -468
- package/build/src/database/record/index.d.ts +17 -15
- package/build/src/database/record/index.d.ts.map +1 -1
- package/build/src/database/record/index.js +3894 -3361
- package/build/src/database/record/instance-relationships/base.js +268 -234
- package/build/src/database/record/instance-relationships/belongs-to.js +73 -58
- package/build/src/database/record/instance-relationships/has-many.js +264 -225
- package/build/src/database/record/instance-relationships/has-one.js +105 -85
- package/build/src/database/record/record-not-found-error.js +2 -3
- package/build/src/database/record/relationships/base.d.ts +2 -2
- package/build/src/database/record/relationships/base.d.ts.map +1 -1
- package/build/src/database/record/relationships/base.js +167 -145
- package/build/src/database/record/relationships/belongs-to.js +51 -44
- package/build/src/database/record/relationships/has-many.js +40 -32
- package/build/src/database/record/relationships/has-one.js +40 -32
- package/build/src/database/record/state-machine.js +208 -156
- package/build/src/database/record/user-module.js +38 -32
- package/build/src/database/record/validators/base.js +24 -22
- package/build/src/database/record/validators/format.js +46 -36
- package/build/src/database/record/validators/presence.js +20 -18
- package/build/src/database/record/validators/uniqueness.js +117 -99
- package/build/src/database/table-data/index.js +231 -199
- package/build/src/database/table-data/table-column.js +382 -338
- package/build/src/database/table-data/table-foreign-key.js +66 -57
- package/build/src/database/table-data/table-index.js +36 -29
- package/build/src/database/table-data/table-reference.js +10 -10
- package/build/src/database/use-database.js +40 -32
- package/build/src/environment-handlers/base.js +544 -484
- package/build/src/environment-handlers/browser.js +294 -241
- package/build/src/environment-handlers/node/cli/commands/background-jobs-main.js +19 -16
- package/build/src/environment-handlers/node/cli/commands/background-jobs-runner.js +21 -18
- package/build/src/environment-handlers/node/cli/commands/background-jobs-worker.js +29 -22
- package/build/src/environment-handlers/node/cli/commands/beacon.js +19 -16
- package/build/src/environment-handlers/node/cli/commands/cli-command-context.js +15 -14
- package/build/src/environment-handlers/node/cli/commands/console.js +120 -99
- package/build/src/environment-handlers/node/cli/commands/db/schema/dump.js +39 -34
- package/build/src/environment-handlers/node/cli/commands/db/schema/load.js +63 -57
- package/build/src/environment-handlers/node/cli/commands/db/seed.js +63 -51
- package/build/src/environment-handlers/node/cli/commands/destroy/migration.js +40 -32
- package/build/src/environment-handlers/node/cli/commands/generate/base-models.d.ts.map +1 -1
- package/build/src/environment-handlers/node/cli/commands/generate/base-models.js +353 -298
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +844 -729
- package/build/src/environment-handlers/node/cli/commands/generate/migration.js +38 -34
- package/build/src/environment-handlers/node/cli/commands/generate/model.js +38 -34
- package/build/src/environment-handlers/node/cli/commands/init.js +61 -56
- package/build/src/environment-handlers/node/cli/commands/routes.js +59 -51
- package/build/src/environment-handlers/node/cli/commands/run-script.js +68 -54
- package/build/src/environment-handlers/node/cli/commands/runner.js +74 -56
- package/build/src/environment-handlers/node/cli/commands/server.js +106 -93
- package/build/src/environment-handlers/node/cli/commands/test.js +113 -97
- package/build/src/environment-handlers/node.js +874 -753
- package/build/src/error-logger.js +21 -22
- package/build/src/frontend-model-controller.d.ts +6 -6
- package/build/src/frontend-model-controller.d.ts.map +1 -1
- package/build/src/frontend-model-controller.js +3288 -2788
- package/build/src/frontend-model-resource/base-resource.d.ts +18 -17
- package/build/src/frontend-model-resource/base-resource.d.ts.map +1 -1
- package/build/src/frontend-model-resource/base-resource.js +869 -759
- package/build/src/frontend-models/base.d.ts +19 -12
- package/build/src/frontend-models/base.d.ts.map +1 -1
- package/build/src/frontend-models/base.js +3602 -3114
- package/build/src/frontend-models/clear-pending-debounced-callback.js +8 -7
- package/build/src/frontend-models/event-hook-models.js +21 -16
- package/build/src/frontend-models/model-registry.js +11 -9
- package/build/src/frontend-models/outgoing-event-buffer.js +17 -10
- package/build/src/frontend-models/preloader.d.ts +6 -6
- package/build/src/frontend-models/preloader.d.ts.map +1 -1
- package/build/src/frontend-models/preloader.js +149 -131
- package/build/src/frontend-models/query.d.ts.map +1 -1
- package/build/src/frontend-models/query.js +1855 -1560
- package/build/src/frontend-models/resource-config-validation.js +37 -27
- package/build/src/frontend-models/resource-definition.js +288 -234
- package/build/src/frontend-models/transport-serialization.js +266 -203
- package/build/src/frontend-models/use-created-event.js +7 -5
- package/build/src/frontend-models/use-destroyed-event.js +93 -80
- package/build/src/frontend-models/use-model-class-event.js +91 -79
- package/build/src/frontend-models/use-updated-event.js +97 -84
- package/build/src/frontend-models/websocket-channel.js +441 -381
- package/build/src/frontend-models/websocket-publishers.js +173 -140
- package/build/src/http-client/header.js +14 -13
- package/build/src/http-client/index.js +132 -116
- package/build/src/http-client/request.js +87 -71
- package/build/src/http-client/response.js +140 -122
- package/build/src/http-client/websocket-client.js +17 -15
- package/build/src/http-server/client/index.js +465 -409
- package/build/src/http-server/client/params-to-object.js +135 -124
- package/build/src/http-server/client/request-buffer/form-data-part.js +132 -111
- package/build/src/http-server/client/request-buffer/header.js +16 -15
- package/build/src/http-server/client/request-buffer/index.js +506 -446
- package/build/src/http-server/client/request-parser.js +186 -163
- package/build/src/http-server/client/request-runner.js +259 -226
- package/build/src/http-server/client/request-timing.js +151 -132
- package/build/src/http-server/client/request.js +108 -96
- package/build/src/http-server/client/response.js +235 -213
- package/build/src/http-server/client/uploaded-file/memory-uploaded-file.js +29 -25
- package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.js +29 -25
- package/build/src/http-server/client/uploaded-file/uploaded-file.js +33 -33
- package/build/src/http-server/client/websocket-request.js +137 -114
- package/build/src/http-server/client/websocket-session.js +1657 -1452
- package/build/src/http-server/cookie.js +236 -216
- package/build/src/http-server/development-reloader.js +221 -190
- package/build/src/http-server/index.js +525 -451
- package/build/src/http-server/remote-address.js +50 -38
- package/build/src/http-server/server-client.js +208 -181
- package/build/src/http-server/server-lock.js +167 -153
- package/build/src/http-server/websocket-channel-subscribers.js +93 -81
- package/build/src/http-server/websocket-channel.js +117 -104
- package/build/src/http-server/websocket-connection.js +104 -96
- package/build/src/http-server/websocket-event-log-store.js +404 -350
- package/build/src/http-server/websocket-events-host.js +164 -145
- package/build/src/http-server/websocket-events.js +47 -47
- package/build/src/http-server/worker-handler/channel-subscriber-dispatch.js +14 -13
- package/build/src/http-server/worker-handler/in-process.js +141 -123
- package/build/src/http-server/worker-handler/index.js +349 -313
- package/build/src/http-server/worker-handler/worker-script.js +5 -4
- package/build/src/http-server/worker-handler/worker-thread.js +269 -240
- package/build/src/initializer.js +36 -31
- package/build/src/jobs/mail-delivery.js +15 -13
- package/build/src/logger/base-logger.js +26 -24
- package/build/src/logger/console-logger.js +23 -21
- package/build/src/logger/file-logger.js +31 -29
- package/build/src/logger/outputs/array-output.js +42 -37
- package/build/src/logger/outputs/console-output.js +24 -20
- package/build/src/logger/outputs/file-output.js +48 -43
- package/build/src/logger/outputs/stdout-output.js +48 -39
- package/build/src/logger.js +394 -338
- package/build/src/mailer/backends/smtp.js +163 -134
- package/build/src/mailer/base.js +251 -211
- package/build/src/mailer/delivery.js +64 -56
- package/build/src/mailer/index.js +22 -4
- package/build/src/mailer.js +13 -4
- package/build/src/plugins/sqljs-wasm-route-controller.js +52 -42
- package/build/src/plugins/sqljs-wasm-route.js +38 -28
- package/build/src/record-payload-values.js +28 -25
- package/build/src/routes/app-routes.js +14 -12
- package/build/src/routes/base-route.js +130 -112
- package/build/src/routes/basic-route.js +102 -83
- package/build/src/routes/built-in/debug/controller.js +10 -10
- package/build/src/routes/built-in/errors/controller.js +5 -5
- package/build/src/routes/get-route.js +63 -50
- package/build/src/routes/hooks/frontend-model-command-route-hook.js +80 -66
- package/build/src/routes/index.js +43 -36
- package/build/src/routes/namespace-route.js +47 -38
- package/build/src/routes/plugin-routes.js +124 -107
- package/build/src/routes/post-route.js +62 -51
- package/build/src/routes/resolver.js +494 -422
- package/build/src/routes/resource-route.js +143 -124
- package/build/src/routes/root-route.js +8 -7
- package/build/src/testing/base-expect.js +14 -13
- package/build/src/testing/browser-frontend-model-event-hook-scenarios.js +405 -329
- package/build/src/testing/browser-test-app.js +29 -23
- package/build/src/testing/expect-to-change.js +50 -41
- package/build/src/testing/expect-utils.js +184 -139
- package/build/src/testing/expect.js +731 -638
- package/build/src/testing/request-client.js +85 -70
- package/build/src/testing/test-files-finder.js +339 -285
- package/build/src/testing/test-filter-parser.js +155 -124
- package/build/src/testing/test-runner.js +1020 -883
- package/build/src/testing/test-suite-splitter.js +142 -114
- package/build/src/testing/test.js +256 -216
- package/build/src/utils/backtrace-cleaner-node.js +69 -62
- package/build/src/utils/backtrace-cleaner.js +216 -188
- package/build/src/utils/ensure-error.js +7 -7
- package/build/src/utils/event-emitter.js +6 -4
- package/build/src/utils/file-exists.js +10 -9
- package/build/src/utils/format-value.js +76 -67
- package/build/src/utils/model-scope.js +31 -27
- package/build/src/utils/nest-callbacks.js +13 -10
- package/build/src/utils/plain-object.js +6 -5
- package/build/src/utils/ransack.d.ts.map +1 -1
- package/build/src/utils/ransack.js +563 -449
- package/build/src/utils/rest-args-error.js +6 -5
- package/build/src/utils/singularize-model-name.js +11 -9
- package/build/src/utils/split-sql-statements.js +79 -68
- package/build/src/utils/to-import-specifier.js +30 -24
- package/build/src/utils/with-tracked-stack-async-hooks.js +74 -60
- package/build/src/utils/with-tracked-stack.js +18 -14
- package/build/src/velocious-error.js +30 -27
- package/index.js +1 -0
- package/package.json +10 -4
- package/scripts/clean-build.js +8 -0
- package/scripts/ensure-bin-executable.js +13 -0
- package/scripts/run-tests.js +37 -0
- package/scripts/test-browser.js +486 -0
- package/src/application.js +229 -0
- package/src/authorization/ability.js +329 -0
- package/src/authorization/base-resource.js +143 -0
- package/src/background-jobs/client.js +50 -0
- package/src/background-jobs/cron-expression.js +277 -0
- package/src/background-jobs/forked-runner-child.js +86 -0
- package/src/background-jobs/job-record.js +13 -0
- package/src/background-jobs/job-registry.js +92 -0
- package/src/background-jobs/job-runner.js +107 -0
- package/src/background-jobs/job.js +77 -0
- package/src/background-jobs/json-socket.js +78 -0
- package/src/background-jobs/main.js +926 -0
- package/src/background-jobs/normalize-error.js +26 -0
- package/src/background-jobs/scheduler.js +274 -0
- package/src/background-jobs/socket-request.js +68 -0
- package/src/background-jobs/status-reporter.js +101 -0
- package/src/background-jobs/store.js +994 -0
- package/src/background-jobs/types.js +70 -0
- package/src/background-jobs/web/authorization.js +89 -0
- package/src/background-jobs/web/controller.js +280 -0
- package/src/background-jobs/web/index.js +57 -0
- package/src/background-jobs/web/path-matcher.js +74 -0
- package/src/background-jobs/web/registry.js +49 -0
- package/src/background-jobs/worker.js +683 -0
- package/src/beacon/client.js +330 -0
- package/src/beacon/in-process-broker.js +71 -0
- package/src/beacon/in-process-client.js +139 -0
- package/src/beacon/server.js +148 -0
- package/src/beacon/types.js +55 -0
- package/src/cli/base-command.js +67 -0
- package/src/cli/browser-cli.js +45 -0
- package/src/cli/commands/background-jobs-main.js +7 -0
- package/src/cli/commands/background-jobs-runner.js +7 -0
- package/src/cli/commands/background-jobs-worker.js +7 -0
- package/src/cli/commands/beacon.js +7 -0
- package/src/cli/commands/console.js +12 -0
- package/src/cli/commands/db/base-command.js +82 -0
- package/src/cli/commands/db/create.js +64 -0
- package/src/cli/commands/db/drop.js +75 -0
- package/src/cli/commands/db/migrate.js +17 -0
- package/src/cli/commands/db/reset.js +22 -0
- package/src/cli/commands/db/rollback.js +15 -0
- package/src/cli/commands/db/schema/dump.js +12 -0
- package/src/cli/commands/db/schema/load.js +12 -0
- package/src/cli/commands/db/seed.js +12 -0
- package/src/cli/commands/db/tenants/check.js +38 -0
- package/src/cli/commands/db/tenants/create.js +33 -0
- package/src/cli/commands/db/tenants/migrate.js +49 -0
- package/src/cli/commands/destroy/migration.js +7 -0
- package/src/cli/commands/generate/base-models.js +7 -0
- package/src/cli/commands/generate/frontend-models.js +12 -0
- package/src/cli/commands/generate/migration.js +7 -0
- package/src/cli/commands/generate/model.js +7 -0
- package/src/cli/commands/init.js +11 -0
- package/src/cli/commands/routes.js +7 -0
- package/src/cli/commands/run-script.js +12 -0
- package/src/cli/commands/runner.js +12 -0
- package/src/cli/commands/server.js +7 -0
- package/src/cli/commands/test.js +9 -0
- package/src/cli/index.js +152 -0
- package/src/cli/tenant-database-command-helper.js +198 -0
- package/src/cli/use-browser-cli.js +30 -0
- package/src/configuration-resolver.js +65 -0
- package/src/configuration-types.js +429 -0
- package/src/configuration.js +2590 -0
- package/src/controller.js +421 -0
- package/src/current-configuration.js +31 -0
- package/src/current.js +80 -0
- package/src/database/annotations-async-hooks.js +47 -0
- package/src/database/annotations.js +40 -0
- package/src/database/drivers/base-column.js +182 -0
- package/src/database/drivers/base-columns-index.js +81 -0
- package/src/database/drivers/base-foreign-key.js +104 -0
- package/src/database/drivers/base-table.js +156 -0
- package/src/database/drivers/base.js +1609 -0
- package/src/database/drivers/mssql/column.js +74 -0
- package/src/database/drivers/mssql/columns-index.js +6 -0
- package/src/database/drivers/mssql/connect-connection.js +16 -0
- package/src/database/drivers/mssql/foreign-key.js +12 -0
- package/src/database/drivers/mssql/index.js +590 -0
- package/src/database/drivers/mssql/options.js +79 -0
- package/src/database/drivers/mssql/query-parser.js +6 -0
- package/src/database/drivers/mssql/sql/alter-table.js +4 -0
- package/src/database/drivers/mssql/sql/create-database.js +36 -0
- package/src/database/drivers/mssql/sql/create-index.js +4 -0
- package/src/database/drivers/mssql/sql/create-table.js +4 -0
- package/src/database/drivers/mssql/sql/delete.js +19 -0
- package/src/database/drivers/mssql/sql/drop-database.js +36 -0
- package/src/database/drivers/mssql/sql/drop-table.js +4 -0
- package/src/database/drivers/mssql/sql/insert.js +4 -0
- package/src/database/drivers/mssql/sql/update.js +31 -0
- package/src/database/drivers/mssql/sql/upsert.js +23 -0
- package/src/database/drivers/mssql/structure-sql.js +120 -0
- package/src/database/drivers/mssql/table.js +145 -0
- package/src/database/drivers/mysql/column.js +112 -0
- package/src/database/drivers/mysql/columns-index.js +22 -0
- package/src/database/drivers/mysql/foreign-key.js +12 -0
- package/src/database/drivers/mysql/index.js +473 -0
- package/src/database/drivers/mysql/options.js +34 -0
- package/src/database/drivers/mysql/query-parser.js +6 -0
- package/src/database/drivers/mysql/query.js +37 -0
- package/src/database/drivers/mysql/sql/alter-table.js +6 -0
- package/src/database/drivers/mysql/sql/create-database.js +39 -0
- package/src/database/drivers/mysql/sql/create-index.js +6 -0
- package/src/database/drivers/mysql/sql/create-table.js +6 -0
- package/src/database/drivers/mysql/sql/delete.js +21 -0
- package/src/database/drivers/mysql/sql/drop-database.js +6 -0
- package/src/database/drivers/mysql/sql/drop-table.js +6 -0
- package/src/database/drivers/mysql/sql/insert.js +6 -0
- package/src/database/drivers/mysql/sql/update.js +33 -0
- package/src/database/drivers/mysql/sql/upsert.js +13 -0
- package/src/database/drivers/mysql/structure-sql.js +93 -0
- package/src/database/drivers/mysql/table.js +121 -0
- package/src/database/drivers/pgsql/column.js +90 -0
- package/src/database/drivers/pgsql/columns-index.js +6 -0
- package/src/database/drivers/pgsql/foreign-key.js +12 -0
- package/src/database/drivers/pgsql/index.js +441 -0
- package/src/database/drivers/pgsql/options.js +32 -0
- package/src/database/drivers/pgsql/query-parser.js +6 -0
- package/src/database/drivers/pgsql/sql/alter-table.js +6 -0
- package/src/database/drivers/pgsql/sql/create-database.js +38 -0
- package/src/database/drivers/pgsql/sql/create-index.js +6 -0
- package/src/database/drivers/pgsql/sql/create-table.js +6 -0
- package/src/database/drivers/pgsql/sql/delete.js +21 -0
- package/src/database/drivers/pgsql/sql/drop-database.js +6 -0
- package/src/database/drivers/pgsql/sql/drop-table.js +6 -0
- package/src/database/drivers/pgsql/sql/insert.js +6 -0
- package/src/database/drivers/pgsql/sql/update.js +33 -0
- package/src/database/drivers/pgsql/sql/upsert.js +14 -0
- package/src/database/drivers/pgsql/structure-sql.js +126 -0
- package/src/database/drivers/pgsql/table.js +135 -0
- package/src/database/drivers/sqlite/base.js +509 -0
- package/src/database/drivers/sqlite/column.js +75 -0
- package/src/database/drivers/sqlite/columns-index.js +30 -0
- package/src/database/drivers/sqlite/connection-sql-js.js +46 -0
- package/src/database/drivers/sqlite/foreign-key.js +24 -0
- package/src/database/drivers/sqlite/index.js +394 -0
- package/src/database/drivers/sqlite/index.native.js +72 -0
- package/src/database/drivers/sqlite/index.web.js +99 -0
- package/src/database/drivers/sqlite/options.js +32 -0
- package/src/database/drivers/sqlite/query-parser.js +6 -0
- package/src/database/drivers/sqlite/query.js +35 -0
- package/src/database/drivers/sqlite/query.native.js +35 -0
- package/src/database/drivers/sqlite/query.web.js +49 -0
- package/src/database/drivers/sqlite/sql/alter-table.js +187 -0
- package/src/database/drivers/sqlite/sql/create-index.js +6 -0
- package/src/database/drivers/sqlite/sql/create-table.js +6 -0
- package/src/database/drivers/sqlite/sql/delete.js +26 -0
- package/src/database/drivers/sqlite/sql/drop-table.js +6 -0
- package/src/database/drivers/sqlite/sql/insert.js +6 -0
- package/src/database/drivers/sqlite/sql/update.js +33 -0
- package/src/database/drivers/sqlite/sql/upsert.js +14 -0
- package/src/database/drivers/sqlite/structure-sql.js +56 -0
- package/src/database/drivers/sqlite/table-rebuilder.js +96 -0
- package/src/database/drivers/sqlite/table.js +131 -0
- package/src/database/drivers/structure-sql/utils.js +35 -0
- package/src/database/handler.js +13 -0
- package/src/database/initializer-from-require-context.js +101 -0
- package/src/database/migration/index.js +438 -0
- package/src/database/migrator/files-finder.js +55 -0
- package/src/database/migrator/types.js +31 -0
- package/src/database/migrator.js +557 -0
- package/src/database/pool/async-tracked-multi-connection.js +1164 -0
- package/src/database/pool/base-methods-forward.js +52 -0
- package/src/database/pool/base.js +380 -0
- package/src/database/pool/single-multi-use.js +118 -0
- package/src/database/query/alter-table-base.js +104 -0
- package/src/database/query/base.js +49 -0
- package/src/database/query/create-database-base.js +42 -0
- package/src/database/query/create-index-base.js +117 -0
- package/src/database/query/create-table-base.js +205 -0
- package/src/database/query/delete-base.js +19 -0
- package/src/database/query/drop-database-base.js +38 -0
- package/src/database/query/drop-table-base.js +58 -0
- package/src/database/query/from-base.js +36 -0
- package/src/database/query/from-plain.js +16 -0
- package/src/database/query/from-table.js +18 -0
- package/src/database/query/index.js +533 -0
- package/src/database/query/insert-base.js +172 -0
- package/src/database/query/join-base.js +43 -0
- package/src/database/query/join-object.js +167 -0
- package/src/database/query/join-plain.js +18 -0
- package/src/database/query/join-tracker.js +93 -0
- package/src/database/query/model-class-query.js +1577 -0
- package/src/database/query/order-base.js +33 -0
- package/src/database/query/order-column.js +77 -0
- package/src/database/query/order-plain.js +28 -0
- package/src/database/query/preloader/belongs-to.js +267 -0
- package/src/database/query/preloader/ensure-model-class-initialized.js +18 -0
- package/src/database/query/preloader/has-many.js +316 -0
- package/src/database/query/preloader/has-one.js +123 -0
- package/src/database/query/preloader/selection.js +152 -0
- package/src/database/query/preloader.js +201 -0
- package/src/database/query/query-data.js +305 -0
- package/src/database/query/select-base.js +30 -0
- package/src/database/query/select-plain.js +18 -0
- package/src/database/query/select-table-and-column.js +28 -0
- package/src/database/query/update-base.js +41 -0
- package/src/database/query/upsert-base.js +103 -0
- package/src/database/query/where-base.js +38 -0
- package/src/database/query/where-combinator.js +31 -0
- package/src/database/query/where-hash.js +77 -0
- package/src/database/query/where-model-class-hash.js +505 -0
- package/src/database/query/where-not.js +23 -0
- package/src/database/query/where-plain.js +20 -0
- package/src/database/query/with-count.js +219 -0
- package/src/database/query-parser/base-query-parser.js +40 -0
- package/src/database/query-parser/from-parser.js +49 -0
- package/src/database/query-parser/group-parser.js +55 -0
- package/src/database/query-parser/joins-parser.js +37 -0
- package/src/database/query-parser/limit-parser.js +77 -0
- package/src/database/query-parser/options.js +94 -0
- package/src/database/query-parser/order-parser.js +45 -0
- package/src/database/query-parser/select-parser.js +67 -0
- package/src/database/query-parser/where-parser.js +46 -0
- package/src/database/record/acts-as-list.js +374 -0
- package/src/database/record/attachments/download.js +49 -0
- package/src/database/record/attachments/handle.js +188 -0
- package/src/database/record/attachments/normalize-input.js +213 -0
- package/src/database/record/attachments/storage-drivers/filesystem.js +114 -0
- package/src/database/record/attachments/storage-drivers/native.js +146 -0
- package/src/database/record/attachments/storage-drivers/s3.js +245 -0
- package/src/database/record/attachments/store.js +591 -0
- package/src/database/record/index.js +3970 -0
- package/src/database/record/instance-relationships/base.js +289 -0
- package/src/database/record/instance-relationships/belongs-to.js +84 -0
- package/src/database/record/instance-relationships/has-many.js +284 -0
- package/src/database/record/instance-relationships/has-one.js +117 -0
- package/src/database/record/record-not-found-error.js +3 -0
- package/src/database/record/relationships/base.js +195 -0
- package/src/database/record/relationships/belongs-to.js +57 -0
- package/src/database/record/relationships/has-many.js +46 -0
- package/src/database/record/relationships/has-one.js +46 -0
- package/src/database/record/state-machine.js +278 -0
- package/src/database/record/user-module.js +43 -0
- package/src/database/record/validators/base.js +27 -0
- package/src/database/record/validators/format.js +50 -0
- package/src/database/record/validators/presence.js +24 -0
- package/src/database/record/validators/uniqueness.js +124 -0
- package/src/database/table-data/index.js +241 -0
- package/src/database/table-data/table-column.js +416 -0
- package/src/database/table-data/table-foreign-key.js +69 -0
- package/src/database/table-data/table-index.js +46 -0
- package/src/database/table-data/table-reference.js +13 -0
- package/src/database/use-database.js +48 -0
- package/src/environment-handlers/base.js +561 -0
- package/src/environment-handlers/browser.js +338 -0
- package/src/environment-handlers/node/cli/commands/background-jobs-main.js +21 -0
- package/src/environment-handlers/node/cli/commands/background-jobs-runner.js +24 -0
- package/src/environment-handlers/node/cli/commands/background-jobs-worker.js +47 -0
- package/src/environment-handlers/node/cli/commands/beacon.js +21 -0
- package/src/environment-handlers/node/cli/commands/cli-command-context.js +31 -0
- package/src/environment-handlers/node/cli/commands/console.js +149 -0
- package/src/environment-handlers/node/cli/commands/db/schema/dump.js +43 -0
- package/src/environment-handlers/node/cli/commands/db/schema/load.js +69 -0
- package/src/environment-handlers/node/cli/commands/db/seed.js +79 -0
- package/src/environment-handlers/node/cli/commands/destroy/migration.js +47 -0
- package/src/environment-handlers/node/cli/commands/generate/base-models.js +367 -0
- package/src/environment-handlers/node/cli/commands/generate/frontend-models.js +872 -0
- package/src/environment-handlers/node/cli/commands/generate/migration.js +45 -0
- package/src/environment-handlers/node/cli/commands/generate/model.js +45 -0
- package/src/environment-handlers/node/cli/commands/init.js +68 -0
- package/src/environment-handlers/node/cli/commands/routes.js +63 -0
- package/src/environment-handlers/node/cli/commands/run-script.js +85 -0
- package/src/environment-handlers/node/cli/commands/runner.js +84 -0
- package/src/environment-handlers/node/cli/commands/server.js +151 -0
- package/src/environment-handlers/node/cli/commands/test.js +118 -0
- package/src/environment-handlers/node.js +887 -0
- package/src/error-logger.js +30 -0
- package/src/frontend-model-controller.js +3491 -0
- package/src/frontend-model-resource/base-resource.js +935 -0
- package/src/frontend-models/base.js +4004 -0
- package/src/frontend-models/clear-pending-debounced-callback.js +16 -0
- package/src/frontend-models/event-hook-models.js +49 -0
- package/src/frontend-models/model-registry.js +28 -0
- package/src/frontend-models/outgoing-event-buffer.js +51 -0
- package/src/frontend-models/preloader.js +169 -0
- package/src/frontend-models/query.js +2245 -0
- package/src/frontend-models/resource-config-validation.js +56 -0
- package/src/frontend-models/resource-definition.js +399 -0
- package/src/frontend-models/transport-serialization.js +369 -0
- package/src/frontend-models/use-created-event.js +21 -0
- package/src/frontend-models/use-destroyed-event.js +148 -0
- package/src/frontend-models/use-model-class-event.js +164 -0
- package/src/frontend-models/use-updated-event.js +152 -0
- package/src/frontend-models/websocket-channel.js +494 -0
- package/src/frontend-models/websocket-publishers.js +224 -0
- package/src/http-client/header.js +17 -0
- package/src/http-client/index.js +139 -0
- package/src/http-client/request.js +94 -0
- package/src/http-client/response.js +151 -0
- package/src/http-client/websocket-client.js +27 -0
- package/src/http-server/client/index.js +507 -0
- package/src/http-server/client/params-to-object.js +152 -0
- package/src/http-server/client/request-buffer/form-data-part.js +139 -0
- package/src/http-server/client/request-buffer/header.js +19 -0
- package/src/http-server/client/request-buffer/index.js +535 -0
- package/src/http-server/client/request-parser.js +195 -0
- package/src/http-server/client/request-runner.js +321 -0
- package/src/http-server/client/request-timing.js +171 -0
- package/src/http-server/client/request.js +114 -0
- package/src/http-server/client/response.js +251 -0
- package/src/http-server/client/uploaded-file/memory-uploaded-file.js +32 -0
- package/src/http-server/client/uploaded-file/temporary-uploaded-file.js +32 -0
- package/src/http-server/client/uploaded-file/uploaded-file.js +36 -0
- package/src/http-server/client/websocket-request.js +147 -0
- package/src/http-server/client/websocket-session.js +1755 -0
- package/src/http-server/cookie.js +245 -0
- package/src/http-server/development-reloader.js +240 -0
- package/src/http-server/index.js +561 -0
- package/src/http-server/remote-address.js +77 -0
- package/src/http-server/server-client.js +222 -0
- package/src/http-server/server-lock.js +178 -0
- package/src/http-server/websocket-channel-subscribers.js +110 -0
- package/src/http-server/websocket-channel.js +137 -0
- package/src/http-server/websocket-connection.js +118 -0
- package/src/http-server/websocket-event-log-store.js +433 -0
- package/src/http-server/websocket-events-host.js +170 -0
- package/src/http-server/websocket-events.js +50 -0
- package/src/http-server/worker-handler/channel-subscriber-dispatch.js +28 -0
- package/src/http-server/worker-handler/in-process.js +155 -0
- package/src/http-server/worker-handler/index.js +370 -0
- package/src/http-server/worker-handler/worker-script.js +6 -0
- package/src/http-server/worker-handler/worker-thread.js +286 -0
- package/src/initializer.js +39 -0
- package/src/jobs/.gitkeep +1 -0
- package/src/jobs/mail-delivery.js +22 -0
- package/src/logger/base-logger.js +34 -0
- package/src/logger/console-logger.js +28 -0
- package/src/logger/file-logger.js +36 -0
- package/src/logger/outputs/array-output.js +50 -0
- package/src/logger/outputs/console-output.js +32 -0
- package/src/logger/outputs/file-output.js +55 -0
- package/src/logger/outputs/stdout-output.js +64 -0
- package/src/logger.js +507 -0
- package/src/mailer/backends/smtp.js +197 -0
- package/src/mailer/base.js +337 -0
- package/src/mailer/delivery.js +70 -0
- package/src/mailer/index.js +24 -0
- package/src/mailer.js +15 -0
- package/src/plugins/sqljs-wasm-route-controller.js +70 -0
- package/src/plugins/sqljs-wasm-route.js +71 -0
- package/src/record-payload-values.js +83 -0
- package/src/routes/app-routes.js +17 -0
- package/src/routes/base-route.js +133 -0
- package/src/routes/basic-route.js +109 -0
- package/src/routes/built-in/debug/controller.js +12 -0
- package/src/routes/built-in/errors/controller.js +7 -0
- package/src/routes/built-in/errors/not-found.ejs +1 -0
- package/src/routes/get-route.js +75 -0
- package/src/routes/hooks/frontend-model-command-route-hook.js +100 -0
- package/src/routes/index.js +50 -0
- package/src/routes/namespace-route.js +51 -0
- package/src/routes/plugin-routes.js +141 -0
- package/src/routes/post-route.js +74 -0
- package/src/routes/resolver.js +535 -0
- package/src/routes/resource-route.js +154 -0
- package/src/routes/root-route.js +11 -0
- package/src/templates/configuration.js +61 -0
- package/src/templates/generate-migration.js +11 -0
- package/src/templates/generate-model.js +6 -0
- package/src/templates/routes.js +11 -0
- package/src/testing/base-expect.js +17 -0
- package/src/testing/browser-frontend-model-event-hook-scenarios.js +520 -0
- package/src/testing/browser-test-app.js +32 -0
- package/src/testing/expect-to-change.js +55 -0
- package/src/testing/expect-utils.js +269 -0
- package/src/testing/expect.js +763 -0
- package/src/testing/request-client.js +90 -0
- package/src/testing/test-files-finder.js +364 -0
- package/src/testing/test-filter-parser.js +198 -0
- package/src/testing/test-runner.js +1168 -0
- package/src/testing/test-suite-splitter.js +177 -0
- package/src/testing/test.js +370 -0
- package/src/types/external-modules.d.ts +57 -0
- package/src/utils/backtrace-cleaner-node.js +87 -0
- package/src/utils/backtrace-cleaner.js +266 -0
- package/src/utils/ensure-error.js +15 -0
- package/src/utils/event-emitter.js +8 -0
- package/src/utils/file-exists.js +18 -0
- package/src/utils/format-value.js +101 -0
- package/src/utils/model-scope.js +56 -0
- package/src/utils/nest-callbacks.js +22 -0
- package/src/utils/plain-object.js +14 -0
- package/src/utils/ransack.js +859 -0
- package/src/utils/rest-args-error.js +14 -0
- package/src/utils/singularize-model-name.js +18 -0
- package/src/utils/split-sql-statements.js +88 -0
- package/src/utils/to-import-specifier.js +53 -0
- package/src/utils/with-tracked-stack-async-hooks.js +103 -0
- package/src/utils/with-tracked-stack.js +38 -0
- package/src/velocious-error.js +34 -0
- package/tsconfig.json +16 -0
|
@@ -1,10 +1,13 @@
|
|
|
1
1
|
// @ts-check
|
|
2
|
-
|
|
3
|
-
import
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
const
|
|
7
|
-
const
|
|
2
|
+
|
|
3
|
+
import {AsyncLocalStorage} from "async_hooks"
|
|
4
|
+
import BasePool, {POOL_CONFIGURATION_KEY} from "./base.js"
|
|
5
|
+
|
|
6
|
+
export const CLOSED_CONNECTION = Symbol("velociousClosedConnection")
|
|
7
|
+
const IDLE_CONNECTION_CHECKED_IN_AT = Symbol("velociousIdleConnectionCheckedInAt")
|
|
8
|
+
const CONNECTION_CHECKED_OUT_AT = Symbol("velociousConnectionCheckedOutAt")
|
|
9
|
+
const DEFAULT_IDLE_TIMEOUT_MILLIS = 5000
|
|
10
|
+
|
|
8
11
|
/**
|
|
9
12
|
* PendingCheckout type.
|
|
10
13
|
* @typedef {object} PendingCheckout
|
|
@@ -15,1000 +18,1147 @@ const DEFAULT_IDLE_TIMEOUT_MILLIS = 5000;
|
|
|
15
18
|
* @property {(connection: import("../drivers/base.js").default) => void} resolve - Resolves with an activated connection.
|
|
16
19
|
* @property {(error: Error) => void} reject - Rejects when checkout cannot complete.
|
|
17
20
|
*/
|
|
21
|
+
|
|
18
22
|
export default class VelociousDatabasePoolAsyncTrackedMultiConnection extends BasePool {
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
23
|
+
/**
|
|
24
|
+
* Global fallback connections keyed by configuration instance and pool identifier.
|
|
25
|
+
* @type {WeakMap<import("../../configuration.js").default, Record<string, import("../drivers/base.js").default>>}
|
|
26
|
+
*/
|
|
27
|
+
static globalConnections = new WeakMap()
|
|
28
|
+
|
|
29
|
+
asyncLocalStorage = new AsyncLocalStorage()
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* When set, returned by getCurrentContextConnection when no async context exists.
|
|
33
|
+
* Used by the test runner to share a connection between test code and HTTP handlers
|
|
34
|
+
* running in the same process (in-process test server mode).
|
|
35
|
+
* @type {import("../drivers/base.js").default | undefined}
|
|
36
|
+
*/
|
|
37
|
+
_testSharedConnection = undefined
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Connections.
|
|
41
|
+
@type {import("../drivers/base.js").default[]} */
|
|
42
|
+
connections = []
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Connections in use.
|
|
46
|
+
@type {Record<number, import("../drivers/base.js").default>} */
|
|
47
|
+
connectionsInUse = {}
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Pending checkouts.
|
|
51
|
+
@type {PendingCheckout[]} */
|
|
52
|
+
pendingCheckouts = []
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Connections being spawned.
|
|
56
|
+
@type {number} */
|
|
57
|
+
connectionsBeingSpawned = 0
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Pending checkout drain promise.
|
|
61
|
+
@type {Promise<void> | undefined} */
|
|
62
|
+
pendingCheckoutDrainPromise = undefined
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Idle connection reaper timer.
|
|
66
|
+
@type {ReturnType<typeof setTimeout> | undefined} */
|
|
67
|
+
idleConnectionReaperTimer = undefined
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* In-flight connection-close promises. The idle reaper is armed on check-in
|
|
71
|
+
* and runs fire-and-forget when its timer fires, so a scheduled reap can be
|
|
72
|
+
* closing a connection while an explicit `reapIdleConnections()` (or
|
|
73
|
+
* `clearIdleConnectionReaperTimer()`) runs. Tracking the in-flight closes lets
|
|
74
|
+
* those callers await them, so once a reap resolves the connections it
|
|
75
|
+
* expired are fully closed instead of half-closed mid-`close()`.
|
|
76
|
+
* @type {Set<Promise<void>>}
|
|
77
|
+
*/
|
|
78
|
+
inflightConnectionCloses = new Set()
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* In-flight close promise per connection, so concurrent closes of the same
|
|
82
|
+
* connection await the same close rather than closing the driver handle twice.
|
|
83
|
+
* @type {WeakMap<object, Promise<void>>}
|
|
84
|
+
*/
|
|
85
|
+
connectionClosePromises = new WeakMap()
|
|
86
|
+
|
|
87
|
+
idSeq = 0
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Runs constructor.
|
|
91
|
+
* @param {object} args - Options object.
|
|
92
|
+
* @param {import("../../configuration.js").default} args.configuration - Configuration instance.
|
|
93
|
+
* @param {string} args.identifier - Identifier.
|
|
94
|
+
*/
|
|
95
|
+
constructor({configuration, identifier}) {
|
|
96
|
+
super({configuration, identifier})
|
|
97
|
+
this._withoutCurrentConnectionContext = /**
|
|
98
|
+
* Narrows the runtime value to the documented type.
|
|
99
|
+
@type {(callback: () => ?) => ?} */ ((callback) => this.asyncLocalStorage.run(undefined, callback))
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Runs checkin.
|
|
104
|
+
* @param {import("../drivers/base.js").default} connection - Database connection instance.
|
|
105
|
+
* @returns {Promise<void>} - Resolves when the connection is checked in or closed.
|
|
106
|
+
*/
|
|
107
|
+
async checkin(connection) {
|
|
108
|
+
const id = connection.getIdSeq()
|
|
109
|
+
const trackedConnection = /**
|
|
110
|
+
* Narrows the runtime value to the documented type.
|
|
111
|
+
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean, [CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
112
|
+
|
|
113
|
+
if (trackedConnection[CLOSED_CONNECTION]) {
|
|
114
|
+
this.untrackConnectionInUse(connection, id)
|
|
115
|
+
await this.drainPendingCheckouts()
|
|
116
|
+
return
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
try {
|
|
120
|
+
await this.rollbackLeftOpenTransaction(connection)
|
|
121
|
+
await connection.clearConnectionCheckoutName()
|
|
122
|
+
} catch (error) {
|
|
123
|
+
await this.closeCheckedOutConnectionAfterCheckinFailure(connection, id, error)
|
|
124
|
+
throw error
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
this.untrackConnectionInUse(connection, id)
|
|
128
|
+
trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT] = Date.now()
|
|
129
|
+
delete trackedConnection[CONNECTION_CHECKED_OUT_AT]
|
|
130
|
+
this.connections.push(connection)
|
|
131
|
+
await this.drainPendingCheckouts()
|
|
132
|
+
if (this.connections.includes(connection)) await this.handleCheckedInIdleConnection()
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* Runs close checked out connection after checkin failure.
|
|
137
|
+
* @param {import("../drivers/base.js").default} connection - Connection that failed check-in cleanup.
|
|
138
|
+
* @param {number | undefined} id - Connection checkout id.
|
|
139
|
+
* @param {?} originalError - Error that caused check-in cleanup to fail.
|
|
140
|
+
* @returns {Promise<void>} - Resolves when cleanup has been attempted.
|
|
141
|
+
*/
|
|
142
|
+
async closeCheckedOutConnectionAfterCheckinFailure(connection, id, originalError) {
|
|
143
|
+
this.untrackConnectionInUse(connection, id)
|
|
144
|
+
|
|
145
|
+
try {
|
|
146
|
+
await this.closeConnection(connection)
|
|
147
|
+
} catch (error) {
|
|
148
|
+
this.logger.warn("Failed to close database connection after check-in cleanup failed", {error, originalError})
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
try {
|
|
152
|
+
await this.drainPendingCheckouts()
|
|
153
|
+
} catch (error) {
|
|
154
|
+
this.logger.warn("Failed to drain pending database checkouts after check-in cleanup failed", {error, originalError})
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
/**
|
|
159
|
+
* Runs untrack connection in use.
|
|
160
|
+
* @param {import("../drivers/base.js").default} connection - Connection being checked in.
|
|
161
|
+
* @param {number | undefined} id - Connection checkout id.
|
|
162
|
+
* @returns {void}
|
|
163
|
+
*/
|
|
164
|
+
untrackConnectionInUse(connection, id) {
|
|
165
|
+
if (typeof id !== "number") {
|
|
166
|
+
throw new Error(`idSeq on connection wasn't set? '${typeof id}' = ${id}`)
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
delete this.connectionsInUse[id]
|
|
170
|
+
connection.setIdSeq(undefined)
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
* Runs handle checked in idle connection.
|
|
175
|
+
* @returns {Promise<void>} - Resolves once idle reaping has been scheduled or run.
|
|
176
|
+
*/
|
|
177
|
+
async handleCheckedInIdleConnection() {
|
|
178
|
+
if (this.idleTimeoutMillis() === 0) {
|
|
179
|
+
await this.reapIdleConnections()
|
|
180
|
+
} else {
|
|
181
|
+
this.scheduleIdleConnectionReaper()
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Runs checkout.
|
|
187
|
+
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
188
|
+
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with the checkout.
|
|
189
|
+
*/
|
|
190
|
+
async checkout(options = {}) {
|
|
191
|
+
const databaseConfig = this.getConfiguration()
|
|
192
|
+
const reuseKey = this.getConfigurationReuseKey()
|
|
193
|
+
let connection = this.takeIdleConnectionForReuseKey(reuseKey)
|
|
194
|
+
|
|
195
|
+
if (connection) return await this.activateConnection(connection, options)
|
|
196
|
+
|
|
197
|
+
await this.reapIdleConnections()
|
|
198
|
+
connection = this.takeIdleConnectionForReuseKey(reuseKey)
|
|
199
|
+
|
|
200
|
+
if (connection) return await this.activateConnection(connection, options)
|
|
201
|
+
|
|
202
|
+
if (this.canSpawnConnection()) {
|
|
203
|
+
// Spawn via spawnConnection() so the tenant-aware configuration is resolved FRESH at
|
|
204
|
+
// spawn time for the current caller. Reusing the databaseConfig captured at the top of
|
|
205
|
+
// checkout() could bind the connection to a stale tenant/database, which breaks
|
|
206
|
+
// per-request isolation (e.g. test truncation appearing not to take effect). The queued
|
|
207
|
+
// path below keeps the waiting caller's captured config via waitForCheckout().
|
|
208
|
+
connection = await this.spawnConnectionForCheckout(this.getConfiguration(), this.getConfigurationReuseKey())
|
|
209
|
+
|
|
210
|
+
return await this.activateConnection(connection, options)
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
return await this.waitForCheckout(databaseConfig, reuseKey, options)
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
/**
|
|
217
|
+
* Runs take idle connection for reuse key.
|
|
218
|
+
* @param {string} reuseKey - Database configuration reuse key.
|
|
219
|
+
* @param {object} [args] - Options.
|
|
220
|
+
* @param {boolean} [args.includeOpenTransactions] - Whether connections with open transactions may be returned.
|
|
221
|
+
* @returns {import("../drivers/base.js").default | undefined} - Matching idle connection.
|
|
222
|
+
*/
|
|
223
|
+
takeIdleConnectionForReuseKey(reuseKey, {includeOpenTransactions = true} = {}) {
|
|
224
|
+
const connectionIndex = this.connections.findIndex((queuedConnection) => {
|
|
225
|
+
if (!includeOpenTransactions && this.connectionHasOpenTransaction(queuedConnection)) return false
|
|
226
|
+
|
|
227
|
+
return this.connectionMatchesReuseKey(queuedConnection, reuseKey)
|
|
228
|
+
})
|
|
229
|
+
const connection = connectionIndex === -1 ? undefined : this.connections.splice(connectionIndex, 1)[0]
|
|
230
|
+
|
|
231
|
+
return connection
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* Runs connection matches reuse key.
|
|
236
|
+
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
237
|
+
* @param {string} reuseKey - Database configuration reuse key.
|
|
238
|
+
* @returns {boolean} - Whether the connection matches the reuse key.
|
|
239
|
+
*/
|
|
240
|
+
connectionMatchesReuseKey(connection, reuseKey) {
|
|
241
|
+
const connectionWithPoolKey = /**
|
|
242
|
+
* Narrows the runtime value to the documented type.
|
|
243
|
+
@type {import("../drivers/base.js").default & {[POOL_CONFIGURATION_KEY]?: string}} */ (connection)
|
|
244
|
+
|
|
245
|
+
return connectionWithPoolKey[POOL_CONFIGURATION_KEY] === reuseKey
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
/**
|
|
249
|
+
* Runs activate connection.
|
|
250
|
+
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
251
|
+
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
252
|
+
* @returns {Promise<import("../drivers/base.js").default>} - Activated connection.
|
|
253
|
+
*/
|
|
254
|
+
async activateConnection(connection, options = {}) {
|
|
255
|
+
if (connection.getIdSeq() !== undefined) throw new Error(`Connection already has an ID-seq - is it in use? ${connection.getIdSeq()}`)
|
|
256
|
+
|
|
257
|
+
const id = this.idSeq++
|
|
258
|
+
|
|
259
|
+
const trackedConnection = /**
|
|
260
|
+
* Narrows the runtime value to the documented type.
|
|
261
|
+
@type {import("../drivers/base.js").default & {[CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
262
|
+
delete trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT]
|
|
263
|
+
trackedConnection[CONNECTION_CHECKED_OUT_AT] = Date.now()
|
|
264
|
+
|
|
265
|
+
connection.setIdSeq(id)
|
|
266
|
+
this.connectionsInUse[id] = connection
|
|
267
|
+
|
|
268
|
+
try {
|
|
269
|
+
await connection.setConnectionCheckoutName(options.name)
|
|
270
|
+
} catch (error) {
|
|
271
|
+
delete this.connectionsInUse[id]
|
|
272
|
+
connection.setIdSeq(undefined)
|
|
273
|
+
await this.closeConnection(connection)
|
|
274
|
+
|
|
275
|
+
throw error
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
return connection
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
/**
|
|
282
|
+
* Runs max connections.
|
|
283
|
+
* @returns {number | undefined} - Configured max live connections.
|
|
284
|
+
*/
|
|
285
|
+
maxConnections() {
|
|
286
|
+
const value = this.getConfiguration().pool?.max
|
|
287
|
+
|
|
288
|
+
if (this.validMaxConnections(value)) return value
|
|
289
|
+
|
|
290
|
+
return
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
/**
|
|
294
|
+
* Runs valid max connections.
|
|
295
|
+
* @param {?} value - Candidate max connection count.
|
|
296
|
+
* @returns {value is number} - Whether the value is a valid max connection count.
|
|
297
|
+
*/
|
|
298
|
+
validMaxConnections(value) {
|
|
299
|
+
return typeof value === "number" && Number.isFinite(value) && value >= 1
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
/**
|
|
303
|
+
* Runs live connection count.
|
|
304
|
+
* @returns {number} - Number of live and in-progress connections.
|
|
305
|
+
*/
|
|
306
|
+
liveConnectionCount() {
|
|
307
|
+
const connections = new Set([
|
|
308
|
+
...this.connections,
|
|
309
|
+
...Object.values(this.connectionsInUse),
|
|
310
|
+
this.getGlobalConnectionForIdentifier()
|
|
311
|
+
].filter(Boolean))
|
|
312
|
+
|
|
313
|
+
return connections.size + this.connectionsBeingSpawned
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
/**
|
|
317
|
+
* Runs can spawn connection.
|
|
318
|
+
* @returns {boolean} - Whether a new connection can be spawned.
|
|
319
|
+
*/
|
|
320
|
+
canSpawnConnection() {
|
|
321
|
+
const maxConnections = this.maxConnections()
|
|
322
|
+
|
|
323
|
+
return maxConnections === undefined || this.liveConnectionCount() < maxConnections
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
/**
|
|
327
|
+
* Runs spawn connection for checkout.
|
|
328
|
+
* @param {import("../../configuration-types.js").DatabaseConfigurationType} databaseConfig - Resolved database config for the checkout.
|
|
329
|
+
* @param {string} reuseKey - Database configuration reuse key for the checkout.
|
|
330
|
+
* @returns {Promise<import("../drivers/base.js").default>} - Spawned connection.
|
|
331
|
+
*/
|
|
332
|
+
async spawnConnectionForCheckout(databaseConfig, reuseKey) {
|
|
333
|
+
this.connectionsBeingSpawned++
|
|
334
|
+
|
|
335
|
+
try {
|
|
336
|
+
const connection = await this.spawnConnectionWithConfiguration(databaseConfig)
|
|
337
|
+
const connectionWithPoolKey = /**
|
|
338
|
+
* Narrows the runtime value to the documented type.
|
|
339
|
+
@type {import("../drivers/base.js").default & {[POOL_CONFIGURATION_KEY]?: string}} */ (connection)
|
|
340
|
+
|
|
341
|
+
connectionWithPoolKey[POOL_CONFIGURATION_KEY] = reuseKey
|
|
342
|
+
connection.setSchemaCacheInvalidator(() => {
|
|
343
|
+
this.clearSchemaCache()
|
|
344
|
+
this.configuration.clearSchemaCachesForReuseKey(reuseKey)
|
|
345
|
+
})
|
|
346
|
+
|
|
347
|
+
return connection
|
|
348
|
+
} finally {
|
|
349
|
+
this.connectionsBeingSpawned--
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
/**
|
|
354
|
+
* Runs wait for checkout.
|
|
355
|
+
* @param {import("../../configuration-types.js").DatabaseConfigurationType} databaseConfig - Resolved database config for the checkout.
|
|
356
|
+
* @param {string} reuseKey - Database configuration reuse key.
|
|
357
|
+
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
358
|
+
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with an activated connection.
|
|
359
|
+
*/
|
|
360
|
+
async waitForCheckout(databaseConfig, reuseKey, options = {}) {
|
|
361
|
+
return await new Promise((resolve, reject) => {
|
|
362
|
+
this.pendingCheckouts.push({databaseConfig, enqueuedAt: Date.now(), options, reject, resolve, reuseKey})
|
|
363
|
+
void this.drainPendingCheckouts().catch((error) => {
|
|
364
|
+
const checkoutError = error instanceof Error ? error : new Error("Failed to drain pending database connection checkouts.", {cause: error})
|
|
365
|
+
|
|
366
|
+
this.rejectPendingCheckouts(checkoutError)
|
|
367
|
+
})
|
|
368
|
+
})
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
/**
|
|
372
|
+
* Runs drain pending checkouts.
|
|
373
|
+
* @returns {Promise<void>} - Resolves when pending checkouts have been drained as far as possible.
|
|
374
|
+
*/
|
|
375
|
+
async drainPendingCheckouts() {
|
|
376
|
+
if (this.pendingCheckoutDrainPromise) {
|
|
377
|
+
await this.pendingCheckoutDrainPromise
|
|
378
|
+
return
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
this.pendingCheckoutDrainPromise = this.drainPendingCheckoutsActual()
|
|
382
|
+
|
|
383
|
+
try {
|
|
384
|
+
await this.pendingCheckoutDrainPromise
|
|
385
|
+
} finally {
|
|
386
|
+
this.pendingCheckoutDrainPromise = undefined
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
/**
|
|
391
|
+
* Runs drain pending checkouts actual.
|
|
392
|
+
* @returns {Promise<void>} - Resolves when pending checkouts have been drained as far as possible.
|
|
393
|
+
*/
|
|
394
|
+
async drainPendingCheckoutsActual() {
|
|
395
|
+
while (this.pendingCheckouts.length > 0) {
|
|
396
|
+
if (await this.resolvePendingCheckoutWithMatchingIdleConnection()) continue
|
|
397
|
+
|
|
398
|
+
const checkout = this.pendingCheckouts[0]
|
|
399
|
+
|
|
400
|
+
if (await this.closeIdleConnectionForPendingCheckoutCapacity(checkout)) continue
|
|
401
|
+
if (this.canSpawnConnection()) {
|
|
402
|
+
this.pendingCheckouts.shift()
|
|
403
|
+
await this.spawnAndResolvePendingCheckout(checkout)
|
|
404
|
+
continue
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
const reapedConnection = await this.idleConnectionForPendingCheckout(checkout)
|
|
408
|
+
|
|
409
|
+
if (!reapedConnection) return
|
|
410
|
+
|
|
411
|
+
this.pendingCheckouts.shift()
|
|
412
|
+
await this.resolvePendingCheckout(checkout, reapedConnection)
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
/**
|
|
417
|
+
* Runs resolve pending checkout with matching idle connection.
|
|
418
|
+
* @returns {Promise<boolean>} - Whether a pending checkout was resolved with an idle connection.
|
|
419
|
+
*/
|
|
420
|
+
async resolvePendingCheckoutWithMatchingIdleConnection() {
|
|
421
|
+
for (let index = 0; index < this.pendingCheckouts.length; index++) {
|
|
422
|
+
const checkout = this.pendingCheckouts[index]
|
|
423
|
+
const connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, {includeOpenTransactions: false})
|
|
424
|
+
|
|
425
|
+
if (!connection) continue
|
|
426
|
+
|
|
427
|
+
this.pendingCheckouts.splice(index, 1)
|
|
428
|
+
await this.resolvePendingCheckout(checkout, connection)
|
|
429
|
+
|
|
430
|
+
return true
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
return false
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
/**
|
|
437
|
+
* Runs close idle connection for pending checkout capacity.
|
|
438
|
+
* @param {PendingCheckout} checkout - Checkout waiting for a connection.
|
|
439
|
+
* @returns {Promise<boolean>} - Whether an idle connection was closed to free capacity.
|
|
440
|
+
*/
|
|
441
|
+
async closeIdleConnectionForPendingCheckoutCapacity(checkout) {
|
|
442
|
+
const connection = this.findIdleConnectionForReuseKey(checkout.reuseKey)
|
|
443
|
+
|
|
444
|
+
if (connection) return false
|
|
445
|
+
|
|
446
|
+
await this.reapIdleConnections()
|
|
447
|
+
|
|
448
|
+
if (this.findIdleConnectionForReuseKey(checkout.reuseKey)) return false
|
|
449
|
+
|
|
450
|
+
return this.canSpawnConnection() ? false : await this.closeOneIdleConnectionForCapacity()
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
/**
|
|
454
|
+
* Runs find idle connection for reuse key.
|
|
455
|
+
* @param {string} reuseKey - Database configuration reuse key.
|
|
456
|
+
* @returns {import("../drivers/base.js").default | undefined} - Matching idle connection, if present.
|
|
457
|
+
*/
|
|
458
|
+
findIdleConnectionForReuseKey(reuseKey) {
|
|
459
|
+
return this.connections.find((connection) => !this.connectionHasOpenTransaction(connection) && this.connectionMatchesReuseKey(connection, reuseKey))
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
/**
|
|
463
|
+
* Runs idle connection for pending checkout.
|
|
464
|
+
* @param {PendingCheckout} checkout - Checkout waiting for a connection.
|
|
465
|
+
* @returns {Promise<import("../drivers/base.js").default | undefined>} - Matching idle connection, if one can be reused.
|
|
466
|
+
*/
|
|
467
|
+
async idleConnectionForPendingCheckout(checkout) {
|
|
468
|
+
let connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, {includeOpenTransactions: false})
|
|
469
|
+
|
|
470
|
+
if (connection) return connection
|
|
471
|
+
|
|
472
|
+
await this.reapIdleConnections()
|
|
473
|
+
connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, {includeOpenTransactions: false})
|
|
474
|
+
|
|
475
|
+
return connection
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
/**
|
|
479
|
+
* Runs spawn and resolve pending checkout.
|
|
480
|
+
* @param {PendingCheckout} checkout - Checkout request to resolve.
|
|
481
|
+
* @returns {Promise<void>} - Resolves when the checkout has been handled.
|
|
482
|
+
*/
|
|
483
|
+
async spawnAndResolvePendingCheckout(checkout) {
|
|
484
|
+
let connection
|
|
485
|
+
|
|
486
|
+
try {
|
|
487
|
+
connection = await this.spawnConnectionForCheckout(checkout.databaseConfig, checkout.reuseKey)
|
|
488
|
+
} catch (error) {
|
|
489
|
+
checkout.reject(error instanceof Error ? error : new Error("Failed to spawn database connection.", {cause: error}))
|
|
490
|
+
return
|
|
491
|
+
}
|
|
492
|
+
|
|
493
|
+
await this.resolvePendingCheckout(checkout, connection)
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
/**
|
|
497
|
+
* Runs resolve pending checkout.
|
|
498
|
+
* @param {PendingCheckout} checkout - Checkout request to resolve.
|
|
499
|
+
* @param {import("../drivers/base.js").default} connection - Connection to activate.
|
|
500
|
+
* @returns {Promise<void>} - Resolves when the checkout has been handled.
|
|
501
|
+
*/
|
|
502
|
+
async resolvePendingCheckout(checkout, connection) {
|
|
503
|
+
try {
|
|
504
|
+
checkout.resolve(await this.activateConnection(connection, checkout.options))
|
|
505
|
+
} catch (error) {
|
|
506
|
+
checkout.reject(error instanceof Error ? error : new Error("Failed to activate database connection.", {cause: error}))
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
/**
|
|
511
|
+
* Runs close one idle connection for capacity.
|
|
512
|
+
* @returns {Promise<boolean>} - Whether an idle connection was closed to free capacity.
|
|
513
|
+
*/
|
|
514
|
+
async closeOneIdleConnectionForCapacity() {
|
|
515
|
+
const connection = this.connections.find((candidate) => !this.connectionHasOpenTransaction(candidate))
|
|
516
|
+
|
|
517
|
+
if (!connection) return false
|
|
518
|
+
|
|
519
|
+
this.connections = this.connections.filter((candidate) => candidate !== connection)
|
|
520
|
+
await this.closeConnection(connection)
|
|
521
|
+
|
|
522
|
+
return true
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
/**
|
|
526
|
+
* Runs with connection.
|
|
527
|
+
* @template T
|
|
528
|
+
* @param {import("./base.js").ConnectionCheckoutOptions | function(import("../drivers/base.js").default) : Promise<T>} optionsOrCallback - Checkout options or callback to invoke with the connection.
|
|
529
|
+
* @param {function(import("../drivers/base.js").default) : Promise<T>} [callback] - Callback to invoke with the connection.
|
|
530
|
+
* @returns {Promise<T>} - Resolves with the callback result.
|
|
531
|
+
*/
|
|
532
|
+
async withConnection(optionsOrCallback, callback) {
|
|
533
|
+
const options = typeof optionsOrCallback == "function" ? {} : optionsOrCallback
|
|
534
|
+
const actualCallback = typeof optionsOrCallback == "function" ? optionsOrCallback : callback
|
|
535
|
+
|
|
536
|
+
if (!actualCallback) throw new Error("withConnection requires a callback")
|
|
537
|
+
|
|
538
|
+
const connection = await this.checkout(options)
|
|
539
|
+
const id = connection.getIdSeq()
|
|
540
|
+
|
|
541
|
+
return await this.asyncLocalStorage.run(id, async () => {
|
|
542
|
+
try {
|
|
543
|
+
return await actualCallback(connection)
|
|
544
|
+
} finally {
|
|
545
|
+
await this.checkin(connection)
|
|
546
|
+
}
|
|
547
|
+
})
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
/**
|
|
551
|
+
* Runs get current connection.
|
|
552
|
+
* @returns {import("../drivers/base.js").default} - The current connection.
|
|
553
|
+
*/
|
|
554
|
+
getCurrentConnection() {
|
|
555
|
+
const id = this.asyncLocalStorage.getStore()
|
|
556
|
+
|
|
557
|
+
if (id === undefined) return this.currentFallbackConnectionOrFail()
|
|
558
|
+
|
|
559
|
+
this.ensureConnectionIsInUse(id)
|
|
560
|
+
|
|
561
|
+
const currentConnection = this.connectionsInUse[id]
|
|
562
|
+
|
|
563
|
+
if (!currentConnection) {
|
|
564
|
+
throw new Error(`Couldn't get current connection from that ID: ${id}`)
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
return currentConnection
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
/**
|
|
571
|
+
* Runs current fallback connection or fail.
|
|
572
|
+
* @returns {import("../drivers/base.js").default} - Fallback connection, if present.
|
|
573
|
+
*/
|
|
574
|
+
currentFallbackConnectionOrFail() {
|
|
575
|
+
const fallbackConnection = this.getGlobalConnection()
|
|
576
|
+
|
|
577
|
+
if (fallbackConnection) return fallbackConnection
|
|
578
|
+
|
|
579
|
+
throw new Error("ID hasn't been set for this async context")
|
|
580
|
+
}
|
|
581
|
+
|
|
582
|
+
/**
|
|
583
|
+
* Runs ensure connection is in use.
|
|
584
|
+
* @param {number} id - Checked-out connection id.
|
|
585
|
+
* @returns {void}
|
|
586
|
+
*/
|
|
587
|
+
ensureConnectionIsInUse(id) {
|
|
588
|
+
if (!(id in this.connectionsInUse)) {
|
|
589
|
+
throw new Error(`Connection ${id} doesn't exist any more - has it been checked in again?`)
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
|
|
593
|
+
/**
|
|
594
|
+
* Registers a fallback connection for this pool identifier that will be used when no async context is available.
|
|
595
|
+
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
596
|
+
* @returns {void} - No return value.
|
|
597
|
+
*/
|
|
598
|
+
setGlobalConnection(connection) {
|
|
599
|
+
const klass = /**
|
|
600
|
+
* Narrows the runtime value to the documented type.
|
|
601
|
+
@type {typeof VelociousDatabasePoolAsyncTrackedMultiConnection} */ (this.constructor)
|
|
602
|
+
let mapForConfiguration = klass.globalConnections.get(this.configuration)
|
|
603
|
+
|
|
604
|
+
if (!mapForConfiguration) {
|
|
605
|
+
mapForConfiguration = {}
|
|
606
|
+
klass.globalConnections.set(this.configuration, mapForConfiguration)
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
mapForConfiguration[this.identifier] = connection
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
/**
|
|
613
|
+
* Ensures a global fallback connection exists for this pool identifier and returns it.
|
|
614
|
+
* If one is already set, it is returned and also made available in the pool queue.
|
|
615
|
+
* Otherwise a new connection is spawned, registered, and queued.
|
|
616
|
+
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with the global connection.
|
|
617
|
+
*/
|
|
618
|
+
async ensureGlobalConnection() {
|
|
619
|
+
const existing = this.getGlobalConnection()
|
|
620
|
+
|
|
621
|
+
if (existing) return existing
|
|
622
|
+
|
|
623
|
+
const connection = await this.spawnConnection()
|
|
624
|
+
|
|
625
|
+
this.setGlobalConnection(connection)
|
|
626
|
+
|
|
627
|
+
return connection
|
|
628
|
+
}
|
|
629
|
+
|
|
630
|
+
/**
|
|
631
|
+
* Set a shared connection for test mode so that HTTP handlers running
|
|
632
|
+
* in the same process can reuse the test runner's database connection.
|
|
633
|
+
* @param {import("../drivers/base.js").default} connection - Shared connection.
|
|
634
|
+
* @returns {void}
|
|
635
|
+
*/
|
|
636
|
+
setTestSharedConnection(connection) {
|
|
637
|
+
this._testSharedConnection = connection
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
/**
|
|
641
|
+
* Runs clear test shared connection.
|
|
642
|
+
@returns {void} */
|
|
643
|
+
clearTestSharedConnection() {
|
|
644
|
+
this._testSharedConnection = undefined
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
/**
|
|
648
|
+
* Returns the connection tied to the current async context, if any.
|
|
649
|
+
* Falls back to the test shared connection when no async context exists.
|
|
650
|
+
* @returns {import("../drivers/base.js").default | undefined} - The current context connection.
|
|
651
|
+
*/
|
|
652
|
+
getCurrentContextConnection() {
|
|
653
|
+
const id = this.asyncLocalStorage.getStore()
|
|
654
|
+
|
|
655
|
+
if (id === undefined) return this._testSharedConnection
|
|
656
|
+
|
|
657
|
+
return this.getCurrentConnection()
|
|
658
|
+
}
|
|
659
|
+
|
|
660
|
+
/**
|
|
661
|
+
* Runs get debug snapshot.
|
|
662
|
+
* @returns {import("./base.js").DatabasePoolDebugSnapshot} - Diagnostic snapshot for this pool.
|
|
663
|
+
*/
|
|
664
|
+
getDebugSnapshot() {
|
|
665
|
+
const snapshot = super.getDebugSnapshot()
|
|
666
|
+
const now = Date.now()
|
|
667
|
+
const {connections} = this.debugConnectionSnapshots(now)
|
|
668
|
+
|
|
669
|
+
return {
|
|
670
|
+
...snapshot,
|
|
671
|
+
connections,
|
|
672
|
+
connectionsBeingSpawned: this.connectionsBeingSpawned,
|
|
673
|
+
idleCount: this.connections.length,
|
|
674
|
+
inUseCount: Object.keys(this.connectionsInUse).length,
|
|
675
|
+
pendingCheckouts: this.pendingCheckoutDebugSnapshots(now),
|
|
676
|
+
pendingCheckoutCount: this.pendingCheckouts.length
|
|
677
|
+
}
|
|
678
|
+
}
|
|
679
|
+
|
|
680
|
+
/**
|
|
681
|
+
* Runs debug connection snapshots.
|
|
682
|
+
* @param {number} now - Current timestamp.
|
|
683
|
+
* @returns {{connections: Array<Record<string, ?>>, seenConnections: Set<import("../drivers/base.js").default>}} - Connection snapshots and seen set.
|
|
684
|
+
*/
|
|
685
|
+
debugConnectionSnapshots(now) {
|
|
32
686
|
/**
|
|
33
687
|
* Connections.
|
|
688
|
+
@type {Array<Record<string, ?>>} */
|
|
689
|
+
const connections = []
|
|
690
|
+
const seenConnections = new Set()
|
|
691
|
+
|
|
692
|
+
this.addInUseDebugConnectionSnapshots({connections, now, seenConnections})
|
|
693
|
+
this.addIdleDebugConnectionSnapshots({connections, now, seenConnections})
|
|
694
|
+
this.addFallbackDebugConnectionSnapshots({connections, seenConnections})
|
|
695
|
+
|
|
696
|
+
return {connections, seenConnections}
|
|
697
|
+
}
|
|
698
|
+
|
|
699
|
+
/**
|
|
700
|
+
* Runs add in use debug connection snapshots.
|
|
701
|
+
* @param {{connections: Array<Record<string, ?>>, now: number, seenConnections: Set<import("../drivers/base.js").default>}} args - Snapshot collection state.
|
|
702
|
+
* @returns {void}
|
|
703
|
+
*/
|
|
704
|
+
addInUseDebugConnectionSnapshots({connections, now, seenConnections}) {
|
|
705
|
+
for (const [id, connection] of Object.entries(this.connectionsInUse)) {
|
|
706
|
+
const trackedConnection = /**
|
|
707
|
+
* Narrows the runtime value to the documented type.
|
|
708
|
+
@type {import("../drivers/base.js").default & {[CONNECTION_CHECKED_OUT_AT]?: number}} */ (connection)
|
|
709
|
+
const checkedOutAt = trackedConnection[CONNECTION_CHECKED_OUT_AT]
|
|
710
|
+
const checkedOutForMs = typeof checkedOutAt === "number" ? Math.max(0, now - checkedOutAt) : undefined
|
|
711
|
+
|
|
712
|
+
seenConnections.add(connection)
|
|
713
|
+
connections.push(this.debugConnectionSnapshot(connection, {checkedOutAt, checkedOutForMs, checkoutId: id, state: "in-use"}))
|
|
714
|
+
}
|
|
715
|
+
}
|
|
716
|
+
|
|
717
|
+
/**
|
|
718
|
+
* Runs add idle debug connection snapshots.
|
|
719
|
+
* @param {{connections: Array<Record<string, ?>>, now: number, seenConnections: Set<import("../drivers/base.js").default>}} args - Snapshot collection state.
|
|
720
|
+
* @returns {void}
|
|
721
|
+
*/
|
|
722
|
+
addIdleDebugConnectionSnapshots({connections, now, seenConnections}) {
|
|
723
|
+
for (const connection of this.connections) {
|
|
724
|
+
if (seenConnections.has(connection)) continue
|
|
725
|
+
|
|
726
|
+
seenConnections.add(connection)
|
|
727
|
+
|
|
728
|
+
const trackedConnection = /**
|
|
729
|
+
* Narrows the runtime value to the documented type.
|
|
730
|
+
@type {import("../drivers/base.js").default & {[IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
731
|
+
const checkedInAt = trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT]
|
|
732
|
+
const idleForMs = typeof checkedInAt === "number" ? Math.max(0, now - checkedInAt) : undefined
|
|
733
|
+
|
|
734
|
+
connections.push(this.debugConnectionSnapshot(connection, {checkedInAt, idleForMs, state: "idle"}))
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
|
|
738
|
+
/**
|
|
739
|
+
* Runs add fallback debug connection snapshots.
|
|
740
|
+
* @param {{connections: Array<Record<string, ?>>, seenConnections: Set<import("../drivers/base.js").default>}} args - Snapshot collection state.
|
|
741
|
+
* @returns {void}
|
|
742
|
+
*/
|
|
743
|
+
addFallbackDebugConnectionSnapshots({connections, seenConnections}) {
|
|
744
|
+
this.addDebugConnectionSnapshotIfUnseen({connection: this.getGlobalConnectionForIdentifier(), connections, reapable: false, seenConnections, state: "global"})
|
|
745
|
+
this.addDebugConnectionSnapshotIfUnseen({connection: this._testSharedConnection, connections, reapable: false, seenConnections, state: "test-shared"})
|
|
746
|
+
}
|
|
747
|
+
|
|
748
|
+
/**
|
|
749
|
+
* Runs add debug connection snapshot if unseen.
|
|
750
|
+
* @param {{connection: import("../drivers/base.js").default | undefined, connections: Array<Record<string, ?>>, reapable?: boolean, seenConnections: Set<import("../drivers/base.js").default>, state: string}} args - Snapshot collection state.
|
|
751
|
+
* @returns {void}
|
|
752
|
+
*/
|
|
753
|
+
addDebugConnectionSnapshotIfUnseen({connection, connections, reapable, seenConnections, state}) {
|
|
754
|
+
if (!connection || seenConnections.has(connection)) return
|
|
755
|
+
|
|
756
|
+
seenConnections.add(connection)
|
|
757
|
+
connections.push(this.debugConnectionSnapshot(connection, {reapable, state}))
|
|
758
|
+
}
|
|
759
|
+
|
|
760
|
+
/**
|
|
761
|
+
* Runs pending checkout debug snapshots.
|
|
762
|
+
* @param {number} now - Current timestamp.
|
|
763
|
+
* @returns {Array<Record<string, ?>>} - Pending checkout snapshots.
|
|
764
|
+
*/
|
|
765
|
+
pendingCheckoutDebugSnapshots(now) {
|
|
766
|
+
return this.pendingCheckouts.map((checkout, index) => ({
|
|
767
|
+
checkoutName: checkout.options.name,
|
|
768
|
+
enqueuedAt: checkout.enqueuedAt,
|
|
769
|
+
index,
|
|
770
|
+
reuseKey: checkout.reuseKey,
|
|
771
|
+
waitingForMs: Math.max(0, now - checkout.enqueuedAt)
|
|
772
|
+
}))
|
|
773
|
+
}
|
|
774
|
+
|
|
775
|
+
/**
|
|
776
|
+
* Runs get global connection.
|
|
777
|
+
* @returns {import("../drivers/base.js").default | undefined} - The global connection.
|
|
778
|
+
*/
|
|
779
|
+
getGlobalConnection() {
|
|
780
|
+
const connection = this.getGlobalConnectionForIdentifier()
|
|
781
|
+
|
|
782
|
+
if (!connection) return
|
|
783
|
+
if (!this.connectionMatchesCurrentConfiguration(connection)) return
|
|
784
|
+
|
|
785
|
+
return connection
|
|
786
|
+
}
|
|
787
|
+
|
|
788
|
+
/**
|
|
789
|
+
* Runs get global connection for identifier.
|
|
790
|
+
* @returns {import("../drivers/base.js").default | undefined} - The global connection for this pool identifier.
|
|
791
|
+
*/
|
|
792
|
+
getGlobalConnectionForIdentifier() {
|
|
793
|
+
const klass = /**
|
|
794
|
+
* Narrows the runtime value to the documented type.
|
|
795
|
+
@type {typeof VelociousDatabasePoolAsyncTrackedMultiConnection} */ (this.constructor)
|
|
796
|
+
const mapForConfiguration = klass.globalConnections.get(this.configuration)
|
|
797
|
+
|
|
798
|
+
return mapForConfiguration?.[this.identifier]
|
|
799
|
+
}
|
|
800
|
+
|
|
801
|
+
/**
|
|
802
|
+
* Runs clear global connection for identifier.
|
|
803
|
+
* @returns {void} - No return value.
|
|
804
|
+
*/
|
|
805
|
+
clearGlobalConnectionForIdentifier() {
|
|
806
|
+
const klass = /**
|
|
807
|
+
* Narrows the runtime value to the documented type.
|
|
808
|
+
@type {typeof VelociousDatabasePoolAsyncTrackedMultiConnection} */ (this.constructor)
|
|
809
|
+
const mapForConfiguration = klass.globalConnections.get(this.configuration)
|
|
810
|
+
|
|
811
|
+
if (!mapForConfiguration) return
|
|
812
|
+
|
|
813
|
+
delete mapForConfiguration[this.identifier]
|
|
814
|
+
}
|
|
815
|
+
|
|
816
|
+
/**
|
|
817
|
+
* Clears schema metadata cached by every live connection owned by this pool.
|
|
818
|
+
* @returns {void} - No return value.
|
|
819
|
+
*/
|
|
820
|
+
clearSchemaCache() {
|
|
821
|
+
const connections = new Set([
|
|
822
|
+
...this.connections,
|
|
823
|
+
...Object.values(this.connectionsInUse),
|
|
824
|
+
this.getGlobalConnection(),
|
|
825
|
+
this._testSharedConnection
|
|
826
|
+
].filter(Boolean))
|
|
827
|
+
|
|
828
|
+
for (const connection of connections) {
|
|
829
|
+
if (connection) this._clearConnectionSchemaCache(connection)
|
|
830
|
+
}
|
|
831
|
+
}
|
|
832
|
+
|
|
833
|
+
/**
|
|
834
|
+
* Runs idle timeout millis.
|
|
835
|
+
* @returns {number | null} - Idle timeout in milliseconds, or null when disabled.
|
|
836
|
+
*/
|
|
837
|
+
idleTimeoutMillis() {
|
|
838
|
+
const value = this.getConfiguration().pool?.idleTimeoutMillis
|
|
839
|
+
|
|
840
|
+
if (value === null) return null
|
|
841
|
+
if (this.validIdleTimeoutMillis(value)) return value
|
|
842
|
+
|
|
843
|
+
return DEFAULT_IDLE_TIMEOUT_MILLIS
|
|
844
|
+
}
|
|
845
|
+
|
|
846
|
+
/**
|
|
847
|
+
* Runs valid idle timeout millis.
|
|
848
|
+
* @param {?} value - Candidate idle timeout value.
|
|
849
|
+
* @returns {value is number} - Whether the value is a valid idle timeout.
|
|
850
|
+
*/
|
|
851
|
+
validIdleTimeoutMillis(value) {
|
|
852
|
+
return typeof value === "number" && Number.isFinite(value) && value >= 0
|
|
853
|
+
}
|
|
854
|
+
|
|
855
|
+
/**
|
|
856
|
+
* Runs schedule idle connection reaper.
|
|
857
|
+
@returns {void} */
|
|
858
|
+
scheduleIdleConnectionReaper() {
|
|
859
|
+
if (this.idleConnectionReaperTimer) return
|
|
860
|
+
if (!this.hasIdleConnectionsToReap()) return
|
|
861
|
+
|
|
862
|
+
const delay = this.nextIdleConnectionReapDelay(/**
|
|
863
|
+
* Narrows the runtime value to the documented type.
|
|
864
|
+
@type {number} */ (this.idleTimeoutMillis()))
|
|
865
|
+
|
|
866
|
+
this.idleConnectionReaperTimer = setTimeout(() => {
|
|
867
|
+
this.idleConnectionReaperTimer = undefined
|
|
868
|
+
void this.reapIdleConnections().catch((error) => {
|
|
869
|
+
this.logger.warn(() => ["Failed to reap idle database connections:", error])
|
|
870
|
+
})
|
|
871
|
+
}, delay)
|
|
872
|
+
|
|
873
|
+
if (typeof this.idleConnectionReaperTimer.unref === "function") {
|
|
874
|
+
this.idleConnectionReaperTimer.unref()
|
|
875
|
+
}
|
|
876
|
+
}
|
|
877
|
+
|
|
878
|
+
/**
|
|
879
|
+
* Runs has idle connections to reap.
|
|
880
|
+
* @returns {boolean} - Whether an idle reaper timer should be scheduled.
|
|
881
|
+
*/
|
|
882
|
+
hasIdleConnectionsToReap() {
|
|
883
|
+
return this.connections.length > 0 && this.idleTimeoutMillis() !== null
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
/**
|
|
887
|
+
* Runs next idle connection reap delay.
|
|
888
|
+
* @param {number} idleTimeoutMillis - Idle timeout in milliseconds.
|
|
889
|
+
* @returns {number} - Delay before the next reap.
|
|
890
|
+
*/
|
|
891
|
+
nextIdleConnectionReapDelay(idleTimeoutMillis) {
|
|
892
|
+
let delay = idleTimeoutMillis
|
|
893
|
+
const now = Date.now()
|
|
894
|
+
|
|
895
|
+
for (const connection of this.connections) {
|
|
896
|
+
if (this.connectionHasOpenTransaction(connection)) continue
|
|
897
|
+
|
|
898
|
+
const trackedConnection = /**
|
|
899
|
+
* Narrows the runtime value to the documented type.
|
|
900
|
+
@type {import("../drivers/base.js").default & {[IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
901
|
+
const checkedInAt = trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT]
|
|
902
|
+
|
|
903
|
+
if (typeof checkedInAt !== "number") continue
|
|
904
|
+
|
|
905
|
+
delay = Math.min(delay, Math.max(0, idleTimeoutMillis - (now - checkedInAt)))
|
|
906
|
+
}
|
|
907
|
+
|
|
908
|
+
return delay
|
|
909
|
+
}
|
|
910
|
+
|
|
911
|
+
/**
|
|
912
|
+
* Closes idle checked-in connections that have exceeded the configured timeout.
|
|
913
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
914
|
+
*/
|
|
915
|
+
async reapIdleConnections() {
|
|
916
|
+
if (this.connections.length === 0) return
|
|
917
|
+
|
|
918
|
+
const idleTimeoutMillis = this.idleTimeoutMillis()
|
|
919
|
+
|
|
920
|
+
if (idleTimeoutMillis === null) return
|
|
921
|
+
|
|
922
|
+
const {expiredConnections, keptConnections} = this.classifyIdleConnectionsForReaping({idleTimeoutMillis, now: Date.now()})
|
|
923
|
+
|
|
924
|
+
this.connections = keptConnections
|
|
925
|
+
await this.closeExpiredIdleConnections(expiredConnections)
|
|
926
|
+
await this.awaitInflightConnectionCloses()
|
|
927
|
+
if (this.connections.length > 0) this.scheduleIdleConnectionReaper()
|
|
928
|
+
}
|
|
929
|
+
|
|
930
|
+
/**
|
|
931
|
+
* Runs close expired idle connections.
|
|
932
|
+
* @param {import("../drivers/base.js").default[]} expiredConnections - Connections to close.
|
|
933
|
+
* @returns {Promise<void>} - Resolves when closed.
|
|
934
|
+
*/
|
|
935
|
+
async closeExpiredIdleConnections(expiredConnections) {
|
|
936
|
+
for (const connection of expiredConnections) {
|
|
937
|
+
await this.closeConnection(connection)
|
|
938
|
+
}
|
|
939
|
+
}
|
|
940
|
+
|
|
941
|
+
/**
|
|
942
|
+
* Runs await inflight connection closes.
|
|
943
|
+
* @returns {Promise<void>} - Resolves once in-flight connection closes settle.
|
|
944
|
+
*/
|
|
945
|
+
async awaitInflightConnectionCloses() {
|
|
946
|
+
if (this.inflightConnectionCloses.size > 0) {
|
|
947
|
+
await Promise.allSettled([...this.inflightConnectionCloses])
|
|
948
|
+
}
|
|
949
|
+
}
|
|
950
|
+
|
|
951
|
+
/**
|
|
952
|
+
* Runs classify idle connections for reaping.
|
|
953
|
+
* @param {{idleTimeoutMillis: number, now: number}} args - Reaper classification inputs.
|
|
954
|
+
* @returns {{expiredConnections: import("../drivers/base.js").default[], keptConnections: import("../drivers/base.js").default[]}} - Classified idle connections.
|
|
955
|
+
*/
|
|
956
|
+
classifyIdleConnectionsForReaping({idleTimeoutMillis, now}) {
|
|
957
|
+
/**
|
|
958
|
+
* Kept connections.
|
|
34
959
|
@type {import("../drivers/base.js").default[]} */
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Connections in use.
|
|
38
|
-
@type {Record<number, import("../drivers/base.js").default>} */
|
|
39
|
-
connectionsInUse = {};
|
|
40
|
-
/**
|
|
41
|
-
* Pending checkouts.
|
|
42
|
-
@type {PendingCheckout[]} */
|
|
43
|
-
pendingCheckouts = [];
|
|
44
|
-
/**
|
|
45
|
-
* Connections being spawned.
|
|
46
|
-
@type {number} */
|
|
47
|
-
connectionsBeingSpawned = 0;
|
|
48
|
-
/**
|
|
49
|
-
* Pending checkout drain promise.
|
|
50
|
-
@type {Promise<void> | undefined} */
|
|
51
|
-
pendingCheckoutDrainPromise = undefined;
|
|
52
|
-
/**
|
|
53
|
-
* Idle connection reaper timer.
|
|
54
|
-
@type {ReturnType<typeof setTimeout> | undefined} */
|
|
55
|
-
idleConnectionReaperTimer = undefined;
|
|
56
|
-
/**
|
|
57
|
-
* In-flight connection-close promises. The idle reaper is armed on check-in
|
|
58
|
-
* and runs fire-and-forget when its timer fires, so a scheduled reap can be
|
|
59
|
-
* closing a connection while an explicit `reapIdleConnections()` (or
|
|
60
|
-
* `clearIdleConnectionReaperTimer()`) runs. Tracking the in-flight closes lets
|
|
61
|
-
* those callers await them, so once a reap resolves the connections it
|
|
62
|
-
* expired are fully closed instead of half-closed mid-`close()`.
|
|
63
|
-
* @type {Set<Promise<void>>}
|
|
64
|
-
*/
|
|
65
|
-
inflightConnectionCloses = new Set();
|
|
66
|
-
/**
|
|
67
|
-
* In-flight close promise per connection, so concurrent closes of the same
|
|
68
|
-
* connection await the same close rather than closing the driver handle twice.
|
|
69
|
-
* @type {WeakMap<object, Promise<void>>}
|
|
70
|
-
*/
|
|
71
|
-
connectionClosePromises = new WeakMap();
|
|
72
|
-
idSeq = 0;
|
|
73
|
-
/**
|
|
74
|
-
* Runs constructor.
|
|
75
|
-
* @param {object} args - Options object.
|
|
76
|
-
* @param {import("../../configuration.js").default} args.configuration - Configuration instance.
|
|
77
|
-
* @param {string} args.identifier - Identifier.
|
|
78
|
-
*/
|
|
79
|
-
constructor({ configuration, identifier }) {
|
|
80
|
-
super({ configuration, identifier });
|
|
81
|
-
this._withoutCurrentConnectionContext = /**
|
|
82
|
-
* Narrows the runtime value to the documented type.
|
|
83
|
-
@type {(callback: () => ?) => ?} */
|
|
84
|
-
((callback) => this.asyncLocalStorage.run(undefined, callback));
|
|
85
|
-
}
|
|
86
|
-
/**
|
|
87
|
-
* Runs checkin.
|
|
88
|
-
* @param {import("../drivers/base.js").default} connection - Database connection instance.
|
|
89
|
-
* @returns {Promise<void>} - Resolves when the connection is checked in or closed.
|
|
90
|
-
*/
|
|
91
|
-
async checkin(connection) {
|
|
92
|
-
const id = connection.getIdSeq();
|
|
93
|
-
const trackedConnection = /**
|
|
94
|
-
* Narrows the runtime value to the documented type.
|
|
95
|
-
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean, [CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection);
|
|
96
|
-
if (trackedConnection[CLOSED_CONNECTION]) {
|
|
97
|
-
this.untrackConnectionInUse(connection, id);
|
|
98
|
-
await this.drainPendingCheckouts();
|
|
99
|
-
return;
|
|
100
|
-
}
|
|
101
|
-
try {
|
|
102
|
-
await this.rollbackLeftOpenTransaction(connection);
|
|
103
|
-
await connection.clearConnectionCheckoutName();
|
|
104
|
-
}
|
|
105
|
-
catch (error) {
|
|
106
|
-
await this.closeCheckedOutConnectionAfterCheckinFailure(connection, id, error);
|
|
107
|
-
throw error;
|
|
108
|
-
}
|
|
109
|
-
this.untrackConnectionInUse(connection, id);
|
|
110
|
-
trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT] = Date.now();
|
|
111
|
-
delete trackedConnection[CONNECTION_CHECKED_OUT_AT];
|
|
112
|
-
this.connections.push(connection);
|
|
113
|
-
await this.drainPendingCheckouts();
|
|
114
|
-
if (this.connections.includes(connection))
|
|
115
|
-
await this.handleCheckedInIdleConnection();
|
|
116
|
-
}
|
|
117
|
-
/**
|
|
118
|
-
* Runs close checked out connection after checkin failure.
|
|
119
|
-
* @param {import("../drivers/base.js").default} connection - Connection that failed check-in cleanup.
|
|
120
|
-
* @param {number | undefined} id - Connection checkout id.
|
|
121
|
-
* @param {?} originalError - Error that caused check-in cleanup to fail.
|
|
122
|
-
* @returns {Promise<void>} - Resolves when cleanup has been attempted.
|
|
123
|
-
*/
|
|
124
|
-
async closeCheckedOutConnectionAfterCheckinFailure(connection, id, originalError) {
|
|
125
|
-
this.untrackConnectionInUse(connection, id);
|
|
126
|
-
try {
|
|
127
|
-
await this.closeConnection(connection);
|
|
128
|
-
}
|
|
129
|
-
catch (error) {
|
|
130
|
-
this.logger.warn("Failed to close database connection after check-in cleanup failed", { error, originalError });
|
|
131
|
-
}
|
|
132
|
-
try {
|
|
133
|
-
await this.drainPendingCheckouts();
|
|
134
|
-
}
|
|
135
|
-
catch (error) {
|
|
136
|
-
this.logger.warn("Failed to drain pending database checkouts after check-in cleanup failed", { error, originalError });
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Runs untrack connection in use.
|
|
141
|
-
* @param {import("../drivers/base.js").default} connection - Connection being checked in.
|
|
142
|
-
* @param {number | undefined} id - Connection checkout id.
|
|
143
|
-
* @returns {void}
|
|
144
|
-
*/
|
|
145
|
-
untrackConnectionInUse(connection, id) {
|
|
146
|
-
if (typeof id !== "number") {
|
|
147
|
-
throw new Error(`idSeq on connection wasn't set? '${typeof id}' = ${id}`);
|
|
148
|
-
}
|
|
149
|
-
delete this.connectionsInUse[id];
|
|
150
|
-
connection.setIdSeq(undefined);
|
|
151
|
-
}
|
|
152
|
-
/**
|
|
153
|
-
* Runs handle checked in idle connection.
|
|
154
|
-
* @returns {Promise<void>} - Resolves once idle reaping has been scheduled or run.
|
|
155
|
-
*/
|
|
156
|
-
async handleCheckedInIdleConnection() {
|
|
157
|
-
if (this.idleTimeoutMillis() === 0) {
|
|
158
|
-
await this.reapIdleConnections();
|
|
159
|
-
}
|
|
160
|
-
else {
|
|
161
|
-
this.scheduleIdleConnectionReaper();
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
/**
|
|
165
|
-
* Runs checkout.
|
|
166
|
-
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
167
|
-
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with the checkout.
|
|
168
|
-
*/
|
|
169
|
-
async checkout(options = {}) {
|
|
170
|
-
const databaseConfig = this.getConfiguration();
|
|
171
|
-
const reuseKey = this.getConfigurationReuseKey();
|
|
172
|
-
let connection = this.takeIdleConnectionForReuseKey(reuseKey);
|
|
173
|
-
if (connection)
|
|
174
|
-
return await this.activateConnection(connection, options);
|
|
175
|
-
await this.reapIdleConnections();
|
|
176
|
-
connection = this.takeIdleConnectionForReuseKey(reuseKey);
|
|
177
|
-
if (connection)
|
|
178
|
-
return await this.activateConnection(connection, options);
|
|
179
|
-
if (this.canSpawnConnection()) {
|
|
180
|
-
// Spawn via spawnConnection() so the tenant-aware configuration is resolved FRESH at
|
|
181
|
-
// spawn time for the current caller. Reusing the databaseConfig captured at the top of
|
|
182
|
-
// checkout() could bind the connection to a stale tenant/database, which breaks
|
|
183
|
-
// per-request isolation (e.g. test truncation appearing not to take effect). The queued
|
|
184
|
-
// path below keeps the waiting caller's captured config via waitForCheckout().
|
|
185
|
-
connection = await this.spawnConnectionForCheckout(this.getConfiguration(), this.getConfigurationReuseKey());
|
|
186
|
-
return await this.activateConnection(connection, options);
|
|
187
|
-
}
|
|
188
|
-
return await this.waitForCheckout(databaseConfig, reuseKey, options);
|
|
189
|
-
}
|
|
190
|
-
/**
|
|
191
|
-
* Runs take idle connection for reuse key.
|
|
192
|
-
* @param {string} reuseKey - Database configuration reuse key.
|
|
193
|
-
* @param {object} [args] - Options.
|
|
194
|
-
* @param {boolean} [args.includeOpenTransactions] - Whether connections with open transactions may be returned.
|
|
195
|
-
* @returns {import("../drivers/base.js").default | undefined} - Matching idle connection.
|
|
196
|
-
*/
|
|
197
|
-
takeIdleConnectionForReuseKey(reuseKey, { includeOpenTransactions = true } = {}) {
|
|
198
|
-
const connectionIndex = this.connections.findIndex((queuedConnection) => {
|
|
199
|
-
if (!includeOpenTransactions && this.connectionHasOpenTransaction(queuedConnection))
|
|
200
|
-
return false;
|
|
201
|
-
return this.connectionMatchesReuseKey(queuedConnection, reuseKey);
|
|
202
|
-
});
|
|
203
|
-
const connection = connectionIndex === -1 ? undefined : this.connections.splice(connectionIndex, 1)[0];
|
|
204
|
-
return connection;
|
|
205
|
-
}
|
|
206
|
-
/**
|
|
207
|
-
* Runs connection matches reuse key.
|
|
208
|
-
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
209
|
-
* @param {string} reuseKey - Database configuration reuse key.
|
|
210
|
-
* @returns {boolean} - Whether the connection matches the reuse key.
|
|
211
|
-
*/
|
|
212
|
-
connectionMatchesReuseKey(connection, reuseKey) {
|
|
213
|
-
const connectionWithPoolKey = /**
|
|
214
|
-
* Narrows the runtime value to the documented type.
|
|
215
|
-
@type {import("../drivers/base.js").default & {[POOL_CONFIGURATION_KEY]?: string}} */ (connection);
|
|
216
|
-
return connectionWithPoolKey[POOL_CONFIGURATION_KEY] === reuseKey;
|
|
217
|
-
}
|
|
218
|
-
/**
|
|
219
|
-
* Runs activate connection.
|
|
220
|
-
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
221
|
-
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
222
|
-
* @returns {Promise<import("../drivers/base.js").default>} - Activated connection.
|
|
223
|
-
*/
|
|
224
|
-
async activateConnection(connection, options = {}) {
|
|
225
|
-
if (connection.getIdSeq() !== undefined)
|
|
226
|
-
throw new Error(`Connection already has an ID-seq - is it in use? ${connection.getIdSeq()}`);
|
|
227
|
-
const id = this.idSeq++;
|
|
228
|
-
const trackedConnection = /**
|
|
229
|
-
* Narrows the runtime value to the documented type.
|
|
230
|
-
@type {import("../drivers/base.js").default & {[CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection);
|
|
231
|
-
delete trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT];
|
|
232
|
-
trackedConnection[CONNECTION_CHECKED_OUT_AT] = Date.now();
|
|
233
|
-
connection.setIdSeq(id);
|
|
234
|
-
this.connectionsInUse[id] = connection;
|
|
235
|
-
try {
|
|
236
|
-
await connection.setConnectionCheckoutName(options.name);
|
|
237
|
-
}
|
|
238
|
-
catch (error) {
|
|
239
|
-
delete this.connectionsInUse[id];
|
|
240
|
-
connection.setIdSeq(undefined);
|
|
241
|
-
await this.closeConnection(connection);
|
|
242
|
-
throw error;
|
|
243
|
-
}
|
|
244
|
-
return connection;
|
|
245
|
-
}
|
|
246
|
-
/**
|
|
247
|
-
* Runs max connections.
|
|
248
|
-
* @returns {number | undefined} - Configured max live connections.
|
|
249
|
-
*/
|
|
250
|
-
maxConnections() {
|
|
251
|
-
const value = this.getConfiguration().pool?.max;
|
|
252
|
-
if (this.validMaxConnections(value))
|
|
253
|
-
return value;
|
|
254
|
-
return;
|
|
255
|
-
}
|
|
256
|
-
/**
|
|
257
|
-
* Runs valid max connections.
|
|
258
|
-
* @param {?} value - Candidate max connection count.
|
|
259
|
-
* @returns {value is number} - Whether the value is a valid max connection count.
|
|
260
|
-
*/
|
|
261
|
-
validMaxConnections(value) {
|
|
262
|
-
return typeof value === "number" && Number.isFinite(value) && value >= 1;
|
|
263
|
-
}
|
|
264
|
-
/**
|
|
265
|
-
* Runs live connection count.
|
|
266
|
-
* @returns {number} - Number of live and in-progress connections.
|
|
267
|
-
*/
|
|
268
|
-
liveConnectionCount() {
|
|
269
|
-
const connections = new Set([
|
|
270
|
-
...this.connections,
|
|
271
|
-
...Object.values(this.connectionsInUse),
|
|
272
|
-
this.getGlobalConnectionForIdentifier()
|
|
273
|
-
].filter(Boolean));
|
|
274
|
-
return connections.size + this.connectionsBeingSpawned;
|
|
275
|
-
}
|
|
276
|
-
/**
|
|
277
|
-
* Runs can spawn connection.
|
|
278
|
-
* @returns {boolean} - Whether a new connection can be spawned.
|
|
279
|
-
*/
|
|
280
|
-
canSpawnConnection() {
|
|
281
|
-
const maxConnections = this.maxConnections();
|
|
282
|
-
return maxConnections === undefined || this.liveConnectionCount() < maxConnections;
|
|
283
|
-
}
|
|
284
|
-
/**
|
|
285
|
-
* Runs spawn connection for checkout.
|
|
286
|
-
* @param {import("../../configuration-types.js").DatabaseConfigurationType} databaseConfig - Resolved database config for the checkout.
|
|
287
|
-
* @param {string} reuseKey - Database configuration reuse key for the checkout.
|
|
288
|
-
* @returns {Promise<import("../drivers/base.js").default>} - Spawned connection.
|
|
289
|
-
*/
|
|
290
|
-
async spawnConnectionForCheckout(databaseConfig, reuseKey) {
|
|
291
|
-
this.connectionsBeingSpawned++;
|
|
292
|
-
try {
|
|
293
|
-
const connection = await this.spawnConnectionWithConfiguration(databaseConfig);
|
|
294
|
-
const connectionWithPoolKey = /**
|
|
295
|
-
* Narrows the runtime value to the documented type.
|
|
296
|
-
@type {import("../drivers/base.js").default & {[POOL_CONFIGURATION_KEY]?: string}} */ (connection);
|
|
297
|
-
connectionWithPoolKey[POOL_CONFIGURATION_KEY] = reuseKey;
|
|
298
|
-
connection.setSchemaCacheInvalidator(() => {
|
|
299
|
-
this.clearSchemaCache();
|
|
300
|
-
this.configuration.clearSchemaCachesForReuseKey(reuseKey);
|
|
301
|
-
});
|
|
302
|
-
return connection;
|
|
303
|
-
}
|
|
304
|
-
finally {
|
|
305
|
-
this.connectionsBeingSpawned--;
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
/**
|
|
309
|
-
* Runs wait for checkout.
|
|
310
|
-
* @param {import("../../configuration-types.js").DatabaseConfigurationType} databaseConfig - Resolved database config for the checkout.
|
|
311
|
-
* @param {string} reuseKey - Database configuration reuse key.
|
|
312
|
-
* @param {import("./base.js").ConnectionCheckoutOptions} [options] - Checkout options.
|
|
313
|
-
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with an activated connection.
|
|
314
|
-
*/
|
|
315
|
-
async waitForCheckout(databaseConfig, reuseKey, options = {}) {
|
|
316
|
-
return await new Promise((resolve, reject) => {
|
|
317
|
-
this.pendingCheckouts.push({ databaseConfig, enqueuedAt: Date.now(), options, reject, resolve, reuseKey });
|
|
318
|
-
void this.drainPendingCheckouts().catch((error) => {
|
|
319
|
-
const checkoutError = error instanceof Error ? error : new Error("Failed to drain pending database connection checkouts.", { cause: error });
|
|
320
|
-
this.rejectPendingCheckouts(checkoutError);
|
|
321
|
-
});
|
|
322
|
-
});
|
|
323
|
-
}
|
|
324
|
-
/**
|
|
325
|
-
* Runs drain pending checkouts.
|
|
326
|
-
* @returns {Promise<void>} - Resolves when pending checkouts have been drained as far as possible.
|
|
327
|
-
*/
|
|
328
|
-
async drainPendingCheckouts() {
|
|
329
|
-
if (this.pendingCheckoutDrainPromise) {
|
|
330
|
-
await this.pendingCheckoutDrainPromise;
|
|
331
|
-
return;
|
|
332
|
-
}
|
|
333
|
-
this.pendingCheckoutDrainPromise = this.drainPendingCheckoutsActual();
|
|
334
|
-
try {
|
|
335
|
-
await this.pendingCheckoutDrainPromise;
|
|
336
|
-
}
|
|
337
|
-
finally {
|
|
338
|
-
this.pendingCheckoutDrainPromise = undefined;
|
|
339
|
-
}
|
|
340
|
-
}
|
|
341
|
-
/**
|
|
342
|
-
* Runs drain pending checkouts actual.
|
|
343
|
-
* @returns {Promise<void>} - Resolves when pending checkouts have been drained as far as possible.
|
|
344
|
-
*/
|
|
345
|
-
async drainPendingCheckoutsActual() {
|
|
346
|
-
while (this.pendingCheckouts.length > 0) {
|
|
347
|
-
if (await this.resolvePendingCheckoutWithMatchingIdleConnection())
|
|
348
|
-
continue;
|
|
349
|
-
const checkout = this.pendingCheckouts[0];
|
|
350
|
-
if (await this.closeIdleConnectionForPendingCheckoutCapacity(checkout))
|
|
351
|
-
continue;
|
|
352
|
-
if (this.canSpawnConnection()) {
|
|
353
|
-
this.pendingCheckouts.shift();
|
|
354
|
-
await this.spawnAndResolvePendingCheckout(checkout);
|
|
355
|
-
continue;
|
|
356
|
-
}
|
|
357
|
-
const reapedConnection = await this.idleConnectionForPendingCheckout(checkout);
|
|
358
|
-
if (!reapedConnection)
|
|
359
|
-
return;
|
|
360
|
-
this.pendingCheckouts.shift();
|
|
361
|
-
await this.resolvePendingCheckout(checkout, reapedConnection);
|
|
362
|
-
}
|
|
363
|
-
}
|
|
364
|
-
/**
|
|
365
|
-
* Runs resolve pending checkout with matching idle connection.
|
|
366
|
-
* @returns {Promise<boolean>} - Whether a pending checkout was resolved with an idle connection.
|
|
367
|
-
*/
|
|
368
|
-
async resolvePendingCheckoutWithMatchingIdleConnection() {
|
|
369
|
-
for (let index = 0; index < this.pendingCheckouts.length; index++) {
|
|
370
|
-
const checkout = this.pendingCheckouts[index];
|
|
371
|
-
const connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, { includeOpenTransactions: false });
|
|
372
|
-
if (!connection)
|
|
373
|
-
continue;
|
|
374
|
-
this.pendingCheckouts.splice(index, 1);
|
|
375
|
-
await this.resolvePendingCheckout(checkout, connection);
|
|
376
|
-
return true;
|
|
377
|
-
}
|
|
378
|
-
return false;
|
|
379
|
-
}
|
|
380
|
-
/**
|
|
381
|
-
* Runs close idle connection for pending checkout capacity.
|
|
382
|
-
* @param {PendingCheckout} checkout - Checkout waiting for a connection.
|
|
383
|
-
* @returns {Promise<boolean>} - Whether an idle connection was closed to free capacity.
|
|
384
|
-
*/
|
|
385
|
-
async closeIdleConnectionForPendingCheckoutCapacity(checkout) {
|
|
386
|
-
const connection = this.findIdleConnectionForReuseKey(checkout.reuseKey);
|
|
387
|
-
if (connection)
|
|
388
|
-
return false;
|
|
389
|
-
await this.reapIdleConnections();
|
|
390
|
-
if (this.findIdleConnectionForReuseKey(checkout.reuseKey))
|
|
391
|
-
return false;
|
|
392
|
-
return this.canSpawnConnection() ? false : await this.closeOneIdleConnectionForCapacity();
|
|
393
|
-
}
|
|
394
|
-
/**
|
|
395
|
-
* Runs find idle connection for reuse key.
|
|
396
|
-
* @param {string} reuseKey - Database configuration reuse key.
|
|
397
|
-
* @returns {import("../drivers/base.js").default | undefined} - Matching idle connection, if present.
|
|
398
|
-
*/
|
|
399
|
-
findIdleConnectionForReuseKey(reuseKey) {
|
|
400
|
-
return this.connections.find((connection) => !this.connectionHasOpenTransaction(connection) && this.connectionMatchesReuseKey(connection, reuseKey));
|
|
401
|
-
}
|
|
402
|
-
/**
|
|
403
|
-
* Runs idle connection for pending checkout.
|
|
404
|
-
* @param {PendingCheckout} checkout - Checkout waiting for a connection.
|
|
405
|
-
* @returns {Promise<import("../drivers/base.js").default | undefined>} - Matching idle connection, if one can be reused.
|
|
406
|
-
*/
|
|
407
|
-
async idleConnectionForPendingCheckout(checkout) {
|
|
408
|
-
let connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, { includeOpenTransactions: false });
|
|
409
|
-
if (connection)
|
|
410
|
-
return connection;
|
|
411
|
-
await this.reapIdleConnections();
|
|
412
|
-
connection = this.takeIdleConnectionForReuseKey(checkout.reuseKey, { includeOpenTransactions: false });
|
|
413
|
-
return connection;
|
|
414
|
-
}
|
|
415
|
-
/**
|
|
416
|
-
* Runs spawn and resolve pending checkout.
|
|
417
|
-
* @param {PendingCheckout} checkout - Checkout request to resolve.
|
|
418
|
-
* @returns {Promise<void>} - Resolves when the checkout has been handled.
|
|
419
|
-
*/
|
|
420
|
-
async spawnAndResolvePendingCheckout(checkout) {
|
|
421
|
-
let connection;
|
|
422
|
-
try {
|
|
423
|
-
connection = await this.spawnConnectionForCheckout(checkout.databaseConfig, checkout.reuseKey);
|
|
424
|
-
}
|
|
425
|
-
catch (error) {
|
|
426
|
-
checkout.reject(error instanceof Error ? error : new Error("Failed to spawn database connection.", { cause: error }));
|
|
427
|
-
return;
|
|
428
|
-
}
|
|
429
|
-
await this.resolvePendingCheckout(checkout, connection);
|
|
430
|
-
}
|
|
431
|
-
/**
|
|
432
|
-
* Runs resolve pending checkout.
|
|
433
|
-
* @param {PendingCheckout} checkout - Checkout request to resolve.
|
|
434
|
-
* @param {import("../drivers/base.js").default} connection - Connection to activate.
|
|
435
|
-
* @returns {Promise<void>} - Resolves when the checkout has been handled.
|
|
436
|
-
*/
|
|
437
|
-
async resolvePendingCheckout(checkout, connection) {
|
|
438
|
-
try {
|
|
439
|
-
checkout.resolve(await this.activateConnection(connection, checkout.options));
|
|
440
|
-
}
|
|
441
|
-
catch (error) {
|
|
442
|
-
checkout.reject(error instanceof Error ? error : new Error("Failed to activate database connection.", { cause: error }));
|
|
443
|
-
}
|
|
444
|
-
}
|
|
445
|
-
/**
|
|
446
|
-
* Runs close one idle connection for capacity.
|
|
447
|
-
* @returns {Promise<boolean>} - Whether an idle connection was closed to free capacity.
|
|
448
|
-
*/
|
|
449
|
-
async closeOneIdleConnectionForCapacity() {
|
|
450
|
-
const connection = this.connections.find((candidate) => !this.connectionHasOpenTransaction(candidate));
|
|
451
|
-
if (!connection)
|
|
452
|
-
return false;
|
|
453
|
-
this.connections = this.connections.filter((candidate) => candidate !== connection);
|
|
454
|
-
await this.closeConnection(connection);
|
|
455
|
-
return true;
|
|
456
|
-
}
|
|
457
|
-
/**
|
|
458
|
-
* Runs with connection.
|
|
459
|
-
* @template T
|
|
460
|
-
* @param {import("./base.js").ConnectionCheckoutOptions | function(import("../drivers/base.js").default) : Promise<T>} optionsOrCallback - Checkout options or callback to invoke with the connection.
|
|
461
|
-
* @param {function(import("../drivers/base.js").default) : Promise<T>} [callback] - Callback to invoke with the connection.
|
|
462
|
-
* @returns {Promise<T>} - Resolves with the callback result.
|
|
463
|
-
*/
|
|
464
|
-
async withConnection(optionsOrCallback, callback) {
|
|
465
|
-
const options = typeof optionsOrCallback == "function" ? {} : optionsOrCallback;
|
|
466
|
-
const actualCallback = typeof optionsOrCallback == "function" ? optionsOrCallback : callback;
|
|
467
|
-
if (!actualCallback)
|
|
468
|
-
throw new Error("withConnection requires a callback");
|
|
469
|
-
const connection = await this.checkout(options);
|
|
470
|
-
const id = connection.getIdSeq();
|
|
471
|
-
return await this.asyncLocalStorage.run(id, async () => {
|
|
472
|
-
try {
|
|
473
|
-
return await actualCallback(connection);
|
|
474
|
-
}
|
|
475
|
-
finally {
|
|
476
|
-
await this.checkin(connection);
|
|
477
|
-
}
|
|
478
|
-
});
|
|
479
|
-
}
|
|
480
|
-
/**
|
|
481
|
-
* Runs get current connection.
|
|
482
|
-
* @returns {import("../drivers/base.js").default} - The current connection.
|
|
483
|
-
*/
|
|
484
|
-
getCurrentConnection() {
|
|
485
|
-
const id = this.asyncLocalStorage.getStore();
|
|
486
|
-
if (id === undefined)
|
|
487
|
-
return this.currentFallbackConnectionOrFail();
|
|
488
|
-
this.ensureConnectionIsInUse(id);
|
|
489
|
-
const currentConnection = this.connectionsInUse[id];
|
|
490
|
-
if (!currentConnection) {
|
|
491
|
-
throw new Error(`Couldn't get current connection from that ID: ${id}`);
|
|
492
|
-
}
|
|
493
|
-
return currentConnection;
|
|
494
|
-
}
|
|
495
|
-
/**
|
|
496
|
-
* Runs current fallback connection or fail.
|
|
497
|
-
* @returns {import("../drivers/base.js").default} - Fallback connection, if present.
|
|
498
|
-
*/
|
|
499
|
-
currentFallbackConnectionOrFail() {
|
|
500
|
-
const fallbackConnection = this.getGlobalConnection();
|
|
501
|
-
if (fallbackConnection)
|
|
502
|
-
return fallbackConnection;
|
|
503
|
-
throw new Error("ID hasn't been set for this async context");
|
|
504
|
-
}
|
|
505
|
-
/**
|
|
506
|
-
* Runs ensure connection is in use.
|
|
507
|
-
* @param {number} id - Checked-out connection id.
|
|
508
|
-
* @returns {void}
|
|
509
|
-
*/
|
|
510
|
-
ensureConnectionIsInUse(id) {
|
|
511
|
-
if (!(id in this.connectionsInUse)) {
|
|
512
|
-
throw new Error(`Connection ${id} doesn't exist any more - has it been checked in again?`);
|
|
513
|
-
}
|
|
514
|
-
}
|
|
515
|
-
/**
|
|
516
|
-
* Registers a fallback connection for this pool identifier that will be used when no async context is available.
|
|
517
|
-
* @param {import("../drivers/base.js").default} connection - Connection.
|
|
518
|
-
* @returns {void} - No return value.
|
|
519
|
-
*/
|
|
520
|
-
setGlobalConnection(connection) {
|
|
521
|
-
const klass = /**
|
|
522
|
-
* Narrows the runtime value to the documented type.
|
|
523
|
-
@type {typeof VelociousDatabasePoolAsyncTrackedMultiConnection} */ (this.constructor);
|
|
524
|
-
let mapForConfiguration = klass.globalConnections.get(this.configuration);
|
|
525
|
-
if (!mapForConfiguration) {
|
|
526
|
-
mapForConfiguration = {};
|
|
527
|
-
klass.globalConnections.set(this.configuration, mapForConfiguration);
|
|
528
|
-
}
|
|
529
|
-
mapForConfiguration[this.identifier] = connection;
|
|
530
|
-
}
|
|
531
|
-
/**
|
|
532
|
-
* Ensures a global fallback connection exists for this pool identifier and returns it.
|
|
533
|
-
* If one is already set, it is returned and also made available in the pool queue.
|
|
534
|
-
* Otherwise a new connection is spawned, registered, and queued.
|
|
535
|
-
* @returns {Promise<import("../drivers/base.js").default>} - Resolves with the global connection.
|
|
536
|
-
*/
|
|
537
|
-
async ensureGlobalConnection() {
|
|
538
|
-
const existing = this.getGlobalConnection();
|
|
539
|
-
if (existing)
|
|
540
|
-
return existing;
|
|
541
|
-
const connection = await this.spawnConnection();
|
|
542
|
-
this.setGlobalConnection(connection);
|
|
543
|
-
return connection;
|
|
544
|
-
}
|
|
960
|
+
const keptConnections = []
|
|
545
961
|
/**
|
|
546
|
-
*
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
670
|
-
|
|
671
|
-
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
684
|
-
|
|
685
|
-
|
|
686
|
-
|
|
687
|
-
|
|
688
|
-
|
|
689
|
-
|
|
690
|
-
|
|
691
|
-
|
|
692
|
-
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
710
|
-
|
|
711
|
-
|
|
712
|
-
|
|
713
|
-
|
|
714
|
-
|
|
715
|
-
|
|
716
|
-
|
|
717
|
-
|
|
718
|
-
|
|
719
|
-
|
|
720
|
-
|
|
721
|
-
|
|
722
|
-
|
|
723
|
-
|
|
724
|
-
|
|
725
|
-
|
|
726
|
-
|
|
727
|
-
|
|
728
|
-
|
|
729
|
-
|
|
730
|
-
|
|
731
|
-
|
|
732
|
-
|
|
733
|
-
|
|
734
|
-
|
|
735
|
-
|
|
736
|
-
|
|
737
|
-
|
|
738
|
-
|
|
739
|
-
|
|
740
|
-
|
|
741
|
-
|
|
742
|
-
|
|
743
|
-
|
|
744
|
-
}
|
|
745
|
-
|
|
746
|
-
|
|
747
|
-
|
|
748
|
-
scheduleIdleConnectionReaper() {
|
|
749
|
-
if (this.idleConnectionReaperTimer)
|
|
750
|
-
return;
|
|
751
|
-
if (!this.hasIdleConnectionsToReap())
|
|
752
|
-
return;
|
|
753
|
-
const delay = this.nextIdleConnectionReapDelay(/**
|
|
754
|
-
* Narrows the runtime value to the documented type.
|
|
755
|
-
@type {number} */ (this.idleTimeoutMillis()));
|
|
756
|
-
this.idleConnectionReaperTimer = setTimeout(() => {
|
|
757
|
-
this.idleConnectionReaperTimer = undefined;
|
|
758
|
-
void this.reapIdleConnections().catch((error) => {
|
|
759
|
-
this.logger.warn(() => ["Failed to reap idle database connections:", error]);
|
|
760
|
-
});
|
|
761
|
-
}, delay);
|
|
762
|
-
if (typeof this.idleConnectionReaperTimer.unref === "function") {
|
|
763
|
-
this.idleConnectionReaperTimer.unref();
|
|
764
|
-
}
|
|
765
|
-
}
|
|
766
|
-
/**
|
|
767
|
-
* Runs has idle connections to reap.
|
|
768
|
-
* @returns {boolean} - Whether an idle reaper timer should be scheduled.
|
|
769
|
-
*/
|
|
770
|
-
hasIdleConnectionsToReap() {
|
|
771
|
-
return this.connections.length > 0 && this.idleTimeoutMillis() !== null;
|
|
772
|
-
}
|
|
773
|
-
/**
|
|
774
|
-
* Runs next idle connection reap delay.
|
|
775
|
-
* @param {number} idleTimeoutMillis - Idle timeout in milliseconds.
|
|
776
|
-
* @returns {number} - Delay before the next reap.
|
|
777
|
-
*/
|
|
778
|
-
nextIdleConnectionReapDelay(idleTimeoutMillis) {
|
|
779
|
-
let delay = idleTimeoutMillis;
|
|
780
|
-
const now = Date.now();
|
|
781
|
-
for (const connection of this.connections) {
|
|
782
|
-
if (this.connectionHasOpenTransaction(connection))
|
|
783
|
-
continue;
|
|
784
|
-
const trackedConnection = /**
|
|
785
|
-
* Narrows the runtime value to the documented type.
|
|
786
|
-
@type {import("../drivers/base.js").default & {[IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection);
|
|
787
|
-
const checkedInAt = trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT];
|
|
788
|
-
if (typeof checkedInAt !== "number")
|
|
789
|
-
continue;
|
|
790
|
-
delay = Math.min(delay, Math.max(0, idleTimeoutMillis - (now - checkedInAt)));
|
|
791
|
-
}
|
|
792
|
-
return delay;
|
|
793
|
-
}
|
|
794
|
-
/**
|
|
795
|
-
* Closes idle checked-in connections that have exceeded the configured timeout.
|
|
796
|
-
* @returns {Promise<void>} - Resolves when complete.
|
|
797
|
-
*/
|
|
798
|
-
async reapIdleConnections() {
|
|
799
|
-
if (this.connections.length === 0)
|
|
800
|
-
return;
|
|
801
|
-
const idleTimeoutMillis = this.idleTimeoutMillis();
|
|
802
|
-
if (idleTimeoutMillis === null)
|
|
803
|
-
return;
|
|
804
|
-
const { expiredConnections, keptConnections } = this.classifyIdleConnectionsForReaping({ idleTimeoutMillis, now: Date.now() });
|
|
805
|
-
this.connections = keptConnections;
|
|
806
|
-
await this.closeExpiredIdleConnections(expiredConnections);
|
|
807
|
-
await this.awaitInflightConnectionCloses();
|
|
808
|
-
if (this.connections.length > 0)
|
|
809
|
-
this.scheduleIdleConnectionReaper();
|
|
810
|
-
}
|
|
811
|
-
/**
|
|
812
|
-
* Runs close expired idle connections.
|
|
813
|
-
* @param {import("../drivers/base.js").default[]} expiredConnections - Connections to close.
|
|
814
|
-
* @returns {Promise<void>} - Resolves when closed.
|
|
815
|
-
*/
|
|
816
|
-
async closeExpiredIdleConnections(expiredConnections) {
|
|
817
|
-
for (const connection of expiredConnections) {
|
|
818
|
-
await this.closeConnection(connection);
|
|
819
|
-
}
|
|
820
|
-
}
|
|
821
|
-
/**
|
|
822
|
-
* Runs await inflight connection closes.
|
|
823
|
-
* @returns {Promise<void>} - Resolves once in-flight connection closes settle.
|
|
824
|
-
*/
|
|
825
|
-
async awaitInflightConnectionCloses() {
|
|
826
|
-
if (this.inflightConnectionCloses.size > 0) {
|
|
827
|
-
await Promise.allSettled([...this.inflightConnectionCloses]);
|
|
828
|
-
}
|
|
829
|
-
}
|
|
830
|
-
/**
|
|
831
|
-
* Runs classify idle connections for reaping.
|
|
832
|
-
* @param {{idleTimeoutMillis: number, now: number}} args - Reaper classification inputs.
|
|
833
|
-
* @returns {{expiredConnections: import("../drivers/base.js").default[], keptConnections: import("../drivers/base.js").default[]}} - Classified idle connections.
|
|
834
|
-
*/
|
|
835
|
-
classifyIdleConnectionsForReaping({ idleTimeoutMillis, now }) {
|
|
836
|
-
/**
|
|
837
|
-
* Kept connections.
|
|
838
|
-
@type {import("../drivers/base.js").default[]} */
|
|
839
|
-
const keptConnections = [];
|
|
840
|
-
/**
|
|
841
|
-
* Expired connections.
|
|
842
|
-
@type {import("../drivers/base.js").default[]} */
|
|
843
|
-
const expiredConnections = [];
|
|
844
|
-
for (const connection of this.connections) {
|
|
845
|
-
this.classifyIdleConnectionForReaping({ connection, expiredConnections, idleTimeoutMillis, keptConnections, now });
|
|
846
|
-
}
|
|
847
|
-
return { expiredConnections, keptConnections };
|
|
848
|
-
}
|
|
849
|
-
/**
|
|
850
|
-
* Runs classify idle connection for reaping.
|
|
851
|
-
* @param {{connection: import("../drivers/base.js").default, expiredConnections: import("../drivers/base.js").default[], idleTimeoutMillis: number, keptConnections: import("../drivers/base.js").default[], now: number}} args - Classification state.
|
|
852
|
-
* @returns {void}
|
|
853
|
-
*/
|
|
854
|
-
classifyIdleConnectionForReaping({ connection, expiredConnections, idleTimeoutMillis, keptConnections, now }) {
|
|
855
|
-
if (this.connectionIsClosed(connection))
|
|
856
|
-
return;
|
|
857
|
-
if (this.connectionHasOpenTransaction(connection)) {
|
|
858
|
-
keptConnections.push(connection);
|
|
859
|
-
return;
|
|
860
|
-
}
|
|
861
|
-
const target = this.idleConnectionExpired({ connection, idleTimeoutMillis, now }) ? expiredConnections : keptConnections;
|
|
862
|
-
target.push(connection);
|
|
863
|
-
}
|
|
864
|
-
/**
|
|
865
|
-
* Runs connection is closed.
|
|
866
|
-
* @param {import("../drivers/base.js").default} connection - Connection to inspect.
|
|
867
|
-
* @returns {boolean} - Whether the connection is marked closed.
|
|
868
|
-
*/
|
|
869
|
-
connectionIsClosed(connection) {
|
|
870
|
-
const trackedConnection = /**
|
|
871
|
-
* Narrows the runtime value to the documented type.
|
|
872
|
-
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean}} */ (connection);
|
|
873
|
-
return Boolean(trackedConnection[CLOSED_CONNECTION]);
|
|
874
|
-
}
|
|
875
|
-
/**
|
|
876
|
-
* Runs idle connection expired.
|
|
877
|
-
* @param {{connection: import("../drivers/base.js").default, idleTimeoutMillis: number, now: number}} args - Expiry inputs.
|
|
878
|
-
* @returns {boolean} - Whether the idle connection expired.
|
|
879
|
-
*/
|
|
880
|
-
idleConnectionExpired({ connection, idleTimeoutMillis, now }) {
|
|
881
|
-
const trackedConnection = /**
|
|
882
|
-
* Narrows the runtime value to the documented type.
|
|
883
|
-
@type {import("../drivers/base.js").default & {[IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection);
|
|
884
|
-
const checkedInAt = trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT];
|
|
885
|
-
return typeof checkedInAt === "number" && now - checkedInAt >= idleTimeoutMillis;
|
|
886
|
-
}
|
|
887
|
-
/**
|
|
888
|
-
* Runs connection has open transaction.
|
|
889
|
-
* @param {import("../drivers/base.js").default} connection - Connection to inspect.
|
|
890
|
-
* @returns {boolean} - Whether the connection has an open transaction.
|
|
891
|
-
*/
|
|
892
|
-
connectionHasOpenTransaction(connection) {
|
|
893
|
-
return connection._transactionsCount > 0;
|
|
894
|
-
}
|
|
895
|
-
/**
|
|
896
|
-
* Rolls back any transaction a previous holder left open before a connection
|
|
897
|
-
* re-enters the idle pool. A connection returned to the pool with an open
|
|
898
|
-
* transaction would otherwise be handed to an unrelated checkout, whose
|
|
899
|
-
* startTransaction() then fails with "A transaction is already running" and
|
|
900
|
-
* poisons every following caller that reuses it.
|
|
901
|
-
* @param {import("../drivers/base.js").default} connection - Connection being checked in.
|
|
902
|
-
* @returns {Promise<void>} - Resolves when the connection holds no open transaction.
|
|
903
|
-
*/
|
|
904
|
-
async rollbackLeftOpenTransaction(connection) {
|
|
905
|
-
if (!this.connectionHasOpenTransaction(connection))
|
|
906
|
-
return;
|
|
907
|
-
this.logger.warn(() => [`Rolling back a transaction left open on a connection being checked in (identifier=${this.identifier}).`]);
|
|
908
|
-
while (this.connectionHasOpenTransaction(connection)) {
|
|
909
|
-
await connection.rollbackTransaction();
|
|
910
|
-
}
|
|
911
|
-
}
|
|
912
|
-
/**
|
|
913
|
-
* Runs close connection.
|
|
914
|
-
* @param {import("../drivers/base.js").default} connection - Connection to close.
|
|
915
|
-
* @returns {Promise<void>} - Resolves when complete.
|
|
916
|
-
*/
|
|
917
|
-
async closeConnection(connection) {
|
|
918
|
-
// Idempotent: a fire-and-forget scheduled reap and an explicit reap can both
|
|
919
|
-
// target the same connection. Await the in-flight close instead of closing
|
|
920
|
-
// twice (which can throw on the driver) or returning while the underlying
|
|
921
|
-
// handle is still open.
|
|
922
|
-
const existingClose = this.connectionClosePromises.get(connection);
|
|
923
|
-
if (existingClose) {
|
|
924
|
-
return await existingClose;
|
|
925
|
-
}
|
|
926
|
-
const trackedConnection = /**
|
|
927
|
-
* Narrows the runtime value to the documented type.
|
|
928
|
-
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean, [CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection);
|
|
929
|
-
trackedConnection[CLOSED_CONNECTION] = true;
|
|
930
|
-
delete trackedConnection[CONNECTION_CHECKED_OUT_AT];
|
|
931
|
-
delete trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT];
|
|
932
|
-
const closePromise = (async () => {
|
|
933
|
-
await trackedConnection.close();
|
|
934
|
-
})();
|
|
935
|
-
this.connectionClosePromises.set(connection, closePromise);
|
|
936
|
-
this.inflightConnectionCloses.add(closePromise);
|
|
937
|
-
try {
|
|
938
|
-
await closePromise;
|
|
939
|
-
}
|
|
940
|
-
finally {
|
|
941
|
-
this.inflightConnectionCloses.delete(closePromise);
|
|
942
|
-
}
|
|
943
|
-
}
|
|
944
|
-
/**
|
|
945
|
-
* Runs clear idle connection reaper timer.
|
|
946
|
-
@returns {void} */
|
|
947
|
-
clearIdleConnectionReaperTimer() {
|
|
948
|
-
if (!this.idleConnectionReaperTimer)
|
|
949
|
-
return;
|
|
950
|
-
clearTimeout(this.idleConnectionReaperTimer);
|
|
951
|
-
this.idleConnectionReaperTimer = undefined;
|
|
952
|
-
}
|
|
953
|
-
/**
|
|
954
|
-
* Closes all active and cached connections for this pool.
|
|
955
|
-
* @returns {Promise<void>} - Resolves when complete.
|
|
956
|
-
*/
|
|
957
|
-
async closeAll() {
|
|
958
|
-
this.clearIdleConnectionReaperTimer();
|
|
959
|
-
this.rejectPendingCheckouts(new Error("Database pool was closed before checkout completed."));
|
|
960
|
-
const connections = new Set([
|
|
961
|
-
...this.connections,
|
|
962
|
-
...Object.values(this.connectionsInUse),
|
|
963
|
-
this.getGlobalConnectionForIdentifier(),
|
|
964
|
-
this._testSharedConnection
|
|
965
|
-
].filter(Boolean));
|
|
966
|
-
this.connections = [];
|
|
967
|
-
this.connectionsInUse = {};
|
|
968
|
-
this._testSharedConnection = undefined;
|
|
969
|
-
this.clearGlobalConnectionForIdentifier();
|
|
970
|
-
for (const connection of connections) {
|
|
971
|
-
if (!connection)
|
|
972
|
-
continue;
|
|
973
|
-
await this.closeConnection(connection);
|
|
974
|
-
}
|
|
975
|
-
}
|
|
976
|
-
/**
|
|
977
|
-
* Runs reject pending checkouts.
|
|
978
|
-
* @param {Error} error - Error to reject pending checkouts with.
|
|
979
|
-
* @returns {void}
|
|
980
|
-
*/
|
|
981
|
-
rejectPendingCheckouts(error) {
|
|
982
|
-
const pendingCheckouts = this.pendingCheckouts;
|
|
983
|
-
this.pendingCheckouts = [];
|
|
984
|
-
for (const checkout of pendingCheckouts) {
|
|
985
|
-
checkout.reject(error);
|
|
986
|
-
}
|
|
987
|
-
}
|
|
988
|
-
/**
|
|
989
|
-
* Replaces all globally registered fallback connections.
|
|
990
|
-
* @param {Record<string, import("../drivers/base.js").default>} [connections] - Connections.
|
|
991
|
-
* @param {import("../../configuration.js").default} [configuration] - Configuration instance.
|
|
992
|
-
* @returns {void} - No return value.
|
|
993
|
-
*/
|
|
994
|
-
static setGlobalConnections(connections, configuration) {
|
|
995
|
-
if (!configuration) {
|
|
996
|
-
this.globalConnections = new WeakMap();
|
|
997
|
-
return;
|
|
998
|
-
}
|
|
999
|
-
this.globalConnections.set(configuration, connections || {});
|
|
1000
|
-
}
|
|
1001
|
-
/**
|
|
1002
|
-
* Clears globally registered fallback connections for all configurations or a single configuration.
|
|
1003
|
-
* @param {import("../../configuration.js").default} [configuration] - Configuration instance.
|
|
1004
|
-
* @returns {void} - No return value.
|
|
1005
|
-
*/
|
|
1006
|
-
static clearGlobalConnections(configuration) {
|
|
1007
|
-
if (!configuration) {
|
|
1008
|
-
this.globalConnections = new WeakMap();
|
|
1009
|
-
return;
|
|
1010
|
-
}
|
|
1011
|
-
this.globalConnections.delete(configuration);
|
|
1012
|
-
}
|
|
962
|
+
* Expired connections.
|
|
963
|
+
@type {import("../drivers/base.js").default[]} */
|
|
964
|
+
const expiredConnections = []
|
|
965
|
+
|
|
966
|
+
for (const connection of this.connections) {
|
|
967
|
+
this.classifyIdleConnectionForReaping({connection, expiredConnections, idleTimeoutMillis, keptConnections, now})
|
|
968
|
+
}
|
|
969
|
+
|
|
970
|
+
return {expiredConnections, keptConnections}
|
|
971
|
+
}
|
|
972
|
+
|
|
973
|
+
/**
|
|
974
|
+
* Runs classify idle connection for reaping.
|
|
975
|
+
* @param {{connection: import("../drivers/base.js").default, expiredConnections: import("../drivers/base.js").default[], idleTimeoutMillis: number, keptConnections: import("../drivers/base.js").default[], now: number}} args - Classification state.
|
|
976
|
+
* @returns {void}
|
|
977
|
+
*/
|
|
978
|
+
classifyIdleConnectionForReaping({connection, expiredConnections, idleTimeoutMillis, keptConnections, now}) {
|
|
979
|
+
if (this.connectionIsClosed(connection)) return
|
|
980
|
+
if (this.connectionHasOpenTransaction(connection)) {
|
|
981
|
+
keptConnections.push(connection)
|
|
982
|
+
return
|
|
983
|
+
}
|
|
984
|
+
|
|
985
|
+
const target = this.idleConnectionExpired({connection, idleTimeoutMillis, now}) ? expiredConnections : keptConnections
|
|
986
|
+
|
|
987
|
+
target.push(connection)
|
|
988
|
+
}
|
|
989
|
+
|
|
990
|
+
/**
|
|
991
|
+
* Runs connection is closed.
|
|
992
|
+
* @param {import("../drivers/base.js").default} connection - Connection to inspect.
|
|
993
|
+
* @returns {boolean} - Whether the connection is marked closed.
|
|
994
|
+
*/
|
|
995
|
+
connectionIsClosed(connection) {
|
|
996
|
+
const trackedConnection = /**
|
|
997
|
+
* Narrows the runtime value to the documented type.
|
|
998
|
+
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean}} */ (connection)
|
|
999
|
+
|
|
1000
|
+
return Boolean(trackedConnection[CLOSED_CONNECTION])
|
|
1001
|
+
}
|
|
1002
|
+
|
|
1003
|
+
/**
|
|
1004
|
+
* Runs idle connection expired.
|
|
1005
|
+
* @param {{connection: import("../drivers/base.js").default, idleTimeoutMillis: number, now: number}} args - Expiry inputs.
|
|
1006
|
+
* @returns {boolean} - Whether the idle connection expired.
|
|
1007
|
+
*/
|
|
1008
|
+
idleConnectionExpired({connection, idleTimeoutMillis, now}) {
|
|
1009
|
+
const trackedConnection = /**
|
|
1010
|
+
* Narrows the runtime value to the documented type.
|
|
1011
|
+
@type {import("../drivers/base.js").default & {[IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
1012
|
+
const checkedInAt = trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT]
|
|
1013
|
+
|
|
1014
|
+
return typeof checkedInAt === "number" && now - checkedInAt >= idleTimeoutMillis
|
|
1015
|
+
}
|
|
1016
|
+
|
|
1017
|
+
/**
|
|
1018
|
+
* Runs connection has open transaction.
|
|
1019
|
+
* @param {import("../drivers/base.js").default} connection - Connection to inspect.
|
|
1020
|
+
* @returns {boolean} - Whether the connection has an open transaction.
|
|
1021
|
+
*/
|
|
1022
|
+
connectionHasOpenTransaction(connection) {
|
|
1023
|
+
return connection._transactionsCount > 0
|
|
1024
|
+
}
|
|
1025
|
+
|
|
1026
|
+
/**
|
|
1027
|
+
* Rolls back any transaction a previous holder left open before a connection
|
|
1028
|
+
* re-enters the idle pool. A connection returned to the pool with an open
|
|
1029
|
+
* transaction would otherwise be handed to an unrelated checkout, whose
|
|
1030
|
+
* startTransaction() then fails with "A transaction is already running" and
|
|
1031
|
+
* poisons every following caller that reuses it.
|
|
1032
|
+
* @param {import("../drivers/base.js").default} connection - Connection being checked in.
|
|
1033
|
+
* @returns {Promise<void>} - Resolves when the connection holds no open transaction.
|
|
1034
|
+
*/
|
|
1035
|
+
async rollbackLeftOpenTransaction(connection) {
|
|
1036
|
+
if (!this.connectionHasOpenTransaction(connection)) return
|
|
1037
|
+
|
|
1038
|
+
this.logger.warn(() => [`Rolling back a transaction left open on a connection being checked in (identifier=${this.identifier}).`])
|
|
1039
|
+
|
|
1040
|
+
while (this.connectionHasOpenTransaction(connection)) {
|
|
1041
|
+
await connection.rollbackTransaction()
|
|
1042
|
+
}
|
|
1043
|
+
}
|
|
1044
|
+
|
|
1045
|
+
/**
|
|
1046
|
+
* Runs close connection.
|
|
1047
|
+
* @param {import("../drivers/base.js").default} connection - Connection to close.
|
|
1048
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
1049
|
+
*/
|
|
1050
|
+
async closeConnection(connection) {
|
|
1051
|
+
// Idempotent: a fire-and-forget scheduled reap and an explicit reap can both
|
|
1052
|
+
// target the same connection. Await the in-flight close instead of closing
|
|
1053
|
+
// twice (which can throw on the driver) or returning while the underlying
|
|
1054
|
+
// handle is still open.
|
|
1055
|
+
const existingClose = this.connectionClosePromises.get(connection)
|
|
1056
|
+
|
|
1057
|
+
if (existingClose) {
|
|
1058
|
+
return await existingClose
|
|
1059
|
+
}
|
|
1060
|
+
|
|
1061
|
+
const trackedConnection = /**
|
|
1062
|
+
* Narrows the runtime value to the documented type.
|
|
1063
|
+
@type {import("../drivers/base.js").default & {[CLOSED_CONNECTION]?: boolean, [CONNECTION_CHECKED_OUT_AT]?: number, [IDLE_CONNECTION_CHECKED_IN_AT]?: number}} */ (connection)
|
|
1064
|
+
|
|
1065
|
+
trackedConnection[CLOSED_CONNECTION] = true
|
|
1066
|
+
delete trackedConnection[CONNECTION_CHECKED_OUT_AT]
|
|
1067
|
+
delete trackedConnection[IDLE_CONNECTION_CHECKED_IN_AT]
|
|
1068
|
+
|
|
1069
|
+
const closePromise = (async () => {
|
|
1070
|
+
await trackedConnection.close()
|
|
1071
|
+
})()
|
|
1072
|
+
|
|
1073
|
+
this.connectionClosePromises.set(connection, closePromise)
|
|
1074
|
+
this.inflightConnectionCloses.add(closePromise)
|
|
1075
|
+
|
|
1076
|
+
try {
|
|
1077
|
+
await closePromise
|
|
1078
|
+
} finally {
|
|
1079
|
+
this.inflightConnectionCloses.delete(closePromise)
|
|
1080
|
+
}
|
|
1081
|
+
}
|
|
1082
|
+
|
|
1083
|
+
/**
|
|
1084
|
+
* Runs clear idle connection reaper timer.
|
|
1085
|
+
@returns {void} */
|
|
1086
|
+
clearIdleConnectionReaperTimer() {
|
|
1087
|
+
if (!this.idleConnectionReaperTimer) return
|
|
1088
|
+
|
|
1089
|
+
clearTimeout(this.idleConnectionReaperTimer)
|
|
1090
|
+
this.idleConnectionReaperTimer = undefined
|
|
1091
|
+
}
|
|
1092
|
+
|
|
1093
|
+
/**
|
|
1094
|
+
* Closes all active and cached connections for this pool.
|
|
1095
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
1096
|
+
*/
|
|
1097
|
+
async closeAll() {
|
|
1098
|
+
this.clearIdleConnectionReaperTimer()
|
|
1099
|
+
this.rejectPendingCheckouts(new Error("Database pool was closed before checkout completed."))
|
|
1100
|
+
|
|
1101
|
+
const connections = new Set([
|
|
1102
|
+
...this.connections,
|
|
1103
|
+
...Object.values(this.connectionsInUse),
|
|
1104
|
+
this.getGlobalConnectionForIdentifier(),
|
|
1105
|
+
this._testSharedConnection
|
|
1106
|
+
].filter(Boolean))
|
|
1107
|
+
|
|
1108
|
+
this.connections = []
|
|
1109
|
+
this.connectionsInUse = {}
|
|
1110
|
+
this._testSharedConnection = undefined
|
|
1111
|
+
this.clearGlobalConnectionForIdentifier()
|
|
1112
|
+
|
|
1113
|
+
for (const connection of connections) {
|
|
1114
|
+
if (!connection) continue
|
|
1115
|
+
|
|
1116
|
+
await this.closeConnection(connection)
|
|
1117
|
+
}
|
|
1118
|
+
|
|
1119
|
+
}
|
|
1120
|
+
|
|
1121
|
+
/**
|
|
1122
|
+
* Runs reject pending checkouts.
|
|
1123
|
+
* @param {Error} error - Error to reject pending checkouts with.
|
|
1124
|
+
* @returns {void}
|
|
1125
|
+
*/
|
|
1126
|
+
rejectPendingCheckouts(error) {
|
|
1127
|
+
const pendingCheckouts = this.pendingCheckouts
|
|
1128
|
+
|
|
1129
|
+
this.pendingCheckouts = []
|
|
1130
|
+
|
|
1131
|
+
for (const checkout of pendingCheckouts) {
|
|
1132
|
+
checkout.reject(error)
|
|
1133
|
+
}
|
|
1134
|
+
}
|
|
1135
|
+
|
|
1136
|
+
/**
|
|
1137
|
+
* Replaces all globally registered fallback connections.
|
|
1138
|
+
* @param {Record<string, import("../drivers/base.js").default>} [connections] - Connections.
|
|
1139
|
+
* @param {import("../../configuration.js").default} [configuration] - Configuration instance.
|
|
1140
|
+
* @returns {void} - No return value.
|
|
1141
|
+
*/
|
|
1142
|
+
static setGlobalConnections(connections, configuration) {
|
|
1143
|
+
if (!configuration) {
|
|
1144
|
+
this.globalConnections = new WeakMap()
|
|
1145
|
+
return
|
|
1146
|
+
}
|
|
1147
|
+
|
|
1148
|
+
this.globalConnections.set(configuration, connections || {})
|
|
1149
|
+
}
|
|
1150
|
+
|
|
1151
|
+
/**
|
|
1152
|
+
* Clears globally registered fallback connections for all configurations or a single configuration.
|
|
1153
|
+
* @param {import("../../configuration.js").default} [configuration] - Configuration instance.
|
|
1154
|
+
* @returns {void} - No return value.
|
|
1155
|
+
*/
|
|
1156
|
+
static clearGlobalConnections(configuration) {
|
|
1157
|
+
if (!configuration) {
|
|
1158
|
+
this.globalConnections = new WeakMap()
|
|
1159
|
+
return
|
|
1160
|
+
}
|
|
1161
|
+
|
|
1162
|
+
this.globalConnections.delete(configuration)
|
|
1163
|
+
}
|
|
1013
1164
|
}
|
|
1014
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXN5bmMtdHJhY2tlZC1tdWx0aS1jb25uZWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2RhdGFiYXNlL3Bvb2wvYXN5bmMtdHJhY2tlZC1tdWx0aS1jb25uZWN0aW9uLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVk7QUFFWixPQUFPLEVBQUMsaUJBQWlCLEVBQUMsTUFBTSxhQUFhLENBQUE7QUFDN0MsT0FBTyxRQUFRLEVBQUUsRUFBQyxzQkFBc0IsRUFBQyxNQUFNLFdBQVcsQ0FBQTtBQUUxRCxNQUFNLENBQUMsTUFBTSxpQkFBaUIsR0FBRyxNQUFNLENBQUMsMkJBQTJCLENBQUMsQ0FBQTtBQUNwRSxNQUFNLDZCQUE2QixHQUFHLE1BQU0sQ0FBQyxvQ0FBb0MsQ0FBQyxDQUFBO0FBQ2xGLE1BQU0seUJBQXlCLEdBQUcsTUFBTSxDQUFDLGlDQUFpQyxDQUFDLENBQUE7QUFDM0UsTUFBTSwyQkFBMkIsR0FBRyxJQUFJLENBQUE7QUFFeEM7Ozs7Ozs7OztHQVNHO0FBRUgsTUFBTSxDQUFDLE9BQU8sT0FBTyxnREFBaUQsU0FBUSxRQUFRO0lBQ3BGOzs7T0FHRztJQUNILE1BQU0sQ0FBQyxpQkFBaUIsR0FBRyxJQUFJLE9BQU8sRUFBRSxDQUFBO0lBRXhDLGlCQUFpQixHQUFHLElBQUksaUJBQWlCLEVBQUUsQ0FBQTtJQUUzQzs7Ozs7T0FLRztJQUNILHFCQUFxQixHQUFHLFNBQVMsQ0FBQTtJQUVqQzs7dURBRW1EO0lBQ25ELFdBQVcsR0FBRyxFQUFFLENBQUE7SUFFaEI7O3FFQUVpRTtJQUNqRSxnQkFBZ0IsR0FBRyxFQUFFLENBQUE7SUFFckI7O2tDQUU4QjtJQUM5QixnQkFBZ0IsR0FBRyxFQUFFLENBQUE7SUFFckI7O3VCQUVtQjtJQUNuQix1QkFBdUIsR0FBRyxDQUFDLENBQUE7SUFFM0I7OzBDQUVzQztJQUN0QywyQkFBMkIsR0FBRyxTQUFTLENBQUE7SUFFdkM7OzBEQUVzRDtJQUN0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LGlCQUFpQixHQUFHLElBQUksT0FBTyxFQUFFLENBQUE7WUFDdEMsT0FBTTtRQUNSLENBQUM7UUFFRCxJQUFJLENBQUMsaUJBQWlCLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBQzlDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IHtBc3luY0xvY2FsU3RvcmFnZX0gZnJvbSBcImFzeW5jX2hvb2tzXCJcbmltcG9ydCBCYXNlUG9vbCwge1BPT0xfQ09ORklHVVJBVElPTl9LRVl9IGZyb20gXCIuL2Jhc2UuanNcIlxuXG5leHBvcnQgY29uc3QgQ0xPU0VEX0NPTk5FQ1RJT04gPSBTeW1ib2woXCJ2ZWxvY2lvdXNDbG9zZWRDb25uZWN0aW9uXCIpXG5jb25zdCBJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVCA9IFN5bWJvbChcInZlbG9jaW91c0lkbGVDb25uZWN0aW9uQ2hlY2tlZEluQXRcIilcbmNvbnN0IENPTk5FQ1RJT05fQ0hFQ0tFRF9PVVRfQVQgPSBTeW1ib2woXCJ2ZWxvY2lvdXNDb25uZWN0aW9uQ2hlY2tlZE91dEF0XCIpXG5jb25zdCBERUZBVUxUX0lETEVfVElNRU9VVF9NSUxMSVMgPSA1MDAwXG5cbi8qKlxuICogUGVuZGluZ0NoZWNrb3V0IHR5cGUuXG4gKiBAdHlwZWRlZiB7b2JqZWN0fSBQZW5kaW5nQ2hlY2tvdXRcbiAqIEBwcm9wZXJ0eSB7aW1wb3J0KFwiLi4vLi4vY29uZmlndXJhdGlvbi10eXBlcy5qc1wiKS5EYXRhYmFzZUNvbmZpZ3VyYXRpb25UeXBlfSBkYXRhYmFzZUNvbmZpZyAtIFJlc29sdmVkIGRhdGFiYXNlIGNvbmZpZ3VyYXRpb24gbmVlZGVkIGJ5IHRoZSBjaGVja291dC5cbiAqIEBwcm9wZXJ0eSB7bnVtYmVyfSBlbnF1ZXVlZEF0IC0gVGltZXN0YW1wIHdoZW4gdGhlIGNoZWNrb3V0IHN0YXJ0ZWQgd2FpdGluZy5cbiAqIEBwcm9wZXJ0eSB7aW1wb3J0KFwiLi9iYXNlLmpzXCIpLkNvbm5lY3Rpb25DaGVja291dE9wdGlvbnN9IG9wdGlvbnMgLSBDaGVja291dCBvcHRpb25zLlxuICogQHByb3BlcnR5IHtzdHJpbmd9IHJldXNlS2V5IC0gRGF0YWJhc2UgY29uZmlndXJhdGlvbiByZXVzZSBrZXkgbmVlZGVkIGJ5IHRoZSBjaGVja291dC5cbiAqIEBwcm9wZXJ0eSB7KGNvbm5lY3Rpb246IGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0KSA9PiB2b2lkfSByZXNvbHZlIC0gUmVzb2x2ZXMgd2l0aCBhbiBhY3RpdmF0ZWQgY29ubmVjdGlvbi5cbiAqIEBwcm9wZXJ0eSB7KGVycm9yOiBFcnJvcikgPT4gdm9pZH0gcmVqZWN0IC0gUmVqZWN0cyB3aGVuIGNoZWNrb3V0IGNhbm5vdCBjb21wbGV0ZS5cbiAqL1xuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBWZWxvY2lvdXNEYXRhYmFzZVBvb2xBc3luY1RyYWNrZWRNdWx0aUNvbm5lY3Rpb24gZXh0ZW5kcyBCYXNlUG9vbCB7XG4gIC8qKlxuICAgKiBHbG9iYWwgZmFsbGJhY2sgY29ubmVjdGlvbnMga2V5ZWQgYnkgY29uZmlndXJhdGlvbiBpbnN0YW5jZSBhbmQgcG9vbCBpZGVudGlmaWVyLlxuICAgKiBAdHlwZSB7V2Vha01hcDxpbXBvcnQoXCIuLi8uLi9jb25maWd1cmF0aW9uLmpzXCIpLmRlZmF1bHQsIFJlY29yZDxzdHJpbmcsIGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pj59XG4gICAqL1xuICBzdGF0aWMgZ2xvYmFsQ29ubmVjdGlvbnMgPSBuZXcgV2Vha01hcCgpXG5cbiAgYXN5bmNMb2NhbFN0b3JhZ2UgPSBuZXcgQXN5bmNMb2NhbFN0b3JhZ2UoKVxuXG4gIC8qKlxuICAgKiBXaGVuIHNldCwgcmV0dXJuZWQgYnkgZ2V0Q3VycmVudENvbnRleHRDb25uZWN0aW9uIHdoZW4gbm8gYXN5bmMgY29udGV4dCBleGlzdHMuXG4gICAqIFVzZWQgYnkgdGhlIHRlc3QgcnVubmVyIHRvIHNoYXJlIGEgY29ubmVjdGlvbiBiZXR3ZWVuIHRlc3QgY29kZSBhbmQgSFRUUCBoYW5kbGVyc1xuICAgKiBydW5uaW5nIGluIHRoZSBzYW1lIHByb2Nlc3MgKGluLXByb2Nlc3MgdGVzdCBzZXJ2ZXIgbW9kZSkuXG4gICAqIEB0eXBlIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCB8IHVuZGVmaW5lZH1cbiAgICovXG4gIF90ZXN0U2hhcmVkQ29ubmVjdGlvbiA9IHVuZGVmaW5lZFxuXG4gIC8qKlxuICAgKiBDb25uZWN0aW9ucy5cbiAgICBAdHlwZSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHRbXX0gKi9cbiAgY29ubmVjdGlvbnMgPSBbXVxuXG4gIC8qKlxuICAgKiBDb25uZWN0aW9ucyBpbiB1c2UuXG4gICAgQHR5cGUge1JlY29yZDxudW1iZXIsIGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn0gKi9cbiAgY29ubmVjdGlvbnNJblVzZSA9IHt9XG5cbiAgLyoqXG4gICAqIFBlbmRpbmcgY2hlY2tvdXRzLlxuICAgIEB0eXBlIHtQZW5kaW5nQ2hlY2tvdXRbXX0gKi9cbiAgcGVuZGluZ0NoZWNrb3V0cyA9IFtdXG5cbiAgLyoqXG4gICAqIENvbm5lY3Rpb25zIGJlaW5nIHNwYXduZWQuXG4gICAgQHR5cGUge251bWJlcn0gKi9cbiAgY29ubmVjdGlvbnNCZWluZ1NwYXduZWQgPSAwXG5cbiAgLyoqXG4gICAqIFBlbmRpbmcgY2hlY2tvdXQgZHJhaW4gcHJvbWlzZS5cbiAgICBAdHlwZSB7UHJvbWlzZTx2b2lkPiB8IHVuZGVmaW5lZH0gKi9cbiAgcGVuZGluZ0NoZWNrb3V0RHJhaW5Qcm9taXNlID0gdW5kZWZpbmVkXG5cbiAgLyoqXG4gICAqIElkbGUgY29ubmVjdGlvbiByZWFwZXIgdGltZXIuXG4gICAgQHR5cGUge1JldHVyblR5cGU8dHlwZW9mIHNldFRpbWVvdXQ+IHwgdW5kZWZpbmVkfSAqL1xuICBpZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyID0gdW5kZWZpbmVkXG5cbiAgLyoqXG4gICAqIEluLWZsaWdodCBjb25uZWN0aW9uLWNsb3NlIHByb21pc2VzLiBUaGUgaWRsZSByZWFwZXIgaXMgYXJtZWQgb24gY2hlY2staW5cbiAgICogYW5kIHJ1bnMgZmlyZS1hbmQtZm9yZ2V0IHdoZW4gaXRzIHRpbWVyIGZpcmVzLCBzbyBhIHNjaGVkdWxlZCByZWFwIGNhbiBiZVxuICAgKiBjbG9zaW5nIGEgY29ubmVjdGlvbiB3aGlsZSBhbiBleHBsaWNpdCBgcmVhcElkbGVDb25uZWN0aW9ucygpYCAob3JcbiAgICogYGNsZWFySWRsZUNvbm5lY3Rpb25SZWFwZXJUaW1lcigpYCkgcnVucy4gVHJhY2tpbmcgdGhlIGluLWZsaWdodCBjbG9zZXMgbGV0c1xuICAgKiB0aG9zZSBjYWxsZXJzIGF3YWl0IHRoZW0sIHNvIG9uY2UgYSByZWFwIHJlc29sdmVzIHRoZSBjb25uZWN0aW9ucyBpdFxuICAgKiBleHBpcmVkIGFyZSBmdWxseSBjbG9zZWQgaW5zdGVhZCBvZiBoYWxmLWNsb3NlZCBtaWQtYGNsb3NlKClgLlxuICAgKiBAdHlwZSB7U2V0PFByb21pc2U8dm9pZD4+fVxuICAgKi9cbiAgaW5mbGlnaHRDb25uZWN0aW9uQ2xvc2VzID0gbmV3IFNldCgpXG5cbiAgLyoqXG4gICAqIEluLWZsaWdodCBjbG9zZSBwcm9taXNlIHBlciBjb25uZWN0aW9uLCBzbyBjb25jdXJyZW50IGNsb3NlcyBvZiB0aGUgc2FtZVxuICAgKiBjb25uZWN0aW9uIGF3YWl0IHRoZSBzYW1lIGNsb3NlIHJhdGhlciB0aGFuIGNsb3NpbmcgdGhlIGRyaXZlciBoYW5kbGUgdHdpY2UuXG4gICAqIEB0eXBlIHtXZWFrTWFwPG9iamVjdCwgUHJvbWlzZTx2b2lkPj59XG4gICAqL1xuICBjb25uZWN0aW9uQ2xvc2VQcm9taXNlcyA9IG5ldyBXZWFrTWFwKClcblxuICBpZFNlcSA9IDBcblxuICAvKipcbiAgICogUnVucyBjb25zdHJ1Y3Rvci5cbiAgICogQHBhcmFtIHtvYmplY3R9IGFyZ3MgLSBPcHRpb25zIG9iamVjdC5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi8uLi9jb25maWd1cmF0aW9uLmpzXCIpLmRlZmF1bHR9IGFyZ3MuY29uZmlndXJhdGlvbiAtIENvbmZpZ3VyYXRpb24gaW5zdGFuY2UuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLmlkZW50aWZpZXIgLSBJZGVudGlmaWVyLlxuICAgKi9cbiAgY29uc3RydWN0b3Ioe2NvbmZpZ3VyYXRpb24sIGlkZW50aWZpZXJ9KSB7XG4gICAgc3VwZXIoe2NvbmZpZ3VyYXRpb24sIGlkZW50aWZpZXJ9KVxuICAgIHRoaXMuX3dpdGhvdXRDdXJyZW50Q29ubmVjdGlvbkNvbnRleHQgPSAvKipcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICogTmFycm93cyB0aGUgcnVudGltZSB2YWx1ZSB0byB0aGUgZG9jdW1lbnRlZCB0eXBlLlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEB0eXBlIHsoY2FsbGJhY2s6ICgpID0+ID8pID0+ID99ICovICgoY2FsbGJhY2spID0+IHRoaXMuYXN5bmNMb2NhbFN0b3JhZ2UucnVuKHVuZGVmaW5lZCwgY2FsbGJhY2spKVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY2hlY2tpbi5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIERhdGFiYXNlIGNvbm5lY3Rpb24gaW5zdGFuY2UuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gdGhlIGNvbm5lY3Rpb24gaXMgY2hlY2tlZCBpbiBvciBjbG9zZWQuXG4gICAqL1xuICBhc3luYyBjaGVja2luKGNvbm5lY3Rpb24pIHtcbiAgICBjb25zdCBpZCA9IGNvbm5lY3Rpb24uZ2V0SWRTZXEoKVxuICAgIGNvbnN0IHRyYWNrZWRDb25uZWN0aW9uID0gLyoqXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEB0eXBlIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCAmIHtbQ0xPU0VEX0NPTk5FQ1RJT05dPzogYm9vbGVhbiwgW0NPTk5FQ1RJT05fQ0hFQ0tFRF9PVVRfQVRdPzogbnVtYmVyLCBbSURMRV9DT05ORUNUSU9OX0NIRUNLRURfSU5fQVRdPzogbnVtYmVyfX0gKi8gKGNvbm5lY3Rpb24pXG5cbiAgICBpZiAodHJhY2tlZENvbm5lY3Rpb25bQ0xPU0VEX0NPTk5FQ1RJT05dKSB7XG4gICAgICB0aGlzLnVudHJhY2tDb25uZWN0aW9uSW5Vc2UoY29ubmVjdGlvbiwgaWQpXG4gICAgICBhd2FpdCB0aGlzLmRyYWluUGVuZGluZ0NoZWNrb3V0cygpXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgdGhpcy5yb2xsYmFja0xlZnRPcGVuVHJhbnNhY3Rpb24oY29ubmVjdGlvbilcbiAgICAgIGF3YWl0IGNvbm5lY3Rpb24uY2xlYXJDb25uZWN0aW9uQ2hlY2tvdXROYW1lKClcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgYXdhaXQgdGhpcy5jbG9zZUNoZWNrZWRPdXRDb25uZWN0aW9uQWZ0ZXJDaGVja2luRmFpbHVyZShjb25uZWN0aW9uLCBpZCwgZXJyb3IpXG4gICAgICB0aHJvdyBlcnJvclxuICAgIH1cblxuICAgIHRoaXMudW50cmFja0Nvbm5lY3Rpb25JblVzZShjb25uZWN0aW9uLCBpZClcbiAgICB0cmFja2VkQ29ubmVjdGlvbltJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF0gPSBEYXRlLm5vdygpXG4gICAgZGVsZXRlIHRyYWNrZWRDb25uZWN0aW9uW0NPTk5FQ1RJT05fQ0hFQ0tFRF9PVVRfQVRdXG4gICAgdGhpcy5jb25uZWN0aW9ucy5wdXNoKGNvbm5lY3Rpb24pXG4gICAgYXdhaXQgdGhpcy5kcmFpblBlbmRpbmdDaGVja291dHMoKVxuICAgIGlmICh0aGlzLmNvbm5lY3Rpb25zLmluY2x1ZGVzKGNvbm5lY3Rpb24pKSBhd2FpdCB0aGlzLmhhbmRsZUNoZWNrZWRJbklkbGVDb25uZWN0aW9uKClcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGNsb3NlIGNoZWNrZWQgb3V0IGNvbm5lY3Rpb24gYWZ0ZXIgY2hlY2tpbiBmYWlsdXJlLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0fSBjb25uZWN0aW9uIC0gQ29ubmVjdGlvbiB0aGF0IGZhaWxlZCBjaGVjay1pbiBjbGVhbnVwLlxuICAgKiBAcGFyYW0ge251bWJlciB8IHVuZGVmaW5lZH0gaWQgLSBDb25uZWN0aW9uIGNoZWNrb3V0IGlkLlxuICAgKiBAcGFyYW0gez99IG9yaWdpbmFsRXJyb3IgLSBFcnJvciB0aGF0IGNhdXNlZCBjaGVjay1pbiBjbGVhbnVwIHRvIGZhaWwuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gY2xlYW51cCBoYXMgYmVlbiBhdHRlbXB0ZWQuXG4gICAqL1xuICBhc3luYyBjbG9zZUNoZWNrZWRPdXRDb25uZWN0aW9uQWZ0ZXJDaGVja2luRmFpbHVyZShjb25uZWN0aW9uLCBpZCwgb3JpZ2luYWxFcnJvcikge1xuICAgIHRoaXMudW50cmFja0Nvbm5lY3Rpb25JblVzZShjb25uZWN0aW9uLCBpZClcblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0aGlzLmNsb3NlQ29ubmVjdGlvbihjb25uZWN0aW9uKVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICB0aGlzLmxvZ2dlci53YXJuKFwiRmFpbGVkIHRvIGNsb3NlIGRhdGFiYXNlIGNvbm5lY3Rpb24gYWZ0ZXIgY2hlY2staW4gY2xlYW51cCBmYWlsZWRcIiwge2Vycm9yLCBvcmlnaW5hbEVycm9yfSlcbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgdGhpcy5kcmFpblBlbmRpbmdDaGVja291dHMoKVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICB0aGlzLmxvZ2dlci53YXJuKFwiRmFpbGVkIHRvIGRyYWluIHBlbmRpbmcgZGF0YWJhc2UgY2hlY2tvdXRzIGFmdGVyIGNoZWNrLWluIGNsZWFudXAgZmFpbGVkXCIsIHtlcnJvciwgb3JpZ2luYWxFcnJvcn0pXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgdW50cmFjayBjb25uZWN0aW9uIGluIHVzZS5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24gYmVpbmcgY2hlY2tlZCBpbi5cbiAgICogQHBhcmFtIHtudW1iZXIgfCB1bmRlZmluZWR9IGlkIC0gQ29ubmVjdGlvbiBjaGVja291dCBpZC5cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqL1xuICB1bnRyYWNrQ29ubmVjdGlvbkluVXNlKGNvbm5lY3Rpb24sIGlkKSB7XG4gICAgaWYgKHR5cGVvZiBpZCAhPT0gXCJudW1iZXJcIikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBpZFNlcSBvbiBjb25uZWN0aW9uIHdhc24ndCBzZXQ/ICcke3R5cGVvZiBpZH0nID0gJHtpZH1gKVxuICAgIH1cblxuICAgIGRlbGV0ZSB0aGlzLmNvbm5lY3Rpb25zSW5Vc2VbaWRdXG4gICAgY29ubmVjdGlvbi5zZXRJZFNlcSh1bmRlZmluZWQpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBoYW5kbGUgY2hlY2tlZCBpbiBpZGxlIGNvbm5lY3Rpb24uXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIG9uY2UgaWRsZSByZWFwaW5nIGhhcyBiZWVuIHNjaGVkdWxlZCBvciBydW4uXG4gICAqL1xuICBhc3luYyBoYW5kbGVDaGVja2VkSW5JZGxlQ29ubmVjdGlvbigpIHtcbiAgICBpZiAodGhpcy5pZGxlVGltZW91dE1pbGxpcygpID09PSAwKSB7XG4gICAgICBhd2FpdCB0aGlzLnJlYXBJZGxlQ29ubmVjdGlvbnMoKVxuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLnNjaGVkdWxlSWRsZUNvbm5lY3Rpb25SZWFwZXIoKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGNoZWNrb3V0LlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4vYmFzZS5qc1wiKS5Db25uZWN0aW9uQ2hlY2tvdXRPcHRpb25zfSBbb3B0aW9uc10gLSBDaGVja291dCBvcHRpb25zLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdD59IC0gUmVzb2x2ZXMgd2l0aCB0aGUgY2hlY2tvdXQuXG4gICAqL1xuICBhc3luYyBjaGVja291dChvcHRpb25zID0ge30pIHtcbiAgICBjb25zdCBkYXRhYmFzZUNvbmZpZyA9IHRoaXMuZ2V0Q29uZmlndXJhdGlvbigpXG4gICAgY29uc3QgcmV1c2VLZXkgPSB0aGlzLmdldENvbmZpZ3VyYXRpb25SZXVzZUtleSgpXG4gICAgbGV0IGNvbm5lY3Rpb24gPSB0aGlzLnRha2VJZGxlQ29ubmVjdGlvbkZvclJldXNlS2V5KHJldXNlS2V5KVxuXG4gICAgaWYgKGNvbm5lY3Rpb24pIHJldHVybiBhd2FpdCB0aGlzLmFjdGl2YXRlQ29ubmVjdGlvbihjb25uZWN0aW9uLCBvcHRpb25zKVxuXG4gICAgYXdhaXQgdGhpcy5yZWFwSWRsZUNvbm5lY3Rpb25zKClcbiAgICBjb25uZWN0aW9uID0gdGhpcy50YWtlSWRsZUNvbm5lY3Rpb25Gb3JSZXVzZUtleShyZXVzZUtleSlcblxuICAgIGlmIChjb25uZWN0aW9uKSByZXR1cm4gYXdhaXQgdGhpcy5hY3RpdmF0ZUNvbm5lY3Rpb24oY29ubmVjdGlvbiwgb3B0aW9ucylcblxuICAgIGlmICh0aGlzLmNhblNwYXduQ29ubmVjdGlvbigpKSB7XG4gICAgICAvLyBTcGF3biB2aWEgc3Bhd25Db25uZWN0aW9uKCkgc28gdGhlIHRlbmFudC1hd2FyZSBjb25maWd1cmF0aW9uIGlzIHJlc29sdmVkIEZSRVNIIGF0XG4gICAgICAvLyBzcGF3biB0aW1lIGZvciB0aGUgY3VycmVudCBjYWxsZXIuIFJldXNpbmcgdGhlIGRhdGFiYXNlQ29uZmlnIGNhcHR1cmVkIGF0IHRoZSB0b3Agb2ZcbiAgICAgIC8vIGNoZWNrb3V0KCkgY291bGQgYmluZCB0aGUgY29ubmVjdGlvbiB0byBhIHN0YWxlIHRlbmFudC9kYXRhYmFzZSwgd2hpY2ggYnJlYWtzXG4gICAgICAvLyBwZXItcmVxdWVzdCBpc29sYXRpb24gKGUuZy4gdGVzdCB0cnVuY2F0aW9uIGFwcGVhcmluZyBub3QgdG8gdGFrZSBlZmZlY3QpLiBUaGUgcXVldWVkXG4gICAgICAvLyBwYXRoIGJlbG93IGtlZXBzIHRoZSB3YWl0aW5nIGNhbGxlcidzIGNhcHR1cmVkIGNvbmZpZyB2aWEgd2FpdEZvckNoZWNrb3V0KCkuXG4gICAgICBjb25uZWN0aW9uID0gYXdhaXQgdGhpcy5zcGF3bkNvbm5lY3Rpb25Gb3JDaGVja291dCh0aGlzLmdldENvbmZpZ3VyYXRpb24oKSwgdGhpcy5nZXRDb25maWd1cmF0aW9uUmV1c2VLZXkoKSlcblxuICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuYWN0aXZhdGVDb25uZWN0aW9uKGNvbm5lY3Rpb24sIG9wdGlvbnMpXG4gICAgfVxuXG4gICAgcmV0dXJuIGF3YWl0IHRoaXMud2FpdEZvckNoZWNrb3V0KGRhdGFiYXNlQ29uZmlnLCByZXVzZUtleSwgb3B0aW9ucylcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHRha2UgaWRsZSBjb25uZWN0aW9uIGZvciByZXVzZSBrZXkuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSByZXVzZUtleSAtIERhdGFiYXNlIGNvbmZpZ3VyYXRpb24gcmV1c2Uga2V5LlxuICAgKiBAcGFyYW0ge29iamVjdH0gW2FyZ3NdIC0gT3B0aW9ucy5cbiAgICogQHBhcmFtIHtib29sZWFufSBbYXJncy5pbmNsdWRlT3BlblRyYW5zYWN0aW9uc10gLSBXaGV0aGVyIGNvbm5lY3Rpb25zIHdpdGggb3BlbiB0cmFuc2FjdGlvbnMgbWF5IGJlIHJldHVybmVkLlxuICAgKiBAcmV0dXJucyB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgfCB1bmRlZmluZWR9IC0gTWF0Y2hpbmcgaWRsZSBjb25uZWN0aW9uLlxuICAgKi9cbiAgdGFrZUlkbGVDb25uZWN0aW9uRm9yUmV1c2VLZXkocmV1c2VLZXksIHtpbmNsdWRlT3BlblRyYW5zYWN0aW9ucyA9IHRydWV9ID0ge30pIHtcbiAgICBjb25zdCBjb25uZWN0aW9uSW5kZXggPSB0aGlzLmNvbm5lY3Rpb25zLmZpbmRJbmRleCgocXVldWVkQ29ubmVjdGlvbikgPT4ge1xuICAgICAgaWYgKCFpbmNsdWRlT3BlblRyYW5zYWN0aW9ucyAmJiB0aGlzLmNvbm5lY3Rpb25IYXNPcGVuVHJhbnNhY3Rpb24ocXVldWVkQ29ubmVjdGlvbikpIHJldHVybiBmYWxzZVxuXG4gICAgICByZXR1cm4gdGhpcy5jb25uZWN0aW9uTWF0Y2hlc1JldXNlS2V5KHF1ZXVlZENvbm5lY3Rpb24sIHJldXNlS2V5KVxuICAgIH0pXG4gICAgY29uc3QgY29ubmVjdGlvbiA9IGNvbm5lY3Rpb25JbmRleCA9PT0gLTEgPyB1bmRlZmluZWQgOiB0aGlzLmNvbm5lY3Rpb25zLnNwbGljZShjb25uZWN0aW9uSW5kZXgsIDEpWzBdXG5cbiAgICByZXR1cm4gY29ubmVjdGlvblxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY29ubmVjdGlvbiBtYXRjaGVzIHJldXNlIGtleS5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24uXG4gICAqIEBwYXJhbSB7c3RyaW5nfSByZXVzZUtleSAtIERhdGFiYXNlIGNvbmZpZ3VyYXRpb24gcmV1c2Uga2V5LlxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gLSBXaGV0aGVyIHRoZSBjb25uZWN0aW9uIG1hdGNoZXMgdGhlIHJldXNlIGtleS5cbiAgICovXG4gIGNvbm5lY3Rpb25NYXRjaGVzUmV1c2VLZXkoY29ubmVjdGlvbiwgcmV1c2VLZXkpIHtcbiAgICBjb25zdCBjb25uZWN0aW9uV2l0aFBvb2xLZXkgPSAvKipcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAdHlwZSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgJiB7W1BPT0xfQ09ORklHVVJBVElPTl9LRVldPzogc3RyaW5nfX0gKi8gKGNvbm5lY3Rpb24pXG5cbiAgICByZXR1cm4gY29ubmVjdGlvbldpdGhQb29sS2V5W1BPT0xfQ09ORklHVVJBVElPTl9LRVldID09PSByZXVzZUtleVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgYWN0aXZhdGUgY29ubmVjdGlvbi5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24uXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi9iYXNlLmpzXCIpLkNvbm5lY3Rpb25DaGVja291dE9wdGlvbnN9IFtvcHRpb25zXSAtIENoZWNrb3V0IG9wdGlvbnMuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn0gLSBBY3RpdmF0ZWQgY29ubmVjdGlvbi5cbiAgICovXG4gIGFzeW5jIGFjdGl2YXRlQ29ubmVjdGlvbihjb25uZWN0aW9uLCBvcHRpb25zID0ge30pIHtcbiAgICBpZiAoY29ubmVjdGlvbi5nZXRJZFNlcSgpICE9PSB1bmRlZmluZWQpIHRocm93IG5ldyBFcnJvcihgQ29ubmVjdGlvbiBhbHJlYWR5IGhhcyBhbiBJRC1zZXEgLSBpcyBpdCBpbiB1c2U/ICR7Y29ubmVjdGlvbi5nZXRJZFNlcSgpfWApXG5cbiAgICBjb25zdCBpZCA9IHRoaXMuaWRTZXErK1xuXG4gICAgY29uc3QgdHJhY2tlZENvbm5lY3Rpb24gPSAvKipcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqIE5hcnJvd3MgdGhlIHJ1bnRpbWUgdmFsdWUgdG8gdGhlIGRvY3VtZW50ZWQgdHlwZS5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0ICYge1tDT05ORUNUSU9OX0NIRUNLRURfT1VUX0FUXT86IG51bWJlciwgW0lETEVfQ09OTkVDVElPTl9DSEVDS0VEX0lOX0FUXT86IG51bWJlcn19ICovIChjb25uZWN0aW9uKVxuICAgIGRlbGV0ZSB0cmFja2VkQ29ubmVjdGlvbltJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF1cbiAgICB0cmFja2VkQ29ubmVjdGlvbltDT05ORUNUSU9OX0NIRUNLRURfT1VUX0FUXSA9IERhdGUubm93KClcblxuICAgIGNvbm5lY3Rpb24uc2V0SWRTZXEoaWQpXG4gICAgdGhpcy5jb25uZWN0aW9uc0luVXNlW2lkXSA9IGNvbm5lY3Rpb25cblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCBjb25uZWN0aW9uLnNldENvbm5lY3Rpb25DaGVja291dE5hbWUob3B0aW9ucy5uYW1lKVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBkZWxldGUgdGhpcy5jb25uZWN0aW9uc0luVXNlW2lkXVxuICAgICAgY29ubmVjdGlvbi5zZXRJZFNlcSh1bmRlZmluZWQpXG4gICAgICBhd2FpdCB0aGlzLmNsb3NlQ29ubmVjdGlvbihjb25uZWN0aW9uKVxuXG4gICAgICB0aHJvdyBlcnJvclxuICAgIH1cblxuICAgIHJldHVybiBjb25uZWN0aW9uXG4gIH1cblxuICAvKipcbiAgICogUnVucyBtYXggY29ubmVjdGlvbnMuXG4gICAqIEByZXR1cm5zIHtudW1iZXIgfCB1bmRlZmluZWR9IC0gQ29uZmlndXJlZCBtYXggbGl2ZSBjb25uZWN0aW9ucy5cbiAgICovXG4gIG1heENvbm5lY3Rpb25zKCkge1xuICAgIGNvbnN0IHZhbHVlID0gdGhpcy5nZXRDb25maWd1cmF0aW9uKCkucG9vbD8ubWF4XG5cbiAgICBpZiAodGhpcy52YWxpZE1heENvbm5lY3Rpb25zKHZhbHVlKSkgcmV0dXJuIHZhbHVlXG5cbiAgICByZXR1cm5cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHZhbGlkIG1heCBjb25uZWN0aW9ucy5cbiAgICogQHBhcmFtIHs/fSB2YWx1ZSAtIENhbmRpZGF0ZSBtYXggY29ubmVjdGlvbiBjb3VudC5cbiAgICogQHJldHVybnMge3ZhbHVlIGlzIG51bWJlcn0gLSBXaGV0aGVyIHRoZSB2YWx1ZSBpcyBhIHZhbGlkIG1heCBjb25uZWN0aW9uIGNvdW50LlxuICAgKi9cbiAgdmFsaWRNYXhDb25uZWN0aW9ucyh2YWx1ZSkge1xuICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09IFwibnVtYmVyXCIgJiYgTnVtYmVyLmlzRmluaXRlKHZhbHVlKSAmJiB2YWx1ZSA+PSAxXG4gIH1cblxuICAvKipcbiAgICogUnVucyBsaXZlIGNvbm5lY3Rpb24gY291bnQuXG4gICAqIEByZXR1cm5zIHtudW1iZXJ9IC0gTnVtYmVyIG9mIGxpdmUgYW5kIGluLXByb2dyZXNzIGNvbm5lY3Rpb25zLlxuICAgKi9cbiAgbGl2ZUNvbm5lY3Rpb25Db3VudCgpIHtcbiAgICBjb25zdCBjb25uZWN0aW9ucyA9IG5ldyBTZXQoW1xuICAgICAgLi4udGhpcy5jb25uZWN0aW9ucyxcbiAgICAgIC4uLk9iamVjdC52YWx1ZXModGhpcy5jb25uZWN0aW9uc0luVXNlKSxcbiAgICAgIHRoaXMuZ2V0R2xvYmFsQ29ubmVjdGlvbkZvcklkZW50aWZpZXIoKVxuICAgIF0uZmlsdGVyKEJvb2xlYW4pKVxuXG4gICAgcmV0dXJuIGNvbm5lY3Rpb25zLnNpemUgKyB0aGlzLmNvbm5lY3Rpb25zQmVpbmdTcGF3bmVkXG4gIH1cblxuICAvKipcbiAgICogUnVucyBjYW4gc3Bhd24gY29ubmVjdGlvbi5cbiAgICogQHJldHVybnMge2Jvb2xlYW59IC0gV2hldGhlciBhIG5ldyBjb25uZWN0aW9uIGNhbiBiZSBzcGF3bmVkLlxuICAgKi9cbiAgY2FuU3Bhd25Db25uZWN0aW9uKCkge1xuICAgIGNvbnN0IG1heENvbm5lY3Rpb25zID0gdGhpcy5tYXhDb25uZWN0aW9ucygpXG5cbiAgICByZXR1cm4gbWF4Q29ubmVjdGlvbnMgPT09IHVuZGVmaW5lZCB8fCB0aGlzLmxpdmVDb25uZWN0aW9uQ291bnQoKSA8IG1heENvbm5lY3Rpb25zXG4gIH1cblxuICAvKipcbiAgICogUnVucyBzcGF3biBjb25uZWN0aW9uIGZvciBjaGVja291dC5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi8uLi9jb25maWd1cmF0aW9uLXR5cGVzLmpzXCIpLkRhdGFiYXNlQ29uZmlndXJhdGlvblR5cGV9IGRhdGFiYXNlQ29uZmlnIC0gUmVzb2x2ZWQgZGF0YWJhc2UgY29uZmlnIGZvciB0aGUgY2hlY2tvdXQuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSByZXVzZUtleSAtIERhdGFiYXNlIGNvbmZpZ3VyYXRpb24gcmV1c2Uga2V5IGZvciB0aGUgY2hlY2tvdXQuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn0gLSBTcGF3bmVkIGNvbm5lY3Rpb24uXG4gICAqL1xuICBhc3luYyBzcGF3bkNvbm5lY3Rpb25Gb3JDaGVja291dChkYXRhYmFzZUNvbmZpZywgcmV1c2VLZXkpIHtcbiAgICB0aGlzLmNvbm5lY3Rpb25zQmVpbmdTcGF3bmVkKytcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCBjb25uZWN0aW9uID0gYXdhaXQgdGhpcy5zcGF3bkNvbm5lY3Rpb25XaXRoQ29uZmlndXJhdGlvbihkYXRhYmFzZUNvbmZpZylcbiAgICAgIGNvbnN0IGNvbm5lY3Rpb25XaXRoUG9vbEtleSA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICogTmFycm93cyB0aGUgcnVudGltZSB2YWx1ZSB0byB0aGUgZG9jdW1lbnRlZCB0eXBlLlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAdHlwZSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgJiB7W1BPT0xfQ09ORklHVVJBVElPTl9LRVldPzogc3RyaW5nfX0gKi8gKGNvbm5lY3Rpb24pXG5cbiAgICAgIGNvbm5lY3Rpb25XaXRoUG9vbEtleVtQT09MX0NPTkZJR1VSQVRJT05fS0VZXSA9IHJldXNlS2V5XG4gICAgICBjb25uZWN0aW9uLnNldFNjaGVtYUNhY2hlSW52YWxpZGF0b3IoKCkgPT4ge1xuICAgICAgICB0aGlzLmNsZWFyU2NoZW1hQ2FjaGUoKVxuICAgICAgICB0aGlzLmNvbmZpZ3VyYXRpb24uY2xlYXJTY2hlbWFDYWNoZXNGb3JSZXVzZUtleShyZXVzZUtleSlcbiAgICAgIH0pXG5cbiAgICAgIHJldHVybiBjb25uZWN0aW9uXG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHRoaXMuY29ubmVjdGlvbnNCZWluZ1NwYXduZWQtLVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHdhaXQgZm9yIGNoZWNrb3V0LlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uLy4uL2NvbmZpZ3VyYXRpb24tdHlwZXMuanNcIikuRGF0YWJhc2VDb25maWd1cmF0aW9uVHlwZX0gZGF0YWJhc2VDb25maWcgLSBSZXNvbHZlZCBkYXRhYmFzZSBjb25maWcgZm9yIHRoZSBjaGVja291dC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IHJldXNlS2V5IC0gRGF0YWJhc2UgY29uZmlndXJhdGlvbiByZXVzZSBrZXkuXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi9iYXNlLmpzXCIpLkNvbm5lY3Rpb25DaGVja291dE9wdGlvbnN9IFtvcHRpb25zXSAtIENoZWNrb3V0IG9wdGlvbnMuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn0gLSBSZXNvbHZlcyB3aXRoIGFuIGFjdGl2YXRlZCBjb25uZWN0aW9uLlxuICAgKi9cbiAgYXN5bmMgd2FpdEZvckNoZWNrb3V0KGRhdGFiYXNlQ29uZmlnLCByZXVzZUtleSwgb3B0aW9ucyA9IHt9KSB7XG4gICAgcmV0dXJuIGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIHRoaXMucGVuZGluZ0NoZWNrb3V0cy5wdXNoKHtkYXRhYmFzZUNvbmZpZywgZW5xdWV1ZWRBdDogRGF0ZS5ub3coKSwgb3B0aW9ucywgcmVqZWN0LCByZXNvbHZlLCByZXVzZUtleX0pXG4gICAgICB2b2lkIHRoaXMuZHJhaW5QZW5kaW5nQ2hlY2tvdXRzKCkuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgICAgIGNvbnN0IGNoZWNrb3V0RXJyb3IgPSBlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IgOiBuZXcgRXJyb3IoXCJGYWlsZWQgdG8gZHJhaW4gcGVuZGluZyBkYXRhYmFzZSBjb25uZWN0aW9uIGNoZWNrb3V0cy5cIiwge2NhdXNlOiBlcnJvcn0pXG5cbiAgICAgICAgdGhpcy5yZWplY3RQZW5kaW5nQ2hlY2tvdXRzKGNoZWNrb3V0RXJyb3IpXG4gICAgICB9KVxuICAgIH0pXG4gIH1cblxuICAvKipcbiAgICogUnVucyBkcmFpbiBwZW5kaW5nIGNoZWNrb3V0cy5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiBwZW5kaW5nIGNoZWNrb3V0cyBoYXZlIGJlZW4gZHJhaW5lZCBhcyBmYXIgYXMgcG9zc2libGUuXG4gICAqL1xuICBhc3luYyBkcmFpblBlbmRpbmdDaGVja291dHMoKSB7XG4gICAgaWYgKHRoaXMucGVuZGluZ0NoZWNrb3V0RHJhaW5Qcm9taXNlKSB7XG4gICAgICBhd2FpdCB0aGlzLnBlbmRpbmdDaGVja291dERyYWluUHJvbWlzZVxuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgdGhpcy5wZW5kaW5nQ2hlY2tvdXREcmFpblByb21pc2UgPSB0aGlzLmRyYWluUGVuZGluZ0NoZWNrb3V0c0FjdHVhbCgpXG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgdGhpcy5wZW5kaW5nQ2hlY2tvdXREcmFpblByb21pc2VcbiAgICB9IGZpbmFsbHkge1xuICAgICAgdGhpcy5wZW5kaW5nQ2hlY2tvdXREcmFpblByb21pc2UgPSB1bmRlZmluZWRcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBkcmFpbiBwZW5kaW5nIGNoZWNrb3V0cyBhY3R1YWwuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gcGVuZGluZyBjaGVja291dHMgaGF2ZSBiZWVuIGRyYWluZWQgYXMgZmFyIGFzIHBvc3NpYmxlLlxuICAgKi9cbiAgYXN5bmMgZHJhaW5QZW5kaW5nQ2hlY2tvdXRzQWN0dWFsKCkge1xuICAgIHdoaWxlICh0aGlzLnBlbmRpbmdDaGVja291dHMubGVuZ3RoID4gMCkge1xuICAgICAgaWYgKGF3YWl0IHRoaXMucmVzb2x2ZVBlbmRpbmdDaGVja291dFdpdGhNYXRjaGluZ0lkbGVDb25uZWN0aW9uKCkpIGNvbnRpbnVlXG5cbiAgICAgIGNvbnN0IGNoZWNrb3V0ID0gdGhpcy5wZW5kaW5nQ2hlY2tvdXRzWzBdXG5cbiAgICAgIGlmIChhd2FpdCB0aGlzLmNsb3NlSWRsZUNvbm5lY3Rpb25Gb3JQZW5kaW5nQ2hlY2tvdXRDYXBhY2l0eShjaGVja291dCkpIGNvbnRpbnVlXG4gICAgICBpZiAodGhpcy5jYW5TcGF3bkNvbm5lY3Rpb24oKSkge1xuICAgICAgICB0aGlzLnBlbmRpbmdDaGVja291dHMuc2hpZnQoKVxuICAgICAgICBhd2FpdCB0aGlzLnNwYXduQW5kUmVzb2x2ZVBlbmRpbmdDaGVja291dChjaGVja291dClcbiAgICAgICAgY29udGludWVcbiAgICAgIH1cblxuICAgICAgY29uc3QgcmVhcGVkQ29ubmVjdGlvbiA9IGF3YWl0IHRoaXMuaWRsZUNvbm5lY3Rpb25Gb3JQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQpXG5cbiAgICAgIGlmICghcmVhcGVkQ29ubmVjdGlvbikgcmV0dXJuXG5cbiAgICAgIHRoaXMucGVuZGluZ0NoZWNrb3V0cy5zaGlmdCgpXG4gICAgICBhd2FpdCB0aGlzLnJlc29sdmVQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQsIHJlYXBlZENvbm5lY3Rpb24pXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgcmVzb2x2ZSBwZW5kaW5nIGNoZWNrb3V0IHdpdGggbWF0Y2hpbmcgaWRsZSBjb25uZWN0aW9uLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxib29sZWFuPn0gLSBXaGV0aGVyIGEgcGVuZGluZyBjaGVja291dCB3YXMgcmVzb2x2ZWQgd2l0aCBhbiBpZGxlIGNvbm5lY3Rpb24uXG4gICAqL1xuICBhc3luYyByZXNvbHZlUGVuZGluZ0NoZWNrb3V0V2l0aE1hdGNoaW5nSWRsZUNvbm5lY3Rpb24oKSB7XG4gICAgZm9yIChsZXQgaW5kZXggPSAwOyBpbmRleCA8IHRoaXMucGVuZGluZ0NoZWNrb3V0cy5sZW5ndGg7IGluZGV4KyspIHtcbiAgICAgIGNvbnN0IGNoZWNrb3V0ID0gdGhpcy5wZW5kaW5nQ2hlY2tvdXRzW2luZGV4XVxuICAgICAgY29uc3QgY29ubmVjdGlvbiA9IHRoaXMudGFrZUlkbGVDb25uZWN0aW9uRm9yUmV1c2VLZXkoY2hlY2tvdXQucmV1c2VLZXksIHtpbmNsdWRlT3BlblRyYW5zYWN0aW9uczogZmFsc2V9KVxuXG4gICAgICBpZiAoIWNvbm5lY3Rpb24pIGNvbnRpbnVlXG5cbiAgICAgIHRoaXMucGVuZGluZ0NoZWNrb3V0cy5zcGxpY2UoaW5kZXgsIDEpXG4gICAgICBhd2FpdCB0aGlzLnJlc29sdmVQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQsIGNvbm5lY3Rpb24pXG5cbiAgICAgIHJldHVybiB0cnVlXG4gICAgfVxuXG4gICAgcmV0dXJuIGZhbHNlXG4gIH1cblxuICAvKipcbiAgICogUnVucyBjbG9zZSBpZGxlIGNvbm5lY3Rpb24gZm9yIHBlbmRpbmcgY2hlY2tvdXQgY2FwYWNpdHkuXG4gICAqIEBwYXJhbSB7UGVuZGluZ0NoZWNrb3V0fSBjaGVja291dCAtIENoZWNrb3V0IHdhaXRpbmcgZm9yIGEgY29ubmVjdGlvbi5cbiAgICogQHJldHVybnMge1Byb21pc2U8Ym9vbGVhbj59IC0gV2hldGhlciBhbiBpZGxlIGNvbm5lY3Rpb24gd2FzIGNsb3NlZCB0byBmcmVlIGNhcGFjaXR5LlxuICAgKi9cbiAgYXN5bmMgY2xvc2VJZGxlQ29ubmVjdGlvbkZvclBlbmRpbmdDaGVja291dENhcGFjaXR5KGNoZWNrb3V0KSB7XG4gICAgY29uc3QgY29ubmVjdGlvbiA9IHRoaXMuZmluZElkbGVDb25uZWN0aW9uRm9yUmV1c2VLZXkoY2hlY2tvdXQucmV1c2VLZXkpXG5cbiAgICBpZiAoY29ubmVjdGlvbikgcmV0dXJuIGZhbHNlXG5cbiAgICBhd2FpdCB0aGlzLnJlYXBJZGxlQ29ubmVjdGlvbnMoKVxuXG4gICAgaWYgKHRoaXMuZmluZElkbGVDb25uZWN0aW9uRm9yUmV1c2VLZXkoY2hlY2tvdXQucmV1c2VLZXkpKSByZXR1cm4gZmFsc2VcblxuICAgIHJldHVybiB0aGlzLmNhblNwYXduQ29ubmVjdGlvbigpID8gZmFsc2UgOiBhd2FpdCB0aGlzLmNsb3NlT25lSWRsZUNvbm5lY3Rpb25Gb3JDYXBhY2l0eSgpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBmaW5kIGlkbGUgY29ubmVjdGlvbiBmb3IgcmV1c2Uga2V5LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gcmV1c2VLZXkgLSBEYXRhYmFzZSBjb25maWd1cmF0aW9uIHJldXNlIGtleS5cbiAgICogQHJldHVybnMge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0IHwgdW5kZWZpbmVkfSAtIE1hdGNoaW5nIGlkbGUgY29ubmVjdGlvbiwgaWYgcHJlc2VudC5cbiAgICovXG4gIGZpbmRJZGxlQ29ubmVjdGlvbkZvclJldXNlS2V5KHJldXNlS2V5KSB7XG4gICAgcmV0dXJuIHRoaXMuY29ubmVjdGlvbnMuZmluZCgoY29ubmVjdGlvbikgPT4gIXRoaXMuY29ubmVjdGlvbkhhc09wZW5UcmFuc2FjdGlvbihjb25uZWN0aW9uKSAmJiB0aGlzLmNvbm5lY3Rpb25NYXRjaGVzUmV1c2VLZXkoY29ubmVjdGlvbiwgcmV1c2VLZXkpKVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgaWRsZSBjb25uZWN0aW9uIGZvciBwZW5kaW5nIGNoZWNrb3V0LlxuICAgKiBAcGFyYW0ge1BlbmRpbmdDaGVja291dH0gY2hlY2tvdXQgLSBDaGVja291dCB3YWl0aW5nIGZvciBhIGNvbm5lY3Rpb24uXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0IHwgdW5kZWZpbmVkPn0gLSBNYXRjaGluZyBpZGxlIGNvbm5lY3Rpb24sIGlmIG9uZSBjYW4gYmUgcmV1c2VkLlxuICAgKi9cbiAgYXN5bmMgaWRsZUNvbm5lY3Rpb25Gb3JQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQpIHtcbiAgICBsZXQgY29ubmVjdGlvbiA9IHRoaXMudGFrZUlkbGVDb25uZWN0aW9uRm9yUmV1c2VLZXkoY2hlY2tvdXQucmV1c2VLZXksIHtpbmNsdWRlT3BlblRyYW5zYWN0aW9uczogZmFsc2V9KVxuXG4gICAgaWYgKGNvbm5lY3Rpb24pIHJldHVybiBjb25uZWN0aW9uXG5cbiAgICBhd2FpdCB0aGlzLnJlYXBJZGxlQ29ubmVjdGlvbnMoKVxuICAgIGNvbm5lY3Rpb24gPSB0aGlzLnRha2VJZGxlQ29ubmVjdGlvbkZvclJldXNlS2V5KGNoZWNrb3V0LnJldXNlS2V5LCB7aW5jbHVkZU9wZW5UcmFuc2FjdGlvbnM6IGZhbHNlfSlcblxuICAgIHJldHVybiBjb25uZWN0aW9uXG4gIH1cblxuICAvKipcbiAgICogUnVucyBzcGF3biBhbmQgcmVzb2x2ZSBwZW5kaW5nIGNoZWNrb3V0LlxuICAgKiBAcGFyYW0ge1BlbmRpbmdDaGVja291dH0gY2hlY2tvdXQgLSBDaGVja291dCByZXF1ZXN0IHRvIHJlc29sdmUuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gdGhlIGNoZWNrb3V0IGhhcyBiZWVuIGhhbmRsZWQuXG4gICAqL1xuICBhc3luYyBzcGF3bkFuZFJlc29sdmVQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQpIHtcbiAgICBsZXQgY29ubmVjdGlvblxuXG4gICAgdHJ5IHtcbiAgICAgIGNvbm5lY3Rpb24gPSBhd2FpdCB0aGlzLnNwYXduQ29ubmVjdGlvbkZvckNoZWNrb3V0KGNoZWNrb3V0LmRhdGFiYXNlQ29uZmlnLCBjaGVja291dC5yZXVzZUtleSlcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgY2hlY2tvdXQucmVqZWN0KGVycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvciA6IG5ldyBFcnJvcihcIkZhaWxlZCB0byBzcGF3biBkYXRhYmFzZSBjb25uZWN0aW9uLlwiLCB7Y2F1c2U6IGVycm9yfSkpXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBhd2FpdCB0aGlzLnJlc29sdmVQZW5kaW5nQ2hlY2tvdXQoY2hlY2tvdXQsIGNvbm5lY3Rpb24pXG4gIH1cblxuICAvKipcbiAgICogUnVucyByZXNvbHZlIHBlbmRpbmcgY2hlY2tvdXQuXG4gICAqIEBwYXJhbSB7UGVuZGluZ0NoZWNrb3V0fSBjaGVja291dCAtIENoZWNrb3V0IHJlcXVlc3QgdG8gcmVzb2x2ZS5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24gdG8gYWN0aXZhdGUuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gdGhlIGNoZWNrb3V0IGhhcyBiZWVuIGhhbmRsZWQuXG4gICAqL1xuICBhc3luYyByZXNvbHZlUGVuZGluZ0NoZWNrb3V0KGNoZWNrb3V0LCBjb25uZWN0aW9uKSB7XG4gICAgdHJ5IHtcbiAgICAgIGNoZWNrb3V0LnJlc29sdmUoYXdhaXQgdGhpcy5hY3RpdmF0ZUNvbm5lY3Rpb24oY29ubmVjdGlvbiwgY2hlY2tvdXQub3B0aW9ucykpXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGNoZWNrb3V0LnJlamVjdChlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IgOiBuZXcgRXJyb3IoXCJGYWlsZWQgdG8gYWN0aXZhdGUgZGF0YWJhc2UgY29ubmVjdGlvbi5cIiwge2NhdXNlOiBlcnJvcn0pKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGNsb3NlIG9uZSBpZGxlIGNvbm5lY3Rpb24gZm9yIGNhcGFjaXR5LlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxib29sZWFuPn0gLSBXaGV0aGVyIGFuIGlkbGUgY29ubmVjdGlvbiB3YXMgY2xvc2VkIHRvIGZyZWUgY2FwYWNpdHkuXG4gICAqL1xuICBhc3luYyBjbG9zZU9uZUlkbGVDb25uZWN0aW9uRm9yQ2FwYWNpdHkoKSB7XG4gICAgY29uc3QgY29ubmVjdGlvbiA9IHRoaXMuY29ubmVjdGlvbnMuZmluZCgoY2FuZGlkYXRlKSA9PiAhdGhpcy5jb25uZWN0aW9uSGFzT3BlblRyYW5zYWN0aW9uKGNhbmRpZGF0ZSkpXG5cbiAgICBpZiAoIWNvbm5lY3Rpb24pIHJldHVybiBmYWxzZVxuXG4gICAgdGhpcy5jb25uZWN0aW9ucyA9IHRoaXMuY29ubmVjdGlvbnMuZmlsdGVyKChjYW5kaWRhdGUpID0+IGNhbmRpZGF0ZSAhPT0gY29ubmVjdGlvbilcbiAgICBhd2FpdCB0aGlzLmNsb3NlQ29ubmVjdGlvbihjb25uZWN0aW9uKVxuXG4gICAgcmV0dXJuIHRydWVcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIHdpdGggY29ubmVjdGlvbi5cbiAgICogQHRlbXBsYXRlIFRcbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuL2Jhc2UuanNcIikuQ29ubmVjdGlvbkNoZWNrb3V0T3B0aW9ucyB8IGZ1bmN0aW9uKGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0KSA6IFByb21pc2U8VD59IG9wdGlvbnNPckNhbGxiYWNrIC0gQ2hlY2tvdXQgb3B0aW9ucyBvciBjYWxsYmFjayB0byBpbnZva2Ugd2l0aCB0aGUgY29ubmVjdGlvbi5cbiAgICogQHBhcmFtIHtmdW5jdGlvbihpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCkgOiBQcm9taXNlPFQ+fSBbY2FsbGJhY2tdIC0gQ2FsbGJhY2sgdG8gaW52b2tlIHdpdGggdGhlIGNvbm5lY3Rpb24uXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPFQ+fSAtIFJlc29sdmVzIHdpdGggdGhlIGNhbGxiYWNrIHJlc3VsdC5cbiAgICovXG4gIGFzeW5jIHdpdGhDb25uZWN0aW9uKG9wdGlvbnNPckNhbGxiYWNrLCBjYWxsYmFjaykge1xuICAgIGNvbnN0IG9wdGlvbnMgPSB0eXBlb2Ygb3B0aW9uc09yQ2FsbGJhY2sgPT0gXCJmdW5jdGlvblwiID8ge30gOiBvcHRpb25zT3JDYWxsYmFja1xuICAgIGNvbnN0IGFjdHVhbENhbGxiYWNrID0gdHlwZW9mIG9wdGlvbnNPckNhbGxiYWNrID09IFwiZnVuY3Rpb25cIiA/IG9wdGlvbnNPckNhbGxiYWNrIDogY2FsbGJhY2tcblxuICAgIGlmICghYWN0dWFsQ2FsbGJhY2spIHRocm93IG5ldyBFcnJvcihcIndpdGhDb25uZWN0aW9uIHJlcXVpcmVzIGEgY2FsbGJhY2tcIilcblxuICAgIGNvbnN0IGNvbm5lY3Rpb24gPSBhd2FpdCB0aGlzLmNoZWNrb3V0KG9wdGlvbnMpXG4gICAgY29uc3QgaWQgPSBjb25uZWN0aW9uLmdldElkU2VxKClcblxuICAgIHJldHVybiBhd2FpdCB0aGlzLmFzeW5jTG9jYWxTdG9yYWdlLnJ1bihpZCwgYXN5bmMgKCkgPT4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgcmV0dXJuIGF3YWl0IGFjdHVhbENhbGxiYWNrKGNvbm5lY3Rpb24pXG4gICAgICB9IGZpbmFsbHkge1xuICAgICAgICBhd2FpdCB0aGlzLmNoZWNraW4oY29ubmVjdGlvbilcbiAgICAgIH1cbiAgICB9KVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgZ2V0IGN1cnJlbnQgY29ubmVjdGlvbi5cbiAgICogQHJldHVybnMge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0fSAtIFRoZSBjdXJyZW50IGNvbm5lY3Rpb24uXG4gICAqL1xuICBnZXRDdXJyZW50Q29ubmVjdGlvbigpIHtcbiAgICBjb25zdCBpZCA9IHRoaXMuYXN5bmNMb2NhbFN0b3JhZ2UuZ2V0U3RvcmUoKVxuXG4gICAgaWYgKGlkID09PSB1bmRlZmluZWQpIHJldHVybiB0aGlzLmN1cnJlbnRGYWxsYmFja0Nvbm5lY3Rpb25PckZhaWwoKVxuXG4gICAgdGhpcy5lbnN1cmVDb25uZWN0aW9uSXNJblVzZShpZClcblxuICAgIGNvbnN0IGN1cnJlbnRDb25uZWN0aW9uID0gdGhpcy5jb25uZWN0aW9uc0luVXNlW2lkXVxuXG4gICAgaWYgKCFjdXJyZW50Q29ubmVjdGlvbikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBDb3VsZG4ndCBnZXQgY3VycmVudCBjb25uZWN0aW9uIGZyb20gdGhhdCBJRDogJHtpZH1gKVxuICAgIH1cblxuICAgIHJldHVybiBjdXJyZW50Q29ubmVjdGlvblxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY3VycmVudCBmYWxsYmFjayBjb25uZWN0aW9uIG9yIGZhaWwuXG4gICAqIEByZXR1cm5zIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gLSBGYWxsYmFjayBjb25uZWN0aW9uLCBpZiBwcmVzZW50LlxuICAgKi9cbiAgY3VycmVudEZhbGxiYWNrQ29ubmVjdGlvbk9yRmFpbCgpIHtcbiAgICBjb25zdCBmYWxsYmFja0Nvbm5lY3Rpb24gPSB0aGlzLmdldEdsb2JhbENvbm5lY3Rpb24oKVxuXG4gICAgaWYgKGZhbGxiYWNrQ29ubmVjdGlvbikgcmV0dXJuIGZhbGxiYWNrQ29ubmVjdGlvblxuXG4gICAgdGhyb3cgbmV3IEVycm9yKFwiSUQgaGFzbid0IGJlZW4gc2V0IGZvciB0aGlzIGFzeW5jIGNvbnRleHRcIilcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGVuc3VyZSBjb25uZWN0aW9uIGlzIGluIHVzZS5cbiAgICogQHBhcmFtIHtudW1iZXJ9IGlkIC0gQ2hlY2tlZC1vdXQgY29ubmVjdGlvbiBpZC5cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqL1xuICBlbnN1cmVDb25uZWN0aW9uSXNJblVzZShpZCkge1xuICAgIGlmICghKGlkIGluIHRoaXMuY29ubmVjdGlvbnNJblVzZSkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgQ29ubmVjdGlvbiAke2lkfSBkb2Vzbid0IGV4aXN0IGFueSBtb3JlIC0gaGFzIGl0IGJlZW4gY2hlY2tlZCBpbiBhZ2Fpbj9gKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlcnMgYSBmYWxsYmFjayBjb25uZWN0aW9uIGZvciB0aGlzIHBvb2wgaWRlbnRpZmllciB0aGF0IHdpbGwgYmUgdXNlZCB3aGVuIG5vIGFzeW5jIGNvbnRleHQgaXMgYXZhaWxhYmxlLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0fSBjb25uZWN0aW9uIC0gQ29ubmVjdGlvbi5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgc2V0R2xvYmFsQ29ubmVjdGlvbihjb25uZWN0aW9uKSB7XG4gICAgY29uc3Qga2xhc3MgPSAvKipcbiAgICAgICAgICAgICAgICAgICAqIE5hcnJvd3MgdGhlIHJ1bnRpbWUgdmFsdWUgdG8gdGhlIGRvY3VtZW50ZWQgdHlwZS5cbiAgICAgICAgICAgICAgICAgICAgQHR5cGUge3R5cGVvZiBWZWxvY2lvdXNEYXRhYmFzZVBvb2xBc3luY1RyYWNrZWRNdWx0aUNvbm5lY3Rpb259ICovICh0aGlzLmNvbnN0cnVjdG9yKVxuICAgIGxldCBtYXBGb3JDb25maWd1cmF0aW9uID0ga2xhc3MuZ2xvYmFsQ29ubmVjdGlvbnMuZ2V0KHRoaXMuY29uZmlndXJhdGlvbilcblxuICAgIGlmICghbWFwRm9yQ29uZmlndXJhdGlvbikge1xuICAgICAgbWFwRm9yQ29uZmlndXJhdGlvbiA9IHt9XG4gICAgICBrbGFzcy5nbG9iYWxDb25uZWN0aW9ucy5zZXQodGhpcy5jb25maWd1cmF0aW9uLCBtYXBGb3JDb25maWd1cmF0aW9uKVxuICAgIH1cblxuICAgIG1hcEZvckNvbmZpZ3VyYXRpb25bdGhpcy5pZGVudGlmaWVyXSA9IGNvbm5lY3Rpb25cbiAgfVxuXG4gIC8qKlxuICAgKiBFbnN1cmVzIGEgZ2xvYmFsIGZhbGxiYWNrIGNvbm5lY3Rpb24gZXhpc3RzIGZvciB0aGlzIHBvb2wgaWRlbnRpZmllciBhbmQgcmV0dXJucyBpdC5cbiAgICogSWYgb25lIGlzIGFscmVhZHkgc2V0LCBpdCBpcyByZXR1cm5lZCBhbmQgYWxzbyBtYWRlIGF2YWlsYWJsZSBpbiB0aGUgcG9vbCBxdWV1ZS5cbiAgICogT3RoZXJ3aXNlIGEgbmV3IGNvbm5lY3Rpb24gaXMgc3Bhd25lZCwgcmVnaXN0ZXJlZCwgYW5kIHF1ZXVlZC5cbiAgICogQHJldHVybnMge1Byb21pc2U8aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQ+fSAtIFJlc29sdmVzIHdpdGggdGhlIGdsb2JhbCBjb25uZWN0aW9uLlxuICAgKi9cbiAgYXN5bmMgZW5zdXJlR2xvYmFsQ29ubmVjdGlvbigpIHtcbiAgICBjb25zdCBleGlzdGluZyA9IHRoaXMuZ2V0R2xvYmFsQ29ubmVjdGlvbigpXG5cbiAgICBpZiAoZXhpc3RpbmcpIHJldHVybiBleGlzdGluZ1xuXG4gICAgY29uc3QgY29ubmVjdGlvbiA9IGF3YWl0IHRoaXMuc3Bhd25Db25uZWN0aW9uKClcblxuICAgIHRoaXMuc2V0R2xvYmFsQ29ubmVjdGlvbihjb25uZWN0aW9uKVxuXG4gICAgcmV0dXJuIGNvbm5lY3Rpb25cbiAgfVxuXG4gIC8qKlxuICAgKiBTZXQgYSBzaGFyZWQgY29ubmVjdGlvbiBmb3IgdGVzdCBtb2RlIHNvIHRoYXQgSFRUUCBoYW5kbGVycyBydW5uaW5nXG4gICAqIGluIHRoZSBzYW1lIHByb2Nlc3MgY2FuIHJldXNlIHRoZSB0ZXN0IHJ1bm5lcidzIGRhdGFiYXNlIGNvbm5lY3Rpb24uXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHR9IGNvbm5lY3Rpb24gLSBTaGFyZWQgY29ubmVjdGlvbi5cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqL1xuICBzZXRUZXN0U2hhcmVkQ29ubmVjdGlvbihjb25uZWN0aW9uKSB7XG4gICAgdGhpcy5fdGVzdFNoYXJlZENvbm5lY3Rpb24gPSBjb25uZWN0aW9uXG4gIH1cblxuICAvKipcbiAgICogUnVucyBjbGVhciB0ZXN0IHNoYXJlZCBjb25uZWN0aW9uLlxuICAgIEByZXR1cm5zIHt2b2lkfSAqL1xuICBjbGVhclRlc3RTaGFyZWRDb25uZWN0aW9uKCkge1xuICAgIHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uID0gdW5kZWZpbmVkXG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyB0aGUgY29ubmVjdGlvbiB0aWVkIHRvIHRoZSBjdXJyZW50IGFzeW5jIGNvbnRleHQsIGlmIGFueS5cbiAgICogRmFsbHMgYmFjayB0byB0aGUgdGVzdCBzaGFyZWQgY29ubmVjdGlvbiB3aGVuIG5vIGFzeW5jIGNvbnRleHQgZXhpc3RzLlxuICAgKiBAcmV0dXJucyB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgfCB1bmRlZmluZWR9IC0gVGhlIGN1cnJlbnQgY29udGV4dCBjb25uZWN0aW9uLlxuICAgKi9cbiAgZ2V0Q3VycmVudENvbnRleHRDb25uZWN0aW9uKCkge1xuICAgIGNvbnN0IGlkID0gdGhpcy5hc3luY0xvY2FsU3RvcmFnZS5nZXRTdG9yZSgpXG5cbiAgICBpZiAoaWQgPT09IHVuZGVmaW5lZCkgcmV0dXJuIHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uXG5cbiAgICByZXR1cm4gdGhpcy5nZXRDdXJyZW50Q29ubmVjdGlvbigpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBnZXQgZGVidWcgc25hcHNob3QuXG4gICAqIEByZXR1cm5zIHtpbXBvcnQoXCIuL2Jhc2UuanNcIikuRGF0YWJhc2VQb29sRGVidWdTbmFwc2hvdH0gLSBEaWFnbm9zdGljIHNuYXBzaG90IGZvciB0aGlzIHBvb2wuXG4gICAqL1xuICBnZXREZWJ1Z1NuYXBzaG90KCkge1xuICAgIGNvbnN0IHNuYXBzaG90ID0gc3VwZXIuZ2V0RGVidWdTbmFwc2hvdCgpXG4gICAgY29uc3Qgbm93ID0gRGF0ZS5ub3coKVxuICAgIGNvbnN0IHtjb25uZWN0aW9uc30gPSB0aGlzLmRlYnVnQ29ubmVjdGlvblNuYXBzaG90cyhub3cpXG5cbiAgICByZXR1cm4ge1xuICAgICAgLi4uc25hcHNob3QsXG4gICAgICBjb25uZWN0aW9ucyxcbiAgICAgIGNvbm5lY3Rpb25zQmVpbmdTcGF3bmVkOiB0aGlzLmNvbm5lY3Rpb25zQmVpbmdTcGF3bmVkLFxuICAgICAgaWRsZUNvdW50OiB0aGlzLmNvbm5lY3Rpb25zLmxlbmd0aCxcbiAgICAgIGluVXNlQ291bnQ6IE9iamVjdC5rZXlzKHRoaXMuY29ubmVjdGlvbnNJblVzZSkubGVuZ3RoLFxuICAgICAgcGVuZGluZ0NoZWNrb3V0czogdGhpcy5wZW5kaW5nQ2hlY2tvdXREZWJ1Z1NuYXBzaG90cyhub3cpLFxuICAgICAgcGVuZGluZ0NoZWNrb3V0Q291bnQ6IHRoaXMucGVuZGluZ0NoZWNrb3V0cy5sZW5ndGhcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBkZWJ1ZyBjb25uZWN0aW9uIHNuYXBzaG90cy5cbiAgICogQHBhcmFtIHtudW1iZXJ9IG5vdyAtIEN1cnJlbnQgdGltZXN0YW1wLlxuICAgKiBAcmV0dXJucyB7e2Nvbm5lY3Rpb25zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCA/Pj4sIHNlZW5Db25uZWN0aW9uczogU2V0PGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn19IC0gQ29ubmVjdGlvbiBzbmFwc2hvdHMgYW5kIHNlZW4gc2V0LlxuICAgKi9cbiAgZGVidWdDb25uZWN0aW9uU25hcHNob3RzKG5vdykge1xuICAgIC8qKlxuICAgICAqIENvbm5lY3Rpb25zLlxuICAgICAgQHR5cGUge0FycmF5PFJlY29yZDxzdHJpbmcsID8+Pn0gKi9cbiAgICBjb25zdCBjb25uZWN0aW9ucyA9IFtdXG4gICAgY29uc3Qgc2VlbkNvbm5lY3Rpb25zID0gbmV3IFNldCgpXG5cbiAgICB0aGlzLmFkZEluVXNlRGVidWdDb25uZWN0aW9uU25hcHNob3RzKHtjb25uZWN0aW9ucywgbm93LCBzZWVuQ29ubmVjdGlvbnN9KVxuICAgIHRoaXMuYWRkSWRsZURlYnVnQ29ubmVjdGlvblNuYXBzaG90cyh7Y29ubmVjdGlvbnMsIG5vdywgc2VlbkNvbm5lY3Rpb25zfSlcbiAgICB0aGlzLmFkZEZhbGxiYWNrRGVidWdDb25uZWN0aW9uU25hcHNob3RzKHtjb25uZWN0aW9ucywgc2VlbkNvbm5lY3Rpb25zfSlcblxuICAgIHJldHVybiB7Y29ubmVjdGlvbnMsIHNlZW5Db25uZWN0aW9uc31cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGFkZCBpbiB1c2UgZGVidWcgY29ubmVjdGlvbiBzbmFwc2hvdHMuXG4gICAqIEBwYXJhbSB7e2Nvbm5lY3Rpb25zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCA/Pj4sIG5vdzogbnVtYmVyLCBzZWVuQ29ubmVjdGlvbnM6IFNldDxpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdD59fSBhcmdzIC0gU25hcHNob3QgY29sbGVjdGlvbiBzdGF0ZS5cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqL1xuICBhZGRJblVzZURlYnVnQ29ubmVjdGlvblNuYXBzaG90cyh7Y29ubmVjdGlvbnMsIG5vdywgc2VlbkNvbm5lY3Rpb25zfSkge1xuICAgIGZvciAoY29uc3QgW2lkLCBjb25uZWN0aW9uXSBvZiBPYmplY3QuZW50cmllcyh0aGlzLmNvbm5lY3Rpb25zSW5Vc2UpKSB7XG4gICAgICBjb25zdCB0cmFja2VkQ29ubmVjdGlvbiA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0ICYge1tDT05ORUNUSU9OX0NIRUNLRURfT1VUX0FUXT86IG51bWJlcn19ICovIChjb25uZWN0aW9uKVxuICAgICAgY29uc3QgY2hlY2tlZE91dEF0ID0gdHJhY2tlZENvbm5lY3Rpb25bQ09OTkVDVElPTl9DSEVDS0VEX09VVF9BVF1cbiAgICAgIGNvbnN0IGNoZWNrZWRPdXRGb3JNcyA9IHR5cGVvZiBjaGVja2VkT3V0QXQgPT09IFwibnVtYmVyXCIgPyBNYXRoLm1heCgwLCBub3cgLSBjaGVja2VkT3V0QXQpIDogdW5kZWZpbmVkXG5cbiAgICAgIHNlZW5Db25uZWN0aW9ucy5hZGQoY29ubmVjdGlvbilcbiAgICAgIGNvbm5lY3Rpb25zLnB1c2godGhpcy5kZWJ1Z0Nvbm5lY3Rpb25TbmFwc2hvdChjb25uZWN0aW9uLCB7Y2hlY2tlZE91dEF0LCBjaGVja2VkT3V0Rm9yTXMsIGNoZWNrb3V0SWQ6IGlkLCBzdGF0ZTogXCJpbi11c2VcIn0pKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGFkZCBpZGxlIGRlYnVnIGNvbm5lY3Rpb24gc25hcHNob3RzLlxuICAgKiBAcGFyYW0ge3tjb25uZWN0aW9uczogQXJyYXk8UmVjb3JkPHN0cmluZywgPz4+LCBub3c6IG51bWJlciwgc2VlbkNvbm5lY3Rpb25zOiBTZXQ8aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQ+fX0gYXJncyAtIFNuYXBzaG90IGNvbGxlY3Rpb24gc3RhdGUuXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKi9cbiAgYWRkSWRsZURlYnVnQ29ubmVjdGlvblNuYXBzaG90cyh7Y29ubmVjdGlvbnMsIG5vdywgc2VlbkNvbm5lY3Rpb25zfSkge1xuICAgIGZvciAoY29uc3QgY29ubmVjdGlvbiBvZiB0aGlzLmNvbm5lY3Rpb25zKSB7XG4gICAgICBpZiAoc2VlbkNvbm5lY3Rpb25zLmhhcyhjb25uZWN0aW9uKSkgY29udGludWVcblxuICAgICAgc2VlbkNvbm5lY3Rpb25zLmFkZChjb25uZWN0aW9uKVxuXG4gICAgICBjb25zdCB0cmFja2VkQ29ubmVjdGlvbiA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0ICYge1tJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF0/OiBudW1iZXJ9fSAqLyAoY29ubmVjdGlvbilcbiAgICAgIGNvbnN0IGNoZWNrZWRJbkF0ID0gdHJhY2tlZENvbm5lY3Rpb25bSURMRV9DT05ORUNUSU9OX0NIRUNLRURfSU5fQVRdXG4gICAgICBjb25zdCBpZGxlRm9yTXMgPSB0eXBlb2YgY2hlY2tlZEluQXQgPT09IFwibnVtYmVyXCIgPyBNYXRoLm1heCgwLCBub3cgLSBjaGVja2VkSW5BdCkgOiB1bmRlZmluZWRcblxuICAgICAgY29ubmVjdGlvbnMucHVzaCh0aGlzLmRlYnVnQ29ubmVjdGlvblNuYXBzaG90KGNvbm5lY3Rpb24sIHtjaGVja2VkSW5BdCwgaWRsZUZvck1zLCBzdGF0ZTogXCJpZGxlXCJ9KSlcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBhZGQgZmFsbGJhY2sgZGVidWcgY29ubmVjdGlvbiBzbmFwc2hvdHMuXG4gICAqIEBwYXJhbSB7e2Nvbm5lY3Rpb25zOiBBcnJheTxSZWNvcmQ8c3RyaW5nLCA/Pj4sIHNlZW5Db25uZWN0aW9uczogU2V0PGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn19IGFyZ3MgLSBTbmFwc2hvdCBjb2xsZWN0aW9uIHN0YXRlLlxuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICovXG4gIGFkZEZhbGxiYWNrRGVidWdDb25uZWN0aW9uU25hcHNob3RzKHtjb25uZWN0aW9ucywgc2VlbkNvbm5lY3Rpb25zfSkge1xuICAgIHRoaXMuYWRkRGVidWdDb25uZWN0aW9uU25hcHNob3RJZlVuc2Vlbih7Y29ubmVjdGlvbjogdGhpcy5nZXRHbG9iYWxDb25uZWN0aW9uRm9ySWRlbnRpZmllcigpLCBjb25uZWN0aW9ucywgcmVhcGFibGU6IGZhbHNlLCBzZWVuQ29ubmVjdGlvbnMsIHN0YXRlOiBcImdsb2JhbFwifSlcbiAgICB0aGlzLmFkZERlYnVnQ29ubmVjdGlvblNuYXBzaG90SWZVbnNlZW4oe2Nvbm5lY3Rpb246IHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uLCBjb25uZWN0aW9ucywgcmVhcGFibGU6IGZhbHNlLCBzZWVuQ29ubmVjdGlvbnMsIHN0YXRlOiBcInRlc3Qtc2hhcmVkXCJ9KVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgYWRkIGRlYnVnIGNvbm5lY3Rpb24gc25hcHNob3QgaWYgdW5zZWVuLlxuICAgKiBAcGFyYW0ge3tjb25uZWN0aW9uOiBpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCB8IHVuZGVmaW5lZCwgY29ubmVjdGlvbnM6IEFycmF5PFJlY29yZDxzdHJpbmcsID8+PiwgcmVhcGFibGU/OiBib29sZWFuLCBzZWVuQ29ubmVjdGlvbnM6IFNldDxpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdD4sIHN0YXRlOiBzdHJpbmd9fSBhcmdzIC0gU25hcHNob3QgY29sbGVjdGlvbiBzdGF0ZS5cbiAgICogQHJldHVybnMge3ZvaWR9XG4gICAqL1xuICBhZGREZWJ1Z0Nvbm5lY3Rpb25TbmFwc2hvdElmVW5zZWVuKHtjb25uZWN0aW9uLCBjb25uZWN0aW9ucywgcmVhcGFibGUsIHNlZW5Db25uZWN0aW9ucywgc3RhdGV9KSB7XG4gICAgaWYgKCFjb25uZWN0aW9uIHx8IHNlZW5Db25uZWN0aW9ucy5oYXMoY29ubmVjdGlvbikpIHJldHVyblxuXG4gICAgc2VlbkNvbm5lY3Rpb25zLmFkZChjb25uZWN0aW9uKVxuICAgIGNvbm5lY3Rpb25zLnB1c2godGhpcy5kZWJ1Z0Nvbm5lY3Rpb25TbmFwc2hvdChjb25uZWN0aW9uLCB7cmVhcGFibGUsIHN0YXRlfSkpXG4gIH1cblxuICAvKipcbiAgICogUnVucyBwZW5kaW5nIGNoZWNrb3V0IGRlYnVnIHNuYXBzaG90cy5cbiAgICogQHBhcmFtIHtudW1iZXJ9IG5vdyAtIEN1cnJlbnQgdGltZXN0YW1wLlxuICAgKiBAcmV0dXJucyB7QXJyYXk8UmVjb3JkPHN0cmluZywgPz4+fSAtIFBlbmRpbmcgY2hlY2tvdXQgc25hcHNob3RzLlxuICAgKi9cbiAgcGVuZGluZ0NoZWNrb3V0RGVidWdTbmFwc2hvdHMobm93KSB7XG4gICAgcmV0dXJuIHRoaXMucGVuZGluZ0NoZWNrb3V0cy5tYXAoKGNoZWNrb3V0LCBpbmRleCkgPT4gKHtcbiAgICAgIGNoZWNrb3V0TmFtZTogY2hlY2tvdXQub3B0aW9ucy5uYW1lLFxuICAgICAgZW5xdWV1ZWRBdDogY2hlY2tvdXQuZW5xdWV1ZWRBdCxcbiAgICAgIGluZGV4LFxuICAgICAgcmV1c2VLZXk6IGNoZWNrb3V0LnJldXNlS2V5LFxuICAgICAgd2FpdGluZ0Zvck1zOiBNYXRoLm1heCgwLCBub3cgLSBjaGVja291dC5lbnF1ZXVlZEF0KVxuICAgIH0pKVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgZ2V0IGdsb2JhbCBjb25uZWN0aW9uLlxuICAgKiBAcmV0dXJucyB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgfCB1bmRlZmluZWR9IC0gVGhlIGdsb2JhbCBjb25uZWN0aW9uLlxuICAgKi9cbiAgZ2V0R2xvYmFsQ29ubmVjdGlvbigpIHtcbiAgICBjb25zdCBjb25uZWN0aW9uID0gdGhpcy5nZXRHbG9iYWxDb25uZWN0aW9uRm9ySWRlbnRpZmllcigpXG5cbiAgICBpZiAoIWNvbm5lY3Rpb24pIHJldHVyblxuICAgIGlmICghdGhpcy5jb25uZWN0aW9uTWF0Y2hlc0N1cnJlbnRDb25maWd1cmF0aW9uKGNvbm5lY3Rpb24pKSByZXR1cm5cblxuICAgIHJldHVybiBjb25uZWN0aW9uXG4gIH1cblxuICAvKipcbiAgICogUnVucyBnZXQgZ2xvYmFsIGNvbm5lY3Rpb24gZm9yIGlkZW50aWZpZXIuXG4gICAqIEByZXR1cm5zIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCB8IHVuZGVmaW5lZH0gLSBUaGUgZ2xvYmFsIGNvbm5lY3Rpb24gZm9yIHRoaXMgcG9vbCBpZGVudGlmaWVyLlxuICAgKi9cbiAgZ2V0R2xvYmFsQ29ubmVjdGlvbkZvcklkZW50aWZpZXIoKSB7XG4gICAgY29uc3Qga2xhc3MgPSAvKipcbiAgICAgICAgICAgICAgICAgICAqIE5hcnJvd3MgdGhlIHJ1bnRpbWUgdmFsdWUgdG8gdGhlIGRvY3VtZW50ZWQgdHlwZS5cbiAgICAgICAgICAgICAgICAgICAgQHR5cGUge3R5cGVvZiBWZWxvY2lvdXNEYXRhYmFzZVBvb2xBc3luY1RyYWNrZWRNdWx0aUNvbm5lY3Rpb259ICovICh0aGlzLmNvbnN0cnVjdG9yKVxuICAgIGNvbnN0IG1hcEZvckNvbmZpZ3VyYXRpb24gPSBrbGFzcy5nbG9iYWxDb25uZWN0aW9ucy5nZXQodGhpcy5jb25maWd1cmF0aW9uKVxuXG4gICAgcmV0dXJuIG1hcEZvckNvbmZpZ3VyYXRpb24/Llt0aGlzLmlkZW50aWZpZXJdXG4gIH1cblxuICAvKipcbiAgICogUnVucyBjbGVhciBnbG9iYWwgY29ubmVjdGlvbiBmb3IgaWRlbnRpZmllci5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgY2xlYXJHbG9iYWxDb25uZWN0aW9uRm9ySWRlbnRpZmllcigpIHtcbiAgICBjb25zdCBrbGFzcyA9IC8qKlxuICAgICAgICAgICAgICAgICAgICogTmFycm93cyB0aGUgcnVudGltZSB2YWx1ZSB0byB0aGUgZG9jdW1lbnRlZCB0eXBlLlxuICAgICAgICAgICAgICAgICAgICBAdHlwZSB7dHlwZW9mIFZlbG9jaW91c0RhdGFiYXNlUG9vbEFzeW5jVHJhY2tlZE11bHRpQ29ubmVjdGlvbn0gKi8gKHRoaXMuY29uc3RydWN0b3IpXG4gICAgY29uc3QgbWFwRm9yQ29uZmlndXJhdGlvbiA9IGtsYXNzLmdsb2JhbENvbm5lY3Rpb25zLmdldCh0aGlzLmNvbmZpZ3VyYXRpb24pXG5cbiAgICBpZiAoIW1hcEZvckNvbmZpZ3VyYXRpb24pIHJldHVyblxuXG4gICAgZGVsZXRlIG1hcEZvckNvbmZpZ3VyYXRpb25bdGhpcy5pZGVudGlmaWVyXVxuICB9XG5cbiAgLyoqXG4gICAqIENsZWFycyBzY2hlbWEgbWV0YWRhdGEgY2FjaGVkIGJ5IGV2ZXJ5IGxpdmUgY29ubmVjdGlvbiBvd25lZCBieSB0aGlzIHBvb2wuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIGNsZWFyU2NoZW1hQ2FjaGUoKSB7XG4gICAgY29uc3QgY29ubmVjdGlvbnMgPSBuZXcgU2V0KFtcbiAgICAgIC4uLnRoaXMuY29ubmVjdGlvbnMsXG4gICAgICAuLi5PYmplY3QudmFsdWVzKHRoaXMuY29ubmVjdGlvbnNJblVzZSksXG4gICAgICB0aGlzLmdldEdsb2JhbENvbm5lY3Rpb24oKSxcbiAgICAgIHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uXG4gICAgXS5maWx0ZXIoQm9vbGVhbikpXG5cbiAgICBmb3IgKGNvbnN0IGNvbm5lY3Rpb24gb2YgY29ubmVjdGlvbnMpIHtcbiAgICAgIGlmIChjb25uZWN0aW9uKSB0aGlzLl9jbGVhckNvbm5lY3Rpb25TY2hlbWFDYWNoZShjb25uZWN0aW9uKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGlkbGUgdGltZW91dCBtaWxsaXMuXG4gICAqIEByZXR1cm5zIHtudW1iZXIgfCBudWxsfSAtIElkbGUgdGltZW91dCBpbiBtaWxsaXNlY29uZHMsIG9yIG51bGwgd2hlbiBkaXNhYmxlZC5cbiAgICovXG4gIGlkbGVUaW1lb3V0TWlsbGlzKCkge1xuICAgIGNvbnN0IHZhbHVlID0gdGhpcy5nZXRDb25maWd1cmF0aW9uKCkucG9vbD8uaWRsZVRpbWVvdXRNaWxsaXNcblxuICAgIGlmICh2YWx1ZSA9PT0gbnVsbCkgcmV0dXJuIG51bGxcbiAgICBpZiAodGhpcy52YWxpZElkbGVUaW1lb3V0TWlsbGlzKHZhbHVlKSkgcmV0dXJuIHZhbHVlXG5cbiAgICByZXR1cm4gREVGQVVMVF9JRExFX1RJTUVPVVRfTUlMTElTXG4gIH1cblxuICAvKipcbiAgICogUnVucyB2YWxpZCBpZGxlIHRpbWVvdXQgbWlsbGlzLlxuICAgKiBAcGFyYW0gez99IHZhbHVlIC0gQ2FuZGlkYXRlIGlkbGUgdGltZW91dCB2YWx1ZS5cbiAgICogQHJldHVybnMge3ZhbHVlIGlzIG51bWJlcn0gLSBXaGV0aGVyIHRoZSB2YWx1ZSBpcyBhIHZhbGlkIGlkbGUgdGltZW91dC5cbiAgICovXG4gIHZhbGlkSWRsZVRpbWVvdXRNaWxsaXModmFsdWUpIHtcbiAgICByZXR1cm4gdHlwZW9mIHZhbHVlID09PSBcIm51bWJlclwiICYmIE51bWJlci5pc0Zpbml0ZSh2YWx1ZSkgJiYgdmFsdWUgPj0gMFxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgc2NoZWR1bGUgaWRsZSBjb25uZWN0aW9uIHJlYXBlci5cbiAgICBAcmV0dXJucyB7dm9pZH0gKi9cbiAgc2NoZWR1bGVJZGxlQ29ubmVjdGlvblJlYXBlcigpIHtcbiAgICBpZiAodGhpcy5pZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyKSByZXR1cm5cbiAgICBpZiAoIXRoaXMuaGFzSWRsZUNvbm5lY3Rpb25zVG9SZWFwKCkpIHJldHVyblxuXG4gICAgY29uc3QgZGVsYXkgPSB0aGlzLm5leHRJZGxlQ29ubmVjdGlvblJlYXBEZWxheSgvKipcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqIE5hcnJvd3MgdGhlIHJ1bnRpbWUgdmFsdWUgdG8gdGhlIGRvY3VtZW50ZWQgdHlwZS5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge251bWJlcn0gKi8gKHRoaXMuaWRsZVRpbWVvdXRNaWxsaXMoKSkpXG5cbiAgICB0aGlzLmlkbGVDb25uZWN0aW9uUmVhcGVyVGltZXIgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIHRoaXMuaWRsZUNvbm5lY3Rpb25SZWFwZXJUaW1lciA9IHVuZGVmaW5lZFxuICAgICAgdm9pZCB0aGlzLnJlYXBJZGxlQ29ubmVjdGlvbnMoKS5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgICAgdGhpcy5sb2dnZXIud2FybigoKSA9PiBbXCJGYWlsZWQgdG8gcmVhcCBpZGxlIGRhdGFiYXNlIGNvbm5lY3Rpb25zOlwiLCBlcnJvcl0pXG4gICAgICB9KVxuICAgIH0sIGRlbGF5KVxuXG4gICAgaWYgKHR5cGVvZiB0aGlzLmlkbGVDb25uZWN0aW9uUmVhcGVyVGltZXIudW5yZWYgPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgdGhpcy5pZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyLnVucmVmKClcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBoYXMgaWRsZSBjb25uZWN0aW9ucyB0byByZWFwLlxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gLSBXaGV0aGVyIGFuIGlkbGUgcmVhcGVyIHRpbWVyIHNob3VsZCBiZSBzY2hlZHVsZWQuXG4gICAqL1xuICBoYXNJZGxlQ29ubmVjdGlvbnNUb1JlYXAoKSB7XG4gICAgcmV0dXJuIHRoaXMuY29ubmVjdGlvbnMubGVuZ3RoID4gMCAmJiB0aGlzLmlkbGVUaW1lb3V0TWlsbGlzKCkgIT09IG51bGxcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIG5leHQgaWRsZSBjb25uZWN0aW9uIHJlYXAgZGVsYXkuXG4gICAqIEBwYXJhbSB7bnVtYmVyfSBpZGxlVGltZW91dE1pbGxpcyAtIElkbGUgdGltZW91dCBpbiBtaWxsaXNlY29uZHMuXG4gICAqIEByZXR1cm5zIHtudW1iZXJ9IC0gRGVsYXkgYmVmb3JlIHRoZSBuZXh0IHJlYXAuXG4gICAqL1xuICBuZXh0SWRsZUNvbm5lY3Rpb25SZWFwRGVsYXkoaWRsZVRpbWVvdXRNaWxsaXMpIHtcbiAgICBsZXQgZGVsYXkgPSBpZGxlVGltZW91dE1pbGxpc1xuICAgIGNvbnN0IG5vdyA9IERhdGUubm93KClcblxuICAgIGZvciAoY29uc3QgY29ubmVjdGlvbiBvZiB0aGlzLmNvbm5lY3Rpb25zKSB7XG4gICAgICBpZiAodGhpcy5jb25uZWN0aW9uSGFzT3BlblRyYW5zYWN0aW9uKGNvbm5lY3Rpb24pKSBjb250aW51ZVxuXG4gICAgICBjb25zdCB0cmFja2VkQ29ubmVjdGlvbiA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKiBOYXJyb3dzIHRoZSBydW50aW1lIHZhbHVlIHRvIHRoZSBkb2N1bWVudGVkIHR5cGUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0ICYge1tJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF0/OiBudW1iZXJ9fSAqLyAoY29ubmVjdGlvbilcbiAgICAgIGNvbnN0IGNoZWNrZWRJbkF0ID0gdHJhY2tlZENvbm5lY3Rpb25bSURMRV9DT05ORUNUSU9OX0NIRUNLRURfSU5fQVRdXG5cbiAgICAgIGlmICh0eXBlb2YgY2hlY2tlZEluQXQgIT09IFwibnVtYmVyXCIpIGNvbnRpbnVlXG5cbiAgICAgIGRlbGF5ID0gTWF0aC5taW4oZGVsYXksIE1hdGgubWF4KDAsIGlkbGVUaW1lb3V0TWlsbGlzIC0gKG5vdyAtIGNoZWNrZWRJbkF0KSkpXG4gICAgfVxuXG4gICAgcmV0dXJuIGRlbGF5XG4gIH1cblxuICAvKipcbiAgICogQ2xvc2VzIGlkbGUgY2hlY2tlZC1pbiBjb25uZWN0aW9ucyB0aGF0IGhhdmUgZXhjZWVkZWQgdGhlIGNvbmZpZ3VyZWQgdGltZW91dC5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiBjb21wbGV0ZS5cbiAgICovXG4gIGFzeW5jIHJlYXBJZGxlQ29ubmVjdGlvbnMoKSB7XG4gICAgaWYgKHRoaXMuY29ubmVjdGlvbnMubGVuZ3RoID09PSAwKSByZXR1cm5cblxuICAgIGNvbnN0IGlkbGVUaW1lb3V0TWlsbGlzID0gdGhpcy5pZGxlVGltZW91dE1pbGxpcygpXG5cbiAgICBpZiAoaWRsZVRpbWVvdXRNaWxsaXMgPT09IG51bGwpIHJldHVyblxuXG4gICAgY29uc3Qge2V4cGlyZWRDb25uZWN0aW9ucywga2VwdENvbm5lY3Rpb25zfSA9IHRoaXMuY2xhc3NpZnlJZGxlQ29ubmVjdGlvbnNGb3JSZWFwaW5nKHtpZGxlVGltZW91dE1pbGxpcywgbm93OiBEYXRlLm5vdygpfSlcblxuICAgIHRoaXMuY29ubmVjdGlvbnMgPSBrZXB0Q29ubmVjdGlvbnNcbiAgICBhd2FpdCB0aGlzLmNsb3NlRXhwaXJlZElkbGVDb25uZWN0aW9ucyhleHBpcmVkQ29ubmVjdGlvbnMpXG4gICAgYXdhaXQgdGhpcy5hd2FpdEluZmxpZ2h0Q29ubmVjdGlvbkNsb3NlcygpXG4gICAgaWYgKHRoaXMuY29ubmVjdGlvbnMubGVuZ3RoID4gMCkgdGhpcy5zY2hlZHVsZUlkbGVDb25uZWN0aW9uUmVhcGVyKClcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGNsb3NlIGV4cGlyZWQgaWRsZSBjb25uZWN0aW9ucy5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdFtdfSBleHBpcmVkQ29ubmVjdGlvbnMgLSBDb25uZWN0aW9ucyB0byBjbG9zZS5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiBjbG9zZWQuXG4gICAqL1xuICBhc3luYyBjbG9zZUV4cGlyZWRJZGxlQ29ubmVjdGlvbnMoZXhwaXJlZENvbm5lY3Rpb25zKSB7XG4gICAgZm9yIChjb25zdCBjb25uZWN0aW9uIG9mIGV4cGlyZWRDb25uZWN0aW9ucykge1xuICAgICAgYXdhaXQgdGhpcy5jbG9zZUNvbm5lY3Rpb24oY29ubmVjdGlvbilcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBhd2FpdCBpbmZsaWdodCBjb25uZWN0aW9uIGNsb3Nlcy5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgb25jZSBpbi1mbGlnaHQgY29ubmVjdGlvbiBjbG9zZXMgc2V0dGxlLlxuICAgKi9cbiAgYXN5bmMgYXdhaXRJbmZsaWdodENvbm5lY3Rpb25DbG9zZXMoKSB7XG4gICAgaWYgKHRoaXMuaW5mbGlnaHRDb25uZWN0aW9uQ2xvc2VzLnNpemUgPiAwKSB7XG4gICAgICBhd2FpdCBQcm9taXNlLmFsbFNldHRsZWQoWy4uLnRoaXMuaW5mbGlnaHRDb25uZWN0aW9uQ2xvc2VzXSlcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBjbGFzc2lmeSBpZGxlIGNvbm5lY3Rpb25zIGZvciByZWFwaW5nLlxuICAgKiBAcGFyYW0ge3tpZGxlVGltZW91dE1pbGxpczogbnVtYmVyLCBub3c6IG51bWJlcn19IGFyZ3MgLSBSZWFwZXIgY2xhc3NpZmljYXRpb24gaW5wdXRzLlxuICAgKiBAcmV0dXJucyB7e2V4cGlyZWRDb25uZWN0aW9uczogaW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHRbXSwga2VwdENvbm5lY3Rpb25zOiBpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdFtdfX0gLSBDbGFzc2lmaWVkIGlkbGUgY29ubmVjdGlvbnMuXG4gICAqL1xuICBjbGFzc2lmeUlkbGVDb25uZWN0aW9uc0ZvclJlYXBpbmcoe2lkbGVUaW1lb3V0TWlsbGlzLCBub3d9KSB7XG4gICAgLyoqXG4gICAgICogS2VwdCBjb25uZWN0aW9ucy5cbiAgICAgIEB0eXBlIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdFtdfSAqL1xuICAgIGNvbnN0IGtlcHRDb25uZWN0aW9ucyA9IFtdXG4gICAgLyoqXG4gICAgICogRXhwaXJlZCBjb25uZWN0aW9ucy5cbiAgICAgIEB0eXBlIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdFtdfSAqL1xuICAgIGNvbnN0IGV4cGlyZWRDb25uZWN0aW9ucyA9IFtdXG5cbiAgICBmb3IgKGNvbnN0IGNvbm5lY3Rpb24gb2YgdGhpcy5jb25uZWN0aW9ucykge1xuICAgICAgdGhpcy5jbGFzc2lmeUlkbGVDb25uZWN0aW9uRm9yUmVhcGluZyh7Y29ubmVjdGlvbiwgZXhwaXJlZENvbm5lY3Rpb25zLCBpZGxlVGltZW91dE1pbGxpcywga2VwdENvbm5lY3Rpb25zLCBub3d9KVxuICAgIH1cblxuICAgIHJldHVybiB7ZXhwaXJlZENvbm5lY3Rpb25zLCBrZXB0Q29ubmVjdGlvbnN9XG4gIH1cblxuICAvKipcbiAgICogUnVucyBjbGFzc2lmeSBpZGxlIGNvbm5lY3Rpb24gZm9yIHJlYXBpbmcuXG4gICAqIEBwYXJhbSB7e2Nvbm5lY3Rpb246IGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0LCBleHBpcmVkQ29ubmVjdGlvbnM6IGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0W10sIGlkbGVUaW1lb3V0TWlsbGlzOiBudW1iZXIsIGtlcHRDb25uZWN0aW9uczogaW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHRbXSwgbm93OiBudW1iZXJ9fSBhcmdzIC0gQ2xhc3NpZmljYXRpb24gc3RhdGUuXG4gICAqIEByZXR1cm5zIHt2b2lkfVxuICAgKi9cbiAgY2xhc3NpZnlJZGxlQ29ubmVjdGlvbkZvclJlYXBpbmcoe2Nvbm5lY3Rpb24sIGV4cGlyZWRDb25uZWN0aW9ucywgaWRsZVRpbWVvdXRNaWxsaXMsIGtlcHRDb25uZWN0aW9ucywgbm93fSkge1xuICAgIGlmICh0aGlzLmNvbm5lY3Rpb25Jc0Nsb3NlZChjb25uZWN0aW9uKSkgcmV0dXJuXG4gICAgaWYgKHRoaXMuY29ubmVjdGlvbkhhc09wZW5UcmFuc2FjdGlvbihjb25uZWN0aW9uKSkge1xuICAgICAga2VwdENvbm5lY3Rpb25zLnB1c2goY29ubmVjdGlvbilcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIGNvbnN0IHRhcmdldCA9IHRoaXMuaWRsZUNvbm5lY3Rpb25FeHBpcmVkKHtjb25uZWN0aW9uLCBpZGxlVGltZW91dE1pbGxpcywgbm93fSkgPyBleHBpcmVkQ29ubmVjdGlvbnMgOiBrZXB0Q29ubmVjdGlvbnNcblxuICAgIHRhcmdldC5wdXNoKGNvbm5lY3Rpb24pXG4gIH1cblxuICAvKipcbiAgICogUnVucyBjb25uZWN0aW9uIGlzIGNsb3NlZC5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24gdG8gaW5zcGVjdC5cbiAgICogQHJldHVybnMge2Jvb2xlYW59IC0gV2hldGhlciB0aGUgY29ubmVjdGlvbiBpcyBtYXJrZWQgY2xvc2VkLlxuICAgKi9cbiAgY29ubmVjdGlvbklzQ2xvc2VkKGNvbm5lY3Rpb24pIHtcbiAgICBjb25zdCB0cmFja2VkQ29ubmVjdGlvbiA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICogTmFycm93cyB0aGUgcnVudGltZSB2YWx1ZSB0byB0aGUgZG9jdW1lbnRlZCB0eXBlLlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAdHlwZSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgJiB7W0NMT1NFRF9DT05ORUNUSU9OXT86IGJvb2xlYW59fSAqLyAoY29ubmVjdGlvbilcblxuICAgIHJldHVybiBCb29sZWFuKHRyYWNrZWRDb25uZWN0aW9uW0NMT1NFRF9DT05ORUNUSU9OXSlcbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGlkbGUgY29ubmVjdGlvbiBleHBpcmVkLlxuICAgKiBAcGFyYW0ge3tjb25uZWN0aW9uOiBpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdCwgaWRsZVRpbWVvdXRNaWxsaXM6IG51bWJlciwgbm93OiBudW1iZXJ9fSBhcmdzIC0gRXhwaXJ5IGlucHV0cy5cbiAgICogQHJldHVybnMge2Jvb2xlYW59IC0gV2hldGhlciB0aGUgaWRsZSBjb25uZWN0aW9uIGV4cGlyZWQuXG4gICAqL1xuICBpZGxlQ29ubmVjdGlvbkV4cGlyZWQoe2Nvbm5lY3Rpb24sIGlkbGVUaW1lb3V0TWlsbGlzLCBub3d9KSB7XG4gICAgY29uc3QgdHJhY2tlZENvbm5lY3Rpb24gPSAvKipcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqIE5hcnJvd3MgdGhlIHJ1bnRpbWUgdmFsdWUgdG8gdGhlIGRvY3VtZW50ZWQgdHlwZS5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQHR5cGUge2ltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0ICYge1tJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF0/OiBudW1iZXJ9fSAqLyAoY29ubmVjdGlvbilcbiAgICBjb25zdCBjaGVja2VkSW5BdCA9IHRyYWNrZWRDb25uZWN0aW9uW0lETEVfQ09OTkVDVElPTl9DSEVDS0VEX0lOX0FUXVxuXG4gICAgcmV0dXJuIHR5cGVvZiBjaGVja2VkSW5BdCA9PT0gXCJudW1iZXJcIiAmJiBub3cgLSBjaGVja2VkSW5BdCA+PSBpZGxlVGltZW91dE1pbGxpc1xuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY29ubmVjdGlvbiBoYXMgb3BlbiB0cmFuc2FjdGlvbi5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24gdG8gaW5zcGVjdC5cbiAgICogQHJldHVybnMge2Jvb2xlYW59IC0gV2hldGhlciB0aGUgY29ubmVjdGlvbiBoYXMgYW4gb3BlbiB0cmFuc2FjdGlvbi5cbiAgICovXG4gIGNvbm5lY3Rpb25IYXNPcGVuVHJhbnNhY3Rpb24oY29ubmVjdGlvbikge1xuICAgIHJldHVybiBjb25uZWN0aW9uLl90cmFuc2FjdGlvbnNDb3VudCA+IDBcbiAgfVxuXG4gIC8qKlxuICAgKiBSb2xscyBiYWNrIGFueSB0cmFuc2FjdGlvbiBhIHByZXZpb3VzIGhvbGRlciBsZWZ0IG9wZW4gYmVmb3JlIGEgY29ubmVjdGlvblxuICAgKiByZS1lbnRlcnMgdGhlIGlkbGUgcG9vbC4gQSBjb25uZWN0aW9uIHJldHVybmVkIHRvIHRoZSBwb29sIHdpdGggYW4gb3BlblxuICAgKiB0cmFuc2FjdGlvbiB3b3VsZCBvdGhlcndpc2UgYmUgaGFuZGVkIHRvIGFuIHVucmVsYXRlZCBjaGVja291dCwgd2hvc2VcbiAgICogc3RhcnRUcmFuc2FjdGlvbigpIHRoZW4gZmFpbHMgd2l0aCBcIkEgdHJhbnNhY3Rpb24gaXMgYWxyZWFkeSBydW5uaW5nXCIgYW5kXG4gICAqIHBvaXNvbnMgZXZlcnkgZm9sbG93aW5nIGNhbGxlciB0aGF0IHJldXNlcyBpdC5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9kcml2ZXJzL2Jhc2UuanNcIikuZGVmYXVsdH0gY29ubmVjdGlvbiAtIENvbm5lY3Rpb24gYmVpbmcgY2hlY2tlZCBpbi5cbiAgICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiB0aGUgY29ubmVjdGlvbiBob2xkcyBubyBvcGVuIHRyYW5zYWN0aW9uLlxuICAgKi9cbiAgYXN5bmMgcm9sbGJhY2tMZWZ0T3BlblRyYW5zYWN0aW9uKGNvbm5lY3Rpb24pIHtcbiAgICBpZiAoIXRoaXMuY29ubmVjdGlvbkhhc09wZW5UcmFuc2FjdGlvbihjb25uZWN0aW9uKSkgcmV0dXJuXG5cbiAgICB0aGlzLmxvZ2dlci53YXJuKCgpID0+IFtgUm9sbGluZyBiYWNrIGEgdHJhbnNhY3Rpb24gbGVmdCBvcGVuIG9uIGEgY29ubmVjdGlvbiBiZWluZyBjaGVja2VkIGluIChpZGVudGlmaWVyPSR7dGhpcy5pZGVudGlmaWVyfSkuYF0pXG5cbiAgICB3aGlsZSAodGhpcy5jb25uZWN0aW9uSGFzT3BlblRyYW5zYWN0aW9uKGNvbm5lY3Rpb24pKSB7XG4gICAgICBhd2FpdCBjb25uZWN0aW9uLnJvbGxiYWNrVHJhbnNhY3Rpb24oKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSdW5zIGNsb3NlIGNvbm5lY3Rpb24uXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHR9IGNvbm5lY3Rpb24gLSBDb25uZWN0aW9uIHRvIGNsb3NlLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgYXN5bmMgY2xvc2VDb25uZWN0aW9uKGNvbm5lY3Rpb24pIHtcbiAgICAvLyBJZGVtcG90ZW50OiBhIGZpcmUtYW5kLWZvcmdldCBzY2hlZHVsZWQgcmVhcCBhbmQgYW4gZXhwbGljaXQgcmVhcCBjYW4gYm90aFxuICAgIC8vIHRhcmdldCB0aGUgc2FtZSBjb25uZWN0aW9uLiBBd2FpdCB0aGUgaW4tZmxpZ2h0IGNsb3NlIGluc3RlYWQgb2YgY2xvc2luZ1xuICAgIC8vIHR3aWNlICh3aGljaCBjYW4gdGhyb3cgb24gdGhlIGRyaXZlcikgb3IgcmV0dXJuaW5nIHdoaWxlIHRoZSB1bmRlcmx5aW5nXG4gICAgLy8gaGFuZGxlIGlzIHN0aWxsIG9wZW4uXG4gICAgY29uc3QgZXhpc3RpbmdDbG9zZSA9IHRoaXMuY29ubmVjdGlvbkNsb3NlUHJvbWlzZXMuZ2V0KGNvbm5lY3Rpb24pXG5cbiAgICBpZiAoZXhpc3RpbmdDbG9zZSkge1xuICAgICAgcmV0dXJuIGF3YWl0IGV4aXN0aW5nQ2xvc2VcbiAgICB9XG5cbiAgICBjb25zdCB0cmFja2VkQ29ubmVjdGlvbiA9IC8qKlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICogTmFycm93cyB0aGUgcnVudGltZSB2YWx1ZSB0byB0aGUgZG9jdW1lbnRlZCB0eXBlLlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBAdHlwZSB7aW1wb3J0KFwiLi4vZHJpdmVycy9iYXNlLmpzXCIpLmRlZmF1bHQgJiB7W0NMT1NFRF9DT05ORUNUSU9OXT86IGJvb2xlYW4sIFtDT05ORUNUSU9OX0NIRUNLRURfT1VUX0FUXT86IG51bWJlciwgW0lETEVfQ09OTkVDVElPTl9DSEVDS0VEX0lOX0FUXT86IG51bWJlcn19ICovIChjb25uZWN0aW9uKVxuXG4gICAgdHJhY2tlZENvbm5lY3Rpb25bQ0xPU0VEX0NPTk5FQ1RJT05dID0gdHJ1ZVxuICAgIGRlbGV0ZSB0cmFja2VkQ29ubmVjdGlvbltDT05ORUNUSU9OX0NIRUNLRURfT1VUX0FUXVxuICAgIGRlbGV0ZSB0cmFja2VkQ29ubmVjdGlvbltJRExFX0NPTk5FQ1RJT05fQ0hFQ0tFRF9JTl9BVF1cblxuICAgIGNvbnN0IGNsb3NlUHJvbWlzZSA9IChhc3luYyAoKSA9PiB7XG4gICAgICBhd2FpdCB0cmFja2VkQ29ubmVjdGlvbi5jbG9zZSgpXG4gICAgfSkoKVxuXG4gICAgdGhpcy5jb25uZWN0aW9uQ2xvc2VQcm9taXNlcy5zZXQoY29ubmVjdGlvbiwgY2xvc2VQcm9taXNlKVxuICAgIHRoaXMuaW5mbGlnaHRDb25uZWN0aW9uQ2xvc2VzLmFkZChjbG9zZVByb21pc2UpXG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgY2xvc2VQcm9taXNlXG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHRoaXMuaW5mbGlnaHRDb25uZWN0aW9uQ2xvc2VzLmRlbGV0ZShjbG9zZVByb21pc2UpXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgY2xlYXIgaWRsZSBjb25uZWN0aW9uIHJlYXBlciB0aW1lci5cbiAgICBAcmV0dXJucyB7dm9pZH0gKi9cbiAgY2xlYXJJZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyKCkge1xuICAgIGlmICghdGhpcy5pZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyKSByZXR1cm5cblxuICAgIGNsZWFyVGltZW91dCh0aGlzLmlkbGVDb25uZWN0aW9uUmVhcGVyVGltZXIpXG4gICAgdGhpcy5pZGxlQ29ubmVjdGlvblJlYXBlclRpbWVyID0gdW5kZWZpbmVkXG4gIH1cblxuICAvKipcbiAgICogQ2xvc2VzIGFsbCBhY3RpdmUgYW5kIGNhY2hlZCBjb25uZWN0aW9ucyBmb3IgdGhpcyBwb29sLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn0gLSBSZXNvbHZlcyB3aGVuIGNvbXBsZXRlLlxuICAgKi9cbiAgYXN5bmMgY2xvc2VBbGwoKSB7XG4gICAgdGhpcy5jbGVhcklkbGVDb25uZWN0aW9uUmVhcGVyVGltZXIoKVxuICAgIHRoaXMucmVqZWN0UGVuZGluZ0NoZWNrb3V0cyhuZXcgRXJyb3IoXCJEYXRhYmFzZSBwb29sIHdhcyBjbG9zZWQgYmVmb3JlIGNoZWNrb3V0IGNvbXBsZXRlZC5cIikpXG5cbiAgICBjb25zdCBjb25uZWN0aW9ucyA9IG5ldyBTZXQoW1xuICAgICAgLi4udGhpcy5jb25uZWN0aW9ucyxcbiAgICAgIC4uLk9iamVjdC52YWx1ZXModGhpcy5jb25uZWN0aW9uc0luVXNlKSxcbiAgICAgIHRoaXMuZ2V0R2xvYmFsQ29ubmVjdGlvbkZvcklkZW50aWZpZXIoKSxcbiAgICAgIHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uXG4gICAgXS5maWx0ZXIoQm9vbGVhbikpXG5cbiAgICB0aGlzLmNvbm5lY3Rpb25zID0gW11cbiAgICB0aGlzLmNvbm5lY3Rpb25zSW5Vc2UgPSB7fVxuICAgIHRoaXMuX3Rlc3RTaGFyZWRDb25uZWN0aW9uID0gdW5kZWZpbmVkXG4gICAgdGhpcy5jbGVhckdsb2JhbENvbm5lY3Rpb25Gb3JJZGVudGlmaWVyKClcblxuICAgIGZvciAoY29uc3QgY29ubmVjdGlvbiBvZiBjb25uZWN0aW9ucykge1xuICAgICAgaWYgKCFjb25uZWN0aW9uKSBjb250aW51ZVxuXG4gICAgICBhd2FpdCB0aGlzLmNsb3NlQ29ubmVjdGlvbihjb25uZWN0aW9uKVxuICAgIH1cblxuICB9XG5cbiAgLyoqXG4gICAqIFJ1bnMgcmVqZWN0IHBlbmRpbmcgY2hlY2tvdXRzLlxuICAgKiBAcGFyYW0ge0Vycm9yfSBlcnJvciAtIEVycm9yIHRvIHJlamVjdCBwZW5kaW5nIGNoZWNrb3V0cyB3aXRoLlxuICAgKiBAcmV0dXJucyB7dm9pZH1cbiAgICovXG4gIHJlamVjdFBlbmRpbmdDaGVja291dHMoZXJyb3IpIHtcbiAgICBjb25zdCBwZW5kaW5nQ2hlY2tvdXRzID0gdGhpcy5wZW5kaW5nQ2hlY2tvdXRzXG5cbiAgICB0aGlzLnBlbmRpbmdDaGVja291dHMgPSBbXVxuXG4gICAgZm9yIChjb25zdCBjaGVja291dCBvZiBwZW5kaW5nQ2hlY2tvdXRzKSB7XG4gICAgICBjaGVja291dC5yZWplY3QoZXJyb3IpXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJlcGxhY2VzIGFsbCBnbG9iYWxseSByZWdpc3RlcmVkIGZhbGxiYWNrIGNvbm5lY3Rpb25zLlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGltcG9ydChcIi4uL2RyaXZlcnMvYmFzZS5qc1wiKS5kZWZhdWx0Pn0gW2Nvbm5lY3Rpb25zXSAtIENvbm5lY3Rpb25zLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uLy4uL2NvbmZpZ3VyYXRpb24uanNcIikuZGVmYXVsdH0gW2NvbmZpZ3VyYXRpb25dIC0gQ29uZmlndXJhdGlvbiBpbnN0YW5jZS5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgc3RhdGljIHNldEdsb2JhbENvbm5lY3Rpb25zKGNvbm5lY3Rpb25zLCBjb25maWd1cmF0aW9uKSB7XG4gICAgaWYgKCFjb25maWd1cmF0aW9uKSB7XG4gICAgICB0aGlzLmdsb2JhbENvbm5lY3Rpb25zID0gbmV3IFdlYWtNYXAoKVxuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgdGhpcy5nbG9iYWxDb25uZWN0aW9ucy5zZXQoY29uZmlndXJhdGlvbiwgY29ubmVjdGlvbnMgfHwge30pXG4gIH1cblxuICAvKipcbiAgICogQ2xlYXJzIGdsb2JhbGx5IHJlZ2lzdGVyZWQgZmFsbGJhY2sgY29ubmVjdGlvbnMgZm9yIGFsbCBjb25maWd1cmF0aW9ucyBvciBhIHNpbmdsZSBjb25maWd1cmF0aW9uLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uLy4uL2NvbmZpZ3VyYXRpb24uanNcIikuZGVmYXVsdH0gW2NvbmZpZ3VyYXRpb25dIC0gQ29uZmlndXJhdGlvbiBpbnN0YW5jZS5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgc3RhdGljIGNsZWFyR2xvYmFsQ29ubmVjdGlvbnMoY29uZmlndXJhdGlvbikge1xuICAgIGlmICghY29uZmlndXJhdGlvbikge1xuICAgICAgdGhpcy5nbG9iYWxDb25uZWN0aW9ucyA9IG5ldyBXZWFrTWFwKClcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIHRoaXMuZ2xvYmFsQ29ubmVjdGlvbnMuZGVsZXRlKGNvbmZpZ3VyYXRpb24pXG4gIH1cbn1cbiJdfQ==
|