KekikStream 0.2.7__tar.gz → 0.2.8__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.
Files changed (62) hide show
  1. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/ExtractorLoader.py +30 -8
  2. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/PluginBase.py +4 -0
  3. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/Odnoklassniki.py +27 -12
  4. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Managers/UIManager.py +1 -0
  5. kekikstream-0.2.8/KekikStream/Plugins/DiziBox.py +115 -0
  6. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/__init__.py +6 -1
  7. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/requirements.txt +1 -0
  8. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/PKG-INFO +2 -1
  9. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/SOURCES.txt +1 -0
  10. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/requires.txt +1 -0
  11. {kekikstream-0.2.7 → kekikstream-0.2.8}/PKG-INFO +2 -1
  12. {kekikstream-0.2.7 → kekikstream-0.2.8}/setup.py +2 -1
  13. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/CLI/__init__.py +0 -0
  14. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/CLI/check_update.py +0 -0
  15. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/ExtractorBase.py +0 -0
  16. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/ExtractorModels.py +0 -0
  17. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/MediaHandler.py +0 -0
  18. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/PluginLoader.py +0 -0
  19. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/PluginModels.py +0 -0
  20. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Core/__init__.py +0 -0
  21. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/CloseLoad.py +0 -0
  22. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/ContentX.py +0 -0
  23. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/FourCX.py +0 -0
  24. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/FourPichive.py +0 -0
  25. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/FourPlayRu.py +0 -0
  26. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/HDStreamAble.py +0 -0
  27. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/Hotlinger.py +0 -0
  28. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/MailRu.py +0 -0
  29. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/MixPlayHD.py +0 -0
  30. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/OkRuHTTP.py +0 -0
  31. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/OkRuSSL.py +0 -0
  32. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/PeaceMakerst.py +0 -0
  33. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/Pichive.py +0 -0
  34. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/PixelDrain.py +0 -0
  35. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/PlayRu.py +0 -0
  36. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/RapidVid.py +0 -0
  37. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/SibNet.py +0 -0
  38. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/Sobreatsesuyp.py +0 -0
  39. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/TRsTX.py +0 -0
  40. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/TauVideo.py +0 -0
  41. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/TurboImgz.py +0 -0
  42. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/VidMoly.py +0 -0
  43. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/VidMoxy.py +0 -0
  44. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Extractors/VideoSeyred.py +0 -0
  45. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Managers/ExtractorManager.py +0 -0
  46. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Managers/MediaManager.py +0 -0
  47. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Managers/PluginManager.py +0 -0
  48. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Managers/__init__.py +0 -0
  49. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/FilmMakinesi.py +0 -0
  50. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/FullHDFilmizlesene.py +0 -0
  51. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/JetFilmizle.py +0 -0
  52. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/SezonlukDizi.py +0 -0
  53. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/SineWix.py +0 -0
  54. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/Plugins/UgurFilm.py +0 -0
  55. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream/__main__.py +0 -0
  56. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/dependency_links.txt +0 -0
  57. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/entry_points.txt +0 -0
  58. {kekikstream-0.2.7 → kekikstream-0.2.8}/KekikStream.egg-info/top_level.txt +0 -0
  59. {kekikstream-0.2.7 → kekikstream-0.2.8}/LICENSE +0 -0
  60. {kekikstream-0.2.7 → kekikstream-0.2.8}/MANIFEST.in +0 -0
  61. {kekikstream-0.2.7 → kekikstream-0.2.8}/README.md +0 -0
  62. {kekikstream-0.2.7 → kekikstream-0.2.8}/setup.cfg +0 -0
@@ -11,33 +11,53 @@ class ExtractorLoader:
11
11
  self.local_extractors_dir = Path(extractors_dir)
12
12
  self.global_extractors_dir = Path(__file__).parent.parent / extractors_dir
13
13
  if not self.local_extractors_dir.exists() and not self.global_extractors_dir.exists():
14
- konsol.log(f"[red][!] Extractor dizini bulunamadı: {self.extractors_dir}[/red]")
14
+ konsol.log(f"[red][!] Extractor dizini bulunamadı: {self.global_extractors_dir}[/red]")
15
15
  cikis_yap(False)
16
16
 
17
17
  def load_all(self) -> list[ExtractorBase]:
18
18
  extractors = []
19
19
 
