Skip to content

Commit c142c6c

Browse files
authored
python: use repr for everything except str and bytes (#378)
2 parents 405b886 + eb58152 commit c142c6c

File tree

8 files changed

+72
-101
lines changed

8 files changed

+72
-101
lines changed

python/example-pytest-selfie/tests/simple_inline_test.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,8 @@
1010

1111

1212
def test_write():
13-
expect_selfie("B").to_be_TODO()
13+
expect_selfie("B").to_be("B")
14+
expect_selfie(20000).to_be(20_000)
15+
expect_selfie([1, 2, 3]).to_be([1, 2, 3])
16+
expect_selfie(("a", 2, 3)).to_be(("a", 2, 3))
17+
expect_selfie({"a": 1, "b": 2}).to_be({"a": 1, "b": 2})
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
{
2+
"folders": [
3+
{
4+
"path": "example-pytest-selfie"
5+
},
6+
{
7+
"path": "pytest-selfie"
8+
},
9+
{
10+
"path": "selfie-lib"
11+
}
12+
],
13+
"settings": {}
14+
}

python/selfie-lib/selfie_lib/CacheSelfie.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -23,10 +23,10 @@ def to_match_disk_TODO(self, sub: str = "") -> T:
2323
return self._to_match_disk_impl(sub, True)
2424

2525
def _to_match_disk_impl(self, sub: str, is_todo: bool) -> T:
26-
from .Selfie import get_system
26+
from .Selfie import _selfieSystem
2727

2828
call = recordCall(False)
29-
system = get_system()
29+
system = _selfieSystem()
3030
if system.mode.can_write(is_todo, call, system):
3131
actual = self.generator()
3232
self.disk.write_disk(
@@ -55,10 +55,10 @@ def to_be(self, expected: str) -> T:
5555
return self._to_be_impl(expected)
5656

5757
def _to_be_impl(self, snapshot: Optional[str]) -> T:
58-
from .Selfie import get_system
58+
from .Selfie import _selfieSystem
5959

6060
call = recordCall(False)
61-
system = get_system()
61+
system = _selfieSystem()
6262
writable = system.mode.can_write(snapshot is None, call, system)
6363
if writable:
6464
actual = self.generator()

python/selfie-lib/selfie_lib/Literals.py

+9-13
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
from calendar import c
22
from enum import Enum, auto
3-
from typing import Protocol, TypeVar
3+
from typing import Any, Protocol, TypeVar
44
from abc import abstractmethod
5+
56
from .EscapeLeadingWhitespace import EscapeLeadingWhitespace
67
import io
78
import re
@@ -247,22 +248,17 @@ def handle_escape_sequences(line: str) -> str:
247248
)
248249

249250

250-
class LiteralBoolean(LiteralFormat[bool]):
251+
class LiteralRepr(LiteralFormat[Any]):
251252
def encode(
252-
self, value: bool, language: Language, encoding_policy: EscapeLeadingWhitespace
253+
self, value: Any, language: Language, encoding_policy: EscapeLeadingWhitespace
253254
) -> str:
254-
return str(value)
255-
256-
def __to_boolean_strict(self, string: str) -> bool:
257-
if string.lower() == "true":
258-
return True
259-
elif string.lower() == "false":
260-
return False
255+
if isinstance(value, int):
256+
return LiteralInt().encode(value, language, encoding_policy)
261257
else:
262-
raise ValueError("String is not a valid boolean representation: " + string)
258+
return repr(value)
263259

264-
def parse(self, string: str, language: Language) -> bool:
265-
return self.__to_boolean_strict(string)
260+
def parse(self, string: str, language: Language) -> Any:
261+
return eval(string)
266262

267263

268264
class TodoStub(Enum):
+19-13
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
from typing import TypeVar, Generic, Protocol, Union
2-
from .SelfieImplementations import StringSelfie, IntSelfie, BooleanSelfie
3-
from .SnapshotSystem import _selfieSystem, SnapshotSystem
1+
from typing import TypeVar, Any, Protocol, Union, overload
2+
from .SelfieImplementations import ReprSelfie, StringSelfie
3+
from .SnapshotSystem import _selfieSystem
44
from .Snapshot import Snapshot
55
from .CacheSelfie import CacheSelfie
66
from .RoundTrip import Roundtrip
@@ -16,21 +16,27 @@ def __call__(self) -> T:
1616
raise NotImplementedError
1717

1818

19-
def get_system() -> SnapshotSystem:
20-
return _selfieSystem()
19+
@overload
20+
def expect_selfie(actual: str) -> StringSelfie: ...
2121

2222

23-
def expect_selfie(actual: Union[str, int, bool]):
24-
if isinstance(actual, int):
25-
return IntSelfie(actual)
26-
elif isinstance(actual, str):
23+
# @overload
24+
# def expect_selfie(actual: bytes) -> BinarySelfie: ...
25+
26+
27+
@overload
28+
def expect_selfie[T](actual: T) -> ReprSelfie[T]: ...
29+
30+
31+
def expect_selfie(
32+
actual: Union[str, Any],
33+
) -> Union[StringSelfie, ReprSelfie]:
34+
if isinstance(actual, str):
2735
snapshot = Snapshot.of(actual)
2836
diskStorage = _selfieSystem().disk_thread_local()
2937
return StringSelfie(snapshot, diskStorage)
30-
elif isinstance(actual, bool):
31-
return BooleanSelfie(actual)
3238
else:
33-
raise NotImplementedError()
39+
return ReprSelfie(actual)
3440

3541

3642
def cache_selfie_string(to_cache: Cacheable[str]) -> CacheSelfie[str]:
@@ -43,5 +49,5 @@ def cache_selfie_generic(
4349
roundtrip: Roundtrip[T, str], to_cache: Cacheable[T]
4450
) -> CacheSelfie[T]:
4551
"""Create a CacheSelfie instance for caching generic objects with specified roundtrip."""
46-
deferred_disk_storage = get_system().disk_thread_local()
52+
deferred_disk_storage = _selfieSystem().disk_thread_local()
4753
return CacheSelfie(deferred_disk_storage, roundtrip, to_cache)

python/selfie-lib/selfie_lib/SelfieImplementations.py

+20-43
Original file line numberDiff line numberDiff line change
@@ -5,20 +5,33 @@
55
from .SnapshotSystem import DiskStorage, SnapshotSystem, _selfieSystem, Mode
66
from .WriteTracker import recordCall as recordCall
77
from .Literals import (
8-
LiteralValue,
9-
LiteralString,
108
LiteralFormat,
9+
LiteralRepr,
10+
LiteralString,
11+
LiteralValue,
1112
TodoStub,
12-
LiteralInt,
13-
LiteralBoolean,
1413
)
1514

1615

1716
from abc import ABC, abstractmethod
18-
from typing import Any, List, Optional, Union
17+
from typing import Any, List, Optional
1918
from itertools import chain
2019

2120

21+
class ReprSelfie[T]:
22+
def __init__(self, actual: T):
23+
self.actual = actual
24+
25+
def to_be_TODO(self, unused_arg: Optional[T] = None) -> T:
26+
return _toBeDidntMatch(None, self.actual, LiteralRepr())
27+
28+
def to_be(self, expected: T) -> T:
29+
if self.actual == expected:
30+
return _checkSrc(self.actual)
31+
else:
32+
return _toBeDidntMatch(expected, self.actual, LiteralRepr())
33+
34+
2235
class FluentFacet(ABC):
2336
@abstractmethod
2437
def facet(self, facet: str) -> "StringFacet":
@@ -95,7 +108,7 @@ def facet_binary(self, facet: str) -> "BinaryFacet":
95108
raise NotImplementedError()
96109

97110

98-
class StringSelfie(DiskSelfie, StringFacet):
111+
class StringSelfie(DiskSelfie, StringFacet, ReprSelfie[str]):
99112
def __init__(
100113
self,
101114
actual: Snapshot,
@@ -147,7 +160,7 @@ def __actual(self) -> str:
147160
def to_be_TODO(self, unused_arg: Any = None) -> str:
148161
return _toBeDidntMatch(None, self.__actual(), LiteralString())
149162

150-
def to_be(self, expected: Union[str, int, bool]) -> str:
163+
def to_be(self, expected: str) -> str:
151164
actual_string = self.__actual()
152165

153166
# Check if expected is a string
@@ -234,39 +247,3 @@ def _serializeOnlyFacets(snapshot: Snapshot, keys: List[str]) -> str:
234247
return writer_str[len(EMPTY_KEY_AND_FACET) : -1]
235248
else:
236249
return writer_str[:-1]
237-
238-
239-
class IntSelfie:
240-
def __init__(self, actual: int):
241-
self.actual = actual
242-
243-
def to_be_TODO(self, unused_arg: Any = None):
244-
return _toBeDidntMatch(None, self.actual, LiteralInt())
245-
246-
def to_be(self, expected: Union[str, int, bool]) -> int:
247-
if not isinstance(expected, int):
248-
raise TypeError("Expected value must be an int")
249-
250-
# Compare actual to expected; handle match or mismatch.
251-
if self.actual == expected:
252-
return _checkSrc(self.actual)
253-
else:
254-
return _toBeDidntMatch(expected, self.actual, LiteralInt())
255-
256-
257-
class BooleanSelfie:
258-
def __init__(self, actual: bool):
259-
self.actual = actual
260-
261-
def to_be_TODO(self, unused_arg: Any = None):
262-
return _toBeDidntMatch(None, self.actual, LiteralBoolean())
263-
264-
def to_be(self, expected: Union[str, int, bool]) -> bool:
265-
if not isinstance(expected, bool):
266-
raise TypeError("Expected value must be a bool")
267-
268-
# Compare actual to expected; handle match or mismatch.
269-
if self.actual == expected:
270-
return _checkSrc(self.actual)
271-
else:
272-
return _toBeDidntMatch(expected, self.actual, LiteralBoolean())

python/selfie-lib/tests/LiteralBoolean_test.py

-26
This file was deleted.

selfie.dev/src/pages/py/get-started.mdx

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ To start snapshot testing in Python, all you need is to add a [single dependency
1818
Selfie snapshot testing works with the following Python test runners:
1919

2020
- Pytest
21-
- PRs welcome for other test runners
21+
- PRs welcome for other test runners (see [here](https://github.com/diffplug/selfie/issues/350) for a guide)
2222

2323
Both disk and inline snapshots can be used with Python test code.
2424

0 commit comments

Comments
 (0)