2019-08-17 15:52:33 +01:00
|
|
|
<?php namespace BookStack\Auth\Access;
|
|
|
|
|
|
|
|
use BookStack\Auth\User;
|
|
|
|
use BookStack\Exceptions\UserTokenExpiredException;
|
|
|
|
use BookStack\Exceptions\UserTokenNotFoundException;
|
|
|
|
use Carbon\Carbon;
|
|
|
|
use Illuminate\Database\Connection as Database;
|
2019-09-13 23:58:40 +01:00
|
|
|
use Illuminate\Support\Str;
|
2019-08-17 15:52:33 +01:00
|
|
|
use stdClass;
|
|
|
|
|
|
|
|
class UserTokenService
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Name of table where user tokens are stored.
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $tokenTable = 'user_tokens';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Token expiry time in hours.
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
protected $expiryTime = 24;
|
|
|
|
|
|
|
|
protected $db;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* UserTokenService constructor.
|
|
|
|
* @param Database $db
|
|
|
|
*/
|
|
|
|
public function __construct(Database $db)
|
|
|
|
{
|
|
|
|
$this->db = $db;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete all email confirmations that belong to a user.
|
|
|
|
* @param User $user
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function deleteByUser(User $user)
|
|
|
|
{
|
|
|
|
return $this->db->table($this->tokenTable)
|
|
|
|
->where('user_id', '=', $user->id)
|
|
|
|
->delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the user id from a token, while check the token exists and has not expired.
|
|
|
|
* @param string $token
|
|
|
|
* @return int
|
|
|
|
* @throws UserTokenNotFoundException
|
|
|
|
* @throws UserTokenExpiredException
|
|
|
|
*/
|
|
|
|
public function checkTokenAndGetUserId(string $token) : int
|
|
|
|
{
|
|
|
|
$entry = $this->getEntryByToken($token);
|
|
|
|
|
|
|
|
if (is_null($entry)) {
|
|
|
|
throw new UserTokenNotFoundException('Token "' . $token . '" not found');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->entryExpired($entry)) {
|
2019-08-18 13:55:28 +01:00
|
|
|
throw new UserTokenExpiredException("Token of id {$entry->id} has expired.", $entry->user_id);
|
2019-08-17 15:52:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return $entry->user_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a unique token within the email confirmation database.
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function generateToken() : string
|
|
|
|
{
|
2019-09-13 23:58:40 +01:00
|
|
|
$token = Str::random(24);
|
2019-08-17 15:52:33 +01:00
|
|
|
while ($this->tokenExists($token)) {
|
2019-09-13 23:58:40 +01:00
|
|
|
$token = Str::random(25);
|
2019-08-17 15:52:33 +01:00
|
|
|
}
|
|
|
|
return $token;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate and store a token for the given user.
|
|
|
|
* @param User $user
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function createTokenForUser(User $user) : string
|
|
|
|
{
|
|
|
|
$token = $this->generateToken();
|
|
|
|
$this->db->table($this->tokenTable)->insert([
|
|
|
|
'user_id' => $user->id,
|
|
|
|
'token' => $token,
|
|
|
|
'created_at' => Carbon::now(),
|
|
|
|
'updated_at' => Carbon::now()
|
|
|
|
]);
|
|
|
|
return $token;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the given token exists.
|
|
|
|
* @param string $token
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function tokenExists(string $token) : bool
|
|
|
|
{
|
|
|
|
return $this->db->table($this->tokenTable)
|
|
|
|
->where('token', '=', $token)->exists();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a token entry for the given token.
|
|
|
|
* @param string $token
|
|
|
|
* @return object|null
|
|
|
|
*/
|
|
|
|
protected function getEntryByToken(string $token)
|
|
|
|
{
|
|
|
|
return $this->db->table($this->tokenTable)
|
|
|
|
->where('token', '=', $token)
|
|
|
|
->first();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the given token entry has expired.
|
|
|
|
* @param stdClass $tokenEntry
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function entryExpired(stdClass $tokenEntry) : bool
|
|
|
|
{
|
|
|
|
return Carbon::now()->subHours($this->expiryTime)
|
|
|
|
->gt(new Carbon($tokenEntry->created_at));
|
|
|
|
}
|
2019-09-15 18:29:51 +01:00
|
|
|
}
|