testcafe 1.13.0 → 1.14.2
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/CHANGELOG.md +88 -10
- package/LICENSE +21 -21
- package/README.md +52 -48
- package/bin/testcafe-with-v8-flag-filter.js +0 -0
- package/lib/api/exportable-lib/index.js +50 -50
- package/lib/api/request-hooks/assert-type.js +7 -7
- package/lib/api/request-hooks/hook-method-names.js +11 -0
- package/lib/api/request-hooks/hook.js +33 -39
- package/lib/api/request-hooks/interfaces.js +3 -3
- package/lib/api/request-hooks/request-logger.js +113 -111
- package/lib/api/request-hooks/request-mock/create-request-mock.js +12 -0
- package/lib/api/request-hooks/request-mock/index.js +46 -0
- package/lib/api/request-hooks/request-mock.js +47 -48
- package/lib/api/structure/base-unit.js +11 -11
- package/lib/api/structure/fixture.js +75 -75
- package/lib/api/structure/interfaces.js +2 -2
- package/lib/api/structure/test-file.js +31 -31
- package/lib/api/structure/test-timeout.js +9 -9
- package/lib/api/structure/test.js +85 -85
- package/lib/api/structure/testing-unit.js +89 -89
- package/lib/api/structure/unit-type.js +9 -9
- package/lib/api/test-controller/assertion.js +88 -88
- package/lib/api/test-controller/execution-context.js +82 -82
- package/lib/api/test-controller/index.js +359 -319
- package/lib/api/test-controller/proxy.js +28 -28
- package/lib/api/test-page-url.js +60 -60
- package/lib/api/test-run-tracker.js +68 -68
- package/lib/api/wrap-test-function.js +49 -49
- package/lib/assertions/executor.js +74 -74
- package/lib/assertions/get-fn.js +46 -46
- package/lib/assertions/type.js +20 -20
- package/lib/assets/content-types.js +9 -9
- package/lib/assets/injectables.js +18 -18
- package/lib/browser/connection/command.js +10 -10
- package/lib/browser/connection/error-hints.js +9 -9
- package/lib/browser/connection/gateway.js +159 -159
- package/lib/browser/connection/get-hints.js +33 -33
- package/lib/browser/connection/heartbeat-status.js +8 -8
- package/lib/browser/connection/index.js +328 -328
- package/lib/browser/connection/remotes-queue.js +46 -46
- package/lib/browser/connection/service-routes.js +12 -12
- package/lib/browser/connection/status.js +12 -12
- package/lib/browser/interfaces.js +2 -2
- package/lib/browser/provider/built-in/dedicated/base.js +80 -80
- package/lib/browser/provider/built-in/dedicated/chrome/browser-client.js +204 -204
- package/lib/browser/provider/built-in/dedicated/chrome/build-chrome-args.js +17 -17
- package/lib/browser/provider/built-in/dedicated/chrome/config.js +110 -110
- package/lib/browser/provider/built-in/dedicated/chrome/create-temp-profile.js +45 -45
- package/lib/browser/provider/built-in/dedicated/chrome/elapsed-upperbounds.js +15 -15
- package/lib/browser/provider/built-in/dedicated/chrome/index.js +102 -102
- package/lib/browser/provider/built-in/dedicated/chrome/interfaces.js +2 -2
- package/lib/browser/provider/built-in/dedicated/chrome/local-chrome.js +24 -24
- package/lib/browser/provider/built-in/dedicated/chrome/runtime-info.js +29 -29
- package/lib/browser/provider/built-in/dedicated/edge/index.js +10 -10
- package/lib/browser/provider/built-in/dedicated/edge/runtime-info.js +29 -29
- package/lib/browser/provider/built-in/dedicated/firefox/config.js +33 -33
- package/lib/browser/provider/built-in/dedicated/firefox/create-temp-profile.js +78 -78
- package/lib/browser/provider/built-in/dedicated/firefox/index.js +73 -73
- package/lib/browser/provider/built-in/dedicated/firefox/local-firefox.js +36 -36
- package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/commands.js +13 -13
- package/lib/browser/provider/built-in/dedicated/firefox/marionette-client/index.js +200 -200
- package/lib/browser/provider/built-in/dedicated/firefox/runtime-info.js +17 -17
- package/lib/browser/provider/built-in/index.js +21 -21
- package/lib/browser/provider/built-in/locally-installed.js +30 -30
- package/lib/browser/provider/built-in/path.js +47 -47
- package/lib/browser/provider/built-in/remote.js +58 -58
- package/lib/browser/provider/index.js +303 -303
- package/lib/browser/provider/parse-provider-name.js +16 -16
- package/lib/browser/provider/plugin-host.js +121 -121
- package/lib/browser/provider/pool.js +115 -115
- package/lib/browser/provider/utils/argument-parsing.js +74 -74
- package/lib/browser/provider/utils/browser-starter.js +34 -34
- package/lib/browser/provider/utils/client-functions.js +24 -24
- package/lib/browser/provider/utils/get-maximized-headless-window-size.js +9 -9
- package/lib/cli/argument-parser.js +284 -279
- package/lib/cli/authentication-helper.js +35 -35
- package/lib/cli/cli.js +134 -134
- package/lib/cli/correct-browsers-and-sources.js +40 -40
- package/lib/cli/index.js +19 -19
- package/lib/cli/log.js +43 -43
- package/lib/cli/remotes-wizard.js +36 -36
- package/lib/cli/termination-handler.js +38 -38
- package/lib/client/automation/index.js +144 -46
- package/lib/client/automation/index.min.js +1 -1
- package/lib/client/browser/idle-page/index.html.mustache +35 -35
- package/lib/client/browser/idle-page/logo.svg +86 -86
- package/lib/client/browser/service-worker.js +4 -6
- package/lib/client/core/index.js +19 -13
- package/lib/client/core/index.min.js +1 -1
- package/lib/client/driver/index.js +1807 -451
- package/lib/client/driver/index.min.js +1 -1
- package/lib/client/driver/internal-properties.js +9 -9
- package/lib/client/test-run/iframe.js.mustache +17 -17
- package/lib/client/test-run/index.js.mustache +51 -51
- package/lib/client/ui/index.js +14 -12
- package/lib/client/ui/index.min.js +1 -1
- package/lib/client/ui/sprite.svg +42 -42
- package/lib/client-functions/builder-symbol.js +4 -4
- package/lib/client-functions/client-function-builder.js +156 -135
- package/lib/client-functions/replicator.js +61 -61
- package/lib/client-functions/return-single-prop-mode.js +8 -8
- package/lib/client-functions/selector-api-execution-mode.js +22 -0
- package/lib/client-functions/selectors/add-api.js +646 -588
- package/lib/client-functions/selectors/create-snapshot-methods.js +13 -13
- package/lib/client-functions/selectors/prepare-api-args.js +20 -20
- package/lib/client-functions/selectors/selector-attribute-filter.js +22 -22
- package/lib/client-functions/selectors/selector-builder.js +154 -150
- package/lib/client-functions/selectors/selector-text-filter.js +43 -43
- package/lib/client-functions/selectors/snapshot-properties.js +48 -48
- package/lib/client-functions/types.js +18 -18
- package/lib/compiler/babel/format-babel-produced-code.js +10 -10
- package/lib/compiler/babel/get-base-babel-options.js +11 -11
- package/lib/compiler/babel/load-libs.js +64 -62
- package/lib/compiler/babel/preset-stage-2.js +19 -19
- package/lib/compiler/compile-client-function.js +73 -73
- package/lib/compiler/compilers.js +33 -33
- package/lib/compiler/index.js +92 -92
- package/lib/compiler/interfaces.js +2 -2
- package/lib/compiler/test-file/api-based.js +146 -146
- package/lib/compiler/test-file/base.js +36 -36
- package/lib/compiler/test-file/exportble-lib-path.js +5 -5
- package/lib/compiler/test-file/formats/coffeescript/compiler.js +38 -38
- package/lib/compiler/test-file/formats/coffeescript/get-test-list.js +29 -29
- package/lib/compiler/test-file/formats/es-next/compiler.js +43 -43
- package/lib/compiler/test-file/formats/es-next/get-test-list.js +166 -166
- package/lib/compiler/test-file/formats/es-next/is-flow-code.js +7 -7
- package/lib/compiler/test-file/formats/raw.js +85 -85
- package/lib/compiler/test-file/formats/typescript/compiler.js +135 -135
- package/lib/compiler/test-file/formats/typescript/get-test-list.js +185 -185
- package/lib/compiler/test-file/test-file-parser-base.js +214 -214
- package/lib/configuration/configuration-base.js +165 -165
- package/lib/configuration/constants.js +9 -9
- package/lib/configuration/customizable-compilers.js +7 -7
- package/lib/configuration/default-values.js +51 -51
- package/lib/configuration/interfaces.js +3 -3
- package/lib/configuration/option-names.js +53 -53
- package/lib/configuration/option-source.js +9 -9
- package/lib/configuration/option.js +14 -14
- package/lib/configuration/quarantine-option-names.js +10 -0
- package/lib/configuration/run-option-names.js +26 -26
- package/lib/configuration/screenshot-option-names.js +10 -10
- package/lib/configuration/testcafe-configuration.js +164 -165
- package/lib/configuration/types.js +2 -2
- package/lib/configuration/typescript-configuration.js +62 -62
- package/lib/custom-client-scripts/assert-type.js +7 -7
- package/lib/custom-client-scripts/client-script-init.js +3 -3
- package/lib/custom-client-scripts/client-script.js +107 -108
- package/lib/custom-client-scripts/get-code.js +11 -11
- package/lib/custom-client-scripts/get-url.js +6 -6
- package/lib/custom-client-scripts/load.js +15 -15
- package/lib/custom-client-scripts/problematic-scripts.js +2 -2
- package/lib/custom-client-scripts/routing.js +36 -36
- package/lib/custom-client-scripts/utils.js +60 -60
- package/lib/embedding-utils.js +83 -83
- package/lib/errors/create-stack-filter.js +18 -18
- package/lib/errors/error-list.js +26 -26
- package/lib/errors/get-callsite.js +31 -31
- package/lib/errors/internal-modules-prefix.js +8 -8
- package/lib/errors/is-internal-stack-frame.js +46 -46
- package/lib/errors/process-test-fn-error.js +37 -37
- package/lib/errors/runtime/index.js +123 -123
- package/lib/errors/runtime/templates.js +116 -114
- package/lib/errors/runtime/type-assertions.js +112 -112
- package/lib/errors/stack-cleaning-hook.js +64 -64
- package/lib/errors/test-run/formattable-adapter.js +59 -59
- package/lib/errors/test-run/index.js +302 -302
- package/lib/errors/test-run/render-error-template.js +31 -31
- package/lib/errors/test-run/templates.js +93 -95
- package/lib/errors/test-run/utils.js +90 -85
- package/lib/errors/types.js +157 -155
- package/lib/index.js +81 -81
- package/lib/live/bootstrapper.js +44 -44
- package/lib/live/controller.js +107 -107
- package/lib/live/file-watcher/index.js +67 -67
- package/lib/live/file-watcher/modules-graph.js +58 -58
- package/lib/live/keyboard-observer.js +76 -76
- package/lib/live/logger/index.js +64 -64
- package/lib/live/test-run-controller.js +96 -96
- package/lib/live/test-run-state.js +6 -6
- package/lib/live/test-run.js +56 -56
- package/lib/live/test-runner.js +168 -168
- package/lib/load-assets.js +29 -29
- package/lib/notifications/add-rendered-warning.js +16 -16
- package/lib/notifications/debug-logger.js +78 -78
- package/lib/notifications/deprecated.js +24 -24
- package/lib/notifications/information-message.js +12 -12
- package/lib/notifications/warning-log.js +32 -31
- package/lib/notifications/warning-message.js +48 -48
- package/lib/reporter/command/command-formatter.js +120 -109
- package/lib/reporter/command/format-command.js +8 -8
- package/lib/reporter/command/interfaces.js +2 -2
- package/lib/reporter/index.js +319 -314
- package/lib/reporter/interfaces.js +2 -2
- package/lib/reporter/plugin-host.js +136 -137
- package/lib/reporter/plugin-methods.js +12 -12
- package/lib/role/index.js +74 -74
- package/lib/role/marker-symbol.js +7 -7
- package/lib/role/phase.js +9 -9
- package/lib/runner/bootstrapper.js +272 -272
- package/lib/runner/browser-job-result.js +9 -9
- package/lib/runner/browser-job.js +153 -123
- package/lib/runner/browser-set.js +114 -114
- package/lib/runner/fixture-hook-controller.js +85 -85
- package/lib/runner/index.js +450 -442
- package/lib/runner/interfaces.js +3 -3
- package/lib/runner/reporter-stream-controller.js +28 -28
- package/lib/runner/task/index.js +152 -133
- package/lib/runner/task/phase.js +9 -9
- package/lib/runner/test-run-controller.js +166 -185
- package/lib/runner/tested-app.js +73 -72
- package/lib/screenshots/capturer.js +141 -141
- package/lib/screenshots/constants.js +11 -11
- package/lib/screenshots/crop.js +111 -111
- package/lib/screenshots/default-extension.js +4 -4
- package/lib/screenshots/index.js +67 -67
- package/lib/screenshots/utils.js +39 -39
- package/lib/services/compiler/host.js +190 -106
- package/lib/services/compiler/io.js +9 -9
- package/lib/services/compiler/protocol.js +17 -16
- package/lib/services/compiler/service.js +225 -106
- package/lib/services/compiler/test-run-proxy.js +111 -73
- package/lib/services/interfaces.js +3 -0
- package/lib/services/process-title.js +8 -8
- package/lib/services/serialization/prepare-options.js +17 -17
- package/lib/services/serialization/replicator/create-replicator.js +27 -0
- package/lib/services/serialization/replicator/custom-error-transform.js +28 -0
- package/lib/services/serialization/test-structure.js +93 -77
- package/lib/services/utils/ipc/interfaces.js +30 -30
- package/lib/services/utils/ipc/io.js +108 -108
- package/lib/services/utils/ipc/message.js +76 -73
- package/lib/services/utils/ipc/packet.js +55 -55
- package/lib/services/utils/ipc/proxy.js +110 -118
- package/lib/services/utils/ipc/transport.js +64 -64
- package/lib/shared/errors/index.js +382 -382
- package/lib/shared/node-modules-folder-name.js +4 -4
- package/lib/test-run/bookmark.js +90 -90
- package/lib/test-run/browser-console-messages.js +73 -73
- package/lib/test-run/browser-manipulation-queue.js +92 -92
- package/lib/test-run/client-messages.js +9 -9
- package/lib/test-run/commands/actions.js +486 -425
- package/lib/test-run/commands/assertion.js +45 -45
- package/lib/test-run/commands/base.js +14 -14
- package/lib/test-run/commands/browser-manipulation.js +95 -95
- package/lib/test-run/commands/from-object.js +82 -82
- package/lib/test-run/commands/observation.js +61 -61
- package/lib/test-run/commands/options.js +231 -215
- package/lib/test-run/commands/service.js +54 -48
- package/lib/test-run/commands/type.js +65 -60
- package/lib/test-run/commands/utils.js +87 -87
- package/lib/test-run/commands/validations/argument.js +90 -90
- package/lib/test-run/commands/validations/factories.js +47 -47
- package/lib/test-run/commands/validations/initializers.js +44 -44
- package/lib/test-run/debug-log.js +32 -32
- package/lib/test-run/execute-js-expression.js +74 -74
- package/lib/test-run/index.js +823 -752
- package/lib/test-run/marker-symbol.js +7 -7
- package/lib/test-run/observed-callsites-storage.js +17 -17
- package/lib/test-run/phase.js +16 -16
- package/lib/test-run/session-controller.js +104 -104
- package/lib/testcafe.js +119 -114
- package/lib/utils/assignable.js +39 -39
- package/lib/utils/async-event-emitter.js +28 -28
- package/lib/utils/async-queue.js +14 -14
- package/lib/utils/browser-connection-timeouts.js +19 -19
- package/lib/utils/callsite.js +17 -17
- package/lib/utils/check-file-path.js +31 -31
- package/lib/utils/check-url.js +51 -51
- package/lib/utils/convert-to-best-fit-type.js +16 -16
- package/lib/utils/correct-file-path.js +21 -21
- package/lib/utils/define-lazy-property.js +13 -13
- package/lib/utils/delay.js +6 -6
- package/lib/utils/delegated-api.js +44 -44
- package/lib/utils/detect-display.js +6 -6
- package/lib/utils/detect-ffmpeg.js +44 -44
- package/lib/utils/diff/colors.js +29 -29
- package/lib/utils/diff/index.js +53 -53
- package/lib/utils/diff/util.js +24 -22
- package/lib/utils/diff.js +29 -29
- package/lib/utils/escape-user-agent.js +10 -10
- package/lib/utils/flag-list.js +17 -17
- package/lib/utils/get-any-key.js +8 -8
- package/lib/utils/get-browser.js +8 -8
- package/lib/utils/get-common-path.js +34 -34
- package/lib/utils/get-filter-fn.js +40 -40
- package/lib/utils/get-options/base.js +36 -36
- package/lib/utils/get-options/compiler.js +33 -33
- package/lib/utils/get-options/grep.js +15 -15
- package/lib/utils/get-options/index.js +21 -19
- package/lib/utils/get-options/meta.js +22 -22
- package/lib/utils/get-options/quarantine.js +92 -0
- package/lib/utils/get-options/screenshot.js +17 -17
- package/lib/utils/get-options/ssl.js +45 -45
- package/lib/utils/get-options/video.js +10 -10
- package/lib/utils/get-viewport-width.js +18 -18
- package/lib/utils/guard-time-execution.js +10 -10
- package/lib/utils/handle-errors.js +74 -74
- package/lib/utils/handle-tag-args.js +8 -8
- package/lib/utils/http.js +30 -30
- package/lib/utils/is-localhost.js +11 -11
- package/lib/utils/is-password-input.js +11 -0
- package/lib/utils/is-repl.js +12 -0
- package/lib/utils/is-window-in-iframe.js +6 -6
- package/lib/utils/limit-number.js +10 -10
- package/lib/utils/make-reg-exp.js +7 -7
- package/lib/utils/moment-loader.js +20 -20
- package/lib/utils/parse-file-list.js +71 -71
- package/lib/utils/parse-user-agent.js +55 -55
- package/lib/utils/path-pattern.js +114 -114
- package/lib/utils/prepare-reporters.js +30 -30
- package/lib/utils/prerender-callsite.js +18 -18
- package/lib/utils/process.js +119 -119
- package/lib/utils/promisified-functions.js +46 -46
- package/lib/utils/re-executable-promise.js +39 -39
- package/lib/utils/render-callsite-sync.js +29 -29
- package/lib/utils/render-template.js +10 -10
- package/lib/utils/reporter.js +30 -30
- package/lib/utils/resolve-path-relatively-cwd.js +7 -7
- package/lib/utils/string.js +105 -105
- package/lib/utils/temp-directory/cleanup-process/commands.js +7 -7
- package/lib/utils/temp-directory/cleanup-process/index.js +143 -143
- package/lib/utils/temp-directory/cleanup-process/worker.js +58 -58
- package/lib/utils/temp-directory/index.js +87 -87
- package/lib/utils/temp-directory/lockfile.js +56 -56
- package/lib/utils/thennable.js +7 -7
- package/lib/utils/timer.js +15 -15
- package/lib/utils/to-posix-path.js +8 -8
- package/lib/utils/types.js +2 -2
- package/lib/video-recorder/interfaces.js +2 -2
- package/lib/video-recorder/process.js +126 -126
- package/lib/video-recorder/recorder.js +136 -136
- package/lib/video-recorder/test-run-video-recorder.js +69 -69
- package/lib/video-recorder/videos.js +37 -37
- package/package.json +7 -7
- package/ts-defs/index.d.ts +109 -17
- package/ts-defs/selectors.d.ts +95 -7
- package/ts-defs/testcafe-scripts.d.ts +96 -8
|
@@ -1,186 +1,186 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
-
};
|
|
5
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.getTypeScriptTestListFromCode = exports.getTypeScriptTestList = void 0;
|
|
7
|
-
const typescript_1 = __importDefault(require("typescript"));
|
|
8
|
-
const lodash_1 = require("lodash");
|
|
9
|
-
const test_file_parser_base_1 = require("../../test-file-parser-base");
|
|
10
|
-
const typescript_configuration_1 = __importDefault(require("../../../../configuration/typescript-configuration"));
|
|
11
|
-
function replaceComments(code) {
|
|
12
|
-
return code.replace(/\/\*[\s\S]*?\*\/|([^\\:]|^)\/\/.*$/gm, match => {
|
|
13
|
-
const lastSymbol = match.indexOf('\n') > -1 ? '\n' : ' ';
|
|
14
|
-
return lodash_1.repeat(' ', match.length + lastSymbol);
|
|
15
|
-
});
|
|
16
|
-
}
|
|
17
|
-
class TypeScriptTestFileParser extends test_file_parser_base_1.TestFileParserBase {
|
|
18
|
-
constructor() {
|
|
19
|
-
super(typescript_1.default.SyntaxKind);
|
|
20
|
-
}
|
|
21
|
-
getComputedNameString({ pos, end }) {
|
|
22
|
-
const templatePos = this.getLocationByOffsets(pos, end);
|
|
23
|
-
return test_file_parser_base_1.TestFileParserBase.formatComputedName(templatePos.loc.start.line);
|
|
24
|
-
}
|
|
25
|
-
getTokenType(token) {
|
|
26
|
-
return token.kind;
|
|
27
|
-
}
|
|
28
|
-
getCalleeToken(token) {
|
|
29
|
-
return token.expression;
|
|
30
|
-
}
|
|
31
|
-
getMemberFnName(token) {
|
|
32
|
-
return token.expression.name.text;
|
|
33
|
-
}
|
|
34
|
-
getKeyValue(prop) {
|
|
35
|
-
const { name, initializer } = prop;
|
|
36
|
-
return {
|
|
37
|
-
key: name.text,
|
|
38
|
-
value: this.getStringValue(initializer)
|
|
39
|
-
};
|
|
40
|
-
}
|
|
41
|
-
getFixedStartOffset(start) {
|
|
42
|
-
let fixedStartOffset = start;
|
|
43
|
-
while (/\s/.test(this.codeWithoutComments[fixedStartOffset]))
|
|
44
|
-
++fixedStartOffset;
|
|
45
|
-
return fixedStartOffset;
|
|
46
|
-
}
|
|
47
|
-
getLocationByOffsets(start, end) {
|
|
48
|
-
const fixedStart = this.getFixedStartOffset(start);
|
|
49
|
-
const codeArr = this.codeArr;
|
|
50
|
-
const loc = { start: null, end: null };
|
|
51
|
-
let line = codeArr[0];
|
|
52
|
-
let startTmp = fixedStart;
|
|
53
|
-
let endTmp = end;
|
|
54
|
-
for (let lineNumber = 1; lineNumber <= codeArr.length; ++lineNumber, line = codeArr[lineNumber - 1]) {
|
|
55
|
-
startTmp -= line.length + 1;
|
|
56
|
-
endTmp -= line.length + 1;
|
|
57
|
-
if (startTmp < 0 && !loc.start)
|
|
58
|
-
loc.start = { line: lineNumber, column: line.length + startTmp + 1 };
|
|
59
|
-
if (endTmp <= 0 || lineNumber === codeArr.length - 1) {
|
|
60
|
-
loc.end = { line: lineNumber, column: line.length + endTmp + 1 };
|
|
61
|
-
break;
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
return { loc, start: fixedStart, end };
|
|
65
|
-
}
|
|
66
|
-
getRValue(token) {
|
|
67
|
-
return token.declarationList.declarations[0].initializer;
|
|
68
|
-
}
|
|
69
|
-
getStringValue(token) {
|
|
70
|
-
const stringTypes = [this.tokenType.StringLiteral, this.tokenType.TemplateExpression];
|
|
71
|
-
if (stringTypes.indexOf(token.kind) > -1 || token.text && token.kind !== this.tokenType.NumericLiteral)
|
|
72
|
-
return this.formatFnArg(token);
|
|
73
|
-
return null;
|
|
74
|
-
}
|
|
75
|
-
isAsyncFn(token) {
|
|
76
|
-
const isGeneratorFn = !!token.asteriskToken;
|
|
77
|
-
const isAsyncFn = token.modifiers &&
|
|
78
|
-
token.modifiers.some(modifier => modifier.kind === this.tokenType.AsyncKeyword);
|
|
79
|
-
return isGeneratorFn || isAsyncFn;
|
|
80
|
-
}
|
|
81
|
-
getFunctionBody(token) {
|
|
82
|
-
return token.body.statements;
|
|
83
|
-
}
|
|
84
|
-
formatFnData(name, value, token, meta = [{}]) {
|
|
85
|
-
const loc = this.getLocationByOffsets(token.pos, token.end);
|
|
86
|
-
return {
|
|
87
|
-
fnName: name,
|
|
88
|
-
value: value,
|
|
89
|
-
loc: loc.loc,
|
|
90
|
-
start: loc.start,
|
|
91
|
-
end: loc.end,
|
|
92
|
-
meta: lodash_1.merge({}, ...meta)
|
|
93
|
-
};
|
|
94
|
-
}
|
|
95
|
-
analyzeMemberExp(token) {
|
|
96
|
-
let exp = token;
|
|
97
|
-
const tokenType = this.tokenType;
|
|
98
|
-
const callStack = [exp];
|
|
99
|
-
while (exp.kind !== this.tokenType.Identifier) {
|
|
100
|
-
exp = exp.expression || exp.tag;
|
|
101
|
-
callStack.push(exp);
|
|
102
|
-
}
|
|
103
|
-
const meta = this.getMetaInfo(callStack.slice());
|
|
104
|
-
if (exp && this.isApiFn(exp.text)) {
|
|
105
|
-
let parentExp = callStack.pop();
|
|
106
|
-
while (parentExp) {
|
|
107
|
-
if (parentExp.kind === tokenType.CallExpression && parentExp.expression) {
|
|
108
|
-
const calleeType = parentExp.expression.kind;
|
|
109
|
-
const calleeMemberFn = calleeType === tokenType.PropertyAccessExpression &&
|
|
110
|
-
parentExp.expression.name.text;
|
|
111
|
-
if (this.checkExpDefineTargetName(calleeType, calleeMemberFn))
|
|
112
|
-
return this.formatFnData(exp.text, this.formatFnArg(parentExp.arguments[0]), token, meta);
|
|
113
|
-
}
|
|
114
|
-
if (parentExp.kind === tokenType.TaggedTemplateExpression && parentExp.tag) {
|
|
115
|
-
const tagType = parentExp.tag.kind;
|
|
116
|
-
const tagMemberFn = tagType === tokenType.PropertyAccessExpression && parentExp.tag.name.text;
|
|
117
|
-
if (this.checkExpDefineTargetName(tagType, tagMemberFn))
|
|
118
|
-
return this.formatFnData(exp.text, this.formatFnArg(parentExp), token, meta);
|
|
119
|
-
}
|
|
120
|
-
parentExp = callStack.pop();
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
return null;
|
|
124
|
-
}
|
|
125
|
-
formatFnArg(arg) {
|
|
126
|
-
if (arg.templateSpans)
|
|
127
|
-
return this.getComputedNameString({ pos: arg.pos, end: arg.end });
|
|
128
|
-
if (arg.head)
|
|
129
|
-
return this.getComputedNameString({ pos: arg.template.pos, end: arg.template.end });
|
|
130
|
-
if (arg.template)
|
|
131
|
-
return arg.template.text || this.getComputedNameString({ pos: arg.template.pos, end: arg.template.end });
|
|
132
|
-
if (arg.kind === this.tokenType.Identifier)
|
|
133
|
-
return this.getComputedNameString({ pos: arg.pos, end: arg.end });
|
|
134
|
-
if (arg.text && arg.kind !== this.tokenType.NumericLiteral)
|
|
135
|
-
return arg.text;
|
|
136
|
-
if (arg.kind === this.tokenType.TypeAssertionExpression)
|
|
137
|
-
return this.formatFnArg(arg.expression);
|
|
138
|
-
return null;
|
|
139
|
-
}
|
|
140
|
-
getFnCall(token) {
|
|
141
|
-
if (this.isApiFn(token.expression.text))
|
|
142
|
-
return this.formatFnData(token.expression.text, this.formatFnArg(token.arguments[0]), token);
|
|
143
|
-
return null;
|
|
144
|
-
}
|
|
145
|
-
getTaggedTemplateExp(token) {
|
|
146
|
-
if (this.isApiFn(token.tag.text))
|
|
147
|
-
return this.formatFnData(token.tag.text, this.formatFnArg(token), token);
|
|
148
|
-
return null;
|
|
149
|
-
}
|
|
150
|
-
analyzeFnCall(token) {
|
|
151
|
-
const tokenType = this.tokenType;
|
|
152
|
-
if (token.kind === tokenType.PropertyAccessExpression)
|
|
153
|
-
return this.analyzeMemberExp(token);
|
|
154
|
-
if (token.kind === tokenType.CallExpression) {
|
|
155
|
-
const expKind = token.expression.kind;
|
|
156
|
-
if (expKind === tokenType.PropertyAccessExpression || expKind === tokenType.CallExpression)
|
|
157
|
-
return this.analyzeMemberExp(token);
|
|
158
|
-
if (expKind === tokenType.ParenthesizedExpression)
|
|
159
|
-
return this.analyzeFnCall(token.expression.expression);
|
|
160
|
-
return this.getFnCall(token);
|
|
161
|
-
}
|
|
162
|
-
if (token.kind === tokenType.FunctionExpression || token.kind === tokenType.ArrowFunction)
|
|
163
|
-
return this.collectTestCafeCalls(this.getFunctionBody(token));
|
|
164
|
-
if (token.kind === tokenType.TaggedTemplateExpression) {
|
|
165
|
-
if (token.tag.kind === tokenType.PropertyAccessExpression || token.tag.kind === tokenType.CallExpression)
|
|
166
|
-
return this.analyzeMemberExp(token);
|
|
167
|
-
return this.getTaggedTemplateExp(token);
|
|
168
|
-
}
|
|
169
|
-
return null;
|
|
170
|
-
}
|
|
171
|
-
parse(code) {
|
|
172
|
-
//NOTE: TypeScript calculates start position of a token incorrectly
|
|
173
|
-
//It doesn't consider spaces and comments between the last token and the current token.
|
|
174
|
-
//So we replace comments with space symbols to calculate fixed position.
|
|
175
|
-
//We just increment position until we meet non whitespace characters
|
|
176
|
-
this.codeArr = code.split('\n');
|
|
177
|
-
this.codeWithoutComments = replaceComments(code);
|
|
178
|
-
const tsConfig = new typescript_configuration_1.default();
|
|
179
|
-
const sourceFile = typescript_1.default.createSourceFile('', code, tsConfig.getOptions(), true);
|
|
180
|
-
return this.analyze(sourceFile.statements);
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
const parser = new TypeScriptTestFileParser();
|
|
184
|
-
exports.getTypeScriptTestList = parser.getTestList.bind(parser);
|
|
185
|
-
exports.getTypeScriptTestListFromCode = parser.getTestListFromCode.bind(parser);
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.getTypeScriptTestListFromCode = exports.getTypeScriptTestList = void 0;
|
|
7
|
+
const typescript_1 = __importDefault(require("typescript"));
|
|
8
|
+
const lodash_1 = require("lodash");
|
|
9
|
+
const test_file_parser_base_1 = require("../../test-file-parser-base");
|
|
10
|
+
const typescript_configuration_1 = __importDefault(require("../../../../configuration/typescript-configuration"));
|
|
11
|
+
function replaceComments(code) {
|
|
12
|
+
return code.replace(/\/\*[\s\S]*?\*\/|([^\\:]|^)\/\/.*$/gm, match => {
|
|
13
|
+
const lastSymbol = match.indexOf('\n') > -1 ? '\n' : ' ';
|
|
14
|
+
return lodash_1.repeat(' ', match.length + lastSymbol);
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
class TypeScriptTestFileParser extends test_file_parser_base_1.TestFileParserBase {
|
|
18
|
+
constructor() {
|
|
19
|
+
super(typescript_1.default.SyntaxKind);
|
|
20
|
+
}
|
|
21
|
+
getComputedNameString({ pos, end }) {
|
|
22
|
+
const templatePos = this.getLocationByOffsets(pos, end);
|
|
23
|
+
return test_file_parser_base_1.TestFileParserBase.formatComputedName(templatePos.loc.start.line);
|
|
24
|
+
}
|
|
25
|
+
getTokenType(token) {
|
|
26
|
+
return token.kind;
|
|
27
|
+
}
|
|
28
|
+
getCalleeToken(token) {
|
|
29
|
+
return token.expression;
|
|
30
|
+
}
|
|
31
|
+
getMemberFnName(token) {
|
|
32
|
+
return token.expression.name.text;
|
|
33
|
+
}
|
|
34
|
+
getKeyValue(prop) {
|
|
35
|
+
const { name, initializer } = prop;
|
|
36
|
+
return {
|
|
37
|
+
key: name.text,
|
|
38
|
+
value: this.getStringValue(initializer)
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
getFixedStartOffset(start) {
|
|
42
|
+
let fixedStartOffset = start;
|
|
43
|
+
while (/\s/.test(this.codeWithoutComments[fixedStartOffset]))
|
|
44
|
+
++fixedStartOffset;
|
|
45
|
+
return fixedStartOffset;
|
|
46
|
+
}
|
|
47
|
+
getLocationByOffsets(start, end) {
|
|
48
|
+
const fixedStart = this.getFixedStartOffset(start);
|
|
49
|
+
const codeArr = this.codeArr;
|
|
50
|
+
const loc = { start: null, end: null };
|
|
51
|
+
let line = codeArr[0];
|
|
52
|
+
let startTmp = fixedStart;
|
|
53
|
+
let endTmp = end;
|
|
54
|
+
for (let lineNumber = 1; lineNumber <= codeArr.length; ++lineNumber, line = codeArr[lineNumber - 1]) {
|
|
55
|
+
startTmp -= line.length + 1;
|
|
56
|
+
endTmp -= line.length + 1;
|
|
57
|
+
if (startTmp < 0 && !loc.start)
|
|
58
|
+
loc.start = { line: lineNumber, column: line.length + startTmp + 1 };
|
|
59
|
+
if (endTmp <= 0 || lineNumber === codeArr.length - 1) {
|
|
60
|
+
loc.end = { line: lineNumber, column: line.length + endTmp + 1 };
|
|
61
|
+
break;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
return { loc, start: fixedStart, end };
|
|
65
|
+
}
|
|
66
|
+
getRValue(token) {
|
|
67
|
+
return token.declarationList.declarations[0].initializer;
|
|
68
|
+
}
|
|
69
|
+
getStringValue(token) {
|
|
70
|
+
const stringTypes = [this.tokenType.StringLiteral, this.tokenType.TemplateExpression];
|
|
71
|
+
if (stringTypes.indexOf(token.kind) > -1 || token.text && token.kind !== this.tokenType.NumericLiteral)
|
|
72
|
+
return this.formatFnArg(token);
|
|
73
|
+
return null;
|
|
74
|
+
}
|
|
75
|
+
isAsyncFn(token) {
|
|
76
|
+
const isGeneratorFn = !!token.asteriskToken;
|
|
77
|
+
const isAsyncFn = token.modifiers &&
|
|
78
|
+
token.modifiers.some(modifier => modifier.kind === this.tokenType.AsyncKeyword);
|
|
79
|
+
return isGeneratorFn || isAsyncFn;
|
|
80
|
+
}
|
|
81
|
+
getFunctionBody(token) {
|
|
82
|
+
return token.body.statements;
|
|
83
|
+
}
|
|
84
|
+
formatFnData(name, value, token, meta = [{}]) {
|
|
85
|
+
const loc = this.getLocationByOffsets(token.pos, token.end);
|
|
86
|
+
return {
|
|
87
|
+
fnName: name,
|
|
88
|
+
value: value,
|
|
89
|
+
loc: loc.loc,
|
|
90
|
+
start: loc.start,
|
|
91
|
+
end: loc.end,
|
|
92
|
+
meta: lodash_1.merge({}, ...meta)
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
analyzeMemberExp(token) {
|
|
96
|
+
let exp = token;
|
|
97
|
+
const tokenType = this.tokenType;
|
|
98
|
+
const callStack = [exp];
|
|
99
|
+
while (exp.kind !== this.tokenType.Identifier) {
|
|
100
|
+
exp = exp.expression || exp.tag;
|
|
101
|
+
callStack.push(exp);
|
|
102
|
+
}
|
|
103
|
+
const meta = this.getMetaInfo(callStack.slice());
|
|
104
|
+
if (exp && this.isApiFn(exp.text)) {
|
|
105
|
+
let parentExp = callStack.pop();
|
|
106
|
+
while (parentExp) {
|
|
107
|
+
if (parentExp.kind === tokenType.CallExpression && parentExp.expression) {
|
|
108
|
+
const calleeType = parentExp.expression.kind;
|
|
109
|
+
const calleeMemberFn = calleeType === tokenType.PropertyAccessExpression &&
|
|
110
|
+
parentExp.expression.name.text;
|
|
111
|
+
if (this.checkExpDefineTargetName(calleeType, calleeMemberFn))
|
|
112
|
+
return this.formatFnData(exp.text, this.formatFnArg(parentExp.arguments[0]), token, meta);
|
|
113
|
+
}
|
|
114
|
+
if (parentExp.kind === tokenType.TaggedTemplateExpression && parentExp.tag) {
|
|
115
|
+
const tagType = parentExp.tag.kind;
|
|
116
|
+
const tagMemberFn = tagType === tokenType.PropertyAccessExpression && parentExp.tag.name.text;
|
|
117
|
+
if (this.checkExpDefineTargetName(tagType, tagMemberFn))
|
|
118
|
+
return this.formatFnData(exp.text, this.formatFnArg(parentExp), token, meta);
|
|
119
|
+
}
|
|
120
|
+
parentExp = callStack.pop();
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
return null;
|
|
124
|
+
}
|
|
125
|
+
formatFnArg(arg) {
|
|
126
|
+
if (arg.templateSpans)
|
|
127
|
+
return this.getComputedNameString({ pos: arg.pos, end: arg.end });
|
|
128
|
+
if (arg.head)
|
|
129
|
+
return this.getComputedNameString({ pos: arg.template.pos, end: arg.template.end });
|
|
130
|
+
if (arg.template)
|
|
131
|
+
return arg.template.text || this.getComputedNameString({ pos: arg.template.pos, end: arg.template.end });
|
|
132
|
+
if (arg.kind === this.tokenType.Identifier)
|
|
133
|
+
return this.getComputedNameString({ pos: arg.pos, end: arg.end });
|
|
134
|
+
if (arg.text && arg.kind !== this.tokenType.NumericLiteral)
|
|
135
|
+
return arg.text;
|
|
136
|
+
if (arg.kind === this.tokenType.TypeAssertionExpression)
|
|
137
|
+
return this.formatFnArg(arg.expression);
|
|
138
|
+
return null;
|
|
139
|
+
}
|
|
140
|
+
getFnCall(token) {
|
|
141
|
+
if (this.isApiFn(token.expression.text))
|
|
142
|
+
return this.formatFnData(token.expression.text, this.formatFnArg(token.arguments[0]), token);
|
|
143
|
+
return null;
|
|
144
|
+
}
|
|
145
|
+
getTaggedTemplateExp(token) {
|
|
146
|
+
if (this.isApiFn(token.tag.text))
|
|
147
|
+
return this.formatFnData(token.tag.text, this.formatFnArg(token), token);
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
analyzeFnCall(token) {
|
|
151
|
+
const tokenType = this.tokenType;
|
|
152
|
+
if (token.kind === tokenType.PropertyAccessExpression)
|
|
153
|
+
return this.analyzeMemberExp(token);
|
|
154
|
+
if (token.kind === tokenType.CallExpression) {
|
|
155
|
+
const expKind = token.expression.kind;
|
|
156
|
+
if (expKind === tokenType.PropertyAccessExpression || expKind === tokenType.CallExpression)
|
|
157
|
+
return this.analyzeMemberExp(token);
|
|
158
|
+
if (expKind === tokenType.ParenthesizedExpression)
|
|
159
|
+
return this.analyzeFnCall(token.expression.expression);
|
|
160
|
+
return this.getFnCall(token);
|
|
161
|
+
}
|
|
162
|
+
if (token.kind === tokenType.FunctionExpression || token.kind === tokenType.ArrowFunction)
|
|
163
|
+
return this.collectTestCafeCalls(this.getFunctionBody(token));
|
|
164
|
+
if (token.kind === tokenType.TaggedTemplateExpression) {
|
|
165
|
+
if (token.tag.kind === tokenType.PropertyAccessExpression || token.tag.kind === tokenType.CallExpression)
|
|
166
|
+
return this.analyzeMemberExp(token);
|
|
167
|
+
return this.getTaggedTemplateExp(token);
|
|
168
|
+
}
|
|
169
|
+
return null;
|
|
170
|
+
}
|
|
171
|
+
parse(code) {
|
|
172
|
+
//NOTE: TypeScript calculates start position of a token incorrectly
|
|
173
|
+
//It doesn't consider spaces and comments between the last token and the current token.
|
|
174
|
+
//So we replace comments with space symbols to calculate fixed position.
|
|
175
|
+
//We just increment position until we meet non whitespace characters
|
|
176
|
+
this.codeArr = code.split('\n');
|
|
177
|
+
this.codeWithoutComments = replaceComments(code);
|
|
178
|
+
const tsConfig = new typescript_configuration_1.default();
|
|
179
|
+
const sourceFile = typescript_1.default.createSourceFile('', code, tsConfig.getOptions(), true);
|
|
180
|
+
return this.analyze(sourceFile.statements);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
const parser = new TypeScriptTestFileParser();
|
|
184
|
+
exports.getTypeScriptTestList = parser.getTestList.bind(parser);
|
|
185
|
+
exports.getTypeScriptTestListFromCode = parser.getTestListFromCode.bind(parser);
|
|
186
186
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LXRlc3QtbGlzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9jb21waWxlci90ZXN0LWZpbGUvZm9ybWF0cy90eXBlc2NyaXB0L2dldC10ZXN0LWxpc3QuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUEsNERBQTRCO0FBQzVCLG1DQUF1QztBQUN2Qyx1RUFBaUU7QUFDakUsa0hBQXlGO0FBRXpGLFNBQVMsZUFBZSxDQUFFLElBQUk7SUFDMUIsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLHNDQUFzQyxFQUFFLEtBQUssQ0FBQyxFQUFFO1FBQ2hFLE1BQU0sVUFBVSxHQUFHLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDO1FBRXpELE9BQU8sZUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsTUFBTSxHQUFHLFVBQVUsQ0FBQyxDQUFDO0lBQ2xELENBQUMsQ0FBQyxDQUFDO0FBQ1AsQ0FBQztBQUVELE1BQU0sd0JBQXlCLFNBQVEsMENBQWtCO0lBQ3JEO1FBQ0ksS0FBSyxDQUFDLG9CQUFFLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDekIsQ0FBQztJQUVELHFCQUFxQixDQUFFLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRTtRQUMvQixNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsb0JBQW9CLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBRXhELE9BQU8sMENBQWtCLENBQUMsa0JBQWtCLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDN0UsQ0FBQztJQUVELFlBQVksQ0FBRSxLQUFLO1FBQ2YsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDO0lBQ3RCLENBQUM7SUFFRCxjQUFjLENBQUUsS0FBSztRQUNqQixPQUFPLEtBQUss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SUFDaEIsQ0FBQztJQUVELEtBQUssQ0FBRSxJQUFJO1FBQ1AsbUVBQW1FO1FBQ25FLHVGQUF1RjtRQUN2Rix3RUFBd0U7UUFDeEUsb0VBQW9FO1FBQ3BFLElBQUksQ0FBQyxPQUFPLEdBQWUsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM1QyxJQUFJLENBQUMsbUJBQW1CLEdBQUcsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRWpELE1BQU0sUUFBUSxHQUFLLElBQUksa0NBQXVCLEVBQUUsQ0FBQztRQUNqRCxNQUFNLFVBQVUsR0FBRyxvQkFBRSxDQUFDLGdCQUFnQixDQUFDLEVBQUUsRUFBRSxJQUFJLEVBQUUsUUFBUSxDQUFDLFVBQVUsRUFBRSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBRTlFLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDL0MsQ0FBQztDQUNKO0FBRUQsTUFBTSxNQUFNLEdBQUcsSUFBSSx3QkFBd0IsRUFBRSxDQUFDO0FBRWpDLFFBQUEscUJBQXFCLEdBQVcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7QUFDaEUsUUFBQSw2QkFBNkIsR0FBRyxNQUFNLENBQUMsbUJBQW1CLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHRzIGZyb20gJ3R5cGVzY3JpcHQnO1xuaW1wb3J0IHsgcmVwZWF0LCBtZXJnZSB9IGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBUZXN0RmlsZVBhcnNlckJhc2UgfSBmcm9tICcuLi8uLi90ZXN0LWZpbGUtcGFyc2VyLWJhc2UnO1xuaW1wb3J0IFR5cGVzY3JpcHRDb25maWd1cmF0aW9uIGZyb20gJy4uLy4uLy4uLy4uL2NvbmZpZ3VyYXRpb24vdHlwZXNjcmlwdC1jb25maWd1cmF0aW9uJztcblxuZnVuY3Rpb24gcmVwbGFjZUNvbW1lbnRzIChjb2RlKSB7XG4gICAgcmV0dXJuIGNvZGUucmVwbGFjZSgvXFwvXFwqW1xcc1xcU10qP1xcKlxcL3woW15cXFxcOl18XilcXC9cXC8uKiQvZ20sIG1hdGNoID0+IHtcbiAgICAgICAgY29uc3QgbGFzdFN5bWJvbCA9IG1hdGNoLmluZGV4T2YoJ1xcbicpID4gLTEgPyAnXFxuJyA6ICcgJztcblxuICAgICAgICByZXR1cm4gcmVwZWF0KCcgJywgbWF0Y2gubGVuZ3RoICsgbGFzdFN5bWJvbCk7XG4gICAgfSk7XG59XG5cbmNsYXNzIFR5cGVTY3JpcHRUZXN0RmlsZVBhcnNlciBleHRlbmRzIFRlc3RGaWxlUGFyc2VyQmFzZSB7XG4gICAgY29uc3RydWN0b3IgKCkge1xuICAgICAgICBzdXBlcih0cy5TeW50YXhLaW5kKTtcbiAgICB9XG5cbiAgICBnZXRDb21wdXRlZE5hbWVTdHJpbmcgKHsgcG9zLCBlbmQgfSkge1xuICAgICAgICBjb25zdCB0ZW1wbGF0ZVBvcyA9IHRoaXMuZ2V0TG9jYXRpb25CeU9mZnNldHMocG9zLCBlbmQpO1xuXG4gICAgICAgIHJldHVybiBUZXN0RmlsZVBhcnNlckJhc2UuZm9ybWF0Q29tcHV0ZWROYW1lKHRlbXBsYXRlUG9zLmxvYy5zdGFydC5saW5lKTtcbiAgICB9XG5cbiAgICBnZXRUb2tlblR5cGUgKHRva2VuKSB7XG4gICAgICAgIHJldHVybiB0b2tlbi5raW5kO1xuICAgIH1cblxuICAgIGdldENhbGxlZVRva2VuICh0b2tlbikge1xuICAgICAgICByZXR1cm4gdG9rZW4uZXhwcmVzc2lvbjtcbiAgICB9XG5cbiAgICBnZXRNZW1iZXJGbk5hbWUgKHRva2VuKSB7XG4gICAgICAgIHJldHVybiB0b2tlbi5leHByZXNzaW9uLm5hbWUudGV4dDtcbiAgICB9XG5cbiAgICBnZXRLZXlWYWx1ZSAocHJvcCkge1xuICAgICAgICBjb25zdCB7IG5hbWUsIGluaXRpYWxpemVyIH0gPSBwcm9wO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBrZXk6ICAgbmFtZS50ZXh0LFxuICAgICAgICAgICAgdmFsdWU6IHRoaXMuZ2V0U3RyaW5nVmFsdWUoaW5pdGlhbGl6ZXIpXG4gICAgICAgIH07XG4gICAgfVxuXG4gICAgZ2V0Rml4ZWRTdGFydE9mZnNldCAoc3RhcnQpIHtcbiAgICAgICAgbGV0IGZpeGVkU3RhcnRPZmZzZXQgPSBzdGFydDtcblxuICAgICAgICB3aGlsZSAoL1xccy8udGVzdCh0aGlzLmNvZGVXaXRob3V0Q29tbWVudHNbZml4ZWRTdGFydE9mZnNldF0pKVxuICAgICAgICAgICAgKytmaXhlZFN0YXJ0T2Zmc2V0O1xuXG4gICAgICAgIHJldHVybiBmaXhlZFN0YXJ0T2Zmc2V0O1xuICAgIH1cblxuICAgIGdldExvY2F0aW9uQnlPZmZzZXRzIChzdGFydCwgZW5kKSB7XG4gICAgICAgIGNvbnN0IGZpeGVkU3RhcnQgPSB0aGlzLmdldEZpeGVkU3RhcnRPZmZzZXQoc3RhcnQpO1xuICAgICAgICBjb25zdCBjb2RlQXJyICAgID0gdGhpcy5jb2RlQXJyO1xuICAgICAgICBjb25zdCBsb2MgICAgICAgID0geyBzdGFydDogbnVsbCwgZW5kOiBudWxsIH07XG5cbiAgICAgICAgbGV0IGxpbmUgICAgID0gY29kZUFyclswXTtcbiAgICAgICAgbGV0IHN0YXJ0VG1wID0gZml4ZWRTdGFydDtcbiAgICAgICAgbGV0IGVuZFRtcCAgID0gZW5kO1xuXG4gICAgICAgIGZvciAobGV0IGxpbmVOdW1iZXIgPSAxOyBsaW5lTnVtYmVyIDw9IGNvZGVBcnIubGVuZ3RoOyArK2xpbmVOdW1iZXIsIGxpbmUgPSBjb2RlQXJyW2xpbmVOdW1iZXIgLSAxXSkge1xuICAgICAgICAgICAgc3RhcnRUbXAgLT0gbGluZS5sZW5ndGggKyAxO1xuICAgICAgICAgICAgZW5kVG1wIC09IGxpbmUubGVuZ3RoICsgMTtcblxuICAgICAgICAgICAgaWYgKHN0YXJ0VG1wIDwgMCAmJiAhbG9jLnN0YXJ0KVxuICAgICAgICAgICAgICAgIGxvYy5zdGFydCA9IHsgbGluZTogbGluZU51bWJlciwgY29sdW1uOiBsaW5lLmxlbmd0aCArIHN0YXJ0VG1wICsgMSB9O1xuXG4gICAgICAgICAgICBpZiAoZW5kVG1wIDw9IDAgfHwgbGluZU51bWJlciA9PT0gY29kZUFyci5sZW5ndGggLSAxKSB7XG4gICAgICAgICAgICAgICAgbG9jLmVuZCA9IHsgbGluZTogbGluZU51bWJlciwgY29sdW1uOiBsaW5lLmxlbmd0aCArIGVuZFRtcCArIDEgfTtcblxuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHsgbG9jLCBzdGFydDogZml4ZWRTdGFydCwgZW5kIH07XG4gICAgfVxuXG4gICAgZ2V0UlZhbHVlICh0b2tlbikge1xuICAgICAgICByZXR1cm4gdG9rZW4uZGVjbGFyYXRpb25MaXN0LmRlY2xhcmF0aW9uc1swXS5pbml0aWFsaXplcjtcbiAgICB9XG5cbiAgICBnZXRTdHJpbmdWYWx1ZSAodG9rZW4pIHtcbiAgICAgICAgY29uc3Qgc3RyaW5nVHlwZXMgPSBbdGhpcy50b2tlblR5cGUuU3RyaW5nTGl0ZXJhbCwgdGhpcy50b2tlblR5cGUuVGVtcGxhdGVFeHByZXNzaW9uXTtcblxuICAgICAgICBpZiAoc3RyaW5nVHlwZXMuaW5kZXhPZih0b2tlbi5raW5kKSA+IC0xIHx8IHRva2VuLnRleHQgJiYgdG9rZW4ua2luZCAhPT0gdGhpcy50b2tlblR5cGUuTnVtZXJpY0xpdGVyYWwpXG4gICAgICAgICAgICByZXR1cm4gdGhpcy5mb3JtYXRGbkFyZyh0b2tlbik7XG5cbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgaXNBc3luY0ZuICh0b2tlbikge1xuICAgICAgICBjb25zdCBpc0dlbmVyYXRvckZuID0gISF0b2tlbi5hc3Rlcmlza1Rva2VuO1xuICAgICAgICBjb25zdCBpc0FzeW5jRm4gICAgID0gdG9rZW4ubW9kaWZpZXJzICYmXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b2tlbi5tb2RpZmllcnMuc29tZShtb2RpZmllciA9PiBtb2RpZmllci5raW5kID09PSB0aGlzLnRva2VuVHlwZS5Bc3luY0tleXdvcmQpO1xuXG4gICAgICAgIHJldHVybiBpc0dlbmVyYXRvckZuIHx8IGlzQXN5bmNGbjtcbiAgICB9XG5cbiAgICBnZXRGdW5jdGlvbkJvZHkgKHRva2VuKSB7XG4gICAgICAgIHJldHVybiB0b2tlbi5ib2R5LnN0YXRlbWVudHM7XG4gICAgfVxuXG4gICAgZm9ybWF0Rm5EYXRhIChuYW1lLCB2YWx1ZSwgdG9rZW4sIG1ldGEgPSBbe31dKSB7XG4gICAgICAgIGNvbnN0IGxvYyA9IHRoaXMuZ2V0TG9jYXRpb25CeU9mZnNldHModG9rZW4ucG9zLCB0b2tlbi5lbmQpO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBmbk5hbWU6IG5hbWUsXG4gICAgICAgICAgICB2YWx1ZTogIHZhbHVlLFxuICAgICAgICAgICAgbG9jOiAgICBsb2MubG9jLFxuICAgICAgICAgICAgc3RhcnQ6ICBsb2Muc3RhcnQsXG4gICAgICAgICAgICBlbmQ6ICAgIGxvYy5lbmQsXG4gICAgICAgICAgICBtZXRhOiAgIG1lcmdlKHt9LCAuLi5tZXRhKVxuICAgICAgICB9O1xuICAgIH1cblxuICAgIGFuYWx5emVNZW1iZXJFeHAgKHRva2VuKSB7XG4gICAgICAgIGxldCBleHAgICAgICAgICA9IHRva2VuO1xuICAgICAgICBjb25zdCB0b2tlblR5cGUgPSB0aGlzLnRva2VuVHlwZTtcbiAgICAgICAgY29uc3QgY2FsbFN0YWNrID0gW2V4cF07XG5cbiAgICAgICAgd2hpbGUgKGV4cC5raW5kICE9PSB0aGlzLnRva2VuVHlwZS5JZGVudGlmaWVyKSB7XG4gICAgICAgICAgICBleHAgPSBleHAuZXhwcmVzc2lvbiB8fCBleHAudGFnO1xuXG4gICAgICAgICAgICBjYWxsU3RhY2sucHVzaChleHApO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgbWV0YSA9IHRoaXMuZ2V0TWV0YUluZm8oY2FsbFN0YWNrLnNsaWNlKCkpO1xuXG4gICAgICAgIGlmIChleHAgJiYgdGhpcy5pc0FwaUZuKGV4cC50ZXh0KSkge1xuICAgICAgICAgICAgbGV0IHBhcmVudEV4cCA9IGNhbGxTdGFjay5wb3AoKTtcblxuICAgICAgICAgICAgd2hpbGUgKHBhcmVudEV4cCkge1xuICAgICAgICAgICAgICAgIGlmIChwYXJlbnRFeHAua2luZCA9PT0gdG9rZW5UeXBlLkNhbGxFeHByZXNzaW9uICYmIHBhcmVudEV4cC5leHByZXNzaW9uKSB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNhbGxlZVR5cGUgICAgID0gcGFyZW50RXhwLmV4cHJlc3Npb24ua2luZDtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgY2FsbGVlTWVtYmVyRm4gPSBjYWxsZWVUeXBlID09PSB0b2tlblR5cGUuUHJvcGVydHlBY2Nlc3NFeHByZXNzaW9uICYmXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcGFyZW50RXhwLmV4cHJlc3Npb24ubmFtZS50ZXh0O1xuXG4gICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLmNoZWNrRXhwRGVmaW5lVGFyZ2V0TmFtZShjYWxsZWVUeXBlLCBjYWxsZWVNZW1iZXJGbikpXG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5mb3JtYXRGbkRhdGEoZXhwLnRleHQsIHRoaXMuZm9ybWF0Rm5BcmcocGFyZW50RXhwLmFyZ3VtZW50c1swXSksIHRva2VuLCBtZXRhKTtcbiAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICBpZiAocGFyZW50RXhwLmtpbmQgPT09IHRva2VuVHlwZS5UYWdnZWRUZW1wbGF0ZUV4cHJlc3Npb24gJiYgcGFyZW50RXhwLnRhZykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCB0YWdUeXBlICAgICA9IHBhcmVudEV4cC50YWcua2luZDtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgdGFnTWVtYmVyRm4gPSB0YWdUeXBlID09PSB0b2tlblR5cGUuUHJvcGVydHlBY2Nlc3NFeHByZXNzaW9uICYmIHBhcmVudEV4cC50YWcubmFtZS50ZXh0O1xuXG4gICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLmNoZWNrRXhwRGVmaW5lVGFyZ2V0TmFtZSh0YWdUeXBlLCB0YWdNZW1iZXJGbikpXG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5mb3JtYXRGbkRhdGEoZXhwLnRleHQsIHRoaXMuZm9ybWF0Rm5BcmcocGFyZW50RXhwKSwgdG9rZW4sIG1ldGEpO1xuICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgIHBhcmVudEV4cCA9IGNhbGxTdGFjay5wb3AoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGZvcm1hdEZuQXJnIChhcmcpIHtcbiAgICAgICAgaWYgKGFyZy50ZW1wbGF0ZVNwYW5zKVxuICAgICAgICAgICAgcmV0dXJuIHRoaXMuZ2V0Q29tcHV0ZWROYW1lU3RyaW5nKHsgcG9zOiBhcmcucG9zLCBlbmQ6IGFyZy5lbmQgfSk7XG5cbiAgICAgICAgaWYgKGFyZy5oZWFkKVxuICAgICAgICAgICAgcmV0dXJuIHRoaXMuZ2V0Q29tcHV0ZWROYW1lU3RyaW5nKHsgcG9zOiBhcmcudGVtcGxhdGUucG9zLCBlbmQ6IGFyZy50ZW1wbGF0ZS5lbmQgfSk7XG5cbiAgICAgICAgaWYgKGFyZy50ZW1wbGF0ZSlcbiAgICAgICAgICAgIHJldHVybiBhcmcudGVtcGxhdGUudGV4dCB8fCB0aGlzLmdldENvbXB1dGVkTmFtZVN0cmluZyh7IHBvczogYXJnLnRlbXBsYXRlLnBvcywgZW5kOiBhcmcudGVtcGxhdGUuZW5kIH0pO1xuXG4gICAgICAgIGlmIChhcmcua2luZCA9PT0gdGhpcy50b2tlblR5cGUuSWRlbnRpZmllcilcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmdldENvbXB1dGVkTmFtZVN0cmluZyh7IHBvczogYXJnLnBvcywgZW5kOiBhcmcuZW5kIH0pO1xuXG4gICAgICAgIGlmIChhcmcudGV4dCAmJiBhcmcua2luZCAhPT0gdGhpcy50b2tlblR5cGUuTnVtZXJpY0xpdGVyYWwpXG4gICAgICAgICAgICByZXR1cm4gYXJnLnRleHQ7XG5cbiAgICAgICAgaWYgKGFyZy5raW5kID09PSB0aGlzLnRva2VuVHlwZS5UeXBlQXNzZXJ0aW9uRXhwcmVzc2lvbilcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmZvcm1hdEZuQXJnKGFyZy5leHByZXNzaW9uKTtcblxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBnZXRGbkNhbGwgKHRva2VuKSB7XG4gICAgICAgIGlmICh0aGlzLmlzQXBpRm4odG9rZW4uZXhwcmVzc2lvbi50ZXh0KSlcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmZvcm1hdEZuRGF0YSh0b2tlbi5leHByZXNzaW9uLnRleHQsIHRoaXMuZm9ybWF0Rm5BcmcodG9rZW4uYXJndW1lbnRzWzBdKSwgdG9rZW4pO1xuXG4gICAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIGdldFRhZ2dlZFRlbXBsYXRlRXhwICh0b2tlbikge1xuICAgICAgICBpZiAodGhpcy5pc0FwaUZuKHRva2VuLnRhZy50ZXh0KSlcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmZvcm1hdEZuRGF0YSh0b2tlbi50YWcudGV4dCwgdGhpcy5mb3JtYXRGbkFyZyh0b2tlbiksIHRva2VuKTtcblxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBhbmFseXplRm5DYWxsICh0b2tlbikge1xuICAgICAgICBjb25zdCB0b2tlblR5cGUgPSB0aGlzLnRva2VuVHlwZTtcblxuICAgICAgICBpZiAodG9rZW4ua2luZCA9PT0gdG9rZW5UeXBlLlByb3BlcnR5QWNjZXNzRXhwcmVzc2lvbilcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmFuYWx5emVNZW1iZXJFeHAodG9rZW4pO1xuXG4gICAgICAgIGlmICh0b2tlbi5raW5kID09PSB0b2tlblR5cGUuQ2FsbEV4cHJlc3Npb24pIHtcbiAgICAgICAgICAgIGNvbnN0IGV4cEtpbmQgPSB0b2tlbi5leHByZXNzaW9uLmtpbmQ7XG5cbiAgICAgICAgICAgIGlmIChleHBLaW5kID09PSB0b2tlblR5cGUuUHJvcGVydHlBY2Nlc3NFeHByZXNzaW9uIHx8IGV4cEtpbmQgPT09IHRva2VuVHlwZS5DYWxsRXhwcmVzc2lvbilcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5hbmFseXplTWVtYmVyRXhwKHRva2VuKTtcblxuICAgICAgICAgICAgaWYgKGV4cEtpbmQgPT09IHRva2VuVHlwZS5QYXJlbnRoZXNpemVkRXhwcmVzc2lvbilcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5hbmFseXplRm5DYWxsKHRva2VuLmV4cHJlc3Npb24uZXhwcmVzc2lvbik7XG5cbiAgICAgICAgICAgIHJldHVybiB0aGlzLmdldEZuQ2FsbCh0b2tlbik7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAodG9rZW4ua2luZCA9PT0gdG9rZW5UeXBlLkZ1bmN0aW9uRXhwcmVzc2lvbiB8fCB0b2tlbi5raW5kID09PSB0b2tlblR5cGUuQXJyb3dGdW5jdGlvbilcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmNvbGxlY3RUZXN0Q2FmZUNhbGxzKHRoaXMuZ2V0RnVuY3Rpb25Cb2R5KHRva2VuKSk7XG5cbiAgICAgICAgaWYgKHRva2VuLmtpbmQgPT09IHRva2VuVHlwZS5UYWdnZWRUZW1wbGF0ZUV4cHJlc3Npb24pIHtcbiAgICAgICAgICAgIGlmICh0b2tlbi50YWcua2luZCA9PT0gdG9rZW5UeXBlLlByb3BlcnR5QWNjZXNzRXhwcmVzc2lvbiB8fCB0b2tlbi50YWcua2luZCA9PT0gdG9rZW5UeXBlLkNhbGxFeHByZXNzaW9uKVxuICAgICAgICAgICAgICAgIHJldHVybiB0aGlzLmFuYWx5emVNZW1iZXJFeHAodG9rZW4pO1xuXG4gICAgICAgICAgICByZXR1cm4gdGhpcy5nZXRUYWdnZWRUZW1wbGF0ZUV4cCh0b2tlbik7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBwYXJzZSAoY29kZSkge1xuICAgICAgICAvL05PVEU6IFR5cGVTY3JpcHQgY2FsY3VsYXRlcyBzdGFydCBwb3NpdGlvbiBvZiBhIHRva2VuIGluY29ycmVjdGx5XG4gICAgICAgIC8vSXQgZG9lc24ndCBjb25zaWRlciBzcGFjZXMgYW5kIGNvbW1lbnRzIGJldHdlZW4gdGhlIGxhc3QgdG9rZW4gYW5kIHRoZSBjdXJyZW50IHRva2VuLlxuICAgICAgICAvL1NvIHdlIHJlcGxhY2UgY29tbWVudHMgd2l0aCBzcGFjZSBzeW1ib2xzIHRvIGNhbGN1bGF0ZSBmaXhlZCBwb3NpdGlvbi5cbiAgICAgICAgLy9XZSBqdXN0IGluY3JlbWVudCBwb3NpdGlvbiB1bnRpbCB3ZSBtZWV0IG5vbiB3aGl0ZXNwYWNlIGNoYXJhY3RlcnNcbiAgICAgICAgdGhpcy5jb2RlQXJyICAgICAgICAgICAgID0gY29kZS5zcGxpdCgnXFxuJyk7XG4gICAgICAgIHRoaXMuY29kZVdpdGhvdXRDb21tZW50cyA9IHJlcGxhY2VDb21tZW50cyhjb2RlKTtcblxuICAgICAgICBjb25zdCB0c0NvbmZpZyAgID0gbmV3IFR5cGVzY3JpcHRDb25maWd1cmF0aW9uKCk7XG4gICAgICAgIGNvbnN0IHNvdXJjZUZpbGUgPSB0cy5jcmVhdGVTb3VyY2VGaWxlKCcnLCBjb2RlLCB0c0NvbmZpZy5nZXRPcHRpb25zKCksIHRydWUpO1xuXG4gICAgICAgIHJldHVybiB0aGlzLmFuYWx5emUoc291cmNlRmlsZS5zdGF0ZW1lbnRzKTtcbiAgICB9XG59XG5cbmNvbnN0IHBhcnNlciA9IG5ldyBUeXBlU2NyaXB0VGVzdEZpbGVQYXJzZXIoKTtcblxuZXhwb3J0IGNvbnN0IGdldFR5cGVTY3JpcHRUZXN0TGlzdCAgICAgICAgID0gcGFyc2VyLmdldFRlc3RMaXN0LmJpbmQocGFyc2VyKTtcbmV4cG9ydCBjb25zdCBnZXRUeXBlU2NyaXB0VGVzdExpc3RGcm9tQ29kZSA9IHBhcnNlci5nZXRUZXN0TGlzdEZyb21Db2RlLmJpbmQocGFyc2VyKTtcbiJdfQ==
|