From shipshitdev-library
Guides NestJS API development with architecture, modules, DI, guards, interceptors, pipes, MongoDB/Mongoose integration, auth, DTOs, error handling, and production patterns.
npx claudepluginhub shipshitdev/skillsThis skill uses the workspace's default tool permissions.
Production NestJS patterns for TypeScript APIs. Stack: NestJS + MongoDB/Mongoose + TypeScript strict mode.
Provides UI/UX resources: 50+ styles, color palettes, font pairings, guidelines, charts for web/mobile across React, Next.js, Vue, Svelte, Tailwind, React Native, Flutter. Aids planning, building, reviewing interfaces.
Fetches up-to-date documentation from Context7 for libraries and frameworks like React, Next.js, Prisma. Use for setup questions, API references, and code examples.
Analyzes competition with Porter's Five Forces, Blue Ocean Strategy, and positioning maps to identify differentiation opportunities and market positioning for startups and pitches.
Production NestJS patterns for TypeScript APIs. Stack: NestJS + MongoDB/Mongoose + TypeScript strict mode.
Every feature is a self-contained module. No cross-module direct imports — use exported providers.
src/
├── app.module.ts # Root — imports feature modules only
├── common/ # Shared guards, pipes, filters, interceptors
│ ├── filters/
│ ├── guards/
│ ├── interceptors/
│ └── pipes/
├── config/ # ConfigModule setup
└── {feature}/
├── {feature}.module.ts
├── {feature}.controller.ts
├── {feature}.service.ts
├── {feature}.repository.ts # optional, wraps Mongoose model
├── dto/
│ ├── create-{feature}.dto.ts
│ └── update-{feature}.dto.ts
├── schemas/
│ └── {feature}.schema.ts
└── {feature}.types.ts
@Injectable({ scope: Scope.DEFAULT }) (singleton) unless you need request-scopedforwardRef only as last resortTest.createTestingModule — always mock external services@Controller('resources')
@UseGuards(JwtAuthGuard)
@UseInterceptors(ResponseTransformInterceptor)
export class ResourceController {
constructor(private readonly resourceService: ResourceService) {}
@Get()
async findAll(@Query() query: PaginationQueryDto) {
return this.resourceService.findAll(query);
}
@Post()
@HttpCode(HttpStatus.CREATED)
async create(@Body() dto: CreateResourceDto, @CurrentUser() user: UserDocument) {
return this.resourceService.create(dto, user._id);
}
}
Rules:
@Body(), @Query(), @Param() with DTOs@CurrentUser() custom decorator, never @Req()import { IsString, IsEnum, IsOptional, MinLength, MaxLength } from 'class-validator';
import { Transform } from 'class-transformer';
export class CreateResourceDto {
@IsString()
@MinLength(1)
@MaxLength(255)
name: string;
@IsEnum(ResourceStatus)
status: ResourceStatus;
@IsOptional()
@IsString()
@Transform(({ value }) => value?.trim())
description?: string;
}
Global validation pipe in main.ts:
app.useGlobalPipes(new ValidationPipe({
whitelist: true, // strip unknown props
forbidNonWhitelisted: true,
transform: true, // auto-transform primitives
transformOptions: { enableImplicitConversion: true },
}));
// schema
@Schema({ timestamps: true, versionKey: false })
export class Resource {
@Prop({ required: true, index: true })
name: string;
@Prop({ type: Types.ObjectId, ref: 'User', required: true, index: true })
userId: Types.ObjectId;
@Prop({ enum: ResourceStatus, default: ResourceStatus.ACTIVE })
status: ResourceStatus;
}
export const ResourceSchema = SchemaFactory.createForClass(Resource);
export type ResourceDocument = Resource & Document;
// service
@Injectable()
export class ResourceService {
constructor(
@InjectModel(Resource.name) private readonly model: Model<ResourceDocument>,
) {}
async findAll(userId: Types.ObjectId, query: PaginationQueryDto) {
const { page = 1, limit = 20 } = query;
return this.model
.find({ userId, deletedAt: null })
.sort({ createdAt: -1 })
.skip((page - 1) * limit)
.limit(limit)
.lean()
.exec();
}
}
Rules:
.lean() for read queries (plain objects, ~30% faster).exec() to get a real PromiseTypes.ObjectId not string for references in service layerdeletedAt: Date | null, never hard delete user data// JWT strategy
@Injectable()
export class JwtStrategy extends PassportStrategy(Strategy) {
constructor(configService: ConfigService) {
super({
jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(),
secretOrKey: configService.get<string>('JWT_SECRET'),
ignoreExpiration: false,
});
}
async validate(payload: JwtPayload): Promise<UserDocument> {
// return value is injected as req.user
return { _id: payload.sub, email: payload.email };
}
}
// custom decorator
export const CurrentUser = createParamDecorator(
(data: unknown, ctx: ExecutionContext) => ctx.switchToHttp().getRequest().user,
);
@Injectable()
export class ResourceOwnerGuard implements CanActivate {
constructor(private readonly resourceService: ResourceService) {}
async canActivate(context: ExecutionContext): Promise<boolean> {
const { user, params } = context.switchToHttp().getRequest();
const resource = await this.resourceService.findById(params.id);
return resource?.userId.equals(user._id) ?? false;
}
}
@Catch()
export class GlobalExceptionFilter implements ExceptionFilter {
private readonly logger = new Logger(GlobalExceptionFilter.name);
catch(exception: unknown, host: ArgumentsHost) {
const ctx = host.switchToHttp();
const response = ctx.getResponse<Response>();
if (exception instanceof HttpException) {
return response.status(exception.getStatus()).json({
statusCode: exception.getStatus(),
message: exception.message,
});
}
this.logger.error('Unhandled exception', exception instanceof Error ? exception.stack : exception);
return response.status(500).json({ statusCode: 500, message: 'Internal server error' });
}
}
// config/app.config.ts
export default registerAs('app', () => ({
port: parseInt(process.env.PORT ?? '3000', 10),
jwtSecret: process.env.JWT_SECRET,
mongoUri: process.env.MONGO_URI,
}));
// access in service
constructor(private config: ConfigService) {}
const port = this.config.get<number>('app.port');
Never use process.env directly outside config files.
lean() on all read queriesfind() filter or sort(){ userId: 1, createdAt: -1 }select() to project only needed fields on large documents@nestjs/cache-manager for expensive reads| Wrong | Right |
|---|---|
| Business logic in controller | Move to service |
any type anywhere | Define interface/DTO |
console.log | new Logger(ClassName.name) |
req.user directly | @CurrentUser() decorator |
| Hard-coding env vars | ConfigService |
.find() without .lean() on reads | Always .lean().exec() |
string for ObjectId refs | Types.ObjectId |
nestjs-queue-architect — BullMQ async job patternsnestjs-testing-expert — Jest unit + integration testingmongodb-migration-expert — schema migrationserror-handling-expert — global error strategy