Jam*_*ter 5 python unit-testing dependency-injection mocking flask
我正在研究一个webapp in flask并使用服务层来抽象数据库查询和操作远离视图和api路由.有人建议这样可以使测试变得更容易,因为你可以模拟出服务层,但我无法找到一个好方法来做到这一点.举个简单的例子,假设我有三个SQLAlchemy模型:
models.py
class User(db.Model):
    id = db.Column(db.Integer, primary_key = True)
    email = db.Column(db.String)
class Group(db.Model):
    id = db.Column(db.Integer, primary_key = True)
    name = db.Column
class Transaction(db.Model):
    id = db.Column(db.Integer, primary_key = True)
    from_id = db.Column(db.Integer, db.ForeignKey('user.id'))
    to_id = db.Column(db.Integer, db.ForeignKey('user.id'))
    group_id = db.Column(db.Integer, db.ForeignKey('group.id'))
    amount = db.Column(db.Numeric(precision = 2))
用户之间存在用户和组以及交易(代表货币转手).现在我有一个services.py,其中包含一系列函数,例如检查某些用户或组是否存在,检查用户是否是特定组的成员等.我在发送JSON的api路由中使用这些服务在请求中并使用它向db添加事务,类似于:
routes.py
import services
@app.route("/addtrans")
def addtrans():
    # get the values out of the json in the request
    args = request.get_json()
    group_id = args['group_id']
    from_id = args['from']
    to_id = args['to'] 
    amount = args['amount']
    # check that both users exist
    if not services.user_exists(to_id) or not services.user_exists(from_id):
        return "no such users"
    # check that the group exists
    if not services.group_exists(to_id):
        return "no such group"
    # add the transaction to the db
    services.add_transaction(from_id,to_id,group_id,amount)
    return "success"
当我试图模拟这些服务进行测试时,问题出现了.我一直在使用模拟库,我必须修补服务模块中的函数,以便将它们重定向到模拟,如下所示:
mock = Mock()
mock.user_exists.return_value = True
mock.group_exists.return_value = True
@patch("services.user_exists",mock.user_exists)
@patch("services.group_exists",mock.group_exists)
def test_addtrans_route(self):
    assert "success" in routes.addtrans()
由于各种原因,这种感觉很糟糕.一,修补感觉很脏; 二,我不喜欢必须修补我单独使用的每种服务方法(据我所知,没有办法修补整个模块).
我想到了几个方法.
routes.services = mymock我在评估这些选项和思考其他选项时遇到了麻烦.进行python web开发的人在测试使用它们的路由时通常会模拟服务吗?
您可以使用依赖注入或控制反转来实现更简单的测试代码.
替换这个:
def addtrans():
    ...
    # check that both users exist
    if not services.user_exists(to_id) or not services.user_exists(from_id):
        return "no such users"
    ...
有:
def addtrans(services=services):
    ...
    # check that both users exist
    if not services.user_exists(to_id) or not services.user_exists(from_id):
        return "no such users"
    ...
发生了什么:
services希望使用相同的接口来解耦代码.例如:
class MockServices:
    def user_exists(id):
        return True
一些资源: