velocious 1.0.171 → 1.0.172
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/README.md +43 -0
- package/build/src/configuration-types.d.ts +14 -0
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +5 -1
- package/build/src/configuration.d.ts +9 -1
- package/build/src/configuration.d.ts.map +1 -1
- package/build/src/configuration.js +14 -2
- package/build/src/environment-handlers/node/cli/commands/test.d.ts.map +1 -1
- package/build/src/environment-handlers/node/cli/commands/test.js +71 -7
- package/build/src/http-server/client/index.d.ts.map +1 -1
- package/build/src/http-server/client/index.js +4 -2
- package/build/src/http-server/client/websocket-session.d.ts +23 -3
- package/build/src/http-server/client/websocket-session.d.ts.map +1 -1
- package/build/src/http-server/client/websocket-session.js +62 -7
- package/build/src/http-server/websocket-channel.d.ts +45 -0
- package/build/src/http-server/websocket-channel.d.ts.map +1 -0
- package/build/src/http-server/websocket-channel.js +64 -0
- package/build/src/http-server/worker-handler/worker-thread.d.ts +2 -2
- package/build/src/http-server/worker-handler/worker-thread.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/worker-thread.js +10 -5
- package/build/src/testing/test-files-finder.d.ts +18 -0
- package/build/src/testing/test-files-finder.d.ts.map +1 -1
- package/build/src/testing/test-files-finder.js +44 -4
- package/build/src/testing/test-runner.d.ts +92 -4
- package/build/src/testing/test-runner.d.ts.map +1 -1
- package/build/src/testing/test-runner.js +153 -11
- package/build/src/testing/test.d.ts +4 -1
- package/build/src/testing/test.d.ts.map +1 -1
- package/build/src/testing/test.js +58 -5
- package/build/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
|
@@ -4,6 +4,7 @@ import TestFilesFinder from "../../../../testing/test-files-finder.js";
|
|
|
4
4
|
import TestRunner from "../../../../testing/test-runner.js";
|
|
5
5
|
const INCLUDE_TAG_FLAGS = new Set(["--tag", "--include-tag", "-t"]);
|
|
6
6
|
const EXCLUDE_TAG_FLAGS = new Set(["--exclude-tag", "--skip-tag", "-x"]);
|
|
7
|
+
const EXAMPLE_FLAGS = new Set(["--example", "--name", "-e"]);
|
|
7
8
|
/**
|
|
8
9
|
* @param {string | undefined} value - Tag argument value.
|
|
9
10
|
* @returns {string[]} - Tags list.
|
|
@@ -16,14 +17,39 @@ function splitTags(value) {
|
|
|
16
17
|
.map((tag) => tag.trim())
|
|
17
18
|
.filter(Boolean);
|
|
18
19
|
}
|
|
20
|
+
/**
|
|
21
|
+
* @param {string} value - Value.
|
|
22
|
+
* @returns {string} - Escaped value for regex.
|
|
23
|
+
*/
|
|
24
|
+
function escapeRegExp(value) {
|
|
25
|
+
return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* @param {string[]} patterns - Patterns.
|
|
29
|
+
* @returns {RegExp[]} - Normalized patterns.
|
|
30
|
+
*/
|
|
31
|
+
function normalizeExamplePatterns(patterns) {
|
|
32
|
+
const normalized = [];
|
|
33
|
+
for (const pattern of patterns) {
|
|
34
|
+
const regexMatch = pattern.match(/^\/(.+)\/([gimsuy]*)$/);
|
|
35
|
+
if (regexMatch) {
|
|
36
|
+
normalized.push(new RegExp(regexMatch[1], regexMatch[2]));
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
normalized.push(new RegExp(escapeRegExp(pattern)));
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
return normalized;
|
|
43
|
+
}
|
|
19
44
|
/**
|
|
20
45
|
* @param {string[]} processArgs - Process args.
|
|
21
|
-
* @returns {{includeTags: string[], excludeTags: string[], filteredProcessArgs: string[]}} - Parsed tags and process args.
|
|
46
|
+
* @returns {{includeTags: string[], excludeTags: string[], examplePatterns: string[], filteredProcessArgs: string[]}} - Parsed tags and process args.
|
|
22
47
|
*/
|
|
23
|
-
function
|
|
48
|
+
function parseFilters(processArgs) {
|
|
24
49
|
const includeTags = [];
|
|
25
50
|
const excludeTags = [];
|
|
26
51
|
const filteredProcessArgs = processArgs.length > 0 ? [processArgs[0]] : [];
|
|
52
|
+
const examplePatterns = [];
|
|
27
53
|
let inRestArgs = false;
|
|
28
54
|
for (let i = 1; i < processArgs.length; i++) {
|
|
29
55
|
const arg = processArgs[i];
|
|
@@ -65,12 +91,29 @@ function parseTagFilters(processArgs) {
|
|
|
65
91
|
excludeTags.push(...splitTags(arg.slice("--skip-tag=".length)));
|
|
66
92
|
continue;
|
|
67
93
|
}
|
|
94
|
+
if (EXAMPLE_FLAGS.has(arg)) {
|
|
95
|
+
const nextValue = processArgs[i + 1];
|
|
96
|
+
if (nextValue && !nextValue.startsWith("-")) {
|
|
97
|
+
examplePatterns.push(nextValue);
|
|
98
|
+
i++;
|
|
99
|
+
}
|
|
100
|
+
continue;
|
|
101
|
+
}
|
|
102
|
+
if (arg.startsWith("--example=")) {
|
|
103
|
+
examplePatterns.push(arg.slice("--example=".length));
|
|
104
|
+
continue;
|
|
105
|
+
}
|
|
106
|
+
if (arg.startsWith("--name=")) {
|
|
107
|
+
examplePatterns.push(arg.slice("--name=".length));
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
68
110
|
}
|
|
69
111
|
filteredProcessArgs.push(arg);
|
|
70
112
|
}
|
|
71
113
|
return {
|
|
72
114
|
includeTags: Array.from(new Set(includeTags)),
|
|
73
115
|
excludeTags: Array.from(new Set(excludeTags)),
|
|
116
|
+
examplePatterns,
|
|
74
117
|
filteredProcessArgs
|
|
75
118
|
};
|
|
76
119
|
}
|
|
@@ -89,10 +132,17 @@ export default class VelociousCliCommandsTest extends BaseCommand {
|
|
|
89
132
|
directories.push(`${this.directory()}/tests`);
|
|
90
133
|
directories.push(`${this.directory()}/spec`);
|
|
91
134
|
}
|
|
92
|
-
const { includeTags, excludeTags, filteredProcessArgs } =
|
|
135
|
+
const { includeTags, excludeTags, examplePatterns, filteredProcessArgs } = parseFilters(this.processArgs || []);
|
|
93
136
|
const testFilesFinder = new TestFilesFinder({ directory, directories, processArgs: filteredProcessArgs });
|
|
94
137
|
const testFiles = await testFilesFinder.findTestFiles();
|
|
95
|
-
const testRunner = new TestRunner({
|
|
138
|
+
const testRunner = new TestRunner({
|
|
139
|
+
configuration: this.getConfiguration(),
|
|
140
|
+
excludeTags,
|
|
141
|
+
includeTags,
|
|
142
|
+
testFiles,
|
|
143
|
+
lineFilters: testFilesFinder.getLineFiltersByFile(),
|
|
144
|
+
examplePatterns: normalizeExamplePatterns(examplePatterns)
|
|
145
|
+
});
|
|
96
146
|
let signalHandled = false;
|
|
97
147
|
const handleSignal = async (signal) => {
|
|
98
148
|
if (signalHandled)
|
|
@@ -117,11 +167,25 @@ export default class VelociousCliCommandsTest extends BaseCommand {
|
|
|
117
167
|
}
|
|
118
168
|
await testRunner.run();
|
|
119
169
|
const executedTests = testRunner.getExecutedTestsCount();
|
|
120
|
-
|
|
121
|
-
|
|
170
|
+
const lineFilters = testRunner.getLineFilters();
|
|
171
|
+
const hasLineFilters = Object.keys(lineFilters).length > 0;
|
|
172
|
+
const hasExampleFilters = examplePatterns.length > 0;
|
|
173
|
+
const hasTagFilters = includeTags.length > 0 || excludeTags.length > 0;
|
|
174
|
+
if ((hasTagFilters || hasLineFilters || hasExampleFilters) && executedTests === 0) {
|
|
175
|
+
console.error("\nNo tests matched the provided filters");
|
|
122
176
|
process.exit(1);
|
|
123
177
|
}
|
|
124
178
|
if (testRunner.isFailed()) {
|
|
179
|
+
const failedTests = testRunner.getFailedTestDetails();
|
|
180
|
+
if (failedTests.length > 0) {
|
|
181
|
+
console.error("\nFailed tests:");
|
|
182
|
+
for (const failed of failedTests) {
|
|
183
|
+
const location = failed.filePath && failed.line
|
|
184
|
+
? ` (${failed.filePath}:${failed.line})`
|
|
185
|
+
: "";
|
|
186
|
+
console.error(`- ${failed.fullDescription}${location}`);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
125
189
|
console.error(`\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`);
|
|
126
190
|
process.exit(1);
|
|
127
191
|
}
|
|
@@ -135,4 +199,4 @@ export default class VelociousCliCommandsTest extends BaseCommand {
|
|
|
135
199
|
}
|
|
136
200
|
}
|
|
137
201
|
}
|
|
138
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseCommand from \"../../../../cli/base-command.js\"\nimport TestFilesFinder from \"../../../../testing/test-files-finder.js\"\nimport TestRunner from \"../../../../testing/test-runner.js\"\n\nconst INCLUDE_TAG_FLAGS = new Set([\"--tag\", \"--include-tag\", \"-t\"])\nconst EXCLUDE_TAG_FLAGS = new Set([\"--exclude-tag\", \"--skip-tag\", \"-x\"])\n\n/**\n * @param {string | undefined} value - Tag argument value.\n * @returns {string[]} - Tags list.\n */\nfunction splitTags(value) {\n  if (!value) return []\n\n  return value\n    .split(\",\")\n    .map((tag) => tag.trim())\n    .filter(Boolean)\n}\n\n/**\n * @param {string[]} processArgs - Process args.\n * @returns {{includeTags: string[], excludeTags: string[], filteredProcessArgs: string[]}} - Parsed tags and process args.\n */\nfunction parseTagFilters(processArgs) {\n  const includeTags = []\n  const excludeTags = []\n  const filteredProcessArgs = processArgs.length > 0 ? [processArgs[0]] : []\n  let inRestArgs = false\n\n  for (let i = 1; i < processArgs.length; i++) {\n    const arg = processArgs[i]\n\n    if (arg === \"--\") {\n      inRestArgs = true\n      filteredProcessArgs.push(arg)\n      continue\n    }\n\n    if (!inRestArgs) {\n      if (INCLUDE_TAG_FLAGS.has(arg)) {\n        const nextValue = processArgs[i + 1]\n\n        if (nextValue && !nextValue.startsWith(\"-\")) {\n          includeTags.push(...splitTags(nextValue))\n          i++\n        }\n        continue\n      }\n\n      if (EXCLUDE_TAG_FLAGS.has(arg)) {\n        const nextValue = processArgs[i + 1]\n\n        if (nextValue && !nextValue.startsWith(\"-\")) {\n          excludeTags.push(...splitTags(nextValue))\n          i++\n        }\n        continue\n      }\n\n      if (arg.startsWith(\"--tag=\")) {\n        includeTags.push(...splitTags(arg.slice(\"--tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--include-tag=\")) {\n        includeTags.push(...splitTags(arg.slice(\"--include-tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--exclude-tag=\")) {\n        excludeTags.push(...splitTags(arg.slice(\"--exclude-tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--skip-tag=\")) {\n        excludeTags.push(...splitTags(arg.slice(\"--skip-tag=\".length)))\n        continue\n      }\n    }\n\n    filteredProcessArgs.push(arg)\n  }\n\n  return {\n    includeTags: Array.from(new Set(includeTags)),\n    excludeTags: Array.from(new Set(excludeTags)),\n    filteredProcessArgs\n  }\n}\n\nexport default class VelociousCliCommandsTest extends BaseCommand {\n  async execute() {\n    this.getConfiguration().setEnvironment(\"test\")\n\n    let directory\n    const directories = []\n\n    if (process.env.VELOCIOUS_TEST_DIR) {\n      directory = process.env.VELOCIOUS_TEST_DIR\n      directories.push(process.env.VELOCIOUS_TEST_DIR)\n    } else {\n      directory = this.directory()\n      directories.push(`${this.directory()}/__tests__`)\n      directories.push(`${this.directory()}/tests`)\n      directories.push(`${this.directory()}/spec`)\n    }\n\n    const {includeTags, excludeTags, filteredProcessArgs} = parseTagFilters(this.processArgs || [])\n    const testFilesFinder = new TestFilesFinder({directory, directories, processArgs: filteredProcessArgs})\n    const testFiles = await testFilesFinder.findTestFiles()\n    const testRunner = new TestRunner({configuration: this.getConfiguration(), excludeTags, includeTags, testFiles})\n    let signalHandled = false\n\n    const handleSignal = async (signal) => {\n      if (signalHandled) return\n      signalHandled = true\n      console.error(`\\nReceived ${signal}, running afterAll hooks before exit...`)\n\n      try {\n        await testRunner.runAfterAllsForActiveScopes()\n      } catch (error) {\n        console.error(\"Failed while running afterAll hooks:\", error)\n      } finally {\n        process.exit(130)\n      }\n    }\n\n    process.once(\"SIGINT\", () => { void handleSignal(\"SIGINT\") })\n    process.once(\"SIGTERM\", () => { void handleSignal(\"SIGTERM\") })\n\n    await testRunner.prepare()\n\n    if (testRunner.getTestsCount() === 0) {\n      throw new Error(`${testRunner.getTestsCount()} tests was found in ${testFiles.length} file(s)`)\n    }\n\n    await testRunner.run()\n\n    const executedTests = testRunner.getExecutedTestsCount()\n\n    if ((includeTags.length > 0 || excludeTags.length > 0) && executedTests === 0) {\n      console.error(\"\\nNo tests matched the provided tag filters\")\n      process.exit(1)\n    }\n\n    if (testRunner.isFailed()) {\n      console.error(`\\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`)\n      process.exit(1)\n    } else if (testRunner.areAnyTestsFocussed()) {\n      console.error(`\\nFocussed run with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`)\n      process.exit(1)\n    } else {\n      console.log(`\\nTest run succeeded with ${testRunner.getSuccessfulTests()} successful tests`)\n      process.exit(0)\n    }\n  }\n}\n"]}
|
|
202
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,WAAW,MAAM,iCAAiC,CAAA;AACzD,OAAO,eAAe,MAAM,0CAA0C,CAAA;AACtE,OAAO,UAAU,MAAM,oCAAoC,CAAA;AAE3D,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC,CAAC,OAAO,EAAE,eAAe,EAAE,IAAI,CAAC,CAAC,CAAA;AACnE,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC,CAAC,eAAe,EAAE,YAAY,EAAE,IAAI,CAAC,CAAC,CAAA;AACxE,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,CAAC,WAAW,EAAE,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAA;AAE5D;;;GAGG;AACH,SAAS,SAAS,CAAC,KAAK;IACtB,IAAI,CAAC,KAAK;QAAE,OAAO,EAAE,CAAA;IAErB,OAAO,KAAK;SACT,KAAK,CAAC,GAAG,CAAC;SACV,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;SACxB,MAAM,CAAC,OAAO,CAAC,CAAA;AACpB,CAAC;AAED;;;GAGG;AACH,SAAS,YAAY,CAAC,KAAK;IACzB,OAAO,KAAK,CAAC,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC,CAAA;AACrD,CAAC;AAED;;;GAGG;AACH,SAAS,wBAAwB,CAAC,QAAQ;IACxC,MAAM,UAAU,GAAG,EAAE,CAAA;IAErB,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,UAAU,GAAG,OAAO,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAA;QAEzD,IAAI,UAAU,EAAE,CAAC;YACf,UAAU,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;QAC3D,CAAC;aAAM,CAAC;YACN,UAAU,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,CAAA;QACpD,CAAC;IACH,CAAC;IAED,OAAO,UAAU,CAAA;AACnB,CAAC;AAED;;;GAGG;AACH,SAAS,YAAY,CAAC,WAAW;IAC/B,MAAM,WAAW,GAAG,EAAE,CAAA;IACtB,MAAM,WAAW,GAAG,EAAE,CAAA;IACtB,MAAM,mBAAmB,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAA;IAC1E,MAAM,eAAe,GAAG,EAAE,CAAA;IAC1B,IAAI,UAAU,GAAG,KAAK,CAAA;IAEtB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,WAAW,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QAC5C,MAAM,GAAG,GAAG,WAAW,CAAC,CAAC,CAAC,CAAA;QAE1B,IAAI,GAAG,KAAK,IAAI,EAAE,CAAC;YACjB,UAAU,GAAG,IAAI,CAAA;YACjB,mBAAmB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;YAC7B,SAAQ;QACV,CAAC;QAED,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,IAAI,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC/B,MAAM,SAAS,GAAG,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAA;gBAEpC,IAAI,SAAS,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC5C,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,SAAS,CAAC,CAAC,CAAA;oBACzC,CAAC,EAAE,CAAA;gBACL,CAAC;gBACD,SAAQ;YACV,CAAC;YAED,IAAI,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC/B,MAAM,SAAS,GAAG,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAA;gBAEpC,IAAI,SAAS,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC5C,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,SAAS,CAAC,CAAC,CAAA;oBACzC,CAAC,EAAE,CAAA;gBACL,CAAC;gBACD,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAC7B,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;gBAC1D,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,gBAAgB,CAAC,EAAE,CAAC;gBACrC,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;gBAClE,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,gBAAgB,CAAC,EAAE,CAAC;gBACrC,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;gBAClE,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,aAAa,CAAC,EAAE,CAAC;gBAClC,WAAW,CAAC,IAAI,CAAC,GAAG,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;gBAC/D,SAAQ;YACV,CAAC;YAED,IAAI,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC3B,MAAM,SAAS,GAAG,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAA;gBAEpC,IAAI,SAAS,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC5C,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;oBAC/B,CAAC,EAAE,CAAA;gBACL,CAAC;gBACD,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;gBACjC,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAA;gBACpD,SAAQ;YACV,CAAC;YAED,IAAI,GAAG,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC9B,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAA;gBACjD,SAAQ;YACV,CAAC;QACH,CAAC;QAED,mBAAmB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;IAC/B,CAAC;IAED,OAAO;QACL,WAAW,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC;QAC7C,WAAW,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC;QAC7C,eAAe;QACf,mBAAmB;KACpB,CAAA;AACH,CAAC;AAED,MAAM,CAAC,OAAO,OAAO,wBAAyB,SAAQ,WAAW;IAC/D,KAAK,CAAC,OAAO;QACX,IAAI,CAAC,gBAAgB,EAAE,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;QAE9C,IAAI,SAAS,CAAA;QACb,MAAM,WAAW,GAAG,EAAE,CAAA;QAEtB,IAAI,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,CAAC;YACnC,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAA;YAC1C,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAA;QAClD,CAAC;aAAM,CAAC;YACN,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,CAAA;YAC5B,WAAW,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAA;YACjD,WAAW,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAA;YAC7C,WAAW,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAA;QAC9C,CAAC;QAED,MAAM,EAAC,WAAW,EAAE,WAAW,EAAE,eAAe,EAAE,mBAAmB,EAAC,GAAG,YAAY,CAAC,IAAI,CAAC,WAAW,IAAI,EAAE,CAAC,CAAA;QAC7G,MAAM,eAAe,GAAG,IAAI,eAAe,CAAC,EAAC,SAAS,EAAE,WAAW,EAAE,WAAW,EAAE,mBAAmB,EAAC,CAAC,CAAA;QACvG,MAAM,SAAS,GAAG,MAAM,eAAe,CAAC,aAAa,EAAE,CAAA;QACvD,MAAM,UAAU,GAAG,IAAI,UAAU,CAAC;YAChC,aAAa,EAAE,IAAI,CAAC,gBAAgB,EAAE;YACtC,WAAW;YACX,WAAW;YACX,SAAS;YACT,WAAW,EAAE,eAAe,CAAC,oBAAoB,EAAE;YACnD,eAAe,EAAE,wBAAwB,CAAC,eAAe,CAAC;SAC3D,CAAC,CAAA;QACF,IAAI,aAAa,GAAG,KAAK,CAAA;QAEzB,MAAM,YAAY,GAAG,KAAK,EAAE,MAAM,EAAE,EAAE;YACpC,IAAI,aAAa;gBAAE,OAAM;YACzB,aAAa,GAAG,IAAI,CAAA;YACpB,OAAO,CAAC,KAAK,CAAC,cAAc,MAAM,yCAAyC,CAAC,CAAA;YAE5E,IAAI,CAAC;gBACH,MAAM,UAAU,CAAC,2BAA2B,EAAE,CAAA;YAChD,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,OAAO,CAAC,KAAK,CAAC,sCAAsC,EAAE,KAAK,CAAC,CAAA;YAC9D,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;YACnB,CAAC;QACH,CAAC,CAAA;QAED,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE,GAAG,KAAK,YAAY,CAAC,QAAQ,CAAC,CAAA,CAAC,CAAC,CAAC,CAAA;QAC7D,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,EAAE,GAAG,KAAK,YAAY,CAAC,SAAS,CAAC,CAAA,CAAC,CAAC,CAAC,CAAA;QAE/D,MAAM,UAAU,CAAC,OAAO,EAAE,CAAA;QAE1B,IAAI,UAAU,CAAC,aAAa,EAAE,KAAK,CAAC,EAAE,CAAC;YACrC,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,CAAC,aAAa,EAAE,uBAAuB,SAAS,CAAC,MAAM,UAAU,CAAC,CAAA;QACjG,CAAC;QAED,MAAM,UAAU,CAAC,GAAG,EAAE,CAAA;QAEtB,MAAM,aAAa,GAAG,UAAU,CAAC,qBAAqB,EAAE,CAAA;QAExD,MAAM,WAAW,GAAG,UAAU,CAAC,cAAc,EAAE,CAAA;QAC/C,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,GAAG,CAAC,CAAA;QAC1D,MAAM,iBAAiB,GAAG,eAAe,CAAC,MAAM,GAAG,CAAC,CAAA;QACpD,MAAM,aAAa,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,CAAA;QAEtE,IAAI,CAAC,aAAa,IAAI,cAAc,IAAI,iBAAiB,CAAC,IAAI,aAAa,KAAK,CAAC,EAAE,CAAC;YAClF,OAAO,CAAC,KAAK,CAAC,yCAAyC,CAAC,CAAA;YACxD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;QAED,IAAI,UAAU,CAAC,QAAQ,EAAE,EAAE,CAAC;YAC1B,MAAM,WAAW,GAAG,UAAU,CAAC,oBAAoB,EAAE,CAAA;YAErD,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC3B,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAA;gBAEhC,KAAK,MAAM,MAAM,IAAI,WAAW,EAAE,CAAC;oBACjC,MAAM,QAAQ,GAAG,MAAM,CAAC,QAAQ,IAAI,MAAM,CAAC,IAAI;wBAC7C,CAAC,CAAC,KAAK,MAAM,CAAC,QAAQ,IAAI,MAAM,CAAC,IAAI,GAAG;wBACxC,CAAC,CAAC,EAAE,CAAA;oBACN,OAAO,CAAC,KAAK,CAAC,KAAK,MAAM,CAAC,eAAe,GAAG,QAAQ,EAAE,CAAC,CAAA;gBACzD,CAAC;YACH,CAAC;YAED,OAAO,CAAC,KAAK,CAAC,0BAA0B,UAAU,CAAC,cAAc,EAAE,qBAAqB,UAAU,CAAC,kBAAkB,EAAE,cAAc,CAAC,CAAA;YACtI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;aAAM,IAAI,UAAU,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAC5C,OAAO,CAAC,KAAK,CAAC,uBAAuB,UAAU,CAAC,cAAc,EAAE,qBAAqB,UAAU,CAAC,kBAAkB,EAAE,cAAc,CAAC,CAAA;YACnI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,6BAA6B,UAAU,CAAC,kBAAkB,EAAE,mBAAmB,CAAC,CAAA;YAC5F,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;IACH,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport BaseCommand from \"../../../../cli/base-command.js\"\nimport TestFilesFinder from \"../../../../testing/test-files-finder.js\"\nimport TestRunner from \"../../../../testing/test-runner.js\"\n\nconst INCLUDE_TAG_FLAGS = new Set([\"--tag\", \"--include-tag\", \"-t\"])\nconst EXCLUDE_TAG_FLAGS = new Set([\"--exclude-tag\", \"--skip-tag\", \"-x\"])\nconst EXAMPLE_FLAGS = new Set([\"--example\", \"--name\", \"-e\"])\n\n/**\n * @param {string | undefined} value - Tag argument value.\n * @returns {string[]} - Tags list.\n */\nfunction splitTags(value) {\n  if (!value) return []\n\n  return value\n    .split(\",\")\n    .map((tag) => tag.trim())\n    .filter(Boolean)\n}\n\n/**\n * @param {string} value - Value.\n * @returns {string} - Escaped value for regex.\n */\nfunction escapeRegExp(value) {\n  return value.replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\")\n}\n\n/**\n * @param {string[]} patterns - Patterns.\n * @returns {RegExp[]} - Normalized patterns.\n */\nfunction normalizeExamplePatterns(patterns) {\n  const normalized = []\n\n  for (const pattern of patterns) {\n    const regexMatch = pattern.match(/^\\/(.+)\\/([gimsuy]*)$/)\n\n    if (regexMatch) {\n      normalized.push(new RegExp(regexMatch[1], regexMatch[2]))\n    } else {\n      normalized.push(new RegExp(escapeRegExp(pattern)))\n    }\n  }\n\n  return normalized\n}\n\n/**\n * @param {string[]} processArgs - Process args.\n * @returns {{includeTags: string[], excludeTags: string[], examplePatterns: string[], filteredProcessArgs: string[]}} - Parsed tags and process args.\n */\nfunction parseFilters(processArgs) {\n  const includeTags = []\n  const excludeTags = []\n  const filteredProcessArgs = processArgs.length > 0 ? [processArgs[0]] : []\n  const examplePatterns = []\n  let inRestArgs = false\n\n  for (let i = 1; i < processArgs.length; i++) {\n    const arg = processArgs[i]\n\n    if (arg === \"--\") {\n      inRestArgs = true\n      filteredProcessArgs.push(arg)\n      continue\n    }\n\n    if (!inRestArgs) {\n      if (INCLUDE_TAG_FLAGS.has(arg)) {\n        const nextValue = processArgs[i + 1]\n\n        if (nextValue && !nextValue.startsWith(\"-\")) {\n          includeTags.push(...splitTags(nextValue))\n          i++\n        }\n        continue\n      }\n\n      if (EXCLUDE_TAG_FLAGS.has(arg)) {\n        const nextValue = processArgs[i + 1]\n\n        if (nextValue && !nextValue.startsWith(\"-\")) {\n          excludeTags.push(...splitTags(nextValue))\n          i++\n        }\n        continue\n      }\n\n      if (arg.startsWith(\"--tag=\")) {\n        includeTags.push(...splitTags(arg.slice(\"--tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--include-tag=\")) {\n        includeTags.push(...splitTags(arg.slice(\"--include-tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--exclude-tag=\")) {\n        excludeTags.push(...splitTags(arg.slice(\"--exclude-tag=\".length)))\n        continue\n      }\n\n      if (arg.startsWith(\"--skip-tag=\")) {\n        excludeTags.push(...splitTags(arg.slice(\"--skip-tag=\".length)))\n        continue\n      }\n\n      if (EXAMPLE_FLAGS.has(arg)) {\n        const nextValue = processArgs[i + 1]\n\n        if (nextValue && !nextValue.startsWith(\"-\")) {\n          examplePatterns.push(nextValue)\n          i++\n        }\n        continue\n      }\n\n      if (arg.startsWith(\"--example=\")) {\n        examplePatterns.push(arg.slice(\"--example=\".length))\n        continue\n      }\n\n      if (arg.startsWith(\"--name=\")) {\n        examplePatterns.push(arg.slice(\"--name=\".length))\n        continue\n      }\n    }\n\n    filteredProcessArgs.push(arg)\n  }\n\n  return {\n    includeTags: Array.from(new Set(includeTags)),\n    excludeTags: Array.from(new Set(excludeTags)),\n    examplePatterns,\n    filteredProcessArgs\n  }\n}\n\nexport default class VelociousCliCommandsTest extends BaseCommand {\n  async execute() {\n    this.getConfiguration().setEnvironment(\"test\")\n\n    let directory\n    const directories = []\n\n    if (process.env.VELOCIOUS_TEST_DIR) {\n      directory = process.env.VELOCIOUS_TEST_DIR\n      directories.push(process.env.VELOCIOUS_TEST_DIR)\n    } else {\n      directory = this.directory()\n      directories.push(`${this.directory()}/__tests__`)\n      directories.push(`${this.directory()}/tests`)\n      directories.push(`${this.directory()}/spec`)\n    }\n\n    const {includeTags, excludeTags, examplePatterns, filteredProcessArgs} = parseFilters(this.processArgs || [])\n    const testFilesFinder = new TestFilesFinder({directory, directories, processArgs: filteredProcessArgs})\n    const testFiles = await testFilesFinder.findTestFiles()\n    const testRunner = new TestRunner({\n      configuration: this.getConfiguration(),\n      excludeTags,\n      includeTags,\n      testFiles,\n      lineFilters: testFilesFinder.getLineFiltersByFile(),\n      examplePatterns: normalizeExamplePatterns(examplePatterns)\n    })\n    let signalHandled = false\n\n    const handleSignal = async (signal) => {\n      if (signalHandled) return\n      signalHandled = true\n      console.error(`\\nReceived ${signal}, running afterAll hooks before exit...`)\n\n      try {\n        await testRunner.runAfterAllsForActiveScopes()\n      } catch (error) {\n        console.error(\"Failed while running afterAll hooks:\", error)\n      } finally {\n        process.exit(130)\n      }\n    }\n\n    process.once(\"SIGINT\", () => { void handleSignal(\"SIGINT\") })\n    process.once(\"SIGTERM\", () => { void handleSignal(\"SIGTERM\") })\n\n    await testRunner.prepare()\n\n    if (testRunner.getTestsCount() === 0) {\n      throw new Error(`${testRunner.getTestsCount()} tests was found in ${testFiles.length} file(s)`)\n    }\n\n    await testRunner.run()\n\n    const executedTests = testRunner.getExecutedTestsCount()\n\n    const lineFilters = testRunner.getLineFilters()\n    const hasLineFilters = Object.keys(lineFilters).length > 0\n    const hasExampleFilters = examplePatterns.length > 0\n    const hasTagFilters = includeTags.length > 0 || excludeTags.length > 0\n\n    if ((hasTagFilters || hasLineFilters || hasExampleFilters) && executedTests === 0) {\n      console.error(\"\\nNo tests matched the provided filters\")\n      process.exit(1)\n    }\n\n    if (testRunner.isFailed()) {\n      const failedTests = testRunner.getFailedTestDetails()\n\n      if (failedTests.length > 0) {\n        console.error(\"\\nFailed tests:\")\n\n        for (const failed of failedTests) {\n          const location = failed.filePath && failed.line\n            ? ` (${failed.filePath}:${failed.line})`\n            : \"\"\n          console.error(`- ${failed.fullDescription}${location}`)\n        }\n      }\n\n      console.error(`\\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`)\n      process.exit(1)\n    } else if (testRunner.areAnyTestsFocussed()) {\n      console.error(`\\nFocussed run with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`)\n      process.exit(1)\n    } else {\n      console.log(`\\nTest run succeeded with ${testRunner.getSuccessfulTests()} successful tests`)\n      process.exit(0)\n    }\n  }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAUA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAChB,EAWA;IAnBD,mEAA2B;IAC3B,cAAiB;IAWf,eAA8B;IAC9B,oBAA8B;IAC9B,wDAAkC;IAClC,sBAAkC;IAElC,8BAA8B;IAC9B,gBADW,aAAa,EAAE,CACF;IAG1B;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED,kCAsBC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAoBhB;IAXG,wBAAoF;IAaxF;;;OAGG;IACH,6BAHW,OAAO,cAAc,EAAE,OAAO,GAC5B,OAAO,CAOnB;IAED,0CAA0C;IAC1C,uBADc,IAAI,
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAUA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAChB,EAWA;IAnBD,mEAA2B;IAC3B,cAAiB;IAWf,eAA8B;IAC9B,oBAA8B;IAC9B,wDAAkC;IAClC,sBAAkC;IAElC,8BAA8B;IAC9B,gBADW,aAAa,EAAE,CACF;IAG1B;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED,kCAsBC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAoBhB;IAXG,wBAAoF;IAaxF;;;OAGG;IACH,6BAHW,OAAO,cAAc,EAAE,OAAO,GAC5B,OAAO,CAOnB;IAED,0CAA0C;IAC1C,uBADc,IAAI,CAsCjB;IAbC,mCAIE;IAWJ,wBAEC;IAED,yBAqBC;IAED;;;OAGG;IACH,4BAHW,aAAa,GACX,IAAI,CAwChB;CACF;uBAxNoB,iBAAiB;0BAEZ,qBAAqB;oBAD3B,cAAc;6BAEL,wBAAwB"}
|
|
@@ -118,7 +118,8 @@ export default class VeoliciousHttpServerClient {
|
|
|
118
118
|
const response = responseLines.join("\r\n");
|
|
119
119
|
this.websocketSession = new WebsocketSession({
|
|
120
120
|
client: this,
|
|
121
|
-
configuration: this.configuration
|
|
121
|
+
configuration: this.configuration,
|
|
122
|
+
upgradeRequest: this.currentRequest
|
|
122
123
|
});
|
|
123
124
|
this.websocketSession.events.on("close", () => {
|
|
124
125
|
this.websocketSession?.destroy();
|
|
@@ -127,6 +128,7 @@ export default class VeoliciousHttpServerClient {
|
|
|
127
128
|
});
|
|
128
129
|
this.state = "websocket";
|
|
129
130
|
this.events.emit("output", response);
|
|
131
|
+
void this.websocketSession.initializeChannel();
|
|
130
132
|
}
|
|
131
133
|
requestDone = () => {
|
|
132
134
|
this.sendDoneRequests();
|
|
@@ -187,4 +189,4 @@ export default class VeoliciousHttpServerClient {
|
|
|
187
189
|
this.events.emit("output", body);
|
|
188
190
|
}
|
|
189
191
|
}
|
|
190
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport crypto from \"crypto\"\nimport {digg} from \"diggerize\"\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport {Logger} from \"../../logger.js\"\nimport Request from \"./request.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketSession from \"./websocket-session.js\"\n\nexport default class VeoliciousHttpServerClient {\n  events = new EventEmitter()\n  state = \"initial\"\n\n  /**\n   * @param {object} args - Options object.\n   * @param {number} args.clientCount - Client count.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.remoteAddress] - Remote address.\n   */\n  constructor({clientCount, configuration, remoteAddress}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n\n    this.logger = new Logger(this)\n    this.clientCount = clientCount\n    this.configuration = configuration\n    this.remoteAddress = remoteAddress\n\n    /** @type {RequestRunner[]} */\n    this.requestRunners = []\n  }\n\n  /**\n   * @param {string} message - Message text.\n   * @returns {void} - No return value.\n   */\n  _sendBadUpgradeResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  executeCurrentRequest = () => {\n    this.logger.debug(\"executeCurrentRequest\")\n\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    if (this._isWebsocketUpgrade(this.currentRequest)) {\n      this._upgradeToWebsocket()\n      return\n    }\n\n    // We are done parsing the given request and can theoretically start parsing a new one, before the current request is done - so reset the state.\n    this.state = \"initial\"\n\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request: this.currentRequest\n    })\n\n    this.requestRunners.push(requestRunner)\n\n    requestRunner.events.on(\"done\", this.requestDone)\n    requestRunner.run()\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onWrite(data) {\n    if (this.websocketSession) {\n      this.websocketSession.onData(data)\n      return\n    }\n\n    if (this.state == \"initial\") {\n      this.currentRequest = new Request({client: this, configuration: this.configuration})\n      this.currentRequest.requestParser.events.on(\"done\", this.executeCurrentRequest)\n      this.currentRequest.feed(data)\n      this.state = \"requestStarted\"\n    } else if (this.state == \"requestStarted\") {\n      if (!this.currentRequest) throw new Error(\"No current request\")\n\n      this.currentRequest.feed(data)\n    } else {\n      throw new Error(`Unknown state for client: ${this.state}`)\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default} request - Request object.\n   * @returns {boolean} - Whether websocket upgrade.\n   */\n  _isWebsocketUpgrade(request) {\n    const upgradeHeader = request.header(\"upgrade\")?.toLowerCase()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()\n\n    return Boolean(upgradeHeader == \"websocket\" && connectionHeader?.includes(\"upgrade\"))\n  }\n\n  /** @returns {void} - No return value.  */\n  _upgradeToWebsocket() {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const secWebsocketKey = this.currentRequest.header(\"sec-websocket-key\")\n\n    if (!secWebsocketKey) {\n      this._sendBadUpgradeResponse(\"Missing Sec-WebSocket-Key header\")\n      return\n    }\n\n    const websocketAcceptKey = crypto.createHash(\"sha1\")\n      .update(`${secWebsocketKey}258EAFA5-E914-47DA-95CA-C5AB0DC85B11`, \"binary\")\n      .digest(\"base64\")\n    const httpVersion = this.currentRequest.httpVersion() || \"1.1\"\n    const responseLines = [\n      `HTTP/${httpVersion} 101 Switching Protocols`,\n      \"Upgrade: websocket\",\n      \"Connection: Upgrade\",\n      `Sec-WebSocket-Accept: ${websocketAcceptKey}`,\n      \"\",\n      \"\"\n    ]\n    const response = responseLines.join(\"\\r\\n\")\n\n    this.websocketSession = new WebsocketSession({\n      client: this,\n      configuration: this.configuration\n    })\n    this.websocketSession.events.on(\"close\", () => {\n      this.websocketSession?.destroy()\n      this.websocketSession = undefined\n      this.events.emit(\"close\")\n    })\n    this.state = \"websocket\"\n    this.events.emit(\"output\", response)\n  }\n\n  requestDone = () => {\n    this.sendDoneRequests()\n  }\n\n  sendDoneRequests() {\n    while (true) {\n      const requestRunner = this.requestRunners[0]\n      const request = requestRunner?.getRequest()\n\n      if (requestRunner?.getState() == \"done\") {\n        const httpVersion = request.httpVersion()\n        const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n\n        this.requestRunners.shift()\n        this.sendResponse(requestRunner)\n        this.logger.debug(() => [\"sendDoneRequests\", {clientCount: this.clientCount, connectionHeader, httpVersion}])\n\n        if (httpVersion == \"1.0\" && connectionHeader != \"keep-alive\") {\n          this.logger.debug(() => [`Closing the connection because ${httpVersion} and connection header ${connectionHeader}`, {clientCount: this.clientCount}])\n          this.events.emit(\"close\")\n        }\n      } else {\n        break\n      }\n    }\n  }\n\n  /**\n   * @param {RequestRunner} requestRunner - Request runner.\n   * @returns {void} - No return value.\n   */\n  sendResponse(requestRunner) {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const response = digg(requestRunner, \"response\")\n    const request = requestRunner.getRequest()\n    const body = response.getBody()\n    const date = new Date()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const httpVersion = request.httpVersion()\n\n    this.logger.debug(\"sendResponse\", {clientCount: this.clientCount, connectionHeader, httpVersion})\n\n    if (httpVersion == \"1.0\" && connectionHeader == \"keep-alive\") {\n      response.setHeader(\"Connection\", \"Keep-Alive\")\n    } else {\n      response.setHeader(\"Connection\", \"Close\")\n    }\n\n    const contentLength = new TextEncoder().encode(body).length\n\n    response.setHeader(\"Content-Length\", contentLength)\n    response.setHeader(\"Date\", date.toUTCString())\n    response.setHeader(\"Server\", \"Velocious\")\n\n    let headers = \"\"\n\n    headers += `HTTP/${this.currentRequest.httpVersion()} ${response.getStatusCode()} ${response.getStatusMessage()}\\r\\n`\n\n    for (const headerKey in response.headers) {\n      for (const headerValue of response.headers[headerKey]) {\n        headers += `${headerKey}: ${headerValue}\\r\\n`\n      }\n    }\n\n    headers += \"\\r\\n\"\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"output\", body)\n  }\n}\n"]}
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport crypto from \"crypto\"\nimport {digg} from \"diggerize\"\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport {Logger} from \"../../logger.js\"\nimport Request from \"./request.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketSession from \"./websocket-session.js\"\n\nexport default class VeoliciousHttpServerClient {\n  events = new EventEmitter()\n  state = \"initial\"\n\n  /**\n   * @param {object} args - Options object.\n   * @param {number} args.clientCount - Client count.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.remoteAddress] - Remote address.\n   */\n  constructor({clientCount, configuration, remoteAddress}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n\n    this.logger = new Logger(this)\n    this.clientCount = clientCount\n    this.configuration = configuration\n    this.remoteAddress = remoteAddress\n\n    /** @type {RequestRunner[]} */\n    this.requestRunners = []\n  }\n\n  /**\n   * @param {string} message - Message text.\n   * @returns {void} - No return value.\n   */\n  _sendBadUpgradeResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  executeCurrentRequest = () => {\n    this.logger.debug(\"executeCurrentRequest\")\n\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    if (this._isWebsocketUpgrade(this.currentRequest)) {\n      this._upgradeToWebsocket()\n      return\n    }\n\n    // We are done parsing the given request and can theoretically start parsing a new one, before the current request is done - so reset the state.\n    this.state = \"initial\"\n\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request: this.currentRequest\n    })\n\n    this.requestRunners.push(requestRunner)\n\n    requestRunner.events.on(\"done\", this.requestDone)\n    requestRunner.run()\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onWrite(data) {\n    if (this.websocketSession) {\n      this.websocketSession.onData(data)\n      return\n    }\n\n    if (this.state == \"initial\") {\n      this.currentRequest = new Request({client: this, configuration: this.configuration})\n      this.currentRequest.requestParser.events.on(\"done\", this.executeCurrentRequest)\n      this.currentRequest.feed(data)\n      this.state = \"requestStarted\"\n    } else if (this.state == \"requestStarted\") {\n      if (!this.currentRequest) throw new Error(\"No current request\")\n\n      this.currentRequest.feed(data)\n    } else {\n      throw new Error(`Unknown state for client: ${this.state}`)\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default} request - Request object.\n   * @returns {boolean} - Whether websocket upgrade.\n   */\n  _isWebsocketUpgrade(request) {\n    const upgradeHeader = request.header(\"upgrade\")?.toLowerCase()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()\n\n    return Boolean(upgradeHeader == \"websocket\" && connectionHeader?.includes(\"upgrade\"))\n  }\n\n  /** @returns {void} - No return value.  */\n  _upgradeToWebsocket() {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const secWebsocketKey = this.currentRequest.header(\"sec-websocket-key\")\n\n    if (!secWebsocketKey) {\n      this._sendBadUpgradeResponse(\"Missing Sec-WebSocket-Key header\")\n      return\n    }\n\n    const websocketAcceptKey = crypto.createHash(\"sha1\")\n      .update(`${secWebsocketKey}258EAFA5-E914-47DA-95CA-C5AB0DC85B11`, \"binary\")\n      .digest(\"base64\")\n    const httpVersion = this.currentRequest.httpVersion() || \"1.1\"\n    const responseLines = [\n      `HTTP/${httpVersion} 101 Switching Protocols`,\n      \"Upgrade: websocket\",\n      \"Connection: Upgrade\",\n      `Sec-WebSocket-Accept: ${websocketAcceptKey}`,\n      \"\",\n      \"\"\n    ]\n    const response = responseLines.join(\"\\r\\n\")\n\n    this.websocketSession = new WebsocketSession({\n      client: this,\n      configuration: this.configuration,\n      upgradeRequest: this.currentRequest\n    })\n    this.websocketSession.events.on(\"close\", () => {\n      this.websocketSession?.destroy()\n      this.websocketSession = undefined\n      this.events.emit(\"close\")\n    })\n    this.state = \"websocket\"\n    this.events.emit(\"output\", response)\n    void this.websocketSession.initializeChannel()\n  }\n\n  requestDone = () => {\n    this.sendDoneRequests()\n  }\n\n  sendDoneRequests() {\n    while (true) {\n      const requestRunner = this.requestRunners[0]\n      const request = requestRunner?.getRequest()\n\n      if (requestRunner?.getState() == \"done\") {\n        const httpVersion = request.httpVersion()\n        const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n\n        this.requestRunners.shift()\n        this.sendResponse(requestRunner)\n        this.logger.debug(() => [\"sendDoneRequests\", {clientCount: this.clientCount, connectionHeader, httpVersion}])\n\n        if (httpVersion == \"1.0\" && connectionHeader != \"keep-alive\") {\n          this.logger.debug(() => [`Closing the connection because ${httpVersion} and connection header ${connectionHeader}`, {clientCount: this.clientCount}])\n          this.events.emit(\"close\")\n        }\n      } else {\n        break\n      }\n    }\n  }\n\n  /**\n   * @param {RequestRunner} requestRunner - Request runner.\n   * @returns {void} - No return value.\n   */\n  sendResponse(requestRunner) {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const response = digg(requestRunner, \"response\")\n    const request = requestRunner.getRequest()\n    const body = response.getBody()\n    const date = new Date()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const httpVersion = request.httpVersion()\n\n    this.logger.debug(\"sendResponse\", {clientCount: this.clientCount, connectionHeader, httpVersion})\n\n    if (httpVersion == \"1.0\" && connectionHeader == \"keep-alive\") {\n      response.setHeader(\"Connection\", \"Keep-Alive\")\n    } else {\n      response.setHeader(\"Connection\", \"Close\")\n    }\n\n    const contentLength = new TextEncoder().encode(body).length\n\n    response.setHeader(\"Content-Length\", contentLength)\n    response.setHeader(\"Date\", date.toUTCString())\n    response.setHeader(\"Server\", \"Velocious\")\n\n    let headers = \"\"\n\n    headers += `HTTP/${this.currentRequest.httpVersion()} ${response.getStatusCode()} ${response.getStatusMessage()}\\r\\n`\n\n    for (const headerKey in response.headers) {\n      for (const headerValue of response.headers[headerKey]) {\n        headers += `${headerKey}: ${headerValue}\\r\\n`\n      }\n    }\n\n    headers += \"\\r\\n\"\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"output\", body)\n  }\n}\n"]}
|
|
@@ -3,16 +3,19 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
3
3
|
* @param {object} args - Options object.
|
|
4
4
|
* @param {import("../../configuration.js").default} args.configuration - Configuration instance.
|
|
5
5
|
* @param {import("./index.js").default} args.client - Client instance.
|
|
6
|
+
* @param {import("./request.js").default | import("./websocket-request.js").default} [args.upgradeRequest] - Initial websocket upgrade request.
|
|
6
7
|
*/
|
|
7
|
-
constructor({ client, configuration }: {
|
|
8
|
+
constructor({ client, configuration, upgradeRequest }: {
|
|
8
9
|
configuration: import("../../configuration.js").default;
|
|
9
10
|
client: import("./index.js").default;
|
|
11
|
+
upgradeRequest?: import("./request.js").default | import("./websocket-request.js").default;
|
|
10
12
|
});
|
|
11
13
|
events: import("eventemitter3").EventEmitter<string | symbol, any>;
|
|
12
14
|
subscriptions: Set<any>;
|
|
13
15
|
buffer: Buffer<ArrayBuffer>;
|
|
14
16
|
client: import("./index.js").default;
|
|
15
17
|
configuration: import("../../configuration.js").default;
|
|
18
|
+
upgradeRequest: WebsocketRequest | import("./request.js").default;
|
|
16
19
|
logger: Logger;
|
|
17
20
|
/**
|
|
18
21
|
* @param {string} channel - Channel name.
|
|
@@ -33,9 +36,14 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
33
36
|
/**
|
|
34
37
|
* @param {string} channel - Channel name.
|
|
35
38
|
* @param {any} payload - Payload data.
|
|
36
|
-
* @returns {void} -
|
|
39
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
40
|
+
*/
|
|
41
|
+
sendEvent(channel: string, payload: any): Promise<void>;
|
|
42
|
+
/**
|
|
43
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
37
44
|
*/
|
|
38
|
-
|
|
45
|
+
initializeChannel(): Promise<void>;
|
|
46
|
+
channel: WebsocketChannel;
|
|
39
47
|
/**
|
|
40
48
|
* @param {import("./index.js").default} client - Client instance.
|
|
41
49
|
* @returns {void} - No return value.
|
|
@@ -61,6 +69,16 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
61
69
|
* @returns {void} - No return value.
|
|
62
70
|
*/
|
|
63
71
|
_sendJson(body: object): void;
|
|
72
|
+
/**
|
|
73
|
+
* @param {string} channel - Channel name.
|
|
74
|
+
* @param {{acknowledge?: boolean}} [options] - Subscribe options.
|
|
75
|
+
* @returns {Promise<boolean>} - Whether the subscription was added.
|
|
76
|
+
*/
|
|
77
|
+
subscribeToChannel(channel: string, { acknowledge }?: {
|
|
78
|
+
acknowledge?: boolean;
|
|
79
|
+
}): Promise<boolean>;
|
|
80
|
+
_handleClose(): void;
|
|
81
|
+
_teardownChannel(): Promise<void>;
|
|
64
82
|
/**
|
|
65
83
|
* @param {Buffer} payload - Payload data.
|
|
66
84
|
* @param {Buffer} mask - Mask.
|
|
@@ -68,5 +86,7 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
68
86
|
*/
|
|
69
87
|
_unmaskPayload(payload: Buffer, mask: Buffer): Buffer;
|
|
70
88
|
}
|
|
89
|
+
import WebsocketRequest from "./websocket-request.js";
|
|
71
90
|
import { Logger } from "../../logger.js";
|
|
91
|
+
import WebsocketChannel from "../websocket-channel.js";
|
|
72
92
|
//# sourceMappingURL=websocket-session.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"websocket-session.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/websocket-session.js"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"websocket-session.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/websocket-session.js"],"names":[],"mappings":"AAcA;IAIE;;;;;OAKG;IACH,uDAJG;QAAuD,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QACL,MAAM,EAAzC,OAAO,YAAY,EAAE,OAAO;QACqD,cAAc,GAA/F,OAAO,cAAc,EAAE,OAAO,GAAG,OAAO,wBAAwB,EAAE,OAAO;KACnF,EAOA;IAfD,mEAA2B;IAC3B,wBAAyB;IASvB,4BAA6B;IAC7B,qCAAoB;IACpB,wDAAkC;IAClC,kEAAoC;IACpC,eAA8B;IAGhC;;;OAGG;IACH,yBAHW,MAAM,GACJ,IAAI,CAIhB;IAED,gBAGC;IAED;;;OAGG;IACH,yBAHW,MAAM,GACJ,OAAO,CAInB;IAED;;;OAGG;IACH,aAHW,MAAM,GACJ,IAAI,CAKhB;IAED;;;;OAIG;IACH,mBAJW,MAAM,WACN,GAAG,GACD,OAAO,CAAC,IAAI,CAAC,CAMzB;IAED;;OAEG;IACH,qBAFa,OAAO,CAAC,IAAI,CAAC,CA6BzB;IAZG,0BAEY;IAYhB;;;OAGG;IACH,oBAHW,OAAO,YAAY,EAAE,OAAO,GAC1B,IAAI,CAMhB;IAED;;;OAGG;IACH,wBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAkDzB;IAED;;OAEG;IACH,kBAFa,IAAI,CAuEhB;IAED;;;;OAIG;IACH,0BAJW,MAAM,WACN,MAAM,GACJ,IAAI,CAShB;IAED;;;OAGG;IACH,gBAHW,MAAM,GACJ,IAAI,CAuBhB;IAED;;;;OAIG;IACH,4BAJW,MAAM,oBACN;QAAC,WAAW,CAAC,EAAE,OAAO,CAAA;KAAC,GACrB,OAAO,CAAC,OAAO,CAAC,CAQ5B;IAED,qBAGC;IAED,kCAMC;IAED;;;;OAIG;IACH,wBAJW,MAAM,QACN,MAAM,GACJ,MAAM,CAWlB;CACF;6BAhU4B,wBAAwB;uBAFhC,iBAAiB;6BAGT,yBAAyB"}
|
|
@@ -3,6 +3,7 @@ import EventEmitter from "../../utils/event-emitter.js";
|
|
|
3
3
|
import { Logger } from "../../logger.js";
|
|
4
4
|
import RequestRunner from "./request-runner.js";
|
|
5
5
|
import WebsocketRequest from "./websocket-request.js";
|
|
6
|
+
import WebsocketChannel from "../websocket-channel.js";
|
|
6
7
|
const WEBSOCKET_FINAL_FRAME = 0x80;
|
|
7
8
|
const WEBSOCKET_OPCODE_TEXT = 0x1;
|
|
8
9
|
const WEBSOCKET_OPCODE_CLOSE = 0x8;
|
|
@@ -15,11 +16,13 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
15
16
|
* @param {object} args - Options object.
|
|
16
17
|
* @param {import("../../configuration.js").default} args.configuration - Configuration instance.
|
|
17
18
|
* @param {import("./index.js").default} args.client - Client instance.
|
|
19
|
+
* @param {import("./request.js").default | import("./websocket-request.js").default} [args.upgradeRequest] - Initial websocket upgrade request.
|
|
18
20
|
*/
|
|
19
|
-
constructor({ client, configuration }) {
|
|
21
|
+
constructor({ client, configuration, upgradeRequest }) {
|
|
20
22
|
this.buffer = Buffer.alloc(0);
|
|
21
23
|
this.client = client;
|
|
22
24
|
this.configuration = configuration;
|
|
25
|
+
this.upgradeRequest = upgradeRequest;
|
|
23
26
|
this.logger = new Logger(this);
|
|
24
27
|
}
|
|
25
28
|
/**
|
|
@@ -30,6 +33,7 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
30
33
|
this.subscriptions.add(channel);
|
|
31
34
|
}
|
|
32
35
|
destroy() {
|
|
36
|
+
void this._teardownChannel();
|
|
33
37
|
this.events.removeAllListeners();
|
|
34
38
|
}
|
|
35
39
|
/**
|
|
@@ -50,13 +54,41 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
50
54
|
/**
|
|
51
55
|
* @param {string} channel - Channel name.
|
|
52
56
|
* @param {any} payload - Payload data.
|
|
53
|
-
* @returns {void} -
|
|
57
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
54
58
|
*/
|
|
55
|
-
sendEvent(channel, payload) {
|
|
59
|
+
async sendEvent(channel, payload) {
|
|
56
60
|
if (!this.hasSubscription(channel))
|
|
57
61
|
return;
|
|
58
62
|
this._sendJson({ channel, payload, type: "event" });
|
|
59
63
|
}
|
|
64
|
+
/**
|
|
65
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
66
|
+
*/
|
|
67
|
+
async initializeChannel() {
|
|
68
|
+
const resolver = this.configuration.getWebsocketChannelResolver?.();
|
|
69
|
+
if (!resolver)
|
|
70
|
+
return;
|
|
71
|
+
try {
|
|
72
|
+
const resolved = await resolver({
|
|
73
|
+
client: this.client,
|
|
74
|
+
configuration: this.configuration,
|
|
75
|
+
request: this.upgradeRequest,
|
|
76
|
+
websocketSession: this
|
|
77
|
+
});
|
|
78
|
+
if (!resolved)
|
|
79
|
+
return;
|
|
80
|
+
this.channel = typeof resolved === "function"
|
|
81
|
+
? new resolved({ client: this.client, configuration: this.configuration, request: this.upgradeRequest, websocketSession: this })
|
|
82
|
+
: resolved;
|
|
83
|
+
if (this.channel && !(this.channel instanceof WebsocketChannel)) {
|
|
84
|
+
throw new Error("Resolved websocket channel must extend WebsocketChannel");
|
|
85
|
+
}
|
|
86
|
+
await this.channel?.subscribed?.();
|
|
87
|
+
}
|
|
88
|
+
catch (error) {
|
|
89
|
+
this.logger.error(() => ["Failed to initialize websocket channel", error]);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
60
92
|
/**
|
|
61
93
|
* @param {import("./index.js").default} client - Client instance.
|
|
62
94
|
* @returns {void} - No return value.
|
|
@@ -74,8 +106,7 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
74
106
|
const { channel } = message;
|
|
75
107
|
if (!channel)
|
|
76
108
|
throw new Error("channel is required for subscribe");
|
|
77
|
-
this.
|
|
78
|
-
this._sendJson({ channel, type: "subscribed" });
|
|
109
|
+
await this.subscribeToChannel(channel, { acknowledge: true });
|
|
79
110
|
return;
|
|
80
111
|
}
|
|
81
112
|
if (message.type && message.type !== "request") {
|
|
@@ -157,7 +188,7 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
157
188
|
continue;
|
|
158
189
|
}
|
|
159
190
|
if (opcode === WEBSOCKET_OPCODE_CLOSE) {
|
|
160
|
-
this.
|
|
191
|
+
this._handleClose();
|
|
161
192
|
this.sendGoodbye(this.client);
|
|
162
193
|
continue;
|
|
163
194
|
}
|
|
@@ -214,6 +245,30 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
214
245
|
header[0] = WEBSOCKET_FINAL_FRAME | WEBSOCKET_OPCODE_TEXT;
|
|
215
246
|
this.client.events.emit("output", Buffer.concat([header, payload]));
|
|
216
247
|
}
|
|
248
|
+
/**
|
|
249
|
+
* @param {string} channel - Channel name.
|
|
250
|
+
* @param {{acknowledge?: boolean}} [options] - Subscribe options.
|
|
251
|
+
* @returns {Promise<boolean>} - Whether the subscription was added.
|
|
252
|
+
*/
|
|
253
|
+
async subscribeToChannel(channel, { acknowledge = true } = {}) {
|
|
254
|
+
this.addSubscription(channel);
|
|
255
|
+
if (acknowledge) {
|
|
256
|
+
this._sendJson({ channel, type: "subscribed" });
|
|
257
|
+
}
|
|
258
|
+
return true;
|
|
259
|
+
}
|
|
260
|
+
_handleClose() {
|
|
261
|
+
void this._teardownChannel();
|
|
262
|
+
this.events.emit("close");
|
|
263
|
+
}
|
|
264
|
+
async _teardownChannel() {
|
|
265
|
+
try {
|
|
266
|
+
await this.channel?.unsubscribed?.();
|
|
267
|
+
}
|
|
268
|
+
catch (error) {
|
|
269
|
+
this.logger.error(() => ["Failed to teardown websocket channel", error]);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
217
272
|
/**
|
|
218
273
|
* @param {Buffer} payload - Payload data.
|
|
219
274
|
* @param {Buffer} mask - Mask.
|
|
@@ -228,4 +283,4 @@ export default class VelociousHttpServerClientWebsocketSession {
|
|
|
228
283
|
return result;
|
|
229
284
|
}
|
|
230
285
|
}
|
|
231
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-session.js","sourceRoot":"","sources":["../../../../src/http-server/client/websocket-session.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport {Logger} from \"../../logger.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketRequest from \"./websocket-request.js\"\n\nconst WEBSOCKET_FINAL_FRAME = 0x80\nconst WEBSOCKET_OPCODE_TEXT = 0x1\nconst WEBSOCKET_OPCODE_CLOSE = 0x8\nconst WEBSOCKET_OPCODE_PING = 0x9\nconst WEBSOCKET_OPCODE_PONG = 0xA\n\nexport default class VelociousHttpServerClientWebsocketSession {\n  events = new EventEmitter()\n  subscriptions = new Set()\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"./index.js\").default} args.client - Client instance.\n   */\n  constructor({client, configuration}) {\n    this.buffer = Buffer.alloc(0)\n    this.client = client\n    this.configuration = configuration\n    this.logger = new Logger(this)\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {void} - No return value.\n   */\n  addSubscription(channel) {\n    this.subscriptions.add(channel)\n  }\n\n  destroy() {\n    this.events.removeAllListeners()\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {boolean} - Whether it has subscription.\n   */\n  hasSubscription(channel) {\n    return this.subscriptions.has(channel)\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onData(data) {\n    this.buffer = Buffer.concat([this.buffer, data])\n    this._processBuffer()\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @param {any} payload - Payload data.\n   * @returns {void} - No return value.\n   */\n  sendEvent(channel, payload) {\n    if (!this.hasSubscription(channel)) return\n\n    this._sendJson({channel, payload, type: \"event\"})\n  }\n\n  /**\n   * @param {import(\"./index.js\").default} client - Client instance.\n   * @returns {void} - No return value.\n   */\n  sendGoodbye(client) {\n    const frame = Buffer.from([WEBSOCKET_FINAL_FRAME | WEBSOCKET_OPCODE_CLOSE, 0x00])\n\n    client.events.emit(\"output\", frame)\n  }\n\n  /**\n   * @param {object} message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _handleMessage(message) {\n    if (message.type === \"subscribe\") {\n      const {channel} = message\n\n      if (!channel) throw new Error(\"channel is required for subscribe\")\n\n      this.addSubscription(channel)\n      this._sendJson({channel, type: \"subscribed\"})\n\n      return\n    }\n\n    if (message.type && message.type !== \"request\") {\n      this._sendJson({error: `Unknown message type: ${message.type}`, type: \"error\"})\n      return\n    }\n\n    const {body, headers, id, method, path} = message\n\n    if (!method) throw new Error(\"method is required\")\n    if (!path) throw new Error(\"path is required\")\n\n    const request = new WebsocketRequest({\n      body,\n      headers,\n      method,\n      path,\n      remoteAddress: this.client.remoteAddress\n    })\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request\n    })\n\n    requestRunner.events.on(\"done\", () => {\n      const response = requestRunner.response\n      const body = response.getBody()\n      const headers = response.headers\n\n      this._sendJson({\n        body,\n        headers,\n        id,\n        statusCode: response.getStatusCode(),\n        statusMessage: response.getStatusMessage(),\n        type: \"response\"\n      })\n    })\n\n    await requestRunner.run()\n  }\n\n  /**\n   * @returns {void} - No return value.\n   */\n  _processBuffer() {\n    while (this.buffer.length >= 2) {\n      const firstByte = this.buffer[0]\n      const secondByte = this.buffer[1]\n      const isFinal = (firstByte & WEBSOCKET_FINAL_FRAME) === WEBSOCKET_FINAL_FRAME\n      const opcode = firstByte & 0x0F\n      const isMasked = (secondByte & 0x80) === 0x80\n      let payloadLength = secondByte & 0x7F\n      let offset = 2\n\n      if (payloadLength === 126) {\n        if (this.buffer.length < offset + 2) return\n        payloadLength = this.buffer.readUInt16BE(offset)\n        offset += 2\n      } else if (payloadLength === 127) {\n        if (this.buffer.length < offset + 8) return\n        const bigLength = this.buffer.readBigUInt64BE(offset)\n\n        payloadLength = Number(bigLength)\n        offset += 8\n      }\n\n      const maskLength = isMasked ? 4 : 0\n\n      if (this.buffer.length < offset + maskLength + payloadLength) return\n\n      /** @type {Buffer} */\n      let payload = this.buffer.slice(offset + maskLength, offset + maskLength + payloadLength)\n\n      if (isMasked) {\n        const mask = this.buffer.slice(offset, offset + maskLength)\n        payload = this._unmaskPayload(payload, mask)\n      }\n\n      this.buffer = this.buffer.slice(offset + maskLength + payloadLength)\n\n      if (!isFinal) {\n        this.logger.warn(\"Fragmented frames are not supported yet\")\n        continue\n      }\n\n      if (opcode === WEBSOCKET_OPCODE_PING) {\n        this._sendControlFrame(WEBSOCKET_OPCODE_PONG, payload)\n        continue\n      }\n\n      if (opcode === WEBSOCKET_OPCODE_CLOSE) {\n        this.events.emit(\"close\")\n        this.sendGoodbye(this.client)\n        continue\n      }\n\n      if (opcode !== WEBSOCKET_OPCODE_TEXT) {\n        this.logger.warn(`Unsupported websocket opcode: ${opcode}`)\n        continue\n      }\n\n      try {\n        const message = JSON.parse(payload.toString(\"utf-8\"))\n\n        this._handleMessage(message).catch((error) => {\n          this.logger.error(() => [\"Websocket message handler failed\", error])\n          this._sendJson({error: error.message, type: \"error\"})\n        })\n      } catch (error) {\n        this.logger.error(() => [\"Failed to parse websocket message\", error])\n        this._sendJson({error: \"Invalid websocket message\", type: \"error\"})\n      }\n    }\n  }\n\n  /**\n   * @param {number} opcode - Opcode.\n   * @param {Buffer} payload - Payload data.\n   * @returns {void} - No return value.\n   */\n  _sendControlFrame(opcode, payload) {\n    const header = Buffer.alloc(2)\n\n    header[0] = WEBSOCKET_FINAL_FRAME | opcode\n    header[1] = payload.length\n\n    this.client.events.emit(\"output\", Buffer.concat([header, payload]))\n  }\n\n  /**\n   * @param {object} body - Request body.\n   * @returns {void} - No return value.\n   */\n  _sendJson(body) {\n    const json = JSON.stringify(body)\n    const payload = Buffer.from(json, \"utf-8\")\n    let header\n\n    if (payload.length < 126) {\n      header = Buffer.alloc(2)\n      header[1] = payload.length\n    } else if (payload.length < 65536) {\n      header = Buffer.alloc(4)\n      header[1] = 126\n      header.writeUInt16BE(payload.length, 2)\n    } else {\n      header = Buffer.alloc(10)\n      header[1] = 127\n      header.writeBigUInt64BE(BigInt(payload.length), 2)\n    }\n\n    header[0] = WEBSOCKET_FINAL_FRAME | WEBSOCKET_OPCODE_TEXT\n\n    this.client.events.emit(\"output\", Buffer.concat([header, payload]))\n  }\n\n  /**\n   * @param {Buffer} payload - Payload data.\n   * @param {Buffer} mask - Mask.\n   * @returns {Buffer} - The unmask payload.\n   */\n  _unmaskPayload(payload, mask) {\n    /** @type {Buffer} */\n    const result = Buffer.alloc(payload.length)\n\n    for (let i = 0; i < payload.length; i++) {\n      result[i] = payload[i] ^ mask[i % 4]\n    }\n\n    return result\n  }\n}\n"]}
|
|
286
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-session.js","sourceRoot":"","sources":["../../../../src/http-server/client/websocket-session.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport {Logger} from \"../../logger.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketRequest from \"./websocket-request.js\"\nimport WebsocketChannel from \"../websocket-channel.js\"\n\nconst WEBSOCKET_FINAL_FRAME = 0x80\nconst WEBSOCKET_OPCODE_TEXT = 0x1\nconst WEBSOCKET_OPCODE_CLOSE = 0x8\nconst WEBSOCKET_OPCODE_PING = 0x9\nconst WEBSOCKET_OPCODE_PONG = 0xA\n\nexport default class VelociousHttpServerClientWebsocketSession {\n  events = new EventEmitter()\n  subscriptions = new Set()\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"./index.js\").default} args.client - Client instance.\n   * @param {import(\"./request.js\").default | import(\"./websocket-request.js\").default} [args.upgradeRequest] - Initial websocket upgrade request.\n   */\n  constructor({client, configuration, upgradeRequest}) {\n    this.buffer = Buffer.alloc(0)\n    this.client = client\n    this.configuration = configuration\n    this.upgradeRequest = upgradeRequest\n    this.logger = new Logger(this)\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {void} - No return value.\n   */\n  addSubscription(channel) {\n    this.subscriptions.add(channel)\n  }\n\n  destroy() {\n    void this._teardownChannel()\n    this.events.removeAllListeners()\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {boolean} - Whether it has subscription.\n   */\n  hasSubscription(channel) {\n    return this.subscriptions.has(channel)\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onData(data) {\n    this.buffer = Buffer.concat([this.buffer, data])\n    this._processBuffer()\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @param {any} payload - Payload data.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async sendEvent(channel, payload) {\n    if (!this.hasSubscription(channel)) return\n\n    this._sendJson({channel, payload, type: \"event\"})\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async initializeChannel() {\n    const resolver = this.configuration.getWebsocketChannelResolver?.()\n\n    if (!resolver) return\n\n    try {\n      const resolved = await resolver({\n        client: this.client,\n        configuration: this.configuration,\n        request: this.upgradeRequest,\n        websocketSession: this\n      })\n\n      if (!resolved) return\n\n      this.channel = typeof resolved === \"function\"\n        ? new resolved({client: this.client, configuration: this.configuration, request: this.upgradeRequest, websocketSession: this})\n        : resolved\n\n      if (this.channel && !(this.channel instanceof WebsocketChannel)) {\n        throw new Error(\"Resolved websocket channel must extend WebsocketChannel\")\n      }\n\n      await this.channel?.subscribed?.()\n    } catch (error) {\n      this.logger.error(() => [\"Failed to initialize websocket channel\", error])\n    }\n  }\n\n  /**\n   * @param {import(\"./index.js\").default} client - Client instance.\n   * @returns {void} - No return value.\n   */\n  sendGoodbye(client) {\n    const frame = Buffer.from([WEBSOCKET_FINAL_FRAME | WEBSOCKET_OPCODE_CLOSE, 0x00])\n\n    client.events.emit(\"output\", frame)\n  }\n\n  /**\n   * @param {object} message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _handleMessage(message) {\n    if (message.type === \"subscribe\") {\n      const {channel} = message\n\n      if (!channel) throw new Error(\"channel is required for subscribe\")\n      await this.subscribeToChannel(channel, {acknowledge: true})\n\n      return\n    }\n\n    if (message.type && message.type !== \"request\") {\n      this._sendJson({error: `Unknown message type: ${message.type}`, type: \"error\"})\n      return\n    }\n\n    const {body, headers, id, method, path} = message\n\n    if (!method) throw new Error(\"method is required\")\n    if (!path) throw new Error(\"path is required\")\n\n    const request = new WebsocketRequest({\n      body,\n      headers,\n      method,\n      path,\n      remoteAddress: this.client.remoteAddress\n    })\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request\n    })\n\n    requestRunner.events.on(\"done\", () => {\n      const response = requestRunner.response\n      const body = response.getBody()\n      const headers = response.headers\n\n      this._sendJson({\n        body,\n        headers,\n        id,\n        statusCode: response.getStatusCode(),\n        statusMessage: response.getStatusMessage(),\n        type: \"response\"\n      })\n    })\n\n    await requestRunner.run()\n  }\n\n  /**\n   * @returns {void} - No return value.\n   */\n  _processBuffer() {\n    while (this.buffer.length >= 2) {\n      const firstByte = this.buffer[0]\n      const secondByte = this.buffer[1]\n      const isFinal = (firstByte & WEBSOCKET_FINAL_FRAME) === WEBSOCKET_FINAL_FRAME\n      const opcode = firstByte & 0x0F\n      const isMasked = (secondByte & 0x80) === 0x80\n      let payloadLength = secondByte & 0x7F\n      let offset = 2\n\n      if (payloadLength === 126) {\n        if (this.buffer.length < offset + 2) return\n        payloadLength = this.buffer.readUInt16BE(offset)\n        offset += 2\n      } else if (payloadLength === 127) {\n        if (this.buffer.length < offset + 8) return\n        const bigLength = this.buffer.readBigUInt64BE(offset)\n\n        payloadLength = Number(bigLength)\n        offset += 8\n      }\n\n      const maskLength = isMasked ? 4 : 0\n\n      if (this.buffer.length < offset + maskLength + payloadLength) return\n\n      /** @type {Buffer} */\n      let payload = this.buffer.slice(offset + maskLength, offset + maskLength + payloadLength)\n\n      if (isMasked) {\n        const mask = this.buffer.slice(offset, offset + maskLength)\n        payload = this._unmaskPayload(payload, mask)\n      }\n\n      this.buffer = this.buffer.slice(offset + maskLength + payloadLength)\n\n      if (!isFinal) {\n        this.logger.warn(\"Fragmented frames are not supported yet\")\n        continue\n      }\n\n      if (opcode === WEBSOCKET_OPCODE_PING) {\n        this._sendControlFrame(WEBSOCKET_OPCODE_PONG, payload)\n        continue\n      }\n\n      if (opcode === WEBSOCKET_OPCODE_CLOSE) {\n        this._handleClose()\n        this.sendGoodbye(this.client)\n        continue\n      }\n\n      if (opcode !== WEBSOCKET_OPCODE_TEXT) {\n        this.logger.warn(`Unsupported websocket opcode: ${opcode}`)\n        continue\n      }\n\n      try {\n        const message = JSON.parse(payload.toString(\"utf-8\"))\n\n        this._handleMessage(message).catch((error) => {\n          this.logger.error(() => [\"Websocket message handler failed\", error])\n          this._sendJson({error: error.message, type: \"error\"})\n        })\n      } catch (error) {\n        this.logger.error(() => [\"Failed to parse websocket message\", error])\n        this._sendJson({error: \"Invalid websocket message\", type: \"error\"})\n      }\n    }\n  }\n\n  /**\n   * @param {number} opcode - Opcode.\n   * @param {Buffer} payload - Payload data.\n   * @returns {void} - No return value.\n   */\n  _sendControlFrame(opcode, payload) {\n    const header = Buffer.alloc(2)\n\n    header[0] = WEBSOCKET_FINAL_FRAME | opcode\n    header[1] = payload.length\n\n    this.client.events.emit(\"output\", Buffer.concat([header, payload]))\n  }\n\n  /**\n   * @param {object} body - Request body.\n   * @returns {void} - No return value.\n   */\n  _sendJson(body) {\n    const json = JSON.stringify(body)\n    const payload = Buffer.from(json, \"utf-8\")\n    let header\n\n    if (payload.length < 126) {\n      header = Buffer.alloc(2)\n      header[1] = payload.length\n    } else if (payload.length < 65536) {\n      header = Buffer.alloc(4)\n      header[1] = 126\n      header.writeUInt16BE(payload.length, 2)\n    } else {\n      header = Buffer.alloc(10)\n      header[1] = 127\n      header.writeBigUInt64BE(BigInt(payload.length), 2)\n    }\n\n    header[0] = WEBSOCKET_FINAL_FRAME | WEBSOCKET_OPCODE_TEXT\n\n    this.client.events.emit(\"output\", Buffer.concat([header, payload]))\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @param {{acknowledge?: boolean}} [options] - Subscribe options.\n   * @returns {Promise<boolean>} - Whether the subscription was added.\n   */\n  async subscribeToChannel(channel, {acknowledge = true} = {}) {\n    this.addSubscription(channel)\n    if (acknowledge) {\n      this._sendJson({channel, type: \"subscribed\"})\n    }\n    return true\n  }\n\n  _handleClose() {\n    void this._teardownChannel()\n    this.events.emit(\"close\")\n  }\n\n  async _teardownChannel() {\n    try {\n      await this.channel?.unsubscribed?.()\n    } catch (error) {\n      this.logger.error(() => [\"Failed to teardown websocket channel\", error])\n    }\n  }\n\n  /**\n   * @param {Buffer} payload - Payload data.\n   * @param {Buffer} mask - Mask.\n   * @returns {Buffer} - The unmask payload.\n   */\n  _unmaskPayload(payload, mask) {\n    /** @type {Buffer} */\n    const result = Buffer.alloc(payload.length)\n\n    for (let i = 0; i < payload.length; i++) {\n      result[i] = payload[i] ^ mask[i % 4]\n    }\n\n    return result\n  }\n}\n"]}
|