1.0.8 • Published 2 years ago
@julian5335/spring-boot-node v1.0.8
spring-boot-node
How to use
Create controller classes in a similar manner to spring boot
user.controller.ts
import { NextFunction, Request, Response } from 'express';
import { Controller, Delete, Get, Middleware, Post, Put } from '@julian5335/spring-boot-node';
interface User {
    id: number
    name: string
}
@Controller('/api/users')
export default class UserController {
    private userId = 2
    private users: User[] = [
        { id: 1, name: 'Julian' },
        { id: 2, name: 'Arnold' },
    ];
    @Get()
    public get(req: Request, res: Response) {
        res.status(200).json(this.users);
    }
    
    @Get('/:id')
    public findById(req: Request, res: Response) {
        const id = Number(req.params.id)
        const user = this.users.find(x => x.id == id);
        if (user) {
            return res.status(200).json({ user });
        }
        return res.status(404).json({ message: 'User not found!' });
    }
    @Post()
    public add(req: Request, res: Response) {
        const name = req.body.name
        this.userId++
        const user = { id: this.userId, name }
        this.users.push(user);
        res.status(200).json(user);
    }
    @Put('/:id')
    public update(req: Request, res: Response) {
        const name = req.body.name
        const id = Number(req.params.id)
        const user = this.users.find(x => x.id == id);
        if (!user) {
            return res.status(400).json({ message: 'User not found!' })
        }
        user.name = name
        res.status(200).json(user);
    }
    @Delete('/:id')
    public delete(req: Request, res: Response) {
        const id = Number(req.params.id)
        const index = this.users.findIndex(x => x.id == id);
        if (index == -1) {
            return res.status(400).json({ message: 'User not found!' })
        }
        this.users.splice(index, 1)
        res.status(200).json();
    }
    @Middleware({ path:'/', order: 1 })
    public log1(req: Request, res: Response, next: NextFunction) {
        console.log("-----1-----")
        next()
    }
    @Middleware({ path:'/', order: 4 })
    public log3(req: Request, res: Response, next: NextFunction) {
        console.log("-----4-----")
        next()
    }
    @Middleware({ path:'/', order: 3 })
    public log4(req: Request, res: Response, next: NextFunction) {
        console.log("-----3-----")
        next()
    }
    @Middleware({ path:'/', order: 2 })
    public log2(req: Request, res: Response, next: NextFunction) {
        console.log("-----2-----")
        next()
    }
}Pass an array of controller classes into the App class and create the server with app.instance
The middleware functions run in the specified order before the http routes
index.ts
import * as http from 'http';
import App, { Config } from '@julian5335/spring-boot-node'
import UserController from './controllers/user.controller'
const config: Config = {
    controllers = [ UserController ]
}
const app = new App(config)
const PORT = 3000;
const server = http.createServer(app.instance);
server.listen(PORT, () => {
    console.log(`Server is listening on :${PORT}`);
});Note: Add "experimentalDecorators": true
tsconfig.json
{
  "compilerOptions": {
    "target": "es2016",
    "module": "commonjs",
    "rootDir": "src",
    "outDir": "dist",
    "esModuleInterop": true,
    "forceConsistentCasingInFileNames": true,
    "strict": true,
    "skipLibCheck": true,
    "experimentalDecorators": true
  }
}Error handling
Catch all errors including promises
Extend the HandledError class to implement custom errors
export class BadRequestError extends HandledError {
    constructor(message: string, key?: string) {
        super({ [key ?? "_"]: message }, 400)
    }
}
export class ForbiddenError extends HandledError {
    constructor(message: string, key?: string) {
        super({ [key ?? "_"]: message }, 403)
    }
}
@Get('/:id')
public async findById(req: Request, res: Response) {
    await this.delay(1000)
    throw new BadRequestError("Not found", "id")
    const id = Number(req.params.id)
    const user = this.users.find(x => x.id == id);
    if (user) {
        return res.status(200).json({ user });
    }
    return res.status(404).json({ message: 'User not found!' });
}The above code results in the following error response
{
    "errors": {
        "id": "Not found"
    }
}Any uncaught errors result in the following error response.
{
    "errors": {
        "_": "Internal server error"
    }
}Further customization is available by creating a custom error handling class
import { Request, Response } from "express"
import { ErrorHandler } from "@julian5335/spring-boot-node"
export class CustomError1 extends Error { }
export class CustomError2 extends Error {
    html = "<h1>Custom Error 2</h1>"
}
export default class CustomErrorHandler {
    @ErrorHandler(CustomError1)
    handleCustomError1(req: Request, res: Response) {
        return res.status(403).json({ "message": "custom error 1" })
    }
    @ErrorHandler(CustomError2)
    handleCustomError2(req: Request, res: Response, e: CustomError2) {
        return res.send(e.html)
    }
}And add the error handling class to App
const config: Config = {
    controllers = [ UserController ],
    errorHandlerClass = CustomErrorHandler
}
const app = new App(config)