{
  "instance_id": "sphinx-doc__sphinx-7975",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-07-18T06:39:32Z",
  "problem_statement": "Two sections called Symbols in index\nWhen using index entries with the following leading characters: _@_, _\u00a3_, and _\u2190_ I get two sections called _Symbols_ in the HTML output, the first containing all _@_ entries before \u201dnormal\u201d words and the second containing _\u00a3_ and _\u2190_ entries after the \u201dnormal\u201d words.  Both have the same anchor in HTML so the links at the top of the index page contain two _Symbols_ links, one before the letters and one after, but both lead to the first section.\n\n",
  "patch": "diff --git a/sphinx/environment/adapters/indexentries.py b/sphinx/environment/adapters/indexentries.py\n--- a/sphinx/environment/adapters/indexentries.py\n+++ b/sphinx/environment/adapters/indexentries.py\n@@ -98,9 +98,8 @@ def keyfunc0(entry: Tuple[str, str]) -> Tuple[bool, str]:\n             for subentry in indexentry[1].values():\n                 subentry[0].sort(key=keyfunc0)  # type: ignore\n \n-        # sort the index entries; put all symbols at the front, even those\n-        # following the letters in ASCII, this is where the chr(127) comes from\n-        def keyfunc(entry: Tuple[str, List]) -> Tuple[str, str]:\n+        # sort the index entries\n+        def keyfunc(entry: Tuple[str, List]) -> Tuple[Tuple[int, str], str]:\n             key, (void, void, category_key) = entry\n             if category_key:\n                 # using specified category key to sort\n@@ -108,11 +107,16 @@ def keyfunc(entry: Tuple[str, List]) -> Tuple[str, str]:\n             lckey = unicodedata.normalize('NFD', key.lower())\n             if lckey.startswith('\\N{RIGHT-TO-LEFT MARK}'):\n                 lckey = lckey[1:]\n+\n             if lckey[0:1].isalpha() or lckey.startswith('_'):\n-                lckey = chr(127) + lckey\n+                # put non-symbol characters at the folloing group (1)\n+                sortkey = (1, lckey)\n+            else:\n+                # put symbols at the front of the index (0)\n+                sortkey = (0, lckey)\n             # ensure a determinstic order *within* letters by also sorting on\n             # the entry itself\n-            return (lckey, entry[0])\n+            return (sortkey, entry[0])\n         newlist = sorted(new.items(), key=keyfunc)\n \n         if group_entries:\n",
  "similar_bug_items": [
    {
      "pr_number": 7226,
      "pr_title": "Fix #7220: genindex: \"main\" index entries are not shown at first",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- refs: #7220 ",
      "issue_id": 7220,
      "issue_title": "Make \"main\" index entries the first",
      "issue_body": "You can mark up \u201cmain\u201d index entries by prefixing them with an exclamation mark. The references to \u201cmain\u201d entries are emphasized in the generated index. But it does not change the order of entries, so the \"main\" index entry can be the last one.\r\n\r\nFor example, look at the index for \"import statement\" in the Python documentation. https://docs.python.org/3/genindex-I.html. There are 5 references. The last one is marked as \"main\".\r\n\r\nWould be nice if references to \"main\" entries be moved to the beginning of the list.\r\n\r\nThis issue was reported on the Python bug tracker: https://bugs.python.org/issue22714.",
      "issue_closed_at": "2020-02-29T17:14:20Z",
      "base_commit": "201455900a5fe938a1aadb4af37a2b13aeaa44bd",
      "changes": [
        {
          "file": "sphinx/environment/adapters/indexentries.py",
          "type": "line",
          "name": "line 7",
          "code": "    :copyright: Copyright 2007-2020 by the Sphinx team, see AUTHORS.\n    :license: BSD, see LICENSE for details.\n\"\"\"\nimport bisect\nimport re\nimport unicodedata\nfrom itertools import groupby"
        },
        {
          "file": "sphinx/environment/adapters/indexentries.py",
          "type": "function",
          "name": "add_entry",
          "class_name": "IndexEntries",
          "code": "def add_entry(word: str, subword: str, main: str, link: bool = True,\n                      dic: Dict = new, key: str = None) -> None:\n            # Force the word to be unicode if it's a ASCII bytestring.\n            # This will solve problems with unicode normalization later.\n            # For instance the RFC role will add bytestrings at the moment\n            word = str(word)\n            entry = dic.get(word)\n            if not entry:\n                dic[word] = entry = [[], {}, key]\n            if subword:\n                add_entry(subword, '', main, link=link, dic=entry[1], key=key)\n            elif link:\n                try:\n                    uri = builder.get_relative_uri('genindex', fn) + '#' + tid\n                except NoUri:\n                    pass\n                else:\n                    # maintain links in sorted/deterministic order\n                    bisect.insort(entry[0], (main, uri))"
        },
        {
          "file": "sphinx/environment/adapters/indexentries.py",
          "type": "function",
          "name": "add_entry",
          "class_name": "IndexEntries",
          "code": "def add_entry(word: str, subword: str, main: str, link: bool = True,\n                      dic: Dict = new, key: str = None) -> None:\n            # Force the word to be unicode if it's a ASCII bytestring.\n            # This will solve problems with unicode normalization later.\n            # For instance the RFC role will add bytestrings at the moment\n            word = str(word)\n            entry = dic.get(word)\n            if not entry:\n                dic[word] = entry = [[], {}, key]\n            if subword:\n                add_entry(subword, '', main, link=link, dic=entry[1], key=key)\n            elif link:\n                try:\n                    uri = builder.get_relative_uri('genindex', fn) + '#' + tid\n                except NoUri:\n                    pass\n                else:\n                    # maintain links in sorted/deterministic order\n                    bisect.insort(entry[0], (main, uri))"
        }
      ]
    },
    {
      "pr_number": 7277,
      "pr_title": "Fix #7276: cpp: objects generate hypertarget names unexpectedly",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n- Refactoring\r\n\r\n### Purpose\r\n- refs: #7276 \r\n- I think this would not be an expected behavior. So I'm going to remove `signode['names']` assignment. Is my understand correct?\r\n- I also thought to change node_id generation mechanism to use `make_id()`. But cpp domain is too complicated to me. And it seems it already has its own ID generation engine. So it does not cause a problem like #6903. So I'd like to keep as is now.\r\n\r\n@jakobandersen Could you review this and give an advice to me.",
      "issue_id": 7276,
      "issue_title": "cpp: objects generate hypertarget names unexpectedly",
      "issue_body": "**Describe the bug**\r\nThe C++ objects generate hyperlink names internally. As a result, it conflicts with normal hyperlink target.\r\n\r\n**To Reproduce**\r\nInput:\r\n```\r\n.. _say_hello:\r\n\r\n.. cpp:function:: say_hello()\r\n```\r\nwarnings:\r\n```\r\n/Users/tkomiya/work/tmp/doc/index.rst:14: WARNING: Duplicate explicit target name: \"say_hello\".\r\n```\r\n\r\nAs a side effect, C++ objects can be refer via hyperlink reference syntax:\r\n```\r\n.. cpp:function:: say_hello()\r\n\r\nBoth :cpp:func:`say_hello` and say_hello_ can refer above definition\r\n```\r\n\r\n**Expected behavior**\r\nI suppose this is not unexpected behavior. So it should not define hyperlink names internally.\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.8.2\r\n- Sphinx version: 2.4.4\r\n- Sphinx extensions:  No\r\n- Extra tools: No\r\n",
      "issue_closed_at": "2020-03-15T05:13:09Z",
      "base_commit": "2b6f06b6228e35564ea789188ae4d47fad3f776e",
      "changes": [
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "add_target_and_index",
          "class_name": "CPPObject",
          "code": "def add_target_and_index(self, ast: ASTDeclaration, sig: str, signode: desc_signature\n                             ) -> None:\n        # general note: name must be lstrip(':')'ed, to remove \"::\"\n        ids = []\n        for i in range(1, _max_id + 1):\n            try:\n                id = ast.get_id(version=i)\n                ids.append(id)\n            except NoOldIdError:\n                assert i < _max_id\n        # let's keep the newest first\n        ids = list(reversed(ids))\n        newestId = ids[0]\n        assert newestId  # shouldn't be None\n        if not re.compile(r'^[a-zA-Z0-9_]*$').match(newestId):\n            self.warn('Index id generation for C++ object \"%s\" failed, please '\n                      'report as bug (id=%s).' % (ast, newestId))\n\n        name = ast.symbol.get_full_nested_name().get_display_string().lstrip(':')\n        # Add index entry, but not if it's a declaration inside a concept\n        isInConcept = False\n        s = ast.symbol.parent\n        while s is not None:\n            decl = s.declaration\n            s = s.parent\n            if decl is None:\n                continue\n            if decl.objectType == 'concept':\n                isInConcept = True\n                break\n        if not isInConcept:\n            strippedName = name\n            for prefix in self.env.config.cpp_index_common_prefix:\n                if name.startswith(prefix):\n                    strippedName = strippedName[len(prefix):]\n                    break\n            indexText = self.get_index_text(strippedName)\n            self.indexnode['entries'].append(('single', indexText, newestId, '', None))\n\n        if newestId not in self.state.document.ids:\n            # if the name is not unique, the first one will win\n            names = self.env.domaindata['cpp']['names']\n            if name not in names:\n                names[name] = ast.symbol.docname\n                signode['names'].append(name)\n            else:\n                # print(\"[CPP] non-unique name:\", name)\n                pass\n            # always add the newest id\n            assert newestId\n            signode['ids'].append(newestId)\n            # only add compatibility ids when there are no conflicts\n            for id in ids[1:]:\n                if not id:  # is None when the element didn't exist in that version\n                    continue\n                if id not in self.state.document.ids:\n                    signode['ids'].append(id)\n            self.state.document.note_explicit_target(signode)"
        }
      ]
    },
    {
      "pr_number": 3154,
      "pr_title": "fix search word splitter in JavaScript",
      "pr_body": "Now I made code generator and it works as same as Python.\r\nIt fix #3150",
      "issue_id": 3150,
      "issue_title": "BugFix to support CJK search",
      "issue_body": "https://github.com/sphinx-doc/sphinx/commit/4ff5cbfaf920700382f809abe5def23e07bd0553\r\n\r\nThis change will break the CJK search. For Chinese:\r\n\r\n```JavaScript\r\nvar query = \"Hello from \u4e2d\u56fd \u4e0a\u6d77\";\r\nvar tmp = query.split(/\\W+/);\r\n// tmp: [ 'Hello', 'from', '' ];\r\n```\r\nFormer one:\r\n```JavaScript\r\nvar query = \"Hello from \u4e2d\u56fd \u4e0a\u6d77\";\r\nvar tmp = query.split(/\\s+/);\r\n// tmp: [ 'Hello', 'from', '\u4e2d\u56fd', '\u4e0a\u6d77' ]\r\n```\r\n\r\nI suggest to revert the commit.\r\n",
      "issue_closed_at": "2016-11-19T15:05:39Z",
      "base_commit": "4583c4b0227e425bca7b3c26b80d904a13a9c714",
      "changes": [
        {
          "file": "sphinx/search/__init__.py",
          "type": "line",
          "name": "line 17",
          "code": "\nfrom sphinx.util import jsdump, rpartition\nfrom sphinx.util.pycompat import htmlescape\n\n\nclass SearchLanguage(object):\n    \"\"\""
        },
        {
          "file": "sphinx/search/__init__.py",
          "type": "function",
          "name": "__init__",
          "class_name": "IndexBuilder",
          "code": "def __init__(self, env, lang, options, scoring):\n        self.env = env\n        # filename -> title\n        self._titles = {}\n        # stemmed word -> set(filenames)\n        self._mapping = {}\n        # stemmed words in titles -> set(filenames)\n        self._title_mapping = {}\n        # word -> stemmed word\n        self._stem_cache = {}\n        # objtype -> index\n        self._objtypes = {}\n        # objtype index -> (domain, type, objname (localized))\n        self._objnames = {}\n        # add language-specific SearchLanguage instance\n        lang_class = languages.get(lang)\n        if lang_class is None:\n            self.lang = SearchEnglish(options)\n        elif isinstance(lang_class, str):\n            module, classname = lang_class.rsplit('.', 1)\n            lang_class = getattr(__import__(module, None, None, [classname]),\n                                 classname)\n            self.lang = lang_class(options)\n        else:\n            # it's directly a class (e.g. added by app.add_search_language)\n            self.lang = lang_class(options)\n\n        if scoring:\n            with open(scoring, 'rb') as fp:\n                self.js_scorer_code = fp.read().decode('utf-8')\n        else:\n            self.js_scorer_code = u''"
        },
        {
          "file": "sphinx/search/__init__.py",
          "type": "function",
          "name": "context_for_searchtool",
          "class_name": "IndexBuilder",
          "code": "def context_for_searchtool(self):\n        return dict(\n            search_language_stemming_code = self.lang.js_stemmer_code,\n            search_language_stop_words = jsdump.dumps(sorted(self.lang.stopwords)),\n            search_scorer_tool = self.js_scorer_code,\n        )"
        }
      ]
    },
    {
      "pr_number": 7823,
      "pr_title": "Fix #7821: autodoc: TypeError is raised for overloaded C-ext function",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7821 ",
      "issue_id": 7821,
      "issue_title": "TypeError: can't set attributes of built-in/extension type 'list'",
      "issue_body": "Previous issue: https://github.com/sphinx-doc/sphinx/issues/7791\r\n\r\n**Describe the bug**\r\nRunning sphinx-build after sphinx-apidoc on our codebase results in the error message:\r\n\r\n```\r\n  File \"/home/danielk/.local/lib/python3.7/site-packages/sphinx/ext/autodoc/__init__.py\", line 1240, in annotate_to_first_argument\r\n    func.__signature__ = sig.replace(parameters=params)  # type: ignore\r\nTypeError: can't set attributes of built-in/extension type 'list'\r\n```\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nRun `make html` in project folder\r\n\r\n**Expected behavior**\r\nNo error\r\n\r\n**Your project**\r\n\r\nn/a\r\n\r\n**Screenshots**\r\nIf applicable, add screenshots to help explain your problem.\r\n\r\n**Environment info**\r\n- OS: Gentoo Linux, KDE 19.12.3, Plasma 5.70\r\n- Python version: 3.7.7\r\n- Sphinx version: See below\r\n- Sphinx extensions: See below\r\n- Extra tools: n/a\r\n\r\n```\r\nalabaster                          0.7.12            \r\nautodoc                            0.5.0             \r\nBabel                              2.8.0             \r\nbeautifulsoup4                     4.9.1             \r\nBrotli                             1.0.7             \r\nbsddb3                             6.2.7             \r\ncertifi                            2020.4.5.1        \r\ncffi                               1.14.0            \r\nchardet                            3.0.4             \r\ncracklib                           2.9.7             \r\ncryptacular                        1.4.1             /home/danielk/Development/cryptacular/src/cryptacular\r\ncryptography                       2.8               \r\nCython                             0.29.15           \r\ndecorator                          4.4.2             \r\ndistro                             1.0.4             \r\ndocutils                           0.16              \r\nextras                             1.0.0             \r\nfile-magic                         0.4.0             \r\nfixtures                           3.0.0             \r\nFormEncode                         1.3.1             \r\ngdbus-codegen                      2.62.6            \r\ngemato                             14.3              \r\ngentoolkit                         0.4.8             \r\ngpg                                1.13.0            \r\ngpodder                            3.10.5            \r\nhtml5lib                           1.0.1             \r\nidna                               2.8               \r\nimagesize                          1.1.0             \r\nisodate                            0.6.0             \r\nisort                              4.3.15            \r\njava-config                        2.2.0             \r\nJinja2                             2.11.1            \r\nlayman                             2.4.3             \r\nlensfun                            0.3.2             \r\nlibsass                            0.20.0            \r\nlinecache2                         1.0.0             \r\nlxml                               4.5.0             \r\nM2Crypto                           0.31.0            \r\nMako                               1.1.2             \r\nMarkdown                           2.4.1             \r\nMarkups                            3.0.0             \r\nMarkupSafe                         1.1.1             \r\nmercurial                          5.3.2             \r\nmeson                              0.52.1            \r\nmimeparse                          1.6.0             \r\nmock                               4.0.2             \r\nMomoko                             2.2.5.1           \r\nmygpoclient                        1.8               \r\nnetlink                            1.0               \r\nnotify2                            0.3.1             \r\nnumpy                              1.17.4            \r\nolefile                            0.46              \r\npackaging                          20.3              \r\npbkdf2                             1.3               \r\npbr                                5.1.1             \r\npdoc                               0.3.2             \r\nPillow                             7.0.0             \r\npip                                20.0.2            \r\nply                                3.11              \r\npodcastparser                      0.6.4             \r\nportage                            2.3.99            \r\npsycopg2                           2.8.5             \r\npsycopg2-binary                    2.8.5             \r\npwquality                          1.4.2             \r\npycairo                            1.18.2            \r\nPycco                              0.6.0             \r\npycountry                          19.8.18           \r\npycparser                          2.20              \r\npycryptodome                       3.9.4             \r\npyelftools                         0.25              \r\npyenchant                          2.0.0             \r\nPygments                           2.5.2             \r\nPyGObject                          3.34.0            \r\npyOpenSSL                          19.1.0            \r\npyparsing                          2.4.6             \r\npyportmidi                         0.0.7             \r\nPyQt5                              5.14.1            \r\nPyQt5-sip                          4.19.21           \r\nPyQtWebEngine                      5.14.0            \r\npyrsistent                         0.15.6            \r\nPySocks                            1.7.1             \r\npystache                           0.5.4             \r\npython-markdown-math               0.6               \r\npython-mimeparse                   1.6.0             \r\npython-subunit                     1.2.0             \r\npytz                               2019.3            \r\npyxdg                              0.26              \r\nPyYAML                             5.3.1             \r\nrdflib                             4.2.2             \r\nreportlab                          3.5.13            \r\nrequests                           2.23.0            \r\nReText                             7.0.4             \r\nscour                              0.37              \r\nsetuptools                         44.1.0            \r\nsimplejson                         3.17.0            \r\nsip                                4.19.21           \r\nsix                                1.14.0            \r\nsmartypants                        2.0.1             \r\nsnowballstemmer                    2.0.0             \r\nsoupsieve                          2.0.1             \r\nSphinx                             3.1.0             \r\nsphinx-rtd-theme                   0.4.3             \r\nsphinx-rtd-theme-http              1.0.0             \r\nsphinxcontrib-apidoc               0.3.0             \r\nsphinxcontrib-applehelp            1.0.2             \r\nsphinxcontrib-autodoc-doxygen      0.6.0             \r\nsphinxcontrib-autodoc-filterparams 0.0.1             \r\nsphinxcontrib-devhelp              1.0.2             \r\nsphinxcontrib-htmlhelp             1.0.3             \r\nsphinxcontrib-httpdomain           1.7.0             \r\nsphinxcontrib-jsmath               1.0.1             \r\nsphinxcontrib-mockautodoc          0.0.1.dev20130518 \r\nsphinxcontrib-qthelp               1.0.3             \r\nsphinxcontrib-serializinghtml      1.1.3             \r\nssl-fetch                          0.4               \r\ntesttools                          2.3.0             \r\ntornado                            5.1.1             \r\ntraceback2                         1.4.0             \r\ntyped-ast                          1.4.1             \r\nunittest2                          1.1.0             \r\nurllib3                            1.25.8            \r\nwaitress                           1.4.4             \r\nwebencodings                       0.5.1             \r\nWebOb                              1.8.6             \r\nWebTest                            2.0.35            \r\nyoutube-dl                         2020.5.29         \r\nzstandard                          0.13.0 \r\n```\r\n\r\n\r\n\r\n**Additional context**\r\n\r\n```\r\n\r\nRunning Sphinx v3.1.0\r\nbuilding [mo]: targets for 0 po files that are out of date\r\nbuilding [html]: targets for 41 source files that are out of date\r\nupdating environment: [new config] 41 added, 0 changed, 0 removed\r\nreading sources... [ 87%] ... Exception occurred:\r\n  File \"/home/danielk/.local/lib/python3.7/site-packages/sphinx/ext/autodoc/__init__.py\", line 1240, in annotate_to_first_argument\r\n    func.__signature__ = sig.replace(parameters=params)  # type: ignore\r\nTypeError: can't set attributes of built-in/extension type 'list'\r\nThe full traceback has been saved in /tmp/sphinx-err-lf0wsvvp.log\r\n\r\n, if you want to report the issue to the developers.\r\nPlease also report this if it was a user error, so that a better error message can be provided next time.\r\nA bug report can be filed in the tracker at <https://github.com/sphinx-doc/sphinx/issues>. Thanks!\r\nmake[1]: *** [Makefile:20: html] Error 2\r\nmake[1]: Leaving directory '/home/danielk/Development/docs'\r\nmake: *** [Makefile:85: docs] Error 2\r\n```\r\n\r\n[sphinx-err-lf0wsvvp.log](https://github.com/sphinx-doc/sphinx/files/4766780/sphinx-err-lf0wsvvp.log)",
      "issue_closed_at": "2020-06-14T02:48:20Z",
      "base_commit": "46f79c55e1242ae8bb49b14c0a9e226d119460e5",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "annotate_to_first_argument",
          "class_name": "MethodDocumenter",
          "code": "def annotate_to_first_argument(self, func: Callable, typ: Type) -> None:\n        \"\"\"Annotate type hint to the first argument of function if needed.\"\"\"\n        try:\n            sig = inspect.signature(func)\n        except TypeError as exc:\n            logger.warning(__(\"Failed to get a method signature for %s: %s\"),\n                           self.fullname, exc)\n            return\n        except ValueError:\n            return\n        if len(sig.parameters) == 1:\n            return\n\n        params = list(sig.parameters.values())\n        if params[1].annotation is Parameter.empty:\n            params[1] = params[1].replace(annotation=typ)\n            func.__signature__ = sig.replace(parameters=params)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "annotate_to_first_argument",
          "class_name": "MethodDocumenter",
          "code": "def annotate_to_first_argument(self, func: Callable, typ: Type) -> None:\n        \"\"\"Annotate type hint to the first argument of function if needed.\"\"\"\n        try:\n            sig = inspect.signature(func)\n        except TypeError as exc:\n            logger.warning(__(\"Failed to get a method signature for %s: %s\"),\n                           self.fullname, exc)\n            return\n        except ValueError:\n            return\n        if len(sig.parameters) == 1:\n            return\n\n        params = list(sig.parameters.values())\n        if params[1].annotation is Parameter.empty:\n            params[1] = params[1].replace(annotation=typ)\n            func.__signature__ = sig.replace(parameters=params)"
        }
      ]
    },
    {
      "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] + ': ')"
        }
      ]
    }
  ]
}