Skillshub nestjs-controllers-services

Controller/Service separation and Custom Decorators. Use when defining NestJS controllers, services, or custom parameter decorators. (triggers: **/*.controller.ts, **/*.service.ts, Controller, Injectable, ExecutionContext, createParamDecorator)

install
source · Clone the upstream repo
git clone https://github.com/ComeOnOliver/skillshub
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/ComeOnOliver/skillshub "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/HoangNguyen0403/agent-skills-standard/nestjs-controllers-services" ~/.claude/skills/comeonoliver-skillshub-nestjs-controllers-services && rm -rf "$T"
manifest: skills/HoangNguyen0403/agent-skills-standard/nestjs-controllers-services/SKILL.md
source content

NestJS Controllers & Services Standards

Priority: P0 (FOUNDATIONAL)

Layer separation standards and dependency injection patterns for NestJS applications.

Controllers

  • Role: Handler only. Delegate all logic to Services.

  • Context: Use

    ExecutionContext
    helpers (
    switchToHttp()
    ) for platform-agnostic code.

  • Custom Decorators:

    • Avoid:
      @Request() req
      ->
      req.user
      (Not type-safe).
    • Pattern: Create typed decorators like
      @CurrentUser()
      ,
      @DeviceIp()
      .
    import { RequestWithUser } from 'src/common/interfaces/request.interface';
    
    export const CurrentUser = createParamDecorator(
      (data: unknown, ctx: ExecutionContext): User => {
        const request = ctx.switchToHttp().getRequest<RequestWithUser>();
        return request.user;
      },
    );
    

DTOs & Validation

  • Strictness:
    • whitelist: true
      : Strip properties without decorators.
    • Critical:
      forbidNonWhitelisted: true
      : Throw error if unknown properties exist.
  • Transformation:
    • transform: true
      : Auto-convert primitives (String '1' -> Number 1) and instantiate DTO classes.
  • Documentation:
    • Automation: Use the
      @nestjs/swagger
      CLI plugin (
      nest-cli.json
      ) to auto-detect DTO properties without manual
      @ApiProperty()
      tags.

Interceptors (Response Mapping)

  • Standardization: specific responses should be mapped in Interceptors, not Controllers.
    • Use
      map()
      to wrap success responses (e.g.
      { data: T }
      ).
    • Refer to API Standards for
      PageDto
      and
      ApiResponse
      .
    • Use
      catchError()
      to map low-level errors (DB constraints) to
      HttpException
      (e.g.
      ConflictException
      ) before they hit the global filter.

Services & Business Logic

  • Singleton: Default.
  • Stateless: Do not store request-specific state in class properties unless identifying as
    Scope.REQUEST
    .

Pipes & Validation

  • Global: Register
    ValidationPipe
    globally.
  • Route Params: Fail fast. Always use
    ParseIntPipe
    ,
    ParseUUIDPipe
    on all ID parameters.
@Get(':id')
findOne(@Param('id', ParseIntPipe) id: number) { ... }

Lifecycle Events

  • Init: Use
    OnModuleInit
    for connection setup.
  • Destroy: Use
    OnApplicationShutdown
    for cleanup. (Requires
    enableShutdownHooks()
    ).

Anti-Patterns

  • No business logic in controllers: Delegate everything to Services; controllers only parse and respond.
  • No req.user access: Create typed
    @CurrentUser()
    decorator instead of accessing raw
    req
    .
  • No REQUEST scope by default: Use SINGLETON; it makes the entire injection chain request-scoped.