20
+ # Global Extractor'ları yükle
20
21
  if self.global_extractors_dir.exists():
21
22
  konsol.log(f"[green][*] Global Extractor dizininden yükleniyor: {self.global_extractors_dir}[/green]")
22
- extractors.extend(self._load_from_directory(self.global_extractors_dir))
23
+ global_extractors = self._load_from_directory(self.global_extractors_dir)
24
+ konsol.log(f"[green]Global Extractor'lar: {[e.__name__ for e in global_extractors]}[/green]")
25
+ extractors.extend(global_extractors)
23
26
 
27
+ # Yerel Extractor'ları yükle
24
28
  if self.local_extractors_dir.exists():
25
29
  konsol.log(f"[green][*] Yerel Extractor dizininden yükleniyor: {self.local_extractors_dir}[/green]")
26
- extractors.extend(self._load_from_directory(self.local_extractors_dir))
30
+ local_extractors = self._load_from_directory(self.local_extractors_dir)
31
+ konsol.log(f"[green]Yerel Extractor'lar: {[e.__name__ for e in local_extractors]}[/green]")
32
+ extractors.extend(local_extractors)
27
33
 
28
- if not extractors:
29
- konsol.print("[yellow][!] Yüklenecek bir Extractor bulunamadı![/yellow]")
34
+ # Benzersizliği sağlama (modül adı + sınıf adı bazında)
35
+ unique_extractors = []
36
+ seen = set()
37
+ for ext in extractors:
38
+ identifier = f"{ext.__module__}.{ext.__name__}"
39
+ if identifier not in seen:
40
+ unique_extractors.append(ext)
41
+ seen.add(identifier)
30
42
 
31
- return extractors
43
+ konsol.log(f"[blue]Sonuç Extractor'lar: {[e.__name__ for e in unique_extractors]}[/blue]")
44
+
45
+ if not unique_extractors:
46
+ konsol.log("[yellow][!] Yüklenecek bir Extractor bulunamadı![/yellow]")
47
+
48
+ return unique_extractors
32
49
 
33
50
  def _load_from_directory(self, directory: Path) -> list[ExtractorBase]:
34
51
  extractors = []
35
52
  for file in os.listdir(directory):
36
53
  if file.endswith(".py") and not file.startswith("__"):
37
54
  module_name = file[:-3]
55
+ konsol.log(f"[cyan]Modül yükleniyor: {module_name}[/cyan]")
38
56
  if extractor := self._load_extractor(directory, module_name):
57
+ konsol.log(f"[magenta]Extractor bulundu: {extractor.__name__}[/magenta]")
39
58
  extractors.append(extractor)
40
59
 
60
+ konsol.log(f"[yellow]{directory} dizininden yüklenen Extractor'lar: {[e.__name__ for e in extractors]}[/yellow]")
41
61
  return extractors
42
62
 
43
63
  def _load_extractor(self, directory: Path, module_name: str):
@@ -50,12 +70,14 @@ class ExtractorLoader:
50
70
  module = importlib.util.module_from_spec(spec)
51
71
  spec.loader.exec_module(module)
52
72
 
73
+ # Yalnızca doğru modülden gelen ExtractorBase sınıflarını yükle
53
74
  for attr in dir(module):
54
75
  obj = getattr(module, attr)
55
- if isinstance(obj, type) and issubclass(obj, ExtractorBase) and obj is not ExtractorBase:
76
+ if obj.__module__ == module_name and isinstance(obj, type) and issubclass(obj, ExtractorBase) and obj is not ExtractorBase:
77
+ konsol.log(f"[green]Yüklenen sınıf: {module_name}.{obj.__name__} ({obj.__module__}.{obj.__name__})[/green]")
56
78
  return obj
57
79
 
58
80
  except Exception as hata:
59
- konsol.print(f"[red][!] Extractor yüklenirken hata oluştu: {module_name}\nHata: {hata}")
81
+ konsol.log(f"[red][!] Extractor yüklenirken hata oluştu: {module_name}\nHata: {hata}")
60
82
 
61
83
  return None
@@ -2,6 +2,7 @@
2
2
 
3
3
  from abc import ABC, abstractmethod
4
4
  from httpx import AsyncClient, Timeout
5
+ from cloudscraper import CloudScraper
5
6
  from .PluginModels import SearchResult, MovieInfo
