
    i!3                       U d 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	 ddl
mZ ddlmZmZmZmZmZ  ej                    dk    Z ej        d          Z eh d          ZddlZdd	lmZmZ dd
lmZ dZdddddZdee          fdZ!de"fdZ#dee          fdZ$de fdZ%dde de fdZ&dde fdZ'dee(         fdZ)ddl*m+Z+ defdZ,defdZ-defdZ.d Z/de"fdZ0d  Z1d!eddfd"Z2d# Z3d!efd$Z4i d%d&d'i d(g d)i d*d+gd,d-d.d/d&d0d1d2d3d4i d5d6d7d0d8d9d:d;d<g d=d>d?g d@i dAdBdCd>dDd>dEdFdGdHdIdJdKdLdMg dNdOdPdLidQdRdSdOdOdTdOidUdVdLdWdXdYdZd[dLd\d]d^d_d`d&dLg dadbd&d&dcdddedfdgdOdhdii djdkd0d&d&d&dRdSdld0d&d&d&dmdnd0d&d&d&dRdnd0d&d&d&dSdnd0d&d&d&dSdnd0d&d&d&dSdnd0d&d&d&dSdnd0d&d&d&dSdndodpdOdqdrdsdOdOdOdLdOdtdLdOi di dudvdwdtidxdydOii dzd{d|d}id~dddddddddddddd&d&ddddddLd6dd&dd%did%didd|ddRdOdddddddddddiddLdLddd&ddd&d&d&d&dWd&ddd&ddg idi dd&ddLd&d&dLdLi ddi ddi iddi iddi iddd/dg i i dLdLdddLdOg g ddddLiddddddOiddZ5g d¢ddgg dŢdgdgdȜZ6ee7ee          f         e8d<   i Z9i dddddOddLdΜdddddLddgddLd՜dddddLddLdΜdddddLddLdΜdddddOddLdΜdddddLddLdΜdddddOddLdΜdddddLddLdΜdddddLddLdΜdddddLddLdΜdddddOddLdΜdddddLddLdΜdddddOddLdΜdddddLddLdΜdddddLddLdΜddd ddOddLdΜdddddLddLdΜi dddddOddLdΜdd	d
ddLddLdΜdddddOddLdΜdddddLd͐ddddd&dOd͐ddddddLd͐ddddd&dOddLdΜddd ddOddLdΜd!d"d#d$dOddLdΜd%d&d'd$dLddLdΜd(d)d*ddOddLdΜd+d,d-d.dLddLdΜd/d0d1ddOddLdΜd2d3d4d.dLddLdΜd5d6d7ddOddLdΜd8d9d:d;dLd͐dd<d=d>ddOddLdΜi d?d@dAdBdLddLdΜdCdDdEddOddLdΜdFdGdHdIdLd͐ddJdKdLddOddLdΜdMdNdOdPdOddLdΜdQdRdSddOddLdΜdTdUdVdWdXdYgdLdZd[d\d]d^d_dXdYgdLdZd[d`dadbdcdXdYgdLdZd[dddedfddOdZdLdΜdgdhdiddOdZdLdΜdjdkdlddOdZdLdΜdmdndoddOdZdLdΜdpdqdrddLdZdLdΜdƐdsdtdug dvdLdZd[dwdxdydzd{d|gdLdZd[d}d~ddzd{d|gdOdZd[i ddddd{d|gdLdZd[dddd{d|gdOdZdddddd{d|gdOdZd[dddddgdLdZd[ddddg ddLdZd[ddddddgdLdZd[dÐdddddgdLdZd[dĐddddLdZddddddLdZddddddLdZddddddgdLdZd[ddddZddddddLdddddddOddddddOddddÐdĐddLdddƐdǐdddOddi dɐdʐdddOddd̐d͐dΐddLdddАdѐdҐddLdddӐdԐdՐddOdddאdؐdddLdddڐdېdddOdddݐdސdddOdddddddOdddddddOdddddddLdddddddOdddddddOdddddddOddLdΜdddddOddLdΜdddddOddLdΜdddddOddLdΜdddddLddLdΜi dd dddOdddddddLdddddddOddd	d
dddddddddddddLddddddddddddddddddddddd d!d"ddd#d$d%ddd&d'd(ddOddLdΜd)d*d+ddOddLdΜd,d-d.ddLddLdΜd/d0d1ddOddLdΜd2d3d4ddOddLdΜd5d6ddOddLdΜd7d8ddOddLdΜd9d:ddLddLdΜd;d<ddOddd=d>ddOddd?d@ddLdddAdBddLdCddDdEddOdCddFdGddOdCddFdHddOdCddIdJddOdCddKdLddOdCddMZ:ddNe"deee ef                  fdOZ;dPe(dQe fdRZ<deee ef                  fdSZ=deee ef                  fdTZ>d&dUdVedWe deee ef                  fdXZ?dYedeee ef                  fdZZ@	 ddPeee ef                  deee ef                  fd[ZAdee7e7f         fd\ZBh d]ZCh d^ZDh d_ZEe	 G d` da                      ZFddPeee ef                  deda         fdbZGddPeee ef                  ddfdcZHddPeee ef                  ddfddZIddee"dfe"dee ef         fdgZJde(dhe(de(fdiZKdj ZLdPee ef         dee ef         fdkZMdPee ef         dee ef         fdlZNdee ef         fdmZOdee ef         fdnZPdoZQdpZRdqZSdPee ef         fdrZTdee e f         fdsZUdteVdeVfduZWde7fdvZXdwe dxe de fdyZYdwe dxe fdzZZdwe de"fd{Z[ddxe fd|Z\dd}Z]ddxe fd~Z^dwe dxe dee ef         fdZ_de7fdZ`dwe dee          fdZadwe de fdZbd Zcd Zddwe dxe fdZed ZfdS (  a  
Configuration management for Hermes Agent.

Config files are stored in ~/.hermes/ for easy access:
- ~/.hermes/config.yaml  - All settings (model, toolsets, terminal, etc.)
- ~/.hermes/.env         - API keys and secrets

This module provides:
- hermes config          - Show current configuration
- hermes config edit     - Open config in editor
- hermes config set      - Set a specific value
- hermes config wizard   - Re-run setup wizard
    N)	dataclass)Path)DictAnyOptionalListTupleWindowsz^[A-Za-z_][A-Za-z0-9_]*$>@   	QQ_APP_IDQQ_STT_MODELTERMINAL_ENVWECOM_BOT_IDWECOM_SECRETWEIXIN_TOKENFEISHU_APP_IDWHATSAPP_MODEOPENAI_API_KEYQQ_STT_API_KEYSIGNAL_ACCOUNTANTHROPIC_TOKENMATRIX_PASSWORDOPENAI_BASE_URLQQ_HOME_CHANNELQQ_STT_BASE_URLSIGNAL_HTTP_URLWEIXIN_BASE_URLMATRIX_DEVICE_IDMATRIX_HOME_ROOMQQ_ALLOWED_USERSQQ_CLIENT_SECRETTERMINAL_SSH_KEYWEIXIN_DM_POLICYWHATSAPP_ENABLEDANTHROPIC_API_KEYFEISHU_APP_SECRETMATRIX_ENCRYPTIONTERMINAL_SSH_PORTWEIXIN_ACCOUNT_IDDINGTALK_CLIENT_IDFEISHU_ENCRYPT_KEYMATRIX_AUTO_THREADQQ_ALLOW_ALL_USERSMATRIX_RECOVERY_KEYQQ_MARKDOWN_SUPPORTWECOM_CALLBACK_HOSTWECOM_CALLBACK_PORTWEIXIN_CDN_BASE_URLWEIXIN_GROUP_POLICYWEIXIN_HOME_CHANNELBLUEBUBBLES_PASSWORDDISCORD_HOME_CHANNELQQ_HOME_CHANNEL_NAMESIGNAL_ALLOWED_USERSWECOM_CALLBACK_TOKENWEIXIN_ALLOWED_USERSMATTERMOST_REPLY_MODETELEGRAM_HOME_CHANNELBLUEBUBBLES_SERVER_URLDINGTALK_CLIENT_SECRETMATRIX_REQUIRE_MENTIONQQ_GROUP_ALLOWED_USERSWECOM_CALLBACK_CORP_IDWEIXIN_ALLOW_ALL_USERSMATTERMOST_HOME_CHANNELWECOM_CALLBACK_AGENT_IDWEIXIN_HOME_CHANNEL_NAMEFEISHU_VERIFICATION_TOKENMATRIX_FREE_RESPONSE_ROOMSSIGNAL_GROUP_ALLOWED_USERSWECOM_CALLBACK_CORP_SECRETWEIXIN_GROUP_ALLOWED_USERSWECOM_CALLBACK_ENCODING_AES_KEY)Colorscolor)DEFAULT_SOUL_MD)true1yesHomebrewNixOS)brewhomebrewnixnixosreturnc                     t          j        dd                                          } | r:|                                 }|t          v rdS t
                              ||           S t                      dz  }|                                rdS dS )z7Return the package manager owning this install, if any.HERMES_MANAGED rR   z.managedN)	osgetenvstriplower_MANAGED_TRUE_VALUES_MANAGED_SYSTEM_NAMESgetget_hermes_homeexists)raw
normalizedmanaged_markers      9/home/agentuser/.hermes/hermes-agent/hermes_cli/config.pyget_managed_systemrh   L   s    
)$b
)
)
/
/
1
1C
 :YY[[
---7$((S999$&&3N w4    c                  "    t                      duS )a  Check if Hermes is running in package-manager-managed mode.

    Two signals: the HERMES_MANAGED env var (set by the systemd service),
    or a .managed marker file in HERMES_HOME (set by the NixOS activation
    script, so interactive shells also see it).
    Nrh    ri   rg   
is_managedrm   [   s     t++ri   c                  B    t                      } | dk    rdS | dk    rdS dS )z;Return the preferred upgrade command for a managed install.rQ   zbrew upgrade hermes-agentrR   zsudo nixos-rebuild switchNrk   )managed_systems    rg   get_managed_update_commandrp   e   s4    '))N##**  **4ri   c                  "    t                      pdS )z<Return the best update command for the current installation.zhermes update)rp   rl   ri   rg   recommended_update_commandrr   o   s    %'':?:ri   modify this Hermes installationactionc                    t                      pd}t          j        dd                                                                          }|dk    r|t
          v rdn|pd}d|  d| dS |d	k    r|pd
}d|  d| dS d|  d| dS )z/Build a user-facing error for managed installs.za package managerrY   rZ   rR   rN   zCannot z?: this Hermes installation is managed by NixOS (HERMES_MANAGED=ze).
Edit services.hermes-agent.settings in your configuration.nix and run:
  sudo nixos-rebuild switchrQ   rT   zB: this Hermes installation is managed by Homebrew (HERMES_MANAGED=z#).
Use:
  brew upgrade hermes-agentz): this Hermes installation is managed by z:.
Use your package manager to upgrade or reinstall Hermes.)rh   r[   r\   r]   r^   r_   )rt   ro   rd   env_hints       rg   format_managed_messagerw   t   s    '))@-@N
)$b
)
)
/
/
1
1
7
7
9
9C   $88866cmV*f * *'* * *	
 ##$**f * *'* * *	
	C& 	C 	C> 	C 	C 	Cri   modify configurationc                 V    t          t          |           t          j                   dS )z+Print user-friendly error for managed mode.fileN)printrw   sysstderr)rt   s    rg   managed_errorr      s&    	
 
(
(sz::::::ri   c                     t           j                            d          dk    rdS ddlm}   |             rdS t                      dz  }	 i }t          |d          5 }|D ]q}|                                }d|v rW|                    d	          sB|	                    d          \  }}}|                                ||                                <   r	 ddd           n# 1 swxY w Y   n# t          $ r Y dS w xY w|                    d
d          }|                    dd          }	|                    dd          }
|                    dd          }||	|
|dS )a  Read container mode metadata from HERMES_HOME/.container-mode.

    Returns a dict with keys: backend, container_name, exec_user, hermes_bin
    or None if container mode is not active, we're already inside the
    container, or HERMES_DEV=1 is set.

    The .container-mode file is written by the NixOS activation script when
    container.enable = true. It tells the host CLI to exec into the container
    instead of running locally.
    
HERMES_DEVrO   Nr   )is_containerz.container-moder=#backenddockercontainer_namezhermes-agent	exec_userhermes
hermes_binz /data/current-package/bin/hermes)r   r   r   r   )r[   environra   hermes_constantsr   rb   openr]   
startswith	partitionFileNotFoundError)r   container_mode_fileinfoflinekey_valuer   r   r   r   s               rg   get_container_exec_infor      s    
z~~l##s**t------|~~ t)++.??	%s++ 	6q 6 6zz||$;;ts';';;$(NN3$7$7MCE(-D%	6	6 	6 	6 	6 	6 	6 	6 	6 	6 	6 	6 	6 	6 	6 	6    tt hhy(++GXX.??Nh//I,(JKKJ ( 	  s7   
C* A5CC* C""C* %C"&C* *
C87C8rb   c                  $    t                      dz  S )zGet the main config file path.zconfig.yamlr   rl   ri   rg   get_config_pathr      s    },,ri   c                  $    t                      dz  S )z&Get the .env file path (for API keys).z.envr   rl   ri   rg   get_env_pathr      s    v%%ri   c                  b    t          t                    j        j                                        S )z'Get the project installation directory.)r   __file__parentresolverl   ri   rg   get_project_rootr      s     >> '//111ri   c                 6   t                      rdS 	 t          j                            dd                                          }|rt          |d          nd}n# t          $ r d}Y nw xY w	 t          j        | |           dS # t          t          f$ r Y dS w xY w)uM  Set directory to owner-only access (0700 by default). No-op on Windows.

    Skipped in managed mode — the NixOS module sets group-readable
    permissions (0750) so interactive users in the hermes group can
    share state with the gateway service.

    The mode can be overridden via the HERMES_HOME_MODE environment variable
    (e.g. HERMES_HOME_MODE=0701) for deployments where a web server (nginx,
    caddy, etc.) needs to traverse HERMES_HOME to reach a served subdirectory.
    The execute-only bit on a directory permits cd-through without exposing
    directory listings.
    NHERMES_HOME_MODErZ      i  )
