@api-client/core 0.3.5 → 0.3.6
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/build/browser.d.ts +2 -0
- package/build/browser.js +8 -0
- package/build/browser.js.map +1 -1
- package/build/index.d.ts +10 -1
- package/build/index.js +19 -1
- package/build/index.js.map +1 -1
- package/build/src/lib/fs/Fs.d.ts +52 -0
- package/build/src/lib/fs/Fs.js +245 -0
- package/build/src/lib/fs/Fs.js.map +1 -0
- package/build/src/lib/timers/Timers.d.ts +5 -0
- package/build/src/lib/timers/Timers.js +10 -0
- package/build/src/lib/timers/Timers.js.map +1 -0
- package/build/src/mocking/ProjectMock.d.ts +13 -0
- package/build/src/mocking/ProjectMock.js +16 -0
- package/build/src/mocking/ProjectMock.js.map +1 -0
- package/build/src/mocking/lib/Request.d.ts +32 -0
- package/build/src/mocking/lib/Request.js +63 -0
- package/build/src/mocking/lib/Request.js.map +1 -0
- package/build/src/mocking/lib/Response.d.ts +33 -0
- package/build/src/mocking/lib/Response.js +79 -0
- package/build/src/mocking/lib/Response.js.map +1 -0
- package/build/src/runtime/node/BaseRunner.d.ts +21 -0
- package/build/src/runtime/node/BaseRunner.js +27 -0
- package/build/src/runtime/node/BaseRunner.js.map +1 -0
- package/build/src/runtime/node/ProjectParallelRunner.d.ts +81 -0
- package/build/src/runtime/node/ProjectParallelRunner.js +173 -0
- package/build/src/runtime/node/ProjectParallelRunner.js.map +1 -0
- package/build/src/runtime/node/ProjectRequestRunner.d.ts +125 -0
- package/build/src/runtime/node/ProjectRequestRunner.js +185 -0
- package/build/src/runtime/node/ProjectRequestRunner.js.map +1 -0
- package/build/src/runtime/node/ProjectRunner.d.ts +164 -62
- package/build/src/runtime/node/ProjectRunner.js +191 -146
- package/build/src/runtime/node/ProjectRunner.js.map +1 -1
- package/build/src/runtime/node/ProjectRunnerWorker.d.ts +1 -0
- package/build/src/runtime/node/ProjectRunnerWorker.js +58 -0
- package/build/src/runtime/node/ProjectRunnerWorker.js.map +1 -0
- package/build/src/runtime/node/ProjectSerialRunner.d.ts +11 -0
- package/build/src/runtime/node/ProjectSerialRunner.js +34 -0
- package/build/src/runtime/node/ProjectSerialRunner.js.map +1 -0
- package/build/src/runtime/reporters/ProjectRunCliReporter.d.ts +7 -0
- package/build/src/runtime/reporters/ProjectRunCliReporter.js +73 -0
- package/build/src/runtime/reporters/ProjectRunCliReporter.js.map +1 -0
- package/build/src/runtime/reporters/Reporter.d.ts +62 -0
- package/build/src/runtime/reporters/Reporter.js +98 -0
- package/build/src/runtime/reporters/Reporter.js.map +1 -0
- package/build/src/testing/TestCliHelper.d.ts +23 -0
- package/build/src/testing/TestCliHelper.js +71 -0
- package/build/src/testing/TestCliHelper.js.map +1 -0
- package/build/src/testing/getPort.d.ts +52 -0
- package/build/src/testing/getPort.js +169 -0
- package/build/src/testing/getPort.js.map +1 -0
- package/package.json +2 -1
- package/src/lib/fs/Fs.ts +258 -0
- package/src/lib/timers/Timers.ts +9 -0
- package/src/mocking/LegacyInterfaces.ts +1 -1
- package/src/mocking/ProjectMock.ts +20 -0
- package/src/mocking/lib/Request.ts +85 -0
- package/src/mocking/lib/Response.ts +101 -0
- package/src/runtime/node/BaseRunner.ts +29 -0
- package/src/runtime/node/ProjectParallelRunner.ts +234 -0
- package/src/runtime/node/ProjectRequestRunner.ts +281 -0
- package/src/runtime/node/ProjectRunner.ts +279 -186
- package/src/runtime/node/ProjectRunnerWorker.ts +62 -0
- package/src/runtime/node/ProjectSerialRunner.ts +36 -0
- package/src/runtime/reporters/ProjectRunCliReporter.ts +79 -0
- package/src/runtime/reporters/Reporter.ts +142 -0
- package/src/testing/TestCliHelper.ts +76 -0
- package/src/testing/getPort.ts +212 -0
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { Table } from 'console-table-printer';
|
|
2
|
+
import chalk from 'chalk';
|
|
3
|
+
import { ErrorResponse, IErrorResponse } from '../../models/ErrorResponse.js';
|
|
4
|
+
import { IArcResponse } from '../../models/ArcResponse.js';
|
|
5
|
+
import { ISerializedError } from '../../models/SerializableError.js';
|
|
6
|
+
import { Reporter } from './Reporter.js';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* HTTP project execution reporter for a terminal output.
|
|
10
|
+
*/
|
|
11
|
+
export class ProjectRunCliReporter extends Reporter {
|
|
12
|
+
async generate(): Promise<void> {
|
|
13
|
+
const { info } = this;
|
|
14
|
+
|
|
15
|
+
const table = new Table({
|
|
16
|
+
title: 'Project execution summary',
|
|
17
|
+
columns: [
|
|
18
|
+
{ name: 'position', title: ' ', alignment: 'left', },
|
|
19
|
+
{ name: 'succeeded', title: 'Succeeded', alignment: 'right', },
|
|
20
|
+
{ name: 'failed', title: 'Failed', alignment: 'right', },
|
|
21
|
+
{ name: 'total', title: 'Total', alignment: 'right', },
|
|
22
|
+
],
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
table.addRow({
|
|
26
|
+
position: 'Iterations',
|
|
27
|
+
succeeded: info.iterations.length,
|
|
28
|
+
failed: 0,
|
|
29
|
+
total: info.iterations.length,
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
const failed = this.computeFailed();
|
|
33
|
+
const succeeded = this.computeSucceeded();
|
|
34
|
+
table.addRow({
|
|
35
|
+
position: 'Requests',
|
|
36
|
+
succeeded,
|
|
37
|
+
failed: failed > 0 ? chalk.redBright(failed) : failed,
|
|
38
|
+
total: failed + succeeded,
|
|
39
|
+
});
|
|
40
|
+
table.printTable();
|
|
41
|
+
process.stdout.write('\n');
|
|
42
|
+
|
|
43
|
+
info.iterations.forEach((run, index) => {
|
|
44
|
+
const itNumber = index + 1;
|
|
45
|
+
if (run.error) {
|
|
46
|
+
process.stdout.write(`Iteration ${itNumber} failed: ${run.error}\n\n`);
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
const failed = run.executed.filter(log => this.isFailedLog(log));
|
|
50
|
+
if (!failed.length) {
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
process.stdout.write(`Iteration ${itNumber} Errors\n`);
|
|
54
|
+
failed.forEach((log) => {
|
|
55
|
+
let url = 'Unknown request URL.';
|
|
56
|
+
if (log.request) {
|
|
57
|
+
url = log.request.url;
|
|
58
|
+
}
|
|
59
|
+
const prefix = chalk.dim(`[${url}] `);
|
|
60
|
+
if (log.response && ErrorResponse.isErrorResponse(log.response)) {
|
|
61
|
+
const response = log.response as IErrorResponse;
|
|
62
|
+
let message = (response.error as ISerializedError).message ? (response.error as Error).message : response.error;
|
|
63
|
+
if (typeof message !== 'string') {
|
|
64
|
+
message = 'Unknown error.';
|
|
65
|
+
}
|
|
66
|
+
process.stdout.write(`${prefix}${message}\n`);
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
if (!log.request) {
|
|
70
|
+
process.stdout.write('Request not executed.\n');
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
const response = log.response as IArcResponse;
|
|
74
|
+
process.stdout.write(`${prefix} Status code is: ${response.status}\n`);
|
|
75
|
+
});
|
|
76
|
+
process.stdout.write('\n\n');
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
}
|
|
@@ -0,0 +1,142 @@
|
|
|
1
|
+
import { IRequestLog } from '../../models/RequestLog.js';
|
|
2
|
+
import { IArcResponse } from '../../models/ArcResponse.js';
|
|
3
|
+
import { ErrorResponse } from '../../models/ErrorResponse.js';
|
|
4
|
+
|
|
5
|
+
export interface IProjectExecutionIteration {
|
|
6
|
+
/**
|
|
7
|
+
* The index of the iteration.
|
|
8
|
+
*/
|
|
9
|
+
index: number;
|
|
10
|
+
/**
|
|
11
|
+
* The list of requests executed in the iteration.
|
|
12
|
+
*/
|
|
13
|
+
executed: IRequestLog[];
|
|
14
|
+
/**
|
|
15
|
+
* Optional general error message.
|
|
16
|
+
*/
|
|
17
|
+
error?: string;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export interface IProjectExecutionLog {
|
|
21
|
+
/**
|
|
22
|
+
* The timestamp when the execution started
|
|
23
|
+
*/
|
|
24
|
+
started: number;
|
|
25
|
+
/**
|
|
26
|
+
* The timestamp when the execution ended
|
|
27
|
+
*/
|
|
28
|
+
ended: number;
|
|
29
|
+
/**
|
|
30
|
+
* The execution logs for each iteration.
|
|
31
|
+
*/
|
|
32
|
+
iterations: IProjectExecutionIteration[];
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Base class for project execution reporters.
|
|
37
|
+
*/
|
|
38
|
+
export abstract class Reporter {
|
|
39
|
+
info: IProjectExecutionLog;
|
|
40
|
+
|
|
41
|
+
constructor(info: IProjectExecutionLog) {
|
|
42
|
+
this.info = info;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Generates the report for the current execution log.
|
|
47
|
+
*/
|
|
48
|
+
abstract generate(): Promise<void>;
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Checks whether the execution log should be considered a failure.
|
|
52
|
+
* @param log The execution log.
|
|
53
|
+
* @returns `true` when the request was a failure.
|
|
54
|
+
*/
|
|
55
|
+
protected isFailedLog(log: IRequestLog): boolean {
|
|
56
|
+
if (!log.response || ErrorResponse.isErrorResponse(log.response)) {
|
|
57
|
+
return true;
|
|
58
|
+
}
|
|
59
|
+
const response = log.response as IArcResponse;
|
|
60
|
+
if (response.status >= 400) {
|
|
61
|
+
return true;
|
|
62
|
+
}
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Computes the number of requests that failed.
|
|
68
|
+
*/
|
|
69
|
+
computeFailed(): number {
|
|
70
|
+
let result = 0;
|
|
71
|
+
const { info } = this;
|
|
72
|
+
const { iterations } = info;
|
|
73
|
+
iterations.forEach((iteration) => {
|
|
74
|
+
iteration.executed.forEach((log) => {
|
|
75
|
+
if (this.isFailedLog(log)) {
|
|
76
|
+
result++;
|
|
77
|
+
}
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
return result;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Computes the number of requests that ended with the status code 399 at the most.
|
|
86
|
+
*/
|
|
87
|
+
computeSucceeded(): number {
|
|
88
|
+
let result = 0;
|
|
89
|
+
const { info } = this;
|
|
90
|
+
const { iterations } = info;
|
|
91
|
+
iterations.forEach((iteration) => {
|
|
92
|
+
iteration.executed.forEach((log) => {
|
|
93
|
+
if (!log.response || ErrorResponse.isErrorResponse(log.response)) {
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
const response = log.response as IArcResponse;
|
|
97
|
+
if (response.status < 400) {
|
|
98
|
+
result++;
|
|
99
|
+
}
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
return result;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Computes the total time of sending each request.
|
|
107
|
+
*/
|
|
108
|
+
computeTotalTime(): number {
|
|
109
|
+
let result = 0;
|
|
110
|
+
const { info } = this;
|
|
111
|
+
const { iterations } = info;
|
|
112
|
+
iterations.forEach((iteration) => {
|
|
113
|
+
iteration.executed.forEach((log) => {
|
|
114
|
+
if (!log.response || ErrorResponse.isErrorResponse(log.response)) {
|
|
115
|
+
return;
|
|
116
|
+
}
|
|
117
|
+
const response = log.response as IArcResponse;
|
|
118
|
+
if (response.loadingTime && response.loadingTime > 0) {
|
|
119
|
+
result += response.loadingTime;
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
return result;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Computes the total size of received data.
|
|
128
|
+
*/
|
|
129
|
+
computeTotalSize(): number {
|
|
130
|
+
let result = 0;
|
|
131
|
+
const { info } = this;
|
|
132
|
+
const { iterations } = info;
|
|
133
|
+
iterations.forEach((iteration) => {
|
|
134
|
+
iteration.executed.forEach((log) => {
|
|
135
|
+
if (log.size && log.size.response) {
|
|
136
|
+
result += log.size.response;
|
|
137
|
+
}
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
return result;
|
|
141
|
+
}
|
|
142
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
export interface ITestRunCommandOptions {
|
|
2
|
+
includeError?: boolean;
|
|
3
|
+
noCleaning?: boolean;
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
export class TestCliHelper {
|
|
7
|
+
static cleanTerminalOutput(s: string): string {
|
|
8
|
+
let result = s.trim();
|
|
9
|
+
result = result.replace(/[^\x20-\x7E\n]/gm, '');
|
|
10
|
+
// result = result.replace(/\[\d+m/gm, '');
|
|
11
|
+
result = result.replace(/\[\d+[a-zA-Z]/gm, '');
|
|
12
|
+
result = result.split('\n').filter(i => !!i.trim()).join('\n');
|
|
13
|
+
return result;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
static splitLines(table: string): string[] {
|
|
17
|
+
const result: string[] = [];
|
|
18
|
+
table.split('\n').forEach((line) => {
|
|
19
|
+
const value = line.trim();
|
|
20
|
+
if (!value) {
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
result.push(value);
|
|
24
|
+
});
|
|
25
|
+
return result;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Executes a passed asynchronous function and captures stdout.
|
|
30
|
+
* When the function fails, it cleans up output listeners and throws the error.
|
|
31
|
+
*
|
|
32
|
+
* ```javascript
|
|
33
|
+
* const out = grabOutput(async () => {
|
|
34
|
+
* // ...
|
|
35
|
+
* });
|
|
36
|
+
* console.log(out); // combined stdout and stderr.
|
|
37
|
+
* ```
|
|
38
|
+
*
|
|
39
|
+
* @param fn The function to execute.
|
|
40
|
+
* @returns The terminal output.
|
|
41
|
+
*/
|
|
42
|
+
static async grabOutput(fn: () => Promise<void>): Promise<string> {
|
|
43
|
+
const messages: string[] = [];
|
|
44
|
+
function noop(): void {
|
|
45
|
+
//
|
|
46
|
+
}
|
|
47
|
+
const origOut = process.stdout.write;
|
|
48
|
+
const origErr = process.stderr.write;
|
|
49
|
+
const origClear = console.clear;
|
|
50
|
+
function messageHandler(buffer: string | Buffer): boolean {
|
|
51
|
+
if (typeof buffer === 'string') {
|
|
52
|
+
messages.push(buffer);
|
|
53
|
+
} else {
|
|
54
|
+
messages.push(buffer.toString('utf8'));
|
|
55
|
+
}
|
|
56
|
+
return true;
|
|
57
|
+
}
|
|
58
|
+
function stop(): void {
|
|
59
|
+
process.stdout.write = origOut;
|
|
60
|
+
process.stderr.write = origErr;
|
|
61
|
+
console.clear = origClear;
|
|
62
|
+
}
|
|
63
|
+
process.stdout.write = messageHandler;
|
|
64
|
+
process.stderr.write = messageHandler;
|
|
65
|
+
console.clear = noop;
|
|
66
|
+
|
|
67
|
+
try {
|
|
68
|
+
await fn();
|
|
69
|
+
stop();
|
|
70
|
+
} catch (e) {
|
|
71
|
+
stop();
|
|
72
|
+
throw e;
|
|
73
|
+
}
|
|
74
|
+
return messages.join('');
|
|
75
|
+
}
|
|
76
|
+
}
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
import net from 'net';
|
|
2
|
+
import os from 'os';
|
|
3
|
+
|
|
4
|
+
/* global NodeJS */
|
|
5
|
+
|
|
6
|
+
export interface IGetPortOptions extends Omit<net.ListenOptions, 'port'> {
|
|
7
|
+
/**
|
|
8
|
+
A preferred port or an iterable of preferred ports to use.
|
|
9
|
+
*/
|
|
10
|
+
readonly port?: number | Iterable<number>;
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
The host on which port resolution should be performed. Can be either an IPv4 or IPv6 address.
|
|
14
|
+
|
|
15
|
+
By default, it checks availability on all local addresses defined in [OS network interfaces](https://nodejs.org/api/os.html#os_os_networkinterfaces). If this option is set, it will only check the given host.
|
|
16
|
+
*/
|
|
17
|
+
readonly host?: string;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
class Locked extends Error {
|
|
21
|
+
constructor(port = 0) {
|
|
22
|
+
super(`${port} is locked`);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
const lockedPorts = {
|
|
27
|
+
old: new Set(),
|
|
28
|
+
young: new Set(),
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
// On this interval, the old locked ports are discarded,
|
|
32
|
+
// the young locked ports are moved to old locked ports,
|
|
33
|
+
// and a new young set for locked ports are created.
|
|
34
|
+
const releaseOldLockedPortsIntervalMs = 1000 * 15;
|
|
35
|
+
|
|
36
|
+
// Lazily create interval on first use
|
|
37
|
+
let interval:NodeJS.Timer;
|
|
38
|
+
|
|
39
|
+
const getLocalHosts = (): Set<string | undefined> => {
|
|
40
|
+
const interfaces = os.networkInterfaces();
|
|
41
|
+
|
|
42
|
+
// Add undefined value for createServer function to use default host,
|
|
43
|
+
// and default IPv4 host in case createServer defaults to IPv6.
|
|
44
|
+
const results = new Set([undefined, '0.0.0.0']);
|
|
45
|
+
|
|
46
|
+
for (const _interface of Object.values(interfaces)) {
|
|
47
|
+
if (_interface) {
|
|
48
|
+
for (const config of _interface) {
|
|
49
|
+
results.add(config.address);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
return results;
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
const checkAvailablePort = (options: net.ListenOptions): Promise<number> =>
|
|
58
|
+
new Promise((resolve, reject) => {
|
|
59
|
+
const server = net.createServer();
|
|
60
|
+
server.unref();
|
|
61
|
+
server.on('error', reject);
|
|
62
|
+
|
|
63
|
+
server.listen(options, () => {
|
|
64
|
+
const {port} = server.address() as net.AddressInfo;
|
|
65
|
+
server.close(() => {
|
|
66
|
+
resolve(port);
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
const getAvailablePort = async (options: net.ListenOptions, hosts: Set<string | undefined>): Promise<number | undefined> => {
|
|
72
|
+
if (options.host || options.port === 0) {
|
|
73
|
+
return checkAvailablePort(options);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
for (const host of hosts) {
|
|
77
|
+
try {
|
|
78
|
+
await checkAvailablePort({port: options.port, host}); // eslint-disable-line no-await-in-loop
|
|
79
|
+
} catch (error) {
|
|
80
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
81
|
+
const typed = error as any;
|
|
82
|
+
if (!['EADDRNOTAVAIL', 'EINVAL'].includes(typed.code)) {
|
|
83
|
+
throw typed;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
return options.port;
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
const portCheckSequence = function * (ports?: number[] | Iterable<number>): IterableIterator<number|undefined> {
|
|
92
|
+
if (ports) {
|
|
93
|
+
yield * ports;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
yield 0; // Fall back to 0 if anything else failed
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
Get an available TCP port number.
|
|
101
|
+
|
|
102
|
+
@returns Port number.
|
|
103
|
+
|
|
104
|
+
@example
|
|
105
|
+
```
|
|
106
|
+
import getPort from 'get-port';
|
|
107
|
+
|
|
108
|
+
console.log(await getPort());
|
|
109
|
+
//=> 51402
|
|
110
|
+
|
|
111
|
+
// Pass in a preferred port
|
|
112
|
+
console.log(await getPort({port: 3000}));
|
|
113
|
+
// Will use 3000 if available, otherwise fall back to a random port
|
|
114
|
+
|
|
115
|
+
// Pass in an array of preferred ports
|
|
116
|
+
console.log(await getPort({port: [3000, 3001, 3002]}));
|
|
117
|
+
// Will use any element in the preferred ports array if available, otherwise fall back to a random port
|
|
118
|
+
```
|
|
119
|
+
*/
|
|
120
|
+
export async function getPort(options?: IGetPortOptions): Promise<number> {
|
|
121
|
+
let ports: number[] | Iterable<number> = [];
|
|
122
|
+
|
|
123
|
+
if (options) {
|
|
124
|
+
if (typeof options.port === 'number') {
|
|
125
|
+
ports = [options.port];
|
|
126
|
+
} else if (options.port) {
|
|
127
|
+
ports = options.port;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
if (interval === undefined) {
|
|
132
|
+
interval = setInterval(() => {
|
|
133
|
+
lockedPorts.old = lockedPorts.young;
|
|
134
|
+
lockedPorts.young = new Set();
|
|
135
|
+
}, releaseOldLockedPortsIntervalMs);
|
|
136
|
+
|
|
137
|
+
// Does not exist in some environments (Electron, Jest jsdom env, browser, etc).
|
|
138
|
+
if (interval.unref) {
|
|
139
|
+
interval.unref();
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
const hosts = getLocalHosts();
|
|
144
|
+
|
|
145
|
+
for (const port of portCheckSequence(ports)) {
|
|
146
|
+
try {
|
|
147
|
+
let availablePort = await getAvailablePort({...options, port}, hosts); // eslint-disable-line no-await-in-loop
|
|
148
|
+
while (lockedPorts.old.has(availablePort) || lockedPorts.young.has(availablePort)) {
|
|
149
|
+
if (port !== 0) {
|
|
150
|
+
throw new Locked(port);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
availablePort = await getAvailablePort({...options, port}, hosts); // eslint-disable-line no-await-in-loop
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
lockedPorts.young.add(availablePort);
|
|
157
|
+
if (!availablePort) {
|
|
158
|
+
throw new Error('No available ports found');
|
|
159
|
+
}
|
|
160
|
+
return availablePort;
|
|
161
|
+
} catch (error) {
|
|
162
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
163
|
+
const typed = error as any;
|
|
164
|
+
if (!['EADDRINUSE', 'EACCES'].includes(typed.code) && !(typed instanceof Locked)) {
|
|
165
|
+
throw error;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
throw new Error('No available ports found');
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
Generate port numbers in the given range `from`...`to`.
|
|
175
|
+
|
|
176
|
+
@param from - The first port of the range. Must be in the range `1024`...`65535`.
|
|
177
|
+
@param to - The last port of the range. Must be in the range `1024`...`65535` and must be greater than `from`.
|
|
178
|
+
@returns The port numbers in the range.
|
|
179
|
+
|
|
180
|
+
@example
|
|
181
|
+
```
|
|
182
|
+
import getPort, {portNumbers} from 'get-port';
|
|
183
|
+
|
|
184
|
+
console.log(await getPort({port: portNumbers(3000, 3100)}));
|
|
185
|
+
// Will use any port from 3000 to 3100, otherwise fall back to a random port
|
|
186
|
+
```
|
|
187
|
+
*/
|
|
188
|
+
export function portNumbers(from: number, to: number): Iterable<number> {
|
|
189
|
+
if (!Number.isInteger(from) || !Number.isInteger(to)) {
|
|
190
|
+
throw new TypeError('`from` and `to` must be integer numbers');
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
if (from < 1024 || from > 65_535) {
|
|
194
|
+
throw new RangeError('`from` must be between 1024 and 65535');
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
if (to < 1024 || to > 65_536) {
|
|
198
|
+
throw new RangeError('`to` must be between 1024 and 65536');
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
if (to < from) {
|
|
202
|
+
throw new RangeError('`to` must be greater than or equal to `from`');
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
const generator = function * (from: number, to: number): IterableIterator<number> {
|
|
206
|
+
for (let port = from; port <= to; port++) {
|
|
207
|
+
yield port;
|
|
208
|
+
}
|
|
209
|
+
};
|
|
210
|
+
|
|
211
|
+
return generator(from, to);
|
|
212
|
+
}
|