quasarr 2.1.5__py3-none-any.whl → 2.3.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.
- quasarr/__init__.py +38 -29
- quasarr/api/__init__.py +94 -23
- quasarr/api/captcha/__init__.py +0 -12
- quasarr/api/config/__init__.py +22 -11
- quasarr/api/packages/__init__.py +26 -34
- quasarr/api/statistics/__init__.py +15 -15
- quasarr/downloads/__init__.py +9 -1
- quasarr/downloads/packages/__init__.py +2 -2
- quasarr/downloads/sources/al.py +6 -0
- quasarr/downloads/sources/by.py +29 -20
- quasarr/downloads/sources/dd.py +9 -1
- quasarr/downloads/sources/dl.py +3 -0
- quasarr/downloads/sources/dt.py +16 -7
- quasarr/downloads/sources/dw.py +22 -17
- quasarr/downloads/sources/he.py +11 -6
- quasarr/downloads/sources/mb.py +9 -3
- quasarr/downloads/sources/nk.py +9 -3
- quasarr/downloads/sources/nx.py +21 -17
- quasarr/downloads/sources/sf.py +21 -13
- quasarr/downloads/sources/sl.py +10 -2
- quasarr/downloads/sources/wd.py +18 -9
- quasarr/downloads/sources/wx.py +7 -11
- quasarr/providers/auth.py +1 -1
- quasarr/providers/cloudflare.py +1 -1
- quasarr/providers/hostname_issues.py +63 -0
- quasarr/providers/html_images.py +1 -18
- quasarr/providers/html_templates.py +104 -12
- quasarr/providers/imdb_metadata.py +288 -75
- quasarr/providers/obfuscated.py +11 -11
- quasarr/providers/sessions/al.py +27 -11
- quasarr/providers/sessions/dd.py +12 -4
- quasarr/providers/sessions/dl.py +19 -11
- quasarr/providers/sessions/nx.py +12 -4
- quasarr/providers/version.py +1 -1
- quasarr/search/__init__.py +5 -0
- quasarr/search/sources/al.py +12 -1
- quasarr/search/sources/by.py +15 -4
- quasarr/search/sources/dd.py +22 -3
- quasarr/search/sources/dj.py +12 -1
- quasarr/search/sources/dl.py +12 -6
- quasarr/search/sources/dt.py +17 -4
- quasarr/search/sources/dw.py +15 -4
- quasarr/search/sources/fx.py +19 -6
- quasarr/search/sources/he.py +22 -3
- quasarr/search/sources/mb.py +15 -4
- quasarr/search/sources/nk.py +19 -3
- quasarr/search/sources/nx.py +15 -4
- quasarr/search/sources/sf.py +25 -8
- quasarr/search/sources/sj.py +14 -1
- quasarr/search/sources/sl.py +17 -2
- quasarr/search/sources/wd.py +15 -4
- quasarr/search/sources/wx.py +16 -18
- quasarr/storage/setup.py +150 -35
- {quasarr-2.1.5.dist-info → quasarr-2.3.0.dist-info}/METADATA +6 -3
- quasarr-2.3.0.dist-info/RECORD +82 -0
- {quasarr-2.1.5.dist-info → quasarr-2.3.0.dist-info}/WHEEL +1 -1
- quasarr-2.1.5.dist-info/RECORD +0 -81
- {quasarr-2.1.5.dist-info → quasarr-2.3.0.dist-info}/entry_points.txt +0 -0
- {quasarr-2.1.5.dist-info → quasarr-2.3.0.dist-info}/licenses/LICENSE +0 -0
- {quasarr-2.1.5.dist-info → quasarr-2.3.0.dist-info}/top_level.txt +0 -0
quasarr/downloads/sources/wd.py
CHANGED
|
@@ -9,29 +9,34 @@ import requests
|
|
|
9
9
|
from bs4 import BeautifulSoup
|
|
10
10
|
|
|
11
11
|
from quasarr.providers.cloudflare import flaresolverr_get, is_cloudflare_challenge
|
|
12
|
+
from quasarr.providers.hostname_issues import mark_hostname_issue
|
|
12
13
|
from quasarr.providers.log import info, debug
|
|
13
14
|
from quasarr.providers.utils import is_flaresolverr_available
|
|
14
15
|
|
|
16
|
+
hostname = "wd"
|
|
17
|
+
|
|
15
18
|
|
|
16
19
|
def resolve_wd_redirect(url, user_agent):
|
|
17
20
|
"""
|
|
18
21
|
Follow redirects for a WD mirror URL and return the final destination.
|
|
19
22
|
"""
|
|
20
23
|
try:
|
|
21
|
-
|
|
24
|
+
r = requests.get(
|
|
22
25
|
url,
|
|
23
26
|
allow_redirects=True,
|
|
24
27
|
timeout=10,
|
|
25
28
|
headers={"User-Agent": user_agent},
|
|
26
29
|
)
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
30
|
+
r.raise_for_status()
|
|
31
|
+
if r.history:
|
|
32
|
+
for resp in r.history:
|
|
33
|
+
debug(f"Redirected from {resp.url} to {r.url}")
|
|
34
|
+
return r.url
|
|
31
35
|
else:
|
|
32
36
|
info(f"WD blocked attempt to resolve {url}. Your IP may be banned. Try again later.")
|
|
33
37
|
except Exception as e:
|
|
34
38
|
info(f"Error fetching redirected URL for {url}: {e}")
|
|
39
|
+
mark_hostname_issue(hostname, "download", str(e) if "e" in dir() else "Download error")
|
|
35
40
|
return None
|
|
36
41
|
|
|
37
42
|
|
|
@@ -46,17 +51,21 @@ def get_wd_download_links(shared_state, url, mirror, title, password):
|
|
|
46
51
|
user_agent = shared_state.values["user_agent"]
|
|
47
52
|
|
|
48
53
|
try:
|
|
49
|
-
|
|
50
|
-
if
|
|
54
|
+
r = requests.get(url)
|
|
55
|
+
if r.status_code >= 400 or is_cloudflare_challenge(r.text):
|
|
51
56
|
if is_flaresolverr_available(shared_state):
|
|
52
57
|
info("WD is protected by Cloudflare. Using FlareSolverr to bypass protection.")
|
|
53
|
-
|
|
58
|
+
r = flaresolverr_get(shared_state, url)
|
|
54
59
|
else:
|
|
55
60
|
info("WD is protected by Cloudflare but FlareSolverr is not configured. "
|
|
56
61
|
"Please configure FlareSolverr in the web UI to access this site.")
|
|
62
|
+
mark_hostname_issue(hostname, "download", "FlareSolverr required but missing.")
|
|
57
63
|
return {"links": [], "imdb_id": None}
|
|
58
64
|
|
|
59
|
-
|
|
65
|
+
if r.status_code >= 400:
|
|
66
|
+
mark_hostname_issue(hostname, "download", f"Download error: {str(r.status_code)}")
|
|
67
|
+
|
|
68
|
+
soup = BeautifulSoup(r.text, "html.parser")
|
|
60
69
|
|
|
61
70
|
# extract IMDb id if present
|
|
62
71
|
imdb_id = None
|
quasarr/downloads/sources/wx.py
CHANGED
|
@@ -6,6 +6,7 @@ import re
|
|
|
6
6
|
|
|
7
7
|
import requests
|
|
8
8
|
|
|
9
|
+
from quasarr.providers.hostname_issues import mark_hostname_issue
|
|
9
10
|
from quasarr.providers.log import info, debug
|
|
10
11
|
from quasarr.providers.utils import check_links_online_status
|
|
11
12
|
|
|
@@ -32,11 +33,8 @@ def get_wx_download_links(shared_state, url, mirror, title, password):
|
|
|
32
33
|
session = requests.Session()
|
|
33
34
|
|
|
34
35
|
# First, load the page to establish session cookies
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
if response.status_code != 200:
|
|
38
|
-
info(f"{hostname.upper()}: Failed to load page: {url} (Status: {response.status_code})")
|
|
39
|
-
return {"links": []}
|
|
36
|
+
r = session.get(url, headers=headers, timeout=30)
|
|
37
|
+
r.raise_for_status()
|
|
40
38
|
|
|
41
39
|
# Extract slug from URL
|
|
42
40
|
slug_match = re.search(r'/detail/([^/?]+)', url)
|
|
@@ -53,13 +51,10 @@ def get_wx_download_links(shared_state, url, mirror, title, password):
|
|
|
53
51
|
}
|
|
54
52
|
|
|
55
53
|
debug(f"{hostname.upper()}: Fetching API data from: {api_url}")
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
if api_response.status_code != 200:
|
|
59
|
-
info(f"{hostname.upper()}: Failed to load API: {api_url} (Status: {api_response.status_code})")
|
|
60
|
-
return {"links": []}
|
|
54
|
+
api_r = session.get(api_url, headers=api_headers, timeout=30)
|
|
55
|
+
api_r.raise_for_status()
|
|
61
56
|
|
|
62
|
-
data =
|
|
57
|
+
data = api_r.json()
|
|
63
58
|
|
|
64
59
|
# Navigate to releases in the API response
|
|
65
60
|
if 'item' not in data or 'releases' not in data['item']:
|
|
@@ -165,4 +160,5 @@ def get_wx_download_links(shared_state, url, mirror, title, password):
|
|
|
165
160
|
|
|
166
161
|
except Exception as e:
|
|
167
162
|
info(f"{hostname.upper()}: Error extracting download links from {url}: {e}")
|
|
163
|
+
mark_hostname_issue(hostname, "download", str(e) if "e" in dir() else "Download error")
|
|
168
164
|
return {"links": []}
|
quasarr/providers/auth.py
CHANGED
|
@@ -273,7 +273,7 @@ def add_auth_routes(app):
|
|
|
273
273
|
return _handle_logout()
|
|
274
274
|
|
|
275
275
|
|
|
276
|
-
def add_auth_hook(app, whitelist_prefixes=
|
|
276
|
+
def add_auth_hook(app, whitelist_prefixes=[], whitelist_suffixes=[]):
|
|
277
277
|
"""Add authentication hook to a Bottle app.
|
|
278
278
|
|
|
279
279
|
Args:
|
quasarr/providers/cloudflare.py
CHANGED
|
@@ -162,7 +162,7 @@ class FlareSolverrResponse:
|
|
|
162
162
|
|
|
163
163
|
def raise_for_status(self):
|
|
164
164
|
if 400 <= self.status_code:
|
|
165
|
-
raise requests.HTTPError(f"{self.status_code} Error
|
|
165
|
+
raise requests.HTTPError(f"{self.status_code} Error at {self.url}")
|
|
166
166
|
|
|
167
167
|
|
|
168
168
|
def flaresolverr_get(shared_state, url, timeout=60):
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
# -*- coding: utf-8 -*-
|
|
2
|
+
# Quasarr
|
|
3
|
+
# Project by https://github.com/rix1337
|
|
4
|
+
|
|
5
|
+
"""
|
|
6
|
+
Hostname Issues Tracker - Uses lazy imports to avoid circular dependency
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import json
|
|
10
|
+
from datetime import datetime
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def _get_db(table_name):
|
|
14
|
+
"""Lazy import to avoid circular dependency."""
|
|
15
|
+
from quasarr.storage.sqlite_database import DataBase
|
|
16
|
+
return DataBase(table_name)
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def mark_hostname_issue(shorthand, operation, error_message):
|
|
20
|
+
shorthand = shorthand.lower()
|
|
21
|
+
db = _get_db("hostname_issues")
|
|
22
|
+
|
|
23
|
+
issue_data = {
|
|
24
|
+
"operation": operation,
|
|
25
|
+
"error": str(error_message)[:500],
|
|
26
|
+
"timestamp": datetime.now().isoformat()
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
db.update_store(shorthand, json.dumps(issue_data))
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
def clear_hostname_issue(shorthand):
|
|
33
|
+
shorthand = shorthand.lower()
|
|
34
|
+
db = _get_db("hostname_issues")
|
|
35
|
+
db.delete(shorthand)
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
def get_hostname_issue(shorthand):
|
|
39
|
+
shorthand = shorthand.lower()
|
|
40
|
+
db = _get_db("hostname_issues")
|
|
41
|
+
data = db.retrieve(shorthand)
|
|
42
|
+
|
|
43
|
+
if data:
|
|
44
|
+
try:
|
|
45
|
+
return json.loads(data)
|
|
46
|
+
except json.JSONDecodeError:
|
|
47
|
+
return None
|
|
48
|
+
return None
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
def get_all_hostname_issues():
|
|
52
|
+
db = _get_db("hostname_issues")
|
|
53
|
+
all_data = db.retrieve_all_titles()
|
|
54
|
+
|
|
55
|
+
issues = {}
|
|
56
|
+
if all_data:
|
|
57
|
+
for shorthand, data in all_data:
|
|
58
|
+
try:
|
|
59
|
+
issues[shorthand] = json.loads(data)
|
|
60
|
+
except json.JSONDecodeError:
|
|
61
|
+
continue
|
|
62
|
+
|
|
63
|
+
return issues
|
quasarr/providers/html_images.py
CHANGED
|
@@ -2,21 +2,4 @@
|
|
|
2
2
|
# Quasarr
|
|
3
3
|
# Project by https://github.com/rix1337
|
|
4
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 = ''
|
|
5
|
+
logo = ''
|
|
@@ -195,18 +195,7 @@ def render_centered_html(inner_content, footer_content=""):
|
|
|
195
195
|
padding: calc(var(--spacing) * 2);
|
|
196
196
|
text-align: center;
|
|
197
197
|
width: 100%;
|
|
198
|
-
max-width:
|
|
199
|
-
}
|
|
200
|
-
/* No padding on the sides for captcha view on small screens */
|
|
201
|
-
@media (max-width: 600px) {
|
|
202
|
-
body:has(iframe) .outer {
|
|
203
|
-
padding-left: 0;
|
|
204
|
-
padding-right: 0;
|
|
205
|
-
}
|
|
206
|
-
body:has(iframe) .inner {
|
|
207
|
-
padding-left: 0;
|
|
208
|
-
padding-right: 0;
|
|
209
|
-
}
|
|
198
|
+
max-width: 600px;
|
|
210
199
|
}
|
|
211
200
|
h2 {
|
|
212
201
|
margin-top: var(--spacing);
|
|
@@ -235,6 +224,13 @@ def render_centered_html(inner_content, footer_content=""):
|
|
|
235
224
|
.captcha-container {
|
|
236
225
|
background-color: var(--secondary);
|
|
237
226
|
}
|
|
227
|
+
/* Responsive scaling for fixed-width CAPTCHA iframe (370px) */
|
|
228
|
+
/* PROBLEM: The drag and drop breaks */
|
|
229
|
+
@media (max-width: 400px) {
|
|
230
|
+
#puzzle-captcha {
|
|
231
|
+
zoom: 0.75;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
238
234
|
button {
|
|
239
235
|
padding: 0.5rem 1rem;
|
|
240
236
|
font-size: 1rem;
|
|
@@ -285,6 +281,54 @@ def render_centered_html(inner_content, footer_content=""):
|
|
|
285
281
|
footer a:hover {
|
|
286
282
|
color: var(--fg-color);
|
|
287
283
|
}
|
|
284
|
+
/* Global Modal Styles */
|
|
285
|
+
.status-modal-overlay {
|
|
286
|
+
position: fixed;
|
|
287
|
+
top: 0;
|
|
288
|
+
left: 0;
|
|
289
|
+
right: 0;
|
|
290
|
+
bottom: 0;
|
|
291
|
+
background: rgba(0, 0, 0, 0.5);
|
|
292
|
+
display: flex;
|
|
293
|
+
align-items: center;
|
|
294
|
+
justify-content: center;
|
|
295
|
+
z-index: 9999;
|
|
296
|
+
visibility: hidden;
|
|
297
|
+
opacity: 0;
|
|
298
|
+
transition: visibility 0s, opacity 0.2s;
|
|
299
|
+
}
|
|
300
|
+
.status-modal-overlay.active {
|
|
301
|
+
visibility: visible;
|
|
302
|
+
opacity: 1;
|
|
303
|
+
}
|
|
304
|
+
.status-modal {
|
|
305
|
+
background: var(--card-bg);
|
|
306
|
+
border-radius: 0.5rem;
|
|
307
|
+
padding: 1.5rem;
|
|
308
|
+
max-width: 400px;
|
|
309
|
+
width: 90%;
|
|
310
|
+
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15);
|
|
311
|
+
transform: scale(0.9);
|
|
312
|
+
transition: transform 0.2s;
|
|
313
|
+
}
|
|
314
|
+
.status-modal-overlay.active .status-modal {
|
|
315
|
+
transform: scale(1);
|
|
316
|
+
}
|
|
317
|
+
.status-modal h3 {
|
|
318
|
+
margin: 0 0 1rem 0;
|
|
319
|
+
display: flex;
|
|
320
|
+
align-items: center;
|
|
321
|
+
gap: 0.5rem;
|
|
322
|
+
}
|
|
323
|
+
.status-modal p {
|
|
324
|
+
margin: 0 0 1rem 0;
|
|
325
|
+
word-break: break-word;
|
|
326
|
+
}
|
|
327
|
+
.status-modal .btn-row {
|
|
328
|
+
display: flex;
|
|
329
|
+
gap: 0.5rem;
|
|
330
|
+
justify-content: flex-end;
|
|
331
|
+
}
|
|
288
332
|
</style>
|
|
289
333
|
</head>'''
|
|
290
334
|
|
|
@@ -295,6 +339,53 @@ def render_centered_html(inner_content, footer_content=""):
|
|
|
295
339
|
else:
|
|
296
340
|
footer_html = version_text
|
|
297
341
|
|
|
342
|
+
# Global modal script
|
|
343
|
+
modal_script = '''
|
|
344
|
+
<script>
|
|
345
|
+
function showModal(title, content, buttonsHtml) {
|
|
346
|
+
let overlay = document.getElementById('global-modal-overlay');
|
|
347
|
+
if (!overlay) {
|
|
348
|
+
overlay = document.createElement('div');
|
|
349
|
+
overlay.id = 'global-modal-overlay';
|
|
350
|
+
overlay.className = 'status-modal-overlay';
|
|
351
|
+
overlay.innerHTML = `
|
|
352
|
+
<div class="status-modal">
|
|
353
|
+
<h3 id="global-modal-title"></h3>
|
|
354
|
+
<div id="global-modal-content"></div>
|
|
355
|
+
<div class="btn-row" id="global-modal-buttons"></div>
|
|
356
|
+
</div>
|
|
357
|
+
`;
|
|
358
|
+
document.body.appendChild(overlay);
|
|
359
|
+
|
|
360
|
+
overlay.addEventListener('click', function(e) {
|
|
361
|
+
if (e.target === overlay) closeModal();
|
|
362
|
+
});
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
document.getElementById('global-modal-title').innerHTML = title;
|
|
366
|
+
document.getElementById('global-modal-content').innerHTML = content;
|
|
367
|
+
|
|
368
|
+
let btns = buttonsHtml || '<button class="btn-secondary" onclick="closeModal()">Close</button>';
|
|
369
|
+
document.getElementById('global-modal-buttons').innerHTML = btns;
|
|
370
|
+
|
|
371
|
+
// Small timeout to allow CSS transition
|
|
372
|
+
requestAnimationFrame(() => {
|
|
373
|
+
overlay.classList.add('active');
|
|
374
|
+
});
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
function closeModal() {
|
|
378
|
+
const overlay = document.getElementById('global-modal-overlay');
|
|
379
|
+
if (overlay) {
|
|
380
|
+
overlay.classList.remove('active');
|
|
381
|
+
setTimeout(() => {
|
|
382
|
+
if (overlay.parentNode) overlay.parentNode.removeChild(overlay);
|
|
383
|
+
}, 200);
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
</script>
|
|
387
|
+
'''
|
|
388
|
+
|
|
298
389
|
body = f'''
|
|
299
390
|
{head}
|
|
300
391
|
<body>
|
|
@@ -306,6 +397,7 @@ def render_centered_html(inner_content, footer_content=""):
|
|
|
306
397
|
<footer>
|
|
307
398
|
{footer_html}
|
|
308
399
|
</footer>
|
|
400
|
+
{modal_script}
|
|
309
401
|
</body>
|
|
310
402
|
'''
|
|
311
403
|
return f'<html>{body}</html>'
|