Files
nexusphp/nexus/PTGen/PTGen.php

493 lines
17 KiB
PHP
Raw Normal View History

2021-01-15 01:12:44 +08:00
<?php
/**
* get media info base on PT-Gen
*
* @since 1.6
* @see https://github.com/Rhilip/pt-gen-cfworker
*/
namespace Nexus\PTGen;
use App\Models\Torrent;
use Carbon\Carbon;
2021-01-15 01:12:44 +08:00
use GuzzleHttp\Client;
use Illuminate\Support\Arr;
use Illuminate\Support\Str;
use Nexus\Imdb\Imdb;
2021-01-15 01:12:44 +08:00
class PTGen
{
private $apiPoint;
const SITE_DOUBAN = 'douban';
const SITE_IMDB = 'imdb';
const SITE_BANGUMI = 'bangumi';
const SITE_STEAM = 'steam';
const SITE_INDIENOVA = 'indienova';
const SITE_EPIC = 'epic';
2021-01-15 01:12:44 +08:00
public static array $validSites = [
2021-01-15 01:12:44 +08:00
self::SITE_IMDB => [
'url_pattern' => '/(?:https?:\/\/)?(?:www\.)?imdb\.com\/title\/(tt\d+)\/?/',
2021-01-15 22:13:46 +08:00
'home_page' => 'https://www.imdb.com/',
2021-01-18 19:45:33 +08:00
'rating_average_img' => 'pic/imdb2.png',
2022-04-15 15:28:30 +08:00
'rating_pattern_in_desc' => "/IMDb评分.*([\d\.]+)\//iU",
2021-01-15 22:13:46 +08:00
],
self::SITE_DOUBAN => [
'url_pattern' => '/(?:https?:\/\/)?(?:(?:movie|www)\.)?douban\.com\/(?:subject|movie)\/(\d+)\/?/',
'home_page' => 'https://www.douban.com/',
2021-01-18 19:45:33 +08:00
'rating_average_img' => 'pic/douban2.png',
2022-04-15 15:28:30 +08:00
'rating_pattern_in_desc' => "/豆瓣评分.*([\d\.]+)\//iU",
2021-01-15 01:12:44 +08:00
],
self::SITE_BANGUMI => [
'url_pattern' => '/(?:https?:\/\/)?(?:bgm\.tv|bangumi\.tv|chii\.in)\/subject\/(\d+)\/?/',
2021-01-15 22:13:46 +08:00
'home_page' => 'https://bangumi.tv/',
2021-01-19 17:35:05 +08:00
'rating_average_img' => 'pic/bangumi.jpg',
2021-01-15 01:12:44 +08:00
],
//Banned !
// self::SITE_STEAM => [
// 'url_pattern' => '/(?:https?:\/\/)?(?:store\.)?steam(?:powered|community)\.com\/app\/(\d+)\/?/',
// 'home_page' => 'https://store.steampowered.com/',
// 'rating_average_img' => 'pic/steam.svg',
// ],
self::SITE_INDIENOVA => [
'url_pattern' => '/(?:https?:\/\/)?indienova\.com\/game\/(\S+)/',
'home_page' => 'https://indienova.com/',
'rating_average_img' => 'pic/invienova.jpg',
],
//seems url_pattern has changed
// self::SITE_EPIC => [
// 'url_pattern' => '/(?:https?:\/\/)?www\.epicgames\.com\/store\/[a-zA-Z-]+\/product\/(\S+)\/\S?/',
// 'home_page' => 'https://store.epicgames.com/',
// 'rating_average_img' => 'pic/epic_game.png',
// ],
2021-01-15 01:12:44 +08:00
];
public function __construct()
{
2021-01-19 17:35:05 +08:00
$setting = get_setting('main');
2021-03-19 14:19:02 +08:00
$this->setApiPoint($setting['pt_gen_api_point'] ?? '');
2021-01-15 01:12:44 +08:00
}
public function getApiPoint(): string
{
return $this->apiPoint;
}
public function setApiPoint(string $apiPoint)
{
$this->apiPoint = trim($apiPoint);
2021-01-15 01:12:44 +08:00
}
2021-01-18 00:41:35 +08:00
public function generate(string $url, bool $withoutCache = false): array
{
$parsed = $this->parse($url);
$targetUrl = trim($this->apiPoint, '/');
if (Str::contains($targetUrl, '?')) {
$targetUrl .= "&";
} else {
$targetUrl .= "?";
}
$targetUrl .= sprintf('site=%s&sid=%s&url=%s', $parsed['site'] , $parsed['id'], urlencode($parsed['url']));
2021-01-18 00:41:35 +08:00
return $this->request($targetUrl, $withoutCache);
}
public function parse(string $url): array
2021-01-15 01:12:44 +08:00
{
2021-01-15 22:13:46 +08:00
foreach (self::$validSites as $site => $info) {
if (preg_match($info['url_pattern'], $url, $matches)) {
2021-01-18 00:41:35 +08:00
return [
'site' => $site,
'url' => $matches[0],
'id' => $matches[1]
];
2021-01-15 22:13:46 +08:00
}
}
throw new PTGenException("invalid url: $url");
2021-01-15 01:12:44 +08:00
}
private function buildDetailsPageTableRow($torrentId, $ptGenArr, $site): string
2021-01-15 01:12:44 +08:00
{
2021-01-15 22:13:46 +08:00
global $lang_details;
if ($this->isRawPTGen($ptGenArr)) {
$ptGenFormatted = $ptGenArr['format'];
} elseif ($this->isIyuu($ptGenArr)) {
$ptGenFormatted = $ptGenArr['data']['format'];
} else {
do_log("Invalid pt gen data", 'error');
return '';
}
2021-01-19 00:55:34 +08:00
$poster = '';
if (!empty($ptGenArr['poster'])) {
$poster = $ptGenArr['poster'];
} elseif (preg_match('/\[img\](.*)\[\/img\]/iU', $ptGenFormatted, $matches)) {
$poster = $matches[1];
}
if ($poster) {
2021-01-19 00:55:34 +08:00
$prefix = sprintf("[img]%s[/img]\n", $poster);
$ptGenFormatted = mb_substr($ptGenFormatted, mb_strlen($prefix, 'utf-8') + 1);
}
2021-01-15 22:13:46 +08:00
$ptGenFormatted = format_comment($ptGenFormatted);
2021-01-18 00:41:35 +08:00
$ptGenFormatted .= sprintf(
'%s %s%s<a href="retriver.php?id=%s&type=1&siteid=%s">%s</a>',
$lang_details['text_information_updated_at'], $ptGenArr['__updated_at'], $lang_details['text_might_be_outdated'],
2021-01-18 00:41:35 +08:00
$torrentId, $site, $lang_details['text_here_to_update']
);
2021-01-15 22:13:46 +08:00
$titleShowOrHide = $lang_details['title_show_or_hide'] ?? '';
$id = 'pt-gen-' . $site;
$posterHtml = "";
if ($poster) {
$posterHtml = sprintf('<div id="poster%s"><img src="%s" width="105" onclick="Preview(this);" title="%s"', $id, $poster, $titleShowOrHide);
}
2021-01-15 22:13:46 +08:00
$html = <<<HTML
<tr>
<td class="rowhead">
<a href="javascript: klappe_ext('{$id}')">
<span class="nowrap">
<img id="pic{$id}" class="minus" src="pic/trans.gif" alt="Show/Hide" title="{$titleShowOrHide}" />
PT-Gen-{$site}
</span>
</a>
$posterHtml
2021-01-15 22:13:46 +08:00
</td>
<td class="rowfollow" align="left">
<div id="k{$id}">
{$ptGenFormatted}
</div>
</td>
</tr>
HTML;
return $html;
2021-01-15 01:12:44 +08:00
}
2021-01-18 00:41:35 +08:00
private function request(string $url, bool $withoutCache = false): array
2021-01-15 01:12:44 +08:00
{
2021-01-15 22:13:46 +08:00
global $Cache;
2021-01-19 00:55:34 +08:00
$begin = microtime(true);
2021-01-15 01:12:44 +08:00
$logPrefix = "url: $url";
2021-01-18 00:41:35 +08:00
$cacheKey = $this->getApiPointResultCacheKey($url);
if (!$withoutCache) {
$cache = $Cache->get_value($cacheKey);
if ($cache) {
do_log("$logPrefix, from cache");
return $cache;
}
2021-01-15 22:13:46 +08:00
}
do_log("$logPrefix, going to send request...");
2021-01-15 01:12:44 +08:00
$http = new Client();
$response = $http->get($url, ['timeout' => 10]);
2021-01-15 01:12:44 +08:00
$statusCode = $response->getStatusCode();
if ($statusCode != 200) {
$msg = "api point response http status code: $statusCode";
do_log("$logPrefix, $msg");
throw new PTGenException($msg);
}
$bodyString = (string)$response->getBody();
if (empty($bodyString)) {
$msg = "response body empty";
do_log("$logPrefix, $msg");
throw new PTGenException($msg);
}
$bodyArr = json_decode($bodyString, true);
if (empty($bodyArr) || !is_array($bodyArr)) {
$msg = "response body error: $bodyString";
do_log("$logPrefix, $msg");
throw new PTGenException($msg);
}
if ($this->isRawPTGen($bodyArr) || $this->isIyuu($bodyArr)) {
$Cache->cache_value($cacheKey, $bodyArr, 24 * 3600);
do_log("$logPrefix, success get from api point, use time: " . (microtime(true) - $begin));
$bodyArr['__updated_at'] = now()->toDateTimeString();
return $bodyArr;
} else {
2021-01-15 01:12:44 +08:00
$msg = "error: " . $bodyArr['error'] ?? '';
do_log("$logPrefix, response: $bodyString");
throw new PTGenException($msg);
}
}
2021-01-18 00:41:35 +08:00
public function deleteApiPointResultCache($url)
{
global $Cache;
$Cache->delete_value($this->getApiPointResultCacheKey($url));
}
private function getApiPointResultCacheKey($url)
{
return __METHOD__ . "_$url";
}
public function renderUploadPageFormInput($ptGen = ''): string
2021-01-15 01:12:44 +08:00
{
$arr = json_decode($ptGen, true);
$link = is_array($arr) ? $arr['__link'] : $ptGen;
$y = $this->buildInput("pt_gen", $link, nexus_trans('ptgen.tooltip', ['sites' => $this->buildTooltip()]), nexus_trans('ptgen.btn_get_desc'));
return tr(nexus_trans('ptgen.label'), $y, 1, '', true);
}
private function buildTooltip(): string
{
$results = [];
2021-01-15 22:13:46 +08:00
foreach (self::$validSites as $site => $info) {
$results[] = sprintf('<a href="%s" target="_blank" /><strong>%s</strong></a>', $info['home_page'], $site);
2021-01-15 01:12:44 +08:00
}
return implode(' / ', $results);
2021-01-15 22:13:46 +08:00
}
public function buildInput($name, $value, $note, $btnText): string
{
$btn = '';
if ($this->apiPoint != '') {
$btn = '<div><input type="button" class="btn-get-pt-gen" value="'.$btnText.'"></div>';
}
$input = <<<HTML
<div style="display: flex">
<div style="display: flex;flex-direction: column;flex-grow: 1">
<input type="text" name="$name" value="{$value}" data-pt-gen="$name">
<span class="medium">{$note}</span>
</div>
$btn
</div>
HTML;
return $input;
}
2022-02-15 17:43:18 +08:00
public function renderDetailsPageDescription($torrentId, $torrentPtGenArr): array
2021-01-15 22:13:46 +08:00
{
$html = '';
$jsonArr = [];
$update = false;
2022-02-15 17:43:18 +08:00
$torrentPtGenArr = (array)$torrentPtGenArr;
2021-01-15 22:13:46 +08:00
foreach (self::$validSites as $site => $info) {
if (empty($torrentPtGenArr[$site]['link'])) {
continue;
}
$link = $torrentPtGenArr[$site]['link'];
$data = $torrentPtGenArr[$site]['data'] ?? [];
if (!empty($data)) {
$jsonArr[$site] = [
'link' => $link,
'data' => $data,
];
2021-01-18 00:41:35 +08:00
$html .= $this->buildDetailsPageTableRow($torrentId, $data, $site);
2021-01-15 22:13:46 +08:00
} else {
2021-01-19 00:55:34 +08:00
try {
$ptGenArr = $this->generate($torrentPtGenArr[$site]['link']);
} catch (\Exception $e) {
$log = $e->getMessage() . ", trace: " . $e->getTraceAsString();
do_log($log,'error');
$ptGenArr = [
'format' => $e->getMessage()
];
}
2021-01-15 22:13:46 +08:00
$jsonArr[$site] = [
'link' => $link,
'data' => $ptGenArr,
];
2021-01-18 00:41:35 +08:00
$html .= $this->buildDetailsPageTableRow($torrentId, $ptGenArr, $site);
2021-01-15 22:13:46 +08:00
if (!$update) {
$update = true;
}
}
2021-01-15 01:12:44 +08:00
}
2021-01-15 22:13:46 +08:00
return ['json_arr' => $jsonArr, 'html' => $html, 'update' => $update];
2021-01-15 01:12:44 +08:00
}
2021-01-18 19:45:33 +08:00
public function buildRatingSpan(array $siteIdAndRating): string
2021-01-18 19:45:33 +08:00
{
2021-03-17 01:21:35 +08:00
$result = '<td class="embedded" style="text-align: right; width: 40px;padding: 4px"><div style="display: flex;flex-direction: column">';
2021-01-18 19:45:33 +08:00
$count = 1;
2021-03-19 14:19:02 +08:00
$ratingIcons = [];
2021-01-18 19:45:33 +08:00
foreach (self::$validSites as $site => $info) {
2021-03-19 14:19:02 +08:00
if (!isset($siteIdAndRating[$site])) {
continue;
}
$rating = $siteIdAndRating[$site];
2021-01-18 19:45:33 +08:00
if (empty($rating)) {
continue;
}
if ($count > 2) {
//only show the first two
break;
}
2021-03-19 14:19:02 +08:00
$ratingIcons[] = $this->getRatingIcon($site, $rating);
2021-01-18 19:45:33 +08:00
$count++;
}
2021-03-19 14:19:02 +08:00
if (empty($ratingIcons)) {
$ratingIcons[] = $this->getRatingIcon(self::SITE_IMDB, 'N/A');
$ratingIcons[] = $this->getRatingIcon(self::SITE_DOUBAN, 'N/A');
}
$result .= implode("", $ratingIcons) . '</div></td>';
return $result;
}
public function getRatingIcon($siteId, $rating): string
2021-03-19 14:19:02 +08:00
{
if (is_numeric($rating)) {
$rating = number_format($rating, 1);
}
2021-03-19 14:19:02 +08:00
$result = sprintf(
'<div style="display: flex;align-content: center;justify-content: space-between;padding: 2px 0"><img src="%s" alt="%s" title="%s" style="max-width: 16px;max-height: 16px"/><span>%s</span></div>',
self::$validSites[$siteId]['rating_average_img'], $siteId, $siteId, $rating
);
2021-01-18 19:45:33 +08:00
return $result;
}
public function isRawPTGen(array $bodyArr): bool
{
return isset($bodyArr['success']) && $bodyArr['success'];
}
public function isIyuu(array $bodyArr): bool
{
return false;
//Not support, due to change frequently
// return isset($bodyArr['ret']) && $bodyArr['ret'] == 200;
}
public function listRatings(array $ptGenData, string $imdbLink, string $desc = ''): array
{
$results = [];
$log = "";
//First, get from PTGen
foreach (self::$validSites as $site => $info) {
if (!isset($ptGenData[$site]['data'])) {
continue;
}
$data = $ptGenData[$site]['data'];
$log .= ", handling site: $site";
$rating = '';
if (isset($data['__rating'])) {
//__rating is new add
$rating = $data['__rating'];
$log .= ", from __rating";
} else {
// from original structure fetch
if ($this->isRawPTGen($data)) {
$log .= ", isRawPTGen";
$rating = $this->getRawPTGenRating($data, $site);
} elseif ($this->isIyuu($data)) {
$log .= ", isIyuu";
$pattern = $info['rating_pattern_in_desc'] ?? null;
if ($pattern && preg_match($pattern,$data['data']['format'], $matches)) {
$rating = $matches[1];
}
}
}
if (!empty($rating)) {
$results[$site] = $rating;
$log .= ", get rating: $rating";
} else {
$log .= ", can't get rating";
}
}
//Second, imdb can get from imdb api
if (!isset($results[self::SITE_IMDB]) && !empty($imdbLink)) {
$imdb = new Imdb();
$imdbRating = $imdb->getRating($imdbLink);
$results[self::SITE_IMDB] = $imdbRating;
$log .= ", again 'imdb' from: $imdbLink -> $imdbRating";
}
//Otherwise, get from desc
if (!empty($desc)) {
foreach (self::$validSites as $site => $info) {
if (isset($results[$site])) {
continue;
}
if (empty($info['rating_pattern_in_desc'])) {
continue;
}
$pattern = $info['rating_pattern_in_desc'];
$log .= ", at last, trying to get '$site' from desc with pattern: $pattern";
if (preg_match($pattern, $desc, $matches)) {
$log .= ", get " . $matches[1];
$results[$site] = $matches[1];
} else {
$log .= ", not match";
}
}
}
do_log($log);
return $results;
}
public function updateTorrentPtGen(array $torrentInfo): bool|array
{
$now = Carbon::now();
$log = "torrent: " . $torrentInfo['id'];
$arr = json_decode($torrentInfo['pt_gen'], true);
if (is_array($arr)) {
if (!empty($arr['__updated_at'])) {
$log .= ", updated_at: " . $arr['__updated_at'];
$updatedAt = Carbon::parse($arr['__updated_at']);
$diffInDays = $now->diffInDays($updatedAt);
$log .= ", diffInDays: $diffInDays";
if ($diffInDays < 30) {
do_log("$log, less 30 days, don't update");
return false;
}
}
$link = $this->getLink($arr);
} else {
$link = $torrentInfo['pt_gen'];
}
if (empty($link)) {
do_log("$log, no link...");
return false;
}
$ptGenInfo = [];
foreach (self::$validSites as $site => $siteConfig) {
if (!preg_match($siteConfig['url_pattern'], $link, $matches)) {
continue;
}
try {
$response = $this->generate($matches[0], true);
$ptGenInfo[$site]['data'] = $response;
} catch (\Exception $exception) {
do_log("$log, site: $site can not be updated: " . $exception->getMessage(), 'error');
}
}
$siteIdAndRating = $this->listRatings($ptGenInfo, $torrentInfo['url'], $torrentInfo['descr']);
foreach ($siteIdAndRating as $key => $value) {
$ptGenInfo[$key]['data']["__rating"] = $value;
}
$ptGenInfo['__link'] = $link;
$ptGenInfo['__updated_at'] = $now->toDateTimeString();
Torrent::query()->where('id', $torrentInfo['id'])->update(['pt_gen' => $ptGenInfo]);
do_log("$log, success update");
return $ptGenInfo;
}
public function getLink(array $ptGenInfo)
{
if (isset($ptGenInfo['__link'])) {
//new
return $ptGenInfo['__link'];
}
$result = '';
foreach ($ptGenInfo as $item) {
if (!empty($item['link'])) {
//old, use the last one
$result = $item['link'];
}
}
return $result;
}
private function getRawPTGenRating(array $ptGenInfo, $site)
{
$key = $site . "_rating_average";
if (isset($ptGenInfo[$key])) {
return $ptGenInfo[$key];
}
if ($site == self::SITE_INDIENOVA) {
$parts = preg_split('/[\s:]+/', $ptGenInfo['rate']);
return Arr::last($parts);
}
return '';
}
2022-02-15 17:43:18 +08:00
}