使用 pytest + Pycharm 的长字符串的烦人 diff 格式

Coy*_*ill 5 python pytest pycharm

你好,有这个非常基本的测试:

def test_long_diff():
    long_str1 = "ABCDEFGHIJ " * 10
    long_str2 = "ABCDEFGHIJ " * 5 + "* " + "ABCDEFGHIJ " * 5
    assert long_str1 == long_str2
Run Code Online (Sandbox Code Playgroud)

使用:Python 3.8.5、pytest-6.2.1、PyCharm 2020.2、MacO

从 shell 运行 pytest,输出是“可用的”,并且错误消息将指出长字符串中的错误字符:

(venv) ~/dev/testdiff/> pytest longdiff.py 
========== test session starts ===========
platform darwin -- Python 3.8.5, pytest-6.2.1, py-1.10.0, pluggy-0.13.1
[...]
>       assert long_str1 == long_str2
E       AssertionError: assert 'ABCDEFGHIJ A...J ABCDEFGHIJ ' == 'ABCDEFGHIJ A...J ABCDEFGHIJ '
E         Skipping 45 identical leading characters in diff, use -v to show
E         - BCDEFGHIJ * ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ 
E         ?          --
E         + BCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ
Run Code Online (Sandbox Code Playgroud)

使用 pytest-clarity 和-vv选项,我得到彩色差异(未在下面呈现)和不同的细节:

E         left:  "ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ "
E         right: "ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ * ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ "
E         
E         left and right have different lengths:
E           len(left) == 110, len(right) == 112
Run Code Online (Sandbox Code Playgroud)

但是,如果我让 Pycharm 运行测试(相同的 Python 版本,相同的 .venv,我只需右键单击测试并选择“运行 'pytest for ...'”),则运行控制台中的输出几乎无法使用,因为“某些东西一路走来”在应用差异之前将长字符串转换为较短字符串的元组:

FAILED                                       [100%]
longdiff.py:0 (test_long_diff)
('ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ '
 'ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ') != ('ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ * ABCDEFGHIJ '
 'ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ABCDEFGHIJ ')

<Click to see difference>
Run Code Online (Sandbox Code Playgroud)

单击<Click to see difference>基本上会在更大的窗口中显示相同的乱码输出

是什么导致了 Pycharm 输出?有没有办法阻止这种行为?理想情况下,我希望在运行控制台中看到 pytest-clarity 的输出。

Coy*_*ill 3

事实证明,这是 PyCharm 使用的 pytest 插件的硬编码行为。该插件始终适用 pprint.pformat()于左侧值和右侧值。

当字符串长度超过 80 个字符并且包含空格时,就会出现问题中描述的行为。

一种可能的解决方法是覆盖插件的pytest_assertrepr_compare钩子。这是一个对我有用的版本。只需将其粘贴到您的conftest.py.

import pprint
import pytest

@pytest.hookimpl(tryfirst=True)
def pytest_assertrepr_compare(config, op, left, right):
    if op in ('==', '!='):
        return ['{0} {1} {2}'.format(pprint.pformat(left, width=999), op, pprint.pformat(right, width=999))]
Run Code Online (Sandbox Code Playgroud)

另一种可能的破解方式是:monkeypatch pprint.pformat

import pytest
import pprint
from functools import partial

@pytest.fixture(scope='function', autouse=True)
def fix_long_string_diffs(monkeypatch):
    monkeypatch.setattr(pprint, 'pformat', partial(pprint.pformat, width=999))
Run Code Online (Sandbox Code Playgroud)