209 lines
6.0 KiB
JavaScript
209 lines
6.0 KiB
JavaScript
const { describe, test } = require("node:test");
|
|
const assert = require("node:assert");
|
|
const { TCPMonitorType } = require("../../../server/monitor-types/tcp");
|
|
const { UP, PENDING } = require("../../../src/util");
|
|
const net = require("net");
|
|
|
|
describe("TCP Monitor", () => {
|
|
/**
|
|
* Creates a TCP server on a specified port
|
|
* @param {number} port - The port number to listen on
|
|
* @returns {Promise<net.Server>} A promise that resolves with the created server
|
|
*/
|
|
async function createTCPServer(port) {
|
|
return new Promise((resolve, reject) => {
|
|
const server = net.createServer();
|
|
|
|
server.listen(port, () => {
|
|
resolve(server);
|
|
});
|
|
|
|
server.on("error", err => {
|
|
reject(err);
|
|
});
|
|
});
|
|
}
|
|
|
|
test("check() sets status to UP when TCP server is reachable", async () => {
|
|
const port = 12345;
|
|
const server = await createTCPServer(port);
|
|
|
|
try {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "localhost",
|
|
port: port,
|
|
isEnabledExpiryNotification: () => false,
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
await tcpMonitor.check(monitor, heartbeat, {});
|
|
|
|
assert.strictEqual(heartbeat.status, UP);
|
|
} finally {
|
|
server.close();
|
|
}
|
|
});
|
|
|
|
test("check() rejects with connection failed when TCP server is not running", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "localhost",
|
|
port: 54321,
|
|
isEnabledExpiryNotification: () => false,
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
await assert.rejects(
|
|
tcpMonitor.check(monitor, heartbeat, {}),
|
|
new Error("Connection failed")
|
|
);
|
|
});
|
|
|
|
test("check() rejects when TLS certificate is expired or invalid", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "expired.badssl.com",
|
|
port: 443,
|
|
smtpSecurity: "secure",
|
|
isEnabledExpiryNotification: () => true,
|
|
handleTlsInfo: async tlsInfo => {
|
|
return tlsInfo;
|
|
},
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
// Regex: contains with "TLS Connection failed:" or "Certificate is invalid"
|
|
const regex = /TLS Connection failed:|Certificate is invalid/;
|
|
|
|
await assert.rejects(
|
|
tcpMonitor.check(monitor, heartbeat, {}),
|
|
regex
|
|
);
|
|
});
|
|
|
|
test("check() sets status to UP when TLS certificate is valid (SSL)", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "smtp.gmail.com",
|
|
port: 465,
|
|
smtpSecurity: "secure",
|
|
isEnabledExpiryNotification: () => true,
|
|
handleTlsInfo: async tlsInfo => {
|
|
return tlsInfo;
|
|
},
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
await tcpMonitor.check(monitor, heartbeat, {});
|
|
|
|
assert.strictEqual(heartbeat.status, UP);
|
|
});
|
|
|
|
test("check() sets status to UP when TLS certificate is valid (STARTTLS)", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "smtp.gmail.com",
|
|
port: 587,
|
|
smtpSecurity: "starttls",
|
|
isEnabledExpiryNotification: () => true,
|
|
handleTlsInfo: async tlsInfo => {
|
|
return tlsInfo;
|
|
},
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
await tcpMonitor.check(monitor, heartbeat, {});
|
|
|
|
assert.strictEqual(heartbeat.status, UP);
|
|
});
|
|
|
|
test("check() rejects when TLS certificate hostname does not match (STARTTLS)", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "wr-in-f108.1e100.net",
|
|
port: 587,
|
|
smtpSecurity: "starttls",
|
|
isEnabledExpiryNotification: () => true,
|
|
handleTlsInfo: async tlsInfo => {
|
|
return tlsInfo;
|
|
},
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
const regex = /does not match certificate/;
|
|
|
|
await assert.rejects(
|
|
tcpMonitor.check(monitor, heartbeat, {}),
|
|
regex
|
|
);
|
|
});
|
|
test("check() sets status to UP for XMPP server with valid certificate (STARTTLS)", async () => {
|
|
const tcpMonitor = new TCPMonitorType();
|
|
|
|
const monitor = {
|
|
hostname: "xmpp.earth",
|
|
port: 5222,
|
|
smtpSecurity: "starttls",
|
|
isEnabledExpiryNotification: () => true,
|
|
handleTlsInfo: async tlsInfo => {
|
|
return tlsInfo;
|
|
},
|
|
};
|
|
|
|
const heartbeat = {
|
|
msg: "",
|
|
status: PENDING,
|
|
};
|
|
|
|
// Retry up to 5 times for external service reliability
|
|
let lastError;
|
|
for (let attempt = 1; attempt <= 5; attempt++) {
|
|
try {
|
|
await tcpMonitor.check(monitor, heartbeat, {});
|
|
assert.strictEqual(heartbeat.status, UP);
|
|
return; // Success, exit test
|
|
} catch (error) {
|
|
lastError = error;
|
|
// Reset heartbeat for next attempt
|
|
heartbeat.msg = "";
|
|
heartbeat.status = PENDING;
|
|
// Wait a bit before retrying
|
|
await new Promise(resolve => setTimeout(resolve, 500 * 2 ** (attempt - 1)));
|
|
}
|
|
}
|
|
// If all retries failed, throw the last error
|
|
throw lastError;
|
|
});
|
|
});
|