2023-03-22 00:43:55 +00:00
|
|
|
// Copyright 2023 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2023-12-12 22:57:09 +00:00
|
|
|
import type {
|
|
|
|
LookupOneOptions,
|
|
|
|
LookupAllOptions,
|
|
|
|
LookupAddress,
|
|
|
|
lookup as nodeLookup,
|
|
|
|
} from 'dns';
|
2023-05-30 23:57:16 +00:00
|
|
|
import { ipcRenderer, net } from 'electron';
|
2024-02-27 02:02:23 +00:00
|
|
|
import type { ResolvedHost, ResolvedEndpoint } from 'electron';
|
2023-03-22 00:43:55 +00:00
|
|
|
|
|
|
|
import { strictAssert } from './assert';
|
2023-05-30 23:57:16 +00:00
|
|
|
import { drop } from './drop';
|
2023-03-22 00:43:55 +00:00
|
|
|
|
2024-02-27 02:02:23 +00:00
|
|
|
const FALLBACK_ADDRS: ReadonlyMap<
|
|
|
|
string,
|
|
|
|
ReadonlyArray<ResolvedEndpoint>
|
|
|
|
> = new Map([
|
|
|
|
[
|
|
|
|
'cdsi.signal.org',
|
|
|
|
[
|
|
|
|
{ family: 'ipv4', address: '40.122.45.194' },
|
|
|
|
{ family: 'ipv6', address: '2603:1030:7::1' },
|
|
|
|
],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'chat.signal.org',
|
|
|
|
[
|
|
|
|
{ family: 'ipv4', address: '13.248.212.111' },
|
|
|
|
{ family: 'ipv4', address: '76.223.92.165' },
|
|
|
|
{ family: 'ipv6', address: '2600:9000:a507:ab6d:4ce3:2f58:25d7:9cbf' },
|
|
|
|
{ family: 'ipv6', address: '2600:9000:a61f:527c:d5eb:a431:5239:3232' },
|
|
|
|
],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'sfu.voip.signal.org',
|
|
|
|
[
|
|
|
|
{ family: 'ipv4', address: '34.49.5.136' },
|
|
|
|
{ family: 'ipv6', address: '2600:1901:0:9c39::' },
|
|
|
|
],
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
|
2023-05-30 23:57:16 +00:00
|
|
|
function lookupAll(
|
2023-03-22 00:43:55 +00:00
|
|
|
hostname: string,
|
2023-05-30 23:57:16 +00:00
|
|
|
opts: LookupOneOptions | LookupAllOptions,
|
2023-03-22 00:43:55 +00:00
|
|
|
callback: (
|
|
|
|
err: NodeJS.ErrnoException | null,
|
2023-05-30 23:57:16 +00:00
|
|
|
addresses: string | Array<LookupAddress>,
|
|
|
|
family?: number
|
2023-03-22 00:43:55 +00:00
|
|
|
) => void
|
|
|
|
): void {
|
|
|
|
// Node.js support various signatures, but we only support one.
|
|
|
|
strictAssert(typeof opts === 'object', 'missing options');
|
|
|
|
strictAssert(typeof callback === 'function', 'missing callback');
|
|
|
|
|
2023-05-30 23:57:16 +00:00
|
|
|
async function run() {
|
2024-02-27 02:02:23 +00:00
|
|
|
let result: Pick<ResolvedHost, 'endpoints'>;
|
2023-03-22 00:43:55 +00:00
|
|
|
|
2024-02-27 02:02:23 +00:00
|
|
|
let queryType: 'A' | 'AAAA' | undefined;
|
|
|
|
if (opts.family === 4) {
|
|
|
|
queryType = 'A';
|
|
|
|
} else if (opts.family === 6) {
|
|
|
|
queryType = 'AAAA';
|
|
|
|
}
|
2023-04-05 21:49:33 +00:00
|
|
|
|
2024-02-27 02:02:23 +00:00
|
|
|
try {
|
2023-05-30 23:57:16 +00:00
|
|
|
if (net) {
|
|
|
|
// Main process
|
|
|
|
result = await net.resolveHost(hostname, {
|
|
|
|
queryType,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// Renderer
|
|
|
|
result = await ipcRenderer.invoke(
|
|
|
|
'net.resolveHost',
|
|
|
|
hostname,
|
|
|
|
queryType
|
|
|
|
);
|
|
|
|
}
|
2024-02-27 02:02:23 +00:00
|
|
|
} catch (error) {
|
|
|
|
const fallback = FALLBACK_ADDRS.get(hostname);
|
|
|
|
if (fallback) {
|
|
|
|
result = { endpoints: fallback.slice() };
|
|
|
|
} else {
|
|
|
|
callback(error, []);
|
2023-05-30 23:57:16 +00:00
|
|
|
return;
|
|
|
|
}
|
2024-02-27 02:02:23 +00:00
|
|
|
}
|
2023-03-22 00:43:55 +00:00
|
|
|
|
2024-02-27 02:02:23 +00:00
|
|
|
const addresses = result.endpoints.map(({ address, family }) => {
|
|
|
|
let numericFamily = -1;
|
|
|
|
if (family === 'ipv4') {
|
|
|
|
numericFamily = 4;
|
|
|
|
} else if (family === 'ipv6') {
|
|
|
|
numericFamily = 6;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
address,
|
|
|
|
family: numericFamily,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!opts.all) {
|
|
|
|
const random = addresses.at(Math.floor(Math.random() * addresses.length));
|
|
|
|
if (random === undefined) {
|
|
|
|
callback(new Error(`Hostname: ${hostname} cannot be resolved`), '', -1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
callback(null, random.address, random.family);
|
|
|
|
return;
|
2023-03-22 00:43:55 +00:00
|
|
|
}
|
2024-02-27 02:02:23 +00:00
|
|
|
|
|
|
|
callback(null, addresses);
|
2023-05-30 23:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
drop(run());
|
2023-03-22 00:43:55 +00:00
|
|
|
}
|
2023-05-30 23:57:16 +00:00
|
|
|
|
2023-08-29 23:58:48 +00:00
|
|
|
export function interleaveAddresses(
|
|
|
|
addresses: ReadonlyArray<LookupAddress>
|
|
|
|
): Array<LookupAddress> {
|
|
|
|
const firstAddr = addresses.find(
|
|
|
|
({ family }) => family === 4 || family === 6
|
|
|
|
);
|
|
|
|
if (!firstAddr) {
|
|
|
|
throw new Error('interleaveAddresses: no addresses to interleave');
|
|
|
|
}
|
|
|
|
|
|
|
|
const v4 = addresses.filter(({ family }) => family === 4);
|
|
|
|
const v6 = addresses.filter(({ family }) => family === 6);
|
|
|
|
|
|
|
|
// Interleave addresses for Happy Eyeballs, but keep the first address
|
|
|
|
// type from the DNS response first in the list.
|
|
|
|
const interleaved = new Array<LookupAddress>();
|
|
|
|
while (v4.length !== 0 || v6.length !== 0) {
|
|
|
|
const v4Entry = v4.pop();
|
|
|
|
const v6Entry = v6.pop();
|
|
|
|
|
|
|
|
if (firstAddr.family === 4) {
|
|
|
|
if (v4Entry !== undefined) {
|
|
|
|
interleaved.push(v4Entry);
|
|
|
|
}
|
|
|
|
if (v6Entry !== undefined) {
|
|
|
|
interleaved.push(v6Entry);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (v6Entry !== undefined) {
|
|
|
|
interleaved.push(v6Entry);
|
|
|
|
}
|
|
|
|
if (v4Entry !== undefined) {
|
|
|
|
interleaved.push(v4Entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return interleaved;
|
|
|
|
}
|
|
|
|
|
2023-05-30 23:57:16 +00:00
|
|
|
// Note: `nodeLookup` has a complicated type due to compatibility requirements.
|
|
|
|
export const electronLookup = lookupAll as typeof nodeLookup;
|