{
  "RepoName": "https://github.com/Infinidat/munch.git",
  "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27",
  "Time": "",
  "Difficulty": "Easy",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "Infinidat_munch/setup.py",
      "content": "from setuptools import setup\n\n\nsetup(\n    setup_requires=['pbr>=3.0', 'setuptools>=61'],\n    pbr=True,\n    long_description_content_type='text/markdown; charset=UTF-8',\n    keywords=['munch', 'dict', 'mapping', 'container', 'collection'],\n    python_requires=\">=3.6\",\n)\n"
    },
    {
      "path": "Infinidat_munch/munch/__init__.py",
      "content": "\"\"\" Munch is a subclass of dict with attribute-style access.\n\n    >>> b = Munch()\n    >>> b.hello = 'world'\n    >>> b.hello\n    'world'\n    >>> b['hello'] += \"!\"\n    >>> b.hello\n    'world!'\n    >>> b.foo = Munch(lol=True)\n    >>> b.foo.lol\n    True\n    >>> b.foo is b['foo']\n    True\n\n    It is safe to import * from this module:\n\n        __all__ = ('Munch', 'munchify','unmunchify')\n\n    un/munchify provide dictionary conversion; Munches can also be\n    converted via Munch.to/fromDict().\n\"\"\"\n\nfrom collections.abc import Mapping\n\ntry:\n    # For python 3.8 and later\n    import importlib.metadata as importlib_metadata\nexcept ImportError:\n    # For everyone else\n    import importlib_metadata\ntry:\n    __version__ = importlib_metadata.version(__name__)\nexcept importlib_metadata.PackageNotFoundError:\n    # package is not installed\n    __version__ = \"0.0.0\"\n\n\ntry:\n    VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\nexcept ValueError:\n    VERSION = (0, 0, 0)\n\n__all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n\n\n\nclass Munch(dict):\n    \"\"\" A dictionary that provides attribute-style access.\n\n        >>> b = Munch()\n        >>> b.hello = 'world'\n        >>> b.hello\n        'world'\n        >>> b['hello'] += \"!\"\n        >>> b.hello\n        'world!'\n        >>> b.foo = Munch(lol=True)\n        >>> b.foo.lol\n        True\n        >>> b.foo is b['foo']\n        True\n\n        A Munch is a subclass of dict; it supports all the methods a dict does...\n\n        >>> sorted(b.keys())\n        ['foo', 'hello']\n\n        Including update()...\n\n        >>> b.update({ 'ponies': 'are pretty!' }, hello=42)\n        >>> print (repr(b))\n        Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n        As well as iteration...\n\n        >>> sorted([ (k,b[k]) for k in b ])\n        [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n\n        And \"splats\".\n\n        >>> \"The {knights} who say {ni}!\".format(**Munch(knights='lolcats', ni='can haz'))\n        'The lolcats who say can haz!'\n\n        See unmunchify/Munch.toDict, munchify/Munch.fromDict for notes about conversion.\n    \"\"\"\n    def __init__(self, *args, **kwargs):  # pylint: disable=super-init-not-called\n        self.update(*args, **kwargs)\n\n    # only called if k not found in normal places\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise throws AttributeError.\n\n            nb. __getattr__ is only called if key is not found in normal places.\n\n            >>> b = Munch(bar='baz', lol={})\n            >>> b.foo\n            Traceback (most recent call last):\n                ...\n            AttributeError: foo\n\n            >>> b.bar\n            'baz'\n            >>> getattr(b, 'bar')\n            'baz'\n            >>> b['bar']\n            'baz'\n\n            >>> b.lol is b['lol']\n            True\n            >>> b.lol is getattr(b, 'lol')\n            True\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            return object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                return self[k]\n            except KeyError:\n                raise AttributeError(k)\n\n    def __setattr__(self, k, v):\n        \"\"\" Sets attribute k if it exists, otherwise sets key k. A KeyError\n            raised by set-item (only likely if you subclass Munch) will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(foo='bar', this_is='useful when subclassing')\n            >>> hasattr(b.values, '__call__')\n            True\n            >>> b.values = 'uh oh'\n            >>> b.values\n            'uh oh'\n            >>> b['values']\n            Traceback (most recent call last):\n                ...\n            KeyError: 'values'\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                self[k] = v\n            except:\n                raise AttributeError(k)\n        else:\n            object.__setattr__(self, k, v)\n\n    def __delattr__(self, k):\n        \"\"\" Deletes attribute k if it exists, otherwise deletes key k. A KeyError\n            raised by deleting the key--such as when the key is missing--will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(lol=42)\n            >>> del b.lol\n            >>> b.lol\n            Traceback (most recent call last):\n                ...\n            AttributeError: lol\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                del self[k]\n            except KeyError:\n                raise AttributeError(k)\n        else:\n            object.__delattr__(self, k)\n\n    def toDict(self):\n        \"\"\" Recursively converts a munch back into a dictionary.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> sorted(b.toDict().items())\n            [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n            See unmunchify for more info.\n        \"\"\"\n        return unmunchify(self)\n\n    @property\n    def __dict__(self):\n        return self.toDict()\n\n    def __repr__(self):\n        \"\"\" Invertible* string-form of a Munch.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> print (repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n            >>> eval(repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n            >>> with_spaces = Munch({1: 2, 'a b': 9, 'c': Munch({'simple': 5})})\n            >>> print (repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n            >>> eval(repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n\n            (*) Invertible so long as collection contents are each repr-invertible.\n        \"\"\"\n        return f'{self.__class__.__name__}({dict.__repr__(self)})'\n\n    def __dir__(self):\n        return list(self.keys())\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return {k: v for k, v in self.items()}\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        self.update(state)\n\n    __members__ = __dir__  # for python2.x compatibility\n\n    @classmethod\n    def fromDict(cls, d):\n        \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n            >>> b = Munch.fromDict({'urmom': {'sez': {'what': 'what'}}})\n            >>> b.urmom.sez.what\n            'what'\n\n            See munchify for more info.\n        \"\"\"\n        return munchify(d, cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n    def update(self, *args, **kwargs):\n        \"\"\"\n        Override built-in method to call custom __setitem__ method that may\n        be defined in subclasses.\n        \"\"\"\n        for k, v in dict(*args, **kwargs).items():\n            self[k] = v\n\n    def get(self, k, d=None):\n        \"\"\"\n        D.get(k[,d]) -> D[k] if k in D, else d.  d defaults to None.\n        \"\"\"\n        if k not in self:\n            return d\n        return self[k]\n\n    def setdefault(self, k, d=None):\n        \"\"\"\n        D.setdefault(k[,d]) -> D.get(k,d), also set D[k]=d if k not in D\n        \"\"\"\n        if k not in self:\n            self[k] = d\n        return self[k]\n\n\nclass AutoMunch(Munch):\n    def __setattr__(self, k, v):\n        \"\"\" Works the same as Munch.__setattr__ but if you supply\n            a dictionary as value it will convert it to another Munch.\n        \"\"\"\n        if isinstance(v, Mapping) and not isinstance(v, (AutoMunch, Munch)):\n            v = munchify(v, AutoMunch)\n        super().__setattr__(k, v)\n\n\nclass DefaultMunch(Munch):\n    \"\"\"\n    A Munch that returns a user-specified value for missing keys.\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        \"\"\" Construct a new DefaultMunch. Like collections.defaultdict, the\n            first argument is the default value; subsequent arguments are the\n            same as those for dict.\n        \"\"\"\n        # Mimic collections.defaultdict constructor\n        if args:\n            default = args[0]\n            args = args[1:]\n        else:\n            default = None\n        super().__init__(*args, **kwargs)\n        self.__default__ = default\n\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getattr__(k)\n        except AttributeError:\n            return self.__default__\n\n    def __setattr__(self, k, v):\n        if k == '__default__':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __getitem__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getitem__(k)\n        except KeyError:\n            return self.__default__\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return (self.__default__, {k: v for k, v in self.items()})\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        default, state_dict = state\n        self.update(state_dict)\n        self.__default__ = default\n\n    @classmethod\n    def fromDict(cls, d, default=None):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default=self.__default__)\n\n    def __repr__(self):\n        return f'{type(self).__name__}({self.__undefined__!r}, {dict.__repr__(self)})'\n\n\nclass DefaultFactoryMunch(Munch):\n    \"\"\" A Munch that calls a user-specified function to generate values for\n        missing keys like collections.defaultdict.\n\n        >>> b = DefaultFactoryMunch(list, {'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        []\n        >>> b.bar.append('hello')\n        >>> b.bar\n        ['hello']\n    \"\"\"\n\n    def __init__(self, default_factory, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.default_factory = default_factory\n\n    @classmethod\n    def fromDict(cls, d, default_factory):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default_factory, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default_factory=self.default_factory)\n\n    def __repr__(self):\n        factory = self.default_factory.__name__\n        return f'{type(self).__name__}({factory}, {dict.__repr__(self)})'\n\n    def __setattr__(self, k, v):\n        if k == 'default_factory':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __missing__(self, k):\n        self[k] = self.default_factory()\n        return self[k]\n\n\nclass RecursiveMunch(DefaultFactoryMunch):\n    \"\"\"A Munch that calls an instance of itself to generate values for\n        missing keys.\n\n        >>> b = RecursiveMunch({'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        RecursiveMunch(RecursiveMunch, {})\n        >>> b.bar.okay = 'hello'\n        >>> b.bar\n        RecursiveMunch(RecursiveMunch, {'okay': 'hello'})\n        >>> b\n        RecursiveMunch(RecursiveMunch, {'hello': 'world!', 'foo': RecursiveMunch(RecursiveMunch, {}),\n        'bar': RecursiveMunch(RecursiveMunch, {'okay': 'hello'})})\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(RecursiveMunch, *args, **kwargs)\n\n    @classmethod\n    def fromDict(cls, d):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n\n# While we could convert abstract types like Mapping or Iterable, I think\n# munchify is more likely to \"do what you mean\" if it is conservative about\n# casting (ex: isinstance(str,Iterable) == True ).\n#\n# Should you disagree, it is not difficult to duplicate this function with\n# more aggressive coercion to suit your own purposes.\n\ndef munchify(x, factory=Munch):\n    \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n        >>> b = munchify({'urmom': {'sez': {'what': 'what'}}})\n        >>> b.urmom.sez.what\n        'what'\n\n        munchify can handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = munchify({ 'lol': ('cats', {'hah':'i win again'}),\n        ...         'hello': [{'french':'salut', 'german':'hallo'}] })\n        >>> b.hello[0].french\n        'salut'\n        >>> b.lol[1].hah\n        'i win again'\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def munchify_cycles(obj):\n        partial, already_seen = pre_munchify_cycles(obj)\n        if already_seen:\n            return partial\n        return post_munchify(partial, obj)\n\n    def pre_munchify_cycles(obj):\n        # If we've already begun munchifying obj, just return the already-created munchified obj\n        try:\n            return seen[id(obj)], True\n        except KeyError:\n            pass\n\n        # Otherwise, first partly munchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_munchify(obj)\n        return partial, False\n\n    def pre_munchify(obj):\n        # Here we return a skeleton of munchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_munchify\n        if isinstance(obj, Mapping):\n            return factory({})\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(pre_munchify_cycles(item)[0] for item in obj)\n        else:\n            return obj\n\n    def post_munchify(partial, obj):\n        # Here we finish munchifying the parts of obj that were deferred by pre_munchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, munchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(munchify_cycles(item) for item in obj)\n        elif isinstance(obj, tuple):\n            for (item_partial, item) in zip(partial, obj):\n                post_munchify(item_partial, item)\n\n        return partial\n\n    return munchify_cycles(x)\n\n\ndef unmunchify(x):\n    \"\"\" Recursively converts a Munch into a dictionary.\n\n        >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n        >>> sorted(unmunchify(b).items())\n        [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n        unmunchify will handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42,\n        ...         ponies=('are pretty!', Munch(lies='are trouble!')))\n        >>> sorted(unmunchify(b).items()) #doctest: +NORMALIZE_WHITESPACE\n        [('foo', ['bar', {'lol': True}]), ('hello', 42), ('ponies', ('are pretty!', {'lies': 'are trouble!'}))]\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def unmunchify_cycles(obj):\n        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n        if id(obj) in seen.keys():\n            return seen[id(obj)]\n        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_unmunchify(obj)\n        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n        return post_unmunchify(partial, obj)\n\n    def pre_unmunchify(obj):\n        # Here we return a skeleton of unmunchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_unmunchify\n        if isinstance(obj, Mapping):\n            return dict()\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(unmunchify_cycles(item) for item in obj)\n        else:\n            return obj\n\n    def post_unmunchify(partial, obj):\n        # Here we finish unmunchifying the parts of obj that were deferred by pre_unmunchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, unmunchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(unmunchify_cycles(v) for v in obj)\n        elif isinstance(obj, tuple):\n            for (value_partial, value) in zip(partial, obj):\n                post_unmunchify(value_partial, value)\n\n        return partial\n\n    return unmunchify_cycles(x)\n\n\n# Serialization\n\ntry:\n    try:\n        import json\n    except ImportError:\n        import simplejson as json\n\n    def toJSON(self, **options):\n        \"\"\" Serializes this Munch to JSON. Accepts the same keyword options as `json.dumps()`.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> json.dumps(b) == b.toJSON()\n            True\n        \"\"\"\n        return json.dumps(self, **options)\n\n    def fromJSON(cls, stream, *args, **kwargs):\n        \"\"\" Deserializes JSON to Munch or any of its subclasses.\n        \"\"\"\n        factory = lambda d: cls(*(args + (d,)), **kwargs)\n        return munchify(json.loads(stream), factory=factory)\n\n    Munch.toJSON = toJSON\n    Munch.fromJSON = classmethod(fromJSON)\n\nexcept ImportError:\n    pass\n\n\ntry:\n    # Attempt to register ourself with PyYAML as a representer\n    import yaml\n    from yaml.representer import Representer, SafeRepresenter\n\n    def from_yaml(loader, node):\n        \"\"\" PyYAML support for Munches using the tag `!munch` and `!munch.Munch`.\n\n            >>> import yaml\n            >>> yaml.load('''\n            ... Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n            ... Block style: !munch\n            ...   Clark : Evans\n            ...   Brian : Ingerson\n            ...   Oren  : Ben-Kiki\n            ... ''') #doctest: +NORMALIZE_WHITESPACE\n            {'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n             'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki')}\n\n            This module registers itself automatically to cover both Munch and any\n            subclasses. Should you want to customize the representation of a subclass,\n            simply register it with PyYAML yourself.\n        \"\"\"\n        data = Munch()\n        yield data\n        value = loader.construct_mapping(node)\n        data.update(value)\n\n    def to_yaml_safe(dumper, data):\n        \"\"\" Converts Munch to a normal mapping node, making it appear as a\n            dict in the YAML output.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.safe_dump(b, default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_dict(data)\n\n    def to_yaml(dumper, data):\n        \"\"\" Converts Munch to a representation node.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.dump(b, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_mapping('!munch.Munch', data)\n\n    for loader_name in (\"BaseLoader\", \"FullLoader\", \"SafeLoader\", \"Loader\", \"UnsafeLoader\", \"DangerLoader\"):\n        LoaderCls = getattr(yaml, loader_name, None)\n        if LoaderCls is None:\n            # This code supports both PyYAML 4.x and 5.x versions\n            continue\n        yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n\n    SafeRepresenter.add_representer(Munch, to_yaml_safe)\n    SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n\n    Representer.add_representer(Munch, to_yaml)\n    Representer.add_multi_representer(Munch, to_yaml)\n\n    # Instance methods for YAML conversion\n    def toYAML(self, **options):\n        \"\"\" Serializes this Munch to YAML, using `yaml.safe_dump()` if\n            no `Dumper` is provided. See the PyYAML documentation for more info.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.safe_dump(b, default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n            >>> b.toYAML(default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n            >>> yaml.dump(b, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n            >>> b.toYAML(Dumper=yaml.Dumper, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n\n        \"\"\"\n        opts = dict(indent=4, default_flow_style=False)\n        opts.update(options)\n        if 'Dumper' not in opts:\n            return yaml.safe_dump(self, **opts)\n        else:\n            return yaml.dump(self, **opts)\n\n    def fromYAML(cls, stream, *args, **kwargs):\n        factory = lambda d: cls(*(args + (d,)), **kwargs)\n        loader_class = kwargs.pop('Loader', yaml.FullLoader)\n        return munchify(yaml.load(stream, Loader=loader_class), factory=factory)\n\n    Munch.toYAML = toYAML\n    Munch.fromYAML = classmethod(fromYAML)\n\nexcept ImportError:\n    pass\n"
    },
    {
      "path": "Infinidat_munch/tests/test_munch.py",
      "content": "# pylint: disable=unnecessary-lambda\nimport json\nimport pickle\nfrom collections import namedtuple\n\nimport pytest\nfrom munch import DefaultFactoryMunch, AutoMunch, DefaultMunch, Munch, munchify, unmunchify\n\n\ndef test_base():\n    b = Munch()\n    b.hello = 'world'\n    assert b.hello == 'world'\n    b['hello'] += \"!\"\n    assert b.hello == 'world!'\n    b.foo = Munch(lol=True)\n    assert b.foo.lol is True\n    assert b.foo is b['foo']\n\n    assert sorted(b.keys()) == ['foo', 'hello']\n\n    b.update({'ponies': 'are pretty!'}, hello=42)\n    assert b == Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n    assert sorted([(k, b[k]) for k in b]) == [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n\n    format_munch = Munch(knights='lolcats', ni='can haz')\n    assert \"The {knights} who say {ni}!\".format(**format_munch) == 'The lolcats who say can haz!'\n\n\ndef test_contains():\n    b = Munch(ponies='are pretty!')\n    assert 'ponies' in b\n    assert ('foo' in b) is False\n\n    b['foo'] = 42\n    assert 'foo' in b\n\n    b.hello = 'hai'\n    assert 'hello' in b\n\n    b[None] = 123\n    assert None in b\n\n    b[False] = 456\n    assert False in b\n\n\ndef test_getattr():\n    b = Munch(bar='baz', lol={})\n\n    with pytest.raises(AttributeError):\n        b.foo  # pylint: disable=pointless-statement\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n\ndef test_setattr():\n    b = Munch(foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n\n    with pytest.raises(KeyError):\n        b['values']  # pylint: disable=pointless-statement\n\n\ndef test_pickle():\n    b = DefaultMunch.fromDict({\"a\": \"b\"})\n    assert pickle.loads(pickle.dumps(b)) == b\n\ndef test_automunch():\n    b = AutoMunch()\n    b.urmom = {'sez': {'what': 'what'}}\n    assert b.urmom.sez.what == 'what'  # pylint: disable=no-member\n\n\ndef test_delattr():\n    b = Munch(lol=42)\n    del b.lol\n\n    with pytest.raises(KeyError):\n        b['lol']  # pylint: disable=pointless-statement\n\n    with pytest.raises(AttributeError):\n        b.lol  # pylint: disable=pointless-statement\n\n\ndef test_toDict():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert sorted(b.toDict().items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n    assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\ndef test_repr():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert repr(b).startswith(\"Munch({'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n    assert \"'hello': 42\" in repr(b)\n    assert \"'foo': Munch({'lol': True})\" in repr(b)\n    assert \"'hello': 42\" in repr(b)\n\n    with_spaces = Munch({1: 2, 'a b': 9, 'c': Munch({'simple': 5})})\n    assert repr(with_spaces).startswith(\"Munch({\")\n    assert \"'a b': 9\" in repr(with_spaces)\n    assert \"1: 2\" in repr(with_spaces)\n    assert \"'c': Munch({'simple': 5})\" in repr(with_spaces)\n\n    assert eval(repr(with_spaces)) == Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})  # pylint: disable=eval-used\n\n\ndef test_dir():\n    m = Munch(a=1, b=2)\n    assert dir(m) == ['a', 'b']\n\n\ndef test_fromDict():\n    b = Munch.fromDict({'urmom': {'sez': {'what': 'what'}}})\n    assert b.urmom.sez.what == 'what'\n\n\ndef test_copy():\n    m = Munch(urmom=Munch(sez=Munch(what='what')))\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n\n\ndef test_munchify():\n    b = munchify({'urmom': {'sez': {'what': 'what'}}})\n    assert b.urmom.sez.what == 'what'\n\n    b = munchify({'lol': ('cats', {'hah': 'i win again'}), 'hello': [{'french': 'salut', 'german': 'hallo'}]})\n    assert b.hello[0].french == 'salut'\n    assert b.lol[1].hah == 'i win again'\n\ndef test_munchify_with_namedtuple():\n    nt = namedtuple('nt', ['prop_a', 'prop_b'])\n\n    b = munchify({'top': nt('in named tuple', 3)})\n    assert b.top.prop_a == 'in named tuple'\n    assert b.top.prop_b == 3\n\n    b = munchify({'top': {'middle': nt(prop_a={'leaf': 'should be munchified'},\n                                       prop_b={'leaf': 'should be munchified'})}})\n    assert b.top.middle.prop_a.leaf == 'should be munchified'\n    assert b.top.middle.prop_b.leaf == 'should be munchified'\n\n\ndef test_unmunchify():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert sorted(unmunchify(b).items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42, ponies=('are pretty!', Munch(lies='are trouble!')))\n    assert sorted(unmunchify(b).items()) == [('foo', ['bar', {'lol': True}]),\n                                             ('hello', 42),\n                                             ('ponies', ('are pretty!', {'lies': 'are trouble!'}))]\n\n\ndef test_unmunchify_namedtuple():\n    nt = namedtuple('nt', ['prop_a', 'prop_b'])\n    b = Munch(foo=Munch(lol=True), hello=nt(prop_a=42, prop_b='yop'), ponies='are pretty!')\n    assert sorted(unmunchify(b).items()) == [('foo', {'lol': True}),\n                                             ('hello', nt(prop_a=42, prop_b='yop')),\n                                             ('ponies', 'are pretty!')]\n\n\ndef test_toJSON_and_fromJSON():\n    # pylint: disable=unidiomatic-typecheck\n    obj = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    obj_json = obj.toJSON()\n    assert json.dumps(obj) == obj_json\n    new_obj = Munch.fromJSON(obj_json)\n    assert type(obj) == Munch\n    assert new_obj == obj\n\n    default_value = object()\n    dm_obj = DefaultMunch.fromJSON(obj_json, default_value)\n    assert type(dm_obj) == DefaultMunch\n    assert dm_obj == obj\n    assert dm_obj['not_exist'] is default_value\n    assert dm_obj.not_exist is default_value\n\n\n@pytest.mark.parametrize(\"attrname\", dir(Munch))\ndef test_reserved_attributes(attrname):\n    # Make sure that the default attributes on the Munch instance are\n    # accessible.\n\n    taken_munch = Munch(**{attrname: 'abc123'})\n\n    # Make sure that the attribute is determined as in the filled collection...\n    assert attrname in taken_munch\n\n    # ...and that it is available using key access...\n    assert taken_munch[attrname] == 'abc123'\n\n    # ...but that it is not available using attribute access.\n    attr = getattr(taken_munch, attrname)\n    assert attr != 'abc123'\n\n    empty_munch = Munch()\n\n    # Make sure that the attribute is not seen contained in the empty\n    # collection...\n    assert attrname not in empty_munch\n\n    # ...and that the attr is of the correct original type.\n    attr = getattr(empty_munch, attrname)\n    if attrname == '__doc__':\n        assert isinstance(attr, str)\n    elif attrname in ('__hash__', '__weakref__'):\n        assert attr is None\n    elif attrname == '__module__':\n        assert attr == 'munch'\n    elif attrname == '__dict__':\n        assert attr == {}\n    else:\n        assert callable(attr)\n\n\ndef test_getattr_default():\n    b = DefaultMunch(bar='baz', lol={})\n    assert b.foo is None\n    assert b['foo'] is None\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n    undefined = object()\n    b = DefaultMunch(undefined, bar='baz', lol={})\n    assert b.foo is undefined\n    assert b['foo'] is undefined\n\n\ndef test_setattr_default():\n    b = DefaultMunch(foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n    assert b['values'] is None\n\n    assert b.__default__ is None\n    assert '__default__' not in b\n\n\ndef test_delattr_default():\n    b = DefaultMunch(lol=42)\n    del b.lol\n\n    assert b.lol is None\n    assert b['lol'] is None\n\n\ndef test_pickle_default():\n    b = DefaultMunch.fromDict({\"a\": \"b\"})\n    assert pickle.loads(pickle.dumps(b)) == b\n\n\ndef test_fromDict_default():\n    undefined = object()\n    b = DefaultMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    assert b.urmom.sez.what == 'what'\n    assert b.urmom.sez.foo is undefined\n\n\ndef test_copy_default():\n    undefined = object()\n    m = DefaultMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n    assert c.urmom.sez.foo is undefined\n    assert c.urmom.sez.__undefined__ is undefined\n\n\ndef test_munchify_default():\n    undefined = object()\n    b = munchify(\n        {'urmom': {'sez': {'what': 'what'}}},\n        lambda d: DefaultMunch(undefined, d))\n    assert b.urmom.sez.what == 'what'\n    assert b.urdad is undefined\n    assert b.urmom.sez.ni is undefined\n\n\ndef test_repr_default():\n    b = DefaultMunch(foo=DefaultMunch(lol=True), ponies='are pretty!')\n    assert repr(b).startswith(\"DefaultMunch(None, {'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n\n\ndef test_getattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, bar='baz', lol={})\n    assert b.foo is None\n    assert b['foo'] is None\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n    undefined = object()\n    default = lambda: undefined\n    b = DefaultFactoryMunch(default, bar='baz', lol={})\n    assert b.foo is undefined\n    assert b['foo'] is undefined\n\n    default = lambda: object()\n    b = DefaultFactoryMunch(default, bar='baz', lol={})\n    assert b.foo is not b.baz\n    assert b.foo is b['foo']\n    assert b.foobar is b.foobar\n\n    b = DefaultFactoryMunch(list)\n    assert b.foo == []\n    b.foo.append('bar')\n    assert b.foo == ['bar']\n    assert b.default_factory is list\n\n\ndef test_setattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n    assert b['values'] is None\n\n    assert b.default_factory() is None\n    assert 'default_factory' not in b\n\n\ndef test_delattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, lol=42)\n    del b.lol\n\n    assert b.lol is None\n    assert b['lol'] is None\n\n\ndef test_fromDict_default_factory():\n    obj = object()\n    undefined = lambda: obj\n    b = DefaultFactoryMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    assert b.urmom.sez.what == 'what'\n    assert b.urmom.sez.foo is undefined()\n\n\ndef test_copy_default_factory():\n    undefined = lambda: object()\n    m = DefaultFactoryMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n\n\ndef test_munchify_default_factory():\n    undefined = lambda: object()\n    b = munchify(\n        {'urmom': {'sez': {'what': 'what'}}},\n        lambda d: DefaultFactoryMunch(undefined, d))\n    assert b.urmom.sez.what == 'what'\n    assert b.urdad is not undefined()\n    assert b.urmom.sez.ni is not b.urdad\n\n\ndef test_munchify_cycle():\n    # dict1 -> dict2 -> dict1\n    x = dict(id=\"x\")\n    y = dict(x=x, id=\"y\")\n    x['y'] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y.id == \"y\"\n    assert m.y.x is m\n\n    # dict -> list -> dict\n    x = dict(id=\"x\")\n    y = [\"y\", x]\n    x[\"y\"] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1] is m\n\n    # dict -> tuple -> dict\n    x = dict(id=\"x\")\n    y = (\"y\", x)\n    x[\"y\"] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1] is m\n\n    # dict1 -> list -> dict2 -> list\n    z = dict(id=\"z\")\n    y = [\"y\", z]\n    z[\"y\"] = y\n    x = dict(id=\"x\", y=y)\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1].id == \"z\"\n    assert m.y[1].y is m.y\n\n    # dict1 -> tuple -> dict2 -> tuple\n    z = dict(id=\"z\")\n    y = (\"y\", z)\n    z[\"y\"] = y\n    x = dict(id=\"x\", y=y)\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1].id == \"z\"\n    assert m.y[1].y is m.y\n\ndef test_unmunchify_cycle():\n    # munch -> munch -> munch\n    x = Munch(id=\"x\")\n    y = Munch(x=x, id=\"y\")\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][\"id\"] == \"y\"\n    assert d[\"y\"][\"x\"] is d\n\n    # munch -> list -> munch\n    x = Munch(id=\"x\")\n    y = [\"y\", x]\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1] is d\n\n    # munch -> tuple -> munch\n    x = Munch(id=\"x\")\n    y = (\"y\", x)\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1] is d\n\n    # munch1 -> list -> munch2 -> list\n    z = Munch(id=\"z\")\n    y = [\"y\", z]\n    z.y = y\n    x = Munch(id=\"x\", y=y)\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1][\"id\"] == \"z\"\n    assert d[\"y\"][1][\"y\"] is d[\"y\"]\n\n    # munch1 -> tuple -> munch2 -> tuple\n    z = Munch(id=\"z\")\n    y = (\"y\", z)\n    z.y = y\n    x = Munch(id=\"x\", y=y)\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1][\"id\"] == \"z\"\n    assert d[\"y\"][1][\"y\"] is d[\"y\"]\n\n\ndef test_repr_default_factory():\n    b = DefaultFactoryMunch(list, foo=DefaultFactoryMunch(list, lol=True), ponies='are pretty!')\n    assert repr(b).startswith(\"DefaultFactoryMunch(list, {'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n\n    assert eval(repr(b)) == b  # pylint: disable=eval-used\n\n\ndef test_pickling_unpickling_nested():\n    m = {'a': {'b': 'c'}}\n    m = munchify(m)\n    assert m == Munch({'a': Munch({'b': 'c'})})\n    assert isinstance(m.a, Munch)\n    result = pickle.loads(pickle.dumps(m))\n    assert result == m\n    assert isinstance(result.a, Munch)\n\n\ndef test_setitem_dunder_for_subclass():\n\n    def test_class(cls, *args):\n        class CustomMunch(cls):\n            def __setitem__(self, k, v):\n                super().__setitem__(k, [v] * 2)\n        custom_munch = CustomMunch(*args, a='foo')\n        assert custom_munch.a == ['foo', 'foo']\n        regular_dict = {}\n        regular_dict.update(custom_munch)\n        assert regular_dict['a'] == ['foo', 'foo']\n        assert repr(regular_dict) == \"{'a': ['foo', 'foo']}\"\n        custom_munch.setdefault('bar', 'baz')\n        assert custom_munch.bar == ['baz', 'baz']\n\n    test_class(Munch)\n    test_class(DefaultFactoryMunch, list)\n    test_class(DefaultMunch, 42)\n\n\ndef test_getitem_dunder_for_subclass():\n    class CustomMunch(Munch):\n        def __getitem__(self, k):\n            return 42\n\n    custom_munch = CustomMunch(a='foo')\n    custom_munch.update({'b': 1})\n    assert custom_munch.a == 42\n    assert custom_munch.get('b') == 42\n    assert custom_munch.copy() == Munch(a=42, b=42)\n\n\n@pytest.mark.usefixtures(\"yaml\")\ndef test_get_default_value(munch_obj):\n    assert munch_obj.get(\"fake_key\", \"default_value\") == \"default_value\"\n    assert isinstance(munch_obj.toJSON(), str)\n    assert isinstance(munch_obj.toYAML(), str)\n    munch_obj.copy()\n    data = munch_obj.toDict()\n    munch_cls = type(munch_obj)\n    kwargs = {} if munch_cls != DefaultFactoryMunch else {\"default_factory\": munch_obj.default_factory}\n    munch_cls.fromDict(data, **kwargs)\n\n\ndef test_munchify_tuple_list():\n    data = ([{'A': 'B'}],)\n    actual = munchify(data)\n    expected = ([Munch(A='B')],)\n    assert actual == expected\n\n\ndef test_munchify_tuple_list_more_elements():\n    data = (1, 2, [{'A': 'B'}])\n    actual = munchify(data)\n    expected = (1, 2, [Munch({'A': 'B'})])\n    assert actual == expected\n"
    },
    {
      "path": "Infinidat_munch/tests/conftest.py",
      "content": "import pytest\nimport munch\n\n\n@pytest.fixture(name='yaml')\ndef yaml_module():\n    try:\n        import yaml  # pylint: disable=import-outside-toplevel\n        return yaml\n    except ImportError:\n        pass\n    pytest.skip(\"Module 'PyYAML' is required\")\n\n\n@pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n                        munch.RecursiveMunch])\ndef munch_obj(request):\n    cls = request.param\n    args = tuple()\n    if cls == munch.DefaultFactoryMunch:\n        args = args + (lambda: None,)\n    return cls(*args, hello=\"world\", number=5)\n"
    },
    {
      "path": "Infinidat_munch/tests/__init__.py",
      "content": ""
    },
    {
      "path": "Infinidat_munch/tests/test_readme.py",
      "content": "import doctest\nimport os\nimport pytest\n\n_HERE = os.path.abspath(os.path.dirname(__file__))\n_README_PATH = os.path.join(_HERE, '..', 'README.md')\nassert os.path.exists(_README_PATH)\n\n\n@pytest.mark.usefixtures(\"yaml\")\ndef test_readme():\n    globs = {\n        'print_function': print\n    }\n    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n    assert not result.failed\n"
    },
    {
      "path": "Infinidat_munch/tests/test_yaml.py",
      "content": "import pytest\nfrom munch import Munch, DefaultMunch\n\n\ndef test_from_yaml(yaml):\n    data = yaml.load('''\n    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n    Block style: !munch\n      Clark : Evans\n      Brian : Ingerson\n      Oren  : Ben-Kiki\n    ''', Loader=yaml.FullLoader)\n    assert data == {\n        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n    }\n\n\ndef test_to_yaml_safe(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    dumped = yaml.safe_dump(b, default_flow_style=True)\n    assert dumped == '{foo: [bar, {lol: true}], hello: 42}\\n'\n\n\ndef test_to_yaml(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    dumped = yaml.dump(b, default_flow_style=True)\n    assert dumped == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n\n\ndef test_toYAML(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    assert yaml.safe_dump(b, default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n    assert b.toYAML(default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n    assert yaml.dump(b, default_flow_style=True) == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n    assert b.toYAML(Dumper=yaml.Dumper, default_flow_style=True) == \\\n        '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n\n\n@pytest.mark.usefixtures('yaml')\ndef test_fromYAML():\n    # pylint: disable=unidiomatic-typecheck\n    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n    obj = Munch.fromYAML(yaml_str)\n    assert type(obj) == Munch\n    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n    assert obj.toYAML() == yaml_str\n\n\n@pytest.mark.usefixtures('yaml')\ndef test_fromYAML_default_munch():\n    # pylint: disable=unidiomatic-typecheck\n    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n    default_value = object()\n    obj = DefaultMunch.fromYAML(yaml_str, default_value)\n    assert type(obj) == DefaultMunch\n    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n    assert obj['not_exist'] is default_value\n    assert obj.not_exist is default_value\n    assert obj.toYAML() == yaml_str\n"
    }
  ],
  "OriginCode": [
    {
      "path": "Infinidat_munch/setup.py",
      "content": "from setuptools import setup\n\n\nsetup(\n    setup_requires=['pbr>=3.0', 'setuptools>=61'],\n    pbr=True,\n    long_description_content_type='text/markdown; charset=UTF-8',\n    keywords=['munch', 'dict', 'mapping', 'container', 'collection'],\n    python_requires=\">=3.6\",\n)\n"
    },
    {
      "path": "Infinidat_munch/munch/__init__.py",
      "content": "\"\"\" Munch is a subclass of dict with attribute-style access.\n\n    >>> b = Munch()\n    >>> b.hello = 'world'\n    >>> b.hello\n    'world'\n    >>> b['hello'] += \"!\"\n    >>> b.hello\n    'world!'\n    >>> b.foo = Munch(lol=True)\n    >>> b.foo.lol\n    True\n    >>> b.foo is b['foo']\n    True\n\n    It is safe to import * from this module:\n\n        __all__ = ('Munch', 'munchify','unmunchify')\n\n    un/munchify provide dictionary conversion; Munches can also be\n    converted via Munch.to/fromDict().\n\"\"\"\n\nfrom collections.abc import Mapping\n\ntry:\n    # For python 3.8 and later\n    import importlib.metadata as importlib_metadata\nexcept ImportError:\n    # For everyone else\n    import importlib_metadata\ntry:\n    __version__ = importlib_metadata.version(__name__)\nexcept importlib_metadata.PackageNotFoundError:\n    # package is not installed\n    __version__ = \"0.0.0\"\n\n\ntry:\n    VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\nexcept ValueError:\n    VERSION = (0, 0, 0)\n\n__all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n\n\n\nclass Munch(dict):\n    \"\"\" A dictionary that provides attribute-style access.\n\n        >>> b = Munch()\n        >>> b.hello = 'world'\n        >>> b.hello\n        'world'\n        >>> b['hello'] += \"!\"\n        >>> b.hello\n        'world!'\n        >>> b.foo = Munch(lol=True)\n        >>> b.foo.lol\n        True\n        >>> b.foo is b['foo']\n        True\n\n        A Munch is a subclass of dict; it supports all the methods a dict does...\n\n        >>> sorted(b.keys())\n        ['foo', 'hello']\n\n        Including update()...\n\n        >>> b.update({ 'ponies': 'are pretty!' }, hello=42)\n        >>> print (repr(b))\n        Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n        As well as iteration...\n\n        >>> sorted([ (k,b[k]) for k in b ])\n        [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n\n        And \"splats\".\n\n        >>> \"The {knights} who say {ni}!\".format(**Munch(knights='lolcats', ni='can haz'))\n        'The lolcats who say can haz!'\n\n        See unmunchify/Munch.toDict, munchify/Munch.fromDict for notes about conversion.\n    \"\"\"\n    def __init__(self, *args, **kwargs):  # pylint: disable=super-init-not-called\n        self.update(*args, **kwargs)\n\n    # only called if k not found in normal places\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise throws AttributeError.\n\n            nb. __getattr__ is only called if key is not found in normal places.\n\n            >>> b = Munch(bar='baz', lol={})\n            >>> b.foo\n            Traceback (most recent call last):\n                ...\n            AttributeError: foo\n\n            >>> b.bar\n            'baz'\n            >>> getattr(b, 'bar')\n            'baz'\n            >>> b['bar']\n            'baz'\n\n            >>> b.lol is b['lol']\n            True\n            >>> b.lol is getattr(b, 'lol')\n            True\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            return object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                return self[k]\n            except KeyError:\n                raise AttributeError(k)\n\n    def __setattr__(self, k, v):\n        \"\"\" Sets attribute k if it exists, otherwise sets key k. A KeyError\n            raised by set-item (only likely if you subclass Munch) will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(foo='bar', this_is='useful when subclassing')\n            >>> hasattr(b.values, '__call__')\n            True\n            >>> b.values = 'uh oh'\n            >>> b.values\n            'uh oh'\n            >>> b['values']\n            Traceback (most recent call last):\n                ...\n            KeyError: 'values'\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                self[k] = v\n            except:\n                raise AttributeError(k)\n        else:\n            object.__setattr__(self, k, v)\n\n    def __delattr__(self, k):\n        \"\"\" Deletes attribute k if it exists, otherwise deletes key k. A KeyError\n            raised by deleting the key--such as when the key is missing--will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(lol=42)\n            >>> del b.lol\n            >>> b.lol\n            Traceback (most recent call last):\n                ...\n            AttributeError: lol\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                del self[k]\n            except KeyError:\n                raise AttributeError(k)\n        else:\n            object.__delattr__(self, k)\n\n    def toDict(self):\n        \"\"\" Recursively converts a munch back into a dictionary.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> sorted(b.toDict().items())\n            [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n            See unmunchify for more info.\n        \"\"\"\n        return unmunchify(self)\n\n    @property\n    def __dict__(self):\n        return self.toDict()\n\n    def __repr__(self):\n        \"\"\" Invertible* string-form of a Munch.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> print (repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n            >>> eval(repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n            >>> with_spaces = Munch({1: 2, 'a b': 9, 'c': Munch({'simple': 5})})\n            >>> print (repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n            >>> eval(repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n\n            (*) Invertible so long as collection contents are each repr-invertible.\n        \"\"\"\n        return f'{self.__class__.__name__}({dict.__repr__(self)})'\n\n    def __dir__(self):\n        return list(self.keys())\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return {k: v for k, v in self.items()}\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        self.update(state)\n\n    __members__ = __dir__  # for python2.x compatibility\n\n    @classmethod\n    def fromDict(cls, d):\n        \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n            >>> b = Munch.fromDict({'urmom': {'sez': {'what': 'what'}}})\n            >>> b.urmom.sez.what\n            'what'\n\n            See munchify for more info.\n        \"\"\"\n        return munchify(d, cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n    def update(self, *args, **kwargs):\n        \"\"\"\n        Override built-in method to call custom __setitem__ method that may\n        be defined in subclasses.\n        \"\"\"\n        for k, v in dict(*args, **kwargs).items():\n            self[k] = v\n\n    def get(self, k, d=None):\n        \"\"\"\n        D.get(k[,d]) -> D[k] if k in D, else d.  d defaults to None.\n        \"\"\"\n        if k not in self:\n            return d\n        return self[k]\n\n    def setdefault(self, k, d=None):\n        \"\"\"\n        D.setdefault(k[,d]) -> D.get(k,d), also set D[k]=d if k not in D\n        \"\"\"\n        if k not in self:\n            self[k] = d\n        return self[k]\n\n\nclass AutoMunch(Munch):\n    def __setattr__(self, k, v):\n        \"\"\" Works the same as Munch.__setattr__ but if you supply\n            a dictionary as value it will convert it to another Munch.\n        \"\"\"\n        if isinstance(v, Mapping) and not isinstance(v, (AutoMunch, Munch)):\n            v = munchify(v, AutoMunch)\n        super().__setattr__(k, v)\n\n\nclass DefaultMunch(Munch):\n    \"\"\"\n    A Munch that returns a user-specified value for missing keys.\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        \"\"\" Construct a new DefaultMunch. Like collections.defaultdict, the\n            first argument is the default value; subsequent arguments are the\n            same as those for dict.\n        \"\"\"\n        # Mimic collections.defaultdict constructor\n        if args:\n            default = args[0]\n            args = args[1:]\n        else:\n            default = None\n        super().__init__(*args, **kwargs)\n        self.__default__ = default\n\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getattr__(k)\n        except AttributeError:\n            return self.__default__\n\n    def __setattr__(self, k, v):\n        if k == '__default__':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __getitem__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getitem__(k)\n        except KeyError:\n            return self.__default__\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return (self.__default__, {k: v for k, v in self.items()})\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        default, state_dict = state\n        self.update(state_dict)\n        self.__default__ = default\n\n    @classmethod\n    def fromDict(cls, d, default=None):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default=self.__default__)\n\n    def __repr__(self):\n        return f'{type(self).__name__}({self.__undefined__!r}, {dict.__repr__(self)})'\n\n\nclass DefaultFactoryMunch(Munch):\n    \"\"\" A Munch that calls a user-specified function to generate values for\n        missing keys like collections.defaultdict.\n\n        >>> b = DefaultFactoryMunch(list, {'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        []\n        >>> b.bar.append('hello')\n        >>> b.bar\n        ['hello']\n    \"\"\"\n\n    def __init__(self, default_factory, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.default_factory = default_factory\n\n    @classmethod\n    def fromDict(cls, d, default_factory):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default_factory, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default_factory=self.default_factory)\n\n    def __repr__(self):\n        factory = self.default_factory.__name__\n        return f'{type(self).__name__}({factory}, {dict.__repr__(self)})'\n\n    def __setattr__(self, k, v):\n        if k == 'default_factory':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __missing__(self, k):\n        self[k] = self.default_factory()\n        return self[k]\n\n\nclass RecursiveMunch(DefaultFactoryMunch):\n    \"\"\"A Munch that calls an instance of itself to generate values for\n        missing keys.\n\n        >>> b = RecursiveMunch({'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        RecursiveMunch(RecursiveMunch, {})\n        >>> b.bar.okay = 'hello'\n        >>> b.bar\n        RecursiveMunch(RecursiveMunch, {'okay': 'hello'})\n        >>> b\n        RecursiveMunch(RecursiveMunch, {'hello': 'world!', 'foo': RecursiveMunch(RecursiveMunch, {}),\n        'bar': RecursiveMunch(RecursiveMunch, {'okay': 'hello'})})\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(RecursiveMunch, *args, **kwargs)\n\n    @classmethod\n    def fromDict(cls, d):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n\n# While we could convert abstract types like Mapping or Iterable, I think\n# munchify is more likely to \"do what you mean\" if it is conservative about\n# casting (ex: isinstance(str,Iterable) == True ).\n#\n# Should you disagree, it is not difficult to duplicate this function with\n# more aggressive coercion to suit your own purposes.\n\ndef munchify(x, factory=Munch):\n    \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n        >>> b = munchify({'urmom': {'sez': {'what': 'what'}}})\n        >>> b.urmom.sez.what\n        'what'\n\n        munchify can handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = munchify({ 'lol': ('cats', {'hah':'i win again'}),\n        ...         'hello': [{'french':'salut', 'german':'hallo'}] })\n        >>> b.hello[0].french\n        'salut'\n        >>> b.lol[1].hah\n        'i win again'\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def munchify_cycles(obj):\n        partial, already_seen = pre_munchify_cycles(obj)\n        if already_seen:\n            return partial\n        return post_munchify(partial, obj)\n\n    def pre_munchify_cycles(obj):\n        # If we've already begun munchifying obj, just return the already-created munchified obj\n        try:\n            return seen[id(obj)], True\n        except KeyError:\n            pass\n\n        # Otherwise, first partly munchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_munchify(obj)\n        return partial, False\n\n    def pre_munchify(obj):\n        # Here we return a skeleton of munchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_munchify\n        if isinstance(obj, Mapping):\n            return factory({})\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(pre_munchify_cycles(item)[0] for item in obj)\n        else:\n            return obj\n\n    def post_munchify(partial, obj):\n        # Here we finish munchifying the parts of obj that were deferred by pre_munchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, munchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(munchify_cycles(item) for item in obj)\n        elif isinstance(obj, tuple):\n            for (item_partial, item) in zip(partial, obj):\n                post_munchify(item_partial, item)\n\n        return partial\n\n    return munchify_cycles(x)\n\n\ndef unmunchify(x):\n    \"\"\" Recursively converts a Munch into a dictionary.\n\n        >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n        >>> sorted(unmunchify(b).items())\n        [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n        unmunchify will handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42,\n        ...         ponies=('are pretty!', Munch(lies='are trouble!')))\n        >>> sorted(unmunchify(b).items()) #doctest: +NORMALIZE_WHITESPACE\n        [('foo', ['bar', {'lol': True}]), ('hello', 42), ('ponies', ('are pretty!', {'lies': 'are trouble!'}))]\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def unmunchify_cycles(obj):\n        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n        try:\n            return seen[id(obj)]\n        except KeyError:\n            pass\n\n        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_unmunchify(obj)\n        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n        return post_unmunchify(partial, obj)\n\n    def pre_unmunchify(obj):\n        # Here we return a skeleton of unmunchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_unmunchify\n        if isinstance(obj, Mapping):\n            return dict()\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(unmunchify_cycles(item) for item in obj)\n        else:\n            return obj\n\n    def post_unmunchify(partial, obj):\n        # Here we finish unmunchifying the parts of obj that were deferred by pre_unmunchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, unmunchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(unmunchify_cycles(v) for v in obj)\n        elif isinstance(obj, tuple):\n            for (value_partial, value) in zip(partial, obj):\n                post_unmunchify(value_partial, value)\n\n        return partial\n\n    return unmunchify_cycles(x)\n\n\n# Serialization\n\ntry:\n    try:\n        import json\n    except ImportError:\n        import simplejson as json\n\n    def toJSON(self, **options):\n        \"\"\" Serializes this Munch to JSON. Accepts the same keyword options as `json.dumps()`.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> json.dumps(b) == b.toJSON()\n            True\n        \"\"\"\n        return json.dumps(self, **options)\n\n    def fromJSON(cls, stream, *args, **kwargs):\n        \"\"\" Deserializes JSON to Munch or any of its subclasses.\n        \"\"\"\n        factory = lambda d: cls(*(args + (d,)), **kwargs)\n        return munchify(json.loads(stream), factory=factory)\n\n    Munch.toJSON = toJSON\n    Munch.fromJSON = classmethod(fromJSON)\n\nexcept ImportError:\n    pass\n\n\ntry:\n    # Attempt to register ourself with PyYAML as a representer\n    import yaml\n    from yaml.representer import Representer, SafeRepresenter\n\n    def from_yaml(loader, node):\n        \"\"\" PyYAML support for Munches using the tag `!munch` and `!munch.Munch`.\n\n            >>> import yaml\n            >>> yaml.load('''\n            ... Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n            ... Block style: !munch\n            ...   Clark : Evans\n            ...   Brian : Ingerson\n            ...   Oren  : Ben-Kiki\n            ... ''') #doctest: +NORMALIZE_WHITESPACE\n            {'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n             'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki')}\n\n            This module registers itself automatically to cover both Munch and any\n            subclasses. Should you want to customize the representation of a subclass,\n            simply register it with PyYAML yourself.\n        \"\"\"\n        data = Munch()\n        yield data\n        value = loader.construct_mapping(node)\n        data.update(value)\n\n    def to_yaml_safe(dumper, data):\n        \"\"\" Converts Munch to a normal mapping node, making it appear as a\n            dict in the YAML output.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.safe_dump(b, default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_dict(data)\n\n    def to_yaml(dumper, data):\n        \"\"\" Converts Munch to a representation node.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.dump(b, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_mapping('!munch.Munch', data)\n\n    for loader_name in (\"BaseLoader\", \"FullLoader\", \"SafeLoader\", \"Loader\", \"UnsafeLoader\", \"DangerLoader\"):\n        LoaderCls = getattr(yaml, loader_name, None)\n        if LoaderCls is None:\n            # This code supports both PyYAML 4.x and 5.x versions\n            continue\n        yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n        yaml.add_constructor('!munch.Munch', from_yaml, Loader=LoaderCls)\n\n    SafeRepresenter.add_representer(Munch, to_yaml_safe)\n    SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n\n    Representer.add_representer(Munch, to_yaml)\n    Representer.add_multi_representer(Munch, to_yaml)\n\n    # Instance methods for YAML conversion\n    def toYAML(self, **options):\n        \"\"\" Serializes this Munch to YAML, using `yaml.safe_dump()` if\n            no `Dumper` is provided. See the PyYAML documentation for more info.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.safe_dump(b, default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n            >>> b.toYAML(default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n            >>> yaml.dump(b, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n            >>> b.toYAML(Dumper=yaml.Dumper, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n\n        \"\"\"\n        opts = dict(indent=4, default_flow_style=False)\n        opts.update(options)\n        if 'Dumper' not in opts:\n            return yaml.safe_dump(self, **opts)\n        else:\n            return yaml.dump(self, **opts)\n\n    def fromYAML(cls, stream, *args, **kwargs):\n        factory = lambda d: cls(*(args + (d,)), **kwargs)\n        loader_class = kwargs.pop('Loader', yaml.FullLoader)\n        return munchify(yaml.load(stream, Loader=loader_class), factory=factory)\n\n    Munch.toYAML = toYAML\n    Munch.fromYAML = classmethod(fromYAML)\n\nexcept ImportError:\n    pass\n"
    },
    {
      "path": "Infinidat_munch/tests/test_munch.py",
      "content": "# pylint: disable=unnecessary-lambda\nimport json\nimport pickle\nfrom collections import namedtuple\n\nimport pytest\nfrom munch import DefaultFactoryMunch, AutoMunch, DefaultMunch, Munch, munchify, unmunchify\n\n\ndef test_base():\n    b = Munch()\n    b.hello = 'world'\n    assert b.hello == 'world'\n    b['hello'] += \"!\"\n    assert b.hello == 'world!'\n    b.foo = Munch(lol=True)\n    assert b.foo.lol is True\n    assert b.foo is b['foo']\n\n    assert sorted(b.keys()) == ['foo', 'hello']\n\n    b.update({'ponies': 'are pretty!'}, hello=42)\n    assert b == Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n    assert sorted([(k, b[k]) for k in b]) == [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n\n    format_munch = Munch(knights='lolcats', ni='can haz')\n    assert \"The {knights} who say {ni}!\".format(**format_munch) == 'The lolcats who say can haz!'\n\n\ndef test_contains():\n    b = Munch(ponies='are pretty!')\n    assert 'ponies' in b\n    assert ('foo' in b) is False\n\n    b['foo'] = 42\n    assert 'foo' in b\n\n    b.hello = 'hai'\n    assert 'hello' in b\n\n    b[None] = 123\n    assert None in b\n\n    b[False] = 456\n    assert False in b\n\n\ndef test_getattr():\n    b = Munch(bar='baz', lol={})\n\n    with pytest.raises(AttributeError):\n        b.foo  # pylint: disable=pointless-statement\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n\ndef test_setattr():\n    b = Munch(foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n\n    with pytest.raises(KeyError):\n        b['values']  # pylint: disable=pointless-statement\n\n\ndef test_pickle():\n    b = DefaultMunch.fromDict({\"a\": \"b\"})\n    assert pickle.loads(pickle.dumps(b)) == b\n\ndef test_automunch():\n    b = AutoMunch()\n    b.urmom = {'sez': {'what': 'what'}}\n    assert b.urmom.sez.what == 'what'  # pylint: disable=no-member\n\n\ndef test_delattr():\n    b = Munch(lol=42)\n    del b.lol\n\n    with pytest.raises(KeyError):\n        b['lol']  # pylint: disable=pointless-statement\n\n    with pytest.raises(AttributeError):\n        b.lol  # pylint: disable=pointless-statement\n\n\ndef test_toDict():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert sorted(b.toDict().items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\ndef test_dict_property():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\ndef test_repr():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert repr(b).startswith(\"Munch({'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n    assert \"'hello': 42\" in repr(b)\n    assert \"'foo': Munch({'lol': True})\" in repr(b)\n    assert \"'hello': 42\" in repr(b)\n\n    with_spaces = Munch({1: 2, 'a b': 9, 'c': Munch({'simple': 5})})\n    assert repr(with_spaces).startswith(\"Munch({\")\n    assert \"'a b': 9\" in repr(with_spaces)\n    assert \"1: 2\" in repr(with_spaces)\n    assert \"'c': Munch({'simple': 5})\" in repr(with_spaces)\n\n    assert eval(repr(with_spaces)) == Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})  # pylint: disable=eval-used\n\n\ndef test_dir():\n    m = Munch(a=1, b=2)\n    assert dir(m) == ['a', 'b']\n\n\ndef test_fromDict():\n    b = Munch.fromDict({'urmom': {'sez': {'what': 'what'}}})\n    assert b.urmom.sez.what == 'what'\n\n\ndef test_copy():\n    m = Munch(urmom=Munch(sez=Munch(what='what')))\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n\n\ndef test_munchify():\n    b = munchify({'urmom': {'sez': {'what': 'what'}}})\n    assert b.urmom.sez.what == 'what'\n\n    b = munchify({'lol': ('cats', {'hah': 'i win again'}), 'hello': [{'french': 'salut', 'german': 'hallo'}]})\n    assert b.hello[0].french == 'salut'\n    assert b.lol[1].hah == 'i win again'\n\ndef test_munchify_with_namedtuple():\n    nt = namedtuple('nt', ['prop_a', 'prop_b'])\n\n    b = munchify({'top': nt('in named tuple', 3)})\n    assert b.top.prop_a == 'in named tuple'\n    assert b.top.prop_b == 3\n\n    b = munchify({'top': {'middle': nt(prop_a={'leaf': 'should be munchified'},\n                                       prop_b={'leaf': 'should be munchified'})}})\n    assert b.top.middle.prop_a.leaf == 'should be munchified'\n    assert b.top.middle.prop_b.leaf == 'should be munchified'\n\n\ndef test_unmunchify():\n    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    assert sorted(unmunchify(b).items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42, ponies=('are pretty!', Munch(lies='are trouble!')))\n    assert sorted(unmunchify(b).items()) == [('foo', ['bar', {'lol': True}]),\n                                             ('hello', 42),\n                                             ('ponies', ('are pretty!', {'lies': 'are trouble!'}))]\n\n\ndef test_unmunchify_namedtuple():\n    nt = namedtuple('nt', ['prop_a', 'prop_b'])\n    b = Munch(foo=Munch(lol=True), hello=nt(prop_a=42, prop_b='yop'), ponies='are pretty!')\n    assert sorted(unmunchify(b).items()) == [('foo', {'lol': True}),\n                                             ('hello', nt(prop_a=42, prop_b='yop')),\n                                             ('ponies', 'are pretty!')]\n\n\ndef test_toJSON_and_fromJSON():\n    # pylint: disable=unidiomatic-typecheck\n    obj = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n    obj_json = obj.toJSON()\n    assert json.dumps(obj) == obj_json\n    new_obj = Munch.fromJSON(obj_json)\n    assert type(obj) == Munch\n    assert new_obj == obj\n\n    default_value = object()\n    dm_obj = DefaultMunch.fromJSON(obj_json, default_value)\n    assert type(dm_obj) == DefaultMunch\n    assert dm_obj == obj\n    assert dm_obj['not_exist'] is default_value\n    assert dm_obj.not_exist is default_value\n\n\n@pytest.mark.parametrize(\"attrname\", dir(Munch))\ndef test_reserved_attributes(attrname):\n    # Make sure that the default attributes on the Munch instance are\n    # accessible.\n\n    taken_munch = Munch(**{attrname: 'abc123'})\n\n    # Make sure that the attribute is determined as in the filled collection...\n    assert attrname in taken_munch\n\n    # ...and that it is available using key access...\n    assert taken_munch[attrname] == 'abc123'\n\n    # ...but that it is not available using attribute access.\n    attr = getattr(taken_munch, attrname)\n    assert attr != 'abc123'\n\n    empty_munch = Munch()\n\n    # Make sure that the attribute is not seen contained in the empty\n    # collection...\n    assert attrname not in empty_munch\n\n    # ...and that the attr is of the correct original type.\n    attr = getattr(empty_munch, attrname)\n    if attrname == '__doc__':\n        assert isinstance(attr, str)\n    elif attrname in ('__hash__', '__weakref__'):\n        assert attr is None\n    elif attrname == '__module__':\n        assert attr == 'munch'\n    elif attrname == '__dict__':\n        assert attr == {}\n    else:\n        assert callable(attr)\n\n\ndef test_getattr_default():\n    b = DefaultMunch(bar='baz', lol={})\n    assert b.foo is None\n    assert b['foo'] is None\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n    undefined = object()\n    b = DefaultMunch(undefined, bar='baz', lol={})\n    assert b.foo is undefined\n    assert b['foo'] is undefined\n\n\ndef test_setattr_default():\n    b = DefaultMunch(foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n    assert b['values'] is None\n\n    assert b.__default__ is None\n    assert '__default__' not in b\n\n\ndef test_delattr_default():\n    b = DefaultMunch(lol=42)\n    del b.lol\n\n    assert b.lol is None\n    assert b['lol'] is None\n\n\ndef test_pickle_default():\n    b = DefaultMunch.fromDict({\"a\": \"b\"})\n    assert pickle.loads(pickle.dumps(b)) == b\n\n\ndef test_fromDict_default():\n    undefined = object()\n    b = DefaultMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    assert b.urmom.sez.what == 'what'\n    assert b.urmom.sez.foo is undefined\n\n\ndef test_copy_default():\n    undefined = object()\n    m = DefaultMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n    assert c.urmom.sez.foo is undefined\n    assert c.urmom.sez.__undefined__ is undefined\n\n\ndef test_munchify_default():\n    undefined = object()\n    b = munchify(\n        {'urmom': {'sez': {'what': 'what'}}},\n        lambda d: DefaultMunch(undefined, d))\n    assert b.urmom.sez.what == 'what'\n    assert b.urdad is undefined\n    assert b.urmom.sez.ni is undefined\n\n\ndef test_repr_default():\n    b = DefaultMunch(foo=DefaultMunch(lol=True), ponies='are pretty!')\n    assert repr(b).startswith(\"DefaultMunch(None, {'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n\n\ndef test_getattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, bar='baz', lol={})\n    assert b.foo is None\n    assert b['foo'] is None\n\n    assert b.bar == 'baz'\n    assert getattr(b, 'bar') == 'baz'\n    assert b['bar'] == 'baz'\n    assert b.lol is b['lol']\n    assert b.lol is getattr(b, 'lol')\n\n    undefined = object()\n    default = lambda: undefined\n    b = DefaultFactoryMunch(default, bar='baz', lol={})\n    assert b.foo is undefined\n    assert b['foo'] is undefined\n\n    default = lambda: object()\n    b = DefaultFactoryMunch(default, bar='baz', lol={})\n    assert b.foo is not b.baz\n    assert b.foo is b['foo']\n    assert b.foobar is b.foobar\n\n    b = DefaultFactoryMunch(list)\n    assert b.foo == []\n    b.foo.append('bar')\n    assert b.foo == ['bar']\n    assert b.default_factory is list\n\n\ndef test_setattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, foo='bar', this_is='useful when subclassing')\n    assert hasattr(b.values, '__call__')\n\n    b.values = 'uh oh'\n    assert b.values == 'uh oh'\n    assert b['values'] is None\n\n    assert b.default_factory() is None\n    assert 'default_factory' not in b\n\n\ndef test_delattr_default_factory():\n    b = DefaultFactoryMunch(lambda: None, lol=42)\n    del b.lol\n\n    assert b.lol is None\n    assert b['lol'] is None\n\n\ndef test_fromDict_default_factory():\n    obj = object()\n    undefined = lambda: obj\n    b = DefaultFactoryMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    assert b.urmom.sez.what == 'what'\n    assert b.urmom.sez.foo is undefined()\n\n\ndef test_copy_default_factory():\n    undefined = lambda: object()\n    m = DefaultFactoryMunch.fromDict({'urmom': {'sez': {'what': 'what'}}}, undefined)\n    c = m.copy()\n    assert c is not m\n    assert c.urmom is not m.urmom\n    assert c.urmom.sez is not m.urmom.sez\n    assert c.urmom.sez.what == 'what'\n    assert c == m\n\n\ndef test_munchify_default_factory():\n    undefined = lambda: object()\n    b = munchify(\n        {'urmom': {'sez': {'what': 'what'}}},\n        lambda d: DefaultFactoryMunch(undefined, d))\n    assert b.urmom.sez.what == 'what'\n    assert b.urdad is not undefined()\n    assert b.urmom.sez.ni is not b.urdad\n\n\ndef test_munchify_cycle():\n    # dict1 -> dict2 -> dict1\n    x = dict(id=\"x\")\n    y = dict(x=x, id=\"y\")\n    x['y'] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y.id == \"y\"\n    assert m.y.x is m\n\n    # dict -> list -> dict\n    x = dict(id=\"x\")\n    y = [\"y\", x]\n    x[\"y\"] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1] is m\n\n    # dict -> tuple -> dict\n    x = dict(id=\"x\")\n    y = (\"y\", x)\n    x[\"y\"] = y\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1] is m\n\n    # dict1 -> list -> dict2 -> list\n    z = dict(id=\"z\")\n    y = [\"y\", z]\n    z[\"y\"] = y\n    x = dict(id=\"x\", y=y)\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1].id == \"z\"\n    assert m.y[1].y is m.y\n\n    # dict1 -> tuple -> dict2 -> tuple\n    z = dict(id=\"z\")\n    y = (\"y\", z)\n    z[\"y\"] = y\n    x = dict(id=\"x\", y=y)\n\n    m = munchify(x)\n    assert m.id == \"x\"\n    assert m.y[0] == \"y\"\n    assert m.y[1].id == \"z\"\n    assert m.y[1].y is m.y\n\ndef test_unmunchify_cycle():\n    # munch -> munch -> munch\n    x = Munch(id=\"x\")\n    y = Munch(x=x, id=\"y\")\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][\"id\"] == \"y\"\n    assert d[\"y\"][\"x\"] is d\n\n    # munch -> list -> munch\n    x = Munch(id=\"x\")\n    y = [\"y\", x]\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1] is d\n\n    # munch -> tuple -> munch\n    x = Munch(id=\"x\")\n    y = (\"y\", x)\n    x.y = y\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1] is d\n\n    # munch1 -> list -> munch2 -> list\n    z = Munch(id=\"z\")\n    y = [\"y\", z]\n    z.y = y\n    x = Munch(id=\"x\", y=y)\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1][\"id\"] == \"z\"\n    assert d[\"y\"][1][\"y\"] is d[\"y\"]\n\n    # munch1 -> tuple -> munch2 -> tuple\n    z = Munch(id=\"z\")\n    y = (\"y\", z)\n    z.y = y\n    x = Munch(id=\"x\", y=y)\n\n    d = unmunchify(x)\n    assert d[\"id\"] == \"x\"\n    assert d[\"y\"][0] == \"y\"\n    assert d[\"y\"][1][\"id\"] == \"z\"\n    assert d[\"y\"][1][\"y\"] is d[\"y\"]\n\n\ndef test_repr_default_factory():\n    b = DefaultFactoryMunch(list, foo=DefaultFactoryMunch(list, lol=True), ponies='are pretty!')\n    assert repr(b).startswith(\"DefaultFactoryMunch(list, {'\")\n    assert \"'ponies': 'are pretty!'\" in repr(b)\n\n    assert eval(repr(b)) == b  # pylint: disable=eval-used\n\n\ndef test_pickling_unpickling_nested():\n    m = {'a': {'b': 'c'}}\n    m = munchify(m)\n    assert m == Munch({'a': Munch({'b': 'c'})})\n    assert isinstance(m.a, Munch)\n    result = pickle.loads(pickle.dumps(m))\n    assert result == m\n    assert isinstance(result.a, Munch)\n\n\ndef test_setitem_dunder_for_subclass():\n\n    def test_class(cls, *args):\n        class CustomMunch(cls):\n            def __setitem__(self, k, v):\n                super().__setitem__(k, [v] * 2)\n        custom_munch = CustomMunch(*args, a='foo')\n        assert custom_munch.a == ['foo', 'foo']\n        regular_dict = {}\n        regular_dict.update(custom_munch)\n        assert regular_dict['a'] == ['foo', 'foo']\n        assert repr(regular_dict) == \"{'a': ['foo', 'foo']}\"\n        custom_munch.setdefault('bar', 'baz')\n        assert custom_munch.bar == ['baz', 'baz']\n\n    test_class(Munch)\n    test_class(DefaultFactoryMunch, list)\n    test_class(DefaultMunch, 42)\n\n\ndef test_getitem_dunder_for_subclass():\n    class CustomMunch(Munch):\n        def __getitem__(self, k):\n            return 42\n\n    custom_munch = CustomMunch(a='foo')\n    custom_munch.update({'b': 1})\n    assert custom_munch.a == 42\n    assert custom_munch.get('b') == 42\n    assert custom_munch.copy() == Munch(a=42, b=42)\n\n\n@pytest.mark.usefixtures(\"yaml\")\ndef test_get_default_value(munch_obj):\n    assert munch_obj.get(\"fake_key\", \"default_value\") == \"default_value\"\n    assert isinstance(munch_obj.toJSON(), str)\n    assert isinstance(munch_obj.toYAML(), str)\n    munch_obj.copy()\n    data = munch_obj.toDict()\n    munch_cls = type(munch_obj)\n    kwargs = {} if munch_cls != DefaultFactoryMunch else {\"default_factory\": munch_obj.default_factory}\n    munch_cls.fromDict(data, **kwargs)\n\n\ndef test_munchify_tuple_list():\n    data = ([{'A': 'B'}],)\n    actual = munchify(data)\n    expected = ([Munch(A='B')],)\n    assert actual == expected\n\n\ndef test_munchify_tuple_list_more_elements():\n    data = (1, 2, [{'A': 'B'}])\n    actual = munchify(data)\n    expected = (1, 2, [Munch({'A': 'B'})])\n    assert actual == expected\n"
    },
    {
      "path": "Infinidat_munch/tests/conftest.py",
      "content": "import pytest\nimport munch\n\n\n@pytest.fixture(name='yaml')\ndef yaml_module():\n    try:\n        import yaml  # pylint: disable=import-outside-toplevel\n        return yaml\n    except ImportError:\n        pass\n    pytest.skip(\"Module 'PyYAML' is required\")\n\n\n@pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n                        munch.RecursiveMunch])\ndef munch_obj(request):\n    cls = request.param\n    args = tuple()\n    if cls == munch.DefaultFactoryMunch:\n        args = args + (lambda: None,)\n    return cls(*args, hello=\"world\", number=5)\n"
    },
    {
      "path": "Infinidat_munch/tests/__init__.py",
      "content": ""
    },
    {
      "path": "Infinidat_munch/tests/test_readme.py",
      "content": "import doctest\nimport os\nimport pytest\n\n_HERE = os.path.abspath(os.path.dirname(__file__))\n_README_PATH = os.path.join(_HERE, '..', 'README.md')\nassert os.path.exists(_README_PATH)\n\n\n@pytest.mark.usefixtures(\"yaml\")\ndef test_readme():\n    globs = {\n        'print_function': print\n    }\n    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n    assert not result.failed\n"
    },
    {
      "path": "Infinidat_munch/tests/test_yaml.py",
      "content": "import pytest\nfrom munch import Munch, DefaultMunch\n\n\ndef test_from_yaml(yaml):\n    data = yaml.load('''\n    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n    Block style: !munch\n      Clark : Evans\n      Brian : Ingerson\n      Oren  : Ben-Kiki\n    ''', Loader=yaml.FullLoader)\n    assert data == {\n        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n    }\n\n\ndef test_to_yaml_safe(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    dumped = yaml.safe_dump(b, default_flow_style=True)\n    assert dumped == '{foo: [bar, {lol: true}], hello: 42}\\n'\n\n\ndef test_to_yaml(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    dumped = yaml.dump(b, default_flow_style=True)\n    assert dumped == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n\n\ndef test_toYAML(yaml):\n    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    assert yaml.safe_dump(b, default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n    assert b.toYAML(default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n    assert yaml.dump(b, default_flow_style=True) == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n    assert b.toYAML(Dumper=yaml.Dumper, default_flow_style=True) == \\\n        '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n\n\n@pytest.mark.usefixtures('yaml')\ndef test_fromYAML():\n    # pylint: disable=unidiomatic-typecheck\n    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n    obj = Munch.fromYAML(yaml_str)\n    assert type(obj) == Munch\n    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n    assert obj.toYAML() == yaml_str\n\n\n@pytest.mark.usefixtures('yaml')\ndef test_fromYAML_default_munch():\n    # pylint: disable=unidiomatic-typecheck\n    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n    default_value = object()\n    obj = DefaultMunch.fromYAML(yaml_str, default_value)\n    assert type(obj) == DefaultMunch\n    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n    assert obj['not_exist'] is default_value\n    assert obj.not_exist is default_value\n    assert obj.toYAML() == yaml_str\n"
    }
  ],
  "ErrorMessage": "============================================================================================== FAILURES ===============================================================================================\n___________________________________________________________________________________________ test_from_yaml ____________________________________________________________________________________________\n\nyaml = <module 'yaml' from '/home/user/anaconda3/envs/py39/lib/python3.9/site-packages/yaml/__init__.py'>\n\n    def test_from_yaml(yaml):\n>       data = yaml.load('''\n        Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n        Block style: !munch\n          Clark : Evans\n          Brian : Ingerson\n          Oren  : Ben-Kiki\n        ''', Loader=yaml.FullLoader)\n\ntests/test_yaml.py:6: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/__init__.py:81: in load\n    return loader.get_single_data()\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:51: in get_single_data\n    return self.construct_document(node)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:60: in construct_document\n    for dummy in generator:\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:413: in construct_yaml_map\n    value = self.construct_mapping(node)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:218: in construct_mapping\n    return super().construct_mapping(node, deep=deep)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:143: in construct_mapping\n    value = self.construct_object(value_node, deep=deep)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:100: in construct_object\n    data = constructor(self, node)\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n\nself = <yaml.loader.FullLoader object at 0x76a0b06717f0>\nnode = MappingNode(tag='!munch.Munch', value=[(ScalarNode(tag='tag:yaml.org,2002:str', value='Clark'), ScalarNode(tag='tag:ya...), (ScalarNode(tag='tag:yaml.org,2002:str', value='Oren'), ScalarNode(tag='tag:yaml.org,2002:str', value='Ben-Kiki'))])\n\n    def construct_undefined(self, node):\n>       raise ConstructorError(None, None,\n                \"could not determine a constructor for the tag %r\" % node.tag,\n                node.start_mark)\nE       yaml.constructor.ConstructorError: could not determine a constructor for the tag '!munch.Munch'\nE         in \"<unicode string>\", line 2, column 17:\nE               Flow style: !munch.Munch { Clark: Evans, Bri ... \nE                           ^\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:427: ConstructorError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_yaml.py::test_from_yaml - yaml.constructor.ConstructorError: could not determine a constructor for the tag '!munch.Munch'\n==================================================================================== 1 failed, 108 passed in 0.09s ====================================================================================",
  "Patch": "--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,8 +510,11 @@\n \n     def unmunchify_cycles(obj):\n         # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n-        if id(obj) in seen.keys():\n+        try:\n             return seen[id(obj)]\n+        except KeyError:\n+            pass\n+\n         # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n         seen[id(obj)] = partial = pre_unmunchify(obj)\n         # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n@@ -631,6 +634,7 @@\n             # This code supports both PyYAML 4.x and 5.x versions\n             continue\n         yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n+        yaml.add_constructor('!munch.Munch', from_yaml, Loader=LoaderCls)\n \n     SafeRepresenter.add_representer(Munch, to_yaml_safe)\n     SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -94,6 +94,9 @@\n def test_toDict():\n     b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n     assert sorted(b.toDict().items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n+\n+def test_dict_property():\n+    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n     assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n \n def test_repr():\n",
  "BuggyCodeLocation": [
    {
      "file": "Infinidat_munch/munch/__init__.py",
      "function": {
        "490": "unmunchify"
      },
      "content_all": {
        "510": "\n",
        "511": "    def unmunchify_cycles(obj):\n",
        "512": "        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n",
        "513": "        if id(obj) in seen.keys():\n",
        "514": "            return seen[id(obj)]\n",
        "515": "        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n",
        "516": "        seen[id(obj)] = partial = pre_unmunchify(obj)\n",
        "517": "        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n"
      },
      "content_change": {
        "513": "        if id(obj) in seen.keys():\n"
      }
    },
    {
      "file": "Infinidat_munch/munch/__init__.py",
      "function": {
        "511": "unmunchify_cycles"
      },
      "content_all": {
        "510": "\n",
        "511": "    def unmunchify_cycles(obj):\n",
        "512": "        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n",
        "513": "        if id(obj) in seen.keys():\n",
        "514": "            return seen[id(obj)]\n",
        "515": "        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n",
        "516": "        seen[id(obj)] = partial = pre_unmunchify(obj)\n",
        "517": "        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n"
      },
      "content_change": {
        "513": "        if id(obj) in seen.keys():\n"
      }
    },
    {
      "file": "Infinidat_munch/munch/__init__.py",
      "function": null,
      "content_all": {
        "631": "            # This code supports both PyYAML 4.x and 5.x versions\n",
        "632": "            continue\n",
        "633": "        yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n",
        "634": "\n",
        "635": "    SafeRepresenter.add_representer(Munch, to_yaml_safe)\n",
        "636": "    SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n"
      },
      "content_change": {}
    },
    {
      "file": "Infinidat_munch/tests/test_munch.py",
      "function": null,
      "content_all": {
        "97": "    assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n"
      },
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Handling of Unmunchify Cycles and YAML Deserialization Issues",
    "description": "### Description\nThere are a couple of issues that need to be addressed related to the unmunchify process and YAML deserialization in the Munch library.\n\n#### 1. Unmunchify Cycles Handling\nCurrently, during the unmunchification process, there is a problem when handling objects that have already been seen. This potentially raises an error due to incorrect handling when the object ID is not found in the `seen` dictionary. This issue affects the robustness of the `unmunchify` function, causing it to fail when it encounters cycles in complex structures, such as nested dictionaries or lists.\n\n#### 2. YAML Deserialization\nThe library does not support some YAML tags like `!munch.Munch`, which are essential for correctly deserializing objects. This leads to incomplete or incorrect deserialization of YAML strings that should convert tags to Munch objects.\n\n### Impact\n- Code that relies on the `unmunchify` function may crash or exhibit undefined behavior when dealing with certain cyclic structures, affecting the reliability of applications using this library.\n- Incorrect or incomplete deserialization of YAML configurations can lead to data loss or unexpected behavior in applications, particularly in those that heavily rely on YAML for configuration or data exchange.\n\n### Steps to Reproduce\n1. Create a Munch object with cyclic references and attempt to unmunchify it. Observe the failure or incorrect results.\n2. Attempt to deserialize a YAML string containing `!munch.Munch` tags and observe that the resulting objects are not properly converted.\n\n### Expected Behavior\n- The `unmunchify` function should handle cyclic structures gracefully without raising errors.\n- YAML deserialization should correctly interpret and convert `!munch.Munch` tags into appropriate Munch objects.\n\n### Suggested Solutions\nWhile specific code changes are not provided here, addressing the above issues will likely involve:\n1. Updating the `unmunchify` function to correctly handle cases where object IDs are not found in the `seen` dictionary.\n2. Adding support for the `!munch.Munch` tag in the YAML deserialization process.\n\n### Additional Information\nRelevant tests should be created or updated to cover these cases, ensuring that both unmunchification and YAML deserialization work as expected.",
    "explanation": "### Summary of the Issue\n\nThe issue involves two key problems related to the Munch library:\n1. **Handling of Unmunchify Cycles**: There is an issue with the `unmunchify` function when it encounters cyclic references in complex structures like nested dictionaries or lists. This can lead to errors when object IDs are not found in a tracking dictionary called `seen`.\n2. **YAML Deserialization**: The library lacks support for deserializing certain YAML tags, specifically `!munch.Munch`. This omission leads to improper conversion of YAML strings into Munch objects, which might result in data loss or unexpected behavior for applications using YAML configurations.\n\n### Content of the Commit\n\nThe commit addresses the two problems as follows:\n\n#### 1. **Updating `unmunchify` Function for Cycles**\n- **Problem**: The original `unmunchify` function could throw errors when it encounters cyclic references, because it did not correctly handle the case when an object ID was not found in the `seen` dictionary.\n- **Solution**: The commit updates the `unmunchify` function to use a `try-except` block. This ensures that instead of raising an error when an object ID is not found in the `seen` dictionary, the function continues processing. This change enhances the function's robustness, allowing it to gracefully handle cyclic references without crashing.\n\n#### 2. **Enhancing YAML Deserialization**\n- **Problem**: The `Munch` library did not support the `!munch.Munch` YAML tag, which is necessary for correctly deserializing YAML strings into Munch objects. Without this support, objects serialized with this tag could not be deserialized correctly, leading to incomplete or incorrect data reconstruction.\n- **Solution**: The commit adds support for the `!munch.Munch` tag by registering a new YAML constructor. This ensures that YAML deserialization correctly converts `!munch.Munch` tags into Munch objects. This enhancement ensures complete and accurate deserialization of YAML strings, maintaining the integrity of data.\n\n### Explanation of the Solution\n\n#### Handling of Unmunchify Cycles\n- **Cause**: The `unmunchify` function was not resilient to cycles in object graphs. It would raise an error if it attempted to access an object ID not present in the `seen` dictionary, which tracks objects that have already been processed.\n- **Solution**: By introducing a `try-except` block, the function now gracefully handles the absence of an object ID in the `seen` dictionary. If an object ID is not found, the function continues processing without raising an error, completing the unmunchification process successfully.\n\n#### YAML Deserialization\n- **Cause**: The absence of support for the `!munch.Munch` tag in YAML deserialization meant that certain YAML-encoded Munch objects could not be reconstructed correctly. This could lead to data loss or inaccuracies.\n- **Solution**: By registering a new YAML constructor for the `!munch.Munch` tag, the commit ensures that the Munch library can correctly interpret and convert these tags during the YAML deserialization process. This guarantees that the objects are reconstructed as intended, preserving data integrity and ensuring that the applications relying on YAML configurations work as expected.\n\n### Conclusion\n\nThe commit effectively addresses both reported issues:\n- It strengthens the `unmunchify` function by improving its handling of cyclic references, thus enhancing the robustness and reliability of the library for complex data structures.\n- It adds necessary support for the `!munch.Munch` YAML tag, ensuring complete and accurate deserialization of YAML-configured Munch objects.\n\nOverall, these changes improve the library's functionality, making it more resilient and reliable for users handling complex data structures and configurations."
  },
  "Explain": "### Summary of the Issue\n\nThe issue involves two key problems related to the Munch library:\n1. **Handling of Unmunchify Cycles**: There is an issue with the `unmunchify` function when it encounters cyclic references in complex structures like nested dictionaries or lists. This can lead to errors when object IDs are not found in a tracking dictionary called `seen`.\n2. **YAML Deserialization**: The library lacks support for deserializing certain YAML tags, specifically `!munch.Munch`. This omission leads to improper conversion of YAML strings into Munch objects, which might result in data loss or unexpected behavior for applications using YAML configurations.\n\n### Content of the Commit\n\nThe commit addresses the two problems as follows:\n\n#### 1. **Updating `unmunchify` Function for Cycles**\n- **Problem**: The original `unmunchify` function could throw errors when it encounters cyclic references, because it did not correctly handle the case when an object ID was not found in the `seen` dictionary.\n- **Solution**: The commit updates the `unmunchify` function to use a `try-except` block. This ensures that instead of raising an error when an object ID is not found in the `seen` dictionary, the function continues processing. This change enhances the function's robustness, allowing it to gracefully handle cyclic references without crashing.\n\n#### 2. **Enhancing YAML Deserialization**\n- **Problem**: The `Munch` library did not support the `!munch.Munch` YAML tag, which is necessary for correctly deserializing YAML strings into Munch objects. Without this support, objects serialized with this tag could not be deserialized correctly, leading to incomplete or incorrect data reconstruction.\n- **Solution**: The commit adds support for the `!munch.Munch` tag by registering a new YAML constructor. This ensures that YAML deserialization correctly converts `!munch.Munch` tags into Munch objects. This enhancement ensures complete and accurate deserialization of YAML strings, maintaining the integrity of data.\n\n### Explanation of the Solution\n\n#### Handling of Unmunchify Cycles\n- **Cause**: The `unmunchify` function was not resilient to cycles in object graphs. It would raise an error if it attempted to access an object ID not present in the `seen` dictionary, which tracks objects that have already been processed.\n- **Solution**: By introducing a `try-except` block, the function now gracefully handles the absence of an object ID in the `seen` dictionary. If an object ID is not found, the function continues processing without raising an error, completing the unmunchification process successfully.\n\n#### YAML Deserialization\n- **Cause**: The absence of support for the `!munch.Munch` tag in YAML deserialization meant that certain YAML-encoded Munch objects could not be reconstructed correctly. This could lead to data loss or inaccuracies.\n- **Solution**: By registering a new YAML constructor for the `!munch.Munch` tag, the commit ensures that the Munch library can correctly interpret and convert these tags during the YAML deserialization process. This guarantees that the objects are reconstructed as intended, preserving data integrity and ensuring that the applications relying on YAML configurations work as expected.\n\n### Conclusion\n\nThe commit effectively addresses both reported issues:\n- It strengthens the `unmunchify` function by improving its handling of cyclic references, thus enhancing the robustness and reliability of the library for complex data structures.\n- It adds necessary support for the `!munch.Munch` YAML tag, ensuring complete and accurate deserialization of YAML-configured Munch objects.\n\nOverall, these changes improve the library's functionality, making it more resilient and reliable for users handling complex data structures and configurations.",
  "Source": "Human",
  "Token": 1498,
  "Command": [
    "pytest tests"
  ],
  "FilteredCode": [
    {
      "path": "Infinidat_munch/munch/__init__.py",
      "content": "1 \"\"\" Munch is a subclass of dict with attribute-style access.\n2 \n3     >>> b = Munch()\n4     >>> b.hello = 'world'\n5     >>> b.hello\n6     'world'\n7     >>> b['hello'] += \"!\"\n8     >>> b.hello\n9     'world!'\n10     >>> b.foo = Munch(lol=True)\n11     >>> b.foo.lol\n12     True\n13     >>> b.foo is b['foo']\n14     True\n15 \n16     It is safe to import * from this module:\n17 \n18         __all__ = ('Munch', 'munchify','unmunchify')\n19 \n20     un/munchify provide dictionary conversion; Munches can also be\n21     converted via Munch.to/fromDict().\n22 \"\"\"\n23 \n24 from collections.abc import Mapping\n25 \n26 try:\n27     # For python 3.8 and later\n28     import importlib.metadata as importlib_metadata\n29 except ImportError:\n30     # For everyone else\n31     import importlib_metadata\n32 try:\n33     __version__ = importlib_metadata.version(__name__)\n34 except importlib_metadata.PackageNotFoundError:\n35     # package is not installed\n36     __version__ = \"0.0.0\"\n37 \n38 \n39 try:\n40     VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\n41 except ValueError:\n42     VERSION = (0, 0, 0)\n43 \n44 __all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n45 \n46 \n47 \n48 class Munch(dict):\n49     \"\"\" A dictionary that provides attribute-style access.\n50 \n51         >>> b = Munch()\n52         >>> b.hello = 'world'\n53         >>> b.hello\n54         'world'\n55         >>> b['hello'] += \"!\"\n56         >>> b.hello\n57         'world!'\n58         >>> b.foo = Munch(lol=True)\n59         >>> b.foo.lol\n60         True\n61         >>> b.foo is b['foo']\n62         True\n63 \n64         A Munch is a subclass of dict; (...truncated)"
    },
    {
      "path": "Infinidat_munch/tests/test_munch.py",
      "content": "1 # pylint: disable=unnecessary-lambda\n2 import json\n3 import pickle\n4 from collections import namedtuple\n5 \n6 import pytest\n7 from munch import DefaultFactoryMunch, AutoMunch, DefaultMunch, Munch, munchify, unmunchify\n8 \n9 \n10 def test_base():\n11     b = Munch()\n12     b.hello = 'world'\n13     assert b.hello == 'world'\n14     b['hello'] += \"!\"\n15     assert b.hello == 'world!'\n16     b.foo = Munch(lol=True)\n17     assert b.foo.lol is True\n18     assert b.foo is b['foo']\n19 \n20     assert sorted(b.keys()) == ['foo', 'hello']\n21 \n22     b.update({'ponies': 'are pretty!'}, hello=42)\n23     assert b == Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n24 \n25     assert sorted([(k, b[k]) for k in b]) == [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n26 \n27     format_munch = Munch(knights='lolcats', ni='can haz')\n28     assert \"The {knights} who say {ni}!\".format(**format_munch) == 'The lolcats who say can haz!'\n29 \n30 \n31 def test_contains():\n32     b = Munch(ponies='are pretty!')\n33     assert 'ponies' in(...truncated)"
    },
    {
      "path": "Infinidat_munch/tests/test_yaml.py",
      "content": "1 import pytest\n2 from munch import Munch, DefaultMunch\n3 \n4 \n5 def test_from_yaml(yaml):\n6     data = yaml.load('''\n7     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n8     Block style: !munch\n9       Clark : Evans\n10       Brian : Ingerson\n11       Oren  : Ben-Kiki\n12     ''', Loader=yaml.FullLoader)\n13     assert data == {\n14         'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n15         'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n16     }\n17 \n18 \n19 def test_to_yaml_safe(yaml):\n20     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n21     dumped = yaml.safe_dump(b, default_flow_st(...truncated)"
    },
    {
      "path": "Infinidat_munch/tests/conftest.py",
      "content": "1 import pytest\n2 import munch\n3 \n4 \n5 @pytest.fixture(name='yaml')\n6 def yaml_module():\n7     try:\n8         import yaml  # pylint: disable=import-outside-toplevel\n9         return yaml\n10     except ImportError:\n11         pass\n12     pytest.skip(\"Module 'PyYAML' is required\")\n13 \n14 \n15 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n16                         munch.RecursiveMunch])\n17 def munch_obj(request):\n18     cls = request.param\n19     args = tuple()\n20     if cls == munch.DefaultFactoryMunch:\n21         args = args + (lambda: None,)\n22     return cls(*args, hello=\"world\", number=5)"
    },
    {
      "path": "Infinidat_munch/tests/test_readme.py",
      "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _HERE = os.path.abspath(os.path.dirname(__file__))\n6 _README_PATH = os.path.join(_HERE, '..', 'README.md')\n7 assert os.path.exists(_README_PATH)\n8 \n9 \n10 @pytest.mark.usefixtures(\"yaml\")\n11 def test_readme():\n12     globs = {\n13         'print_function': print\n14     }\n15     result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n16     assert not result.failed"
    },
    {
      "path": "Infinidat_munch/setup.py",
      "content": "1 from setuptools import setup\n2 \n3 \n4 setup(\n5     setup_requires=['pbr>=3.0', 'setuptool(...truncated)"
    }
  ],
  "TokenAll": 13453,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Easy",
    "issue_origin": {
      "title": "Unconditional use of external 'yaml' module",
      "description": "In the 'Infinidat_munch/tests/test_yaml.py' file, the tests assume that the 'yaml' module is always available. However, the 'yaml' module is not a standard library and may not be installed in the environment. This could lead to test failures. Consider adding a conditional import or a check to verify if 'yaml' is available before running the tests.",
      "explanation": "### Issue Summary\n\nThe issue raised concerns about the `yaml` module's use in the test suite, specifically within the file `Infinidat_munch/tests/test_yaml.py`. The code assumes unconditionally that the `yaml` module is available for import and use. However, since `yaml` is not a standard library module but an external one, it may not be present in the environment where tests are run, which leads to test failures. This issue can be resolved by adding a conditional import or a check to ensure that the `yaml` module is available before executing the tests.\n\n### Commit Details\n\nBased on the provided code, a commit has been made to address this issue by modifying files to implement conditional checking for the `yaml` module. Here are the modified files and their relevant content:\n\n1. **`Infinidat_munch/tests/conftest.py`**:\n    ```python\n    1 import pytest\n    2 import munch\n    3 \n    4 \n    5 @pytest.fixture(name='yaml')\n    6 def yaml_module():\n    7     try:\n    8         import yaml  # pylint: disable=import-outside-toplevel\n    9         return yaml\n    10     except ImportError:\n    11         pass\n    12     pytest.skip(\"Module 'PyYAML' is required\")\n    13 \n    14 \n    15 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n    16                         munch.RecursiveMunch])\n    17 def munch_obj(request):\n    18     cls = request.param\n    19     args = tuple()\n    20     if cls == munch.DefaultFactoryMunch:\n    21         args = args + (lambda: None,)\n    22     return cls(*args, hello=\"world\", number=5)\n    ```\n\n2. **`Infinidat_munch/tests/test_yaml.py`**:\n    ```python\n    1 import pytest\n    2 from munch import Munch, DefaultMunch\n    3 \n    4 \n    5 def test_from_yaml(yaml):\n    6     data = yaml.load('''\n    7     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n    8     Block style: !munch\n    9       Clark : Evans\n    10       Brian : Ingerson\n    11       Oren  : Ben-Kiki\n    12     ''', Loader=yaml.FullLoader)\n    13     assert data == {\n    14         'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n    15         'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n    16     }\n    17 \n    18 \n    19 def test_to_yaml_safe(yaml):\n    20     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n    21     dumped = yaml.safe_dump(b, default_flow_st...\n    ```\n\nIn the `conftest.py` file, a fixture named `yaml_module` is created under the `@pytest.fixture(name='yaml')` decorator. This fixture attempts to import the `yaml` module inside a `try` block:\n\n- If the import is successful, it returns the `yaml` module.\n- If the `ImportError` is raised (indicating that the module is not installed), it skips the test using `pytest.skip(\"Module 'PyYAML' is required\")`.\n\nThis fixture is transparently used in the test functions in `test_yaml.py` via dependency injection. Thus, before the tests in `test_yaml.py` run, the `yaml` fixture ensures that the `yaml` module is available. If it isn’t, the tests are skipped, avoiding any import errors and unintended test failures.\n\n### Solution Explanation\n\nThe solution leverages Pytest's fixture mechanism to create a controlled environment for the tests that require external modules. Here's the breakdown of how this commit solves the issue:\n\n1. **Conditional Import Handling**:\n   - Inside the `yaml_module` fixture function (`conftest.py`), the import for `yaml` is wrapped within a `try-except` block.\n   - If the module fails to import (raises `ImportError`), `pytest.skip()` is called, resulting in the test being skipped with a message indicating the requirement.\n   - This ensures the test suite continues running other tests if `yaml` is unavailable, instead of outright failing.\n\n2. **Fixture Usage in Tests**:\n   - The `test_yaml.py` file is modified to utilize the `yaml` fixture in test functions such as `test_from_yaml`.\n   - The test"
    },
    "issue_message": {
      "title": "YAML ConstructorError for !munch.Munch tag",
      "description": "The test 'test_from_yaml' in 'tests/test_yaml.py' fails with a ConstructorError because the 'yaml' module could not determine a constructor for the '!munch.Munch' tag. This indicates that a custom constructor for the !munch.Munch tag is not defined or registered with the PyYAML loader. Please define and add the appropriate constructor for the !munch.Munch tag to the YAML loader.",
      "explanation": "### Issue Summary\n\nThe issue at hand is a `ConstructorError` during the execution of the test `test_from_yaml` located in `tests/test_yaml.py`. The error message indicates that the PyYAML loader could not determine a constructor for the custom tag `!munch.Munch`. This error suggests that a custom constructor for the `!munch.Munch` tag is either not defined or not registered with the PyYAML loader. Consequently, when PyYAML encounters this tag during YAML deserialization, it doesn't know how to process it, leading to the failure of the test.\n\n### Detailed Analysis\n\n#### Problematic Code Segment\n\nThe error originates from the code segment in the `test_from_yaml` function:\n\n```python\ndata = yaml.load('''\n    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n    Block style: !munch\n      Clark : Evans\n      Brian : Ingerson\n      Oren  : Ben-Kiki\n    ''', Loader=yaml.FullLoader)\n```\n\n#### Error Message\n\nThe error message provides context about where the failure occurs:\n\n```\nE       yaml.constructor.ConstructorError: could not determine a constructor for the tag '!munch.Munch'\n```\n\nThis indicates that the YAML loader does not have a predefined constructor for the `!munch.Munch` tag.\n\n### Solution: Registering a Custom Constructor\n\n#### Commit Content\n\nThe solution involves defining a custom constructor for the `!munch.Munch` tag and registering it with the PyYAML loader. The necessary changes might look like the following:\n\n1. **Defining a Constructor Function**:\n   A constructor function is responsible for converting a YAML node into a `Munch` object.\n\n   ```python\n   def munch_constructor(loader, node):\n       # Parse the YAML node as a dictionary\n       value = loader.construct_mapping(node)\n       # Convert the dictionary into a Munch object\n       return Munch(value)\n   ```\n   \n2. **Registering the Constructor**:\n   The `add_constructor` method of the PyYAML library will be used to associate the `!munch.Munch` tag with the custom constructor function, typically in the module where the `Munch` class is defined.\n\n   ```python\n   import yaml\n   from munch import Munch\n\n   yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)\n   ```\n\n### Detailed Explanation of the Fix\n\nLet's break down how this commit will resolve the issue:\n\n1. **Custom Constructor Function**:\n    - The function `munch_constructor` takes `loader` and `node` as arguments.\n    - The `loader.construct_mapping(node)` method processes the YAML node and converts it into a standard Python dictionary.\n    - The dictionary is then passed to the `Munch` class constructor, creating a `Munch` object.\n  \n2. **Adding the Constructor**:\n    - The `yaml.add_constructor` function is used to register the `munch_constructor` with the `yaml.FullLoader`.\n    - This step binds the `!munch.Munch` tag in YAML to the `munch_constructor`.\n\n### How the Fix Solves the Issue\n\nBy defining and registering a custom constructor, the YAML loader will now recognize the `!munch.Munch` tag and correctly convert nodes tagged with it into `Munch` objects. Here's how it works during the test execution:\n\n1. When `yaml.load` is called with the provided string, the `FullLoader` encounters the `!munch.Munch` tag.\n2. The loader uses the `munch_constructor` registered for the `!munch.Munch` tag to convert the corresponding YAML nodes into `Munch` objects.\n3. The YAML is successfully loaded into a dictionary containing `Munch` objects, matching the expected output of the test case.\n\n### Updated Test Case with the Fix\n\nAfter adding the custom constructor definition and registration, the `test_from_yaml` function should pass as the YAML content will be correctly parsed into `Munch` objects.\n\nHere is an example of where you might place these modifications within the project structure:\n\n1. **Constructor Definition in `__init__.py`** (or another central module within `munch`):\n\n   ```python\n   import yaml\n   from munch import Munch\n\n   def munch_constructor(loader, node):\n       value = loader.construct_mapping(node)\n       return Munch(value)\n\n   yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)\n   ```\n\nThis update ensures that the `!munch.Munch` tag is properly handled by the PyYAML loader, allowing the `test_from_yaml` to pass.\n\nBy following these steps, the `ConstructorError` will be resolved, and the `test_from_yaml` function will function correctly."
    },
    "issue_ground": {
      "title": "Handling of Unmunchify Cycles and YAML Deserialization Issues",
      "description": "### Description\nThere are a couple of issues that need to be addressed related to the unmunchify process and YAML deserialization in the Munch library.\n\n#### 1. Unmunchify Cycles Handling\nCurrently, during the unmunchification process, there is a problem when handling objects that have already been seen. This potentially raises an error due to incorrect handling when the object ID is not found in the `seen` dictionary. This issue affects the robustness of the `unmunchify` function, causing it to fail when it encounters cycles in complex structures, such as nested dictionaries or lists.\n\n#### 2. YAML Deserialization\nThe library does not support some YAML tags like `!munch.Munch`, which are essential for correctly deserializing objects. This leads to incomplete or incorrect deserialization of YAML strings that should convert tags to Munch objects.\n\n### Impact\n- Code that relies on the `unmunchify` function may crash or exhibit undefined behavior when dealing with certain cyclic structures, affecting the reliability of applications using this library.\n- Incorrect or incomplete deserialization of YAML configurations can lead to data loss or unexpected behavior in applications, particularly in those that heavily rely on YAML for configuration or data exchange.\n\n### Steps to Reproduce\n1. Create a Munch object with cyclic references and attempt to unmunchify it. Observe the failure or incorrect results.\n2. Attempt to deserialize a YAML string containing `!munch.Munch` tags and observe that the resulting objects are not properly converted.\n\n### Expected Behavior\n- The `unmunchify` function should handle cyclic structures gracefully without raising errors.\n- YAML deserialization should correctly interpret and convert `!munch.Munch` tags into appropriate Munch objects.\n\n### Suggested Solutions\nWhile specific code changes are not provided here, addressing the above issues will likely involve:\n1. Updating the `unmunchify` function to correctly handle cases where object IDs are not found in the `seen` dictionary.\n2. Adding support for the `!munch.Munch` tag in the YAML deserialization process.\n\n### Additional Information\nRelevant tests should be created or updated to cover these cases, ensuring that both unmunchification and YAML deserialization work as expected.",
      "explanation": "### Summary of the Issue\n\nThe issue identified in the `Munch` library involves two main problems: \n1. **Unmunchify Cycles Handling**: The `unmunchify` function fails to handle cyclic structures properly, resulting in errors during unmunchification when the same object is encountered more than once.\n2. **YAML Deserialization**: The library does not support deserialization of YAML tags like `!munch.Munch`, resulting in errors and incorrect deserialization of YAML strings that rely on these tags.\n\n### Commit Content\n\nTo address these issues, the following changes were implemented:\n\n#### `Infinidat_munch/munch/__init__.py`\n\nHere, additional utility functions can be introduced to handle cyclic references properly during unmunchification. We could also define custom YAML constructors to handle `!munch.Munch` tags during deserialization.\n\n```python\nimport yaml\n\n# Fix for Unmunchify Cycles Handling\ndef unmunchify(value, seen=None):\n    if seen is None:\n        seen = {}\n\n    # To handle cycles, store each seen object by id\n    obj_id = id(value)\n    if obj_id in seen:\n        return seen[obj_id]\n\n    if isinstance(value, Munch):\n        result = {k: unmunchify(v, seen) for k, v in value.items()}\n        seen[obj_id] = result\n        return result\n    elif isinstance(value, list):\n        result = [unmunchify(i, seen) for i in value]\n        seen[obj_id] = result\n        return result\n    elif isinstance(value, dict):\n        result = {k: unmunchify(v, seen) for k, v in value.items()}\n        seen[obj_id] = result\n        return result\n    else:\n        return value\n\n# Custom constructor for YAML deserialization\ndef munch_constructor(loader, node):\n    value = loader.construct_mapping(node)\n    return Munch(value)\n\n# Add the custom constructor to the PyYAML loader\nyaml.add_constructor(u'!munch.Munch', munch_constructor)\nyaml.add_constructor(u'!munch', munch_constructor)\n```\n\n#### `Infinidat_munch/tests/test_munch.py`\n\nTests to cover the corrected behavior for cyclic structures.\n\n```python\nimport pytest\n\ndef test_unmunchify_cycle():\n    a = Munch()\n    b = Munch(ref=a)\n    a.ref = b\n    result = unmunchify(a)\n    assert result['ref']['ref'] == result\n```\n\n#### `Infinidat_munch/tests/test_yaml.py`\n\nTest cases to ensure correct YAML deserialization.\n\n```python\ndef test_from_yaml(yaml):\n    yaml_string = '''\n    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n    Block style: !munch\n      Clark : Evans\n      Brian : Ingerson\n      Oren  : Ben-Kiki\n    '''\n    data = yaml.load(yaml_string, Loader=yaml.FullLoader)\n    assert data == {\n        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n    }\n```\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Unmunchify Cycles Handling**:\n   - **Problem**: The original `unmunchify` function did not correctly manage objects that it had already encountered, leading to infinite loops and crashes when cyclic structures were present.\n   - **Solution**: By introducing a `seen` dictionary that maps object IDs to their unmunchified counterparts, the modified `unmunchify` function can now detect and handle cycles properly. When a cycle is detected (i.e., an object's ID is already present in the `seen` dictionary), the function returns the previously unmunchified object instead of processing it again. This eliminates infinite loops and crashes.\n\n2. **YAML Deserialization**:\n   - **Problem**: The `yaml` library did not know how to handle `!munch.Munch` tags since no constructors were defined for these tags, leading to `ConstructorError`.\n   - **Solution**: By defining the `munch_constructor` function and associating it with the `!munch.Munch` and `!munch` tags using `yaml.add_constructor`, the YAML deserializer can now convert these tags into appropriate `Munch` objects. This ensures that YAML configurations containing these tags are correctly deserialized into `Munch` instances, preventing data loss or unexpected behavior.\n\nThe modifications ensure that:\n- **Unmunchify**: The function can handle cyclic structures gracefully by using a memoization technique, making it robust against complex nested structures with cycles.\n- **YAML Deserialization**: YAML tags `!munch.Munch` and `!munch"
    },
    "issue_ground_truth": {
      "title": "Handling of Unmunchify Cycles and YAML Deserialization Issues",
      "description": "### Description\nThere are a couple of issues that need to be addressed related to the unmunchify process and YAML deserialization in the Munch library.\n\n#### 1. Unmunchify Cycles Handling\nCurrently, during the unmunchification process, there is a problem when handling objects that have already been seen. This potentially raises an error due to incorrect handling when the object ID is not found in the `seen` dictionary. This issue affects the robustness of the `unmunchify` function, causing it to fail when it encounters cycles in complex structures, such as nested dictionaries or lists.\n\n#### 2. YAML Deserialization\nThe library does not support some YAML tags like `!munch.Munch`, which are essential for correctly deserializing objects. This leads to incomplete or incorrect deserialization of YAML strings that should convert tags to Munch objects.\n\n### Impact\n- Code that relies on the `unmunchify` function may crash or exhibit undefined behavior when dealing with certain cyclic structures, affecting the reliability of applications using this library.\n- Incorrect or incomplete deserialization of YAML configurations can lead to data loss or unexpected behavior in applications, particularly in those that heavily rely on YAML for configuration or data exchange.\n\n### Steps to Reproduce\n1. Create a Munch object with cyclic references and attempt to unmunchify it. Observe the failure or incorrect results.\n2. Attempt to deserialize a YAML string containing `!munch.Munch` tags and observe that the resulting objects are not properly converted.\n\n### Expected Behavior\n- The `unmunchify` function should handle cyclic structures gracefully without raising errors.\n- YAML deserialization should correctly interpret and convert `!munch.Munch` tags into appropriate Munch objects.\n\n### Suggested Solutions\nWhile specific code changes are not provided here, addressing the above issues will likely involve:\n1. Updating the `unmunchify` function to correctly handle cases where object IDs are not found in the `seen` dictionary.\n2. Adding support for the `!munch.Munch` tag in the YAML deserialization process.\n\n### Additional Information\nRelevant tests should be created or updated to cover these cases, ensuring that both unmunchification and YAML deserialization work as expected.",
      "explanation": "### Summary of the Issue\n\nThe issue involves two key problems related to the Munch library:\n1. **Handling of Unmunchify Cycles**: There is an issue with the `unmunchify` function when it encounters cyclic references in complex structures like nested dictionaries or lists. This can lead to errors when object IDs are not found in a tracking dictionary called `seen`.\n2. **YAML Deserialization**: The library lacks support for deserializing certain YAML tags, specifically `!munch.Munch`. This omission leads to improper conversion of YAML strings into Munch objects, which might result in data loss or unexpected behavior for applications using YAML configurations.\n\n### Content of the Commit\n\nThe commit addresses the two problems as follows:\n\n#### 1. **Updating `unmunchify` Function for Cycles**\n- **Problem**: The original `unmunchify` function could throw errors when it encounters cyclic references, because it did not correctly handle the case when an object ID was not found in the `seen` dictionary.\n- **Solution**: The commit updates the `unmunchify` function to use a `try-except` block. This ensures that instead of raising an error when an object ID is not found in the `seen` dictionary, the function continues processing. This change enhances the function's robustness, allowing it to gracefully handle cyclic references without crashing.\n\n#### 2. **Enhancing YAML Deserialization**\n- **Problem**: The `Munch` library did not support the `!munch.Munch` YAML tag, which is necessary for correctly deserializing YAML strings into Munch objects. Without this support, objects serialized with this tag could not be deserialized correctly, leading to incomplete or incorrect data reconstruction.\n- **Solution**: The commit adds support for the `!munch.Munch` tag by registering a new YAML constructor. This ensures that YAML deserialization correctly converts `!munch.Munch` tags into Munch objects. This enhancement ensures complete and accurate deserialization of YAML strings, maintaining the integrity of data.\n\n### Explanation of the Solution\n\n#### Handling of Unmunchify Cycles\n- **Cause**: The `unmunchify` function was not resilient to cycles in object graphs. It would raise an error if it attempted to access an object ID not present in the `seen` dictionary, which tracks objects that have already been processed.\n- **Solution**: By introducing a `try-except` block, the function now gracefully handles the absence of an object ID in the `seen` dictionary. If an object ID is not found, the function continues processing without raising an error, completing the unmunchification process successfully.\n\n#### YAML Deserialization\n- **Cause**: The absence of support for the `!munch.Munch` tag in YAML deserialization meant that certain YAML-encoded Munch objects could not be reconstructed correctly. This could lead to data loss or inaccuracies.\n- **Solution**: By registering a new YAML constructor for the `!munch.Munch` tag, the commit ensures that the Munch library can correctly interpret and convert these tags during the YAML deserialization process. This guarantees that the objects are reconstructed as intended, preserving data integrity and ensuring that the applications relying on YAML configurations work as expected.\n\n### Conclusion\n\nThe commit effectively addresses both reported issues:\n- It strengthens the `unmunchify` function by improving its handling of cyclic references, thus enhancing the robustness and reliability of the library for complex data structures.\n- It adds necessary support for the `!munch.Munch` YAML tag, ensuring complete and accurate deserialization of YAML-configured Munch objects.\n\nOverall, these changes improve the library's functionality, making it more resilient and reliable for users handling complex data structures and configurations."
    },
    "location_origin": [
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "5": "test_from_yaml"
        },
        "content_all": {
          "4": "\n",
          "5": "def test_from_yaml(yaml):\n",
          "6": "    data = yaml.load('''\n",
          "7": "    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n",
          "8": "    Block style: !munch\n",
          "9": "      Clark : Evans\n",
          "10": "      Brian : Ingerson\n",
          "11": "      Oren  : Ben-Kiki\n",
          "12": "    ''', Loader=yaml.FullLoader)\n",
          "13": "    assert data == {\n",
          "14": "        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "15": "        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "16": "    }\n"
        },
        "content_change": {
          "5": "def test_from_yaml(yaml):\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "19": "test_to_yaml_safe"
        },
        "content_all": {
          "18": "\n",
          "19": "def test_to_yaml_safe(yaml):\n",
          "20": "    b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n",
          "21": "    dumped = yaml.safe_dump(b, default_flow_style=False)\n",
          "22": "\n",
          "23": "\n",
          "24": "\n"
        },
        "content_change": {
          "19": "def test_to_yaml_safe(yaml):\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/conftest.py",
        "function": {
          "6": "yaml_module"
        },
        "content_all": {
          "5": "\n",
          "6": "@pytest.fixture(name='yaml')\n",
          "7": "def yaml_module():\n",
          "8": "    try:\n",
          "9": "        import yaml  # pylint: disable=import-outside-toplevel\n",
          "10": "        return yaml\n",
          "11": "    except ImportError:\n",
          "12": "        pass\n",
          "13": "    pytest.skip(\"Module 'PyYAML' is required\")\n"
        },
        "content_change": {
          "6": "@pytest.fixture(name='yaml')\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "47": "register_yaml_constructor"
        },
        "content_all": {
          "44": "__all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')",
          "45": "\n",
          "46": "\n",
          "47": "def register_yaml_constructor():",
          "48": "    import yaml",
          "49": "    from munch import Munch",
          "50": "    def munch_constructor(loader, node):",
          "51": "        value = loader.construct_mapping(node)",
          "52": "        return Munch(value)",
          "53": "    yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)",
          "54": "\n",
          "55": "\n"
        },
        "content_change": {
          "47": "def register_yaml_constructor():",
          "48": "    import yaml",
          "49": "    from munch import Munch",
          "50": "    def munch_constructor(loader, node):",
          "51": "        value = loader.construct_mapping(node)",
          "52": "        return Munch(value)",
          "53": "    yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)"
        }
      },
      {
        "file": "Infinidat_munch/tests/conftest.py",
        "function": {
          "5": "yaml_module"
        },
        "content_all": {
          "1": "import pytest",
          "2": "import munch",
          "3": "\n",
          "4": "\n",
          "5": "@pytest.fixture(name='yaml')",
          "6": "def yaml_module():",
          "7": "    try:",
          "8": "        import yaml  # pylint: disable=import-outside-toplevel",
          "9": "        return yaml",
          "10": "    except ImportError:",
          "11": "        pass",
          "12": "    pytest.skip(\"Module 'PyYAML' is required\")",
          "13": "\n",
          "14": "\n",
          "15": "@pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,",
          "16": "                        munch.RecursiveMunch])",
          "17": "def munch_obj(request):",
          "18": "    cls = request.param",
          "19": "    args = tuple()",
          "20": "    if cls == munch.DefaultFactoryMunch:",
          "21": "        args = args + (lambda: None,)",
          "22": "    return cls(*args, hello=\"world\", number=5)"
        },
        "content_change": {
          "9": "        register_yaml_constructor()",
          "9.5": "        return yaml"
        }
      }
    ],
    "location_ground": [
      {
        "location": "```json\n[\n    {\n        \"file\": \"Infinidat_munch/munch/__init__.py\",\n        \"function\": {\n            \"1\": \"__init__.py\"\n        },\n        \"content_all\": {\n            \"1\": \"\\\"\\\"\\\" Munch is a subclass of dict with attribute-style access.\\n\",\n            \"2\": \"\\n\",\n            \"3\": \"    >>> b = Munch()\\n\",\n            \"4\": \"    >>> b.hello = 'world'\\n\",\n            \"5\": \"    >>> b.hello\\n\",\n            \"6\": \"    'world'\\n\",\n            \"7\": \"    >>> b['hello'] += \\\"!\\\"\\n\",\n            \"8\": \"    >>> b.hello\\n\",\n            \"9\": \"    'world!'\\n\",\n            \"10\": \"    >>> b.foo = Munch(lol=True)\\n\",\n            \"11\": \"    >>> b.foo.lol\\n\",\n            \"12\": \"    True\\n\",\n            \"13\": \"    >>> b.foo is b['foo']\\n\",\n            \"14\": \"    True\\n\",\n            \"15\": \"\\n\",\n            \"16\": \"    It is safe to import * from this module:\\n\",\n            \"17\": \"\\n\",\n            \"18\": \"        __all__ = ('Munch', 'munchify','unmunchify')\\n\",\n            \"19\": \"\\n\",\n            \"20\": \"    un/munchify provide dictionary conversion; Munches can also be\\n\",\n            \"21\": \"    converted via Munch.to/fromDict().\\n\",\n            \"22\": \"\\\"\\\"\\\"\\n\",\n            \"23\": \"\\n\",\n            \"24\": \"from collections.abc import Mapping\\n\",\n            \"25\": \"\\n\",\n            \"26\": \"try:\\n\",\n            \"27\": \"    # For python 3.8 and later\\n\",\n            \"28\": \"    import importlib.metadata as importlib_metadata\\n\",\n            \"29\": \"except ImportError:\\n\",\n            \"30\": \"    # For everyone else\\n\",\n            \"31\": \"    import importlib_metadata\\n\",\n            \"32\": \"try:\\n\",\n            \"33\": \"    __version__ = importlib_metadata.version(__name__)\\n\",\n            \"34\": \"except importlib_metadata.PackageNotFoundError:\\n\",\n            \"35\": \"    # package is not installed\\n\",\n            \"36\": \"    __version__ = \\\"0.0.0\\\"\\n\",\n            \"37\": \"\\n\",\n            \"38\": \"\\n\",\n            \"39\": \"try:\\n\",\n            \"40\": \"    VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\\n\",\n            \"41\": \"except ValueError:\\n\",\n            \"42\": \"    VERSION = (0, 0, 0)\\n\",\n            \"43\": \"\\n\",\n            \"44\": \"__all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\\n\",\n            \"45\": \"\\n\",\n            \"46\": \"\\n\",\n            \"47\": \"\\n\",\n            \"48\": \"class Munch(dict):\\n\",\n            \"49\": \"    \\\"\\\"\\\" A dictionary that provides attribute-style access.\\n\",\n            \"50\": \"\\n\",\n            \"51\": \"        >>> b = Munch()\\n\",\n            \"52\": \"        >>> b.hello = 'world'\\n\",\n            \"53\": \"        >>> b.hello\\n\",\n            \"54\": \"        'world'\\n\",\n            \"55\": \"        >>> b['hello'] += \\\"!\\\"\\n\",\n            \"56\": \"        >>> b.hello\\n\",\n            \"57\": \"        'world!'\\n\",\n            \"58\": \"        >>> b.foo = Munch(lol=True)\\n\",\n            \"59\": \"        >>> b.foo.lol\\n\",\n            \"60\": \"        True\\n\",\n            \"61\": \"        >>> b.foo is b['foo']\\n\",\n            \"62\": \"        True\\n\",\n            \"63\": \"\\n\",\n            \"64\": \"        A Munch is a subclass of dict; it has all the methods (...truncated)\",\n            \"65\": \"\\n\",\n            \"66\": \"        # Fix for Unmunchify Cycles Handling\\n\",\n            \"67\": \"        def unmunchify(value, seen=None):\\n\",\n            \"68\": \"            if seen is None:\\n\",\n            \"69\": \"                seen = {}\\n\",\n            \"70\": \"\\n\",\n            \"71\": \"            # To handle cycles, store each seen object by id\\n\",\n            \"72\": \"            obj_id = id(value)\\n\",\n            \"73\": \"            if"
      }
    ],
    "location_ground_exp": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "113": "unmunchify"
        },
        "content_all": {
          "110": "    seen = {}\n",
          "111": "    def pre_unmunchify(obj):\n",
          "112": "        if isinstance(obj, Munch):\n",
          "113": "            try: seen.setdefault(id(obj), 'placeholder')\n",
          "114": "            obj = {k: pre_unmunchify(v) for k, v in obj.items()}\n",
          "115": "            seen[id(obj)] = obj\n",
          "116": "        elif isinstance(obj, list):\n"
        },
        "content_change": {
          "113": "            try: seen.setdefault(id(obj), 'placeholder')\n"
        }
      },
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "118": "unmunchify"
        },
        "content_all": {
          "115": "            seen[id(obj)] = obj\n",
          "116": "        elif isinstance(obj, list):\n",
          "117": "            obj = [pre_unmunchify(v) for v in obj]\n",
          "118": "        elif isinstance(obj, dict):\n",
          "119": "            try: seen.setdefault(id(obj), 'placeholder')\n",
          "120": "            obj = {k: pre_unmunchify(v) for k, v in obj.items()}\n",
          "121": "            seen[id(obj)] = obj\n",
          "122": "        return obj\n"
        },
        "content_change": {
          "119": "            try: seen.setdefault(id(obj), 'placeholder')\n"
        }
      },
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "123": "unmunchify"
        },
        "content_all": {
          "120": "            obj = {k: pre_unmunchify(v) for k, v in obj.items()}\n",
          "121": "            seen[id(obj)] = obj\n",
          "122": "        return obj\n",
          "123": "\n",
          "124": "    return pre_unmunchify(obj)\n",
          "125": "\n",
          "126": "\ndef day_name(self, obj):\n"
        },
        "content_change": {
          "121": "            seen[id(obj)] = obj\n"
        }
      },
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "130": "register_yaml_constructors"
        },
        "content_all": {
          "127": "\n",
          "128": "import yaml\n",
          "129": "from yaml.constructor import ConstructorError\n",
          "130": "\n",
          "131": "\n",
          "132": "def register_yaml_constructors():\n",
          "133": "    def munch_constructor(loader, node):\n",
          "134": "        values = loader.construct_mapping(node)\n",
          "135": "        return Munch(values)\n"
        },
        "content_change": {
          "132": "def register_yaml_constructors():\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "6": "test_from_yaml"
        },
        "content_all": {
          "3": "\n",
          "4": "\n",
          "5": "def test_from_yaml(yaml):\n",
          "6": "    data = yaml.load('''\n",
          "7": "    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n",
          "8": "    Block style: !munch\n",
          "9": "      Clark : Evans\n",
          "10": "      Brian : Ingerson\n",
          "11": "      Oren  : Ben-Kiki\n"
        },
        "content_change": {
          "6": "    data = yaml.load('''\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "490": "unmunchify"
        },
        "content_all": {
          "510": "\n",
          "511": "    def unmunchify_cycles(obj):\n",
          "512": "        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n",
          "513": "        if id(obj) in seen.keys():\n",
          "514": "            return seen[id(obj)]\n",
          "515": "        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n",
          "516": "        seen[id(obj)] = partial = pre_unmunchify(obj)\n",
          "517": "        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n"
        },
        "content_change": {
          "513": "        if id(obj) in seen.keys():\n"
        }
      },
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "511": "unmunchify_cycles"
        },
        "content_all": {
          "510": "\n",
          "511": "    def unmunchify_cycles(obj):\n",
          "512": "        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n",
          "513": "        if id(obj) in seen.keys():\n",
          "514": "            return seen[id(obj)]\n",
          "515": "        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n",
          "516": "        seen[id(obj)] = partial = pre_unmunchify(obj)\n",
          "517": "        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n"
        },
        "content_change": {
          "513": "        if id(obj) in seen.keys():\n"
        }
      },
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": null,
        "content_all": {
          "631": "            # This code supports both PyYAML 4.x and 5.x versions\n",
          "632": "            continue\n",
          "633": "        yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n",
          "634": "\n",
          "635": "    SafeRepresenter.add_representer(Munch, to_yaml_safe)\n",
          "636": "    SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n"
        },
        "content_change": {}
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": null,
        "content_all": {
          "97": "    assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n"
        },
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/Infinidat_munch/tests/conftest.py\n+++ b/Infinidat_munch/tests/conftest.py\n@@ -0,0 +1,22 @@\n+import pytest\n+import munch\n+\n+\n+@pytest.fixture(name='yaml')\n+def yaml_module():\n+    try:\n+        import yaml  # pylint: disable=import-outside-toplevel\n+        return yaml\n+    except ImportError:\n+        pass\n+    pytest.skip(\"Module 'PyYAML' is required\")\n+\n+\n+@pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n+                        munch.RecursiveMunch])\n+def munch_obj(request):\n+    cls = request.param\n+    args = tuple()\n+    if cls == munch.DefaultFactoryMunch:\n+        args = args + (lambda: None,)\n+    return cls(*args, hello=\"world\", number=5)\n",
    "patch_im": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -43,6 +43,14 @@\n \n __all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n \n+import yaml\n+from munch import Munch\n+\n+def munch_constructor(loader, node):\n+    value = loader.construct_mapping(node)\n+    return Munch(value)\n+\n+yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)\n \n class Munch(dict):\n     \"\"\" A dictionary that provides attribute-style access.\n",
    "patch_il": "\n--- a/Infinidat_munch/tests/conftest.py\n+++ b/Infinidat_munch/tests/conftest.py\n@@ -0,0 +1,23 @@\n+import pytest\n+import munch\n+ \n+ \n+@pytest.fixture(name='yaml')\n+def yaml_module():\n+    try:\n+        import yaml  # pylint: disable=import-outside-toplevel\n+        return yaml\n+    except ImportError:\n+        pass\n+    pytest.skip(\"Module 'PyYAML' is required\")\n+ \n+ \n+@pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n+                        munch.RecursiveMunch])\n+def munch_obj(request):\n+    cls = request.param\n+    args = tuple()\n+    if cls == munch.DefaultFactoryMunch:\n+        args = args + (lambda: None,)\n+    return cls(*args, hello=\"world\", number=5)\n+\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -1,4 +1,5 @@\n import pytest\n from munch import Munch, DefaultMunch\n \n \n def test_from_yaml(yaml):\n     data = yaml.load('''\n     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n     Block style: !munch\n       Clark : Evans\n       Brian : Ingerson\n       Oren  : Ben-Kiki\n     ''', Loader=yaml.FullLoader)\n     assert data == {\n         'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n         'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n     }\n \n \n def test_to_yaml_safe(yaml):\n     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n     dumped = yaml.safe_dump(b, default_flow_style=False)\n     assert dumped\n",
    "patch_iml": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -44,6 +44,12 @@ __all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'Recursi\n \n \n def register_yaml_constructor():\n+    import yaml\n+    from munch import Munch\n+    def munch_constructor(loader, node):\n+        value = loader.construct_mapping(node)\n+        return Munch(value)\n+    yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)\n \n \n class Munch(dict):\n--- a/Infinidat_munch/tests/conftest.py\n+++ b/Infinidat_munch/tests/conftest.py\n@@ -8,6 +8,7 @@ def yaml_module():\n         import yaml  # pylint: disable=import-outside-toplevel\n+        from munch import register_yaml_constructor\n+        register_yaml_constructor()\n         return yaml\n     except ImportError:\n         pass\n",
    "patch_ground": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -42,6 +42,8 @@ except ValueError:\n     VERSION = (0, 0, 0)\n \n __all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n+\n+import yaml\n \n class Munch(dict):\n     \"\"\" A dictionary that provides attribute-style access.\n@@ -415,6 +417,63 @@ def muched(cls, dct=None, *args, **kwargs):\n \n \n def unmunchify(d):\n+    \"\"\"Recursively converts a Munch into a dictionary.\n+\n+    Args:\n+        d (Munch): The Munch instance to convert.\n+\n+    Returns:\n+        A dictionary representation of `d`.\n+    \"\"\"\n+    seen = {}\n+\n+    def _unmunchify(value):\n+        obj_id = id(value)\n+        if obj_id in seen:\n+            return seen[obj_id]\n+\n+        if isinstance(value, Munch):\n+            result = {k: _unmunchify(v) for k, v in value.items()}\n+            seen[obj_id] = result\n+            return result\n+        elif isinstance(value, dict):\n+            result = {k: _unmunchify(v) for k, v in value.items()}\n+            seen[obj_id] = result\n+            return result\n+        elif isinstance(value, list):\n+            result = [_unmunchify(v) for v in value]\n+            seen[obj_id] = result\n+            return result\n+        else:\n+            return value\n+\n+    return _unmunchify(d)\n+\n+# Custom constructor for YAML deserialization\n+def munch_constructor(loader, node):\n+    value = loader.construct_mapping(node)\n+    return Munch(value)\n+\n+# Add the custom constructor to the PyYAML loader\n+yaml.add_constructor(u'!munch.Munch', munch_constructor)\n+yaml.add_constructor(u'!munch', munch_constructor)\n+\n \n def toYaml(d, **kwargs):\n     return yaml.dump(munchify(d), **kwargs)\n",
    "patch_ground_location": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -24,6 +24,7 @@ from collections.abc import Mapping\n import yaml\n \n \n+def unmunchify(value, seen=None):\n     if seen is None:\n         seen = {}\n \n@@ -31,6 +32,7 @@ def unmunchify(value, seen=None):\n \n     if isinstance(value, Munch):\n         result = {k: unmunchify(v, seen) for k, v in value.items()}\n+        seen[obj_id] = result\n         return result\n     elif isinstance(value, list):\n         result = [unmunchify(i, seen) for i in value]\n@@ -69,6 +71,15 @@ def unmunchify(value, seen=None):\n         seen[obj_id] = result\n         return result\n     else:\n         return value\n \n \n+def munch_constructor(loader, node):\n+    value = loader.construct_mapping(node)\n+    return Munch(value)\n+\n+\n+yaml.add_constructor(u'!munch.Munch', munch_constructor)\n+yaml.add_constructor(u'!munch', munch_constructor)\n+\n class Munch(dict):\n     \"\"\" A dictionary that provides attribute-style access.\n \n---\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -31,6 +31,12 @@ def test_contains():\n def test_unmunchify_cycle():\n     a = Munch()\n     b = Munch(ref=a)\n     a.ref = b\n     result = unmunchify(a)\n     assert result['ref']['ref'] == result\n \n \n---\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -5,12 +5,23 @@ def test_from_yaml(yaml):\n def test_from_yaml(yaml):\n     yaml_string = '''\n     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n     Block style: !munch\n       Clark : Evans\n       Brian : Ingerson\n       Oren  : Ben-Kiki\n     '''\n     data = yaml.load(yaml_string, Loader=yaml.FullLoader)\n     assert data == {\n         'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n         'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n     }\n \n",
    "patch_ground_exp": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -110,7 +110,7 @@\n     seen = {}\n     def pre_unmunchify(obj):\n         if isinstance(obj, Munch):\n-            try: seen.setdefault(id(obj), 'placeholder')\n+            seen.setdefault(id(obj), 'placeholder')\n             obj = {k: pre_unmunchify(v) for k, v in obj.items()}\n             seen[id(obj)] = obj\n         elif isinstance(obj, list):\n@@ -119,7 +119,7 @@\n             obj = [pre_unmunchify(v) for v in obj]\n         elif isinstance(obj, dict):\n             try:\n-                seen.setdefault(id(obj), 'placeholder')\n+                seen[id(obj)]\n             except KeyError:\n                 seen[id(obj)] = 'placeholder'\n                 obj = {k: pre_unmunchify(v) for k, v in obj.items()}\n@@ -124,4 +124,21 @@\n \n     return pre_unmunchify(obj)\n \n+\n+import yaml\n+\n+def register_yaml_constructors():\n+    def munch_constructor(loader, node):\n+        values = loader.construct_mapping(node)\n+        return Munch(values)\n+\n+    yaml.add_constructor('!munch.Munch', munch_constructor, Loader=yaml.FullLoader)\n+    yaml.add_constructor('!munch', munch_constructor, Loader=yaml.FullLoader)\n+\n+register_yaml_constructors()\n+\n+\n+def day_name(self, obj):\n+    pass\n+\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -1,5 +1,6 @@\n import pytest\n from munch import Munch, DefaultMunch\n+import yaml\n \n \n def test_from_yaml(yaml):\n",
    "patch_ground_all": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -438,6 +438,17 @@ def munchify(x, factory=None):\n \n def unmunchify(x, seen=None):\n     \"\"\"Recursively convert a Munch into a dictionary.\"\"\"\n+    if seen is None:\n+        seen = {}\n+    obj_id = id(x)\n+    if obj_id in seen:\n+        return seen[obj_id]\n+\n+    if isinstance(x, Munch):\n+        result = {}\n+        seen[obj_id] = result\n+        for k, v in x.items():\n+            result[k] = unmunchify(v, seen)\n+        return result\n     if isinstance(x, dict):\n         return {k: unmunchify(v, seen) for k, v in x.items()}\n     if isinstance(x, (list, tuple)):\n@@ -464,3 +475,18 @@ def unmunchify(x, seen=None):\n     }\n     vars(munch).update(factories)\n     return factories\n+\n+import yaml\n+from yaml.constructor import ConstructorError\n+\n+def from_yaml(loader, node):\n+    \"\"\" Convert YAML node to Munch \"\"\"\n+    return Munch(loader.construct_mapping(node, deep=True))\n+\n+def munch_yaml_representer(dumper, data):\n+    \"\"\" Represent Munch object as YAML \"\"\"\n+    return dumper.represent_mapping('tag:yaml.org,2002:map', data)\n+\n+yaml.add_constructor('!munch.Munch', from_yaml, Loader=yaml.FullLoader)\n+yaml.add_representer(Munch, munch_yaml_representer)\n+yaml.SafeRepresenter.add_representer(Munch, munch_yaml_representer)\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -145,6 +145,23 @@ def test_unmunchify():\n     a.b.c = Munch(d='e')\n     a.ddd = a\n     res = unmunchify(a)\n+    assert res['a']['b']['c']['d'] == 'e'\n+    assert res['a']['ddd'] is res['a']\n \n def test_cyclic_unmunchify():\n     a = Munch()\n     a['self'] = a\n     res = unmunchify(a)\n     assert res['self'] is res\n \n+import yaml\n+\n+def test_from_yaml(yaml):\n+    data = yaml.load('''\n+    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n+    Block style: !munch\n+      Clark : Evans\n+      Brian : Ingerson\n+      Oren  : Ben-Kiki\n+    ''', Loader=yaml.FullLoader)\n+    assert data == {\n+        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n+        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n+    }\n",
    "patch_ground_truth": "--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,8 +510,11 @@\n \n     def unmunchify_cycles(obj):\n         # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n-        if id(obj) in seen.keys():\n+        try:\n             return seen[id(obj)]\n+        except KeyError:\n+            pass\n+\n         # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n         seen[id(obj)] = partial = pre_unmunchify(obj)\n         # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n@@ -631,6 +634,7 @@\n             # This code supports both PyYAML 4.x and 5.x versions\n             continue\n         yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n+        yaml.add_constructor('!munch.Munch', from_yaml, Loader=LoaderCls)\n \n     SafeRepresenter.add_representer(Munch, to_yaml_safe)\n     SafeRepresenter.add_multi_representer(Munch, to_yaml_safe)\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -94,6 +94,9 @@\n def test_toDict():\n     b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n     assert sorted(b.toDict().items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n+\n+def test_dict_property():\n+    b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n     assert sorted(b.__dict__.items()) == [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n \n def test_repr():\n",
    "message": "============================================================================================== FAILURES ===============================================================================================\n___________________________________________________________________________________________ test_from_yaml ____________________________________________________________________________________________\n\nyaml = <module 'yaml' from '/home/user/anaconda3/envs/py39/lib/python3.9/site-packages/yaml/__init__.py'>\n\n    def test_from_yaml(yaml):\n>       data = yaml.load('''\n        Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n        Block style: !munch\n          Clark : Evans\n          Brian : Ingerson\n          Oren  : Ben-Kiki\n        ''', Loader=yaml.FullLoader)\n\ntests/test_yaml.py:6: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/__init__.py:81: in load\n    return loader.get_single_data()\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:51: in get_single_data\n    return self.construct_document(node)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:60: in construct_document\n    for dummy in generator:\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:413: in construct_yaml_map\n    value = self.construct_mapping(node)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:218: in construct_mapping\n    return super().construct_mapping(node, deep=deep)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:143: in construct_mapping\n    value = self.construct_object(value_node, deep=deep)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:100: in construct_object\n    data = constructor(self, node)\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n\nself = <yaml.loader.FullLoader object at 0x76a0b06717f0>\nnode = MappingNode(tag='!munch.Munch', value=[(ScalarNode(tag='tag:yaml.org,2002:str', value='Clark'), ScalarNode(tag='tag:ya...), (ScalarNode(tag='tag:yaml.org,2002:str', value='Oren'), ScalarNode(tag='tag:yaml.org,2002:str', value='Ben-Kiki'))])\n\n    def construct_undefined(self, node):\n>       raise ConstructorError(None, None,\n                \"could not determine a constructor for the tag %r\" % node.tag,\n                node.start_mark)\nE       yaml.constructor.ConstructorError: could not determine a constructor for the tag '!munch.Munch'\nE         in \"<unicode string>\", line 2, column 17:\nE               Flow style: !munch.Munch { Clark: Evans, Bri ... \nE                           ^\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/yaml/constructor.py:427: ConstructorError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_yaml.py::test_from_yaml - yaml.constructor.ConstructorError: could not determine a constructor for the tag '!munch.Munch'\n==================================================================================== 1 failed, 108 passed in 0.09s ====================================================================================",
    "CodeBase": [
      {
        "path": "Infinidat_munch/munch/__init__.py",
        "content": "1 \"\"\" Munch is a subclass of dict with attribute-style access.\n2 \n3     >>> b = Munch()\n4     >>> b.hello = 'world'\n5     >>> b.hello\n6     'world'\n7     >>> b['hello'] += \"!\"\n8     >>> b.hello\n9     'world!'\n10     >>> b.foo = Munch(lol=True)\n11     >>> b.foo.lol\n12     True\n13     >>> b.foo is b['foo']\n14     True\n15 \n16     It is safe to import * from this module:\n17 \n18         __all__ = ('Munch', 'munchify','unmunchify')\n19 \n20     un/munchify provide dictionary conversion; Munches can also be\n21     converted via Munch.to/fromDict().\n22 \"\"\"\n23 \n24 from collections.abc import Mapping\n25 \n26 try:\n27     # For python 3.8 and later\n28     import importlib.metadata as importlib_metadata\n29 except ImportError:\n30     # For everyone else\n31     import importlib_metadata\n32 try:\n33     __version__ = importlib_metadata.version(__name__)\n34 except importlib_metadata.PackageNotFoundError:\n35     # package is not installed\n36     __version__ = \"0.0.0\"\n37 \n38 \n39 try:\n40     VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\n41 except ValueError:\n42     VERSION = (0, 0, 0)\n43 \n44 __all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n45 \n46 \n47 \n48 class Munch(dict):\n49     \"\"\" A dictionary that provides attribute-style access.\n50 \n51         >>> b = Munch()\n52         >>> b.hello = 'world'\n53         >>> b.hello\n54         'world'\n55         >>> b['hello'] += \"!\"\n56         >>> b.hello\n57         'world!'\n58         >>> b.foo = Munch(lol=True)\n59         >>> b.foo.lol\n60         True\n61         >>> b.foo is b['foo']\n62         True\n63 \n64         A Munch is a subclass of dict; (...truncated)"
      },
      {
        "path": "Infinidat_munch/tests/test_munch.py",
        "content": "1 # pylint: disable=unnecessary-lambda\n2 import json\n3 import pickle\n4 from collections import namedtuple\n5 \n6 import pytest\n7 from munch import DefaultFactoryMunch, AutoMunch, DefaultMunch, Munch, munchify, unmunchify\n8 \n9 \n10 def test_base():\n11     b = Munch()\n12     b.hello = 'world'\n13     assert b.hello == 'world'\n14     b['hello'] += \"!\"\n15     assert b.hello == 'world!'\n16     b.foo = Munch(lol=True)\n17     assert b.foo.lol is True\n18     assert b.foo is b['foo']\n19 \n20     assert sorted(b.keys()) == ['foo', 'hello']\n21 \n22     b.update({'ponies': 'are pretty!'}, hello=42)\n23     assert b == Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n24 \n25     assert sorted([(k, b[k]) for k in b]) == [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n26 \n27     format_munch = Munch(knights='lolcats', ni='can haz')\n28     assert \"The {knights} who say {ni}!\".format(**format_munch) == 'The lolcats who say can haz!'\n29 \n30 \n31 def test_contains():\n32     b = Munch(ponies='are pretty!')\n33     assert 'ponies' in(...truncated)"
      },
      {
        "path": "Infinidat_munch/tests/test_yaml.py",
        "content": "1 import pytest\n2 from munch import Munch, DefaultMunch\n3 \n4 \n5 def test_from_yaml(yaml):\n6     data = yaml.load('''\n7     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n8     Block style: !munch\n9       Clark : Evans\n10       Brian : Ingerson\n11       Oren  : Ben-Kiki\n12     ''', Loader=yaml.FullLoader)\n13     assert data == {\n14         'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n15         'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n16     }\n17 \n18 \n19 def test_to_yaml_safe(yaml):\n20     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n21     dumped = yaml.safe_dump(b, default_flow_st(...truncated)"
      },
      {
        "path": "Infinidat_munch/tests/conftest.py",
        "content": "1 import pytest\n2 import munch\n3 \n4 \n5 @pytest.fixture(name='yaml')\n6 def yaml_module():\n7     try:\n8         import yaml  # pylint: disable=import-outside-toplevel\n9         return yaml\n10     except ImportError:\n11         pass\n12     pytest.skip(\"Module 'PyYAML' is required\")\n13 \n14 \n15 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n16                         munch.RecursiveMunch])\n17 def munch_obj(request):\n18     cls = request.param\n19     args = tuple()\n20     if cls == munch.DefaultFactoryMunch:\n21         args = args + (lambda: None,)\n22     return cls(*args, hello=\"world\", number=5)"
      },
      {
        "path": "Infinidat_munch/tests/test_readme.py",
        "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _HERE = os.path.abspath(os.path.dirname(__file__))\n6 _README_PATH = os.path.join(_HERE, '..', 'README.md')\n7 assert os.path.exists(_README_PATH)\n8 \n9 \n10 @pytest.mark.usefixtures(\"yaml\")\n11 def test_readme():\n12     globs = {\n13         'print_function': print\n14     }\n15     result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n16     assert not result.failed"
      },
      {
        "path": "Infinidat_munch/setup.py",
        "content": "1 from setuptools import setup\n2 \n3 \n4 setup(\n5     setup_requires=['pbr>=3.0', 'setuptool(...truncated)"
      }
    ],
    "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27"
  },
  "Score": {
    "Difficulty": "Easy",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 6,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 9,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 9,
      "Explanation": 9,
      "Overall": 9
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Handling of Unmunchify Cycles and YAML Deserialization Issues",
      "description": "### Description\nThere are a couple of issues that need to be addressed related to the unmunchify process and YAML deserialization in the Munch library.\n\n#### 1. Unmunchify Cycles Handling\nCurrently, during the unmunchification process, there is a problem when handling objects that have already been seen. This potentially raises an error due to incorrect handling when the object ID is not found in the `seen` dictionary. This issue affects the robustness of the `unmunchify` function, causing it to fail when it encounters cycles in complex structures, such as nested dictionaries or lists.\n\n#### 2. YAML Deserialization\nThe library does not support some YAML tags like `!munch.Munch`, which are essential for correctly deserializing objects. This leads to incomplete or incorrect deserialization of YAML strings that should convert tags to Munch objects.\n\n### Impact\n- Code that relies on the `unmunchify` function may crash or exhibit undefined behavior when dealing with certain cyclic structures, affecting the reliability of applications using this library.\n- Incorrect or incomplete deserialization of YAML configurations can lead to data loss or unexpected behavior in applications, particularly in those that heavily rely on YAML for configuration or data exchange.\n\n### Steps to Reproduce\n1. Create a Munch object with cyclic references and attempt to unmunchify it. Observe the failure or incorrect results.\n2. Attempt to deserialize a YAML string containing `!munch.Munch` tags and observe that the resulting objects are not properly converted.\n\n### Expected Behavior\n- The `unmunchify` function should handle cyclic structures gracefully without raising errors.\n- YAML deserialization should correctly interpret and convert `!munch.Munch` tags into appropriate Munch objects.\n\n### Suggested Solutions\nWhile specific code changes are not provided here, addressing the above issues will likely involve:\n1. Updating the `unmunchify` function to correctly handle cases where object IDs are not found in the `seen` dictionary.\n2. Adding support for the `!munch.Munch` tag in the YAML deserialization process.\n\n### Additional Information\nRelevant tests should be created or updated to cover these cases, ensuring that both unmunchification and YAML deserialization work as expected.",
      "explanation": "### Summary of the Issue\n\nThe issue involves two key problems related to the Munch library:\n1. **Handling of Unmunchify Cycles**: There is an issue with the `unmunchify` function when it encounters cyclic references in complex structures like nested dictionaries or lists. This can lead to errors when object IDs are not found in a tracking dictionary called `seen`.\n2. **YAML Deserialization**: The library lacks support for deserializing certain YAML tags, specifically `!munch.Munch`. This omission leads to improper conversion of YAML strings into Munch objects, which might result in data loss or unexpected behavior for applications using YAML configurations.\n\n### Content of the Commit\n\nThe commit addresses the two problems as follows:\n\n#### 1. **Updating `unmunchify` Function for Cycles**\n- **Problem**: The original `unmunchify` function could throw errors when it encounters cyclic references, because it did not correctly handle the case when an object ID was not found in the `seen` dictionary.\n- **Solution**: The commit updates the `unmunchify` function to use a `try-except` block. This ensures that instead of raising an error when an object ID is not found in the `seen` dictionary, the function continues processing. This change enhances the function's robustness, allowing it to gracefully handle cyclic references without crashing.\n\n#### 2. **Enhancing YAML Deserialization**\n- **Problem**: The `Munch` library did not support the `!munch.Munch` YAML tag, which is necessary for correctly deserializing YAML strings into Munch objects. Without this support, objects serialized with this tag could not be deserialized correctly, leading to incomplete or incorrect data reconstruction.\n- **Solution**: The commit adds support for the `!munch.Munch` tag by registering a new YAML constructor. This ensures that YAML deserialization correctly converts `!munch.Munch` tags into Munch objects. This enhancement ensures complete and accurate deserialization of YAML strings, maintaining the integrity of data.\n\n### Explanation of the Solution\n\n#### Handling of Unmunchify Cycles\n- **Cause**: The `unmunchify` function was not resilient to cycles in object graphs. It would raise an error if it attempted to access an object ID not present in the `seen` dictionary, which tracks objects that have already been processed.\n- **Solution**: By introducing a `try-except` block, the function now gracefully handles the absence of an object ID in the `seen` dictionary. If an object ID is not found, the function continues processing without raising an error, completing the unmunchification process successfully.\n\n#### YAML Deserialization\n- **Cause**: The absence of support for the `!munch.Munch` tag in YAML deserialization meant that certain YAML-encoded Munch objects could not be reconstructed correctly. This could lead to data loss or inaccuracies.\n- **Solution**: By registering a new YAML constructor for the `!munch.Munch` tag, the commit ensures that the Munch library can correctly interpret and convert these tags during the YAML deserialization process. This guarantees that the objects are reconstructed as intended, preserving data integrity and ensuring that the applications relying on YAML configurations work as expected.\n\n### Conclusion\n\nThe commit effectively addresses both reported issues:\n- It strengthens the `unmunchify` function by improving its handling of cyclic references, thus enhancing the robustness and reliability of the library for complex data structures.\n- It adds necessary support for the `!munch.Munch` YAML tag, ensuring complete and accurate deserialization of YAML-configured Munch objects.\n\nOverall, these changes improve the library's functionality, making it more resilient and reliable for users handling complex data structures and configurations."
    }
  }
}