This commit is contained in:
qist
2026-05-21 18:24:36 +08:00
parent defd328b8b
commit 651f77d144
3 changed files with 419 additions and 13 deletions

View File

@@ -37,16 +37,22 @@ class Spider(Spider):
self._ua_fallback = "Dalvik/2.1.0 (Linux; U; Android 10)"
self._class_map = [
("最新电影", "/zuixindianying"),
("豆瓣Top250", "/dbtop250"),
("国产剧", "/gcj"),
("美剧", "/meijutt"),
("韩剧", "/hanjutv"),
("日剧", "/riju"),
("番剧", "/fanju"),
("剧场版", "/dongmanjuchangban"),
("海外剧", "/haiwaijuqita"),
("最新电影", "zuixindianying", "/zuixindianying"),
("豆瓣Top250", "dbtop250", "/dbtop250"),
("国产剧", "gcj", "/gcj"),
("美剧", "meijutt", "/meijutt"),
("韩剧", "hanjutv", "/hanjutv"),
("日剧", "riju", "/riju"),
("番剧", "fanju", "/fanju"),
("剧场版", "dongmanjuchangban", "/dongmanjuchangban"),
("海外剧", "haiwaijuqita", "/haiwaijuqita"),
]
self._tid_map = {}
for n, tid, p in self._class_map:
self._tid_map[tid] = p
self._tid_map[n] = p
self._tid_map[p] = p
self._tid_map[p.lstrip("/")] = p
def getName(self):
return "厂长资源"
@@ -236,8 +242,8 @@ class Spider(Spider):
def homeContent(self, filter):
result = {"class": [], "list": []}
for name, path in self._class_map:
result["class"].append({"type_name": name, "type_id": path})
for name, tid, _ in self._class_map:
result["class"].append({"type_name": name, "type_id": tid})
if self._use_api:
return result
@@ -247,7 +253,10 @@ class Spider(Spider):
return result
def homeVideoContent(self):
return {}
if self._use_api:
return {}
html = self._fetch_text(self.host + "/")
return {"list": self._parse_vod_list(html)[:24]}
def categoryContent(self, tid, pg, filter, extend):
pg = int(pg or 1)
@@ -256,6 +265,7 @@ class Spider(Spider):
if self._use_api:
return result
tid = self._tid_map.get(tid, tid)
if tid.startswith("http"):
base = tid
else:

396
py/czzyv_v2.py Normal file
View File