rm   r[   r   ra   r]   int
ValueErrorchmodOSErrorNotImplementedError)pathmode_strmodes      rg   _secure_dirr      s     || :>>"4b99??AA#+6s8Q   
t()   s$   AA A('A(,B BBc                     t           j                            d          st           j                            d          rdS t           j                            d          rdS 	 t          dd          5 } |                                 }ddd           n# 1 swxY w Y   d|v sd	|v sd
|v rdS n# t          t          f$ r Y nw xY wdS )a3  Detect if we're running inside a Docker/Podman/LXC container.

    When Hermes runs in a container with volume-mounted config files, forcing
    0o600 permissions breaks multi-process setups where the gateway and
    dashboard run as different UIDs or the volume mount requires broader
    permissions.
    HERMES_CONTAINERHERMES_SKIP_CHMODTz/.dockerenvz/proc/1/cgroupr   Nr   lxckubepodsF)	r[   r   ra   r   rc   r   readr   IOError)r   cgroup_contents     rg   _is_containerr      s    
z~~()) RZ^^<O-P-P t	w~~m$$ t"C(( 	&AVVXXN	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	&~%%.)@)@JR`D`D`4 EaW   5s6   #B. 3BB. BB. BB. .CCc                     t                      st                      rdS 	 t          j                            t          |                     rt          j        | d           dS dS # t          t          f$ r Y dS w xY w)uQ  Set file to owner-only read/write (0600). No-op on Windows.

    Skipped in managed mode — the NixOS activation script sets
    group-readable permissions (0640) on config files.

    Skipped in containers — Docker/Podman volume mounts often need broader
    permissions.  Set HERMES_SKIP_CHMOD=1 to force-skip on other systems.
    Ni  )	rm   r   r[   r   rc   strr   r   r   )r   s    rg   _secure_filer     s     || } 7>>#d))$$ 	"HT5!!!!!	" 	"()   s   AA% %A:9A:homec                     | dz  }|                                 rdS |                    t          d           t          |           dS )zISeed a default SOUL.md into HERMES_HOME if the user doesn't have one yet.zSOUL.mdNutf-8encoding)rc   
write_textrM   r   )r   	soul_paths     rg   _ensure_default_soul_mdr     sR    y I 7;;;ri   c                     t                      } t                      rSt          j        d          }	 t	          |            t          j        |           dS # t          j        |           w xY w|                     dd           t          |            dD ]-}| |z  }|                    dd           t          |           .t          |            dS )a   Ensure ~/.hermes directory structure exists with secure permissions.

    In managed mode (NixOS), dirs are created by the activation script with
    setgid + group-writable (2770). We skip mkdir and set umask(0o007) so
    any files created (e.g. SOUL.md) are group-writable (0660).
       T)parentsexist_okcronsessionslogsmemoriesN)rb   rm   r[   umask_ensure_hermes_home_managedmkdirr   r   )r   	old_umasksubdirds       rg   ensure_hermes_homer   '  s     D|| &HUOO		 '---HYBHY

4$
///D> 	 	FvAGGD4G000NNNN%%%%%s   A A-c                     |                                  st          d|  d          dD ]-}| |z  }|                                 st          | d          .t          |            dS )zPManaged-mode variant: verify dirs exist (activation creates them), seed SOUL.md.zHERMES_HOME z7 does not exist. Run 'sudo nixos-rebuild switch' first.r   N)is_dirRuntimeErrorr   )r   r   r   s      rg   r   r   ?  s    ;;== 
54 5 5 5
 
 	
 ;  6Mxxzz 	 9 9 9  	 D!!!!!ri   modelrZ   	providersfallback_providerscredential_pool_strategiestoolsetsz
hermes-cliagentZ   i  <   autoi  iX  )	max_turnsgateway_timeoutrestart_drain_timeoutservice_tiertool_use_enforcementgateway_timeout_warninggateway_notify_intervalterminalr   local
modal_modecwd.timeout   env_passthroughdocker_image*nikolaik/python-nodejs:python3.11-nodejs20docker_forward_env
docker_envsingularity_image3docker://nikolaik/python-nodejs:python3.11-nodejs20modal_imagedaytona_imagecontainer_cpu   container_memoryi   container_diski   container_persistentTdocker_volumesdocker_mount_cwd_to_workspaceFpersistent_shellbrowserx      managed_persistence)inactivity_timeoutcommand_timeoutrecord_sessionsallow_private_urlscamofoxcheckpoints2   )enabledmax_snapshotsfile_read_max_charsi compression      ?皙?   )r
  	thresholdtarget_ratioprotect_last_nbedrocki  )r
  provider_filterrefresh_intervalasyncdisabled)guardrail_identifierguardrail_versionstream_processing_modetrace)region	discovery	guardrailsmart_model_routing      )r
  max_simple_charsmax_simple_wordscheap_model	auxiliary)providerr   base_urlapi_keyr   download_timeoutih  )r'  r   r(  r)  r   )visionweb_extractr  session_search
skills_hubapprovalmcpflush_memoriesdisplaykawaiifull	interruptdefault)compactpersonalityresume_displaybusy_input_modebell_on_completeshow_reasoning	streaminginline_diffs	show_costskininterim_assistant_messagestool_progress_commandtool_progress_overridestool_preview_length	platforms	dashboardthemeprivacy
redact_piittsedgevoicezen-US-AriaNeuralpNInz6obpgDQGcFmaJgBeleven_multilingual_v2)voice_idmodel_idzgpt-4o-mini-ttsalloy)r   rL  eveeni]  i  )rO  languagesample_ratebit_ratezvoxtral-mini-tts-2603z$c69964a6-ab8b-4f8a-9465-ec0925096ec8)r   rO  zneuphonic/neutts-air-q4-ggufcpu)	ref_audioref_textr   device)r'  rK  
elevenlabsopenaixaimistralneuttssttbase)r   rT  z	whisper-1zvoxtral-mini-latest)r
  r'  r   r\  r^  zctrl+b   g      @)
record_keymax_recording_secondsauto_ttssilence_thresholdsilence_durationhuman_delayoffi   i	  )r   min_msmax_mscontextengine
compressormemoryi  i_  )memory_enableduser_profile_enabledmemory_char_limituser_char_limitr'  
delegation)r   r'  r(  r)  max_iterationsreasoning_effortprefill_messages_fileskillsexternal_dirshonchotimezonediscord)require_mentionfree_response_channelsallowed_channelsauto_thread	reactionschannel_promptswhatsapptelegramr  slack
mattermost	approvalsmanual)r   r   tirith   )r
  domainsshared_files)redact_secretstirith_enabledtirith_pathtirith_timeouttirith_fail_openwebsite_blocklistwrap_responseINFO   )levelmax_size_mbbackup_count
force_ipv4   )command_allowlistquick_commandspersonalitiessecurityr   loggingnetwork_config_version)FIRECRAWL_API_KEYBROWSERBASE_API_KEYBROWSERBASE_PROJECT_IDFAL_KEYVOICE_TOOLS_OPENAI_KEYELEVENLABS_API_KEY)r#   r   WHATSAPP_ALLOWED_USERSSLACK_BOT_TOKENSLACK_APP_TOKENSLACK_ALLOWED_USERSTAVILY_API_KEYTERMINAL_MODAL_MODE)r     r  
      ENV_VARS_BY_VERSIONNOUS_BASE_URLzNous Portal base URL overridez.Nous Portal base URL (leave empty for default)r'  )descriptionprompturlpasswordcategoryadvancedOPENROUTER_API_KEYz>OpenRouter API key (for vision, web scraping helpers, and MoA)zOpenRouter API keyzhttps://openrouter.ai/keysvision_analyzemixture_of_agents)r  r  r  r  toolsr  r  GOOGLE_API_KEYz<Google AI Studio API key (also recognized as GEMINI_API_KEY)zGoogle AI Studio API keyz&https://aistudio.google.com/app/apikeyGEMINI_API_KEYz3Google AI Studio API key (alias for GOOGLE_API_KEY)zGemini API keyGEMINI_BASE_URLz"Google AI Studio base URL overridez)Gemini base URL (leave empty for default)XAI_API_KEYzxAI API keyzhttps://console.x.ai/XAI_BASE_URLzxAI base URL overridez&xAI base URL (leave empty for default)GLM_API_KEYzBZ.AI / GLM API key (also recognized as ZAI_API_KEY / Z_AI_API_KEY)zZ.AI / GLM API keyzhttps://z.ai/ZAI_API_KEYz$Z.AI API key (alias for GLM_API_KEY)zZ.AI API keyZ_AI_API_KEYGLM_BASE_URLzZ.AI / GLM base URL overridez-Z.AI / GLM base URL (leave empty for default)KIMI_API_KEYzKimi / Moonshot API keyzKimi API keyzhttps://platform.moonshot.cn/KIMI_BASE_URLz!Kimi / Moonshot base URL overridez'Kimi base URL (leave empty for default)KIMI_CN_API_KEYzKimi / Moonshot China API keyzKimi (China) API keyARCEEAI_API_KEYzArcee AI API keyzhttps://chat.arcee.ai/ARCEE_BASE_URLzArcee AI base URL overridez(Arcee base URL (leave empty for default)MINIMAX_API_KEYzMiniMax API key (international)zMiniMax API keyzhttps://www.minimax.io/MINIMAX_BASE_URLzMiniMax base URL overridez*MiniMax base URL (leave empty for default)MINIMAX_CN_API_KEYz MiniMax API key (China endpoint)zMiniMax (China) API keyzhttps://www.minimaxi.com/MINIMAX_CN_BASE_URLz!MiniMax (China) base URL overridez2MiniMax (China) base URL (leave empty for default)DEEPSEEK_API_KEYz+DeepSeek API key for direct DeepSeek accesszDeepSeek API Keyz&https://platform.deepseek.com/api_keys)r  r  r  r  r  DEEPSEEK_BASE_URLz'Custom DeepSeek API base URL (advanced)zDeepSeek Base URLDASHSCOPE_API_KEYz>Alibaba Cloud DashScope API key (Qwen + multi-provider models)zDashScope API Keyz-https://modelstudio.console.alibabacloud.com/DASHSCOPE_BASE_URLzGCustom DashScope base URL (default: coding-intl OpenAI-compat endpoint)zDashScope Base URLHERMES_QWEN_BASE_URLzBQwen Portal base URL override (default: https://portal.qwen.ai/v1)z.Qwen Portal base URL (leave empty for default)HERMES_GEMINI_CLIENT_IDzfGoogle OAuth client ID for google-gemini-cli (optional; defaults to Google's public gemini-cli client)uK   Google OAuth client ID (optional — leave empty to use the public default)z1https://console.cloud.google.com/apis/credentialsHERMES_GEMINI_CLIENT_SECRETz;Google OAuth client secret for google-gemini-cli (optional)z%Google OAuth client secret (optional)HERMES_GEMINI_PROJECT_IDz@GCP project ID for paid Gemini tiers (free tier auto-provisions)z;GCP project ID for Gemini OAuth (leave empty for free tier)OPENCODE_ZEN_API_KEYz=OpenCode Zen API key (pay-as-you-go access to curated models)zOpenCode Zen API keyzhttps://opencode.ai/authOPENCODE_ZEN_BASE_URLzOpenCode Zen base URL overridez/OpenCode Zen base URL (leave empty for default)OPENCODE_GO_API_KEYz<OpenCode Go API key ($10/month subscription for open models)zOpenCode Go API keyOPENCODE_GO_BASE_URLzOpenCode Go base URL overridez.OpenCode Go base URL (leave empty for default)HF_TOKENzVHugging Face token for Inference Providers (20+ open models via router.huggingface.co)zHugging Face Tokenz&https://huggingface.co/settings/tokensHF_BASE_URLz2Hugging Face Inference Providers base URL overridez%HF base URL (leave empty for default)OLLAMA_API_KEYu>   Ollama Cloud API key (ollama.com — cloud-hosted open models)zOllama Cloud API keyzhttps://ollama.com/settingsOLLAMA_BASE_URLz?Ollama Cloud base URL override (default: https://ollama.com/v1)z)Ollama base URL (leave empty for default)XIAOMI_API_KEYzNXiaomi MiMo API key for MiMo models (mimo-v2-pro, mimo-v2-omni, mimo-v2-flash)zXiaomi MiMo API Keyzhttps://platform.xiaomimimo.comXIAOMI_BASE_URLzFXiaomi MiMo base URL override (default: https://api.xiaomimimo.com/v1)z)Xiaomi base URL (leave empty for default)
