문돌이 존버/Django 스터디
장고(Django), Form Validation
애뚱
2022. 2. 14. 14:41
반응형
본 글은 Holix의 "리액트와 함께 장고 시작하기 Complete" 강의를 듣고 작성한 일지입니다.
이번 시간엔 Form 유효성 검사에 대해 자세히 알아보겠습니다.
먼저 Form 유효성 검사는 .is_valid() 함수에서 수행되며, 유효성 검사 호출 로직은 아래와 같습니다.
1. form.full_clean() 호출
- 각 필드 객체별: 각 필드객체.clean() 호출을 통해 각 필드 타입에 맞춰 유효성 검사
- Form 객체 내: 필드 이름 별로 Form 객체.clean_필드명() 함수가 있다면 호출해서 유효성 검사
Form객체.clean() 함수가 있다면 호출해서 유효성 검사(필드 다수를 묶어서 검사할 필요가 있을 경우)
2. 에러 유무에 따른 True/False 리턴
Form에서는 2가지 유효성 검사를 수행합니다.
1. Validator 함수를 통한 유효성 검사
- 값이 원하는 조건에 맞지 않을 때 ValidationError 예외 발생
- 리턴값은 사용되지 않음
2. Form 클래스 내 clean, clean_ 멤버함수를 통한 유효성 검사 및 값 변경
- 값이 원하는 조건에 맞지 않을 때 ValidationError 예외 발생
- 리턴값을 통해 값 반환
# forms.py
from django import forms
from .models import Post
import re
class PostForm(forms.ModelForm):
class Meta:
model = Post
fields = [
'content', 'photo', 'tag_set', 'is_public'
]
# content 필드 예시
def clean_content(self):
content = self.cleaned_data.get('content')
if content:
content = re.sub(r'[a-zA-Z]+', '', content) # 영어는 모두 제거
return content
영어로 ahew... 를 타이핑했지만 아래 결과에선 영어가 삭제된 것을 확인할 수 있습니다.
Validators는 함수형과 클래스형으로 나뉘어서 어떤 형태로든 모두 커스텀으로 만드는 것이 가능합니다. 다만 빌트인 Validators(RegexValidator, EmailValidator, URLValidator, MaxValueValidator 등)를 최대한 활용하는 것을 추천합니다.
모델 필드에 디폴트로 적용된 validators는 아래와 같습니다.
models.EmailField
- validators.validate_email 적용
models.URLField
- validators.URLValidator 적용
models.GenericIPAddressField
- validators.ip_address_validators 적용
models.SlugField
- validators.validate_slug 적용
Form clean 멤버함수의 역할은 2가지로 정리할 수 있습니다.
1. "필드별 Error 기록" 또는 "Non 필드 Error 기록"
- 값이 조건에 안 맞으면 ValidationError 예외를 통해 오류 기록
- add_error(필드명, 오류내용) 직접 호출을 통해 오류 기록
2. 원하는 포맷으로 값 변경
- 리턴값을 통해 값 변경
필드별 Error / Non 필드 Error 기록에 대해 조금 더 살펴보겠습니다.
clean_필드명() 멤버함수
- 특정 필드별 검사/변경의 책임
- ValidationError 예외 발생 시, 해당 필드 Error로 분류
clean() 멤버함수
- 다수 필드에 대한 검사/변경의 책임
- ValidationError 예외 발생 시, non_field_errors로 분류
- add_error() 함수를 통해 필드별 Error 기록도 가능
그렇다면 언제 validators를 사용하고, 언제 clean을 사용할까요?
validators
- 지속적인 유효성 검사 루틴이 필요할 때
- 가급적 모든 validators는 모델에 정의하고(Fat Model), ModelForm을 통해 모델의 validators 정보도 같이 가져오기
clean
- 특정 Form에서 1회성 유효성 검사 루틴이 필요할 때
- 다수 필드값에 걸쳐서 유효성 검사가 필요할 때
- 필드값을 변경할 필요가 있을 때
마지막으로 게임에서 서버와 닉네임을 선택할 때 유니크함을 체크하는 예제 코드를 소개드리겠습니다.
# models.py
from django.core.validators import MinLengthValidator
class GameUser(models.Model):
server = models.CharField(max_length=10)
username = models.CharField(max_length=20, validators=[MinLengthValidator(3)]
# DB에서 server-username 합쳐서 유니크함 확인
class Meta:
# 장고에서 제공하는 기본 탬플릿
unique_together = [
('server', 'username')
]
# forms.py
class GameUserSignupForm(forms.ModelForm):
class Meta:
model = GameUser
fields = ['server', 'username']
728x90
반응형