server.js 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. "use strict";
  2. var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
  3. if (k2 === undefined) k2 = k;
  4. var desc = Object.getOwnPropertyDescriptor(m, k);
  5. if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
  6. desc = { enumerable: true, get: function() { return m[k]; } };
  7. }
  8. Object.defineProperty(o, k2, desc);
  9. }) : (function(o, m, k, k2) {
  10. if (k2 === undefined) k2 = k;
  11. o[k2] = m[k];
  12. }));
  13. var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
  14. Object.defineProperty(o, "default", { enumerable: true, value: v });
  15. }) : function(o, v) {
  16. o["default"] = v;
  17. });
  18. var __importStar = (this && this.__importStar) || function (mod) {
  19. if (mod && mod.__esModule) return mod;
  20. var result = {};
  21. if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
  22. __setModuleDefault(result, mod);
  23. return result;
  24. };
  25. var __importDefault = (this && this.__importDefault) || function (mod) {
  26. return (mod && mod.__esModule) ? mod : { "default": mod };
  27. };
  28. Object.defineProperty(exports, "__esModule", { value: true });
  29. exports.createSSRServer = exports.createServer = void 0;
  30. const fs_1 = __importDefault(require("fs"));
  31. const os_1 = __importDefault(require("os"));
  32. const path_1 = __importDefault(require("path"));
  33. const picocolors_1 = __importDefault(require("picocolors"));
  34. const express_1 = __importDefault(require("express"));
  35. const uni_cli_shared_1 = require("@dcloudio/uni-cli-shared");
  36. const utils_1 = require("./utils");
  37. function createLogger(level, options) {
  38. return Promise.resolve().then(() => __importStar(require('vite'))).then(({ createLogger }) => createLogger(level, options));
  39. }
  40. function createViteServer(inlineConfig) {
  41. return Promise.resolve().then(() => __importStar(require('vite'))).then(({ createServer }) => createServer(inlineConfig));
  42. }
  43. async function createServer(options) {
  44. const server = await createViteServer((0, utils_1.addConfigFile)({
  45. root: process.env.VITE_ROOT_DIR,
  46. configFile: options.config,
  47. base: options.base,
  48. mode: options.mode,
  49. logLevel: options.logLevel || 'info',
  50. clearScreen: options.clearScreen,
  51. server: (0, utils_1.cleanOptions)(options),
  52. }));
  53. await server.listen();
  54. const logger = server.config.logger;
  55. logger.info(picocolors_1.default.cyan(`\n vite v${require('vite/package.json').version}`) +
  56. picocolors_1.default.green(` dev server running at:\n`), {
  57. clear: !server.config.logger.hasWarned,
  58. });
  59. server.printUrls();
  60. // printUrls 会在 nextTick 中输出
  61. process.nextTick(() => (0, utils_1.printStartupDuration)(logger));
  62. return server;
  63. }
  64. exports.createServer = createServer;
  65. async function createSSRServer(options) {
  66. const app = (0, express_1.default)();
  67. /**
  68. * @type {import('vite').ViteDevServer}
  69. */
  70. const vite = await createViteServer((0, utils_1.addConfigFile)({
  71. // custom: don't include HTML middlewares
  72. appType: 'custom',
  73. root: process.env.VITE_ROOT_DIR,
  74. configFile: options.config,
  75. base: options.base,
  76. mode: options.mode,
  77. logLevel: options.logLevel || 'info',
  78. clearScreen: options.clearScreen,
  79. server: {
  80. middlewareMode: true,
  81. watch: {
  82. // During tests we edit the files too fast and sometimes chokidar
  83. // misses change events, so enforce polling for consistency
  84. usePolling: true,
  85. interval: 100,
  86. },
  87. },
  88. }));
  89. // use vite's connect instance as middleware
  90. app.use(vite.middlewares);
  91. app.use('*', async (req, res) => {
  92. try {
  93. const { h5 } = (0, uni_cli_shared_1.parseManifestJson)(process.env.UNI_INPUT_DIR);
  94. const base = (h5 && h5.router && h5.router.base) || '';
  95. const url = req.originalUrl.replace(base, '');
  96. const template = await vite.transformIndexHtml(url, fs_1.default.readFileSync(path_1.default.resolve(process.env.VITE_ROOT_DIR, 'index.html'), 'utf-8'));
  97. const render = (await vite.ssrLoadModule(path_1.default.resolve(process.env.UNI_INPUT_DIR, 'entry-server.js'))).render;
  98. const { title, headMeta, preloadLinks, appHtml, appContext } = await render(url);
  99. const icon = template.includes('rel="icon"')
  100. ? ''
  101. : '<link rel="icon" href="data:," />\n';
  102. const html = template
  103. .replace(/<title>(.*?)<\/title>/, `${icon}<title>${title}</title>`)
  104. .replace(`<!--head-meta-->`, headMeta)
  105. .replace(`<!--preload-links-->`, preloadLinks)
  106. .replace(`<!--app-html-->`, appHtml)
  107. .replace(`<!--app-context-->`, appContext);
  108. res.status(200).set({ 'Content-Type': 'text/html' }).end(html);
  109. }
  110. catch (e) {
  111. vite && vite.ssrFixStacktrace(e);
  112. res.status(500).end(e.stack);
  113. }
  114. });
  115. const logger = await createLogger(options.logLevel);
  116. const serverOptions = vite.config.server || {};
  117. let port = options.port || serverOptions.port || 5173;
  118. let hostname;
  119. if (options.host === 'localhost') {
  120. // Use a secure default
  121. hostname = '127.0.0.1';
  122. }
  123. else if (options.host === undefined || options.host === true) {
  124. // probably passed --host in the CLI, without arguments
  125. hostname = undefined; // undefined typically means 0.0.0.0 or :: (listen on all IPs)
  126. }
  127. else {
  128. hostname = options.host;
  129. }
  130. return new Promise((resolve, reject) => {
  131. const onSuccess = () => {
  132. printHttpServerUrls(server, vite.config);
  133. process.nextTick(() => (0, utils_1.printStartupDuration)(logger));
  134. resolve(vite);
  135. };
  136. const onError = (e) => {
  137. if (e.code === 'EADDRINUSE') {
  138. if (options.strictPort) {
  139. server.off('error', onError);
  140. reject(new Error(`Port ${port} is already in use`));
  141. }
  142. else {
  143. logger.info(`Port ${port} is in use, trying another one...`);
  144. app.listen(++port, hostname, onSuccess).on('error', onError);
  145. }
  146. }
  147. else {
  148. server.off('error', onError);
  149. reject(e);
  150. }
  151. };
  152. const server = app.listen(port, hostname, onSuccess).on('error', onError);
  153. });
  154. }
  155. exports.createSSRServer = createSSRServer;
  156. function printHttpServerUrls(server, config) {
  157. printCommonServerUrls(server, config.server, config);
  158. }
  159. function printCommonServerUrls(server, options, config) {
  160. const address = server.address();
  161. const isAddressInfo = (x) => x?.address;
  162. if (isAddressInfo(address)) {
  163. const hostname = resolveHostname(options.host);
  164. const protocol = options.https ? 'https' : 'http';
  165. printServerUrls(hostname, protocol, address.port, config.base, config.logger.info);
  166. }
  167. }
  168. function printServerUrls(hostname, protocol, port, base, info) {
  169. if (hostname.host === '127.0.0.1') {
  170. const url = `${protocol}://${hostname.name}:${picocolors_1.default.bold(port)}${base}`;
  171. info(` - Local: ${picocolors_1.default.cyan(url)}`);
  172. if (hostname.name !== '127.0.0.1') {
  173. info(` - Network: ${picocolors_1.default.dim('use `--host` to expose')}`);
  174. }
  175. }
  176. else {
  177. Object.values(os_1.default.networkInterfaces())
  178. .flatMap((nInterface) => nInterface ?? [])
  179. .filter((detail) => detail &&
  180. detail.address &&
  181. // Node < v18
  182. ((typeof detail.family === 'string' && detail.family === 'IPv4') ||
  183. // Node >= v18
  184. // @ts-ignore
  185. (typeof detail.family === 'number' && detail.family === 4)))
  186. .map((detail) => {
  187. const type = detail.address.includes('127.0.0.1')
  188. ? ' - Local: '
  189. : ' * Network: ';
  190. const host = detail.address.replace('127.0.0.1', hostname.name);
  191. const url = `${protocol}://${host}:${picocolors_1.default.bold(port)}${base}`;
  192. return `${type} ${picocolors_1.default.cyan(url)}`;
  193. })
  194. .sort((msg1) => {
  195. return msg1.indexOf('- Local') > -1 ? -1 : 1;
  196. })
  197. .forEach((msg, index, arr) => {
  198. if (arr.length - 1 === index) {
  199. info(msg.replace('* Network', '- Network'));
  200. }
  201. else {
  202. info(msg);
  203. }
  204. });
  205. }
  206. }
  207. function resolveHostname(optionsHost) {
  208. let host;
  209. if (optionsHost === undefined || optionsHost === false) {
  210. // Use a secure default
  211. host = '127.0.0.1';
  212. }
  213. else if (optionsHost === true) {
  214. // If passed --host in the CLI without arguments
  215. host = undefined; // undefined typically means 0.0.0.0 or :: (listen on all IPs)
  216. }
  217. else {
  218. host = optionsHost;
  219. }
  220. // Set host name to localhost when possible, unless the user explicitly asked for '127.0.0.1'
  221. const name = (optionsHost !== '127.0.0.1' && host === '127.0.0.1') ||
  222. host === '0.0.0.0' ||
  223. host === '::' ||
  224. host === undefined
  225. ? 'localhost'
  226. : host;
  227. return { host, name };
  228. }