
    i                   Z   d Z ddlmZ ddlZ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mZmZ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mZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'm(Z(m)Z)  ej*        e+          Z,dPdZ-dZ.dZ/dZ0dZ1dZ2dZ3dZ4dZ5dZ6e3e4e5e6hZ7dZ8dZ9dZ: e;h d          Z<e
 G d d                      Z=dQdZ>dRd"Z?dSd%Z@dTd(ZAdUd,ZBdVd.ZCdWd2ZDdXd4ZEdYd6ZFdZd[d8ZGd\d;ZHd]d=ZId^d?ZJd_dAZKdBZL G dC dD          ZMd`dFZNdadGZOdbdIZPdbdJZQdcdMZRdddNZSdedOZTdS )fz<Persistent multi-credential pool for same-provider failover.    )annotationsN)	dataclassfieldsreplacedatetime)AnyDictListOptionalSetTuple)OPENROUTER_BASE_URL)get_env_value)'CODEX_ACCESS_TOKEN_REFRESH_SKEW_SECONDS!DEFAULT_AGENT_KEY_MIN_TTL_SECONDSPROVIDER_REGISTRY_auth_store_lock_codex_access_token_is_expiring_decode_jwt_claims_load_auth_store_load_provider_state_resolve_kimi_base_url_resolve_zai_base_url_save_auth_store_save_provider_stateread_credential_poolwrite_credential_poolreturnOptional[dict]c                 F    	 ddl m}   |             S # t          $ r Y dS w xY w)z.Load config.yaml, returning None on any error.r   load_configN)hermes_cli.configr#   	Exceptionr"   s    </home/longshao/.hermes/hermes-agent/agent/credential_pool.py_load_config_safer'   &   sG    111111{}}   tts    
  ok	exhaustedoauthapi_keymanual
fill_firstround_robinrandom
least_usedi  zcustom:>   tlsscope	client_id
expires_in
token_typeobtained_atagent_key_idportal_base_urlagent_key_reusedagent_key_expires_inagent_key_obtained_atc                     e Zd ZU ded<   ded<   ded<   ded<   ded<   ded<   ded	<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   d
Z	ded<   d
Z
ded<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   d
Zded<   dZded<   d
Zded<   d  Zd*d"Zed+d&            Zd,d'Zed-d(            Zed.d)            Zd
S )/PooledCredentialstrprovideridlabel	auth_typeintprioritysourceaccess_tokenNOptional[str]refresh_tokenlast_statusOptional[float]last_status_atOptional[int]last_error_codelast_error_reasonlast_error_messagelast_error_reset_atbase_url
expires_atexpires_at_mslast_refreshinference_base_url	agent_keyagent_key_expires_atr   request_countDict[str, Any]extrac                &    | j         	i | _         d S d S N)rZ   selfs    r&   __post_init__zPooledCredential.__post_init__s   s    :DJJJ     namec                    |t           v r| j                            |          S t          dt	          |           j         d|          )N'z' object has no attribute )_EXTRA_KEYSrZ   getAttributeErrortype__name__)r^   ra   s     r&   __getattr__zPooledCredential.__getattr__w   sJ    ;:>>$'''Xd!4XXPTXXYYYr`   payloadr   'PooledCredential'c                   d t          |           D             }fd|D             }fdt          D             }||d<   |                    dt          j                    j        d d                    |                    d                    d|                     |                    d	t                     |                    d
d           |                    dt                     |                    dd            | dd|i|S )Nc                2    h | ]}|j         d k    |j         S )r?   ra   .0fs     r&   	<setcomp>z-PooledCredential.from_dict.<locals>.<setcomp>~   s&    KKK!af
6J6Jqv6J6J6Jr`   c                D    i | ]}|v |                     |          S  )re   rp   krj   s     r&   
<dictcomp>z.PooledCredential.from_dict.<locals>.<dictcomp>   s*    GGGa!w,,7;;q>>,,,r`   c                :    i | ]}|v |         ||         S r\   rt   ru   s     r&   rw   z.PooledCredential.from_dict.<locals>.<dictcomp>   s1    ___1qG||PQ
H^GAJH^H^H^r`   rZ   r@      rA   rE   rB   rD   r   rF    r?   rt   )	r   rd   
setdefaultuuiduuid4hexre   AUTH_TYPE_API_KEYSOURCE_MANUAL)clsr?   rj   field_namesdatarZ   s     `   r&   	from_dictzPooledCredential.from_dict|   s   KKvc{{KKKGGGG;GGG_______Wdjll.rr2333Xx!@!@AAA%6777
A&&&-000+++s--H----r`   c                    h d}i }t          |           D ]6}|j        dv rt          | |j                  }|	|j        |v r
|||j        <   7| j                                        D ]\  }}||||<   |S )N>   rI   rK   rM   rN   rO   rP   )r?   rZ   )r   ra   getattrrZ   items)r^   _ALWAYS_EMITresult	field_defvaluerv   vs          r&   to_dictzPooledCredential.to_dict   s    
 
 
 "$ 	/ 	/I~!666D).11E INl$B$B).y~&J$$&& 	 	DAq}q	r`   c                ~    | j         dk    rt          | j        p| j        pd          S t          | j        pd          S )Nnousrz   )r?   r>   rV   rF   r]   s    r&   runtime_api_keyz PooledCredential.runtime_api_key   sB    =F""t~@):@bAAA4$*+++r`   c                B    | j         dk    r| j        p| j        S | j        S )Nr   )r?   rU   rQ   r]   s    r&   runtime_base_urlz!PooledCredential.runtime_base_url   s'    =F""*;dm;}r`   )ra   r>   )r?   r>   rj   rY   r   rk   )r   rY   )r   r>   )r   rG   )rh   
__module____qualname____annotations__rH   rI   rK   rM   rN   rO   rP   rQ   rR   rS   rT   rU   rV   rW   rX   rZ   r_   ri   classmethodr   r   propertyr   r   rt   r`   r&   r=   r=   Y   s        MMMGGGJJJNNNMMMKKK#'M''''!%K%%%%&*N****%)O))))'+++++(,,,,,+/////"H"""" $J$$$$#'M''''"&L&&&&(,,,,,#I####*.....M E      Z Z Z Z
 . . . [.   * , , , X,
    X  r`   r=   tokenr>   fallbackc                    t          |           }dD ]V}|                    |          }t          |t                    r*|                                r|                                c S W|S )N)emailpreferred_usernameupn)r   re   
isinstancer>   strip)r   r   claimskeyr   s        r&   label_from_tokenr      sh    &&F5 ! !

3eS!! 	!ekkmm 	!;;==   Or`   entriesList[PooledCredential]rC   c                >    t          d | D             d          dz   S )Nc              3  $   K   | ]}|j         V  d S r\   rD   rp   entrys     r&   	<genexpr>z!_next_priority.<locals>.<genexpr>   s$      445444444r`   )default   )max)r   s    r&   _next_priorityr      s)    44G444bAAAAEEr`   rE   boolc                    | pd                                                                 }|t          k    p|                    t           d          S )Nrz   :)r   lowerr   
startswith)rE   
normalizeds     r&   _is_manual_sourcer      sJ    ,B%%''--//J&T**?*?=@S@S@S*T*TTr`   
error_coderL   c                *    | dk    rt           S t          S )zHReturn cooldown seconds based on the HTTP status that caused exhaustion.i  )EXHAUSTED_TTL_429_SECONDSEXHAUSTED_TTL_DEFAULT_SECONDS)r   s    r&   _exhausted_ttlr      s    S((((r`   r   r	   rJ   c                   | | dk    rdS t          | t          t          f          r$t          |           }|dk    rdS |dk    r|dz  n|S t          | t                    r|                                 }|sdS 	 t          |          }n# t
          $ r d}Y nw xY w||dk    r|dz  n|S 	 t          j        |                    dd                    	                                S # t
          $ r Y dS w xY wdS )zBest-effort parse for provider reset timestamps.

    Accepts epoch seconds, epoch milliseconds, and ISO-8601 strings.
    Returns seconds since epoch.
    Nrz   r   l    J)     @@Zz+00:00)
r   rC   floatr>   r   
ValueErrorr   fromisoformatr   	timestamp)r   numericraws      r&   _parse_absolute_timestampr      s;    }t%#u&& L,,a<<4#*->#>#>wGK% kkmm 	4	CjjGG 	 	 	GGG	'.1B'B'B7V##O	)#++c8*D*DEEOOQQQ 	 	 	44	4s$   9B	 	BB+9C% %
C32C3messagec                   | sd S t          j        d| t           j                  }|rTt          |                    d                    }|                    d                                          dk    r|dz  n|S t          j        d| t           j                  }|r"t          |                    d                    S d S )Nz,quotaResetDelay[:\s\"]+(\d+(?:\.\d+)?)(ms|s)r      msr   z?retry\s+(?:after\s+)?(\d+(?:\.\d+)?)\s*(?:sec|secs|seconds|s\b))research
IGNORECASEr   groupr   )r   delay_matchr   	sec_matchs       r&   _extract_retry_delay_secondsr      s     t)KWVXVcddK Qk''**++!,!2!21!5!5!;!;!=!=!E!Euv~~5P	\^egigtuuI )Y__Q''(((4r`   error_contextOptional[Dict[str, Any]]rY   c                   t          | t                    si S i }|                     d          }t          |t                    r+|                                r|                                |d<   |                     d          }t          |t                    r+|                                r|                                |d<   |                     d          p)|                     d          p|                     d          }t          |          }|<t          |t                    r't          |          }|t          j                    |z   }|||d<   |S )Nreasonr   reset_at	resets_atretry_until)r   dictre   r>   r   r   r   time)r   r   r   r   r   parsed_reset_atretry_delay_secondss          r&   _normalize_error_contextr      sK   mT** 	!#Jx((F&# .6<<>> .%||~~
8	**G'3 0GMMOO 0 '
9*%% 	,[))	,]++ 
 099O:gs#;#;:7CC*"ikk,??O"!0
