forked from WolframResearch/WolframClientForPython
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbase.py
186 lines (144 loc) · 6.05 KB
/
base.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
from __future__ import absolute_import, print_function, unicode_literals
import datetime
import inspect
import re
from itertools import chain
from wolframclient.serializers.encoder import Encoder
from wolframclient.serializers.wxfencoder.constants import WXF_HEADER_SEPARATOR, WXF_VERSION
from wolframclient.serializers.wxfencoder.utils import (
numeric_array_to_wxf,
packed_array_to_wxf,
)
from wolframclient.utils import six
from wolframclient.utils.encoding import concatenate_bytes, force_text
from wolframclient.utils.functional import first
if hasattr(inspect, "getfullargspec"):
inspect_args = inspect.getfullargspec
elif hasattr(inspect, "getargspec"):
inspect_args = inspect.getargspec
else:
def inspect_args(f):
raise TypeError()
class FormatSerializer(Encoder):
def generate_bytes(self, data):
raise NotImplementedError
def export(self, data, stream=None):
if stream:
if isinstance(stream, six.string_types):
with open(stream, "wb") as file:
for token in self.generate_bytes(data):
file.write(token)
return stream
for token in self.generate_bytes(data):
stream.write(token)
return stream
return concatenate_bytes(self.generate_bytes(data))
# implementation of several methods
def serialize_function(self, head, args, **opts):
raise NotImplementedError
def serialize_symbol(self, symbol):
raise NotImplementedError
def serialize_string(self, obj):
raise NotImplementedError
def serialize_float(self, obj):
raise NotImplementedError
def serialize_decimal(self, obj):
raise NotImplementedError
def serialize_int(self, obj):
raise NotImplementedError
def serialize_bytes(self, bytes, as_byte_array=not six.PY2):
raise NotImplementedError
def serialize_input_form(self, string):
return self.serialize_function(
self.serialize_symbol(b"ToExpression"), (self.serialize_string(string),)
)
def _serialize_as_wxf(self, data, shape, wl_type, constructor):
payload = concatenate_bytes(
chain((WXF_VERSION, WXF_HEADER_SEPARATOR), constructor(data, shape, wl_type))
)
return self.serialize_function(
self.serialize_symbol(b"BinaryDeserialize"),
(self.serialize_bytes(payload, as_byte_array=True),),
)
def serialize_numeric_array(self, data, shape, wl_type):
return self._serialize_as_wxf(data, shape, wl_type, numeric_array_to_wxf)
def serialize_packed_array(self, data, shape, wl_type):
return self._serialize_as_wxf(data, shape, wl_type, packed_array_to_wxf)
def serialize_iterable(self, iterable, **opts):
return self.serialize_function(self.serialize_symbol(b"List"), iterable, **opts)
def serialize_mapping(self, mappable, **opts):
return self.serialize_function(
self.serialize_symbol(b"Association"),
(self.serialize_rule(key, value) for key, value in mappable),
**opts
)
def serialize_association(self, mappable, **opts):
return self.serialize_function(
self.serialize_symbol(b"Association"),
(self.serialize_rule(key, value) for key, value in mappable),
**opts
)
def serialize_fraction(self, o):
return self.serialize_function(
self.serialize_symbol(b"Rational"),
(self.serialize_int(o.numerator), self.serialize_int(o.denominator)),
)
def serialize_complex(self, o):
return self.serialize_function(
self.serialize_symbol(b"Complex"),
(self.serialize_float(o.real), self.serialize_float(o.imag)),
)
def serialize_rule(self, lhs, rhs):
return self.serialize_function(self.serialize_symbol(b"Rule"), (lhs, rhs))
def serialize_rule_delayed(self, lhs, rhs):
return self.serialize_function(self.serialize_symbol(b"RuleDelayed"), (lhs, rhs))
def serialize_tzinfo(
self, tzinfo, date=None, name_match=re.compile("^([A-Za-z]+/[A-Za-z]+?|UTC)$")
):
if tzinfo is None:
return self.serialize_symbol(
self.target_kernel_version >= 12 and b"None" or b"$TimeZone"
)
if name_match:
name = tzinfo.tzname(None)
if name and name_match.match(name):
return self.serialize_string(name)
return self.serialize_float(
tzinfo.utcoffset(date or datetime.datetime.utcnow()).total_seconds() / 3600
)
def _serialize_external_object(self, o):
yield "System", "Python"
if callable(o):
yield "Type", "PythonFunction"
try:
# force tuple to avoid calling this method again on `map`.
yield "Arguments", tuple(map(force_text, first(inspect_args(o))))
except TypeError:
# this function can fail with TypeError unsupported callable
pass
else:
yield "Type", "PythonObject"
if hasattr(o, "__name__"):
yield "Command", force_text(o.__name__)
else:
yield "Command", force_text(o.__class__.__name__)
is_module = inspect.ismodule(o)
yield "IsModule", is_module
if not is_module:
module = inspect.getmodule(o)
if module:
yield "Module", force_text(module.__name__)
yield "IsClass", inspect.isclass(o),
yield "IsFunction", inspect.isfunction(o),
yield "IsMethod", inspect.ismethod(o),
yield "IsCallable", callable(o),
def serialize_external_object(self, obj):
return self.serialize_function(
self.serialize_symbol(callable(obj) and b"ExternalFunction" or b"ExternalObject"),
(
self.serialize_mapping(
(self.encode(key), self.encode(value))
for key, value in self._serialize_external_object(obj)
),
),
)