validate-gfwlist.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. import { processLine } from './lib/process-line';
  2. import { fastNormalizeDomain } from './lib/normalize-domain';
  3. import { HostnameSmolTrie } from './lib/trie';
  4. import yauzl from 'yauzl-promise';
  5. import { fetchRemoteTextByLine } from './lib/fetch-text-by-line';
  6. import path from 'node:path';
  7. import { OUTPUT_SURGE_DIR, SOURCE_DIR } from './constants/dir';
  8. import { createRetrieKeywordFilter as createKeywordFilter } from 'foxts/retrie';
  9. import { $$fetch } from './lib/fetch-retry';
  10. import runAgainstSourceFile from './lib/run-against-source-file';
  11. import { nullthrow } from 'foxts/guard';
  12. import { Buffer } from 'node:buffer';
  13. import { GLOBAL } from '../Source/non_ip/global';
  14. export async function getTopOneMillionDomains() {
  15. const { parse: csvParser } = await import('csv-parse');
  16. const topDomainTrie = new HostnameSmolTrie();
  17. const csvParse = csvParser({ columns: false, skip_empty_lines: true });
  18. const topDomainsZipBody = await (await $$fetch('https://tranco-list.eu/top-1m.csv.zip', {
  19. headers: {
  20. accept: '*/*',
  21. 'user-agent': 'curl/8.12.1'
  22. }
  23. })).arrayBuffer();
  24. let entry: yauzl.Entry | null = null;
  25. for await (const e of await yauzl.fromBuffer(Buffer.from(topDomainsZipBody))) {
  26. if (e.filename === 'top-1m.csv') {
  27. entry = e;
  28. break;
  29. }
  30. }
  31. const { promise, resolve, reject } = Promise.withResolvers<HostnameSmolTrie>();
  32. const readable = await nullthrow(entry, 'top-1m.csv entry not found').openReadStream();
  33. const parser = readable.pipe(csvParse);
  34. parser.on('readable', () => {
  35. let record;
  36. while ((record = parser.read()) !== null) {
  37. topDomainTrie.add(record[1]);
  38. }
  39. });
  40. parser.on('end', () => {
  41. resolve(topDomainTrie);
  42. });
  43. parser.on('error', (err) => {
  44. reject(err);
  45. });
  46. return promise;
  47. }
  48. export async function parseGfwList() {
  49. const whiteSet = new Set<string>();
  50. const gfwListTrie = new HostnameSmolTrie();
  51. let totalGfwSize = 0;
  52. const gfwlistIgnoreLineKwfilter = createKeywordFilter([
  53. '.*',
  54. '*',
  55. '=',
  56. '[',
  57. '/',
  58. '?'
  59. ]);
  60. const text = await (await $$fetch('https://raw.githubusercontent.com/gfwlist/gfwlist/master/gfwlist.txt')).text();
  61. for (const l of atob(text).split('\n')) {
  62. const line = processLine(l);
  63. if (!line) continue;
  64. if (gfwlistIgnoreLineKwfilter(line)) {
  65. continue;
  66. }
  67. if (line.startsWith('@@||')) {
  68. whiteSet.add('.' + line.slice(4));
  69. continue;
  70. }
  71. if (line.startsWith('@@|http://')) {
  72. whiteSet.add(line.slice(10));
  73. continue;
  74. }
  75. if (line.startsWith('@@|https://')) {
  76. whiteSet.add(line.slice(11));
  77. continue;
  78. }
  79. if (line.startsWith('||')) {
  80. gfwListTrie.add('.' + line.slice(2));
  81. continue;
  82. }
  83. if (line.startsWith('|')) {
  84. gfwListTrie.add(line.slice(1));
  85. continue;
  86. }
  87. if (line.startsWith('.')) {
  88. gfwListTrie.add(line);
  89. continue;
  90. }
  91. const d = fastNormalizeDomain(line);
  92. if (d) {
  93. totalGfwSize++;
  94. gfwListTrie.add(d);
  95. continue;
  96. }
  97. }
  98. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/cn-blocked-domain/release/domains.txt', true)) {
  99. totalGfwSize++;
  100. gfwListTrie.add(l);
  101. }
  102. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/v2ray-rules-dat/release/gfw.txt', true)) {
  103. totalGfwSize++;
  104. gfwListTrie.add(l);
  105. }
  106. const topDomainTrie = await getTopOneMillionDomains();
  107. const keywordSet = new Set<string>();
  108. const callback = (domain: string, includeAllSubdomain: boolean) => {
  109. gfwListTrie.whitelist(domain, includeAllSubdomain);
  110. topDomainTrie.whitelist(domain, includeAllSubdomain);
  111. };
  112. await Promise.all([
  113. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/global.conf'), callback, 'ruleset', keywordSet),
  114. // runAgainstSourceFile(path.join(OUTPUT_SURGE_DIR, 'non_ip/domestic.conf'), callback, 'ruleset', keywordSet),
  115. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/reject.conf'), callback, 'ruleset', keywordSet),
  116. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/telegram.conf'), callback, 'ruleset', keywordSet),
  117. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'non_ip/stream.conf'), callback, 'ruleset', keywordSet),
  118. runAgainstSourceFile(path.resolve(SOURCE_DIR, 'non_ip/ai.conf'), callback, 'ruleset', keywordSet),
  119. runAgainstSourceFile(path.resolve(SOURCE_DIR, 'non_ip/microsoft.conf'), callback, 'ruleset', keywordSet),
  120. runAgainstSourceFile(path.resolve(SOURCE_DIR, 'non_ip/apple_services.conf'), callback, 'ruleset', keywordSet),
  121. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject.conf'), callback, 'domainset'),
  122. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject_extra.conf'), callback, 'domainset'),
  123. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/cdn.conf'), callback, 'domainset')
  124. ]);
  125. Object.values(GLOBAL).forEach(({ domains }) => {
  126. domains.forEach(domain => {
  127. if (domain[0] === '$') {
  128. callback(domain.slice(1), false);
  129. } else if (domain[0] === '+') {
  130. callback(domain.slice(1), true);
  131. } else {
  132. callback(domain, true);
  133. }
  134. });
  135. });
  136. whiteSet.forEach(domain => gfwListTrie.whitelist(domain, true));
  137. let gfwListSize = 0;
  138. gfwListTrie.dump(() => gfwListSize++);
  139. const kwfilter = createKeywordFilter([...keywordSet]);
  140. const missingTop10000Gfwed = new Set<string>();
  141. topDomainTrie.dump((domain) => {
  142. if (gfwListTrie.has(domain) && !kwfilter(domain)) {
  143. missingTop10000Gfwed.add(domain);
  144. }
  145. });
  146. console.log(Array.from(missingTop10000Gfwed).join('\n'));
  147. console.log({ totalGfwSize, gfwListSize, missingSize: missingTop10000Gfwed.size });
  148. return [
  149. whiteSet,
  150. gfwListTrie,
  151. missingTop10000Gfwed
  152. ] as const;
  153. }
  154. if (require.main === module) {
  155. parseGfwList().catch(console.error);
  156. }
  157. // python.com waiting-for-sell