:r`   r   c                    | j         t          k    rd S t          t          | dd                     }||S | j        r| j        t          | j                  z   S d S )NrP   )rI   STATUS_EXHAUSTEDr   r   rK   r   rM   )r   r   s     r&   _exhausted_untilr   	  se    ,,,t(8Mt)T)TUUH L#nU5J&K&KKK4r`   ra   c                v    |                                                                                      dd          S )z>Normalize a custom provider name for use as a pool key suffix. -)r   r   r   rn   s    r&   _normalize_custom_pool_namer     s,    ::<<''S111r`   configc              #    K   | t                      } | dS |                     d          }t          |t                    s$	 ddlm}  ||           }n# t          $ r Y dS w xY w|sdS |D ]V}t          |t                    s|                    d          }t          |t                    sCt          |          |fV  WdS )zJYield (normalized_name, entry_dict) for each valid custom_providers entry.Ncustom_providersr   )get_compatible_custom_providersra   )
r'   re   r   listr$   r   r%   r   r>   r   )r   r   r   r   ra   s        r&   _iter_custom_providersr     s     ~"$$~zz"455&-- 	IIIIII>>vFF 	 	 	FF	 ! 7 7%&& 	yy  $$$ 	)$//666667 7s   A 
A"!A"rQ   rG   c                @   | sdS |                                                      d          }t                      D ]c\  }}t          |                    d          pd                                                               d          }|r||k    rt
           | c S ddS )zLook up the custom_providers list in config.yaml and return 'custom:<name>' for a matching base_url.

    Returns None if no match is found.
    N/rQ   rz   )r   rstripr   r>   re   CUSTOM_POOL_PREFIX)rQ   normalized_url	norm_namer   	entry_urls        r&   get_custom_provider_pool_keyr   3  s    
  t^^%%,,S11N244 6 6	5		*--344::<<CCCHH	 	6n44(5)555554r`   	List[str]c                 X     t          d           t           fd D                       S )z?Return all 'custom:*' pool keys that have entries in auth.json.Nc              3     K   | ]P}|                     t                    t                              |          t                    D|         L|V  Qd S r\   )r   r   r   re   r   )rp   r   	pool_datas     r&   r   z-list_custom_pool_providers.<locals>.<genexpr>E  sx        >>,-- y}}S))400 cN	     r`   )r   sorted)r   s   @r&   list_custom_pool_providersr   B  sH    $T**I          r`   pool_keyc                    |                      t                    sdS | t          t                    d         }t                      D ]\  }}||k    r|c S dS )zWReturn the custom_providers config entry matching a pool key like 'custom:together.ai'.N)r   r   lenr   )r   suffixr   r   s       r&   _get_custom_provider_configr   M  so    122 tc,--../F244  	5LLL 4r`   r?   c                H   t                      }|t          S |                    d          }t          |t                    st          S t          |                    | d          pd                                                                          }|t          v r|S t          S )z8Return the configured selection strategy for a provider.Ncredential_pool_strategiesrz   )	r'   STRATEGY_FILL_FIRSTre   r   r   r>   r   r   SUPPORTED_POOL_STRATEGIES)r?   r   
strategiesstrategys       r&   get_pool_strategyr  X  s      F~""899Jj$'' #"":>>(B//5266<<>>DDFFH,,,r`   r   c                      e 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@dAdZ
dBdZdBdZdBdZdCdZdDdZdEd Zd=d!Zd"d"d#dFd&Zd=d'Zd=d(Zdd)dGd*Zd@dHd-ZdId.Zd=d/Zd=d0ZdJd2ZdKd4ZdLd8ZdBd9ZdS )MCredentialPoolr?   r>   r   r   c                    || _         t          |d           | _        d | _        t	          |          | _        t          j                    | _        i | _	        t          | _        d S )Nc                    | j         S r\   r   r   s    r&   <lambda>z)CredentialPool.__init__.<locals>.<lambda>n  s    %. r`   r   )r?   r   _entries_current_idr  	_strategy	threadingLock_lock_active_leases%DEFAULT_MAX_CONCURRENT_PER_CREDENTIAL_max_concurrent)r^   r?   r   s      r&   __init__zCredentialPool.__init__l  s]     w,H,HIII*.*844^%%
.0Dr`   r   r   c                *    t          | j                  S r\   )r   r  r]   s    r&   has_credentialszCredentialPool.has_credentialsu      DM"""r`   c                D    t          |                                           S )zCTrue if at least one entry is not currently in exhaustion cooldown.)r   _available_entriesr]   s    r&   has_availablezCredentialPool.has_availablex  s    D++--...r`   c                *    t          | j                  S r\   )r   r  r]   s    r&   r   zCredentialPool.entries|  r  r`   Optional[PooledCredential]c                X      j         sd S t           fd j        D             d           S )Nc              3  <   K   | ]}|j         j        k    |V  d S r\   )r@   r  rp   r   r^   s     r&   r   z)CredentialPool.current.<locals>.<genexpr>  s2      VVuTEU9U9UU9U9U9U9UVVr`   )r  nextr  r]   s   `r&   currentzCredentialPool.current  s;     	4VVVVVVVX\]]]r`   oldr=   newNonec                t    t          | j                  D ]"\  }}|j        |j        k    r|| j        |<    dS #dS )z4Swap an entry in-place by id, preserving sort order.N)	enumerater  r@   )r^   r"  r#  idxr   s        r&   _replace_entryzCredentialPool._replace_entry  sP    #DM22 	 	JCx36!!%(c" "	 	r`   c                N    t          | j        d | j        D                        d S )Nc                6    g | ]}|                                 S rt   r   r   s     r&   
<listcomp>z+CredentialPool._persist.<locals>.<listcomp>  s     888U]]__888r`   )r   r?   r  r]   s    r&   _persistzCredentialPool._persist  s6    M88$-888	
 	
 	
 	
 	
r`   Nr   status_coderL   r   r   c                B   t          |          }t          |t          t          j                    ||                    d          |                    d          |                    d                    }|                     ||           |                                  |S )Nr   r   r   rI   rK   rM   rN   rO   rP   )r   r   r   r   re   r(  r-  )r^   r   r.  r   normalized_errorupdateds         r&   _mark_exhaustedzCredentialPool._mark_exhausted  s     4MBB(9;;'.228<</33I>> 0 4 4Z @ @
 
 
 	E7+++r`   c           	     (   | j         dk    s|j        dk    r|S 	 ddlm}  |            }|s|S |                    dd          }|                    dd          }|                    dd          }|rm||j        k    rbt                              d	|j                   t          ||||d
d
d
          }| 
                    ||           |                                  |S n2# t          $ r%}t                              d|           Y d
}~nd
}~ww xY w|S )a  Sync a claude_code pool entry from ~/.claude/.credentials.json if tokens differ.

        OAuth refresh tokens are single-use. When something external (e.g.
        Claude Code CLI, or another profile's pool) refreshes the token, it
        writes the new pair to ~/.claude/.credentials.json. The pool entry's
        refresh token becomes stale. This method detects that and syncs.
        	anthropicclaude_coder   )read_claude_code_credentialsrefreshTokenrz   accessToken	expiresAtzKPool entry %s: syncing tokens from credentials file (refresh token changed)NrF   rH   rS   rI   rK   rM   z(Failed to sync from credentials file: %s)r?   rE   agent.anthropic_adapterr7  re   rH   loggerdebugr@   r   r(  r-  r%   )	r^   r   r7  credsfile_refreshfile_accessfile_expiresr2  excs	            r&   +_sync_anthropic_entry_from_credentials_filez:CredentialPool._sync_anthropic_entry_from_credentials_file  s\    =K''5<=+H+HL	JLLLLLL0022E  99^R88L))M266K 99[!44L 0C C Cjlqltuuu!!,".". $#'$(   ##E7333 	J 	J 	JLLCSIIIIIIII	Js   C  B0C   
D*D

Dc           	     ^   | j         dk    s|j        dk    r|S 	 t                      5  t                      }t	          |d          }ddd           n# 1 swxY w Y   t          |t                    s|S |                    d          }t          |t                    s|S |                    dd          }|                    dd          }|j        pd}|j	        pd}|r||k    s|r||k    rt                              d|j                   ||p|j	        ddddddd	}	|                    d
          r|d
         |	d
<   t          |fi |	}
|                     ||
           |                                  |
S n2# t           $ r%}t                              d|           Y d}~nd}~ww xY w|S )u  Sync a Codex device_code pool entry from auth.json if tokens differ.

        When a Codex OAuth access token expires (or the ChatGPT account hits
        its 5h/weekly quota), the pool entry gets marked ``STATUS_EXHAUSTED``
        with a ``last_error_reset_at`` that can be many hours in the future.
        Meanwhile the user may run ``hermes model`` / ``hermes auth`` which
        performs a fresh device-code login and writes new tokens to
        ``auth.json`` under ``_auth_store_lock``.  Without this sync the pool
        entry stays frozen until ``last_error_reset_at`` elapses — even
        though fresh credentials are sitting on disk — and every request
        fails with "no available entries (all exhausted or empty)".

        Mirrors the Nous/Anthropic resync paths above.  Only applies to
        device_code-sourced entries; env/API-key-sourced entries have no
        auth.json shadow to sync from.
        openai-codexdevice_codeNtokensrF   rz   rH   zQPool entry %s: syncing Codex tokens from auth.json (refreshed by another process))rF   rH   rI   rK   rM   rN   rO   rP   rT   z-Failed to sync Codex entry from auth.json: %s)r?   rE   r   r   r   r   r   re   rF   rH   r=  r>  r@   r   r(  r-  r%   )r^   r   
auth_storestaterH  store_accessstore_refreshentry_accessentry_refreshfield_updatesr2  rC  s               r&   !_sync_codex_entry_from_auth_storez0CredentialPool._sync_codex_entry_from_auth_store  s[   " =N**elm.K.KL*	O!## I I-//
,ZHHI I I I I I I I I I I I I I I eT** YYx((Ffd++ !::nb99L"JJ;;M !-3L!/52M ,,! -&3}&D&D5H   %1%2%Ie6I#'&*'+)-*.+/	1 	1 99^,, J49.4IM.1!%99=99##E7333 	O 	O 	OLLH#NNNNNNNN	OsG   E; AE; AE; AE; 5+E; !CE; ;
F*F%%F*c                   | j         dk    s|j        dk    r|S 	 t                      5  t                      }t	          |d          }ddd           n# 1 swxY w Y   |s|S |                    dd          }|                    dd          }|r$||j        k    rt                              d|j	                   ||dddd}|                    d	          r|d	         |d	<   |                    d
          r|d
         |d
<   |                    d          r|d         |d<   |                    d          r|d         |d<   t          |j                  }dD ]}|                    |          }	|	|	||<   t          |fd|i|}
|                     ||
           |                                  |
S n2# t          $ r%}t                              d|           Y d}~nd}~ww xY w|S )a  Sync a Nous pool entry from auth.json if tokens differ.

        Nous OAuth refresh tokens are single-use.  When another process
        (e.g. a concurrent cron) refreshes the token via
        ``resolve_nous_runtime_credentials``, it writes fresh tokens to
        auth.json under ``_auth_store_lock``.  The pool entry's tokens
        become stale.  This method detects that and adopts the newer pair,
        avoiding a "refresh token reuse" revocation on the Nous Portal.
        r   rG  NrH   rz   rF   zIPool entry %s: syncing tokens from auth.json (Nous refresh token changed))rF   rH   rI   rK   rM   rR   rV   rW   rU   r6   r4   r7   r:   r9   r;   rZ   z,Failed to sync Nous entry from auth.json: %s)r?   rE   r   r   r   re   rH   r=  r>  r@   r   rZ   r   r(  r-  r%   )r^   r   rI  rJ  rL  rK  rO  extra_updates	extra_keyvalr2  rC  s               r&    _sync_nous_entry_from_auth_storez/CredentialPool._sync_nous_entry_from_auth_store
  s    =F""elm&C&CL(	N!## A A-//
,Z@@A A A A A A A A A A A A A A A  !IIor::M 99^R88L %2E!E!E_H  
 %1%2#'&*'+1 1 99\** F272EM,/99[)) D16{1CM+.99344 Z<ABX<YM"8999122 V:?@T:UM"67 $U[ 1 1"; 7 7I  ))I..C36i0!%NN}NNN##E7333 	N 	N 	NLLGMMMMMMMM	NsA   F6 AF6 AF6 AF6 "EF6 6
G% G  G%c                $   |j         dk    rdS 	 t                      5  t                      }| j        dk    rt	          |d          }|	 ddd           dS |j        |d<   |j        r
|j        |d<   |j        r
|j        |d<   |j        r
|j        |d<   |j	        r
|j	        |d<   d	D ]#}|j
                            |          }||||<   $|j        r
|j        |d
<   t          |d|           n| j        dk    rt	          |d          }t          |t                    s	 ddd           dS |                    d          }t          |t                    s	 ddd           dS |j        |d<   |j        r
|j        |d<   |j        r
|j        |d<   t          |d|           n	 ddd           dS t#          |           ddd           dS # 1 swxY w Y   dS # t$          $ r,}t&                              d| j        |           Y d}~dS d}~ww xY w)u  Write refreshed pool entry tokens back to auth.json providers.

        After a pool-level refresh, the pool entry has fresh tokens but
        auth.json's ``providers.<id>`` still holds the pre-refresh state.
        On the next ``load_pool()``, ``_seed_from_singletons()`` reads that
        stale state and can overwrite the fresh pool entry — potentially
        re-seeding a consumed single-use refresh token.

        Applies to any OAuth provider whose singleton lives in auth.json
        (currently Nous and OpenAI Codex).
        rG  Nr   rF   rH   rR   rV   rW   rR  rU   rF  rH  rT   z3Failed to sync %s pool entry back to auth store: %s)rE   r   r   r?   r   rF   rH   rR   rV   rW   rZ   re   rU   r   r   r   rT   r   r%   r=  r>  )r^   r   rI  rJ  rT  rU  rH  rC  s           r&   %_sync_device_code_entry_to_auth_storez4CredentialPool._sync_device_code_entry_to_auth_storeA  sL    <=((F-	d!## *- *--//
=F**0VDDE}*- *- *- *- *- *- *- *- -2,>E.)* E161Do.' ?.3.>l+ =-2_k*1 S8=8R45&? 3 3	 $kooi88?/2E),/ O6;6N23(VUCCCC]n440^LLE%eT22 9*- *- *- *- *- *- *- *-: #YYx00F%fd33 ?*- *- *- *- *- *- *- *-@ .3-?F>** F272E/) C050Bn-(^UKKKK Q*- *- *- *- *- *- *- *-T !,,,U*- *- *- *- *- *- *- *- *- *- *- *- *- *- *- *- *- *-V  	d 	d 	dLLNPTP]_bccccccccc	dsk   G -G
G CGG ,+GG $?G#G 0G?G GG GG 
H#!H

Hforcec               
   |j         t          k    s|j        s|r|                     |d            d S 	 | j        dk    rddlm}  ||j        |j                            d                    }t          ||d         |d         |d         	          }|j        d
k    r[	 ddlm
}  ||d         |d         |d                    n# t          $ r&}t                              d|           Y d }~nid }~ww xY wn_| j        dk    rSt          j        |j        |j                  }t          ||d         |d         |                    d                    }n| j        dk    r|                     |          }||ur|}|j        |j        |j        |j        |j        |j        |j        |j        |j        |j        |j        |j        d}	t          j        |	t<          ||          }i }
t?          |j                   }d tC          |          D             }|"                                D ]\  }}||v r||
|<   |tF          v r|||<   t          |fd|i|
}n|S n# t          $ r}t                              d| j        |j$        |           | j        dk    r|j        d
k    r| %                    |          }|j        |j        k    r4t                              d           	 ddlm}  ||j        |j                            d                    }t          ||d         |d         |d         tL          d d           }| '                    ||           | (                                 	 ddlm
}  ||d         |d         |d                    n2# t          $ r%}t                              d|           Y d }~nd }~ww xY w|cY d }~S # t          $ r%}t                              d|           Y d }~n>d }~ww xY w| )                    |          s!t                              d           |cY d }~S | j        dk    r|                     |          }|j        |j        k    r{t                              d           t          |tL          d d d d d           }| '                    ||           | (                                 | *                    |           |cY d }~S |                     |d            Y d }~d S d }~ww xY wt          |tL          d d d d d           }| '                    ||           | (                                 | *                    |           |S )Nr5  r   )refresh_anthropic_oauth_purehermes_pkce)use_jsonrF   rH   rS   )rF   rH   rS   r6  )_write_claude_code_credentialsz7Failed to write refreshed token to credentials file: %srF  rT   )rF   rH   rT   r   )rF   rH   r3   r8   rU   r5   r2   r6   rR   rV   rW   r1   )min_key_ttl_secondsforce_refresh
force_mintc                    h | ]	}|j         
S rt   rn   ro   s     r&   rr   z0CredentialPool._refresh_entry.<locals>.<setcomp>  s    >>>1>>>r`   rZ   z'Credential refresh failed for %s/%s: %sz8Retrying refresh with synced token from credentials filer;  zDFailed to write refreshed token to credentials file (retry path): %szRetry refresh also failed: %sz7Credentials file has valid token, using without refreshu?   Nous refresh failed but auth.json has newer tokens — adoptingr0  )+rB   AUTH_TYPE_OAUTHrH   r3  r?   r<  r[  rE   endswithr   r^  r%   r=  r>  auth_modrefresh_codex_oauth_purerF   re   rV  r3   r8   rU   r5   r2   r6   rR   rV   rW   r1   refresh_nous_oauth_from_stater   r   rZ   r   r   rd   r@   rD  	STATUS_OKr(  r-  _entry_needs_refreshrX  )r^   r   rY  r[  	refreshedr2  r^  wexcsynced
nous_staterO  rS  _field_namesrv   r   rC  	retry_excs                    r&   _refresh_entryzCredentialPool._refresh_entry~  s   ?o--U5H- 2$$UD1114I	}++PPPPPP88'"\22=AA  	 "!*>!:"+O"<"+O"<	   <=00fZZZZZZ66%n5%o6%o6   
 % f f f%^`deeeeeeeef 1 .00$=&' 	 "!*>!:"+O"<!*~!>!>	   &((>>uEE&&"E$)$6%*%8!&','<*/*B"'"2"[#(#4"'"2!&,1,F 9 
 %B(I"'$	  	 !# $U[ 1 1>>u>>>%OO-- - -DAqL((+,a((k))+,a(!%NN}NNN ?	 ?	 ?	LLBDMSXS[]`aaa }++0M0MII%PP'5+>>>LL![\\\QXXXXXX$@$@"0%+]%;%;M%J%J% % %	 #*")2>)B*3O*D*3O*D(1+/,0# # # ++FG<<<w^^^^^^:: ). 9 )/ : )/ :   
  ) w w w"LL)oquvvvvvvvvw&$ Q Q Q%DiPPPPPPPPQ226:: "LL!Z[[[!MMMMMM }&&>>uEE'5+>>>LL!bccc%$-'+(,*.+/,0  G ''888MMOOO>>wGGG"NNNNNN  ---44444?	B ! "# $
 
 
 	E7+++ 	227;;;s   A+I %C I 
C6C1+I 1C66E#I T'A?S='BO/%NO
ON?:O?OOT
O=O83S=8O==3S=0T6B%S=T!S==Tc                .   |j         t          k    rdS | j        dk    rE|j        dS t	          |j                  t	          t          j                    dz            dz   k    S | j        dk    rt          |j        t                    S | j        dk    rdS dS )NFr5    i rF  r   )	rB   rc  r?   rS   rC   r   r   rF   r   r^   r   s     r&   ri  z#CredentialPool._entry_needs_refresh   s    ?o--5=K''"*uu*++s49;;3E/F/F/PPP=N**2"7   =F"" 5ur`   c                l    | j         5  |                                 cd d d            S # 1 swxY w Y   d S r\   )r  _select_unlockedr]   s    r&   selectzCredentialPool.select3  s}    Z 	+ 	+((**	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+   )--Fclear_expiredrefreshry  rz  c          
     Z   t          j                     }d}g }| j        D ]t}| j        dk    r8|j        dk    r-|j        t
          k    r|                     |          }||ur|}d}| j        dk    r8|j        dk    r-|j        t
          k    r|                     |          }||ur|}d}| j        dk    r8|j        dk    r-|j        t
          k    r|                     |          }||ur|}d}|j        t
          k    rOt          |          }|||k     r|r5t          |t          ddddd	          }	|                     ||	           |	}d}|r2|                     |          r|                     |d
          }
