{
  "RepoName": "https://github.com/Infinidat/munch.git",
  "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "Infinidat_munch/setup.py",
      "content": "from setuptools import setup\n\n\nsetup(\n    setup_requires=['pbr>=3.0', 'setuptools>=61'],\n    pbr=True,\n    long_description_content_type='text/markdown; charset=UTF-8',\n    keywords=['munch', 'dict', 'mapping', 'container', 'collection'],\n    python_requires=\">=3.6\",\n)\n"
    },
    {
      "path": "Infinidat_munch/munch/__init__.py",
      "content": "\"\"\" Munch is a subclass of dict with attribute-style access.\n\n    >>> b = Munch()\n    >>> b.hello = 'world'\n    >>> b.hello\n    'world'\n    >>> b['hello'] += \"!\"\n    >>> b.hello\n    'world!'\n    >>> b.foo = Munch(lol=True)\n    >>> b.foo.lol\n    True\n    >>> b.foo is b['foo']\n    True\n\n    It is safe to import * from this module:\n\n        __all__ = ('Munch', 'munchify','unmunchify')\n\n    un/munchify provide dictionary conversion; Munches can also be\n    converted via Munch.to/fromDict().\n\"\"\"\n\nfrom collections.abc import Mapping\n\ntry:\n    # For python 3.8 and later\n    import importlib.metadata as importlib_metadata\nexcept ImportError:\n    # For everyone else\n    import importlib_metadata\ntry:\n    __version__ = importlib_metadata.version(__name__)\nexcept importlib_metadata.PackageNotFoundError:\n    # package is not installed\n    __version__ = \"0.0.0\"\n\n\ntry:\n    VERSION = tuple(map(int, __version__.split('+')[0].split('.')[:3]))\nexcept ValueError:\n    VERSION = (0, 0, 0)\n\n__all__ = ('Munch', 'munchify', 'DefaultMunch', 'DefaultFactoryMunch', 'RecursiveMunch', 'unmunchify')\n\n\n\nclass Munch(dict):\n    \"\"\" A dictionary that provides attribute-style access.\n\n        >>> b = Munch()\n        >>> b.hello = 'world'\n        >>> b.hello\n        'world'\n        >>> b['hello'] += \"!\"\n        >>> b.hello\n        'world!'\n        >>> b.foo = Munch(lol=True)\n        >>> b.foo.lol\n        True\n        >>> b.foo is b['foo']\n        True\n\n        A Munch is a subclass of dict; it supports all the methods a dict does...\n\n        >>> sorted(b.keys())\n        ['foo', 'hello']\n\n        Including update()...\n\n        >>> b.update({ 'ponies': 'are pretty!' }, hello=42)\n        >>> print (repr(b))\n        Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n        As well as iteration...\n\n        >>> sorted([ (k,b[k]) for k in b ])\n        [('foo', Munch({'lol': True})), ('hello', 42), ('ponies', 'are pretty!')]\n\n        And \"splats\".\n\n        >>> \"The {knights} who say {ni}!\".format(**Munch(knights='lolcats', ni='can haz'))\n        'The lolcats who say can haz!'\n\n        See unmunchify/Munch.toDict, munchify/Munch.fromDict for notes about conversion.\n    \"\"\"\n    def __init__(self, *args, **kwargs):  # pylint: disable=super-init-not-called\n        self.update(*args, **kwargs)\n\n    # only called if k not found in normal places\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise throws AttributeError.\n\n            nb. __getattr__ is only called if key is not found in normal places.\n\n            >>> b = Munch(bar='baz', lol={})\n            >>> b.foo\n            Traceback (most recent call last):\n                ...\n            AttributeError: foo\n\n            >>> b.bar\n            'baz'\n            >>> getattr(b, 'bar')\n            'baz'\n            >>> b['bar']\n            'baz'\n\n            >>> b.lol is b['lol']\n            True\n            >>> b.lol is getattr(b, 'lol')\n            True\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            return object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                return self[k]\n            except KeyError:\n                raise AttributeError(k)\n\n    def __setattr__(self, k, v):\n        \"\"\" Sets attribute k if it exists, otherwise sets key k. A KeyError\n            raised by set-item (only likely if you subclass Munch) will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(foo='bar', this_is='useful when subclassing')\n            >>> hasattr(b.values, '__call__')\n            True\n            >>> b.values = 'uh oh'\n            >>> b.values\n            'uh oh'\n            >>> b['values']\n            Traceback (most recent call last):\n                ...\n            KeyError: 'values'\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                self[k] = v\n            except:\n                raise AttributeError(k)\n        else:\n            object.__setattr__(self, k, v)\n\n    def __delattr__(self, k):\n        \"\"\" Deletes attribute k if it exists, otherwise deletes key k. A KeyError\n            raised by deleting the key--such as when the key is missing--will\n            propagate as an AttributeError instead.\n\n            >>> b = Munch(lol=42)\n            >>> del b.lol\n            >>> b.lol\n            Traceback (most recent call last):\n                ...\n            AttributeError: lol\n        \"\"\"\n        try:\n            # Throws exception if not in prototype chain\n            object.__getattribute__(self, k)\n        except AttributeError:\n            try:\n                del self[k]\n            except KeyError:\n                raise AttributeError(k)\n        else:\n            object.__delattr__(self, k)\n\n    def toDict(self):\n        \"\"\" Recursively converts a munch back into a dictionary.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> sorted(b.toDict().items())\n            [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n            See unmunchify for more info.\n        \"\"\"\n        return unmunchify(self)\n\n    @property\n    def __dict__(self):\n        return self.toDict()\n\n    def __repr__(self):\n        \"\"\" Invertible* string-form of a Munch.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> print (repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n            >>> eval(repr(b))\n            Munch({'ponies': 'are pretty!', 'foo': Munch({'lol': True}), 'hello': 42})\n\n            >>> with_spaces = Munch({1: 2, 'a b': 9, 'c': Munch({'simple': 5})})\n            >>> print (repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n            >>> eval(repr(with_spaces))\n            Munch({'a b': 9, 1: 2, 'c': Munch({'simple': 5})})\n\n            (*) Invertible so long as collection contents are each repr-invertible.\n        \"\"\"\n        return f'{self.__class__.__name__}({dict.__repr__(self)})'\n\n    def __dir__(self):\n        return list(self.keys())\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return {k: v for k, v in self.items()}\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        self.update(state)\n\n    __members__ = __dir__  # for python2.x compatibility\n\n    @classmethod\n    def fromDict(cls, d):\n        \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n            >>> b = Munch.fromDict({'urmom': {'sez': {'what': 'what'}}})\n            >>> b.urmom.sez.what\n            'what'\n\n            See munchify for more info.\n        \"\"\"\n        return munchify(d, cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n    def update(self, *args, **kwargs):\n        \"\"\"\n        Override built-in method to call custom __setitem__ method that may\n        be defined in subclasses.\n        \"\"\"\n        for k, v in dict(*args, **kwargs).items():\n            self[k] = v\n\n    def get(self, k, d=None):\n        \"\"\"\n        D.get(k[,d]) -> D[k] if k in D, else d.  d defaults to None.\n        \"\"\"\n        if k not in self:\n            return d\n        return self[k]\n\n    def setdefault(self, k, d=None):\n        \"\"\"\n        D.setdefault(k[,d]) -> D.get(k,d), also set D[k]=d if k not in D\n        \"\"\"\n        if k not in self:\n            self[k] = d\n        return self[k]\n\n\nclass AutoMunch(Munch):\n    def __setattr__(self, k, v):\n        \"\"\" Works the same as Munch.__setattr__ but if you supply\n            a dictionary as value it will convert it to another Munch.\n        \"\"\"\n        if isinstance(v, Mapping) and not isinstance(v, (AutoMunch, Munch)):\n            v = munchify(v, AutoMunch)\n        super().__setattr__(k, v)\n\n\nclass DefaultMunch(Munch):\n    \"\"\"\n    A Munch that returns a user-specified value for missing keys.\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        \"\"\" Construct a new DefaultMunch. Like collections.defaultdict, the\n            first argument is the default value; subsequent arguments are the\n            same as those for dict.\n        \"\"\"\n        # Mimic collections.defaultdict constructor\n        if args:\n            default = args[0]\n            args = args[1:]\n        else:\n            default = None\n        super().__init__(*args, **kwargs)\n        self.__default__ = default\n\n    def __getattr__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getattr__(k)\n        except AttributeError:\n            return self.__default__\n\n    def __setattr__(self, k, v):\n        if k == '__default__':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __getitem__(self, k):\n        \"\"\" Gets key if it exists, otherwise returns the default value.\"\"\"\n        try:\n            return super().__getitem__(k)\n        except KeyError:\n            return self.__default__\n\n    def __getstate__(self):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        return (self.__default__, {k: v for k, v in self.items()})\n\n    def __setstate__(self, state):\n        \"\"\" Implement a serializable interface used for pickling.\n\n        See https://docs.python.org/3.6/library/pickle.html.\n        \"\"\"\n        self.clear()\n        default, state_dict = state\n        self.update(state_dict)\n        self.__default__ = default\n\n    @classmethod\n    def fromDict(cls, d, default=None):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default=self.__default__)\n\n    def __repr__(self):\n        return f'{type(self).__name__}({self.__undefined__!r}, {dict.__repr__(self)})'\n\n\nclass DefaultFactoryMunch(Munch):\n    \"\"\" A Munch that calls a user-specified function to generate values for\n        missing keys like collections.defaultdict.\n\n        >>> b = DefaultFactoryMunch(list, {'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        []\n        >>> b.bar.append('hello')\n        >>> b.bar\n        ['hello']\n    \"\"\"\n\n    def __init__(self, default_factory, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.default_factory = default_factory\n\n    @classmethod\n    def fromDict(cls, d, default_factory):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=lambda d_: cls(default_factory, d_))\n\n    def copy(self):\n        return type(self).fromDict(self, default_factory=self.default_factory)\n\n    def __repr__(self):\n        factory = self.default_factory.__name__\n        return f'{type(self).__name__}({factory}, {dict.__repr__(self)})'\n\n    def __setattr__(self, k, v):\n        if k == 'default_factory':\n            object.__setattr__(self, k, v)\n        else:\n            super().__setattr__(k, v)\n\n    def __missing__(self, k):\n        self[k] = self.default_factory()\n        return self[k]\n\n\nclass RecursiveMunch(DefaultFactoryMunch):\n    \"\"\"A Munch that calls an instance of itself to generate values for\n        missing keys.\n\n        >>> b = RecursiveMunch({'hello': 'world!'})\n        >>> b.hello\n        'world!'\n        >>> b.foo\n        RecursiveMunch(RecursiveMunch, {})\n        >>> b.bar.okay = 'hello'\n        >>> b.bar\n        RecursiveMunch(RecursiveMunch, {'okay': 'hello'})\n        >>> b\n        RecursiveMunch(RecursiveMunch, {'hello': 'world!', 'foo': RecursiveMunch(RecursiveMunch, {}),\n        'bar': RecursiveMunch(RecursiveMunch, {'okay': 'hello'})})\n    \"\"\"\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(RecursiveMunch, *args, **kwargs)\n\n    @classmethod\n    def fromDict(cls, d):\n        # pylint: disable=arguments-differ\n        return munchify(d, factory=cls)\n\n    def copy(self):\n        return type(self).fromDict(self)\n\n\n# While we could convert abstract types like Mapping or Iterable, I think\n# munchify is more likely to \"do what you mean\" if it is conservative about\n# casting (ex: isinstance(str,Iterable) == True ).\n#\n# Should you disagree, it is not difficult to duplicate this function with\n# more aggressive coercion to suit your own purposes.\n\ndef munchify(x, factory=Munch):\n    \"\"\" Recursively transforms a dictionary into a Munch via copy.\n\n        >>> b = munchify({'urmom': {'sez': {'what': 'what'}}})\n        >>> b.urmom.sez.what\n        'what'\n\n        munchify can handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = munchify({ 'lol': ('cats', {'hah':'i win again'}),\n        ...         'hello': [{'french':'salut', 'german':'hallo'}] })\n        >>> b.hello[0].french\n        'salut'\n        >>> b.lol[1].hah\n        'i win again'\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def munchify_cycles(obj):\n        partial, already_seen = pre_munchify_cycles(obj)\n        if already_seen:\n            return partial\n        return post_munchify(partial, obj)\n\n    def pre_munchify_cycles(obj):\n        # If we've already begun munchifying obj, just return the already-created munchified obj\n        try:\n            return seen[id(obj)], True\n        except KeyError:\n            pass\n\n        # Otherwise, first partly munchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_munchify(obj)\n        return partial, False\n\n    def pre_munchify(obj):\n        # Here we return a skeleton of munchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_munchify\n        if isinstance(obj, Mapping):\n            return factory({})\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(pre_munchify_cycles(item)[0] for item in obj)\n        else:\n            return obj\n\n    def post_munchify(partial, obj):\n        # Here we finish munchifying the parts of obj that were deferred by pre_munchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, munchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(munchify_cycles(item) for item in obj)\n        elif isinstance(obj, tuple):\n            for (item_partial, item) in zip(partial, obj):\n                post_munchify(item_partial, item)\n\n        return partial\n\n    return munchify_cycles(x)\n\n\ndef unmunchify(x):\n    \"\"\" Recursively converts a Munch into a dictionary.\n\n        >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n        >>> sorted(unmunchify(b).items())\n        [('foo', {'lol': True}), ('hello', 42), ('ponies', 'are pretty!')]\n\n        unmunchify will handle intermediary dicts, lists and tuples (as well as\n        their subclasses), but ymmv on custom datatypes.\n\n        >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42,\n        ...         ponies=('are pretty!', Munch(lies='are trouble!')))\n        >>> sorted(unmunchify(b).items()) #doctest: +NORMALIZE_WHITESPACE\n        [('foo', ['bar', {'lol': True}]), ('hello', 42), ('ponies', ('are pretty!', {'lies': 'are trouble!'}))]\n\n        nb. As dicts are not hashable, they cannot be nested in sets/frozensets.\n    \"\"\"\n\n    # Munchify x, using `seen` to track object cycles\n    seen = dict()\n\n    def unmunchify_cycles(obj):\n        # If we've already begun unmunchifying obj, just return the already-created unmunchified obj\n        if id(obj) in seen.keys():\n            return seen[id(obj)]\n        # Otherwise, first partly unmunchify obj (but without descending into any lists or dicts) and save that\n        seen[id(obj)] = partial = pre_unmunchify(obj)\n        # Then finish unmunchifying lists and dicts inside obj (reusing unmunchified obj if cycles are encountered)\n        return post_unmunchify(partial, obj)\n\n    def pre_unmunchify(obj):\n        # Here we return a skeleton of unmunchified obj, which is enough to save for later (in case\n        # we need to break cycles) but it needs to filled out in post_unmunchify\n        if isinstance(obj, Mapping):\n            return dict()\n        elif isinstance(obj, list):\n            return type(obj)()\n        elif isinstance(obj, tuple):\n            type_factory = getattr(obj, \"_make\", type(obj))\n            return type_factory(unmunchify_cycles(item) for item in obj)\n        else:\n            return obj\n\n    def post_unmunchify(partial, obj):\n        # Here we finish unmunchifying the parts of obj that were deferred by pre_unmunchify because they\n        # might be involved in a cycle\n        if isinstance(obj, Mapping):\n            partial.update((k, unmunchify_cycles(obj[k])) for k in obj.keys())\n        elif isinstance(obj, list):\n            partial.extend(unmunchify_cycles(v) for v in obj)\n        elif isinstance(obj, tuple):\n            for (value_partial, value) in zip(partial, obj):\n                post_unmunchify(value_partial, value)\n\n        return partial\n\n    return unmunchify_cycles(x)\n\n\n# Serialization\n\ntry:\n    try:\n        import json\n    except ImportError:\n        import simplejson as json\n\n    def toJSON(self, **options):\n        \"\"\" Serializes this Munch to JSON. Accepts the same keyword options as `json.dumps()`.\n\n            >>> b = Munch(foo=Munch(lol=True), hello=42, ponies='are pretty!')\n            >>> json.dumps(b) == b.toJSON()\n            True\n        \"\"\"\n        return json.dumps(self, **options)\n\n    def fromJSON(cls, stream, *args, **kwargs):\n        \"\"\" Deserializes JSON to Munch or any of its subclasses.\n        \"\"\"\n        factory = lambda d: cls(*(args + (d,)), **kwargs)\n        return munchify(json.loads(stream), factory=factory)\n\n    Munch.toJSON = toJSON\n    Munch.fromJSON = classmethod(fromJSON)\n\nexcept ImportError:\n    pass\n\n\ntry:\n    # Attempt to register ourself with PyYAML as a representer\n    import yaml\n    from yaml.representer import Representer, SafeRepresenter\n\n    def from_yaml(loader, node):\n        \"\"\" PyYAML support for Munches using the tag `!munch` and `!munch.Munch`.\n\n            >>> import yaml\n            >>> yaml.load('''\n            ... Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n            ... Block style: !munch\n            ...   Clark : Evans\n            ...   Brian : Ingerson\n            ...   Oren  : Ben-Kiki\n            ... ''') #doctest: +NORMALIZE_WHITESPACE\n            {'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n             'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki')}\n\n            This module registers itself automatically to cover both Munch and any\n            subclasses. Should you want to customize the representation of a subclass,\n            simply register it with PyYAML yourself.\n        \"\"\"\n        data = Munch()\n        yield data\n        value = loader.construct_mapping(node)\n        data.update(value)\n\n    def to_yaml_safe(dumper, data):\n        \"\"\" Converts Munch to a normal mapping node, making it appear as a\n            dict in the YAML output.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.safe_dump(b, default_flow_style=True)\n            '{foo: [bar, {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_dict(data)\n\n    def to_yaml(dumper, data):\n        \"\"\" Converts Munch to a representation node.\n\n            >>> b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n            >>> import yaml\n            >>> yaml.dump(b, default_flow_style=True)\n            '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\\\n'\n        \"\"\"\n        return dumper.represent_mapping('!munch.Munch', data)\n\n    for loader_name in (\"BaseLoader\", \"FullLoader\", \"SafeLoader\", \"Loader\", \"UnsafeLoader\", \"DangerLoader\"):\n        LoaderCls = getattr(yaml, loader_name, None)\n        if LoaderCls is None:\n            # This code supports both PyYAML 4.x and 5.x versions\n            continue\n        yaml.add_constructor('!munch', from_yaml, Loader=LoaderCls)\n        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].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_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)\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_fromYAML_default_munch _____________________________________________________________________________________\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)\n        assert type(obj) == DefaultMunch\n        assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n>       assert obj['not_exist']\nE       assert {}\n\ntests/test_yaml.py:57: AssertionError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_yaml.py::test_fromYAML_default_munch - assert {}\n==================================================================================== 1 failed, 108 passed in 0.08s ====================================================================================",
  "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--- 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)\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"
      },
      "content_change": {}
    },
    {
      "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)\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)\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": "Address Cyclic References, Unhandled KeyErrors, and Improve Consistency in `Munch` Library",
    "description": "### Issue Overview\n\n1. **Cyclic References in `unmunchify_cycles` Method:** \n   - The `unmunchify_cycles` method previously used `seen.keys()` to check for cyclic references during recursion. This is inefficient and can be replaced with a more error-free approach using a `try-except` block to handle `KeyError` exceptions.\n\n2. **Inconsistent Equality Checks in Tests:** \n   - In various tests within `test_yaml.py`, equality checks like `assert obj.__eq__(Munch(...))` should be replaced with the more Pythonic `assert obj == Munch(...)`. This ensures consistency and better readability.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`:** \n   - The `test_fromYAML_default_munch` function needed to ensure that default values for non-existent keys in `DefaultMunch` are correctly handled and asserted against.\n\n4. **Documentation Path Handling in `test_readme.py`:**\n   - The hard-coded relative path for the `README.md` file should be replaced with a more robust path resolution to avoid directory structure issues.\n\n### Potential Impact\n\nThe above issues can lead to unexpected behavior, inefficient processing, and difficulties in understanding the code base. These can negatively affect both developers and users of the `Munch` library by introducing increased likelihood of runtime errors, making the code harder to maintain, and causing mismatches in expected library behavior.\n\n### Suggested Solution\n\n- Update the `unmunchify_cycles` method to handle cyclic references efficiently using a `try-except` block.\n- Replace equality assertions with the idiomatic form in relevant test files for better consistency.\n- Add and update tests to ensure that default values in `DefaultMunch` are appropriately handled.\n- Refactor the path resolution in tests to be more flexible and robust.\n\nThese changes will improve the overall reliability, maintainability, and clarity of the `Munch` library, thereby enhancing developer and user experience.",
    "explanation": "### Summary of the Issue\n\nThe `Munch` library had several identified issues that potentially lead to inefficiencies, inconsistencies, and difficulties for developers and users. The specific problems were:\n1. Inefficient handling of cyclic references using keys in the `unmunchify_cycles` method.\n2. Inconsistent use of equality checks in the tests.\n3. Insufficient handling and testing of default values for inexistent keys in `DefaultMunch`.\n4. Hard-coded relative paths in tests, particularly in `test_readme.py`, which could cause directory structure issues.\n\n### Overview of the Commit\n\nThe commit made several changes to address these problems:\n1. Updated `unmunchify_cycles` to handle cyclic references more efficiently by replacing key lookups with a `try-except` block.\n2. Improved the consistency of equality checks in the test files by replacing specific methods with more idiomatic Python expressions.\n3. Enhanced the default value handling in tests related to `DefaultMunch` to ensure non-existent keys are correctly managed.\n4. Refactored the path handling in tests to use a more robust path resolution technique, hence avoiding directory structure issues.\n\n### Detailed Explanation\n\n#### 1. Cyclic References in `unmunchify_cycles`\n\n**Problem:**\nThe method `unmunchify_cycles` previously checked for cyclic references by looking up keys in a dictionary using `.keys()`. This was inefficient and error-prone.\n\n**Solution:**\nThe updated method now uses a `try-except` block for handling cyclic references. By attempting to access the required key and catching a potential `KeyError`, it handles cycles more effectively and simplifies the logic.\n\n**Impact:**\nThis change makes the method more efficient by reducing unnecessary key lookups and ensures that cyclic references are detected and managed properly, thus enhancing the robustness of the method.\n\n#### 2. Inconsistent Equality Checks in Tests\n\n**Problem:**\nThe tests in files like `test_yaml.py` used the `__eq__()` method directly to check for equality, which is not Pythonic and hinders readability.\n\n**Solution:**\nThe commit replaced direct calls to `__eq__()` with the `==` operator, which internally calls the `__eq__()` method but is more idiomatic and clearer.\n\n**Impact:**\nImproving these equality checks increases code readability and consistency, allowing easier understanding and maintenance of the test code.\n\n#### 3. Default Value Handling in `DefaultMunch`\n\n**Problem:**\nTests related to `DefaultMunch` did not adequately ensure that non-existent keys were managed as expected.\n\n**Solution:**\nAdditional tests were added and existing ones were modified to assert that default values for non-existent keys are correctly applied and handled. This ensures that `DefaultMunch` behaves as expected whenever it encounters a key that doesn't exist.\n\n**Impact:**\nThese improvements make the tests more comprehensive, guaranteeing the intended behavior of `DefaultMunch`, thus providing more reliable functionality when using `DefaultMunch`.\n\n#### 4. Path Handling in Tests\n\n**Problem:**\nThe tests had hard-coded relative paths for the README file, potentially leading to issues if directory structures changed.\n\n**Solution:**\nThe commit refactored the path resolution using more robust methods by computing paths based on the current file's absolute path, making it robust against directory changes.\n\n**Impact:**\nThis ensures that the tests do not fail due to modifications in the project directory structure, enhancing the test suite's stability and reliability.\n\n### Final Solution Explanation\n\nOverall, the commit introduces a series of refinements aimed at making the `Munch` library more efficient, consistent, and reliable. By updating the handling of cyclic references, the method performance and correctness are both improved. The shift to more idiomatic equality checks in tests polish the code and enhance readability. Ensuring default values for non-existent keys are correctly handled in `DefaultMunch` improves test coverage and functionality reliability. Lastly, refactoring path handling in tests makes them more robust against directory changes, ensuring consistent and stable test performance. This comprehensive approach addresses the core issues, leading to a more maintainable and robust codebase for the `Munch` library."
  },
  "Explain": "### Summary of the Issue\n\nThe `Munch` library had several identified issues that potentially lead to inefficiencies, inconsistencies, and difficulties for developers and users. The specific problems were:\n1. Inefficient handling of cyclic references using keys in the `unmunchify_cycles` method.\n2. Inconsistent use of equality checks in the tests.\n3. Insufficient handling and testing of default values for inexistent keys in `DefaultMunch`.\n4. Hard-coded relative paths in tests, particularly in `test_readme.py`, which could cause directory structure issues.\n\n### Overview of the Commit\n\nThe commit made several changes to address these problems:\n1. Updated `unmunchify_cycles` to handle cyclic references more efficiently by replacing key lookups with a `try-except` block.\n2. Improved the consistency of equality checks in the test files by replacing specific methods with more idiomatic Python expressions.\n3. Enhanced the default value handling in tests related to `DefaultMunch` to ensure non-existent keys are correctly managed.\n4. Refactored the path handling in tests to use a more robust path resolution technique, hence avoiding directory structure issues.\n\n### Detailed Explanation\n\n#### 1. Cyclic References in `unmunchify_cycles`\n\n**Problem:**\nThe method `unmunchify_cycles` previously checked for cyclic references by looking up keys in a dictionary using `.keys()`. This was inefficient and error-prone.\n\n**Solution:**\nThe updated method now uses a `try-except` block for handling cyclic references. By attempting to access the required key and catching a potential `KeyError`, it handles cycles more effectively and simplifies the logic.\n\n**Impact:**\nThis change makes the method more efficient by reducing unnecessary key lookups and ensures that cyclic references are detected and managed properly, thus enhancing the robustness of the method.\n\n#### 2. Inconsistent Equality Checks in Tests\n\n**Problem:**\nThe tests in files like `test_yaml.py` used the `__eq__()` method directly to check for equality, which is not Pythonic and hinders readability.\n\n**Solution:**\nThe commit replaced direct calls to `__eq__()` with the `==` operator, which internally calls the `__eq__()` method but is more idiomatic and clearer.\n\n**Impact:**\nImproving these equality checks increases code readability and consistency, allowing easier understanding and maintenance of the test code.\n\n#### 3. Default Value Handling in `DefaultMunch`\n\n**Problem:**\nTests related to `DefaultMunch` did not adequately ensure that non-existent keys were managed as expected.\n\n**Solution:**\nAdditional tests were added and existing ones were modified to assert that default values for non-existent keys are correctly applied and handled. This ensures that `DefaultMunch` behaves as expected whenever it encounters a key that doesn't exist.\n\n**Impact:**\nThese improvements make the tests more comprehensive, guaranteeing the intended behavior of `DefaultMunch`, thus providing more reliable functionality when using `DefaultMunch`.\n\n#### 4. Path Handling in Tests\n\n**Problem:**\nThe tests had hard-coded relative paths for the README file, potentially leading to issues if directory structures changed.\n\n**Solution:**\nThe commit refactored the path resolution using more robust methods by computing paths based on the current file's absolute path, making it robust against directory changes.\n\n**Impact:**\nThis ensures that the tests do not fail due to modifications in the project directory structure, enhancing the test suite's stability and reliability.\n\n### Final Solution Explanation\n\nOverall, the commit introduces a series of refinements aimed at making the `Munch` library more efficient, consistent, and reliable. By updating the handling of cyclic references, the method performance and correctness are both improved. The shift to more idiomatic equality checks in tests polish the code and enhance readability. Ensuring default values for non-existent keys are correctly handled in `DefaultMunch` improves test coverage and functionality reliability. Lastly, refactoring path handling in tests makes them more robust against directory changes, ensuring consistent and stable test performance. This comprehensive approach addresses the core issues, leading to a more maintainable and robust codebase for the `Munch` library.",
  "Source": "Human",
  "Token": 1521,
  "Command": [
    "pytest tests"
  ],
  "FilteredCode": [
    {
      "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_style=True)\n22     assert dumped == '{foo: [bar, {lol: true}], hello: 42}\\n'\n23 \n24 \n25 def test_to_yaml(yaml):\n26     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n27     dumped = yaml.dump(b, default_flow_style=True)\n28     assert dumped == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n29 \n30 \n31 def test_toYAML(yaml):\n32     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n33     assert yaml.safe_dump(b, default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n34     assert b.toYAML(default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n35     assert yaml.dump(b, default_flow_style=True) == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n36     assert b.toYAML(Dumper=yaml.Dumper, default_flow_style=True) == \\\n37         '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n38 \n39 \n40 @pytest.mark.usefixtures('yaml')\n41 def test_fromYAML():\n42     # pylint: disable=unidiomatic-typecheck\n43     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n44     obj = Munch.fromYAML(yaml_str)\n45     assert type(obj) == Munch\n46     assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n47     assert obj.toYAML().__eq__(yaml_str)\n48 \n49 \n50 @pytest.mark.usefixtures('yaml')\n51 def test_fromYAML_default_munch():\n52     # pylint: disable=unidiomatic-typecheck\n53     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n54     obj = DefaultMunch.fromYAML(yaml_str)\n55     assert type(obj) == DefaultMunch\n56     assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n57     assert obj['not_exist']\n58     assert obj.not_exist\n59     assert obj.toYAML().__eq__(yaml_str)"
    },
    {
      "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    (...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.k(...truncated)"
    },
    {
      "path": "Infinidat_munch/tests/conftest.py",
      "content": "1 import pytest\n2 import munch\n3 \n4 \n5 @pytest.fixture(name='yaml')\n6 def yaml_module():\n7     try:\n8         import yaml  # pylint: disable=import-outside-toplevel\n9         return yaml\n10     except ImportError:\n11         pass\n12     pytest.skip(\"Module 'PyYAML' is required\")\n13 \n14 \n15 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n16                         munch.RecursiveMunch])\n17 def munch_obj(request):\n18     cls = request.param\n19     args = tuple()\n20     if cls == munch.DefaultFactoryMunch:\n21         args = args + (lambda: None,)\n22     return cls(*args, hello=\"world\", number=5)"
    },
    {
      "path": "Infinidat_munch/tests/test_readme.py",
      "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _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"
    }
  ],
  "TokenAll": 13453,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Potential unhandled ImportError for 'yaml' module",
      "description": "In the file 'Infinidat_munch/tests/conftest.py', if the 'yaml' module import fails, the test is skipped using 'pytest.skip' without ensuring that other tests dependent on the 'yaml' fixture are also skipped. This could lead to test failures if other tests rely on the 'yaml' fixture.",
      "explanation": "### Issue Summary\nThe issue revolves around the potential unhandled `ImportError` for the `yaml` module in the `Infinidat_munch/tests/conftest.py` file. If the `yaml` module fails to be imported, the current setup skips one specific test using `pytest.skip`. However, other tests that rely on the `yaml` fixture also need to be skipped because they will encounter failures due to the missing dependency. The current mechanism doesn't ensure these dependent tests are also skipped, leading to possible test failures.\n\n### Analysis of Code and Commit\n#### Relevant Code Snippets:\n\n1. **conftest.py** - Fixture Definition:\n    ```python\n    1 import pytest\n    2 import munch\n    3 \n    4 \n    5 @pytest.fixture(name='yaml')\n    6 def yaml_module():\n    7     try:\n    8         import yaml  # pylint: disable=import-outside-toplevel\n    9         return yaml\n    10     except ImportError:\n    11         pass\n    12     pytest.skip(\"Module 'PyYAML' is required\")\n    ```\n\n2. **test_yaml.py** - Tests Depending on `yaml` Fixture:\n    ```python\n    1 import pytest\n    2 from munch import Munch, DefaultMunch\n    3 \n    4 \n    5 def test_from_yaml(yaml):\n    6     ...\n    7 \n    8 def test_to_yaml_safe(yaml):\n    9     ...\n    10 \n    11 def test_to_yaml(yaml):\n    12     ...\n    13 \n    14 def test_toYAML(yaml):\n    15     ...\n    16 \n    17 @pytest.mark.usefixtures('yaml')\n    18 def test_fromYAML():\n    19     ...\n    20 \n    21 @pytest.mark.usefixtures('yaml')\n    22 def test_fromYAML_default_munch():\n    23     ...\n    ```\n\n3. **test_readme.py** - Test Depending on `yaml` Fixture:\n    ```python\n    1 import doctest\n    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 @pytest.mark.usefixtures(\"yaml\")\n    9 def test_readme():\n    10    ...\n    ```\n\n#### Commit Changes (Hypothetical Since Not Given in Input):\nSuppose a commit was made as follows:\n1. Updated the `conftest.py` to ensure `pytest.skip` is raised promptly when `yaml` import fails:\n\n    ```python\n    1 import pytest\n    2 import munch\n    3 \n    4 @pytest.fixture(name='yaml')\n    5 def yaml_module():\n    6     try:\n    7         import yaml  # pylint: disable=import-outside-toplevel\n    8         return yaml\n    9     except ImportError:\n    10         pytest.skip(\"Module 'PyYAML' is required\")\n\n    ```\n\n2. Removed unnecessary pass statement and directly raise the skip condition on import failure.\n\n### Detailed Explanation\n#### Cause of the Issue:\n1. **Initial Setup**: The original setup employs a fixture named `yaml` to import the `yaml` module. If this import fails, the fixture uses `pytest.skip` to skip the test.\n2. **Unintended Behavior**: However, the tests in various test files (`test_yaml.py` and `test_readme.py`) that depend on this `yaml` fixture through direct function argument or through `pytest.mark.usefixtures('yaml')` do not get automatically skipped when the import fails.\n\n#### Content of the Commit:\n1. **Refactor and Cleanup in `conftest.py`**:\n    - The fixture `yaml_module` has been streamlined to immediately skip the test if the `yaml` module cannot be imported. The `pytest.skip` is now raised directly within the `except ImportError` block ensuring that any test relying on this fixture will be skipped.\n\n    ```python\n    @pytest.fixture(name='yaml')\n    def yaml_module():\n        try:\n            import yaml  # pylint: disable=import-outside-toplevel\n            return yaml\n        except ImportError:\n            pytest.skip(\"Module 'PyYAML' is required\")\n    ```\n\n### How the Commit Solves the Issue:\n1. **Unified Skipping Mechanism**: By raising `pytest.skip` directly within the fixture when `yaml` import fails, any test that depends on the `yaml` fixture (by either passing it directly like `test_from_yaml` or through `pytest.mark.usefixtures` like `test_fromYAML_default_munch`) will be skipped automatically.\n2."
    },
    "issue_message": {
      "title": "Assertion for non-existent key fails",
      "description": "In the test function 'test_fromYAML_default_munch' located in file 'tests/test_yaml.py', line 57 contains an assertion 'assert obj['not_exist']' which fails because the key 'not_exist' does not exist in the DefaultMunch object, leading to an empty dictionary assertion. Consider revising the test to check for the presence or default value of non-existent keys in a DefaultMunch object.",
      "explanation": "### Issue Summary\n\nThe reported issue is that the test `test_fromYAML_default_munch` fails because it contains an assertion for a non-existent key (`'not_exist'`) in a `DefaultMunch` object. Specifically, the assertion `assert obj['not_exist']` fails, as the key does not exist in the `DefaultMunch` object, leading to an empty dictionary assertion. The goal is to revise the test so that it correctly checks for the presence or the default value of non-existent keys in a `DefaultMunch` object.\n\n### Commit Analysis\n\nThe commit likely modified the failing test in the `tests/test_yaml.py` file to properly handle assertions for non-existent keys in a `DefaultMunch` object. Here's the relevant part of the test function, including the erroneous assertion:\n\n```python\n51  def test_fromYAML_default_munch():\n52      # pylint: disable=unidiomatic-typecheck\n53      yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n54      obj = DefaultMunch.fromYAML(yaml_str)\n55      assert type(obj) == DefaultMunch\n56      assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n57      assert obj['not_exist']\n58      assert obj.not_exist\n59      assert obj.toYAML().__eq__(yaml_str)\n```\n\nThe key part of the erroneous assertion is line 57:\n```python\n57      assert obj['not_exist']\n```\nIn `DefaultMunch`, when a key does not exist, it should return a default value (typically an empty instance of a specified type or a default value defined during the initialization of the `DefaultMunch`). However, this assertion blindly checks whether `obj['not_exist']` is truthy, without specifying what it expects.\n\n### Solution Explanation\n\n#### Revised Code\n\nThe revised code to fix the issue could look like this:\n```python\n57      assert 'not_exist' in obj  # Check if the key exists\n58      assert obj['not_exist'] == {}  # Ensure the key returns the expected default value\n59      assert obj.not_exist == {}\n```\n\n#### Explanation\n\n1. **Check for Key Presence:**\n   ```python\n   assert 'not_exist' in obj\n   ```\n   This assertion verifies that the key `'not_exist'` is present in the `DefaultMunch` object. In a `DefaultMunch`, accessing a non-existent key should ideally initialize it with a default value (such as an empty dictionary). This check ensures that requesting the key from `DefaultMunch` does not cause errors.\n\n2. **Check for Default Value:**\n   ```python\n   assert obj['not_exist'] == {}\n   ```\n   Instead of simply checking if `obj['not_exist']` is truthy, this assertion explicitly ensures that the value returned for the non-existent key is an empty dictionary, which is the expected default for a `DefaultMunch`.\n\n3. **Check Attribute Access:**\n   ```python\n   assert obj.not_exist == {}\n   ```\n   The same check as above but using attribute-style access, which `DefaultMunch` also supports. Ensures that accessing the key via dot notation returns the expected default value.\n\n### Addressing the Issue\n\nThe updated assertions ensure:\n- The test does not fail due to the existence check of the key.\n- It validates the behavior of `DefaultMunch` in returning a default value for non-existent keys.\n\nBy making these changes, the test accurately reflects the expected behavior of the `DefaultMunch` class when handling non-existent keys, ensuring it returns default values instead of causing assertion failures. This approach promotes better coding practices and yields more meaningful test outcomes."
    },
    "issue_ground": {
      "title": "Address Cyclic References, Unhandled KeyErrors, and Improve Consistency in `Munch` Library",
      "description": "### Issue Overview\n\n1. **Cyclic References in `unmunchify_cycles` Method:** \n   - The `unmunchify_cycles` method previously used `seen.keys()` to check for cyclic references during recursion. This is inefficient and can be replaced with a more error-free approach using a `try-except` block to handle `KeyError` exceptions.\n\n2. **Inconsistent Equality Checks in Tests:** \n   - In various tests within `test_yaml.py`, equality checks like `assert obj.__eq__(Munch(...))` should be replaced with the more Pythonic `assert obj == Munch(...)`. This ensures consistency and better readability.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`:** \n   - The `test_fromYAML_default_munch` function needed to ensure that default values for non-existent keys in `DefaultMunch` are correctly handled and asserted against.\n\n4. **Documentation Path Handling in `test_readme.py`:**\n   - The hard-coded relative path for the `README.md` file should be replaced with a more robust path resolution to avoid directory structure issues.\n\n### Potential Impact\n\nThe above issues can lead to unexpected behavior, inefficient processing, and difficulties in understanding the code base. These can negatively affect both developers and users of the `Munch` library by introducing increased likelihood of runtime errors, making the code harder to maintain, and causing mismatches in expected library behavior.\n\n### Suggested Solution\n\n- Update the `unmunchify_cycles` method to handle cyclic references efficiently using a `try-except` block.\n- Replace equality assertions with the idiomatic form in relevant test files for better consistency.\n- Add and update tests to ensure that default values in `DefaultMunch` are appropriately handled.\n- Refactor the path resolution in tests to be more flexible and robust.\n\nThese changes will improve the overall reliability, maintainability, and clarity of the `Munch` library, thereby enhancing developer and user experience.",
      "explanation": "### Issue Summary\n\nThe issue at hand is related to multiple areas of the `Munch` library:\n\n1. **Cyclic References in `unmunchify_cycles` Method**:\n   - The `unmunchify_cycles` method has an inefficient implementation for handling cyclic references.\n   \n2. **Inconsistent Equality Checks in Tests**:\n   - Equality assertions in the test file `test_yaml.py` use an unidiomatic form (`__eq__` method) instead of the more Pythonic `==`.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`**:\n   - The `test_fromYAML_default_munch` test function does not properly handle default values for non-existent keys in `DefaultMunch`.\n\n4. **Documentation Path Handling in `test_readme.py`**:\n   - The path to the `README.md` file is hard-coded, which can lead to issues with different directory structures.\n\n### Commit Content Analysis\n\nLet's break down the changes needed to address each of these issues:\n\n1. **Cyclic References in `unmunchify_cycles` Method**:\n   - Update `unmunchify_cycles` to use a `try-except` block to efficiently handle cyclic references.\n   \n2. **Inconsistent Equality Checks in Tests**:\n   - Replace `assert obj.__eq__(Munch(...))` with `assert obj == Munch(...)` for better consistency and readability in `test_yaml.py`.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`**:\n   - Ensure that default values are correctly asserted in the `test_fromYAML_default_munch` function.\n\n4. **Documentation Path Handling in `test_readme.py`**:\n   - Use a more robust path resolution method to handle different directory structures.\n\n### Detailed Explanation of the Solution\n\n#### Solution for Cyclic References Handling:\n\n1. **Current Inefficient Handling**:\n   ```python\n   def unmunchify_cycles(obj, seen=None):\n       if seen is None:\n           seen = {}\n       if id(obj) in seen.keys():\n           raise KeyError('Cyclic reference detected')\n       seen[id(obj)] = obj\n       ...\n   ```\n\n2. **Proposed Efficient Handling**:\n   ```python\n   def unmunchify_cycles(obj, seen=None):\n       if seen is None:\n           seen = {}\n       try:\n           seen[id(obj)]\n           raise KeyError('Cyclic reference detected')\n       except KeyError:\n           seen[id(obj)] = obj\n       ...\n   ```\n\n   * By using `try-except`, we avoid checking `seen.keys()` which is inefficient. Instead, we attempt to find the key and handle the exception if it's not there.\n\n#### Solution for Inconsistent Equality Checks in Tests:\n\n1. **Current Inconsistent Checks**:\n   ```python\n   assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n   ```\n\n2. **Proposed Consistent Checks**:\n   ```python\n   assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n   ```\n\n   * This change improves readability and consistency across tests by using idiomatic Python equality checks.\n\n#### Solution for Default Value Handling in `DefaultMunch.fromYAML`:\n\n1. **Current Handling Causing Failure**:\n   ```python\n   assert obj['not_exist']\n   ```\n\n2. **Proposed Handling**:\n   ```python\n   assert 'not_exist' in obj\n   assert obj['not_exist'] is not None\n   ```\n\n   * Ensures that default values are properly set and correctly asserted.\n\n#### Solution for Documentation Path Handling:\n\n1. **Current Path Handling**:\n   ```python\n   _README_PATH = os.path.join(os.path.dirname(__file__), '..', 'README.md')\n   ```\n\n2. **Proposed Path Handling**:\n   ```python\n   _README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n   ```\n\n   * Using `os.path.abspath` ensures the path resolution is more robust to different directory structures.\n\n### How the Commit Solves the Issue\n\n1. **Cyclic References**: The new try-except mechanism in `unmunchify_cycles` efficiently checks for and handles cyclic references without the overhead of `seen.keys()` lookups.\n   \n2. **Equality Checks**: Using `==` instead of `__eq__` provides a cleaner, more Pythonic way to check object equality, making the tests more maintainable and understandable.\n\n3. **Default Value Handling**: Ensuring the existence and correctness of default values in the `test_fromYAML_default_munch` function addresses where the test was previously failing.\n\n4. **Documentation Path Handling**: The robust path resolution in `test_readme.py` eliminates potential path issues, making the test setup more reliable.\n\nTogether, these changes improve the reliability, efficiency, and readability of the `Munch`"
    },
    "issue_ground_truth": {
      "title": "Address Cyclic References, Unhandled KeyErrors, and Improve Consistency in `Munch` Library",
      "description": "### Issue Overview\n\n1. **Cyclic References in `unmunchify_cycles` Method:** \n   - The `unmunchify_cycles` method previously used `seen.keys()` to check for cyclic references during recursion. This is inefficient and can be replaced with a more error-free approach using a `try-except` block to handle `KeyError` exceptions.\n\n2. **Inconsistent Equality Checks in Tests:** \n   - In various tests within `test_yaml.py`, equality checks like `assert obj.__eq__(Munch(...))` should be replaced with the more Pythonic `assert obj == Munch(...)`. This ensures consistency and better readability.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`:** \n   - The `test_fromYAML_default_munch` function needed to ensure that default values for non-existent keys in `DefaultMunch` are correctly handled and asserted against.\n\n4. **Documentation Path Handling in `test_readme.py`:**\n   - The hard-coded relative path for the `README.md` file should be replaced with a more robust path resolution to avoid directory structure issues.\n\n### Potential Impact\n\nThe above issues can lead to unexpected behavior, inefficient processing, and difficulties in understanding the code base. These can negatively affect both developers and users of the `Munch` library by introducing increased likelihood of runtime errors, making the code harder to maintain, and causing mismatches in expected library behavior.\n\n### Suggested Solution\n\n- Update the `unmunchify_cycles` method to handle cyclic references efficiently using a `try-except` block.\n- Replace equality assertions with the idiomatic form in relevant test files for better consistency.\n- Add and update tests to ensure that default values in `DefaultMunch` are appropriately handled.\n- Refactor the path resolution in tests to be more flexible and robust.\n\nThese changes will improve the overall reliability, maintainability, and clarity of the `Munch` library, thereby enhancing developer and user experience.",
      "explanation": "### Summary of the Issue\n\nThe `Munch` library had several identified issues that potentially lead to inefficiencies, inconsistencies, and difficulties for developers and users. The specific problems were:\n1. Inefficient handling of cyclic references using keys in the `unmunchify_cycles` method.\n2. Inconsistent use of equality checks in the tests.\n3. Insufficient handling and testing of default values for inexistent keys in `DefaultMunch`.\n4. Hard-coded relative paths in tests, particularly in `test_readme.py`, which could cause directory structure issues.\n\n### Overview of the Commit\n\nThe commit made several changes to address these problems:\n1. Updated `unmunchify_cycles` to handle cyclic references more efficiently by replacing key lookups with a `try-except` block.\n2. Improved the consistency of equality checks in the test files by replacing specific methods with more idiomatic Python expressions.\n3. Enhanced the default value handling in tests related to `DefaultMunch` to ensure non-existent keys are correctly managed.\n4. Refactored the path handling in tests to use a more robust path resolution technique, hence avoiding directory structure issues.\n\n### Detailed Explanation\n\n#### 1. Cyclic References in `unmunchify_cycles`\n\n**Problem:**\nThe method `unmunchify_cycles` previously checked for cyclic references by looking up keys in a dictionary using `.keys()`. This was inefficient and error-prone.\n\n**Solution:**\nThe updated method now uses a `try-except` block for handling cyclic references. By attempting to access the required key and catching a potential `KeyError`, it handles cycles more effectively and simplifies the logic.\n\n**Impact:**\nThis change makes the method more efficient by reducing unnecessary key lookups and ensures that cyclic references are detected and managed properly, thus enhancing the robustness of the method.\n\n#### 2. Inconsistent Equality Checks in Tests\n\n**Problem:**\nThe tests in files like `test_yaml.py` used the `__eq__()` method directly to check for equality, which is not Pythonic and hinders readability.\n\n**Solution:**\nThe commit replaced direct calls to `__eq__()` with the `==` operator, which internally calls the `__eq__()` method but is more idiomatic and clearer.\n\n**Impact:**\nImproving these equality checks increases code readability and consistency, allowing easier understanding and maintenance of the test code.\n\n#### 3. Default Value Handling in `DefaultMunch`\n\n**Problem:**\nTests related to `DefaultMunch` did not adequately ensure that non-existent keys were managed as expected.\n\n**Solution:**\nAdditional tests were added and existing ones were modified to assert that default values for non-existent keys are correctly applied and handled. This ensures that `DefaultMunch` behaves as expected whenever it encounters a key that doesn't exist.\n\n**Impact:**\nThese improvements make the tests more comprehensive, guaranteeing the intended behavior of `DefaultMunch`, thus providing more reliable functionality when using `DefaultMunch`.\n\n#### 4. Path Handling in Tests\n\n**Problem:**\nThe tests had hard-coded relative paths for the README file, potentially leading to issues if directory structures changed.\n\n**Solution:**\nThe commit refactored the path resolution using more robust methods by computing paths based on the current file's absolute path, making it robust against directory changes.\n\n**Impact:**\nThis ensures that the tests do not fail due to modifications in the project directory structure, enhancing the test suite's stability and reliability.\n\n### Final Solution Explanation\n\nOverall, the commit introduces a series of refinements aimed at making the `Munch` library more efficient, consistent, and reliable. By updating the handling of cyclic references, the method performance and correctness are both improved. The shift to more idiomatic equality checks in tests polish the code and enhance readability. Ensuring default values for non-existent keys are correctly handled in `DefaultMunch` improves test coverage and functionality reliability. Lastly, refactoring path handling in tests makes them more robust against directory changes, ensuring consistent and stable test performance. This comprehensive approach addresses the core issues, leading to a more maintainable and robust codebase for the `Munch` library."
    },
    "location_origin": [
      {
        "file": "Infinidat_munch/tests/conftest.py",
        "function": {
          "5": "yaml_module"
        },
        "content_all": {
          "4": " \n",
          "5": "@pytest.fixture(name='yaml')\n",
          "6": "def yaml_module():\n",
          "7": "    try:\n",
          "8": "        import yaml  # pylint: disable=import-outside-toplevel\n",
          "9": "        return yaml\n",
          "10": "    except ImportError:\n",
          "11": "        pass\n",
          "12": "    pytest.skip(\"Module 'PyYAML' is required\")\n",
          "13": " \n"
        },
        "content_change": {
          "11": "        pass\n",
          "12": "    pytest.skip(\"Module 'PyYAML' is required\")\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "5": "test_from_yaml"
        },
        "content_all": {
          "4": " \n",
          "5": "def test_from_yaml(yaml):\n",
          "6": "    data = yaml.load('''\n",
          "7": "    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n",
          "8": "    Block style: !munch\n",
          "9": "      Clark : Evans\n",
          "10": "      Brian : Ingerson\n",
          "11": "      Oren  : Ben-Kiki\n",
          "12": "    ''', Loader=yaml.FullLoader)\n",
          "13": "    assert data == {\n",
          "14": "        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "15": "        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "16": "    }\n",
          "17": " \n",
          "18": " \n"
        },
        "content_change": {
          "4": " \n",
          "5": "def test_from_yaml(yaml):\n",
          "6": "    data = yaml.load('''\n",
          "7": "    Flow style: !munch.Munch { Clark: Evans, Brian: Ingerson, Oren: Ben-Kiki }\n",
          "8": "    Block style: !munch\n",
          "9": "      Clark : Evans\n",
          "10": "      Brian : Ingerson\n",
          "11": "      Oren  : Ben-Kiki\n",
          "12": "    ''', Loader=yaml.FullLoader)\n",
          "13": "    assert data == {\n",
          "14": "        'Flow style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "15": "        'Block style': Munch(Brian='Ingerson', Clark='Evans', Oren='Ben-Kiki'),\n",
          "16": "    }\n",
          "17": " \n",
          "18": " \n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": {
          "9": "test_readme"
        },
        "content_all": {
          "8": " \n",
          "9": "@pytest.mark.usefixtures(\"yaml\")\n",
          "10": "def test_readme():\n",
          "11": "    globs = {\n",
          "12": "        'print_function': print\n",
          "13": "    }\n",
          "14": "    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n",
          "15": "    assert not result.failed\n"
        },
        "content_change": {
          "8": " \n",
          "9": "@pytest.mark.usefixtures(\"yaml\")\n",
          "10": "def test_readme():\n",
          "11": "    globs = {\n",
          "12": "        'print_function': print\n",
          "13": "    }\n",
          "14": "    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n",
          "15": "    assert not result.failed\n"
        }
      }
    ],
    "location_message": [
      {
        "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)\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",
          "59": "        assert obj.toYAML().__eq__(yaml_str)\n"
        },
        "content_change": {
          "57": "        assert 'not_exist' in obj\n",
          "58": "        assert obj['not_exist'] == {}\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "Infinidat_munch/munch/__init__.py",
        "function": {
          "511": "unmunchify_cycles"
        },
        "content_all": {
          "510": "\n",
          "511": "    def unmunchify_cycles(obj, seen=None):\n",
          "512": "        if seen is None:\n",
          "513": "            seen = {}\n",
          "514": "        if id(obj) in seen.keys():\n",
          "515": "            raise KeyError('Cyclic reference detected')\n",
          "516": "        seen[id(obj)] = obj\n",
          "517": "        # Continue processing...\n"
        },
        "content_change": {
          "514": "        try:\n",
          "515": "            seen[id(obj)]\n",
          "516": "            raise KeyError('Cyclic reference detected')\n",
          "517": "        except KeyError:\n",
          "518": "            seen[id(obj)] = obj\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "40": "test_fromYAML"
        },
        "content_all": {
          "39": "\n",
          "40": "@pytest.mark.usefixtures('yaml')\n",
          "41": "def test_fromYAML():\n",
          "42": "    # pylint: disable=unidiomatic-typecheck\n",
          "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",
          "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
          "48": "\n"
        },
        "content_change": {
          "46": "    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n",
          "47": "    assert obj.toYAML() == yaml_str\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "50": "test_fromYAML_default_munch"
        },
        "content_all": {
          "49": "\n",
          "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)\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",
          "59": "    assert obj.toYAML() == yaml_str\n"
        },
        "content_change": {
          "57": "    assert 'not_exist' in obj\n",
          "58": "    assert obj['not_exist'] is not None\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": {
          "9": "test_readme"
        },
        "content_all": {
          "8": "\n",
          "9": "@pytest.mark.usefixtures(\"yaml\")\n",
          "10": "def test_readme():\n",
          "11": "    globs = {\n",
          "12": "        'print_function': print\n",
          "13": "    }\n",
          "14": "    result = doctest.testfile(_README_PATH, module_relative=False, globs=globs)\n",
          "15": "    assert not result.failed\n"
        },
        "content_change": {
          "5": "README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n",
          "6": "assert os.path.exists(_README_PATH)\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "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": "        try:\n",
          "514": "            return seen[id(obj)]\n",
          "515": "        except KeyError:\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "46": "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",
          "47": "    assert obj.toYAML().__eq__(yaml_str)\n",
          "48": "\n",
          "49": "\n"
        },
        "content_change": {
          "46": "    assert obj == Munch(foo=Munch(bar=[1, 2], hello='world'))\n",
          "47": "    assert obj.toYAML() == yaml_str\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_readme.py",
        "function": {
          "5": "test_readme"
        },
        "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",
          "9": "@pytest.mark.usefixtures(\"yaml\")\n",
          "10": "def test_readme():\n",
          "11": "    globs = {\n",
          "12": "        'print_function': print\n",
          "13": "    }\n"
        },
        "content_change": {
          "5": "_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n"
        }
      },
      {
        "file": "Infinidat_munch/tests/test_yaml.py",
        "function": {
          "56": "test_fromYAML_default_munch"
        },
        "content_all": {
          "53": "    yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n",
          "54": "    obj = DefaultMunch.fromYAML(yaml_str)\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",
          "59": "    assert obj.toYAML().__eq__(yaml_str)"
        },
        "content_change": {
          "57": "    assert 'not_exist' not in obj\n",
          "58": "    assert obj.not_exist is None"
        }
      }
    ],
    "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"
        },
        "content_change": {}
      },
      {
        "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)\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)\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/conftest.py\n+++ b/Infinidat_munch/tests/conftest.py\n@@ -8,7 +8,7 @@ def yaml_module():\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 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n",
    "patch_im": "\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -56,4 +56,5 @@ def test_fromYAML_default_munch():\n      assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n-     assert obj['not_exist']\n+     assert 'not_exist' in obj\n+     assert obj['not_exist'] == {}\n      assert obj.not_exist\n      assert obj.toYAML().__eq__(yaml_str)\n",
    "patch_il": "\n--- a/Infinidat_munch/tests/conftest.py\n+++ b/Infinidat_munch/tests/conftest.py\n@@ -10,8 +10,7 @@\n         import yaml  # pylint: disable=import-outside-toplevel\n         return yaml\n     except ImportError:\n-        pass\n-    pytest.skip(\"Module 'PyYAML' is required\")\n+        pytest.skip(\"Module 'PyYAML' is required\")\n \n @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n                         munch.RecursiveMunch])\n",
    "patch_iml": "\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -54,6 +54,8 @@\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 'not_exist' in obj  # Check if the key exists\n+    assert obj['not_exist'] == {}  # Ensure the key returns the expected default value\n+    assert obj.not_exist == {}  # Check attribute access\n     assert obj.toYAML().__eq__(yaml_str)\n",
    "patch_ground": "\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -45,8 +45,8 @@ def test_fromYAML():\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@@ -56,5 +56,6 @@ def test_fromYAML_default_munch():\n     assert type(obj) == DefaultMunch\n-    assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n-    assert obj['not_exist']\n+    assert obj == DefaultMunch(foo=DefaultMunch(bar=[1, 2], hello='world'))\n+    assert 'not_exist' in obj  # Ensure key is in the object\n+    assert obj['not_exist'] is not None  # Ensure default value handling\n     assert obj.not_exist\n@@ -58,5 +59,5 @@ def test_fromYAML_default_munch():\n     assert obj.toYAML().__eq__(yaml_str)\n",
    "patch_ground_location": "\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, seen=None):\n         if seen is None:\n-            seen = {}\n-        if id(obj) in seen.keys(): \n-            raise KeyError('Cyclic reference detected')\n-        seen[id(obj)] = obj\n+            seen = {}\n+        try:\n+            seen[id(obj)]\n+            raise KeyError('Cyclic reference detected')\n+        except KeyError:\n+            seen[id(obj)] = obj\n+        # Continue processing...\n",
    "patch_ground_exp": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -510,1 +510,1 @@\n \n def unmunchify_cycles(obj):\n-    if id(obj) in seen.keys():\n-        return seen[id(obj)]\n+    try:\n+        return seen[id(obj)]\n+    except KeyError:\n     seen[id(obj)] = partial = pre_unmunchify(obj)\n--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -46,1 +46,1 @@\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@@ -57,2 +57,2 @@\n-    assert obj['not_exist']\n-    assert obj.not_exist\n+    assert 'not_exist' not in obj\n+    assert obj.not_exist is None\n--- a/Infinidat_munch/tests/test_readme.py\n+++ b/Infinidat_munch/tests/test_readme.py\n@@ -5,1 +5,1 @@\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",
    "patch_ground_all": "\n--- a/Infinidat_munch/munch/__init__.py\n+++ b/Infinidat_munch/munch/__init__.py\n@@ -509,7 +509,8 @@\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-        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--- a/Infinidat_munch/tests/test_yaml.py\n+++ b/Infinidat_munch/tests/test_yaml.py\n@@ -45,8 +45,8 @@\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 @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)\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.toYAML() == yaml_str\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+_README_PATH = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'README.md'))\n assert os.path.exists(_README_PATH)\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--- 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)\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_fromYAML_default_munch _____________________________________________________________________________________\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)\n        assert type(obj) == DefaultMunch\n        assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n>       assert obj['not_exist']\nE       assert {}\n\ntests/test_yaml.py:57: AssertionError\n======================================================================================= short test summary info =======================================================================================\nFAILED tests/test_yaml.py::test_fromYAML_default_munch - assert {}\n==================================================================================== 1 failed, 108 passed in 0.08s ====================================================================================",
    "CodeBase": [
      {
        "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_style=True)\n22     assert dumped == '{foo: [bar, {lol: true}], hello: 42}\\n'\n23 \n24 \n25 def test_to_yaml(yaml):\n26     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n27     dumped = yaml.dump(b, default_flow_style=True)\n28     assert dumped == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n29 \n30 \n31 def test_toYAML(yaml):\n32     b = Munch(foo=['bar', Munch(lol=True)], hello=42)\n33     assert yaml.safe_dump(b, default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n34     assert b.toYAML(default_flow_style=True) == '{foo: [bar, {lol: true}], hello: 42}\\n'\n35     assert yaml.dump(b, default_flow_style=True) == '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n36     assert b.toYAML(Dumper=yaml.Dumper, default_flow_style=True) == \\\n37         '!munch.Munch {foo: [bar, !munch.Munch {lol: true}], hello: 42}\\n'\n38 \n39 \n40 @pytest.mark.usefixtures('yaml')\n41 def test_fromYAML():\n42     # pylint: disable=unidiomatic-typecheck\n43     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n44     obj = Munch.fromYAML(yaml_str)\n45     assert type(obj) == Munch\n46     assert obj.__eq__(Munch(foo=Munch(bar=[1, 2], hello='world')))\n47     assert obj.toYAML().__eq__(yaml_str)\n48 \n49 \n50 @pytest.mark.usefixtures('yaml')\n51 def test_fromYAML_default_munch():\n52     # pylint: disable=unidiomatic-typecheck\n53     yaml_str = 'foo:\\n    bar:\\n    - 1\\n    - 2\\n    hello: world\\n'\n54     obj = DefaultMunch.fromYAML(yaml_str)\n55     assert type(obj) == DefaultMunch\n56     assert obj == DefaultMunch(foo=Munch(bar=[1, 2], hello='world'))\n57     assert obj['not_exist']\n58     assert obj.not_exist\n59     assert obj.toYAML().__eq__(yaml_str)"
      },
      {
        "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    (...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.k(...truncated)"
      },
      {
        "path": "Infinidat_munch/tests/conftest.py",
        "content": "1 import pytest\n2 import munch\n3 \n4 \n5 @pytest.fixture(name='yaml')\n6 def yaml_module():\n7     try:\n8         import yaml  # pylint: disable=import-outside-toplevel\n9         return yaml\n10     except ImportError:\n11         pass\n12     pytest.skip(\"Module 'PyYAML' is required\")\n13 \n14 \n15 @pytest.fixture(params=[munch.Munch, munch.AutoMunch, munch.DefaultMunch, munch.DefaultFactoryMunch,\n16                         munch.RecursiveMunch])\n17 def munch_obj(request):\n18     cls = request.param\n19     args = tuple()\n20     if cls == munch.DefaultFactoryMunch:\n21         args = args + (lambda: None,)\n22     return cls(*args, hello=\"world\", number=5)"
      },
      {
        "path": "Infinidat_munch/tests/test_readme.py",
        "content": "1 import doctest\n2 import os\n3 import pytest\n4 \n5 _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"
      }
    ],
    "CommitSHA": "d6a54cac0e0d913e27f7cf59064d0e4c13f82e27"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 6,
      "Description": 7,
      "Reproducibility": 7,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 7,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Address Cyclic References, Unhandled KeyErrors, and Improve Consistency in `Munch` Library",
      "description": "### Issue Overview\n\n1. **Cyclic References in `unmunchify_cycles` Method:** \n   - The `unmunchify_cycles` method previously used `seen.keys()` to check for cyclic references during recursion. This is inefficient and can be replaced with a more error-free approach using a `try-except` block to handle `KeyError` exceptions.\n\n2. **Inconsistent Equality Checks in Tests:** \n   - In various tests within `test_yaml.py`, equality checks like `assert obj.__eq__(Munch(...))` should be replaced with the more Pythonic `assert obj == Munch(...)`. This ensures consistency and better readability.\n\n3. **Default Value Handling in `DefaultMunch.fromYAML`:** \n   - The `test_fromYAML_default_munch` function needed to ensure that default values for non-existent keys in `DefaultMunch` are correctly handled and asserted against.\n\n4. **Documentation Path Handling in `test_readme.py`:**\n   - The hard-coded relative path for the `README.md` file should be replaced with a more robust path resolution to avoid directory structure issues.\n\n### Potential Impact\n\nThe above issues can lead to unexpected behavior, inefficient processing, and difficulties in understanding the code base. These can negatively affect both developers and users of the `Munch` library by introducing increased likelihood of runtime errors, making the code harder to maintain, and causing mismatches in expected library behavior.\n\n### Suggested Solution\n\n- Update the `unmunchify_cycles` method to handle cyclic references efficiently using a `try-except` block.\n- Replace equality assertions with the idiomatic form in relevant test files for better consistency.\n- Add and update tests to ensure that default values in `DefaultMunch` are appropriately handled.\n- Refactor the path resolution in tests to be more flexible and robust.\n\nThese changes will improve the overall reliability, maintainability, and clarity of the `Munch` library, thereby enhancing developer and user experience.",
      "explanation": "### Summary of the Issue\n\nThe `Munch` library had several identified issues that potentially lead to inefficiencies, inconsistencies, and difficulties for developers and users. The specific problems were:\n1. Inefficient handling of cyclic references using keys in the `unmunchify_cycles` method.\n2. Inconsistent use of equality checks in the tests.\n3. Insufficient handling and testing of default values for inexistent keys in `DefaultMunch`.\n4. Hard-coded relative paths in tests, particularly in `test_readme.py`, which could cause directory structure issues.\n\n### Overview of the Commit\n\nThe commit made several changes to address these problems:\n1. Updated `unmunchify_cycles` to handle cyclic references more efficiently by replacing key lookups with a `try-except` block.\n2. Improved the consistency of equality checks in the test files by replacing specific methods with more idiomatic Python expressions.\n3. Enhanced the default value handling in tests related to `DefaultMunch` to ensure non-existent keys are correctly managed.\n4. Refactored the path handling in tests to use a more robust path resolution technique, hence avoiding directory structure issues.\n\n### Detailed Explanation\n\n#### 1. Cyclic References in `unmunchify_cycles`\n\n**Problem:**\nThe method `unmunchify_cycles` previously checked for cyclic references by looking up keys in a dictionary using `.keys()`. This was inefficient and error-prone.\n\n**Solution:**\nThe updated method now uses a `try-except` block for handling cyclic references. By attempting to access the required key and catching a potential `KeyError`, it handles cycles more effectively and simplifies the logic.\n\n**Impact:**\nThis change makes the method more efficient by reducing unnecessary key lookups and ensures that cyclic references are detected and managed properly, thus enhancing the robustness of the method.\n\n#### 2. Inconsistent Equality Checks in Tests\n\n**Problem:**\nThe tests in files like `test_yaml.py` used the `__eq__()` method directly to check for equality, which is not Pythonic and hinders readability.\n\n**Solution:**\nThe commit replaced direct calls to `__eq__()` with the `==` operator, which internally calls the `__eq__()` method but is more idiomatic and clearer.\n\n**Impact:**\nImproving these equality checks increases code readability and consistency, allowing easier understanding and maintenance of the test code.\n\n#### 3. Default Value Handling in `DefaultMunch`\n\n**Problem:**\nTests related to `DefaultMunch` did not adequately ensure that non-existent keys were managed as expected.\n\n**Solution:**\nAdditional tests were added and existing ones were modified to assert that default values for non-existent keys are correctly applied and handled. This ensures that `DefaultMunch` behaves as expected whenever it encounters a key that doesn't exist.\n\n**Impact:**\nThese improvements make the tests more comprehensive, guaranteeing the intended behavior of `DefaultMunch`, thus providing more reliable functionality when using `DefaultMunch`.\n\n#### 4. Path Handling in Tests\n\n**Problem:**\nThe tests had hard-coded relative paths for the README file, potentially leading to issues if directory structures changed.\n\n**Solution:**\nThe commit refactored the path resolution using more robust methods by computing paths based on the current file's absolute path, making it robust against directory changes.\n\n**Impact:**\nThis ensures that the tests do not fail due to modifications in the project directory structure, enhancing the test suite's stability and reliability.\n\n### Final Solution Explanation\n\nOverall, the commit introduces a series of refinements aimed at making the `Munch` library more efficient, consistent, and reliable. By updating the handling of cyclic references, the method performance and correctness are both improved. The shift to more idiomatic equality checks in tests polish the code and enhance readability. Ensuring default values for non-existent keys are correctly handled in `DefaultMunch` improves test coverage and functionality reliability. Lastly, refactoring path handling in tests makes them more robust against directory changes, ensuring consistent and stable test performance. This comprehensive approach addresses the core issues, leading to a more maintainable and robust codebase for the `Munch` library."
    }
  }
}