quasarr 1.18.0__py3-none-any.whl → 1.20.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of quasarr might be problematic. Click here for more details.

@@ -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, filecrypt_quasarr_helper_user_js
19
+ from quasarr.providers.obfuscated import captcha_js, captcha_values, filecrypt_user_js, junkies_user_js
20
20
  from quasarr.providers.statistics import StatsHelper
21
21
 
22
22
 
@@ -58,7 +58,7 @@ def setup_captcha_routes(app):
58
58
  desired_mirror = None
59
59
 
60
60
  # This is set for circle CAPTCHAs
61
- session = data.get("session", None)
61
+ filecrypt_session = data.get("session", None)
62
62
 
63
63
  # This is required for cutcaptcha
64
64
  rapid = [ln for ln in links if "rapidgator" in ln[1].lower()]
@@ -73,15 +73,21 @@ def setup_captcha_routes(app):
73
73
  "title": title,
74
74
  "password": password,
75
75
  "mirror": desired_mirror,
76
- "session": session,
76
+ "session": filecrypt_session,
77
77
  "links": prioritized_links,
78
78
  "original_url": original_url
79
79
  }
80
80
 
81
81
  encoded_payload = urlsafe_b64encode(json.dumps(payload).encode()).decode()
82
82
 
83
- if session:
84
- debug(f'Session "{session}" found, redirecting to circle CAPTCHA')
83
+ sj = True
84
+ dj = False
85
+
86
+ if sj or dj:
87
+ debug("Redirecting to Junkies CAPTCHA")
88
+ redirect(f"/captcha/junkies?data={quote(encoded_payload)}")
89
+ elif filecrypt_session:
90
+ debug(f'Redirecting to circle CAPTCHA')
85
91
  redirect(f"/captcha/circle?data={quote(encoded_payload)}")
86
92
  else:
87
93
  debug(f"Redirecting to cutcaptcha")
@@ -101,13 +107,140 @@ def setup_captcha_routes(app):
101
107
  except Exception as e:
102
108
  return {"error": f"Failed to decode payload: {str(e)}"}
103
109
 
