quasarr 1.17.3__tar.gz → 1.19.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 (74) hide show
  1. {quasarr-1.17.3 → quasarr-1.19.0}/PKG-INFO +1 -1
  2. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/captcha/__init__.py +56 -1
  3. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/__init__.py +6 -0
  4. quasarr-1.19.0/quasarr/downloads/sources/he.py +112 -0
  5. quasarr-1.19.0/quasarr/downloads/sources/nk.py +51 -0
  6. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/html_images.py +2 -0
  7. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/obfuscated.py +18 -0
  8. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/version.py +1 -1
  9. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/__init__.py +8 -0
  10. quasarr-1.19.0/quasarr/search/sources/he.py +196 -0
  11. quasarr-1.19.0/quasarr/search/sources/nk.py +188 -0
  12. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/storage/config.py +2 -0
  13. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/PKG-INFO +1 -1
  14. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/SOURCES.txt +4 -0
  15. {quasarr-1.17.3 → quasarr-1.19.0}/LICENSE +0 -0
  16. {quasarr-1.17.3 → quasarr-1.19.0}/README.md +0 -0
  17. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/__init__.py +0 -0
  18. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/__init__.py +0 -0
  19. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/arr/__init__.py +0 -0
  20. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/config/__init__.py +0 -0
  21. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/sponsors_helper/__init__.py +0 -0
  22. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/api/statistics/__init__.py +0 -0
  23. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/__init__.py +0 -0
  24. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/al.py +0 -0
  25. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/filecrypt.py +0 -0
  26. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/hide.py +0 -0
  27. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/packages/__init__.py +0 -0
  28. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/__init__.py +0 -0
  29. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/al.py +0 -0
  30. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/by.py +0 -0
  31. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/dd.py +0 -0
  32. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/dt.py +0 -0
  33. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/dw.py +0 -0
  34. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/mb.py +0 -0
  35. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/nx.py +0 -0
  36. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/sf.py +0 -0
  37. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/sl.py +0 -0
  38. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/downloads/sources/wd.py +0 -0
  39. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/__init__.py +0 -0
  40. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/cloudflare.py +0 -0
  41. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/html_templates.py +0 -0
  42. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/imdb_metadata.py +0 -0
  43. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/log.py +0 -0
  44. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/myjd_api.py +0 -0
  45. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/notifications.py +0 -0
  46. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/sessions/__init__.py +0 -0
  47. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/sessions/al.py +0 -0
  48. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/sessions/dd.py +0 -0
  49. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/sessions/nx.py +0 -0
  50. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/shared_state.py +0 -0
  51. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/statistics.py +0 -0
  52. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/providers/web_server.py +0 -0
  53. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/__init__.py +0 -0
  54. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/al.py +0 -0
  55. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/by.py +0 -0
  56. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/dd.py +0 -0
  57. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/dt.py +0 -0
  58. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/dw.py +0 -0
  59. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/fx.py +0 -0
  60. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/mb.py +0 -0
  61. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/nx.py +0 -0
  62. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/sf.py +0 -0
  63. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/sl.py +0 -0
  64. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/search/sources/wd.py +0 -0
  65. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/storage/__init__.py +0 -0
  66. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/storage/setup.py +0 -0
  67. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr/storage/sqlite_database.py +0 -0
  68. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/dependency_links.txt +0 -0
  69. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/entry_points.txt +0 -0
  70. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/not-zip-safe +0 -0
  71. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/requires.txt +0 -0
  72. {quasarr-1.17.3 → quasarr-1.19.0}/quasarr.egg-info/top_level.txt +0 -0
  73. {quasarr-1.17.3 → quasarr-1.19.0}/setup.cfg +0 -0
  74. {quasarr-1.17.3 → quasarr-1.19.0}/setup.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: quasarr
3
- Version: 1.17.3
3
+ Version: 1.19.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
@@ -16,7 +16,7 @@ from quasarr.downloads.packages import delete_package
16
16
  from quasarr.providers import shared_state
17
17
  from quasarr.providers.html_templates import render_button, render_centered_html
