
    o;i                    &   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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        e          Z ej                    dk    Z ej        d          Zi Zeeef         ed<   i Zeeeeeeeef         f         f         ed	<   i Z eeeeeeeef         f         f         ed
<    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/d1dedefdZ0d2defdZ1dee2         fdZ3ddl4m5Z5 ddl6m7Z7 defdZ8defd Z9defd!Z:d" Z;de,fd#Z<d$ Z=d%eddfd&Z>d' Z?d%efd(Z@i d)d*d+i d,g d-i d.d/gd0d1d2d3d4d*d5d6d3d7d5g d8d9i d:d;d<d5d=d>d?d3d@g dAg dBdCdDdEdFg dGi dHdIdJdEdKdEdLdMdNdOdPdQdRdSdCg dTdTdCdUdVdWdXdTdTdCd*dYdZd[dTid\	d]dCd^dTd_dCd`dadbdcdddedfdfdgdhdCdTdid4didjdkdldkdjdmdndCdodpdqdrdsdtdudvidwdCdZdxdyd*dCg d7dzd*d*d{d|d}d~dd5d*d*d*dWi dXdd5d*d*d*di dd5d*d*d*dWi dd5d*d*d*dXi d4dd5d*d*d*dXi dd5d*d*d*dXi dd5d*d*d*dXi dd5d*d*d*dXi dd5d*d*d*di dd	i di ddTddddddddTddTddTddTddddCddTddddddididddCddTdi ddi dTg ddddddidddTiddddiddddddddddddddd*d*dddddidddCd;dd*dd)did)diddddWdTdCdddƜddddd˜dddiddCdCddd*dҜdd*d*d*d*dCd^dd*d4dOdCdTdԜdd*dddqidg dCdTddTdڜddCddidXd1dCdkdݜdޜdi dd*ddCd*d*dCdCi d*di di ddTi dddi iddi idddddCddg di di ddTdi ddTdTdCddkdCdTg g ddddCddddCdddddidddkd4d ddCdd`i ddddTidTd1dCd`ddi idTdkdd	d
ZAg dddgg ddgdgdZBeeee         f         ed<   i ZCi dddddTddCddddddCddgddCddd d!d"dCddCdd#d$d%d"dCddCdd&d'd(ddTddCdd)d*d*d+dCddCdd,d-d.ddTddCdd/d0d1d2dCddCdd3d4d5ddTddCdd6d7d8ddCddCdd9d:d;ddTddCdd<d=d>d?dCddCdd@dAdBd?dCddCddCdAdBd?dCddCddDdEdFddTddCddGdHdIdJdCddCddKdLdMddTddCdi dNdOdPdJdCddCddQdRdRdSdCddCddTdUdVddTddCddWdXdXdYdCddCddZd[d\ddTddCdd]d^d^d_dCddCdd`dadbddTddCddcdddedfdCddCddgdhdiddTddCddjdkdldmdCddCddndodpddTddCddqdrdsdtdCddudvdwdxd*dTddudydzd{d|dCddud}d~dd*dTddCddddddTddCddddddTddCdi dddddCddCddddddTddCddddddCddCddddddTddCddddddCddCddddddTddCddddddCddudddddTddCddddddCddCddddddTddCddddddCddudddddTddCddddddTddCddddddTddCddddddCddudddddTddCdddddddgdCdÐdĜi dŐdƐdǐdȐddgdCdÐdĜdɐdʐdːd̐ddgdCdÐdĜd͐dΐdddTddCddАdѐdddTddCddӐdԐdddTddCdd֐dאdddTddCddِdڐdddCddCdddܐdݐdg dߢdCdÐdĜddddddgdCdÐdĜddddddgdTdÐdĜddddddgdCdÐdĜdddddgdTdÐdddddddgdTdÐdĜdddddgdCdÐdĜddddg ddCdÐdĜdd ddddgdCdÐdĜdddddd	gdCdÐdĜi dd
dddCdÐdudddddCdÐdudddddCdÐdudddddCddCddddddCddCdddd d!dCddCdd"d#d$d%dCddCdd&d'd(d)d*gdCdÐdĜd+d,d-dÐd.d/d0d1d2dTdÐdud3d4d5d2dCdÐdud6d7d8ddTddCdd9d:d;d<dCd=dud>d?d@dAdTd=dudBdCdDdTd=dEdFdGdHdIdCd=dudJdKdLddTd=dui dMdNdOddTd=dudPdQdRdSdCd=dudTdUdVdSdCd=dudWdXdYdZdTd=dud[d\d]ddCd=dud^d_d`ddTd=dudadbdcddTd=dudddedfddTd=dudgdhdidjdTd=dudkdldmddCd=dudndodpddTd=dudqdrdsddTd=dudtdudvddTd=dCddwdxdyddTd=dCddzd{d|ddTd=dCdd}d~dddTd=dCddddddTd=dCdi dddddCd=dCddddddTd=dudddddCd=dudddddTd=dudddd=d.ddddd=dddddCd=dE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dddTd=dudddddTd=dudddddTd=dudddddCd=dCdi dddddCd=dCddddddTd=dCddddddTd=dCdddÐdddCd=dCddŐdƐdddTd=dCddȐdɐdddTd=dCddːd̐dddTd=dCddΐdϐdddTd=dCddѐdҐdddCd=dCddԐdՐdddTd=dudאdؐdddTd=dudڐdېdddCd=dudݐdސdddCddudddddTddudddddTddudddddTddudddddTddudddddTdduiZDd3de,deeeef                  fdZEdefdZFdeeeef                  fdZGdeeeef                  fdZHd*ddededeeeef                  fdZIdedeeeef                  fdZJ	 d4deeeef                  deeeef                  fdZK	 	 d5d)ededeeeeef                           deeeef                  dee         f
dZLdeeef         fd ZMh dZNh dZOh dZPe G d d                      ZQd4deeeef                  ded         fdZRd4deeeef                  ddfdZSd4deeeef                  ddfdZTd6d	e,d
e,deeef         fdZUde2de2de2fdZVd ZWd ZXd4dZYdeeef         deeef         fdZZdeeef         deeef         fdZ[dddeeeef                  dededefdZ\deeef         fdZ]deeef         fdZ^dZ_dZ`dZadeeef         fdZbdeeef         fdZcdeddedfdZedefd Zfd!ed"edefd#Zgd!ed"efd$Zhd!ede,fd%Zid4d"efd&Zjd4d'Zkd4d"efd(Zld!ed"edeeef         fd)Zmdefd*Znd!edee         fd+Zod!edefd,Zpd- Zqd. Zrd!ed"efd/Zsd0 ZtdS (7  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_]*$_LAST_EXPANDED_CONFIG_BY_PATH_LOAD_CONFIG_CACHE_RAW_CONFIG_CACHE>e   IRC_PORT	QQ_APP_ID
IRC_SERVERIRC_CHANNELIRC_USE_TLSIRC_NICKNAME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SMS_HOME_CHANNELTERMINAL_SSH_KEYWEIXIN_DM_POLICYWHATSAPP_ENABLEDANTHROPIC_API_KEYFEISHU_APP_SECRETLANGFUSE_BASE_URLMATRIX_ENCRYPTIONTERMINAL_SSH_PORTWEIXIN_ACCOUNT_IDDINGTALK_CLIENT_IDFEISHU_ENCRYPT_KEYMATRIX_AUTO_THREADQQBOT_HOME_CHANNELQQ_ALLOW_ALL_USERSSLACK_HOME_CHANNELWECOM_HOME_CHANNELFEISHU_HOME_CHANNELHERMES_LANGFUSE_ENVIRC_SERVER_PASSWORDLANGFUSE_PUBLIC_KEYLANGFUSE_SECRET_KEYMATRIX_RECOVERY_KEYQQ_MARKDOWN_SUPPORTSIGNAL_HOME_CHANNEL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YUANBAO_HOME_CHANNELDINGTALK_HOME_CHANNELHERMES_LANGFUSE_DEBUGIRC_NICKSERV_PASSWORDMATRIX_DM_AUTO_THREADMATTERMOST_REPLY_MODESMS_HOME_CHANNEL_NAMETELEGRAM_HOME_CHANNELBLUEBUBBLES_SERVER_URLDINGTALK_CLIENT_SECRETMATRIX_REQUIRE_MENTIONQQ_GROUP_ALLOWED_USERSWECOM_CALLBACK_CORP_IDWEIXIN_ALLOW_ALL_USERSHERMES_LANGFUSE_RELEASEMATTERMOST_HOME_CHANNELQQBOT_HOME_CHANNEL_NAMESLACK_HOME_CHANNEL_NAMEWECOM_CALLBACK_AGENT_IDWECOM_HOME_CHANNEL_NAMEBLUEBUBBLES_HOME_CHANNELFEISHU_HOME_CHANNEL_NAMESIGNAL_HOME_CHANNEL_NAMEWEIXIN_HOME_CHANNEL_NAMEDISCORD_HOME_CHANNEL_NAMEFEISHU_VERIFICATION_TOKENHERMES_LANGFUSE_MAX_CHARSYUANBAO_HOME_CHANNEL_NAMEDINGTALK_HOME_CHANNEL_NAMEMATRIX_FREE_RESPONSE_ROOMSSIGNAL_GROUP_ALLOWED_USERSTELEGRAM_HOME_CHANNEL_NAMEWECOM_CALLBACK_CORP_SECRETWEIXIN_GROUP_ALLOWED_USERSHERMES_LANGFUSE_SAMPLE_RATEMATTERMOST_HOME_CHANNEL_NAMEBLUEBUBBLES_HOME_CHANNEL_NAME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 rz   z.managedN)	osgetenvstriplower_MANAGED_TRUE_VALUES_MANAGED_SYSTEM_NAMESgetget_hermes_homeexists)raw
normalizedmanaged_markers      6/home/ubuntu/.hermes/hermes-agent/hermes_cli/config.pyget_managed_systemr   t   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r    r   r   
is_managedr      s     t++r   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.ry   zbrew upgrade hermes-agentrz   zsudo nixos-rebuild switchNr   )managed_systems    r   get_managed_update_commandr      s4    '))N##**  **4r   c                  "    t                      pdS )z<Return the best update command for the current installation.zhermes update)r   r   r   r   recommended_update_commandr      s    %'':?:r   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 managerr   r   rz   rv   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 switchry   r|   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.)r   r   r   r   r   r   )r   r   r   env_hints       r   format_managed_messager      s    '))@-@N
)$b
)
)
/
/
1
1
7
7
9
9C   $88866cmV*f * *'* * *	
 ##$**f * *'* * *	
	C& 	C 	C> 	C 	C 	Cr   modify configurationc                 V    t          t          |           t          j                   dS )z+Print user-friendly error for managed mode.fileN)printr   sysstderr)r   s    r   managed_errorr      s&    	
 
(
(sz::::::r   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_DEVrw   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   environr   hermes_constantsr   r   openr   
startswith	partitionFileNotFoundError)r   container_mode_fileinfoflinekey_valuer   r   r   r   s               r   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r   )atomic_replacec                  $    t                      dz  S )zGet the main config file path.zconfig.yamlr   r   r   r   get_config_pathr      s    },,r   c                  $    t                      dz  S )z&Get the .env file path (for API keys).z.envr   r   r   r   get_env_pathr      s    v%%r   c                  b    t          t                    j        j                                        S )z'Get the project installation directory.)r   __file__parentresolver   r   r   get_project_rootr      s     >> '//111r   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_MODEr      i  )
r   r   r   r   r   int
ValueErrorchmodOSErrorNotImplementedError)pathmode_strmodes      r   _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   r   r   r   r   readr   IOError)r   cgroup_contents     r   _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  )	r   r   r   r   r   strr   r   r   )r   s    r   _secure_filer   5  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)r   
write_textru   r   )r   	soul_paths     r   _ensure_default_soul_mdr   G  sR    y I 7;;;r   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logszlogs/curatormemoriesN)r   r   r   umask_ensure_hermes_home_managedmkdirr   r   )r   	old_umasksubdirds       r   ensure_hermes_homer   P  s     D|| &HUOO		 '---HYBHY

4$
///DN 	 	FvAGGD4G000NNNN%%%%%s   A A-c                    |                                  st          d|  d          dD ]-}| |z  }|                                 st          | d          .| dz  dz                      d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   r   r   r   r   curatorTr   N)is_dirRuntimeErrorr   r   )r   r   r   s      r   r   r   h  s    ;;== 
54 5 5 5
 
 	
 ;  6Mxxzz 	 9 9 9  	 
F]Y%%dT%BBBD!!!!!r   modelr   	providersfallback_providerscredential_pool_strategiestoolsetsz
hermes-cliagentZ   i        autoi  i  )	max_turnsgateway_timeoutrestart_drain_timeoutapi_max_retriesservice_tiertool_use_enforcementgateway_timeout_warninggateway_notify_intervalgateway_auto_continue_freshnessimage_input_modedisabled_toolsetsterminalr   local
modal_modecwd.timeoutenv_passthroughshell_init_filesauto_source_bashrcT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vercel_runtimenode24container_cpu   container_memoryi   container_diski   F)container_persistentdocker_volumesdocker_mount_cwd_to_workspacedocker_run_as_host_userpersistent_shellbrowserx      must_respondi,  managed_persistence)	inactivity_timeoutcommand_timeoutrecord_sessionsallow_private_urlsauto_local_for_private_urlscdp_urldialog_policydialog_timeout_scamofoxcheckpoints2   r      )enabledmax_snapshots
auto_pruneretention_daysdelete_orphansmin_interval_hoursfile_read_max_charsi tool_outputiP  i  )	max_bytes	max_linesmax_line_lengthtool_loop_guardrails   )exact_failuresame_tool_failureidempotent_no_progress   r   )warnings_enabledhard_stop_enabled
warn_afterhard_stop_aftercompression      ?皙?   i  )rC  	thresholdtarget_ratioprotect_last_nhygiene_hard_message_limitprompt_caching	cache_ttl5m
openrouter)response_cacheresponse_cache_ttlbedrock)rC  provider_filterrefresh_intervalasyncdisabled)guardrail_identifierguardrail_versionstream_processing_modetrace)region	discovery	guardrail	auxiliary)providerr  base_urlapi_keyr  
extra_bodydownload_timeoutih  )rs  r  rt  ru  r  rv  )rs  r  rt  ru  r  rv  max_concurrencyiX  )	visionweb_extractrX  session_search
skills_hubapprovalmcptitle_generationr   displaycompactpersonalitykawaiiresume_displayfullbusy_input_mode	interrupttui_auto_resume_recentbell_on_completeshow_reasoning	streamingfinal_response_markdownr   inline_diffs	show_costskindefaulttui_status_indicatorkaomojiuser_message_preview)first_lines
last_linesinterim_assistant_messagestool_progress_commandtool_progress_overrides)r  context_pctr  )rC  fields)tool_preview_lengthephemeral_system_ttl	platformsruntime_footer	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  r  eveeni]  i  )r  languagesample_ratebit_ratezvoxtral-mini-tts-2603z$c69964a6-ab8b-4f8a-9465-ec0925096ec8)r  r  zneuphonic/neutts-air-q4-ggufcpu)	ref_audioref_textr  devicezen_US-lessac-medium)rs  r  
elevenlabsopenaixaimistralneuttspipersttbase)r  r  z	whisper-1zvoxtral-mini-latest)rC  rs  r  r  r  zctrl+b   g      @)
record_keymax_recording_secondsauto_ttsbeep_enabled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_limitrs  
delegation)r  rs  rt  ru  inherit_mcp_toolsetsmax_iterationschild_timeout_secondsreasoning_effortmax_concurrent_childrenmax_spawn_depthorchestrator_enabledsubagent_auto_approveprefill_messages_filegoalsr  skills
   )external_dirstemplate_varsinline_shellinline_shell_timeoutguard_agent_createdr      )rC  keep)rC  interval_hoursmin_idle_hoursstale_after_daysarchive_after_daysbackuphonchotimezonediscord)require_mentionfree_response_channelsallowed_channelsauto_thread	reactionschannel_promptsserver_actionswhatsapptelegram)r  r  slackr  
mattermost	approvalsmanual<   deny)r   r  	cron_modemcp_reload_confirmcommand_allowlistquick_commandshookshooks_auto_acceptpersonalitiessecuritytirith)rC  domainsshared_files)r:  redact_secretstirith_enabledtirith_pathtirith_timeouttirith_fail_openwebsite_blocklistr   )wrap_responsemax_parallel_jobskanban)dispatch_in_gatewaydispatch_interval_secondscode_executionr   projectloggingINFO)levelmax_size_mbbackup_countmodel_catalogzAhttps://hermes-agent.nousresearch.com/docs/api/model-catalog.json)rC  url	ttl_hoursr  network
force_ipv4)rE  rF  vacuum_after_prunerH  seen)pre_update_backupbackup_keep   )r   
onboardingupdates_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	     rS  r     ENV_VARS_BY_VERSIONNOUS_BASE_URLzNous Portal base URL overridez.Nous Portal base URL (leave empty for default)rs  )descriptionpromptr  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)r4  r5  r  r6  toolsr7  r8  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)NVIDIA_API_KEYz;NVIDIA NIM API key (build.nvidia.com or local NIM endpoint)zNVIDIA NIM API keyzhttps://build.nvidia.com/NVIDIA_BASE_URLzJNVIDIA NIM base URL override (e.g. http://localhost:8000/v1 for local NIM)z-NVIDIA NIM base URL (leave empty for default)
LM_API_KEYz5LM Studio bearer token for auth-enabled local serversz LM Studio API key / bearer tokenLM_BASE_URLzLM Studio base URL overridez,LM Studio 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STEPFUN_API_KEYzStepFun Step Plan API keyzhttps://platform.stepfun.com/STEPFUN_BASE_URLz#StepFun Step Plan base URL overridez4StepFun Step Plan base URL (leave empty for default)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)GMI_API_KEYzGMI Cloud API keyzhttps://www.gmicloud.ai/GMI_BASE_URLzGMI Cloud base URL overridez,GMI Cloud 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)r4  r5  r  r6  r7  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_KEYzhXiaomi MiMo API key for MiMo models (mimo-v2.5-pro, mimo-v2.5, 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AZURE_FOUNDRY_API_KEYz0Azure Foundry API key for custom Azure endpointszAzure Foundry API Keyzhttps://ai.azure.com/AZURE_FOUNDRY_BASE_URLzLAzure Foundry base URL (set via 'hermes model' for endpoint-specific config)zAzure Foundry base URLEXA_API_KEYz1Exa API key for AI-native web search and contentszExa API keyzhttps://exa.ai/
web_searchrz  tool)r4  r5  r  r<  r6  r7  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)rn  rz  	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))r4  r5  r<  r6  r7  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NOTION_API_KEYz5Notion integration token (used by the `notion` skill)zNotion API keyz%https://www.notion.so/my-integrationsskillLINEAR_API_KEYz4Linear personal API key (used by the `linear` skill)zLinear API keyzhttps://linear.app/settings/apiAIRTABLE_API_KEYz=Airtable personal access token (used by the `airtable` skill)zAirtable API keyz"https://airtable.com/create/tokensTENOR_API_KEYz=Tenor API key for GIF search (used by the `gif-search` skill)zTenor API keyz5https://developers.google.com/tenor/guides/quickstart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))r4  r5  r7  HERMES_LANGFUSE_PUBLIC_KEYz'Langfuse project public key (pk-lf-...)zLangfuse public keyzhttps://cloud.langfuse.comHERMES_LANGFUSE_SECRET_KEYz'Langfuse project secret key (sk-lf-...)zLangfuse secret keyHERMES_LANGFUSE_BASE_URLz9Langfuse server URL (default: https://cloud.langfuse.com)z8Langfuse server URL (leave empty for cloud.langfuse.com)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))r4  r5  r6  r7  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)rW   zZRequire @mention in Matrix rooms (default: true). Set to false to respond to all messages.z&Require @mention in rooms (true/false)rj   zCComma-separated Matrix room IDs where bot responds without @mentionz(Free-response room IDs (comma-separated)r5   z@Auto-create threads for messages in Matrix rooms (default: true)z)Auto-create threads in rooms (true/false)rQ   z>Auto-create threads for DM messages in Matrix (default: false)z'Auto-create threads in DMs (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 keyrU   zOBlueBubbles server URL for iMessage integration (e.g. http://192.168.1.10:1234)zBlueBubbles server URLzhttps://bluebubbles.app/rG   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)r4  r5  r  r7  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 UsersrX   z=Comma-separated QQ group IDs allowed to interact with the botzQQ Group Allowed Usersr7   z4Allow all QQ users without an allowlist (true/false)zAllow All QQ Usersr6   z<Default QQ channel/group for cron delivery and notificationszQQ Home Channelr]   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 Moder   z*IRC server hostname (e.g. irc.libera.chat)z
IRC serverr   z"IRC channel to join (e.g. #hermes)zIRC channelr   z)Bot nickname on IRC (default: hermes-bot)zIRC nicknamer<   z!IRC server password (if required)zIRC server passwordrP   z)NickServ password for nick identificationzNickServ password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 hostAPI_SERVER_MODEL_NAMEzModel 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 nameGATEWAY_PROXY_URLu  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)GATEWAY_PROXY_KEYzBearer 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 keyWEBHOOK_ENABLEDzREnable the webhook platform adapter for receiving events from GitHub, GitLab, etc.zEnable webhooks (true/false)WEBHOOK_PORTz1Port for the webhook HTTP server (default: 8644).zWebhook portWEBHOOK_SECRETz[Global HMAC secret for webhook signature validation (overridable per route in config.yaml).zWebhook secretSUDO_PASSWORDzySudo password for terminal commands requiring root access; set to an explicit empty string to try empty without promptingzSudo passwordsettingHERMES_MAX_ITERATIONSz>Maximum tool-calling iterations per conversation (default: 90)zMax iterationsHERMES_TOOL_PROGRESSz=(deprecated) Use display.tool_progress in config.yaml insteadu.   Tool progress (deprecated — use config.yaml)HERMES_TOOL_PROGRESS_MODEu.   Progress mode (deprecated — use config.yaml)HERMES_PREFILL_MESSAGES_FILEzFPath to JSON file with ephemeral prefill messages for few-shot primingzPrefill messages file pathHERMES_EPHEMERAL_SYSTEM_PROMPTzOEphemeral system prompt injected at API-call time (never persisted to sessions)z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       r   get_missing_env_varsr  H	  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Nr   
dotted_keyc           	      j   |                     d          }| }|dd         D ]}t          |t                    rE	 t          |          }n+# t          t
          f$ r t	          d|d|d          w xY w||         }\t          |t                    rC|                    |          }||vst          |t          t          f          si ||<   ||         }t	          dt          |          j	         d|          |d         }t          |t                    r||t          |          <   dS |||<   dS )	u  Set a value at an arbitrarily nested dotted key path.

    Supports both dict and list navigation:
      _set_nested(c, "a.b.c", 1)     → c["a"]["b"]["c"] = 1
      _set_nested(c, "a.0.b", 1)     → c["a"][0]["b"] = 1
      _set_nested(c, "providers.1", "x") → c["providers"][1] = "x"

    Intermediate dicts are created on demand.  List indices are parsed
    from numeric path segments; the referenced index must already exist
    (we do not grow lists — the user is navigating into structure they
    wrote themselves).  If a segment targets a non-container leaf
    (scalar), the leaf is replaced with a fresh dict so the write can
    proceed — this preserves the pre-existing behavior for bare scalar
    overrides (e.g. setting ``a.b.c`` where ``a.b`` was previously a
    string).

    Guards against #17876: before this fix the code unconditionally
    replaced any non-dict value (including lists) with ``{}``, silently
    destroying list-typed config like ``custom_providers`` whenever a
    caller used an indexed path.
    r  Nz!Cannot navigate into list at key z
: segment z is not a numeric indexzCannot navigate into z at key )
split
isinstancelistr   	TypeErrorr   dictr   type__name__)	configr  r   partscurrentpartidxexistinglasts	            r   _set_nestedr  ^	  s|   , S!!EGcrc
  gt$$ 	$iiz*   ?
 ? ?#? ? ?  
 clGG&& 		{{4((H7""*Xd|*L*L" "dmGGVW(>VV
VV   9D'4   "D		s   A

(A2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.
    r   defaultsr  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   r  r4  )r  r   r  r  r  r   )r  r  r  r   default_valuefull_key_checkr  s         r   r  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===	> 	>r   )r   )load_configr  r   DEFAULT_CONFIG)r  r  r  s    @@r   get_missing_config_fieldsr  	  sj     ]]FG> > > >c > > > > > > > F>6"""Nr   c                  6   	 ddl m} m} n# t          $ r g cY S w xY w	  |             }nF# t          $ r9}ddl} |j        t                                        d|           g cY d}~S d}~w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_PREFIXNz)discover_all_skill_config_vars failed: %sr  r   )agent.skill_utilsr  r  	Exceptionr  	getLoggerr  debugr  r  r  r  r   r   r  )r  r  all_varser  r  r  varstorage_keyr  r  r   r  s                r   get_missing_skill_config_varsr  	  s   YYYYYYYYY   			
1133    	(##))7	
 	
 	
 						  	]]F$&G    ,;;s5z;;!!#&& 	 	D'4(( TW__!$-=Zs33=EKKMM=NN3Ns&    
) 
A,.A'!A,'A,provider_keyentryr  c          	      *	   t          | t                    sdS ddddddddd	}d
| v rd| vr| d
         | d<   h d}|                                D ]7\  }}|| v r.|| vr*t                              d|pd||           | |         | |<   8t          |                                           |z
  t          |                                          z
  }|r>t                              d|pdd                    t          |                               ddl	m
} d}dD ]}	|                     |	          }
t          |
t                    rd|
                                rP|
                                } ||          }|j        r|j        r|} n t                              d|pd|	|           |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<   n&t          |t                     r|rd |D             |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``.Nru  rt  api_modekey_envdefault_modelcontext_lengthrate_limit_delay)apiKeybaseUrlapiModekeyEnv	apiKeyEnvdefaultModelcontextLengthrateLimitDelayapi_key_env>   apir  r  r  modelsru  r  r  rt  	transportr  r  r  r  stale_timeout_secondsrequest_timeout_secondsz[providers.%s: camelCase key '%s' auto-mapped to '%s' (use snake_case to avoid this warning)?z-providers.%s: unknown config keys ignored: %s, r   urlparser   )rt  r  r  uP   providers.%s: '%s' value '%s' is not a valid URL (no scheme or host) — skippedr  )r  rt  r  r  r  r  c                     i | ];}t          |t                    |                                +t          |          i <S r   )r  r   r   ).0ms     r   
<dictcomp>z4_normalize_custom_provider_entry.<locals>.<dictcomp>H
  sO      
  
  
*Q*<*< 
AB 
FFB 
  
  
r   )r  r  r  loggerwarningsetkeysjoinsortedurllib.parser  r   r   r   schemenetlocr  r   float)r  r  _CAMEL_ALIASES_KNOWN_KEYScamelsnakeunknownr  rt  url_keyraw_url	candidateparsedr  raw_namer   ru  r  r  
model_namer  r  r  s                          r    _normalize_custom_provider_entryr  	  s    eT"" t '),	& 	&N )5"8"8 /i  K ',,.. ( (uE>>e500NN9#UE  
 !<E%L%**,,+-N4G4G4I4I0J0JJG 
;C6'??!;!;	
 	
 	

 &%%%%%H-  ))G$$gs## 	 	IXi((F}  $6 'C)  
  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	FD	!	! 
f 

 
  
 & 
  
  

8 YY/00N.#&& 6>A+=+='5
#$yy!344"S%L11 :6F!6K6K)9
%&r   providers_dictc                     t          | t                    sg S g }|                                 D ]:\  }}t          |t	          |                    }||                    |           ;|S )zMNormalize ``providers`` config entries into the legacy custom-provider shape.r  )r  r  r  r  r   r  )r   custom_providersr   r  r   s        r   "providers_dict_to_custom_providersr#  W
  sw    nd++ 	-/$**,, 0 0
U5e#c((SSS
!##J///r   r  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.
    Nr  r   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 )Nr  r   r  rt  /r  )r   r   r   r   rstripr  add)	r  r  r  rt  r  pair
compatibleseen_name_url_pairsseen_provider_keyss	         r   _append_if_newz7get_compatible_custom_providers.<locals>._append_if_newv
  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)))))	* 	* 	* 	*r   r"  r  )r  r  r   r   r   r   r   r  r  r  r#  )r  r-  r"  r  r*  r+  r,  s       @@@r   get_compatible_custom_providersr.  e
  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r   rt  r"  c                    | r|sdS |T	 t          |          }nC# t          $ r6 |Y dS |                    d          }t          |t                    r|ng }Y nw xY wt          |t                    sdS |pd                    d          }|sdS |D ]}t          |t                    s|                    d          pd                    d          }|r||k    rK|                    d          }t          |t                    sv|                    |           }	t          |	t                    s|	                    d          }
|
	 t          |
          }n# t          t          f$ r Y w xY w|dk    r|c S dS )	a  Look up a per-model ``context_length`` override from ``custom_providers``.

    Matches any entry whose ``base_url`` equals ``base_url`` (trailing-slash
    insensitive) and returns ``custom_providers[i].models.<model>.context_length``
    if present and valid.  Returns ``None`` when no override applies.

    This is the single source of truth for custom-provider context overrides,
    used by:
      * ``AIAgent.__init__`` (startup resolution)
      * ``AIAgent.switch_model`` (mid-session ``/model`` switch)
      * ``hermes_cli.model_switch.resolve_display_context_length`` (``/model`` confirmation display)
      * ``gateway.run._format_session_info`` (``/info`` display)
      * ``agent.model_metadata.get_model_context_length`` (when custom_providers is threaded through)

    Before this helper existed, the lookup was duplicated in ``run_agent.py``'s
    startup path only; every other path (notably ``/model`` switch) fell back
    to the 128K default.  See #15779.
    Nr"  r   r&  rt  r  r  r   )
r.  r  r   r  r  r'  r  r   r  r   )r  rt  r"  r  r   
target_urlr  	entry_urlr  	model_cfgraw_ctxctxs               r   "get_custom_provider_context_lengthr5  
  s   0   t	D>vFF 	D 	D 	D~tt**/00C&0d&;&;Css		D
 &-- t.b((--J t!  %&& 	YYz**0b88==	 	I338$$&$'' 	JJu%%	)T** 	-- 011?	g,,CC:& 	 	 	H	77JJJ 4s&    A.AAEE/.E/c                      t                      } |                     dd          }t                              dd          }||fS )zS
    Check config version.
    
    Returns (current_version, latest_version).
    r#  r   r*  )r  r   r  )r  r  latests      r   check_config_versionr8  
  sA     ]]Fjj*A..G 1155FF?r   >   r  r  r  r  r  gatewayr   r  r  rr  r  r  rX  fallback_modelr#  r"  r  r  >	   r  r  r  ru  r  r  rt  r  r  >   ru  r  rt  r  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)r  
__module____qualname____doc__r   __annotations__r   r   r   r<  r<  
  s1         ..MMMLLL
IIIIIr   r<  c                 d	   | 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                    rt          |          D ]\  }}t	          |t
                    s=|                    t          dd| dt          |          j         d                     W|                    d          s(|                    t          dd| dd                     |                    d          s(|                    t          dd| dd                     nt	          |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 configr"  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: ...r  z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-providerrt  z] is missing 'base_url' fieldzDAdd the API endpoint URL, e.g.: base_url: https://api.example.com/v1r:  zfallback_model[z] should be a dict, got z!Each entry needs provider + modelrs  z] is missing 'provider' fieldz/Add: provider: openrouter (or another provider)r  z] is missing 'model' fieldzAdd: model: <model-name>zAfallback_model should be a dict with 'provider' and 'model', got zZChange to:
  fallback_model:
    provider: openrouter
    model: anthropic/claude-sonnet-4uH   fallback_model is missing 'provider' field — fallback will be disableduE   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)r  r  r<  r   r  r  r  r  r  _CUSTOM_PROVIDER_LIKE_FIELDSr  r  	enumerater  r  r   _KNOWN_ROOT_KEYS)
r  issuescpcp_keys
suspiciousir  fbr2  r   s
             r   validate_config_structurerP    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	~b$ +	%bMM  5!%.. MM+[![[T%[[EY[[;# #     !99Z00 k%NaNNNM' '   
 !99W-- k%KaKKK6' '   ( B%% 	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                 T   	 t          |           }n# t          $ r Y dS w xY w|sdS dg}|D ]1}|j        dk    rdnd}|                    d| d|j                    2|                    d           t
          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.
    Nu3   [33m⚠ Config issues detected in config.yaml:[0mrE  u   [31m✗[0mu   [33m⚠[0m   z2  [2mRun 'hermes doctor' for fix suggestions.[0m


)	rP  r  r=  r  r>  r   r   writer  )r  rJ  linescimarkers        r   print_config_warningsrZ    s    *622    HIE 1 1)+)?)?%%EY/&//2://0000	LLKLLLJTYYu%%./////s    
  c                    t           j                            d          }t           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	           |rt           j                            dd          }|                    dd           |                    d           |                    d| d           t          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.
    MESSAGING_CWDTERMINAL_CWDNr  r  r  )r  r
  r  r   u     [33m⚠[0m MESSAGING_CWD=u&    found in .env — this is deprecated.u     [33m⚠[0m TERMINAL_CWD=HERMES_HOMEz	~/.hermesr   u/   [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[0mrT  rU  )r   r   r   r  r  r  r  r  insertr   r   rV  r  )r  messaging_cwdterminal_cwd_envterminal_cfg
config_cwdconfig_has_explicit_cwdrW  	hint_paths           r   warn_deprecated_cwd_env_varsrf    s    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           
      f0  Y 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_           |d`k     r$t/                      }|                    da          }1t          |1t                    si }1db|1vr|1                    dcg           pg }2t          |2t                     sg }2t9          |2          }3g }4	 t;                      daz  }5|5                                rt?          |5                                           D ]}6|6                                s|6ddz  }7|7!                                s|6dez  }7|7!                                sJ	 tE          |7          5 }8tG          j$        |8          pi }9d7d7d7           n# 1 swxY w Y   n# t          $ r i }9Y nw xY w|9                    d#          p|6j%        }:|:|3v r|4
                    |:           n# t          $ r g }4Y nw xY w|4|1db<   |1|da<   t          |           |d         
                    dftM          |4           dg           |s2|4r!t          dhtM          |4           di           nt          dj           |dkk     r	 t;                      dlz  dmz  };|;'                    dIdIn           n8# t          $ r+}<|do         
                    dp|; d;|<            Y d7}<~<nd7}<~<ww xY wt/                      }dq}=tP                              dmi           }>|                    dm          }?t          |?t                    si }?g }@|>                                D ]5\  }A}B|A|?vr,tS          j*        |B          |?|A<   |@
                    |A           6|@r|?|dm<   dI}=tP                              dXi                               dmi           }C|                    dX          }Dt          |Dt                    si }D|D                    dm          }Et          |Et                    si }Eg }F|C                                D ]5\  }A}B|A|Evr,tS          j*        |B          |E|A<   |F
                    |A           6|Fr|E|Ddm<   |D|dX<   dI}=|=rt          |           |@rS|d         
                    drtM          |@           ds           |s%t          dtdO                    |@                      |FrS|d         
                    dutM          |F           ds           |s%t          dvdO                    |F                      ||k     r|st          dw| dx|            tW          dIy          }G|Gr7|s5t          dz           |GD ]#}Ht          d{|Hd#          d;|Hd|                     $| r/|Gr,t          d}           |GD ]}H|H                    d%          rt          d~|Hd%                     |H                    d          r$d"d7l,}I|I,                    d|Hd          d;          }Jn+t[          d|Hd          d;                                          }J|JrPt          |Hd#         |J           |d         
                    |Hd#                    t          d|Hd#                     n%|do         
                    d|Hd#          d           t                       tW          dqy          }K|Grd |GD             nt9                      YYfd|KD             }Kt9                      }Lt]          |d6z   |d6z             D ]0}M|L/                    t`                              |Mg                      1|LrC| r@|s=d t?          |L          D             }N|Nr!t          dtM          |N           d           |ND ].\  }:}Ot          d|: d|O                    d|d                      /t                       	 t[          d                                          	                                }Pn# tb          td          f$ r d}PY nw xY w|Pdv r`t                       |ND ]M\  }:}O|O                    d%          r?t          d|O                    d||:                      t          d~|Od%                     n&t          d|O                    d||:                      |O                    d          r2d"d7l,}I|I,                    d|O                    d|:           d          }Jn9t[          d|O                    d|:           d                                          }J|Jr=t          |:|J           |d         
                    |:           t          d|:            t                       Ont          d           tg                      }Q|Qr{t	                      }|QD ]U}R|Rd         }|Rd         }Sti          |||S           |d         
                    |           |st          d| d|S            V||d<   t          |           n(||k     r"t	                      }||d<   t          |           tk                      }T|Tr*| r'|s$t          dtM          |T           d           |TD ]=}H|H                    dd          }Ut          d|Hd          d|Hd|          d|U d*           >t                       	 t[          d                                          	                                }Pn# tb          td          f$ r d}PY nw xY w|Pdv rTt                       t	                      }	 d"dl6m7}V n# t          $ r d}VY nw xY w|TD ]
}H|H                    dd          }S|Srd|S d*nd}Wt[          d|Hd          |W d;                                          }J|Js|Srt7          |S          }J|Jr[|V d.|Hd          }Xti          ||X|J           |d         
                    |Hd                    t          d|Hd          d|J            n<|do         
                    d|Hd          d|H                    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)r0  r  tool_progressr  r  )false0nor  rk  z;display.tool_progress=off (from HERMES_TOOL_PROGRESS=false))newallzdisplay.tool_progress=z! (from HERMES_TOOL_PROGRESS_MODE)rr  z#display.tool_progress=all (default)u-     ✓ Migrated tool progress to config.yaml: rS  r  HERMES_TIMEZONEr   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)   r"  r  r   r  rt  r  ru  rS  -(rF  z--r  endpointr  z	endpoint-r  )zno-keyzno-key-requiredr   r  r  r  r  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  rs  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.5r  tinylargesmallturbomediumu;     ✓ Migrated legacy stt.model to provider-specific config   r  Tz1display.interim_assistant_messages=true (default)u3     ✓ Added display.interim_assistant_messages=true   r  r  r  c              3   *   K   | ]\  }}| d | V  dS )r   Nr   )r  pr  s      r   	<genexpr>z!migrate_config.<locals>.<genexpr>  s0      $R$RDAqZZAZZ$R$R$R$R$R$Rr   u>     ✓ Migrated tool_progress_overrides → display.platforms: z9display.platforms (migrated from tool_progress_overrides)   rX  summary_modelsummary_providersummary_base_urlrr  model=)r   r
  r
  	provider=z	base_url=u@     ✓ Migrated compression.summary_* → auxiliary.compression: u/     ✓ Removed unused compression.summary_* keys   pluginsrC  rj  zplugin.yamlz
