build-cdn-conf.js 2.7 KB

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