quasarr 1.18.0__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.
- {quasarr-1.18.0 → quasarr-1.19.0}/PKG-INFO +1 -1
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/__init__.py +3 -0
- quasarr-1.19.0/quasarr/downloads/sources/he.py +112 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/nk.py +3 -7
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/html_images.py +1 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/version.py +1 -1
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/__init__.py +4 -0
- quasarr-1.19.0/quasarr/search/sources/he.py +196 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/nk.py +6 -7
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/storage/config.py +1 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/PKG-INFO +1 -1
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/SOURCES.txt +2 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/LICENSE +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/README.md +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/arr/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/captcha/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/config/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/sponsors_helper/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/api/statistics/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/al.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/filecrypt.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/linkcrypters/hide.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/packages/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/al.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/by.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/dd.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/dt.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/dw.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/mb.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/nx.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/sf.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/sl.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/downloads/sources/wd.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/cloudflare.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/html_templates.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/imdb_metadata.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/log.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/myjd_api.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/notifications.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/obfuscated.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/sessions/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/sessions/al.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/sessions/dd.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/sessions/nx.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/shared_state.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/statistics.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/providers/web_server.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/al.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/by.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/dd.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/dt.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/dw.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/fx.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/mb.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/nx.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/sf.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/sl.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/search/sources/wd.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/storage/__init__.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/storage/setup.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr/storage/sqlite_database.py +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/dependency_links.txt +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/entry_points.txt +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/not-zip-safe +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/requires.txt +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/quasarr.egg-info/top_level.txt +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/setup.cfg +0 -0
- {quasarr-1.18.0 → quasarr-1.19.0}/setup.py +0 -0
|
@@ -13,6 +13,7 @@ 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
|
|
17
18
|
from quasarr.downloads.sources.nk import get_nk_download_links
|
|
18
19
|
from quasarr.downloads.sources.nx import get_nx_download_links
|
|
@@ -202,6 +203,7 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
|
|
|
202
203
|
'DD': config.get("dd"),
|
|
203
204
|
'DT': config.get("dt"),
|
|
204
205
|
'DW': config.get("dw"),
|
|
206
|
+
'HE': config.get("he"),
|
|
205
207
|
'MB': config.get("mb"),
|
|
206
208
|
'NK': config.get("nk"),
|
|
207
209
|
'NX': config.get("nx"),
|
|
@@ -216,6 +218,7 @@ def download(shared_state, request_from, title, url, mirror, size_mb, password,
|
|
|
216
218
|
(flags['DD'], lambda *a: handle_unprotected(*a, func=get_dd_download_links, label='DD')),
|
|
217
219
|
(flags['DT'], lambda *a: handle_unprotected(*a, func=get_dt_download_links, label='DT')),
|
|
218
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')),
|
|
219
222
|
(flags['MB'], lambda *a: handle_protected(*a, func=get_mb_download_links, label='MB')),
|
|
220
223
|
(flags['NK'], lambda *a: handle_protected(*a, func=get_nk_download_links, label='NK')),
|
|
221
224
|
(flags['NX'], lambda *a: handle_unprotected(*a, func=get_nx_download_links, label='NX')),
|
|
@@ -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
|
|
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
|
|
35
|
+
href = 'https://' + host + href
|
|
40
36
|
|
|
41
37
|
try:
|
|
42
38
|
href = requests.head(href, headers=headers, allow_redirects=True, timeout=20).url
|
|
@@ -10,6 +10,7 @@ dt = '
|
|
|
10
10
|
dw = ''
|
|
11
11
|
fx = ''
|
|
12
12
|
nk = ''
|
|
13
|
+
he = ''
|
|
13
14
|
mb = ''
|
|
14
15
|
nx = ''
|
|
15
16
|
sf = ''
|
|
@@ -12,6 +12,7 @@ from quasarr.search.sources.dd import dd_search, dd_feed
|
|
|
12
12
|
from quasarr.search.sources.dt import dt_feed, dt_search
|
|
13
13
|
from quasarr.search.sources.dw import dw_feed, dw_search
|
|
14
14
|
from quasarr.search.sources.fx import fx_feed, fx_search
|
|
15
|
+
from quasarr.search.sources.he import he_feed, he_search
|
|
15
16
|
from quasarr.search.sources.mb import mb_feed, mb_search
|
|
16
17
|
from quasarr.search.sources.nk import nk_feed, nk_search
|
|
17
18
|
from quasarr.search.sources.nx import nx_feed, nx_search
|
|
@@ -34,6 +35,7 @@ def get_search_results(shared_state, request_from, imdb_id="", search_phrase="",
|
|
|
34
35
|
dt = shared_state.values["config"]("Hostnames").get("dt")
|
|
35
36
|
dw = shared_state.values["config"]("Hostnames").get("dw")
|
|
36
37
|
fx = shared_state.values["config"]("Hostnames").get("fx")
|
|
38
|
+
he = shared_state.values["config"]("Hostnames").get("he")
|
|
37
39
|
mb = shared_state.values["config"]("Hostnames").get("mb")
|
|
38
40
|
nk = shared_state.values["config"]("Hostnames").get("nk")
|
|
39
41
|
nx = shared_state.values["config"]("Hostnames").get("nx")
|
|
@@ -53,6 +55,7 @@ def get_search_results(shared_state, request_from, imdb_id="", search_phrase="",
|
|
|
53
55
|
(dt, dt_search),
|
|
54
56
|
(dw, dw_search),
|
|
55
57
|
(fx, fx_search),
|
|
58
|
+
(he, he_search),
|
|
56
59
|
(mb, mb_search),
|
|
57
60
|
(nk, nk_search),
|
|
58
61
|
(nx, nx_search),
|
|
@@ -78,6 +81,7 @@ def get_search_results(shared_state, request_from, imdb_id="", search_phrase="",
|
|
|
78
81
|
(dt, dt_feed),
|
|
79
82
|
(dw, dw_feed),
|
|
80
83
|
(fx, fx_feed),
|
|
84
|
+
(he, he_feed),
|
|
81
85
|
(mb, mb_feed),
|
|
82
86
|
(nk, nk_feed),
|
|
83
87
|
(nx, nx_feed),
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
# -*- coding: utf-8 -*-
|
|
2
|
+
# Quasarr
|
|
3
|
+
# Project by https://github.com/rix1337
|
|
4
|
+
|
|
5
|
+
import re
|
|
6
|
+
import time
|
|
7
|
+
from base64 import urlsafe_b64encode
|
|
8
|
+
from datetime import datetime, timedelta
|
|
9
|
+
from html import unescape
|
|
10
|
+
|
|
11
|
+
import requests
|
|
12
|
+
from bs4 import BeautifulSoup
|
|
13
|
+
|
|
14
|
+
from quasarr.providers.imdb_metadata import get_localized_title
|
|
15
|
+
from quasarr.providers.log import info, debug
|
|
16
|
+
|
|
17
|
+
hostname = "he"
|
|
18
|
+
supported_mirrors = ["rapidgator", "nitroflare"]
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
def parse_posted_ago(txt):
|
|
22
|
+
try:
|
|
23
|
+
m = re.search(r"(\d+)\s*(sec|min|hour|day|week|month|year)s?", txt, re.IGNORECASE)
|
|
24
|
+
if not m:
|
|
25
|
+
return ''
|
|
26
|
+
value = int(m.group(1))
|
|
27
|
+
unit = m.group(2).lower()
|
|
28
|
+
now = datetime.utcnow()
|
|
29
|
+
if unit.startswith('sec'):
|
|
30
|
+
delta = timedelta(seconds=value)
|
|
31
|
+
elif unit.startswith('min'):
|
|
32
|
+
delta = timedelta(minutes=value)
|
|
33
|
+
elif unit.startswith('hour'):
|
|
34
|
+
delta = timedelta(hours=value)
|
|
35
|
+
elif unit.startswith('day'):
|
|
36
|
+
delta = timedelta(days=value)
|
|
37
|
+
elif unit.startswith('week'):
|
|
38
|
+
delta = timedelta(weeks=value)
|
|
39
|
+
elif unit.startswith('month'):
|
|
40
|
+
delta = timedelta(days=30 * value)
|
|
41
|
+
else:
|
|
42
|
+
delta = timedelta(days=365 * value)
|
|
43
|
+
return (datetime.utcnow() - delta).strftime("%a, %d %b %Y %H:%M:%S +0000")
|
|
44
|
+
except Exception:
|
|
45
|
+
return ''
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
def extract_size(text: str) -> dict:
|
|
49
|
+
match = re.search(r"(\d+(?:[\.,]\d+)?)\s*([A-Za-z]+)", text)
|
|
50
|
+
if match:
|
|
51
|
+
size = match.group(1).replace(',', '.')
|
|
52
|
+
unit = match.group(2)
|
|
53
|
+
return {"size": size, "sizeunit": unit}
|
|
54
|
+
return {"size": "0", "sizeunit": "MB"}
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def he_feed(*args, **kwargs):
|
|
58
|
+
return he_search(*args, **kwargs)
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
def he_search(shared_state, start_time, request_from, search_string="", mirror=None, season=None, episode=None):
|
|
62
|
+
releases = []
|
|
63
|
+
host = shared_state.values["config"]("Hostnames").get(hostname)
|
|
64
|
+
|
|
65
|
+
if not "arr" in request_from.lower():
|
|
66
|
+
debug(f'Skipping {request_from} search on "{hostname.upper()}" (unsupported media type)!')
|
|
67
|
+
return releases
|
|
68
|
+
|
|
69
|
+
if "radarr" in request_from.lower():
|
|
70
|
+
tag = "movies"
|
|
71
|
+
else:
|
|
72
|
+
tag = "tv-shows"
|
|
73
|
+
|
|
74
|
+
if mirror and mirror not in supported_mirrors:
|
|
75
|
+
debug(f'Mirror "{mirror}" not supported by {hostname}.')
|
|
76
|
+
return releases
|
|
77
|
+
|
|
78
|
+
source_search = ""
|
|
79
|
+
if search_string != "":
|
|
80
|
+
imdb_id = shared_state.is_imdb_id(search_string)
|
|
81
|
+
if imdb_id:
|
|
82
|
+
local_title = get_localized_title(shared_state, imdb_id, 'en')
|
|
83
|
+
if not local_title:
|
|
84
|
+
info(f"{hostname}: no title for IMDb {imdb_id}")
|
|
85
|
+
return releases
|
|
86
|
+
source_search = local_title
|
|
87
|
+
else:
|
|
88
|
+
return releases
|
|
89
|
+
source_search = unescape(source_search)
|
|
90
|
+
else:
|
|
91
|
+
imdb_id = None
|
|
92
|
+
|
|
93
|
+
url = f'https://{host}/tag/{tag}/'
|
|
94
|
+
|
|
95
|
+
headers = {"User-Agent": shared_state.values["user_agent"]}
|
|
96
|
+
params = {"s": source_search}
|
|
97
|
+
|
|
98
|
+
try:
|
|
99
|
+
r = requests.get(url, headers=headers, params=params, timeout=10)
|
|
100
|
+
soup = BeautifulSoup(r.content, 'html.parser')
|
|
101
|
+
results = soup.find_all('div', class_='item')
|
|
102
|
+
except Exception as e:
|
|
103
|
+
info(f"{hostname}: search load error: {e}")
|
|
104
|
+
return releases
|
|
105
|
+
|
|
106
|
+
if not results:
|
|
107
|
+
return releases
|
|
108
|
+
|
|
109
|
+
for result in results:
|
|
110
|
+
try:
|
|
111
|
+
data = result.find('div', class_='data')
|
|
112
|
+
if not data:
|
|
113
|
+
continue
|
|
114
|
+
|
|
115
|
+
headline = data.find('h5')
|
|
116
|
+
if not headline:
|
|
117
|
+
continue
|
|
118
|
+
|
|
119
|
+
a = headline.find('a', href=True)
|
|
120
|
+
if not a:
|
|
121
|
+
continue
|
|
122
|
+
|
|
123
|
+
source = a['href'].strip()
|
|
124
|
+
|
|
125
|
+
head_title = a.get_text(strip=True)
|
|
126
|
+
if not head_title:
|
|
127
|
+
continue
|
|
128
|
+
|
|
129
|
+
head_split = head_title.split(" – ")
|
|
130
|
+
title = head_split[0].strip()
|
|
131
|
+
|
|
132
|
+
if not shared_state.is_valid_release(title, request_from, search_string, season, episode):
|
|
133
|
+
continue
|
|
134
|
+
|
|
135
|
+
size_item = extract_size(head_split[1].strip())
|
|
136
|
+
mb = shared_state.convert_to_mb(size_item)
|
|
137
|
+
|
|
138
|
+
size = mb * 1024 * 1024
|
|
139
|
+
|
|
140
|
+
published = None
|
|
141
|
+
p_meta = data.find('p', class_='meta')
|
|
142
|
+
if p_meta:
|
|
143
|
+
posted_span = None
|
|
144
|
+
for sp in p_meta.find_all('span'):
|
|
145
|
+
txt = sp.get_text(' ', strip=True)
|
|
146
|
+
if txt.lower().startswith('posted') or 'ago' in txt.lower():
|
|
147
|
+
posted_span = txt
|
|
148
|
+
break
|
|
149
|
+
|
|
150
|
+
if posted_span:
|
|
151
|
+
published = parse_posted_ago(posted_span)
|
|
152
|
+
|
|
153
|
+
if published is None:
|
|
154
|
+
continue
|
|
155
|
+
|
|
156
|
+
release_imdb_id = None
|
|
157
|
+
try:
|
|
158
|
+
r = requests.get(source, headers=headers, timeout=10)
|
|
159
|
+
soup = BeautifulSoup(r.content, 'html.parser')
|
|
160
|
+
imdb_link = soup.find('a', href=re.compile(r"imdb\.com/title/tt\d+", re.IGNORECASE))
|
|
161
|
+
if imdb_link:
|
|
162
|
+
release_imdb_id = re.search(r'tt\d+', imdb_link['href']).group()
|
|
163
|
+
if imdb_id and release_imdb_id != imdb_id:
|
|
164
|
+
debug(f"{hostname}: IMDb ID mismatch: expected {imdb_id}, found {release_imdb_id}")
|
|
165
|
+
continue
|
|
166
|
+
else:
|
|
167
|
+
debug(f"{hostname}: imdb link not found for title {title}")
|
|
168
|
+
except Exception as e:
|
|
169
|
+
debug(f"{hostname}: failed to determine imdb_id for title {title}")
|
|
170
|
+
continue
|
|
171
|
+
|
|
172
|
+
password = None
|
|
173
|
+
payload = urlsafe_b64encode(
|
|
174
|
+
f"{title}|{source}|{mirror}|{mb}|{password}|{release_imdb_id}".encode("utf-8")).decode()
|
|
175
|
+
link = f"{shared_state.values['internal_address']}/download/?payload={payload}"
|
|
176
|
+
|
|
177
|
+
releases.append({
|
|
178
|
+
"details": {
|
|
179
|
+
"title": title,
|
|
180
|
+
"hostname": hostname,
|
|
181
|
+
"imdb_id": release_imdb_id,
|
|
182
|
+
"link": link,
|
|
183
|
+
"mirror": mirror,
|
|
184
|
+
"size": size,
|
|
185
|
+
"date": published,
|
|
186
|
+
"source": source
|
|
187
|
+
},
|
|
188
|
+
"type": "protected"
|
|
189
|
+
})
|
|
190
|
+
except Exception as e:
|
|
191
|
+
debug(f"{hostname}: error parsing search result: {e}")
|
|
192
|
+
continue
|
|
193
|
+
|
|
194
|
+
elapsed = time.time() - start_time
|
|
195
|
+
debug(f"Time taken: {elapsed:.2f}s ({hostname})")
|
|
196
|
+
return releases
|
|
@@ -9,8 +9,8 @@ from datetime import datetime
|
|
|
9
9
|
from html import unescape
|
|
10
10
|
from urllib.parse import urljoin
|
|
11
11
|
|
|
12
|
-
from bs4 import BeautifulSoup
|
|
13
12
|
import requests
|
|
13
|
+
from bs4 import BeautifulSoup
|
|
14
14
|
|
|
15
15
|
from quasarr.providers.imdb_metadata import get_localized_title
|
|
16
16
|
from quasarr.providers.log import info, debug
|
|
@@ -65,7 +65,6 @@ def nk_search(shared_state, start_time, request_from, search_string="", mirror=N
|
|
|
65
65
|
if mirror and mirror not in supported_mirrors:
|
|
66
66
|
debug(f'Mirror "{mirror}" not supported by {hostname}.')
|
|
67
67
|
return releases
|
|
68
|
-
|
|
69
68
|
|
|
70
69
|
source_search = ""
|
|
71
70
|
if search_string != "":
|
|
@@ -94,7 +93,6 @@ def nk_search(shared_state, start_time, request_from, search_string="", mirror=N
|
|
|
94
93
|
info(f"{hostname}: search load error: {e}")
|
|
95
94
|
return releases
|
|
96
95
|
|
|
97
|
-
|
|
98
96
|
if not results:
|
|
99
97
|
return releases
|
|
100
98
|
|
|
@@ -118,7 +116,7 @@ def nk_search(shared_state, start_time, request_from, search_string="", mirror=N
|
|
|
118
116
|
a = result.find('a', class_='release-details', href=True)
|
|
119
117
|
if not a:
|
|
120
118
|
continue
|
|
121
|
-
|
|
119
|
+
|
|
122
120
|
sub_title = result.find('span', class_='subtitle')
|
|
123
121
|
if sub_title:
|
|
124
122
|
title = sub_title.get_text(strip=True)
|
|
@@ -163,9 +161,10 @@ def nk_search(shared_state, start_time, request_from, search_string="", mirror=N
|
|
|
163
161
|
|
|
164
162
|
published = convert_to_rss_date(date_text) if date_text else ""
|
|
165
163
|
|
|
166
|
-
payload = urlsafe_b64encode(
|
|
164
|
+
payload = urlsafe_b64encode(
|
|
165
|
+
f"{title}|{source}|{mirror}|{mb}|{password}|{release_imdb_id}".encode("utf-8")).decode()
|
|
167
166
|
link = f"{shared_state.values['internal_address']}/download/?payload={payload}"
|
|
168
|
-
|
|
167
|
+
|
|
169
168
|
releases.append({
|
|
170
169
|
"details": {
|
|
171
170
|
"title": title,
|
|
@@ -186,4 +185,4 @@ def nk_search(shared_state, start_time, request_from, search_string="", mirror=N
|
|
|
186
185
|
|
|
187
186
|
elapsed = time.time() - start_time
|
|
188
187
|
debug(f"Time taken: {elapsed:.2f}s ({hostname})")
|
|
189
|
-
return releases
|
|
188
|
+
return releases
|
|
@@ -27,6 +27,7 @@ quasarr/downloads/sources/by.py
|
|
|
27
27
|
quasarr/downloads/sources/dd.py
|
|
28
28
|
quasarr/downloads/sources/dt.py
|
|
29
29
|
quasarr/downloads/sources/dw.py
|
|
30
|
+
quasarr/downloads/sources/he.py
|
|
30
31
|
quasarr/downloads/sources/mb.py
|
|
31
32
|
quasarr/downloads/sources/nk.py
|
|
32
33
|
quasarr/downloads/sources/nx.py
|
|
@@ -58,6 +59,7 @@ quasarr/search/sources/dd.py
|
|
|
58
59
|
quasarr/search/sources/dt.py
|
|
59
60
|
quasarr/search/sources/dw.py
|
|
60
61
|
quasarr/search/sources/fx.py
|
|
62
|
+
quasarr/search/sources/he.py
|
|
61
63
|
quasarr/search/sources/mb.py
|
|
62
64
|
quasarr/search/sources/nk.py
|
|
63
65
|
quasarr/search/sources/nx.py
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|