@anansi/core 0.7.2 → 0.8.0
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 +30 -0
- package/dist/server.js +27 -23
- package/dist/server.js.map +1 -1
- package/lib/scripts/index.d.ts +3 -0
- package/lib/scripts/index.d.ts.map +1 -0
- package/lib/scripts/index.js +15 -0
- package/lib/scripts/serve.d.ts +4 -0
- package/lib/scripts/serve.d.ts.map +1 -0
- package/lib/scripts/serve.js +162 -0
- package/lib/scripts/startDevserver.d.ts +1 -0
- package/lib/scripts/startDevserver.d.ts.map +1 -1
- package/lib/scripts/startDevserver.js +175 -161
- package/lib/spouts/DocumentComponent.d.ts +3 -1
- package/lib/spouts/DocumentComponent.d.ts.map +1 -1
- package/lib/spouts/DocumentComponent.js +10 -6
- package/lib/spouts/document.server.d.ts +2 -0
- package/lib/spouts/document.server.d.ts.map +1 -1
- package/lib/spouts/document.server.js +1 -1
- package/package.json +8 -3
- package/src/scripts/index.ts +2 -0
- package/src/scripts/serve.ts +168 -0
- package/src/scripts/startDevserver.ts +220 -195
- package/src/spouts/DocumentComponent.tsx +8 -2
- package/src/spouts/document.server.tsx +2 -0
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,36 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
## [0.8.0](https://github.com/ntucker/anansi/compare/@anansi/core@0.7.4...@anansi/core@0.8.0) (2022-05-29)
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
### 🚀 Features
|
|
10
|
+
|
|
11
|
+
* Add anansi serve command ([#1525](https://github.com/ntucker/anansi/issues/1525)) ([ac5a396](https://github.com/ntucker/anansi/commit/ac5a396f9640ce18058813c1594d49367a8aa468))
|
|
12
|
+
* Add charSet option to documentSpout ([51d3168](https://github.com/ntucker/anansi/commit/51d31681d27144d06b8ed4c7da9aae43fd10f12b))
|
|
13
|
+
* Add production node service ([bceeb56](https://github.com/ntucker/anansi/commit/bceeb56c23c8f8c3dc4a15d35dc8877f277e3d2f))
|
|
14
|
+
* Handle devserver proxy ([e7c5b38](https://github.com/ntucker/anansi/commit/e7c5b38cdb60b236db15ffb28622889b0b771515))
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
### [0.7.4](https://github.com/ntucker/anansi/compare/@anansi/core@0.7.3...@anansi/core@0.7.4) (2022-05-28)
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
### 💅 Enhancement
|
|
22
|
+
|
|
23
|
+
* Stall SSR requests until build is ready ([fe2616a](https://github.com/ntucker/anansi/commit/fe2616ac2d957f7243310d896f4a2ad5cb7d910e))
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
### [0.7.3](https://github.com/ntucker/anansi/compare/@anansi/core@0.7.2...@anansi/core@0.7.3) (2022-05-28)
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
### 💅 Enhancement
|
|
31
|
+
|
|
32
|
+
* Require, then import fresh ([7de7d8d](https://github.com/ntucker/anansi/commit/7de7d8d73e533b69bdf78698965121a35375692c))
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
|
|
6
36
|
### [0.7.2](https://github.com/ntucker/anansi/compare/@anansi/core@0.7.1...@anansi/core@0.7.2) (2022-05-28)
|
|
7
37
|
|
|
8
38
|
|
package/dist/server.js
CHANGED
|
@@ -128,51 +128,62 @@ function laySpouts(spouts, {
|
|
|
128
128
|
const external_react_namespaceObject = require("react");
|
|
129
129
|
var external_react_default = /*#__PURE__*/__webpack_require__.n(external_react_namespaceObject);
|
|
130
130
|
;// CONCATENATED MODULE: ./src/spouts/DocumentComponent.tsx
|
|
131
|
-
var _jsxFileName = "/home/ntucker/src/anansi/packages/core/src/spouts/DocumentComponent.tsx"
|
|
131
|
+
var _jsxFileName = "/home/ntucker/src/anansi/packages/core/src/spouts/DocumentComponent.tsx",
|
|
132
|
+
_process$env$WEBPACK_;
|
|
133
|
+
|
|
132
134
|
|
|
133
135
|
function Document({
|
|
134
136
|
assets,
|
|
135
137
|
head,
|
|
136
138
|
children,
|
|
137
139
|
title,
|
|
138
|
-
rootId
|
|
140
|
+
rootId,
|
|
141
|
+
charSet
|
|
139
142
|
}) {
|
|
140
143
|
return /*#__PURE__*/external_react_default().createElement("html", {
|
|
141
144
|
__self: this,
|
|
142
145
|
__source: {
|
|
143
146
|
fileName: _jsxFileName,
|
|
144
|
-
lineNumber:
|
|
147
|
+
lineNumber: 19,
|
|
145
148
|
columnNumber: 5
|
|
146
149
|
}
|
|
147
150
|
}, /*#__PURE__*/external_react_default().createElement("head", {
|
|
148
151
|
__self: this,
|
|
149
152
|
__source: {
|
|
150
153
|
fileName: _jsxFileName,
|
|
151
|
-
lineNumber:
|
|
154
|
+
lineNumber: 20,
|
|
152
155
|
columnNumber: 7
|
|
153
156
|
}
|
|
154
|
-
},
|
|
157
|
+
}, /*#__PURE__*/external_react_default().createElement("meta", {
|
|
158
|
+
charSet: charSet,
|
|
159
|
+
__self: this,
|
|
160
|
+
__source: {
|
|
161
|
+
fileName: _jsxFileName,
|
|
162
|
+
lineNumber: 21,
|
|
163
|
+
columnNumber: 9
|
|
164
|
+
}
|
|
165
|
+
}), head, assets.map((asset, i) => /*#__PURE__*/external_react_default().createElement("link", {
|
|
155
166
|
key: i,
|
|
156
167
|
rel: "preload",
|
|
157
168
|
...asset,
|
|
158
169
|
__self: this,
|
|
159
170
|
__source: {
|
|
160
171
|
fileName: _jsxFileName,
|
|
161
|
-
lineNumber:
|
|
172
|
+
lineNumber: 24,
|
|
162
173
|
columnNumber: 11
|
|
163
174
|
}
|
|
164
175
|
})), /*#__PURE__*/external_react_default().createElement("title", {
|
|
165
176
|
__self: this,
|
|
166
177
|
__source: {
|
|
167
178
|
fileName: _jsxFileName,
|
|
168
|
-
lineNumber:
|
|
179
|
+
lineNumber: 26,
|
|
169
180
|
columnNumber: 9
|
|
170
181
|
}
|
|
171
182
|
}, title)), /*#__PURE__*/external_react_default().createElement("body", {
|
|
172
183
|
__self: this,
|
|
173
184
|
__source: {
|
|
174
185
|
fileName: _jsxFileName,
|
|
175
|
-
lineNumber:
|
|
186
|
+
lineNumber: 28,
|
|
176
187
|
columnNumber: 7
|
|
177
188
|
}
|
|
178
189
|
}, /*#__PURE__*/external_react_default().createElement("div", {
|
|
@@ -180,7 +191,7 @@ function Document({
|
|
|
180
191
|
__self: this,
|
|
181
192
|
__source: {
|
|
182
193
|
fileName: _jsxFileName,
|
|
183
|
-
lineNumber:
|
|
194
|
+
lineNumber: 29,
|
|
184
195
|
columnNumber: 9
|
|
185
196
|
}
|
|
186
197
|
}, children), /*#__PURE__*/external_react_default().createElement("script", {
|
|
@@ -190,7 +201,7 @@ function Document({
|
|
|
190
201
|
__self: this,
|
|
191
202
|
__source: {
|
|
192
203
|
fileName: _jsxFileName,
|
|
193
|
-
lineNumber:
|
|
204
|
+
lineNumber: 31,
|
|
194
205
|
columnNumber: 9
|
|
195
206
|
}
|
|
196
207
|
}), assets.filter(({
|
|
@@ -204,39 +215,32 @@ function Document({
|
|
|
204
215
|
__self: this,
|
|
205
216
|
__source: {
|
|
206
217
|
fileName: _jsxFileName,
|
|
207
|
-
lineNumber:
|
|
218
|
+
lineNumber: 39,
|
|
208
219
|
columnNumber: 13
|
|
209
220
|
}
|
|
210
221
|
}))));
|
|
211
222
|
}
|
|
212
223
|
Document.defaultProps = {
|
|
213
224
|
head: /*#__PURE__*/external_react_default().createElement((external_react_default()).Fragment, null, /*#__PURE__*/external_react_default().createElement("meta", {
|
|
214
|
-
charSet: "utf-8",
|
|
215
|
-
__self: undefined,
|
|
216
|
-
__source: {
|
|
217
|
-
fileName: _jsxFileName,
|
|
218
|
-
lineNumber: 45,
|
|
219
|
-
columnNumber: 7
|
|
220
|
-
}
|
|
221
|
-
}), /*#__PURE__*/external_react_default().createElement("meta", {
|
|
222
225
|
name: "viewport",
|
|
223
226
|
content: "width=device-width, initial-scale=1",
|
|
224
227
|
__self: undefined,
|
|
225
228
|
__source: {
|
|
226
229
|
fileName: _jsxFileName,
|
|
227
|
-
lineNumber:
|
|
230
|
+
lineNumber: 48,
|
|
228
231
|
columnNumber: 7
|
|
229
232
|
}
|
|
230
233
|
}), /*#__PURE__*/external_react_default().createElement("link", {
|
|
231
234
|
rel: "shortcut icon",
|
|
232
|
-
href:
|
|
235
|
+
href: `${(_process$env$WEBPACK_ = process.env.WEBPACK_PUBLIC_PATH) != null ? _process$env$WEBPACK_ : '/'}favicon.ico`,
|
|
233
236
|
__self: undefined,
|
|
234
237
|
__source: {
|
|
235
238
|
fileName: _jsxFileName,
|
|
236
|
-
lineNumber:
|
|
239
|
+
lineNumber: 49,
|
|
237
240
|
columnNumber: 7
|
|
238
241
|
}
|
|
239
242
|
})),
|
|
243
|
+
charSet: 'utf-8',
|
|
240
244
|
rootId: 'anansi-root'
|
|
241
245
|
};
|
|
242
246
|
;// CONCATENATED MODULE: ./src/spouts/document.server.tsx
|
|
@@ -289,7 +293,7 @@ function DocumentSpout(options) {
|
|
|
289
293
|
__self: this,
|
|
290
294
|
__source: {
|
|
291
295
|
fileName: document_server_jsxFileName,
|
|
292
|
-
lineNumber:
|
|
296
|
+
lineNumber: 54,
|
|
293
297
|
columnNumber: 14
|
|
294
298
|
}
|
|
295
299
|
}, nextProps.app)
|
package/dist/server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.js","mappings":";;AAAA;AACA;AACA;;;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACPA;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACNA;;ACAA;AAKA;AAIA;AAAA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;AAEA;;AAnBA;AAuBA;;AACA;AACA;;AACA;AACA;;AC3DA;;;;;ACQA;AACA;AACA;AACA;AACA;AACA;AALA;AAOA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAGA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAGA;AACA;AADA;AADA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AAAA;AACA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAKA;AACA;AACA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAGA;AARA;;;ACzCA;AAKA;AAOA;AAKA;AAGA;AAAA;;AACA;AAEA;AAEA;AAOA;;AACA;AACA;AAAA;;AAEA;AACA;AACA;;AACA;AAFA;AAKA;AAGA;AAAA;AAGA;;AAAA;AAGA;AAHA;;AASA;AAQA;AAAA;AAAA;AAEA;AAAA;AAAA;AACA;AAAA;AAGA;AAEA;AAGA;AACA;AACA;AAJA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAHA;AAnDA;AAHA;AAqEA;;AAEA;AACA;AAEA;;AAAA;AAAA;AAGA;;AC9FA;;ACAA;;;;ACAA;AACA;AAMA;AAGA;AAAA;AAEA;AAGA;AACA;AAIA;AAEA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAHA;AAKA;AACA;AACA;;AC7BA;;ACAA;;;ACAA;AACA;AACA;AAMA;AAKA;AAGA;AAAA;AACA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA;;AAEA;AAGA;AACA;AACA;AACA;AAAA;AAEA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAJA;AAMA;AACA;AACA;;ACvCA;AACA;AAOA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA","sources":["/home/ntucker/src/anansi/packages/core/webpack/bootstrap","/home/ntucker/src/anansi/packages/core/webpack/runtime/compat get default export","/home/ntucker/src/anansi/packages/core/webpack/runtime/define property getters","/home/ntucker/src/anansi/packages/core/webpack/runtime/hasOwnProperty shorthand","/home/ntucker/src/anansi/packages/core/webpack/runtime/make namespace object","/home/ntucker/src/anansi/packages/core/external commonjs \"react-dom/server\"","/home/ntucker/src/anansi/packages/core/src/laySpouts.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"react\"","/home/ntucker/src/anansi/packages/core/src/spouts/DocumentComponent.tsx","/home/ntucker/src/anansi/packages/core/src/spouts/document.server.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"@rest-hooks/core\"","/home/ntucker/src/anansi/packages/core/external commonjs \"@rest-hooks/ssr\"","/home/ntucker/src/anansi/packages/core/src/spouts/restHooks.server.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"@anansi/router\"","/home/ntucker/src/anansi/packages/core/external commonjs \"history\"","/home/ntucker/src/anansi/packages/core/src/spouts/router.server.tsx","/home/ntucker/src/anansi/packages/core/src/spouts/prefetch.server.tsx","/home/ntucker/src/anansi/packages/core/src/index.server.ts"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"react-dom/server\");","import { renderToPipeableStream as reactRender } from 'react-dom/server';\n\nimport { Render } from './scripts/types';\nimport { ServerProps } from './spouts/types';\n\nexport default function laySpouts(\n spouts: (props: ServerProps) => Promise<{\n app: JSX.Element;\n }>,\n { timeoutMS = 200 }: { timeoutMS?: number } = {},\n) {\n const render: Render = async (clientManifest, req, res) => {\n const { app } = await spouts({ clientManifest, req, res });\n let didError = false;\n const { pipe, abort } = reactRender(\n app,\n /*\n This is not documented, so included the types here for reference:\ntype Options = {|\n identifierPrefix?: string,\n namespaceURI?: string,\n nonce?: string,\n bootstrapScriptContent?: string,\n bootstrapScripts?: Array<string>,\n bootstrapModules?: Array<string>,\n progressiveChunkSize?: number,\n onShellReady?: () => void,\n onShellError?: () => void,\n onAllReady?: () => void,\n onError?: (error: mixed) => void,\n|};\n */\n {\n //bootstrapScripts: assets.filter(asset => asset.endsWith('.js')),\n onShellReady() {\n //managers.forEach(manager => manager.cleanup());\n // If something errored before we started streaming, we set the error code appropriately.\n res.statusCode = didError ? 500 : 200;\n res.setHeader('Content-type', 'text/html');\n pipe(res);\n },\n onShellError() {\n didError = true;\n res.statusCode = 500;\n pipe(res);\n },\n onError(x: any) {\n didError = true;\n console.error(x);\n res.statusCode = 500;\n //pipe(res); Removing this avoids, \"React currently only supports piping to one writable stream.\"\n },\n },\n );\n // Abandon and switch to client rendering if enough time passes.\n // Try lowering this to see the client recover.\n setTimeout(abort, timeoutMS);\n };\n return render;\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"react\");","type Props = {\n children: React.ReactNode;\n assets: { href: string; as?: string; rel?: string }[];\n head: React.ReactNode;\n title: string;\n rootId: string;\n};\n\nexport default function Document({\n assets,\n head,\n children,\n title,\n rootId,\n}: Props) {\n return (\n <html>\n <head>\n {head}\n {assets.map((asset, i) => (\n <link key={i} rel=\"preload\" {...asset} />\n ))}\n <title>{title}</title>\n </head>\n <body>\n <div id={rootId}>{children}</div>\n {/* this ensures the client can hydrate the assets prop */}\n <script\n dangerouslySetInnerHTML={{\n __html: `assetManifest = ${JSON.stringify(assets)};`,\n }}\n />\n {assets\n .filter(({ href }) => href.endsWith('.js'))\n .map(({ href }, i) => (\n <script key={i} src={href} async />\n ))}\n </body>\n </html>\n );\n}\nDocument.defaultProps = {\n head: (\n <>\n <meta charSet=\"utf-8\" />\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\" />\n <link rel=\"shortcut icon\" href=\"/assets/favicon.ico\" />\n </>\n ),\n rootId: 'anansi-root',\n};\n","import React from 'react';\nimport type { Route } from '@anansi/router';\nimport { StatsChunkGroup } from 'webpack';\n\nimport type { ServerProps, ResolveProps } from './types';\nimport Document from './DocumentComponent';\n\ntype NeededProps = {\n matchedRoutes: Route<any>[];\n title?: string;\n} & ResolveProps;\n\nexport default function DocumentSpout(options: {\n head?: React.ReactNode;\n title: string;\n rootId: string;\n}) {\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const nextProps = await next(props);\n\n const publicPath = props.clientManifest.publicPath;\n\n if (\n Object.keys(props.clientManifest?.entrypoints ?? {}).length < 1 ||\n publicPath === undefined\n )\n throw new Error('Manifest missing entries needed');\n\n // TODO: consider using this package for build stats in future:\n // https://github.com/facebook/react/tree/main/packages/react-server-dom-webpack\n const assetMap = (assets: { name: string; size?: number }[]) =>\n assets.map(({ name }) => `${publicPath}${name}`);\n\n const assetList: string[] = [];\n Object.values(props.clientManifest?.entrypoints ?? {}).forEach(\n entrypoint => {\n assetList.push(...assetMap(entrypoint.assets ?? []));\n },\n );\n new Set(\n assetMap(\n Object.values(props.clientManifest.namedChunkGroups ?? {})\n .filter(({ name }) =>\n nextProps.matchedRoutes.some(route => name?.includes(route.name)),\n )\n .flatMap(chunk => [\n ...(chunk.assets ?? []),\n // any chunk preloads\n ...childrenAssets(chunk),\n ]),\n ),\n ).forEach(asset => assetList.push(asset));\n\n // find additional assets to preload based on matched route\n const assets: {\n href: string;\n as?: string | undefined;\n rel?: string | undefined;\n }[] = assetList\n .filter(asset => !asset.endsWith('.hot-update.js'))\n .map(asset =>\n asset.endsWith('.css')\n ? { href: asset, rel: 'stylesheet' }\n : asset.endsWith('.js')\n ? { href: asset, as: 'script' }\n : { href: asset },\n );\n\n return {\n ...nextProps,\n app: (\n <Document\n {...options}\n title={nextProps.title ?? options.title}\n assets={assets}\n rootId={options.rootId}\n >\n {nextProps.app}\n </Document>\n ),\n };\n };\n };\n}\n\nfunction childrenAssets(chunk: StatsChunkGroup) {\n return chunk.children\n ? Object.values(chunk.children).flatMap(preload =>\n preload.flatMap(c => c.assets ?? []),\n )\n : [];\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@rest-hooks/core\");","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@rest-hooks/ssr\");","import { Manager, NetworkManager } from '@rest-hooks/core';\nimport { createPersistedStore } from '@rest-hooks/ssr';\n\nimport type { ResolveProps, ServerProps } from './types';\n\ntype NeededProps = ResolveProps;\n\nexport default function restHooksSpout(\n options: {\n getManagers: () => Manager[];\n } = { getManagers: () => [new NetworkManager()] },\n) {\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const [ServerCacheProvider, controller] = createPersistedStore(\n options.getManagers(),\n );\n\n const nextProps = await next(props);\n\n return {\n ...nextProps,\n controller,\n app: <ServerCacheProvider>{nextProps.app}</ServerCacheProvider>,\n };\n };\n };\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@anansi/router\");","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"history\");","import { Route, RouteProvider, RouteController } from '@anansi/router';\nimport React from 'react';\nimport { createMemoryHistory } from 'history';\n\nimport type { ResolveProps, ServerProps, CreateRouter } from './types';\n\ntype NeededProps = ResolveProps;\n\nexport default function routerSpout<ResolveWith>(options: {\n resolveWith?: any;\n useResolveWith: () => ResolveWith;\n createRouter: CreateRouter<ResolveWith>;\n}) {\n const createRouteComponent = (\n router: RouteController<Route<ResolveWith, any>>,\n ) =>\n function Router({ children }: { children: React.ReactNode }) {\n const resolveWith = options.useResolveWith();\n\n return (\n <RouteProvider router={router} resolveWith={resolveWith}>\n {children}\n </RouteProvider>\n );\n };\n\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const url = props.req.url || '';\n const router = options.createRouter(\n createMemoryHistory({ initialEntries: [url] }),\n );\n const matchedRoutes: Route<ResolveWith>[] = router.getMatchedRoutes(url);\n\n const nextProps = await next(props);\n\n const Router = createRouteComponent(router);\n return {\n ...nextProps,\n matchedRoutes,\n router,\n app: <Router>{nextProps.app}</Router>,\n };\n };\n };\n}\n","import { Route } from '@anansi/router';\n\nimport type { ResolveProps, ServerProps } from './types';\n\ntype NeededProps<RouteWith> = {\n matchedRoutes: Route<RouteWith>[];\n} & ResolveProps;\n\nexport default function prefetchSpout<F extends string>(field: F) {\n return function <RouteWith, T extends NeededProps<RouteWith>>(\n next: (props: ServerProps) => Promise<\n {\n [K in F]: RouteWith;\n } & T\n >,\n ) {\n return async (props: ServerProps) => {\n const nextProps = await next(props);\n\n try {\n const toFetch: Promise<unknown>[] = [];\n nextProps.matchedRoutes.forEach(route => {\n if (typeof route.resolveData === 'function') {\n toFetch.push(route.resolveData(nextProps[field], route));\n }\n });\n await Promise.all(toFetch);\n } catch (e) {\n console.error(e);\n }\n return nextProps;\n };\n };\n}\n","export { default as laySpouts } from './laySpouts';\nexport { default as documentSpout } from './spouts/document.server';\nexport { default as restHooksSpout } from './spouts/restHooks.server';\nexport { default as routerSpout } from './spouts/router.server';\nexport { default as prefetchSpout } from './spouts/prefetch.server';\n"],"names":[],"sourceRoot":""}
|
|
1
|
+
{"version":3,"file":"server.js","mappings":";;AAAA;AACA;AACA;;;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACPA;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACNA;;ACAA;AAKA;AAIA;AAAA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;AAEA;;AAnBA;AAuBA;;AACA;AACA;;AACA;AACA;;AC3DA;;;;;;;ACSA;AACA;AACA;AACA;AACA;AACA;AACA;AANA;AAQA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAGA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAGA;AACA;AADA;AADA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AAAA;AACA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAKA;AACA;AACA;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA;AACA;AAFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AAXA;;;AC5CA;AAKA;AAQA;AAMA;AAGA;AAAA;;AACA;AAEA;AAEA;AAOA;;AACA;AACA;AAAA;;AAEA;AACA;AACA;;AACA;AAFA;AAKA;AAGA;AAAA;AAGA;;AAAA;AAGA;AAHA;;AASA;AAQA;AAAA;AAAA;AAEA;AAAA;AAAA;AACA;AAAA;AAGA;AAEA;AAGA;AACA;AACA;AAJA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAHA;AAnDA;AAHA;AAqEA;;AAEA;AACA;AAEA;;AAAA;AAAA;AAGA;;AChGA;;ACAA;;;;ACAA;AACA;AAMA;AAGA;AAAA;AAEA;AAGA;AACA;AAIA;AAEA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAHA;AAKA;AACA;AACA;;AC7BA;;ACAA;;;ACAA;AACA;AACA;AAMA;AAKA;AAGA;AAAA;AACA;AAEA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA;;AAEA;AAGA;AACA;AACA;AACA;AAAA;AAEA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAJA;AAMA;AACA;AACA;;ACvCA;AACA;AAOA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA","sources":["/home/ntucker/src/anansi/packages/core/webpack/bootstrap","/home/ntucker/src/anansi/packages/core/webpack/runtime/compat get default export","/home/ntucker/src/anansi/packages/core/webpack/runtime/define property getters","/home/ntucker/src/anansi/packages/core/webpack/runtime/hasOwnProperty shorthand","/home/ntucker/src/anansi/packages/core/webpack/runtime/make namespace object","/home/ntucker/src/anansi/packages/core/external commonjs \"react-dom/server\"","/home/ntucker/src/anansi/packages/core/src/laySpouts.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"react\"","/home/ntucker/src/anansi/packages/core/src/spouts/DocumentComponent.tsx","/home/ntucker/src/anansi/packages/core/src/spouts/document.server.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"@rest-hooks/core\"","/home/ntucker/src/anansi/packages/core/external commonjs \"@rest-hooks/ssr\"","/home/ntucker/src/anansi/packages/core/src/spouts/restHooks.server.tsx","/home/ntucker/src/anansi/packages/core/external commonjs \"@anansi/router\"","/home/ntucker/src/anansi/packages/core/external commonjs \"history\"","/home/ntucker/src/anansi/packages/core/src/spouts/router.server.tsx","/home/ntucker/src/anansi/packages/core/src/spouts/prefetch.server.tsx","/home/ntucker/src/anansi/packages/core/src/index.server.ts"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"react-dom/server\");","import { renderToPipeableStream as reactRender } from 'react-dom/server';\n\nimport { Render } from './scripts/types';\nimport { ServerProps } from './spouts/types';\n\nexport default function laySpouts(\n spouts: (props: ServerProps) => Promise<{\n app: JSX.Element;\n }>,\n { timeoutMS = 200 }: { timeoutMS?: number } = {},\n) {\n const render: Render = async (clientManifest, req, res) => {\n const { app } = await spouts({ clientManifest, req, res });\n let didError = false;\n const { pipe, abort } = reactRender(\n app,\n /*\n This is not documented, so included the types here for reference:\ntype Options = {|\n identifierPrefix?: string,\n namespaceURI?: string,\n nonce?: string,\n bootstrapScriptContent?: string,\n bootstrapScripts?: Array<string>,\n bootstrapModules?: Array<string>,\n progressiveChunkSize?: number,\n onShellReady?: () => void,\n onShellError?: () => void,\n onAllReady?: () => void,\n onError?: (error: mixed) => void,\n|};\n */\n {\n //bootstrapScripts: assets.filter(asset => asset.endsWith('.js')),\n onShellReady() {\n //managers.forEach(manager => manager.cleanup());\n // If something errored before we started streaming, we set the error code appropriately.\n res.statusCode = didError ? 500 : 200;\n res.setHeader('Content-type', 'text/html');\n pipe(res);\n },\n onShellError() {\n didError = true;\n res.statusCode = 500;\n pipe(res);\n },\n onError(x: any) {\n didError = true;\n console.error(x);\n res.statusCode = 500;\n //pipe(res); Removing this avoids, \"React currently only supports piping to one writable stream.\"\n },\n },\n );\n // Abandon and switch to client rendering if enough time passes.\n // Try lowering this to see the client recover.\n setTimeout(abort, timeoutMS);\n };\n return render;\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"react\");","type Props = {\n children: React.ReactNode;\n assets: { href: string; as?: string; rel?: string }[];\n head: React.ReactNode;\n title: string;\n rootId: string;\n charSet: string;\n};\n\nexport default function Document({\n assets,\n head,\n children,\n title,\n rootId,\n charSet,\n}: Props) {\n return (\n <html>\n <head>\n <meta charSet={charSet} />\n {head}\n {assets.map((asset, i) => (\n <link key={i} rel=\"preload\" {...asset} />\n ))}\n <title>{title}</title>\n </head>\n <body>\n <div id={rootId}>{children}</div>\n {/* this ensures the client can hydrate the assets prop */}\n <script\n dangerouslySetInnerHTML={{\n __html: `assetManifest = ${JSON.stringify(assets)};`,\n }}\n />\n {assets\n .filter(({ href }) => href.endsWith('.js'))\n .map(({ href }, i) => (\n <script key={i} src={href} async />\n ))}\n </body>\n </html>\n );\n}\nDocument.defaultProps = {\n head: (\n <>\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\" />\n <link\n rel=\"shortcut icon\"\n href={`${process.env.WEBPACK_PUBLIC_PATH ?? '/'}favicon.ico`}\n />\n </>\n ),\n charSet: 'utf-8',\n rootId: 'anansi-root',\n};\n","import React from 'react';\nimport type { Route } from '@anansi/router';\nimport { StatsChunkGroup } from 'webpack';\n\nimport type { ServerProps, ResolveProps } from './types';\nimport Document from './DocumentComponent';\n\ntype NeededProps = {\n matchedRoutes: Route<any>[];\n title?: string;\n charSet?: string;\n} & ResolveProps;\n\nexport default function DocumentSpout(options: {\n head?: React.ReactNode;\n title: string;\n rootId: string;\n charSet: string;\n}) {\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const nextProps = await next(props);\n\n const publicPath = props.clientManifest.publicPath;\n\n if (\n Object.keys(props.clientManifest?.entrypoints ?? {}).length < 1 ||\n publicPath === undefined\n )\n throw new Error('Manifest missing entries needed');\n\n // TODO: consider using this package for build stats in future:\n // https://github.com/facebook/react/tree/main/packages/react-server-dom-webpack\n const assetMap = (assets: { name: string; size?: number }[]) =>\n assets.map(({ name }) => `${publicPath}${name}`);\n\n const assetList: string[] = [];\n Object.values(props.clientManifest?.entrypoints ?? {}).forEach(\n entrypoint => {\n assetList.push(...assetMap(entrypoint.assets ?? []));\n },\n );\n new Set(\n assetMap(\n Object.values(props.clientManifest.namedChunkGroups ?? {})\n .filter(({ name }) =>\n nextProps.matchedRoutes.some(route => name?.includes(route.name)),\n )\n .flatMap(chunk => [\n ...(chunk.assets ?? []),\n // any chunk preloads\n ...childrenAssets(chunk),\n ]),\n ),\n ).forEach(asset => assetList.push(asset));\n\n // find additional assets to preload based on matched route\n const assets: {\n href: string;\n as?: string | undefined;\n rel?: string | undefined;\n }[] = assetList\n .filter(asset => !asset.endsWith('.hot-update.js'))\n .map(asset =>\n asset.endsWith('.css')\n ? { href: asset, rel: 'stylesheet' }\n : asset.endsWith('.js')\n ? { href: asset, as: 'script' }\n : { href: asset },\n );\n\n return {\n ...nextProps,\n app: (\n <Document\n {...options}\n title={nextProps.title ?? options.title}\n assets={assets}\n rootId={options.rootId}\n >\n {nextProps.app}\n </Document>\n ),\n };\n };\n };\n}\n\nfunction childrenAssets(chunk: StatsChunkGroup) {\n return chunk.children\n ? Object.values(chunk.children).flatMap(preload =>\n preload.flatMap(c => c.assets ?? []),\n )\n : [];\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@rest-hooks/core\");","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@rest-hooks/ssr\");","import { Manager, NetworkManager } from '@rest-hooks/core';\nimport { createPersistedStore } from '@rest-hooks/ssr';\n\nimport type { ResolveProps, ServerProps } from './types';\n\ntype NeededProps = ResolveProps;\n\nexport default function restHooksSpout(\n options: {\n getManagers: () => Manager[];\n } = { getManagers: () => [new NetworkManager()] },\n) {\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const [ServerCacheProvider, controller] = createPersistedStore(\n options.getManagers(),\n );\n\n const nextProps = await next(props);\n\n return {\n ...nextProps,\n controller,\n app: <ServerCacheProvider>{nextProps.app}</ServerCacheProvider>,\n };\n };\n };\n}\n","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"@anansi/router\");","const __WEBPACK_NAMESPACE_OBJECT__ = require(\"history\");","import { Route, RouteProvider, RouteController } from '@anansi/router';\nimport React from 'react';\nimport { createMemoryHistory } from 'history';\n\nimport type { ResolveProps, ServerProps, CreateRouter } from './types';\n\ntype NeededProps = ResolveProps;\n\nexport default function routerSpout<ResolveWith>(options: {\n resolveWith?: any;\n useResolveWith: () => ResolveWith;\n createRouter: CreateRouter<ResolveWith>;\n}) {\n const createRouteComponent = (\n router: RouteController<Route<ResolveWith, any>>,\n ) =>\n function Router({ children }: { children: React.ReactNode }) {\n const resolveWith = options.useResolveWith();\n\n return (\n <RouteProvider router={router} resolveWith={resolveWith}>\n {children}\n </RouteProvider>\n );\n };\n\n return function <T extends NeededProps>(\n next: (props: ServerProps) => Promise<T>,\n ) {\n return async (props: ServerProps) => {\n const url = props.req.url || '';\n const router = options.createRouter(\n createMemoryHistory({ initialEntries: [url] }),\n );\n const matchedRoutes: Route<ResolveWith>[] = router.getMatchedRoutes(url);\n\n const nextProps = await next(props);\n\n const Router = createRouteComponent(router);\n return {\n ...nextProps,\n matchedRoutes,\n router,\n app: <Router>{nextProps.app}</Router>,\n };\n };\n };\n}\n","import { Route } from '@anansi/router';\n\nimport type { ResolveProps, ServerProps } from './types';\n\ntype NeededProps<RouteWith> = {\n matchedRoutes: Route<RouteWith>[];\n} & ResolveProps;\n\nexport default function prefetchSpout<F extends string>(field: F) {\n return function <RouteWith, T extends NeededProps<RouteWith>>(\n next: (props: ServerProps) => Promise<\n {\n [K in F]: RouteWith;\n } & T\n >,\n ) {\n return async (props: ServerProps) => {\n const nextProps = await next(props);\n\n try {\n const toFetch: Promise<unknown>[] = [];\n nextProps.matchedRoutes.forEach(route => {\n if (typeof route.resolveData === 'function') {\n toFetch.push(route.resolveData(nextProps[field], route));\n }\n });\n await Promise.all(toFetch);\n } catch (e) {\n console.error(e);\n }\n return nextProps;\n };\n };\n}\n","export { default as laySpouts } from './laySpouts';\nexport { default as documentSpout } from './spouts/document.server';\nexport { default as restHooksSpout } from './spouts/restHooks.server';\nexport { default as routerSpout } from './spouts/router.server';\nexport { default as prefetchSpout } from './spouts/prefetch.server';\n"],"names":[],"sourceRoot":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/scripts/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,IAAI,KAAK,EAAE,MAAM,SAAS,CAAC;AAC3C,OAAO,EAAE,OAAO,IAAI,QAAQ,EAAE,MAAM,kBAAkB,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
|
|
4
|
+
|
|
5
|
+
exports.__esModule = true;
|
|
6
|
+
exports.serve = exports.devServe = void 0;
|
|
7
|
+
|
|
8
|
+
var _serve = _interopRequireDefault(require("./serve"));
|
|
9
|
+
|
|
10
|
+
exports.serve = _serve.default;
|
|
11
|
+
|
|
12
|
+
var _startDevserver = _interopRequireDefault(require("./startDevserver"));
|
|
13
|
+
|
|
14
|
+
exports.devServe = _startDevserver.default;
|
|
15
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmlwdHMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgZGVmYXVsdCBhcyBzZXJ2ZSB9IGZyb20gJy4vc2VydmUnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBkZXZTZXJ2ZSB9IGZyb20gJy4vc3RhcnREZXZzZXJ2ZXInO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7Ozs7QUFDQSJ9
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serve.d.ts","sourceRoot":"","sources":["../../src/scripts/serve.ts"],"names":[],"mappings":";AAWA,OAAO,sBAAsB,CAAC;AAc9B,MAAM,CAAC,OAAO,UAAU,KAAK,CAAC,UAAU,EAAE,MAAM,QA0H/C"}
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
"use strict";
|
|
3
|
+
|
|
4
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
|
|
5
|
+
|
|
6
|
+
exports.__esModule = true;
|
|
7
|
+
exports.default = serve;
|
|
8
|
+
|
|
9
|
+
var _util = require("util");
|
|
10
|
+
|
|
11
|
+
var _fs = _interopRequireDefault(require("fs"));
|
|
12
|
+
|
|
13
|
+
var _path = _interopRequireDefault(require("path"));
|
|
14
|
+
|
|
15
|
+
var _express = _interopRequireDefault(require("express"));
|
|
16
|
+
|
|
17
|
+
var _ora = _interopRequireDefault(require("ora"));
|
|
18
|
+
|
|
19
|
+
var _compression = _interopRequireDefault(require("compression"));
|
|
20
|
+
|
|
21
|
+
require("cross-fetch/polyfill");
|
|
22
|
+
|
|
23
|
+
// run directly from node
|
|
24
|
+
if (require.main === module) {
|
|
25
|
+
const entrypoint = process.argv[2];
|
|
26
|
+
|
|
27
|
+
if (!entrypoint) {
|
|
28
|
+
console.log(`Usage: ${process.argv[0]} <server-entrypoint>`);
|
|
29
|
+
process.exit(-1);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
serve(entrypoint);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function serve(entrypoint) {
|
|
36
|
+
const PORT = process.env.PORT || 8080;
|
|
37
|
+
const loader = (0, _ora.default)('Initializing').start();
|
|
38
|
+
const manifestPath = getManifestPathFromWebpackconfig();
|
|
39
|
+
const readFile = (0, _util.promisify)(_fs.default.readFile);
|
|
40
|
+
let server;
|
|
41
|
+
|
|
42
|
+
function handleErrors(fn) {
|
|
43
|
+
return async function (req, res, next) {
|
|
44
|
+
try {
|
|
45
|
+
return await fn(req, res);
|
|
46
|
+
} catch (x) {
|
|
47
|
+
next(x);
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
} // Start the express server after the first compilation
|
|
51
|
+
|
|
52
|
+
|
|
53
|
+
function initializeApp(clientManifest) {
|
|
54
|
+
loader.info('Launching server');
|
|
55
|
+
|
|
56
|
+
if (!clientManifest) {
|
|
57
|
+
loader.fail('Manifest not found'); // TODO: handle more gracefully
|
|
58
|
+
|
|
59
|
+
process.exit(-1);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const wrappingApp = (0, _express.default)(); // eslint-disable-next-line
|
|
63
|
+
//@ts-ignore
|
|
64
|
+
|
|
65
|
+
wrappingApp.use((0, _compression.default)()); // ASSETS
|
|
66
|
+
|
|
67
|
+
const assetRoute = async (req, res) => {
|
|
68
|
+
var _req$url$substr, _req$url, _clientManifest$outpu;
|
|
69
|
+
|
|
70
|
+
const filename = (_req$url$substr = (_req$url = req.url) === null || _req$url === void 0 ? void 0 : _req$url.substr(process.env.WEBPACK_PUBLIC_PATH.length)) != null ? _req$url$substr : '';
|
|
71
|
+
|
|
72
|
+
const assetPath = _path.default.join((_clientManifest$outpu = clientManifest.outputPath) != null ? _clientManifest$outpu : '', filename);
|
|
73
|
+
|
|
74
|
+
try {
|
|
75
|
+
const fileContent = (await readFile(assetPath)).toString();
|
|
76
|
+
res.contentType(filename);
|
|
77
|
+
res.send(fileContent);
|
|
78
|
+
} catch (e) {
|
|
79
|
+
res.status(404);
|
|
80
|
+
res.send(e);
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
};
|
|
84
|
+
|
|
85
|
+
wrappingApp.get(`${process.env.WEBPACK_PUBLIC_PATH}*`, assetRoute); // SERVER SIDE RENDERING
|
|
86
|
+
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
87
|
+
|
|
88
|
+
const render = require(_path.default.join(process.cwd(), entrypoint)).default;
|
|
89
|
+
|
|
90
|
+
wrappingApp.get('/*', handleErrors(async function (req, res) {
|
|
91
|
+
if (req.url.endsWith('favicon.ico')) {
|
|
92
|
+
res.statusCode = 404;
|
|
93
|
+
res.setHeader('Content-type', 'text/html');
|
|
94
|
+
res.send('not found');
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
res.socket.on('error', error => {
|
|
99
|
+
console.error('Fatal', error);
|
|
100
|
+
});
|
|
101
|
+
await render(clientManifest, req, res);
|
|
102
|
+
}));
|
|
103
|
+
server = wrappingApp.listen(PORT, () => {
|
|
104
|
+
loader.info(`Listening at ${PORT}...`);
|
|
105
|
+
}).on('error', function (error) {
|
|
106
|
+
if (error.syscall !== 'listen') {
|
|
107
|
+
throw error;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
const isPipe = portOrPipe => Number.isNaN(portOrPipe);
|
|
111
|
+
|
|
112
|
+
const bind = isPipe(PORT) ? 'Pipe ' + PORT : 'Port ' + PORT;
|
|
113
|
+
|
|
114
|
+
switch (error.code) {
|
|
115
|
+
case 'EACCES':
|
|
116
|
+
loader.fail(bind + ' requires elevated privileges');
|
|
117
|
+
process.exit(1);
|
|
118
|
+
// eslint-disable-next-line no-fallthrough
|
|
119
|
+
|
|
120
|
+
case 'EADDRINUSE':
|
|
121
|
+
loader.fail(bind + ' is already in use');
|
|
122
|
+
process.exit(1);
|
|
123
|
+
// eslint-disable-next-line no-fallthrough
|
|
124
|
+
|
|
125
|
+
default:
|
|
126
|
+
throw error;
|
|
127
|
+
}
|
|
128
|
+
});
|
|
129
|
+
} // eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
initializeApp(require(manifestPath));
|
|
133
|
+
process.on('SIGINT', () => {
|
|
134
|
+
var _server;
|
|
135
|
+
|
|
136
|
+
loader.warn('Received SIGINT, devserver shutting down');
|
|
137
|
+
if (server) console.log('Closing server');
|
|
138
|
+
(_server = server) === null || _server === void 0 ? void 0 : _server.close(() => {
|
|
139
|
+
loader.info('Server closed');
|
|
140
|
+
});
|
|
141
|
+
process.exit(-1);
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function getManifestPathFromWebpackconfig() {
|
|
146
|
+
var _opts$filename, _webpackConfig$plugin, _webpackConfig$plugin2, _webpackConfig$plugin3, _webpackConfig$output, _webpackConfig$output2;
|
|
147
|
+
|
|
148
|
+
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
149
|
+
const webpackConfig = require(require.resolve( // TODO: use normal resolution algorithm to find webpack file
|
|
150
|
+
_path.default.join(process.cwd(), 'webpack.config')))({}, {
|
|
151
|
+
mode: 'production'
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
const manifestFilename = (_opts$filename = webpackConfig === null || webpackConfig === void 0 ? void 0 : (_webpackConfig$plugin = webpackConfig.plugins) === null || _webpackConfig$plugin === void 0 ? void 0 : (_webpackConfig$plugin2 = _webpackConfig$plugin.find(plugin => {
|
|
155
|
+
return plugin.constructor.name === 'StatsWriterPlugin';
|
|
156
|
+
})) === null || _webpackConfig$plugin2 === void 0 ? void 0 : (_webpackConfig$plugin3 = _webpackConfig$plugin2.opts) === null || _webpackConfig$plugin3 === void 0 ? void 0 : _webpackConfig$plugin3.filename) != null ? _opts$filename : 'manifest.json';
|
|
157
|
+
|
|
158
|
+
const manifestPath = _path.default.join((_webpackConfig$output = webpackConfig === null || webpackConfig === void 0 ? void 0 : (_webpackConfig$output2 = webpackConfig.output) === null || _webpackConfig$output2 === void 0 ? void 0 : _webpackConfig$output2.path) != null ? _webpackConfig$output : '', manifestFilename);
|
|
159
|
+
|
|
160
|
+
return manifestPath;
|
|
161
|
+
}
|
|
162
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["require","main","module","entrypoint","process","argv","console","log","exit","serve","PORT","env","loader","ora","start","manifestPath","getManifestPathFromWebpackconfig","readFile","promisify","diskFs","server","handleErrors","fn","req","res","next","x","initializeApp","clientManifest","info","fail","wrappingApp","express","use","compress","assetRoute","filename","url","substr","WEBPACK_PUBLIC_PATH","length","assetPath","path","join","outputPath","fileContent","toString","contentType","send","e","status","get","render","cwd","default","endsWith","statusCode","setHeader","socket","on","error","listen","syscall","isPipe","portOrPipe","Number","isNaN","bind","code","warn","close","webpackConfig","resolve","mode","manifestFilename","plugins","find","plugin","constructor","name","opts","output"],"sources":["../../src/scripts/serve.ts"],"sourcesContent":["#!/usr/bin/env node\n\nimport { promisify } from 'util';\nimport diskFs from 'fs';\nimport path from 'path';\nimport webpack, { web } from 'webpack';\nimport { Server, IncomingMessage, ServerResponse } from 'http';\nimport express, { NextFunction } from 'express';\nimport ora from 'ora';\nimport compress from 'compression';\n\nimport 'cross-fetch/polyfill';\nimport { Render } from './types';\n\n// run directly from node\nif (require.main === module) {\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 function serve(entrypoint: string) {\n  const PORT = process.env.PORT || 8080;\n\n  const loader = ora('Initializing').start();\n\n  const manifestPath = getManifestPathFromWebpackconfig();\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  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    const assetRoute = async (req: Request | IncomingMessage, res: any) => {\n      const filename =\n        req.url?.substr((process.env.WEBPACK_PUBLIC_PATH as string).length) ??\n        '';\n      const assetPath = path.join(clientManifest.outputPath ?? '', filename);\n\n      try {\n        const fileContent = (await readFile(assetPath)).toString();\n        res.contentType(filename);\n        res.send(fileContent);\n      } catch (e) {\n        res.status(404);\n        res.send(e);\n        return;\n      }\n    };\n    wrappingApp.get(`${process.env.WEBPACK_PUBLIC_PATH}*`, assetRoute);\n\n    // SERVER SIDE RENDERING\n    // eslint-disable-next-line @typescript-eslint/no-var-requires\n    const render: Render = require(path.join(\n      process.cwd(),\n      entrypoint,\n    )).default;\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  // eslint-disable-next-line @typescript-eslint/no-var-requires\n  initializeApp(require(manifestPath));\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  // eslint-disable-next-line @typescript-eslint/no-var-requires\n  const webpackConfig: webpack.Configuration = require(require.resolve(\n    // TODO: use normal resolution algorithm to find webpack file\n    path.join(process.cwd(), 'webpack.config'),\n  ))({}, { mode: 'production' });\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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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";AAiBA,OAAO,sBAAsB,CAAC"}
|
|
1
|
+
{"version":3,"file":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";AAiBA,OAAO,sBAAsB,CAAC;AAiB9B,MAAM,CAAC,OAAO,UAAU,cAAc,CACpC,UAAU,EAAE,MAAM,EAClB,GAAG,GAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAM,QA0NlC"}
|