{
  "instance_id": "sympy__sympy-24152",
  "repo": "sympy/sympy",
  "created_at": "2022-10-21T13:47:03Z",
  "problem_statement": "Bug in expand of TensorProduct + Workaround + Fix\n### Error description\r\nThe expansion of a TensorProduct object stops incomplete if summands in the tensor product factors have (scalar) factors, e.g.\r\n```\r\nfrom sympy import *\r\nfrom sympy.physics.quantum import *\r\nU = Operator('U')\r\nV = Operator('V')\r\nP = TensorProduct(2*U - V, U + V)\r\nprint(P) \r\n# (2*U - V)x(U + V)\r\nprint(P.expand(tensorproduct=True)) \r\n#result: 2*Ux(U + V) - Vx(U + V) #expansion has missed 2nd tensor factor and is incomplete\r\n```\r\nThis is clearly not the expected behaviour. It also effects other functions that rely on .expand(tensorproduct=True), as e.g. qapply() .\r\n\r\n### Work around\r\nRepeat .expand(tensorproduct=True) as may times as there are tensor factors, resp. until the expanded term does no longer change. This is however only reasonable in interactive session and not in algorithms.\r\n\r\n### Code Fix\r\n.expand relies on the method TensorProduct._eval_expand_tensorproduct(). The issue arises from an inprecise check in TensorProduct._eval_expand_tensorproduct() whether a recursive call is required; it fails when the creation of a TensorProduct object returns commutative (scalar) factors up front: in that case the constructor returns a Mul(c_factors, TensorProduct(..)).\r\nI thus propose the following  code fix in TensorProduct._eval_expand_tensorproduct() in quantum/tensorproduct.py.  I have marked the four lines to be added / modified:\r\n```\r\n    def _eval_expand_tensorproduct(self, **hints):\r\n                ...\r\n                for aa in args[i].args:\r\n                    tp = TensorProduct(*args[:i] + (aa,) + args[i + 1:])\r\n                    c_part, nc_part = tp.args_cnc() #added\r\n                    if len(nc_part)==1 and isinstance(nc_part[0], TensorProduct): #modified\r\n                        nc_part = (nc_part[0]._eval_expand_tensorproduct(), ) #modified\r\n                    add_args.append(Mul(*c_part)*Mul(*nc_part)) #modified\r\n                break\r\n                ...\r\n```\r\nThe fix splits of commutative (scalar) factors from the tp returned. The TensorProduct object will be the one nc factor in nc_part (see TensorProduct.__new__ constructor), if any. Note that the constructor will return 0 if a tensor factor is 0, so there is no guarantee that tp contains a TensorProduct object (e.g. TensorProduct(U-U, U+V).\r\n\r\n\r\n\n",
  "patch": "diff --git a/sympy/physics/quantum/tensorproduct.py b/sympy/physics/quantum/tensorproduct.py\n--- a/sympy/physics/quantum/tensorproduct.py\n+++ b/sympy/physics/quantum/tensorproduct.py\n@@ -246,9 +246,12 @@ def _eval_expand_tensorproduct(self, **hints):\n             if isinstance(args[i], Add):\n                 for aa in args[i].args:\n                     tp = TensorProduct(*args[:i] + (aa,) + args[i + 1:])\n-                    if isinstance(tp, TensorProduct):\n-                        tp = tp._eval_expand_tensorproduct()\n-                    add_args.append(tp)\n+                    c_part, nc_part = tp.args_cnc()\n+                    # Check for TensorProduct object: is the one object in nc_part, if any:\n+                    # (Note: any other object type to be expanded must be added here)\n+                    if len(nc_part) == 1 and isinstance(nc_part[0], TensorProduct):\n+                        nc_part = (nc_part[0]._eval_expand_tensorproduct(), )\n+                    add_args.append(Mul(*c_part)*Mul(*nc_part))\n                 break\n \n         if add_args:\n",
  "similar_bug_items": [
    {
      "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": 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": 21182,
      "pr_title": "Fixed the error in residue computation and added test",
      "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 #21176\r\n\r\n#### Brief description of what is fixed or changed\r\nChanged the assignment value of variable `res` in `_eval_nseries()` in mul.py because the attribute `removeO()` is only there for sympyfied expressions. `res = 0` assigned the integer equivalent of `0` in python to `res`, to which `removeO()` could not be applied. Also, code credits to @oscarbenjamin .\r\n\r\n#### Other comments\r\nAlso updated the test file.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below between the BEGIN and END\r\nstatements. The basic format is a bulleted list with the name of the subpackage\r\nand the release note for this PR. For example:\r\n\r\n* solvers\r\n  * Added a new solver for logarithmic equations.\r\n\r\n* functions\r\n  * Fixed a bug with log of integers.\r\n\r\nor if no release note(s) should be included use:\r\n\r\nNO ENTRY\r\n\r\nSee https://github.com/sympy/sympy/wiki/Writing-Release-Notes for more\r\ninformation on how to write release notes. The bot will check your release\r\nnotes automatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 21176,
      "issue_title": "Incorrect residue of `x**2*cot(pi*x)/(x**4 + 1)`",
      "issue_body": "```\r\nx = Symbol('x')\r\nresidue(x**2*cot(pi*x)/(x**4 + 1), x, -sqrt(2)/2 - sqrt(2)*I/2)\r\n```\r\n~Gives `AttributeError: 'int' object has no attribute 'removeO'`~\r\nNow it gives wrong result",
      "issue_closed_at": "2021-06-01T05:44:42Z",
      "base_commit": "43ee3d9099b3422276ada33ebdc981a7a58498c6",
      "changes": [
        {
          "file": "sympy/core/mul.py",
          "type": "function",
          "name": "coeff_exp",
          "class_name": "Mul",
          "code": "def coeff_exp(term, x):\n            coeff, exp = S.One, S.Zero\n            for factor in Mul.make_args(term):\n                if factor.has(x):\n                    base, exp = factor.as_base_exp()\n                    if base != x:\n                        try:\n                            return term.leadterm(x)\n                        except ValueError:\n                            return term, S.Zero\n                else:\n                    coeff *= factor\n            return coeff, exp"
        }
      ]
    },
    {
      "pr_number": 17769,
      "pr_title": "Fix linsolve  for immutable matrix",
      "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 #17762 \r\nFixes #17722 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nI've fixed the type checking for `linearsolve`, and also fixed some usage of the mutability of a matrix in `gauss_jordan_solve`.\r\n\r\n#### Other comments\r\n\r\nI'd also try to check `linsolve` whether it works with matrix expressions.\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- matrices\r\n  - Fixed `Matrix.gauss_jordan_solve` raising `ShapeError` for immutable matrices.\r\n- solvers\r\n  - Fixed `linsolve` raising errors for `ImmutableMatrix`.\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 17722,
      "issue_title": "linsolve error; ValueError: Got rows of variable lengths: [1, 3]",
      "issue_body": "Can anyone tell me if this is a bug and if there is a method to get around this please, cause it seems like the linsolve() function is broken?\r\nA similar question was asked in issue #17430 but this doesn't solve my problem when used in jupyter or qtconsole or spyder. This function would be really useful if it worked for my uni study.\r\n\r\nParameters given are: *system* in form of Ax=b and variables *c* in a tuple of Sympy Symbols\r\n![image](https://user-images.githubusercontent.com/42601907/66694509-028ba580-ed00-11e9-85db-f8e1567a5306.png)\r\n![image](https://user-images.githubusercontent.com/42601907/66694519-1505df00-ed00-11e9-9861-02ea8f1dd249.png)\r\n",
      "issue_closed_at": "2019-10-21T21:12:36Z",
      "base_commit": "4c87ea3ec5a36b0d3364b9bc52f88653282bf9aa",
      "changes": [
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/solvers/solveset.py",
          "type": "function",
          "name": "linsolve",
          "class_name": null,
          "code": "def linsolve(system, *symbols):\n    r\"\"\"\n    Solve system of N linear equations with M variables; both\n    underdetermined and overdetermined systems are supported.\n    The possible number of solutions is zero, one or infinite.\n    Zero solutions throws a ValueError, whereas infinite\n    solutions are represented parametrically in terms of the given\n    symbols. For unique solution a FiniteSet of ordered tuples\n    is returned.\n\n    All Standard input formats are supported:\n    For the given set of Equations, the respective input types\n    are given below:\n\n    .. math:: 3x + 2y -   z = 1\n    .. math:: 2x - 2y + 4z = -2\n    .. math:: 2x -   y + 2z = 0\n\n    * Augmented Matrix Form, `system` given below:\n\n    ::\n\n              [3   2  -1  1]\n     system = [2  -2   4 -2]\n              [2  -1   2  0]\n\n    * List Of Equations Form\n\n    `system  =  [3x + 2y - z - 1, 2x - 2y + 4z + 2, 2x - y + 2z]`\n\n    * Input A & b Matrix Form (from Ax = b) are given as below:\n\n    ::\n\n         [3   2  -1 ]         [  1 ]\n     A = [2  -2   4 ]    b =  [ -2 ]\n         [2  -1   2 ]         [  0 ]\n\n    `system = (A, b)`\n\n    Symbols can always be passed but are actually only needed\n    when 1) a system of equations is being passed and 2) the\n    system is passed as an underdetermined matrix and one wants\n    to control the name of the free variables in the result.\n    An error is raised if no symbols are used for case 1, but if\n    no symbols are provided for case 2, internally generated symbols\n    will be provided. When providing symbols for case 2, there should\n    be at least as many symbols are there are columns in matrix A.\n\n    The algorithm used here is Gauss-Jordan elimination, which\n    results, after elimination, in a row echelon form matrix.\n\n    Returns\n    =======\n\n    A FiniteSet containing an ordered tuple of values for the\n    unknowns for which the `system` has a solution. (Wrapping\n    the tuple in FiniteSet is used to maintain a consistent\n    output format throughout solveset.)\n\n    Returns EmptySet(), if the linear system is inconsistent.\n\n    Raises\n    ======\n\n    ValueError\n        The input is not valid.\n        The symbols are not given.\n\n    Examples\n    ========\n\n    >>> from sympy import Matrix, S, linsolve, symbols\n    >>> x, y, z = symbols(\"x, y, z\")\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n    >>> b = Matrix([3, 6, 9])\n    >>> A\n    Matrix([\n    [1, 2,  3],\n    [4, 5,  6],\n    [7, 8, 10]])\n    >>> b\n    Matrix([\n    [3],\n    [6],\n    [9]])\n    >>> linsolve((A, b), [x, y, z])\n    {(-1, 2, 0)}\n\n    * Parametric Solution: In case the system is underdetermined, the\n      function will return a parametric solution in terms of the given\n      symbols. Those that are free will be returned unchanged. e.g. in\n      the system below, `z` is returned as the solution for variable z;\n      it can take on any value.\n\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])\n    >>> b = Matrix([3, 6, 9])\n    >>> linsolve((A, b), x, y, z)\n    {(z - 1, 2 - 2*z, z)}\n\n    If no symbols are given, internally generated symbols will be used.\n    The `tau0` in the 3rd position indicates (as before) that the 3rd\n    variable -- whatever it's named -- can take on any value:\n\n    >>> linsolve((A, b))\n    {(tau0 - 1, 2 - 2*tau0, tau0)}\n\n    * List of Equations as input\n\n    >>> Eqns = [3*x + 2*y - z - 1, 2*x - 2*y + 4*z + 2, - x + y/2 - z]\n    >>> linsolve(Eqns, x, y, z)\n    {(1, -2, -2)}\n\n    * Augmented Matrix as input\n\n    >>> aug = Matrix([[2, 1, 3, 1], [2, 6, 8, 3], [6, 8, 18, 5]])\n    >>> aug\n    Matrix([\n    [2, 1,  3, 1],\n    [2, 6,  8, 3],\n    [6, 8, 18, 5]])\n    >>> linsolve(aug, x, y, z)\n    {(3/10, 2/5, 0)}\n\n    * Solve for symbolic coefficients\n\n    >>> a, b, c, d, e, f = symbols('a, b, c, d, e, f')\n    >>> eqns = [a*x + b*y - c, d*x + e*y - f]\n    >>> linsolve(eqns, x, y)\n    {((-b*f + c*e)/(a*e - b*d), (a*f - c*d)/(a*e - b*d))}\n\n    * A degenerate system returns solution as set of given\n      symbols.\n\n    >>> system = Matrix(([0, 0, 0], [0, 0, 0], [0, 0, 0]))\n    >>> linsolve(system, x, y)\n    {(x, y)}\n\n    * For an empty system linsolve returns empty set\n\n    >>> linsolve([], x)\n    EmptySet()\n\n    * An error is raised if, after expansion, any nonlinearity\n      is detected:\n\n    >>> linsolve([x*(1/x - 1), (y - 1)**2 - y**2 + 1], x, y)\n    {(1, 1)}\n    >>> linsolve([x**2 - 1], x)\n    Traceback (most recent call last):\n    ...\n    ValueError:\n    The term x**2 is nonlinear in {x}\n    \"\"\"\n    if not system:\n        return S.EmptySet\n\n    # If second argument is an iterable\n    if symbols and hasattr(symbols[0], '__iter__'):\n        symbols = symbols[0]\n    sym_gen = isinstance(symbols, GeneratorType)\n\n    swap = {}\n    b = None  # if we don't get b the input was bad\n    syms_needed_msg = None\n\n    # unpack system\n\n    if hasattr(system, '__iter__'):\n\n        # 1). (A, b)\n        if len(system) == 2 and isinstance(system[0], Matrix):\n            A, b = system\n\n        # 2). (eq1, eq2, ...)\n        if not isinstance(system[0], Matrix):\n            if sym_gen or not symbols:\n                raise ValueError(filldedent('''\n                    When passing a system of equations, the explicit\n                    symbols for which a solution is being sought must\n                    be given as a sequence, too.\n                '''))\n            system = [\n                _mexpand(i.lhs - i.rhs if isinstance(i, Eq) else i,\n                recursive=True) for i in system]\n            system, symbols, swap = recast_to_symbols(system, symbols)\n            A, b = linear_eq_to_matrix(system, symbols)\n            syms_needed_msg = 'free symbols in the equations provided'\n\n    elif isinstance(system, Matrix) and not (\n            symbols and not isinstance(symbols, GeneratorType) and\n            isinstance(symbols[0], Matrix)):\n        # 3). A augmented with b\n        A, b = system[:, :-1], system[:, -1:]\n\n    if b is None:\n        raise ValueError(\"Invalid arguments\")\n\n    syms_needed_msg  = syms_needed_msg or 'columns of A'\n\n    if sym_gen:\n        symbols = [next(symbols) for i in range(A.cols)]\n        if any(set(symbols) & (A.free_symbols | b.free_symbols)):\n            raise ValueError(filldedent('''\n                At least one of the symbols provided\n                already appears in the system to be solved.\n                One way to avoid this is to use Dummy symbols in\n                the generator, e.g. numbered_symbols('%s', cls=Dummy)\n            ''' % symbols[0].name.rstrip('1234567890')))\n\n    try:\n        solution, params, free_syms = A.gauss_jordan_solve(b, freevar=True)\n    except ValueError:\n        # No solution\n        return S.EmptySet\n\n    # Replace free parameters with free symbols\n    if params:\n        if not symbols:\n            symbols = [_ for _ in params]\n            # re-use the parameters but put them in order\n            # params       [x, y, z]\n            # free_symbols [2, 0, 4]\n            # idx          [1, 0, 2]\n            idx = list(zip(*sorted(zip(free_syms, range(len(free_syms))))))[1]\n            # simultaneous replacements {y: x, x: y, z: z}\n            replace_dict = dict(zip(symbols, [symbols[i] for i in idx]))\n        elif len(symbols) >= A.cols:\n            replace_dict = {v: symbols[free_syms[k]] for k, v in enumerate(params)}\n        else:\n            raise IndexError(filldedent('''\n                the number of symbols passed should have a length\n                equal to the number of %s.\n                ''' % syms_needed_msg))\n        solution = [sol.xreplace(replace_dict) for sol in solution]\n\n    solution = [simplify(sol).xreplace(swap) for sol in solution]\n    return FiniteSet(tuple(solution))"
        },
        {
          "file": "sympy/solvers/solveset.py",
          "type": "function",
          "name": "linsolve",
          "class_name": null,
          "code": "def linsolve(system, *symbols):\n    r\"\"\"\n    Solve system of N linear equations with M variables; both\n    underdetermined and overdetermined systems are supported.\n    The possible number of solutions is zero, one or infinite.\n    Zero solutions throws a ValueError, whereas infinite\n    solutions are represented parametrically in terms of the given\n    symbols. For unique solution a FiniteSet of ordered tuples\n    is returned.\n\n    All Standard input formats are supported:\n    For the given set of Equations, the respective input types\n    are given below:\n\n    .. math:: 3x + 2y -   z = 1\n    .. math:: 2x - 2y + 4z = -2\n    .. math:: 2x -   y + 2z = 0\n\n    * Augmented Matrix Form, `system` given below:\n\n    ::\n\n              [3   2  -1  1]\n     system = [2  -2   4 -2]\n              [2  -1   2  0]\n\n    * List Of Equations Form\n\n    `system  =  [3x + 2y - z - 1, 2x - 2y + 4z + 2, 2x - y + 2z]`\n\n    * Input A & b Matrix Form (from Ax = b) are given as below:\n\n    ::\n\n         [3   2  -1 ]         [  1 ]\n     A = [2  -2   4 ]    b =  [ -2 ]\n         [2  -1   2 ]         [  0 ]\n\n    `system = (A, b)`\n\n    Symbols can always be passed but are actually only needed\n    when 1) a system of equations is being passed and 2) the\n    system is passed as an underdetermined matrix and one wants\n    to control the name of the free variables in the result.\n    An error is raised if no symbols are used for case 1, but if\n    no symbols are provided for case 2, internally generated symbols\n    will be provided. When providing symbols for case 2, there should\n    be at least as many symbols are there are columns in matrix A.\n\n    The algorithm used here is Gauss-Jordan elimination, which\n    results, after elimination, in a row echelon form matrix.\n\n    Returns\n    =======\n\n    A FiniteSet containing an ordered tuple of values for the\n    unknowns for which the `system` has a solution. (Wrapping\n    the tuple in FiniteSet is used to maintain a consistent\n    output format throughout solveset.)\n\n    Returns EmptySet(), if the linear system is inconsistent.\n\n    Raises\n    ======\n\n    ValueError\n        The input is not valid.\n        The symbols are not given.\n\n    Examples\n    ========\n\n    >>> from sympy import Matrix, S, linsolve, symbols\n    >>> x, y, z = symbols(\"x, y, z\")\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n    >>> b = Matrix([3, 6, 9])\n    >>> A\n    Matrix([\n    [1, 2,  3],\n    [4, 5,  6],\n    [7, 8, 10]])\n    >>> b\n    Matrix([\n    [3],\n    [6],\n    [9]])\n    >>> linsolve((A, b), [x, y, z])\n    {(-1, 2, 0)}\n\n    * Parametric Solution: In case the system is underdetermined, the\n      function will return a parametric solution in terms of the given\n      symbols. Those that are free will be returned unchanged. e.g. in\n      the system below, `z` is returned as the solution for variable z;\n      it can take on any value.\n\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])\n    >>> b = Matrix([3, 6, 9])\n    >>> linsolve((A, b), x, y, z)\n    {(z - 1, 2 - 2*z, z)}\n\n    If no symbols are given, internally generated symbols will be used.\n    The `tau0` in the 3rd position indicates (as before) that the 3rd\n    variable -- whatever it's named -- can take on any value:\n\n    >>> linsolve((A, b))\n    {(tau0 - 1, 2 - 2*tau0, tau0)}\n\n    * List of Equations as input\n\n    >>> Eqns = [3*x + 2*y - z - 1, 2*x - 2*y + 4*z + 2, - x + y/2 - z]\n    >>> linsolve(Eqns, x, y, z)\n    {(1, -2, -2)}\n\n    * Augmented Matrix as input\n\n    >>> aug = Matrix([[2, 1, 3, 1], [2, 6, 8, 3], [6, 8, 18, 5]])\n    >>> aug\n    Matrix([\n    [2, 1,  3, 1],\n    [2, 6,  8, 3],\n    [6, 8, 18, 5]])\n    >>> linsolve(aug, x, y, z)\n    {(3/10, 2/5, 0)}\n\n    * Solve for symbolic coefficients\n\n    >>> a, b, c, d, e, f = symbols('a, b, c, d, e, f')\n    >>> eqns = [a*x + b*y - c, d*x + e*y - f]\n    >>> linsolve(eqns, x, y)\n    {((-b*f + c*e)/(a*e - b*d), (a*f - c*d)/(a*e - b*d))}\n\n    * A degenerate system returns solution as set of given\n      symbols.\n\n    >>> system = Matrix(([0, 0, 0], [0, 0, 0], [0, 0, 0]))\n    >>> linsolve(system, x, y)\n    {(x, y)}\n\n    * For an empty system linsolve returns empty set\n\n    >>> linsolve([], x)\n    EmptySet()\n\n    * An error is raised if, after expansion, any nonlinearity\n      is detected:\n\n    >>> linsolve([x*(1/x - 1), (y - 1)**2 - y**2 + 1], x, y)\n    {(1, 1)}\n    >>> linsolve([x**2 - 1], x)\n    Traceback (most recent call last):\n    ...\n    ValueError:\n    The term x**2 is nonlinear in {x}\n    \"\"\"\n    if not system:\n        return S.EmptySet\n\n    # If second argument is an iterable\n    if symbols and hasattr(symbols[0], '__iter__'):\n        symbols = symbols[0]\n    sym_gen = isinstance(symbols, GeneratorType)\n\n    swap = {}\n    b = None  # if we don't get b the input was bad\n    syms_needed_msg = None\n\n    # unpack system\n\n    if hasattr(system, '__iter__'):\n\n        # 1). (A, b)\n        if len(system) == 2 and isinstance(system[0], Matrix):\n            A, b = system\n\n        # 2). (eq1, eq2, ...)\n        if not isinstance(system[0], Matrix):\n            if sym_gen or not symbols:\n                raise ValueError(filldedent('''\n                    When passing a system of equations, the explicit\n                    symbols for which a solution is being sought must\n                    be given as a sequence, too.\n                '''))\n            system = [\n                _mexpand(i.lhs - i.rhs if isinstance(i, Eq) else i,\n                recursive=True) for i in system]\n            system, symbols, swap = recast_to_symbols(system, symbols)\n            A, b = linear_eq_to_matrix(system, symbols)\n            syms_needed_msg = 'free symbols in the equations provided'\n\n    elif isinstance(system, Matrix) and not (\n            symbols and not isinstance(symbols, GeneratorType) and\n            isinstance(symbols[0], Matrix)):\n        # 3). A augmented with b\n        A, b = system[:, :-1], system[:, -1:]\n\n    if b is None:\n        raise ValueError(\"Invalid arguments\")\n\n    syms_needed_msg  = syms_needed_msg or 'columns of A'\n\n    if sym_gen:\n        symbols = [next(symbols) for i in range(A.cols)]\n        if any(set(symbols) & (A.free_symbols | b.free_symbols)):\n            raise ValueError(filldedent('''\n                At least one of the symbols provided\n                already appears in the system to be solved.\n                One way to avoid this is to use Dummy symbols in\n                the generator, e.g. numbered_symbols('%s', cls=Dummy)\n            ''' % symbols[0].name.rstrip('1234567890')))\n\n    try:\n        solution, params, free_syms = A.gauss_jordan_solve(b, freevar=True)\n    except ValueError:\n        # No solution\n        return S.EmptySet\n\n    # Replace free parameters with free symbols\n    if params:\n        if not symbols:\n            symbols = [_ for _ in params]\n            # re-use the parameters but put them in order\n            # params       [x, y, z]\n            # free_symbols [2, 0, 4]\n            # idx          [1, 0, 2]\n            idx = list(zip(*sorted(zip(free_syms, range(len(free_syms))))))[1]\n            # simultaneous replacements {y: x, x: y, z: z}\n            replace_dict = dict(zip(symbols, [symbols[i] for i in idx]))\n        elif len(symbols) >= A.cols:\n            replace_dict = {v: symbols[free_syms[k]] for k, v in enumerate(params)}\n        else:\n            raise IndexError(filldedent('''\n                the number of symbols passed should have a length\n                equal to the number of %s.\n                ''' % syms_needed_msg))\n        solution = [sol.xreplace(replace_dict) for sol in solution]\n\n    solution = [simplify(sol).xreplace(swap) for sol in solution]\n    return FiniteSet(tuple(solution))"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}