{
  "Selected_candidate": {
    "pr_number": 17447,
    "pr_title": "exp.as_leading_term() returns wrong results",
    "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 #4615 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\n`as_leading_term` does not work properly on exponential functions.\r\nAs noted in the issue #4615 , `exp((x+1)/x**2).as_leading_term(x)`\r\nreturns `exp(x**(-2))`, which is not the correct result.\r\nIndeed when the argument of `exp` tends to `oo`, then it is not\r\ncorrect doing the leading term of the `arg` and then applying the exponential.\r\n\r\nAlso SymPy in the case\r\n`exp((2*x + 3) / (x+1)).as_leading_term(x)`\r\nreturns `S(1)`, which is not correct here, since the argument is closed to `3`.\r\n\r\nIn `test_exponential.py` the following tests are added\r\n```\r\n    assert exp((x + 1) / x**2).as_leading_term(x) == exp((x + 1) / x**2)\r\n    assert exp((2*x + 3) / (x+1)).as_leading_term(x) == exp(3)\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\n\r\n- functions\r\n   - remove a bug for `as_leading_term()` for exponential functions.\r\n\r\n<!-- END RELEASE NOTES -->\r\n",
    "issue_id": 4615,
    "issue_title": "exp.as_leading_term returns wrong results",
    "issue_body": "```\nas_leading_term is implicitly understood (at least by me) to obey the\nfollowing mathematical contract: expr.as_leading_term(x) ~ expr as x -> 0.\nHowever, it fails on the common pitfall that f(x) ~ g(x) doesn't imply\nexp(f(x)) ~ exp(g(x)). Concretely, we have:\n\n>>>exp((x+1)/x**2).as_leading_term(x)\nexp(x**(-2))\n\nbut, of course,\n\n>>>limit(exp((x+1)/x**2)/exp(x**(-2)), x, 0)\noo\n\nObtaining a correct answer requires performing a full asympotic expansion\nto order o(1) which might be outside the scope of this method, so maybe it\nshould return the expression unchanged unless it can easily get a correct\nanswer.\n```\n\nOriginal issue for #4615: http://code.google.com/p/sympy/issues/detail?id=1516\nOriginal author: https://code.google.com/u/101272611947379421629/\n",
    "issue_closed_at": "2019-08-20T14:27:02Z",
    "base_commit": "fc6f766ab588fecbb69ad85eb01ca28b44715e5c",
    "changes": [
      {
        "file": "sympy/functions/elementary/exponential.py",
        "type": "function",
        "name": "_eval_as_leading_term",
        "class_name": "log",
        "code": "def _eval_as_leading_term(self, x):\n        arg = self.args[0].as_leading_term(x)\n        if arg is S.One:\n            return (self.args[0] - 1).as_leading_term(x)\n        return self.func(arg)"
      }
    ]
  },
  "Justification": "Candidate D is selected because it addresses a fundamental issue with the `as_leading_term` method in exponential functions, which is closely related to the CURRENT bug involving inconsistent simplification of expressions with `ceiling`. Both issues revolve around the complexity of symbolic expressions and how they are evaluated under different settings. The potential fixes and testing enhancements implemented in Candidate D may provide valuable insights into ensuring consistent behavior across different evaluation contexts, just as the CURRENT bug seeks to achieve consistency in behavior regarding the `evaluate` parameter. Leveraging the resolution strategies from Candidate D can significantly help in analyzing the evaluation inconsistencies observed in the CURRENT bug report.",
  "instance_id": "sympy__sympy-20322",
  "repo": "sympy/sympy",
  "created_at": "2020-10-22T20:39:24Z",
  "problem_statement": "Inconsistent behavior for sympify/simplify with ceiling\nIn sympy v1.5.1:\r\n```python\r\nIn [16]: sympy.sympify('4*ceiling(x/4 - 3/4)', evaluate=False).simplify()\r\nOut[16]: 4*ceiling(x/4 - 3/4)\r\n\r\nIn [17]: sympy.sympify('4*ceiling(x/4 - 3/4)', evaluate=True).simplify()\r\nOut[17]: 4*ceiling(x/4 - 3/4)\r\n```\r\n\r\nIn sympy v.1.6.2:\r\n```python\r\nIn [16]: sympy.sympify('4*ceiling(x/4 - 3/4)', evaluate=False).simplify()\r\nOut[16]: 4*ceiling(x/4) - 3\r\n\r\nIn [17]: sympy.sympify('4*ceiling(x/4 - 3/4)', evaluate=True).simplify()\r\nOut [17]: 4*ceiling(x/4 - 3/4)\r\n```\r\n\r\nIs there a way to ensure that the behavior is consistent, even though evaluate is equal to `False` when parsing?\n",
  "patch": "diff --git a/sympy/core/mul.py b/sympy/core/mul.py\n--- a/sympy/core/mul.py\n+++ b/sympy/core/mul.py\n@@ -7,7 +7,7 @@\n from .singleton import S\n from .operations import AssocOp, AssocOpDispatcher\n from .cache import cacheit\n-from .logic import fuzzy_not, _fuzzy_group, fuzzy_and\n+from .logic import fuzzy_not, _fuzzy_group\n from .compatibility import reduce\n from .expr import Expr\n from .parameters import global_parameters\n@@ -1262,27 +1262,47 @@ def _eval_is_zero(self):\n                     zero = None\n         return zero\n \n+    # without involving odd/even checks this code would suffice:\n+    #_eval_is_integer = lambda self: _fuzzy_group(\n+    #    (a.is_integer for a in self.args), quick_exit=True)\n     def _eval_is_integer(self):\n-        from sympy import fraction\n-        from sympy.core.numbers import Float\n-\n         is_rational = self._eval_is_rational()\n         if is_rational is False:\n             return False\n \n-        # use exact=True to avoid recomputing num or den\n-        n, d = fraction(self, exact=True)\n-        if is_rational:\n-            if d is S.One:\n-                return True\n-        if d.is_even:\n-            if d.is_prime:  # literal or symbolic 2\n-                return n.is_even\n-            if n.is_odd:\n-                return False  # true even if d = 0\n-        if n == d:\n-            return fuzzy_and([not bool(self.atoms(Float)),\n-            fuzzy_not(d.is_zero)])\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\n \n     def _eval_is_polar(self):\n         has_polar = any(arg.is_polar for arg in self.args)\n"
}