unrealon 2.0.7__py3-none-any.whl → 2.0.9__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.7
3
+ Version: 2.0.9
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,10 +33,12 @@ 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/enums/__init__.py,sha256=C-fqNVSWofnaq2Ed6GbGtck8WkN8gpHYA_NiIlaEbTI,520
36
+ unrealon_core/config/urls.py,sha256=FjdXI5WJzdveNCYS8FryRQX9gxjSSH-Qu30dT-USUts,2869
37
+ unrealon_core/enums/__init__.py,sha256=wOH3DK3E1xUO3jD4Cu0UFzTL3HaHq9UEdwP70YdnyI0,736
38
+ unrealon_core/enums/events.py,sha256=rlidf5M2-EDZCRLv4tBkBQ0--qsrgKFt5WrAgw7FRGg,2704
39
+ unrealon_core/enums/jobs.py,sha256=sTSjhgnwE9O6tkgecYSI2BFxECEW0Ro5lkPF1oiEy40,787
37
40
  unrealon_core/enums/status.py,sha256=DT6rth7Ve7chsYhJtccY6plsgQQ8jEMhfhcZTpK308c,6861
38
- unrealon_core/enums/types.py,sha256=o4df_zPVE5ByF4P5t_dnwmr1GVQytl5ppKNQBCu0i2A,8784
41
+ unrealon_core/enums/types.py,sha256=SAUPz1P7ZMyoMqvrDc_h04Ogk8nLl6dQNr2by-fo_7E,8864
39
42
  unrealon_core/error_handling/__init__.py,sha256=s7VXUJG7nvD6Od76-ZfFmn5e6r29mxNusP8YPba73GM,1248
40
43
  unrealon_core/error_handling/circuit_breaker.py,sha256=IPT7I8nrDm4eDojrjMgM3-vLIaz-0raS-9W3LUpFnhY,10066
41
44
  unrealon_core/error_handling/error_context.py,sha256=jfluYy1FVyyU_8dwjBN1w2vPq9oLRYLg2fgndlc_8As,9798
@@ -48,9 +51,10 @@ unrealon_core/exceptions/driver.py,sha256=YB3_oRgOMzD4cCSEhqpqpP5iD6ChbZmaOtnw4B
48
51
  unrealon_core/exceptions/proxy.py,sha256=oqy3gV8jha1a2JrOtbJU8zTpWq_XA8Ooou2cl4eHF0o,246
49
52
  unrealon_core/exceptions/task.py,sha256=uNSyJgZvBSHtimQ-ZLPjjDZrVzq6sXscJUQDkCmQ0No,285
50
53
  unrealon_core/exceptions/validation.py,sha256=f7nWbi5ag1nqo7kwN0QaXYNfRrIwHhYVNlKHGiyhuQ8,351
51
- unrealon_core/models/__init__.py,sha256=msWqpIVlqWb0mX3jAwX83i78hFIoqZj8XgE4TgL_uu4,1379
54
+ unrealon_core/models/__init__.py,sha256=2iBHYutI-FAUVYbTj2-G3Kn7dHrz5aqGuy2j1H6dI-Y,1542
52
55
  unrealon_core/models/arq_context.py,sha256=9a6yJgdQ3ZifoCgEEiN90q5qRzQ9pk0ivHnowhzLF7c,8389
53
56
  unrealon_core/models/arq_responses.py,sha256=OYEIjLZ1iEoWEKAOqAuudYWT4I_ZVeftDFOOmWjgEaM,4471
57
+ unrealon_core/models/authentication.py,sha256=bZTTubIS07xXGPP-EWVXcct97O11CXCl7yQ8IHMxdMo,996
54
58
  unrealon_core/models/base.py,sha256=5x5Ie77wH_qRrezd8kSQ23DgOffSmCmICmVqEeXX4Hc,8964
55
59
  unrealon_core/models/bridge_stats.py,sha256=trVoAQkFG5fc8TSzUmH133Xwf8htKYM5Xyl_3-5ApnE,1878
56
60
  unrealon_core/models/communication.py,sha256=YFdiR-J0ncjmvg4yudU1bTfR61BEmV6sQ1zCXWN_xR0,1152
@@ -111,7 +115,7 @@ unrealon_driver/driver/utilities/logging.py,sha256=2my2QnkAa6Hdw-TfO4oOQ94yGc-Cj
111
115
  unrealon_driver/driver/utilities/serialization.py,sha256=wTCSVrEloykiGN4K1JXbk2aqNKm7W90aWXmzhcLyAZc,2123
