quasarr 2.1.5__py3-none-any.whl → 2.2.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.

Files changed (57) hide show
  1. quasarr/api/__init__.py +94 -23
  2. quasarr/api/captcha/__init__.py +0 -12
  3. quasarr/api/config/__init__.py +22 -11
  4. quasarr/api/packages/__init__.py +26 -34
  5. quasarr/api/statistics/__init__.py +15 -15
  6. quasarr/downloads/__init__.py +9 -1
  7. quasarr/downloads/packages/__init__.py +2 -2
  8. quasarr/downloads/sources/al.py +6 -0
  9. quasarr/downloads/sources/by.py +29 -20
  10. quasarr/downloads/sources/dd.py +9 -1
  11. quasarr/downloads/sources/dl.py +3 -0
  12. quasarr/downloads/sources/dt.py +16 -7
  13. quasarr/downloads/sources/dw.py +22 -17
  14. quasarr/downloads/sources/he.py +11 -6
  15. quasarr/downloads/sources/mb.py +9 -3
  16. quasarr/downloads/sources/nk.py +9 -3
  17. quasarr/downloads/sources/nx.py +21 -17
  18. quasarr/downloads/sources/sf.py +21 -13
  19. quasarr/downloads/sources/sl.py +10 -2
  20. quasarr/downloads/sources/wd.py +18 -9
  21. quasarr/downloads/sources/wx.py +7 -11
  22. quasarr/providers/auth.py +1 -1
  23. quasarr/providers/cloudflare.py +1 -1
  24. quasarr/providers/hostname_issues.py +63 -0
  25. quasarr/providers/html_images.py +1 -18
  26. quasarr/providers/html_templates.py +104 -12
  27. quasarr/providers/obfuscated.py +11 -11
  28. quasarr/providers/sessions/al.py +27 -11
  29. quasarr/providers/sessions/dd.py +12 -4
  30. quasarr/providers/sessions/dl.py +19 -11
  31. quasarr/providers/sessions/nx.py +12 -4
  32. quasarr/providers/version.py +1 -1
  33. quasarr/search/sources/al.py +12 -1
  34. quasarr/search/sources/by.py +15 -4
  35. quasarr/search/sources/dd.py +22 -3
  36. quasarr/search/sources/dj.py +12 -1
  37. quasarr/search/sources/dl.py +12 -6
  38. quasarr/search/sources/dt.py +17 -4
  39. quasarr/search/sources/dw.py +15 -4
  40. quasarr/search/sources/fx.py +19 -6
  41. quasarr/search/sources/he.py +15 -2
  42. quasarr/search/sources/mb.py +15 -4
  43. quasarr/search/sources/nk.py +15 -2
  44. quasarr/search/sources/nx.py +15 -4
  45. quasarr/search/sources/sf.py +25 -8
  46. quasarr/search/sources/sj.py +14 -1
  47. quasarr/search/sources/sl.py +17 -2
  48. quasarr/search/sources/wd.py +15 -4
  49. quasarr/search/sources/wx.py +16 -18
  50. quasarr/storage/setup.py +150 -35
  51. {quasarr-2.1.5.dist-info → quasarr-2.2.0.dist-info}/METADATA +6 -3
  52. quasarr-2.2.0.dist-info/RECORD +82 -0
  53. {quasarr-2.1.5.dist-info → quasarr-2.2.0.dist-info}/WHEEL +1 -1
  54. quasarr-2.1.5.dist-info/RECORD +0 -81
  55. {quasarr-2.1.5.dist-info → quasarr-2.2.0.dist-info}/entry_points.txt +0 -0
  56. {quasarr-2.1.5.dist-info → quasarr-2.2.0.dist-info}/licenses/LICENSE +0 -0
  57. {quasarr-2.1.5.dist-info → quasarr-2.2.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,63 @@
1
+ # -*- coding: utf-8 -*-
2
+ # Quasarr
3
+ # Project by https://github.com/rix1337
4
+
5
+ """
6
+ Hostname Issues Tracker - Uses lazy imports to avoid circular dependency
7
+ """
8
+
9
+ import json
10
+ from datetime import datetime
11
+
12
+
13
+ def _get_db(table_name):
14
+ """Lazy import to avoid circular dependency."""
15
+ from quasarr.storage.sqlite_database import DataBase
16
+ return DataBase(table_name)
17
+
18
+
19
+ def mark_hostname_issue(shorthand, operation, error_message):
20
+ shorthand = shorthand.lower()
21
+ db = _get_db("hostname_issues")
22
+
23
+ issue_data = {
24
+ "operation": operation,
25
+ "error": str(error_message)[:500],
26
+ "timestamp": datetime.now().isoformat()
27
+ }
28
+
29
+ db.update_store(shorthand, json.dumps(issue_data))
30
+
31
+
32
+ def clear_hostname_issue(shorthand):
33
+ shorthand = shorthand.lower()
34
+ db = _get_db("hostname_issues")
35
+ db.delete(shorthand)
36
+
37
+
38
+ def get_hostname_issue(shorthand):
39
+ shorthand = shorthand.lower()
40
+ db = _get_db("hostname_issues")
41
+ data = db.retrieve(shorthand)
42
+
43
+ if data:
44
+ try:
45
+ return json.loads(data)
46
+ except json.JSONDecodeError:
47
+ return None
48
+ return None
49
+
50
+
51
+ def get_all_hostname_issues():
52
+ db = _get_db("hostname_issues")
53
+ all_data = db.retrieve_all_titles()
54
+
55
+ issues = {}
56
+ if all_data:
57
+ for shorthand, data in all_data:
58
+ try:
59
+ issues[shorthand] = json.loads(data)
60
+ except json.JSONDecodeError:
61
+ continue
62
+
63
+ return issues
@@ -2,21 +2,4 @@
2
2
  # Quasarr
3
3
  # Project by https://github.com/rix1337
4
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 = ''
5
+ logo = ''
@@ -195,18 +195,7 @@ def render_centered_html(inner_content, footer_content=""):
195
195
  padding: calc(var(--spacing) * 2);
196
196
  text-align: center;
197
197
  width: 100%;
198
- max-width: fit-content;
199
- }
200
- /* No padding on the sides for captcha view on small screens */
201
- @media (max-width: 600px) {
202
- body:has(iframe) .outer {
203
- padding-left: 0;
204
- padding-right: 0;
205
- }
206
- body:has(iframe) .inner {
207
- padding-left: 0;
208
- padding-right: 0;
209
- }
198
+ max-width: 600px;
210
199
  }
