202 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			202 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import base64
 | ||
| import json
 | ||
| import re
 | ||
| from typing import Tuple, List
 | ||
| 
 | ||
| from ruamel.yaml import CommentedMap
 | ||
| 
 | ||
| from app.core.config import settings
 | ||
| from app.db.systemconfig_oper import SystemConfigOper
 | ||
| from app.log import logger
 | ||
| from app.schemas import MediaType
 | ||
| from app.utils.http import RequestUtils
 | ||
| from app.utils.string import StringUtils
 | ||
| 
 | ||
| 
 | ||
| class MTorrentSpider:
 | ||
|     """
 | ||
|     mTorrent API,需要缓存ApiKey
 | ||
|     """
 | ||
|     _indexerid = None
 | ||
|     _domain = None
 | ||
|     _name = ""
 | ||
|     _proxy = None
 | ||
|     _cookie = None
 | ||
|     _ua = None
 | ||
|     _size = 100
 | ||
|     _searchurl = "%sapi/torrent/search"
 | ||
|     _downloadurl = "%sapi/torrent/genDlToken"
 | ||
|     _pageurl = "%sdetail/%s"
 | ||
| 
 | ||
|     # 电影分类
 | ||
|     _movie_category = ['401', '419', '420', '421', '439', '405', '404']
 | ||
|     _tv_category = ['403', '402', '435', '438', '404', '405']
 | ||
| 
 | ||
|     # API KEY
 | ||
|     _apikey = None
 | ||
|     # JWT Token
 | ||
|     _token = None
 | ||
| 
 | ||
|     # 标签
 | ||
|     _labels = {
 | ||
|         "0": "",
 | ||
|         "1": "DIY",
 | ||
|         "2": "国配",
 | ||
|         "3": "DIY 国配",
 | ||
|         "4": "中字",
 | ||
|         "5": "DIY 中字",
 | ||
|         "6": "国配 中字",
 | ||
|         "7": "DIY 国配 中字"
 | ||
|     }
 | ||
| 
 | ||
|     def __init__(self, indexer: CommentedMap):
 | ||
|         self.systemconfig = SystemConfigOper()
 | ||
|         if indexer:
 | ||
|             self._indexerid = indexer.get('id')
 | ||
|             self._domain = indexer.get('domain')
 | ||
|             self._searchurl = self._searchurl % self._domain
 | ||
|             self._name = indexer.get('name')
 | ||
|             if indexer.get('proxy'):
 | ||
|                 self._proxy = settings.PROXY
 | ||
|             self._cookie = indexer.get('cookie')
 | ||
|             self._ua = indexer.get('ua')
 | ||
|             self._apikey = indexer.get('apikey')
 | ||
|             self._token = indexer.get('token')
 | ||
| 
 | ||
|     def search(self, keyword: str, mtype: MediaType = None, page: int = 0) -> Tuple[bool, List[dict]]:
 | ||
|         """
 | ||
|         搜索
 | ||
|         """
 | ||
|         # 检查ApiKey
 | ||
|         if not self._apikey:
 | ||
|             return True, []
 | ||
| 
 | ||
|         if not mtype:
 | ||
|             categories = []
 | ||
|         elif mtype == MediaType.TV:
 | ||
|             categories = self._tv_category
 | ||
|         else:
 | ||
|             categories = self._movie_category
 | ||
|         params = {
 | ||
|             "keyword": keyword,
 | ||
|             "categories": categories,
 | ||
|             "pageNumber": int(page) + 1,
 | ||
|             "pageSize": self._size,
 | ||
|             "visible": 1
 | ||
|         }
 | ||
|         res = RequestUtils(
 | ||
|             headers={
 | ||
|                 "Content-Type": "application/json",
 | ||
|                 "User-Agent": f"{self._ua}",
 | ||
|                 "x-api-key": self._apikey
 | ||
|             },
 | ||
|             proxies=self._proxy,
 | ||
|             referer=f"{self._domain}browse",
 | ||
|             timeout=15
 | ||
|         ).post_res(url=self._searchurl, json=params)
 | ||
|         torrents = []
 | ||
|         if res and res.status_code == 200:
 | ||
|             results = res.json().get('data', {}).get("data") or []
 | ||
|             for result in results:
 | ||
|                 category_value = result.get('category')
 | ||
|                 if category_value in self._tv_category \
 | ||
|                         and category_value not in self._movie_category:
 | ||
|                     category = MediaType.TV.value
 | ||
|                 elif category_value in self._movie_category:
 | ||
|                     category = MediaType.MOVIE.value
 | ||
