| # ISC License | |
| # | |
| # Copyright (c) 2018-2025, Andrea Giammarchi, @WebReflection | |
| # | |
| # Permission to use, copy, modify, and/or distribute this software for any | |
| # purpose with or without fee is hereby granted, provided that the above | |
| # copyright notice and this permission notice appear in all copies. | |
| # | |
| # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH | |
| # REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY | |
| # AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, | |
| # INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM | |
| # LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE | |
| # OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR | |
| # PERFORMANCE OF THIS SOFTWARE. | |
| 
 | |
| import json as _json | |
| 
 | |
| class _Known: | |
|     def __init__(self): | |
|         self.key = [] | |
|         self.value = [] | |
| 
 | |
| class _String: | |
|     def __init__(self, value): | |
|         self.value = value | |
| 
 | |
| 
 | |
| def _array_keys(value): | |
|     keys = [] | |
|     i = 0 | |
|     for _ in value: | |
|         keys.append(i) | |
|         i += 1 | |
|     return keys | |
| 
 | |
| def _object_keys(value): | |
|     keys = [] | |
|     for key in value: | |
|         keys.append(key) | |
|     return keys | |
| 
 | |
| def _is_array(value): | |
|     return isinstance(value, (list, tuple)) | |
| 
 | |
| def _is_object(value): | |
|     return isinstance(value, dict) | |
| 
 | |
| def _is_string(value): | |
|     return isinstance(value, str) | |
| 
 | |
| def _index(known, input, value): | |
|     input.append(value) | |
|     index = str(len(input) - 1) | |
|     known.key.append(value) | |
|     known.value.append(index) | |
|     return index | |
| 
 | |
| def _loop(keys, input, known, output): | |
|     for key in keys: | |
|         value = output[key] | |
|         if isinstance(value, _String): | |
|             _ref(key, input[int(value.value)], input, known, output) | |
| 
 | |
|     return output | |
| 
 | |
| def _ref(key, value, input, known, output): | |
|     if _is_array(value) and value not in known: | |
|         known.append(value) | |
|         value = _loop(_array_keys(value), input, known, value) | |
|     elif _is_object(value) and value not in known: | |
|         known.append(value) | |
|         value = _loop(_object_keys(value), input, known, value) | |
| 
 | |
|     output[key] = value | |
| 
 | |
| def _relate(known, input, value): | |
|     if _is_string(value) or _is_array(value) or _is_object(value): | |
|         try: | |
|             return known.value[known.key.index(value)] | |
|         except: | |
|             return _index(known, input, value) | |
| 
 | |
|     return value | |
| 
 | |
| def _transform(known, input, value): | |
|     if _is_array(value): | |
|         output = [] | |
|         for val in value: | |
|             output.append(_relate(known, input, val)) | |
|         return output | |
| 
 | |
|     if _is_object(value): | |
|         obj = {} | |
|         for key in value: | |
|             obj[key] = _relate(known, input, value[key]) | |
|         return obj | |
| 
 | |
|     return value | |
| 
 | |
| def _wrap(value): | |
|     if _is_string(value): | |
|         return _String(value) | |
| 
 | |
|     if _is_array(value): | |
|         i = 0 | |
|         for val in value: | |
|             value[i] = _wrap(val) | |
|             i += 1 | |
| 
 | |
|     elif _is_object(value): | |
|         for key in value: | |
|             value[key] = _wrap(value[key]) | |
| 
 | |
|     return value | |
| 
 | |
| def parse(value, *args, **kwargs): | |
|     json = _json.loads(value, *args, **kwargs) | |
|     wrapped = [] | |
|     for value in json: | |
|         wrapped.append(_wrap(value)) | |
| 
 | |
|     input = [] | |
|     for value in wrapped: | |
|         if isinstance(value, _String): | |
|             input.append(value.value) | |
|         else: | |
|             input.append(value) | |
| 
 | |
|     value = input[0] | |
| 
 | |
|     if _is_array(value): | |
|         return _loop(_array_keys(value), input, [value], value) | |
| 
 | |
|     if _is_object(value): | |
|         return _loop(_object_keys(value), input, [value], value) | |
| 
 | |
|     return value | |
| 
 | |
| 
 | |
| def stringify(value, *args, **kwargs): | |
|     known = _Known() | |
|     input = [] | |
|     output = [] | |
|     i = int(_index(known, input, value)) | |
|     while i < len(input): | |
|         output.append(_transform(known, input, input[i])) | |
|         i += 1 | |
|     return _json.dumps(output, *args, **kwargs)
 |