Skip to content

Commit

Permalink
Add a few testing helper utilities to pueblo.testing
Browse files Browse the repository at this point in the history
  • Loading branch information
amotl committed Nov 7, 2023
1 parent b42f504 commit c4f1082
Show file tree
Hide file tree
Showing 10 changed files with 240 additions and 4 deletions.
1 change: 1 addition & 0 deletions CHANGES.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
# Changes for pueblo

## Unreleased
- Add a few testing helper utilities to `pueblo.testing`

## 2023-11-06 v0.0.3
- ngr: Fix `contextlib.chdir` only available on Python 3.11 and newer
Expand Down
35 changes: 35 additions & 0 deletions pueblo/testing/folder.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
from pathlib import Path


def list_files(path: Path, pattern: str):
"""
Enumerate all files in given directory.
"""
files = path.glob(pattern)
return [item.relative_to(path) for item in files]


def list_notebooks(path: Path, pattern: str = "*.ipynb"):
"""
Enumerate all Jupyter Notebook files found in given directory.
"""
return list_files(path, pattern)


def list_python_files(path: Path, pattern: str = "*.py"):
"""
Enumerate all regular Python files found in given directory.
"""
pyfiles = []
for item in list_files(path, pattern):
if item.name in ["conftest.py"] or item.name.startswith("test"):
continue

Check warning on line 26 in pueblo/testing/folder.py

View check run for this annotation

Codecov / codecov/patch

pueblo/testing/folder.py#L26

Added line #L26 was not covered by tests
pyfiles.append(item)
return pyfiles


def str_list(things):
"""
Converge list to list of strings.
"""
return list(map(str, things))
13 changes: 13 additions & 0 deletions pueblo/testing/nlp.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import pytest


@pytest.fixture(scope="session", autouse=True)
def nltk_init():
"""
Initialize nltk upfront, so that it does not run stray output into Jupyter Notebooks.
"""
download_items = ["averaged_perceptron_tagger", "punkt"]
import nltk

