{
  "instance_id": "sympy__sympy-24066",
  "repo": "sympy/sympy",
  "created_at": "2022-09-16T22:58:15Z",
  "problem_statement": "SI._collect_factor_and_dimension() cannot properly detect that exponent is dimensionless\nHow to reproduce:\r\n\r\n```python\r\nfrom sympy import exp\r\nfrom sympy.physics import units\r\nfrom sympy.physics.units.systems.si import SI\r\n\r\nexpr = units.second / (units.ohm * units.farad)\r\ndim = SI._collect_factor_and_dimension(expr)[1]\r\n\r\nassert SI.get_dimension_system().is_dimensionless(dim)\r\n\r\nbuggy_expr = 100 + exp(expr)\r\nSI._collect_factor_and_dimension(buggy_expr)\r\n\r\n# results in ValueError: Dimension of \"exp(second/(farad*ohm))\" is Dimension(time/(capacitance*impedance)), but it should be Dimension(1)\r\n```\n",
  "patch": "diff --git a/sympy/physics/units/unitsystem.py b/sympy/physics/units/unitsystem.py\n--- a/sympy/physics/units/unitsystem.py\n+++ b/sympy/physics/units/unitsystem.py\n@@ -190,10 +190,9 @@ def _collect_factor_and_dimension(self, expr):\n                 dim /= idim**count\n             return factor, dim\n         elif isinstance(expr, Function):\n-            fds = [self._collect_factor_and_dimension(\n-                arg) for arg in expr.args]\n-            return (expr.func(*(f[0] for f in fds)),\n-                    *(d[1] for d in fds))\n+            fds = [self._collect_factor_and_dimension(arg) for arg in expr.args]\n+            dims = [Dimension(1) if self.get_dimension_system().is_dimensionless(d[1]) else d[1] for d in fds]\n+            return (expr.func(*(f[0] for f in fds)), *dims)\n         elif isinstance(expr, Dimension):\n             return S.One, expr\n         else:\n",
  "similar_bug_items": [
    {
      "pr_number": 15519,
      "pr_title": "Exit _eval_derivative_n_times when None is obtained",
      "pr_body": "#### References to other Issues or PRs\r\n\r\nFixes #15518 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nThe method `_eval_derivative_n_times` contains a loop that involves calling `_eval_derivative`. The latter may return None when evaluation is not implemented. In such a case the loop should be aborted (returning None) instead of trying to differentiate None.\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* core\r\n  * Fixed a bug in the creation of higher order derivatives that cannot be evaluated.  \r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15518,
      "issue_title": "Higher order derivatives that cannot be evaluated, like re(x).diff(x, 2), raise an exception",
      "issue_body": "This issue arose in a [comment on another issue]( https://github.com/sympy/sympy/issues/15457#issuecomment-436453327) but since it's actually unrelated, is posted here separately. \r\n```\r\n>>> re(x).diff(x, 2)\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/ubuntu/sympy/sympy/core/expr.py\", line 3047, in diff\r\n    return Derivative(self, *symbols, **assumptions)\r\n  File \"/home/ubuntu/sympy/sympy/core/function.py\", line 1370, in __new__\r\n    obj = expr._eval_derivative_n_times(v, count)\r\n  File \"/home/ubuntu/sympy/sympy/core/basic.py\", line 1688, in _eval_derivative_n_times\r\n    obj2 = obj._accept_eval_derivative(s)\r\nAttributeError: 'NoneType' object has no attribute '_accept_eval_derivative'\r\n```\r\nThis happens whenever a higher-order derivative is requested for a function that does not have an explicit derivative. The expected output is \r\n```\r\nDerivative(re(x), (x, 2))\r\n```\r\nA PR is forthcoming. \r\n",
      "issue_closed_at": "2018-11-24T11:36:42Z",
      "base_commit": "61e5c20c02328815270ddda385f0108a2b40d24d",
      "changes": [
        {
          "file": "sympy/core/basic.py",
          "type": "function",
          "name": "_eval_derivative_n_times",
          "class_name": "Basic",
          "code": "def _eval_derivative_n_times(self, s, n):\n        # This is the default evaluator for derivatives (as called by `diff`\n        # and `Derivative`), it will attempt a loop to derive the expression\n        # `n` times by calling the corresponding `_eval_derivative` method,\n        # while leaving the derivative unevaluated if `n` is symbolic.  This\n        # method should be overridden if the object has a closed form for its\n        # symbolic n-th derivative.\n        from sympy import Integer\n        if isinstance(n, (int, Integer)):\n            obj = self\n            for i in range(n):\n                obj2 = obj._accept_eval_derivative(s)\n                if obj == obj2:\n                    break\n                obj = obj2\n            return obj\n        else:\n            return None"
        }
      ]
    },
    {
      "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": 21928,
      "pr_title": "Fixed issue with determining is_integer for round functions (#21651)",
      "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\n\r\nCloses #21651\r\n\r\n#### Brief description of what is fixed or changed\r\nFor some odd cases, `Mul.is_integer` returned `True` where it should have been `None`.\r\n\r\n#### Other comments\r\nThere are three tests added:\r\n1. To check that the final result in #21651 is correct.\r\n2. To check that the floor is not removed from the expression.\r\n3. To check that `is_integer` returns correctly.\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\n* core\r\n   * Fixed issue where `is_integer` returned incorrectly for multiplications.\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 21651,
      "issue_title": "doit() method *sometimes* ignores floor and ceiling within Sum",
      "issue_body": "**Summary of problem**: the `doit()` method will sometimes (depending on the expression) ignore any `floor` or `ceiling` that is located inside a `Sum`.\r\n\r\n**Example detailing the bug**\r\n\r\nThe following code sets up the problem:\r\n\r\n```\r\nfrom sympy import floor, Sum, Symbol\r\ni = Symbol('i')\r\na = Sum(floor(2*2**(-i)), (i, 1, 2))\r\nb = floor(2*2**(-1)) + floor(2*2**(-2))\r\n```\r\n\r\nNote that these two expressions `a` and `b` are the same, just `a` is written with the summation notation and `b` is written out explicitly. They both equal `1`. \r\n\r\nPrinting these variables give us the expected results:\r\n\r\n![image](https://user-images.githubusercontent.com/10404731/122962576-da35c480-d385-11eb-8927-94dff7f43a0c.png)\r\n\r\nHowever, if we run the `doit()` method on them we see a discrepancy:\r\n\r\n![image](https://user-images.githubusercontent.com/10404731/122962993-2bde4f00-d386-11eb-8fc3-cb365bb5790f.png)\r\n\r\nwhere we see that the `floor` function inside the summation has been ignored. The same issue also arises when using the `ceiling` method (i.e. the `doit()` method ignores it when it is in a summation).\r\n\r\nHowever it seems like this problem is not consistent, e.g. if you remove the `2*` from the expression it works fine. More precisely if you run\r\n\r\n```\r\nfrom sympy import floor, Sum, Symbol\r\ni = Symbol('i')\r\na = Sum(floor(2**(-i)), (i, 1, 2))\r\nb = floor(2**(-1)) + floor(2**(-2))\r\nprint(a.doit())\r\nprint(b.doit())\r\n```\r\nyou get the exptected results (this works for `ceiling` too)\r\n\r\n(bug discovered on sympy version 1.8 on python 3.7.3)",
      "issue_closed_at": "2021-08-24T21:25:21Z",
      "base_commit": "8dd210b000100cb01dc128b5680795c190b247fe",
      "changes": [
        {
          "file": "sympy/core/mul.py",
          "type": "function",
          "name": "_eval_is_integer",
          "class_name": "Mul",
          "code": "def _eval_is_integer(self):\n        is_rational = self._eval_is_rational()\n        if is_rational is False:\n            return False\n\n        numerators = []\n        denominators = []\n        for a in self.args:\n            if a.is_integer:\n                numerators.append(a)\n            elif a.is_Rational:\n                n, d = a.as_numer_denom()\n                numerators.append(n)\n                denominators.append(d)\n            elif a.is_Pow:\n                b, e = a.as_base_exp()\n                if not b.is_integer or not e.is_integer: return\n                if e.is_negative:\n                    denominators.append(b)\n                else:\n                    # for integer b and positive integer e: a = b**e would be integer\n                    assert not e.is_positive\n                    # for self being rational and e equal to zero: a = b**e would be 1\n                    assert not e.is_zero\n                    return # sign of e unknown -> self.is_integer cannot be decided\n            else:\n                return\n\n        if not denominators:\n            return True\n\n        odd = lambda ints: all(i.is_odd for i in ints)\n        even = lambda ints: any(i.is_even for i in ints)\n\n        if odd(numerators) and even(denominators):\n            return False\n        elif even(numerators) and denominators == [2]:\n            return True"
        }
      ]
    },
    {
      "pr_number": 17826,
      "pr_title": "Fix exp.is_complex for infinite extended reals",
      "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\n\r\nFixes #17789 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nFixes exp.is_complex so that it understands that infinite non-complex inputs like oo and -oo do not necessarily mean that exp(x) is not complex.\r\n\r\n#### Other comments\r\n\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 17789,
      "issue_title": "Intermittent test failure in assumptions",
      "issue_body": "This test fails intermittently on Python 3.5 or 3.6:\r\n```console\r\n$ pytest sympy/core/tests/test_arit.py -k test_Add_Mul_is_finite\r\n==================================================================== test session starts =====================================================================\r\nplatform darwin -- Python 3.5.7, pytest-4.3.1, py-1.8.0, pluggy-0.9.0\r\nhypothesis profile 'default' -> database=DirectoryBasedExampleDatabase('/Users/enojb/current/sympy/sympy/.hypothesis/examples')\r\narchitecture: 64-bit\r\ncache:        yes\r\nground types: gmpy 1.17\r\n\r\nrootdir: /Users/enojb/current/sympy/sympy, inifile: pytest.ini\r\nplugins: xdist-1.27.0, instafail-0.4.1, forked-1.0.2, doctestplus-0.3.0, cov-2.7.1, hypothesis-4.32.3\r\ncollected 88 items / 87 deselected / 1 selected                                                                                                              \r\n\r\nsympy/core/tests/test_arit.py F                                                                                                                        [100%]\r\n\r\n========================================================================== FAILURES ==========================================================================\r\n___________________________________________________________________ test_Add_Mul_is_finite ___________________________________________________________________\r\n\r\n    def test_Add_Mul_is_finite():\r\n        x = Symbol('x', extended_real=True, finite=False)\r\n    \r\n        assert sin(x).is_finite is True\r\n        assert (x*sin(x)).is_finite is None\r\n        assert (x*atan(x)).is_finite is False\r\n        assert (1024*sin(x)).is_finite is True\r\n>       assert (sin(x)*exp(x)).is_finite is None\r\nE       assert False is None\r\nE        +  where False = (sin(x) * exp(x)).is_finite\r\nE        +    where sin(x) = sin(x)\r\nE        +    and   exp(x) = exp(x)\r\n\r\nsympy/core/tests/test_arit.py:390: AssertionError\r\n```",
      "issue_closed_at": "2019-10-31T11:56:38Z",
      "base_commit": "cf43f87f5d145f2589cee8a79d0f20bbdf9eb075",
      "changes": [
        {
          "file": "sympy/functions/elementary/exponential.py",
          "type": "line",
          "name": "line 6",
          "code": "from sympy.core.compatibility import range\nfrom sympy.core.function import (Function, ArgumentIndexError, _coeff_isneg,\n        expand_mul)\nfrom sympy.core.logic import fuzzy_and, fuzzy_not\nfrom sympy.core.mul import Mul\nfrom sympy.core.numbers import Integer, Rational\nfrom sympy.core.power import Pow"
        },
        {
          "file": "sympy/functions/elementary/exponential.py",
          "type": "function",
          "name": "_eval_is_extended_real",
          "class_name": "LambertW",
          "code": "def _eval_is_extended_real(self):\n        x = self.args[0]\n        if len(self.args) == 1:\n            k = S.Zero\n        else:\n            k = self.args[1]\n        if k.is_zero:\n            if (x + 1/S.Exp1).is_positive:\n                return True\n            elif (x + 1/S.Exp1).is_nonpositive:\n                return False\n        elif (k + 1).is_zero:\n            if x.is_negative and (x + 1/S.Exp1).is_positive:\n                return True\n            elif x.is_nonpositive or (x + 1/S.Exp1).is_nonnegative:\n                return False\n        elif fuzzy_not(k.is_zero) and fuzzy_not((k + 1).is_zero):\n            if x.is_extended_real:\n                return False"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}