{
  "Selected_candidate": {
    "pr_number": 13103,
    "pr_title": "fixed cos.rewrite(sqrt) sometimes not fully rewriting",
    "pr_body": "See #13066\r\n`functions.elementary.trigonometric.cos._eval_rewrite_as_sqrt` would fail to fully rewrite `cos(pi/51)` on python 3.6 only. This turned out to be an issue with the `_fermatCoords` function which is supposed to factor 51 into fermat primes. Trigonometric expressions with these fermat primes as denominators of pi can be rewritten using squareroots.\r\n\r\nThe function uses `divmod` repeatedly to get the factors. `n` was set to the whole number result of the `divmod`. This is where the bug was, since that would mean that if `n` was not divisible by that particular factor, then `n` would still be changed to the whole number result, which usually ended up being 0, in which case the function would miss the other factors and return `False`. Or in rare cases it would have led to a false factorization.\r\n\r\nThe reason this bug only revealed itself in 3.6 is because the ordering of dicts was changed. Previously for n=51 it so happened that python ordered the dict with 17 first, then 3, which are both divisible.\r\n\r\n<!-- Please give this pull request a descriptive title. Pull requests with descriptive titles are more likely to receive reviews. Describe what you changed! A title that only references an issue number is not descriptive. -->\r\n\r\n<!-- If this pull request fixes an issue please indicate which issue by typing \"Fixes #NNNN\" below. -->\r\nFixes #13066",
    "issue_id": 13066,
    "issue_title": "cos(pi/51).rewrite(sqrt) does not fully rewrite the expression on python 3.6 (test failure)",
    "issue_body": "It currently returns `cos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2`, even though both `cos(6*pi/17)` and `sin(6*pi/17)` can be rewritten using `sqrt`s. The slow test `test_sincos_rewrite_sqrt` in test_trigonometric.py currently fails because it tests that `cos(pi/51).rewrite(sqrt)` returns the full expression in terms of `sqrt`s. I tested on python 3.4 64 bit and python 3.6 64 bit, and it only happens on 3.6, which is odd. I tried a fresh installation for 3.6 to be sure it wasn't anything I changed and it still happened.\r\n```\r\n>>> cos(pi/51).rewrite(sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n>>> cos(pi/51).rewrite(cos, sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n>>> cos(pi/51).rewrite([cos, sin], sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n```",
    "issue_closed_at": "2017-08-10T09:16:46Z",
    "base_commit": "8ff744d843f9dc78c560e61e3edc01f878d1960b",
    "changes": [
      {
        "file": "sympy/functions/elementary/trigonometric.py",
        "type": "function",
        "name": "_fermatCoords",
        "class_name": "cos",
        "code": "def _fermatCoords(n):\n            # if n can be factored in terms of Fermat primes with\n            # multiplicity of each being 1, return those primes, else\n            # False\n            from sympy import chebyshevt\n            primes = []\n            for p_i in cst_table_some:\n                n, r = divmod(n, p_i)\n                if not r:\n                    primes.append(p_i)\n                    if n == 1:\n                        return tuple(primes)\n            return False"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful because it deals with the `rewrite` functionality in Sympy, which shares a conceptual similarity with the `lambdify` function's handling of symbolic expressions. Both involve symbolic manipulation and accuracy in handling complex expressions, which is critical for understanding the failure related to specific symbol names like `MatrixSymbol` and `curly` symbols. The patch also provides insights into dealing with potential edge cases that resemble the current issue of name compatibility with `lambdify`. Additionally, Candidate E includes a fix for how rewriting is handled, which could inspire ways to manage the symbol handling issue in the current bug report effectively.",
  "instance_id": "sympy__sympy-15011",
  "repo": "sympy/sympy",
  "created_at": "2018-08-02T12:54:02Z",
  "problem_statement": "lambdify does not work with certain MatrixSymbol names even with dummify=True\n`lambdify` is happy with curly braces in a symbol name and with `MatrixSymbol`s, but not with both at the same time, even if `dummify` is `True`.\r\n\r\nHere is some basic code that gives the error.\r\n```\r\nimport sympy as sy\r\ncurlyx = sy.symbols(\"{x}\")\r\nv = sy.MatrixSymbol(\"v\", 2, 1)\r\ncurlyv = sy.MatrixSymbol(\"{v}\", 2, 1)\r\n```\r\n\r\nThe following two lines of code work:\r\n```\r\ncurlyScalarId = sy.lambdify(curlyx, curlyx)\r\nvectorId = sy.lambdify(v,v)\r\n```\r\n\r\nThe following two lines of code give a `SyntaxError`:\r\n```\r\ncurlyVectorId = sy.lambdify(curlyv, curlyv)\r\ncurlyVectorIdDummified = sy.lambdify(curlyv, curlyv, dummify=True)\r\n```\r\n\r\n\n",
  "patch": "diff --git a/sympy/utilities/lambdify.py b/sympy/utilities/lambdify.py\n--- a/sympy/utilities/lambdify.py\n+++ b/sympy/utilities/lambdify.py\n@@ -700,14 +700,13 @@ def _is_safe_ident(cls, ident):\n             return isinstance(ident, str) and cls._safe_ident_re.match(ident) \\\n                 and not (keyword.iskeyword(ident) or ident == 'None')\n \n-\n     def _preprocess(self, args, expr):\n         \"\"\"Preprocess args, expr to replace arguments that do not map\n         to valid Python identifiers.\n \n         Returns string form of args, and updated expr.\n         \"\"\"\n-        from sympy import Dummy, Symbol, Function, flatten\n+        from sympy import Dummy, Symbol, MatrixSymbol, Function, flatten\n         from sympy.matrices import DeferredVector\n \n         dummify = self._dummify\n@@ -725,7 +724,7 @@ def _preprocess(self, args, expr):\n                 argstrs.append(nested_argstrs)\n             elif isinstance(arg, DeferredVector):\n                 argstrs.append(str(arg))\n-            elif isinstance(arg, Symbol):\n+            elif isinstance(arg, Symbol) or isinstance(arg, MatrixSymbol):\n                 argrep = self._argrepr(arg)\n \n                 if dummify or not self._is_safe_ident(argrep):\n@@ -739,7 +738,14 @@ def _preprocess(self, args, expr):\n                 argstrs.append(self._argrepr(dummy))\n                 expr = self._subexpr(expr, {arg: dummy})\n             else:\n-                argstrs.append(str(arg))\n+                argrep = self._argrepr(arg)\n+\n+                if dummify:\n+                    dummy = Dummy()\n+                    argstrs.append(self._argrepr(dummy))\n+                    expr = self._subexpr(expr, {arg: dummy})\n+                else:\n+                    argstrs.append(str(arg))\n \n         return argstrs, expr\n \n"
}