vite-plugin-api-routes
![Build Status](https://travis-ci.org/yracnet/vite-plugin-api-routes.svg?branch=main)
Apology for Project Renaming
🙏 Dear Community,
We sincerely apologize for the recent project name changes. After careful consideration and feedback, we've settled on the name vite-plugin-api-routes. We understand that these changes might have caused confusion, and we appreciate your understanding.
Thank you for your continued support and flexibility.
Best regards,
Willyams Yujra
Additional Resources
For more detailed information and resources related to vite-plugin-api-routes
, please refer to the following:
Vision
Enhance API routing in ViteJS based on directory structure for improved visibility and project structure in Node.js and Express.
See the tutorial
Motivation
- Simplify project configuration.
- Convert the directory tree into route rules.
Example Structure:
> tree src/api/
src/api/:
├───admin
│ ├───auth
│ │ ├───login.js
│ │ └───status.js
│ └───user
│ ├───index.js
│ └───[userId] //Remix Format
│ ├───index.js
│ └───detail.js
├───site
│ ├───article
│ │ ├───$articleId.js //NextJS Format
│ │ └───new.js
│ └───page
│ ├───$pageId.js
│ └───new.js
└───index.js
The directory tree is exported as router rules tree:
GET /api/site/
GET /api/routers
USE /api/admin/user
GET /api/admin/user
GET /api/admin/user/
POST /api/admin/user/
GET /api/admin/auth/login
POST /api/admin/auth/login
GET /api/site/article/new
GET /api/admin/auth/status
POST /api/admin/auth/status
GET /api/site/page/:pageId
GET /api/admin/user/:userId/
PUT /api/admin/user/:userId/
DELETE /api/admin/user/:userId/
GET /api/site/article/:articleId
GET /api/admin/user/:userId/detail
For example, the src/api/admin/user/$userId.js
file exports allowed request methods:
export const DELETE = (req, res, next) => {
res.send("DELETE REQUEST");
};
export const PUT = async (req, res, next) => {
res.send("PUT REQUEST");
};
export default [authMiddleware, secondMiddleware ];
Similarly, the [userId].js
or $userId.js
file name is exported as a request parameter /user/:userId
, following the Next.js/Remix framework.
How to Use
Install
yarn add vite-plugin-api-routes
Configuration
In vite.config.ts
:
import { defineConfig } from "vite";
import { pluginAPIRoutes } from "vite-plugin-api-routes";
export default defineConfig({
plugins: [
pluginAPIRoutes({
}),
],
});
Parameters
- moduleId: Name of the virtual module, default @api (used for imports, change if conflicts occur).
- server: The main file to build as the server app. See default file.
- handler: The main file to register the API. It is called in viteServer and is the default entry. See default file.
- configure: The configureFile centralizes server configuration for both development (viteServer) and production (express). This file is invoked in various lifecycle hooks of the server. See default file.
- routeBase: Base name route for all routes. The default value is api.
- dirs: List of directories to be scanned. The default value is [ { dir: 'src/api', route: '', exclude: []} ].
- include: Files and directories to include in the scan process. The default value is ["\\/.js", "\\/.ts"].
- exclude: Files and directories to exclude from the scan process. The default value is ["node_modules", ".git"].
- mapper: Mapping rules from exported functions to server instance methods.
- cacheDir: Cache Directory target to write temp files.
Mapper
Default Value
mapper: {
default: "use",
GET: "get",
POST: "post",
PUT: "put",
PATCH: "patch",
DELETE: "delete",
...mapper,
};
Custom Mapping
/vite.config.js
export default defineConfig({
plugins: [
pluginAPIRoutes({
mapper: {
PING: "get",
OTHER_POST: "post2",
PATCH: false,
},
}),
],
});
You can disable a method by setting its value to false. In the example PATCH: false
, the PATCH method is disabled.
/src/api/index.js
export const PING = (req, res, next)=>{
res.send({name:"Ping Service"});
}
export const OTHER_POST = (req, res, next)=>{
res.send({name:"Other Service"});
}
export const PATCH = (req, res, next)=>{
res.send({name:"Path Service"});
}
/src/handler.js or see handler.js
import express from "express";
import { applyRouters } from "@api/routers";
import * as configure from "@api/configure";
export const handler = express();
configure.handlerBefore?.(handler);
applyRouters(
(props) => {
const { method, route, path, cb } = props;
if (handler[method]) {
if(Array.isArray(cb)) {
handler[method](route, ...cb);
} else {
handler[method](route, cb);
}
} else {
console.log("Not Support", method, "for", route, "in", handler);
}
}
);
configure.handlerAfter?.(handler);
/src/server.ts or see server.ts
import { handler } from "@api/handler";
import { endpoints } from "@api/routers";
import * as configure from "@api/configure";
import express from "express";
const server = express();
configure.serverBefore?.(server);
const { PORT = 3000, PUBLIC_DIR = "import.meta.env.PUBLIC_DIR" } = process.env;
server.use("import.meta.env.BASE", express.static(PUBLIC_DIR));
server.use("import.meta.env.BASE_API", handler);
configure.serverAfter?.(server);
server.on("error", (error) => {
console.error(`Error at http://localhost:${PORT}`, error);
configure.serverError?.(server, error);
});
server.on("listening", () => {
console.log(`Ready at http://localhost:${PORT}`);
configure.serverListening?.(server, endpoints);
});
server.listen(PORT);
/src/configure.ts or see configure.ts
import { NextFunction, Request, Response, } from "express";
const devServerBefore = (viteServer: any, vite: any): void => {
console.log("devServerBefore");
};
const serverBefore = (server: Express) => {
console.log("serverBefore");
};
const serverAfter = (server: Express) => {
console.log("serverAfter");
}
const handlerBefore = (handler: Express) => {
handler.use(express.json());
handler.use(express.urlencoded({ extended: true }));
};
const handlerAfter = (server: Express) => {
console.log("handlerAfter");
}
type RouteModule = (req: Request, res: Response, next: NextFunction) => void;
const callbackBefore = (callback: RouteModule, route: any) => {
console.log("callbackBefore", callback);
return callback;
};
export default {
devServerBefore,
serverBefore,
serverAfter,
handlerBefore,
handlerAfter,
callbackBefore,
}
TypeScript Support
To leverage TypeScript models within your Vite.js project, follow these steps:
Reference the TypeScript Definitions:
Add a reference to the TypeScript definitions file [moduleId]/types.d.ts within your vite-env.d.ts file.
src/vite-env.d.ts
Utilize the TypeScript Models in Your Code:
Once you've referenced the required TypeScript definitions, you can incorporate them directly into your TypeScript code.
Incorporating a ViteServerHook model from vite-plugin-api-routes:
import { ViteServerHook } from "vite-plugin-api-routes/model";
export const viteServerBefore: ViteServerHook = (server, viteServer) => {
console.log("VITEJS SERVER");
};
NOTE:
In the server file, we do not explicitly declare the basic configuration. Instead, this responsibility is delegated to the configure file, ensuring a more modular and centralized approach to server setup and initialization.
TO DO:
- Duplicate declaration (GET in /user.ts and /user/index.ts). Handler definition is required.
- Extend the
mapper
attribute to support custom HTTP methods using a header attribute.