o
    8Va9                     @   s   d 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
 i ddddd	d	d
d
ddddddddddddddddddddddddddddddddddd d!d"
Zg d#ZG d$d% d%eZd+d'd(Zd)d* Zd&S ),z
R code printer

The RCodePrinter converts single sympy expressions into single R expressions,
using the functions defined in math.h where possible.



    )AnyDict)CodePrinter)
precedence
PRECEDENCE)RangeZAbsabssinZcosZtanZasinZacosZatanZatan2explogZerfZsinhZcoshZtanhZasinhZacoshZatanhfloorceilingsignmaxmin	factorialgammadigammatrigammabeta)
r   r   r   ZMaxZMinr   r   r   r   r   )ifelserepeatwhilefunctionforinnextbreakZTRUEZFALSEZNULLInfZNaNZNAZNA_integer_ZNA_real_ZNA_complex_ZNA_character_Zvolatilec                
       s  e Zd ZdZdZdZdddi dde dd	d
	ZddddZi Z	i fddZ
dd Zdd Zdd Zdd Zdd Zdd Zdd Zdd  Zd!d" Zd#d$ Zd%d& Zd'd( Zd)d* Zd+d, Zd-d. Zd/d0 Zd1d2 Zd3d4 Zd5d6 Z fd7d8Zd9d: Zd;d< Z d=d> Z!d?d@ Z"dAdB Z#  Z$S )CRCodePrinterz<A printer to convert python expressions to strings of R codeZ_rcodeRNauto   TF_)	ZorderZ	full_precZ	precisionuser_functionsZhumancontractdereferenceZerror_on_reservedZreserved_word_suffix&|!)andornotc                 C   sN   t | | tt| _|di }| j| t|dg | _tt| _d S )Nr%   r'   )	r   __init__dictknown_functionsgetupdateset_dereferencereserved_words)selfsettingsZ	userfuncs r8   6/usr/lib/python3/dist-packages/sympy/printing/rcode.pyr.   d   s   
zRCodePrinter.__init__c                 C   s   |d S )N   r8   )r6   pr8   r8   r9   _rate_index_positionl      z!RCodePrinter._rate_index_positionc                 C   s   d| S )Nz%s;r8   )r6   Z
codestringr8   r8   r9   _get_statemento   r=   zRCodePrinter._get_statementc                 C   s
   d |S )Nz// {}format)r6   textr8   r8   r9   _get_commentr      
zRCodePrinter._get_commentc                 C   s   d ||S )Nz{} = {};r?   )r6   namevaluer8   r8   r9   _declare_number_constu      z"RCodePrinter._declare_number_constc                 C   s
   |  |S N)indent_code)r6   linesr8   r8   r9   _format_codex   rC   zRCodePrinter._format_codec                    s    |j \}  fddt|D S )Nc                 3   s&    | ]}t  D ]}||fV  qqd S rH   )range).0ijZcolsr8   r9   	<genexpr>}   s   $ z8RCodePrinter._traverse_matrix_indices.<locals>.<genexpr>)shaperL   )r6   ZmatZrowsr8   rP   r9   _traverse_matrix_indices{   s   
z%RCodePrinter._traverse_matrix_indicesc              
   C   s^   g }g }d}|D ]"}| || |j| |jd | |jd d  | d q||fS )zPReturns a tuple (open_lines, close_lines) containing lists of codelines
        z#for (%(var)s in %(start)s:%(end)s){   )varstartend})append_printlabellowerupper)r6   indicesZ
open_linesZclose_linesZ	loopstartrN   r8   r8   r9   _get_loop_opening_ending   s   

z%RCodePrinter._get_loop_opening_endingc                 C   sr   d| j v r
| |S t|}|jdkrd| |j| S |jdkr)d| |j S d| |j|| |j|f S )NZPowz1.0/%sg      ?zsqrt(%s)z%s^%s)r0   Z_print_Functionr   r
   parenthesizebaserZ   )r6   exprZPRECr8   r8   r9   
_print_Pow   s   



zRCodePrinter._print_Powc                 C   s"   t |jt |j}}d||f S )Nz	%d.0/%d.0)intr;   q)r6   rc   r;   rf   r8   r8   r9   _print_Rational   s   zRCodePrinter._print_Rationalc                    s0    fdd|j D }d |jjd|f S )Nc                    s   g | ]}  |qS r8   )rZ   )rM   rN   r6   r8   r9   
<listcomp>       z/RCodePrinter._print_Indexed.<locals>.<listcomp>z%s[%s]z, )r^   rZ   rb   r[   join)r6   rc   Zindsr8   rh   r9   _print_Indexed   s   zRCodePrinter._print_Indexedc                 C   s   |  |jS rH   )rZ   r[   r6   rc   r8   r8   r9   
_print_Idx   rG   zRCodePrinter._print_Idxc                 C      dS )Nzexp(1)r8   rm   r8   r8   r9   _print_Exp1      zRCodePrinter._print_Exp1c                 C   ro   )NZpir8   rm   r8   r8   r9   	_print_Pi   rq   zRCodePrinter._print_Pic                 C   ro   )Nr   r8   rm   r8   r8   r9   _print_Infinity   rq   zRCodePrinter._print_Infinityc                 C   ro   )Nz-Infr8   rm   r8   r8   r9   _print_NegativeInfinity   rq   z$RCodePrinter._print_NegativeInfinityc                 C   s   ddl m} ddlm} ddlm} |j}|j}t||rEg }| 	|D ]\}}	||||	f |||	f }
| 
|
}|| q$d|S | jd rZ||sT||rZ| ||S | 
|}| 
|}| d||f S )Nr   )
Assignment)MatrixSymbol)IndexedBase
r&   z%s = %s)Zsympy.codegen.astru   Z"sympy.matrices.expressions.matexprrv   Zsympy.tensor.indexedrw   lhsrhs
isinstancerS   rZ   rY   rk   Z	_settingsZhasZ_doprint_loopsr>   )r6   rc   ru   rv   rw   ry   rz   rJ   rN   rO   ZtempZcode0lhs_coderhs_coder8   r8   r9   _print_Assignment   s&   