|
]|
}|                    |           v|r|                                  |S )a  Return entries not currently in exhaustion cooldown.

        When *clear_expired* is True, entries whose cooldown has elapsed are
        reset to STATUS_OK and persisted.  When *refresh* is True, entries
        that need a token refresh are refreshed (skipped on failure).
        Fr5  r6  Tr   rG  rF  Nr0  rY  )r   r  r?   rE   rI   r   rD  rV  rP  r   r   rh  r(  ri  rp  appendr-  )r^   ry  rz  nowcleared_any	availabler   rl  exhausted_untilclearedrj  s              r&   r  z!CredentialPool._available_entries7  s    ikk,.	] 7	$ 7	$E ,,1N1N)-===II%PP&&"E"&K
 ''55)-===>>uEE&&"E"&K //55)-===??FF&&"E"&K $444"25"9"9".33H3H  '%$-'+(,*.+/,0  G ''w777#E"&K "444U;; " //U/CC	$!U#### 	MMOOOr`   c                   |                      dd          }|s#d | _        t                              d           d S | j        t
          k    r"t          j        |          j        | _        S | j        t          k    rbt          |          dk    rOt          |d           t          j        dz             }|                     |           j        | _        |S | j        t          k    rt          |          dk    r|d         fd	| j        D             }|                    t          t          | j                  dz
  
                     d t%          |          D             | _        |                                  j        | _        |                                 pS |d         j        | _        S )NTrx  z>credential pool: no available entries (all exhausted or empty)r   c                    | j         S r\   rX   )es    r&   r	  z1CredentialPool._select_unlocked.<locals>.<lambda>  s     r`   r
  r  r   c                4    g | ]}|j         j         k    |S rt   r@   )rp   	candidater   s     r&   r,  z3CredentialPool._select_unlocked.<locals>.<listcomp>  s)    \\\Y9<SXS[C[C[yC[C[C[r`   r   c                6    g | ]\  }}t          ||           S r   r   )rp   r'  r  s      r&   r,  z3CredentialPool._select_unlocked.<locals>.<listcomp>  s)    ggg.#yWY===gggr`   )r  r  r=  infor  STRATEGY_RANDOMr/   choicer@   STRATEGY_LEAST_USEDr   minr   rX   r(  STRATEGY_ROUND_ROBINr  r}  r&  r-  r!  )r^   r  r2  rotatedr   s       @r&   ru  zCredentialPool._select_unlocked}  s   ++$+MM	 	#DKKXYYY4>_,,M),,E$xDL>000S^^a5G5G	'@'@AAAEe53F3JKKKGw///$xDN>111c)nnq6H6HaLE\\\\$-\\\GNN753t}3E3E3IJJJKKKggT]^eTfTfgggDMMMOOO$xD<<>>*U*! 8r`   c                r    |                                  }||S |                                 }|r|d         nd S Nr   )r!  r  )r^   r!  r  s      r&   peekzCredentialPool.peek  s>    ,,..N++--	(2y||d2r`   )r   c                  | j         5  |                                 p|                                 }|	 d d d            d S |j        p|j        d d         }t
                              d||           |                     |||           d | _        |                                 }|r1|j        p|j        d d         }t
                              d|           |cd d d            S # 1 swxY w Y   d S )N   z;credential pool: marking %s exhausted (status=%s), rotatingzcredential pool: rotated to %s)	r  r!  ru  rA   r@   r=  r  r3  r  )r^   r.  r   r   _label
