webscout 8.3.1__py3-none-any.whl → 8.3.3__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.
- webscout/AIutel.py +180 -78
- webscout/Bing_search.py +417 -0
- webscout/Extra/gguf.py +706 -177
- webscout/Provider/AISEARCH/__init__.py +1 -0
- webscout/Provider/AISEARCH/genspark_search.py +7 -7
- webscout/Provider/AISEARCH/stellar_search.py +132 -0
- webscout/Provider/ExaChat.py +84 -58
- webscout/Provider/GeminiProxy.py +140 -0
- webscout/Provider/HeckAI.py +85 -80
- webscout/Provider/Jadve.py +56 -50
- webscout/Provider/MCPCore.py +78 -75
- webscout/Provider/MiniMax.py +207 -0
- webscout/Provider/Nemotron.py +41 -13
- webscout/Provider/Netwrck.py +34 -51
- webscout/Provider/OPENAI/BLACKBOXAI.py +0 -4
- webscout/Provider/OPENAI/GeminiProxy.py +328 -0
- webscout/Provider/OPENAI/MiniMax.py +298 -0
- webscout/Provider/OPENAI/README.md +32 -29
- webscout/Provider/OPENAI/README_AUTOPROXY.md +238 -0
- webscout/Provider/OPENAI/TogetherAI.py +4 -17
- webscout/Provider/OPENAI/__init__.py +17 -1
- webscout/Provider/OPENAI/autoproxy.py +1067 -39
- webscout/Provider/OPENAI/base.py +17 -76
- webscout/Provider/OPENAI/deepinfra.py +42 -108
- webscout/Provider/OPENAI/e2b.py +0 -1
- webscout/Provider/OPENAI/flowith.py +179 -166
- webscout/Provider/OPENAI/friendli.py +233 -0
- webscout/Provider/OPENAI/mcpcore.py +109 -70
- webscout/Provider/OPENAI/monochat.py +329 -0
- webscout/Provider/OPENAI/pydantic_imports.py +1 -172
- webscout/Provider/OPENAI/scirachat.py +59 -51
- webscout/Provider/OPENAI/toolbaz.py +3 -9
- webscout/Provider/OPENAI/typegpt.py +1 -1
- webscout/Provider/OPENAI/utils.py +19 -42
- webscout/Provider/OPENAI/x0gpt.py +14 -2
- webscout/Provider/OPENAI/xenai.py +514 -0
- webscout/Provider/OPENAI/yep.py +8 -2
- webscout/Provider/OpenGPT.py +54 -32
- webscout/Provider/PI.py +58 -84
- webscout/Provider/StandardInput.py +32 -13
- webscout/Provider/TTI/README.md +9 -9
- webscout/Provider/TTI/__init__.py +3 -1
- webscout/Provider/TTI/aiarta.py +92 -78
- webscout/Provider/TTI/bing.py +231 -0
- webscout/Provider/TTI/infip.py +212 -0
- webscout/Provider/TTI/monochat.py +220 -0
- webscout/Provider/TTS/speechma.py +45 -39
- webscout/Provider/TeachAnything.py +11 -3
- webscout/Provider/TextPollinationsAI.py +78 -70
- webscout/Provider/TogetherAI.py +350 -0
- webscout/Provider/Venice.py +37 -46
- webscout/Provider/VercelAI.py +27 -24
- webscout/Provider/WiseCat.py +35 -35
- webscout/Provider/WrDoChat.py +22 -26
- webscout/Provider/WritingMate.py +26 -22
- webscout/Provider/XenAI.py +324 -0
- webscout/Provider/__init__.py +10 -5
- webscout/Provider/deepseek_assistant.py +378 -0
- webscout/Provider/granite.py +48 -57
- webscout/Provider/koala.py +51 -39
- webscout/Provider/learnfastai.py +49 -64
- webscout/Provider/llmchat.py +79 -93
- webscout/Provider/llmchatco.py +63 -78
- webscout/Provider/multichat.py +51 -40
- webscout/Provider/oivscode.py +1 -1
- webscout/Provider/scira_chat.py +159 -96
- webscout/Provider/scnet.py +13 -13
- webscout/Provider/searchchat.py +13 -13
- webscout/Provider/sonus.py +12 -11
- webscout/Provider/toolbaz.py +25 -8
- webscout/Provider/turboseek.py +41 -42
- webscout/Provider/typefully.py +27 -12
- webscout/Provider/typegpt.py +41 -46
- webscout/Provider/uncovr.py +55 -90
- webscout/Provider/x0gpt.py +33 -17
- webscout/Provider/yep.py +79 -96
- webscout/auth/__init__.py +55 -0
- webscout/auth/api_key_manager.py +189 -0
- webscout/auth/auth_system.py +100 -0
- webscout/auth/config.py +76 -0
- webscout/auth/database.py +400 -0
- webscout/auth/exceptions.py +67 -0
- webscout/auth/middleware.py +248 -0
- webscout/auth/models.py +130 -0
- webscout/auth/providers.py +279 -0
- webscout/auth/rate_limiter.py +254 -0
- webscout/auth/request_models.py +127 -0
- webscout/auth/request_processing.py +226 -0
- webscout/auth/routes.py +550 -0
- webscout/auth/schemas.py +103 -0
- webscout/auth/server.py +367 -0
- webscout/client.py +121 -70
- webscout/litagent/Readme.md +68 -55
- webscout/litagent/agent.py +99 -9
- webscout/scout/core/scout.py +104 -26
- webscout/scout/element.py +139 -18
- webscout/swiftcli/core/cli.py +14 -3
- webscout/swiftcli/decorators/output.py +59 -9
- webscout/update_checker.py +31 -49
- webscout/version.py +1 -1
- webscout/webscout_search.py +4 -12
- webscout/webscout_search_async.py +3 -10
- webscout/yep_search.py +2 -11
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/METADATA +141 -99
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/RECORD +109 -83
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/entry_points.txt +1 -1
- webscout/Provider/HF_space/__init__.py +0 -0
- webscout/Provider/HF_space/qwen_qwen2.py +0 -206
- webscout/Provider/OPENAI/api.py +0 -1320
- webscout/Provider/TTI/fastflux.py +0 -233
- webscout/Provider/Writecream.py +0 -246
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/WHEEL +0 -0
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/licenses/LICENSE.md +0 -0
- {webscout-8.3.1.dist-info → webscout-8.3.3.dist-info}/top_level.txt +0 -0
webscout/Provider/WiseCat.py
CHANGED
|
@@ -106,7 +106,6 @@ class WiseCat(Provider):
|
|
|
106
106
|
raise Exception(
|
|
107
107
|
f"Optimizer is not one of {self.__available_optimizers}"
|
|
108
108
|
)
|
|
109
|
-
|
|
110
109
|
payload = {
|
|
111
110
|
"id": "ephemeral",
|
|
112
111
|
"messages": [
|
|
@@ -121,52 +120,49 @@ class WiseCat(Provider):
|
|
|
121
120
|
],
|
|
122
121
|
"selectedChatModel": self.model
|
|
123
122
|
}
|
|
124
|
-
|
|
125
123
|
def for_stream():
|
|
126
|
-
try:
|
|
127
|
-
# Use curl_cffi session post with impersonate
|
|
124
|
+
try:
|
|
128
125
|
response = self.session.post(
|
|
129
126
|
self.api_endpoint,
|
|
130
127
|
headers=self.headers,
|
|
131
128
|
json=payload,
|
|
132
129
|
stream=True,
|
|
133
130
|
timeout=self.timeout,
|
|
134
|
-
impersonate="chrome120"
|
|
131
|
+
impersonate="chrome120"
|
|
135
132
|
)
|
|
136
133
|
if not response.ok:
|
|
137
134
|
error_msg = f"Failed to generate response - ({response.status_code}, {response.reason}) - {response.text}"
|
|
138
135
|
raise exceptions.FailedToGenerateResponseError(error_msg)
|
|
139
|
-
|
|
140
136
|
streaming_text = ""
|
|
141
|
-
# Use sanitize_stream with the custom extractor
|
|
142
137
|
processed_stream = sanitize_stream(
|
|
143
|
-
data=response.iter_content(chunk_size=None),
|
|
144
|
-
intro_value=None,
|
|
145
|
-
to_json=False,
|
|
146
|
-
content_extractor=self._wisecat_extractor
|
|
138
|
+
data=response.iter_content(chunk_size=None),
|
|
139
|
+
intro_value=None,
|
|
140
|
+
to_json=False,
|
|
141
|
+
content_extractor=self._wisecat_extractor,
|
|
142
|
+
raw=raw
|
|
147
143
|
)
|
|
148
|
-
|
|
149
144
|
for content_chunk in processed_stream:
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
145
|
+
# Always yield as string, even in raw mode
|
|
146
|
+
if isinstance(content_chunk, bytes):
|
|
147
|
+
content_chunk = content_chunk.decode('utf-8', errors='ignore')
|
|
148
|
+
if raw:
|
|
149
|
+
yield content_chunk
|
|
150
|
+
else:
|
|
151
|
+
if content_chunk and isinstance(content_chunk, str):
|
|
152
|
+
streaming_text += content_chunk
|
|
153
|
+
yield dict(text=content_chunk)
|
|
154
|
+
self.last_response.update(dict(text=streaming_text))
|
|
155
155
|
self.conversation.update_chat_history(
|
|
156
156
|
prompt, self.get_message(self.last_response)
|
|
157
157
|
)
|
|
158
|
-
except CurlError as e:
|
|
158
|
+
except CurlError as e:
|
|
159
159
|
raise exceptions.FailedToGenerateResponseError(f"Request failed (CurlError): {e}")
|
|
160
|
-
except Exception as e:
|
|
160
|
+
except Exception as e:
|
|
161
161
|
raise exceptions.FailedToGenerateResponseError(f"An unexpected error occurred ({type(e).__name__}): {e}")
|
|
162
|
-
|
|
163
|
-
|
|
164
162
|
def for_non_stream():
|
|
165
|
-
# This function implicitly uses the updated for_stream
|
|
166
163
|
for _ in for_stream():
|
|
167
164
|
pass
|
|
168
165
|
return self.last_response
|
|
169
|
-
|
|
170
166
|
return for_stream() if stream else for_non_stream()
|
|
171
167
|
|
|
172
168
|
def chat(
|
|
@@ -175,24 +171,28 @@ class WiseCat(Provider):
|
|
|
175
171
|
stream: bool = False,
|
|
176
172
|
optimizer: str = None,
|
|
177
173
|
conversationally: bool = False,
|
|
174
|
+
raw: bool = False, # Added raw parameter
|
|
178
175
|
) -> str:
|
|
179
|
-
"""Generate response `str`"""
|
|
180
176
|
def for_stream():
|
|
181
177
|
for response in self.ask(
|
|
182
|
-
prompt, True, optimizer=optimizer, conversationally=conversationally
|
|
178
|
+
prompt, True, raw=raw, optimizer=optimizer, conversationally=conversationally
|
|
183
179
|
):
|
|
184
|
-
|
|
185
|
-
|
|
180
|
+
if raw:
|
|
181
|
+
yield response
|
|
182
|
+
else:
|
|
183
|
+
yield self.get_message(response)
|
|
186
184
|
def for_non_stream():
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
)
|
|
185
|
+
result = self.ask(
|
|
186
|
+
prompt,
|
|
187
|
+
False,
|
|
188
|
+
raw=raw,
|
|
189
|
+
optimizer=optimizer,
|
|
190
|
+
conversationally=conversationally,
|
|
194
191
|
)
|
|
195
|
-
|
|
192
|
+
if raw:
|
|
193
|
+
return result
|
|
194
|
+
else:
|
|
195
|
+
return self.get_message(result)
|
|
196
196
|
return for_stream() if stream else for_non_stream()
|
|
197
197
|
|
|
198
198
|
def get_message(self, response: dict) -> str:
|
webscout/Provider/WrDoChat.py
CHANGED
|
@@ -244,7 +244,6 @@ class WrDoChat(Provider):
|
|
|
244
244
|
def for_stream():
|
|
245
245
|
try:
|
|
246
246
|
self.headers["referer"] = f"https://oi.wr.do/chat/{chat_id}"
|
|
247
|
-
|
|
248
247
|
response = self.session.post(
|
|
249
248
|
self.api_endpoint,
|
|
250
249
|
json=payload,
|
|
@@ -252,31 +251,27 @@ class WrDoChat(Provider):
|
|
|
252
251
|
timeout=self.timeout,
|
|
253
252
|
impersonate="chrome110"
|
|
254
253
|
)
|
|
255
|
-
|
|
256
254
|
if response.status_code == 401:
|
|
257
255
|
raise exceptions.AuthenticationError("Authentication failed. Please check your cookies.")
|
|
258
|
-
|
|
259
256
|
response.raise_for_status()
|
|
260
|
-
|
|
261
257
|
streaming_response = ""
|
|
262
258
|
has_content = False
|
|
263
|
-
|
|
264
|
-
# Use sanitize_stream with the custom extractor
|
|
265
259
|
processed_stream = sanitize_stream(
|
|
266
260
|
data=response.iter_lines(),
|
|
267
261
|
intro_value=None, # No intro to remove
|
|
268
262
|
to_json=False, # Response is not JSON
|
|
269
263
|
content_extractor=self._wrdo_extractor,
|
|
270
|
-
yield_raw_on_error=False
|
|
264
|
+
yield_raw_on_error=False,
|
|
265
|
+
raw=raw
|
|
271
266
|
)
|
|
272
|
-
|
|
273
267
|
for content in processed_stream:
|
|
268
|
+
# Always yield as string, even in raw mode
|
|
269
|
+
if isinstance(content, bytes):
|
|
270
|
+
content = content.decode('utf-8', errors='ignore')
|
|
274
271
|
if content and isinstance(content, str):
|
|
275
272
|
streaming_response += content
|
|
276
273
|
has_content = True
|
|
277
|
-
yield
|
|
278
|
-
|
|
279
|
-
# Only update conversation history if we received content
|
|
274
|
+
yield content if raw else {"text": content}
|
|
280
275
|
if has_content:
|
|
281
276
|
self.last_response = {"text": streaming_response}
|
|
282
277
|
self.conversation.update_chat_history(
|
|
@@ -286,12 +281,10 @@ class WrDoChat(Provider):
|
|
|
286
281
|
raise exceptions.FailedToGenerateResponseError(
|
|
287
282
|
"No content received from API"
|
|
288
283
|
)
|
|
289
|
-
|
|
290
284
|
except CurlError as e:
|
|
291
285
|
raise exceptions.FailedToGenerateResponseError(f"Request failed (CurlError): {e}")
|
|
292
286
|
except Exception as e:
|
|
293
287
|
raise exceptions.FailedToGenerateResponseError(f"An error occurred: {str(e)}")
|
|
294
|
-
|
|
295
288
|
def for_non_stream():
|
|
296
289
|
response_text = ""
|
|
297
290
|
try:
|
|
@@ -303,9 +296,7 @@ class WrDoChat(Provider):
|
|
|
303
296
|
except Exception as e:
|
|
304
297
|
if not response_text:
|
|
305
298
|
raise exceptions.FailedToGenerateResponseError(f"Failed to get response: {str(e)}")
|
|
306
|
-
|
|
307
299
|
return response_text if raw else {"text": response_text}
|
|
308
|
-
|
|
309
300
|
return for_stream() if stream else for_non_stream()
|
|
310
301
|
|
|
311
302
|
def chat(
|
|
@@ -314,6 +305,7 @@ class WrDoChat(Provider):
|
|
|
314
305
|
stream: bool = False,
|
|
315
306
|
optimizer: str = None,
|
|
316
307
|
conversationally: bool = False,
|
|
308
|
+
raw: bool = False, # Added raw parameter
|
|
317
309
|
) -> Union[str, Generator[str, None, None]]:
|
|
318
310
|
"""
|
|
319
311
|
Generate a response to a prompt.
|
|
@@ -329,20 +321,24 @@ class WrDoChat(Provider):
|
|
|
329
321
|
"""
|
|
330
322
|
def for_stream():
|
|
331
323
|
for response in self.ask(
|
|
332
|
-
prompt, True, optimizer=optimizer, conversationally=conversationally
|
|
324
|
+
prompt, True, raw=raw, optimizer=optimizer, conversationally=conversationally
|
|
333
325
|
):
|
|
334
|
-
|
|
335
|
-
|
|
326
|
+
if raw:
|
|
327
|
+
yield response
|
|
328
|
+
else:
|
|
329
|
+
yield self.get_message(response)
|
|
336
330
|
def for_non_stream():
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
)
|
|
331
|
+
result = self.ask(
|
|
332
|
+
prompt,
|
|
333
|
+
False,
|
|
334
|
+
raw=raw,
|
|
335
|
+
optimizer=optimizer,
|
|
336
|
+
conversationally=conversationally,
|
|
344
337
|
)
|
|
345
|
-
|
|
338
|
+
if raw:
|
|
339
|
+
return result
|
|
340
|
+
else:
|
|
341
|
+
return self.get_message(result)
|
|
346
342
|
return for_stream() if stream else for_non_stream()
|
|
347
343
|
|
|
348
344
|
def get_message(self, response: dict) -> str:
|
webscout/Provider/WritingMate.py
CHANGED
|
@@ -178,13 +178,20 @@ class WritingMate(Provider):
|
|
|
178
178
|
data=response.iter_content(chunk_size=None), # Pass byte iterator
|
|
179
179
|
intro_value=None, # No simple prefix
|
|
180
180
|
to_json=False, # Content is not JSON
|
|
181
|
-
content_extractor=self._writingmate_extractor # Use the specific extractor
|
|
181
|
+
content_extractor=self._writingmate_extractor, # Use the specific extractor
|
|
182
|
+
raw=raw
|
|
182
183
|
)
|
|
183
184
|
|
|
184
185
|
for content_chunk in processed_stream:
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
186
|
+
# Always yield as string, even in raw mode
|
|
187
|
+
if isinstance(content_chunk, bytes):
|
|
188
|
+
content_chunk = content_chunk.decode('utf-8', errors='ignore')
|
|
189
|
+
if raw:
|
|
190
|
+
yield content_chunk
|
|
191
|
+
else:
|
|
192
|
+
if content_chunk and isinstance(content_chunk, str):
|
|
193
|
+
streaming_text += content_chunk
|
|
194
|
+
yield dict(text=content_chunk)
|
|
188
195
|
|
|
189
196
|
self.last_response.update(dict(text=streaming_text))
|
|
190
197
|
self.conversation.update_chat_history(
|
|
@@ -196,12 +203,10 @@ class WritingMate(Provider):
|
|
|
196
203
|
raise exceptions.FailedToGenerateResponseError(f"An unexpected error occurred ({type(e).__name__}): {e}")
|
|
197
204
|
|
|
198
205
|
def for_non_stream():
|
|
199
|
-
# This function implicitly uses the updated for_stream
|
|
200
206
|
for _ in for_stream():
|
|
201
207
|
pass
|
|
202
208
|
return self.last_response
|
|
203
209
|
|
|
204
|
-
# Ensure stream defaults to True if not provided, matching original behavior
|
|
205
210
|
effective_stream = stream if stream is not None else True
|
|
206
211
|
return for_stream() if effective_stream else for_non_stream()
|
|
207
212
|
|
|
@@ -210,36 +215,35 @@ class WritingMate(Provider):
|
|
|
210
215
|
prompt: str,
|
|
211
216
|
stream: bool = False, # Default stream to False as per original chat method
|
|
212
217
|
optimizer: str = None,
|
|
213
|
-
conversationally: bool = False
|
|
218
|
+
conversationally: bool = False,
|
|
219
|
+
raw: bool = False, # Added raw parameter
|
|
214
220
|
) -> Union[str, Generator[str,None,None]]:
|
|
215
221
|
if stream:
|
|
216
|
-
# yield decoded text chunks
|
|
217
222
|
def text_stream():
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
prompt, stream=True, raw=False,
|
|
223
|
+
for response in self.ask(
|
|
224
|
+
prompt, stream=True, raw=raw,
|
|
221
225
|
optimizer=optimizer, conversationally=conversationally
|
|
222
226
|
):
|
|
223
|
-
|
|
224
|
-
|
|
227
|
+
if raw:
|
|
228
|
+
yield response
|
|
229
|
+
else:
|
|
230
|
+
yield self.get_message(response)
|
|
225
231
|
return text_stream()
|
|
226
|
-
else:
|
|
227
|
-
# Call ask with stream=False, raw=False
|
|
232
|
+
else:
|
|
228
233
|
response_data = self.ask(
|
|
229
234
|
prompt,
|
|
230
235
|
stream=False,
|
|
231
|
-
raw=
|
|
236
|
+
raw=raw,
|
|
232
237
|
optimizer=optimizer,
|
|
233
238
|
conversationally=conversationally,
|
|
234
239
|
)
|
|
235
|
-
|
|
240
|
+
if raw:
|
|
241
|
+
return response_data
|
|
236
242
|
if isinstance(response_data, dict):
|
|
237
|
-
|
|
243
|
+
return self.get_message(response_data)
|
|
238
244
|
else:
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
full_text = "".join(self.get_message(chunk) for chunk in response_data if isinstance(chunk, dict))
|
|
242
|
-
return full_text
|
|
245
|
+
full_text = "".join(self.get_message(chunk) for chunk in response_data if isinstance(chunk, dict))
|
|
246
|
+
return full_text
|
|
243
247
|
|
|
244
248
|
|
|
245
249
|
def get_message(self, response: dict) -> str:
|
|
@@ -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)}")
|