{
  "instance_id": "sphinx-doc__sphinx-8435",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-11-15T17:12:24Z",
  "problem_statement": "autodoc_type_aliases does not effect to variables and attributes\n**Describe the bug**\r\nautodoc_type_aliases does not effect to variables and attributes\r\n\r\n**To Reproduce**\r\n\r\n```\r\n# example.py\r\nfrom __future__ import annotations\r\n\r\n\r\n#: blah blah blah\r\nvar: String\r\n\r\n\r\nclass MyString:\r\n    \"mystring\"\r\n\r\n    #: blah blah blah\r\n    var: String\r\n```\r\n```\r\n# index.rst\r\n.. automodule:: example\r\n   :members:\r\n   :undoc-members:\r\n```\r\n```\r\n# conf.py\r\nautodoc_type_aliases = {\r\n    'String': 'example.MyString'\r\n}\r\n```\r\n\r\n**Expected behavior**\r\n`autodoc_type_aliases` should be applied to `example.var` and `example.MyString.var`.\r\n\r\n**Your project**\r\nN/A\r\n\r\n**Screenshots**\r\nN/A\r\n\r\n**Environment info**\r\n- OS: Mac\r\n- Python version: 3.9.0\r\n- Sphinx version: HEAD of 3.x branch\r\n- Sphinx extensions: sphinx.ext.autodoc\r\n- Extra tools: Nothing\r\n\r\n**Additional context**\r\nN/A\n",
  "patch": "diff --git a/sphinx/ext/autodoc/__init__.py b/sphinx/ext/autodoc/__init__.py\n--- a/sphinx/ext/autodoc/__init__.py\n+++ b/sphinx/ext/autodoc/__init__.py\n@@ -1702,7 +1702,8 @@ def add_directive_header(self, sig: str) -> None:\n         if not self.options.annotation:\n             # obtain annotation for this data\n             try:\n-                annotations = get_type_hints(self.parent)\n+                annotations = get_type_hints(self.parent, None,\n+                                             self.config.autodoc_type_aliases)\n             except NameError:\n                 # Failed to evaluate ForwardRef (maybe TYPE_CHECKING)\n                 annotations = safe_getattr(self.parent, '__annotations__', {})\n@@ -2093,7 +2094,8 @@ def add_directive_header(self, sig: str) -> None:\n         if not self.options.annotation:\n             # obtain type annotation for this attribute\n             try:\n-                annotations = get_type_hints(self.parent)\n+                annotations = get_type_hints(self.parent, None,\n+                                             self.config.autodoc_type_aliases)\n             except NameError:\n                 # Failed to evaluate ForwardRef (maybe TYPE_CHECKING)\n                 annotations = safe_getattr(self.parent, '__annotations__', {})\n",
  "similar_bug_items": [
    {
      "pr_number": 7515,
      "pr_title": "Close #2044: autodoc: Suppress default value for instance attributes",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #2044 \r\n- This also fixes uninitialized global variables.",
      "issue_id": 2044,
      "issue_title": "None by by default for instance attributes?",
      "issue_body": "When I'm included docstrings for instance attributes in the `__init__` of a class then using autoclass, I always end up with\n\n```\nattr = None\n    Documentation of attr\n```\n\ndue to this line\n\nhttps://github.com/sphinx-doc/sphinx/blob/add4c2467d9f98cadadff51593ca1727b4a81ca1/sphinx/ext/autodoc.py#L1465\n\nIs there a use case for documenting the `= None`. Can it just always be suppressed? There's no way right now to pass a `annotation` option to autoclass AFAICT.\n",
      "issue_closed_at": "2020-04-23T12:57:37Z",
      "base_commit": "12cb90c3fa5451e2c99100d8ce1d4d0a8bb7f527",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "identity",
          "class_name": null,
          "code": "def identity(x: Any) -> Any:\n    return x"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "add_directive_header",
          "class_name": "PropertyDocumenter",
          "code": "def add_directive_header(self, sig: str) -> None:\n        super().add_directive_header(sig)\n        sourcename = self.get_sourcename()\n        if inspect.isabstractmethod(self.object):\n            self.add_line('   :abstractmethod:', sourcename)\n        self.add_line('   :property:', sourcename)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "import_object",
          "class_name": "SlotsAttributeDocumenter",
          "code": "def import_object(self) -> Any:\n        \"\"\"Never import anything.\"\"\"\n        # disguise as an attribute\n        self.objtype = 'attribute'\n        self._datadescriptor = True\n\n        with mock(self.env.config.autodoc_mock_imports):\n            try:\n                ret = import_object(self.modname, self.objpath[:-1], 'class',\n                                    attrgetter=self.get_attr,\n                                    warningiserror=self.env.config.autodoc_warningiserror)\n                self.module, _, _, self.parent = ret\n                return True\n            except ImportError as exc:\n                logger.warning(exc.args[0], type='autodoc', subtype='import_object')\n                self.env.note_reread()\n                return False"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "add_directive_header",
          "class_name": "PropertyDocumenter",
          "code": "def add_directive_header(self, sig: str) -> None:\n        super().add_directive_header(sig)\n        sourcename = self.get_sourcename()\n        if inspect.isabstractmethod(self.object):\n            self.add_line('   :abstractmethod:', sourcename)\n        self.add_line('   :property:', sourcename)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "import_object",
          "class_name": "SlotsAttributeDocumenter",
          "code": "def import_object(self) -> Any:\n        \"\"\"Never import anything.\"\"\"\n        # disguise as an attribute\n        self.objtype = 'attribute'\n        self._datadescriptor = True\n\n        with mock(self.env.config.autodoc_mock_imports):\n            try:\n                ret = import_object(self.modname, self.objpath[:-1], 'class',\n                                    attrgetter=self.get_attr,\n                                    warningiserror=self.env.config.autodoc_warningiserror)\n                self.module, _, _, self.parent = ret\n                return True\n            except ImportError as exc:\n                logger.warning(exc.args[0], type='autodoc', subtype='import_object')\n                self.env.note_reread()\n                return False"
        }
      ]
    },
    {
      "pr_number": 5445,
      "pr_title": "Fix #5433: latex: ImportError: cannot import name 'DEFAULT_SETTINGS'",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #5433 \r\n",
      "issue_id": 5433,
      "issue_title": "ImportError: cannot import name 'DEFAULT_SETTINGS'",
      "issue_body": "Subject: ImportError: cannot import name 'DEFAULT_SETTINGS'\r\n\r\n### Problem\r\n\r\nWith the release of Sphinx 1.8.0, our builds have started to fail with the error message `ImportError: cannot import name 'DEFAULT_SETTINGS'`.\r\n\r\n#### Error logs / results\r\n\r\nSee:\r\nhttps://travis-ci.org/Pylons/pyramid/jobs/428634951#L497\r\n\r\nSee also https://github.com/Pylons/pyramid/issues/3350 and https://github.com/sphinx-doc/sphinx/issues/109",
      "issue_closed_at": "2018-09-18T14:33:46Z",
      "base_commit": "5d509d69edde6bfa7c2ed70a62762cc89f7c301e",
      "changes": [
        {
          "file": "sphinx/builders/latex/nodes.py",
          "type": "class",
          "name": "math_reference",
          "code": "class math_reference(nodes.Inline, nodes.Referential, nodes.TextElement):\n    \"\"\"A node for a reference for equation.\"\"\"\n    pass"
        },
        {
          "file": "sphinx/writers/latex.py",
          "type": "line",
          "name": "line 24",
          "code": "\nfrom sphinx import addnodes\nfrom sphinx import highlighting\nfrom sphinx.builders.latex.nodes import captioned_literal_block, footnotetext\nfrom sphinx.deprecation import RemovedInSphinx30Warning\nfrom sphinx.errors import SphinxError\nfrom sphinx.locale import admonitionlabels, _, __"
        },
        {
          "file": "sphinx/writers/latex.py",
          "type": "line",
          "name": "line 57",
          "code": "MAX_CITATION_LABEL_LENGTH = 8\nLATEXSECTIONNAMES = [\"part\", \"chapter\", \"section\", \"subsection\",\n                     \"subsubsection\", \"paragraph\", \"subparagraph\"]\nHYPERLINK_SUPPORT_NODES = (\n    nodes.figure,\n    nodes.literal_block,\n    nodes.table,\n    nodes.section,\n    captioned_literal_block,\n)\nENUMERATE_LIST_STYLE = defaultdict(lambda: r'\\arabic',\n                                   {\n                                       'arabic': r'\\arabic',"
        },
        {
          "file": "sphinx/writers/latex.py",
          "type": "function",
          "name": "check_latex_elements",
          "class_name": "LaTeXTranslator",
          "code": "def check_latex_elements(self):\n        # type: () -> None\n        warnings.warn('check_latex_elements() is deprecated.',\n                      RemovedInSphinx30Warning)\n\n        for key in self.builder.config.latex_elements:\n            if key not in self.elements:\n                msg = __(\"Unknown configure key: latex_elements[%r] is ignored.\")\n                logger.warning(msg % key)"
        }
      ]
    },
    {
      "pr_number": 6869,
      "pr_title": "Fix #6867: text: extra spaces are inserted to hyphenated words on folding lines",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #6867 \r\n- Not to call needless `do_format()` twice, I changed the rule for admonition. After this change, it starts after a blank line.\r\n\r\nBefore:\r\n```\r\nSee also: blah blah blah ...\r\n  blah blah blah ...\r\n```\r\nAfter:\r\n```\r\nSee also:\r\n\r\n  blah blah blah ...\r\n  blah blah blah ...\r\n```",
      "issue_id": 6867,
      "issue_title": "Text writer breaks on hyphens",
      "issue_body": "The text writer uses a custom `TextWrapper` class that breaks hyphenated words. The [parent class][1] from Python standard library has a [`break_on_hyphens`][2] option that allows to disable that, however Sphinx\u2019 `TextWrapper` has a custom `wordsep_re` regular expression so it is not easy to override that:\r\nhttps://github.com/sphinx-doc/sphinx/blob/8c7faed6fcbc6b7d40f497698cb80fc10aee1ab3/sphinx/writers/text.py#L259-L263\r\n\r\nThis leads to two issues.\r\n\r\n1. Some technical terms, program names, email addresses, etc., are being cut at their hypen at the end of line. This makes copy&paste more difficult, and it reads confusingly.\r\n\r\n2. The second issue is related to footnotes. When processing a footnote reference, the `add_text` method is called with `first` argument being `'[N]'`, where N is the footnote number.\r\n\r\n   When `first` is not None, the text is wrapped:\r\n   https://github.com/sphinx-doc/sphinx/blob/8c7faed6fcbc6b7d40f497698cb80fc10aee1ab3/sphinx/writers/text.py#L451\r\n\r\n   then joined by space and wrapped again:\r\n   https://github.com/sphinx-doc/sphinx/blob/8c7faed6fcbc6b7d40f497698cb80fc10aee1ab3/sphinx/writers/text.py#L456-L457\r\n\r\n   Because of this, the words that had a hyphen get an extra space after the hyphen.\r\n\r\n[1]: https://docs.python.org/3/library/textwrap.html#textwrap.TextWrapper\r\n[2]: https://docs.python.org/3/library/textwrap.html#textwrap.TextWrapper.break_on_hyphens\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n```bash\r\n$ cat >index.rst <<EOF\r\nPlease see the footnote. [#]_\r\n\r\n.. [#]\r\n   Another common way to do this is for ``build`` to depend on\r\n   ``build-stamp`` and to do nothing else, and for the ``build-stamp``\r\n   target to do the building and to ``touch build-stamp`` on completion.\r\nEOF\r\n$ touch conf.py\r\n$ sphinx-build -b text . _build/text\r\n```\r\n\r\nThe result is:\r\n```\r\n$ cat _build/text/index.txt \r\nPlease see the footnote. [1]\r\n\r\n[1] Another common way to do this is for \"build\" to depend on\r\n    \"build- stamp\" and to do nothing else, and for the \"build-stamp\"\r\n    target to do the building and to \"touch build-stamp\" on\r\n    completion.\r\n```\r\n\r\n**Expected behavior**\r\nIn the second line of footnote there should be no space between `build-` and `stamp`.\r\n\r\nI see two possible solutions to achieve that:\r\n- Do not break words on hyphens, or make that configurable.\r\n- Refactor the `end_state` method to avoid calling `do_format()` twice.\r\n\r\n**Environment info**\r\n- OS: Debian GNU/Linux sid\r\n- Python version: 3.7.5\r\n- Sphinx version: latest master\r\n\r\nThis issue is based on two bugs in Debian: [#944330] and [#944331].\r\n\r\n[#944330]: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=944330\r\n[#944331]: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=944331",
      "issue_closed_at": "2020-02-09T06:20:47Z",
      "base_commit": "cf37af27cfef1b620d7d3cd981184ddaedc4922a",
      "changes": [
        {
          "file": "sphinx/writers/text.py",
          "type": "function",
          "name": "do_format",
          "class_name": "TextTranslator",
          "code": "def do_format() -> None:\n            if not toformat:\n                return\n            if wrap:\n                res = my_wrap(''.join(toformat), width=MAXWIDTH - maxindent)\n            else:\n                res = ''.join(toformat).splitlines()\n            if end:\n                res += end\n            result.append((indent, res))"
        },
        {
          "file": "sphinx/writers/text.py",
          "type": "function",
          "name": "depart_admonition",
          "class_name": "TextTranslator",
          "code": "def depart_admonition(self, node: Element) -> None:\n            self.end_state(first=admonitionlabels[name] + ': ')"
        }
      ]
    },
    {
      "pr_number": 6536,
      "pr_title": "Close #6475: Add override argument to app.add_autodocumenter()",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- refs: #6475 ",
      "issue_id": 6475,
      "issue_title": "add override keyword to add_autodocumenter",
      "issue_body": "Subject: <short purpose of this pull request>\r\n\r\n### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- Allow override for `add_directive` contained in `add_autodocumenter`\r\n\r\n### Detail\r\n- Add keyword arg `override`\r\n\r\n### Relates\r\n- n/a\r\n\r\n",
      "issue_closed_at": "2019-06-29T16:50:39Z",
      "base_commit": "afefeebb52d2ec1fffc48f1d0691eb3b135c8edf",
      "changes": [
        {
          "file": "sphinx/application.py",
          "type": "function",
          "name": "add_lexer",
          "class_name": "Sphinx",
          "code": "def add_lexer(self, alias, lexer):\n        # type: (str, Union[Lexer, Type[Lexer]]) -> None\n        \"\"\"Register a new lexer for source code.\n\n        Use *lexer* to highlight code blocks with the given language *alias*.\n\n        .. versionadded:: 0.6\n        .. versionchanged:: 2.1\n           Take a lexer class as an argument.  An instance of lexers are\n           still supported until Sphinx-3.x.\n        \"\"\"\n        logger.debug('[app] adding lexer: %r', (alias, lexer))\n        if isinstance(lexer, Lexer):\n            warnings.warn('app.add_lexer() API changed; '\n                          'Please give lexer class instead instance',\n                          RemovedInSphinx40Warning)\n            lexers[alias] = lexer\n        else:\n            lexer_classes[alias] = lexer"
        },
        {
          "file": "sphinx/application.py",
          "type": "function",
          "name": "add_autodocumenter",
          "class_name": "Sphinx",
          "code": "def add_autodocumenter(self, cls):\n        # type: (Any) -> None\n        \"\"\"Register a new documenter class for the autodoc extension.\n\n        Add *cls* as a new documenter class for the :mod:`sphinx.ext.autodoc`\n        extension.  It must be a subclass of\n        :class:`sphinx.ext.autodoc.Documenter`.  This allows to auto-document\n        new types of objects.  See the source of the autodoc module for\n        examples on how to subclass :class:`Documenter`.\n\n        .. todo:: Add real docs for Documenter and subclassing\n\n        .. versionadded:: 0.6\n        \"\"\"\n        logger.debug('[app] adding autodocumenter: %r', cls)\n        from sphinx.ext.autodoc.directive import AutodocDirective\n        self.registry.add_documenter(cls.objtype, cls)\n        self.add_directive('auto' + cls.objtype, AutodocDirective)"
        }
      ]
    },
    {
      "pr_number": 4396,
      "pr_title": "Fix #1922: Upper characters problem in French HTML Search",
      "pr_body": "refs: #1922 ",
      "issue_id": 1922,
      "issue_title": "Upper characters problem in French stemmer",
      "issue_body": "I've found a bug in the HTML / Javascript search engine with the French language. The bug is actually related to the indexing of the doctree. The HTML / Javascript search engine expects only lower characters in the index. When indexing, the English stemmer removes upper characters and replace them with lower ones (with .lower()). However, the French stemmer (in sphinx/search/fr.py) keeps the upper characters. Consequently, any word with an upper character in wrongly indexed in French.\nThis can easily be fixed by adding \".lower()\" in SearchFrench.stem().\n",
      "issue_closed_at": "2018-01-15T01:15:45Z",
      "base_commit": "79501767d8a3e7433847af924fa4b9ff5d872612",
      "changes": [
        {
          "file": "sphinx/search/fr.py",
          "type": "function",
          "name": "init",
          "class_name": "SearchFrench",
          "code": "def init(self, options):\n        self.stemmer = snowballstemmer.stemmer('french')"
        }
      ]
    }
  ]
}