next_entry_next_labels          r&   mark_exhausted_and_rotatez(CredentialPool.mark_exhausted_and_rotate  s`    Z 	 	LLNN=d&;&;&=&=E}	 	 	 	 	 	 	 	 [0EHRaRLFKKM     ]CCC#D..00J K(.C*-2C<kJJJ	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	s   ,C&BC&&C*-C*credential_idrG   c                     j         5  |r; j                            |d          dz    j        |<   | _        |cddd           S                      dd          }|s	 ddd           dS  fd|D             }|r|n|}t          | fd          } j                            |j        d          dz    j        |j        <   |j         _        |j        cddd           S # 1 swxY w Y   dS )	a\  Acquire a soft lease on a credential.

        If a specific credential_id is provided, lease that entry directly.
        Otherwise prefer the least-leased available credential, using priority as
        a stable tie-breaker. When every credential is already at the soft cap,
        still return the least-leased one instead of blocking.
        r   r   NTrx  c                f    g | ]-}j                             |j        d           j        k     +|.S )r   )r  re   r@   r  r  s     r&   r,  z0CredentialPool.acquire_lease.<locals>.<listcomp>  sF       &**58Q77$:NNN NNNr`   c                R    j                             | j        d          | j        fS r  )r  re   r@   rD   )r   r^   s    r&   r	  z.CredentialPool.acquire_lease.<locals>.<lambda>  s#    4#6#:#:58Q#G#G"X r`   r
  )r  r  re   r  r  r  r@   )r^   r  r  	below_cap