zRCodePrinter._print_Assignmentc                 C   s   |j d jdkrd| |j d j }nd| |j d j| |j d jf }|}t|j d d D ]\}}d| || |f | d }q3|S )Nr`   Tz%szifelse(%s,%s,NA)zifelse(%s,%s,))argsZcondrZ   rc   reversed)r6   rc   	last_linecodeecr8   r8   r9   _print_Piecewise   s   ("zRCodePrinter._print_Piecewisec                 C   s:   ddl m} ||jd |jd f|jd df}| |S )Nr   	PiecewiserT      T)sympy.functionsr   r   rZ   )r6   rc   r   
_piecewiser8   r8   r9   
_print_ITE   s   $
zRCodePrinter._print_ITEc                 C   s2   d | j|jtd dd|j|j|jjd   S )Nz{}[{}]ZAtomT)strictrT   )r@   ra   parentr   rO   rN   rR   rm   r8   r8   r9   _print_MatrixElement   s
   z!RCodePrinter._print_MatrixElementc                    s$   t  |}|| jv rd|S |S )Nz(*{}))super_print_Symbolr4   r@   )r6   rc   rD   	__class__r8   r9   r      s   

zRCodePrinter._print_Symbolc                 C   s,   |  |j}|  |j}|j}d|||S )Nz{} {} {})rZ   ry   rz   Zrel_opr@   )r6   rc   r|   r}   opr8   r8   r9   _print_Relational   s   zRCodePrinter._print_Relationalc                 C   s\   ddl m} ddlm} ddlm} |||jd |jd  ||jd dfd}| |S )Nr   )r	   )Ner   )rT   T)Z(sympy.functions.elementary.trigonometricr	   Zsympy.core.relationalr   r   r   r   rZ   )r6   rc   r	   r   r   r   r8   r8   r9   _print_sinc   s   (
zRCodePrinter._print_sincc                 C   s,   |  |j}|j}|  |j}d|||S )Nz	{} {} {};)rZ   ry   r   rz   r@   )r6   rc   r|   r   r}   r8   r8   r9   _print_AugmentedAssignment  s   z'RCodePrinter._print_AugmentedAssignmentc                 C   sP   |  |j}t|jtr|jj\}}}ntd|  |j}dj|||||dS )Nz*Only iterable currently supported is RangezLfor ({target} = {start}; {target} < {stop}; {target} += {step}) {{
{body}
}})targetrV   stopstepbody)	rZ   r   r{   iterabler   r   NotImplementedErrorr   r@   )r6   rc   r   rV   r   r   r   r8   r8   r9   
_print_For	  s   zRCodePrinter._print_Forc           
         s   t |tr| |d}d|S d}dd dd |D }fdd|D } fd	d|D }g }d
}t|D ])\}}	|	dksE|	dkrK||	 q9||| 8 }|d|| |	f  ||| 7 }q9|S )z0Accepts a string of code or a list of code linesT z   ){(z{
z(
)rX   r   c                 S   s   g | ]}| d qS )z 	)lstriprM   liner8   r8   r9   ri      rj   z,RCodePrinter.indent_code.<locals>.<listcomp>c                        g | ]}t tt|j qS r8   )re   anymapendswithr   )	inc_tokenr8   r9   ri   "  s     c                    r   r8   )re   r   r   
startswithr   )	dec_tokenr8   r9   ri   #  s    r   rx   z%s%s)r{   strrI   
splitlinesrk   	enumeraterY   )
r6   r   Z
code_linesZtabZincreaseZdecreaseZprettylevelnr   r8   )r   r   r9   rI     s*   



zRCodePrinter.indent_code)%__name__
__module____qualname____doc__Zprintmethodlanguager3   Z_default_settingsZ
_operatorsZ_relationalsr.   r<   r>   rB   rF   rK   rS   r_   rd   rg   rl   rn   rp   rr   rs   rt   r~   r   r   r   r   r   r   r   r   rI   __classcell__r8   r8   r   r9   r    K   sZ    'r    Nc                 K   s   t || |S )a  Converts an expr to a string of r code

    Parameters
    ==========

    expr : Expr
        A sympy expression to be converted.
    assign_to : optional
        When given, the argument is used as the name of the variable to which
        the expression is assigned. Can be a string, ``Symbol``,
        ``MatrixSymbol``, or ``Indexed`` type. This is helpful in case of
        line-wrapping, or for expressions that generate multi-line statements.
    precision : integer, optional
        The precision for numbers such as pi [default=15].
    user_functions : dict, optional
        A dictionary where the keys are string representations of either
        ``FunctionClass`` or ``UndefinedFunction`` instances and the values
        are their desired R string representations. Alternatively, the
        dictionary value can be a list of tuples i.e. [(argument_test,
        rfunction_string)] or [(argument_test, rfunction_formater)]. See below
        for examples.
    human : bool, optional
        If True, the result is a single string that may contain some constant
        declarations for the number symbols. If False, the same information is
        returned in a tuple of (symbols_to_declare, not_supported_functions,
        code_text). [default=True].
    contract: bool, optional
        If True, ``Indexed`` instances are assumed to obey tensor contraction
        rules and the corresponding nested loops over indices are generated.
        Setting contract=False will not generate loops, instead the user is
        responsible to provide values for the indices in the code.
        [default=True].

    Examples
    ========

    >>> from sympy import rcode, symbols, Rational, sin, ceiling, Abs, Function
    >>> x, tau = symbols("x, tau")
    >>> rcode((2*tau)**Rational(7, 2))
    '8*sqrt(2)*tau^(7.0/2.0)'
    >>> rcode(sin(x), assign_to="s")
    's = sin(x);'

    Simple custom printing can be defined for certain types by passing a
    dictionary of {"type" : "function"} to the ``user_functions`` kwarg.
    Alternatively, the dictionary value can be a list of tuples i.e.
    [(argument_test, cfunction_string)].

    >>> custom_functions = {
    ...   "ceiling": "CEIL",
    ...   "Abs": [(lambda x: not x.is_integer, "fabs"),
    ...           (lambda x: x.is_integer, "ABS")],
    ...   "func": "f"
    ... }
    >>> func = Function('func')
    >>> rcode(func(Abs(x) + ceiling(x)), user_functions=custom_functions)
    'f(fabs(x) + CEIL(x))'

    or if the R-function takes a subset of the original arguments:

    >>> rcode(2**x + 3**x, user_functions={'Pow': [
    ...   (lambda b, e: b == 2, lambda b, e: 'exp2(%s)' % e),
    ...   (lambda b, e: b != 2, 'pow')]})
    'exp2(x) + pow(3, x)'

    ``Piecewise`` expressions are converted into conditionals. If an
    ``assign_to`` variable is provided an if statement is created, otherwise
    the ternary operator is used. Note that if the ``Piecewise`` lacks a
    default term, represented by ``(expr, True)`` then an error will be thrown.
    This is to prevent generating an expression that may not evaluate to
    anything.

    >>> from sympy import Piecewise
    >>> expr = Piecewise((x + 1, x > 0), (x, True))
    >>> print(rcode(expr, assign_to=tau))
    tau = ifelse(x > 0,x + 1,x);

    Support for loops is provided through ``Indexed`` types. With
    ``contract=True`` these expressions will be turned into loops, whereas
    ``contract=False`` will just print the assignment expression that should be
    looped over:

    >>> from sympy import Eq, IndexedBase, Idx
    >>> len_y = 5
    >>> y = IndexedBase('y', shape=(len_y,))
    >>> t = IndexedBase('t', shape=(len_y,))
    >>> Dy = IndexedBase('Dy', shape=(len_y-1,))
    >>> i = Idx('i', len_y-1)
    >>> e=Eq(Dy[i], (y[i+1]-y[i])/(t[i+1]-t[i]))
    >>> rcode(e.rhs, assign_to=e.lhs, contract=False)
    'Dy[i] = (y[i + 1] - y[i])/(t[i + 1] - t[i]);'

    Matrices are also supported, but a ``MatrixSymbol`` of the same dimensions
    must be provided to ``assign_to``. Note that any expression that can be
    generated normally can also exist inside a Matrix:

    >>> from sympy import Matrix, MatrixSymbol
    >>> mat = Matrix([x**2, Piecewise((x + 1, x > 0), (x, True)), sin(x)])
    >>> A = MatrixSymbol('A', 3, 1)
    >>> print(rcode(mat, A))
    A[0] = x^2;
    A[1] = ifelse(x > 0,x + 1,x);
    A[2] = sin(x);

    )r    Zdoprint)rc   Z	assign_tor7   r8   r8   r9   rcode2  s   kr   c                 K   s   t t| fi | dS )z0Prints R representation of the given expression.N)printr   )rc   r7   r8   r8   r9   print_rcode  s   r   rH   )r   typingr   r   Zsympy.printing.codeprinterr   Zsympy.printing.precedencer   r   Zsympy.sets.fancysetsr   r0   r5   r    r   r   r8   r8   r8   r9   <module>   sp    
	
" 
hn