Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Getting an uninitialized error when using pytest_deselected hook #13149

Closed
pfunk1978 opened this issue Jan 20, 2025 · 6 comments
Closed

Getting an uninitialized error when using pytest_deselected hook #13149

pfunk1978 opened this issue Jan 20, 2025 · 6 comments
Labels
type: docs documentation improvement, missing or needing clarification

Comments

@pfunk1978
Copy link

When using the included hook. I'm getting the following error:

INTERNALERROR>     config.hook.pytest_deselected(do_not_run)
INTERNALERROR> TypeError: HookCaller.__call__() takes 1 positional argument but 2 were given

I'm not sure if I was supposed to initialize something somewhere. I didn't see anything in the docs about it.

Error produced from the following code:

def pytest_configure(config):
    config.addinivalue_line(
        "markers",
        "requirement(jira_issue): The jira issue that is being tested"
    )


def _is_wanted_test(item, run_tags) -> bool:
    maybe_wanted = list(item.iter_markers(name="requirment"))
    if maybe_wanted:
        return (maybe_wanted[0].args[0] in run_tags)
    return False


def pytest_collection_modifyitems(session, config, items):
    req_tags_str = config.getoption("--requirements")
    if req_tags_str:
        req_tags = [i.strip() for i in req_tags_str.split(",")]
        do_not_run = filter(
            lambda item: not _is_wanted_test(item, req_tags),
            items
        )
        config.hook.pytest_deselected(do_not_run)

pytest 8.3.4
os Fedora 41
pip list:

allure-pytest         2.13.5
allure-python-commons 2.13.5
annotated-types       0.7.0
argon2-cffi           23.1.0
argon2-cffi-bindings  21.2.0
attrs                 23.2.0
bitarray              2.9.3
bitstring             4.2.3
bitstruct             8.19.0
certifi               2024.2.2
cffi                  1.16.0
charset-normalizer    3.3.2
click                 8.1.8
colorama              0.4.6
cryptography          42.0.7
dotmap                1.3.30
enum-compat           0.0.3
hypothesis            6.102.4
idna                  3.7
iniconfig             2.0.0
kyber                 0.1.0
liboqs                0.9.1
markdown-it-py        3.0.0
mdurl                 0.1.2
ntplib                0.4.0
numpy                 2.2.1
orjson                3.10.14
packaging             24.0
pip                   24.3.1
pluggy                1.5.0
protobuf              5.29.1
pyasn1                0.6.1
pyasn1_modules        0.4.1
pycparser             2.22
pycryptodome          3.21.0
pycryptodomex         3.21.0
pydantic              2.10.5
pydantic_core         2.27.2
Pygments              2.19.1
pyparsing             3.2.0
pytest                8.3.4
pytest-jira-xray      0.8.11
requests              2.32.2
rich                  13.9.4
shellingham           1.5.4
sortedcontainers      2.4.0
typer                 0.9.4
types-zxcvbn          4.4.1.20240106
typing_extensions     4.12.2
urllib3               2.2.1
zxcvbn                4.4.28
@RonnyPfannschmidt
Copy link
Member

Please use keyword arguments

@pfunk1978
Copy link
Author

dang. that was it.

However, pytest is still running everything, not just the ones that I have marked. I've confirmed the ones that I want are not in the list that I'm sending to the hook.

@nicoddemus
Copy link
Member

nicoddemus commented Jan 20, 2025

As described in the docs, you need to change the items list in-place leaving only the items that should be run.

def pytest_collection_modifyitems(session, config, items):
    req_tags_str = config.getoption("--requirements")
    if req_tags_str:
        req_tags = [i.strip() for i in req_tags_str.split(",")]
        run = filter(
            lambda item: _is_wanted_test(item, req_tags),
            items
        )
        do_not_run = filter(
            lambda item: not _is_wanted_test(item, req_tags),
            items
        )
        items[:] = run
        config.hook.pytest_deselected(items=do_not_run)

Or:

def pytest_collection_modifyitems(session, config, items):
    req_tags_str = config.getoption("--requirements")
    if req_tags_str:
        req_tags = [i.strip() for i in req_tags_str.split(",")]
        run = [i for i in items if _is_wanted_test(item)]   
        do_not_run = [i for i in items if not _is_wanted_test(item)]   
        items[:] = run
        config.hook.pytest_deselected(items=do_not_run)

@The-Compiler
Copy link
Member

For what it's worth, exactly the docs you linked to claim:

[...] the hook pytest_deselected must be called [...], e.g. with config.hook.pytest_deselected(deselected_items).

which evidently doesn't work, so that seems like a docs bug to me.

@The-Compiler The-Compiler added the type: docs documentation improvement, missing or needing clarification label Jan 21, 2025
nicoddemus added a commit that referenced this issue Jan 21, 2025
Ref #13149
@nicoddemus
Copy link
Member

Indeed, #13152. 👍

nicoddemus added a commit that referenced this issue Jan 21, 2025

Verified

This commit was signed with the committer’s verified signature.
Groverkss Kunwar Grover
Ref #13149
patchback bot pushed a commit that referenced this issue Jan 21, 2025
Ref #13149

(cherry picked from commit 4a6a512)
nicoddemus added a commit that referenced this issue Jan 21, 2025
Ref #13149

(cherry picked from commit 4a6a512)

Co-authored-by: Bruno Oliveira <[email protected]>
@pfunk1978
Copy link
Author

Thanks everyone.

For future reference, this is the final code that worked for me:

def pytest_configure(config):
    config.addinivalue_line(
        "markers",
        "{requirement}(jira_issue): The jira issue that is being tested"
    )


@curry
def _is_wanted_test(run_tags, marker, item) -> bool:
    maybe_wanted = list(item.iter_markers(name=marker))
    if maybe_wanted:
        for x in maybe_wanted:
            if x.args[0] in run_tags:
                return True
    return False


def _sort_tests(items, checker: callable):
    discard = []
    keep = []
    for item in items:
        if checker(item):
            keep.append(item)
        else:
            discard.append(item)
    return keep, discard


def pytest_collection_modifyitems(session, config, items):
    req_tags_str = config.getoption("--requirements")
    if req_tags_str:
        req_tags = [i.strip() for i in req_tags_str.split(",")]
        keep, discard = _sort_tests(
            items, _is_wanted_test(req_tags, "requirement")
        )
        items[:] = keep
        config.hook.pytest_deselected(items=discard)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
type: docs documentation improvement, missing or needing clarification
Projects
None yet
Development

No branches or pull requests

4 participants