Python REST API | Flask/Django로 API 서버 만들기
이 글의 핵심
Python REST API에 대해 정리한 개발 블로그 글입니다. from flask import Flask, jsonify, request
들어가며
”백엔드 개발의 핵심”
REST API는 프론트엔드와 백엔드를 연결하는 표준 방식입니다.
1. REST API 기본
RESTful 설계 원칙
리소스 중심 설계:
GET /api/users - 사용자 목록
GET /api/users/1 - 사용자 1 조회
POST /api/users - 사용자 생성
PUT /api/users/1 - 사용자 1 수정
DELETE /api/users/1 - 사용자 1 삭제
2. Flask로 REST API
CRUD API 구현
아래 예제는 메모리에 있는 users 장바구니 리스트를 JSON으로 주고받는 미니 API입니다. GET은 목록·단건 조회, POST는 새 항목 추가, PUT/DELETE는 수정·삭제에 대응합니다. 실제 서비스에서는 이 자리를 DB와 바꿉니다.
from flask import Flask, jsonify, request
app = Flask(__name__)
users = [
{'id': 1, 'name': '철수', 'email': '[email protected]'},
{'id': 2, 'name': '영희', 'email': '[email protected]'}
]
# 목록 조회 (GET)
@app.route('/api/users', methods=['GET'])
def get_users():
return jsonify({'users': users, 'count': len(users)})
# 단일 조회 (GET)
@app.route('/api/users/<int:user_id>', methods=['GET'])
def get_user(user_id):
user = next((u for u in users if u['id'] == user_id), None)
if user:
return jsonify(user)
return jsonify({'error': 'User not found'}), 404
# 생성 (POST)
@app.route('/api/users', methods=['POST'])
def create_user():
data = request.get_json()
if not data or 'name' not in data or 'email' not in data:
return jsonify({'error': 'Invalid data'}), 400
new_user = {
'id': max(u['id'] for u in users) + 1 if users else 1,
'name': data['name'],
'email': data['email']
}
users.append(new_user)
return jsonify(new_user), 201
# 수정 (PUT)
@app.route('/api/users/<int:user_id>', methods=['PUT'])
def update_user(user_id):
user = next((u for u in users if u['id'] == user_id), None)
if not user:
return jsonify({'error': 'User not found'}), 404
data = request.get_json()
user.update(data)
return jsonify(user)
# 삭제 (DELETE)
@app.route('/api/users/<int:user_id>', methods=['DELETE'])
def delete_user(user_id):
global users
users = [u for u in users if u['id'] != user_id]
return '', 204
if __name__ == '__main__':
app.run(debug=True)
3. Django REST Framework
설치
pip install djangorestframework
시리얼라이저
# blog/serializers.py
from rest_framework import serializers
from .models import Post
class PostSerializer(serializers.ModelSerializer):
class Meta:
model = Post
fields = ['id', 'title', 'content', 'author', 'created_at']
read_only_fields = ['id', 'created_at']
ViewSet
# blog/views.py
from rest_framework import viewsets
from .models import Post
from .serializers import PostSerializer
class PostViewSet(viewsets.ModelViewSet):
queryset = Post.objects.all()
serializer_class = PostSerializer
URL 라우팅
# blog/urls.py
from rest_framework.routers import DefaultRouter
from .views import PostViewSet
router = DefaultRouter()
router.register(r'posts', PostViewSet)
urlpatterns = router.urls
4. 인증과 권한
JWT 인증
pip install djangorestframework-simplejwt
# settings.py
REST_FRAMEWORK = {
'DEFAULT_AUTHENTICATION_CLASSES': [
'rest_framework_simplejwt.authentication.JWTAuthentication',
],
}
# urls.py
from rest_framework_simplejwt.views import (
TokenObtainPairView,
TokenRefreshView,
)
urlpatterns = [
path('api/token/', TokenObtainPairView.as_view()),
path('api/token/refresh/', TokenRefreshView.as_view()),
]
5. 에러 처리
커스텀 에러 응답
from flask import jsonify
@app.errorhandler(404)
def not_found(error):
return jsonify({
'error': 'Not Found',
'message': '리소스를 찾을 수 없습니다'
}), 404
@app.errorhandler(500)
def internal_error(error):
return jsonify({
'error': 'Internal Server Error',
'message': '서버 오류가 발생했습니다'
}), 500
6. 실전 예제
완전한 Flask API
from flask import Flask, jsonify, request
from functools import wraps
app = Flask(__name__)
# 간단한 인증
API_KEY = "secret-key-123"
def require_api_key(f):
@wraps(f)
def decorated(*args, **kwargs):
api_key = request.headers.get('X-API-Key')
if api_key != API_KEY:
return jsonify({'error': 'Unauthorized'}), 401
return f(*args, **kwargs)
return decorated
# 데이터베이스 대신 메모리 사용
posts = []
@app.route('/api/posts', methods=['GET'])
def get_posts():
page = request.args.get('page', 1, type=int)
per_page = request.args.get('per_page', 10, type=int)
start = (page - 1) * per_page
end = start + per_page
return jsonify({
'posts': posts[start:end],
'page': page,
'total': len(posts)
})
@app.route('/api/posts', methods=['POST'])
@require_api_key
def create_post():
data = request.get_json()
required_fields = ['title', 'content']
if not all(field in data for field in required_fields):
return jsonify({'error': 'Missing required fields'}), 400
post = {
'id': len(posts) + 1,
'title': data['title'],
'content': data['content'],
'author': data.get('author', 'Anonymous')
}
posts.append(post)
return jsonify(post), 201
if __name__ == '__main__':
app.run(debug=True)
URL 설계와 HTTP 상태 코드 (REST 관례)
REST API는 주소(URL)만 봐도 어떤 자원인지 드러나게 짓는 것이 일반적입니다. 동사를 경로에 넣기보다 명사·복수형·계층으로 자원을 표현하고, 성공·실패는 HTTP 상태 코드로 구분하면 클라이언트가 응답을 처리하기 쉽습니다.
# ✅ 명사 사용, 동사 X
GET /api/users # O
GET /api/getUsers # X
# ✅ 복수형 사용
GET /api/users # O
GET /api/user # X
# ✅ 계층 구조
GET /api/users/1/posts # 사용자 1의 포스트
# ✅ HTTP 상태 코드
200 OK # 성공
201 Created # 생성 성공
400 Bad Request # 잘못된 요청
401 Unauthorized # 인증 필요
404 Not Found # 리소스 없음
500 Server Error # 서버 오류
정리
핵심 요약
- REST: HTTP + 자원 중심 설계
- Flask: 가볍고 유연한 API 서버
- Django REST Framework: 강력한 기능
- 인증: JWT, API Key
- 에러 처리: 적절한 HTTP 상태 코드
다음 단계
- 데이터베이스 연동
- 배포
관련 글
- Flask 기초 | Python 웹 프레임워크 시작하기