@anansi/core 0.21.0 → 0.21.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +12 -0
- package/dist/server.js +91 -5
- package/lib/laySpouts.d.ts.map +1 -1
- package/lib/laySpouts.js +9 -5
- package/lib/scripts/laySpouts.js +9 -5
- package/lib/scripts/scripts/serve.js +17 -7
- package/lib/scripts/scripts/ssrErrorHandler.js +82 -0
- package/lib/scripts/scripts/startDevserver.js +18 -4
- package/lib/scripts/serve.d.ts.map +1 -1
- package/lib/scripts/serve.js +17 -7
- package/lib/scripts/ssrErrorHandler.d.ts +26 -0
- package/lib/scripts/ssrErrorHandler.d.ts.map +1 -0
- package/lib/scripts/ssrErrorHandler.js +82 -0
- package/lib/scripts/startDevserver.d.ts.map +1 -1
- package/lib/scripts/startDevserver.js +18 -4
- package/package.json +3 -3
- package/src/laySpouts.tsx +9 -5
- package/src/scripts/__tests__/ssrErrorHandler.test.ts +249 -0
- package/src/scripts/serve.ts +18 -6
- package/src/scripts/ssrErrorHandler.ts +98 -0
- package/src/scripts/startDevserver.ts +19 -3
|
@@ -7,10 +7,10 @@ import express from 'express';
|
|
|
7
7
|
import diskFs from 'fs';
|
|
8
8
|
import ora from 'ora';
|
|
9
9
|
import path from 'path';
|
|
10
|
-
import { promisify } from 'util';
|
|
11
10
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
12
11
|
import getProxyMiddlewares from './getProxyMiddlewares.js';
|
|
13
12
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
13
|
+
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
14
14
|
// run directly from node
|
|
15
15
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
16
16
|
// @ts-ignore
|
|
@@ -29,14 +29,24 @@ export default async function serve(entrypoint, options = {}) {
|
|
|
29
29
|
const manifestPath = getManifestPathFromWebpackconfig(webpackConfig({}, {
|
|
30
30
|
mode: 'production'
|
|
31
31
|
}));
|
|
32
|
-
const readFile = promisify(diskFs.readFile);
|
|
33
32
|
let server;
|
|
34
33
|
function handleErrors(fn) {
|
|
35
|
-
return async function (req, res,
|
|
34
|
+
return async function (req, res, _next) {
|
|
36
35
|
try {
|
|
37
36
|
return await fn(req, res);
|
|
38
|
-
} catch (
|
|
39
|
-
|
|
37
|
+
} catch (error) {
|
|
38
|
+
console.error('SSR rendering error:', error);
|
|
39
|
+
|
|
40
|
+
// Return error response with status from error if available
|
|
41
|
+
const expressRes = res;
|
|
42
|
+
if (!expressRes.headersSent) {
|
|
43
|
+
const statusCode = getErrorStatus(error);
|
|
44
|
+
expressRes.status(statusCode);
|
|
45
|
+
expressRes.setHeader('Content-Type', 'text/html');
|
|
46
|
+
expressRes.send(renderErrorPage(error, req.url ?? '/', statusCode, {
|
|
47
|
+
showStack: process.env.NODE_ENV !== 'production'
|
|
48
|
+
}));
|
|
49
|
+
}
|
|
40
50
|
}
|
|
41
51
|
};
|
|
42
52
|
}
|
|
@@ -62,7 +72,7 @@ export default async function serve(entrypoint, options = {}) {
|
|
|
62
72
|
if (diskFs.existsSync(assetPath) && !diskFs.lstatSync(assetPath).isDirectory()) {
|
|
63
73
|
try {
|
|
64
74
|
res.sendFile(assetPath);
|
|
65
|
-
} catch (
|
|
75
|
+
} catch (_e) {
|
|
66
76
|
return next();
|
|
67
77
|
}
|
|
68
78
|
} else {
|
|
@@ -154,4 +164,4 @@ function getManifestPathFromWebpackconfig(webpackConfig) {
|
|
|
154
164
|
const manifestPath = path.join(webpackConfig?.output?.path ?? '', manifestFilename);
|
|
155
165
|
return manifestPath;
|
|
156
166
|
}
|
|
157
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","compress","express","diskFs","ora","path","promisify","getProxyMiddlewares","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","serve","options","PORT","env","loader","start","webpackConfig","manifestPath","getManifestPathFromWebpackconfig","mode","readFile","server","handleErrors","fn","req","res","next","x","initializeApp","clientManifest","info","fail","wrappingApp","use","serveAssets","filename","url","substring","WEBPACK_PUBLIC_PATH","length","assetPath","join","outputPath","existsSync","lstatSync","isDirectory","sendFile","e","serveProxy","devConfig","devServer","proxy","middlewares","map","middleware","render","cwd","default","Error","get","endsWith","statusCode","setHeader","send","socket","on","error","listen","syscall","isPipe","portOrPipe","Number","isNaN","bind","code","manifest","with","type","warn","close","manifestFilename","plugins","find","plugin","constructor","name","opts","output"],"sources":["../../../src/scripts/serve.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport compress from 'compression';\nimport express, { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { Server, IncomingMessage, ServerResponse } from 'http';\nimport ora from 'ora';\nimport path from 'path';\nimport { promisify } from 'util';\nimport webpack from 'webpack';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport getProxyMiddlewares from './getProxyMiddlewares.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { Render } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: ${process.argv[0]} <server-entrypoint>`);\n    process.exit(-1);\n  }\n  serve(entrypoint);\n}\n\nexport default async function serve(\n  entrypoint: string,\n  options: { serveAssets?: boolean; serveProxy?: boolean } = {},\n) {\n  const PORT = process.env.PORT || 8080;\n\n  const loader = ora('Initializing').start();\n\n  const webpackConfig = await getWebpackConfig();\n\n  const manifestPath = getManifestPathFromWebpackconfig(\n    webpackConfig({}, { mode: 'production' }),\n  );\n\n  const readFile = promisify(diskFs.readFile);\n  let server: Server | undefined;\n\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  // Start the express server after the first compilation\n  async function initializeApp(clientManifest: webpack.StatsCompilation) {\n    loader.info('Launching server');\n    if (!clientManifest) {\n      loader.fail('Manifest not found');\n      // TODO: handle more gracefully\n      process.exit(-1);\n    }\n\n    const wrappingApp = express();\n    // eslint-disable-next-line\n    //@ts-ignore\n    wrappingApp.use(compress());\n\n    // ASSETS\n    if (options.serveAssets) {\n      wrappingApp.use(\n        async (\n          req: Request | IncomingMessage,\n          res: any,\n          next: NextFunction,\n        ) => {\n          const filename =\n            req.url?.substring(\n              (process.env.WEBPACK_PUBLIC_PATH as string).length,\n            ) ?? '';\n          const assetPath = path.join(\n            clientManifest.outputPath ?? '',\n            filename,\n          );\n\n          if (\n            diskFs.existsSync(assetPath) &&\n            !diskFs.lstatSync(assetPath).isDirectory()\n          ) {\n            try {\n              res.sendFile(assetPath);\n            } catch (e) {\n              return next();\n            }\n          } else {\n            next();\n          }\n        },\n      );\n    }\n\n    // PROXIES\n    if (options.serveProxy) {\n      const devConfig: webpack.Configuration = webpackConfig(\n        {},\n        { mode: 'development' },\n      );\n      if (devConfig.devServer?.proxy) {\n        const middlewares = getProxyMiddlewares(devConfig.devServer?.proxy);\n        if (middlewares) {\n          wrappingApp.use(...middlewares.map(({ middleware }) => middleware));\n        }\n      }\n    }\n\n    // SERVER SIDE RENDERING\n    let render: Render = (await import(path.join(process.cwd(), entrypoint)))\n      .default;\n\n    if ('default' in render) {\n      render = render.default as any;\n    }\n\n    if (typeof render !== 'function') {\n      throw new Error(\n        `default export of ${path.join(\n          process.cwd(),\n          entrypoint,\n        )} is not a function`,\n      );\n    }\n\n    wrappingApp.get(\n      '/*',\n      handleErrors(async function (req: any, res: any) {\n        if (req.url.endsWith('favicon.ico')) {\n          res.statusCode = 404;\n          res.setHeader('Content-type', 'text/html');\n          res.send('not found');\n          return;\n        }\n        res.socket.on('error', (error: unknown) => {\n          console.error('Fatal', error);\n        });\n\n        await render(clientManifest, req, res);\n      }),\n    );\n\n    server = wrappingApp\n      .listen(PORT, () => {\n        loader.info(`Listening at ${PORT}...`);\n      })\n      .on('error', function (error: any) {\n        if (error.syscall !== 'listen') {\n          throw error;\n        }\n        const isPipe = (portOrPipe: string | number) =>\n          Number.isNaN(portOrPipe);\n        const bind = isPipe(PORT) ? 'Pipe ' + PORT : 'Port ' + PORT;\n        switch (error.code) {\n          case 'EACCES':\n            loader.fail(bind + ' requires elevated privileges');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          case 'EADDRINUSE':\n            loader.fail(bind + ' is already in use');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          default:\n            throw error;\n        }\n      });\n  }\n\n  let manifest = await import(manifestPath, { with: { type: 'json' } });\n  // handle inconsistent import conditions\n  if ('default' in manifest) {\n    manifest = manifest.default;\n  }\n  await initializeApp(manifest);\n\n  process.on('SIGINT', () => {\n    loader.warn('Received SIGINT, devserver shutting down');\n    if (server) console.log('Closing server');\n    server?.close(() => {\n      loader.info('Server closed');\n    });\n    process.exit(-1);\n  });\n}\n\nfunction getManifestPathFromWebpackconfig(\n  webpackConfig: webpack.Configuration,\n) {\n  const manifestFilename: string =\n    (\n      webpackConfig?.plugins?.find(plugin => {\n        return plugin?.constructor.name === 'StatsWriterPlugin';\n      }) as any\n    )?.opts?.filename ?? 'manifest.json';\n\n  const manifestPath = path.join(\n    webpackConfig?.output?.path ?? '',\n    manifestFilename,\n  );\n  return manifestPath;\n}\n"],"mappings":"AAAA;AACAA,MAAM,CAACC,MAAM,GACXD,MAAM,CAACC,MAAM,IACb,0BAA2B,SAASA,MAAMA,CAACC,EAAE,EAAEC,GAAG,EAAE;EAClD,OAAOH,MAAM,CAACI,SAAS,CAACC,cAAc,CAACC,IAAI,CAACJ,EAAE,EAAEC,GAAG,CAAC;AACtD,CAAC;AACH,OAAOI,QAAQ,MAAM,aAAa;AAClC,OAAOC,OAAO,MAAwB,SAAS;AAC/C,OAAOC,MAAM,MAAM,IAAI;AAEvB,OAAOC,GAAG,MAAM,KAAK;AACrB,OAAOC,IAAI,MAAM,MAAM;AACvB,SAASC,SAAS,QAAQ,MAAM;AAGhC,OAAO,mCAAmC;AAC1C,OAAOC,mBAAmB,MAAM,0BAA0B;AAC1D,SAASC,gBAAgB,QAAQ,uBAAuB;AAGxD;AACA;AACA;AACA,IAAIC,MAAM,CAACC,IAAI,CAACC,IAAI,EAAE;EACpB,MAAMC,UAAU,GAAGC,OAAO,CAACC,IAAI,CAAC,CAAC,CAAC;EAElC,IAAI,CAACF,UAAU,EAAE;IACfG,OAAO,CAACC,GAAG,CAAC,UAAUH,OAAO,CAACC,IAAI,CAAC,CAAC,CAAC,sBAAsB,CAAC;IAC5DD,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB;EACAC,KAAK,CAACN,UAAU,CAAC;AACnB;AAEA,eAAe,eAAeM,KAAKA,CACjCN,UAAkB,EAClBO,OAAwD,GAAG,CAAC,CAAC,EAC7D;EACA,MAAMC,IAAI,GAAGP,OAAO,CAACQ,GAAG,CAACD,IAAI,IAAI,IAAI;EAErC,MAAME,MAAM,GAAGlB,GAAG,CAAC,cAAc,CAAC,CAACmB,KAAK,CAAC,CAAC;EAE1C,MAAMC,aAAa,GAAG,MAAMhB,gBAAgB,CAAC,CAAC;EAE9C,MAAMiB,YAAY,GAAGC,gCAAgC,CACnDF,aAAa,CAAC,CAAC,CAAC,EAAE;IAAEG,IAAI,EAAE;EAAa,CAAC,CAC1C,CAAC;EAED,MAAMC,QAAQ,GAAGtB,SAAS,CAACH,MAAM,CAACyB,QAAQ,CAAC;EAC3C,IAAIC,MAA0B;EAE9B,SAASC,YAAYA,CAKnBC,EAAK,EAAE;IACP,OAAO,gBACLC,GAA8B,EAC9BC,GAA8B,EAC9BC,IAAkB,EAClB;MACA,IAAI;QACF,OAAO,MAAMH,EAAE,CAACC,GAAG,EAAEC,GAAG,CAAC;MAC3B,CAAC,CAAC,OAAOE,CAAC,EAAE;QACVD,IAAI,CAACC,CAAC,CAAC;MACT;IACF,CAAC;EACH;;EAEA;EACA,eAAeC,aAAaA,CAACC,cAAwC,EAAE;IACrEf,MAAM,CAACgB,IAAI,CAAC,kBAAkB,CAAC;IAC/B,IAAI,CAACD,cAAc,EAAE;MACnBf,MAAM,CAACiB,IAAI,CAAC,oBAAoB,CAAC;MACjC;MACA1B,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB;IAEA,MAAMuB,WAAW,GAAGtC,OAAO,CAAC,CAAC;IAC7B;IACA;IACAsC,WAAW,CAACC,GAAG,CAACxC,QAAQ,CAAC,CAAC,CAAC;;IAE3B;IACA,IAAIkB,OAAO,CAACuB,WAAW,EAAE;MACvBF,WAAW,CAACC,GAAG,CACb,OACET,GAA8B,EAC9BC,GAAQ,EACRC,IAAkB,KACf;QACH,MAAMS,QAAQ,GACZX,GAAG,CAACY,GAAG,EAAEC,SAAS,CACfhC,OAAO,CAACQ,GAAG,CAACyB,mBAAmB,CAAYC,MAC9C,CAAC,IAAI,EAAE;QACT,MAAMC,SAAS,GAAG3C,IAAI,CAAC4C,IAAI,CACzBZ,cAAc,CAACa,UAAU,IAAI,EAAE,EAC/BP,QACF,CAAC;QAED,IACExC,MAAM,CAACgD,UAAU,CAACH,SAAS,CAAC,IAC5B,CAAC7C,MAAM,CAACiD,SAAS,CAACJ,SAAS,CAAC,CAACK,WAAW,CAAC,CAAC,EAC1C;UACA,IAAI;YACFpB,GAAG,CAACqB,QAAQ,CAACN,SAAS,CAAC;UACzB,CAAC,CAAC,OAAOO,CAAC,EAAE;YACV,OAAOrB,IAAI,CAAC,CAAC;UACf;QACF,CAAC,MAAM;UACLA,IAAI,CAAC,CAAC;QACR;MACF,CACF,CAAC;IACH;;IAEA;IACA,IAAIf,OAAO,CAACqC,UAAU,EAAE;MACtB,MAAMC,SAAgC,GAAGjC,aAAa,CACpD,CAAC,CAAC,EACF;QAAEG,IAAI,EAAE;MAAc,CACxB,CAAC;MACD,IAAI8B,SAAS,CAACC,SAAS,EAAEC,KAAK,EAAE;QAC9B,MAAMC,WAAW,GAAGrD,mBAAmB,CAACkD,SAAS,CAACC,SAAS,EAAEC,KAAK,CAAC;QACnE,IAAIC,WAAW,EAAE;UACfpB,WAAW,CAACC,GAAG,CAAC,GAAGmB,WAAW,CAACC,GAAG,CAAC,CAAC;YAAEC;UAAW,CAAC,KAAKA,UAAU,CAAC,CAAC;QACrE;MACF;IACF;;IAEA;IACA,IAAIC,MAAc,GAAG,CAAC,MAAM,MAAM,CAAC1D,IAAI,CAAC4C,IAAI,CAACpC,OAAO,CAACmD,GAAG,CAAC,CAAC,EAAEpD,UAAU,CAAC,CAAC,EACrEqD,OAAO;IAEV,IAAI,SAAS,IAAIF,MAAM,EAAE;MACvBA,MAAM,GAAGA,MAAM,CAACE,OAAc;IAChC;IAEA,IAAI,OAAOF,MAAM,KAAK,UAAU,EAAE;MAChC,MAAM,IAAIG,KAAK,CACb,qBAAqB7D,IAAI,CAAC4C,IAAI,CAC5BpC,OAAO,CAACmD,GAAG,CAAC,CAAC,EACbpD,UACF,CAAC,oBACH,CAAC;IACH;IAEA4B,WAAW,CAAC2B,GAAG,CACb,IAAI,EACJrC,YAAY,CAAC,gBAAgBE,GAAQ,EAAEC,GAAQ,EAAE;MAC/C,IAAID,GAAG,CAACY,GAAG,CAACwB,QAAQ,CAAC,aAAa,CAAC,EAAE;QACnCnC,GAAG,CAACoC,UAAU,GAAG,GAAG;QACpBpC,GAAG,CAACqC,SAAS,CAAC,cAAc,EAAE,WAAW,CAAC;QAC1CrC,GAAG,CAACsC,IAAI,CAAC,WAAW,CAAC;QACrB;MACF;MACAtC,GAAG,CAACuC,MAAM,CAACC,EAAE,CAAC,OAAO,EAAGC,KAAc,IAAK;QACzC3D,OAAO,CAAC2D,KAAK,CAAC,OAAO,EAAEA,KAAK,CAAC;MAC/B,CAAC,CAAC;MAEF,MAAMX,MAAM,CAAC1B,cAAc,EAAEL,GAAG,EAAEC,GAAG,CAAC;IACxC,CAAC,CACH,CAAC;IAEDJ,MAAM,GAAGW,WAAW,CACjBmC,MAAM,CAACvD,IAAI,EAAE,MAAM;MAClBE,MAAM,CAACgB,IAAI,CAAC,gBAAgBlB,IAAI,KAAK,CAAC;IACxC,CAAC,CAAC,CACDqD,EAAE,CAAC,OAAO,EAAE,UAAUC,KAAU,EAAE;MACjC,IAAIA,KAAK,CAACE,OAAO,KAAK,QAAQ,EAAE;QAC9B,MAAMF,KAAK;MACb;MACA,MAAMG,MAAM,GAAIC,UAA2B,IACzCC,MAAM,CAACC,KAAK,CAACF,UAAU,CAAC;MAC1B,MAAMG,IAAI,GAAGJ,MAAM,CAACzD,IAAI,CAAC,GAAG,OAAO,GAAGA,IAAI,GAAG,OAAO,GAAGA,IAAI;MAC3D,QAAQsD,KAAK,CAACQ,IAAI;QAChB,KAAK,QAAQ;UACX5D,MAAM,CAACiB,IAAI,CAAC0C,IAAI,GAAG,+BAA+B,CAAC;UACnDpE,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC;QACjB;QACA,KAAK,YAAY;UACfK,MAAM,CAACiB,IAAI,CAAC0C,IAAI,GAAG,oBAAoB,CAAC;UACxCpE,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC;QACjB;QACA;UACE,MAAMyD,KAAK;MACf;IACF,CAAC,CAAC;EACN;EAEA,IAAIS,QAAQ,GAAG,MAAM,MAAM,CAAC1D,YAAY,EAAE;IAAE2D,IAAI,EAAE;MAAEC,IAAI,EAAE;IAAO;EAAE,CAAC,CAAC;EACrE;EACA,IAAI,SAAS,IAAIF,QAAQ,EAAE;IACzBA,QAAQ,GAAGA,QAAQ,CAAClB,OAAO;EAC7B;EACA,MAAM7B,aAAa,CAAC+C,QAAQ,CAAC;EAE7BtE,OAAO,CAAC4D,EAAE,CAAC,QAAQ,EAAE,MAAM;IACzBnD,MAAM,CAACgE,IAAI,CAAC,0CAA0C,CAAC;IACvD,IAAIzD,MAAM,EAAEd,OAAO,CAACC,GAAG,CAAC,gBAAgB,CAAC;IACzCa,MAAM,EAAE0D,KAAK,CAAC,MAAM;MAClBjE,MAAM,CAACgB,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC,CAAC;IACFzB,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB,CAAC,CAAC;AACJ;AAEA,SAASS,gCAAgCA,CACvCF,aAAoC,EACpC;EACA,MAAMgE,gBAAwB,GAE1BhE,aAAa,EAAEiE,OAAO,EAAEC,IAAI,CAACC,MAAM,IAAI;IACrC,OAAOA,MAAM,EAAEC,WAAW,CAACC,IAAI,KAAK,mBAAmB;EACzD,CAAC,CAAC,EACDC,IAAI,EAAEnD,QAAQ,IAAI,eAAe;EAEtC,MAAMlB,YAAY,GAAGpB,IAAI,CAAC4C,IAAI,CAC5BzB,aAAa,EAAEuE,MAAM,EAAE1F,IAAI,IAAI,EAAE,EACjCmF,gBACF,CAAC;EACD,OAAO/D,YAAY;AACrB","ignoreList":[]}
|
|
167
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","compress","express","diskFs","ora","path","getProxyMiddlewares","getWebpackConfig","getErrorStatus","renderErrorPage","import","meta","main","entrypoint","process","argv","console","log","exit","serve","options","PORT","env","loader","start","webpackConfig","manifestPath","getManifestPathFromWebpackconfig","mode","server","handleErrors","fn","req","res","_next","error","expressRes","headersSent","statusCode","status","setHeader","send","url","showStack","NODE_ENV","initializeApp","clientManifest","info","fail","wrappingApp","use","serveAssets","next","filename","substring","WEBPACK_PUBLIC_PATH","length","assetPath","join","outputPath","existsSync","lstatSync","isDirectory","sendFile","_e","serveProxy","devConfig","devServer","proxy","middlewares","map","middleware","render","cwd","default","Error","get","endsWith","socket","on","listen","syscall","isPipe","portOrPipe","Number","isNaN","bind","code","manifest","with","type","warn","close","manifestFilename","plugins","find","plugin","constructor","name","opts","output"],"sources":["../../../src/scripts/serve.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport compress from 'compression';\nimport express, { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { Server, IncomingMessage, ServerResponse } from 'http';\nimport ora from 'ora';\nimport path from 'path';\nimport webpack from 'webpack';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport getProxyMiddlewares from './getProxyMiddlewares.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';\nimport { Render } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: ${process.argv[0]} <server-entrypoint>`);\n    process.exit(-1);\n  }\n  serve(entrypoint);\n}\n\nexport default async function serve(\n  entrypoint: string,\n  options: { serveAssets?: boolean; serveProxy?: boolean } = {},\n) {\n  const PORT = process.env.PORT || 8080;\n\n  const loader = ora('Initializing').start();\n\n  const webpackConfig = await getWebpackConfig();\n\n  const manifestPath = getManifestPathFromWebpackconfig(\n    webpackConfig({}, { mode: 'production' }),\n  );\n\n  let server: Server | undefined;\n\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      _next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (error: unknown) {\n        console.error('SSR rendering error:', error);\n\n        // Return error response with status from error if available\n        const expressRes = res as express.Response;\n        if (!expressRes.headersSent) {\n          const statusCode = getErrorStatus(error);\n          expressRes.status(statusCode);\n          expressRes.setHeader('Content-Type', 'text/html');\n          expressRes.send(\n            renderErrorPage(error, req.url ?? '/', statusCode, {\n              showStack: process.env.NODE_ENV !== 'production',\n            }),\n          );\n        }\n      }\n    };\n  }\n\n  // Start the express server after the first compilation\n  async function initializeApp(clientManifest: webpack.StatsCompilation) {\n    loader.info('Launching server');\n    if (!clientManifest) {\n      loader.fail('Manifest not found');\n      // TODO: handle more gracefully\n      process.exit(-1);\n    }\n\n    const wrappingApp = express();\n    // eslint-disable-next-line\n    //@ts-ignore\n    wrappingApp.use(compress());\n\n    // ASSETS\n    if (options.serveAssets) {\n      wrappingApp.use(\n        async (\n          req: Request | IncomingMessage,\n          res: any,\n          next: NextFunction,\n        ) => {\n          const filename =\n            req.url?.substring(\n              (process.env.WEBPACK_PUBLIC_PATH as string).length,\n            ) ?? '';\n          const assetPath = path.join(\n            clientManifest.outputPath ?? '',\n            filename,\n          );\n\n          if (\n            diskFs.existsSync(assetPath) &&\n            !diskFs.lstatSync(assetPath).isDirectory()\n          ) {\n            try {\n              res.sendFile(assetPath);\n            } catch (_e) {\n              return next();\n            }\n          } else {\n            next();\n          }\n        },\n      );\n    }\n\n    // PROXIES\n    if (options.serveProxy) {\n      const devConfig: webpack.Configuration = webpackConfig(\n        {},\n        { mode: 'development' },\n      );\n      if (devConfig.devServer?.proxy) {\n        const middlewares = getProxyMiddlewares(devConfig.devServer?.proxy);\n        if (middlewares) {\n          wrappingApp.use(...middlewares.map(({ middleware }) => middleware));\n        }\n      }\n    }\n\n    // SERVER SIDE RENDERING\n    let render: Render = (await import(path.join(process.cwd(), entrypoint)))\n      .default;\n\n    if ('default' in render) {\n      render = render.default as any;\n    }\n\n    if (typeof render !== 'function') {\n      throw new Error(\n        `default export of ${path.join(\n          process.cwd(),\n          entrypoint,\n        )} is not a function`,\n      );\n    }\n\n    wrappingApp.get(\n      '/*',\n      handleErrors(async function (req: any, res: any) {\n        if (req.url.endsWith('favicon.ico')) {\n          res.statusCode = 404;\n          res.setHeader('Content-type', 'text/html');\n          res.send('not found');\n          return;\n        }\n        res.socket.on('error', (error: unknown) => {\n          console.error('Fatal', error);\n        });\n\n        await render(clientManifest, req, res);\n      }),\n    );\n\n    server = wrappingApp\n      .listen(PORT, () => {\n        loader.info(`Listening at ${PORT}...`);\n      })\n      .on('error', function (error: any) {\n        if (error.syscall !== 'listen') {\n          throw error;\n        }\n        const isPipe = (portOrPipe: string | number) =>\n          Number.isNaN(portOrPipe);\n        const bind = isPipe(PORT) ? 'Pipe ' + PORT : 'Port ' + PORT;\n        switch (error.code) {\n          case 'EACCES':\n            loader.fail(bind + ' requires elevated privileges');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          case 'EADDRINUSE':\n            loader.fail(bind + ' is already in use');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          default:\n            throw error;\n        }\n      });\n  }\n\n  let manifest = await import(manifestPath, { with: { type: 'json' } });\n  // handle inconsistent import conditions\n  if ('default' in manifest) {\n    manifest = manifest.default;\n  }\n  await initializeApp(manifest);\n\n  process.on('SIGINT', () => {\n    loader.warn('Received SIGINT, devserver shutting down');\n    if (server) console.log('Closing server');\n    server?.close(() => {\n      loader.info('Server closed');\n    });\n    process.exit(-1);\n  });\n}\n\nfunction getManifestPathFromWebpackconfig(\n  webpackConfig: webpack.Configuration,\n) {\n  const manifestFilename: string =\n    (\n      webpackConfig?.plugins?.find(plugin => {\n        return plugin?.constructor.name === 'StatsWriterPlugin';\n      }) as any\n    )?.opts?.filename ?? 'manifest.json';\n\n  const manifestPath = path.join(\n    webpackConfig?.output?.path ?? '',\n    manifestFilename,\n  );\n  return manifestPath;\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Utilities for handling SSR errors gracefully
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Extract HTTP status code from an error object.
|
|
7
|
+
* Looks for a `status` property that is a number or parseable string.
|
|
8
|
+
* Returns 500 if no valid status found.
|
|
9
|
+
*/
|
|
10
|
+
export function getErrorStatus(error) {
|
|
11
|
+
if (error && typeof error === 'object' && 'status' in error) {
|
|
12
|
+
const status = error.status;
|
|
13
|
+
if (typeof status === 'number' && status >= 100 && status < 600) {
|
|
14
|
+
return status;
|
|
15
|
+
}
|
|
16
|
+
if (typeof status === 'string') {
|
|
17
|
+
const parsed = parseInt(status, 10);
|
|
18
|
+
if (!isNaN(parsed) && parsed >= 100 && parsed < 600) {
|
|
19
|
+
return parsed;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return 500;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Escape HTML special characters to prevent XSS
|
|
28
|
+
*/
|
|
29
|
+
export function escapeHtml(str) {
|
|
30
|
+
return str.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/"/g, '"').replace(/'/g, ''');
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Render an HTML error page for SSR failures
|
|
34
|
+
*/
|
|
35
|
+
export function renderErrorPage(error, url, statusCode, options = {}) {
|
|
36
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
37
|
+
const stack = error instanceof Error ? error.stack : undefined;
|
|
38
|
+
const {
|
|
39
|
+
showStack = false,
|
|
40
|
+
hint,
|
|
41
|
+
badge
|
|
42
|
+
} = options;
|
|
43
|
+
return `<!DOCTYPE html>
|
|
44
|
+
<html lang="en">
|
|
45
|
+
<head>
|
|
46
|
+
<meta charset="UTF-8">
|
|
47
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
48
|
+
<title>${statusCode} - Server Error</title>
|
|
49
|
+
<style>
|
|
50
|
+
* { box-sizing: border-box; margin: 0; padding: 0; }
|
|
51
|
+
body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; background: #1a1a2e; color: #eee; min-height: 100vh; display: flex; align-items: center; justify-content: center; padding: 2rem; }
|
|
52
|
+
.container { max-width: 800px; width: 100%; }
|
|
53
|
+
h1 { color: #ff6b6b; font-size: 2.5rem; margin-bottom: 1rem; }
|
|
54
|
+
.badge { display: inline-block; background: #4ecdc4; color: #1a1a2e; padding: 0.25rem 0.5rem; border-radius: 4px; font-size: 0.75rem; font-weight: 600; margin-bottom: 1rem; }
|
|
55
|
+
.url { color: #888; font-size: 0.9rem; margin-bottom: 1.5rem; word-break: break-all; }
|
|
56
|
+
.message { background: #16213e; border-left: 4px solid #ff6b6b; padding: 1rem 1.5rem; border-radius: 0 8px 8px 0; margin-bottom: 1.5rem; }
|
|
57
|
+
.message code { color: #ff6b6b; font-size: 1.1rem; }
|
|
58
|
+
.stack { background: #0f0f23; border-radius: 8px; padding: 1.5rem; overflow-x: auto; font-family: 'Monaco', 'Menlo', monospace; font-size: 0.85rem; line-height: 1.6; color: #aaa; white-space: pre-wrap; word-break: break-word; }
|
|
59
|
+
.retry { margin-top: 2rem; }
|
|
60
|
+
.retry a { color: #4ecdc4; text-decoration: none; padding: 0.75rem 1.5rem; border: 2px solid #4ecdc4; border-radius: 6px; display: inline-block; transition: all 0.2s; }
|
|
61
|
+
.retry a:hover { background: #4ecdc4; color: #1a1a2e; }
|
|
62
|
+
.hint { margin-top: 1.5rem; color: #888; font-size: 0.9rem; }
|
|
63
|
+
</style>
|
|
64
|
+
</head>
|
|
65
|
+
<body>
|
|
66
|
+
<div class="container">
|
|
67
|
+
${badge ? `<span class="badge">${escapeHtml(badge)}</span>` : ''}
|
|
68
|
+
<h1>${statusCode} - Server Error</h1>
|
|
69
|
+
<p class="url">Error rendering: ${escapeHtml(url)}</p>
|
|
70
|
+
<div class="message">
|
|
71
|
+
<code>${escapeHtml(errorMessage)}</code>
|
|
72
|
+
</div>
|
|
73
|
+
${showStack && stack ? `<pre class="stack">${escapeHtml(stack)}</pre>` : ''}
|
|
74
|
+
<div class="retry">
|
|
75
|
+
<a href="${escapeHtml(url)}">Retry</a>
|
|
76
|
+
</div>
|
|
77
|
+
${hint ? `<p class="hint">${escapeHtml(hint)}</p>` : ''}
|
|
78
|
+
</div>
|
|
79
|
+
</body>
|
|
80
|
+
</html>`;
|
|
81
|
+
}
|
|
82
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJnZXRFcnJvclN0YXR1cyIsImVycm9yIiwic3RhdHVzIiwicGFyc2VkIiwicGFyc2VJbnQiLCJpc05hTiIsImVzY2FwZUh0bWwiLCJzdHIiLCJyZXBsYWNlIiwicmVuZGVyRXJyb3JQYWdlIiwidXJsIiwic3RhdHVzQ29kZSIsIm9wdGlvbnMiLCJlcnJvck1lc3NhZ2UiLCJFcnJvciIsIm1lc3NhZ2UiLCJTdHJpbmciLCJzdGFjayIsInVuZGVmaW5lZCIsInNob3dTdGFjayIsImhpbnQiLCJiYWRnZSJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zY3JpcHRzL3NzckVycm9ySGFuZGxlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFV0aWxpdGllcyBmb3IgaGFuZGxpbmcgU1NSIGVycm9ycyBncmFjZWZ1bGx5XG4gKi9cblxuLyoqXG4gKiBFeHRyYWN0IEhUVFAgc3RhdHVzIGNvZGUgZnJvbSBhbiBlcnJvciBvYmplY3QuXG4gKiBMb29rcyBmb3IgYSBgc3RhdHVzYCBwcm9wZXJ0eSB0aGF0IGlzIGEgbnVtYmVyIG9yIHBhcnNlYWJsZSBzdHJpbmcuXG4gKiBSZXR1cm5zIDUwMCBpZiBubyB2YWxpZCBzdGF0dXMgZm91bmQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRFcnJvclN0YXR1cyhlcnJvcjogdW5rbm93bik6IG51bWJlciB7XG4gIGlmIChlcnJvciAmJiB0eXBlb2YgZXJyb3IgPT09ICdvYmplY3QnICYmICdzdGF0dXMnIGluIGVycm9yKSB7XG4gICAgY29uc3Qgc3RhdHVzID0gKGVycm9yIGFzIHsgc3RhdHVzOiB1bmtub3duIH0pLnN0YXR1cztcbiAgICBpZiAodHlwZW9mIHN0YXR1cyA9PT0gJ251bWJlcicgJiYgc3RhdHVzID49IDEwMCAmJiBzdGF0dXMgPCA2MDApIHtcbiAgICAgIHJldHVybiBzdGF0dXM7XG4gICAgfVxuICAgIGlmICh0eXBlb2Ygc3RhdHVzID09PSAnc3RyaW5nJykge1xuICAgICAgY29uc3QgcGFyc2VkID0gcGFyc2VJbnQoc3RhdHVzLCAxMCk7XG4gICAgICBpZiAoIWlzTmFOKHBhcnNlZCkgJiYgcGFyc2VkID49IDEwMCAmJiBwYXJzZWQgPCA2MDApIHtcbiAgICAgICAgcmV0dXJuIHBhcnNlZDtcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcmV0dXJuIDUwMDtcbn1cblxuLyoqXG4gKiBFc2NhcGUgSFRNTCBzcGVjaWFsIGNoYXJhY3RlcnMgdG8gcHJldmVudCBYU1NcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGVzY2FwZUh0bWwoc3RyOiBzdHJpbmcpOiBzdHJpbmcge1xuICByZXR1cm4gc3RyXG4gICAgLnJlcGxhY2UoLyYvZywgJyZhbXA7JylcbiAgICAucmVwbGFjZSgvPC9nLCAnJmx0OycpXG4gICAgLnJlcGxhY2UoLz4vZywgJyZndDsnKVxuICAgIC5yZXBsYWNlKC9cIi9nLCAnJnF1b3Q7JylcbiAgICAucmVwbGFjZSgvJy9nLCAnJiMwMzk7Jyk7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVuZGVyRXJyb3JQYWdlT3B0aW9ucyB7XG4gIC8qKiBTaG93IHN0YWNrIHRyYWNlIGluIG91dHB1dCAqL1xuICBzaG93U3RhY2s/OiBib29sZWFuO1xuICAvKiogQWRkaXRpb25hbCBoaW50IG1lc3NhZ2UgdG8gZGlzcGxheSAqL1xuICBoaW50Pzogc3RyaW5nO1xuICAvKiogQmFkZ2UgdGV4dCB0byBkaXNwbGF5IChlLmcuLCBcIkRFViBNT0RFXCIpICovXG4gIGJhZGdlPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlbmRlciBhbiBIVE1MIGVycm9yIHBhZ2UgZm9yIFNTUiBmYWlsdXJlc1xuICovXG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyRXJyb3JQYWdlKFxuICBlcnJvcjogdW5rbm93bixcbiAgdXJsOiBzdHJpbmcsXG4gIHN0YXR1c0NvZGU6IG51bWJlcixcbiAgb3B0aW9uczogUmVuZGVyRXJyb3JQYWdlT3B0aW9ucyA9IHt9LFxuKTogc3RyaW5nIHtcbiAgY29uc3QgZXJyb3JNZXNzYWdlID0gZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpO1xuICBjb25zdCBzdGFjayA9IGVycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5zdGFjayA6IHVuZGVmaW5lZDtcbiAgY29uc3QgeyBzaG93U3RhY2sgPSBmYWxzZSwgaGludCwgYmFkZ2UgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIGA8IURPQ1RZUEUgaHRtbD5cbjxodG1sIGxhbmc9XCJlblwiPlxuPGhlYWQ+XG4gIDxtZXRhIGNoYXJzZXQ9XCJVVEYtOFwiPlxuICA8bWV0YSBuYW1lPVwidmlld3BvcnRcIiBjb250ZW50PVwid2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMFwiPlxuICA8dGl0bGU+JHtzdGF0dXNDb2RlfSAtIFNlcnZlciBFcnJvcjwvdGl0bGU+XG4gIDxzdHlsZT5cbiAgICAqIHsgYm94LXNpemluZzogYm9yZGVyLWJveDsgbWFyZ2luOiAwOyBwYWRkaW5nOiAwOyB9XG4gICAgYm9keSB7IGZvbnQtZmFtaWx5OiAtYXBwbGUtc3lzdGVtLCBCbGlua01hY1N5c3RlbUZvbnQsICdTZWdvZSBVSScsIFJvYm90bywgc2Fucy1zZXJpZjsgYmFja2dyb3VuZDogIzFhMWEyZTsgY29sb3I6ICNlZWU7IG1pbi1oZWlnaHQ6IDEwMHZoOyBkaXNwbGF5OiBmbGV4OyBhbGlnbi1pdGVtczogY2VudGVyOyBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjsgcGFkZGluZzogMnJlbTsgfVxuICAgIC5jb250YWluZXIgeyBtYXgtd2lkdGg6IDgwMHB4OyB3aWR0aDogMTAwJTsgfVxuICAgIGgxIHsgY29sb3I6ICNmZjZiNmI7IGZvbnQtc2l6ZTogMi41cmVtOyBtYXJnaW4tYm90dG9tOiAxcmVtOyB9XG4gICAgLmJhZGdlIHsgZGlzcGxheTogaW5saW5lLWJsb2NrOyBiYWNrZ3JvdW5kOiAjNGVjZGM0OyBjb2xvcjogIzFhMWEyZTsgcGFkZGluZzogMC4yNXJlbSAwLjVyZW07IGJvcmRlci1yYWRpdXM6IDRweDsgZm9udC1zaXplOiAwLjc1cmVtOyBmb250LXdlaWdodDogNjAwOyBtYXJnaW4tYm90dG9tOiAxcmVtOyB9XG4gICAgLnVybCB7IGNvbG9yOiAjODg4OyBmb250LXNpemU6IDAuOXJlbTsgbWFyZ2luLWJvdHRvbTogMS41cmVtOyB3b3JkLWJyZWFrOiBicmVhay1hbGw7IH1cbiAgICAubWVzc2FnZSB7IGJhY2tncm91bmQ6ICMxNjIxM2U7IGJvcmRlci1sZWZ0OiA0cHggc29saWQgI2ZmNmI2YjsgcGFkZGluZzogMXJlbSAxLjVyZW07IGJvcmRlci1yYWRpdXM6IDAgOHB4IDhweCAwOyBtYXJnaW4tYm90dG9tOiAxLjVyZW07IH1cbiAgICAubWVzc2FnZSBjb2RlIHsgY29sb3I6ICNmZjZiNmI7IGZvbnQtc2l6ZTogMS4xcmVtOyB9XG4gICAgLnN0YWNrIHsgYmFja2dyb3VuZDogIzBmMGYyMzsgYm9yZGVyLXJhZGl1czogOHB4OyBwYWRkaW5nOiAxLjVyZW07IG92ZXJmbG93LXg6IGF1dG87IGZvbnQtZmFtaWx5OiAnTW9uYWNvJywgJ01lbmxvJywgbW9ub3NwYWNlOyBmb250LXNpemU6IDAuODVyZW07IGxpbmUtaGVpZ2h0OiAxLjY7IGNvbG9yOiAjYWFhOyB3aGl0ZS1zcGFjZTogcHJlLXdyYXA7IHdvcmQtYnJlYWs6IGJyZWFrLXdvcmQ7IH1cbiAgICAucmV0cnkgeyBtYXJnaW4tdG9wOiAycmVtOyB9XG4gICAgLnJldHJ5IGEgeyBjb2xvcjogIzRlY2RjNDsgdGV4dC1kZWNvcmF0aW9uOiBub25lOyBwYWRkaW5nOiAwLjc1cmVtIDEuNXJlbTsgYm9yZGVyOiAycHggc29saWQgIzRlY2RjNDsgYm9yZGVyLXJhZGl1czogNnB4OyBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7IHRyYW5zaXRpb246IGFsbCAwLjJzOyB9XG4gICAgLnJldHJ5IGE6aG92ZXIgeyBiYWNrZ3JvdW5kOiAjNGVjZGM0OyBjb2xvcjogIzFhMWEyZTsgfVxuICAgIC5oaW50IHsgbWFyZ2luLXRvcDogMS41cmVtOyBjb2xvcjogIzg4ODsgZm9udC1zaXplOiAwLjlyZW07IH1cbiAgPC9zdHlsZT5cbjwvaGVhZD5cbjxib2R5PlxuICA8ZGl2IGNsYXNzPVwiY29udGFpbmVyXCI+XG4gICAgJHtiYWRnZSA/IGA8c3BhbiBjbGFzcz1cImJhZGdlXCI+JHtlc2NhcGVIdG1sKGJhZGdlKX08L3NwYW4+YCA6ICcnfVxuICAgIDxoMT4ke3N0YXR1c0NvZGV9IC0gU2VydmVyIEVycm9yPC9oMT5cbiAgICA8cCBjbGFzcz1cInVybFwiPkVycm9yIHJlbmRlcmluZzogJHtlc2NhcGVIdG1sKHVybCl9PC9wPlxuICAgIDxkaXYgY2xhc3M9XCJtZXNzYWdlXCI+XG4gICAgICA8Y29kZT4ke2VzY2FwZUh0bWwoZXJyb3JNZXNzYWdlKX08L2NvZGU+XG4gICAgPC9kaXY+XG4gICAgJHtzaG93U3RhY2sgJiYgc3RhY2sgPyBgPHByZSBjbGFzcz1cInN0YWNrXCI+JHtlc2NhcGVIdG1sKHN0YWNrKX08L3ByZT5gIDogJyd9XG4gICAgPGRpdiBjbGFzcz1cInJldHJ5XCI+XG4gICAgICA8YSBocmVmPVwiJHtlc2NhcGVIdG1sKHVybCl9XCI+UmV0cnk8L2E+XG4gICAgPC9kaXY+XG4gICAgJHtoaW50ID8gYDxwIGNsYXNzPVwiaGludFwiPiR7ZXNjYXBlSHRtbChoaW50KX08L3A+YCA6ICcnfVxuICA8L2Rpdj5cbjwvYm9keT5cbjwvaHRtbD5gO1xufVxuIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLE9BQU8sU0FBU0EsY0FBY0EsQ0FBQ0MsS0FBYyxFQUFVO0VBQ3JELElBQUlBLEtBQUssSUFBSSxPQUFPQSxLQUFLLEtBQUssUUFBUSxJQUFJLFFBQVEsSUFBSUEsS0FBSyxFQUFFO0lBQzNELE1BQU1DLE1BQU0sR0FBSUQsS0FBSyxDQUF5QkMsTUFBTTtJQUNwRCxJQUFJLE9BQU9BLE1BQU0sS0FBSyxRQUFRLElBQUlBLE1BQU0sSUFBSSxHQUFHLElBQUlBLE1BQU0sR0FBRyxHQUFHLEVBQUU7TUFDL0QsT0FBT0EsTUFBTTtJQUNmO0lBQ0EsSUFBSSxPQUFPQSxNQUFNLEtBQUssUUFBUSxFQUFFO01BQzlCLE1BQU1DLE1BQU0sR0FBR0MsUUFBUSxDQUFDRixNQUFNLEVBQUUsRUFBRSxDQUFDO01BQ25DLElBQUksQ0FBQ0csS0FBSyxDQUFDRixNQUFNLENBQUMsSUFBSUEsTUFBTSxJQUFJLEdBQUcsSUFBSUEsTUFBTSxHQUFHLEdBQUcsRUFBRTtRQUNuRCxPQUFPQSxNQUFNO01BQ2Y7SUFDRjtFQUNGO0VBQ0EsT0FBTyxHQUFHO0FBQ1o7O0FBRUE7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFTRyxVQUFVQSxDQUFDQyxHQUFXLEVBQVU7RUFDOUMsT0FBT0EsR0FBRyxDQUNQQyxPQUFPLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUN0QkEsT0FBTyxDQUFDLElBQUksRUFBRSxNQUFNLENBQUMsQ0FDckJBLE9BQU8sQ0FBQyxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQ3JCQSxPQUFPLENBQUMsSUFBSSxFQUFFLFFBQVEsQ0FBQyxDQUN2QkEsT0FBTyxDQUFDLElBQUksRUFBRSxRQUFRLENBQUM7QUFDNUI7QUFXQTtBQUNBO0FBQ0E7QUFDQSxPQUFPLFNBQVNDLGVBQWVBLENBQzdCUixLQUFjLEVBQ2RTLEdBQVcsRUFDWEMsVUFBa0IsRUFDbEJDLE9BQStCLEdBQUcsQ0FBQyxDQUFDLEVBQzVCO0VBQ1IsTUFBTUMsWUFBWSxHQUFHWixLQUFLLFlBQVlhLEtBQUssR0FBR2IsS0FBSyxDQUFDYyxPQUFPLEdBQUdDLE1BQU0sQ0FBQ2YsS0FBSyxDQUFDO0VBQzNFLE1BQU1nQixLQUFLLEdBQUdoQixLQUFLLFlBQVlhLEtBQUssR0FBR2IsS0FBSyxDQUFDZ0IsS0FBSyxHQUFHQyxTQUFTO0VBQzlELE1BQU07SUFBRUMsU0FBUyxHQUFHLEtBQUs7SUFBRUMsSUFBSTtJQUFFQztFQUFNLENBQUMsR0FBR1QsT0FBTztFQUVsRCxPQUFPO0FBQ1Q7QUFDQTtBQUNBO0FBQ0E7QUFDQSxXQUFXRCxVQUFVO0FBQ3JCO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLE1BQU1VLEtBQUssR0FBRyx1QkFBdUJmLFVBQVUsQ0FBQ2UsS0FBSyxDQUFDLFNBQVMsR0FBRyxFQUFFO0FBQ3BFLFVBQVVWLFVBQVU7QUFDcEIsc0NBQXNDTCxVQUFVLENBQUNJLEdBQUcsQ0FBQztBQUNyRDtBQUNBLGNBQWNKLFVBQVUsQ0FBQ08sWUFBWSxDQUFDO0FBQ3RDO0FBQ0EsTUFBTU0sU0FBUyxJQUFJRixLQUFLLEdBQUcsc0JBQXNCWCxVQUFVLENBQUNXLEtBQUssQ0FBQyxRQUFRLEdBQUcsRUFBRTtBQUMvRTtBQUNBLGlCQUFpQlgsVUFBVSxDQUFDSSxHQUFHLENBQUM7QUFDaEM7QUFDQSxNQUFNVSxJQUFJLEdBQUcsbUJBQW1CZCxVQUFVLENBQUNjLElBQUksQ0FBQyxNQUFNLEdBQUcsRUFBRTtBQUMzRDtBQUNBO0FBQ0EsUUFBUTtBQUNSIiwiaWdub3JlTGlzdCI6W119
|
|
@@ -15,6 +15,7 @@ import WebpackDevServer from 'webpack-dev-server';
|
|
|
15
15
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
16
16
|
import { createHybridRequire } from './createHybridRequire.js';
|
|
17
17
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
18
|
+
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
18
19
|
// run directly from node
|
|
19
20
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
20
21
|
// @ts-ignore
|
|
@@ -88,11 +89,24 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
88
89
|
return path.join(serverJson.outputPath ?? '', 'server.js');
|
|
89
90
|
}
|
|
90
91
|
function handleErrors(fn) {
|
|
91
|
-
return async function (req, res,
|
|
92
|
+
return async function (req, res, _next) {
|
|
92
93
|
try {
|
|
93
94
|
return await fn(req, res);
|
|
94
|
-
} catch (
|
|
95
|
-
|
|
95
|
+
} catch (error) {
|
|
96
|
+
log.error('SSR rendering error:', error);
|
|
97
|
+
|
|
98
|
+
// Return error response with status from error if available
|
|
99
|
+
const expressRes = res;
|
|
100
|
+
if (!expressRes.headersSent) {
|
|
101
|
+
const statusCode = getErrorStatus(error);
|
|
102
|
+
expressRes.status(statusCode);
|
|
103
|
+
expressRes.setHeader('Content-Type', 'text/html');
|
|
104
|
+
expressRes.send(renderErrorPage(error, req.url ?? '/', statusCode, {
|
|
105
|
+
showStack: true,
|
|
106
|
+
badge: 'DEV MODE',
|
|
107
|
+
hint: 'The dev server is still running. Fix the error and retry, or check the console for more details.'
|
|
108
|
+
}));
|
|
109
|
+
}
|
|
96
110
|
}
|
|
97
111
|
};
|
|
98
112
|
}
|
|
@@ -229,4 +243,4 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
229
243
|
});
|
|
230
244
|
runServer();
|
|
231
245
|
}
|
|
232
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","createHybridRequire","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","next","x","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","url","endsWith","statusCode","setHeader","send","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport { createFsFromVolume, Volume } from 'memfs';\nimport path from 'path';\nimport sourceMapSupport from 'source-map-support';\nimport tmp from 'tmp';\nimport { ufs } from 'unionfs';\nimport { promisify } from 'util';\nimport webpack, { type Configuration, type MultiConfiguration } from 'webpack';\nimport logging from 'webpack/lib/logging/runtime.js';\nimport WebpackDevServer from 'webpack-dev-server';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport { createHybridRequire } from './createHybridRequire.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createHybridRequire(ufs);\n\n  const readFile = promisify(ufs.readFile);\n  // Generate a temporary file so we can hot reload from the root of the application\n  function hotEntry(entryPath: string) {\n    // eslint-disable-next-line\n    // @ts-ignore for some reason it's not picking up that other options are optional\n    const generatedEntrypoint = tmp.fileSync({ postfix: '.js' });\n    diskFs.writeSync(\n      generatedEntrypoint.fd,\n      `\n  import entry from \"${path.resolve(process.cwd(), entryPath)}\";\n\n  if (import.meta.webpackHot) {\n    import.meta.webpackHot.accept();\n  }\n\n  export default entry;\n    `,\n    );\n    return generatedEntrypoint;\n  }\n\n  const webpackConfigs: Configuration[] = [\n    webpackConfig(\n      {\n        ...env,\n        entrypath: hotEntry(entrypoint).name,\n        name: 'client',\n      },\n      { mode: 'development' },\n    ),\n    webpackConfig(\n      {\n        ...env,\n        entrypath: entrypoint.replace('.tsx', '.server.tsx'),\n        name: 'server',\n        BROWSERSLIST_ENV: 'current node',\n      },\n      { mode: 'development', target: 'node' },\n    ),\n  ];\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs as unknown as MultiConfiguration);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"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","ignoreList":[]}
|
|
246
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","createHybridRequire","getWebpackConfig","getErrorStatus","renderErrorPage","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","_next","expressRes","headersSent","statusCode","status","setHeader","send","url","showStack","badge","hint","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","endsWith","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport { createFsFromVolume, Volume } from 'memfs';\nimport path from 'path';\nimport sourceMapSupport from 'source-map-support';\nimport tmp from 'tmp';\nimport { ufs } from 'unionfs';\nimport { promisify } from 'util';\nimport webpack, { type Configuration, type MultiConfiguration } from 'webpack';\nimport logging from 'webpack/lib/logging/runtime.js';\nimport WebpackDevServer from 'webpack-dev-server';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport { createHybridRequire } from './createHybridRequire.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createHybridRequire(ufs);\n\n  const readFile = promisify(ufs.readFile);\n  // Generate a temporary file so we can hot reload from the root of the application\n  function hotEntry(entryPath: string) {\n    // eslint-disable-next-line\n    // @ts-ignore for some reason it's not picking up that other options are optional\n    const generatedEntrypoint = tmp.fileSync({ postfix: '.js' });\n    diskFs.writeSync(\n      generatedEntrypoint.fd,\n      `\n  import entry from \"${path.resolve(process.cwd(), entryPath)}\";\n\n  if (import.meta.webpackHot) {\n    import.meta.webpackHot.accept();\n  }\n\n  export default entry;\n    `,\n    );\n    return generatedEntrypoint;\n  }\n\n  const webpackConfigs: Configuration[] = [\n    webpackConfig(\n      {\n        ...env,\n        entrypath: hotEntry(entrypoint).name,\n        name: 'client',\n      },\n      { mode: 'development' },\n    ),\n    webpackConfig(\n      {\n        ...env,\n        entrypath: entrypoint.replace('.tsx', '.server.tsx'),\n        name: 'server',\n        BROWSERSLIST_ENV: 'current node',\n      },\n      { mode: 'development', target: 'node' },\n    ),\n  ];\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs as unknown as MultiConfiguration);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      _next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (error: unknown) {\n        log.error('SSR rendering error:', error);\n\n        // Return error response with status from error if available\n        const expressRes = res as any;\n        if (!expressRes.headersSent) {\n          const statusCode = getErrorStatus(error);\n          expressRes.status(statusCode);\n          expressRes.setHeader('Content-Type', 'text/html');\n          expressRes.send(\n            renderErrorPage(error, req.url ?? '/', statusCode, {\n              showStack: true,\n              badge: 'DEV MODE',\n              hint: 'The dev server is still running. Fix the error and retry, or check the console for more details.',\n            }),\n          );\n        }\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"serve.d.ts","sourceRoot":"","sources":["../../src/scripts/serve.ts"],"names":[],"mappings":";
|
|
1
|
+
{"version":3,"file":"serve.d.ts","sourceRoot":"","sources":["../../src/scripts/serve.ts"],"names":[],"mappings":";AAcA,OAAO,mCAAmC,CAAC;AAmB3C,wBAA8B,KAAK,CACjC,UAAU,EAAE,MAAM,EAClB,OAAO,GAAE;IAAE,WAAW,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAO,iBAsL9D"}
|
package/lib/scripts/serve.js
CHANGED
|
@@ -7,10 +7,10 @@ import express from 'express';
|
|
|
7
7
|
import diskFs from 'fs';
|
|
8
8
|
import ora from 'ora';
|
|
9
9
|
import path from 'path';
|
|
10
|
-
import { promisify } from 'util';
|
|
11
10
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
12
11
|
import getProxyMiddlewares from './getProxyMiddlewares.js';
|
|
13
12
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
13
|
+
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
14
14
|
// run directly from node
|
|
15
15
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
16
16
|
// @ts-ignore
|
|
@@ -29,14 +29,24 @@ export default async function serve(entrypoint, options = {}) {
|
|
|
29
29
|
const manifestPath = getManifestPathFromWebpackconfig(webpackConfig({}, {
|
|
30
30
|
mode: 'production'
|
|
31
31
|
}));
|
|
32
|
-
const readFile = promisify(diskFs.readFile);
|
|
33
32
|
let server;
|
|
34
33
|
function handleErrors(fn) {
|
|
35
|
-
return async function (req, res,
|
|
34
|
+
return async function (req, res, _next) {
|
|
36
35
|
try {
|
|
37
36
|
return await fn(req, res);
|
|
38
|
-
} catch (
|
|
39
|
-
|
|
37
|
+
} catch (error) {
|
|
38
|
+
console.error('SSR rendering error:', error);
|
|
39
|
+
|
|
40
|
+
// Return error response with status from error if available
|
|
41
|
+
const expressRes = res;
|
|
42
|
+
if (!expressRes.headersSent) {
|
|
43
|
+
const statusCode = getErrorStatus(error);
|
|
44
|
+
expressRes.status(statusCode);
|
|
45
|
+
expressRes.setHeader('Content-Type', 'text/html');
|
|
46
|
+
expressRes.send(renderErrorPage(error, req.url ?? '/', statusCode, {
|
|
47
|
+
showStack: process.env.NODE_ENV !== 'production'
|
|
48
|
+
}));
|
|
49
|
+
}
|
|
40
50
|
}
|
|
41
51
|
};
|
|
42
52
|
}
|
|
@@ -62,7 +72,7 @@ export default async function serve(entrypoint, options = {}) {
|
|
|
62
72
|
if (diskFs.existsSync(assetPath) && !diskFs.lstatSync(assetPath).isDirectory()) {
|
|
63
73
|
try {
|
|
64
74
|
res.sendFile(assetPath);
|
|
65
|
-
} catch (
|
|
75
|
+
} catch (_e) {
|
|
66
76
|
return next();
|
|
67
77
|
}
|
|
68
78
|
} else {
|
|
@@ -154,4 +164,4 @@ function getManifestPathFromWebpackconfig(webpackConfig) {
|
|
|
154
164
|
const manifestPath = path.join(webpackConfig?.output?.path ?? '', manifestFilename);
|
|
155
165
|
return manifestPath;
|
|
156
166
|
}
|
|
157
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","compress","express","diskFs","ora","path","promisify","getProxyMiddlewares","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","serve","options","PORT","env","loader","start","webpackConfig","manifestPath","getManifestPathFromWebpackconfig","mode","readFile","server","handleErrors","fn","req","res","next","x","initializeApp","clientManifest","info","fail","wrappingApp","use","serveAssets","filename","url","substring","WEBPACK_PUBLIC_PATH","length","assetPath","join","outputPath","existsSync","lstatSync","isDirectory","sendFile","e","serveProxy","devConfig","devServer","proxy","middlewares","map","middleware","render","cwd","default","Error","get","endsWith","statusCode","setHeader","send","socket","on","error","listen","syscall","isPipe","portOrPipe","Number","isNaN","bind","code","manifest","with","type","warn","close","manifestFilename","plugins","find","plugin","constructor","name","opts","output"],"sources":["../../src/scripts/serve.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport compress from 'compression';\nimport express, { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { Server, IncomingMessage, ServerResponse } from 'http';\nimport ora from 'ora';\nimport path from 'path';\nimport { promisify } from 'util';\nimport webpack from 'webpack';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport getProxyMiddlewares from './getProxyMiddlewares.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { Render } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: ${process.argv[0]} <server-entrypoint>`);\n    process.exit(-1);\n  }\n  serve(entrypoint);\n}\n\nexport default async function serve(\n  entrypoint: string,\n  options: { serveAssets?: boolean; serveProxy?: boolean } = {},\n) {\n  const PORT = process.env.PORT || 8080;\n\n  const loader = ora('Initializing').start();\n\n  const webpackConfig = await getWebpackConfig();\n\n  const manifestPath = getManifestPathFromWebpackconfig(\n    webpackConfig({}, { mode: 'production' }),\n  );\n\n  const readFile = promisify(diskFs.readFile);\n  let server: Server | undefined;\n\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  // Start the express server after the first compilation\n  async function initializeApp(clientManifest: webpack.StatsCompilation) {\n    loader.info('Launching server');\n    if (!clientManifest) {\n      loader.fail('Manifest not found');\n      // TODO: handle more gracefully\n      process.exit(-1);\n    }\n\n    const wrappingApp = express();\n    // eslint-disable-next-line\n    //@ts-ignore\n    wrappingApp.use(compress());\n\n    // ASSETS\n    if (options.serveAssets) {\n      wrappingApp.use(\n        async (\n          req: Request | IncomingMessage,\n          res: any,\n          next: NextFunction,\n        ) => {\n          const filename =\n            req.url?.substring(\n              (process.env.WEBPACK_PUBLIC_PATH as string).length,\n            ) ?? '';\n          const assetPath = path.join(\n            clientManifest.outputPath ?? '',\n            filename,\n          );\n\n          if (\n            diskFs.existsSync(assetPath) &&\n            !diskFs.lstatSync(assetPath).isDirectory()\n          ) {\n            try {\n              res.sendFile(assetPath);\n            } catch (e) {\n              return next();\n            }\n          } else {\n            next();\n          }\n        },\n      );\n    }\n\n    // PROXIES\n    if (options.serveProxy) {\n      const devConfig: webpack.Configuration = webpackConfig(\n        {},\n        { mode: 'development' },\n      );\n      if (devConfig.devServer?.proxy) {\n        const middlewares = getProxyMiddlewares(devConfig.devServer?.proxy);\n        if (middlewares) {\n          wrappingApp.use(...middlewares.map(({ middleware }) => middleware));\n        }\n      }\n    }\n\n    // SERVER SIDE RENDERING\n    let render: Render = (await import(path.join(process.cwd(), entrypoint)))\n      .default;\n\n    if ('default' in render) {\n      render = render.default as any;\n    }\n\n    if (typeof render !== 'function') {\n      throw new Error(\n        `default export of ${path.join(\n          process.cwd(),\n          entrypoint,\n        )} is not a function`,\n      );\n    }\n\n    wrappingApp.get(\n      '/*',\n      handleErrors(async function (req: any, res: any) {\n        if (req.url.endsWith('favicon.ico')) {\n          res.statusCode = 404;\n          res.setHeader('Content-type', 'text/html');\n          res.send('not found');\n          return;\n        }\n        res.socket.on('error', (error: unknown) => {\n          console.error('Fatal', error);\n        });\n\n        await render(clientManifest, req, res);\n      }),\n    );\n\n    server = wrappingApp\n      .listen(PORT, () => {\n        loader.info(`Listening at ${PORT}...`);\n      })\n      .on('error', function (error: any) {\n        if (error.syscall !== 'listen') {\n          throw error;\n        }\n        const isPipe = (portOrPipe: string | number) =>\n          Number.isNaN(portOrPipe);\n        const bind = isPipe(PORT) ? 'Pipe ' + PORT : 'Port ' + PORT;\n        switch (error.code) {\n          case 'EACCES':\n            loader.fail(bind + ' requires elevated privileges');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          case 'EADDRINUSE':\n            loader.fail(bind + ' is already in use');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          default:\n            throw error;\n        }\n      });\n  }\n\n  let manifest = await import(manifestPath, { with: { type: 'json' } });\n  // handle inconsistent import conditions\n  if ('default' in manifest) {\n    manifest = manifest.default;\n  }\n  await initializeApp(manifest);\n\n  process.on('SIGINT', () => {\n    loader.warn('Received SIGINT, devserver shutting down');\n    if (server) console.log('Closing server');\n    server?.close(() => {\n      loader.info('Server closed');\n    });\n    process.exit(-1);\n  });\n}\n\nfunction getManifestPathFromWebpackconfig(\n  webpackConfig: webpack.Configuration,\n) {\n  const manifestFilename: string =\n    (\n      webpackConfig?.plugins?.find(plugin => {\n        return plugin?.constructor.name === 'StatsWriterPlugin';\n      }) as any\n    )?.opts?.filename ?? 'manifest.json';\n\n  const manifestPath = path.join(\n    webpackConfig?.output?.path ?? '',\n    manifestFilename,\n  );\n  return manifestPath;\n}\n"],"mappings":"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","ignoreList":[]}
|
|
167
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","compress","express","diskFs","ora","path","getProxyMiddlewares","getWebpackConfig","getErrorStatus","renderErrorPage","import","meta","main","entrypoint","process","argv","console","log","exit","serve","options","PORT","env","loader","start","webpackConfig","manifestPath","getManifestPathFromWebpackconfig","mode","server","handleErrors","fn","req","res","_next","error","expressRes","headersSent","statusCode","status","setHeader","send","url","showStack","NODE_ENV","initializeApp","clientManifest","info","fail","wrappingApp","use","serveAssets","next","filename","substring","WEBPACK_PUBLIC_PATH","length","assetPath","join","outputPath","existsSync","lstatSync","isDirectory","sendFile","_e","serveProxy","devConfig","devServer","proxy","middlewares","map","middleware","render","cwd","default","Error","get","endsWith","socket","on","listen","syscall","isPipe","portOrPipe","Number","isNaN","bind","code","manifest","with","type","warn","close","manifestFilename","plugins","find","plugin","constructor","name","opts","output"],"sources":["../../src/scripts/serve.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport compress from 'compression';\nimport express, { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { Server, IncomingMessage, ServerResponse } from 'http';\nimport ora from 'ora';\nimport path from 'path';\nimport webpack from 'webpack';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport getProxyMiddlewares from './getProxyMiddlewares.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';\nimport { Render } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: ${process.argv[0]} <server-entrypoint>`);\n    process.exit(-1);\n  }\n  serve(entrypoint);\n}\n\nexport default async function serve(\n  entrypoint: string,\n  options: { serveAssets?: boolean; serveProxy?: boolean } = {},\n) {\n  const PORT = process.env.PORT || 8080;\n\n  const loader = ora('Initializing').start();\n\n  const webpackConfig = await getWebpackConfig();\n\n  const manifestPath = getManifestPathFromWebpackconfig(\n    webpackConfig({}, { mode: 'production' }),\n  );\n\n  let server: Server | undefined;\n\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      _next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (error: unknown) {\n        console.error('SSR rendering error:', error);\n\n        // Return error response with status from error if available\n        const expressRes = res as express.Response;\n        if (!expressRes.headersSent) {\n          const statusCode = getErrorStatus(error);\n          expressRes.status(statusCode);\n          expressRes.setHeader('Content-Type', 'text/html');\n          expressRes.send(\n            renderErrorPage(error, req.url ?? '/', statusCode, {\n              showStack: process.env.NODE_ENV !== 'production',\n            }),\n          );\n        }\n      }\n    };\n  }\n\n  // Start the express server after the first compilation\n  async function initializeApp(clientManifest: webpack.StatsCompilation) {\n    loader.info('Launching server');\n    if (!clientManifest) {\n      loader.fail('Manifest not found');\n      // TODO: handle more gracefully\n      process.exit(-1);\n    }\n\n    const wrappingApp = express();\n    // eslint-disable-next-line\n    //@ts-ignore\n    wrappingApp.use(compress());\n\n    // ASSETS\n    if (options.serveAssets) {\n      wrappingApp.use(\n        async (\n          req: Request | IncomingMessage,\n          res: any,\n          next: NextFunction,\n        ) => {\n          const filename =\n            req.url?.substring(\n              (process.env.WEBPACK_PUBLIC_PATH as string).length,\n            ) ?? '';\n          const assetPath = path.join(\n            clientManifest.outputPath ?? '',\n            filename,\n          );\n\n          if (\n            diskFs.existsSync(assetPath) &&\n            !diskFs.lstatSync(assetPath).isDirectory()\n          ) {\n            try {\n              res.sendFile(assetPath);\n            } catch (_e) {\n              return next();\n            }\n          } else {\n            next();\n          }\n        },\n      );\n    }\n\n    // PROXIES\n    if (options.serveProxy) {\n      const devConfig: webpack.Configuration = webpackConfig(\n        {},\n        { mode: 'development' },\n      );\n      if (devConfig.devServer?.proxy) {\n        const middlewares = getProxyMiddlewares(devConfig.devServer?.proxy);\n        if (middlewares) {\n          wrappingApp.use(...middlewares.map(({ middleware }) => middleware));\n        }\n      }\n    }\n\n    // SERVER SIDE RENDERING\n    let render: Render = (await import(path.join(process.cwd(), entrypoint)))\n      .default;\n\n    if ('default' in render) {\n      render = render.default as any;\n    }\n\n    if (typeof render !== 'function') {\n      throw new Error(\n        `default export of ${path.join(\n          process.cwd(),\n          entrypoint,\n        )} is not a function`,\n      );\n    }\n\n    wrappingApp.get(\n      '/*',\n      handleErrors(async function (req: any, res: any) {\n        if (req.url.endsWith('favicon.ico')) {\n          res.statusCode = 404;\n          res.setHeader('Content-type', 'text/html');\n          res.send('not found');\n          return;\n        }\n        res.socket.on('error', (error: unknown) => {\n          console.error('Fatal', error);\n        });\n\n        await render(clientManifest, req, res);\n      }),\n    );\n\n    server = wrappingApp\n      .listen(PORT, () => {\n        loader.info(`Listening at ${PORT}...`);\n      })\n      .on('error', function (error: any) {\n        if (error.syscall !== 'listen') {\n          throw error;\n        }\n        const isPipe = (portOrPipe: string | number) =>\n          Number.isNaN(portOrPipe);\n        const bind = isPipe(PORT) ? 'Pipe ' + PORT : 'Port ' + PORT;\n        switch (error.code) {\n          case 'EACCES':\n            loader.fail(bind + ' requires elevated privileges');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          case 'EADDRINUSE':\n            loader.fail(bind + ' is already in use');\n            process.exit(1);\n          // eslint-disable-next-line no-fallthrough\n          default:\n            throw error;\n        }\n      });\n  }\n\n  let manifest = await import(manifestPath, { with: { type: 'json' } });\n  // handle inconsistent import conditions\n  if ('default' in manifest) {\n    manifest = manifest.default;\n  }\n  await initializeApp(manifest);\n\n  process.on('SIGINT', () => {\n    loader.warn('Received SIGINT, devserver shutting down');\n    if (server) console.log('Closing server');\n    server?.close(() => {\n      loader.info('Server closed');\n    });\n    process.exit(-1);\n  });\n}\n\nfunction getManifestPathFromWebpackconfig(\n  webpackConfig: webpack.Configuration,\n) {\n  const manifestFilename: string =\n    (\n      webpackConfig?.plugins?.find(plugin => {\n        return plugin?.constructor.name === 'StatsWriterPlugin';\n      }) as any\n    )?.opts?.filename ?? 'manifest.json';\n\n  const manifestPath = path.join(\n    webpackConfig?.output?.path ?? '',\n    manifestFilename,\n  );\n  return manifestPath;\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Utilities for handling SSR errors gracefully
|
|
3
|
+
*/
|
|
4
|
+
/**
|
|
5
|
+
* Extract HTTP status code from an error object.
|
|
6
|
+
* Looks for a `status` property that is a number or parseable string.
|
|
7
|
+
* Returns 500 if no valid status found.
|
|
8
|
+
*/
|
|
9
|
+
export declare function getErrorStatus(error: unknown): number;
|
|
10
|
+
/**
|
|
11
|
+
* Escape HTML special characters to prevent XSS
|
|
12
|
+
*/
|
|
13
|
+
export declare function escapeHtml(str: string): string;
|
|
14
|
+
export interface RenderErrorPageOptions {
|
|
15
|
+
/** Show stack trace in output */
|
|
16
|
+
showStack?: boolean;
|
|
17
|
+
/** Additional hint message to display */
|
|
18
|
+
hint?: string;
|
|
19
|
+
/** Badge text to display (e.g., "DEV MODE") */
|
|
20
|
+
badge?: string;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Render an HTML error page for SSR failures
|
|
24
|
+
*/
|
|
25
|
+
export declare function renderErrorPage(error: unknown, url: string, statusCode: number, options?: RenderErrorPageOptions): string;
|
|
26
|
+
//# sourceMappingURL=ssrErrorHandler.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ssrErrorHandler.d.ts","sourceRoot":"","sources":["../../src/scripts/ssrErrorHandler.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH;;;;GAIG;AACH,wBAAgB,cAAc,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAcrD;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAO9C;AAED,MAAM,WAAW,sBAAsB;IACrC,iCAAiC;IACjC,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,yCAAyC;IACzC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,+CAA+C;IAC/C,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED;;GAEG;AACH,wBAAgB,eAAe,CAC7B,KAAK,EAAE,OAAO,EACd,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,OAAO,GAAE,sBAA2B,GACnC,MAAM,CA2CR"}
|