a
    <b                    @   s  d Z ddlmZmZ ddlmZ ddlmZmZm	Z	m
Z
 ddlmZ ddlmZ ddlmZmZmZ ddlmZmZmZmZmZ dd	lmZ dd
lmZmZ ddlmZm Z m!Z! ddl"m#Z#m$Z$ ddl%m&Z& ddl'm(Z(m)Z) ddl*m+Z+m,Z, ddl-m.Z.m/Z/ ddl0m1Z1 ddl2m3Z4 ddl5m6Z6 ddl7m8Z8m9Z9m:Z: ddl;m<Z< ddl=m>Z> ddl?m@ZA ddlBmCZC ddlDmEZE ddlFmGZGmHZHmIZI ddlJmKZKmLZLmMZMmNZNmOZOmPZPmQZQmRZRmSZSmTZTmUZU ddlVmWZWmXZXmYZYmZZZm[Z[m\Z\ ddl]m^Z^ ddl_m`Z` ddlambZbmcZcmdZd dd lemfZf dd!lgmhZhmiZi dd"l2Zjdd"lkZkdd#llmmZm d$d% ZnecG d&d' d'eZoecG d(d) d)eoZpecd*d+ Zqd,d- Zrecd.d/ Zsd0d1 Ztd2d3 Zuecdd4d5Zvecd6d7 Zwecd8d9 Zxecd:d; Zyecd<d= Zzecd>d? Z{ecd@dA Z|ecdBdC Z}ecdDdE Z~ecdFdG ZecdHdI ZecdJdK ZecdLdM ZecdNdO ZecdPdQ ZecdRdS ZecdTdU ZecdVdW ZecdXdYdZd[Zecd\d] Zecd^d_ Zecd`da ZecddbdcZecddde ZecddfdgZecdhdi Zecdjdk Zecdldm Zecdndo Zecdpdq Zecdrds Zecdtdu Zecdvdw Zecdxdy Zecdzd{ Zecd|d} Zecd~d Zdd Zdd Zdd Zdd Zdd Zdd Zdd Zdd Zecdd Zecdd Zecdd ZecdXdddZecdddZecdddZecdddZecdddZecdddZecdd Zecdd ZecddddZecdd Zecdd Z@ecdd ZecG dd deZecdd Zd"S )z8User-friendly public interface to polynomial functions.     )wrapsreducemul)SExprAddTuple)Basic)
_sympifyit)Factors	factor_ncfactor_terms)pure_complexevalffastlog_evalf_with_bounded_errorquad_to_mpmath)
Derivative)Mul_keep_coeff)ilcmIInteger)
RelationalEquality)ordered)DummySymbol)sympify_sympify)preorder_traversal	bottom_up)BooleanAtom)polyoptions)construct_domain)FFQQZZ)DomainElement)matrix_fglm)groebner)Monomial)monomial_key)DMPDMFANP)OperationNotSupportedDomainErrorCoercionFailedUnificationFailedGeneratorsNeededPolynomialErrorMultivariatePolynomialErrorExactQuotientFailedPolificationFailedComputationFailedGeneratorsError)basic_from_dict
_sort_gens_unify_gens_dict_reorder_dict_from_expr_parallel_dict_from_expr)together)dup_isolate_real_roots_list)grouppublic
filldedent)sympy_deprecation_warning)iterablesiftN)NoConvergencec                    s   t   fdd}|S )Nc                    s   t |}t|tr | |S t|trz| j|g| jR  }W nR ty   |jrZt Y S t	| 
  j}||}|turtdddd | Y S 0  | |S ntS d S )Na@  
                        Mixing Poly with non-polynomial expressions in binary
                        operations is deprecated. Either explicitly convert
                        the non-Poly operand to a Poly with as_poly() or
                        convert the Poly to an Expr with as_expr().
                        z1.6z)deprecated-poly-nonpoly-binary-operations)Zdeprecated_since_versionZactive_deprecations_target)r    
isinstancePolyr   	from_exprgensr6   Z	is_MatrixNotImplementedgetattras_expr__name__rG   )fgZexpr_methodresultfunc e/Users/vegardjervell/Documents/master/model/venv/lib/python3.9/site-packages/sympy/polys/polytools.pywrapperC   s(    




z_polifyit.<locals>.wrapper)r   )rW   rZ   rX   rV   rY   	_polifyitB   s    r[   c                       sB  e Zd ZdZdZdZdZdZdd Ze	dd Z
ed	d
 Zedd Zdd Ze	dd Ze	dd Ze	dd Ze	dd Ze	dd Ze	dd Ze	dd Ze	dd Ze	dd  Z fd!d"Zed#d$ Zed%d& Zed'd( Zed)d* Zed+d, Zed-d. Zed/d0 Zd1d2 Z d3d4 Z!dtd6d7Z"d8d9 Z#d:d; Z$d<d= Z%d>d? Z&d@dA Z'dBdC Z(dudDdEZ)dFdG Z*dHdI Z+dJdK Z,dLdM Z-dNdO Z.dPdQ Z/dRdS Z0dvdTdUZ1dwdVdWZ2dxdXdYZ3dydZd[Z4dzd\d]Z5d^d_ Z6d`da Z7dbdc Z8ddde Z9dfdg Z:d{didjZ;d|dkdlZ<dmdn Z=dodp Z>dqdr Z?dsdt Z@d}dudvZAdwdx ZBdydz ZCd{d| ZDd}d~ ZEdd ZFdd ZGdd ZHdd ZIdd ZJdd ZKdd ZLdd ZMdd ZNdd ZOdd ZPdd ZQdd ZRdd ZSd~ddZTdddZUdddZVdddZWdd ZXdddZYdd ZZdd Z[dd Z\dd Z]dddZ^dd Z_dddZ`dd Zadd ZbdddZcdddZddddZedddZfdddĄZgddƄ ZhddȄ ZidddʄZjdd̄ Zkdd΄ ZlddЄ ZmemZnddd҄ZoddԄ ZpdddքZqddd؄ZrdddڄZsdd܄ Ztddބ ZudddZvdd ZwdddZxdddZydd Zzdd Z{dd Z|dd Z}dddZ~dd Zdd Zdd Zdd Zdd Zdd ZdddZdd  Zdd Zdd Zdd ZdddZdd	d
Zdd Zdd ZdddZdddZdddZdddZdddZdddZdddZdd  Zd!d" Zd#d$ Zdd%d&Zed'd( Zed)d* Zed+d, Zed-d. Zed/d0 Zed1d2 Zed3d4 Zed5d6 Zed7d8 Zed9d: Zed;d< Zed=d> Zed?d@ ZedAdB ZdCdD ZdEdF ZedGdH ZedIdJ ZedKdL ZedMdN ZedOdP ZedQdR ZedSedTdU ZedVdW ZedXdY ZedZd[ Zed\d] Zed^d_ Zed`da Zedbedcdd Zedbededf Zedgedhdi Zedbedjdk Zdldm ZddndoZddpdqZdrds Z  ZS (  rL   aP  
    Generic class for representing and operating on polynomial expressions.

    See :ref:`polys-docs` for general documentation.

    Poly is a subclass of Basic rather than Expr but instances can be
    converted to Expr with the :py:meth:`~.Poly.as_expr` method.

    .. deprecated:: 1.6

       Combining Poly with non-Poly objects in binary operations is
       deprecated. Explicitly convert both objects to either Poly or Expr
       first. See :ref:`deprecated-poly-nonpoly-binary-operations`.

    Examples
    ========

    >>> from sympy import Poly
    >>> from sympy.abc import x, y

    Create a univariate polynomial:

    >>> Poly(x*(x**2 + x - 1)**2)
    Poly(x**5 + 2*x**4 - x**3 - 2*x**2 + x, x, domain='ZZ')

    Create a univariate polynomial with specific domain:

    >>> from sympy import sqrt
    >>> Poly(x**2 + 2*x + sqrt(3), domain='R')
    Poly(1.0*x**2 + 2.0*x + 1.73205080756888, x, domain='RR')

    Create a multivariate polynomial:

    >>> Poly(y*x**2 + x*y + 1)
    Poly(x**2*y + x*y + 1, x, y, domain='ZZ')

    Create a univariate polynomial, where y is a constant:

    >>> Poly(y*x**2 + x*y + 1,x)
    Poly(y*x**2 + y*x + 1, x, domain='ZZ[y]')

    You can evaluate the above polynomial as a function of y:

    >>> Poly(y*x**2 + x*y + 1,x).eval(2)
    6*y + 1

    See Also
    ========

    sympy.core.expr.Expr

    reprN   Tgn $@c                 O   s   t ||}d|v rtdt|ttttfr:| ||S t	|t
drnt|tr\| ||S | t||S n&t|}|jr| ||S | ||S dS )z:Create a new polynomial instance out of something useful. orderz&'order' keyword is not implemented yet)excludeN)optionsbuild_optionsNotImplementedErrorrK   r.   r/   r0   r)   _from_domain_elementrH   strdict
_from_dict
_from_listlistr   is_Poly
_from_poly
_from_exprclsr]   rN   argsoptrX   rX   rY   __new__   s    
zPoly.__new__c                 G   sT   t |tstd| n"|jt|d kr:td||f t| }||_||_|S )z:Construct :class:`Poly` instance from raw representation. z%invalid polynomial representation: %s   zinvalid arguments: %s, %s)	rK   r.   r6   levlenr
   rp   r]   rN   )rm   r]   rN   objrX   rX   rY   new   s    

zPoly.newc                 C   s   t | j g| jR  S N)r<   r]   to_sympy_dictrN   selfrX   rX   rY   expr   s    z	Poly.exprc                 C   s   | j f| j S rv   )rz   rN   rx   rX   rX   rY   rn      s    z	Poly.argsc                 C   s   | j f| j S rv   r\   rx   rX   rX   rY   _hashable_content   s    zPoly._hashable_contentc                 O   s   t ||}| ||S )(Construct a polynomial from a ``dict``. )r`   ra   rf   rl   rX   rX   rY   	from_dict   s    zPoly.from_dictc                 O   s   t ||}| ||S )(Construct a polynomial from a ``list``. )r`   ra   rg   rl   rX   rX   rY   	from_list   s    zPoly.from_listc                 O   s   t ||}| ||S )*Construct a polynomial from a polynomial. )r`   ra   rj   rl   rX   rX   rY   	from_poly   s    zPoly.from_polyc                 O   s   t ||}| ||S +Construct a polynomial from an expression. )r`   ra   rk   rl   rX   rX   rY   rM      s    zPoly.from_exprc                 C   sz   |j }|stdt|d }|j}|du r>t||d\}}n | D ]\}}||||< qF| jt	|||g|R  S )r|   z0Cannot initialize from 'dict' without generatorsrq   Nro   )
rN   r5   rs   domainr%   itemsconvertru   r.   r}   )rm   r]   ro   rN   levelr   monomcoeffrX   rX   rY   rf      s    zPoly._from_dictc                 C   s   |j }|stdnt|dkr(tdt|d }|j}|du rTt||d\}}ntt|j|}| j	t
|||g|R  S )r~   z0Cannot initialize from 'list' without generatorsrq   z#'list' representation not supportedNr   )rN   r5   rs   r7   r   r%   rh   mapr   ru   r.   r   )rm   r]   ro   rN   r   r   rX   rX   rY   rg     s    zPoly._from_listc                 C   s   | |j kr | j|jg|jR  }|j}|j}|j}|rl|j|krlt|jt|krb| | |S |j	| }d|v r|r|
|}n|du r| }|S )r   r   T)	__class__ru   r]   rN   fieldr   setrk   rQ   reorder
set_domainto_field)rm   r]   ro   rN   r   r   rX   rX   rY   rj     s    

zPoly._from_polyc                 C   s   t ||\}}| ||S r   )r@   rf   )rm   r]   ro   rX   rX   rY   rk   3  s    zPoly._from_exprc                 C   s@   |j }|j}t|d }||g}| jt|||g|R  S Nrq   )rN   r   rs   r   ru   r.   r   )rm   r]   ro   rN   r   r   rX   rX   rY   rc   9  s
    zPoly._from_domain_elementc                    s
   t   S rv   super__hash__rx   r   rX   rY   r   C  s    zPoly.__hash__c                 C   sP   t  }| j}tt|D ],}|  D ]}|| r$||| jO } qq$q|| jB S )a  
        Free symbols of a polynomial expression.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> Poly(x**2 + 1).free_symbols
        {x}
        >>> Poly(x**2 + y).free_symbols
        {x, y}
        >>> Poly(x**2 + y, x).free_symbols
        {x, y}
        >>> Poly(x**2 + y, x, z).free_symbols
        {x, y}

        )r   rN   rangers   monomsfree_symbolsfree_symbols_in_domain)ry   symbolsrN   ir   rX   rX   rY   r   F  s    zPoly.free_symbolsc                 C   sP   | j jt  }}|jr.|jD ]}||jO }qn|jrL|  D ]}||jO }q<|S )aj  
        Free symbols of the domain of ``self``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 1).free_symbols_in_domain
        set()
        >>> Poly(x**2 + y).free_symbols_in_domain
        set()
        >>> Poly(x**2 + y, x).free_symbols_in_domain
        {y}

        )r]   domr   is_Compositer   r   is_EXcoeffs)ry   r   r   genr   rX   rX   rY   r   e  s    
zPoly.free_symbols_in_domainc                 C   s
   | j d S )z
        Return the principal generator.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).gen
        x

        r   rN   rx   rX   rX   rY   r     s    zPoly.genc                 C   s   |   S )a  Get the ground domain of a :py:class:`~.Poly`

        Returns
        =======

        :py:class:`~.Domain`:
            Ground domain of the :py:class:`~.Poly`.

        Examples
        ========

        >>> from sympy import Poly, Symbol
        >>> x = Symbol('x')
        >>> p = Poly(x**2 + x)
        >>> p
        Poly(x**2 + x, x, domain='ZZ')
        >>> p.domain
        ZZ
        )
get_domainrx   rX   rX   rY   r     s    zPoly.domainc                 C   s&   | j | j| jj| jjg| jR  S )z3Return zero polynomial with ``self``'s properties. )ru   r]   zerorr   r   rN   rx   rX   rX   rY   r     s    z	Poly.zeroc                 C   s&   | j | j| jj| jjg| jR  S )z2Return one polynomial with ``self``'s properties. )ru   r]   onerr   r   rN   rx   rX   rX   rY   r     s    zPoly.onec                 C   s&   | j | j| jj| jjg| jR  S )z3Return unit polynomial with ``self``'s properties. )ru   r]   unitrr   r   rN   rx   rX   rX   rY   r     s    z	Poly.unitc                 C   s"   |  |\}}}}||||fS )a  
        Make ``f`` and ``g`` belong to the same domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f, g = Poly(x/2 + 1), Poly(2*x + 1)

        >>> f
        Poly(1/2*x + 1, x, domain='QQ')
        >>> g
        Poly(2*x + 1, x, domain='ZZ')

        >>> F, G = f.unify(g)

        >>> F
        Poly(1/2*x + 1, x, domain='QQ')
        >>> G
        Poly(2*x + 1, x, domain='QQ')

        )_unify)rS   rT   _perFGrX   rX   rY   unify  s    z
Poly.unifyc                    s  t jsZz(jjjjjjjfW S  tyX   tdf Y n0 tjt	rtjt	rt
jj}jjjj|t|d  }j|krtj j|\}}jjkrfdd|D }t	ttt|||}nj}j|krttj j|\}}jjkrXfdd|D }t	ttt|||}	nj}	ntdf j |d f fdd	}
|
||	fS )NCannot unify %s with %srq   c                    s   g | ]}  |jjqS rX   r   r]   r   .0c)r   rS   rX   rY   
<listcomp>      zPoly._unify.<locals>.<listcomp>c                    s   g | ]}  |jjqS rX   r   r   )r   rT   rX   rY   r     r   c                    sD   |d ur2|d | ||d d   }|s2| | S  j| g|R  S r   to_sympyru   r]   r   rN   removerm   rX   rY   r     s
    
zPoly._unify.<locals>.per)r   ri   r]   r   r   
from_sympyr3   r4   rK   r.   r>   rN   r   rs   r?   to_dictre   rh   zipr   r   )rS   rT   rN   rr   Zf_monomsZf_coeffsr   Zg_monomsZg_coeffsr   r   rX   )rm   r   rS   rT   rY   r     s:    ("	zPoly._unifyNc                 C   sX   |du r| j }|durD|d| ||d d  }|sD| jj|S | jj|g|R  S )ab  
        Create a Poly out of the given representation.

        Examples
        ========

        >>> from sympy import Poly, ZZ
        >>> from sympy.abc import x, y

        >>> from sympy.polys.polyclasses import DMP

        >>> a = Poly(x**2 + 1)

        >>> a.per(DMP([ZZ(1), ZZ(1)], ZZ), gens=[y])
        Poly(y + 1, y, domain='ZZ')

        Nrq   )rN   r]   r   r   r   ru   )rS   r]   rN   r   rX   rX   rY   r   	  s    zPoly.perc                 C   s&   t | jd|i}| | j|jS )z Set the ground domain of ``f``. r   )r`   ra   rN   r   r]   r   r   )rS   r   ro   rX   rX   rY   r   &  s    zPoly.set_domainc                 C   s   | j jS )z Get the ground domain of ``f``. )r]   r   rS   rX   rX   rY   r   +  s    zPoly.get_domainc                 C   s   t j|}| t|S )z
        Set the modulus of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(5*x**2 + 2*x - 1, x).set_modulus(2)
        Poly(x**2 + 1, x, modulus=2)

        )r`   ZModulus
preprocessr   r&   )rS   modulusrX   rX   rY   set_modulus/  s    zPoly.set_modulusc                 C   s&   |   }|jrt| S tddS )z
        Get the modulus of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, modulus=2).get_modulus()
        2

        z$not a polynomial over a Galois fieldN)r   Zis_FiniteFieldr   Zcharacteristicr6   )rS   r   rX   rX   rY   get_modulus@  s    zPoly.get_modulusc                 C   sN   || j v r>|jr| ||S z| ||W S  ty<   Y n0 |  ||S )z)Internal implementation of :func:`subs`. )rN   	is_numberevalreplacer6   rQ   subs)rS   oldru   rX   rX   rY   
_eval_subsU  s    
zPoly._eval_subsc                 C   sL   | j  \}}g }tt| jD ]}||vr || j|  q | j||dS )a  
        Remove unnecessary generators from ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import a, b, c, d, x

        >>> Poly(a + x, a, b, c, d, x).exclude()
        Poly(a + x, a, x, domain='ZZ')

        r   )r]   r_   r   rs   rN   appendr   )rS   Jru   rN   jrX   rX   rY   r_   b  s    zPoly.excludec                 K   s   |du r$| j r| j| }}ntd||ks6|| jvr:| S || jv r|| jvr|  }|jrf||jvrt| j}||||< | j	| j
|dS td||| f dS )a  
        Replace ``x`` with ``y`` in generators list.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 1, x).replace(x, y)
        Poly(y**2 + 1, y, domain='ZZ')

        Nz(syntax supported only in univariate caser   zCannot replace %s with %s in %s)is_univariater   r6   rN   r   r   r   rh   indexr   r]   )rS   xy_ignorer   rN   rX   rX   rY   r   y  s    
zPoly.replacec                 O   s   |   j|i |S )z-Match expression from Poly. See Basic.match())rQ   match)rS   rn   kwargsrX   rX   rY   r     s    z
Poly.matchc                 O   s|   t d|}|s t| j|d}nt| jt|kr:tdtttt	| j
 | j| }| jt|| j
