
    i                    ^   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	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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)m*Z*  ej+        e,          Z-dOdZ.dZ/dZ0dZ1dZ2dZ3dZ4dZ5dZ6dZ7e4e5e6e7hZ8dZ9dZ:dZ; e<h d          Z=e G d d                      Z>dPdZ?dQd!Z@dRd$ZAdSd'ZBdTd+ZCdUd-ZDdVd1ZEdWd3ZFdXd5ZGdYdZd7ZHd[d:ZId\d<ZJd]d>ZKd^d@ZLdAZM G dB dC          ZNd_dEZOd`dFZPdadHZQdadIZRdbdLZSdcdMZTdddNZUdS )ez<Persistent multi-credential pool for same-provider failover.    )annotationsN)	dataclassfieldsreplace)datetime)AnyDictListOptionalSetTuple)OPENROUTER_BASE_URL)'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_import_codex_cli_tokens_write_codex_cli_tokens_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/agentuser/.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__u   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__y   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=   [   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   5  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>G  sx        >>,-- y}}S))400 cN	     r`   )r   sorted)r   s   @r&   list_custom_pool_providersr   D  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   O  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  Z  s      F~""899Jj$'' #"":>>(B//5266<<>>DDFFH,,,r`   r   c                      e Zd Zd9dZd:dZd:d	Zd;d
Zd<dZd=dZd>dZ		 d?d@dZ
dAdZdAdZdBdZdCdZdDdZd<d Zd!d!d"dEd%Zd<d&Zd<d'Zdd(dFd)Zd?dGd,ZdHd-Zd<d.Zd<d/ZdId1ZdJd3ZdKd7ZdAd8ZdS )L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>p  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__n  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_credentialsw      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_availablez  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    r|S 	 t                      }|s|S |                    dd          }|                    dd          }|rl||j        k    rat                              d|j                   t          |||ddd          }|                     ||           | 	                                 |S n2# t          $ r%}t                              d|           Y d}~nd}~ww xY w|S )	a  Sync an openai-codex pool entry from ~/.codex/auth.json if tokens differ.

        OpenAI OAuth refresh tokens are single-use and rotate on every refresh.
        When the Codex CLI (or another Hermes profile) refreshes its token,
        the pool entry's refresh_token becomes stale.  This method detects that
        by comparing against ~/.codex/auth.json and syncing the fresh pair.
        openai-codexrH   rz   rF   zMPool entry %s: syncing tokens from ~/.codex/auth.json (refresh token changed)N)rF   rH   rI   rK   rM   z*Failed to sync from ~/.codex/auth.json: %s)r?   r   re   rH   r=  r>  r@   r   r(  r-  r%   )r^   r   
cli_tokenscli_refresh
cli_accessr2  rC  s          r&   _sync_codex_entry_from_cliz)CredentialPool._sync_codex_entry_from_cli  s%    =N**L	L133J $.."==K#;;J {e.AAAlnsnvwww!!+"- $#'$(   ##E7333 	L 	L 	LLLEsKKKKKKKK	Ls   B< BB< <
C+C&&C+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).
        device_codeNr   rF   rH   rR   rV   rW   )r6   r4   r7   r:   r9   r;   rU   rF  tokensrT   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   