6
7
  from .MediaHandler import MediaHandler
7
8
  from urllib.parse import urljoin
@@ -21,6 +22,9 @@ class PluginBase(ABC):
21
22
  timeout = Timeout(10.0),
22
23
  )
23
24
  self.media_handler = MediaHandler()
25
+ self.cloudscraper = CloudScraper()
26
+ self.oturum.headers.update(self.cloudscraper.headers)
27
+ self.oturum.cookies.update(self.cloudscraper.cookies)
24
28
 
25
29
  @abstractmethod
26
30
  async def search(self, query: str) -> list[SearchResult]:
@@ -1,5 +1,6 @@
1
1
  # Bu araç @keyiflerolsun tarafından | @KekikAkademi için yazılmıştır.
2
2
 
3
+ from KekikStream.CLI import konsol
3
4
  from KekikStream.Core import ExtractorBase, ExtractResult
4
5
  import re
5
6
  import json
@@ -16,18 +17,11 @@ class Odnoklassniki(ExtractorBase):
16
17
  "User-Agent": "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Mobile Safari/537.36"
17
18
  })
18
19
 
19
- try:
20
- # İlk isteği yap, ancak yönlendirmeyi manuel olarak kontrol et.
21
- istek = await self.oturum.get(url, follow_redirects=False)
22
- if istek.status_code == 302: # Yönlendirme varsa
23
- redirected_url = istek.headers.get("Location")
24
- if not redirected_url:
25
- raise ValueError("Redirect location not found.")
26
-
27
- # Yönlendirilmiş URL'yi kullanarak isteği yeniden yap.
28
- url = redirected_url if redirected_url.startswith("http") else f"https://{redirected_url}"
29
- istek = await self.oturum.get(url)
20
+ if "/video/" in url:
21
+ url = url.replace("/video/", "/videoembed/")
30
22
 
23
+ try:
24
+ istek = await self.fetch_with_redirects(url)
31
25
  istek.raise_for_status()
32
26
  except Exception as e:
33
27
  raise RuntimeError(f"Failed to fetch the URL: {url}, Error: {e}")
@@ -91,4 +85,25 @@ class Odnoklassniki(ExtractorBase):
91
85
  url = best_video,
92
86
  referer = self.main_url,
93
87
  subtitles = []
94
- )
88
+ )
89
+
90
+ async def fetch_with_redirects(self, url, max_redirects=5):
91
+ """Yönlendirmeleri takip eden bir fonksiyon"""
92
+ redirects = 0
93
+ while redirects < max_redirects:
94
+ istek = await self.oturum.get(url, follow_redirects=False)
95
+
96
+ if istek.status_code in [301, 302]: # Yönlendirme varsa
97
+ redirected_url = istek.headers.get("Location")
98
+ print(redirected_url)
99
+ if not redirected_url:
100
+ raise ValueError("Redirect location not found.")
101
+
102
+ url = redirected_url if redirected_url.startswith("http") else f"https://{redirected_url}"
103
+ redirects += 1
104
+ else:
105
+ break # Yönlendirme yoksa çık
106
+
107
+ if redirects == max_redirects:
108
+ raise RuntimeError(f"Max redirects ({max_redirects}) reached.")
109
+ return istek
@@ -9,6 +9,7 @@ import os
9
9
  class UIManager:
10
10
  @staticmethod
11
11
  def clear_console():
12
+ # return True
12
13
  os.system("cls" if os.name == "nt" else "clear")
13
14
 
14
15
  @staticmethod
