{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 8548,
      "pr_title": "Issue #6988: expand_log(exp(x), force=True) = x",
      "pr_body": "This is a try to fix issue #6988. If I made some mistake just let me know, as I'm still new to sympy development.\n\n```\nIn [9]: expand_log(log(exp(x)), force=True)\nOut[9]: x\n\nIn [10]: expand_log(log(y**(x)), force=True)\nOut[10]: x\u22c5log(y)\n```\n",
      "issue_id": 6988,
      "issue_title": "expand_log(exp(x), force=True) should give x",
      "issue_body": "```\nIn [5]: expand_log(log(exp(x)), force=True)\nOut[5]:\n   \u239b x\u239e\nlog\u239d\u212f \u23a0\n\nIn [6]: expand_log(log(y**(x)), force=True)\nOut[6]: x\u22c5log(y) Issue 1799 is probably to blame.\n```\n\nOriginal issue for #6988: http://code.google.com/p/sympy/issues/detail?id=3889\nOriginal author: https://code.google.com/u/asmeurer@gmail.com/\n",
      "issue_closed_at": "2014-12-03T13:34:14Z",
      "base_commit": "e6fc53f27ee872b27bc79b96529fc4bf34d4f023",
      "changes": [
        {
          "file": "sympy/functions/elementary/exponential.py",
          "type": "function",
          "name": "_eval_expand_log",
          "class_name": "log",
          "code": "def _eval_expand_log(self, deep=True, **hints):\n        from sympy import unpolarify\n        from sympy.concrete import Sum, Product\n        force = hints.get('force', False)\n        arg = self.args[0]\n        if arg.is_Integer:\n            # remove perfect powers\n            p = perfect_power(int(arg))\n            if p is not False:\n                return p[1]*self.func(p[0])\n        elif arg.is_Mul:\n            expr = []\n            nonpos = []\n            for x in arg.args:\n                if force or x.is_positive or x.is_polar:\n                    a = self.func(x)\n                    if isinstance(a, log):\n                        expr.append(self.func(x)._eval_expand_log(**hints))\n                    else:\n                        expr.append(a)\n                elif x.is_negative:\n                    a = self.func(-x)\n                    expr.append(a)\n                    nonpos.append(S.NegativeOne)\n                else:\n                    nonpos.append(x)\n            return Add(*expr) + log(Mul(*nonpos))\n        elif arg.is_Pow:\n            if force or (arg.exp.is_real and arg.base.is_positive) or \\\n                    arg.base.is_polar:\n                b = arg.base\n                e = arg.exp\n                a = self.func(b)\n                if isinstance(a, log):\n                    return unpolarify(e) * a._eval_expand_log(**hints)\n                else:\n                    return unpolarify(e) * a\n        elif isinstance(arg, Product):\n            if arg.function.is_positive:\n                return Sum(log(arg.function), *arg.limits)\n\n        return self.func(arg)"
        }
      ]
    },
    {
      "pr_number": 13744,
      "pr_title": "group theory: correct FreeGroupElement multiplication",
      "pr_body": "At the moment, multiplication of `FreeGroupElement`s treats two elements as non-equal unless the symbols representing them are the same object. But it is sufficient for the symbols to be equal via `==`, and the current behaviour causes problems. For example, the added test fails in master.\r\n\r\nFixes #13737.",
      "issue_id": 13737,
      "issue_title": "coset_table doctest failure (Fatal Python error: Cannot recover from stack overflow.)",
      "issue_body": "See https://travis-ci.org/sympy/sympy/jobs/315588322\r\n\r\n```\r\nsympy/combinatorics/coset_table.py[3] \r\n.\r\nFatal Python error: Cannot recover from stack overflow.\r\nCurrent thread 0x00007f7fef7dd740 (most recent call first):\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/symbol.py\", line 257 in assumptions0\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/symbol.py\", line 252 in _hashable_content\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/basic.py\", line 331 in __eq__\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/cache.py\", line 93 in wrapper\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/core/expr.py\", line 111 in __neg__\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 435 in <listcomp>\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 435 in letter_form\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 927 in subword_index\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 964 in is_dependent\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 976 in is_independent\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 663 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 684 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  File \"/home/travis/virtualenv/python3.5.4/lib/python3.5/site-packages/sympy/combinatorics/free_groups.py\", line 687 in eliminate_word\r\n  ...\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 4, in <module>\r\nException: Tests failed\r\n```\r\n\r\nHere is the test header\r\n\r\n```\r\n============================= test process starts ==============================\r\nexecutable:         /home/travis/virtualenv/python3.5.4/bin/python  (3.5.4-final-0) [CPython]\r\narchitecture:       64-bit\r\ncache:              yes\r\nground types:       python \r\nnumpy:              None\r\nhash randomization: on (PYTHONHASHSEED=935966445)\r\n```",
      "issue_closed_at": "2017-12-15T08:43:52Z",
      "base_commit": "ae3d6f647f080240318d114f2c397b99be5cef5f",
      "changes": [
        {
          "file": "sympy/combinatorics/free_groups.py",
          "type": "function",
          "name": "letter_form_to_array_form",
          "class_name": null,
          "code": "def letter_form_to_array_form(array_form, group):\n    \"\"\"\n    This method converts a list given with possible repetitions of elements in\n    it. It returns a new list such that repetitions of consecutive elements is\n    removed and replace with a tuple element of size two such that the first\n    index contains `value` and the second index contains the number of\n    consecutive repetitions of `value`.\n\n    \"\"\"\n    a = list(array_form[:])\n    new_array = []\n    n = 1\n    symbols = group.symbols\n    for i in range(len(a)):\n        if i == len(a) - 1:\n            if a[i] == a[i - 1]:\n                if (-a[i]) in symbols:\n                    new_array.append((-a[i], -n))\n                else:\n                    new_array.append((a[i], n))\n            else:\n                if (-a[i]) in symbols:\n                    new_array.append((-a[i], -1))\n                else:\n                    new_array.append((a[i], 1))\n            return new_array\n        elif a[i] == a[i + 1]:\n            n += 1\n        else:\n            if (-a[i]) in symbols:\n                new_array.append((-a[i], -n))\n            else:\n                new_array.append((a[i], n))\n            n = 1"
        },
        {
          "file": "sympy/combinatorics/rewritingsystem.py",
          "type": "function",
          "name": "add_rule",
          "class_name": "RewritingSystem",
          "code": "def add_rule(self, w1, w2, check=False):\n        new_keys = set()\n\n        if w1 == w2:\n            return new_keys\n\n        if w1 < w2:\n            w1, w2 = w2, w1\n\n        if (w1, w2) in self.rules_cache:\n            return new_keys\n        self.rules_cache.append((w1, w2))\n\n        s1, s2 = w1, w2\n\n        # The following is the equivalent of checking\n        # s1 for overlaps with the implicit reductions\n        # {g*g**-1 -> <identity>} and {g**-1*g -> <identity>}\n        # for any generator g without installing the\n        # redundant rules that would result from processing\n        # the overlaps. See [1], Section 3 for details.\n\n        if len(s1) - len(s2) < 3:\n            if s1 not in self.rules:\n                new_keys.add(s1)\n                if not check:\n                    self._add_rule(s1, s2)\n            if s2**-1 > s1**-1 and s2**-1 not in self.rules:\n                new_keys.add(s2**-1)\n                if not check:\n                    self._add_rule(s2**-1, s1**-1)\n\n        # overlaps on the right\n        while len(s1) - len(s2) > -1:\n            g = s1[len(s1)-1]\n            s1 = s1.subword(0, len(s1)-1)\n            s2 = s2*g**-1\n            if len(s1) - len(s2) < 0:\n                #print(\"this\", len(s1)-len(s2))\n                if s2 not in self.rules:\n                    if not check:\n                        self._add_rule(s2, s1)\n                    new_keys.add(s2)\n            elif len(s1) - len(s2) < 3:\n                new = self.add_rule(s1, s2, check)\n                new_keys.update(new)\n\n        # overlaps on the left\n        while len(w1) - len(w2) > -1:\n            g = w1[0]\n            w1 = w1.subword(1, len(w1))\n            w2 = g**-1*w2\n            if len(w1) - len(w2) < 0:\n                if w2 not in self.rules:\n                    if not check:\n                        self._add_rule(w2, w1)\n                    new_keys.add(w2)\n            elif len(w1) - len(w2) < 3:\n                new = self.add_rule(w1, w2, check)\n                new_keys.update(new)\n\n        return new_keys"
        }
      ]
    },
    {
      "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": 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]"
        }
      ]
    },
    {
      "pr_number": 12882,
      "pr_title": "Checker functions for orthogonality of coordinate system",
      "pr_body": "This PR introduce additional condition for transformation equation which must be fulfill. \r\nIn `vector` module we are dealing with orthogonal curvilinear coordinate system only, so we need to check if transformation equations are correctly defined.\r\n\r\nCloses #12852",
      "issue_id": 12852,
      "issue_title": "Vector module: make sure that the base vectors remain orthogonal",
      "issue_body": "Given a generic transformation equation, make sure that the base vectors remain orthogonal, if not, raise an exception.",
      "issue_closed_at": "2017-07-05T15:10:49Z",
      "base_commit": "c87c0fbf1223c28db8590f2591f64db75d5bdd66",
      "changes": [
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "line",
          "name": "line 4",
          "code": "from sympy.core.cache import cacheit\nfrom sympy.core import S\nfrom sympy.vector.scalar import BaseScalar\nfrom sympy import eye, trigsimp, ImmutableMatrix as Matrix, Symbol, sin, cos, sqrt, diff, Tuple\nimport sympy.vector\nfrom sympy import simplify\nfrom sympy.vector.orienters import (Orienter, AxisOrienter, BodyOrienter,"
        },
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "function",
          "name": "_connect_to_standard_cartesian",
          "class_name": "CoordSys3D",
          "code": "def _connect_to_standard_cartesian(self, curv_coord_type):\n        \"\"\"\n        Change the type of orthogonal curvilinear system. It could be done\n        by tuple of transformation equations or by choosing one of pre-defined\n        coordinate system.\n\n        Parameters\n        ==========\n\n        :param curv_coord_type: str, tuple\n\n        \"\"\"\n        if isinstance(curv_coord_type, string_types):\n            self._set_transformation_equations_mapping(curv_coord_type)\n            self._set_lame_coefficient_mapping(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 3:\n            self._transformation_eqs = curv_coord_type\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 2:\n            self._transformation_eqs = \\\n            tuple([eq.subs({curv_coord_type[0][0]: self.x,\n                            curv_coord_type[0][1]: self.y,\n                            curv_coord_type[0][2]: self.z}) for eq in curv_coord_type[1]])\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(self._transformation_equations())\n\n        else:\n            raise ValueError(\"Wrong set of parameter.\")"
        }
      ]
    }
  ]
}