Bundling and packaging
After creating your universal middleware, you need to bundle it before publishing.
You can use either rollup-based tools (such as rollup or vite) or esbuild-based tools (like esbuild or tsup) for bundling.
Bundling
ts
import { rollup } from "rollup";
import { nodeResolve } from "@rollup/plugin-node-resolve";
import typescript from "@rollup/plugin-typescript";
import universalMiddleware from "universal-middleware/rollup";
// See https://github.com/magne4000/universal-middleware/blob/main/packages/universal-middleware/test/rollup.test.ts
// for a list of possible options and usage
const result = await rollup({
// Your middlewares and handlers must all be added as input
input: "./src/middleware/demo.middleware.ts",
plugins: [
universalMiddleware(),
// usually required
nodeResolve(),
// required if using typescript
typescript(),
]
});
ts
// vite.config.js
import { resolve } from "node:path";
import { defineConfig } from "vite";
import universalMiddleware from "universal-middleware/rollup";
export default defineConfig({
plugins: [universalMiddleware()],
build: {
lib: {
// Could also be a dictionary or array of multiple entry points
entry: resolve(__dirname, 'src/middleware/demo.middleware.ts'),
name: 'My Awesome Middleware',
fileName: 'some-lib',
}
},
});
ts
import { build } from "esbuild";
import universalMiddleware from "universal-middleware/esbuild";
// See https://github.com/magne4000/universal-middleware/blob/main/packages/universal-middleware/test/esbuild.test.ts
// for a list of possible options and usage
const result = await build({
entryPoints: ["./src/middleware/demo.middleware.ts"],
plugins: [
universalMiddleware(),
],
outdir: "dist",
bundle: true,
platform: "neutral",
format: "esm",
target: "es2022",
splitting: true,
});
ts
// tsup.config.ts
import { defineConfig } from "tsup";
import universalMiddleware from "universal-middleware/esbuild";
export default defineConfig([
{
entry: {
"middlewares/demo": "./src/middlewares/demo.middleware.ts",
},
format: ["esm"],
platform: "neutral",
target: "es2022",
esbuildPlugins: [universalMiddleware()],
esbuildOptions(opts) {
opts.outbase = "src";
},
bundle: true,
},
]);
Plugin options
The bundler plugin also accepts the following options:
ts
universalMiddleware({
// Only generate files for selected servers. All enabled by default
servers?: ('hono' | 'express' | 'hattip' | 'fastify' | 'h3' | 'webroute' | 'cloudflare-pages' | 'cloudflare-worker' | 'elysia')[];
// akin to esbuild `entryNames` for generated "exports" in package.json
serversExportNames?: string;
// akin to ebsuild `entryNames` for generated "exports" in package.json
entryExportNames?: string;
// Disables some warning
ignoreRecommendations?: boolean;
// No auto writing in package.json. All info available to do it manually in `buildEnd`
doNotEditPackageJson?: boolean;
// Generate typings. true by default
dts?: boolean;
// By default, generated exports are self-contained.
// You can opt-out of this behaviour and have @universal-middleware/* packages added as `dependencies` instead
externalDependencies?: boolean;
// Hook called when bundle is generated
// For details, check https://github.com/magne4000/universal-middleware/blob/main/packages/universal-middleware/src/plugin.ts
buildEnd?: (report: Report[]) => void | Promise<void>;
});
Building
Once the build is executed, your dist folder will contain the following files:
dist
└─ middlewares
# raw middleware, compiled to js + types
├─ demo.d.ts
├─ demo.js
# compiled adapters + types
├─ universal-express-middleware-demo.middleware.d.ts
├─ universal-express-middleware-demo.middleware.js
├─ universal-h3-middleware-demo.middleware.d.ts
├─ universal-h3-middleware-demo.middleware.js
├─ universal-hono-middleware-demo.middleware.d.ts
├─ universal-hono-middleware-demo.middleware.js
└─ ...
Your project's package.json
will be updated to contains the necessary exports
.
Details
json5
{
// ...
"exports": {
"./middlewares/demo-middleware": {
"types": "./dist/middlewares/demo.d.ts",
"import": "./dist/middlewares/demo.js",
"default": "./dist/middlewares/demo.js"
},
"./middlewares/demo-middleware-hono": {
"types": "./dist/middlewares/universal-hono-middleware-demo.middleware.d.ts",
"import": "./dist/middlewares/universal-hono-middleware-demo.middleware.js",
"default": "./dist/middlewares/universal-hono-middleware-demo.middleware.js"
},
"./middlewares/demo-middleware-express": {
"types": "./dist/middlewares/universal-express-middleware-demo.middleware.d.ts",
"import": "./dist/middlewares/universal-express-middleware-demo.middleware.js",
"default": "./dist/middlewares/universal-express-middleware-demo.middleware.js"
},
"./middlewares/demo-middleware-h3": {
"types": "./dist/middlewares/universal-h3-middleware-demo.middleware.d.ts",
"import": "./dist/middlewares/universal-h3-middleware-demo.middleware.js",
"default": "./dist/middlewares/universal-h3-middleware-demo.middleware.js"
},
// ...
}
}
TIP
You can opt out of this behaviour by setting doNotEditPackageJson: true
in the plugin options.
Publishing
Your package should now be ready to be published! 🚀