quasarr 1.25.0__tar.gz → 1.26.0__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.

Files changed (85) hide show
  1. {quasarr-1.25.0 → quasarr-1.26.0}/PKG-INFO +1 -1
  2. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/captcha/__init__.py +73 -22
  3. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/__init__.py +2 -1
  4. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/hide.py +22 -8
  5. quasarr-1.26.0/quasarr/providers/html_images.py +22 -0
  6. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/html_templates.py +48 -1
  7. quasarr-1.26.0/quasarr/providers/obfuscated.py +104 -0
  8. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/version.py +1 -1
  9. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/PKG-INFO +1 -1
  10. quasarr-1.25.0/quasarr/providers/html_images.py +0 -22
  11. quasarr-1.25.0/quasarr/providers/obfuscated.py +0 -86
  12. {quasarr-1.25.0 → quasarr-1.26.0}/LICENSE +0 -0
  13. {quasarr-1.25.0 → quasarr-1.26.0}/README.md +0 -0
  14. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/__init__.py +0 -0
  15. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/__init__.py +0 -0
  16. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/arr/__init__.py +0 -0
  17. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/config/__init__.py +0 -0
  18. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/sponsors_helper/__init__.py +0 -0
  19. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/api/statistics/__init__.py +0 -0
  20. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/__init__.py +0 -0
  21. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/al.py +0 -0
  22. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/filecrypt.py +0 -0
  23. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/packages/__init__.py +0 -0
  24. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/__init__.py +0 -0
  25. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/al.py +0 -0
  26. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/by.py +0 -0
  27. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/dd.py +0 -0
  28. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/dj.py +0 -0
  29. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/dl.py +0 -0
  30. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/dt.py +0 -0
  31. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/dw.py +0 -0
  32. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/he.py +0 -0
  33. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/mb.py +0 -0
  34. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/nk.py +0 -0
  35. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/nx.py +0 -0
  36. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/sf.py +0 -0
  37. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/sj.py +0 -0
  38. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/sl.py +0 -0
  39. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/wd.py +0 -0
  40. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/downloads/sources/wx.py +0 -0
  41. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/__init__.py +0 -0
  42. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/cloudflare.py +0 -0
  43. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/imdb_metadata.py +0 -0
  44. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/log.py +0 -0
  45. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/myjd_api.py +0 -0
  46. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/notifications.py +0 -0
  47. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/sessions/__init__.py +0 -0
  48. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/sessions/al.py +0 -0
  49. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/sessions/dd.py +0 -0
  50. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/sessions/dl.py +0 -0
  51. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/sessions/nx.py +0 -0
  52. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/shared_state.py +0 -0
  53. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/statistics.py +0 -0
  54. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/providers/web_server.py +0 -0
  55. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/__init__.py +0 -0
  56. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/__init__.py +0 -0
  57. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/al.py +0 -0
  58. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/by.py +0 -0
  59. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/dd.py +0 -0
  60. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/dj.py +0 -0
  61. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/dl.py +0 -0
  62. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/dt.py +0 -0
  63. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/dw.py +0 -0
  64. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/fx.py +0 -0
  65. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/he.py +0 -0
  66. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/mb.py +0 -0
  67. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/nk.py +0 -0
  68. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/nx.py +0 -0
  69. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/sf.py +0 -0
  70. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/sj.py +0 -0
  71. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/sl.py +0 -0
  72. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/wd.py +0 -0
  73. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/search/sources/wx.py +0 -0
  74. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/storage/__init__.py +0 -0
  75. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/storage/config.py +0 -0
  76. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/storage/setup.py +0 -0
  77. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr/storage/sqlite_database.py +0 -0
  78. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/SOURCES.txt +0 -0
  79. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/dependency_links.txt +0 -0
  80. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/entry_points.txt +0 -0
  81. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/not-zip-safe +0 -0
  82. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/requires.txt +0 -0
  83. {quasarr-1.25.0 → quasarr-1.26.0}/quasarr.egg-info/top_level.txt +0 -0
  84. {quasarr-1.25.0 → quasarr-1.26.0}/setup.cfg +0 -0
  85. {quasarr-1.25.0 → quasarr-1.26.0}/setup.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: quasarr
3
- Version: 1.25.0
3
+ Version: 1.26.0
4
4
  Summary: Quasarr connects JDownloader with Radarr, Sonarr and LazyLibrarian. It also decrypts links protected by CAPTCHAs.
5
5
  Home-page: https://github.com/rix1337/Quasarr
6
6
  Author: rix1337
@@ -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 has_junkies_links:
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 ToLink pages
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 style="background: #1a3a1a; border: 2px solid #2d5a2d; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
187
- <h3 style="margin-top: 0; color: #6fbf6f;">ℹ️ How This Works:</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" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
201
- <h3 style="margin-top: 0; color: #58a6ff;">📦 First Time Setup:</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" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
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" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
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 style="margin-top: 20px; padding-top: 20px; border-top: 2px solid #444;">
237
+ <div class="section-divider">
229
238
  <details id="manualSubmitDetails">
230
- <summary id="manualSubmitSummary" style="cursor: pointer; color: #888;">Show Manual Submission</summary>
239
+ <summary id="manualSubmitSummary" style="cursor: pointer;">Show Manual Submission</summary>
231
240
  <div style="margin-top: 16px;">
232
- <p style="color: #888; font-size: 0.9em;">
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 style="margin-top: 40px; padding-top: 20px; border-top: 2px solid #ccc; max-width: 370px; margin-left: auto; margin-right: auto;">
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 style="background: #1a3a1a; border: 2px solid #2d5a2d; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
454
- <h3 style="margin-top: 0; color: #6fbf6f;">ℹ️ How This Works:</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" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
468
- <h3 style="margin-top: 0; color: #58a6ff;">📦 First Time Setup:</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" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
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" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
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 style="margin-top: 24px; padding-top: 20px; border-top: 2px solid #444;">
496
- <p style="color: #888; font-size: 0.9em; margin-bottom: 16px;">
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}, checking for protected fallback...")
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
- link_id = link.get("id")
34
- if not link_id:
35
- continue
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
- final_url = link_data.get("url")
42
- if final_url and final_url not in links:
43
- links.append(final_url)
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,50 @@ 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 */
70
+ .btn-subtle {
71
+ background: var(--btn-subtle-bg);
72
+ color: var(--fg-color);
73
+ border: 1px solid var(--btn-subtle-border);
74
+ padding: 6px 12px;
75
+ border-radius: 4px;
76
+ cursor: pointer;
77
+ }
78
+ /* Divider styling */
79
+ .section-divider {
80
+ margin-top: 20px;
81
+ padding-top: 20px;
82
+ border-top: 1px solid var(--divider-color);
83
+ }
37
84
  /* Logo and heading alignment */
38
85
  h1 {
39
86
  display: inline-flex;
@@ -159,7 +206,7 @@ def render_centered_html(inner_content):
159
206
  text-decoration: none;
160
207
  }
161
208
  a:hover {
162
-
209
+
163
210
  }
164
211
  /* footer styling */
165
212
  footer {