# This file is part of Hypothesis, which may be found at # https://github.com/HypothesisWorks/hypothesis/ # # Copyright the Hypothesis Authors. # Individual contributors are listed in AUTHORS.rst and the git log. # # This Source Code Form is subject to the terms of the Mozilla Public License, # v. 2.0. If a copy of the MPL was not distributed with this file, You can # obtain one at https://mozilla.org/MPL/2.0/. import math from collections import defaultdict, namedtuple from dataclasses import dataclass from functools import partial from inspect import Parameter, Signature, signature from typing import ForwardRef, Optional, Union import pytest from hypothesis.internal.compat import ( add_note, ceil, dataclass_asdict, extract_bits, floor, get_type_hints, ) floor_ceil_values = [ -10.7, -10.3, -0.5, -0.0, 0, 0.5, 10.3, 10.7, ] @pytest.mark.parametrize("value", floor_ceil_values) def test_our_floor_agrees_with_math_floor(value): assert floor(value) == math.floor(value) @pytest.mark.parametrize("value", floor_ceil_values) def test_our_ceil_agrees_with_math_ceil(value): assert ceil(value) == math.ceil(value) class WeirdSig: __signature__ = Signature( parameters=[Parameter(name="args", kind=Parameter.VAR_POSITIONAL)] ) def test_no_type_hints(): assert get_type_hints(WeirdSig) == {} @dataclass class Foo: x: "Foo" = None # type: ignore Foo.__signature__ = signature(Foo).replace( # type: ignore parameters=[ Parameter( "x", Parameter.POSITIONAL_OR_KEYWORD, annotation=ForwardRef("Foo"), default=None, ) ] ) @dataclass class Bar: x: Optional[Union[int, "Bar"]] Bar.__signature__ = signature(Bar).replace( # type: ignore parameters=[ Parameter( "x", Parameter.POSITIONAL_OR_KEYWORD, annotation=Optional[Union[int, ForwardRef("Bar")]], # type: ignore ) ] ) @pytest.mark.parametrize( "obj,expected", [(Foo, Optional[Foo]), (Bar, Optional[Union[int, Bar]])] ) def test_resolve_fwd_refs(obj, expected): # See: https://github.com/HypothesisWorks/hypothesis/issues/3519 assert get_type_hints(obj)["x"] == expected def func(a, b: int, *c: str, d: Optional[int] = None): pass @pytest.mark.parametrize( "pf, names", [ (partial(func, 1), "b c d"), (partial(func, 1, 2), "c d"), (partial(func, 1, 2, 3, 4, 5), "c d"), # varargs don't fill (partial(func, 1, 2, 3, d=4), "c d"), # kwonly args just get new defaults ], ) def test_get_hints_through_partial(pf, names): assert set(get_type_hints(pf)) == set(names.split()) @dataclass class FilledWithStuff: a: list b: tuple c: namedtuple d: dict e: defaultdict def test_dataclass_asdict(): ANamedTuple = namedtuple("ANamedTuple", ("with_some_field")) obj = FilledWithStuff(a=[1], b=(2), c=ANamedTuple(3), d={4: 5}, e=defaultdict(list)) assert dataclass_asdict(obj) == { "a": [1], "b": (2), "c": ANamedTuple(3), "d": {4: 5}, "e": {}, } @pytest.mark.parametrize("width", [None, 8]) @pytest.mark.parametrize("x", [0, 2, 123]) def test_extract_bits_roundtrip(width, x): bits = extract_bits(x, width=width) if width is not None: assert len(bits) == width assert x == sum(v << p for p, v in enumerate(reversed(bits))) @dataclass(frozen=True) class ImmutableError: msg: str def test_add_note_fails_gracefully_on_frozen_instance(): add_note(ImmutableError("msg"), "some note")