Compare commits

...

8 Commits

Author SHA1 Message Date
8225969e37 settings and last update line 2026-01-11 01:48:55 +05:00
088ec57da6 Merge branch 'stable' of https://git.svoboda.works/mirzaev/pot-php-telegram into stable 2026-01-07 14:58:37 +05:00
a0ca54bf25 fixed by garden/campanula 2026-01-07 14:57:15 +05:00
a76e11fb9f fixed settings installer 2026-01-07 12:54:54 +05:00
190ef8eabd exaples 2026-01-06 14:20:24 +05:00
41fcf9319b localizations 2026-01-06 14:17:36 +05:00
8b5a49747b why so shroomious 2026-01-06 14:04:51 +05:00
f2eb59336e created from kodorvan/neurobot 2026-01-06 14:02:18 +05:00
17 changed files with 1932 additions and 14 deletions

View File

@@ -1,3 +1,3 @@
!.gitignore
!*.php
*.baza
*.php

View File

@@ -0,0 +1,71 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME};
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\account,
${REPO_OWNER}\${REPO_NAME}\models\authorizations,
${REPO_OWNER}\${REPO_NAME}\models\chat,
${REPO_OWNER}\${REPO_NAME}\models\tariff;
// Svoboda time
use svoboda\time\statement as svoboda;
// Baza database
use mirzaev\baza\record;
// Enabling debugging
/* ini_set('error_reporting', E_ALL);
ini_set('display_errors', 1);
ini_set('display_startup_errors', 1); */
// Initializing path to the public directory
define('INDEX', __DIR__ . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'public');
// Initializing path to the root directory
define('ROOT', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR);
// Initializing path to the settings directory
define('SETTINGS', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'settings');
// Initializing path to the storage directory
define('STORAGE', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'storage');
// Initializing path to the databases directory
define('DATABASES', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'databases');
// Initializing path to the localizations directory
define('LOCALIZATIONS', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'localizations');
// Initiailizing telegram data
require(SETTINGS . DIRECTORY_SEPARATOR . 'telegram.php');
// Initializing dependencies
require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
// Initializing the account model
$$account_model = new account();
// Searching for the account
$$account = $$account_model->database->read(
filter: fn(record $$record) => $$record->domain === 'buddy_volkodav',
amount: 1,
offset: 0
)[0] ?? null;
var_dump($$account);
// Initializing the account authorizations model
$$authorizations_model = new authorizations();
// Searching for the account authorizations
$$authorizations = $$authorizations_model->database->read(
filter: fn(record $$record) => $$record->account === $$account->identifier,
amount: 1,
offset: 0
)[0] ?? null;
var_dump($$authorizations);

View File

@@ -2,5 +2,42 @@
// Exit (success)
return [
'' => ''
]
// System
'empty' => 'Empty',
'yes' => 'Yes',
'no' => 'No',
// Main menu
'menu_title' => 'Main menu',
'menu_update' => 'Last update',
// Account
'account_title' => 'Account',
'account_authorized_system' => 'Access to the system',
'account_authorized_settings' => 'Access to settings',
'account_authorized_system_settings' => 'System access to the system settings',
// Language setting
'settings_select_language_title' => 'Select language',
'settings_select_language_description' => 'The selected language will be writed in your account settings',
'settings_language_update_success' => 'Language replaced:',
'settings_language_update_fail' => 'Failed to replace language',
// Language selection
'select_language_title' => 'Select language',
'select_language_description' => 'The selected language will be used in the current process',
'select_language_button_add' => 'Add a language',
// Authorization
'not_authorized_system' => 'You do not have access to the system',
'not_authorized_settings' => 'You do not have access to the settings',
'not_authorized_system_settings' => 'You do not have access to the system settings',
// Messages
'message_initialization_fail' => 'Не удалось инициализировать сообщение Телеграм',
'message_text_initialization_fail' => 'Не удалось инициализировать текст сообщения Телеграм',
// Other
'why_so_shroomious' => 'why so shroomious'
];

View File

@@ -1,6 +1,42 @@
<?php
// Exit (success)
// Выход (успех)
return [
'' => ''
]
// Система
'empty' => 'Пусто',
'yes' => 'Да',
'no' => 'Нет',
// Главное меню
'menu_title' => 'Главное меню',
'menu_update' => 'Последнее обновление',
// Аккаунт
'account_title' => 'Аккаунт',
'account_authorized_system' => 'Доступ к системе',
'account_authorized_settings' => 'Доступ к изменению настроек',
'account_authorized_system_settings' => 'Системный доступ к системным настройкам',
// Настройки языка
'settings_select_language_title' => 'Выбери язык',
'settings_select_language_description' => 'Выбранный язык будет записан в настройки аккаунта',
'settings_language_update_success' => 'Язык заменён:',
'settings_language_update_fail' => 'Не удалось заменить язык',
// Выбор языка
'select_language_title' => 'Выбери язык',
'select_language_description' => 'Выбранный язык будет использован в текущем процессе',
'select_language_button_add' => 'Добавить язык',
// Авторизация
'not_authorized_system' => 'У тебя нет доступа к системе',
'not_authorized_settings' => 'У тебя нет доступа к настройкам',
'not_authorized_system_settings' => 'У тебя нет доступа к системным настройкам',
// Сообщения
'message_initialization_fail' => 'Не удалось инициализировать сообщение Телеграм',
'message_text_initialization_fail' => 'Не удалось инициализировать текст сообщения Телеграм',
// Прочее
'why_so_shroomious' => 'почему такой грибъёзный'
];

View File

@@ -0,0 +1,367 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core,
${REPO_OWNER}\${REPO_NAME}\models\authorizations,
${REPO_OWNER}\${REPO_NAME}\models\settings;
// 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 Zanzara\Telegram\Type\User as telegram_user;
// Built-in libraries
use Exception as exception,
RuntimeException as exception_runtime;
/**
* Account
*
* @package ${REPO_OWNER}\${REPO_NAME}\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;
/**
* 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('robot', type::char),
/* new column('', type::), */
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->getId(), amount: 1)[0] ?? null;
if ($$account instanceof record) {
// Found the account record
if (
$$account->domain !== (string) $$telegram->getUsername() ||
$$account->name_first !== (string) $$telegram->getFirstName() ||
$$account->name_second !== (string) $$telegram->getLastName()
) {
// The telegram account was updated
// Updating the account in the database
$$updated = $$this->database->read(
filter: fn(record $$record) => $$record->identifier_telegram === $$telegram->getId(),
update: function (record &$$record) use ($$telegram) {
// Writing new values into the record
$$record->domain = (string) $$telegram->getUsername();
$$record->name_first = (string) $$telegram->getFirstName();
$$record->name_second = (string) $$telegram->getLastName();
$$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->getId(), 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->getId(),
name_first: (string) $$telegram->getFirstName(),
name_second: (string) $$telegram->getLastName(),
domain: (string) $$telegram->getUsername(),
language: (string) $$telegram->getLanguageCode(),
robot: (bool) $$telegram->isBot()
);
if ($$record instanceof record) {
// The record was writed into the database
// 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);
// 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;
// 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,
/* */
(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
{
// 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;
// Exit (success)
return $$this;
}
/**
* Deserialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function deserialize(): self
{
// 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;
// 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;
}
}

View File

@@ -0,0 +1,157 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core;
// 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 Zanzara\Telegram\Type\User as telegram;
// Built-in libraries
use Exception as exception,
RuntimeException as exception_runtime;
/**
* Authorizations
*
* @package ${REPO_OWNER}\${REPO_NAME}\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 authorizations extends core implements record_interface
{
use record_trait;
/**
* File
*
* @var string $$database Path to the database file
*/
protected string $$file = DATABASES . DIRECTORY_SEPARATOR . 'authorizations.baza';
/**
* Database
*
* @var database $$database The database
*/
public protected(set) database $$database;
/**
* 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('account', type::long_long_unsigned),
new column('system', type::char),
new column('settings', type::char),
/* new column('', type::char), */
new column('system_settings', 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;
}
/**
* Write
*
* @param int $$account The account identifier
* @param bool $$system
* @param bool $$settings
* @param bool $$system_settings
* @param bool $$active Is the record active?
*
* @return int|false The record identifier, if created
*/
public function write(
int $$account,
bool $$system = true,
bool $$settings = true,
bool $$system_settings = false,
bool $$active = true,
): int|false
{
$$record = $$this->database->record(
$$this->database->count() + 1,
$$account,
(int) $$system,
(int) $$settings,
(int) $$system_settings,
(int) $$active,
svoboda::timestamp(),
svoboda::timestamp()
);
// Writing the record into the database
$$created = $$this->database->write($$record);
// Exit (success)
return $$created ? $$record->identifier : false;
}
/**
* Serialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function serialize(): self
{
// Serializing the record parameters
$$this->record->system = (int) $$this->record->system;
$$this->record->settings = (int) $$this->record->settings;
$$this->record->system_settings = (int) $$this->record->system_settings;
$$this->record->active = (int) $$this->record->active;
// Exit (success)
return $$this;
}
/**
* Deserialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function deserialize(): self
{
// Deserializing the record parameters
$$this->record->system = (bool) $$this->record->system;
$$this->record->settings = (bool) $$this->record->settings;
$$this->record->system_settings = (bool) $$this->record->system_settings;
$$this->record->active = (bool) $$this->record->active;
// Exit (success)
return $$this;
}
}

View File

@@ -0,0 +1,135 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core;
// 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;
// Built-in libraries
use Exception as exception,
RuntimeException as exception_runtime;
/**
* Settings
*
* @package ${REPO_OWNER}\${REPO_NAME}\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 settings extends core implements record_interface
{
use record_trait;
/**
* File
*
* @var string $$database Path to the database file
*/
protected string $$file = DATABASES . DIRECTORY_SEPARATOR . 'settings.baza';
/**
* Database
*
* @var database $$database The database
*/
public protected(set) database $$database;
/**
* 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('account', type::long_long_unsigned),
/* new column('', type::), */
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;
}
/**
* Write
*
* @param int $$account The account identifier
* @param int $$active Is the record active?
*
* @return int|false The record identifier, if created
*/
public function write(
int $$account,
bool $$active = true,
): int|false {
$$record = $$this->database->record(
$$this->database->count() + 1,
$$account,
(int) $$active,
svoboda::timestamp(),
svoboda::timestamp()
);
// Writing the record into the database
$$created = $$this->database->write($$record);
// Exit (success)
return $$created ? $$record->identifier : false;
}
/**
* Serialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function serialize(): self
{
// Serializing the record parameters
$$this->record->active = (int) $$this->record->active;
// Exit (success)
return $$this;
}
/**
* Deserialize
*
* @return self The instance from which the method was called (fluent interface)
*/
public function deserialize(): self
{
// Deserializing the record parameters
$$this->record->active = (bool) $$this->record->active;
// Exit (success)
return $$this;
}
}

View File

@@ -0,0 +1,360 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models\telegram;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core,
${REPO_OWNER}\${REPO_NAME}\models\account,
${REPO_OWNER}\${REPO_NAME}\models\settings,
${REPO_OWNER}\${REPO_NAME}\models\telegram\processes\language\select as process_language_select;
// Library for languages support
use mirzaev\languages\language;
// The library for escaping all markdown symbols
use function mirzaev\unmarkdown;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message,
Zanzara\Telegram\Type\Input\InputFile as file_input;
/**
* Telegram commands
*
* @package ${REPO_OWNER}\${REPO_NAME}\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class commands extends core
{
/**
* Start
*
* Responce for command: "/start"
*
* @param context $$context Request data from Telegram
*
* @return void
*/
public static function start(context $$context): void
{
static::menu($$context);
}
/**
* Menu
*
* Responce for command: '/menu'
*
* @param context $$context Request data from Telegram
*
* @return void
*/
public static function menu(context $$context): void
{
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing the language
$$language = $$context->get('language');
if ($$language instanceof language) {
// Initialized the language
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Initializing the message title
$$title = '📋 *' . $$localization['menu_title'] . '*';
// Initializing the message last update text
exec(command: 'git log --oneline $$(git describe --tags --abbrev=0 @^ --always)..@ -1 --format="%at" | xargs -I{} date -d @{} "+%Y.%m.%d %H:%M"', output: $$git);
$$update = empty($$git[0]) ? '' : "\n\n🔏 *" . $$localization['menu_update'] . ':* ' . unmarkdown($$git[0]);
// Sending the message
$$context->sendMessage(
<<<TXT
$$title$$update
TXT,
[
'reply_markup' => [
'inline_keyboard' => [
/* [
[
'text' => '⚙️ ' . $$localization[''],
'callback_data' => ''
]
] */
],
'disable_notification' => true,
'remove_keyboard' => true
],
]
)->then(function (message $$message) use ($$context) {
// Sended the message
});
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Account
*
* Responce for the command: "/account"
*
* Sends information about account with menu
*
* @param context $$context Request data from Telegram
*
* @return void
*/
public static function account(context $$context): void
{
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Initializing title for the message
$$title = '🫵 ' . $$localization['account_title'];
// Declaring buufer of rows about authorizations
$$authorizations = '';
// Initializing rows about authorization
foreach ($$account->values() as $$key => $$value) {
// Iterating over account parameters
if (str_starts_with($$key, 'authorized_')) {
// Iterating over account authorizations
// Skipping system authorizations
if (str_starts_with($$key, 'authorized_system_')) continue;
// Writing into buffer of rows about authorizations
$$authorizations .= ($$value ? '✅' : '❎') . ' *' . ($$localization["account_$$key"] ?? $$key) . ':* ' . ($$value ? $$localization['yes'] : $$localization['no']) . "\n";
}
}
// Trimming the last line break character
$$authorizations = trim($$authorizations, "\n");
// Sending the message
$$context->sendMessage(
<<<TXT
$$title
$$authorizations
TXT,
[
'reply_markup' => [
'remove_keyboard' => true,
'disable_notification' => true
],
'link_preview_options' => [
'is_disabled' => true
]
]
);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Language
*
* Responce for the command: "/language"
*
* Send the language selection menu
*
* @param context $$context Request data from Telegram
*
* @return void
*/
public static function language(context $$context): void
{
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing language
$$language = $$context->get('language');
if ($$language instanceof language) {
// Initialized language
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Sending the language selection
process_language_select::menu(
context: $$context,
prefix: 'settings_language_',
title: '🌏 *' . $$localization['settings_select_language_title'] . '*',
description: '🌏 *' . $$localization['settings_select_language_description'] . '*'
);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Society
*
* Responce for the command: "/society"
*
* Sends the "mushroom" image and the localized text "why so shroomious"
*
* @param context $$context Request data from Telegram
*
* @return void
*/
public static function society(context $$context): void
{
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Sending the message
$$context->sendPhoto(
new file_input(STORAGE . DIRECTORY_SEPARATOR . 'images' . DIRECTORY_SEPARATOR . 'mushroom.jpg'),
[
'caption' => $$localization['why_so_shroomious'],
'disable_notification' => true
]
);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Ending the conversation process
$$context->endConversation();
});
}
}
}

View File

@@ -0,0 +1,469 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models\telegram;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core,
${REPO_OWNER}\${REPO_NAME}\models\account,
${REPO_OWNER}\${REPO_NAME}\models\authorizations;
// The library for languages support
use mirzaev\languages\language;
// The library for escaping all markdown symbols
use function mirzaev\unmarkdown;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message,
Zanzara\Middleware\MiddlewareNode as node;
// Built-in libraries
use Error as error;
/**
* Telegram middlewares
*
* @package ${REPO_OWNER}\${REPO_NAME}\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class middlewares extends core
{
/**
* Account (middleware)
*
* Initialize or registrate the account and write it into the `account` variable inside the `$$context`
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function account(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the telegram account
$$telegram = $$context->getEffectiveUser();
// Initializing the account
$$account = new account()->initialize($$telegram);
if ($$account instanceof account) {
// Initialized the account
// Writing the account into the context variable
$$context->set('account', $$account);
// Continuation of the process
$$next($$context);
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Authorizations (middleware)
*
* Initialize the account authorizations and write them into the `authorizations` variable inside the `$$context`
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function authorizations(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing the account authorizations
$$authorizations = $$account->authorizations();
if ($$authorizations instanceof authorizations) {
// Initialized the account authorizations
// Writing the account authorizations into the context variable
$$context->set('authorizations', $$authorizations);
// Continuation of the process
$$next($$context);
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account authorizations*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Language (middleware)
*
* Implement the account language
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function language(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
if ($$account->language instanceof language) {
// Initialized the language parameter
try {
// Writing the account language into the context variable
$$context->set('language', $$account->language);
} catch (error $$error) {
// Not initialized the language
// Writing the default language into the context variable
$$context->set('language', LANGUAGE_DEFAULT ?? language::en);
}
} else {
// Not initialized the language parameter
// Writing the default language into the context variable
$$context->set('language', LANGUAGE_DEFAULT ?? language::en);
}
// Continuation of the process
$$next($$context);
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Localization (middleware)
*
* Implement the account language and initialize the localization file
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function localization(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing the language
$$language = $$context->get('language');
if ($$language instanceof language) {
// Initialized the language
// Initializing path to the localization file
$$file = LOCALIZATIONS . DIRECTORY_SEPARATOR . strtolower($$language->label()) . '.php';
if (file_exists($$file) && is_readable($$file)) {
// Found the localization file
// Initializing localization
$$localization = require($$file);
if (is_array($$localization)) {
// Initialized the localization
// Writing localization into the context variable
$$context->set('localization', $$localization);
// Continuation of the process
$$next($$context);
} else {
// Not initialized the localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not found the localization file
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize the localization file*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* Settings (middleware)
*
* Check the account for access to the settings
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function settings(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing the account authorizations
$$authorizations = $$context->get('authorizations');
if ($$authorizations instanceof authorizations) {
// Initialized the account authorizations
if ($$authorizations->settings) {
// Authorized the account to the settings
// Continuation of the process
$$next($$context);
} else {
// Not authorized the account to the settings
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Sending the message
$$context->sendMessage('⛔ *' . $$localization['not_authorized_settings'] . '*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
// Stopping the process
$$context->set('stop', true);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
} else {
// Not initialized the account authorizations
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account authorizations*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
/**
* System settings (middleware)
*
* Check the account for access to the system settings
*
* @param context $$context
* @param node $$next
*
* @return void
*/
public static function system_settings(context $$context, node $$next): void
{
// Is the process stopped?
if ($$context->get('stop')) return;
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof account) {
// Initialized the account
// Initializing the account authorizations
$$authorizations = $$context->get('authorizations');
if ($$authorizations instanceof authorizations) {
// Initialized the account authorizations
if ($$authorizations->system_settings) {
// Authorized the account to the system settings
// Continuation of the process
$$next($$context);
} else {
// Not authorized the account to the system settings
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Sending the message
$$context->sendMessage('⛔ *' . $$localization['not_authorized_system_settings'] . '*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
// Stopping the process
$$context->set('stop', true);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account authorizations*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
}

View File

@@ -0,0 +1,152 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME}\models\telegram\processes\language;
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\core;
// Library for languages support
use mirzaev\languages\language;
// Baza database
use mirzaev\baza\record;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message;
/**
* Telegram language select
*
* @package ${REPO_OWNER}\${REPO_NAME}\models\telegram\processes\language
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class select extends core
{
/**
* Language
*
* Send the language choose menu
*
* @param context $$context Request data from Telegram
* @param string $$prefix Prefix for 'callback_data' (`$$prefix . $$language->name`)
* @param string $$title Title of the message
* @param string $$description Description of the message
* @param array $$exclude Languages that will be excluded ['ru', 'en'...]
*
* @return void
*/
public static function menu(context $$context, string $$prefix, string $$title, string $$description, array $$exclude = []): void
{
// Initializing the account
$$account = $$context->get('account');
if ($$account instanceof record) {
// Initialized the account
// Initializing language
$$language = $$context->get('language');
if ($$language) {
// Initialized language
// Initializing localization
$$localization = $$context->get('localization');
if ($$localization) {
// Initialized localization
// Declaring the buffer of generated keyboard with languages
$$keyboard = [];
// Initializing the iterator of rows
$$row = 0;
// Initializing buffer of languages
$$languages = language::cases();
// Deleting the actual language from buffer of languages
unset($$languages[array_search($$language, $$languages, strict: true)]);
// Sorting buffer of languages by the actual language
$$languages = [$$language, ...$$languages];
foreach ($$languages as $$language) {
// Iterating over languages
// Skipping excluded languages
if (array_search($$language->name, $$exclude, strict: true) !== false) continue;
// Initializing the row
$$keyboard[$$row] ??= [];
// Writing the language choose button into the buffer of generated keyboard with languages
$$keyboard[$$row][] = [
'text' => ($$language->flag() ? $$language->flag() . ' ' : '') . $$language->label($$language),
'callback_data' => $$prefix . $$language->name
];
// When reaching 4 buttons in a row, move to the next row
if (count($$keyboard[$$row]) === 4) ++$$row;
}
// Writing the button for helping lozalizing
$$keyboard[$$row === 0 && empty($$keyboard[0]) ? 0 : ++$$row] = [
[
'text' => '🗂 ' . $$localization['select_language_button_add'],
'url' => 'https://git.svoboda.works/${REPO_OWNER}\${REPO_NAME}/src/branch/stable/kodorvan/neurobot/system/localizations'
]
];
// Sending the message
$$context->sendMessage(
$$title ?? '🌏 *' . $$localization['select_language_title'] . "*\n" . ($$description ?? $$localization['select_language_description']),
[
'reply_markup' => [
'inline_keyboard' => $$keyboard,
'disable_notification' => true
],
]
);
} else {
// Not initialized localization
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $$message) use ($$context) {
// Sended the message
// Ending the conversation process
$$context->endConversation();
});
}
}
}

View File

@@ -0,0 +1,92 @@
<?php
declare(strict_types=1);
namespace ${REPO_OWNER}\${REPO_NAME};
// Files of the project
use ${REPO_OWNER}\${REPO_NAME}\models\account,
${REPO_OWNER}\${REPO_NAME}\models\telegram\middlewares,
${REPO_OWNER}\${REPO_NAME}\models\telegram\commands,
${REPO_OWNER}\${REPO_NAME}\models\telegram\settings;
// Library for languages support
use mirzaev\languages\language;
// Framework for PHP
use mirzaev\minimal\core,
mirzaev\minimal\route;
// Framework for Telegram
use Zanzara\Zanzara as zanzara,
Zanzara\Context as context,
Zanzara\Config as config;
// Enabling debugging
/* ini_set('error_reporting', E_ALL);
ini_set('display_errors', 1);
ini_set('display_startup_errors', 1); */
// Initializing path to the public directory
define('INDEX', __DIR__);
// Initializing path to the project root directory
define('ROOT', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR);
// Initializing path to the directory of views
define('VIEWS', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'views');
// Initializing path to the directory of settings
define('SETTINGS', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'settings');
// Initializing system settings
require SETTINGS . DIRECTORY_SEPARATOR . 'system.php';
// Initializing path to the directory of the storage
define('STORAGE', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'storage');
// Initializing path to the databases directory
define('DATABASES', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'databases');
// Initializing path to the localizations directory
define('LOCALIZATIONS', INDEX . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'localizations');
// Initiailizing Telegram chat-robot settings
require(SETTINGS . DIRECTORY_SEPARATOR . 'telegram.php');
// Initializing dependencies
require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
// Initializing the configuration
$$config = new config();
$$config->setParseMode(config::PARSE_MODE_MARKDOWN);
$$config->useReactFileSystem(true);
// Initializing the robot
$$robot = new Zanzara(ROBOT['key'], $$config);
// Initializing the updates listener
$$robot->onUpdate(function (context $$context): void {});
// Initializing the robot middlewares
$$robot->middleware([middlewares::class, 'account']);
$$robot->middleware([middlewares::class, 'language']);
$$robot->middleware([middlewares::class, 'localization']);
$$robot->middleware([middlewares::class, 'authorizations']);
// Initializing the robot commands handlers
$$robot->onCommand('start', [commands::class, 'start']);
$$robot->onCommand('menu', [commands::class, 'menu']);
$$robot->onCommand('language', [commands::class, 'language'])->middleware([middlewares::class, 'settings']);
$$robot->onCommand('society', [commands::class, 'society']);
// Initializing the robot settings language buttons handlers
foreach (language::cases() as $$language) {
// Iterating over languages
// Initializing language buttons
$$robot->onCbQueryData(["settings_language_$$language->name"], fn(context $$context) => settings::language($$context, $$language));
};
// Starting chat-robot
$$robot->run();

View File

@@ -1,4 +1,15 @@
<?php
// The library for languages support
use mirzaev\languages\language;
// Initializing dependencies
require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
define('PROJECT_NAME', '');
define('PROJECT_DOMAIN', '');
define('LANGUAGE_DEFAULT', language::en);
// Initializing default theme for the views templater
define('THEME', 'default');
define('THEME', 'default');

View File

@@ -0,0 +1,7 @@
<?php
// Robot
define('ROBOT', [
'identifier' => null,
'key' => ''
]);

Binary file not shown.

After

Width:  |  Height:  |  Size: 46 KiB

View File

@@ -22,13 +22,19 @@
"issues": "https://git.svoboda.works${REPO_LINK}/issues"
},
"require": {
"php": "^8.4",
"php": "^8.5",
"mirzaev/minimal": "^3.7",
"mirzaev/baza": "^3.3",
"mirzaev/languages": "^1",
"mirzaev/record": "^1.0",
"mirzaev/languages": "^1.0",
"mirzaev/currencies": "^2.0",
"mirzaev/unmarkdown": "^1.0",
"svoboda/time": "^1.0",
"badfarm/zanzara": "^0.9.1",
"twig/twig": "^3.2",
"twig/extra-bundle": "^3.7",
"twig/intl-extra": "^3.10"
"twig/intl-extra": "^3.10",
"react/filesystem": "^0.1.2"
},
"suggest": {
"mirzaev/files": "Easy working with files",

View File

@@ -0,0 +1,16 @@
[Unit]
Description=Telegram chat-robot: @robot_bot
Wants=network.target
After=syslog.target network-online.target
[Service]
ExecStart=sudo -u www-data /usr/bin/php /var/www/${REPO_NAME}/${REPO_OWNER}/${REPO_NAME}/system/public/telegram.php
PIDFile=/var/run/php/telegram.pid
RemainAfterExit=no
RuntimeMaxSec=3600s
Restart=always
RestartSec=5s
[Install]
WantedBy=multi-user.target

View File

@@ -8,8 +8,10 @@ if [ -d author ]; then
mv author ${REPO_OWNER}
fi
if [ -e ${REPO_OWNER}/${REPO_NAME}/system/settings/*.sample ]; then
for i in ${REPO_OWNER}/${REPO_NAME}/system/settings/*.sample; do
for i in ${REPO_OWNER}/${REPO_NAME}/system/settings/*.sample; do
echo $$i;
if [ ! -f "$${i/.sample/}" ]; then
cp "$$i" "$${i/.sample/}";
done
fi
echo $${i/.sample/};
fi
done