기본 콘텐츠로 건너뛰기

django-filter로 필터링 API 우아하게 만들기

Django-filter로 API 필터링, 쉽게 해결하기! (내 경험담과 함께)

요즘 Django 기반 웹 애플리케이션 개발하면서 느낀 건데, 데이터 관리가 정말 중요하더라고요. 특히 사용자에게 딱 필요한 정보만 깔끔하게 보여주는 게 얼마나 중요한지… 그래서 제가 애용하는 필터링 도구가 바로 Django-filter입니다! Django REST framework랑 같이 쓰면 정말 효율적인데, 이걸 어떻게 활용하는지 제 경험을 바탕으로 쉽게 설명해 드릴게요. 복잡한 쿼리 씨름 안 하고도 간단하게 필터링 로직을 만들 수 있답니다.

먼저 Django-filter의 핵심 개념부터 살펴볼까요? 쉽게 말해, 모델의 필드를 기준으로 필터링 조건을 정의하는 사용자 정의 필터 클래스를 만드는 거예요. FilterSet 클래스를 상속받아서 모델의 필드랑 필터링 방식을 정의하면 되는데, 여기서 lookup expressions라고 하는 다양한 필터링 옵션을 쓸 수 있어요. 예를 들어, exact (정확히 일치), icontains (대소문자 구분 없이 포함), startswith (시작 문자열 일치), range (범위 지정) 등이 있죠. 이걸 Django REST framework와 연결하면 API 요청 파라미터로 필터링 조건을 받아서 처리할 수 있답니다.

이해가 잘 안 가시나요? 그럼 제가 실제로 썼던 코드 예제를 보여드릴게요. models.py, filters.py, serializers.py, views.py 이렇게 네 개의 파일을 만들었어요. (이름만 봐도 어떤 역할을 하는지 감이 오시죠?)

models.py (상품 모델 정의):

from django.db import models

class Product(models.Model):
    name = models.CharField(max_length=255)
    category = models.CharField(max_length=100)
    price = models.DecimalField(max_digits=10, decimal_places=2)
    stock = models.IntegerField()

    def __str__(self):
        return self.name

이건 상품 정보를 저장하는 모델이에요. 상품 이름, 카테고리, 가격, 재고량을 저장하죠. 익숙하시죠?

filters.py (필터 정의):

import django_filters
from .models import Product

class ProductFilter(django_filters.FilterSet):
    name = django_filters.CharFilter(lookup_expr='icontains')
    price = django_filters.RangeFilter()
    category = django_filters.CharFilter(lookup_expr='exact')

    class Meta:
        model = Product
        fields = ['name', 'price', 'category', 'stock']

여기서부터 Django-filter가 빛을 발하는데요. ProductFilter 클래스는 상품 이름(icontains로 부분 일치 검색), 가격(RangeFilter로 범위 지정), 카테고리(exact로 정확히 일치)를 필터링하도록 정의했어요. 실제로 제가 상품 검색 기능을 만들 때 정말 유용하게 썼던 부분이에요.

serializers.py (데이터 직렬화):

from rest_framework import serializers
from .models import Product

class ProductSerializer(serializers.ModelSerializer):
    class Meta:
        model = Product
        fields = '__all__'

이건 API 응답으로 보낼 데이터를 정리하는 부분이에요. 모든 필드를 그대로 보내도록 설정했어요.

views.py (API 뷰 정의):

from rest_framework import generics
from .models import Product
from .serializers import ProductSerializer
from .filters import ProductFilter
from django_filters.rest_framework import DjangoFilterBackend

class ProductList(generics.ListAPIView):
    queryset = Product.objects.all()
    serializer_class = ProductSerializer
    filter_backends = [DjangoFilterBackend]
    filterset_class = ProductFilter

마지막으로, API 뷰를 정의하는데요. DjangoFilterBackend를 사용해서 ProductFilter를 적용하고, /products/?name=example&price__range=10-20&category=electronics 같은 URL로 필터링 요청을 보낼 수 있도록 설정했어요. 예를 들어, name=example은 상품 이름에 "example"이 포함된 상품을, price__range=10-20은 가격이 10~20 사이인 상품을 찾는다는 뜻이죠.

자, 이제 실제로 돌려보면 어떻게 작동하는지 바로 알 수 있을 거예요. 하지만 몇 가지 주의할 점이 있어요.

  • 필터링에 쓰는 필드의 데이터 타입이랑 필터링 방식을 정확하게 맞춰야 해요. 안 그러면 예상치 못한 결과가 나올 수 있으니까요. 제가 처음에 이 부분 때문에 꽤 애먹었거든요.
  • 성능이 중요하다면 prefetch_relatedselect_related를 사용해서 데이터베이스 쿼리 효율을 높일 수 있어요. 데이터 양이 많다면 꼭 고려해 보세요!
  • 필터링 로직이 너무 복잡해지면 필터 클래스를 여러 개로 나누는 걸 고려해 보세요. 코드 관리가 훨씬 수월해져요.
  • 마지막으로, 보안을 위해 사용자 입력에 대한 유효성 검사는 필수입니다! 이 부분 꼭 잊지 마세요.

결론적으로, Django-filter는 Django 애플리케이션에서 API 필터링을 쉽고 효율적으로 구현하는 데 정말 유용한 도구입니다. 제가 설명한 방법대로 따라 해보시면 사용자에게 최적화된 API를 만들 수 있을 거예요. 복잡한 필터링 로직을 만들 때는 미리 설계를 잘 하고, 테스트를 충분히 해서 문제를 미리 방지하는 게 중요하다는 점, 꼭 기억해주세요!