candidateschosens   `     r&   acquire_leasezCredentialPool.acquire_lease  s    Z 	 	 %595H5L5L]\]5^5^ab5b#M2#0 $		 	 	 	 	 	 	 	 //dD/QQI 	 	 	 	 	 	 	 	   #,  I '0>YJXXXX  F .2-@-D-DVYPQ-R-RUV-VD	*%yD9+	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	s   1C%C%.A*C%%C),C)c                    | j         5  | j                            |d          }|dk    r| j                            |d           n|dz
  | j        |<   ddd           dS # 1 swxY w Y   dS )z/Release a previously acquired credential lease.r   r   N)r  r  re   pop)r^   r  counts      r&   release_leasezCredentialPool.release_lease  s    Z 	? 	?'++M1==Ezz#''t<<<<5:QY#M2	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	? 	?s   AA  A$'A$c                l    | j         5  |                                 cd d d            S # 1 swxY w Y   d S r\   )r  _try_refresh_current_unlockedr]   s    r&   try_refresh_currentz"CredentialPool.try_refresh_current  s}    Z 	8 	85577	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8 	8rw  c                    |                                  }|d S |                     |d          }||j        | _        |S )NTr|  )r!  rp  r@   r  )r^   r   rj  s      r&   r  z,CredentialPool._try_refresh_current_unlocked  sG    =4''T'::	 (|Dr`   rC   c                   d}g }| j         D ][}|j        s|j        s|j        r/|                    t          |d d d d d d                      |dz  }F|                    |           \|r|| _         |                                  |S )Nr   r0  r   )r  rI   rK   rM   r}  r   r-  )r^   r  new_entriesr   s       r&   reset_statuseszCredentialPool.reset_statuses  s    ] 	* 	*E  *E$8 *E<Q *""$('+(,*.+/,0  
 
 
 
""5)))) 	'DMMMOOOr`   indexc                   |dk     s|t          | j                  k    rd S | j                            |dz
            }d t          | j                  D             | _        |                                  | j        |j        k    rd | _        |S )Nr   c                6    g | ]\  }}t          ||           S r  r  )rp   new_priorityr   s      r&   r,  z/CredentialPool.remove_index.<locals>.<listcomp>  s9     
 
 