104
- @app.get('/captcha/quasarr.user.js')
105
- def serve_quasarr_user_js():
106
- content = filecrypt_quasarr_helper_user_js()
110
+ @app.get("/captcha/junkies")
111
+ def serve_junkies_captcha():
112
+ payload = decode_payload()
113
+
114
+ if "error" in payload:
115
+ return render_centered_html(f'''<h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
116
+ <p>{payload["error"]}</p>
117
+ <p>
118
+ {render_button("Back", "secondary", {"onclick": "location.href='/'"})}
119
+ </p>''')
120
+
121
+ package_id = payload.get("package_id")
122
+ title = payload.get("title")
123
+ password = payload.get("password")
124
+ urls = payload.get("links")
125
+ url = urls[0]
126
+
127
+ return render_centered_html(f"""
128
+ <!DOCTYPE html>
129
+ <html>
130
+ <body>
131
+ <h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
132
+ <p><b>Package:</b> {title}</p>
133
+ {render_junkies_section(url, package_id, title, password)}
134
+ <p>
135
+ {render_button("Delete Package", "secondary", {"onclick": f"location.href='/captcha/delete/{package_id}'"})}
136
+ </p>
137
+ <p>
138
+ {render_button("Back", "secondary", {"onclick": "location.href='/'"})}
139
+ </p>
140
+
141
+ </body>
142
+ </html>""")
143
+
144
+ def render_junkies_section(url, package_id, title, password):
145
+ """Render the UI section for SJ and DJ pages"""
146
+
147
+ # Generate userscript URL with transfer params
148
+ # Get base URL of current request
149
+ base_url = request.urlparts.scheme + '://' + request.urlparts.netloc
150
+ transfer_url = f"{base_url}/captcha/quick-transfer"
151
+
152
+ url_with_quick_transfer_params = (
153
+ f"{url}?"
154
+ f"transfer_url={quote(transfer_url)}&"
155
+ f"pkg_id={quote(package_id)}&"
156
+ f"pkg_title={quote(title)}&"
157
+ f"pkg_pass={quote(password)}"
158
+ )
159
+
160
+ return f'''
161
+ <div>
162
+ <!-- One-time setup section - visually separated -->
163
+ <div id="setup-instructions" style="background: #2a2a2a; border: 2px solid #444; border-radius: 8px; padding: 16px; margin-bottom: 24px;">
164
+ <h3 style="margin-top: 0; color: #58a6ff;">First Time Setup:</h3>
165
+ <p style="margin-bottom: 8px;">
166
+ <a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
167
+ </p>
168
+ <p style="margin-top: 0; margin-bottom: 12px;">
169
+ <a href="/captcha/junkies.user.js" target="_blank">2. Install this userscript</a>
170
+ </p>
171
+ <p style="margin-top: 0;">
172
+ <button id="hide-setup-btn" type="button" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
173
+ ✅ Don't show this again
174
+ </button>
175
+ </p>
176
+ </div>
177
+
178
+ <!-- Hidden "show instructions" link -->
179
+ <div id="show-instructions-link" style="display: none; margin-bottom: 16px;">
180
+ <a href="#" id="show-setup-btn" style="color: #58a6ff;">ℹ️ Show instructions again</a>
181
+ </div>
182
+
183
+ <strong><a href="{url_with_quick_transfer_params}" target="_blank">🔗 Obtain the download links here!</a></strong><br><br>
184
+
185
+ <form id="bypass-form" action="/captcha/bypass-submit" method="post" enctype="multipart/form-data">
186
+ <input type="hidden" name="package_id" value="{package_id}" />
187
+ <input type="hidden" name="title" value="{title}" />
188
+ <input type="hidden" name="password" value="{password}" />
189
+
190
+ <div>
191
+ <strong>Paste the download links (one per line):</strong>
192
+ <textarea id="links-input" name="links" rows="5" style="width: 100%; padding: 8px; font-family: monospace; resize: vertical;"></textarea>
193
+ </div>
194
+
195
+ <div>
196
+ {render_button("Submit", "primary", {"type": "submit"})}
197
+ </div>
198
+ </form>
199
+ </div>
200
+ <script>
201
+ // Handle setup instructions hide/show
202
+ const hideSetup = localStorage.getItem('hideJunkiesSetupInstructions');
203
+ const setupBox = document.getElementById('setup-instructions');
204
+ const showLink = document.getElementById('show-instructions-link');
205
+
206
+ if (hideSetup === 'true') {{
207
+ setupBox.style.display = 'none';
208
+ showLink.style.display = 'block';
209
+ }}
210
+
211
+ // Hide setup instructions
212
+ document.getElementById('hide-setup-btn').addEventListener('click', function() {{
213
+ localStorage.setItem('hideJunkiesSetupInstructions', 'true');
214
+ setupBox.style.display = 'none';
215
+ showLink.style.display = 'block';
216
+ }});
217
+
218
+ // Show setup instructions again
219
+ document.getElementById('show-setup-btn').addEventListener('click', function(e) {{
220
+ e.preventDefault();
221
+ localStorage.setItem('hideJunkiesSetupInstructions', 'false');
222
+ setupBox.style.display = 'block';
223
+ showLink.style.display = 'none';
224
+ }});
225
+ </script>
226
+ '''
227
+
228
+ @app.get('/captcha/junkies.user.js')
229
+ def serve_junkies_user_js():
230
+ sj = shared_state.values["config"]("Hostnames").get("sj")
231
+ dj = shared_state.values["config"]("Hostnames").get("dj")
232
+
233
+ content = junkies_user_js(sj, dj)
234
+ response.content_type = 'application/javascript'
235
+ return content
236
+
237
+ @app.get('/captcha/filecrypt.user.js')
238
+ def serve_filecrypt_user_js():
239
+ content = filecrypt_user_js()
107
240
  response.content_type = 'application/javascript'
108
241
  return content
109
242
 
110
- def render_bypass_section(url, package_id, title, password):
243
+ def render_filecrypt_bypass_section(url, package_id, title, password):
111
244
  """Render the bypass UI section for both cutcaptcha and circle captcha pages"""
112
245
 
113
246
  # Generate userscript URL with transfer params
@@ -135,7 +268,7 @@ def setup_captcha_routes(app):
135
268
  <a href="https://www.tampermonkey.net/" target="_blank" rel="noopener noreferrer">1. Install Tampermonkey</a>
136
269
  </p>
137
270
  <p style="margin-top: 0; margin-bottom: 12px;">
138
- <a href="/captcha/quasarr.user.js" target="_blank">2. Install this userscript</a>
271
+ <a href="/captcha/filecrypt.user.js" target="_blank">2. Install this userscript</a>
139
272
  </p>
140
273
  <p style="margin-top: 0;">
141
274
  <button id="hide-setup-btn" type="button" style="background: #444; color: #fff; border: 1px solid #666; padding: 6px 12px; border-radius: 4px; cursor: pointer;">
@@ -146,7 +279,7 @@ def setup_captcha_routes(app):
146
279
 
