{
  "instance_id": "sympy__sympy-17022",
  "repo": "sympy/sympy",
  "created_at": "2019-06-12T21:54:57Z",
  "problem_statement": "Lambdify misinterprets some matrix expressions\nUsing lambdify on an expression containing an identity matrix gives us an unexpected result:\r\n\r\n```python\r\n>>> import numpy as np\r\n>>> n = symbols('n', integer=True)\r\n>>> A = MatrixSymbol(\"A\", n, n)\r\n>>> a = np.array([[1, 2], [3, 4]])\r\n>>> f = lambdify(A, A + Identity(n))\r\n>>> f(a)\r\narray([[1.+1.j, 2.+1.j],\r\n       [3.+1.j, 4.+1.j]])\r\n```\r\n\r\nInstead, the output should be  `array([[2, 2], [3, 5]])`, since we're adding an identity matrix to the array. Inspecting the globals and source code of `f` shows us why we get the result:\r\n\r\n```python\r\n>>> import inspect\r\n>>> print(inspect.getsource(f))\r\ndef _lambdifygenerated(A):\r\n    return (I + A)\r\n>>> f.__globals__['I']\r\n1j\r\n```\r\n\r\nThe code printer prints `I`, which is currently being interpreted as a Python built-in complex number. The printer should support printing identity matrices, and signal an error for unsupported expressions that might be misinterpreted.\n",
  "patch": "diff --git a/sympy/printing/pycode.py b/sympy/printing/pycode.py\n--- a/sympy/printing/pycode.py\n+++ b/sympy/printing/pycode.py\n@@ -608,6 +608,13 @@ def _print_MatrixBase(self, expr):\n             func = self._module_format('numpy.array')\n         return \"%s(%s)\" % (func, self._print(expr.tolist()))\n \n+    def _print_Identity(self, expr):\n+        shape = expr.shape\n+        if all([dim.is_Integer for dim in shape]):\n+            return \"%s(%s)\" % (self._module_format('numpy.eye'), self._print(expr.shape[0]))\n+        else:\n+            raise NotImplementedError(\"Symbolic matrix dimensions are not yet supported for identity matrices\")\n+\n     def _print_BlockMatrix(self, expr):\n         return '{0}({1})'.format(self._module_format('numpy.block'),\n                                  self._print(expr.args[0].tolist()))\n",
  "similar_bug_items": [
    {
      "pr_number": 11947,
      "pr_title": "aligned middle of matrix with baseline",
      "pr_body": "**Previous Output**: \r\n\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n2\u22c5\u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\n\r\n**Output after this PR**:\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n  \u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n2.\u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\nFixes #10472 . ",
      "issue_id": 10472,
      "issue_title": "pprint should align the middle of the matrix to the baseline?",
      "issue_body": "Consider the current behaviour, where operators align with the top of the matrix:\n\n```\n>>> Z = Matrix(5,5, lambda i,j: i+j)\n>>> pprint(MatMul(2,Z))\n2\u22c5\u23a10  1  2  3  4\u23a4\n  \u23a2             \u23a5\n  \u23a21  2  3  4  5\u23a5\n  \u23a2             \u23a5\n  \u23a22  3  4  5  6\u23a5\n  \u23a2             \u23a5\n  \u23a23  4  5  6  7\u23a5\n  \u23a2             \u23a5\n  \u23a34  5  6  7  8\u23a6\n>>> pprint(MatAdd(Z,Z))\n\u23a10  1  2  3  4\u23a4 + \u23a10  1  2  3  4\u23a4\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a21  2  3  4  5\u23a5   \u23a21  2  3  4  5\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a22  3  4  5  6\u23a5   \u23a22  3  4  5  6\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a23  4  5  6  7\u23a5   \u23a23  4  5  6  7\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a34  5  6  7  8\u23a6   \u23a34  5  6  7  8\u23a6\n```\n\nThis looks poor, but I figured it might be the desired behaviour...\n\nHere's the example that makes me think its wrong:\n\n```\n>>> pprint(sin((1+1/x)/(1+1/y)) + det(MatMul(2,Z)))\n   \u239b    1\u239e                       \n   \u239c1 + \u2500\u239f                       \n   \u239c    x\u239f                       \nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + 32\u22c5\u2502\u23a10  1  2  3  4\u23a4\u2502\n   \u239c    1\u239f      \u2502\u23a2             \u23a5\u2502\n   \u239c1 + \u2500\u239f      \u2502\u23a21  2  3  4  5\u23a5\u2502\n   \u239d    y\u23a0      \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a22  3  4  5  6\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a23  4  5  6  7\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a34  5  6  7  8\u23a6\u2502\n```\n\nHere's how I think it should work, using the proposed #10423:\n\n```\npprint(sin((1+1/x)/y) + Trace(Z))\n               \u239b\u23a10  1  2  3  4\u23a4\u239e\n   \u239b    1\u239e     \u239c\u23a2             \u23a5\u239f\n   \u239c1 + \u2500\u239f     \u239c\u23a21  2  3  4  5\u23a5\u239f\n   \u239c    x\u239f     \u239c\u23a2             \u23a5\u239f\nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + tr\u239c\u23a22  3  4  5  6\u23a5\u239f\n   \u239d  y  \u23a0     \u239c\u23a2             \u23a5\u239f\n               \u239c\u23a23  4  5  6  7\u23a5\u239f\n               \u239c\u23a2             \u23a5\u239f\n               \u239d\u23a34  5  6  7  8\u23a6\u23a0\n```\n\nwhere note the `sin` alignment is by-design, and (relevant to this bug) the `+` and `tr` operator line up with `sin`.\n\nAt least, det, MatAdd, MatMul, MatPow would need fixed, maybe others I haven't thought of.\n",
      "issue_closed_at": "2016-12-22T06:52:11Z",
      "base_commit": "9a724a42c033c1aae97064947a0f44ec3b922d73",
      "changes": [
        {
          "file": "sympy/matrices/expressions/matadd.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matadd):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatAdd, pprint\n    >>> from sympy.matrices.expressions.matadd import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = eye(2)\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatAdd(A, B, C)\n    >>> pprint(X)\n    A + [1  0] + [1  2]\n        [    ]   [    ]\n        [0  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    A + [2  2]\n        [    ]\n        [3  5]\n    \"\"\"\n    groups = sift(matadd.args, lambda arg: isinstance(arg, MatrixBase))\n    if len(groups[True]) > 1:\n        return MatAdd(*(groups[False] + [reduce(add, groups[True])]))\n    else:\n        return matadd"
        },
        {
          "file": "sympy/matrices/expressions/matmul.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matmul):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatMul, pprint\n    >>> from sympy.matrices.expressions.matmul import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = Matrix([[1, 1], [1, 1]])\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatMul(A, B, C)\n    >>> pprint(X)\n    A*[1  1]*[1  2]\n      [    ] [    ]\n      [1  1] [3  4]\n    >>> pprint(merge_explicit(X))\n    A*[4  6]\n      [    ]\n      [4  6]\n\n    >>> X = MatMul(B, A, C)\n    >>> pprint(X)\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    \"\"\"\n    if not any(isinstance(arg, MatrixBase) for arg in matmul.args):\n        return matmul\n    newargs = []\n    last = matmul.args[0]\n    for arg in matmul.args[1:]:\n        if isinstance(arg, (MatrixBase, Number)) and isinstance(last, (MatrixBase, Number)):\n            last = last * arg\n        else:\n            newargs.append(last)\n            last = arg\n    newargs.append(last)\n\n    return MatMul(*newargs)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_matrix_contents",
          "class_name": "PrettyPrinter",
          "code": "def _print_matrix_contents(self, e):\n        \"\"\"\n        This method factors out what is essentially grid printing.\n        \"\"\"\n        M = e   # matrix\n        Ms = {}  # i,j -> pretty(M[i,j])\n        for i in range(M.rows):\n            for j in range(M.cols):\n                Ms[i, j] = self._print(M[i, j])\n\n        # h- and v- spacers\n        hsep = 2\n        vsep = 1\n\n        # max width for columns\n        maxw = [-1] * M.cols\n\n        for j in range(M.cols):\n            maxw[j] = max([Ms[i, j].width() for i in range(M.rows)] or [0])\n\n        # drawing result\n        D = None\n\n        for i in range(M.rows):\n\n            D_row = None\n            for j in range(M.cols):\n                s = Ms[i, j]\n\n                # reshape s to maxw\n                # XXX this should be generalized, and go to stringPict.reshape ?\n                assert s.width() <= maxw[j]\n\n                # hcenter it, +0.5 to the right                        2\n                # ( it's better to align formula starts for say 0 and r )\n                # XXX this is not good in all cases -- maybe introduce vbaseline?\n                wdelta = maxw[j] - s.width()\n                wleft = wdelta // 2\n                wright = wdelta - wleft\n\n                s = prettyForm(*s.right(' '*wright))\n                s = prettyForm(*s.left(' '*wleft))\n\n                # we don't need vcenter cells -- this is automatically done in\n                # a pretty way because when their baselines are taking into\n                # account in .right()\n\n                if D_row is None:\n                    D_row = s   # first box in a row\n                    continue\n\n                D_row = prettyForm(*D_row.right(' '*hsep))  # h-spacer\n                D_row = prettyForm(*D_row.right(s))\n\n            if D is None:\n                D = D_row       # first row in a picture\n                continue\n\n            # v-spacer\n            for _ in range(vsep):\n                D = prettyForm(*D.below(' '))\n\n            D = prettyForm(*D.below(D_row))\n\n        if D is None:\n            D = prettyForm('')  # Empty Matrix\n\n        return D"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_MatrixElement",
          "class_name": "PrettyPrinter",
          "code": "def _print_MatrixElement(self, expr):\n        from sympy.matrices import MatrixSymbol\n        from sympy import Symbol\n        if (isinstance(expr.parent, MatrixSymbol)\n                and expr.i.is_number and expr.j.is_number):\n            return self._print(\n                    Symbol(expr.parent.name + '_%d%d'%(expr.i, expr.j)))\n        else:\n            prettyFunc = self._print(expr.parent)\n            prettyIndices = self._print_seq((expr.i, expr.j), delimiter=', '\n                    ).parens(left='[', right=']')[0]\n            pform = prettyForm(binding=prettyForm.FUNC,\n                    *stringPict.next(prettyFunc, prettyIndices))\n\n            # store pform parts so it can be reassembled e.g. when powered\n            pform.prettyFunc = prettyFunc\n            pform.prettyArgs = prettyIndices\n\n            return pform"
        }
      ]
    },
    {
      "pr_number": 16255,
      "pr_title": "Replace non-ascii in docstring",
      "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 #16254\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nNon-ascii code is removed.\r\n\r\n#### Other comments\r\n\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 16254,
      "issue_title": "isympy broken under Py2 on master",
      "issue_body": "```\r\n$ isympy\r\nTraceback (most recent call last):\r\n  File \"/Users/enojb/current/sympy/27venv/bin/isympy\", line 11, in <module>\r\n    load_entry_point('sympy', 'console_scripts', 'isympy')()\r\n  File \"/Users/enojb/current/sympy/sympy/isympy.py\", line 338, in main\r\n    from sympy.interactive import init_session\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/__init__.py\", line 60, in <module>\r\n    from .polys import *\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/__init__.py\", line 5, in <module>\r\n    from . import polytools\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/polytools.py\", line 18, in <module>\r\n    from sympy.polys.constructor import construct_domain\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/constructor.py\", line 6, in <module>\r\n    from sympy.polys.domains import ZZ, QQ, EX\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/domains/__init__.py\", line 9, in <module>\r\n    from . import finitefield\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/domains/finitefield.py\", line 6, in <module>\r\n    from sympy.polys.domains.groundtypes import SymPyInteger\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/domains/groundtypes.py\", line 13, in <module>\r\n    from .pythonrational import PythonRational\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/polys/domains/pythonrational.py\", line 12, in <module>\r\n    from sympy.printing.defaults import DefaultPrinting\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/printing/__init__.py\", line 41, in <module>\r\n    from .mathematica import mathematica_code\r\n  File \"/Users/enojb/current/sympy/sympy/sympy/printing/mathematica.py\", line 187\r\nSyntaxError: Non-ASCII character '\\xe2' in file /Users/enojb/current/sympy/sympy/sympy/printing/mathematica.py on line 188, but no encoding declared; see http://python.org/dev/peps/pep-0263/ for details\r\n```\r\n\r\nI think a recent PR has added a non-ascii character to the code.",
      "issue_closed_at": "2019-03-14T13:16:04Z",
      "base_commit": "93a65b9bb8a615906e73d5885ff03076bcabc555",
      "changes": [
        {
          "file": "sympy/printing/mathematica.py",
          "type": "function",
          "name": "print_data",
          "class_name": "MCodePrinter",
          "code": "def print_data():\n            \"\"\"Helper function to print data part of Mathematica\n            sparse array.\n\n            It uses the fourth notation ``SparseArray[data,{d1,d2,\u2026}]``\n            from\n            https://reference.wolfram.com/language/ref/SparseArray.html\n\n            ``data`` must be formatted with rule.\n            \"\"\"\n            return print_string_list(\n                [print_rule(\n                    to_mathematica_index(*(expr._get_tuple_index(key))),\n                    value)\n                for key, value in sorted(expr._sparse_array.items())]\n            )"
        },
        {
          "file": "sympy/printing/mathematica.py",
          "type": "function",
          "name": "print_dims",
          "class_name": "MCodePrinter",
          "code": "def print_dims():\n            \"\"\"Helper function to print dimensions part of Mathematica\n            sparse array.\n\n            It uses the fourth notation ``SparseArray[data,{d1,d2,\u2026}]``\n            from\n            https://reference.wolfram.com/language/ref/SparseArray.html\n            \"\"\"\n            return self.doprint(expr.shape)"
        }
      ]
    },
    {
      "pr_number": 15519,
      "pr_title": "Exit _eval_derivative_n_times when None is obtained",
      "pr_body": "#### References to other Issues or PRs\r\n\r\nFixes #15518 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nThe method `_eval_derivative_n_times` contains a loop that involves calling `_eval_derivative`. The latter may return None when evaluation is not implemented. In such a case the loop should be aborted (returning None) instead of trying to differentiate None.\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* core\r\n  * Fixed a bug in the creation of higher order derivatives that cannot be evaluated.  \r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15518,
      "issue_title": "Higher order derivatives that cannot be evaluated, like re(x).diff(x, 2), raise an exception",
      "issue_body": "This issue arose in a [comment on another issue]( https://github.com/sympy/sympy/issues/15457#issuecomment-436453327) but since it's actually unrelated, is posted here separately. \r\n```\r\n>>> re(x).diff(x, 2)\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/ubuntu/sympy/sympy/core/expr.py\", line 3047, in diff\r\n    return Derivative(self, *symbols, **assumptions)\r\n  File \"/home/ubuntu/sympy/sympy/core/function.py\", line 1370, in __new__\r\n    obj = expr._eval_derivative_n_times(v, count)\r\n  File \"/home/ubuntu/sympy/sympy/core/basic.py\", line 1688, in _eval_derivative_n_times\r\n    obj2 = obj._accept_eval_derivative(s)\r\nAttributeError: 'NoneType' object has no attribute '_accept_eval_derivative'\r\n```\r\nThis happens whenever a higher-order derivative is requested for a function that does not have an explicit derivative. The expected output is \r\n```\r\nDerivative(re(x), (x, 2))\r\n```\r\nA PR is forthcoming. \r\n",
      "issue_closed_at": "2018-11-24T11:36:42Z",
      "base_commit": "61e5c20c02328815270ddda385f0108a2b40d24d",
      "changes": [
        {
          "file": "sympy/core/basic.py",
          "type": "function",
          "name": "_eval_derivative_n_times",
          "class_name": "Basic",
          "code": "def _eval_derivative_n_times(self, s, n):\n        # This is the default evaluator for derivatives (as called by `diff`\n        # and `Derivative`), it will attempt a loop to derive the expression\n        # `n` times by calling the corresponding `_eval_derivative` method,\n        # while leaving the derivative unevaluated if `n` is symbolic.  This\n        # method should be overridden if the object has a closed form for its\n        # symbolic n-th derivative.\n        from sympy import Integer\n        if isinstance(n, (int, Integer)):\n            obj = self\n            for i in range(n):\n                obj2 = obj._accept_eval_derivative(s)\n                if obj == obj2:\n                    break\n                obj = obj2\n            return obj\n        else:\n            return None"
        }
      ]
    },
    {
      "pr_number": 7707,
      "pr_title": "Fix the matplotlib warnings",
      "pr_body": "See #2822.\n",
      "issue_id": 2822,
      "issue_title": "sympy_printing produces odd output on ipython",
      "issue_body": "Hi all,\n\nI have recently run into problems when trying to use the interactive printing in the ipython notebook (version 2.0). It seems to me that lists/matrices and the like are not recognised properly by sympy.interactive.printing. Here is a screen shot:\n\n![screenshot from 2014-01-23 10 38 07](https://f.cloud.github.com/assets/1310472/1983286/80ec8e74-8412-11e3-9aad-e4686df0a7ea.png)\n\nEventually, the result is rendered properly, but printing throws an exception first. I haven't found any references to this error, so I would like to raise the issue. Or is there anything that can be wrong with my installation/settings?\n\nZolt\u00e1n\n",
      "issue_closed_at": "2014-07-10T13:36:56Z",
      "base_commit": "06e3c8edf2fade1a35094bbd5d9e500defc61df2",
      "changes": [
        {
          "file": "sympy/interactive/printing.py",
          "type": "function",
          "name": "_print_latex_matplotlib",
          "class_name": null,
          "code": "def _print_latex_matplotlib(o):\n        \"\"\"\n        A function that returns a png rendered by mathtext\n        \"\"\"\n        if _can_print_latex(o):\n            s = latex(o, mode='inline')\n            return _matplotlib_wrapper(s)"
        },
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "_print_Identity",
          "class_name": "LatexPrinter",
          "code": "def _print_Identity(self, I):\n        return r\"\\mathbb{I}\""
        },
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "_print_dict",
          "class_name": "LatexPrinter",
          "code": "def _print_dict(self, d):\n        keys = sorted(d.keys(), key=default_sort_key)\n        items = []\n\n        for key in keys:\n            val = d[key]\n            items.append(\"%s : %s\" % (self._print(key), self._print(val)))\n\n        return r\"\\begin{Bmatrix}%s\\end{Bmatrix}\" % r\", & \".join(items)"
        },
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "latex",
          "class_name": null,
          "code": "def latex(expr, **settings):\n    r\"\"\"\n    Convert the given expression to LaTeX representation.\n\n    >>> from sympy import latex, pi, sin, asin, Integral, Matrix, Rational\n    >>> from sympy.abc import x, y, mu, r, tau\n\n    >>> print(latex((2*tau)**Rational(7,2)))\n    8 \\sqrt{2} \\tau^{\\frac{7}{2}}\n\n    order: Any of the supported monomial orderings (currently \"lex\", \"grlex\", or\n    \"grevlex\"), \"old\", and \"none\". This parameter does nothing for Mul objects.\n    Setting order to \"old\" uses the compatibility ordering for Add defined in\n    Printer. For very large expressions, set the 'order' keyword to 'none' if\n    speed is a concern.\n\n    mode: Specifies how the generated code will be delimited. 'mode' can be one\n    of 'plain', 'inline', 'equation' or 'equation*'.  If 'mode' is set to\n    'plain', then the resulting code will not be delimited at all (this is the\n    default). If 'mode' is set to 'inline' then inline LaTeX $ $ will be used.\n    If 'mode' is set to 'equation' or 'equation*', the resulting code will be\n    enclosed in the 'equation' or 'equation*' environment (remember to import\n    'amsmath' for 'equation*'), unless the 'itex' option is set. In the latter\n    case, the ``$$ $$`` syntax is used.\n\n    >>> print(latex((2*mu)**Rational(7,2), mode='plain'))\n    8 \\sqrt{2} \\mu^{\\frac{7}{2}}\n\n    >>> print(latex((2*tau)**Rational(7,2), mode='inline'))\n    $8 \\sqrt{2} \\tau^{\\frac{7}{2}}$\n\n    >>> print(latex((2*mu)**Rational(7,2), mode='equation*'))\n    \\begin{equation*}8 \\sqrt{2} \\mu^{\\frac{7}{2}}\\end{equation*}\n\n    >>> print(latex((2*mu)**Rational(7,2), mode='equation'))\n    \\begin{equation}8 \\sqrt{2} \\mu^{\\frac{7}{2}}\\end{equation}\n\n    itex: Specifies if itex-specific syntax is used, including emitting ``$$ $$``.\n\n    >>> print(latex((2*mu)**Rational(7,2), mode='equation', itex=True))\n    $$8 \\sqrt{2} \\mu^{\\frac{7}{2}}$$\n\n    fold_frac_powers: Emit \"^{p/q}\" instead of \"^{\\frac{p}{q}}\" for fractional\n    powers.\n\n    >>> print(latex((2*tau)**Rational(7,2), fold_frac_powers=True))\n    8 \\sqrt{2} \\tau^{7/2}\n\n    fold_func_brackets: Fold function brackets where applicable.\n\n    >>> print(latex((2*tau)**sin(Rational(7,2))))\n    \\left(2 \\tau\\right)^{\\sin{\\left (\\frac{7}{2} \\right )}}\n    >>> print(latex((2*tau)**sin(Rational(7,2)), fold_func_brackets = True))\n    \\left(2 \\tau\\right)^{\\sin {\\frac{7}{2}}}\n\n    fold_short_frac: Emit \"p / q\" instead of \"\\frac{p}{q}\" when the\n    denominator is simple enough (at most two terms and no powers).\n    The default value is `True` for inline mode, False otherwise.\n\n    >>> print(latex(3*x**2/y))\n    \\frac{3 x^{2}}{y}\n    >>> print(latex(3*x**2/y, fold_short_frac=True))\n    3 x^{2} / y\n\n    long_frac_ratio: The allowed ratio of the width of the numerator to the\n    width of the denominator before we start breaking off long fractions.\n    The default value is 2.\n\n    >>> print(latex(Integral(r, r)/2/pi, long_frac_ratio=2))\n    \\frac{\\int r\\, dr}{2 \\pi}\n    >>> print(latex(Integral(r, r)/2/pi, long_frac_ratio=0))\n    \\frac{1}{2 \\pi} \\int r\\, dr\n\n    mul_symbol: The symbol to use for multiplication. Can be one of None,\n    \"ldot\", \"dot\", or \"times\".\n\n    >>> print(latex((2*tau)**sin(Rational(7,2)), mul_symbol=\"times\"))\n    \\left(2 \\times \\tau\\right)^{\\sin{\\left (\\frac{7}{2} \\right )}}\n\n    inv_trig_style: How inverse trig functions should be displayed. Can be one\n    of \"abbreviated\", \"full\", or \"power\". Defaults to \"abbreviated\".\n\n    >>> print(latex(asin(Rational(7,2))))\n    \\operatorname{asin}{\\left (\\frac{7}{2} \\right )}\n    >>> print(latex(asin(Rational(7,2)), inv_trig_style=\"full\"))\n    \\arcsin{\\left (\\frac{7}{2} \\right )}\n    >>> print(latex(asin(Rational(7,2)), inv_trig_style=\"power\"))\n    \\sin^{-1}{\\left (\\frac{7}{2} \\right )}\n\n    mat_str: Which matrix environment string to emit. \"smallmatrix\", \"matrix\",\n    \"array\", etc. Defaults to \"smallmatrix\" for inline mode, \"matrix\" for\n    matrices of no more than 10 columns, and \"array\" otherwise.\n\n    >>> print(latex(Matrix(2, 1, [x, y])))\n    \\left[\\begin{matrix}x\\\\y\\end{matrix}\\right]\n\n    >>> print(latex(Matrix(2, 1, [x, y]), mat_str = \"array\"))\n    \\left[\\begin{array}{c}x\\\\y\\end{array}\\right]\n\n    mat_delim: The delimiter to wrap around matrices. Can be one of \"[\", \"(\",\n    or the empty string. Defaults to \"[\".\n\n    >>> print(latex(Matrix(2, 1, [x, y]), mat_delim=\"(\"))\n    \\left(\\begin{matrix}x\\\\y\\end{matrix}\\right)\n\n    symbol_names: Dictionary of symbols and the custom strings they should be\n    emitted as.\n\n    >>> print(latex(x**2, symbol_names={x:'x_i'}))\n    x_i^{2}\n\n    Besides all Basic based expressions, you can recursively\n    convert Python containers (lists, tuples and dicts) and\n    also SymPy matrices:\n\n    >>> print(latex([2/x, y], mode='inline'))\n    $\\begin{bmatrix}2 / x, & y\\end{bmatrix}$\n\n    \"\"\"\n\n    return LatexPrinter(settings).doprint(expr)"
        }
      ]
    },
    {
      "pr_number": 9009,
      "pr_title": "MatrixExpr: add MatPow doit(), changes to doit for MatAdd, MatMul, Trace",
      "pr_body": "I'm been looking at `doit()` support in MatrixExpr.  In particular, I'm interested in cases where these have ImmutableMatrices inside and they could be evaluated (e.g., after a substitution).\n\nHere are some fixes:\n- MatPow: added `doit()`.\n- MatPow: refactor my older `.as_explicit()` code to use the the `doit()`.\n- MatAdd: had rather ineffective `doit()`, improved.\n- MatMul: change the default from `deep=False` to `deep=True`.  I cannot find a reason for it, so I've switched it.\n- MatMul: clean up test code.\n- Trace: change the default to `deep=True`.\n\nThe `deep=True` changes might be controversial: if someone knows why it `deep=False` before, I could always revert that (and add a comment to the code!)\n\nAnyway, here is what it does:\n\n```\nIn [6]: A = Matrix([[1,2],[3,4]])\n\nIn [7]: B = Matrix([[2,3],[4,5]])\n\nIn [8]: n = Symbol('n', integer=True)\n\nIn [9]: w = MatPow(A, n) + MatMul(A, MatPow(B,2*n))\n\nIn [10]: pprint(w)\n               2\u22c5n           n\n\u23a11  2\u23a4\u22c5\u239b\u23a12  3\u23a4\u239e    + \u239b\u23a11  2\u23a4\u239e \n\u23a2    \u23a5 \u239c\u23a2    \u23a5\u239f      \u239c\u23a2    \u23a5\u239f \n\u23a33  4\u23a6 \u239d\u23a34  5\u23a6\u23a0      \u239d\u23a33  4\u23a6\u23a0 \n\nIn [21]: pprint(w.subs(n, 1))\n               2           1\n\u23a11  2\u23a4\u22c5\u239b\u23a12  3\u23a4\u239e  + \u239b\u23a11  2\u23a4\u239e \n\u23a2    \u23a5 \u239c\u23a2    \u23a5\u239f    \u239c\u23a2    \u23a5\u239f \n\u23a33  4\u23a6 \u239d\u23a34  5\u23a6\u23a0    \u239d\u23a33  4\u23a6\u23a0 \n\nIn [22]: pprint(w.subs(n, 1).doit())    % these fixes make this work\n\u23a173   97 \u23a4\n\u23a2        \u23a5\n\u23a3163  215\u23a6\n```\n",
      "issue_id": 8138,
      "issue_title": "MatExpr.doit: deep default should be True for Expr (and still False for MatExpr?)",
      "issue_body": "I notice `MatExpr.doit()` has deep defaulting to False.  I assume that is deliberate (forget where but recall it looked deliberate).\n\nAnyway, deep should probably still default to True for any `Expr` (i.e., non-Matrix) it encounters.  \n\nConsider the following current behaviour:\n\n```\nx = S('x')\nn = S('n')\nA = MatrixSymbol('A', n, n)\nfive = Add(2, 3, evaluate=False)\n\nf = five*x\nG = five*A\n\nf.doit()\n# gives: 5*x\n\nG.doit()\n# gives: (2 + 3)*A\n```\n\nSame for `MatPow` where I saw this.\n",
      "issue_closed_at": "2015-02-19T04:28:17Z",
      "base_commit": "cf171b2f088bc6985f5973972425fa0a9fc65f31",
      "changes": [
        {
          "file": "sympy/matrices/expressions/matadd.py",
          "type": "function",
          "name": "_eval_trace",
          "class_name": "MatAdd",
          "code": "def _eval_trace(self):\n        from trace import Trace\n        return MatAdd(*[Trace(arg) for arg in self.args]).doit()"
        },
        {
          "file": "sympy/matrices/expressions/matmul.py",
          "type": "function",
          "name": "_eval_inverse",
          "class_name": "MatMul",
          "code": "def _eval_inverse(self):\n        try:\n            return MatMul(*[\n                arg.inverse() if isinstance(arg, MatrixExpr) else arg**-1\n                    for arg in self.args[::-1]]).doit()\n        except ShapeError:\n            from sympy.matrices.expressions.inverse import Inverse\n            return Inverse(self)"
        },
        {
          "file": "sympy/matrices/expressions/matpow.py",
          "type": "function",
          "name": "shape",
          "class_name": "MatPow",
          "code": "def shape(self):\n        return self.base.shape"
        },
        {
          "file": "sympy/matrices/expressions/trace.py",
          "type": "function",
          "name": "arg",
          "class_name": "Trace",
          "code": "def arg(self):\n        return self.args[0]"
        }
      ]
    }
  ]
}