{
  "Selected_candidate": {
    "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⋅n           n\n⎡1  2⎤⋅⎛⎡2  3⎤⎞    + ⎛⎡1  2⎤⎞ \n⎢    ⎥ ⎜⎢    ⎥⎟      ⎜⎢    ⎥⎟ \n⎣3  4⎦ ⎝⎣4  5⎦⎠      ⎝⎣3  4⎦⎠ \n\nIn [21]: pprint(w.subs(n, 1))\n               2           1\n⎡1  2⎤⋅⎛⎡2  3⎤⎞  + ⎛⎡1  2⎤⎞ \n⎢    ⎥ ⎜⎢    ⎥⎟    ⎜⎢    ⎥⎟ \n⎣3  4⎦ ⎝⎣4  5⎦⎠    ⎝⎣3  4⎦⎠ \n\nIn [22]: pprint(w.subs(n, 1).doit())    % these fixes make this work\n⎡73   97 ⎤\n⎢        ⎥\n⎣163  215⎦\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]"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful report as it deals with issues in matrix expressions, specifically with functions related to matrix addition and multiplication, which are relevant to the current problem involving the identity matrix and its interaction with matrix operations. The patch focuses on improving the behavior of the `doit()` method for matrix expressions, which relates to how matrix operations are evaluated and represented. This context is crucial for understanding the misinterpretation issue in the CURRENT bug report where identity matrices are concerned. Moreover, the shared files among the patch details arc towards areas likely influencing the current bug report, providing targeted information that could assist in debugging and fixing the issue.",
  "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"
}