build-speedtest-domainset.ts 5.6 KB

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