{
    "Selected_candidate": {
        "pr_number": 19888,
        "pr_title": "Fix errorbar drawstyle",
        "pr_body": "## PR Summary\r\nclose #19879 \r\n\r\n## PR Checklist\r\n\r\n<!-- Please mark any checkboxes that do not apply to this PR as [N/A]. -->\r\n\r\n- [x] Has pytest style unit tests (and `pytest` passes).\r\n- [x] Is [Flake 8](https://flake8.pycqa.org/en/latest/) compliant (run `flake8` on changed files to check).\r\n- [N/A] New features are documented, with examples if plot related.\r\n- [x] Documentation is sphinx and numpydoc compliant (the docs should [build](https://matplotlib.org/devel/documenting_mpl.html#building-the-docs) without error).\r\n- [x] Conforms to Matplotlib style conventions (install `flake8-docstrings` and run `flake8 --docstring-convention=all`).\r\n- [N/A] New features have an entry in `doc/users/next_whats_new/` (follow instructions in README.rst there).\r\n- [N/A] API changes documented in `doc/api/next_api_changes/` (follow instructions in README.rst there).\r\n\r\n<!--\r\nThank you so much for your PR!  To help us review your contribution, please\r\nconsider the following points:\r\n\r\n- A development guide is available at https://matplotlib.org/devdocs/devel/index.html.\r\n\r\n- Help with git and github is available at\r\n  https://matplotlib.org/devel/gitwash/development_workflow.html.\r\n\r\n- Do not create the PR out of master, but out of a separate branch.\r\n\r\n- The PR title should summarize the changes, for example \"Raise ValueError on\r\n  non-numeric input to set_xlim\".  Avoid non-descriptive titles such as\r\n  \"Addresses issue #8576\".\r\n\r\n- The summary should provide at least 1-2 sentences describing the pull request\r\n  in detail (Why is this change required?  What problem does it solve?) and\r\n  link to any relevant issues.\r\n\r\n- If you are contributing fixes to docstrings, please pay attention to\r\n  http://matplotlib.org/devel/documenting_mpl.html#formatting.  In particular,\r\n  note the difference between using single backquotes, double backquotes, and\r\n  asterisks in the markup.\r\n\r\nWe understand that PRs can sometimes be overwhelming, especially as the\r\nreviews start coming in.  Please let us know if the reviews are unclear or\r\nthe recommended next step seems overly demanding, if you would like help in\r\naddressing a reviewer's comments, or if you have been waiting too long to hear\r\nback on your PR.\r\n-->\r\n",
        "issue_id": 19879,
        "issue_title": "Using \"drawstyle\" raises AttributeError in errorbar, when yerr is specified.",
        "issue_body": "<!--To help us understand and resolve your issue, please fill out the form to the best of your ability.-->\r\n<!--You can feel free to delete the sections that do not apply.-->\r\n\r\n### Bug report\r\n\r\nHello maintainers! \r\nStarting from 3.4.1, I'm having failures in CI related to a failure in `errorbar` (e.g. https://github.com/StingraySoftware/HENDRICS/runs/2267849177?check_suite_focus=true). Until at least 3.3.4, this code line\r\n\r\n```\r\nplt.errorbar(x, profile, yerr=profile_err, drawstyle=\"steps-mid\")\r\n```\r\n\r\nused to work, now it doesn't. After a little debugging, I realized that `yerr` and `drawstyle` seem incompatible. I write a MWE below\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n\r\n**Code for reproduction**\r\n\r\n<!--A minimum code snippet required to reproduce the bug.\r\nPlease make sure to minimize the number of dependencies required, and provide\r\nany necessary plotted data.\r\nAvoid using threads, as Matplotlib is (explicitly) not thread-safe.-->\r\n\r\n```python\r\nIn [1]: import numpy as np\r\n\r\nIn [2]: import matplotlib.pyplot as plt\r\n\r\nIn [3]: x = np.arange(10)\r\n\r\nIn [4]: y = np.random.poisson(100, 10)\r\n\r\nIn [5]: plt.errorbar(x, y, drawstyle=\"steps-mid\")\r\nOut[5]: <ErrorbarContainer object of 3 artists>\r\n\r\nIn [6]: plt.show()  # It works\r\n\r\nIn [7]: yerr = np.zeros_like(y) + 10\r\n\r\nIn [8]: plt.errorbar(x, y, yerr=yerr, drawstyle=\"steps-mid\")\r\n---------------------------------------------------------------------------\r\nAttributeError                            Traceback (most recent call last)\r\n<ipython-input-8-a70dfad4683c> in <module>\r\n----> 1 plt.errorbar(x, y, yerr=yerr, drawstyle=\"steps-mid\")\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/pyplot.py in errorbar(x, y, yerr, xerr, fmt, ecolor, elinewidth, capsize, barsabove, lolims, uplims, xlolims, xuplims, errorevery, capthick, data, **kwargs)\r\n   2739         uplims=False, xlolims=False, xuplims=False, errorevery=1,\r\n   2740         capthick=None, *, data=None, **kwargs):\r\n-> 2741     return gca().errorbar(\r\n   2742         x, y, yerr=yerr, xerr=xerr, fmt=fmt, ecolor=ecolor,\r\n   2743         elinewidth=elinewidth, capsize=capsize, barsabove=barsabove,\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/__init__.py in inner(ax, data, *args, **kwargs)\r\n   1350     def inner(ax, *args, data=None, **kwargs):\r\n   1351         if data is None:\r\n-> 1352             return func(ax, *map(sanitize_sequence, args), **kwargs)\r\n   1353 \r\n   1354         bound = new_sig.bind(ax, *args, **kwargs)\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/axes/_axes.py in errorbar(self, x, y, yerr, xerr, fmt, ecolor, elinewidth, capsize, barsabove, lolims, uplims, xlolims, xuplims, errorevery, capthick, **kwargs)\r\n   3496         if yerr is not None:\r\n   3497             lower, upper = extract_err('y', yerr, y, lolims, uplims)\r\n-> 3498             barcols.append(self.vlines(\r\n   3499                 *apply_mask([x, lower, upper], everymask), **eb_lines_style))\r\n   3500             # select points without upper/lower limits in y and\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/__init__.py in inner(ax, data, *args, **kwargs)\r\n   1350     def inner(ax, *args, data=None, **kwargs):\r\n   1351         if data is None:\r\n-> 1352             return func(ax, *map(sanitize_sequence, args), **kwargs)\r\n   1353 \r\n   1354         bound = new_sig.bind(ax, *args, **kwargs)\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/axes/_axes.py in vlines(self, x, ymin, ymax, colors, linestyles, label, **kwargs)\r\n   1118                                      linestyles=linestyles, label=label)\r\n   1119         self.add_collection(lines, autolim=False)\r\n-> 1120         lines.update(kwargs)\r\n   1121 \r\n   1122         if len(x) > 0:\r\n\r\n~/anaconda3/envs/py38/lib/python3.8/site-packages/matplotlib/artist.py in update(self, props)\r\n   1060                     func = getattr(self, f\"set_{k}\", None)\r\n   1061                     if not callable(func):\r\n-> 1062                         raise AttributeError(f\"{type(self).__name__!r} object \"\r\n   1063                                              f\"has no property {k!r}\")\r\n   1064                     ret.append(func(v))\r\n\r\nAttributeError: 'LineCollection' object has no property 'drawstyle'\r\n\r\n```\r\n\r\n**Matplotlib version**\r\n<!--Please specify your platform and versions of the relevant libraries you are using:-->\r\n  * Operating system: Mac OS, Linux, Windows (all tested on CI)\r\n  * Matplotlib version (`import matplotlib; print(matplotlib.__version__)`): 3.4.1\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): various\r\n  * Python version: 3.8\r\n  * Jupyter version (if applicable): \r\n  * Other libraries: \r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda--> \r\nI installed matplotlib from pip (on CI) and conda (my laptop)\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
        "issue_closed_at": "2021-04-09T00:06:34Z",
        "base_commit": "0c29bc0dd8169870dd93d656f33b62df0e6e96a6",
        "changes": [
            {
                "file": "lib/matplotlib/axes/_axes.py",
                "type": "function",
                "name": "errorbar",
                "class_name": "Axes",
                "code": "def errorbar(self, x, y, yerr=None, xerr=None,\n                 fmt='', ecolor=None, elinewidth=None, capsize=None,\n                 barsabove=False, lolims=False, uplims=False,\n                 xlolims=False, xuplims=False, errorevery=1, capthick=None,\n                 **kwargs):\n        \"\"\"\n        Plot y versus x as lines and/or markers with attached errorbars.\n\n        *x*, *y* define the data locations, *xerr*, *yerr* define the errorbar\n        sizes. By default, this draws the data markers/lines as well the\n        errorbars. Use fmt='none' to draw errorbars without any data markers.\n\n        Parameters\n        ----------\n        x, y : float or array-like\n            The data positions.\n\n        xerr, yerr : float or array-like, shape(N,) or shape(2, N), optional\n            The errorbar sizes:\n\n            - scalar: Symmetric +/- values for all data points.\n            - shape(N,): Symmetric +/-values for each data point.\n            - shape(2, N): Separate - and + values for each bar. First row\n              contains the lower errors, the second row contains the upper\n              errors.\n            - *None*: No errorbar.\n\n            Note that all error arrays should have *positive* values.\n\n            See :doc:`/gallery/statistics/errorbar_features`\n            for an example on the usage of ``xerr`` and ``yerr``.\n\n        fmt : str, default: ''\n            The format for the data points / data lines. See `.plot` for\n            details.\n\n            Use 'none' (case insensitive) to plot errorbars without any data\n            markers.\n\n        ecolor : color, default: None\n            The color of the errorbar lines.  If None, use the color of the\n            line connecting the markers.\n\n        elinewidth : float, default: None\n            The linewidth of the errorbar lines. If None, the linewidth of\n            the current style is used.\n\n        capsize : float, default: :rc:`errorbar.capsize`\n            The length of the error bar caps in points.\n\n        capthick : float, default: None\n            An alias to the keyword argument *markeredgewidth* (a.k.a. *mew*).\n            This setting is a more sensible name for the property that\n            controls the thickness of the error bar cap in points. For\n            backwards compatibility, if *mew* or *markeredgewidth* are given,\n            then they will over-ride *capthick*. This may change in future\n            releases.\n\n        barsabove : bool, default: False\n            If True, will plot the errorbars above the plot\n            symbols. Default is below.\n\n        lolims, uplims, xlolims, xuplims : bool, default: False\n            These arguments can be used to indicate that a value gives only\n            upper/lower limits.  In that case a caret symbol is used to\n            indicate this. *lims*-arguments may be scalars, or array-likes of\n            the same length as *xerr* and *yerr*.  To use limits with inverted\n            axes, `~.Axes.set_xlim` or `~.Axes.set_ylim` must be called before\n            :meth:`errorbar`.  Note the tricky parameter names: setting e.g.\n            *lolims* to True means that the y-value is a *lower* limit of the\n            True value, so, only an *upward*-pointing arrow will be drawn!\n\n        errorevery : int or (int, int), default: 1\n            draws error bars on a subset of the data. *errorevery* =N draws\n            error bars on the points (x[::N], y[::N]).\n            *errorevery* =(start, N) draws error bars on the points\n            (x[start::N], y[start::N]). e.g. errorevery=(6, 3)\n            adds error bars to the data at (x[6], x[9], x[12], x[15], ...).\n            Used to avoid overlapping error bars when two series share x-axis\n            values.\n\n        Returns\n        -------\n        `.ErrorbarContainer`\n            The container contains:\n\n            - plotline: `.Line2D` instance of x, y plot markers and/or line.\n            - caplines: A tuple of `.Line2D` instances of the error bar caps.\n            - barlinecols: A tuple of `.LineCollection` with the horizontal and\n              vertical error ranges.\n\n        Other Parameters\n        ----------------\n        **kwargs\n            All other keyword arguments are passed on to the `~.Axes.plot` call\n            drawing the markers. For example, this code makes big red squares\n            with thick green edges::\n\n                x, y, yerr = rand(3, 10)\n                errorbar(x, y, yerr, marker='s', mfc='red',\n                         mec='green', ms=20, mew=4)\n\n            where *mfc*, *mec*, *ms* and *mew* are aliases for the longer\n            property names, *markerfacecolor*, *markeredgecolor*, *markersize*\n            and *markeredgewidth*.\n\n            Valid kwargs for the marker properties are `.Line2D` properties:\n\n            %(Line2D_kwdoc)s\n        \"\"\"\n        kwargs = cbook.normalize_kwargs(kwargs, mlines.Line2D)\n        # anything that comes in as 'None', drop so the default thing\n        # happens down stream\n        kwargs = {k: v for k, v in kwargs.items() if v is not None}\n        kwargs.setdefault('zorder', 2)\n\n        # Casting to object arrays preserves units.\n        if not isinstance(x, np.ndarray):\n            x = np.asarray(x, dtype=object)\n        if not isinstance(y, np.ndarray):\n            y = np.asarray(y, dtype=object)\n        if xerr is not None and not isinstance(xerr, np.ndarray):\n            xerr = np.asarray(xerr, dtype=object)\n        if yerr is not None and not isinstance(yerr, np.ndarray):\n            yerr = np.asarray(yerr, dtype=object)\n        x, y = np.atleast_1d(x, y)  # Make sure all the args are iterable.\n        if len(x) != len(y):\n            raise ValueError(\"'x' and 'y' must have the same size\")\n\n        if isinstance(errorevery, Integral):\n            errorevery = (0, errorevery)\n        if isinstance(errorevery, tuple):\n            if (len(errorevery) == 2 and\n                    isinstance(errorevery[0], Integral) and\n                    isinstance(errorevery[1], Integral)):\n                errorevery = slice(errorevery[0], None, errorevery[1])\n            else:\n                raise ValueError(\n                    f'errorevery={errorevery!r} is a not a tuple of two '\n                    f'integers')\n        elif isinstance(errorevery, slice):\n            pass\n        elif not isinstance(errorevery, str) and np.iterable(errorevery):\n            # fancy indexing\n            try:\n                x[errorevery]\n            except (ValueError, IndexError) as err:\n                raise ValueError(\n                    f\"errorevery={errorevery!r} is iterable but not a valid \"\n                    f\"NumPy fancy index to match 'xerr'/'yerr'\") from err\n        else:\n            raise ValueError(\n                f\"errorevery={errorevery!r} is not a recognized value\")\n        everymask = np.zeros(len(x), bool)\n        everymask[errorevery] = True\n\n        label = kwargs.pop(\"label\", None)\n        kwargs['label'] = '_nolegend_'\n\n        # Create the main line and determine overall kwargs for child artists.\n        # We avoid calling self.plot() directly, or self._get_lines(), because\n        # that would call self._process_unit_info again, and do other indirect\n        # data processing.\n        (data_line, base_style), = self._get_lines._plot_args(\n            (x, y) if fmt == '' else (x, y, fmt), kwargs, return_kwargs=True)\n\n        # Do this after creating `data_line` to avoid modifying `base_style`.\n        if barsabove:\n            data_line.set_zorder(kwargs['zorder'] - .1)\n        else:\n            data_line.set_zorder(kwargs['zorder'] + .1)\n\n        # Add line to plot, or throw it away and use it to determine kwargs.\n        if fmt.lower() != 'none':\n            self.add_line(data_line)\n        else:\n            data_line = None\n            # Remove alpha=0 color that _get_lines._plot_args returns for\n            # 'none' format, and replace it with user-specified color, if\n            # supplied.\n            base_style.pop('color')\n            if 'color' in kwargs:\n                base_style['color'] = kwargs.pop('color')\n\n        if 'color' not in base_style:\n            base_style['color'] = 'C0'\n        if ecolor is None:\n            ecolor = base_style['color']\n\n        # Eject any marker information from line format string, as it's not\n        # needed for bars or caps.\n        base_style.pop('marker', None)\n        base_style.pop('markersize', None)\n        base_style.pop('markerfacecolor', None)\n        base_style.pop('markeredgewidth', None)\n        base_style.pop('markeredgecolor', None)\n        base_style.pop('markevery', None)\n        base_style.pop('linestyle', None)\n        base_style.pop('fillstyle', None)\n\n        # Make the style dict for the line collections (the bars).\n        eb_lines_style = {**base_style, 'color': ecolor}\n\n        if elinewidth:\n            eb_lines_style['linewidth'] = elinewidth\n        elif 'linewidth' in kwargs:\n            eb_lines_style['linewidth'] = kwargs['linewidth']\n\n        for key in ('transform', 'alpha', 'zorder', 'rasterized'):\n            if key in kwargs:\n                eb_lines_style[key] = kwargs[key]\n\n        # Make the style dict for caps (the \"hats\").\n        eb_cap_style = {**base_style, 'linestyle': 'none'}\n        if capsize is None:\n            capsize = rcParams[\"errorbar.capsize\"]\n        if capsize > 0:\n            eb_cap_style['markersize'] = 2. * capsize\n        if capthick is not None:\n            eb_cap_style['markeredgewidth'] = capthick\n\n        # For backwards-compat, allow explicit setting of\n        # 'markeredgewidth' to over-ride capthick.\n        for key in ('markeredgewidth', 'transform', 'alpha',\n                    'zorder', 'rasterized'):\n            if key in kwargs:\n                eb_cap_style[key] = kwargs[key]\n        eb_cap_style['color'] = ecolor\n\n        barcols = []\n        caplines = []\n\n        # arrays fine here, they are booleans and hence not units\n        lolims = np.broadcast_to(lolims, len(x)).astype(bool)\n        uplims = np.broadcast_to(uplims, len(x)).astype(bool)\n        xlolims = np.broadcast_to(xlolims, len(x)).astype(bool)\n        xuplims = np.broadcast_to(xuplims, len(x)).astype(bool)\n\n        # Vectorized fancy-indexer.\n        def apply_mask(arrays, mask): return [array[mask] for array in arrays]\n\n        def extract_err(name, err, data, lolims, uplims):\n            \"\"\"\n            Private function to compute error bars.\n\n            Parameters\n            ----------\n            name : {'x', 'y'}\n                Name used in the error message.\n            err : array-like\n                xerr or yerr from errorbar().\n            data : array-like\n                x or y from errorbar().\n            lolims : array-like\n                Error is only applied on **upper** side when this is True.  See\n                the note in the main docstring about this parameter's name.\n            uplims : array-like\n                Error is only applied on **lower** side when this is True.  See\n                the note in the main docstring about this parameter's name.\n            \"\"\"\n            try:\n                np.broadcast_to(err, (2, len(data)))\n            except ValueError:\n                raise ValueError(\n                    f\"'{name}err' (shape: {np.shape(err)}) must be a scalar \"\n                    f\"or a 1D or (2, n) array-like whose shape matches \"\n                    f\"'{name}' (shape: {np.shape(data)})\") from None\n            # This is like\n            #     low, high = np.broadcast_to(...)\n            #     return data - low * ~lolims, data + high * ~uplims\n            # except that broadcast_to would strip units.\n            return data + np.row_stack([-(1 - lolims), 1 - uplims]) * err\n\n        if xerr is not None:\n            left, right = extract_err('x', xerr, x, xlolims, xuplims)\n            barcols.append(self.hlines(\n                *apply_mask([y, left, right], everymask), **eb_lines_style))\n            # select points without upper/lower limits in x and\n            # draw normal errorbars for these points\n            noxlims = ~(xlolims | xuplims)\n            if noxlims.any() and capsize > 0:\n                yo, lo, ro = apply_mask([y, left, right], noxlims & everymask)\n                caplines.extend([\n                    mlines.Line2D(lo, yo, marker='|', **eb_cap_style),\n                    mlines.Line2D(ro, yo, marker='|', **eb_cap_style)])\n            if xlolims.any():\n                xo, yo, ro = apply_mask([x, y, right], xlolims & everymask)\n                if self.xaxis_inverted():\n                    marker = mlines.CARETLEFTBASE\n                else:\n                    marker = mlines.CARETRIGHTBASE\n                caplines.append(mlines.Line2D(\n                    ro, yo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='|', **eb_cap_style))\n            if xuplims.any():\n                xo, yo, lo = apply_mask([x, y, left], xuplims & everymask)\n                if self.xaxis_inverted():\n                    marker = mlines.CARETRIGHTBASE\n                else:\n                    marker = mlines.CARETLEFTBASE\n                caplines.append(mlines.Line2D(\n                    lo, yo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='|', **eb_cap_style))\n\n        if yerr is not None:\n            lower, upper = extract_err('y', yerr, y, lolims, uplims)\n            barcols.append(self.vlines(\n                *apply_mask([x, lower, upper], everymask), **eb_lines_style))\n            # select points without upper/lower limits in y and\n            # draw normal errorbars for these points\n            noylims = ~(lolims | uplims)\n            if noylims.any() and capsize > 0:\n                xo, lo, uo = apply_mask([x, lower, upper], noylims & everymask)\n                caplines.extend([\n                    mlines.Line2D(xo, lo, marker='_', **eb_cap_style),\n                    mlines.Line2D(xo, uo, marker='_', **eb_cap_style)])\n            if lolims.any():\n                xo, yo, uo = apply_mask([x, y, upper], lolims & everymask)\n                if self.yaxis_inverted():\n                    marker = mlines.CARETDOWNBASE\n                else:\n                    marker = mlines.CARETUPBASE\n                caplines.append(mlines.Line2D(\n                    xo, uo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='_', **eb_cap_style))\n            if uplims.any():\n                xo, yo, lo = apply_mask([x, y, lower], uplims & everymask)\n                if self.yaxis_inverted():\n                    marker = mlines.CARETUPBASE\n                else:\n                    marker = mlines.CARETDOWNBASE\n                caplines.append(mlines.Line2D(\n                    xo, lo, ls='None', marker=marker, **eb_cap_style))\n                if capsize > 0:\n                    caplines.append(mlines.Line2D(\n                        xo, yo, marker='_', **eb_cap_style))\n\n        for l in caplines:\n            self.add_line(l)\n\n        self._request_autoscale_view()\n        errorbar_container = ErrorbarContainer(\n            (data_line, tuple(caplines), tuple(barcols)),\n            has_xerr=(xerr is not None), has_yerr=(yerr is not None),\n            label=label)\n        self.containers.append(errorbar_container)\n\n        return errorbar_container"
            }
        ]
    },
    "Justification": "Candidate E provides a direct comparison in functionality, specifically regarding the `drawstyle` parameter in the `errorbar` function, which is reported to raise an `AttributeError` when `yerr` is specified. This is somewhat analogous to the current bug that mentions an issue with the `axis` attribute in different axes classes. Both issues pertain to parameter incompatibility and method behavior, making Candidate E particularly helpful for debugging similar attribute errors in class methods. The fix from Candidate E could offer insights into resolving type errors arising from method compatibility, essential for the current bug scenario."
}