147
280
  <!-- Hidden "show instructions" link -->
148
281
  <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>
282
+ <a href="#" id="show-setup-btn" style="color: #58a6ff;">ℹ️ Show instructions again</a>
150
283
  </div>
151
284
 
152
285
  <strong><a href="{url_with_quick_transfer_params}" target="_blank">🔗 Obtain the download links here!</a></strong><br><br>
@@ -188,7 +321,7 @@ def setup_captcha_routes(app):
188
321
  }}
189
322
 
190
323
  // Handle setup instructions hide/show
191
- const hideSetup = localStorage.getItem('hideSetupInstructions');
324
+ const hideSetup = localStorage.getItem('hideFileCryptSetupInstructions');
192
325
  const setupBox = document.getElementById('setup-instructions');
193
326
  const showLink = document.getElementById('show-instructions-link');
194
327
 
@@ -199,7 +332,7 @@ def setup_captcha_routes(app):
199
332
 
200
333
  // Hide setup instructions
201
334
  document.getElementById('hide-setup-btn').addEventListener('click', function() {{
202
- localStorage.setItem('hideSetupInstructions', 'true');
335
+ localStorage.setItem('hideFileCryptSetupInstructions', 'true');
203
336
  setupBox.style.display = 'none';
204
337
  showLink.style.display = 'block';
205
338
  }});
@@ -207,7 +340,7 @@ def setup_captcha_routes(app):
207
340
  // Show setup instructions again
208
341
  document.getElementById('show-setup-btn').addEventListener('click', function(e) {{
209
342
  e.preventDefault();
210
- localStorage.setItem('hideSetupInstructions', 'false');
343
+ localStorage.setItem('hideFileCryptSetupInstructions', 'false');
211
344
  setupBox.style.display = 'block';
212
345
  showLink.style.display = 'none';
213
346
  }});
@@ -426,7 +559,7 @@ def setup_captcha_routes(app):
426
559
  url = prioritized_links[0][0]
427
560
 
428
561
  # Add bypass section
429
- bypass_section = render_bypass_section(url, package_id, title, password)
562
+ bypass_section = render_filecrypt_bypass_section(url, package_id, title, password)
430
563
 
431
564
  content = render_centered_html(r'''
432
565
  <script type="text/javascript">
@@ -607,6 +740,14 @@ def setup_captcha_routes(app):
607
740
  {render_button("Back", "secondary", {"onclick": "location.href='/captcha'"})}
608
741
  </p>''')
609
742
 
743
+ package_exists = shared_state.get_db("protected").retrieve(package_id)
744
+ if not package_exists:
745
+ return render_centered_html(f'''<h1><img src="{images.logo}" type="image/png" alt="Quasarr logo" class="logo"/>Quasarr</h1>
746
+ <p><b>Error:</b> Package not found or already solved.</p>
747
+ <p>
748
+ {render_button("Back", "secondary", {"onclick": "location.href='/captcha'"})}
749
+ </p>''')
750
+
610
751
  # Process links input
611
752
  if links_input:
612
753
  info(f"Processing direct links bypass for {title}")
@@ -787,7 +928,7 @@ def setup_captcha_routes(app):
787
928
  return "Missing required parameters"
788
929
 
789
930
  # Add bypass section
790
- bypass_section = render_bypass_section(original_url, package_id, title, password)
931
+ bypass_section = render_filecrypt_bypass_section(original_url, package_id, title, password)
791
932
 