댓글

이 블로그의 인기 게시물

Django에서 트랜잭션 관리하기

Django에서 트랜잭션 관리하기 안녕하세요! 오늘은 Django에서 데이터베이스 트랜잭션을 효과적으로 관리하는 방법에 대해 알아보겠습니다. 1. 트랜잭션의 중요성 트랜잭션은 데이터베이스의 일관성과 무결성을 보장하는 중요한 개념입니다. Django에서는 여러 가지 방법으로 트랜잭션을 관리할 수 있습니다. 1.1 기본 개념 원자성(Atomicity) : 트랜잭션은 모두 실행되거나 모두 실행되지 않아야 합니다. 일관성(Consistency) : 트랜잭션 전후로 데이터베이스의 일관성이 유지되어야 합니다. 격리성(Isolation) : 동시에 실행되는 트랜잭션들이 서로 영향을 주지 않아야 합니다. 지속성(Durability) : 완료된 트랜잭션의 결과는 영구적으로 저장되어야 합니다. 2. Django의 트랜잭션 관리 2.1 기본 설정 # settings.py DATABASES = { 'default': { 'ENGINE': 'django.db.backends.postgresql', 'NAME': 'mydatabase', 'USER': 'myuser', 'PASSWORD': 'mypassword', 'HOST': 'localhost', 'PORT': '5432', 'ATOMIC_REQUESTS': True, # 모든 뷰를 트랜잭션으로 래핑 } } 2.2 데코레이터 사용 from django.db import transaction @transaction.atomic def create_order(user, items): order = Order.objects.create(user=...

AWS S3 + CloudFront로 정적 파일 서빙 완전 가이드

AWS S3 + CloudFront로 정적 파일 서빙 완전 가이드 안녕하세요! 오늘은 AWS S3와 CloudFront를 사용하여 정적 파일을 효율적으로 서빙하는 방법에 대해 알아보겠습니다. 왜 S3와 CloudFront를 사용할까요? 높은 가용성 : AWS의 글로벌 인프라를 활용 빠른 전송 속도 : CloudFront의 CDN 기능으로 전 세계 사용자에게 빠른 전송 비용 효율성 : 사용한 만큼만 지불 보안 : AWS의 보안 기능 활용 확장성 : 트래픽 증가에 자동 대응 1. S3 버킷 설정 1.1 버킷 생성 및 설정 import boto3 def create_s3_bucket(): s3 = boto3.client('s3') # 버킷 생성 bucket_name = 'your-static-files-bucket' s3.create_bucket( Bucket=bucket_name, CreateBucketConfiguration={ 'LocationConstraint': 'ap-northeast-2' } ) # 버킷 정책 설정 bucket_policy = { "Version": "2012-10-17", "Statement": [ { "Sid": "PublicReadGetObject", "Effect": "Allow", "Principal": "*", "Action": "s3:GetObje...

Python에서 asyncio 완전 정복 (await, async, gather 등)

어휴, 요즘 파이썬으로 비동기 프로그래밍 하는 재미에 푹 빠졌어요! 특히 asyncio 는 정말 마법 같더라고요. 처음엔 좀 낯설었는데, 익숙해지니까 속도 향상이 눈에 띄게 느껴져서 완전 반해버렸습니다. 이 글에선 제가 asyncio 를 배우면서 깨달은 점들을 풀어놓을게요. 혹시 비동기 프로그래밍이 뭔지 잘 모르시겠다면, 간단히 말해 여러 작업을 동시에 처리해서 프로그램 속도를 엄청나게 높이는 기술이라고 생각하시면 돼요. 마치 여러 요리사가 동시에 음식을 만들어서 손님에게 빨리 제공하는 것과 비슷하죠! 일단 async 와 await 라는 녀석들이 핵심인데요, async 는 함수 앞에 붙여서 "얘는 비동기 함수야!"라고 선언하는 거예요. 그리고 await 는 다른 비동기 함수가 끝날 때까지 기다리라고 지시하는 역할을 하죠. 예를 들어, 네트워크에서 데이터를 가져오는 함수가 있다면, await 를 사용해서 데이터가 다 가져올 때까지 기다렸다가 다음 작업을 진행할 수 있어요. 그 동안 다른 작업을 처리할 수 있으니, 마치 멀티태스킹을 하는 것처럼 느껴져요. 신기하지 않나요? 그리고 asyncio.gather 는 여러 비동기 함수를 동시에 실행하고 결과를 모아주는 아주 유용한 친구입니다. 제가 웹사이트 여러 개에서 데이터를 동시에 가져와야 할 때 정말 요긴하게 썼어요. 하나씩 순서대로 가져오는 것보다 훨씬 빠르더라고요! 마치 여러 개의 탭을 동시에 열어놓고 작업하는 것과 같다고 생각하시면 될 것 같아요. 실제로 제가 썼던 코드를 보여드릴게요. 세 개의 웹사이트에서 데이터를 가져오는 예제인데요. (아래 코드 삽입) 이 코드를 보시면, fetch_data 함수가 각 웹사이트에서 데이터를 가져오는 역할을 하고, asyncio.gather 가 이 함수들을 동시에 실행하도록 도와주는 것을 볼 수 있을 거예요. asyncio.sleep(2) 는 네트워크 지연을 시뮬레이션하기 위해 넣...