import { domainDeduper } from './lib/domain-deduper'; import path from 'path'; import { createRuleset } from './lib/create-file'; import { sortDomains } from './lib/stable-sort-domain'; import { Sema } from 'async-sema'; import * as tldts from 'tldts'; import { task } from './trace'; import { fetchWithRetry } from './lib/fetch-retry'; import { SHARED_DESCRIPTION } from './lib/constants'; import { getGorhillPublicSuffixPromise } from './lib/get-gorhill-publicsuffix'; import picocolors from 'picocolors'; import { fetchRemoteTextByLine } from './lib/fetch-text-by-line'; import { processLine } from './lib/process-line'; import { TTL, deserializeArray, fsCache, serializeArray } from './lib/cache-filesystem'; const s = new Sema(2); const latestTopUserAgentsPromise = fsCache.apply( 'https://unpkg.com/top-user-agents@latest/src/desktop.json', () => fetchWithRetry('https://unpkg.com/top-user-agents@latest/src/desktop.json') .then(res => res.json()) .then(userAgents => userAgents.filter(ua => ua.startsWith('Mozilla/5.0 '))), { serializer: serializeArray, deserializer: deserializeArray, ttl: TTL.THREE_DAYS() } ); const querySpeedtestApi = async (keyword: string): Promise> => { const topUserAgents = await latestTopUserAgentsPromise; const url = `https://www.speedtest.net/api/js/servers?engine=js&search=${keyword}&limit=100`; try { const randomUserAgent = topUserAgents[Math.floor(Math.random() * topUserAgents.length)]; return await fsCache.apply( url, () => s.acquire().then(() => fetchWithRetry(url, { headers: { dnt: '1', Referer: 'https://www.speedtest.net/', accept: 'application/json, text/plain, */*', 'User-Agent': randomUserAgent, 'Accept-Language': 'en-US,en;q=0.9', ...(randomUserAgent.includes('Chrome') ? { 'Sec-Ch-Ua-Mobile': '?0', 'Sec-Fetch-Dest': 'empty', 'Sec-Fetch-Mode': 'cors', 'Sec-Fetch-Site': 'same-origin', 'Sec-Gpc': '1' } : {}) }, signal: AbortSignal.timeout(1000 * 4), retry: { retries: 2 } })).then(r => r.json>()).then(data => data.reduce( (prev, cur) => { const hn = tldts.getHostname(cur.url, { detectIp: false }); if (hn) { prev.push(hn); } return prev; }, [] )).finally(() => s.release()), { ttl: TTL.ONE_WEEK(), serializer: serializeArray, deserializer: deserializeArray } ); } catch (e) { console.error(e); return []; } }; export const buildSpeedtestDomainSet = task(import.meta.path, async (span) => { // Predefined domainset /** @type {Set} */ const domains = new Set([ '.speedtest.net', '.speedtestcustom.com', '.ooklaserver.net', '.speed.misaka.one', '.speed.cloudflare.com', '.speedtest.rt.ru', '.speedtest.aptg.com.tw', '.speedtest.gslnetworks.com', '.speedtest.jsinfo.net', '.speedtest.i3d.net', '.speedtestkorea.com', '.speedtest.telus.com', '.speedtest.telstra.net', '.speedtest.clouvider.net', '.speedtest.idv.tw', '.speedtest.frontier.com', '.speedtest.orange.fr', '.speedtest.centurylink.net', '.srvr.bell.ca', '.speedtest.contabo.net', 'speedtest.hk.chinamobile.com', 'speedtestbb.hk.chinamobile.com', '.hizinitestet.com', '.linknetspeedtest.net.br', 'speedtest.rit.edu', 'speedtest.ropa.de', 'speedtest.sits.su', 'speedtest.tigo.cr', 'speedtest.upp.com', '.speedtest.pni.tw', '.speed.pfm.gg', '.speedtest.faelix.net', '.speedtest.labixe.net', '.speedtest.warian.net', '.speedtest.starhub.com', '.speedtest.gibir.net.tr', '.speedtest.ozarksgo.net', '.speedtest.exetel.com.au', '.speedtest.sbcglobal.net', '.speedtest.leaptel.com.au', '.speedtest.windstream.net', '.speedtest.vodafone.com.au', '.speedtest.rascom.ru', '.speedtest.dchost.com', '.speedtest.highnet.com', '.speedtest.seattle.wa.limewave.net', '.speedtest.optitel.com.au', '.speednet.net.tr', '.speedtest.angolacables.co.ao', // Wi-Fi Man '.wifiman.com', '.wifiman.me', // Fast.com '.fast.com', // MacPaw 'speedtest.macpaw.com', // speedtestmaster '.netspeedtestmaster.com', // Google Search Result of "speedtest", powered by this '.measurement-lab.org', '.measurementlab.net', // Google Fiber legacy speedtest site (new fiber speedtest use speedtestcustom.com) '.speed.googlefiber.net', // librespeed '.backend.librespeed.org' ]); // Download previous speedtest domainset for await (const l of await fetchRemoteTextByLine('https://ruleset.skk.moe/List/domainset/speedtest.conf')) { const line = processLine(l); if (line) { domains.add(line); } } await new Promise((resolve) => { const pMap = ([ 'Hong Kong', 'Taiwan', 'China Telecom', 'China Mobile', 'China Unicom', 'Japan', 'Tokyo', 'Singapore', 'Korea', 'Canada', 'Toronto', 'Montreal', 'Los Ang', 'San Jos', 'Seattle', 'New York', 'Dallas', 'Miami', 'Berlin', 'Frankfurt', 'London', 'Paris', 'Amsterdam', 'Moscow', 'Australia', 'Sydney', 'Brazil', 'Turkey' ]).reduce>>((pMap, keyword) => { pMap[keyword] = span.traceChild(`fetch speedtest endpoints: ${keyword}`).traceAsyncFn(() => querySpeedtestApi(keyword)).then(hostnameGroup => { hostnameGroup.forEach(hostname => { if (hostname) { domains.add(hostname); } }); }); return pMap; }, {}); const timer = setTimeout(() => { console.error(picocolors.red('Task timeout!')); Object.entries(pMap).forEach(([name, p]) => { console.log(`[${name}]`, Bun.peek.status(p)); }); resolve(); }, 1000 * 60 * 2); Promise.all(Object.values(pMap)).then(() => { clearTimeout(timer); resolve(); }); }); const gorhill = await getGorhillPublicSuffixPromise(); const deduped = span.traceChild('sort result').traceSyncFn(() => sortDomains(domainDeduper(Array.from(domains)), gorhill)); const description = [ ...SHARED_DESCRIPTION, '', 'This file contains common speedtest endpoints.' ]; return Promise.all(createRuleset( 'Sukka\'s Ruleset - Speedtest Domains', description, new Date(), deduped, 'domainset', path.resolve(import.meta.dir, '../List/domainset/speedtest.conf'), path.resolve(import.meta.dir, '../Clash/domainset/speedtest.txt') )); }); if (import.meta.main) { buildSpeedtestDomainSet(); }