@@ -0,0 +1,396 @@
"""
czzyv.com - 厂长资源
"""
import re
import json
import time
from urllib.parse import urljoin, quote, unquote, urlparse, parse_qs
import requests
from base.spider import Spider
class Spider(Spider):
def __init__(self):
self.host = "https://czzyv.com"
self.timeout = 20
self._hosts = [
"https://czzyv.com",
"https://www.czzy.site",
"https://www.cz4k.com",
"https://cz01.vip",
"https://cz01.tv",
]
self.headers = {
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36",
"Referer": "https://czzyv.com/",
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8",
"Accept-Language": "zh-CN,zh;q=0.9",
}
self.session = None
self._text_cache = {}
self._text_cache_ttl = 300
self._api_base = ""
self._use_api = False
self._ua_fallback = "Dalvik/2.1.0 (Linux; U; Android 10)"
self._class_map = [
("最新电影", "zuixindianying", "/zuixindianying"),
("豆瓣Top250", "dbtop250", "/dbtop250"),
("国产剧", "gcj", "/gcj"),
("美剧", "meijutt", "/meijutt"),
("韩剧", "hanjutv", "/hanjutv"),
("日剧", "riju", "/riju"),
("番剧", "fanju", "/fanju"),
("剧场版", "dongmanjuchangban", "/dongmanjuchangban"),
("海外剧", "haiwaijuqita", "/haiwaijuqita"),
]
self._tid_map = {}
for n, tid, p in self._class_map:
self._tid_map[tid] = p
self._tid_map[n] = p
self._tid_map[p] = p
self._tid_map[p.lstrip("/")] = p
def getName(self):
return "厂长资源"
def init(self, extend=""):
if isinstance(extend, dict):
host = (extend.get("host") or extend.get("site") or "").strip()
if host:
self.host = host.rstrip("/")
elif isinstance(extend, str) and extend.strip():
ext_str = extend.strip()
if (ext_str.startswith("{") and ext_str.endswith("}")) or (ext_str.startswith("[") and ext_str.endswith("]")):
try:
ext_obj = json.loads(ext_str)
if isinstance(ext_obj, dict):
host = (ext_obj.get("host") or ext_obj.get("site") or "").strip()
if host:
self.host = host.rstrip("/")
except Exception:
pass
elif ext_str.startswith("http"):
self.host = ext_str.rstrip("/")
self.headers["Referer"] = self.host + "/"
self.headers["Origin"] = self.host
self.session = requests.Session()
self.session.headers.update(self.headers)
self._choose_host()
self._detect_api()
self._warmup()
def isVideoFormat(self, url):
pass
def manualVideoCheck(self):
pass
def destroy(self):
pass
def _choose_host(self):
if not self.session:
return
candidates = []
if self.host:
candidates.append(self.host.rstrip("/"))
for h in self._hosts:
h = (h or "").rstrip("/")
if h and h not in candidates:
candidates.append(h)
for h in candidates:
try:
r = self.session.get(h + "/", timeout=self.timeout, allow_redirects=True, verify=False)
if not r or r.status_code != 200:
continue
r.encoding = "utf-8"
text = r.text or ""
if "访问已被拦截" in text or "已被拦截" in text:
continue
self.host = h
self.headers["Referer"] = self.host + "/"
self.headers["Origin"] = self.host
self.session.headers.update(self.headers)
self._text_cache.clear()
return
except Exception:
continue
def _detect_api(self):
candidates = [
"/api.php/provide/vod?ac=list",
"/api.php/provide/vod/?ac=list",
"/index.php/api/vod?ac=list",
]
for p in candidates:
url = urljoin(self.host + "/", p.lstrip("/"))
try:
r = self.session.get(url, timeout=self.timeout, allow_redirects=True, verify=False)
ct = (r.headers.get("Content-Type") or "").lower()
if r.status_code == 200 and (("json" in ct) or ("xml" in ct) or r.text.strip().startswith(("{", "<"))):
self._api_base = url.split("?", 1)[0]
self._use_api = True
return
except Exception:
continue
self._api_base = ""
self._use_api = False
def _warmup(self):
try:
self.session.get(self.host + "/", timeout=self.timeout, allow_redirects=True, verify=False)
except Exception:
pass
def _fetch_text(self, url):
now = time.time()
cached = self._text_cache.get(url)
if cached and cached[0] > now:
return cached[1]
try:
r = None
for _ in range(3):
try:
r = self.session.get(url, timeout=self.timeout, allow_redirects=True, verify=False)
break
except Exception:
time.sleep(1)
if not r or r.status_code != 200:
if r and r.status_code in (403, 406, 412):
self.session.headers["User-Agent"] = self._ua_fallback
try:
r = self.session.get(url, timeout=self.timeout, allow_redirects=True, verify=False)
except Exception:
r = None
if not r or r.status_code != 200:
return ""
r.encoding = "utf-8"
text = r.text or ""
if "访问已被拦截" in text or "已被拦截" in text:
self.session.headers["User-Agent"] = self._ua_fallback
try:
r2 = self.session.get(url, timeout=self.timeout, allow_redirects=True, verify=False)
if r2 and r2.status_code == 200:
r2.encoding = "utf-8"
text = r2.text or ""
except Exception:
pass
if text:
self._text_cache[url] = (now + self._text_cache_ttl, text)
return text
except Exception:
return ""
def _abs(self, href):
return urljoin(self.host + "/", href or "")
def _parse_pagecount(self, html, current_pg):
m = re.findall(r"/page/(\d+)", html or "")
nums = [int(x) for x in m if x.isdigit()]
if nums:
return max(max(nums), current_pg)
return 999
def _parse_vod_list(self, html):
html = html or ""
vods = []
blocks = re.findall(r"(?is)<li\b[^>]*>.*?</li>", html)
for b in blocks:
if "/movie/" not in b:
continue
m_id = re.search(r'(?is)href=["\'](?:https?://[^"\']+)?/movie/(\d+)\.html["\']', b)
if not m_id:
continue
mid = m_id.group(1)
m_alt = re.search(r'(?is)<img\b[^>]*\balt=["\']([^"\']+)["\']', b)
name = (m_alt.group(1).strip() if m_alt else "") or mid
m_pic = re.search(r'(?is)<img\b[^>]*(?:data-original|data-src|data-lazy-src|src)=["\']([^"\']+)["\']', b)
pic = (m_pic.group(1).strip() if m_pic else "")
remark = ""
m_qb = re.search(r'(?is)<div\b[^>]*class=["\'][^"\']*\bhdinfo\b[^"\']*["\'][^>]*>.*?<span\b[^>]*>(.*?)</span>', b)
if m_qb:
remark = re.sub(r"(?is)<[^>]+>", "", m_qb.group(1)).strip()
m_score = re.search(r'(?is)<div\b[^>]*class=["\'][^"\']*\brating\b[^"\']*["\'][^>]*>\s*([^<]+)\s*</div>', b)
score = (m_score.group(1).strip() if m_score else "")
if remark and score and score not in remark:
remark = f"{remark} {score}"
elif not remark:
remark = score
vods.append({"vod_id": mid, "vod_name": name, "vod_pic": pic, "vod_remarks": remark})
seen = set()
unique = []
for v in vods:
if v["vod_id"] in seen:
continue
seen.add(v["vod_id"])
unique.append(v)
return unique
def homeContent(self, filter):
result = {"class": [], "list": []}
for name, tid, _ in self._class_map:
result["class"].append({"type_name": name, "type_id": tid})
if self._use_api:
return result
html = self._fetch_text(self.host + "/")
result["list"] = self._parse_vod_list(html)[:24]
return result
def homeVideoContent(self):
if self._use_api:
return {}
html = self._fetch_text(self.host + "/")
return {"list": self._parse_vod_list(html)[:24]}
def categoryContent(self, tid, pg, filter, extend):
pg = int(pg or 1)
result = {"list": [], "page": pg, "pagecount": 999, "limit": 24, "total": 0}
if self._use_api:
return result
tid = self._tid_map.get(tid, tid)
if tid.startswith("http"):
base = tid
else:
base = self._abs(tid)
if pg > 1:
if base.endswith("/"):
url = base + f"page/{pg}"
else:
url = base + f"/page/{pg}"
else:
url = base
html = self._fetch_text(url)
result["list"] = self._parse_vod_list(html)
result["pagecount"] = self._parse_pagecount(html, pg)
result["total"] = result["pagecount"] * result["limit"]
return result
def detailContent(self, ids):
if not ids or not ids[0]:
return {"list": []}
vid = ids[0]
if vid.startswith("http"):
url = vid
else:
url = f"{self.host}/movie/{vid}.html"
html = self._fetch_text(url)
name = ""
m_title = re.search(r"(?is)<h1[^>]*>\s*([^<]+)\s*</h1>", html or "")
if m_title:
name = m_title.group(1).strip()
pic = ""
m_pic = re.search(r'(?is)<div\b[^>]*class=["\'][^"\']*\bdyimg\b[^"\']*["\'][^>]*>[\s\S]*?<img\b[^>]*(?:data-original|data-src|data-lazy-src|src)=["\']([^"\']+)["\']', html or "")
if not m_pic:
m_pic = re.search(r'(?is)<img\b[^>]*(?:data-original|data-src|data-lazy-src|src)=["\']([^"\']+)["\']', html or "")
if m_pic:
pic = m_pic.group(1).strip()
desc = ""
m_desc = re.search(r'(?is)<div\b[^>]*class=["\'][^"\']*\byp_context\b[^"\']*["\'][^>]*>\s*([\s\S]*?)\s*</div>', html or "")
if m_desc:
desc = re.sub(r"(?is)<[^>]+>", "", m_desc.group(1)).strip()
actor = ""
director = ""
m_actor = re.search(r"(?is)主演:\s*([^<\n\r]+)", html or "")
if m_actor:
actor = m_actor.group(1).strip()
m_director = re.search(r"(?is)导演:\s*([^<\n\r]+)", html or "")
if m_director:
director = m_director.group(1).strip()
play_items = []
for m in re.finditer(r'(?is)<a\b[^>]*href=["\']([^"\']*/v_play/[^"\']+)["\'][^>]*>\s*([^<]+)\s*</a>', html or ""):
href = self._abs(m.group(1).strip())
t = m.group(2).strip()
if t and href:
play_items.append(f"{t}${href}")
if not play_items:
play_items.append(f"播放${url}")
return {
"list": [
{
"vod_id": vid,
"vod_name": name or vid,
"vod_pic": pic,
"vod_remarks": "",
"vod_year": "",
"type_name": "",
"vod_content": desc,
"vod_actor": actor,
"vod_director": director,
"vod_play_from": "厂长资源",
"vod_play_url": "#".join(play_items),
}
]
}
def searchContent(self, key, quick, pg="1"):
if not key:
return {"list": []}
if self._use_api:
return {"list": []}
pg = str(pg or "1")
url = f"{self.host}/boss1O1?q={quote(key)}"
if pg != "1":
url += f"&page={quote(pg)}"
html = self._fetch_text(url)
return {"list": self._parse_vod_list(html)}
def playerContent(self, flag, id, vipFlags):
h = {"User-Agent": self.headers.get("User-Agent", ""), "Referer": self.host + "/"}
if not id:
return {"parse": 1, "url": "", "header": h, "playUrl": ""}
if isinstance(id, str) and (".m3u8" in id or ".mp4" in id) and id.startswith("http"):
return {"parse": 0, "url": id, "header": h, "playUrl": ""}
play_url = id if id.startswith("http") else self._abs(id)
if "/v_play/" not in play_url:
return {"parse": 1, "url": play_url, "header": h, "playUrl": ""}
html = self._fetch_text(play_url)
m_iframe = re.search(r'(?is)<iframe\b[^>]*\bsrc=["\']([^"\']+)["\']', html or "")
if m_iframe:
src = m_iframe.group(1).strip()
if src:
qs = parse_qs(urlparse(src).query)
raw = (qs.get("url") or [""])[0]
raw = unquote(raw).strip()
if raw.startswith("http"):
return {"parse": 0, "url": raw, "header": h, "playUrl": ""}
return {"parse": 1, "url": src, "header": h, "playUrl": ""}
m = re.findall(r'https?://[^\s"\']+?\.(?:m3u8|mp4)(?:\?[^\s"\']*)?', html or "")
if m:
return {"parse": 0, "url": m[0], "header": h, "playUrl": ""}
return {"parse": 1, "url": play_url, "header": h, "playUrl": ""}
def localProxy(self, param):
return None

