flask-sqlalchemy、pytest 的单元测试和事务自动回滚

使用 flask-sqlalchemy 做数据库时,单元测试可以帮助发现一些可能意想不到的问题,像 delete-cascade 、数据长度、多对多关联等等。如果使用 alembic 管理数据库版本,还可以写些跟迁移相关的单元测试。在团队中实现规范的单元测试,再配合 flake8 / pep8 之类的代码规范工具,有助于提高代码的质量,让开发人员有意识去主动发现问题,在新功能进行回归测试时可重复使用单元测试的代码,避免中断已有功能。

在数据库的单元测试前,先要把数据库表结构创建起来。有两种方式来做这个事,一是用一个专门用于单元测试的数据库,重新进行 create_all 的建表操作;或者直接用开发用的数据库, upgrade 到最新的版本,然后在单元测试完成后再进行 downgrade 。两种方法都可以使用 pytest 的 fixture 实现,都可以使用。

做数据库的单元测试需要注意的是事务回滚。通常每个单元测试会针对某个单独的类,而为了避免单元测试所插入或删除的数据对其它单元测试的影响,需要把整个单元测试包含在一个事务中,单元测试结束时自动回滚。事务所针对的层次可以是一个模块,或者是单元测试函数。

假如项目的目录结构如下 ::

.
├── config.py
├── db.sqlite
├── manage.py
├── maze
├── migrations
├── requirements-dev.txt
├── requirements-test.txt
├── requirements.txt
├── tests
├── tox.ini

db.sqlite 为开发用的 sqlite 数据库,migrations 为数据库迁移文件,tests 中放单元测试。在配置 sqlite 路径时,需要注意使用绝对路径,否则会把 sqlite 文件写在 maze 目录下。

现在有 model 类的定义如下 ::

class User(db.Model, Base, UserMixin):
"""User model.""" __tablename__ = 'users'
username = db.Column(db.String(20), unique=True, nullable=False)
email = db.Column(db.String(64), unique=True, nullable=False)
password_hash = db.Column(db.String(128), nullable=False) @property
def password(self):
raise AttributeError('Not readable') @password.setter
def password(self, value):
self.password_hash = generate_password_hash(value) def save(self):
db.session.add(self)
db.session.flush()

为了测试这个类,需要先编写一些可行的测试框架代码,以在 pytest 中支持对 flask 的测试。

在 tests 中创建 conftest.py 文件,包含我们在整个测试中都需要使用的 fixture 。首先,需要创建一个 flask 的 fixture ,并创建一个测试上下文。

# -*- coding: utf-8 -*-

import pytest

@pytest.yield_fixture(scope='session',autouse=True)
def app():
from maze import create_app
app = create_app()
ctx = app.test_request_context()
with ctx:
yield app

如果使用单独测试数据库的方式进行单元测试,那么加一个创建数据库的 fixture ,并在测试::

@pytest.yield_fixture(scope='session', autouse=True)
@pytest.mark.usefixtures('app')
def create_schema():
from maze import db
db.create_all()
try:
yield
finally:
db.drop_all()

那么在单元测试中,就已经可以通过 @pytest.mark.usefixtures('app') ,flask 的相关代码就能进行测试。为了支持自动回滚,还需要加一个 rollback 的 fixture ,如下 ::

@pytest.yield_fixture(autouse=True)
def rollback(app):
from maze import db
ctx = app.test_request_context()
with ctx:
db.session.begin(subtransactions=True)
try:
yield
finally:
db.session.rollback()
db.session.remove()

rollback 可使用默认的 function scope ,如果需要在模块范围内才进行 rollback ,那么设置 scope='module' 。但要注意,原始代码中应用 db.session.flush() 的方式刷新 session 数据,而不是用 commitrollback 对本事务上次 commit 之后的数据才生效,因此如果单元测试中有 commit 的代码,部分数据会被写入数据库,从而达不到 rollback 的目的。

OK,现在在 test_user.py 的单元测试模块中写些测试代码 ::

@pytest.usefixtures('rollback')
def test_user():
user = User(username='test', password='test123')
user.save()
assert user.id is not None user2 = User.query.filter_by(username='test').first()
assert user2 is not None
assert user == user2 with pytest.raises(AttributeError):
p = user.password # noqa:F841

最后,执行 pytest 单元测试 ::

pytest tests

整个过程有几点要注意的:

(1) 为了支持自动回滚,原始代码不使用 commit 而要用 flush ,但要在 flask 中用 @app.teardown_request 注册一个处理函数,在函数中进行最终的 commit ;

(2) fixture 需要写在 conftest.py 文件中(或在 conftest.py 中进行 import),而单元测试模块不需要 import 这些 fixture ,否则会出现找不到 fixture (尤其在 pycharm IDE 直接调用 pytest 测试时非常容易出现);

(3) 注意 flask 的上下文处理,可使用 pytest-flask 这个插件处理一些问题。

pytest 的 fixture 编写方式和 unittest 、 nose 、 xUnit 等有点区别,它是以函数的方式出现,在一开始用的时候会不习惯,也没有用 setup 、 teardown 等常见的约定函数来注入依赖,而是用简单的函数参数的方式,用过一小段时间后,感觉也还挺好,简单明了处理的方式在编写代码时少了很多麻烦。

