实战项目(二)-嘉宾发布会签到系统

先前学习了虫师老师的接口自动化测试教程,总结一下项目实战思路

近期分享: 接口自动化测试从零开始学习集锦: https://www.jianshu.com/nb/49125734 --欢迎探讨!!

1、Django开发一个发布会嘉宾签到系统;
2、学习Html前端优化展示;
3、保证开发的功能点测试通过;
4、开发接口视图层代码;
5、安装Pymsql,搭建本地数据库, 设计测试数据初始化,并对数据的插入删除做了封装
6、安装Requests库调用接口;
7、unittest单元测试框架运行测试;
8、HTMLTestRunner生成接口测试报告;

步骤1: db_fixture.mysql_db.py

Created on 2017年8月29日
@author: Yvon_早安阳光

from pymysql import  connect,cursors
from pymysql.err import OperationalError
import os
import configparser as cparser
 
#============读取db_config.ini文件设置============
'''获取当前路径'''
base_dir = str(os.path.dirname(os.path.dirname(__file__)))
print(base_dir)
'''转换路径格式'''
base_dir = base_dir.replace('\\', '/')
'''获取数据库配置文件的路径'''
file_path = base_dir + '/db_config.ini'
print(file_path)
 
cf = cparser.ConfigParser()
cf.read(file_path)
host = cf.get("mysqlconf", "host")
print(host)
port = cf.get("mysqlconf", "port")
user = cf.get("mysqlconf", "user")
print(user)
password = cf.get("mysqlconf", "password")
db = cf.get("mysqlconf", "db_name")
 
#============封装MySQL基本操作============ 
class DB:
    def __init__(self):    
        try:
            #连接数据库
            self.connection = connect(host=host,
                                port=int(port),
                                user=user,
                                password=password,
                                db=db,
                                charset='utf8mb4',
                                cursorclass=cursors.DictCursor)
        except OperationalError as e:
            print("Mysql Error %d: %s" % (e.args[0], e.args[1]))
             
    #清除表数据
    def clear(self,table_name):
        real_sql = 'delete from ' + table_name + ';'
        print(real_sql)
        with self.connection.cursor() as cursor:
            cursor.execute("SET FOREIGN_KEY_CHECKS=0;")
            cursor.execute(real_sql)
        self.connection.commit()
         
    #插入数据
    def insert(self,table_name,table_data):
        for key in table_data:
            table_data[key] = "'" + str(table_data[key]) + "'"
        key = ','.join(table_data.keys())
        value = ','.join(table_data.values())
        real_sql = 'INSERT INTO ' + table_name + " (" + key + ") VALUES (" + value + ") "
        print(real_sql)
        with self.connection.cursor() as cursor:   
            cursor.execute(real_sql)
        self.connection.commit()
 
    #关闭数据库连接
    def close(self):
        self.connection.close()
         
    # 插入数据
    def init_data(self, datas):
        for table, data in datas.items():
            self.clear(table)
            for d in data:
                self.insert(table, d)
        self.close()
 
if __name__ == '__main__':
    db =DB()
    table_name = "sign_event"
    data = {'id':2,'name':'土豆','`limit`':3000,'status':1,'address':'杭州博物馆','start_time':'2018-05-15 14:25:42','create_time':'2016-07-20 00:25:42'}
    table_name2 = "sign_guest"
    data2 = {'realname':'alen','phone':12312341234,'email':'alen@mail.com','sign':0,'event_id':1}
  
    db.clear(table_name)
    db.insert(table_name, data)
    db.close()

步骤2: db_fixture.test_data.py

'''
Created on 2017年8月29日
@author: Yvon_早安阳光
'''
import sys
sys.path.append('../db_fixture')
try:
    from mysql_db import DB
except ImportError:
    from .mysql_db import DB
 
