Files
wallabag/src/Entity/User.php

388 lines
8.8 KiB
PHP
Raw Normal View History

2015-01-22 17:18:56 +01:00
<?php
2024-02-19 01:30:12 +01:00
namespace Wallabag\Entity;
2015-01-22 17:18:56 +01:00
use Doctrine\Common\Collections\ArrayCollection;
2015-01-22 17:18:56 +01:00
use Doctrine\ORM\Mapping as ORM;
2017-07-01 09:52:38 +02:00
use FOS\UserBundle\Model\User as BaseUser;
use JMS\Serializer\Annotation\Accessor;
use JMS\Serializer\Annotation\Groups;
use JMS\Serializer\Annotation\XmlRoot;
use Nelmio\ApiDocBundle\Annotation\Model;
use OpenApi\Annotations as OA;
2018-12-03 06:51:06 +01:00
use Scheb\TwoFactorBundle\Model\BackupCodeInterface;
use Scheb\TwoFactorBundle\Model\Email\TwoFactorInterface as EmailTwoFactorInterface;
use Scheb\TwoFactorBundle\Model\Google\TwoFactorInterface as GoogleTwoFactorInterface;
use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity;
2024-02-19 01:30:12 +01:00
use Wallabag\Entity\Api\Client;
use Wallabag\Helper\EntityTimestampsTrait;
2025-04-05 12:55:51 +02:00
use Wallabag\Repository\UserRepository;
2015-01-22 17:18:56 +01:00
/**
2015-05-30 13:52:26 +02:00
* User.
2015-01-22 17:18:56 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Table(name: '`user`')]
#[ORM\Entity(repositoryClass: UserRepository::class)]
#[ORM\HasLifecycleCallbacks]
2025-04-05 15:06:57 +02:00
#[UniqueEntity('email')]
#[UniqueEntity('username')]
2025-04-05 15:12:30 +02:00
#[XmlRoot('user')]
2018-12-03 06:51:06 +01:00
class User extends BaseUser implements EmailTwoFactorInterface, GoogleTwoFactorInterface, BackupCodeInterface
2015-01-22 17:18:56 +01:00
{
use EntityTimestampsTrait;
/** @Serializer\XmlAttribute */
2015-01-22 17:18:56 +01:00
/**
2015-05-30 13:52:26 +02:00
* @var int
2015-01-22 17:18:56 +01:00
*
* @OA\Property(
* description="The unique numeric id of the user",
* type="int",
* example=12,
* )
2015-01-22 17:18:56 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(name: 'id', type: 'integer')]
#[ORM\Id]
#[ORM\GeneratedValue(strategy: 'AUTO')]
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
2015-08-18 11:08:45 +02:00
protected $id;
2015-01-22 17:18:56 +01:00
/**
2023-11-16 09:36:47 +01:00
* @var string|null
2015-01-22 17:18:56 +01:00
*
* @OA\Property(
* description="The personal Name of the user",
* type="string",
* example="Walla Baggger",
* )
2015-01-22 17:18:56 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(name: 'name', type: 'text', nullable: true)]
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
2015-08-18 11:08:45 +02:00
protected $name;
2015-03-07 23:25:36 +01:00
/**
* @var string
*
* @OA\Property(
* description="The unique username of the user",
* type="string",
* example="wallabag",
* )
*/
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
protected $username;
/**
* @var string
*
* @OA\Property(
* description="E-mail address of the user",
* type="string",
* example="wallabag@wallabag.io",
* )
*/
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
protected $email;
2015-02-06 14:18:01 +01:00
/**
2017-06-01 09:30:20 +02:00
* @var \DateTime
2015-02-06 14:18:01 +01:00
*
* @OA\Property(
* description="Creation date of the user account. (In ISO 8601 format)",
* type="string",
* example="2023-06-27T19:25:44+0000",
* )
2015-02-06 14:18:01 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(name: 'created_at', type: 'datetime')]
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
2015-08-18 11:08:45 +02:00
protected $createdAt;
2015-02-06 14:18:01 +01:00
/**
2017-06-01 09:30:20 +02:00
* @var \DateTime
2015-02-06 14:18:01 +01:00
*
* @OA\Property(
* description="Update date of the user account. (In ISO 8601 format)",
* type="string",
* example="2023-06-27T19:37:30+0000",
* )
2015-02-06 14:18:01 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(name: 'updated_at', type: 'datetime')]
2025-04-05 15:12:30 +02:00
#[Groups(['user_api', 'user_api_with_client'])]
2015-08-18 11:08:45 +02:00
protected $updatedAt;
2015-02-06 14:18:01 +01:00
2025-04-05 12:55:51 +02:00
#[ORM\OneToMany(targetEntity: Entry::class, mappedBy: 'user', cascade: ['remove'])]
2015-08-18 11:08:45 +02:00
protected $entries;
2025-04-05 12:55:51 +02:00
#[ORM\OneToOne(targetEntity: Config::class, mappedBy: 'user', cascade: ['remove'])]
2015-08-18 11:08:45 +02:00
protected $config;
/**
2024-11-25 09:27:39 +01:00
* @var ArrayCollection&iterable<SiteCredential>
*/
2025-04-05 12:55:51 +02:00
#[ORM\OneToMany(targetEntity: SiteCredential::class, mappedBy: 'user', cascade: ['remove'])]
2017-07-03 11:49:46 +02:00
protected $siteCredentials;
2015-10-13 22:43:15 +02:00
/**
2024-11-25 09:27:39 +01:00
* @var ArrayCollection&iterable<Client>
2015-10-13 22:43:15 +02:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\OneToMany(targetEntity: Client::class, mappedBy: 'user', cascade: ['remove'])]
2017-07-01 09:52:38 +02:00
protected $clients;
2015-10-13 22:43:15 +02:00
/**
2017-07-01 09:52:38 +02:00
* @see getFirstClient() below
*
* @OA\Property(
* description="Default client created during user registration. Used for further authorization",
* ref=@Model(type=Client::class, groups={"user_api_with_client"})
* )
2015-10-13 22:43:15 +02:00
*/
2025-04-05 15:12:30 +02:00
#[Groups(['user_api_with_client'])]
#[Accessor(getter: 'getFirstClient')]
2017-07-01 09:52:38 +02:00
protected $default_client;
2015-10-13 22:43:15 +02:00
2025-04-05 12:55:51 +02:00
#[ORM\Column(type: 'integer', nullable: true)]
2017-07-01 09:52:38 +02:00
private $authCode;
2015-10-13 22:43:15 +02:00
2025-04-05 12:55:51 +02:00
#[ORM\Column(name: 'googleAuthenticatorSecret', type: 'string', nullable: true)]
private $googleAuthenticatorSecret;
2018-12-03 06:51:06 +01:00
/**
* @var array
2018-12-03 06:51:06 +01:00
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(type: 'json', nullable: true)]
2018-12-03 06:51:06 +01:00
private $backupCodes;
/**
* @var bool
*/
2025-04-05 12:55:51 +02:00
#[ORM\Column(type: 'boolean')]
private $emailTwoFactor = false;
2015-01-31 15:14:10 +01:00
public function __construct()
{
2015-08-18 11:08:45 +02:00
parent::__construct();
2015-09-20 22:37:27 +02:00
$this->entries = new ArrayCollection();
$this->roles = ['ROLE_USER'];
2015-01-31 15:14:10 +01:00
}
2015-02-06 14:18:01 +01:00
2015-01-22 17:18:56 +01:00
/**
2015-05-30 13:52:26 +02:00
* Set name.
*
* @param string $name
2015-01-22 17:18:56 +01:00
*
2015-02-06 14:18:01 +01:00
* @return User
2015-01-22 17:18:56 +01:00
*/
public function setName($name)
{
$this->name = $name;
return $this;
}
/**
2015-05-30 13:52:26 +02:00
* Get name.
2015-01-22 17:18:56 +01:00
*
2015-01-31 19:09:34 +01:00
* @return string
2015-01-22 17:18:56 +01:00
*/
public function getName()
{
return $this->name;
}
2015-02-06 14:18:01 +01:00
/**
2017-06-01 09:36:01 +02:00
* @return \DateTime
2015-02-06 14:18:01 +01:00
*/
public function getCreatedAt()
{
return $this->createdAt;
}
/**
2017-06-01 09:36:01 +02:00
* @return \DateTime
2015-02-06 14:18:01 +01:00
*/
public function getUpdatedAt()
{
return $this->updatedAt;
}
/**
* @return User
*/
public function addEntry(Entry $entry)
{
$this->entries[] = $entry;
return $this;
}
/**
* @return ArrayCollection<Entry>
*/
public function getEntries()
{
return $this->entries;
}
/**
2015-05-30 13:52:26 +02:00
* Set config.
*
* @return User
*/
2024-02-19 09:31:30 +01:00
public function setConfig(?Config $config = null)
{
$this->config = $config;
return $this;
}
/**
2015-05-30 13:52:26 +02:00
* Get config.
*
* @return Config
*/
public function getConfig()
{
return $this->config;
}
2015-10-13 22:43:15 +02:00
/**
* @return bool
*/
public function isEmailTwoFactor()
{
return $this->emailTwoFactor;
}
/**
* @param bool $emailTwoFactor
*/
public function setEmailTwoFactor($emailTwoFactor)
2015-10-13 22:43:15 +02:00
{
$this->emailTwoFactor = $emailTwoFactor;
2015-10-13 22:43:15 +02:00
}
/**
* Used in the user config form to be "like" the email option.
2015-10-13 22:43:15 +02:00
*/
public function isGoogleTwoFactor()
2015-10-13 22:43:15 +02:00
{
return $this->isGoogleAuthenticatorEnabled();
2015-10-13 22:43:15 +02:00
}
public function isEmailAuthEnabled(): bool
2015-10-13 22:43:15 +02:00
{
return $this->emailTwoFactor;
2015-10-13 22:43:15 +02:00
}
public function getEmailAuthCode(): string
2015-10-13 22:43:15 +02:00
{
return $this->authCode;
}
public function setEmailAuthCode(string $authCode): void
2015-10-13 22:43:15 +02:00
{
$this->authCode = $authCode;
}
public function getEmailAuthRecipient(): string
2015-10-13 22:43:15 +02:00
{
return $this->email;
2015-10-13 22:43:15 +02:00
}
public function isGoogleAuthenticatorEnabled(): bool
2015-10-13 22:43:15 +02:00
{
return $this->googleAuthenticatorSecret ? true : false;
}
2015-10-13 22:43:15 +02:00
public function getGoogleAuthenticatorUsername(): string
{
return $this->username;
}
2015-10-13 22:43:15 +02:00
public function getGoogleAuthenticatorSecret(): string
{
return $this->googleAuthenticatorSecret;
}
public function setGoogleAuthenticatorSecret(?string $googleAuthenticatorSecret): void
{
$this->googleAuthenticatorSecret = $googleAuthenticatorSecret;
2015-10-13 22:43:15 +02:00
}
2024-02-19 09:31:30 +01:00
public function setBackupCodes(?array $codes = null)
2018-12-03 06:51:06 +01:00
{
$this->backupCodes = $codes;
}
public function getBackupCodes()
{
return $this->backupCodes;
}
public function isBackupCode(string $code): bool
{
return false === $this->findBackupCode($code) ? false : true;
2018-12-03 06:51:06 +01:00
}
public function invalidateBackupCode(string $code): void
{
$key = $this->findBackupCode($code);
2018-12-03 06:51:06 +01:00
if (false !== $key) {
unset($this->backupCodes[$key]);
}
}
/**
* @return User
*/
public function addClient(Client $client)
{
$this->clients[] = $client;
return $this;
}
/**
2023-11-16 09:36:47 +01:00
* @return ArrayCollection<Client>
*/
public function getClients()
{
return $this->clients;
}
/**
* Only used by the API when creating a new user it'll also return the first client (which was also created at the same time).
*
* @return Client|false
*/
public function getFirstClient()
{
2017-06-07 23:23:34 +02:00
if (!empty($this->clients)) {
return $this->clients->first();
}
return false;
}
/**
* Try to find a backup code from the list of backup codes of the current user.
*
* @param string $code Given code from the user
*
* @return string|false
*/
private function findBackupCode(string $code)
{
foreach ($this->backupCodes as $key => $backupCode) {
// backup code are hashed using `password_hash`
// see ConfigController->otpAppAction
2025-04-05 14:01:48 +02:00
if (password_verify($code, (string) $backupCode)) {
return $key;
}
}
return false;
}
2015-01-22 17:18:56 +01:00
}