AWS_REGIONz?AWS region for Bedrock API calls (e.g. us-east-1, eu-central-1)z
AWS RegionzIhttps://docs.aws.amazon.com/bedrock/latest/userguide/bedrock-regions.htmlAWS_PROFILEzFAWS named profile for Bedrock authentication (from ~/.aws/credentials)zAWS ProfileEXA_API_KEYz1Exa API key for AI-native web search and contentszExa API keyzhttps://exa.ai/
web_searchr,  tool)r  r  r  r  r  r  PARALLEL_API_KEYz5Parallel API key for AI-native web search and extractzParallel API keyzhttps://parallel.ai/r  z-Firecrawl API key for web search and scrapingzFirecrawl API keyzhttps://firecrawl.dev/FIRECRAWL_API_URLz6Firecrawl API URL for self-hosted instances (optional)z)Firecrawl API URL (leave empty for cloud)FIRECRAWL_GATEWAY_URLzQExact Firecrawl tool-gateway origin override for Nous Subscribers only (optional)z9Firecrawl gateway URL (leave empty to derive from domain)TOOL_GATEWAY_DOMAINzShared tool-gateway domain suffix for Nous Subscribers only, used to derive vendor hosts, e.g. nousresearch.com -> firecrawl-gateway.nousresearch.comzTool-gateway domain suffixTOOL_GATEWAY_SCHEMEzShared tool-gateway URL scheme for Nous Subscribers only, used to derive vendor hosts (`https` by default, set `http` for local gateway testing)zTool-gateway URL schemeTOOL_GATEWAY_USER_TOKENzuExplicit Nous Subscriber access token for tool-gateway requests (optional; otherwise read from the Hermes auth store)zTool-gateway user tokenz;Tavily API key for AI-native web search, extract, and crawlzTavily API keyzhttps://app.tavily.com/home)r  r,  	web_crawlr  uU   Browserbase API key for cloud browser (optional — local browser works without this)zBrowserbase API keyzhttps://browserbase.com/browser_navigatebrowser_clickr  uC   Browserbase project ID (optional — only needed for cloud browser)zBrowserbase project IDBROWSER_USE_API_KEYuU   Browser Use API key for cloud browser (optional — local browser works without this)zBrowser Use API keyzhttps://browser-use.com/FIRECRAWL_BROWSER_TTLz@Firecrawl browser session TTL in seconds (optional, default 300)zBrowser session TTL (seconds))r  r  r  r  r  CAMOFOX_URLzYCamofox browser server URL for local anti-detection browsing (e.g. http://localhost:9377)zCamofox server URLz)https://github.com/jo-inc/camofox-browserr  z FAL API key for image generationzFAL API keyzhttps://fal.ai/image_generateTINKER_API_KEYzTinker API key for RL trainingzTinker API keyz/https://tinker-console.thinkingmachines.ai/keys)rl_start_trainingrl_check_statusrl_stop_trainingWANDB_API_KEYz0Weights & Biases API key for experiment trackingzWandB API keyzhttps://wandb.ai/authorizerl_get_resultsr  z?OpenAI API key for voice transcription (Whisper) and OpenAI TTSz&OpenAI API Key (for Whisper STT + TTS)z$https://platform.openai.com/api-keysvoice_transcription
openai_ttsz4ElevenLabs API key for premium text-to-speech voiceszElevenLabs API keyzhttps://elevenlabs.io/MISTRAL_API_KEYz7Mistral API key for Voxtral TTS and transcription (STT)zMistral API keyzhttps://console.mistral.ai/GITHUB_TOKENzCGitHub token for Skills Hub (higher API rate limits, skill publish)zGitHub Tokenz"https://github.com/settings/tokensHONCHO_API_KEYz.Honcho API key for AI-native persistent memoryzHoncho API keyzhttps://app.honcho.devhoncho_contextHONCHO_BASE_URLz=Base URL for self-hosted Honcho instances (no API key needed)z,Honcho base URL (e.g. http://localhost:8000))r  r  r  TELEGRAM_BOT_TOKENz"Telegram bot token from @BotFatherzTelegram bot tokenzhttps://t.me/BotFather	messagingTELEGRAM_ALLOWED_USERSzSComma-separated Telegram user IDs allowed to use the bot (get ID from @userinfobot)z+Allowed Telegram user IDs (comma-separated)zhttps://t.me/userinfobotTELEGRAM_PROXYzaProxy URL for Telegram connections (overrides HTTPS_PROXY). Supports http://, https://, socks5://zTelegram proxy URL (optional))r  r  r  r  DISCORD_BOT_TOKENz'Discord bot token from Developer PortalzDiscord bot tokenz+https://discord.com/developers/applicationsDISCORD_ALLOWED_USERSz7Comma-separated Discord user IDs allowed to use the botz*Allowed Discord user IDs (comma-separated)DISCORD_REPLY_TO_MODEzDiscord reply threading mode: 'off' (no reply references), 'first' (reply on first message only, default), 'all' (reply on every chunk)z"Discord reply mode (off/first/all)r  zSlack bot token (xoxb-). Get from OAuth & Permissions after installing your app. Required scopes: chat:write, app_mentions:read, channels:history, groups:history, im:history, im:read, im:write, users:read, files:read, files:writezSlack Bot Token (xoxb-...)zhttps://api.slack.com/appsr  u   Slack app-level token (xapp-) for Socket Mode. Get from Basic Information → App-Level Tokens. Also ensure Event Subscriptions include: message.im, message.channels, message.groups, app_mentionzSlack App Token (xapp-...)MATTERMOST_URLz3Mattermost server URL (e.g. https://mm.example.com)zMattermost server URLzhttps://mattermost.com/deploy/MATTERMOST_TOKENz-Mattermost bot token or personal access tokenzMattermost bot tokenMATTERMOST_ALLOWED_USERSz:Comma-separated Mattermost user IDs allowed to use the botz-Allowed Mattermost user IDs (comma-separated)MATTERMOST_REQUIRE_MENTIONzaRequire @mention in Mattermost channels (default: true). Set to false to respond to all messages.zRequire @mention in channels!MATTERMOST_FREE_RESPONSE_CHANNELSzJComma-separated Mattermost channel IDs where bot responds without @mentionz+Free-response channel IDs (comma-separated)MATRIX_HOMESERVERz7Matrix homeserver URL (e.g. https://matrix.example.org)zMatrix homeserver URLz%https://matrix.org/ecosystem/servers/MATRIX_ACCESS_TOKENz3Matrix access token (preferred over password login)zMatrix access tokenMATRIX_USER_IDz)Matrix user ID (e.g. @hermes:example.org)zMatrix user ID (@user:server)MATRIX_ALLOWED_USERSzLComma-separated Matrix user IDs allowed to use the bot (@user:server format)z)Allowed Matrix user IDs (comma-separated)r>   zZRequire @mention in Matrix rooms (default: true). Set to false to respond to all messages.z&Require @mention in rooms (true/false)rF   zCComma-separated Matrix room IDs where bot responds without @mentionz(Free-response room IDs (comma-separated)r+   z@Auto-create threads for messages in Matrix rooms (default: true)z)Auto-create threads in rooms (true/false)r   zNStable Matrix device ID for E2EE persistence across restarts (e.g. HERMES_BOT)z)Matrix device ID (stable across restarts)r-   u   Matrix recovery key for cross-signing verification after device key rotation (from Element: Settings → Security → Recovery Key)zMatrix recovery keyr<   zOBlueBubbles server URL for iMessage integration (e.g. http://192.168.1.10:1234)zBlueBubbles server URLzhttps://bluebubbles.app/r4   uJ   BlueBubbles server password (from BlueBubbles Server → Settings → API)zBlueBubbles server passwordBLUEBUBBLES_ALLOWED_USERSzJComma-separated iMessage addresses (email or phone) allowed to use the botz,Allowed iMessage addresses (comma-separated)BLUEBUBBLES_ALLOW_ALL_USERSz-Allow all BlueBubbles users without allowlistzAllow All BlueBubbles Usersr   z.QQ Bot App ID from QQ Open Platform (q.qq.com)z	QQ App IDzhttps://q.qq.com)r  r  r  r  r    z*QQ Bot Client Secret from QQ Open PlatformzQQ Client Secretr   z2Comma-separated QQ user IDs allowed to use the botzQQ Allowed Usersr?   z=Comma-separated QQ group IDs allowed to interact with the botzQQ Group Allowed Usersr,   z4Allow all QQ users without an allowlist (true/false)zAllow All QQ Usersr   z<Default QQ channel/group for cron delivery and notificationszQQ Home Channelr6   z$Display name for the QQ home channelzQQ Home Channel Name
QQ_SANDBOXz;Enable QQ sandbox mode for development testing (true/false)zQQ Sandbox ModeGATEWAY_ALLOW_ALL_USERSzMAllow all users to interact with messaging bots (true/false). Default: false.zAllow all users (true/false)API_SERVER_ENABLEDzrEnable the OpenAI-compatible API server (true/false). Allows frontends like Open WebUI, LobeChat, etc. to connect.zEnable API server (true/false)API_SERVER_KEYzBearer token for API server authentication. Required for non-loopback binding; server refuses to start without it. On loopback (127.0.0.1), all requests are allowed if empty.z1API server auth key (required for network access)API_SERVER_PORTz(Port for the API server (default: 8642).zAPI server portAPI_SERVER_HOSTu   Host/bind address for the API server (default: 127.0.0.1). Use 0.0.0.0 for network access — server refuses to start without API_SERVER_KEY.zAPI server hostzModel name advertised on /v1/models. Defaults to the profile name (or 'hermes-agent' for the default profile). Useful for multi-user setups with OpenWebUI.zAPI server model nameu  URL of a remote Hermes API server to forward messages to (proxy mode). When set, the gateway handles platform I/O only — all agent work is delegated to the remote server. Use for Docker E2EE containers that relay to a host agent. Also configurable via gateway.proxy_url in config.yaml.z=Remote Hermes API server URL (e.g. http://192.168.1.100:8642)zBearer token for authenticating with the remote Hermes API server (proxy mode). Must match the API_SERVER_KEY on the remote host.zRemote API server auth keyzREnable the webhook platform adapter for receiving events from GitHub, GitLab, etc.zEnable webhooks (true/false)z1Port for the webhook HTTP server (default: 8644).zWebhook portz[Global HMAC secret for webhook signature validation (overridable per route in config.yaml).zWebhook secretzySudo password for terminal commands requiring root access; set to an explicit empty string to try empty without promptingzSudo passwordsettingz>Maximum tool-calling iterations per conversation (default: 90)zMax iterationsz=(deprecated) Use display.tool_progress in config.yaml insteadu.   Tool progress (deprecated — use config.yaml)u.   Progress mode (deprecated — use config.yaml)zFPath to JSON file with ephemeral prefill messages for few-shot primingzPrefill messages file pathzOEphemeral system prompt injected at API-call time (never persisted to sessions)zEphemeral system prompt)API_SERVER_MODEL_NAMEGATEWAY_PROXY_URLGATEWAY_PROXY_KEYWEBHOOK_ENABLEDWEBHOOK_PORTWEBHOOK_SECRETSUDO_PASSWORDHERMES_MAX_ITERATIONSHERMES_TOOL_PROGRESSHERMES_TOOL_PROGRESS_MODEHERMES_PREFILL_MESSAGES_FILEHERMES_EPHEMERAL_SYSTEM_PROMPTrequired_onlyc                 :   g }t                                           D ]1\  }}t          |          s|                    d|i|ddi           2| sKt                                          D ]1\  }}t          |          s|                    d|i|ddi           2|S )z|
    Check which environment variables are missing.
    
    Returns list of dicts with var info for missing variables.
    nameis_requiredTF)REQUIRED_ENV_VARSitemsget_env_valueappendOPTIONAL_ENV_VARS)r%  missingvar_namer   s       rg   get_missing_env_varsr0    s     G ,1133 L L$X&& 	LNNFHJJmTJJKKK  Q/5577 	Q 	QNHd ** QODO-OOPPPNri   config
dotted_keyc                     |                     d          }| }|dd         D ];}||vs(t          |                    |          t                    si ||<   ||         }<|||d         <   dS )zSet a value at an arbitrarily nested dotted key path.

    Creates intermediate dicts as needed, e.g. ``_set_nested(c, "a.b.c", 1)``
    ensures ``c["a"]["b"]["c"] == 1``.
    r   N)split
isinstancera   dict)r1  r2  r   partscurrentparts         rg   _set_nestedr;    s     S!!EGcrc
    wjT1B1BD&I&IGDM$-GE"Iri   c                      t                      } g ddt          dt          dt          ffd t          |            S )z
    Check which config fields are missing or outdated (recursive).
    
    Walks the DEFAULT_CONFIG tree at arbitrary depth and reports any keys
    present in defaults but absent from the user's loaded config.
    rZ   defaultsr9  prefixc                 `   |                                  D ]\  }}|                    d          r|s|n| d| }||vr                    ||d| d           Gt          |t                    r;t          |                    |          t                    r |||         |           d S )Nr   r   zNew config option: )r   r6  r  )r*  r   r,  r6  r7  ra   )r=  r9  r>  r   default_valuefull_key_checkr.  s         rg   rB  z)get_missing_config_fields.<locals>._check  s    "*.."2"2 	> 	>C~~c"" "(?ss.?.?#.?.?H'!!#,#C#C#C       
 M400 >ZC@P@PRV5W5W >}gclH===	> 	>ri   )rZ   )load_configr7  r   DEFAULT_CONFIG)r1  rB  r.  s    @@rg   get_missing_config_fieldsrE    sj     ]]FG> > > >c > > > > > > > F>6"""Nri   c                     	 ddl m} m} n# t          $ r g cY S w xY w |             }|sg S t	                      }g }|D ]}| d|d          }|                    d          }|}d}	|D ](}
t          |t                    r|
|v r||
         }|}	&d}	 |	)t          |	t                    r)|		                                s|
                    |           |S )a3  Return skill-declared config vars that are missing or empty in config.yaml.

    Scans all enabled skills for ``metadata.hermes.config`` entries, then checks
    which ones are absent or empty under ``skills.config.<key>`` in the user's
    config.yaml.  Returns a list of dicts suitable for prompting.
    r   )discover_all_skill_config_varsSKILL_CONFIG_PREFIXr   r   N)agent.skill_utilsrG  rH  	ExceptionrC  r5  r6  r7  r   r]   r,  )rG  rH  all_varsr1  r.  varstorage_keyr8  r9  r   r:  s              rg   get_missing_skill_config_varsrN    s,   YYYYYYYYY   			 .-//H 	]]F$&G    ,;;s5z;;!!#&& 	 	D'4(( TW__!$-=Zs33=EKKMM=NN3Ns    provider_keyentryrP  c                t   t          | t                    sdS d}dD ]V}|                     |          }t          |t                    r*|                                r|                                } nW|sdS d}|                     d          }t          |t                    r)|                                r|                                }n(|                                r|                                }|sdS ||d}|                                }|r||d<   |                     d          }t          |t                    r+|                                r|                                |d<   |                     d          }	t          |	t                    r+|	                                r|	                                |d<   |                     d	          p|                     d
          }
t          |
t                    r+|
                                r|
                                |d	<   |                     d          p|                     d          }t          |t                    r+|                                r|                                |d<   |                     d          }t          |t                    r|r||d<   |                     d          }t          |t
                    r|dk    r||d<   |                     d          }t          |t
          t          f          r|dk    r||d<   |S )z>Return a runtime-compatible custom provider entry or ``None``.NrZ   )apir  r(  r'  )r'  r(  rP  r)  key_envapi_mode	transportr   default_modelmodelscontext_lengthr   rate_limit_delay)r6  r7  ra   r   r]   r   float)rQ  rP  r(  url_keyraw_urlr'  raw_namere   r)  rT  rU  
model_namerX  rY  rZ  s                  rg    _normalize_custom_provider_entryr`    s'    eT"" tH-  ))G$$gs## 	 	}}HE tDyy  H(C   $X^^%5%5 $~~					 $!!## t " "J
  %%''L 2%1
