StreamingCommunity 1.7.6__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of StreamingCommunity might be problematic. Click here for more details.

Files changed (97) hide show
  1. StreamingCommunity/Src/Api/Player/Helper/Vixcloud/js_parser.py +140 -0
  2. StreamingCommunity/Src/Api/Player/Helper/Vixcloud/util.py +166 -0
  3. StreamingCommunity/Src/Api/Player/ddl.py +89 -0
  4. StreamingCommunity/Src/Api/Player/maxstream.py +151 -0
  5. StreamingCommunity/Src/Api/Player/supervideo.py +194 -0
  6. StreamingCommunity/Src/Api/Player/vixcloud.py +212 -0
  7. StreamingCommunity/Src/Api/Site/1337xx/__init__.py +50 -0
  8. StreamingCommunity/Src/Api/Site/1337xx/costant.py +15 -0
  9. StreamingCommunity/Src/Api/Site/1337xx/site.py +84 -0
  10. StreamingCommunity/Src/Api/Site/1337xx/title.py +66 -0
  11. StreamingCommunity/Src/Api/Site/altadefinizione/__init__.py +50 -0
  12. StreamingCommunity/Src/Api/Site/altadefinizione/costant.py +15 -0
  13. StreamingCommunity/Src/Api/Site/altadefinizione/film.py +69 -0
  14. StreamingCommunity/Src/Api/Site/altadefinizione/site.py +86 -0
  15. StreamingCommunity/Src/Api/Site/animeunity/__init__.py +50 -0
  16. StreamingCommunity/Src/Api/Site/animeunity/anime.py +126 -0
  17. StreamingCommunity/Src/Api/Site/animeunity/costant.py +15 -0
  18. StreamingCommunity/Src/Api/Site/animeunity/film_serie.py +131 -0
  19. StreamingCommunity/Src/Api/Site/animeunity/site.py +165 -0
  20. StreamingCommunity/Src/Api/Site/animeunity/util/ScrapeSerie.py +97 -0
  21. StreamingCommunity/Src/Api/Site/bitsearch/__init__.py +51 -0
  22. StreamingCommunity/Src/Api/Site/bitsearch/costant.py +15 -0
  23. StreamingCommunity/Src/Api/Site/bitsearch/site.py +84 -0
  24. StreamingCommunity/Src/Api/Site/bitsearch/title.py +47 -0
  25. StreamingCommunity/Src/Api/Site/cb01new/__init__.py +51 -0
  26. StreamingCommunity/Src/Api/Site/cb01new/costant.py +15 -0
  27. StreamingCommunity/Src/Api/Site/cb01new/film.py +69 -0
  28. StreamingCommunity/Src/Api/Site/cb01new/site.py +74 -0
  29. StreamingCommunity/Src/Api/Site/ddlstreamitaly/Player/ScrapeSerie.py +83 -0
  30. StreamingCommunity/Src/Api/Site/ddlstreamitaly/__init__.py +57 -0
  31. StreamingCommunity/Src/Api/Site/ddlstreamitaly/costant.py +16 -0
  32. StreamingCommunity/Src/Api/Site/ddlstreamitaly/series.py +142 -0
  33. StreamingCommunity/Src/Api/Site/ddlstreamitaly/site.py +93 -0
  34. StreamingCommunity/Src/Api/Site/ddlstreamitaly/util/ScrapeSerie.py +83 -0
  35. StreamingCommunity/Src/Api/Site/guardaserie/Player/ScrapeSerie.py +110 -0
  36. StreamingCommunity/Src/Api/Site/guardaserie/__init__.py +52 -0
  37. StreamingCommunity/Src/Api/Site/guardaserie/costant.py +15 -0
  38. StreamingCommunity/Src/Api/Site/guardaserie/series.py +195 -0
  39. StreamingCommunity/Src/Api/Site/guardaserie/site.py +84 -0
  40. StreamingCommunity/Src/Api/Site/guardaserie/util/ScrapeSerie.py +110 -0
  41. StreamingCommunity/Src/Api/Site/mostraguarda/__init__.py +48 -0
  42. StreamingCommunity/Src/Api/Site/mostraguarda/costant.py +15 -0
  43. StreamingCommunity/Src/Api/Site/mostraguarda/film.py +94 -0
  44. StreamingCommunity/Src/Api/Site/piratebays/__init__.py +50 -0
  45. StreamingCommunity/Src/Api/Site/piratebays/costant.py +15 -0
  46. StreamingCommunity/Src/Api/Site/piratebays/site.py +89 -0
  47. StreamingCommunity/Src/Api/Site/piratebays/title.py +45 -0
  48. StreamingCommunity/Src/Api/Site/streamingcommunity/__init__.py +55 -0
  49. StreamingCommunity/Src/Api/Site/streamingcommunity/costant.py +15 -0
  50. StreamingCommunity/Src/Api/Site/streamingcommunity/film.py +70 -0
  51. StreamingCommunity/Src/Api/Site/streamingcommunity/series.py +203 -0
  52. StreamingCommunity/Src/Api/Site/streamingcommunity/site.py +126 -0
  53. StreamingCommunity/Src/Api/Site/streamingcommunity/util/ScrapeSerie.py +113 -0
  54. StreamingCommunity/Src/Api/Template/Class/SearchType.py +101 -0
  55. StreamingCommunity/Src/Api/Template/Util/__init__.py +5 -0
  56. StreamingCommunity/Src/Api/Template/Util/get_domain.py +137 -0
  57. StreamingCommunity/Src/Api/Template/Util/manage_ep.py +153 -0
  58. StreamingCommunity/Src/Api/Template/Util/recall_search.py +37 -0
  59. StreamingCommunity/Src/Api/Template/__init__.py +3 -0
  60. StreamingCommunity/Src/Api/Template/site.py +87 -0
  61. StreamingCommunity/Src/Lib/Downloader/HLS/downloader.py +968 -0
  62. StreamingCommunity/Src/Lib/Downloader/HLS/proxyes.py +110 -0
  63. StreamingCommunity/Src/Lib/Downloader/HLS/segments.py +540 -0
  64. StreamingCommunity/Src/Lib/Downloader/MP4/downloader.py +156 -0
  65. StreamingCommunity/Src/Lib/Downloader/TOR/downloader.py +222 -0
  66. StreamingCommunity/Src/Lib/Downloader/__init__.py +5 -0
  67. StreamingCommunity/Src/Lib/Driver/driver_1.py +76 -0
  68. StreamingCommunity/Src/Lib/FFmpeg/__init__.py +4 -0
  69. StreamingCommunity/Src/Lib/FFmpeg/capture.py +170 -0
  70. StreamingCommunity/Src/Lib/FFmpeg/command.py +292 -0
  71. StreamingCommunity/Src/Lib/FFmpeg/util.py +242 -0
  72. StreamingCommunity/Src/Lib/M3U8/__init__.py +6 -0
  73. StreamingCommunity/Src/Lib/M3U8/decryptor.py +129 -0
  74. StreamingCommunity/Src/Lib/M3U8/estimator.py +173 -0
  75. StreamingCommunity/Src/Lib/M3U8/parser.py +666 -0
  76. StreamingCommunity/Src/Lib/M3U8/url_fixer.py +52 -0
  77. StreamingCommunity/Src/Lib/TMBD/__init__.py +2 -0
  78. StreamingCommunity/Src/Lib/TMBD/obj_tmbd.py +39 -0
  79. StreamingCommunity/Src/Lib/TMBD/tmdb.py +346 -0
  80. StreamingCommunity/Src/Upload/update.py +64 -0
  81. StreamingCommunity/Src/Upload/version.py +5 -0
  82. StreamingCommunity/Src/Util/_jsonConfig.py +204 -0
  83. StreamingCommunity/Src/Util/call_stack.py +42 -0
  84. StreamingCommunity/Src/Util/color.py +20 -0
  85. StreamingCommunity/Src/Util/console.py +12 -0
  86. StreamingCommunity/Src/Util/headers.py +147 -0
  87. StreamingCommunity/Src/Util/logger.py +53 -0
  88. StreamingCommunity/Src/Util/message.py +46 -0
  89. StreamingCommunity/Src/Util/os.py +417 -0
  90. StreamingCommunity/Src/Util/table.py +163 -0
  91. StreamingCommunity/run.py +196 -0
  92. StreamingCommunity-1.7.6.dist-info/LICENSE +674 -0
  93. StreamingCommunity-1.7.6.dist-info/METADATA +348 -0
  94. StreamingCommunity-1.7.6.dist-info/RECORD +97 -0
  95. StreamingCommunity-1.7.6.dist-info/WHEEL +5 -0
  96. StreamingCommunity-1.7.6.dist-info/entry_points.txt +2 -0
  97. StreamingCommunity-1.7.6.dist-info/top_level.txt +1 -0