792
933
  return render_centered_html(f"""
793
934
  <!DOCTYPE html>
@@ -11,12 +11,15 @@ from quasarr.downloads.linkcrypters.hide import decrypt_links_if_hide
11
11
  from quasarr.downloads.sources.al import get_al_download_links
12
12
  from quasarr.downloads.sources.by import get_by_download_links
13
13
  from quasarr.downloads.sources.dd import get_dd_download_links
14
+ from quasarr.downloads.sources.dj import get_dj_download_links
14
15
  from quasarr.downloads.sources.dt import get_dt_download_links
15
16
  from quasarr.downloads.sources.dw import get_dw_download_links
17
+ from quasarr.downloads.sources.he import get_he_download_links
16
18
  from quasarr.downloads.sources.mb import get_mb_download_links
17
19
  from quasarr.downloads.sources.nk import get_nk_download_links
18
20
  from quasarr.downloads.sources.nx import get_nx_download_links
19
21
  from quasarr.downloads.sources.sf import get_sf_download_links, resolve_sf_redirect
22
+ from quasarr.downloads.sources.sj import get_sj_download_links
20
23
  from quasarr.downloads.sources.sl import get_sl_download_links
21
24
  from quasarr.downloads.sources.wd import get_wd_download_links
22
25
  from quasarr.providers.log import info
@@ -200,12 +203,15 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
200
203
  'AL': config.get("al"),
201
204
  'BY': config.get("by"),
202
205
  'DD': config.get("dd"),
206
+ 'DJ': config.get("dj"),
203
207
  'DT': config.get("dt"),
204
208
  'DW': config.get("dw"),
209
+ 'HE': config.get("he"),
205
210
  'MB': config.get("mb"),
206
211
  'NK': config.get("nk"),
207
212
  'NX': config.get("nx"),
208
213
  'SF': config.get("sf"),
214
+ 'SJ': config.get("sj"),
209
215
  'SL': config.get("sl"),
210
216
  'WD': config.get("wd")
211
217
  }
@@ -214,12 +220,15 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
214
220
  (flags['AL'], handle_al),
215
221
  (flags['BY'], handle_by),
216
222
  (flags['DD'], lambda *a: handle_unprotected(*a, func=get_dd_download_links, label='DD')),
223
+ (flags['DJ'], lambda *a: handle_protected(*a, func=get_dj_download_links, label='DJ')),
217
224
  (flags['DT'], lambda *a: handle_unprotected(*a, func=get_dt_download_links, label='DT')),
218
225
  (flags['DW'], lambda *a: handle_protected(*a, func=get_dw_download_links, label='DW')),
226
+ (flags['HE'], lambda *a: handle_unprotected(*a, func=get_he_download_links, label='HE')),
219
227
  (flags['MB'], lambda *a: handle_protected(*a, func=get_mb_download_links, label='MB')),
220
228
  (flags['NK'], lambda *a: handle_protected(*a, func=get_nk_download_links, label='NK')),
221
229
  (flags['NX'], lambda *a: handle_unprotected(*a, func=get_nx_download_links, label='NX')),
222
230
  (flags['SF'], handle_sf),
231
+ (flags['SJ'], lambda *a: handle_protected(*a, func=get_sj_download_links, label='SJ')),
223
232
  (flags['SL'], handle_sl),
224
233
  (flags['WD'], handle_wd),
225
234
  ]
@@ -0,0 +1,7 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+
6
+ def get_dj_download_links(shared_state, url, mirror, title): # signature must align with other download link functions!
7
+ return [url]
@@ -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
+ }
@@ -2,13 +2,10 @@
2
2
  # Quasarr
3
3
  # Project by https://github.com/rix1337
4
4
 
5
- import re
6
-
7
5
  import requests
8
6
  from bs4 import BeautifulSoup
9
7
 
10
- from quasarr.providers.log import info, debug
11
- from urllib.parse import urlparse, urljoin
8
+ from quasarr.providers.log import info
12
9
 
13
10
  hostname = "nk"
14
11
 
@@ -28,15 +25,14 @@ def get_nk_download_links(shared_state, url, mirror, title):
28
25
  info(f"{hostname}: could not fetch release page for {title}: {e}")
29
26
  return False
30
27
 
31
- # download links are provided as anchors with class 'dl-button'
32
28
  anchors = soup.select('a.btn-orange')
33
29
  candidates = []
34
30
  for a in anchors:
35
-
31
+
36
32
  href = a.get('href', '').strip()
37
33
  hoster = href.split('/')[3].lower()
38
34
  if not href.lower().startswith(('http://', 'https://')):
39
- href = 'https://' + host + href
35
+ href = 'https://' + host + href
40
36
 
41
37
  try:
42
38
  href = requests.head(href, headers=headers, allow_redirects=True, timeout=20).url
@@ -0,0 +1,7 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+
6
+ def get_sj_download_links(shared_state, url, mirror, title): # signature must align with other download link functions!
7
+ return [url]
@@ -6,12 +6,15 @@ logo = '
6
6
  al = ''
7
7
  by = ''
8
8
  dd = ''
9
+ dj = ''
9
10
  dt = ''
10
11
  dw = ''
11
12
  fx = ''
12
13
  nk = ''
14
+ he = ''
13
15
  mb = ''
14
16
  nx = ''
15
17
  sf = ''
18
+ sj = ''
16
19
  sl = ''
17
20
  wd = ''
@@ -159,7 +159,7 @@ def render_centered_html(inner_content):
159
159
  text-decoration: none;
160
160
  }
161
161
  a:hover {
162
- text-decoration: underline;
162
+
163
163
  }
164
164
  /* footer styling */
165
165
  footer {