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

Fix for #54 - allow regexes when matching expected message text #55

Merged
merged 11 commits into from
Aug 9, 2021
4 changes: 3 additions & 1 deletion .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,9 @@ jobs:
pip install -U pip setuptools wheel
pip install -r dev-requirements.txt
- name: Run tests
run: pytest
run: pytest --ignore-glob="*.shouldfail.yml"
- name: Run test with expected failures
run: pytest pytest_mypy_plugins/tests/*.shouldfail.yml 2>&1 | grep "5 failed"

lint:
runs-on: ubuntu-latest
Expand Down
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -4,3 +4,5 @@
__pycache__
dist/
build/
.pytest_cache/
venv/
42 changes: 32 additions & 10 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -52,16 +52,17 @@ You can also specify `PYTHONPATH`, `MYPYPATH`, or any other environment variable
In general each test case is just an element in an array written in a properly formatted `YAML` file.
On top of that, each case must comply to following types:

| Property | Type | Description |
| --------------- | ------------------------------------------------------ | --------------------------------------------------------------------------------------------------------------- |
| `case` | `str` | Name of the test case, complies to `[a-zA-Z0-9]` pattern |
| `main` | `str` | Portion of the code as if written in `.py` file |
| `files` | `Optional[List[File]]=[]`\* | List of extra files to simulate imports if needed |
| `disable_cache` | `Optional[bool]=False` | Set to `true` disables `mypy` caching |
| `mypy_config` | `Optional[Dict[str, Union[str, int, bool, float]]]={}` | Inline `mypy` configuration, passed directly to `mypy` as `--config-file` option |
| `env` | `Optional[Dict[str, str]]={}` | Environmental variables to be provided inside of test run |
| `parametrized` | `Optional[List[Parameter]]=[]`\* | List of parameters, similar to [`@pytest.mark.parametrize`](https://docs.pytest.org/en/stable/parametrize.html) |
| `skip` | `str` | Expression evaluated with following globals set: `sys`, `os`, `pytest` and `platform` |
| Property | Type | Description |
| --------------- | ------------------------------------------------------ | ------------------------------------------------------------------------------------------------------------------- |
| `case` | `str` | Name of the test case, complies to `[a-zA-Z0-9]` pattern |
| `main` | `str` | Portion of the code as if written in `.py` file |
| `files` | `Optional[List[File]]=[]`\* | List of extra files to simulate imports if needed |
| `disable_cache` | `Optional[bool]=False` | Set to `true` disables `mypy` caching |
| `mypy_config` | `Optional[Dict[str, Union[str, int, bool, float]]]={}` | Inline `mypy` configuration, passed directly to `mypy` as `--config-file` option |
| `env` | `Optional[Dict[str, str]]={}` | Environmental variables to be provided inside of test run |
| `parametrized` | `Optional[List[Parameter]]=[]`\* | List of parameters, similar to [`@pytest.mark.parametrize`](https://docs.pytest.org/en/stable/parametrize.html) |
| `skip` | `str` | Expression evaluated with following globals set: `sys`, `os`, `pytest` and `platform` |
| `regex` | `str` | Allow regular expressions in comments to be matched against actual output. Defaults to "no", i.e. matches full text.|

(*) Appendix to **pseudo** types used above:

Expand Down Expand Up @@ -126,6 +127,27 @@ Implementation notes:
main:1: note: Revealed type is 'builtins.str'
```

#### 4. Regular expressions in expectations

```yaml
- case: expected_message_regex_with_out
regex: yes
main: |
a = 'abc'
reveal_type(a)
out: |
main:2: note: .*str.*
```

#### 5. Regular expressions specific lines of output.

```yaml
- case: expected_single_message_regex
main: |
a = 'hello'
reveal_type(a) # NR: .*str.*
```

## Options

```
Expand Down
15 changes: 10 additions & 5 deletions pytest_mypy_plugins/collect.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,16 +101,21 @@ def collect(self) -> Iterator["YamlTestItem"]:
test_name = f"{test_name_prefix}{test_name_suffix}"
main_file = File(path="main.py", content=pystache.render(raw_test["main"], params))
test_files = [main_file] + parse_test_files(raw_test.get("files", []))
regex = raw_test.get("regex", False)

output_from_comments = []
expected_output = []
for test_file in test_files:
output_lines = utils.extract_errors_from_comments(test_file.path, test_file.content.split("\n"))
output_from_comments.extend(output_lines)
output_lines = utils.extract_output_matchers_from_comments(
test_file.path, test_file.content.split("\n"), regex=regex
)
expected_output.extend(output_lines)

starting_lineno = raw_test["__line__"]
extra_environment_variables = parse_environment_variables(raw_test.get("env", []))
disable_cache = raw_test.get("disable_cache", False)
expected_output_lines = pystache.render(raw_test.get("out", ""), params).split("\n")
expected_output.extend(
utils.extract_output_matchers_from_out(raw_test.get("out", ""), params, regex=regex)
)
additional_mypy_config = raw_test.get("mypy_config", "")

skip = self._eval_skip(str(raw_test.get("skip", "False")))
Expand All @@ -122,7 +127,7 @@ def collect(self) -> Iterator["YamlTestItem"]:
starting_lineno=starting_lineno,
environment_variables=extra_environment_variables,
disable_cache=disable_cache,
expected_output_lines=output_from_comments + expected_output_lines,
expected_output=expected_output,
parsed_test_data=raw_test,
mypy_config=additional_mypy_config,
)
Expand Down
9 changes: 5 additions & 4 deletions pytest_mypy_plugins/item.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,9 @@
from pytest_mypy_plugins import utils
from pytest_mypy_plugins.collect import File, YamlTestFile
from pytest_mypy_plugins.utils import (
OutputMatcher,
TypecheckAssertionError,
assert_string_arrays_equal,
assert_expected_matched_actual,
capture_std_streams,
fname_to_module,
)
Expand Down Expand Up @@ -124,7 +125,7 @@ def __init__(
*,
files: List[File],
starting_lineno: int,
expected_output_lines: List[str],
expected_output: List[OutputMatcher],
environment_variables: Dict[str, Any],
disable_cache: bool,
mypy_config: str,
Expand All @@ -134,7 +135,7 @@ def __init__(
self.files = files
self.environment_variables = environment_variables
self.disable_cache = disable_cache
self.expected_output_lines = expected_output_lines
self.expected_output = expected_output
self.starting_lineno = starting_lineno
self.additional_mypy_config = mypy_config
self.parsed_test_data = parsed_test_data
Expand Down Expand Up @@ -279,7 +280,7 @@ def runtest(self) -> None:
for line in mypy_output.splitlines():
output_line = replace_fpath_with_module_name(line, rootdir=execution_path)
output_lines.append(output_line)
assert_string_arrays_equal(expected=self.expected_output_lines, actual=output_lines)
assert_expected_matched_actual(expected=self.expected_output, actual=output_lines)
finally:
temp_dir.cleanup()
# remove created modules and all their dependants from cache
Expand Down
16 changes: 16 additions & 0 deletions pytest_mypy_plugins/tests/test-regex_assertions.shouldfail.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
- case: rexex_but_not_turned_on
main: |
a = 'hello'
reveal_type(a) # N: .*str.*

- case: rexex_but_turned_off
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's add one more test that should fail: with wrong regex. Something that does not match the output 👍

regex: no
main: |
a = 'hello'
reveal_type(a) # N: .*str.*

- case: regext_does_not_match
regex: no
main: |
a = 'hello'
reveal_type(a) # NR: .*banana.*
22 changes: 22 additions & 0 deletions pytest_mypy_plugins/tests/test-regex_assertions.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
- case: expected_message_regex
regex: yes
main: |
a = 1
b = 'hello'

reveal_type(a) # N: Revealed type is "builtins.int"
reveal_type(b) # N: .*str.*

- case: expected_message_regex_with_out
Copy link
Member

@sobolevn sobolevn Aug 7, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We also need a test without regex attribute and with regex: no where we try to use regex output

Copy link
Contributor Author

@brunns brunns Aug 9, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure how to do negative tests - ie tests where we expect a failure. Are there any examples of this kind of test?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For example, if I added this test:

- case: should_fail
  regex: no
  main: |
    a = 'hello'
    reveal_type(a)  # N: .*str.*

I'd expect the test to fail - but that fails the build. Is there a way to express that I expect this to fail?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nope, see #46

The easiest way for now is to add all failing tests into a single non-collectable file. And run it with pytest after the main test case. We can grep that failed X tests

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you approve me to run workflows? I have something which works locally, but I'd like to see if it works in CI.

regex: yes
main: |
a = 'abc'
reveal_type(a)
out: |
main:2: note: .*str.*

- case: expected_single_message_regex
regex: no
main: |
a = 'hello'
reveal_type(a) # NR: .*str.*
12 changes: 12 additions & 0 deletions pytest_mypy_plugins/tests/test-simple-cases.shouldfail.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
- case: fail_if_message_does_not_match
main: |
a = 'hello'
reveal_type(a) # N: Some other message

- case: fail_if_message_from_outdoes_not_match
regex: yes
main: |
a = 'abc'
reveal_type(a)
out: |
main:2: note: Some other message
Loading