>"ii	""G'3 0GMMOO 0 '
9ii	""G'3 0GMMOO 0 '
9yy$$>		+(>(>H(C   2X^^%5%5 2!)!1!1
:7##Auyy'A'AJ*c"" 1z'7'7'9'9 1(..00
7YYx  F&$ &F &%
8YY/00N.#&& 6>A+=+='5
#$yy!344"S%L11 :6F!6K6K)9
%&ri   providers_dictc                     t          | t                    sg S g }|                                 D ]:\  }}t          |t	          |                    }||                    |           ;|S )zMNormalize ``providers`` config entries into the legacy custom-provider shape.rO  )r6  r7  r*  r`  r   r,  )ra  custom_providersr   rQ  re   s        rg   "providers_dict_to_custom_providersrd  5  sw    nd++ 	-/$**,, 0 0
U5e#c((SSS
!##J///ri   c                    | t                      } g t                      t                      dt          t          t          t
          f                  ddffd}|                     d          }|4t          |t                    sg S |D ]} |t          |                     t          |                     d                    D ]} ||           S )ay  Return a deduplicated custom-provider view across legacy and v12+ config.

    ``custom_providers`` remains the on-disk legacy format, while ``providers``
    is the newer keyed schema.  Runtime and picker flows still need a single
    list-shaped view, but we should not materialise that compatibility layer
    back into config.yaml because it duplicates entries in UIs.
    NrQ  rW   c                 >   | d S t          |                     dd          pd                                                                          }t          |                     dd          pd                                                                          }t          |                     dd          pd                                                              d                                          }t          |                     dd          pd                                                                          }|||f}|r|v rd S |r|r|v rd S                     |            |r                    |           |r|r                    |           d S d S d S )NrP  rZ   r'  r(  /r   )r   ra   r]   r^   rstripr,  add)	rQ  rP  r'  r(  r   pair
compatibleseen_name_url_pairsseen_provider_keyss	         rg   _append_if_newz7get_compatible_custom_providers.<locals>._append_if_newT  s   =F599^R88>B??EEGGMMOO599VR((.B//5577==??uyyR006B77==??FFsKKQQSSEIIgr**0b117799??AAh& 	L,>>>F 	H 	)<!<!<F%    	1""<000 	*H 	*##D)))))	* 	* 	* 	*ri   rc  r   )rC  setr   r   r   r   ra   r6  listr`  rd  )r1  rn  rc  rQ  rk  rl  rm  s       @@@rg   get_compatible_custom_providersrq  C  s    ~')J!ee"uu*htCH~6 *4 * * * * * * * *( zz"455#*D11 	I% 	D 	DEN;EBBCCCC3FJJ{4K4KLL  uri   c                      t                      } |                     dd          }t                              dd          }||fS )zS
    Check config version.
    
    Returns (current_version, latest_version).
    r  r   r   )rC  ra   rD  )r1  r9  latests      rg   check_config_versionrt  u  sA     ]]Fjj*A..G 1155FF?ri   >   r   r   ro  rl  r2  gatewayr   r   r&  r   rt  r  fallback_modelr  rc  r   r   >   r'  r   rX  r)  rU  r(  rY  rZ  >   r)  rU  r(  rZ  c                   2    e Zd ZU dZeed<   eed<   eed<   dS )ConfigIssuez$A detected config structure problem.severitymessagehintN)__name__
__module____qualname____doc__r   __annotations__rl   ri   rg   rx  rx    s1         ..MMMLLL
IIIIIri   rx  c                 t   | 2	 t                      } n"# t          $ r t          ddd          gcY S w xY wg }|                     d          }|t	          |t
                    r|                    t          ddd                     t	          |t
                    r!t          |                                          nt                      }|t          z  }|r5|                    t          dd	t          |           d
d                     nt	          |t                    rt          |          D ]\  }}t	          |t
                    s>|                    t          dd| dt          |          j         dd                     X|                    d          s(|                    t          dd| dd                     |                    d          s(|                    t          dd| dd                     |                     d          }|t	          |t
                    s:|                    t          ddt          |          j         d                     nt|rr|                    d          s$|                    t          ddd                     |                    d          s$|                    t          ddd                     t	          |t
                    r.d| vr*d|pi v r$|                    t          ddd                      |                     d          }|r&|s$|                    t          dd!d"                     | D ]V}	|	                    d#          r|	t           vr5|	t          v r,|                    t          dd$|	 d%d&|	 d'                     W|S )(a"  Validate config.yaml structure and return a list of detected issues.

    Catches common YAML formatting mistakes that produce confusing runtime
    errors (like "Unknown provider") instead of clear diagnostics.

    Can be called with a pre-loaded config dict, or will load from disk.
    NerrorzCould not load config.yamlz+Run 'hermes setup' to create a valid configrc  uO   custom_providers is a dict — it must be a YAML list (items prefixed with '-')zeChange to:
  custom_providers:
    - name: my-provider
      base_url: https://...
      api_key: ...warningzRoot-level keys z( look like custom_providers entry fieldszLThese should be indented under a '- name: ...' list entry, not at root levelzcustom_providers[z] is not a dict (got )z1Each entry should have at minimum: name, base_urlr'  z] is missing 'name' fieldz#Add a name, e.g.: name: my-providerr(  z] is missing 'base_url' fieldzDAdd the API endpoint URL, e.g.: base_url: https://api.example.com/v1rv  zAfallback_model should be a dict with 'provider' and 'model', got zZChange to:
  fallback_model:
    provider: openrouter
    model: anthropic/claude-sonnet-4r'  uH   fallback_model is missing 'provider' field — fallback will be disabledz/Add: provider: openrouter (or another provider)r   uE   fallback_model is missing 'model' field — fallback will be disabledz8Add: model: anthropic/claude-sonnet-4 (or another model)zGfallback_model appears inside custom_providers instead of at root levelzDMove fallback_model to the top level of config.yaml (no indentation)u[   custom_providers defined but no 'model' section — Hermes won't know which provider to useziAdd a model section:
  model:
    provider: custom
    default: your-model-name
    base_url: https://...r   zRoot-level key 'uW   ' looks misplaced — should it be under 'model:' or inside a 'custom_providers' entry?zMove 'z' under the appropriate section)rC  rJ  rx  ra   r6  r7  r,  ro  keys_CUSTOM_PROVIDER_LIKE_FIELDSsortedrp  	enumeratetyper|  r   _KNOWN_ROOT_KEYS)
r1  issuescpcp_keys
suspiciousirQ  fb	model_cfgr   s
             rg   validate_config_structurer    s    ~	w ]]FF 	w 	w 	w)EGtuuvvvv	w !#F 
&	'	'B	~b$ (	MM+a%     )32t(<(<Gc"''))nnn#%%G #??J kcvj'9'9cccb    
 D!! 	%bMM  5!%.. MM+![A[[DKKDX[[[K# #   
 yy(( MM+!HAHHH=# #   
 yy,, MM+!LALLL^# #    
$	%	%B	~"d## 	MM+gTXY[T\T\Tegg7       	66*%% k^E    
 66'?? k[N     "d  0 > >CSXZX`^`CaCakUR
 
 	 	 	 

7##I	 	) 	ki(
 
 	 	 	   >># 	&&&32N+N+NMM+3====     Ms    22c                 R   	 t          |           }n# t          $ r Y dS w xY w|sdS ddl}dg}|D ]1}|j        dk    rdnd}|                    d| d|j                    2|                    d	           |j                            d
                    |          dz              dS )zPrint config structure warnings to stderr at startup.

    Called early in CLI and gateway init so users see problems before
    they hit cryptic "Unknown provider" errors.  Prints nothing if
    config is healthy.
    Nr   u3   [33m⚠ Config issues detected in config.yaml:[0mr  u   [31m✗[0mu   [33m⚠[0m   z2  [2mRun 'hermes doctor' for fix suggestions.[0m


)	r  rJ  r}   ry  r,  rz  r~   writejoin)r1  r  r}   linescimarkers         rg   print_config_warningsr    s    *622    JJJHIE 1 1)+)?)?%%EY/&//2://0000	LLKLLLJTYYu%%./////s    
  c                    ddl }ddl}|j                            d          }|j                            d          }| !	 t	                      } n# t
          $ r Y dS w xY w|                     di           }t          |t                    r|                    dd          nd}|dv}g }|r|                    d	| d
           |r|s|                    d| d
           |r|j                            dd          }	|	                    dd           |                    d           |                    d|	 d           |j
                            d                    |          dz              dS dS )u   Warn if MESSAGING_CWD or TERMINAL_CWD is set in .env instead of config.yaml.

    These env vars are deprecated — the canonical setting is terminal.cwd
    in config.yaml.  Prints a migration hint to stderr.
    r   NMESSAGING_CWDTERMINAL_CWDr   r   r   )r   r   r   rZ   u     [33m⚠[0m MESSAGING_CWD=u&    found in .env — this is deprecated.u     [33m⚠[0m TERMINAL_CWD=HERMES_HOMEz	~/.hermesu/   [33m⚠ Deprecated .env settings detected:[0mzN  [2mMove to config.yaml instead:  terminal:\n    cwd: /your/project/path[0mz'  [2mThen remove the old entries from z	/.env[0mr  r  )r[   r}   r   ra   rC  rJ  r6  r7  r,  insertr~   r  r  )
r1  r[   r}   messaging_cwdterminal_cwd_envterminal_cfg
config_cwdconfig_has_explicit_cwdr  	hint_paths
             rg   warn_deprecated_cwd_env_varsr  1  s    NNNNNNNJNN?33Mz~~n55~	 ]]FF 	 	 	FF	 ::j"--L1;L$1O1OX!!%---UXJ(0HHE 
#- # # #	
 	
 	
  
 7 
#1A # # #	
 	
 	
  
4JNN=+>>	QOPPP>	
 	
 	
 	PPPP	
 	
 	
 	
5))F233333
4 
4s    A 
AAinteractivequietc           
      &  D g g g d}	 t                      }|r|st          d| d           n# t          $ r Y nw xY wt                      \  }}|dk     rFt	                      }|                    di           }t          |t                    si }d|vrt          d          }t          d          }	|r7|	                                d	v r!d
|d<   |d         
                    d           n|	r_|		                                dv rI|		                                |d<   |d         
                    d|		                                 d           n d|d<   |d         
                    d           ||d<   t          |           |st          d|d                     |dk     rt	                      }d|vrt          j        dd          }
