StreamingCommunity 1.7.6__py3-none-any.whl → 1.8.0__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.
- StreamingCommunity/Src/Api/Player/Helper/Vixcloud/js_parser.py +4 -1
- StreamingCommunity/Src/Api/Player/Helper/Vixcloud/util.py +166 -166
- StreamingCommunity/Src/Api/Player/ddl.py +89 -89
- StreamingCommunity/Src/Api/Player/maxstream.py +151 -151
- StreamingCommunity/Src/Api/Player/supervideo.py +193 -193
- StreamingCommunity/Src/Api/Player/vixcloud.py +224 -212
- StreamingCommunity/Src/Api/Site/1337xx/__init__.py +50 -50
- StreamingCommunity/Src/Api/Site/1337xx/costant.py +14 -14
- StreamingCommunity/Src/Api/Site/1337xx/site.py +83 -83
- StreamingCommunity/Src/Api/Site/1337xx/title.py +66 -66
- StreamingCommunity/Src/Api/Site/altadefinizione/__init__.py +50 -50
- StreamingCommunity/Src/Api/Site/altadefinizione/costant.py +14 -14
- StreamingCommunity/Src/Api/Site/altadefinizione/film.py +69 -69
- StreamingCommunity/Src/Api/Site/altadefinizione/site.py +86 -86
- StreamingCommunity/Src/Api/Site/animeunity/__init__.py +50 -50
- StreamingCommunity/Src/Api/Site/animeunity/costant.py +15 -15
- StreamingCommunity/Src/Api/Site/animeunity/film_serie.py +131 -131
- StreamingCommunity/Src/Api/Site/animeunity/site.py +164 -164
- StreamingCommunity/Src/Api/Site/bitsearch/__init__.py +51 -51
- StreamingCommunity/Src/Api/Site/bitsearch/costant.py +15 -15
- StreamingCommunity/Src/Api/Site/bitsearch/site.py +84 -84
- StreamingCommunity/Src/Api/Site/bitsearch/title.py +47 -47
- StreamingCommunity/Src/Api/Site/cb01new/__init__.py +51 -51
- StreamingCommunity/Src/Api/Site/cb01new/costant.py +15 -15
- StreamingCommunity/Src/Api/Site/cb01new/film.py +69 -69
- StreamingCommunity/Src/Api/Site/cb01new/site.py +74 -74
- StreamingCommunity/Src/Api/Site/ddlstreamitaly/__init__.py +57 -57
- StreamingCommunity/Src/Api/Site/ddlstreamitaly/costant.py +16 -16
- StreamingCommunity/Src/Api/Site/ddlstreamitaly/series.py +142 -142
- StreamingCommunity/Src/Api/Site/ddlstreamitaly/site.py +92 -92
- StreamingCommunity/Src/Api/Site/ddlstreamitaly/util/ScrapeSerie.py +82 -82
- StreamingCommunity/Src/Api/Site/guardaserie/__init__.py +52 -52
- StreamingCommunity/Src/Api/Site/guardaserie/costant.py +15 -15
- StreamingCommunity/Src/Api/Site/guardaserie/series.py +195 -195
- StreamingCommunity/Src/Api/Site/guardaserie/site.py +84 -84
- StreamingCommunity/Src/Api/Site/guardaserie/util/ScrapeSerie.py +110 -110
- StreamingCommunity/Src/Api/Site/mostraguarda/__init__.py +48 -48
- StreamingCommunity/Src/Api/Site/mostraguarda/costant.py +14 -14
- StreamingCommunity/Src/Api/Site/mostraguarda/film.py +94 -94
- StreamingCommunity/Src/Api/Site/piratebays/__init__.py +50 -50
- StreamingCommunity/Src/Api/Site/piratebays/costant.py +14 -14
- StreamingCommunity/Src/Api/Site/piratebays/site.py +88 -88
- StreamingCommunity/Src/Api/Site/piratebays/title.py +45 -45
- StreamingCommunity/Src/Api/Site/streamingcommunity/__init__.py +55 -55
- StreamingCommunity/Src/Api/Site/streamingcommunity/costant.py +14 -14
- StreamingCommunity/Src/Api/Site/streamingcommunity/film.py +70 -70
- StreamingCommunity/Src/Api/Site/streamingcommunity/series.py +203 -203
- StreamingCommunity/Src/Api/Site/streamingcommunity/site.py +125 -125
- StreamingCommunity/Src/Api/Template/Class/SearchType.py +101 -101
- StreamingCommunity/Src/Api/Template/Util/__init__.py +4 -4
- StreamingCommunity/Src/Api/Template/Util/get_domain.py +137 -137
- StreamingCommunity/Src/Api/Template/Util/manage_ep.py +153 -153
- StreamingCommunity/Src/Api/Template/Util/recall_search.py +37 -37
- StreamingCommunity/Src/Api/Template/__init__.py +2 -2
- StreamingCommunity/Src/Api/Template/site.py +87 -87
- StreamingCommunity/Src/Lib/Downloader/HLS/downloader.py +968 -968
- StreamingCommunity/Src/Lib/Downloader/HLS/proxyes.py +110 -110
- StreamingCommunity/Src/Lib/Downloader/HLS/segments.py +540 -540
- StreamingCommunity/Src/Lib/Downloader/MP4/downloader.py +156 -156
- StreamingCommunity/Src/Lib/Downloader/TOR/downloader.py +222 -222
- StreamingCommunity/Src/Lib/Downloader/__init__.py +4 -4
- StreamingCommunity/Src/Lib/Driver/driver_1.py +76 -76
- StreamingCommunity/Src/Lib/FFmpeg/__init__.py +4 -4
- StreamingCommunity/Src/Lib/FFmpeg/capture.py +170 -170
- StreamingCommunity/Src/Lib/FFmpeg/command.py +292 -292
- StreamingCommunity/Src/Lib/FFmpeg/util.py +241 -241
- StreamingCommunity/Src/Lib/M3U8/__init__.py +5 -5
- StreamingCommunity/Src/Lib/M3U8/decryptor.py +128 -128
- StreamingCommunity/Src/Lib/M3U8/estimator.py +172 -172
- StreamingCommunity/Src/Lib/M3U8/parser.py +666 -666
- StreamingCommunity/Src/Lib/M3U8/url_fixer.py +51 -51
- StreamingCommunity/Src/Lib/TMBD/__init__.py +1 -1
- StreamingCommunity/Src/Lib/TMBD/obj_tmbd.py +39 -39
- StreamingCommunity/Src/Lib/TMBD/tmdb.py +345 -345
- StreamingCommunity/Src/Upload/update.py +64 -64
- StreamingCommunity/Src/Upload/version.py +5 -5
- StreamingCommunity/Src/Util/_jsonConfig.py +204 -204
- StreamingCommunity/Src/Util/call_stack.py +42 -42
- StreamingCommunity/Src/Util/color.py +20 -20
- StreamingCommunity/Src/Util/console.py +12 -12
- StreamingCommunity/Src/Util/headers.py +147 -147
- StreamingCommunity/Src/Util/logger.py +53 -53
- StreamingCommunity/Src/Util/message.py +46 -46
- StreamingCommunity/Src/Util/os.py +417 -417
- StreamingCommunity/Src/Util/table.py +163 -163
- StreamingCommunity/run.py +196 -196
- {StreamingCommunity-1.7.6.dist-info → StreamingCommunity-1.8.0.dist-info}/METADATA +1 -1
- StreamingCommunity-1.8.0.dist-info/RECORD +97 -0
- StreamingCommunity-1.7.6.dist-info/RECORD +0 -97
- {StreamingCommunity-1.7.6.dist-info → StreamingCommunity-1.8.0.dist-info}/LICENSE +0 -0
- {StreamingCommunity-1.7.6.dist-info → StreamingCommunity-1.8.0.dist-info}/WHEEL +0 -0
- {StreamingCommunity-1.7.6.dist-info → StreamingCommunity-1.8.0.dist-info}/entry_points.txt +0 -0
- {StreamingCommunity-1.7.6.dist-info → StreamingCommunity-1.8.0.dist-info}/top_level.txt +0 -0
|
@@ -1,203 +1,203 @@
|
|
|
1
|
-
# 3.12.23
|
|
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 .util.ScrapeSerie import ScrapeSerie
|
|
18
|
-
from StreamingCommunity.Src.Api.Template.Util import manage_selection, map_episode_title, validate_selection, validate_episode_selection, execute_search
|
|
19
|
-
from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaItem
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
# Player
|
|
23
|
-
from StreamingCommunity.Src.Api.Player.vixcloud import VideoSource
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
# Variable
|
|
27
|
-
from .costant import ROOT_PATH, SITE_NAME, SERIES_FOLDER
|
|
28
|
-
scrape_serie = ScrapeSerie(SITE_NAME)
|
|
29
|
-
video_source = VideoSource(SITE_NAME, True)
|
|
30
|
-
table_show_manager = TVShowManager()
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
def download_video(tv_name: str, index_season_selected: int, index_episode_selected: int) -> None:
|
|
35
|
-
"""
|
|
36
|
-
Download a single episode video.
|
|
37
|
-
|
|
38
|
-
Parameters:
|
|
39
|
-
- tv_name (str): Name of the TV series.
|
|
40
|
-
- index_season_selected (int): Index of the selected season.
|
|
41
|
-
- index_episode_selected (int): Index of the selected episode.
|
|
42
|
-
"""
|
|
43
|
-
|
|
44
|
-
start_message()
|
|
45
|
-
|
|
46
|
-
# Get info about episode
|
|
47
|
-
obj_episode = scrape_serie.obj_episode_manager.episodes[index_episode_selected - 1]
|
|
48
|
-
console.print(f"[yellow]Download: [red]{index_season_selected}:{index_episode_selected} {obj_episode.name}")
|
|
49
|
-
print()
|
|
50
|
-
|
|
51
|
-
# Define filename and path for the downloaded video
|
|
52
|
-
mp4_name = f"{map_episode_title(tv_name, index_season_selected, index_episode_selected, obj_episode.name)}.mp4"
|
|
53
|
-
mp4_path = os.path.join(ROOT_PATH, SITE_NAME, SERIES_FOLDER, tv_name, f"S{index_season_selected}")
|
|
54
|
-
|
|
55
|
-
# Retrieve scws and if available master playlist
|
|
56
|
-
video_source.get_iframe(obj_episode.id)
|
|
57
|
-
video_source.get_content()
|
|
58
|
-
master_playlist = video_source.get_playlist()
|
|
59
|
-
|
|
60
|
-
# Download the episode
|
|
61
|
-
r_proc = HLS_Downloader(
|
|
62
|
-
m3u8_playlist=master_playlist,
|
|
63
|
-
output_filename=os.path.join(mp4_path, mp4_name)
|
|
64
|
-
).start()
|
|
65
|
-
|
|
66
|
-
if r_proc == 404:
|
|
67
|
-
time.sleep(2)
|
|
68
|
-
|
|
69
|
-
# Re call search function
|
|
70
|
-
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":
|
|
71
|
-
frames = get_call_stack()
|
|
72
|
-
execute_search(frames[-4])
|
|
73
|
-
|
|
74
|
-
if r_proc != None:
|
|
75
|
-
console.print("[green]Result: ")
|
|
76
|
-
console.print(r_proc)
|
|
77
|
-
|
|
78
|
-
def download_episode(tv_name: str, index_season_selected: int, download_all: bool = False) -> None:
|
|
79
|
-
"""
|
|
80
|
-
Download episodes of a selected season.
|
|
81
|
-
|
|
82
|
-
Parameters:
|
|
83
|
-
- tv_name (str): Name of the TV series.
|
|
84
|
-
- index_season_selected (int): Index of the selected season.
|
|
85
|
-
- download_all (bool): Download all episodes in the season.
|
|
86
|
-
"""
|
|
87
|
-
|
|
88
|
-
# Clean memory of all episodes and get the number of the season
|
|
89
|
-
scrape_serie.obj_episode_manager.clear()
|
|
90
|
-
season_number = scrape_serie.obj_season_manager.seasons[index_season_selected - 1].number
|
|
91
|
-
|
|
92
|
-
# Start message and collect information about episodes
|
|
93
|
-
start_message()
|
|
94
|
-
scrape_serie.collect_title_season(season_number)
|
|
95
|
-
episodes_count = scrape_serie.obj_episode_manager.get_length()
|
|
96
|
-
|
|
97
|
-
if download_all:
|
|
98
|
-
|
|
99
|
-
# Download all episodes without asking
|
|
100
|
-
for i_episode in range(1, episodes_count + 1):
|
|
101
|
-
download_video(tv_name, index_season_selected, i_episode)
|
|
102
|
-
console.print(f"\n[red]End downloaded [yellow]season: [red]{index_season_selected}.")
|
|
103
|
-
|
|
104
|
-
else:
|
|
105
|
-
|
|
106
|
-
# Display episodes list and manage user selection
|
|
107
|
-
last_command = display_episodes_list()
|
|
108
|
-
list_episode_select = manage_selection(last_command, episodes_count)
|
|
109
|
-
|
|
110
|
-
try:
|
|
111
|
-
list_episode_select = validate_episode_selection(list_episode_select, episodes_count)
|
|
112
|
-
except ValueError as e:
|
|
113
|
-
console.print(f"[red]{str(e)}")
|
|
114
|
-
return
|
|
115
|
-
|
|
116
|
-
# Download selected episodes
|
|
117
|
-
for i_episode in list_episode_select:
|
|
118
|
-
download_video(tv_name, index_season_selected, i_episode)
|
|
119
|
-
|
|
120
|
-
def download_series(select_season: MediaItem, version: str) -> None:
|
|
121
|
-
"""
|
|
122
|
-
Download episodes of a TV series based on user selection.
|
|
123
|
-
|
|
124
|
-
Parameters:
|
|
125
|
-
- select_season (MediaItem): Selected media item (TV series).
|
|
126
|
-
- domain (str): Domain from which to download.
|
|
127
|
-
- version (str): Version of the site.
|
|
128
|
-
"""
|
|
129
|
-
|
|
130
|
-
# Start message and set up video source
|
|
131
|
-
start_message()
|
|
132
|
-
|
|
133
|
-
# Setup video source
|
|
134
|
-
scrape_serie.setup(version, select_season.id, select_season.slug)
|
|
135
|
-
video_source.setup(select_season.id)
|
|
136
|
-
|
|
137
|
-
# Collect information about seasons
|
|
138
|
-
scrape_serie.collect_info_seasons()
|
|
139
|
-
seasons_count = scrape_serie.obj_season_manager.get_length()
|
|
140
|
-
|
|
141
|
-
# Prompt user for season selection and download episodes
|
|
142
|
-
console.print(f"\n[green]Seasons found: [red]{seasons_count}")
|
|
143
|
-
index_season_selected = msg.ask(
|
|
144
|
-
"\n[cyan]Insert season number [yellow](e.g., 1), [red]* [cyan]to download all seasons, "
|
|
145
|
-
"[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"
|
|
146
|
-
)
|
|
147
|
-
|
|
148
|
-
# Manage and validate the selection
|
|
149
|
-
list_season_select = manage_selection(index_season_selected, seasons_count)
|
|
150
|
-
|
|
151
|
-
try:
|
|
152
|
-
list_season_select = validate_selection(list_season_select, seasons_count)
|
|
153
|
-
except ValueError as e:
|
|
154
|
-
console.print(f"[red]{str(e)}")
|
|
155
|
-
return
|
|
156
|
-
|
|
157
|
-
# Loop through the selected seasons and download episodes
|
|
158
|
-
for i_season in list_season_select:
|
|
159
|
-
if len(list_season_select) > 1 or index_season_selected == "*":
|
|
160
|
-
|
|
161
|
-
# Download all episodes if multiple seasons are selected or if '*' is used
|
|
162
|
-
download_episode(select_season.slug, i_season, download_all=True)
|
|
163
|
-
else:
|
|
164
|
-
|
|
165
|
-
# Otherwise, let the user select specific episodes for the single season
|
|
166
|
-
download_episode(select_season.slug, i_season, download_all=False)
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
def display_episodes_list() -> str:
|
|
170
|
-
"""
|
|
171
|
-
Display episodes list and handle user input.
|
|
172
|
-
|
|
173
|
-
Returns:
|
|
174
|
-
last_command (str): Last command entered by the user.
|
|
175
|
-
"""
|
|
176
|
-
|
|
177
|
-
# Set up table for displaying episodes
|
|
178
|
-
table_show_manager.set_slice_end(10)
|
|
179
|
-
|
|
180
|
-
# Add columns to the table
|
|
181
|
-
column_info = {
|
|
182
|
-
"Index": {'color': 'red'},
|
|
183
|
-
"Name": {'color': 'magenta'},
|
|
184
|
-
"Duration": {'color': 'green'}
|
|
185
|
-
}
|
|
186
|
-
table_show_manager.add_column(column_info)
|
|
187
|
-
|
|
188
|
-
# Populate the table with episodes information
|
|
189
|
-
for i, media in enumerate(scrape_serie.obj_episode_manager.episodes):
|
|
190
|
-
table_show_manager.add_tv_show({
|
|
191
|
-
'Index': str(media.number),
|
|
192
|
-
'Name': media.name,
|
|
193
|
-
'Duration': str(media.duration)
|
|
194
|
-
})
|
|
195
|
-
|
|
196
|
-
# Run the table and handle user input
|
|
197
|
-
last_command = table_show_manager.run()
|
|
198
|
-
|
|
199
|
-
if last_command == "q":
|
|
200
|
-
console.print("\n[red]Quit [white]...")
|
|
201
|
-
sys.exit(0)
|
|
202
|
-
|
|
203
|
-
return last_command
|
|
1
|
+
# 3.12.23
|
|
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 .util.ScrapeSerie import ScrapeSerie
|
|
18
|
+
from StreamingCommunity.Src.Api.Template.Util import manage_selection, map_episode_title, validate_selection, validate_episode_selection, execute_search
|
|
19
|
+
from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaItem
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
# Player
|
|
23
|
+
from StreamingCommunity.Src.Api.Player.vixcloud import VideoSource
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
# Variable
|
|
27
|
+
from .costant import ROOT_PATH, SITE_NAME, SERIES_FOLDER
|
|
28
|
+
scrape_serie = ScrapeSerie(SITE_NAME)
|
|
29
|
+
video_source = VideoSource(SITE_NAME, True)
|
|
30
|
+
table_show_manager = TVShowManager()
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
def download_video(tv_name: str, index_season_selected: int, index_episode_selected: int) -> None:
|
|
35
|
+
"""
|
|
36
|
+
Download a single episode video.
|
|
37
|
+
|
|
38
|
+
Parameters:
|
|
39
|
+
- tv_name (str): Name of the TV series.
|
|
40
|
+
- index_season_selected (int): Index of the selected season.
|
|
41
|
+
- index_episode_selected (int): Index of the selected episode.
|
|
42
|
+
"""
|
|
43
|
+
|
|
44
|
+
start_message()
|
|
45
|
+
|
|
46
|
+
# Get info about episode
|
|
47
|
+
obj_episode = scrape_serie.obj_episode_manager.episodes[index_episode_selected - 1]
|
|
48
|
+
console.print(f"[yellow]Download: [red]{index_season_selected}:{index_episode_selected} {obj_episode.name}")
|
|
49
|
+
print()
|
|
50
|
+
|
|
51
|
+
# Define filename and path for the downloaded video
|
|
52
|
+
mp4_name = f"{map_episode_title(tv_name, index_season_selected, index_episode_selected, obj_episode.name)}.mp4"
|
|
53
|
+
mp4_path = os.path.join(ROOT_PATH, SITE_NAME, SERIES_FOLDER, tv_name, f"S{index_season_selected}")
|
|
54
|
+
|
|
55
|
+
# Retrieve scws and if available master playlist
|
|
56
|
+
video_source.get_iframe(obj_episode.id)
|
|
57
|
+
video_source.get_content()
|
|
58
|
+
master_playlist = video_source.get_playlist()
|
|
59
|
+
|
|
60
|
+
# Download the episode
|
|
61
|
+
r_proc = HLS_Downloader(
|
|
62
|
+
m3u8_playlist=master_playlist,
|
|
63
|
+
output_filename=os.path.join(mp4_path, mp4_name)
|
|
64
|
+
).start()
|
|
65
|
+
|
|
66
|
+
if r_proc == 404:
|
|
67
|
+
time.sleep(2)
|
|
68
|
+
|
|
69
|
+
# Re call search function
|
|
70
|
+
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":
|
|
71
|
+
frames = get_call_stack()
|
|
72
|
+
execute_search(frames[-4])
|
|
73
|
+
|
|
74
|
+
if r_proc != None:
|
|
75
|
+
console.print("[green]Result: ")
|
|
76
|
+
console.print(r_proc)
|
|
77
|
+
|
|
78
|
+
def download_episode(tv_name: str, index_season_selected: int, download_all: bool = False) -> None:
|
|
79
|
+
"""
|
|
80
|
+
Download episodes of a selected season.
|
|
81
|
+
|
|
82
|
+
Parameters:
|
|
83
|
+
- tv_name (str): Name of the TV series.
|
|
84
|
+
- index_season_selected (int): Index of the selected season.
|
|
85
|
+
- download_all (bool): Download all episodes in the season.
|
|
86
|
+
"""
|
|
87
|
+
|
|
88
|
+
# Clean memory of all episodes and get the number of the season
|
|
89
|
+
scrape_serie.obj_episode_manager.clear()
|
|
90
|
+
season_number = scrape_serie.obj_season_manager.seasons[index_season_selected - 1].number
|
|
91
|
+
|
|
92
|
+
# Start message and collect information about episodes
|
|
93
|
+
start_message()
|
|
94
|
+
scrape_serie.collect_title_season(season_number)
|
|
95
|
+
episodes_count = scrape_serie.obj_episode_manager.get_length()
|
|
96
|
+
|
|
97
|
+
if download_all:
|
|
98
|
+
|
|
99
|
+
# Download all episodes without asking
|
|
100
|
+
for i_episode in range(1, episodes_count + 1):
|
|
101
|
+
download_video(tv_name, index_season_selected, i_episode)
|
|
102
|
+
console.print(f"\n[red]End downloaded [yellow]season: [red]{index_season_selected}.")
|
|
103
|
+
|
|
104
|
+
else:
|
|
105
|
+
|
|
106
|
+
# Display episodes list and manage user selection
|
|
107
|
+
last_command = display_episodes_list()
|
|
108
|
+
list_episode_select = manage_selection(last_command, episodes_count)
|
|
109
|
+
|
|
110
|
+
try:
|
|
111
|
+
list_episode_select = validate_episode_selection(list_episode_select, episodes_count)
|
|
112
|
+
except ValueError as e:
|
|
113
|
+
console.print(f"[red]{str(e)}")
|
|
114
|
+
return
|
|
115
|
+
|
|
116
|
+
# Download selected episodes
|
|
117
|
+
for i_episode in list_episode_select:
|
|
118
|
+
download_video(tv_name, index_season_selected, i_episode)
|
|
119
|
+
|
|
120
|
+
def download_series(select_season: MediaItem, version: str) -> None:
|
|
121
|
+
"""
|
|
122
|
+
Download episodes of a TV series based on user selection.
|
|
123
|
+
|
|
124
|
+
Parameters:
|
|
125
|
+
- select_season (MediaItem): Selected media item (TV series).
|
|
126
|
+
- domain (str): Domain from which to download.
|
|
127
|
+
- version (str): Version of the site.
|
|
128
|
+
"""
|
|
129
|
+
|
|
130
|
+
# Start message and set up video source
|
|
131
|
+
start_message()
|
|
132
|
+
|
|
133
|
+
# Setup video source
|
|
134
|
+
scrape_serie.setup(version, select_season.id, select_season.slug)
|
|
135
|
+
video_source.setup(select_season.id)
|
|
136
|
+
|
|
137
|
+
# Collect information about seasons
|
|
138
|
+
scrape_serie.collect_info_seasons()
|
|
139
|
+
seasons_count = scrape_serie.obj_season_manager.get_length()
|
|
140
|
+
|
|
141
|
+
# Prompt user for season selection and download episodes
|
|
142
|
+
console.print(f"\n[green]Seasons found: [red]{seasons_count}")
|
|
143
|
+
index_season_selected = msg.ask(
|
|
144
|
+
"\n[cyan]Insert season number [yellow](e.g., 1), [red]* [cyan]to download all seasons, "
|
|
145
|
+
"[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"
|
|
146
|
+
)
|
|
147
|
+
|
|
148
|
+
# Manage and validate the selection
|
|
149
|
+
list_season_select = manage_selection(index_season_selected, seasons_count)
|
|
150
|
+
|
|
151
|
+
try:
|
|
152
|
+
list_season_select = validate_selection(list_season_select, seasons_count)
|
|
153
|
+
except ValueError as e:
|
|
154
|
+
console.print(f"[red]{str(e)}")
|
|
155
|
+
return
|
|
156
|
+
|
|
157
|
+
# Loop through the selected seasons and download episodes
|
|
158
|
+
for i_season in list_season_select:
|
|
159
|
+
if len(list_season_select) > 1 or index_season_selected == "*":
|
|
160
|
+
|
|
161
|
+
# Download all episodes if multiple seasons are selected or if '*' is used
|
|
162
|
+
download_episode(select_season.slug, i_season, download_all=True)
|
|
163
|
+
else:
|
|
164
|
+
|
|
165
|
+
# Otherwise, let the user select specific episodes for the single season
|
|
166
|
+
download_episode(select_season.slug, i_season, download_all=False)
|
|
167
|
+
|
|
168
|
+
|
|
169
|
+
def display_episodes_list() -> str:
|
|
170
|
+
"""
|
|
171
|
+
Display episodes list and handle user input.
|
|
172
|
+
|
|
173
|
+
Returns:
|
|
174
|
+
last_command (str): Last command entered by the user.
|
|
175
|
+
"""
|
|
176
|
+
|
|
177
|
+
# Set up table for displaying episodes
|
|
178
|
+
table_show_manager.set_slice_end(10)
|
|
179
|
+
|
|
180
|
+
# Add columns to the table
|
|
181
|
+
column_info = {
|
|
182
|
+
"Index": {'color': 'red'},
|
|
183
|
+
"Name": {'color': 'magenta'},
|
|
184
|
+
"Duration": {'color': 'green'}
|
|
185
|
+
}
|
|
186
|
+
table_show_manager.add_column(column_info)
|
|
187
|
+
|
|
188
|
+
# Populate the table with episodes information
|
|
189
|
+
for i, media in enumerate(scrape_serie.obj_episode_manager.episodes):
|
|
190
|
+
table_show_manager.add_tv_show({
|
|
191
|
+
'Index': str(media.number),
|
|
192
|
+
'Name': media.name,
|
|
193
|
+
'Duration': str(media.duration)
|
|
194
|
+
})
|
|
195
|
+
|
|
196
|
+
# Run the table and handle user input
|
|
197
|
+
last_command = table_show_manager.run()
|
|
198
|
+
|
|
199
|
+
if last_command == "q":
|
|
200
|
+
console.print("\n[red]Quit [white]...")
|
|
201
|
+
sys.exit(0)
|
|
202
|
+
|
|
203
|
+
return last_command
|
|
@@ -1,126 +1,126 @@
|
|
|
1
|
-
# 10.12.23
|
|
2
|
-
|
|
3
|
-
import sys
|
|
4
|
-
import json
|
|
5
|
-
import logging
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
# External libraries
|
|
9
|
-
import httpx
|
|
10
|
-
from bs4 import BeautifulSoup
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
# Internal utilities
|
|
14
|
-
from StreamingCommunity.Src.Util.console import console
|
|
15
|
-
from StreamingCommunity.Src.Util._jsonConfig import config_manager
|
|
16
|
-
from StreamingCommunity.Src.Util.headers import get_headers
|
|
17
|
-
from StreamingCommunity.Src.Util.table import TVShowManager
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
# Logic class
|
|
22
|
-
from StreamingCommunity.Src.Api.Template import get_select_title
|
|
23
|
-
from StreamingCommunity.Src.Api.Template.Util import search_domain
|
|
24
|
-
from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaManager
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
# Config
|
|
28
|
-
from .costant import SITE_NAME
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
# Variable
|
|
32
|
-
media_search_manager = MediaManager()
|
|
33
|
-
table_show_manager = TVShowManager()
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
def get_version(text: str):
|
|
38
|
-
"""
|
|
39
|
-
Extracts the version from the HTML text of a webpage.
|
|
40
|
-
|
|
41
|
-
Parameters:
|
|
42
|
-
- text (str): The HTML text of the webpage.
|
|
43
|
-
|
|
44
|
-
Returns:
|
|
45
|
-
str: The version extracted from the webpage.
|
|
46
|
-
list: Top 10 titles headlines for today.
|
|
47
|
-
"""
|
|
48
|
-
try:
|
|
49
|
-
|
|
50
|
-
# Parse request to site
|
|
51
|
-
soup = BeautifulSoup(text, "html.parser")
|
|
52
|
-
|
|
53
|
-
# Extract version
|
|
54
|
-
version = json.loads(soup.find("div", {"id": "app"}).get("data-page"))['version']
|
|
55
|
-
#console.print(f"[cyan]Get version [white]=> [red]{version} \n")
|
|
56
|
-
|
|
57
|
-
return version
|
|
58
|
-
|
|
59
|
-
except Exception as e:
|
|
60
|
-
logging.error(f"Error extracting version: {e}")
|
|
61
|
-
raise
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
def get_version_and_domain():
|
|
65
|
-
"""
|
|
66
|
-
Retrieve the current version and domain of the site.
|
|
67
|
-
|
|
68
|
-
This function performs the following steps:
|
|
69
|
-
- Determines the correct domain to use for the site by searching for a specific meta tag.
|
|
70
|
-
- Fetches the content of the site to extract the version information.
|
|
71
|
-
"""
|
|
72
|
-
|
|
73
|
-
# Find new domain if prev dont work
|
|
74
|
-
domain_to_use, base_url = search_domain(SITE_NAME, f"https://{SITE_NAME}")
|
|
75
|
-
|
|
76
|
-
# Extract version from the response
|
|
77
|
-
version = get_version(httpx.get(base_url, headers={'user-agent': get_headers()}).text)
|
|
78
|
-
|
|
79
|
-
return version, domain_to_use
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
def title_search(title_search: str, domain: str) -> int:
|
|
83
|
-
"""
|
|
84
|
-
Search for titles based on a search query.
|
|
85
|
-
|
|
86
|
-
Parameters:
|
|
87
|
-
- title_search (str): The title to search for.
|
|
88
|
-
- domain (str): The domain to search on.
|
|
89
|
-
|
|
90
|
-
Returns:
|
|
91
|
-
int: The number of titles found.
|
|
92
|
-
"""
|
|
93
|
-
|
|
94
|
-
max_timeout = config_manager.get_int("REQUESTS", "timeout")
|
|
95
|
-
|
|
96
|
-
# Send request to search for titles ( replace à to a and space to "+" )
|
|
97
|
-
try:
|
|
98
|
-
response = httpx.get(
|
|
99
|
-
url=f"https://{SITE_NAME}.{domain}/api/search?q={title_search.replace(' ', '+')}",
|
|
100
|
-
headers={'user-agent': get_headers()},
|
|
101
|
-
timeout=max_timeout
|
|
102
|
-
)
|
|
103
|
-
response.raise_for_status()
|
|
104
|
-
|
|
105
|
-
except Exception as e:
|
|
106
|
-
console.print(f"Site: {SITE_NAME}, request search error: {e}")
|
|
107
|
-
|
|
108
|
-
# Add found titles to media search manager
|
|
109
|
-
for dict_title in response.json()['data']:
|
|
110
|
-
media_search_manager.add_media({
|
|
111
|
-
'id': dict_title.get('id'),
|
|
112
|
-
'slug': dict_title.get('slug'),
|
|
113
|
-
'name': dict_title.get('name'),
|
|
114
|
-
'type': dict_title.get('type'),
|
|
115
|
-
'score': dict_title.get('score')
|
|
116
|
-
})
|
|
117
|
-
|
|
118
|
-
# Return the number of titles found
|
|
119
|
-
return media_search_manager.get_length()
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
def run_get_select_title():
|
|
123
|
-
"""
|
|
124
|
-
Display a selection of titles and prompt the user to choose one.
|
|
125
|
-
"""
|
|
1
|
+
# 10.12.23
|
|
2
|
+
|
|
3
|
+
import sys
|
|
4
|
+
import json
|
|
5
|
+
import logging
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
# External libraries
|
|
9
|
+
import httpx
|
|
10
|
+
from bs4 import BeautifulSoup
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
# Internal utilities
|
|
14
|
+
from StreamingCommunity.Src.Util.console import console
|
|
15
|
+
from StreamingCommunity.Src.Util._jsonConfig import config_manager
|
|
16
|
+
from StreamingCommunity.Src.Util.headers import get_headers
|
|
17
|
+
from StreamingCommunity.Src.Util.table import TVShowManager
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
# Logic class
|
|
22
|
+
from StreamingCommunity.Src.Api.Template import get_select_title
|
|
23
|
+
from StreamingCommunity.Src.Api.Template.Util import search_domain
|
|
24
|
+
from StreamingCommunity.Src.Api.Template.Class.SearchType import MediaManager
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
# Config
|
|
28
|
+
from .costant import SITE_NAME
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
# Variable
|
|
32
|
+
media_search_manager = MediaManager()
|
|
33
|
+
table_show_manager = TVShowManager()
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def get_version(text: str):
|
|
38
|
+
"""
|
|
39
|
+
Extracts the version from the HTML text of a webpage.
|
|
40
|
+
|
|
41
|
+
Parameters:
|
|
42
|
+
- text (str): The HTML text of the webpage.
|
|
43
|
+
|
|
44
|
+
Returns:
|
|
45
|
+
str: The version extracted from the webpage.
|
|
46
|
+
list: Top 10 titles headlines for today.
|
|
47
|
+
"""
|
|
48
|
+
try:
|
|
49
|
+
|
|
50
|
+
# Parse request to site
|
|
51
|
+
soup = BeautifulSoup(text, "html.parser")
|
|
52
|
+
|
|
53
|
+
# Extract version
|
|
54
|
+
version = json.loads(soup.find("div", {"id": "app"}).get("data-page"))['version']
|
|
55
|
+
#console.print(f"[cyan]Get version [white]=> [red]{version} \n")
|
|
56
|
+
|
|
57
|
+
return version
|
|
58
|
+
|
|
59
|
+
except Exception as e:
|
|
60
|
+
logging.error(f"Error extracting version: {e}")
|
|
61
|
+
raise
|
|
62
|
+
|
|
63
|
+
|
|
64
|
+
def get_version_and_domain():
|
|
65
|
+
"""
|
|
66
|
+
Retrieve the current version and domain of the site.
|
|
67
|
+
|
|
68
|
+
This function performs the following steps:
|
|
69
|
+
- Determines the correct domain to use for the site by searching for a specific meta tag.
|
|
70
|
+
- Fetches the content of the site to extract the version information.
|
|
71
|
+
"""
|
|
72
|
+
|
|
73
|
+
# Find new domain if prev dont work
|
|
74
|
+
domain_to_use, base_url = search_domain(SITE_NAME, f"https://{SITE_NAME}")
|
|
75
|
+
|
|
76
|
+
# Extract version from the response
|
|
77
|
+
version = get_version(httpx.get(base_url, headers={'user-agent': get_headers()}).text)
|
|
78
|
+
|
|
79
|
+
return version, domain_to_use
|
|
80
|
+
|
|
81
|
+
|
|
82
|
+
def title_search(title_search: str, domain: str) -> int:
|
|
83
|
+
"""
|
|
84
|
+
Search for titles based on a search query.
|
|
85
|
+
|
|
86
|
+
Parameters:
|
|
87
|
+
- title_search (str): The title to search for.
|
|
88
|
+
- domain (str): The domain to search on.
|
|
89
|
+
|
|
90
|
+
Returns:
|
|
91
|
+
int: The number of titles found.
|
|
92
|
+
"""
|
|
93
|
+
|
|
94
|
+
max_timeout = config_manager.get_int("REQUESTS", "timeout")
|
|
95
|
+
|
|
96
|
+
# Send request to search for titles ( replace à to a and space to "+" )
|
|
97
|
+
try:
|
|
98
|
+
response = httpx.get(
|
|
99
|
+
url=f"https://{SITE_NAME}.{domain}/api/search?q={title_search.replace(' ', '+')}",
|
|
100
|
+
headers={'user-agent': get_headers()},
|
|
101
|
+
timeout=max_timeout
|
|
102
|
+
)
|
|
103
|
+
response.raise_for_status()
|
|
104
|
+
|
|
105
|
+
except Exception as e:
|
|
106
|
+
console.print(f"Site: {SITE_NAME}, request search error: {e}")
|
|
107
|
+
|
|
108
|
+
# Add found titles to media search manager
|
|
109
|
+
for dict_title in response.json()['data']:
|
|
110
|
+
media_search_manager.add_media({
|
|
111
|
+
'id': dict_title.get('id'),
|
|
112
|
+
'slug': dict_title.get('slug'),
|
|
113
|
+
'name': dict_title.get('name'),
|
|
114
|
+
'type': dict_title.get('type'),
|
|
115
|
+
'score': dict_title.get('score')
|
|
116
|
+
})
|
|
117
|
+
|
|
118
|
+
# Return the number of titles found
|
|
119
|
+
return media_search_manager.get_length()
|
|
120
|
+
|
|
121
|
+
|
|
122
|
+
def run_get_select_title():
|
|
123
|
+
"""
|
|
124
|
+
Display a selection of titles and prompt the user to choose one.
|
|
125
|
+
"""
|
|
126
126
|
return get_select_title(table_show_manager, media_search_manager)
|