스킬
백엔드 패턴

백엔드 패턴

다운로드 후 ~/.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
    })
  }
}

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