auth_storestate	extra_keyvalrM  rC  s           r&   %_sync_device_code_entry_to_auth_storez4CredentialPool._sync_device_code_entry_to_auth_store  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 }~nhd }~ww xY wn^| j        dk    rm|                     |          }||ur|}t          j        |j        |j                  }t          ||d         |d         |                    d                    }n| j        dk    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    rAt                              d           	 t          j        |j        |j                  }t          ||d         |d         |                    d          tL          d d           }| '                    ||           | (                                 | *                    |           	 tW          |j        |j        |j,                   n2# t          $ r%}t                              d|           Y d }~nd }~ww xY w|cY d }~S # t          $ r%}t                              d|           Y d }~nSd }~ww xY w| )                    |          s6t                              d            | *                    |           |cY d }~S |                     |d            Y d }~d S d }~ww xY wt          |tL          d d d d d !          }| '                    ||           | (                                 | *                    |           | j        dk    rU	 tW          |j        |j        |j,                   n2# t          $ r%}t                              d"|           Y d }~nd }~ww xY w|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>u  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 refreshz@Retrying Codex refresh with synced token from ~/.codex/auth.json)rF   rH   rT   rI   rK   rM   )rT   z>Failed to write refreshed Codex tokens to CLI file (retry): %sz#Codex retry refresh also failed: %sz0Codex CLI has valid token, using without refreshr0  z6Failed to write refreshed Codex tokens to CLI file: %s)-rB   AUTH_TYPE_OAUTHrH   r3  r?   r<  rU  rE   endswithr   rX  r%   r=  r>  rJ  auth_modrefresh_codex_oauth_purerF   re   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_refreshrR  r   rT   )r^   r   rS  rU  	refreshedr2  rX  wexcsynced
nous_statefield_updatesextra_updates_field_namesrv   r   rC  	retry_excs                    r&   _refresh_entryzCredentialPool._refresh_entry*  s   ?o--U5H- 2$$UD1114`	}++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
 88??&&"E$=&' 	 "!*>!:"+O"<!*~!>!>	   &(($)$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 R	 R	 R	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 }..88??'5+>>>LL!cdddW$,$E"/"0% %	 #*")2>)B*3O*D)2~)F)F(1+/,0# # # ++FG<<<BB7KKKq3 ' 4 ' 5-4-A    
  ) q q q"LL)ikoppppppppq&$ W W W%JIVVVVVVVVW226:: "LL!STTT>>vFFF!MMMMMM  ---44444eR	h ! "# $
 
 
 	E7+++ 	227;;; =N**]'()!(!5    
  ] ] ]UW[\\\\\\\\]s   A+I %C I 
C6C1+I 1C66E"I X&A?X&BO.%NO
ON>9O>OOX
O<O72X7O<<3X/X5AXBU6!T=<U6=
U,U'"U6'U,,U60X6
V% V X V%%AX-X3XX=!Z 
[)[		[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  i rF  r   )	rB   r]  r?   rS   rC   r   r   rF   r   r^   r   s     r&   rc  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.select  s}    Z 	+ 	+((**	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+   )--Fclear_expiredrefreshrt  ru  c          
        t          j                     }d}g }| j        D ]-}| j        dk    r8|j        dk    r-|j        t
          k    r|                     |          }||ur|}d}| j        dk    r4|j        t
          k    r$|j        r|                     |          }||ur|}d}|j        t
          k    rOt          |          }|||k     r|r5t          |t          ddddd          }	|                     ||	           |	}d}|r2|                     |          r|                     |d          }
|
|
}|                    |           /|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  TrF  Nr0  rS  )r   r  r?   rE   rI   r   rD  rH   rJ  r   r   rb  r(  rc  rl  appendr-  )r^   rt  ru  nowcleared_any	availabler   rf  exhausted_untilclearedrd  s              r&   r  z!CredentialPool._available_entries  s    ikk,.	] *	$ *	$E ,,1N1N)-===II%PP&&"E"&K //)-===+ >88??&&"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    r3t          |          dk    r t          |d           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 )NTrs  z>credential pool: no available entries (all exhausted or empty)r   c                    | j         S r\   )rX   )es    r&   r	  z1CredentialPool._select_unlocked.<locals>.<lambda>K  s     r`   r
  r   c                4    g | ]}|j         j         k    |S rt   r@   )rp   	candidater   s     r&   r,  z3CredentialPool._select_unlocked.<locals>.<listcomp>Q  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  s)    ggg.#yWY===gggr`   )r  r  r=  infor  STRATEGY_RANDOMr/   choicer@   STRATEGY_LEAST_USEDr   minSTRATEGY_ROUND_ROBINr  rx  r   r&  r-  r!  )r^   r{  rotatedr   s      @r&   rp  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$xDL>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!  rp  rA   r@   r=  r  r3  r  )r^   r.  r   r   _label
