unrealon 2.0.8__py3-none-any.whl → 2.0.10__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: unrealon
3
- Version: 2.0.8
3
+ Version: 2.0.10
4
4
  Summary: Enterprise-grade web scraping platform with AI-powered automation and real-time orchestration capabilities
5
5
  Author-email: UnrealOn Team <team@unrealon.com>
6
6
  License: MIT
@@ -6,7 +6,7 @@ unrealon_browser/cli/cookies_cli.py,sha256=yhZvGrg8bknlH4zlySdi8ue-25Ue-1rI_u1G0
6
6
  unrealon_browser/cli/interactive_mode.py,sha256=gLn9bMH0h0tPX3dP4i4QQxQK4Htkyg5r4KcqdMBaP6Q,12125
7
7
  unrealon_browser/cli/main.py,sha256=XCYcTxJUqaz320KCU_JPKizYMk6bdljb8Boyok3uO-4,1353
8
8
  unrealon_browser/core/__init__.py,sha256=uVL_t4sZelUzflWPdgrwoXGnAkSV1WNQ98-eu0QB2eM,151
9
- unrealon_browser/core/browser_manager.py,sha256=9xxo0kqbUcssFBNpvIXavWBya2E3TmgHKplQQ2kkZeU,29598
9
+ unrealon_browser/core/browser_manager.py,sha256=G7jZ9490j4yAsfW-VM1QXyRJGCsESX4dgaSO4-5EHko,29864
10
10
  unrealon_browser/dto/__init__.py,sha256=bApqcLz-KanEi0_MCiFPrQmGBoX3VBijP7XtBUyIfjo,1636
11
11
  unrealon_browser/dto/bot_detection.py,sha256=qXfC0HghV7m4L6qA87t3STi-166jM-QgoP6OYbCb4o4,6884
12
12
  unrealon_browser/dto/models/config.py,sha256=Why5H3rtFclmwbdczuDfhlgf-LDz72Aa8LhDX4_ayfw,1752
@@ -15,15 +15,16 @@ unrealon_browser/dto/models/dataclasses.py,sha256=zqhJVyzp4CvtuTBsZwm6n6TodVWrZf
15
15
  unrealon_browser/dto/models/detection.py,sha256=ma9ZNIjPR7HnjqZaAj6ZoskiewPFiSn_FgFXSkgiQc8,2715
16
16
  unrealon_browser/dto/models/enums.py,sha256=Q4WzHdfSKf7dhKyX00i_Pvl2U8w3lBsxOYfSIoaQY3Q,1219
17
17
  unrealon_browser/dto/models/statistics.py,sha256=aIzJNV5r23VBxjhEoja4tXwI1Z7_UCw5zOaxuPya2E8,2728
18
- unrealon_browser/managers/__init__.py,sha256=WMzrYGrCXn467XWBr8msrvT43ckX_EQSe9Ye9cfMozg,528
18
+ unrealon_browser/managers/__init__.py,sha256=lpa93ggEN93ucoi4FqnCG_sn-_aRlP1As7DBRogDSsQ,591
19
19
  unrealon_browser/managers/captcha.py,sha256=KGBO7sfq9XusAlcPByUFdIg-v6rlruzS2oHx-Zx28wo,21453
20
20
  unrealon_browser/managers/cookies.py,sha256=r4VVnKLXH82vhU7qgtY-dF7KPf0Ie3QxGD3FEi6geFA,15085
21
21
  unrealon_browser/managers/logger_bridge.py,sha256=I3KNUmxSV9xL1JHpc6WOVCzn_kLlOxJzfZy76rsxr48,10912
22
22
  unrealon_browser/managers/page_wait_manager.py,sha256=UyZqiSfkjzahrxp9x1odXFIT_sFhZGvdECxWuIMCVBY,7876
23
23
  unrealon_browser/managers/profile.py,sha256=HjddlSeUry_65WPtF8CMkT7cfJ6X3Jap9kJaaZpwtAA,18956
