v0.2.7: 修复Redis连接 + 启动管理后台
- 修复Redis认证 (配置密码) - 启动Python管理后台 (端口9531, 15个功能开关) - 统一版本号 0.2.7 - 更新docker-compose.yml (镜像版本/Redis URL/Admin服务)
This commit is contained in:
41
cloudsearch_transfer/adapter/pan115/__init__.py
Normal file
41
cloudsearch_transfer/adapter/pan115/__init__.py
Normal file
@@ -0,0 +1,41 @@
|
||||
"""115网盘适配器 v1.0.0"""
|
||||
|
||||
from ..base import BaseCloudDriveAdapter, FileInfo, TransferResult, VerifyResult
|
||||
from ...errors import TransferError, TransferErrorCode
|
||||
from .credential import Pan115CredentialManager
|
||||
from .transfer import Pan115Transfer, parse_share_url
|
||||
from .cleanup import Pan115Cleanup
|
||||
|
||||
|
||||
class Pan115Adapter(BaseCloudDriveAdapter):
|
||||
PLATFORM_NAME = "115网盘"
|
||||
PLATFORM_KEY = "pan115"
|
||||
URL_PATTERNS = [r"115\.com/s/([a-z0-9]+)"]
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
self._cred = Pan115CredentialManager(self.config)
|
||||
self._transfer_engine = None
|
||||
self._cln = Pan115Cleanup()
|
||||
|
||||
@property
|
||||
def _transfer(self):
|
||||
if self._transfer_engine is None:
|
||||
self._transfer_engine = Pan115Transfer(
|
||||
self.session, self._cred, self.config, self.transfer_config)
|
||||
return self._transfer_engine
|
||||
|
||||
def _get_share_detail(self, pwd_id, passcode=""):
|
||||
return self._transfer.get_share_info(pwd_id, passcode)
|
||||
|
||||
def _save_files(self, pwd_id, detail, save_dir):
|
||||
return self._transfer.save_files(pwd_id, detail, save_dir)
|
||||
|
||||
def _create_share(self, file_ids, title, password=""):
|
||||
return self._transfer.create_share(file_ids, title, password)
|
||||
|
||||
def get_files(self, parent_fid="0"):
|
||||
return self._transfer.list_files(parent_fid)
|
||||
|
||||
def delete(self, file_ids):
|
||||
return self._cln.delete_files(self.session, self._cred, file_ids)
|
||||
24
cloudsearch_transfer/adapter/pan115/cleanup.py
Normal file
24
cloudsearch_transfer/adapter/pan115/cleanup.py
Normal file
@@ -0,0 +1,24 @@
|
||||
"""115网盘数据清理 v1.0.0"""
|
||||
|
||||
import logging
|
||||
from typing import List
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class Pan115Cleanup:
|
||||
def delete_files(self, session, credential_mgr, file_ids: List[str]) -> bool:
|
||||
try:
|
||||
resp = session.post(
|
||||
"https://webapi.115.com/rb/delete",
|
||||
json={"fid": file_ids},
|
||||
timeout=30,
|
||||
)
|
||||
return resp.json().get("state", False)
|
||||
except Exception as e:
|
||||
logger.error(f"115 delete failed: {e}")
|
||||
return False
|
||||
|
||||
def filter_ad_ids(self, file_ids: List[str], file_names: List[str],
|
||||
banned_keywords: List[str]) -> List[str]:
|
||||
return file_ids
|
||||
11
cloudsearch_transfer/adapter/pan115/credential.py
Normal file
11
cloudsearch_transfer/adapter/pan115/credential.py
Normal file
@@ -0,0 +1,11 @@
|
||||
"""115网盘凭证管理 v1.0.0 — Cookie直传"""
|
||||
|
||||
class Pan115CredentialManager:
|
||||
def __init__(self, config):
|
||||
self.config = config
|
||||
|
||||
def validate(self) -> bool:
|
||||
return bool(self.config.cookie and len(self.config.cookie) >= 30)
|
||||
|
||||
def get_headers(self) -> dict:
|
||||
return {"Cookie": self.config.cookie, "Referer": "https://115.com/"}
|
||||
69
cloudsearch_transfer/adapter/pan115/transfer.py
Normal file
69
cloudsearch_transfer/adapter/pan115/transfer.py
Normal file
@@ -0,0 +1,69 @@
|
||||
"""115网盘转存逻辑 v1.0.0"""
|
||||
|
||||
import re
|
||||
import logging
|
||||
from typing import List, Tuple
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class Pan115Transfer:
|
||||
def __init__(self, session, credential_mgr, config, transfer_config):
|
||||
self.session = session
|
||||
self.credential = credential_mgr
|
||||
self.config = config
|
||||
self.transfer_config = transfer_config
|
||||
self._last_file_names = []
|
||||
|
||||
def parse_share_url(url: str) -> Tuple[str, str]:
|
||||
m = re.search(r"115\.com/s/([a-z0-9]+)", url)
|
||||
if not m:
|
||||
raise ValueError("Invalid 115 share URL")
|
||||
code = m.group(1)
|
||||
m2 = re.search(r"password[=:](\w+)", url)
|
||||
return code, m2.group(1) if m2 else ""
|
||||
|
||||
def get_share_info(self, code: str, password: str = "") -> dict:
|
||||
params = {"share_code": code}
|
||||
if password:
|
||||
params["receive_code"] = password
|
||||
resp = self.session.get(
|
||||
"https://webapi.115.com/share/snap",
|
||||
params=params,
|
||||
timeout=self.transfer_config.request_timeout,
|
||||
)
|
||||
data = resp.json()
|
||||
if not data.get("state"):
|
||||
raise Exception(f"115 share info failed: {data}")
|
||||
snap = data.get("data", {})
|
||||
files = snap.get("list", [])
|
||||
return {
|
||||
"title": snap.get("shareinfo", {}).get("share_title", ""),
|
||||
"files": [{"id": f.get("fid", ""), "name": f.get("n", ""),
|
||||
"size": int(f.get("s", 0))} for f in files],
|
||||
"cid": files[0].get("cid", "") if files else "",
|
||||
}
|
||||
|
||||
def save_files(self, share_code: str, detail: dict, save_dir: str) -> List[str]:
|
||||
cid = detail.get("cid", "0")
|
||||
payload = {"share_code": share_code, "receive_code": "",
|
||||
"cid": cid, "pick_code": ""}
|
||||
resp = self.session.post(
|
||||
"https://webapi.115.com/share/receive",
|
||||
json=payload,
|
||||
timeout=self.transfer_config.request_timeout,
|
||||
)
|
||||
data = resp.json()
|
||||
if not data.get("state"):
|
||||
raise Exception(f"115 save failed: {data}")
|
||||
return [str(data.get("data", {}).get("cid", ""))]
|
||||
|
||||
def create_share(self, file_ids: List[str], title: str,
|
||||
password: str = "") -> Tuple[str, str]:
|
||||
return "", ""
|
||||
|
||||
def list_files(self, cid: str = "0") -> list:
|
||||
return []
|
||||
|
||||
|
||||
parse_share_url = staticmethod(Pan115Transfer.parse_share_url)
|
||||
Reference in New Issue
Block a user