plugin.ymlz$plugins.enabled (opt-in allow-list, z grandfathered)u(     ✓ Plugins now opt-in: grandfathered z( existing plugin(s) into plugins.enabledum     ✓ Plugins now opt-in: no existing plugins to grandfather. Use `hermes plugins enable <name>` to activate.r   r   r   r   rl  zCould not create Fz	curator (z default key(s))u.     ✓ Seeded curator defaults in config.yaml: zauxiliary.curator (u8     ✓ Seeded auxiliary.curator defaults in config.yaml: zConfig version:     → r  u0   
⚠️  Missing required environment variables:u      • r4  z
Let's configure them now:
z  Get your key at: r6  rR  r5  rj  u     ✓ Saved zSkipped z - some features may not workc                     h | ]
}|d          S )r  r   r  vs     r   	<setcomp>z!migrate_config.<locals>.<setcomp>  s    555Aai555r   c                 R    g | ]#}|d          v|                     d          !|$S )r  r8  r   )r  r  required_namess     r   
<listcomp>z"migrate_config.<locals>.<listcomp>  sC       V9N**1553D3D* 	
***r   c                 \    g | ])}t          |          s|t          v |t          |         f*S r   )r  r  )r  r  s     r   r  z"migrate_config.<locals>.<listcomp>  sK     
 
 
 &&
 ,03D+D+D $T*++D+D+Dr   
  z$ new optional key(s) in this update:u       • u    — z  Configure new keys? [y/N]: n)yrx   z (Enter to skip): z1  Set later with: hermes config set <key> <value>r   r  u     ✓ Added  = r#  z! skill setting(s) not configured:r  r  z (from skill: z#  Configure skill settings? [y/N]: )r  zskills.configz (default: u    — skill 'r  z' may ask for it later)8sanitize_env_filer   r  r8  r  r   r  r  r  r   r  save_configr   r   r   save_env_valuer  replacer  r  hostnamepopr  read_raw_config
