StreamingCommunity 2.4.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 (91) hide show
  1. {StreamingCommunity-2.4.0.dist-info → StreamingCommunity-2.5.0.dist-info}/METADATA +9 -6
  2. StreamingCommunity-2.5.0.dist-info/RECORD +8 -0
  3. StreamingCommunity/Api/Player/Helper/Vixcloud/js_parser.py +0 -143
  4. StreamingCommunity/Api/Player/Helper/Vixcloud/util.py +0 -136
  5. StreamingCommunity/Api/Player/ddl.py +0 -89
  6. StreamingCommunity/Api/Player/maxstream.py +0 -151
  7. StreamingCommunity/Api/Player/supervideo.py +0 -194
  8. StreamingCommunity/Api/Player/vixcloud.py +0 -273
  9. StreamingCommunity/Api/Site/1337xx/__init__.py +0 -51
  10. StreamingCommunity/Api/Site/1337xx/costant.py +0 -15
  11. StreamingCommunity/Api/Site/1337xx/site.py +0 -89
  12. StreamingCommunity/Api/Site/1337xx/title.py +0 -66
  13. StreamingCommunity/Api/Site/altadefinizionegratis/__init__.py +0 -51
  14. StreamingCommunity/Api/Site/altadefinizionegratis/costant.py +0 -19
  15. StreamingCommunity/Api/Site/altadefinizionegratis/film.py +0 -74
  16. StreamingCommunity/Api/Site/altadefinizionegratis/site.py +0 -95
  17. StreamingCommunity/Api/Site/animeunity/__init__.py +0 -51
  18. StreamingCommunity/Api/Site/animeunity/costant.py +0 -19
  19. StreamingCommunity/Api/Site/animeunity/film_serie.py +0 -135
  20. StreamingCommunity/Api/Site/animeunity/site.py +0 -175
  21. StreamingCommunity/Api/Site/animeunity/util/ScrapeSerie.py +0 -97
  22. StreamingCommunity/Api/Site/cb01new/__init__.py +0 -52
  23. StreamingCommunity/Api/Site/cb01new/costant.py +0 -19
  24. StreamingCommunity/Api/Site/cb01new/film.py +0 -73
  25. StreamingCommunity/Api/Site/cb01new/site.py +0 -83
  26. StreamingCommunity/Api/Site/ddlstreamitaly/__init__.py +0 -56
  27. StreamingCommunity/Api/Site/ddlstreamitaly/costant.py +0 -20
  28. StreamingCommunity/Api/Site/ddlstreamitaly/series.py +0 -146
  29. StreamingCommunity/Api/Site/ddlstreamitaly/site.py +0 -99
  30. StreamingCommunity/Api/Site/ddlstreamitaly/util/ScrapeSerie.py +0 -85
  31. StreamingCommunity/Api/Site/guardaserie/__init__.py +0 -51
  32. StreamingCommunity/Api/Site/guardaserie/costant.py +0 -19
  33. StreamingCommunity/Api/Site/guardaserie/series.py +0 -198
  34. StreamingCommunity/Api/Site/guardaserie/site.py +0 -90
  35. StreamingCommunity/Api/Site/guardaserie/util/ScrapeSerie.py +0 -110
  36. StreamingCommunity/Api/Site/ilcorsaronero/__init__.py +0 -52
  37. StreamingCommunity/Api/Site/ilcorsaronero/costant.py +0 -19
  38. StreamingCommunity/Api/Site/ilcorsaronero/site.py +0 -72
  39. StreamingCommunity/Api/Site/ilcorsaronero/title.py +0 -46
  40. StreamingCommunity/Api/Site/ilcorsaronero/util/ilCorsarScraper.py +0 -149
  41. StreamingCommunity/Api/Site/mostraguarda/__init__.py +0 -49
  42. StreamingCommunity/Api/Site/mostraguarda/costant.py +0 -19
  43. StreamingCommunity/Api/Site/mostraguarda/film.py +0 -101
  44. StreamingCommunity/Api/Site/streamingcommunity/__init__.py +0 -56
  45. StreamingCommunity/Api/Site/streamingcommunity/costant.py +0 -19
  46. StreamingCommunity/Api/Site/streamingcommunity/film.py +0 -75
  47. StreamingCommunity/Api/Site/streamingcommunity/series.py +0 -206
  48. StreamingCommunity/Api/Site/streamingcommunity/site.py +0 -142
  49. StreamingCommunity/Api/Site/streamingcommunity/util/ScrapeSerie.py +0 -123
  50. StreamingCommunity/Api/Template/Class/SearchType.py +0 -101
  51. StreamingCommunity/Api/Template/Util/__init__.py +0 -5
  52. StreamingCommunity/Api/Template/Util/get_domain.py +0 -203
  53. StreamingCommunity/Api/Template/Util/manage_ep.py +0 -179
  54. StreamingCommunity/Api/Template/Util/recall_search.py +0 -37
  55. StreamingCommunity/Api/Template/__init__.py +0 -3
  56. StreamingCommunity/Api/Template/site.py +0 -87
  57. StreamingCommunity/Lib/Downloader/HLS/downloader.py +0 -965
  58. StreamingCommunity/Lib/Downloader/HLS/proxyes.py +0 -110
  59. StreamingCommunity/Lib/Downloader/HLS/segments.py +0 -573
  60. StreamingCommunity/Lib/Downloader/MP4/downloader.py +0 -155
  61. StreamingCommunity/Lib/Downloader/TOR/downloader.py +0 -296
  62. StreamingCommunity/Lib/Downloader/__init__.py +0 -5
  63. StreamingCommunity/Lib/FFmpeg/__init__.py +0 -4
  64. StreamingCommunity/Lib/FFmpeg/capture.py +0 -170
  65. StreamingCommunity/Lib/FFmpeg/command.py +0 -296
  66. StreamingCommunity/Lib/FFmpeg/util.py +0 -249
  67. StreamingCommunity/Lib/M3U8/__init__.py +0 -6
  68. StreamingCommunity/Lib/M3U8/decryptor.py +0 -165
  69. StreamingCommunity/Lib/M3U8/estimator.py +0 -229
  70. StreamingCommunity/Lib/M3U8/parser.py +0 -666
  71. StreamingCommunity/Lib/M3U8/url_fixer.py +0 -58
  72. StreamingCommunity/Lib/TMBD/__init__.py +0 -2
  73. StreamingCommunity/Lib/TMBD/obj_tmbd.py +0 -39
  74. StreamingCommunity/Lib/TMBD/tmdb.py +0 -346
  75. StreamingCommunity/Upload/update.py +0 -67
  76. StreamingCommunity/Upload/version.py +0 -5
  77. StreamingCommunity/Util/_jsonConfig.py +0 -228
  78. StreamingCommunity/Util/call_stack.py +0 -42
  79. StreamingCommunity/Util/color.py +0 -20
  80. StreamingCommunity/Util/console.py +0 -12
  81. StreamingCommunity/Util/ffmpeg_installer.py +0 -368
  82. StreamingCommunity/Util/headers.py +0 -160
  83. StreamingCommunity/Util/logger.py +0 -62
  84. StreamingCommunity/Util/message.py +0 -64
  85. StreamingCommunity/Util/os.py +0 -507
  86. StreamingCommunity/Util/table.py +0 -229
  87. StreamingCommunity-2.4.0.dist-info/RECORD +0 -92
  88. {StreamingCommunity-2.4.0.dist-info → StreamingCommunity-2.5.0.dist-info}/LICENSE +0 -0
  89. {StreamingCommunity-2.4.0.dist-info → StreamingCommunity-2.5.0.dist-info}/WHEEL +0 -0
  90. {StreamingCommunity-2.4.0.dist-info → StreamingCommunity-2.5.0.dist-info}/entry_points.txt +0 -0
  91. {StreamingCommunity-2.4.0.dist-info → StreamingCommunity-2.5.0.dist-info}/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: StreamingCommunity