for item in download_items:
nltk.download(item)
2 changes: 1 addition & 1 deletion pueblo/testing/notebook.py
Original file line number Diff line number Diff line change
Expand Up @@ -46,4 +46,4 @@ async def async_execute_cell(
else: # noqa: RET506
raise

NotebookClient.async_execute_cell = async_execute_cell # type: ignore[method-assign]
NotebookClient.async_execute_cell = async_execute_cell # type: ignore[method-assign,unused-ignore]
48 changes: 48 additions & 0 deletions pueblo/testing/snippet.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import importlib
import typing as t
from pathlib import Path

import pytest


def pytest_module_function(request: pytest.FixtureRequest, filepath: t.Union[str, Path], entrypoint: str = "main"):
"""
From individual Python file, collect and wrap the `main` function into a test case.
"""
from _pytest.monkeypatch import MonkeyPatch
from _pytest.python import Function

path = Path(filepath)

# Temporarily add parent directory to module search path.
with MonkeyPatch.context() as m:
m.syspath_prepend(path.parent)

# Import file as Python module.
mod = importlib.import_module(path.stem)
fun = getattr(mod, entrypoint)

# Wrap the entrypoint function into a pytest test case, and run it.
test = Function.from_parent(request.node, name=entrypoint, callobj=fun)
test.runtest()
return test.reportinfo()


def pytest_notebook(request: pytest.FixtureRequest, filepath: t.Union[str, Path]):
"""
From individual Jupyter Notebook file, collect cells as pytest
test cases, and run them.
Not using `NBRegressionFixture`, because it would manually need to be configured.
"""
from _pytest._py.path import LocalPath
from pytest_notebook.plugin import pytest_collect_file

tests = pytest_collect_file(LocalPath(filepath), request.node)
if not tests:
raise ValueError(f"No tests collected from notebook: {filepath}")

Check warning on line 43 in pueblo/testing/snippet.py

View check run for this annotation

Codecov / codecov/patch

pueblo/testing/snippet.py#L43

Added line #L43 was not covered by tests
infos = []
for test in tests.collect():
test.runtest()
infos.append(test.reportinfo())
return infos
27 changes: 24 additions & 3 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ dependencies = [

[project.optional-dependencies]
all = [
"pueblo[cli,ngr,nlp,web]",
"pueblo[cli,ngr,nlp,testing,web]",
]
cli = [
"click<9",
Expand All @@ -78,7 +78,7 @@ cli = [
"python-dotenv<2",
]
develop = [
"black<24",
"black[jupyter]<24",
"mypy==1.6.1",
"poethepoet<1",
"pyproject-fmt<1.5,>=1.3",
Expand All @@ -102,6 +102,14 @@ test = [
"pytest-cov<5",
"pytest-mock<4",
]
testing = [
"coverage~=7.3",
"ipykernel",
"pytest<8",
"pytest-cov<5",
"pytest-env<2",
"pytest-notebook<0.9",
]
web = [
"requests-cache<2",
]
Expand Down Expand Up @@ -131,7 +139,7 @@ skip_gitignore = false
minversion = "2.0"
addopts = """
-rfEX -p pytester --strict-markers --verbosity=3
--cov --cov-report=term-missing --cov-report=xml
--cov=pueblo --cov-report=term-missing --cov-report=xml
--capture=no
--ignore=tests/ngr
"""
Expand All @@ -142,6 +150,19 @@ xfail_strict = true
markers = [
"ngr",
]
env = [
"PYDEVD_DISABLE_FILE_VALIDATION=1",
]


# pytest-notebook settings
nb_test_files = "true"
nb_coverage = "false"
nb_diff_ignore = [
"/metadata/language_info",
"/cells/*/execution_count",
"/cells/*/outputs/*/execution_count",
]

[tool.coverage.run]
branch = false
Expand Down
13 changes: 13 additions & 0 deletions tests/test_nlp.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
# TODO: Publish as real package.
from pueblo.testing.nlp import nltk_init # noqa: F401


def test_cached_web_resource():
from pueblo.nlp.resource import CachedWebResource

Expand All @@ -8,3 +12,12 @@ def test_cached_web_resource():
from langchain.schema import Document

assert isinstance(docs[0], Document)


def test_nltk_init(nltk_init): # noqa: F811
"""
Just _use_ the fixture to check if it works well.
TODO: Anything else that could be verified here?
"""
pass
40 changes: 40 additions & 0 deletions tests/test_testing.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,45 @@
from pathlib import Path

from pueblo.testing.notebook import monkeypatch_pytest_notebook_treat_cell_exit_as_notebook_skip
from pueblo.testing.snippet import pytest_module_function, pytest_notebook

HERE = Path(__file__).parent


def test_monkeypatch_pytest_notebook_treat_cell_exit_as_notebook_skip():
monkeypatch_pytest_notebook_treat_cell_exit_as_notebook_skip()


def test_pytest_module_function(request, capsys):
outcome = pytest_module_function(request=request, filepath=HERE / "testing" / "dummy.py")
assert isinstance(outcome[0], Path)
assert outcome[0].name == "dummy.py"
assert outcome[1] == 0
assert outcome[2] == "test_pytest_module_function.main"

out, err = capsys.readouterr()
assert out == "Hallo, Räuber Hotzenplotz.\n"


def test_pytest_notebook(request):
from _pytest._py.path import LocalPath

outcomes = pytest_notebook(request=request, filepath=HERE / "testing" / "dummy.ipynb")
assert isinstance(outcomes[0][0], LocalPath)
assert outcomes[0][0].basename == "dummy.ipynb"
assert outcomes[0][1] == 0
assert outcomes[0][2] == "notebook: nbregression(dummy)"


def test_list_python_files():
from pueblo.testing.folder import list_python_files, str_list

outcome = str_list(list_python_files(HERE / "testing"))
assert outcome == ["dummy.py"]


def test_list_notebooks():
from pueblo.testing.folder import list_notebooks, str_list

outcome = str_list(list_notebooks(HERE / "testing"))
assert outcome == ["dummy.ipynb"]
62 changes: 62 additions & 0 deletions tests/testing/dummy.ipynb
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
{
"cells": [
{
"cell_type": "markdown",
"source": [
"# A little notebook."
],
"metadata": {
"collapsed": false
}
},
{
"cell_type": "code",
"execution_count": 1,
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"Hallo, Räuber Hotzenplotz.\n"
]
}
],
"source": [
"print(\"Hallo, Räuber Hotzenplotz.\")"
],
"metadata": {
"collapsed": false
}
},
{
"cell_type": "code",
"execution_count": null,
"outputs": [],
"source": [],
"metadata": {
"collapsed": false
}
}
],
"metadata": {
"kernelspec": {
"display_name": "Python 3",
"language": "python",
"name": "python3"
},
"language_info": {
"codemirror_mode": {
"name": "ipython",
"version": 2
},
"file_extension": ".py",
"mimetype": "text/x-python",
"name": "python",
"nbconvert_exporter": "python",
"pygments_lexer": "ipython2",
"version": "2.7.6"
}
},
"nbformat": 4,
"nbformat_minor": 0
}
3 changes: 3 additions & 0 deletions tests/testing/dummy.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
def main():
print("Hallo, Räuber Hotzenplotz.") # noqa: T201
return 42

0 comments on commit c4f1082

Please sign in to comment.