webscout 8.3__py3-none-any.whl → 8.3.2__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.

Potentially problematic release.


This version of webscout might be problematic. Click here for more details.

Files changed (120) hide show
  1. webscout/AIauto.py +4 -4
  2. webscout/AIbase.py +61 -1
  3. webscout/AIutel.py +46 -53
  4. webscout/Bing_search.py +418 -0
  5. webscout/Extra/YTToolkit/ytapi/patterns.py +45 -45
  6. webscout/Extra/YTToolkit/ytapi/stream.py +1 -1
  7. webscout/Extra/YTToolkit/ytapi/video.py +10 -10
  8. webscout/Extra/autocoder/autocoder_utiles.py +1 -1
  9. webscout/Extra/gguf.py +706 -177
  10. webscout/Litlogger/formats.py +9 -0
  11. webscout/Litlogger/handlers.py +18 -0
  12. webscout/Litlogger/logger.py +43 -1
  13. webscout/Provider/AISEARCH/genspark_search.py +7 -7
  14. webscout/Provider/AISEARCH/scira_search.py +3 -2
  15. webscout/Provider/GeminiProxy.py +140 -0
  16. webscout/Provider/LambdaChat.py +7 -1
  17. webscout/Provider/MCPCore.py +78 -75
  18. webscout/Provider/OPENAI/BLACKBOXAI.py +1046 -1017
  19. webscout/Provider/OPENAI/GeminiProxy.py +328 -0
  20. webscout/Provider/OPENAI/Qwen3.py +303 -303
  21. webscout/Provider/OPENAI/README.md +5 -0
  22. webscout/Provider/OPENAI/README_AUTOPROXY.md +238 -0
  23. webscout/Provider/OPENAI/TogetherAI.py +355 -0
  24. webscout/Provider/OPENAI/__init__.py +16 -1
  25. webscout/Provider/OPENAI/autoproxy.py +332 -0
  26. webscout/Provider/OPENAI/base.py +101 -14
  27. webscout/Provider/OPENAI/chatgpt.py +15 -2
  28. webscout/Provider/OPENAI/chatgptclone.py +14 -3
  29. webscout/Provider/OPENAI/deepinfra.py +339 -328
  30. webscout/Provider/OPENAI/e2b.py +295 -74
  31. webscout/Provider/OPENAI/mcpcore.py +109 -70
  32. webscout/Provider/OPENAI/opkfc.py +18 -6
  33. webscout/Provider/OPENAI/scirachat.py +59 -50
  34. webscout/Provider/OPENAI/toolbaz.py +2 -10
  35. webscout/Provider/OPENAI/writecream.py +166 -166
  36. webscout/Provider/OPENAI/x0gpt.py +367 -367
  37. webscout/Provider/OPENAI/xenai.py +514 -0
  38. webscout/Provider/OPENAI/yep.py +389 -383
  39. webscout/Provider/STT/__init__.py +3 -0
  40. webscout/Provider/STT/base.py +281 -0
  41. webscout/Provider/STT/elevenlabs.py +265 -0
  42. webscout/Provider/TTI/__init__.py +4 -1
  43. webscout/Provider/TTI/aiarta.py +399 -365
  44. webscout/Provider/TTI/base.py +74 -2
  45. webscout/Provider/TTI/bing.py +231 -0
  46. webscout/Provider/TTI/fastflux.py +63 -30
  47. webscout/Provider/TTI/gpt1image.py +149 -0
  48. webscout/Provider/TTI/imagen.py +196 -0
  49. webscout/Provider/TTI/magicstudio.py +60 -29
  50. webscout/Provider/TTI/piclumen.py +43 -32
  51. webscout/Provider/TTI/pixelmuse.py +232 -225
  52. webscout/Provider/TTI/pollinations.py +43 -32
  53. webscout/Provider/TTI/together.py +287 -0
  54. webscout/Provider/TTI/utils.py +2 -1
  55. webscout/Provider/TTS/README.md +1 -0
  56. webscout/Provider/TTS/__init__.py +2 -1
  57. webscout/Provider/TTS/freetts.py +140 -0
  58. webscout/Provider/TTS/speechma.py +45 -39
  59. webscout/Provider/TogetherAI.py +366 -0
  60. webscout/Provider/UNFINISHED/ChutesAI.py +314 -0
  61. webscout/Provider/UNFINISHED/fetch_together_models.py +95 -0
  62. webscout/Provider/XenAI.py +324 -0
  63. webscout/Provider/__init__.py +8 -0
  64. webscout/Provider/deepseek_assistant.py +378 -0
  65. webscout/Provider/scira_chat.py +3 -2
  66. webscout/Provider/toolbaz.py +0 -1
  67. webscout/auth/__init__.py +44 -0
  68. webscout/auth/api_key_manager.py +189 -0
  69. webscout/auth/auth_system.py +100 -0
  70. webscout/auth/config.py +76 -0
  71. webscout/auth/database.py +400 -0
  72. webscout/auth/exceptions.py +67 -0
  73. webscout/auth/middleware.py +248 -0
  74. webscout/auth/models.py +130 -0
  75. webscout/auth/providers.py +257 -0
  76. webscout/auth/rate_limiter.py +254 -0
  77. webscout/auth/request_models.py +127 -0
  78. webscout/auth/request_processing.py +226 -0
  79. webscout/auth/routes.py +526 -0
  80. webscout/auth/schemas.py +103 -0
  81. webscout/auth/server.py +312 -0
  82. webscout/auth/static/favicon.svg +11 -0
  83. webscout/auth/swagger_ui.py +203 -0
  84. webscout/auth/templates/components/authentication.html +237 -0
  85. webscout/auth/templates/components/base.html +103 -0
  86. webscout/auth/templates/components/endpoints.html +750 -0
  87. webscout/auth/templates/components/examples.html +491 -0
  88. webscout/auth/templates/components/footer.html +75 -0
  89. webscout/auth/templates/components/header.html +27 -0
  90. webscout/auth/templates/components/models.html +286 -0
  91. webscout/auth/templates/components/navigation.html +70 -0
  92. webscout/auth/templates/static/api.js +455 -0
  93. webscout/auth/templates/static/icons.js +168 -0
  94. webscout/auth/templates/static/main.js +784 -0
  95. webscout/auth/templates/static/particles.js +201 -0
  96. webscout/auth/templates/static/styles.css +3353 -0
  97. webscout/auth/templates/static/ui.js +374 -0
  98. webscout/auth/templates/swagger_ui.html +170 -0
  99. webscout/client.py +49 -3
  100. webscout/litagent/Readme.md +12 -3
  101. webscout/litagent/agent.py +99 -62
  102. webscout/scout/core/scout.py +104 -26
  103. webscout/scout/element.py +139 -18
  104. webscout/swiftcli/core/cli.py +14 -3
  105. webscout/swiftcli/decorators/output.py +59 -9
  106. webscout/update_checker.py +31 -49
  107. webscout/version.py +1 -1
  108. webscout/webscout_search.py +4 -12
  109. webscout/webscout_search_async.py +3 -10
  110. webscout/yep_search.py +2 -11
  111. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/METADATA +41 -11
  112. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/RECORD +116 -68
  113. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/entry_points.txt +1 -1
  114. webscout/Provider/HF_space/__init__.py +0 -0
  115. webscout/Provider/HF_space/qwen_qwen2.py +0 -206
  116. webscout/Provider/OPENAI/api.py +0 -1035
  117. webscout/Provider/TTI/artbit.py +0 -0
  118. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/WHEEL +0 -0
  119. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/licenses/LICENSE.md +0 -0
  120. {webscout-8.3.dist-info → webscout-8.3.2.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,324 @@
1
+ import json
2
+ import uuid
3
+ import random
4
+ import string
5
+ from typing import Any, Dict, Generator, Union
6
+ import requests
7
+ import warnings
8
+ import urllib3
9
+
10
+ from webscout.AIutel import Optimizers
11
+ from webscout.AIutel import Conversation
12
+ from webscout.AIutel import AwesomePrompts, sanitize_stream
13
+ from webscout.AIbase import Provider
14
+ from webscout import exceptions
15
+ from webscout.litagent import LitAgent
16
+
17
+ # Suppress only the single InsecureRequestWarning from urllib3 needed for verify=False
18
+ warnings.filterwarnings("ignore", category=urllib3.exceptions.InsecureRequestWarning)
19
+
20
+ class XenAI(Provider):
21
+
22
+ # Add more models if known, starting with the one from the example
23
+ AVAILABLE_MODELS = [
24
+ "gemini-2.5-pro-preview-05-06",
25
+ "gemini-2.5-flash-preview-05-20",
26
+ "o4-mini-high",
27
+ "grok-3-mini-fast-beta",
28
+ "grok-3-fast-beta",
29
+ "gpt-4.1",
30
+ "o3-high",
31
+ "gpt-4o-search-preview",
32
+ "gpt-4o",
33
+ "claude-sonnet-4-20250514",
34
+ "claude-sonnet-4-20250514-thinking",
35
+ "deepseek-ai/DeepSeek-V3-0324",
36
+ "deepseek-ai/DeepSeek-R1-0528",
37
+ "groq/deepseek-r1-distill-llama-70b",
38
+ "deepseek-ai/DeepSeek-Prover-V2-671B",
39
+ "meta-llama/llama-4-maverick-17b-128e-instruct",
40
+ "meta-llama/llama-4-scout-17b-16e-instruct",
41
+ "cognitivecomputations/Dolphin3.0-Mistral-24B",
42
+ "sonar-pro",
43
+ "gpt-4o-mini",
44
+ "gemini-2.0-flash-lite-preview-02-05",
45
+ "claude-3-7-sonnet-20250219",
46
+ "claude-3-7-sonnet-20250219-thinking",
47
+ "claude-opus-4-20250514",
48
+ "claude-opus-4-20250514-thinking",
49
+ "chutesai/Llama-4-Maverick-17B-128E-Instruct-FP8",
50
+ "chutesai/Llama-4-Scout-17B-16E-Instruct",
51
+ ]
52
+
53
+ def __init__(
54
+ self,
55
+ is_conversation: bool = True,
56
+ max_tokens: int = 2048,
57
+ timeout: int = 60,
58
+ intro: str = None,
59
+ filepath: str = None,
60
+ update_file: bool = True,
61
+ proxies: dict = {},
62
+ history_offset: int = 10250,
63
+ act: str = None,
64
+ model: str = "gemini-2.5-pro-preview-05-06",
65
+ system_prompt: str = "You are a helpful assistant.",
66
+ ):
67
+ """Initializes the xenai API client."""
68
+ if model not in self.AVAILABLE_MODELS:
69
+ print(f"Warning: Model '{model}' is not listed in AVAILABLE_MODELS. Proceeding with the provided model.")
70
+
71
+ self.api_endpoint = "https://chat.xenai.tech/api/chat/completions"
72
+
73
+ self.model = model
74
+ self.system_prompt = system_prompt
75
+
76
+ # Initialize requests Session
77
+ self.session = requests.Session()
78
+
79
+ # Set up headers based on the provided request
80
+ self.headers = {
81
+ **LitAgent().generate_fingerprint(),
82
+ 'origin': 'https://chat.xenai.tech',
83
+ 'referer': 'https://chat.xenai.tech/',
84
+ }
85
+
86
+ # Apply headers, proxies, and cookies to the session
87
+ self.session.headers.update(self.headers)
88
+ self.session.proxies.update(proxies)
89
+ # Always disable SSL verification for this session
90
+ self.session.verify = False
91
+
92
+ # Provider settings
93
+ self.is_conversation = is_conversation
94
+ self.max_tokens_to_sample = max_tokens
95
+ self.timeout = timeout
96
+ self.last_response = {}
97
+
98
+ # Initialize optimizers
99
+ self.__available_optimizers = (
100
+ method
101
+ for method in dir(Optimizers)
102
+ if callable(getattr(Optimizers, method))
103
+ and not method.startswith("__")
104
+ )
105
+ Conversation.intro = (
106
+ AwesomePrompts().get_act(
107
+ act, raise_not_found=True, default=None, case_insensitive=True
108
+ )
109
+ if act
110
+ else intro or Conversation.intro
111
+ )
112
+ self.conversation = Conversation(
113
+ is_conversation, self.max_tokens_to_sample, filepath, update_file
114
+ )
115
+ self.conversation.history_offset = history_offset
116
+
117
+ # Token handling: always auto-fetch token, no cookies logic
118
+ self.token = self._auto_fetch_token()
119
+
120
+ # Set the Authorization header for the session
121
+ self.session.headers.update({
122
+ 'authorization': f'Bearer {self.token}',
123
+ })
124
+
125
+ def _auto_fetch_token(self):
126
+ """Automatically fetch a token from the signup endpoint using requests."""
127
+ session = requests.Session()
128
+ session.verify = False # Always disable SSL verification for this session
129
+ def random_string(length=8):
130
+ return ''.join(random.choices(string.ascii_lowercase, k=length))
131
+ name = random_string(8)
132
+ email = f"{name}@gmail.com"
133
+ password = email
134
+ profile_image_url = ""
135
+ payload = {
136
+ "name": name,
137
+ "email": email,
138
+ "password": password,
139
+ "profile_image_url": profile_image_url
140
+ }
141
+ headers = {
142
+ **LitAgent().generate_fingerprint(),
143
+ 'origin': 'https://chat.xenai.tech',
144
+ 'referer': 'https://chat.xenai.tech/auth',
145
+ }
146
+ try:
147
+ resp = session.post(
148
+ "https://chat.xenai.tech/api/v1/auths/signup",
149
+ headers=headers,
150
+ json=payload,
151
+ timeout=30,
152
+ verify=False # Disable SSL verification for testing
153
+ )
154
+ if resp.ok:
155
+ data = resp.json()
156
+ token = data.get("token")
157
+ if token:
158
+ return token
159
+ set_cookie = resp.headers.get("set-cookie", "")
160
+ if "token=" in set_cookie:
161
+ return set_cookie.split("token=")[1].split(";")[0]
162
+ raise exceptions.FailedToGenerateResponseError(f"Failed to auto-fetch token: {resp.status_code} {resp.text}")
163
+ except Exception as e:
164
+ raise exceptions.FailedToGenerateResponseError(f"Token auto-fetch failed: {e}")
165
+
166
+ def ask(
167
+ self,
168
+ prompt: str,
169
+ stream: bool = False,
170
+ raw: bool = False,
171
+ optimizer: str = None,
172
+ conversationally: bool = False,
173
+ **kwargs
174
+ ) -> Union[Dict[str, Any], Generator]:
175
+ """Sends a prompt to the xenai API and returns the response."""
176
+
177
+ conversation_prompt = self.conversation.gen_complete_prompt(prompt)
178
+
179
+ if optimizer:
180
+ if optimizer in self.__available_optimizers:
181
+ conversation_prompt = getattr(Optimizers, optimizer)(
182
+ conversation_prompt if conversationally else prompt
183
+ )
184
+ else:
185
+ raise exceptions.InvalidOptimizerError(
186
+ f"Optimizer is not one of {self.__available_optimizers}"
187
+ )
188
+
189
+ chat_id = kwargs.get("chat_id", str(uuid.uuid4()))
190
+ message_id = str(uuid.uuid4())
191
+
192
+ payload = {
193
+ "stream": stream,
194
+ "model": self.model,
195
+ "messages": [
196
+ {"role": "system", "content": self.system_prompt},
197
+ {"role": "user", "content": conversation_prompt}
198
+ ],
199
+ "params": kwargs.get("params", {}),
200
+ "tool_servers": kwargs.get("tool_servers", []),
201
+ "features": kwargs.get("features", {"web_search": False}),
202
+ "chat_id": chat_id,
203
+ "id": message_id,
204
+ "stream_options": kwargs.get("stream_options", {"include_usage": True})
205
+ }
206
+
207
+ def for_stream():
208
+ streaming_text = ""
209
+ try:
210
+ response = self.session.post(
211
+ self.api_endpoint,
212
+ json=payload,
213
+ stream=True,
214
+ timeout=self.timeout,
215
+ verify=False # Always disable SSL verification for this request
216
+ )
217
+ response.raise_for_status()
218
+
219
+ # Use sanitize_stream
220
+ processed_stream = sanitize_stream(
221
+ data=response.iter_content(chunk_size=None), # Pass byte iterator
222
+ intro_value="data:",
223
+ to_json=True, # Stream sends JSON
224
+ skip_markers=["[DONE]"],
225
+ content_extractor=lambda chunk: chunk.get('choices', [{}])[0].get('delta', {}).get('content') if isinstance(chunk, dict) else None,
226
+ yield_raw_on_error=False # Skip non-JSON or lines where extractor fails
227
+ )
228
+
229
+ for content_chunk in processed_stream:
230
+ # content_chunk is the string extracted by the content_extractor
231
+ if content_chunk and isinstance(content_chunk, str):
232
+ streaming_text += content_chunk
233
+ yield dict(text=content_chunk) if not raw else content_chunk
234
+
235
+ self.last_response = {"text": streaming_text}
236
+ self.conversation.update_chat_history(prompt, self.get_message(self.last_response))
237
+
238
+ except requests.RequestException as e:
239
+ raise exceptions.FailedToGenerateResponseError(f"Request failed (requests): {e}") from e
240
+ except Exception as e:
241
+ err_text = getattr(e, 'response', None) and getattr(e.response, 'text', '')
242
+ raise exceptions.FailedToGenerateResponseError(f"An unexpected error occurred ({type(e).__name__}): {e} - {err_text}") from e
243
+
244
+ def for_non_stream():
245
+ full_text = ""
246
+ try:
247
+ stream_generator = self.ask(
248
+ prompt, stream=True, raw=False, optimizer=optimizer, conversationally=conversationally, **kwargs
249
+ )
250
+ for chunk_data in stream_generator:
251
+ if isinstance(chunk_data, dict):
252
+ full_text += chunk_data["text"]
253
+ elif isinstance(chunk_data, str):
254
+ full_text += chunk_data
255
+ except requests.RequestException as e:
256
+ raise exceptions.FailedToGenerateResponseError(f"Failed to aggregate non-stream response (requests): {str(e)}") from e
257
+ except Exception as e:
258
+ raise exceptions.FailedToGenerateResponseError(f"Failed to aggregate non-stream response: {str(e)}") from e
259
+
260
+ return full_text if raw else self.last_response
261
+
262
+ return for_stream() if stream else for_non_stream()
263
+
264
+ def chat(
265
+ self,
266
+ prompt: str,
267
+ stream: bool = False,
268
+ optimizer: str = None,
269
+ conversationally: bool = False,
270
+ **kwargs
271
+ ) -> Union[str, Generator[str, None, None]]:
272
+ """Generates a response from the xenai API."""
273
+
274
+ def for_stream_chat() -> Generator[str, None, None]:
275
+ gen = self.ask(
276
+ prompt, stream=True, raw=False,
277
+ optimizer=optimizer, conversationally=conversationally, **kwargs
278
+ )
279
+ for response_dict in gen:
280
+ yield self.get_message(response_dict)
281
+
282
+ def for_non_stream_chat() -> str:
283
+ response_data = self.ask(
284
+ prompt, stream=False, raw=False,
285
+ optimizer=optimizer, conversationally=conversationally, **kwargs
286
+ )
287
+ return self.get_message(response_data)
288
+
289
+ return for_stream_chat() if stream else for_non_stream_chat()
290
+
291
+ def get_message(self, response: Dict[str, Any]) -> str:
292
+ """Extracts the message from the API response."""
293
+ assert isinstance(response, dict), "Response should be of dict data-type only"
294
+ return response.get("text", "")
295
+
296
+ # Example usage (no cookies file needed)
297
+ if __name__ == "__main__":
298
+ from rich import print
299
+
300
+ print("-" * 80)
301
+ print(f"{'Model':<50} {'Status':<10} {'Response'}")
302
+ print("-" * 80)
303
+
304
+ for model in XenAI.AVAILABLE_MODELS:
305
+ try:
306
+ test_ai = XenAI(model=model, timeout=60)
307
+ response = test_ai.chat("Say 'Hello' in one word", stream=True)
308
+ response_text = ""
309
+ # Accumulate the response text without printing in the loop
310
+ for chunk in response:
311
+ response_text += chunk
312
+
313
+ if response_text and len(response_text.strip()) > 0:
314
+ status = "✓"
315
+ # Truncate response if too long
316
+ display_text = response_text.strip()[:50] + "..." if len(response_text.strip()) > 50 else response_text.strip()
317
+ else:
318
+ status = "✗"
319
+ display_text = "Empty or invalid response"
320
+ # Print the final status and response, overwriting the "Testing..." line
321
+ print(f"\r{model:<50} {status:<10} {display_text}")
322
+ except Exception as e:
323
+ # Print error, overwriting the "Testing..." line
324
+ print(f"\r{model:<50} {'✗':<10} {str(e)}")
@@ -83,10 +83,18 @@ from .Flowith import Flowith
83
83
  from .samurai import samurai
84
84
  from .lmarena import lmarena
85
85
  from .oivscode import oivscode
86
+ from .XenAI import XenAI
87
+ from .deepseek_assistant import DeepSeekAssistant
88
+ from .GeminiProxy import GeminiProxy
89
+ from .TogetherAI import TogetherAI
86
90
  __all__ = [
87
91
  'SCNet',
92
+ 'GeminiProxy',
93
+ 'TogetherAI',
88
94
  'oivscode',
95
+ 'DeepSeekAssistant',
89
96
  'lmarena',
97
+ 'XenAI',
90
98
  'NEMOTRON',
91
99
  'Flowith',
92
100
  'samurai',