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?>]>
context
はrender
に渡してたやつ
ということは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をチェックしてないのはなにかのポリシーによるもの?- サーバ動かしてなくていいというのは楽だなあ
データベースは各テストメソッドごとにリセットされる
そうなんだ