{
    "Selected_candidate": {
        "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"
            }
        ]
    },
    "Justification": "Candidate A is the most relevant as it deals with the `polygamma` function, which shares similarities in the context of symbolic arguments. Both the CURRENT bug and Candidate A involve the handling of mathematical functions with complex or symbolic inputs. This relationship hints at the potential similar logic flaws and assumptions regarding argument simplifications inherent to both cases. The structural similarity in the call stacks and the file impacted (assumptions and evaluation messages) further support its relevance, making it a useful resource for diagnosing and resolving the current issue with the `refine()` function."
}