setdefaultr  r  r   r  r   r   r  iterdirr   r   yaml	safe_loadr  lenr   r  copydeepcopyr  getpassinputrangeupdater2  EOFErrorKeyboardInterruptr  r  r  r  r  )Zrg  rh  resultsfixescurrent_ver
latest_verr  r  old_enabledold_modeold_tz
tz_display	old_tokencustom_listr   migrated_countr  old_nameold_urlold_keyr   r  r  	new_entryepdead_varold_valr   raw_sttlegacy_modelrs  r  _local_models	raw_local	local_cfgraw_providerprovider_cfgold_overridesr  platr   migratedcomps_model
s_provider
s_base_urlmigrated_keysauxaux_compplugins_cfgrj  disabled_setgrandfathereduser_plugins_dirchildmanifest_file_mfmanifestr  curator_dirr  touched_curator_defaultsraw_curatoradded_curatorkr  _aux_curator_defaultsraw_auxraw_aux_curator	added_auxmissing_envr  r  r   missing_optionalnew_var_namesvernew_and_unsetr   answermissing_configfieldr  missing_skill_config
skill_namer  default_hintr  r  sZ                                                                                            @r   migrate_configr    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 R ""jj+++t,, 	KK''"z266<"Hh-- x==L (*M##2#4#4y#@ #**,, 3!'(8(@(@(B(B!C!C 3 3$||~~ %$(-(=,3355 A,1L,@M,3355 %$*!%m!4!4 E+/>#+>+>+D"E E E E E E E E E E E E E E E( * * *')HHH*'||F33Auz<//$%,,T2222 # # # "# &3K	" +F9N#**Zs=7I7IZZZ    
  	X}--X X X   
 J  0 R	O)++f4y@KdT:::: 	O 	O 	OJ&&'M;'M'M!'M'MNNNNNNNN	O !"" +..y"==jj+++t,, 	K#%%++-- 	( 	(DAq##!%q!1!1A$$Q''' 	 +F9G {B//33IrBB 	 **[))'4(( 	G!++i00/400 	! O!	)//11 	$ 	$DAq''%)]1%5%5"  ### 	!0GI")F;G 	 '..DM 2 2DDD    699]336 6    '..J#i..JJJ    299Y//2 2  
 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%>Bv uu5uu	uu		uv uv u:v v&%v&*+y 
z !zzN3AN6 N6AOOAOY<3AZ0 Z0A[[A[[+A[2 [2A\\ A\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.
    )r  r  r  r  _deep_merge)r  r  resultr   r   s        r   r  r    s     YY[[Fnn&&    
U6MM6#;-- 5$''  &fSk599F3KKF3KKMr   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   r   group)r  s    r   <lambda>z"_expand_env_vars.<locals>.<lambda>;  s*    bjnnQWWQZZ<< r   c                 4    i | ]\  }}|t          |          S r   _expand_env_vars)r  r  r  s      r   r	  z$_expand_env_vars.<locals>.<dictcomp>?  s'    ???41a#A&&???r   c                 ,    g | ]}t          |          S r   r  )r  items     r   r  z$_expand_env_vars.<locals>.<listcomp>A  s!    7774 &&777r   )r  r   resubr  r  r  )objs    r   r  r  1  s     #s 
v<<
 
 	

 #t @??399;;????#t 87737777Jr   c                     t          | t                    sdS i }| D ]V}t          |t                    r(t          |                    d          t                    s dS |d         }||v r dS |||<   W|S )zHReturn a name-indexed dict only when all items have unique string names.Nr  )r  r  r  r   r   )r  indexedr  r  s       r   _items_by_unique_namer  E  s    eT"" tG  $%% 	Z8H8H#-N-N 	44F|7??44Nr   c                    t          | t                    rft          t                    rQt          j        d          r<| k    rS t          t                    r| k    rS t	                    | k    rS | S t          | t
                    r6t          t
                    r!fd|                                 D             S t          | t                    rqt          t                    r\t          |           }t                    t                    |fd| D             S fdt          |           D             S | S )a  Restore raw ``${VAR}`` templates when a value is otherwise unchanged.

    ``load_config()`` expands env refs for runtime use. When a caller later
    persists that config after modifying some unrelated setting, keep the
    original on-disk template instead of writing the expanded plaintext
    secret back to ``config.yaml``.

    Prefer preserving the raw template when ``current`` still matches either
    the value previously returned by ``load_config()`` for this config path or
    the current environment expansion of ``raw``. This handles env-var
    rotation between load and save while still treating mixed literal/template
    string edits as caller-owned once their rendered value diverges.
    z	\${[^}]+}c                     i | ]T\  }}|t          |                    |          t          t                    r                    |          nd           US N)_preserve_env_ref_templatesr   r  r  )r  r   r   loaded_expandedr   s      r   r	  z/_preserve_env_ref_templates.<locals>.<dictcomp>l  sr     
 
 
 U ,,6,M,MW##C(((SW 
 
 
