webscout 8.2.9__py3-none-any.whl → 2026.1.19__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.
- webscout/AIauto.py +524 -251
- webscout/AIbase.py +247 -319
- webscout/AIutel.py +68 -703
- webscout/Bard.py +1072 -1026
- webscout/Extra/GitToolkit/__init__.py +10 -10
- webscout/Extra/GitToolkit/gitapi/__init__.py +20 -12
- webscout/Extra/GitToolkit/gitapi/gist.py +142 -0
- webscout/Extra/GitToolkit/gitapi/organization.py +91 -0
- webscout/Extra/GitToolkit/gitapi/repository.py +308 -195
- webscout/Extra/GitToolkit/gitapi/search.py +162 -0
- webscout/Extra/GitToolkit/gitapi/trending.py +236 -0
- webscout/Extra/GitToolkit/gitapi/user.py +128 -96
- webscout/Extra/GitToolkit/gitapi/utils.py +82 -62
- webscout/Extra/YTToolkit/README.md +443 -375
- webscout/Extra/YTToolkit/YTdownloader.py +953 -957
- webscout/Extra/YTToolkit/__init__.py +3 -3
- webscout/Extra/YTToolkit/transcriber.py +595 -476
- webscout/Extra/YTToolkit/ytapi/README.md +230 -44
- webscout/Extra/YTToolkit/ytapi/__init__.py +22 -6
- webscout/Extra/YTToolkit/ytapi/captions.py +190 -0
- webscout/Extra/YTToolkit/ytapi/channel.py +302 -307
- webscout/Extra/YTToolkit/ytapi/errors.py +13 -13
- webscout/Extra/YTToolkit/ytapi/extras.py +178 -118
- webscout/Extra/YTToolkit/ytapi/hashtag.py +120 -0
- webscout/Extra/YTToolkit/ytapi/https.py +89 -88
- webscout/Extra/YTToolkit/ytapi/patterns.py +61 -61
- webscout/Extra/YTToolkit/ytapi/playlist.py +59 -59
- webscout/Extra/YTToolkit/ytapi/pool.py +8 -8
- webscout/Extra/YTToolkit/ytapi/query.py +143 -40
- webscout/Extra/YTToolkit/ytapi/shorts.py +122 -0
- webscout/Extra/YTToolkit/ytapi/stream.py +68 -63
- webscout/Extra/YTToolkit/ytapi/suggestions.py +97 -0
- webscout/Extra/YTToolkit/ytapi/utils.py +66 -62
- webscout/Extra/YTToolkit/ytapi/video.py +403 -232
- webscout/Extra/__init__.py +2 -3
- webscout/Extra/gguf.py +1298 -684
- webscout/Extra/tempmail/README.md +487 -487
- webscout/Extra/tempmail/__init__.py +28 -28
- webscout/Extra/tempmail/async_utils.py +143 -141
- webscout/Extra/tempmail/base.py +172 -161
- webscout/Extra/tempmail/cli.py +191 -187
- webscout/Extra/tempmail/emailnator.py +88 -84
- webscout/Extra/tempmail/mail_tm.py +378 -361
- webscout/Extra/tempmail/temp_mail_io.py +304 -292
- webscout/Extra/weather.py +196 -194
- webscout/Extra/weather_ascii.py +17 -15
- webscout/Provider/AISEARCH/PERPLEXED_search.py +175 -0
- webscout/Provider/AISEARCH/Perplexity.py +292 -333
- webscout/Provider/AISEARCH/README.md +106 -279
- webscout/Provider/AISEARCH/__init__.py +16 -9
- webscout/Provider/AISEARCH/brave_search.py +298 -0
- webscout/Provider/AISEARCH/iask_search.py +357 -410
- webscout/Provider/AISEARCH/monica_search.py +200 -220
- webscout/Provider/AISEARCH/webpilotai_search.py +242 -255
- webscout/Provider/Algion.py +413 -0
- webscout/Provider/Andi.py +74 -69
- webscout/Provider/Apriel.py +313 -0
- webscout/Provider/Ayle.py +323 -0
- webscout/Provider/ChatSandbox.py +329 -342
- webscout/Provider/ClaudeOnline.py +365 -0
- webscout/Provider/Cohere.py +232 -208
- webscout/Provider/DeepAI.py +367 -0
- webscout/Provider/Deepinfra.py +467 -340
- webscout/Provider/EssentialAI.py +217 -0
- webscout/Provider/ExaAI.py +274 -261
- webscout/Provider/Gemini.py +175 -169
- webscout/Provider/GithubChat.py +385 -369
- webscout/Provider/Gradient.py +286 -0
- webscout/Provider/Groq.py +556 -801
- webscout/Provider/HadadXYZ.py +323 -0
- webscout/Provider/HeckAI.py +392 -375
- webscout/Provider/HuggingFace.py +387 -0
- webscout/Provider/IBM.py +340 -0
- webscout/Provider/Jadve.py +317 -291
- webscout/Provider/K2Think.py +306 -0
- webscout/Provider/Koboldai.py +221 -384
- webscout/Provider/Netwrck.py +273 -270
- webscout/Provider/Nvidia.py +310 -0
- webscout/Provider/OPENAI/DeepAI.py +489 -0
- webscout/Provider/OPENAI/K2Think.py +423 -0
- webscout/Provider/OPENAI/PI.py +463 -0
- webscout/Provider/OPENAI/README.md +890 -952
- webscout/Provider/OPENAI/TogetherAI.py +405 -0
- webscout/Provider/OPENAI/TwoAI.py +255 -357
- webscout/Provider/OPENAI/__init__.py +148 -40
- webscout/Provider/OPENAI/ai4chat.py +348 -293
- webscout/Provider/OPENAI/akashgpt.py +436 -0
- webscout/Provider/OPENAI/algion.py +303 -0
- webscout/Provider/OPENAI/{exachat.py → ayle.py} +365 -444
- webscout/Provider/OPENAI/base.py +253 -249
- webscout/Provider/OPENAI/cerebras.py +296 -0
- webscout/Provider/OPENAI/chatgpt.py +870 -556
- webscout/Provider/OPENAI/chatsandbox.py +233 -173
- webscout/Provider/OPENAI/deepinfra.py +403 -322
- webscout/Provider/OPENAI/e2b.py +2370 -1414
- webscout/Provider/OPENAI/elmo.py +278 -0
- webscout/Provider/OPENAI/exaai.py +452 -417
- webscout/Provider/OPENAI/freeassist.py +446 -0
- webscout/Provider/OPENAI/gradient.py +448 -0
- webscout/Provider/OPENAI/groq.py +380 -364
- webscout/Provider/OPENAI/hadadxyz.py +292 -0
- webscout/Provider/OPENAI/heckai.py +333 -308
- webscout/Provider/OPENAI/huggingface.py +321 -0
- webscout/Provider/OPENAI/ibm.py +425 -0
- webscout/Provider/OPENAI/llmchat.py +253 -0
- webscout/Provider/OPENAI/llmchatco.py +378 -335
- webscout/Provider/OPENAI/meta.py +541 -0
- webscout/Provider/OPENAI/netwrck.py +374 -357
- webscout/Provider/OPENAI/nvidia.py +317 -0
- webscout/Provider/OPENAI/oivscode.py +348 -287
- webscout/Provider/OPENAI/openrouter.py +328 -0
- webscout/Provider/OPENAI/pydantic_imports.py +1 -172
- webscout/Provider/OPENAI/sambanova.py +397 -0
- webscout/Provider/OPENAI/sonus.py +305 -304
- webscout/Provider/OPENAI/textpollinations.py +370 -339
- webscout/Provider/OPENAI/toolbaz.py +375 -413
- webscout/Provider/OPENAI/typefully.py +419 -355
- webscout/Provider/OPENAI/typliai.py +279 -0
- webscout/Provider/OPENAI/utils.py +314 -318
- webscout/Provider/OPENAI/wisecat.py +359 -387
- webscout/Provider/OPENAI/writecream.py +185 -163
- webscout/Provider/OPENAI/x0gpt.py +462 -365
- webscout/Provider/OPENAI/zenmux.py +380 -0
- webscout/Provider/OpenRouter.py +386 -0
- webscout/Provider/Openai.py +337 -496
- webscout/Provider/PI.py +443 -429
- webscout/Provider/QwenLM.py +346 -254
- webscout/Provider/STT/__init__.py +28 -0
- webscout/Provider/STT/base.py +303 -0
- webscout/Provider/STT/elevenlabs.py +264 -0
- webscout/Provider/Sambanova.py +317 -0
- webscout/Provider/TTI/README.md +69 -82
- webscout/Provider/TTI/__init__.py +37 -7
- webscout/Provider/TTI/base.py +147 -64
- webscout/Provider/TTI/claudeonline.py +393 -0
- webscout/Provider/TTI/magicstudio.py +292 -201
- webscout/Provider/TTI/miragic.py +180 -0
- webscout/Provider/TTI/pollinations.py +331 -221
- webscout/Provider/TTI/together.py +334 -0
- webscout/Provider/TTI/utils.py +14 -11
- webscout/Provider/TTS/README.md +186 -192
- webscout/Provider/TTS/__init__.py +43 -10
- webscout/Provider/TTS/base.py +523 -159
- webscout/Provider/TTS/deepgram.py +286 -156
- webscout/Provider/TTS/elevenlabs.py +189 -111
- webscout/Provider/TTS/freetts.py +218 -0
- webscout/Provider/TTS/murfai.py +288 -113
- webscout/Provider/TTS/openai_fm.py +364 -129
- webscout/Provider/TTS/parler.py +203 -111
- webscout/Provider/TTS/qwen.py +334 -0
- webscout/Provider/TTS/sherpa.py +286 -0
- webscout/Provider/TTS/speechma.py +693 -580
- webscout/Provider/TTS/streamElements.py +275 -333
- webscout/Provider/TTS/utils.py +280 -280
- webscout/Provider/TextPollinationsAI.py +331 -308
- webscout/Provider/TogetherAI.py +450 -0
- webscout/Provider/TwoAI.py +309 -475
- webscout/Provider/TypliAI.py +311 -305
- webscout/Provider/UNFINISHED/ChatHub.py +219 -209
- webscout/Provider/{OPENAI/glider.py → UNFINISHED/ChutesAI.py} +331 -326
- webscout/Provider/{GizAI.py → UNFINISHED/GizAI.py} +300 -295
- webscout/Provider/{Marcus.py → UNFINISHED/Marcus.py} +218 -198
- webscout/Provider/UNFINISHED/Qodo.py +481 -0
- webscout/Provider/{MCPCore.py → UNFINISHED/XenAI.py} +330 -315
- webscout/Provider/UNFINISHED/Youchat.py +347 -330
- webscout/Provider/UNFINISHED/aihumanizer.py +41 -0
- webscout/Provider/UNFINISHED/grammerchecker.py +37 -0
- webscout/Provider/UNFINISHED/liner.py +342 -0
- webscout/Provider/UNFINISHED/liner_api_request.py +246 -263
- webscout/Provider/{samurai.py → UNFINISHED/samurai.py} +231 -224
- webscout/Provider/WiseCat.py +256 -233
- webscout/Provider/WrDoChat.py +390 -370
- webscout/Provider/__init__.py +115 -174
- webscout/Provider/ai4chat.py +181 -174
- webscout/Provider/akashgpt.py +330 -335
- webscout/Provider/cerebras.py +397 -290
- webscout/Provider/cleeai.py +236 -213
- webscout/Provider/elmo.py +291 -283
- webscout/Provider/geminiapi.py +343 -208
- webscout/Provider/julius.py +245 -223
- webscout/Provider/learnfastai.py +333 -325
- webscout/Provider/llama3mitril.py +230 -215
- webscout/Provider/llmchat.py +308 -258
- webscout/Provider/llmchatco.py +321 -306
- webscout/Provider/meta.py +996 -801
- webscout/Provider/oivscode.py +332 -309
- webscout/Provider/searchchat.py +316 -292
- webscout/Provider/sonus.py +264 -258
- webscout/Provider/toolbaz.py +359 -353
- webscout/Provider/turboseek.py +332 -266
- webscout/Provider/typefully.py +262 -202
- webscout/Provider/x0gpt.py +332 -299
- webscout/__init__.py +31 -39
- webscout/__main__.py +5 -5
- webscout/cli.py +585 -524
- webscout/client.py +1497 -70
- webscout/conversation.py +140 -436
- webscout/exceptions.py +383 -362
- webscout/litagent/__init__.py +29 -29
- webscout/litagent/agent.py +492 -455
- webscout/litagent/constants.py +60 -60
- webscout/models.py +505 -181
- webscout/optimizers.py +74 -420
- webscout/prompt_manager.py +376 -288
- webscout/sanitize.py +1514 -0
- webscout/scout/README.md +452 -404
- webscout/scout/__init__.py +8 -8
- webscout/scout/core/__init__.py +7 -7
- webscout/scout/core/crawler.py +330 -210
- webscout/scout/core/scout.py +800 -607
- webscout/scout/core/search_result.py +51 -96
- webscout/scout/core/text_analyzer.py +64 -63
- webscout/scout/core/text_utils.py +412 -277
- webscout/scout/core/web_analyzer.py +54 -52
- webscout/scout/element.py +872 -478
- webscout/scout/parsers/__init__.py +70 -69
- webscout/scout/parsers/html5lib_parser.py +182 -172
- webscout/scout/parsers/html_parser.py +238 -236
- webscout/scout/parsers/lxml_parser.py +203 -178
- webscout/scout/utils.py +38 -37
- webscout/search/__init__.py +47 -0
- webscout/search/base.py +201 -0
- webscout/search/bing_main.py +45 -0
- webscout/search/brave_main.py +92 -0
- webscout/search/duckduckgo_main.py +57 -0
- webscout/search/engines/__init__.py +127 -0
- webscout/search/engines/bing/__init__.py +15 -0
- webscout/search/engines/bing/base.py +35 -0
- webscout/search/engines/bing/images.py +114 -0
- webscout/search/engines/bing/news.py +96 -0
- webscout/search/engines/bing/suggestions.py +36 -0
- webscout/search/engines/bing/text.py +109 -0
- webscout/search/engines/brave/__init__.py +19 -0
- webscout/search/engines/brave/base.py +47 -0
- webscout/search/engines/brave/images.py +213 -0
- webscout/search/engines/brave/news.py +353 -0
- webscout/search/engines/brave/suggestions.py +318 -0
- webscout/search/engines/brave/text.py +167 -0
- webscout/search/engines/brave/videos.py +364 -0
- webscout/search/engines/duckduckgo/__init__.py +25 -0
- webscout/search/engines/duckduckgo/answers.py +80 -0
- webscout/search/engines/duckduckgo/base.py +189 -0
- webscout/search/engines/duckduckgo/images.py +100 -0
- webscout/search/engines/duckduckgo/maps.py +183 -0
- webscout/search/engines/duckduckgo/news.py +70 -0
- webscout/search/engines/duckduckgo/suggestions.py +22 -0
- webscout/search/engines/duckduckgo/text.py +221 -0
- webscout/search/engines/duckduckgo/translate.py +48 -0
- webscout/search/engines/duckduckgo/videos.py +80 -0
- webscout/search/engines/duckduckgo/weather.py +84 -0
- webscout/search/engines/mojeek.py +61 -0
- webscout/search/engines/wikipedia.py +77 -0
- webscout/search/engines/yahoo/__init__.py +41 -0
- webscout/search/engines/yahoo/answers.py +19 -0
- webscout/search/engines/yahoo/base.py +34 -0
- webscout/search/engines/yahoo/images.py +323 -0
- webscout/search/engines/yahoo/maps.py +19 -0
- webscout/search/engines/yahoo/news.py +258 -0
- webscout/search/engines/yahoo/suggestions.py +140 -0
- webscout/search/engines/yahoo/text.py +273 -0
- webscout/search/engines/yahoo/translate.py +19 -0
- webscout/search/engines/yahoo/videos.py +302 -0
- webscout/search/engines/yahoo/weather.py +220 -0
- webscout/search/engines/yandex.py +67 -0
- webscout/search/engines/yep/__init__.py +13 -0
- webscout/search/engines/yep/base.py +34 -0
- webscout/search/engines/yep/images.py +101 -0
- webscout/search/engines/yep/suggestions.py +38 -0
- webscout/search/engines/yep/text.py +99 -0
- webscout/search/http_client.py +172 -0
- webscout/search/results.py +141 -0
- webscout/search/yahoo_main.py +57 -0
- webscout/search/yep_main.py +48 -0
- webscout/server/__init__.py +48 -0
- webscout/server/config.py +78 -0
- webscout/server/exceptions.py +69 -0
- webscout/server/providers.py +286 -0
- webscout/server/request_models.py +131 -0
- webscout/server/request_processing.py +404 -0
- webscout/server/routes.py +642 -0
- webscout/server/server.py +351 -0
- webscout/server/ui_templates.py +1171 -0
- webscout/swiftcli/__init__.py +79 -95
- webscout/swiftcli/core/__init__.py +7 -7
- webscout/swiftcli/core/cli.py +574 -297
- webscout/swiftcli/core/context.py +98 -104
- webscout/swiftcli/core/group.py +268 -241
- webscout/swiftcli/decorators/__init__.py +28 -28
- webscout/swiftcli/decorators/command.py +243 -221
- webscout/swiftcli/decorators/options.py +247 -220
- webscout/swiftcli/decorators/output.py +392 -252
- webscout/swiftcli/exceptions.py +21 -21
- webscout/swiftcli/plugins/__init__.py +9 -9
- webscout/swiftcli/plugins/base.py +134 -135
- webscout/swiftcli/plugins/manager.py +269 -269
- webscout/swiftcli/utils/__init__.py +58 -59
- webscout/swiftcli/utils/formatting.py +251 -252
- webscout/swiftcli/utils/parsing.py +368 -267
- webscout/update_checker.py +280 -136
- webscout/utils.py +28 -14
- webscout/version.py +2 -1
- webscout/version.py.bak +3 -0
- webscout/zeroart/__init__.py +218 -135
- webscout/zeroart/base.py +70 -66
- webscout/zeroart/effects.py +155 -101
- webscout/zeroart/fonts.py +1799 -1239
- webscout-2026.1.19.dist-info/METADATA +638 -0
- webscout-2026.1.19.dist-info/RECORD +312 -0
- {webscout-8.2.9.dist-info → webscout-2026.1.19.dist-info}/WHEEL +1 -1
- {webscout-8.2.9.dist-info → webscout-2026.1.19.dist-info}/entry_points.txt +1 -1
- webscout/DWEBS.py +0 -520
- webscout/Extra/Act.md +0 -309
- webscout/Extra/GitToolkit/gitapi/README.md +0 -110
- webscout/Extra/autocoder/__init__.py +0 -9
- webscout/Extra/autocoder/autocoder.py +0 -1105
- webscout/Extra/autocoder/autocoder_utiles.py +0 -332
- webscout/Extra/gguf.md +0 -430
- webscout/Extra/weather.md +0 -281
- webscout/Litlogger/README.md +0 -10
- webscout/Litlogger/__init__.py +0 -15
- webscout/Litlogger/formats.py +0 -4
- webscout/Litlogger/handlers.py +0 -103
- webscout/Litlogger/levels.py +0 -13
- webscout/Litlogger/logger.py +0 -92
- webscout/Provider/AI21.py +0 -177
- webscout/Provider/AISEARCH/DeepFind.py +0 -254
- webscout/Provider/AISEARCH/felo_search.py +0 -202
- webscout/Provider/AISEARCH/genspark_search.py +0 -324
- webscout/Provider/AISEARCH/hika_search.py +0 -186
- webscout/Provider/AISEARCH/scira_search.py +0 -298
- webscout/Provider/Aitopia.py +0 -316
- webscout/Provider/AllenAI.py +0 -440
- webscout/Provider/Blackboxai.py +0 -791
- webscout/Provider/ChatGPTClone.py +0 -237
- webscout/Provider/ChatGPTGratis.py +0 -194
- webscout/Provider/Cloudflare.py +0 -324
- webscout/Provider/ExaChat.py +0 -358
- webscout/Provider/Flowith.py +0 -217
- webscout/Provider/FreeGemini.py +0 -250
- webscout/Provider/Glider.py +0 -225
- webscout/Provider/HF_space/__init__.py +0 -0
- webscout/Provider/HF_space/qwen_qwen2.py +0 -206
- webscout/Provider/HuggingFaceChat.py +0 -469
- webscout/Provider/Hunyuan.py +0 -283
- webscout/Provider/LambdaChat.py +0 -411
- webscout/Provider/Llama3.py +0 -259
- webscout/Provider/Nemotron.py +0 -218
- webscout/Provider/OLLAMA.py +0 -396
- webscout/Provider/OPENAI/BLACKBOXAI.py +0 -766
- webscout/Provider/OPENAI/Cloudflare.py +0 -378
- webscout/Provider/OPENAI/FreeGemini.py +0 -283
- webscout/Provider/OPENAI/NEMOTRON.py +0 -232
- webscout/Provider/OPENAI/Qwen3.py +0 -283
- webscout/Provider/OPENAI/api.py +0 -969
- webscout/Provider/OPENAI/c4ai.py +0 -373
- webscout/Provider/OPENAI/chatgptclone.py +0 -494
- webscout/Provider/OPENAI/copilot.py +0 -242
- webscout/Provider/OPENAI/flowith.py +0 -162
- webscout/Provider/OPENAI/freeaichat.py +0 -359
- webscout/Provider/OPENAI/mcpcore.py +0 -389
- webscout/Provider/OPENAI/multichat.py +0 -376
- webscout/Provider/OPENAI/opkfc.py +0 -496
- webscout/Provider/OPENAI/scirachat.py +0 -477
- webscout/Provider/OPENAI/standardinput.py +0 -433
- webscout/Provider/OPENAI/typegpt.py +0 -364
- webscout/Provider/OPENAI/uncovrAI.py +0 -463
- webscout/Provider/OPENAI/venice.py +0 -431
- webscout/Provider/OPENAI/yep.py +0 -382
- webscout/Provider/OpenGPT.py +0 -209
- webscout/Provider/Perplexitylabs.py +0 -415
- webscout/Provider/Reka.py +0 -214
- webscout/Provider/StandardInput.py +0 -290
- webscout/Provider/TTI/aiarta.py +0 -365
- webscout/Provider/TTI/artbit.py +0 -0
- webscout/Provider/TTI/fastflux.py +0 -200
- webscout/Provider/TTI/piclumen.py +0 -203
- webscout/Provider/TTI/pixelmuse.py +0 -225
- webscout/Provider/TTS/gesserit.py +0 -128
- webscout/Provider/TTS/sthir.py +0 -94
- webscout/Provider/TeachAnything.py +0 -229
- webscout/Provider/UNFINISHED/puterjs.py +0 -635
- webscout/Provider/UNFINISHED/test_lmarena.py +0 -119
- webscout/Provider/Venice.py +0 -258
- webscout/Provider/VercelAI.py +0 -253
- webscout/Provider/Writecream.py +0 -246
- webscout/Provider/WritingMate.py +0 -269
- webscout/Provider/asksteve.py +0 -220
- webscout/Provider/chatglm.py +0 -215
- webscout/Provider/copilot.py +0 -425
- webscout/Provider/freeaichat.py +0 -285
- webscout/Provider/granite.py +0 -235
- webscout/Provider/hermes.py +0 -266
- webscout/Provider/koala.py +0 -170
- webscout/Provider/lmarena.py +0 -198
- webscout/Provider/multichat.py +0 -364
- webscout/Provider/scira_chat.py +0 -299
- webscout/Provider/scnet.py +0 -243
- webscout/Provider/talkai.py +0 -194
- webscout/Provider/typegpt.py +0 -289
- webscout/Provider/uncovr.py +0 -368
- webscout/Provider/yep.py +0 -389
- webscout/litagent/Readme.md +0 -276
- webscout/litprinter/__init__.py +0 -59
- webscout/swiftcli/Readme.md +0 -323
- webscout/tempid.py +0 -128
- webscout/webscout_search.py +0 -1184
- webscout/webscout_search_async.py +0 -654
- webscout/yep_search.py +0 -347
- webscout/zeroart/README.md +0 -89
- webscout-8.2.9.dist-info/METADATA +0 -1033
- webscout-8.2.9.dist-info/RECORD +0 -289
- {webscout-8.2.9.dist-info → webscout-2026.1.19.dist-info}/licenses/LICENSE.md +0 -0
- {webscout-8.2.9.dist-info → webscout-2026.1.19.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,365 @@
|
|
|
1
|
+
from typing import Any, Dict, Generator, Optional, Union, cast
|
|
2
|
+
|
|
3
|
+
from curl_cffi import CurlError
|
|
4
|
+
from curl_cffi.requests import Session
|
|
5
|
+
|
|
6
|
+
from webscout import exceptions
|
|
7
|
+
from webscout.AIbase import Provider, Response
|
|
8
|
+
from webscout.AIutel import AwesomePrompts, Conversation, Optimizers
|
|
9
|
+
from webscout.litagent import LitAgent
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
class ClaudeOnline(Provider):
|
|
13
|
+
"""
|
|
14
|
+
A class to interact with the Claude Online API (claude.online/chat).
|
|
15
|
+
|
|
16
|
+
This provider implements the reverse-engineered API from claude.online,
|
|
17
|
+
providing access to Claude AI through their web interface backend.
|
|
18
|
+
"""
|
|
19
|
+
required_auth = False
|
|
20
|
+
AVAILABLE_MODELS = ["claude-online"]
|
|
21
|
+
|
|
22
|
+
def __init__(
|
|
23
|
+
self,
|
|
24
|
+
is_conversation: bool = True,
|
|
25
|
+
max_tokens: int = 2049,
|
|
26
|
+
timeout: int = 30,
|
|
27
|
+
intro: Optional[str] = None,
|
|
28
|
+
filepath: Optional[str] = None,
|
|
29
|
+
update_file: bool = True,
|
|
30
|
+
proxies: dict = {},
|
|
31
|
+
history_offset: int = 10250,
|
|
32
|
+
act: Optional[str] = None,
|
|
33
|
+
system_prompt: str = "You are a helpful assistant.",
|
|
34
|
+
model: str = "claude-online"
|
|
35
|
+
):
|
|
36
|
+
"""
|
|
37
|
+
Initializes the Claude Online API client.
|
|
38
|
+
|
|
39
|
+
Args:
|
|
40
|
+
is_conversation: Whether the provider is in conversation mode.
|
|
41
|
+
max_tokens: Maximum number of tokens to sample.
|
|
42
|
+
timeout: Timeout for API requests.
|
|
43
|
+
intro: Introduction message for the conversation.
|
|
44
|
+
filepath: Filepath for storing conversation history.
|
|
45
|
+
update_file: Whether to update the conversation history file.
|
|
46
|
+
proxies: Proxies for the API requests.
|
|
47
|
+
history_offset: Offset for conversation history.
|
|
48
|
+
act: Act for the conversation.
|
|
49
|
+
system_prompt: The system prompt to define the assistant's role.
|
|
50
|
+
model: Model to use (only "claude-online" supported).
|
|
51
|
+
"""
|
|
52
|
+
if model not in self.AVAILABLE_MODELS:
|
|
53
|
+
raise ValueError(f"Invalid model: {model}. Choose from: {self.AVAILABLE_MODELS}")
|
|
54
|
+
|
|
55
|
+
# API endpoints
|
|
56
|
+
self.chat_url = "https://wewordle.org/gptapi/v1/web/turbo"
|
|
57
|
+
self.limit_url = "https://wewordle.org/gptapi/v1/web/get_limit"
|
|
58
|
+
|
|
59
|
+
# Initialize LitAgent for user agent generation
|
|
60
|
+
self.agent = LitAgent()
|
|
61
|
+
|
|
62
|
+
# Generate browser fingerprint
|
|
63
|
+
self.fingerprint = self.agent.generate_fingerprint(browser="chrome")
|
|
64
|
+
|
|
65
|
+
# Setup headers to mimic browser requests
|
|
66
|
+
self.headers = {
|
|
67
|
+
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.0.0 Safari/537.36',
|
|
68
|
+
'Accept': 'application/json',
|
|
69
|
+
'Content-Type': 'application/json',
|
|
70
|
+
'sec-ch-ua': '"Google Chrome";v="141", "Not?A_Brand";v="8", "Chromium";v="141"',
|
|
71
|
+
'sec-ch-ua-mobile': '?0',
|
|
72
|
+
'sec-ch-ua-platform': '"Windows"',
|
|
73
|
+
'Referer': 'https://claude.online/',
|
|
74
|
+
'Origin': 'https://claude.online',
|
|
75
|
+
'Sec-Fetch-Dest': 'empty',
|
|
76
|
+
'Sec-Fetch-Mode': 'cors',
|
|
77
|
+
'Sec-Fetch-Site': 'cross-site',
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
# Initialize curl_cffi Session
|
|
81
|
+
self.session = Session()
|
|
82
|
+
self.session.headers.update(self.headers)
|
|
83
|
+
if proxies:
|
|
84
|
+
self.session.proxies.update(proxies)
|
|
85
|
+
|
|
86
|
+
self.system_prompt = system_prompt
|
|
87
|
+
self.is_conversation = is_conversation
|
|
88
|
+
self.max_tokens_to_sample = max_tokens
|
|
89
|
+
self.timeout = timeout
|
|
90
|
+
self.last_response = {}
|
|
91
|
+
self.model = model
|
|
92
|
+
|
|
93
|
+
self.__available_optimizers = (
|
|
94
|
+
method
|
|
95
|
+
for method in dir(Optimizers)
|
|
96
|
+
if callable(getattr(Optimizers, method)) and not method.startswith("__")
|
|
97
|
+
)
|
|
98
|
+
|
|
99
|
+
self.conversation = Conversation(
|
|
100
|
+
is_conversation, self.max_tokens_to_sample, filepath, update_file
|
|
101
|
+
)
|
|
102
|
+
self.conversation.history_offset = history_offset
|
|
103
|
+
|
|
104
|
+
if act:
|
|
105
|
+
self.conversation.intro = AwesomePrompts().get_act(cast(Union[str, int], act), default=self.conversation.intro, case_insensitive=True
|
|
106
|
+
) or self.conversation.intro
|
|
107
|
+
elif intro:
|
|
108
|
+
self.conversation.intro = intro
|
|
109
|
+
|
|
110
|
+
def _make_request(self, url: str, payload: Optional[Dict] = None, method: str = "POST") -> Dict:
|
|
111
|
+
"""
|
|
112
|
+
Make a request to the API with error handling.
|
|
113
|
+
|
|
114
|
+
Args:
|
|
115
|
+
url: The URL to request
|
|
116
|
+
payload: Request payload (for POST requests)
|
|
117
|
+
method: HTTP method
|
|
118
|
+
|
|
119
|
+
Returns:
|
|
120
|
+
Parsed JSON response
|
|
121
|
+
|
|
122
|
+
Raises:
|
|
123
|
+
FailedToGenerateResponseError: If request fails
|
|
124
|
+
"""
|
|
125
|
+
try:
|
|
126
|
+
if method == "POST":
|
|
127
|
+
response = self.session.post(
|
|
128
|
+
url,
|
|
129
|
+
json=payload,
|
|
130
|
+
timeout=self.timeout,
|
|
131
|
+
impersonate="chrome110"
|
|
132
|
+
)
|
|
133
|
+
else:
|
|
134
|
+
response = self.session.get(
|
|
135
|
+
url,
|
|
136
|
+
timeout=self.timeout,
|
|
137
|
+
impersonate="chrome110"
|
|
138
|
+
)
|
|
139
|
+
|
|
140
|
+
response.raise_for_status()
|
|
141
|
+
|
|
142
|
+
# Check rate limit headers
|
|
143
|
+
remaining_str = response.headers.get('ratelimit-remaining')
|
|
144
|
+
if remaining_str is not None:
|
|
145
|
+
try:
|
|
146
|
+
remaining = int(remaining_str)
|
|
147
|
+
if remaining <= 0:
|
|
148
|
+
reset_time_str = response.headers.get('ratelimit-reset', '60')
|
|
149
|
+
reset_time = int(reset_time_str) if reset_time_str else 60
|
|
150
|
+
raise exceptions.FailedToGenerateResponseError(
|
|
151
|
+
f"Rate limit exceeded. Resets in {reset_time} seconds."
|
|
152
|
+
)
|
|
153
|
+
except (ValueError, TypeError):
|
|
154
|
+
pass
|
|
155
|
+
|
|
156
|
+
return response.json()
|
|
157
|
+
|
|
158
|
+
except CurlError as e:
|
|
159
|
+
raise exceptions.FailedToGenerateResponseError(f"Request failed (CurlError): {str(e)}")
|
|
160
|
+
except Exception as e:
|
|
161
|
+
raise exceptions.FailedToGenerateResponseError(f"Request failed ({type(e).__name__}): {str(e)}")
|
|
162
|
+
|
|
163
|
+
def get_remaining_limit(self) -> Dict[str, Union[int, float]]:
|
|
164
|
+
"""
|
|
165
|
+
Get the current rate limit status.
|
|
166
|
+
|
|
167
|
+
Returns:
|
|
168
|
+
Dict containing limit information:
|
|
169
|
+
- limit: Remaining requests
|
|
170
|
+
- fullLimit: Unknown parameter (possibly daily limit)
|
|
171
|
+
"""
|
|
172
|
+
try:
|
|
173
|
+
response = self._make_request(self.limit_url, method="GET")
|
|
174
|
+
return {
|
|
175
|
+
"limit": response.get("limit", 0),
|
|
176
|
+
"fullLimit": response.get("fullLimit", 0)
|
|
177
|
+
}
|
|
178
|
+
except Exception:
|
|
179
|
+
# Return default values if request fails
|
|
180
|
+
return {"limit": 0, "fullLimit": 0}
|
|
181
|
+
|
|
182
|
+
def create_message(self, content: str, role: str = "user") -> Dict[str, str]:
|
|
183
|
+
"""
|
|
184
|
+
Create a message dictionary.
|
|
185
|
+
|
|
186
|
+
Args:
|
|
187
|
+
content: Message content
|
|
188
|
+
role: Message role (user/assistant)
|
|
189
|
+
|
|
190
|
+
Returns:
|
|
191
|
+
Message dictionary
|
|
192
|
+
"""
|
|
193
|
+
return {
|
|
194
|
+
"content": content,
|
|
195
|
+
"role": role
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
def ask(
|
|
199
|
+
self,
|
|
200
|
+
prompt: str,
|
|
201
|
+
stream: bool = False,
|
|
202
|
+
raw: bool = False,
|
|
203
|
+
optimizer: Optional[str] = None,
|
|
204
|
+
conversationally: bool = False,
|
|
205
|
+
**kwargs: Any,
|
|
206
|
+
) -> Response:
|
|
207
|
+
"""
|
|
208
|
+
Send a chat message and get response.
|
|
209
|
+
|
|
210
|
+
Args:
|
|
211
|
+
prompt: The message to send
|
|
212
|
+
stream: Whether to stream the response (not supported by this API)
|
|
213
|
+
raw: Whether to return raw response
|
|
214
|
+
optimizer: Optimizer to use for the prompt
|
|
215
|
+
conversationally: Whether to generate the prompt conversationally
|
|
216
|
+
|
|
217
|
+
Returns:
|
|
218
|
+
Dict containing response or Generator for streaming
|
|
219
|
+
"""
|
|
220
|
+
conversation_prompt = self.conversation.gen_complete_prompt(prompt)
|
|
221
|
+
if optimizer:
|
|
222
|
+
if optimizer in self.__available_optimizers:
|
|
223
|
+
conversation_prompt = getattr(Optimizers, optimizer)(
|
|
224
|
+
conversation_prompt if conversationally else prompt
|
|
225
|
+
)
|
|
226
|
+
else:
|
|
227
|
+
raise Exception(f"Optimizer is not one of {self.__available_optimizers}")
|
|
228
|
+
|
|
229
|
+
# Prepare messages (only current message supported by this API)
|
|
230
|
+
messages = [self.create_message(conversation_prompt, "user")]
|
|
231
|
+
|
|
232
|
+
payload = {
|
|
233
|
+
"messages": messages
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
def for_stream():
|
|
237
|
+
# This API doesn't support streaming, so simulate it
|
|
238
|
+
try:
|
|
239
|
+
response = self._make_request(self.chat_url, payload)
|
|
240
|
+
|
|
241
|
+
# Extract message content
|
|
242
|
+
if "message" in response and "content" in response["message"]:
|
|
243
|
+
content = response["message"]["content"]
|
|
244
|
+
|
|
245
|
+
# Simulate streaming by yielding chunks
|
|
246
|
+
words = content.split()
|
|
247
|
+
current_chunk = ""
|
|
248
|
+
for word in words:
|
|
249
|
+
current_chunk += word + " "
|
|
250
|
+
if len(current_chunk) > 50: # Yield chunks of ~50 characters
|
|
251
|
+
yield dict(text=current_chunk.strip())
|
|
252
|
+
current_chunk = ""
|
|
253
|
+
if current_chunk:
|
|
254
|
+
yield dict(text=current_chunk.strip())
|
|
255
|
+
|
|
256
|
+
self.last_response = {"text": content}
|
|
257
|
+
self.conversation.update_chat_history(prompt, content)
|
|
258
|
+
else:
|
|
259
|
+
raise exceptions.FailedToGenerateResponseError("Unexpected response format")
|
|
260
|
+
|
|
261
|
+
except Exception as e:
|
|
262
|
+
raise exceptions.FailedToGenerateResponseError(f"Chat request failed: {str(e)}")
|
|
263
|
+
|
|
264
|
+
def for_non_stream():
|
|
265
|
+
try:
|
|
266
|
+
response = self._make_request(self.chat_url, payload)
|
|
267
|
+
|
|
268
|
+
# Extract message content
|
|
269
|
+
if "message" in response and "content" in response["message"]:
|
|
270
|
+
content = response["message"]["content"]
|
|
271
|
+
self.last_response = {"text": content}
|
|
272
|
+
self.conversation.update_chat_history(prompt, content)
|
|
273
|
+
return self.last_response if not raw else content
|
|
274
|
+
else:
|
|
275
|
+
raise exceptions.FailedToGenerateResponseError("Unexpected response format")
|
|
276
|
+
|
|
277
|
+
except Exception as e:
|
|
278
|
+
raise exceptions.FailedToGenerateResponseError(f"Chat request failed: {str(e)}")
|
|
279
|
+
|
|
280
|
+
return for_stream() if stream else for_non_stream()
|
|
281
|
+
|
|
282
|
+
def chat(
|
|
283
|
+
self,
|
|
284
|
+
prompt: str,
|
|
285
|
+
stream: bool = False,
|
|
286
|
+
optimizer: Optional[str] = None,
|
|
287
|
+
conversationally: bool = False,
|
|
288
|
+
**kwargs: Any,
|
|
289
|
+
) -> Union[str, Generator[str, None, None]]:
|
|
290
|
+
"""
|
|
291
|
+
Generate a response from Claude Online.
|
|
292
|
+
|
|
293
|
+
Args:
|
|
294
|
+
prompt: The prompt to send
|
|
295
|
+
stream: Whether to stream the response
|
|
296
|
+
optimizer: Optimizer to use
|
|
297
|
+
conversationally: Whether to generate conversationally
|
|
298
|
+
**kwargs: Additional parameters including raw.
|
|
299
|
+
|
|
300
|
+
Returns:
|
|
301
|
+
Response string or generator for streaming
|
|
302
|
+
"""
|
|
303
|
+
raw = kwargs.get("raw", False)
|
|
304
|
+
if stream:
|
|
305
|
+
def for_stream_chat():
|
|
306
|
+
gen = self.ask(
|
|
307
|
+
prompt, stream=True, raw=raw,
|
|
308
|
+
optimizer=optimizer, conversationally=conversationally
|
|
309
|
+
)
|
|
310
|
+
if hasattr(gen, "__iter__"):
|
|
311
|
+
for response in gen:
|
|
312
|
+
if raw:
|
|
313
|
+
yield cast(str, response)
|
|
314
|
+
else:
|
|
315
|
+
yield self.get_message(response)
|
|
316
|
+
return for_stream_chat()
|
|
317
|
+
else:
|
|
318
|
+
response_data = self.ask(
|
|
319
|
+
prompt, stream=False, raw=raw,
|
|
320
|
+
optimizer=optimizer, conversationally=conversationally
|
|
321
|
+
)
|
|
322
|
+
if raw:
|
|
323
|
+
return cast(str, response_data)
|
|
324
|
+
return self.get_message(response_data)
|
|
325
|
+
|
|
326
|
+
def get_message(self, response: Response) -> str:
|
|
327
|
+
"""
|
|
328
|
+
Extract message from response.
|
|
329
|
+
|
|
330
|
+
Args:
|
|
331
|
+
response: Response dictionary
|
|
332
|
+
|
|
333
|
+
Returns:
|
|
334
|
+
Message content
|
|
335
|
+
"""
|
|
336
|
+
if not isinstance(response, dict):
|
|
337
|
+
return str(response)
|
|
338
|
+
return cast(Dict[str, Any], response).get("text", "")
|
|
339
|
+
|
|
340
|
+
|
|
341
|
+
if __name__ == "__main__":
|
|
342
|
+
print("-" * 80)
|
|
343
|
+
print(f"{'Model':<50} {'Status':<10} {'Response'}")
|
|
344
|
+
print("-" * 80)
|
|
345
|
+
|
|
346
|
+
try:
|
|
347
|
+
ai = ClaudeOnline(timeout=60)
|
|
348
|
+
limits = ai.get_remaining_limit()
|
|
349
|
+
print(f"Rate limits - Remaining: {limits['limit']}, Full limit: {limits['fullLimit']}")
|
|
350
|
+
|
|
351
|
+
if limits['limit'] > 0:
|
|
352
|
+
response = ai.chat("Say 'Hello World' in one word", stream=False)
|
|
353
|
+
if isinstance(response, str) and response.strip():
|
|
354
|
+
status = "✓"
|
|
355
|
+
clean_text = response.strip().encode('utf-8', errors='ignore').decode('utf-8')
|
|
356
|
+
display_text = clean_text[:50] + "..." if len(clean_text) > 50 else clean_text
|
|
357
|
+
else:
|
|
358
|
+
status = "✗"
|
|
359
|
+
display_text = "Empty response or invalid type"
|
|
360
|
+
print(f"{'claude-online':<50} {status:<10} {display_text}")
|
|
361
|
+
else:
|
|
362
|
+
print(f"{'claude-online':<50} {'✗':<10} Rate limit exceeded")
|
|
363
|
+
|
|
364
|
+
except Exception as e:
|
|
365
|
+
print(f"{'claude-online':<50} {'✗':<10} {str(e)}")
|