@anansi/core 0.20.38 → 0.20.40
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 +20 -0
- package/lib/scripts/createHybridRequire.d.ts +16 -0
- package/lib/scripts/createHybridRequire.d.ts.map +1 -0
- package/lib/scripts/createHybridRequire.js +36 -0
- package/lib/scripts/getWebpackConfig.d.ts +2 -2
- package/lib/scripts/getWebpackConfig.d.ts.map +1 -1
- package/lib/scripts/getWebpackConfig.js +1 -1
- package/lib/scripts/scripts/createHybridRequire.js +36 -0
- package/lib/scripts/scripts/getWebpackConfig.js +1 -1
- package/lib/scripts/scripts/startDevserver.js +3 -5
- package/lib/scripts/startDevserver.d.ts.map +1 -1
- package/lib/scripts/startDevserver.js +3 -5
- package/package.json +4 -4
- package/src/scripts/__tests__/hybridRequire.cwd.test.ts +172 -0
- package/src/scripts/__tests__/hybridRequire.test.ts +243 -0
- package/src/scripts/__tests__/tsconfig.json +4 -0
- package/src/scripts/createHybridRequire.ts +42 -0
- package/src/scripts/getWebpackConfig.ts +2 -2
- package/src/scripts/startDevserver.ts +8 -9
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,26 @@
|
|
|
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.20.40](/github.com/ntucker/anansi/compare/@anansi/core@0.20.39...@anansi/core@0.20.40) (2025-11-15)
|
|
7
|
+
|
|
8
|
+
### 🐛 Bug Fix
|
|
9
|
+
|
|
10
|
+
* Properly handle fallbacks to node_modules in multiple directories ([#2864](/github.com/ntucker/anansi/issues/2864)) ([136bcb5](/github.com/ntucker/anansi/commit/136bcb5a2a3a895931200f16eeccd7c3b071b11b))
|
|
11
|
+
|
|
12
|
+
### 📦 Package
|
|
13
|
+
|
|
14
|
+
* Update all non-major dependencies ([#2866](/github.com/ntucker/anansi/issues/2866)) ([930d19a](/github.com/ntucker/anansi/commit/930d19aa7ff8768fb2ba9f220543374435c25fbf))
|
|
15
|
+
|
|
16
|
+
## [0.20.39](/github.com/ntucker/anansi/compare/@anansi/core@0.20.38...@anansi/core@0.20.39) (2025-11-09)
|
|
17
|
+
|
|
18
|
+
### 🐛 Bug Fix
|
|
19
|
+
|
|
20
|
+
* Update types ([7b36890](/github.com/ntucker/anansi/commit/7b3689004a45de9ac14091a130eae8177d4baed4))
|
|
21
|
+
|
|
22
|
+
### 📦 Package
|
|
23
|
+
|
|
24
|
+
* Update all non-major dependencies ([#2858](/github.com/ntucker/anansi/issues/2858)) ([ea62701](/github.com/ntucker/anansi/commit/ea6270186daa1634ef55e2cbd158879bcfc6bbdd))
|
|
25
|
+
|
|
6
26
|
## [0.20.38](/github.com/ntucker/anansi/compare/@anansi/core@0.20.37...@anansi/core@0.20.38) (2025-10-14)
|
|
7
27
|
|
|
8
28
|
### 📦 Package
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { createFsRequire } from 'fs-require';
|
|
2
|
+
import { type IUnionFs } from 'unionfs';
|
|
3
|
+
/**
|
|
4
|
+
* Creates a hybrid require function that combines fs-require (for in-memory files)
|
|
5
|
+
* with Node's native require (for bare specifiers from project node_modules).
|
|
6
|
+
*
|
|
7
|
+
* This solves the issue where fs-require cannot resolve bare specifiers like
|
|
8
|
+
* `@anansi/core/server` from workspace node_modules when they're not in the
|
|
9
|
+
* memfs bundle.
|
|
10
|
+
*
|
|
11
|
+
* @param ufs - The unionfs instance (disk first, then memfs)
|
|
12
|
+
* @param projectRoot - Optional project root directory. Defaults to process.cwd()
|
|
13
|
+
* @returns A require function that tries memfs first, then falls back to project node_modules
|
|
14
|
+
*/
|
|
15
|
+
export declare function createHybridRequire(ufs: IUnionFs, projectRoot?: string): ReturnType<typeof createFsRequire>;
|
|
16
|
+
//# sourceMappingURL=createHybridRequire.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"createHybridRequire.d.ts","sourceRoot":"","sources":["../../src/scripts/createHybridRequire.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAG7C,OAAO,EAAE,KAAK,QAAQ,EAAE,MAAM,SAAS,CAAC;AAExC;;;;;;;;;;;GAWG;AACH,wBAAgB,mBAAmB,CACjC,GAAG,EAAE,QAAQ,EACb,WAAW,GAAE,MAAsB,GAClC,UAAU,CAAC,OAAO,eAAe,CAAC,CAqBpC"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { createFsRequire } from 'fs-require';
|
|
2
|
+
import { createRequire } from 'module';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
/**
|
|
5
|
+
* Creates a hybrid require function that combines fs-require (for in-memory files)
|
|
6
|
+
* with Node's native require (for bare specifiers from project node_modules).
|
|
7
|
+
*
|
|
8
|
+
* This solves the issue where fs-require cannot resolve bare specifiers like
|
|
9
|
+
* `@anansi/core/server` from workspace node_modules when they're not in the
|
|
10
|
+
* memfs bundle.
|
|
11
|
+
*
|
|
12
|
+
* @param ufs - The unionfs instance (disk first, then memfs)
|
|
13
|
+
* @param projectRoot - Optional project root directory. Defaults to process.cwd()
|
|
14
|
+
* @returns A require function that tries memfs first, then falls back to project node_modules
|
|
15
|
+
*/
|
|
16
|
+
export function createHybridRequire(ufs, projectRoot = process.cwd()) {
|
|
17
|
+
const fsRequire = createFsRequire(ufs);
|
|
18
|
+
const projectRequire = createRequire(path.join(projectRoot, 'package.json'));
|
|
19
|
+
function hybridRequire(id) {
|
|
20
|
+
try {
|
|
21
|
+
return fsRequire(id);
|
|
22
|
+
} catch (error) {
|
|
23
|
+
const isBare = !id.startsWith('.') && !path.isAbsolute(id);
|
|
24
|
+
if (isBare && error?.code === 'MODULE_NOT_FOUND') {
|
|
25
|
+
return projectRequire(id);
|
|
26
|
+
}
|
|
27
|
+
throw error;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// Preserve cache and resolve properties from fsRequire
|
|
32
|
+
hybridRequire.cache = fsRequire.cache;
|
|
33
|
+
hybridRequire.resolve = fsRequire.resolve;
|
|
34
|
+
return hybridRequire;
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjcmVhdGVGc1JlcXVpcmUiLCJjcmVhdGVSZXF1aXJlIiwicGF0aCIsImNyZWF0ZUh5YnJpZFJlcXVpcmUiLCJ1ZnMiLCJwcm9qZWN0Um9vdCIsInByb2Nlc3MiLCJjd2QiLCJmc1JlcXVpcmUiLCJwcm9qZWN0UmVxdWlyZSIsImpvaW4iLCJoeWJyaWRSZXF1aXJlIiwiaWQiLCJlcnJvciIsImlzQmFyZSIsInN0YXJ0c1dpdGgiLCJpc0Fic29sdXRlIiwiY29kZSIsImNhY2hlIiwicmVzb2x2ZSJdLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JpcHRzL2NyZWF0ZUh5YnJpZFJlcXVpcmUudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3JlYXRlRnNSZXF1aXJlIH0gZnJvbSAnZnMtcmVxdWlyZSc7XG5pbXBvcnQgeyBjcmVhdGVSZXF1aXJlIH0gZnJvbSAnbW9kdWxlJztcbmltcG9ydCBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgdHlwZSBJVW5pb25GcyB9IGZyb20gJ3VuaW9uZnMnO1xuXG4vKipcbiAqIENyZWF0ZXMgYSBoeWJyaWQgcmVxdWlyZSBmdW5jdGlvbiB0aGF0IGNvbWJpbmVzIGZzLXJlcXVpcmUgKGZvciBpbi1tZW1vcnkgZmlsZXMpXG4gKiB3aXRoIE5vZGUncyBuYXRpdmUgcmVxdWlyZSAoZm9yIGJhcmUgc3BlY2lmaWVycyBmcm9tIHByb2plY3Qgbm9kZV9tb2R1bGVzKS5cbiAqXG4gKiBUaGlzIHNvbHZlcyB0aGUgaXNzdWUgd2hlcmUgZnMtcmVxdWlyZSBjYW5ub3QgcmVzb2x2ZSBiYXJlIHNwZWNpZmllcnMgbGlrZVxuICogYEBhbmFuc2kvY29yZS9zZXJ2ZXJgIGZyb20gd29ya3NwYWNlIG5vZGVfbW9kdWxlcyB3aGVuIHRoZXkncmUgbm90IGluIHRoZVxuICogbWVtZnMgYnVuZGxlLlxuICpcbiAqIEBwYXJhbSB1ZnMgLSBUaGUgdW5pb25mcyBpbnN0YW5jZSAoZGlzayBmaXJzdCwgdGhlbiBtZW1mcylcbiAqIEBwYXJhbSBwcm9qZWN0Um9vdCAtIE9wdGlvbmFsIHByb2plY3Qgcm9vdCBkaXJlY3RvcnkuIERlZmF1bHRzIHRvIHByb2Nlc3MuY3dkKClcbiAqIEByZXR1cm5zIEEgcmVxdWlyZSBmdW5jdGlvbiB0aGF0IHRyaWVzIG1lbWZzIGZpcnN0LCB0aGVuIGZhbGxzIGJhY2sgdG8gcHJvamVjdCBub2RlX21vZHVsZXNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUh5YnJpZFJlcXVpcmUoXG4gIHVmczogSVVuaW9uRnMsXG4gIHByb2plY3RSb290OiBzdHJpbmcgPSBwcm9jZXNzLmN3ZCgpLFxuKTogUmV0dXJuVHlwZTx0eXBlb2YgY3JlYXRlRnNSZXF1aXJlPiB7XG4gIGNvbnN0IGZzUmVxdWlyZSA9IGNyZWF0ZUZzUmVxdWlyZSh1ZnMpO1xuICBjb25zdCBwcm9qZWN0UmVxdWlyZSA9IGNyZWF0ZVJlcXVpcmUocGF0aC5qb2luKHByb2plY3RSb290LCAncGFja2FnZS5qc29uJykpO1xuXG4gIGZ1bmN0aW9uIGh5YnJpZFJlcXVpcmUoaWQ6IHN0cmluZykge1xuICAgIHRyeSB7XG4gICAgICByZXR1cm4gZnNSZXF1aXJlKGlkKTtcbiAgICB9IGNhdGNoIChlcnJvcjogYW55KSB7XG4gICAgICBjb25zdCBpc0JhcmUgPSAhaWQuc3RhcnRzV2l0aCgnLicpICYmICFwYXRoLmlzQWJzb2x1dGUoaWQpO1xuICAgICAgaWYgKGlzQmFyZSAmJiBlcnJvcj8uY29kZSA9PT0gJ01PRFVMRV9OT1RfRk9VTkQnKSB7XG4gICAgICAgIHJldHVybiBwcm9qZWN0UmVxdWlyZShpZCk7XG4gICAgICB9XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9XG4gIH1cblxuICAvLyBQcmVzZXJ2ZSBjYWNoZSBhbmQgcmVzb2x2ZSBwcm9wZXJ0aWVzIGZyb20gZnNSZXF1aXJlXG4gIGh5YnJpZFJlcXVpcmUuY2FjaGUgPSBmc1JlcXVpcmUuY2FjaGU7XG4gIGh5YnJpZFJlcXVpcmUucmVzb2x2ZSA9IGZzUmVxdWlyZS5yZXNvbHZlO1xuXG4gIHJldHVybiBoeWJyaWRSZXF1aXJlIGFzIFJldHVyblR5cGU8dHlwZW9mIGNyZWF0ZUZzUmVxdWlyZT47XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLGVBQWUsUUFBUSxZQUFZO0FBQzVDLFNBQVNDLGFBQWEsUUFBUSxRQUFRO0FBQ3RDLE9BQU9DLElBQUksTUFBTSxNQUFNO0FBR3ZCO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLE9BQU8sU0FBU0MsbUJBQW1CQSxDQUNqQ0MsR0FBYSxFQUNiQyxXQUFtQixHQUFHQyxPQUFPLENBQUNDLEdBQUcsQ0FBQyxDQUFDLEVBQ0M7RUFDcEMsTUFBTUMsU0FBUyxHQUFHUixlQUFlLENBQUNJLEdBQUcsQ0FBQztFQUN0QyxNQUFNSyxjQUFjLEdBQUdSLGFBQWEsQ0FBQ0MsSUFBSSxDQUFDUSxJQUFJLENBQUNMLFdBQVcsRUFBRSxjQUFjLENBQUMsQ0FBQztFQUU1RSxTQUFTTSxhQUFhQSxDQUFDQyxFQUFVLEVBQUU7SUFDakMsSUFBSTtNQUNGLE9BQU9KLFNBQVMsQ0FBQ0ksRUFBRSxDQUFDO0lBQ3RCLENBQUMsQ0FBQyxPQUFPQyxLQUFVLEVBQUU7TUFDbkIsTUFBTUMsTUFBTSxHQUFHLENBQUNGLEVBQUUsQ0FBQ0csVUFBVSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUNiLElBQUksQ0FBQ2MsVUFBVSxDQUFDSixFQUFFLENBQUM7TUFDMUQsSUFBSUUsTUFBTSxJQUFJRCxLQUFLLEVBQUVJLElBQUksS0FBSyxrQkFBa0IsRUFBRTtRQUNoRCxPQUFPUixjQUFjLENBQUNHLEVBQUUsQ0FBQztNQUMzQjtNQUNBLE1BQU1DLEtBQUs7SUFDYjtFQUNGOztFQUVBO0VBQ0FGLGFBQWEsQ0FBQ08sS0FBSyxHQUFHVixTQUFTLENBQUNVLEtBQUs7RUFDckNQLGFBQWEsQ0FBQ1EsT0FBTyxHQUFHWCxTQUFTLENBQUNXLE9BQU87RUFFekMsT0FBT1IsYUFBYTtBQUN0QiIsImlnbm9yZUxpc3QiOltdfQ==
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
import
|
|
2
|
-
export declare function getWebpackConfig(): Promise<(env: any, argv: any) =>
|
|
1
|
+
import type { Configuration } from 'webpack';
|
|
2
|
+
export declare function getWebpackConfig(): Promise<(env: any, argv: any) => Configuration>;
|
|
3
3
|
//# sourceMappingURL=getWebpackConfig.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"getWebpackConfig.d.ts","sourceRoot":"","sources":["../../src/scripts/getWebpackConfig.ts"],"names":[],"mappings":"AACA,OAAO,
|
|
1
|
+
{"version":3,"file":"getWebpackConfig.d.ts","sourceRoot":"","sources":["../../src/scripts/getWebpackConfig.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAS7C,wBAAsB,gBAAgB,IAAI,OAAO,CAC/C,CAAC,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,GAAG,KAAK,aAAa,CACvC,CAOA"}
|
|
@@ -12,4 +12,4 @@ export async function getWebpackConfig() {
|
|
|
12
12
|
}
|
|
13
13
|
return (await import(configPath)).default;
|
|
14
14
|
}
|
|
15
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
15
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJkZWZhdWx0IiwiZW5oYW5jZWQiLCJyZXNvbHZlIiwiY3JlYXRlIiwic3luYyIsIm1vZHVsZXMiLCJleHRlbnNpb25zIiwiZ2V0V2VicGFja0NvbmZpZyIsImNvbmZpZ1BhdGgiLCJwcm9jZXNzIiwiY3dkIiwiRXJyb3IiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvc2NyaXB0cy9nZXRXZWJwYWNrQ29uZmlnLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlZmF1bHQgYXMgZW5oYW5jZWQgfSBmcm9tICdlbmhhbmNlZC1yZXNvbHZlJztcbmltcG9ydCB0eXBlIHsgQ29uZmlndXJhdGlvbiB9IGZyb20gJ3dlYnBhY2snO1xuXG5jb25zdCByZXNvbHZlID0gZW5oYW5jZWQuY3JlYXRlLnN5bmMoe1xuICBtb2R1bGVzOiBbJy4nXSxcbiAgLy8gb3IgcmVzb2x2ZS5jcmVhdGUuc3luY1xuICBleHRlbnNpb25zOiBbJy5qcycsICcuY2pzJywgJy5tanMnXSxcbiAgLy8gc2VlIG1vcmUgb3B0aW9ucyBiZWxvd1xufSk7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRXZWJwYWNrQ29uZmlnKCk6IFByb21pc2U8XG4gIChlbnY6IGFueSwgYXJndjogYW55KSA9PiBDb25maWd1cmF0aW9uXG4+IHtcbiAgY29uc3QgY29uZmlnUGF0aCA9IHJlc29sdmUoe30sIHByb2Nlc3MuY3dkKCksICd3ZWJwYWNrLmNvbmZpZycpO1xuICBpZiAoIWNvbmZpZ1BhdGgpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ0Nhbm5vdCBmaW5kIHdlYnBhY2suY29uZmlnIGluICcgKyBwcm9jZXNzLmN3ZCgpKTtcbiAgfVxuXG4gIHJldHVybiAoYXdhaXQgaW1wb3J0KGNvbmZpZ1BhdGgpKS5kZWZhdWx0O1xufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxPQUFPLElBQUlDLFFBQVEsUUFBUSxrQkFBa0I7QUFHdEQsTUFBTUMsT0FBTyxHQUFHRCxRQUFRLENBQUNFLE1BQU0sQ0FBQ0MsSUFBSSxDQUFDO0VBQ25DQyxPQUFPLEVBQUUsQ0FBQyxHQUFHLENBQUM7RUFDZDtFQUNBQyxVQUFVLEVBQUUsQ0FBQyxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU07RUFDbEM7QUFDRixDQUFDLENBQUM7QUFFRixPQUFPLGVBQWVDLGdCQUFnQkEsQ0FBQSxFQUVwQztFQUNBLE1BQU1DLFVBQVUsR0FBR04sT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFTyxPQUFPLENBQUNDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsZ0JBQWdCLENBQUM7RUFDL0QsSUFBSSxDQUFDRixVQUFVLEVBQUU7SUFDZixNQUFNLElBQUlHLEtBQUssQ0FBQyxnQ0FBZ0MsR0FBR0YsT0FBTyxDQUFDQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0VBQ25FO0VBRUEsT0FBTyxDQUFDLE1BQU0sTUFBTSxDQUFDRixVQUFVLENBQUMsRUFBRVIsT0FBTztBQUMzQyIsImlnbm9yZUxpc3QiOltdfQ==
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { createFsRequire } from 'fs-require';
|
|
2
|
+
import { createRequire } from 'module';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
/**
|
|
5
|
+
* Creates a hybrid require function that combines fs-require (for in-memory files)
|
|
6
|
+
* with Node's native require (for bare specifiers from project node_modules).
|
|
7
|
+
*
|
|
8
|
+
* This solves the issue where fs-require cannot resolve bare specifiers like
|
|
9
|
+
* `@anansi/core/server` from workspace node_modules when they're not in the
|
|
10
|
+
* memfs bundle.
|
|
11
|
+
*
|
|
12
|
+
* @param ufs - The unionfs instance (disk first, then memfs)
|
|
13
|
+
* @param projectRoot - Optional project root directory. Defaults to process.cwd()
|
|
14
|
+
* @returns A require function that tries memfs first, then falls back to project node_modules
|
|
15
|
+
*/
|
|
16
|
+
export function createHybridRequire(ufs, projectRoot = process.cwd()) {
|
|
17
|
+
const fsRequire = createFsRequire(ufs);
|
|
18
|
+
const projectRequire = createRequire(path.join(projectRoot, 'package.json'));
|
|
19
|
+
function hybridRequire(id) {
|
|
20
|
+
try {
|
|
21
|
+
return fsRequire(id);
|
|
22
|
+
} catch (error) {
|
|
23
|
+
const isBare = !id.startsWith('.') && !path.isAbsolute(id);
|
|
24
|
+
if (isBare && error?.code === 'MODULE_NOT_FOUND') {
|
|
25
|
+
return projectRequire(id);
|
|
26
|
+
}
|
|
27
|
+
throw error;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// Preserve cache and resolve properties from fsRequire
|
|
32
|
+
hybridRequire.cache = fsRequire.cache;
|
|
33
|
+
hybridRequire.resolve = fsRequire.resolve;
|
|
34
|
+
return hybridRequire;
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjcmVhdGVGc1JlcXVpcmUiLCJjcmVhdGVSZXF1aXJlIiwicGF0aCIsImNyZWF0ZUh5YnJpZFJlcXVpcmUiLCJ1ZnMiLCJwcm9qZWN0Um9vdCIsInByb2Nlc3MiLCJjd2QiLCJmc1JlcXVpcmUiLCJwcm9qZWN0UmVxdWlyZSIsImpvaW4iLCJoeWJyaWRSZXF1aXJlIiwiaWQiLCJlcnJvciIsImlzQmFyZSIsInN0YXJ0c1dpdGgiLCJpc0Fic29sdXRlIiwiY29kZSIsImNhY2hlIiwicmVzb2x2ZSJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zY3JpcHRzL2NyZWF0ZUh5YnJpZFJlcXVpcmUudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3JlYXRlRnNSZXF1aXJlIH0gZnJvbSAnZnMtcmVxdWlyZSc7XG5pbXBvcnQgeyBjcmVhdGVSZXF1aXJlIH0gZnJvbSAnbW9kdWxlJztcbmltcG9ydCBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgdHlwZSBJVW5pb25GcyB9IGZyb20gJ3VuaW9uZnMnO1xuXG4vKipcbiAqIENyZWF0ZXMgYSBoeWJyaWQgcmVxdWlyZSBmdW5jdGlvbiB0aGF0IGNvbWJpbmVzIGZzLXJlcXVpcmUgKGZvciBpbi1tZW1vcnkgZmlsZXMpXG4gKiB3aXRoIE5vZGUncyBuYXRpdmUgcmVxdWlyZSAoZm9yIGJhcmUgc3BlY2lmaWVycyBmcm9tIHByb2plY3Qgbm9kZV9tb2R1bGVzKS5cbiAqXG4gKiBUaGlzIHNvbHZlcyB0aGUgaXNzdWUgd2hlcmUgZnMtcmVxdWlyZSBjYW5ub3QgcmVzb2x2ZSBiYXJlIHNwZWNpZmllcnMgbGlrZVxuICogYEBhbmFuc2kvY29yZS9zZXJ2ZXJgIGZyb20gd29ya3NwYWNlIG5vZGVfbW9kdWxlcyB3aGVuIHRoZXkncmUgbm90IGluIHRoZVxuICogbWVtZnMgYnVuZGxlLlxuICpcbiAqIEBwYXJhbSB1ZnMgLSBUaGUgdW5pb25mcyBpbnN0YW5jZSAoZGlzayBmaXJzdCwgdGhlbiBtZW1mcylcbiAqIEBwYXJhbSBwcm9qZWN0Um9vdCAtIE9wdGlvbmFsIHByb2plY3Qgcm9vdCBkaXJlY3RvcnkuIERlZmF1bHRzIHRvIHByb2Nlc3MuY3dkKClcbiAqIEByZXR1cm5zIEEgcmVxdWlyZSBmdW5jdGlvbiB0aGF0IHRyaWVzIG1lbWZzIGZpcnN0LCB0aGVuIGZhbGxzIGJhY2sgdG8gcHJvamVjdCBub2RlX21vZHVsZXNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUh5YnJpZFJlcXVpcmUoXG4gIHVmczogSVVuaW9uRnMsXG4gIHByb2plY3RSb290OiBzdHJpbmcgPSBwcm9jZXNzLmN3ZCgpLFxuKTogUmV0dXJuVHlwZTx0eXBlb2YgY3JlYXRlRnNSZXF1aXJlPiB7XG4gIGNvbnN0IGZzUmVxdWlyZSA9IGNyZWF0ZUZzUmVxdWlyZSh1ZnMpO1xuICBjb25zdCBwcm9qZWN0UmVxdWlyZSA9IGNyZWF0ZVJlcXVpcmUocGF0aC5qb2luKHByb2plY3RSb290LCAncGFja2FnZS5qc29uJykpO1xuXG4gIGZ1bmN0aW9uIGh5YnJpZFJlcXVpcmUoaWQ6IHN0cmluZykge1xuICAgIHRyeSB7XG4gICAgICByZXR1cm4gZnNSZXF1aXJlKGlkKTtcbiAgICB9IGNhdGNoIChlcnJvcjogYW55KSB7XG4gICAgICBjb25zdCBpc0JhcmUgPSAhaWQuc3RhcnRzV2l0aCgnLicpICYmICFwYXRoLmlzQWJzb2x1dGUoaWQpO1xuICAgICAgaWYgKGlzQmFyZSAmJiBlcnJvcj8uY29kZSA9PT0gJ01PRFVMRV9OT1RfRk9VTkQnKSB7XG4gICAgICAgIHJldHVybiBwcm9qZWN0UmVxdWlyZShpZCk7XG4gICAgICB9XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9XG4gIH1cblxuICAvLyBQcmVzZXJ2ZSBjYWNoZSBhbmQgcmVzb2x2ZSBwcm9wZXJ0aWVzIGZyb20gZnNSZXF1aXJlXG4gIGh5YnJpZFJlcXVpcmUuY2FjaGUgPSBmc1JlcXVpcmUuY2FjaGU7XG4gIGh5YnJpZFJlcXVpcmUucmVzb2x2ZSA9IGZzUmVxdWlyZS5yZXNvbHZlO1xuXG4gIHJldHVybiBoeWJyaWRSZXF1aXJlIGFzIFJldHVyblR5cGU8dHlwZW9mIGNyZWF0ZUZzUmVxdWlyZT47XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLGVBQWUsUUFBUSxZQUFZO0FBQzVDLFNBQVNDLGFBQWEsUUFBUSxRQUFRO0FBQ3RDLE9BQU9DLElBQUksTUFBTSxNQUFNO0FBR3ZCO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLE9BQU8sU0FBU0MsbUJBQW1CQSxDQUNqQ0MsR0FBYSxFQUNiQyxXQUFtQixHQUFHQyxPQUFPLENBQUNDLEdBQUcsQ0FBQyxDQUFDLEVBQ0M7RUFDcEMsTUFBTUMsU0FBUyxHQUFHUixlQUFlLENBQUNJLEdBQUcsQ0FBQztFQUN0QyxNQUFNSyxjQUFjLEdBQUdSLGFBQWEsQ0FBQ0MsSUFBSSxDQUFDUSxJQUFJLENBQUNMLFdBQVcsRUFBRSxjQUFjLENBQUMsQ0FBQztFQUU1RSxTQUFTTSxhQUFhQSxDQUFDQyxFQUFVLEVBQUU7SUFDakMsSUFBSTtNQUNGLE9BQU9KLFNBQVMsQ0FBQ0ksRUFBRSxDQUFDO0lBQ3RCLENBQUMsQ0FBQyxPQUFPQyxLQUFVLEVBQUU7TUFDbkIsTUFBTUMsTUFBTSxHQUFHLENBQUNGLEVBQUUsQ0FBQ0csVUFBVSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUNiLElBQUksQ0FBQ2MsVUFBVSxDQUFDSixFQUFFLENBQUM7TUFDMUQsSUFBSUUsTUFBTSxJQUFJRCxLQUFLLEVBQUVJLElBQUksS0FBSyxrQkFBa0IsRUFBRTtRQUNoRCxPQUFPUixjQUFjLENBQUNHLEVBQUUsQ0FBQztNQUMzQjtNQUNBLE1BQU1DLEtBQUs7SUFDYjtFQUNGOztFQUVBO0VBQ0FGLGFBQWEsQ0FBQ08sS0FBSyxHQUFHVixTQUFTLENBQUNVLEtBQUs7RUFDckNQLGFBQWEsQ0FBQ1EsT0FBTyxHQUFHWCxTQUFTLENBQUNXLE9BQU87RUFFekMsT0FBT1IsYUFBYTtBQUN0QiIsImlnbm9yZUxpc3QiOltdfQ==
|
|
@@ -12,4 +12,4 @@ export async function getWebpackConfig() {
|
|
|
12
12
|
}
|
|
13
13
|
return (await import(configPath)).default;
|
|
14
14
|
}
|
|
15
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
15
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJkZWZhdWx0IiwiZW5oYW5jZWQiLCJyZXNvbHZlIiwiY3JlYXRlIiwic3luYyIsIm1vZHVsZXMiLCJleHRlbnNpb25zIiwiZ2V0V2VicGFja0NvbmZpZyIsImNvbmZpZ1BhdGgiLCJwcm9jZXNzIiwiY3dkIiwiRXJyb3IiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2NyaXB0cy9nZXRXZWJwYWNrQ29uZmlnLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlZmF1bHQgYXMgZW5oYW5jZWQgfSBmcm9tICdlbmhhbmNlZC1yZXNvbHZlJztcbmltcG9ydCB0eXBlIHsgQ29uZmlndXJhdGlvbiB9IGZyb20gJ3dlYnBhY2snO1xuXG5jb25zdCByZXNvbHZlID0gZW5oYW5jZWQuY3JlYXRlLnN5bmMoe1xuICBtb2R1bGVzOiBbJy4nXSxcbiAgLy8gb3IgcmVzb2x2ZS5jcmVhdGUuc3luY1xuICBleHRlbnNpb25zOiBbJy5qcycsICcuY2pzJywgJy5tanMnXSxcbiAgLy8gc2VlIG1vcmUgb3B0aW9ucyBiZWxvd1xufSk7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRXZWJwYWNrQ29uZmlnKCk6IFByb21pc2U8XG4gIChlbnY6IGFueSwgYXJndjogYW55KSA9PiBDb25maWd1cmF0aW9uXG4+IHtcbiAgY29uc3QgY29uZmlnUGF0aCA9IHJlc29sdmUoe30sIHByb2Nlc3MuY3dkKCksICd3ZWJwYWNrLmNvbmZpZycpO1xuICBpZiAoIWNvbmZpZ1BhdGgpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ0Nhbm5vdCBmaW5kIHdlYnBhY2suY29uZmlnIGluICcgKyBwcm9jZXNzLmN3ZCgpKTtcbiAgfVxuXG4gIHJldHVybiAoYXdhaXQgaW1wb3J0KGNvbmZpZ1BhdGgpKS5kZWZhdWx0O1xufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxPQUFPLElBQUlDLFFBQVEsUUFBUSxrQkFBa0I7QUFHdEQsTUFBTUMsT0FBTyxHQUFHRCxRQUFRLENBQUNFLE1BQU0sQ0FBQ0MsSUFBSSxDQUFDO0VBQ25DQyxPQUFPLEVBQUUsQ0FBQyxHQUFHLENBQUM7RUFDZDtFQUNBQyxVQUFVLEVBQUUsQ0FBQyxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU07RUFDbEM7QUFDRixDQUFDLENBQUM7QUFFRixPQUFPLGVBQWVDLGdCQUFnQkEsQ0FBQSxFQUVwQztFQUNBLE1BQU1DLFVBQVUsR0FBR04sT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFTyxPQUFPLENBQUNDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsZ0JBQWdCLENBQUM7RUFDL0QsSUFBSSxDQUFDRixVQUFVLEVBQUU7SUFDZixNQUFNLElBQUlHLEtBQUssQ0FBQyxnQ0FBZ0MsR0FBR0YsT0FBTyxDQUFDQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0VBQ25FO0VBRUEsT0FBTyxDQUFDLE1BQU0sTUFBTSxDQUFDRixVQUFVLENBQUMsRUFBRVIsT0FBTztBQUMzQyIsImlnbm9yZUxpc3QiOltdfQ==
|
|
@@ -3,7 +3,6 @@ Object.hasOwn = Object.hasOwn || /* istanbul ignore next */function hasOwn(it, k
|
|
|
3
3
|
return Object.prototype.hasOwnProperty.call(it, key);
|
|
4
4
|
};
|
|
5
5
|
import diskFs from 'fs';
|
|
6
|
-
import { createFsRequire } from 'fs-require';
|
|
7
6
|
import { createFsFromVolume, Volume } from 'memfs';
|
|
8
7
|
import path from 'path';
|
|
9
8
|
import sourceMapSupport from 'source-map-support';
|
|
@@ -14,6 +13,7 @@ import webpack from 'webpack';
|
|
|
14
13
|
import logging from 'webpack/lib/logging/runtime.js';
|
|
15
14
|
import WebpackDevServer from 'webpack-dev-server';
|
|
16
15
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
16
|
+
import { createHybridRequire } from './createHybridRequire.js';
|
|
17
17
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
18
18
|
// run directly from node
|
|
19
19
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
@@ -36,10 +36,8 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
36
36
|
const volume = new Volume();
|
|
37
37
|
const fs = createFsFromVolume(volume);
|
|
38
38
|
ufs.use(diskFs).use(fs);
|
|
39
|
-
const fsRequire =
|
|
39
|
+
const fsRequire = createHybridRequire(ufs);
|
|
40
40
|
const readFile = promisify(ufs.readFile);
|
|
41
|
-
let server;
|
|
42
|
-
|
|
43
41
|
// Generate a temporary file so we can hot reload from the root of the application
|
|
44
42
|
function hotEntry(entryPath) {
|
|
45
43
|
// eslint-disable-next-line
|
|
@@ -231,4 +229,4 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
231
229
|
});
|
|
232
230
|
runServer();
|
|
233
231
|
}
|
|
234
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsRequire","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","server","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","next","x","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","url","endsWith","statusCode","setHeader","send","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { createFsRequire } from 'fs-require';\nimport { Server, 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, { MultiCompiler } 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 { getWebpackConfig } from './getWebpackConfig.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createFsRequire(ufs);\n  const readFile = promisify(ufs.readFile);\n  let server: Server | undefined;\n\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 = [\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  ] as const;\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"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","ignoreList":[]}
|
|
232
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","createHybridRequire","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","next","x","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","url","endsWith","statusCode","setHeader","send","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport { createFsFromVolume, Volume } from 'memfs';\nimport path from 'path';\nimport sourceMapSupport from 'source-map-support';\nimport tmp from 'tmp';\nimport { ufs } from 'unionfs';\nimport { promisify } from 'util';\nimport webpack, { type Configuration, type MultiConfiguration } from 'webpack';\nimport logging from 'webpack/lib/logging/runtime.js';\nimport WebpackDevServer from 'webpack-dev-server';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport { createHybridRequire } from './createHybridRequire.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createHybridRequire(ufs);\n\n  const readFile = promisify(ufs.readFile);\n  // Generate a temporary file so we can hot reload from the root of the application\n  function hotEntry(entryPath: string) {\n    // eslint-disable-next-line\n    // @ts-ignore for some reason it's not picking up that other options are optional\n    const generatedEntrypoint = tmp.fileSync({ postfix: '.js' });\n    diskFs.writeSync(\n      generatedEntrypoint.fd,\n      `\n  import entry from \"${path.resolve(process.cwd(), entryPath)}\";\n\n  if (import.meta.webpackHot) {\n    import.meta.webpackHot.accept();\n  }\n\n  export default entry;\n    `,\n    );\n    return generatedEntrypoint;\n  }\n\n  const webpackConfigs: Configuration[] = [\n    webpackConfig(\n      {\n        ...env,\n        entrypath: hotEntry(entrypoint).name,\n        name: 'client',\n      },\n      { mode: 'development' },\n    ),\n    webpackConfig(\n      {\n        ...env,\n        entrypath: entrypoint.replace('.tsx', '.server.tsx'),\n        name: 'server',\n        BROWSERSLIST_ENV: 'current node',\n      },\n      { mode: 'development', target: 'node' },\n    ),\n  ];\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs as unknown as MultiConfiguration);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"AAAA;AACAA,MAAM,CAACC,MAAM,GACXD,MAAM,CAACC,MAAM,IACb,0BAA2B,SAASA,MAAMA,CAACC,EAAE,EAAEC,GAAG,EAAE;EAClD,OAAOH,MAAM,CAACI,SAAS,CAACC,cAAc,CAACC,IAAI,CAACJ,EAAE,EAAEC,GAAG,CAAC;AACtD,CAAC;AAEH,OAAOI,MAAM,MAAM,IAAI;AAEvB,SAASC,kBAAkB,EAAEC,MAAM,QAAQ,OAAO;AAClD,OAAOC,IAAI,MAAM,MAAM;AACvB,OAAOC,gBAAgB,MAAM,oBAAoB;AACjD,OAAOC,GAAG,MAAM,KAAK;AACrB,SAASC,GAAG,QAAQ,SAAS;AAC7B,SAASC,SAAS,QAAQ,MAAM;AAChC,OAAOC,OAAO,MAAuD,SAAS;AAC9E,OAAOC,OAAO,MAAM,gCAAgC;AACpD,OAAOC,gBAAgB,MAAM,oBAAoB;AAEjD,OAAO,mCAAmC;AAC1C,SAASC,mBAAmB,QAAQ,0BAA0B;AAC9D,SAASC,gBAAgB,QAAQ,uBAAuB;AAGxD;AACA;AACA;AACA,IAAIC,MAAM,CAACC,IAAI,CAACC,IAAI,EAAE;EACpB,MAAMC,UAAU,GAAGC,OAAO,CAACC,IAAI,CAAC,CAAC,CAAC;EAElC,IAAI,CAACF,UAAU,EAAE;IACfG,OAAO,CAACC,GAAG,CAAC,uCAAuC,CAAC;IACpDH,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB;EAEAC,cAAc,CAACN,UAAU,CAAC;AAC5B;AAEA,IAAIO,kBAAmC,GAAGC,OAAO,CAACC,OAAO,CAAC,EAAE,CAAC;AAC7D,IAAIC,WAAW,GAAG,EAAE;AAEpB,eAAe,eAAeJ,cAAcA,CAC1CN,UAAkB,EAClBW,GAA4B,GAAG,CAAC,CAAC,EACjC;EACA,MAAMC,aAAa,GAAG,MAAMhB,gBAAgB,CAAC,CAAC;EAE9C,MAAMQ,GAAG,GAAGX,OAAO,CAACoB,SAAS,CAAC,kBAAkB,CAAC;;EAEjD;EACA,MAAMC,MAAM,GAAG,IAAI5B,MAAM,CAAC,CAAC;EAC3B,MAAM6B,EAAE,GAAG9B,kBAAkB,CAAC6B,MAAM,CAAC;EACrCxB,GAAG,CAAC0B,GAAG,CAAChC,MAAM,CAAC,CAACgC,GAAG,CAACD,EAAS,CAAC;EAE9B,MAAME,SAAS,GAAGtB,mBAAmB,CAACL,GAAG,CAAC;EAE1C,MAAM4B,QAAQ,GAAG3B,SAAS,CAACD,GAAG,CAAC4B,QAAQ,CAAC;EACxC;EACA,SAASC,QAAQA,CAACC,SAAiB,EAAE;IACnC;IACA;IACA,MAAMC,mBAAmB,GAAGhC,GAAG,CAACiC,QAAQ,CAAC;MAAEC,OAAO,EAAE;IAAM,CAAC,CAAC;IAC5DvC,MAAM,CAACwC,SAAS,CACdH,mBAAmB,CAACI,EAAE,EACtB;AACN,uBAAuBtC,IAAI,CAACsB,OAAO,CAACR,OAAO,CAACyB,GAAG,CAAC,CAAC,EAAEN,SAAS,CAAC;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,KACI,CAAC;IACD,OAAOC,mBAAmB;EAC5B;EAEA,MAAMM,cAA+B,GAAG,CACtCf,aAAa,CACX;IACE,GAAGD,GAAG;IACNiB,SAAS,EAAET,QAAQ,CAACnB,UAAU,CAAC,CAAC6B,IAAI;IACpCA,IAAI,EAAE;EACR,CAAC,EACD;IAAEC,IAAI,EAAE;EAAc,CACxB,CAAC,EACDlB,aAAa,CACX;IACE,GAAGD,GAAG;IACNiB,SAAS,EAAE5B,UAAU,CAAC+B,OAAO,CAAC,MAAM,EAAE,aAAa,CAAC;IACpDF,IAAI,EAAE,QAAQ;IACdG,gBAAgB,EAAE;EACpB,CAAC,EACD;IAAEF,IAAI,EAAE,aAAa;IAAEG,MAAM,EAAE;EAAO,CACxC,CAAC,CACF;;EAED;EACA,MAAMC,QAAQ,GAAG1C,OAAO,CAACmC,cAA+C,CAAC;EACzE,IAAI,CAACO,QAAQ,EAAE;IACb9B,GAAG,CAAC+B,KAAK,CAAC,2CAA2C,CAAC;IACtDlC,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB;EAEAjB,gBAAgB,CAACgD,OAAO,CAAC;IAAEC,WAAW,EAAE;EAAK,CAAC,CAAC;EAE/C,SAASC,eAAeA,CAACC,WAA0B,EAAE;IACnD,MAAMC,UAAU,GAAGD,WAAW,CAACE,MAAM,CAAC;MAAEC,MAAM,EAAE;IAAK,CAAC,CAAC;IACvD,OAAOvD,IAAI,CAACwD,IAAI,CAACH,UAAU,CAACI,UAAU,IAAI,EAAE,EAAE,WAAW,CAAC;EAC5D;EACA,SAASC,YAAYA,CAKnBC,EAAK,EAAE;IACP,OAAO,gBACLC,GAA8B,EAC9BC,GAA8B,EAC9BC,IAAkB,EAClB;MACA,IAAI;QACF,OAAO,MAAMH,EAAE,CAACC,GAAG,EAAEC,GAAG,CAAC;MAC3B,CAAC,CAAC,OAAOE,CAAC,EAAE;QACVD,IAAI,CAACC,CAAC,CAAC;MACT;IACF,CAAC;EACH;EAEA,IAAIC,UAES,GAAG,EAAE;EAClB,IAAIC,MAAmB,GAAGA,CAAC,GAAGC,IAAI,KAChC,IAAI7C,OAAO,CAACC,OAAO,IAAI;IACrB0C,UAAU,EAAEG,IAAI,CAAC;MAAED,IAAI;MAAE5C;IAAQ,CAAC,CAAC;EACrC,CAAC,CAAC;EAEJ,SAAS8C,YAAYA,CAACC,KAAsB,EAAE;IAC5C,MAAM,CAACC,WAAW,EAAElB,WAAW,CAAC,GAAGiB,KAAK;IACxC,IACEC,WAAW,EAAEC,WAAW,EAAEC,MAAM,EAAEC,MAAM,IACxCrB,WAAW,EAAEmB,WAAW,EAAEC,MAAM,EAAEC,MAAM,EACxC;MACAxD,GAAG,CAAC+B,KAAK,CAAC,2BAA2B,GAAGsB,WAAW,CAACC,WAAW,CAACC,MAAM,CAAC;MACvEvD,GAAG,CAAC+B,KAAK,CAAC,2BAA2B,GAAGI,WAAW,CAACmB,WAAW,CAACC,MAAM,CAAC;MACvE;MACA,IAAIE,KAAK,CAACC,OAAO,CAACX,UAAU,CAAC,EAAE;QAC7BlD,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;MAClB;MACAD,GAAG,CAAC+B,KAAK,CAAC,uCAAuC,CAAC;MAClD;IACF,CAAC,MAAM;MACL/B,GAAG,CAAC2D,IAAI,CAAC,eAAe,CAAC;IAC3B;;IAEA;IACA,MAAMC,cAAc,GAAGP,WAAW,CAAChB,MAAM,CAAC,CAAC;IAE3C/B,WAAW,GAAG4B,eAAe,CAACC,WAAW,CAAC;IAC1ChC,kBAAkB,GAAGW,QAAQ,CAACR,WAAW,CAAC,CAACuD,IAAI,CAACC,GAAG,IAAIA,GAAG,CAACC,QAAQ,CAAC,CAAC,CAAC;IACtE;IACA1F,MAAM,CAAC2F,IAAI,CAACnD,SAAS,CAACoD,KAAK,CAAC,CAACC,OAAO,CAAC1F,GAAG,IAAI;MAC1C,OAAOqC,SAAS,CAACoD,KAAK,CAACzF,GAAG,CAAC;IAC7B,CAAC,CAAC;IACFwE,MAAM,GAAInC,SAAS,CAACP,WAAW,CAAC,CAAS6D,OAAO,CAACC,IAAI,CACnDC,SAAS,EACTT,cACF,CAAC;IACD;IACA,IAAIH,KAAK,CAACC,OAAO,CAACX,UAAU,CAAC,EAAE;MAC7BA,UAAU,CAACmB,OAAO,CAAC,MAAMI,IAAI,IAAI;QAC/B,IAAI;UACFtE,GAAG,CAAC2D,IAAI,CAAC,2BAA2B,CAAC;UACrC,MAAMX,MAAM,CAAC,GAAGsB,IAAI,CAACrB,IAAI,CAAC;UAC1BqB,IAAI,CAACjE,OAAO,CAAC,CAAC;QAChB,CAAC,CAAC,OAAOkE,CAAC,EAAE;UACVvE,GAAG,CAAC+B,KAAK,CAAC,iDAAiD,CAAC;UAC5D/B,GAAG,CAAC+B,KAAK,CAACwC,CAAC,CAAC;QACd;MACF,CAAC,CAAC;MACFxB,UAAU,GAAGsB,SAAS;IACxB;EACF;EAEA,MAAMG,SAAS,GAAG,IAAIlF,gBAAgB;EACpC;EACA;IACE,GAAGiC,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS;IAC9BC,aAAa,EAAE;MACb,GAAGlD,cAAc,CAAC,CAAC,CAAC,EAAEiD,SAAS,EAAEC,aAAa;MAC9CC,gBAAgB,EAAE;QAChB,GAAG/D,EAAE;QACL4B,IAAI,EAAExD,IAAI,CAACwD;MACb;IACF,CAAC;IACDoC,gBAAgB,EAAEA,CAACC,WAAW,EAAEJ,SAAS,KAAK;MAC5C,IAAI,CAACA,SAAS,EAAE;QACd,MAAM,IAAIK,KAAK,CAAC,mCAAmC,CAAC;MACtD;MAEA,MAAMC,WAAW,GAAG,CAClBjF,OAAO,CAACU,GAAG,CAACwE,mBAAmB,EAC/B,IAAIxD,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,EAAEQ,KAAK,EAClCC,MAAM,CAACD,KAAK,IAAI,OAAOA,KAAK,KAAK,QAAQ,CAAC,EAC1CE,OAAO,CAACF,KAAK,IAAIA,KAAK,CAACG,OAAO,CAAC,IAAI,EAAE,CAAC,CAC3C;MACD;MACAX,SAAS,CAACY,GAAG,EAAEC,GAAG,CAChB,IAAIC,MAAM,CAAC,OAAOR,WAAW,CAACvC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAC3CE,YAAY,CAAC,gBAAgBE,GAAQ,EAAEC,GAAQ,EAAE;QAC/C,IAAID,GAAG,CAAC4C,GAAG,CAACC,QAAQ,CAAC,aAAa,CAAC,EAAE;UACnC5C,GAAG,CAAC6C,UAAU,GAAG,GAAG;UACpB7C,GAAG,CAAC8C,SAAS,CAAC,cAAc,EAAE,WAAW,CAAC;UAC1C9C,GAAG,CAAC+C,IAAI,CAAC,WAAW,CAAC;UACrB;QACF;QACA/C,GAAG,CAACgD,MAAM,CAACC,EAAE,CAAC,OAAO,EAAG9D,KAAc,IAAK;UACzC/B,GAAG,CAAC+B,KAAK,CAAC,QAAQ,EAAEA,KAAK,CAAC;UAC1B,IAAKA,KAAK,CAAS+D,IAAI,KAAK,YAAY,EAAE;YACxC9F,GAAG,CAAC+B,KAAK,CACP,6DACF,CAAC;UACH;QACF,CAAC,CAAC;QAEF,MAAMiB,MAAM,CAACL,GAAG,EAAEC,GAAG,CAAC;MACxB,CAAC,CACH,CAAC;MAED,IAAIrB,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,EAAEG,gBAAgB,EAAE;QACjD,OAAOpD,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,CAACG,gBAAgB,CACjDC,WAAW,EACXJ,SACF,CAAC;MACH;MAEA,OAAOI,WAAW;IACpB;EACF,CAAC,EACD9C,QACF,CAAC;EACD,MAAMiE,SAAS,GAAG,MAAAA,CAAA,KAAY;IAC5B,MAAMvB,SAAS,CAACwB,KAAK,CAAC,CAAC;IACvBxB,SAAS,CAAC1C,QAAQ,CAACmE,KAAK,CAACC,IAAI,CAACC,GAAG,CAC/B,eAAe,EACdC,UAA8C,IAAK;MAClD,IAAI,CAACA,UAAU,EAAE;QACfpG,GAAG,CAAC+B,KAAK,CAAC,gBAAgB,CAAC;QAC3BlC,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;MAClB;MAEA,IAAI,CAAC5B,MAAM,CAACC,MAAM,CAAC8H,UAAU,EAAE,OAAO,CAAC,EAAE;MACzC,IAAKA,UAAU,CAAwBhD,KAAK,CAACI,MAAM,GAAG,CAAC,EAAE;QACvD,IAAI;UACFL,YAAY,CAAEiD,UAAU,CAAwBhD,KAAK,CAAC;QACxD,CAAC,CAAC,OAAOmB,CAAM,EAAE;UACfvE,GAAG,CAAC+B,KAAK,CAAC,iCAAiC,CAAC;UAC5C,MAAMsE,MAAM,GAAG,IAAIf,MAAM,CAAC,GAAGhF,WAAW,oBAAoB,EAAE,GAAG,CAAC;UAClEH,kBAAkB,CAAC0D,IAAI,CAACyC,QAAQ,IAAI;YAClC,MAAMC,QAAQ,GAAGD,QAAQ,CAACE,KAAK,CAAC,IAAI,CAAC;YACrCxG,GAAG,CAAC+B,KAAK,CAAC,0CAA0C,CAAC;YACrD,KAAK,MAAM0E,KAAK,IAAIlC,CAAC,CAACmC,KAAK,CAACC,QAAQ,CAACN,MAAM,CAAC,IAAI,EAAE,EAAE;cAClD,MAAMO,GAAG,GAAGC,MAAM,CAACC,QAAQ,CAACL,KAAK,CAAC,CAAC,CAAC,CAAC;cACrC,MAAMM,GAAG,GAAGF,MAAM,CAACC,QAAQ,CAACL,KAAK,CAAC,CAAC,CAAC,CAAC;cACrCzG,GAAG,CAAC+B,KAAK,CAAChD,IAAI,CAACiI,QAAQ,CAAC1G,WAAW,CAAC,GAAG,GAAG,GAAGsG,GAAG,GAAG,GAAG,GAAGG,GAAG,CAAC;cAC7D/G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAAC0B,KAAK,CAACsD,GAAG,CAAC,CAACxE,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;cACrCvC,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,CAAC,CAAC;cACxB5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;YAC9B;YACAhI,MAAM,CAACqI,aAAa,CAAC3G,WAAW,EAAEgG,QAAQ,CAAC;UAC7C,CAAC,CAAC;UAEF,MAAM/B,CAAC;QACT;MACF,CAAC,MAAM;QACLvE,GAAG,CAAC+B,KAAK,CAAC,wBAAwB,CAAC;MACrC;IACF,CACF,CAAC;EACH,CAAC;EACD,MAAMmF,UAAU,GAAG,MAAAA,CAAA,KAAY;IAC7BlH,GAAG,CAAC2D,IAAI,CAAC,oBAAoB,CAAC;IAC9B,MAAMa,SAAS,CAAC2C,IAAI,CAAC,CAAC;IACtBnH,GAAG,CAAC2D,IAAI,CAAC,eAAe,CAAC;EAC3B,CAAC;EAED9D,OAAO,CAACgG,EAAE,CAAC,QAAQ,EAAE,MAAM;IACzB7F,GAAG,CAACoH,IAAI,CAAC,0CAA0C,CAAC;IACpDF,UAAU,CAAC,CAAC;IACZrH,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB,CAAC,CAAC;EAEF8F,SAAS,CAAC,CAAC;AACb","ignoreList":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";
|
|
1
|
+
{"version":3,"file":"startDevserver.d.ts","sourceRoot":"","sources":["../../src/scripts/startDevserver.ts"],"names":[],"mappings":";AAmBA,OAAO,mCAAmC,CAAC;AAsB3C,wBAA8B,cAAc,CAC1C,UAAU,EAAE,MAAM,EAClB,GAAG,GAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAM,iBA6PlC"}
|
|
@@ -3,7 +3,6 @@ Object.hasOwn = Object.hasOwn || /* istanbul ignore next */function hasOwn(it, k
|
|
|
3
3
|
return Object.prototype.hasOwnProperty.call(it, key);
|
|
4
4
|
};
|
|
5
5
|
import diskFs from 'fs';
|
|
6
|
-
import { createFsRequire } from 'fs-require';
|
|
7
6
|
import { createFsFromVolume, Volume } from 'memfs';
|
|
8
7
|
import path from 'path';
|
|
9
8
|
import sourceMapSupport from 'source-map-support';
|
|
@@ -14,6 +13,7 @@ import webpack from 'webpack';
|
|
|
14
13
|
import logging from 'webpack/lib/logging/runtime.js';
|
|
15
14
|
import WebpackDevServer from 'webpack-dev-server';
|
|
16
15
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
16
|
+
import { createHybridRequire } from './createHybridRequire.js';
|
|
17
17
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
18
18
|
// run directly from node
|
|
19
19
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
@@ -36,10 +36,8 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
36
36
|
const volume = new Volume();
|
|
37
37
|
const fs = createFsFromVolume(volume);
|
|
38
38
|
ufs.use(diskFs).use(fs);
|
|
39
|
-
const fsRequire =
|
|
39
|
+
const fsRequire = createHybridRequire(ufs);
|
|
40
40
|
const readFile = promisify(ufs.readFile);
|
|
41
|
-
let server;
|
|
42
|
-
|
|
43
41
|
// Generate a temporary file so we can hot reload from the root of the application
|
|
44
42
|
function hotEntry(entryPath) {
|
|
45
43
|
// eslint-disable-next-line
|
|
@@ -231,4 +229,4 @@ export default async function startDevServer(entrypoint, env = {}) {
|
|
|
231
229
|
});
|
|
232
230
|
runServer();
|
|
233
231
|
}
|
|
234
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsRequire","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","server","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","next","x","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","url","endsWith","statusCode","setHeader","send","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { createFsRequire } from 'fs-require';\nimport { Server, 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, { MultiCompiler } 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 { getWebpackConfig } from './getWebpackConfig.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createFsRequire(ufs);\n  const readFile = promisify(ufs.readFile);\n  let server: Server | undefined;\n\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 = [\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  ] as const;\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"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","ignoreList":[]}
|
|
232
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Object","hasOwn","it","key","prototype","hasOwnProperty","call","diskFs","createFsFromVolume","Volume","path","sourceMapSupport","tmp","ufs","promisify","webpack","logging","WebpackDevServer","createHybridRequire","getWebpackConfig","import","meta","main","entrypoint","process","argv","console","log","exit","startDevServer","serverFileContents","Promise","resolve","serverEntry","env","webpackConfig","getLogger","volume","fs","use","fsRequire","readFile","hotEntry","entryPath","generatedEntrypoint","fileSync","postfix","writeSync","fd","cwd","webpackConfigs","entrypath","name","mode","replace","BROWSERSLIST_ENV","target","compiler","error","install","hookRequire","getServerBundle","serverStats","serverJson","toJson","assets","join","outputPath","handleErrors","fn","req","res","next","x","initRender","render","args","push","importRender","stats","clientStats","compilation","errors","length","Array","isArray","info","clientManifest","then","buf","toString","keys","cache","forEach","default","bind","undefined","init","e","devServer","devMiddleware","outputFileSystem","setupMiddlewares","middlewares","Error","otherRoutes","WEBPACK_PUBLIC_PATH","proxy","filter","flatMap","context","app","get","RegExp","url","endsWith","statusCode","setHeader","send","socket","on","code","runServer","start","hooks","done","tap","multiStats","finder","fileText","textRows","split","match","stack","matchAll","row","Number","parseInt","col","basename","writeFileSync","stopServer","stop","warn"],"sources":["../../src/scripts/startDevserver.ts"],"sourcesContent":["#!/usr/bin/env node\nObject.hasOwn =\n  Object.hasOwn ||\n  /* istanbul ignore next */ function hasOwn(it, key) {\n    return Object.prototype.hasOwnProperty.call(it, key);\n  };\nimport type { NextFunction } from 'express';\nimport diskFs from 'fs';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport { createFsFromVolume, Volume } from 'memfs';\nimport path from 'path';\nimport sourceMapSupport from 'source-map-support';\nimport tmp from 'tmp';\nimport { ufs } from 'unionfs';\nimport { promisify } from 'util';\nimport webpack, { type Configuration, type MultiConfiguration } from 'webpack';\nimport logging from 'webpack/lib/logging/runtime.js';\nimport WebpackDevServer from 'webpack-dev-server';\n\nimport 'cross-fetch/dist/node-polyfill.js';\nimport { createHybridRequire } from './createHybridRequire.js';\nimport { getWebpackConfig } from './getWebpackConfig.js';\nimport { BoundRender } from './types.js';\n\n// run directly from node\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\nif (import.meta.main) {\n  const entrypoint = process.argv[2];\n\n  if (!entrypoint) {\n    console.log(`Usage: start-anansi <entrypoint-file>`);\n    process.exit(-1);\n  }\n\n  startDevServer(entrypoint);\n}\n\nlet serverFileContents: Promise<string> = Promise.resolve('');\nlet serverEntry = '';\n\nexport default async function startDevServer(\n  entrypoint: string,\n  env: Record<string, unknown> = {},\n) {\n  const webpackConfig = await getWebpackConfig();\n\n  const log = logging.getLogger('anansi-devserver');\n\n  // Set up in memory filesystem\n  const volume = new Volume();\n  const fs = createFsFromVolume(volume);\n  ufs.use(diskFs).use(fs as any);\n\n  const fsRequire = createHybridRequire(ufs);\n\n  const readFile = promisify(ufs.readFile);\n  // Generate a temporary file so we can hot reload from the root of the application\n  function hotEntry(entryPath: string) {\n    // eslint-disable-next-line\n    // @ts-ignore for some reason it's not picking up that other options are optional\n    const generatedEntrypoint = tmp.fileSync({ postfix: '.js' });\n    diskFs.writeSync(\n      generatedEntrypoint.fd,\n      `\n  import entry from \"${path.resolve(process.cwd(), entryPath)}\";\n\n  if (import.meta.webpackHot) {\n    import.meta.webpackHot.accept();\n  }\n\n  export default entry;\n    `,\n    );\n    return generatedEntrypoint;\n  }\n\n  const webpackConfigs: Configuration[] = [\n    webpackConfig(\n      {\n        ...env,\n        entrypath: hotEntry(entrypoint).name,\n        name: 'client',\n      },\n      { mode: 'development' },\n    ),\n    webpackConfig(\n      {\n        ...env,\n        entrypath: entrypoint.replace('.tsx', '.server.tsx'),\n        name: 'server',\n        BROWSERSLIST_ENV: 'current node',\n      },\n      { mode: 'development', target: 'node' },\n    ),\n  ];\n\n  // initialize the webpack compiler\n  const compiler = webpack(webpackConfigs as unknown as MultiConfiguration);\n  if (!compiler) {\n    log.error('Failed to initialize the webpack compiler');\n    process.exit(-1);\n  }\n\n  sourceMapSupport.install({ hookRequire: true });\n\n  function getServerBundle(serverStats: webpack.Stats) {\n    const serverJson = serverStats.toJson({ assets: true });\n    return path.join(serverJson.outputPath ?? '', 'server.js');\n  }\n  function handleErrors<\n    F extends (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n    ) => Promise<void>,\n  >(fn: F) {\n    return async function (\n      req: Request | IncomingMessage,\n      res: Response | ServerResponse,\n      next: NextFunction,\n    ) {\n      try {\n        return await fn(req, res);\n      } catch (x) {\n        next(x);\n      }\n    };\n  }\n\n  let initRender:\n    | { args: Parameters<BoundRender>; resolve: () => void }[]\n    | undefined = [];\n  let render: BoundRender = (...args) =>\n    new Promise(resolve => {\n      initRender?.push({ args, resolve });\n    });\n\n  function importRender(stats: webpack.Stats[]) {\n    const [clientStats, serverStats] = stats;\n    if (\n      clientStats?.compilation?.errors?.length ||\n      serverStats?.compilation?.errors?.length\n    ) {\n      log.error('Errors for client build: ' + clientStats.compilation.errors);\n      log.error('Errors for server build: ' + serverStats.compilation.errors);\n      // first time, rather than re-render\n      if (Array.isArray(initRender)) {\n        process.exit(-1);\n      }\n      log.error('Above compiler errors blocking reload');\n      return;\n    } else {\n      log.info('Launching SSR');\n    }\n\n    // ASSETS\n    const clientManifest = clientStats.toJson();\n\n    serverEntry = getServerBundle(serverStats);\n    serverFileContents = readFile(serverEntry).then(buf => buf.toString());\n    // reload modules\n    Object.keys(fsRequire.cache).forEach(key => {\n      delete fsRequire.cache[key];\n    });\n    render = (fsRequire(serverEntry) as any).default.bind(\n      undefined,\n      clientManifest,\n    );\n    // SERVER SIDE ENTRYPOINT\n    if (Array.isArray(initRender)) {\n      initRender.forEach(async init => {\n        try {\n          log.info('Resolving queued requests');\n          await render(...init.args);\n          init.resolve();\n        } catch (e) {\n          log.error('Error when attempting to render queued requests');\n          log.error(e);\n        }\n      });\n      initRender = undefined;\n    }\n  }\n\n  const devServer = new WebpackDevServer(\n    // write to memory filesystem so we can import\n    {\n      ...webpackConfigs[0].devServer,\n      devMiddleware: {\n        ...webpackConfigs[0]?.devServer?.devMiddleware,\n        outputFileSystem: {\n          ...fs,\n          join: path.join as any,\n        } as any,\n      },\n      setupMiddlewares: (middlewares, devServer) => {\n        if (!devServer) {\n          throw new Error('webpack-dev-server is not defined');\n        }\n\n        const otherRoutes = [\n          process.env.WEBPACK_PUBLIC_PATH,\n          ...(webpackConfigs[0].devServer?.proxy\n            ?.filter(proxy => typeof proxy === 'object')\n            ?.flatMap(proxy => proxy.context) ?? []),\n        ];\n        // serve SSR for non-WEBPACK_PUBLIC_PATH\n        devServer.app?.get(\n          new RegExp(`^(?!${otherRoutes.join('|')})`),\n          handleErrors(async function (req: any, res: any) {\n            if (req.url.endsWith('favicon.ico')) {\n              res.statusCode = 404;\n              res.setHeader('Content-type', 'text/html');\n              res.send('not found');\n              return;\n            }\n            res.socket.on('error', (error: unknown) => {\n              log.error('Fatal:', error);\n              if ((error as any).code === 'ECONNRESET') {\n                log.error(\n                  'ECONNRESET is usually due to browser closing the connection',\n                );\n              }\n            });\n\n            await render(req, res);\n          }),\n        );\n\n        if (webpackConfigs[0].devServer?.setupMiddlewares) {\n          return webpackConfigs[0].devServer.setupMiddlewares(\n            middlewares,\n            devServer,\n          );\n        }\n\n        return middlewares;\n      },\n    },\n    compiler,\n  );\n  const runServer = async () => {\n    await devServer.start();\n    devServer.compiler.hooks.done.tap(\n      'Anansi Server',\n      (multiStats: webpack.MultiStats | webpack.Stats) => {\n        if (!multiStats) {\n          log.error('stats not send');\n          process.exit(-1);\n        }\n\n        if (!Object.hasOwn(multiStats, 'stats')) return;\n        if ((multiStats as webpack.MultiStats).stats.length > 1) {\n          try {\n            importRender((multiStats as webpack.MultiStats).stats);\n          } catch (e: any) {\n            log.error('Failed to load serve entrypoint');\n            const finder = new RegExp(`${serverEntry}:([\\\\d]+):([\\\\d]+)`, 'g');\n            serverFileContents.then(fileText => {\n              const textRows = fileText.split('\\n');\n              log.error('>>> Stack Context [serve entrypoint] <<<');\n              for (const match of e.stack.matchAll(finder) ?? []) {\n                const row = Number.parseInt(match[1]);\n                const col = Number.parseInt(match[2]);\n                log.error(path.basename(serverEntry) + ' ' + row + ':' + col);\n                log.error(textRows[row - 2]);\n                log.error(textRows[row - 1]);\n                log.error(Array(col).join(' ') + '^');\n                log.error(textRows[row]);\n                log.error(textRows[row + 1]);\n                log.error(textRows[row + 2]);\n              }\n              diskFs.writeFileSync(serverEntry, fileText);\n            });\n\n            throw e;\n          }\n        } else {\n          log.error('Only compiler one stat');\n        }\n      },\n    );\n  };\n  const stopServer = async () => {\n    log.info('Stopping server...');\n    await devServer.stop();\n    log.info('Server closed');\n  };\n\n  process.on('SIGINT', () => {\n    log.warn('Received SIGINT, devserver shutting down');\n    stopServer();\n    process.exit(-1);\n  });\n\n  runServer();\n}\n"],"mappings":"AAAA;AACAA,MAAM,CAACC,MAAM,GACXD,MAAM,CAACC,MAAM,IACb,0BAA2B,SAASA,MAAMA,CAACC,EAAE,EAAEC,GAAG,EAAE;EAClD,OAAOH,MAAM,CAACI,SAAS,CAACC,cAAc,CAACC,IAAI,CAACJ,EAAE,EAAEC,GAAG,CAAC;AACtD,CAAC;AAEH,OAAOI,MAAM,MAAM,IAAI;AAEvB,SAASC,kBAAkB,EAAEC,MAAM,QAAQ,OAAO;AAClD,OAAOC,IAAI,MAAM,MAAM;AACvB,OAAOC,gBAAgB,MAAM,oBAAoB;AACjD,OAAOC,GAAG,MAAM,KAAK;AACrB,SAASC,GAAG,QAAQ,SAAS;AAC7B,SAASC,SAAS,QAAQ,MAAM;AAChC,OAAOC,OAAO,MAAuD,SAAS;AAC9E,OAAOC,OAAO,MAAM,gCAAgC;AACpD,OAAOC,gBAAgB,MAAM,oBAAoB;AAEjD,OAAO,mCAAmC;AAC1C,SAASC,mBAAmB,QAAQ,0BAA0B;AAC9D,SAASC,gBAAgB,QAAQ,uBAAuB;AAGxD;AACA;AACA;AACA,IAAIC,MAAM,CAACC,IAAI,CAACC,IAAI,EAAE;EACpB,MAAMC,UAAU,GAAGC,OAAO,CAACC,IAAI,CAAC,CAAC,CAAC;EAElC,IAAI,CAACF,UAAU,EAAE;IACfG,OAAO,CAACC,GAAG,CAAC,uCAAuC,CAAC;IACpDH,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB;EAEAC,cAAc,CAACN,UAAU,CAAC;AAC5B;AAEA,IAAIO,kBAAmC,GAAGC,OAAO,CAACC,OAAO,CAAC,EAAE,CAAC;AAC7D,IAAIC,WAAW,GAAG,EAAE;AAEpB,eAAe,eAAeJ,cAAcA,CAC1CN,UAAkB,EAClBW,GAA4B,GAAG,CAAC,CAAC,EACjC;EACA,MAAMC,aAAa,GAAG,MAAMhB,gBAAgB,CAAC,CAAC;EAE9C,MAAMQ,GAAG,GAAGX,OAAO,CAACoB,SAAS,CAAC,kBAAkB,CAAC;;EAEjD;EACA,MAAMC,MAAM,GAAG,IAAI5B,MAAM,CAAC,CAAC;EAC3B,MAAM6B,EAAE,GAAG9B,kBAAkB,CAAC6B,MAAM,CAAC;EACrCxB,GAAG,CAAC0B,GAAG,CAAChC,MAAM,CAAC,CAACgC,GAAG,CAACD,EAAS,CAAC;EAE9B,MAAME,SAAS,GAAGtB,mBAAmB,CAACL,GAAG,CAAC;EAE1C,MAAM4B,QAAQ,GAAG3B,SAAS,CAACD,GAAG,CAAC4B,QAAQ,CAAC;EACxC;EACA,SAASC,QAAQA,CAACC,SAAiB,EAAE;IACnC;IACA;IACA,MAAMC,mBAAmB,GAAGhC,GAAG,CAACiC,QAAQ,CAAC;MAAEC,OAAO,EAAE;IAAM,CAAC,CAAC;IAC5DvC,MAAM,CAACwC,SAAS,CACdH,mBAAmB,CAACI,EAAE,EACtB;AACN,uBAAuBtC,IAAI,CAACsB,OAAO,CAACR,OAAO,CAACyB,GAAG,CAAC,CAAC,EAAEN,SAAS,CAAC;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,KACI,CAAC;IACD,OAAOC,mBAAmB;EAC5B;EAEA,MAAMM,cAA+B,GAAG,CACtCf,aAAa,CACX;IACE,GAAGD,GAAG;IACNiB,SAAS,EAAET,QAAQ,CAACnB,UAAU,CAAC,CAAC6B,IAAI;IACpCA,IAAI,EAAE;EACR,CAAC,EACD;IAAEC,IAAI,EAAE;EAAc,CACxB,CAAC,EACDlB,aAAa,CACX;IACE,GAAGD,GAAG;IACNiB,SAAS,EAAE5B,UAAU,CAAC+B,OAAO,CAAC,MAAM,EAAE,aAAa,CAAC;IACpDF,IAAI,EAAE,QAAQ;IACdG,gBAAgB,EAAE;EACpB,CAAC,EACD;IAAEF,IAAI,EAAE,aAAa;IAAEG,MAAM,EAAE;EAAO,CACxC,CAAC,CACF;;EAED;EACA,MAAMC,QAAQ,GAAG1C,OAAO,CAACmC,cAA+C,CAAC;EACzE,IAAI,CAACO,QAAQ,EAAE;IACb9B,GAAG,CAAC+B,KAAK,CAAC,2CAA2C,CAAC;IACtDlC,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB;EAEAjB,gBAAgB,CAACgD,OAAO,CAAC;IAAEC,WAAW,EAAE;EAAK,CAAC,CAAC;EAE/C,SAASC,eAAeA,CAACC,WAA0B,EAAE;IACnD,MAAMC,UAAU,GAAGD,WAAW,CAACE,MAAM,CAAC;MAAEC,MAAM,EAAE;IAAK,CAAC,CAAC;IACvD,OAAOvD,IAAI,CAACwD,IAAI,CAACH,UAAU,CAACI,UAAU,IAAI,EAAE,EAAE,WAAW,CAAC;EAC5D;EACA,SAASC,YAAYA,CAKnBC,EAAK,EAAE;IACP,OAAO,gBACLC,GAA8B,EAC9BC,GAA8B,EAC9BC,IAAkB,EAClB;MACA,IAAI;QACF,OAAO,MAAMH,EAAE,CAACC,GAAG,EAAEC,GAAG,CAAC;MAC3B,CAAC,CAAC,OAAOE,CAAC,EAAE;QACVD,IAAI,CAACC,CAAC,CAAC;MACT;IACF,CAAC;EACH;EAEA,IAAIC,UAES,GAAG,EAAE;EAClB,IAAIC,MAAmB,GAAGA,CAAC,GAAGC,IAAI,KAChC,IAAI7C,OAAO,CAACC,OAAO,IAAI;IACrB0C,UAAU,EAAEG,IAAI,CAAC;MAAED,IAAI;MAAE5C;IAAQ,CAAC,CAAC;EACrC,CAAC,CAAC;EAEJ,SAAS8C,YAAYA,CAACC,KAAsB,EAAE;IAC5C,MAAM,CAACC,WAAW,EAAElB,WAAW,CAAC,GAAGiB,KAAK;IACxC,IACEC,WAAW,EAAEC,WAAW,EAAEC,MAAM,EAAEC,MAAM,IACxCrB,WAAW,EAAEmB,WAAW,EAAEC,MAAM,EAAEC,MAAM,EACxC;MACAxD,GAAG,CAAC+B,KAAK,CAAC,2BAA2B,GAAGsB,WAAW,CAACC,WAAW,CAACC,MAAM,CAAC;MACvEvD,GAAG,CAAC+B,KAAK,CAAC,2BAA2B,GAAGI,WAAW,CAACmB,WAAW,CAACC,MAAM,CAAC;MACvE;MACA,IAAIE,KAAK,CAACC,OAAO,CAACX,UAAU,CAAC,EAAE;QAC7BlD,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;MAClB;MACAD,GAAG,CAAC+B,KAAK,CAAC,uCAAuC,CAAC;MAClD;IACF,CAAC,MAAM;MACL/B,GAAG,CAAC2D,IAAI,CAAC,eAAe,CAAC;IAC3B;;IAEA;IACA,MAAMC,cAAc,GAAGP,WAAW,CAAChB,MAAM,CAAC,CAAC;IAE3C/B,WAAW,GAAG4B,eAAe,CAACC,WAAW,CAAC;IAC1ChC,kBAAkB,GAAGW,QAAQ,CAACR,WAAW,CAAC,CAACuD,IAAI,CAACC,GAAG,IAAIA,GAAG,CAACC,QAAQ,CAAC,CAAC,CAAC;IACtE;IACA1F,MAAM,CAAC2F,IAAI,CAACnD,SAAS,CAACoD,KAAK,CAAC,CAACC,OAAO,CAAC1F,GAAG,IAAI;MAC1C,OAAOqC,SAAS,CAACoD,KAAK,CAACzF,GAAG,CAAC;IAC7B,CAAC,CAAC;IACFwE,MAAM,GAAInC,SAAS,CAACP,WAAW,CAAC,CAAS6D,OAAO,CAACC,IAAI,CACnDC,SAAS,EACTT,cACF,CAAC;IACD;IACA,IAAIH,KAAK,CAACC,OAAO,CAACX,UAAU,CAAC,EAAE;MAC7BA,UAAU,CAACmB,OAAO,CAAC,MAAMI,IAAI,IAAI;QAC/B,IAAI;UACFtE,GAAG,CAAC2D,IAAI,CAAC,2BAA2B,CAAC;UACrC,MAAMX,MAAM,CAAC,GAAGsB,IAAI,CAACrB,IAAI,CAAC;UAC1BqB,IAAI,CAACjE,OAAO,CAAC,CAAC;QAChB,CAAC,CAAC,OAAOkE,CAAC,EAAE;UACVvE,GAAG,CAAC+B,KAAK,CAAC,iDAAiD,CAAC;UAC5D/B,GAAG,CAAC+B,KAAK,CAACwC,CAAC,CAAC;QACd;MACF,CAAC,CAAC;MACFxB,UAAU,GAAGsB,SAAS;IACxB;EACF;EAEA,MAAMG,SAAS,GAAG,IAAIlF,gBAAgB;EACpC;EACA;IACE,GAAGiC,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS;IAC9BC,aAAa,EAAE;MACb,GAAGlD,cAAc,CAAC,CAAC,CAAC,EAAEiD,SAAS,EAAEC,aAAa;MAC9CC,gBAAgB,EAAE;QAChB,GAAG/D,EAAE;QACL4B,IAAI,EAAExD,IAAI,CAACwD;MACb;IACF,CAAC;IACDoC,gBAAgB,EAAEA,CAACC,WAAW,EAAEJ,SAAS,KAAK;MAC5C,IAAI,CAACA,SAAS,EAAE;QACd,MAAM,IAAIK,KAAK,CAAC,mCAAmC,CAAC;MACtD;MAEA,MAAMC,WAAW,GAAG,CAClBjF,OAAO,CAACU,GAAG,CAACwE,mBAAmB,EAC/B,IAAIxD,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,EAAEQ,KAAK,EAClCC,MAAM,CAACD,KAAK,IAAI,OAAOA,KAAK,KAAK,QAAQ,CAAC,EAC1CE,OAAO,CAACF,KAAK,IAAIA,KAAK,CAACG,OAAO,CAAC,IAAI,EAAE,CAAC,CAC3C;MACD;MACAX,SAAS,CAACY,GAAG,EAAEC,GAAG,CAChB,IAAIC,MAAM,CAAC,OAAOR,WAAW,CAACvC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAC3CE,YAAY,CAAC,gBAAgBE,GAAQ,EAAEC,GAAQ,EAAE;QAC/C,IAAID,GAAG,CAAC4C,GAAG,CAACC,QAAQ,CAAC,aAAa,CAAC,EAAE;UACnC5C,GAAG,CAAC6C,UAAU,GAAG,GAAG;UACpB7C,GAAG,CAAC8C,SAAS,CAAC,cAAc,EAAE,WAAW,CAAC;UAC1C9C,GAAG,CAAC+C,IAAI,CAAC,WAAW,CAAC;UACrB;QACF;QACA/C,GAAG,CAACgD,MAAM,CAACC,EAAE,CAAC,OAAO,EAAG9D,KAAc,IAAK;UACzC/B,GAAG,CAAC+B,KAAK,CAAC,QAAQ,EAAEA,KAAK,CAAC;UAC1B,IAAKA,KAAK,CAAS+D,IAAI,KAAK,YAAY,EAAE;YACxC9F,GAAG,CAAC+B,KAAK,CACP,6DACF,CAAC;UACH;QACF,CAAC,CAAC;QAEF,MAAMiB,MAAM,CAACL,GAAG,EAAEC,GAAG,CAAC;MACxB,CAAC,CACH,CAAC;MAED,IAAIrB,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,EAAEG,gBAAgB,EAAE;QACjD,OAAOpD,cAAc,CAAC,CAAC,CAAC,CAACiD,SAAS,CAACG,gBAAgB,CACjDC,WAAW,EACXJ,SACF,CAAC;MACH;MAEA,OAAOI,WAAW;IACpB;EACF,CAAC,EACD9C,QACF,CAAC;EACD,MAAMiE,SAAS,GAAG,MAAAA,CAAA,KAAY;IAC5B,MAAMvB,SAAS,CAACwB,KAAK,CAAC,CAAC;IACvBxB,SAAS,CAAC1C,QAAQ,CAACmE,KAAK,CAACC,IAAI,CAACC,GAAG,CAC/B,eAAe,EACdC,UAA8C,IAAK;MAClD,IAAI,CAACA,UAAU,EAAE;QACfpG,GAAG,CAAC+B,KAAK,CAAC,gBAAgB,CAAC;QAC3BlC,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;MAClB;MAEA,IAAI,CAAC5B,MAAM,CAACC,MAAM,CAAC8H,UAAU,EAAE,OAAO,CAAC,EAAE;MACzC,IAAKA,UAAU,CAAwBhD,KAAK,CAACI,MAAM,GAAG,CAAC,EAAE;QACvD,IAAI;UACFL,YAAY,CAAEiD,UAAU,CAAwBhD,KAAK,CAAC;QACxD,CAAC,CAAC,OAAOmB,CAAM,EAAE;UACfvE,GAAG,CAAC+B,KAAK,CAAC,iCAAiC,CAAC;UAC5C,MAAMsE,MAAM,GAAG,IAAIf,MAAM,CAAC,GAAGhF,WAAW,oBAAoB,EAAE,GAAG,CAAC;UAClEH,kBAAkB,CAAC0D,IAAI,CAACyC,QAAQ,IAAI;YAClC,MAAMC,QAAQ,GAAGD,QAAQ,CAACE,KAAK,CAAC,IAAI,CAAC;YACrCxG,GAAG,CAAC+B,KAAK,CAAC,0CAA0C,CAAC;YACrD,KAAK,MAAM0E,KAAK,IAAIlC,CAAC,CAACmC,KAAK,CAACC,QAAQ,CAACN,MAAM,CAAC,IAAI,EAAE,EAAE;cAClD,MAAMO,GAAG,GAAGC,MAAM,CAACC,QAAQ,CAACL,KAAK,CAAC,CAAC,CAAC,CAAC;cACrC,MAAMM,GAAG,GAAGF,MAAM,CAACC,QAAQ,CAACL,KAAK,CAAC,CAAC,CAAC,CAAC;cACrCzG,GAAG,CAAC+B,KAAK,CAAChD,IAAI,CAACiI,QAAQ,CAAC1G,WAAW,CAAC,GAAG,GAAG,GAAGsG,GAAG,GAAG,GAAG,GAAGG,GAAG,CAAC;cAC7D/G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAAC0B,KAAK,CAACsD,GAAG,CAAC,CAACxE,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;cACrCvC,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,CAAC,CAAC;cACxB5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;cAC5B5G,GAAG,CAAC+B,KAAK,CAACwE,QAAQ,CAACK,GAAG,GAAG,CAAC,CAAC,CAAC;YAC9B;YACAhI,MAAM,CAACqI,aAAa,CAAC3G,WAAW,EAAEgG,QAAQ,CAAC;UAC7C,CAAC,CAAC;UAEF,MAAM/B,CAAC;QACT;MACF,CAAC,MAAM;QACLvE,GAAG,CAAC+B,KAAK,CAAC,wBAAwB,CAAC;MACrC;IACF,CACF,CAAC;EACH,CAAC;EACD,MAAMmF,UAAU,GAAG,MAAAA,CAAA,KAAY;IAC7BlH,GAAG,CAAC2D,IAAI,CAAC,oBAAoB,CAAC;IAC9B,MAAMa,SAAS,CAAC2C,IAAI,CAAC,CAAC;IACtBnH,GAAG,CAAC2D,IAAI,CAAC,eAAe,CAAC;EAC3B,CAAC;EAED9D,OAAO,CAACgG,EAAE,CAAC,QAAQ,EAAE,MAAM;IACzB7F,GAAG,CAACoH,IAAI,CAAC,0CAA0C,CAAC;IACpDF,UAAU,CAAC,CAAC;IACZrH,OAAO,CAACI,IAAI,CAAC,CAAC,CAAC,CAAC;EAClB,CAAC,CAAC;EAEF8F,SAAS,CAAC,CAAC;AACb","ignoreList":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@anansi/core",
|
|
3
|
-
"version": "0.20.
|
|
3
|
+
"version": "0.20.40",
|
|
4
4
|
"description": "React 19 Framework",
|
|
5
5
|
"homepage": "https://github.com/ntucker/anansi/tree/master/packages/core#readme",
|
|
6
6
|
"repository": {
|
|
@@ -71,11 +71,11 @@
|
|
|
71
71
|
"@types/compression": "1.8.1",
|
|
72
72
|
"@types/express": "^4.17.17",
|
|
73
73
|
"@types/node": "^24.0.0",
|
|
74
|
-
"@types/react": "19.2.
|
|
75
|
-
"@types/react-dom": "19.2.
|
|
74
|
+
"@types/react": "19.2.5",
|
|
75
|
+
"@types/react-dom": "19.2.3",
|
|
76
76
|
"@types/source-map-support": "0.5.10",
|
|
77
77
|
"@types/tmp": "0.2.6",
|
|
78
|
-
"@types/webpack-hot-middleware": "2.25.
|
|
78
|
+
"@types/webpack-hot-middleware": "2.25.12",
|
|
79
79
|
"@types/webpack-node-externals": "^3",
|
|
80
80
|
"react": "19.2.0",
|
|
81
81
|
"react-dom": "19.2.0"
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import diskFs from 'fs';
|
|
2
|
+
import { createFsFromVolume, Volume } from 'memfs';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
import tmp from 'tmp';
|
|
5
|
+
import { ufs } from 'unionfs';
|
|
6
|
+
|
|
7
|
+
import { createHybridRequire } from '../createHybridRequire';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Regression test to ensure hybridRequire resolves from the initial
|
|
11
|
+
* working directory even if process.cwd() changes after initialization.
|
|
12
|
+
* This guards against future refactors that might call process.chdir.
|
|
13
|
+
*/
|
|
14
|
+
describe('hybridRequire - process.cwd() variance regression', () => {
|
|
15
|
+
let tempDir: string;
|
|
16
|
+
let originalCwd: string;
|
|
17
|
+
|
|
18
|
+
beforeEach(() => {
|
|
19
|
+
originalCwd = process.cwd();
|
|
20
|
+
tempDir = tmp.dirSync({ unsafeCleanup: true }).name;
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
afterEach(() => {
|
|
24
|
+
process.chdir(originalCwd);
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
it('should capture projectRequire path at initialization time', () => {
|
|
28
|
+
const projectRoot = path.join(tempDir, 'project');
|
|
29
|
+
const otherDir = path.join(tempDir, 'other');
|
|
30
|
+
|
|
31
|
+
// Create project structure
|
|
32
|
+
diskFs.mkdirSync(projectRoot, { recursive: true });
|
|
33
|
+
diskFs.mkdirSync(otherDir, { recursive: true });
|
|
34
|
+
|
|
35
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
36
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package-cwd');
|
|
37
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
38
|
+
|
|
39
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
40
|
+
diskFs.writeFileSync(
|
|
41
|
+
testFile,
|
|
42
|
+
"module.exports = { default: 'from-project-root' };",
|
|
43
|
+
);
|
|
44
|
+
diskFs.writeFileSync(
|
|
45
|
+
path.join(testPackageDir, 'package.json'),
|
|
46
|
+
JSON.stringify({ name: 'test-package-cwd', main: 'index.js' }),
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
diskFs.writeFileSync(
|
|
50
|
+
path.join(projectRoot, 'package.json'),
|
|
51
|
+
JSON.stringify({ name: 'project' }),
|
|
52
|
+
);
|
|
53
|
+
|
|
54
|
+
// Create another package.json in otherDir (should not be used)
|
|
55
|
+
diskFs.writeFileSync(
|
|
56
|
+
path.join(otherDir, 'package.json'),
|
|
57
|
+
JSON.stringify({ name: 'other' }),
|
|
58
|
+
);
|
|
59
|
+
|
|
60
|
+
// Set up memfs WITHOUT diskFs to test fallback
|
|
61
|
+
const volume = new Volume();
|
|
62
|
+
const fs = createFsFromVolume(volume);
|
|
63
|
+
ufs.use(fs as any);
|
|
64
|
+
|
|
65
|
+
// Initialize hybridRequire while in projectRoot
|
|
66
|
+
process.chdir(projectRoot);
|
|
67
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
68
|
+
|
|
69
|
+
// Change to other directory
|
|
70
|
+
process.chdir(otherDir);
|
|
71
|
+
|
|
72
|
+
// Verify process.cwd() changed
|
|
73
|
+
expect(process.cwd()).toBe(otherDir);
|
|
74
|
+
|
|
75
|
+
// hybridRequire should still resolve from projectRoot (captured at init)
|
|
76
|
+
const result = hybridRequire('test-package-cwd');
|
|
77
|
+
expect(result.default).toBe('from-project-root');
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('should work correctly when process.cwd() is different from projectRoot', () => {
|
|
81
|
+
const projectRoot = path.join(tempDir, 'workspace', 'examples', 'app');
|
|
82
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
83
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package-nested');
|
|
84
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
85
|
+
|
|
86
|
+
// Create nested structure
|
|
87
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
88
|
+
diskFs.writeFileSync(
|
|
89
|
+
testFile,
|
|
90
|
+
"module.exports = { default: 'from-nested-workspace' };",
|
|
91
|
+
);
|
|
92
|
+
diskFs.writeFileSync(
|
|
93
|
+
path.join(testPackageDir, 'package.json'),
|
|
94
|
+
JSON.stringify({ name: 'test-package-nested', main: 'index.js' }),
|
|
95
|
+
);
|
|
96
|
+
|
|
97
|
+
diskFs.writeFileSync(
|
|
98
|
+
path.join(projectRoot, 'package.json'),
|
|
99
|
+
JSON.stringify({ name: 'app' }),
|
|
100
|
+
);
|
|
101
|
+
|
|
102
|
+
// Set up memfs WITHOUT diskFs to test fallback
|
|
103
|
+
const volume = new Volume();
|
|
104
|
+
const fs = createFsFromVolume(volume);
|
|
105
|
+
ufs.use(fs as any);
|
|
106
|
+
|
|
107
|
+
// Initialize from a different directory (simulating monorepo root)
|
|
108
|
+
const monorepoRoot = path.join(tempDir, 'workspace');
|
|
109
|
+
process.chdir(monorepoRoot);
|
|
110
|
+
|
|
111
|
+
// Create hybridRequire with explicit projectRoot (not process.cwd())
|
|
112
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
113
|
+
|
|
114
|
+
// Verify it resolves from projectRoot, not monorepoRoot
|
|
115
|
+
const result = hybridRequire('test-package-nested');
|
|
116
|
+
expect(result.default).toBe('from-nested-workspace');
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('should handle multiple chdir calls without breaking', () => {
|
|
120
|
+
const projectRoot = path.join(tempDir, 'project');
|
|
121
|
+
const dir1 = path.join(tempDir, 'dir1');
|
|
122
|
+
const dir2 = path.join(tempDir, 'dir2');
|
|
123
|
+
|
|
124
|
+
diskFs.mkdirSync(projectRoot, { recursive: true });
|
|
125
|
+
diskFs.mkdirSync(dir1, { recursive: true });
|
|
126
|
+
diskFs.mkdirSync(dir2, { recursive: true });
|
|
127
|
+
|
|
128
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
129
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package-stable');
|
|
130
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
131
|
+
|
|
132
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
133
|
+
diskFs.writeFileSync(
|
|
134
|
+
testFile,
|
|
135
|
+
"module.exports = { default: 'stable-reference' };",
|
|
136
|
+
);
|
|
137
|
+
diskFs.writeFileSync(
|
|
138
|
+
path.join(testPackageDir, 'package.json'),
|
|
139
|
+
JSON.stringify({ name: 'test-package-stable', main: 'index.js' }),
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
diskFs.writeFileSync(
|
|
143
|
+
path.join(projectRoot, 'package.json'),
|
|
144
|
+
JSON.stringify({ name: 'project' }),
|
|
145
|
+
);
|
|
146
|
+
|
|
147
|
+
// Set up memfs WITHOUT diskFs to test fallback
|
|
148
|
+
const volume = new Volume();
|
|
149
|
+
const fs = createFsFromVolume(volume);
|
|
150
|
+
ufs.use(fs as any);
|
|
151
|
+
|
|
152
|
+
// Initialize
|
|
153
|
+
process.chdir(projectRoot);
|
|
154
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
155
|
+
|
|
156
|
+
// Multiple chdir calls
|
|
157
|
+
process.chdir(dir1);
|
|
158
|
+
expect(hybridRequire('test-package-stable').default).toBe(
|
|
159
|
+
'stable-reference',
|
|
160
|
+
);
|
|
161
|
+
|
|
162
|
+
process.chdir(dir2);
|
|
163
|
+
expect(hybridRequire('test-package-stable').default).toBe(
|
|
164
|
+
'stable-reference',
|
|
165
|
+
);
|
|
166
|
+
|
|
167
|
+
process.chdir(projectRoot);
|
|
168
|
+
expect(hybridRequire('test-package-stable').default).toBe(
|
|
169
|
+
'stable-reference',
|
|
170
|
+
);
|
|
171
|
+
});
|
|
172
|
+
});
|
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
import diskFs from 'fs';
|
|
2
|
+
import { createFsRequire } from 'fs-require';
|
|
3
|
+
import { createFsFromVolume, Volume } from 'memfs';
|
|
4
|
+
import path from 'path';
|
|
5
|
+
import tmp from 'tmp';
|
|
6
|
+
import { ufs } from 'unionfs';
|
|
7
|
+
|
|
8
|
+
import { createHybridRequire } from '../createHybridRequire';
|
|
9
|
+
|
|
10
|
+
describe('hybridRequire wrapper', () => {
|
|
11
|
+
let tempDir: string;
|
|
12
|
+
let originalCwd: string;
|
|
13
|
+
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
originalCwd = process.cwd();
|
|
16
|
+
tempDir = tmp.dirSync({ unsafeCleanup: true }).name;
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
afterEach(() => {
|
|
20
|
+
process.chdir(originalCwd);
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
describe('unit tests - fallback resolver', () => {
|
|
24
|
+
it('should resolve bare specifiers from project node_modules when not in memfs', () => {
|
|
25
|
+
// Set up a temp directory with node_modules containing a test package
|
|
26
|
+
const projectRoot = tempDir;
|
|
27
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
28
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package');
|
|
29
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
30
|
+
|
|
31
|
+
// Create directory structure with a unique package name to avoid conflicts
|
|
32
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
33
|
+
diskFs.writeFileSync(
|
|
34
|
+
testFile,
|
|
35
|
+
"module.exports = { default: 'test-export-value', test: 'value' };",
|
|
36
|
+
);
|
|
37
|
+
|
|
38
|
+
// Create package.json for the test package
|
|
39
|
+
diskFs.writeFileSync(
|
|
40
|
+
path.join(testPackageDir, 'package.json'),
|
|
41
|
+
JSON.stringify({ name: 'test-package', main: 'index.js' }),
|
|
42
|
+
);
|
|
43
|
+
|
|
44
|
+
// Create package.json in project root
|
|
45
|
+
diskFs.writeFileSync(
|
|
46
|
+
path.join(projectRoot, 'package.json'),
|
|
47
|
+
JSON.stringify({ name: 'test-project' }),
|
|
48
|
+
);
|
|
49
|
+
|
|
50
|
+
// Set up memfs WITHOUT diskFs to simulate memfs-only scenario
|
|
51
|
+
// This way fsRequire will fail (module not in memfs)
|
|
52
|
+
// and hybridRequire will fall back to projectRequire
|
|
53
|
+
const volume = new Volume();
|
|
54
|
+
const fs = createFsFromVolume(volume);
|
|
55
|
+
// Only use memfs, not disk - this simulates the webpack output scenario
|
|
56
|
+
ufs.use(fs as any);
|
|
57
|
+
|
|
58
|
+
const fsRequire = createFsRequire(ufs);
|
|
59
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
60
|
+
|
|
61
|
+
// Mock process.cwd to return our temp directory
|
|
62
|
+
const originalCwd = process.cwd;
|
|
63
|
+
process.cwd = jest.fn(() => projectRoot);
|
|
64
|
+
|
|
65
|
+
try {
|
|
66
|
+
// fsRequire should fail because test-package is not in memfs
|
|
67
|
+
expect(() => {
|
|
68
|
+
fsRequire('test-package');
|
|
69
|
+
}).toThrow();
|
|
70
|
+
|
|
71
|
+
// hybridRequire should succeed by falling back to project node_modules
|
|
72
|
+
const result = hybridRequire('test-package');
|
|
73
|
+
expect(result).toBeDefined();
|
|
74
|
+
expect(result.default).toBe('test-export-value');
|
|
75
|
+
expect(result.test).toBe('value');
|
|
76
|
+
} finally {
|
|
77
|
+
process.cwd = originalCwd;
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
it('should work with unionfs when module exists in both disk and memfs', () => {
|
|
82
|
+
const projectRoot = tempDir;
|
|
83
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
84
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package-union');
|
|
85
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
86
|
+
|
|
87
|
+
// Create directory structure on disk
|
|
88
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
89
|
+
diskFs.writeFileSync(
|
|
90
|
+
testFile,
|
|
91
|
+
"module.exports = { default: 'disk-version' };",
|
|
92
|
+
);
|
|
93
|
+
diskFs.writeFileSync(
|
|
94
|
+
path.join(testPackageDir, 'package.json'),
|
|
95
|
+
JSON.stringify({ name: 'test-package-union', main: 'index.js' }),
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
diskFs.writeFileSync(
|
|
99
|
+
path.join(projectRoot, 'package.json'),
|
|
100
|
+
JSON.stringify({ name: 'test-project' }),
|
|
101
|
+
);
|
|
102
|
+
|
|
103
|
+
// Set up memfs with the same module (different version)
|
|
104
|
+
const volume = new Volume();
|
|
105
|
+
const fs = createFsFromVolume(volume);
|
|
106
|
+
const memfsPath = path.join('/node_modules/test-package-union/index.js');
|
|
107
|
+
fs.mkdirSync(path.dirname(memfsPath), { recursive: true });
|
|
108
|
+
fs.writeFileSync(
|
|
109
|
+
memfsPath,
|
|
110
|
+
"module.exports = { default: 'memfs-version' };",
|
|
111
|
+
);
|
|
112
|
+
|
|
113
|
+
// unionfs checks disk first, then memfs (same as dev server)
|
|
114
|
+
ufs.use(diskFs).use(fs as any);
|
|
115
|
+
|
|
116
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
117
|
+
|
|
118
|
+
// hybridRequire should get disk version (unionfs checks disk first)
|
|
119
|
+
// This verifies that hybridRequire correctly uses fsRequire with unionfs
|
|
120
|
+
const result = hybridRequire('test-package-union');
|
|
121
|
+
expect(result.default).toBe('disk-version');
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('should handle relative paths without fallback', () => {
|
|
125
|
+
const projectRoot = tempDir;
|
|
126
|
+
diskFs.writeFileSync(
|
|
127
|
+
path.join(projectRoot, 'package.json'),
|
|
128
|
+
JSON.stringify({ name: 'test-project' }),
|
|
129
|
+
);
|
|
130
|
+
|
|
131
|
+
const volume = new Volume();
|
|
132
|
+
const fs = createFsFromVolume(volume);
|
|
133
|
+
const relativeFile = '/relative.js';
|
|
134
|
+
fs.writeFileSync(
|
|
135
|
+
relativeFile,
|
|
136
|
+
"module.exports = { default: 'relative-export' };",
|
|
137
|
+
);
|
|
138
|
+
|
|
139
|
+
ufs.use(diskFs).use(fs as any);
|
|
140
|
+
|
|
141
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
142
|
+
|
|
143
|
+
// Relative paths should work through memfs
|
|
144
|
+
const result = hybridRequire(relativeFile);
|
|
145
|
+
expect(result.default).toBe('relative-export');
|
|
146
|
+
|
|
147
|
+
// If relative path fails, it should throw (not fallback)
|
|
148
|
+
expect(() => hybridRequire('./nonexistent.js')).toThrow();
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
it('should throw non-MODULE_NOT_FOUND errors', () => {
|
|
152
|
+
const projectRoot = tempDir;
|
|
153
|
+
diskFs.writeFileSync(
|
|
154
|
+
path.join(projectRoot, 'package.json'),
|
|
155
|
+
JSON.stringify({ name: 'test-project' }),
|
|
156
|
+
);
|
|
157
|
+
|
|
158
|
+
const volume = new Volume();
|
|
159
|
+
const fs = createFsFromVolume(volume);
|
|
160
|
+
ufs.use(diskFs).use(fs as any);
|
|
161
|
+
|
|
162
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
163
|
+
|
|
164
|
+
// Create a file that will cause a different error
|
|
165
|
+
const badFile = '/bad.js';
|
|
166
|
+
fs.writeFileSync(badFile, 'invalid syntax !!!');
|
|
167
|
+
|
|
168
|
+
// Should throw the original error, not attempt fallback
|
|
169
|
+
expect(() => hybridRequire(badFile)).toThrow();
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
it('should preserve cache and resolve properties', () => {
|
|
173
|
+
const projectRoot = tempDir;
|
|
174
|
+
diskFs.writeFileSync(
|
|
175
|
+
path.join(projectRoot, 'package.json'),
|
|
176
|
+
JSON.stringify({ name: 'test-project' }),
|
|
177
|
+
);
|
|
178
|
+
|
|
179
|
+
const volume = new Volume();
|
|
180
|
+
const fs = createFsFromVolume(volume);
|
|
181
|
+
ufs.use(diskFs).use(fs as any);
|
|
182
|
+
|
|
183
|
+
// Create hybridRequire (which creates fsRequire internally)
|
|
184
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
185
|
+
|
|
186
|
+
// Verify cache and resolve are preserved from the internal fsRequire
|
|
187
|
+
expect(hybridRequire.cache).toBeDefined();
|
|
188
|
+
expect(typeof hybridRequire.cache).toBe('object');
|
|
189
|
+
expect(hybridRequire.resolve).toBeDefined();
|
|
190
|
+
expect(typeof hybridRequire.resolve).toBe('function');
|
|
191
|
+
// Verify resolve works
|
|
192
|
+
expect(() => hybridRequire.resolve('some-module')).not.toThrow();
|
|
193
|
+
});
|
|
194
|
+
});
|
|
195
|
+
|
|
196
|
+
describe('regression test - process.cwd variance', () => {
|
|
197
|
+
it('should resolve from initial working directory even after process.chdir', () => {
|
|
198
|
+
const projectRoot = tempDir;
|
|
199
|
+
const nodeModulesDir = path.join(projectRoot, 'node_modules');
|
|
200
|
+
const testPackageDir = path.join(nodeModulesDir, 'test-package-3');
|
|
201
|
+
const testFile = path.join(testPackageDir, 'index.js');
|
|
202
|
+
|
|
203
|
+
// Create directory structure
|
|
204
|
+
diskFs.mkdirSync(testPackageDir, { recursive: true });
|
|
205
|
+
diskFs.writeFileSync(
|
|
206
|
+
testFile,
|
|
207
|
+
"module.exports = { default: 'test-export-value' };",
|
|
208
|
+
);
|
|
209
|
+
diskFs.writeFileSync(
|
|
210
|
+
path.join(testPackageDir, 'package.json'),
|
|
211
|
+
JSON.stringify({ name: 'test-package-3', main: 'index.js' }),
|
|
212
|
+
);
|
|
213
|
+
|
|
214
|
+
diskFs.writeFileSync(
|
|
215
|
+
path.join(projectRoot, 'package.json'),
|
|
216
|
+
JSON.stringify({ name: 'test-project' }),
|
|
217
|
+
);
|
|
218
|
+
|
|
219
|
+
// Create another directory that we'll chdir to
|
|
220
|
+
const otherDir = tmp.dirSync({ unsafeCleanup: true }).name;
|
|
221
|
+
diskFs.writeFileSync(
|
|
222
|
+
path.join(otherDir, 'package.json'),
|
|
223
|
+
JSON.stringify({ name: 'other-project' }),
|
|
224
|
+
);
|
|
225
|
+
|
|
226
|
+
// Set up memfs WITHOUT diskFs to test fallback
|
|
227
|
+
const volume = new Volume();
|
|
228
|
+
const fs = createFsFromVolume(volume);
|
|
229
|
+
ufs.use(fs as any);
|
|
230
|
+
|
|
231
|
+
// Capture projectRequire at initial cwd
|
|
232
|
+
const hybridRequire = createHybridRequire(ufs, projectRoot);
|
|
233
|
+
|
|
234
|
+
// Change directory
|
|
235
|
+
process.chdir(otherDir);
|
|
236
|
+
|
|
237
|
+
// hybridRequire should still resolve from the original projectRoot
|
|
238
|
+
// because projectRequire was created with that path
|
|
239
|
+
const result = hybridRequire('test-package-3');
|
|
240
|
+
expect(result.default).toBe('test-export-value');
|
|
241
|
+
});
|
|
242
|
+
});
|
|
243
|
+
});
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { createFsRequire } from 'fs-require';
|
|
2
|
+
import { createRequire } from 'module';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
import { type IUnionFs } from 'unionfs';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Creates a hybrid require function that combines fs-require (for in-memory files)
|
|
8
|
+
* with Node's native require (for bare specifiers from project node_modules).
|
|
9
|
+
*
|
|
10
|
+
* This solves the issue where fs-require cannot resolve bare specifiers like
|
|
11
|
+
* `@anansi/core/server` from workspace node_modules when they're not in the
|
|
12
|
+
* memfs bundle.
|
|
13
|
+
*
|
|
14
|
+
* @param ufs - The unionfs instance (disk first, then memfs)
|
|
15
|
+
* @param projectRoot - Optional project root directory. Defaults to process.cwd()
|
|
16
|
+
* @returns A require function that tries memfs first, then falls back to project node_modules
|
|
17
|
+
*/
|
|
18
|
+
export function createHybridRequire(
|
|
19
|
+
ufs: IUnionFs,
|
|
20
|
+
projectRoot: string = process.cwd(),
|
|
21
|
+
): ReturnType<typeof createFsRequire> {
|
|
22
|
+
const fsRequire = createFsRequire(ufs);
|
|
23
|
+
const projectRequire = createRequire(path.join(projectRoot, 'package.json'));
|
|
24
|
+
|
|
25
|
+
function hybridRequire(id: string) {
|
|
26
|
+
try {
|
|
27
|
+
return fsRequire(id);
|
|
28
|
+
} catch (error: any) {
|
|
29
|
+
const isBare = !id.startsWith('.') && !path.isAbsolute(id);
|
|
30
|
+
if (isBare && error?.code === 'MODULE_NOT_FOUND') {
|
|
31
|
+
return projectRequire(id);
|
|
32
|
+
}
|
|
33
|
+
throw error;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Preserve cache and resolve properties from fsRequire
|
|
38
|
+
hybridRequire.cache = fsRequire.cache;
|
|
39
|
+
hybridRequire.resolve = fsRequire.resolve;
|
|
40
|
+
|
|
41
|
+
return hybridRequire as ReturnType<typeof createFsRequire>;
|
|
42
|
+
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { default as enhanced } from 'enhanced-resolve';
|
|
2
|
-
import
|
|
2
|
+
import type { Configuration } from 'webpack';
|
|
3
3
|
|
|
4
4
|
const resolve = enhanced.create.sync({
|
|
5
5
|
modules: ['.'],
|
|
@@ -9,7 +9,7 @@ const resolve = enhanced.create.sync({
|
|
|
9
9
|
});
|
|
10
10
|
|
|
11
11
|
export async function getWebpackConfig(): Promise<
|
|
12
|
-
(env: any, argv: any) =>
|
|
12
|
+
(env: any, argv: any) => Configuration
|
|
13
13
|
> {
|
|
14
14
|
const configPath = resolve({}, process.cwd(), 'webpack.config');
|
|
15
15
|
if (!configPath) {
|
|
@@ -6,19 +6,19 @@ Object.hasOwn =
|
|
|
6
6
|
};
|
|
7
7
|
import type { NextFunction } from 'express';
|
|
8
8
|
import diskFs from 'fs';
|
|
9
|
-
import {
|
|
10
|
-
import { Server, IncomingMessage, ServerResponse } from 'http';
|
|
9
|
+
import { IncomingMessage, ServerResponse } from 'http';
|
|
11
10
|
import { createFsFromVolume, Volume } from 'memfs';
|
|
12
11
|
import path from 'path';
|
|
13
12
|
import sourceMapSupport from 'source-map-support';
|
|
14
13
|
import tmp from 'tmp';
|
|
15
14
|
import { ufs } from 'unionfs';
|
|
16
15
|
import { promisify } from 'util';
|
|
17
|
-
import webpack, {
|
|
16
|
+
import webpack, { type Configuration, type MultiConfiguration } from 'webpack';
|
|
18
17
|
import logging from 'webpack/lib/logging/runtime.js';
|
|
19
18
|
import WebpackDevServer from 'webpack-dev-server';
|
|
20
19
|
|
|
21
20
|
import 'cross-fetch/dist/node-polyfill.js';
|
|
21
|
+
import { createHybridRequire } from './createHybridRequire.js';
|
|
22
22
|
import { getWebpackConfig } from './getWebpackConfig.js';
|
|
23
23
|
import { BoundRender } from './types.js';
|
|
24
24
|
|
|
@@ -52,10 +52,9 @@ export default async function startDevServer(
|
|
|
52
52
|
const fs = createFsFromVolume(volume);
|
|
53
53
|
ufs.use(diskFs).use(fs as any);
|
|
54
54
|
|
|
55
|
-
const fsRequire =
|
|
56
|
-
const readFile = promisify(ufs.readFile);
|
|
57
|
-
let server: Server | undefined;
|
|
55
|
+
const fsRequire = createHybridRequire(ufs);
|
|
58
56
|
|
|
57
|
+
const readFile = promisify(ufs.readFile);
|
|
59
58
|
// Generate a temporary file so we can hot reload from the root of the application
|
|
60
59
|
function hotEntry(entryPath: string) {
|
|
61
60
|
// eslint-disable-next-line
|
|
@@ -76,7 +75,7 @@ export default async function startDevServer(
|
|
|
76
75
|
return generatedEntrypoint;
|
|
77
76
|
}
|
|
78
77
|
|
|
79
|
-
const webpackConfigs = [
|
|
78
|
+
const webpackConfigs: Configuration[] = [
|
|
80
79
|
webpackConfig(
|
|
81
80
|
{
|
|
82
81
|
...env,
|
|
@@ -94,10 +93,10 @@ export default async function startDevServer(
|
|
|
94
93
|
},
|
|
95
94
|
{ mode: 'development', target: 'node' },
|
|
96
95
|
),
|
|
97
|
-
]
|
|
96
|
+
];
|
|
98
97
|
|
|
99
98
|
// initialize the webpack compiler
|
|
100
|
-
const compiler = webpack(webpackConfigs);
|
|
99
|
+
const compiler = webpack(webpackConfigs as unknown as MultiConfiguration);
|
|
101
100
|
if (!compiler) {
|
|
102
101
|
log.error('Failed to initialize the webpack compiler');
|
|
103
102
|
process.exit(-1);
|