build-speedtest-domainset.ts 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  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. // Wi-Fi Man
  134. '.wifiman.com',
  135. '.wifiman.me',
  136. // Fast.com
  137. '.fast.com',
  138. // MacPaw
  139. 'speedtest.macpaw.com',
  140. // speedtestmaster
  141. '.netspeedtestmaster.com',
  142. // Google Search Result of "speedtest", powered by this
  143. '.measurement-lab.org',
  144. '.measurementlab.net',
  145. // Google Fiber legacy speedtest site (new fiber speedtest use speedtestcustom.com)
  146. '.speed.googlefiber.net',
  147. // librespeed
  148. '.backend.librespeed.org'
  149. ]);
  150. // Download previous speedtest domainset
  151. for await (const l of await fetchRemoteTextByLine('https://ruleset.skk.moe/List/domainset/speedtest.conf')) {
  152. const line = processLine(l);
  153. if (line) {
  154. domains.add(line);
  155. }
  156. }
  157. await new Promise<void>((resolve) => {
  158. const pMap = ([
  159. 'Hong Kong',
  160. 'Taiwan',
  161. 'China Telecom',
  162. 'China Mobile',
  163. 'China Unicom',
  164. 'Japan',
  165. 'Tokyo',
  166. 'Singapore',
  167. 'Korea',
  168. 'Canada',
  169. 'Toronto',
  170. 'Montreal',
  171. 'Los Ang',
  172. 'San Jos',
  173. 'Seattle',
  174. 'New York',
  175. 'Dallas',
  176. 'Miami',
  177. 'Berlin',
  178. 'Frankfurt',
  179. 'London',
  180. 'Paris',
  181. 'Amsterdam',
  182. 'Moscow',
  183. 'Australia',
  184. 'Sydney',
  185. 'Brazil',
  186. 'Turkey'
  187. ]).reduce<Record<string, Promise<void>>>((pMap, keyword) => {
  188. pMap[keyword] = querySpeedtestApi(keyword).then(hostnameGroup => {
  189. hostnameGroup.forEach(hostname => {
  190. if (hostname) {
  191. domains.add(hostname);
  192. }
  193. });
  194. });
  195. return pMap;
  196. }, {});
  197. const timer = setTimeout(() => {
  198. console.error(picocolors.red('Task timeout!'));
  199. Object.entries(pMap).forEach(([name, p]) => {
  200. console.log(`[${name}]`, Bun.peek.status(p));
  201. });
  202. resolve();
  203. }, 1000 * 60 * 2);
  204. Promise.all(Object.values(pMap)).then(() => {
  205. clearTimeout(timer);
  206. resolve();
  207. });
  208. });
  209. const gorhill = await getGorhillPublicSuffixPromise();
  210. const deduped = sortDomains(domainDeduper(Array.from(domains)), gorhill);
  211. const description = [
  212. ...SHARED_DESCRIPTION,
  213. '',
  214. 'This file contains common speedtest endpoints.'
  215. ];
  216. return Promise.all(createRuleset(
  217. 'Sukka\'s Ruleset - Speedtest Domains',
  218. description,
  219. new Date(),
  220. deduped,
  221. 'domainset',
  222. path.resolve(import.meta.dir, '../List/domainset/speedtest.conf'),
  223. path.resolve(import.meta.dir, '../Clash/domainset/speedtest.txt')
  224. ));
  225. });
  226. if (import.meta.main) {
  227. buildSpeedtestDomainSet();
  228. }