24
+ unrealon_browser/managers/script_manager.py,sha256=z2TxYKlhXIsXNz6bCStLF7cxthBOZEESkhoFI1TCu_Y,11453
24
25
  unrealon_browser/stealth/__init__.py,sha256=zUfkPPafYlPANLVQIy-Se11R_UjcJakUb3krCxxUK5Q,842
25
26
  unrealon_browser/stealth/bypass_techniques.pyc,sha256=Tys_I4tnJmL9aQLB1k1mL-4OtzedhpYTmW4XPEIb3cI,25790
26
- unrealon_browser/stealth/manager.pyc,sha256=prGC5fWE-T4ROGSVrc65CJ_qFpb8dScuN83FotZ1oj4,21183
27
+ unrealon_browser/stealth/manager.pyc,sha256=snye2EmZWo36PRy-dG9MrB837gVhVRjt6MgawA1mAF4,21540
27
28
  unrealon_browser/stealth/nodriver_stealth.pyc,sha256=SnDMdeG_W2LDK_3UfrjDeolMpqxTMDNjRbId_YC_cKA,15286
28
29
  unrealon_browser/stealth/playwright_stealth.pyc,sha256=Y-IiRTu136R6LIstGdKbns1j-CODX3sTGVT0IGx_MF4,6529
29
30
  unrealon_browser/stealth/scanner_tester.pyc,sha256=ki_Lp7zBp12iWgj3LGPnXTwemwgaGG1UrWeM1IXB6Gs,21711
@@ -32,7 +33,7 @@ unrealon_core/__init__.py,sha256=ZAinyQDsAS63xqnl5uqUKTna9M-xFDs6fLwy8-Hh-X0,559
32
33
  unrealon_core/version.py,sha256=qZOlKA_Hsz7_KXsCLO_0l9Mf0u_iNTxvHZPV21DwCqs,5803
33
34
  unrealon_core/config/__init__.py,sha256=57-KZaTDya0oyfOCfZ3pU1xLLbnZPBgqwfP9VrfhcKE,395
34
35
  unrealon_core/config/environment.py,sha256=TACbyjr3lxrA9R_Ve0LLqyLnRk3FShm2imQjhiP9i98,4589
35
- unrealon_core/config/urls.py,sha256=7ScpWEhSu1kewhMaKhuqgiMHIv_fyI1TyRiS-f2wrLo,2821
36
+ unrealon_core/config/urls.py,sha256=FjdXI5WJzdveNCYS8FryRQX9gxjSSH-Qu30dT-USUts,2869
36
37
  unrealon_core/enums/__init__.py,sha256=wOH3DK3E1xUO3jD4Cu0UFzTL3HaHq9UEdwP70YdnyI0,736
37
38
  unrealon_core/enums/events.py,sha256=rlidf5M2-EDZCRLv4tBkBQ0--qsrgKFt5WrAgw7FRGg,2704
38
39
  unrealon_core/enums/jobs.py,sha256=sTSjhgnwE9O6tkgecYSI2BFxECEW0Ro5lkPF1oiEy40,787
@@ -114,7 +115,7 @@ unrealon_driver/driver/utilities/logging.py,sha256=2my2QnkAa6Hdw-TfO4oOQ94yGc-Cj
114
115
  unrealon_driver/driver/utilities/serialization.py,sha256=wTCSVrEloykiGN4K1JXbk2aqNKm7W90aWXmzhcLyAZc,2123
115
116
  unrealon_driver/managers/__init__.py,sha256=zJJsOb6Oodg7l00v4ncKUytnyeaZM887pHY8-eSuWdU,981
116
117
  unrealon_driver/managers/base.py,sha256=GkuXillg9uqqnx6RL682fmKgK-7JyqYlH6DFUgyN4F8,5445
