{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 8707,
      "pr_title": "Fix real assumption for gamma function.",
      "pr_body": "- [x] Updated code as suggested by Sergey.\n\n@pelegm @skirpichev \n",
      "issue_id": 8657,
      "issue_title": "gamma(x) is assumed (wrongly) to be real when x is real",
      "issue_body": "It may be that `x` is -1, for example, where `gamma` is not defined (or rather is `zoo`). So, unless it is known that `x` is not a nonpositive integer, `gamma(x).is_real` should return `None` (currently it returns `True`).\n",
      "issue_closed_at": "2014-12-31T11:14:57Z",
      "base_commit": "ed054cc55f714dab9e809036151f0ca136397604",
      "changes": [
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "function",
          "name": "_eval_conjugate",
          "class_name": "loggamma",
          "code": "def _eval_conjugate(self):\n        z = self.args[0]\n        if not z in (S.Zero, S.NegativeInfinity):\n            return self.func(z.conjugate())"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 12882,
      "pr_title": "Checker functions for orthogonality of coordinate system",
      "pr_body": "This PR introduce additional condition for transformation equation which must be fulfill. \r\nIn `vector` module we are dealing with orthogonal curvilinear coordinate system only, so we need to check if transformation equations are correctly defined.\r\n\r\nCloses #12852",
      "issue_id": 12852,
      "issue_title": "Vector module: make sure that the base vectors remain orthogonal",
      "issue_body": "Given a generic transformation equation, make sure that the base vectors remain orthogonal, if not, raise an exception.",
      "issue_closed_at": "2017-07-05T15:10:49Z",
      "base_commit": "c87c0fbf1223c28db8590f2591f64db75d5bdd66",
      "changes": [
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "line",
          "name": "line 4",
          "code": "from sympy.core.cache import cacheit\nfrom sympy.core import S\nfrom sympy.vector.scalar import BaseScalar\nfrom sympy import eye, trigsimp, ImmutableMatrix as Matrix, Symbol, sin, cos, sqrt, diff, Tuple\nimport sympy.vector\nfrom sympy import simplify\nfrom sympy.vector.orienters import (Orienter, AxisOrienter, BodyOrienter,"
        },
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "function",
          "name": "_connect_to_standard_cartesian",
          "class_name": "CoordSys3D",
          "code": "def _connect_to_standard_cartesian(self, curv_coord_type):\n        \"\"\"\n        Change the type of orthogonal curvilinear system. It could be done\n        by tuple of transformation equations or by choosing one of pre-defined\n        coordinate system.\n\n        Parameters\n        ==========\n\n        :param curv_coord_type: str, tuple\n\n        \"\"\"\n        if isinstance(curv_coord_type, string_types):\n            self._set_transformation_equations_mapping(curv_coord_type)\n            self._set_lame_coefficient_mapping(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 3:\n            self._transformation_eqs = curv_coord_type\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 2:\n            self._transformation_eqs = \\\n            tuple([eq.subs({curv_coord_type[0][0]: self.x,\n                            curv_coord_type[0][1]: self.y,\n                            curv_coord_type[0][2]: self.z}) for eq in curv_coord_type[1]])\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(self._transformation_equations())\n\n        else:\n            raise ValueError(\"Wrong set of parameter.\")"
        }
      ]
    },
    {
      "pr_number": 8682,
      "pr_title": "_random: added ValueError to handled exceptions (fixes #8662)",
      "pr_body": "I think this completely resolves the issue, and it does not break any tests. I'm not sure it is the best solution because I'm not overwhelmingly familiar with the codebase. I could have also fixed this by making the _new_args_filter() function in miscellaneous.py throw a TypeError - which _random already handles - instead of a ValueError.\n",
      "issue_id": 8662,
      "issue_title": "Expr._random() fails to catch ValueError exception",
      "issue_body": "When I run the following code, it fails with a \"ValueError\" exception. The problem is that is_constant calls _random, which eventually calls MinMaxBase._new_args_filter in miscellaneous.py. This function has an explicit check for conditions which can result in a ValueError, but _random only catches TypeError.\n\nI'm happy to fix this and make a pull request, but I am not sure what the correct fix is. I currently added ValueError to the list of exceptions caught by _random so that it just returns None in this case. It could be that instead I should make _new_args_filter throw a TypeError instead of a ValueError.\n\n```\n# test sympy problem\nfrom sympy import S\nproblem_expr_s = \"\"\"\n    -SPANCALC_OFF + Piecewise((SPANCALC_PCT, And(Pcl_Loop_Type > 0, \n    sqrt((-Pcl_Kink1_XY_Offset - (-Neck1L_x1*Pcl_Loop_Height - \n    Pcl_Loop_Height**2*neck1L_x2 + Pcl_Loop_Height - neck1L_c)*\n    tan(pi*(-Pcl_Loop_Height**2*neck2A_x2/180 - Pcl_Loop_Height*neck2A_x1/180 \n    - neck2A_c/180 + 1/2)) + Piecewise((Pcl_Span01_XY_Offset + \n    Piecewise((Min(bond_dist, Max(Pcl_Span_Percent*bond_dist, (-Neck1L_x1*\n    Pcl_Loop_Height - Pcl_Loop_Height**2*neck1L_x2 + Pcl_Loop_Height - neck1L_c)\n    *tan(pi*(-Pcl_Loop_Height**2*neck2A_x2/180 - Pcl_Loop_Height*neck2A_x1/180 - \n    neck2A_c/180 + 1/2)))), Pcl_Loop_Type > 0), ((-Neck1L_x1*Pcl_Loop_Height - \n    Pcl_Loop_Height**2*neck1L_x2 + Pcl_Loop_Height - neck1L_c)*tan(pi*(\n    -Pcl_Loop_Height**2*neck2A_x2/180 - Pcl_Loop_Height*neck2A_x1/180 - \n    neck2A_c/180 + 1/2)), True)), Pcl_Loop_Type > 0), (Pcl_Kink1_XY_Offset + \n    (-Neck1L_x1*Pcl_Loop_Height - Pcl_Loop_Height**2*neck1L_x2 + Pcl_Loop_Height\n    - neck1L_c)*tan(pi*(-Pcl_Loop_Height**2*neck2A_x2/180 - Pcl_Loop_Height*\n    neck2A_x1/180 - neck2A_c/180 + 1/2)), True)))**2 + (-Pcl_Kink1_Z_Offset - \n    Pcl_Loop_Height + Max(-pcl_bond_height_diff, Pcl_Span01_Z_Offset + Max(\n    Pcl_Loop_Height, -pcl_bond_height_diff)))**2) > sqrt((Pcl_Kink1_Z_Offset + \n    Pcl_Loop_Height - Max(0, Neck1L_x1*Pcl_Loop_Height + Pcl_Kink2_Z_Offset + \n    Pcl_Loop_Height**2*neck1L_x2 + neck1L_c))**2 + (Pcl_Kink1_XY_Offset - \n    Pcl_Kink2_XY_Offset - pcl_dflt_kink2_xy + (-Neck1L_x1*Pcl_Loop_Height - \n    Pcl_Loop_Height**2*neck1L_x2 + Pcl_Loop_Height - neck1L_c)*tan(pi*(\n    -Pcl_Loop_Height**2*neck2A_x2/180 - Pcl_Loop_Height*neck2A_x1/180 - \n    neck2A_c/180 + 1/2)))**2))), (SPANCALC_OFF, True))\n\"\"\"\nproblem_expr = S(problem_expr_s)\n# the real problem occured in a different context, but this will replicate the problem\nproblem_expr.is_constant(simplify=False)\n# this is the problem call:\nproblem_expr._random()\n```\n",
      "issue_closed_at": "2015-01-10T09:44:19Z",
      "base_commit": "1fff22f3950a2125580fcc4b1760b18f3ce8874b",
      "changes": [
        {
          "file": "sympy/core/expr.py",
          "type": "function",
          "name": "_random",
          "class_name": "Expr",
          "code": "def _random(self, n=None, re_min=-1, im_min=-1, re_max=1, im_max=1):\n        \"\"\"Return self evaluated, if possible, replacing free symbols with\n        random complex values, if necessary.\n\n        The random complex value for each free symbol is generated\n        by the random_complex_number routine giving real and imaginary\n        parts in the range given by the re_min, re_max, im_min, and im_max\n        values. The returned value is evaluated to a precision of n\n        (if given) else the maximum of 15 and the precision needed\n        to get more than 1 digit of precision. If the expression\n        could not be evaluated to a number, or could not be evaluated\n        to more than 1 digit of precision, then None is returned.\n\n        Examples\n        ========\n\n        >>> from sympy import sqrt\n        >>> from sympy.abc import x, y\n        >>> x._random()                         # doctest: +SKIP\n        0.0392918155679172 + 0.916050214307199*I\n        >>> x._random(2)                        # doctest: +SKIP\n        -0.77 - 0.87*I\n        >>> (x + y/2)._random(2)                # doctest: +SKIP\n        -0.57 + 0.16*I\n        >>> sqrt(2)._random(2)\n        1.4\n\n        See Also\n        ========\n\n        sympy.utilities.randtest.random_complex_number\n        \"\"\"\n\n        free = self.free_symbols\n        prec = 1\n        if free:\n            from sympy.utilities.randtest import random_complex_number\n            a, c, b, d = re_min, re_max, im_min, im_max\n            reps = dict(list(zip(free, [random_complex_number(a, b, c, d, rational=True)\n                           for zi in free])))\n            try:\n                nmag = abs(self.evalf(2, subs=reps))\n            except TypeError:\n                # if an out of range value resulted in evalf problems\n                # then return None -- XXX is there a way to know how to\n                # select a good random number for a given expression?\n                # e.g. when calculating n! negative values for n should not\n                # be used\n                return None\n        else:\n            reps = {}\n            nmag = abs(self.evalf(2))\n\n        if not hasattr(nmag, '_prec'):\n            # e.g. exp_polar(2*I*pi) doesn't evaluate but is_number is True\n            return None\n\n        if nmag._prec == 1:\n            # increase the precision up to the default maximum\n            # precision to see if we can get any significance\n\n            from mpmath.libmp.libintmath import giant_steps\n            from sympy.core.evalf import DEFAULT_MAXPREC as target\n\n            # evaluate\n            for prec in giant_steps(2, target):\n                nmag = abs(self.evalf(prec, subs=reps))\n                if nmag._prec != 1:\n                    break\n\n        if nmag._prec != 1:\n            if n is None:\n                n = max(prec, 15)\n            return self.evalf(n, subs=reps)\n\n        # never got any significance\n        return None"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}