{
  "instance_id": "sympy__sympy-21171",
  "repo": "sympy/sympy",
  "created_at": "2021-03-26T07:48:35Z",
  "problem_statement": "_print_SingularityFunction() got an unexpected keyword argument 'exp'\nOn a Jupyter Notebook cell, type the following:\r\n\r\n```python\r\nfrom sympy import *\r\nfrom sympy.physics.continuum_mechanics import Beam\r\n# Young's modulus\r\nE = symbols(\"E\")\r\n# length of the beam\r\nL = symbols(\"L\")\r\n# concentrated load at the end tip of the beam\r\nF = symbols(\"F\")\r\n# square cross section\r\nB, H = symbols(\"B, H\")\r\nI = B * H**3 / 12\r\n# numerical values (material: steel)\r\nd = {B: 1e-02, H: 1e-02, E: 210e09, L: 0.2, F: 100}\r\n\r\nb2 = Beam(L, E, I)\r\nb2.apply_load(-F, L / 2, -1)\r\nb2.apply_support(0, \"fixed\")\r\nR0, M0 = symbols(\"R_0, M_0\")\r\nb2.solve_for_reaction_loads(R0, M0)\r\n```\r\n\r\nThen:\r\n\r\n```\r\nb2.shear_force()\r\n```\r\n\r\nThe following error appears:\r\n```\r\n---------------------------------------------------------------------------\r\nTypeError                                 Traceback (most recent call last)\r\n/usr/local/lib/python3.8/dist-packages/IPython/core/formatters.py in __call__(self, obj)\r\n    343             method = get_real_method(obj, self.print_method)\r\n    344             if method is not None:\r\n--> 345                 return method()\r\n    346             return None\r\n    347         else:\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/interactive/printing.py in _print_latex_png(o)\r\n    184         \"\"\"\r\n    185         if _can_print(o):\r\n--> 186             s = latex(o, mode=latex_mode, **settings)\r\n    187             if latex_mode == 'plain':\r\n    188                 s = '$\\\\displaystyle %s$' % s\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in __call__(self, *args, **kwargs)\r\n    371 \r\n    372     def __call__(self, *args, **kwargs):\r\n--> 373         return self.__wrapped__(*args, **kwargs)\r\n    374 \r\n    375     @property\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in latex(expr, **settings)\r\n   2913 \r\n   2914     \"\"\"\r\n-> 2915     return LatexPrinter(settings).doprint(expr)\r\n   2916 \r\n   2917 \r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in doprint(self, expr)\r\n    252 \r\n    253     def doprint(self, expr):\r\n--> 254         tex = Printer.doprint(self, expr)\r\n    255 \r\n    256         if self._settings['mode'] == 'plain':\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in doprint(self, expr)\r\n    289     def doprint(self, expr):\r\n    290         \"\"\"Returns printer's representation for expr (as a string)\"\"\"\r\n--> 291         return self._str(self._print(expr))\r\n    292 \r\n    293     def _print(self, expr, **kwargs):\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Add(self, expr, order)\r\n    381             else:\r\n    382                 tex += \" + \"\r\n--> 383             term_tex = self._print(term)\r\n    384             if self._needs_add_brackets(term):\r\n    385                 term_tex = r\"\\left(%s\\right)\" % term_tex\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Mul(self, expr)\r\n    565             # use the original expression here, since fraction() may have\r\n    566             # altered it when producing numer and denom\r\n--> 567             tex += convert(expr)\r\n    568 \r\n    569         else:\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in convert(expr)\r\n    517                                isinstance(x.base, Quantity)))\r\n    518 \r\n--> 519                 return convert_args(args)\r\n    520 \r\n    521         def convert_args(args):\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in convert_args(args)\r\n    523 \r\n    524                 for i, term in enumerate(args):\r\n--> 525                     term_tex = self._print(term)\r\n    526 \r\n    527                     if self._needs_mul_brackets(term, first=(i == 0),\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Add(self, expr, order)\r\n    381             else:\r\n    382                 tex += \" + \"\r\n--> 383             term_tex = self._print(term)\r\n    384             if self._needs_add_brackets(term):\r\n    385                 term_tex = r\"\\left(%s\\right)\" % term_tex\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Mul(self, expr)\r\n    569         else:\r\n    570             snumer = convert(numer)\r\n--> 571             sdenom = convert(denom)\r\n    572             ldenom = len(sdenom.split())\r\n    573             ratio = self._settings['long_frac_ratio']\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in convert(expr)\r\n    505         def convert(expr):\r\n    506             if not expr.is_Mul:\r\n--> 507                 return str(self._print(expr))\r\n    508             else:\r\n    509                 if self.order not in ('old', 'none'):\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Add(self, expr, order)\r\n    381             else:\r\n    382                 tex += \" + \"\r\n--> 383             term_tex = self._print(term)\r\n    384             if self._needs_add_brackets(term):\r\n    385                 term_tex = r\"\\left(%s\\right)\" % term_tex\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/latex.py in _print_Pow(self, expr)\r\n    649         else:\r\n    650             if expr.base.is_Function:\r\n--> 651                 return self._print(expr.base, exp=self._print(expr.exp))\r\n    652             else:\r\n    653                 tex = r\"%s^{%s}\"\r\n\r\n/usr/local/lib/python3.8/dist-packages/sympy/printing/printer.py in _print(self, expr, **kwargs)\r\n    327                 printmethod = '_print_' + cls.__name__\r\n    328                 if hasattr(self, printmethod):\r\n--> 329                     return getattr(self, printmethod)(expr, **kwargs)\r\n    330             # Unknown object, fall back to the emptyPrinter.\r\n    331             return self.emptyPrinter(expr)\r\n\r\nTypeError: _print_SingularityFunction() got an unexpected keyword argument 'exp'\r\n```\n",
  "patch": "diff --git a/sympy/printing/latex.py b/sympy/printing/latex.py\n--- a/sympy/printing/latex.py\n+++ b/sympy/printing/latex.py\n@@ -1968,10 +1968,12 @@ def _print_DiracDelta(self, expr, exp=None):\n             tex = r\"\\left(%s\\right)^{%s}\" % (tex, exp)\n         return tex\n \n-    def _print_SingularityFunction(self, expr):\n+    def _print_SingularityFunction(self, expr, exp=None):\n         shift = self._print(expr.args[0] - expr.args[1])\n         power = self._print(expr.args[2])\n         tex = r\"{\\left\\langle %s \\right\\rangle}^{%s}\" % (shift, power)\n+        if exp is not None:\n+            tex = r\"{\\left({\\langle %s \\rangle}^{%s}\\right)}^{%s}\" % (shift, power, exp)\n         return tex\n \n     def _print_Heaviside(self, expr, exp=None):\n",
  "similar_bug_items": [
    {
      "pr_number": 19705,
      "pr_title": "Fix handling of addition and multiplication of Dimensions for get_dimensional_dependencies",
      "pr_body": "#### References to other Issues or PRs\r\nFixes #18738\r\n\r\n#### Brief description of what is fixed or changed\r\nFixes some cases where the method get_dimensional_dependencies of DimensionSystem raises exceptions when handling adding and multiplying dimensions. This addresses issue #18738 (at least the first half dealing with Dimensions, the second comment dealing with Quantities still doesn't work).\r\n\r\nThe following example previously raised an exception and now works properly:\r\n```\r\nIn [3]: from sympy.physics.units import length, mass, acceleration, time, pressure, force\r\nIn [4]: from sympy.physics.units.systems.si import dimsys_SI\r\nIn [5]: dimsys_SI.get_dimensional_dependencies(mass * length / time**2 + force - pressure * length**2)\r\nOut[5]: {'mass': 1, 'length': 1, 'time': -2}\r\n```\r\n\r\n#### Other comments\r\nAs part of the fix, the following two lines needed to be moved from the beginning of get_dimensional_dependencies to the beginning of _get_dimensional_dependencies_from_name:\r\n```\r\n        if isinstance(name, Dimension):\r\n            name = name.name\r\n\r\n        if isinstance(name, str):\r\n            name = Symbol(name)\r\n```\r\n\r\nThis had to be done to allow the recursion of the multiplication and addition operators to work properly. This is what fixed #18738 even though I was just trying to get the addition case above to work.  Additionally, an exception now gets raised if _def_dimensional_dependencies_for_name does not handle the input so that more useful error messages are provided then what is shown in #18738.\r\n\r\nIf the user attempts to add incompatible dimensions and then calls get_dimensional_dependencies, an exception is raised.  For example the following will raise an exception:\r\n\r\n`In [4]: dimsys_SI.get_dimensional_dependencies(mass * length / time**2 + pressure)`\r\n\r\nTests have been added for issue #18738 and for the addition cases listed above.\r\n\r\n#### Release Notes\r\n<!-- BEGIN RELEASE NOTES -->\r\n* physics.units\r\n  * Fixed some dimensional analysis bugs with the addition and multiplication operators.\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 18738,
      "issue_title": "Dimensional analysis: AttributeError in get_dimensional_dependencies with simple example",
      "issue_body": "Hi all, \r\n\r\nI am trying to get a simple example of a dimensional analysis (as described in [this tutorial](https://docs.sympy.org/latest/modules/physics/units/examples.html#dimensional-analysis) ) to work. Unfortunately, an AttributeError is raised inside the method `get_dimensional_dependencies`.\r\n\r\nHere is a minimal example:\r\n```\r\nfrom sympy import init_printing, symbols, sqrt\r\nfrom sympy.physics.units.systems.si import dimsys_SI\r\nfrom sympy.physics.units import length\r\ninit_printing()\r\n\r\na, b = symbols(\"a b\")\r\nc = sqrt(a**2 + b**2)\r\nc_dim = c.subs({a: length, b: length})\r\n\r\nprint(c_dim) \r\n# >> sqrt(2)*Dimension(sqrt(length**2))\r\n\r\ndimsys_SI.equivalent_dims(c_dim, length) \r\n# >> AttributeError: 'NoneType' object has no attribute 'items' \r\n```\r\n\r\nIt seems that `get_dimensional_dependencies` has problems handling non-dimensional terms (i.e. the `sqrt(2)` term in `c_dim` in the above example). Oddly, when I try to do the substitution 'manually', this term is eliminated:\r\n```\r\nc_dim = sqrt(length**2 + length**2)\r\nprint(c_dim) \r\n# >> Dimension(sqrt(length**2))\r\n```\r\n\r\nIs this intended behavior? It does make sense that multiplying a dimension with a dimensionless value should just give back the dimension as result. However, how come the `sqrt(2)`-term is not eliminated when substituting symbols for dimensions using `subs`?\r\n\r\nI hope I haven't just misunderstood the tutorial/docs. Happy for any help or hints. Also, I would be willing to dive into the code if someone would point me into the right direction.\r\n\r\nThis may be related to the bug discussed [here](https://gitter.im/sympy/sympy?at=5dc0c052fb4dab784a65dfb1).\r\n\r\nThe above example was tested with Python 2.7 and 3.7 and sympy 1.5.1.",
      "issue_closed_at": "2020-07-17T11:37:09Z",
      "base_commit": "f46f4488c7cb7aff9666e26ce1bfb980f98e97a7",
      "changes": [
        {
          "file": "sympy/physics/units/dimensions.py",
          "type": "function",
          "name": "dimensional_dependencies",
          "class_name": "DimensionSystem",
          "code": "def dimensional_dependencies(self):\n        return self.args[2]"
        },
        {
          "file": "sympy/physics/units/dimensions.py",
          "type": "function",
          "name": "_get_dimensional_dependencies_for_name",
          "class_name": "DimensionSystem",
          "code": "def _get_dimensional_dependencies_for_name(self, name):\n\n        if name.is_Symbol:\n            # Dimensions not included in the dependencies are considered\n            # as base dimensions:\n            return dict(self.dimensional_dependencies.get(name, {name: 1}))\n\n        if name.is_Number:\n            return {}\n\n        get_for_name = self._get_dimensional_dependencies_for_name\n\n        if name.is_Mul:\n            ret = collections.defaultdict(int)\n            dicts = [get_for_name(i) for i in name.args]\n            for d in dicts:\n                for k, v in d.items():\n                    ret[k] += v\n            return {k: v for (k, v) in ret.items() if v != 0}\n\n        if name.is_Pow:\n            dim = get_for_name(name.base)\n            return {k: v*name.exp for (k, v) in dim.items()}\n\n        if name.is_Function:\n            args = (Dimension._from_dimensional_dependencies(\n                get_for_name(arg)) for arg in name.args)\n            result = name.func(*args)\n\n            if isinstance(result, Dimension):\n                return self.get_dimensional_dependencies(result)\n            elif result.func == name.func:\n                return {}\n            else:\n                return get_for_name(result)"
        },
        {
          "file": "sympy/physics/units/dimensions.py",
          "type": "function",
          "name": "_get_dimensional_dependencies_for_name",
          "class_name": "DimensionSystem",
          "code": "def _get_dimensional_dependencies_for_name(self, name):\n\n        if name.is_Symbol:\n            # Dimensions not included in the dependencies are considered\n            # as base dimensions:\n            return dict(self.dimensional_dependencies.get(name, {name: 1}))\n\n        if name.is_Number:\n            return {}\n\n        get_for_name = self._get_dimensional_dependencies_for_name\n\n        if name.is_Mul:\n            ret = collections.defaultdict(int)\n            dicts = [get_for_name(i) for i in name.args]\n            for d in dicts:\n                for k, v in d.items():\n                    ret[k] += v\n            return {k: v for (k, v) in ret.items() if v != 0}\n\n        if name.is_Pow:\n            dim = get_for_name(name.base)\n            return {k: v*name.exp for (k, v) in dim.items()}\n\n        if name.is_Function:\n            args = (Dimension._from_dimensional_dependencies(\n                get_for_name(arg)) for arg in name.args)\n            result = name.func(*args)\n\n            if isinstance(result, Dimension):\n                return self.get_dimensional_dependencies(result)\n            elif result.func == name.func:\n                return {}\n            else:\n                return get_for_name(result)"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 15849,
      "pr_title": "Fixed vector derivatives printing ",
      "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\nFixes #10173\r\n\r\n#### Brief description of what is fixed or changed\r\nFixed the fourth-order printing issue.\r\n\r\nThis has been attempted in #10283 and #15022 but no unit tests were added. It wasn't clear which one of these I should base it on, but as seen, the current PR deals with higher order derivatives as well. And fixes some unused/redundant imports.\r\n\r\n#### Other comments\r\nI choose to keep two separate `if`-clauses to make the comments stand alone, while the newly added `if dot_i >= 5:` could easily have been added with the previous `if`.\r\n\r\nThe result looks different in different renders, see https://github.com/sympy/sympy/issues/14425#issuecomment-457834386 Hence, it may look a bit \"ugly\" in certain circumstances.\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* physics.vector\r\n   * printing of arbitrary order vector derivatives is fixed\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 10173,
      "issue_title": "mechanics_printing module does not display 4th derivatives correctly ",
      "issue_body": "When I run the following code snippet in an IPython notebook, the 1st, 2nd, and 3rd derivatives display correctly, but the 4th derivative does not. The 4th derivative simply displays as p, rather than as \\ddddot{p}.\n\n```\n%pylab inline\n\nimport sympy.physics.mechanics.functions\n\nsympy.physics.mechanics.functions.mechanics_printing(use_latex=\"mathjax\", latex_mode=\"equation\")\n\nt = sympy.Symbol(\"t\")\np = sympy.Symbol(\"p\")(t)\n\np, p.diff(t), p.diff(t).diff(t), p.diff(t).diff(t).diff(t), p.diff(t).diff(t).diff(t).diff(t)\n```\n\nAt first glance, this behavior seems to be caused by the `_print_Derivative` function in sympy/sympy/physics/vector/printing.py not handling 4th derivatives. I'll try hacking this function locally to support 4th derivatives and report back if this works around the issue.\n\nIdeally, mechanics_printing should gracefully fall back to Leibniz notation in the case of high-order derivatives, rather than printing them incorrectly.\n",
      "issue_closed_at": "2019-01-29T15:53:05Z",
      "base_commit": "1e999fc5159b830a9872b86675bc5f3692a9c1be",
      "changes": [
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        }
      ]
    },
    {
      "pr_number": 20639,
      "pr_title": "Pretty printing bug fix",
      "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://tinyurl.com/auto-closing for more information). Also, please\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #17616 \r\nCloses https://github.com/sympy/sympy/pull/17620\r\nCloses https://github.com/sympy/sympy/pull/17628\r\n\r\n#### Brief description of what is fixed or changed\r\nEarlier, \r\n```\r\n>>> pprint(pi**(1/exp(1)))\r\n-1___\r\n\u2572\u2571 \u03c0 \r\n```\r\nNow,\r\n```\r\n>>> pprint(pi**(1/exp(1)))\r\n \u239b -1\u239e\r\n \u239d\u212f  \u23a0\r\n\u03c0     \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* printing\r\n    * irrational powers are no longer printed with square root sign, they are printed as fractional powers\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 17616,
      "issue_title": "inaccurate rendering of pi**(1/E)",
      "issue_body": "This claims to be version 1.5.dev; I just merged from the project master, so I hope this is current.  I didn't notice this bug among others in printing.pretty.\r\n\r\n```\r\nIn [52]: pi**(1/E)                                                               \r\nOut[52]: \r\n-1___\r\n\u2572\u2571 \u03c0 \r\n\r\n```\r\nLaTeX and str not fooled:\r\n```\r\nIn [53]: print(latex(pi**(1/E)))                                                 \r\n\\pi^{e^{-1}}\r\n\r\nIn [54]: str(pi**(1/E))                                                          \r\nOut[54]: 'pi**exp(-1)'\r\n```\r\n",
      "issue_closed_at": "2021-01-10T06:33:26Z",
      "base_commit": "eb926a1d0c1158bf43f01eaf673dc84416b5ebb1",
      "changes": [
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_Mul",
          "class_name": "PrettyPrinter",
          "code": "def _print_Mul(self, product):\n        from sympy.physics.units import Quantity\n\n        # Check for unevaluated Mul. In this case we need to make sure the\n        # identities are visible, multiple Rational factors are not combined\n        # etc so we display in a straight-forward form that fully preserves all\n        # args and their order.\n        args = product.args\n        if args[0] is S.One or any(isinstance(arg, Number) for arg in args[1:]):\n            strargs = list(map(self._print, args))\n            # XXX: This is a hack to work around the fact that\n            # prettyForm.__mul__ absorbs a leading -1 in the args. Probably it\n            # would be better to fix this in prettyForm.__mul__ instead.\n            negone = strargs[0] == '-1'\n            if negone:\n                strargs[0] = prettyForm('1', 0, 0)\n            obj = prettyForm.__mul__(*strargs)\n            if negone:\n                obj = prettyForm('-' + obj.s, obj.baseline, obj.binding)\n            return obj\n\n        a = []  # items in the numerator\n        b = []  # items that are in the denominator (if any)\n\n        if self.order not in ('old', 'none'):\n            args = product.as_ordered_factors()\n        else:\n            args = list(product.args)\n\n        # If quantities are present append them at the back\n        args = sorted(args, key=lambda x: isinstance(x, Quantity) or\n                     (isinstance(x, Pow) and isinstance(x.base, Quantity)))\n\n        # Gather terms for numerator/denominator\n        for item in args:\n            if item.is_commutative and item.is_Pow and item.exp.is_Rational and item.exp.is_negative:\n                if item.exp != -1:\n                    b.append(Pow(item.base, -item.exp, evaluate=False))\n                else:\n                    b.append(Pow(item.base, -item.exp))\n            elif item.is_Rational and item is not S.Infinity:\n                if item.p != 1:\n                    a.append( Rational(item.p) )\n                if item.q != 1:\n                    b.append( Rational(item.q) )\n            else:\n                a.append(item)\n\n        from sympy import Integral, Piecewise, Product, Sum\n\n        # Convert to pretty forms. Add parens to Add instances if there\n        # is more than one term in the numer/denom\n        for i in range(0, len(a)):\n            if (a[i].is_Add and len(a) > 1) or (i != len(a) - 1 and\n                    isinstance(a[i], (Integral, Piecewise, Product, Sum))):\n                a[i] = prettyForm(*self._print(a[i]).parens())\n            elif a[i].is_Relational:\n                a[i] = prettyForm(*self._print(a[i]).parens())\n            else:\n                a[i] = self._print(a[i])\n\n        for i in range(0, len(b)):\n            if (b[i].is_Add and len(b) > 1) or (i != len(b) - 1 and\n                    isinstance(b[i], (Integral, Piecewise, Product, Sum))):\n                b[i] = prettyForm(*self._print(b[i]).parens())\n            else:\n                b[i] = self._print(b[i])\n\n        # Construct a pretty form\n        if len(b) == 0:\n            return prettyForm.__mul__(*a)\n        else:\n            if len(a) == 0:\n                a.append( self._print(S.One) )\n            return prettyForm.__mul__(*a)/prettyForm.__mul__(*b)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_nth_root",
          "class_name": "PrettyPrinter",
          "code": "def _print_nth_root(self, base, expt):\n        bpretty = self._print(base)\n\n        # In very simple cases, use a single-char root sign\n        if (self._settings['use_unicode_sqrt_char'] and self._use_unicode\n            and expt is S.Half and bpretty.height() == 1\n            and (bpretty.width() == 1\n                 or (base.is_Integer and base.is_nonnegative))):\n            return prettyForm(*bpretty.left('\\N{SQUARE ROOT}'))\n\n        # Construct root sign, start with the \\/ shape\n        _zZ = xobj('/', 1)\n        rootsign = xobj('\\\\', 1) + _zZ\n        # Make exponent number to put above it\n        if isinstance(expt, Rational):\n            exp = str(expt.q)\n            if exp == '2':\n                exp = ''\n        else:\n            exp = str(expt.args[0])\n        exp = exp.ljust(2)\n        if len(exp) > 2:\n            rootsign = ' '*(len(exp) - 2) + rootsign\n        # Stack the exponent\n        rootsign = stringPict(exp + '\\n' + rootsign)\n        rootsign.baseline = 0\n        # Diagonal: length is one less than height of base\n        linelength = bpretty.height() - 1\n        diagonal = stringPict('\\n'.join(\n            ' '*(linelength - i - 1) + _zZ + ' '*i\n            for i in range(linelength)\n        ))\n        # Put baseline just below lowest line: next to exp\n        diagonal.baseline = linelength - 1\n        # Make the root symbol\n        rootsign = prettyForm(*rootsign.right(diagonal))\n        # Det the baseline to match contents to fix the height\n        # but if the height of bpretty is one, the rootsign must be one higher\n        rootsign.baseline = max(1, bpretty.baseline)\n        #build result\n        s = prettyForm(hobj('_', 2 + bpretty.width()))\n        s = prettyForm(*bpretty.above(s))\n        s = prettyForm(*s.left(rootsign))\n        return s"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_Pow",
          "class_name": "PrettyPrinter",
          "code": "def _print_Pow(self, power):\n        from sympy.simplify.simplify import fraction\n        b, e = power.as_base_exp()\n        if power.is_commutative:\n            if e is S.NegativeOne:\n                return prettyForm(\"1\")/self._print(b)\n            n, d = fraction(e)\n            if n is S.One and d.is_Atom and not e.is_Integer and self._settings['root_notation']:\n                return self._print_nth_root(b, e)\n            if e.is_Rational and e < 0:\n                return prettyForm(\"1\")/self._print(Pow(b, -e, evaluate=False))\n\n        if b.is_Relational:\n            return prettyForm(*self._print(b).parens()).__pow__(self._print(e))\n\n        return self._print(b)**self._print(e)"
        }
      ]
    },
    {
      "pr_number": 13564,
      "pr_title": "Resolves RecursionError in latex representation",
      "pr_body": "fixes #13559. It was due to factor of `1/1`  \r\n```python\r\nexpr = parse_expr('5/1', evaluate=False)\r\nexpr.args\r\n```\r\nit gives `(5 , 1/1)` , which was not handled properly in latex conversion . So everytime an expression was divided by 1 , it resulted in RecursionError",
      "issue_id": 13559,
      "issue_title": "LaTeX representation raises RecursionError",
      "issue_body": "LaTeX representation of an expression that has division by 1 causes `RecursionError`. SymPy version: 1.1.1.\r\n\r\nCode to reproduce the bug:\r\n```python\r\nfrom sympy import latex\r\nfrom sympy.parsing.sympy_parser import parse_expr\r\n\r\nexpr = parse_expr('2/1', evaluate=False)\r\nlatex(expr)\r\n```\r\nOutput:\r\n```\r\n...\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/printer.py in _print(self, expr, *args, **kwargs)\r\n    255                 printmethod = '_print_' + cls.__name__\r\n    256                 if hasattr(self, printmethod):\r\n--> 257                     return getattr(self, printmethod)(expr, *args, **kwargs)\r\n    258             # Unknown object, fall back to the emptyPrinter.\r\n    259             return self.emptyPrinter(expr)\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in _print_Pow(self, expr)\r\n    466         elif expr.exp.is_Rational and expr.exp.is_negative and expr.base.is_commutative:\r\n    467             # Things like 1/x\r\n--> 468             return self._print_Mul(expr)\r\n    469         else:\r\n    470             if expr.base.is_Function:\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in _print_Mul(self, expr)\r\n    396             # use the original expression here, since fraction() may have\r\n    397             # altered it when producing numer and denom\r\n--> 398             tex += convert(expr)\r\n    399         else:\r\n    400             snumer = convert(numer)\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in convert(expr)\r\n    366         def convert(expr):\r\n    367             if not expr.is_Mul:\r\n--> 368                 return str(self._print(expr))\r\n    369             else:\r\n    370                 _tex = last_term_tex = \"\"\r\n\r\n... last 4 frames repeated, from the frame below ...\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/printer.py in _print(self, expr, *args, **kwargs)\r\n    255                 printmethod = '_print_' + cls.__name__\r\n    256                 if hasattr(self, printmethod):\r\n--> 257                     return getattr(self, printmethod)(expr, *args, **kwargs)\r\n    258             # Unknown object, fall back to the emptyPrinter.\r\n    259             return self.emptyPrinter(expr)\r\n\r\nRecursionError: maximum recursion depth exceeded in comparison\r\n```\r\n",
      "issue_closed_at": "2017-11-09T16:49:04Z",
      "base_commit": "36d9c850c642bec047e2cbad0a07c48f4fc7c779",
      "changes": [
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "_print_Gradient",
          "class_name": "LatexPrinter",
          "code": "def _print_Gradient(self, expr):\n        func = expr._expr\n        return r\"\\nabla\\cdot %s\" % self.parenthesize(func, PRECEDENCE['Mul'])"
        },
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "convert",
          "class_name": "LatexPrinter",
          "code": "def convert(expr):\n            if not expr.is_Mul:\n                return str(self._print(expr))\n            else:\n                _tex = last_term_tex = \"\"\n\n                if self.order not in ('old', 'none'):\n                    args = expr.as_ordered_factors()\n                else:\n                    args = expr.args\n\n                for i, term in enumerate(args):\n                    term_tex = self._print(term)\n\n                    if self._needs_mul_brackets(term, first=(i == 0),\n                                                last=(i == len(args) - 1)):\n                        term_tex = r\"\\left(%s\\right)\" % term_tex\n\n                    if _between_two_numbers_p[0].search(last_term_tex) and \\\n                            _between_two_numbers_p[1].match(term_tex):\n                        # between two numbers\n                        _tex += numbersep\n                    elif _tex:\n                        _tex += separator\n\n                    _tex += term_tex\n                    last_term_tex = term_tex\n                return _tex"
        }
      ]
    }
  ]
}