Vite Plugin Node
A vite plugin to allow you to use vite as node dev server.
Features
- All the perks from Vite plus:
- Node server HMR! (hot module replacement)
- Support Express, Fastify, Koa and Nest out of box
- Support Custom Request Adapter
- You can choose to use
esbuild
or swc
to compile your typescript files
Get started
-
Install vite and this plugin with your favorite package manager, here use npm as example:
npm install vite vite-plugin-node -D
-
Create a vite.config.ts
file in your project root to config vite to actually use this plugin:
import { defineConfig } from 'vite';
import { VitePluginNode } from 'vite-plugin-node';
export default defineConfig({
server: {
port: 3000
},
plugins: [
...VitePluginNode({
adapter: 'express',
appPath: './app.ts',
exportName: 'viteNodeApp',
tsCompiler: 'esbuild',
})
]
}
-
Update your server entry to export your app named viteNodeApp
or the name you configured.
ExpressJs
const app = express();
if (process.env.NODE_ENV === 'production') {
app.listen(3000)
}
export const viteNodeApp = app;
KoaJs
import Koa from 'koa';
const app = new Koa();
if (process.env.NODE_ENV === 'production') {
app.listen(3000)
}
export const viteNodeApp = app;
Fastify
import fastify from 'fastify';
const app = fastify();
if (process.env.NODE_ENV === 'production') {
app.listen(3000)
}
export const viteNodeApp = app;
if the app created by an async factory function you can just export the promise.
import fastify from 'fastify';
const app = async (options) => {
const app = fastify(options);
return app
}
export const viteNodeApp = app(options);
NestJs
import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
if (process.env.NODE_ENV === 'production') {
async function bootstrap() {
const app = await NestFactory.create(AppModule);
await app.listen(3000);
}
bootstrap();
}
export const viteNodeApp = NestFactory.create(AppModule);
-
Add a npm script to run the dev server:
"scripts": {
"dev": "vite"
},
-
Run the script! npm run dev
Custom Adapter
If your favorite framework not supported yet, you can either create an issue to request it or use the adapter
option to tell the plugin how to pass down the request to your app. You can take a look how the supported frameworks implementations from the ./src/server
folder.
Example:
import { defineConfig } from 'vite';
import { VitePluginNode } from 'vite-plugin-node';
export default defineConfig({
plugins: [
...VitePluginNode({
adapter: function(app, req, res) {
app(res, res)
},
appPath: './app.ts'
})
]
})
Examples
See the examples folder.
Why?
While frontend development tooling is evolving rapidly in recent years, backend DX is still like in stone age. No hot module replacement; Typescript recompiling slow as funk; Lack of plugin system etc. Thanks to Vitejs created by Evan You (A.K.A creator of vuejs; my biggest idol developer), makes all those dreams for backend development come true!
How?
Vite by design have a middlewareMode which allow us to use vite programmatically inside other module. It's originally made for SSR web app originally. So that for each request, vite can load the renderer to render the latest changes you made to your app. This plugin leverage this feature to load and execute your server app entry.
You may ask isn't super slow since it re-compile/reload entire app from the entry? The answer is NO, because vite is smart. Vite has a builtin module graph as a cache layer, the graph is built up at the first time your app load. After that, when you update one file, vite will only invalidate itself and its parents modules, so that for next request, only those invalidated module need to be re-compiled which is super fast thanks to esbuild and swc.
To-Do
As this plugin just fresh developed, there are still lots ideas need to be implemented, including:
Bugs
Create an issue if you found any bugs to help me to improve this project please!