인프런 영문 브랜드 로고
인프런 영문 브랜드 로고

인프런 커뮤니티 질문&답변

pplkjh2님의 프로필 이미지
pplkjh2

작성한 질문수

파이썬/장고 웹서비스 개발 완벽 가이드 with 리액트 (장고 4.2 기준)

13-14 API 테스트 코드 작성 (Feat. pytest) #1 - pytest 설정 및 Factory, Fixture 구현

13-14 pytest 코드 실행 오류

작성

·

54

·

수정됨

0

  • 질문을 온전히 이해할 수 있도록, 모든 맥락을 전달해주세요.

  • 질문은 질문자가 번거로워야 보다 좋은 답변을 얻으실 수 있습니다.

  • 시행착오를 알려주시면 곧바로 원하는 문제에 집중할 수 있습니다.

  • 오류 메시지는 일부만 알려주시기보다 전체 오류 메시지를 캡처해서 주시면, 오류 파악에 도움이 됩니다.

     

당신의 파이썬/장고 페이스메이커가 되겠습니다. ;-)


인프런 서비스 운영 관련 문의는 1:1 문의하기를 이용해주세요.

 

postgresql 을 다른 서버에 연결하여 사용중인데요.

runserver 시에는 정상 작동하나 강사님이 올려주신 pytest를 똑같이 설정하고 실행하면, schema가 없어서 table을 생성할 수 없다는 오류가 발생합니다.

MigrationSchemaMissing("Unable to create the django_migrations table 

일단은 settings.py를 sqlite3로 변경하여 정상 작동은 시켰는데요 어떻게 수정할 수 있을까요?

 

 


또한 올려주신 소스코드에서

import factory
from accounts.models import User


class UserFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = User
        django_get_or_create = ("username",)

    username = factory.Sequence(lambda n: f"user{n}")
    email = factory.LazyAttribute(lambda user: f"{user.username}@example.com")

    # UserFactory 인스턴스 생성 시에, 암호 입력을 지원하기 위함
    # raw_password 필드가 아니라 password 필드로 지정하면, 암호화없이 입력값 그대로 저장됩니다.
    raw_password = factory.Faker("password")

    @classmethod
    def _create(cls, model_class, *args, **kwargs):
        # raw_password 필드값을 암호화하여 password 필드에 저장합니다.
        # User 모델에는 raw_password 이름의 필드가 없으므로 kwargs 사전에서 제거해줘야만 합니다.
        raw_password = kwargs.pop("raw_password", None)
        if raw_password:
            kwargs["password"] = make_password(raw_password)

        return super()._create(model_class, *args, **kwargs)

make_password 부분이 import 되어 있지 않습니다. git에서는 수정되어 있는 모양인데 참고 부탁드립니다.

답변 1

0

이진석님의 프로필 이미지
이진석
지식공유자

안녕하세요.

장고 테스트에서는 settings.DATABASES["default"]의 "NAME"에 지정한 데이터베이스를 사용하지 않고, 이 이름에 접두어 "test_"를 붙인 데이터베이스를 디폴트 테스트 데이터베이스로 사용합니다.

이는 테스트 시에 서비스 데이터를 변경하지 않기 위함이고, 테스트를 수행할 때마다 매번 데이터베이스를 새롭게 생성해서, 원본 데이터베이스와 독립적인 테스트 환경을 유지합니다

지정하신 PostgreSQL 서버에서 test 데이터베이스 생성권한이 없거나, test 데이터베이스에서 데이터베이스 테이블 생성 권한이 없어서 발생하는 이슈 같습니다. 해당 PostgreSQL 서버에 test_ 접두어를 붙인 데이터베이스를 생성하신 후에 테스트를 수행해보시면 어떤가요?

혹은 settings.DATABASES["default"]에 "NAME" 외에 "TEST" 항목으로 { "NAME": "원하시는_TEST_데이터베이스이름" } 지정하시어, 원하는 테스트 데이터베이스 이름을 지정하실 수도 있습니다.

관련 문서 : https://docs.djangoproject.com/en/5.1/topics/testing/overview/#the-test-database

살펴보시고, 댓글 부탁드립니다. :-)

pplkjh2님의 프로필 이미지
pplkjh2
질문자

테스트를 여러가지 해 보았는데요. django로 postgresql의 schema를 자동 생성하는 방법을 모르겠습니다.

postgresql 버전은 16.3이구요. admin 계정으로 접속했으니 권한은 전부 있습니다.

settings.py의 options은 아래와 같이 주었는데요

'OPTIONS': {
            'options': '-c search_path=django,public'
        },
  1. 해당 옵션은 "TEST" 값 내부에서는 인식되지 않았고요

  2. 그 값마저도 public 일 경우만 정상 인식 되었습니다.

결국 schema는 public으로 하고 DB 이름을 변경하여 테스트 수행하였습니다.

같은 DB 내에서 schema만 변경하여 테스트 하는 경우는 없는건가요?

그리고 schema 이름이 public인 경우만 장고에서 인식되는게 정상적인건가요?

이진석님의 프로필 이미지
이진석
지식공유자

방금 확인해보니, 장고 마이그레이션 기능에서는 테이블을 생성할 뿐 postgresql schema는 사용할 뿐 자동 생성되지는 않습니다.

말씀하신 대로 아래처럼 search_path를 통해 사용할 schema를 지정할 수 있구요.

image.png

위에서 지정한 new_schema가 데이터베이스에 없다면, 테스트 수행 시에도 마이그레이션 과정에서 테이블 생성을 시도하면서 아래 오류를 만나시게 됩니다.

psycopg2.errors.InvalidSchemaName: no schema has been selected to create in
LINE 1: CREATE TABLE "django_migrations" ("id" bigint NOT NULL PRIMA...

settings.py 에서 아래와 같이 환경변수를 통해 SCHEMA 설정을 주입받으셔서, 그 값이 있을 때 search_path 설정에 반영토록 해보시면 어떨까요?

image.png

pytest 에서는 pytest-env 라이브러리를 설치하시고 나서 pytest.ini 에서 env= 설정을 통해 환경변수를 지정하실 수 있습니다. // 단, pytest.ini 환경변수를 지정하시고 나서, .env 에도 지정하시면 .env 로딩 코드가 앞서 로딩한 환경변수를 덮어쓸 수도 있으니 주의해주세요.

그럼 테스트 시에만 사용하실 schema를 따로 지정하실 수 있는 거죠.

살펴보시고 댓글 부탁드립니다.

화이팅입니다. :-)

이진석님의 프로필 이미지
이진석
지식공유자

내용을 정리해서 간략히 포스팅도 해봤습니다.

https://pyhub.kr/recipe/31e9RWj4QPOV2/

pplkjh2님의 프로필 이미지
pplkjh2

작성한 질문수

질문하기