jt|d |dS )a  
        Efficiently apply new order of generators.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + x*y**2, x, y).reorder(y, x)
        Poly(y**2*x + x**2, y, x, domain='ZZ')

        rX   r   z7generators list can differ only up to order of elementsrq   r   )r`   Optionsr=   rN   r   r6   re   rh   r   r?   r]   r   r   r.   r   rs   )rS   rN   rn   ro   r]   rX   rX   rY   r     s     zPoly.reorderc                 C   s   | j dd}| |}i }| D ]4\}}t|d| rFtd|  ||||d < q"| j|d }| jt|t	|d | j
jg|R  S )a(  
        Remove dummy generators from ``f`` that are to the left of
        specified ``gen`` in the generators as ordered. When ``gen``
        is an integer, it refers to the generator located at that
        position within the tuple of generators of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> Poly(y**2 + y*z**2, x, y, z).ltrim(y)
        Poly(y**2 + y*z**2, y, z, domain='ZZ')
        >>> Poly(z, x, y, z).ltrim(-1)
        Poly(z, z, domain='ZZ')

        T)nativeNzCannot left trim %srq   )as_dict_gen_to_levelr   anyr6   rN   ru   r.   r}   rs   r]   r   )rS   r   r]   r   termsr   r   rN   rX   rX   rY   ltrim  s    
z
Poly.ltrimc              	   G   s   t  }|D ]B}z| j|}W n" ty@   td| |f Y q
0 || q
|  D ]*}t|D ]\}}||vrb|rb  dS qbqVdS )aJ  
        Return ``True`` if ``Poly(f, *gens)`` retains ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> Poly(x*y + 1, x, y, z).has_only_gens(x, y)
        True
        >>> Poly(x*y + z, x, y, z).has_only_gens(x, y)
        False

        %s doesn't have %s as generatorFT)r   rN   r   
ValueErrorr;   addr   	enumerate)rS   rN   indicesr   r   r   r   eltrX   rX   rY   has_only_gens  s    

zPoly.has_only_gensc                 C   s,   t | jdr| j }n
t| d| |S )z
        Make the ground domain a ring.

        Examples
        ========

        >>> from sympy import Poly, QQ
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, domain=QQ).to_ring()
        Poly(x**2 + 1, x, domain='ZZ')

        to_ring)hasattrr]   r   r1   r   rS   rU   rX   rX   rY   r      s    
zPoly.to_ringc                 C   s,   t | jdr| j }n
t| d| |S )z
        Make the ground domain a field.

        Examples
        ========

        >>> from sympy import Poly, ZZ
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x, domain=ZZ).to_field()
        Poly(x**2 + 1, x, domain='QQ')

        r   )r   r]   r   r1   r   r   rX   rX   rY   r     s    
zPoly.to_fieldc                 C   s,   t | jdr| j }n
t| d| |S )z
        Make the ground domain exact.

        Examples
        ========

        >>> from sympy import Poly, RR
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1.0, x, domain=RR).to_exact()
        Poly(x**2 + 1, x, domain='QQ')

        to_exact)r   r]   r   r1   r   r   rX   rX   rY   r   *  s    
zPoly.to_exactc                 C   s4   t | jdd|| jjpdd\}}| j|| j|dS )a  
        Recalculate the ground domain of a polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = Poly(x**2 + 1, x, domain='QQ[y]')
        >>> f
        Poly(x**2 + 1, x, domain='QQ[y]')

        >>> f.retract()
        Poly(x**2 + 1, x, domain='ZZ')
        >>> f.retract(field=True)
        Poly(x**2 + 1, x, domain='QQ')

        Tr   N)r   Z	compositer   )r%   r   r   r   r}   rN   )rS   r   r   r]   rX   rX   rY   retract?  s    
zPoly.retractc                 C   sh   |du rd||  }}}n
|  |}t|t| }}t| jdrT| j|||}n
t| d| |S )z1Take a continuous subsequence of terms of ``f``. Nr   slice)r   intr   r]   r   r1   r   )rS   r   mnr   rU   rX   rX   rY   r   W  s    

z
Poly.slicec                    s    fdd j j|dD S )aQ  
        Returns all non-zero coefficients from ``f`` in lex order.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x + 3, x).coeffs()
        [1, 2, 3]

        See Also
        ========
        all_coeffs
        coeff_monomial
        nth

        c                    s   g | ]} j j|qS rX   r]   r   r   r   r   rX   rY   r   {  r   zPoly.coeffs.<locals>.<listcomp>r^   )r]   r   rS   r^   rX   r   rY   r   g  s    zPoly.coeffsc                 C   s   | j j|dS )aU  
        Returns all non-zero monomials from ``f`` in lex order.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 2*x*y**2 + x*y + 3*y, x, y).monoms()
        [(2, 0), (1, 2), (1, 1), (0, 1)]

        See Also
        ========
        all_monoms

        r   )r]   r   r   rX   rX   rY   r   }  s    zPoly.monomsc                    s    fdd j j|dD S )ac  
        Returns all non-zero terms from ``f`` in lex order.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 2*x*y**2 + x*y + 3*y, x, y).terms()
        [((2, 0), 1), ((1, 2), 2), ((1, 1), 1), ((0, 1), 3)]

        See Also
        ========
        all_terms

        c                    s"   g | ]\}}| j j|fqS rX   r   r   r   r   r   rX   rY   r     r   zPoly.terms.<locals>.<listcomp>r   )r]   r   r   rX   r   rY   r     s    z
Poly.termsc                    s    fdd j  D S )a  
        Returns all coefficients from a univariate polynomial ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x - 1, x).all_coeffs()
        [1, 0, 2, -1]

        c                    s   g | ]} j j|qS rX   r   r   r   rX   rY   r     r   z#Poly.all_coeffs.<locals>.<listcomp>)r]   
all_coeffsr   rX   r   rY   r     s    zPoly.all_coeffsc                 C   s
   | j  S )a?  
        Returns all monomials from a univariate polynomial ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x - 1, x).all_monoms()
        [(3,), (2,), (1,), (0,)]

        See Also
        ========
        all_terms

        )r]   
all_monomsr   rX   rX   rY   r     s    zPoly.all_monomsc                    s    fdd j  D S )a  
        Returns all terms from a univariate polynomial ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x - 1, x).all_terms()
        [((3,), 1), ((2,), 0), ((1,), 2), ((0,), -1)]

        c                    s"   g | ]\}}| j j|fqS rX   r   r   r   rX   rY   r     r   z"Poly.all_terms.<locals>.<listcomp>)r]   	all_termsr   rX   r   rY   r     s    zPoly.all_termsc                 O   sx   i }|   D ]L\}}|||}t|tr2|\}}n|}|r||vrL|||< qtd| q| j|g|pj| jR i |S )ah  
        Apply a function to all terms of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> def func(k, coeff):
        ...     k = k[0]
        ...     return coeff//10**(2-k)

        >>> Poly(x**2 + 20*x + 400).termwise(func)
        Poly(x**2 + 2*x + 4, x, domain='ZZ')

        z%s monomial was generated twice)r   rK   tupler6   r}   rN   )rS   rW   rN   rn   r   r   r   rU   rX   rX   rY   termwise  s    



zPoly.termwisec                 C   s   t |  S )z
        Returns the number of non-zero terms in ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 2*x - 1).length()
        3

        )rs   r   r   rX   rX   rY   length  s    zPoly.lengthFc                 C   s$   |r| j j|dS | j j|dS dS )a  
        Switch to a ``dict`` representation.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 2*x*y**2 - y, x, y).as_dict()
        {(0, 1): -1, (1, 2): 2, (2, 0): 1}

        r   N)r]   r   rw   )rS   r   r   rX   rX   rY   r     s    zPoly.as_dictc                 C   s   |r| j  S | j  S dS )z%Switch to a ``list`` representation. N)r]   Zto_listZto_sympy_list)rS   r   rX   rX   rY   as_list!  s    
zPoly.as_listc              	   G   s   |s
| j S t|dkrt|d tr|d }t| j}| D ]B\}}z||}W n" tyv   t	d| |f Y q>0 |||< q>t
| j g|R  S )ar  
        Convert a Poly instance to an Expr instance.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = Poly(x**2 + 2*x*y**2 - y, x, y)

        >>> f.as_expr()
        x**2 + 2*x*y**2 - y
        >>> f.as_expr({x: 5})
        10*y**2 - y + 25
        >>> f.as_expr(5, 6)
        379

        rq   r   r   )rz   rs   rK   re   rh   rN   r   r   r   r;   r<   r]   rw   )rS   rN   mappingr   valuer   rX   rX   rY   rQ   (  s    



zPoly.as_exprc                 O   sF   z,t | g|R i |}|js$W dS |W S W n ty@   Y dS 0 dS )a{  Converts ``self`` to a polynomial or returns ``None``.

        >>> from sympy import sin
        >>> from sympy.abc import x, y

        >>> print((x**2 + x*y).as_poly())
        Poly(x**2 + x*y, x, y, domain='ZZ')

        >>> print((x**2 + x*y).as_poly(x, y))
        Poly(x**2 + x*y, x, y, domain='ZZ')

        >>> print((x**2 + sin(y)).as_poly(x, y))
        None

        N)rL   ri   r6   )ry   rN   rn   polyrX   rX   rY   as_polyN  s    
zPoly.as_polyc                 C   s,   t | jdr| j }n
t| d| |S )a  
        Convert algebraic coefficients to rationals.

        Examples
        ========

        >>> from sympy import Poly, I
        >>> from sympy.abc import x

        >>> Poly(x**2 + I*x + 1, x, extension=I).lift()
        Poly(x**4 + 3*x**2 + 1, x, domain='QQ')

        lift)r   r]   r   r1   r   r   rX   rX   rY   r   h  s    
z	Poly.liftc                 C   s4   t | jdr| j \}}n
t| d|| |fS )a+  
        Reduce degree of ``f`` by mapping ``x_i**m`` to ``y_i``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**6*y**2 + x**3 + 1, x, y).deflate()
        ((3, 2), Poly(x**2*y + x + 1, x, y, domain='ZZ'))

        deflate)r   r]   r   r1   r   rS   r   rU   rX   rX   rY   r   }  s    
zPoly.deflatec                 C   sz   | j j}|jr| S |js$td| t| j dr@| j j|d}n
t| d|r\|j| j	 }n| j	|j }| j
|g|R  S )a  
        Inject ground domain generators into ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = Poly(x**2*y + x*y**3 + x*y + 1, x)

        >>> f.inject()
        Poly(x**2*y + x*y**3 + x*y + 1, x, y, domain='ZZ')
        >>> f.inject(front=True)
        Poly(y**3*x + y*x**2 + y*x + 1, y, x, domain='ZZ')

        z Cannot inject generators over %sinjectfront)r]   r   is_Numericalri   r2   r   r   r1   r   rN   ru   )rS   r   r   rU   rN   rX   rX   rY   r     s    
zPoly.injectc                 G   s   | j j}|jstd| t|}| jd| |krJ| j|d d }}n4| j| d |krv| jd|  d }}ntd|j| }t| j dr| j j	||d}n
t
| d| j|g|R  S )a  
        Eject selected generators into the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = Poly(x**2*y + x*y**3 + x*y + 1, x, y)

        >>> f.eject(x)
        Poly(x*y**3 + (x**2 + x)*y + 1, y, domain='ZZ[x]')
        >>> f.eject(y)
        Poly(y*x**2 + (y**3 + y)*x + 1, x, domain='ZZ[y]')

        zCannot eject generators over %sNTFz'can only eject front or back generatorsejectr   )r]   r   r   r2   rs   rN   rb   r   r   r   r1   ru   )rS   rN   r   kZ_gensr   rU   rX   rX   rY   r     s     

z
Poly.ejectc                 C   s4   t | jdr| j \}}n
t| d|| |fS )a  
        Remove GCD of terms from the polynomial ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**6*y**2 + x**3*y, x, y).terms_gcd()
        ((3, 1), Poly(x**3*y + 1, x, y, domain='ZZ'))

        	terms_gcd)r   r]   r  r1   r   r   rX   rX   rY   r    s    
zPoly.terms_gcdc                 C   s.   t | jdr| j|}n
t| d| |S )z
        Add an element of the ground domain to ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x + 1).add_ground(2)
        Poly(x + 3, x, domain='ZZ')

        
add_ground)r   r]   r  r1   r   rS   r   rU   rX   rX   rY   r    s    
zPoly.add_groundc                 C   s.   t | jdr| j|}n
t| d| |S )z
        Subtract an element of the ground domain from ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x + 1).sub_ground(2)
        Poly(x - 1, x, domain='ZZ')

        
sub_ground)r   r]   r  r1   r   r  rX   rX   rY   r    s    
zPoly.sub_groundc                 C   s.   t | jdr| j|}n
t| d| |S )z
        Multiply ``f`` by a an element of the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x + 1).mul_ground(2)
        Poly(2*x + 2, x, domain='ZZ')

        
mul_ground)r   r]   r  r1   r   r  rX   rX   rY   r     s    
zPoly.mul_groundc                 C   s.   t | jdr| j|}n
t| d| |S )aO  
        Quotient of ``f`` by a an element of the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x + 4).quo_ground(2)
        Poly(x + 2, x, domain='ZZ')

        >>> Poly(2*x + 3).quo_ground(2)
        Poly(x + 1, x, domain='ZZ')

        
quo_ground)r   r]   r  r1   r   r  rX   rX   rY   r  5  s    
zPoly.quo_groundc                 C   s.   t | jdr| j|}n
t| d| |S )a  
        Exact quotient of ``f`` by a an element of the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x + 4).exquo_ground(2)
        Poly(x + 2, x, domain='ZZ')

        >>> Poly(2*x + 3).exquo_ground(2)
        Traceback (most recent call last):
        ...
        ExactQuotientFailed: 2 does not divide 3 in ZZ

        exquo_ground)r   r]   r  r1   r   r  rX   rX   rY   r  M  s    
zPoly.exquo_groundc                 C   s,   t | jdr| j }n
t| d| |S )z
        Make all coefficients in ``f`` positive.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).abs()
        Poly(x**2 + 1, x, domain='ZZ')

        abs)r   r]   r	  r1   r   r   rX   rX   rY   r	  g  s    
zPoly.absc                 C   s,   t | jdr| j }n
t| d| |S )a4  
        Negate all coefficients in ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).neg()
        Poly(-x**2 + 1, x, domain='ZZ')

        >>> -Poly(x**2 - 1, x)
        Poly(-x**2 + 1, x, domain='ZZ')

        neg)r   r]   r
  r1   r   r   rX   rX   rY   r
  |  s    
zPoly.negc                 C   sT   t |}|js| |S | |\}}}}t| jdrB||}n
t| d||S )a[  
        Add two polynomials ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).add(Poly(x - 2, x))
        Poly(x**2 + x - 1, x, domain='ZZ')

        >>> Poly(x**2 + 1, x) + Poly(x - 2, x)
        Poly(x**2 + x - 1, x, domain='ZZ')

        r   )r   ri   r  r   r   r]   r   r1   rS   rT   r   r   r   r   rU   rX   rX   rY   r     s    

zPoly.addc                 C   sT   t |}|js| |S | |\}}}}t| jdrB||}n
t| d||S )a`  
        Subtract two polynomials ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).sub(Poly(x - 2, x))
        Poly(x**2 - x + 3, x, domain='ZZ')

        >>> Poly(x**2 + 1, x) - Poly(x - 2, x)
        Poly(x**2 - x + 3, x, domain='ZZ')

        sub)r   ri   r  r   r   r]   r  r1   r  rX   rX   rY   r    s    

zPoly.subc                 C   sT   t |}|js| |S | |\}}}}t| jdrB||}n
t| d||S )ap  
        Multiply two polynomials ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).mul(Poly(x - 2, x))
        Poly(x**3 - 2*x**2 + x - 2, x, domain='ZZ')

        >>> Poly(x**2 + 1, x)*Poly(x - 2, x)
        Poly(x**3 - 2*x**2 + x - 2, x, domain='ZZ')

        r   )r   ri   r  r   r   r]   r   r1   r  rX   rX   rY   r     s    

zPoly.mulc                 C   s,   t | jdr| j }n
t| d| |S )a3  
        Square a polynomial ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x - 2, x).sqr()
        Poly(x**2 - 4*x + 4, x, domain='ZZ')

        >>> Poly(x - 2, x)**2
        Poly(x**2 - 4*x + 4, x, domain='ZZ')

        sqr)r   r]   r  r1   r   r   rX   rX   rY   r    s    
zPoly.sqrc                 C   s6   t |}t| jdr"| j|}n
t| d| |S )aX  
        Raise ``f`` to a non-negative power ``n``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x - 2, x).pow(3)
        Poly(x**3 - 6*x**2 + 12*x - 8, x, domain='ZZ')

        >>> Poly(x - 2, x)**3
        Poly(x**3 - 6*x**2 + 12*x - 8, x, domain='ZZ')

        pow)r   r   r]   r  r1   r   rS   r   rU   rX   rX   rY   r  	  s
    
zPoly.powc                 C   sH   |  |\}}}}t| jdr.||\}}n
t| d||||fS )a#  
        Polynomial pseudo-division of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).pdiv(Poly(2*x - 4, x))
        (Poly(2*x + 4, x, domain='ZZ'), Poly(20, x, domain='ZZ'))

        pdiv)r   r   r]   r  r1   )rS   rT   r   r   r   r   qrrX   rX   rY   r  #  s
    
z	Poly.pdivc                 C   s<   |  |\}}}}t| jdr*||}n
t| d||S )aN  
        Polynomial pseudo-remainder of ``f`` by ``g``.

        Caveat: The function prem(f, g, x) can be safely used to compute
          in Z[x] _only_ subresultant polynomial remainder sequences (prs's).

          To safely compute Euclidean and Sturmian prs's in Z[x]
          employ anyone of the corresponding functions found in
          the module sympy.polys.subresultants_qq_zz. The functions
          in the module with suffix _pg compute prs's in Z[x] employing
          rem(f, g, x), whereas the functions with suffix _amv
          compute prs's in Z[x] employing rem_z(f, g, x).

          The function rem_z(f, g, x) differs from prem(f, g, x) in that
          to compute the remainder polynomials in Z[x] it premultiplies
          the divident times the absolute value of the leading coefficient
          of the divisor raised to the power degree(f, x) - degree(g, x) + 1.


        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).prem(Poly(2*x - 4, x))
        Poly(20, x, domain='ZZ')

        prem)r   r   r]   r  r1   r  rX   rX   rY   r  :  s
    
z	Poly.premc                 C   s<   |  |\}}}}t| jdr*||}n
t| d||S )a  
        Polynomial pseudo-quotient of ``f`` by ``g``.

        See the Caveat note in the function prem(f, g).

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).pquo(Poly(2*x - 4, x))
        Poly(2*x + 4, x, domain='ZZ')

        >>> Poly(x**2 - 1, x).pquo(Poly(2*x - 2, x))
        Poly(2*x + 2, x, domain='ZZ')

        pquo)r   r   r]   r  r1   r  rX   rX   rY   r  a  s
    
z	Poly.pquoc              
   C   sz   |  |\}}}}t| jdrhz||}W qr tyd } z ||  | W Y d}~qrd}~0 0 n
t| d||S )a  
        Polynomial exact pseudo-quotient of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).pexquo(Poly(2*x - 2, x))
        Poly(2*x + 2, x, domain='ZZ')

        >>> Poly(x**2 + 1, x).pexquo(Poly(2*x - 4, x))
        Traceback (most recent call last):
        ...
        ExactQuotientFailed: 2*x - 4 does not divide x**2 + 1

        pexquoN)r   r   r]   r  r8   ru   rQ   r1   )rS   rT   r   r   r   r   rU   excrX   rX   rY   r  }  s    ,
zPoly.pexquoc                 C   s   |  |\}}}}d}|r<|jr<|js<| |  }}d}t| jdrX||\}}	n
t| d|rz| |	  }
}W n t	y   Y n0 |
| }}	||||	fS )a  
        Polynomial division with remainder of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).div(Poly(2*x - 4, x))
        (Poly(1/2*x + 1, x, domain='QQ'), Poly(5, x, domain='QQ'))

        >>> Poly(x**2 + 1, x).div(Poly(2*x - 4, x), auto=False)
        (Poly(0, x, domain='ZZ'), Poly(x**2 + 1, x, domain='ZZ'))

        FTdiv)
r   is_Ringis_Fieldr   r   r]   r  r1   r   r3   )rS   rT   autor   r   r   r   r   r  r  QRrX   rX   rY   r    s    

zPoly.divc           	      C   s   |  |\}}}}d}|r<|jr<|js<| |  }}d}t| jdrT||}n
t| d|rz| }W n t	y   Y n0 ||S )ao  
        Computes the polynomial remainder of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).rem(Poly(2*x - 4, x))
        Poly(5, x, domain='ZZ')

        >>> Poly(x**2 + 1, x).rem(Poly(2*x - 4, x), auto=False)
        Poly(x**2 + 1, x, domain='ZZ')

        FTrem)
r   r  r  r   r   r]   r  r1   r   r3   )	rS   rT   r  r   r   r   r   r   r  rX   rX   rY   r    s    
zPoly.remc           	      C   s   |  |\}}}}d}|r<|jr<|js<| |  }}d}t| jdrT||}n
t| d|rz| }W n t	y   Y n0 ||S )aa  
        Computes polynomial quotient of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).quo(Poly(2*x - 4, x))
        Poly(1/2*x + 1, x, domain='QQ')

        >>> Poly(x**2 - 1, x).quo(Poly(x - 1, x))
        Poly(x + 1, x, domain='ZZ')

        FTquo)
r   r  r  r   r   r]   r  r1   r   r3   )	rS   rT   r  r   r   r   r   r   r  rX   rX   rY   r    s    
zPoly.quoc           
   
   C   s   |  |\}}}}d}|r<|jr<|js<| |  }}d}t| jdrz||}W q ty }	 z |	| 	 |	 W Y d}	~	qd}	~	0 0 n
t
| d|rz| }W n ty   Y n0 ||S )a  
        Computes polynomial exact quotient of ``f`` by ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).exquo(Poly(x - 1, x))
        Poly(x + 1, x, domain='ZZ')

        >>> Poly(x**2 + 1, x).exquo(Poly(2*x - 4, x))
        Traceback (most recent call last):
        ...
        ExactQuotientFailed: 2*x - 4 does not divide x**2 + 1

        FTexquoN)r   r  r  r   r   r]   r  r8   ru   rQ   r1   r   r3   )
rS   rT   r  r   r   r   r   r   r  r  rX   rX   rY   r    s"    ,
z
Poly.exquoc                 C   s   t |trXt| j}| |  kr*|k rDn n|dk r>|| S |S qtd|||f n2z| jt|W S  ty   td| Y n0 dS )z3Returns level associated with the given generator. r   z -%s <= gen < %s expected, got %sz"a valid generator expected, got %sN)rK   r   rs   rN   r6   r   r   r   )rS   r   r   rX   rX   rY   r   7  s    

zPoly._gen_to_levelr   c                 C   s0   |  |}t| jdr"| j|S t| ddS )au  
        Returns degree of ``f`` in ``x_j``.

        The degree of 0 is negative infinity.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + y*x + 1, x, y).degree()
        2
        >>> Poly(x**2 + y*x + y, x, y).degree(y)
        1
        >>> Poly(0, x).degree()
        -oo

        degreeN)r   r   r]   r   r1   )rS   r   r   rX   rX   rY   r   K  s    
