"use strict"; const {isIP} = require("net"); const {networkInterfaces} = require("os"); const execa = require("execa"); const gwArgs = "path Win32_NetworkAdapterConfiguration where IPEnabled=true get DefaultIPGateway,GatewayCostMetric,IPConnectionMetric,Index /format:table".split(" "); const ifArgs = index => `path Win32_NetworkAdapter where Index=${index} get NetConnectionID,MACAddress /format:table`.split(" "); const spawnOpts = { windowsHide: true, }; // Parsing tables like this. The final metric is GatewayCostMetric + IPConnectionMetric // // DefaultIPGateway GatewayCostMetric Index IPConnectionMetric // {"1.2.3.4", "2001:db8::1"} {0, 256} 12 25 // {"2.3.4.5"} {25} 12 55 function parseGwTable(gwTable, family) { let [bestGw, bestMetric, bestId] = [null, null, null]; for (let line of (gwTable || "").trim().split(/\r?\n/).splice(1)) { line = line.trim(); const [_, gwArr, gwCostsArr, id, ipMetric] = /({.+?}) +?({.+?}) +?([0-9]+) +?([0-9]+)/g.exec(line) || []; if (!gwArr) continue; const gateways = (gwArr.match(/"(.+?)"/g) || []).map(match => match.substring(1, match.length - 1)); const gatewayCosts = (gwCostsArr.match(/[0-9]+/g) || []); for (const [index, gateway] of Object.entries(gateways)) { if (!gateway || `v${isIP(gateway)}` !== family) continue; const metric = parseInt(gatewayCosts[index]) + parseInt(ipMetric); if (!bestGw || metric < bestMetric) { [bestGw, bestMetric, bestId] = [gateway, metric, id]; } } } if (bestGw) return [bestGw, bestId]; } function parseIfTable(ifTable) { const line = (ifTable || "").trim().split("\n")[1]; let [mac, name] = line.trim().split(/\s+/); mac = mac.toLowerCase(); // try to get the interface name by matching the mac to os.networkInterfaces to avoid wmic's encoding issues // https://github.com/silverwind/default-gateway/issues/14 for (const [osname, addrs] of Object.entries(networkInterfaces())) { for (const addr of addrs) { if (addr && addr.mac && addr.mac.toLowerCase() === mac) { return osname; } } } return name; } const promise = async family => { const {stdout} = await execa("wmic", gwArgs, spawnOpts); const [gateway, id] = parseGwTable(stdout, family) || []; if (!gateway) { throw new Error("Unable to determine default gateway"); } let name; if (id) { const {stdout} = await execa("wmic", ifArgs(id), spawnOpts); name = parseIfTable(stdout); } return {gateway, interface: name ? name : null}; }; const sync = family => { const {stdout} = execa.sync("wmic", gwArgs, spawnOpts); const [gateway, id] = parseGwTable(stdout, family) || []; if (!gateway) { throw new Error("Unable to determine default gateway"); } let name; if (id) { const {stdout} = execa.sync("wmic", ifArgs(id), spawnOpts); name = parseIfTable(stdout); } return {gateway, interface: name ? name : null}; }; module.exports.v4 = () => promise("v4"); module.exports.v6 = () => promise("v6"); module.exports.v4.sync = () => sync("v4"); module.exports.v6.sync = () => sync("v6");