webscout 7.9__tar.gz → 8.0__tar.gz
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-7.9/webscout.egg-info → webscout-8.0}/PKG-INFO +2 -2
- {webscout-7.9 → webscout-8.0}/README.md +1 -1
- webscout-8.0/webscout/Extra/GitToolkit/__init__.py +10 -0
- webscout-8.0/webscout/Extra/GitToolkit/gitapi/__init__.py +12 -0
- webscout-8.0/webscout/Extra/GitToolkit/gitapi/repository.py +195 -0
- webscout-8.0/webscout/Extra/GitToolkit/gitapi/user.py +96 -0
- webscout-8.0/webscout/Extra/GitToolkit/gitapi/utils.py +62 -0
- webscout-8.0/webscout/Extra/YTToolkit/ytapi/video.py +232 -0
- webscout-8.0/webscout/Provider/AISEARCH/__init__.py +8 -0
- webscout-8.0/webscout/Provider/AISEARCH/hika_search.py +194 -0
- webscout-8.0/webscout/Provider/AISEARCH/monica_search.py +246 -0
- webscout-8.0/webscout/Provider/AISEARCH/scira_search.py +320 -0
- webscout-8.0/webscout/Provider/AISEARCH/webpilotai_search.py +281 -0
- webscout-8.0/webscout/Provider/AllenAI.py +413 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/DeepSeek.py +1 -2
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Deepinfra.py +17 -9
- webscout-8.0/webscout/Provider/ExaAI.py +261 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ExaChat.py +8 -1
- {webscout-7.9 → webscout-8.0}/webscout/Provider/GithubChat.py +2 -1
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Netwrck.py +3 -2
- webscout-8.0/webscout/Provider/OpenGPT.py +199 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/PI.py +39 -24
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Youchat.py +326 -296
- {webscout-7.9 → webscout-8.0}/webscout/Provider/__init__.py +10 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ai4chat.py +58 -56
- {webscout-7.9 → webscout-8.0}/webscout/Provider/akashgpt.py +34 -22
- {webscout-7.9 → webscout-8.0}/webscout/Provider/freeaichat.py +1 -1
- {webscout-7.9 → webscout-8.0}/webscout/Provider/labyrinth.py +121 -20
- webscout-8.0/webscout/Provider/llmchatco.py +306 -0
- webscout-8.0/webscout/Provider/scira_chat.py +271 -0
- webscout-8.0/webscout/Provider/typefully.py +280 -0
- {webscout-7.9 → webscout-8.0}/webscout/version.py +1 -1
- {webscout-7.9 → webscout-8.0}/webscout/webscout_search.py +118 -54
- {webscout-7.9 → webscout-8.0}/webscout/webscout_search_async.py +109 -45
- {webscout-7.9 → webscout-8.0/webscout.egg-info}/PKG-INFO +2 -2
- {webscout-7.9 → webscout-8.0}/webscout.egg-info/SOURCES.txt +14 -0
- webscout-7.9/webscout/Extra/YTToolkit/ytapi/video.py +0 -103
- webscout-7.9/webscout/Provider/AISEARCH/__init__.py +0 -4
- webscout-7.9/webscout/Provider/AllenAI.py +0 -280
- {webscout-7.9 → webscout-8.0}/LICENSE.md +0 -0
- {webscout-7.9 → webscout-8.0}/setup.cfg +0 -0
- {webscout-7.9 → webscout-8.0}/setup.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/AIauto.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/AIbase.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/AIutel.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Bard.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/DWEBS.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/YTdownloader.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/transcriber.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/channel.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/errors.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/extras.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/https.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/patterns.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/playlist.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/pool.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/query.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/stream.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/YTToolkit/ytapi/utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/autocoder/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/autocoder/autocoder.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/autocoder/autocoder_utiles.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/gguf.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/async_utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/base.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/cli.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/mail_tm.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/tempmail/temp_mail_io.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/weather.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Extra/weather_ascii.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/LLM.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/core/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/core/level.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/core/logger.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/handlers/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/handlers/console.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/handlers/file.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/handlers/network.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/styles/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/styles/colors.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/styles/formats.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/styles/text.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/utils/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/utils/detectors.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Litlogger/utils/formatters.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/AI21.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/AISEARCH/DeepFind.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/AISEARCH/ISou.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/AISEARCH/felo_search.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/AISEARCH/genspark_search.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Aitopia.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Andi.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Blackboxai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/C4ai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ChatGPTClone.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ChatGPTES.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ChatGPTGratis.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Chatify.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Cloudflare.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Cohere.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/ElectronHub.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Free2GPT.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/GPTWeb.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Gemini.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Glider.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Groq.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/HF_space/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/HF_space/qwen_qwen2.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/HeckAI.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/HuggingFaceChat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Hunyuan.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Jadve.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Koboldai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/LambdaChat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Llama.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Llama3.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Marcus.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/OLLAMA.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Openai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Perplexitylabs.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Phind.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/PizzaGPT.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/QwenLM.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Reka.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/AiForce/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/AiForce/async_aiforce.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/AiForce/sync_aiforce.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/FreeAIPlayground/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/FreeAIPlayground/async_freeaiplayground.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/FreeAIPlayground/sync_freeaiplayground.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/ImgSys/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/ImgSys/async_imgsys.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/ImgSys/sync_imgsys.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/MagicStudio/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/MagicStudio/async_magicstudio.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/MagicStudio/sync_magicstudio.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/Nexra/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/Nexra/async_nexra.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/Nexra/sync_nexra.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/PollinationsAI/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/PollinationsAI/async_pollinations.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/PollinationsAI/sync_pollinations.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/aiarta/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/aiarta/async_aiarta.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/aiarta/sync_aiarta.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/artbit/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/artbit/async_artbit.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/artbit/sync_artbit.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/fastflux/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/fastflux/async_fastflux.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/fastflux/sync_fastflux.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/huggingface/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/huggingface/async_huggingface.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/huggingface/sync_huggingface.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/piclumen/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/piclumen/async_piclumen.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/piclumen/sync_piclumen.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/pixelmuse/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/pixelmuse/async_pixelmuse.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/pixelmuse/sync_pixelmuse.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/talkai/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/talkai/async_talkai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTI/talkai/sync_talkai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/deepgram.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/elevenlabs.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/gesserit.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/murfai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/parler.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/speechma.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/streamElements.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TTS/utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TeachAnything.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TextPollinationsAI.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/TwoAI.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/Venice.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/VercelAI.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/WebSim.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/WiseCat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/aimathgpt.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/askmyai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/asksteve.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/bagoodex.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/cerebras.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/chatglm.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/cleeai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/copilot.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/elmo.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/flowith.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/gaurish.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/geminiapi.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/geminiprorealtime.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/granite.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/hermes.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/julius.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/koala.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/learnfastai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/lepton.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/llama3mitril.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/llamatutor.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/llmchat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/meta.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/multichat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/promptrefine.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/searchchat.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/sonus.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/talkai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/turboseek.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/tutorai.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/typegpt.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/uncovr.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/x0gpt.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/Provider/yep.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/__main__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/cli.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/conversation.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/exceptions.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/litagent/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/litagent/agent.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/litagent/constants.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/litprinter/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/models.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/optimizers.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/prompt_manager.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/crawler.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/scout.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/search_result.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/text_analyzer.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/text_utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core/web_analyzer.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/core.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/element.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/parsers/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/parsers/html5lib_parser.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/parsers/html_parser.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/parsers/lxml_parser.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/scout/utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/swiftcli/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/tempid.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/update_checker.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/utils.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/yep_search.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/zeroart/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/zeroart/base.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/zeroart/effects.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout/zeroart/fonts.py +0 -0
- {webscout-7.9 → webscout-8.0}/webscout.egg-info/dependency_links.txt +0 -0
- {webscout-7.9 → webscout-8.0}/webscout.egg-info/entry_points.txt +0 -0
- {webscout-7.9 → webscout-8.0}/webscout.egg-info/requires.txt +0 -0
- {webscout-7.9 → webscout-8.0}/webscout.egg-info/top_level.txt +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/__init__.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/classifier.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/keywords.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/language.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/ner.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/normalizer.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/processor.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/sentiment.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/stemmer.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/tagger.py +0 -0
- {webscout-7.9 → webscout-8.0}/webstoken/tokenizer.py +0 -0
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: webscout
|
|
3
|
-
Version:
|
|
3
|
+
Version: 8.0
|
|
4
4
|
Summary: Search for anything using Google, DuckDuckGo, phind.com, Contains AI models, can transcribe yt videos, temporary email and phone number generation, has TTS support, webai (terminal gpt and open interpreter) and offline LLMs and more
|
|
5
5
|
Author: OEvortex
|
|
6
6
|
Author-email: helpingai5@gmail.com
|
|
@@ -868,7 +868,7 @@ a = AndiSearch()
|
|
|
868
868
|
print(a.chat("HelpingAI-9B"))
|
|
869
869
|
```
|
|
870
870
|
|
|
871
|
-
### `LLAMA`, `C4ai`, `Venice`, `Copilot`, `HuggingFaceChat`, `TwoAI`, `HeckAI`, `AllenAI`, `PerplexityLabs`, `AkashGPT`, `DeepSeek`, `WiseCat`, `IBMGranite`, `QwenLM`, `ChatGPTGratis`, `TextPollinationsAI`, `GliderAI`, `Cohere`, `REKA`, `GROQ`, `AsyncGROQ`, `OPENAI`, `AsyncOPENAI`, `KOBOLDAI`, `AsyncKOBOLDAI`, `BLACKBOXAI`, `PhindSearch`, `GEMINI`, `DeepInfra`, `AI4Chat`, `Phindv2`, `OLLAMA`, `AndiSearch`, `PIZZAGPT`, `Sambanova`, `KOALA`, `Meta`, `AskMyAI`, `PiAI`, `Julius`, `YouChat`, `YEPCHAT`, `Cloudflare`, `TurboSeek`, `TeachAnything`, `AI21`, `Chatify`, `X0GPT`, `Cerebras`, `Lepton`, `GEMINIAPI`, `Cleeai`, `Elmo`, `Free2GPT`, `GPTWeb`, `Netwrck`, `LlamaTutor`, `PromptRefine`, `TutorAI`, `ChatGPTES`, `Bagoodex`, `AIMathGPT`, `GaurishCerebras`, `GeminiPro`, `LLMChat`, `Talkai`, `Llama3Mitril`, `Marcus`, `TypeGPT`, `Netwrck`, `MultiChatAI`, `JadveOpenAI`, `ChatGLM`, `NousHermes`, `FreeAIChat`, `ElectronHub`, `GithubChat`, `Flowith`, `SonusAI`, `UncovrAI`, `LabyrinthAI`, `WebSim`, `LambdaChat`, `ChatGPTClone`, `VercelAI`, `ExaChat`, `AskSteve`, `Aitopia`, `SearchChatAI`
|
|
871
|
+
### `LLAMA`, `C4ai`, `Venice`, `Copilot`, `HuggingFaceChat`, `TwoAI`, `HeckAI`, `AllenAI`, `PerplexityLabs`, `AkashGPT`, `DeepSeek`, `WiseCat`, `IBMGranite`, `QwenLM`, `ChatGPTGratis`, `TextPollinationsAI`, `GliderAI`, `Cohere`, `REKA`, `GROQ`, `AsyncGROQ`, `OPENAI`, `AsyncOPENAI`, `KOBOLDAI`, `AsyncKOBOLDAI`, `BLACKBOXAI`, `PhindSearch`, `GEMINI`, `DeepInfra`, `AI4Chat`, `Phindv2`, `OLLAMA`, `AndiSearch`, `PIZZAGPT`, `Sambanova`, `KOALA`, `Meta`, `AskMyAI`, `PiAI`, `Julius`, `YouChat`, `YEPCHAT`, `Cloudflare`, `TurboSeek`, `TeachAnything`, `AI21`, `Chatify`, `X0GPT`, `Cerebras`, `Lepton`, `GEMINIAPI`, `Cleeai`, `Elmo`, `Free2GPT`, `GPTWeb`, `Netwrck`, `LlamaTutor`, `PromptRefine`, `TutorAI`, `ChatGPTES`, `Bagoodex`, `AIMathGPT`, `GaurishCerebras`, `GeminiPro`, `LLMChat`, `Talkai`, `Llama3Mitril`, `Marcus`, `TypeGPT`, `Netwrck`, `MultiChatAI`, `JadveOpenAI`, `ChatGLM`, `NousHermes`, `FreeAIChat`, `ElectronHub`, `GithubChat`, `Flowith`, `SonusAI`, `UncovrAI`, `LabyrinthAI`, `WebSim`, `LambdaChat`, `ChatGPTClone`, `VercelAI`, `ExaChat`, `AskSteve`, `Aitopia`, `SearchChatAI`, `LLMChatCo`, `TypefullyAI`, `ExaAI`, `OpenGPT`, `SciraAI`
|
|
872
872
|
|
|
873
873
|
Code is similar to other providers.
|
|
874
874
|
|
|
@@ -840,7 +840,7 @@ a = AndiSearch()
|
|
|
840
840
|
print(a.chat("HelpingAI-9B"))
|
|
841
841
|
```
|
|
842
842
|
|
|
843
|
-
### `LLAMA`, `C4ai`, `Venice`, `Copilot`, `HuggingFaceChat`, `TwoAI`, `HeckAI`, `AllenAI`, `PerplexityLabs`, `AkashGPT`, `DeepSeek`, `WiseCat`, `IBMGranite`, `QwenLM`, `ChatGPTGratis`, `TextPollinationsAI`, `GliderAI`, `Cohere`, `REKA`, `GROQ`, `AsyncGROQ`, `OPENAI`, `AsyncOPENAI`, `KOBOLDAI`, `AsyncKOBOLDAI`, `BLACKBOXAI`, `PhindSearch`, `GEMINI`, `DeepInfra`, `AI4Chat`, `Phindv2`, `OLLAMA`, `AndiSearch`, `PIZZAGPT`, `Sambanova`, `KOALA`, `Meta`, `AskMyAI`, `PiAI`, `Julius`, `YouChat`, `YEPCHAT`, `Cloudflare`, `TurboSeek`, `TeachAnything`, `AI21`, `Chatify`, `X0GPT`, `Cerebras`, `Lepton`, `GEMINIAPI`, `Cleeai`, `Elmo`, `Free2GPT`, `GPTWeb`, `Netwrck`, `LlamaTutor`, `PromptRefine`, `TutorAI`, `ChatGPTES`, `Bagoodex`, `AIMathGPT`, `GaurishCerebras`, `GeminiPro`, `LLMChat`, `Talkai`, `Llama3Mitril`, `Marcus`, `TypeGPT`, `Netwrck`, `MultiChatAI`, `JadveOpenAI`, `ChatGLM`, `NousHermes`, `FreeAIChat`, `ElectronHub`, `GithubChat`, `Flowith`, `SonusAI`, `UncovrAI`, `LabyrinthAI`, `WebSim`, `LambdaChat`, `ChatGPTClone`, `VercelAI`, `ExaChat`, `AskSteve`, `Aitopia`, `SearchChatAI`
|
|
843
|
+
### `LLAMA`, `C4ai`, `Venice`, `Copilot`, `HuggingFaceChat`, `TwoAI`, `HeckAI`, `AllenAI`, `PerplexityLabs`, `AkashGPT`, `DeepSeek`, `WiseCat`, `IBMGranite`, `QwenLM`, `ChatGPTGratis`, `TextPollinationsAI`, `GliderAI`, `Cohere`, `REKA`, `GROQ`, `AsyncGROQ`, `OPENAI`, `AsyncOPENAI`, `KOBOLDAI`, `AsyncKOBOLDAI`, `BLACKBOXAI`, `PhindSearch`, `GEMINI`, `DeepInfra`, `AI4Chat`, `Phindv2`, `OLLAMA`, `AndiSearch`, `PIZZAGPT`, `Sambanova`, `KOALA`, `Meta`, `AskMyAI`, `PiAI`, `Julius`, `YouChat`, `YEPCHAT`, `Cloudflare`, `TurboSeek`, `TeachAnything`, `AI21`, `Chatify`, `X0GPT`, `Cerebras`, `Lepton`, `GEMINIAPI`, `Cleeai`, `Elmo`, `Free2GPT`, `GPTWeb`, `Netwrck`, `LlamaTutor`, `PromptRefine`, `TutorAI`, `ChatGPTES`, `Bagoodex`, `AIMathGPT`, `GaurishCerebras`, `GeminiPro`, `LLMChat`, `Talkai`, `Llama3Mitril`, `Marcus`, `TypeGPT`, `Netwrck`, `MultiChatAI`, `JadveOpenAI`, `ChatGLM`, `NousHermes`, `FreeAIChat`, `ElectronHub`, `GithubChat`, `Flowith`, `SonusAI`, `UncovrAI`, `LabyrinthAI`, `WebSim`, `LambdaChat`, `ChatGPTClone`, `VercelAI`, `ExaChat`, `AskSteve`, `Aitopia`, `SearchChatAI`, `LLMChatCo`, `TypefullyAI`, `ExaAI`, `OpenGPT`, `SciraAI`
|
|
844
844
|
|
|
845
845
|
Code is similar to other providers.
|
|
846
846
|
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
from typing import List, Dict, Any, Optional
|
|
2
|
+
from .utils import request
|
|
3
|
+
from urllib.parse import quote
|
|
4
|
+
|
|
5
|
+
class Repository:
|
|
6
|
+
"""Class for interacting with GitHub repositories"""
|
|
7
|
+
|
|
8
|
+
def __init__(self, owner: str, repo: str):
|
|
9
|
+
"""
|
|
10
|
+
Initialize repository client
|
|
11
|
+
|
|
12
|
+
Args:
|
|
13
|
+
owner: Repository owner
|
|
14
|
+
repo: Repository name
|
|
15
|
+
"""
|
|
16
|
+
self.owner = owner
|
|
17
|
+
self.repo = repo
|
|
18
|
+
self.base_url = f"https://api.github.com/repos/{owner}/{repo}"
|
|
19
|
+
|
|
20
|
+
def get_info(self) -> Dict[str, Any]:
|
|
21
|
+
"""Get basic repository information"""
|
|
22
|
+
return request(self.base_url)
|
|
23
|
+
|
|
24
|
+
def get_commits(self, page: int = 1, per_page: int = 30, sha: str = None) -> List[Dict[str, Any]]:
|
|
25
|
+
"""
|
|
26
|
+
Get repository commits
|
|
27
|
+
|
|
28
|
+
Args:
|
|
29
|
+
page: Page number
|
|
30
|
+
per_page: Items per page
|
|
31
|
+
sha: SHA or branch name to start listing commits from
|
|
32
|
+
"""
|
|
33
|
+
url = f"{self.base_url}/commits?page={page}&per_page={per_page}"
|
|
34
|
+
if sha:
|
|
35
|
+
url += f"&sha={sha}"
|
|
36
|
+
return request(url)
|
|
37
|
+
|
|
38
|
+
def get_commit(self, sha: str) -> Dict[str, Any]:
|
|
39
|
+
"""Get a specific commit details"""
|
|
40
|
+
url = f"{self.base_url}/commits/{sha}"
|
|
41
|
+
return request(url)
|
|
42
|
+
|
|
43
|
+
def get_pull_requests(self, state: str = "all", page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
44
|
+
"""
|
|
45
|
+
Get repository pull requests
|
|
46
|
+
|
|
47
|
+
Args:
|
|
48
|
+
state: State of PRs to return (open/closed/all)
|
|
49
|
+
page: Page number
|
|
50
|
+
per_page: Items per page
|
|
51
|
+
"""
|
|
52
|
+
url = f"{self.base_url}/pulls?state={state}&page={page}&per_page={per_page}"
|
|
53
|
+
return request(url)
|
|
54
|
+
|
|
55
|
+
def get_pull_request(self, pr_number: int) -> Dict[str, Any]:
|
|
56
|
+
"""Get specific pull request details"""
|
|
57
|
+
url = f"{self.base_url}/pulls/{pr_number}"
|
|
58
|
+
return request(url)
|
|
59
|
+
|
|
60
|
+
def get_pull_request_commits(self, pr_number: int) -> List[Dict[str, Any]]:
|
|
61
|
+
"""Get commits in a specific pull request"""
|
|
62
|
+
url = f"{self.base_url}/pulls/{pr_number}/commits"
|
|
63
|
+
return request(url)
|
|
64
|
+
|
|
65
|
+
def get_pull_request_files(self, pr_number: int) -> List[Dict[str, Any]]:
|
|
66
|
+
"""Get files changed in a specific pull request"""
|
|
67
|
+
url = f"{self.base_url}/pulls/{pr_number}/files"
|
|
68
|
+
return request(url)
|
|
69
|
+
|
|
70
|
+
def get_issues(self, state: str = "all", page: int = 1, per_page: int = 30, labels: str = None) -> List[Dict[str, Any]]:
|
|
71
|
+
"""
|
|
72
|
+
Get repository issues
|
|
73
|
+
|
|
74
|
+
Args:
|
|
75
|
+
state: State of issues to return (open/closed/all)
|
|
76
|
+
page: Page number
|
|
77
|
+
per_page: Items per page
|
|
78
|
+
labels: Comma-separated list of label names
|
|
79
|
+
"""
|
|
80
|
+
url = f"{self.base_url}/issues?state={state}&page={page}&per_page={per_page}"
|
|
81
|
+
if labels:
|
|
82
|
+
url += f"&labels={labels}"
|
|
83
|
+
return request(url)
|
|
84
|
+
|
|
85
|
+
def get_issue(self, issue_number: int) -> Dict[str, Any]:
|
|
86
|
+
"""Get specific issue details"""
|
|
87
|
+
url = f"{self.base_url}/issues/{issue_number}"
|
|
88
|
+
return request(url)
|
|
89
|
+
|
|
90
|
+
def get_issue_comments(self, issue_number: int) -> List[Dict[str, Any]]:
|
|
91
|
+
"""Get comments on a specific issue"""
|
|
92
|
+
url = f"{self.base_url}/issues/{issue_number}/comments"
|
|
93
|
+
return request(url)
|
|
94
|
+
|
|
95
|
+
def get_labels(self) -> List[Dict[str, Any]]:
|
|
96
|
+
"""Get repository labels"""
|
|
97
|
+
url = f"{self.base_url}/labels"
|
|
98
|
+
return request(url)
|
|
99
|
+
|
|
100
|
+
def get_milestones(self, state: str = "all") -> List[Dict[str, Any]]:
|
|
101
|
+
"""Get repository milestones"""
|
|
102
|
+
url = f"{self.base_url}/milestones?state={state}"
|
|
103
|
+
return request(url)
|
|
104
|
+
|
|
105
|
+
def get_contributors(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
106
|
+
"""Get repository contributors"""
|
|
107
|
+
url = f"{self.base_url}/contributors?page={page}&per_page={per_page}"
|
|
108
|
+
return request(url)
|
|
109
|
+
|
|
110
|
+
def get_languages(self) -> Dict[str, int]:
|
|
111
|
+
"""Get repository language breakdown"""
|
|
112
|
+
url = f"{self.base_url}/languages"
|
|
113
|
+
return request(url)
|
|
114
|
+
|
|
115
|
+
def get_teams(self) -> List[Dict[str, Any]]:
|
|
116
|
+
"""Get teams with access to repository"""
|
|
117
|
+
url = f"{self.base_url}/teams"
|
|
118
|
+
return request(url)
|
|
119
|
+
|
|
120
|
+
def get_tags(self) -> List[Dict[str, Any]]:
|
|
121
|
+
"""Get repository tags"""
|
|
122
|
+
url = f"{self.base_url}/tags"
|
|
123
|
+
return request(url)
|
|
124
|
+
|
|
125
|
+
def get_releases(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
126
|
+
"""Get repository releases"""
|
|
127
|
+
url = f"{self.base_url}/releases?page={page}&per_page={per_page}"
|
|
128
|
+
return request(url)
|
|
129
|
+
|
|
130
|
+
def get_release(self, release_id: str) -> Dict[str, Any]:
|
|
131
|
+
"""Get specific release details"""
|
|
132
|
+
url = f"{self.base_url}/releases/{release_id}"
|
|
133
|
+
return request(url)
|
|
134
|
+
|
|
135
|
+
def get_latest_release(self) -> Dict[str, Any]:
|
|
136
|
+
"""Get latest release"""
|
|
137
|
+
url = f"{self.base_url}/releases/latest"
|
|
138
|
+
return request(url)
|
|
139
|
+
|
|
140
|
+
def get_branches(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
141
|
+
"""Get repository branches"""
|
|
142
|
+
url = f"{self.base_url}/branches?page={page}&per_page={per_page}"
|
|
143
|
+
return request(url)
|
|
144
|
+
|
|
145
|
+
def get_branch(self, branch: str) -> Dict[str, Any]:
|
|
146
|
+
"""Get specific branch details"""
|
|
147
|
+
url = f"{self.base_url}/branches/{branch}"
|
|
148
|
+
return request(url)
|
|
149
|
+
|
|
150
|
+
def get_contents(self, path: str = "", ref: Optional[str] = None) -> Dict[str, Any]:
|
|
151
|
+
"""Get contents of file or directory"""
|
|
152
|
+
url = f"{self.base_url}/contents/{quote(path)}"
|
|
153
|
+
if ref:
|
|
154
|
+
url += f"?ref={ref}"
|
|
155
|
+
return request(url)
|
|
156
|
+
|
|
157
|
+
def get_collaborators(self) -> List[Dict[str, Any]]:
|
|
158
|
+
"""Get repository collaborators"""
|
|
159
|
+
url = f"{self.base_url}/collaborators"
|
|
160
|
+
return request(url)
|
|
161
|
+
|
|
162
|
+
def get_workflows(self) -> List[Dict[str, Any]]:
|
|
163
|
+
"""Get repository GitHub Actions workflows"""
|
|
164
|
+
url = f"{self.base_url}/actions/workflows"
|
|
165
|
+
return request(url)
|
|
166
|
+
|
|
167
|
+
def get_workflow_runs(self, workflow_id: str) -> List[Dict[str, Any]]:
|
|
168
|
+
"""Get workflow runs for a specific workflow"""
|
|
169
|
+
url = f"{self.base_url}/actions/workflows/{workflow_id}/runs"
|
|
170
|
+
return request(url)
|
|
171
|
+
|
|
172
|
+
def get_deployments(self) -> List[Dict[str, Any]]:
|
|
173
|
+
"""Get repository deployments"""
|
|
174
|
+
url = f"{self.base_url}/deployments"
|
|
175
|
+
return request(url)
|
|
176
|
+
|
|
177
|
+
def get_traffic(self) -> Dict[str, Any]:
|
|
178
|
+
"""Get repository traffic data"""
|
|
179
|
+
url = f"{self.base_url}/traffic/views"
|
|
180
|
+
return request(url)
|
|
181
|
+
|
|
182
|
+
def get_code_frequency(self) -> List[List[int]]:
|
|
183
|
+
"""Get weekly commit activity"""
|
|
184
|
+
url = f"{self.base_url}/stats/code_frequency"
|
|
185
|
+
return request(url)
|
|
186
|
+
|
|
187
|
+
def get_commit_activity(self) -> List[Dict[str, Any]]:
|
|
188
|
+
"""Get last year of commit activity"""
|
|
189
|
+
url = f"{self.base_url}/stats/commit_activity"
|
|
190
|
+
return request(url)
|
|
191
|
+
|
|
192
|
+
def get_community_profile(self) -> Dict[str, Any]:
|
|
193
|
+
"""Get community profile metrics"""
|
|
194
|
+
url = f"{self.base_url}/community/profile"
|
|
195
|
+
return request(url)
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
from typing import List, Dict, Any
|
|
2
|
+
from .utils import request
|
|
3
|
+
|
|
4
|
+
class User:
|
|
5
|
+
"""Class for interacting with GitHub user data"""
|
|
6
|
+
|
|
7
|
+
def __init__(self, username: str):
|
|
8
|
+
"""
|
|
9
|
+
Initialize user client
|
|
10
|
+
|
|
11
|
+
Args:
|
|
12
|
+
username: GitHub username
|
|
13
|
+
"""
|
|
14
|
+
self.username = username
|
|
15
|
+
self.base_url = f"https://api.github.com/users/{username}"
|
|
16
|
+
|
|
17
|
+
def get_profile(self) -> Dict[str, Any]:
|
|
18
|
+
"""Get user profile information"""
|
|
19
|
+
return request(self.base_url)
|
|
20
|
+
|
|
21
|
+
def get_repositories(self, page: int = 1, per_page: int = 30, type: str = "all") -> List[Dict[str, Any]]:
|
|
22
|
+
"""
|
|
23
|
+
Get user's public repositories
|
|
24
|
+
|
|
25
|
+
Args:
|
|
26
|
+
page: Page number
|
|
27
|
+
per_page: Items per page
|
|
28
|
+
type: Type of repositories (all/owner/member)
|
|
29
|
+
"""
|
|
30
|
+
url = f"{self.base_url}/repos?page={page}&per_page={per_page}&type={type}"
|
|
31
|
+
return request(url)
|
|
32
|
+
|
|
33
|
+
def get_starred(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
34
|
+
"""Get repositories starred by user"""
|
|
35
|
+
url = f"{self.base_url}/starred?page={page}&per_page={per_page}"
|
|
36
|
+
return request(url)
|
|
37
|
+
|
|
38
|
+
def get_followers(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
39
|
+
"""Get user's followers"""
|
|
40
|
+
url = f"{self.base_url}/followers?page={page}&per_page={per_page}"
|
|
41
|
+
return request(url)
|
|
42
|
+
|
|
43
|
+
def get_following(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
44
|
+
"""Get users followed by this user"""
|
|
45
|
+
url = f"{self.base_url}/following?page={page}&per_page={per_page}"
|
|
46
|
+
return request(url)
|
|
47
|
+
|
|
48
|
+
def get_gists(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
49
|
+
"""Get user's public gists"""
|
|
50
|
+
url = f"{self.base_url}/gists?page={page}&per_page={per_page}"
|
|
51
|
+
return request(url)
|
|
52
|
+
|
|
53
|
+
def get_organizations(self) -> List[Dict[str, Any]]:
|
|
54
|
+
"""Get user's organizations"""
|
|
55
|
+
url = f"{self.base_url}/orgs"
|
|
56
|
+
return request(url)
|
|
57
|
+
|
|
58
|
+
def get_received_events(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
59
|
+
"""Get events received by user"""
|
|
60
|
+
url = f"{self.base_url}/received_events?page={page}&per_page={per_page}"
|
|
61
|
+
return request(url)
|
|
62
|
+
|
|
63
|
+
def get_public_events(self, page: int = 1, per_page: int = 30) -> List[Dict[str, Any]]:
|
|
64
|
+
"""Get user's public events"""
|
|
65
|
+
url = f"{self.base_url}/events/public?page={page}&per_page={per_page}"
|
|
66
|
+
return request(url)
|
|
67
|
+
|
|
68
|
+
def get_starred_gists(self) -> List[Dict[str, Any]]:
|
|
69
|
+
"""Get gists starred by user"""
|
|
70
|
+
url = f"{self.base_url}/starred_gists"
|
|
71
|
+
return request(url)
|
|
72
|
+
|
|
73
|
+
def get_subscriptions(self) -> List[Dict[str, Any]]:
|
|
74
|
+
"""Get repositories user is watching"""
|
|
75
|
+
url = f"{self.base_url}/subscriptions"
|
|
76
|
+
return request(url)
|
|
77
|
+
|
|
78
|
+
def get_hovercard(self) -> Dict[str, Any]:
|
|
79
|
+
"""Get user's hovercard information"""
|
|
80
|
+
url = f"{self.base_url}/hovercard"
|
|
81
|
+
return request(url)
|
|
82
|
+
|
|
83
|
+
def get_installation(self) -> Dict[str, Any]:
|
|
84
|
+
"""Get user's GitHub App installations"""
|
|
85
|
+
url = f"{self.base_url}/installation"
|
|
86
|
+
return request(url)
|
|
87
|
+
|
|
88
|
+
def get_keys(self) -> List[Dict[str, Any]]:
|
|
89
|
+
"""Get user's public SSH keys"""
|
|
90
|
+
url = f"{self.base_url}/keys"
|
|
91
|
+
return request(url)
|
|
92
|
+
|
|
93
|
+
def get_gpg_keys(self) -> List[Dict[str, Any]]:
|
|
94
|
+
"""Get user's public GPG keys"""
|
|
95
|
+
url = f"{self.base_url}/gpg_keys"
|
|
96
|
+
return request(url)
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
from urllib.request import Request, urlopen
|
|
2
|
+
from urllib.error import HTTPError
|
|
3
|
+
import json
|
|
4
|
+
from typing import Dict, Any
|
|
5
|
+
from webscout.litagent import LitAgent
|
|
6
|
+
|
|
7
|
+
class GitError(Exception):
|
|
8
|
+
"""Base exception for GitHub API errors"""
|
|
9
|
+
pass
|
|
10
|
+
|
|
11
|
+
class RateLimitError(GitError):
|
|
12
|
+
"""Raised when hitting GitHub API rate limits"""
|
|
13
|
+
pass
|
|
14
|
+
|
|
15
|
+
class NotFoundError(GitError):
|
|
16
|
+
"""Raised when resource is not found"""
|
|
17
|
+
pass
|
|
18
|
+
|
|
19
|
+
class RequestError(GitError):
|
|
20
|
+
"""Raised for general request errors"""
|
|
21
|
+
pass
|
|
22
|
+
|
|
23
|
+
_USER_AGENT_GENERATOR = LitAgent()
|
|
24
|
+
|
|
25
|
+
def request(url: str, retry_attempts: int = 3) -> Dict[str, Any]:
|
|
26
|
+
"""
|
|
27
|
+
Send a request to GitHub API with retry mechanism
|
|
28
|
+
|
|
29
|
+
Args:
|
|
30
|
+
url: GitHub API endpoint URL
|
|
31
|
+
retry_attempts: Number of retry attempts
|
|
32
|
+
|
|
33
|
+
Returns:
|
|
34
|
+
Parsed JSON response
|
|
35
|
+
|
|
36
|
+
Raises:
|
|
37
|
+
NotFoundError: If resource not found
|
|
38
|
+
RateLimitError: If rate limited
|
|
39
|
+
RequestError: For other request errors
|
|
40
|
+
"""
|
|
41
|
+
headers = {
|
|
42
|
+
"User-Agent": _USER_AGENT_GENERATOR.random(),
|
|
43
|
+
"Accept": "application/vnd.github+json"
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
for attempt in range(retry_attempts):
|
|
47
|
+
try:
|
|
48
|
+
req = Request(url, headers=headers)
|
|
49
|
+
response = urlopen(req)
|
|
50
|
+
return json.loads(response.read().decode('utf-8'))
|
|
51
|
+
|
|
52
|
+
except HTTPError as e:
|
|
53
|
+
if e.code == 404:
|
|
54
|
+
raise NotFoundError(f"Resource not found: {url}")
|
|
55
|
+
if e.code == 429:
|
|
56
|
+
raise RateLimitError(f"Rate limited on attempt {attempt + 1}")
|
|
57
|
+
if attempt == retry_attempts - 1:
|
|
58
|
+
raise RequestError(f"HTTP Error {e.code}: {e.reason}")
|
|
59
|
+
|
|
60
|
+
except Exception as e:
|
|
61
|
+
if attempt == retry_attempts - 1:
|
|
62
|
+
raise RequestError(f"Request failed: {str(e)}")
|
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
import re
|
|
2
|
+
import json
|
|
3
|
+
from typing import Dict, Any
|
|
4
|
+
from .https import video_data
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
class Video:
|
|
8
|
+
|
|
9
|
+
_HEAD = 'https://www.youtube.com/watch?v='
|
|
10
|
+
|
|
11
|
+
def __init__(self, video_id: str):
|
|
12
|
+
"""
|
|
13
|
+
Represents a YouTube video
|
|
14
|
+
|
|
15
|
+
Parameters
|
|
16
|
+
----------
|
|
17
|
+
video_id : str
|
|
18
|
+
The id or url of the video
|
|
19
|
+
"""
|
|
20
|
+
pattern = re.compile('.be/(.*?)$|=(.*?)$|^(\w{11})$') # noqa
|
|
21
|
+
match = pattern.search(video_id)
|
|
22
|
+
|
|
23
|
+
if not match:
|
|
24
|
+
raise ValueError('Invalid YouTube video ID or URL')
|
|
25
|
+
|
|
26
|
+
self._matched_id = (
|
|
27
|
+
match.group(1)
|
|
28
|
+
or match.group(2)
|
|
29
|
+
or match.group(3)
|
|
30
|
+
)
|
|
31
|
+
|
|
32
|
+
if self._matched_id:
|
|
33
|
+
self._url = self._HEAD + self._matched_id
|
|
34
|
+
self._video_data = video_data(self._matched_id)
|
|
35
|
+
# Extract basic info for fallback
|
|
36
|
+
title_match = re.search('<title>(.*?) - YouTube</title>', self._video_data)
|
|
37
|
+
self.title = title_match.group(1) if title_match else None
|
|
38
|
+
self.id = self._matched_id
|
|
39
|
+
else:
|
|
40
|
+
raise ValueError('Invalid YouTube video ID or URL')
|
|
41
|
+
|
|
42
|
+
def __repr__(self):
|
|
43
|
+
return f'<Video {self._url}>'
|
|
44
|
+
|
|
45
|
+
@property
|
|
46
|
+
def metadata(self) -> Dict[str, Any]:
|
|
47
|
+
"""
|
|
48
|
+
Fetches video metadata in a dict format
|
|
49
|
+
|
|
50
|
+
Returns
|
|
51
|
+
-------
|
|
52
|
+
Dict
|
|
53
|
+
Video metadata in a dict format containing keys: title, id, views, duration, author_id,
|
|
54
|
+
upload_date, url, thumbnails, tags, description, likes, genre, etc.
|
|
55
|
+
"""
|
|
56
|
+
# Multiple patterns to try for video details extraction for robustness
|
|
57
|
+
details_patterns = [
|
|
58
|
+
re.compile('videoDetails\":(.*?)\"isLiveContent\":.*?}'),
|
|
59
|
+
re.compile('videoDetails\":(.*?),\"playerConfig'),
|
|
60
|
+
re.compile('videoDetails\":(.*?),\"playabilityStatus')
|
|
61
|
+
]
|
|
62
|
+
|
|
63
|
+
# Other metadata patterns
|
|
64
|
+
upload_date_pattern = re.compile("<meta itemprop=\"uploadDate\" content=\"(.*?)\">")
|
|
65
|
+
genre_pattern = re.compile("<meta itemprop=\"genre\" content=\"(.*?)\">")
|
|
66
|
+
like_count_patterns = [
|
|
67
|
+
re.compile("iconType\":\"LIKE\"},\"defaultText\":(.*?)}"),
|
|
68
|
+
re.compile('\"likeCount\":\"(\\d+)\"')
|
|
69
|
+
]
|
|
70
|
+
channel_name_pattern = re.compile('"ownerChannelName":"(.*?)"')
|
|
71
|
+
|
|
72
|
+
# Try each pattern for video details
|
|
73
|
+
raw_details_match = None
|
|
74
|
+
for pattern in details_patterns:
|
|
75
|
+
match = pattern.search(self._video_data)
|
|
76
|
+
if match:
|
|
77
|
+
raw_details_match = match
|
|
78
|
+
break
|
|
79
|
+
|
|
80
|
+
if not raw_details_match:
|
|
81
|
+
# Fallback metadata for search results or incomplete video data
|
|
82
|
+
return {
|
|
83
|
+
'title': getattr(self, 'title', None),
|
|
84
|
+
'id': getattr(self, 'id', None),
|
|
85
|
+
'views': getattr(self, 'views', None),
|
|
86
|
+
'streamed': False,
|
|
87
|
+
'duration': None,
|
|
88
|
+
'author_id': None,
|
|
89
|
+
'author_name': None,
|
|
90
|
+
'upload_date': None,
|
|
91
|
+
'url': f"https://www.youtube.com/watch?v={getattr(self, 'id', '')}" if hasattr(self, 'id') else None,
|
|
92
|
+
'thumbnails': None,
|
|
93
|
+
'tags': None,
|
|
94
|
+
'description': None,
|
|
95
|
+
'likes': None,
|
|
96
|
+
'genre': None,
|
|
97
|
+
'is_age_restricted': 'age-restricted' in self._video_data.lower(),
|
|
98
|
+
'is_unlisted': 'unlisted' in self._video_data.lower()
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
raw_details = raw_details_match.group(0)
|
|
102
|
+
|
|
103
|
+
# Extract upload date
|
|
104
|
+
upload_date_match = upload_date_pattern.search(self._video_data)
|
|
105
|
+
upload_date = upload_date_match.group(1) if upload_date_match else None
|
|
106
|
+
|
|
107
|
+
# Extract channel name
|
|
108
|
+
channel_name_match = channel_name_pattern.search(self._video_data)
|
|
109
|
+
channel_name = channel_name_match.group(1) if channel_name_match else None
|
|
110
|
+
|
|
111
|
+
# Parse video details
|
|
112
|
+
try:
|
|
113
|
+
# Clean up the JSON string for parsing
|
|
114
|
+
clean_json = raw_details.replace('videoDetails\":', '')
|
|
115
|
+
# Handle potential JSON parsing issues
|
|
116
|
+
if clean_json.endswith(','):
|
|
117
|
+
clean_json = clean_json[:-1]
|
|
118
|
+
metadata = json.loads(clean_json)
|
|
119
|
+
|
|
120
|
+
data = {
|
|
121
|
+
'title': metadata.get('title'),
|
|
122
|
+
'id': metadata.get('videoId', self._matched_id),
|
|
123
|
+
'views': metadata.get('viewCount'),
|
|
124
|
+
'streamed': metadata.get('isLiveContent', False),
|
|
125
|
+
'duration': metadata.get('lengthSeconds'),
|
|
126
|
+
'author_id': metadata.get('channelId'),
|
|
127
|
+
'author_name': channel_name or metadata.get('author'),
|
|
128
|
+
'upload_date': upload_date,
|
|
129
|
+
'url': f"https://www.youtube.com/watch?v={metadata.get('videoId', self._matched_id)}",
|
|
130
|
+
'thumbnails': metadata.get('thumbnail', {}).get('thumbnails'),
|
|
131
|
+
'tags': metadata.get('keywords'),
|
|
132
|
+
'description': metadata.get('shortDescription'),
|
|
133
|
+
'is_age_restricted': metadata.get('isAgeRestricted', False) or 'age-restricted' in self._video_data.lower(),
|
|
134
|
+
'is_unlisted': 'unlisted' in self._video_data.lower(),
|
|
135
|
+
'is_family_safe': metadata.get('isFamilySafe', True),
|
|
136
|
+
'is_private': metadata.get('isPrivate', False),
|
|
137
|
+
'is_live_content': metadata.get('isLiveContent', False),
|
|
138
|
+
'is_crawlable': metadata.get('isCrawlable', True),
|
|
139
|
+
'allow_ratings': metadata.get('allowRatings', True)
|
|
140
|
+
}
|
|
141
|
+
except (json.JSONDecodeError, KeyError, TypeError) as e:
|
|
142
|
+
# Fallback to basic metadata if JSON parsing fails
|
|
143
|
+
return {
|
|
144
|
+
'title': getattr(self, 'title', None),
|
|
145
|
+
'id': self._matched_id,
|
|
146
|
+
'url': self._url,
|
|
147
|
+
'error': f"Failed to parse video details: {str(e)}"
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
# Try to extract likes count
|
|
151
|
+
likes = None
|
|
152
|
+
for pattern in like_count_patterns:
|
|
153
|
+
try:
|
|
154
|
+
likes_match = pattern.search(self._video_data)
|
|
155
|
+
if likes_match:
|
|
156
|
+
likes_text = likes_match.group(1)
|
|
157
|
+
# Handle different formats of like count
|
|
158
|
+
if '{' in likes_text:
|
|
159
|
+
likes = json.loads(likes_text + '}}}')['accessibility']['accessibilityData']['label'].split(' ')[0].replace(',', '')
|
|
160
|
+
else:
|
|
161
|
+
likes = likes_text
|
|
162
|
+
break
|
|
163
|
+
except (AttributeError, KeyError, json.decoder.JSONDecodeError):
|
|
164
|
+
continue
|
|
165
|
+
|
|
166
|
+
data['likes'] = likes
|
|
167
|
+
|
|
168
|
+
# Try to extract genre
|
|
169
|
+
try:
|
|
170
|
+
genre_match = genre_pattern.search(self._video_data)
|
|
171
|
+
data['genre'] = genre_match.group(1) if genre_match else None
|
|
172
|
+
except AttributeError:
|
|
173
|
+
data['genre'] = None
|
|
174
|
+
|
|
175
|
+
return data
|
|
176
|
+
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
@property
|
|
180
|
+
def embed_html(self) -> str:
|
|
181
|
+
"""
|
|
182
|
+
Get the embed HTML code for this video
|
|
183
|
+
|
|
184
|
+
Returns:
|
|
185
|
+
HTML iframe code for embedding the video
|
|
186
|
+
"""
|
|
187
|
+
return f'<iframe width="560" height="315" src="https://www.youtube.com/embed/{self._matched_id}" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" allowfullscreen></iframe>'
|
|
188
|
+
|
|
189
|
+
@property
|
|
190
|
+
def embed_url(self) -> str:
|
|
191
|
+
"""
|
|
192
|
+
Get the embed URL for this video
|
|
193
|
+
|
|
194
|
+
Returns:
|
|
195
|
+
URL for embedding the video
|
|
196
|
+
"""
|
|
197
|
+
return f'https://www.youtube.com/embed/{self._matched_id}'
|
|
198
|
+
|
|
199
|
+
@property
|
|
200
|
+
def thumbnail_url(self) -> str:
|
|
201
|
+
"""
|
|
202
|
+
Get the thumbnail URL for this video
|
|
203
|
+
|
|
204
|
+
Returns:
|
|
205
|
+
URL of the video thumbnail (high quality)
|
|
206
|
+
"""
|
|
207
|
+
return f'https://i.ytimg.com/vi/{self._matched_id}/hqdefault.jpg'
|
|
208
|
+
|
|
209
|
+
@property
|
|
210
|
+
def thumbnail_urls(self) -> Dict[str, str]:
|
|
211
|
+
"""
|
|
212
|
+
Get all thumbnail URLs for this video in different qualities
|
|
213
|
+
|
|
214
|
+
Returns:
|
|
215
|
+
Dictionary of thumbnail URLs with quality labels
|
|
216
|
+
"""
|
|
217
|
+
return {
|
|
218
|
+
'default': f'https://i.ytimg.com/vi/{self._matched_id}/default.jpg',
|
|
219
|
+
'medium': f'https://i.ytimg.com/vi/{self._matched_id}/mqdefault.jpg',
|
|
220
|
+
'high': f'https://i.ytimg.com/vi/{self._matched_id}/hqdefault.jpg',
|
|
221
|
+
'standard': f'https://i.ytimg.com/vi/{self._matched_id}/sddefault.jpg',
|
|
222
|
+
'maxres': f'https://i.ytimg.com/vi/{self._matched_id}/maxresdefault.jpg'
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
if __name__ == '__main__':
|
|
226
|
+
video = Video('https://www.youtube.com/watch?v=9bZkp7q19f0')
|
|
227
|
+
print(video.metadata)
|
|
228
|
+
|
|
229
|
+
# Example of getting comments
|
|
230
|
+
print("\nFirst 3 comments:")
|
|
231
|
+
for i, comment in enumerate(video.stream_comments(3), 1):
|
|
232
|
+
print(f"{i}. {comment['author']}: {comment['text'][:50]}...")
|