18
18
  from quasarr.providers.log import info, debug
19
- from quasarr.providers.obfuscated import captcha_js, captcha_values
19
+ from quasarr.providers.obfuscated import captcha_js, captcha_values, filecrypt_quasarr_helper_user_js
20
20
  from quasarr.providers.statistics import StatsHelper
21
21
 
22
22
 
@@ -101,6 +101,12 @@ def setup_captcha_routes(app):
101
101
  except Exception as e:
102
102
  return {"error": f"Failed to decode payload: {str(e)}"}
103
103
 
104
+ @app.get('/captcha/quasarr.user.js')
105
+ def serve_quasarr_user_js():
106
+ content = filecrypt_quasarr_helper_user_js()
107
+ response.content_type = 'application/javascript'
108
+ return content
109
+
104
110
  def render_bypass_section(url, package_id, title, password):
105
111
  """Render the bypass UI section for both cutcaptcha and circle captcha pages"""
106
112
 
@@ -121,7 +127,30 @@ def setup_captcha_routes(app):
121
127
  <div style="margin-top: 40px; padding-top: 20px; border-top: 2px solid #ccc;">
122
128
  <details id="bypassDetails">
123
129
  <summary id="bypassSummary">Show CAPTCHA Bypass</summary><br>
130
+
131
+ <!-- One-time setup section - visually separated -->
132
+ <div id="setup-instructions" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
133
+ <h3 style="margin-top: 0; color: #58a6ff;">First Time Setup:</h3>
134
+ <p style="margin-bottom: 8px;">
135
+ <a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
136
+ </p>
137
+ <p style="margin-top: 0; margin-bottom: 12px;">
138
+ <a href="/captcha/quasarr.user.js" target="_blank">2. Install this userscript</a>
139
+ </p>
140
+ <p style="margin-top: 0;">
141
+ <button id="hide-setup-btn" type="button" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
142
+ ✅ Don't show this again
143
+ </button>
144
+ </p>
145
+ </div>
146
+
147
+ <!-- Hidden "show instructions" link -->
148
+ <div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
149
+ <a href="#" id="show-setup-btn" style="color: #58a6ff; text-decoration: underline;">ℹ️ Show instructions again</a>
150
+ </div>
151
+
124
152
  <strong><a href="{url_with_quick_transfer_params}" target="_blank">🔗 Obtain the download links here!</a></strong><br><br>
153
+
125
154
  <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
126
155
  <input type="hidden" name="package_id" value="{package_id}" />
127
156
  <input type="hidden" name="title" value="{title}" />
@@ -144,6 +173,7 @@ def setup_captcha_routes(app):
144
173
  </details>
145
174
  </div>
146
175
  <script>
176
+ // Handle CAPTCHA Bypass toggle
147
177
  const bypassDetails = document.getElementById('bypassDetails');
148
178
  const bypassSummary = document.getElementById('bypassSummary');
149
179
 
@@ -156,6 +186,31 @@ def setup_captcha_routes(app):
156
186
  }}
157
187
  }});
158
188
  }}
189
+
190
+ // Handle setup instructions hide/show
191
+ const hideSetup = localStorage.getItem('hideSetupInstructions');
192
+ const setupBox = document.getElementById('setup-instructions');
193
+ const showLink = document.getElementById('show-instructions-link');
194
+
195
+ if (hideSetup === 'true') {{
196
+ setupBox.style.display = 'none';
197
+ showLink.style.display = 'block';
198
+ }}
199
+
200
+ // Hide setup instructions
201
+ document.getElementById('hide-setup-btn').addEventListener('click', function() {{
202
+ localStorage.setItem('hideSetupInstructions', 'true');
203
+ setupBox.style.display = 'none';
204
+ showLink.style.display = 'block';
205
+ }});
206
+
207
+ // Show setup instructions again
208
+ document.getElementById('show-setup-btn').addEventListener('click', function(e) {{
209
+ e.preventDefault();
210
+ localStorage.setItem('hideSetupInstructions', 'false');
211
+ setupBox.style.display = 'block';
212
+ showLink.style.display = 'none';
213
+ }});
159
214
  </script>
