
    jcY                    8   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 ed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d01          d2 ed3d41          d5 edd6d78          d9 edd:;          d< edd=d>d?@          dA ed3dB;          dC ed3ddD          dE ed3dF;          i dG eddH;          dI eddJ;          dK eddL;          dM edddNO          dP edddQO          dR edddSO          dT edddUO          dV edddWO          dX eddd          dY eddZd[          d\ edd];          d^ edd_;          d` eddadb          dc eddddedf@          dg eddhdi          dj eddk;          dl edmdno          Zdpedq<   e G dr ds                      Zi dtddud5dvd5dwd5dxd5dydXdzdXd{dXd|dddd}dd~dddYddYddYddYdd9i dd9dd9dd9dd<dd<ddEddEdd2dd2dd/dd/dd+ddMddMddPddPddRi ddRddRddGddIddIddIddIddKddKddKdd#dd#ddTddTddTddVddVi dd\dd\dd^dd^dd^dd^ddlddlddlddldd`dd`ddcddcd%d%dd%dd%ddddddZded<   dddddddddddddǜZded<   dd3ddmdʜ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       3/usr/local/lib/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
openai-apizhttps://api.openai.com/v1OPENAI_BASE_URL)r   r   r   	xai-oauthzhttps://api.x.ai/v1XAI_BASE_URL)r   r   r   r   z
qwen-oauthzhttps://portal.qwen.ai/v1HERMES_QWEN_BASE_URLz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_URLzalibaba-coding-planALIBABA_CODING_PLAN_BASE_URLopencodeOPENCODE_ZEN_BASE_URL)r   r   r   zopencode-goOPENCODE_GO_BASE_URLkiloKILOCODE_BASE_URLhuggingfaceHF_BASE_URLnovitaNOVITA_BASE_URLxainvidiaz#https://integrate.api.nvidia.com/v1NVIDIA_BASE_URLxiaomiXIAOMI_BASE_URLtencent-tokenhubTOKENHUB_BASE_URLarceezhttps://api.arcee.ai/api/v1ARCEE_BASE_URLgmi)GMI_API_KEYzhttps://api.gmi-serving.com/v1GMI_BASE_URLollama-cloudzhttps://ollama.com/v1OLLAMA_BASE_URLzazure-foundryAZURE_FOUNDRY_BASE_URLbedrockbedrock_converseaws_sdk)r   r   z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    rm   r   r   r   rn   ro   r   r!   r"   ri   ri      s         CCGGGIIINNN%%%%HMICMMMMFr!   ri   openaiglmzz-aizz.aizhipuzx-aizx.aigrokz
grok-oauthz
x-ai-oauthzxai-grok-oauthnimz
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opencode-zenzengozopencode-go-subkilocodez	kilo-codezkilo-gatewayz	deep-seek	dashscopealiyunqwenzalibaba-cloudalibaba_codingzalibaba-codingalibaba_coding_planz
gemini-clizgemini-oauthhfzhugging-facezhuggingface-hubz	novita-ainovitaaimimozxiaomi-mimotencenttokenhubztencent-cloudtencentmaasawszaws-bedrockzamazon-bedrockamazonzarcee-aiarceeaiz	gmi-cloudgmicloudz	lm-studio	lm_studiocustomlocal)ollamavllmllamacppz	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 Cloudz%xAI Grok OAuth (SuperGrok / Premium+))r&   r(   r2   rA   rX   r^   rZ   r/   r   rd   ra   r,   _LABEL_OVERRIDESchat_completions)r   r8   r)   re   TRANSPORT_TO_API_MODErk   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)rk   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)
rj   rk   r   rl   rm   r   r   r   rn   ro   hermes)	rj   rk   r   rl   rm   r   r   r   ro   )r   agent.models_devr   	Exceptionrg   r   r   r   r   r   r   listenvr   appendri   rk   tupleapirn   r   )rk   	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   rk   )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   rm   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/codingr8   z
/anthropiczapi.anthropic.comzapi.openai.comr)   r   rd   re   zapi.kimi.comz/codingzbedrock-runtime.zamazonaws.com)
r   rstripr   endswithr   r   r   r   
startswithr
   )r   rm   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.
    Nrk   r   r   urlrm   key_envr   r   Fr   user-configrj   rk   r   rl   rm   r   r   ro   )
isinstancedictr   r   ri   r   )rk   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   G  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   rk   rm   r   r   r   r   Fr   r   r   )r   r   r   r   r   r   r   ri   )rk   r   	requestedbare_custom_fallbackfirst_validr   r   r   slugdnameaurls              r"   resolve_custom_providerr   Q  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   )rj   rk   r   rl   rm   ro   )r   r   r   r   r   r   r   r   ri   rk   r   r   r   )	rk   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)rk   r   r   r   )rk   r   r   r   )r   r   r   r   )r   r   r   r   )r   )r   r   rm   r   r   r   )rk   r   r   r   r   r   )r   r   r   r   )rk   r   r   r   r   r   )NN)rk   r   r   r   r   r   r   r   )"r   
__future__r   loggingdataclassesr   typingr   r   r   r   r	   utilsr
   r   	getLoggerr   loggerr   rg   r    ri   r   r   r   r   r   r   r   r   r   r   r   r   r   r!   r"   <module>r      s3    & # " " " " "  ! ! ! ! ! ! 3 3 3 3 3 3 3 3 3 3 3 3 3 3 : : : : : : : :		8	$	$ $       i---*.	  i- MM%E  i- MM#"A  i-" --#5*  #i-, #"/'	  -i-8 --"5/	  9i-D "1  Ei-N &4&  Oi-\ ==#$)/	  ]i-h mm;  ii-p &E  qi-x 
==E'  yi-B }}(  Ci-J }}+?+	  Ki-V }}&+  Wi-^ ]]&"<  _i-h --&.  ii- i-p ,  qi-x }}-  yi-@ ==7  Ai-H 0  Ii-R ==/  Si-\ MM,  ]i-f ==&  gi-p mm*  qi-z 
==#/'  {i-D mm?*  Ei-N mm*  Oi-V ,  Wi-^ ]]7)  _i-h 
==':'	  ii-t MM1*  ui-B ]]1  Ci-J }}$  Ki- i- i i i i^        &xlx
 
5x Ex Ex Ux Ex Ex Ex +x x  +!x" k#x( 
8)x* (+x, H-x. /x4 5x x6 $7x8 '9x: !;x@ IAxB 9CxH \IxJ ,KxP kQxR ;SxX YxZ [x\ -]xb Jcxd 
:exj 	-kxl }mxr sx x xt uxv Fwx| }xB CxD iExF IGxH YIxJ +KxL +MxN 0OxT %UxV 'Wx^ 	-_x` Maxb }cxh ixj kx x xp Hqxr 8sxx !yxz "{x| '}x~ %xD 
9ExF 9GxH iIxJ iKxP QxR wSxX YxZ [x` 
axb cxd ex xf ox x x x x x x@ "'"*"8$ $     & &.(*	) )     ! ! ! !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!   