StreamingCommunity 2.3.0__py3-none-any.whl → 2.5.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/run.py +61 -7
- {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/METADATA +88 -18
- StreamingCommunity-2.5.0.dist-info/RECORD +8 -0
- StreamingCommunity/Api/Player/Helper/Vixcloud/js_parser.py +0 -143
- StreamingCommunity/Api/Player/Helper/Vixcloud/util.py +0 -136
- StreamingCommunity/Api/Player/ddl.py +0 -89
- StreamingCommunity/Api/Player/maxstream.py +0 -151
- StreamingCommunity/Api/Player/supervideo.py +0 -194
- StreamingCommunity/Api/Player/vixcloud.py +0 -273
- StreamingCommunity/Api/Site/1337xx/__init__.py +0 -51
- StreamingCommunity/Api/Site/1337xx/costant.py +0 -15
- StreamingCommunity/Api/Site/1337xx/site.py +0 -89
- StreamingCommunity/Api/Site/1337xx/title.py +0 -66
- StreamingCommunity/Api/Site/altadefinizione/__init__.py +0 -51
- StreamingCommunity/Api/Site/altadefinizione/costant.py +0 -19
- StreamingCommunity/Api/Site/altadefinizione/film.py +0 -74
- StreamingCommunity/Api/Site/altadefinizione/site.py +0 -95
- StreamingCommunity/Api/Site/animeunity/__init__.py +0 -51
- StreamingCommunity/Api/Site/animeunity/costant.py +0 -19
- StreamingCommunity/Api/Site/animeunity/film_serie.py +0 -135
- StreamingCommunity/Api/Site/animeunity/site.py +0 -175
- StreamingCommunity/Api/Site/animeunity/util/ScrapeSerie.py +0 -97
- StreamingCommunity/Api/Site/cb01new/__init__.py +0 -52
- StreamingCommunity/Api/Site/cb01new/costant.py +0 -19
- StreamingCommunity/Api/Site/cb01new/film.py +0 -73
- StreamingCommunity/Api/Site/cb01new/site.py +0 -83
- StreamingCommunity/Api/Site/ddlstreamitaly/__init__.py +0 -56
- StreamingCommunity/Api/Site/ddlstreamitaly/costant.py +0 -20
- StreamingCommunity/Api/Site/ddlstreamitaly/series.py +0 -146
- StreamingCommunity/Api/Site/ddlstreamitaly/site.py +0 -99
- StreamingCommunity/Api/Site/ddlstreamitaly/util/ScrapeSerie.py +0 -85
- StreamingCommunity/Api/Site/guardaserie/__init__.py +0 -51
- StreamingCommunity/Api/Site/guardaserie/costant.py +0 -19
- StreamingCommunity/Api/Site/guardaserie/series.py +0 -198
- StreamingCommunity/Api/Site/guardaserie/site.py +0 -90
- StreamingCommunity/Api/Site/guardaserie/util/ScrapeSerie.py +0 -110
- StreamingCommunity/Api/Site/ilcorsaronero/__init__.py +0 -52
- StreamingCommunity/Api/Site/ilcorsaronero/costant.py +0 -19
- StreamingCommunity/Api/Site/ilcorsaronero/site.py +0 -72
- StreamingCommunity/Api/Site/ilcorsaronero/title.py +0 -46
- StreamingCommunity/Api/Site/ilcorsaronero/util/ilCorsarScraper.py +0 -149
- StreamingCommunity/Api/Site/mostraguarda/__init__.py +0 -49
- StreamingCommunity/Api/Site/mostraguarda/costant.py +0 -19
- StreamingCommunity/Api/Site/mostraguarda/film.py +0 -101
- StreamingCommunity/Api/Site/streamingcommunity/__init__.py +0 -56
- StreamingCommunity/Api/Site/streamingcommunity/costant.py +0 -19
- StreamingCommunity/Api/Site/streamingcommunity/film.py +0 -75
- StreamingCommunity/Api/Site/streamingcommunity/series.py +0 -206
- StreamingCommunity/Api/Site/streamingcommunity/site.py +0 -139
- StreamingCommunity/Api/Site/streamingcommunity/util/ScrapeSerie.py +0 -123
- StreamingCommunity/Api/Template/Class/SearchType.py +0 -101
- StreamingCommunity/Api/Template/Util/__init__.py +0 -5
- StreamingCommunity/Api/Template/Util/get_domain.py +0 -137
- StreamingCommunity/Api/Template/Util/manage_ep.py +0 -179
- StreamingCommunity/Api/Template/Util/recall_search.py +0 -37
- StreamingCommunity/Api/Template/__init__.py +0 -3
- StreamingCommunity/Api/Template/site.py +0 -87
- StreamingCommunity/Lib/Downloader/HLS/downloader.py +0 -955
- StreamingCommunity/Lib/Downloader/HLS/proxyes.py +0 -110
- StreamingCommunity/Lib/Downloader/HLS/segments.py +0 -564
- StreamingCommunity/Lib/Downloader/MP4/downloader.py +0 -155
- StreamingCommunity/Lib/Downloader/TOR/downloader.py +0 -296
- StreamingCommunity/Lib/Downloader/__init__.py +0 -5
- StreamingCommunity/Lib/FFmpeg/__init__.py +0 -4
- StreamingCommunity/Lib/FFmpeg/capture.py +0 -170
- StreamingCommunity/Lib/FFmpeg/command.py +0 -296
- StreamingCommunity/Lib/FFmpeg/util.py +0 -249
- StreamingCommunity/Lib/M3U8/__init__.py +0 -6
- StreamingCommunity/Lib/M3U8/decryptor.py +0 -164
- StreamingCommunity/Lib/M3U8/estimator.py +0 -229
- StreamingCommunity/Lib/M3U8/parser.py +0 -666
- StreamingCommunity/Lib/M3U8/url_fixer.py +0 -52
- StreamingCommunity/Lib/TMBD/__init__.py +0 -2
- StreamingCommunity/Lib/TMBD/obj_tmbd.py +0 -39
- StreamingCommunity/Lib/TMBD/tmdb.py +0 -346
- StreamingCommunity/Upload/update.py +0 -67
- StreamingCommunity/Upload/version.py +0 -5
- StreamingCommunity/Util/_jsonConfig.py +0 -204
- StreamingCommunity/Util/call_stack.py +0 -42
- StreamingCommunity/Util/color.py +0 -20
- StreamingCommunity/Util/console.py +0 -12
- StreamingCommunity/Util/ffmpeg_installer.py +0 -351
- StreamingCommunity/Util/headers.py +0 -147
- StreamingCommunity/Util/logger.py +0 -53
- StreamingCommunity/Util/message.py +0 -64
- StreamingCommunity/Util/os.py +0 -545
- StreamingCommunity/Util/table.py +0 -229
- StreamingCommunity-2.3.0.dist-info/RECORD +0 -92
- {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/LICENSE +0 -0
- {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/WHEEL +0 -0
- {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/entry_points.txt +0 -0
- {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/top_level.txt +0 -0
StreamingCommunity/run.py
CHANGED
|
@@ -125,7 +125,6 @@ def initialize():
|
|
|
125
125
|
|
|
126
126
|
|
|
127
127
|
def main():
|
|
128
|
-
|
|
129
128
|
start = time.time()
|
|
130
129
|
|
|
131
130
|
# Create logger
|
|
@@ -136,9 +135,39 @@ def main():
|
|
|
136
135
|
search_functions = load_search_functions()
|
|
137
136
|
logging.info(f"Load module in: {time.time() - start} s")
|
|
138
137
|
|
|
139
|
-
# Create
|
|
140
|
-
parser = argparse.ArgumentParser(
|
|
141
|
-
|
|
138
|
+
# Create argument parser
|
|
139
|
+
parser = argparse.ArgumentParser(
|
|
140
|
+
description='Script to download movies and series from the internet. Use these commands to configure the script and control its behavior.'
|
|
141
|
+
)
|
|
142
|
+
|
|
143
|
+
# Add arguments for the main configuration parameters
|
|
144
|
+
parser.add_argument(
|
|
145
|
+
'--add_siteName', type=bool, help='Enable or disable adding the site name to the file name (e.g., true/false).'
|
|
146
|
+
)
|
|
147
|
+
parser.add_argument(
|
|
148
|
+
'--disable_searchDomain', type=bool, help='Enable or disable searching in configured domains (e.g., true/false).'
|
|
149
|
+
)
|
|
150
|
+
parser.add_argument(
|
|
151
|
+
'--not_close', type=bool, help='If set to true, the script will not close the console after execution (e.g., true/false).'
|
|
152
|
+
)
|
|
153
|
+
|
|
154
|
+
# Add arguments for M3U8 configuration
|
|
155
|
+
parser.add_argument(
|
|
156
|
+
'--default_video_worker', type=int, help='Number of workers for video during M3U8 download (default: 12).'
|
|
157
|
+
)
|
|
158
|
+
parser.add_argument(
|
|
159
|
+
'--default_audio_worker', type=int, help='Number of workers for audio during M3U8 download (default: 12).'
|
|
160
|
+
)
|
|
161
|
+
|
|
162
|
+
# Add options for audio and subtitles
|
|
163
|
+
parser.add_argument(
|
|
164
|
+
'--specific_list_audio', type=str, help='Comma-separated list of specific audio languages to download (e.g., ita,eng).'
|
|
165
|
+
)
|
|
166
|
+
parser.add_argument(
|
|
167
|
+
'--specific_list_subtitles', type=str, help='Comma-separated list of specific subtitle languages to download (e.g., eng,spa).'
|
|
168
|
+
)
|
|
169
|
+
|
|
170
|
+
# Add arguments for search functions
|
|
142
171
|
color_map = {
|
|
143
172
|
"anime": "red",
|
|
144
173
|
"film_serie": "yellow",
|
|
@@ -153,10 +182,35 @@ def main():
|
|
|
153
182
|
long_option = alias
|
|
154
183
|
parser.add_argument(f'-{short_option}', f'--{long_option}', action='store_true', help=f'Search for {alias.split("_")[0]} on streaming platforms.')
|
|
155
184
|
|
|
156
|
-
# Parse command
|
|
185
|
+
# Parse command-line arguments
|
|
157
186
|
args = parser.parse_args()
|
|
158
187
|
|
|
159
|
-
#
|
|
188
|
+
# Map command-line arguments to the config values
|
|
189
|
+
config_updates = {}
|
|
190
|
+
|
|
191
|
+
if args.add_siteName is not None:
|
|
192
|
+
config_updates['DEFAULT.add_siteName'] = args.add_siteName
|
|
193
|
+
if args.disable_searchDomain is not None:
|
|
194
|
+
config_updates['DEFAULT.disable_searchDomain'] = args.disable_searchDomain
|
|
195
|
+
if args.not_close is not None:
|
|
196
|
+
config_updates['DEFAULT.not_close'] = args.not_close
|
|
197
|
+
if args.default_video_worker is not None:
|
|
198
|
+
config_updates['M3U8_DOWNLOAD.default_video_worker'] = args.default_video_worker
|
|
199
|
+
if args.default_audio_worker is not None:
|
|
200
|
+
config_updates['M3U8_DOWNLOAD.default_audio_worker'] = args.default_audio_worker
|
|
201
|
+
if args.specific_list_audio is not None:
|
|
202
|
+
config_updates['M3U8_DOWNLOAD.specific_list_audio'] = args.specific_list_audio.split(',')
|
|
203
|
+
if args.specific_list_subtitles is not None:
|
|
204
|
+
config_updates['M3U8_DOWNLOAD.specific_list_subtitles'] = args.specific_list_subtitles.split(',')
|
|
205
|
+
|
|
206
|
+
# Apply the updates to the config file
|
|
207
|
+
for key, value in config_updates.items():
|
|
208
|
+
section, option = key.split('.')
|
|
209
|
+
config_manager.set_key(section, option, value)
|
|
210
|
+
|
|
211
|
+
config_manager.write_config()
|
|
212
|
+
|
|
213
|
+
# Map command-line arguments to functions
|
|
160
214
|
arg_to_function = {alias: func for alias, (func, _) in search_functions.items()}
|
|
161
215
|
|
|
162
216
|
# Check which argument is provided and run the corresponding function
|
|
@@ -188,4 +242,4 @@ def main():
|
|
|
188
242
|
run_function(input_to_function[category])
|
|
189
243
|
else:
|
|
190
244
|
console.print("[red]Invalid category.")
|
|
191
|
-
sys.exit(0)
|
|
245
|
+
sys.exit(0)
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: StreamingCommunity
|
|
3
|
-
Version: 2.
|
|
3
|
+
Version: 2.5.0
|
|
4
4
|
Summary: UNKNOWN
|
|
5
5
|
Home-page: https://github.com/Lovi-0/StreamingCommunity
|
|
6
6
|
Author: Lovi-0
|
|
@@ -21,10 +21,10 @@ Requires-Dist: unidecode
|
|
|
21
21
|
Requires-Dist: jsbeautifier
|
|
22
22
|
Requires-Dist: pathvalidate
|
|
23
23
|
Requires-Dist: pycryptodomex
|
|
24
|
-
Requires-Dist:
|
|
24
|
+
Requires-Dist: googlesearch-python
|
|
25
|
+
Requires-Dist: fake-useragent
|
|
25
26
|
Requires-Dist: qbittorrent-api
|
|
26
27
|
Requires-Dist: python-qbittorrent
|
|
27
|
-
Requires-Dist: googlesearch-python
|
|
28
28
|
|
|
29
29
|
<p align="center">
|
|
30
30
|
<img src="https://i.ibb.co/PFnjvBc/immagine-2024-12-26-180318047.png" alt="Project Logo" width="700"/>
|
|
@@ -37,9 +37,6 @@ Requires-Dist: googlesearch-python
|
|
|
37
37
|
<a href="https://www.paypal.com/donate/?hosted_button_id=UXTWMT8P6HE2C">
|
|
38
38
|
<img src="https://img.shields.io/badge/_-Donate-red.svg?logo=githubsponsors&labelColor=555555&style=for-the-badge" alt="Donate"/>
|
|
39
39
|
</a>
|
|
40
|
-
<a href="https://github.com/Lovi-0/StreamingCommunity/blob/main/LICENSE">
|
|
41
|
-
<img src="https://img.shields.io/badge/License-GPL_3.0-blue.svg?style=for-the-badge" alt="License"/>
|
|
42
|
-
</a>
|
|
43
40
|
<a href="https://github.com/Lovi-0/StreamingCommunity/commits">
|
|
44
41
|
<img src="https://img.shields.io/github/commit-activity/m/Lovi-0/StreamingCommunity?label=commits&style=for-the-badge" alt="Commits"/>
|
|
45
42
|
</a>
|
|
@@ -49,17 +46,14 @@ Requires-Dist: googlesearch-python
|
|
|
49
46
|
</p>
|
|
50
47
|
|
|
51
48
|
<p align="center">
|
|
49
|
+
<a href="https://github.com/Lovi-0/StreamingCommunity/blob/main/LICENSE">
|
|
50
|
+
<img src="https://img.shields.io/badge/License-GPL_3.0-blue.svg?style=for-the-badge" alt="License"/>
|
|
51
|
+
</a>
|
|
52
52
|
<a href="https://pypi.org/project/streamingcommunity">
|
|
53
53
|
<img src="https://img.shields.io/pypi/dm/streamingcommunity?style=for-the-badge" alt="PyPI Downloads"/>
|
|
54
54
|
</a>
|
|
55
|
-
<a href="https://github.com/Lovi-0/StreamingCommunity/network/members">
|
|
56
|
-
<img src="https://img.shields.io/github/forks/Lovi-0/StreamingCommunity?style=for-the-badge" alt="Forks"/>
|
|
57
|
-
</a>
|
|
58
55
|
<a href="https://github.com/Lovi-0/StreamingCommunity">
|
|
59
|
-
<img src="https://img.shields.io/
|
|
60
|
-
</a>
|
|
61
|
-
<a href="https://github.com/Lovi-0/StreamingCommunity">
|
|
62
|
-
<img src="https://img.shields.io/github/repo-size/Lovi-0/StreamingCommunity?style=for-the-badge" alt="Repo Size"/>
|
|
56
|
+
<img src="https://img.shields.io/endpoint?url=https://raw.githubusercontent.com/Lovi-0/StreamingCommunity/main/Test/Util/loc-badge.json&style=for-the-badge" alt="Lines of Code"/>
|
|
63
57
|
</a>
|
|
64
58
|
</p>
|
|
65
59
|
|
|
@@ -69,6 +63,7 @@ Requires-Dist: googlesearch-python
|
|
|
69
63
|
- 🛠️ [Installation](#installation)
|
|
70
64
|
- 📦 [PyPI Installation](#1-pypi-installation)
|
|
71
65
|
- 🔄 [Automatic Installation](#2-automatic-installation)
|
|
66
|
+
- 🔧 [Binary Location](#binary-location)
|
|
72
67
|
- 📝 [Manual Installation](#3-manual-installation)
|
|
73
68
|
- 💻 [Win 7](https://github.com/Ghost6446/StreamingCommunity_api/wiki/Installation#win-7)
|
|
74
69
|
- 📱 [Termux](https://github.com/Ghost6446/StreamingCommunity_api/wiki/Termux)
|
|
@@ -127,7 +122,7 @@ Run the script:
|
|
|
127
122
|
python run_streaming.py
|
|
128
123
|
```
|
|
129
124
|
|
|
130
|
-
|
|
125
|
+
### Updating via PyPI
|
|
131
126
|
|
|
132
127
|
```bash
|
|
133
128
|
pip install --upgrade StreamingCommunity
|
|
@@ -182,6 +177,69 @@ source .venv/bin/activate && python test_run.py && deactivate
|
|
|
182
177
|
./test_run.py
|
|
183
178
|
```
|
|
184
179
|
|
|
180
|
+
## Binary Location
|
|
181
|
+
|
|
182
|
+
### Default Locations
|
|
183
|
+
- **Windows**: `C:\binary`
|
|
184
|
+
- **MacOS**: `~/Applications/binary`
|
|
185
|
+
- **Linux**: `~/.local/bin/binary`
|
|
186
|
+
|
|
187
|
+
You can customize these locations by following these steps for your operating system:
|
|
188
|
+
|
|
189
|
+
#### Windows
|
|
190
|
+
1. Move the binary folder from `C:\binary` to your desired location
|
|
191
|
+
2. Add the new path to Windows environment variables:
|
|
192
|
+
- Open Start menu and search for "Environment Variables"
|
|
193
|
+
- Click "Edit the system environment variables"
|
|
194
|
+
- Click "Environment Variables" button
|
|
195
|
+
- Under "System Variables", find and select "Path"
|
|
196
|
+
- Click "Edit"
|
|
197
|
+
- Add the new binary folder path
|
|
198
|
+
- Click "OK" to save changes
|
|
199
|
+
|
|
200
|
+
For detailed Windows PATH instructions, see the [Windows PATH guide](https://www.eukhost.com/kb/how-to-add-to-the-path-on-windows-10-and-windows-11/).
|
|
201
|
+
|
|
202
|
+
#### MacOS
|
|
203
|
+
1. Move the binary folder from `~/Applications/binary` to your desired location
|
|
204
|
+
2. Add the new path to your shell's configuration file:
|
|
205
|
+
```bash
|
|
206
|
+
# For bash (edit ~/.bash_profile)
|
|
207
|
+
export PATH="/your/custom/path:$PATH"
|
|
208
|
+
|
|
209
|
+
# For zsh (edit ~/.zshrc)
|
|
210
|
+
export PATH="/your/custom/path:$PATH"
|
|
211
|
+
```
|
|
212
|
+
3. Reload your shell configuration:
|
|
213
|
+
```bash
|
|
214
|
+
# For bash
|
|
215
|
+
source ~/.bash_profile
|
|
216
|
+
|
|
217
|
+
# For zsh
|
|
218
|
+
source ~/.zshrc
|
|
219
|
+
```
|
|
220
|
+
|
|
221
|
+
#### Linux
|
|
222
|
+
1. Move the binary folder from `~/.local/bin/binary` to your desired location
|
|
223
|
+
2. Add the new path to your shell's configuration file:
|
|
224
|
+
```bash
|
|
225
|
+
# For bash (edit ~/.bashrc)
|
|
226
|
+
export PATH="/your/custom/path:$PATH"
|
|
227
|
+
|
|
228
|
+
# For zsh (edit ~/.zshrc)
|
|
229
|
+
export PATH="/your/custom/path:$PATH"
|
|
230
|
+
```
|
|
231
|
+
3. Apply the changes:
|
|
232
|
+
```bash
|
|
233
|
+
source ~/.bashrc # for bash
|
|
234
|
+
# or
|
|
235
|
+
source ~/.zshrc # for zsh
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
> [!IMPORTANT]
|
|
239
|
+
> After moving the binary folder, ensure that all executables (ffmpeg, ffprobe, ffplay) are present in the new location and have the correct permissions:
|
|
240
|
+
> - Windows: `.exe` extensions required
|
|
241
|
+
> - MacOS/Linux: Ensure files have execute permissions (`chmod +x filename`)
|
|
242
|
+
|
|
185
243
|
## 3. Manual Installation
|
|
186
244
|
|
|
187
245
|
### Requirements 📋
|
|
@@ -346,6 +404,10 @@ forced-ita hin - Hindi pol - Polish tur - Turkish
|
|
|
346
404
|
ind - Indonesian vie - Vietnamese
|
|
347
405
|
```
|
|
348
406
|
|
|
407
|
+
> [!NOTE]
|
|
408
|
+
> When using subtitles on Windows, please note that the default Windows Media Player may not display them correctly. We recommend using VLC Media Player for proper subtitle display and optimal playback experience.
|
|
409
|
+
|
|
410
|
+
|
|
349
411
|
> [!IMPORTANT]
|
|
350
412
|
> Language code availability may vary by site. Some platforms might:
|
|
351
413
|
>
|
|
@@ -363,6 +425,14 @@ forced-ita hin - Hindi pol - Polish tur - Turkish
|
|
|
363
425
|
> "specific_list_subtitles": ["ita", "eng", "spa"]
|
|
364
426
|
> ```
|
|
365
427
|
|
|
428
|
+
For the best viewing experience, we recommend using VLC Media Player:
|
|
429
|
+
- Supports all video formats and codecs
|
|
430
|
+
- Properly displays embedded subtitles
|
|
431
|
+
- Available for all major operating systems
|
|
432
|
+
- Free and open-source
|
|
433
|
+
|
|
434
|
+
You can download VLC Media Player from the [official website](https://www.videolan.org/vlc/).
|
|
435
|
+
|
|
366
436
|
## M3U8_PARSER Settings
|
|
367
437
|
|
|
368
438
|
```json
|
|
@@ -429,14 +499,14 @@ The `run-container` command mounts also the `config.json` file, so any change to
|
|
|
429
499
|
| Website | Status |
|
|
430
500
|
|:-------------------|:------:|
|
|
431
501
|
| [1337xx](https://1337xx.to/) | ✅ |
|
|
432
|
-
| [
|
|
502
|
+
| [AltadefinizioneGratis](https://altadefinizionegratis.site/) | ✅ |
|
|
433
503
|
| [AnimeUnity](https://animeunity.so/) | ✅ |
|
|
434
504
|
| [Ilcorsaronero](https://ilcorsaronero.link/) | ✅ |
|
|
435
|
-
| [CB01New](https://cb01new.
|
|
505
|
+
| [CB01New](https://cb01new.video/) | ✅ |
|
|
436
506
|
| [DDLStreamItaly](https://ddlstreamitaly.co/) | ✅ |
|
|
437
|
-
| [GuardaSerie](https://guardaserie.
|
|
507
|
+
| [GuardaSerie](https://guardaserie.meme/) | ✅ |
|
|
438
508
|
| [MostraGuarda](https://mostraguarda.stream/) | ✅ |
|
|
439
|
-
| [StreamingCommunity](https://streamingcommunity.
|
|
509
|
+
| [StreamingCommunity](https://streamingcommunity.paris/) | ✅ |
|
|
440
510
|
|
|
441
511
|
|
|
442
512
|
# Tutorials
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
StreamingCommunity/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
2
|
+
StreamingCommunity/run.py,sha256=cWm5QFKMQU7zVuNJzlL7Fv2fT1Apl9JVHm2ZBDiKYh4,9011
|
|
3
|
+
StreamingCommunity-2.5.0.dist-info/LICENSE,sha256=IwGE9guuL-ryRPEKi6wFPI_zOhg7zDZbTYuHbSt_SAk,35823
|
|
4
|
+
StreamingCommunity-2.5.0.dist-info/METADATA,sha256=jcuxKg4KZOPwqqB44PABoMBmBvnlaNmHJ4Nyg8rHZGE,17016
|
|
5
|
+
StreamingCommunity-2.5.0.dist-info/WHEEL,sha256=tZoeGjtWxWRfdplE7E3d45VPlLNQnvbKiYnx7gwAy8A,92
|
|
6
|
+
StreamingCommunity-2.5.0.dist-info/entry_points.txt,sha256=-iQU6qfeHFwauAg4iZhifWhNZAkiV-x3XuEauo_EjUc,68
|
|
7
|
+
StreamingCommunity-2.5.0.dist-info/top_level.txt,sha256=YsOcxKP-WOhWpIWgBlh0coll9XUx7aqmRPT7kmt3fH0,19
|
|
8
|
+
StreamingCommunity-2.5.0.dist-info/RECORD,,
|
|
@@ -1,143 +0,0 @@
|
|
|
1
|
-
# 26.11.24
|
|
2
|
-
# !!! DIO CANErino
|
|
3
|
-
|
|
4
|
-
import re
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
class JavaScriptParser:
|
|
8
|
-
@staticmethod
|
|
9
|
-
def fix_string(ss):
|
|
10
|
-
if ss is None:
|
|
11
|
-
return None
|
|
12
|
-
|
|
13
|
-
ss = str(ss)
|
|
14
|
-
ss = ss.encode('utf-8').decode('unicode-escape')
|
|
15
|
-
ss = ss.strip("\"'")
|
|
16
|
-
ss = ss.strip()
|
|
17
|
-
|
|
18
|
-
return ss
|
|
19
|
-
|
|
20
|
-
@staticmethod
|
|
21
|
-
def fix_url(url):
|
|
22
|
-
if url is None:
|
|
23
|
-
return None
|
|
24
|
-
|
|
25
|
-
url = url.replace('\\/', '/')
|
|
26
|
-
return url
|
|
27
|
-
|
|
28
|
-
@staticmethod
|
|
29
|
-
def parse_value(value):
|
|
30
|
-
value = JavaScriptParser.fix_string(value)
|
|
31
|
-
|
|
32
|
-
if 'http' in str(value) or 'https' in str(value):
|
|
33
|
-
return JavaScriptParser.fix_url(value)
|
|
34
|
-
|
|
35
|
-
if value is None or str(value).lower() == 'null':
|
|
36
|
-
return None
|
|
37
|
-
if str(value).lower() == 'true':
|
|
38
|
-
return True
|
|
39
|
-
if str(value).lower() == 'false':
|
|
40
|
-
return False
|
|
41
|
-
|
|
42
|
-
try:
|
|
43
|
-
return int(value)
|
|
44
|
-
except ValueError:
|
|
45
|
-
try:
|
|
46
|
-
return float(value)
|
|
47
|
-
except ValueError:
|
|
48
|
-
pass
|
|
49
|
-
|
|
50
|
-
return value
|
|
51
|
-
|
|
52
|
-
@staticmethod
|
|
53
|
-
def parse_object(obj_str):
|
|
54
|
-
obj_str = obj_str.strip('{}').strip()
|
|
55
|
-
|
|
56
|
-
result = {}
|
|
57
|
-
key_value_pairs = re.findall(r'([\'"]?[\w]+[\'"]?)\s*:\s*([^,{}]+|{[^}]*}|\[[^\]]*\]|\'[^\']*\'|"[^"]*")', obj_str)
|
|
58
|
-
|
|
59
|
-
for key, value in key_value_pairs:
|
|
60
|
-
key = JavaScriptParser.fix_string(key)
|
|
61
|
-
value = value.strip()
|
|
62
|
-
|
|
63
|
-
if value.startswith('{'):
|
|
64
|
-
result[key] = JavaScriptParser.parse_object(value)
|
|
65
|
-
elif value.startswith('['):
|
|
66
|
-
result[key] = JavaScriptParser.parse_array(value)
|
|
67
|
-
else:
|
|
68
|
-
result[key] = JavaScriptParser.parse_value(value)
|
|
69
|
-
|
|
70
|
-
return result
|
|
71
|
-
|
|
72
|
-
@staticmethod
|
|
73
|
-
def parse_array(arr_str):
|
|
74
|
-
arr_str = arr_str.strip('[]').strip()
|
|
75
|
-
result = []
|
|
76
|
-
|
|
77
|
-
elements = []
|
|
78
|
-
current_elem = ""
|
|
79
|
-
brace_count = 0
|
|
80
|
-
in_string = False
|
|
81
|
-
quote_type = None
|
|
82
|
-
|
|
83
|
-
for char in arr_str:
|
|
84
|
-
if char in ['"', "'"]:
|
|
85
|
-
if not in_string:
|
|
86
|
-
in_string = True
|
|
87
|
-
quote_type = char
|
|
88
|
-
elif quote_type == char:
|
|
89
|
-
in_string = False
|
|
90
|
-
quote_type = None
|
|
91
|
-
|
|
92
|
-
if not in_string:
|
|
93
|
-
if char == '{':
|
|
94
|
-
brace_count += 1
|
|
95
|
-
elif char == '}':
|
|
96
|
-
brace_count -= 1
|
|
97
|
-
elif char == ',' and brace_count == 0:
|
|
98
|
-
elements.append(current_elem.strip())
|
|
99
|
-
current_elem = ""
|
|
100
|
-
continue
|
|
101
|
-
|
|
102
|
-
current_elem += char
|
|
103
|
-
|
|
104
|
-
if current_elem.strip():
|
|
105
|
-
elements.append(current_elem.strip())
|
|
106
|
-
|
|
107
|
-
for elem in elements:
|
|
108
|
-
elem = elem.strip()
|
|
109
|
-
|
|
110
|
-
if elem.startswith('{'):
|
|
111
|
-
result.append(JavaScriptParser.parse_object(elem))
|
|
112
|
-
elif 'active' in elem or 'url' in elem:
|
|
113
|
-
key_value_match = re.search(r'([\w]+)\":([^,}]+)', elem)
|
|
114
|
-
|
|
115
|
-
if key_value_match:
|
|
116
|
-
key = key_value_match.group(1)
|
|
117
|
-
value = key_value_match.group(2)
|
|
118
|
-
result[-1][key] = JavaScriptParser.parse_value(value.strip('"\\'))
|
|
119
|
-
else:
|
|
120
|
-
result.append(JavaScriptParser.parse_value(elem))
|
|
121
|
-
|
|
122
|
-
return result
|
|
123
|
-
|
|
124
|
-
@classmethod
|
|
125
|
-
def parse(cls, js_string):
|
|
126
|
-
assignments = re.findall(r'window\.(\w+)\s*=\s*([^;]+);?', js_string, re.DOTALL)
|
|
127
|
-
result = {}
|
|
128
|
-
|
|
129
|
-
for var_name, value in assignments:
|
|
130
|
-
value = value.strip()
|
|
131
|
-
|
|
132
|
-
if value.startswith('{'):
|
|
133
|
-
result[var_name] = cls.parse_object(value)
|
|
134
|
-
elif value.startswith('['):
|
|
135
|
-
result[var_name] = cls.parse_array(value)
|
|
136
|
-
else:
|
|
137
|
-
result[var_name] = cls.parse_value(value)
|
|
138
|
-
|
|
139
|
-
can_play_fhd_match = re.search(r'window\.canPlayFHD\s*=\s*(\w+);?', js_string)
|
|
140
|
-
if can_play_fhd_match:
|
|
141
|
-
result['canPlayFHD'] = cls.parse_value(can_play_fhd_match.group(1))
|
|
142
|
-
|
|
143
|
-
return result
|
|
@@ -1,136 +0,0 @@
|
|
|
1
|
-
# 23.11.24
|
|
2
|
-
|
|
3
|
-
from typing import Dict, Any, List, Union
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
class Episode:
|
|
7
|
-
def __init__(self, data: Dict[str, Any]):
|
|
8
|
-
self.data = data
|
|
9
|
-
|
|
10
|
-
self.id: int = data.get('id', 0)
|
|
11
|
-
self.scws_id: int = data.get('scws_id', 0)
|
|
12
|
-
self.number: int = data.get('number', 1)
|
|
13
|
-
self.name: str = data.get('name', '')
|
|
14
|
-
self.plot: str = data.get('plot', '')
|
|
15
|
-
self.duration: int = data.get('duration', 0)
|
|
16
|
-
|
|
17
|
-
def __str__(self):
|
|
18
|
-
return f"Episode(id={self.id}, number={self.number}, name='{self.name}', plot='{self.plot}', duration={self.duration} sec)"
|
|
19
|
-
|
|
20
|
-
class EpisodeManager:
|
|
21
|
-
def __init__(self):
|
|
22
|
-
self.episodes: List[Episode] = []
|
|
23
|
-
|
|
24
|
-
def add(self, episode_data: Dict[str, Any]):
|
|
25
|
-
"""
|
|
26
|
-
Add a new episode to the manager.
|
|
27
|
-
|
|
28
|
-
Parameters:
|
|
29
|
-
- episode_data (Dict[str, Any]): A dictionary containing data for the new episode.
|
|
30
|
-
"""
|
|
31
|
-
episode = Episode(episode_data)
|
|
32
|
-
self.episodes.append(episode)
|
|
33
|
-
|
|
34
|
-
def get(self, index: int) -> Episode:
|
|
35
|
-
"""
|
|
36
|
-
Retrieve an episode by its index in the episodes list.
|
|
37
|
-
|
|
38
|
-
Parameters:
|
|
39
|
-
- index (int): The zero-based index of the episode to retrieve.
|
|
40
|
-
|
|
41
|
-
Returns:
|
|
42
|
-
Episode: The Episode object at the specified index.
|
|
43
|
-
"""
|
|
44
|
-
return self.episodes[index]
|
|
45
|
-
|
|
46
|
-
def length(self) -> int:
|
|
47
|
-
"""
|
|
48
|
-
Get the number of episodes in the manager.
|
|
49
|
-
|
|
50
|
-
Returns:
|
|
51
|
-
int: Number of episodes.
|
|
52
|
-
"""
|
|
53
|
-
return len(self.episodes)
|
|
54
|
-
|
|
55
|
-
def clear(self) -> None:
|
|
56
|
-
"""
|
|
57
|
-
This method clears the episodes list.
|
|
58
|
-
|
|
59
|
-
Parameters:
|
|
60
|
-
- self: The object instance.
|
|
61
|
-
"""
|
|
62
|
-
self.episodes.clear()
|
|
63
|
-
|
|
64
|
-
def __str__(self):
|
|
65
|
-
return f"EpisodeManager(num_episodes={len(self.episodes)})"
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
class Season:
|
|
69
|
-
def __init__(self, season_data: Dict[str, Union[int, str, None]]):
|
|
70
|
-
self.season_data = season_data
|
|
71
|
-
|
|
72
|
-
self.id: int = season_data.get('id', 0)
|
|
73
|
-
self.scws_id: int = season_data.get('scws_id', 0)
|
|
74
|
-
self.imdb_id: int = season_data.get('imdb_id', 0)
|
|
75
|
-
self.number: int = season_data.get('number', 0)
|
|
76
|
-
self.name: str = season_data.get('name', '')
|
|
77
|
-
self.slug: str = season_data.get('slug', '')
|
|
78
|
-
self.plot: str = season_data.get('plot', '')
|
|
79
|
-
self.type: str = season_data.get('type', '')
|
|
80
|
-
self.seasons_count: int = season_data.get('seasons_count', 0)
|
|
81
|
-
self.episodes: EpisodeManager = EpisodeManager()
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
class Stream:
|
|
85
|
-
def __init__(self, name: str, url: str, active: bool):
|
|
86
|
-
self.name = name
|
|
87
|
-
self.url = url
|
|
88
|
-
self.active = active
|
|
89
|
-
|
|
90
|
-
def __repr__(self):
|
|
91
|
-
return f"Stream(name={self.name!r}, url={self.url!r}, active={self.active!r})"
|
|
92
|
-
|
|
93
|
-
class StreamsCollection:
|
|
94
|
-
def __init__(self, streams: list):
|
|
95
|
-
self.streams = [Stream(**stream) for stream in streams]
|
|
96
|
-
|
|
97
|
-
def __repr__(self):
|
|
98
|
-
return f"StreamsCollection(streams={self.streams})"
|
|
99
|
-
|
|
100
|
-
def add_stream(self, name: str, url: str, active: bool):
|
|
101
|
-
self.streams.append(Stream(name, url, active))
|
|
102
|
-
|
|
103
|
-
def get_streams(self):
|
|
104
|
-
return self.streams
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
class WindowVideo:
|
|
108
|
-
def __init__(self, data: Dict[str, Any]):
|
|
109
|
-
self.data = data
|
|
110
|
-
self.id: int = data.get('id', '')
|
|
111
|
-
self.name: str = data.get('name', '')
|
|
112
|
-
self.filename: str = data.get('filename', '')
|
|
113
|
-
self.size: str = data.get('size', '')
|
|
114
|
-
self.quality: str = data.get('quality', '')
|
|
115
|
-
self.duration: str = data.get('duration', '')
|
|
116
|
-
self.views: int = data.get('views', '')
|
|
117
|
-
self.is_viewable: bool = data.get('is_viewable', '')
|
|
118
|
-
self.status: str = data.get('status', '')
|
|
119
|
-
self.fps: float = data.get('fps', '')
|
|
120
|
-
self.legacy: bool = data.get('legacy', '')
|
|
121
|
-
self.folder_id: int = data.get('folder_id', '')
|
|
122
|
-
self.created_at_diff: str = data.get('created_at_diff', '')
|
|
123
|
-
|
|
124
|
-
def __str__(self):
|
|
125
|
-
return f"WindowVideo(id={self.id}, name='{self.name}', filename='{self.filename}', size='{self.size}', quality='{self.quality}', duration='{self.duration}', views={self.views}, is_viewable={self.is_viewable}, status='{self.status}', fps={self.fps}, legacy={self.legacy}, folder_id={self.folder_id}, created_at_diff='{self.created_at_diff}')"
|
|
126
|
-
|
|
127
|
-
class WindowParameter:
|
|
128
|
-
def __init__(self, data: Dict[str, Any]):
|
|
129
|
-
self.data = data
|
|
130
|
-
params = data.get('params', {})
|
|
131
|
-
self.token: str = params.get('token', '')
|
|
132
|
-
self.expires: str = str(params.get('expires', ''))
|
|
133
|
-
self.url = data.get('url')
|
|
134
|
-
|
|
135
|
-
def __str__(self):
|
|
136
|
-
return (f"WindowParameter(token='{self.token}', expires='{self.expires}', url='{self.url}', data={self.data})")
|
|
@@ -1,89 +0,0 @@
|
|
|
1
|
-
# 14.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.Util._jsonConfig import config_manager
|
|
13
|
-
from StreamingCommunity.Util.headers import get_headers
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
# Variable
|
|
17
|
-
from StreamingCommunity.Api.Site.ddlstreamitaly.costant import COOKIE
|
|
18
|
-
max_timeout = config_manager.get_int("REQUESTS", "timeout")
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
class VideoSource:
|
|
22
|
-
def __init__(self) -> None:
|
|
23
|
-
"""
|
|
24
|
-
Initializes the VideoSource object with default values.
|
|
25
|
-
"""
|
|
26
|
-
self.headers = {'user-agent': get_headers()}
|
|
27
|
-
self.cookie = COOKIE
|
|
28
|
-
|
|
29
|
-
def setup(self, url: str) -> None:
|
|
30
|
-
"""
|
|
31
|
-
Sets up the video source with the provided URL.
|
|
32
|
-
|
|
33
|
-
Parameters:
|
|
34
|
-
- url (str): The URL of the video source.
|
|
35
|
-
"""
|
|
36
|
-
self.url = url
|
|
37
|
-
|
|
38
|
-
def make_request(self, url: str) -> str:
|
|
39
|
-
"""
|
|
40
|
-
Make an HTTP GET request to the provided URL.
|
|
41
|
-
|
|
42
|
-
Parameters:
|
|
43
|
-
- url (str): The URL to make the request to.
|
|
44
|
-
|
|
45
|
-
Returns:
|
|
46
|
-
- str: The response content if successful, None otherwise.
|
|
47
|
-
"""
|
|
48
|
-
try:
|
|
49
|
-
response = httpx.get(
|
|
50
|
-
url=url,
|
|
51
|
-
headers=self.headers,
|
|
52
|
-
cookies=self.cookie,
|
|
53
|
-
timeout=max_timeout
|
|
54
|
-
)
|
|
55
|
-
response.raise_for_status()
|
|
56
|
-
|
|
57
|
-
return response.text
|
|
58
|
-
|
|
59
|
-
except Exception as err:
|
|
60
|
-
logging.error(f"An error occurred: {err}")
|
|
61
|
-
|
|
62
|
-
return None
|
|
63
|
-
|
|
64
|
-
def get_playlist(self):
|
|
65
|
-
"""
|
|
66
|
-
Retrieves the playlist URL from the video source.
|
|
67
|
-
|
|
68
|
-
Returns:
|
|
69
|
-
- tuple: The mp4 link if found, None otherwise.
|
|
70
|
-
"""
|
|
71
|
-
try:
|
|
72
|
-
text = self.make_request(self.url)
|
|
73
|
-
|
|
74
|
-
if text:
|
|
75
|
-
soup = BeautifulSoup(text, "html.parser")
|
|
76
|
-
source = soup.find("source")
|
|
77
|
-
|
|
78
|
-
if source:
|
|
79
|
-
mp4_link = source.get("src")
|
|
80
|
-
return mp4_link
|
|
81
|
-
|
|
82
|
-
else:
|
|
83
|
-
logging.error("No <source> tag found in the HTML.")
|
|
84
|
-
|
|
85
|
-
else:
|
|
86
|
-
logging.error("Failed to retrieve content from the URL.")
|
|
87
|
-
|
|
88
|
-
except Exception as e:
|
|
89
|
-
logging.error(f"An error occurred while parsing the playlist: {e}")
|