{
  "instance_id": "sympy__sympy-21612",
  "repo": "sympy/sympy",
  "created_at": "2021-06-14T04:31:24Z",
  "problem_statement": "Latex parsing of fractions yields wrong expression due to missing brackets\nProblematic latex expression: `\"\\\\frac{\\\\frac{a^3+b}{c}}{\\\\frac{1}{c^2}}\"`\r\n\r\nis parsed to: `((a**3 + b)/c)/1/(c**2)`.\r\n\r\nExpected is: `((a**3 + b)/c)/(1/(c**2))`. \r\n\r\nThe missing brackets in the denominator result in a wrong expression.\r\n\r\n## Tested on\r\n\r\n- 1.8\r\n- 1.6.2\r\n\r\n## Reproduce:\r\n\r\n```\r\nroot@d31ef1c26093:/# python3\r\nPython 3.6.9 (default, Jan 26 2021, 15:33:00)\r\n[GCC 8.4.0] on linux\r\nType \"help\", \"copyright\", \"credits\" or \"license\" for more information.\r\n>>> from sympy.parsing.latex import parse_latex\r\n>>> parse_latex(\"\\\\frac{\\\\frac{a^3+b}{c}}{\\\\frac{1}{c^2}}\")\r\n((a**3 + b)/c)/1/(c**2)\r\n\r\n\n",
  "patch": "diff --git a/sympy/printing/str.py b/sympy/printing/str.py\n--- a/sympy/printing/str.py\n+++ b/sympy/printing/str.py\n@@ -333,7 +333,7 @@ def apow(i):\n                     b.append(apow(item))\n                 else:\n                     if (len(item.args[0].args) != 1 and\n-                            isinstance(item.base, Mul)):\n+                            isinstance(item.base, (Mul, Pow))):\n                         # To avoid situations like #14160\n                         pow_paren.append(item)\n                     b.append(item.base)\n",
  "similar_bug_items": [
    {
      "pr_number": 13437,
      "pr_title": "Fix bell(n) for n=oo, and negative/non-integer 'n'",
      "pr_body": "This PR uses commits of #9198 (maintaining ownership to original author). And tries to address the requested changes on PR #9198 .\r\nFixes #9184",
      "issue_id": 9184,
      "issue_title": "bell(n).limit(n, oo) should be oo rather than bell(oo)",
      "issue_body": "`bell(n).limit(n,oo)` should take the value infinity, but the current output is `bell(oo)`. As the Bell numbers represent the number of partitions of a set, it seems natural that `bell(oo)` should be able to be evaluated rather than be returned unevaluated. This issue is also in line with the recent fixes to the corresponding limit for the Fibonacci numbers and Lucas numbers.\n\n```\nfrom sympy import *\nn = symbols('n')\nbell(n).limit(n,oo)\n\nOutput:\nbell(oo)\n```\n\nI'm new to Sympy, so I'd appreciate the opportunity to fix this bug myself if that's alright.\n",
      "issue_closed_at": "2017-10-13T04:35:13Z",
      "base_commit": "674afc619d7f5c519b6a5393a8b0532a131e57e0",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/numbers.py",
          "type": "function",
          "name": "_bell_incomplete_poly",
          "class_name": "bell",
          "code": "def _bell_incomplete_poly(n, k, symbols):\n        r\"\"\"\n        The second kind of Bell polynomials (incomplete Bell polynomials).\n\n        Calculated by recurrence formula:\n\n        .. math:: B_{n,k}(x_1, x_2, \\dotsc, x_{n-k+1}) =\n                \\sum_{m=1}^{n-k+1}\n                \\x_m \\binom{n-1}{m-1} B_{n-m,k-1}(x_1, x_2, \\dotsc, x_{n-m-k})\n\n        where\n            B_{0,0} = 1;\n            B_{n,0} = 0; for n>=1\n            B_{0,k} = 0; for k>=1\n\n        \"\"\"\n        if (n == 0) and (k == 0):\n            return S.One\n        elif (n == 0) or (k == 0):\n            return S.Zero\n        s = S.Zero\n        a = S.One\n        for m in range(1, n - k + 2):\n            s += a * bell._bell_incomplete_poly(\n                n - m, k - 1, symbols) * symbols[m - 1]\n            a = a * (n - m) / m\n        return expand_mul(s)"
        }
      ]
    },
    {
      "pr_number": 13131,
      "pr_title": "Fix powsimp issues #9183, #10095, and KeyError",
      "pr_body": "- Fixed powsimp raising ValueError when base is float and Pow is argument of Mul. Since multiplicity expects integer or rational but powsimp doesn't checks it, it raises ValueError: fixes #9183\r\n- Fixed Pow.as_numer_denom() returning nan when either numerator or denominator of base is 1 and exp is infinite. Since 1**oo is NaN, numerator or denominator becomes NaN: fixes #10095\r\n- Fixed powsimp raising KeyError in common_b[b] = common_b[b] + e, when one base is Float and another is equal Rational(Integer). This seems to be caused by #11707.\r\n\r\n~**Thing(s) to consider:** Should we automatically evaluate k**oo to zero where k is in (-1, 1), i.e., `(1/(2*E))**oo`?~",
      "issue_id": 10095,
      "issue_title": "simplify((1/(2*E))**oo) returns `nan`",
      "issue_body": "```\n>>> (1/(2*E))**oo\n(1/(2*E))**oo\n>>> simplify((1/(2*E))**oo)\nnan\n>>> powsimp((1/(2*E))**oo)\n(1/(2*E))**oo\n```\n",
      "issue_closed_at": "2017-09-09T23:38:26Z",
      "base_commit": "dceb708ca035c568c816d9457af1b7ca9e57c0a5",
      "changes": [
        {
          "file": "sympy/core/power.py",
          "type": "function",
          "name": "as_numer_denom",
          "class_name": "Pow",
          "code": "def as_numer_denom(self):\n        if not self.is_commutative:\n            return self, S.One\n        base, exp = self.as_base_exp()\n        n, d = base.as_numer_denom()\n        # this should be the same as ExpBase.as_numer_denom wrt\n        # exponent handling\n        neg_exp = exp.is_negative\n        if not neg_exp and not (-exp).is_negative:\n            neg_exp = _coeff_isneg(exp)\n        int_exp = exp.is_integer\n        # the denominator cannot be separated from the numerator if\n        # its sign is unknown unless the exponent is an integer, e.g.\n        # sqrt(a/b) != sqrt(a)/sqrt(b) when a=1 and b=-1. But if the\n        # denominator is negative the numerator and denominator can\n        # be negated and the denominator (now positive) separated.\n        if not (d.is_real or int_exp):\n            n = base\n            d = S.One\n        dnonpos = d.is_nonpositive\n        if dnonpos:\n            n, d = -n, -d\n        elif dnonpos is None and not int_exp:\n            n = base\n            d = S.One\n        if neg_exp:\n            n, d = d, n\n            exp = -exp\n        return self.func(n, exp), self.func(d, exp)"
        },
        {
          "file": "sympy/simplify/powsimp.py",
          "type": "function",
          "name": "recurse",
          "class_name": null,
          "code": "def recurse(arg, **kwargs):\n        _deep = kwargs.get('deep', deep)\n        _combine = kwargs.get('combine', combine)\n        _force = kwargs.get('force', force)\n        _measure = kwargs.get('measure', measure)\n        return powsimp(arg, _deep, _combine, _force, _measure)"
        },
        {
          "file": "sympy/simplify/powsimp.py",
          "type": "function",
          "name": "update",
          "class_name": null,
          "code": "def update(b):\n            '''Decide what to do with base, b. If its exponent is now an\n            integer multiple of the Rational denominator, then remove it\n            and put the factors of its base in the common_b dictionary or\n            update the existing bases if necessary. If it has been zeroed\n            out, simply remove the base.\n            '''\n            newe, r = divmod(common_b[b], b[1])\n            if not r:\n                common_b.pop(b)\n                if newe:\n                    for m in Mul.make_args(b[0]**newe):\n                        b, e = bkey(m)\n                        if b not in common_b:\n                            common_b[b] = 0\n                        common_b[b] += e\n                        if b[1] != 1:\n                            bases.append(b)"
        }
      ]
    },
    {
      "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": 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": 11947,
      "pr_title": "aligned middle of matrix with baseline",
      "pr_body": "**Previous Output**: \r\n\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n2\u22c5\u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\n\r\n**Output after this PR**:\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n  \u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n2.\u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\nFixes #10472 . ",
      "issue_id": 10472,
      "issue_title": "pprint should align the middle of the matrix to the baseline?",
      "issue_body": "Consider the current behaviour, where operators align with the top of the matrix:\n\n```\n>>> Z = Matrix(5,5, lambda i,j: i+j)\n>>> pprint(MatMul(2,Z))\n2\u22c5\u23a10  1  2  3  4\u23a4\n  \u23a2             \u23a5\n  \u23a21  2  3  4  5\u23a5\n  \u23a2             \u23a5\n  \u23a22  3  4  5  6\u23a5\n  \u23a2             \u23a5\n  \u23a23  4  5  6  7\u23a5\n  \u23a2             \u23a5\n  \u23a34  5  6  7  8\u23a6\n>>> pprint(MatAdd(Z,Z))\n\u23a10  1  2  3  4\u23a4 + \u23a10  1  2  3  4\u23a4\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a21  2  3  4  5\u23a5   \u23a21  2  3  4  5\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a22  3  4  5  6\u23a5   \u23a22  3  4  5  6\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a23  4  5  6  7\u23a5   \u23a23  4  5  6  7\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a34  5  6  7  8\u23a6   \u23a34  5  6  7  8\u23a6\n```\n\nThis looks poor, but I figured it might be the desired behaviour...\n\nHere's the example that makes me think its wrong:\n\n```\n>>> pprint(sin((1+1/x)/(1+1/y)) + det(MatMul(2,Z)))\n   \u239b    1\u239e                       \n   \u239c1 + \u2500\u239f                       \n   \u239c    x\u239f                       \nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + 32\u22c5\u2502\u23a10  1  2  3  4\u23a4\u2502\n   \u239c    1\u239f      \u2502\u23a2             \u23a5\u2502\n   \u239c1 + \u2500\u239f      \u2502\u23a21  2  3  4  5\u23a5\u2502\n   \u239d    y\u23a0      \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a22  3  4  5  6\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a23  4  5  6  7\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a34  5  6  7  8\u23a6\u2502\n```\n\nHere's how I think it should work, using the proposed #10423:\n\n```\npprint(sin((1+1/x)/y) + Trace(Z))\n               \u239b\u23a10  1  2  3  4\u23a4\u239e\n   \u239b    1\u239e     \u239c\u23a2             \u23a5\u239f\n   \u239c1 + \u2500\u239f     \u239c\u23a21  2  3  4  5\u23a5\u239f\n   \u239c    x\u239f     \u239c\u23a2             \u23a5\u239f\nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + tr\u239c\u23a22  3  4  5  6\u23a5\u239f\n   \u239d  y  \u23a0     \u239c\u23a2             \u23a5\u239f\n               \u239c\u23a23  4  5  6  7\u23a5\u239f\n               \u239c\u23a2             \u23a5\u239f\n               \u239d\u23a34  5  6  7  8\u23a6\u23a0\n```\n\nwhere note the `sin` alignment is by-design, and (relevant to this bug) the `+` and `tr` operator line up with `sin`.\n\nAt least, det, MatAdd, MatMul, MatPow would need fixed, maybe others I haven't thought of.\n",
      "issue_closed_at": "2016-12-22T06:52:11Z",
      "base_commit": "9a724a42c033c1aae97064947a0f44ec3b922d73",
      "changes": [
        {
          "file": "sympy/matrices/expressions/matadd.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matadd):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatAdd, pprint\n    >>> from sympy.matrices.expressions.matadd import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = eye(2)\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatAdd(A, B, C)\n    >>> pprint(X)\n    A + [1  0] + [1  2]\n        [    ]   [    ]\n        [0  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    A + [2  2]\n        [    ]\n        [3  5]\n    \"\"\"\n    groups = sift(matadd.args, lambda arg: isinstance(arg, MatrixBase))\n    if len(groups[True]) > 1:\n        return MatAdd(*(groups[False] + [reduce(add, groups[True])]))\n    else:\n        return matadd"
        },
        {
          "file": "sympy/matrices/expressions/matmul.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matmul):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatMul, pprint\n    >>> from sympy.matrices.expressions.matmul import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = Matrix([[1, 1], [1, 1]])\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatMul(A, B, C)\n    >>> pprint(X)\n    A*[1  1]*[1  2]\n      [    ] [    ]\n      [1  1] [3  4]\n    >>> pprint(merge_explicit(X))\n    A*[4  6]\n      [    ]\n      [4  6]\n\n    >>> X = MatMul(B, A, C)\n    >>> pprint(X)\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    \"\"\"\n    if not any(isinstance(arg, MatrixBase) for arg in matmul.args):\n        return matmul\n    newargs = []\n    last = matmul.args[0]\n    for arg in matmul.args[1:]:\n        if isinstance(arg, (MatrixBase, Number)) and isinstance(last, (MatrixBase, Number)):\n            last = last * arg\n        else:\n            newargs.append(last)\n            last = arg\n    newargs.append(last)\n\n    return MatMul(*newargs)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_matrix_contents",
          "class_name": "PrettyPrinter",
          "code": "def _print_matrix_contents(self, e):\n        \"\"\"\n        This method factors out what is essentially grid printing.\n        \"\"\"\n        M = e   # matrix\n        Ms = {}  # i,j -> pretty(M[i,j])\n        for i in range(M.rows):\n            for j in range(M.cols):\n                Ms[i, j] = self._print(M[i, j])\n\n        # h- and v- spacers\n        hsep = 2\n        vsep = 1\n\n        # max width for columns\n        maxw = [-1] * M.cols\n\n        for j in range(M.cols):\n            maxw[j] = max([Ms[i, j].width() for i in range(M.rows)] or [0])\n\n        # drawing result\n        D = None\n\n        for i in range(M.rows):\n\n            D_row = None\n            for j in range(M.cols):\n                s = Ms[i, j]\n\n                # reshape s to maxw\n                # XXX this should be generalized, and go to stringPict.reshape ?\n                assert s.width() <= maxw[j]\n\n                # hcenter it, +0.5 to the right                        2\n                # ( it's better to align formula starts for say 0 and r )\n                # XXX this is not good in all cases -- maybe introduce vbaseline?\n                wdelta = maxw[j] - s.width()\n                wleft = wdelta // 2\n                wright = wdelta - wleft\n\n                s = prettyForm(*s.right(' '*wright))\n                s = prettyForm(*s.left(' '*wleft))\n\n                # we don't need vcenter cells -- this is automatically done in\n                # a pretty way because when their baselines are taking into\n                # account in .right()\n\n                if D_row is None:\n                    D_row = s   # first box in a row\n                    continue\n\n                D_row = prettyForm(*D_row.right(' '*hsep))  # h-spacer\n                D_row = prettyForm(*D_row.right(s))\n\n            if D is None:\n                D = D_row       # first row in a picture\n                continue\n\n            # v-spacer\n            for _ in range(vsep):\n                D = prettyForm(*D.below(' '))\n\n            D = prettyForm(*D.below(D_row))\n\n        if D is None:\n            D = prettyForm('')  # Empty Matrix\n\n        return D"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_MatrixElement",
          "class_name": "PrettyPrinter",
          "code": "def _print_MatrixElement(self, expr):\n        from sympy.matrices import MatrixSymbol\n        from sympy import Symbol\n        if (isinstance(expr.parent, MatrixSymbol)\n                and expr.i.is_number and expr.j.is_number):\n            return self._print(\n                    Symbol(expr.parent.name + '_%d%d'%(expr.i, expr.j)))\n        else:\n            prettyFunc = self._print(expr.parent)\n            prettyIndices = self._print_seq((expr.i, expr.j), delimiter=', '\n                    ).parens(left='[', right=']')[0]\n            pform = prettyForm(binding=prettyForm.FUNC,\n                    *stringPict.next(prettyFunc, prettyIndices))\n\n            # store pform parts so it can be reassembled e.g. when powered\n            pform.prettyFunc = prettyFunc\n            pform.prettyArgs = prettyIndices\n\n            return pform"
        }
      ]
    }
  ]
}