{
  "instance_id": "sympy__sympy-22005",
  "repo": "sympy/sympy",
  "created_at": "2021-09-02T13:05:27Z",
  "problem_statement": "detection of infinite solution request\n```python\r\n>>> solve_poly_system((x - 1,), x, y)\r\nTraceback (most recent call last):\r\n...\r\nNotImplementedError:\r\nonly zero-dimensional systems supported (finite number of solutions)\r\n>>> solve_poly_system((y - 1,), x, y)  <--- this is not handled correctly\r\n[(1,)]\r\n```\r\n```diff\r\ndiff --git a/sympy/solvers/polysys.py b/sympy/solvers/polysys.py\r\nindex b9809fd4e9..674322d4eb 100644\r\n--- a/sympy/solvers/polysys.py\r\n+++ b/sympy/solvers/polysys.py\r\n@@ -240,7 +240,7 @@ def _solve_reduced_system(system, gens, entry=False):\r\n \r\n         univariate = list(filter(_is_univariate, basis))\r\n \r\n-        if len(univariate) == 1:\r\n+        if len(univariate) == 1 and len(gens) == 1:\r\n             f = univariate.pop()\r\n         else:\r\n             raise NotImplementedError(filldedent('''\r\ndiff --git a/sympy/solvers/tests/test_polysys.py b/sympy/solvers/tests/test_polysys.py\r\nindex 58419f8762..9e674a6fe6 100644\r\n--- a/sympy/solvers/tests/test_polysys.py\r\n+++ b/sympy/solvers/tests/test_polysys.py\r\n@@ -48,6 +48,10 @@ def test_solve_poly_system():\r\n     raises(NotImplementedError, lambda: solve_poly_system(\r\n         [z, -2*x*y**2 + x + y**2*z, y**2*(-z - 4) + 2]))\r\n     raises(PolynomialError, lambda: solve_poly_system([1/x], x))\r\n+    raises(NotImplementedError, lambda: solve_poly_system(\r\n+        Poly(x - 1, x, y), (x, y)))\r\n+    raises(NotImplementedError, lambda: solve_poly_system(\r\n+        Poly(y - 1, x, y), (x, y)))\r\n \r\n \r\n def test_solve_biquadratic():\r\n```\n",
  "patch": "diff --git a/sympy/solvers/polysys.py b/sympy/solvers/polysys.py\n--- a/sympy/solvers/polysys.py\n+++ b/sympy/solvers/polysys.py\n@@ -240,6 +240,12 @@ def _solve_reduced_system(system, gens, entry=False):\n \n         univariate = list(filter(_is_univariate, basis))\n \n+        if len(basis) < len(gens):\n+            raise NotImplementedError(filldedent('''\n+                only zero-dimensional systems supported\n+                (finite number of solutions)\n+                '''))\n+\n         if len(univariate) == 1:\n             f = univariate.pop()\n         else:\n",
  "similar_bug_items": [
    {
      "pr_number": 13437,
      "pr_title": "Fix bell(n) for n=oo, and negative/non-integer 'n'",
      "pr_body": "This PR uses commits of #9198 (maintaining ownership to original author). And tries to address the requested changes on PR #9198 .\r\nFixes #9184",
      "issue_id": 9184,
      "issue_title": "bell(n).limit(n, oo) should be oo rather than bell(oo)",
      "issue_body": "`bell(n).limit(n,oo)` should take the value infinity, but the current output is `bell(oo)`. As the Bell numbers represent the number of partitions of a set, it seems natural that `bell(oo)` should be able to be evaluated rather than be returned unevaluated. This issue is also in line with the recent fixes to the corresponding limit for the Fibonacci numbers and Lucas numbers.\n\n```\nfrom sympy import *\nn = symbols('n')\nbell(n).limit(n,oo)\n\nOutput:\nbell(oo)\n```\n\nI'm new to Sympy, so I'd appreciate the opportunity to fix this bug myself if that's alright.\n",
      "issue_closed_at": "2017-10-13T04:35:13Z",
      "base_commit": "674afc619d7f5c519b6a5393a8b0532a131e57e0",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/numbers.py",
          "type": "function",
          "name": "_bell_incomplete_poly",
          "class_name": "bell",
          "code": "def _bell_incomplete_poly(n, k, symbols):\n        r\"\"\"\n        The second kind of Bell polynomials (incomplete Bell polynomials).\n\n        Calculated by recurrence formula:\n\n        .. math:: B_{n,k}(x_1, x_2, \\dotsc, x_{n-k+1}) =\n                \\sum_{m=1}^{n-k+1}\n                \\x_m \\binom{n-1}{m-1} B_{n-m,k-1}(x_1, x_2, \\dotsc, x_{n-m-k})\n\n        where\n            B_{0,0} = 1;\n            B_{n,0} = 0; for n>=1\n            B_{0,k} = 0; for k>=1\n\n        \"\"\"\n        if (n == 0) and (k == 0):\n            return S.One\n        elif (n == 0) or (k == 0):\n            return S.Zero\n        s = S.Zero\n        a = S.One\n        for m in range(1, n - k + 2):\n            s += a * bell._bell_incomplete_poly(\n                n - m, k - 1, symbols) * symbols[m - 1]\n            a = a * (n - m) / m\n        return expand_mul(s)"
        }
      ]
    },
    {
      "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": 13103,
      "pr_title": "fixed cos.rewrite(sqrt) sometimes not fully rewriting",
      "pr_body": "See #13066\r\n`functions.elementary.trigonometric.cos._eval_rewrite_as_sqrt` would fail to fully rewrite `cos(pi/51)` on python 3.6 only. This turned out to be an issue with the `_fermatCoords` function which is supposed to factor 51 into fermat primes. Trigonometric expressions with these fermat primes as denominators of pi can be rewritten using squareroots.\r\n\r\nThe function uses `divmod` repeatedly to get the factors. `n` was set to the whole number result of the `divmod`. This is where the bug was, since that would mean that if `n` was not divisible by that particular factor, then `n` would still be changed to the whole number result, which usually ended up being 0, in which case the function would miss the other factors and return `False`. Or in rare cases it would have led to a false factorization.\r\n\r\nThe reason this bug only revealed itself in 3.6 is because the ordering of dicts was changed. Previously for n=51 it so happened that python ordered the dict with 17 first, then 3, which are both divisible.\r\n\r\n<!-- Please give this pull request a descriptive title. Pull requests with descriptive titles are more likely to receive reviews. Describe what you changed! A title that only references an issue number is not descriptive. -->\r\n\r\n<!-- If this pull request fixes an issue please indicate which issue by typing \"Fixes #NNNN\" below. -->\r\nFixes #13066",
      "issue_id": 13066,
      "issue_title": "cos(pi/51).rewrite(sqrt) does not fully rewrite the expression on python 3.6 (test failure)",
      "issue_body": "It currently returns `cos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2`, even though both `cos(6*pi/17)` and `sin(6*pi/17)` can be rewritten using `sqrt`s. The slow test `test_sincos_rewrite_sqrt` in test_trigonometric.py currently fails because it tests that `cos(pi/51).rewrite(sqrt)` returns the full expression in terms of `sqrt`s. I tested on python 3.4 64 bit and python 3.6 64 bit, and it only happens on 3.6, which is odd. I tried a fresh installation for 3.6 to be sure it wasn't anything I changed and it still happened.\r\n```\r\n>>> cos(pi/51).rewrite(sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n>>> cos(pi/51).rewrite(cos, sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n>>> cos(pi/51).rewrite([cos, sin], sqrt)\r\ncos(6*pi/17)/2 + sqrt(3)*sin(6*pi/17)/2\r\n```",
      "issue_closed_at": "2017-08-10T09:16:46Z",
      "base_commit": "8ff744d843f9dc78c560e61e3edc01f878d1960b",
      "changes": [
        {
          "file": "sympy/functions/elementary/trigonometric.py",
          "type": "function",
          "name": "_fermatCoords",
          "class_name": "cos",
          "code": "def _fermatCoords(n):\n            # if n can be factored in terms of Fermat primes with\n            # multiplicity of each being 1, return those primes, else\n            # False\n            from sympy import chebyshevt\n            primes = []\n            for p_i in cst_table_some:\n                n, r = divmod(n, p_i)\n                if not r:\n                    primes.append(p_i)\n                    if n == 1:\n                        return tuple(primes)\n            return False"
        }
      ]
    },
    {
      "pr_number": 17375,
      "pr_title": "polygamma improvements (evaluation, real and positivity determination)",
      "pr_body": "Closes #17350. Fixes #12569.\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\n- Add `_eval_evalf` to prevent evaluation when the first argument is not a nonnegative integer.\r\n- Remove `_eval_is_real` since the logic was wrong.\r\n- Update `_eval_is_positive` and `_eval_is_negative` to avoid doing comparisons with `>`.\r\n\r\n#### Release Notes\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 17350,
      "issue_title": "Should polygamma(a, b) raise NotImplementedError for noninteger (numeric) a?",
      "issue_body": "mpmath's implementation of polygamma is valid only for integer first arguments. SymPy's implementation does not care, but this leads to many issues if you actually try to do things with such objects:\r\n\r\n```\r\n>>> (I*polygamma(I, pi)).as_real_imag()\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'zero'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'finite'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 262, in getit\r\n    return self._assumptions[fact]\r\nKeyError: 'extended_positive'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1308, in evalf\r\n    rf = evalf_table[x.func]\r\nKeyError: polygamma\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/eward/se2/sympy/core/mul.py\", line 798, in as_real_imag\r\n    if i.is_zero:\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 321, in _ask\r\n    _ask(pk, obj)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 864, in _eval_is_negative\r\n    finite = self.is_finite\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 321, in _ask\r\n    _ask(pk, obj)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 857, in _eval_is_positive\r\n    extended_positive = self.is_extended_positive\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 266, in getit\r\n    return _ask(fact, self)\r\n  File \"/home/eward/se2/sympy/core/assumptions.py\", line 309, in _ask\r\n    a = evaluate(obj)\r\n  File \"/home/eward/se2/sympy/core/expr.py\", line 882, in _eval_is_extended_positive\r\n    n2 = self._eval_evalf(2)\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 565, in _eval_evalf\r\n    args = [arg._to_mpmath(prec + 5) for arg in self.args]\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 565, in <listcomp>\r\n    args = [arg._to_mpmath(prec + 5) for arg in self.args]\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1489, in _to_mpmath\r\n    re, im, _, _ = evalf(self, prec, {})\r\n  File \"/home/eward/se2/sympy/core/evalf.py\", line 1314, in evalf\r\n    xe = x._eval_evalf(prec)\r\n  File \"/home/eward/se2/sympy/core/function.py\", line 590, in _eval_evalf\r\n    v = func(*args)\r\n  File \"/usr/local/lib/python3.6/dist-packages/mpmath/ctx_mp.py\", line 265, in psi\r\n    m = int(m)\r\nTypeError: int() argument must be a string, a bytes-like object or a number, not 'mpc'\r\n>>> (tanh(polygamma(I, 1))).rewrite(exp)\r\n...\r\n    m = int(m)\r\nTypeError: int() argument must be a string, a bytes-like object or a number, not 'mpc'\r\n>>> (I / polygamma(I, 4)).rewrite(exp)\r\n...\r\n    raise TypeError(\"Invalid comparison of complex %s\" % me)\r\nTypeError: Invalid comparison of complex I\r\n```",
      "issue_closed_at": "2019-08-23T06:49:24Z",
      "base_commit": "fc6f766ab588fecbb69ad85eb01ca28b44715e5c",
      "changes": [
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "class",
          "name": "polygamma",
          "code": "class polygamma(Function):\n    r\"\"\"\n    The function ``polygamma(n, z)`` returns ``log(gamma(z)).diff(n + 1)``.\n\n    It is a meromorphic function on `\\mathbb{C}` and defined as the (n+1)-th\n    derivative of the logarithm of the gamma function:\n\n    .. math::\n        \\psi^{(n)} (z) := \\frac{\\mathrm{d}^{n+1}}{\\mathrm{d} z^{n+1}} \\log\\Gamma(z).\n\n    Examples\n    ========\n\n    Several special values are known:\n\n    >>> from sympy import S, polygamma\n    >>> polygamma(0, 1)\n    -EulerGamma\n    >>> polygamma(0, 1/S(2))\n    -2*log(2) - EulerGamma\n    >>> polygamma(0, 1/S(3))\n    -log(3) - sqrt(3)*pi/6 - EulerGamma - log(sqrt(3))\n    >>> polygamma(0, 1/S(4))\n    -pi/2 - log(4) - log(2) - EulerGamma\n    >>> polygamma(0, 2)\n    1 - EulerGamma\n    >>> polygamma(0, 23)\n    19093197/5173168 - EulerGamma\n\n    >>> from sympy import oo, I\n    >>> polygamma(0, oo)\n    oo\n    >>> polygamma(0, -oo)\n    oo\n    >>> polygamma(0, I*oo)\n    oo\n    >>> polygamma(0, -I*oo)\n    oo\n\n    Differentiation with respect to x is supported:\n\n    >>> from sympy import Symbol, diff\n    >>> x = Symbol(\"x\")\n    >>> diff(polygamma(0, x), x)\n    polygamma(1, x)\n    >>> diff(polygamma(0, x), x, 2)\n    polygamma(2, x)\n    >>> diff(polygamma(0, x), x, 3)\n    polygamma(3, x)\n    >>> diff(polygamma(1, x), x)\n    polygamma(2, x)\n    >>> diff(polygamma(1, x), x, 2)\n    polygamma(3, x)\n    >>> diff(polygamma(2, x), x)\n    polygamma(3, x)\n    >>> diff(polygamma(2, x), x, 2)\n    polygamma(4, x)\n\n    >>> n = Symbol(\"n\")\n    >>> diff(polygamma(n, x), x)\n    polygamma(n + 1, x)\n    >>> diff(polygamma(n, x), x, 2)\n    polygamma(n + 2, x)\n\n    We can rewrite polygamma functions in terms of harmonic numbers:\n\n    >>> from sympy import harmonic\n    >>> polygamma(0, x).rewrite(harmonic)\n    harmonic(x - 1) - EulerGamma\n    >>> polygamma(2, x).rewrite(harmonic)\n    2*harmonic(x - 1, 3) - 2*zeta(3)\n    >>> ni = Symbol(\"n\", integer=True)\n    >>> polygamma(ni, x).rewrite(harmonic)\n    (-1)**(n + 1)*(-harmonic(x - 1, n + 1) + zeta(n + 1))*factorial(n)\n\n    See Also\n    ========\n\n    gamma: Gamma function.\n    lowergamma: Lower incomplete gamma function.\n    uppergamma: Upper incomplete gamma function.\n    loggamma: Log Gamma function.\n    digamma: Digamma function.\n    trigamma: Trigamma function.\n    sympy.functions.special.beta_functions.beta: Euler Beta function.\n\n    References\n    ==========\n\n    .. [1] https://en.wikipedia.org/wiki/Polygamma_function\n    .. [2] http://mathworld.wolfram.com/PolygammaFunction.html\n    .. [3] http://functions.wolfram.com/GammaBetaErf/PolyGamma/\n    .. [4] http://functions.wolfram.com/GammaBetaErf/PolyGamma2/\n    \"\"\"\n\n\n    def fdiff(self, argindex=2):\n        if argindex == 2:\n            n, z = self.args[:2]\n            return polygamma(n + 1, z)\n        else:\n            raise ArgumentIndexError(self, argindex)\n\n    def _eval_is_positive(self):\n        if self.args[1].is_positive and (self.args[0] > 0) == True:\n            return self.args[0].is_odd\n\n    def _eval_is_negative(self):\n        if self.args[1].is_positive and (self.args[0] > 0) == True:\n            return self.args[0].is_even\n\n    def _eval_is_real(self):\n        return self.args[0].is_real\n\n    def _eval_aseries(self, n, args0, x, logx):\n        from sympy import Order\n        if args0[1] != oo or not \\\n                (self.args[0].is_Integer and self.args[0].is_nonnegative):\n            return super(polygamma, self)._eval_aseries(n, args0, x, logx)\n        z = self.args[1]\n        N = self.args[0]\n\n        if N == 0:\n            # digamma function series\n            # Abramowitz & Stegun, p. 259, 6.3.18\n            r = log(z) - 1/(2*z)\n            o = None\n            if n < 2:\n                o = Order(1/z, x)\n            else:\n                m = ceiling((n + 1)//2)\n                l = [bernoulli(2*k) / (2*k*z**(2*k)) for k in range(1, m)]\n                r -= Add(*l)\n                o = Order(1/z**(2*m), x)\n            return r._eval_nseries(x, n, logx) + o\n        else:\n            # proper polygamma function\n            # Abramowitz & Stegun, p. 260, 6.4.10\n            # We return terms to order higher than O(x**n) on purpose\n            # -- otherwise we would not be able to return any terms for\n            #    quite a long time!\n            fac = gamma(N)\n            e0 = fac + N*fac/(2*z)\n            m = ceiling((n + 1)//2)\n            for k in range(1, m):\n                fac = fac*(2*k + N - 1)*(2*k + N - 2) / ((2*k)*(2*k - 1))\n                e0 += bernoulli(2*k)*fac/z**(2*k)\n            o = Order(1/z**(2*m), x)\n            if n == 0:\n                o = Order(1/z, x)\n            elif n == 1:\n                o = Order(1/z**2, x)\n            r = e0._eval_nseries(z, n, logx) + o\n            return (-1 * (-1/z)**N * r)._eval_nseries(x, n, logx)\n\n    @classmethod\n    def eval(cls, n, z):\n        n, z = map(sympify, (n, z))\n        from sympy import unpolarify\n\n        if n.is_integer:\n            if n.is_nonnegative:\n                nz = unpolarify(z)\n                if z != nz:\n                    return polygamma(n, nz)\n\n            if n == -1:\n                return loggamma(z)\n            else:\n                if z.is_Number:\n                    if z is S.NaN:\n                        return S.NaN\n                    elif z is S.Infinity:\n                        if n.is_Number:\n                            if n is S.Zero:\n                                return S.Infinity\n                            else:\n                                return S.Zero\n                    elif z.is_Integer:\n                        if z.is_nonpositive:\n                            return S.ComplexInfinity\n                        else:\n                            if n is S.Zero:\n                                return -S.EulerGamma + harmonic(z - 1, 1)\n                            elif n.is_odd:\n                                return (-1)**(n + 1)*factorial(n)*zeta(n + 1, z)\n\n        if n == 0:\n            if z is S.NaN:\n                return S.NaN\n            elif z.is_Rational:\n\n                p, q = z.as_numer_denom()\n\n                # only expand for small denominators to avoid creating long expressions\n                if q <= 5:\n                    return expand_func(polygamma(n, z, evaluate=False))\n\n            elif z in (S.Infinity, S.NegativeInfinity):\n                return S.Infinity\n            else:\n                t = z.extract_multiplicatively(S.ImaginaryUnit)\n                if t in (S.Infinity, S.NegativeInfinity):\n                    return S.Infinity\n\n        # TODO n == 1 also can do some rational z\n\n    def _eval_expand_func(self, **hints):\n        n, z = self.args\n\n        if n.is_Integer and n.is_nonnegative:\n            if z.is_Add:\n                coeff = z.args[0]\n                if coeff.is_Integer:\n                    e = -(n + 1)\n                    if coeff > 0:\n                        tail = Add(*[Pow(\n                            z - i, e) for i in range(1, int(coeff) + 1)])\n                    else:\n                        tail = -Add(*[Pow(\n                            z + i, e) for i in range(0, int(-coeff))])\n                    return polygamma(n, z - coeff) + (-1)**n*factorial(n)*tail\n\n            elif z.is_Mul:\n                coeff, z = z.as_two_terms()\n                if coeff.is_Integer and coeff.is_positive:\n                    tail = [ polygamma(n, z + Rational(\n                        i, coeff)) for i in range(0, int(coeff)) ]\n                    if n == 0:\n                        return Add(*tail)/coeff + log(coeff)\n                    else:\n                        return Add(*tail)/coeff**(n + 1)\n                z *= coeff\n\n        if n == 0 and z.is_Rational:\n            p, q = z.as_numer_denom()\n\n            # Reference:\n            #   Values of the polygamma functions at rational arguments, J. Choi, 2007\n            part_1 = -S.EulerGamma - pi * cot(p * pi / q) / 2 - log(q) + Add(\n                *[cos(2 * k * pi * p / q) * log(2 * sin(k * pi / q)) for k in range(1, q)])\n\n            if z > 0:\n                n = floor(z)\n                z0 = z - n\n                return part_1 + Add(*[1 / (z0 + k) for k in range(n)])\n            elif z < 0:\n                n = floor(1 - z)\n                z0 = z + n\n                return part_1 - Add(*[1 / (z0 - 1 - k) for k in range(n)])\n\n        return polygamma(n, z)\n\n    def _eval_rewrite_as_zeta(self, n, z, **kwargs):\n        if n >= S.One:\n            return (-1)**(n + 1)*factorial(n)*zeta(n + 1, z)\n        else:\n            return self\n\n    def _eval_rewrite_as_harmonic(self, n, z, **kwargs):\n        if n.is_integer:\n            if n == S.Zero:\n                return harmonic(z - 1) - S.EulerGamma\n            else:\n                return S.NegativeOne**(n+1) * factorial(n) * (zeta(n+1) - harmonic(z-1, n+1))\n\n    def _eval_as_leading_term(self, x):\n        from sympy import Order\n        n, z = [a.as_leading_term(x) for a in self.args]\n        o = Order(z, x)\n        if n == 0 and o.contains(1/x):\n            return o.getn() * log(x)\n        else:\n            return self.func(n, z)"
        },
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "function",
          "name": "fdiff",
          "class_name": "multigamma",
          "code": "def fdiff(self, argindex=2):\n        from sympy import Sum\n        if argindex == 2:\n            x, p = self.args\n            k = Dummy(\"k\")\n            return self.func(x, p)*Sum(polygamma(0, x + (1 - k)/2), (k, 1, p))\n        else:\n            raise ArgumentIndexError(self, argindex)"
        },
        {
          "file": "sympy/functions/special/gamma_functions.py",
          "type": "function",
          "name": "_eval_expand_func",
          "class_name": "loggamma",
          "code": "def _eval_expand_func(self, **hints):\n        from sympy import Sum\n        z = self.args[0]\n\n        if z.is_Rational:\n            p, q = z.as_numer_denom()\n            # General rational arguments (u + p/q)\n            # Split z as n + p/q with p < q\n            n = p // q\n            p = p - n*q\n            if p.is_positive and q.is_positive and p < q:\n                k = Dummy(\"k\")\n                if n.is_positive:\n                    return loggamma(p / q) - n*log(q) + Sum(log((k - 1)*q + p), (k, 1, n))\n                elif n.is_negative:\n                    return loggamma(p / q) - n*log(q) + S.Pi*S.ImaginaryUnit*n - Sum(log(k*q - p), (k, 1, -n))\n                elif n.is_zero:\n                    return loggamma(p / q)\n\n        return self"
        }
      ]
    },
    {
      "pr_number": 18578,
      "pr_title": "Fixes bug in Series",
      "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\n#Fixes #10503\r\n#### Brief description of what is fixed or changed\r\nNow, \r\n`f=exp(x**3)*cos(x**6)`\r\n`f.series(x, 0, 14)`\r\nreturn correct result.\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\nNO ENTRY\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 10503,
      "issue_title": "Series return an incorrect result",
      "issue_body": "The latest sympy master:\n\n```\nIn [1]: f=exp(x**3)*cos(x**6)\n\nIn [2]: f.series(x, 0, 14)\nOut[2]: \n          6    9    12         \n     3   x    x    x      \u239b 14\u239e\n1 + x  + \u2500\u2500 + \u2500\u2500 + \u2500\u2500\u2500 + O\u239dx  \u23a0\n         2    6     24         \n\nIn [3]: f.series(x, 0, 19)\nOut[3]: \n          6    9       12       15        18         \n     3   x    x    11\u22c5x     59\u22c5x     179\u22c5x      \u239b 19\u239e\n1 + x  + \u2500\u2500 + \u2500\u2500 - \u2500\u2500\u2500\u2500\u2500\u2500 - \u2500\u2500\u2500\u2500\u2500\u2500 - \u2500\u2500\u2500\u2500\u2500\u2500\u2500 + O\u239dx  \u23a0\n         2    6      24      120       720           \n```\n",
      "issue_closed_at": "2020-02-06T12:07:56Z",
      "base_commit": "a8a3a3b026cc55aa14010fc7cd7909806b6e116c",
      "changes": [
        {
          "file": "sympy/core/function.py",
          "type": "function",
          "name": "_eval_nseries",
          "class_name": "Derivative",
          "code": "def _eval_nseries(self, x, n, logx):\n        if x in self.point:\n            # x is the variable being substituted into\n            apos = self.point.index(x)\n            other = self.variables[apos]\n        else:\n            other = x\n        arg = self.expr.nseries(other, n=n, logx=logx)\n        o = arg.getO()\n        terms = Add.make_args(arg.removeO())\n        rv = Add(*[self.func(a, *self.args[1:]) for a in terms])\n        if o:\n            rv += o.subs(other, x)\n        return rv"
        }
      ]
    }
  ]
}