112
116
  unrealon_driver/managers/__init__.py,sha256=zJJsOb6Oodg7l00v4ncKUytnyeaZM887pHY8-eSuWdU,981
113
117
  unrealon_driver/managers/base.py,sha256=GkuXillg9uqqnx6RL682fmKgK-7JyqYlH6DFUgyN4F8,5445
114
- unrealon_driver/managers/browser.py,sha256=9kwdmWPxWFpNQ1KejtpzdN615V8PBJu8Y2nUOewQaeQ,3489
118
+ unrealon_driver/managers/browser.py,sha256=qM9-bBDSfxheQrjvPMRKDASeVEzy6T_-tH4XF2gc15w,4433
115
119
  unrealon_driver/managers/cache.py,sha256=c0tPKQ5KFd_Un1U8mw3j1WPuycxg863MMWNMveVF_2I,3506
116
120
  unrealon_driver/managers/http.py,sha256=EjlpoTRuhpsgzzrEARxRlbGczzua7hnKFVq06bvCgTM,3624
117
121
  unrealon_driver/managers/logger.py,sha256=PL3rA9ZQl12jJU0EiPAkLwJ6eDHQfIzr8-nc8bVivKQ,10526
@@ -121,9 +125,9 @@ unrealon_driver/managers/threading.py,sha256=djw5cSC99dfBKmep3IJ_8IgxQceMXtNvCp5
121
125
  unrealon_driver/managers/update.py,sha256=-hohVxGXpj5bZ6ZTQN6NH1RK9Pd6GVzCMtu3GS2SdcQ,3582
122
126
  unrealon_driver/utils/__init__.py,sha256=qxXVoQJVdLJhaLBXk_LZV_062AhrvBrMPXWAKfEc3C4,104
123
127
  unrealon_driver/utils/time.py,sha256=Oxk1eicKeZl8ZWbf7gu1Ll716k6CpXmVj67FHSnPIsA,184
124
- unrealon-2.0.7.dist-info/LICENSE,sha256=eEH8mWZW49YMpl4Sh5MtKqkZ8aVTzKQXiNPEnvL14ns,1070
125
- unrealon-2.0.7.dist-info/METADATA,sha256=6emT2rPYUexfKJNIKu4CuuhuSCYptvwMahqhmiXqzRs,15688
126
- unrealon-2.0.7.dist-info/WHEEL,sha256=pL8R0wFFS65tNSRnaOVrsw9EOkOqxLrlUPenUYnJKNo,91
127
- unrealon-2.0.7.dist-info/entry_points.txt,sha256=k0qM-eotpajkKUq-almJmxj9afhXprZ6IkvQkSdcKhI,104
128
- unrealon-2.0.7.dist-info/top_level.txt,sha256=Gu8IeIfIVfUxdi-h-F0nKMQxo15pjhHZ0aTadXTpRE8,47
129
- unrealon-2.0.7.dist-info/RECORD,,
128
+ unrealon-2.0.9.dist-info/LICENSE,sha256=eEH8mWZW49YMpl4Sh5MtKqkZ8aVTzKQXiNPEnvL14ns,1070
129
+ unrealon-2.0.9.dist-info/METADATA,sha256=xxugB_yHPD7JBa7ZfHPdMZn8wUZ_Fxx0OtWXnbc6SN4,15688
130
+ unrealon-2.0.9.dist-info/WHEEL,sha256=pL8R0wFFS65tNSRnaOVrsw9EOkOqxLrlUPenUYnJKNo,91
131
+ unrealon-2.0.9.dist-info/entry_points.txt,sha256=k0qM-eotpajkKUq-almJmxj9afhXprZ6IkvQkSdcKhI,104
132
+ unrealon-2.0.9.dist-info/top_level.txt,sha256=Gu8IeIfIVfUxdi-h-F0nKMQxo15pjhHZ0aTadXTpRE8,47
133
+ unrealon-2.0.9.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:
@@ -9,6 +9,8 @@ Phase 1: Foundation enums with strict validation
9
9
 
10
10
  from .status import DriverStatus, TaskStatus, ProxyStatus, LogLevel
11
11
  from .types import MessageType, ProxyType, TaskPriority
12
+ from .events import EventType, SystemEventType, RedisEventType
13
+ from .jobs import ARQJobName
12
14
 