r   Nc                     g | ]c}t          |                    |                    d                     (                    |                    d                     nd          dS )r  N)r  r   )r  r  loaded_by_nameraw_by_names     r   r  z/_preserve_env_ref_templates.<locals>.<listcomp>~  s{         ,OODHHV$4$455<J<VN&&txx'7'7888\`   r   c           
          g | ]a\  }}t          ||t                    k     r|         nd t          t                    r|t                    k     r|         nd           bS r  )r  r  r  r  )r  indexr  r  r   s      r   r  z/_preserve_env_ref_templates.<locals>.<listcomp>  s     	
 	
 	
 t (#c#hh..E

Dot449>_AUAU9U9U  && 	
 	
 	
r   )
r  r   r  searchr  r  r  r  r  rH  )r  r   r  current_by_namer  r  s    `` @@r   r  r  T  s    '3 JsC$8$8 RY|UX=Y=Y c>>Jos++ 	?0J0JJC  G++J'4   
ZT%:%: 

 
 
 
 
 &mmoo
 
 
 	
 '4   
ZT%:%: 

 088+C00.??&;+B     $   	
 	
 	
 	
 	
  )11	
 	
 	
 		
 Nr   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/context_length into model section.

    Some users (or older code) placed ``provider:``, ``base_url:``, or
    ``context_length:`` 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 value.
    After migration the root-level keys are removed so they can't cause
    confusion on subsequent loads.
    c              3   B   K   | ]}                     |          V  d S r  r  )r  r  r  s     r   r  z-_normalize_root_model_keys.<locals>.<genexpr>  s-      UUQ6::a==UUUUUUr   )rs  rt  r  r  r  N)anyr  r   r  r  )r  has_rootr  r   root_vals   `    r   _normalize_root_model_keysr'    s     UUUU*TUUUUUH &\\FJJwEeT""  &+3E""w9  ::c?? 	"EIIcNN 	"!E#J

3Mr   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)r  r   r  r  )r  agent_configs     r   _normalize_max_turns_configr*    s    &\\F

