{
  "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 C is the most relevant because it deals with the evaluation of expressions within matrices, specifically focusing on the `doit()` functionality for matrix expressions, which is closely related to the current bug involving the sum of elements in an identity matrix. This candidate shares a similar operational context, as both involve matrix calculations and manipulation of symbolic variables. The fixes described in Candidate C could help the developer understand how to adjust the behavior of matrix evaluations and potentially resolve similar issues related to symbolic summation in the CURRENT bug's context.",
  "instance_id": "sympy__sympy-12419",
  "repo": "sympy/sympy",
  "created_at": "2017-03-25T15:02:26Z",
  "problem_statement": "Sum of the elements of an identity matrix is zero\nI think this is a bug.\r\n\r\nI created a matrix by M.T * M under an assumption that M is orthogonal.  SymPy successfully recognized that the result is an identity matrix.  I tested its identity-ness by element-wise, queries, and sum of the diagonal elements and received expected results.\r\n\r\nHowever, when I attempt to evaluate the total sum of the elements the result was 0 while 'n' is expected.\r\n\r\n```\r\nfrom sympy import *\r\nfrom sympy import Q as Query\r\n\r\nn = Symbol('n', integer=True, positive=True)\r\ni, j = symbols('i j', integer=True)\r\nM = MatrixSymbol('M', n, n)\r\n\r\ne = None\r\nwith assuming(Query.orthogonal(M)):\r\n    e = refine((M.T * M).doit())\r\n\r\n# Correct: M.T * M is an identity matrix.\r\nprint(e, e[0, 0], e[0, 1], e[1, 0], e[1, 1])\r\n\r\n# Correct: The output is True True\r\nprint(ask(Query.diagonal(e)), ask(Query.integer_elements(e)))\r\n\r\n# Correct: The sum of the diagonal elements is n\r\nprint(Sum(e[i, i], (i, 0, n-1)).doit())\r\n\r\n# So far so good\r\n# Total sum of the elements is expected to be 'n' but the answer is 0!\r\nprint(Sum(Sum(e[i, j], (i, 0, n-1)), (j, 0, n-1)).doit())\r\n```\n",
  "patch": "diff --git a/sympy/matrices/expressions/matexpr.py b/sympy/matrices/expressions/matexpr.py\n--- a/sympy/matrices/expressions/matexpr.py\n+++ b/sympy/matrices/expressions/matexpr.py\n@@ -2,11 +2,12 @@\n \n from functools import wraps\n \n-from sympy.core import S, Symbol, Tuple, Integer, Basic, Expr\n+from sympy.core import S, Symbol, Tuple, Integer, Basic, Expr, Eq\n from sympy.core.decorators import call_highest_priority\n from sympy.core.compatibility import range\n from sympy.core.sympify import SympifyError, sympify\n from sympy.functions import conjugate, adjoint\n+from sympy.functions.special.tensor_functions import KroneckerDelta\n from sympy.matrices import ShapeError\n from sympy.simplify import simplify\n \n@@ -375,7 +376,6 @@ def _eval_derivative(self, v):\n         if self.args[0] != v.args[0]:\n             return S.Zero\n \n-        from sympy import KroneckerDelta\n         return KroneckerDelta(self.args[1], v.args[1])*KroneckerDelta(self.args[2], v.args[2])\n \n \n@@ -476,10 +476,12 @@ def conjugate(self):\n         return self\n \n     def _entry(self, i, j):\n-        if i == j:\n+        eq = Eq(i, j)\n+        if eq is S.true:\n             return S.One\n-        else:\n+        elif eq is S.false:\n             return S.Zero\n+        return KroneckerDelta(i, j)\n \n     def _eval_determinant(self):\n         return S.One\n"
}