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

Add hypothesis-powered tests #469

Closed
wants to merge 28 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
5a45e64
Add test dependency on hypothesis
hexagonrecursion Mar 1, 2021
df1c2dd
Start testing with hypothesis
hexagonrecursion Feb 24, 2021
1838bf1
Add integers to property based testing
hexagonrecursion Feb 24, 2021
cbfe680
Add floats to property based testing
hexagonrecursion Feb 24, 2021
521dc9b
Add str to property based testing
hexagonrecursion Feb 24, 2021
a681f7f
Add bytes to property based testing
hexagonrecursion Feb 24, 2021
c44d4c3
Add TODO
hexagonrecursion Feb 24, 2021
c68f247
Add list to to property based testing
hexagonrecursion Feb 24, 2021
7d52fcf
Add dict to property based testing
hexagonrecursion Feb 24, 2021
f36b286
Test all non-composite types as map keys
hexagonrecursion Feb 24, 2021
af9deb5
Fix style
hexagonrecursion Feb 24, 2021
fccea37
Add max_size
hexagonrecursion Feb 24, 2021
f05ec6e
Fix style
hexagonrecursion Feb 24, 2021
f9d7cad
Fix error: Data generation is extremely slow
hexagonrecursion Feb 24, 2021
c427cc1
Add test_roudtrip
hexagonrecursion Feb 24, 2021
c6c2670
This is now obsolete
hexagonrecursion Feb 24, 2021
fe50123
Clarify comment
hexagonrecursion Feb 24, 2021
3997376
Add TODO
hexagonrecursion Feb 24, 2021
151e8ad
Fix: Data generation is extremely slow
hexagonrecursion Mar 1, 2021
d2ee79f
Add test_extension_and_fallback_unpack_identically
hexagonrecursion Mar 1, 2021
59596d2
Blacken test/test_property_based.py
hexagonrecursion Mar 1, 2021
b6a4bfc
Use unpackb instead of unpacker
hexagonrecursion Mar 1, 2021
681f79f
Use unpackb instead of Unpacker
hexagonrecursion Mar 1, 2021
fd28b0c
Add test dependency: hypothesis
hexagonrecursion Mar 1, 2021
4e66cf1
Add test dependency: hypothesis
hexagonrecursion Mar 1, 2021
6f15eda
Fix: Data generation is extremely slow
hexagonrecursion Mar 1, 2021
aad7b96
Add test dependency: hypothesis
hexagonrecursion Mar 1, 2021
9622b83
Add test dependency: hypothesis
hexagonrecursion Mar 1, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 4 additions & 4 deletions .github/workflows/linux.yml
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ jobs:

- name: Run test (3.8)
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: --no-index -f dist/wheelhouse
pytest -v test

Expand All @@ -54,7 +54,7 @@ jobs:

- name: Run test (3.9)
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: --no-index -f dist/wheelhouse
pytest -v test

Expand All @@ -65,7 +65,7 @@ jobs:

- name: Run test (3.7)
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: --no-index -f dist/wheelhouse
pytest -v test

Expand All @@ -76,7 +76,7 @@ jobs:

- name: Run test (3.6)
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: --no-index -f dist/wheelhouse
pytest -v test

Expand Down
8 changes: 4 additions & 4 deletions .github/workflows/mac.yml
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ jobs:

- name: Run test
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: -f dist/ --no-index
pytest -v test

Expand All @@ -47,7 +47,7 @@ jobs:

- name: Run test
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: -f dist/ --no-index
pytest -v test

Expand All @@ -64,7 +64,7 @@ jobs:

- name: Run test
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: -f dist/ --no-index
pytest -v test

Expand All @@ -81,7 +81,7 @@ jobs:

- name: Run test
run: |
pip install pytest
pip install pytest hypothesis
pip install -v msgpack --only-binary :all: -f dist/ --no-index
pytest -v test

