{
  "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 is the most helpful because it deals with relevant functionality within the matplotlib library, specifically addressing issues related to plotting which can also yield deprecation warnings when invalid arguments are passed due to changes in NumPy handling. The symptoms highlighted in Candidate A revolve around unexpected behavior during plotting, which has parallels to the deprecation warnings observed in the CURRENT bug report. Furthermore, it provides insights into structural and parameter handling that can be directly useful while adapting to the changes introduced in NumPy 1.24.",
  "instance_id": "matplotlib__matplotlib-24970",
  "repo": "matplotlib/matplotlib",
  "created_at": "2023-01-13T14:23:39Z",
  "problem_statement": "[Bug]: NumPy 1.24 deprecation warnings\n### Bug summary\r\n\r\nStarting NumPy 1.24 I observe several deprecation warnings.\r\n\r\n\r\n### Code for reproduction\r\n\r\n```python\r\nimport matplotlib.pyplot as plt\r\nimport numpy as np\r\n\r\nplt.get_cmap()(np.empty((0, ), dtype=np.uint8))\r\n```\r\n\r\n\r\n### Actual outcome\r\n\r\n```\r\n/usr/lib/python3.10/site-packages/matplotlib/colors.py:730: DeprecationWarning: NumPy will stop allowing conversion of out-of-bound Python integers to integer arrays.  The conversion of 257 to uint8 will fail in the future.\r\nFor the old behavior, usually:\r\n    np.array(value).astype(dtype)`\r\nwill give the desired result (the cast overflows).\r\n  xa[xa > self.N - 1] = self._i_over\r\n/usr/lib/python3.10/site-packages/matplotlib/colors.py:731: DeprecationWarning: NumPy will stop allowing conversion of out-of-bound Python integers to integer arrays.  The conversion of 256 to uint8 will fail in the future.\r\nFor the old behavior, usually:\r\n    np.array(value).astype(dtype)`\r\nwill give the desired result (the cast overflows).\r\n  xa[xa < 0] = self._i_under\r\n/usr/lib/python3.10/site-packages/matplotlib/colors.py:732: DeprecationWarning: NumPy will stop allowing conversion of out-of-bound Python integers to integer arrays.  The conversion of 258 to uint8 will fail in the future.\r\nFor the old behavior, usually:\r\n    np.array(value).astype(dtype)`\r\nwill give the desired result (the cast overflows).\r\n  xa[mask_bad] = self._i_bad\r\n```\r\n\r\n### Expected outcome\r\n\r\nNo warnings.\r\n\r\n### Additional information\r\n\r\n_No response_\r\n\r\n### Operating system\r\n\r\nArchLinux\r\n\r\n### Matplotlib Version\r\n\r\n3.6.2\r\n\r\n### Matplotlib Backend\r\n\r\nQtAgg\r\n\r\n### Python version\r\n\r\nPython 3.10.9\r\n\r\n### Jupyter version\r\n\r\n_No response_\r\n\r\n### Installation\r\n\r\nLinux package manager\n",
  "patch": "diff --git a/lib/matplotlib/colors.py b/lib/matplotlib/colors.py\n--- a/lib/matplotlib/colors.py\n+++ b/lib/matplotlib/colors.py\n@@ -715,16 +715,17 @@ def __call__(self, X, alpha=None, bytes=False):\n         if not xa.dtype.isnative:\n             xa = xa.byteswap().newbyteorder()  # Native byteorder is faster.\n         if xa.dtype.kind == \"f\":\n-            with np.errstate(invalid=\"ignore\"):\n-                xa *= self.N\n-                # Negative values are out of range, but astype(int) would\n-                # truncate them towards zero.\n-                xa[xa < 0] = -1\n-                # xa == 1 (== N after multiplication) is not out of range.\n-                xa[xa == self.N] = self.N - 1\n-                # Avoid converting large positive values to negative integers.\n-                np.clip(xa, -1, self.N, out=xa)\n-                xa = xa.astype(int)\n+            xa *= self.N\n+            # Negative values are out of range, but astype(int) would\n+            # truncate them towards zero.\n+            xa[xa < 0] = -1\n+            # xa == 1 (== N after multiplication) is not out of range.\n+            xa[xa == self.N] = self.N - 1\n+            # Avoid converting large positive values to negative integers.\n+            np.clip(xa, -1, self.N, out=xa)\n+        with np.errstate(invalid=\"ignore\"):\n+            # We need this cast for unsigned ints as well as floats\n+            xa = xa.astype(int)\n         # Set the over-range indices before the under-range;\n         # otherwise the under-range values get converted to over-range.\n         xa[xa > self.N - 1] = self._i_over\n"
}