# 创建测试数据
datas = {
    #发布会表数据
    'sign_event':[
        #有效的发布会数据
        {'id':1,'name':'OPPO发布会','`limit`':2000,'status':1,'address':'杭州江干区','start_time':'2019-05-20 14:00:00','create_time':'2016-07-20 00:25:42'},
        {'id':2,'name':'联想发布会','`limit`':1500,'status':1,'address':'北京博远路','start_time':'2019-05-25 14:25:38','create_time':'2016-07-20 00:25:42'},
        #参加人数限制
        {'id':3,'name':'ViVo发布会','`limit`':0,'status':1,'address':'苏州通园路','start_time':'2019-05-20 14:00:00','create_time':'2016-07-20 00:25:42'},
        #发布会未开始,状态关闭
        {'id':5,'name':'三星发布会','`limit`':2000,'status':0,'address':'南京博物馆','start_time':'2019-07-25 14:00:00','create_time':'2016-07-20 00:25:42'},
        {'id':6,'name':'锤子发布会','`limit`':1000,'status':0,'address':'南通泗水路','start_time':'2019-05-10 14:00:00','create_time':'2016-07-20 00:25:42'},
        #发布会已结束
        {'id':7,'name':'小米发布会','`limit`':800,'status':1,'address':'北京水立方','start_time':'2016-05-20 14:00:00','create_time':'2016-07-20 00:25:42'},
        {'id':8,'name':'大米发布会','`limit`':1500,'status':1,'address':'淮安高教区','start_time':'2017-05-20 14:34:56','create_time':'2016-07-20 00:25:42'},
    ],
     #嘉宾表数据
    'sign_guest':[
        #未签到有效数据
        {'id':1,'realname':'张强','phone':18612341234,'email':'andy@mail.com','sign':0,'event_id':1,'create_time':'2016-07-20 00:25:42'},
        {'id':3,'realname':'刘强','phone':17701231234,'email':'marry@163.com','sign':0,'event_id':2,'create_time':'2016-07-20 00:25:42'},
        #已签到
        {'id':2,'realname':'李强','phone':15040013210,'email':'dugu@126.com','sign':1,'event_id':1,'create_time':'2016-07-20 00:25:42'},
        #发布会人数限制、发布会未激活,发布会已结束
        {'id':5,'realname':'郭强','phone':18108090705,'email':'xiaowu@mail.com','sign':0,'event_id':3,'create_time':'2016-07-20 00:25:42'},
        {'id':6,'realname':'金俊','phone':13064931234,'email':'jinj@163.com','sign':0,'event_id':5,'create_time':'2017-07-20 00:00:00'},
 
    ],
}
 
# 将测试数据插入表
def init_data():
    DB().init_data(datas)

步骤3: Views_interface.py

思路1: 创建新的一条发布会

访问地址:http://127.0.0.1:8000/api/add_event](http://127.0.0.1:8000/api/add_event)

1、首先,判断eid、name、limit、address、start_time 等字段均不能为空,否则JsonResponse()返回相应的状态码和提示。JsonResponse()是一个非常有用的方法,它可以直接将字典转化成Json 格式返回到客户端。
2、接下来,判断发布会id 是否存在,以及发布会名称(name)是否存在;如果存在将返回相应的状态码和提示信息。将数据字典作为整个返回字典中data 对应的值返回。
3、再接下来,判断发布会状态是否为空,如果为空,将状态设置为1(True)。
4、最后,将数据插入到Event表,在插入的过程中如果日期格式错误,将抛出ValidationError异常,接收该异常并返回相应的状态和提示,否则,插入成功,返回状态码200 和“add event success”的提示。

思路2: 查询发布会接口

访问地址: http://127.0.0.1:8000/api/get_event_list/?eid=4](http://127.0.0.1:8000/api/get_event_list/?eid=4
http://127.0.0.1:8000/api/get_event_list/?name=](http://127.0.0.1:8000/api/get_event_list/?name=发布会

