quasarr 1.24.1__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.24.1 → quasarr-1.26.0}/PKG-INFO +3 -3
  2. {quasarr-1.24.1 → quasarr-1.26.0}/README.md +2 -2
  3. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/captcha/__init__.py +181 -63
  4. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/__init__.py +2 -1
  5. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/hide.py +22 -8
  6. quasarr-1.26.0/quasarr/providers/html_images.py +22 -0
  7. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/html_templates.py +48 -1
  8. quasarr-1.26.0/quasarr/providers/obfuscated.py +104 -0
  9. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/version.py +1 -1
  10. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/web_server.py +8 -1
  11. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/PKG-INFO +3 -3
  12. quasarr-1.24.1/quasarr/providers/html_images.py +0 -22
  13. quasarr-1.24.1/quasarr/providers/obfuscated.py +0 -86
  14. {quasarr-1.24.1 → quasarr-1.26.0}/LICENSE +0 -0
  15. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/__init__.py +0 -0
  16. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/__init__.py +0 -0
  17. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/arr/__init__.py +0 -0
  18. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/config/__init__.py +0 -0
  19. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/sponsors_helper/__init__.py +0 -0
  20. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/api/statistics/__init__.py +0 -0
  21. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/__init__.py +0 -0
  22. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/al.py +0 -0
  23. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/linkcrypters/filecrypt.py +0 -0
  24. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/packages/__init__.py +0 -0
  25. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/__init__.py +0 -0
  26. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/al.py +0 -0
  27. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/by.py +0 -0
  28. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/dd.py +0 -0
  29. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/dj.py +0 -0
  30. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/dl.py +0 -0
  31. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/dt.py +0 -0
  32. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/dw.py +0 -0
  33. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/he.py +0 -0
  34. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/mb.py +0 -0
  35. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/nk.py +0 -0
  36. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/nx.py +0 -0
  37. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/sf.py +0 -0
  38. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/sj.py +0 -0
  39. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/sl.py +0 -0
  40. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/wd.py +0 -0
  41. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/downloads/sources/wx.py +0 -0
  42. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/__init__.py +0 -0
  43. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/cloudflare.py +0 -0
  44. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/imdb_metadata.py +0 -0
  45. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/log.py +0 -0
  46. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/myjd_api.py +0 -0
  47. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/notifications.py +0 -0
  48. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/sessions/__init__.py +0 -0
  49. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/sessions/al.py +0 -0
  50. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/sessions/dd.py +0 -0
  51. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/sessions/dl.py +0 -0
  52. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/sessions/nx.py +0 -0
  53. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/shared_state.py +0 -0
  54. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/providers/statistics.py +0 -0
  55. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/__init__.py +0 -0
  56. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/__init__.py +0 -0
  57. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/al.py +0 -0
  58. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/by.py +0 -0
  59. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/dd.py +0 -0
  60. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/dj.py +0 -0
  61. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/dl.py +0 -0
  62. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/dt.py +0 -0
  63. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/dw.py +0 -0
  64. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/fx.py +0 -0
  65. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/he.py +0 -0
  66. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/mb.py +0 -0
  67. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/nk.py +0 -0
  68. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/nx.py +0 -0
  69. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/sf.py +0 -0
  70. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/sj.py +0 -0
  71. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/sl.py +0 -0
  72. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/wd.py +0 -0
  73. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/search/sources/wx.py +0 -0
  74. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/storage/__init__.py +0 -0
  75. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/storage/config.py +0 -0
  76. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/storage/setup.py +0 -0
  77. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr/storage/sqlite_database.py +0 -0
  78. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/SOURCES.txt +0 -0
  79. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/dependency_links.txt +0 -0
  80. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/entry_points.txt +0 -0
  81. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/not-zip-safe +0 -0
  82. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/requires.txt +0 -0
  83. {quasarr-1.24.1 → quasarr-1.26.0}/quasarr.egg-info/top_level.txt +0 -0
  84. {quasarr-1.24.1 → quasarr-1.26.0}/setup.cfg +0 -0
  85. {quasarr-1.24.1 → 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.24.1
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
@@ -36,12 +36,12 @@ Quasarr connects JDownloader with Radarr, Sonarr and LazyLibrarian. It also decr
36
36
  [![GitHub Sponsorship](https://img.shields.io/badge/support-me-red.svg)](https://github.com/users/rix1337/sponsorship)
37
37
 
38
38
  Quasarr pretends to be both `Newznab Indexer` and `SABnzbd client`. Therefore, do not try to use it with real usenet
39
- indexers or download clients. It simply does not know what NZB or torrent files are.
39
+ indexers. It simply does not know what NZB files are.
40
40
 
41
41
  Quasarr includes a solution to quickly and easily decrypt protected links.
42
42
  [Active monthly Sponsors get access to SponsorsHelper to do so automatically.](https://github.com/rix1337/Quasarr?tab=readme-ov-file#sponsorshelper)
43
43
  Alternatively, follow the link from the console output (or discord notification) to solve CAPTCHAs manually.
44
- Quasarr will confidently handle the rest.
44
+ Quasarr will confidently handle the rest. Some CAPTCHA types require [Tampermonkey](https://www.tampermonkey.net/) to be installed in your browser.
45
45
 
46
46
  # Instructions
47
47
  1. Set up and run [FlareSolverr 3](https://github.com/FlareSolverr/FlareSolverr).
@@ -9,12 +9,12 @@ Quasarr connects JDownloader with Radarr, Sonarr and LazyLibrarian. It also decr
9
9
  [![GitHub Sponsorship](https://img.shields.io/badge/support-me-red.svg)](https://github.com/users/rix1337/sponsorship)
10
10
 
11
11
  Quasarr pretends to be both `Newznab Indexer` and `SABnzbd client`. Therefore, do not try to use it with real usenet
12
- indexers or download clients. It simply does not know what NZB or torrent files are.
12
+ indexers. It simply does not know what NZB files are.
13
13
 
14
14
  Quasarr includes a solution to quickly and easily decrypt protected links.
15
15
  [Active monthly Sponsors get access to SponsorsHelper to do so automatically.](https://github.com/rix1337/Quasarr?tab=readme-ov-file#sponsorshelper)
16
16
  Alternatively, follow the link from the console output (or discord notification) to solve CAPTCHAs manually.
17
- Quasarr will confidently handle the rest.
17
+ Quasarr will confidently handle the rest. Some CAPTCHA types require [Tampermonkey](https://www.tampermonkey.net/) to be installed in your browser.
18
18
 
19
19
  # Instructions
20
20
  1. Set up and run [FlareSolverr 3](https://github.com/FlareSolverr/FlareSolverr).
@@ -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"
@@ -182,45 +191,89 @@ def setup_captcha_routes(app):
182
191
 
183
192
  return f'''
184
193
  <div>
194
+ <!-- Info section explaining the process -->
195
+ <div class="info-box">
196
+ <h3>ℹ️ How This Works:</h3>
197
+ <p style="margin-bottom: 8px;">
198
+ 1. Click the link below to open {provider_name}
199
+ </p>
200
+ <p style="margin-top: 0; margin-bottom: 8px;">
201
+ 2. Solve any CAPTCHAs on their site to reveal the download links
202
+ </p>
203
+ <p style="margin-top: 0; margin-bottom: 0;">
204
+ 3. <b>With the userscript installed</b>, links are automatically sent back to Quasarr!
205
+ </p>
206
+ </div>
207
+
185
208
  <!-- One-time setup section - visually separated -->
186
- <div id="setup-instructions" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
187
- <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>
188
211
  <p style="margin-bottom: 8px;">
189
212
  <a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
190
213
  </p>
191
214
  <p style="margin-top: 0; margin-bottom: 12px;">
192
- <a href="{userscript_url}" target="_blank">2. Install this userscript</a>
215
+ <a href="{userscript_url}" target="_blank">2. Install the {provider_name} userscript</a>
193
216
  </p>
194
217
  <p style="margin-top: 0;">
195
- <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">
196
219
  ✅ Don't show this again
197
220
  </button>
198
221
  </p>
199
222
  </div>
200
223
 
201
- <!-- Hidden "show instructions" link -->
224
+ <!-- Hidden "show instructions" button -->
202
225
  <div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
203
- <a href="#" id="show-setup-btn" style="color: #58a6ff;">ℹ️ Show instructions again</a>
226
+ <button id="show-setup-btn" type="button" class="btn-subtle">
227
+ ℹ️ Show setup instructions
228
+ </button>
204
229
  </div>
205
230
 
206
- <strong><a href="{url_with_quick_transfer_params}" target="_self">🔗 Obtain the download links here!</a></strong><br><br>
207
-
208
- <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
209
- <input type="hidden" name="package_id" value="{package_id}" />
210
- <input type="hidden" name="title" value="{title}" />
211
- <input type="hidden" name="password" value="{password}" />
212
-
213
- <div>
214
- <strong>Paste the download links (one per line):</strong>
215
- <textarea id="links-input" name="links" rows="5" style="width: 100%; padding: 8px; font-family: monospace; resize: vertical;"></textarea>
216
- </div>
231
+ <!-- Primary action - the quick transfer link -->
232
+ <p>
233
+ {render_button(f"Open {provider_name} & Get Download Links", "primary", {"onclick": f"location.href='{url_with_quick_transfer_params}'"})}
234
+ </p>
217
235
 
218
- <div>
219
- {render_button("Submit", "primary", {"type": "submit"})}
220
- </div>
221
- </form>
236
+ <!-- Manual submission - collapsible -->
237
+ <div class="section-divider">
238
+ <details id="manualSubmitDetails">
239
+ <summary id="manualSubmitSummary" style="cursor: pointer;">Show Manual Submission</summary>
240
+ <div style="margin-top: 16px;">
241
+ <p style="font-size: 0.9em;">
242
+ If the userscript doesn't work, you can manually paste the links below:
243
+ </p>
244
+ <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
245
+ <input type="hidden" name="package_id" value="{package_id}" />
246
+ <input type="hidden" name="title" value="{title}" />
247
+ <input type="hidden" name="password" value="{password}" />
248
+
249
+ <div>
250
+ <strong>Paste the download links (one per line):</strong>
251
+ <textarea id="links-input" name="links" rows="5" style="width: 100%; padding: 8px; font-family: monospace; resize: vertical;"></textarea>
252
+ </div>
253
+
254
+ <div>
255
+ {render_button("Submit", "primary", {"type": "submit"})}
256
+ </div>
257
+ </form>
258
+ </div>
259
+ </details>
260
+ </div>
222
261
  </div>
223
262
  <script>
263
+ // Handle manual submission toggle text
264
+ const manualDetails = document.getElementById('manualSubmitDetails');
265
+ const manualSummary = document.getElementById('manualSubmitSummary');
266
+
267
+ if (manualDetails && manualSummary) {{
268
+ manualDetails.addEventListener('toggle', () => {{
269
+ if (manualDetails.open) {{
270
+ manualSummary.textContent = 'Hide Manual Submission';
271
+ }} else {{
272
+ manualSummary.textContent = 'Show Manual Submission';
273
+ }}
274
+ }});
275
+ }}
276
+
224
277
  // Handle setup instructions hide/show
225
278
  const hideSetup = localStorage.getItem('{storage_key}');
226
279
  const setupBox = document.getElementById('setup-instructions');
@@ -239,8 +292,7 @@ def setup_captcha_routes(app):
239
292
  }});
240
293
 
241
294
  // Show setup instructions again
242
- document.getElementById('show-setup-btn').addEventListener('click', function(e) {{
243
- e.preventDefault();
295
+ document.getElementById('show-setup-btn').addEventListener('click', function() {{
244
296
  localStorage.setItem('{storage_key}', 'false');
245
297
  setupBox.style.display = 'block';
246
298
  showLink.style.display = 'none';
@@ -248,6 +300,42 @@ def setup_captcha_routes(app):
248
300
  </script>
249
301
  '''
250
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
+
251
339
  @app.get("/captcha/junkies")
252
340
  def serve_junkies_captcha():
253
341
  payload = decode_payload()
@@ -358,6 +446,18 @@ def setup_captcha_routes(app):
358
446
  </body>
359
447
  </html>""")
360
448
 
449
+ @app.get('/captcha/filecrypt.user.js')
450
+ def serve_filecrypt_user_js():
451
+ content = obfuscated.filecrypt_user_js()
452
+ response.content_type = 'application/javascript'
453
+ return content
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
+
361
461
  @app.get('/captcha/junkies.user.js')
362
462
  def serve_junkies_user_js():
363
463
  sj = shared_state.values["config"]("Hostnames").get("sj")
@@ -379,12 +479,6 @@ def setup_captcha_routes(app):
379
479
  response.content_type = 'application/javascript'
380
480
  return content
381
481
 
382
- @app.get('/captcha/filecrypt.user.js')
383
- def serve_filecrypt_user_js():
384
- content = obfuscated.filecrypt_user_js()
385
- response.content_type = 'application/javascript'
386
- return content
387
-
388
482
  def render_filecrypt_bypass_section(url, package_id, title, password):
389
483
  """Render the bypass UI section for both cutcaptcha and circle captcha pages"""
390
484
 
@@ -402,52 +496,77 @@ def setup_captcha_routes(app):
402
496
  )
403
497
 
404
498
  return f'''
405
- <div style="margin-top: 40px; padding-top: 20px; border-top: 2px solid #ccc;">
499
+ <div class="section-divider" style="max-width: 370px; margin-left: auto; margin-right: auto;">
406
500
  <details id="bypassDetails">
407
501
  <summary id="bypassSummary">Show CAPTCHA Bypass</summary><br>
408
502
 
503
+ <!-- Info section explaining the process -->
504
+ <div class="info-box">
505
+ <h3>ℹ️ How This Works:</h3>
506
+ <p style="margin-bottom: 8px;">
507
+ 1. Click the button below to open FileCrypt directly
508
+ </p>
509
+ <p style="margin-top: 0; margin-bottom: 8px;">
510
+ 2. Solve any CAPTCHAs on their site to reveal the download links
511
+ </p>
512
+ <p style="margin-top: 0; margin-bottom: 0;">
513
+ 3. <b>With the userscript installed</b>, links are automatically sent back to Quasarr!
514
+ </p>
515
+ </div>
516
+
409
517
  <!-- One-time setup section - visually separated -->
410
- <div id="setup-instructions" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
411
- <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>
412
520
  <p style="margin-bottom: 8px;">
413
521
  <a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
414
522
  </p>
415
523
  <p style="margin-top: 0; margin-bottom: 12px;">
416
- <a href="/captcha/filecrypt.user.js" target="_blank">2. Install this userscript</a>
524
+ <a href="/captcha/filecrypt.user.js" target="_blank">2. Install the FileCrypt userscript</a>
417
525
  </p>
418
526
  <p style="margin-top: 0;">
419
- <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">
420
528
  ✅ Don't show this again
421
529
  </button>
422
530
  </p>
423
531
  </div>
424
532
 
425
- <!-- Hidden "show instructions" link -->
533
+ <!-- Hidden "show instructions" button -->
426
534
  <div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
427
- <a href="#" id="show-setup-btn" style="color: #58a6ff;">ℹ️ Show instructions again</a>
535
+ <button id="show-setup-btn" type="button" class="btn-subtle">
536
+ ℹ️ Show setup instructions
537
+ </button>
428
538
  </div>
429
539
 
430
- <strong><a href="{url_with_quick_transfer_params}" target="_self">🔗 Obtain the download links here!</a></strong><br><br>
431
-
432
- <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
433
- <input type="hidden" name="package_id" value="{package_id}" />
434
- <input type="hidden" name="title" value="{title}" />
435
- <input type="hidden" name="password" value="{password}" />
436
-
437
- <div>
438
- <strong>Paste the download links (one per line):</strong>
439
- <textarea id="links-input" name="links" rows="5" style="width: 100%; padding: 8px; font-family: monospace; resize: vertical;"></textarea>
440
- </div>
441
-
442
- <div>
443
- <strong>Or upload DLC file:</strong><br>
444
- <input type="file" id="dlc-file" name="dlc_file" accept=".dlc" />
445
- </div>
540
+ <!-- Primary action button -->
541
+ <p>
542
+ {render_button("Open FileCrypt & Get Download Links", "primary", {"onclick": f"location.href='{url_with_quick_transfer_params}'"})}
543
+ </p>
446
544
 
447
- <div>
448
- {render_button("Submit", "primary", {"type": "submit"})}
449
- </div>
450
- </form>
545
+ <!-- Manual submission section -->
546
+ <div class="section-divider">
547
+ <p style="font-size: 0.9em; margin-bottom: 16px;">
548
+ If the userscript doesn't work, you can manually paste the links or upload a DLC file:
549
+ </p>
550
+ <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
551
+ <input type="hidden" name="package_id" value="{package_id}" />
552
+ <input type="hidden" name="title" value="{title}" />
553
+ <input type="hidden" name="password" value="{password}" />
554
+
555
+ <div>
556
+ <strong>Paste the download links (one per line):</strong>
557
+ <textarea id="links-input" name="links" rows="5" style="width: 100%; padding: 8px; font-family: monospace; resize: vertical;"></textarea>
558
+ </div>
559
+
560
+ <div>
561
+ <strong>Or upload DLC file:</strong><br>
562
+ <input type="file" id="dlc-file" name="dlc_file" accept=".dlc" />
563
+ </div>
564
+
565
+ <div>
566
+ {render_button("Submit", "primary", {"type": "submit"})}
567
+ </div>
568
+ </form>
569
+ </div>
451
570
  </details>
452
571
  </div>
453
572
  <script>
@@ -483,8 +602,7 @@ def setup_captcha_routes(app):
483
602
  }});
484
603
 
485
604
  // Show setup instructions again
486
- document.getElementById('show-setup-btn').addEventListener('click', function(e) {{
487
- e.preventDefault();
605
+ document.getElementById('show-setup-btn').addEventListener('click', function() {{
488
606
  localStorage.setItem('hideFileCryptSetupInstructions', 'false');
489
607
  setupBox.style.display = 'block';
490
608
  showLink.style.display = 'none';
@@ -764,7 +882,7 @@ def setup_captcha_routes(app):
764
882
  reloadSection.style.display = "block";
765
883
  });
766
884
  }
767
- ''' + obfuscated.captcha_js() + f'''</script>
885
+ ''' + obfuscated.cutcaptcha_custom_js() + f'''</script>
768
886
  <div>
769
887
  <h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
770
888
  <p id="package-title" style="max-width: 370px; word-wrap: break-word; overflow-wrap: break-word;"><b>Package:</b> {title}</p>
@@ -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 {