211
200
  h2 {
212
201
  margin-top: var(--spacing);
@@ -235,6 +224,13 @@ def render_centered_html(inner_content, footer_content=""):
235
224
  .captcha-container {
236
225
  background-color: var(--secondary);
237
226
  }
227
+ /* Responsive scaling for fixed-width CAPTCHA iframe (370px) */
228
+ /* PROBLEM: The drag and drop breaks */
229
+ @media (max-width: 400px) {
230
+ #puzzle-captcha {
231
+ zoom: 0.75;
232
+ }
233
+ }
238
234
  button {
239
235
  padding: 0.5rem 1rem;
240
236
  font-size: 1rem;
@@ -285,6 +281,54 @@ def render_centered_html(inner_content, footer_content=""):
285
281
  footer a:hover {
286
282
  color: var(--fg-color);
287
283
  }
284
+ /* Global Modal Styles */
285
+ .status-modal-overlay {
286
+ position: fixed;
287
+ top: 0;
288
+ left: 0;
289
+ right: 0;
290
+ bottom: 0;
291
+ background: rgba(0, 0, 0, 0.5);
292
+ display: flex;
293
+ align-items: center;
294
+ justify-content: center;
295
+ z-index: 9999;
296
+ visibility: hidden;
297
+ opacity: 0;
298
+ transition: visibility 0s, opacity 0.2s;
299
+ }
300
+ .status-modal-overlay.active {
301
+ visibility: visible;
302
+ opacity: 1;
303
+ }
304
+ .status-modal {
305
+ background: var(--card-bg);
306
+ border-radius: 0.5rem;
307
+ padding: 1.5rem;
308
+ max-width: 400px;
309
+ width: 90%;
310
+ box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15);
311
+ transform: scale(0.9);
312
+ transition: transform 0.2s;
313
+ }
314
+ .status-modal-overlay.active .status-modal {
315
+ transform: scale(1);
316
+ }
317
+ .status-modal h3 {
318
+ margin: 0 0 1rem 0;
319
+ display: flex;
320
+ align-items: center;
321
+ gap: 0.5rem;
322
+ }
323
+ .status-modal p {
324
+ margin: 0 0 1rem 0;
325
+ word-break: break-word;
326
+ }
327
+ .status-modal .btn-row {
328
+ display: flex;
329
+ gap: 0.5rem;
330
+ justify-content: flex-end;
331
+ }
288
332
  </style>
289
333
  </head>'''
290
334
 
@@ -295,6 +339,53 @@ def render_centered_html(inner_content, footer_content=""):
295
339
  else:
296
340
  footer_html = version_text
297
341
 
342
+ # Global modal script
343
+ modal_script = '''
344
+ <script>
345
+ function showModal(title, content, buttonsHtml) {
346
+ let overlay = document.getElementById('global-modal-overlay');
347
+ if (!overlay) {
348
+ overlay = document.createElement('div');
349
+ overlay.id = 'global-modal-overlay';
350
+ overlay.className = 'status-modal-overlay';
351
+ overlay.innerHTML = `
352
+ <div class="status-modal">
353
+ <h3 id="global-modal-title"></h3>
354
+ <div id="global-modal-content"></div>
355
+ <div class="btn-row" id="global-modal-buttons"></div>
356
+ </div>
357
+ `;
358
+ document.body.appendChild(overlay);
359
+
360
+ overlay.addEventListener('click', function(e) {
361
+ if (e.target === overlay) closeModal();
362
+ });
363
+ }
364
+
365
+ document.getElementById('global-modal-title').innerHTML = title;
366
+ document.getElementById('global-modal-content').innerHTML = content;
367
+
368
+ let btns = buttonsHtml || '<button class="btn-secondary" onclick="closeModal()">Close</button>';
369
+ document.getElementById('global-modal-buttons').innerHTML = btns;
370
+
371
+ // Small timeout to allow CSS transition
372
+ requestAnimationFrame(() => {
373
+ overlay.classList.add('active');
374
+ });
375
+ }
376
+
377
+ function closeModal() {
378
+ const overlay = document.getElementById('global-modal-overlay');
379
+ if (overlay) {
380
+ overlay.classList.remove('active');
381
+ setTimeout(() => {
382
+ if (overlay.parentNode) overlay.parentNode.removeChild(overlay);
383
+ }, 200);
384
+ }
385
+ }
386
+ </script>
387
+ '''
388
+
298
389
  body = f'''
299
390
  {head}
300
391
  <body>
@@ -306,6 +397,7 @@ def render_centered_html(inner_content, footer_content=""):
306
397
  <footer>
307
398
  {footer_html}
308
399
  </footer>
400
+ {modal_script}
309
401
  </body>
310
402
  '''
311
403
  return f'<html>{body}</html>'