From e6f14d5a44999b73223194616b9815a11f19557b 2013-11-25 19:55:34 From: MinRK Date: 2013-11-25 19:55:34 Subject: [PATCH] re-cast int/float subclasses to int/float in json_clean Protects against custom `__str__` in int subclasses (e.g. enums). Works around a [bug in the stdlib](http://bugs.python.org/issue12657). closes #4598 --- diff --git a/IPython/utils/jsonutil.py b/IPython/utils/jsonutil.py index 565b1bc..6de6709 100644 --- a/IPython/utils/jsonutil.py +++ b/IPython/utils/jsonutil.py @@ -194,9 +194,8 @@ def json_clean(obj): >>> json_clean(True) True """ - # types that are 'atomic' and ok in json as-is. bool doesn't need to be - # listed explicitly because bools pass as int instances - atomic_ok = (unicode_type, int, type(None)) + # types that are 'atomic' and ok in json as-is. + atomic_ok = (unicode_type, type(None)) # containers that we need to convert into lists container_to_list = (tuple, set, types.GeneratorType) @@ -205,7 +204,14 @@ def json_clean(obj): # cast out-of-range floats to their reprs if math.isnan(obj) or math.isinf(obj): return repr(obj) - return obj + return float(obj) + + if isinstance(obj, int): + # cast int to int, in case subclasses override __str__ (e.g. boost enum, #4598) + if isinstance(obj, bool): + # bools are ints, but we don't want to cast them to 0,1 + return obj + return int(obj) if isinstance(obj, atomic_ok): return obj