build-cdn-conf.ts 2.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  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. const line = processLine(l);
  57. if (line) {
  58. cdnDomainsList.push(line);
  59. }
  60. }
  61. (await getS3OSSDomainsPromise).forEach((domain: string) => { cdnDomainsList.push(`DOMAIN-SUFFIX,${domain}`); });
  62. const description: string[] = [
  63. ...SHARED_DESCRIPTION,
  64. '',
  65. 'This file contains object storage and static assets CDN domains.'
  66. ];
  67. return Promise.all(createRuleset(
  68. 'Sukka\'s Ruleset - CDN Domains',
  69. description,
  70. new Date(),
  71. cdnDomainsList,
  72. 'ruleset',
  73. path.resolve(import.meta.dir, '../List/non_ip/cdn.conf'),
  74. path.resolve(import.meta.dir, '../Clash/non_ip/cdn.txt')
  75. ));
  76. });
  77. export { buildCdnConf };
  78. if (import.meta.main) {
  79. buildCdnConf();
  80. }