This repository has been archived on 2024-06-26. You can view files and clone it, but cannot push or open issues or pull requests.
sakura/libraries/User.php

1202 lines
32 KiB
PHP
Raw Normal View History

2015-08-18 23:29:45 +00:00
<?php
2016-02-03 22:22:56 +00:00
/**
* Holds the user object class.
*
* @package Sakura
*/
2015-08-18 23:29:45 +00:00
namespace Sakura;
2015-12-29 21:52:19 +00:00
use Sakura\Perms;
use Sakura\Perms\Site;
/**
2016-02-02 21:04:15 +00:00
* Everything you'd ever need from a specific user.
*
* @package Sakura
2016-02-02 21:04:15 +00:00
* @author Julian van de Groep <me@flash.moe>
*/
class User
{
2016-02-02 21:04:15 +00:00
/**
* The User's ID.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $id = 0;
2016-02-02 21:04:15 +00:00
/**
* The user's username.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $username = 'User';
2016-02-02 21:04:15 +00:00
/**
* A cleaned version of the username.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $usernameClean = 'user';
2016-02-02 21:04:15 +00:00
/**
* The user's password hash.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $passwordHash = '';
2016-02-02 21:04:15 +00:00
/**
* The user's password salt.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $passwordSalt = '';
2016-02-02 21:04:15 +00:00
/**
* The user's password algorithm.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $passwordAlgo = 'disabled';
2016-02-02 21:04:15 +00:00
/**
* The password iterations.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $passwordIter = 0;
2016-02-02 21:04:15 +00:00
/**
* UNIX timestamp of last time the password was changed.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $passwordChan = 0;
2016-02-02 21:04:15 +00:00
/**
* The user's e-mail address.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $email = 'user@sakura';
2016-02-02 21:04:15 +00:00
/**
* The rank object of the user's main rank.
*
* @var Rank
*/
2016-01-17 01:58:31 +00:00
public $mainRank = null;
2016-02-02 21:04:15 +00:00
/**
* The ID of the main rank.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $mainRankId = 1;
2016-02-02 21:04:15 +00:00
/**
* The index of rank objects.
*
* @var array
*/
2016-01-17 01:58:31 +00:00
public $ranks = [];
2016-02-02 21:04:15 +00:00
/**
* The user's username colour.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $colour = '';
2016-02-02 21:04:15 +00:00
/**
* The IP the user registered from.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $registerIp = '0.0.0.0';
2016-02-02 21:04:15 +00:00
/**
* The IP the user was last active from.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $lastIp = '0.0.0.0';
2016-02-02 21:04:15 +00:00
/**
* A user's title.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $title = '';
2016-02-02 21:04:15 +00:00
/**
* The UNIX timestamp of when the user registered.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $registered = 0;
2016-02-02 21:04:15 +00:00
/**
* The UNIX timestamp of when the user was last online.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $lastOnline = 0;
2016-02-02 21:04:15 +00:00
/**
* The 2 character country code of a user.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $country = 'XX';
2016-02-02 21:04:15 +00:00
/**
* The File id of the user's avatar.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $avatar = 0;
2016-02-02 21:04:15 +00:00
/**
* The File id of the user's background.
*
* @var int
*/
2016-01-17 01:58:31 +00:00
public $background = 0;
2016-02-02 21:04:15 +00:00
/**
* The FIle id of the user's header.
* @var mixed
*/
2016-01-17 01:58:31 +00:00
public $header = 0;
2016-02-02 21:04:15 +00:00
/**
* The raw userpage of the user.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $page = '';
2016-02-02 21:04:15 +00:00
/**
* The raw signature of the user.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
public $signature = '';
2016-02-02 21:04:15 +00:00
/**
* The user's birthday.
*
* @var string
*/
2016-01-17 01:58:31 +00:00
private $birthday = '0000-00-00';
2016-02-02 21:04:15 +00:00
/**
* The user's permission container.
*
* @var Perms
*/
2015-12-29 21:52:19 +00:00
private $permissions;
2016-02-02 21:04:15 +00:00
/**
* The user's option fields.
*
* @var array
*/
2016-01-17 01:58:31 +00:00
private $optionFields = null;
2016-02-02 21:04:15 +00:00
/**
* The user's profile fields.
*
* @var array
*/
2016-01-17 01:58:31 +00:00
private $profileFields = null;
2016-02-02 21:04:15 +00:00
/**
* The User instance cache array.
*
* @var array
*/
2015-12-29 01:27:49 +00:00
protected static $_userCache = [];
2016-02-02 21:04:15 +00:00
/**
* Cached constructor.
*
* @param int|string $uid The user ID or clean username.
* @param bool $forceRefresh Force a recreation.
*
* @return User Returns a user object.
*/
public static function construct($uid, $forceRefresh = false)
{
2015-12-29 01:27:49 +00:00
// Check if a user object isn't present in cache
if ($forceRefresh || !array_key_exists($uid, self::$_userCache)) {
// If not create a new object and cache it
self::$_userCache[$uid] = new User($uid);
}
// Return the cached object
return self::$_userCache[$uid];
}
2015-08-18 23:29:45 +00:00
2016-02-02 21:04:15 +00:00
/**
* Create a new user.
*
* @param string $username The username of the user.
* @param string $password The password of the user.
* @param string $email The e-mail, used primarily for activation.
* @param array $ranks The ranks assigned to the user on creation.
*
* @return User The newly created user's object.
*/
2016-01-04 20:14:09 +00:00
public static function create($username, $password, $email, $ranks = [2])
{
// Set a few variables
2016-01-17 01:58:31 +00:00
$usernameClean = Utils::cleanString($username, true);
$emailClean = Utils::cleanString($email, true);
2016-01-04 20:14:09 +00:00
$password = Hashing::createHash($password);
2016-01-17 02:08:08 +00:00
2016-01-04 20:14:09 +00:00
// Insert the user into the database
Database::insert('users', [
'username' => $username,
'username_clean' => $usernameClean,
'password_hash' => $password[3],
'password_salt' => $password[2],
'password_algo' => $password[0],
'password_iter' => $password[1],
'email' => $emailClean,
'rank_main' => 0,
2016-01-17 01:58:31 +00:00
'register_ip' => Utils::getRemoteIP(),
'last_ip' => Utils::getRemoteIP(),
2016-01-04 20:14:09 +00:00
'user_registered' => time(),
'user_last_online' => 0,
2016-01-17 01:58:31 +00:00
'user_country' => Utils::getCountryCode(),
2016-01-04 20:14:09 +00:00
]);
// Get the last id
$userId = Database::lastInsertID();
// Create a user object
$user = self::construct($userId);
// Assign the default rank
$user->addRanks($ranks);
// Set the default rank
$user->setMainRank($ranks[0]);
// Return the user object
return $user;
}
2016-02-02 21:04:15 +00:00
/**
* The actual constructor
*
* @param int|string $uid The user ID or clean username.
*/
2015-12-29 01:27:49 +00:00
private function __construct($uid)
{
2015-08-18 23:29:45 +00:00
// Get the user database row
2016-01-17 01:58:31 +00:00
$userRow = Database::fetch(
2015-09-14 21:41:43 +00:00
'users',
false,
[
'user_id' => [$uid, '=', true],
2016-01-17 01:58:31 +00:00
'username_clean' => [Utils::cleanString($uid, true), '=', true],
2015-09-14 21:41:43 +00:00
]
);
2015-08-21 22:07:45 +00:00
2016-01-17 01:58:31 +00:00
// Populate the variables
if ($userRow) {
$this->id = $userRow['user_id'];
$this->username = $userRow['username'];
$this->usernameClean = $userRow['username_clean'];
$this->passwordHash = $userRow['password_hash'];
$this->passwordSalt = $userRow['password_salt'];
$this->passwordAlgo = $userRow['password_algo'];
$this->passwordIter = $userRow['password_iter'];
$this->passwordChan = $userRow['password_chan'];
$this->email = $userRow['email'];
$this->mainRankId = $userRow['rank_main'];
$this->colour = $userRow['user_colour'];
$this->registerIp = $userRow['register_ip'];
$this->lastIp = $userRow['last_ip'];
$this->title = $userRow['user_title'];
$this->registered = $userRow['user_registered'];
$this->lastOnline = $userRow['user_last_online'];
$this->birthday = $userRow['user_birthday'];
$this->country = $userRow['user_country'];
$this->avatar = $userRow['user_avatar'];
$this->background = $userRow['user_background'];
$this->header = $userRow['user_header'];
$this->page = $userRow['user_page'];
$this->signature = $userRow['user_signature'];
2015-08-19 02:37:45 +00:00
}
2016-01-03 21:19:37 +00:00
// Get all ranks
2016-01-17 01:58:31 +00:00
$ranks = Database::fetch('user_ranks', true, ['user_id' => [$this->id, '=']]);
2015-08-18 23:29:45 +00:00
// Get the rows for all the ranks
2016-01-03 21:19:37 +00:00
foreach ($ranks as $rank) {
2015-08-19 02:37:45 +00:00
// Store the database row in the array
2016-01-17 01:58:31 +00:00
$this->ranks[$rank['rank_id']] = Rank::construct($rank['rank_id']);
2015-08-19 02:37:45 +00:00
}
// Check if ranks were set
if (empty($this->ranks)) {
2015-08-19 02:37:45 +00:00
// If not assign the fallback rank
2016-01-04 20:14:09 +00:00
$this->ranks[1] = Rank::construct(1);
2015-08-19 02:37:45 +00:00
}
2016-01-17 01:58:31 +00:00
// Check if the rank is actually assigned to this user
if (!array_key_exists($this->mainRankId, $this->ranks)) {
$this->mainRankId = array_keys($this->ranks)[0];
$this->setMainRank($this->mainRankId);
}
2016-01-17 01:58:31 +00:00
// Assign the main rank to its own var
$this->mainRank = $this->ranks[$this->mainRankId];
2016-01-17 01:58:31 +00:00
// Set user colour
$this->colour = $this->colour ? $this->colour : $this->mainRank->colour;
2016-01-17 01:58:31 +00:00
// Set user title
$this->title = $this->title ? $this->title : $this->mainRank->title;
2015-08-21 22:07:45 +00:00
2016-01-17 01:58:31 +00:00
// Init the permissions
$this->permissions = new Perms(Perms::SITE);
2015-08-19 02:37:45 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Commit changed to database, doesn't do anything yet.
*/
2016-01-22 20:07:44 +00:00
public function update()
{
// placeholder
}
2016-02-02 21:04:15 +00:00
/**
* Get the user's birthday.
*
* @param bool $age Just get the age.
*
* @return int|string Return the birthday.
*/
2016-01-17 01:58:31 +00:00
public function birthday($age = false)
{
2016-01-17 01:58:31 +00:00
// If age is requested calculate it
if ($age) {
// Create dates
$birthday = date_create($this->birthday);
$now = date_create(date('Y-m-d'));
2016-01-17 01:58:31 +00:00
// Get the difference
$diff = date_diff($birthday, $now);
2015-08-19 02:37:45 +00:00
2016-01-17 01:58:31 +00:00
// Return the difference in years
2016-01-22 12:46:52 +00:00
return (int) $diff->format('%Y');
2016-01-17 01:58:31 +00:00
}
2016-01-17 01:58:31 +00:00
// Otherwise just return the birthday value
return $this->birthday;
2015-08-19 02:37:45 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Get the user's country.
*
* @param bool $long Get the full country name.
*
* @return string The country.
*/
2016-01-17 01:58:31 +00:00
public function country($long = false)
{
2016-01-17 01:58:31 +00:00
return $long ? Utils::getCountryName($this->country) : $this->country;
}
2016-02-02 21:04:15 +00:00
/**
* Check if a user is online.
*
* @return bool Are they online?
*/
public function isOnline()
{
// Get all sessions
2016-01-17 01:58:31 +00:00
$sessions = Database::fetch('sessions', true, ['user_id' => [$this->id, '=']]);
// If there's no entries just straight up return false
2015-11-11 21:56:03 +00:00
if (!$sessions) {
return false;
}
// Otherwise use the standard method
2016-01-17 01:58:31 +00:00
return $this->lastOnline > (time() - Config::get('max_online_time'));
}
2016-02-02 21:04:15 +00:00
/**
* Get a few forum statistics.
*
* @return array Post and thread counts.
*/
public function forumStats()
{
return [
'posts' => Database::count(
'posts',
2016-01-17 01:58:31 +00:00
['poster_id' => [$this->id, '=']]
)[0],
'topics' => count(Database::fetch(
'posts',
true,
2016-01-17 01:58:31 +00:00
['poster_id' => [$this->id, '=']],
['post_time'],
null,
['topic_id']
)),
];
2015-08-19 12:13:38 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Get the elapsed string for some of the user's dates
*
* @param string $append Append to the value.
* @param string $none Replace the 0 value with this.
*
* @return array The times.
*/
public function elapsed($append = ' ago', $none = 'Just now')
{
$times = [];
2016-01-17 01:58:31 +00:00
$dates = [
'joined' => $this->registered,
'lastOnline' => $this->lastOnline,
];
2016-01-17 01:58:31 +00:00
foreach ($dates as $key => $val) {
$times[$key] = Utils::timeElapsed($val, $append, $none);
}
return $times;
}
2016-02-02 21:04:15 +00:00
/**
* Add ranks to a user.
*
* @param array $ranks Array containing the rank IDs.
*/
2015-11-11 21:56:03 +00:00
public function addRanks($ranks)
{
// Update the ranks array
2016-01-17 01:58:31 +00:00
$ranks = array_diff(
array_unique(
array_merge(
array_keys($this->ranks),
$ranks)
),
array_keys($this->ranks)
);
// Save to the database
2016-01-03 21:19:37 +00:00
foreach ($ranks as $rank) {
Database::insert('user_ranks', [
'rank_id' => $rank,
2016-01-17 01:58:31 +00:00
'user_id' => $this->id,
2016-01-03 21:19:37 +00:00
]);
}
}
2016-02-02 21:04:15 +00:00
/**
* Remove a set of ranks from a user.
*
* @param array $ranks Array containing the IDs of ranks to remove.
*/
public function removeRanks($ranks)
{
// Current ranks
2016-01-17 01:58:31 +00:00
$remove = array_intersect(array_keys($this->ranks), $ranks);
// Iterate over the ranks
2016-01-03 21:19:37 +00:00
foreach ($remove as $rank) {
2016-01-17 01:58:31 +00:00
Database::delete('user_ranks', ['user_id' => [$this->id, '='], 'rank_id' => [$rank, '=']]);
}
}
2016-02-02 21:04:15 +00:00
/**
* Change the main rank of a user.
*
* @param int $rank The ID of the new main rank.
*
* @return bool Always true.
*/
2015-11-08 22:31:52 +00:00
public function setMainRank($rank)
{
// If it does exist update their row
Database::update('users', [
2015-11-08 22:31:52 +00:00
[
'rank_main' => $rank,
],
[
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
2015-11-08 22:31:52 +00:00
],
]);
// Return true if everything was successful
2015-11-08 22:31:52 +00:00
return true;
}
2016-02-02 21:04:15 +00:00
/**
* Check if a user has a certain set of rank.
*
* @param array $ranks Ranks IDs to check.
*
* @return bool Successful?
*/
2015-11-08 22:31:52 +00:00
public function hasRanks($ranks)
{
// Check if the main rank is the specified rank
2016-01-17 01:58:31 +00:00
if (in_array($this->mainRankId, $ranks)) {
return true;
}
// If not go over all ranks and check if the user has them
foreach ($ranks as $rank) {
// We check if $rank is in $this->ranks and if yes return true
2016-01-17 01:58:31 +00:00
if (in_array($rank, array_keys($this->ranks))) {
return true;
}
}
// If all fails return false
return false;
2015-11-08 22:31:52 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Add a new friend.
*
* @param int $uid The ID of the friend.
*
* @return array Status indicator.
*/
2015-10-12 18:25:37 +00:00
public function addFriend($uid)
{
// Create the foreign object
2015-12-29 01:27:49 +00:00
$user = User::construct($uid);
2015-10-12 18:25:37 +00:00
// Validate that the user exists
2015-12-29 21:52:19 +00:00
if ($user->permission(Site::DEACTIVATED)) {
2015-10-12 18:25:37 +00:00
return [0, 'USER_NOT_EXIST'];
}
// Check if the user already has this user a friend
if ($this->isFriends($uid)) {
2015-10-12 18:25:37 +00:00
return [0, 'ALREADY_FRIENDS'];
}
// Add friend
Database::insert('friends', [
2016-01-17 01:58:31 +00:00
'user_id' => $this->id,
2015-10-12 18:25:37 +00:00
'friend_id' => $uid,
'friend_timestamp' => time(),
]);
// Return true because yay
2016-01-17 01:58:31 +00:00
return [1, $user->isFriends($this->id) ? 'FRIENDS' : 'NOT_MUTUAL'];
2015-10-12 18:25:37 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Remove a friend.
*
* @param int $uid The friend Id
* @param bool $deleteRequest Delete the open request as well (remove you from their friends list).
*
* @return array Status indicator.
*/
2015-10-12 18:25:37 +00:00
public function removeFriend($uid, $deleteRequest = false)
{
// Create the foreign object
2015-12-29 01:27:49 +00:00
$user = User::construct($uid);
2015-10-12 18:25:37 +00:00
// Validate that the user exists
2015-12-29 21:52:19 +00:00
if ($user->permission(Site::DEACTIVATED)) {
2015-10-12 18:25:37 +00:00
return [0, 'USER_NOT_EXIST'];
}
// Remove friend
Database::delete('friends', [
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
2015-10-12 18:25:37 +00:00
'friend_id' => [$uid, '='],
]);
// Attempt to remove the request
if ($deleteRequest) {
Database::delete('friends', [
2016-01-17 01:58:31 +00:00
'friend_id' => [$this->id, '='],
2015-10-12 18:25:37 +00:00
'user_id' => [$uid, '='],
]);
}
// Return true because yay
return [1, 'REMOVED'];
}
2016-02-02 21:04:15 +00:00
/**
* Check if this user is friends with another user.
*
* @param int $with ID of the other user.
*
* @return int 0 = no, 1 = pending request, 2 = mutual
*/
public function isFriends($with)
{
// Accepted from this user
$user = Database::count('friends', [
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
'friend_id' => [$with, '='],
])[0];
// And the other user
$friend = Database::count('friends', [
'user_id' => [$with, '='],
2016-01-17 01:58:31 +00:00
'friend_id' => [$this->id, '='],
])[0];
if ($user && $friend) {
return 2; // Mutual friends
} elseif ($user) {
return 1; // Pending request
}
// Else return 0
return 0;
}
2015-10-12 18:25:37 +00:00
2016-02-02 21:04:15 +00:00
/**
* Get all the friends from this user.
*
* @param int $level Friend level; (figure out what the levels are at some point)
* @param bool $noObj Just return IDs.
*
* @return array The array with either the objects or the ids.
*/
2015-11-11 00:30:22 +00:00
public function friends($level = 0, $noObj = false)
{
// User ID container
$users = [];
// Select the correct level
switch ($level) {
case 2:
// Get all the current user's friends
2016-01-17 01:58:31 +00:00
$self = array_column(Database::fetch('friends', true, ['user_id' => [$this->id, '=']]), 'friend_id');
// Get all the people that added this user as a friend
2016-01-17 01:58:31 +00:00
$others = array_column(Database::fetch('friends', true, ['friend_id' => [$this->id, '=']]), 'user_id');
// Create a difference map
$users = array_intersect($self, $others);
break;
case 1:
2016-01-17 01:58:31 +00:00
$users = array_column(Database::fetch('friends', true, ['user_id' => [$this->id, '=']]), 'friend_id');
break;
case 0:
default:
// Get all the current user's friends
2016-01-17 01:58:31 +00:00
$self = array_column(Database::fetch('friends', true, ['user_id' => [$this->id, '=']]), 'friend_id');
// Get all the people that added this user as a friend
2016-01-17 01:58:31 +00:00
$others = array_column(Database::fetch('friends', true, ['friend_id' => [$this->id, '=']]), 'user_id');
// Create a difference map
$users = array_merge($others, $self);
break;
case -1:
// Get all the current user's friends
2016-01-17 01:58:31 +00:00
$self = array_column(Database::fetch('friends', true, ['user_id' => [$this->id, '=']]), 'friend_id');
// Get all the people that added this user as a friend
2016-01-17 01:58:31 +00:00
$others = array_column(Database::fetch('friends', true, ['friend_id' => [$this->id, '=']]), 'user_id');
// Create a difference map
$users = array_diff($others, $self);
break;
2015-10-12 18:25:37 +00:00
}
2015-11-11 00:30:22 +00:00
// Check if we only requested the IDs
if ($noObj) {
// If so just return $users
return $users;
}
// Create the storage array
$objects = [];
2015-11-11 00:30:22 +00:00
// Create the user objects
foreach ($users as $user) {
// Create new object
2015-12-29 01:27:49 +00:00
$objects[$user] = User::construct($user);
2015-10-12 18:25:37 +00:00
}
// Return the objects
return $objects;
2015-08-19 12:13:38 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Check if the user is banned.
*
* @return array|bool Ban status.
*/
public function checkBan()
{
2016-01-17 01:58:31 +00:00
return Bans::checkBan($this->id);
2015-08-19 12:13:38 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Check if the user has a certaing permission flag.
*
* @param int $flag The permission flag.
* @param string $mode The permission mode.
*
* @return bool Success?
*/
2015-12-29 21:52:19 +00:00
public function permission($flag, $mode = null)
{
2015-12-29 21:52:19 +00:00
// Set mode
$this->permissions->mode($mode ? $mode : Perms::SITE);
// Set default permission value
$perm = 0;
// Bitwise OR it with the permissions for this forum
2016-01-17 01:58:31 +00:00
$perm = $this->permissions->user($this->id);
2015-12-29 21:52:19 +00:00
return $this->permissions->check($flag, $perm);
2015-08-21 22:07:45 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Get the comments from the user's profile.
* @return Comments
*/
public function profileComments()
{
2016-01-17 01:58:31 +00:00
return new Comments('profile-' . $this->id);
}
2016-02-02 21:04:15 +00:00
/**
* Get the user's profile fields.
*
* @return array The profile fields.
*/
public function profileFields()
{
2016-01-17 01:58:31 +00:00
// Check if we have cached data
if ($this->profileFields) {
return $this->profileFields;
2015-08-19 02:37:45 +00:00
}
2016-01-17 01:58:31 +00:00
// Create array and get values
2015-08-19 02:37:45 +00:00
$profile = [];
2016-01-17 01:58:31 +00:00
$profileFields = Database::fetch('profilefields');
$profileValuesRaw = Database::fetch('user_profilefields', true, ['user_id' => [$this->id, '=']]);
$profileValueKeys = array_map(function ($a) {
return $a['field_name'];
}, $profileValuesRaw);
$profileValueVals = array_map(function ($a) {
return $a['field_value'];
}, $profileValuesRaw);
$profileValues = array_combine($profileValueKeys, $profileValueVals);
// Check if anything was returned
if (!$profileFields || !$profileValues) {
return $profile;
}
2015-08-19 02:37:45 +00:00
// Check if profile fields aren't fake
foreach ($profileFields as $field) {
2015-08-19 02:37:45 +00:00
// Completely strip all special characters from the field name
2016-01-17 01:58:31 +00:00
$fieldName = Utils::cleanString($field['field_name'], true, true);
2015-08-19 02:37:45 +00:00
// Check if the user has the current field set otherwise continue
2016-01-17 01:58:31 +00:00
if (!array_key_exists($fieldName, $profileValues)) {
2015-08-19 02:37:45 +00:00
continue;
}
// Assign field to output with value
$profile[$fieldName] = [];
$profile[$fieldName]['name'] = $field['field_name'];
2016-01-17 01:58:31 +00:00
$profile[$fieldName]['value'] = $profileValues[$fieldName];
$profile[$fieldName]['islink'] = $field['field_link'];
2015-08-19 02:37:45 +00:00
// If the field is set to be a link add a value for that as well
if ($field['field_link']) {
2015-09-14 21:41:43 +00:00
$profile[$fieldName]['link'] = str_replace(
'{{ VAL }}',
2016-01-17 01:58:31 +00:00
$profileValues[$fieldName],
$field['field_linkformat']
2015-09-14 21:41:43 +00:00
);
2015-08-19 02:37:45 +00:00
}
// Check if we have additional options as well
if ($field['field_additional'] != null) {
2015-08-19 02:37:45 +00:00
// Decode the json of the additional stuff
$additional = json_decode($field['field_additional'], true);
2015-08-19 02:37:45 +00:00
// Go over all additional forms
foreach ($additional as $subName => $subField) {
2015-08-19 02:37:45 +00:00
// Check if the user has the current field set otherwise continue
2016-01-17 01:58:31 +00:00
if (!array_key_exists($subName, $profileValues)) {
2015-08-19 02:37:45 +00:00
continue;
}
// Assign field to output with value
2016-01-17 01:58:31 +00:00
$profile[$fieldName][$subName] = $profileValues[$subName];
2015-08-19 02:37:45 +00:00
}
}
}
2016-01-17 01:58:31 +00:00
// Assign cache
$this->profileFields = $profile;
2015-08-19 02:37:45 +00:00
// Return appropiate profile data
return $profile;
2015-08-18 23:29:45 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Get a user's option fields.
*
* @return array The array containing the fields.
*/
public function optionFields()
{
2016-01-17 01:58:31 +00:00
// Check if we have cached data
if ($this->optionFields) {
return $this->optionFields;
2015-08-21 22:07:45 +00:00
}
2016-01-17 01:58:31 +00:00
// Create array and get values
2015-08-21 22:07:45 +00:00
$options = [];
2016-01-17 01:58:31 +00:00
$optionFields = Database::fetch('optionfields');
$optionValuesRaw = Database::fetch('user_optionfields', true, ['user_id' => [$this->id, '=']]);
$optionValueKeys = array_map(function ($a) {
return $a['field_name'];
}, $optionValuesRaw);
$optionValueVals = array_map(function ($a) {
return $a['field_value'];
}, $optionValuesRaw);
$optionValues = array_combine($optionValueKeys, $optionValueVals);
2016-01-17 02:08:08 +00:00
2016-01-17 01:58:31 +00:00
// Check if anything was returned
if (!$optionFields || !$optionValues) {
return $options;
}
2015-08-21 22:07:45 +00:00
2016-01-17 01:58:31 +00:00
// Check if option fields aren't fake
foreach ($optionFields as $field) {
2015-08-21 22:07:45 +00:00
// Check if the user has the current field set otherwise continue
2016-01-17 01:58:31 +00:00
if (!array_key_exists($field['option_id'], $optionValues)) {
2015-08-21 22:07:45 +00:00
continue;
}
// Make sure the user has the proper permissions to use this option
2015-12-29 21:52:19 +00:00
if (!$this->permission(constant('Sakura\Perms\Site::' . $field['option_permission']))) {
2015-08-21 22:07:45 +00:00
continue;
}
// Assign field to output with value
2016-01-17 01:58:31 +00:00
$options[$field['option_id']] = $optionValues[$field['option_id']];
2015-08-21 22:07:45 +00:00
}
2016-01-17 02:08:08 +00:00
2016-01-17 01:58:31 +00:00
// Assign cache
$this->optionFields = $options;
2016-01-17 02:08:08 +00:00
2016-01-17 01:58:31 +00:00
// Return appropiate option data
2015-08-21 22:07:45 +00:00
return $options;
}
2016-02-02 21:04:15 +00:00
/**
* Does this user have premium?
*
* @return array Premium status information.
*/
public function isPremium()
{
2015-08-19 12:13:38 +00:00
// Check if the user has static premium
2015-12-29 21:52:19 +00:00
if ($this->permission(Site::STATIC_PREMIUM)) {
2015-08-19 12:13:38 +00:00
return [2, 0, time() + 1];
}
// Attempt to retrieve the premium record from the database
$getRecord = Database::fetch('premium', false, [
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
2015-08-19 12:13:38 +00:00
]);
// If nothing was returned just return false
if (empty($getRecord)) {
2015-08-19 12:13:38 +00:00
return [0];
}
// Check if the Tenshi hasn't expired
if ($getRecord['premium_expire'] < time()) {
return [0, $getRecord['premium_start'], $getRecord['premium_expire']];
2015-08-19 12:13:38 +00:00
}
// Else return the start and expiration date
return [1, $getRecord['premium_start'], $getRecord['premium_expire']];
2015-08-19 12:13:38 +00:00
}
2016-02-02 21:04:15 +00:00
/**
* Get the open warnings on this user.
*
* @return array The warnings.
*/
public function getWarnings()
{
2015-08-19 12:13:38 +00:00
// Do the database query
$getWarnings = Database::fetch('warnings', true, [
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
2015-08-19 12:13:38 +00:00
]);
// Storage array
$warnings = [];
// Add special stuff
foreach ($getWarnings as $warning) {
// Check if it hasn't expired
if ($warning['warning_expires'] < time()) {
Database::delete('warnings', ['warning_id' => [$warning['warning_id'], '=']]);
continue;
}
// Text action
switch ($warning['warning_action']) {
default:
case '0':
$warning['warning_action_text'] = 'Warning';
break;
case '1':
$warning['warning_action_text'] = 'Silence';
break;
case '2':
$warning['warning_action_text'] = 'Restriction';
break;
case '3':
$warning['warning_action_text'] = 'Ban';
break;
case '4':
$warning['warning_action_text'] = 'Abyss';
break;
}
// Text expiration
$warning['warning_length'] = round(($warning['warning_expires'] - $warning['warning_issued']) / 60);
// Add to array
$warnings[$warning['warning_id']] = $warning;
}
2015-08-19 12:13:38 +00:00
// Return all the warnings
return $warnings;
}
2016-02-02 21:04:15 +00:00
/**
* Parse the user's userpage.
*
* @return string The parsed page.
*/
public function userPage()
{
2016-01-20 23:06:21 +00:00
return BBcode::toHTML(htmlentities($this->page));
}
2016-02-02 21:04:15 +00:00
/**
* Parse a user's signature
*
* @return string The parsed signature.
*/
public function signature()
{
2016-01-20 23:06:21 +00:00
return BBcode::toHTML(htmlentities($this->signature));
}
2016-02-02 21:04:15 +00:00
/**
* Get a user's username history.
*
* @return array The history.
*/
public function getUsernameHistory()
{
// Do the database query
$changes = Database::fetch('username_history', true, [
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
], ['change_id', true]);
// Return all the warnings
return $changes;
}
2016-02-02 21:04:15 +00:00
/**
* Alter the user's username
*
* @param string $username The new username.
*
* @return array Status indicator.
*/
public function setUsername($username)
{
// Create a cleaned version
2016-01-17 01:58:31 +00:00
$username_clean = Utils::cleanString($username, true);
// Check if the username is too short
2015-12-04 14:19:10 +00:00
if (strlen($username_clean) < Config::get('username_min_length')) {
return [0, 'TOO_SHORT'];
}
// Check if the username is too long
2015-12-04 14:19:10 +00:00
if (strlen($username_clean) > Config::get('username_max_length')) {
return [0, 'TOO_LONG'];
}
// Check if this username hasn't been used in the last amount of days set in the config
$getOld = Database::fetch('username_history', false, [
'username_old_clean' => [$username_clean, '='],
2015-12-04 14:19:10 +00:00
'change_time' => [(Config::get('old_username_reserve') * 24 * 60 * 60), '>'],
], ['change_id', true]);
// Check if anything was returned
2016-01-17 01:58:31 +00:00
if ($getOld && $getOld['user_id'] != $this->id) {
return [0, 'TOO_RECENT', $getOld['change_time']];
}
// Check if the username is already in use
$getInUse = Database::fetch('users', false, [
'username_clean' => [$username_clean, '='],
]);
// Check if anything was returned
if ($getInUse) {
return [0, 'IN_USE', $getInUse['user_id']];
}
// Insert into username_history table
Database::insert('username_history', [
'change_time' => time(),
2016-01-17 01:58:31 +00:00
'user_id' => $this->id,
'username_new' => $username,
'username_new_clean' => $username_clean,
2016-01-17 01:58:31 +00:00
'username_old' => $this->username,
'username_old_clean' => $this->usernameClean,
]);
// Update userrow
Database::update('users', [
[
'username' => $username,
'username_clean' => $username_clean,
],
[
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
],
]);
// Return success
return [1, 'SUCCESS', $username];
}
2016-02-02 21:04:15 +00:00
/**
* Alter a user's e-mail address
*
* @param string $email The new e-mail address.
*
* @return array Status indicator.
*/
public function setEMailAddress($email)
{
// Validate e-mail address
if (!filter_var($email, FILTER_VALIDATE_EMAIL)) {
return [0, 'INVALID'];
}
// Check if the username is already in use
$getInUse = Database::fetch('users', false, [
'email' => [$email, '='],
]);
// Check if anything was returned
if ($getInUse) {
return [0, 'IN_USE', $getInUse['user_id']];
}
// Update userrow
Database::update('users', [
[
'email' => $email,
],
[
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
],
]);
// Return success
return [1, 'SUCCESS', $email];
}
2016-02-02 21:04:15 +00:00
/**
* Change the user's password
*
* @param string $old The old password.
* @param string $new The new password
* @param string $confirm The new one again.
*
* @return array Status indicator.
*/
public function setPassword($old, $new, $confirm)
{
// Validate password
2016-01-17 01:58:31 +00:00
switch ($this->passwordAlgo) {
// Disabled account
case 'disabled':
return [0, 'NO_LOGIN'];
// Default hashing method
default:
if (!Hashing::validatePassword($old, [
2016-01-17 01:58:31 +00:00
$this->passwordAlgo,
$this->passwordIter,
$this->passwordSalt,
$this->passwordHash,
])) {
2016-01-17 01:58:31 +00:00
return [0, 'INCORRECT_PASSWORD', $this->passwordChan];
}
}
// Check password entropy
2016-01-17 01:58:31 +00:00
if (Utils::pwdEntropy($new) < Config::get('min_entropy')) {
return [0, 'PASS_TOO_SHIT'];
}
// Passwords do not match
if ($new != $confirm) {
return [0, 'PASS_NOT_MATCH'];
}
// Create hash
$password = Hashing::createHash($new);
// Update userrow
Database::update('users', [
[
'password_hash' => $password[3],
'password_salt' => $password[2],
'password_algo' => $password[0],
'password_iter' => $password[1],
'password_chan' => time(),
],
[
2016-01-17 01:58:31 +00:00
'user_id' => [$this->id, '='],
],
]);
// Return success
return [1, 'SUCCESS'];
}
2015-08-18 23:29:45 +00:00
}