{
  "RepoName": "https://github.com/Infinidat/munch.git",
  "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "undefined objects",
  "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        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    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].cats == '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_README_PATH = os.path.join(os.path.dirname(__file__), '..', '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.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n    assert obj.toYAML().__eq__(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    obj = DefaultMunch.fromYAML(yaml_str, object())\n    assert type(obj) == DefaultMunch\n    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n    assert obj['not_exist']\n    assert obj.not_exist\n    assert obj.toYAML().__eq__(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_munchify ____________________________________________________________________________________________\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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)\nE           AttributeError: 'Munch' object has no attribute 'cats'\n\nmunch/__init__.py:116: AttributeError\n\nDuring handling of the above exception, another exception occurred:\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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]\nE               KeyError: 'cats'\n\nmunch/__init__.py:119: KeyError\n\nDuring handling of the above exception, another exception occurred:\n\n    def 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].cats == 'i win again'\n\ntests/test_munch.py:142: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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)\nE               AttributeError: cats\n\nmunch/__init__.py:121: AttributeError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_munch.py::test_munchify - AttributeError: cats\n==================================================================================== 1 failed, 108 passed in 0.12s ====================================================================================",
  "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--- 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@@ -139,7 +142,7 @@\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].cats == 'i win again'\n+    assert b.lol[1].hah == 'i win again'\n \n def test_munchify_with_namedtuple():\n     nt = namedtuple('nt', ['prop_a', 'prop_b'])\n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -2,7 +2,8 @@\n import os\n import pytest\n \n-_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n+_HERE = os.path.abspath(os.path.dirname(__file__))\n+_README_PATH = os.path.join(_HERE, '..', 'README.md')\n assert os.path.exists(_README_PATH)\n \n \n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -43,17 +43,18 @@\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.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n-    assert obj.toYAML().__eq__(yaml_str)\n+    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n+    assert obj.toYAML() == yaml_str\n \n \n @pytest.mark.usefixtures('yaml')\n def test_fromYAML_default_munch():\n     # pylint: disable=unidiomatic-typecheck\n     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n-    obj = DefaultMunch.fromYAML(yaml_str, object())\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']\n-    assert obj.not_exist\n-    assert obj.toYAML().__eq__(yaml_str)\n+    assert obj['not_exist'] is default_value\n+    assert obj.not_exist is default_value\n+    assert obj.toYAML() == yaml_str\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": {},
      "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",
        "142": "    assert b.lol[1].cats == 'i win again'\n"
      },
      "content_change": {
        "142": "    assert b.lol[1].cats == 'i win again'\n"
      }
    },
    {
      "file": "Infinidat_munch/tests/test_readme.py",
      "function": null,
      "content_all": {
        "2": "import os\n",
        "3": "import pytest\n",
        "4": "\n",
        "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n",
        "6": "assert os.path.exists(_README_PATH)\n",
        "7": "\n",
        "8": "\n"
      },
      "content_change": {
        "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n"
      }
    },
    {
      "file": "Infinidat_munch/tests/test_yaml.py",
      "function": {
        "41": "test_fromYAML"
      },
      "content_all": {
        "43": "    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n",
        "44": "    obj = Munch.fromYAML(yaml_str)\n",
        "45": "    assert type(obj) == Munch\n",
        "46": "    assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n"
      },
      "content_change": {
        "46": "    assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n"
      }
    },
    {
      "file": "Infinidat_munch/tests/test_yaml.py",
      "function": {
        "51": "test_fromYAML_default_munch"
      },
      "content_all": {
        "50": "@pytest.mark.usefixtures('yaml')\n",
        "51": "def test_fromYAML_default_munch():\n",
        "52": "    # pylint: disable=unidiomatic-typecheck\n",
        "53": "    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n",
        "54": "    obj = DefaultMunch.fromYAML(yaml_str, object())\n",
        "55": "    assert type(obj) == DefaultMunch\n",
        "56": "    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n",
        "57": "    assert obj['not_exist']\n",
        "58": "    assert obj.not_exist\n"
      },
      "content_change": {
        "54": "    obj = DefaultMunch.fromYAML(yaml_str, object())\n",
        "57": "    assert obj['not_exist']\n",
        "58": "    assert obj.not_exist\n"
      }
    },
    {
      "file": "Infinidat_munch/tests/test_yaml.py",
      "function": null,
      "content_all": {
        "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
        "48": "\n",
        "49": "\n",
        "59": "    assert obj.toYAML().__eq__(yaml_str)\n"
      },
      "content_change": {
        "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
        "59": "    assert obj.toYAML().__eq__(yaml_str)\n"
      }
    }
  ],
  "Issue": {
    "title": "Fix Unmunchify Function KeyError and Add Improper Assertion Error Fixes",
    "description": "### Issue Description\nSeveral issues have been identified that affect the functionality and reliability of the `Infinidat_munch` project.\n\n1. **`unmunchify_cycles` Function KeyError**: The `unmunchify_cycles` function in `munch/__init__.py` does not handle cases where an object is not found in the `seen` dictionary correctly. This results in a KeyError which interrupts the unmunchify process and can potentially cause the application to crash.\n\n2. **Incorrect Assertion in Tests**: There are instances of incorrect assertions within the test suite.\n   - The `test_munchify` test incorrectly asserts the `lol[1].cats` attribute, which leads to false positives or unresolved tests.\n   - The equality checks in tests use `__eq__`, which can be replaced with the `==` operator for clarity and readability.\n\n3. **Path Issue in Test Files**: There are issues with path resolution in one of the test files (`test_readme.py`). The path `_README_PATH` is constructed using `os.path.join`, which may lead to `FileNotFoundError` if the current directory is not set correctly.\n\n4. **Default Value Handling in DefaultMunch**: The tests for the `DefaultMunch` class assume that non-existent keys return `True`, which is not correct. It should instead verify that the expected default values are returned for missing keys.\n\n### Impact\n- **Unmunchify Function**: The unmunchify process will fail when encountering objects not tracked in the `seen` dictionary, resulting in incomplete or inaccurate conversions and potential application crashes.\n- **Test Suite Failures**: Incorrect assertions and comparisons lead to unreliable test results, contributing to an unstable codebase and masking potential bugs.\n- **Path Issues**: The path issue in the tests can lead to failures when the tests are run in environments where the current directory varies, impacting the reliability of the test suite.\n- **Default Value Handling**: Incorrect assumptions about the default value behavior can cause misleading test results and unexpected behavior in application code.\n\n### Steps to Reproduce\n1. Call the `unmunchify` function with cyclical data structures and observe `KeyError` exceptions.\n2. Run the test suite and observe the incorrect assertions in `test_munch.py` and path-related issues in `test_readme.py`.\n3. Verify the default value behavior by using `DefaultMunch` and checking non-existent keys.\n\n### Expected Behavior\n- **Unmunchify Function**: The function should handle the absence of objects in the `seen` dictionary gracefully without raising `KeyError`.\n- **Correct Assertions**: The tests should correctly assert the presence and values of attributes within Munch objects.\n- **Correct Path Resolution**: The test paths should be constructed in a manner that is robust to changes in the current directory.\n- **Accurate Default Value Handling**: The `DefaultMunch` tests should accurately assess the behavior of default values for non-existent keys.\n\n### Suggestions\n- Implement exception handling in the `unmunchify_cycles` function to address cycles and missing references gracefully.\n- Update the assertions in the test suite to reflect accurate and intended behavior, particularly for attribute checks and equality comparisons.\n- Refactor path construction in the tests to guarantee path resolution correctness.\n- Adjust tests for `DefaultMunch` to verify that default values are returned for keys that do not exist.\n",
    "explanation": "### Summary of the Issue\n\nThe issue reported multiple problems affecting the `Infinidat_munch` project, primarily related to:\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n    - The `unmunchify_cycles` function can throw a `KeyError` if an object isn't found in the `seen` dictionary, causing the app to crash.\n\n2. **Incorrect Assertions and Path Issues in Tests**:\n    - Some test assertions, particularly in `test_munch.py`, were incorrect.\n    - Path construction in `test_readme.py` was unreliable and could cause `FileNotFoundError`.\n    - Tests for `DefaultMunch` class made incorrect assumptions about default value handling, leading to unreliable results.\n\n3. **Improper Assertion Error Fixes**:\n    - The test assertions for some keys and attributes were incorrectly performed.\n\n### Detailed Commit Content and Solutions\n\nThe commit makes several changes to address these issues:\n\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n   - **Issue**: If an object is not tracked in `seen`, the function throws a `KeyError`.\n   - **Solution**: Introduce exception handling to gracefully manage cases where an object isn't found in `seen`, thereby preventing the `KeyError` and ensuring the function completes normally.\n\n2. **Incorrect Assertions in Tests**:\n   - **Issue**: Assertions in `test_munch.py` were incorrectly asserting attributes, causing false positives.\n   - **Solution**: Correct the assertions to check for the proper attributes and use the equality operator (`==`) instead of direct method calls, enhancing readability and correctness.\n   - This addresses the testing reliability by ensuring tests are accurately verifying the expected behavior.\n\n3. **Path Issue in `test_readme.py`**:\n   - **Issue**: Path construction could result in `FileNotFoundError` if the directory is not correctly set.\n   - **Solution**: Refactor the path construction method to use an absolute path derived from the directory of the file itself, making the path resolution robust and independent of the current working directory.\n\n4. **Default Value Handling in `DefaultMunch`**:\n   - **Issue**: Tests assumed that non-existent keys return `True`, which was incorrect.\n   - **Solution**: Adjust tests to accurately verify that the expected default values are returned for missing keys, ensuring the actual behavior matches the intended design.\n\n### Explanation from Developer's Perspective\n\nThe developer recognized that the current implementation had certain fragile elements:\n- The `unmunchify_cycles` was not robust against missing objects in `seen`.\n- Test cases had incorrect assertions, which undermined the reliability of the results.\n- Path constructions in test scripts were susceptible to the changing current directory.\n- `DefaultMunch` class tests were based on incorrect assumptions about default value handling.\n\n**Detailed Solution Approach**:\n- **Robust Error Handling**: By adding a `try-except` block around the dictionary access in `unmunchify_cycles`, any missing objects will not cause an abrupt termination of the function. This ensures that the conversion process can complete without crashing due to a `KeyError`.\n- **Correct Assertions**: Fixing the test assertions to accurately point to the correct attributes and properties ensures that tests will capture the correct functional behavior and prevent missed bugs or misleading success reports.\n- **Reliable Path Construction**: Refactoring the path resolution to use absolute paths derived from the current file location ensures that tests will run correctly regardless of the current directory from which the tests are initiated. This increases the robustness of the test execution environment.\n- **Accurate Default Handling Tests**: By adjusting the tests for the `DefaultMunch` class to check the returned values against expected behavior, the likelihood of undetected issues in the default value behavior is minimized, thus ensuring features work as designed.\n\n### Conclusion\n\nThe commit addresses and resolves the identified issues by:\n- Implementing robust handling of cyclical unmunchification.\n- Updating and correcting the test assertions to ensure reliability.\n- Fixing path issues to ensure consistent test results.\n- Correcting the default value handling tests to align with the expected functionality.\n\nThese changes fortify the codebase against potential faults and improve the reliability and accuracy of the test suite, leading to a more robust and maintainable project."
  },
  "Explain": "### Summary of the Issue\n\nThe issue reported multiple problems affecting the `Infinidat_munch` project, primarily related to:\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n    - The `unmunchify_cycles` function can throw a `KeyError` if an object isn't found in the `seen` dictionary, causing the app to crash.\n\n2. **Incorrect Assertions and Path Issues in Tests**:\n    - Some test assertions, particularly in `test_munch.py`, were incorrect.\n    - Path construction in `test_readme.py` was unreliable and could cause `FileNotFoundError`.\n    - Tests for `DefaultMunch` class made incorrect assumptions about default value handling, leading to unreliable results.\n\n3. **Improper Assertion Error Fixes**:\n    - The test assertions for some keys and attributes were incorrectly performed.\n\n### Detailed Commit Content and Solutions\n\nThe commit makes several changes to address these issues:\n\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n   - **Issue**: If an object is not tracked in `seen`, the function throws a `KeyError`.\n   - **Solution**: Introduce exception handling to gracefully manage cases where an object isn't found in `seen`, thereby preventing the `KeyError` and ensuring the function completes normally.\n\n2. **Incorrect Assertions in Tests**:\n   - **Issue**: Assertions in `test_munch.py` were incorrectly asserting attributes, causing false positives.\n   - **Solution**: Correct the assertions to check for the proper attributes and use the equality operator (`==`) instead of direct method calls, enhancing readability and correctness.\n   - This addresses the testing reliability by ensuring tests are accurately verifying the expected behavior.\n\n3. **Path Issue in `test_readme.py`**:\n   - **Issue**: Path construction could result in `FileNotFoundError` if the directory is not correctly set.\n   - **Solution**: Refactor the path construction method to use an absolute path derived from the directory of the file itself, making the path resolution robust and independent of the current working directory.\n\n4. **Default Value Handling in `DefaultMunch`**:\n   - **Issue**: Tests assumed that non-existent keys return `True`, which was incorrect.\n   - **Solution**: Adjust tests to accurately verify that the expected default values are returned for missing keys, ensuring the actual behavior matches the intended design.\n\n### Explanation from Developer's Perspective\n\nThe developer recognized that the current implementation had certain fragile elements:\n- The `unmunchify_cycles` was not robust against missing objects in `seen`.\n- Test cases had incorrect assertions, which undermined the reliability of the results.\n- Path constructions in test scripts were susceptible to the changing current directory.\n- `DefaultMunch` class tests were based on incorrect assumptions about default value handling.\n\n**Detailed Solution Approach**:\n- **Robust Error Handling**: By adding a `try-except` block around the dictionary access in `unmunchify_cycles`, any missing objects will not cause an abrupt termination of the function. This ensures that the conversion process can complete without crashing due to a `KeyError`.\n- **Correct Assertions**: Fixing the test assertions to accurately point to the correct attributes and properties ensures that tests will capture the correct functional behavior and prevent missed bugs or misleading success reports.\n- **Reliable Path Construction**: Refactoring the path resolution to use absolute paths derived from the current file location ensures that tests will run correctly regardless of the current directory from which the tests are initiated. This increases the robustness of the test execution environment.\n- **Accurate Default Handling Tests**: By adjusting the tests for the `DefaultMunch` class to check the returned values against expected behavior, the likelihood of undetected issues in the default value behavior is minimized, thus ensuring features work as designed.\n\n### Conclusion\n\nThe commit addresses and resolves the identified issues by:\n- Implementing robust handling of cyclical unmunchification.\n- Updating and correcting the test assertions to ensure reliability.\n- Fixing path issues to ensure consistent test results.\n- Correcting the default value handling tests to align with the expected functionality.\n\nThese changes fortify the codebase against potential faults and improve the reliability and accuracy of the test suite, leading to a more robust and maintainable project.",
  "Source": "Human",
  "Token": 1547,
  "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/test_readme.py",
      "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n6 assert os.path.exists(_README_PATH)\n7 \n8 \n9 @pytest.mark.usefixtures(\"yaml\")\n10 def test_readme():\n11     globs = {\n12         'print_function': print\n13     }\n14     result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n15     assert not result.failed"
    },
    {
      "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/setup.py",
      "content": "1 from setuptools import setup\n2 \n3 \n4 setup(\n5     setup_requires=['pbr>=3.0', 'setuptools>=61'],\n6     pbr=True,\n7     long_description_content_type='text/markdown; charset=UTF-8',\n8     keywords=['munch', 'dict', 'mapping', 'container', 'collection'],\n9     python_requires=\">=3.6\",\n10 )"
    }
  ],
  "TokenAll": 13456,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Potential security risk with yaml.load()",
      "description": "The code uses yaml.load() in 'Infinidat_munch/tests/test_yaml.py' on line 6 to load YAML data. This can be a security risk as it can execute arbitrary code. It is recommended to use yaml.safe_load() instead.",
      "explanation": "### Issue Summary\n\nThe issue identifies a potential security risk in the `Infinidat_munch/tests/test_yaml.py` file. Specifically, the code is using `yaml.load()` to load YAML data, which can execute arbitrary code, posing a security risk. The recommended approach is to use `yaml.safe_load()` instead, which mitigates this risk by only loading simple, safe YAML constructs.\n\n### Commit Summary and Content\n\nWhile the specific content of the commit is not provided, the fix for this issue would involve replacing the call to `yaml.load()` with `yaml.safe_load()` in the relevant part of the code. The specific line of interest in the provided code base is:\n\n```python\n6 data = yaml.load(''' ... ''', Loader=yaml.FullLoader)\n```\n\nThis line would be updated to:\n\n```python\n6 data = yaml.safe_load(''' ... ''')\n```\n\n### Detailed Explanation\n\n#### Original Issue\n\nThe function `yaml.load()` from the PyYAML library is used to parse a YAML string and convert it into a Python object. However, `yaml.load()` has the capability to execute arbitrary code embedded in the YAML content, which can be exploited by an attacker. This is particularly dangerous when loading YAML content from untrusted sources.\n\nFor example, a YAML string could potentially include Python code that is executed, thus performing malicious actions such as deleting files or leaking sensitive data.\n\n#### Changes Introduced by the Commit\n\nThe proposed change involves using `yaml.safe_load()` rather than `yaml.load()`. Here's how the proposed alteration in `Infinidat_munch/tests/test_yaml.py` would look like:\n\n```python\n1 import pytest\n2 from munch import Munch, DefaultMunch\n3 \n4 \n5 def test_from_yaml(yaml):\n6     data = yaml.safe_load('''  # Changed from yaml.load to yaml.safe_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     ''')\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_style=False)\n22     assert dumped == '''foo:\n23   - bar\n24   - lol: true\n25 hello: 42\n26 '''\n```\n\n#### Solution Explanation\n\n1. **Security Improvement**: By switching from `yaml.load()` to `yaml.safe_load()`, the risk of executing arbitrary code from within the YAML content is eliminated. `yaml.safe_load()` only allows a subset of YAML that is safe for loading. It prevents code execution by disallowing tags that could execute arbitrary code.\n\n2. **Maintaining Functionality**: The logic of the test remains unchanged. `yaml.safe_load()` ensures that YAML strings are safely loaded into their corresponding Python objects without the risk of executing malicious payloads. The functionality required for the test, i.e., interpreting the special `!munch` tags to convert them into Munch instances, should still operate correctly as long as these tags fall within the acceptable subset of YAML types that `safe_load` can handle.\n\n#### Conclusion\n\nBy making this change from `yaml.load()` to `yaml.safe_load()`, the security vulnerability is mitigated without sacrificing the functionality of the test. This ensures that the code remains safe and robust, adhering to best practices for handling YAML data."
    },
    "issue_message": {
      "title": "Incorrect attribute access",
      "description": "The test `test_munchify` in `tests/test_munch.py` fails because it attempts to access a non-existent attribute 'cats' on a Munch object. This results in an AttributeError being raised. Modify the test to ensure the correct attributes are being accessed.",
      "explanation": "The issue reported is titled \"Incorrect attribute access\" and pertains to a failing test, `test_munchify`, in the `tests/test_munch.py` file. The test fails because it attempts to access a non-existent attribute 'cats' on a Munch object, leading to an `AttributeError`.\n\n### Summary of the Issue:\nThe issue is due to an attempt to access an attribute 'cats' on a `Munch` object, which does not exist. This results in an `AttributeError`. The `Munch` class is designed to provide dictionary-like access using attribute-style access. The failing code is part of the `test_munchify` function within the test suite.\n\n### Code Analysis:\n1. **Initialization in `__init__.py`**:\n    The `Munch` class is defined in `Infinidat_munch/munch/__init__.py`, which provides functionality for attribute-style access on dictionary items.\n    ```python\n    48 class Munch(dict):  # Munch class extending dictionary\n    ```\n\n2. **Existing Tests in `test_munch.py`**:\n    The existing tests in `tests/test_munch.py` cover various functionalities of `Munch` such as initialization, attribute access, and dictionary updates.\n\n3. **Problematic Test**:\n    ```python\n    def 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].cats == 'i win again'  # This line is incorrect\n    ```\n    - The line `assert b.lol[1].cats == 'i win again'` incorrectly tries to access `b.lol[1].cats`. The actual structure does not support this access pattern.\n\n### Cause of the Issue:\nThe `munchify` function is likely converting a dictionary to a `Munch` object, but the access pattern `b.lol[1].cats` is incorrect. Based on `munchify({'lol': ('cats', {'hah': 'i win again'})})`, `b.lol` should reference a tuple `('cats', Munch({'hah': 'i win again'}))`. Thus, `b.lol[1]` is a `Munch` object but does not have any `cats` attribute; the correct attribute should be `hah`.\n\n### Solution:\nModify the test to correctly verify the attributes based on the real structure of the `Munch` object.\n\n### Commit Content:\nHere's the proposed modification in `test_munch.py`:\n```diff\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].cats == 'i win again'\n+    assert b.lol[1].hah == 'i win again'  # Corrected attribute access\n```\n\n### Explanation of the Commit:\nThe changes made in the commit correct the line where the error is occurring. Instead of trying to access the `cats` attribute on the second element of the tuple `b.lol`, we now correctly access the `hah` attribute:\n- `assert b.lol[1].hah == 'i win again'`\n\nThis correction ensures that the test aligns with the actual structure of the `Munch` object, thus preventing the `AttributeError` by accessing an existing attribute 'hah' in the nested `Munch` object.\n\nThis solution is validated by ensuring that:\n1. `munchify({'lol': ('cats', {'hah': 'i win again'}), ... })` creates `b.lol` as a tuple where:\n   - `b.lol[0]` is `'cats'`\n   - `b.lol[1]` is a `Munch` object with `{'hah': 'i win again'}`.\n2. Accessing `b.lol[1].hah` correctly retrieves the value 'i win again'.\n\nWith this fix in place, running the test suite should result in no `_AttributeError_` being thrown, confirming that the correct attributes are being accessed."
    },
    "issue_ground": {
      "title": "Fix Unmunchify Function KeyError and Add Improper Assertion Error Fixes",
      "description": "### Issue Description\nSeveral issues have been identified that affect the functionality and reliability of the `Infinidat_munch` project.\n\n1. **`unmunchify_cycles` Function KeyError**: The `unmunchify_cycles` function in `munch/__init__.py` does not handle cases where an object is not found in the `seen` dictionary correctly. This results in a KeyError which interrupts the unmunchify process and can potentially cause the application to crash.\n\n2. **Incorrect Assertion in Tests**: There are instances of incorrect assertions within the test suite.\n   - The `test_munchify` test incorrectly asserts the `lol[1].cats` attribute, which leads to false positives or unresolved tests.\n   - The equality checks in tests use `__eq__`, which can be replaced with the `==` operator for clarity and readability.\n\n3. **Path Issue in Test Files**: There are issues with path resolution in one of the test files (`test_readme.py`). The path `_README_PATH` is constructed using `os.path.join`, which may lead to `FileNotFoundError` if the current directory is not set correctly.\n\n4. **Default Value Handling in DefaultMunch**: The tests for the `DefaultMunch` class assume that non-existent keys return `True`, which is not correct. It should instead verify that the expected default values are returned for missing keys.\n\n### Impact\n- **Unmunchify Function**: The unmunchify process will fail when encountering objects not tracked in the `seen` dictionary, resulting in incomplete or inaccurate conversions and potential application crashes.\n- **Test Suite Failures**: Incorrect assertions and comparisons lead to unreliable test results, contributing to an unstable codebase and masking potential bugs.\n- **Path Issues**: The path issue in the tests can lead to failures when the tests are run in environments where the current directory varies, impacting the reliability of the test suite.\n- **Default Value Handling**: Incorrect assumptions about the default value behavior can cause misleading test results and unexpected behavior in application code.\n\n### Steps to Reproduce\n1. Call the `unmunchify` function with cyclical data structures and observe `KeyError` exceptions.\n2. Run the test suite and observe the incorrect assertions in `test_munch.py` and path-related issues in `test_readme.py`.\n3. Verify the default value behavior by using `DefaultMunch` and checking non-existent keys.\n\n### Expected Behavior\n- **Unmunchify Function**: The function should handle the absence of objects in the `seen` dictionary gracefully without raising `KeyError`.\n- **Correct Assertions**: The tests should correctly assert the presence and values of attributes within Munch objects.\n- **Correct Path Resolution**: The test paths should be constructed in a manner that is robust to changes in the current directory.\n- **Accurate Default Value Handling**: The `DefaultMunch` tests should accurately assess the behavior of default values for non-existent keys.\n\n### Suggestions\n- Implement exception handling in the `unmunchify_cycles` function to address cycles and missing references gracefully.\n- Update the assertions in the test suite to reflect accurate and intended behavior, particularly for attribute checks and equality comparisons.\n- Refactor path construction in the tests to guarantee path resolution correctness.\n- Adjust tests for `DefaultMunch` to verify that default values are returned for keys that do not exist.\n",
      "explanation": "### Summary of the Issue\n\nThe issue addresses several bugs and discrepancies in the `Infinidat_munch` project, including:\n\n1. **KeyError in `unmunchify_cycles`:** The function does not handle objects not found in the `seen` dictionary correctly, resulting in a `KeyError`.\n2. **Incorrect Assertions in Tests:** Specific assertions in the test suite are incorrect. For example, the `test_munchify` test incorrectly asserts the `lol[1].cats` attribute.\n3. **Path Issue in Test Files:** The path `_README_PATH` is constructed using `os.path.join`, which can lead to `FileNotFoundError`.\n4. **Default Value Handling in `DefaultMunch`:** Tests assume non-existent keys return `True`, rather than expected default values.\n\n### Explanation of the Commit\n\n**1. Handling `KeyError` in `unmunchify_cycles`:**\n\nIn the `munch/__init__.py` file, the `unmunchify_cycles` function would need exception handling to manage objects not found in the `seen` dictionary gracefully.\n```python\n# Pseudo-code for fix\ndef unmunchify_cycles(obj, seen):\n    try:\n        return seen[id(obj)]\n    except KeyError:\n        pass\n    # handle cycle breaking or other logic\n```\n\n**2. Correcting Assertions in Tests:**\n\nThe incorrect assertions within tests should be updated. For instance, changing the assertion to correctly reflect the presence and values of attributes.\n```python\n# Updating the test_munchify function\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'  # Corrected assertion\n```\n\n**3. Fixing Path Issue in Test Files:**\n\nModify the `_README_PATH` construction in `test_readme.py` to ensure correctness regardless of the current directory.\n```python\nimport os\n\n# Absolute path resolution for README.md\n_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\nassert os.path.exists(_README_PATH)\n```\n\n**4. Fixing Default Value Handling in `DefaultMunch`:**\n\nTests should reflect correct behavior for missing keys in `DefaultMunch`.\n```python\ndef test_default_munch():\n    dm = DefaultMunch(lambda: 'default')\n    assert dm.nonexistent_key == 'default'\n```\n\n### Explanation of How the Commit Solves the Issues\n\n**1. Handling `KeyError` in `unmunchify_cycles`:**\n\nBy adding a try-except block around the `seen` dictionary access, the function can catch and handle `KeyError`. This prevents the application from crashing when an object is not found, thus ensuring a smoother `unmunchify` process.\n\n**2. Correcting Assertions in Tests:**\n\nUpdating the assertions in the tests ensures that they correctly reflect the present attributes and their values. This leads to more reliable and accurate test results, uncovering actual bugs instead of false positives.\n\n**3. Fixing Path Issue in Test Files:**\n\nConstructing the path in an absolute manner guarantees that the file is found regardless of changes to the current working directory. This prevents `FileNotFoundError` and improves the reliability of the tests.\n\n**4. Fixing Default Value Handling in `DefaultMunch`:**\n\nBy verifying the correct default values are returned for nonexistent keys, the tests now accurately assess and reflect the expected behavior of `DefaultMunch`. This ensures that default behavior is properly tested and verified, leading to a more consistent and predictable application code.\n\n### Summary\n\nThe commit addresses these core issues through targeted changes - adding exception handling, updating assertions, correcting path resolution, and refining test logic. These adjustments improve the robustness of the `Infinidat_munch` project both in functionality and test reliability."
    },
    "issue_ground_truth": {
      "title": "Fix Unmunchify Function KeyError and Add Improper Assertion Error Fixes",
      "description": "### Issue Description\nSeveral issues have been identified that affect the functionality and reliability of the `Infinidat_munch` project.\n\n1. **`unmunchify_cycles` Function KeyError**: The `unmunchify_cycles` function in `munch/__init__.py` does not handle cases where an object is not found in the `seen` dictionary correctly. This results in a KeyError which interrupts the unmunchify process and can potentially cause the application to crash.\n\n2. **Incorrect Assertion in Tests**: There are instances of incorrect assertions within the test suite.\n   - The `test_munchify` test incorrectly asserts the `lol[1].cats` attribute, which leads to false positives or unresolved tests.\n   - The equality checks in tests use `__eq__`, which can be replaced with the `==` operator for clarity and readability.\n\n3. **Path Issue in Test Files**: There are issues with path resolution in one of the test files (`test_readme.py`). The path `_README_PATH` is constructed using `os.path.join`, which may lead to `FileNotFoundError` if the current directory is not set correctly.\n\n4. **Default Value Handling in DefaultMunch**: The tests for the `DefaultMunch` class assume that non-existent keys return `True`, which is not correct. It should instead verify that the expected default values are returned for missing keys.\n\n### Impact\n- **Unmunchify Function**: The unmunchify process will fail when encountering objects not tracked in the `seen` dictionary, resulting in incomplete or inaccurate conversions and potential application crashes.\n- **Test Suite Failures**: Incorrect assertions and comparisons lead to unreliable test results, contributing to an unstable codebase and masking potential bugs.\n- **Path Issues**: The path issue in the tests can lead to failures when the tests are run in environments where the current directory varies, impacting the reliability of the test suite.\n- **Default Value Handling**: Incorrect assumptions about the default value behavior can cause misleading test results and unexpected behavior in application code.\n\n### Steps to Reproduce\n1. Call the `unmunchify` function with cyclical data structures and observe `KeyError` exceptions.\n2. Run the test suite and observe the incorrect assertions in `test_munch.py` and path-related issues in `test_readme.py`.\n3. Verify the default value behavior by using `DefaultMunch` and checking non-existent keys.\n\n### Expected Behavior\n- **Unmunchify Function**: The function should handle the absence of objects in the `seen` dictionary gracefully without raising `KeyError`.\n- **Correct Assertions**: The tests should correctly assert the presence and values of attributes within Munch objects.\n- **Correct Path Resolution**: The test paths should be constructed in a manner that is robust to changes in the current directory.\n- **Accurate Default Value Handling**: The `DefaultMunch` tests should accurately assess the behavior of default values for non-existent keys.\n\n### Suggestions\n- Implement exception handling in the `unmunchify_cycles` function to address cycles and missing references gracefully.\n- Update the assertions in the test suite to reflect accurate and intended behavior, particularly for attribute checks and equality comparisons.\n- Refactor path construction in the tests to guarantee path resolution correctness.\n- Adjust tests for `DefaultMunch` to verify that default values are returned for keys that do not exist.\n",
      "explanation": "### Summary of the Issue\n\nThe issue reported multiple problems affecting the `Infinidat_munch` project, primarily related to:\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n    - The `unmunchify_cycles` function can throw a `KeyError` if an object isn't found in the `seen` dictionary, causing the app to crash.\n\n2. **Incorrect Assertions and Path Issues in Tests**:\n    - Some test assertions, particularly in `test_munch.py`, were incorrect.\n    - Path construction in `test_readme.py` was unreliable and could cause `FileNotFoundError`.\n    - Tests for `DefaultMunch` class made incorrect assumptions about default value handling, leading to unreliable results.\n\n3. **Improper Assertion Error Fixes**:\n    - The test assertions for some keys and attributes were incorrectly performed.\n\n### Detailed Commit Content and Solutions\n\nThe commit makes several changes to address these issues:\n\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n   - **Issue**: If an object is not tracked in `seen`, the function throws a `KeyError`.\n   - **Solution**: Introduce exception handling to gracefully manage cases where an object isn't found in `seen`, thereby preventing the `KeyError` and ensuring the function completes normally.\n\n2. **Incorrect Assertions in Tests**:\n   - **Issue**: Assertions in `test_munch.py` were incorrectly asserting attributes, causing false positives.\n   - **Solution**: Correct the assertions to check for the proper attributes and use the equality operator (`==`) instead of direct method calls, enhancing readability and correctness.\n   - This addresses the testing reliability by ensuring tests are accurately verifying the expected behavior.\n\n3. **Path Issue in `test_readme.py`**:\n   - **Issue**: Path construction could result in `FileNotFoundError` if the directory is not correctly set.\n   - **Solution**: Refactor the path construction method to use an absolute path derived from the directory of the file itself, making the path resolution robust and independent of the current working directory.\n\n4. **Default Value Handling in `DefaultMunch`**:\n   - **Issue**: Tests assumed that non-existent keys return `True`, which was incorrect.\n   - **Solution**: Adjust tests to accurately verify that the expected default values are returned for missing keys, ensuring the actual behavior matches the intended design.\n\n### Explanation from Developer's Perspective\n\nThe developer recognized that the current implementation had certain fragile elements:\n- The `unmunchify_cycles` was not robust against missing objects in `seen`.\n- Test cases had incorrect assertions, which undermined the reliability of the results.\n- Path constructions in test scripts were susceptible to the changing current directory.\n- `DefaultMunch` class tests were based on incorrect assumptions about default value handling.\n\n**Detailed Solution Approach**:\n- **Robust Error Handling**: By adding a `try-except` block around the dictionary access in `unmunchify_cycles`, any missing objects will not cause an abrupt termination of the function. This ensures that the conversion process can complete without crashing due to a `KeyError`.\n- **Correct Assertions**: Fixing the test assertions to accurately point to the correct attributes and properties ensures that tests will capture the correct functional behavior and prevent missed bugs or misleading success reports.\n- **Reliable Path Construction**: Refactoring the path resolution to use absolute paths derived from the current file location ensures that tests will run correctly regardless of the current directory from which the tests are initiated. This increases the robustness of the test execution environment.\n- **Accurate Default Handling Tests**: By adjusting the tests for the `DefaultMunch` class to check the returned values against expected behavior, the likelihood of undetected issues in the default value behavior is minimized, thus ensuring features work as designed.\n\n### Conclusion\n\nThe commit addresses and resolves the identified issues by:\n- Implementing robust handling of cyclical unmunchification.\n- Updating and correcting the test assertions to ensure reliability.\n- Fixing path issues to ensure consistent test results.\n- Correcting the default value handling tests to align with the expected functionality.\n\nThese changes fortify the codebase against potential faults and improve the reliability and accuracy of the test suite, leading to a more robust and maintainable project."
    },
    "location_origin": [
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "5": "test_from_yaml"
        },
        "content_all": {
          "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"
        },
        "content_change": {
          "6": "    data = yaml.safe_load('''\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "141": "test_munchify"
        },
        "content_all": {
          "138": "    b = munchify({'lol': ('cats', {'hah': 'i win again'}), 'hello': [{'french': 'salut', 'german': 'hallo'}]})\n",
          "139": "    assert b.hello[0].french == 'salut'\n",
          "140": "\n",
          "141": "    # Incorrect line\n",
          "142": "    assert b.lol[1].cats == 'i win again'\n",
          "143": "\n",
          "144": "    # Correct line\n",
          "145": "    assert b.lol[1].hah == 'i win again'\n"
        },
        "content_change": {
          "142": "    assert b.lol[1].cats == 'i win again'",
          "145": "    assert b.lol[1].hah == 'i win again'"
        }
      }
    ],
    "location_ground": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "511": "unmunchify_cycles"
        },
        "content_all": {
          "510": "",
          "511": "    def unmunchify_cycles(obj):",
          "512": "        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj",
          "513": "        if id(obj) in seen.keys():",
          "514": "            return seen[id(obj)]",
          "515": "        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that",
          "516": "        seen[id(obj)] = partial = pre_unmunchify(obj)",
          "517": "        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)"
        },
        "content_change": {
          "513": "        try:",
          "514": "            return seen[id(obj)]",
          "515": "        except KeyError:",
          "516": "            pass"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "31": "test_munchify"
        },
        "content_all": {
          "30": "",
          "31": "def test_munchify():",
          "32": "    b = munchify({'urmom': {'sez': {'what': 'what'}}})",
          "33": "    assert b.urmom.sez.what == 'what'",
          "34": "",
          "35": "    b = munchify({'lol': ('cats', {'hah': 'i win again'}), 'hello': [{'french': 'salut', 'german': 'hallo'}]})",
          "36": "    assert b.hello[0].french == 'salut'",
          "37": "    assert b.lol[1].cats == 'i win again'"
        },
        "content_change": {
          "37": "    assert b.lol[1].hah == 'i win again'"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "25": "equality_test"
        },
        "content_all": {
          "24": "def equality_test():",
          "25": "    m1 = Munch({'key': 'value'})",
          "26": "    m2 = Munch({'key': 'value'})",
          "27": "    assert m1.__eq__(m2)"
        },
        "content_change": {
          "27": "    assert m1 == m2"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": {
          "5": "global"
        },
        "content_all": {
          "4": "",
          "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')",
          "6": "assert os.path.exists(_README_PATH)",
          "7": "",
          "8": "@pytest.mark.usefixtures(\"yaml\")",
          "9": "def test_readme():",
          "10": "    globs = {",
          "11": "        'print_function': print",
          "12": "    }"
        },
        "content_change": {
          "5": "_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "41": "test_default_munch"
        },
        "content_all": {
          "40": "",
          "41": "def test_default_munch():",
          "42": "    dm = DefaultMunch(lambda: 'default')",
          "43": "    assert dm.nonexistent_key",
          "44": "    assert dm.nonexistent_key == 'default'",
          "45": "    dm.nonexistent_key = 'something'",
          "46": "    assert dm.nonexistent_key == 'something'"
        },
        "content_change": {
          "43": "    assert dm.nonexistent_key == 'default'"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "511": "unmunchify_cycles"
        },
        "content_all": {
          "510": "",
          "511": "def unmunchify_cycles(obj):",
          "512": "    # If we've already begun unmunchifying obj, just return the already-created unmunchified obj",
          "513": "    if id(obj) in seen.keys():",
          "514": "        return seen[id(obj)]",
          "515": "    # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that",
          "516": "    seen[id(obj)] = partial = pre_unmunchify(obj)",
          "517": "    # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)",
          "518": "    return partial"
        },
        "content_change": {
          "513": "    if id(obj) in seen:",
          "514": "        return seen[id(obj)]"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "10": "test_base"
        },
        "content_all": {
          "7": "from munch import DefaultFactoryMunch, AutoMunch, DefaultMunch, Munch, munchify, unmunchify",
          "8": "",
          "9": "def test_base():",
          "10": "    b = Munch()",
          "11": "    b.hello = 'world'",
          "12": "",
          "13": "    assert b.hello == 'world'",
          "14": "    b['hello'] += \"!\"",
          "15": "    assert b.hello == 'world!'",
          "16": "    b.foo = Munch(lol=True)",
          "17": "    assert b.foo.lol is True",
          "18": "    assert b.foo is b['foo']"
        },
        "content_change": {
          "14": "    b['hello'] += \"!\""
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": {
          "9": "test_readme"
        },
        "content_all": {
          "4": "",
          "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')",
          "6": "assert os.path.exists(_README_PATH)",
          "7": "",
          "8": "@pytest.mark.usefixtures(\"yaml\")",
          "9": "def test_readme():",
          "10": "    globs = {",
          "11": "        \"print_function\": print",
          "12": "    }",
          "13": "    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)",
          "14": "    assert not result.failed"
        },
        "content_change": {
          "5": "_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_munch.py",
        "function": {
          "10": "test_default_munch"
        },
        "content_all": {
          "31": "def test_contains():",
          "32": "    b = Munch(ponies='are pretty!')",
          "33": "",
          "34": "    assert 'ponies' in b",
          "35": "    assert 'foo' not in b",
          "36": "",
          "37": "def test_default_munch():",
          "38": "    dm = DefaultMunch(lambda: 'default')",
          "39": "    assert dm.non_existent == True"
        },
        "content_change": {
          "39": "    assert dm.non_existent == 'default'"
        }
      }
    ],
    "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": {},
        "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",
          "142": "    assert b.lol[1].cats == 'i win again'\n"
        },
        "content_change": {
          "142": "    assert b.lol[1].cats == 'i win again'\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": null,
        "content_all": {
          "2": "import os\n",
          "3": "import pytest\n",
          "4": "\n",
          "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n",
          "6": "assert os.path.exists(_README_PATH)\n",
          "7": "\n",
          "8": "\n"
        },
        "content_change": {
          "5": "_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "41": "test_fromYAML"
        },
        "content_all": {
          "43": "    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n",
          "44": "    obj = Munch.fromYAML(yaml_str)\n",
          "45": "    assert type(obj) == Munch\n",
          "46": "    assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n"
        },
        "content_change": {
          "46": "    assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "51": "test_fromYAML_default_munch"
        },
        "content_all": {
          "50": "@pytest.mark.usefixtures('yaml')\n",
          "51": "def test_fromYAML_default_munch():\n",
          "52": "    # pylint: disable=unidiomatic-typecheck\n",
          "53": "    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n",
          "54": "    obj = DefaultMunch.fromYAML(yaml_str, object())\n",
          "55": "    assert type(obj) == DefaultMunch\n",
          "56": "    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n",
          "57": "    assert obj['not_exist']\n",
          "58": "    assert obj.not_exist\n"
        },
        "content_change": {
          "54": "    obj = DefaultMunch.fromYAML(yaml_str, object())\n",
          "57": "    assert obj['not_exist']\n",
          "58": "    assert obj.not_exist\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": null,
        "content_all": {
          "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
          "48": "\n",
          "49": "\n",
          "59": "    assert obj.toYAML().__eq__(yaml_str)\n"
        },
        "content_change": {
          "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
          "59": "    assert obj.toYAML().__eq__(yaml_str)\n"
        }
      }
    ],
    "patch_i": "\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -5,7 +5,7 @@\n def test_from_yaml(yaml):\n-    data = yaml.load('''\n+    data = yaml.safe_load('''\n     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n     Block style: !munch\n       Clark : Evans\n",
    "patch_im": "\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -31,7 +31,7 @@ def test_contains():\n \n def 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].cats == 'i win again'  # This line is incorrect\n+    assert b.lol[1].hah == 'i win again'  # Corrected attribute access\n",
    "patch_il": "\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -3,7 +3,7 @@\n \n \n def test_from_yaml(yaml):\n-    data = yaml.load('''\n+    data = yaml.safe_load('''\n     Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n     Block style: !munch\n       Clark : Evans\n",
    "patch_iml": "\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -138,7 +138,7 @@\n     b = munchify({'lol': ('cats', {'hah': 'i win again'}), 'hello': [{'french': 'salut', 'german': 'hallo'}]})\n     assert b.hello[0].french == 'salut'\n \n-    assert b.lol[1].cats == 'i win again'\n+    assert b.lol[1].hah == 'i win again'\n \n     assert isinstance(b.lol, tuple)\n     assert b.lol[0] == 'cats'\n",
    "patch_ground": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -515,6 +515,10 @@ def unmunchify(obj):\n         return obj\n     seen = {}\n     def convert(obj):\n+        try:\n+            return seen[id(obj)]\n+        except KeyError:\n+            pass\n         if isinstance(obj, Munch):\n             result = {}\n             seen[id(obj)] = result\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -54,7 +54,7 @@ def 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.lol[1].cats.hah == 'i win again'  # Incorrect assertion\n+    assert b.lol[1].hah == 'i win again'  # Corrected assertion\n\n def test_unmunchify():\n     b = munchify({'urmom': {'sez': {'what': 'what'}}})\n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -5,7 +5,7 @@ import pytest\n\n-_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n+_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n assert os.path.exists(_README_PATH)\n\n @pytest.mark.usefixtures(\"yaml\")\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -25,6 +25,12 @@ def test_from_yaml(yaml):\n def test_default_munch():\n     dm = DefaultMunch(lambda: 'default')\n     assert dm.nonexistent_key == 'default'\n",
    "patch_ground_location": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,7 +510,8 @@\n \n def unmunchify_cycles(obj, seen):\n     try:\n-        return seen[id(obj)]\n+        return seen[id(obj)]\n+    except KeyError:\n         pass\n     # If obj is a Munch, convert it by unmunchifying all items\n     if isinstance(obj, Munch):\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -31,7 +31,7 @@\n def 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+    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].cats == 'i win again'\n+    assert b.lol[1].hah == 'i win again'\n \n def equality_test():\n     m1 = Munch({'key': 'value'})\n@@ -41,7 +41,7 @@\n \n def test_default_munch():\n     dm = DefaultMunch(lambda: 'default')\n-    assert dm.nonexistent_key\n+    assert dm.nonexistent_key == 'default'\n     assert dm.nonexistent_key == 'default'\n     dm.nonexistent_key = 'something'\n     assert dm.nonexistent_key == 'something'\n@@ -27,7 +27,7 @@\n     assert m1.__eq__(m2)\n+    assert m1 == m2\n \n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -5,7 +5,7 @@\n _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n assert os.path.exists(_README_PATH)\n+_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n assert os.path.exists(_README_PATH)\n",
    "patch_ground_exp": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,7 +510,7 @@\n \n def unmunchify_cycles(obj, seen=None):\n     if seen is None:\n         seen = {}\n-    if id(obj) in seen.keys():\n+    if id(obj) in seen:\n         return seen[id(obj)]\n     seen[id(obj)] = partial = pre_unmunchify(obj)\n     return partial\n--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -7,6 +7,7 @@\n \n def test_base():\n     b = Munch()\n     b.hello = 'world'\n+    \n     assert b.hello == 'world'\n     b['hello'] += \"!\"\n     assert b.hello == 'world!'\n     b.foo = Munch(lol=True)\n@@ -37,7 +38,7 @@\n \n def test_default_munch():\n     dm = DefaultMunch(lambda: 'default')\n-    assert dm.non_existent == True\n+    assert dm.non_existent == 'default'\n \n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -4,7 +4,7 @@\n \n _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n assert os.path.exists(_README_PATH)\n \n def test_readme():\n     globs = {\n         'print_function': print\n     }\n\n",
    "patch_ground_all": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,7 +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--- a/Infinidat_munch/tests/test_munch.py\n+++ b/Infinidat_munch/tests/test_munch.py\n@@ -97,8 +97,8 @@\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+    assert \"The {} who say {}!\".format(format_munch.knights, format_munch.ni) == 'The lolcats who say can haz!'\n \n-def test_contains():\n-    b = Munch(ponies='are pretty!')\n+def test_munchify():\n+    b = Munch(lol=[{}, Munch(cats='i win again')])\n     assert 'ponies' in b\n     assert 'lol' in b\n     assert b.lol[1].cats == 'i win again'\n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -5,7 +5,7 @@\n \n _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n assert os.path.exists(_README_PATH)\n \n @pytest.mark.usefixtures(\"yaml\")\n def test_readme():\n     globs = {\n         'print_function': print\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -56,7 +56,8 @@\n     assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n-    assert obj['not_exist']\n-    assert obj.not_exist\n+    assert obj.get('not_exist') is None\n+    assert hasattr(obj, 'not_exist') == False\n \n def test_word_exists():\n     ym = DefaultMunch.fromYAML('No Exist: world', '')\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--- 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@@ -139,7 +142,7 @@\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].cats == 'i win again'\n+    assert b.lol[1].hah == 'i win again'\n \n def test_munchify_with_namedtuple():\n     nt = namedtuple('nt', ['prop_a', 'prop_b'])\n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -2,7 +2,8 @@\n import os\n import pytest\n \n-_README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n+_HERE = os.path.abspath(os.path.dirname(__file__))\n+_README_PATH = os.path.join(_HERE, '..', 'README.md')\n assert os.path.exists(_README_PATH)\n \n \n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -43,17 +43,18 @@\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.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n-    assert obj.toYAML().__eq__(yaml_str)\n+    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n+    assert obj.toYAML() == yaml_str\n \n \n @pytest.mark.usefixtures('yaml')\n def test_fromYAML_default_munch():\n     # pylint: disable=unidiomatic-typecheck\n     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n-    obj = DefaultMunch.fromYAML(yaml_str, object())\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']\n-    assert obj.not_exist\n-    assert obj.toYAML().__eq__(yaml_str)\n+    assert obj['not_exist'] is default_value\n+    assert obj.not_exist is default_value\n+    assert obj.toYAML() == yaml_str\n",
    "message": "============================================================================================== FAILURES ===============================================================================================\n____________________________________________________________________________________________ test_munchify ____________________________________________________________________________________________\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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)\nE           AttributeError: 'Munch' object has no attribute 'cats'\n\nmunch/__init__.py:116: AttributeError\n\nDuring handling of the above exception, another exception occurred:\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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]\nE               KeyError: 'cats'\n\nmunch/__init__.py:119: KeyError\n\nDuring handling of the above exception, another exception occurred:\n\n    def 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].cats == 'i win again'\n\ntests/test_munch.py:142: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n\nself = Munch({'hah': 'i win again'}), k = 'cats'\n\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)\nE               AttributeError: cats\n\nmunch/__init__.py:121: AttributeError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_munch.py::test_munchify - AttributeError: cats\n==================================================================================== 1 failed, 108 passed in 0.12s ====================================================================================",
    "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/test_readme.py",
        "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n6 assert os.path.exists(_README_PATH)\n7 \n8 \n9 @pytest.mark.usefixtures(\"yaml\")\n10 def test_readme():\n11     globs = {\n12         'print_function': print\n13     }\n14     result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n15     assert not result.failed"
      },
      {
        "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/setup.py",
        "content": "1 from setuptools import setup\n2 \n3 \n4 setup(\n5     setup_requires=['pbr>=3.0', 'setuptools>=61'],\n6     pbr=True,\n7     long_description_content_type='text/markdown; charset=UTF-8',\n8     keywords=['munch', 'dict', 'mapping', 'container', 'collection'],\n9     python_requires=\">=3.6\",\n10 )"
      }
    ],
    "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 4,
      "Description": 4,
      "Reproducibility": 3,
      "Relevance": 6,
      "Explanation": 7,
      "Overall": 5
    },
    "issue_ground": {
      "Title": 9,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 9,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Fix Unmunchify Function KeyError and Add Improper Assertion Error Fixes",
      "description": "### Issue Description\nSeveral issues have been identified that affect the functionality and reliability of the `Infinidat_munch` project.\n\n1. **`unmunchify_cycles` Function KeyError**: The `unmunchify_cycles` function in `munch/__init__.py` does not handle cases where an object is not found in the `seen` dictionary correctly. This results in a KeyError which interrupts the unmunchify process and can potentially cause the application to crash.\n\n2. **Incorrect Assertion in Tests**: There are instances of incorrect assertions within the test suite.\n   - The `test_munchify` test incorrectly asserts the `lol[1].cats` attribute, which leads to false positives or unresolved tests.\n   - The equality checks in tests use `__eq__`, which can be replaced with the `==` operator for clarity and readability.\n\n3. **Path Issue in Test Files**: There are issues with path resolution in one of the test files (`test_readme.py`). The path `_README_PATH` is constructed using `os.path.join`, which may lead to `FileNotFoundError` if the current directory is not set correctly.\n\n4. **Default Value Handling in DefaultMunch**: The tests for the `DefaultMunch` class assume that non-existent keys return `True`, which is not correct. It should instead verify that the expected default values are returned for missing keys.\n\n### Impact\n- **Unmunchify Function**: The unmunchify process will fail when encountering objects not tracked in the `seen` dictionary, resulting in incomplete or inaccurate conversions and potential application crashes.\n- **Test Suite Failures**: Incorrect assertions and comparisons lead to unreliable test results, contributing to an unstable codebase and masking potential bugs.\n- **Path Issues**: The path issue in the tests can lead to failures when the tests are run in environments where the current directory varies, impacting the reliability of the test suite.\n- **Default Value Handling**: Incorrect assumptions about the default value behavior can cause misleading test results and unexpected behavior in application code.\n\n### Steps to Reproduce\n1. Call the `unmunchify` function with cyclical data structures and observe `KeyError` exceptions.\n2. Run the test suite and observe the incorrect assertions in `test_munch.py` and path-related issues in `test_readme.py`.\n3. Verify the default value behavior by using `DefaultMunch` and checking non-existent keys.\n\n### Expected Behavior\n- **Unmunchify Function**: The function should handle the absence of objects in the `seen` dictionary gracefully without raising `KeyError`.\n- **Correct Assertions**: The tests should correctly assert the presence and values of attributes within Munch objects.\n- **Correct Path Resolution**: The test paths should be constructed in a manner that is robust to changes in the current directory.\n- **Accurate Default Value Handling**: The `DefaultMunch` tests should accurately assess the behavior of default values for non-existent keys.\n\n### Suggestions\n- Implement exception handling in the `unmunchify_cycles` function to address cycles and missing references gracefully.\n- Update the assertions in the test suite to reflect accurate and intended behavior, particularly for attribute checks and equality comparisons.\n- Refactor path construction in the tests to guarantee path resolution correctness.\n- Adjust tests for `DefaultMunch` to verify that default values are returned for keys that do not exist.\n",
      "explanation": "### Summary of the Issue\n\nThe issue reported multiple problems affecting the `Infinidat_munch` project, primarily related to:\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n    - The `unmunchify_cycles` function can throw a `KeyError` if an object isn't found in the `seen` dictionary, causing the app to crash.\n\n2. **Incorrect Assertions and Path Issues in Tests**:\n    - Some test assertions, particularly in `test_munch.py`, were incorrect.\n    - Path construction in `test_readme.py` was unreliable and could cause `FileNotFoundError`.\n    - Tests for `DefaultMunch` class made incorrect assumptions about default value handling, leading to unreliable results.\n\n3. **Improper Assertion Error Fixes**:\n    - The test assertions for some keys and attributes were incorrectly performed.\n\n### Detailed Commit Content and Solutions\n\nThe commit makes several changes to address these issues:\n\n1. **Handling Missing Objects in `unmunchify_cycles`**:\n   - **Issue**: If an object is not tracked in `seen`, the function throws a `KeyError`.\n   - **Solution**: Introduce exception handling to gracefully manage cases where an object isn't found in `seen`, thereby preventing the `KeyError` and ensuring the function completes normally.\n\n2. **Incorrect Assertions in Tests**:\n   - **Issue**: Assertions in `test_munch.py` were incorrectly asserting attributes, causing false positives.\n   - **Solution**: Correct the assertions to check for the proper attributes and use the equality operator (`==`) instead of direct method calls, enhancing readability and correctness.\n   - This addresses the testing reliability by ensuring tests are accurately verifying the expected behavior.\n\n3. **Path Issue in `test_readme.py`**:\n   - **Issue**: Path construction could result in `FileNotFoundError` if the directory is not correctly set.\n   - **Solution**: Refactor the path construction method to use an absolute path derived from the directory of the file itself, making the path resolution robust and independent of the current working directory.\n\n4. **Default Value Handling in `DefaultMunch`**:\n   - **Issue**: Tests assumed that non-existent keys return `True`, which was incorrect.\n   - **Solution**: Adjust tests to accurately verify that the expected default values are returned for missing keys, ensuring the actual behavior matches the intended design.\n\n### Explanation from Developer's Perspective\n\nThe developer recognized that the current implementation had certain fragile elements:\n- The `unmunchify_cycles` was not robust against missing objects in `seen`.\n- Test cases had incorrect assertions, which undermined the reliability of the results.\n- Path constructions in test scripts were susceptible to the changing current directory.\n- `DefaultMunch` class tests were based on incorrect assumptions about default value handling.\n\n**Detailed Solution Approach**:\n- **Robust Error Handling**: By adding a `try-except` block around the dictionary access in `unmunchify_cycles`, any missing objects will not cause an abrupt termination of the function. This ensures that the conversion process can complete without crashing due to a `KeyError`.\n- **Correct Assertions**: Fixing the test assertions to accurately point to the correct attributes and properties ensures that tests will capture the correct functional behavior and prevent missed bugs or misleading success reports.\n- **Reliable Path Construction**: Refactoring the path resolution to use absolute paths derived from the current file location ensures that tests will run correctly regardless of the current directory from which the tests are initiated. This increases the robustness of the test execution environment.\n- **Accurate Default Handling Tests**: By adjusting the tests for the `DefaultMunch` class to check the returned values against expected behavior, the likelihood of undetected issues in the default value behavior is minimized, thus ensuring features work as designed.\n\n### Conclusion\n\nThe commit addresses and resolves the identified issues by:\n- Implementing robust handling of cyclical unmunchification.\n- Updating and correcting the test assertions to ensure reliability.\n- Fixing path issues to ensure consistent test results.\n- Correcting the default value handling tests to align with the expected functionality.\n\nThese changes fortify the codebase against potential faults and improve the reliability and accuracy of the test suite, leading to a more robust and maintainable project."
    }
  }
}