13
15
  __all__ = [
14
16
  # Status enums
@@ -21,4 +23,12 @@ __all__ = [
21
23
  "MessageType",
22
24
  "ProxyType",
23
25
  "TaskPriority",
26
+
27
+ # Event enums
28
+ "EventType",
29
+ "SystemEventType",
30
+ "RedisEventType",
31
+
32
+ # Job enums
33
+ "ARQJobName",
24
34
  ]
@@ -0,0 +1,98 @@
1
+ """
2
+ Event type enums for UnrealOn system.
3
+
4
+ Centralized event type definitions used across all services.
5
+ """
6
+ from enum import Enum
7
+
8
+
9
+ class EventType(str, Enum):
10
+ """
11
+ Event types used in Redis PubSub and system events.
12
+
13
+ These events are used for communication between services:
14
+ - RPC server publishes events
15
+ - Django consumes events via Redis PubSub
16
+ - System events are stored in database
17
+ """
18
+
19
+ # Driver lifecycle events
20
+ DRIVER_REGISTER = "driver_register"
21
+ DRIVER_DISCONNECT = "driver_disconnect"
22
+
23
+ # Parser events (Django-specific naming)
24
+ PARSER_REGISTERED = "parser_registered"
25
+ PARSER_HEARTBEAT = "parser_heartbeat"
26
+ PARSER_DISCONNECTED = "parser_disconnected"
27
+ PARSER_ERROR = "parser_error"
28
+ PARSER_LOG = "parser_log"
29
+
30
+ # Session events
31
+ SESSION_STARTED = "session_started"
32
+ SESSION_COMPLETED = "session_completed"
33
+ SESSION_FAILED = "session_failed"
34
+
35
+ # Command events
36
+ COMMAND_ISSUED = "command_issued"
37
+ COMMAND_COMPLETED = "command_completed"
38
+ COMMAND_FAILED = "command_failed"
39
+
40
+ # System events
41
+ SYSTEM_STARTUP = "system_startup"
42
+ SYSTEM_SHUTDOWN = "system_shutdown"
43
+ SYSTEM_ERROR = "system_error"
44
+
45
+
46
+ class SystemEventType(str, Enum):
47
+ """
48
+ System event types for Django database storage.
49
+
50
+ These match the choices in Django's SystemEvent model.
51
+ """
52
+
53
+ # Parser events
54
+ PARSER_REGISTERED = "parser_registered"
55
+ PARSER_HEARTBEAT = "parser_heartbeat"
56
+ PARSER_DISCONNECTED = "parser_disconnected"
57
+ PARSER_ERROR = "parser_error"
58
+
59
+ # Session events
60
+ SESSION_STARTED = "session_started"
61
+ SESSION_COMPLETED = "session_completed"
62
+ SESSION_FAILED = "session_failed"
63
+
64
+ # Command events
65
+ COMMAND_ISSUED = "command_issued"
66
+ COMMAND_COMPLETED = "command_completed"
67
+ COMMAND_FAILED = "command_failed"
68
+
69
+ # System events
70
+ SYSTEM_STARTUP = "system_startup"
71
+ SYSTEM_SHUTDOWN = "system_shutdown"
72
+ SYSTEM_ERROR = "system_error"
73
+
74
+
75
+ class RedisEventType(str, Enum):
76
+ """
77
+ Redis PubSub event types.
78
+
79
+ These are the event types published to Redis channels
80
+ for inter-service communication.
81
+ """
82
+
83
+ # Driver lifecycle (from RPC to Django)
84
+ DRIVER_REGISTER = "driver_register"
85
+ DRIVER_DISCONNECT = "driver_disconnect"
86
+
87
+ # Parser events (internal Django events)
88
+ PARSER_HEARTBEAT = "parser_heartbeat"
89
+ PARSER_LOG = "parser_log"
90
+
91
+ # Session events
92
+ SESSION_STARTED = "session_started"
93
+ SESSION_COMPLETED = "session_completed"
94
+ SESSION_FAILED = "session_failed"
95
+
96
+ # Command events
97
+ COMMAND_COMPLETED = "command_completed"
98
+ COMMAND_FAILED = "command_failed"
@@ -0,0 +1,32 @@
1
+ """
2
+ ARQ job name enums for UnrealOn system.
3
+
4
+ Centralized job name definitions used across all services.
5
+ """
6
+ from enum import Enum
7
+
8
+
9
+ class ARQJobName(str, Enum):
10
+ """
11
+ ARQ job names used in the system.
12
+
13
+ These are the job function names that can be enqueued
14
+ and processed by ARQ workers.
15
+ """
16
+
17
+ # Driver management jobs
18
+ REGISTER_DRIVER = "register_driver"
19
+ DRIVER_DISCONNECT = "driver_disconnect"
20
+ GET_DRIVER_STATUS = "get_driver_status"
21
+ LIST_AVAILABLE_DRIVERS = "list_available_drivers"
22
+
23
+ # Task management jobs
24
+ ASSIGN_TASK_TO_DRIVER = "assign_task_to_driver"
25
+
26
+ # System jobs
27
+ PING = "ping"
28
+ PROCESS_DRIVER_HEARTBEAT = "process_driver_heartbeat"
29
+
30
+ # Lifecycle jobs
31
+ STARTUP = "startup"
32
+ SHUTDOWN = "shutdown"
@@ -58,6 +58,7 @@ class MessageType(str, Enum):
58
58
  PONG = "pong" # Pong response
59
59
  ERROR = "error" # Error message
60
60
  ACK = "ack" # Acknowledgment
61
+ MONITOR_REGISTER = "monitor_register" # Monitor client registration
61
62
 
62
63
  def is_driver_lifecycle(self) -> bool:
63
64
  """Check if message is related to driver lifecycle."""
@@ -46,6 +46,11 @@ from .logging import (
46
46
  LogMetrics
47
47
  )
48
48
 
49
+ from .authentication import (
50
+ APIKeyAuthRequest,
51
+ APIKeyAuthResponse
52
+ )
53
+
49
54
  __all__ = [
50
55
  # Base models
51
56
  "UnrealOnBaseModel",
@@ -76,4 +81,8 @@ __all__ = [
76
81
  "LogEntry",
77
82
  "LogQuery",
78
83
  "LogMetrics",
84
+
85
+ # Authentication models
86
+ "APIKeyAuthRequest",
87
+ "APIKeyAuthResponse",
79
88
  ]
@@ -0,0 +1,24 @@
1
+ """
2
+ Authentication models for UnrealOn system.
3
+
4
+ Provides Pydantic models for API key authentication between services.
5
+ """
6
+ from typing import Optional, List
7
+ from pydantic import Field
8
+
9
+ from .base import UnrealOnBaseModel
10
+
11
+
12
+ class APIKeyAuthRequest(UnrealOnBaseModel):
13
+ """API key authentication request."""
14
+ api_key: str = Field(min_length=1, description="API key for authentication")
15
+ parser_id: str = Field(min_length=1, description="Parser requesting authentication")
16
+
17
+
18
+ class APIKeyAuthResponse(UnrealOnBaseModel):
19
+ """API key authentication response."""
20
+ success: bool = Field(description="Whether authentication was successful")
21
+ user_id: Optional[int] = Field(default=None, description="Authenticated user ID")
22
+ username: Optional[str] = Field(default=None, description="Authenticated username")
23
+ permissions: List[str] = Field(default_factory=list, description="User permissions")
24
+ error: Optional[str] = Field(default=None, description="Error message if failed")
@@ -14,6 +14,7 @@ class BrowserManagerConfig(ManagerConfig):
14
14
 
15
15
  headless: bool = Field(default=True, description="Run headless")
16
16
  parser_name: str = Field(..., description="Parser name for browser")
17
+ stealth_warmup_enabled: bool = Field(default=False, description="Enable stealth warmup")
17
18
 
18
19
 
19
20
  class BrowserManager(BaseManager):
@@ -51,7 +52,10 @@ class BrowserManager(BaseManager):
51
52
  self.logger.info("🚀 Starting browser (lazy initialization)...")
52
53
 
53
54
  # Create browser config
54
- browser_config = BrowserConfig(parser_name=self.config.parser_name)
55
+ browser_config = BrowserConfig(
56
+ parser_name=self.config.parser_name,
57
+ stealth_warmup_enabled=self.config.stealth_warmup_enabled # Disable stealth warmup to avoid scanner navigation
58
+ )
55
59
 
56
60
  # Create browser manager
57
61
  self.browser = CoreBrowserManager(browser_config)
@@ -96,3 +100,19 @@ class BrowserManager(BaseManager):
96
100
  self.logger.error(f"Get HTML failed: {e}")
97
101
  self.stats.record_operation(False, 0.0)
98
102
  return None
103
+
104
+ async def execute_script_async(self, script: str) -> any:
105
+ """Execute JavaScript on current page via ScriptManager."""
106
+ # Ensure browser is initialized
107
+ if not await self._ensure_browser_initialized():
108
+ raise RuntimeError("Failed to initialize browser")
109
+
110
+ try:
111
+ # Use ScriptManager from CoreBrowserManager
112
+ result = await self.browser.script_manager.execute_script(script)
113
+ self.stats.record_operation(True, 0.0)
114
+ return result
115
+ except Exception as e:
116
+ self.logger.error(f"Script execution failed: {e}")
117
+ self.stats.record_operation(False, 0.0)
118
+ raise