{
  "instance_id": "sympy__sympy-21055",
  "repo": "sympy/sympy",
  "created_at": "2021-03-07T21:08:36Z",
  "problem_statement": "`refine()` does not understand how to simplify complex arguments\nJust learned about the refine-function, which would come in handy frequently for me.  But\r\n`refine()` does not recognize that argument functions simplify for real numbers.\r\n\r\n```\r\n>>> from sympy import *                                                     \r\n>>> var('a,x')                                                              \r\n>>> J = Integral(sin(x)*exp(-a*x),(x,0,oo))                                     \r\n>>> J.doit()\r\n\tPiecewise((1/(a**2 + 1), 2*Abs(arg(a)) < pi), (Integral(exp(-a*x)*sin(x), (x, 0, oo)), True))\r\n>>> refine(J.doit(),Q.positive(a))                                                 \r\n        Piecewise((1/(a**2 + 1), 2*Abs(arg(a)) < pi), (Integral(exp(-a*x)*sin(x), (x, 0, oo)), True))\r\n>>> refine(abs(a),Q.positive(a))                                            \r\n\ta\r\n>>> refine(arg(a),Q.positive(a))                                            \r\n\targ(a)\r\n```\r\nI cann't find any open issues identifying this.  Easy to fix, though.\r\n\r\n\n",
  "patch": "diff --git a/sympy/assumptions/refine.py b/sympy/assumptions/refine.py\n--- a/sympy/assumptions/refine.py\n+++ b/sympy/assumptions/refine.py\n@@ -297,6 +297,28 @@ def refine_im(expr, assumptions):\n         return - S.ImaginaryUnit * arg\n     return _refine_reim(expr, assumptions)\n \n+def refine_arg(expr, assumptions):\n+    \"\"\"\n+    Handler for complex argument\n+\n+    Explanation\n+    ===========\n+\n+    >>> from sympy.assumptions.refine import refine_arg\n+    >>> from sympy import Q, arg\n+    >>> from sympy.abc import x\n+    >>> refine_arg(arg(x), Q.positive(x))\n+    0\n+    >>> refine_arg(arg(x), Q.negative(x))\n+    pi\n+    \"\"\"\n+    rg = expr.args[0]\n+    if ask(Q.positive(rg), assumptions):\n+        return S.Zero\n+    if ask(Q.negative(rg), assumptions):\n+        return S.Pi\n+    return None\n+\n \n def _refine_reim(expr, assumptions):\n     # Helper function for refine_re & refine_im\n@@ -379,6 +401,7 @@ def refine_matrixelement(expr, assumptions):\n     'atan2': refine_atan2,\n     're': refine_re,\n     'im': refine_im,\n+    'arg': refine_arg,\n     'sign': refine_sign,\n     'MatrixElement': refine_matrixelement\n }  # type: Dict[str, Callable[[Expr, Boolean], Expr]]\n",
  "similar_bug_items": [
    {
      "pr_number": 17375,
      "pr_title": "polygamma improvements (evaluation, real and positivity determination)",
      "pr_body": "Closes #17350. Fixes #12569.\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\n- Add `_eval_evalf` to prevent evaluation when the first argument is not a nonnegative integer.\r\n- Remove `_eval_is_real` since the logic was wrong.\r\n- Update `_eval_is_positive` and `_eval_is_negative` to avoid doing comparisons with `>`.\r\n\r\n#### Release Notes\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 17350,
      "issue_title": "Should polygamma(a, b) raise NotImplementedError for noninteger (numeric) a?",
      "issue_body": "mpmath's implementation of polygamma is valid only for integer first arguments. SymPy's implementation does not care, but this leads to many issues if you actually try to do things with such objects:\r\n\r\n```\r\n>>> (I*polygamma(I, pi)).as_real_imag()\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'zero'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'finite'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'extended_positive'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1308, in evalf\r\n    rf = evalf_table[x.func]\r\nKeyError: polygamma\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/eward/se2/sympy/core/mul.py\", line 798, in as_real_imag\r\n    if i.is_zero:\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 321, in _ask\r\n    _ask(pk, obj)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 864, in _eval_is_negative\r\n    finite = self.is_finite\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 321, in _ask\r\n    _ask(pk, obj)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 857, in _eval_is_positive\r\n    extended_positive = self.is_extended_positive\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 882, in _eval_is_extended_positive\r\n    n2 = self._eval_evalf(2)\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 565, in _eval_evalf\r\n    args = [arg._to_mpmath(prec + 5) for arg in self.args]\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 565, in <listcomp>\r\n    args = [arg._to_mpmath(prec + 5) for arg in self.args]\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1489, in _to_mpmath\r\n    re, im, _, _ = evalf(self, prec, {})\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1314, in evalf\r\n    xe = x._eval_evalf(prec)\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 590, in _eval_evalf\r\n    v = func(*args)\r\n  File \"/usr/local/lib/python3.6/dist-packages/mpmath/ctx_mp.py\", line 265, in psi\r\n    m = int(m)\r\nTypeError: int() argument must be a string, a bytes-like object or a number, not 'mpc'\r\n>>> (tanh(polygamma(I, 1))).rewrite(exp)\r\n...\r\n    m = int(m)\r\nTypeError: int() argument must be a string, a bytes-like object or a number, not 'mpc'\r\n>>> (I / polygamma(I, 4)).rewrite(exp)\r\n...\r\n    raise TypeError(\"Invalid comparison of complex %s\" % me)\r\nTypeError: Invalid comparison of complex I\r\n```",
      "issue_closed_at": "2019-08-23T06:49:24Z",
      "base_commit": "fc6f766ab588fecbb69ad85eb01ca28b44715e5c",
      "changes": [
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "class",
          "name": "polygamma",
          "code": "class polygamma(Function):\n    r\"\"\"\n    The function ``polygamma(n, z)`` returns ``log(gamma(z)).diff(n + 1)``.\n\n    It is a meromorphic function on `\\mathbb{C}` and defined as the (n+1)-th\n    derivative of the logarithm of the gamma function:\n\n    .. math::\n        \\psi^{(n)} (z) := \\frac{\\mathrm{d}^{n+1}}{\\mathrm{d} z^{n+1}} \\log\\Gamma(z).\n\n    Examples\n    ========\n\n    Several special values are known:\n\n    >>> from sympy import S, polygamma\n    >>> polygamma(0, 1)\n    -EulerGamma\n    >>> polygamma(0, 1/S(2))\n    -2*log(2) - EulerGamma\n    >>> polygamma(0, 1/S(3))\n    -log(3) - sqrt(3)*pi/6 - EulerGamma - log(sqrt(3))\n    >>> polygamma(0, 1/S(4))\n    -pi/2 - log(4) - log(2) - EulerGamma\n    >>> polygamma(0, 2)\n    1 - EulerGamma\n    >>> polygamma(0, 23)\n    19093197/5173168 - EulerGamma\n\n    >>> from sympy import oo, I\n    >>> polygamma(0, oo)\n    oo\n    >>> polygamma(0, -oo)\n    oo\n    >>> polygamma(0, I*oo)\n    oo\n    >>> polygamma(0, -I*oo)\n    oo\n\n    Differentiation with respect to x is supported:\n\n    >>> from sympy import Symbol, diff\n    >>> x = Symbol(\"x\")\n    >>> diff(polygamma(0, x), x)\n    polygamma(1, x)\n    >>> diff(polygamma(0, x), x, 2)\n    polygamma(2, x)\n    >>> diff(polygamma(0, x), x, 3)\n    polygamma(3, x)\n    >>> diff(polygamma(1, x), x)\n    polygamma(2, x)\n    >>> diff(polygamma(1, x), x, 2)\n    polygamma(3, x)\n    >>> diff(polygamma(2, x), x)\n    polygamma(3, x)\n    >>> diff(polygamma(2, x), x, 2)\n    polygamma(4, x)\n\n    >>> n = Symbol(\"n\")\n    >>> diff(polygamma(n, x), x)\n    polygamma(n + 1, x)\n    >>> diff(polygamma(n, x), x, 2)\n    polygamma(n + 2, x)\n\n    We can rewrite polygamma functions in terms of harmonic numbers:\n\n    >>> from sympy import harmonic\n    >>> polygamma(0, x).rewrite(harmonic)\n    harmonic(x - 1) - EulerGamma\n    >>> polygamma(2, x).rewrite(harmonic)\n    2*harmonic(x - 1, 3) - 2*zeta(3)\n    >>> ni = Symbol(\"n\", integer=True)\n    >>> polygamma(ni, x).rewrite(harmonic)\n    (-1)**(n + 1)*(-harmonic(x - 1, n + 1) + zeta(n + 1))*factorial(n)\n\n    See Also\n    ========\n\n    gamma: Gamma function.\n    lowergamma: Lower incomplete gamma function.\n    uppergamma: Upper incomplete gamma function.\n    loggamma: Log Gamma function.\n    digamma: Digamma function.\n    trigamma: Trigamma function.\n    sympy.functions.special.beta_functions.beta: Euler Beta function.\n\n    References\n    ==========\n\n    .. [1] https://en.wikipedia.org/wiki/Polygamma_function\n    .. [2] http://mathworld.wolfram.com/PolygammaFunction.html\n    .. [3] http://functions.wolfram.com/GammaBetaErf/PolyGamma/\n    .. [4] http://functions.wolfram.com/GammaBetaErf/PolyGamma2/\n    \"\"\"\n\n\n    def fdiff(self, argindex=2):\n        if argindex == 2:\n            n, z = self.args[:2]\n            return polygamma(n + 1, z)\n        else:\n            raise ArgumentIndexError(self, argindex)\n\n    def _eval_is_positive(self):\n        if self.args[1].is_positive and (self.args[0] > 0) == True:\n            return self.args[0].is_odd\n\n    def _eval_is_negative(self):\n        if self.args[1].is_positive and (self.args[0] > 0) == True:\n            return self.args[0].is_even\n\n    def _eval_is_real(self):\n        return self.args[0].is_real\n\n    def _eval_aseries(self, n, args0, x, logx):\n        from sympy import Order\n        if args0[1] != oo or not \\\n                (self.args[0].is_Integer and self.args[0].is_nonnegative):\n            return super(polygamma, self)._eval_aseries(n, args0, x, logx)\n        z = self.args[1]\n        N = self.args[0]\n\n        if N == 0:\n            # digamma function series\n            # Abramowitz & Stegun, p. 259, 6.3.18\n            r = log(z) - 1/(2*z)\n            o = None\n            if n < 2:\n                o = Order(1/z, x)\n            else:\n                m = ceiling((n + 1)//2)\n                l = [bernoulli(2*k) / (2*k*z**(2*k)) for k in range(1, m)]\n                r -= Add(*l)\n                o = Order(1/z**(2*m), x)\n            return r._eval_nseries(x, n, logx) + o\n        else:\n            # proper polygamma function\n            # Abramowitz & Stegun, p. 260, 6.4.10\n            # We return terms to order higher than O(x**n) on purpose\n            # -- otherwise we would not be able to return any terms for\n            #    quite a long time!\n            fac = gamma(N)\n            e0 = fac + N*fac/(2*z)\n            m = ceiling((n + 1)//2)\n            for k in range(1, m):\n                fac = fac*(2*k + N - 1)*(2*k + N - 2) / ((2*k)*(2*k - 1))\n                e0 += bernoulli(2*k)*fac/z**(2*k)\n            o = Order(1/z**(2*m), x)\n            if n == 0:\n                o = Order(1/z, x)\n            elif n == 1:\n                o = Order(1/z**2, x)\n            r = e0._eval_nseries(z, n, logx) + o\n            return (-1 * (-1/z)**N * r)._eval_nseries(x, n, logx)\n\n    @classmethod\n    def eval(cls, n, z):\n        n, z = map(sympify, (n, z))\n        from sympy import unpolarify\n\n        if n.is_integer:\n            if n.is_nonnegative:\n                nz = unpolarify(z)\n                if z != nz:\n                    return polygamma(n, nz)\n\n            if n == -1:\n                return loggamma(z)\n            else:\n                if z.is_Number:\n                    if z is S.NaN:\n                        return S.NaN\n                    elif z is S.Infinity:\n                        if n.is_Number:\n                            if n is S.Zero:\n                                return S.Infinity\n                            else:\n                                return S.Zero\n                    elif z.is_Integer:\n                        if z.is_nonpositive:\n                            return S.ComplexInfinity\n                        else:\n                            if n is S.Zero:\n                                return -S.EulerGamma + harmonic(z - 1, 1)\n                            elif n.is_odd:\n                                return (-1)**(n + 1)*factorial(n)*zeta(n + 1, z)\n\n        if n == 0:\n            if z is S.NaN:\n                return S.NaN\n            elif z.is_Rational:\n\n                p, q = z.as_numer_denom()\n\n                # only expand for small denominators to avoid creating long expressions\n                if q <= 5:\n                    return expand_func(polygamma(n, z, evaluate=False))\n\n            elif z in (S.Infinity, S.NegativeInfinity):\n                return S.Infinity\n            else:\n                t = z.extract_multiplicatively(S.ImaginaryUnit)\n                if t in (S.Infinity, S.NegativeInfinity):\n                    return S.Infinity\n\n        # TODO n == 1 also can do some rational z\n\n    def _eval_expand_func(self, **hints):\n        n, z = self.args\n\n        if n.is_Integer and n.is_nonnegative:\n            if z.is_Add:\n                coeff = z.args[0]\n                if coeff.is_Integer:\n                    e = -(n + 1)\n                    if coeff > 0:\n                        tail = Add(*[Pow(\n                            z - i, e) for i in range(1, int(coeff) + 1)])\n                    else:\n                        tail = -Add(*[Pow(\n                            z + i, e) for i in range(0, int(-coeff))])\n                    return polygamma(n, z - coeff) + (-1)**n*factorial(n)*tail\n\n            elif z.is_Mul:\n                coeff, z = z.as_two_terms()\n                if coeff.is_Integer and coeff.is_positive:\n                    tail = [ polygamma(n, z + Rational(\n                        i, coeff)) for i in range(0, int(coeff)) ]\n                    if n == 0:\n                        return Add(*tail)/coeff + log(coeff)\n                    else:\n                        return Add(*tail)/coeff**(n + 1)\n                z *= coeff\n\n        if n == 0 and z.is_Rational:\n            p, q = z.as_numer_denom()\n\n            # Reference:\n            #   Values of the polygamma functions at rational arguments, J. Choi, 2007\n            part_1 = -S.EulerGamma - pi * cot(p * pi / q) / 2 - log(q) + Add(\n                *[cos(2 * k * pi * p / q) * log(2 * sin(k * pi / q)) for k in range(1, q)])\n\n            if z > 0:\n                n = floor(z)\n                z0 = z - n\n                return part_1 + Add(*[1 / (z0 + k) for k in range(n)])\n            elif z < 0:\n                n = floor(1 - z)\n                z0 = z + n\n                return part_1 - Add(*[1 / (z0 - 1 - k) for k in range(n)])\n\n        return polygamma(n, z)\n\n    def _eval_rewrite_as_zeta(self, n, z, **kwargs):\n        if n >= S.One:\n            return (-1)**(n + 1)*factorial(n)*zeta(n + 1, z)\n        else:\n            return self\n\n    def _eval_rewrite_as_harmonic(self, n, z, **kwargs):\n        if n.is_integer:\n            if n == S.Zero:\n                return harmonic(z - 1) - S.EulerGamma\n            else:\n                return S.NegativeOne**(n+1) * factorial(n) * (zeta(n+1) - harmonic(z-1, n+1))\n\n    def _eval_as_leading_term(self, x):\n        from sympy import Order\n        n, z = [a.as_leading_term(x) for a in self.args]\n        o = Order(z, x)\n        if n == 0 and o.contains(1/x):\n            return o.getn() * log(x)\n        else:\n            return self.func(n, z)"
        },
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "function",
          "name": "fdiff",
          "class_name": "multigamma",
          "code": "def fdiff(self, argindex=2):\n        from sympy import Sum\n        if argindex == 2:\n            x, p = self.args\n            k = Dummy(\"k\")\n            return self.func(x, p)*Sum(polygamma(0, x + (1 - k)/2), (k, 1, p))\n        else:\n            raise ArgumentIndexError(self, argindex)"
        },
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "function",
          "name": "_eval_expand_func",
          "class_name": "loggamma",
          "code": "def _eval_expand_func(self, **hints):\n        from sympy import Sum\n        z = self.args[0]\n\n        if z.is_Rational:\n            p, q = z.as_numer_denom()\n            # General rational arguments (u + p/q)\n            # Split z as n + p/q with p < q\n            n = p // q\n            p = p - n*q\n            if p.is_positive and q.is_positive and p < q:\n                k = Dummy(\"k\")\n                if n.is_positive:\n                    return loggamma(p / q) - n*log(q) + Sum(log((k - 1)*q + p), (k, 1, n))\n                elif n.is_negative:\n                    return loggamma(p / q) - n*log(q) + S.Pi*S.ImaginaryUnit*n - Sum(log(k*q - p), (k, 1, -n))\n                elif n.is_zero:\n                    return loggamma(p / q)\n\n        return self"
        }
      ]
    },
    {
      "pr_number": 19091,
      "pr_title": "Fixed tensor contraction problem",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\" (see\r\nhttps://tinyurl.com/auto-closing for more information). Also, please\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #18465 \r\n\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nTensor contractions using `replace_with_arrays` would prematurely contract the data arrays before applying the metric\u2014yielding incorrect results. The solution implemented here is a rather trivial one in that it simply separates out an existing function for doing matrix multiplication with an existing data array with the metric and utilizes it in `Tensor._extract_data` to apply the metric before dummy indices are contracted.\r\n\r\n#### Other comments\r\n\r\nThis is my first pull request with Sympy. Let me know if I should make any adjustments to my code.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* tensor\r\n  * Fixed a bug where tensor contractions in calls to `replace_with_arrays` would fail to apply the metric.\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 18465,
      "issue_title": "Tensor contractions are wrong",
      "issue_body": "This is essentially a generalization of #17328.\r\n\r\nThe problem in the current implementation is that contractions are handled before applications of the metric, which leads to incorrect results such as in #17328.\r\n\r\nIn `tensor/tensor.py`:\r\n```python\r\nclass Tensor(TensExpr):\r\n# ...\r\n    def _extract_data(self, replacement_dict):\r\n    # ...\r\n        if len(dum1) > 0:\r\n            indices2 = other.get_indices()\r\n            repl = {}\r\n            for p1, p2 in dum1:\r\n                repl[indices2[p2]] = -indices2[p1]\r\n            other = other.xreplace(repl).doit()\r\n            array = _TensorDataLazyEvaluator.data_contract_dum([array], dum1, len(indices2))\r\n\r\n        free_ind1 = self.get_free_indices()\r\n        free_ind2 = other.get_free_indices()\r\n\r\n        return self._match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict)\r\n```\r\nAnd thus, the issue is that `_TensorDataLazyEvaluator.data_contract_dum` is being called prior to `self._match_indices_with_other_tensor` (where the metric is applied).\r\n\r\nThe reason that this ordering matters is because tensor contraction is itself the abstraction of applying the metric to the tensors that represent psuedo-riemannian manifolds. In essence, it means that we must have it that ![equation](https://latex.codecogs.com/svg.latex?T^\\mu_\\mu=g_{\\mu\\nu}T^{\\mu\\nu}); however, this isn't the case here.\r\n\r\nI've tried tampering with the code above, but by the way tensors have been designed, this bug is essentially unavoidable. As a consequence, the tensor module needs to be refactored in order to get accurate results. (Also, I couldn't help but notice that the last argument to `_TensorDataLazyEvaluator.data_contract_dum` isn't used).\r\n\r\n@drybalka had mentioned that he had this sort of refactoring in the works, but based on his fork, progress seems to be slow. I think discussions should be in order for reorganizing how tensors actually represent their components in this module.",
      "issue_closed_at": "2020-04-20T15:15:50Z",
      "base_commit": "64d28fe0534f6993695d11244ea740f783958dc8",
      "changes": [
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "recursor",
          "class_name": "TensExpr",
          "code": "def recursor(expr, pos):\n            if isinstance(expr, TensorIndex):\n                yield (expr, pos)\n            elif isinstance(expr, (Tuple, TensExpr)):\n                for p, arg in enumerate(expr.args):\n                    for i in recursor(arg, pos+(p,)):\n                        yield i"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "_match_indices_with_other_tensor",
          "class_name": "TensExpr",
          "code": "def _match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict):\n        from .array import tensorcontraction, tensorproduct, permutedims\n\n        index_types1 = [i.tensor_index_type for i in free_ind1]\n\n        # Check if variance of indices needs to be fixed:\n        pos2up = []\n        pos2down = []\n        free2remaining = free_ind2[:]\n        for pos1, index1 in enumerate(free_ind1):\n            if index1 in free2remaining:\n                pos2 = free2remaining.index(index1)\n                free2remaining[pos2] = None\n                continue\n            if -index1 in free2remaining:\n                pos2 = free2remaining.index(-index1)\n                free2remaining[pos2] = None\n                free_ind2[pos2] = index1\n                if index1.is_up:\n                    pos2up.append(pos2)\n                else:\n                    pos2down.append(pos2)\n            else:\n                index2 = free2remaining[pos1]\n                if index2 is None:\n                    raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n                free2remaining[pos1] = None\n                free_ind2[pos1] = index1\n                if index1.is_up ^ index2.is_up:\n                    if index1.is_up:\n                        pos2up.append(pos1)\n                    else:\n                        pos2down.append(pos1)\n\n        if len(set(free_ind1) & set(free_ind2)) < len(free_ind1):\n            raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n\n        # TODO: add possibility of metric after (spinors)\n        def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)\n\n        # Raise indices:\n        for pos in pos2up:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            metric_inverse = _TensorDataLazyEvaluator.inverse_matrix(metric)\n            array = contract_and_permute(metric_inverse, array, pos)\n        # Lower indices:\n        for pos in pos2down:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            array = contract_and_permute(metric, array, pos)\n\n        if free_ind1:\n            permutation = TensExpr._get_indices_permutation(free_ind2, free_ind1)\n            array = permutedims(array, permutation)\n\n        if hasattr(array, \"rank\") and array.rank() == 0:\n            array = array[()]\n\n        return free_ind2, array"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "contract_and_permute",
          "class_name": "TensExpr",
          "code": "def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "_extract_data",
          "class_name": "TensorElement",
          "code": "def _extract_data(self, replacement_dict):\n        ret_indices, array = self.expr._extract_data(replacement_dict)\n        index_map = self.index_map\n        slice_tuple = tuple(index_map.get(i, slice(None)) for i in ret_indices)\n        ret_indices = [i for i in ret_indices if i not in index_map]\n        array = array.__getitem__(slice_tuple)\n        return ret_indices, array"
        }
      ]
    },
    {
      "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": 15849,
      "pr_title": "Fixed vector derivatives printing ",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests . Please also\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #10173\r\n\r\n#### Brief description of what is fixed or changed\r\nFixed the fourth-order printing issue.\r\n\r\nThis has been attempted in #10283 and #15022 but no unit tests were added. It wasn't clear which one of these I should base it on, but as seen, the current PR deals with higher order derivatives as well. And fixes some unused/redundant imports.\r\n\r\n#### Other comments\r\nI choose to keep two separate `if`-clauses to make the comments stand alone, while the newly added `if dot_i >= 5:` could easily have been added with the previous `if`.\r\n\r\nThe result looks different in different renders, see https://github.com/sympy/sympy/issues/14425#issuecomment-457834386 Hence, it may look a bit \"ugly\" in certain circumstances.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* physics.vector\r\n   * printing of arbitrary order vector derivatives is fixed\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 10173,
      "issue_title": "mechanics_printing module does not display 4th derivatives correctly ",
      "issue_body": "When I run the following code snippet in an IPython notebook, the 1st, 2nd, and 3rd derivatives display correctly, but the 4th derivative does not. The 4th derivative simply displays as p, rather than as \\ddddot{p}.\n\n```\n%pylab inline\n\nimport sympy.physics.mechanics.functions\n\nsympy.physics.mechanics.functions.mechanics_printing(use_latex=\"mathjax\", latex_mode=\"equation\")\n\nt = sympy.Symbol(\"t\")\np = sympy.Symbol(\"p\")(t)\n\np, p.diff(t), p.diff(t).diff(t), p.diff(t).diff(t).diff(t), p.diff(t).diff(t).diff(t).diff(t)\n```\n\nAt first glance, this behavior seems to be caused by the `_print_Derivative` function in sympy/sympy/physics/vector/printing.py not handling 4th derivatives. I'll try hacking this function locally to support 4th derivatives and report back if this works around the issue.\n\nIdeally, mechanics_printing should gracefully fall back to Leibniz notation in the case of high-order derivatives, rather than printing them incorrectly.\n",
      "issue_closed_at": "2019-01-29T15:53:05Z",
      "base_commit": "1e999fc5159b830a9872b86675bc5f3692a9c1be",
      "changes": [
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        }
      ]
    },
    {
      "pr_number": 15349,
      "pr_title": "quaternions: fix a sign mistake in the definition of the rotation matrix",
      "pr_body": "quaternions: fix a sign mistake in definition of rotation matrix\r\n<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests .-->\r\n\r\nFixes #15193.\r\n#### Brief description of what is fixed or changed\r\nSign was corrected. A unit test added. Other unit tests fixed with the correct values.\r\n\r\n#### Other comments\r\n\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* algebras\r\n    * fix sign mistake in Quaternion.to_rotation_matrix method\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15193,
      "issue_title": "Incorrect result with Quaterniont.to_rotation_matrix()",
      "issue_body": "https://github.com/sympy/sympy/blob/ab14b02dba5a7e3e4fb1e807fc8a954f1047a1a1/sympy/algebras/quaternion.py#L489\r\n\r\nThere appears to be an error in the `Quaternion.to_rotation_matrix()` output.  The simplest example I created to illustrate the problem is as follows:\r\n\r\n```\r\n>>import sympy\r\n>>print('Sympy version: ', sympy.__version__)\r\nSympy version: 1.2\r\n\r\n>> from sympy import *\r\n>> x = symbols('x')\r\n>> q = Quaternion(cos(x/2), sin(x/2), 0, 0)\r\n>> trigsimp(q.to_rotation_matrix())\r\nMatrix([\r\n[1,      0,      0],\r\n[0, cos(x), sin(x)],\r\n[0, sin(x), cos(x)]])\r\n```\r\nOne of the `sin(x)` functions should be negative.  What was the reference of the original equations?  ",
      "issue_closed_at": "2018-10-07T10:54:51Z",
      "base_commit": "768da1c6f6ec907524b8ebbf6bf818c92b56101b",
      "changes": [
        {
          "file": "sympy/algebras/quaternion.py",
          "type": "function",
          "name": "to_rotation_matrix",
          "class_name": "Quaternion",
          "code": "def to_rotation_matrix(self, v=None):\n        \"\"\"Returns the equivalent rotation transformation matrix of the quaternion\n        which represents rotation about the origin if v is not passed.\n\n        Example\n        ========\n\n        >>> from sympy.algebras.quaternion import Quaternion\n        >>> from sympy import symbols, trigsimp, cos, sin\n        >>> x = symbols('x')\n        >>> q = Quaternion(cos(x/2), 0, 0, sin(x/2))\n        >>> trigsimp(q.to_rotation_matrix())\n        Matrix([\n        [cos(x), -sin(x), 0],\n        [sin(x),  cos(x), 0],\n        [     0,       0, 1]])\n\n        Generates a 4x4 transformation matrix (used for rotation about a point\n        other than the origin) if the point(v) is passed as an argument.\n\n        Example\n        ========\n\n        >>> from sympy.algebras.quaternion import Quaternion\n        >>> from sympy import symbols, trigsimp, cos, sin\n        >>> x = symbols('x')\n        >>> q = Quaternion(cos(x/2), 0, 0, sin(x/2))\n        >>> trigsimp(q.to_rotation_matrix((1, 1, 1)))\n         Matrix([\n        [cos(x), -sin(x), 0,  sin(x) - cos(x) + 1],\n        [sin(x),  cos(x), 0, -sin(x) - cos(x) + 1],\n        [     0,       0, 1,                    0],\n        [     0,       0, 0,                    1]])\n        \"\"\"\n\n        q = self\n        s = q.norm()**-2\n        m00 = 1 - 2*s*(q.c**2 + q.d**2)\n        m01 = 2*s*(q.b*q.c - q.d*q.a)\n        m02 = 2*s*(q.b*q.d + q.c*q.a)\n\n        m10 = 2*s*(q.b*q.c + q.d*q.a)\n        m11 = 1 - 2*s*(q.b**2 + q.d**2)\n        m12 = 2*s*(q.c*q.d + q.b*q.a)\n\n        m20 = 2*s*(q.b*q.d - q.c*q.a)\n        m21 = 2*s*(q.c*q.d + q.b*q.a)\n        m22 = 1 - 2*s*(q.b**2 + q.c**2)\n\n        if not v:\n            return Matrix([[m00, m01, m02], [m10, m11, m12], [m20, m21, m22]])\n\n        else:\n            (x, y, z) = v\n\n            m03 = x - x*m00 - y*m01 - z*m02\n            m13 = y - x*m10 - y*m11 - z*m12\n            m23 = z - x*m20 - y*m21 - z*m22\n            m30 = m31 = m32 = 0\n            m33 = 1\n\n            return Matrix([[m00, m01, m02, m03], [m10, m11, m12, m13],\n                          [m20, m21, m22, m23], [m30, m31, m32, m33]])"
        }
      ]
    }
  ]
}