#e EL111
 
 
r`   )r   r  r  r&  r-  r  r@   )r^   r  removeds      r&   remove_indexzCredentialPool.remove_index	  s    199DM 2 2224-##EAI..
 
'0'?'?
 
 
 	wz))#Dr`   targetr	   ?Tuple[Optional[int], Optional[PooledCredential], Optional[str]]c                ^   t          |pd                                          sdS t          | j        d          D ]\  }}|j        k    r||d fc S fdt          | j        d          D             }t          |          dk    r|d         d         |d         d         d fS t          |          dk    r	d d d dfS                                 rMt                    }d|cxk    rt          | j                  k    rn n|| j        |dz
           d fS d d d	| d
fS d d d dfS )Nrz   )NNzNo credential target provided.r   )startc                    g | ]J\  }}|j                                                                                                         k    F||fKS rt   )rA   r   r   )rp   r'  r   r   s      r&   r,  z1CredentialPool.resolve_target.<locals>.<listcomp>  sX     
 
 
U{  ""((**ciikk99 %L999r`   r   zAmbiguous credential label "z-". Use the numeric index or entry id instead.zNo credential #.zNo credential matching "z".)r>   r   r&  r  r@   r   isdigitrC   )r^   r  r'  r   label_matchesr  r   s         @r&   resolve_targetzCredentialPool.resolve_target  s   &,B%%'' 	@??#DM;;; 	( 	(JCx3E4'''' 
 
 
 
'Q???
 
 

 }"" #A&a(8(;TAA}!!pcppppp;;== 	:HHEE////S///////dmEAI6<<999999T=c=====r`   c                    t          |t          | j                            }| j                            |           |                                  |S )Nr   )r   r   r  r}  r-  rs  s     r&   	add_entryzCredentialPool.add_entry/  sF    t}(E(EFFFU###r`   )r?   r>   r   r   )r   r   )r   r   )r   r  )r"  r=   r#  r=   r   r$  )r   r$  r\   )r   r=   r.  rL   r   r   r   r=   )r   r=   r   r=   )r   r=   r   r$  )r   r=   rY  r   r   r  )r   r=   r   r   )ry  r   rz  r   r   r   )r.  rL   r   r   r   r  )r  rG   r   rG   )r  r>   r   r$  )r   rC   )r  rC   r   r  )r  r	   r   r  )rh   r   r   r  r  r  r   r!  r(  r-  r3  rD  rP  rV  rX  rp  ri  rv  r  ru  r  r  r  r  r  r  r  r  r  r  rt   r`   r&   r  r  k  s~       E E E E# # # #/ / / /# # # #^ ^ ^ ^
   
 
 
 
 37	    (# # # #J> > > >@5 5 5 5n;d ;d ;d ;dz` ` ` `D   &+ + + + ;@QV D D D D D DL   B3 3 3 3 37	     .    >? ? ? ?8 8 8 8      0   > > > >2     r`   r  rj   c                B   d }t          |           D ]\  }}|j        |k    r|} n||                    dt          j                    j        d d                    |                    dt          |                      |                    d|                    d          p|           |                     t          
                    ||                     dS | |         }i }i }	d t          |          D             }
|                                D ]d\  }}|dv s||dk    r|j        r||
v rt          ||          |k    r|||<   8|t          v r#|j                            |          |k    r||	|<   e|s|	r"|	ri |j        |	|d<   t#          |fi || |<   dS d	S )
Nr@   ry   rD   rA   Tc                    h | ]	}|j         
S rt   rn   ro   s     r&   rr   z _upsert_entry.<locals>.<setcomp>G  s    555qAF555r`   >   r@   rD   rZ   F)r&  rE   r{   r|   r}   r~   r   re   r}  r=   r   r   r   rA   r   rd   rZ   r   )r   r?   rE   rj   existing_idxr'  r   existingrO  rS  rn  r   r   s                r&   _upsert_entryr  6  s   L((  
U<6!!LE " 4!1"1"!5666:~g'>'>???7GKK$8$8$BFCCC'11(GDDEEEt|$HMM55F8$4$4555Lmmoo 
+ 
+
U$$$'>>hn>,x%%..%*c"K~!!#&&%//%*c"   	I%H%H-%HM'" ' B BM B Bt5r`   c                `  	 | dk    rdS dddddd	t          d	 |D             d
           }t          d |D             	fd          }g ||}d t          |          D             }d}t          |          D ]1\  }}|j        |k    r!t          ||          |||j                 <   d}2|S )Nr5  Fr   r   r         )zenv:ANTHROPIC_TOKENzenv:CLAUDE_CODE_OAUTH_TOKENr\  r6  zenv:ANTHROPIC_API_KEYc              3  B   K   | ]}t          |j                  |V  d S r\   r   rE   r   s     r&   r   z-_normalize_pool_priorities.<locals>.<genexpr>g  s2      GG5'8'F'FGGGGGGGr`   c                    | j         S r\   r   r  s    r&   r	  z,_normalize_pool_priorities.<locals>.<lambda>h  s    %. r`   r
  c              3  B   K   | ]}t          |j                  |V  d S r\   r  r   s     r&   r   z-_normalize_pool_priorities.<locals>.<genexpr>k  s2      KK5+<U\+J+JKKKKKKKr`   c                n                         | j        t                              | j        | j        fS r\   )re   rE   r   rD   rA   )r   source_ranks    r&   r	  z,_normalize_pool_priorities.<locals>.<lambda>l  s/    OOEL#k*:*:;;NK
 r`   c                $    i | ]\  }}|j         |S rt   r  )rp   r'  r   s      r&   rw   z._normalize_pool_priorities.<locals>.<dictcomp>t  s     DDD:33DDDr`   r   T)r   r&  rD   r   r@   )
