build-public.ts 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. import path from 'node:path';
  2. import fs from 'node:fs';
  3. import fsp from 'node:fs/promises';
  4. import { task } from './trace';
  5. import { treeDir, TreeFileType } from './lib/tree-dir';
  6. import type { TreeType, TreeTypeArray } from './lib/tree-dir';
  7. import { OUTPUT_MOCK_DIR, OUTPUT_MODULES_RULES_DIR, PUBLIC_DIR, ROOT_DIR } from './constants/dir';
  8. import { fastStringCompare, mkdirp, writeFile } from './lib/misc';
  9. import type { VoidOrVoidArray } from './lib/misc';
  10. import picocolors from 'picocolors';
  11. import { tagged as html } from 'foxts/tagged';
  12. import { compareAndWriteFile } from './lib/create-file';
  13. const mockDir = path.join(ROOT_DIR, 'Mock');
  14. const modulesDir = path.join(ROOT_DIR, 'Modules');
  15. const priorityOrder: Record<'default' | string & {}, number> = {
  16. LICENSE: 0,
  17. domainset: 10,
  18. non_ip: 20,
  19. ip: 30,
  20. List: 40,
  21. Surge: 50,
  22. Clash: 60,
  23. 'sing-box': 70,
  24. Surfboard: 80,
  25. LegacyClashPremium: 81,
  26. Modules: 90,
  27. Script: 100,
  28. Mock: 110,
  29. Assets: 120,
  30. Internal: 130,
  31. default: Number.MAX_VALUE
  32. };
  33. async function copyDirContents(srcDir: string, destDir: string, promises: Array<Promise<VoidOrVoidArray>> = []): Promise<Array<Promise<VoidOrVoidArray>>> {
  34. for await (const entry of await fsp.opendir(srcDir)) {
  35. const src = path.join(srcDir, entry.name);
  36. const dest = path.join(destDir, entry.name);
  37. if (entry.isDirectory()) {
  38. console.warn(picocolors.red('[build public] cant copy directory'), src);
  39. } else {
  40. promises.push(fsp.copyFile(src, dest, fs.constants.COPYFILE_FICLONE));
  41. }
  42. }
  43. return promises;
  44. }
  45. export const buildPublic = task(require.main === module, __filename)(async (span) => {
  46. await span.traceChildAsync('copy rest of the files', async () => {
  47. const p: Array<Promise<any>> = [];
  48. let pt = mkdirp(OUTPUT_MODULES_RULES_DIR);
  49. if (pt) {
  50. p.push(pt.then(() => { copyDirContents(modulesDir, OUTPUT_MODULES_RULES_DIR, p); }));
  51. } else {
  52. p.push(copyDirContents(modulesDir, OUTPUT_MODULES_RULES_DIR, p));
  53. }
  54. pt = mkdirp(OUTPUT_MOCK_DIR);
  55. if (pt) {
  56. p.push(pt.then(() => { copyDirContents(mockDir, OUTPUT_MOCK_DIR, p); }));
  57. } else {
  58. p.push(copyDirContents(mockDir, OUTPUT_MOCK_DIR, p));
  59. }
  60. await Promise.all(p);
  61. });
  62. const html = await span
  63. .traceChild('generate index.html')
  64. .traceAsyncFn(() => treeDir(PUBLIC_DIR).then(generateHtml));
  65. await Promise.all([
  66. compareAndWriteFile(
  67. span,
  68. [
  69. '/*',
  70. ' cache-control: public, max-age=240, stale-while-revalidate=60, stale-if-error=15',
  71. 'https://:project.pages.dev/*',
  72. ' X-Robots-Tag: noindex',
  73. ...Object.keys(priorityOrder)
  74. .map((name) => `/${name}/*\n content-type: text/plain; charset=utf-8\n X-Robots-Tag: noindex`)
  75. ],
  76. path.join(PUBLIC_DIR, '_headers')
  77. ),
  78. compareAndWriteFile(
  79. span,
  80. [
  81. '# <pre>',
  82. '#########################################',
  83. '# Sukka\'s Ruleset - 404 Not Found',
  84. '################## EOF ##################</pre>'
  85. ],
  86. path.join(PUBLIC_DIR, '404.html')
  87. ),
  88. compareAndWriteFile(
  89. span,
  90. [
  91. '# The source code is located at [Sukkaw/Surge](https://github.com/Sukkaw/Surge)',
  92. '',
  93. '![GitHub repo size](https://img.shields.io/github/repo-size/sukkalab/ruleset.skk.moe?style=flat-square)'
  94. ],
  95. path.join(PUBLIC_DIR, 'README.md')
  96. )
  97. ]);
  98. return writeFile(path.join(PUBLIC_DIR, 'index.html'), html);
  99. });
  100. const prioritySorter = (a: TreeType, b: TreeType) => ((priorityOrder[a.name] || priorityOrder.default) - (priorityOrder[b.name] || priorityOrder.default)) || fastStringCompare(a.name, b.name);
  101. function treeHtml(tree: TreeTypeArray) {
  102. let result = '';
  103. tree.sort(prioritySorter);
  104. for (let i = 0, len = tree.length; i < len; i++) {
  105. const entry = tree[i];
  106. if (entry.type === TreeFileType.DIRECTORY) {
  107. result += html`
  108. <li class="folder">
  109. ${entry.name}
  110. <ul>${treeHtml(entry.children)}</ul>
  111. </li>
  112. `;
  113. } else if (/* entry.type === 'file' && */ !entry.name.endsWith('.html') && !entry.name.startsWith('_')) {
  114. result += html`<li><a class="file directory-list-file" href="${entry.path}">${entry.name}</a></li>`;
  115. }
  116. }
  117. return result;
  118. }
  119. function generateHtml(tree: TreeTypeArray) {
  120. return html`
  121. <!DOCTYPE html>
  122. <html lang="en">
  123. <head>
  124. <meta charset="utf-8">
  125. <title>Surge Ruleset Server | Sukka (@SukkaW)</title>
  126. <meta name="viewport" content="width=device-width,initial-scale=1,viewport-fit=cover">
  127. <link href="https://cdn.skk.moe/favicon.ico" rel="icon" type="image/ico">
  128. <link href="https://cdn.skk.moe/favicon/apple-touch-icon.png" rel="apple-touch-icon" sizes="180x180">
  129. <link href="https://cdn.skk.moe/favicon/android-chrome-192x192.png" rel="icon" type="image/png" sizes="192x192">
  130. <link href="https://cdn.skk.moe/favicon/favicon-32x32.png" rel="icon" type="image/png" sizes="32x32">
  131. <link href="https://cdn.skk.moe/favicon/favicon-16x16.png" rel="icon" type="image/png" sizes="16x16">
  132. <meta name="description" content="Sukka 自用的 Surge / Clash Premium 规则组">
  133. <link rel="stylesheet" href="https://cdn.skk.moe/ruleset/css/21d8777a.css" />
  134. <meta property="og:title" content="Surge Ruleset | Sukka (@SukkaW)">
  135. <meta property="og:type" content="Website">
  136. <meta property="og:url" content="https://ruleset.skk.moe/">
  137. <meta property="og:image" content="https://cdn.skk.moe/favicon/android-chrome-192x192.png">
  138. <meta property="og:description" content="Sukka 自用的 Surge / Clash Premium 规则组">
  139. <meta name="twitter:card" content="summary">
  140. <link rel="canonical" href="https://ruleset.skk.moe/">
  141. </head>
  142. <body>
  143. <main class="container">
  144. <h1>Sukka Ruleset Server</h1>
  145. <p>
  146. Made by <a href="https://skk.moe">Sukka</a> | <a href="https://github.com/SukkaW/Surge/">Source @ GitHub</a> | Licensed under <a href="/LICENSE" target="_blank">AGPL-3.0</a>
  147. </p>
  148. <p>Last Build: ${new Date().toISOString()}</p>
  149. <br>
  150. <ul class="directory-list">${treeHtml(tree)}</ul>
  151. </main>
  152. </body>
  153. </html>
  154. `;
  155. }