|                 else:
 | ||
|                     category = MediaType.UNKNOWN.value
 | ||
|                 labels_value = self._labels.get(result.get('labels') or "0") or ""
 | ||
|                 if labels_value:
 | ||
|                     labels = labels_value.split()
 | ||
|                 else:
 | ||
|                     labels = []
 | ||
|                 torrent = {
 | ||
|                     'title': result.get('name'),
 | ||
|                     'description': result.get('smallDescr'),
 | ||
|                     'enclosure': self.__get_download_url(result.get('id')),
 | ||
|                     'pubdate': StringUtils.format_timestamp(result.get('createdDate')),
 | ||
|                     'size': int(result.get('size') or '0'),
 | ||
|                     'seeders': int(result.get('status', {}).get("seeders") or '0'),
 | ||
|                     'peers': int(result.get('status', {}).get("leechers") or '0'),
 | ||
|                     'grabs': int(result.get('status', {}).get("timesCompleted") or '0'),
 | ||
|                     'downloadvolumefactor': self.__get_downloadvolumefactor(result.get('status', {}).get("discount")),
 | ||
|                     'uploadvolumefactor': self.__get_uploadvolumefactor(result.get('status', {}).get("discount")),
 | ||
|                     'page_url': self._pageurl % (self._domain, result.get('id')),
 | ||
|                     'imdbid': self.__find_imdbid(result.get('imdb')),
 | ||
|                     'labels': labels,
 | ||
|                     'category': category
 | ||
|                 }
 | ||
|                 torrents.append(torrent)
 | ||
|         elif res is not None:
 | ||
|             logger.warn(f"{self._name} 搜索失败,错误码:{res.status_code}")
 | ||
|             return True, []
 | ||
|         else:
 | ||
|             logger.warn(f"{self._name} 搜索失败,无法连接 {self._domain}")
 | ||
|             return True, []
 | ||
|         return False, torrents
 | ||
| 
 | ||
|     @staticmethod
 | ||
|     def __find_imdbid(imdb: str) -> str:
 | ||
|         """
 | ||
|         从imdb链接中提取imdbid
 | ||
|         """
 | ||
|         if imdb:
 | ||
|             m = re.search(r"tt\d+", imdb)
 | ||
|             if m:
 | ||
|                 return m.group(0)
 | ||
|         return ""
 | ||
| 
 | ||
|     @staticmethod
 | ||
|     def __get_downloadvolumefactor(discount: str) -> float:
 | ||
|         """
 | ||
|         获取下载系数
 | ||
|         """
 | ||
|         discount_dict = {
 | ||
|             "FREE": 0,
 | ||
|             "PERCENT_50": 0.5,
 | ||
|             "PERCENT_70": 0.3,
 | ||
|             "_2X_FREE": 0,
 | ||
|             "_2X_PERCENT_50": 0.5
 | ||
|         }
 | ||
|         if discount:
 | ||
|             return discount_dict.get(discount, 1)
 | ||
|         return 1
 | ||
| 
 | ||
|     @staticmethod
 | ||
|     def __get_uploadvolumefactor(discount: str) -> float:
 | ||
|         """
 | ||
|         获取上传系数
 | ||
|         """
 | ||
|         uploadvolumefactor_dict = {
 | ||
|             "_2X": 2.0,
 | ||
|             "_2X_FREE": 2.0,
 | ||
|             "_2X_PERCENT_50": 2.0
 | ||
|         }
 | ||
|         if discount:
 | ||
|             return uploadvolumefactor_dict.get(discount, 1)
 | ||
|         return 1
 | ||
| 
 | ||
|     def __get_download_url(self, torrent_id: str) -> str:
 | ||
|         """
 | ||
|         获取下载链接,返回base64编码的json字符串及URL
 | ||
|         """
 | ||
|         url = self._downloadurl % self._domain
 | ||
|         params = {
 | ||
|             'method': 'post',
 | ||
|             'cookie': False,
 | ||
|             'params': {
 | ||
|                 'id': torrent_id
 | ||
|             },
 | ||
|             'header': {
 | ||
|                 'Content-Type': 'application/json',
 | ||
|                 'User-Agent': f'{self._ua}',
 | ||
|                 'Accept': 'application/json, text/plain, */*',
 | ||
|                 'x-api-key': self._apikey
 | ||
|             },
 | ||
|             'result': 'data'
 | ||
|         }
 | ||
|         # base64编码
 | ||
|         base64_str = base64.b64encode(json.dumps(params).encode('utf-8')).decode('utf-8')
 | ||
|         return f"[{base64_str}]{url}"
 |