B
    fcQV              	   @   s  d Z ddlZddlZdZdZdd ZG dd dZd	d
 Zdd Zdd Z	dd Z
dd Zdd ZG dd deZddddddddhZdedejfd ed!d"e d#fd$ed%fd&ed'fd(ed)ejfgZd*d+ ZdJd,d-ZdKd.d/Zd0d1 Zd2d3 Zd4d5 Zd6d7 ZG d8d9 d9Zd:d; Zd<d= Zd>d? ZG d@dA dAZG dBdC dCeZ G dDdE dEeZ!G dFdG dGe!Z"G dHdI dIeZ#dS )Lz
    babel.numbers
    ~~~~~~~~~~~~~

    CLDR Plural support.  See UTS #35.

    :copyright: (c) 2013-2022 by the Babel Team.
    :license: BSD, see LICENSE for more details.
    N)ZzeroZoneZtwoZfewZmanyotherr   c             C   s   t | }t|}t|tr6||kr(|}ntt|}t|tjr| }|j}|dk rf|j	|d nd}d
dd |D }|d}t|}t|}	t|pd}
t|pd}nd } }	 }
}d }}||||	|
|||fS )u  Extract operands from a decimal, a float or an int, according to `CLDR rules`_.

    The result is a 8-tuple (n, i, v, w, f, t, c, e), where those symbols are as follows:

    ====== ===============================================================
    Symbol Value
    ------ ---------------------------------------------------------------
    n      absolute value of the source number (integer and decimals).
    i      integer digits of n.
    v      number of visible fraction digits in n, with trailing zeros.
    w      number of visible fraction digits in n, without trailing zeros.
    f      visible fractional digits in n, with trailing zeros.
    t      visible fractional digits in n, without trailing zeros.
    c      compact decimal exponent value: exponent of the power of 10 used in compact decimal formatting.
    e      currently, synonym for ‘c’. however, may be redefined in the future.
    ====== ===============================================================

    .. _`CLDR rules`: https://www.unicode.org/reports/tr35/tr35-61/tr35-numbers.html#Operands

    :param source: A real number
    :type source: int|float|decimal.Decimal
    :return: A n-i-v-w-f-t-c-e tuple
    :rtype: tuple[decimal.Decimal, int, int, int, int, int, int, int]
    r   N  c             s   s   | ]}t |V  qd S )N)str).0dr   r   +lib/python3.7/site-packages/babel/plural.py	<genexpr>A   s    z#extract_operands.<locals>.<genexpr>0)absint
isinstancefloatdecimalZDecimalr   Zas_tupleZexponentdigitsjoinrstriplen)sourceniZ	dec_tupleZexpZfraction_digitsZtrailingZno_trailingvwftcer   r   r   extract_operands   s&    

r   c               @   sd   e Zd ZdZdZdd Zdd Zedd Ze	d	d
 Z
e	dd ddZdd Zdd Zdd ZdS )
PluralRuleaf  Represents a set of language pluralization rules.  The constructor
    accepts a list of (tag, expr) tuples or a dict of `CLDR rules`_. The
    resulting object is callable and accepts one parameter with a positive or
    negative number (both integer and float) for the number that indicates the
    plural form for a string and returns the tag for the format:

    >>> rule = PluralRule({'one': 'n is 1'})
    >>> rule(1)
    'one'
    >>> rule(2)
    'other'

    Currently the CLDR defines these tags: zero, one, two, few, many and
    other where other is an implicit default.  Rules should be mutually
    exclusive; for a given numeric value, only one rule should apply (i.e.
    the condition should only be true for one of the plural rule elements.

    .. _`CLDR rules`: https://www.unicode.org/reports/tr35/tr35-33/tr35-numbers.html#Language_Plural_Rules
    )abstract_funcc             C   s   t |tr| }t }g | _xjtt|D ]Z\}}|tkrJtd| n||kr^td| |	| t
|j}|r,| j||f q,W dS )a$  Initialize the rule instance.

        :param rules: a list of ``(tag, expr)``) tuples with the rules
                      conforming to UTS #35 or a dict with the tags as keys
                      and expressions as values.
        :raise RuleError: if the expression is malformed
        zunknown tag %rztag %r defined twiceN)r   dictitemssetr   sortedlist_plural_tags
ValueErroradd_Parserastappend)selfrulesfoundkeyexprr*   r   r   r   __init__d   s    