117
- unrealon_driver/managers/browser.py,sha256=9kwdmWPxWFpNQ1KejtpzdN615V8PBJu8Y2nUOewQaeQ,3489
118
+ unrealon_driver/managers/browser.py,sha256=v2FjgtgzTM94Qobll9_7TGRVbxsMlVvirP_sj3u152Q,4608
118
119
  unrealon_driver/managers/cache.py,sha256=c0tPKQ5KFd_Un1U8mw3j1WPuycxg863MMWNMveVF_2I,3506
119
120
  unrealon_driver/managers/http.py,sha256=EjlpoTRuhpsgzzrEARxRlbGczzua7hnKFVq06bvCgTM,3624
120
121
  unrealon_driver/managers/logger.py,sha256=PL3rA9ZQl12jJU0EiPAkLwJ6eDHQfIzr8-nc8bVivKQ,10526
@@ -124,9 +125,9 @@ unrealon_driver/managers/threading.py,sha256=djw5cSC99dfBKmep3IJ_8IgxQceMXtNvCp5
124
125
  unrealon_driver/managers/update.py,sha256=-hohVxGXpj5bZ6ZTQN6NH1RK9Pd6GVzCMtu3GS2SdcQ,3582
125
126
  unrealon_driver/utils/__init__.py,sha256=qxXVoQJVdLJhaLBXk_LZV_062AhrvBrMPXWAKfEc3C4,104
126
127
  unrealon_driver/utils/time.py,sha256=Oxk1eicKeZl8ZWbf7gu1Ll716k6CpXmVj67FHSnPIsA,184
127
- unrealon-2.0.8.dist-info/LICENSE,sha256=eEH8mWZW49YMpl4Sh5MtKqkZ8aVTzKQXiNPEnvL14ns,1070
128
- unrealon-2.0.8.dist-info/METADATA,sha256=sCydlGvYhaf2-tzj-0twQUib9nTG5LR-4bHPAqwuDWI,15688
129
- unrealon-2.0.8.dist-info/WHEEL,sha256=pL8R0wFFS65tNSRnaOVrsw9EOkOqxLrlUPenUYnJKNo,91
130
- unrealon-2.0.8.dist-info/entry_points.txt,sha256=k0qM-eotpajkKUq-almJmxj9afhXprZ6IkvQkSdcKhI,104
131
- unrealon-2.0.8.dist-info/top_level.txt,sha256=Gu8IeIfIVfUxdi-h-F0nKMQxo15pjhHZ0aTadXTpRE8,47
132
- unrealon-2.0.8.dist-info/RECORD,,
128
+ unrealon-2.0.10.dist-info/LICENSE,sha256=eEH8mWZW49YMpl4Sh5MtKqkZ8aVTzKQXiNPEnvL14ns,1070
129
+ unrealon-2.0.10.dist-info/METADATA,sha256=fatBlrWFUSdwc5LjMflYHXNVlalO65qX482hwKToNpE,15689
130
+ unrealon-2.0.10.dist-info/WHEEL,sha256=pL8R0wFFS65tNSRnaOVrsw9EOkOqxLrlUPenUYnJKNo,91
131
+ unrealon-2.0.10.dist-info/entry_points.txt,sha256=k0qM-eotpajkKUq-almJmxj9afhXprZ6IkvQkSdcKhI,104
132
+ unrealon-2.0.10.dist-info/top_level.txt,sha256=Gu8IeIfIVfUxdi-h-F0nKMQxo15pjhHZ0aTadXTpRE8,47
133
+ unrealon-2.0.10.dist-info/RECORD,,
@@ -30,6 +30,7 @@ from unrealon_browser.managers import (
30
30
  CaptchaDetector,
31
31
  create_browser_logger_bridge,
32
32
  PageWaitManager,
33
+ ScriptManager,
33
34
  )
34
35
 
35
36
 
@@ -66,6 +67,7 @@ class BrowserManager:
66
67
  self.cookie_manager = None
67
68
  self.captcha_manager = CaptchaDetector()
68
69
  self.page_wait = PageWaitManager(None, self.logger_bridge)
70
+ self.script_manager = ScriptManager(None, self.logger_bridge)
69
71
 
