build-cdn-conf.ts 2.8 KB

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