@@ -0,0 +1,115 @@
1
+ # Bu araç @keyiflerolsun tarafından | @KekikAkademi için yazılmıştır.
2
+
3
+ from KekikStream.Core import PluginBase, SearchResult, SeriesInfo, Episode
4
+ from Kekik.Sifreleme import CryptoJS
5
+ from parsel import Selector
6
+ import re, urllib.parse, base64
7
+
8
+ class DiziBox(PluginBase):
9
+ name = "DiziBox"
10
+ main_url = "https://www.dizibox.plus"
11
+
12
+ async def search(self, query: str) -> list[SearchResult]:
13
+ self.oturum.cookies.update({
14
+ "LockUser" : "true",
15
+ "isTrustedUser" : "true",
16
+ "dbxu" : "1722403730363"
17
+ })
18
+ istek = await self.oturum.get(f"{self.main_url}/?s={query}")
19
+ secici = Selector(istek.text)
20
+
21
+ return [
22
+ SearchResult(
23
+ title = item.css("h3 a::text").get(),
24
+ url = self.fix_url(item.css("h3 a::attr(href)").get()),
25
+ poster = self.fix_url(item.css("img::attr(src)").get()),
26
+ )
27
+ for item in secici.css("article.detailed-article")
28
+ ]
29
+
30
+ async def load_item(self, url: str) -> SeriesInfo:
31
+ istek = await self.oturum.get(url)
32
+ secici = Selector(istek.text)
33
+
34
+ title = secici.css("div.tv-overview h1 a::text").get()
35
+ poster = self.fix_url(secici.css("div.tv-overview figure img::attr(src)").get())
36
+ description = secici.css("div.tv-story p::text").get()
37
+ year = secici.css("a[href*='/yil/']::text").re_first(r"(\d{4})")
38
+ tags = secici.css("a[href*='/tur/']::text").getall()
39
+ rating = secici.css("span.label-imdb b::text").re_first(r"[\d.,]+")
40
+ actors = [actor.css("::text").get() for actor in secici.css("a[href*='/oyuncu/']")]
41
+
42
+ episodes = []
43
+ for sezon_link in secici.css("div#seasons-list a::attr(href)").getall():
44
+ sezon_url = self.fix_url(sezon_link)
45
+ sezon_istek = await self.oturum.get(sezon_url)
46
+ sezon_secici = Selector(sezon_istek.text)
47
+
48
+ for bolum in sezon_secici.css("article.grid-box"):
49
+ ep_title = bolum.css("div.post-title a::text").get()
50
+ ep_href = self.fix_url(bolum.css("div.post-title a::attr(href)").get())
51
+ ep_season = bolum.css("div.post-title a::text").re_first(r"(\d+)\. ?Sezon")
52
+ ep_episode = bolum.css("div.post-title a::text").re_first(r"(\d+)\. ?Bölüm")
53
+
54
+ if ep_title and ep_href:
55
+ episodes.append(Episode(
56
+ season = ep_season,
57
+ episode = ep_episode,
58
+ title = ep_title.strip(),
59
+ url = ep_href,
60
+ ))
61
+
62
+ return SeriesInfo(
63
+ url = url,
64
+ poster = poster,
65
+ title = title,
66
+ description = description,
67
+ tags = tags,
68
+ rating = rating,
69
+ year = year,
70
+ episodes = episodes,
71
+ actors = actors,
72
+ )
73
+
74
+ async def _iframe_decode(self, name:str, iframe_link:str, referer:str) -> list[str]:
75
+ results = []
76
+
77
+ if "/player/king/king.php" in iframe_link:
78
+ print(name, iframe_link)
79
+ elif "/player/moly/moly.php" in iframe_link:
80
+ print(name, iframe_link)
81
+ elif "/player/haydi.php" in iframe_link:
82
+ okru_url = base64.b64decode(iframe_link.split("?v=")[-1]).decode("utf-8")
83
+ results.append(okru_url)
84
+
85
+ return results
86
+
87
+ async def load_links(self, url: str) -> list[str]:
88
+ istek = await self.oturum.get(url)
89
+ secici = Selector(istek.text)
90
+
91
+ iframes = []
92
+ main_iframe = secici.css("div#video-area iframe::attr(src)").get()
93
+ if main_iframe:
94
+ if decoded := await self._iframe_decode(self.name, main_iframe, url):
95
+ iframes.extend(decoded)
96
+
97
+ for alternatif in secici.css("div.video-toolbar option[value]"):
98
+ alt_name = alternatif.css("::text").get()
99
+ alt_link = alternatif.css("::attr(value)").get()
100
+
101
+ if not alt_link:
102
+ continue
103
+
104
+ self.oturum.headers.update({"Referer": url})
105
+ alt_istek = await self.oturum.get(alt_link)
106
+ alt_istek.raise_for_status()
107
+
108
+ alt_secici = Selector(alt_istek.text)
109
+ alt_iframe = alt_secici.css("div#video-area iframe::attr(src)").get()
110
+
111
+ if alt_iframe:
112
+ if decoded := await self._iframe_decode(alt_name, alt_iframe, url):
113
+ iframes.extend(decoded)
114
+
115
+ return iframes
@@ -185,7 +185,12 @@ class KekikStream:
185
185
  konsol.print("[bold red]Uygun Extractor bulunamadı.[/bold red]")