160
215
  '''
161
216
 
@@ -13,7 +13,9 @@ from quasarr.downloads.sources.by import get_by_download_links
13
13
  from quasarr.downloads.sources.dd import get_dd_download_links
14
14
  from quasarr.downloads.sources.dt import get_dt_download_links
15
15
  from quasarr.downloads.sources.dw import get_dw_download_links
16
+ from quasarr.downloads.sources.he import get_he_download_links
16
17
  from quasarr.downloads.sources.mb import get_mb_download_links
18
+ from quasarr.downloads.sources.nk import get_nk_download_links
17
19
  from quasarr.downloads.sources.nx import get_nx_download_links
18
20
  from quasarr.downloads.sources.sf import get_sf_download_links, resolve_sf_redirect
19
21
  from quasarr.downloads.sources.sl import get_sl_download_links
@@ -201,7 +203,9 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
201
203
  'DD': config.get("dd"),
202
204
  'DT': config.get("dt"),
203
205
  'DW': config.get("dw"),
206
+ 'HE': config.get("he"),
204
207
  'MB': config.get("mb"),
208
+ 'NK': config.get("nk"),
205
209
  'NX': config.get("nx"),
206
210
  'SF': config.get("sf"),
207
211
  'SL': config.get("sl"),
@@ -214,7 +218,9 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
214
218
  (flags['DD'], lambda *a: handle_unprotected(*a, func=get_dd_download_links, label='DD')),
215
219
  (flags['DT'], lambda *a: handle_unprotected(*a, func=get_dt_download_links, label='DT')),
216
220
  (flags['DW'], lambda *a: handle_protected(*a, func=get_dw_download_links, label='DW')),
221
+ (flags['HE'], lambda *a: handle_unprotected(*a, func=get_he_download_links, label='HE')),
217
222
  (flags['MB'], lambda *a: handle_protected(*a, func=get_mb_download_links, label='MB')),
223
+ (flags['NK'], lambda *a: handle_protected(*a, func=get_nk_download_links, label='NK')),
218
224
  (flags['NX'], lambda *a: handle_unprotected(*a, func=get_nx_download_links, label='NX')),
219
225
  (flags['SF'], handle_sf),
220
226
  (flags['SL'], handle_sl),
@@ -0,0 +1,112 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ import re
6
+ from urllib.parse import urlparse, urljoin
7
+
8
+ import requests
9
+ from bs4 import BeautifulSoup
10
+
11
+ from quasarr.providers.log import info, debug
12
+
13
+ hostname = "he"
14
+
15
+
16
+ def get_he_download_links(shared_state, url, mirror, title):
17
+ headers = {
18
+ 'User-Agent': shared_state.values["user_agent"],
19
+ }
20
+
21
+ session = requests.Session()
22
+
23
+ try:
24
+ resp = session.get(url, headers=headers, timeout=30)
25
+ soup = BeautifulSoup(resp.text, 'html.parser')
26
+ except Exception as e:
27
+ info(f"{hostname}: could not fetch release for {title}: {e}")
28
+ return False
29
+
30
+ imdb_id = None
31
+ try:
32
+ imdb_link = soup.find('a', href=re.compile(r"imdb\.com/title/tt\d+", re.IGNORECASE))
33
+ if imdb_link:
34
+ href = imdb_link['href'].strip()
35
+ m = re.search(r"(tt\d{4,7})", href)
36
+ if m:
37
+ imdb_id = m.group(1)
38
+ else:
39
+ debug(f"{hostname}: imdb_id not found for title {title} in link href.")
40
+ else:
41
+ debug(f"{hostname}: imdb_id link href not found for title {title}.")
42
+ except Exception:
43
+ debug(f"{hostname}: failed to extract imdb_id for title {title}.")
44
+
45
+ anchors = []
46
+ for retries in range(10):
47
+ form = soup.find('form', id=re.compile(r'content-protector-access-form'))
48
+ if not form:
49
+ return False
50
+
51
+ action = form.get('action') or url
52
+ action_url = urljoin(resp.url, action)
53
+
54
+ payload = {}
55
+ for inp in form.find_all('input'):
56
+ name = inp.get('name')
57
+ if not name:
58
+ continue
59
+ value = inp.get('value', '')
60
+ payload[name] = value
61
+
62
+ append_patt = re.compile(r"append\(\s*[\'\"](?P<key>[^\'\"]+)[\'\"]\s*,\s*[\'\"](?P<val>[^\'\"]+)[\'\"]\s*\)",
63
+ re.IGNORECASE)
64
+
65
+ for script in soup.find_all('script'):
66
+ txt = script.string if script.string is not None else script.get_text()
67
+ if not txt:
68
+ continue
69
+ for m in append_patt.finditer(txt):
70
+ payload[m.group('key')] = m.group('val')
71
+
72
+ post_headers = headers.copy()
73
+ post_headers.update({'Referer': resp.url})
74
+ try:
75
+ resp = session.post(action_url, data=payload, headers=post_headers, timeout=30)
76
+ soup = BeautifulSoup(resp.text, 'html.parser')
77
+ except Exception as e:
78
+ info(f"{hostname}: could not submit protector form for {title}: {e}")
79
+ break
80
+
81
+ unlocked = soup.select('.content-protector-access-form')
82
+ if unlocked:
83
+ for u in unlocked:
84
+ anchors.extend(u.find_all('a', href=True))
85
+
86
+ if anchors:
87
+ break
88
+
89
+ links = []
90
+ for a in anchors:
91
+ try:
92
+ href = a['href'].strip()
93
+
94
+ netloc = urlparse(href).netloc
95
+ hoster = netloc.split(':')[0].lower()
96
+ parts = hoster.split('.')
97
+ if len(parts) >= 2:
98
+ hoster = parts[-2]
99
+
100
+ links.append([href, hoster])
101
+ except Exception:
102
+ debug(f"{hostname}: could not resolve download link hoster for {title}")
103
+ continue
104
+
105
+ if not links:
106
+ info(f"No external download links found on {hostname} page for {title}")
107
+ return False
108
+
109
+ return {
110
+ "links": links,
111
+ "imdb_id": imdb_id,
112
+ }
@@ -0,0 +1,51 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ import requests
6
+ from bs4 import BeautifulSoup
7
+
8
+ from quasarr.providers.log import info
9
+
10
+ hostname = "nk"
11
+
12
+
13
+ def get_nk_download_links(shared_state, url, mirror, title):
14
+ host = shared_state.values["config"]("Hostnames").get(hostname)
15
+ headers = {
16
+ 'User-Agent': shared_state.values["user_agent"],
17
+ }
18
+
19
+ session = requests.Session()
20
+
21
+ try:
22
+ resp = session.get(url, headers=headers, timeout=20)
23
+ soup = BeautifulSoup(resp.text, 'html.parser')
24
+ except Exception as e:
25
+ info(f"{hostname}: could not fetch release page for {title}: {e}")
26
+ return False
27
+
28
+ anchors = soup.select('a.btn-orange')
29
+ candidates = []
30
+ for a in anchors:
31
+
32
+ href = a.get('href', '').strip()
33
+ hoster = href.split('/')[3].lower()
34
+ if not href.lower().startswith(('http://', 'https://')):
35
+ href = 'https://' + host + href
36
+
37
+ try:
38
+ href = requests.head(href, headers=headers, allow_redirects=True, timeout=20).url
39
+ except Exception as e:
40
+ info(f"{hostname}: could not resolve download link for {title}: {e}")
41
+ continue
42
+
43
+ if hoster == 'ddl.to':
44
+ hoster = 'ddownload'
45
+
46
+ candidates.append([href, hoster])
47
+
48
+ if not candidates:
49
+ info(f"No external download links found on {hostname} page for {title}")
50
+
51
+ return candidates
@@ -9,6 +9,8 @@ dd = '
9
9
  dt = ''
10
10
  dw = ''
11
11
  fx = ''
12
+ nk = ''
13
+ he = ''
12
14
  mb = ''
13
15
  nx = ''
14
16
  sf = ''