1、通过GET 请求接收发布会id 和name 参数。两个参数都是可选的。首先,判断当两个参数同时为空,接口返回状态码10021,参数错误。
2、如果发布会id 不为空,优先通过id 查询,因为id 的唯一性,所以,查询结果只会有一条,将查询结果以key:value 对的方式存放到定义的event 字典中,并将数据字典作为整个返回字典中data 对应的值返回。
3、name 查询为模糊查询,查询数据可能会有多条,返回的数据稍显复杂;首先将查询的每一条数据放到一个字典event 中,再把每一个字典再放到数组datas 中,最后再将整个数组做为返回字典中data 对应的值返回。

思路3: 添加嘉宾接口

访问地址: http://127.0.0.1:8000/api/add_guest/](http://127.0.0.1:8000/api/add_guest/

通过POST 请求接收嘉宾参数:关联的发布会id、姓名、手机号和邮箱等参数。
1、首先,判断eid、realname、phone 等参数均不能为空。
2、接下来,判断嘉宾关联的发布会id 是否存在,以及关联的发布会状态是否为True(即1),如果不存在或不为True,将返回相应的状态码和提示信息。
3、再接下来的步骤是判断当前时间是否大于发布会时间,如果大于则说明发布已开始,或者早已经结束。那么该发布会就应该不能允许再添加嘉宾。
4、最后,插入嘉宾数据,如果发布会的手机号重复则抛IntegrityError 异常,接收该异常并返回相应的状态码和提示信息。如果添加成功,则返回状态码200 和“add guest success”的提示。

代码参考

<meta charset="utf-8">
from django.http.response import HttpResponse, JsonResponse

'''查询数据库'''
from sign.models import Event,Guest
from django.core.exceptions import ValidationError, ObjectDoesNotExist
from django.db.utils import IntegrityError
import time
from django.template.context_processors import request

#添加发布会接口
def add_event(request):
    eid = request.POST.get('eid','')     #发布会id
    name = request.POST.get('name','')   #发布会标题
    limit = request.POST.get('limit','')  #限制人数
    status = request.POST.get('status','')  #状态
    address = request.POST.get('address','')  #地址
    start_time = request.POST.get('start_time','')  #发布会时间

    '''必填项为空'''
    if eid =='' or name == '' or limit == '' or address == '' or start_time == '':
        return JsonResponse({'status':207,'message':'parameter error'})
    '''event id重复'''
    results = Event.objects.filter(id = eid)
    if results:
        return JsonResponse({'status':201,'message':'event id already exists'})
    '''event name 重复'''
    result = Event.objects.filter(name = name)
    if result:
        return JsonResponse({'status':201,'message':'event name already exists'})

    '''创建新的一条发布会''' 
    if status == '': 
        status = 1  
    try:
        Event.objects.create(id=eid,name=name,limit=limit,address=address,status=int(status),start_time=start_time)
    except ValidationError as e:
        error = 'start_time format error. It must be in YYYY-MM-DD HH:MM:SS format.' 
        return JsonResponse({'status':202,'message':'start_tiem format error'})
    return JsonResponse({'status':200,'message':'add event success'})

#查询发布会接口
def get_event_list(request):
    eid = request.GET.get('eid','')
    name = request.GET.get('name','')
    if eid == '' and name == '':
        return JsonResponse({'status':207,'message':'parameter error'})
    if eid !='':
        event = {}
        try:
            result = Event.objects.get(id = eid)
        except ObjectDoesNotExist:
            return JsonResponse({'status':201,'message':'query result is empty'})
        else:
            event['name'] = result.name
            event['limit'] = result.limit
            event['status'] = result.status
            event['address'] = result.address
            event['start_time'] = result.start_time
            return JsonResponse({'status':200,'message':'success','data':event})

    if name !='':
        datas = []
        results = Event.objects.filter(name__contains = name)  #发布会名称模糊查询
        if results:
            for r in results:
                event = {}
                event['name'] = r.name
                event['limit'] = r.limit
                event['status'] = r.status
                event['address'] = r.address
                event['start_time'] = r.start_time
                datas.append(event)
            return JsonResponse({'status':200,'message':'success','data':datas})
        else:
            return JsonResponse({'status':201,'message':'query result is empty'})