@@ -0,0 +1,142 @@
1
+ # 13.06.24
2
+
3
+ import os
4
+ import sys
5
+ from urllib.parse import urlparse
6
+
7
+
8
+ # Internal utilities
9
+ from StreamingCommunity.Src.Util.console import console
10
+ from StreamingCommunity.Src.Util.message import start_message
11
+ from StreamingCommunity.Src.Util.os import os_manager
12
+ from StreamingCommunity.Src.Util.table import TVShowManager
13
+ from StreamingCommunity.Src.Lib.Downloader import MP4_downloader
14
+
15
+
16
+ # Logic class
17
+ from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaItem
18
+ from StreamingCommunity.Src.Api.Template.Util import manage_selection, map_episode_title, validate_episode_selection
19
+
20
+
21
+ # Player
22
+ from .util.ScrapeSerie import GetSerieInfo
23
+ from StreamingCommunity.Src.Api.Player.ddl import VideoSource
24
+
25
+
26
+ # Variable
27
+ from .costant import ROOT_PATH, SITE_NAME, SERIES_FOLDER
28
+ table_show_manager = TVShowManager()
29
+ video_source = VideoSource()
30
+
31
+
32
+
33
+ def download_video(scape_info_serie: GetSerieInfo, index_episode_selected: int) -> None:
34
+ """
35
+ Download a single episode video.
36
+
37
+ Parameters:
38
+ - tv_name (str): Name of the TV series.
39
+ - index_episode_selected (int): Index of the selected episode.
40
+ """
41
+
42
+ start_message()
43
+
44
+ # Get info about episode
45
+ obj_episode = scape_info_serie.list_episodes[index_episode_selected - 1]
46
+ console.print(f"[yellow]Download: [red]{obj_episode.get('name')}")
47
+ print()
48
+
49
+ # Define filename and path for the downloaded video
50
+ title_name = os_manager.get_sanitize_file(
51
+ f"{map_episode_title(scape_info_serie.tv_name, None, index_episode_selected, obj_episode.get('name'))}.mp4"
52
+ )
53
+ mp4_path = os.path.join(ROOT_PATH, SITE_NAME, SERIES_FOLDER, scape_info_serie.tv_name)
54
+
55
+ # Create output folder
56
+ os_manager.create_path(mp4_path)
57
+
58
+ # Setup video source
59
+ video_source.setup(obj_episode.get('url'))
60
+
61
+ # Get m3u8 master playlist
62
+ master_playlist = video_source.get_playlist()
63
+
64
+ # Parse start page url
65
+ parsed_url = urlparse(obj_episode.get('url'))
66
+
67
+ # Start download
68
+ r_proc = MP4_downloader(
69
+ url = master_playlist,
70
+ path = os.path.join(mp4_path, title_name),
71
+ referer = f"{parsed_url.scheme}://{parsed_url.netloc}/",
72
+ )
73
+
74
+ if r_proc != None:
75
+ console.print("[green]Result: ")
76
+ console.print(r_proc)
77
+
78
+
79
+ def download_thread(dict_serie: MediaItem):
80
+ """
81
+ Download all episode of a thread
82
+ """
83
+
84
+ # Start message and set up video source
85
+ start_message()
86
+
87
+ # Init class
88
+ scape_info_serie = GetSerieInfo(dict_serie)
89
+
90
+ # Collect information about thread
91
+ list_dict_episode = scape_info_serie.get_episode_number()
92
+ episodes_count = len(list_dict_episode)
93
+
94
+ # Display episodes list and manage user selection
95
+ last_command = display_episodes_list(scape_info_serie.list_episodes)
96
+ list_episode_select = manage_selection(last_command, episodes_count)
97
+
98
+ try:
99
+ list_episode_select = validate_episode_selection(list_episode_select, episodes_count)
100
+ except ValueError as e:
101
+ console.print(f"[red]{str(e)}")
102
+ return
103
+
104
+ # Download selected episodes
105
+ for i_episode in list_episode_select:
106
+ download_video(scape_info_serie, i_episode)
107
+
108
+
109
+
110
+ def display_episodes_list(obj_episode_manager) -> str:
111
+ """
112
+ Display episodes list and handle user input.
113
+
114
+ Returns:
115
+ last_command (str): Last command entered by the user.
116
+ """
117
+
118
+ # Set up table for displaying episodes
119
+ table_show_manager.set_slice_end(10)
120
+
121
+ # Add columns to the table
122
+ column_info = {
123
+ "Index": {'color': 'red'},
124
+ "Name": {'color': 'magenta'},
125
+ }
126
+ table_show_manager.add_column(column_info)
127
+
128
+ # Populate the table with episodes information
129
+ for i, media in enumerate(obj_episode_manager):
130
+ table_show_manager.add_tv_show({
131
+ 'Index': str(i+1),
132
+ 'Name': media.get('name'),
133
+ })
134
+
135
+ # Run the table and handle user input
136
+ last_command = table_show_manager.run()
137
+
138
+ if last_command == "q":
139
+ console.print("\n[red]Quit [white]...")
140
+ sys.exit(0)
141
+
142
+ return last_command
@@ -0,0 +1,93 @@
1
+ # 09.06.24
2
+
3
+ import logging
4
+
5
+
6
+ # External libraries
7
+ import httpx
8
+ from bs4 import BeautifulSoup
9
+
10
+
11
+ # Internal utilities
12
+ from StreamingCommunity.Src.Util.console import console
13
+ from StreamingCommunity.Src.Util._jsonConfig import config_manager
14
+ from StreamingCommunity.Src.Util.headers import get_headers
15
+ from StreamingCommunity.Src.Util.table import TVShowManager
16
+
17
+
18
+ # Logic class
19
+ from StreamingCommunity.Src.Api.Template import get_select_title
20
+ from StreamingCommunity.Src.Api.Template.Util import search_domain
21
+ from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaManager
22
+
23
+
24
+ # Variable
25
+ from .costant import SITE_NAME
26
+ media_search_manager = MediaManager()
27
+ table_show_manager = TVShowManager()
28
+
29
+
30
+ def title_search(word_to_search: str) -> int:
31
+ """
32
+ Search for titles based on a search query.
33
+
34
+ Parameters:
35
+ - title_search (str): The title to search for.
36
+
37
+ Returns:
38
+ - int: The number of titles found.
39
+ """
40
+
41
+ # Find new domain if prev dont work
42
+ max_timeout = config_manager.get_int("REQUESTS", "timeout")
43
+ domain_to_use, _ = search_domain(SITE_NAME, f"https://{SITE_NAME}")
44
+
45
+ # Send request to search for titles
46
+ try:
47
+ response = httpx.get(
48
+ url=f"https://{SITE_NAME}.{domain_to_use}/search/?&q={word_to_search}&quick=1&type=videobox_video&nodes=11",
49
+ headers={'user-agent': get_headers()},
50
+ timeout=max_timeout
51
+ )
52
+ response.raise_for_status()
53
+
54
+ except Exception as e:
55
+ console.print(f"Site: {SITE_NAME}, request search error: {e}")
56
+
57
+ # Create soup and find table
58
+ soup = BeautifulSoup(response.text, "html.parser")
59
+ table_content = soup.find('ol', class_="ipsStream")
60
+
61
+ if table_content:
62
+ for title_div in table_content.find_all('li', class_='ipsStreamItem'):
63
+ try:
64
+
65
+ title_type = title_div.find("p", class_="ipsType_reset").find_all("a")[-1].get_text(strip=True)
66
+ name = title_div.find("span", class_="ipsContained").find("a").get_text(strip=True)
67
+ link = title_div.find("span", class_="ipsContained").find("a").get("href")
68
+
69
+ title_info = {
70
+ 'name': name,
71
+ 'url': link,
72
+ 'type': title_type
73
+ }
74
+
75
+ media_search_manager.add_media(title_info)
76
+
77
+ except Exception as e:
78
+ logging.error(f"Error processing title div: {e}")
79
+
80
+ return media_search_manager.get_length()
81
+
82
+ else:
83
+ logging.error("No table content found.")
84
+ return -999
85
+
86
+ return -9999
87
+
88
+
89
+ def run_get_select_title():
90
+ """
91
+ Display a selection of titles and prompt the user to choose one.
92
+ """
93
+ return get_select_title(table_show_manager, media_search_manager)
@@ -0,0 +1,83 @@
1
+ # 13.06.24
2
+
3
+ import sys
4
+ import logging
5
+ from typing import List, Dict
6
+
7
+
8
+ # External libraries
9
+ import httpx
10
+ from bs4 import BeautifulSoup
11
+
12
+
13
+ # Internal utilities
14
+ from StreamingCommunity.Src.Util.headers import get_headers
15
+
16
+
17
+ # Logic class
18
+ from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaItem
19
+
20
+
21
+ # Variable
22
+ from ..costant import COOKIE
23
+
24
+
25
+ class GetSerieInfo:
26
+ def __init__(self, dict_serie: MediaItem) -> None:
27
+ """
28
+ Initializes the GetSerieInfo object with default values.
29
+
30
+ Parameters:
31
+ - dict_serie (MediaItem): Dictionary containing series information (optional).
32
+ """
33
+ self.headers = {'user-agent': get_headers()}
34
+ self.cookies = COOKIE
35
+ self.url = dict_serie.url
36
+ self.tv_name = None
37
+ self.list_episodes = None
38
+
39
+ def get_episode_number(self) -> List[Dict[str, str]]:
40
+ """
41
+ Retrieves the number of episodes for a specific season.
42
+
43
+ Parameters:
44
+ n_season (int): The season number.
45
+
46
+ Returns:
47
+ List[Dict[str, str]]: List of dictionaries containing episode information.
48
+ """
49
+
50
+ try:
51
+ response = httpx.get(f"{self.url}?area=online", cookies=self.cookies, headers=self.headers, timeout=10)
52
+ response.raise_for_status()
53
+
54
+ except Exception as e:
55
+ logging.error(f"Insert value for [ips4_device_key, ips4_member_id, ips4_login_key] in config.json file SITE \\ ddlstreamitaly \\ cookie. Use browser debug and cookie request with a valid account, filter by DOC. Error: {e}")
56
+ sys.exit(0)
57
+
58
+ # Parse HTML content of the page
59
+ soup = BeautifulSoup(response.text, "html.parser")
60
+
61
+ # Get tv name
62
+ self.tv_name = soup.find("span", class_= "ipsType_break").get_text(strip=True)
63
+
64
+ # Find the container of episodes for the specified season
65
+ table_content = soup.find('div', class_='ipsMargin_bottom:half')
66
+ list_dict_episode = []
67
+
68
+ for episode_div in table_content.find_all('a', href=True):
69
+
70
+ # Get text of episode
71
+ part_name = episode_div.get_text(strip=True)
72
+
73
+ if part_name:
74
+ obj_episode = {
75
+ 'name': part_name,
76
+ 'url': episode_div['href']
77
+ }
78
+
79
+ list_dict_episode.append(obj_episode)
80
+
81
+ self.list_episodes = list_dict_episode
82
+ return list_dict_episode
83
+
@@ -0,0 +1,110 @@
1
+ # 13.06.24
2
+
3
+ import logging
4
+ from typing import List, Dict
5
+
6
+
7
+ # External libraries
8
+ import httpx
9
+ from bs4 import BeautifulSoup
10
+
11
+
12
+ # Internal utilities
13
+ from StreamingCommunity.Src.Util.headers import get_headers
14
+
15
+
16
+ # Logic class
17
+ from StreamingCommunity.Src.Api.Template .Class.SearchType import MediaItem
18
+
19
+
20
+ class GetSerieInfo:
21
+ def __init__(self, dict_serie: MediaItem) -> None:
22
+ """
23
+ Initializes the GetSerieInfo object with default values.
24
+
25
+ Parameters:
26
+ dict_serie (MediaItem): Dictionary containing series information (optional).
27
+ """
28
+ self.headers = {'user-agent': get_headers()}
29
+ self.url = dict_serie.url
30
+ self.tv_name = None
31
+ self.list_episodes = None
32
+
33
+ def get_seasons_number(self) -> int:
34
+ """
35
+ Retrieves the number of seasons of a TV series.
36
+
37
+ Returns:
38
+ int: Number of seasons of the TV series.
39
+ """
40
+ try:
41
+
42
+ # Make an HTTP request to the series URL
43
+ response = httpx.get(self.url, headers=self.headers, timeout=15)
44
+ response.raise_for_status()
45
+
46
+ # Parse HTML content of the page
47
+ soup = BeautifulSoup(response.text, "html.parser")
48
+
49
+ # Find the container of seasons
50
+ table_content = soup.find('div', class_="tt_season")
51
+
52
+ # Count the number of seasons
53
+ seasons_number = len(table_content.find_all("li"))
54
+
55
+ # Extract the name of the series
56
+ self.tv_name = soup.find("h1", class_="front_title").get_text(strip=True)
57
+
58
+ return seasons_number
59
+
60
+ except Exception as e:
61
+ logging.error(f"Error parsing HTML page: {e}")
62
+
63
+ return -999
64
+
65
+ def get_episode_number(self, n_season: int) -> List[Dict[str, str]]:
66
+ """
67
+ Retrieves the number of episodes for a specific season.
68
+
69
+ Parameters:
70
+ n_season (int): The season number.
71
+
72
+ Returns:
73
+ List[Dict[str, str]]: List of dictionaries containing episode information.
74
+ """
75
+ try:
76
+
77
+ # Make an HTTP request to the series URL
78
+ response = httpx.get(self.url, headers=self.headers, timeout=15)
79
+ response.raise_for_status()
80
+
81
+ # Parse HTML content of the page
82
+ soup = BeautifulSoup(response.text, "html.parser")
83
+
84
+ # Find the container of episodes for the specified season
85
+ table_content = soup.find('div', class_="tab-pane", id=f"season-{n_season}")
86
+
87
+ # Extract episode information
88
+ episode_content = table_content.find_all("li")
89
+ list_dict_episode = []
90
+
91
+ for episode_div in episode_content:
92
+ index = episode_div.find("a").get("data-num")
93
+ link = episode_div.find("a").get("data-link")
94
+ name = episode_div.find("a").get("data-title")
95
+
96
+ obj_episode = {
97
+ 'number': index,
98
+ 'name': name,
99
+ 'url': link
100
+ }
101
+
102
+ list_dict_episode.append(obj_episode)
103
+
104
+ self.list_episodes = list_dict_episode
105
+ return list_dict_episode
106
+
107
+ except Exception as e:
108
+ logging.error(f"Error parsing HTML page: {e}")
109
+
110
+ return []
@@ -0,0 +1,52 @@
1
+ # 09.06.24
2
+
3
+ from unidecode import unidecode
4
+
5
+
6
+ # Internal utilities
7
+ from StreamingCommunity.Src.Util.console import console, msg
8
+
9
+
10
+ # Logic class
11
+ from .site import title_search, run_get_select_title, media_search_manager
12
+ from .series import download_series
13
+
14
+
15
+ # Variable
16
+ indice = 4
17
+ _useFor = "serie"
18
+ _deprecate = False
19
+ _priority = 2
20
+ _engineDownload = "hls"
21
+
22
+
23
+ def search(string_to_search: str = None, get_onylDatabase: bool = False):
24
+ """
25
+ Main function of the application for film and series.
26
+ """
27
+
28
+ if string_to_search is None:
29
+
30
+ # Make request to site to get content that corrsisponde to that string
31
+ string_to_search = msg.ask("\n[purple]Insert word to search in all site").strip()
32
+
33
+ # Search on database
34
+ len_database = title_search(unidecode(string_to_search))
35
+
36
+ # Return list of elements
37
+ if get_onylDatabase:
38
+ return media_search_manager
39
+
40
+ if len_database > 0:
41
+
42
+ # Select title from list
43
+ select_title = run_get_select_title()
44
+
45
+ # Download only film
46
+ download_series(select_title)
47
+
48
+ else:
49
+ console.print(f"\n[red]Nothing matching was found for[white]: [purple]{string_to_search}")
50
+
51
+ # Retry
52
+ search()
@@ -0,0 +1,15 @@
1
+ # 09.06.24
2
+
3
+ import os
4
+
5
+
6
+ # Internal utilities
7
+ from StreamingCommunity.Src.Util._jsonConfig import config_manager
8
+
9
+
10
+ SITE_NAME = os.path.basename(os.path.dirname(os.path.abspath(__file__)))
11
+ ROOT_PATH = config_manager.get('DEFAULT', 'root_path')
12
+ DOMAIN_NOW = config_manager.get_dict('SITE', SITE_NAME)['domain']
13
+
14
+ SERIES_FOLDER = "Serie"
15
+ MOVIE_FOLDER = "Film"
@@ -0,0 +1,195 @@
1
+ # 13.06.24
2
+
3
+ import os
4
+ import sys
5
+ import time
6
+
7
+
8
+ # Internal utilities
9
+ from StreamingCommunity.Src.Util.console import console, msg
10
+ from StreamingCommunity.Src.Util.message import start_message
11
+ from StreamingCommunity.Src.Util.call_stack import get_call_stack
12
+ from StreamingCommunity.Src.Util.table import TVShowManager
13
+ from StreamingCommunity.Src.Lib.Downloader import HLS_Downloader
14
+
15
+
16
+ # Logic class
17
+ from StreamingCommunity.Src.Api.Template.Util import manage_selection, map_episode_title, validate_selection, validate_episode_selection, execute_search
18
+ from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaItem
19
+
20
+
21
+ # Player
22
+ from .util.ScrapeSerie import GetSerieInfo
23
+ from StreamingCommunity.Src.Api.Player.supervideo import VideoSource
24
+
25
+
26
+ # Variable
27
+ from .costant import ROOT_PATH, SITE_NAME, SERIES_FOLDER
28
+ table_show_manager = TVShowManager()
29
+
30
+
31
+
32
+ def download_video(scape_info_serie: GetSerieInfo, index_season_selected: int, index_episode_selected: int) -> None:
33
+ """
34
+ Download a single episode video.
35
+
36
+ Parameters:
37
+ - tv_name (str): Name of the TV series.
38
+ - index_season_selected (int): Index of the selected season.
39
+ - index_episode_selected (int): Index of the selected episode.
40
+ """
41
+
42
+ start_message()
43
+
44
+ # Get info about episode
45
+ obj_episode = scape_info_serie.list_episodes[index_episode_selected - 1]
46
+ console.print(f"[yellow]Download: [red]{index_season_selected}:{index_episode_selected} {obj_episode.get('name')}")
47
+ print()
48
+
49
+ # Define filename and path for the downloaded video
50
+ mp4_name = f"{map_episode_title(scape_info_serie.tv_name, index_season_selected, index_episode_selected, obj_episode.get('name'))}.mp4"
51
+ mp4_path = os.path.join(ROOT_PATH, SITE_NAME, SERIES_FOLDER, scape_info_serie.tv_name, f"S{index_season_selected}")
52
+
53
+ # Setup video source
54
+ video_source = VideoSource(obj_episode.get('url'))
55
+
56
+ # Get m3u8 master playlist
57
+ master_playlist = video_source.get_playlist()
58
+
59
+ # Download the film using the m3u8 playlist, and output filename
60
+ r_proc = HLS_Downloader(
61
+ m3u8_playlist=master_playlist,
62
+ output_filename=os.path.join(mp4_path, mp4_name)
63
+ ).start()
64
+
65
+ if r_proc == 404:
66
+ time.sleep(2)
67
+
68
+ # Re call search function
69
+ if msg.ask("[green]Do you want to continue [white]([red]y[white])[green] or return at home[white]([red]n[white]) ", choices=['y', 'n'], default='y', show_choices=True) == "n":
70
+ frames = get_call_stack()
71
+ execute_search(frames[-4])
72
+
73
+ if r_proc != None:
74
+ console.print("[green]Result: ")
75
+ console.print(r_proc)
76
+
77
+
78
+
79
+ def download_episode(scape_info_serie: GetSerieInfo, index_season_selected: int, download_all: bool = False) -> None:
80
+ """
81
+ Download all episodes of a season.
82
+
83
+ Parameters:
84
+ - tv_name (str): Name of the TV series.
85
+ - index_season_selected (int): Index of the selected season.
86
+ - download_all (bool): Download all seasons episodes
87
+ """
88
+
89
+ # Start message and collect information about episodes
90
+ start_message()
91
+ list_dict_episode = scape_info_serie.get_episode_number(index_season_selected)
92
+ episodes_count = len(list_dict_episode)
93
+
94
+ if download_all:
95
+
96
+ # Download all episodes without asking
97
+ for i_episode in range(1, episodes_count + 1):
98
+ download_video(scape_info_serie, index_season_selected, i_episode)
99
+ console.print(f"\n[red]End downloaded [yellow]season: [red]{index_season_selected}.")
100
+
101
+ else:
102
+
103
+ # Display episodes list and manage user selection
104
+ last_command = display_episodes_list(scape_info_serie.list_episodes)
105
+ list_episode_select = manage_selection(last_command, episodes_count)
106
+
107
+ try:
108
+ list_episode_select = validate_episode_selection(list_episode_select, episodes_count)
109
+ except ValueError as e:
110
+ console.print(f"[red]{str(e)}")
111
+ return
112
+
113
+ # Download selected episodes
114
+ for i_episode in list_episode_select:
115
+ download_video(scape_info_serie, index_season_selected, i_episode)
116
+
117
+
118
+ def download_series(dict_serie: MediaItem) -> None:
119
+ """
120
+ Download all episodes of a TV series.
121
+
122
+ Parameters:
123
+ - dict_serie (MediaItem): obj with url name type and score
124
+ """
125
+
126
+ # Start message and set up video source
127
+ start_message()
128
+
129
+ # Init class
130
+ scape_info_serie = GetSerieInfo(dict_serie)
131
+
132
+ # Collect information about seasons
133
+ seasons_count = scape_info_serie.get_seasons_number()
134
+
135
+ # Prompt user for season selection and download episodes
136
+ console.print(f"\n[green]Seasons found: [red]{seasons_count}")
137
+ index_season_selected = msg.ask(
138
+ "\n[cyan]Insert season number [yellow](e.g., 1), [red]* [cyan]to download all seasons, "
139
+ "[yellow](e.g., 1-2) [cyan]for a range of seasons, or [yellow](e.g., 3-*) [cyan]to download from a specific season to the end"
140
+ )
141
+
142
+ # Manage and validate the selection
143
+ list_season_select = manage_selection(index_season_selected, seasons_count)
144
+
145
+ try:
146
+ list_season_select = validate_selection(list_season_select, seasons_count)
147
+ except ValueError as e:
148
+ console.print(f"[red]{str(e)}")
149
+ return
150
+
151
+ # Loop through the selected seasons and download episodes
152
+ for i_season in list_season_select:
153
+ if len(list_season_select) > 1 or index_season_selected == "*":
154
+
155
+ # Download all episodes if multiple seasons are selected or if '*' is used
156
+ download_episode(scape_info_serie, i_season, download_all=True)
157
+ else:
158
+
159
+ # Otherwise, let the user select specific episodes for the single season
160
+ download_episode(scape_info_serie, i_season, download_all=False)
161
+
162
+
163
+ def display_episodes_list(obj_episode_manager) -> str:
164
+ """
165
+ Display episodes list and handle user input.
166
+
167
+ Returns:
168
+ last_command (str): Last command entered by the user.
169
+ """
170
+
171
+ # Set up table for displaying episodes
172
+ table_show_manager.set_slice_end(10)
173
+
174
+ # Add columns to the table
175
+ column_info = {
176
+ "Index": {'color': 'red'},
177
+ "Name": {'color': 'magenta'},
178
+ }
179
+ table_show_manager.add_column(column_info)
180
+
181
+ # Populate the table with episodes information
182
+ for media in obj_episode_manager:
183
+ table_show_manager.add_tv_show({
184
+ 'Index': str(media.get('number')),
185
+ 'Name': media.get('name'),
186
+ })
187
+
188
+ # Run the table and handle user input
189
+ last_command = table_show_manager.run()
190
+
191
+ if last_command == "q":
192
+ console.print("\n[red]Quit [white]...")
193
+ sys.exit(0)
194
+
195
+ return last_command