a
    hP                     @   s:  d dl Z d dlZd dlZd dlZd dlZd dlmZmZmZ d dl	m
Z
 d dlmZ d dlmZ d dlmZ d dlmZ d dlmZmZmZmZmZmZmZ d dlZed	eZed
eZdd Zeeegef dddZ dde j!e j!ddeeeee f  ee eeg ef ee" e j#dddZ$G dd deZ%dS )    N)ArgumentDefaultsHelpFormatterArgumentParserArgumentTypeError)Iterable)copy)Enum)isclassPath)AnyCallableLiteralNewTypeOptionalUnionget_type_hints	DataClassDataClassTypec                 C   sB   t | tr| S |  dv rdS |  dv r.dS td|  dd S )N)yestruety1T)nofalsefn0FzTruthy value expected: got zJ but expected one of yes/no, true/false, t/f, y/n, 1/0 (case insensitive).)
isinstanceboollowerr   )v r"   U/var/www/html/assistant/venv/lib/python3.9/site-packages/transformers/hf_argparser.pystring_to_bool$   s    

r$   )choicesreturnc                    s   dd | D   fddS )aN  
    Creates a mapping function from each choices string representation to the actual value. Used to support multiple
    value types for a single argument.

    Args:
        choices (list): List of choices.

    Returns:
        Callable[[str], Any]: Mapping function from string representation to actual value for each choice.
    c                 S   s   i | ]}t ||qS r"   )str).0choicer"   r"   r#   
<dictcomp><       z-make_choice_type_function.<locals>.<dictcomp>c                    s     | | S )N)get)argZstr_to_choicer"   r#   <lambda>=   r+   z+make_choice_type_function.<locals>.<lambda>r"   )r%   r"   r.   r#   make_choice_type_function1   s    r0   )aliaseshelpdefaultdefault_factorymetadata)r1   r2   r3   r4   r5   r&   c                 K   sD   |du ri }| dur| |d< |dur,||d< t jf |||d|S )a  Argument helper enabling a concise syntax to create dataclass fields for parsing with `HfArgumentParser`.

    Example comparing the use of `HfArg` and `dataclasses.field`:
    ```
    @dataclass
    class Args:
        regular_arg: str = dataclasses.field(default="Huggingface", metadata={"aliases": ["--example", "-e"], "help": "This syntax could be better!"})
        hf_arg: str = HfArg(default="Huggingface", aliases=["--example", "-e"], help="What a nice syntax!")
    ```

    Args:
        aliases (Union[str, list[str]], optional):
            Single string or list of strings of aliases to pass on to argparse, e.g. `aliases=["--example", "-e"]`.
            Defaults to None.
        help (str, optional): Help string to pass on to argparse that can be displayed with --help. Defaults to None.
        default (Any, optional):
            Default value for the argument. If not default or default_factory is specified, the argument is required.
            Defaults to dataclasses.MISSING.
        default_factory (Callable[[], Any], optional):
            The default_factory is a 0-argument function called to initialize a field's value. It is useful to provide
            default values for mutable types, e.g. lists: `default_factory=list`. Mutually exclusive with `default=`.
            Defaults to dataclasses.MISSING.
        metadata (dict, optional): Further metadata to pass on to `dataclasses.field`. Defaults to None.

    Returns:
        Field: A `dataclasses.Field` with the desired properties.
    Nr1   r2   )r5   r3   r4   )dataclassesfield)r1   r2   r3   r4   r5   kwargsr"   r"   r#   HfArg@   s    $r9   c                       s   e Zd ZU dZee ed< deeeee f  d fddZ	e
