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,140 @@
|
|
|
1
|
+
"""Yahoo search suggestions engine."""
|
|
2
|
+
|
|
3
|
+
from __future__ import annotations
|
|
4
|
+
|
|
5
|
+
import json
|
|
6
|
+
from typing import Any, Optional
|
|
7
|
+
|
|
8
|
+
from .base import YahooSearchEngine
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
class YahooSuggestions(YahooSearchEngine[str]):
|
|
12
|
+
"""Yahoo search suggestions engine.
|
|
13
|
+
|
|
14
|
+
Provides autocomplete suggestions as you type.
|
|
15
|
+
"""
|
|
16
|
+
|
|
17
|
+
name = "yahoo"
|
|
18
|
+
category = "suggestions"
|
|
19
|
+
|
|
20
|
+
search_url = "https://search.yahoo.com/sugg/gossip/gossip-us-ura"
|
|
21
|
+
search_method = "GET"
|
|
22
|
+
|
|
23
|
+
def build_payload(
|
|
24
|
+
self,
|
|
25
|
+
query: str,
|
|
26
|
+
region: str,
|
|
27
|
+
safesearch: str,
|
|
28
|
+
timelimit: str | None,
|
|
29
|
+
page: int = 1,
|
|
30
|
+
**kwargs: Any,
|
|
31
|
+
) -> dict[str, Any]:
|
|
32
|
+
"""Build suggestions payload.
|
|
33
|
+
|
|
34
|
+
Args:
|
|
35
|
+
query: Partial search query
|
|
36
|
+
region: Region code
|
|
37
|
+
safesearch: Safe search level (unused)
|
|
38
|
+
timelimit: Time limit (unused)
|
|
39
|
+
page: Page number (unused)
|
|
40
|
+
**kwargs: Additional parameters
|
|
41
|
+
|
|
42
|
+
Returns:
|
|
43
|
+
Query parameters
|
|
44
|
+
"""
|
|
45
|
+
payload = {
|
|
46
|
+
"command": query,
|
|
47
|
+
"output": "sd1",
|
|
48
|
+
"nresults": kwargs.get("max_suggestions", 10),
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
return payload
|
|
52
|
+
|
|
53
|
+
def extract_results(self, html_text: str) -> list[str]:
|
|
54
|
+
"""Extract suggestions from JSON response.
|
|
55
|
+
|
|
56
|
+
Args:
|
|
57
|
+
html_text: JSON response text
|
|
58
|
+
|
|
59
|
+
Returns:
|
|
60
|
+
List of suggestion strings
|
|
61
|
+
"""
|
|
62
|
+
try:
|
|
63
|
+
data = json.loads(html_text)
|
|
64
|
+
|
|
65
|
+
# Yahoo returns suggestions in 'r' key
|
|
66
|
+
if "r" in data and isinstance(data["r"], list):
|
|
67
|
+
suggestions = []
|
|
68
|
+
for item in data["r"]:
|
|
69
|
+
if isinstance(item, dict) and "k" in item:
|
|
70
|
+
suggestions.append(item["k"])
|
|
71
|
+
elif isinstance(item, str):
|
|
72
|
+
suggestions.append(item)
|
|
73
|
+
return suggestions
|
|
74
|
+
|
|
75
|
+
return []
|
|
76
|
+
except (json.JSONDecodeError, KeyError, TypeError):
|
|
77
|
+
return []
|
|
78
|
+
|
|
79
|
+
def search(
|
|
80
|
+
self,
|
|
81
|
+
query: str,
|
|
82
|
+
region: str = "us-en",
|
|
83
|
+
safesearch: str = "moderate",
|
|
84
|
+
timelimit: str | None = None,
|
|
85
|
+
page: int = 1,
|
|
86
|
+
max_results: int | None = None,
|
|
87
|
+
**kwargs: Any,
|
|
88
|
+
) -> list[str] | None:
|
|
89
|
+
"""Get search suggestions for a query.
|
|
90
|
+
|
|
91
|
+
Args:
|
|
92
|
+
query: Partial search query
|
|
93
|
+
region: Region code
|
|
94
|
+
safesearch: Safe search level
|
|
95
|
+
timelimit: Time limit
|
|
96
|
+
page: Page number
|
|
97
|
+
max_results: Maximum suggestions
|
|
98
|
+
**kwargs: Additional parameters
|
|
99
|
+
|
|
100
|
+
Returns:
|
|
101
|
+
List of suggestion strings
|
|
102
|
+
"""
|
|
103
|
+
if max_results:
|
|
104
|
+
kwargs["max_suggestions"] = max_results
|
|
105
|
+
|
|
106
|
+
payload = self.build_payload(
|
|
107
|
+
query=query,
|
|
108
|
+
region=region,
|
|
109
|
+
safesearch=safesearch,
|
|
110
|
+
timelimit=timelimit,
|
|
111
|
+
page=page,
|
|
112
|
+
**kwargs
|
|
113
|
+
)
|
|
114
|
+
|
|
115
|
+
response = self.request(self.search_method, self.search_url, params=payload)
|
|
116
|
+
if not response:
|
|
117
|
+
return None
|
|
118
|
+
|
|
119
|
+
suggestions = self.extract_results(response)
|
|
120
|
+
|
|
121
|
+
if max_results:
|
|
122
|
+
suggestions = suggestions[:max_results]
|
|
123
|
+
|
|
124
|
+
return suggestions if suggestions else None
|
|
125
|
+
|
|
126
|
+
def run(self, keywords: str, region: str = "us-en") -> list[str]:
|
|
127
|
+
"""Run suggestions search and return results.
|
|
128
|
+
|
|
129
|
+
Args:
|
|
130
|
+
keywords: Search query.
|
|
131
|
+
region: Region code.
|
|
132
|
+
|
|
133
|
+
Returns:
|
|
134
|
+
List of suggestion strings.
|
|
135
|
+
"""
|
|
136
|
+
results = self.search(
|
|
137
|
+
query=keywords,
|
|
138
|
+
region=region,
|
|
139
|
+
)
|
|
140
|
+
return results if results else []
|
|
@@ -0,0 +1,273 @@
|
|
|
1
|
+
"""Yahoo text search engine with pagination support."""
|
|
2
|
+
|
|
3
|
+
from __future__ import annotations
|
|
4
|
+
|
|
5
|
+
from collections.abc import Mapping
|
|
6
|
+
from typing import Any, Optional
|
|
7
|
+
from urllib.parse import unquote_plus
|
|
8
|
+
|
|
9
|
+
from ...results import TextResult
|
|
10
|
+
from .base import YahooSearchEngine
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def extract_url(u: str) -> str:
|
|
14
|
+
"""Extract and sanitize URL from Yahoo redirect.
|
|
15
|
+
|
|
16
|
+
Yahoo uses /RU= redirect URLs that need to be decoded.
|
|
17
|
+
Example: /url?sa=t&url=https%3A%2F%2Fexample.com
|
|
18
|
+
"""
|
|
19
|
+
if not u:
|
|
20
|
+
return u
|
|
21
|
+
|
|
22
|
+
# Handle /RU= redirect format
|
|
23
|
+
if "/RU=" in u:
|
|
24
|
+
start = u.find("/RU=") + 4
|
|
25
|
+
end = u.find("/RK=", start)
|
|
26
|
+
if end == -1:
|
|
27
|
+
end = len(u)
|
|
28
|
+
return unquote_plus(u[start:end])
|
|
29
|
+
|
|
30
|
+
return u
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
class YahooText(YahooSearchEngine[TextResult]):
|
|
34
|
+
"""Yahoo text search engine with full pagination support.
|
|
35
|
+
|
|
36
|
+
Features:
|
|
37
|
+
- Multi-page navigation like a human
|
|
38
|
+
- Automatic next page detection
|
|
39
|
+
- Clean result extraction
|
|
40
|
+
- Time filter support
|
|
41
|
+
- Region support
|
|
42
|
+
"""
|
|
43
|
+
|
|
44
|
+
name = "yahoo"
|
|
45
|
+
category = "text"
|
|
46
|
+
|
|
47
|
+
search_url = "https://search.yahoo.com/search"
|
|
48
|
+
search_method = "GET"
|
|
49
|
+
|
|
50
|
+
# XPath selectors for result extraction
|
|
51
|
+
items_xpath = "//div[contains(@class, 'compTitle')]"
|
|
52
|
+
elements_xpath: Mapping[str, str] = {
|
|
53
|
+
"title": ".//h3//span//text()",
|
|
54
|
+
"href": ".//a/@href",
|
|
55
|
+
"body": "./following-sibling::div[contains(@class, 'compText')]//text()",
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
def build_payload(
|
|
59
|
+
self,
|
|
60
|
+
query: str,
|
|
61
|
+
region: str,
|
|
62
|
+
safesearch: str,
|
|
63
|
+
timelimit: str | None,
|
|
64
|
+
page: int = 1,
|
|
65
|
+
**kwargs: Any
|
|
66
|
+
) -> dict[str, Any]:
|
|
67
|
+
"""Build search payload for Yahoo.
|
|
68
|
+
|
|
69
|
+
Args:
|
|
70
|
+
query: Search query string
|
|
71
|
+
region: Region code (e.g., 'us-en')
|
|
72
|
+
safesearch: Safe search level
|
|
73
|
+
timelimit: Time limit filter (d=day, w=week, m=month)
|
|
74
|
+
page: Page number (1-indexed)
|
|
75
|
+
**kwargs: Additional parameters
|
|
76
|
+
|
|
77
|
+
Returns:
|
|
78
|
+
Dictionary of query parameters
|
|
79
|
+
"""
|
|
80
|
+
payload = {
|
|
81
|
+
"p": query,
|
|
82
|
+
"ei": "UTF-8",
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
# Pagination: Yahoo uses 'b' parameter for offset
|
|
86
|
+
# Page 1: no b parameter or b=1
|
|
87
|
+
# Page 2: b=8 (shows results 8-14)
|
|
88
|
+
# Page 3: b=15, etc.
|
|
89
|
+
if page > 1:
|
|
90
|
+
payload["b"] = f"{(page - 1) * 7 + 1}"
|
|
91
|
+
|
|
92
|
+
# Time filter
|
|
93
|
+
if timelimit:
|
|
94
|
+
payload["btf"] = timelimit
|
|
95
|
+
|
|
96
|
+
return payload
|
|
97
|
+
|
|
98
|
+
def post_extract_results(self, results: list[TextResult]) -> list[TextResult]:
|
|
99
|
+
"""Post-process and clean extracted results.
|
|
100
|
+
|
|
101
|
+
Args:
|
|
102
|
+
results: Raw extracted results
|
|
103
|
+
|
|
104
|
+
Returns:
|
|
105
|
+
Cleaned and filtered results
|
|
106
|
+
"""
|
|
107
|
+
cleaned_results = []
|
|
108
|
+
|
|
109
|
+
for result in results:
|
|
110
|
+
# Extract real URL from redirect
|
|
111
|
+
if result.href:
|
|
112
|
+
result.href = extract_url(result.href)
|
|
113
|
+
|
|
114
|
+
# Filter out empty results
|
|
115
|
+
if result.title and result.href:
|
|
116
|
+
cleaned_results.append(result)
|
|
117
|
+
|
|
118
|
+
return cleaned_results
|
|
119
|
+
|
|
120
|
+
def search(
|
|
121
|
+
self,
|
|
122
|
+
query: str,
|
|
123
|
+
region: str = "us-en",
|
|
124
|
+
safesearch: str = "moderate",
|
|
125
|
+
timelimit: str | None = None,
|
|
126
|
+
page: int = 1,
|
|
127
|
+
max_results: int | None = None,
|
|
128
|
+
**kwargs: Any,
|
|
129
|
+
) -> list[TextResult] | None:
|
|
130
|
+
"""Search Yahoo with automatic pagination like a human browser.
|
|
131
|
+
|
|
132
|
+
This method automatically follows pagination links to gather results
|
|
133
|
+
across multiple pages, similar to how a human would browse search results.
|
|
134
|
+
|
|
135
|
+
Args:
|
|
136
|
+
query: Search query string
|
|
137
|
+
region: Region code
|
|
138
|
+
safesearch: Safe search level
|
|
139
|
+
timelimit: Time filter (d=day, w=week, m=month, y=year)
|
|
140
|
+
page: Starting page number
|
|
141
|
+
max_results: Maximum number of results to return
|
|
142
|
+
**kwargs: Additional search parameters
|
|
143
|
+
|
|
144
|
+
Returns:
|
|
145
|
+
List of TextResult objects, or None if search fails
|
|
146
|
+
"""
|
|
147
|
+
results = []
|
|
148
|
+
current_page = page
|
|
149
|
+
max_pages = kwargs.get("max_pages", 10) # Limit to prevent infinite loops
|
|
150
|
+
|
|
151
|
+
while current_page <= max_pages:
|
|
152
|
+
# Build payload for current page
|
|
153
|
+
payload = self.build_payload(
|
|
154
|
+
query=query,
|
|
155
|
+
region=region,
|
|
156
|
+
safesearch=safesearch,
|
|
157
|
+
timelimit=timelimit,
|
|
158
|
+
page=current_page,
|
|
159
|
+
**kwargs
|
|
160
|
+
)
|
|
161
|
+
|
|
162
|
+
# Make request
|
|
163
|
+
html_text = self.request(self.search_method, self.search_url, params=payload)
|
|
164
|
+
if not html_text:
|
|
165
|
+
break
|
|
166
|
+
|
|
167
|
+
# Pre-process HTML
|
|
168
|
+
html_text = self.pre_process_html(html_text)
|
|
169
|
+
|
|
170
|
+
# Extract results from current page
|
|
171
|
+
page_results = self.extract_results(html_text)
|
|
172
|
+
if not page_results:
|
|
173
|
+
break
|
|
174
|
+
|
|
175
|
+
results.extend(page_results)
|
|
176
|
+
|
|
177
|
+
# Check if we have enough results
|
|
178
|
+
if max_results and len(results) >= max_results:
|
|
179
|
+
break
|
|
180
|
+
|
|
181
|
+
# Look for next page link
|
|
182
|
+
tree = self.extract_tree(html_text)
|
|
183
|
+
next_links = tree.xpath("//a[contains(text(), 'Next') or contains(@class, 'next')]/@href")
|
|
184
|
+
|
|
185
|
+
if not next_links:
|
|
186
|
+
# Try to find numbered page links
|
|
187
|
+
page_links = tree.xpath(f"//a[contains(text(), '{current_page + 1}')]/@href")
|
|
188
|
+
if not page_links:
|
|
189
|
+
break
|
|
190
|
+
|
|
191
|
+
current_page += 1
|
|
192
|
+
|
|
193
|
+
# Post-process all results
|
|
194
|
+
results = self.post_extract_results(results)
|
|
195
|
+
|
|
196
|
+
# Trim to max_results if specified
|
|
197
|
+
if max_results:
|
|
198
|
+
results = results[:max_results]
|
|
199
|
+
|
|
200
|
+
return results if results else None
|
|
201
|
+
|
|
202
|
+
def search_page(
|
|
203
|
+
self,
|
|
204
|
+
query: str,
|
|
205
|
+
region: str = "us-en",
|
|
206
|
+
safesearch: str = "moderate",
|
|
207
|
+
timelimit: str | None = None,
|
|
208
|
+
page: int = 1,
|
|
209
|
+
**kwargs: Any,
|
|
210
|
+
) -> list[TextResult] | None:
|
|
211
|
+
"""Search a single page (for compatibility).
|
|
212
|
+
|
|
213
|
+
Args:
|
|
214
|
+
query: Search query
|
|
215
|
+
region: Region code
|
|
216
|
+
safesearch: Safe search level
|
|
217
|
+
timelimit: Time filter
|
|
218
|
+
page: Page number
|
|
219
|
+
**kwargs: Additional parameters
|
|
220
|
+
|
|
221
|
+
Returns:
|
|
222
|
+
List of results from the specified page
|
|
223
|
+
"""
|
|
224
|
+
payload = self.build_payload(
|
|
225
|
+
query=query,
|
|
226
|
+
region=region,
|
|
227
|
+
safesearch=safesearch,
|
|
228
|
+
timelimit=timelimit,
|
|
229
|
+
page=page,
|
|
230
|
+
**kwargs
|
|
231
|
+
)
|
|
232
|
+
|
|
233
|
+
html_text = self.request(self.search_method, self.search_url, params=payload)
|
|
234
|
+
if not html_text:
|
|
235
|
+
return None
|
|
236
|
+
|
|
237
|
+
html_text = self.pre_process_html(html_text)
|
|
238
|
+
results = self.extract_results(html_text)
|
|
239
|
+
|
|
240
|
+
return self.post_extract_results(results) if results else None
|
|
241
|
+
|
|
242
|
+
def run(
|
|
243
|
+
self,
|
|
244
|
+
keywords: str,
|
|
245
|
+
region: str = "us-en",
|
|
246
|
+
safesearch: str = "moderate",
|
|
247
|
+
timelimit: str | None = None,
|
|
248
|
+
backend: str = "auto",
|
|
249
|
+
max_results: int | None = None,
|
|
250
|
+
) -> list[dict[str, str]]:
|
|
251
|
+
"""Run text search and return results as dictionaries.
|
|
252
|
+
|
|
253
|
+
Args:
|
|
254
|
+
keywords: Search query.
|
|
255
|
+
region: Region code.
|
|
256
|
+
safesearch: Safe search level.
|
|
257
|
+
timelimit: Time filter.
|
|
258
|
+
backend: Backend type (ignored for Yahoo).
|
|
259
|
+
max_results: Maximum number of results.
|
|
260
|
+
|
|
261
|
+
Returns:
|
|
262
|
+
List of search result dictionaries.
|
|
263
|
+
"""
|
|
264
|
+
results = self.search(
|
|
265
|
+
query=keywords,
|
|
266
|
+
region=region,
|
|
267
|
+
safesearch=safesearch,
|
|
268
|
+
timelimit=timelimit,
|
|
269
|
+
max_results=max_results,
|
|
270
|
+
)
|
|
271
|
+
if results is None:
|
|
272
|
+
return []
|
|
273
|
+
return [result.to_dict() for result in results]
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"""Yahoo translate search."""
|
|
2
|
+
|
|
3
|
+
from __future__ import annotations
|
|
4
|
+
|
|
5
|
+
from .base import YahooSearchEngine
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
class YahooTranslate(YahooSearchEngine):
|
|
9
|
+
"""Yahoo translation."""
|
|
10
|
+
|
|
11
|
+
def build_payload(self, *args, **kwargs) -> dict:
|
|
12
|
+
return {}
|
|
13
|
+
|
|
14
|
+
def run(self, *args, **kwargs) -> list[dict[str, str]]:
|
|
15
|
+
"""Translate text using Yahoo.
|
|
16
|
+
|
|
17
|
+
Not supported.
|
|
18
|
+
"""
|
|
19
|
+
raise NotImplementedError("Yahoo does not support translation")
|