
    iW                       U d Z ddlmZ ddlZddlmZ ddlmZmZm	Z	m
Z
mZ ddlmZmZ  ej        e          Z ed           G d	 d
                      Zi d edddd          d eddd          d eddd          d edddd          d eddd          d eddd d!d"#          d$ edd%d&d'          d( edd)*          d+ ed,d-*          d. edd/d01          d2 edd34          d5 edd6d7d89          d: ed,d;4          d< ed,dd=          d> ed,d?4          d@ eddA4          dB eddC4           eddD4           eddE           edddFG           edddHG           edddIG           edddJG           eddKdLM           eddNdOM           eddP4           eddQ4           eddRdSM           eddTdUdV9           eddW4           eddX4           edYdZ[          d\Zd]ed^<   e G d_ d`                      Zi daddbd.dcd.ddd.ded.dfdgdhdgdidgdjdkdldkdmdkdndkdod2dpd2dqd2drd2dsd5i dtd5dud>dvd>dwd+dxd+dyd(dzd(d{d$d|d}d~d}dd}ddddddddddddi dddd@ddBddBddBddBddddddddddddddddddddddi dddddddddddddddddddddddddddddddddddddZded<   ddddddddddddZded<   dd,ddYdĜZded<   ddɄZdd˄Zdd̈́ZddЄZdddӄZddքZdd؄ZddۄZ 	 	 dddބZ!dS )u  
Single source of truth for provider identity in Hermes Agent.

Two data sources, merged at runtime:

1. **models.dev catalog** — 109+ providers with base URLs, env vars, display
   names, and full model metadata (context, cost, capabilities).  This is
   the primary database.

2. **Hermes overlays** — transport type, auth patterns, aggregator flags,
   and additional env vars that models.dev doesn't track.  Small dict,
   maintained here.

3. **User config** (``providers:`` section in config.yaml) — user-defined
   endpoints and overrides.  Merged on top of everything else.

Other modules import from this file.  No parallel registries.
    )annotationsN)	dataclass)AnyDictListOptionalTuple)base_url_host_matchesbase_url_hostnameT)frozenc                  h    e Zd ZU dZ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S )HermesOverlayz?Hermes-specific provider metadata layered on top of models.dev.openai_chatstr	transportFboolis_aggregatorapi_key	auth_type Tuple[str, ...]extra_env_vars base_url_overridebase_url_env_varN)__name__
