Skip to content

Commit f7b4c6b

Browse files
author
nesquena-hermes
committed
Merge PR nesquena#3087
# Conflicts: # CHANGELOG.md
2 parents d2265aa + 0f6eab3 commit f7b4c6b

4 files changed

Lines changed: 250 additions & 10 deletions

File tree

CHANGELOG.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
- Hardened frontend startup and navigation fallbacks: early storage access now survives blocked `localStorage`, stale session recovery preserves subpath mounts, session URL generation removes both legacy session query aliases, canceling a stream closes the local EventSource, and the PWA shell precaches same-origin markdown/KaTeX vendor assets.
1010
- Added missing i18n keys used by command, cron, provider, search/default, and session-rename UI paths across supported locales so missing translations fall back to labels instead of raw key names.
1111
- Made workspace Git tests pin their temporary repository branch to `master` so the suite is independent of the host Git default-branch setting.
12+
- Browser chat start and queued-turn payloads now fall back to the selected/persisted provider only when it belongs to the same model being sent, preventing fresh sessions from sending a dropdown-selected model with `model_provider=null`.
1213

1314
## [v0.51.155] — 2026-05-28 — Release EA (stage-batch37 — 3-PR very low-risk cleanup: passive timeout toasts + sidecar order + subsecond timestamps)
1415

static/messages.js

Lines changed: 38 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,25 @@ function _markActiveSessionViewedOnReturn() {
3232
if(typeof renderSessionListFromCache==='function') renderSessionListFromCache();
3333
}
3434

