博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
第五章 保存用户输入
阅读量:5360 次
发布时间:2019-06-15

本文共 8818 字,大约阅读时间需要 29 分钟。

5.1编写表单,发送POST请求

调整一下lists/templates/home.html中的模版

            To-Do lists                

Your To-Do list

此时运行,在Django的调试页面,会显示有CSRF错误,所以,使用‘模版标签’’,添加CSRF令牌

            To-Do lists                

Your To-Do list

{
% csrf_token %}

运行功能测试,出现预期的失败

5.2在服务器中处理POST请求

修改视图函数,让它能够处理POST请求

#tests.py# -*- coding: utf-8 -*-from django.test import TestCasefrom django.http import HttpRequestfrom django.core.urlresolvers import resolvefrom lists.views import home_pagefrom django.template.loader import render_to_stringclass HomePageTest(TestCase):    def test_root_url_resolves_to_home_page_view(self):        found = resolve('/')        self.assertEqual(found.func,home_page)    def test_home_page_return_correct_html(self):        request = HttpRequest()        response = home_page(request)        except_html = render_to_string('home_html')        self.assertEqual(response.content.decode(),except_html)        # .deocde()把字节转换为unicode字符串        def test_home_page_can_return_a_POST_request(self):        request = HttpRequest()        request.method = 'POST'        request.POST['item_text'] = 'A new list item'                response = home_page(request)        self.assertIn('A new list item',response.content.decode())

python3 manage.py test 出现预期失败

为了让测试通过,添加一个if语句,为POST请求提供不同的代码执行路径

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsedef home_page(request):    if request.method == 'POST':        return HttpResponse(request.POST['item_text'])    return render(request,'home.html')

接下来,把POST请求的数据添加到首页模版的表格里

5.3Python变量传入模版渲染

在模版中引入python对象,使用{{}}

            To-Do lists                

Your To-Do list

{
% csrf_token %}
{
{ new_item_text }}

测试new_item_text的值是否正确,在tests.py添加

self.assertIn('A new list item',response.content.decode()) expect_html = render_to_string('home.html',{
'new_item_text':'A new list item'})self.assertEqual(response.content.decode(),expect_html)

重写视图函数,把POST请求中的参数传入模版

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsedef home_page(request):    return render(request,'home.html',{
'new_item_text':request.POST.get('item_text',''),})

修改functioncal.py中的any用法,如果把1:去掉的话,测试应该能通过

#self.assertTrue(any(row.text == '1: Buy peacock feathers' for row in rows))#替代为 self.assertIn('1: Buy peacock feathers',[row.text for row in rows])

接下来,扩充功能测试,在functioncal.py中检查表格中添加的第二个待办事项。

事不过三,三则重构,只有test_开头的方法才会作为测试运行,我们添加辅助函数的方法,置于tearDown 和第一个测试之间,用于添加待办事项

# functional_tests.py# -*- coding: utf-8 -*-from selenium import webdriverfrom selenium.webdriver.common.keys import Keysimport unittestclass NewVisitorTest(unittest.TestCase):    #setup 和tearDowm是特殊的方法,分别在测试的前后运行,这两个方法与try/except相似    def setUp(self):        self.browser = webdriver.Chrome()        self.browser.implicitly_wait(3)                            #隐式等待 3秒    def tearDown(self):        self.browser.quit()    def check_for_row_in_item(self,row_text):        table = self.browser.find_element_by_id('id_list_table')        rows = table.find_elements_by_tag_name('tr')        self.assertIn(row_text,[row.text for row in rows])            def test_can_start_a_list_and_retrieve_it_later(self):         #名字以test开头的函数都是测试方法        self.browser.get('http://localhost:8000')        #网页头部和标题是否含有To-Do这个词        self.assertIn('To-Do',self.browser.title)        header_text = self.browser.find_element_by_id('id_new_item').text        #self.assertIn('To-Do', header_text)    #不知道为什么,错误        #待办事项        inputbox = self.browser.find_element_by_id('id_new_item')        self.assertEqual(inputbox.get_attribute('placeholder'),'Enter a to-do item')   #获取属性placeholder的值        #发送第一个        inputbox.send_keys('Buy peacock feathers')        inputbox.send_keys(Keys.ENTER)        self.check_for_row_in_item('1: Buy peacock feathers')        #第二个        inputbox = self.browser.find_element_by_id('id_new_item')        inputbox.send_keys('Use peacock feathers to make a fly')        inputbox.send_keys(Keys.ENTER)        self.check_for_row_in_item('1: Buy peacock feathers')        self.check_for_row_in_item('2: Use peacock feathers to make a fly')        self.fail('Finish the test!')if __name__ == '__main__':    unittest.main(warnings='ignore')                                #warnings='ignore'为禁止抛出resourceWarning异常

5.3Django ORM和第一个模型

“对象关系映射器”(ORM) 是一个数据抽象层,描述储存在数据库中的表,列,行。接下来在单元测试按照指定的方式中使用ORM

在tests.py中新建一个类

