{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 8627,
      "pr_title": "Bug Fix #8626",
      "pr_body": "Let me know if a test case is needed\n",
      "issue_id": 8626,
      "issue_title": "bug in physics/mechanics/lagrange.py",
      "issue_body": "When providing a `forcelist` with more than one force, not all terms are computed,\nthis comes from the fact that the iterator `flist = zip(*_f_list_parser(self.forcelist, N))` is consumed in the first iteration of the loop.\n\nI propose a following PR that fixes the bug, let me know if I need to also provide a test case.\n\nBest,\nGuillaume\n",
      "issue_closed_at": "2015-01-06T16:57:16Z",
      "base_commit": "65f7c8c2c9c1927eaa8520c4ce06864f93a20ad1",
      "changes": [
        {
          "file": "sympy/physics/mechanics/lagrange.py",
          "type": "function",
          "name": "form_lagranges_equations",
          "class_name": "LagrangesMethod",
          "code": "def form_lagranges_equations(self):\n        \"\"\"Method to form Lagrange's equations of motion.\n\n        Returns a vector of equations of motion using Lagrange's equations of\n        the second kind.\n        \"\"\"\n\n        qds = self._qdots\n        qdd_zero = dict((i, 0) for i in self._qdoubledots)\n        n = len(self.q)\n\n        # Internally we represent the EOM as four terms:\n        # EOM = term1 - term2 - term3 - term4 = 0\n\n        # First term\n        self._term1 = self._L.jacobian(qds)\n        self._term1 = self._term1.diff(dynamicsymbols._t).T\n\n        # Second term\n        self._term2 = self._L.jacobian(self.q).T\n\n        # Third term\n        if self.coneqs:\n            coneqs = self.coneqs\n            m = len(coneqs)\n            # Creating the multipliers\n            self.lam_vec = Matrix(dynamicsymbols('lam1:' + str(m + 1)))\n            self.lam_coeffs = -coneqs.jacobian(qds)\n            self._term3 = self.lam_coeffs.T * self.lam_vec\n            # Extracting the coeffecients of the qdds from the diff coneqs\n            diffconeqs = coneqs.diff(dynamicsymbols._t)\n            self._m_cd = diffconeqs.jacobian(self._qdoubledots)\n            # The remaining terms i.e. the 'forcing' terms in diff coneqs\n            self._f_cd = -diffconeqs.subs(qdd_zero)\n        else:\n            self._term3 = zeros(n, 1)\n\n        # Fourth term\n        if self.forcelist:\n            N = self.inertial\n            self._term4 = zeros(n, 1)\n            flist = zip(*_f_list_parser(self.forcelist, N))\n            for i, qd in enumerate(qds):\n                for obj, force in self.forcelist:\n                    self._term4[i] = sum(v.diff(qd, N) & f for (v, f) in flist)\n        else:\n            self._term4 = zeros(n, 1)\n\n        # Form the dynamic mass and forcing matrices\n        without_lam = self._term1 - self._term2 - self._term4\n        self._m_d = without_lam.jacobian(self._qdoubledots)\n        self._f_d = -without_lam.subs(qdd_zero)\n\n        # Form the EOM\n        self.eom = without_lam - self._term3\n        return self.eom"
        }
      ]
    },
    {
      "pr_number": 13437,
      "pr_title": "Fix bell(n) for n=oo, and negative/non-integer 'n'",
      "pr_body": "This PR uses commits of #9198 (maintaining ownership to original author). And tries to address the requested changes on PR #9198 .\r\nFixes #9184",
      "issue_id": 9184,
      "issue_title": "bell(n).limit(n, oo) should be oo rather than bell(oo)",
      "issue_body": "`bell(n).limit(n,oo)` should take the value infinity, but the current output is `bell(oo)`. As the Bell numbers represent the number of partitions of a set, it seems natural that `bell(oo)` should be able to be evaluated rather than be returned unevaluated. This issue is also in line with the recent fixes to the corresponding limit for the Fibonacci numbers and Lucas numbers.\n\n```\nfrom sympy import *\nn = symbols('n')\nbell(n).limit(n,oo)\n\nOutput:\nbell(oo)\n```\n\nI'm new to Sympy, so I'd appreciate the opportunity to fix this bug myself if that's alright.\n",
      "issue_closed_at": "2017-10-13T04:35:13Z",
      "base_commit": "674afc619d7f5c519b6a5393a8b0532a131e57e0",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/numbers.py",
          "type": "function",
          "name": "_bell_incomplete_poly",
          "class_name": "bell",
          "code": "def _bell_incomplete_poly(n, k, symbols):\n        r\"\"\"\n        The second kind of Bell polynomials (incomplete Bell polynomials).\n\n        Calculated by recurrence formula:\n\n        .. math:: B_{n,k}(x_1, x_2, \\dotsc, x_{n-k+1}) =\n                \\sum_{m=1}^{n-k+1}\n                \\x_m \\binom{n-1}{m-1} B_{n-m,k-1}(x_1, x_2, \\dotsc, x_{n-m-k})\n\n        where\n            B_{0,0} = 1;\n            B_{n,0} = 0; for n>=1\n            B_{0,k} = 0; for k>=1\n\n        \"\"\"\n        if (n == 0) and (k == 0):\n            return S.One\n        elif (n == 0) or (k == 0):\n            return S.Zero\n        s = S.Zero\n        a = S.One\n        for m in range(1, n - k + 2):\n            s += a * bell._bell_incomplete_poly(\n                n - m, k - 1, symbols) * symbols[m - 1]\n            a = a * (n - m) / m\n        return expand_mul(s)"
        }
      ]
    },
    {
      "pr_number": 11947,
      "pr_title": "aligned middle of matrix with baseline",
      "pr_body": "**Previous Output**: \r\n\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n2\u22c5\u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\n\r\n**Output after this PR**:\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n  \u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n2.\u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\nFixes #10472 . ",
      "issue_id": 10472,
      "issue_title": "pprint should align the middle of the matrix to the baseline?",
      "issue_body": "Consider the current behaviour, where operators align with the top of the matrix:\n\n```\n>>> Z = Matrix(5,5, lambda i,j: i+j)\n>>> pprint(MatMul(2,Z))\n2\u22c5\u23a10  1  2  3  4\u23a4\n  \u23a2             \u23a5\n  \u23a21  2  3  4  5\u23a5\n  \u23a2             \u23a5\n  \u23a22  3  4  5  6\u23a5\n  \u23a2             \u23a5\n  \u23a23  4  5  6  7\u23a5\n  \u23a2             \u23a5\n  \u23a34  5  6  7  8\u23a6\n>>> pprint(MatAdd(Z,Z))\n\u23a10  1  2  3  4\u23a4 + \u23a10  1  2  3  4\u23a4\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a21  2  3  4  5\u23a5   \u23a21  2  3  4  5\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a22  3  4  5  6\u23a5   \u23a22  3  4  5  6\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a23  4  5  6  7\u23a5   \u23a23  4  5  6  7\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a34  5  6  7  8\u23a6   \u23a34  5  6  7  8\u23a6\n```\n\nThis looks poor, but I figured it might be the desired behaviour...\n\nHere's the example that makes me think its wrong:\n\n```\n>>> pprint(sin((1+1/x)/(1+1/y)) + det(MatMul(2,Z)))\n   \u239b    1\u239e                       \n   \u239c1 + \u2500\u239f                       \n   \u239c    x\u239f                       \nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + 32\u22c5\u2502\u23a10  1  2  3  4\u23a4\u2502\n   \u239c    1\u239f      \u2502\u23a2             \u23a5\u2502\n   \u239c1 + \u2500\u239f      \u2502\u23a21  2  3  4  5\u23a5\u2502\n   \u239d    y\u23a0      \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a22  3  4  5  6\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a23  4  5  6  7\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a34  5  6  7  8\u23a6\u2502\n```\n\nHere's how I think it should work, using the proposed #10423:\n\n```\npprint(sin((1+1/x)/y) + Trace(Z))\n               \u239b\u23a10  1  2  3  4\u23a4\u239e\n   \u239b    1\u239e     \u239c\u23a2             \u23a5\u239f\n   \u239c1 + \u2500\u239f     \u239c\u23a21  2  3  4  5\u23a5\u239f\n   \u239c    x\u239f     \u239c\u23a2             \u23a5\u239f\nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + tr\u239c\u23a22  3  4  5  6\u23a5\u239f\n   \u239d  y  \u23a0     \u239c\u23a2             \u23a5\u239f\n               \u239c\u23a23  4  5  6  7\u23a5\u239f\n               \u239c\u23a2             \u23a5\u239f\n               \u239d\u23a34  5  6  7  8\u23a6\u23a0\n```\n\nwhere note the `sin` alignment is by-design, and (relevant to this bug) the `+` and `tr` operator line up with `sin`.\n\nAt least, det, MatAdd, MatMul, MatPow would need fixed, maybe others I haven't thought of.\n",
      "issue_closed_at": "2016-12-22T06:52:11Z",
      "base_commit": "9a724a42c033c1aae97064947a0f44ec3b922d73",
      "changes": [
        {
          "file": "sympy/matrices/expressions/matadd.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matadd):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatAdd, pprint\n    >>> from sympy.matrices.expressions.matadd import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = eye(2)\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatAdd(A, B, C)\n    >>> pprint(X)\n    A + [1  0] + [1  2]\n        [    ]   [    ]\n        [0  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    A + [2  2]\n        [    ]\n        [3  5]\n    \"\"\"\n    groups = sift(matadd.args, lambda arg: isinstance(arg, MatrixBase))\n    if len(groups[True]) > 1:\n        return MatAdd(*(groups[False] + [reduce(add, groups[True])]))\n    else:\n        return matadd"
        },
        {
          "file": "sympy/matrices/expressions/matmul.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matmul):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatMul, pprint\n    >>> from sympy.matrices.expressions.matmul import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = Matrix([[1, 1], [1, 1]])\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatMul(A, B, C)\n    >>> pprint(X)\n    A*[1  1]*[1  2]\n      [    ] [    ]\n      [1  1] [3  4]\n    >>> pprint(merge_explicit(X))\n    A*[4  6]\n      [    ]\n      [4  6]\n\n    >>> X = MatMul(B, A, C)\n    >>> pprint(X)\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    \"\"\"\n    if not any(isinstance(arg, MatrixBase) for arg in matmul.args):\n        return matmul\n    newargs = []\n    last = matmul.args[0]\n    for arg in matmul.args[1:]:\n        if isinstance(arg, (MatrixBase, Number)) and isinstance(last, (MatrixBase, Number)):\n            last = last * arg\n        else:\n            newargs.append(last)\n            last = arg\n    newargs.append(last)\n\n    return MatMul(*newargs)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_matrix_contents",
          "class_name": "PrettyPrinter",
          "code": "def _print_matrix_contents(self, e):\n        \"\"\"\n        This method factors out what is essentially grid printing.\n        \"\"\"\n        M = e   # matrix\n        Ms = {}  # i,j -> pretty(M[i,j])\n        for i in range(M.rows):\n            for j in range(M.cols):\n                Ms[i, j] = self._print(M[i, j])\n\n        # h- and v- spacers\n        hsep = 2\n        vsep = 1\n\n        # max width for columns\n        maxw = [-1] * M.cols\n\n        for j in range(M.cols):\n            maxw[j] = max([Ms[i, j].width() for i in range(M.rows)] or [0])\n\n        # drawing result\n        D = None\n\n        for i in range(M.rows):\n\n            D_row = None\n            for j in range(M.cols):\n                s = Ms[i, j]\n\n                # reshape s to maxw\n                # XXX this should be generalized, and go to stringPict.reshape ?\n                assert s.width() <= maxw[j]\n\n                # hcenter it, +0.5 to the right                        2\n                # ( it's better to align formula starts for say 0 and r )\n                # XXX this is not good in all cases -- maybe introduce vbaseline?\n                wdelta = maxw[j] - s.width()\n                wleft = wdelta // 2\n                wright = wdelta - wleft\n\n                s = prettyForm(*s.right(' '*wright))\n                s = prettyForm(*s.left(' '*wleft))\n\n                # we don't need vcenter cells -- this is automatically done in\n                # a pretty way because when their baselines are taking into\n                # account in .right()\n\n                if D_row is None:\n                    D_row = s   # first box in a row\n                    continue\n\n                D_row = prettyForm(*D_row.right(' '*hsep))  # h-spacer\n                D_row = prettyForm(*D_row.right(s))\n\n            if D is None:\n                D = D_row       # first row in a picture\n                continue\n\n            # v-spacer\n            for _ in range(vsep):\n                D = prettyForm(*D.below(' '))\n\n            D = prettyForm(*D.below(D_row))\n\n        if D is None:\n            D = prettyForm('')  # Empty Matrix\n\n        return D"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_MatrixElement",
          "class_name": "PrettyPrinter",
          "code": "def _print_MatrixElement(self, expr):\n        from sympy.matrices import MatrixSymbol\n        from sympy import Symbol\n        if (isinstance(expr.parent, MatrixSymbol)\n                and expr.i.is_number and expr.j.is_number):\n            return self._print(\n                    Symbol(expr.parent.name + '_%d%d'%(expr.i, expr.j)))\n        else:\n            prettyFunc = self._print(expr.parent)\n            prettyIndices = self._print_seq((expr.i, expr.j), delimiter=', '\n                    ).parens(left='[', right=']')[0]\n            pform = prettyForm(binding=prettyForm.FUNC,\n                    *stringPict.next(prettyFunc, prettyIndices))\n\n            # store pform parts so it can be reassembled e.g. when powered\n            pform.prettyFunc = prettyFunc\n            pform.prettyArgs = prettyIndices\n\n            return pform"
        }
      ]
    },
    {
      "pr_number": 13744,
      "pr_title": "group theory: correct FreeGroupElement multiplication",
      "pr_body": "At the moment, multiplication of `FreeGroupElement`s treats two elements as non-equal unless the symbols representing them are the same object. But it is sufficient for the symbols to be equal via `==`, and the current behaviour causes problems. For example, the added test fails in master.\r\n\r\nFixes #13737.",
      "issue_id": 13737,
      "issue_title": "coset_table doctest failure (Fatal Python error: Cannot recover from stack overflow.)",
      "issue_body": "See https://travis-ci.org/sympy/sympy/jobs/315588322\r\n\r\n```\r\nsympy/combinatorics/coset_table.py[3] \r\n.\r\nFatal Python error: Cannot recover from stack overflow.\r\nCurrent thread 0x00007f7fef7dd740 (most recent call first):\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/symbol.py\", line 257 in assumptions0\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/symbol.py\", line 252 in _hashable_content\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/basic.py\", line 331 in __eq__\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/cache.py\", line 93 in wrapper\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/expr.py\", line 111 in __neg__\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 435 in <listcomp>\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 435 in letter_form\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 927 in subword_index\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 964 in is_dependent\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 976 in is_independent\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 663 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 684 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  ...\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 4, in <module>\r\nException: Tests failed\r\n```\r\n\r\nHere is the test header\r\n\r\n```\r\n============================= test process starts ==============================\r\nexecutable:         /home/travis/virtualenv/python3.5.4/bin/python  (3.5.4-final-0) [CPython]\r\narchitecture:       64-bit\r\ncache:              yes\r\nground types:       python \r\nnumpy:              None\r\nhash randomization: on (PYTHONHASHSEED=935966445)\r\n```",
      "issue_closed_at": "2017-12-15T08:43:52Z",
      "base_commit": "ae3d6f647f080240318d114f2c397b99be5cef5f",
      "changes": [
        {
          "file": "sympy/combinatorics/free_groups.py",
          "type": "function",
          "name": "letter_form_to_array_form",
          "class_name": null,
          "code": "def letter_form_to_array_form(array_form, group):\n    \"\"\"\n    This method converts a list given with possible repetitions of elements in\n    it. It returns a new list such that repetitions of consecutive elements is\n    removed and replace with a tuple element of size two such that the first\n    index contains `value` and the second index contains the number of\n    consecutive repetitions of `value`.\n\n    \"\"\"\n    a = list(array_form[:])\n    new_array = []\n    n = 1\n    symbols = group.symbols\n    for i in range(len(a)):\n        if i == len(a) - 1:\n            if a[i] == a[i - 1]:\n                if (-a[i]) in symbols:\n                    new_array.append((-a[i], -n))\n                else:\n                    new_array.append((a[i], n))\n            else:\n                if (-a[i]) in symbols:\n                    new_array.append((-a[i], -1))\n                else:\n                    new_array.append((a[i], 1))\n            return new_array\n        elif a[i] == a[i + 1]:\n            n += 1\n        else:\n            if (-a[i]) in symbols:\n                new_array.append((-a[i], -n))\n            else:\n                new_array.append((a[i], n))\n            n = 1"
        },
        {
          "file": "sympy/combinatorics/rewritingsystem.py",
          "type": "function",
          "name": "add_rule",
          "class_name": "RewritingSystem",
          "code": "def add_rule(self, w1, w2, check=False):\n        new_keys = set()\n\n        if w1 == w2:\n            return new_keys\n\n        if w1 < w2:\n            w1, w2 = w2, w1\n\n        if (w1, w2) in self.rules_cache:\n            return new_keys\n        self.rules_cache.append((w1, w2))\n\n        s1, s2 = w1, w2\n\n        # The following is the equivalent of checking\n        # s1 for overlaps with the implicit reductions\n        # {g*g**-1 -> <identity>} and {g**-1*g -> <identity>}\n        # for any generator g without installing the\n        # redundant rules that would result from processing\n        # the overlaps. See [1], Section 3 for details.\n\n        if len(s1) - len(s2) < 3:\n            if s1 not in self.rules:\n                new_keys.add(s1)\n                if not check:\n                    self._add_rule(s1, s2)\n            if s2**-1 > s1**-1 and s2**-1 not in self.rules:\n                new_keys.add(s2**-1)\n                if not check:\n                    self._add_rule(s2**-1, s1**-1)\n\n        # overlaps on the right\n        while len(s1) - len(s2) > -1:\n            g = s1[len(s1)-1]\n            s1 = s1.subword(0, len(s1)-1)\n            s2 = s2*g**-1\n            if len(s1) - len(s2) < 0:\n                #print(\"this\", len(s1)-len(s2))\n                if s2 not in self.rules:\n                    if not check:\n                        self._add_rule(s2, s1)\n                    new_keys.add(s2)\n            elif len(s1) - len(s2) < 3:\n                new = self.add_rule(s1, s2, check)\n                new_keys.update(new)\n\n        # overlaps on the left\n        while len(w1) - len(w2) > -1:\n            g = w1[0]\n            w1 = w1.subword(1, len(w1))\n            w2 = g**-1*w2\n            if len(w1) - len(w2) < 0:\n                if w2 not in self.rules:\n                    if not check:\n                        self._add_rule(w2, w1)\n                    new_keys.add(w2)\n            elif len(w1) - len(w2) < 3:\n                new = self.add_rule(w1, w2, check)\n                new_keys.update(new)\n\n        return new_keys"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}