Expand Down
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,4 @@ msgpack/*.cpp
/tags
/docs/_build
.cache
.hypothesis
6 changes: 4 additions & 2 deletions .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ python:
_pure: &pure
install:
- pip install -U pip
- pip install -U pytest pytest-cov codecov
- pip install -U pytest pytest-cov codecov hypothesis
- pip install .
script:
- pytest --cov=msgpack -v test
Expand All @@ -33,6 +33,7 @@ matrix:
- DOCKER_IMAGE=quay.io/pypa/manylinux1_i686
install:
- pip install -U pip
- pip install -U hypothesis
- pip install -r requirements.txt
- make cython
- docker pull $DOCKER_IMAGE
Expand All @@ -48,6 +49,7 @@ matrix:
- DOCKER_IMAGE=quay.io/pypa/manylinux2014_aarch64
install:
- pip install -U pip
- pip install -U hypothesis
- pip install -r requirements.txt
- make cython
- docker pull $DOCKER_IMAGE
Expand All @@ -70,7 +72,7 @@ matrix:

install:
- pip install -U pip
- pip install -U pytest pytest-cov codecov
- pip install -U pytest pytest-cov codecov hypothesis
- pip install -r requirements.txt # Cython
- make cython
- pip install -e .
Expand Down
2 changes: 1 addition & 1 deletion ci/runtests.bat
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
%PYTHON%\python.exe -m pip install -U pip wheel pytest
%PYTHON%\python.exe -m pip install -U pip wheel pytest hypothesis
%PYTHON%\python.exe setup.py build_ext -i
%PYTHON%\python.exe setup.py install
%PYTHON%\python.exe -c "import sys; print(hex(sys.maxsize))"
Expand Down
2 changes: 1 addition & 1 deletion ci/runtests.sh
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#!/bin/bash
set -ex
${PYTHON} -VV
${PYTHON} -m pip install setuptools wheel pytest
${PYTHON} -m pip install setuptools wheel pytest hypothesis
${PYTHON} setup.py build_ext -if
${PYTHON} -c "from msgpack import _cmsgpack"
${PYTHON} setup.py bdist_wheel
Expand Down
2 changes: 1 addition & 1 deletion docker/runtests.sh
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ for V in "${PYTHON_VERSIONS[@]}"; do
PYBIN=/opt/python/$V/bin
$PYBIN/python setup.py install
rm -rf build/ # Avoid lib build by narrow Python is used by wide python
$PYBIN/pip install pytest
$PYBIN/pip install pytest hypothesis
pushd test # prevent importing msgpack package in current directory.
$PYBIN/python -c 'import sys; print(hex(sys.maxsize))'
$PYBIN/python -c 'from msgpack import _cmsgpack' # Ensure extension is available
Expand Down
84 changes: 84 additions & 0 deletions test/test_property_based.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
import pytest
from hypothesis import given, assume, strategies as st

import msgpack

try:
from msgpack import _cmsgpack
except ImportError:
_cmsgpack = None
from msgpack import fallback

HYPOTHESIS_MAX = 3

# https://github.com/msgpack/msgpack/blob/master/spec.md#type-system
# TODO: test timestamps
# TODO: test the extension type
simple_types = (
st.integers(min_value=-(2 ** 63), max_value=2 ** 64 - 1)
| st.none()
| st.booleans()
| st.floats()
# TODO: The msgpack speck says that string objects may contain invalid byte sequence
| st.text(max_size=HYPOTHESIS_MAX)
| st.binary(max_size=HYPOTHESIS_MAX)
)


def composite_types(any_type):
return st.lists(any_type, max_size=HYPOTHESIS_MAX) | st.dictionaries(
simple_types, any_type, max_size=HYPOTHESIS_MAX
)


any_type = st.recursive(simple_types, composite_types)


@pytest.mark.skipif(_cmsgpack is None, reason="C extension is not available")
@given(any_type)
def test_extension_and_fallback_pack_identically(obj):
extension_packer = _cmsgpack.Packer()
fallback_packer = fallback.Packer()

assert extension_packer.pack(obj) == fallback_packer.pack(obj)


# TODO: also test with strict_map_key=True
@pytest.mark.parametrize("impl", [fallback, _cmsgpack])
@given(obj=any_type)
def test_roudtrip(obj, impl):
if impl is None:
pytest.skip("C extension is not available")
packer = impl.Packer()
buf = packer.pack(obj)
got = impl.unpackb(buf, strict_map_key=False)
# using obj == got fails because NaN != NaN
assert repr(obj) == repr(got)


# TODO: also test with strict_map_key=True
@pytest.mark.skipif(_cmsgpack is None, reason="C extension is not available")
@given(st.binary(max_size=HYPOTHESIS_MAX))
def test_extension_and_fallback_unpack_identically(buf):
try:
from_extension = _cmsgpack.unpackb(buf)
except (msgpack.ExtraData, ValueError) as e:
# Ignore the exception message. This avoids:
#
# Falsifying example: buf=b'\x00\x00'
# Error: ExtraData(0, bytearray(b'\x00')) != ExtraData(0, b'\x00')
#
# Falsifying example: buf=b'\xa2'
# Error: ValueError('2 exceeds max_str_len(1)') != ValueError('Unpack failed: incomplete input')
# See https://github.com/msgpack/msgpack-python/pull/464
from_extension = type(e)
except Exception as e:
from_extension = e
try:
from_fallback = fallback.unpackb(buf)
except (msgpack.ExtraData, ValueError) as e:
from_fallback = type(e)
except Exception as e:
from_fallback = e

assert repr(from_extension) == repr(from_fallback)
3 changes: 3 additions & 0 deletions tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ isolated_build = true
[testenv]
deps=
pytest
hypothesis

changedir=test
commands=
Expand All @@ -23,6 +24,7 @@ setenv=
basepython=python2.7-x86
deps=
pytest
hypothesis

changedir=test
commands=
Expand All @@ -34,6 +36,7 @@ commands=
basepython=python3.4-x86
deps=
pytest
hypothesis

changedir=test
commands=
Expand Down