7++1r22LfL!@!@$*;$7[!,&&$27$;K$H[!"F7O
JJ{D!!!Mr   )r  cfgr  c                    t          | t                    s|S | }|D ]+}t          |t                    s|c S ||vr|c S ||         },|S )u%  Traverse nested dict keys safely, returning ``default`` on any miss.

    Canonical helper for the ``cfg.get("X", {}).get("Y", default)`` pattern
    that appears 50+ times across the codebase. Handles three common gotchas
    in one place:

      1. Missing intermediate keys (returns ``default``, no KeyError).
      2. An intermediate value that's not a dict (e.g. a user wrote a string
         where a section was expected). Returns ``default`` instead of
         AttributeError on ``.get()``.
      3. ``cfg is None`` (callers sometimes pass ``load_config() or None``).

    Named ``cfg_get`` rather than ``cfg_path`` to avoid shadowing the
    ubiquitous ``cfg_path = _hermes_home / "config.yaml"`` local variable
    that appears in gateway/run.py, cron/scheduler.py, main.py, etc.

    Explicit ``None`` values are returned as-is (matches ``dict.get(key,
    default)`` semantics — ``default`` is only returned when the key is
    *absent*, not when it's present but set to ``None``).

    Examples:
        >>> cfg_get({"agent": {"reasoning_effort": "high"}}, "agent", "reasoning_effort")
        'high'
        >>> cfg_get({}, "agent", "reasoning_effort", default="medium")
        'medium'
        >>> cfg_get({"agent": "oops_a_string"}, "agent", "reasoning_effort", default="low")
        'low'
        >>> cfg_get(None, "anything", default=42)
        42
        >>> cfg_get({"a": {"b": None}}, "a", "b", default="def")  # explicit None preserved
        >>> cfg_get({"a": {"b": False}}, "a", "b", default=True)  # falsy values preserved
        False
    )r  r  )r+  r  r  noder   s        r   cfg_getr.    sm    D c4   D  $%% 	NNNd??NNNCyKr   c                  l   	 t                      } |                                 }|j        |j        f}n# t          t
          f$ r i cY S w xY wt          |           }t                              |          }|(|dd         |k    rt          j
        |d                   S 	 t          | d          5 }t          j        |          pi }ddd           n# 1 swxY w Y   n# t          $ r i cY S w xY wt          |t                     si }|d         |d         t          j
        |          ft          |<   |S )u  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.

    Cached on the config file's (mtime_ns, size) — same strategy as
    ``load_config()``. Returns a deepcopy on every call since some callers
    mutate the result before passing to ``save_config()``.
    NrO  r   r   r   r*  )r   statst_mtime_nsst_sizer   r   r   r   r   r  r  r   r  r  r  r  r  )config_pathst	cache_keypath_keycachedr   datas          r   r  r    s   %''^RZ0		w'   			 ;H""8,,FfRaRjI55}VAY'''+000 	+A>!$$*D	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+ 	+   			 dD!! #,Q<1t}T?R?R"ShKsE   03 A	A	 C  1CC  CC  CC   C/.C/c                  d   t                       t                      } t          |           }	 |                                 }|j        |j        f}n# t          $ r d}Y nw xY wt                              |          }|*|(|dd         |k    rt          j
        |d                   S t          j
        t                    }|	 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-          |
          }t          j
        |          t.          |<   |,|d         |d	         t          j
        |          ft          |<   nt                              |d           |S )