View File

@@ -13,7 +13,7 @@
{"key":"闪雷","name":"闪雷┃MP4","type":3,"api":"./lib/drpy2.min.js","ext":"./js/678.js","header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0"}},
{"key":"fyyy","name":"飞宇影院","type":3,"api":"csp_XBPQ","searchable":1,"quickSearch":1,"filterable":1,"changeable":1,"ext":{"分类url":"http://ntfeiyu.com/nt/{cateId}/area/{area}/by/{by}/class/{class}/lang/{lang}/page/{catePg}/year/{year}.html","分类":"电影$1#电视剧$2#综艺$3#动漫$4"},"header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0"}},
{"key":"cupfox_in","name":"茶杯狐┃cupfox.in","type":3,"api":"./lib/drpy2.min.js","ext":"./js/茶杯狐.js","searchable":1,"quickSearch":1,"filterable":0,"changeable":1,"header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0"}},
{"key": "czzyv","name": "厂长 | 影视","type": 3,"api": "./py/czzyv.py","searchable": 1,"quickSearch": 1,"filterable": 0,"changeable": 1,"header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36","Referer":"https://czzyv.com/"}},
{"key": "czzyv","name": "厂长 | 影视","type": 3,"api": "./py/czzyv_v2.py","searchable": 1,"quickSearch": 1,"filterable": 0,"changeable": 1,"header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36","Referer":"https://czzyv.com/"}},
{"key": "libvio","name": "libvio | 影视","type": 3,"api": "./py/libvio.py","searchable": 1,"quickSearch": 1,"filterable": 0,"changeable": 1},
{"key":"italkbbtv","name":"ITalkBB | 外","type":3,"api":"./py/ITalkBBTV.py","searchable":1,"quickSearch":1,"filterable":0,"changeable":1},
{"key":"qiletv","name":"奇乐影视","type":3,"api":"csp_XBPQ","searchable":1,"quickSearch":1,"filterable":1,"changeable":1,"ext":"./json/奇乐.json","header":{"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0"}},