index.ts 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. import process from 'node:process';
  2. import os from 'node:os';
  3. import fs from 'node:fs';
  4. import fsp from 'node:fs/promises';
  5. import { downloadPreviousBuild } from './download-previous-build';
  6. import { buildCommon } from './build-common';
  7. import { buildRejectIPList } from './build-reject-ip-list';
  8. import { buildAppleCdn } from './build-apple-cdn';
  9. import { buildCdnDownloadConf } from './build-cdn-download-conf';
  10. import { buildRejectDomainSet } from './build-reject-domainset';
  11. import { buildTelegramCIDR } from './build-telegram-cidr';
  12. import { buildChnCidr } from './build-chn-cidr';
  13. import { buildSpeedtestDomainSet } from './build-speedtest-domainset';
  14. import { buildDomesticRuleset } from './build-domestic-direct-lan-ruleset-dns-mapping-module';
  15. import { buildGlobalRuleset } from './build-global-server-dns-mapping';
  16. import { buildStreamService } from './build-stream-service';
  17. import { buildRedirectModule } from './build-sgmodule-redirect';
  18. import { buildAlwaysRealIPModule } from './build-sgmodule-always-realip';
  19. import { buildMicrosoftCdn } from './build-microsoft-cdn';
  20. import { buildSSPanelUIMAppProfile } from './build-sspanel-appprofile';
  21. import { buildPublic } from './build-public';
  22. import { downloadMockAssets } from './download-mock-assets';
  23. import { buildCloudMounterRules } from './build-cloudmounter-rules';
  24. import { createSpan, printTraceResult, whyIsNodeRunning } from './trace';
  25. import { buildDeprecateFiles } from './build-deprecate-files';
  26. import path from 'node:path';
  27. import { CACHE_DIR, ROOT_DIR } from './constants/dir';
  28. import { isCI } from 'ci-info';
  29. process.on('uncaughtException', (error) => {
  30. console.error('Uncaught exception:', error);
  31. process.exit(1);
  32. });
  33. process.on('unhandledRejection', (reason) => {
  34. console.error('Unhandled rejection:', reason);
  35. process.exit(1);
  36. });
  37. const removesFiles = [
  38. path.join(CACHE_DIR, '.cache.db'),
  39. path.join(CACHE_DIR, '.cache.db-shm'),
  40. path.join(CACHE_DIR, '.cache.db-wal')
  41. ];
  42. const buildFinishedLock = path.join(ROOT_DIR, '.BUILD_FINISHED');
  43. (async () => {
  44. console.log('Version:', process.version);
  45. console.log(`OS: ${os.type()} ${os.release()} ${os.arch()}`);
  46. console.log(`Node.js: ${process.versions.node}`);
  47. console.log(`V8: ${process.versions.v8}`);
  48. const cpus = os.cpus()
  49. .reduce<Record<string, number>>((o, cpu) => {
  50. o[cpu.model] = (o[cpu.model] || 0) + 1;
  51. return o;
  52. }, {});
  53. console.log(`CPU: ${
  54. Object.keys(cpus)
  55. .map((key) => `${key} x ${cpus[key]}`)
  56. .join('\n')
  57. }`);
  58. if ('availableParallelism' in os) {
  59. console.log(`Available parallelism: ${os.availableParallelism()}`);
  60. }
  61. console.log(`Memory: ${os.totalmem() / (1024 * 1024)} MiB`);
  62. const rootSpan = createSpan('root');
  63. if (fs.existsSync(buildFinishedLock)) {
  64. fs.unlinkSync(buildFinishedLock);
  65. }
  66. try {
  67. // only enable why-is-node-running in GitHub Actions debug mode
  68. if (isCI && process.env.RUNNER_DEBUG === '1') {
  69. await import('why-is-node-running');
  70. }
  71. const downloadPreviousBuildPromise = downloadPreviousBuild(rootSpan);
  72. const buildCommonPromise = downloadPreviousBuildPromise.then(() => buildCommon(rootSpan));
  73. await Promise.all([
  74. ...removesFiles.map(file => fsp.rm(file, { force: true })),
  75. downloadPreviousBuildPromise,
  76. buildCommonPromise,
  77. downloadPreviousBuildPromise.then(() => buildRejectIPList(rootSpan)),
  78. downloadPreviousBuildPromise.then(() => buildAppleCdn(rootSpan)),
  79. downloadPreviousBuildPromise.then(() => buildCdnDownloadConf(rootSpan)),
  80. downloadPreviousBuildPromise.then(() => buildRejectDomainSet(rootSpan)),
  81. downloadPreviousBuildPromise.then(() => buildTelegramCIDR(rootSpan)),
  82. downloadPreviousBuildPromise.then(() => buildChnCidr(rootSpan)),
  83. downloadPreviousBuildPromise.then(() => buildSpeedtestDomainSet(rootSpan)),
  84. downloadPreviousBuildPromise.then(() => buildDomesticRuleset(rootSpan)),
  85. downloadPreviousBuildPromise.then(() => buildGlobalRuleset(rootSpan)),
  86. downloadPreviousBuildPromise.then(() => buildRedirectModule(rootSpan)),
  87. downloadPreviousBuildPromise.then(() => buildAlwaysRealIPModule(rootSpan)),
  88. downloadPreviousBuildPromise.then(() => buildStreamService(rootSpan)),
  89. downloadPreviousBuildPromise.then(() => buildMicrosoftCdn(rootSpan)),
  90. Promise.all([
  91. downloadPreviousBuildPromise,
  92. buildCommonPromise
  93. ]).then(() => buildSSPanelUIMAppProfile(rootSpan)),
  94. downloadPreviousBuildPromise.then(() => buildCloudMounterRules(rootSpan)),
  95. downloadMockAssets(rootSpan)
  96. ]);
  97. await buildDeprecateFiles(rootSpan);
  98. await buildPublic(rootSpan);
  99. rootSpan.stop();
  100. printTraceResult(rootSpan.traceResult);
  101. // write a file to demonstrate that the build is finished
  102. fs.writeFileSync(buildFinishedLock, 'BUILD_FINISHED\n');
  103. // Finish the build to avoid leaking timer/fetch ref
  104. await whyIsNodeRunning();
  105. process.exit(0);
  106. } catch (e) {
  107. console.error('Something went wrong!');
  108. console.trace(e);
  109. process.exit(1);
  110. }
  111. })();