{
    "Selected_candidate": {
        "pr_number": 10772,
        "pr_title": "API: check locator and formatter args when passed",
        "pr_body": "## PR Summary\r\n\r\nRight now, `ax.set_major_formatter` and `ax.set_major_locator` throw pretty deep error messages if the wrong type of data is passed to them.  This catches at entry. \r\n\r\nCloses #10769\r\n\r\n```python\r\nimport matplotlib\r\nimport matplotlib.pyplot as plt\r\n\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\nplt.show()\r\n```\r\nNow returns:\r\n```\r\nTraceback (most recent call last):\r\n  File \"testEngTicker.py\", line 6, in <module>\r\n    ax.xaxis.set_major_formatter(matplotlib.ticker.LogLocator())\r\n  File \"/Users/jklymak/matplotlib/lib/matplotlib/axis.py\", line 1573, in set_major_formatter\r\n    \"formatter argument must be matplotlib.ticker.Formatter\")\r\nValueError: formatter argument must be matplotlib.ticker.Formatter\r\n```\r\ninstead of something down in the bowels of drawing the ticker...\r\n\r\n## PR Checklist\r\n\r\n- [x] Code is PEP 8 compliant\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-->",
        "issue_id": 10769,
        "issue_title": "DOC: set_major_locator could check that its getting a Locator (was EngFormatter broken?)",
        "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\n`matplotlib.ticker.EngFormatter` seems to be broken:\r\n\r\n<!--A short 1-2 sentences that succinctly describes the bug-->\r\n```\r\nf, ax = plt.subplots(1, 1, figsize=(15, 10))\r\nplt.plot([10**4, 11*10**4], [100, 150])\r\nax.xaxis.set_major_locator(matplotlib.ticker.EngFormatter())\r\n```\r\n```\r\nError in callback <function install_repl_displayhook.<locals>.post_execute at 0x7f60a9b120d0> (for post_execute):\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/pyplot.py in post_execute()\r\n    147             def post_execute():\r\n    148                 if matplotlib.is_interactive():\r\n--> 149                     draw_all()\r\n    150 \r\n    151             # IPython >= 2\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/_pylab_helpers.py in draw_all(cls, force)\r\n    134         for f_mgr in cls.get_all_fig_managers():\r\n    135             if force or f_mgr.canvas.figure.stale:\r\n--> 136                 f_mgr.canvas.draw_idle()\r\n    137 \r\n    138 atexit.register(Gcf.destroy_all)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in draw_idle(self, *args, **kwargs)\r\n   2053         if not self._is_idle_drawing:\r\n   2054             with self._idle_draw_cntx():\r\n-> 2055                 self.draw(*args, **kwargs)\r\n   2056 \r\n   2057     def draw_cursor(self, event):\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/formatters.py in __call__(self, obj)\r\n    339                 pass\r\n    340             else:\r\n--> 341                 return printer(obj)\r\n    342             # Finally look for special method names\r\n    343             method = get_real_method(obj, self.print_method)\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in <lambda>(fig)\r\n    236 \r\n    237     if 'png' in formats:\r\n--> 238         png_formatter.for_type(Figure, lambda fig: print_figure(fig, 'png', **kwargs))\r\n    239     if 'retina' in formats or 'png2x' in formats:\r\n    240         png_formatter.for_type(Figure, lambda fig: retina_figure(fig, **kwargs))\r\n\r\n/usr/local/lib/python3.5/dist-packages/IPython/core/pylabtools.py in print_figure(fig, fmt, bbox_inches, **kwargs)\r\n    120 \r\n    121     bytes_io = BytesIO()\r\n--> 122     fig.canvas.print_figure(bytes_io, **kw)\r\n    123     data = bytes_io.getvalue()\r\n    124     if fmt == 'svg':\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backend_bases.py in print_figure(self, filename, dpi, facecolor, edgecolor, orientation, format, **kwargs)\r\n   2208                     orientation=orientation,\r\n   2209                     dryrun=True,\r\n-> 2210                     **kwargs)\r\n   2211                 renderer = self.figure._cachedRenderer\r\n   2212                 bbox_inches = self.figure.get_tightbbox(renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in print_png(self, filename_or_obj, *args, **kwargs)\r\n    509 \r\n    510     def print_png(self, filename_or_obj, *args, **kwargs):\r\n--> 511         FigureCanvasAgg.draw(self)\r\n    512         renderer = self.get_renderer()\r\n    513         original_dpi = renderer.dpi\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py in draw(self)\r\n    429             # if toolbar:\r\n    430             #     toolbar.set_cursor(cursors.WAIT)\r\n--> 431             self.figure.draw(self.renderer)\r\n    432             # A GUI class may be need to update a window using this draw, so\r\n    433             # don't forget to call the superclass.\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py in draw(self, renderer)\r\n   1473 \r\n   1474             mimage._draw_list_compositing_images(\r\n-> 1475                 renderer, self, artists, self.suppressComposite)\r\n   1476 \r\n   1477             renderer.close_group('figure')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py in draw(self, renderer, inframe)\r\n   2605             renderer.stop_rasterizing()\r\n   2606 \r\n-> 2607         mimage._draw_list_compositing_images(renderer, self, artists)\r\n   2608 \r\n   2609         renderer.close_group('axes')\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/image.py in _draw_list_compositing_images(renderer, parent, artists, suppress_composite)\r\n    139     if not_composite or not has_images:\r\n    140         for a in artists:\r\n--> 141             a.draw(renderer)\r\n    142     else:\r\n    143         # Composite any adjacent images together\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py in draw_wrapper(artist, renderer, *args, **kwargs)\r\n     53                 renderer.start_filter()\r\n     54 \r\n---> 55             return draw(artist, renderer, *args, **kwargs)\r\n     56         finally:\r\n     57             if artist.get_agg_filter() is not None:\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in draw(self, renderer, *args, **kwargs)\r\n   1185         renderer.open_group(__name__)\r\n   1186 \r\n-> 1187         ticks_to_draw = self._update_ticks(renderer)\r\n   1188         ticklabelBoxes, ticklabelBoxes2 = self._get_tick_bboxes(ticks_to_draw,\r\n   1189                                                                 renderer)\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in _update_ticks(self, renderer)\r\n   1023 \r\n   1024         interval = self.get_view_interval()\r\n-> 1025         tick_tups = list(self.iter_ticks())  # iter_ticks calls the locator\r\n   1026         if self._smart_bounds and tick_tups:\r\n   1027             # handle inverted limits\r\n\r\n/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py in iter_ticks(self)\r\n    966         Iterate through all of the major and minor ticks.\r\n    967         \"\"\"\r\n--> 968         majorLocs = self.major.locator()\r\n    969         majorTicks = self.get_major_ticks(len(majorLocs))\r\n    970         self.major.formatter.set_locs(majorLocs)\r\n\r\nTypeError: __call__() missing 1 required positional argument: 'x'\r\n\r\n<Figure size 1080x720 with 1 Axes>\r\n```\r\n<!--A minimum code snippet required to reproduce the bug, also minimizing the number of dependencies required-->\r\n\r\n\r\n<!--A description of the expected outcome from the code snippet-->\r\n<!--If this used to work in an earlier version of Matplotlib, please note the version it used to work on-->\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: Ubuntu 16.04\r\n  * Matplotlib version: 2.2.0\r\n  * Matplotlib backend (`print(matplotlib.get_backend())`): module://ipykernel.pylab.backend_inline\r\n  * Python version: 3.5.2 (IPython 6.2.1)\r\n  * Jupyter version (if applicable): Jupyter Lab beta 0.31.8\r\n\r\n\r\n<!--Please tell us how you installed matplotlib and python e.g., from source, pip, conda-->\r\n<!--If you installed from conda, please specify which channel you used if not the default-->\r\n\r\n",
        "issue_closed_at": "2018-03-14T06:41:19Z",
        "base_commit": "8af1fac8baf9bbf06160d8a91575d3d6d8fdca04",
        "changes": [
            {
                "file": "lib/matplotlib/axis.py",
                "type": "function",
                "name": "set_major_formatter",
                "class_name": "Axis",
                "code": "def set_major_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the major ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_majfmt = False\n        self.major.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
            },
            {
                "file": "lib/matplotlib/axis.py",
                "type": "function",
                "name": "set_minor_formatter",
                "class_name": "Axis",
                "code": "def set_minor_formatter(self, formatter):\n        \"\"\"\n        Set the formatter of the minor ticker\n\n        ACCEPTS: A :class:`~matplotlib.ticker.Formatter` instance\n        \"\"\"\n        self.isDefault_minfmt = False\n        self.minor.formatter = formatter\n        formatter.set_axis(self)\n        self.stale = True"
            },
            {
                "file": "lib/matplotlib/axis.py",
                "type": "function",
                "name": "set_major_locator",
                "class_name": "Axis",
                "code": "def set_major_locator(self, locator):\n        \"\"\"\n        Set the locator of the major ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_majloc = False\n        self.major.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
            },
            {
                "file": "lib/matplotlib/axis.py",
                "type": "function",
                "name": "set_minor_locator",
                "class_name": "Axis",
                "code": "def set_minor_locator(self, locator):\n        \"\"\"\n        Set the locator of the minor ticker\n\n        ACCEPTS: a :class:`~matplotlib.ticker.Locator` instance\n        \"\"\"\n        self.isDefault_minloc = False\n        self.minor.locator = locator\n        locator.set_axis(self)\n        self.stale = True"
            }
        ]
    },
    "Justification": "Candidate A addresses a relevant aspect of the Matplotlib library concerning how formatters and locators are used, which is closely related to the error raised in the CURRENT bug report regarding attributes of a `Poly3DCollection`. Both involve object attributes that are expected but are not found, hinting at issues within the library's structure or its functionality. Furthermore, the structural similarity is clear as both reports deal with Matplotlib's handling of graphical elements and closely examine return values from object methods, making this candidate particularly insightful for diagnosing the current bug's root cause."
}