zPluralRule.__init__c                s,   | j  dt| jd fddtD f S )Nz<%s %r>z, c                s$   g | ]}| krd | | f qS )z%s: %sr   )r   tag)r-   r   r   
<listcomp>~   s    z'PluralRule.__repr__.<locals>.<listcomp>)r-   type__name__r   r&   )r,   r   )r-   r   __repr__z   s    zPluralRule.__repr__c             C   s   t || r|S | |S )a
  Create a `PluralRule` instance for the given rules.  If the rules
        are a `PluralRule` object, that object is returned.

        :param rules: the rules as list or dict, or a `PluralRule` object
        :raise RuleError: if the expression is malformed
        )r   )clsr-   r   r   r   parse   s    
zPluralRule.parsec                s   t  j  fdd| jD S )zThe `PluralRule` as a dict of unicode plural rules.

        >>> rule = PluralRule({'one': 'n is 1'})
        >>> rule.rules
        {'one': 'n is 1'}
        c                s   i | ]\}} ||qS r   r   )r   r2   r*   )_compiler   r   
<dictcomp>   s    z$PluralRule.rules.<locals>.<dictcomp>)_UnicodeCompilercompiler   )r,   r   )r9   r   r-      s    zPluralRule.rulesc             C   s   t dd | jD S )Nc             s   s   | ]}|d  V  qdS )r   Nr   )r   r   r   r   r   r	      s    z&PluralRule.<lambda>.<locals>.<genexpr>)	frozensetr   )xr   r   r   <lambda>       zPluralRule.<lambda>z
        A set of explicitly defined tags in this rule.  The implicit default
        ``'other'`` rules is not part of this set unless there is an explicit
        rule for it.)docc             C   s   | j S )N)r   )r,   r   r   r   __getstate__   s    zPluralRule.__getstate__c             C   s
   || _ d S )N)r   )r,   r   r   r   r   __setstate__   s    zPluralRule.__setstate__c             C   s   t | dst| | _| |S )Nr    )hasattr	to_pythonr    )r,   r   r   r   r   __call__   s    

zPluralRule.__call__N)r5   
__module____qualname____doc__	__slots__r1   r6   classmethodr8   propertyr-   tagsrB   rC   rF   r   r   r   r   r   M   s   r   c             C   sV   t  j}dg}x.t| jD ]\}}|d|||f  qW |dt  d|S )a  Convert a list/dict of rules or a `PluralRule` object into a JavaScript
    function.  This function depends on no external library:

    >>> to_javascript({'one': 'n is 1'})
    "(function(n) { return (n == 1) ? 'one' : 'other'; })"

    Implementation detail: The function generated will probably evaluate
    expressions involved into range operations multiple times.  This has the
    advantage that external helper functions are not required and is not a
    big performance hit for these simple calculations.

    :param rule: the rules as list or dict, or a `PluralRule` object
    :raise RuleError: if the expression is malformed
    z(function(n) { return z
%s ? %r : z%r; })r   )_JavaScriptCompilerr<   r   r8   r   r+   _fallback_tagr   )ruleZto_jsresultr2   r*   r   r   r   to_javascript   s    rR   c             C   s   t tttd}t j}ddg}x2t| jD ]"\}}|	d||t
|f  q,W |	dt  td|dd}t|| |d	 S )
a<  Convert a list/dict of rules or a `PluralRule` object into a regular
    Python function.  This is useful in situations where you need a real
    function and don't are about the actual rule object:

    >>> func = to_python({'one': 'n is 1', 'few': 'n in 2..4'})
    >>> func(1)
    'one'
    >>> func(3)
    'few'
    >>> func = to_python({'one': 'n in 1,11', 'few': 'n in 3..10,13..19'})
    >>> func(11)
    'one'
    >>> func(15)
    'few'

    :param rule: the rules as list or dict, or a `PluralRule` object
    :raise RuleError: if the expression is malformed
    )INZWITHINZMODr   zdef evaluate(n):z- n, i, v, w, f, t, c, e = extract_operands(n)z if (%s): return %rz
 return %r
