-
Notifications
You must be signed in to change notification settings - Fork 109
Expand file tree
/
Copy pathtest_local_server.py
More file actions
174 lines (135 loc) · 5.68 KB
/
test_local_server.py
File metadata and controls
174 lines (135 loc) · 5.68 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
from __future__ import annotations
import json
import httpx
import pytest
from respx import MockRouter
from stagehand import Stagehand, AsyncStagehand
from stagehand._exceptions import StagehandError
class _DummySeaServer:
def __init__(self, base_url: str) -> None:
self._base_url = base_url
self.started = 0
self.closed = 0
def ensure_running_sync(self) -> str:
self.started += 1
return self._base_url
async def ensure_running_async(self) -> str:
self.started += 1
return self._base_url
def close(self) -> None:
self.closed += 1
async def aclose(self) -> None:
self.closed += 1
def _set_required_env(monkeypatch: pytest.MonkeyPatch) -> None:
monkeypatch.setenv("BROWSERBASE_API_KEY", "bb_key")
monkeypatch.setenv("BROWSERBASE_PROJECT_ID", "bb_project")
monkeypatch.setenv("MODEL_API_KEY", "model_key")
@pytest.mark.respx(base_url="http://127.0.0.1:43123")
def test_sync_local_mode_starts_before_first_request(respx_mock: MockRouter, monkeypatch: pytest.MonkeyPatch) -> None:
_set_required_env(monkeypatch)
dummy = _DummySeaServer("http://127.0.0.1:43123")
respx_mock.post("/v1/sessions/start").mock(
return_value=httpx.Response(
200,
json={
"success": True,
"data": {
"available": True,
"connectUrl": "ws://example",
"sessionId": "00000000-0000-0000-0000-000000000000",
},
},
)
)
client = Stagehand(server="local", _local_stagehand_binary_path="/does/not/matter/in/test")
# Swap in a dummy server so we don't spawn a real binary in unit tests.
client._sea_server = dummy # type: ignore[attr-defined]
resp = client.sessions.start(model_name="openai/gpt-5-nano")
assert resp.success is True
assert dummy.started == 1
client.close()
assert dummy.closed == 1
@pytest.mark.respx(base_url="http://127.0.0.1:43127")
def test_sync_local_mode_defaults_browser_type_local_when_omitted(
respx_mock: MockRouter, monkeypatch: pytest.MonkeyPatch
) -> None:
_set_required_env(monkeypatch)
dummy = _DummySeaServer("http://127.0.0.1:43127")
request_body: dict[str, object] = {}
def _capture_start_request(request: httpx.Request) -> httpx.Response:
request_body["json"] = json.loads(request.content.decode("utf-8"))
return httpx.Response(
200,
json={
"success": True,
"data": {
"available": True,
"connectUrl": "ws://example",
"sessionId": "00000000-0000-0000-0000-000000000001",
},
},
)
respx_mock.post("/v1/sessions/start").mock(side_effect=_capture_start_request)
client = Stagehand(server="local", _local_stagehand_binary_path="/does/not/matter/in/test")
client._sea_server = dummy # type: ignore[attr-defined]
resp = client.sessions.start(model_name="openai/gpt-5-nano")
assert resp.success is True
assert request_body["json"] == {
"modelName": "openai/gpt-5-nano",
"browser": {"type": "local"},
}
client.close()
assert dummy.closed == 1
@pytest.mark.respx(base_url="http://127.0.0.1:43124")
@pytest.mark.asyncio
async def test_async_local_mode_starts_before_first_request(
respx_mock: MockRouter, monkeypatch: pytest.MonkeyPatch
) -> None:
_set_required_env(monkeypatch)
dummy = _DummySeaServer("http://127.0.0.1:43124")
respx_mock.post("/v1/sessions/start").mock(
return_value=httpx.Response(
200,
json={
"success": True,
"data": {
"available": True,
"connectUrl": "ws://example",
"sessionId": "00000000-0000-0000-0000-000000000000",
},
},
)
)
async with AsyncStagehand(server="local", _local_stagehand_binary_path="/does/not/matter/in/test") as client:
client._sea_server = dummy # type: ignore[attr-defined]
resp = await client.sessions.start(model_name="openai/gpt-5-nano")
assert resp.success is True
assert dummy.started == 1
assert dummy.closed == 1
def test_local_server_requires_browserbase_keys_for_browserbase_sessions(
monkeypatch: pytest.MonkeyPatch,
) -> None:
_set_required_env(monkeypatch)
monkeypatch.delenv("BROWSERBASE_API_KEY", raising=False)
monkeypatch.delenv("BROWSERBASE_PROJECT_ID", raising=False)
client = Stagehand(server="local", _local_stagehand_binary_path="/does/not/matter/in/test")
client._sea_server = _DummySeaServer("http://127.0.0.1:43125") # type: ignore[attr-defined]
with pytest.raises(StagehandError):
client.sessions.start(model_name="openai/gpt-5-nano")
def test_local_server_allows_local_browser_without_browserbase_keys(
monkeypatch: pytest.MonkeyPatch,
) -> None:
_set_required_env(monkeypatch)
monkeypatch.delenv("BROWSERBASE_API_KEY", raising=False)
monkeypatch.delenv("BROWSERBASE_PROJECT_ID", raising=False)
client = Stagehand(server="local", _local_stagehand_binary_path="/does/not/matter/in/test")
client._sea_server = _DummySeaServer("http://127.0.0.1:43126") # type: ignore[attr-defined]
def _post(*_args: object, **_kwargs: object) -> object:
raise RuntimeError("post called")
client.sessions._post = _post # type: ignore[method-assign]
client.base_url = httpx.URL("http://127.0.0.1:43126")
with pytest.raises(RuntimeError, match="post called"):
client.sessions.start(
model_name="openai/gpt-5-nano",
browser={"type": "local"},
)