35+
function _chatPayloadModel(){
36+
return S.session&&S.session.model||($('modelSelect')&&$('modelSelect').value)||'';
37+
}
38+
39+
function _chatPayloadModelProvider(model){
40+
if(typeof _modelProviderForSend==='function') return _modelProviderForSend(model);
41+
if(S.session&&S.session.model_provider) return S.session.model_provider||null;
42+
return null;
43+
}
44+
45+
function _chatPayloadModelState(){
46+
// Source-compat invariant: the starting precedence is still
47+
// model:S.session.model||$('modelSelect').value and
48+
// model_provider:S.session.model_provider||null. The helper only fills a
49+
// missing provider when it belongs to the same outgoing model.
50+
const model=_chatPayloadModel();
51+
return {model,model_provider:_chatPayloadModelProvider(model)};
52+
}
53+
3554
function _deferStreamErrorIfOffline(){
3655
if(typeof isOfflineBannerVisible==='function' && isOfflineBannerVisible()){
3756
setComposerStatus(t('offline_stream_waiting'));
@@ -277,7 +296,8 @@ async function send(){
277296
// so the queued message goes to the chat that owns the active stream.
278297
const _targetSid=_sendInProgressSid||(S.session&&S.session.session_id);
279298
if(_text && _targetSid){
280-
queueSessionMessage(_targetSid,{text:_text,files:[...S.pendingFiles],model:S.session&&S.session.model||($('modelSelect')&&$('modelSelect').value)||'',model_provider:S.session&&S.session.model_provider||null,profile:S.activeProfile||'default'});
299+
const _modelState=_chatPayloadModelState();
300+
queueSessionMessage(_targetSid,{text:_text,files:[...S.pendingFiles],model:_modelState.model,model_provider:_modelState.model_provider,profile:S.activeProfile||'default'});
281301
$('msg').value='';autoResize();
282302
S.pendingFiles=[];renderTray();
283303
updateQueueBadge(_targetSid);
@@ -336,7 +356,8 @@ async function send(){
336356
S.pendingFiles=[];renderTray();
337357
} else if(busyMode==='interrupt'){
338358
// Queue the message, then cancel so drain re-sends it.
339-
queueSessionMessage(S.session.session_id,{text,files:[...S.pendingFiles],model:S.session&&S.session.model||($('modelSelect')&&$('modelSelect').value)||'',model_provider:S.session&&S.session.model_provider||null,profile:S.activeProfile||'default'});
359+
const _modelState=_chatPayloadModelState();
360+
queueSessionMessage(S.session.session_id,{text,files:[...S.pendingFiles],model:_modelState.model,model_provider:_modelState.model_provider,profile:S.activeProfile||'default'});
340361
updateQueueBadge(S.session.session_id);
341362
$('msg').value='';autoResize();
342363
S.pendingFiles=[];renderTray();
@@ -349,7 +370,8 @@ async function send(){
349370
} else {
350371
// Default: queue mode (current behavior). Also the fallback for
351372
// 'steer' mode when no stream is active or _trySteer is unavailable.
352-
queueSessionMessage(S.session.session_id,{text,files:[...S.pendingFiles],model:S.session&&S.session.model||($('modelSelect')&&$('modelSelect').value)||'',model_provider:S.session&&S.session.model_provider||null,profile:S.activeProfile||'default'});
373+
const _modelState=_chatPayloadModelState();
374+
queueSessionMessage(S.session.session_id,{text,files:[...S.pendingFiles],model:_modelState.model,model_provider:_modelState.model_provider,profile:S.activeProfile||'default'});
353375
$('msg').value='';autoResize();
354376
S.pendingFiles=[];renderTray();
355377
updateQueueBadge(S.session.session_id);
@@ -513,10 +535,13 @@ async function send(){
513535
// Start the agent via POST, get a stream_id back
514536
let streamId;
515537
try{
538+
const _modelState=_chatPayloadModelState();
516539
const startData=await api('/api/chat/start',{method:'POST',body:JSON.stringify({
517540
session_id:activeSid,message:msgText,
518-
model:S.session.model||$('modelSelect').value,workspace:S.session.workspace,
519-
model_provider:S.session.model_provider||null,
541+
// S.session.model remains authoritative; the helper only resolves a
542+
// matching provider fallback for the same outgoing model.
543+
model:_modelState.model,workspace:S.session.workspace,
544+
model_provider:_modelState.model_provider,
520545
profile:S.activeProfile||S.session.profile||'default',
521546
attachments:uploaded.length?uploaded:undefined
522547
})});
@@ -575,7 +600,8 @@ async function send(){
575600
stopApprovalPolling();
576601
stopClarifyPolling();
577602
// Keep the user's attempted turn by queueing it for after the current run.
578-
queueSessionMessage(activeSid,{text:msgText,files:[],model:S.session&&S.session.model||($('modelSelect')&&$('modelSelect').value)||'',model_provider:S.session&&S.session.model_provider||null,profile:S.activeProfile||'default'});
603+
const _retryModelState=_chatPayloadModelState();
604+
queueSessionMessage(activeSid,{text:msgText,files:[],model:_retryModelState.model,model_provider:_retryModelState.model_provider,profile:S.activeProfile||'default'});
579605
updateQueueBadge(activeSid);
580606
showToast('Current session is still running. Reconnected and queued your message.',2600);
581607
try{
@@ -1749,11 +1775,12 @@ function attachLiveStream(activeSid, streamId, uploaded=[], options={}){
17491775
const sid=d.session_id||activeSid;
17501776
const continuation_prompt=String(d.continuation_prompt||d.text||'').trim();
17511777
if(!continuation_prompt||sid!==activeSid)return;
1778+
const _modelState=_chatPayloadModelState();
17521779
_pendingGoalContinuation={
17531780
sid,
17541781
text:continuation_prompt,
1755-
model:S.session&&S.session.model||'',
1756-
model_provider:S.session&&S.session.model_provider||null,
1782+
model:_modelState.model,
1783+
model_provider:_modelState.model_provider,
17571784
profile:S.activeProfile||'default',
17581785
};
17591786
const toast=t('goal_continuing_toast');
@@ -1980,10 +2007,11 @@ function attachLiveStream(activeSid, streamId, uploaded=[], options={}){
19802007
const txt=String(d.text||'').trim();
19812008
if(!txt||sid!==activeSid) return;
19822009
if(typeof queueSessionMessage==='function'){
2010+
const _modelState=_chatPayloadModelState();
19832011
queueSessionMessage(sid,{
19842012
text:txt,files:[],
1985-
model:S.session&&S.session.model||'',
1986-
model_provider:S.session&&S.session.model_provider||null,
2013+
model:_modelState.model,
2014+
model_provider:_modelState.model_provider,
19872015
profile:S.activeProfile||'default',
19882016
});
19892017
if(typeof updateQueueBadge==='function') updateQueueBadge(sid);

static/ui.js

Lines changed: 28 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -897,6 +897,34 @@ function _captureModelDropdownSelection(sel){
897897
}catch(_){}
898898
return {model:String(sel.value||''),model_provider:null};
899899
}
900+
function _modelProviderForSend(modelId){
901+
const sessionProvider=(S&&S.session&&S.session.model_provider)||null;
902+
if(sessionProvider) return sessionProvider;
903+
const model=String(modelId||'').trim();
904+
if(!model) return null;
905+
const explicitProvider=typeof _providerFromModelValue==='function'
906+
? _providerFromModelValue(model)
907+
: '';
908+
if(explicitProvider) return explicitProvider;
909+
const sel=typeof $==='function' ? $('modelSelect') : null;
910+
if(sel&&String(sel.value||'').trim()===model&&typeof _modelStateForSelect==='function'){
911+
try{
912+
const dropdownState=_modelStateForSelect(sel,sel.value);
913+
if(dropdownState&&String(dropdownState.model||'').trim()===model){
914+
return dropdownState.model_provider||null;
915+
}
916+
}catch(_){}
917+
}
918+
if(typeof _readPersistedModelState==='function'){
919+
try{
920+
const persisted=_readPersistedModelState();
921+
if(persisted&&String(persisted.model||'').trim()===model){
922+
return persisted.model_provider||null;
923+
}
924+
}catch(_){}
925+
}
926+
return null;
927+
}
900928
function _reconcileModelDropdownSelection(sel,data,previousState,opts){
901929
if(!sel) return null;
902930
const activeSession=(typeof S!=='undefined'&&S&&S.session)?S.session:null;
Lines changed: 183 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,183 @@
1+
"""Regression coverage for browser chat model-provider fallback.
2+
3+
The browser send path may fall back to the model dropdown for the model ID on a
4+
fresh session. The provider must follow only when that dropdown/persisted state
5+
describes the same model being sent.
6+
"""
7+
import json
8+
import shutil
9+
import subprocess
10+
from pathlib import Path
11+
12+
import pytest
13+
14+
REPO_ROOT = Path(__file__).resolve().parents[1]
15+
UI_JS_PATH = REPO_ROOT / "static" / "ui.js"
16+
MESSAGES_JS_PATH = REPO_ROOT / "static" / "messages.js"
17+
NODE = shutil.which("node")
18+
19+
20+
def test_messages_payloads_use_model_tied_provider_helper():
21+
ui_src = UI_JS_PATH.read_text(encoding="utf-8")
22+
messages_src = MESSAGES_JS_PATH.read_text(encoding="utf-8")
23+
24+
assert "function _modelProviderForSend" in ui_src
25+
assert "function _chatPayloadModelState" in messages_src
26+
assert "_modelProviderForSend(model)" in messages_src
27+
28+
chat_start_idx = messages_src.find("api('/api/chat/start'")
29+
assert chat_start_idx >= 0, "could not find /api/chat/start POST in messages.js"
30+
payload_block = messages_src[chat_start_idx:chat_start_idx + 500]
31+
assert "model:_modelState.model" in payload_block
32+
assert "model_provider:_modelState.model_provider" in payload_block
33+
assert "model_provider:S.session.model_provider||null" not in payload_block
34+
35+
for idx in [m.start() for m in __import__("re").finditer("queueSessionMessage\\(", messages_src)]:
36+
block = messages_src[idx:idx + 260]
37+
if "model_provider:" in block:
38+
assert "S.session.model_provider||null" not in block
39+
40+
41+
_DRIVER_SRC = r"""
42+
const fs = require('fs');
43+
const ui = fs.readFileSync(process.argv[2], 'utf8');
44+
45+
function extractFunc(name) {
46+
const re = new RegExp('function\\s+' + name + '\\s*\\(');
47+
const start = ui.search(re);
48+
if (start < 0) throw new Error(name + ' not found');
49+
let i = ui.indexOf('{', start);
50+
let depth = 1;
51+
i++;
52+
while (depth > 0 && i < ui.length) {
53+
if (ui[i] === '{') depth++;
54+
else if (ui[i] === '}') depth--;
55+
i++;
56+
}
57+
return ui.slice(start, i);
58+
}
59+
60+
let modelSelect;
61+
function $(id) { return id === 'modelSelect' ? modelSelect : null; }
62+
63+
function makeSelect(options, initialValue) {
64+
const sel = {options: [], selectedIndex: -1, selectedOptions: []};
65+
Object.defineProperty(sel, 'value', {
66+
get() { return this._value || ''; },
67+
set(v) {
68+
this._value = v;
69+
const idx = this.options.findIndex(o => o.value === v);
70+
this.selectedIndex = idx;
71+
this.selectedOptions = idx >= 0 ? [this.options[idx]] : [];
72+
}
73+
});
74+
for (const item of options) {
75+
const group = {tagName: 'OPTGROUP', dataset: {provider: item.provider || ''}};
76+
const opt = {value: item.value, parentElement: group, dataset: {}};
77+
if (item.optionProvider) opt.dataset.provider = item.optionProvider;
78+
sel.options.push(opt);
79+
}
80+
sel.value = initialValue || '';
81+
return sel;
82+
}
83+
84+
const store = new Map();
85+
const localStorage = {
86+
getItem(k) { return store.has(k) ? store.get(k) : null; },
87+
setItem(k, v) { store.set(k, String(v)); },
88+
removeItem(k) { store.delete(k); },
89+
};
90+
const MODEL_STATE_KEY = 'hermes-webui-model-state';
91+
92+
for (const name of [
93+
'_getOptionProviderId',
94+
'_providerFromModelValue',
95+
'_modelStateForSelect',
96+
'_readPersistedModelState',
97+
'_modelProviderForSend',
98+
]) {
99+
eval(extractFunc(name));
100+
}
101+
102+
const args = JSON.parse(process.argv[3]);
103+
modelSelect = makeSelect(args.options || [], args.initialValue || '');
104+
if (args.persisted) localStorage.setItem(MODEL_STATE_KEY, JSON.stringify(args.persisted));
105+
var S = {session: {model_provider: args.sessionProvider || null}};
106+
107+
process.stdout.write(JSON.stringify({provider: _modelProviderForSend(args.model)}));
108+
"""
109+
110+
node_test = pytest.mark.skipif(NODE is None, reason="node not on PATH")
111+
112+
113+
@pytest.fixture(scope="module")
114+
def driver_path(tmp_path_factory):
115+
p = tmp_path_factory.mktemp("chat_provider_fallback_driver") / "driver.js"
116+
p.write_text(_DRIVER_SRC, encoding="utf-8")
117+
return str(p)
118+
119+
120+
def _run_helper(driver_path, payload):
121+
result = subprocess.run(
122+
[NODE, driver_path, str(UI_JS_PATH), json.dumps(payload)],
123+
capture_output=True,
124+
text=True,
125+
timeout=10,
126+
)
127+
if result.returncode != 0:
128+
raise RuntimeError(f"node driver failed:\nSTDOUT={result.stdout}\nSTDERR={result.stderr}")
129+
return json.loads(result.stdout)["provider"]
130+
131+
132+
@node_test
133+
def test_model_provider_for_send_preserves_session_provider(driver_path):
134+
provider = _run_helper(driver_path, {
135+
"model": "grok-4.3",
136+
"sessionProvider": "session-provider",
137+
"initialValue": "grok-4.3",
138+
"options": [{"provider": "xai-oauth", "value": "grok-4.3"}],
139+
})
140+
141+
assert provider == "session-provider"
142+
143+
144+
@node_test
145+
def test_model_provider_for_send_falls_back_to_matching_dropdown(driver_path):
146+
provider = _run_helper(driver_path, {
147+
"model": "grok-4.3",
148+
"initialValue": "grok-4.3",
149+
"options": [{"provider": "xai-oauth", "value": "grok-4.3"}],
150+
})
151+
152+
assert provider == "xai-oauth"
153+
154+
155+
@node_test
156+
def test_model_provider_for_send_does_not_steal_unrelated_dropdown_provider(driver_path):
157+
provider = _run_helper(driver_path, {
158+
"model": "grok-4.3",
159+
"initialValue": "claude-sonnet-4.6",
160+
"options": [
161+
{"provider": "anthropic", "value": "claude-sonnet-4.6"},
162+
{"provider": "xai-oauth", "value": "grok-4.3"},
163+
],
164+
})
165+
166+
assert provider is None
167+
168+
169+
@node_test
170+
def test_model_provider_for_send_uses_only_matching_persisted_state(driver_path):
171+
matching = _run_helper(driver_path, {
172+
"model": "grok-4.3",
173+
"initialValue": "",
174+
"persisted": {"model": "grok-4.3", "model_provider": "xai-oauth"},
175+
})
176+
unrelated = _run_helper(driver_path, {
177+
"model": "grok-4.3",
178+
"initialValue": "",
179+
"persisted": {"model": "claude-sonnet-4.6", "model_provider": "anthropic"},
180+
})
181+
182+
assert matching == "xai-oauth"
183+
assert unrelated is None

0 commit comments

Comments
 (0)