quasarr 0.1.6__py3-none-any.whl → 1.23.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 quasarr might be problematic. Click here for more details.

Files changed (77) hide show
  1. quasarr/__init__.py +316 -42
  2. quasarr/api/__init__.py +187 -0
  3. quasarr/api/arr/__init__.py +387 -0
  4. quasarr/api/captcha/__init__.py +1189 -0
  5. quasarr/api/config/__init__.py +23 -0
  6. quasarr/api/sponsors_helper/__init__.py +166 -0
  7. quasarr/api/statistics/__init__.py +196 -0
  8. quasarr/downloads/__init__.py +319 -256
  9. quasarr/downloads/linkcrypters/__init__.py +0 -0
  10. quasarr/downloads/linkcrypters/al.py +237 -0
  11. quasarr/downloads/linkcrypters/filecrypt.py +444 -0
  12. quasarr/downloads/linkcrypters/hide.py +123 -0
  13. quasarr/downloads/packages/__init__.py +476 -0
  14. quasarr/downloads/sources/al.py +697 -0
  15. quasarr/downloads/sources/by.py +106 -0
  16. quasarr/downloads/sources/dd.py +76 -0
  17. quasarr/downloads/sources/dj.py +7 -0
  18. quasarr/downloads/sources/dl.py +199 -0
  19. quasarr/downloads/sources/dt.py +66 -0
  20. quasarr/downloads/sources/dw.py +14 -7
  21. quasarr/downloads/sources/he.py +112 -0
  22. quasarr/downloads/sources/mb.py +47 -0
  23. quasarr/downloads/sources/nk.py +54 -0
  24. quasarr/downloads/sources/nx.py +42 -83
  25. quasarr/downloads/sources/sf.py +159 -0
  26. quasarr/downloads/sources/sj.py +7 -0
  27. quasarr/downloads/sources/sl.py +90 -0
  28. quasarr/downloads/sources/wd.py +110 -0
  29. quasarr/downloads/sources/wx.py +127 -0
  30. quasarr/providers/cloudflare.py +204 -0
  31. quasarr/providers/html_images.py +22 -0
  32. quasarr/providers/html_templates.py +211 -104
  33. quasarr/providers/imdb_metadata.py +108 -3
  34. quasarr/providers/log.py +19 -0
  35. quasarr/providers/myjd_api.py +201 -40
  36. quasarr/providers/notifications.py +99 -11
  37. quasarr/providers/obfuscated.py +65 -0
  38. quasarr/providers/sessions/__init__.py +0 -0
  39. quasarr/providers/sessions/al.py +286 -0
  40. quasarr/providers/sessions/dd.py +78 -0
  41. quasarr/providers/sessions/dl.py +175 -0
  42. quasarr/providers/sessions/nx.py +76 -0
  43. quasarr/providers/shared_state.py +656 -79
  44. quasarr/providers/statistics.py +154 -0
  45. quasarr/providers/version.py +60 -1
  46. quasarr/providers/web_server.py +1 -1
  47. quasarr/search/__init__.py +144 -15
  48. quasarr/search/sources/al.py +448 -0
  49. quasarr/search/sources/by.py +204 -0
  50. quasarr/search/sources/dd.py +135 -0
  51. quasarr/search/sources/dj.py +213 -0
  52. quasarr/search/sources/dl.py +354 -0
  53. quasarr/search/sources/dt.py +265 -0
  54. quasarr/search/sources/dw.py +94 -67
  55. quasarr/search/sources/fx.py +89 -33
  56. quasarr/search/sources/he.py +196 -0
  57. quasarr/search/sources/mb.py +195 -0
  58. quasarr/search/sources/nk.py +188 -0
  59. quasarr/search/sources/nx.py +75 -21
  60. quasarr/search/sources/sf.py +374 -0
  61. quasarr/search/sources/sj.py +213 -0
  62. quasarr/search/sources/sl.py +246 -0
  63. quasarr/search/sources/wd.py +208 -0
  64. quasarr/search/sources/wx.py +337 -0
  65. quasarr/storage/config.py +39 -10
  66. quasarr/storage/setup.py +269 -97
  67. quasarr/storage/sqlite_database.py +6 -1
  68. quasarr-1.23.0.dist-info/METADATA +306 -0
  69. quasarr-1.23.0.dist-info/RECORD +77 -0
  70. {quasarr-0.1.6.dist-info → quasarr-1.23.0.dist-info}/WHEEL +1 -1
  71. quasarr/arr/__init__.py +0 -423
  72. quasarr/captcha_solver/__init__.py +0 -284
  73. quasarr-0.1.6.dist-info/METADATA +0 -81
  74. quasarr-0.1.6.dist-info/RECORD +0 -31
  75. {quasarr-0.1.6.dist-info → quasarr-1.23.0.dist-info}/entry_points.txt +0 -0
  76. {quasarr-0.1.6.dist-info → quasarr-1.23.0.dist-info/licenses}/LICENSE +0 -0
  77. {quasarr-0.1.6.dist-info → quasarr-1.23.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,127 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ import re
6
+
7
+ import requests
8
+
9
+ from quasarr.providers.log import info, debug
10
+
11
+ hostname = "wx"
12
+
13
+
14
+ def get_wx_download_links(shared_state, url, mirror, title):
15
+ """
16
+ Get download links from API based on title and mirror.
17
+
18
+ Returns:
19
+ list of [url, hoster] pairs where hoster is the actual mirror (e.g., 'ddownload.com', 'rapidgator.net')
20
+ """
21
+ host = shared_state.values["config"]("Hostnames").get(hostname)
22
+
23
+ headers = {
24
+ 'User-Agent': shared_state.values["user_agent"],
25
+ 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8'
26
+ }
27
+
28
+ try:
29
+ session = requests.Session()
30
+
31
+ # First, load the page to establish session cookies
32
+ response = session.get(url, headers=headers, timeout=30)
33
+
34
+ if response.status_code != 200:
35
+ info(f"{hostname.upper()}: Failed to load page: {url} (Status: {response.status_code})")
36
+ return []
37
+
38
+ # Extract slug from URL
39
+ slug_match = re.search(r'/detail/([^/]+)', url)
40
+ if not slug_match:
41
+ info(f"{hostname.upper()}: Could not extract slug from URL: {url}")
42
+ return []
43
+
44
+ api_url = f'https://api.{host}/start/d/{slug_match.group(1)}'
45
+
46
+ # Update headers for API request
47
+ api_headers = {
48
+ 'User-Agent': shared_state.values["user_agent"],
49
+ 'Accept': 'application/json'
50
+ }
51
+
52
+ debug(f"{hostname.upper()}: Fetching API data from: {api_url}")
53
+ api_response = session.get(api_url, headers=api_headers, timeout=30)
54
+
55
+ if api_response.status_code != 200:
56
+ info(f"{hostname.upper()}: Failed to load API: {api_url} (Status: {api_response.status_code})")
57
+ return []
58
+
59
+ data = api_response.json()
60
+
61
+ # Navigate to releases in the API response
62
+ if 'item' not in data or 'releases' not in data['item']:
63
+ info(f"{hostname.upper()}: No releases found in API response")
64
+ return []
65
+
66
+ releases = data['item']['releases']
67
+
68
+ # Find the release matching the title
69
+ matching_release = None
70
+ for release in releases:
71
+ if release.get('fulltitle') == title:
72
+ matching_release = release
73
+ break
74
+
75
+ if not matching_release:
76
+ info(f"{hostname.upper()}: No release found matching title: {title}")
77
+ return []
78
+
79
+ # Extract crypted_links based on mirror
80
+ crypted_links = matching_release.get('crypted_links', {})
81
+
82
+ if not crypted_links:
83
+ info(f"{hostname.upper()}: No crypted_links found for: {title}")
84
+ return []
85
+
86
+ links = []
87
+
88
+ # If mirror is specified, find matching hoster (handle partial matches like 'ddownload' -> 'ddownload.com')
89
+ if mirror:
90
+ matched_hoster = None
91
+ for hoster in crypted_links.keys():
92
+ if mirror.lower() in hoster.lower() or hoster.lower() in mirror.lower():
93
+ matched_hoster = hoster
94
+ break
95
+
96
+ if matched_hoster:
97
+ link = crypted_links[matched_hoster]
98
+ # Prefer hide over filecrypt
99
+ if re.search(r'hide\.', link, re.IGNORECASE):
100
+ links.append([link, matched_hoster])
101
+ debug(f"{hostname.upper()}: Found hide link for mirror {matched_hoster}")
102
+ elif re.search(r'filecrypt\.', link, re.IGNORECASE):
103
+ links.append([link, matched_hoster])
104
+ debug(f"{hostname.upper()}: Found filecrypt link for mirror {matched_hoster}")
105
+ else:
106
+ info(
107
+ f"{hostname.upper()}: Mirror '{mirror}' not found in available hosters: {list(crypted_links.keys())}")
108
+ else:
109
+ # If no mirror specified, get all available crypted links (prefer hide over filecrypt)
110
+ for hoster, link in crypted_links.items():
111
+ if re.search(r'hide\.', link, re.IGNORECASE):
112
+ links.append([link, hoster])
113
+ debug(f"{hostname.upper()}: Found hide link for hoster {hoster}")
114
+ elif re.search(r'filecrypt\.', link, re.IGNORECASE):
115
+ links.append([link, hoster])
116
+ debug(f"{hostname.upper()}: Found filecrypt link for hoster {hoster}")
117
+
118
+ if not links:
119
+ info(f"{hostname.upper()}: No supported crypted links found for: {title}")
120
+ return []
121
+
122
+ debug(f"{hostname.upper()}: Found {len(links)} crypted link(s) for: {title}")
123
+ return links
124
+
125
+ except Exception as e:
126
+ info(f"{hostname.upper()}: Error extracting download links from {url}: {e}")
127
+ return []
@@ -0,0 +1,204 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ import requests
6
+ from bs4 import BeautifulSoup
7
+
8
+
9
+ def is_cloudflare_challenge(html: str) -> bool:
10
+ soup = BeautifulSoup(html, "html.parser")
11
+
12
+ # Check <title>
13
+ title = (soup.title.string or "").strip().lower() if soup.title else ""
14
+ if "just a moment" in title or "attention required" in title:
15
+ return True
16
+
17
+ # Check known Cloudflare elements
18
+ if soup.find(id="challenge-form"):
19
+ return True
20
+ if soup.find("div", {"class": "cf-browser-verification"}):
21
+ return True
22
+ if soup.find("div", {"id": "cf-challenge-running"}):
23
+ return True
24
+
25
+ # Check scripts referencing Cloudflare
26
+ for script in soup.find_all("script", src=True):
27
+ if "cdn-cgi/challenge-platform" in script["src"]:
28
+ return True
29
+
30
+ # Optional: look for Cloudflare comment or beacon
31
+ if "data-cf-beacon" in html or "<!-- cloudflare -->" in html.lower():
32
+ return True
33
+
34
+ return False
35
+
36
+
37
+ def update_session_via_flaresolverr(info,
38
+ shared_state,
39
+ sess,
40
+ target_url: str,
41
+ timeout: int = 60):
42
+ flaresolverr_url = shared_state.values["config"]('FlareSolverr').get('url')
43
+ if not flaresolverr_url:
44
+ info("Cannot proceed without FlareSolverr. Please set it up to try again!")
45
+ return False
46
+
47
+ fs_payload = {
48
+ "cmd": "request.get",
49
+ "url": target_url,
50
+ "maxTimeout": timeout * 1000,
51
+ }
52
+
53
+ # Send the JSON request to FlareSolverr
54
+ fs_headers = {"Content-Type": "application/json"}
55
+ try:
56
+ resp = requests.post(
57
+ flaresolverr_url,
58
+ headers=fs_headers,
59
+ json=fs_payload,
60
+ timeout=timeout + 10
61
+ )
62
+ resp.raise_for_status()
63
+ except requests.exceptions.RequestException as e:
64
+ info(f"Could not reach FlareSolverr: {e}")
65
+ return {
66
+ "status_code": None,
67
+ "headers": {},
68
+ "json": None,
69
+ "text": "",
70
+ "cookies": [],
71
+ "error": f"FlareSolverr request failed: {e}"
72
+ }
73
+ except Exception as e:
74
+ raise RuntimeError(f"Could not reach FlareSolverr: {e}")
75
+
76
+ fs_json = resp.json()
77
+ if fs_json.get("status") != "ok" or "solution" not in fs_json:
78
+ raise RuntimeError(f"FlareSolverr did not return a valid solution: {fs_json.get('message', '<no message>')}")
79
+
80
+ solution = fs_json["solution"]
81
+
82
+ # Replace our requests.Session cookies with whatever FlareSolverr solved
83
+ sess.cookies.clear()
84
+ for ck in solution.get("cookies", []):
85
+ sess.cookies.set(
86
+ ck.get("name"),
87
+ ck.get("value"),
88
+ domain=ck.get("domain"),
89
+ path=ck.get("path", "/")
90
+ )
91
+ return {"session": sess, "user_agent": solution.get("userAgent", None)}
92
+
93
+
94
+ def ensure_session_cf_bypassed(info, shared_state, session, url, headers):
95
+ """
96
+ Performs a GET and, if Cloudflare challenge or 403 is present, tries FlareSolverr.
97
+ Returns tuple: (session, headers, response) or (None, None, None) on failure.
98
+ """
99
+ try:
100
+ resp = session.get(url, headers=headers, timeout=30)
101
+ except requests.RequestException as e:
102
+ info(f"Initial GET failed: {e}")
103
+ return None, None, None
104
+
105
+ # If page is protected, try FlareSolverr
106
+ if resp.status_code == 403 or is_cloudflare_challenge(resp.text):
107
+ info("Encountered Cloudflare protection. Solving challenge with FlareSolverr...")
108
+ flaresolverr_result = update_session_via_flaresolverr(info, shared_state, session, url)
109
+ if not flaresolverr_result:
110
+ info("FlareSolverr did not return a result.")
111
+ return None, None, None
112
+
113
+ # update session and possibly user-agent
114
+ session = flaresolverr_result.get("session", session)
115
+ user_agent = flaresolverr_result.get("user_agent")
116
+ if user_agent and user_agent != shared_state.values.get("user_agent"):
117
+ info("Updating User-Agent from FlareSolverr solution: " + user_agent)
118
+ shared_state.update("user_agent", user_agent)
119
+ headers = {'User-Agent': shared_state.values["user_agent"]}
120
+
121
+ # re-fetch using the new session/headers
122
+ try:
123
+ resp = session.get(url, headers=headers, timeout=30)
124
+ except requests.RequestException as e:
125
+ info(f"GET after FlareSolverr failed: {e}")
126
+ return None, None, None
127
+
128
+ if resp.status_code == 403 or is_cloudflare_challenge(resp.text):
129
+ info("Could not bypass Cloudflare protection with FlareSolverr!")
130
+ return None, None, None
131
+
132
+ return session, headers, resp
133
+
134
+
135
+ class FlareSolverrResponse:
136
+ """
137
+ Minimal Response-like object so it behaves like requests.Response.
138
+ """
139
+
140
+ def __init__(self, url, status_code, headers, text):
141
+ self.url = url
142
+ self.status_code = status_code
143
+ self.headers = headers or {}
144
+ self.text = text or ""
145
+ self.content = self.text.encode("utf-8")
146
+
147
+ # Cloudflare cookies are irrelevant here, but keep attribute for compatibility
148
+ self.cookies = requests.cookies.RequestsCookieJar()
149
+
150
+ def raise_for_status(self):
151
+ if 400 <= self.status_code:
152
+ raise requests.HTTPError(f"{self.status_code} Error for URL: {self.url}")
153
+
154
+
155
+ def flaresolverr_get(shared_state, url, timeout=60):
156
+ """
157
+ Core function for performing a GET request via FlareSolverr only.
158
+ Used internally by FlareSolverrSession.get()
159
+ """
160
+ flaresolverr_url = shared_state.values["config"]('FlareSolverr').get('url')
161
+ if not flaresolverr_url:
162
+ raise RuntimeError("FlareSolverr URL not configured in shared_state.")
163
+
164
+ payload = {
165
+ "cmd": "request.get",
166
+ "url": url,
167
+ "maxTimeout": timeout * 1000
168
+ }
169
+
170
+ try:
171
+ resp = requests.post(
172
+ flaresolverr_url,
173
+ json=payload,
174
+ headers={"Content-Type": "application/json"},
175
+ timeout=timeout + 10
176
+ )
177
+ resp.raise_for_status()
178
+ except Exception as e:
179
+ raise RuntimeError(f"Error communicating with FlareSolverr: {e}")
180
+
181
+ data = resp.json()
182
+
183
+ if data.get("status") != "ok":
184
+ raise RuntimeError(f"FlareSolverr returned error: {data.get('message')}")
185
+
186
+ solution = data.get("solution", {})
187
+ html = solution.get("response", "")
188
+ status_code = solution.get("status", 200)
189
+ url = solution.get("url", url)
190
+
191
+ # headers → convert list-of-keyvals to dict
192
+ fs_headers = {h["name"]: h["value"] for h in solution.get("headers", [])}
193
+
194
+ # Update global UA if provided
195
+ user_agent = solution.get("userAgent")
196
+ if user_agent and user_agent != shared_state.values.get("user_agent"):
197
+ shared_state.update("user_agent", user_agent)
198
+
199
+ return FlareSolverrResponse(
200
+ url=url,
201
+ status_code=status_code,
202
+ headers=fs_headers,
203
+ text=html
204
+ )
@@ -0,0 +1,22 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ logo = ''
6
+ al = ''
7
+ by = ''
8
+ dd = ''
9
+ dj = ''
10
+ dl = ''
11
+ dt = ''
12
+ dw = ''
13
+ fx = ''
14
+ nk = ''
15
+ he = ''
16
+ mb = ''
17
+ nx = ''
18
+ sf = ''
19
+ sj = ''
20
+ sl = ''
21
+ wd = ''
22
+ wx = ''