|
| 1 | +import { Net } from "../../../services/net-service"; |
| 2 | +import { assert } from "chai"; |
| 3 | +import { Yok } from "../../../yok"; |
| 4 | +import { ErrorsStub, CommonLoggerStub } from "../stubs"; |
| 5 | +import { EOL } from "os"; |
| 6 | + |
| 7 | +describe("net", () => { |
| 8 | + const createTestInjector = (platform: string): IInjector => { |
| 9 | + const testInjector = new Yok(); |
| 10 | + testInjector.register("errors", ErrorsStub); |
| 11 | + testInjector.register("childProcess", {}); |
| 12 | + testInjector.register("logger", CommonLoggerStub); |
| 13 | + testInjector.register("osInfo", { |
| 14 | + platform: () => platform |
| 15 | + }); |
| 16 | + |
| 17 | + return testInjector; |
| 18 | + }; |
| 19 | + |
| 20 | + describe("waitForPortToListen", () => { |
| 21 | + let execCalledCount = 0; |
| 22 | + beforeEach(() => { |
| 23 | + execCalledCount = 0; |
| 24 | + }); |
| 25 | + |
| 26 | + const createNetStatResult = (testInjector: IInjector, platform: string, port?: number, iteration?: number): void => { |
| 27 | + const childProcess = testInjector.resolve<IChildProcess>("childProcess"); |
| 28 | + |
| 29 | + childProcess.exec = async (command: string, options?: any, execOptions?: IExecOptions): Promise<any> => { |
| 30 | + const platformsData: IDictionary<any> = { |
| 31 | + linux: { |
| 32 | + data: `Active Internet connections (only servers) |
| 33 | +Proto Recv-Q Send-Q Local Address Foreign Address State |
| 34 | +tcp 0 0 192.168.122.1:53 0.0.0.0:* LISTEN |
| 35 | +tcp 0 0 127.0.1.1:53 0.0.0.0:* LISTEN |
| 36 | +tcp 0 0 127.0.0.1:631 0.0.0.0:* LISTEN |
| 37 | +tcp6 0 0 :::60433 :::* LISTEN |
| 38 | +tcp6 0 0 ::1:631 :::* LISTEN`, |
| 39 | + portData: `tcp6 0 0 :::${port} :::* LISTEN` |
| 40 | + }, |
| 41 | + |
| 42 | + darwin: { |
| 43 | + data: `Current listen queue sizes (qlen/incqlen/maxqlen) |
| 44 | +Listen Local Address |
| 45 | +0/0/1 127.0.0.1.9335 |
| 46 | +0/0/1 127.0.0.1.9334 |
| 47 | +0/0/1 127.0.0.1.9333 |
| 48 | +0/0/128 *.3283 |
| 49 | +0/0/128 *.88 |
| 50 | +0/0/128 *.22`, |
| 51 | + portData: `0/0/128 *.${port}` |
| 52 | + }, |
| 53 | + win32: { |
| 54 | + data: ` |
| 55 | +Active Connections |
| 56 | +
|
| 57 | + Proto Local Address Foreign Address State |
| 58 | + TCP 0.0.0.0:80 0.0.0.0:0 LISTENING |
| 59 | + TCP 0.0.0.0:135 0.0.0.0:0 LISTENING |
| 60 | + TCP 0.0.0.0:60061 0.0.0.0:0 LISTENING |
| 61 | + TCP 127.0.0.1:5037 127.0.0.1:54737 ESTABLISHED |
| 62 | + TCP 127.0.0.1:5037 127.0.0.1:54741 ESTABLISHED |
| 63 | + TCP 127.0.0.1:5354 0.0.0.0:0 LISTENING`, |
| 64 | + portData: ` TCP 127.0.0.1:${port} 0.0.0.0:0 LISTENING` |
| 65 | + } |
| 66 | + }; |
| 67 | + |
| 68 | + execCalledCount++; |
| 69 | + |
| 70 | + let data = platformsData[platform].data; |
| 71 | + |
| 72 | + if (port) { |
| 73 | + data += `${EOL}${platformsData[platform].portData}`; |
| 74 | + } |
| 75 | + |
| 76 | + if (iteration) { |
| 77 | + return iteration === execCalledCount ? data : ""; |
| 78 | + } |
| 79 | + return data; |
| 80 | + }; |
| 81 | + }; |
| 82 | + |
| 83 | + _.each(["linux", "darwin", "win32"], platform => { |
| 84 | + describe(`for ${platform}`, () => { |
| 85 | + it("returns true when netstat returns port is listening", async () => { |
| 86 | + const port = 18181; |
| 87 | + const testInjector = createTestInjector(platform); |
| 88 | + createNetStatResult(testInjector, platform, port); |
| 89 | + |
| 90 | + const net = testInjector.resolve<INet>(Net); |
| 91 | + const isPortListening = await net.waitForPortToListen({ port, timeout: 10, interval: 1 }); |
| 92 | + assert.isTrue(isPortListening); |
| 93 | + assert.equal(execCalledCount, 1); |
| 94 | + }); |
| 95 | + |
| 96 | + it("returns false when netstat does not return the port as not listening", async () => { |
| 97 | + const testInjector = createTestInjector(platform); |
| 98 | + createNetStatResult(testInjector, platform); |
| 99 | + |
| 100 | + const net = testInjector.resolve<INet>(Net); |
| 101 | + const isPortListening = await net.waitForPortToListen({ port: 18181, timeout: 5, interval: 1 }); |
| 102 | + assert.isFalse(isPortListening); |
| 103 | + assert.isTrue(execCalledCount > 1); |
| 104 | + }); |
| 105 | + |
| 106 | + it("returns true when netstat finds the port after some time", async () => { |
| 107 | + const port = 18181; |
| 108 | + const testInjector = createTestInjector(platform); |
| 109 | + const iterations = 2; |
| 110 | + createNetStatResult(testInjector, platform, port, iterations); |
| 111 | + |
| 112 | + const net = testInjector.resolve<INet>(Net); |
| 113 | + const isPortListening = await net.waitForPortToListen({ port, timeout: 10, interval: 1 }); |
| 114 | + assert.isTrue(isPortListening); |
| 115 | + assert.equal(execCalledCount, iterations); |
| 116 | + }); |
| 117 | + |
| 118 | + it("returns false when netstat command fails", async () => { |
| 119 | + const testInjector = createTestInjector(platform); |
| 120 | + const childProcess = testInjector.resolve<IChildProcess>("childProcess"); |
| 121 | + const error = new Error("test error"); |
| 122 | + childProcess.exec = async (command: string, options?: any, execOptions?: IExecOptions): Promise<any> => { |
| 123 | + execCalledCount++; |
| 124 | + return Promise.reject(error); |
| 125 | + }; |
| 126 | + |
| 127 | + const net = testInjector.resolve<INet>(Net); |
| 128 | + const isPortListening = await net.waitForPortToListen({ port: 18181, timeout: 50, interval: 1 }); |
| 129 | + assert.isFalse(isPortListening); |
| 130 | + assert.isTrue(execCalledCount > 1); |
| 131 | + }); |
| 132 | + }); |
| 133 | + }); |
| 134 | + |
| 135 | + it("throws correct error when current operating system is not supported", async () => { |
| 136 | + const invalidPlatform = "invalid_platform"; |
| 137 | + const testInjector = createTestInjector(invalidPlatform); |
| 138 | + |
| 139 | + const net = testInjector.resolve<INet>(Net); |
| 140 | + await assert.isRejected(net.waitForPortToListen({ port: 18181, timeout: 50, interval: 1 }), `Unable to check for free ports on ${invalidPlatform}. Supported platforms are: darwin, linux, win32`); |
| 141 | + }); |
| 142 | + |
| 143 | + it("throws correct error when null is passed", async () => { |
| 144 | + const invalidPlatform = "invalid_platform"; |
| 145 | + const testInjector = createTestInjector(invalidPlatform); |
| 146 | + |
| 147 | + const net = testInjector.resolve<INet>(Net); |
| 148 | + await assert.isRejected(net.waitForPortToListen({ port: 18181, timeout: 50, interval: 1 }), "You must pass port and timeout for check."); |
| 149 | + |
| 150 | + }); |
| 151 | + }); |
| 152 | +}); |
0 commit comments