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.

Files changed (92) hide show
  1. StreamingCommunity/run.py +61 -7
  2. {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/METADATA +88 -18
  3. StreamingCommunity-2.5.0.dist-info/RECORD +8 -0
  4. StreamingCommunity/Api/Player/Helper/Vixcloud/js_parser.py +0 -143
  5. StreamingCommunity/Api/Player/Helper/Vixcloud/util.py +0 -136
  6. StreamingCommunity/Api/Player/ddl.py +0 -89
  7. StreamingCommunity/Api/Player/maxstream.py +0 -151
  8. StreamingCommunity/Api/Player/supervideo.py +0 -194
  9. StreamingCommunity/Api/Player/vixcloud.py +0 -273
  10. StreamingCommunity/Api/Site/1337xx/__init__.py +0 -51
  11. StreamingCommunity/Api/Site/1337xx/costant.py +0 -15
  12. StreamingCommunity/Api/Site/1337xx/site.py +0 -89
  13. StreamingCommunity/Api/Site/1337xx/title.py +0 -66
  14. StreamingCommunity/Api/Site/altadefinizione/__init__.py +0 -51
  15. StreamingCommunity/Api/Site/altadefinizione/costant.py +0 -19
  16. StreamingCommunity/Api/Site/altadefinizione/film.py +0 -74
  17. StreamingCommunity/Api/Site/altadefinizione/site.py +0 -95
  18. StreamingCommunity/Api/Site/animeunity/__init__.py +0 -51
  19. StreamingCommunity/Api/Site/animeunity/costant.py +0 -19
  20. StreamingCommunity/Api/Site/animeunity/film_serie.py +0 -135
  21. StreamingCommunity/Api/Site/animeunity/site.py +0 -175
  22. StreamingCommunity/Api/Site/animeunity/util/ScrapeSerie.py +0 -97
  23. StreamingCommunity/Api/Site/cb01new/__init__.py +0 -52
  24. StreamingCommunity/Api/Site/cb01new/costant.py +0 -19
  25. StreamingCommunity/Api/Site/cb01new/film.py +0 -73
  26. StreamingCommunity/Api/Site/cb01new/site.py +0 -83
  27. StreamingCommunity/Api/Site/ddlstreamitaly/__init__.py +0 -56
  28. StreamingCommunity/Api/Site/ddlstreamitaly/costant.py +0 -20
  29. StreamingCommunity/Api/Site/ddlstreamitaly/series.py +0 -146
  30. StreamingCommunity/Api/Site/ddlstreamitaly/site.py +0 -99
  31. StreamingCommunity/Api/Site/ddlstreamitaly/util/ScrapeSerie.py +0 -85
  32. StreamingCommunity/Api/Site/guardaserie/__init__.py +0 -51
  33. StreamingCommunity/Api/Site/guardaserie/costant.py +0 -19
  34. StreamingCommunity/Api/Site/guardaserie/series.py +0 -198
  35. StreamingCommunity/Api/Site/guardaserie/site.py +0 -90
  36. StreamingCommunity/Api/Site/guardaserie/util/ScrapeSerie.py +0 -110
  37. StreamingCommunity/Api/Site/ilcorsaronero/__init__.py +0 -52
  38. StreamingCommunity/Api/Site/ilcorsaronero/costant.py +0 -19
  39. StreamingCommunity/Api/Site/ilcorsaronero/site.py +0 -72
  40. StreamingCommunity/Api/Site/ilcorsaronero/title.py +0 -46
  41. StreamingCommunity/Api/Site/ilcorsaronero/util/ilCorsarScraper.py +0 -149
  42. StreamingCommunity/Api/Site/mostraguarda/__init__.py +0 -49
  43. StreamingCommunity/Api/Site/mostraguarda/costant.py +0 -19
  44. StreamingCommunity/Api/Site/mostraguarda/film.py +0 -101
  45. StreamingCommunity/Api/Site/streamingcommunity/__init__.py +0 -56
  46. StreamingCommunity/Api/Site/streamingcommunity/costant.py +0 -19
  47. StreamingCommunity/Api/Site/streamingcommunity/film.py +0 -75
  48. StreamingCommunity/Api/Site/streamingcommunity/series.py +0 -206
  49. StreamingCommunity/Api/Site/streamingcommunity/site.py +0 -139
  50. StreamingCommunity/Api/Site/streamingcommunity/util/ScrapeSerie.py +0 -123
  51. StreamingCommunity/Api/Template/Class/SearchType.py +0 -101
  52. StreamingCommunity/Api/Template/Util/__init__.py +0 -5
  53. StreamingCommunity/Api/Template/Util/get_domain.py +0 -137
  54. StreamingCommunity/Api/Template/Util/manage_ep.py +0 -179
  55. StreamingCommunity/Api/Template/Util/recall_search.py +0 -37
  56. StreamingCommunity/Api/Template/__init__.py +0 -3
  57. StreamingCommunity/Api/Template/site.py +0 -87
  58. StreamingCommunity/Lib/Downloader/HLS/downloader.py +0 -955
  59. StreamingCommunity/Lib/Downloader/HLS/proxyes.py +0 -110
  60. StreamingCommunity/Lib/Downloader/HLS/segments.py +0 -564
  61. StreamingCommunity/Lib/Downloader/MP4/downloader.py +0 -155
  62. StreamingCommunity/Lib/Downloader/TOR/downloader.py +0 -296
  63. StreamingCommunity/Lib/Downloader/__init__.py +0 -5
  64. StreamingCommunity/Lib/FFmpeg/__init__.py +0 -4
  65. StreamingCommunity/Lib/FFmpeg/capture.py +0 -170
  66. StreamingCommunity/Lib/FFmpeg/command.py +0 -296
  67. StreamingCommunity/Lib/FFmpeg/util.py +0 -249
  68. StreamingCommunity/Lib/M3U8/__init__.py +0 -6
  69. StreamingCommunity/Lib/M3U8/decryptor.py +0 -164
  70. StreamingCommunity/Lib/M3U8/estimator.py +0 -229
  71. StreamingCommunity/Lib/M3U8/parser.py +0 -666
  72. StreamingCommunity/Lib/M3U8/url_fixer.py +0 -52
  73. StreamingCommunity/Lib/TMBD/__init__.py +0 -2
  74. StreamingCommunity/Lib/TMBD/obj_tmbd.py +0 -39
  75. StreamingCommunity/Lib/TMBD/tmdb.py +0 -346
  76. StreamingCommunity/Upload/update.py +0 -67
  77. StreamingCommunity/Upload/version.py +0 -5
  78. StreamingCommunity/Util/_jsonConfig.py +0 -204
  79. StreamingCommunity/Util/call_stack.py +0 -42
  80. StreamingCommunity/Util/color.py +0 -20
  81. StreamingCommunity/Util/console.py +0 -12
  82. StreamingCommunity/Util/ffmpeg_installer.py +0 -351
  83. StreamingCommunity/Util/headers.py +0 -147
  84. StreamingCommunity/Util/logger.py +0 -53
  85. StreamingCommunity/Util/message.py +0 -64
  86. StreamingCommunity/Util/os.py +0 -545
  87. StreamingCommunity/Util/table.py +0 -229
  88. StreamingCommunity-2.3.0.dist-info/RECORD +0 -92
  89. {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/LICENSE +0 -0
  90. {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/WHEEL +0 -0
  91. {StreamingCommunity-2.3.0.dist-info → StreamingCommunity-2.5.0.dist-info}/entry_points.txt +0 -0
  92. {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 dynamic argument parser
140
- parser = argparse.ArgumentParser(description='Script to download film and series from the internet.')
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 line arguments
185
+ # Parse command-line arguments
157
186
  args = parser.parse_args()
158
187
 
159
- # Mapping command-line arguments to functions
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.0
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: fake-useragent==1.1.3
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/github/languages/code-size/Lovi-0/StreamingCommunity?style=for-the-badge" alt="Code Size"/>
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
- ## Updating via PyPI
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
- | [Altadefinizione](https://altadefinizione.prof/) | ✅ |
502
+ | [AltadefinizioneGratis](https://altadefinizionegratis.site/) | ✅ |
433
503
  | [AnimeUnity](https://animeunity.so/) | ✅ |
434
504
  | [Ilcorsaronero](https://ilcorsaronero.link/) | ✅ |
435
- | [CB01New](https://cb01new.pics/) | ✅ |
505
+ | [CB01New](https://cb01new.video/) | ✅ |
436
506
  | [DDLStreamItaly](https://ddlstreamitaly.co/) | ✅ |
437
- | [GuardaSerie](https://guardaserie.academy/) | ✅ |
507
+ | [GuardaSerie](https://guardaserie.meme/) | ✅ |
438
508
  | [MostraGuarda](https://mostraguarda.stream/) | ✅ |
439
- | [StreamingCommunity](https://streamingcommunity.prof/) | ✅ |
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}")