mirror of
https://github.com/OPSnet/Gazelle.git
synced 2026-01-16 18:04:34 -05:00
395 lines
17 KiB
PHP
395 lines
17 KiB
PHP
<?php
|
|
// phpcs:disable PSR1.Files.SideEffects.FoundWithSymbols
|
|
// phpcs:disable Generic.WhiteSpace.ScopeIndent.IncorrectExact
|
|
// phpcs:disable Generic.WhiteSpace.ScopeIndent.Incorrect
|
|
/** @phpstan-var \Gazelle\User $Viewer */
|
|
/** @phpstan-var \Gazelle\Cache $Cache */
|
|
/** @phpstan-var \Twig\Environment $Twig */
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Gazelle;
|
|
|
|
use Gazelle\Enum\BonusItemPurchaseStatus;
|
|
use Gazelle\Enum\UserTokenType;
|
|
use Gazelle\User\Vote;
|
|
|
|
$userMan = new Manager\User();
|
|
$user = $userMan->findById((int)$_GET['id']);
|
|
if (is_null($user)) {
|
|
header("Location: log.php?search=User+" . (int)$_GET['id']);
|
|
exit;
|
|
}
|
|
|
|
$userId = $user->id;
|
|
$username = $user->username();
|
|
$Class = $user->primaryClass();
|
|
$donor = new User\Donor($user);
|
|
$userBonus = new User\Bonus($user);
|
|
$viewerBonus = new User\Bonus($Viewer);
|
|
$history = new User\History($user);
|
|
$limiter = new User\UserclassRateLimit($user);
|
|
$bonusMan = new Manager\Bonus();
|
|
$donorMan = new Manager\Donation();
|
|
$ipv4 = new Manager\IPv4();
|
|
$tgMan = new Manager\TGroup();
|
|
|
|
$resetToken = $Viewer->permitted('users_mod')
|
|
? new Manager\UserToken()->findByUser($user, UserTokenType::password)
|
|
: null;
|
|
|
|
if (isset($_POST['flsubmit'], $_POST['fltype']) && ($_POST['action'] ?? '') === 'fltoken') {
|
|
authorize();
|
|
if (str_starts_with($_POST['fltype'], 'fl-')) {
|
|
// backwards compatibility
|
|
$_POST['fltype'] = substr($_POST['fltype'], 3);
|
|
}
|
|
$status = $bonusMan->purchaseTokenOther(
|
|
$Viewer, $user, $_POST['fltype'], $_POST['message'] ?? ''
|
|
);
|
|
if ($status === BonusItemPurchaseStatus::success) {
|
|
header("Location: {$user->location()}");
|
|
exit;
|
|
} else {
|
|
Error400::error(
|
|
match ($status) { /* @phpstan-ignore match.unhandled (alreadyPurchased) */
|
|
BonusItemPurchaseStatus::declined
|
|
=> "{$user->username()} does not wish to receive tokens",
|
|
BonusItemPurchaseStatus::insufficientFunds
|
|
=> "You do not have enough bonus points to buy that",
|
|
BonusItemPurchaseStatus::forbidden
|
|
=> "{$user->username()} cannot receive tokens from you",
|
|
BonusItemPurchaseStatus::incomplete
|
|
=> "Something was wrong with the parameters provided",
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
if ($userId == $Viewer->id) {
|
|
$Preview = (bool)($_GET['preview'] ?? false);
|
|
$OwnProfile = !$Preview;
|
|
$user->forceCacheFlush(true);
|
|
} else {
|
|
$OwnProfile = false;
|
|
// Don't allow any kind of previewing on other profiles
|
|
$Preview = false;
|
|
}
|
|
$previewer = $Preview ? $userMan->findById(PARANOIA_PREVIEW_USER) : $Viewer;
|
|
$Paranoia = $Preview ? explode(',', $_GET['paranoia']) : $user->paranoia();
|
|
|
|
function check_paranoia_here(?string $Setting): int|false {
|
|
global $Paranoia, $Class, $userId, $Preview;
|
|
if (!$Setting) {
|
|
return PARANOIA_ALLOWED;
|
|
}
|
|
if ($Preview) {
|
|
return check_paranoia($Setting, $Paranoia ?? [], $Class);
|
|
} else {
|
|
return check_paranoia($Setting, $Paranoia ?? [], $Class, $userId);
|
|
}
|
|
}
|
|
|
|
View::show_header($username, [
|
|
'js' => 'vendor/jquery.imagesloaded,vendor/jquery.wookmark,bbcode,comments,lastfm,requests,user'
|
|
. ($Viewer->isStaff() ? ',info_paster' : '')
|
|
. ($Viewer->permitted('users_view_ips') ? ',resolve-ip' : '')
|
|
. ($Viewer->permitted('users_mod') ? ',reports' : ''),
|
|
'css' => 'tiles'
|
|
]);
|
|
echo $Twig->render('user/header.twig', [
|
|
'bonus' => $userBonus,
|
|
'donor' => $donor,
|
|
'freeleech' => [
|
|
'offer' => $OwnProfile ? [] : $bonusMan->offerTokenOther($Viewer),
|
|
'latest' => $viewerBonus->otherLatest($user),
|
|
],
|
|
'friend' => new User\Friend($Viewer),
|
|
'preview_user' => $previewer,
|
|
'user' => $user,
|
|
'viewer' => $Viewer,
|
|
]);
|
|
|
|
echo $Twig->render('user/sidebar.twig', [
|
|
'ancestry' => $userMan->ancestry($user),
|
|
'applicant' => new Manager\Applicant(),
|
|
'invite_source' => $Viewer->permitted('admin_manage_invite_source')
|
|
? new Manager\InviteSource()->findSourceNameByUser($user) : null,
|
|
'next_class' => $user->nextClass($userMan),
|
|
'user' => $user,
|
|
'viewer' => $Viewer,
|
|
]);
|
|
|
|
// Last.fm statistics and comparability
|
|
$lastfmInfo = new Util\LastFM()->userInfo($user);
|
|
if ($lastfmInfo) {
|
|
echo $Twig->render('user/lastfm.twig', [
|
|
'can_reload' => ($OwnProfile && $Cache->get_value("lastfm_clear_cache_$userId") === false) || $Viewer->permitted('users_mod'),
|
|
'info' => $lastfmInfo,
|
|
'own_profile' => $OwnProfile,
|
|
]);
|
|
}
|
|
|
|
$vote = new Vote($user);
|
|
$stats = $user->stats();
|
|
$Uploads = check_paranoia_here('uploads+') ? $stats->uploadTotal() : 0;
|
|
$rank = new UserRank(
|
|
new UserRank\Configuration(RANKING_WEIGHT),
|
|
[
|
|
'uploaded' => $user->uploadedSize(),
|
|
'downloaded' => $user->downloadedSize(),
|
|
'uploads' => $Uploads,
|
|
'requests' => $stats->requestBountyTotal(),
|
|
'posts' => $stats->forumPostTotal(),
|
|
'bounty' => $stats->requestVoteSize(),
|
|
'artists' => check_paranoia_here('artistsadded') ? $stats->artistAddedTotal() : 0,
|
|
'collage-add' => check_paranoia_here('collagecontribs+') ? $stats->collageContrib() : 0,
|
|
'collage-create' => check_paranoia_here('collagecontribs+') ? $stats->collageTotal() : 0,
|
|
'votes' => $vote->userTotal(Vote::UPVOTE | Vote::DOWNVOTE),
|
|
'bonus' => $userBonus->pointsSpent(),
|
|
'comment-t' => check_paranoia_here('torrentcomments++') ? $stats->commentTotal('torrents') : 0,
|
|
],
|
|
);
|
|
|
|
$byteFormatter = fn ($value) => byte_format($value);
|
|
$numberFormatter = fn ($value) => number_format($value);
|
|
|
|
$statList = [
|
|
// [dimension, permission, title, formatter, tooltip suffix]
|
|
['uploaded', 'uploaded', 'Data uploaded', $byteFormatter, 'uploaded'],
|
|
['downloaded', 'downloaded', 'Data downloaded', $byteFormatter, 'downloaded'],
|
|
['uploads', 'uploads+', 'Torrents uploaded', $numberFormatter, 'uploads'],
|
|
['requests', 'requestsfilled_count', 'Requests filled', $numberFormatter, 'filled'],
|
|
['bounty', 'requestsvoted_bounty', 'Request votes', $byteFormatter, 'spent'],
|
|
['posts', null, 'Forum posts made', $numberFormatter, 'posts'],
|
|
['comment-t', 'torrentcomments++', 'Torrent comments', $numberFormatter, 'posted'],
|
|
['collage-add', 'collagecontribs+', 'Collage contributions', $numberFormatter, 'contributions'],
|
|
['collage-create', 'collagecontribs+', 'Collages created', $numberFormatter, 'created'],
|
|
['artists', 'artistsadded', 'Artists added', $numberFormatter, 'added'],
|
|
['votes', null, 'Release votes cast', $numberFormatter, 'votes'],
|
|
]
|
|
?>
|
|
<div class="box box_info box_userinfo_percentile">
|
|
<div class="head colhead_dark">Percentile Rankings (hover for values)</div>
|
|
<ul class="stats nobullet">
|
|
<?php
|
|
foreach ($statList as $item) {
|
|
if (($Override = check_paranoia_here($item[1]))) {
|
|
?>
|
|
<li class="tooltip<?=($Override === 2 ? ' paranoia_override' : '')?>" title="<?= $item[3]($rank->raw($item[0])) ?> <?= $item[4] ?>"><?= $item[2] ?>: <?= $rank->rank($item[0]) ?></li>
|
|
<?php
|
|
}
|
|
}
|
|
if ($OwnProfile || $Viewer->permitted('admin_bp_history')) { ?>
|
|
<li class="tooltip<?= !$OwnProfile && $Viewer->permitted('admin_bp_history') ? ' paranoia_override' : '' ?>" title="<?=number_format($rank->raw('bonus')) ?> spent">Bonus points spent: <?= $rank->rank('bonus') ?></li>
|
|
<?php
|
|
}
|
|
if ($user->propertyVisibleMulti($previewer, ['artistsadded', 'collagecontribs+', 'downloaded', 'requestsfilled_count', 'requestsvoted_bounty', 'torrentcomments++', 'uploaded', 'uploads+', ])) {
|
|
?>
|
|
<li<?= $user->classLevel() >= 900 ? ' title="Infinite"' : '' ?>><strong>Overall rank: <?= is_null($rank->score())
|
|
? 'Server busy'
|
|
: ($user->classLevel() >= 900 ? ' ∞' : number_format($rank->score() * $user->rankFactor())) ?></strong></li>
|
|
<?php } ?>
|
|
</ul>
|
|
</div>
|
|
<?php if ($Viewer->permitted('users_mod') || $Viewer->permitted('users_view_ips') || $Viewer->permitted('users_view_keys')) { ?>
|
|
<div class="box box_info box_userinfo_history">
|
|
<div class="head colhead_dark">History</div>
|
|
<ul class="stats nobullet">
|
|
<?php if ($Viewer->permitted('users_view_email')) { ?>
|
|
<li>Emails: <?=number_format($history->emailTotal())?> <a href="userhistory.php?action=email&userid=<?=$userId?>" class="brackets">View</a></li>
|
|
<?php
|
|
}
|
|
if ($Viewer->permitted('users_view_ips')) {
|
|
?>
|
|
<li>IPs: <?=number_format($ipv4->userTotal($user))?> <a href="userhistory.php?action=ips&userid=<?=$userId?>" class="brackets">View</a> <a href="userhistory.php?action=ips&userid=<?=$userId?>&usersonly=1" class="brackets">View users</a></li>
|
|
<?php if ($Viewer->permitted('users_mod')) { ?>
|
|
<li>Tracker IPs: <?=number_format($user->trackerIPCount())?> <a href="userhistory.php?action=tracker_ips&userid=<?=$userId?>" class="brackets">View</a></li>
|
|
<?php
|
|
}
|
|
}
|
|
if ($Viewer->permitted('users_view_keys')) {
|
|
?>
|
|
<li>Announce keys: <?=number_format($user->history()->announceKeyTotal())?> <a href="userhistory.php?action=passkeys&userid=<?=$userId?>" class="brackets">View</a></li>
|
|
<?php
|
|
}
|
|
if ($Viewer->permitted('users_mod')) {
|
|
if ($resetToken instanceof User\Token) {
|
|
?>
|
|
<li><span class="tooltip" title="User requested a password reset by email">Password reset expiry: <?= time_diff($resetToken->expiry()) ?></li>
|
|
<?php } ?>
|
|
<li>Password history: <?=number_format($user->history()->passwordTotal())?> <a href="userhistory.php?action=passwords&userid=<?=$userId?>" class="brackets">View</a></li>
|
|
<li>Stats: N/A <a href="userhistory.php?action=stats&userid=<?=$userId?>" class="brackets">View</a></li>
|
|
<?php } ?>
|
|
</ul>
|
|
</div>
|
|
<?php } ?>
|
|
|
|
<?php
|
|
|
|
if (check_paranoia_here('snatched')) {
|
|
echo $Twig->render('user/tag-snatch.twig', [
|
|
'user' => $user,
|
|
]);
|
|
}
|
|
|
|
echo $Twig->render('user/sidebar-stats.twig', [
|
|
'prl' => $limiter,
|
|
'upload_total' => $Uploads,
|
|
'user' => $user,
|
|
'viewer' => $Viewer,
|
|
'visible' => [
|
|
'collages+' => check_paranoia_here('collages+'),
|
|
'collages' => check_paranoia_here('collages'),
|
|
'collagescontrib+' => check_paranoia_here('collagecontribs+'),
|
|
'collagecontribs' => check_paranoia_here('collagecontribs'),
|
|
'downloaded' => $OwnProfile || $Viewer->permitted('site_view_torrent_snatchlist'),
|
|
'invitedcount' => check_paranoia_here('invitedcount'),
|
|
'leeching+' => check_paranoia_here('leeching+'),
|
|
'leeching' => check_paranoia_here('leeching'),
|
|
'perfectflacs+' => check_paranoia_here('perfectflacs+'),
|
|
'perfectflacs' => check_paranoia_here('perfectflacs'),
|
|
'seeding+' => check_paranoia_here('seeding+'),
|
|
'seeding' => check_paranoia_here('seeding'),
|
|
'snatched+' => check_paranoia_here('snatched+'),
|
|
'snatched' => check_paranoia_here('snatched'),
|
|
'torrentcomments+' => check_paranoia_here('torrentcomments+'),
|
|
'torrentcomments' => check_paranoia_here('torrentcomments'),
|
|
'requestsfilled_list' => check_paranoia_here('requestsfilled_list'),
|
|
'requestsfilled_count' => check_paranoia_here('requestsfilled_count'),
|
|
'requestsfilled_bounty' => check_paranoia_here('requestsfilled_bounty'),
|
|
'requestsvoted_list' => check_paranoia_here('requestsvoted_list'),
|
|
'requestsvoted_count' => check_paranoia_here('requestsvoted_count'),
|
|
'requestsvoted_bounty' => check_paranoia_here('requestsvoted_bounty'),
|
|
'uniquegroups+' => check_paranoia_here('uniquegroups+'),
|
|
'uniquegroups' => check_paranoia_here('uniquegroups'),
|
|
'uploads+' => check_paranoia_here('uploads+'),
|
|
'uploads' => check_paranoia_here('uploads'),
|
|
],
|
|
]);
|
|
|
|
if ($Viewer->permitted("users_mod") || $OwnProfile || $donor->isVisible()) {
|
|
echo $Twig->render('donation/stats.twig', [
|
|
'donor' => $donor,
|
|
'viewer' => $Viewer,
|
|
]);
|
|
}
|
|
?>
|
|
</div>
|
|
<div class="main_column">
|
|
<?php
|
|
if ($Viewer->permitted('users_mod') && $user->onRatioWatch()) {
|
|
echo $Twig->render('user/ratio-watch.twig', [
|
|
'user' => $user,
|
|
]);
|
|
}
|
|
?>
|
|
<div class="box">
|
|
<div class="head">
|
|
<?= html_escape($user->profileTitle()) ?>
|
|
<span style="float: right;"><a href="#" onclick="$('#profilediv').gtoggle(); this.innerHTML = (this.innerHTML == 'Hide' ? 'Show' : 'Hide'); return false;" class="brackets">Hide</a></span>
|
|
</div>
|
|
<div class="pad profileinfo" id="profilediv">
|
|
<?= $user->profileInfo() ? \Text::full_format($user->profileInfo()) : 'This profile is currently empty.' ?>
|
|
</div>
|
|
</div>
|
|
<?php
|
|
foreach (range(1, 4) as $level) {
|
|
$profileInfo = $donor->profileInfo($level);
|
|
if (is_string($profileInfo)) {
|
|
?>
|
|
<div class="box">
|
|
<div class="head">
|
|
<?= html_escape($donor->profileTitle($level) ?? "Extra Info $level") ?>
|
|
<span style="float: right;"><a href="#" onclick="$('#profilediv_<?= $level ?>').gtoggle(); this.innerHTML = (this.innerHTML == 'Hide' ? 'Show' : 'Hide'); return false;" class="brackets">Hide</a></span>
|
|
</div>
|
|
<div class="pad profileinfo" id="profilediv_<?= $level ?>"><?= \Text::full_format($profileInfo) ?></div>
|
|
</div>
|
|
<?php
|
|
}
|
|
}
|
|
|
|
if (check_paranoia_here('snatched')) {
|
|
echo $Twig->render('user/recent.twig', [
|
|
'id' => $userId,
|
|
'recent' => array_map(fn ($id) => $tgMan->findById($id), $user->snatch()->recentSnatchList()),
|
|
'title' => 'Snatches',
|
|
'type' => 'snatched',
|
|
'thing' => 'snatches',
|
|
]);
|
|
}
|
|
|
|
if (check_paranoia_here('uploads')) {
|
|
echo $Twig->render('user/recent.twig', [
|
|
'id' => $userId,
|
|
'recent' => array_map(fn ($id) => $tgMan->findById($id), $user->recentUploadList()),
|
|
'title' => 'Uploads',
|
|
'type' => 'uploaded',
|
|
'thing' => 'uploads',
|
|
]);
|
|
}
|
|
|
|
if ($OwnProfile || !$user->hasAttr('hide-vote-recent') || $Viewer->permitted('view-release-votes')) {
|
|
echo $Twig->render('user/recent-vote.twig', [
|
|
'recent' => $vote->recent($tgMan),
|
|
'show_link' => $OwnProfile || !$user->hasAttr('hide-vote-history') || $Viewer->permitted('view-release-votes'),
|
|
'user_id' => $userId,
|
|
]);
|
|
}
|
|
|
|
echo $Twig->render('user/collage-list.twig', [
|
|
'list' => new Manager\Collage()->findPersonalByUser($user),
|
|
'manager' => $tgMan,
|
|
]);
|
|
|
|
echo $Twig->render('user/information.twig', [
|
|
'hash' => signature($comments ?? '', USER_EDIT_SALT),
|
|
'previewer' => $previewer,
|
|
'user' => $user,
|
|
'viewer' => $Viewer,
|
|
|
|
// optional (possibly expensive) elements requiring additional privileges
|
|
'donor_history' => $Viewer->permitted('users_give_donor')
|
|
? $donor->historyList() : [],
|
|
|
|
'has_invitees' => $Viewer->permitted('users_view_invites')
|
|
? new User\InviteTree($user)->hasInvitees() : [],
|
|
|
|
'report_list' => $Viewer->permitted('admin_reports')
|
|
? new Manager\Report($userMan)->findByReportedUser($user) : [],
|
|
|
|
'user_link' => $Viewer->permitted('users_linked_users')
|
|
? new User\UserLink($user)->info() : [],
|
|
|
|
'request_list' => !$Viewer->disableRequests() && $user->propertyVisible($previewer, 'requestsvoted_list')
|
|
? new Manager\Request()->findUnfilledByUser($user, 100) : [],
|
|
|
|
'staffpm_list' => $Viewer->permitted('users_mod') || $Viewer->isStaffPMReader()
|
|
? new Staff($Viewer)->userStaffPmList($user) : [],
|
|
|
|
'warning_list' => $Viewer->permitted('users_warn')
|
|
? $user->forumWarning() : [],
|
|
|
|
'auto_list' => $Viewer->permitted('users_auto_reports')
|
|
? (function ($u) {
|
|
$raTypeMan = new \Gazelle\Manager\ReportAutoType();
|
|
$raSearch = new Search\ReportAuto(new \Gazelle\Manager\ReportAuto($raTypeMan), $raTypeMan);
|
|
return $raSearch->setUser($u)->setState(Enum\ReportAutoState::open)->userTotalList(0);
|
|
})($user)
|
|
: [],
|
|
]);
|
|
|
|
echo $Twig->render('user/main-column.twig', [
|
|
'asn' => new Search\ASN(),
|
|
'class_list' => $userMan->classLevelList(),
|
|
'donor' => $donor,
|
|
'forum_man' => new Manager\Forum(),
|
|
'history' => $history,
|
|
'invite_source' => new Manager\InviteSource()->inviterConfiguration($user),
|
|
'is_traced' => $Viewer->permitted('admin_tracker') && new Tracker()->isTraced($user),
|
|
'over_usercap' => new Stats\Users()->overUsercap(),
|
|
'prl' => $limiter,
|
|
'user' => $user,
|
|
'viewer' => $Viewer,
|
|
]);
|