|
r]|
                                rI|
                                |d<   |d         
                    d|
                                 d           n d|d<   |d         
                    d           t          |           |s|d         pd}t          d|            |dk     rD	 t          d          }|r!t          dd           |st          d           n# t          $ r Y nw xY w|dk     rt	                      }|                    d           }t          |t                     r|r|                    d!i           }t          |t                    si }d"}|D ]}t          |t                    s|                    d#d          }|                    d$d          p|                    d%d          pd}|                    d&d          }|sv|                                	                                                    d'd(                              d)d                              d*d          }d+|v r|                    d+d(          }d+|v |                    d(          }|sE	 d"d,lm}  ||          }|j        pd-                    d.d(          }n# t          $ r d/| }Y nw xY w||v r| d(| }d0|i}|r||d#<   |r	|d1vr||d&<   |                    d2          r|d2         |d3<   |                    d4          r|d4         |d5<   |||<   |d6z  }|d"k    r||d!<   |                    d d7           t          |           |sqt          d8| d9           t!          |                                          | d7         D ]3}||         }t          d:| d;|                    d0d                      4|d<k     rLd=D ]I}	 t          |          }|r%t          |d           |st          d>| d?           :# t          $ r Y Fw xY w|d@k     r]t/                      }|                    dAi           }t          |t                    r#d2|v r|d2         }|                    dBdC          }t	                      }|                    dAi           }|                    d2d7           |dDv rSh dE} || v rJ|                    dCi           }!t          |!t                    rd2|!vr|                    dCi           }"||"d2<   nJ|                    |i           }#t          |#t                    rd2|#vr|                    |i           }$||$d2<   ||dA<   t          |           |st          dF           |dGk     rt/                      }|                    di           }t          |t                    si }dH|vrEdI|dH<   ||d<   |d         
                    dJ           t          |           |st          dK           |dLk     rEt/                      }|                    di           }t          |t                    si }|                    dM          }%t          |%t                    r|%r|                    dNi           }&t          |&t                    si }&|%                                D ]#\  }'}(|'|&vri |&|'<   d|&|'         vr|(|&|'         d<   $|&|dN<   ||d<   t          |           |sCdO                    dP |%                                D                       })t          dQ|)            |d         
                    dR           |dSk     rt/                      }|                    dTi           }*t          |*t                    r|*                    dUd7          }+|*                    dVd7          },|*                    dWd7          }-g }.|+rt7          |+                                          r}|                    dXi           }/|/                    dTi           }0|0                    d2          s<t7          |+                                          |0d2<   |.
                    dY|+            |,rt7          |,                                          dZvr|                    dXi           }/|/                    dTi           }0|0                    dB          r|0                    dB          d[k    r<t7          |,                                          |0dB<   |.
                    d\|,            |-rt7          |-                                          r}|                    dXi           }/|/                    dTi           }0|0                    d$          s<t7          |-                                          |0d$<   |.
                    d]|-            |.s|+|,|-M|*|dT<   t          |           |s7|.r&t          d^dO                    |.                      nt          d_           ||k     r|st          d`| da|            t9          dIb          }1|1r7|s5t          dc           |1D ]#}2t          dd|2d#          d;|2de                     $| r/|1r,t          df           |1D ]}2|2                    d%          rt          dg|2d%                     |2                    dh          r$d"d7l}3|3                    di|2dj          d;          }4n+t=          di|2dj          d;                                          }4|4rPt          |2d#         |4           |dk         
                    |2d#                    t          dl|2d#                     n%|dm         
                    dn|2d#          do           t                       t9          dpb          }5|1rdq |1D             nt?                      DDfdr|5D             }5t?                      }6tA          |d6z   |d6z             D ]0}7|6!                    tD                              |7g                      1|6rC| r@|s=ds tG          |6          D             }8|8r!t          dttI          |8           du           |8D ].\  }9}:t          dv|9 dw|:                    ded                      /t                       	 t=          dx                                          	                                };n# tJ          tL          f$ r dy};Y nw xY w|;dzv r`t                       |8D ]M\  }9}:|:                    d%          r?t          di|:                    de|9                      t          dg|:d%                     n&t          di|:                    de|9                      |:                    dh          r2d"d7l}3|3                    di|:                    dj|9           d{          }4n9t=          di|:                    dj|9           d{                                          }4|4r=t          |9|4           |dk         
                    |9           t          dl|9            t                       Ont          d|           tO                      }<|<r{t	                      }|<D ]U}=|=d}         }|=d~         }>tQ          |||>           |d         
                    |           |st          d| d|>            V||d<   t          |           n(||k     r"t	                      }||d<   t          |           tS                      }?|?r*| r'|s$t          dttI          |?           d           |?D ]=}2|2                    dd          }@t          dv|2d}          dw|2de          d|@ d*           >t                       	 t=          d                                          	                                };n# tJ          tL          f$ r dy};Y nw xY w|;dzv rTt                       t	                      }	 d"dl*m+}A n# t          $ r d}AY nw xY w|?D ]
}2|2                    d~d          }>|>rd|> d*nd}Bt=          di|2dj          |B d;                                          }4|4s|>rt7          |>          }4|4r[|A d.|2d}          }CtQ          ||C|4           |d         
                    |2d}                    t          dl|2d}          d|4            n<|dm         
                    dn|2d}          d|2                    dd           d           t                       t          |           nt          d|           |S )a<  
    Migrate config to latest version, prompting for new required fields.
    
    Args:
        interactive: If True, prompt user for missing values
        quiet: If True, suppress output
        
    Returns:
        Dict with migration results: {"env_added": [...], "config_added": [...], "warnings": [...]}
    )	env_addedconfig_addedwarningsu     ✓ Repaired .env file (z corrupted entries fixed)r  r2  tool_progressr!  r"  )false0nori  r  z;display.tool_progress=off (from HERMES_TOOL_PROGRESS=false))newallzdisplay.tool_progress=z! (from HERMES_TOOL_PROGRESS_MODE)r  z#display.tool_progress=all (default)u-     ✓ Migrated tool progress to config.yaml: r  r{  HERMES_TIMEZONErZ   z	timezone=z (from HERMES_TIMEZONE)z$timezone= (empty, uses server-local)(server-local)u%     ✓ Added timezone to config.yaml: 	   r   u8     ✓ Cleared ANTHROPIC_TOKEN from .env (no longer used)   rc  r   r   r'  r(  r  r)  r  -(r  z--)urlparseendpointr   z	endpoint-rS  )zno-keyzno-key-requiredrZ   r   rW  rU  rV  r   Nu     ✓ Migrated z) custom provider(s) to providers: sectionu       → :    )	LLM_MODELOPENAI_MODELu     ✓ Cleared u>    from .env (no longer used — config.yaml is source of truth)   r`  r'  r   )r   local_command>   base.entiny.enlarge-v1large-v2large-v3small.en	medium.enlarge-v3-turbodistil-large-v2distil-large-v3distil-small.endistil-medium.endistil-large-v3.5ra  tinylargesmallturbomediumu;     ✓ Migrated legacy stt.model to provider-specific config   rA  Tz1display.interim_assistant_messages=true (default)u3     ✓ Added display.interim_assistant_messages=true   rC  rE  , c              3   *   K   | ]\  }}| d | V  dS )r   Nrl   ).0pms      rg   	<genexpr>z!migrate_config.<locals>.<genexpr>L	  s0      $R$RDAqZZAZZ$R$R$R$R$R$Rri   u>     ✓ Migrated tool_progress_overrides → display.platforms: z9display.platforms (migrated from tool_progress_overrides)   r  summary_modelsummary_providersummary_base_urlr&  model=)rZ   r   r   	provider=z	base_url=u@     ✓ Migrated compression.summary_* → auxiliary.compression: u/     ✓ Removed unused compression.summary_* keyszConfig version:     → r%  u0   
⚠️  Missing required environment variables:u      • r  z
Let's configure them now:
z  Get your key at: r  r  r  r  u     ✓ Saved r  zSkipped z - some features may not workFc                     h | ]
}|d          S )r'  rl   r  vs     rg   	<setcomp>z!migrate_config.<locals>.<setcomp>	  s    555Aai555ri   c                 R    g | ]#}|d          v|                     d          !|$S )r'  r  ra   )r  r  required_namess     rg   
<listcomp>z"migrate_config.<locals>.<listcomp>	  sC       V9N**1553D3D* 	
***ri   c                 \    g | ])}t          |          s|t          v |t          |         f*S rl   )r+  r-  )r  r'  s     rg   r  z"migrate_config.<locals>.<listcomp>	  sK     
 
 
 &&
 ,03D+D+D $T*++D+D+Dri   
  z$ new optional key(s) in this update:u       • u    — z  Configure new keys? [y/N]: n)yrP   z (Enter to skip): z1  Set later with: hermes config set <key> <value>r   r6  u     ✓ Added  = r  z! skill setting(s) not configured:skillunknownz (from skill: z#  Configure skill settings? [y/N]: )rH  zskills.configz (default: u    — skill '?z' may ask for it later),sanitize_env_filer|   rJ  rt  rC  ra   r6  r7  r+  r^   r,  save_configr[   r\   r]   save_env_valuerp  replaceurllib.parser  hostnamepopr  read_raw_config
setdefaultr*  r  r   r0  getpassinputro  rangeupdater  r  lenEOFErrorKeyboardInterruptrE  r;  rN  rI  rH  )Er  r  resultsfixescurrent_ver
latest_verr1  r2  old_enabledold_modeold_tz
tz_display	old_tokencustom_listra  migrated_countrQ  old_nameold_urlold_keyr   r  parsed	new_entryepdead_varold_valrd   raw_sttlegacy_modelr'  r`  _local_models	raw_local	local_cfgraw_providerprovider_cfgold_overridesrE  platr   migratedcomps_model
s_provider
s_base_urlmigrated_keysauxaux_compmissing_envrL  r  r   missing_optionalnew_var_namesvernew_and_unsetr'  r   answermissing_configfieldr6  missing_skill_config
skill_namerH  default_hintrM  r  sE                                                                       @rg   migrate_configr5  _  s    CCG!## 	Q 	QOuOOOPPP    344K Q**Y++'4(( 	G'))'(>??K$%@AAH V{00226JJJ+0('../lmmmm Vhnn...@@+3>>+;+;('../{HXHX/{/{/{||||+0('../TUUU 'F9 b`goF^``aaa QV##Y0"55F W&,,.. W%+\\^^z"'../b6<<>>/b/b/bcccc%'z"'../UVVV L#J/C3C
JjJJKKK Q	%&788I V0"555 VTUUU 	 	 	D	 Rjj!344k4(( 9	E[ 9	E#ZZR88Nnd33 $!#N$ )$ )$!%..  99VR00))J33Quyy7K7KQr))Ir22  nn&&,,..66sC@@HHbQQYYZ]_abbckk++dC00C ckkiinn ;;999999!)'!2!2%<*EEc3OO$ ; ; ;:.::; .(( 33>33C"G,	 1(0If% 3w.OOO+2Ii( 99W%% @16wIo.99Z(( ?-2:->Ik*&/s#!#!!&4{#

-t444F### EeNeeefff#N$7$7$9$9::N?;K;KL E E+C0CCCub0A0ACCDDDD R5 	 	H'11 y"8R000  ywxwwwxxx    R ''%$$gt$$ %	VG););"7+L{{:w77H ]]F**UB''CGGGT""" 555! ! !  =00 !(GR 8 8I%i66 :':R:R$'NN7B$?$?	-9	'*  '{{8R88!,55 99T9T#&>>(B#?#?L,8L)F5M VTUUU R ""**Y++'4(( 	G'w6648G01 'F9N#**+^___ MKLLL R ""**Y++'4(( 	G$=>>mT** 	h} 	hK44Ii.. 	+1133 < <
dy((&(IdO")D/997;IdOO4#,GK  'F9 c99$R$RM<O<O<Q<Q$R$R$RRRaW_aabbbN#**+fggg R ""zz-,,dD!! 	Qhh55G"4d;;J"4d;;JM =3w<<--// =''R88>>-<<||G,, =(+G(:(:(<(<HW%!(();'););<<< Cc*oo3355\II''R88>>-<<||J// C8<<
3K3Kv3U3U+.z??+@+@+B+BHZ(!(()AZ)A)ABBB Cc*oo3355 C''R88>>-<<||J// C+.z??+@+@+B+BHZ(!(()AZ)A)ABBB Q 3z7MQ[Qg(,}%F### Q$ Q{aeajajkxayay{{||||OPPPZ???:??@@@ 'T:::K A5 AABBB 	A 	AC?CK??3}+=??@@@@ { -... 	 	Cwwu~~ :8CJ88999wwz"" >(>S](>(>(>??43x=44455;;== bs6{E222$++CK8882S[223333
#**+`c&k+`+`+`aaaGGGG ,%@@@9DO555555#%%N   #   EEM[1_j1n55 ? ?044S"==>>>> #K #KU #K
 
}--
 
 

  	KQ]++QQQRRR+ K K
dIIIDHH]B,G,GIIJJJJGGG>??EEGGMMOO/0    %%"/  JD$xx DB488M4#@#@BBCCCADKAABBBBB488M4#@#@BBCCCxx
++ a& '0aTXXh5M5M0a0a0a b b %&W488Hd+C+C&W&W&W X X ^ ^ ` ` 5&tU333,33D9993T33444GGGG" IJJJ /00N # 	8 	8E,CI&GW---N#**3/// 86S66W66777 %/ !F	z	!	!$. !F 9:: $G $GE $GQS-..QQQRRR' 	_ 	_C)44J]SZ]]c-.@]]PZ]]]^^^^	@AAGGIIOOQQFF+, 	 	 	FFF	 \!!GGG ]]F6AAAAAAA 6 6 6&5###6+  '')R00;BJ7W7777B3x=B,BBBCCIIKK ) )LLE %8"G"G3u:"G"GKU;;;N+223u:>>>?U????@@@@J'..h3u:hh3777C;P;Phhh   EFFFNs   %/ 
<<=2J0 0
J=<J=.Q11RR!6W
W%$W%#3{ {-,{-F3AG GAG'G&AG'HAH HAH"H!AH"overridec                    |                                  }|                                D ]X\  }}||v rJt          ||         t                    r/t          |t                    rt	          ||         |          ||<   S|||<   Y|S )a  Recursively merge *override* into *base*, preserving nested defaults.

    Keys in *override* take precedence. If both values are dicts the merge
    recurses, so a user who overrides only ``tts.elevenlabs.voice_id`` will
    keep the default ``tts.elevenlabs.model_id`` intact.
    )copyr*  r6  r7  _deep_merge)ra  r6  resultr   r   s        rg   r9  r9  
  s     YY[[Fnn&&    
U6MM6#;-- 5$''  &fSk599F3KKF3KKMri   c                    t          | t                    rt          j        dd |           S t          | t                    rd |                                 D             S t          | t                    rd | D             S | S )a  Recursively expand ``${VAR}`` references in config values.

    Only string values are processed; dict keys, numbers, booleans, and
    None are left untouched.  Unresolved references (variable not in
    ``os.environ``) are kept verbatim so callers can detect them.
    z\${([^}]+)}c                     t           j                            |                     d          |                     d                    S )Nr   r   )r[   r   ra   group)r  s    rg   <lambda>z"_expand_env_vars.<locals>.<lambda>+
  s*    bjnnQWWQZZ<< ri   c                 4    i | ]\  }}|t          |          S rl   _expand_env_vars)r  kr  s      rg   
<dictcomp>z$_expand_env_vars.<locals>.<dictcomp>/
  s'    ???41a#A&&???ri   c                 ,    g | ]}t          |          S rl   r@  )r  items     rg   r  z$_expand_env_vars.<locals>.<listcomp>1
  s!    7774 &&777ri   )r6  r   resubr7  r*  rp  )objs    rg   rA  rA  !
  s     #s 
v<<
 
 	

 #t @??399;;????#t 87737777Jri   c                 j    t           fddD                       }|s S t                                           d          }t          |t                    s|rd|ini }| d<   dD ]I}                     |          }|r|                    |          s|||<                        |d           J S )u  Move stale root-level provider/base_url into model section.

    Some users (or older code) placed ``provider:`` and ``base_url:`` at the
    config root instead of inside ``model:``.  These root-level keys are only
    used as a fallback when the corresponding ``model.*`` key is empty — they
    never override an existing ``model.provider`` or ``model.base_url``.
    After migration the root-level keys are removed so they can't cause
    confusion on subsequent loads.
    c              3   B   K   | ]}                     |          V  d S Nr  )r  rB  r1  s     rg   r  z-_normalize_root_model_keys.<locals>.<genexpr>@
  s-      CCQ6::a==CCCCCCri   )r'  r(  r   r6  N)anyr7  ra   r6  r  )r1  has_rootr   r   root_vals   `    rg   _normalize_root_model_keysrO  5
  s     CCCC*BCCCCCH &\\FJJwEeT""  &+3E""w'  ::c?? 	"EIIcNN 	"!E#J

