build-cdn-conf.ts 2.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. import path from 'path';
  2. import { createRuleset } from './lib/create-file';
  3. import { fetchRemoteTextAndReadByLine, readFileByLine } from './lib/fetch-text-by-line';
  4. import { createTrie } from './lib/trie';
  5. import { task } from './lib/trace-runner';
  6. import { processLine } from './lib/process-line';
  7. import { SHARED_DESCRIPTION } from './lib/constants';
  8. const publicSuffixPath: string = path.resolve(import.meta.dir, '../node_modules/.cache/public_suffix_list_dat.txt');
  9. const getS3OSSDomains = async (): Promise<Set<string>> => {
  10. const trie = createTrie();
  11. const publicSuffixFile = Bun.file(publicSuffixPath);
  12. if (await publicSuffixFile.exists()) {
  13. for await (const line of readFileByLine(publicSuffixFile)) {
  14. trie.add(line);
  15. }
  16. } else {
  17. console.log('public_suffix_list.dat not found, fetch directly from remote.');
  18. for await (const line of await fetchRemoteTextAndReadByLine('https://publicsuffix.org/list/public_suffix_list.dat')) {
  19. trie.add(line);
  20. }
  21. }
  22. /**
  23. * Extract OSS domain from publicsuffix list
  24. */
  25. const S3OSSDomains = new Set<string>();
  26. trie.find('.amazonaws.com').forEach((line: string) => {
  27. if (
  28. (line.startsWith('s3-') || line.startsWith('s3.'))
  29. && !line.includes('cn-')
  30. ) {
  31. S3OSSDomains.add(line);
  32. }
  33. });
  34. trie.find('.scw.cloud').forEach((line: string) => {
  35. if (
  36. (line.startsWith('s3-') || line.startsWith('s3.'))
  37. && !line.includes('cn-')
  38. ) {
  39. S3OSSDomains.add(line);
  40. }
  41. });
  42. trie.find('sakurastorage.jp').forEach((line: string) => {
  43. if (
  44. (line.startsWith('s3-') || line.startsWith('s3.'))
  45. ) {
  46. S3OSSDomains.add(line);
  47. }
  48. });
  49. return S3OSSDomains;
  50. };
  51. const buildCdnConf = task(import.meta.path, async () => {
  52. /** @type {string[]} */
  53. const cdnDomainsList: string[] = [];
  54. const getS3OSSDomainsPromise: Promise<Set<string>> = getS3OSSDomains();
  55. for await (const l of readFileByLine(path.resolve(import.meta.dir, '../Source/non_ip/cdn.conf'))) {
  56. if (l === '# --- [AWS S3 Replace Me] ---') {
  57. (await getS3OSSDomainsPromise).forEach((domain: string) => { cdnDomainsList.push(`DOMAIN-SUFFIX,${domain}`); });
  58. continue;
  59. }
  60. const line = processLine(l);
  61. if (line) {
  62. cdnDomainsList.push(line);
  63. }
  64. }
  65. const description: string[] = [
  66. ...SHARED_DESCRIPTION,
  67. '',
  68. 'This file contains object storage and static assets CDN domains.'
  69. ];
  70. return Promise.all(createRuleset(
  71. 'Sukka\'s Ruleset - CDN Domains',
  72. description,
  73. new Date(),
  74. cdnDomainsList,
  75. 'ruleset',
  76. path.resolve(import.meta.dir, '../List/non_ip/cdn.conf'),
  77. path.resolve(import.meta.dir, '../Clash/non_ip/cdn.txt')
  78. ));
  79. });
  80. export { buildCdnConf };
  81. if (import.meta.main) {
  82. buildCdnConf();
  83. }