{
  "Selected_candidate": {
    "pr_number": 7501,
    "pr_title": "Fix #7418: std domain: term role role could not match case-insensitively",
    "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7418 ",
    "issue_id": 7418,
    "issue_title": "glossary duplicate term with a different case",
    "issue_body": "**Describe the bug**\r\n```\r\nWarning, treated as error:\r\ndoc/glossary.rst:243:duplicate term description of mysql, other instance in glossary\r\n```\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n[.travis.yml#L168](https://github.com/phpmyadmin/phpmyadmin/blob/f7cc383674b7099190771b1db510c62bfbbf89a7/.travis.yml#L168)\r\n```\r\n$ git clone --depth 1 https://github.com/phpmyadmin/phpmyadmin.git\r\n$ cd doc\r\n$ pip install 'Sphinx'\r\n$ make html\r\n```\r\n\r\n**Expected behavior**\r\nMySQL != mysql term right ?\r\n\r\n**Your project**\r\nhttps://github.com/phpmyadmin/phpmyadmin/blame/master/doc/glossary.rst#L234\r\n\r\n\r\n**Environment info**\r\n- OS: Unix\r\n- Python version: 3.6\r\n- Sphinx version: 3.0.0\r\n\r\n**Additional context**\r\nDid occur some hours ago, maybe you just released the version\r\n\r\n- https://travis-ci.org/github/williamdes/phpmyadmintest/jobs/671352365#L328\r\n\r\n",
    "issue_closed_at": "2020-04-09T15:12:11Z",
    "base_commit": "50d2d289e150cb429de15770bdd48a723de8c45d",
    "changes": [
      {
        "file": "sphinx/domains/std.py",
        "type": "function",
        "name": "resolve_xref",
        "class_name": "StandardDomain",
        "code": "def resolve_xref(self, env: \"BuildEnvironment\", fromdocname: str, builder: \"Builder\",\n                     typ: str, target: str, node: pending_xref, contnode: Element) -> Element:\n        if typ == 'ref':\n            resolver = self._resolve_ref_xref\n        elif typ == 'numref':\n            resolver = self._resolve_numref_xref\n        elif typ == 'keyword':\n            resolver = self._resolve_keyword_xref\n        elif typ == 'doc':\n            resolver = self._resolve_doc_xref\n        elif typ == 'option':\n            resolver = self._resolve_option_xref\n        elif typ == 'citation':\n            warnings.warn('pending_xref(domain=std, type=citation) is deprecated: %r' % node,\n                          RemovedInSphinx40Warning)\n            domain = env.get_domain('citation')\n            return domain.resolve_xref(env, fromdocname, builder, typ, target, node, contnode)\n        else:\n            resolver = self._resolve_obj_xref\n\n        return resolver(env, fromdocname, builder, typ, target, node, contnode)"
      },
      {
        "file": "sphinx/domains/std.py",
        "type": "function",
        "name": "_resolve_option_xref",
        "class_name": "StandardDomain",
        "code": "def _resolve_option_xref(self, env: \"BuildEnvironment\", fromdocname: str,\n                             builder: \"Builder\", typ: str, target: str,\n                             node: pending_xref, contnode: Element) -> Element:\n        progname = node.get('std:program')\n        target = target.strip()\n        docname, labelid = self.progoptions.get((progname, target), ('', ''))\n        if not docname:\n            commands = []\n            while ws_re.search(target):\n                subcommand, target = ws_re.split(target, 1)\n                commands.append(subcommand)\n                progname = \"-\".join(commands)\n\n                docname, labelid = self.progoptions.get((progname, target), ('', ''))\n                if docname:\n                    break\n            else:\n                return None\n\n        return make_refnode(builder, fromdocname, docname,\n                            labelid, contnode)"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant because it features a structural warning from Sphinx, similar to the warning in the CURRENT bug report regarding option syntax. Both reports deal with Sphinx's strict parsing rules, with Candidate A presenting an instance of how a specific format is incorrect, which resonates with the CURRENT bug's complaint of Sphinx not accepting certain directive syntax that previously worked. Both emphasize the challenge of maintaining documentation compatibility across Sphinx versions, making this candidate particularly useful for troubleshooting the CURRENT issue.",
  "instance_id": "sphinx-doc__sphinx-8506",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-11-28T17:28:05Z",
  "problem_statement": "Sphinx 3.2 complains about option:: syntax that earlier versions accepted\nSphinx 3.2 complains about use of the option:: directive that earlier versions accepted without complaint.\r\n\r\nThe QEMU documentation includes this:\r\n```\r\n.. option:: [enable=]PATTERN\r\n\r\n   Immediately enable events matching *PATTERN*\r\n```\r\n\r\nas part of the documentation of the command line options of one of its programs. Earlier versions of Sphinx were fine with this, but Sphinx 3.2 complains:\r\n\r\n```\r\nWarning, treated as error:\r\n../../docs/qemu-option-trace.rst.inc:4:Malformed option description '[enable=]PATTERN', should look like \"opt\", \"-opt args\", \"--opt args\", \"/opt args\" or \"+opt args\"\r\n```\r\n\r\nSphinx ideally shouldn't change in ways that break the building of documentation that worked in older versions, because this makes it unworkably difficult to have documentation that builds with whatever the Linux distro's sphinx-build is.\r\n\r\nThe error message suggests that Sphinx has a very restrictive idea of what option syntax is; it would be better if it just accepted any string, because not all programs and OSes have option syntax that matches the limited list the error message indicates.\r\n\n",
  "patch": "diff --git a/sphinx/domains/std.py b/sphinx/domains/std.py\n--- a/sphinx/domains/std.py\n+++ b/sphinx/domains/std.py\n@@ -43,7 +43,7 @@\n \n \n # RE for option descriptions\n-option_desc_re = re.compile(r'((?:/|--|-|\\+)?[^\\s=[]+)(=?\\s*.*)')\n+option_desc_re = re.compile(r'((?:/|--|-|\\+)?[^\\s=]+)(=?\\s*.*)')\n # RE for grammar tokens\n token_re = re.compile(r'`(\\w+)`', re.U)\n \n@@ -197,6 +197,11 @@ def handle_signature(self, sig: str, signode: desc_signature) -> str:\n                                location=signode)\n                 continue\n             optname, args = m.groups()\n+            if optname.endswith('[') and args.endswith(']'):\n+                # optional value surrounded by brackets (ex. foo[=bar])\n+                optname = optname[:-1]\n+                args = '[' + args\n+\n             if count:\n                 signode += addnodes.desc_addname(', ', ', ')\n             signode += addnodes.desc_name(optname, optname)\n"
}