3Mri   c                     t          |           } t          |                     d          pi           }d| v rd|vr| d         |d<   d|vrt          d         d         |d<   || d<   |                     dd           | S )z;Normalize legacy root-level max_turns into agent.max_turns.r   r   N)r7  ra   rD  r  )r1  agent_configs     rg   _normalize_max_turns_configrR  S
  s    &\\F

7++1r22LfL!@!@$*;$7[!,&&$27$;K$H[!"F7O
JJ{D!!!Mri   c                      	 t                      } |                                 r?t          | d          5 }t          j        |          pi cddd           S # 1 swxY w Y   n# t
          $ r Y nw xY wi S )aO  Read ~/.hermes/config.yaml as-is, without merging defaults or migrating.

    Returns the raw YAML dict, or ``{}`` if the file doesn't exist or can't
    be parsed.  Use this for lightweight config reads where you just need a
    single value and don't want the overhead of ``load_config()``'s deep-merge
    + migration pipeline.
    r   r   N)r   rc   r   yaml	safe_loadrJ  )config_pathr   s     rg   r  r  d
  s    %'' 	/kG444 /~a((.B/ / / / / / / / / / / / / / / /   Is4   3A$ AA$ AA$ A A$ $
A10A1c                     ddl } t                       t                      }|                     t                    }|                                r	 t          |d          5 }t          j        |          pi }ddd           n# 1 swxY w Y   d|v r_t          |
                    d          pi           }|
                    d          |d         |d<   ||d<   |                    dd           t          ||          }n)# t          $ r}t          d|            Y d}~nd}~ww xY wt          t!          t#          |                              S )z.Load configuration from ~/.hermes/config.yaml.r   Nr   r   r   r   z Warning: Failed to load config: )r8  r   r   deepcopyrD  rc   r   rT  rU  r7  ra   r  r9  rJ  r|   rA  rO  rR  )r8  rV  r1  r   user_configagent_user_configes          rg   rC  rC  v
  s   KKK!##K]]>**F :	:kG444 6"nQ//526 6 6 6 6 6 6 6 6 6 6 6 6 6 6 k))$()A)A)GR$H$H!$((55=5@5M%k2'8G$T222 55FF 	: 	: 	:8Q8899999999	: 67RSY7Z7Z[[\\\s=   D !B8D BD BA6D 
D)D$$D)u  
# ── Security ──────────────────────────────────────────────────────────
# API keys, tokens, and passwords are redacted from tool output by default.
# Set to false to see full values (useful for debugging auth issues).
# tirith pre-exec scanning is enabled by default when the tirith binary
# is available. Configure via security.tirith_* keys or env vars
# (TIRITH_ENABLED, TIRITH_BIN, TIRITH_TIMEOUT, TIRITH_FAIL_OPEN).
#
# security:
#   redact_secrets: false
#   tirith_enabled: true
#   tirith_path: "tirith"
#   tirith_timeout: 5
#   tirith_fail_open: true
u	  
# ── Fallback Model ────────────────────────────────────────────────────
# Automatic provider failover when primary is unavailable.
# Uncomment and configure to enable. Triggers on rate limits (429),
# overload (529), service errors (503), or connection failures.
#
# Supported providers:
#   openrouter   (OPENROUTER_API_KEY)  — routes to any model
#   openai-codex (OAuth — hermes auth) — OpenAI Codex
#   nous         (OAuth — hermes auth) — Nous Portal
#   zai          (ZAI_API_KEY)         — Z.AI / GLM
#   kimi-coding  (KIMI_API_KEY)        — Kimi / Moonshot
#   kimi-coding-cn (KIMI_CN_API_KEY)   — Kimi / Moonshot (China)
#   minimax      (MINIMAX_API_KEY)     — MiniMax
#   minimax-cn   (MINIMAX_CN_API_KEY)  — MiniMax (China)
#
# For custom OpenAI-compatible endpoints, add base_url and api_key_env.
#
# fallback_model:
#   provider: openrouter
#   model: anthropic/claude-sonnet-4
#
# ── Smart Model Routing ────────────────────────────────────────────────
# Optional cheap-vs-strong routing for simple turns.
# Keeps the primary model for complex work, but can route short/simple
# messages to a cheaper model across providers.
#
# smart_model_routing:
#   enabled: true
#   max_simple_chars: 160
#   max_simple_words: 28
#   cheap_model:
#     provider: openrouter
#     model: google/gemini-2.5-flash
u  
# ── Security ──────────────────────────────────────────────────────────
# API keys, tokens, and passwords are redacted from tool output by default.
# Set to false to see full values (useful for debugging auth issues).
#
# security:
#   redact_secrets: false

# ── Fallback Model ────────────────────────────────────────────────────
# Automatic provider failover when primary is unavailable.
# Uncomment and configure to enable. Triggers on rate limits (429),
# overload (529), service errors (503), or connection failures.
#
# Supported providers:
#   openrouter   (OPENROUTER_API_KEY)  — routes to any model
#   openai-codex (OAuth — hermes auth) — OpenAI Codex
#   nous         (OAuth — hermes auth) — Nous Portal
#   zai          (ZAI_API_KEY)         — Z.AI / GLM
#   kimi-coding  (KIMI_API_KEY)        — Kimi / Moonshot
#   kimi-coding-cn (KIMI_CN_API_KEY)   — Kimi / Moonshot (China)
#   minimax      (MINIMAX_API_KEY)     — MiniMax
#   minimax-cn   (MINIMAX_CN_API_KEY)  — MiniMax (China)
#
# For custom OpenAI-compatible endpoints, add base_url and api_key_env.
#
# fallback_model:
#   provider: openrouter
#   model: anthropic/claude-sonnet-4
#
# ── Smart Model Routing ────────────────────────────────────────────────
# Optional cheap-vs-strong routing for simple turns.
# Keeps the primary model for complex work, but can route short/simple
# messages to a cheaper model across providers.
#
# smart_model_routing:
#   enabled: true
#   max_simple_chars: 160
#   max_simple_words: 28
#   cheap_model:
#     provider: openrouter
#     model: google/gemini-2.5-flash
c                 r   t                      rt          d           dS ddlm} t	                       t                      }t          t          |                     }g }|                    di           }|r|                    d          |	                    t                     |                    di           }|r*|                    d          r|                    d	          s|	                    t                      ||||rd
                    |          nd           t          |           dS )z,Save configuration to ~/.hermes/config.yaml.zsave configurationNr   atomic_yaml_writer  r  rv  r'  r   rZ   )extra_content)rm   r   utilsr^  r   r   rO  rR  ra   r,  _SECURITY_COMMENT_FALLBACK_COMMENTr  r   )r1  r^  rV  re   r8  secr  s          rg   r  r  
  sG   || *+++''''''!##K+,G,O,OPPJ E
..R
(
(C (#''*++3&'''	("	-	-B (bffZ(( (RVVG__ (&'''(-7bggennn4   
 ri   c                     t                      } i }|                                 rt          rdddni }t          | fi |5 }|                                }ddd           n# 1 swxY w Y   t          |          }|D ]}|                                }|rn|                    d          sYd|v rU|                    d          \  }}}	|	                                                    d          ||                                <   |S )a  Load environment variables from ~/.hermes/.env.

    Sanitizes lines before parsing so that corrupted files (e.g.
    concatenated KEY=VALUE pairs on a single line) are handled
    gracefully instead of producing mangled values such as duplicated
    bot tokens.  See #8908.
    r   r  r   errorsNr   r   z"')	r   rc   _IS_WINDOWSr   	readlines_sanitize_env_linesr]   r   r   )
env_pathenv_varsopen_kwr   	raw_linesr  r   r   r   r   s
             rg   load_envrn    sM    ~~HH C ALSw)<<<QS(&&g&& 	&!I	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& 	& $I.. 	C 	CD::<<D CDOOC00 CSD[[ $s 3 3Q(-(;(;E(B(B%Os    A!!A%(A%r  c           	         t          t                                                    t          z  }g }| D ]}|                    d          }|                                }|r|                    d          r|                    |dz              \g }|D ]c}|dz   }|                    |          }	|	dk    rA|                    |	           |                    ||	t          |          z             }	|	dk    Adt          |          dk    r|
                                 t          t          |                    }t          |          D ]k\  }
}|
dz   t          |          k     r||
dz            nt          |          }|||                                         }|r|                    |dz              l|                    |dz              |S )a  Fix corrupted .env lines before reading or writing.

    Handles two known corruption patterns:
    1. Concatenated KEY=VALUE pairs on a single line (missing newline between
       entries, e.g. ``ANTHROPIC_API_KEY=sk-...OPENAI_BASE_URL=https://...``).
    2. Stale ``KEY=***`` placeholder entries left by incomplete setup runs.

    Uses a known-keys set (OPTIONAL_ENV_VARS + _EXTRA_ENV_KEYS) so we only
    split on real Hermes env var names, avoiding false positives from values
    that happen to contain uppercase text with ``=``.
    z
r   r  r   r   r   )ro  r-  r  _EXTRA_ENV_KEYSrh  r]   r   r,  findr  sortr  r  )r  
known_keys	sanitizedr   rd   strippedsplit_positionskey_nameneedleidxr  posendr:  s                 rg   ri  ri  ,  s    &++--..@JI . .kk&!!99;;  	8..s33 	S4Z((( " 	? 	?H^F--''C((&&s+++mmFC#f++,=>> (( !##  """$S%9%9::O#O44 2 2301AO8L8L0L0Loa!e,,RUV^R_R_C(..00 2$$TD[111	2 X_----ri   c                  6   t                      } |                                 sdS t          rdddni }t          rddini }t          | fi |5 }|                                }ddd           n# 1 swxY w Y   t          |          }||k    rdS t          t          |          t          |          z
            }|dk    rVt          d t          ||          D                       }|t          t          |          t          |          z
            z  }t          j        t          | j                  dd	
          \  }}	 t          j        |dfi |5 }|                    |           |                                 t          j        |                                           ddd           n# 1 swxY w Y   t          j        ||            n5# t*          $ r( 	 t          j        |           n# t.          $ r Y nw xY w w xY wt1          |            |S )zRead, sanitize, and rewrite ~/.hermes/.env in place.

    Returns the number of lines that were fixed (concatenation splits +
    placeholder removals).  Returns 0 when no changes are needed.
    r   r   r  re  r   Nc              3   ,   K   | ]\  }}||k    d V  dS )r   Nrl   )r  abs      rg   r  z$sanitize_env_file.<locals>.<genexpr>x  s*      KK$!QAFFAFFFFKKri   .tmp.env_dirsuffixr>  w)r   rc   rg  r   rh  ri  absr  sumziptempfilemkstempr   r   r[   fdopen
writelinesflushfsyncfilenor  BaseExceptionunlinkr   r   )	rj  read_kwwrite_kwr   original_linesrt  r  fdtmp_paths	            rg   r  r  _  s    ~~H?? q<GO7i888RG(3;
G$$H	h	"	"'	"	" 'a' ' ' ' ' ' ' ' ' ' ' ' ' ' ' $N33IN""q I^!4!4455EzzKK#ni"@"@KKKKKS^^c.&9&99:::#HO(<(<VT[\\\LBYr3++(++ 	!qLL###GGIIIHQXXZZ   	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	
8X&&&&   	Ih 	 	 	D	 Lsg   A..A25A2G AF4(G 4F88G ;F8<G 
H G54H5
H?HHHr   r   c                    	 |                     d           |S # t          $ r Y nw xY wg }t          |          D ]E\  }}t          |          dk    r-|                    d| d|dt          |          dd           F|                     dd	                              d          }d
dl}t          d|  dd                    d |dd         D                       z   t          |          dk    rdndz   dz   |j
                   |S )u1  Warn and strip non-ASCII characters from credential values.

    API keys and tokens must be pure ASCII — they are sent as HTTP header
    values which httpx/httpcore encode as ASCII.  Non-ASCII characters
    (commonly introduced by copy-pasting from rich-text editors or PDFs
    that substitute lookalike Unicode glyphs for ASCII letters) cause
    ``UnicodeEncodeError: 'ascii' codec can't encode character`` at
    request time.

    Returns the sanitized (ASCII-only) value.  Prints a warning if any
    non-ASCII characters were found and removed.
    ascii   z  position r  z (U+04Xr  ignore)rf  r   Nz
  Warning: z contains non-ASCII characters that will break API requests.
  This usually happens when copy-pasting from a PDF, rich-text editor,
  or web page that substitutes lookalike Unicode glyphs for ASCII letters.

r  c              3       K   | ]	}d | V  
dS )r  Nrl   )r  r   s     rg   r  z._check_non_ascii_credential.<locals>.<genexpr>  s(      ::DKKK::::::ri   r  z
  ... and morerZ   z

  The non-ASCII characters have been stripped automatically.
  If authentication fails, re-copy the key from the provider's dashboard.
rz   )encodeUnicodeEncodeErrorr  ordr,  decoder}   r|   r  r  r~   )r   r   	bad_charsr  chrt  r}   s          rg   _check_non_ascii_credentialr    sl   W    I5!! J J2r77S==H1HHHH#b''HHHHIIIWX66==gFFIJJJ		 	 	 	 ))::IbqbM:::
:
:		;
 "%Y!!3!3	=W	W Z
 
 
 
 s    
&&c                    t                      rt          d|             dS t                              |           st	          d|           |                    dd                              dd          }t          | |          }t                       t                      }t          rddd	ni }t          rd
dini }g }|
                                rHt          |fi |5 }|                                }ddd           n# 1 swxY w Y   t          |          }d}t          |          D ]>\  }}	|	                                                    |  d          r|  d| d||<   d} n?|sH|r+|d                             d          s|dxx         dz  cc<   |                    |  d| d           t'          j        t+          |j                  dd          \  }
}d}|
                                r=	 t/          j        |                                j                  }n# t4          $ r Y nw xY w	 t7          j        |