#添加嘉宾接口
def add_guest(request):
    eid = request.POST.get('eid','')    #关联发布会id 
    realname = request.POST.get('realname','') #姓名
    phone = request.POST.get('phone','') #手机号
    email = request.POST.get('email','') #邮箱

    '''必填项为空'''
    if eid =='' or realname == '' or phone == '' :
        return JsonResponse({'status':207,'message':'parameter error'})
    '''eid不存在'''
    result = Event.objects.filter(id=eid)   
    if not result:
        return JsonResponse({'status':202,'message':'event id null'})
    '''status状态为False,不可再添加嘉宾'''
    result = Event.objects.get(id=eid).status
    if not result:
        return JsonResponse({'status':203,'message':'event status is not available'})

    '''限制发布会人数'''
    event_limit = Event.objects.get(id=eid).limit    #发布会限制人数
    guest_limit = Guest.objects.filter(event_id=eid) #发布会已添加的嘉宾数
    if len(guest_limit) >= event_limit:
        return JsonResponse({'status':204,'message':'event number is full'})
    '''发布会已结束'''
    event_time = Event.objects.get(id=eid).start_time #获取发布会开始时间
    print(event_time)
#     etime = str(event_time).split(".")[0]  #截取小数点左边的时间数据,精确到秒,并转成字符串
    etime = str(event_time).split('+')[0]  # 根据实际的打印结果截取右侧的数据,打印出来的结果是"+00:00",故截取+号右侧的数据
    print(etime)
    timeArray = time.strptime(str(etime), "%Y-%m-%d %H:%M:%S") #转换成时间戳的格式
    e_time = int(time.mktime(timeArray)) #获取最终的发布会的开始时间
    print(e_time)
    now_time = str(time.time())  #获取当前时间
    print(now_time)
    ntime = now_time.split(".")[0] #截取小数点左边的时间数据
    n_time = int(ntime)
    print(n_time)

    #当前时间大于发布会时间
    if n_time >= e_time:
        return JsonResponse({'status':205,'message':'event has started'})
    try:
        Guest.objects.create(realname=realname,phone=int(phone),email=email,sign=0,event_id=int(eid))
    except IntegrityError:
        return JsonResponse({'status':206,'message':'the event guest phone number repeat'}) #嘉宾重复添加手机号
    return JsonResponse({'status':200,'message':'add guest success'})

步骤4: interface.add_event_test.py

<meta charset="utf-8">
'''
Created on 2017年8月29日
@author: Yvon_早安阳光
'''
#coding:utf-8

import unittest
import requests
import os, sys

parentdir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.insert(0, parentdir)
from db_fixture import test_data

class AddEventTest(unittest.TestCase):
    ''' 添加发布会 '''
    def setUp(self):
        self.base_url = "[http://127.0.0.1:8000/api/add_event/](http://127.0.0.1:8000/api/add_event/)"
    def tearDown(self):
        print(self.result)
    def test_add_event_all_null(self):
        '''所有参数为空'''
        payload = {'eid':'','name':'','status':'','limit':'','address':"",'start_time':''}
        r = requests.post(self.base_url,data = payload)
        self.result = r.json()
        self.assertEqual(self.result['status'], 207)
        self.assertEqual(self.result['message'], 'parameter error')

    def test_add_event_eid_exist(self):
        '''eid 已存在'''
        payload = {'eid':1,'name':'小金理财发布会','status':'1','limit':'2000',
                   'address':"江苏建湖",'start_time':'2020-05-20 14:00:00'}
        r = requests.post(self.base_url,data = payload)
        self.result = r.json()
        self.assertEqual(self.result['status'], 201)
        self.assertEqual(self.result['message'], 'event id already exists')

    def test_add_event_name_exist(self):
        '''名称已存在'''
        payload = {'eid':9,'name':'大米发布会','status':'1','limit':'1500',
                   'address':"杭州江干庆春路",'start_time':'2019-05-20 15:00:00'}
        r = requests.post(self.base_url,data = payload)
        self.result = r.json()
        self.assertEqual(self.result['status'], 201)
        self.assertEqual(self.result['message'], 'event name already exists')

    def test_add_event_data_type_error(self):
        '''日期格式错误'''
        payload = {'eid':9,'name':'悟空理财发布会','status':'1','limit':'1000',
                   'address':"扬州中华路",'start_time':'2019'}

        r = requests.post(self.base_url,data = payload)
        self.result = r.json()
        self.assertEqual(self.result['status'], 202)
        self.assertEqual(self.result['message'],'start_tiem format error')

    def test_add_event_success(self):
        payload = {'eid':10,'name':'金立发布会','status':'1','limit':'1500',
                   'address':"盐城中华路",'start_time':'2020-05-10 13:00:00'}
        r = requests.post(self.base_url,data = payload)
        self.result = r.json()
        self.assertEqual(self.result['status'], 200)
        self.assertEqual(self.result['message'],'add event success')

