@secjs/http v1.0.6
Http 📶
Simple http server for any NodeJS Project
The intention behind this repository is to always maintain a Http package to any NodeJS project.
Installation
To use the high potential from this package you need to install first this other packages from SecJS, it keeps as dev dependency because one day
@secjs/corewill install everything once.
npm install @secjs/ioc @secjs/env @secjs/utils @secjs/exceptionsThen you can install the package using:
npm install @secjs/httpUsage
Http
Use Http class to create the http server and map all your routes
import { Http, ContextContract } from '@secjs/http'
const server = new Http()
// Middleware
server.use((ctx: ContextContract) => ctx.data.param = 'param')
server.get('/', ({ response }) => {
  response
    .status(200)
    .send({ hello: 'world!', param: ctx.data.param })
})
server.listen().then(address => `Server running on address ${address}`)Router
Use Router class to map all the groups, resources and normal routes of the application
import { TestController } from './TestController'
import { TestMiddleware } from './TestMiddleware'
import { Http, Router, ContextContract } from '@secjs/http'
// With router class you can map your routes inside groups and create resources
const http = new Http() // First you need to create the Http server
const Route = new Router(http)
// Create a route group to set the API version as prefix
Route.group(() => {
  Route.get('posts', (ctx: ContextContract) => {
    ctx.response.status(200).send({
      hello: 'world',
      userId: ctx.data.userId,
      postId: ctx.data.postId
    })
  })
    .middleware((ctx: ContextContract) => {
      ctx.data.postId = 1
      ctx.next()
    })
  // You can create a Resource route to create all the Http methods (index, store, show, update and delete)
  Route.resource('tests', new TestController()).except(['show']) // You can use except to create all minus show method
})
  .prefix('/api/v1')
  // You can how many middlewares you want using builder pattern, .middleware, .middleware, .middleware ....
  .middleware((ctx: ContextContract) => {
    ctx.data.userId = 1
    ctx.next()
  })
// You can also use middlewares 
// You need to call register method in the end to register all the routes in the Http server
Route.register()
http.listen()Registering routes like this could be a little difficult, so you can use the global Container from @secjs/ioc to register controllers and middlewares in the container
import '@secjs/ioc/src/utils/global' // Will load the class Container in global runtime and in TS types
Container.singleton(TestController, 'TestController')
Container.singleton(
  // Named middlewares
  { 
    // Is extremelly important that middleware implement MiddlewareContract from @secjs/http
    'test.auth': new TestMiddleware(), 
    'test.hello': new TestMiddleware() 
  },
  'Middlewares',
)
// Now you can start using string names in routes
Route.group(() => {
  Route.resource('posts', 'TestController').only(['index', 'store']).middleware('test.auth')
})
  .prefix('/api/v2')
  .middleware('test.hello')
Route.register()
http.listen()Creating a Middleware
With Http you can define three different execution times for one middleware
import { 
  Router,
  MiddlewareContract,
  HandleContextContract,
  InterceptContextContract,
  TerminateContextContract 
} from '@secjs/http'
export class Middleware implements MiddlewareContract {
  // Handle method will be executed before the controller method handler
  // This is the normal middleware
  async handle(ctx: HandleContextContract) {
    ctx.data.userId = '1'
    ctx.next()
  }
  // Intercept method will be executed before the response goes to client
  async intercept(ctx: InterceptContextContract) {
    // You can use intercept to rewrite or add some information to the response
    ctx.body.intercepted = true
    ctx.response.status(304)
    // Example
    if ('intercept method logic changes the body') {
      // In intercept method, next function needs to receive the new body payload as parameter.
      ctx.next({ hello: 'intercepted', ...ctx.body })
    } else {
      // If your logic does not change the body you can just do like this
      ctx.next(ctx.body)
    }
  }
  // Terminate method will be executed after the response goes to client
  async terminate(ctx: TerminateContextContract) {
    // You can use terminate to save metrics of the request in an Elastic for example
    console.log('Terminate middleware executed!')
    ctx.next()
  }
}Now we can use Router to set the middleware in some route
Container.singleton(
  {
    'middleware': new Middleware(),
  },
  'Middlewares',
)
// If you use named middlewares in Router, he will register all the three methods of Middleware class.
Route.get('middlewares', 'TestController.index').middleware('middleware')
// But you can instantiate the middleware and will register all the three methods
Route
  // You can use controller method to set the default controller of Router
  .controller(new TestController())
  .get('middlewares', 'index')
  .middleware(new Middleware())
// Or you can set only the method and as second parameter the middleware type
Route
  .get('middlewares', new TestController().index)
  .middleware(new Middleware().intercept, 'intercept')License
Made with 🖤 by jlenon7 :wave: