Files
edabil/kodorvan/neurobot/system/models/account.php
2026-02-28 18:29:31 +05:00

607 lines
15 KiB
PHP
Executable File

<?php
declare(strict_types=1);
namespace kodorvan\neurobot\models;
// Files of the project
use kodorvan\neurobot\models\core,
kodorvan\neurobot\models\authorizations,
kodorvan\neurobot\models\chat,
kodorvan\neurobot\models\settings,
kodorvan\neurobot\models\tariff,
kodorvan\neurobot\models\code,
kodorvan\neurobot\models\enumerations\tariff as tariff_type;
// The library for languages support
use mirzaev\languages\language;
// Baza database
use mirzaev\baza\database,
mirzaev\baza\column,
mirzaev\baza\record,
mirzaev\baza\enumerations\encoding,
mirzaev\baza\enumerations\type;
// Active Record pattern
use mirzaev\record\interfaces\record as record_interface,
mirzaev\record\traits\record as record_trait;
// Svoboda time
use svoboda\time\statement as svoboda;
// Framework for Telegram
use SergiX44\Nutgram\Telegram\Types\User\User as telegram_user;
// Built-in libraries
use Exception as exception,
RuntimeException as exception_runtime;
/**
* Account
*
* @package kodorvan\neurobot\models
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class account extends core implements record_interface
{
use record_trait;
/**
* File
*
* @var string $database Path to the database file
*/
protected string $file = DATABASES . DIRECTORY_SEPARATOR . 'accounts.baza';
/**
* Database
*
* @var database $database The database
*/
public protected(set) database $database;
/**
* Serialized
*
* @var bool $serialized Is the implementator object serialized?
*/
private bool $serialized = true;
/**
* Constructor
*
* @method record|null $record The record
*
* @return void
*/
public function __construct(?record $record = null)
{
// Initializing the database
$this->database = new database()
->encoding(encoding::utf8)
->columns(
new column('identifier', type::long_long_unsigned),
new column('identifier_telegram', type::long_long_unsigned),
new column('domain', type::string, ['length' => 32]),
new column('name_first', type::string, ['length' => 64]),
new column('name_second', type::string, ['length' => 64]),
new column('language', type::string, ['length' => 2]),
/* new column('currency', type::string, ['length' => 3]), */
new column('robot', type::char),
new column('chat', type::long_long_unsigned),
new column('tariff', type::long_long_unsigned),
new column('welcome', type::char),
new column('active', type::char),
new column('updated', type::integer_unsigned),
new column('created', type::integer_unsigned)
)
->connect($this->file);
// Initializing the record
$record instanceof record and $this->record = $record;
}
/**
* Initialize
*
* @param telegram_user $telegram The telegram account
*
* @throws exception_runtime if update the account record in the database by the telegram account values
* @throws exception_runtime if failed to find the registered account
* @throws exception_runtime if failed to registrate the account
*
* @return static|null The account, if found, updated or created
*/
public function initialize(telegram_user $telegram): ?static
{
// Searching for the account in the database
$account = $this->database->read(filter: fn(record $record) => $record->identifier_telegram === $telegram->id, amount: 1)[0] ?? null;
if ($account instanceof record) {
// Found the account record
if (
$account->domain !== (string) $telegram->username ||
$account->name_first !== (string) $telegram->first_name ||
$account->name_second !== (string) $telegram->last_name
) {
// The telegram account was updated
// Updating the account in the database
$updated = $this->database->read(
filter: fn(record $record) => $record->identifier_telegram === $telegram->id,
update: function (record &$record) use ($telegram) {
// Writing new values into the record
$record->domain = (string) $telegram->username;
$record->name_first = (string) $telegram->first_name;
$record->name_second = (string) $telegram->last_name;
$record->updated = svoboda::timestamp();
},
amount: 1
)[0] ?? null;
if ($updated instanceof record && $updated->values() !== $account->values()) {
// Updated the account in the database
// Writing the updated record into the account object
$this->record = $updated;
// Deserializing parameters
$this->deserialize();
// Exit (success)
return $this;
} else {
// Not updated the account in the database
// Exit (fail)
throw new exception_runtime('Failed to update the account record in the database by the telegram account values');
}
}
// Writing the found record into the account object
$this->record = $account;
// Deserializing parameters
$this->deserialize();
// Exit (success)
return $this;
} else {
// Not found the account record
if ($this->registrate($telegram)) {
// Registered the account
// Searching for the registered account in the database
$registered = $this->database->read(filter: fn(record $record) => $record->identifier_telegram === $telegram->id, amount: 1)[0] ?? null;
if ($registered instanceof record) {
// Found the registered account
// Writing the registered record into the account object
$this->record = $registered;
// Deserializing parameters
$this->deserialize();
// Exit (success)
return $this;
} else {
// Not found the registered account
// Exit (fail)
throw new exception_runtime('Failed to find the registered account');
}
} else {
// Not registered the account
// Exit (fail)
throw new exception_runtime('Failed to registrate the account');
}
}
}
/**
* Registrate
*
* Create the account by the telegram account data
*
* @param telegram_user $telegram The telegram account
*
* @return record|false The record, if created
*/
public function registrate(telegram_user $telegram): record|false
{
// Creating the record
$record = $this->write(
telegram_identifier: (int) $telegram->id,
name_first: (string) $telegram->first_name,
name_second: (string) $telegram->last_name,
domain: (string) $telegram->username,
language: (string) $telegram->language_code,
robot: (bool) $telegram->is_bot
);
if ($record instanceof record) {
// The record was writed into the database
if (defined('TARIFF_DEFAULT')) {
// Initialized the default tariff
// Creating the account tariff
$tariff = new tariff()->write(
account: $record->identifier,
invoice: 0,
type: TARIFF_DEFAULT,
active: true
);
// Writing the tariff into the record
$record->tariff = $tariff->identifier;
// Writing the record into the database
$record = $this->database->read(
filter: fn(record $_record) => $_record->identifier === $record->identifier,
update: fn(record &$_record) => $_record = $record,
amount: 1
)[0] ?? null;
}
// Initializing the authorizations model
$authorizations = new authorizations();
// Creating the authorizations record
$authorizations->write(account: $record->identifier);
// Initializing the settings model
$settings = new settings();
// Creating the account settings
$settings->write(account: $record->identifier);
// Exit (success)
return $record;
}
// Exit (fail)
return false;
}
/**
* Write
*
* @param int $telegram_identifier The telegram account identifier
* @param string $name_first
* @param string $name_second
* @param string $domain
* @param language|string $language
* @param bool $robot Is a robot?
* @param bool $active Is the record active?
*
* @return record|false The record, if created
*/
public function write(
int $telegram_identifier,
string $domain = '',
string $name_first = '',
string $name_second = '',
language|string $language = LANGUAGE_DEFAULT ?? language::en,
bool $robot = false,
bool $active = true,
): record|false {
// Initializing the record
$record = $this->database->record(
$this->database->count() + 1,
(int) $telegram_identifier,
$domain,
$name_first,
$name_second,
$language instanceof language ? $language->name : (string) $language,
(int) $robot,
0,
0,
0,
/* */
(int) $active,
svoboda::timestamp(),
svoboda::timestamp()
);
// Writing the record into the database
$created = $this->database->write($record);
// Exit (success)
return $created ? $record : false;
}
/**
* Serialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function serialize(): self
{
if ($this->serialized) {
// The record implementor is serialized
// Exit (fail)
throw new exception_runtime('The record implementator object is already serialized');
}
// Serializing the record parameters
$this->record->language = $this->record->language->name;
$this->record->robot = (int) $this->record->robot;
$this->record->active = (int) $this->record->active;
// Writing the serializing status
$this->serialized = true;
// Exit (success)
return $this;
}
/**
* Deserialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function deserialize(): self
{
if (!$this->serialized) {
// The record implementor is deserialized
// Exit (fail)
throw new exception_runtime('The record implementator object is already deserialized');
}
// Deserializing the record parameters
$this->record->language = language::{$this->record->language} ?? LANGUAGE_DEFAULT ?? language::en;
$this->record->robot = (bool) $this->record->robot;
$this->record->active = (bool) $this->record->active;
// Writing the serialized status
$this->serialized = false;
// Exit (success)
return $this;
}
/**
* Authorizations
*
* Search for the account authorizations
*
* @return authorizations|null The account authorizations
*/
public function authorizations(): ?authorizations
{
// Search for the account authorizations
$authorizations = new authorizations()->read(filter: fn(record $record) => $record->active === 1 && $record->account === $this->identifier);
if ($authorizations instanceof authorizations) {
// Found the account authorizations
// Exit (success)
return $authorizations;
}
// Exit (fail)
return null;
}
/**
* Settings
*
* Search for the account settings
*
* @return settings|null The account settings
*/
public function settings(): ?settings
{
// Search for the account settings
$settings = new settings()->read(filter: fn(record $record) => $record->active === 1 && $record->account === $this->identifier);
if ($settings instanceof settings) {
// Found the account settings
// Exit (success)
return $settings;
}
// Exit (fail)
return null;
}
/**
* Chat
*
* Search for the account chat (or create)
*
* @return chat|null The account chat
*/
public function chat(): ?chat
{
// Search for the account chat
$chat = new chat()->read(filter: fn(record $record) => $record->identifier === $this->record->chat && $record->active === 1);
if ($chat instanceof chat) {
// Found the account chat
// Deserializing the chat
$chat->deserialize();
// Exit (success)
return $chat;
} else {
// Not found the account chat
// Initializing the chat model
$chat = new chat();
// @todo проверку на то что чат создан добавить здесь и при регистрации аккаунта, затем сделать то же самое для тарифов и прочего
// Creating the account chat
$this->chat = $chat->write(account: $this->identifier, network: NETWORK_DEFAULT);
// Serializing the account
$this->serialize();
// Writing the record into the database
$record = $this->update();
// Deserializing the account
$this->deserialize();
// Search for the account chat
$chat = new chat()->read(filter: fn(record $record) => $record->identifier === $this->record->chat && $record->active === 1);
if ($chat instanceof chat) {
// Found the account chat
// Deserializing the chat
$chat->deserialize();
// Exit (success)
return $chat;
}
}
// Exit (fail)
return null;
}
/**
* Tariff
*
* Search for the account tariff
*
* @return tariff|null The account tariff
*/
public function tariff(): ?tariff
{
// Search for the account tariff
$tariff = new tariff()->read(
filter: fn(record $record) =>
$record->identifier === $this->record->tariff
&& $record->account === $this->record->identifier
&& $record->active === 1
&& $record->used <= $record->tokens
);
if ($tariff instanceof tariff) {
// Found the account tariff
// Deserializing the tariff
$tariff->deserialize();
// Exit (success)
return $tariff;
}
// Exit (fail)
return null;
}
/**
* Code
*
* Activate the tariff activation code
*
* @param string $code The code
*
* @return bool Is the code was activated?
*/
public function code(string $code): bool
{
// Search for the code
$record = new code()->database->read(
filter: fn(record $record) => $record->value === $code && $record->activated === 0 && $record->account === 0,
update: function (record &$record) {
$record->account = $this->record->identifier;
$record->activated = 1;
$record->updated = svoboda::timestamp();
},
amount: 1
)[0] ?? null;
if ($record instanceof record) {
// Found and activated the code
// Initializing the code instance
$code = new code($record);
// Writing the code record into the code instance
$code->record = $record;
// Initializing the "activated" registry
$activated = false;
if ($code->tariff !== 0) {
// The code has a tariff
// Search for the tariff
$tariff = new tariff()->database->read(
filter: fn(record $record) => $record->identifier === $code->tariff,
update: function (record &$record) {
$record->account = $this->record->identifier;
$record->active = 1;
$record->updated = svoboda::timestamp();
},
amount: 1
)[0] ?? null;
if ($tariff instanceof record) {
// Found and connected the tariff
// Writing the tariff into the account (select the active tariff)
$this->record->tariff = $tariff->identifier;
// Serializing the account data
$this->serialize();
// Writing the updated account unto the database
$this->update();
// Deserializing the account data
$this->deserialize();
// Writing the "activated" registry
$activated = true;
}
}
if ($code->bundle !== 0) {
// The code has a bundle
// Search for the bundle
$bundle = new bundle()->database->read(
filter: fn(record $record) => $record->identifier === $code->bundle,
update: function (record &$record) {
$record->account = $this->recoed->identifier;
$record->updated = svoboda::timestamp();
},
amount: 1
)[0] ?? null;
if ($bundle instanceof record) {
// Found and connected the bundle
// Writing the "activated" registry
$activated = true;
}
}
// Exit (success/fail)
return $activated;
}
// Exit (fail)
return false;
}
}