Skip to Content
스킬백엔드 패턴

백엔드 패턴

다운로드 후 ~/.claude/skills/ 폴더에 복사하여 사용하세요

확장 가능한 서버 사이드 애플리케이션을 위한 백엔드 아키텍처 패턴과 모범 사례입니다.

API 설계 패턴

RESTful API 구조

// 리소스 기반 URL GET /api/markets # 리소스 목록 GET /api/markets/:id # 단일 리소스 조회 POST /api/markets # 리소스 생성 PUT /api/markets/:id # 리소스 교체 PATCH /api/markets/:id # 리소스 업데이트 DELETE /api/markets/:id # 리소스 삭제 // 필터링, 정렬, 페이지네이션을 위한 쿼리 파라미터 GET /api/markets?status=active&sort=volume&limit=20&offset=0

레포지토리 패턴

// 데이터 접근 로직 추상화 interface MarketRepository { findAll(filters?: MarketFilters): Promise<Market[]> findById(id: string): Promise<Market | null> create(data: CreateMarketDto): Promise<Market> update(id: string, data: UpdateMarketDto): Promise<Market> delete(id: string): Promise<void> } class SupabaseMarketRepository implements MarketRepository { async findAll(filters?: MarketFilters): Promise<Market[]> { let query = supabase.from('markets').select('*') if (filters?.status) { query = query.eq('status', filters.status) } if (filters?.limit) { query = query.limit(filters.limit) } const { data, error } = await query if (error) throw new Error(error.message) return data } }

서비스 레이어 패턴

// 비즈니스 로직을 데이터 접근과 분리 class MarketService { constructor(private marketRepo: MarketRepository) {} async searchMarkets(query: string, limit: number = 10): Promise<Market[]> { // 비즈니스 로직 const embedding = await generateEmbedding(query) const results = await this.vectorSearch(embedding, limit) // 전체 데이터 조회 const markets = await this.marketRepo.findByIds(results.map(r => r.id)) // 유사도로 정렬 return markets.sort((a, b) => { const scoreA = results.find(r => r.id === a.id)?.score || 0 const scoreB = results.find(r => r.id === b.id)?.score || 0 return scoreA - scoreB }) } }

미들웨어 패턴

// 요청/응답 처리 파이프라인 export function withAuth(handler: NextApiHandler): NextApiHandler { return async (req, res) => { const token = req.headers.authorization?.replace('Bearer ', '') if (!token) { return res.status(401).json({ error: 'Unauthorized' }) } try { const user = await verifyToken(token) req.user = user return handler(req, res) } catch (error) { return res.status(401).json({ error: 'Invalid token' }) } } } // 사용 예 export default withAuth(async (req, res) => { // 핸들러가 req.user에 접근 가능 })

데이터베이스 패턴

쿼리 최적화

// 필요한 컬럼만 선택 const { data } = await supabase .from('markets') .select('id, name, status, volume') .eq('status', 'active') .order('volume', { ascending: false }) .limit(10)

N+1 쿼리 방지

// 배치 조회 const markets = await getMarkets() const creatorIds = markets.map(m => m.creator_id) const creators = await getUsers(creatorIds) // 1개 쿼리 const creatorMap = new Map(creators.map(c => [c.id, c])) markets.forEach(market => { market.creator = creatorMap.get(market.creator_id) })

트랜잭션 패턴

