@anansi/core 0.22.2 → 0.22.4
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 +16 -0
- package/README.md +340 -44
- package/lib/scripts/proxyUtils.d.ts +17 -0
- package/lib/scripts/proxyUtils.d.ts.map +1 -0
- package/lib/scripts/proxyUtils.js +26 -0
- package/lib/scripts/scripts/proxyUtils.js +26 -0
- package/lib/scripts/scripts/startDevserver.js +3 -2
- package/lib/scripts/startDevserver.d.ts.map +1 -1
- package/lib/scripts/startDevserver.js +3 -2
- package/package.json +2 -2
- package/src/scripts/__tests__/proxyUtils.test.ts +89 -0
- package/src/scripts/proxyUtils.ts +35 -0
- package/src/scripts/startDevserver.ts +2 -3
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,22 @@
|
|
|
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.22.4](/github.com/ntucker/anansi/compare/@anansi/core@0.22.3...@anansi/core@0.22.4) (2026-01-10)
|
|
7
|
+
|
|
8
|
+
### 📦 Package
|
|
9
|
+
|
|
10
|
+
* Update all non-major dependencies ([#2930](/github.com/ntucker/anansi/issues/2930)) ([6162e2b](/github.com/ntucker/anansi/commit/6162e2bfff24989e670d57e62ec93d6a1d42d839))
|
|
11
|
+
|
|
12
|
+
## [0.22.3](/github.com/ntucker/anansi/compare/@anansi/core@0.22.2...@anansi/core@0.22.3) (2026-01-02)
|
|
13
|
+
|
|
14
|
+
### 💅 Enhancement
|
|
15
|
+
|
|
16
|
+
* Improve webpack proxy handling ([0413bc6](/github.com/ntucker/anansi/commit/0413bc6ff3ea9eb107d4ca5259c21e1b7efef29c))
|
|
17
|
+
|
|
18
|
+
### 📝 Documentation
|
|
19
|
+
|
|
20
|
+
* Update readme ([b42d500](/github.com/ntucker/anansi/commit/b42d50069157979ac2c21fce92c4750f3219461a))
|
|
21
|
+
|
|
6
22
|
## [0.22.2](/github.com/ntucker/anansi/compare/@anansi/core@0.22.1...@anansi/core@0.22.2) (2025-12-31)
|
|
7
23
|
|
|
8
24
|
### 🐛 Bug Fix
|
package/README.md
CHANGED
|
@@ -1,7 +1,5 @@
|
|
|
1
1
|
# @anansi/core
|
|
2
2
|
|
|
3
|
-
<!--[](https://circleci.com/gh/notwillk/pojo-router)-->
|
|
4
|
-
|
|
5
3
|
[](https://www.npmjs.com/package/@anansi/core)
|
|
6
4
|
[](https://bundlephobia.com/result?p=@anansi/core)
|
|
7
5
|
[](https://www.npmjs.com/package/@anansi/core)
|
|
@@ -12,106 +10,404 @@
|
|
|
12
10
|
> Out came the sun, and dried up all the rain,
|
|
13
11
|
> and the itsy bitsy spider went up the spout again
|
|
14
12
|
|
|
15
|
-
|
|
13
|
+
React 19 framework with streaming SSR support, built on a composable "spouts" architecture.
|
|
14
|
+
|
|
15
|
+
## Installation
|
|
16
|
+
|
|
17
|
+
```bash
|
|
18
|
+
yarn add @anansi/core
|
|
19
|
+
```
|
|
20
|
+
|
|
21
|
+
## Quick Start
|
|
22
|
+
|
|
23
|
+
Anansi uses a dual-entry pattern for SSR: one entry for the server and one for the client.
|
|
16
24
|
|
|
17
25
|
```bash
|
|
18
|
-
yarn start-anansi./src/index.tsx
|
|
26
|
+
yarn start-anansi ./src/index.tsx
|
|
19
27
|
```
|
|
20
28
|
|
|
21
|
-
This
|
|
29
|
+
This automatically uses `./src/index.tsx` for the client and `./src/index.server.tsx` for the server.
|
|
30
|
+
|
|
31
|
+
## Concepts
|
|
32
|
+
|
|
33
|
+
### Spouts
|
|
34
|
+
|
|
35
|
+
Spouts are composable middleware for building React applications. They handle concerns like routing, data fetching, document structure, and more.
|
|
22
36
|
|
|
23
|
-
|
|
37
|
+
- **Server**: `laySpouts()` - Lays out the spouts for SSR, streaming React to the response
|
|
38
|
+
- **Client**: `floodSpouts()` - Hydrates the application on the client
|
|
39
|
+
|
|
40
|
+
The spout pattern allows you to compose functionality in a declarative, nested structure where each spout can:
|
|
41
|
+
- Inject props to downstream spouts
|
|
42
|
+
- Wrap the rendered application with providers
|
|
43
|
+
- Serialize data for hydration
|
|
44
|
+
|
|
45
|
+
## Entry Points
|
|
46
|
+
|
|
47
|
+
### Client Entry (`index.tsx`)
|
|
24
48
|
|
|
25
49
|
```tsx
|
|
26
50
|
import { useController } from '@data-client/react';
|
|
27
51
|
import {
|
|
28
|
-
|
|
52
|
+
floodSpouts,
|
|
29
53
|
documentSpout,
|
|
30
54
|
dataClientSpout,
|
|
31
|
-
prefetchSpout,
|
|
32
55
|
routerSpout,
|
|
33
56
|
JSONSpout,
|
|
34
57
|
appSpout,
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
import app from 'app';
|
|
58
|
+
navigatorSpout,
|
|
59
|
+
} from '@anansi/core';
|
|
38
60
|
|
|
61
|
+
import App from './App';
|
|
39
62
|
import { createRouter } from './routing';
|
|
40
63
|
|
|
41
|
-
const spouts =
|
|
42
|
-
|
|
43
|
-
|
|
64
|
+
const spouts = documentSpout({ title: 'My App' })(
|
|
65
|
+
JSONSpout()(
|
|
66
|
+
navigatorSpout()(
|
|
44
67
|
dataClientSpout()(
|
|
45
68
|
routerSpout({ useResolveWith: useController, createRouter })(
|
|
46
|
-
appSpout(
|
|
69
|
+
appSpout(<App />),
|
|
47
70
|
),
|
|
48
71
|
),
|
|
49
72
|
),
|
|
50
73
|
),
|
|
51
74
|
);
|
|
52
75
|
|
|
53
|
-
|
|
76
|
+
floodSpouts(spouts);
|
|
54
77
|
```
|
|
55
78
|
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
<details open><summary>index.tsx</summary>
|
|
79
|
+
### Server Entry (`index.server.tsx`)
|
|
59
80
|
|
|
60
81
|
```tsx
|
|
61
82
|
import { useController } from '@data-client/react';
|
|
62
83
|
import {
|
|
63
|
-
|
|
84
|
+
laySpouts,
|
|
64
85
|
documentSpout,
|
|
65
86
|
dataClientSpout,
|
|
87
|
+
prefetchSpout,
|
|
66
88
|
routerSpout,
|
|
67
89
|
JSONSpout,
|
|
68
90
|
appSpout,
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
import app from 'app';
|
|
91
|
+
navigatorSpout,
|
|
92
|
+
} from '@anansi/core/server';
|
|
72
93
|
|
|
94
|
+
import App from './App';
|
|
73
95
|
import { createRouter } from './routing';
|
|
74
96
|
|
|
75
|
-
const
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
97
|
+
const spouts = prefetchSpout('controller')(
|
|
98
|
+
documentSpout({ title: 'My App' })(
|
|
99
|
+
JSONSpout()(
|
|
100
|
+
navigatorSpout()(
|
|
101
|
+
dataClientSpout()(
|
|
102
|
+
routerSpout({ useResolveWith: useController, createRouter })(
|
|
103
|
+
appSpout(<App />),
|
|
104
|
+
),
|
|
105
|
+
),
|
|
82
106
|
),
|
|
83
107
|
),
|
|
84
108
|
),
|
|
85
109
|
);
|
|
86
110
|
|
|
87
|
-
|
|
111
|
+
export default laySpouts(spouts);
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
## Spouts Reference
|
|
115
|
+
|
|
116
|
+
### appSpout
|
|
117
|
+
|
|
118
|
+
The innermost spout that wraps your React application.
|
|
119
|
+
|
|
120
|
+
```tsx
|
|
121
|
+
import { appSpout } from '@anansi/core';
|
|
122
|
+
|
|
123
|
+
appSpout(<App />)
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
### documentSpout
|
|
127
|
+
|
|
128
|
+
Generates the HTML document structure with proper asset loading.
|
|
129
|
+
|
|
130
|
+
```tsx
|
|
131
|
+
import { documentSpout } from '@anansi/core';
|
|
132
|
+
|
|
133
|
+
documentSpout({
|
|
134
|
+
title: 'My App', // Page title
|
|
135
|
+
head?: ReactNode, // Additional head elements
|
|
136
|
+
lang?: string, // HTML lang attribute (default: undefined)
|
|
137
|
+
rootId?: string, // Root element ID (default: 'anansi-root')
|
|
138
|
+
charSet?: string, // Character set (default: 'utf-8')
|
|
139
|
+
csPolicy?: CSPolicy, // Content Security Policy
|
|
140
|
+
})
|
|
141
|
+
```
|
|
142
|
+
|
|
143
|
+
#### Content Security Policy
|
|
144
|
+
|
|
145
|
+
The `csPolicy` option configures CSP headers. In production, it sets `Content-Security-Policy`; in development, it uses `Content-Security-Policy-Report-Only`.
|
|
146
|
+
|
|
147
|
+
```tsx
|
|
148
|
+
const csPolicy = {
|
|
149
|
+
'base-uri': "'self'",
|
|
150
|
+
'object-src': "'none'",
|
|
151
|
+
'script-src': ["'self'", "'unsafe-inline'"],
|
|
152
|
+
'style-src': ["'unsafe-inline'", "'self'"],
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
documentSpout({ title: 'My App', csPolicy })
|
|
88
156
|
```
|
|
89
157
|
|
|
90
|
-
|
|
158
|
+
Nonces are automatically injected into `script-src` for inline scripts.
|
|
159
|
+
|
|
160
|
+
### routerSpout
|
|
91
161
|
|
|
92
|
-
|
|
162
|
+
Integrates [@anansi/router](https://github.com/ntucker/anansi/tree/master/packages/router) for client-side navigation.
|
|
93
163
|
|
|
94
|
-
|
|
164
|
+
```tsx
|
|
165
|
+
import { routerSpout } from '@anansi/core';
|
|
166
|
+
|
|
167
|
+
routerSpout({
|
|
168
|
+
useResolveWith: () => any, // Hook returning data passed to route resolvers
|
|
169
|
+
createRouter: (history) => RouteController, // Router factory function
|
|
170
|
+
onChange?: (update, callback) => void, // Client-only: navigation callback
|
|
171
|
+
})
|
|
172
|
+
```
|
|
95
173
|
|
|
96
|
-
|
|
174
|
+
**Provides to downstream spouts:**
|
|
175
|
+
- `matchedRoutes` - Array of matched route objects
|
|
176
|
+
- `router` - RouteController instance
|
|
177
|
+
- `searchParams` - URLSearchParams from the current URL
|
|
97
178
|
|
|
179
|
+
### dataClientSpout
|
|
180
|
+
|
|
181
|
+
Integrates [@data-client/react](https://dataclient.io) for data fetching with automatic SSR hydration.
|
|
182
|
+
|
|
183
|
+
```tsx
|
|
184
|
+
import { dataClientSpout } from '@anansi/core';
|
|
185
|
+
|
|
186
|
+
dataClientSpout({
|
|
187
|
+
getManagers?: () => Manager[], // Custom managers (default: [NetworkManager])
|
|
188
|
+
})
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
Server-side, it creates a persisted store and serializes state for hydration. Client-side, it hydrates from the serialized state.
|
|
192
|
+
|
|
193
|
+
### JSONSpout
|
|
194
|
+
|
|
195
|
+
Serializes data from upstream spouts into `<script type="application/json">` tags for hydration.
|
|
196
|
+
|
|
197
|
+
```tsx
|
|
198
|
+
import { JSONSpout } from '@anansi/core';
|
|
199
|
+
|
|
200
|
+
JSONSpout({
|
|
201
|
+
id?: string, // Base ID for script tags (default: 'anansi-json')
|
|
202
|
+
})
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
### navigatorSpout
|
|
206
|
+
|
|
207
|
+
Provides browser navigator-like properties (language preferences) that work on both server and client.
|
|
208
|
+
|
|
209
|
+
```tsx
|
|
210
|
+
import { navigatorSpout, useNavigator } from '@anansi/core';
|
|
211
|
+
|
|
212
|
+
// In spouts config
|
|
213
|
+
navigatorSpout()
|
|
214
|
+
|
|
215
|
+
// In components
|
|
216
|
+
function MyComponent() {
|
|
217
|
+
const { language, languages } = useNavigator();
|
|
218
|
+
// language: string - Primary language (e.g., 'en-US')
|
|
219
|
+
// languages: readonly string[] - All accepted languages by preference
|
|
220
|
+
}
|
|
221
|
+
```
|
|
98
222
|
|
|
99
|
-
|
|
223
|
+
### prefetchSpout (Server Only)
|
|
224
|
+
|
|
225
|
+
Prefetches route data before rendering. Must wrap `routerSpout`.
|
|
226
|
+
|
|
227
|
+
```tsx
|
|
228
|
+
import { prefetchSpout } from '@anansi/core/server';
|
|
229
|
+
|
|
230
|
+
// 'controller' is the field name from dataClientSpout to use for resolving routes
|
|
231
|
+
prefetchSpout('controller')(
|
|
232
|
+
// ... other spouts including routerSpout
|
|
233
|
+
)
|
|
234
|
+
```
|
|
235
|
+
|
|
236
|
+
This calls `route.resolveData()` and `route.component.preload()` for all matched routes before SSR.
|
|
237
|
+
|
|
238
|
+
### antdSpout (Ant Design)
|
|
239
|
+
|
|
240
|
+
Integrates Ant Design's CSS-in-JS with SSR support.
|
|
241
|
+
|
|
242
|
+
```tsx
|
|
243
|
+
// Client
|
|
244
|
+
import { antdSpout } from '@anansi/core/antd';
|
|
245
|
+
|
|
246
|
+
// Server
|
|
247
|
+
import { antdSpout } from '@anansi/core/antd/server';
|
|
248
|
+
|
|
249
|
+
antdSpout()
|
|
250
|
+
```
|
|
251
|
+
|
|
252
|
+
## Core Functions
|
|
253
|
+
|
|
254
|
+
### laySpouts (Server)
|
|
255
|
+
|
|
256
|
+
Wraps spouts for server-side rendering with streaming support.
|
|
257
|
+
|
|
258
|
+
```tsx
|
|
259
|
+
import { laySpouts } from '@anansi/core/server';
|
|
260
|
+
|
|
261
|
+
export default laySpouts(spouts, {
|
|
262
|
+
timeoutMS?: number, // SSR timeout (default: 10000)
|
|
263
|
+
onError?: (error) => void, // Error callback
|
|
264
|
+
});
|
|
265
|
+
```
|
|
266
|
+
|
|
267
|
+
Returns a render function compatible with Express: `(clientManifest, req, res) => Promise<void>`
|
|
268
|
+
|
|
269
|
+
### floodSpouts (Client)
|
|
270
|
+
|
|
271
|
+
Hydrates the application on the client.
|
|
272
|
+
|
|
273
|
+
```tsx
|
|
274
|
+
import { floodSpouts } from '@anansi/core';
|
|
275
|
+
|
|
276
|
+
floodSpouts(spouts, {
|
|
277
|
+
rootId?: string, // Root element ID (default: 'anansi-root')
|
|
278
|
+
});
|
|
279
|
+
```
|
|
280
|
+
|
|
281
|
+
## CLI Commands
|
|
282
|
+
|
|
283
|
+
### start-anansi
|
|
284
|
+
|
|
285
|
+
Development server with hot module replacement for both client and server.
|
|
286
|
+
|
|
287
|
+
```bash
|
|
288
|
+
yarn start-anansi ./src/index.tsx
|
|
289
|
+
```
|
|
290
|
+
|
|
291
|
+
Features:
|
|
292
|
+
- Dual webpack compilation (client + server)
|
|
293
|
+
- In-memory filesystem for fast rebuilds
|
|
294
|
+
- Hot reloading for both bundles
|
|
295
|
+
- Automatic SSR on all non-asset routes
|
|
296
|
+
- Proxy support from webpack devServer config
|
|
297
|
+
|
|
298
|
+
### serve-anansi
|
|
299
|
+
|
|
300
|
+
Production server for pre-built applications.
|
|
301
|
+
|
|
302
|
+
```bash
|
|
303
|
+
yarn serve-anansi ./dist-server/App.js
|
|
304
|
+
```
|
|
305
|
+
|
|
306
|
+
## Scripts API
|
|
307
|
+
|
|
308
|
+
For programmatic usage:
|
|
309
|
+
|
|
310
|
+
```ts
|
|
311
|
+
import { serve, devServe } from '@anansi/core/scripts';
|
|
312
|
+
```
|
|
100
313
|
|
|
101
314
|
### serve(entry, options?)
|
|
102
315
|
|
|
316
|
+
Starts a production server.
|
|
317
|
+
|
|
318
|
+
```ts
|
|
319
|
+
serve('./dist-server/App.js', {
|
|
320
|
+
serveAssets?: boolean, // Serve static assets (default: false)
|
|
321
|
+
serveProxy?: boolean, // Enable proxy from webpack config (default: false)
|
|
322
|
+
});
|
|
323
|
+
```
|
|
324
|
+
|
|
325
|
+
**Environment Variables:**
|
|
326
|
+
- `PORT` - Server port (default: 8080)
|
|
327
|
+
- `WEBPACK_PUBLIC_PATH` - Public path for assets
|
|
328
|
+
|
|
329
|
+
#### Options
|
|
330
|
+
|
|
331
|
+
| Option | Type | Description |
|
|
332
|
+
|--------|------|-------------|
|
|
333
|
+
| `serveAssets` | `boolean` | Serve static assets from the build output. Useful for local validation; use a dedicated HTTP server in production. |
|
|
334
|
+
| `serveProxy` | `boolean` | Enable proxying based on webpack devServer config. Useful for local validation; use a reverse proxy in production. |
|
|
335
|
+
|
|
336
|
+
### devServe(entry, env?)
|
|
337
|
+
|
|
338
|
+
Starts a development server with HMR.
|
|
339
|
+
|
|
340
|
+
```ts
|
|
341
|
+
devServe('./src/index.tsx', {
|
|
342
|
+
// Additional env variables passed to webpack config
|
|
343
|
+
});
|
|
344
|
+
```
|
|
345
|
+
|
|
346
|
+
## Types
|
|
347
|
+
|
|
348
|
+
### ServerProps
|
|
349
|
+
|
|
350
|
+
Props available to server-side spouts:
|
|
351
|
+
|
|
352
|
+
```ts
|
|
353
|
+
interface ServerProps {
|
|
354
|
+
req: Request | IncomingMessage;
|
|
355
|
+
res: Response | ServerResponse;
|
|
356
|
+
clientManifest: StatsCompilation;
|
|
357
|
+
nonce: string;
|
|
358
|
+
}
|
|
359
|
+
```
|
|
360
|
+
|
|
361
|
+
### Spout Types
|
|
362
|
+
|
|
103
363
|
```ts
|
|
104
|
-
import {
|
|
364
|
+
import type { Spout, ServerProps, NavigatorProperties } from '@anansi/core';
|
|
365
|
+
```
|
|
366
|
+
|
|
367
|
+
### CSPolicy
|
|
368
|
+
|
|
369
|
+
Content Security Policy configuration:
|
|
105
370
|
|
|
106
|
-
|
|
371
|
+
```ts
|
|
372
|
+
interface CSPolicy {
|
|
373
|
+
[directive: string]: string | string[];
|
|
374
|
+
}
|
|
107
375
|
```
|
|
108
376
|
|
|
109
|
-
|
|
377
|
+
## Exports
|
|
378
|
+
|
|
379
|
+
### `@anansi/core`
|
|
380
|
+
|
|
381
|
+
Client-side exports:
|
|
382
|
+
|
|
383
|
+
- `floodSpouts` - Hydrate the application
|
|
384
|
+
- `documentSpout` - Document structure
|
|
385
|
+
- `dataClientSpout` - Data Client integration
|
|
386
|
+
- `routerSpout` - Router integration
|
|
387
|
+
- `JSONSpout` - JSON serialization for hydration
|
|
388
|
+
- `appSpout` - Application wrapper
|
|
389
|
+
- `navigatorSpout` - Navigator properties
|
|
390
|
+
- `useNavigator` - Hook for navigator properties
|
|
391
|
+
|
|
392
|
+
### `@anansi/core/server`
|
|
393
|
+
|
|
394
|
+
Server-side exports (all client exports plus):
|
|
395
|
+
|
|
396
|
+
- `laySpouts` - SSR render function
|
|
397
|
+
- `prefetchSpout` - Route data prefetching
|
|
398
|
+
- `CSPolicy` - CSP type
|
|
399
|
+
|
|
400
|
+
### `@anansi/core/scripts`
|
|
401
|
+
|
|
402
|
+
Build/dev scripts:
|
|
403
|
+
|
|
404
|
+
- `serve` - Production server
|
|
405
|
+
- `devServe` - Development server
|
|
406
|
+
|
|
407
|
+
### `@anansi/core/antd`
|
|
110
408
|
|
|
111
|
-
|
|
112
|
-
typically want to use a dedicated HTTP server for static assets in production.
|
|
409
|
+
Ant Design integration (client-side).
|
|
113
410
|
|
|
114
|
-
|
|
411
|
+
### `@anansi/core/antd/server`
|
|
115
412
|
|
|
116
|
-
|
|
117
|
-
In production it is much more performant to use a separate reverse proxy.
|
|
413
|
+
Ant Design integration (server-side).
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import type { ProxyConfigArray } from 'webpack-dev-server';
|
|
2
|
+
/**
|
|
3
|
+
* Extracts route patterns from webpack-dev-server proxy configuration.
|
|
4
|
+
*
|
|
5
|
+
* Handles the webpack-dev-server proxy array format:
|
|
6
|
+
* ```
|
|
7
|
+
* proxy: [
|
|
8
|
+
* { context: ['/api'], target: 'http://localhost:3000' },
|
|
9
|
+
* { context: '/ws', target: 'http://localhost:3001' },
|
|
10
|
+
* { path: '/legacy', target: 'http://localhost:3002' },
|
|
11
|
+
* ]
|
|
12
|
+
* ```
|
|
13
|
+
*
|
|
14
|
+
* @see https://webpack.js.org/configuration/dev-server/#devserverproxy
|
|
15
|
+
*/
|
|
16
|
+
export declare function extractProxyRoutes(proxy: ProxyConfigArray | undefined): string[];
|
|
17
|
+
//# sourceMappingURL=proxyUtils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"proxyUtils.d.ts","sourceRoot":"","sources":["../../src/scripts/proxyUtils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAE3D;;;;;;;;;;;;;GAaG;AACH,wBAAgB,kBAAkB,CAChC,KAAK,EAAE,gBAAgB,GAAG,SAAS,GAClC,MAAM,EAAE,CAgBV"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Extracts route patterns from webpack-dev-server proxy configuration.
|
|
3
|
+
*
|
|
4
|
+
* Handles the webpack-dev-server proxy array format:
|
|
5
|
+
* ```
|
|
6
|
+
* proxy: [
|
|
7
|
+
* { context: ['/api'], target: 'http://localhost:3000' },
|
|
8
|
+
* { context: '/ws', target: 'http://localhost:3001' },
|
|
9
|
+
* { path: '/legacy', target: 'http://localhost:3002' },
|
|
10
|
+
* ]
|
|
11
|
+
* ```
|
|
12
|
+
*
|
|
13
|
+
* @see https://webpack.js.org/configuration/dev-server/#devserverproxy
|
|
14
|
+
*/
|
|
15
|
+
export function extractProxyRoutes(proxy) {
|
|
16
|
+
if (!proxy) return [];
|
|
17
|
+
return proxy.filter(item => typeof item === 'object' && item !== null).flatMap(item => {
|
|
18
|
+
// webpack-dev-server proxy supports both 'context' and 'path' properties
|
|
19
|
+
// and each can be a string or an array of strings
|
|
20
|
+
const context = item.context ?? item.path;
|
|
21
|
+
if (Array.isArray(context)) return context;
|
|
22
|
+
if (typeof context === 'string') return [context];
|
|
23
|
+
return [];
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleHRyYWN0UHJveHlSb3V0ZXMiLCJwcm94eSIsImZpbHRlciIsIml0ZW0iLCJmbGF0TWFwIiwiY29udGV4dCIsInBhdGgiLCJBcnJheSIsImlzQXJyYXkiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvc2NyaXB0cy9wcm94eVV0aWxzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgUHJveHlDb25maWdBcnJheSB9IGZyb20gJ3dlYnBhY2stZGV2LXNlcnZlcic7XG5cbi8qKlxuICogRXh0cmFjdHMgcm91dGUgcGF0dGVybnMgZnJvbSB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgY29uZmlndXJhdGlvbi5cbiAqXG4gKiBIYW5kbGVzIHRoZSB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgYXJyYXkgZm9ybWF0OlxuICogYGBgXG4gKiBwcm94eTogW1xuICogICB7IGNvbnRleHQ6IFsnL2FwaSddLCB0YXJnZXQ6ICdodHRwOi8vbG9jYWxob3N0OjMwMDAnIH0sXG4gKiAgIHsgY29udGV4dDogJy93cycsIHRhcmdldDogJ2h0dHA6Ly9sb2NhbGhvc3Q6MzAwMScgfSxcbiAqICAgeyBwYXRoOiAnL2xlZ2FjeScsIHRhcmdldDogJ2h0dHA6Ly9sb2NhbGhvc3Q6MzAwMicgfSxcbiAqIF1cbiAqIGBgYFxuICpcbiAqIEBzZWUgaHR0cHM6Ly93ZWJwYWNrLmpzLm9yZy9jb25maWd1cmF0aW9uL2Rldi1zZXJ2ZXIvI2RldnNlcnZlcnByb3h5XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBleHRyYWN0UHJveHlSb3V0ZXMoXG4gIHByb3h5OiBQcm94eUNvbmZpZ0FycmF5IHwgdW5kZWZpbmVkLFxuKTogc3RyaW5nW10ge1xuICBpZiAoIXByb3h5KSByZXR1cm4gW107XG5cbiAgcmV0dXJuIHByb3h5XG4gICAgLmZpbHRlcihcbiAgICAgIChpdGVtKTogaXRlbSBpcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PlxuICAgICAgICB0eXBlb2YgaXRlbSA9PT0gJ29iamVjdCcgJiYgaXRlbSAhPT0gbnVsbCxcbiAgICApXG4gICAgLmZsYXRNYXAoaXRlbSA9PiB7XG4gICAgICAvLyB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgc3VwcG9ydHMgYm90aCAnY29udGV4dCcgYW5kICdwYXRoJyBwcm9wZXJ0aWVzXG4gICAgICAvLyBhbmQgZWFjaCBjYW4gYmUgYSBzdHJpbmcgb3IgYW4gYXJyYXkgb2Ygc3RyaW5nc1xuICAgICAgY29uc3QgY29udGV4dCA9IGl0ZW0uY29udGV4dCA/PyBpdGVtLnBhdGg7XG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShjb250ZXh0KSkgcmV0dXJuIGNvbnRleHQgYXMgc3RyaW5nW107XG4gICAgICBpZiAodHlwZW9mIGNvbnRleHQgPT09ICdzdHJpbmcnKSByZXR1cm4gW2NvbnRleHRdO1xuICAgICAgcmV0dXJuIFtdO1xuICAgIH0pO1xufVxuIl0sIm1hcHBpbmdzIjoiQUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFTQSxrQkFBa0JBLENBQ2hDQyxLQUFtQyxFQUN6QjtFQUNWLElBQUksQ0FBQ0EsS0FBSyxFQUFFLE9BQU8sRUFBRTtFQUVyQixPQUFPQSxLQUFLLENBQ1RDLE1BQU0sQ0FDSkMsSUFBSSxJQUNILE9BQU9BLElBQUksS0FBSyxRQUFRLElBQUlBLElBQUksS0FBSyxJQUN6QyxDQUFDLENBQ0FDLE9BQU8sQ0FBQ0QsSUFBSSxJQUFJO0lBQ2Y7SUFDQTtJQUNBLE1BQU1FLE9BQU8sR0FBR0YsSUFBSSxDQUFDRSxPQUFPLElBQUlGLElBQUksQ0FBQ0csSUFBSTtJQUN6QyxJQUFJQyxLQUFLLENBQUNDLE9BQU8sQ0FBQ0gsT0FBTyxDQUFDLEVBQUUsT0FBT0EsT0FBTztJQUMxQyxJQUFJLE9BQU9BLE9BQU8sS0FBSyxRQUFRLEVBQUUsT0FBTyxDQUFDQSxPQUFPLENBQUM7SUFDakQsT0FBTyxFQUFFO0VBQ1gsQ0FBQyxDQUFDO0FBQ04iLCJpZ25vcmVMaXN0IjpbXX0=
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Extracts route patterns from webpack-dev-server proxy configuration.
|
|
3
|
+
*
|
|
4
|
+
* Handles the webpack-dev-server proxy array format:
|
|
5
|
+
* ```
|
|
6
|
+
* proxy: [
|
|
7
|
+
* { context: ['/api'], target: 'http://localhost:3000' },
|
|
8
|
+
* { context: '/ws', target: 'http://localhost:3001' },
|
|
9
|
+
* { path: '/legacy', target: 'http://localhost:3002' },
|
|
10
|
+
* ]
|
|
11
|
+
* ```
|
|
12
|
+
*
|
|
13
|
+
* @see https://webpack.js.org/configuration/dev-server/#devserverproxy
|
|
14
|
+
*/
|
|
15
|
+
export function extractProxyRoutes(proxy) {
|
|
16
|
+
if (!proxy) return [];
|
|
17
|
+
return proxy.filter(item => typeof item === 'object' && item !== null).flatMap(item => {
|
|
18
|
+
// webpack-dev-server proxy supports both 'context' and 'path' properties
|
|
19
|
+
// and each can be a string or an array of strings
|
|
20
|
+
const context = item.context ?? item.path;
|
|
21
|
+
if (Array.isArray(context)) return context;
|
|
22
|
+
if (typeof context === 'string') return [context];
|
|
23
|
+
return [];
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleHRyYWN0UHJveHlSb3V0ZXMiLCJwcm94eSIsImZpbHRlciIsIml0ZW0iLCJmbGF0TWFwIiwiY29udGV4dCIsInBhdGgiLCJBcnJheSIsImlzQXJyYXkiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2NyaXB0cy9wcm94eVV0aWxzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgUHJveHlDb25maWdBcnJheSB9IGZyb20gJ3dlYnBhY2stZGV2LXNlcnZlcic7XG5cbi8qKlxuICogRXh0cmFjdHMgcm91dGUgcGF0dGVybnMgZnJvbSB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgY29uZmlndXJhdGlvbi5cbiAqXG4gKiBIYW5kbGVzIHRoZSB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgYXJyYXkgZm9ybWF0OlxuICogYGBgXG4gKiBwcm94eTogW1xuICogICB7IGNvbnRleHQ6IFsnL2FwaSddLCB0YXJnZXQ6ICdodHRwOi8vbG9jYWxob3N0OjMwMDAnIH0sXG4gKiAgIHsgY29udGV4dDogJy93cycsIHRhcmdldDogJ2h0dHA6Ly9sb2NhbGhvc3Q6MzAwMScgfSxcbiAqICAgeyBwYXRoOiAnL2xlZ2FjeScsIHRhcmdldDogJ2h0dHA6Ly9sb2NhbGhvc3Q6MzAwMicgfSxcbiAqIF1cbiAqIGBgYFxuICpcbiAqIEBzZWUgaHR0cHM6Ly93ZWJwYWNrLmpzLm9yZy9jb25maWd1cmF0aW9uL2Rldi1zZXJ2ZXIvI2RldnNlcnZlcnByb3h5XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBleHRyYWN0UHJveHlSb3V0ZXMoXG4gIHByb3h5OiBQcm94eUNvbmZpZ0FycmF5IHwgdW5kZWZpbmVkLFxuKTogc3RyaW5nW10ge1xuICBpZiAoIXByb3h5KSByZXR1cm4gW107XG5cbiAgcmV0dXJuIHByb3h5XG4gICAgLmZpbHRlcihcbiAgICAgIChpdGVtKTogaXRlbSBpcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PlxuICAgICAgICB0eXBlb2YgaXRlbSA9PT0gJ29iamVjdCcgJiYgaXRlbSAhPT0gbnVsbCxcbiAgICApXG4gICAgLmZsYXRNYXAoaXRlbSA9PiB7XG4gICAgICAvLyB3ZWJwYWNrLWRldi1zZXJ2ZXIgcHJveHkgc3VwcG9ydHMgYm90aCAnY29udGV4dCcgYW5kICdwYXRoJyBwcm9wZXJ0aWVzXG4gICAgICAvLyBhbmQgZWFjaCBjYW4gYmUgYSBzdHJpbmcgb3IgYW4gYXJyYXkgb2Ygc3RyaW5nc1xuICAgICAgY29uc3QgY29udGV4dCA9IGl0ZW0uY29udGV4dCA/PyBpdGVtLnBhdGg7XG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShjb250ZXh0KSkgcmV0dXJuIGNvbnRleHQgYXMgc3RyaW5nW107XG4gICAgICBpZiAodHlwZW9mIGNvbnRleHQgPT09ICdzdHJpbmcnKSByZXR1cm4gW2NvbnRleHRdO1xuICAgICAgcmV0dXJuIFtdO1xuICAgIH0pO1xufVxuIl0sIm1hcHBpbmdzIjoiQUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFTQSxrQkFBa0JBLENBQ2hDQyxLQUFtQyxFQUN6QjtFQUNWLElBQUksQ0FBQ0EsS0FBSyxFQUFFLE9BQU8sRUFBRTtFQUVyQixPQUFPQSxLQUFLLENBQ1RDLE1BQU0sQ0FDSkMsSUFBSSxJQUNILE9BQU9BLElBQUksS0FBSyxRQUFRLElBQUlBLElBQUksS0FBSyxJQUN6QyxDQUFDLENBQ0FDLE9BQU8sQ0FBQ0QsSUFBSSxJQUFJO0lBQ2Y7SUFDQTtJQUNBLE1BQU1FLE9BQU8sR0FBR0YsSUFBSSxDQUFDRSxPQUFPLElBQUlGLElBQUksQ0FBQ0csSUFBSTtJQUN6QyxJQUFJQyxLQUFLLENBQUNDLE9BQU8sQ0FBQ0gsT0FBTyxDQUFDLEVBQUUsT0FBT0EsT0FBTztJQUMxQyxJQUFJLE9BQU9BLE9BQU8sS0FBSyxRQUFRLEVBQUUsT0FBTyxDQUFDQSxPQUFPLENBQUM7SUFDakQsT0FBTyxFQUFFO0VBQ1gsQ0FBQyxDQUFDO0FBQ04iLCJpZ25vcmVMaXN0IjpbXX0=
|
|
@@ -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 { extractProxyRoutes } from './proxyUtils.js';
|
|
18
19
|
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
19
20
|
// run directly from node
|
|
20
21
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
@@ -171,7 +172,7 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
171
172
|
if (!devServer) {
|
|
172
173
|
throw new Error('webpack-dev-server is not defined');
|
|
173
174
|
}
|
|
174
|
-
const otherRoutes = [process.env.WEBPACK_PUBLIC_PATH, ...(webpackConfigs[0].devServer?.proxy
|
|
175
|
+
const otherRoutes = [process.env.WEBPACK_PUBLIC_PATH, ...extractProxyRoutes(webpackConfigs[0].devServer?.proxy)];
|
|
175
176
|
// serve SSR for non-WEBPACK_PUBLIC_PATH
|
|
176
177
|
devServer.app?.get(new RegExp(`^(?!${otherRoutes.join('|')})`), handleErrors(async function (req, res) {
|
|
177
178
|
if (req.url.endsWith('favicon.ico')) {
|
|
@@ -243,4 +244,4 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
243
244
|
});
|
|
244
245
|
runServer();
|
|
245
246
|
}
|
|
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":[]}
|
|
247
|
+
//# 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","extractProxyRoutes","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","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 { extractProxyRoutes } from './proxyUtils.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          ...extractProxyRoutes(webpackConfigs[0].devServer?.proxy),\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":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";AAmBA,OAAO,mCAAmC,CAAC;
|
|
1
|
+
{"version":3,"file":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";AAmBA,OAAO,mCAAmC,CAAC;AAwB3C,wBAA8B,cAAc,CAC1C,UAAU,EAAE,MAAM,EAClB,GAAG,GAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAM,iBA0QlC"}
|
|
@@ -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 { extractProxyRoutes } from './proxyUtils.js';
|
|
18
19
|
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
19
20
|
// run directly from node
|
|
20
21
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
@@ -171,7 +172,7 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
171
172
|
if (!devServer) {
|
|
172
173
|
throw new Error('webpack-dev-server is not defined');
|
|
173
174
|
}
|
|
174
|
-
const otherRoutes = [process.env.WEBPACK_PUBLIC_PATH, ...(webpackConfigs[0].devServer?.proxy
|
|
175
|
+
const otherRoutes = [process.env.WEBPACK_PUBLIC_PATH, ...extractProxyRoutes(webpackConfigs[0].devServer?.proxy)];
|
|
175
176
|
// serve SSR for non-WEBPACK_PUBLIC_PATH
|
|
176
177
|
devServer.app?.get(new RegExp(`^(?!${otherRoutes.join('|')})`), handleErrors(async function (req, res) {
|
|
177
178
|
if (req.url.endsWith('favicon.ico')) {
|
|
@@ -243,4 +244,4 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
243
244
|
});
|
|
244
245
|
runServer();
|
|
245
246
|
}
|
|
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":[]}
|
|
247
|
+
//# 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","extractProxyRoutes","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","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 { extractProxyRoutes } from './proxyUtils.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          ...extractProxyRoutes(webpackConfigs[0].devServer?.proxy),\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":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@anansi/core",
|
|
3
|
-
"version": "0.22.
|
|
3
|
+
"version": "0.22.4",
|
|
4
4
|
"description": "React 19 Framework",
|
|
5
5
|
"homepage": "https://github.com/ntucker/anansi/tree/master/packages/core#readme",
|
|
6
6
|
"repository": {
|
|
@@ -82,7 +82,7 @@
|
|
|
82
82
|
"@types/compression": "1.8.1",
|
|
83
83
|
"@types/express": "^4.17.17",
|
|
84
84
|
"@types/node": "^24.0.0",
|
|
85
|
-
"@types/react": "19.2.
|
|
85
|
+
"@types/react": "19.2.8",
|
|
86
86
|
"@types/react-dom": "19.2.3",
|
|
87
87
|
"@types/source-map-support": "0.5.10",
|
|
88
88
|
"@types/tmp": "0.2.6",
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { extractProxyRoutes } from '../proxyUtils';
|
|
2
|
+
|
|
3
|
+
describe('extractProxyRoutes', () => {
|
|
4
|
+
it('should return empty array for undefined proxy', () => {
|
|
5
|
+
expect(extractProxyRoutes(undefined)).toEqual([]);
|
|
6
|
+
});
|
|
7
|
+
|
|
8
|
+
it('should return empty array for empty proxy array', () => {
|
|
9
|
+
expect(extractProxyRoutes([])).toEqual([]);
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
it('should extract context as array of strings', () => {
|
|
13
|
+
const proxy = [{ context: ['/api'], target: 'http://localhost:3000' }];
|
|
14
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api']);
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
it('should extract multiple contexts from array', () => {
|
|
18
|
+
const proxy = [
|
|
19
|
+
{ context: ['/api', '/graphql'], target: 'http://localhost:3000' },
|
|
20
|
+
];
|
|
21
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api', '/graphql']);
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it('should extract context as single string', () => {
|
|
25
|
+
const proxy = [{ context: '/api', target: 'http://localhost:3000' }];
|
|
26
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api']);
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
it('should extract path property (legacy format)', () => {
|
|
30
|
+
const proxy = [{ path: '/legacy', target: 'http://localhost:3000' }];
|
|
31
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/legacy']);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('should extract path as array', () => {
|
|
35
|
+
const proxy = [
|
|
36
|
+
{ path: ['/legacy', '/old-api'], target: 'http://localhost:3000' },
|
|
37
|
+
];
|
|
38
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/legacy', '/old-api']);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it('should prefer context over path', () => {
|
|
42
|
+
const proxy = [
|
|
43
|
+
{ context: '/api', path: '/legacy', target: 'http://localhost:3000' },
|
|
44
|
+
];
|
|
45
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api']);
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it('should handle multiple proxy entries', () => {
|
|
49
|
+
const proxy = [
|
|
50
|
+
{ context: ['/api'], target: 'http://localhost:3000' },
|
|
51
|
+
{ context: '/ws', target: 'http://localhost:3001' },
|
|
52
|
+
{ path: '/legacy', target: 'http://localhost:3002' },
|
|
53
|
+
];
|
|
54
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api', '/ws', '/legacy']);
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
it('should skip function entries', () => {
|
|
58
|
+
const proxy = [
|
|
59
|
+
{ context: ['/api'], target: 'http://localhost:3000' },
|
|
60
|
+
() => ({ context: '/dynamic', target: 'http://localhost:3001' }),
|
|
61
|
+
];
|
|
62
|
+
// Functions are filtered out since we can't statically analyze them
|
|
63
|
+
expect(extractProxyRoutes(proxy as any)).toEqual(['/api']);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it('should skip entries without context or path', () => {
|
|
67
|
+
const proxy = [
|
|
68
|
+
{ context: ['/api'], target: 'http://localhost:3000' },
|
|
69
|
+
{ target: 'http://localhost:3001', router: {} }, // router-based proxy without context
|
|
70
|
+
];
|
|
71
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api']);
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
it('should handle null entries gracefully', () => {
|
|
75
|
+
const proxy = [
|
|
76
|
+
{ context: ['/api'], target: 'http://localhost:3000' },
|
|
77
|
+
null as any,
|
|
78
|
+
];
|
|
79
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api']);
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it('should handle mixed context types across entries', () => {
|
|
83
|
+
const proxy = [
|
|
84
|
+
{ context: ['/api', '/graphql'], target: 'http://localhost:3000' },
|
|
85
|
+
{ context: '/ws', target: 'http://localhost:3001' },
|
|
86
|
+
];
|
|
87
|
+
expect(extractProxyRoutes(proxy)).toEqual(['/api', '/graphql', '/ws']);
|
|
88
|
+
});
|
|
89
|
+
});
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import type { ProxyConfigArray } from 'webpack-dev-server';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Extracts route patterns from webpack-dev-server proxy configuration.
|
|
5
|
+
*
|
|
6
|
+
* Handles the webpack-dev-server proxy array format:
|
|
7
|
+
* ```
|
|
8
|
+
* proxy: [
|
|
9
|
+
* { context: ['/api'], target: 'http://localhost:3000' },
|
|
10
|
+
* { context: '/ws', target: 'http://localhost:3001' },
|
|
11
|
+
* { path: '/legacy', target: 'http://localhost:3002' },
|
|
12
|
+
* ]
|
|
13
|
+
* ```
|
|
14
|
+
*
|
|
15
|
+
* @see https://webpack.js.org/configuration/dev-server/#devserverproxy
|
|
16
|
+
*/
|
|
17
|
+
export function extractProxyRoutes(
|
|
18
|
+
proxy: ProxyConfigArray | undefined,
|
|
19
|
+
): string[] {
|
|
20
|
+
if (!proxy) return [];
|
|
21
|
+
|
|
22
|
+
return proxy
|
|
23
|
+
.filter(
|
|
24
|
+
(item): item is Record<string, unknown> =>
|
|
25
|
+
typeof item === 'object' && item !== null,
|
|
26
|
+
)
|
|
27
|
+
.flatMap(item => {
|
|
28
|
+
// webpack-dev-server proxy supports both 'context' and 'path' properties
|
|
29
|
+
// and each can be a string or an array of strings
|
|
30
|
+
const context = item.context ?? item.path;
|
|
31
|
+
if (Array.isArray(context)) return context as string[];
|
|
32
|
+
if (typeof context === 'string') return [context];
|
|
33
|
+
return [];
|
|
34
|
+
});
|
|
35
|
+
}
|
|
@@ -20,6 +20,7 @@ import WebpackDevServer from 'webpack-dev-server';
|
|
|
20
20
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
21
21
|
import { createHybridRequire } from './createHybridRequire.js';
|
|
22
22
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
23
|
+
import { extractProxyRoutes } from './proxyUtils.js';
|
|
23
24
|
import { getErrorStatus, renderErrorPage } from './ssrErrorHandler.js';
|
|
24
25
|
import { BoundRender } from './types.js';
|
|
25
26
|
|
|
@@ -216,9 +217,7 @@ export default async function startDevServer(
|
|
|
216
217
|
|
|
217
218
|
const otherRoutes = [
|
|
218
219
|
process.env.WEBPACK_PUBLIC_PATH,
|
|
219
|
-
...(webpackConfigs[0].devServer?.proxy
|
|
220
|
-
?.filter(proxy => typeof proxy === 'object')
|
|
221
|
-
?.flatMap(proxy => proxy.context) ?? []),
|
|
220
|
+
...extractProxyRoutes(webpackConfigs[0].devServer?.proxy),
|
|
222
221
|
];
|
|
223
222
|
// serve SSR for non-WEBPACK_PUBLIC_PATH
|
|
224
223
|
devServer.app?.get(
|