if __name__ == '__main__':
    test_data.init_data() #初始化接口测试数据
    unittest.ma

步骤5: Send_mail_report.py

'''
Created on 2017年9月4日
@author: Yvon_早安阳光
'''
import time, sys, unittest, smtplib, os
sys.path.append('./interface')
sys.path.append('./db_fixture')
from HTMLTestRunner import HTMLTestRunner
from db_fixture import test_data
from email.mime.multipart import MIMEMultipart
from email.header import Header
# from email.mime.application import MIMEApplication
from email.mime.text import MIMEText
 
#======================定义发送邮件======================
 
def send_mail(file_new):
    '''file_new是指最新的测试报告'''
    f = open(file_new,'rb')
    mail_body = f.read()
    f.close()
     
    msg = MIMEMultipart()
    '''邮件标题'''
    msg['Subject'] = Header('发布会嘉宾签到接口测试报告' ,'utf-8')
     
    '''邮件正文'''
    zhengwen = "各位,详情测试结果请见附件文档"
    msg.attach(MIMEText(zhengwen, 'plain')) 
     
    '''附件测试报告 '''
    att= MIMEText(mail_body,'html','utf-8')
    att.add_header('Content-Disposition', 'attachment', filename='Guest Manage System Interface_TestReport.html')
    msg.attach(att) 
        #发送邮件
    smtp = smtplib.SMTP()
    smtp.connect('smtp.exmail.qq.com')
    smtp.login('fajin@bertadata.com','36#20qa20A')
    smtp.sendmail('fajin@bertadata.com','huiselanse@yeah.net',msg.as_string())
    print('email has send out !')
    smtp.quit()
     
#======================查找测试报告的目录,找到最新生成的测试报告======================
'''按时间获取最新的测试报告'''
def new_report(testreport):
    lists = os.listdir(testreport)
    lists.sort(key = lambda fn: os.path.getatime(testreport + '\\' + fn))
    file_new = os.path.join(testreport,lists[-1])
    print(file_new)
    return file_new
 
#指定测试用例为当前文件夹下的目录
test_dir = './interface'
discover = unittest.defaultTestLoader.discover(test_dir, pattern='*_test.py')
 
if __name__ == '__main__':
    test_data.init_data() #初始化测试数据     
    #报告存放的路径
    result_report_dir= 'result_report/'
    '''实时当前的时间'''
    now = time.strftime("%Y-%m-%d %H_%M_%S")
    file_name = result_report_dir + '\\' + now +'result.html'
    fp = open(file_name,'wb')    
    #定义测试报告
    runner =HTMLTestRunner(stream = fp,title ='嘉宾签到接口测试报告',description='接口用例执行情况:')                          
    runner.run(discover)
    fp.close() #关闭报告文件
    new_report = new_report(result_report_dir)
    send_mail(new_report)
测试报告1.png

测试报告2.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容