quasarr 1.25.0__tar.gz → 1.26.1__tar.gz
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-1.25.0 → quasarr-1.26.1}/PKG-INFO +1 -1
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/captcha/__init__.py +73 -22
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/__init__.py +2 -1
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/linkcrypters/hide.py +22 -8
- quasarr-1.26.1/quasarr/providers/html_images.py +22 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/html_templates.py +68 -3
- quasarr-1.26.1/quasarr/providers/obfuscated.py +119 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/version.py +1 -1
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/PKG-INFO +1 -1
- quasarr-1.25.0/quasarr/providers/html_images.py +0 -22
- quasarr-1.25.0/quasarr/providers/obfuscated.py +0 -86
- {quasarr-1.25.0 → quasarr-1.26.1}/LICENSE +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/README.md +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/arr/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/config/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/sponsors_helper/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/api/statistics/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/linkcrypters/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/linkcrypters/al.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/linkcrypters/filecrypt.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/packages/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/al.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/by.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/dd.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/dj.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/dl.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/dt.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/dw.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/he.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/mb.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/nk.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/nx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/sf.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/sj.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/sl.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/wd.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/downloads/sources/wx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/cloudflare.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/imdb_metadata.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/log.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/myjd_api.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/notifications.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/sessions/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/sessions/al.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/sessions/dd.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/sessions/dl.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/sessions/nx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/shared_state.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/statistics.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/providers/web_server.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/al.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/by.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/dd.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/dj.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/dl.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/dt.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/dw.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/fx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/he.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/mb.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/nk.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/nx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/sf.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/sj.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/sl.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/wd.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/search/sources/wx.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/storage/__init__.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/storage/config.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/storage/setup.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr/storage/sqlite_database.py +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/SOURCES.txt +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/dependency_links.txt +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/entry_points.txt +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/not-zip-safe +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/requires.txt +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/quasarr.egg-info/top_level.txt +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/setup.cfg +0 -0
- {quasarr-1.25.0 → quasarr-1.26.1}/setup.py +0 -0
|
@@ -108,6 +108,12 @@ def setup_captcha_routes(app):
|
|
|
108
108
|
|
|
109
109
|
has_junkies_links = any(is_junkies_link(link) for link in prioritized_links)
|
|
110
110
|
|
|
111
|
+
# Hide uses nested arrays like FileCrypt: [["url", "mirror"]]
|
|
112
|
+
has_hide_links = any(
|
|
113
|
+
("hide." in link[0] if isinstance(link, (list, tuple)) else "hide." in link)
|
|
114
|
+
for link in prioritized_links
|
|
115
|
+
)
|
|
116
|
+
|
|
111
117
|
# KeepLinks uses nested arrays like FileCrypt: [["url", "mirror"]]
|
|
112
118
|
has_keeplinks_links = any(
|
|
113
119
|
("keeplinks." in link[0] if isinstance(link, (list, tuple)) else "keeplinks." in link)
|
|
@@ -120,7 +126,10 @@ def setup_captcha_routes(app):
|
|
|
120
126
|
for link in prioritized_links
|
|
121
127
|
)
|
|
122
128
|
|
|
123
|
-
if
|
|
129
|
+
if has_hide_links:
|
|
130
|
+
debug("Redirecting to Hide page")
|
|
131
|
+
redirect(f"/captcha/hide?data={quote(encoded_payload)}")
|
|
132
|
+
elif has_junkies_links:
|
|
124
133
|
debug("Redirecting to Junkies CAPTCHA")
|
|
125
134
|
redirect(f"/captcha/junkies?data={quote(encoded_payload)}")
|
|
126
135
|
elif has_keeplinks_links:
|
|
@@ -151,7 +160,7 @@ def setup_captcha_routes(app):
|
|
|
151
160
|
return {"error": f"Failed to decode payload: {str(e)}"}
|
|
152
161
|
|
|
153
162
|
def render_userscript_section(url, package_id, title, password, provider_type="junkies"):
|
|
154
|
-
"""Render the userscript UI section for Junkies, KeepLinks, or
|
|
163
|
+
"""Render the userscript UI section for Junkies, KeepLinks, ToLink, or Hide pages
|
|
155
164
|
|
|
156
165
|
This is the MAIN solution for these providers (not a bypass/fallback).
|
|
157
166
|
|
|
@@ -160,10 +169,10 @@ def setup_captcha_routes(app):
|
|
|
160
169
|
package_id: Package identifier
|
|
161
170
|
title: Package title
|
|
162
171
|
password: Package password
|
|
163
|
-
provider_type: Either "junkies", "keeplinks", or "tolink"
|
|
172
|
+
provider_type: Either "hide", "junkies", "keeplinks", or "tolink"
|
|
164
173
|
"""
|
|
165
174
|
|
|
166
|
-
provider_names = {"junkies": "Junkies", "keeplinks": "KeepLinks", "tolink": "ToLink"}
|
|
175
|
+
provider_names = {"hide": "Hide", "junkies": "Junkies", "keeplinks": "KeepLinks", "tolink": "ToLink"}
|
|
167
176
|
provider_name = provider_names.get(provider_type, "Provider")
|
|
168
177
|
userscript_url = f"/captcha/{provider_type}.user.js"
|
|
169
178
|
storage_key = f"hide{provider_name}SetupInstructions"
|
|
@@ -183,8 +192,8 @@ def setup_captcha_routes(app):
|
|
|
183
192
|
return f'''
|
|
184
193
|
<div>
|
|
185
194
|
<!-- Info section explaining the process -->
|
|
186
|
-
<div
|
|
187
|
-
<h3
|
|
195
|
+
<div class="info-box">
|
|
196
|
+
<h3>ℹ️ How This Works:</h3>
|
|
188
197
|
<p style="margin-bottom: 8px;">
|
|
189
198
|
1. Click the link below to open {provider_name}
|
|
190
199
|
</p>
|
|
@@ -197,8 +206,8 @@ def setup_captcha_routes(app):
|
|
|
197
206
|
</div>
|
|
198
207
|
|
|
199
208
|
<!-- One-time setup section - visually separated -->
|
|
200
|
-
<div id="setup-instructions"
|
|
201
|
-
<h3
|
|
209
|
+
<div id="setup-instructions" class="setup-box">
|
|
210
|
+
<h3>📦 First Time Setup:</h3>
|
|
202
211
|
<p style="margin-bottom: 8px;">
|
|
203
212
|
<a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
|
|
204
213
|
</p>
|
|
@@ -206,7 +215,7 @@ def setup_captcha_routes(app):
|
|
|
206
215
|
<a href="{userscript_url}" target="_blank">2. Install the {provider_name} userscript</a>
|
|
207
216
|
</p>
|
|
208
217
|
<p style="margin-top: 0;">
|
|
209
|
-
<button id="hide-setup-btn" type="button"
|
|
218
|
+
<button id="hide-setup-btn" type="button" class="btn-subtle">
|
|
210
219
|
✅ Don't show this again
|
|
211
220
|
</button>
|
|
212
221
|
</p>
|
|
@@ -214,7 +223,7 @@ def setup_captcha_routes(app):
|
|
|
214
223
|
|
|
215
224
|
<!-- Hidden "show instructions" button -->
|
|
216
225
|
<div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
|
|
217
|
-
<button id="show-setup-btn" type="button"
|
|
226
|
+
<button id="show-setup-btn" type="button" class="btn-subtle">
|
|
218
227
|
ℹ️ Show setup instructions
|
|
219
228
|
</button>
|
|
220
229
|
</div>
|
|
@@ -225,11 +234,11 @@ def setup_captcha_routes(app):
|
|
|
225
234
|
</p>
|
|
226
235
|
|
|
227
236
|
<!-- Manual submission - collapsible -->
|
|
228
|
-
<div
|
|
237
|
+
<div class="section-divider">
|
|
229
238
|
<details id="manualSubmitDetails">
|
|
230
|
-
<summary id="manualSubmitSummary" style="cursor: pointer;
|
|
239
|
+
<summary id="manualSubmitSummary" style="cursor: pointer;">Show Manual Submission</summary>
|
|
231
240
|
<div style="margin-top: 16px;">
|
|
232
|
-
<p style="
|
|
241
|
+
<p style="font-size: 0.9em;">
|
|
233
242
|
If the userscript doesn't work, you can manually paste the links below:
|
|
234
243
|
</p>
|
|
235
244
|
<form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
|
|
@@ -291,6 +300,42 @@ def setup_captcha_routes(app):
|
|
|
291
300
|
</script>
|
|
292
301
|
'''
|
|
293
302
|
|
|
303
|
+
@app.get("/captcha/hide")
|
|
304
|
+
def serve_hide_captcha():
|
|
305
|
+
payload = decode_payload()
|
|
306
|
+
|
|
307
|
+
if "error" in payload:
|
|
308
|
+
return render_centered_html(f'''<h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
|
|
309
|
+
<p>{payload["error"]}</p>
|
|
310
|
+
<p>
|
|
311
|
+
{render_button("Back", "secondary", {"onclick": "location.href='/'"})}
|
|
312
|
+
</p>''')
|
|
313
|
+
|
|
314
|
+
package_id = payload.get("package_id")
|
|
315
|
+
title = payload.get("title")
|
|
316
|
+
password = payload.get("password")
|
|
317
|
+
urls = payload.get("links")
|
|
318
|
+
url = urls[0][0] if isinstance(urls[0], (list, tuple)) else urls[0]
|
|
319
|
+
|
|
320
|
+
check_package_exists(package_id)
|
|
321
|
+
|
|
322
|
+
return render_centered_html(f"""
|
|
323
|
+
<!DOCTYPE html>
|
|
324
|
+
<html>
|
|
325
|
+
<body>
|
|
326
|
+
<h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
|
|
327
|
+
<p><b>Package:</b> {title}</p>
|
|
328
|
+
{render_userscript_section(url, package_id, title, password, "hide")}
|
|
329
|
+
<p>
|
|
330
|
+
{render_button("Delete Package", "secondary", {"onclick": f"location.href='/captcha/delete/{package_id}'"})}
|
|
331
|
+
</p>
|
|
332
|
+
<p>
|
|
333
|
+
{render_button("Back", "secondary", {"onclick": "location.href='/'"})}
|
|
334
|
+
</p>
|
|
335
|
+
|
|
336
|
+
</body>
|
|
337
|
+
</html>""")
|
|
338
|
+
|
|
294
339
|
@app.get("/captcha/junkies")
|
|
295
340
|
def serve_junkies_captcha():
|
|
296
341
|
payload = decode_payload()
|
|
@@ -407,6 +452,12 @@ def setup_captcha_routes(app):
|
|
|
407
452
|
response.content_type = 'application/javascript'
|
|
408
453
|
return content
|
|
409
454
|
|
|
455
|
+
@app.get('/captcha/hide.user.js')
|
|
456
|
+
def serve_hide_user_js():
|
|
457
|
+
content = obfuscated.hide_user_js()
|
|
458
|
+
response.content_type = 'application/javascript'
|
|
459
|
+
return content
|
|
460
|
+
|
|
410
461
|
@app.get('/captcha/junkies.user.js')
|
|
411
462
|
def serve_junkies_user_js():
|
|
412
463
|
sj = shared_state.values["config"]("Hostnames").get("sj")
|
|
@@ -445,13 +496,13 @@ def setup_captcha_routes(app):
|
|
|
445
496
|
)
|
|
446
497
|
|
|
447
498
|
return f'''
|
|
448
|
-
<div
|
|
499
|
+
<div class="section-divider" style="max-width: 370px; margin-left: auto; margin-right: auto;">
|
|
449
500
|
<details id="bypassDetails">
|
|
450
501
|
<summary id="bypassSummary">Show CAPTCHA Bypass</summary><br>
|
|
451
502
|
|
|
452
503
|
<!-- Info section explaining the process -->
|
|
453
|
-
<div
|
|
454
|
-
<h3
|
|
504
|
+
<div class="info-box">
|
|
505
|
+
<h3>ℹ️ How This Works:</h3>
|
|
455
506
|
<p style="margin-bottom: 8px;">
|
|
456
507
|
1. Click the button below to open FileCrypt directly
|
|
457
508
|
</p>
|
|
@@ -464,8 +515,8 @@ def setup_captcha_routes(app):
|
|
|
464
515
|
</div>
|
|
465
516
|
|
|
466
517
|
<!-- One-time setup section - visually separated -->
|
|
467
|
-
<div id="setup-instructions"
|
|
468
|
-
<h3
|
|
518
|
+
<div id="setup-instructions" class="setup-box">
|
|
519
|
+
<h3>📦 First Time Setup:</h3>
|
|
469
520
|
<p style="margin-bottom: 8px;">
|
|
470
521
|
<a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
|
|
471
522
|
</p>
|
|
@@ -473,7 +524,7 @@ def setup_captcha_routes(app):
|
|
|
473
524
|
<a href="/captcha/filecrypt.user.js" target="_blank">2. Install the FileCrypt userscript</a>
|
|
474
525
|
</p>
|
|
475
526
|
<p style="margin-top: 0;">
|
|
476
|
-
<button id="hide-setup-btn" type="button"
|
|
527
|
+
<button id="hide-setup-btn" type="button" class="btn-subtle">
|
|
477
528
|
✅ Don't show this again
|
|
478
529
|
</button>
|
|
479
530
|
</p>
|
|
@@ -481,7 +532,7 @@ def setup_captcha_routes(app):
|
|
|
481
532
|
|
|
482
533
|
<!-- Hidden "show instructions" button -->
|
|
483
534
|
<div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
|
|
484
|
-
<button id="show-setup-btn" type="button"
|
|
535
|
+
<button id="show-setup-btn" type="button" class="btn-subtle">
|
|
485
536
|
ℹ️ Show setup instructions
|
|
486
537
|
</button>
|
|
487
538
|
</div>
|
|
@@ -492,8 +543,8 @@ def setup_captcha_routes(app):
|
|
|
492
543
|
</p>
|
|
493
544
|
|
|
494
545
|
<!-- Manual submission section -->
|
|
495
|
-
<div
|
|
496
|
-
<p style="
|
|
546
|
+
<div class="section-divider">
|
|
547
|
+
<p style="font-size: 0.9em; margin-bottom: 16px;">
|
|
497
548
|
If the userscript doesn't work, you can manually paste the links or upload a DLC file:
|
|
498
549
|
</p>
|
|
499
550
|
<form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
|
|
@@ -202,7 +202,8 @@ def process_links(shared_state, source_result, title, password, package_id, imdb
|
|
|
202
202
|
if result["success"]:
|
|
203
203
|
send_discord_message(shared_state, title=title, case="unprotected", imdb_id=imdb_id, source=source_url)
|
|
204
204
|
return {"success": True, "title": title}
|
|
205
|
-
info(f"Auto-decrypt failed for {title},
|
|
205
|
+
info(f"Auto-decrypt failed for {title}, falling back to manual CAPTCHA...")
|
|
206
|
+
classified['protected'].extend(classified['auto'])
|
|
206
207
|
|
|
207
208
|
# PRIORITY 3: Protected (filecrypt, tolink, keeplinks, junkies)
|
|
208
209
|
if classified['protected']:
|
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
# Project by https://github.com/rix1337
|
|
4
4
|
|
|
5
5
|
import re
|
|
6
|
+
from concurrent.futures import ThreadPoolExecutor, as_completed
|
|
6
7
|
from typing import List, Dict, Any
|
|
7
8
|
|
|
8
9
|
import requests
|
|
@@ -29,18 +30,31 @@ def unhide_links(shared_state, url):
|
|
|
29
30
|
response = requests.get(container_url, headers=headers)
|
|
30
31
|
data = response.json()
|
|
31
32
|
|
|
32
|
-
for link in data.get("links", [])
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
33
|
+
link_ids = [link.get("id") for link in data.get("links", []) if link.get("id")]
|
|
34
|
+
|
|
35
|
+
if not link_ids:
|
|
36
|
+
debug(f"No link IDs found in container {container_id}")
|
|
37
|
+
return []
|
|
36
38
|
|
|
39
|
+
def fetch_link(link_id):
|
|
37
40
|
debug(f"Fetching hide.cx link with ID: {link_id}")
|
|
38
41
|
link_url = f"https://api.hide.cx/containers/{container_id}/links/{link_id}"
|
|
39
42
|
link_data = requests.get(link_url, headers=headers).json()
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
43
|
+
return link_data.get("url")
|
|
44
|
+
|
|
45
|
+
# Process links in batches of 10
|
|
46
|
+
batch_size = 10
|
|
47
|
+
for i in range(0, len(link_ids), batch_size):
|
|
48
|
+
batch = link_ids[i:i + batch_size]
|
|
49
|
+
with ThreadPoolExecutor(max_workers=batch_size) as executor:
|
|
50
|
+
futures = [executor.submit(fetch_link, link_id) for link_id in batch]
|
|
51
|
+
for future in as_completed(futures):
|
|
52
|
+
try:
|
|
53
|
+
final_url = future.result()
|
|
54
|
+
if final_url and final_url not in links:
|
|
55
|
+
links.append(final_url)
|
|
56
|
+
except Exception as e:
|
|
57
|
+
info(f"Error fetching link: {e}")
|
|
44
58
|
|
|
45
59
|
success = bool(links)
|
|
46
60
|
if success:
|
|
@@ -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 = ''
|
|
@@ -24,6 +24,11 @@ def render_centered_html(inner_content):
|
|
|
24
24
|
--secondary: #6c757d;
|
|
25
25
|
--code-bg: #f8f9fa;
|
|
26
26
|
--spacing: 1rem;
|
|
27
|
+
--info-border: #2d5a2d;
|
|
28
|
+
--setup-border: var(--primary);
|
|
29
|
+
--divider-color: #dee2e6;
|
|
30
|
+
--btn-subtle-bg: #e9ecef;
|
|
31
|
+
--btn-subtle-border: #ced4da;
|
|
27
32
|
}
|
|
28
33
|
@media (prefers-color-scheme: dark) {
|
|
29
34
|
:root {
|
|
@@ -32,8 +37,57 @@ def render_centered_html(inner_content):
|
|
|
32
37
|
--card-bg: #242526;
|
|
33
38
|
--card-shadow: rgba(0, 0, 0, 0.5);
|
|
34
39
|
--code-bg: #2c2f33;
|
|
40
|
+
--info-border: #4a8c4a;
|
|
41
|
+
--setup-border: var(--primary);
|
|
42
|
+
--divider-color: #444;
|
|
43
|
+
--btn-subtle-bg: #444;
|
|
44
|
+
--btn-subtle-border: #666;
|
|
35
45
|
}
|
|
36
46
|
}
|
|
47
|
+
/* Info box styling */
|
|
48
|
+
.info-box {
|
|
49
|
+
border: 1px solid var(--info-border);
|
|
50
|
+
border-radius: 8px;
|
|
51
|
+
padding: 16px;
|
|
52
|
+
margin-bottom: 24px;
|
|
53
|
+
}
|
|
54
|
+
.info-box h3 {
|
|
55
|
+
margin-top: 0;
|
|
56
|
+
color: var(--info-border);
|
|
57
|
+
}
|
|
58
|
+
/* Setup box styling */
|
|
59
|
+
.setup-box {
|
|
60
|
+
border: 1px solid var(--setup-border);
|
|
61
|
+
border-radius: 8px;
|
|
62
|
+
padding: 16px;
|
|
63
|
+
margin-bottom: 24px;
|
|
64
|
+
}
|
|
65
|
+
.setup-box h3 {
|
|
66
|
+
margin-top: 0;
|
|
67
|
+
color: var(--setup-border);
|
|
68
|
+
}
|
|
69
|
+
/* Subtle button styling (ghost style) */
|
|
70
|
+
.btn-subtle {
|
|
71
|
+
background: transparent;
|
|
72
|
+
color: var(--fg-color);
|
|
73
|
+
border: 1.5px solid var(--btn-subtle-border);
|
|
74
|
+
padding: 8px 16px;
|
|
75
|
+
border-radius: 0.5rem;
|
|
76
|
+
cursor: pointer;
|
|
77
|
+
font-size: 1rem;
|
|
78
|
+
font-weight: 500;
|
|
79
|
+
transition: background-color 0.2s ease, border-color 0.2s ease;
|
|
80
|
+
}
|
|
81
|
+
.btn-subtle:hover {
|
|
82
|
+
background: var(--btn-subtle-bg);
|
|
83
|
+
border-color: var(--fg-color);
|
|
84
|
+
}
|
|
85
|
+
/* Divider styling */
|
|
86
|
+
.section-divider {
|
|
87
|
+
margin-top: 20px;
|
|
88
|
+
padding-top: 20px;
|
|
89
|
+
border-top: 1px solid var(--divider-color);
|
|
90
|
+
}
|
|
37
91
|
/* Logo and heading alignment */
|
|
38
92
|
h1 {
|
|
39
93
|
display: inline-flex;
|
|
@@ -142,24 +196,35 @@ def render_centered_html(inner_content):
|
|
|
142
196
|
border-radius: 0.5rem;
|
|
143
197
|
font-weight: 500;
|
|
144
198
|
cursor: pointer;
|
|
145
|
-
transition: background-color 0.2s ease, border-color 0.2s ease;
|
|
146
|
-
border: none;
|
|
199
|
+
transition: background-color 0.2s ease, border-color 0.2s ease, box-shadow 0.2s ease;
|
|
147
200
|
margin-top: 0.5rem;
|
|
148
201
|
}
|
|
149
202
|
.btn-primary {
|
|
150
203
|
background-color: var(--primary);
|
|
151
204
|
color: #fff;
|
|
205
|
+
border: 1.5px solid #0856c7;
|
|
206
|
+
}
|
|
207
|
+
.btn-primary:hover {
|
|
208
|
+
background-color: #0b5ed7;
|
|
209
|
+
border-color: #084298;
|
|
210
|
+
box-shadow: 0 2px 6px rgba(13, 110, 253, 0.4);
|
|
152
211
|
}
|
|
153
212
|
.btn-secondary {
|
|
154
213
|
background-color: var(--secondary);
|
|
155
214
|
color: #fff;
|
|
215
|
+
border: 1.5px solid #565e64;
|
|
216
|
+
}
|
|
217
|
+
.btn-secondary:hover {
|
|
218
|
+
background-color: #5c636a;
|
|
219
|
+
border-color: #41464b;
|
|
220
|
+
box-shadow: 0 2px 6px rgba(108, 117, 125, 0.4);
|
|
156
221
|
}
|
|
157
222
|
a {
|
|
158
223
|
color: var(--primary);
|
|
159
224
|
text-decoration: none;
|
|
160
225
|
}
|
|
161
226
|
a:hover {
|
|
162
|
-
|
|
227
|
+
|
|
163
228
|
}
|
|
164
229
|
/* footer styling */
|
|
165
230
|
footer {
|