{
  "Selected_candidate": {
    "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]])"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant as it deals with a specific mathematical operation (conversion of quaternions to rotation matrices), similarly to how the current bug deals with the calculation of velocities between reference frames in mechanics. Both bugs indicate that certain calculations are not yielding the expected results, which suggests a potential misunderstanding or misimplementation of mathematical principles. The structural similarity in terms of evaluating mathematical expressions within the SymPy environment shows a commonality that can help illuminate how to address the unresolved handling of velocities in the CURRENT bug.",
  "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"
}