build-speedtest-domainset.ts 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. import path from 'node:path';
  2. import { Sema } from 'async-sema';
  3. import { getHostname } from 'tldts-experimental';
  4. import { task } from './trace';
  5. import { fetchWithRetry } from './lib/fetch-retry';
  6. import { SHARED_DESCRIPTION } from './lib/constants';
  7. import { readFileIntoProcessedArray } from './lib/fetch-text-by-line';
  8. import { TTL, deserializeArray, fsFetchCache, serializeArray, createCacheKey } from './lib/cache-filesystem';
  9. import { DomainsetOutput } from './lib/create-file';
  10. import { OUTPUT_SURGE_DIR } from './constants/dir';
  11. const KEYWORDS = [
  12. 'Hong Kong',
  13. 'Taiwan',
  14. 'China Telecom',
  15. 'China Mobile',
  16. 'China Unicom',
  17. 'Japan',
  18. 'Tokyo',
  19. 'Singapore',
  20. 'Korea',
  21. 'Seoul',
  22. 'Canada',
  23. 'Toronto',
  24. 'Montreal',
  25. 'Los Ang',
  26. 'San Jos',
  27. 'Seattle',
  28. 'New York',
  29. 'Dallas',
  30. 'Miami',
  31. 'Berlin',
  32. 'Frankfurt',
  33. 'London',
  34. 'Paris',
  35. 'Amsterdam',
  36. 'Moscow',
  37. 'Australia',
  38. 'Sydney',
  39. 'Brazil',
  40. 'Turkey'
  41. ];
  42. const PREDEFINE_DOMAINS = [
  43. // speedtest.net
  44. '.speedtest.net',
  45. '.speedtestcustom.com',
  46. '.ooklaserver.net',
  47. '.speed.misaka.one',
  48. '.speedtest.rt.ru',
  49. '.speedtest.aptg.com.tw',
  50. '.speedtest.gslnetworks.com',
  51. '.speedtest.jsinfo.net',
  52. '.speedtest.i3d.net',
  53. '.speedtestkorea.com',
  54. '.speedtest.telus.com',
  55. '.speedtest.telstra.net',
  56. '.speedtest.clouvider.net',
  57. '.speedtest.idv.tw',
  58. '.speedtest.frontier.com',
  59. '.speedtest.orange.fr',
  60. '.speedtest.centurylink.net',
  61. '.srvr.bell.ca',
  62. '.speedtest.contabo.net',
  63. 'speedtest.hk.chinamobile.com',
  64. 'speedtestbb.hk.chinamobile.com',
  65. '.hizinitestet.com',
  66. '.linknetspeedtest.net.br',
  67. 'speedtest.rit.edu',
  68. 'speedtest.ropa.de',
  69. 'speedtest.sits.su',
  70. 'speedtest.tigo.cr',
  71. 'speedtest.upp.com',
  72. '.speedtest.pni.tw',
  73. '.speed.pfm.gg',
  74. '.speedtest.faelix.net',
  75. '.speedtest.labixe.net',
  76. '.speedtest.warian.net',
  77. '.speedtest.starhub.com',
  78. '.speedtest.gibir.net.tr',
  79. '.speedtest.ozarksgo.net',
  80. '.speedtest.exetel.com.au',
  81. '.speedtest.sbcglobal.net',
  82. '.speedtest.leaptel.com.au',
  83. '.speedtest.windstream.net',
  84. '.speedtest.vodafone.com.au',
  85. '.speedtest.rascom.ru',
  86. '.speedtest.dchost.com',
  87. '.speedtest.highnet.com',
  88. '.speedtest.seattle.wa.limewave.net',
  89. '.speedtest.optitel.com.au',
  90. '.speednet.net.tr',
  91. '.speedtest.angolacables.co.ao',
  92. '.ookla-speedtest.fsr.com',
  93. '.speedtest.comnet.com.tr',
  94. '.speedtest.gslnetworks.com.au',
  95. '.test.gslnetworks.com.au',
  96. '.speedtest.gslnetworks.com',
  97. '.speedtestunonet.com.br',
  98. '.speedtest.alagas.net',
  99. 'speedtest.surfshark.com',
  100. '.speedtest.aarnet.net.au',
  101. '.ookla.rcp.net',
  102. '.ookla-speedtests.e2ro.com',
  103. '.speedtest.com.sg',
  104. '.ookla.ddnsgeek.com',
  105. '.speedtest.pni.tw',
  106. '.speedtest.cmcnetworks.net',
  107. '.speedtestwnet.com.br',
  108. // Cloudflare
  109. '.speed.cloudflare.com',
  110. // Wi-Fi Man
  111. '.wifiman.com',
  112. '.wifiman.me',
  113. '.wifiman.ubncloud.com',
  114. // Fast.com
  115. '.fast.com',
  116. // MacPaw
  117. 'speedtest.macpaw.com',
  118. // speedtestmaster
  119. '.netspeedtestmaster.com',
  120. // Google Search Result of "speedtest", powered by this
  121. '.measurement-lab.org',
  122. '.measurementlab.net',
  123. // Google Fiber legacy speedtest site (new fiber speedtest use speedtestcustom.com)
  124. '.speed.googlefiber.net',
  125. // librespeed
  126. '.backend.librespeed.org',
  127. // Apple,
  128. 'mensura.cdn-apple.com', // From netQuality command
  129. // OpenSpeedtest
  130. 'open.cachefly.net' // This is also used for openspeedtest server download
  131. ];
  132. const s = new Sema(2);
  133. const cacheKey = createCacheKey(__filename);
  134. const latestTopUserAgentsPromise = fsFetchCache.apply(
  135. cacheKey('https://cdn.jsdelivr.net/npm/top-user-agents@latest/src/desktop.json'),
  136. () => fetchWithRetry(
  137. 'https://cdn.jsdelivr.net/npm/top-user-agents@latest/src/desktop.json',
  138. { signal: AbortSignal.timeout(1000 * 60) }
  139. )
  140. .then(res => res.json() as Promise<string[]>)
  141. .then((userAgents) => userAgents.filter(ua => ua.startsWith('Mozilla/5.0 '))),
  142. {
  143. serializer: serializeArray,
  144. deserializer: deserializeArray,
  145. ttl: TTL.THREE_DAYS()
  146. }
  147. );
  148. const querySpeedtestApi = async (keyword: string): Promise<Array<string | null>> => {
  149. const topUserAgents = await latestTopUserAgentsPromise;
  150. const url = `https://www.speedtest.net/api/js/servers?engine=js&search=${keyword}&limit=100`;
  151. try {
  152. const randomUserAgent = topUserAgents[Math.floor(Math.random() * topUserAgents.length)];
  153. return await fsFetchCache.apply(
  154. cacheKey(url),
  155. () => s.acquire().then(() => fetchWithRetry(url, {
  156. headers: {
  157. dnt: '1',
  158. Referer: 'https://www.speedtest.net/',
  159. accept: 'application/json, text/plain, */*',
  160. 'User-Agent': randomUserAgent,
  161. 'Accept-Language': 'en-US,en;q=0.9',
  162. ...(randomUserAgent.includes('Chrome')
  163. ? {
  164. 'Sec-Ch-Ua-Mobile': '?0',
  165. 'Sec-Fetch-Dest': 'empty',
  166. 'Sec-Fetch-Mode': 'cors',
  167. 'Sec-Fetch-Site': 'same-origin',
  168. 'Sec-Gpc': '1'
  169. }
  170. : {})
  171. },
  172. signal: AbortSignal.timeout(1000 * 60),
  173. retry: {
  174. retries: 2
  175. }
  176. })).then(r => r.json() as any).then((data: Array<{ url: string, host: string }>) => data.reduce<string[]>(
  177. (prev, cur) => {
  178. const line = cur.host || cur.url;
  179. const hn = getHostname(line, { detectIp: false, validateHostname: true });
  180. if (hn) {
  181. prev.push(hn);
  182. }
  183. return prev;
  184. }, []
  185. )).finally(() => s.release()),
  186. {
  187. ttl: TTL.ONE_WEEK(),
  188. serializer: serializeArray,
  189. deserializer: deserializeArray
  190. }
  191. );
  192. } catch (e) {
  193. console.error(e);
  194. return [];
  195. }
  196. };
  197. export const buildSpeedtestDomainSet = task(require.main === module, __filename)(async (span) => {
  198. const output = new DomainsetOutput(span, 'speedtest')
  199. .withTitle('Sukka\'s Ruleset - Speedtest Domains')
  200. .withDescription([
  201. ...SHARED_DESCRIPTION,
  202. '',
  203. 'This file contains common speedtest endpoints.'
  204. ])
  205. .addFromDomainset(PREDEFINE_DOMAINS)
  206. .addFromDomainset(await readFileIntoProcessedArray(path.resolve(OUTPUT_SURGE_DIR, 'domainset/speedtest.conf')));
  207. await Promise.all(KEYWORDS.map((keyword) => span.traceChildAsync(
  208. `fetch speedtest endpoints: ${keyword}`,
  209. () => querySpeedtestApi(keyword)
  210. ).then(hostnameGroup => hostnameGroup.forEach(hostname => {
  211. if (hostname) {
  212. output.addDomain(hostname);
  213. }
  214. }))));
  215. return output.write();
  216. });