build-speedtest-domainset.ts 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. import { domainDeduper } from './lib/domain-deduper';
  2. import path from 'path';
  3. import { createRuleset } from './lib/create-file';
  4. import { sortDomains } from './lib/stable-sort-domain';
  5. import { Sema } from 'async-sema';
  6. import * as tldts from 'tldts';
  7. import { task } from './lib/trace-runner';
  8. import { fetchWithRetry } from './lib/fetch-retry';
  9. import { SHARED_DESCRIPTION } from './lib/constants';
  10. import { getGorhillPublicSuffixPromise } from './lib/get-gorhill-publicsuffix';
  11. import picocolors from 'picocolors';
  12. import { fetchRemoteTextByLine } from './lib/fetch-text-by-line';
  13. import { processLine } from './lib/process-line';
  14. import { TTL, deserializeArray, fsCache, serializeArray } from './lib/cache-filesystem';
  15. const s = new Sema(2);
  16. const latestTopUserAgentsPromise = fsCache.apply(
  17. 'https://unpkg.com/top-user-agents@latest/src/desktop.json',
  18. () => fetchWithRetry('https://unpkg.com/top-user-agents@latest/src/desktop.json')
  19. .then(res => res.json<string[]>())
  20. .then(userAgents => userAgents.filter(ua => ua.startsWith('Mozilla/5.0 '))),
  21. {
  22. serializer: serializeArray,
  23. deserializer: deserializeArray,
  24. ttl: TTL.ONE_DAY()
  25. }
  26. );
  27. const querySpeedtestApi = async (keyword: string): Promise<Array<string | null>> => {
  28. const topUserAgents = await latestTopUserAgentsPromise;
  29. const url = `https://www.speedtest.net/api/js/servers?engine=js&search=${keyword}&limit=100`;
  30. try {
  31. const randomUserAgent = topUserAgents[Math.floor(Math.random() * topUserAgents.length)];
  32. const key = `fetch speedtest endpoints: ${keyword}`;
  33. console.log(key);
  34. console.time(key);
  35. const json = await fsCache.apply(
  36. url,
  37. () => s.acquire().then(() => fetchWithRetry(url, {
  38. headers: {
  39. dnt: '1',
  40. Referer: 'https://www.speedtest.net/',
  41. accept: 'application/json, text/plain, */*',
  42. 'User-Agent': randomUserAgent,
  43. 'Accept-Language': 'en-US,en;q=0.9',
  44. ...(randomUserAgent.includes('Chrome')
  45. ? {
  46. 'Sec-Ch-Ua-Mobile': '?0',
  47. 'Sec-Fetch-Dest': 'empty',
  48. 'Sec-Fetch-Mode': 'cors',
  49. 'Sec-Fetch-Site': 'same-origin',
  50. 'Sec-Gpc': '1'
  51. }
  52. : {})
  53. },
  54. signal: AbortSignal.timeout(1000 * 4),
  55. retry: {
  56. retries: 2
  57. }
  58. })).then(r => r.json<Array<{ url: string }>>()).then(data => data.reduce<string[]>(
  59. (prev, cur) => {
  60. const hn = tldts.getHostname(cur.url, { detectIp: false });
  61. if (hn) {
  62. prev.push(hn);
  63. }
  64. return prev;
  65. }, []
  66. )).finally(() => s.release()),
  67. {
  68. ttl: TTL.ONE_WEEK(),
  69. serializer: serializeArray,
  70. deserializer: deserializeArray
  71. }
  72. );
  73. console.timeEnd(key);
  74. return json;
  75. } catch (e) {
  76. console.log(e);
  77. return [];
  78. }
  79. };
  80. export const buildSpeedtestDomainSet = task(import.meta.path, async () => {
  81. // Predefined domainset
  82. /** @type {Set<string>} */
  83. const domains = new Set<string>([
  84. '.speedtest.net',
  85. '.speedtestcustom.com',
  86. '.ooklaserver.net',
  87. '.speed.misaka.one',
  88. '.speed.cloudflare.com',
  89. '.speedtest.rt.ru',
  90. '.speedtest.aptg.com.tw',
  91. '.speedtest.gslnetworks.com',
  92. '.speedtest.jsinfo.net',
  93. '.speedtest.i3d.net',
  94. '.speedtestkorea.com',
  95. '.speedtest.telus.com',
  96. '.speedtest.telstra.net',
  97. '.speedtest.clouvider.net',
  98. '.speedtest.idv.tw',
  99. '.speedtest.frontier.com',
  100. '.speedtest.orange.fr',
  101. '.speedtest.centurylink.net',
  102. '.srvr.bell.ca',
  103. '.speedtest.contabo.net',
  104. 'speedtest.hk.chinamobile.com',
  105. 'speedtestbb.hk.chinamobile.com',
  106. '.hizinitestet.com',
  107. '.linknetspeedtest.net.br',
  108. 'speedtest.rit.edu',
  109. 'speedtest.ropa.de',
  110. 'speedtest.sits.su',
  111. 'speedtest.tigo.cr',
  112. 'speedtest.upp.com',
  113. '.speedtest.pni.tw',
  114. '.speed.pfm.gg',
  115. '.speedtest.faelix.net',
  116. '.speedtest.labixe.net',
  117. '.speedtest.warian.net',
  118. '.speedtest.starhub.com',
  119. '.speedtest.gibir.net.tr',
  120. '.speedtest.ozarksgo.net',
  121. '.speedtest.exetel.com.au',
  122. '.speedtest.sbcglobal.net',
  123. '.speedtest.leaptel.com.au',
  124. '.speedtest.windstream.net',
  125. '.speedtest.vodafone.com.au',
  126. '.speedtest.rascom.ru',
  127. '.speedtest.dchost.com',
  128. '.speedtest.highnet.com',
  129. '.speedtest.seattle.wa.limewave.net',
  130. '.speedtest.optitel.com.au',
  131. '.speednet.net.tr',
  132. '.speedtest.angolacables.co.ao',
  133. // Fast.com
  134. '.fast.com',
  135. // MacPaw
  136. 'speedtest.macpaw.com',
  137. // speedtestmaster
  138. '.netspeedtestmaster.com',
  139. // Google Search Result of "speedtest", powered by this
  140. '.measurement-lab.org',
  141. // Google Fiber legacy speedtest site (new fiber speedtest use speedtestcustom.com)
  142. '.speed.googlefiber.net',
  143. // librespeed
  144. '.backend.librespeed.org'
  145. ]);
  146. // Download previous speedtest domainset
  147. for await (const l of await fetchRemoteTextByLine('https://ruleset.skk.moe/List/domainset/speedtest.conf')) {
  148. const line = processLine(l);
  149. if (line) {
  150. domains.add(line);
  151. }
  152. }
  153. await new Promise<void>((resolve) => {
  154. const pMap = ([
  155. 'Hong Kong',
  156. 'Taiwan',
  157. 'China Telecom',
  158. 'China Mobile',
  159. 'China Unicom',
  160. 'Japan',
  161. 'Tokyo',
  162. 'Singapore',
  163. 'Korea',
  164. 'Canada',
  165. 'Toronto',
  166. 'Montreal',
  167. 'Los Ang',
  168. 'San Jos',
  169. 'Seattle',
  170. 'New York',
  171. 'Dallas',
  172. 'Miami',
  173. 'Berlin',
  174. 'Frankfurt',
  175. 'London',
  176. 'Paris',
  177. 'Amsterdam',
  178. 'Moscow',
  179. 'Australia',
  180. 'Sydney',
  181. 'Brazil',
  182. 'Turkey'
  183. ]).reduce<Record<string, Promise<void>>>((pMap, keyword) => {
  184. pMap[keyword] = querySpeedtestApi(keyword).then(hostnameGroup => {
  185. hostnameGroup.forEach(hostname => {
  186. if (hostname) {
  187. domains.add(hostname);
  188. }
  189. });
  190. });
  191. return pMap;
  192. }, {});
  193. const timer = setTimeout(() => {
  194. console.error(picocolors.red('Task timeout!'));
  195. Object.entries(pMap).forEach(([name, p]) => {
  196. console.log(`[${name}]`, Bun.peek.status(p));
  197. });
  198. resolve();
  199. }, 1000 * 60 * 2);
  200. Promise.all(Object.values(pMap)).then(() => {
  201. clearTimeout(timer);
  202. resolve();
  203. });
  204. });
  205. const gorhill = await getGorhillPublicSuffixPromise();
  206. const deduped = sortDomains(domainDeduper(Array.from(domains)), gorhill);
  207. const description = [
  208. ...SHARED_DESCRIPTION,
  209. '',
  210. 'This file contains common speedtest endpoints.'
  211. ];
  212. return Promise.all(createRuleset(
  213. 'Sukka\'s Ruleset - Speedtest Domains',
  214. description,
  215. new Date(),
  216. deduped,
  217. 'domainset',
  218. path.resolve(import.meta.dir, '../List/domainset/speedtest.conf'),
  219. path.resolve(import.meta.dir, '../Clash/domainset/speedtest.txt')
  220. ));
  221. });
  222. if (import.meta.main) {
  223. buildSpeedtestDomainSet();
  224. }