{
  "instance_id": "sympy__sympy-13031",
  "repo": "sympy/sympy",
  "created_at": "2017-07-23T15:48:13Z",
  "problem_statement": "Behavior of Matrix hstack and vstack changed in sympy 1.1\nIn sympy 1.0:\r\n```\r\nimport sympy as sy\r\nM1 = sy.Matrix.zeros(0, 0)\r\nM2 = sy.Matrix.zeros(0, 1)\r\nM3 = sy.Matrix.zeros(0, 2)\r\nM4 = sy.Matrix.zeros(0, 3)\r\nsy.Matrix.hstack(M1, M2, M3, M4).shape\r\n```\r\nreturns \r\n`(0, 6)`\r\n\r\nNow, same in sympy 1.1:\r\n```\r\nimport sympy as sy\r\nM1 = sy.Matrix.zeros(0, 0)\r\nM2 = sy.Matrix.zeros(0, 1)\r\nM3 = sy.Matrix.zeros(0, 2)\r\nM4 = sy.Matrix.zeros(0, 3)\r\nsy.Matrix.hstack(M1, M2, M3, M4).shape\r\n```\r\nreturns\r\n`(0, 3)\r\n`\r\nwhereas:\r\n```\r\nimport sympy as sy\r\nM1 = sy.Matrix.zeros(1, 0)\r\nM2 = sy.Matrix.zeros(1, 1)\r\nM3 = sy.Matrix.zeros(1, 2)\r\nM4 = sy.Matrix.zeros(1, 3)\r\nsy.Matrix.hstack(M1, M2, M3, M4).shape\r\n```\r\nreturns\r\n`(1, 6)\r\n`\n",
  "patch": "diff --git a/sympy/matrices/sparse.py b/sympy/matrices/sparse.py\n--- a/sympy/matrices/sparse.py\n+++ b/sympy/matrices/sparse.py\n@@ -985,8 +985,10 @@ def col_join(self, other):\n         >>> C == A.row_insert(A.rows, Matrix(B))\n         True\n         \"\"\"\n-        if not self:\n-            return type(self)(other)\n+        # A null matrix can always be stacked (see  #10770)\n+        if self.rows == 0 and self.cols != other.cols:\n+            return self._new(0, other.cols, []).col_join(other)\n+\n         A, B = self, other\n         if not A.cols == B.cols:\n             raise ShapeError()\n@@ -1191,8 +1193,10 @@ def row_join(self, other):\n         >>> C == A.col_insert(A.cols, B)\n         True\n         \"\"\"\n-        if not self:\n-            return type(self)(other)\n+        # A null matrix can always be stacked (see  #10770)\n+        if self.cols == 0 and self.rows != other.rows:\n+            return self._new(other.rows, 0, []).row_join(other)\n+\n         A, B = self, other\n         if not A.rows == B.rows:\n             raise ShapeError()\n",
  "similar_bug_items": [
    {
      "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": 12270,
      "pr_title": "fix bug in extract_multiplicatively()",
      "pr_body": "Fixes #12254 .\r\nCan you review this and suggest improvements @smichr, @cbm755 ?\r\n\r\nAnd consider this test case : \r\n```\r\n>>> (4*y**2*(-x*y-2*y)).extract_multiplicatively(-2*y)\r\n```\r\nI am not sure whether we should return ```None``` or ```2*x*y**2 + 4*y**2``` ,  currently it returns ```None```",
      "issue_id": 12254,
      "issue_title": "cannot extract_multiplicatively(-2) from (-2*x - 2*y)",
      "issue_body": "I think this might be a bug.\r\n````\r\n>>> (2+4*I).extract_multiplicatively(2)    # yes\r\n1 + 2*I\r\n>>> (-2-4*I).extract_multiplicatively(-1)   # yes\r\n2 + 4*I\r\n>>> (-2-4*I).extract_multiplicatively(-2)   # bug?\r\n````\r\nsimilarly:\r\n````\r\n>>> (2*x + 4*y + 8).extract_multiplicatively(2)   # yes\r\nx + 2*y + 4\r\n>>> (-2*x - 4*y - 8).extract_multiplicatively(2)    # yes\r\n-x - 2*y - 4\r\n>>> (-2*x - 4*y - 8).extract_multiplicatively(-1)    # yes\r\n2*x + 4*y + 8\r\n>>> (-2*x - 4*y - 8).extract_multiplicatively(-2)    # bug?\r\n````\r\n\r\nAssuming it is indeed a bug, here is why it happens:\r\n\r\nLook in `core/expr.py` where:\r\n````\r\n>>> (-2*x - 4*y - 8).primitive()\r\n(2, -x - 2*y - 4)\r\n````\r\nwhich is then made into a *non-evaluated* `Mul`, from which `-2` cannot be multiplicatively extracted; for example:\r\n````\r\n>>> Mul(2, -x).extract_multiplicatively(-2)\r\nx\r\n>>> Mul(2, -x, evaluate=False).extract_multiplicatively(-2)\r\n````\r\n@smichr do you think this is bug? (see your commit 8968b85310506c0a2b34f3d7aeb8e0d88f87885b: not clear whether we still need this special case anyway)",
      "issue_closed_at": "2017-03-11T15:24:28Z",
      "base_commit": "a79801c044c2b0ed74176e4abc18f4ca2b38ac58",
      "changes": [
        {
          "file": "sympy/core/expr.py",
          "type": "line",
          "name": "line 11",
          "code": "\nfrom collections import defaultdict\n\n\nclass Expr(Basic, EvalfMixin):\n    \"\"\"\n    Base class for algebraic expressions."
        },
        {
          "file": "sympy/core/expr.py",
          "type": "function",
          "name": "extract_multiplicatively",
          "class_name": "Expr",
          "code": "def extract_multiplicatively(self, c):\n        \"\"\"Return None if it's not possible to make self in the form\n           c * something in a nice way, i.e. preserving the properties\n           of arguments of self.\n\n           >>> from sympy import symbols, Rational\n\n           >>> x, y = symbols('x,y', real=True)\n\n           >>> ((x*y)**3).extract_multiplicatively(x**2 * y)\n           x*y**2\n\n           >>> ((x*y)**3).extract_multiplicatively(x**4 * y)\n\n           >>> (2*x).extract_multiplicatively(2)\n           x\n\n           >>> (2*x).extract_multiplicatively(3)\n\n           >>> (Rational(1, 2)*x).extract_multiplicatively(3)\n           x/6\n\n        \"\"\"\n        c = sympify(c)\n        if self is S.NaN:\n            return None\n        if c is S.One:\n            return self\n        elif c == self:\n            return S.One\n        if c.is_Add:\n            cc, pc = c.primitive()\n            if cc is not S.One:\n                c = Mul(cc, pc, evaluate=False)\n        if c.is_Mul:\n            a, b = c.as_two_terms()\n            x = self.extract_multiplicatively(a)\n            if x is not None:\n                return x.extract_multiplicatively(b)\n        quotient = self / c\n        if self.is_Number:\n            if self is S.Infinity:\n                if c.is_positive:\n                    return S.Infinity\n            elif self is S.NegativeInfinity:\n                if c.is_negative:\n                    return S.Infinity\n                elif c.is_positive:\n                    return S.NegativeInfinity\n            elif self is S.ComplexInfinity:\n                if not c.is_zero:\n                    return S.ComplexInfinity\n            elif self.is_Integer:\n                if not quotient.is_Integer:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Rational:\n                if not quotient.is_Rational:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Float:\n                if not quotient.is_Float:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n        elif self.is_NumberSymbol or self.is_Symbol or self is S.ImaginaryUnit:\n            if quotient.is_Mul and len(quotient.args) == 2:\n                if quotient.args[0].is_Integer and quotient.args[0].is_positive and quotient.args[1] == self:\n                    return quotient\n            elif quotient.is_Integer and c.is_Number:\n                return quotient\n        elif self.is_Add:\n            cs, ps = self.primitive()\n            if cs is not S.One:\n                return Mul(cs, ps, evaluate=False).extract_multiplicatively(c)\n            newargs = []\n            for arg in self.args:\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    newargs.append(newarg)\n                else:\n                    return None\n            return Add(*newargs)\n        elif self.is_Mul:\n            args = list(self.args)\n            for i, arg in enumerate(args):\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    args[i] = newarg\n                    return Mul(*args)\n        elif self.is_Pow:\n            if c.is_Pow and c.base == self.base:\n                new_exp = self.exp.extract_additively(c.exp)\n                if new_exp is not None:\n                    return self.base ** (new_exp)\n            elif c == self.base:\n                new_exp = self.exp.extract_additively(1)\n                if new_exp is not None:\n                    return self.base ** (new_exp)"
        },
        {
          "file": "sympy/core/expr.py",
          "type": "function",
          "name": "extract_multiplicatively",
          "class_name": "Expr",
          "code": "def extract_multiplicatively(self, c):\n        \"\"\"Return None if it's not possible to make self in the form\n           c * something in a nice way, i.e. preserving the properties\n           of arguments of self.\n\n           >>> from sympy import symbols, Rational\n\n           >>> x, y = symbols('x,y', real=True)\n\n           >>> ((x*y)**3).extract_multiplicatively(x**2 * y)\n           x*y**2\n\n           >>> ((x*y)**3).extract_multiplicatively(x**4 * y)\n\n           >>> (2*x).extract_multiplicatively(2)\n           x\n\n           >>> (2*x).extract_multiplicatively(3)\n\n           >>> (Rational(1, 2)*x).extract_multiplicatively(3)\n           x/6\n\n        \"\"\"\n        c = sympify(c)\n        if self is S.NaN:\n            return None\n        if c is S.One:\n            return self\n        elif c == self:\n            return S.One\n        if c.is_Add:\n            cc, pc = c.primitive()\n            if cc is not S.One:\n                c = Mul(cc, pc, evaluate=False)\n        if c.is_Mul:\n            a, b = c.as_two_terms()\n            x = self.extract_multiplicatively(a)\n            if x is not None:\n                return x.extract_multiplicatively(b)\n        quotient = self / c\n        if self.is_Number:\n            if self is S.Infinity:\n                if c.is_positive:\n                    return S.Infinity\n            elif self is S.NegativeInfinity:\n                if c.is_negative:\n                    return S.Infinity\n                elif c.is_positive:\n                    return S.NegativeInfinity\n            elif self is S.ComplexInfinity:\n                if not c.is_zero:\n                    return S.ComplexInfinity\n            elif self.is_Integer:\n                if not quotient.is_Integer:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Rational:\n                if not quotient.is_Rational:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Float:\n                if not quotient.is_Float:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n        elif self.is_NumberSymbol or self.is_Symbol or self is S.ImaginaryUnit:\n            if quotient.is_Mul and len(quotient.args) == 2:\n                if quotient.args[0].is_Integer and quotient.args[0].is_positive and quotient.args[1] == self:\n                    return quotient\n            elif quotient.is_Integer and c.is_Number:\n                return quotient\n        elif self.is_Add:\n            cs, ps = self.primitive()\n            if cs is not S.One:\n                return Mul(cs, ps, evaluate=False).extract_multiplicatively(c)\n            newargs = []\n            for arg in self.args:\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    newargs.append(newarg)\n                else:\n                    return None\n            return Add(*newargs)\n        elif self.is_Mul:\n            args = list(self.args)\n            for i, arg in enumerate(args):\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    args[i] = newarg\n                    return Mul(*args)\n        elif self.is_Pow:\n            if c.is_Pow and c.base == self.base:\n                new_exp = self.exp.extract_additively(c.exp)\n                if new_exp is not None:\n                    return self.base ** (new_exp)\n            elif c == self.base:\n                new_exp = self.exp.extract_additively(1)\n                if new_exp is not None:\n                    return self.base ** (new_exp)"
        },
        {
          "file": "sympy/core/expr.py",
          "type": "function",
          "name": "extract_multiplicatively",
          "class_name": "Expr",
          "code": "def extract_multiplicatively(self, c):\n        \"\"\"Return None if it's not possible to make self in the form\n           c * something in a nice way, i.e. preserving the properties\n           of arguments of self.\n\n           >>> from sympy import symbols, Rational\n\n           >>> x, y = symbols('x,y', real=True)\n\n           >>> ((x*y)**3).extract_multiplicatively(x**2 * y)\n           x*y**2\n\n           >>> ((x*y)**3).extract_multiplicatively(x**4 * y)\n\n           >>> (2*x).extract_multiplicatively(2)\n           x\n\n           >>> (2*x).extract_multiplicatively(3)\n\n           >>> (Rational(1, 2)*x).extract_multiplicatively(3)\n           x/6\n\n        \"\"\"\n        c = sympify(c)\n        if self is S.NaN:\n            return None\n        if c is S.One:\n            return self\n        elif c == self:\n            return S.One\n        if c.is_Add:\n            cc, pc = c.primitive()\n            if cc is not S.One:\n                c = Mul(cc, pc, evaluate=False)\n        if c.is_Mul:\n            a, b = c.as_two_terms()\n            x = self.extract_multiplicatively(a)\n            if x is not None:\n                return x.extract_multiplicatively(b)\n        quotient = self / c\n        if self.is_Number:\n            if self is S.Infinity:\n                if c.is_positive:\n                    return S.Infinity\n            elif self is S.NegativeInfinity:\n                if c.is_negative:\n                    return S.Infinity\n                elif c.is_positive:\n                    return S.NegativeInfinity\n            elif self is S.ComplexInfinity:\n                if not c.is_zero:\n                    return S.ComplexInfinity\n            elif self.is_Integer:\n                if not quotient.is_Integer:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Rational:\n                if not quotient.is_Rational:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n            elif self.is_Float:\n                if not quotient.is_Float:\n                    return None\n                elif self.is_positive and quotient.is_negative:\n                    return None\n                else:\n                    return quotient\n        elif self.is_NumberSymbol or self.is_Symbol or self is S.ImaginaryUnit:\n            if quotient.is_Mul and len(quotient.args) == 2:\n                if quotient.args[0].is_Integer and quotient.args[0].is_positive and quotient.args[1] == self:\n                    return quotient\n            elif quotient.is_Integer and c.is_Number:\n                return quotient\n        elif self.is_Add:\n            cs, ps = self.primitive()\n            if cs is not S.One:\n                return Mul(cs, ps, evaluate=False).extract_multiplicatively(c)\n            newargs = []\n            for arg in self.args:\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    newargs.append(newarg)\n                else:\n                    return None\n            return Add(*newargs)\n        elif self.is_Mul:\n            args = list(self.args)\n            for i, arg in enumerate(args):\n                newarg = arg.extract_multiplicatively(c)\n                if newarg is not None:\n                    args[i] = newarg\n                    return Mul(*args)\n        elif self.is_Pow:\n            if c.is_Pow and c.base == self.base:\n                new_exp = self.exp.extract_additively(c.exp)\n                if new_exp is not None:\n                    return self.base ** (new_exp)\n            elif c == self.base:\n                new_exp = self.exp.extract_additively(1)\n                if new_exp is not None:\n                    return self.base ** (new_exp)"
        }
      ]
    },
    {
      "pr_number": 8627,
      "pr_title": "Bug Fix #8626",
      "pr_body": "Let me know if a test case is needed\n",
      "issue_id": 8626,
      "issue_title": "bug in physics/mechanics/lagrange.py",
      "issue_body": "When providing a `forcelist` with more than one force, not all terms are computed,\nthis comes from the fact that the iterator `flist = zip(*_f_list_parser(self.forcelist, N))` is consumed in the first iteration of the loop.\n\nI propose a following PR that fixes the bug, let me know if I need to also provide a test case.\n\nBest,\nGuillaume\n",
      "issue_closed_at": "2015-01-06T16:57:16Z",
      "base_commit": "65f7c8c2c9c1927eaa8520c4ce06864f93a20ad1",
      "changes": [
        {
          "file": "sympy/physics/mechanics/lagrange.py",
          "type": "function",
          "name": "form_lagranges_equations",
          "class_name": "LagrangesMethod",
          "code": "def form_lagranges_equations(self):\n        \"\"\"Method to form Lagrange's equations of motion.\n\n        Returns a vector of equations of motion using Lagrange's equations of\n        the second kind.\n        \"\"\"\n\n        qds = self._qdots\n        qdd_zero = dict((i, 0) for i in self._qdoubledots)\n        n = len(self.q)\n\n        # Internally we represent the EOM as four terms:\n        # EOM = term1 - term2 - term3 - term4 = 0\n\n        # First term\n        self._term1 = self._L.jacobian(qds)\n        self._term1 = self._term1.diff(dynamicsymbols._t).T\n\n        # Second term\n        self._term2 = self._L.jacobian(self.q).T\n\n        # Third term\n        if self.coneqs:\n            coneqs = self.coneqs\n            m = len(coneqs)\n            # Creating the multipliers\n            self.lam_vec = Matrix(dynamicsymbols('lam1:' + str(m + 1)))\n            self.lam_coeffs = -coneqs.jacobian(qds)\n            self._term3 = self.lam_coeffs.T * self.lam_vec\n            # Extracting the coeffecients of the qdds from the diff coneqs\n            diffconeqs = coneqs.diff(dynamicsymbols._t)\n            self._m_cd = diffconeqs.jacobian(self._qdoubledots)\n            # The remaining terms i.e. the 'forcing' terms in diff coneqs\n            self._f_cd = -diffconeqs.subs(qdd_zero)\n        else:\n            self._term3 = zeros(n, 1)\n\n        # Fourth term\n        if self.forcelist:\n            N = self.inertial\n            self._term4 = zeros(n, 1)\n            flist = zip(*_f_list_parser(self.forcelist, N))\n            for i, qd in enumerate(qds):\n                for obj, force in self.forcelist:\n                    self._term4[i] = sum(v.diff(qd, N) & f for (v, f) in flist)\n        else:\n            self._term4 = zeros(n, 1)\n\n        # Form the dynamic mass and forcing matrices\n        without_lam = self._term1 - self._term2 - self._term4\n        self._m_d = without_lam.jacobian(self._qdoubledots)\n        self._f_d = -without_lam.subs(qdd_zero)\n\n        # Form the EOM\n        self.eom = without_lam - self._term3\n        return self.eom"
        }
      ]
    },
    {
      "pr_number": 7928,
      "pr_title": "give MaxMinBase an evalf routine",
      "pr_body": "Although Or, And, etc... will not have a number as an\nargument, Min and Max can. When the operations version\nof _eval_evalf is called a call to as_independent is\nmade. This creates an x (=1) that is assumed to be an\nargument of the object so Max(0,y).n() becomes Max(1, 0, y) -> Max(1, y).\nTo avoid this an evalf routine was added to handle MaxMinBase separately.\n\nfixes #7233\n",
      "issue_id": 7233,
      "issue_title": "N(Max(0.0, n + m)) returns Max(1.0, n + m)",
      "issue_body": "```\nThe following code does not return what is expected,\n\nn = sympy.Symbol('n')\nm = sympy.Symbol('m')\na = sympy.Max(0.0, n + m)\nprint a\nprint sympy.N(a)\n\noutputs:\n>> Max(0.0, m + n)\n>> Max(1.0, m + n)\n```\n\nOriginal issue for #7233: http://code.google.com/p/sympy/issues/detail?id=4134\nOriginal author: https://code.google.com/u/100679098312105897492/\n",
      "issue_closed_at": "2014-08-29T23:58:03Z",
      "base_commit": "d365f932a26c28c4c57b2b9f6dddee2ca94a5c66",
      "changes": [
        {
          "file": "sympy/functions/elementary/miscellaneous.py",
          "type": "function",
          "name": "_eval_derivative",
          "class_name": "MinMaxBase",
          "code": "def _eval_derivative(self, s):\n        # f(x).diff(s) -> x.diff(s) * f.fdiff(1)(s)\n        i = 0\n        l = []\n        for a in self.args:\n            i += 1\n            da = a.diff(s)\n            if da is S.Zero:\n                continue\n            try:\n                df = self.fdiff(i)\n            except ArgumentIndexError:\n                df = Function.fdiff(self, i)\n            l.append(df * da)\n        return Add(*l)"
        }
      ]
    },
    {
      "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)"
        }
      ]
    }
  ]
}