Source code for xsdata.formats.dataclass.parsers.json

import io
import json
from dataclasses import dataclass
from typing import Any
from typing import Dict
from typing import Type
from typing import TypeVar

from xsdata.exceptions import ParserError
from xsdata.formats.bindings import AbstractParser
from xsdata.formats.dataclass.mixins import ModelInspect
from xsdata.formats.dataclass.models import AnyElement
from xsdata.models.inspect import ClassVar

T = TypeVar("T")


[docs]@dataclass class JsonParser(AbstractParser, ModelInspect):
[docs] def parse(self, source: io.BytesIO, clazz: Type[T]) -> T: """Parse the JSON input stream and return the resulting object tree.""" ctx = json.load(source) return self.parse_context(ctx, clazz)
[docs] def parse_context(self, data: Dict, clazz: Type[T]) -> T: """ Recursively build the given model from the input dict data. :raise TypeError: When parsing fails for any reason """ params = {} if isinstance(data, list) and len(data) == 1: data = data[0] for var in self.class_meta(clazz).vars.values(): value = self.get_value(data, var) if value is None: continue elif var.is_list: params[var.name] = [self.bind_value(var, val) for val in value] else: params[var.name] = self.bind_value(var, value) try: return clazz(**params) # type: ignore except Exception: raise ParserError("Parsing failed")
[docs] def bind_value(self, var: ClassVar, value) -> Any: """ Bind value according to the class var. The return value can be: - a dataclass instance - a dictionary with unknown attributes - a list of unknown elements - an enumeration - a primitive value """ if var.dataclass: return self.parse_context(value, var.clazz) elif var.is_any_attribute: return dict(value) elif var.is_any_element: return ( value if isinstance(value, str) else self.parse_context(value, AnyElement) ) else: return self.parse_value(var.types, value, var.default)
[docs] @staticmethod def get_value(data: Dict, field: ClassVar): """Find the field value in the given dictionary or return the default field value.""" if field.qname.localname in data: value = data[field.qname.localname] elif field.name in data: value = data[field.name] else: return None if field.is_list and not isinstance(value, list): value = [value] return value