Consider I have a special object which may hold a literal json string, that I intend to use as a field in a larger JSON object, as the literal value itself (not a string containing the JSON).
I want to write a custom encoder that can accomplish this, ie:
> encoder.encode({
> 'a': LiteralJson('{}')
> })
{"a": {}}
I don't believe subclassing JSONEncoder and overriding default will work, because at best there, I can return the string, which would make the result {"a": "{}"}
.
Overriding encode also appears not to work when the LiteralJson is nested somewhere inside another dictionary.
The background for this, if you are interested, is that I am storing JSON-encoded values in a cache, and it seems to me to be a waste to deserialize then reserialize all the time. It works that way, but some of these values are fairly long and it just seems like a huge waste.
The following encoder would accomplish what I like (but seems unnecessarily slow):
class MagicEncoder(json.JSONEncoder):
def default(self, obj):
if isinstance(obj, LiteralJson):
return json.loads(obj.content)
else:
return json.JSONEncoder.default(self, obj)
I've just realised I had a similar question recently. The answer suggested to use a replacement token.
It's possible to integrate this logic more or less transparently using a custom JSONEncoder
that generates these tokens internally using a random UUID. (What I've called "RawJavaScriptText
" is the equivalent of your "LiteralJson
".)
You can then use json.dumps(testvar, cls=RawJsJSONEncoder)
directly.
import json
import uuid
class RawJavaScriptText:
def __init__(self, jstext):
self._jstext = jstext
def get_jstext(self):
return self._jstext
class RawJsJSONEncoder(json.JSONEncoder):
def __init__(self, *args, **kwargs):
json.JSONEncoder.__init__(self, *args, **kwargs)
self._replacement_map = {}
def default(self, o):
if isinstance(o, RawJavaScriptText):
key = uuid.uuid4().hex
self._replacement_map[key] = o.get_jstext()
return key
else:
return json.JSONEncoder.default(self, o)
def encode(self, o):
result = json.JSONEncoder.encode(self, o)
for k, v in self._replacement_map.iteritems():
result = result.replace('"%s"' % (k,), v)
return result
testvar = {
'a': 1,
'b': 'abc',
'c': RawJavaScriptText('{ "x": [ 1, 2, 3 ] }')
}
print json.dumps(testvar, cls=RawJsJSONEncoder)
Result (using Python 2.6 and 2.7):
{"a": 1, "c": { "x": [ 1, 2, 3 ] }, "b": "abc"}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With