70
72
  # Signal handlers for graceful shutdown
71
73
  self._setup_signal_handlers()
@@ -231,6 +233,9 @@ class BrowserManager:
231
233
 
232
234
  # Update page wait manager with new page
233
235
  self.page_wait.update_page(self._page)
236
+
237
+ # Update script manager with new page
238
+ self.script_manager.update_page(self._page)
234
239
 
235
240
  # 🔥 STEALTH ALWAYS APPLIED TO EVERY PAGE!
236
241
  stealth_success = await self.stealth_manager.apply_stealth(self._page)
@@ -644,6 +649,7 @@ class BrowserManager:
644
649
  finally:
645
650
  self._page = None
646
651
  self.page_wait.update_page(None)
652
+ self.script_manager.update_page(None)
647
653
 
648
654
  # Close context with safety checks
649
655
  if self._context:
@@ -8,6 +8,7 @@ from .logger_bridge import BrowserLoggerBridge, create_browser_logger_bridge
8
8
  from .cookies import CookieManager
9
9
  from .captcha import CaptchaDetector
10
10
  from .page_wait_manager import PageWaitManager
11
+ from .script_manager import ScriptManager
11
12
 
12
13
 
13
14
  __all__ = [
@@ -18,4 +19,5 @@ __all__ = [
18
19
  "CookieManager",
19
20
  "CaptchaDetector",
20
21
  "PageWaitManager",
22
+ "ScriptManager",
21
23
  ]
@@ -0,0 +1,314 @@
1
+ """
2
+ Script Manager - JavaScript execution and evaluation manager
3
+ Layer 2.5: JavaScript Integration - Handles script execution, API calls, and result processing
4
+ """
5
+
6
+ import asyncio
7
+ import json
8
+ from typing import Any, Dict, Optional, Union, List
9
+ from datetime import datetime, timezone
10
+ from playwright.async_api import Page
11
+
12
+ from .logger_bridge import BrowserLoggerBridge as LoggingBridge
13
+
14
+
15
+ class ScriptManager:
16
+ """Manager for JavaScript execution and evaluation"""
17
+
18
+ def __init__(self, page: Optional[Page], logger_bridge: LoggingBridge):
19
+ self._page = page
20
+ self.logger_bridge = logger_bridge
21
+
22
+ # Statistics
23
+ self._scripts_executed = 0
24
+ self._scripts_successful = 0
25
+ self._scripts_failed = 0
26
+ self._api_calls_made = 0
27
+ self._execution_history: List[Dict[str, Any]] = []
28
+
29
+ def update_page(self, page: Optional[Page]):
30
+ """Update the page reference"""
31
+ self._page = page
32
+
33
+ async def execute_script(self, script: str, timeout: int = 30000) -> Any:
34
+ """
35
+ Execute JavaScript code and return result
36
+
37
+ Args:
38
+ script: JavaScript code to execute
39
+ timeout: Timeout in milliseconds
40
+
41
+ Returns:
42
+ Script execution result
43
+ """
44
+ if not self._page:
45
+ raise RuntimeError("No page available for script execution")
46
+
47
+ start_time = datetime.now()
48
+ self._scripts_executed += 1
49
+
50
+ try:
51
+ self.logger_bridge.log_info(f"🔧 Executing JavaScript (timeout: {timeout}ms)")
52
+ self.logger_bridge.log_debug(f"Script preview: {script[:100]}...")
53
+
54
+ # Execute script with timeout
55
+ result = await asyncio.wait_for(
56
+ self._page.evaluate(script),
57
+ timeout=timeout / 1000
58
+ )
59
+
60
+ duration_ms = (datetime.now() - start_time).total_seconds() * 1000
61
+ self._scripts_successful += 1
62
+
63
+ # Log execution details
64
+ execution_record = {
65
+ "timestamp": start_time.isoformat(),
66
+ "duration_ms": duration_ms,
67
+ "success": True,
68
+ "result_type": type(result).__name__,
69
+ "script_length": len(script),
70
+ }
71
+ self._execution_history.append(execution_record)
72
+
73
+ self.logger_bridge.log_info(f"✅ Script executed successfully ({duration_ms:.1f}ms)")
74
+ self.logger_bridge.log_debug(f"Result type: {type(result).__name__}")
75
+
76
+ return result
77
+
78
+ except asyncio.TimeoutError:
79
+ duration_ms = (datetime.now() - start_time).total_seconds() * 1000
80
+ self._scripts_failed += 1
81
+
82
+ execution_record = {
83
+ "timestamp": start_time.isoformat(),
84
+ "duration_ms": duration_ms,
85
+ "success": False,
86
+ "error": "Timeout",
87
+ "script_length": len(script),
88
+ }
89
+ self._execution_history.append(execution_record)
90
+
91
+ self.logger_bridge.log_error(f"⏰ Script execution timeout ({timeout}ms)")
92
+ raise
93
+
94
+ except Exception as e:
95
+ duration_ms = (datetime.now() - start_time).total_seconds() * 1000
96
+ self._scripts_failed += 1
97
+
98
+ execution_record = {
99
+ "timestamp": start_time.isoformat(),
100
+ "duration_ms": duration_ms,
101
+ "success": False,
102
+ "error": str(e),
103
+ "script_length": len(script),
104
+ }
105
+ self._execution_history.append(execution_record)
106
+
107
+ self.logger_bridge.log_error(f"❌ Script execution failed: {e}")
108
+ raise
109
+
110
+ async def execute_api_call(self, api_url: str, headers: Dict[str, str], method: str = "GET", timeout: int = 30000) -> Dict[str, Any]:
111
+ """
112
+ Execute API call via JavaScript fetch
113
+
114
+ Args:
115
+ api_url: API endpoint URL
116
+ headers: HTTP headers
117
+ method: HTTP method
118
+ timeout: Timeout in milliseconds
119
+
120
+ Returns:
121
+ API response data
122
+ """
123
+ self._api_calls_made += 1
124
+
125
+ # Build fetch script
126
+ headers_json = json.dumps(headers)
127
+
128
+ script = f"""
129
+ (async function() {{
130
+ try {{
131
+ const response = await fetch('{api_url}', {{
132
+ method: '{method}',
133
+ headers: {headers_json}
134
+ }});
135
+
136
+ if (!response.ok) {{
137
+ throw new Error('HTTP error! status: ' + response.status);
138
+ }}
139
+
140
+ const data = await response.json();
141
+ return data;
142
+ }} catch (error) {{
143
+ return {{ error: error.message }};
144
+ }}
145
+ }})()
146
+ """
147
+
148
+ self.logger_bridge.log_info(f"🌐 Making API call: {method} {api_url}")
149
+
150
+ result = await self.execute_script(script, timeout)
151
+
152
+ if isinstance(result, dict) and 'error' in result:
153
+ self.logger_bridge.log_error(f"❌ API call failed: {result['error']}")
154
+ else:
155
+ self.logger_bridge.log_info(f"✅ API call successful")
156
+
157
+ return result
158
+
159
+
160
+ async def wait_for_element(self, selector: str, timeout: int = 10000) -> bool:
161
+ """
162
+ Wait for element using JavaScript
163
+
164
+ Args:
165
+ selector: CSS selector
166
+ timeout: Timeout in milliseconds
167
+
168
+ Returns:
169
+ True if element found
170
+ """
171
+ script = f"""
172
+ (function() {{
173
+ return new Promise((resolve) => {{
174
+ const element = document.querySelector('{selector}');
175
+ if (element) {{
176
+ resolve(true);
177
+ return;
178
+ }}
179
+
180
+ const observer = new MutationObserver(() => {{
181
+ const element = document.querySelector('{selector}');
182
+ if (element) {{
183
+ observer.disconnect();
184
+ resolve(true);
185
+ }}
186
+ }});
187
+
188
+ observer.observe(document.body, {{
189
+ childList: true,
190
+ subtree: true
191
+ }});
192
+
193
+ setTimeout(() => {{
194
+ observer.disconnect();
195
+ resolve(false);
196
+ }}, {timeout});
197
+ }});
198
+ }})()
199
+ """
200
+
201
+ self.logger_bridge.log_info(f"🎯 Waiting for element: {selector}")
202
+
203
+ try:
204
+ result = await self.execute_script(script, timeout + 1000)
205
+ if result:
206
+ self.logger_bridge.log_info(f"✅ Element found: {selector}")
207
+ else:
208
+ self.logger_bridge.log_warning(f"⏰ Element timeout: {selector}")
209
+ return result
210
+ except Exception as e:
211
+ self.logger_bridge.log_error(f"❌ Element wait failed: {selector} - {e}")
212
+ return False
213
+
214
+ async def inject_helper_functions(self) -> bool:
215
+ """
216
+ Inject helper JavaScript functions into page
217
+
218
+ Returns:
219
+ True if injection successful
220
+ """
221
+ helper_script = """
222
+ window.unrealonHelpers = {
223
+ // Wait for element with promise
224
+ waitForElement: function(selector, timeout = 10000) {
225
+ return new Promise((resolve) => {
226
+ const element = document.querySelector(selector);
227
+ if (element) {
228
+ resolve(element);
229
+ return;
230
+ }
231
+
232
+ const observer = new MutationObserver(() => {
233
+ const element = document.querySelector(selector);
234
+ if (element) {
235
+ observer.disconnect();
236
+ resolve(element);
237
+ }
238
+ });
239
+
240
+ observer.observe(document.body, {
241
+ childList: true,
242
+ subtree: true
243
+ });
244
+
245
+ setTimeout(() => {
246
+ observer.disconnect();
247
+ resolve(null);
248
+ }, timeout);
249
+ });
250
+ },
251
+
252
+ // Get page info
253
+ getPageInfo: function() {
254
+ return {
255
+ url: window.location.href,
256
+ title: document.title,
257
+ readyState: document.readyState,
258
+ timestamp: new Date().toISOString()
259
+ };
260
+ },
261
+
262
+ // Check if SPA loaded
263
+ isSPAReady: function() {
264
+ return document.readyState === 'complete' &&
265
+ document.querySelector('body').children.length > 0;
266
+ }
267
+ };
268
+
269
+ console.log('🔧 UnrealOn helper functions injected');
270
+ """
271
+
272
+ try:
273
+ await self.execute_script(helper_script)
274
+ self.logger_bridge.log_info("🔧 Helper functions injected successfully")
275
+ return True
276
+ except Exception as e:
277
+ self.logger_bridge.log_error(f"❌ Failed to inject helper functions: {e}")
278
+ return False
279
+
280
+ def get_statistics(self) -> Dict[str, Any]:
281
+ """Get script execution statistics"""
282
+ success_rate = (self._scripts_successful / self._scripts_executed * 100) if self._scripts_executed > 0 else 0
283
+
284
+ return {
285
+ "scripts_executed": self._scripts_executed,
286
+ "scripts_successful": self._scripts_successful,
287
+ "scripts_failed": self._scripts_failed,
288
+ "success_rate": success_rate,
289
+ "api_calls_made": self._api_calls_made,
290
+ "execution_history_count": len(self._execution_history),
291
+ }
292
+
293
+ def print_statistics(self) -> None:
294
+ """Print script execution statistics"""
295
+ stats = self.get_statistics()
296
+
297
+ self.logger_bridge.log_info("\n🔧 Script Manager Statistics:")
298
+ self.logger_bridge.log_info(f" Scripts executed: {stats['scripts_executed']}")
299
+ self.logger_bridge.log_info(f" Successful: {stats['scripts_successful']}")
300
+ self.logger_bridge.log_info(f" Failed: {stats['scripts_failed']}")
301
+ self.logger_bridge.log_info(f" Success rate: {stats['success_rate']:.1f}%")
302
+ self.logger_bridge.log_info(f" API calls made: {stats['api_calls_made']}")
303
+
304
+ # Show recent executions
305
+ if self._execution_history:
306
+ self.logger_bridge.log_info(" Recent executions:")
307
+ for execution in self._execution_history[-3:]: # Show last 3
308
+ status = "✅" if execution["success"] else "❌"
309
+ self.logger_bridge.log_info(f" {status} {execution['duration_ms']:.1f}ms")
310
+
311
+ def clear_history(self) -> None:
312
+ """Clear execution history"""
313
+ self._execution_history.clear()
314
+ self.logger_bridge.log_info("🧹 Cleared script execution history")
Binary file
@@ -24,6 +24,7 @@ class URLConfig(BaseModel):
24
24
 
25
25
  # Cloud platform URLs
26
26
  cloud_base_url: str = Field(
27
+ default="https://cloud.unrealon.com",
27
28
  description="Base URL for UnrealOn Cloud platform"
28
29
  )
29
30
 
@@ -41,7 +42,7 @@ class URLConfig(BaseModel):
41
42
  if environment == Environment.PRODUCTION:
42
43
  return cls(
43
44
  # scanner_url="https://cloud.unrealon.com/scanner",
44
- cloud_base_url="https://cloud.unrealon.com",
45
+ # cloud_base_url="https://cloud.unrealon.com",
45
46
  api_base_url="https://api.unrealon.com"
46
47
  )
47
48
  # elif environment == Environment.TESTING:
@@ -6,6 +6,8 @@ from typing import Optional
6
6
  from pydantic import Field
7
7
 
8
8
  from unrealon_browser import BrowserManager as CoreBrowserManager, BrowserConfig
9
+ from unrealon_browser.dto.models.enums import BrowserMode
10
+
9
11
  from .base import BaseManager, ManagerConfig
10
12
 
11
13
 
@@ -14,6 +16,7 @@ class BrowserManagerConfig(ManagerConfig):
14
16
 
15
17
  headless: bool = Field(default=True, description="Run headless")
16
18
  parser_name: str = Field(..., description="Parser name for browser")
19
+ stealth_warmup_enabled: bool = Field(default=False, description="Enable stealth warmup")
17
20
 
18
21
 
19
22
  class BrowserManager(BaseManager):
@@ -50,8 +53,14 @@ class BrowserManager(BaseManager):
50
53
  try:
51
54
  self.logger.info("🚀 Starting browser (lazy initialization)...")
52
55
 
53
- # Create browser config
54
- browser_config = BrowserConfig(parser_name=self.config.parser_name)
56
+ # Create browser config with proper headless mode
57
+ browser_mode = BrowserMode.HEADLESS if self.config.headless else BrowserMode.HEADED
58
+
59
+ browser_config = BrowserConfig(
60
+ parser_name=self.config.parser_name,
61
+ mode=browser_mode,
62
+ stealth_warmup_enabled=self.config.stealth_warmup_enabled
63
+ )
55
64
 
56
65
  # Create browser manager
57
66
  self.browser = CoreBrowserManager(browser_config)
@@ -96,3 +105,19 @@ class BrowserManager(BaseManager):
96
105
  self.logger.error(f"Get HTML failed: {e}")
97
106
  self.stats.record_operation(False, 0.0)
98
107
  return None
108
+
109
+ async def execute_script_async(self, script: str) -> any:
110
+ """Execute JavaScript on current page via ScriptManager."""
111
+ # Ensure browser is initialized
112
+ if not await self._ensure_browser_initialized():
113
+ raise RuntimeError("Failed to initialize browser")
114
+
115
+ try:
116
+ # Use ScriptManager from CoreBrowserManager
117
+ result = await self.browser.script_manager.execute_script(script)
118
+ self.stats.record_operation(True, 0.0)
119
+ return result
120
+ except Exception as e:
121
+ self.logger.error(f"Script execution failed: {e}")
122
+ self.stats.record_operation(False, 0.0)
123
+ raise