r?   r   manual_entriesseeded_entriesordered	id_to_idxchangedr  r   r  s
            @r&   _normalize_pool_prioritiesr  [  s   ;u  !'(!" K GGGGGG((  N KKGKKK
 
 
 
  N 100GDD71C1CDDDIG(11  e>\))+25<+P+P+PGIeh'(GNr`   Tuple[bool, Set[str]]c                P   d}t                      }t                      }	 ddlm} n# t          $ r d }Y nw xY w| dk    r	 ddlm}  |d          s||fS n# t          $ r Y nw xY wddlm}m} d |            fd	 |            ffD ]\  }	}
|
r|
	                    d
          r || |	          r)|
                    |	           |t          || |	|	t          |
	                    d
d          |
	                    d          |
	                    d          t          |
	                    d
d          |	          d          z  }n| dk    rTt          |d          }|r? || d          s2|
                    d           t          |	                    d          pd                                          }|p#t          |	                    dd          d          }|t          || di dddt          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#          t#          |	                    d$          t$                    r|	                    d$          nd |d%          z  }n| d&k    r	 dd'lm}m}  |            \  }}|r ||          }d(|                                v rd)nd*| }	 || |	          sR|
                    |	           t/          j	        |           }|t          || |	|	t0          ||r|j        nd|d+          z  }n# t4          $ r&}t6                              d,|           Y d }~nd }~ww xY w| d-k    r	 dd.lm}  |d/          }