3
- Version: 2.4.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: googlesearch-python
24
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"/>
@@ -50,7 +50,10 @@ Requires-Dist: googlesearch-python
50
50
  <img src="https://img.shields.io/badge/License-GPL_3.0-blue.svg?style=for-the-badge" alt="License"/>
51
51
  </a>
52
52
  <a href="https://pypi.org/project/streamingcommunity">
53
- <img src="https://img.shields.io/pypi/dw/streamingcommunity?style=for-the-badge" alt="PyPI Downloads"/>
53
+ <img src="https://img.shields.io/pypi/dm/streamingcommunity?style=for-the-badge" alt="PyPI Downloads"/>
54
+ </a>
55
+ <a href="https://github.com/Lovi-0/StreamingCommunity">
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"/>
54
57
  </a>
55
58
  </p>
56
59
 
@@ -496,14 +499,14 @@ The `run-container` command mounts also the `config.json` file, so any change to
496
499
  | Website | Status |
497
500
  |:-------------------|:------:|
498
501
  | [1337xx](https://1337xx.to/) | ✅ |
499
- | [AltadefinizioneGratis](https://altadefinizionegratis.info/) | ✅ |
502
+ | [AltadefinizioneGratis](https://altadefinizionegratis.site/) | ✅ |
500
503
  | [AnimeUnity](https://animeunity.so/) | ✅ |
501
504
  | [Ilcorsaronero](https://ilcorsaronero.link/) | ✅ |
502
505
  | [CB01New](https://cb01new.video/) | ✅ |
503
506
  | [DDLStreamItaly](https://ddlstreamitaly.co/) | ✅ |
504
- | [GuardaSerie](https://guardaserie.academy/) | ✅ |
507
+ | [GuardaSerie](https://guardaserie.meme/) | ✅ |
505
508
  | [MostraGuarda](https://mostraguarda.stream/) | ✅ |
506
- | [StreamingCommunity](https://streamingcommunity.ooo/) | ✅ |
509
+ | [StreamingCommunity](https://streamingcommunity.paris/) | ✅ |
507
510
 
508
511
 
509
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}")
@@ -1,151 +0,0 @@
1
- # 05.07.24
2
-
3
- import re
4
- import logging
5
-
6
-
7
- # External libraries
8
- import httpx
9
- import jsbeautifier
10
- from bs4 import BeautifulSoup
11
-
12
-
13
- # Internal utilities
14
- from StreamingCommunity.Util._jsonConfig import config_manager
15
- from StreamingCommunity.Util.headers import get_headers
16
-
17
-
18
- # Variable
19
- max_timeout = config_manager.get_int("REQUESTS", "timeout")
20
-
21
-
22
- class VideoSource:
23
- def __init__(self, url: str):
24
- """
25
- Sets up the video source with the provided URL.
26
-
27
- Parameters:
28
- - url (str): The URL of the video.
29
- """
30
- self.url = url
31
- self.redirect_url = None
32
- self.maxstream_url = None
33
- self.m3u8_url = None
34
- self.headers = {'user-agent': get_headers()}
35
-
36
- def get_redirect_url(self):
37
- """
38
- Sends a request to the initial URL and extracts the redirect URL.
39
- """
40
- try:
41
-
42
- # Send a GET request to the initial URL
43
- response = httpx.get(self.url, headers=self.headers, follow_redirects=True, timeout=max_timeout)
44
- response.raise_for_status()
45
-
46
- # Extract the redirect URL from the HTML
47
- soup = BeautifulSoup(response.text, "html.parser")
48
- self.redirect_url = soup.find("div", id="iframen1").get("data-src")
49
- logging.info(f"Redirect URL: {self.redirect_url}")
50
-
51
- return self.redirect_url
52
-
53
- except httpx.RequestError as e:
54
- logging.error(f"Error during the initial request: {e}")
55
- raise
56
-
57
- except AttributeError as e:
58
- logging.error(f"Error parsing HTML: {e}")
59
- raise
60
-
61
- def get_maxstream_url(self):
62
- """
63
- Sends a request to the redirect URL and extracts the Maxstream URL.
64
- """
65
- try:
66
-
67
- # Send a GET request to the redirect URL
68
- response = httpx.get(self.redirect_url, headers=self.headers, follow_redirects=True, timeout=max_timeout)
69
- response.raise_for_status()
70
-
71
- # Extract the Maxstream URL from the HTML
72
- soup = BeautifulSoup(response.text, "html.parser")
73
- maxstream_url = soup.find("a")
74
-
75
- if maxstream_url is None:
76
-
77
- # If no anchor tag is found, try the alternative method
78
- logging.warning("Anchor tag not found. Trying the alternative method.")
79
- headers = {
80
- 'origin': 'https://stayonline.pro',
81
- 'user-agent': get_headers(),
82
- 'x-requested-with': 'XMLHttpRequest',
83
- }
84
-
85
- # Make request to stayonline api
86
- data = {'id': self.redirect_url.split("/")[-2], 'ref': ''}
87
- response = httpx.post('https://stayonline.pro/ajax/linkEmbedView.php', headers=headers, data=data)
88
- response.raise_for_status()
89
- uprot_url = response.json()['data']['value']
90
-
91
- # Retry getting maxtstream url
92
- response = httpx.get(uprot_url, headers=self.headers, follow_redirects=True, timeout=max_timeout)
93
- response.raise_for_status()
94
- soup = BeautifulSoup(response.text, "html.parser")
95
- maxstream_url = soup.find("a").get("href")
96
-
97
- else:
98
- maxstream_url = maxstream_url.get("href")
99
-
100
- self.maxstream_url = maxstream_url
101
- logging.info(f"Maxstream URL: {self.maxstream_url}")
102
-
103
- return self.maxstream_url
104
-
105
- except httpx.RequestError as e:
106
- logging.error(f"Error during the request to the redirect URL: {e}")
107
- raise
108
-
109
- except AttributeError as e:
110
- logging.error(f"Error parsing HTML: {e}")
111
- raise
112
-
113
- def get_m3u8_url(self):
114
- """
115
- Sends a request to the Maxstream URL and extracts the .m3u8 file URL.
116
- """
117
- try:
118
-
119
- # Send a GET request to the Maxstream URL
120
- response = httpx.get(self.maxstream_url, headers=self.headers, follow_redirects=True, timeout=max_timeout)
121
- response.raise_for_status()
122
- soup = BeautifulSoup(response.text, "html.parser")
123
-
124
- # Iterate over all script tags in the HTML
125
- for script in soup.find_all("script"):
126
- if "eval(function(p,a,c,k,e,d)" in script.text:
127
-
128
- # Execute the script using
129
- data_js = jsbeautifier.beautify(script.text)
130
-
131
- # Extract the .m3u8 URL from the script's output
132
- match = re.search(r'sources:\s*\[\{\s*src:\s*"([^"]+)"', data_js)
133
-
134
- if match:
135
- self.m3u8_url = match.group(1)
136
- logging.info(f"M3U8 URL: {self.m3u8_url}")
137
- break
138
-
139
- return self.m3u8_url
140
-
141
- except Exception as e:
142
- logging.error(f"Error executing the Node.js script: {e}")
143
- raise
144
-
145
- def get_playlist(self):
146
- """
147
- Executes the entire flow to obtain the final .m3u8 file URL.
148
- """
149
- self.get_redirect_url()
150
- self.get_maxstream_url()
151
- return self.get_m3u8_url()