{
  "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 D is the most relevant because it directly addresses the `MatMul`, a fundamental operation related to the current bug involving the matrix multiplication operator (`@`). The changes made to the `doit()` method in `MatMul`, especially regarding the handling of deep defaults, highlight adjustments made to the matrix multiplication process itself, which is closely related to the issue of type validation for matrix multiplication versus scalar multiplication. The structural similarity in handling matrix operations alongside similar contexts of user interactions with matrices provides a valuable reference for debugging the current bug.",
  "instance_id": "sympy__sympy-13773",
  "repo": "sympy/sympy",
  "created_at": "2017-12-19T10:44:38Z",
  "problem_statement": "@ (__matmul__) should fail if one argument is not a matrix\n```\r\n>>> A = Matrix([[1, 2], [3, 4]])\r\n>>> B = Matrix([[2, 3], [1, 2]])\r\n>>> A@B\r\nMatrix([\r\n[ 4,  7],\r\n[10, 17]])\r\n>>> 2@B\r\nMatrix([\r\n[4, 6],\r\n[2, 4]])\r\n```\r\n\r\nRight now `@` (`__matmul__`) just copies `__mul__`, but it should actually only work if the multiplication is actually a matrix multiplication. \r\n\r\nThis is also how NumPy works\r\n\r\n```\r\n>>> import numpy as np\r\n>>> a = np.array([[1, 2], [3, 4]])\r\n>>> 2*a\r\narray([[2, 4],\r\n       [6, 8]])\r\n>>> 2@a\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\nValueError: Scalar operands are not allowed, use '*' instead\r\n```\n",
  "patch": "diff --git a/sympy/matrices/common.py b/sympy/matrices/common.py\n--- a/sympy/matrices/common.py\n+++ b/sympy/matrices/common.py\n@@ -1973,6 +1973,10 @@ def __div__(self, other):\n \n     @call_highest_priority('__rmatmul__')\n     def __matmul__(self, other):\n+        other = _matrixify(other)\n+        if not getattr(other, 'is_Matrix', False) and not getattr(other, 'is_MatrixLike', False):\n+            return NotImplemented\n+\n         return self.__mul__(other)\n \n     @call_highest_priority('__rmul__')\n@@ -2066,6 +2070,10 @@ def __radd__(self, other):\n \n     @call_highest_priority('__matmul__')\n     def __rmatmul__(self, other):\n+        other = _matrixify(other)\n+        if not getattr(other, 'is_Matrix', False) and not getattr(other, 'is_MatrixLike', False):\n+            return NotImplemented\n+\n         return self.__rmul__(other)\n \n     @call_highest_priority('__mul__')\n"
}