zPoly.degreec                 C   s$   t | jdr| j S t| ddS )z
        Returns a list of degrees of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + y*x + 1, x, y).degree_list()
        (2, 1)

        degree_listN)r   r]   r!  r1   r   rX   rX   rY   r!  f  s    
zPoly.degree_listc                 C   s$   t | jdr| j S t| ddS )a  
        Returns the total degree of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + y*x + 1, x, y).total_degree()
        2
        >>> Poly(x + y**5, x, y).total_degree()
        5

        total_degreeN)r   r]   r"  r1   r   rX   rX   rY   r"  y  s    
zPoly.total_degreec                 C   s~   t |tstdt| || jv r8| j|}| j}nt| j}| j|f }t| jdrp| j	| j
||dS t| ddS )a  
        Returns the homogeneous polynomial of ``f``.

        A homogeneous polynomial is a polynomial whose all monomials with
        non-zero coefficients have the same total degree. If you only
        want to check if a polynomial is homogeneous, then use
        :func:`Poly.is_homogeneous`. If you want not only to check if a
        polynomial is homogeneous but also compute its homogeneous order,
        then use :func:`Poly.homogeneous_order`.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> f = Poly(x**5 + 2*x**2*y**2 + 9*x*y**3)
        >>> f.homogenize(z)
        Poly(x**5 + 2*x**2*y**2*z + 9*x*y**3*z, x, y, z, domain='ZZ')

        z``Symbol`` expected, got %s
homogenizer   homogeneous_orderN)rK   r   	TypeErrortyperN   r   rs   r   r]   r   r#  r1   )rS   sr   rN   rX   rX   rY   r#    s    


zPoly.homogenizec                 C   s$   t | jdr| j S t| ddS )a-  
        Returns the homogeneous order of ``f``.

        A homogeneous polynomial is a polynomial whose all monomials with
        non-zero coefficients have the same total degree. This degree is
        the homogeneous order of ``f``. If you only want to check if a
        polynomial is homogeneous, then use :func:`Poly.is_homogeneous`.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = Poly(x**5 + 2*x**3*y**2 + 9*x*y**4)
        >>> f.homogeneous_order()
        5

        r$  N)r   r]   r$  r1   r   rX   rX   rY   r$    s    
zPoly.homogeneous_orderc                 C   sF   |dur|  |d S t| jdr.| j }n
t| d| jj|S )z
        Returns the leading coefficient of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(4*x**3 + 2*x**2 + 3*x, x).LC()
        4

        Nr   LC)r   r   r]   r(  r1   r   r   )rS   r^   rU   rX   rX   rY   r(    s    
zPoly.LCc                 C   s0   t | jdr| j }n
t| d| jj|S )z
        Returns the trailing coefficient of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x**2 + 3*x, x).TC()
        0

        TC)r   r]   r)  r1   r   r   r   rX   rX   rY   r)    s    
zPoly.TCc                 C   s(   t | jdr| |d S t| ddS )z
        Returns the last non-zero coefficient of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 + 2*x**2 + 3*x, x).EC()
        3

        r   ECN)r   r]   r   r1   r   rX   rX   rY   r+    s    zPoly.ECc                 C   s   | j t|| jj S )aE  
        Returns the coefficient of ``monom`` in ``f`` if there, else None.

        Examples
        ========

        >>> from sympy import Poly, exp
        >>> from sympy.abc import x, y

        >>> p = Poly(24*x*y*exp(8) + 23*x, x, y)

        >>> p.coeff_monomial(x)
        23
        >>> p.coeff_monomial(y)
        0
        >>> p.coeff_monomial(x*y)
        24*exp(8)

        Note that ``Expr.coeff()`` behaves differently, collecting terms
        if possible; the Poly must be converted to an Expr to use that
        method, however:

        >>> p.as_expr().coeff(x)
        24*y*exp(8) + 23
        >>> p.as_expr().coeff(y)
        24*x*exp(8)
        >>> p.as_expr().coeff(x*y)
        24*exp(8)

        See Also
        ========
        nth: more efficient query using exponents of the monomial's generators

        )nthr,   rN   	exponents)rS   r   rX   rX   rY   coeff_monomial	  s    #zPoly.coeff_monomialc                 G   sV   t | jdr>t|t| jkr&td| jjttt| }n
t	| d| jj
|S )a.  
        Returns the ``n``-th coefficient of ``f`` where ``N`` are the
        exponents of the generators in the term of interest.

        Examples
        ========

        >>> from sympy import Poly, sqrt
        >>> from sympy.abc import x, y

        >>> Poly(x**3 + 2*x**2 + 3*x, x).nth(2)
        2
        >>> Poly(x**3 + 2*x*y**2 + y**2, x, y).nth(1, 2)
        2
        >>> Poly(4*sqrt(x)*y)
        Poly(4*y*(sqrt(x)), y, sqrt(x), domain='ZZ')
        >>> _.nth(1, 1)
        4

        See Also
        ========
        coeff_monomial

        r,  z,exponent of each generator must be specified)r   r]   rs   rN   r   r,  rh   r   r   r1   r   r   )rS   NrU   rX   rX   rY   r,  .  s    
zPoly.nthrq   c                 C   s   t dd S )NzyEither convert to Expr with `as_expr` method to use Expr's coeff method or else use the `coeff_monomial` method of Polys.rb   )rS   r   r   rightrX   rX   rY   r   P  s    z
Poly.coeffc                 C   s   t | |d | jS )a  
        Returns the leading monomial of ``f``.

        The Leading monomial signifies the monomial having
        the highest power of the principal generator in the
        expression f.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(4*x**2 + 2*x*y**2 + x*y + 3*y, x, y).LM()
        x**2*y**0

        r   r,   r   rN   r   rX   rX   rY   LM\  s    zPoly.LMc                 C   s   t | |d | jS )z
        Returns the last non-zero monomial of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(4*x**2 + 2*x*y**2 + x*y + 3*y, x, y).EM()
        x**0*y**1

        r*  r2  r   rX   rX   rY   EMp  s    zPoly.EMc                 C   s"   |  |d \}}t|| j|fS )a  
        Returns the leading term of ``f``.

        The Leading term signifies the term having
        the highest power of the principal generator in the
        expression f along with its coefficient.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(4*x**2 + 2*x*y**2 + x*y + 3*y, x, y).LT()
        (x**2*y**0, 4)

        r   r   r,   rN   rS   r^   r   r   rX   rX   rY   LT  s    zPoly.LTc                 C   s"   |  |d \}}t|| j|fS )z
        Returns the last non-zero term of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(4*x**2 + 2*x*y**2 + x*y + 3*y, x, y).ET()
        (x**0*y**1, 3)

        r*  r5  r6  rX   rX   rY   ET  s    zPoly.ETc                 C   s0   t | jdr| j }n
t| d| jj|S )z
        Returns maximum norm of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(-x**2 + 2*x - 3, x).max_norm()
        3

        max_norm)r   r]   r9  r1   r   r   r   rX   rX   rY   r9    s    
zPoly.max_normc                 C   s0   t | jdr| j }n
t| d| jj|S )z
        Returns l1 norm of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(-x**2 + 2*x - 3, x).l1_norm()
        6

        l1_norm)r   r]   r:  r1   r   r   r   rX   rX   rY   r:    s    
zPoly.l1_normc                 C   s   | }|j jjstj|fS | }|jr2|j j }t|j drN|j 	 \}}n
t
|d|||| }}|rx|js||fS || fS dS )a  
        Clear denominators, but keep the ground domain.

        Examples
        ========

        >>> from sympy import Poly, S, QQ
        >>> from sympy.abc import x

        >>> f = Poly(x/2 + S(1)/3, x, domain=QQ)

        >>> f.clear_denoms()
        (6, Poly(3*x + 2, x, domain='QQ'))
        >>> f.clear_denoms(convert=True)
        (6, Poly(3*x + 2, x, domain='ZZ'))

        clear_denomsN)r]   r   r  r   Oner   has_assoc_Ringget_ringr   r;  r1   r   r   r   )ry   r   rS   r   r   rU   rX   rX   rY   r;    s    



zPoly.clear_denomsc                 C   sv   | }| |\}}}}||}||}|jr2|js:||fS |jdd\}}|jdd\}}||}||}||fS )a  
        Clear denominators in a rational function ``f/g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = Poly(x**2/y + 1, x)
        >>> g = Poly(x**3 + y, x)

        >>> p, q = f.rat_clear_denoms(g)

        >>> p
        Poly(x**2 + y, x, domain='ZZ[y]')
        >>> q
        Poly(y*x**3 + y**2, x, domain='ZZ[y]')

        Tr   )r   r  r=  r;  r  )ry   rT   rS   r   r   abrX   rX   rY   rat_clear_denoms  s    

zPoly.rat_clear_denomsc                 O   s   | }| ddr"|jjjr"| }t|jdr|sF||jjddS |j}|D ]8}t|t	rh|\}}n
|d }}|t
|||}qP||S t|ddS )a  
        Computes indefinite integral of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 2*x + 1, x).integrate()
        Poly(1/3*x**3 + x**2 + x, x, domain='QQ')

        >>> Poly(x*y**2 + x, x, y).integrate((0, 1), (1, 0))
        Poly(1/2*x**2*y**2 + 1/2*x**2, x, y, domain='QQ')

        r  T	integraterq   r   N)getr]   r   r  r   r   r   rC  rK   r   r   r   r1   )ry   specsrn   rS   r]   specr   r   rX   rX   rY   rC  	  s    



zPoly.integratec                 O   s   | dds"t| g|R i |S t| jdr|sF| | jjddS | j}|D ]8}t|trh|\}}n
|d }}|t|| 	|}qP| |S t
| ddS )aX  
        Computes partial derivative of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + 2*x + 1, x).diff()
        Poly(2*x + 2, x, domain='ZZ')

        >>> Poly(x*y**2 + x, x, y).diff((0, 0), (1, 1))
        Poly(2*x*y, x, y, domain='ZZ')

        evaluateTdiffrq   rD  N)rE  r   r   r]   r   rI  rK   r   r   r   r1   )rS   rF  r   r]   rG  r   r   rX   rX   rY   rI  F	  s    



z	Poly.diffc                 C   sP  | }|du rt |tr<|}| D ]\}}|||}q"|S t |ttfr|}t|t|jkrhtdt	|j|D ]\}}|||}qt|S d| }	}n
|
|}	t|jdst|dz|j||	}
W nt ty@   |std||jjf nFt|g\}\}| ||j}||}|||}|j||	}
Y n0 |j|
|	dS )a  
        Evaluate ``f`` at ``a`` in the given variable.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> Poly(x**2 + 2*x + 3, x).eval(2)
        11

        >>> Poly(2*x*y + 3*x + y + 2, x, y).eval(x, 2)
        Poly(5*y + 8, y, domain='ZZ')

        >>> f = Poly(2*x*y + 3*x + y + 2*z, x, y, z)

        >>> f.eval({x: 2})
        Poly(5*y + 2*z + 6, y, z, domain='ZZ')
        >>> f.eval({x: 2, y: 5})
        Poly(2*z + 31, z, domain='ZZ')
        >>> f.eval({x: 2, y: 5, z: 7})
        45

        >>> f.eval((2, 5))
        Poly(2*z + 31, z, domain='ZZ')
        >>> f(2, 5)
        Poly(2*z + 31, z, domain='ZZ')

        Nztoo many values providedr   r   zCannot evaluate at %s in %sr   )rK   re   r   r   r   rh   rs   rN   r   r   r   r   r]   r1   r3   r2   r   r%   r   Zunify_with_symbolsr   r   r   )ry   r   r@  r  rS   r   r   r   valuesr   rU   Za_domainZ
new_domainrX   rX   rY   r   n	  s:    



z	Poly.evalc                 G   s
   |  |S )az  
        Evaluate ``f`` at the give values.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y, z

        >>> f = Poly(2*x*y + 3*x + y + 2*z, x, y, z)

        >>> f(2)
        Poly(5*y + 2*z + 6, y, z, domain='ZZ')
        >>> f(2, 5)
        Poly(2*z + 31, z, domain='ZZ')
        >>> f(2, 5, 7)
        45

        )r   )rS   rK  rX   rX   rY   __call__	  s    zPoly.__call__c           	      C   sd   |  |\}}}}|r.|jr.| |  }}t| jdrJ||\}}n
t| d||||fS )a  
        Half extended Euclidean algorithm of ``f`` and ``g``.

        Returns ``(s, h)`` such that ``h = gcd(f, g)`` and ``s*f = h (mod g)``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = x**4 - 2*x**3 - 6*x**2 + 12*x + 15
        >>> g = x**3 + x**2 - 4*x - 4

        >>> Poly(f).half_gcdex(Poly(g))
        (Poly(-1/5*x + 3/5, x, domain='QQ'), Poly(x + 1, x, domain='QQ'))

        
half_gcdex)r   r  r   r   r]   rM  r1   )	rS   rT   r  r   r   r   r   r'  hrX   rX   rY   rM  	  s    

zPoly.half_gcdexc           
      C   sl   |  |\}}}}|r.|jr.| |  }}t| jdrL||\}}}	n
t| d||||||	fS )a  
        Extended Euclidean algorithm of ``f`` and ``g``.

        Returns ``(s, t, h)`` such that ``h = gcd(f, g)`` and ``s*f + t*g = h``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = x**4 - 2*x**3 - 6*x**2 + 12*x + 15
        >>> g = x**3 + x**2 - 4*x - 4

        >>> Poly(f).gcdex(Poly(g))
        (Poly(-1/5*x + 3/5, x, domain='QQ'),
         Poly(1/5*x**2 - 6/5*x + 2, x, domain='QQ'),
         Poly(x + 1, x, domain='QQ'))

        gcdex)r   r  r   r   r]   rO  r1   )
rS   rT   r  r   r   r   r   r'  trN  rX   rX   rY   rO  	  s    

z
Poly.gcdexc                 C   sX   |  |\}}}}|r.|jr.| |  }}t| jdrF||}n
t| d||S )a  
        Invert ``f`` modulo ``g`` when possible.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).invert(Poly(2*x - 1, x))
        Poly(-4/3, x, domain='QQ')

        >>> Poly(x**2 - 1, x).invert(Poly(x - 1, x))
        Traceback (most recent call last):
        ...
        NotInvertible: zero divisor

        invert)r   r  r   r   r]   rQ  r1   )rS   rT   r  r   r   r   r   rU   rX   rX   rY   rQ  
  s    

zPoly.invertc                 C   s2   t | jdr| jt|}n
t| d| |S )ad  
        Compute ``f**(-1)`` mod ``x**n``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(1, x).revert(2)
        Poly(1, x, domain='ZZ')

        >>> Poly(1 + x, x).revert(1)
        Poly(1, x, domain='ZZ')

        >>> Poly(x**2 - 2, x).revert(2)
        Traceback (most recent call last):
        ...
        NotReversible: only units are reversible in a ring

        >>> Poly(1/x, x).revert(1)
        Traceback (most recent call last):
        ...
        PolynomialError: 1/x contains an element of the generators set

        revert)r   r]   rR  r   r1   r   r  rX   rX   rY   rR  .
  s    
zPoly.revertc                 C   sB   |  |\}}}}t| jdr*||}n
t| dtt||S )ad  
        Computes the subresultant PRS of ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 1, x).subresultants(Poly(x**2 - 1, x))
        [Poly(x**2 + 1, x, domain='ZZ'),
         Poly(x**2 - 1, x, domain='ZZ'),
         Poly(-2, x, domain='ZZ')]

        subresultants)r   r   r]   rS  r1   rh   r   r  rX   rX   rY   rS  P
  s
    
zPoly.subresultantsc           	      C   sv   |  |\}}}}t| jdrB|r6|j||d\}}qL||}n
t| d|rj||ddtt||fS ||ddS )a  
        Computes the resultant of ``f`` and ``g`` via PRS.

        If includePRS=True, it includes the subresultant PRS in the result.
        Because the PRS is used to calculate the resultant, this is more
        efficient than calling :func:`subresultants` separately.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = Poly(x**2 + 1, x)

        >>> f.resultant(Poly(x**2 - 1, x))
        4
        >>> f.resultant(Poly(x**2 - 1, x), includePRS=True)
        (4, [Poly(x**2 + 1, x, domain='ZZ'), Poly(x**2 - 1, x, domain='ZZ'),
             Poly(-2, x, domain='ZZ')])

        	resultant
includePRSr   rJ  )r   r   r]   rT  r1   rh   r   )	rS   rT   rV  r   r   r   r   rU   r  rX   rX   rY   rT  i
  s    
zPoly.resultantc                 C   s0   t | jdr| j }n
t| d| j|ddS )z
        Computes the discriminant of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + 2*x + 3, x).discriminant()
        -8

        discriminantr   rJ  )r   r]   rW  r1   r   r   rX   rX   rY   rW  
  s    
zPoly.discriminantc                 C   s   ddl m} || |S )a  Compute the *dispersion set* of two polynomials.

        For two polynomials `f(x)` and `g(x)` with `\deg f > 0`
        and `\deg g > 0` the dispersion set `\operatorname{J}(f, g)` is defined as:

        .. math::
            \operatorname{J}(f, g)
            & := \{a \in \mathbb{N}_0 | \gcd(f(x), g(x+a)) \neq 1\} \\
            &  = \{a \in \mathbb{N}_0 | \deg \gcd(f(x), g(x+a)) \geq 1\}

        For a single polynomial one defines `\operatorname{J}(f) := \operatorname{J}(f, f)`.

        Examples
        ========

        >>> from sympy import poly
        >>> from sympy.polys.dispersion import dispersion, dispersionset
        >>> from sympy.abc import x

        Dispersion set and dispersion of a simple polynomial:

        >>> fp = poly((x - 3)*(x + 3), x)
        >>> sorted(dispersionset(fp))
        [0, 6]
        >>> dispersion(fp)
        6

        Note that the definition of the dispersion is not symmetric:

        >>> fp = poly(x**4 - 3*x**2 + 1, x)
        >>> gp = fp.shift(-3)
        >>> sorted(dispersionset(fp, gp))
        [2, 3, 4]
        >>> dispersion(fp, gp)
        4
        >>> sorted(dispersionset(gp, fp))
        []
        >>> dispersion(gp, fp)
        -oo

        Computing the dispersion also works over field extensions:

        >>> from sympy import sqrt
        >>> fp = poly(x**2 + sqrt(5)*x - 1, x, domain='QQ<sqrt(5)>')
        >>> gp = poly(x**2 + (2 + sqrt(5))*x + sqrt(5), x, domain='QQ<sqrt(5)>')
        >>> sorted(dispersionset(fp, gp))
        [2]
        >>> sorted(dispersionset(gp, fp))
        [1, 4]

        We can even perform the computations for polynomials
        having symbolic coefficients:

        >>> from sympy.abc import a
        >>> fp = poly(4*x**4 + (4*a + 8)*x**3 + (a**2 + 6*a + 4)*x**2 + (a**2 + 2*a)*x, x)
        >>> sorted(dispersionset(fp))
        [0, 1]

        See Also
        ========

        dispersion

        References
        ==========

        1. [ManWright94]_
        2. [Koepf98]_
        3. [Abramov71]_
        4. [Man93]_
        r   )dispersionset)sympy.polys.dispersionrX  )rS   rT   rX  rX   rX   rY   rX  
  s    HzPoly.dispersionsetc                 C   s   ddl m} || |S )a  Compute the *dispersion* of polynomials.

        For two polynomials `f(x)` and `g(x)` with `\deg f > 0`
        and `\deg g > 0` the dispersion `\operatorname{dis}(f, g)` is defined as:

        .. math::
            \operatorname{dis}(f, g)
            & := \max\{ J(f,g) \cup \{0\} \} \\
            &  = \max\{ \{a \in \mathbb{N} | \gcd(f(x), g(x+a)) \neq 1\} \cup \{0\} \}

        and for a single polynomial `\operatorname{dis}(f) := \operatorname{dis}(f, f)`.

        Examples
        ========

        >>> from sympy import poly
        >>> from sympy.polys.dispersion import dispersion, dispersionset
        >>> from sympy.abc import x

        Dispersion set and dispersion of a simple polynomial:

        >>> fp = poly((x - 3)*(x + 3), x)
        >>> sorted(dispersionset(fp))
        [0, 6]
        >>> dispersion(fp)
        6

        Note that the definition of the dispersion is not symmetric:

        >>> fp = poly(x**4 - 3*x**2 + 1, x)
        >>> gp = fp.shift(-3)
        >>> sorted(dispersionset(fp, gp))
        [2, 3, 4]
        >>> dispersion(fp, gp)
        4
        >>> sorted(dispersionset(gp, fp))
        []
        >>> dispersion(gp, fp)
        -oo

        Computing the dispersion also works over field extensions:

        >>> from sympy import sqrt
        >>> fp = poly(x**2 + sqrt(5)*x - 1, x, domain='QQ<sqrt(5)>')
        >>> gp = poly(x**2 + (2 + sqrt(5))*x + sqrt(5), x, domain='QQ<sqrt(5)>')
        >>> sorted(dispersionset(fp, gp))
        [2]
        >>> sorted(dispersionset(gp, fp))
        [1, 4]

        We can even perform the computations for polynomials
        having symbolic coefficients:

        >>> from sympy.abc import a
        >>> fp = poly(4*x**4 + (4*a + 8)*x**3 + (a**2 + 6*a + 4)*x**2 + (a**2 + 2*a)*x, x)
        >>> sorted(dispersionset(fp))
        [0, 1]

        See Also
        ========

        dispersionset

        References
        ==========

        1. [ManWright94]_
        2. [Koepf98]_
        3. [Abramov71]_
        4. [Man93]_
        r   )
