{
  "Selected_candidate": {
    "pr_number": 8820,
    "pr_title": "Check for k=-1 branch real solutions of the Lambert equation",
    "pr_body": "fixes #8708\n",
    "issue_id": 8708,
    "issue_title": "Wrong solution for a simple inequality",
    "issue_body": "I have tried to solve the following rather simple inequality:\n\n``` python\n>>> x = symbols('x', real=True, positive=True)\n>>> solve(1-x-exp(-2*x)>0)\nFalse\n```\n\nand the result (`False`) was quite surprising, and indeed wrong, as, for example,\n\n``` python\n>>> (1-x-exp(-2*x)).subs(x, S(1)/2).evalf()\n0.132120558828558\n```\n\nwhich looks positive...\n",
    "issue_closed_at": "2015-01-15T13:13:59Z",
    "base_commit": "d0d234b90289c5ed4b214fa38aed901244751729",
    "changes": [
      {
        "file": "sympy/functions/elementary/exponential.py",
        "type": "function",
        "name": "eval",
        "class_name": "LambertW",
        "code": "def eval(cls, x, k=None):\n        if k is S.Zero:\n            return cls(x)\n        elif k is None:\n            k = S.Zero\n\n        if k is S.Zero:\n            if x is S.Zero:\n                return S.Zero\n            if x is S.Exp1:\n                return S.One\n            if x == -1/S.Exp1:\n                return S.NegativeOne\n            if x == -log(2)/2:\n                return -log(2)\n            if x is S.Infinity:\n                return S.Infinity\n\n        if k.is_nonzero:\n            if x is S.Zero:\n                return S.NegativeInfinity\n        if k is S.NegativeOne:\n            if x == -S.Pi/2:\n                return -S.ImaginaryUnit*S.Pi/2\n            elif x == -1/S.Exp1:\n                return S.NegativeOne"
      },
      {
        "file": "sympy/functions/elementary/exponential.py",
        "type": "function",
        "name": "_eval_is_real",
        "class_name": "LambertW",
        "code": "def _eval_is_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            return (x + 1/S.Exp1).is_positive\n        elif (k + 1).is_zero:\n            from sympy.core.logic import fuzzy_and\n            return fuzzy_and([x.is_negative, (x + 1/S.Exp1).is_positive])\n        elif k.is_nonzero and (k + 1).is_nonzero:\n            return False"
      },
      {
        "file": "sympy/solvers/bivariate.py",
        "type": "function",
        "name": "_lambert",
        "class_name": null,
        "code": "def _lambert(eq, x):\n    \"\"\"\n    Given an expression assumed to be in the form\n        ``F(X, a..f) = a*log(b*X + c) + d*X + f = 0``\n    where X = g(x) and x = g^-1(X), return the Lambert solution if possible:\n        ``x = g^-1(-c/b + (a/d)*W(d/(a*b)*exp(c*d/a/b)*exp(-f/a)))``.\n    \"\"\"\n    eq = _mexpand(expand_log(eq))\n    mainlog = _mostfunc(eq, log, x)\n    if not mainlog:\n        return []  # violated assumptions\n    other = eq.subs(mainlog, 0)\n    if (-other).func is log:\n        eq = (eq - other).subs(mainlog, mainlog.args[0])\n        mainlog = mainlog.args[0]\n        if mainlog.func is not log:\n            return []  # violated assumptions\n        other = -(-other).args[0]\n        eq += other\n    if not x in other.free_symbols:\n        return [] # violated assumptions\n    d, f, X2 = _linab(other, x)\n    logterm = collect(eq - other, mainlog)\n    a = logterm.as_coefficient(mainlog)\n    if a is None or x in a.free_symbols:\n        return []  # violated assumptions\n    logarg = mainlog.args[0]\n    b, c, X1 = _linab(logarg, x)\n    if X1 != X2:\n        return []  # violated assumptions\n\n    u = Dummy('rhs')\n    rhs = -c/b + (a/d)*LambertW(d/(a*b)*exp(c*d/a/b)*exp(-f/a))\n\n    # if W's arg is between -1/e and 0 there is a -1 branch solution, too.\n\n    # Check here to see if exp(W(s)) appears and return s/W(s) instead?\n\n    solns = solve(X1 - u, x)\n    for i, tmp in enumerate(solns):\n        solns[i] = tmp.subs(u, rhs)\n    return solns"
      }
    ]
  },
  "Justification": "Candidate E is selected because it addresses a bug related to assumptions and evaluations of expressions involving real symbols, which directly relates to the CURRENT bug report's issue with piecewise expressions and symbolic evaluation. Both reports involve the underlying assumptions about the variables (realness, positivity) leading to unexpected errors when using symbolic functions like `subs`. This similarity in the nature of the errors and the resolution approach—understanding how assumptions affect the behavior of symbolic expressions—makes this report particularly helpful for debugging the CURRENT bug.",
  "instance_id": "sympy__sympy-21379",
  "repo": "sympy/sympy",
  "created_at": "2021-04-24T19:49:52Z",
  "problem_statement": "Unexpected `PolynomialError` when using simple `subs()` for particular expressions\nI am seeing weird behavior with `subs` for particular expressions with hyperbolic sinusoids with piecewise arguments. When applying `subs`, I obtain an unexpected `PolynomialError`. For context, I was umbrella-applying a casting from int to float of all int atoms for a bunch of random expressions before using a tensorflow lambdify to avoid potential tensorflow type errors. You can pretend the expression below has a `+ 1` at the end, but below is the MWE that I could produce.\r\n\r\nSee the expression below, and the conditions in which the exception arises.\r\n\r\nSympy version: 1.8.dev\r\n\r\n```python\r\nfrom sympy import *\r\nfrom sympy.core.cache import clear_cache\r\n\r\nx, y, z = symbols('x y z')\r\n\r\nclear_cache()\r\nexpr = exp(sinh(Piecewise((x, y > x), (y, True)) / z))\r\n# This works fine\r\nexpr.subs({1: 1.0})\r\n\r\nclear_cache()\r\nx, y, z = symbols('x y z', real=True)\r\nexpr = exp(sinh(Piecewise((x, y > x), (y, True)) / z))\r\n# This fails with \"PolynomialError: Piecewise generators do not make sense\"\r\nexpr.subs({1: 1.0})  # error\r\n# Now run it again (isympy...) w/o clearing cache and everything works as expected without error\r\nexpr.subs({1: 1.0})\r\n```\r\n\r\nI am not really sure where the issue is, but I think it has something to do with the order of assumptions in this specific type of expression. Here is what I found-\r\n\r\n- The error only (AFAIK) happens with `cosh` or `tanh` in place of `sinh`, otherwise it succeeds\r\n- The error goes away if removing the division by `z`\r\n- The error goes away if removing `exp` (but stays for most unary functions, `sin`, `log`, etc.)\r\n- The error only happens with real symbols for `x` and `y` (`z` does not have to be real)\r\n\r\nNot too sure how to debug this one.\n",
  "patch": "diff --git a/sympy/core/mod.py b/sympy/core/mod.py\n--- a/sympy/core/mod.py\n+++ b/sympy/core/mod.py\n@@ -40,6 +40,7 @@ def eval(cls, p, q):\n         from sympy.core.mul import Mul\n         from sympy.core.singleton import S\n         from sympy.core.exprtools import gcd_terms\n+        from sympy.polys.polyerrors import PolynomialError\n         from sympy.polys.polytools import gcd\n \n         def doit(p, q):\n@@ -166,10 +167,13 @@ def doit(p, q):\n         # XXX other possibilities?\n \n         # extract gcd; any further simplification should be done by the user\n-        G = gcd(p, q)\n-        if G != 1:\n-            p, q = [\n-                gcd_terms(i/G, clear=False, fraction=False) for i in (p, q)]\n+        try:\n+            G = gcd(p, q)\n+            if G != 1:\n+                p, q = [gcd_terms(i/G, clear=False, fraction=False)\n+                        for i in (p, q)]\n+        except PolynomialError:  # issue 21373\n+            G = S.One\n         pwas, qwas = p, q\n \n         # simplify terms\n"
}