__module____qualname____doc__r   __annotations__r   r   r   r   r   r       9/home/ubuntu/.hermes/hermes-agent/hermes_cli/providers.pyr   r   "   s         II"I""""MI&(N((((r!   r   
openrouterr   )OPENAI_API_KEYOPENROUTER_BASE_URL)r   r   r   r   nousoauth_device_codez)https://inference-api.nousresearch.com/v1)r   r   r   openai-codexcodex_responsesoauth_externalz%https://chatgpt.com/backend-api/codexz
qwen-oauthzhttps://portal.qwen.ai/v1HERMES_QWEN_BASE_URL)r   r   r   r   zgoogle-gemini-clizcloudcode-pa://googlelmstudior   )
LM_API_KEYzhttp://127.0.0.1:1234/v1LM_BASE_URL)r   r   r   r   r   copilot-acpexternal_processzacp://copilotCOPILOT_ACP_BASE_URLzgithub-copilot)COPILOT_GITHUB_TOKENGH_TOKEN)r   r   	anthropicanthropic_messages)ANTHROPIC_TOKENCLAUDE_CODE_OAUTH_TOKENzai)GLM_API_KEYZAI_API_KEYZ_AI_API_KEYGLM_BASE_URL)r   r   r   zkimi-for-codingKIMI_BASE_URL)r   r   stepfun)STEPFUN_API_KEYz#https://api.stepfun.ai/step_plan/v1STEPFUN_BASE_URL)r   r   r   r   minimaxMINIMAX_BASE_URLzminimax-oauthz https://api.minimax.io/anthropicz
minimax-cnMINIMAX_CN_BASE_URLdeepseekDEEPSEEK_BASE_URLalibabaDASHSCOPE_BASE_URLALIBABA_CODING_PLAN_BASE_URL)r   r   OPENCODE_ZEN_BASE_URL)r   r   r   OPENCODE_GO_BASE_URLKILOCODE_BASE_URLHF_BASE_URLzhttps://api.x.ai/v1XAI_BASE_URL)r   r   r   z#https://integrate.api.nvidia.com/v1NVIDIA_BASE_URLXIAOMI_BASE_URLTOKENHUB_BASE_URLzhttps://api.arcee.ai/api/v1ARCEE_BASE_URL)GMI_API_KEYzhttps://api.gmi-serving.com/v1GMI_BASE_URLOLLAMA_BASE_URLAZURE_FOUNDRY_BASE_URLbedrock_converseaws_sdk)r   r   )alibaba-coding-planvercelopencodeopencode-gokilohuggingfacexainvidiaxiaomitencent-tokenhubarceegmiollama-cloudzazure-foundrybedrockzDict[str, HermesOverlay]HERMES_OVERLAYSc                      e Zd ZU dZ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S )ProviderDefu9   Complete provider definition — merged from all sources.r   idnamer   r   api_key_env_varsr   base_urlr   Fr   r   r   r   docsourceN)r   r   r   r   r    rl   r   r   r   rm   rn   r   r!   r"   rh   rh      s         CCGGGIIINNN%%%%HMICMMMMFr!   rh   openaiglmzz-aizz.aizhipuzx-air^   zx.aigroknimr_   z
nvidia-nimzbuild-nvidianemotronkimizkimi-codingzkimi-coding-cnmoonshotstepzstepfun-coding-planzminimax-china
minimax_cnclaudezclaude-codecopilotgithubzgithub-copilot-acpz
ai-gatewayrY   	aigatewayzvercel-ai-gatewayzopencode-zenrZ   zengor[   zopencode-go-subkilocoder\   z	kilo-codezkilo-gatewayz	deep-seek	dashscopealiyunqwenzalibaba-cloudalibaba_codingrX   zalibaba-codingalibaba_coding_planz
gemini-clizgemini-oauthhfr]   zhugging-facezhuggingface-hubmimor`   zxiaomi-mimotencentra   tokenhubztencent-cloudtencentmaasawsre   zaws-bedrockzamazon-bedrockamazonzarcee-airb   arceeaiz	gmi-cloudrc   gmicloudz	lm-studio	lm_studioollamacustomvllmlocalllamacpp)z	llama.cppz	llama-cppzDict[str, str]ALIASESzNous PortalzOpenAI CodexzGitHub Copilot ACPzStepFun Step PlanzXiaomi MiMoz	GMI CloudzTencent TokenHubz	LM StudiozLocal endpointzAWS BedrockzOllama Cloud)r&   r(   r/   r>   r`   rc   ra   r,   r   re   rd   _LABEL_OVERRIDESchat_completions)r   r5   r)   rV   TRANSPORT_TO_API_MODErj   r   returnc                    |                                                                  }t                              ||          S )zResolve aliases and normalise casing to a canonical provider id.

    Returns the canonical id string.  Does *not* validate that the id
    corresponds to a known provider.
    )striplowerr   get)rj   keys     r"   normalize_providerr   {  s1     **,,



C;;sC   r!   Optional[ProviderDef]c                   t          |           }	 ddlm}  ||          }n# t          $ r d}Y nw xY wt                              |          }||r|j        nd}|r|j        nd}|r|j        nd}|r|j	        nd}|r|j
        nd}	t          |j                  }
|r*|j        r#|j        D ]}||
vr|
                    |           t          ||j        |t#          |
          |	p|j        ||||j        d	
  
        S |Ot          |t(                              ||          |j        |j        |j
        |j	        |j        |j        d
	  	        S dS )aA  Look up a built-in provider by id or alias.

    Resolution order:
      1. Hermes overlays (for providers not in models.dev: nous, openai-codex, etc.)
      2. models.dev catalog + Hermes overlay

    User-defined providers from config.yaml (``providers:`` / ``custom_providers:``)
    are resolved by :func:`resolve_provider_full`, which layers ``resolve_user_provider``
    and ``resolve_custom_provider`` on top of this function. Callers that need
    user-config support should use ``resolve_provider_full`` instead.

    Returns a fully-resolved ProviderDef or None.
    r   get_provider_infoNr   Fr   r   
models.dev)
ri   rj   r   rk   rl   r   r   r   rm   rn   hermes)	ri   rj   r   rk   rl   r   r   r   rn   )r   agent.models_devr   	Exceptionrf   r   r   r   r   r   r   listenvr   appendrh   rj   tupleapirm   r   )rj   	canonical_mdev_provider	mdev_infooverlayr   is_aggauthbase_url_envr   env_varsevs               r"   get_providerr     s    #4((IHHHHHH"N9--		   			 !!),,G)0CG%%m	*1<&&u$+:w  3:Bw//9@HG55b 	&& 	(w- 	(, ( (X%%OOB'''"8__&7)-) 
 
 
 	
 !%%i;;'$3.$5!/'

 

 

 
	
 4s   # 22provider_idc                    t          |           }|t          v rt          |         S t          |          }|r|j        S |S )z1Get a human-readable display name for a provider.)r   r   r   rj   )r   r   pdefs      r"   	get_labelr     sM    ";//I $$$	** 	""D yr!   providerr   c                6    t          |           }|r|j        ndS )z:Return True when the provider is a multi-model aggregator.F)r   r   )r   r   s     r"   r   r     s"    !!D!%0450r!   r   rl   c                8   t          |           }|p|rN|                    d                                          }d|v rdS |                    d          sd|v rdS d|v rdS t                              |j        d	          S | d
k    rdS |r|                    d                                          }t          |          }|                    d          s|dk    rdS |dk    rd|v rdS |dk    rdS |                    d          rt          |d          rdS d	S )u   Determine the API mode (wire protocol) for a provider/endpoint.

    Resolution order:
      1. Known provider → transport → TRANSPORT_TO_API_MODE.
      2. URL heuristics for unknown / custom providers.
      3. Default: 'chat_completions'.
    N/zapi.kimi.com/codingr5   z
/anthropiczapi.anthropic.comzapi.openai.comr)   r   re   rV   zapi.kimi.comz/codingzbedrock-runtime.zamazonaws.com)
r   rstripr   endswithr   r   r   r   
startswithr
   )r   rl   r   	url_lowerhostnames        r"   determine_api_moder     sq    !!D  	) ,,2244I$	11++!!,// ,3F)3S3S++9,,(($((9KLLL 9!!  
&OOC((..00	$X..l++ 	(x;N/N/N''~%%)y*@*@'''''$$122 	&7LXWf7g7g 	&%%r!   user_configDict[str, Any]c           
        |rt          |t                    sdS |                    |           }t          |t                    sdS |                    dd          p| }|                    dd          p-|                    dd          p|                    dd          pd}|                    dd          pd}|                    dd	          pd	}g }|r|                    |           t	          | ||t          |          |d
dd          S )zResolve a provider from the user's config.yaml ``providers:`` section.

    Args:
        name: Provider name as given by the user.
        user_config: The ``providers:`` dict from config.yaml.

    Returns:
        ProviderDef if found, else None.
    Nrj   r   r   urlrl   key_envr   r   Fr   user-configri   rj   r   rk   rl   r   r   rn   )
isinstancedictr   r   rh   r   )rj   r   entrydisplay_nameapi_urlr   r   r   s           r"   resolve_user_providerr     s(     jd;; tOOD!!EeT"" t 99VR((0DLiir""]eiir&:&:]eii
TV>W>W][]Gii	2&&,"G		+}55FIH !   x	 	 	 	r!   r   c                |    d|                                                                                      dd          z   S )zBuild a canonical slug for a custom_providers entry.

    Matches the convention used by runtime_provider and credential_pool
    (``custom:<normalized-name>``).  Centralised here so all call-sites
    produce identical slugs.
    zcustom: -)r   r   replace)r   s    r"   custom_provider_slugr   6  s7     |))++1133;;CEEEEr!   custom_providersOptional[List[Dict[str, Any]]]c                   |rt          |t                    sdS | pd                                                                }|sdS |dk    }d}|D ]}t          |t                    s|                    d          pd                                }|                    dd          p-|                    dd          p|                    dd          pd                                }|r|s|||f}t          |          }||                                |hvrt          ||dd	|d
dd          c S |r-|r+|\  }	}
t          |	          }t          ||	dd	|
d
dd          S dS )zIResolve a provider from the user's config.yaml ``custom_providers`` list.Nr   r   rj   rl   r   r   r   r   Fr   r   r   )r   r   r   r   r   r   r   rh   )rj   r   	requestedbare_custom_fallbackfirst_validr   r   r   slugdnameaurls              r"   resolve_custom_providerr   @  s   
  :.>#E#E t""$$**,,I t
 %0K! 
 
%&& 			&))/R6688IIj"%% yy##yy## 
%'' 	  	7 	 '1K#L11\//114888# 	
 	
 	
 		
 		
 		
  
 
!t#E**# 	
 	
 	
 		
 4r!   user_providersOptional[Dict[str, Any]]c                   t          |           }t          |          }||S |rLt          ||          }||S t          |                                                                 |          }||S t          | |          }||S 	 ddlm}  ||          }|$t          ||j	        d|j
        |j        d          S n# t          $ r Y nw xY wdS )u  Full resolution chain: built-in → models.dev → user config.

    This is the main entry point for --provider flag resolution.

    Args:
        name: Provider name or alias.
        user_providers: The ``providers:`` dict from config.yaml (optional).
        custom_providers: The ``custom_providers:`` list from config.yaml (optional).

    Returns:
        ProviderDef if found, else None.
    Nr   r   r   r   )ri   rj   r   rk   rl   rn   )r   r   r   r   r   r   r   r   rh   rj   r   r   r   )	rj   r   r   r   r   	user_pdefcustom_pdefr   r   s	            r"   resolve_provider_fullr     s&   " #4((I 	""D  ))^DD	 )$**,,*<*<*>*>OO	  *$0@AAKHHHHHH"N9--	 ^'!*"#    !     4s   6B> >
C
C)rj   r   r   r   )rj   r   r   r   )r   r   r   r   )r   r   r   r   )r   )r   r   rl   r   r   r   )rj   r   r   r   r   r   )r   r   r   r   )rj   r   r   r   r   r   )NN)rj   r   r   r   r   r   r   r   )"r   
__future__r   loggingdataclassesr   typingr   r   r   r   r	   utilsr
   r   	getLoggerr   loggerr   rf   r    rh   r   r   r   r   r   r   r   r   r   r   r   r   r   r!   r"   <module>r      sB    & # " " " " "  ! ! ! ! ! ! 3 3 3 3 3 3 3 3 3 3 3 3 3 3 : : : : : : : :		8	$	$ $       \---*.	  \- MM%E  \- MM#"A  \-" --"5/	  #\-. "1  /\-8 &4&  9\-F ==#$)/	  G\-R mm;  S\-Z &E  [\-b 
==E'  c\-l }}(  m\-t }}+?+	  u\-@ }}&+  A\-H ]]&"<  I\-R --&.  S\-Z ,  [\-b }}-  c\-j )=7   m   0  
 !=/  
 M,  
 !=&  
 =#/'  
 m?*  
 m*   &,   ]7)  
 =':'	   "M*   #]1   }$  q\- \- \- \ \ \ \D        &ulu
 
5u Eu Eu Uu Eu Eu Eu  
8!u" (#u$ H%u& 'u, -u. $/u0 '1u2 !3u8 I9u u: 9;u@ \AuB ,CuH kIuJ ;KuP QuR SuT -UuZ ([u\ ]u^ _ud Jeuf 
:gul 	-mun }out uuv wu u ux Fyu~ uD EuF iGuH IIuJ YKuL +MuN +OuP 0QuV %WuX 'Yu` 	-aub Mcud }euj Hkul 8mur !su u ut "uuv 'wux %yu~ 
9u@ 9AuB iCuD iEuJ KuL wMuR SuT UuZ 
[u\ ]u^ _u` haub Gcud eu uf iu u u u u u uz "'"*"$ $     $ &.(*	) )     ! ! ! !C C C CL   $1 1 1 1' ' ' ' 'X$ $ $ $NF F F FB B B BN 047;8 8 8 8 8 8 8r!   