dispersion)rY  rZ  )rS   rT   rZ  rX   rX   rY   rZ  
  s    HzPoly.dispersionc           	      C   sP   |  |\}}}}t| jdr0||\}}}n
t| d||||||fS )a#  
        Returns the GCD of ``f`` and ``g`` and their cofactors.

        Returns polynomials ``(h, cff, cfg)`` such that ``h = gcd(f, g)``, and
        ``cff = quo(f, h)`` and ``cfg = quo(g, h)`` are, so called, cofactors
        of ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).cofactors(Poly(x**2 - 3*x + 2, x))
        (Poly(x - 1, x, domain='ZZ'),
         Poly(x + 1, x, domain='ZZ'),
         Poly(x - 2, x, domain='ZZ'))

        	cofactors)r   r   r]   r[  r1   )	rS   rT   r   r   r   r   rN  cffcfgrX   rX   rY   r[  9  s
    
zPoly.cofactorsc                 C   s<   |  |\}}}}t| jdr*||}n
t| d||S )a  
        Returns the polynomial GCD of ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).gcd(Poly(x**2 - 3*x + 2, x))
        Poly(x - 1, x, domain='ZZ')

        gcd)r   r   r]   r^  r1   r  rX   rX   rY   r^  V  s
    
zPoly.gcdc                 C   s<   |  |\}}}}t| jdr*||}n
t| d||S )a  
        Returns polynomial LCM of ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 1, x).lcm(Poly(x**2 - 3*x + 2, x))
        Poly(x**3 - 2*x**2 - x + 2, x, domain='ZZ')

        lcm)r   r   r]   r_  r1   r  rX   rX   rY   r_  m  s
    
zPoly.lcmc                 C   s<   | j j|}t| j dr(| j |}n
t| d| |S )a  
        Reduce ``f`` modulo a constant ``p``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**3 + 3*x**2 + 5*x + 7, x).trunc(3)
        Poly(-x**3 - x + 1, x, domain='ZZ')

        trunc)r]   r   r   r   r`  r1   r   )rS   prU   rX   rX   rY   r`    s
    
z
Poly.truncc                 C   sF   | }|r|j jjr| }t|j dr2|j  }n
t|d||S )az  
        Divides all coefficients by ``LC(f)``.

        Examples
        ========

        >>> from sympy import Poly, ZZ
        >>> from sympy.abc import x

        >>> Poly(3*x**2 + 6*x + 9, x, domain=ZZ).monic()
        Poly(x**2 + 2*x + 3, x, domain='QQ')

        >>> Poly(3*x**2 + 4*x + 2, x, domain=ZZ).monic()
        Poly(x**2 + 4/3*x + 2/3, x, domain='QQ')

        monic)r]   r   r  r   r   rb  r1   r   ry   r  rS   rU   rX   rX   rY   rb    s    
z
Poly.monicc                 C   s0   t | jdr| j }n
t| d| jj|S )z
        Returns the GCD of polynomial coefficients.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(6*x**2 + 8*x + 12, x).content()
        2

        content)r   r]   rd  r1   r   r   r   rX   rX   rY   rd    s    
zPoly.contentc                 C   s>   t | jdr| j \}}n
t| d| jj|| |fS )a  
        Returns the content and a primitive form of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**2 + 8*x + 12, x).primitive()
        (2, Poly(x**2 + 4*x + 6, x, domain='ZZ'))

        	primitive)r   r]   re  r1   r   r   r   )rS   contrU   rX   rX   rY   re    s    
zPoly.primitivec                 C   s<   |  |\}}}}t| jdr*||}n
t| d||S )a  
        Computes the functional composition of ``f`` and ``g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + x, x).compose(Poly(x - 1, x))
        Poly(x**2 - x, x, domain='ZZ')

        compose)r   r   r]   rg  r1   r  rX   rX   rY   rg    s
    
zPoly.composec                 C   s2   t | jdr| j }n
t| dtt| j|S )a=  
        Computes a functional decomposition of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**4 + 2*x**3 - x - 1, x, domain='ZZ').decompose()
        [Poly(x**2 - x - 1, x, domain='ZZ'), Poly(x**2 + x, x, domain='ZZ')]

        	decompose)r   r]   rh  r1   rh   r   r   r   rX   rX   rY   rh    s    
zPoly.decomposec                 C   s.   t | jdr| j|}n
t| d| |S )a  
        Efficiently compute Taylor shift ``f(x + a)``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 2*x + 1, x).shift(2)
        Poly(x**2 + 2*x + 1, x, domain='ZZ')

        shift)r   r]   ri  r1   r   )rS   r@  rU   rX   rX   rY   ri    s    
z
Poly.shiftc                 C   s^   | |\}}|  |\}}| |\}}t|jdrJ|j|j|j}n
t|d||S )a3  
        Efficiently evaluate the functional transformation ``q**n * f(p/q)``.


        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 2*x + 1, x).transform(Poly(x + 1, x), Poly(x - 1, x))
        Poly(4, x, domain='ZZ')

        	transform)r   r   r]   rj  r1   r   )rS   ra  r  Pr  r   rU   rX   rX   rY   rj  #  s    
zPoly.transformc                 C   sL   | }|r|j jjr| }t|j dr2|j  }n
t|dtt|j	|S )a  
        Computes the Sturm sequence of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 - 2*x**2 + x - 3, x).sturm()
        [Poly(x**3 - 2*x**2 + x - 3, x, domain='QQ'),
         Poly(3*x**2 - 4*x + 1, x, domain='QQ'),
         Poly(2/9*x + 25/9, x, domain='QQ'),
         Poly(-2079/4, x, domain='QQ')]

        sturm)
r]   r   r  r   r   rl  r1   rh   r   r   rc  rX   rX   rY   rl  =  s    
z
Poly.sturmc                    s4   t  jdr j }n
t d fdd|D S )aI  
        Computes greatest factorial factorization of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = x**5 + 2*x**4 - x**3 - 2*x**2

        >>> Poly(f).gff_list()
        [(Poly(x, x, domain='ZZ'), 1), (Poly(x + 2, x, domain='ZZ'), 4)]

        gff_listc                    s   g | ]\}}  ||fqS rX   r   r   rT   r  r   rX   rY   r   o  r   z!Poly.gff_list.<locals>.<listcomp>)r   r]   rm  r1   r   rX   r   rY   rm  Z  s    
zPoly.gff_listc                 C   s,   t | jdr| j }n
t| d| |S )a  
        Computes the product, ``Norm(f)``, of the conjugates of
        a polynomial ``f`` defined over a number field ``K``.

        Examples
        ========

        >>> from sympy import Poly, sqrt
        >>> from sympy.abc import x

        >>> a, b = sqrt(2), sqrt(3)

        A polynomial over a quadratic extension.
        Two conjugates x - a and x + a.

        >>> f = Poly(x - a, x, extension=a)
        >>> f.norm()
        Poly(x**2 - 2, x, domain='QQ')

        A polynomial over a quartic extension.
        Four conjugates x - a, x - a, x + a and x + a.

        >>> f = Poly(x - a, x, extension=(a, b))
        >>> f.norm()
        Poly(x**4 - 4*x**2 + 4, x, domain='QQ')

        norm)r   r]   rp  r1   r   )rS   r  rX   rX   rY   rp  q  s    
z	Poly.normc                 C   s>   t | jdr| j \}}}n
t| d|| || |fS )af  
        Computes square-free norm of ``f``.

        Returns ``s``, ``f``, ``r``, such that ``g(x) = f(x-sa)`` and
        ``r(x) = Norm(g(x))`` is a square-free polynomial over ``K``,
        where ``a`` is the algebraic extension of the ground domain.

        Examples
        ========

        >>> from sympy import Poly, sqrt
        >>> from sympy.abc import x

        >>> s, f, r = Poly(x**2 + 1, x, extension=[sqrt(3)]).sqf_norm()

        >>> s
        1
        >>> f
        Poly(x**2 - 2*sqrt(3)*x + 4, x, domain='QQ<sqrt(3)>')
        >>> r
        Poly(x**4 - 4*x**2 + 16, x, domain='QQ')

        sqf_norm)r   r]   rq  r1   r   )rS   r'  rT   r  rX   rX   rY   rq    s    
zPoly.sqf_normc                 C   s,   t | jdr| j }n
t| d| |S )z
        Computes square-free part of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**3 - 3*x - 2, x).sqf_part()
        Poly(x**2 - x - 2, x, domain='ZZ')

        sqf_part)r   r]   rr  r1   r   r   rX   rX   rY   rr    s    
zPoly.sqf_partc                    sH   t  jdr j|\}}n
t d jj| fdd|D fS )a   
        Returns a list of square-free factors of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = 2*x**5 + 16*x**4 + 50*x**3 + 76*x**2 + 56*x + 16

        >>> Poly(f).sqf_list()
        (2, [(Poly(x + 1, x, domain='ZZ'), 2),
             (Poly(x + 2, x, domain='ZZ'), 3)])

        >>> Poly(f).sqf_list(all=True)
        (2, [(Poly(1, x, domain='ZZ'), 1),
             (Poly(x + 1, x, domain='ZZ'), 2),
             (Poly(x + 2, x, domain='ZZ'), 3)])

        sqf_listc                    s   g | ]\}}  ||fqS rX   rn  ro  r   rX   rY   r     r   z!Poly.sqf_list.<locals>.<listcomp>)r   r]   rs  r1   r   r   )rS   allr   factorsrX   r   rY   rs    s    
zPoly.sqf_listc                    s6   t  jdr j|}n
t d fdd|D S )a  
        Returns a list of square-free factors of ``f``.

        Examples
        ========

        >>> from sympy import Poly, expand
        >>> from sympy.abc import x

        >>> f = expand(2*(x + 1)**3*x**4)
        >>> f
        2*x**7 + 6*x**6 + 6*x**5 + 2*x**4

        >>> Poly(f).sqf_list_include()
        [(Poly(2, x, domain='ZZ'), 1),
         (Poly(x + 1, x, domain='ZZ'), 3),
         (Poly(x, x, domain='ZZ'), 4)]

        >>> Poly(f).sqf_list_include(all=True)
        [(Poly(2, x, domain='ZZ'), 1),
         (Poly(1, x, domain='ZZ'), 2),
         (Poly(x + 1, x, domain='ZZ'), 3),
         (Poly(x, x, domain='ZZ'), 4)]

        sqf_list_includec                    s   g | ]\}}  ||fqS rX   rn  ro  r   rX   rY   r     r   z)Poly.sqf_list_include.<locals>.<listcomp>)r   r]   rv  r1   )rS   rt  ru  rX   r   rY   rv    s    
zPoly.sqf_list_includec                    sn   t  jdrDz j \}}W qN ty@   tj dfgf Y S 0 n
t d jj| fdd|D fS )a~  
        Returns a list of irreducible factors of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = 2*x**5 + 2*x**4*y + 4*x**3 + 4*x**2*y + 2*x + 2*y

        >>> Poly(f).factor_list()
        (2, [(Poly(x + y, x, y, domain='ZZ'), 1),
             (Poly(x**2 + 1, x, y, domain='ZZ'), 2)])

        factor_listrq   c                    s   g | ]\}}  ||fqS rX   rn  ro  r   rX   rY   r     r   z$Poly.factor_list.<locals>.<listcomp>)	r   r]   rw  r2   r   r<  r1   r   r   )rS   r   ru  rX   r   rY   rw    s    
zPoly.factor_listc                    sV   t  jdr:z j }W qD ty6    dfg Y S 0 n
t d fdd|D S )a  
        Returns a list of irreducible factors of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> f = 2*x**5 + 2*x**4*y + 4*x**3 + 4*x**2*y + 2*x + 2*y

        >>> Poly(f).factor_list_include()
        [(Poly(2*x + 2*y, x, y, domain='ZZ'), 1),
         (Poly(x**2 + 1, x, y, domain='ZZ'), 2)]

        factor_list_includerq   c                    s   g | ]\}}  ||fqS rX   rn  ro  r   rX   rY   r   :  r   z,Poly.factor_list_include.<locals>.<listcomp>)r   r]   rx  r2   r1   )rS   ru  rX   r   rY   rx  !  s    
zPoly.factor_list_includec                 C   s
  |dur"t |}|dkr"td|dur4t |}|durFt |}t| jdrl| jj||||||d}n
t| d|rdd }|stt||S dd	 }	|\}
}tt||
tt|	|fS d
d }|stt||S dd	 }	|\}
}tt||
tt|	|fS dS )a  
        Compute isolating intervals for roots of ``f``.

        For real roots the Vincent-Akritas-Strzebonski (VAS) continued fractions method is used.

        References
        ==========
        .. [#] Alkiviadis G. Akritas and Adam W. Strzebonski: A Comparative Study of Two Real Root
            Isolation Methods . Nonlinear Analysis: Modelling and Control, Vol. 10, No. 4, 297-304, 2005.
        .. [#] Alkiviadis G. Akritas, Adam W. Strzebonski and Panagiotis S. Vigklas: Improving the
            Performance of the Continued Fractions Method Using new Bounds of Positive Roots. Nonlinear
            Analysis: Modelling and Control, Vol. 13, No. 3, 265-279, 2008.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 3, x).intervals()
        [((-2, -1), 1), ((1, 2), 1)]
        >>> Poly(x**2 - 3, x).intervals(eps=1e-2)
        [((-26/15, -19/11), 1), ((19/11, 26/15), 1)]

        Nr   !'eps' must be a positive rational	intervalsrt  epsinfsupfastsqfc                 S   s   | \}}t |t |fS rv   r'   r   )intervalr'  rP  rX   rX   rY   _realh  s    zPoly.intervals.<locals>._realc                 S   s@   | \\}}\}}t |tt |  t |tt |  fS rv   r'   r   r   )	rectangleuvr'  rP  rX   rX   rY   _complexo  s    z Poly.intervals.<locals>._complexc                 S   s$   | \\}}}t |t |f|fS rv   r  )r  r'  rP  r  rX   rX   rY   r  x  s    c                 S   sH   | \\\}}\}}}t |tt |  t |tt |  f|fS rv   r  )r  r  r  r'  rP  r  rX   rX   rY   r    s    )	r'   r   r   r   r]   rz  r1   rh   r   )rS   rt  r|  r}  r~  r  r  rU   r  r  Z	real_partZcomplex_partrX   rX   rY   rz  <  s4    



zPoly.intervalsc           	      C   s   |r| j stdt|t| }}|durJt|}|dkrJtd|dur\t|}n|du rhd}t| jdr| jj|||||d\}}n
t	| dt
|t
|fS )a  
        Refine an isolating interval of a root to the given precision.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 3, x).refine_root(1, 2, eps=1e-2)
        (19/11, 26/15)

        z&only square-free polynomials supportedNr   ry  rq   refine_root)r|  stepsr  )is_sqfr6   r'   r   r   r   r   r]   r  r1   r   )	rS   r'  rP  r|  r  r  	check_sqfr   TrX   rX   rY   r    s    



zPoly.refine_rootc                 C   sL  d\}}|dur^t |}|tju r(d}n6| \}}|sDt|}ntttj||fd }}|durt |}|tju r~d}n6| \}}|st|}ntttj||fd }}|r|rt	| j
dr| j
j||d}n
t| dn^|r |dur |tjf}|r|dur|tjf}t	| j
dr:| j
j||d}n
t| dt|S )a<  
        Return the number of roots of ``f`` in ``[inf, sup]`` interval.

        Examples
        ========

        >>> from sympy import Poly, I
        >>> from sympy.abc import x

        >>> Poly(x**4 - 4, x).count_roots(-3, 3)
        2
        >>> Poly(x**4 - 4, x).count_roots(0, 1 + 3*I)
        1

        )TTNFcount_real_rootsr}  r~  count_complex_roots)r   r   NegativeInfinityas_real_imagr'   r   rh   r   Infinityr   r]   r  r1   r   r  r   )rS   r}  r~  Zinf_realZsup_realreZimcountrX   rX   rY   count_roots  s:    




zPoly.count_rootsc                 C   s   t jjj| ||dS )a  
        Get an indexed root of a polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = Poly(2*x**3 - 7*x**2 + 4*x + 4)

        >>> f.root(0)
        -1/2
        >>> f.root(1)
        2
        >>> f.root(2)
        2
        >>> f.root(3)
        Traceback (most recent call last):
        ...
        IndexError: root index out of [-3, 2] range, got 3

        >>> Poly(x**5 + x + 1).root(0)
        CRootOf(x**3 - x**2 + 1, 0)

        radicals)sympypolysrootoftoolsZrootof)rS   r   r  rX   rX   rY   root  s    z	Poly.rootc                 C   s,   t jjjj| |d}|r|S t|ddS dS )aL  
        Return a list of real roots with multiplicities.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**3 - 7*x**2 + 4*x + 4).real_roots()
        [-1/2, 2, 2]
        >>> Poly(x**3 + x + 1).real_roots()
        [CRootOf(x**3 + x + 1, 0)]

        r  FmultipleN)r  r  r  CRootOf
real_rootsrD   )rS   r  r  ZrealsrX   rX   rY   r  	  s    zPoly.real_rootsc                 C   s,   t jjjj| |d}|r|S t|ddS dS )a  
        Return a list of real and complex roots with multiplicities.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**3 - 7*x**2 + 4*x + 4).all_roots()
        [-1/2, 2, 2]
        >>> Poly(x**3 + x + 1).all_roots()
        [CRootOf(x**3 + x + 1, 0),
         CRootOf(x**3 + x + 1, 1),
         CRootOf(x**3 + x + 1, 2)]

        r  Fr  N)r  r  r  r  	all_rootsrD   )rS   r  r  rootsrX   rX   rY   r     s    zPoly.all_roots   2   c                    s  ddl m | jrtd|  |  dkr.g S | jjtu rNdd |  D }n| jjt	u rdd |  D }t