186
186
  return
187
187
 
188
- extract_data = await extractor.extract(selected_link, referer=self.current_plugin.main_url)
188
+ try:
189
+ extract_data = await extractor.extract(selected_link, referer=self.current_plugin.main_url)
190
+ except Exception as hata:
191
+ konsol.print(f"[bold red]{extractor.name} » hata oluştu: {hata}[/bold red]")
192
+ await self.handle_no_results()
193
+ return
189
194
 
190
195
  if isinstance(extract_data, list):
191
196
  selected_data = await self.ui_manager.select_from_list(
@@ -3,6 +3,7 @@ wheel
3
3
 
4
4
  Kekik
5
5
  httpx
6
+ cloudscraper
6
7
  parsel
7
8
 
8
9
  pydantic
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: KekikStream
3
- Version: 0.2.7
3
+ Version: 0.2.8
4
4
  Summary: terminal üzerinden medya içeriği aramanızı ve VLC/MPV gibi popüler medya oynatıcılar aracılığıyla doğrudan izlemenizi sağlayan modüler ve genişletilebilir bir bıdı bıdı
5
5
  Home-page: https://github.com/keyiflerolsun/KekikStream
6
6
  Author: keyiflerolsun
@@ -17,6 +17,7 @@ Requires-Dist: setuptools
17
17
  Requires-Dist: wheel
18
18
  Requires-Dist: Kekik
19
19
  Requires-Dist: httpx
20
+ Requires-Dist: cloudscraper
20
21
  Requires-Dist: parsel
21
22
  Requires-Dist: pydantic
22
23
  Requires-Dist: InquirerPy
@@ -51,6 +51,7 @@ KekikStream/Managers/MediaManager.py
51
51
  KekikStream/Managers/PluginManager.py
52
52
  KekikStream/Managers/UIManager.py
53
53
  KekikStream/Managers/__init__.py
54
+ KekikStream/Plugins/DiziBox.py
54
55
  KekikStream/Plugins/FilmMakinesi.py
55
56
  KekikStream/Plugins/FullHDFilmizlesene.py
56
57
  KekikStream/Plugins/JetFilmizle.py
@@ -2,6 +2,7 @@ setuptools
2
2
  wheel
3
3
  Kekik
4
4
  httpx
5
+ cloudscraper
5
6
  parsel
6
7
  pydantic
7
8
  InquirerPy
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: KekikStream
3
- Version: 0.2.7
3
+ Version: 0.2.8
4
4
  Summary: terminal üzerinden medya içeriği aramanızı ve VLC/MPV gibi popüler medya oynatıcılar aracılığıyla doğrudan izlemenizi sağlayan modüler ve genişletilebilir bir bıdı bıdı
5
5
  Home-page: https://github.com/keyiflerolsun/KekikStream
6
6
  Author: keyiflerolsun
@@ -17,6 +17,7 @@ Requires-Dist: setuptools
17
17
  Requires-Dist: wheel
18
18
  Requires-Dist: Kekik
19
19
  Requires-Dist: httpx
20
+ Requires-Dist: cloudscraper
20
21
  Requires-Dist: parsel
21
22
  Requires-Dist: pydantic
22
23
  Requires-Dist: InquirerPy
@@ -6,7 +6,7 @@ from io import open
6
6
  setup(
7
7
  # ? Genel Bilgiler
8
8
  name = "KekikStream",
9
- version = "0.2.7",
9
+ version = "0.2.8",
10
10
  url = "https://github.com/keyiflerolsun/KekikStream",
11
11
  description = "terminal üzerinden medya içeriği aramanızı ve VLC/MPV gibi popüler medya oynatıcılar aracılığıyla doğrudan izlemenizi sağlayan modüler ve genişletilebilir bir bıdı bıdı",
12
12
  keywords = ["KekikStream", "KekikAkademi", "keyiflerolsun"],
@@ -29,6 +29,7 @@ setup(
29
29
  "wheel",
30
30
  "Kekik",
31
31
  "httpx",
32
+ "cloudscraper",
32
33
  "parsel",
33
34
  "pydantic",
34
35
  "InquirerPy",
File without changes
File without changes
File without changes
File without changes