eejddd	Zed
ddZdeedf dddZdeeef eeedf dddZdeeejf eeedf dddZd eeejf eeedf dddZ  ZS )!HfArgumentParsera  
    This subclass of `argparse.ArgumentParser` uses type hints on dataclasses to generate arguments.

    The class is designed to play well with the native argparse. In particular, you can add more (non-dataclass backed)
    arguments to the parser after initialization and you'll get the output back after parsing as an additional
    namespace. Optional: To create sub argument groups use the `_argument_group_name` attribute in the dataclass.

    Args:
        dataclass_types (`DataClassType` or `Iterable[DataClassType]`, *optional*):
            Dataclass type, or list of dataclass types for which we will "fill" instances with the parsed args.
        kwargs (`dict[str, Any]`, *optional*):
            Passed to `argparse.ArgumentParser()` in the regular way.
    dataclass_typesN)r;   c                    st   |d u rg }nt |ts|g}d|vr.t|d< t jf i | t|rP|g}t|| _| jD ]}| 	| q`d S )Nformatter_class)
r   r   r   super__init__r6   is_dataclasslistr;   _add_dataclass_arguments)selfr;   r8   dtype	__class__r"   r#   r>      s    



zHfArgumentParser.__init__)parserr7   c                 C   s  d|j  g}d|j v r2|d|j dd  |j }t|jtrPtd|	dg }t|trl|g}t
|jd|j}|tu sttdrxt|tjrxt|jjvrt|jjdkstd |jjvrtd	|j  d
td |jjvr,|jjd tu r|jjd n
|jjd |_t
|jd|j}nLt|jjvrxtd |jjd rZ|jjd n
|jjd |_t
|jd|j}i }|tu st|jtrt|jtr|tu r|jj|d< ndd |jD |d< t|d |d< |jtjur|j|d< nd|d< n.|jtu s |jtt krt|}t|d< |jtu sV|jd ur2|jtjur2|jtju rhdn|j}||d< d|d< d|d< nt|rt|tr|jjd |d< d|d< |jtjur| |d< n|jtju r2d|d< nH|j|d< |jtjur|j|d< n$|jtjur*| |d< nd|d< | jg ||R i | |jdu r|jtu st|jtt krd|d< | jd|j  d|j dd fd|j d| d S )Nz--_-zpUnresolved type detected, which should have been done with the help of `typing.get_type_hints` method by defaultr1   
__origin__	UnionType   zOnly `Union[X, NoneType]` (i.e., `Optional[X]`) is allowed for `Union` because the argument parser only supports one type per argument. Problem encountered in field 'z'.   r   r%   c                 S   s   g | ]
}|j qS r"   )value)r(   xr"   r"   r#   
<listcomp>   r+   z;HfArgumentParser._parse_dataclass_field.<locals>.<listcomp>typer3   TrequiredF?nargsconst+z--no_z--no-store_false)actiondest) nameappendreplacer5   r   r   rP   r'   RuntimeErrorpopgetattrr   hasattrtypesrJ   __args__len
ValueErrorr   r   
issubclassr   r0   r3   r6   MISSINGr   r$   r   r@   r4   add_argument)rF   r7   Zlong_optionsr8   r1   Zorigin_typeZbool_kwargsr3   r"   r"   r#   _parse_dataclass_field   s    


",*&
&


(
z'HfArgumentParser._parse_dataclass_field)rC   c              
   C   s   t |dr| |j}n| }zt|}W n tyJ   td| dY nv ty } z^tjd d dk rdt	|v rd
tt	tjd d }td| d	| d
| W Y d }~n
d }~0 0 t|D ]$}|jsq||j |_| || qd S )N_argument_group_namezType resolution failed for z. Try declaring the class in global scope or removing line of `from __future__ import annotations` which opts in Postponed Evaluation of Annotations (PEP 563)rK   )   
   z!unsupported operand type(s) for |.ri   z on Python a6  . Try removing line of `from __future__ import annotations` which opts in union types as `X | Y` (PEP 604) via Postponed Evaluation of Annotations (PEP 563). To support Python versions that lower than 3.10, you need to use `typing.Union[X, Y]` instead of `X | Y` and `typing.Optional[X]` instead of `X | None`.)r_   add_argument_grouprh   r   	NameErrorr\   	TypeErrorsysversion_infor'   joinmapr6   fieldsinitrY   rP   rg   )rB   rC   rF   Z
type_hintsexpython_versionr7   r"   r"   r#   rA      s.    

z)HfArgumentParser._add_dataclass_argumentsFT.)r&   c                    s  |s|s|rt tjrg }|r.|t| n(|rVt tjrV|ttjd d |rt }|j|tdd |j	|d\}}t
||dd}	|	r|dd	 |	D  g }
|D ]}| r|
|  7 }
q|dur|
| n|
tjd
d  }| j	|d\}}g }| jD ]b}dd t|D   fddt
| D } D ]}t|| qF|f i |}|| qt |jdkr|| |rg ||R S |rtd| g |R S dS )a%  
        Parse command-line args into instances of the specified dataclass types.

        This relies on argparse's `ArgumentParser.parse_known_args`. See the doc at:
        docs.python.org/3/library/argparse.html#argparse.ArgumentParser.parse_args

        Args:
            args:
                List of strings to parse. The default is taken from sys.argv. (same as argparse.ArgumentParser)
            return_remaining_strings:
                If true, also return a list of remaining argument strings.
            look_for_args_file:
                If true, will look for a ".args" file with the same base name as the entry point script for this
                process, and will append its potential content to the command line args.
            args_filename:
                If not None, will uses this file instead of the ".args" file specified in the previous argument.
            args_file_flag:
                If not None, will look for a file in the command-line args specified with this flag. The flag can be
                specified multiple times and precedence is determined by the order (last one wins).

        Returns:
            Tuple consisting of:

                - the dataclass instances in the same order as they were passed to the initializer.abspath
                - if applicable, an additional namespace for more (non-dataclass backed) arguments added to the parser
                  after initialization.
                - The potential list of remaining argument strings. (same as argparse.ArgumentParser.parse_known_args)
        r   z.argsrZ   )rP   rW   )argsrH   Nc                 S   s   g | ]}t |qS r"   r	   )r(   pr"   r"   r#   rO   U  r+   z@HfArgumentParser.parse_args_into_dataclasses.<locals>.<listcomp>rL   c                 S   s   h | ]}|j r|jqS r"   rt   rY   r(   r   r"   r"   r#   	<setcomp>b  r+   z?HfArgumentParser.parse_args_into_dataclasses.<locals>.<setcomp>c                    s   i | ]\}}| v r||qS r"   r"   r(   kr!   keysr"   r#   r*   c  r+   z@HfArgumentParser.parse_args_into_dataclasses.<locals>.<dictcomp>z?Some specified arguments are not used by the HfArgumentParser: )rb   ro   argvrZ   r
   with_suffixr   rf   r'   parse_known_argsvarsr,   lstripextendexists	read_textsplitr;   r6   rs   itemsdelattr__dict__rc   )rB   rw   Zreturn_remaining_stringsZlook_for_args_fileZargs_filenameZargs_file_flagZ
args_filesZargs_file_parsercfgZcmd_args_file_pathsZ	file_args	args_file	namespaceremaining_argsoutputsrC   inputsr}   objr"   r~   r#   parse_args_into_dataclasses  sD    %"

z,HfArgumentParser.parse_args_into_dataclasses)rw   allow_extra_keysr&   c                    s   t | }g }| jD ]T}dd t|D   fdd| D }||  |f i |}|| q|s|rtdt	| t
|S )a<  
        Alternative helper method that does not use `argparse` at all, instead uses a dict and populating the dataclass
        types.

        Args:
            args (`dict`):
                dict containing config values
            allow_extra_keys (`bool`, *optional*, defaults to `False`):
                Defaults to False. If False, will raise an exception if the dict contains keys that are not parsed.

        Returns:
            Tuple consisting of:

                - the dataclass instances in the same order as they were passed to the initializer.
        c                 S   s   h | ]}|j r|jqS r"   ry   rz   r"   r"   r#   r{     r+   z.HfArgumentParser.parse_dict.<locals>.<setcomp>c                    s   i | ]\}}| v r||qS r"   r"   r|   r~   r"   r#   r*     r+   z/HfArgumentParser.parse_dict.<locals>.<dictcomp>z0Some keys are not used by the HfArgumentParser: )setr   r;   r6   rs   r   difference_updaterZ   rc   sortedtuple)rB   rw   r   Zunused_keysr   rC   r   r   r"   r~   r#   
parse_dicts  s    
zHfArgumentParser.parse_dict)	json_filer   r&   c                 C   sT   t t|dd}t| }W d   n1 s40    Y  | j||d}t|S )at  
        Alternative helper method that does not use `argparse` at all, instead loading a json file and populating the
        dataclass types.

        Args:
            json_file (`str` or `os.PathLike`):
                File name of the json file to parse
            allow_extra_keys (`bool`, *optional*, defaults to `False`):
                Defaults to False. If False, will raise an exception if the json file contains keys that are not
                parsed.

        Returns:
            Tuple consisting of:

                - the dataclass instances in the same order as they were passed to the initializer.
        zutf-8)encodingNr   )openr
   jsonloadsreadr   r   )rB   r   r   Zopen_json_filedatar   r"   r"   r#   parse_json_file  s    ,z HfArgumentParser.parse_json_file)	yaml_filer   r&   c                 C   s$   | j tt| |d}t|S )at  
        Alternative helper method that does not use `argparse` at all, instead loading a yaml file and populating the
        dataclass types.

        Args:
            yaml_file (`str` or `os.PathLike`):
                File name of the yaml file to parse
            allow_extra_keys (`bool`, *optional*, defaults to `False`):
                Defaults to False. If False, will raise an exception if the json file contains keys that are not
                parsed.

        Returns:
            Tuple consisting of:

                - the dataclass instances in the same order as they were passed to the initializer.
        r   )r   yamlZ	safe_loadr
   r   r   )rB   r   r   r   r"   r"   r#   parse_yaml_file  s    z HfArgumentParser.parse_yaml_file)N)NFTNN)F)F)F)__name__
__module____qualname____doc__r   r   __annotations__r   r   r>   staticmethodr   r6   Fieldrg   rA   r   r   r   dictr'   r   r   r   osPathLiker   r   __classcell__r"   r"   rD   r#   r:   o   s2   
$i$     
V$ 
 
r:   )&r6   r   r   ro   r`   argparser   r   r   collections.abcr   r   enumr   inspectr   pathlibr
   typingr   r   r   r   r   r   r   r   r   r   r$   r@   r'   r0   re   r   r   r9   r:   r"   r"   r"   r#   <module>   s<   $


/