|
	                    d0d          }|r|
	                    dd1          }	 || |	          sq|
                    |	           |t          || |	|	t          ||
	                    d2          |
	                    d3d          |
	                    d4|	          d5          z  }n# t4          $ r&}t6                              d6|           Y d }~nd }~ww xY w| d7k    r	 dd8lm}  |d7          }|rP|	                    d          r:d9}	 || |	          s+|
                    |	           d }	 dd:lm} |	                    dd          }|r7tA          |!                    |          "                                d;z            }n# t4          $ r d }Y nw xY wt          |	                    dd          pd          #                    d<          }|t          || |	|	t          |d         |	                    d          |||	                    dd          p#t          |	                    dd          |	          d=          z  }nC# t4          $ r&}t6                              d>|           Y d }~nd }~ww xY w| d?k    r	 || d          r||fS t          |d?          }t#          |t$                    r|	                    d@          nd }t#          |t$                    r|	                    d          r|
                    d           |t          || ddt          |	                    dd          |	                    d          dA|	                    dB          t          |	                    dd          d          dC          z  }||fS )DNFr   is_source_suppressedc                    dS NFrt   _p_ss     r&   _is_suppressedz-_seed_from_singletons.<locals>._is_suppressed      5r`   r5  )!is_provider_explicitly_configured)r7  read_hermes_oauth_credentialsr\  r6  r9  rz   r8  r:  )rE   rB   rF   rH   rS   rA   r   rG  rA   rF   rE   rB   rH   rR   r5   r2   r3   r8   rU   rV   rW   r6   r4   r7   r:   r9   r;   r1   )r;   r1   rA   copilot)resolve_copilot_tokenget_copilot_api_tokenghgh_clienv:rE   rB   rF   rQ   rA   zCopilot token seed failed: %sz
qwen-oauth) resolve_qwen_runtime_credentials)refresh_if_expiringr+   zqwen-clirS   rQ   	auth_file)rE   rB   rF   rS   rQ   rA   z Qwen OAuth token seed failed: %szminimax-oauth)get_provider_auth_stater*   r   rr  r   )rE   rB   rF   rH   rS   rQ   rA   z#MiniMax OAuth token seed failed: %srF  rH  z%https://chatgpt.com/backend-api/codexrT   )rE   rB   rF   rH   rQ   rT   rA   )$setr   hermes_cli.authr  ImportErrorr  r<  r7  r  re   addr  rc  r   r   r>   r   r   r   hermes_cli.copilot_authr  r  r   r   r   rU   r%   r=  r>  r  r  r   rC   r   r   r   )r?   r   r  active_sourcesrI  r  r  r7  r  source_namer?  rJ  custom_labelseeded_labelr  r  r   rE   	api_tokenpconfigrC  r  r  rS   _dtr   rQ   rH  s                               r&   _seed_from_singletonsr  }  s.
   G"uuN!##JJJJJJJJ   	 	 	 	 	 ;
	IIIIII44[AA /../ 	 	 	D	 	hggggggg 99;;<88::;#
 	 	K  =11 !>(K88 "";///="-%4(-		-(D(D).>)B)B).;)?)?!1%))M22N2NP[!\!\ 	  	, 
V		$Z88 *	-@@ *	}---
 uyy117R88>>@@L' +;		."--}, ,L }m #EIInb$A$A $UYY%?%?	
 !%))L"9"9 !%))L"9"9 UYYw//  ;!7!7 &uyy1B'C'C )%))4H*I*I  ;!7!7 +EII6L,M,M& "599]#;#;'( !%))L"9"9)* #EIIn$=$=+, +EII6L,M,M-. '		2D(E(E/0 .3YY7N-O-O/9%))E:J:JD/Q/Q[599U+++W[)5  	     GD 
Y			?\\\\\\\\1133ME6 11%88	*.&,,..*@*@hhoVoo%~h<< "&&{333/3H==G} #&1):,5FM(U(B(BSU%+ 	    G  	? 	? 	?LL8#>>>>>>>>	? 
\	!	!	BHHHHHH44OOOEIIi,,E #ii*==%~h<< "&&{333} #&1)8,1-2YY-G-G(-		*b(A(A%*YY{K%H%H 	    G  	B 	B 	BLL;SAAAAAAAA	B 
_	$	$!	E??????++O<<E >22 %%~h<< "&&{333$(M-<<<<<<#iib99 [,/0A0A#0F0F0P0P0R0RUY0Y,Z,ZM$ - - -(,-"599-A2#F#F#L"MMTTUXYYH} #&1)8,1.,A-2YY-G-G-:(0%*YYw%;%; &?O %		." = ={@ @
 
	    G   	E 	E 	ELL>DDDDDDDD	E 
^	#	#
 >(M22 	+N**$Z@@(25$(?(?I8$$$T fd## 	

>(B(B 	}---}+!0$*JJ~r$B$B%+ZZ%@%@ G$)IIn$=$=-fjj.L.Lm\\ 	  G N""s   ' 77A 
A%$A%(BQ 
Q7Q22Q7B=U   
U0
U++U0;A[! AX" ![! "X1.[! 0X11B.[! !
\+\\c                   d}t                      }	 ddlm} n# t          $ r d }Y nw xY w| dk    rut	          d          pd                                }|rLd} || |          r||fS |                    |           |t          || ||t          |t          dd	          z  }||fS t          j        |           }|r|j        t          k    r||fS d}|j        r;t	          |j                  pd                                                    d
          }t          |j                  }	| dk    rg d}	|	D ]}
t	          |
          pd                                }|s(d|
 } || |          r:|                    |           | dk    r|                    d          st$          nt          }|p|j        }| dk    rt)          ||j        |          }n| dk    rt+          ||j        |          }|t          || ||||||
d	          z  }||fS )NFr   r  c                    dS r  rt   r  s     r&   _is_source_suppressedz-_seed_from_env.<locals>._is_source_suppressedo  r  r`   
openrouterOPENROUTER_API_KEYrz   zenv:OPENROUTER_API_KEYr  r   r5  )ANTHROPIC_TOKENCLAUDE_CODE_OAUTH_TOKENANTHROPIC_API_KEYr  z
sk-ant-apizkimi-codingzai)r  r  r  r  r   r   r  r  r   r   r   re   rB   base_url_env_varr   r   api_key_env_varsr   rc  rU   r   r   )r?   r   r  r  r  r   rE   r  env_urlenv_varsenv_varrB   rQ   s                r&   _seed_from_envr	  d  s   G"uuNQQQQQQQ   	 	 	 	 	 <344:AACC 	-F$$Xv66 /..v&&&}$!2$) 31 	  G &&#H--G 'g'+<<<&&G V !9::@bGGIIPPQTUUG,--H;
 
 
  
 
w''-24466 	!!!  622 	6"""'/;'>'>uGWGWXdGeGe'>OOk|	8g8}$$-eW5OQXYYHH,UG4NPWXXH= & %$  	
 
 	
 N""s    ))r  Set[str]c                v    fd| D             }t          |          t          |           k    rdS || d d <   dS )Nc                    g | ]D}t          |j                  s,|j        v s#|j                            d           s|j        dv B|ES )r  >   r6  r\  )r   rE   r   )rp   r   r  s     r&   r,  z/_prune_stale_seeded_entries.<locals>.<listcomp>  sn     	 	 	U\**	 <>))L##F++ * |=== 	 >==r`   FT)r   )r   r  retaineds    ` r&   _prune_stale_seeded_entriesr    s[    	 	 	 		 	 	H 8}}G$$uGAAAJ4r`   c                   d}t                      }	 ddlm} n# t          $ r d }Y nw xY wt	          |           }|rt          |                    d          pd                                          }t          |                    d          pd                                                              d          }t          |                    d	          pd                                          }|rHd
| }	 || |	          s7|	                    |	           |t          || |	|	t          |||p|	d          z  }	 t                      }
|
r|
                    d          nd}t          |t                    rNt          |                    d          pd                                                                          }t          |                    d          pd                                                              d          }d}dD ]V}|                    |          }t          |t
                    r*|                                r|                                } nW|dk    r\|rZ|rXt!          |          }|| k    rCd}	 || |	          s5|	                    |	           |t          || |	|	t          ||dd          z  }n# t"          $ r Y nw xY w||fS )zJSeed a custom endpoint pool from custom_providers config and model config.Fr   r  c                    dS r  rt   r  s     r&   r  z)_seed_custom_pool.<locals>._is_suppressed  r  r`   r+   rz   rQ   r   ra   zconfig:r  modelNr?   )r+   apicustommodel_config)r  r  r  r  r   r>   re   r   r   r  r  r   r'   r   r   r   r   r%   )r   r   r  r  r  	cp_configr+   rQ   ra   rE   r   	model_cfgmodel_providermodel_base_urlmodel_api_keyrv   r   matched_keys                     r&   _seed_custom_poolr    sA   G"uuNJJJJJJJ   	 	 	 	 	
 ,H55I immI..4"55;;==y}}Z006B77==??FFsKK9==((.B//5577 	%t%%F!>(F33 ""6***="(%6(/$,!% 	   "$$+1;FJJw'''t	i&& 	 z!:!:!@bAAGGIIOOQQN z!:!:!@bAAGGIIPPQTUUNM'  MM!$$a%% !'')) $%GGIIME))n)):>JJ(**+F)>(F;; &**6222=#$"*0->0=,:)7 	$ $      N""s    ))=FK	 	
KKc                2     pd                                                                  t                     } fd|D             }                     t                    r)t           |          \  }}|}|t          ||          z  }nSt           |          \  }}t           |          \  }}	|p|}|t          |||	z            z  }|t           |          z  }|r*t           d t          |d           D                        t           |          S )Nrz   c                F    g | ]}t                               |          S rt   )r=   r   )rp   rj   r?   s     r&   r,  zload_pool.<locals>.<listcomp>  s*    XXX))(G<<XXXr`   c                6    g | ]}|                                 S rt   r+  r   s     r&   r,  zload_pool.<locals>.<listcomp>#  s     ZZZU]]__ZZZr`   c                    | j         S r\   r   )items    r&   r	  zload_pool.<locals>.<lambda>#  s    4= r`   r
  )r   r   r   r   r   r  r  r  r	  r  r   r   r  )
r?   raw_entriesr   custom_changedcustom_sourcesr  singleton_changedsingleton_sourcesenv_changedenv_sourcess
   `         r&   	load_poolr(    sG   B%%''--//H&x00KXXXXKXXXG-.. 
A):8W)M)M& .wGGG/DXw/W/W,,#1(G#D#D [#2{.w8IK8WXXX-h@@@ 
ZZ&>X>X*Y*Y*YZZZ	
 	
 	
 (G,,,r`   )r   r    )r   r>   r   r>   r   r>   )r   r   r   rC   )rE   r>   r   r   )r   rL   r   rC   )r   r	   r   rJ   )r   r>   r   rJ   )r   r   r   rY   )r   r=   r   rJ   )ra   r>   r   r>   r\   )r   r    )rQ   r>   r   rG   )r   r   )r   r>   r   r   )r?   r>   r   r>   )
r   r   r?   r>   rE   r>   rj   rY   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  )U__doc__
__future__r   loggingr/   r  r   r|   r   dataclassesr   r   r   r   typingr	   r
   r   r   r   r   hermes_constantsr   r$   r   r  authre  r   r   r   r   r   r   r   r   r   r   r   r   r   r   	getLoggerrh   r=  r'   rh  r   rc  r   r   r   r  r  r  r   r   r   r   	frozensetrd   r=   r   r   r   r   r   r   r   r   r   r   r   r   r   r  r  r  r  r  r  r	  r  r  r(  rt   r`   r&   <module>r2     sv   B B " " " " " "         				 2 2 2 2 2 2 2 2 2 2       8 8 8 8 8 8 8 8 8 8 8 8 8 8 8 8 0 0 0 0 0 0 + + + + + + " " " " " "                               " 
	8	$	$    	  " $ " 	  $  ' 
   i      O O O O O O O Od   F F F FU U U U
) ) ) )   <
 
 
 
   2   2 2 2 2
7 7 7 7 74              )* %H H H H H H H HV" " " "J   Dd# d# d# d#NN# N# N# N#b   "F# F# F# F#R- - - - - -r`   