from lists.models import Itemclass HomePageTest(TestCase):。。。class ItemModelTest(TestCase):    def test_saving_and_retrieving_items(self):        first_item = Item()        first_item.text = 'The first list item'        first_item.save()        second_item = Item()        second_item.text = 'The second list item'        second_item.save()        saved_items = Item.objects.all()        self.assertEqual(saved_items.count(),2)        first_save_item = saved_items[0]        second_save_item = saved_items[1]        self.assertEqual(first_save_item.text,'The first list item')        self.assertEqual(second_save_item.text, 'The second list item')

在models.py中写入一些代码,添加Item 这个类

 

#models.pyfrom django.db import modelsclass Item(models.Model):    text = models.TextField(default='')   #设置字段类型,并为字段设置默认值,不然会失败

此时运行,会看到一个数据库错误。于是要进行数据库迁徙

python3 manage.py makemigrations 进行数据库迁徙,因为有2个新字段,要迁徙2,

python3 manage.py makemigrations

python3 manage.py test lists运行测试,成功

5.4把POST请求中的数据存入数据库

希望视图把待办事项存入数据库,而不是直接传给响应,于是在测试方法中的test_home_page_can_return_a_POST_request中,添加新代码

def test_home_page_can_return_a_POST_request(self):        request = HttpRequest()        request.method = 'POST'        request.POST['item_text'] = 'A new list item'        response = home_page(request)                self.assertEqual(Item.count(),1)        new_item = Item.objects.first()        self.assertEqual(new_item.text,'A new list item')                self.assertIn('A new list item',response.content.decode())        expect_html = render_to_string('home.html',{
'new_item_text':'A new list item'}) self.assertEqual(response.content.decode(),expect_html)

运行报错,0!= 1,

修改一下视图

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsefrom lists.models import Itemdef home_page(request):    item = Item()    item.text = request.POST.get('item_text','')    item.save()        return render(request,'home.html',item.text})

运行通过

然后,在定义一个新测试方法

class HomePageTest(TestCase):。。。def test_home_page_only_save_items_when_necessary(self):        request = HttpRequest()        home_page(request)        self.assertEqual(Item.objects.count(),0)

再更改视图文件

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsefrom lists.models import Itemdef home_page(request):    if request.method == 'POST':        new_item_text = request.POST['item_text']        Item.objects.create(text=new_item_text) #使用.obejcts.create是创建新Item对象的简化方式,无需再用.save()方法    else:        new_item_text = ''    return render(request,'home.html',{
'new_item_text':new_item_text})

运行。通过

5.5处理完POST请求后重定向到首页

修改views.py

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsefrom lists.models import Itemdef home_page(request):    if request.method == 'POST':        Item.objects.creat(text=request.POST['item_text'])        redirect('/')            return render(request,'home.html')

5.6在模版中渲染待办事项

编写一个单元测试,检测模版是否显示多个待办事项

#test.py class HomePageTest(TestCase):。。。    def test_home_page_displays_all_list_items(self):        Item.objects.create(text='itemy 1')        Item.objects.create(text='itemy 2')        request = HttpRequest()        response = home_page(request)        self.assertIn('itemy 1', response.content.decode())        self.assertIn('itemy 2', response.content.decode())

运行。失败

在home.html中编写遍历标签

{
% for item in items %}
{
% endfor %}
{
{ forloop.counter }}: {
{ item.text }}

同时,把视图待办事项传入模版

#viewsfrom django.shortcuts import render,redirectfrom django.http import HttpResponsefrom lists.models import Itemdef home_page(request):    if request.method == 'POST':        Item.objects.create(text=request.POST['item_text'])        #print(redirect('/'))        return redirect('/')    items = Item.objects.all()    return render(request,'home.html',{
'items':items})

运行,出现'no such table lists_item'错误

5.7使用迁徙创建生产数据库

pyhon3 manage.py migrate创建数据库

同时使用forloop.counter解决序号问题

{% for item in items %}        {
{ forloop.counter }}: {
{ item.text }}{
% endfor %}

运行。成功。。。

 

 

 

转载于:https://www.cnblogs.com/fg2312/p/7625591.html

你可能感兴趣的文章
2018年3月份
查看>>
SSH-struts2的异常处理
查看>>
《30天自制操作系统》学习笔记--第14天
查看>>
LGPL协议的理解
查看>>
1、Python基础
查看>>
Unity The Tag Attribute Matching Rule
查看>>
试着理解下kvm
查看>>
WebService学习总结(二)--使用JDK开发WebService
查看>>
Tizen参考手机RD-210和RD-PQ
查看>>
竞价广告系统-位置拍卖理论
查看>>
策略模式 C#
查看>>
[模板]树状数组
查看>>
[HDU 6447][2018CCPC网络选拔赛 1010][YJJ's Salesman][离散化+线段树+DP]
查看>>
设计模式学习的好方法
查看>>
感谢Leslie Ma
查看>>
几种排序方法
查看>>
查看数据库各表的信息
查看>>
第一阶段测试题
查看>>
第二轮冲刺第五天
查看>>
图片压缩
查看>>