AttributeError:使用pytest断言异常时,“ ExceptionInfo”对象没有属性“ traceback”

dai*_*yue 3 python exception pytest python-3.x pandas

我需要使用来声明一条错误消息py.test

import pandas as pd
import numpy as np

from inv_exception_store import InvAmtValError

MAX_INV_VAL = 10000000.0
MIN_INV_VAL = 0.0


class Invoices:

    def __init__(self, data=None):
        if data is None:
            self.__invoices = pd.Series([], dtype=np.float32)
        else:
            self.__invoices = pd.Series(pd.Series(data).astype(np.float32))

    def addInvoice(self, amount):
        try:
            if self.__invoices.size > MAX_INV_SIZE:
                raise InvNumError
            elif amount > MAX_INV_VAL or amount < MIN_INV_VAL:
                raise InvAmtValError(amount)
            else:
                self.__invoices = self.__invoices.append(pd.Series(amount).astype(np.float32), ignore_index=True)
        except (InvNumError, InvAmtValError) as e:
            print(str(e))


class InvAmtValError(Exception):
    def __init__(self, amount, message=None):
        if message is None:
            if amount > 100000000.0:
                message = 'The invoice amount(s) {} is invalid since it is > $100,000,00.00'.format(amount)
            elif amount < 0.0:
                message = 'The invoice amount(s) {} is invalid since it is < $0.00'.format(amount)
            else:
                message = 'The invoice amount(s) {} is invalid'.format(amount)

        super(InvAmtValError, self).__init__(str(self.__class__.__name__) + ': ' + message)
        self.message = message

    def __str__(self):
        return self.message

class TestInvoice(object):
        def test_invalid_inv_amount_err(self):
            with pytest.raises(InvAmtValError) as e:
                invoices = Invoices()

                invoices.addInvoice(-1.2)

                assert str(e) == 'The invoice amount(s) -1.2 is invalid since it is < $0.00'

                invoices.addInvoice(100000000.1)

                assert str(e) == 'The invoice amount(s) 100000000.1 is invalid since it is > $100,000,00.00'
Run Code Online (Sandbox Code Playgroud)

通过运行测试,我得到了

self = <ExceptionInfo AttributeError tblen=2>

    def __str__(self):
>       entry = self.traceback[-1]
E       AttributeError: 'ExceptionInfo' object has no attribute 'traceback'
Run Code Online (Sandbox Code Playgroud)

我想知道如何在py.test这里声明异常。

更新。尝试了建议的解决方案,

    def test_invalid_min_inv_amount_err(self):
        with pytest.raises(InvAmtValError) as e:
            invoices = Invoices()

            invoices.addInvoice(-1.2)
        assert str(e) == 'The invoice amount(s) -1.2 is invalid since it is < $0.00'
        assert e.type == InvAmtValError
Run Code Online (Sandbox Code Playgroud)

得到

>           invoices.addInvoice(-1.2)
E           Failed: DID NOT RAISE
Run Code Online (Sandbox Code Playgroud)

hoe*_*ing 5

您不能使用ExceptionInfo内部with pytest.raises上下文。运行预期在上下文中引发的代码,并在外部处理异常信息:

with pytest.raises(InvAmtValError) as e:
    invoices = InvoiceStats()
    invoices.addInvoice(-1.2)

assert str(e) == 'The invoice amount(s) -1.2 is invalid since it is < $0.00'
assert e.type == InvAmtValError  # etc
Run Code Online (Sandbox Code Playgroud)

但是,如果您只想声明异常消息,那么惯用的方式是将期望的消息直接传递给pytest.raises

expected = 'The invoice amount(s) -1.2 is invalid since it is < $0.00'
with pytest.raises(InvAmtValError, message=expected):
    invoices = InvoiceStats()
    invoices.addInvoice(-1.2)

expected = 'The invoice amount(s) 100000000.1 is invalid since it is > $100,000,00.00'
with pytest.raises(InvAmtValError, message=expected):
    invoices = InvoiceStats()
    invoices.addInvoice(100000000.1)
Run Code Online (Sandbox Code Playgroud)

更新。尝试了建议的解决方案,得到了:

>           invoices.addInvoice(-1.2)
E           Failed: DID NOT RAISE
Run Code Online (Sandbox Code Playgroud)

那是因为在addInvoice方法中确实没有引发异常-异常在try块内引发,然后立即被捕获。要么try完全删除该块,要么重新引发异常:

try:
    raise InvAmtValError(amount)
except InvAmtValError as e:
    print(str(e))
    raise e
Run Code Online (Sandbox Code Playgroud)