{
  "Selected_candidate": {
    "pr_number": 5521,
    "pr_title": "Fix #5493: gettext: crashed with broken template",
    "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #5493 \r\n",
    "issue_id": 5493,
    "issue_title": "Encountered unknown tag 'end'",
    "issue_body": "### Problem\r\nFile \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 59, in fail\r\n    raise exc(msg, lineno, self.name, self.filename)\r\njinja2.exceptions.TemplateSyntaxError: Encountered unknown tag 'end'. Jinja was looking for the following tags: 'endfor' or 'else'. The innermost block that needs to be closed is 'for'.\r\n\r\n#### Procedure to reproduce the problem\r\n```\r\nmake gettext\r\n```\r\n\r\n#### Error logs / results\r\n```\r\n# Sphinx version: 1.8.1\r\n# Python version: 3.6.0 (CPython)\r\n# Docutils version: 0.14 \r\n# Jinja2 version: 2.10\r\n# Last messages:\r\n#   读取模板……[  4%] /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/alabaster/donate.html\r\n#   \r\n#   读取模板……[  6%] /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/alabaster/layout.html\r\n#   \r\n#   读取模板……[  8%] /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/alabaster/navigation.html\r\n#   \r\n#   读取模板……[ 10%] /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/alabaster/relations.html\r\n#   \r\n#   读取模板……[ 13%] /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/matplotlib/backends/web_backend/all_figures.html\r\n#   \r\n# Loaded extensions:\r\n#   sphinx.ext.mathjax (1.8.1) from /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/ext/mathjax.py\r\n#   alabaster (0.7.11) from /Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/alabaster/__init__.py\r\nTraceback (most recent call last):\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/cmd/build.py\", line 304, in build_main\r\n    app.build(args.force_all, filenames)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/application.py\", line 341, in build\r\n    self.builder.build_update()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/builders/__init__.py\", line 347, in build_update\r\n    len(to_build))\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/builders/gettext.py\", line 258, in build\r\n    self._extract_from_template()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/sphinx/builders/gettext.py\", line 252, in _extract_from_template\r\n    for line, meth, msg in extract_translations(context):\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/ext.py\", line 214, in _extract\r\n    source = self.environment.parse(source)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/environment.py\", line 493, in parse\r\n    self.handle_exception(exc_info, source_hint=source)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/environment.py\", line 780, in handle_exception\r\n    reraise(exc_type, exc_value, tb)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/_compat.py\", line 37, in reraise\r\n    raise value.with_traceback(tb)\r\n  File \"<unknown>\", line 31, in template\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/environment.py\", line 497, in _parse\r\n    return Parser(self, source, name, encode_filename(filename)).parse()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 901, in parse\r\n    result = nodes.Template(self.subparse(), lineno=1)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 883, in subparse\r\n    rv = self.parse_statement()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 130, in parse_statement\r\n    return getattr(self, 'parse_' + self.stream.current.value)()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 199, in parse_for\r\n    body = self.parse_statements(('name:endfor', 'name:else'))\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 165, in parse_statements\r\n    result = self.subparse(end_tokens)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 883, in subparse\r\n    rv = self.parse_statement()\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 144, in parse_statement\r\n    self.fail_unknown_tag(token.value, token.lineno)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 97, in fail_unknown_tag\r\n    return self._fail_ut_eof(name, self._end_token_stack, lineno)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 90, in _fail_ut_eof\r\n    self.fail(' '.join(message), lineno)\r\n  File \"/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages/jinja2/parser.py\", line 59, in fail\r\n    raise exc(msg, lineno, self.name, self.filename)\r\njinja2.exceptions.TemplateSyntaxError: Encountered unknown tag 'end'. Jinja was looking for the following tags: 'endfor' or 'else'. The innermost block that needs to be closed is 'for'.\r\n```\r\n\r\n### Environment info\r\n- OS: Mac\r\n- Python version: 3.6.0 (CPython)\r\n- Sphinx version: 1.8.1\r\n",
    "issue_closed_at": "2018-10-12T13:38:02Z",
    "base_commit": "551ce9e759e6fc28d1631a12401955dcb15679ba",
    "changes": [
      {
        "file": "sphinx/builders/gettext.py",
        "type": "line",
        "name": "line 22",
        "code": "\nfrom sphinx.builders import Builder\nfrom sphinx.domains.python import pairindextypes\nfrom sphinx.locale import __\nfrom sphinx.util import split_index_msg, logging, status_iterator\nfrom sphinx.util.console import bold  # type: ignore"
      },
      {
        "file": "sphinx/builders/gettext.py",
        "type": "function",
        "name": "_extract_from_template",
        "class_name": "MessageCatalogBuilder",
        "code": "def _extract_from_template(self):\n        # type: () -> None\n        files = list(self._collect_templates())\n        files.sort()\n        logger.info(bold(__('building [%s]: ') % self.name), nonl=1)\n        logger.info(__('targets for %d template files'), len(files))\n\n        extract_translations = self.templates.environment.extract_translations\n\n        for template in status_iterator(files, __('reading templates... '), \"purple\",  # type: ignore  # NOQA\n                                        len(files), self.app.verbosity):\n            with open(template, 'r', encoding='utf-8') as f:  # type: ignore\n                context = f.read()\n            for line, meth, msg in extract_translations(context):\n                origin = MsgOrigin(template, line)\n                self.catalogs['sphinx'].add(msg, origin)"
      }
    ]
  },
  "Justification": "Candidate B addresses template failures which directly relate to the current issue of incorrect member listing in generated outputs. The shared components of Jinja2 parsing and potential similar failures in how members are processed in templates will likely provide valuable insights. Understanding the errors and fixes in Candidate B could illuminate potential issues with handling the `members` variable in the CURRENT bug, making it particularly helpful.",
  "instance_id": "sphinx-doc__sphinx-7686",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-05-17T14:09:10Z",
  "problem_statement": "autosummary: The members variable for module template contains imported members\n**Describe the bug**\r\nautosummary: The members variable for module template contains imported members even if autosummary_imported_members is False.\r\n\r\n**To Reproduce**\r\n\r\n```\r\n# _templates/autosummary/module.rst\r\n{{ fullname | escape | underline }}\r\n\r\n.. automodule:: {{ fullname }}\r\n\r\n   .. autosummary::\r\n   {% for item in members %}\r\n      {{ item }}\r\n   {%- endfor %}\r\n\r\n```\r\n```\r\n# example.py\r\nimport os\r\n```\r\n```\r\n# index.rst\r\n.. autosummary::\r\n   :toctree: generated\r\n\r\n   example\r\n```\r\n```\r\n# conf.py\r\nautosummary_generate = True\r\nautosummary_imported_members = False\r\n```\r\n\r\nAs a result, I got following output:\r\n```\r\n# generated/example.rst\r\nexample\r\n=======\r\n\r\n.. automodule:: example\r\n\r\n   .. autosummary::\r\n\r\n      __builtins__\r\n      __cached__\r\n      __doc__\r\n      __file__\r\n      __loader__\r\n      __name__\r\n      __package__\r\n      __spec__\r\n      os\r\n```\r\n\r\n**Expected behavior**\r\nThe template variable `members` should not contain imported members when `autosummary_imported_members` is False.\r\n\r\n**Your project**\r\nNo\r\n\r\n**Screenshots**\r\nNo\r\n\r\n**Environment info**\r\n- OS: Mac\r\n- Python version: 3.8.2\r\n- Sphinx version: 3.1.0dev\r\n- Sphinx extensions:  sphinx.ext.autosummary\r\n- Extra tools: No\r\n\r\n**Additional context**\r\nNo\r\n\n",
  "patch": "diff --git a/sphinx/ext/autosummary/generate.py b/sphinx/ext/autosummary/generate.py\n--- a/sphinx/ext/autosummary/generate.py\n+++ b/sphinx/ext/autosummary/generate.py\n@@ -18,6 +18,7 @@\n \"\"\"\n \n import argparse\n+import inspect\n import locale\n import os\n import pkgutil\n@@ -176,6 +177,56 @@ def render(self, template_name: str, context: Dict) -> str:\n # -- Generating output ---------------------------------------------------------\n \n \n+class ModuleScanner:\n+    def __init__(self, app: Any, obj: Any) -> None:\n+        self.app = app\n+        self.object = obj\n+\n+    def get_object_type(self, name: str, value: Any) -> str:\n+        return get_documenter(self.app, value, self.object).objtype\n+\n+    def is_skipped(self, name: str, value: Any, objtype: str) -> bool:\n+        try:\n+            return self.app.emit_firstresult('autodoc-skip-member', objtype,\n+                                             name, value, False, {})\n+        except Exception as exc:\n+            logger.warning(__('autosummary: failed to determine %r to be documented, '\n+                              'the following exception was raised:\\n%s'),\n+                           name, exc, type='autosummary')\n+            return False\n+\n+    def scan(self, imported_members: bool) -> List[str]:\n+        members = []\n+        for name in dir(self.object):\n+            try:\n+                value = safe_getattr(self.object, name)\n+            except AttributeError:\n+                value = None\n+\n+            objtype = self.get_object_type(name, value)\n+            if self.is_skipped(name, value, objtype):\n+                continue\n+\n+            try:\n+                if inspect.ismodule(value):\n+                    imported = True\n+                elif safe_getattr(value, '__module__') != self.object.__name__:\n+                    imported = True\n+                else:\n+                    imported = False\n+            except AttributeError:\n+                imported = False\n+\n+            if imported_members:\n+                # list all members up\n+                members.append(name)\n+            elif imported is False:\n+                # list not-imported members up\n+                members.append(name)\n+\n+        return members\n+\n+\n def generate_autosummary_content(name: str, obj: Any, parent: Any,\n                                  template: AutosummaryRenderer, template_name: str,\n                                  imported_members: bool, app: Any,\n@@ -246,7 +297,8 @@ def get_modules(obj: Any) -> Tuple[List[str], List[str]]:\n     ns.update(context)\n \n     if doc.objtype == 'module':\n-        ns['members'] = dir(obj)\n+        scanner = ModuleScanner(app, obj)\n+        ns['members'] = scanner.scan(imported_members)\n         ns['functions'], ns['all_functions'] = \\\n             get_members(obj, {'function'}, imported=imported_members)\n         ns['classes'], ns['all_classes'] = \\\n"
}