a  Load configuration from ~/.hermes/config.yaml.

    Cached on the config file's (mtime_ns, size). Returns a deepcopy of
    the cached value when unchanged, since most call sites mutate the
    result (e.g. ``cfg["model"]["default"] = ...`` before ``save_config``).
    The cache is keyed on ``str(config_path)`` so profile switches
    (which change ``HERMES_HOME`` and therefore ``get_config_path()``)
    don't collide.
    NrO  r   r   r  r  z Warning: Failed to load config: r   r*  )r   r   r   r0  r1  r2  r   r   r   r  r  r  r   r  r  r  r  r  r  r   r'  r*  r  r   )r3  r6  r4  r5  r7  r  r   user_configagent_user_configr  r   expandeds               r   r  r    s    !##K;H02
/K		   			  ##H--Fi3rr
i8O8O}VAY''']>**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	: ,,G,O,OPPJ
++H.2mH.E.E!(+(1!ilDMRZD[D['\8$$x...OsN   "A AAE7 C8,E7 8C<<E7 ?C< A6E7 7
FFFu(  
# ── Security ──────────────────────────────────────────────────────────
# Secret redaction is OFF by default — tool output (terminal stdout,
# read_file results, web content) passes through unmodified. Set
# redact_secrets to true to mask strings that look like API keys, tokens,
# and passwords before they enter the model context and logs.
# 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: true
#   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 key_env.
#
# fallback_model:
#   provider: openrouter
#   model: anthropic/claude-sonnet-4
uo  
# ── Security ──────────────────────────────────────────────────────────
# Secret redaction is OFF by default. Set to true to mask strings that
# look like API keys, tokens, and passwords in tool output and logs.
#
# security:
#   redact_secrets: true

# ── 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 key_env.
#
# fallback_model:
#   provider: openrouter
#   model: anthropic/claude-sonnet-4
c           	      .   t                      rt          d           dS ddlm} t	                       t                      }t          t          |                     }|}t          t          t                                          }|r6t          ||t                              t          |                              }g }|                    di           }|r|                    d          |                    t                     |                    di           }d}	t          |t                     rt#          d	 |D                       }	nLt          |t$                    r7t'          |                    d
          o|                    d                    }	|	s|                    t(                      ||||rd                    |          nd           t-          |           t/          j        |          t          t          |          <   dS )z,Save configuration to ~/.hermes/config.yaml.zsave configurationNr   atomic_yaml_writer  r  r:  Fc              3      K   | ]C}t          |t                    o)|                    d           o|                    d          V  DdS )rs  r  N)r  r  r   )r  r  s     r   r  zsave_config.<locals>.<genexpr>  sM      cc[\*Q--V!%%
2C2CVgccccccr   rs  r  r   )extra_content)r   r   utilsr?  r   r   r'  r*  r  r  r   r   r   r  _SECURITY_COMMENTr  r  r$  r  bool_FALLBACK_COMMENTr  r   r  r  )
r  r?  r3  current_normalizedr   raw_existingr  secrO  fb_is_valids
             r   r  r    s   || *+++''''''!##K34OPV4W4WXX#J-.I/J[J[.\.\]]L 
0)--c+.>.>??
 

 E
..R
(
(C (#''*++3&'''	("	-	-BK"d Ccc`bccccc	B		 C266*--A"&&//BB (&'''(-7bggennn4   
 6:mDV6W6W!#k"2"2333r   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   r   _IS_WINDOWSr   	readlines_sanitize_env_linesr   r   r   )
env_pathenv_varsopen_kwr   	raw_linesrW  r   r   r   r   s
             r   load_envrT    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%rW  c           	         t          t                                                    t          z  }g }| D ]}|                    d          }|                                }|r|                    d          r|                    |dz              \g |D ]u}|dz   }|                    |          }|dk    rS                    ||t          |          z   f           |                    ||t          |          z             }|dk    Svt          fdD                       }	t          |	          dk    r}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   rT  r   r   c                 V    h | ]#\  t          fd D                       !$S )c              3   J   K   | ]\  }}|k    o|k    o	||ffk    V  d S r  r   )r  s2e2r  ss      r   r  z0_sanitize_env_lines.<locals>.<setcomp>.<genexpr>  sZ        B a:B!G:RQF(:     r   )r$  )r  r  rZ  match_rangess    @@r   r  z&_sanitize_env_lines.<locals>.<setcomp>  sj     "
 "
 "
!Q     *    "
"
 "
 "
r   r*  )r  r  r  _EXTRA_ENV_KEYSr'  r   r   r  findr  r  rH  )rW  
known_keys	sanitizedr   r   strippedkey_nameneedler  split_positionsrN  posendr  r[  s                 @r   rO  rO    s    &++--..@JI &. &.kk&!!99;;  	8..s33 	S4Z((( /1" 	? 	?H^F--''C((##S#F*;$<===mmFC#f++,=>> (( ! "
 "
 "
 "
&"
 "
 "
   !###O44 2 2301AO8L8L0L0Loa!e,,RUV^R_R_C(..00 2$$TD[111	2 X_----r   c                  ,   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)          ||            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  rK  r   Nc              3   ,   K   | ]\  }}||k    d V  dS )r*  Nr   )r  abs      r   r  z$sanitize_env_file.<locals>.<genexpr>/  s*      KK$!QAFFAFFFFKKr   .tmp.env_dirsuffixr  w)r   r   rM  r   rN  rO  absr  sumziptempfilemkstempr   r   r   fdopen
writelinesflushfsyncfilenor   BaseExceptionunlinkr   r   )	rP  read_kwwrite_kwr   original_linesr_  r  fdtmp_paths	            r   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   	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	x****   	Ih 	 	 	D	 Lsg   A..A25A2G AF4(G 4F88G ;F8<G 
HG0/H0
G=:H<G==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          }t          d
|  dd                    d |dd         D                       z   t          |          dk    rdndz   dz   t          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 rz  z (U+04XrF  ignore)rL  z
  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.

rT  c              3       K   | ]	}d | V  
dS )rR  Nr   )r  r   s     r   r  z._check_non_ascii_credential.<locals>.<genexpr>b  s(      ::DKKK::::::r   NrS  z
  ... and morer   z

  The non-ASCII characters have been stripped automatically.
  If authentication fails, re-copy the key from the provider's dashboard.
r   )encodeUnicodeEncodeErrorrH  ordr  decoder   r  r  r   r   )r   r   	bad_charsrN  chr_  s         r   _check_non_ascii_credentialr  C  sc   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		 	 	 	 ))::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   tC          ||           |'	 t7          j"        ||           n# t4          $ r Y nw xY wn5# tF          $ r( 	 t7          j$        |           n# t4          $ r Y nw xY w w xY wtK          |           |t6          j&        | <   dS )z)Save or update a value in ~/.hermes/.env.zset N#Invalid environment variable name: rT  r   r   r  rK  r   Fr   Tr  rj  rk  rl  ro  )'r   r   _ENV_VAR_NAME_REmatchr   r  r  r   r   rM  r   r   rN  rO  rH  r   r   endswithr  rs  rt  r   r   r0  S_IMODEst_moder   r   ru  rv  rw  rx  ry  r   r   rz  r{  r   r   )r   r   rP  r|  r}  rW  r   foundrN  r   r  r  original_modes                r   r  r  k  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   	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	! 	x***$=1111      	Ih 	 	 	D	 BJsOOOs   #DDD;+H' '
H43H48K, AJ'K, 'J++K, .J+/K, K K, 
K(%K, 'K((K, ,
L7LL
LLLLc                 \    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     }|rdt          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   t9          |	|           |
'	 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  rK  r   c                 h    g | ].}|                                                      d           ,|/S )r   )r   r   )r  r   r   s     r   r  z$remove_env_value.<locals>.<listcomp>  s:    RRR$tzz||/F/F#yyy/Q/QRRRRr   rj  rk  rl  ro  )!r   r   r  r  r   r   r   r   r   r  rM  r   rN  rO  r  rs  rt  r   r   r0  r  r  r   ru  rv  rw  rx  ry  r   r   rz  r{  r   )r   rP  r|  r}  r   rW  	new_linesr  r  r  r  s   `          r   remove_env_valuer    sZ   
 || 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'''			$$$% % % % % % % % % % % % % % % 8X...(HX}5555   D 	 	 		(####   	 	XJNN3Ls   <CC!$C!+F 
FFI 'AH7I HI 
HI !H7 6I 7
II II 
I:I('I:(
I52I:4I55I:c                 H    |pt           } |d|             |dd           dS )zBPersist an Anthropic OAuth/setup token and clear the API-key slot.r   r-   r   Nr  r   save_fnwriters      r   save_anthropic_oauth_tokenr    s8    &F
Fe$$$
F#####r   c                 H    | pt           } |dd            |dd           dS )zHUse Claude Code's own credential files instead of persisting env tokens.r   r   r-   Nr  )r  r  s     r   %use_anthropic_claude_code_credentialsr    s8    &F
Fb!!!
F#####r   c                 H    |pt           } |d|             |dd           dS )zBPersist an Anthropic API key and clear the OAuth/setup-token slot.r-   r   r   Nr  r  s      r   save_anthropic_api_keyr    s8    &F
F&&&
Fb!!!!!r   c                 .    t          | |           d| ddS )NTF)success	stored_as	validatedr  )r   r   s     r   save_env_value_securer    s*    3  r   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*  )	rT  r  r  r  r\  r  r   r   r   )rQ  r^  countr   r   s        r   
reload_envr    s     zzH&++--..@JEnn&&  
U:>>#%''#BJsOQJE  h3"*#4#4
3QJELr   c                     | t           j        v rt           j        |          S t                      }|                    |           S )z/Get a value from ~/.hermes/.env or environment.)r   r   rT  r   )r   rQ  s     r   r  r    s:     bjz# zzH<<r   c                 X    ddl m}  || t          dt          j                            S )u   Redact an API key for display.

    Thin wrapper over :func:`agent.redact.mask_secret` — preserves the
    "(not set)" placeholder in dim color for the empty case.
    r   )mask_secret	(not set))empty)agent.redactr  rt   rs   DIM)r   r  s     r   
redact_keyr  -  s7     )(((((;s%VZ"@"@AAAAr   c                     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          |                    d#i           t                     r|                    d#i           ni }|                    d$d%          }	|                    d&d%          }
t          d'|	 d(|
 d)           t                       t          t          d*t          j        t          j                             |                     d+i           }t          d,|                    d-d.                      t          d/|                    d0d1                      t          d2|                    d3d4           d5           |                    d-          d6k    r(t          d7|                    d8d9                      n|                    d-          d:k    r(t          d;|                    d<d=                      n|                    d-          d>k    rMt          d?|                    d@d9                      t          dA          }t          dB|rdCndD            nT|                    d-          dEk    rLt          dF|                    dGd9                      t          dH          }t          dI|rdCndD            n|                    d-          dJk    rwt          dK|                    dLdM                      t          dNt          dO          s-t          dP          r t          dQ          rt          dR          rdCndD            n_|                    d-          dSk    rFt          dT          }t          dU          }t          dV|pdD            t          dW|pdD            t                       t          t          dXt          j        t          j                             |                     dYdZ          }|rt          d[|            n*t          d[t          d\t          j                              t                       t          t          d]t          j        t          j                             |                     d^i           }|                    d_d`          }t          da|rdbndc            |rt          dd|                    dedf          dgz  dhdi           t          dj|                    dkdl          dgz  dhdm           t          dn|                    dodp           dq           |                     dri                               d^i           }|                    ddZ          pds}t          d|            |                    dtdu          }|r|duk    rt          dv|            |                     dri           }|                    dwi           |                    dxi           dy}t%          dz |                                D                       }|rt                       t          t          d{t          j        t          j                             |                                D ]\  }}|                    dtdu          }|                    ddZ          }|duk    s|rId|| g}|r|                    d}|            t          d
|d~d
d                    |                      t                       t          t          dt          j        t          j                             t          d          }t          d          }t          d|rdCnt          dt          j                              t          d|rdCnt          dt          j                              	 ddlm} m}!  |             }"|"r |!|"          }#t                       t          t          dt          j        t          j                             |"D ]}$|$d         }%|#                    |%dZ          }|$                    ddZ          }&|rt5          |          nt          dDt          j                  }'t          d
|%dd|' d
t          d|& dt          j                              n# t6          $ r Y nw xY wt                       t          t          dt          j                             t          t          dt          j                             t          t          dt          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)	)r9  
OpenRouter)r(  zOpenAI (STT/TTS))rm  Exa)rp  Parallel)r$  	Firecrawl)r.  Tavily)r%  Browserbase)ry  zBrowser Use)r'  FALrR  z<14rS  r   )get_anthropic_key	Anthropicu	   ◆ Modelz  Model:        r  znot setz  Max turns:    r  r  u   ◆ Displayr  z  Personality:  r  r  z  Reasoning:    r  Fonr  z  Bell:         r  r  r  rO  r  z  User preview: first z line(s), last z line(s)u   ◆ Terminalr  z  Backend:      r   r  z  Working dir:  r  r  z  Timeout:      r  r  rZ  r   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:      vercel_sandboxz  Vercel runtime: r'  r(  z  Vercel auth:    VERCEL_OIDC_TOKENVERCEL_TOKENVERCEL_PROJECT_IDVERCEL_TEAM_IDsshTERMINAL_SSH_HOSTTERMINAL_SSH_USERz  SSH host:     z  SSH user:     u   ◆ Timezoner  r   z  Timezone:     rt  u   ◆ Context CompressionrX  rC  Tz  Enabled:      rx   rp  z  Threshold:    r\  rY  d   z.0f%z  Target ratio: r]  rZ  z% of threshold preservedz  Protect last: r^  r[  z	 messagesrr  z(auto)rs  r
  z  Provider:     ry  rz  )VisionzWeb extractc              3   v   K   | ]4}|                     d d          dk    p|                     dd          V  5dS )rs  r
  r  r   Nr  )r  ts     r   r  zshow_config.<locals>.<genexpr>  s\         	
j&!!V+AquuWb/A/A     r   u    ◆ Auxiliary Models (overrides)r  r  12sr  u   ◆ Messaging Platformsr  r  z  Telegram:     znot configuredz  Discord:      )r  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)r  r   rt   rs   CYANBOLDr   r   r   r  r  hermes_cli.authr  r   r  r  r  r  r$  valuesr  r  r  r  r  r  r   r  )(r  r  env_keyr  r   r  anthropic_valuer  ump	ump_firstump_lastr  modal_tokendaytona_keyssh_hostssh_usertzrX  rC  	_aux_comp_smcomp_providerrr  	aux_taskshas_overrideslabeltask_cfgprovmdlr  telegram_tokendiscord_tokenr  r  
skill_varsresolvedr  r   r   display_vals(                                           r   show_configr  7  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Y5?Lbdf@g@gim5n5n
v'++,b
1
1
1tvCq))Iww|Q''H	
O9
O
OX
O
O
OPPP 
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	 	 $4	4	4M8<<0@(#K#KMMNNN  W=AT3U3U  $UZghvZwZw  $U  }J  K^  }_  }_  $U  dq  rB  dC  dC  $U<<  JU  W  W  	X  	X  	X  	X	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   C8o< <
p	p	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:rR  zOpening  in z...)r   r   r   r   r  r  r   r   r   shutilwhich
subprocessrunr   )r3  editorcmdr  s       r   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,,-.....r   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
v rd}nu|                                dv rd}n\|                                rt!          |          }n8|                    ddd                                          rt%          |          }t'          || |           t)                       ddlm}  |||d           i 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/d0d1d2d3d4d5}| |v r#t          ||          t/          |                     t          d|  d6| d|            dS )7zSet a configuration value.zset configuration valuesN)r9  r   r-   r(  rm  rp  r$  rq  rr  rs  rt  ru  r.  r%  r&  ry  r'  r  r  r  r  r*   r  r+  r,  r  r  r  r}  )_API_KEY_TOKENTERMINAL_SSHu   ✓ Set r  r   r   )rv   rx   r  T)rn  rp  r  Fr  r   r*  r   r>  )	sort_keyszterminal.backendr   zterminal.modal_moder/  zterminal.docker_imageTERMINAL_DOCKER_IMAGEzterminal.singularity_imageTERMINAL_SINGULARITY_IMAGEzterminal.modal_imageTERMINAL_MODAL_IMAGEzterminal.daytona_imageTERMINAL_DAYTONA_IMAGEzterminal.vercel_runtimeTERMINAL_VERCEL_RUNTIMEz&terminal.docker_mount_cwd_to_workspace&TERMINAL_DOCKER_MOUNT_CWD_TO_WORKSPACEz terminal.docker_run_as_host_user TERMINAL_DOCKER_RUN_AS_HOST_USERzterminal.cwdr]  zterminal.timeoutTERMINAL_TIMEOUTzterminal.sandbox_dirTERMINAL_SANDBOX_DIRzterminal.persistent_shellTERMINAL_PERSISTENT_SHELLzterminal.container_cpuTERMINAL_CONTAINER_CPUzterminal.container_memoryTERMINAL_CONTAINER_MEMORYzterminal.container_diskTERMINAL_CONTAINER_DISKzterminal.container_persistentTERMINAL_CONTAINER_PERSISTENTr  )r   r   upperr  r   r  r   r   r   r   r   r  r  r  r   isdigitr   r  r  r  r   rB  r?  r   )r   r   api_keysr3  r:  r   r?  _config_to_env_syncs           r   set_config_valuer     s   || 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	 {{}}---	0	0	0	 E

	sB	"	"	*	*	,	, eS%((( ''''''k;%@@@@N4 	 !8 	%&B	
 	 6 	!": 	"#< 	12Z 	+,N 	 	. 	 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editr  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     r   r  z"config_command.<locals>.<listcomp>  s)    KKK!aeeM6J6JKAKKKr   c                 f    g | ].}|                     d           |                     d          ,|/S )r  r8  r  r  s     r   r  z"config_command.<locals>.<listcomp>  sN     
 
 
55''
01j0A0A

 
 
r   u   
  ⚠️  z required API key(s) missing:u	        • r  u   
  ℹ️  z$ optional API key(s) not configured:r<  z (enables: r  rO  rF  r   T)rg  rh  rj  rk  u   ✓ Configuration updated!rl  u
     ⚠️  checku   📋 Configuration Statusu    ✓z (update available)z  Required:u       ✓ u       ✗ z
 (missing)z  Optional:u       ○ rR  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   rt   rs   r  r  r  r  r8  GREENr  r   r  r  YELLOWr  r  REDr  r  r  )argssubcmdr   r   r  r  r  r  required_missingoptional_missingr  r<  	tools_strr  r  r  r   s                    r   r  r  Y  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?@@@=>>>r   )r   )r   )Fr  )NN)TF)urB  r  r  r   platformr  r0  r   r   rs  dataclassesr   pathlibr   typingr   r   r   r   r	   r  r  r
  systemrM  compiler  r   r   rC  r   r   r   	frozensetr\  r  hermes_cli.colorsrs   rt   hermes_cli.default_soulru   r   r   r   rD  r   r   r   r   r   r  r   r   r   rB  r   r   r   r   r   r   r   r   r   r   r  r2  r  r  r  r  r  r  r  r#  r.  r5  r8  rI  _VALID_CUSTOM_PROVIDER_FIELDSrG  r<  rP  rZ  rf  r  r  r  r  r  r'  r*  r.  r  r  rC  rE  _COMMENTED_SECTIONSr  rT  r  rO  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r  r   r   r   <module>r:     s+       				  				      



  ! ! ! ! ! !       3 3 3 3 3 3 3 3 3 3 3 3 3 3		8	$	$ho9,2:9:: 02 tCH~ 2 2 2 BD DeCd38n$<==> C C C AC 4U3T#s(^#;<<= B B B ) 1 1 1 1 1d  + + + + + + + + 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 " " " "4MRMM "M !"	M
 M 
   "%  !' $' $' ,0 #SJ JMd  N7NfN 	sN 	3	N 	2N* 	B+ND 	dENF 	DGNH 	bINT 	bUNV 	RWNX 	CYNZ 	E[N\ 	(]N` 	aNb 	DcNd 	%eNf !%  */ $)
 ![N N NeMD ! #'+
 (
 "5	
 EMt    uM^ 7_M~  MP  "!"&'
 
 !"&'
 
 QMn &) oMB TCMX ! YMf ! $
 
 %'!#&-
 
 gMV	  "
 
 
 
 
 
  
 
 
 
 
 
 
 
 
 
 
 
QP PW	M Mz  -5-x- 	&- 	;	- 	!%- 	E- 	%- 	U- 	"7- 	- 	U-  		!-& 		'-( 	!
 !
)-0 	%d1-2 	 3-4 	"25-6  ! !" 555
 
S- - -{MZ [Md eeMv 
'

 /0
 

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

 3	
 
 *	
?, ,wMR 

 

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