z<rule>execZevaluate)in_range_listwithin_range_listcldr_modulor   _PythonCompilerr<   r   r8   r   r+   r   rO   r   eval)rP   	namespaceZto_python_funcrQ   r2   r*   coder   r   r   rE      s    
rE   c                s   t | } | jthB  t j} fddtD j}dt  g}x,| j	D ]"\}}|
d||||f  qHW |
d|t  d|S )a  The plural rule as gettext expression.  The gettext expression is
    technically limited to integers and returns indices rather than tags.

    >>> to_gettext({'one': 'n is 1', 'two': 'n is 2'})
    'nplurals=3; plural=((n == 1) ? 0 : (n == 2) ? 1 : 2);'

    :param rule: the rules as list or dict, or a `PluralRule` object
    :raise RuleError: if the expression is malformed
    c                s   g | ]}| kr|qS r   r   )r   r2   )	used_tagsr   r   r3      s    zto_gettext.<locals>.<listcomp>znplurals=%d; plural=(z
%s ? %d : z%d);r   )r   r8   rM   rO   _GettextCompilerr<   r&   indexr   r   r+   r   )rP   r9   Z
_get_indexrQ   r2   r*   r   )r]   r   
to_gettext   s    

r`   c             C   s   | t | kot| |S )a  Integer range list test.  This is the callback for the "in" operator
    of the UTS #35 pluralization rule language:

    >>> in_range_list(1, [(1, 3)])
    True
    >>> in_range_list(3, [(1, 3)])
    True
    >>> in_range_list(3, [(1, 3), (5, 8)])
    True
    >>> in_range_list(1.2, [(1, 4)])
    False
    >>> in_range_list(10, [(1, 4)])
    False
    >>> in_range_list(10, [(1, 4), (6, 8)])
    False
    )r   rW   )num
range_listr   r   r   rV      s    rV   c                s   t  fdd|D S )a  Float range test.  This is the callback for the "within" operator
    of the UTS #35 pluralization rule language:

    >>> within_range_list(1, [(1, 3)])
    True
    >>> within_range_list(1.0, [(1, 3)])
    True
    >>> within_range_list(1.2, [(1, 4)])
    True
    >>> within_range_list(8.8, [(1, 4), (7, 15)])
    True
    >>> within_range_list(10, [(1, 4)])
    False
    >>> within_range_list(10.5, [(1, 4), (20, 30)])
    False
    c             3   s"   | ]\}} |ko |kV  qd S )Nr   )r   Zmin_Zmax_)ra   r   r   r	   %  s    z$within_range_list.<locals>.<genexpr>)any)ra   rb   r   )ra   r   rW     s    rW   c             C   s@   d}| dk r| d9 } d}|dk r(|d9 }| | }|r<|d9 }|S )zJavaish modulo.  This modulo operator returns the value with the sign
    of the dividend rather than the divisor like Python does:

    >>> cldr_modulo(-3, 5)
    -3
    >>> cldr_modulo(-3, -5)
    -3
    >>> cldr_modulo(3, 5)
    3
    r      r   )abreverservr   r   r   rX   (  s    rX   c               @   s   e Zd ZdZdS )	RuleErrorzRaised if a rule is malformed.N)r5   rG   rH   rI   r   r   r   r   rj   ?  s   rj   r   r   r   r   r   r   r   r   z\s+wordz"\b(and|or|is|(?:with)?in|not|mod|[r   z])\bvaluez\d+symbolz%|,|!=|=ellipsisz\.{2,3}|\u2026c             C   s   |  dd } g }d}t| }xb||k rxTtD ]<\}}|| |}|d k	r.| }|rh||| f P q.W td| |  q W |d d d S )N@r   z5malformed CLDR pluralization rule.  Got unexpected %rrd   )splitr   _RULESmatchendr+   grouprj   )srQ   posrs   tokrP   rr   r   r   r   tokenize_ruleV  s    
rx   c             C   s,   | o*| d d |ko*|d kp*| d d |kS )Nrd   r   re   r   )tokenstype_rl   r   r   r   test_next_tokeni  s    r{   c             C   s   t | ||r|  S d S )N)r{   pop)ry   rz   rl   r   r   r   
skip_tokenn  s    r}   c             C   s
   d| ffS )Nrl   r   )rl   r   r   r   
value_nodes  s    r~   c             C   s   | dfS )Nr   r   )namer   r   r   
ident_nodew  s    r   c             C   s   d| fS )Nrb   r   )rb   r   r   r   range_list_node{  s    r   c             C   s
   d| ffS )Nnotr   )ri   r   r   r   negate  s    r   c               @   sb   e Zd ZdZdd Zd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S )r)   u  Internal parser.  This class can translate a single rule into an abstract
    tree of tuples. It implements the following grammar::

        condition     = and_condition ('or' and_condition)*
                        ('@integer' samples)?
                        ('@decimal' samples)?
        and_condition = relation ('and' relation)*
        relation      = is_relation | in_relation | within_relation
        is_relation   = expr 'is' ('not')? value
        in_relation   = expr (('not')? 'in' | '=' | '!=') range_list
        within_relation = expr ('not')? 'within' range_list
        expr          = operand (('mod' | '%') value)?
        operand       = 'n' | 'i' | 'f' | 't' | 'v' | 'w'
        range_list    = (range | value) (',' range_list)*
        value         = digit+
        digit         = 0|1|2|3|4|5|6|7|8|9
        range         = value'..'value
        samples       = sampleRange (',' sampleRange)* (',' ('…'|'...'))?
        sampleRange   = decimalValue '~' decimalValue
        decimalValue  = value ('.' value)?

    - Whitespace can occur between or around any of the above tokens.
    - Rules should be mutually exclusive; for a given numeric value, only one
      rule should apply (i.e. the condition should only be true for one of
      the plural rule elements).
    - The in and within relations can take comma-separated lists, such as:
      'n in 3,5,7..15'.
    - Samples are ignored.

    The translator parses the expression on instanciation into an attribute
    called `ast`.
    c             C   sD   t || _| jsd | _d S |  | _| jr@td| jd d  d S )NzExpected end of rule, got %rrd   re   )rx   ry   r*   	conditionrj   )r,   stringr   r   r   r1     s    

z_Parser.__init__Nc             C   sf   t | j||}|d k	r|S |d kr6t|d kr0|p2|}| jsHtd| td|| jd d f d S )Nz#expected %s but end of rule reachedzexpected %s but got %rrd   re   )r}   ry   reprrj   )r,   rz   rl   Ztermtokenr   r   r   expect  s    z_Parser.expectc             C   s0   |   }x"t| jddr*d||   ff}q
W |S )Nrk   or)and_conditionr}   ry   )r,   opr   r   r   r     s    z_Parser.conditionc             C   s0   |   }x"t| jddr*d||   ff}q
W |S )Nrk   and)relationr}   ry   )r,   r   r   r   r   r     s    z_Parser.and_conditionc             C   s   |   }t| jddr8t| jddr(dp*d||  ffS t| jdd}d}t| jddr^d}n$t| jdds|rxtd| |S d|||  ff}|rt|S |S )	Nrk   isr   ZisnotinZwithinz#Cannot negate operator based rules.r   )r0   r}   ry   rl   rj   newfangled_relationrb   r   )r,   leftnegatedmethodri   r   r   r   r     s    
z_Parser.relationc             C   sR   t | jddrd}nt | jddr(d}ntddd||  ff}|rNt|S |S )	Nrm   =Fz!=Tz'Expected "=" or "!=" or legacy relationr   r   )r}   ry   rj   rb   r   )r,   r   r   ri   r   r   r   r     s    z_Parser.newfangled_relationc             C   s,   |   }t| jdr ||   fS ||fS d S )Nrn   )rl   r}   ry   )r,   r   r   r   r   range_or_value  s    z_Parser.range_or_valuec             C   s4   |   g}x t| jddr*||    qW t|S )Nrm   ,)r   r}   ry   r+   r   )r,   rb   r   r   r   rb     s    
z_Parser.range_listc             C   s|   t | jd}|d ks |d tkr(td|d }t | jddrRd|df|  ffS t | jddrtd|df|  ffS t|S )Nrk   re   zExpected identifier variablemodr   rm   %)r}   ry   _VARSrj   rl   r   )r,   rk   r   r   r   r   r0     s    z_Parser.exprc             C   s   t t| dd S )Nrl   re   )r~   r   r   )r,   r   r   r   rl     s    z_Parser.value)NN)r5   rG   rH   rI   r1   r   r   r   r   r   r   rb   r0   rl   r   r   r   r   r)     s    


r)   c                s    fddS )z%Compiler factory for the `_Compiler`.c                s    |  ||  |f S )N)r<   )r,   lr)tmplr   r   r?      r@   z"_binary_compiler.<locals>.<lambda>r   )r   r   )r   r   _binary_compiler  s    r   c                s    fddS )z%Compiler factory for the `_Compiler`.c                s    |  | S )N)r<   )r,   r>   )r   r   r   r?     r@   z!_unary_compiler.<locals>.<lambda>r   )r   r   )r   r   _unary_compiler  s    r   c             C   s   dS )Nr
   r   )r>   r   r   r   r?     r@   r?   c               @   s   e Zd Z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edZedZedZedZedZedZdd ZdS )	_CompilerzZThe compilers are able to transform the expressions into multiple
    output formats.
    c             C   s   |\}}t | d| | S )NZcompile_)getattr)r,   argr   argsr   r   r   r<     s    z_Compiler.compilec             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   z_Compiler.<lambda>c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   dS )Nr   r   )r>   r   r   r   r?     r@   c             C   s   t |S )N)r   )r>   r   r   r   r   r?     r@   z
(%s && %s)z
(%s || %s)z(!%s)z
(%s %% %s)z
(%s == %s)z
(%s != %s)c             C   s
   t  d S )N)NotImplementedError)r,   r   r0   rb   r   r   r   compile_relation$  s    z_Compiler.compile_relationN)r5   rG   rH   rI   r<   	compile_n	compile_i	compile_v	compile_w	compile_f	compile_tZ	compile_cZ	compile_eZcompile_valuer   compile_and
compile_orr   compile_notcompile_mod
compile_iscompile_isnotr   r   r   r   r   r     s$   r   c               @   s8   e Zd ZdZedZedZedZedZ	dd Z
dS )	rY   z!Compiles an expression to Python.z(%s and %s)z
(%s or %s)z(not %s)zMOD(%s, %s)c                s8   dd  fdd|d D  }d|  ||f S )Nz[%s]r   c                s    g | ]}d t t j| qS )z(%s, %s))tuplemapr<   )r   Zrange_)r,   r   r   r3   2  s   z4_PythonCompiler.compile_relation.<locals>.<listcomp>re   z
%s(%s, %s))r   upperr<   )r,   r   r0   rb   Zcompile_range_listr   )r,   r   r   0  s
    
z _PythonCompiler.compile_relationN)r5   rG   rH   rI   r   r   r   r   r   r   r   r   r   r   r   rY   (  s   rY   c               @   s.   e Zd ZdZejZeZeZ	eZ
eZdd ZdS )r^   z)Compile into a gettext plural expression.c          	   C   s   g }|  |}xd|d D ]X}|d |d krJ|d||  |d f  qt| j |\}}|d||||f  qW dd| S )Nre   r   z
(%s == %s)z(%s >= %s && %s <= %s)z(%s)z || )r<   r+   r   r   )r,   r   r0   rb   ri   itemminmaxr   r   r   r   A  s    
z!_GettextCompiler.compile_relationN)r5   rG   rH   rI   r   r   r   compile_zeror   r   r   r   r   r   r   r   r   r^   8  s   r^   c               @   s0   e Zd ZdZdd ZeZeZeZeZ	dd Z
dS )rN   z/Compiles the expression to plain of JavaScript.c             C   s   dS )NzparseInt(n, 10)r   )r>   r   r   r   r?   Z  r@   z_JavaScriptCompiler.<lambda>c             C   s4   t | |||}|dkr0| |}d|||f }|S )Nr   z(parseInt(%s, 10) == %s && %s))r^   r   r<   )r,   r   r0   rb   r\   r   r   r   r   `  s    
z$_JavaScriptCompiler.compile_relationN)r5   rG   rH   rI   r   r   r   r   r   r   r   r   r   r   r   rN   U  s   rN   c               @   sJ   e Zd ZdZedZedZedZedZedZ	dd Z
dd
dZdS )r;   z+Returns a unicode pluralization rule again.z%s is %sz%s is not %sz	%s and %sz%s or %sz	%s mod %sc             C   s   | j |d ddiS )Nre   r   T)r   )r,   r   r   r   r   r   v  s    z_UnicodeCompiler.compile_notFc             C   sz   g }xP|d D ]D}|d |d kr8| | |d  q| dtt| j|  qW d| ||rhdpjd|d|f S )Nre   r   z%s..%sz
%s%s %s %sz notr   r   )r+   r<   r   r   r   )r,   r   r0   rb   r   Zrangesr   r   r   r   r   y  s    z!_UnicodeCompiler.compile_relationN)F)r5   rG   rH   rI   r   r   r   r   r   r   r   r   r   r   r   r   r;   i  s   r;   )N)N)$rI   r   rer&   rO   r   r   rR   rE   r`   rV   rW   rX   	Exceptionrj   r   r<   UNICODEr   rq   rx   r{   r}   r~   r   r   r   r)   r   r   r   r   rY   r^   rN   r;   r   r   r   r   <module>	   sT   ;](

{