next_entry_next_labels          r&   mark_exhausted_and_rotatez(CredentialPool.mark_exhausted_and_rotatec  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   NTrs  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_leasez  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 	8rr  c                    |                                  }|d S |                     |d          }||j        | _        |S )NTrw  )r!  rl  r@   r  )r^   r   rd  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   rx  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  rx  r-  rn  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=   rS  r   r   r  )r   r=   r   r   )rt  r   ru  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  rJ  rR  rl  rc  rq  r  rp  r  r  r  r  r  r  r  r  r  r  rt   r`   r&   r  r  m  sb       E E E E# # # #/ / / /# # # #^ ^ ^ ^
   
 
 
 
 37	    (# # # #J   B;d ;d ;d ;dzB B B BH   &+ + + + ;@QV 7 7 7 7 7 7r   <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>  s    555qAF555r`   >   r@   rD   rZ   F)r&  rE   r{   r|   r}   r~   r   re   rx  r=   r   r   r   rA   r   rd   rZ   r   )r   r?   rE   rj   existing_idxr'  r   existingrh  ri  rj  r   r   s                r&   _upsert_entryr    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_TOKENrV  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>%  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>&  s    %. r`   r
  c              3  B   K   | ]}t          |j                  |V  d S r\   r  r   s     r&   r   z-_normalize_pool_priorities.<locals>.<genexpr>)  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>*  s/    OOEL#k*:*:;;NK
 r`   c                $    i | ]\  }}|j         |S rt   r  )rp   r'  r   s      r&   rw   z._normalize_pool_priorities.<locals>.<dictcomp>2  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                   d}t                      }t                      }| 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	 dd	lm	}
  |
| |          r0n# t          $ r Y nw xY w|
                    |           |t          || ||t          |	                    dd
          |	                    d          |	                    d          t          |	                    dd
          |          d          z  }Ԑn| dk    rqt          |d          }|r\|
                    d           |t          || ddt          |                    dd
          |                    d          |                    d          |                    d          |                    d          |                    d          |                    d          |                    d          |                    d          |                    d          t          |                    d          t                     r|                    d          nd t          |                    dd
          d          d          z  }n| dk    r	 ddlm}  |            \  }}|rod|                                v rdnd | }|
                    |           t)          j        |           }|t          || ||t*          ||r|j        nd
|d!          z  }n# t.          $ r&}t0                              d"|           Y d }~nd }~ww xY w| d#k    r	 dd$lm}  |d%          }	|	                    d&d
          }|r|	                    d'd(          }|
                    |           |t          || ||t          ||	                    d)          |	                    d*d
          |	                    d+|          d,          z  }n# t.          $ r&}t0                              d-|           Y d }~nd }~ww xY w| d.k    rt          |d.          }t          |t                     r|                    d/          nd }t          |t                     r|                    d          s	 dd0lm}m}  |            }|rot0                              d1            ||           t                      }t          |d.          }t          |t                     r|                    d/          nd }n2# t.          $ r%}t0                              d2|           Y d }~nd }~ww xY wt          |t                     r|                    d          r|
                    d           |t          || ddt          |                    dd
          |                    d          d3|                    d4          t          |                    dd
          d          d5          z  }||fS )6NFr5  r   )!is_provider_explicitly_configured)r7  read_hermes_oauth_credentialsrV  r6  r9  )is_source_suppressedrz   r8  r:  )rE   rB   rF   rH   rS   rA   r   rL  rF   rH   rR   r5   r2   r3   r8   rU   rV   rW   r1   )rE   rB   rF   rH   rR   r5   r2   r3   r8   rU   rV   rW   r1   rA   copilot)resolve_copilot_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+   rE   zqwen-clirS   rQ   	auth_file)rE   rB   rF   rS   rQ   rA   z Qwen OAuth token seed failed: %srF  rM  )r   _save_codex_tokensz2Importing Codex CLI tokens into Hermes auth store.z!Codex CLI token import failed: %sz%https://chatgpt.com/backend-api/codexrT   )rE   rB   rF   rH   rQ   rT   rA   )setr   hermes_cli.authr  ImportErrorr<  r7  r  re   r  addr  r]  r   r   r   r   hermes_cli.copilot_authr  r   r   r   rU   r%   r=  r>  r  r   r  r  )r?   r   r  active_sourcesrN  r  r7  r  source_namer?  r  rO  r  r   rE   pconfigrC  r  rM  r   r  rG  s                         r&   _seed_from_singletonsr  ;  sl   G"uuN!##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 DDDDDD++HkBB ! !"   D"";///="-%4(-		-(D(D).>)B)B).;)?)?!1%))M22N2NP[!\!\ 	  	6 
V		$Z88 	}---}+!0$)IInb$A$A%*YY%?%?"'))L"9"9"'))L"9"9"YYw//!&;!7!7',yy1B'C'C*/))4H*I*I!&;!7!7,1II6L,M,M/9%))E:J:JD/Q/Q[599U+++W[-eii.K.K][[ 	  G, 
Y			?EEEEEE1133ME6 *.&,,..*@*@hhoVoo"";///+/99="-%6(-BI$QG$>$>r!' 	    	? 	? 	?LL8#>>>>>>>>	? 
\	!	!	BHHHHHH44OOOEIIi,,E #ii*=="";///="-%4(-).?)C)C$)IIj"$=$=!&;!D!D 	    	B 	B 	BLL;SAAAAAAAA	B 
^	#	#$Z@@(25$(?(?I8$$$T
 64(( 	GVZZ-G-G 	GGXXXXXXXX5577
 VKK TUUU&&z222!1!3!3J0^LLE4>ud4K4KUUYYx000QUF G G G@#FFFFFFFFGfd## 	

>(B(B 	}---}+!0$*JJ~r$B$B%+ZZ%@%@ G$)IIn$=$=-fjj.L.Lm\\ 	  G N""sl   = 
A
	A

B
B+*B+BM 
M8M33M8B1P5 5
Q%?Q  Q%BU 
V	$VV	c                   d}t                      }| dk    rit          j        dd                                          }|r<d}|                    |           |t          || ||t          |t          dd          z  }||fS t          j	        |           }|r|j
        t          k    r||fS d}|j        r?t          j        |j        d                                                              d          }t          |j                  }| dk    rg d	}|D ]}	t          j        |	d                                          }|s,d
|	 }|                    |           | 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 )NF
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  osgetenvr   r  r  r   r   r   re   rB   base_url_env_varr   r   api_key_env_varsr   r]  rU   r   r   )r?   r   r  r  r   rE   r  env_urlenv_varsenv_varrB   rQ   s               r&   _seed_from_envr    sS   G"uuN<	.3399;; 	-Fv&&&}$!2$) 31 	  G &&#H--G 'g'+<<<&&G N)G4b99??AAHHMMG,--H;
 
 
  
 
	'2&&,,.. 	!!!6"""'/;'>'>uGWGWXdGeGe'>OOk|	8g8}$$-eW5OQXYYHH,UG4NPWXXH= & %$  	
 
 	
 N""r`   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  rV  )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                T   d}t                      }t          |           }|rt          |                    d          pd                                          }t          |                    d          pd                                                              d          }t          |                    d          pd                                          }|r<d| }|                    |           |t          || ||t          |||p|d          z  }	 t                      }	|	r|	                    d	          nd
}
t          |
t                    rBt          |
                    d          pd                                                                          }t          |
                    d          pd                                                              d          }d}dD ]V}|
                    |          }t          |t                    r*|                                r|                                } nW|dk    rP|rN|rLt          |          }|| k    r7d}|                    |           |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+   rz   rQ   r   ra   zconfig:r  modelNr?   )r+   apicustommodel_config)r  r   r>   re   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  4  s   G"uuN ,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v&&&}$!2$+ (!^V 	  G"$$+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"&&v...} &,):,9(6%3 	    G     N""s   E?J 
J#"J#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>w  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  t  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  )V__doc__
__future__r   loggingr/   r  r   r|   r  r   dataclassesr   r   r   r   typingr   r	   r
   r   r   r   hermes_constantsr   r  authr_  r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   	getLoggerrh   r=  r'   rb  r   r]  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>r     sy   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              )* %D
 D
 D
 D
 D
 D
 D
 D
N" " " "J   Dd# d# d# d#N># ># ># >#B   "=# =# =# =#@- - - - - -r`   