{
  "instance_id": "sympy__sympy-20049",
  "repo": "sympy/sympy",
  "created_at": "2020-09-05T09:37:44Z",
  "problem_statement": "Point.vel() should calculate the velocity if possible\nIf you specify the orientation of two reference frames and then ask for the angular velocity between the two reference frames the angular velocity will be calculated. But if you try to do the same thing with velocities, this doesn't work. See below:\r\n\r\n```\r\nIn [1]: import sympy as sm                                                                               \r\n\r\nIn [2]: import sympy.physics.mechanics as me                                                             \r\n\r\nIn [3]: A = me.ReferenceFrame('A')                                                                       \r\n\r\nIn [5]: q = me.dynamicsymbols('q')                                                                       \r\n\r\nIn [6]: B = A.orientnew('B', 'Axis', (q, A.x))                                                           \r\n\r\nIn [7]: B.ang_vel_in(A)                                                                                  \r\nOut[7]: q'*A.x\r\n\r\nIn [9]: P = me.Point('P')                                                                                \r\n\r\nIn [10]: Q = me.Point('Q')                                                                               \r\n\r\nIn [11]: r = q*A.x + 2*q*A.y                                                                             \r\n\r\nIn [12]: Q.set_pos(P, r)                                                                                 \r\n\r\nIn [13]: Q.vel(A)                                                                                        \r\n---------------------------------------------------------------------------\r\nValueError                                Traceback (most recent call last)\r\n<ipython-input-13-0fc8041904cc> in <module>\r\n----> 1 Q.vel(A)\r\n\r\n~/miniconda3/lib/python3.6/site-packages/sympy/physics/vector/point.py in vel(self, frame)\r\n    453         if not (frame in self._vel_dict):\r\n    454             raise ValueError('Velocity of point ' + self.name + ' has not been'\r\n--> 455                              ' defined in ReferenceFrame ' + frame.name)\r\n    456         return self._vel_dict[frame]\r\n    457 \r\n\r\nValueError: Velocity of point Q has not been defined in ReferenceFrame A\r\n```\r\n\r\nThe expected result of the `Q.vel(A)` should be:\r\n\r\n```\r\nIn [14]: r.dt(A)                                                                                         \r\nOut[14]: q'*A.x + 2*q'*A.y\r\n```\r\n\r\nI think that this is possible. Maybe there is a reason it isn't implemented. But we should try to implement it because it is confusing why this works for orientations and not positions.\r\n\r\n\n",
  "patch": "diff --git a/sympy/physics/vector/point.py b/sympy/physics/vector/point.py\n--- a/sympy/physics/vector/point.py\n+++ b/sympy/physics/vector/point.py\n@@ -483,19 +483,49 @@ def vel(self, frame):\n         Examples\n         ========\n \n-        >>> from sympy.physics.vector import Point, ReferenceFrame\n+        >>> from sympy.physics.vector import Point, ReferenceFrame, dynamicsymbols\n         >>> N = ReferenceFrame('N')\n         >>> p1 = Point('p1')\n         >>> p1.set_vel(N, 10 * N.x)\n         >>> p1.vel(N)\n         10*N.x\n \n+        Velocities will be automatically calculated if possible, otherwise a ``ValueError`` will be returned. If it is possible to calculate multiple different velocities from the relative points, the points defined most directly relative to this point will be used. In the case of inconsistent relative positions of points, incorrect velocities may be returned. It is up to the user to define prior relative positions and velocities of points in a self-consistent way.\n+\n+        >>> p = Point('p')\n+        >>> q = dynamicsymbols('q')\n+        >>> p.set_vel(N, 10 * N.x)\n+        >>> p2 = Point('p2')\n+        >>> p2.set_pos(p, q*N.x)\n+        >>> p2.vel(N)\n+        (Derivative(q(t), t) + 10)*N.x\n+\n         \"\"\"\n \n         _check_frame(frame)\n         if not (frame in self._vel_dict):\n-            raise ValueError('Velocity of point ' + self.name + ' has not been'\n+            visited = []\n+            queue = [self]\n+            while queue: #BFS to find nearest point\n+                node = queue.pop(0)\n+                if node not in visited:\n+                    visited.append(node)\n+                    for neighbor, neighbor_pos in node._pos_dict.items():\n+                        try:\n+                            neighbor_pos.express(frame) #Checks if pos vector is valid\n+                        except ValueError:\n+                            continue\n+                        try :\n+                            neighbor_velocity = neighbor._vel_dict[frame] #Checks if point has its vel defined in req frame\n+                        except KeyError:\n+                            queue.append(neighbor)\n+                            continue\n+                        self.set_vel(frame, self.pos_from(neighbor).dt(frame) + neighbor_velocity)\n+                        return self._vel_dict[frame]\n+            else:\n+                raise ValueError('Velocity of point ' + self.name + ' has not been'\n                              ' defined in ReferenceFrame ' + frame.name)\n+\n         return self._vel_dict[frame]\n \n     def partial_velocity(self, frame, *gen_speeds):\n",
  "similar_bug_items": [
    {
      "pr_number": 15349,
      "pr_title": "quaternions: fix a sign mistake in the definition of the rotation matrix",
      "pr_body": "quaternions: fix a sign mistake in definition of rotation matrix\r\n<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests .-->\r\n\r\nFixes #15193.\r\n#### Brief description of what is fixed or changed\r\nSign was corrected. A unit test added. Other unit tests fixed with the correct values.\r\n\r\n#### Other comments\r\n\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* algebras\r\n    * fix sign mistake in Quaternion.to_rotation_matrix method\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15193,
      "issue_title": "Incorrect result with Quaterniont.to_rotation_matrix()",
      "issue_body": "https://github.com/sympy/sympy/blob/ab14b02dba5a7e3e4fb1e807fc8a954f1047a1a1/sympy/algebras/quaternion.py#L489\r\n\r\nThere appears to be an error in the `Quaternion.to_rotation_matrix()` output.  The simplest example I created to illustrate the problem is as follows:\r\n\r\n```\r\n>>import sympy\r\n>>print('Sympy version: ', sympy.__version__)\r\nSympy version: 1.2\r\n\r\n>> from sympy import *\r\n>> x = symbols('x')\r\n>> q = Quaternion(cos(x/2), sin(x/2), 0, 0)\r\n>> trigsimp(q.to_rotation_matrix())\r\nMatrix([\r\n[1,      0,      0],\r\n[0, cos(x), sin(x)],\r\n[0, sin(x), cos(x)]])\r\n```\r\nOne of the `sin(x)` functions should be negative.  What was the reference of the original equations?  ",
      "issue_closed_at": "2018-10-07T10:54:51Z",
      "base_commit": "768da1c6f6ec907524b8ebbf6bf818c92b56101b",
      "changes": [
        {
          "file": "sympy/algebras/quaternion.py",
          "type": "function",
          "name": "to_rotation_matrix",
          "class_name": "Quaternion",
          "code": "def to_rotation_matrix(self, v=None):\n        \"\"\"Returns the equivalent rotation transformation matrix of the quaternion\n        which represents rotation about the origin if v is not passed.\n\n        Example\n        ========\n\n        >>> from sympy.algebras.quaternion import Quaternion\n        >>> from sympy import symbols, trigsimp, cos, sin\n        >>> x = symbols('x')\n        >>> q = Quaternion(cos(x/2), 0, 0, sin(x/2))\n        >>> trigsimp(q.to_rotation_matrix())\n        Matrix([\n        [cos(x), -sin(x), 0],\n        [sin(x),  cos(x), 0],\n        [     0,       0, 1]])\n\n        Generates a 4x4 transformation matrix (used for rotation about a point\n        other than the origin) if the point(v) is passed as an argument.\n\n        Example\n        ========\n\n        >>> from sympy.algebras.quaternion import Quaternion\n        >>> from sympy import symbols, trigsimp, cos, sin\n        >>> x = symbols('x')\n        >>> q = Quaternion(cos(x/2), 0, 0, sin(x/2))\n        >>> trigsimp(q.to_rotation_matrix((1, 1, 1)))\n         Matrix([\n        [cos(x), -sin(x), 0,  sin(x) - cos(x) + 1],\n        [sin(x),  cos(x), 0, -sin(x) - cos(x) + 1],\n        [     0,       0, 1,                    0],\n        [     0,       0, 0,                    1]])\n        \"\"\"\n\n        q = self\n        s = q.norm()**-2\n        m00 = 1 - 2*s*(q.c**2 + q.d**2)\n        m01 = 2*s*(q.b*q.c - q.d*q.a)\n        m02 = 2*s*(q.b*q.d + q.c*q.a)\n\n        m10 = 2*s*(q.b*q.c + q.d*q.a)\n        m11 = 1 - 2*s*(q.b**2 + q.d**2)\n        m12 = 2*s*(q.c*q.d + q.b*q.a)\n\n        m20 = 2*s*(q.b*q.d - q.c*q.a)\n        m21 = 2*s*(q.c*q.d + q.b*q.a)\n        m22 = 1 - 2*s*(q.b**2 + q.c**2)\n\n        if not v:\n            return Matrix([[m00, m01, m02], [m10, m11, m12], [m20, m21, m22]])\n\n        else:\n            (x, y, z) = v\n\n            m03 = x - x*m00 - y*m01 - z*m02\n            m13 = y - x*m10 - y*m11 - z*m12\n            m23 = z - x*m20 - y*m21 - z*m22\n            m30 = m31 = m32 = 0\n            m33 = 1\n\n            return Matrix([[m00, m01, m02, m03], [m10, m11, m12, m13],\n                          [m20, m21, m22, m23], [m30, m31, m32, m33]])"
        }
      ]
    },
    {
      "pr_number": 17769,
      "pr_title": "Fix linsolve  for immutable matrix",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests . Please also\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #17762 \r\nFixes #17722 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nI've fixed the type checking for `linearsolve`, and also fixed some usage of the mutability of a matrix in `gauss_jordan_solve`.\r\n\r\n#### Other comments\r\n\r\nI'd also try to check `linsolve` whether it works with matrix expressions.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n- matrices\r\n  - Fixed `Matrix.gauss_jordan_solve` raising `ShapeError` for immutable matrices.\r\n- solvers\r\n  - Fixed `linsolve` raising errors for `ImmutableMatrix`.\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 17722,
      "issue_title": "linsolve error; ValueError: Got rows of variable lengths: [1, 3]",
      "issue_body": "Can anyone tell me if this is a bug and if there is a method to get around this please, cause it seems like the linsolve() function is broken?\r\nA similar question was asked in issue #17430 but this doesn't solve my problem when used in jupyter or qtconsole or spyder. This function would be really useful if it worked for my uni study.\r\n\r\nParameters given are: *system* in form of Ax=b and variables *c* in a tuple of Sympy Symbols\r\n![image](https://user-images.githubusercontent.com/42601907/66694509-028ba580-ed00-11e9-85db-f8e1567a5306.png)\r\n![image](https://user-images.githubusercontent.com/42601907/66694519-1505df00-ed00-11e9-9861-02ea8f1dd249.png)\r\n",
      "issue_closed_at": "2019-10-21T21:12:36Z",
      "base_commit": "4c87ea3ec5a36b0d3364b9bc52f88653282bf9aa",
      "changes": [
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/matrices/matrices.py",
          "type": "function",
          "name": "gauss_jordan_solve",
          "class_name": "MatrixBase",
          "code": "def gauss_jordan_solve(self, B, freevar=False):\n        \"\"\"\n        Solves ``Ax = B`` using Gauss Jordan elimination.\n\n        There may be zero, one, or infinite solutions.  If one solution\n        exists, it will be returned. If infinite solutions exist, it will\n        be returned parametrically. If no solutions exist, It will throw\n        ValueError.\n\n        Parameters\n        ==========\n\n        B : Matrix\n            The right hand side of the equation to be solved for.  Must have\n            the same number of rows as matrix A.\n\n        freevar : List\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            values of free variables. Then the index of the free variables\n            in the solutions (column Matrix) will be returned by freevar, if\n            the flag `freevar` is set to `True`.\n\n        Returns\n        =======\n\n        x : Matrix\n            The matrix that will satisfy ``Ax = B``.  Will have as many rows as\n            matrix A has columns, and as many columns as matrix B.\n\n        params : Matrix\n            If the system is underdetermined (e.g. A has more columns than\n            rows), infinite solutions are possible, in terms of arbitrary\n            parameters. These arbitrary parameters are returned as params\n            Matrix.\n\n        Examples\n        ========\n\n        >>> from sympy import Matrix\n        >>> A = Matrix([[1, 2, 1, 1], [1, 2, 2, -1], [2, 4, 0, 6]])\n        >>> B = Matrix([7, 12, 4])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-2*tau0 - 3*tau1 + 2],\n        [                 tau0],\n        [           2*tau1 + 5],\n        [                 tau1]])\n        >>> params\n        Matrix([\n        [tau0],\n        [tau1]])\n        >>> taus_zeroes = { tau:0 for tau in params }\n        >>> sol_unique = sol.xreplace(taus_zeroes)\n        >>> sol_unique\n         Matrix([\n        [2],\n        [0],\n        [5],\n        [0]])\n\n\n        >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n        >>> B = Matrix([3, 6, 9])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [-1],\n        [ 2],\n        [ 0]])\n        >>> params\n        Matrix(0, 1, [])\n\n        >>> A = Matrix([[2, -7], [-1, 4]])\n        >>> B = Matrix([[-21, 3], [12, -2]])\n        >>> sol, params = A.gauss_jordan_solve(B)\n        >>> sol\n        Matrix([\n        [0, -2],\n        [3, -1]])\n        >>> params\n        Matrix(0, 2, [])\n\n        See Also\n        ========\n\n        lower_triangular_solve\n        upper_triangular_solve\n        cholesky_solve\n        diagonal_solve\n        LDLsolve\n        LUsolve\n        QRsolve\n        pinv\n\n        References\n        ==========\n\n        .. [1] https://en.wikipedia.org/wiki/Gaussian_elimination\n\n        \"\"\"\n        from sympy.matrices import Matrix, zeros\n\n        aug = self.hstack(self.copy(), B.copy())\n        B_cols = B.cols\n        row, col = aug[:, :-B_cols].shape\n\n        # solve by reduced row echelon form\n        A, pivots = aug.rref(simplify=True)\n        A, v = A[:, :-B_cols], A[:, -B_cols:]\n        pivots = list(filter(lambda p: p < col, pivots))\n        rank = len(pivots)\n\n        # Bring to block form\n        permutation = Matrix(range(col)).T\n\n        for i, c in enumerate(pivots):\n            permutation.col_swap(i, c)\n\n\n        # check for existence of solutions\n        # rank of aug Matrix should be equal to rank of coefficient matrix\n        if not v[rank:, :].is_zero:\n            raise ValueError(\"Linear system has no solution\")\n\n        # Get index of free symbols (free parameters)\n        free_var_index = permutation[\n                         len(pivots):]  # non-pivots columns are free variables\n\n        # Free parameters\n        # what are current unnumbered free symbol names?\n        name = _uniquely_named_symbol('tau', aug,\n            compare=lambda i: str(i).rstrip('1234567890')).name\n        gen = numbered_symbols(name)\n        tau = Matrix([next(gen) for k in range((col - rank)*B_cols)]).reshape(\n            col - rank, B_cols)\n\n        # Full parametric solution\n        V = A[:rank,:]\n        for c in reversed(pivots):\n            V.col_del(c)\n        vt = v[:rank, :]\n        free_sol = tau.vstack(vt - V * tau, tau)\n\n        # Undo permutation\n        sol = zeros(col, B_cols)\n        for k in range(col):\n            sol[permutation[k], :] = free_sol[k,:]\n\n        if freevar:\n            return sol, tau, free_var_index\n        else:\n            return sol, tau"
        },
        {
          "file": "sympy/solvers/solveset.py",
          "type": "function",
          "name": "linsolve",
          "class_name": null,
          "code": "def linsolve(system, *symbols):\n    r\"\"\"\n    Solve system of N linear equations with M variables; both\n    underdetermined and overdetermined systems are supported.\n    The possible number of solutions is zero, one or infinite.\n    Zero solutions throws a ValueError, whereas infinite\n    solutions are represented parametrically in terms of the given\n    symbols. For unique solution a FiniteSet of ordered tuples\n    is returned.\n\n    All Standard input formats are supported:\n    For the given set of Equations, the respective input types\n    are given below:\n\n    .. math:: 3x + 2y -   z = 1\n    .. math:: 2x - 2y + 4z = -2\n    .. math:: 2x -   y + 2z = 0\n\n    * Augmented Matrix Form, `system` given below:\n\n    ::\n\n              [3   2  -1  1]\n     system = [2  -2   4 -2]\n              [2  -1   2  0]\n\n    * List Of Equations Form\n\n    `system  =  [3x + 2y - z - 1, 2x - 2y + 4z + 2, 2x - y + 2z]`\n\n    * Input A & b Matrix Form (from Ax = b) are given as below:\n\n    ::\n\n         [3   2  -1 ]         [  1 ]\n     A = [2  -2   4 ]    b =  [ -2 ]\n         [2  -1   2 ]         [  0 ]\n\n    `system = (A, b)`\n\n    Symbols can always be passed but are actually only needed\n    when 1) a system of equations is being passed and 2) the\n    system is passed as an underdetermined matrix and one wants\n    to control the name of the free variables in the result.\n    An error is raised if no symbols are used for case 1, but if\n    no symbols are provided for case 2, internally generated symbols\n    will be provided. When providing symbols for case 2, there should\n    be at least as many symbols are there are columns in matrix A.\n\n    The algorithm used here is Gauss-Jordan elimination, which\n    results, after elimination, in a row echelon form matrix.\n\n    Returns\n    =======\n\n    A FiniteSet containing an ordered tuple of values for the\n    unknowns for which the `system` has a solution. (Wrapping\n    the tuple in FiniteSet is used to maintain a consistent\n    output format throughout solveset.)\n\n    Returns EmptySet(), if the linear system is inconsistent.\n\n    Raises\n    ======\n\n    ValueError\n        The input is not valid.\n        The symbols are not given.\n\n    Examples\n    ========\n\n    >>> from sympy import Matrix, S, linsolve, symbols\n    >>> x, y, z = symbols(\"x, y, z\")\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n    >>> b = Matrix([3, 6, 9])\n    >>> A\n    Matrix([\n    [1, 2,  3],\n    [4, 5,  6],\n    [7, 8, 10]])\n    >>> b\n    Matrix([\n    [3],\n    [6],\n    [9]])\n    >>> linsolve((A, b), [x, y, z])\n    {(-1, 2, 0)}\n\n    * Parametric Solution: In case the system is underdetermined, the\n      function will return a parametric solution in terms of the given\n      symbols. Those that are free will be returned unchanged. e.g. in\n      the system below, `z` is returned as the solution for variable z;\n      it can take on any value.\n\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])\n    >>> b = Matrix([3, 6, 9])\n    >>> linsolve((A, b), x, y, z)\n    {(z - 1, 2 - 2*z, z)}\n\n    If no symbols are given, internally generated symbols will be used.\n    The `tau0` in the 3rd position indicates (as before) that the 3rd\n    variable -- whatever it's named -- can take on any value:\n\n    >>> linsolve((A, b))\n    {(tau0 - 1, 2 - 2*tau0, tau0)}\n\n    * List of Equations as input\n\n    >>> Eqns = [3*x + 2*y - z - 1, 2*x - 2*y + 4*z + 2, - x + y/2 - z]\n    >>> linsolve(Eqns, x, y, z)\n    {(1, -2, -2)}\n\n    * Augmented Matrix as input\n\n    >>> aug = Matrix([[2, 1, 3, 1], [2, 6, 8, 3], [6, 8, 18, 5]])\n    >>> aug\n    Matrix([\n    [2, 1,  3, 1],\n    [2, 6,  8, 3],\n    [6, 8, 18, 5]])\n    >>> linsolve(aug, x, y, z)\n    {(3/10, 2/5, 0)}\n\n    * Solve for symbolic coefficients\n\n    >>> a, b, c, d, e, f = symbols('a, b, c, d, e, f')\n    >>> eqns = [a*x + b*y - c, d*x + e*y - f]\n    >>> linsolve(eqns, x, y)\n    {((-b*f + c*e)/(a*e - b*d), (a*f - c*d)/(a*e - b*d))}\n\n    * A degenerate system returns solution as set of given\n      symbols.\n\n    >>> system = Matrix(([0, 0, 0], [0, 0, 0], [0, 0, 0]))\n    >>> linsolve(system, x, y)\n    {(x, y)}\n\n    * For an empty system linsolve returns empty set\n\n    >>> linsolve([], x)\n    EmptySet()\n\n    * An error is raised if, after expansion, any nonlinearity\n      is detected:\n\n    >>> linsolve([x*(1/x - 1), (y - 1)**2 - y**2 + 1], x, y)\n    {(1, 1)}\n    >>> linsolve([x**2 - 1], x)\n    Traceback (most recent call last):\n    ...\n    ValueError:\n    The term x**2 is nonlinear in {x}\n    \"\"\"\n    if not system:\n        return S.EmptySet\n\n    # If second argument is an iterable\n    if symbols and hasattr(symbols[0], '__iter__'):\n        symbols = symbols[0]\n    sym_gen = isinstance(symbols, GeneratorType)\n\n    swap = {}\n    b = None  # if we don't get b the input was bad\n    syms_needed_msg = None\n\n    # unpack system\n\n    if hasattr(system, '__iter__'):\n\n        # 1). (A, b)\n        if len(system) == 2 and isinstance(system[0], Matrix):\n            A, b = system\n\n        # 2). (eq1, eq2, ...)\n        if not isinstance(system[0], Matrix):\n            if sym_gen or not symbols:\n                raise ValueError(filldedent('''\n                    When passing a system of equations, the explicit\n                    symbols for which a solution is being sought must\n                    be given as a sequence, too.\n                '''))\n            system = [\n                _mexpand(i.lhs - i.rhs if isinstance(i, Eq) else i,\n                recursive=True) for i in system]\n            system, symbols, swap = recast_to_symbols(system, symbols)\n            A, b = linear_eq_to_matrix(system, symbols)\n            syms_needed_msg = 'free symbols in the equations provided'\n\n    elif isinstance(system, Matrix) and not (\n            symbols and not isinstance(symbols, GeneratorType) and\n            isinstance(symbols[0], Matrix)):\n        # 3). A augmented with b\n        A, b = system[:, :-1], system[:, -1:]\n\n    if b is None:\n        raise ValueError(\"Invalid arguments\")\n\n    syms_needed_msg  = syms_needed_msg or 'columns of A'\n\n    if sym_gen:\n        symbols = [next(symbols) for i in range(A.cols)]\n        if any(set(symbols) & (A.free_symbols | b.free_symbols)):\n            raise ValueError(filldedent('''\n                At least one of the symbols provided\n                already appears in the system to be solved.\n                One way to avoid this is to use Dummy symbols in\n                the generator, e.g. numbered_symbols('%s', cls=Dummy)\n            ''' % symbols[0].name.rstrip('1234567890')))\n\n    try:\n        solution, params, free_syms = A.gauss_jordan_solve(b, freevar=True)\n    except ValueError:\n        # No solution\n        return S.EmptySet\n\n    # Replace free parameters with free symbols\n    if params:\n        if not symbols:\n            symbols = [_ for _ in params]\n            # re-use the parameters but put them in order\n            # params       [x, y, z]\n            # free_symbols [2, 0, 4]\n            # idx          [1, 0, 2]\n            idx = list(zip(*sorted(zip(free_syms, range(len(free_syms))))))[1]\n            # simultaneous replacements {y: x, x: y, z: z}\n            replace_dict = dict(zip(symbols, [symbols[i] for i in idx]))\n        elif len(symbols) >= A.cols:\n            replace_dict = {v: symbols[free_syms[k]] for k, v in enumerate(params)}\n        else:\n            raise IndexError(filldedent('''\n                the number of symbols passed should have a length\n                equal to the number of %s.\n                ''' % syms_needed_msg))\n        solution = [sol.xreplace(replace_dict) for sol in solution]\n\n    solution = [simplify(sol).xreplace(swap) for sol in solution]\n    return FiniteSet(tuple(solution))"
        },
        {
          "file": "sympy/solvers/solveset.py",
          "type": "function",
          "name": "linsolve",
          "class_name": null,
          "code": "def linsolve(system, *symbols):\n    r\"\"\"\n    Solve system of N linear equations with M variables; both\n    underdetermined and overdetermined systems are supported.\n    The possible number of solutions is zero, one or infinite.\n    Zero solutions throws a ValueError, whereas infinite\n    solutions are represented parametrically in terms of the given\n    symbols. For unique solution a FiniteSet of ordered tuples\n    is returned.\n\n    All Standard input formats are supported:\n    For the given set of Equations, the respective input types\n    are given below:\n\n    .. math:: 3x + 2y -   z = 1\n    .. math:: 2x - 2y + 4z = -2\n    .. math:: 2x -   y + 2z = 0\n\n    * Augmented Matrix Form, `system` given below:\n\n    ::\n\n              [3   2  -1  1]\n     system = [2  -2   4 -2]\n              [2  -1   2  0]\n\n    * List Of Equations Form\n\n    `system  =  [3x + 2y - z - 1, 2x - 2y + 4z + 2, 2x - y + 2z]`\n\n    * Input A & b Matrix Form (from Ax = b) are given as below:\n\n    ::\n\n         [3   2  -1 ]         [  1 ]\n     A = [2  -2   4 ]    b =  [ -2 ]\n         [2  -1   2 ]         [  0 ]\n\n    `system = (A, b)`\n\n    Symbols can always be passed but are actually only needed\n    when 1) a system of equations is being passed and 2) the\n    system is passed as an underdetermined matrix and one wants\n    to control the name of the free variables in the result.\n    An error is raised if no symbols are used for case 1, but if\n    no symbols are provided for case 2, internally generated symbols\n    will be provided. When providing symbols for case 2, there should\n    be at least as many symbols are there are columns in matrix A.\n\n    The algorithm used here is Gauss-Jordan elimination, which\n    results, after elimination, in a row echelon form matrix.\n\n    Returns\n    =======\n\n    A FiniteSet containing an ordered tuple of values for the\n    unknowns for which the `system` has a solution. (Wrapping\n    the tuple in FiniteSet is used to maintain a consistent\n    output format throughout solveset.)\n\n    Returns EmptySet(), if the linear system is inconsistent.\n\n    Raises\n    ======\n\n    ValueError\n        The input is not valid.\n        The symbols are not given.\n\n    Examples\n    ========\n\n    >>> from sympy import Matrix, S, linsolve, symbols\n    >>> x, y, z = symbols(\"x, y, z\")\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 10]])\n    >>> b = Matrix([3, 6, 9])\n    >>> A\n    Matrix([\n    [1, 2,  3],\n    [4, 5,  6],\n    [7, 8, 10]])\n    >>> b\n    Matrix([\n    [3],\n    [6],\n    [9]])\n    >>> linsolve((A, b), [x, y, z])\n    {(-1, 2, 0)}\n\n    * Parametric Solution: In case the system is underdetermined, the\n      function will return a parametric solution in terms of the given\n      symbols. Those that are free will be returned unchanged. e.g. in\n      the system below, `z` is returned as the solution for variable z;\n      it can take on any value.\n\n    >>> A = Matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])\n    >>> b = Matrix([3, 6, 9])\n    >>> linsolve((A, b), x, y, z)\n    {(z - 1, 2 - 2*z, z)}\n\n    If no symbols are given, internally generated symbols will be used.\n    The `tau0` in the 3rd position indicates (as before) that the 3rd\n    variable -- whatever it's named -- can take on any value:\n\n    >>> linsolve((A, b))\n    {(tau0 - 1, 2 - 2*tau0, tau0)}\n\n    * List of Equations as input\n\n    >>> Eqns = [3*x + 2*y - z - 1, 2*x - 2*y + 4*z + 2, - x + y/2 - z]\n    >>> linsolve(Eqns, x, y, z)\n    {(1, -2, -2)}\n\n    * Augmented Matrix as input\n\n    >>> aug = Matrix([[2, 1, 3, 1], [2, 6, 8, 3], [6, 8, 18, 5]])\n    >>> aug\n    Matrix([\n    [2, 1,  3, 1],\n    [2, 6,  8, 3],\n    [6, 8, 18, 5]])\n    >>> linsolve(aug, x, y, z)\n    {(3/10, 2/5, 0)}\n\n    * Solve for symbolic coefficients\n\n    >>> a, b, c, d, e, f = symbols('a, b, c, d, e, f')\n    >>> eqns = [a*x + b*y - c, d*x + e*y - f]\n    >>> linsolve(eqns, x, y)\n    {((-b*f + c*e)/(a*e - b*d), (a*f - c*d)/(a*e - b*d))}\n\n    * A degenerate system returns solution as set of given\n      symbols.\n\n    >>> system = Matrix(([0, 0, 0], [0, 0, 0], [0, 0, 0]))\n    >>> linsolve(system, x, y)\n    {(x, y)}\n\n    * For an empty system linsolve returns empty set\n\n    >>> linsolve([], x)\n    EmptySet()\n\n    * An error is raised if, after expansion, any nonlinearity\n      is detected:\n\n    >>> linsolve([x*(1/x - 1), (y - 1)**2 - y**2 + 1], x, y)\n    {(1, 1)}\n    >>> linsolve([x**2 - 1], x)\n    Traceback (most recent call last):\n    ...\n    ValueError:\n    The term x**2 is nonlinear in {x}\n    \"\"\"\n    if not system:\n        return S.EmptySet\n\n    # If second argument is an iterable\n    if symbols and hasattr(symbols[0], '__iter__'):\n        symbols = symbols[0]\n    sym_gen = isinstance(symbols, GeneratorType)\n\n    swap = {}\n    b = None  # if we don't get b the input was bad\n    syms_needed_msg = None\n\n    # unpack system\n\n    if hasattr(system, '__iter__'):\n\n        # 1). (A, b)\n        if len(system) == 2 and isinstance(system[0], Matrix):\n            A, b = system\n\n        # 2). (eq1, eq2, ...)\n        if not isinstance(system[0], Matrix):\n            if sym_gen or not symbols:\n                raise ValueError(filldedent('''\n                    When passing a system of equations, the explicit\n                    symbols for which a solution is being sought must\n                    be given as a sequence, too.\n                '''))\n            system = [\n                _mexpand(i.lhs - i.rhs if isinstance(i, Eq) else i,\n                recursive=True) for i in system]\n            system, symbols, swap = recast_to_symbols(system, symbols)\n            A, b = linear_eq_to_matrix(system, symbols)\n            syms_needed_msg = 'free symbols in the equations provided'\n\n    elif isinstance(system, Matrix) and not (\n            symbols and not isinstance(symbols, GeneratorType) and\n            isinstance(symbols[0], Matrix)):\n        # 3). A augmented with b\n        A, b = system[:, :-1], system[:, -1:]\n\n    if b is None:\n        raise ValueError(\"Invalid arguments\")\n\n    syms_needed_msg  = syms_needed_msg or 'columns of A'\n\n    if sym_gen:\n        symbols = [next(symbols) for i in range(A.cols)]\n        if any(set(symbols) & (A.free_symbols | b.free_symbols)):\n            raise ValueError(filldedent('''\n                At least one of the symbols provided\n                already appears in the system to be solved.\n                One way to avoid this is to use Dummy symbols in\n                the generator, e.g. numbered_symbols('%s', cls=Dummy)\n            ''' % symbols[0].name.rstrip('1234567890')))\n\n    try:\n        solution, params, free_syms = A.gauss_jordan_solve(b, freevar=True)\n    except ValueError:\n        # No solution\n        return S.EmptySet\n\n    # Replace free parameters with free symbols\n    if params:\n        if not symbols:\n            symbols = [_ for _ in params]\n            # re-use the parameters but put them in order\n            # params       [x, y, z]\n            # free_symbols [2, 0, 4]\n            # idx          [1, 0, 2]\n            idx = list(zip(*sorted(zip(free_syms, range(len(free_syms))))))[1]\n            # simultaneous replacements {y: x, x: y, z: z}\n            replace_dict = dict(zip(symbols, [symbols[i] for i in idx]))\n        elif len(symbols) >= A.cols:\n            replace_dict = {v: symbols[free_syms[k]] for k, v in enumerate(params)}\n        else:\n            raise IndexError(filldedent('''\n                the number of symbols passed should have a length\n                equal to the number of %s.\n                ''' % syms_needed_msg))\n        solution = [sol.xreplace(replace_dict) for sol in solution]\n\n    solution = [simplify(sol).xreplace(swap) for sol in solution]\n    return FiniteSet(tuple(solution))"
        }
      ]
    },
    {
      "pr_number": 19091,
      "pr_title": "Fixed tensor contraction problem",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\" (see\r\nhttps://tinyurl.com/auto-closing for more information). Also, please\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #18465 \r\n\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nTensor contractions using `replace_with_arrays` would prematurely contract the data arrays before applying the metric\u2014yielding incorrect results. The solution implemented here is a rather trivial one in that it simply separates out an existing function for doing matrix multiplication with an existing data array with the metric and utilizes it in `Tensor._extract_data` to apply the metric before dummy indices are contracted.\r\n\r\n#### Other comments\r\n\r\nThis is my first pull request with Sympy. Let me know if I should make any adjustments to my code.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* tensor\r\n  * Fixed a bug where tensor contractions in calls to `replace_with_arrays` would fail to apply the metric.\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 18465,
      "issue_title": "Tensor contractions are wrong",
      "issue_body": "This is essentially a generalization of #17328.\r\n\r\nThe problem in the current implementation is that contractions are handled before applications of the metric, which leads to incorrect results such as in #17328.\r\n\r\nIn `tensor/tensor.py`:\r\n```python\r\nclass Tensor(TensExpr):\r\n# ...\r\n    def _extract_data(self, replacement_dict):\r\n    # ...\r\n        if len(dum1) > 0:\r\n            indices2 = other.get_indices()\r\n            repl = {}\r\n            for p1, p2 in dum1:\r\n                repl[indices2[p2]] = -indices2[p1]\r\n            other = other.xreplace(repl).doit()\r\n            array = _TensorDataLazyEvaluator.data_contract_dum([array], dum1, len(indices2))\r\n\r\n        free_ind1 = self.get_free_indices()\r\n        free_ind2 = other.get_free_indices()\r\n\r\n        return self._match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict)\r\n```\r\nAnd thus, the issue is that `_TensorDataLazyEvaluator.data_contract_dum` is being called prior to `self._match_indices_with_other_tensor` (where the metric is applied).\r\n\r\nThe reason that this ordering matters is because tensor contraction is itself the abstraction of applying the metric to the tensors that represent psuedo-riemannian manifolds. In essence, it means that we must have it that ![equation](https://latex.codecogs.com/svg.latex?T^\\mu_\\mu=g_{\\mu\\nu}T^{\\mu\\nu}); however, this isn't the case here.\r\n\r\nI've tried tampering with the code above, but by the way tensors have been designed, this bug is essentially unavoidable. As a consequence, the tensor module needs to be refactored in order to get accurate results. (Also, I couldn't help but notice that the last argument to `_TensorDataLazyEvaluator.data_contract_dum` isn't used).\r\n\r\n@drybalka had mentioned that he had this sort of refactoring in the works, but based on his fork, progress seems to be slow. I think discussions should be in order for reorganizing how tensors actually represent their components in this module.",
      "issue_closed_at": "2020-04-20T15:15:50Z",
      "base_commit": "64d28fe0534f6993695d11244ea740f783958dc8",
      "changes": [
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "recursor",
          "class_name": "TensExpr",
          "code": "def recursor(expr, pos):\n            if isinstance(expr, TensorIndex):\n                yield (expr, pos)\n            elif isinstance(expr, (Tuple, TensExpr)):\n                for p, arg in enumerate(expr.args):\n                    for i in recursor(arg, pos+(p,)):\n                        yield i"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "_match_indices_with_other_tensor",
          "class_name": "TensExpr",
          "code": "def _match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict):\n        from .array import tensorcontraction, tensorproduct, permutedims\n\n        index_types1 = [i.tensor_index_type for i in free_ind1]\n\n        # Check if variance of indices needs to be fixed:\n        pos2up = []\n        pos2down = []\n        free2remaining = free_ind2[:]\n        for pos1, index1 in enumerate(free_ind1):\n            if index1 in free2remaining:\n                pos2 = free2remaining.index(index1)\n                free2remaining[pos2] = None\n                continue\n            if -index1 in free2remaining:\n                pos2 = free2remaining.index(-index1)\n                free2remaining[pos2] = None\n                free_ind2[pos2] = index1\n                if index1.is_up:\n                    pos2up.append(pos2)\n                else:\n                    pos2down.append(pos2)\n            else:\n                index2 = free2remaining[pos1]\n                if index2 is None:\n                    raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n                free2remaining[pos1] = None\n                free_ind2[pos1] = index1\n                if index1.is_up ^ index2.is_up:\n                    if index1.is_up:\n                        pos2up.append(pos1)\n                    else:\n                        pos2down.append(pos1)\n\n        if len(set(free_ind1) & set(free_ind2)) < len(free_ind1):\n            raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n\n        # TODO: add possibility of metric after (spinors)\n        def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)\n\n        # Raise indices:\n        for pos in pos2up:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            metric_inverse = _TensorDataLazyEvaluator.inverse_matrix(metric)\n            array = contract_and_permute(metric_inverse, array, pos)\n        # Lower indices:\n        for pos in pos2down:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            array = contract_and_permute(metric, array, pos)\n\n        if free_ind1:\n            permutation = TensExpr._get_indices_permutation(free_ind2, free_ind1)\n            array = permutedims(array, permutation)\n\n        if hasattr(array, \"rank\") and array.rank() == 0:\n            array = array[()]\n\n        return free_ind2, array"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "contract_and_permute",
          "class_name": "TensExpr",
          "code": "def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)"
        },
        {
          "file": "sympy/tensor/tensor.py",
          "type": "function",
          "name": "_extract_data",
          "class_name": "TensorElement",
          "code": "def _extract_data(self, replacement_dict):\n        ret_indices, array = self.expr._extract_data(replacement_dict)\n        index_map = self.index_map\n        slice_tuple = tuple(index_map.get(i, slice(None)) for i in ret_indices)\n        ret_indices = [i for i in ret_indices if i not in index_map]\n        array = array.__getitem__(slice_tuple)\n        return ret_indices, array"
        }
      ]
    },
    {
      "pr_number": 15286,
      "pr_title": "Implemented finding equation of Ellipse using slope as parameter and another method for calculation of circumference of ellipse - continued",
      "pr_body": "Implemented finding equation of Ellipse using slope as parameter.\r\n\r\nWhen the rotated ellipse is supported, this calculation can still be used but the `_slope` keyword will not be needed: the slope will be an attribute of the rotated ellipse.\r\n\r\nFixes #2815\r\nFixes #6952\r\nCloses #15053 \r\n\r\nThis PR uses the approach to finding equation of ellipse using slope, length of semi minor axis and length of semi major axis as inputs given [here](https://math.stackexchange.com/questions/108270/what-is-the-equation-of-an-ellipse-that-is-not-aligned-with-the-axis/646971#646971)\r\nThis could be an added functionality to the equation finding method in class `Ellipse`.\r\nThanks to @smichr  for providing the approach.\r\n\r\nNote : This is a continuation of #15053 \r\nPlease take a look at this PR and suggest changes. I will be glad to implement them.\r\nThanks.\r\n\r\n#### Release Notes\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* geometry\r\n   * Implemented private method for finding equation of Ellipse using `_slope` as parameter\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 6952,
      "issue_title": "recognize elliptical integrals",
      "issue_body": "```\nThis requires about 2 minutes\n\n>>> Ellipse((0,0),3,1).circumference.n()\n13.3648932205553\n\n\nThis is nearly instantaneous\n\n>>> def EllipseCircumference(a, b):\n...    \"\"\"\n...    Compute the circumference of an ellipse with semi-axes a and b.\n...    Require a >= 0 and b >= 0.  Relative accuracy is about 0.5^53.\n...    \"\"\"\n...    import math\n...    x, y = max(a, b), min(a, b)\n...    digits = 53; tol = math.sqrt(math.pow(0.5, digits))\n...    if digits * y < tol * x: return 4 * x\n...    s = 0; m = 1\n...    while x - y > tol * y:\n...       x, y = 0.5 * (x + y), math.sqrt(x * y)\n...       m *= 2; s += m * math.pow(x - y, 2)\n...    return math.pi * (math.pow(a + b, 2) - s) / (x + y)\n...\n>>> EllipseCircumference(3,1)\n13.364893220555258\n>>>\n\nPerhaps recognition of such integrals by integrate/Integral.evalf would be a good idea.\n```\n\nOriginal issue for #6952: http://code.google.com/p/sympy/issues/detail?id=3853\nOriginal author: https://code.google.com/u/117933771799683895267/\n",
      "issue_closed_at": "2018-10-01T21:27:53Z",
      "base_commit": "5997e30a33f92e6b4b4d351e835feb7379a0e31d",
      "changes": [
        {
          "file": "sympy/geometry/ellipse.py",
          "type": "function",
          "name": "encloses_point",
          "class_name": "Ellipse",
          "code": "def encloses_point(self, p):\n        \"\"\"\n        Return True if p is enclosed by (is inside of) self.\n\n        Notes\n        -----\n        Being on the border of self is considered False.\n\n        Parameters\n        ==========\n\n        p : Point\n\n        Returns\n        =======\n\n        encloses_point : True, False or None\n\n        See Also\n        ========\n\n        sympy.geometry.point.Point\n\n        Examples\n        ========\n\n        >>> from sympy import Ellipse, S\n        >>> from sympy.abc import t\n        >>> e = Ellipse((0, 0), 3, 2)\n        >>> e.encloses_point((0, 0))\n        True\n        >>> e.encloses_point(e.arbitrary_point(t).subs(t, S.Half))\n        False\n        >>> e.encloses_point((4, 0))\n        False\n\n        \"\"\"\n        p = Point(p, dim=2)\n        if p in self:\n            return False\n\n        if len(self.foci) == 2:\n            # if the combined distance from the foci to p (h1 + h2) is less\n            # than the combined distance from the foci to the minor axis\n            # (which is the same as the major axis length) then p is inside\n            # the ellipse\n            h1, h2 = [f.distance(p) for f in self.foci]\n            test = 2*self.major - (h1 + h2)\n        else:\n            test = self.radius - self.center.distance(p)\n\n        return fuzzy_bool(test.is_positive)"
        },
        {
          "file": "sympy/geometry/ellipse.py",
          "type": "function",
          "name": "equation",
          "class_name": "Circle",
          "code": "def equation(self, x='x', y='y'):\n        \"\"\"The equation of the circle.\n\n        Parameters\n        ==========\n\n        x : str or Symbol, optional\n            Default value is 'x'.\n        y : str or Symbol, optional\n            Default value is 'y'.\n\n        Returns\n        =======\n\n        equation : SymPy expression\n\n        Examples\n        ========\n\n        >>> from sympy import Point, Circle\n        >>> c1 = Circle(Point(0, 0), 5)\n        >>> c1.equation()\n        x**2 + y**2 - 25\n\n        \"\"\"\n        x = _symbol(x, real=True)\n        y = _symbol(y, real=True)\n        t1 = (x - self.center.x)**2\n        t2 = (y - self.center.y)**2\n        return t1 + t2 - self.major**2"
        },
        {
          "file": "sympy/geometry/ellipse.py",
          "type": "function",
          "name": "equation",
          "class_name": "Circle",
          "code": "def equation(self, x='x', y='y'):\n        \"\"\"The equation of the circle.\n\n        Parameters\n        ==========\n\n        x : str or Symbol, optional\n            Default value is 'x'.\n        y : str or Symbol, optional\n            Default value is 'y'.\n\n        Returns\n        =======\n\n        equation : SymPy expression\n\n        Examples\n        ========\n\n        >>> from sympy import Point, Circle\n        >>> c1 = Circle(Point(0, 0), 5)\n        >>> c1.equation()\n        x**2 + y**2 - 25\n\n        \"\"\"\n        x = _symbol(x, real=True)\n        y = _symbol(y, real=True)\n        t1 = (x - self.center.x)**2\n        t2 = (y - self.center.y)**2\n        return t1 + t2 - self.major**2"
        }
      ]
    },
    {
      "pr_number": 16668,
      "pr_title": "speed up Line/Segment intersections",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests . Please also\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\n\r\ncloses #16628\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nIntersections will return more quickly for Line/Segment interactions in 2D because a redundant check that the point is in the line has been eliminated.\r\n\r\nPoint containment in a 2D Segment now uses a quick check before using the triangle inequality.\r\n\r\n#### Other comments\r\n\r\nSome tests which were quite fast already were moved out of the slow test.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n- geometry\r\n  * improved response time of 2D Line/Segement interactions and Segment containment of Point\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 16628,
      "issue_title": "Geometry: Line-Segment-Intersection can run into memory problems",
      "issue_body": "Hi,\r\n\r\nwhen determining a line-segment-intersection, depending on the geometry, sympy can take up som much memory that the script will crash.\r\n\r\nI'm on Windows, using CPython 3.7.3 on sympy commit b97cbe99, although the issue exists in version 1.3 too.\r\n\r\nExample code:\r\n<pre>\r\n#!Python -u\r\n\r\nimport sys, os, time, math\r\n\r\nsys.stderr.write(str(os.getpid())+time.strftime(': starting @ %d%b%y, %H:%M\\n'))\r\n\t\r\nfrom sympy.geometry import Point, Point2D, Line, Segment, intersection\r\nfrom sympy import *\r\nimport sympy\r\n\r\nsys.stderr.write('sympy version: '+sympy.__version__+'\\n')\r\n\r\np0=Point2D(249/5, 497999/10000)\r\np1x=3*(-19659028262*sqrt(405639795226) + 676896692394731 + 6704069269*sqrt(630547164901) + 33200*sqrt(255775022850776494562626))/(2000*(sqrt(255775022850776494562626) + 995999*sqrt(405639795226) + 995999*sqrt(630547164901) + 811280586451))\r\np1y=(-498000*sqrt(255775022850776494562626) - 995999*sqrt(630547164901) + 90004251917891999 + 496005510002*sqrt(405639795226))/(10000*(sqrt(255775022850776494562626) + 995999*sqrt(405639795226) + 995999*sqrt(630547164901) + 811280586451))\r\np1=Point2D(p1x, p1y)\r\n\r\n\r\nsys.stderr.write('p1: '+str(p1)+'\\n')\r\n\r\np2=Point2D(497/10, -497/10)\r\np3=Point2D(-497/10, -497/10)\r\n\r\nl=Line(p0,p1)\r\ns=Segment(p2,p3)\r\nints=intersection(l,s)\r\n\r\n\r\nsys.stderr.write(str(os.getpid())+time.strftime(': done @ %d%b%y, %H:%M\\n'))\r\n</pre>",
      "issue_closed_at": "2019-04-17T22:08:24Z",
      "base_commit": "96925f549e10ee2d95e592d1fe84cbdd39cec3f4",
      "changes": [
        {
          "file": "sympy/geometry/line.py",
          "type": "function",
          "name": "intersect_parallel_segments",
          "class_name": "LinearEntity",
          "code": "def intersect_parallel_segments(seg1, seg2):\n            if seg1.contains(seg2):\n                return [seg2]\n            if seg2.contains(seg1):\n                return [seg1]\n\n            # direct the segments so they're oriented the same way\n            if seg1.direction.dot(seg2.direction) < 0:\n                seg2 = Segment(seg2.p2, seg2.p1)\n            # order the segments so seg1 is \"behind\" seg2\n            if seg1._span_test(seg2.p1) < 0:\n                seg1, seg2 = seg2, seg1\n            if seg2._span_test(seg1.p2) < 0:\n                return []\n            return [Segment(seg2.p1, seg1.p2)]"
        },
        {
          "file": "sympy/geometry/line.py",
          "type": "function",
          "name": "contains",
          "class_name": "Segment",
          "code": "def contains(self, other):\n        \"\"\"\n        Is the other GeometryEntity contained within this Segment?\n\n        Examples\n        ========\n\n        >>> from sympy import Point, Segment\n        >>> p1, p2 = Point(0, 1), Point(3, 4)\n        >>> s = Segment(p1, p2)\n        >>> s2 = Segment(p2, p1)\n        >>> s.contains(s2)\n        True\n        >>> from sympy import Point3D, Segment3D\n        >>> p1, p2 = Point3D(0, 1, 1), Point3D(3, 4, 5)\n        >>> s = Segment3D(p1, p2)\n        >>> s2 = Segment3D(p2, p1)\n        >>> s.contains(s2)\n        True\n        >>> s.contains((p1 + p2) / 2)\n        True\n        \"\"\"\n\n        if not isinstance(other, GeometryEntity):\n            other = Point(other, dim=self.ambient_dimension)\n        if isinstance(other, Point):\n            if Point.is_collinear(other, self.p1, self.p2):\n                d1, d2 = other - self.p1, other - self.p2\n                d = self.p2 - self.p1\n                # without the call to simplify, sympy cannot tell that an expression\n                # like (a+b)*(a/2+b/2) is always non-negative.  If it cannot be\n                # determined, raise an Undecidable error\n                try:\n                    # the triangle inequality says that |d1|+|d2| >= |d| and is strict\n                    # only if other lies in the line segment\n                    return bool(simplify(Eq(abs(d1) + abs(d2) - abs(d), 0)))\n                except TypeError:\n                    raise Undecidable(\"Cannot determine if {} is in {}\".format(other, self))\n        if isinstance(other, Segment):\n            return other.p1 in self and other.p2 in self\n\n        return False"
        },
        {
          "file": "sympy/geometry/polygon.py",
          "type": "function",
          "name": "bisectors",
          "class_name": "Triangle",
          "code": "def bisectors(self):\n        \"\"\"The angle bisectors of the triangle.\n\n        An angle bisector of a triangle is a straight line through a vertex\n        which cuts the corresponding angle in half.\n\n        Returns\n        =======\n\n        bisectors : dict\n            Each key is a vertex (Point) and each value is the corresponding\n            bisector (Segment).\n\n        See Also\n        ========\n\n        sympy.geometry.point.Point, sympy.geometry.line.Segment\n\n        Examples\n        ========\n\n        >>> from sympy.geometry import Point, Triangle, Segment\n        >>> p1, p2, p3 = Point(0, 0), Point(1, 0), Point(0, 1)\n        >>> t = Triangle(p1, p2, p3)\n        >>> from sympy import sqrt\n        >>> t.bisectors()[p2] == Segment(Point(1, 0), Point(0, sqrt(2) - 1))\n        True\n\n        \"\"\"\n        # use lines containing sides so containment check during\n        # intersection calculation can be avoided\n        # This would reduce the processing time for calculating the bisectors\n        s = [Line(l) for l in self.sides]\n        v = self.vertices\n        c = self.incenter\n        l1 = Segment(v[0], Line(v[0], c).intersection(s[1])[0])\n        l2 = Segment(v[1], Line(v[1], c).intersection(s[2])[0])\n        l3 = Segment(v[2], Line(v[2], c).intersection(s[0])[0])\n        return {v[0]: l1, v[1]: l2, v[2]: l3}"
        }
      ]
    }
  ]
}