validate-gfwlist.ts 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. import { processLine } from './lib/process-line';
  2. import { fastNormalizeDomain } from './lib/normalize-domain';
  3. import { HostnameSmolTrie } from './lib/trie';
  4. // import { Readable } from 'stream';
  5. import { parse } from 'csv-parse/sync';
  6. import { fetchRemoteTextByLine } from './lib/fetch-text-by-line';
  7. import path from 'node:path';
  8. import { OUTPUT_SURGE_DIR } from './constants/dir';
  9. import { createRetrieKeywordFilter as createKeywordFilter } from 'foxts/retrie';
  10. import { $$fetch } from './lib/fetch-retry';
  11. import runAgainstSourceFile from './lib/run-against-source-file';
  12. export async function parseGfwList() {
  13. const whiteSet = new Set<string>();
  14. const trie = new HostnameSmolTrie();
  15. const excludeGfwList = createKeywordFilter([
  16. '.*',
  17. '*',
  18. '=',
  19. '[',
  20. '/',
  21. '?'
  22. ]);
  23. const text = await (await $$fetch('https://raw.githubusercontent.com/gfwlist/gfwlist/master/gfwlist.txt')).text();
  24. for (const l of atob(text).split('\n')) {
  25. const line = processLine(l);
  26. if (!line) continue;
  27. if (excludeGfwList(line)) {
  28. continue;
  29. }
  30. if (line.startsWith('@@||')) {
  31. whiteSet.add('.' + line.slice(4));
  32. continue;
  33. }
  34. if (line.startsWith('@@|http://')) {
  35. whiteSet.add(line.slice(10));
  36. continue;
  37. }
  38. if (line.startsWith('@@|https://')) {
  39. whiteSet.add(line.slice(11));
  40. continue;
  41. }
  42. if (line.startsWith('||')) {
  43. trie.add('.' + line.slice(2));
  44. continue;
  45. }
  46. if (line.startsWith('|')) {
  47. trie.add(line.slice(1));
  48. continue;
  49. }
  50. if (line.startsWith('.')) {
  51. trie.add(line);
  52. continue;
  53. }
  54. const d = fastNormalizeDomain(line);
  55. if (d) {
  56. trie.add(d);
  57. continue;
  58. }
  59. }
  60. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/cn-blocked-domain/release/domains.txt', true)) {
  61. trie.add(l);
  62. }
  63. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/v2ray-rules-dat/release/gfw.txt', true)) {
  64. trie.add(l);
  65. }
  66. const topDomainsRes = await (await $$fetch('https://downloads.majestic.com/majestic_million.csv', {
  67. headers: {
  68. accept: '*/*',
  69. 'user-agent': 'curl/8.12.1'
  70. }
  71. })).text();
  72. const topDomains = parse(topDomainsRes);
  73. const keywordSet = new Set<string>();
  74. const callback = (domain: string, includeAllSubdomain: boolean) => {
  75. trie.whitelist(domain, includeAllSubdomain);
  76. };
  77. await Promise.all([
  78. runAgainstSourceFile(path.join(OUTPUT_SURGE_DIR, 'non_ip/global.conf'), callback, 'ruleset'),
  79. runAgainstSourceFile(path.join(OUTPUT_SURGE_DIR, 'non_ip/reject.conf'), callback, 'ruleset'),
  80. runAgainstSourceFile(path.join(OUTPUT_SURGE_DIR, 'non_ip/telegram.conf'), callback, 'ruleset'),
  81. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'non_ip/stream.conf'), callback, 'ruleset'),
  82. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'non_ip/ai.conf'), callback, 'ruleset'),
  83. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'non_ip/microsoft.conf'), callback, 'ruleset'),
  84. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject.conf'), callback, 'domainset'),
  85. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject_extra.conf'), callback, 'domainset'),
  86. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/cdn.conf'), callback, 'domainset')
  87. ]);
  88. whiteSet.forEach(domain => trie.whitelist(domain));
  89. const kwfilter = createKeywordFilter([...keywordSet]);
  90. const missingTop10000Gfwed = new Set<string>();
  91. for await (const [domain] of topDomains) {
  92. if (trie.has(domain) && !kwfilter(domain)) {
  93. missingTop10000Gfwed.add(domain);
  94. }
  95. }
  96. console.log(missingTop10000Gfwed.size, '');
  97. console.log(Array.from(missingTop10000Gfwed).join('\n'));
  98. return [
  99. whiteSet,
  100. trie,
  101. missingTop10000Gfwed
  102. ] as const;
  103. }
  104. if (require.main === module) {
  105. parseGfwList().catch(console.error);
  106. }