Claude Code: Claude Code 활용 가이드
claude code: Claude Code 활용. 실용적인 팁과 코드 예시를 포함합니다.
Go言語개발でClaude Codeを활용するメリット
GoはシンプルなAPI설계、強力な並行処理、高速な컴파일が特徴です。Claude Code는 Goのイディオムを理解し、에러 핸들링、ゴルーチン、チャネルなどGo特有のパターンを正確に구현할 수 있습니다。
Web APIの개발
> Goの標準라이브러리とchiルーターでREST API를 생성해줘。
> 미들웨어、유효성 검사、構造化로그も구현して。
// main.go
package main
import (
"context"
"encoding/json"
"log/slog"
"net/http"
"os"
"os/signal"
"time"
"github.com/go-chi/chi/v5"
"github.com/go-chi/chi/v5/middleware"
"github.com/google/uuid"
)
type Task struct {
ID string `json:"id"`
Title string `json:"title"`
Description string `json:"description,omitempty"`
Completed bool `json:"completed"`
CreatedAt time.Time `json:"created_at"`
}
type CreateTaskRequest struct {
Title string `json:"title"`
Description string `json:"description,omitempty"`
}
type TaskStore struct {
tasks map[string]*Task
}
func NewTaskStore() *TaskStore {
return &TaskStore{tasks: make(map[string]*Task)}
}
func main() {
logger := slog.New(slog.NewJSONHandler(os.Stdout, nil))
slog.SetDefault(logger)
store := NewTaskStore()
r := chi.NewRouter()
r.Use(middleware.RequestID)
r.Use(middleware.RealIP)
r.Use(requestLogger)
r.Use(middleware.Recoverer)
r.Use(middleware.Timeout(30 * time.Second))
r.Route("/api/tasks", func(r chi.Router) {
r.Get("/", store.listTasks)
r.Post("/", store.createTask)
r.Route("/{id}", func(r chi.Router) {
r.Get("/", store.getTask)
r.Put("/", store.updateTask)
r.Delete("/", store.deleteTask)
})
})
r.Get("/health", func(w http.ResponseWriter, r *http.Request) {
writeJSON(w, http.StatusOK, map[string]string{"status": "ok"})
})
srv := &http.Server{
Addr: ":8080",
Handler: r,
ReadTimeout: 10 * time.Second,
WriteTimeout: 30 * time.Second,
IdleTimeout: 60 * time.Second,
}
// Graceful shutdown
go func() {
slog.Info("Server starting", "addr", srv.Addr)
if err := srv.ListenAndServe(); err != http.ErrServerClosed {
slog.Error("Server error", "error", err)
os.Exit(1)
}
}()
quit := make(chan os.Signal, 1)
signal.Notify(quit, os.Interrupt)
<-quit
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
slog.Info("Shutting down server...")
if err := srv.Shutdown(ctx); err != nil {
slog.Error("Shutdown error", "error", err)
}
}
func (s *TaskStore) listTasks(w http.ResponseWriter, r *http.Request) {
tasks := make([]*Task, 0, len(s.tasks))
for _, t := range s.tasks {
tasks = append(tasks, t)
}
writeJSON(w, http.StatusOK, tasks)
}
func (s *TaskStore) createTask(w http.ResponseWriter, r *http.Request) {
var req CreateTaskRequest
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
writeError(w, http.StatusBadRequest, "Invalid request body")
return
}
if req.Title == "" {
writeError(w, http.StatusBadRequest, "Title is required")
return
}
task := &Task{
ID: uuid.New().String(),
Title: req.Title,
Description: req.Description,
Completed: false,
CreatedAt: time.Now(),
}
s.tasks[task.ID] = task
writeJSON(w, http.StatusCreated, task)
}
func (s *TaskStore) getTask(w http.ResponseWriter, r *http.Request) {
id := chi.URLParam(r, "id")
task, ok := s.tasks[id]
if !ok {
writeError(w, http.StatusNotFound, "Task not found")
return
}
writeJSON(w, http.StatusOK, task)
}
func (s *TaskStore) updateTask(w http.ResponseWriter, r *http.Request) {
id := chi.URLParam(r, "id")
task, ok := s.tasks[id]
if !ok {
writeError(w, http.StatusNotFound, "Task not found")
return
}
var req struct {
Title *string `json:"title"`
Description *string `json:"description"`
Completed *bool `json:"completed"`
}
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
writeError(w, http.StatusBadRequest, "Invalid request body")
return
}
if req.Title != nil {
task.Title = *req.Title
}
if req.Description != nil {
task.Description = *req.Description
}
if req.Completed != nil {
task.Completed = *req.Completed
}
writeJSON(w, http.StatusOK, task)
}
func (s *TaskStore) deleteTask(w http.ResponseWriter, r *http.Request) {
id := chi.URLParam(r, "id")
if _, ok := s.tasks[id]; !ok {
writeError(w, http.StatusNotFound, "Task not found")
return
}
delete(s.tasks, id)
w.WriteHeader(http.StatusNoContent)
}
func writeJSON(w http.ResponseWriter, status int, data interface{}) {
w.Header().Set("Content-Type", "application/json")
w.WriteHeader(status)
json.NewEncoder(w).Encode(data)
}
func writeError(w http.ResponseWriter, status int, message string) {
writeJSON(w, status, map[string]string{"error": message})
}
func requestLogger(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
start := time.Now()
ww := middleware.NewWrapResponseWriter(w, r.ProtoMajor)
next.ServeHTTP(ww, r)
slog.Info("Request",
"method", r.Method,
"path", r.URL.Path,
"status", ww.Status(),
"duration_ms", time.Since(start).Milliseconds(),
"request_id", middleware.GetReqID(r.Context()),
)
})
}
並行処理のパターン
> 複数のAPIを並行に呼び出して結果を集約する함수를 생성해줘。
> コンテキスト에 의한취소とタイムアウトも구현して。
// concurrent.go
package service
import (
"context"
"fmt"
"sync"
"time"
)
type AggregatedData struct {
Users []User `json:"users"`
Products []Product `json:"products"`
Orders []Order `json:"orders"`
}
func FetchDashboardData(ctx context.Context) (*AggregatedData, error) {
ctx, cancel := context.WithTimeout(ctx, 5*time.Second)
defer cancel()
var (
result AggregatedData
mu sync.Mutex
wg sync.WaitGroup
errs []error
)
// 사용자취득
wg.Add(1)
go func() {
defer wg.Done()
users, err := fetchUsers(ctx)
mu.Lock()
defer mu.Unlock()
if err != nil {
errs = append(errs, fmt.Errorf("fetch users: %w", err))
return
}
result.Users = users
}()
// 상품취득
wg.Add(1)
go func() {
defer wg.Done()
products, err := fetchProducts(ctx)
mu.Lock()
defer mu.Unlock()
if err != nil {
errs = append(errs, fmt.Errorf("fetch products: %w", err))
return
}
result.Products = products
}()
// 注文취득
wg.Add(1)
go func() {
defer wg.Done()
orders, err := fetchOrders(ctx)
mu.Lock()
defer mu.Unlock()
if err != nil {
errs = append(errs, fmt.Errorf("fetch orders: %w", err))
return
}
result.Orders = orders
}()
wg.Wait()
if len(errs) > 0 {
return &result, fmt.Errorf("partial failure: %v", errs)
}
return &result, nil
}
테이블駆動테스트
// main_test.go
package main
import (
"bytes"
"encoding/json"
"net/http"
"net/http/httptest"
"testing"
"github.com/go-chi/chi/v5"
)
func TestCreateTask(t *testing.T) {
tests := []struct {
name string
body map[string]interface{}
wantStatus int
}{
{
name: "valid task",
body: map[string]interface{}{"title": "Test Task"},
wantStatus: http.StatusCreated,
},
{
name: "empty title",
body: map[string]interface{}{"title": ""},
wantStatus: http.StatusBadRequest,
},
{
name: "missing title",
body: map[string]interface{}{},
wantStatus: http.StatusBadRequest,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
store := NewTaskStore()
body, _ := json.Marshal(tt.body)
req := httptest.NewRequest(http.MethodPost, "/api/tasks", bytes.NewReader(body))
req.Header.Set("Content-Type", "application/json")
rec := httptest.NewRecorder()
store.createTask(rec, req)
if rec.Code != tt.wantStatus {
t.Errorf("got status %d, want %d", rec.Code, tt.wantStatus)
}
})
}
}
정리
Claude Code를 활용하면 GoのWeb API、並行処理、CLIツール、테스트を효율적으로개발할 수 있습니다。Goのイディオムである에러 핸들링パターンや테이블駆動테스트もClaude Code는 正確に생성します。개발の기본적인효율화에 대해서는生産性を3倍にするTips를 참고하세요.프로젝트のGo規約はCLAUDE.mdに記述しておくと一貫性が保てます。
Claude Code의 상세 정보는Anthropic공식 문서를 확인하세요.Goの公式가이드はGo공식 사이트를 참고하세요.
Claude Code 워크플로우를 한 단계 업그레이드하세요
지금 바로 Claude Code에 복사해 쓸 수 있는 검증된 프롬프트 템플릿 50선.
이 글을 작성한 사람
Masa
Claude Code를 적극 활용하는 엔지니어. 10개 언어, 2,000페이지 이상의 테크 미디어 claudecode-lab.com을 운영 중.
관련 글
Claude Code로 리팩토링을 자동화하는 방법
Claude Code를 활용해 코드 리팩토링을 효율적으로 자동화하는 방법을 알아봅니다. 실전 프롬프트와 구체적인 리팩토링 패턴을 소개합니다.
Claude Code로 사이드 프로젝트 개발 속도를 극대화하는 방법 [예제 포함]
Claude Code를 활용해 개인 프로젝트 개발 속도를 획기적으로 높이는 방법을 알아봅니다. 실전 예제와 아이디어부터 배포까지의 워크플로를 포함합니다.
Complete CORS Configuration Guide: Claude Code 활용 가이드
complete cors configuration guide: Claude Code 활용. 실용적인 팁과 코드 예시를 포함합니다.