{
  "Selected_candidate": {
    "pr_number": 4050,
    "pr_title": "Fix masked array handling",
    "pr_body": "This fixes #4049, and is an alternative to #4008.  Unlike #4008, this fixes the problem directly in `_fast_from_vertices_and_codes` rather than at a number of call sites that call it.  While it does make that function slower, it apparently needs the masked array treatment from all places that call it anyway, so sticking it right inside the function seems fine.\n\nIn the process of fixing this bug, it was discovered that a fix to handle NaNs in draw_markers (cf43d57) was not brought over from the 1.4.x branch (probably my fault, since merges of C++ code are really hairy right now).  This missing bugfix was papered over by the fact that NaNs were no longer being passed to `draw_markers`.\n\nLastly, this adds an optimization to the log scaling to use NaNs rather than masked arrays.  The original array is not touched, so this should be effectively the same thing, just without the large time and memory overhead of masked arrays.\n\nThis includes a test I shamelessly cherry-picked from #4008.\n",
    "issue_id": 4049,
    "issue_title": "Issue with plotting zeros in log space",
    "issue_body": "The following simple example shows an issues with zero values when calling `loglog`:\n\n```\nimport matplotlib.pyplot as plt\n\nnu = [1e2, 1e3, 1e4, 1e5, 1e6, 1e7]\nfnu = [1, 1e-30, 1e-45, 0., 0., 0.]\n\nfig = plt.figure()\nax = fig.add_subplot(1,1,1)\nax.loglog(nu, fnu, color='blue')\nfig.savefig('mpl.png')\n```\n\nThe output is:\n\n![mpl](https://cloud.githubusercontent.com/assets/314716/5959827/8d0677cc-a7d1-11e4-8cd3-7227dee4a37a.png)\n\nOf course, zero is undefined in log space, but I don't understand why it defaults to plotting a value of 0.1 instead. Doing:\n\n```\nfnu = [1, 1e-30, 1e-45, np.nan, np.nan, np.nan]\n```\n\nDoes the right thing on the other hand. So maybe the best solution is to replace 0 values by np.nan inside `loglog`?\n\nI am using matplotlib 1.5.x with Python 3.4.\n",
    "issue_closed_at": "2015-01-30T23:18:44Z",
    "base_commit": "dd4b2bc76c800f2fae69953c888681fd67788192",
    "changes": [
      {
        "file": "lib/matplotlib/path.py",
        "type": "function",
        "name": "_fast_from_codes_and_verts",
        "class_name": "Path",
        "code": "def _fast_from_codes_and_verts(cls, verts, codes, internals=None):\n        \"\"\"\n        Creates a Path instance without the expense of calling the constructor\n\n        Parameters\n        ----------\n        verts : numpy array\n        codes : numpy array (may not be None)\n        internals : dict or None\n            The attributes that the resulting path should have.\n            Allowed keys are ``readonly``, ``should_simplify``,\n            ``simplify_threshold``, ``has_nonfinite`` and\n            ``interpolation_steps``.\n\n        \"\"\"\n        internals = internals or {}\n        pth = cls.__new__(cls)\n        pth._vertices = verts\n        pth._codes = codes\n        pth._readonly = internals.pop('readonly', False)\n        pth.should_simplify = internals.pop('should_simplify', True)\n        pth.simplify_threshold = (\n            internals.pop('simplify_threshold',\n                          rcParams['path.simplify_threshold'])\n        )\n        pth._has_nonfinite = internals.pop('has_nonfinite', False)\n        pth._interpolation_steps = internals.pop('interpolation_steps', 1)\n        if internals:\n            raise ValueError('Unexpected internals provided to '\n                             '_fast_from_codes_and_verts: '\n                             '{0}'.format('\\n *'.join(six.iterkeys(\n                                 internals\n                             ))))\n        return pth"
      },
      {
        "file": "lib/matplotlib/path.py",
        "type": "function",
        "name": "_fast_from_codes_and_verts",
        "class_name": "Path",
        "code": "def _fast_from_codes_and_verts(cls, verts, codes, internals=None):\n        \"\"\"\n        Creates a Path instance without the expense of calling the constructor\n\n        Parameters\n        ----------\n        verts : numpy array\n        codes : numpy array (may not be None)\n        internals : dict or None\n            The attributes that the resulting path should have.\n            Allowed keys are ``readonly``, ``should_simplify``,\n            ``simplify_threshold``, ``has_nonfinite`` and\n            ``interpolation_steps``.\n\n        \"\"\"\n        internals = internals or {}\n        pth = cls.__new__(cls)\n        pth._vertices = verts\n        pth._codes = codes\n        pth._readonly = internals.pop('readonly', False)\n        pth.should_simplify = internals.pop('should_simplify', True)\n        pth.simplify_threshold = (\n            internals.pop('simplify_threshold',\n                          rcParams['path.simplify_threshold'])\n        )\n        pth._has_nonfinite = internals.pop('has_nonfinite', False)\n        pth._interpolation_steps = internals.pop('interpolation_steps', 1)\n        if internals:\n            raise ValueError('Unexpected internals provided to '\n                             '_fast_from_codes_and_verts: '\n                             '{0}'.format('\\n *'.join(six.iterkeys(\n                                 internals\n                             ))))\n        return pth"
      },
      {
        "file": "lib/matplotlib/scale.py",
        "type": "function",
        "name": "get_transform",
        "class_name": "SymmetricalLogScale",
        "code": "def get_transform(self):\n        \"\"\"\n        Return a :class:`SymmetricalLogTransform` instance.\n        \"\"\"\n        return self._transform"
      }
    ]
  },
  "Justification": "Candidate A addresses similar core functionalities in matplotlib regarding figure representation and has relevant insights on handling data for output, which can aid in debugging the pickling issue seen in the CURRENT bug. Its analysis and fix indirectly relate to the serialization process, potentially guiding users on proper management of figure data.",
  "instance_id": "matplotlib__matplotlib-25332",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-02-26T11:18:40Z",
  "problem_statement": "[Bug]: Unable to pickle figure with aligned labels\n### Bug summary\r\n\r\n Unable to pickle figure after calling `align_labels()`\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport pickle\r\n\r\nfig = plt.figure()\r\nax1 = fig.add_subplot(211)\r\nax2 = fig.add_subplot(212)\r\ntime=[0,1,2,3,4]\r\nspeed=[40000,4300,4500,4700,4800]\r\nacc=[10,11,12,13,14]\r\nax1.plot(time,speed)\r\nax1.set_ylabel('speed')\r\nax2.plot(time,acc)\r\nax2.set_ylabel('acc')\r\n\r\nfig.align_labels() ##pickling works after removing this line \r\n\r\npickle.dumps(fig)\r\nplt.show()\r\n```\r\n\r\n\r\n### Actual outcome\r\n```\r\nalign.py\", line 16\r\npickle.dumps(fig)\r\nTypeError: cannot pickle 'weakref.ReferenceType' object\r\n```\r\n### Expected outcome\r\n\r\nPickling successful\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nWindows\r\n\r\n### Matplotlib Version\r\n\r\n3.7.0\r\n\r\n### Matplotlib Backend\r\n\r\n_No response_\r\n\r\n### Python version\r\n\r\n_No response_\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nNone\n",
  "patch": "diff --git a/lib/matplotlib/cbook.py b/lib/matplotlib/cbook.py\n--- a/lib/matplotlib/cbook.py\n+++ b/lib/matplotlib/cbook.py\n@@ -788,6 +788,19 @@ class Grouper:\n     def __init__(self, init=()):\n         self._mapping = {weakref.ref(x): [weakref.ref(x)] for x in init}\n \n+    def __getstate__(self):\n+        return {\n+            **vars(self),\n+            # Convert weak refs to strong ones.\n+            \"_mapping\": {k(): [v() for v in vs] for k, vs in self._mapping.items()},\n+        }\n+\n+    def __setstate__(self, state):\n+        vars(self).update(state)\n+        # Convert strong refs to weak ones.\n+        self._mapping = {weakref.ref(k): [*map(weakref.ref, vs)]\n+                         for k, vs in self._mapping.items()}\n+\n     def __contains__(self, item):\n         return weakref.ref(item) in self._mapping\n \n"
}