validate-gfwlist.ts 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  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. export async function parseGfwList() {
  14. const { parse: csvParser } = await import('csv-parse');
  15. const whiteSet = new Set<string>();
  16. const gfwListTrie = new HostnameSmolTrie();
  17. const excludeGfwList = createKeywordFilter([
  18. '.*',
  19. '*',
  20. '=',
  21. '[',
  22. '/',
  23. '?'
  24. ]);
  25. const text = await (await $$fetch('https://raw.githubusercontent.com/gfwlist/gfwlist/master/gfwlist.txt')).text();
  26. for (const l of atob(text).split('\n')) {
  27. const line = processLine(l);
  28. if (!line) continue;
  29. if (excludeGfwList(line)) {
  30. continue;
  31. }
  32. if (line.startsWith('@@||')) {
  33. whiteSet.add('.' + line.slice(4));
  34. continue;
  35. }
  36. if (line.startsWith('@@|http://')) {
  37. whiteSet.add(line.slice(10));
  38. continue;
  39. }
  40. if (line.startsWith('@@|https://')) {
  41. whiteSet.add(line.slice(11));
  42. continue;
  43. }
  44. if (line.startsWith('||')) {
  45. gfwListTrie.add('.' + line.slice(2));
  46. continue;
  47. }
  48. if (line.startsWith('|')) {
  49. gfwListTrie.add(line.slice(1));
  50. continue;
  51. }
  52. if (line.startsWith('.')) {
  53. gfwListTrie.add(line);
  54. continue;
  55. }
  56. const d = fastNormalizeDomain(line);
  57. if (d) {
  58. gfwListTrie.add(d);
  59. continue;
  60. }
  61. }
  62. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/cn-blocked-domain/release/domains.txt', true)) {
  63. gfwListTrie.add(l);
  64. }
  65. for await (const l of await fetchRemoteTextByLine('https://raw.githubusercontent.com/Loyalsoldier/v2ray-rules-dat/release/gfw.txt', true)) {
  66. gfwListTrie.add(l);
  67. }
  68. const topDomainTrie = new HostnameSmolTrie();
  69. const csvParse = csvParser({ columns: false, skip_empty_lines: true });
  70. const topDomainsZipBody = await (await $$fetch('https://s3-us-west-1.amazonaws.com/umbrella-static/top-1m.csv.zip', {
  71. headers: {
  72. accept: '*/*',
  73. 'user-agent': 'curl/8.12.1'
  74. }
  75. })).arrayBuffer();
  76. let entry: yauzl.Entry | null = null;
  77. for await (const e of await yauzl.fromBuffer(Buffer.from(topDomainsZipBody))) {
  78. if (e.filename === 'top-1m.csv') {
  79. entry = e;
  80. break;
  81. }
  82. }
  83. const { promise, resolve, reject } = Promise.withResolvers<HostnameSmolTrie>();
  84. const readable = await nullthrow(entry, 'top-1m.csv entry not found').openReadStream();
  85. const parser = readable.pipe(csvParse);
  86. parser.on('readable', () => {
  87. let record;
  88. while ((record = parser.read()) !== null) {
  89. topDomainTrie.add(record[1]);
  90. }
  91. });
  92. parser.on('end', () => {
  93. resolve(topDomainTrie);
  94. });
  95. parser.on('error', (err) => {
  96. reject(err);
  97. });
  98. await promise;
  99. const keywordSet = new Set<string>();
  100. const callback = (domain: string, includeAllSubdomain: boolean) => {
  101. gfwListTrie.whitelist(domain, includeAllSubdomain);
  102. topDomainTrie.whitelist(domain, includeAllSubdomain);
  103. };
  104. await Promise.all([
  105. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/global.conf'), callback, 'ruleset', keywordSet),
  106. runAgainstSourceFile(path.join(OUTPUT_SURGE_DIR, 'non_ip/domestic.conf'), callback, 'ruleset', keywordSet),
  107. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/reject.conf'), callback, 'ruleset', keywordSet),
  108. runAgainstSourceFile(path.join(SOURCE_DIR, 'non_ip/telegram.conf'), callback, 'ruleset', keywordSet),
  109. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'non_ip/stream.conf'), callback, 'ruleset', keywordSet),
  110. runAgainstSourceFile(path.resolve(SOURCE_DIR, 'non_ip/ai.conf'), callback, 'ruleset', keywordSet),
  111. runAgainstSourceFile(path.resolve(SOURCE_DIR, 'non_ip/microsoft.conf'), callback, 'ruleset', keywordSet),
  112. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject.conf'), callback, 'domainset'),
  113. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/reject_extra.conf'), callback, 'domainset'),
  114. runAgainstSourceFile(path.resolve(OUTPUT_SURGE_DIR, 'domainset/cdn.conf'), callback, 'domainset')
  115. ]);
  116. whiteSet.forEach(domain => gfwListTrie.whitelist(domain));
  117. const kwfilter = createKeywordFilter([...keywordSet]);
  118. const missingTop10000Gfwed = new Set<string>();
  119. topDomainTrie.dump((domain) => {
  120. if (gfwListTrie.has(domain) && !kwfilter(domain)) {
  121. missingTop10000Gfwed.add(domain);
  122. }
  123. });
  124. console.log(missingTop10000Gfwed.size, '');
  125. console.log(Array.from(missingTop10000Gfwed).join('\n'));
  126. return [
  127. whiteSet,
  128. gfwListTrie,
  129. missingTop10000Gfwed
  130. ] as const;
  131. }
  132. if (require.main === module) {
  133. parseGfwList().catch(console.error);
  134. }