[tor-commits] [sbws/maint-1.1] chg: json: Create custom JSON encoder/decoder
juga at torproject.org
juga at torproject.org
Tue Apr 14 13:53:19 UTC 2020
commit f19738f1c24316399c0928f62130ec133cee5db0
Author: juga0 <juga at riseup.net>
Date: Sat Mar 14 16:52:21 2020 +0000
chg: json: Create custom JSON encoder/decoder
to be able to serialize/deserailize datetime in the state file.
---
sbws/util/json.py | 44 ++++++++++++++++++++++++++++++++++++
tests/unit/util/test_json.py | 54 ++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 98 insertions(+)
diff --git a/sbws/util/json.py b/sbws/util/json.py
new file mode 100644
index 0000000..f4f8130
--- /dev/null
+++ b/sbws/util/json.py
@@ -0,0 +1,44 @@
+"""JSON custom serializers and deserializers."""
+import datetime
+import json
+
+from .timestamps import DateTimeSeq, DateTimeIntSeq
+
+
+class CustomEncoder(json.JSONEncoder):
+ """JSONEncoder that serializes datetime to ISO 8601 string."""
+
+ def default(self, obj):
+ if isinstance(obj, DateTimeSeq) or isinstance(obj, DateTimeIntSeq):
+ return [self.default(i) for i in obj.list()]
+ if isinstance(obj, datetime.datetime):
+ return obj.replace(microsecond=0).isoformat()
+ else:
+ return super().default(obj)
+
+
+class CustomDecoder(json.JSONDecoder):
+ """JSONDecoder that deserializes ISO 8601 string to datetime."""
+
+ def decode(self, s, **kwargs):
+ decoded = super().decode(s, **kwargs)
+ return self.process(decoded)
+
+ def process(self, obj):
+ if isinstance(obj, list) and obj:
+ return [self.process(item) for item in obj]
+ if isinstance(obj, dict):
+ return {key: self.process(value) for key, value in obj.items()}
+ if isinstance(obj, str):
+ try:
+ return datetime.datetime.strptime(obj, "%Y-%m-%dT%H:%M:%S")
+ except ValueError:
+ try:
+ datetime.datetime.strptime(
+ obj, "%Y-%m-%dT%H:%M:%S.%f"
+ ).replace(microsecond=0)
+ except ValueError:
+ pass
+ except TypeError:
+ pass
+ return obj
diff --git a/tests/unit/util/test_json.py b/tests/unit/util/test_json.py
new file mode 100644
index 0000000..05e0bf3
--- /dev/null
+++ b/tests/unit/util/test_json.py
@@ -0,0 +1,54 @@
+"""json.py unit tests."""
+import json
+
+from sbws.util.json import CustomDecoder, CustomEncoder
+
+STATE = """{
+ "min_perc_reached": null,
+ "recent_consensus_count": [
+ "2020-03-04T10:00:00",
+ "2020-03-05T10:00:00",
+ "2020-03-06T10:00:00"
+ ],
+ "recent_measurement_attempt": [
+ [
+ "2020-03-04T10:00:00",
+ 2
+ ],
+ [
+ "2020-03-05T10:00:00",
+ 2
+ ],
+ [
+ "2020-03-06T10:00:00",
+ 2
+ ]
+ ],
+ "recent_priority_list": [
+ "2020-03-04T10:00:00",
+ "2020-03-05T10:00:00",
+ "2020-03-06T10:00:00"
+ ],
+ "recent_priority_relay": [
+ [
+ "2020-03-04T10:00:00",
+ 2
+ ],
+ [
+ "2020-03-05T10:00:00",
+ 2
+ ],
+ [
+ "2020-03-06T10:00:00",
+ 2
+ ]
+ ],
+ "scanner_started": "2020-03-14T16:15:22",
+ "uuid": "x"
+}"""
+
+
+def test_decode_encode_roundtrip():
+ d = json.loads(STATE, cls=CustomDecoder)
+ s = json.dumps(d, cls=CustomEncoder, indent=4, sort_keys=True)
+ assert s == STATE
More information about the tor-commits
mailing list