|   fdd|  D }nLfdd|  D }zd	d |D }W n" ty   td
| jj Y n0 tjj}tj_zz>tj|||d|  d d}tttt|fddd}W nx ty   z>tj|||d|  d d}tttt|fddd}W n$ ty   td|f Y n0 Y n0 W |tj_n
|tj_0 |S )a  
        Compute numerical approximations of roots of ``f``.

        Parameters
        ==========

        n ... the number of digits to calculate
        maxsteps ... the maximum number of iterations to do

        If the accuracy `n` cannot be reached in `maxsteps`, it will raise an
        exception. You need to rerun with higher maxsteps.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 3).nroots(n=15)
        [-1.73205080756888, 1.73205080756888]
        >>> Poly(x**2 - 3).nroots(n=30)
        [-1.73205080756887729352744634151, 1.73205080756887729352744634151]

        r   signz$Cannot compute numerical roots of %sc                 S   s   g | ]}t |qS rX   r   r   r   rX   rX   rY   r   ^  r   zPoly.nroots.<locals>.<listcomp>c                 S   s   g | ]
}|j qS rX   )r  r  rX   rX   rY   r   `  r   c                    s   g | ]}t |  qS rX   r  r  )facrX   rY   r   b  r   c                    s   g | ]}|j  d  qS )r   )r   r  r  r  rX   rY   r   d  s   c                 S   s   g | ]}t j| qS rX   )mpmathZmpcr  rX   rX   rY   r   g  r   z!Numerical domain expected, got %sF
   )maxstepscleanuperrorZ	extraprecc                    s$   | j r
dnd| jt| j  | j fS Nrq   r   imagrealr	  r  r  rX   rY   <lambda>x  r   zPoly.nroots.<locals>.<lambda>keyr  c                    s$   | j r
dnd| jt| j  | j fS r  r  r  r  rX   rY   r    r   z7convergence to root failed; try n < %s or maxsteps > %s)Z$sympy.functions.elementary.complexesr  is_multivariater7   r   r]   r   r(   r   r'   r   r%  r2   r  mpdpsZ	polyrootsrh   r   r   sortedrJ   )rS   r   r  r  r   Zdenomsr  r  rX   )r  r   r  rY   nroots9  s^    


zPoly.nrootsc                 C   sP   | j rtd|  i }|  d D ](\}}|jr"| \}}||| | < q"|S )a  
        Compute roots of ``f`` by factorization in the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**6 - 4*x**4 + 4*x**3 - x**2).ground_roots()
        {0: 2, 1: 2}

        z!Cannot compute ground roots of %srq   )r  r7   rw  	is_linearr   )rS   r  factorr  r@  rA  rX   rX   rY   ground_roots  s    zPoly.ground_rootsc                 C   sr   | j rtdt|}|jr.|dkr.t|}ntd| | j}td}| | j	
|| | ||}|||S )af  
        Construct a polynomial with n-th powers of roots of ``f``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = Poly(x**4 - x**2 + 1)

        >>> f.nth_power_roots_poly(2)
        Poly(x**4 - 2*x**3 + 3*x**2 - 2*x + 1, x, domain='ZZ')
        >>> f.nth_power_roots_poly(3)
        Poly(x**4 + 2*x**2 + 1, x, domain='ZZ')
        >>> f.nth_power_roots_poly(4)
        Poly(x**4 + 2*x**3 + 3*x**2 + 2*x + 1, x, domain='ZZ')
        >>> f.nth_power_roots_poly(12)
        Poly(x**4 - 4*x**3 + 6*x**2 - 4*x + 1, x, domain='ZZ')

        zmust be a univariate polynomialrq   z&'n' must an integer and n >= 1, got %srP  )r  r7   r   
is_Integerr   r   r   r   rT  r   rM   r   )rS   r   r/  r   rP  r  rX   rX   rY   nth_power_roots_poly  s    
zPoly.nth_power_roots_polyc                    s   | j rtd| j }| j |}|d }td| di \}}}}t|}|d |d    fdd}	|	||	| }
}|
j	|j	 d |
j
|j
 d  |k S )a  
        Decide whether two roots of this polynomial are equal.

        Examples
        ========

        >>> from sympy import Poly, cyclotomic_poly, exp, I, pi
        >>> f = Poly(cyclotomic_poly(5))
        >>> r0 = exp(2*I*pi/5)
        >>> indices = [i for i, r in enumerate(f.all_roots()) if f.same_root(r, r0)]
        >>> print(indices)
        [3]

        Raises
        ======

        DomainError
            If the domain of the polynomial is not :ref:`ZZ`, :ref:`QQ`,
            :ref:`RR`, or :ref:`CC`.
        MultivariatePolynomialError
            If the polynomial is not univariate.
        PolynomialError
            If the polynomial is of degree < 2.

        zMust be a univariate polynomial	   rq      c                    s   t t|  dS )NrD  )r   r   r   rD  rX   rY   r    r   z Poly.same_root.<locals>.<lambda>)r  r7   r]   Zmignotte_sep_bound_squaredr   	get_fieldr   r   r   r  r  )rS   r@  rA  Zdom_delta_sqZdelta_sqZeps_sqr  r   r   ZevABrX   rD  rY   	same_root  s    
zPoly.same_rootc                 C   s   |  |\}}}}t|dr,|j||d}n
t| d|s~|jrH| }|\}}	}
}||}||	}	||	 ||
||fS tt||S dS )a  
        Cancel common factors in a rational function ``f/g``.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**2 - 2, x).cancel(Poly(x**2 - 2*x + 1, x))
        (1, Poly(2*x + 2, x, domain='ZZ'), Poly(x - 1, x, domain='ZZ'))

        >>> Poly(2*x**2 - 2, x).cancel(Poly(x**2 - 2*x + 1, x), include=True)
        (Poly(2*x + 2, x, domain='ZZ'), Poly(x - 1, x, domain='ZZ'))

        cancel)includeN)	r   r   r  r1   r=  r>  r   r   r   )rS   rT   r  r   r   r   r   rU   cpcqra  r  rX   rX   rY   r    s    



zPoly.cancelc                 C   s   | j jS )a  
        Returns ``True`` if ``f`` is a zero polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(0, x).is_zero
        True
        >>> Poly(1, x).is_zero
        False

        )r]   is_zeror   rX   rX   rY   r  $  s    zPoly.is_zeroc                 C   s   | j jS )a  
        Returns ``True`` if ``f`` is a unit polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(0, x).is_one
        False
        >>> Poly(1, x).is_one
        True

        )r]   is_oner   rX   rX   rY   r  7  s    zPoly.is_onec                 C   s   | j jS )a   
        Returns ``True`` if ``f`` is a square-free polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 - 2*x + 1, x).is_sqf
        False
        >>> Poly(x**2 - 1, x).is_sqf
        True

        )r]   r  r   rX   rX   rY   r  J  s    zPoly.is_sqfc                 C   s   | j jS )a   
        Returns ``True`` if the leading coefficient of ``f`` is one.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x + 2, x).is_monic
        True
        >>> Poly(2*x + 2, x).is_monic
        False

        )r]   is_monicr   rX   rX   rY   r  ]  s    zPoly.is_monicc                 C   s   | j jS )a;  
        Returns ``True`` if GCD of the coefficients of ``f`` is one.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(2*x**2 + 6*x + 12, x).is_primitive
        False
        >>> Poly(x**2 + 3*x + 6, x).is_primitive
        True

        )r]   is_primitiver   rX   rX   rY   r  p  s    zPoly.is_primitivec                 C   s   | j jS )aJ  
        Returns ``True`` if ``f`` is an element of the ground domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x, x).is_ground
        False
        >>> Poly(2, x).is_ground
        True
        >>> Poly(y, x).is_ground
        True

        )r]   	is_groundr   rX   rX   rY   r    s    zPoly.is_groundc                 C   s   | j jS )a,  
        Returns ``True`` if ``f`` is linear in all its variables.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x + y + 2, x, y).is_linear
        True
        >>> Poly(x*y + 2, x, y).is_linear
        False

        )r]   r  r   rX   rX   rY   r    s    zPoly.is_linearc                 C   s   | j jS )a6  
        Returns ``True`` if ``f`` is quadratic in all its variables.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x*y + 2, x, y).is_quadratic
        True
        >>> Poly(x*y**2 + 2, x, y).is_quadratic
        False

        )r]   is_quadraticr   rX   rX   rY   r    s    zPoly.is_quadraticc                 C   s   | j jS )a%  
        Returns ``True`` if ``f`` is zero or has only one term.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(3*x**2, x).is_monomial
        True
        >>> Poly(3*x**2 + 1, x).is_monomial
        False

        )r]   is_monomialr   rX   rX   rY   r    s    zPoly.is_monomialc                 C   s   | j jS )aZ  
        Returns ``True`` if ``f`` is a homogeneous polynomial.

        A homogeneous polynomial is a polynomial whose all monomials with
        non-zero coefficients have the same total degree. If you want not
        only to check if a polynomial is homogeneous but also compute its
        homogeneous order, then use :func:`Poly.homogeneous_order`.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + x*y, x, y).is_homogeneous
        True
        >>> Poly(x**3 + x*y, x, y).is_homogeneous
        False

        )r]   is_homogeneousr   rX   rX   rY   r    s    zPoly.is_homogeneousc                 C   s   | j jS )aG  
        Returns ``True`` if ``f`` has no factors over its domain.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> Poly(x**2 + x + 1, x, modulus=2).is_irreducible
        True
        >>> Poly(x**2 + 1, x, modulus=2).is_irreducible
        False

        )r]   is_irreducibler   rX   rX   rY   r    s    zPoly.is_irreduciblec                 C   s   t | jdkS )a  
        Returns ``True`` if ``f`` is a univariate polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + x + 1, x).is_univariate
        True
        >>> Poly(x*y**2 + x*y + 1, x, y).is_univariate
        False
        >>> Poly(x*y**2 + x*y + 1, x).is_univariate
        True
        >>> Poly(x**2 + x + 1, x, y).is_univariate
        False

        rq   rs   rN   r   rX   rX   rY   r     s    zPoly.is_univariatec                 C   s   t | jdkS )a  
        Returns ``True`` if ``f`` is a multivariate polynomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x, y

        >>> Poly(x**2 + x + 1, x).is_multivariate
        False
        >>> Poly(x*y**2 + x*y + 1, x, y).is_multivariate
        True
        >>> Poly(x*y**2 + x*y + 1, x).is_multivariate
        False
        >>> Poly(x**2 + x + 1, x, y).is_multivariate
        True

        rq   r  r   rX   rX   rY   r    s    zPoly.is_multivariatec                 C   s   | j jS )a  
        Returns ``True`` if ``f`` is a cyclotomic polnomial.

        Examples
        ========

        >>> from sympy import Poly
        >>> from sympy.abc import x

        >>> f = x**16 + x**14 - x**10 + x**8 - x**6 + x**2 + 1

        >>> Poly(f).is_cyclotomic
        False

        >>> g = x**16 + x**14 - x**10 - x**8 - x**6 + x**2 + 1

        >>> Poly(g).is_cyclotomic
        True

        )r]   is_cyclotomicr   rX   rX   rY   r  *  s    zPoly.is_cyclotomicc                 C   s   |   S rv   )r	  r   rX   rX   rY   __abs__B  s    zPoly.__abs__c                 C   s   |   S rv   )r
  r   rX   rX   rY   __neg__E  s    zPoly.__neg__c                 C   s
   |  |S rv   r   rS   rT   rX   rX   rY   __add__H  s    zPoly.__add__c                 C   s
   | | S rv   r  r  rX   rX   rY   __radd__L  s    zPoly.__radd__c                 C   s
   |  |S rv   r  r  rX   rX   rY   __sub__P  s    zPoly.__sub__c                 C   s
   | | S rv   r  r  rX   rX   rY   __rsub__T  s    zPoly.__rsub__c                 C   s
   |  |S rv   r   r  rX   rX   rY   __mul__X  s    zPoly.__mul__c                 C   s
   | | S rv   r   r  rX   rX   rY   __rmul__\  s    zPoly.__rmul__r   c                 C   s    |j r|dkr| |S tS d S )Nr   )r  r  rO   )rS   r   rX   rX   rY   __pow__`  s    
zPoly.__pow__c                 C   s
   |  |S rv   r  r  rX   rX   rY   
__divmod__g  s    zPoly.__divmod__c                 C   s
   | | S rv   r  r  rX   rX   rY   __rdivmod__k  s    zPoly.__rdivmod__c                 C   s
   |  |S rv   r  r  rX   rX   rY   __mod__o  s    zPoly.__mod__c                 C   s
   | | S rv   r  r  rX   rX   rY   __rmod__s  s    zPoly.__rmod__c                 C   s
   |  |S rv   r  r  rX   rX   rY   __floordiv__w  s    zPoly.__floordiv__c                 C   s
   | | S rv   r  r  rX   rX   rY   __rfloordiv__{  s    zPoly.__rfloordiv__rT   c                 C   s   |   |   S rv   rQ   r  rX   rX   rY   __truediv__  s    zPoly.__truediv__c                 C   s   |  |    S rv   r  r  rX   rX   rY   __rtruediv__  s    zPoly.__rtruediv__otherc              
   C   sv   | | }}|j sFz|j||j| d}W n tttfyD   Y dS 0 |j|jkrVdS |jj|jjkrjdS |j|jkS Nr   F)	ri   r   rN   r   r6   r2   r3   r]   r   )ry   r  rS   rT   rX   rX   rY   __eq__  s    
zPoly.__eq__c                 C   s
   | |k S rv   rX   r  rX   rX   rY   __ne__  s    zPoly.__ne__c                 C   s   | j  S rv   )r  r   rX   rX   rY   __bool__  s    zPoly.__bool__c                 C   s   |s| |kS |  t|S d S rv   )
_strict_eqr   rS   rT   strictrX   rX   rY   eq  s    zPoly.eqc                 C   s   | j ||d S )Nr  )r  r  rX   rX   rY   ne  s    zPoly.nec                 C   s*   t || jo(| j|jko(| jj|jddS NTr  )rK   r   rN   r]   r  r  rX   rX   rY   r    s    zPoly._strict_eq)NN)N)N)N)N)N)N)FF)F)F)T)T)T)T)r   )N)N)rq   F)N)N)N)N)F)NT)T)T)T)F)N)N)T)T)F)F)FNNNFF)NNFF)NN)T)TT)TT)r  r  T)F)F)F)rR   
__module____qualname____doc__	__slots__is_commutativeri   Z_op_priorityrp   classmethodru   propertyrz   rn   r{   r}   r   r   rM   rf   rg   rj   rk   rc   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r_   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   rQ   r   r   r   r   r   r  r  r  r  r  r  r	  r
  r   r  r   r  r  r  r  r  r  r  r  r  r  r   r   r!  r"  r#  r$  r(  r)  r+  r.  r,  r   r3  r4  r7  r8  r9  r:  r;  rB  rC  rI  Z_eval_derivativer   rL  rM  rO  rQ  rR  rS  rT  rW  rX  rZ  r[  r^  r_  r`  rb  rd  re  rg  rh  ri  rj  rl  rm  rp  rq  rr  rs  rv  rw  rx  rz  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r   r  r  r  r  r[   r  r  r  r  r  r  r   rO   r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  __classcell__rX   rX   r   rY   rL   d   s  5











	






3#$"%&%*''%%*"%"''(&K!"%KK
#!L%?P(3%




rL   c                       sV   e Zd ZdZdd Z fddZedd Zede	d	d
 Z
dd Zdd Z  ZS )PurePolyz)Class for representing pure polynomials. c                 C   s   | j fS )z$Allow SymPy to hash Poly instances. )r]   rx   rX   rX   rY   r{     s    zPurePoly._hashable_contentc                    s
   t   S rv   r   rx   r   rX   rY   r     s    zPurePoly.__hash__c                 C   s   | j S )aR  
        Free symbols of a polynomial.

        Examples
        ========

        >>> from sympy import PurePoly
        >>> from sympy.abc import x, y

        >>> PurePoly(x**2 + 1).free_symbols
        set()
        >>> PurePoly(x**2 + y).free_symbols
        set()
        >>> PurePoly(x**2 + y, x).free_symbols
        {y}

        )r   rx   rX   rX   rY   r     s    zPurePoly.free_symbolsr  c              
   C   s   | | }}|j sFz|j||j| d}W n tttfyD   Y dS 0 t|jt|jkr^dS |jj	|jj	krz|jj	
|jj	|j}W n ty   Y dS 0 ||}||}|j|jkS r  )ri   r   rN   r   r6   r2   r3   rs   r]   r   r   r4   r   )ry   r  rS   rT   r   rX   rX   rY   r    s     


zPurePoly.__eq__c                 C   s   t || jo| jj|jddS r  )rK   r   r]   r  r  rX   rX   rY   r    s    zPurePoly._strict_eqc                    s   t |}|jsZz(| jj| j| j| j| jj|fW S  tyX   td| |f Y n0 t| j	t|j	kr~td| |f t
| jtrt
|jtstd| |f | j | j	}| jj|jj|}| j|}|j|}||d f fdd	}||||fS )Nr   c                    sD   |d ur2|d | ||d d   }|s2| | S  j| g|R  S r   r   r   r   rX   rY   r     s
    
zPurePoly._unify.<locals>.per)r   ri   r]   r   r   r   r3   r4   rs   rN   rK   r.   r   r   r   )rS   rT   rN   r   r   r   r   rX   r   rY   r     s"    (	zPurePoly._unify)rR   r  r  r  r{   r   r  r   r   rO   r  r  r   r  rX   rX   r   rY   r    s   

r  c                 O   s   t ||}t| |S r   )r`   ra   _poly_from_exprrz   rN   rn   ro   rX   rX   rY   poly_from_expr  s    r  c                 C   s  | t |  }} t| ts&t||| nJ| jrb| j| |}|j|_|j|_|j	du rZd|_	||fS |j
rp| 
 } t| |\}}|jst||| ttt|  \}}|j}|du rt||d\|_}ntt|j|}ttt||}t||}|j	du r
d|_	||fS )r   NTr   F)r   rK   r
   r9   ri   r   rj   rN   r   r  expandr@   rh   r   r   r%   r   r   re   rL   rf   )rz   ro   origr   r]   r   r   r   rX   rX   rY   r    s2    

r  c                 O   s   t ||}t| |S )(Construct polynomials from expressions. )r`   ra   _parallel_poly_from_expr)exprsrN   rn   ro   rX   rX   rY   parallel_poly_from_expr:  s    r  c                 C   s  ddl m} t| dkr| \}}t|trt|tr|j||}|j||}||\}}|j|_|j	|_	|j
du r~d|_
||g|fS t| g  }} g g  }}d}t|D ]T\}	}
t|
}
t|
tr|
jr||	 q||	 |jr|
 }
nd}| |
 q|rt||| d|r:|D ]}	| |	  | |	< q"t| |\}}|js^t||| d|jD ]}t||rdtdqdg g  }}g }g }|D ]@}ttt|  \}}|| || |t| q|j	}|du rt||d\|_	}ntt|j|}|D ]$}||d|  ||d }qg }t||D ]2\}}ttt||}t||}|| qD|j
du rt||_
||fS )	r
  r   	Piecewiser  NTFz&Piecewise generators do not make senser   )$sympy.functions.elementary.piecewiser  rs   rK   rL   r   rj   r   rN   r   r  rh   r   r   r
   ri   r   r  r9   rQ   rA   r6   r   r   extendr%   r   r   re   rf   bool)r  ro   r  rS   rT   ZorigsZ_exprsZ_polysfailedr   rz   repsr  Zcoeffs_listlengthsr   r   r]   r   r   r   r  r   rX   rX   rY   r  A  sv    










r  c                 C   s   t | } || vr|| |< | S )z7Add a new ``(key, value)`` pair to arguments ``dict``. )re   )rn   r  r   rX   rX   rY   _update_args  s    r  c                 C   s   t | dd} t |ddj}| jr0| }| j}n*| j}|sZ|rLt| \}}nt| |\}}|rn| rhtjS tjS |s| jr||jvrt|  \}}||jvrtjS n4| jst	| j
dkrttd| tt| j
|f ||}t|trt|S tjS )a  
    Return the degree of ``f`` in the given variable.

    The degree of 0 is negative infinity.

    Examples
    ========

    >>> from sympy import degree
    >>> from sympy.abc import x, y

    >>> degree(x**2 + y*x + 1, gen=x)
    2
    >>> degree(x**2 + y*x + 1, gen=y)
    1
    >>> degree(0, x)
    -oo

    See also
    ========

    sympy.polys.polytools.Poly.total_degree
    degree_list
    Tr  rq   z
         A symbolic generator of interest is required for a multivariate
         expression like func = %s, e.g. degree(func, gen = %s) instead of
         degree(func, gen = %s).
        )r   	is_Numberri   rQ   r  r   Zeror  rN   rs   r   r%  rF   nextr   r   rK   r   r   )rS   r   Z
gen_is_Numra  ZisNumr   rU   rX   rX   rY   r     s.    

r   c                 G   sH   t | }|jr| }|jr"d}n| jr2|p0| j}t|| }t|S )a  
    Return the total_degree of ``f`` in the given variables.

    Examples
    ========
    >>> from sympy import total_degree, Poly
    >>> from sympy.abc import x, y

    >>> total_degree(1)
    0
    >>> total_degree(x + x*y)
    2
    >>> total_degree(x + x*y, x)
    1

    If the expression is a Poly and no variables are given
    then the generators of the Poly will be used:

    >>> p = Poly(x + x*y, y)
    >>> total_degree(p)
    1

    To deal with the underlying expression of the Poly, convert
    it to an Expr:

    >>> total_degree(p.as_expr())
    2

    This is done automatically if any variables are given:

    >>> total_degree(p, x)
    1

    See also
    ========
    degree
    r   )r   ri   rQ   r  rN   rL   r"  r   )rS   rN   ra  rvrX   rX   rY   r"    s    (
r"  c              
   O   st   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | }ttt|S )z
    Return a list of degrees of ``f`` in all variables.

    Examples
    ========

    >>> from sympy import degree_list
    >>> from sympy.abc import x, y

    >>> degree_list(x**2 + y*x + 1)
    (2, 1)

    r  r!  rq   N)	r`   allowed_flagsr  r9   r:   r!  r   r   r   )rS   rN   rn   r   ro   r  degreesrX   rX   rY   r!    s    "r!  c              
   O   sl   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 |j|jdS )z
    Return the leading coefficient of ``f``.

    Examples
    ========

    >>> from sympy import LC
    >>> from sympy.abc import x, y

    >>> LC(4*x**2 + 2*x*y**2 + x*y + 3*y)
    4

    r  r(  rq   Nr   )r`   r  r  r9   r:   r(  r^   rS   rN   rn   r   ro   r  rX   rX   rY   r(  5  s    "r(  c              
   O   st   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 |j|jd}| S )z
    Return the leading monomial of ``f``.

    Examples
    ========

    >>> from sympy import LM
    >>> from sympy.abc import x, y

    >>> LM(4*x**2 + 2*x*y**2 + x*y + 3*y)
    x**2

    r  r3  rq   Nr   )r`   r  r  r9   r:   r3  r^   rQ   )rS   rN   rn   r   ro   r  r   rX   rX   rY   r3  N  s    "r3  c              
   O   s|   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 |j|jd\}}||  S )z
    Return the leading term of ``f``.

    Examples
    ========

    >>> from sympy import LT
    >>> from sympy.abc import x, y

    >>> LT(4*x**2 + 2*x*y**2 + x*y + 3*y)
    4*x**2

    r  r7  rq   Nr   )r`   r  r  r9   r:   r7  r^   rQ   )rS   rN   rn   r   ro   r  r   r   rX   rX   rY   r7  h  s    "r7  c           
   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 ||\}}	|js| |	 fS ||	fS dS )z
    Compute polynomial pseudo-division of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import pdiv
    >>> from sympy.abc import x

    >>> pdiv(x**2 + 1, 2*x - 4)
    (2*x + 4, 20)

    r  r  r  N)r`   r  r  r9   r:   r  r  rQ   
rS   rT   rN   rn   r   r   ro   r  r  r  rX   rX   rY   r    s    &"r  c           	   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 ||}|js~| S |S dS )z
    Compute polynomial pseudo-remainder of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import prem
    >>> from sympy.abc import x

    >>> prem(x**2 + 1, 2*x - 4)
    20

    r  r  r  N)r`   r  r  r9   r:   r  r  rQ   	rS   rT   rN   rn   r   r   ro   r  r  rX   rX   rY   r    s    &"
