kb84tkhrのブログ

何を書こうか考え中です あ、あと組織とは関係ないってやつです 個人的なやつ

Djangoチュートリアル(11)

テストふたつ追加
境界値のテストを追加したってことだな

    def test_was_published_recently_with_old_question(self):
        time = timezone.now() - datetime.timedelta(days=1, seconds=1)
        old_question = Question(pub_date=time)
        self.assertIs(old_question.was_published_recently(), False)

    def test_was_published_recently_with_recent_question(self):
        time = timezone.now() - datetime.timedelta(hours=23, minutes=59, seconds=59)
        recent_question = Question(pub_date=time)
        self.assertIs(recent_question.was_published_recently(), True)

テスト界隈だとこういうやけに自然言語っぽい関数名使ってますね
そういうのがいいプラクティスってことなんでしょうけど日本人にはちょっとつらい
日本語で名前つけてるよっていうのも見たことあるかなあ

普通のコードだとコードをそのまま自然言語にしただけみたいなのは
よくないってされてて、こういうコメントってほとんどそれじゃね?って
気もするけどまあこれはかんたんなテストだからそう思うだけかもしれないね

    """
    was_published_recently() returns False for questions whose pub_date
    is older than 1 day.
    """

あるいはテスト=仕様っていう思想で仕様を記述しましたってことなのかな
いや単に関数にはdocstringをつける、ってだけなのが一番ありそうな話か

モデルのテストはここまで
次はビューのテスト
なんだけどまずshellで練習

$ python manage.py shell
>>> from django.test.utils import setup_test_environment
>>> setup_test_environment()
>>> from django.test import Client
>>> client = Client()

ここでpython2.7が動いててしまったvenvになってないいつからだ?となりました
venvは私のような人間には難しい
vscodeが勝手にやってくれるときもあるんだけどどういうときなんだ
それはそれとして

ここまではテスト用の環境の準備
いわゆるおまじない
っていうほど意味がわからないものじゃないけど

>>> response = client.get('/')
Not Found: /
>>> response.status_code
404

そういえばrunserverしてなくていいんだっけ?と思ったけどいいんだな
HTTP通信をしてるわけじゃなくて直接viewsの関数を呼んでるんだろうから

/はないので404

>>> from django.urls import reverse
>>> response = client.get(reverse('polls:index'))

reverse('polls:index')じゃなくて直接'/polls/'と書いても成功するけど、
自動テストのことを考えるとちゃんとしておいたほうがいい

>>> reverse('polls:index')
u'/polls/'

いろいろ取れる

>>> response.status_code
200
>>> response.content
'\n    <ul>\n    \n        <li><a href="/polls/3/">What is your favourite color?</a></li>\n    \n    </ul>\n\n'
>>> response.context['latest_question_list']
<QuerySet [<Question: What is your favourite color?>]>

contextrenderに渡してたやつ
ということはresponseってのはrenderの戻り値かな?

ではviews.pyを修正
pub_dateが未来なら表示しないようにする

class IndexView(generic.ListView):
    def get_queryset(self):
        return Question.objects.filter(
            pub_date__lte=timezone.now()
        ).order_by('-pub_date')[:5]

lteっていうのはless than or equalってことかな

ここでデータベースにあたらしいQuestionを登録するのかな、と思ったら
テストの中でレコードを作ってた
なるほどそうきたか
一度も全体を動かしてないってのもちょっと足元が不安な感じだけど
ユニットテストってそういうものなんだろう

class QuestionIndexViewTests(TestCase):

    def test_no_questions(self):
        response = self.client.get(reverse('polls:index'))
        self.assertEqual(response.status_code, 200)
        self.asserContains(response, "No polls are available.")
        self.assertQuerysetEqual(response.context['latest_question_list'], [])

    def test_past_question(self):
        create_question(question_text="Past question.", days=-30)
        response = self.client.get(reverse('polls:index'))
        self.assertQuerysetEqual(
            response.context['latest_question_list'],
            ['<Question: Past question.>']
        )

    ...
  • test_past_question以降でstatusをチェックしてないのはなにかのポリシーによるもの?
  • サーバ動かしてなくていいというのは楽だなあ

データベースは各テストメソッドごとにリセットされる

そうなんだ