dfi |5 }|                    |           |                                 t7          j        |                                            ddd           n# 1 swxY w Y   t7          j        ||           |'	 t7          j!        ||           n# t4          $ r Y nw xY wn5# tD          $ r( 	 t7          j#        |           n# t4          $ r Y nw xY w w xY wtI          |           |t6          j%        | <   dS )z)Save or update a value in ~/.hermes/.env.zset N#Invalid environment variable name: r  rZ   r   r  re  r   Fr   Tr4  r  r  r  r  )&rm   r   _ENV_VAR_NAME_REmatchr   r  r  r   r   rg  rc   r   rh  ri  r  r]   r   endswithr,  r  r  r   r   statS_IMODEst_moder   r[   r  r  r  r  r  r   r  r  r   r   )r   r   rj  r  r  r  r   foundr  r   r  r  original_modes                rg   r  r    s(   || lSll###!!#&& HFsFFGGGMM$##++D"55E'U33E~~H =HO7i888RG(3;
G$$HE +(&&g&& 	"!KKMME	" 	" 	" 	" 	" 	" 	" 	" 	" 	" 	" 	" 	" 	" 	" $E** EU##  4::<<""c999-- 	)))))E!HEE	
  ) 	r++D11 	"IIIIII''e'''(((#HO(<(<VT[\\\LBM 	 L)@AAMM 	 	 	D	Yr3++(++ 	!qLLGGIIIHQXXZZ   	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	
8X&&&$=1111      	Ih 	 	 	D	 BJsOOOs   #DDD;+H' '
H43H48K1 AJ'K1 'J++K1 .J+/K1 
K  K1  
K-*K1 ,K--K1 1
L#<LL#
LL#LL#c                 f    t                      rt          d             dS t                                         st	          d           t                      }|                                s"t          j        	                     d           dS t          rdddni }t          rddini }t          |fi |5 }|                                }ddd           n# 1 swxY w Y   t          |          } fd	|D             }t          |          t          |          k     }|rit          j        t#          |j                  d
d          \  }}	d}
	 t'          j        |                                j                  }
n# t,          $ r Y nw xY w	 t          j        |dfi |5 }|                    |           |                                 t          j        |                                           ddd           n# 1 swxY w Y   t          j        |	|           |
'	 t          j        ||
           n# t,          $ r Y nw xY wn5# t<          $ r( 	 t          j        |	           n# t,          $ r Y nw xY w w xY wtA          |           t          j        	                     d           |S )zzRemove a key from ~/.hermes/.env and os.environ.

    Returns True if the key was found and removed, False otherwise.
    zremove Fr  Nr   r  re  r   c                 h    g | ].}|                                                      d           ,|/S )r   )r]   r   )r  r   r   s     rg   r  z$remove_env_value.<locals>.<listcomp>  s:    RRR$tzz||/F/F#yyy/Q/QRRRRri   r  r  r  r  )!rm   r   r  r  r   r   rc   r[   r   r  rg  r   rh  ri  r  r  r  r   r   r  r  r  r   r  r  r  r  r  r  r   r  r  r   )r   rj  r  r  r   r  	new_linesr  r  r  r  s   `          rg   remove_env_valuer    s\   
 || ooo&&&u!!#&& HFsFFGGG~~H?? 

sD!!!u<GO7i888RG(3;
G$$H	h	"	"'	"	" a              &&ERRRR%RRRI	NNSZZ'E 'C,@,@X_```H	 L)@AAMM 	 	 	D		2s//h// %1Y'''			$$$% % % % % % % % % % % % % % % Jx***(HX}5555   D 	 	 		(####   	 	XJNN3Ls   <CC!$C!+F 
FFI 'AH7I HI 
HI &H< ;I <
I	I I		I 
I?I-,I?-
I:7I?9I::I?c                 H    |pt           } |d|             |dd           dS )zBPersist an Anthropic OAuth/setup token and clear the API-key slot.r   r$   rZ   Nr  r   save_fnwriters      rg   save_anthropic_oauth_tokenr  3  s8    &F
Fe$$$
F#####ri   c                 H    | pt           } |dd            |dd           dS )zHUse Claude Code's own credential files instead of persisting env tokens.r   rZ   r$   Nr  )r  r  s     rg   %use_anthropic_claude_code_credentialsr  :  s8    &F
Fb!!!
F#####ri   c                 H    |pt           } |d|             |dd           dS )zBPersist an Anthropic API key and clear the OAuth/setup-token slot.r$   r   rZ   Nr  r  s      rg   save_anthropic_api_keyr  A  s8    &F
F&&&
Fb!!!!!ri   c                 .    t          | |           d| ddS )NTF)success	stored_as	validatedr  )r   r   s     rg   save_env_value_securer  H  s*    3  ri   c                  v   t                      } t          t                                                    t          z  }d}|                                 D ]<\  }}t          j                            |          |k    r|t          j        |<   |dz  }=|D ]&}|| vr |t          j        v rt          j        |= |dz  }'|S )u&  Re-read ~/.hermes/.env into os.environ. Returns count of vars updated.

    Adds/updates vars that changed and removes vars that were deleted from
    the .env file (but only vars known to Hermes — OPTIONAL_ENV_VARS and
    _EXTRA_ENV_KEYS — to avoid clobbering unrelated environment).
    r   r   )	rn  ro  r-  r  rp  r*  r[   r   ra   )rk  rs  countr   r   s        rg   
reload_envr  R  s     zzH&++--..@JEnn&&  
U:>>#%''#BJsOQJE  h3"*#4#4
3QJELri   c                     | t           j        v rt           j        |          S t                      }|                    |           S )z/Get a value from ~/.hermes/.env or environment.)r[   r   rn  ra   )r   rk  s     rg   r+  r+  h  s:     bjz# zzH<<ri   c                     | st          dt          j                  S t          |           dk     rdS | dd         dz   | dd         z   S )zRedact an API key for display.	(not set)r  z***Nr  ...)rL   rK   DIMr  )r   s    rg   
redact_keyr  w  sO     .[&*---
3xx"}}urr7U?SX%%ri   c                  d   t                      } t                       t          t          dt          j                             t          t          dt          j                             t          t          dt          j                             t                       t          t          dt          j        t          j                             t          dt                                  t          dt                                  t          dt                                  t                       t          t          dt          j        t          j                             g d	}|D ]7\  }}t          |          }t          d
|ddt          |                      8ddlm}  |            }t          d
dddt          |                      t                       t          t          dt          j        t          j                             t          d|                     dd                      t          d|                     di                               dt          d         d                               t                       t          t          dt          j        t          j                             |                     di           }t          d|                    dd                      t          d|                    dd          rdnd             t          d!|                    d"d          rdnd             t                       t          t          d#t          j        t          j                             |                     d$i           }t          d%|                    d&d'                      t          d(|                    d)d*                      t          d+|                    d,d-           d.           |                    d&          d/k    r(t          d0|                    d1d2                      nj|                    d&          d3k    r(t          d4|                    d5d6                      n)|                    d&          d7k    rLt          d8|                    d9d2                      t          d:          }	t          d;|	rd<nd=            n|                    d&          d>k    rLt          d?|                    d@d2                      t          dA          }
t          dB|
rd<nd=            n_|                    d&          dCk    rFt          dD          }t          dE          }t          dF|pd=            t          dG|pd=            t                       t          t          dHt          j        t          j                             |                     dIdJ          }|rt          dK|            n*t          dKt          dLt          j                              t                       t          t          dMt          j        t          j                             |                     dNi           }|                    dOdP          }t          dQ|rdRndS            |rt          dT|                    dUdV          dWz  dXdY           t          dZ|                    d[d\          dWz  dXd]           t          d^|                    d_d`           da           |                     dbi                               dNi           }|                    ddJ          pdc}t          d|            |                    ddde          }|r|dek    rt          df|            |                     dbi           }|                    dgi           |                    dhi           di}t!          dj |                                D                       }|rt                       t          t          dkt          j        t          j                             |                                D ]\  }}|                    ddde          }|                    ddJ          }|dek    s|rIdl| g}|r|                    dm|            t          d
|dnd
do                    |                      t                       t          t          dpt          j        t          j                             t          dq          }t          dr          }t          ds|rd<nt          dtt          j                              t          du|rd<nt          dtt          j                              	 ddvlm}m}  |            }|r ||          } t                       t          t          dwt          j        t          j                             |D ]}!|!dx         }"|                     |"dJ          }|!                    dydJ          }#|rt1          |          nt          d=t          j                  }$t          d
|"dzd|$ d
t          d{|# d|t          j                              n# t2          $ r Y nw xY wt                       t          t          d}t          j                             t          t          d~t          j                             t          t          dt          j                             t          t          dt          j                             t                       dS )zDisplay current configuration.u   ┌─────────────────────────────────────────────────────────┐u@   │              ⚕ Hermes Configuration                    │u   └─────────────────────────────────────────────────────────┘u	   ◆ Pathsz  Config:       z  Secrets:      z  Install:      u   ◆ API Keys)	)r  
OpenRouter)r  zOpenAI (STT/TTS))r  Exa)r  Parallel)r  	Firecrawl)r  Tavily)r  Browserbase)r  zBrowser Use)r  FALr  z<14r  r   )get_anthropic_key	Anthropicu	   ◆ Modelz  Model:        r   znot setz  Max turns:    r   r   u   ◆ Displayr2  z  Personality:  r8  r3  z  Reasoning:    r<  Fonri  z  Bell:         r;  u   ◆ Terminalr   z  Backend:      r   r   z  Working dir:  r   r   z  Timeout:      r   r   sr   z  Docker image: r   r   singularityz  Image:        r   r   modalz  Modal image:  r   MODAL_TOKEN_IDz  Modal token:  
configuredr  daytonaz  Daytona image: r   DAYTONA_API_KEYz  API key:      sshTERMINAL_SSH_HOSTTERMINAL_SSH_USERz  SSH host:     z  SSH user:     u   ◆ Timezoner{  rZ   z  Timezone:     r  u   ◆ Context Compressionr  r
  Tz  Enabled:      rP   r  z  Threshold:    r  r  d   z.0f%z  Target ratio: r  r  z% of threshold preservedz  Protect last: r  r  z	 messagesr&  z(auto)r'  r   z  Provider:     r+  r,  )VisionzWeb extractc              3   v   K   | ]4}|                     d d          dk    p|                     dd          V  5dS )r'  r   r   rZ   Nr  )r  ts     rg   r  zshow_config.<locals>.<genexpr>  s\         	
j&!!V+AquuWb/A/A     ri   u    ◆ Auxiliary Models (overrides)r  r  12sr  u   ◆ Messaging Platformsr   r  z  Telegram:     znot configuredz  Discord:      )rG  resolve_skill_config_valuesu   ◆ Skill Settingsr   r  z<20s[]u   ────────────────────────────────────────────────────────────z+  hermes config edit     # Edit config filez!  hermes config set <key> <value>z+  hermes setup           # Run setup wizardN)rC  r|   rL   rK   CYANBOLDr   r   r   r+  r  hermes_cli.authr  ra   rD  r  rL  valuesr*  r,  r  rI  rG  r  r   rJ  )%r1  r  env_keyr'  r   r  anthropic_valuer2  r   modal_tokendaytona_keyssh_hostssh_usertzr  r
  	_aux_comp_smcomp_providerr&  	aux_taskshas_overrideslabeltask_cfgprovmdlr8  telegram_tokendiscord_tokenrG  r  
skill_varsresolvedrL  r   r3  display_vals%                                        rg   show_configr    s~   ]]F	GGG	%  D  FL  FQ  R  R  S  S  S	%RTZT_
`
`aaa	%  D  FL  FQ  R  R  S  S  S 
GGG	%V[&+
6
6777	
0_..
0
0111	
-\^^
-
-...	
1-//
1
1222 
GGG	%V[
9
9:::
 
 
D  3 3g&&14111j//112222111111''))O	
>{
>
>
>O!<!<
>
>??? 
GGG	%V[&+
6
6777	
=VZZ;;
=
=>>>	
mVZZ4488nU\F]^iFjkk
m
mnnn 
GGG	%v{FK
8
8999jjB''G	
CW[[AA
C
CDDD	
VW[[1A5%I%ITTTu
V
VWWW	
XW[[1CU%K%KVTTQV
X
XYYY 
GGG	%V[
9
9:::zz*b))H	
?X\\)W==
?
?@@@	
7X\\%55
7
7888	
;X\\)R88
;
;
;<<<||I(**mn>j!k!kmmnnnn	i	 	 M	1	1{.ACx!y!y{{||||	i	 	 G	+	+lm=i!j!jllmmm#$455O!M+OOPPPP	i	 	 I	-	-o(,,@l"m"mooppp#$566O!M+OOPPPP	i	 	 E	)	) !455 !455:!8[::;;;:!8[::;;; 
GGG	%V[
9
9:::	J	#	#B	 H%%%&&&&F'7!D!DFFGGG 
GGG	%)6;
D
DEEE**]B//Kooi..G	
9g7UU4
9
9::: 	6Pd!C!Cc!IPPPPQQQj!F!F!LjjjjkkkQ1A2!F!FQQQRRRJJ{B//33M2FF	mmGR((4H&&&'''!j&99 	6]f444]44555 

;++I }}Xr22 }}]B77 I   !!##    M  
<e6V[QQRRR(00 	< 	<OE8<<
F33D,,w++Cv~~~+T++, 1LL#000:5:::		%(8(8::;;; 
GGG	%)6;
D
DEEE"#788N!"566M	
f^d\\GWY_YcAdAd
f
fggg	
e]c\\FVX^Xb@c@c
e
efffaaaaaaaa3355
 		]22:>>HGGG%,fk6;GGHHH! ] ]%j S"-- WWWb11
,1Uc%jjju[&*7U7U[3[[[k[[U;Lz;L;L;Lfj5Y5Y[[\\\\    
GGG	%
FJ
'
'(((	%=vz
J
JKKK	%3VZ
@
@AAA	%=vz
J
JKKK	GGGGGs   /C8k( (
k54k5c                  (   t                      rt          d           dS t                      } |                                 s&t	          t
                     t          d|             t          j        d          pt          j        d          }|s"dD ]}ddl	}|
                    |          r|} n |s#t          d           t          d	|             dS t          d
|  d| d           t          j        |t          |           g           dS )z"Open config file in user's editor.zedit configurationNzCreated EDITORVISUAL)nanovimvicodenotepadr   z#No editor found. Config file is at:r  zOpening  in r  )rm   r   r   rc   r  rD  r|   r[   r\   shutilwhich
subprocessrunr   )rV  editorcmdr  s       rg   edit_configr  "  sE   || *+++!##K  (N###&&&''' Yx  7BIh$7$7F ; 	 	CMMM||C     3444 ;  !!!	
1[
1
1f
1
1
1222NFC,,-.....ri   c                 ^   t                      rt          d           dS g d}|                                 |v sN|                                                     d          s'|                                                     d          rEt          |                                 |           t          d|  dt                                  dS t                      }i }|	                                rS	 t          |d	          5 }t          j        |          pi }ddd           n# 1 swxY w Y   n# t          $ r i }Y nw xY w|                     d
          }|}|dd         D ];}||vs(t          |                    |          t"                    si ||<   ||         }<|                                dv rd}nu|                                dv rd}n\|                                rt)          |          }n8|                    d
dd                                          rt-          |          }|||d         <   t/                       ddlm}	  |	||d           dddddddddddd d!d"d#d$}
| |
v r#t          |
|          t5          |                     t          d|  d%| d|            dS )&zSet a configuration value.zset configuration valuesN)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  )_API_KEY_TOKENTERMINAL_SSHu   ✓ Set r  r   r   r   r4  )rN   rP   r  T)r  r  ri  FrZ   r   r   r]  )	sort_keysr   r  TERMINAL_DOCKER_IMAGETERMINAL_SINGULARITY_IMAGETERMINAL_MODAL_IMAGETERMINAL_DAYTONA_IMAGE&TERMINAL_DOCKER_MOUNT_CWD_TO_WORKSPACEr  TERMINAL_TIMEOUTTERMINAL_SANDBOX_DIRTERMINAL_PERSISTENT_SHELLTERMINAL_CONTAINER_CPUTERMINAL_CONTAINER_MEMORYTERMINAL_CONTAINER_DISKTERMINAL_CONTAINER_PERSISTENT)zterminal.backendzterminal.modal_modezterminal.docker_imagezterminal.singularity_imagezterminal.modal_imagezterminal.daytona_imagez&terminal.docker_mount_cwd_to_workspacezterminal.cwdzterminal.timeoutzterminal.sandbox_dirzterminal.persistent_shellzterminal.container_cpuzterminal.container_memoryzterminal.container_diskzterminal.container_persistentr  )rm   r   upperr  r   r  r|   r   r   rc   r   rT  rU  rJ  r5  r6  ra   r7  r^   isdigitr   r  r[  r   r`  r^  r   )r   r   api_keysrV  rY  r   r8  r9  r:  r^  _config_to_env_syncs              rg   set_config_valuer)  B  sR   || 0111  H yy{{h#))++"6"67M"N"NRUR[R[R]R]RhRhiwRxRxsyy{{E***222,..22333
 "##KK 	kG444 6"nQ//526 6 6 6 6 6 6 6 6 6 6 6 6 6 6 	 	 	KKK	 IIcNNEGcrc
    wjT1B1BD&I&IGDM$- {{}}---	0	0	0	 E

	sB	"	"	*	*	,	, eGE"I ''''''k;%@@@@
 +4!8&B 6":2Z&. 6%@":%@#<)H " !!!*3/U<<<	
5S
5
5U
5
5
5
566666s6   2D2 D&D2 &D**D2 -D*.D2 2E Ec           	         t          | dd          }||dk    rt                       dS |dk    rt                       dS |dk    rt          | dd          }t          | dd          }|r|mt          d           t                       t          d	           t          d
           t          d           t          d           t	          j        d           t          ||           dS |dk    rt          t                                 dS |dk    rt          t                                 dS |dk    rt                       t          t          dt          j        t          j                             t                       t          d          }t                      }t                      \  }}|s?|s=||k    r7t          t          dt          j                             t                       dS ||k     rt          d| d|            |r t          dt#          |           d           d |D             }d |D             }	|r=t          dt#          |           d           |D ]}
