{
  "Selected_candidate": {
    "pr_number": 2707,
    "pr_title": "remove practice of evaluating factorial as 0 for negative numbers",
    "pr_body": "it is infinity for negative integers and remains unevaluated for non-integers. fix gosper test, which had a missing term.\nfix string representations\n",
    "issue_id": 2703,
    "issue_title": "uncommon convention for factorials of negative numbers",
    "issue_body": "from the docstring of the factorial function:\n\n\"For the sake of convenience and simplicity of procedures using\nthis function it is defined for negative integers and returns\nzero in this case.\"\n\nBy convention (and by some definitions), n! is usually infinity for n < 0. This is consistent with the Gamma function identity n! = Gamma(n+1) and with the convention that binomial coefficients are 0 outside of Pascal's triangle. In fact, in the same module, we have, in the doc string for binomial coefficients:\n\n\"For the sake of convenience for negative 'k' this function\nwill return zero no matter what valued is the other argument.\"\n\nThis is incompatible with sympy's convention on factorials, given the definition of the binomial coefficients: C(n,k) = n!/(k!(n-k)!)\n\nFor example take \"2 choose -1\", C(2,-1) = 0. However, going to sympy's definition of the factorial, we would have 2!/((-1)! \\* 3!) == 1/0 == oo, which is obviously wrong. Switching over to the Gamma function gives us the correct answer.\n\nDoes anyone know what \"procedures using\" the factorial function rely on the \"convenience and simplicity\" of the incorrect convention n! = 0  for n < 0? It would nice to bring the factorial function to consistency with other parts of sympy (like the binomial coefficients) and the more common convention in the math and cs communities.\n",
    "issue_closed_at": "2014-01-05T13:28:55Z",
    "base_commit": "5879fa2318e853807a6cf63981f2bca441cceb9f",
    "changes": [
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "_eval_simplify",
        "class_name": "CombinatorialFunction",
        "code": "def _eval_simplify(self, ratio, measure):\n        from sympy.simplify.simplify import combsimp\n        expr = combsimp(self)\n        if measure(expr) <= ratio*measure(self):\n            return expr\n        return self"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "class",
        "name": "factorial",
        "code": "class factorial(CombinatorialFunction):\n    \"\"\"Implementation of factorial function over nonnegative integers.\n       For the sake of convenience and simplicity of procedures using\n       this function it is defined for negative integers and returns\n       zero in this case.\n\n       The factorial is very important in combinatorics where it gives\n       the number of ways in which 'n' objects can be permuted. It also\n       arises in calculus, probability, number theory etc.\n\n       There is strict relation of factorial with gamma function. In\n       fact n! = gamma(n+1) for nonnegative integers. Rewrite of this\n       kind is very useful in case of combinatorial simplification.\n\n       Computation of the factorial is done using two algorithms. For\n       small arguments naive product is evaluated. However for bigger\n       input algorithm Prime-Swing is used. It is the fastest algorithm\n       known and computes n! via prime factorization of special class\n       of numbers, called here the 'Swing Numbers'.\n\n       Examples\n       ========\n\n       >>> from sympy import Symbol, factorial\n       >>> n = Symbol('n', integer=True)\n\n       >>> factorial(-2)\n       0\n\n       >>> factorial(0)\n       1\n\n       >>> factorial(7)\n       5040\n\n       >>> factorial(n)\n       factorial(n)\n\n       >>> factorial(2*n)\n       factorial(2*n)\n\n       See Also\n       ========\n\n       factorial2, RisingFactorial, FallingFactorial\n    \"\"\"\n\n    nargs = 1\n\n    def fdiff(self, argindex=1):\n        if argindex == 1:\n            return C.gamma(self.args[0] + 1)*C.polygamma(0, self.args[0] + 1)\n        else:\n            raise ArgumentIndexError(self, argindex)\n\n    _small_swing = [\n        1, 1, 1, 3, 3, 15, 5, 35, 35, 315, 63, 693, 231, 3003, 429, 6435, 6435, 109395,\n        12155, 230945, 46189, 969969, 88179, 2028117, 676039, 16900975, 1300075,\n        35102025, 5014575, 145422675, 9694845, 300540195, 300540195\n    ]\n\n    @classmethod\n    def _swing(cls, n):\n        if n < 33:\n            return cls._small_swing[n]\n        else:\n            N, primes = int(_sqrt(n)), []\n\n            for prime in sieve.primerange(3, N + 1):\n                p, q = 1, n\n\n                while True:\n                    q //= prime\n\n                    if q > 0:\n                        if q & 1 == 1:\n                            p *= prime\n                    else:\n                        break\n\n                if p > 1:\n                    primes.append(p)\n\n            for prime in sieve.primerange(N + 1, n//3 + 1):\n                if (n // prime) & 1 == 1:\n                    primes.append(prime)\n\n            L_product = R_product = 1\n\n            for prime in sieve.primerange(n//2 + 1, n + 1):\n                L_product *= prime\n\n            for prime in primes:\n                R_product *= prime\n\n            return L_product*R_product\n\n    @classmethod\n    def _recursive(cls, n):\n        if n < 2:\n            return 1\n        else:\n            return (cls._recursive(n//2)**2)*cls._swing(n)\n\n    @classmethod\n    def eval(cls, n):\n        n = sympify(n)\n\n        if n.is_Number:\n            if n is S.Zero:\n                return S.One\n            elif n is S.Infinity:\n                return S.Infinity\n            elif n.is_Integer:\n                if n.is_negative:\n                    return S.Zero\n                else:\n                    n, result = n.p, 1\n\n                    if n < 20:\n                        for i in range(2, n + 1):\n                            result *= i\n                    else:\n                        N, bits = n, 0\n\n                        while N != 0:\n                            if N & 1 == 1:\n                                bits += 1\n\n                            N = N >> 1\n\n                        result = cls._recursive(n)*2**(n - bits)\n\n                    return C.Integer(result)\n\n        if n.is_negative:\n            return S.Zero\n\n    def _eval_rewrite_as_gamma(self, n):\n        return C.gamma(n + 1)\n\n    def _eval_is_integer(self):\n        return self.args[0].is_integer\n\n    def _eval_is_positive(self):\n        if self.args[0].is_integer and self.args[0].is_positive:\n            return True"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "eval",
        "class_name": "binomial",
        "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "eval",
        "class_name": "binomial",
        "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
      }
    ]
  },
  "Justification": "Candidate D is the most helpful because it addresses similar issues of inconsistent definitions in mathematical functions, just like the CURRENT bug report discusses the expected behavior of `expr.atoms()`. Both bugs involve clarifying expected outcomes (the behavior of factorial for negative integers versus leaf nodes in expression trees), which can help the developer understand the principles behind correct function designs and anticipated output. The patch type for D involves ensuring consistency in function behavior, which aligns with the need to review performance and correctness in the CURRENT bug report, making it particularly relevant for debugging and fixing.",
  "instance_id": "sympy__sympy-18532",
  "repo": "sympy/sympy",
  "created_at": "2020-02-01T17:26:30Z",
  "problem_statement": "expr.atoms() should return objects with no args instead of subclasses of Atom\n`expr.atoms()` with no arguments returns subclasses of `Atom` in `expr`. But the correct definition of a leaf node should be that it has no `.args`. \n\nThis should be easy to fix, but one needs to check that this doesn't affect the performance. \n\n",
  "patch": "diff --git a/sympy/core/basic.py b/sympy/core/basic.py\n--- a/sympy/core/basic.py\n+++ b/sympy/core/basic.py\n@@ -503,12 +503,11 @@ def atoms(self, *types):\n         if types:\n             types = tuple(\n                 [t if isinstance(t, type) else type(t) for t in types])\n+        nodes = preorder_traversal(self)\n+        if types:\n+            result = {node for node in nodes if isinstance(node, types)}\n         else:\n-            types = (Atom,)\n-        result = set()\n-        for expr in preorder_traversal(self):\n-            if isinstance(expr, types):\n-                result.add(expr)\n+            result = {node for node in nodes if not node.args}\n         return result\n \n     @property\n"
}