{
  "instance_id": "sympy__sympy-13915",
  "repo": "sympy/sympy",
  "created_at": "2018-01-13T20:41:07Z",
  "problem_statement": "Issue with a substitution that leads to an undefined expression\n```\r\nPython 3.6.4 |Anaconda custom (64-bit)| (default, Dec 21 2017, 15:39:08) \r\nType 'copyright', 'credits' or 'license' for more information\r\nIPython 6.2.1 -- An enhanced Interactive Python. Type '?' for help.\r\n\r\nIn [1]: from sympy import *\r\n\r\nIn [2]: a,b = symbols('a,b')\r\n\r\nIn [3]: r = (1/(a+b) + 1/(a-b))/(1/(a+b) - 1/(a-b))\r\n\r\nIn [4]: r.subs(b,a)\r\nOut[4]: 1\r\n\r\nIn [6]: import sympy\r\n\r\nIn [7]: sympy.__version__\r\nOut[7]: '1.1.1'\r\n```\r\n\r\nIf b is substituted by a, r is undefined. It is possible to calculate the limit\r\n`r.limit(b,a) # -1`\r\n\r\nBut whenever a subexpression of r is undefined, r itself is undefined.\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@@ -423,6 +423,11 @@ def _gather(c_powers):\n             changed = False\n             for b, e in c_powers:\n                 if e.is_zero:\n+                    # canceling out infinities yields NaN\n+                    if (b.is_Add or b.is_Mul) and any(infty in b.args\n+                        for infty in (S.ComplexInfinity, S.Infinity,\n+                                      S.NegativeInfinity)):\n+                        return [S.NaN], [], None\n                     continue\n                 if e is S.One:\n                     if b.is_Number:\n",
  "similar_bug_items": [
    {
      "pr_number": 13437,
      "pr_title": "Fix bell(n) for n=oo, and negative/non-integer 'n'",
      "pr_body": "This PR uses commits of #9198 (maintaining ownership to original author). And tries to address the requested changes on PR #9198 .\r\nFixes #9184",
      "issue_id": 9184,
      "issue_title": "bell(n).limit(n, oo) should be oo rather than bell(oo)",
      "issue_body": "`bell(n).limit(n,oo)` should take the value infinity, but the current output is `bell(oo)`. As the Bell numbers represent the number of partitions of a set, it seems natural that `bell(oo)` should be able to be evaluated rather than be returned unevaluated. This issue is also in line with the recent fixes to the corresponding limit for the Fibonacci numbers and Lucas numbers.\n\n```\nfrom sympy import *\nn = symbols('n')\nbell(n).limit(n,oo)\n\nOutput:\nbell(oo)\n```\n\nI'm new to Sympy, so I'd appreciate the opportunity to fix this bug myself if that's alright.\n",
      "issue_closed_at": "2017-10-13T04:35:13Z",
      "base_commit": "674afc619d7f5c519b6a5393a8b0532a131e57e0",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/numbers.py",
          "type": "function",
          "name": "_bell_incomplete_poly",
          "class_name": "bell",
          "code": "def _bell_incomplete_poly(n, k, symbols):\n        r\"\"\"\n        The second kind of Bell polynomials (incomplete Bell polynomials).\n\n        Calculated by recurrence formula:\n\n        .. math:: B_{n,k}(x_1, x_2, \\dotsc, x_{n-k+1}) =\n                \\sum_{m=1}^{n-k+1}\n                \\x_m \\binom{n-1}{m-1} B_{n-m,k-1}(x_1, x_2, \\dotsc, x_{n-m-k})\n\n        where\n            B_{0,0} = 1;\n            B_{n,0} = 0; for n>=1\n            B_{0,k} = 0; for k>=1\n\n        \"\"\"\n        if (n == 0) and (k == 0):\n            return S.One\n        elif (n == 0) or (k == 0):\n            return S.Zero\n        s = S.Zero\n        a = S.One\n        for m in range(1, n - k + 2):\n            s += a * bell._bell_incomplete_poly(\n                n - m, k - 1, symbols) * symbols[m - 1]\n            a = a * (n - m) / m\n        return expand_mul(s)"
        }
      ]
    },
    {
      "pr_number": 13131,
      "pr_title": "Fix powsimp issues #9183, #10095, and KeyError",
      "pr_body": "- Fixed powsimp raising ValueError when base is float and Pow is argument of Mul. Since multiplicity expects integer or rational but powsimp doesn't checks it, it raises ValueError: fixes #9183\r\n- Fixed Pow.as_numer_denom() returning nan when either numerator or denominator of base is 1 and exp is infinite. Since 1**oo is NaN, numerator or denominator becomes NaN: fixes #10095\r\n- Fixed powsimp raising KeyError in common_b[b] = common_b[b] + e, when one base is Float and another is equal Rational(Integer). This seems to be caused by #11707.\r\n\r\n~**Thing(s) to consider:** Should we automatically evaluate k**oo to zero where k is in (-1, 1), i.e., `(1/(2*E))**oo`?~",
      "issue_id": 10095,
      "issue_title": "simplify((1/(2*E))**oo) returns `nan`",
      "issue_body": "```\n>>> (1/(2*E))**oo\n(1/(2*E))**oo\n>>> simplify((1/(2*E))**oo)\nnan\n>>> powsimp((1/(2*E))**oo)\n(1/(2*E))**oo\n```\n",
      "issue_closed_at": "2017-09-09T23:38:26Z",
      "base_commit": "dceb708ca035c568c816d9457af1b7ca9e57c0a5",
      "changes": [
        {
          "file": "sympy/core/power.py",
          "type": "function",
          "name": "as_numer_denom",
          "class_name": "Pow",
          "code": "def as_numer_denom(self):\n        if not self.is_commutative:\n            return self, S.One\n        base, exp = self.as_base_exp()\n        n, d = base.as_numer_denom()\n        # this should be the same as ExpBase.as_numer_denom wrt\n        # exponent handling\n        neg_exp = exp.is_negative\n        if not neg_exp and not (-exp).is_negative:\n            neg_exp = _coeff_isneg(exp)\n        int_exp = exp.is_integer\n        # the denominator cannot be separated from the numerator if\n        # its sign is unknown unless the exponent is an integer, e.g.\n        # sqrt(a/b) != sqrt(a)/sqrt(b) when a=1 and b=-1. But if the\n        # denominator is negative the numerator and denominator can\n        # be negated and the denominator (now positive) separated.\n        if not (d.is_real or int_exp):\n            n = base\n            d = S.One\n        dnonpos = d.is_nonpositive\n        if dnonpos:\n            n, d = -n, -d\n        elif dnonpos is None and not int_exp:\n            n = base\n            d = S.One\n        if neg_exp:\n            n, d = d, n\n            exp = -exp\n        return self.func(n, exp), self.func(d, exp)"
        },
        {
          "file": "sympy/simplify/powsimp.py",
          "type": "function",
          "name": "recurse",
          "class_name": null,
          "code": "def recurse(arg, **kwargs):\n        _deep = kwargs.get('deep', deep)\n        _combine = kwargs.get('combine', combine)\n        _force = kwargs.get('force', force)\n        _measure = kwargs.get('measure', measure)\n        return powsimp(arg, _deep, _combine, _force, _measure)"
        },
        {
          "file": "sympy/simplify/powsimp.py",
          "type": "function",
          "name": "update",
          "class_name": null,
          "code": "def update(b):\n            '''Decide what to do with base, b. If its exponent is now an\n            integer multiple of the Rational denominator, then remove it\n            and put the factors of its base in the common_b dictionary or\n            update the existing bases if necessary. If it has been zeroed\n            out, simply remove the base.\n            '''\n            newe, r = divmod(common_b[b], b[1])\n            if not r:\n                common_b.pop(b)\n                if newe:\n                    for m in Mul.make_args(b[0]**newe):\n                        b, e = bkey(m)\n                        if b not in common_b:\n                            common_b[b] = 0\n                        common_b[b] += e\n                        if b[1] != 1:\n                            bases.append(b)"
        }
      ]
    },
    {
      "pr_number": 8548,
      "pr_title": "Issue #6988: expand_log(exp(x), force=True) = x",
      "pr_body": "This is a try to fix issue #6988. If I made some mistake just let me know, as I'm still new to sympy development.\n\n```\nIn [9]: expand_log(log(exp(x)), force=True)\nOut[9]: x\n\nIn [10]: expand_log(log(y**(x)), force=True)\nOut[10]: x\u22c5log(y)\n```\n",
      "issue_id": 6988,
      "issue_title": "expand_log(exp(x), force=True) should give x",
      "issue_body": "```\nIn [5]: expand_log(log(exp(x)), force=True)\nOut[5]:\n   \u239b x\u239e\nlog\u239d\u212f \u23a0\n\nIn [6]: expand_log(log(y**(x)), force=True)\nOut[6]: x\u22c5log(y) Issue 1799 is probably to blame.\n```\n\nOriginal issue for #6988: http://code.google.com/p/sympy/issues/detail?id=3889\nOriginal author: https://code.google.com/u/asmeurer@gmail.com/\n",
      "issue_closed_at": "2014-12-03T13:34:14Z",
      "base_commit": "e6fc53f27ee872b27bc79b96529fc4bf34d4f023",
      "changes": [
        {
          "file": "sympy/functions/elementary/exponential.py",
          "type": "function",
          "name": "_eval_expand_log",
          "class_name": "log",
          "code": "def _eval_expand_log(self, deep=True, **hints):\n        from sympy import unpolarify\n        from sympy.concrete import Sum, Product\n        force = hints.get('force', False)\n        arg = self.args[0]\n        if arg.is_Integer:\n            # remove perfect powers\n            p = perfect_power(int(arg))\n            if p is not False:\n                return p[1]*self.func(p[0])\n        elif arg.is_Mul:\n            expr = []\n            nonpos = []\n            for x in arg.args:\n                if force or x.is_positive or x.is_polar:\n                    a = self.func(x)\n                    if isinstance(a, log):\n                        expr.append(self.func(x)._eval_expand_log(**hints))\n                    else:\n                        expr.append(a)\n                elif x.is_negative:\n                    a = self.func(-x)\n                    expr.append(a)\n                    nonpos.append(S.NegativeOne)\n                else:\n                    nonpos.append(x)\n            return Add(*expr) + log(Mul(*nonpos))\n        elif arg.is_Pow:\n            if force or (arg.exp.is_real and arg.base.is_positive) or \\\n                    arg.base.is_polar:\n                b = arg.base\n                e = arg.exp\n                a = self.func(b)\n                if isinstance(a, log):\n                    return unpolarify(e) * a._eval_expand_log(**hints)\n                else:\n                    return unpolarify(e) * a\n        elif isinstance(arg, Product):\n            if arg.function.is_positive:\n                return Sum(log(arg.function), *arg.limits)\n\n        return self.func(arg)"
        }
      ]
    },
    {
      "pr_number": 7928,
      "pr_title": "give MaxMinBase an evalf routine",
      "pr_body": "Although Or, And, etc... will not have a number as an\nargument, Min and Max can. When the operations version\nof _eval_evalf is called a call to as_independent is\nmade. This creates an x (=1) that is assumed to be an\nargument of the object so Max(0,y).n() becomes Max(1, 0, y) -> Max(1, y).\nTo avoid this an evalf routine was added to handle MaxMinBase separately.\n\nfixes #7233\n",
      "issue_id": 7233,
      "issue_title": "N(Max(0.0, n + m)) returns Max(1.0, n + m)",
      "issue_body": "```\nThe following code does not return what is expected,\n\nn = sympy.Symbol('n')\nm = sympy.Symbol('m')\na = sympy.Max(0.0, n + m)\nprint a\nprint sympy.N(a)\n\noutputs:\n>> Max(0.0, m + n)\n>> Max(1.0, m + n)\n```\n\nOriginal issue for #7233: http://code.google.com/p/sympy/issues/detail?id=4134\nOriginal author: https://code.google.com/u/100679098312105897492/\n",
      "issue_closed_at": "2014-08-29T23:58:03Z",
      "base_commit": "d365f932a26c28c4c57b2b9f6dddee2ca94a5c66",
      "changes": [
        {
          "file": "sympy/functions/elementary/miscellaneous.py",
          "type": "function",
          "name": "_eval_derivative",
          "class_name": "MinMaxBase",
          "code": "def _eval_derivative(self, s):\n        # f(x).diff(s) -> x.diff(s) * f.fdiff(1)(s)\n        i = 0\n        l = []\n        for a in self.args:\n            i += 1\n            da = a.diff(s)\n            if da is S.Zero:\n                continue\n            try:\n                df = self.fdiff(i)\n            except ArgumentIndexError:\n                df = Function.fdiff(self, i)\n            l.append(df * da)\n        return Add(*l)"
        }
      ]
    },
    {
      "pr_number": 13441,
      "pr_title": "address issues with _commutative_match and _combine_inverse",
      "pr_body": "fixes #10979 (as best as I can see how); I suspect the slowdown is coming from the `_should_evalf` procedure rather than `count_ops`.",
      "issue_id": 10979,
      "issue_title": "count_ops is slow for large expressions",
      "issue_body": "It seems that this script was hanging inside `count_ops`:\n\n```\nmoorepants@garuda:pydy.wiki(master)$ SYMPY_CACHE_SIZE=10000 ipython\nPython 3.5.1 |Continuum Analytics, Inc.| (default, Dec  7 2015, 11:16:01) \nType \"copyright\", \"credits\" or \"license\" for more information.\n\nIPython 4.1.2 -- An enhanced Interactive Python.\n?         -> Introduction and overview of IPython's features.\n%quickref -> Quick reference.\nhelp      -> Python's own help system.\nobject?   -> Details about 'object', use 'object??' for extra details.\n\nIn [1]: %paste\n   In [1]: from pydy.models import n_link_pendulum_on_cart\n\n   In [2]: sys = n_link_pendulum_on_cart(3)\n\n   In [3]: x_dot = sys.eom_method.rhs()\n\n   In [4]: %time jac = x_dot.jacobian(sys.states)\n## -- End pasted text --\nCPU times: user 2.2 s, sys: 4 ms, total: 2.21 s\nWall time: 2.2 s\n\nIn [2]: %paste\n   In [5]: sys = n_link_pendulum_on_cart(4)\n\n   In [6]: x_dot = sys.eom_method.rhs()\n\n   In [7]: %time jac = x_dot.jacobian(sys.states)\n## -- End pasted text --\n^C---------------------------------------------------------------------------\nKeyboardInterrupt                         Traceback (most recent call last)\n<ipython-input-2-1039ec729c05> in <module>()\n      3 x_dot = sys.eom_method.rhs()\n      4 \n----> 5 get_ipython().magic('time jac = x_dot.jacobian(sys.states)')\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/interactiveshell.py in magic(self, arg_s)\n   2161         magic_name, _, magic_arg_s = arg_s.partition(' ')\n   2162         magic_name = magic_name.lstrip(prefilter.ESC_MAGIC)\n-> 2163         return self.run_line_magic(magic_name, magic_arg_s)\n   2164 \n   2165     #-------------------------------------------------------------------------\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/interactiveshell.py in run_line_magic(self, magic_name, line)\n   2082                 kwargs['local_ns'] = sys._getframe(stack_depth).f_locals\n   2083             with self.builtin_trap:\n-> 2084                 result = fn(*args,**kwargs)\n   2085             return result\n   2086 \n\n<decorator-gen-60> in time(self, line, cell, local_ns)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/magic.py in <lambda>(f, *a, **k)\n    191     # but it's overkill for just that one bit of state.\n    192     def magic_deco(arg):\n--> 193         call = lambda f, *a, **k: f(*a, **k)\n    194 \n    195         if callable(arg):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/magics/execution.py in time(self, line, cell, local_ns)\n   1175         else:\n   1176             st = clock2()\n-> 1177             exec(code, glob, local_ns)\n   1178             end = clock2()\n   1179             out = None\n\n<timed exec> in <module>()\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in jacobian(self, X)\n   1551         # m is the number of functions and n is the number of variables\n   1552         # computing the Jacobian is now easy:\n-> 1553         return self._new(m, n, lambda j, i: self[j].diff(X[i]))\n   1554 \n   1555     def QRdecomposition(self):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/dense.py in _new(cls, *args, **kwargs)\n    601     @classmethod\n    602     def _new(cls, *args, **kwargs):\n--> 603         rows, cols, flat_list = cls._handle_creation_inputs(*args, **kwargs)\n    604         self = object.__new__(cls)\n    605         self.rows = rows\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in _handle_creation_inputs(cls, *args, **kwargs)\n    207                     flat_list.extend(\n    208                         [cls._sympify(op(cls._sympify(i), cls._sympify(j)))\n--> 209                         for j in range(cols)])\n    210 \n    211             # Matrix(2, 2, [1, 2, 3, 4])\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in <listcomp>(.0)\n    207                     flat_list.extend(\n    208                         [cls._sympify(op(cls._sympify(i), cls._sympify(j)))\n--> 209                         for j in range(cols)])\n    210 \n    211             # Matrix(2, 2, [1, 2, 3, 4])\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in <lambda>(j, i)\n   1551         # m is the number of functions and n is the number of variables\n   1552         # computing the Jacobian is now easy:\n-> 1553         return self._new(m, n, lambda j, i: self[j].diff(X[i]))\n   1554 \n   1555     def QRdecomposition(self):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/power.py in _eval_derivative(self, s)\n    982     def _eval_derivative(self, s):\n    983         from sympy import log\n--> 984         dbase = self.base.diff(s)\n    985         dexp = self.exp.diff(s)\n    986         return self * (dexp * log(self.base) + dbase * self.exp/self.base)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/power.py in _eval_derivative(self, s)\n    984         dbase = self.base.diff(s)\n    985         dexp = self.exp.diff(s)\n--> 986         return self * (dexp * log(self.base) + dbase * self.exp/self.base)\n    987 \n    988     def _eval_evalf(self, prec):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, *args, **options)\n    388 \n    389         pr = max(cls._should_evalf(a) for a in result.args)\n--> 390         pr2 = min(cls._should_evalf(a) for a in result.args)\n    391         if pr2 > 0:\n    392             return result.evalf(mlib.libmpf.prec_to_dps(pr))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in <genexpr>(.0)\n    388 \n    389         pr = max(cls._should_evalf(a) for a in result.args)\n--> 390         pr2 = min(cls._should_evalf(a) for a in result.args)\n    391         if pr2 > 0:\n    392             return result.evalf(mlib.libmpf.prec_to_dps(pr))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in _should_evalf(cls, arg)\n    411         # Don't use as_real_imag() here, that's too much work\n    412         a, b = Wild('a'), Wild('b')\n--> 413         m = arg.match(a + b*S.ImaginaryUnit)\n    414         if not m or not (m[a].is_Float or m[b].is_Float):\n    415             return -1\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/basic.py in match(self, pattern, old)\n   1489         \"\"\"\n   1490         pattern = sympify(pattern)\n-> 1491         return pattern.matches(self, old=old)\n   1492 \n   1493     def count_ops(self, visual=None):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in matches(self, expr, repl_dict, old)\n    365 \n    366     def matches(self, expr, repl_dict={}, old=False):\n--> 367         return AssocOp._matches_commutative(self, expr, repl_dict, old)\n    368 \n    369     @staticmethod\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/operations.py in _matches_commutative(self, expr, repl_dict, old)\n    215                     d1 = w.matches(last_op, repl_dict)\n    216                     if d1 is not None:\n--> 217                         d2 = self.xreplace(d1).matches(expr, d1)\n    218                         if d2 is not None:\n    219                             return d2\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in matches(self, expr, repl_dict, old)\n    365 \n    366     def matches(self, expr, repl_dict={}, old=False):\n--> 367         return AssocOp._matches_commutative(self, expr, repl_dict, old)\n    368 \n    369     @staticmethod\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/operations.py in _matches_commutative(self, expr, repl_dict, old)\n    201             newexpr = self._combine_inverse(expr, exact)\n    202             if not old and (expr.is_Add or expr.is_Mul):\n--> 203                 if newexpr.count_ops() > expr.count_ops():\n    204                     return None\n    205             return newpattern.matches(newexpr, repl_dict)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in count_ops(self, visual)\n   1017         \"\"\"wrapper for count_ops that returns the operation count.\"\"\"\n   1018         from .function import count_ops\n-> 1019         return count_ops(self, visual)\n   1020 \n   1021     def args_cnc(self, cset=False, warn=True, split_1=True):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in count_ops(expr, visual)\n   2378                 a.is_Pow or\n   2379                 a.is_Function or\n-> 2380                 isinstance(a, Derivative) or\n   2381                     isinstance(a, Integral)):\n   2382 \n\nKeyboardInterrupt: \n\nIn [3]: \n```\n",
      "issue_closed_at": "2017-10-13T23:48:13Z",
      "base_commit": "e0cd7d65857a90376a9b49529840f96908dd774f",
      "changes": [
        {
          "file": "sympy/core/add.py",
          "type": "function",
          "name": "matches",
          "class_name": "Add",
          "code": "def matches(self, expr, repl_dict={}, old=False):\n        return AssocOp._matches_commutative(self, expr, repl_dict, old)"
        },
        {
          "file": "sympy/core/function.py",
          "type": "function",
          "name": "_should_evalf",
          "class_name": "Function",
          "code": "def _should_evalf(cls, arg):\n        \"\"\"\n        Decide if the function should automatically evalf().\n\n        By default (in this implementation), this happens if (and only if) the\n        ARG is a floating point number.\n        This function is used by __new__.\n\n        Returns the precision to evalf to, or -1 if it shouldn't evalf.\n        \"\"\"\n        from sympy.core.symbol import Wild\n        if arg.is_Float:\n            return arg._prec\n        if not arg.is_Add:\n            return -1\n        # Don't use as_real_imag() here, that's too much work\n        a, b = Wild('a'), Wild('b')\n        m = arg.match(a + b*S.ImaginaryUnit)\n        if not m or not (m[a].is_Float or m[b].is_Float):\n            return -1\n        l = [m[i]._prec for i in m if m[i].is_Float]\n        l.append(-1)\n        return max(l)"
        },
        {
          "file": "sympy/core/operations.py",
          "type": "function",
          "name": "_new_rawargs",
          "class_name": "AssocOp",
          "code": "def _new_rawargs(self, *args, **kwargs):\n        \"\"\"Create new instance of own class with args exactly as provided by\n        caller but returning the self class identity if args is empty.\n\n           This is handy when we want to optimize things, e.g.\n\n               >>> from sympy import Mul, S\n               >>> from sympy.abc import x, y\n               >>> e = Mul(3, x, y)\n               >>> e.args\n               (3, x, y)\n               >>> Mul(*e.args[1:])\n               x*y\n               >>> e._new_rawargs(*e.args[1:])  # the same as above, but faster\n               x*y\n\n           Note: use this with caution. There is no checking of arguments at\n           all. This is best used when you are rebuilding an Add or Mul after\n           simply removing one or more terms. If modification which result,\n           for example, in extra 1s being inserted (as when collecting an\n           expression's numerators and denominators) they will not show up in\n           the result but a Mul will be returned nonetheless:\n\n               >>> m = (x*y)._new_rawargs(S.One, x); m\n               x\n               >>> m == x\n               False\n               >>> m.is_Mul\n               True\n\n           Another issue to be aware of is that the commutativity of the result\n           is based on the commutativity of self. If you are rebuilding the\n           terms that came from a commutative object then there will be no\n           problem, but if self was non-commutative then what you are\n           rebuilding may now be commutative.\n\n           Although this routine tries to do as little as possible with the\n           input, getting the commutativity right is important, so this level\n           of safety is enforced: commutativity will always be recomputed if\n           self is non-commutative and kwarg `reeval=False` has not been\n           passed.\n        \"\"\"\n        if kwargs.pop('reeval', True) and self.is_commutative is False:\n            is_commutative = None\n        else:\n            is_commutative = self.is_commutative\n        return self._from_args(args, is_commutative)"
        },
        {
          "file": "sympy/core/operations.py",
          "type": "function",
          "name": "_matches_commutative",
          "class_name": "AssocOp",
          "code": "def _matches_commutative(self, expr, repl_dict={}, old=False):\n        \"\"\"\n        Matches Add/Mul \"pattern\" to an expression \"expr\".\n\n        repl_dict ... a dictionary of (wild: expression) pairs, that get\n                      returned with the results\n\n        This function is the main workhorse for Add/Mul.\n\n        For instance:\n\n        >>> from sympy import symbols, Wild, sin\n        >>> a = Wild(\"a\")\n        >>> b = Wild(\"b\")\n        >>> c = Wild(\"c\")\n        >>> x, y, z = symbols(\"x y z\")\n        >>> (a+sin(b)*c)._matches_commutative(x+sin(y)*z)\n        {a_: x, b_: y, c_: z}\n\n        In the example above, \"a+sin(b)*c\" is the pattern, and \"x+sin(y)*z\" is\n        the expression.\n\n        The repl_dict contains parts that were already matched. For example\n        here:\n\n        >>> (x+sin(b)*c)._matches_commutative(x+sin(y)*z, repl_dict={a: x})\n        {a_: x, b_: y, c_: z}\n\n        the only function of the repl_dict is to return it in the\n        result, e.g. if you omit it:\n\n        >>> (x+sin(b)*c)._matches_commutative(x+sin(y)*z)\n        {b_: y, c_: z}\n\n        the \"a: x\" is not returned in the result, but otherwise it is\n        equivalent.\n\n        \"\"\"\n        # make sure expr is Expr if pattern is Expr\n        from .expr import Add, Expr\n        from sympy import Mul\n        if isinstance(self, Expr) and not isinstance(expr, Expr):\n            return None\n\n        # handle simple patterns\n        if self == expr:\n            return repl_dict\n\n        d = self._matches_simple(expr, repl_dict)\n        if d is not None:\n            return d\n\n        # eliminate exact part from pattern: (2+a+w1+w2).matches(expr) -> (w1+w2).matches(expr-a-2)\n        from .function import WildFunction\n        from .symbol import Wild\n        wild_part = []\n        exact_part = []\n        for p in ordered(self.args):\n            if p.has(Wild, WildFunction) and (not expr.has(p)):\n                # not all Wild should stay Wilds, for example:\n                # (w2+w3).matches(w1) -> (w1+w3).matches(w1) -> w3.matches(0)\n                wild_part.append(p)\n            else:\n                exact_part.append(p)\n\n        if exact_part:\n            exact = self.func(*exact_part)\n            free = expr.free_symbols\n            if free and (exact.free_symbols - free):\n                # there are symbols in the exact part that are not\n                # in the expr; but if there are no free symbols, let\n                # the matching continue\n                return None\n            newpattern = self.func(*wild_part)\n            newexpr = self._combine_inverse(expr, exact)\n            if not old and (expr.is_Add or expr.is_Mul):\n                if newexpr.count_ops() > expr.count_ops():\n                    return None\n            return newpattern.matches(newexpr, repl_dict)\n\n        # now to real work ;)\n        i = 0\n        saw = set()\n        while expr not in saw:\n            saw.add(expr)\n            expr_list = (self.identity,) + tuple(ordered(self.make_args(expr)))\n            for last_op in reversed(expr_list):\n                for w in reversed(wild_part):\n                    d1 = w.matches(last_op, repl_dict)\n                    if d1 is not None:\n                        d2 = self.xreplace(d1).matches(expr, d1)\n                        if d2 is not None:\n                            return d2\n\n            if i == 0:\n                if self.is_Mul:\n                    # make e**i look like Mul\n                    if expr.is_Pow and expr.exp.is_Integer:\n                        if expr.exp > 0:\n                            expr = Mul(*[expr.base, expr.base**(expr.exp - 1)], evaluate=False)\n                        else:\n                            expr = Mul(*[1/expr.base, expr.base**(expr.exp + 1)], evaluate=False)\n                        i += 1\n                        continue\n\n                elif self.is_Add:\n                    # make i*e look like Add\n                    c, e = expr.as_coeff_Mul()\n                    if abs(c) > 1:\n                        if c > 0:\n                            expr = Add(*[e, (c - 1)*e], evaluate=False)\n                        else:\n                            expr = Add(*[-e, (c + 1)*e], evaluate=False)\n                        i += 1\n                        continue\n\n                    # try collection on non-Wild symbols\n                    from sympy.simplify.radsimp import collect\n                    was = expr\n                    did = set()\n                    for w in reversed(wild_part):\n                        c, w = w.as_coeff_mul(Wild)\n                        free = c.free_symbols - did\n                        if free:\n                            did.update(free)\n                            expr = collect(expr, free)\n                    if expr != was:\n                        i += 0\n                        continue\n\n                break  # if we didn't continue, there is nothing more to do\n\n        return"
        }
      ]
    }
  ]
}