async function createMarketWithPosition( marketData: CreateMarketDto, positionData: CreatePositionDto ) { // Supabase 트랜잭션 사용 const { data, error } = await supabase.rpc('create_market_with_position', { market_data: marketData, position_data: positionData }) if (error) throw new Error('Transaction failed') return data }

캐싱 전략

Redis 캐싱 레이어

class CachedMarketRepository implements MarketRepository { constructor( private baseRepo: MarketRepository, private redis: RedisClient ) {} async findById(id: string): Promise<Market | null> { // 캐시 먼저 확인 const cached = await this.redis.get(`market:${id}`) if (cached) { return JSON.parse(cached) } // 캐시 미스 - 데이터베이스에서 조회 const market = await this.baseRepo.findById(id) if (market) { // 5분간 캐시 await this.redis.setex(`market:${id}`, 300, JSON.stringify(market)) } return market } async invalidateCache(id: string): Promise<void> { await this.redis.del(`market:${id}`) } }

Cache-Aside 패턴

async function getMarketWithCache(id: string): Promise<Market> { const cacheKey = `market:${id}` // 캐시 시도 const cached = await redis.get(cacheKey) if (cached) return JSON.parse(cached) // 캐시 미스 - DB에서 조회 const market = await db.markets.findUnique({ where: { id } }) if (!market) throw new Error('Market not found') // 캐시 업데이트 await redis.setex(cacheKey, 300, JSON.stringify(market)) return market }

오류 처리 패턴

중앙집중식 오류 핸들러

class ApiError extends Error { constructor( public statusCode: number, public message: string, public isOperational = true ) { super(message) Object.setPrototypeOf(this, ApiError.prototype) } } export function errorHandler(error: unknown, req: Request): Response { if (error instanceof ApiError) { return NextResponse.json({ success: false, error: error.message }, { status: error.statusCode }) } if (error instanceof z.ZodError) { return NextResponse.json({ success: false, error: '검증 실패', details: error.errors }, { status: 400 }) } // 예상치 못한 오류 로깅 console.error('Unexpected error:', error) return NextResponse.json({ success: false, error: '내부 서버 오류' }, { status: 500 }) }

지수 백오프 재시도

async function fetchWithRetry<T>( fn: () => Promise<T>, maxRetries = 3 ): Promise<T> { let lastError: Error for (let i = 0; i < maxRetries; i++) { try { return await fn() } catch (error) { lastError = error as Error if (i < maxRetries - 1) { // 지수 백오프: 1초, 2초, 4초 const delay = Math.pow(2, i) * 1000 await new Promise(resolve => setTimeout(resolve, delay)) } } } throw lastError! }

인증 & 인가

JWT 토큰 검증

import jwt from 'jsonwebtoken' interface JWTPayload { userId: string email: string role: 'admin' | 'user' } export function verifyToken(token: string): JWTPayload { try { const payload = jwt.verify(token, process.env.JWT_SECRET!) as JWTPayload return payload } catch (error) { throw new ApiError(401, 'Invalid token') } }

역할 기반 접근 제어

type Permission = 'read' | 'write' | 'delete' | 'admin' const rolePermissions: Record<User['role'], Permission[]> = { admin: ['read', 'write', 'delete', 'admin'], moderator: ['read', 'write', 'delete'], user: ['read', 'write'] } export function hasPermission(user: User, permission: Permission): boolean { return rolePermissions[user.role].includes(permission) } export function requirePermission(permission: Permission) { return async (request: Request) => { const user = await requireAuth(request) if (!hasPermission(user, permission)) { throw new ApiError(403, '권한이 없습니다') } return user } }

레이트 리미팅

class RateLimiter { private requests = new Map<string, number[]>() async checkLimit( identifier: string, maxRequests: number, windowMs: number ): Promise<boolean> { const now = Date.now() const requests = this.requests.get(identifier) || [] // 윈도우 밖의 오래된 요청 제거 const recentRequests = requests.filter(time => now - time < windowMs) if (recentRequests.length >= maxRequests) { return false // 레이트 리밋 초과 } // 현재 요청 추가 recentRequests.push(now) this.requests.set(identifier, recentRequests) return true } }

로깅 & 모니터링

interface LogContext { userId?: string requestId?: string method?: string path?: string [key: string]: unknown } class Logger { log(level: 'info' | 'warn' | 'error', message: string, context?: LogContext) { const entry = { timestamp: new Date().toISOString(), level, message, ...context } console.log(JSON.stringify(entry)) } info(message: string, context?: LogContext) { this.log('info', message, context) } error(message: string, error: Error, context?: LogContext) { this.log('error', message, { ...context, error: error.message, stack: error.stack }) } }

백엔드 패턴은 확장 가능하고 유지보수 가능한 서버 사이드 애플리케이션을 가능하게 합니다. 복잡도 수준에 맞는 패턴을 선택하세요.

Last updated on