t          d|
d                     |	rzt          dt#          |	           d            |	D ]W}
|
                    d!g           }|r!d"d#                    |dd$                    d%nd&}t          d|
d          |            Xt                       t)          d'd(          }t                       |d)         s|d*         r't          t          d+t          j                             |d,         rCt                       |d,         D ],}t          t          d-| t          j                             -t                       dS |d.k    rt                       t          t          d/t          j        t          j                             t                       t                      \  }}||k    rt          d| d0           n.t          t          d| d| d1t          j                             t                       t          t          d2t          j                             t,          D ]O}t/          |          rt          d3|            $t          t          d4| d5t          j                             Pt                       t          t          d6t          j                             t2                                          D ]\  }}t/          |          rt          d3|            '|                    d!g           }|r dd#                    |dd$                    nd&}t          t          d7| | t          j                             t                      }|rUt                       t          t          d8t#          |           d9t          j                             t          d:           t                       dS t          d;|            t                       t          d<           t          d=           t          d>           t          d?           t          d@           t          dA           t          dB           t          dC           t	          j        d           dS )DzHandle config subcommands.config_commandNshoweditro  r   r   z&Usage: hermes config set <key> <value>z	Examples:z3  hermes config set model anthropic/claude-sonnet-4z+  hermes config set terminal.backend dockerz0  hermes config set OPENROUTER_API_KEY sk-or-...r   r   zenv-pathmigrateu*   🔄 Checking configuration for updates...Fr  u    ✓ Configuration is up to date!z  Config version: r  r  z1 new config option(s) will be added with defaultsc                 <    g | ]}|                     d           |S )r(  r  r  s     rg   r  z"config_command.<locals>.<listcomp>  s)    KKK!aeeM6J6JKAKKKri   c                 f    g | ].}|                     d           |                     d          ,|/S )r(  r  r  r  s     rg   r  z"config_command.<locals>.<listcomp>  sN     
 
 
55''
01j0A0A

 
 
ri   u   
  ⚠️  z required API key(s) missing:u	        • r'  u   
  ℹ️  z$ optional API key(s) not configured:r  z (enables: r     r  rZ   T)r  r  r  r  u   ✓ Configuration updated!r  u
     ⚠️  checku   📋 Configuration Statusu    ✓z (update available)z  Required:u       ✓ u       ✗ z
 (missing)z  Optional:u       ○ r  z new config option(s) availablez+    Run 'hermes config migrate' to add themzUnknown config command: zAvailable commands:z4  hermes config           Show current configurationz/  hermes config edit      Open config in editorz6  hermes config set <key> <value>   Set a config valuez;  hermes config check     Check for missing/outdated configz8  hermes config migrate   Update config with new optionsz/  hermes config path      Show config file pathz-  hermes config env-path  Show .env file path)getattrr  r  r|   r}   exitr)  r   r   rL   rK   r  r  r0  rE  rt  GREENr  ra   r  r5  YELLOWr)  r+  REDr-  r*  r  )argssubcmdr   r   r*  r0  r  r	  required_missingoptional_missingrL  r  	tools_strr  r  r/  r   s                    rg   r+  r+    sA   T+T22F~6))	6			5dE4((gt,, 	em:;;;GGG+GHHH?@@@DEEEHQKKKe$$$$$	6		o     	:		lnn	9		e@&+v{[[\\\ +???244"6"8"8Z 	> 	kZ6O6O%:FLIIJJJGGGF ##E{EEEEFFF 	a_^,,___```KK{KKK
 
"
 
 

  	1U%5!6!6UUUVVV' 1 1/#f+//0000 	<\%5!6!6\\\]]]' < <,,EJRA$))E"1"I*>*>AAAAPR	:#f+:y::;;;; !T???; 	E7>#: 	E%4flCCDDD: 	DGGG":. D De222FMBBCCCC	7		e/fkJJKKK"6"8"8Z*$$8{8889999%^[^^z^^^`f`mnnoooeM6;//000) 	J 	JHX&& J+++,,,,e;x;;;VZHHIIIIeM6;//000/5577 	K 	KNHdX&& K+++,,,,"-->CK:DIIeBQBi$8$8:::	e<x<<<fjIIJJJJ244 	AGGG%QS00QQQSYS`aabbb?@@@ 	111222#$$$DEEE?@@@FGGGKLLLHIII?@@@=>>>ri   )rs   )rx   )FrK  )TF)gr  r[   platformrF  r  r  r}   r  dataclassesr   pathlibr   typingr   r   r   r   r	   systemrg  compiler  	frozensetrp  rT  hermes_cli.colorsrK   rL   hermes_cli.default_soulrM   r_   r`   r   rh   boolrm   rp   rr   rw   r   r7  r   r   rb   r   r   r   r   r   r   r   r   r   rD  r  r   r  r)  r-  r0  r;  rE  rN  r`  rd  rq  rt  r  _VALID_CUSTOM_PROVIDER_FIELDSr  rx  r  r  r  r5  r9  rA  rO  rR  r  rC  ra  rb  _COMMENTED_SECTIONSr  rn  rp  ri  r  r  r  r  r  r  r  r  r  r+  r  r  r  r)  r+  rl   ri   rg   <module>rI     s"     
			  				      



  ! ! ! ! ! !       3 3 3 3 3 3 3 3 3 3 3 3 3 3 ho9,2:9::  )     4  + + + + + + + + 3 3 3 3 3 3 , 	  HSM    ,D , , , ,HSM    ;C ; ; ; ;
 3 s    8; ;# ; ; ; ;*$ * * * *d - , , , , ,- - - - -&d & & & &2$ 2 2 2 2  4t    2  $$ 4    & & &0"d " " " ",xRxx "x !"	x
 x 
  
 "$ !' $' $'5 xF  %7%f% 	s% 	3	% 	2% 	D% 	b%  	b!%" 	R#%$ 	C%%& 	E'%* 	+%, 	D-%. 	%/%0 	1%8 	"9%> 	(?%H 	DI% %GxT ! #
 "5	
 Uxt  uxD 7ExH 	 Ix\ ! $
 
 %'!#&-
 
 ]x@ 	 AxZ  "
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
g: :[xR  &!&*!&#%  Sxx yxB eCx xL 
'

 /0
 

 '
 
  	
 
 ->
 

 3	
 
3 MxN 

 

 [
 *
 Oxl !$  mx|  }xT	 ,U	x^	  $!
 
 
_	x~	 	 		xZ
 R[
xd
 e
xr
 bs
xz
 {
x@ "$ AxT  Uxd 2exn 2oxx 2yxH  Ix xT    
 
   	   	e	 ox x xB YXX "67E E E		- - T#tCy.)     a6B a W&+"$78 a& U,7 'a6 L"7 7aF ;= GaV $& Waf .: gav [& waF =  GaV =  Waf 5A gav 0 . waF :; GaV 6(. Waf )$' gav 3< waF 8#( Ga aV 2> Waf 9+* gav :F waF D$7 GaT @% Uab W%> cap `& qa@ [B AaP _B   Qa` "T9B$ $aap YO! !qa@ V() AaP 7C Qa` U') aap 6B qa@ o&7 AaN K9 Oa a a^ W(, _an X= oa~ g'0 aL	 _= M	a\	 XZ ]	al	 _ m	a@
 J . A
aP
 N$%. Q
a`
 F%'. a
ap
 O= q
a@ jM AaP  o. Qa`  j+ aap  O+   qa@ T",;;; AaP n')$o6 Qa` \*)$o6 aa a ap n')$o6 qa@ Y1$o6 AaN r&:$o6 Oa^ 9 "# _an 7"@MMM oa~ I!+"$56 aN X:5'6 Oa^ M&' _al P#, maz \ 3 {aL G"'"# Ma\ V@ ]aj ;&' kax l?) yaF z1	 GaR @%< Sa` P> aa a an  a6 oa| \ /+ }aN G /+ Oa` L)/ aan F( oa| SA! !}aJ !z0# #KaX (c?* *Yaf P)6 gat L' uaB B1 CaP e= Qa^ s: _an !\<# #oa~ Y= aN g= Oa^  ]' _a a an h*) oa| c/ }aJ  c@" "KaX "F/$ $Yab G!	 can C$	 oaz K$ {aD V* EaN M& OaX U# Yab =( cal T# mav f0   waF  L2 GaV  HE Waf A# gav  g# wa aH u)  yQ  [.  l0  K   u"  S!  X"  WB  WB" " `.% % i+' 'ua a a N  d38n9M    , #    4S#X#7    8"tDcN'; " " " "P @ @ @@ @ d38n	@ @ @ @Fs tDcN?S     (,/ /T#s(^$/	$sCx./ / / /d	eCHo 	 	 	 	"   ! ! !   WVV         w whtCH~&> w$}J] w w w wt0 0(4S>": 0d 0 0 0 0.+4 +4$sCx.)A +4T +4 +4 +4 +4\k k kD kT#s(^ k k k k\d d t    (  (tCH~ $sCx.    <S#X 4S>    "c3h    $]T#s(^ ] ] ] ]6  " J) XS#X    :$sCx.    :0t 0 0 0 0 0f*3 * * * *Z&S & & & & & &RC CC C C C CL5# 5$ 5 5 5 5p$ $c $ $ $ $$ $ $ $" "# " " " "s 3 4S>    C    ,s x}    &C &C & & & &_ _ _D/ / /@T7# T7c T7 T7 T7 T7vF F F F Fri   