{
  "instance_id": "sympy__sympy-19254",
  "repo": "sympy/sympy",
  "created_at": "2020-05-04T13:38:13Z",
  "problem_statement": "sympy.polys.factortools.dmp_zz_mignotte_bound improvement\nThe method `dup_zz_mignotte_bound(f, K)` can be significantly improved by using the **Knuth-Cohen bound** instead. After our research with Prof. Ag.Akritas we have implemented the Knuth-Cohen bound among others, and compare them among dozens of polynomials with different degree, density and coefficients range. Considering the results and the feedback from Mr.Kalevi Suominen, our proposal is that the mignotte_bound should be replaced by the knuth-cohen bound.\r\nAlso, `dmp_zz_mignotte_bound(f, u, K)` for mutli-variants polynomials should be replaced appropriately.\n",
  "patch": "diff --git a/sympy/polys/factortools.py b/sympy/polys/factortools.py\n--- a/sympy/polys/factortools.py\n+++ b/sympy/polys/factortools.py\n@@ -124,13 +124,64 @@ def dmp_trial_division(f, factors, u, K):\n \n \n def dup_zz_mignotte_bound(f, K):\n-    \"\"\"Mignotte bound for univariate polynomials in `K[x]`. \"\"\"\n-    a = dup_max_norm(f, K)\n-    b = abs(dup_LC(f, K))\n-    n = dup_degree(f)\n+    \"\"\"\n+    The Knuth-Cohen variant of Mignotte bound for\n+    univariate polynomials in `K[x]`.\n \n-    return K.sqrt(K(n + 1))*2**n*a*b\n+    Examples\n+    ========\n+\n+    >>> from sympy.polys import ring, ZZ\n+    >>> R, x = ring(\"x\", ZZ)\n+\n+    >>> f = x**3 + 14*x**2 + 56*x + 64\n+    >>> R.dup_zz_mignotte_bound(f)\n+    152\n+\n+    By checking `factor(f)` we can see that max coeff is 8\n+\n+    Also consider a case that `f` is irreducible for example `f = 2*x**2 + 3*x + 4`\n+    To avoid a bug for these cases, we return the bound plus the max coefficient of `f`\n+\n+    >>> f = 2*x**2 + 3*x + 4\n+    >>> R.dup_zz_mignotte_bound(f)\n+    6\n+\n+    Lastly,To see the difference between the new and the old Mignotte bound\n+    consider the irreducible polynomial::\n+\n+    >>> f = 87*x**7 + 4*x**6 + 80*x**5 + 17*x**4 + 9*x**3 + 12*x**2 + 49*x + 26\n+    >>> R.dup_zz_mignotte_bound(f)\n+    744\n+\n+    The new Mignotte bound is 744 whereas the old one (SymPy 1.5.1) is 1937664.\n+\n+\n+    References\n+    ==========\n+\n+    ..[1] [Abbott2013]_\n+\n+    \"\"\"\n+    from sympy import binomial\n+\n+    d = dup_degree(f)\n+    delta = _ceil(d / 2)\n+    delta2 = _ceil(delta / 2)\n+\n+    # euclidean-norm\n+    eucl_norm = K.sqrt( sum( [cf**2 for cf in f] ) )\n+\n+    # biggest values of binomial coefficients (p. 538 of reference)\n+    t1 = binomial(delta - 1, delta2)\n+    t2 = binomial(delta - 1, delta2 - 1)\n+\n+    lc = K.abs(dup_LC(f, K))   # leading coefficient\n+    bound = t1 * eucl_norm + t2 * lc   # (p. 538 of reference)\n+    bound += dup_max_norm(f, K) # add max coeff for irreducible polys\n+    bound = _ceil(bound / 2) * 2   # round up to even integer\n \n+    return bound\n \n def dmp_zz_mignotte_bound(f, u, K):\n     \"\"\"Mignotte bound for multivariate polynomials in `K[X]`. \"\"\"\n",
  "similar_bug_items": [
    {
      "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": 13564,
      "pr_title": "Resolves RecursionError in latex representation",
      "pr_body": "fixes #13559. It was due to factor of `1/1`  \r\n```python\r\nexpr = parse_expr('5/1', evaluate=False)\r\nexpr.args\r\n```\r\nit gives `(5 , 1/1)` , which was not handled properly in latex conversion . So everytime an expression was divided by 1 , it resulted in RecursionError",
      "issue_id": 13559,
      "issue_title": "LaTeX representation raises RecursionError",
      "issue_body": "LaTeX representation of an expression that has division by 1 causes `RecursionError`. SymPy version: 1.1.1.\r\n\r\nCode to reproduce the bug:\r\n```python\r\nfrom sympy import latex\r\nfrom sympy.parsing.sympy_parser import parse_expr\r\n\r\nexpr = parse_expr('2/1', evaluate=False)\r\nlatex(expr)\r\n```\r\nOutput:\r\n```\r\n...\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/printer.py in _print(self, expr, *args, **kwargs)\r\n    255                 printmethod = '_print_' + cls.__name__\r\n    256                 if hasattr(self, printmethod):\r\n--> 257                     return getattr(self, printmethod)(expr, *args, **kwargs)\r\n    258             # Unknown object, fall back to the emptyPrinter.\r\n    259             return self.emptyPrinter(expr)\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in _print_Pow(self, expr)\r\n    466         elif expr.exp.is_Rational and expr.exp.is_negative and expr.base.is_commutative:\r\n    467             # Things like 1/x\r\n--> 468             return self._print_Mul(expr)\r\n    469         else:\r\n    470             if expr.base.is_Function:\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in _print_Mul(self, expr)\r\n    396             # use the original expression here, since fraction() may have\r\n    397             # altered it when producing numer and denom\r\n--> 398             tex += convert(expr)\r\n    399         else:\r\n    400             snumer = convert(numer)\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/latex.py in convert(expr)\r\n    366         def convert(expr):\r\n    367             if not expr.is_Mul:\r\n--> 368                 return str(self._print(expr))\r\n    369             else:\r\n    370                 _tex = last_term_tex = \"\"\r\n\r\n... last 4 frames repeated, from the frame below ...\r\n\r\n~/.virtualenvs/sympy/lib/python3.5/site-packages/sympy/printing/printer.py in _print(self, expr, *args, **kwargs)\r\n    255                 printmethod = '_print_' + cls.__name__\r\n    256                 if hasattr(self, printmethod):\r\n--> 257                     return getattr(self, printmethod)(expr, *args, **kwargs)\r\n    258             # Unknown object, fall back to the emptyPrinter.\r\n    259             return self.emptyPrinter(expr)\r\n\r\nRecursionError: maximum recursion depth exceeded in comparison\r\n```\r\n",
      "issue_closed_at": "2017-11-09T16:49:04Z",
      "base_commit": "36d9c850c642bec047e2cbad0a07c48f4fc7c779",
      "changes": [
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "_print_Gradient",
          "class_name": "LatexPrinter",
          "code": "def _print_Gradient(self, expr):\n        func = expr._expr\n        return r\"\\nabla\\cdot %s\" % self.parenthesize(func, PRECEDENCE['Mul'])"
        },
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "convert",
          "class_name": "LatexPrinter",
          "code": "def convert(expr):\n            if not expr.is_Mul:\n                return str(self._print(expr))\n            else:\n                _tex = last_term_tex = \"\"\n\n                if self.order not in ('old', 'none'):\n                    args = expr.as_ordered_factors()\n                else:\n                    args = expr.args\n\n                for i, term in enumerate(args):\n                    term_tex = self._print(term)\n\n                    if self._needs_mul_brackets(term, first=(i == 0),\n                                                last=(i == len(args) - 1)):\n                        term_tex = r\"\\left(%s\\right)\" % term_tex\n\n                    if _between_two_numbers_p[0].search(last_term_tex) and \\\n                            _between_two_numbers_p[1].match(term_tex):\n                        # between two numbers\n                        _tex += numbersep\n                    elif _tex:\n                        _tex += separator\n\n                    _tex += term_tex\n                    last_term_tex = term_tex\n                return _tex"
        }
      ]
    },
    {
      "pr_number": 12882,
      "pr_title": "Checker functions for orthogonality of coordinate system",
      "pr_body": "This PR introduce additional condition for transformation equation which must be fulfill. \r\nIn `vector` module we are dealing with orthogonal curvilinear coordinate system only, so we need to check if transformation equations are correctly defined.\r\n\r\nCloses #12852",
      "issue_id": 12852,
      "issue_title": "Vector module: make sure that the base vectors remain orthogonal",
      "issue_body": "Given a generic transformation equation, make sure that the base vectors remain orthogonal, if not, raise an exception.",
      "issue_closed_at": "2017-07-05T15:10:49Z",
      "base_commit": "c87c0fbf1223c28db8590f2591f64db75d5bdd66",
      "changes": [
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "line",
          "name": "line 4",
          "code": "from sympy.core.cache import cacheit\nfrom sympy.core import S\nfrom sympy.vector.scalar import BaseScalar\nfrom sympy import eye, trigsimp, ImmutableMatrix as Matrix, Symbol, sin, cos, sqrt, diff, Tuple\nimport sympy.vector\nfrom sympy import simplify\nfrom sympy.vector.orienters import (Orienter, AxisOrienter, BodyOrienter,"
        },
        {
          "file": "sympy/vector/coordsysrect.py",
          "type": "function",
          "name": "_connect_to_standard_cartesian",
          "class_name": "CoordSys3D",
          "code": "def _connect_to_standard_cartesian(self, curv_coord_type):\n        \"\"\"\n        Change the type of orthogonal curvilinear system. It could be done\n        by tuple of transformation equations or by choosing one of pre-defined\n        coordinate system.\n\n        Parameters\n        ==========\n\n        :param curv_coord_type: str, tuple\n\n        \"\"\"\n        if isinstance(curv_coord_type, string_types):\n            self._set_transformation_equations_mapping(curv_coord_type)\n            self._set_lame_coefficient_mapping(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 3:\n            self._transformation_eqs = curv_coord_type\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(curv_coord_type)\n\n        elif isinstance(curv_coord_type, (tuple, list, Tuple)) and len(curv_coord_type) == 2:\n            self._transformation_eqs = \\\n            tuple([eq.subs({curv_coord_type[0][0]: self.x,\n                            curv_coord_type[0][1]: self.y,\n                            curv_coord_type[0][2]: self.z}) for eq in curv_coord_type[1]])\n            self._h1, self._h2, self._h3 = self._calculate_lame_coefficients(self._transformation_equations())\n\n        else:\n            raise ValueError(\"Wrong set of parameter.\")"
        }
      ]
    },
    {
      "pr_number": 15519,
      "pr_title": "Exit _eval_derivative_n_times when None is obtained",
      "pr_body": "#### References to other Issues or PRs\r\n\r\nFixes #15518 \r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nThe method `_eval_derivative_n_times` contains a loop that involves calling `_eval_derivative`. The latter may return None when evaluation is not implemented. In such a case the loop should be aborted (returning None) instead of trying to differentiate None.\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* core\r\n  * Fixed a bug in the creation of higher order derivatives that cannot be evaluated.  \r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15518,
      "issue_title": "Higher order derivatives that cannot be evaluated, like re(x).diff(x, 2), raise an exception",
      "issue_body": "This issue arose in a [comment on another issue]( https://github.com/sympy/sympy/issues/15457#issuecomment-436453327) but since it's actually unrelated, is posted here separately. \r\n```\r\n>>> re(x).diff(x, 2)\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/ubuntu/sympy/sympy/core/expr.py\", line 3047, in diff\r\n    return Derivative(self, *symbols, **assumptions)\r\n  File \"/home/ubuntu/sympy/sympy/core/function.py\", line 1370, in __new__\r\n    obj = expr._eval_derivative_n_times(v, count)\r\n  File \"/home/ubuntu/sympy/sympy/core/basic.py\", line 1688, in _eval_derivative_n_times\r\n    obj2 = obj._accept_eval_derivative(s)\r\nAttributeError: 'NoneType' object has no attribute '_accept_eval_derivative'\r\n```\r\nThis happens whenever a higher-order derivative is requested for a function that does not have an explicit derivative. The expected output is \r\n```\r\nDerivative(re(x), (x, 2))\r\n```\r\nA PR is forthcoming. \r\n",
      "issue_closed_at": "2018-11-24T11:36:42Z",
      "base_commit": "61e5c20c02328815270ddda385f0108a2b40d24d",
      "changes": [
        {
          "file": "sympy/core/basic.py",
          "type": "function",
          "name": "_eval_derivative_n_times",
          "class_name": "Basic",
          "code": "def _eval_derivative_n_times(self, s, n):\n        # This is the default evaluator for derivatives (as called by `diff`\n        # and `Derivative`), it will attempt a loop to derive the expression\n        # `n` times by calling the corresponding `_eval_derivative` method,\n        # while leaving the derivative unevaluated if `n` is symbolic.  This\n        # method should be overridden if the object has a closed form for its\n        # symbolic n-th derivative.\n        from sympy import Integer\n        if isinstance(n, (int, Integer)):\n            obj = self\n            for i in range(n):\n                obj2 = obj._accept_eval_derivative(s)\n                if obj == obj2:\n                    break\n                obj = obj2\n            return obj\n        else:\n            return None"
        }
      ]
    },
    {
      "pr_number": 2707,
      "pr_title": "remove practice of evaluating factorial as 0 for negative numbers",
      "pr_body": "it is infinity for negative integers and remains unevaluated for non-integers. fix gosper test, which had a missing term.\nfix string representations\n",
      "issue_id": 2703,
      "issue_title": "uncommon convention for factorials of negative numbers",
      "issue_body": "from the docstring of the factorial function:\n\n\"For the sake of convenience and simplicity of procedures using\nthis function it is defined for negative integers and returns\nzero in this case.\"\n\nBy convention (and by some definitions), n! is usually infinity for n < 0. This is consistent with the Gamma function identity n! = Gamma(n+1) and with the convention that binomial coefficients are 0 outside of Pascal's triangle. In fact, in the same module, we have, in the doc string for binomial coefficients:\n\n\"For the sake of convenience for negative 'k' this function\nwill return zero no matter what valued is the other argument.\"\n\nThis is incompatible with sympy's convention on factorials, given the definition of the binomial coefficients: C(n,k) = n!/(k!(n-k)!)\n\nFor example take \"2 choose -1\", C(2,-1) = 0. However, going to sympy's definition of the factorial, we would have 2!/((-1)! \\* 3!) == 1/0 == oo, which is obviously wrong. Switching over to the Gamma function gives us the correct answer.\n\nDoes anyone know what \"procedures using\" the factorial function rely on the \"convenience and simplicity\" of the incorrect convention n! = 0  for n < 0? It would nice to bring the factorial function to consistency with other parts of sympy (like the binomial coefficients) and the more common convention in the math and cs communities.\n",
      "issue_closed_at": "2014-01-05T13:28:55Z",
      "base_commit": "5879fa2318e853807a6cf63981f2bca441cceb9f",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/factorials.py",
          "type": "function",
          "name": "_eval_simplify",
          "class_name": "CombinatorialFunction",
          "code": "def _eval_simplify(self, ratio, measure):\n        from sympy.simplify.simplify import combsimp\n        expr = combsimp(self)\n        if measure(expr) <= ratio*measure(self):\n            return expr\n        return self"
        },
        {
          "file": "sympy/functions/combinatorial/factorials.py",
          "type": "class",
          "name": "factorial",
          "code": "class factorial(CombinatorialFunction):\n    \"\"\"Implementation of factorial function over nonnegative integers.\n       For the sake of convenience and simplicity of procedures using\n       this function it is defined for negative integers and returns\n       zero in this case.\n\n       The factorial is very important in combinatorics where it gives\n       the number of ways in which 'n' objects can be permuted. It also\n       arises in calculus, probability, number theory etc.\n\n       There is strict relation of factorial with gamma function. In\n       fact n! = gamma(n+1) for nonnegative integers. Rewrite of this\n       kind is very useful in case of combinatorial simplification.\n\n       Computation of the factorial is done using two algorithms. For\n       small arguments naive product is evaluated. However for bigger\n       input algorithm Prime-Swing is used. It is the fastest algorithm\n       known and computes n! via prime factorization of special class\n       of numbers, called here the 'Swing Numbers'.\n\n       Examples\n       ========\n\n       >>> from sympy import Symbol, factorial\n       >>> n = Symbol('n', integer=True)\n\n       >>> factorial(-2)\n       0\n\n       >>> factorial(0)\n       1\n\n       >>> factorial(7)\n       5040\n\n       >>> factorial(n)\n       factorial(n)\n\n       >>> factorial(2*n)\n       factorial(2*n)\n\n       See Also\n       ========\n\n       factorial2, RisingFactorial, FallingFactorial\n    \"\"\"\n\n    nargs = 1\n\n    def fdiff(self, argindex=1):\n        if argindex == 1:\n            return C.gamma(self.args[0] + 1)*C.polygamma(0, self.args[0] + 1)\n        else:\n            raise ArgumentIndexError(self, argindex)\n\n    _small_swing = [\n        1, 1, 1, 3, 3, 15, 5, 35, 35, 315, 63, 693, 231, 3003, 429, 6435, 6435, 109395,\n        12155, 230945, 46189, 969969, 88179, 2028117, 676039, 16900975, 1300075,\n        35102025, 5014575, 145422675, 9694845, 300540195, 300540195\n    ]\n\n    @classmethod\n    def _swing(cls, n):\n        if n < 33:\n            return cls._small_swing[n]\n        else:\n            N, primes = int(_sqrt(n)), []\n\n            for prime in sieve.primerange(3, N + 1):\n                p, q = 1, n\n\n                while True:\n                    q //= prime\n\n                    if q > 0:\n                        if q & 1 == 1:\n                            p *= prime\n                    else:\n                        break\n\n                if p > 1:\n                    primes.append(p)\n\n            for prime in sieve.primerange(N + 1, n//3 + 1):\n                if (n // prime) & 1 == 1:\n                    primes.append(prime)\n\n            L_product = R_product = 1\n\n            for prime in sieve.primerange(n//2 + 1, n + 1):\n                L_product *= prime\n\n            for prime in primes:\n                R_product *= prime\n\n            return L_product*R_product\n\n    @classmethod\n    def _recursive(cls, n):\n        if n < 2:\n            return 1\n        else:\n            return (cls._recursive(n//2)**2)*cls._swing(n)\n\n    @classmethod\n    def eval(cls, n):\n        n = sympify(n)\n\n        if n.is_Number:\n            if n is S.Zero:\n                return S.One\n            elif n is S.Infinity:\n                return S.Infinity\n            elif n.is_Integer:\n                if n.is_negative:\n                    return S.Zero\n                else:\n                    n, result = n.p, 1\n\n                    if n < 20:\n                        for i in range(2, n + 1):\n                            result *= i\n                    else:\n                        N, bits = n, 0\n\n                        while N != 0:\n                            if N & 1 == 1:\n                                bits += 1\n\n                            N = N >> 1\n\n                        result = cls._recursive(n)*2**(n - bits)\n\n                    return C.Integer(result)\n\n        if n.is_negative:\n            return S.Zero\n\n    def _eval_rewrite_as_gamma(self, n):\n        return C.gamma(n + 1)\n\n    def _eval_is_integer(self):\n        return self.args[0].is_integer\n\n    def _eval_is_positive(self):\n        if self.args[0].is_integer and self.args[0].is_positive:\n            return True"
        },
        {
          "file": "sympy/functions/combinatorial/factorials.py",
          "type": "function",
          "name": "eval",
          "class_name": "binomial",
          "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
        },
        {
          "file": "sympy/functions/combinatorial/factorials.py",
          "type": "function",
          "name": "eval",
          "class_name": "binomial",
          "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
        }
      ]
    }
  ]
}