r  c           	   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 z||}W n ty   t| |Y n0 |js| S |S dS )z
    Compute polynomial pseudo-quotient of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import pquo
    >>> from sympy.abc import x

    >>> pquo(x**2 + 1, 2*x - 4)
    2*x + 4
    >>> pquo(x**2 - 1, 2*x - 1)
    2*x + 1

    r  r  r  N)	r`   r  r  r9   r:   r  r8   r  rQ   	rS   rT   rN   rn   r   r   ro   r  r  rX   rX   rY   r    s    &"r  c           	   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 ||}|js~| S |S dS )a_  
    Compute polynomial exact pseudo-quotient of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import pexquo
    >>> from sympy.abc import x

    >>> pexquo(x**2 - 1, 2*x - 2)
    2*x + 2

    >>> pexquo(x**2 + 1, 2*x - 4)
    Traceback (most recent call last):
    ...
    ExactQuotientFailed: 2*x - 4 does not divide x**2 + 1

    r  r  r  N)r`   r  r  r9   r:   r  r  rQ   r   rX   rX   rY   r    s    &"
r  c           
   
   O   s   t |ddg z&t| |fg|R i |\\}}}W n0 tyf } ztdd|W Y d}~n
d}~0 0 |j||jd\}}	|js| |	 fS ||	fS dS )a  
    Compute polynomial division of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import div, ZZ, QQ
    >>> from sympy.abc import x

    >>> div(x**2 + 1, 2*x - 4, domain=ZZ)
    (0, x**2 + 1)
    >>> div(x**2 + 1, 2*x - 4, domain=QQ)
    (x/2 + 1, 5)

    r  r  r  r  Nr  )	r`   r  r  r9   r:   r  r  r  rQ   r  rX   rX   rY   r    s    &"r  c           	   
   O   s   t |ddg z&t| |fg|R i |\\}}}W n0 tyf } ztdd|W Y d}~n
d}~0 0 |j||jd}|js| S |S dS )a  
    Compute polynomial remainder of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import rem, ZZ, QQ
    >>> from sympy.abc import x

    >>> rem(x**2 + 1, 2*x - 4, domain=ZZ)
    x**2 + 1
    >>> rem(x**2 + 1, 2*x - 4, domain=QQ)
    5

    r  r  r  r  Nr!  )	r`   r  r  r9   r:   r  r  r  rQ   r  rX   rX   rY   r  $  s    &"r  c           	   
   O   s   t |ddg z&t| |fg|R i |\\}}}W n0 tyf } ztdd|W Y d}~n
d}~0 0 |j||jd}|js| S |S dS )z
    Compute polynomial quotient of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import quo
    >>> from sympy.abc import x

    >>> quo(x**2 + 1, 2*x - 4)
    x/2 + 1
    >>> quo(x**2 - 1, x - 1)
    x + 1

    r  r  r  r  Nr!  )	r`   r  r  r9   r:   r  r  r  rQ   r   rX   rX   rY   r  D  s    &"r  c           	   
   O   s   t |ddg z&t| |fg|R i |\\}}}W n0 tyf } ztdd|W Y d}~n
d}~0 0 |j||jd}|js| S |S dS )aQ  
    Compute polynomial exact quotient of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import exquo
    >>> from sympy.abc import x

    >>> exquo(x**2 - 1, x - 1)
    x + 1

    >>> exquo(x**2 + 1, 2*x - 4)
    Traceback (most recent call last):
    ...
    ExactQuotientFailed: 2*x - 4 does not divide x**2 + 1

    r  r  r  r  Nr!  )	r`   r  r  r9   r:   r  r  r  rQ   r   rX   rX   rY   r  d  s    &"r  c                 O   s   t |ddg z&t| |fg|R i |\\}}}W n ty } zrt|j\}\}	}
z||	|
\}}W n ty   tdd|Y n"0 |	||	|fW  Y d}~S W Y d}~n
d}~0 0 |j||j
d\}}|js| | fS ||fS dS )aT  
    Half extended Euclidean algorithm of ``f`` and ``g``.

    Returns ``(s, h)`` such that ``h = gcd(f, g)`` and ``s*f = h (mod g)``.

    Examples
    ========

    >>> from sympy import half_gcdex
    >>> from sympy.abc import x

    >>> half_gcdex(x**4 - 2*x**3 - 6*x**2 + 12*x + 15, x**3 + x**2 - 4*x - 4)
    (3/5 - x/5, x + 1)

    r  r  rM  r  Nr!  )r`   r  r  r9   r%   r  rM  rb   r:   r   r  r  rQ   )rS   rT   rN   rn   r   r   ro   r  r   r@  rA  r'  rN  rX   rX   rY   rM    s    &6rM  c                 O   s  t |ddg z&t| |fg|R i |\\}}}W n ty } z|t|j\}\}	}
z||	|
\}}}W n ty   tdd|Y n*0 |	||	||	|fW  Y d}~S W Y d}~n
d}~0 0 |j||j
d\}}}|js | | | fS |||fS dS )aZ  
    Extended Euclidean algorithm of ``f`` and ``g``.

    Returns ``(s, t, h)`` such that ``h = gcd(f, g)`` and ``s*f + t*g = h``.

    Examples
    ========

    >>> from sympy import gcdex
    >>> from sympy.abc import x

    >>> gcdex(x**4 - 2*x**3 - 6*x**2 + 12*x + 15, x**3 + x**2 - 4*x - 4)
    (3/5 - x/5, x**2/5 - 6*x/5 + 2, x + 1)

    r  r  rO  r  Nr!  )r`   r  r  r9   r%   r  rO  rb   r:   r   r  r  rQ   )rS   rT   rN   rn   r   r   ro   r  r   r@  rA  r'  rP  rN  rX   rX   rY   rO    s    &>rO  c                 O   s   t |ddg z&t| |fg|R i |\\}}}W nv ty } z^t|j\}\}	}
z |||	|
W W  Y d}~S  ty   t	dd|Y n0 W Y d}~n
d}~0 0 |j||j
d}|js| S |S dS )a  
    Invert ``f`` modulo ``g`` when possible.

    Examples
    ========

    >>> from sympy import invert, S, mod_inverse
    >>> from sympy.abc import x

    >>> invert(x**2 - 1, 2*x - 1)
    -4/3

    >>> invert(x**2 - 1, x - 1)
    Traceback (most recent call last):
    ...
    NotInvertible: zero divisor

    For more efficient inversion of Rationals,
    use the :obj:`~.mod_inverse` function:

    >>> mod_inverse(3, 5)
    2
    >>> (S(2)/5).invert(S(7)/3)
    5/2

    See Also
    ========

    sympy.core.numbers.mod_inverse

    r  r  NrQ  r  r!  )r`   r  r  r9   r%   r  r   rQ  rb   r:   r  r  rQ   )rS   rT   rN   rn   r   r   ro   r  r   r@  rA  rN  rX   rX   rY   rQ    s    !& (rQ  c           	   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 ||}|jsdd |D S |S dS )z
    Compute subresultant PRS of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import subresultants
    >>> from sympy.abc import x

    >>> subresultants(x**2 + 1, x**2 - 1)
    [x**2 + 1, x**2 - 1, -2]

    r  rS  r  Nc                 S   s   g | ]}|  qS rX   r  r   r  rX   rX   rY   r   #  r   z!subresultants.<locals>.<listcomp>)r`   r  r  r9   r:   rS  r  	rS   rT   rN   rn   r   r   ro   r  rU   rX   rX   rY   rS  
  s    &"
rS  FrU  c             
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 |r~|j||d\}	}
n
||}	|js|r|	 dd |
D fS |	 S |r|	|
fS |	S dS )z
    Compute resultant of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import resultant
    >>> from sympy.abc import x

    >>> resultant(x**2 + 1, x**2 - 1)
    4

    r  rT  r  NrU  c                 S   s   g | ]}|  qS rX   r  r"  rX   rX   rY   r   E  r   zresultant.<locals>.<listcomp>)r`   r  r  r9   r:   rT  r  rQ   )rS   rT   rV  rN   rn   r   r   ro   r  rU   r  rX   rX   rY   rT  (  s    &"
rT  c              
   O   s|   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | }|jst| S |S dS )z
    Compute discriminant of ``f``.

    Examples
    ========

    >>> from sympy import discriminant
    >>> from sympy.abc import x

    >>> discriminant(x**2 + 2*x + 3)
    -8

    r  rW  rq   N)r`   r  r  r9   r:   rW  r  rQ   rS   rN   rn   r   ro   r  rU   rX   rX   rY   rW  M  s    "rW  c                 O   s  t |dg z&t| |fg|R i |\\}}}W n ty } z|t|j\}\}	}
z||	|
\}}}W n ty   tdd|Y n*0 |	||	||	|fW  Y d}~S W Y d}~n
d}~0 0 ||\}}}|j
s| | | fS |||fS dS )a  
    Compute GCD and cofactors of ``f`` and ``g``.

    Returns polynomials ``(h, cff, cfg)`` such that ``h = gcd(f, g)``, and
    ``cff = quo(f, h)`` and ``cfg = quo(g, h)`` are, so called, cofactors
    of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import cofactors
    >>> from sympy.abc import x

    >>> cofactors(x**2 - 1, x**2 - 3*x + 2)
    (x - 1, x + 1, x - 2)

    r  r[  r  N)r`   r  r  r9   r%   r  r[  rb   r:   r   r  rQ   )rS   rT   rN   rn   r   r   ro   r  r   r@  rA  rN  r\  r]  rX   rX   rY   r[  k  s    &>r[  c              
      s  t | } fdd}|| }|dur*|S tdg zt| gR i \}}t| dkrtdd | D r| d   fd	d
| dd D }tdd |D rd}|D ]}	t||	 d }qt | W S W nZ t	y0 }
 z@||
j
}|dur|W  Y d}
~
S tdt| |
W Y d}
~
n
d}
~
0 0 |sR|jsFtjS td|dS |d |dd  }}|D ]}||}|jrl qql|js| S |S dS )z
    Compute GCD of a list of polynomials.

    Examples
    ========

    >>> from sympy import gcd_list
    >>> from sympy.abc import x

    >>> gcd_list([x**3 - 1, x**2 - 1, x**2 - 3*x + 2])
    x - 1

    c                    sl   sh sht | \}}|s|jS |jrh|d |dd   }}|D ]}|||}||r> q^q>||S d S Nr   rq   )r%   r   r   r^  r  r   seqr   ZnumbersrU   numberrn   rN   rX   rY   try_non_polynomial_gcd  s    

z(gcd_list.<locals>.try_non_polynomial_gcdNr  rq   c                 s   s   | ]}|j o|jV  qd S rv   is_algebraicis_irrationalr   r   rX   rX   rY   	<genexpr>  r   zgcd_list.<locals>.<genexpr>r*  c                    s   g | ]} |   qS rX   ratsimpr.  r@  rX   rY   r     r   zgcd_list.<locals>.<listcomp>c                 s   s   | ]}|j V  qd S rv   is_rationalr   frcrX   rX   rY   r/    r   r   gcd_listr   )r   r`   r  r  rs   rt  r_  as_numer_denomr	  r9   r  r:   r  r   r  rL   r^  r  rQ   )r'  rN   rn   r*  rU   r  ro   lstlcr6  r  r   rX   r@  rn   rN   rY   r7    sB    

&

r7  c                 O   sf  t | dr2|dur|f| }t| g|R i |S |du rBtdt|dg zxt| |fg|R i |\\}}}tt| |f\}}|jr|j	r|jr|j	r|| 
 }	|	jrt||	 d  W S W nz tyB }
 z`t|
j\}\}}z ||||W W  Y d}
~
S  ty,   tdd|
Y n0 W Y d}
~
n
d}
~
0 0 ||}|js^| S |S dS )z
    Compute GCD of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import gcd
    >>> from sympy.abc import x

    >>> gcd(x**2 - 1, x**2 - 3*x + 2)
    x - 1

    __iter__Nz2gcd() takes 2 arguments or a sequence of argumentsr  r   r^  r  )r   r7  r%  r`   r  r  r   r   r,  r-  r1  r4  r	  r8  r9   r%   r  r   r^  rb   r:   r  rQ   rS   rT   rN   rn   r   r   ro   r@  rA  r6  r  r   rU   rX   rX   rY   r^    s0    

" (
r^  c              
      s  t | } fdd}|| }|dur*|S tdg zt| gR i \}}t| dkrtdd | D r| d   fd	d
| dd D }tdd |D rd}|D ]}	t||	 d }q | W S W nZ ty, }
 z@||
j	}|dur|W  Y d}
~
S t
dt| |
W Y d}
~
n
d}
~
0 0 |sN|jsBtjS td|dS |d |dd  }}|D ]}||}qh|js| S |S dS )z
    Compute LCM of a list of polynomials.

    Examples
    ========

    >>> from sympy import lcm_list
    >>> from sympy.abc import x

    >>> lcm_list([x**3 - 1, x**2 - 1, x**2 - 3*x + 2])
    x**5 - x**4 - 2*x**3 - x**2 + x + 2

    c                    s^   sZ sZt | \}}|s|jS |jrZ|d |dd   }}|D ]}|||}q>||S d S r%  )r%   r   r   r_  r   r&  r)  rX   rY   try_non_polynomial_lcm.  s    
z(lcm_list.<locals>.try_non_polynomial_lcmNr  rq   c                 s   s   | ]}|j o|jV  qd S rv   r+  r.  rX   rX   rY   r/  I  r   zlcm_list.<locals>.<genexpr>r*  c                    s   g | ]} |   qS rX   r0  r.  r2  rX   rY   r   K  r   zlcm_list.<locals>.<listcomp>c                 s   s   | ]}|j V  qd S rv   r3  r5  rX   rX   rY   r/  L  r   lcm_listr   r   )r   r`   r  r  rs   rt  r_  r8  r9   r  r:   r  r   r<  rL   rQ   )r'  rN   rn   r>  rU   r  ro   r9  r:  r6  r  r   rX   r;  rY   r?    s>    

&r?  c                 O   sb  t | dr2|dur|f| }t| g|R i |S |du rBtdt|dg ztt| |fg|R i |\\}}}tt| |f\}}|jr|j	r|jr|j	r|| 
 }	|	jr||	 d  W S W nz ty> }
 z`t|
j\}\}}z ||||W W  Y d}
~
S  ty(   tdd|
Y n0 W Y d}
~
n
d}
~
0 0 ||}|jsZ| S |S dS )z
    Compute LCM of ``f`` and ``g``.

    Examples
    ========

    >>> from sympy import lcm
    >>> from sympy.abc import x

    >>> lcm(x**2 - 1, x**2 - 3*x + 2)
    x**3 - 2*x**2 - x + 2

    r<  Nz2lcm() takes 2 arguments or a sequence of argumentsr  rq   r_  r  )r   r?  r%  r`   r  r  r   r   r,  r-  r1  r4  r8  r9   r%   r  r   r_  rb   r:   r  rQ   r=  rX   rX   rY   r_  k  s0    

" (
r_  c              
      s  t | }t| tr2t fdd| j| jfD  S t| trJtd| f t| trZ| jr^|S  	ddr| j
 fdd| jD  } d d d< t|gR i  S  d	d
}t dg zt| gR i  \}}W n, ty } z|jW  Y d}~S d}~0 0 | \}	} |jjrd|jjrD| jd
d\}
} |  \}} |jjrj||
 }ntj}tdd t| j|	D  }|dkrtj}|dkr|S |rt|||   S t||  dd \}} t|||  ddS )az  
    Remove GCD of terms from ``f``.

    If the ``deep`` flag is True, then the arguments of ``f`` will have
    terms_gcd applied to them.

    If a fraction is factored out of ``f`` and ``f`` is an Add, then
    an unevaluated Mul will be returned so that automatic simplification
    does not redistribute it. The hint ``clear``, when set to False, can be
    used to prevent such factoring when all coefficients are not fractions.

    Examples
    ========

    >>> from sympy import terms_gcd, cos
    >>> from sympy.abc import x, y
    >>> terms_gcd(x**6*y**2 + x**3*y, x, y)
    x**3*y*(x**3*y + 1)

    The default action of polys routines is to expand the expression
    given to them. terms_gcd follows this behavior:

    >>> terms_gcd((3+3*x)*(x+x*y))
    3*x*(x*y + x + y + 1)

    If this is not desired then the hint ``expand`` can be set to False.
    In this case the expression will be treated as though it were comprised
    of one or more terms:

    >>> terms_gcd((3+3*x)*(x+x*y), expand=False)
    (3*x + 3)*(x*y + x)

    In order to traverse factors of a Mul or the arguments of other
    functions, the ``deep`` hint can be used:

    >>> terms_gcd((3 + 3*x)*(x + x*y), expand=False, deep=True)
    3*x*(x + 1)*(y + 1)
    >>> terms_gcd(cos(x + x*y), deep=True)
    cos(x*(y + 1))

    Rationals are factored out by default:

    >>> terms_gcd(x + y/2)
    (2*x + y)/2

    Only the y-term had a coefficient that was a fraction; if one
    does not want to factor out the 1/2 in cases like this, the
    flag ``clear`` can be set to False:

    >>> terms_gcd(x + y/2, clear=False)
    x + y/2
    >>> terms_gcd(x*y/2 + y**2, clear=False)
    y*(x/2 + y)

    The ``clear`` flag is ignored if all coefficients are fractions:

    >>> terms_gcd(x/3 + y/2, clear=False)
    (2*x + 3*y)/6

    See Also
    ========
    sympy.core.exprtools.gcd_terms, sympy.core.exprtools.factor_terms

    c                 3   s$   | ]}t |gR i  V  qd S rv   r  )r   r'  r)  rX   rY   r/    r   zterms_gcd.<locals>.<genexpr>z5Inequalities cannot be used with terms_gcd. Found: %sdeepFc                    s"   g | ]}t |gR i  qS rX   r@  )r   r@  r)  rX   rY   r     r   zterms_gcd.<locals>.<listcomp>r  clearTr  Nr?  c                 S   s   g | ]\}}|| qS rX   rX   )r   r   r   rX   rX   rY   r     r   rq   )rB  ) r   rK   r   lhsrhsr   r%  r   is_AtomrE  rW   rn   popr  r`   r  r  r9   rz   r   r  r  r;  re  r   r<  r   r   rN   r   rQ   Zas_coeff_Mul)rS   rN   rn   r	  ru   rB  r   ro   r  r   denomr   termrX   r)  rY   r    sF    C
 







r  c              
   O   s   t |ddg zt| g|R i |\}}W n0 ty^ } ztdd|W Y d}~n
d}~0 0 |t|}|js|| S |S dS )z
    Reduce ``f`` modulo a constant ``p``.

    Examples
    ========

    >>> from sympy import trunc
    >>> from sympy.abc import x

    >>> trunc(2*x**3 + 3*x**2 + 5*x + 7, 3)
    -x**3 - x + 1

    r  r  r`  rq   N)	r`   r  r  r9   r:   r`  r   r  rQ   )rS   ra  rN   rn   r   ro   r  rU   rX   rX   rY   r`    s    "r`  c              
   O   s   t |ddg zt| g|R i |\}}W n0 ty^ } ztdd|W Y d}~n
d}~0 0 |j|jd}|js|| S |S dS )z
    Divide all coefficients of ``f`` by ``LC(f)``.

    Examples
    ========

    >>> from sympy import monic
    >>> from sympy.abc import x

    >>> monic(3*x**2 + 4*x + 2)
    x**2 + 4*x/3 + 2/3

    r  r  rb  rq   Nr!  )	r`   r  r  r9   r:   rb  r  r  rQ   r$  rX   rX   rY   rb  2  s    "rb  c              
   O   sf   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | S )z
    Compute GCD of coefficients of ``f``.

    Examples
    ========

    >>> from sympy import content
    >>> from sympy.abc import x

    >>> content(6*x**2 + 8*x + 12)
    2

    r  rd  rq   N)r`   r  r  r9   r:   rd  r  rX   rX   rY   rd  P  s    "rd  c              
   O   s   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | \}}|js||| fS ||fS dS )a  
    Compute content and the primitive form of ``f``.

    Examples
    ========

    >>> from sympy.polys.polytools import primitive
    >>> from sympy.abc import x

    >>> primitive(6*x**2 + 8*x + 12)
    (2, 3*x**2 + 4*x + 6)

    >>> eq = (2 + 2*x)*x + 2

    Expansion is performed by default:

    >>> primitive(eq)
    (2, x**2 + x + 1)

    Set ``expand`` to False to shut this off. Note that the
    extraction will not be recursive; use the as_content_primitive method
    for recursive, non-destructive Rational extraction.

    >>> primitive(eq, expand=False)
    (1, x*(2*x + 2) + 2)

    >>> eq.as_content_primitive()
    (2, x*(x + 1) + 1)

    r  re  rq   N)r`   r  r  r9   r:   re  r  rQ   )rS   rN   rn   r   ro   r  rf  rU   rX   rX   rY   re  i  s     "re  c           	   
   O   s   t |dg z&t| |fg|R i |\\}}}W n0 tyd } ztdd|W Y d}~n
d}~0 0 ||}|js~| S |S dS )z
    Compute functional composition ``f(g)``.

    Examples
    ========

    >>> from sympy import compose
    >>> from sympy.abc import x

    >>> compose(x**2 + x, x - 1)
    x**2 - x

    r  rg  r  N)r`   r  r  r9   r:   rg  r  rQ   r#  rX   rX   rY   rg    s    &"
rg  c              
   O   s   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | }|jszdd |D S |S dS )z
    Compute functional decomposition of ``f``.

    Examples
    ========

    >>> from sympy import decompose
    >>> from sympy.abc import x

    >>> decompose(x**4 + 2*x**3 - x - 1)
    [x**2 - x - 1, x**2 + x]

    r  rh  rq   Nc                 S   s   g | ]}|  qS rX   r  r"  rX   rX   rY   r     r   zdecompose.<locals>.<listcomp>)r`   r  r  r9   r:   rh  r  r$  rX   rX   rY   rh    s    "rh  c              
   O   s   t |ddg zt| g|R i |\}}W n0 ty^ } ztdd|W Y d}~n
