{
  "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 relevant report as it directly pertains to `MatAdd`, which is implicated in the CURRENT bug. The bug revolves around pretty printing Matrix addition that encounters issues when the expression is not in the expected format. The fixes in Candidate E target the `doit()` function enhancements for `MatAdd`, addressing how matrix expressions are evaluated and printed. This alignment on the same component, along with structural similarities in error handling and expression parsing, provides critical insights for resolving the CURRENT bug efficiently.",
  "instance_id": "sympy__sympy-14817",
  "repo": "sympy/sympy",
  "created_at": "2018-06-21T08:34:37Z",
  "problem_statement": "Error pretty printing MatAdd\n```py\r\n>>> pprint(MatrixSymbol('x', n, n) + MatrixSymbol('y*', n, n))\r\nTraceback (most recent call last):\r\n  File \"./sympy/core/sympify.py\", line 368, in sympify\r\n    expr = parse_expr(a, local_dict=locals, transformations=transformations, evaluate=evaluate)\r\n  File \"./sympy/parsing/sympy_parser.py\", line 950, in parse_expr\r\n    return eval_expr(code, local_dict, global_dict)\r\n  File \"./sympy/parsing/sympy_parser.py\", line 863, in eval_expr\r\n    code, global_dict, local_dict)  # take local objects in preference\r\n  File \"<string>\", line 1\r\n    Symbol ('y' )*\r\n                 ^\r\nSyntaxError: unexpected EOF while parsing\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"./sympy/printing/pretty/pretty.py\", line 2371, in pretty_print\r\n    use_unicode_sqrt_char=use_unicode_sqrt_char))\r\n  File \"./sympy/printing/pretty/pretty.py\", line 2331, in pretty\r\n    return pp.doprint(expr)\r\n  File \"./sympy/printing/pretty/pretty.py\", line 62, in doprint\r\n    return self._print(expr).render(**self._settings)\r\n  File \"./sympy/printing/printer.py\", line 274, in _print\r\n    return getattr(self, printmethod)(expr, *args, **kwargs)\r\n  File \"./sympy/printing/pretty/pretty.py\", line 828, in _print_MatAdd\r\n    if S(item.args[0]).is_negative:\r\n  File \"./sympy/core/sympify.py\", line 370, in sympify\r\n    raise SympifyError('could not parse %r' % a, exc)\r\nsympy.core.sympify.SympifyError: Sympify of expression 'could not parse 'y*'' failed, because of exception being raised:\r\nSyntaxError: unexpected EOF while parsing (<string>, line 1)\r\n```\r\n\r\nThe code shouldn't be using sympify to handle string arguments from MatrixSymbol.\r\n\r\nI don't even understand what the code is doing. Why does it omit the `+` when the first argument is negative? This seems to assume that the arguments of MatAdd have a certain form, and that they will always print a certain way if they are negative. \n",
  "patch": "diff --git a/sympy/printing/pretty/pretty.py b/sympy/printing/pretty/pretty.py\n--- a/sympy/printing/pretty/pretty.py\n+++ b/sympy/printing/pretty/pretty.py\n@@ -825,7 +825,8 @@ def _print_MatAdd(self, expr):\n             if s is None:\n                 s = pform     # First element\n             else:\n-                if S(item.args[0]).is_negative:\n+                coeff = item.as_coeff_mmul()[0]\n+                if _coeff_isneg(S(coeff)):\n                     s = prettyForm(*stringPict.next(s, ' '))\n                     pform = self._print(item)\n                 else:\n"
}