@rooted-adapters/express 1.0.0-alpha.0 → 1.0.0-alpha.1
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/dist/index.d.mts +49 -2
- package/dist/index.mjs +54 -7
- package/package.json +64 -63
- package/readme.md +22 -0
- package/LICENSE +0 -21
package/dist/index.d.mts
CHANGED
|
@@ -1,7 +1,30 @@
|
|
|
1
1
|
import { AdapterRoutes } from "@rooted/adapter";
|
|
2
|
+
import { Express } from "express";
|
|
2
3
|
import { Plugin } from "vite";
|
|
3
4
|
|
|
4
5
|
//#region src/index.d.mts
|
|
6
|
+
/**
|
|
7
|
+
* An Express middleware function for use with {@link ExpressAdapterOptions.middlewarePath}.
|
|
8
|
+
* Receives the Express instance and may register middleware or routes on it.
|
|
9
|
+
*/
|
|
10
|
+
type ExpressMiddleware = (app: Express) => Promise<void> | void;
|
|
11
|
+
/**
|
|
12
|
+
* Identity helper that types a middleware function for the express adapter.
|
|
13
|
+
* Use it as the default export of a file under your `middlewarePath` folder so
|
|
14
|
+
* editors pick up the Express instance type without extra annotations.
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```ts
|
|
18
|
+
* // src/server-middleware/01-api-proxy.mts
|
|
19
|
+
* import { createMiddleware } from '@rooted-adapters/express'
|
|
20
|
+
* import { createProxyMiddleware } from 'http-proxy-middleware'
|
|
21
|
+
*
|
|
22
|
+
* export default createMiddleware((app) => {
|
|
23
|
+
* app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))
|
|
24
|
+
* })
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
declare function createMiddleware(handler: ExpressMiddleware): ExpressMiddleware;
|
|
5
28
|
/**
|
|
6
29
|
* Options for {@link expressAdapter}.
|
|
7
30
|
*/
|
|
@@ -11,6 +34,30 @@ type ExpressAdapterOptions = {
|
|
|
11
34
|
* See {@link AdapterRoutes}.
|
|
12
35
|
*/
|
|
13
36
|
routes?: AdapterRoutes;
|
|
37
|
+
/**
|
|
38
|
+
* Path to a folder of middleware files, relative to the Vite project root.
|
|
39
|
+
* Files can be `.mts`, `.ts`, `.mjs`, or `.js` -- TypeScript is transpiled with
|
|
40
|
+
* rolldown at build time. Each file must export a default `function(app)` that
|
|
41
|
+
* registers middleware on the Express instance. Files are loaded in lexicographic
|
|
42
|
+
* order, so numeric prefixes (`01-auth.mts`, `02-proxy.mts`) control load order.
|
|
43
|
+
* Middleware runs before the rooted static-file and route handlers.
|
|
44
|
+
*
|
|
45
|
+
* @example
|
|
46
|
+
* ```ts
|
|
47
|
+
* expressAdapter({ middlewarePath: './src/server-middleware' })
|
|
48
|
+
* ```
|
|
49
|
+
*
|
|
50
|
+
* ```ts
|
|
51
|
+
* // src/server-middleware/01-api-proxy.mts
|
|
52
|
+
* import { createMiddleware } from '@rooted-adapters/express'
|
|
53
|
+
* import { createProxyMiddleware } from 'http-proxy-middleware'
|
|
54
|
+
*
|
|
55
|
+
* export default createMiddleware((app) => {
|
|
56
|
+
* app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))
|
|
57
|
+
* })
|
|
58
|
+
* ```
|
|
59
|
+
*/
|
|
60
|
+
middlewarePath?: string;
|
|
14
61
|
};
|
|
15
62
|
/**
|
|
16
63
|
* Adapter for server-side hosting with Express.
|
|
@@ -42,5 +89,5 @@ type ExpressAdapterOptions = {
|
|
|
42
89
|
*/
|
|
43
90
|
declare function expressAdapter(options?: ExpressAdapterOptions): Plugin;
|
|
44
91
|
//#endregion
|
|
45
|
-
export { ExpressAdapterOptions, expressAdapter };
|
|
46
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
92
|
+
export { ExpressAdapterOptions, ExpressMiddleware, createMiddleware, expressAdapter };
|
|
93
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC5tdHMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4Lm10cyJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQWVBOztLQUFZLGlCQUFBLElBQXFCLEdBQUEsRUFBSyxPQUFBLEtBQVksT0FBQTs7Ozs7OztBQWtCbEQ7Ozs7Ozs7OztBQU9BO2lCQVBnQixnQkFBQSxDQUFpQixPQUFBLEVBQVMsaUJBQUEsR0FBb0IsaUJBQUE7Ozs7S0FPbEQscUJBQUE7RUE2Qlg7OztBQStCRDtFQXZEQyxNQUFBLEdBQVMsYUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0VBd0JULGNBQUE7QUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7aUJBK0JlLGNBQUEsQ0FBZSxPQUFBLEdBQVUscUJBQUEsR0FBd0IsTUFBQSJ9
|
package/dist/index.mjs
CHANGED
|
@@ -1,8 +1,28 @@
|
|
|
1
|
-
import { writeFile } from "node:fs/promises";
|
|
1
|
+
import { mkdir, readdir, writeFile } from "node:fs/promises";
|
|
2
2
|
import path from "node:path";
|
|
3
|
+
import { build } from "rolldown";
|
|
3
4
|
import { routedAdapter } from "@rooted/adapter";
|
|
4
5
|
//#region src/index.mts
|
|
5
6
|
/**
|
|
7
|
+
* Identity helper that types a middleware function for the express adapter.
|
|
8
|
+
* Use it as the default export of a file under your `middlewarePath` folder so
|
|
9
|
+
* editors pick up the Express instance type without extra annotations.
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```ts
|
|
13
|
+
* // src/server-middleware/01-api-proxy.mts
|
|
14
|
+
* import { createMiddleware } from '@rooted-adapters/express'
|
|
15
|
+
* import { createProxyMiddleware } from 'http-proxy-middleware'
|
|
16
|
+
*
|
|
17
|
+
* export default createMiddleware((app) => {
|
|
18
|
+
* app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))
|
|
19
|
+
* })
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
function createMiddleware(handler) {
|
|
23
|
+
return handler;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
6
26
|
* Adapter for server-side hosting with Express.
|
|
7
27
|
*
|
|
8
28
|
* Writes `routes.json` and a ready-to-run `server.mjs` to the output directory.
|
|
@@ -35,13 +55,32 @@ function expressAdapter(options) {
|
|
|
35
55
|
name: "rooted:express",
|
|
36
56
|
routes: options?.routes,
|
|
37
57
|
async setup({ outputDirectory }) {
|
|
38
|
-
|
|
58
|
+
if (options?.middlewarePath) {
|
|
59
|
+
const sourceDirectory = path.resolve(process.cwd(), options.middlewarePath);
|
|
60
|
+
const files = (await readdir(sourceDirectory)).filter((f) => MIDDLEWARE_EXTENSIONS.test(f));
|
|
61
|
+
if (files.length === 0) throw new Error(`[rooted:express] No middleware files (.mts, .ts, .mjs, .js) found in middlewarePath "${options.middlewarePath}"`);
|
|
62
|
+
const middlewareDirectory = path.join(outputDirectory, "middleware");
|
|
63
|
+
await mkdir(middlewareDirectory, { recursive: true });
|
|
64
|
+
for (const file of files) await build({
|
|
65
|
+
input: path.join(sourceDirectory, file),
|
|
66
|
+
platform: "node",
|
|
67
|
+
external: (id) => !id.startsWith(".") && !path.isAbsolute(id),
|
|
68
|
+
logLevel: "silent",
|
|
69
|
+
output: {
|
|
70
|
+
file: path.join(middlewareDirectory, file.replace(MIDDLEWARE_EXTENSIONS, ".mjs")),
|
|
71
|
+
format: "esm"
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
await writeFile(path.join(outputDirectory, "server.mjs"), buildExpressTemplate(!!options?.middlewarePath), "utf8");
|
|
39
76
|
}
|
|
40
77
|
});
|
|
41
78
|
}
|
|
42
|
-
const
|
|
79
|
+
const MIDDLEWARE_EXTENSIONS = /\.(mts|ts|mjs|js)$/;
|
|
80
|
+
function buildExpressTemplate(hasMiddleware) {
|
|
81
|
+
return `\
|
|
43
82
|
import express from 'express'
|
|
44
|
-
import { readFileSync } from 'node:fs'
|
|
83
|
+
${hasMiddleware ? `import { readFileSync, readdirSync } from 'node:fs'` : `import { readFileSync } from 'node:fs'`}
|
|
45
84
|
import { fileURLToPath } from 'node:url'
|
|
46
85
|
import path from 'node:path'
|
|
47
86
|
|
|
@@ -52,7 +91,14 @@ const { base, dynamicRoutes, fallback } = JSON.parse(
|
|
|
52
91
|
|
|
53
92
|
const prefix = base.replace(/\\/$/, '')
|
|
54
93
|
const app = express()
|
|
55
|
-
|
|
94
|
+
${hasMiddleware ? `
|
|
95
|
+
// User middleware -- applied before rooted handlers
|
|
96
|
+
const middlewareDir = path.join(__dirname, 'middleware')
|
|
97
|
+
for (const file of readdirSync(middlewareDir).filter(f => f.endsWith('.mjs')).sort()) {
|
|
98
|
+
const mod = await import(path.join(middlewareDir, file))
|
|
99
|
+
if (mod.default) await mod.default(app)
|
|
100
|
+
}
|
|
101
|
+
` : ""}
|
|
56
102
|
// Serves all pre-rendered HTML files and static assets automatically
|
|
57
103
|
app.use(base, express.static(__dirname))
|
|
58
104
|
|
|
@@ -69,7 +115,8 @@ app.use((_req, res) => res.sendFile(path.join(__dirname, fallback)))
|
|
|
69
115
|
const port = Number(process.env.PORT ?? 3000)
|
|
70
116
|
app.listen(port, '0.0.0.0', () => console.log(\`Listening on http://0.0.0.0:\${port}\`))
|
|
71
117
|
`;
|
|
118
|
+
}
|
|
72
119
|
//#endregion
|
|
73
|
-
export { expressAdapter };
|
|
120
|
+
export { createMiddleware, expressAdapter };
|
|
74
121
|
|
|
75
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgubWpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5tdHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgd3JpdGVGaWxlIH0gZnJvbSAnbm9kZTpmcy9wcm9taXNlcydcbmltcG9ydCBwYXRoIGZyb20gJ25vZGU6cGF0aCdcblxuaW1wb3J0IHsgcm91dGVkQWRhcHRlciB9IGZyb20gJ0Byb290ZWQvYWRhcHRlcidcblxuaW1wb3J0IHR5cGUgeyBBZGFwdGVyUm91dGVzIH0gZnJvbSAnQHJvb3RlZC9hZGFwdGVyJ1xuaW1wb3J0IHR5cGUgeyBQbHVnaW4gfSBmcm9tICd2aXRlJ1xuXG4vKipcbiAqIE9wdGlvbnMgZm9yIHtAbGluayBleHByZXNzQWRhcHRlcn0uXG4gKi9cbmV4cG9ydCB0eXBlIEV4cHJlc3NBZGFwdGVyT3B0aW9ucyA9IHtcblx0LyoqXG5cdCAqIE1hbnVhbCByb3V0ZSBsaXN0IGZvciBwcm9qZWN0cyB0aGF0IGRvbid0IHVzZSBgZ2VuZXJhdGVSb3V0ZU1hbmlmZXN0YC5cblx0ICogU2VlIHtAbGluayBBZGFwdGVyUm91dGVzfS5cblx0ICovXG5cdHJvdXRlcz86IEFkYXB0ZXJSb3V0ZXNcbn1cblxuLyoqXG4gKiBBZGFwdGVyIGZvciBzZXJ2ZXItc2lkZSBob3N0aW5nIHdpdGggRXhwcmVzcy5cbiAqXG4gKiBXcml0ZXMgYHJvdXRlcy5qc29uYCBhbmQgYSByZWFkeS10by1ydW4gYHNlcnZlci5tanNgIHRvIHRoZSBvdXRwdXQgZGlyZWN0b3J5LlxuICogVGhlIHNlcnZlciB1c2VzIGBleHByZXNzLnN0YXRpY2AgdG8gc2VydmUgcHJlLXJlbmRlcmVkIEhUTUwgZmlsZXMgYW5kIHJlZ2lzdGVyc1xuICogZXhwbGljaXQgaGFuZGxlcnMgZm9yIHBhcmFtZXRlcml6ZWQgcm91dGVzICh3aGljaCBzZXJ2ZSB0aGUgYDQwNC5odG1sYCBTUEEgc2hlbGxcbiAqIHNvIHRoZSBicm93c2VyLXNpZGUgcm91dGVyIHJlbmRlcnMgdGhlIGNvcnJlY3QgY29udGVudCkuIEV4cHJlc3MgdXNlcyB0aGUgc2FtZVxuICogYDpwYXJhbWAgc3ludGF4IGFzIHRoZSByb290ZWQgcm91dGVyLCBzbyBwYXR0ZXJucyBtYXAgZGlyZWN0bHkuXG4gKlxuICogVXNlcnMgc3RhcnQgdGhlIHNlcnZlciB3aXRoIGBub2RlIGRpc3Qvc2VydmVyLm1qc2AuIFRoZSBgUE9SVGAgZW52aXJvbm1lbnQgdmFyaWFibGVcbiAqIGNvbnRyb2xzIHRoZSBwb3J0IChkZWZhdWx0OiAzMDAwKS5cbiAqXG4gKiBSZXF1aXJlcyBgZXhwcmVzcyA+PSA1LjAuMGAgaW4gdGhlIHByb2plY3QuXG4gKlxuICogQGV4YW1wbGUgYHZpdGUuY29uZmlnLnRzYFxuICogYGBgdHNcbiAqIGltcG9ydCB7IHJvb3RlZE1hbmlmZXN0IH0gZnJvbSAnQHJvb3RlZC9hcHBsaWNhdGlvbidcbiAqIGltcG9ydCB7IGdlbmVyYXRlUm91dGVNYW5pZmVzdCB9IGZyb20gJ0Byb290ZWQvcm91dGVyL21hbmlmZXN0J1xuICogaW1wb3J0IHsgZXhwcmVzc0FkYXB0ZXIgfSBmcm9tICdAcm9vdGVkLWFkYXB0ZXJzL2V4cHJlc3MnXG4gKlxuICogZXhwb3J0IGRlZmF1bHQgcm9vdGVkTWFuaWZlc3Qoe1xuICogICBwbHVnaW5zOiBbXG4gKiAgICAgZ2VuZXJhdGVSb3V0ZU1hbmlmZXN0KHsgZ2xvYjogJy4vc3JjLyoqXFwvX3JvdXRlcy5tdHMnLCByb290OiAnLi9zcmMvX3JvdXRlcy5nLm10cycgfSksXG4gKiAgICAgZXhwcmVzc0FkYXB0ZXIoKSxcbiAqICAgXSxcbiAqIH0pXG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGV4cHJlc3NBZGFwdGVyKG9wdGlvbnM/OiBFeHByZXNzQWRhcHRlck9wdGlvbnMpOiBQbHVnaW4ge1xuXHRyZXR1cm4gcm91dGVkQWRhcHRlcih7XG5cdFx0bmFtZTogJ3Jvb3RlZDpleHByZXNzJyxcblx0XHRyb3V0ZXM6IG9wdGlvbnM/LnJvdXRlcyxcblx0XHRhc3luYyBzZXR1cCh7IG91dHB1dERpcmVjdG9yeSB9KSB7XG5cdFx0XHRhd2FpdCB3cml0ZUZpbGUoXG5cdFx0XHRcdHBhdGguam9pbihvdXRwdXREaXJlY3RvcnksICdzZXJ2ZXIubWpzJyksXG5cdFx0XHRcdEVYUFJFU1NfU0VSVkVSX1RFTVBMQVRFLFxuXHRcdFx0XHQndXRmOCcsXG5cdFx0XHQpXG5cdFx0fSxcblx0fSlcbn1cblxuY29uc3QgRVhQUkVTU19TRVJWRVJfVEVNUExBVEUgPSBgXFxcbmltcG9ydCBleHByZXNzIGZyb20gJ2V4cHJlc3MnXG5pbXBvcnQgeyByZWFkRmlsZVN5bmMgfSBmcm9tICdub2RlOmZzJ1xuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gJ25vZGU6dXJsJ1xuaW1wb3J0IHBhdGggZnJvbSAnbm9kZTpwYXRoJ1xuXG5jb25zdCBfX2Rpcm5hbWUgPSBwYXRoLmRpcm5hbWUoZmlsZVVSTFRvUGF0aChpbXBvcnQubWV0YS51cmwpKVxuY29uc3QgeyBiYXNlLCBkeW5hbWljUm91dGVzLCBmYWxsYmFjayB9ID0gSlNPTi5wYXJzZShcbiAgcmVhZEZpbGVTeW5jKHBhdGguam9pbihfX2Rpcm5hbWUsICdyb3V0ZXMuanNvbicpLCAndXRmOCcpXG4pXG5cbmNvbnN0IHByZWZpeCA9IGJhc2UucmVwbGFjZSgvXFxcXC8kLywgJycpXG5jb25zdCBhcHAgPSBleHByZXNzKClcblxuLy8gU2VydmVzIGFsbCBwcmUtcmVuZGVyZWQgSFRNTCBmaWxlcyBhbmQgc3RhdGljIGFzc2V0cyBhdXRvbWF0aWNhbGx5XG5hcHAudXNlKGJhc2UsIGV4cHJlc3Muc3RhdGljKF9fZGlybmFtZSkpXG5cbi8vIFBhcmFtZXRlcml6ZWQgcm91dGVzOiBFeHByZXNzIHVzZXMgdGhlIHNhbWUgOnBhcmFtIHN5bnRheCBhcyB0aGUgcm9vdGVkIHJvdXRlclxuZm9yIChjb25zdCByb3V0ZSBvZiBkeW5hbWljUm91dGVzKSB7XG4gIGFwcC5nZXQocHJlZml4ICsgcm91dGUsIChfcmVxLCByZXMpID0+XG4gICAgcmVzLnNlbmRGaWxlKHBhdGguam9pbihfX2Rpcm5hbWUsIGZhbGxiYWNrKSlcbiAgKVxufVxuXG4vLyBBbnl0aGluZyBlbHNlOiBTUEEgc2hlbGwgKGJyb3dzZXItc2lkZSByb3V0ZXIgc2hvd3MgdGhlIGNvcnJlY3QgY29udGVudCBvciA0MDQpXG5hcHAudXNlKChfcmVxLCByZXMpID0+IHJlcy5zZW5kRmlsZShwYXRoLmpvaW4oX19kaXJuYW1lLCBmYWxsYmFjaykpKVxuXG5jb25zdCBwb3J0ID0gTnVtYmVyKHByb2Nlc3MuZW52LlBPUlQgPz8gMzAwMClcbmFwcC5saXN0ZW4ocG9ydCwgJzAuMC4wLjAnLCAoKSA9PiBjb25zb2xlLmxvZyhcXGBMaXN0ZW5pbmcgb24gaHR0cDovLzAuMC4wLjA6XFwke3BvcnR9XFxgKSlcbmBcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUErQ0EsU0FBZ0IsZUFBZSxTQUF5QztDQUN2RSxPQUFPLGNBQWM7RUFDcEIsTUFBTTtFQUNOLFFBQVEsU0FBUztFQUNqQixNQUFNLE1BQU0sRUFBRSxtQkFBbUI7R0FDaEMsTUFBTSxVQUNMLEtBQUssS0FBSyxpQkFBaUIsYUFBYSxFQUN4Qyx5QkFDQSxPQUNBOztFQUVGLENBQUM7O0FBR0gsTUFBTSwwQkFBMEIifQ==
|
|
122
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.mjs","names":[],"sources":["../src/index.mts"],"sourcesContent":["import { mkdir, readdir, writeFile } from 'node:fs/promises'\nimport path from 'node:path'\n\nimport { build } from 'rolldown'\n\nimport { routedAdapter } from '@rooted/adapter'\n\nimport type { AdapterRoutes } from '@rooted/adapter'\nimport type { Express } from 'express'\nimport type { Plugin } from 'vite'\n\n/**\n * An Express middleware function for use with {@link ExpressAdapterOptions.middlewarePath}.\n * Receives the Express instance and may register middleware or routes on it.\n */\nexport type ExpressMiddleware = (app: Express) => Promise<void> | void\n\n/**\n * Identity helper that types a middleware function for the express adapter.\n * Use it as the default export of a file under your `middlewarePath` folder so\n * editors pick up the Express instance type without extra annotations.\n *\n * @example\n * ```ts\n * // src/server-middleware/01-api-proxy.mts\n * import { createMiddleware } from '@rooted-adapters/express'\n * import { createProxyMiddleware } from 'http-proxy-middleware'\n *\n * export default createMiddleware((app) => {\n *   app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))\n * })\n * ```\n */\nexport function createMiddleware(handler: ExpressMiddleware): ExpressMiddleware {\n\treturn handler\n}\n\n/**\n * Options for {@link expressAdapter}.\n */\nexport type ExpressAdapterOptions = {\n\t/**\n\t * Manual route list for projects that don't use `generateRouteManifest`.\n\t * See {@link AdapterRoutes}.\n\t */\n\troutes?: AdapterRoutes\n\t/**\n\t * Path to a folder of middleware files, relative to the Vite project root.\n\t * Files can be `.mts`, `.ts`, `.mjs`, or `.js` -- TypeScript is transpiled with\n\t * rolldown at build time. Each file must export a default `function(app)` that\n\t * registers middleware on the Express instance. Files are loaded in lexicographic\n\t * order, so numeric prefixes (`01-auth.mts`, `02-proxy.mts`) control load order.\n\t * Middleware runs before the rooted static-file and route handlers.\n\t *\n\t * @example\n\t * ```ts\n\t * expressAdapter({ middlewarePath: './src/server-middleware' })\n\t * ```\n\t *\n\t * ```ts\n\t * // src/server-middleware/01-api-proxy.mts\n\t * import { createMiddleware } from '@rooted-adapters/express'\n\t * import { createProxyMiddleware } from 'http-proxy-middleware'\n\t *\n\t * export default createMiddleware((app) => {\n\t *   app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))\n\t * })\n\t * ```\n\t */\n\tmiddlewarePath?: string\n}\n\n/**\n * Adapter for server-side hosting with Express.\n *\n * Writes `routes.json` and a ready-to-run `server.mjs` to the output directory.\n * The server uses `express.static` to serve pre-rendered HTML files and registers\n * explicit handlers for parameterized routes (which serve the `404.html` SPA shell\n * so the browser-side router renders the correct content). Express uses the same\n * `:param` syntax as the rooted router, so patterns map directly.\n *\n * Users start the server with `node dist/server.mjs`. The `PORT` environment variable\n * controls the port (default: 3000).\n *\n * Requires `express >= 5.0.0` in the project.\n *\n * @example `vite.config.ts`\n * ```ts\n * import { rootedManifest } from '@rooted/application'\n * import { generateRouteManifest } from '@rooted/router/manifest'\n * import { expressAdapter } from '@rooted-adapters/express'\n *\n * export default rootedManifest({\n *   plugins: [\n *     generateRouteManifest({ glob: './src/**\\/_routes.mts', root: './src/_routes.g.mts' }),\n *     expressAdapter(),\n *   ],\n * })\n * ```\n */\nexport function expressAdapter(options?: ExpressAdapterOptions): Plugin {\n\treturn routedAdapter({\n\t\tname: 'rooted:express',\n\t\troutes: options?.routes,\n\t\tasync setup({ outputDirectory }) {\n\t\t\tif (options?.middlewarePath) {\n\t\t\t\tconst sourceDirectory = path.resolve(process.cwd(), options.middlewarePath)\n\t\t\t\tconst files = (await readdir(sourceDirectory)).filter(f => MIDDLEWARE_EXTENSIONS.test(f))\n\t\t\t\tif (files.length === 0)\n\t\t\t\t\tthrow new Error(\n\t\t\t\t\t\t`[rooted:express] No middleware files (.mts, .ts, .mjs, .js) found in middlewarePath \"${options.middlewarePath}\"`,\n\t\t\t\t\t)\n\t\t\t\tconst middlewareDirectory = path.join(outputDirectory, 'middleware')\n\t\t\t\tawait mkdir(middlewareDirectory, { recursive: true })\n\t\t\t\tfor (const file of files) {\n\t\t\t\t\tawait build({\n\t\t\t\t\t\tinput: path.join(sourceDirectory, file),\n\t\t\t\t\t\tplatform: 'node',\n\t\t\t\t\t\texternal: id => !id.startsWith('.') && !path.isAbsolute(id),\n\t\t\t\t\t\tlogLevel: 'silent',\n\t\t\t\t\t\toutput: {\n\t\t\t\t\t\t\tfile: path.join(middlewareDirectory, file.replace(MIDDLEWARE_EXTENSIONS, '.mjs')),\n\t\t\t\t\t\t\tformat: 'esm',\n\t\t\t\t\t\t},\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t}\n\t\t\tawait writeFile(\n\t\t\t\tpath.join(outputDirectory, 'server.mjs'),\n\t\t\t\tbuildExpressTemplate(!!options?.middlewarePath),\n\t\t\t\t'utf8',\n\t\t\t)\n\t\t},\n\t})\n}\n\nconst MIDDLEWARE_EXTENSIONS = /\\.(mts|ts|mjs|js)$/\n\nfunction buildExpressTemplate(hasMiddleware: boolean): string {\n\tconst fsImport = hasMiddleware\n\t\t? `import { readFileSync, readdirSync } from 'node:fs'`\n\t\t: `import { readFileSync } from 'node:fs'`\n\n\tconst middlewareBlock = hasMiddleware\n\t\t? `\n// User middleware -- applied before rooted handlers\nconst middlewareDir = path.join(__dirname, 'middleware')\nfor (const file of readdirSync(middlewareDir).filter(f => f.endsWith('.mjs')).sort()) {\n  const mod = await import(path.join(middlewareDir, file))\n  if (mod.default) await mod.default(app)\n}\n`\n\t\t: ''\n\n\treturn `\\\nimport express from 'express'\n${fsImport}\nimport { fileURLToPath } from 'node:url'\nimport path from 'node:path'\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url))\nconst { base, dynamicRoutes, fallback } = JSON.parse(\n  readFileSync(path.join(__dirname, 'routes.json'), 'utf8')\n)\n\nconst prefix = base.replace(/\\\\/$/, '')\nconst app = express()\n${middlewareBlock}\n// Serves all pre-rendered HTML files and static assets automatically\napp.use(base, express.static(__dirname))\n\n// Parameterized routes: Express uses the same :param syntax as the rooted router\nfor (const route of dynamicRoutes) {\n  app.get(prefix + route, (_req, res) =>\n    res.sendFile(path.join(__dirname, fallback))\n  )\n}\n\n// Anything else: SPA shell (browser-side router shows the correct content or 404)\napp.use((_req, res) => res.sendFile(path.join(__dirname, fallback)))\n\nconst port = Number(process.env.PORT ?? 3000)\napp.listen(port, '0.0.0.0', () => console.log(\\`Listening on http://0.0.0.0:\\${port}\\`))\n`\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAiCA,SAAgB,iBAAiB,SAA+C;CAC/E,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAkER,SAAgB,eAAe,SAAyC;CACvE,OAAO,cAAc;EACpB,MAAM;EACN,QAAQ,SAAS;EACjB,MAAM,MAAM,EAAE,mBAAmB;GAChC,IAAI,SAAS,gBAAgB;IAC5B,MAAM,kBAAkB,KAAK,QAAQ,QAAQ,KAAK,EAAE,QAAQ,eAAe;IAC3E,MAAM,SAAS,MAAM,QAAQ,gBAAgB,EAAE,QAAO,MAAK,sBAAsB,KAAK,EAAE,CAAC;IACzF,IAAI,MAAM,WAAW,GACpB,MAAM,IAAI,MACT,wFAAwF,QAAQ,eAAe,GAC/G;IACF,MAAM,sBAAsB,KAAK,KAAK,iBAAiB,aAAa;IACpE,MAAM,MAAM,qBAAqB,EAAE,WAAW,MAAM,CAAC;IACrD,KAAK,MAAM,QAAQ,OAClB,MAAM,MAAM;KACX,OAAO,KAAK,KAAK,iBAAiB,KAAK;KACvC,UAAU;KACV,WAAU,OAAM,CAAC,GAAG,WAAW,IAAI,IAAI,CAAC,KAAK,WAAW,GAAG;KAC3D,UAAU;KACV,QAAQ;MACP,MAAM,KAAK,KAAK,qBAAqB,KAAK,QAAQ,uBAAuB,OAAO,CAAC;MACjF,QAAQ;MACR;KACD,CAAC;;GAGJ,MAAM,UACL,KAAK,KAAK,iBAAiB,aAAa,EACxC,qBAAqB,CAAC,CAAC,SAAS,eAAe,EAC/C,OACA;;EAEF,CAAC;;AAGH,MAAM,wBAAwB;AAE9B,SAAS,qBAAqB,eAAgC;CAgB7D,OAAO;;EAfU,gBACd,wDACA,yCAeO;;;;;;;;;;;EAbc,gBACrB;;;;;;;IAQA,GAec"}
|
package/package.json
CHANGED
|
@@ -1,64 +1,65 @@
|
|
|
1
1
|
{
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
}
|
|
2
|
+
"name": "@rooted-adapters/express",
|
|
3
|
+
"version": "1.0.0-alpha.1",
|
|
4
|
+
"description": "Adapter for server-side hosting with Express",
|
|
5
|
+
"keywords": [
|
|
6
|
+
"web-components",
|
|
7
|
+
"typescript",
|
|
8
|
+
"rooted"
|
|
9
|
+
],
|
|
10
|
+
"license": "MIT",
|
|
11
|
+
"repository": {
|
|
12
|
+
"type": "git",
|
|
13
|
+
"url": "git+https://github.com/Marvin-Brouwer/rooted.git",
|
|
14
|
+
"directory": "packages/adapters/express"
|
|
15
|
+
},
|
|
16
|
+
"homepage": "https://github.com/Marvin-Brouwer/rooted#readme",
|
|
17
|
+
"bugs": "https://github.com/Marvin-Brouwer/rooted/issues",
|
|
18
|
+
"sideEffects": false,
|
|
19
|
+
"publishConfig": {
|
|
20
|
+
"registry": "https://registry.npmjs.org/",
|
|
21
|
+
"access": "public",
|
|
22
|
+
"provenance": true
|
|
23
|
+
},
|
|
24
|
+
"files": [
|
|
25
|
+
"dist",
|
|
26
|
+
"readme.md"
|
|
27
|
+
],
|
|
28
|
+
"engines": {
|
|
29
|
+
"node": ">=22.0.0",
|
|
30
|
+
"pnpm": ">=10.0.0"
|
|
31
|
+
},
|
|
32
|
+
"exports": {
|
|
33
|
+
".": {
|
|
34
|
+
"import": "./dist/index.mjs",
|
|
35
|
+
"types": "./dist/index.d.mts"
|
|
36
|
+
}
|
|
37
|
+
},
|
|
38
|
+
"peerDependencies": {
|
|
39
|
+
"@rooted/adapter": "1.0.0-alpha.1",
|
|
40
|
+
"express": ">= 5.0.0",
|
|
41
|
+
"rolldown": ">= 1.0.0",
|
|
42
|
+
"vite": ">= 8.0.1"
|
|
43
|
+
},
|
|
44
|
+
"peerDependenciesMeta": {
|
|
45
|
+
"express": {
|
|
46
|
+
"optional": false
|
|
47
|
+
}
|
|
48
|
+
},
|
|
49
|
+
"devDependencies": {
|
|
50
|
+
"@rooted/adapter": "1.0.0-alpha.1",
|
|
51
|
+
"@rooted/development": "workspace:*",
|
|
52
|
+
"@rooted/tsdown": "workspace:^",
|
|
53
|
+
"@types/node": "^25.6.2",
|
|
54
|
+
"tsx": "^4.21.0",
|
|
55
|
+
"typescript": "^6.0.3",
|
|
56
|
+
"vite": "^8.0.12"
|
|
57
|
+
},
|
|
58
|
+
"scripts": {
|
|
59
|
+
"build": "pnpm build:ci",
|
|
60
|
+
"build:dev": "pnpm build:ci",
|
|
61
|
+
"build:ci": "tsdown",
|
|
62
|
+
"watch": "tsdown --watch --no-clean",
|
|
63
|
+
"clean": "rm -rf dist"
|
|
64
|
+
}
|
|
65
|
+
}
|
package/readme.md
CHANGED
|
@@ -27,4 +27,26 @@ export default rootedManifest({
|
|
|
27
27
|
})
|
|
28
28
|
```
|
|
29
29
|
|
|
30
|
+
## Adding middleware
|
|
31
|
+
|
|
32
|
+
If you need to register Express middleware (proxies, auth, rate-limiting), point `middlewarePath` at a folder of `.mjs` files. Use the `createMiddleware` helper so editors pick up the Express instance type:
|
|
33
|
+
|
|
34
|
+
```ts
|
|
35
|
+
expressAdapter({ middlewarePath: './src/server-middleware' })
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
```ts
|
|
39
|
+
// src/server-middleware/01-api-proxy.mts
|
|
40
|
+
import { createMiddleware } from '@rooted-adapters/express'
|
|
41
|
+
import { createProxyMiddleware } from 'http-proxy-middleware'
|
|
42
|
+
|
|
43
|
+
export default createMiddleware((app) => {
|
|
44
|
+
app.use('/api', createProxyMiddleware({ target: process.env.API_URL }))
|
|
45
|
+
})
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
Files can be `.mts`, `.ts`, `.mjs`, or `.js` -- TypeScript is transpiled with rolldown at build time.
|
|
49
|
+
|
|
50
|
+
Full details in [advanced/server-middleware](https://github.com/Marvin-Brouwer/rooted/blob/main/docs/advanced/server-middleware.md).
|
|
51
|
+
|
|
30
52
|
More in the [adapters guide](https://github.com/Marvin-Brouwer/rooted/blob/main/docs/guide/adapters.md).
|
package/LICENSE
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
MIT License
|
|
2
|
-
|
|
3
|
-
Copyright (c) 2026 Marvin Brouwer
|
|
4
|
-
|
|
5
|
-
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
-
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
-
in the Software without restriction, including without limitation the rights
|
|
8
|
-
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
-
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
-
furnished to do so, subject to the following conditions:
|
|
11
|
-
|
|
12
|
-
The above copyright notice and this permission notice shall be included in all
|
|
13
|
-
copies or substantial portions of the Software.
|
|
14
|
-
|
|
15
|
-
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
-
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
-
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
-
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
-
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
-
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
-
SOFTWARE.
|