d}~0 0 |j|jd}|jsdd |D S |S dS )	z
    Compute Sturm sequence of ``f``.

    Examples
    ========

    >>> from sympy import sturm
    >>> from sympy.abc import x

    >>> sturm(x**3 - 2*x**2 + x - 3)
    [x**3 - 2*x**2 + x - 3, 3*x**2 - 4*x + 1, 2*x/9 + 25/9, -2079/4]

    r  r  rl  rq   Nr!  c                 S   s   g | ]}|  qS rX   r  r"  rX   rX   rY   r     r   zsturm.<locals>.<listcomp>)r`   r  r  r9   r:   rl  r  r  r$  rX   rX   rY   rl    s    "rl  c              
   O   s   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | }|jszdd |D S |S dS )a&  
    Compute a list of greatest factorial factors of ``f``.

    Note that the input to ff() and rf() should be Poly instances to use the
    definitions here.

    Examples
    ========

    >>> from sympy import gff_list, ff, Poly
    >>> from sympy.abc import x

    >>> f = Poly(x**5 + 2*x**4 - x**3 - 2*x**2, x)

    >>> gff_list(f)
    [(Poly(x, x, domain='ZZ'), 1), (Poly(x + 2, x, domain='ZZ'), 4)]

    >>> (ff(Poly(x), 1)*ff(Poly(x + 2), 4)) == f
    True

    >>> f = Poly(x**12 + 6*x**11 - 11*x**10 - 56*x**9 + 220*x**8 + 208*x**7 -         1401*x**6 + 1090*x**5 + 2715*x**4 - 6720*x**3 - 1092*x**2 + 5040*x, x)

    >>> gff_list(f)
    [(Poly(x**3 + 7, x, domain='ZZ'), 2), (Poly(x**2 + 5*x, x, domain='ZZ'), 3)]

    >>> ff(Poly(x**3 + 7, x), 2)*ff(Poly(x**2 + 5*x, x), 3) == f
    True

    r  rm  rq   Nc                 S   s   g | ]\}}|  |fqS rX   r  ro  rX   rX   rY   r     r   zgff_list.<locals>.<listcomp>)r`   r  r  r9   r:   rm  r  )rS   rN   rn   r   ro   r  ru  rX   rX   rY   rm    s     "rm  c                 O   s   t ddS )z3Compute greatest factorial factorization of ``f``. zsymbolic falling factorialNr0  rS   rN   rn   rX   rX   rY   gff   s    rJ  c           	   
   O   s   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | \}}}|jst|| | fS t|||fS dS )a  
    Compute square-free norm of ``f``.

    Returns ``s``, ``f``, ``r``, such that ``g(x) = f(x-sa)`` and
    ``r(x) = Norm(g(x))`` is a square-free polynomial over ``K``,
    where ``a`` is the algebraic extension of the ground domain.

    Examples
    ========

    >>> from sympy import sqf_norm, sqrt
    >>> from sympy.abc import x

    >>> sqf_norm(x**2 + 1, extension=[sqrt(3)])
    (1, x**2 - 2*sqrt(3)*x + 4, x**4 - 4*x**2 + 16)

    r  rq  rq   N)	r`   r  r  r9   r:   rq  r  r   rQ   )	rS   rN   rn   r   ro   r  r'  rT   r  rX   rX   rY   rq  &  s    "rq  c              
   O   s|   t |dg zt| g|R i |\}}W n0 ty\ } ztdd|W Y d}~n
d}~0 0 | }|jst| S |S dS )z
    Compute square-free part of ``f``.

    Examples
    ========

    >>> from sympy import sqf_part
    >>> from sympy.abc import x

    >>> sqf_part(x**3 - 3*x - 2)
    x**2 - x - 2

    r  rr  rq   N)r`   r  r  r9   r:   rr  r  rQ   r$  rX   rX   rY   rr  H  s    "rr  c                 C   s&   |dkrdd }ndd }t | |dS )z&Sort a list of ``(expr, exp)`` pairs. r  c                 S   s.   | \}}|j j }|t|t|jt|j|fS rv   r]   rs   rN   rd   r   rt   r   expr]   rX   rX   rY   r  i  s    z_sorted_factors.<locals>.keyc                 S   s.   | \}}|j j }t|t|j|t|j|fS rv   rK  rL  rX   rX   rY   r  n  s    r  )r  )ru  methodr  rX   rX   rY   _sorted_factorsf  s    
rO  c                 C   s   t dd | D  S )z*Multiply a list of ``(expr, exp)`` pairs. c                 S   s   g | ]\}}|  | qS rX   r  r   rS   r  rX   rX   rY   r   x  r   z$_factors_product.<locals>.<listcomp>)r   ru  rX   rX   rY   _factors_productv  s    rR  c                    s  t jg  }dd t| D }|D ]}|jsBt|trNt|rN||9 }q$nV|jr|j	t j
kr|j\} |jr jr||9 }q$|jr| f q$n|t j } zt||\}}W n4 ty }	 z|	j f W Y d}	~	q$d}	~	0 0 t||d }
|
 \}}|t jurN jr&||  9 }n(|jr>| f n||t jf  t ju rf| q$ jr fdd|D  q$g }|D ]8\}}| jr||  f n|||f qt| f q$|dkrfdddd	 D D |fS )
z.Helper function for :func:`_symbolic_factor`. c                 S   s"   g | ]}t |d r| n|qS )_eval_factor)r   rS  r   r   rX   rX   rY   r     s   z)_symbolic_factor_list.<locals>.<listcomp>NZ_listc                    s   g | ]\}}||  fqS rX   rX   rP  )rM  rX   rY   r     r   r  c                    s(   g | ]  t t fd dD  fqS )c                 3   s   | ]\}}| kr|V  qd S rv   rX   )r   rS   r   r  rX   rY   r/    r   z3_symbolic_factor_list.<locals>.<listcomp>.<genexpr>)r   r   )r   rQ  rU  rY   r     s   c                 S   s   h | ]\}}|qS rX   rX   )r   r   r   rX   rX   rY   	<setcomp>  r   z(_symbolic_factor_list.<locals>.<setcomp>)r   r<  r   	make_argsr  rK   r   r   is_PowbaseZExp1rn   r   r  r9   rz   rP   r  Zis_positiver  
is_integerrQ   rR  )rz   ro   rN  r   rn   argrY  r   r   r  rW   Z_coeffZ_factorsr  rS   r  rX   )rM  ru  rY   _symbolic_factor_list{  sX    

&


r\  c                    s   t | trFt| dr|  S tt| d d \}}t|t|S t| drl| j fdd| j	D  S t| dr| 
 fdd| D S | S d	S )
z%Helper function for :func:`_factor`. rS  fraction)r]  rn   c                    s   g | ]}t | qS rX   _symbolic_factorr   r[  rN  ro   rX   rY   r     r   z$_symbolic_factor.<locals>.<listcomp>r<  c                    s   g | ]}t | qS rX   r^  r`  ra  rX   rY   r     r   N)rK   r   r   rS  r\  rB   r   rR  rW   rn   r   )rz   ro   rN  r   ru  rX   ra  rY   r_    s    



r_  c                 C   sP  t |ddg t ||}t| } t| ttfr@t| trJ| d }}nt|  \}}t	|||\}}t	|||\}	}
|
r|j
std|  |tdd}||
fD ]:}t|D ],\}\}}|jst||\}}||f||< qqt||}t|
|}
|jsdd |D }d	d |
D }
||	 }|j
s4||fS |||
fS ntd|  d
S )z>Helper function for :func:`sqf_list` and :func:`factor_list`. fracr  rq   za polynomial expected, got %sT)r  c                 S   s   g | ]\}}|  |fqS rX   r  rP  rX   rX   rY   r     r   z(_generic_factor_list.<locals>.<listcomp>c                 S   s   g | ]\}}|  |fqS rX   r  rP  rX   rX   rY   r     r   N)r`   r  ra   r   rK   r   rL   rB   r8  r\  rb  r6   clonere   r   ri   r  rO  r  )rz   rN   rn   rN  ro   ZnumerrG  r  fpr  ZfqZ_optru  r   rS   r  r   r   rX   rX   rY   _generic_factor_list  s6    



re  c                 C   s<   | dd}t|g  t||}||d< tt| ||S )z4Helper function for :func:`sqf` and :func:`factor`. r]  T)rF  r`   r  ra   r_  r   )rz   rN   rn   rN  r]  ro   rX   rX   rY   _generic_factor  s
    rf  c                    s   ddl m  d fdd	}d fdd	}dd	 }|  jr|| r|  }|| |}|rp|d |d
 d|d fS || |}|rdd|d |d
 fS dS )a!  
    try to transform a polynomial to have rational coefficients

    try to find a transformation ``x = alpha*y``

    ``f(x) = lc*alpha**n * g(y)`` where ``g`` is a polynomial with
    rational coefficients, ``lc`` the leading coefficient.

    If this fails, try ``x = y + beta``
    ``f(x) = g(y)``

    Returns ``None`` if ``g`` not found;
    ``(lc, alpha, None, g)`` in case of rescaling
    ``(None, None, beta, g)`` in case of translation

    Notes
    =====

    Currently it transforms only polynomials without roots larger than 2.

    Examples
    ========

    >>> from sympy import sqrt, Poly, simplify
    >>> from sympy.polys.polytools import to_rational_coeffs
    >>> from sympy.abc import x
    >>> p = Poly(((x**2-1)*(x-2)).subs({x:x*(1 + sqrt(2))}), x, domain='EX')
    >>> lc, r, _, g = to_rational_coeffs(p)
    >>> lc, r
    (7 + 5*sqrt(2), 2 - 2*sqrt(2))
    >>> g
    Poly(x**3 + x**2 - 1/4*x - 1/4, x, domain='QQ')
    >>> r1 = simplify(1/r)
    >>> Poly(lc*r**3*(g.as_expr()).subs({x:x*r1}), x, domain='EX') == p
    True

    r   simplifyNc                    s@  t | jdkr| jd js"d| fS |  }|  }|p<| }| dd } fdd|D }t |dkr<|d r< |d |d  }g }tt |D ]2} || ||d   }|js q<|	| q d| }	| jd }
|
| g}td|d D ]"}|	||d  |
||    qt
| } t| } ||	| fS dS )a$  
        try rescaling ``x -> alpha*x`` to convert f to a polynomial
        with rational coefficients.
        Returns ``alpha, f``; if the rescaling is successful,
        ``alpha`` is the rescaling factor, and ``f`` is the rescaled
        polynomial; else ``alpha`` is ``None``.
        rq   r   Nc                    s   g | ]} |qS rX   rX   )r   coeffxrg  rX   rY   r   /  r   z<to_rational_coeffs.<locals>._try_rescale.<locals>.<listcomp>r*  )rs   rN   rE  r   r(  rb  r   r   r4  r   r   rL   )rS   f1r   r:  r   Z
rescale1_xZcoeffs1r   ri  Z	rescale_xr   r  rg  rX   rY   _try_rescale!  s0    

 
z(to_rational_coeffs.<locals>._try_rescalec           	         s   t | jdkr| jd js"d| fS |  }|p4| }| dd } |d }|jr|jst|j	dd dd\}}|j
|  | }||}||fS dS )a+  
        try translating ``x -> x + alpha`` to convert f to a polynomial
        with rational coefficients.
        Returns ``alpha, f``; if the translating is successful,
        ``alpha`` is the translating factor, and ``f`` is the shifted
        polynomial; else ``alpha`` is ``None``.
        rq   r   Nc                 S   s
   | j du S NTr3  )zrX   rX   rY   r  S  r   z<to_rational_coeffs.<locals>._try_translate.<locals>.<lambda>Tbinary)rs   rN   rE  r   rb  r   is_Addr4  rI   rn   rW   ri  )	rS   rk  r   r   r   ZratZnonratalphaf2rg  rX   rY   _try_translateC  s    

z*to_rational_coeffs.<locals>._try_translatec                 S   sp   |   }d}|D ]Z}t|D ]J}t|j}dd | D }|sDqt|dkrTd}t|dkr  dS qq|S )zS
        Return True if ``f`` is a sum with square roots but no other root
        Fc                 S   s,   g | ]$\}}|j r|jr|jd kr|jqS )r  )r   Zis_Rationalr  )r   rA  ZwxrX   rX   rY   r   b  s   zAto_rational_coeffs.<locals>._has_square_roots.<locals>.<listcomp>r  T)r   r   rW  r   ru  r   minmax)ra  r   Zhas_sqr   r   rS   r  rX   rX   rY   _has_square_rootsY  s    
z-to_rational_coeffs.<locals>._has_square_rootsrq   r  )N)N)sympy.simplify.simplifyrh  r   r   rb  )rS   rl  rt  rw  rk  r  rX   rg  rY   to_rational_coeffs  s    &"

ry  c              	   C   s  ddl m} t| |dd}| }t|}|s2dS |\}}}}	t|	 }
|r||
d | ||  }|d| }g }|
dd d D ],}|||d ||| i|d f qnF|
d }g }|
dd d D ](}||d ||| i|d f q||fS )a  
    helper function to factor polynomial using to_rational_coeffs

    Examples
    ========

    >>> from sympy.polys.polytools import _torational_factor_list
    >>> from sympy.abc import x
    >>> from sympy import sqrt, expand, Mul
    >>> p = expand(((x**2-1)*(x-2)).subs({x:x*(1 + sqrt(2))}))
    >>> factors = _torational_factor_list(p, x); factors
    (-2, [(-x*(1 + sqrt(2))/2 + 1, 1), (-x*(1 + sqrt(2)) - 1, 1), (-x*(1 + sqrt(2)) + 1, 1)])
    >>> expand(factors[0]*Mul(*[z[0] for z in factors[1]])) == p
    True
    >>> p = expand(((x**2-1)*(x-2)).subs({x:x + sqrt(2)}))
    >>> factors = _torational_factor_list(p, x); factors
    (1, [(x - 2 + sqrt(2), 1), (x - 1 + sqrt(2), 1), (x + 1 + sqrt(2), 1)])
    >>> expand(factors[0]*Mul(*[z[0] for z in factors[1]])) == p
    True

    r   rg  ZEXr   Nrq   )	rx  rh  rL   r   ry  rw  rQ   r   r   )ra  r   rh  p1r   resr:  r  rP  rT   ru  r   Zr1r@  rn  rX   rX   rY   _torational_factor_listx  s&    ,&r|  c                 O   s   t | ||ddS )z
    Compute a list of square-free factors of ``f``.

    Examples
    ========

    >>> from sympy import sqf_list
    >>> from sympy.abc import x

    >>> sqf_list(2*x**5 + 16*x**4 + 50*x**3 + 76*x**2 + 56*x + 16)
    (2, [(x + 1, 2), (x + 2, 3)])

    r  rN  re  rI  rX   rX   rY   rs    s    rs  c                 O   s   t | ||ddS )z
    Compute square-free factorization of ``f``.

    Examples
    ========

    >>> from sympy import sqf
    >>> from sympy.abc import x

    >>> sqf(2*x**5 + 16*x**4 + 50*x**3 + 76*x**2 + 56*x + 16)
    2*(x + 1)**2*(x + 2)**3

    r  r}  )rf  rI  rX   rX   rY   r    s    r  c                 O   s   t | ||ddS )a  
    Compute a list of irreducible factors of ``f``.

    Examples
    ========

    >>> from sympy import factor_list
    >>> from sympy.abc import x, y

    >>> factor_list(2*x**5 + 2*x**4*y + 4*x**3 + 4*x**2*y + 2*x + 2*y)
    (2, [(x + y, 1), (x**2 + 1, 2)])

    r  r}  r~  rI  rX   rX   rY   rw    s    rw  )rA  c          
   
      s   t | } |rz fdd}t| |} i }| tt}|D ]6}t|gR i  }|js^|jr8||kr8|||< q8| |S zt	|  ddW S  t
y }	 z.| jst| W  Y d}	~	S t
|	W Y d}	~	n
d}	~	0 0 dS )a  
    Compute the factorization of expression, ``f``, into irreducibles. (To
    factor an integer into primes, use ``factorint``.)

    There two modes implemented: symbolic and formal. If ``f`` is not an
    instance of :class:`Poly` and generators are not specified, then the
    former mode is used. Otherwise, the formal mode is used.

    In symbolic mode, :func:`factor` will traverse the expression tree and
    factor its components without any prior expansion, unless an instance
    of :class:`~.Add` is encountered (in this case formal factorization is
    used). This way :func:`factor` can handle large or symbolic exponents.

    By default, the factorization is computed over the rationals. To factor
    over other domain, e.g. an algebraic or finite field, use appropriate
    options: ``extension``, ``modulus`` or ``domain``.

    Examples
    ========

    >>> from sympy import factor, sqrt, exp
    >>> from sympy.abc import x, y

    >>> factor(2*x**5 + 2*x**4*y + 4*x**3 + 4*x**2*y + 2*x + 2*y)
    2*(x + y)*(x**2 + 1)**2

    >>> factor(x**2 + 1)
    x**2 + 1
    >>> factor(x**2 + 1, modulus=2)
    (x + 1)**2
    >>> factor(x**2 + 1, gaussian=True)
    (x - I)*(x + I)

    >>> factor(x**2 - 2, extension=sqrt(2))
    (x - sqrt(2))*(x + sqrt(2))

    >>> factor((x**2 - 1)/(x**2 + 4*x + 4))
    (x - 1)*(x + 1)/(x + 2)**2
    >>> factor((x**2 + 4*x + 4)**10000000*(x**2 + 1))
    (x + 2)**20000000*(x**2 + 1)

    By default, factor deals with an expression as a whole:

    >>> eq = 2**(x**2 + 2*x + 1)
    >>> factor(eq)
    2**(x**2 + 2*x + 1)

    If the ``deep`` flag is True then subexpressions will
    be factored:

    >>> factor(eq, deep=True)
    2**((x + 1)**2)

    If the ``fraction`` flag is False then rational expressions
    will not be combined. By default it is True.

    >>> factor(5*x + 3*exp(2 - 7*x), deep=True)
    (5*x*exp(7*x) + 3*exp(2))*exp(-7*x)
    >>> factor(5*x + 3*exp(2 - 7*x), deep=True, fraction=False)
    5*x + 3*exp(2)*exp(-7*x)

    See Also
    ========
    sympy.ntheory.factor_.factorint

    c                    s*   t | gR i  }|js"|jr&|S | S )zS
            Factor, but avoid changing the expression when unable to.
            )r  is_MulrX  )rz   r  r)  rX   rY   _try_factor   s    zfactor.<locals>._try_factorr  r}  N)r   r"   Zatomsr   r   r  r  rX  xreplacerf  r6   r   r   )
rS   rA  rN   rn   r  ZpartialsZmuladdra  r  msgrX   r)  rY   r    s"    D	


r  c              	   C   s2  t | dsDzt| } W n ty,   g  Y S 0 | j||||||dS t| dd\}}	t|	jdkrftt|D ]\}
}|j	j	||
< qn|dur|	j
|}|dkrtd|dur|	j
|}|dur|	j
|}t||	j
|||||d	}g }|D ]8\\}}}|	j
||	j
| }}|||f|f q|S dS )
a/  
    Compute isolating intervals for roots of ``f``.

    Examples
    ========

    >>> from sympy import intervals
    >>> from sympy.abc import x

    >>> intervals(x**2 - 3)
    [((-2, -1), 1), ((1, 2), 1)]
    >>> intervals(x**2 - 3, eps=1e-2)
    [((-26/15, -19/11), 1), ((19/11, 26/15), 1)]

    r<  r{  r'   r   rq   Nr   ry  )r|  r}  r~  r  r  )r   rL   r5   rz  r  rs   rN   r7   r   r]   r   r   r   rC   r   r   )r   rt  r|  r}  r~  r  r  r  r  ro   r   r   rz  rU   r'  rP  r   rX   rX   rY   rz  =  s6    


rz  c                 C   s\   z&t | }t| t s$|jjs$tdW n tyD   td|  Y n0 |j||||||dS )z
    Refine an isolating interval of a root to the given precision.

    Examples
    ========

    >>> from sympy import refine_root
    >>> from sympy.abc import x

    >>> refine_root(x**2 - 3, 1, 2, eps=1e-2)
    (19/11, 26/15)

    generator must be a Symbolz,Cannot refine a root of %s, not a polynomial)r|  r  r  r  )rL   rK   r   	is_Symbolr6   r5   r  )rS   r'  rP  r|  r  r  r  r   rX   rX   rY   r  u  s    
r  c                 C   sX   z*t | dd}t| t s(|jjs(tdW n tyH   td|  Y n0 |j||dS )a  
    Return the number of roots of ``f`` in ``[inf, sup]`` interval.

    If one of ``inf`` or ``sup`` is complex, it will return the number of roots
    in the complex rectangle with corners at ``inf`` and ``sup``.

    Examples
    ========

    >>> from sympy import count_roots, I
    >>> from sympy.abc import x

    >>> count_roots(x**4 - 4, -3, 3)
    2
    >>> count_roots(x**4 - 4, 0, 1 + 3*I)
    1

    FZgreedyr  z*Cannot count roots of %s, not a polynomialr  )rL   rK   r   r  r6   r5   r  )rS   r}  r~  r   rX   rX   rY   r    s    r  Tc                 C   sV   z*t | dd}t| t s(|jjs(tdW n tyH   td|  Y n0 |j|dS )z
    Return a list of real roots with multiplicities of ``f``.

    Examples
    ========

    >>> from sympy import real_roots
    >>> from sympy.abc import x

    >>> real_roots(2*x**3 - 7*x**2 + 4*x + 4)
    [-1/2, 2, 2]
    Fr  r  z1Cannot compute real roots of %s, not a polynomialr  )rL   rK   r   r  r6   r5   r  )rS   r  r   rX   rX   rY   r    s    