flask-sqlalchemy、pytest 的单元测试和事务自动回滚的更多相关文章

  1. MockMvc 进行 controller层单元测试 事务自动回滚 完整实例

    package com.ieou.ms_backend.controller; import com.google.gson.Gson; import com.ieou.ms_backend.dto. ...

  2. Spring事务为什么不会自动回滚?Spring事务怎样才会自动回滚?事务自动回滚条件及手动回滚

    原文:https://blog.csdn.net/qq_32331073/article/details/76508147 更多Spring事务问题请访问链接:Spring事务回滚问题疑难详解 在此, ...

  3. mysql 事务中如果有sql语句出错,会导致自动回滚吗?

    事务,我们都知道具有原子性,操作要么全部成功,要么全部失败.但是有可能会造成误解. 我们先准备一张表,来进行测试 CREATE TABLE `name` ( `id` int(11) unsigned ...

  4. spring事务什么时候会自动回滚

    在java中异常的基类为Throwable,他有两个子类xception与Errors.同时RuntimeException就是Exception的子类,只有RuntimeException才会进行回 ...

  5. 事务框架之声明事务(自动开启,自动提交,自动回滚)Spring AOP 封装

    利用Spring AOP 封装事务类,自己的在方法前begin 事务,完成后提交事务,有异常回滚事务 比起之前的编程式事务,AOP将事务的开启与提交写在了环绕通知里面,回滚写在异常通知里面,找到指定的 ...

  6. DDL, DML不是所有SQL都是可以自动回滚的

    因为DDL没有事务性,所以DDL不能回滚. 要实现自动回滚.(begin,commit,rollback),则SQL语句中只能包括DML. 这样,自动化发布就会受限规范格式. 故而,一刀切的办法是,假 ...

  7. Spring,SpringMvc配置常见的坑,注解的使用注意事项,applicationContext.xml和spring.mvc.xml配置注意事项,spring中的事务失效,事务不回滚原因

    1.Spring中的applicationContext.xml配置错误导致的异常 异常信息: org.apache.ibatis.binding.BindingException: Invalid ...

  8. Spring声明式事务不回滚问题

    疑问,确实像往常一样在service上添加了注解 @Transactional,为什么查询数据库时还是发现有数据不一致的情况,想想肯定是事务没起作用,出现异常的时候数据没有回滚.于是就对相关代码进行了 ...

  9. spring 在service中需要抛出异常才能自动回滚

    在spring 事务配置中,如果service方法捕获了异常,则程序报错也不会自动回滚, 1.手动开启关闭事务 2.抛出异常,可以先捕获异常,然后自定义runtime异常,可不用声明

随机推荐

  1. 【Java EE 学习 24 下】【注解在数据库开发中的使用】【反射+注解+动态代理在事务中的应用service层】

    一.使用注解可以解决JavaBean和数据库中表名不一致.字段名不一致.字段数量不一致的问题. 1.Sun公司给jdbc提供的注解 @Table.@Column.@Id.@OneToMany.@One ...

  2. [BI项目记]-搭建代码管理环境之客户端

    前面已经介绍了如何搭建代码管理环境的服务器端安装和配置,这里介绍对于客户端的几个场景. 首先对于开发人员来说,可以直接使用Visual Studio来连接,这里主要演示Visual Studio 20 ...

  3. android tab选项卡的使用

    项目做完了,写写博客,在项目中遇到的一些问题,或者是自己觉得很不错的东西.这一篇主要是想和大家分享一下我在项目中封装的一个东西,就是tab选项卡.先看看效果图: 我在网上看了很多有关选项卡的demo, ...

  4. hdu2196 树形dp

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=2196 Problem Description A school bought the fi ...

  5. tomcat的debug模式启动不了

    这个问题可能是由于eclipse和tomcat的交互而产生的,在以debug模式启动tomcat时,发生了读取文件错误,eclipse自动设置了断点,导致tomcat不能正常启动.解决方法如下,打开b ...

  6. 关于jqgrid数据不显示问题

    近日有个需求要用到jqgrid,原本用着一切都很顺利,但是在需求变动后,只是修改部分字段名称jqgrid就不显示数据了,后台数据也能传到前台,但是就是不给我显示,到嘴的肉就是没法吃,蛋疼,郁闷都无法形 ...

  7. 【MongoDB初识】-增删改

    1.切换数据库 admin数据库:use admin test数据库:use test 2.新增: 方法一(首选) c} db.class.save(c) 或者db.class.insert(c) 方 ...

  8. [转]Nodejs基础中间件Connect

    Nodejs基础中间件Connect 从零开始nodejs系列文章,将介绍如何利Javascript做为服务端脚本,通过Nodejs框架web开发.Nodejs框架是基于V8的引擎,是目前速度最快的J ...

  9. 如何将U盘内文件拷入VMware Linux CentOS6.5虚拟机

    之前在Linux CentOS下安装Oracle这篇随笔中我提到要将下载到的安装文件解压缩 那么,问题来了! 如何把下载到的文件拷入虚拟机中呢? 我是这样做的: 1.将下载到的文件拷入U盘 2.以ro ...

  10. NOI 题库 8465

    8465  马走日 描述 马在中国象棋以日字形规则移动. 请编写一段程序,给定n*m大小的棋盘,以及马的初始位置(x,y),要求不能重复经过棋盘上的同一个点,计算马可以有多少途径遍历棋盘上的所有点. ...