r  r  r  c                 C   sZ   z*t | dd}t| t s(|jjs(tdW n tyH   td|  Y n0 |j|||dS )aL  
    Compute numerical approximations of roots of ``f``.

    Examples
    ========

    >>> from sympy import nroots
    >>> from sympy.abc import x

    >>> nroots(x**2 - 3, n=15)
    [-1.73205080756888, 1.73205080756888]
    >>> nroots(x**2 - 3, n=30)
    [-1.73205080756887729352744634151, 1.73205080756887729352744634151]

    Fr  r  z6Cannot compute numerical roots of %s, not a polynomial)r   r  r  )rL   rK   r   r  r6   r5   r  )rS   r   r  r  r   rX   rX   rY   r    s    
r  c              
   O   s~   t |g  z8t| g|R i |\}}t| tsB|jjsBtdW n0 tyt } zt	dd|W Y d}~n
d}~0 0 |
 S )z
    Compute roots of ``f`` by factorization in the ground domain.

    Examples
    ========

    >>> from sympy import ground_roots
    >>> from sympy.abc import x

    >>> ground_roots(x**6 - 4*x**4 + 4*x**3 - x**2)
    {0: 2, 1: 2}

    r  r  rq   N)r`   r  r  rK   rL   r   r  r6   r9   r:   r  r  rX   rX   rY   r    s    "r  c              
   O   s   t |g  z8t| g|R i |\}}t| tsB|jjsBtdW n0 tyt } zt	dd|W Y d}~n
d}~0 0 |
|}|js| S |S dS )a  
    Construct a polynomial with n-th powers of roots of ``f``.

    Examples
    ========

    >>> from sympy import nth_power_roots_poly, factor, roots
    >>> from sympy.abc import x

    >>> f = x**4 - x**2 + 1
    >>> g = factor(nth_power_roots_poly(f, 2))

    >>> g
    (x**2 - x + 1)**2

    >>> R_f = [ (r**2).expand() for r in roots(f) ]
    >>> R_g = roots(g).keys()

    >>> set(R_f) == set(R_g)
    True

    r  r  rq   N)r`   r  r  rK   rL   r   r  r6   r9   r:   r  r  rQ   )rS   r   rN   rn   r   ro   r  rU   rX   rX   rY   r    s    "
r  )	_signsimpc                   sr  ddl m} ddlm  ddlm} t|dg t| } |rF|| } i }d|v r^|d |d< t	| t
tfs| jst	| tst	| ts| S t| dd} |  \}}nt| dkr| \}}t	|trt	|tr|j|d	< |j|d
< |dd|d< | |  }}n t	| trt| S td|  zj|  r:t |||fg|R i |\}	\}
}|	jst	| t
tfs~|  W S tj||fW S W n* ty } z| jr|  st|| js| j r"t!| j" fdddd\}}dd |D }| j#t$| j#| g|R  W  Y d}~S g }t%| }t&| |D ]P}t	|t
tt'frTq:z|(|t$|f |)  W n t*y   Y n0 q:| +t,|W  Y d}~S W Y d}~n
d}~0 0 d|
$| }\}}|ddrd	|vr|	j-|d	< t	| t
tfs|| |   S | |  }}|dds@|||fS |t|g|R i |t|g|R i |fS dS )a[  
    Cancel common factors in a rational function ``f``.

    Examples
    ========

    >>> from sympy import cancel, sqrt, Symbol, together
    >>> from sympy.abc import x
    >>> A = Symbol('A', commutative=False)

    >>> cancel((2*x**2 - 2)/(x**2 - 2*x + 1))
    (2*x + 2)/(x - 1)
    >>> cancel((sqrt(3) + sqrt(15)*A)/(sqrt(2) + sqrt(10)*A))
    sqrt(6)/2

    Note: due to automatic distribution of Rationals, a sum divided by an integer
    will appear as a sum. To recover a rational form use `together` on the result:

    >>> cancel(x/2 + 1)
    x/2 + 1
    >>> together(_)
    (x + 2)/2
    r   )signsimpr  )sringr  T)radicalr  rN   r   zunexpected argument: %sc                    s   | j du o|   S rm  )r   hasr  r  rX   rY   r  }  s    zcancel.<locals>.<lambda>ro  c                 S   s   g | ]}t |qS rX   )r  rT  rX   rX   rY   r     r   zcancel.<locals>.<listcomp>Nrq   F).rx  r  r  r  sympy.polys.ringsr  r`   r  r   rK   r   r	   r  r   r   r   r8  rs   rL   rN   r   rE  rQ   r   r  r6   Zngensr  r   r<  r   rq  r  rI   rn   rW   r  r!   r  r#   r   skiprb   r  re   r   )rS   r  rN   rn   r  r  ro   ra  r  r  r   r   r  r   Zncr  Zpoterk  r  rX   r  rY   r  8  s~    

"

(
0

r  c              
      s  t |ddg z(t| gt| g|R i |\} W n0 tyh } ztdd|W Y d}~n
d}~0 0  j}d} jr|jr|j	s 
t| d d}dd	lm} | j j j\}	}
t|D ](\}}| jj }|	|||< q|d |d
d \}} fdd|D }tt| }|rpzdd |D |  }}W n tyd   Y n0 || }} jsdd |D | fS ||fS dS )a<  
    Reduces a polynomial ``f`` modulo a set of polynomials ``G``.

    Given a polynomial ``f`` and a set of polynomials ``G = (g_1, ..., g_n)``,
    computes a set of quotients ``q = (q_1, ..., q_n)`` and the remainder ``r``
    such that ``f = q_1*g_1 + ... + q_n*g_n + r``, where ``r`` vanishes or ``r``
    is a completely reduced polynomial with respect to ``G``.

    Examples
    ========

    >>> from sympy import reduced
    >>> from sympy.abc import x, y

    >>> reduced(2*x**4 + y**2 - x**2 + y**3, [x**3 - x, y**3 - y])
    ([2*x, 1], x**2 + y**2 + y)

    r  r  reducedr   NFr   Txringrq   c                    s   g | ]}t t| qS rX   rL   rf   re   r   r  r   rX   rY   r     r   zreduced.<locals>.<listcomp>c                 S   s   g | ]}|  qS rX   r   r  rX   rX   rY   r     r   c                 S   s   g | ]}|  qS rX   r  r  rX   rX   rY   r     r   )r`   r  r  rh   r9   r:   r   r  r  r  rc  re   r  r  r  rN   r^   r   r   r]   r   r}   r  rL   rf   r   r3   r  rQ   )rS   r   rN   rn   r  r  r   r   r  _ringr   r   r   r  r  _Q_rrX   r   rY   r    s6    ("
r  c                 O   s   t | g|R i |S )a  
    Computes the reduced Groebner basis for a set of polynomials.

    Use the ``order`` argument to set the monomial ordering that will be
    used to compute the basis. Allowed orders are ``lex``, ``grlex`` and
    ``grevlex``. If no order is specified, it defaults to ``lex``.

    For more information on Groebner bases, see the references and the docstring
    of :func:`~.solve_poly_system`.

    Examples
    ========

    Example taken from [1].

    >>> from sympy import groebner
    >>> from sympy.abc import x, y

    >>> F = [x*y - 2*y, 2*y**2 - x**2]

    >>> groebner(F, x, y, order='lex')
    GroebnerBasis([x**2 - 2*y**2, x*y - 2*y, y**3 - 2*y], x, y,
                  domain='ZZ', order='lex')
    >>> groebner(F, x, y, order='grlex')
    GroebnerBasis([y**3 - 2*y, x**2 - 2*y**2, x*y - 2*y], x, y,
                  domain='ZZ', order='grlex')
    >>> groebner(F, x, y, order='grevlex')
    GroebnerBasis([y**3 - 2*y, x**2 - 2*y**2, x*y - 2*y], x, y,
                  domain='ZZ', order='grevlex')

    By default, an improved implementation of the Buchberger algorithm is
    used. Optionally, an implementation of the F5B algorithm can be used. The
    algorithm can be set using the ``method`` flag or with the
    :func:`sympy.polys.polyconfig.setup` function.

    >>> F = [x**2 - x - 1, (2*x - 1) * y - (x**10 - (1 - x)**10)]

    >>> groebner(F, x, y, method='buchberger')
    GroebnerBasis([x**2 - x - 1, y - 55], x, y, domain='ZZ', order='lex')
    >>> groebner(F, x, y, method='f5b')
    GroebnerBasis([x**2 - x - 1, y - 55], x, y, domain='ZZ', order='lex')

    References
    ==========

    1. [Buchberger01]_
    2. [Cox97]_

    )GroebnerBasisr   rN   rn   rX   rX   rY   r+     s    3r+   c                 O   s   t | g|R i |jS )a[  
    Checks if the ideal generated by a Groebner basis is zero-dimensional.

    The algorithm checks if the set of monomials not divisible by the
    leading monomial of any element of ``F`` is bounded.

    References
    ==========

    David A. Cox, John B. Little, Donal O'Shea. Ideals, Varieties and
    Algorithms, 3rd edition, p. 230

    )r  is_zero_dimensionalr  rX   rX   rY   r    s    r  c                   @   s   e Zd ZdZdd Zedd Zedd Zedd	 Z	ed
d Z
edd Zedd Zedd Zdd Zdd Zdd Zdd Zdd Zdd Zedd Zd d! Zd(d#d$Zd%d& Zd'S ))r  z%Represents a reduced Groebner basis. c              
      s   t |ddg zt|g|R i |\} W n4 tyb } ztdt||W Y d}~n
d}~0 0 ddlm} | j j	 j
fdd|D }t| jd	} fd
d|D }| | S )z>Compute a reduced Groebner basis for a system of polynomials. r  rN  r+   Nr   )PolyRingc                    s    g | ]}|r  |j qS rX   )r}   r]   r   r   r   )ringrX   rY   r   3  r   z)GroebnerBasis.__new__.<locals>.<listcomp>r}  c                    s   g | ]}t | qS rX   )rL   rf   r   rT   r   rX   rY   r   6  r   )r`   r  r  r9   r:   rs   r  r  rN   r   r^   	_groebnerrN  _new)rm   r   rN   rn   r  r  r  r   rX   )ro   r  rY   rp   '  s    &zGroebnerBasis.__new__c                 C   s   t | }t||_||_|S rv   )r
   rp   r   _basis_options)rm   basisr`   rt   rX   rX   rY   r  :  s    

zGroebnerBasis._newc                 C   s$   dd | j D }t| t| jj fS )Nc                 s   s   | ]}|  V  qd S rv   r  )r   ra  rX   rX   rY   r/  E  r   z%GroebnerBasis.args.<locals>.<genexpr>)r  r	   r  rN   )ry   r  rX   rX   rY   rn   C  s    zGroebnerBasis.argsc                 C   s   dd | j D S )Nc                 S   s   g | ]}|  qS rX   r  r  rX   rX   rY   r   J  r   z'GroebnerBasis.exprs.<locals>.<listcomp>)r  rx   rX   rX   rY   r  H  s    zGroebnerBasis.exprsc                 C   s
   t | jS rv   )rh   r  rx   rX   rX   rY   r  L  s    zGroebnerBasis.polysc                 C   s   | j jS rv   )r  rN   rx   rX   rX   rY   rN   P  s    zGroebnerBasis.gensc                 C   s   | j jS rv   )r  r   rx   rX   rX   rY   r   T  s    zGroebnerBasis.domainc                 C   s   | j jS rv   )r  r^   rx   rX   rX   rY   r^   X  s    zGroebnerBasis.orderc                 C   s
   t | jS rv   )rs   r  rx   rX   rX   rY   __len__\  s    zGroebnerBasis.__len__c                 C   s    | j jrt| jS t| jS d S rv   )r  r  iterr  rx   rX   rX   rY   r<  _  s    
zGroebnerBasis.__iter__c                 C   s   | j jr| j}n| j}|| S rv   )r  r  r  )ry   itemr  rX   rX   rY   __getitem__e  s    zGroebnerBasis.__getitem__c                 C   s   t | jt| j fS rv   )hashr  r   r  r   rx   rX   rX   rY   r   m  s    zGroebnerBasis.__hash__c                 C   sP   t || jr$| j|jko"| j|jkS t|rH| jt|kpF| jt|kS dS d S )NF)rK   r   r  r  rH   r  rh   r  ry   r  rX   rX   rY   r  p  s
    zGroebnerBasis.__eq__c                 C   s
   | |k S rv   rX   r  rX   rX   rY   r  x  s    zGroebnerBasis.__ne__c                 C   sT   dd }t dgt| j }| jj}| jD ] }|j|d}||r*||9 }q*t|S )a{  
        Checks if the ideal generated by a Groebner basis is zero-dimensional.

        The algorithm checks if the set of monomials not divisible by the
        leading monomial of any element of ``F`` is bounded.

        References
        ==========

        David A. Cox, John B. Little, Donal O'Shea. Ideals, Varieties and
        Algorithms, 3rd edition, p. 230

        c                 S   s   t tt| dkS r   )sumr   r  )monomialrX   rX   rY   
single_var  s    z5GroebnerBasis.is_zero_dimensional.<locals>.single_varr   r   )r,   rs   rN   r  r^   r  r3  rt  )ry   r  r-  r^   r   r  rX   rX   rY   r  {  s    

z!GroebnerBasis.is_zero_dimensionalc                    s   | j   j}t|}||kr | S | js.tdt| j} j} t	|
 |d ddlm} | j j|\}}t|D ](\}	}
|
 jj }
||
||	< q|t|||} fdd|D }|jsdd |D }| _| | S )a  
        Convert a Groebner basis from one ordering to another.

        The FGLM algorithm converts reduced Groebner bases of zero-dimensional
        ideals from one ordering to another. This method is often used when it
        is infeasible to compute a Groebner basis with respect to a particular
        ordering directly.

        Examples
        ========

        >>> from sympy.abc import x, y
        >>> from sympy import groebner

        >>> F = [x**2 - 3*y - x + 1, y**2 - 2*x + y - 1]
        >>> G = groebner(F, x, y, order='grlex')

        >>> list(G.fglm('lex'))
        [2*x - y**2 - y + 1, y**4 + 2*y**3 - 3*y**2 - 16*y + 7]
        >>> list(groebner(F, x, y, order='lex'))
        [2*x - y**2 - y + 1, y**4 + 2*y**3 - 3*y**2 - 16*y + 7]

        References
        ==========

        .. [1] J.C. Faugere, P. Gianni, D. Lazard, T. Mora (1994). Efficient
               Computation of Zero-dimensional Groebner Bases by Change of
               Ordering

        z?Cannot convert Groebner bases of ideals with positive dimension)r   r^   r   r  c                    s   g | ]}t t| qS rX   r  r  r   rX   rY   r     r   z&GroebnerBasis.fglm.<locals>.<listcomp>c                 S   s   g | ]}|j d dd qS )Tr?  rq   )r;  r  rX   rX   rY   r     r   )r  r^   r-   r  rb   rh   r  r   rc  re   r  r  r  rN   r   r   r]   r   r}   r*   r  r  )ry   r^   Z	src_orderZ	dst_orderr  r   r  r  r   r   r   r   rX   r   rY   fglm  s0    
zGroebnerBasis.fglmTc                    sR  t || j}|gt| j }| j  j}d}|rV|jrV|jsV t	|
 d d}ddlm} | j j j\}}	t|D ](\}
}| jj }||||
< q|d |dd \}} fdd	|D }t t	| }|r(zd
d	 |D |  }}W n ty   Y n0 || }} jsFdd	 |D | fS ||fS dS )a#  
        Reduces a polynomial modulo a Groebner basis.

        Given a polynomial ``f`` and a set of polynomials ``G = (g_1, ..., g_n)``,
        computes a set of quotients ``q = (q_1, ..., q_n)`` and the remainder ``r``
        such that ``f = q_1*f_1 + ... + q_n*f_n + r``, where ``r`` vanishes or ``r``
        is a completely reduced polynomial with respect to ``G``.

        Examples
        ========

        >>> from sympy import groebner, expand
        >>> from sympy.abc import x, y

        >>> f = 2*x**4 - x**2 + y**3 + y**2
        >>> G = groebner([x**3 - x, y**3 - y])

        >>> G.reduce(f)
        ([2*x, 1], x**2 + y**2 + y)
        >>> Q, r = _

        >>> expand(sum(q*g for q, g in zip(Q, G)) + r)
        2*x**4 - x**2 + y**3 + y**2
        >>> _ == f
        True

        Fr   Tr   r  rq   Nc                    s   g | ]}t t| qS rX   r  r  r   rX   rY   r     r   z(GroebnerBasis.reduce.<locals>.<listcomp>c                 S   s   g | ]}|  qS rX   r  r  rX   rX   rY   r     r   c                 S   s   g | ]}|  qS rX   r  r  rX   rX   rY   r     r   )rL   rk   r  rh   r  r   r  r  rc  re   r  r  r  rN   r^   r   r   r]   r   r}   r  rf   r   r3   r  rQ   )ry   rz   r  r   r  r   r   r  r  r   r   r  r  r  r  rX   r   rY   r     s2    
zGroebnerBasis.reducec                 C   s   |  |d dkS )am  
        Check if ``poly`` belongs the ideal generated by ``self``.

        Examples
        ========

        >>> from sympy import groebner
        >>> from sympy.abc import x, y

        >>> f = 2*x**3 + y**3 + 3*y
        >>> G = groebner([x**2 + y**2 - 1, x*y - 2])

        >>> G.contains(f)
        True
        >>> G.contains(f + 1)
        False

        rq   r   )r   )ry   r   rX   rX   rY   contains  s    zGroebnerBasis.containsN)T)rR   r  r  r  rp   r  r  r  rn   r  r  rN   r   r^   r  r<  r  r   r  r  r  r  r   r  rX   rX   rX   rY   r  #  s6   







B
Ar  c                    sd   t g   fdd t| } | jr>t| g|R i S dvrNdd< t |} | |S )z
    Efficiently transform an expression into a polynomial.

    Examples
    ========

    >>> from sympy import poly
    >>> from sympy.abc import x

    >>> poly(x*(x**2 + x - 1)**2)
    Poly(x**5 + 2*x**4 - x**3 - 2*x**2 + x, x, domain='ZZ')

    c           
         s  g g  }}t | D ]}g g  }}t|D ]b}|jrH| || q,|jr|jjr|jjr|jdkr| |j|	|j q,|| q,|s|| q|d }|dd  D ]}|
|}q|rt| }|jr|
|}n|
t||}|| q|st| |}	nZ|d }	|dd  D ]}|	|}	q(|rnt | }|jr\|	|}	n|	t||}	|	j|ddi S )Nr   rq   rN   rX   )r   rW  r   rq  r   rX  rY  rM  r  r  r   r  rL   rk   r   r   rE  )
rz   ro   r   Z
poly_termsrH  ru  Zpoly_factorsr  productrU   _polyrn   rX   rY   r  E  sJ    

zpoly.<locals>._polyr  F)r`   r  r   ri   rL   ra   r  rX   r  rY   r   4  s    4r   )r   )N)N)FNNNFFF)NNFF)NN)T)r  r  T)r  	functoolsr   r   operatorr   Z
sympy.corer   r   r   r	   Zsympy.core.basicr
   Zsympy.core.decoratorsr   Zsympy.core.exprtoolsr   r   r   Zsympy.core.evalfr   r   r   r   r   Zsympy.core.functionr   Zsympy.core.mulr   r   Zsympy.core.numbersr   r   r   Zsympy.core.relationalr   r   Zsympy.core.sortingr   Zsympy.core.symbolr   r   Zsympy.core.sympifyr   r    Zsympy.core.traversalr!   r"   Zsympy.logic.boolalgr#   Zsympy.polysr$   r`   Zsympy.polys.constructorr%   Zsympy.polys.domainsr&   r'   r(   Z!sympy.polys.domains.domainelementr)   Zsympy.polys.fglmtoolsr*   Zsympy.polys.groebnertoolsr+   r  Zsympy.polys.monomialsr,   Zsympy.polys.orderingsr-   Zsympy.polys.polyclassesr.   r/   r0   Zsympy.polys.polyerrorsr1   r2   r3   r4   r5   r6   r7   r8   r9   r:   r;   Zsympy.polys.polyutilsr<   r=   r>   r?   r@   rA   Zsympy.polys.rationaltoolsrB   Zsympy.polys.rootisolationrC   Zsympy.utilitiesrD   rE   rF   Zsympy.utilities.exceptionsrG   Zsympy.utilities.iterablesrH   rI   r  r  Zmpmath.libmp.libhyperrJ   r[   rL   r  r  r  r  r  r  r   r"  r!  r(  r3  r7  r  r  r  r  r  r  r  r  rM  rO  rQ  rS  rT  rW  r[  r7  r^  r?  r_  r  r`  rb  rd  re  rg  rh  rl  rm  rJ  rq  rr  rO  rR  r\  r_  re  rf  ry  r|  rs  r  rw  r  rz  r  r  r  r  r  r  r  r  r  r  r   rX   rX   rX   rY   <module>   s  4
 "                                h]
(
_
:
4






"
"



"
&
&
4
$

(
T3
M2
u



-



.

!
:,	,


b7 

+f
;
5
  