deals, project maker and sex

This commit is contained in:
2026-03-09 22:06:33 +05:00
parent 4757a81d95
commit f1f73564a0
18 changed files with 1884 additions and 4862 deletions

View File

@@ -23,18 +23,13 @@
}, },
"require": { "require": {
"php": "^8.5", "php": "^8.5",
"mirzaev/minimal": "^3.7", "mirzaev/minimal": "^3",
"mirzaev/baza": "^3.4", "mirzaev/baza": "^3",
"mirzaev/record": "^1.0", "mirzaev/record": "^2.0",
"mirzaev/languages": "^1.0", "mirzaev/languages": "^1.0",
"mirzaev/currencies": "^2.0", "mirzaev/currencies": "^2.0",
"mirzaev/unmarkdown": "^1.0", "mirzaev/unmarkdown": "^1.0",
"svoboda/time": "^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",
"react/filesystem": "^0.1.2",
"nyholm/psr7": "^1.8", "nyholm/psr7": "^1.8",
"irazasyed/telegram-bot-sdk": "^3.15", "irazasyed/telegram-bot-sdk": "^3.15",
"nutgram/nutgram": "^4.42", "nutgram/nutgram": "^4.42",
@@ -44,7 +39,6 @@
"suggest": { "suggest": {
"mirzaev/files": "Easy working with files", "mirzaev/files": "Easy working with files",
"mirzaev/currencies": "Easy currencies integration" "mirzaev/currencies": "Easy currencies integration"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {

4342
composer.lock generated

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,77 @@
<?php
declare(strict_types=1);
namespace kodorvan\constructor;
// Files of the project
use kodorvan\constructor\models\account,
kodorvan\constructor\models\authorizations,
kodorvan\constructor\models\chat,
kodorvan\constructor\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,
update: function (record $record) {
$record->system_settings = 1;
$record->system_deals = 1;
$record->system_projects = 1;
$record->system_invoices = 1;
$record->updated = svoboda::timestamp();
},
amount: 1,
offset: 0
)[0] ?? null;
var_dump($authorizations);

View File

@@ -10,7 +10,8 @@ return [
// Главное меню // Главное меню
'menu_title' => 'Главное меню', 'menu_title' => 'Главное меню',
'menu_description_guest' => "🔥 *Создайте ваш первый проект* и получите *ориентировочную стоимость* всего за 2 минуты", 'menu_description_guest' => "🔥 *Создайте ваш первый проект* и получите *ориентировочную стоимость* всего за 2 минуты",
'menu_description_partner' => "*Благодарю за выбор нашей команды*\. Теперь Вы один из наших %d партнёров!", 'menu_description_partner' => 'Благодарим за выбор нашей команды',
'menu_cooperation' => '_Стараемся, чтобы сотрудничество с нами было максимально *комфортным* и *эффективным*_',
'menu_update' => 'Последнее обновление', 'menu_update' => 'Последнее обновление',
'menu_button_project_new' => 'Создать', 'menu_button_project_new' => 'Создать',
'menu_button_projects' => 'Проекты', 'menu_button_projects' => 'Проекты',
@@ -23,63 +24,87 @@ return [
'account_authorized_system_settings' => 'Системный доступ к системным настройкам', 'account_authorized_system_settings' => 'Системный доступ к системным настройкам',
// Проект: создание // Проект: создание
'project_create_title' => 'Создание проекта', 'project_title' => 'Создание проекта',
/* 'project_create_description' => "Расчитайте ориентировочное время разработки, затем выберите разработчиков и получите стоимость\n\nПосле расчётов можно будет отправить проект в заказ разработчикам и приложить ТЗ, либо краткое описание задачи\n\nМы погружаемся в проекты полностью, поэтому стараемся не распыляться - от степени нагрузки меняется коэффициент стоимости!", */ /* 'project_description' => "Расчитайте ориентировочное время разработки, затем выберите разработчиков и получите стоимость\n\nПосле расчётов можно будет отправить проект в заказ разработчикам и приложить ТЗ, либо краткое описание задачи\n\nМы погружаемся в проекты полностью, поэтому стараемся не распыляться - от степени нагрузки меняется коэффициент стоимости!", */
'project_create_description' => "Установите параметры и получите ориентировочное время разработки\n\nосле расчётов можно отправить проект в заказ приложив ТЗ, либо описание задачи_", 'project_description' => "Установите параметры и получите ориентировочное время разработки\n\nосле расчётов можно отправить проект в заказ приложив ТЗ, либо описание задачи_",
'project_create_warning_cost' => "Стоимость и сроки выполнения предоставлены для планирования и *не являются публичной офертой*", 'project_warning_cost' => "Стоимость и сроки выполнения предоставлены для планирования и *не являются публичной офертой*",
'project_create_team_warning' => "", 'project_team_warning' => "",
'project_create_time' => 'Время', 'project_time' => 'Время',
'project_create_time_hours' => 'ч', 'project_time_hours' => 'ч',
'project_create_time_hours_from' => 'от', 'project_time_hours_from' => 'от',
'project_create_time_days' => 'дн', 'project_time_days' => 'дн',
'project_create_cost' => 'Стоимость', 'project_cost' => 'Стоимость',
'project_create_cost_prepayment' => 'Предоплата', 'project_cost_prepayment' => 'Предоплата',
'project_create_team_title' => 'Сбор команды', 'project_team_title' => 'Сбор команды',
'project_create_team_description' => '_Вы можете *убрать* или *добавить* специалистов, создавая максимально удобное сотрудничество под вашу систему_', 'project_team_description' => '_Вы можете *убрать* или *добавить* специалистов, создавая максимально удобное сотрудничество под вашу систему_',
'project_create_team_warning_cost' => '_*Не влияет на время разработки*_', 'project_team_warning_cost' => '_*Не влияет на время разработки*_',
'project_create_team_button_programmers' => 'Программисты: %d%s', 'project_team_button_programmers' => 'Программисты: %d%s',
'project_create_team_button_designers' => 'Дизайнеры: %d%s', 'project_team_button_designers' => 'Дизайнеры: %d%s',
'project_create_team_button_boosters' => 'Бустеры: %d%s', 'project_team_button_boosters' => 'Бустеры: %d%s',
'project_create_cost_title' => 'Стоимость разработки', 'project_team_programmers_title' => 'Количество программистов',
'project_create_cost_description' => 'Введите предложение по *оплате за 1 час* разработки', 'project_team_programmers_description' => 'Никаких копий по шаблонам и конструкторов, только *чистый код с нуля* и наши *уникальные технологии* с *многолетней кодовой базой*',
'project_create_cost_default' => '*`%d%s`* \- средняя стоимость с учётом *компетенции*, совокупного опыта, *уникальных* разработок\. Цена уже включает\: *наши сервера*, документирование кода, *передачу кода*, ведение репозитория, *техподдержку* и репутационные гарантии', 'project_team_programmers_request' => '_Отправьте предложение по *количеству программистов*_',
'project_create_cost_warning' => '_Если цена окажется *недостаточной*, мы выставим *справедливое* и *конкурентное* встречное предложение_', 'project_team_programmers_error_amount' => 'Мы можем предоставить до %d программистов',
'project_create_cost_error_distance' => 'Длина сообщения должна быть от %d и до %d символов', 'project_team_designers_title' => 'Количество дизайнеров',
'project_create_cost_error_not_a_number' => 'Значение должно быть числом', 'project_team_designers_description' => 'У каждого дизайнера своё *уникальное видение*, поэтому мы стараемся находить максимально подходящих под проект, но зачастую мы привлекаем сторонних дизайнеров\. Если у заказчика имеются свои проверенные дизайнеры, то мы готовы с ними сотрудничать',
'project_create_button_back' => 'Назад', 'project_team_designers_request' => '_Отправьте предложение по *количеству дизайнеров*_',
'project_create_button_cost_per_hour' => 'Час', 'project_team_designers_error_amount' => 'Мы можем предоставить до %d дизайнеров',
'project_create_button_request' => 'Заказать', 'project_team_boosters_title' => 'Количество бустеров',
'project_team_boosters_description' => '*Бустер* \- обобщение для специалистов которые занимаются *маркетингом*, *рекламными кабинетами*, дошлифовкой интерфейса, *user\-experience*, А\/Б тестированием, сбором фокус\-группы, развитием сообщества и в целом *SMM*',
'project_team_boosters_request' => '_Отправьте предложение по *количеству бустеров*_',
'project_team_boosters_error_amount' => 'Мы можем предоставить до %d бустеров',
'project_team_error_not_a_number' => 'Значение должно быть числом',
'project_cost_title' => 'Стоимость разработки',
'project_cost_description' => '*`%d%s`* \- средняя стоимость с учётом *компетенции*, совокупного опыта, *уникальных* разработок\. Цена уже включает\: *наши сервера*, документирование кода, *передачу кода*, ведение репозитория, *техподдержку* и репутационные гарантии',
'project_cost_request' => '_Отправьте предложение по *оплате за 1 час* разработки_',
'project_cost_warning' => '_Если цена окажется *недостаточной*, мы выставим *справедливое* и *конкурентное* встречное предложение_',
'project_cost_error_distance' => 'Длина сообщения должна быть от %d и до %d символов',
'project_cost_error_not_a_number' => 'Значение должно быть числом',
'project_button_back' => 'Назад',
'project_button_cost_per_hour' => 'Час',
'project_button_request' => 'Заказать',
'project_create_architectures_title' => 'Выбор архитектуры проекта', 'project_architectures_title' => 'Выбор архитектуры проекта',
'project_create_architectures_description' => 'Каждая архитектура имеет уникальные параметры и коэффициенты \- это основа дальнейших расчётов\!', 'project_architectures_description' => 'Каждая архитектура имеет уникальные параметры и коэффициенты \- это основа дальнейших расчётов\!',
'project_create_button_architecture' => 'Архитектура', 'project_button_architecture' => 'Архитектура',
'project_create_button_architecture_selected' => 'Архитектура', 'project_button_architecture_selected' => 'Архитектура',
'project_create_purposes_title' => 'Выбор назначения', 'project_purposes_title' => 'Выбор назначения',
'project_create_purposes_description' => 'Вектор разработки, основание проекта', 'project_purposes_description' => 'Вектор разработки, основание проекта',
'project_create_button_purpose' => 'Назначение', 'project_button_purpose' => 'Назначение',
'project_create_button_purpose_selected' => 'Назнач.', 'project_button_purpose_selected' => 'Назнач.',
'project_create_integrations_title' => 'Выбор интеграций', 'project_integrations_title' => 'Выбор интеграций',
'project_create_integrations_description' => "Синхронизация заказов, товаров, публикаций и прочего в реальном времени, скачивание, загрузка, админ\-панель, рассылка сообщений, подключение аккаунтов\.\.\.\n\n_Отправка запросов в *API*, генерация и перехват *HTTP\-сообщений*, *эмуляция действий пользователя*_", 'project_integrations_description' => "Синхронизация заказов, товаров, публикаций и прочего в реальном времени, скачивание, загрузка, админ\-панель, рассылка сообщений, подключение аккаунтов\.\.\.\n\n_Отправка запросов в *API*, генерация и перехват *HTTP\-сообщений*, *эмуляция действий пользователя*_",
'project_create_button_integrations' => 'Интеграции', 'project_button_integrations' => 'Интеграции',
'project_create_button_integrations_selected' => 'Интеграции', 'project_button_integrations_selected' => 'Интеграции',
'project_create_button_team' => 'Команда: %d%s', 'project_button_team' => 'Команда: %d%s',
'project_create_peoples' => '', 'project_peoples' => '',
'project_create_requested' => 'Проект создан и отправлен оператору', 'project_cancelled' => 'Создание проекта отменено',
'project_create_cancelled' => 'Создание проекта отменено', 'project_deal_requested' => 'Проект создан и отправлен оператору',
'project_deal_title' => 'Заказ #%d',
'project_request_title' => 'Заказ #%d', 'project_deal_architecture' => 'Архитектура',
'project_request_architecture' => 'Архитектура', 'project_deal_purpose' => 'Назначение',
'project_request_purpose' => 'Назначение', 'project_deal_time' => 'Время',
'project_request_hours' => 'Часы', 'project_deal_time_hours' => 'ч',
'project_request_cost' => 'Стоимость', 'project_deal_time_hours_from' => 'от',
'project_request_team' => 'Команда', 'project_deal_time_days' => 'дн',
'project_request_empty' => 'Пусто', 'project_deal_cost' => 'Стоимость',
'project_request_button_accept' => 'Принять', 'project_deal_team' => 'Команда',
'project_request_button_refuse' => 'Отказать', 'project_deal_empty' => 'Пусто',
'project_request_button_edit' => 'Редактировать', 'project_deal_button_accept' => 'Принять',
'project_request_button_chat' => 'Чат с заказчиком', 'project_deal_button_refuse' => 'Отказать',
'project_deal_button_edit' => 'Редактировать',
'project_deal_button_chat' => 'Чат с заказчиком',
'project_accepted_deal_not_found' => 'Не удалось найти заявку',
'project_accepted_project_not_found' => 'Не удалось найти проект',
'project_accepted_already_confirmed' => 'Сделка уже проведена',
'project_accepted_title' => 'Заказ подтверждён\!',
'project_accepted_description' => 'Мы изучили предложение и *согласились* на условия',
'project_accepted_prepayment' => 'Предоплата',
'project_accepted_documents' => '_При желании оформить договор о неразглашении и передаче прав, *обратитесь к оператору*_',
'project_accepted_confirmed' => 'Подтверждено',
'project_accepted_button_prepayment' => 'Оплатить через СБП',
// Проект: типы // Проект: типы
'project_architecture_chat_robot' => 'Чат-робот', 'project_architecture_chat_robot' => 'Чат-робот',
@@ -126,14 +151,24 @@ return [
// Авторизация // Авторизация
'authorization_system' => 'Система', 'authorization_system' => 'Система',
'authorization_settings' => 'Настройки', 'authorization_settings' => 'Настройки',
'not_authorized_system' => 'У тебя нет доступа к системе', 'not_authorized_system' => 'Запрещён доступ к системе',
'not_authorized_settings' => 'У тебя нет доступа к настройкам', 'not_authorized_settings' => 'Запрещён доступ к настройкам',
'not_authorized_system_settings' => 'У тебя нет доступа к системным настройкам', 'not_authorized_system_settings' => 'Запрещён системный доступ к настройкам',
'not_authorized_system_deals' => 'Запрещён системный доступ к сделкам',
'not_authorized_system_projects' => 'Запрещён системный доступ к проектам ',
'not_authorized_system_invoices' => 'Запрещён системный доступ к счетам',
// Сообщения // Сообщения
'message_initialization_fail' => 'Не удалось инициализировать сообщение Телеграм', 'message_initialization_fail' => 'Не удалось инициализировать сообщение Телеграм',
'message_text_initialization_fail' => 'Не удалось инициализировать текст сообщения Телеграм', 'message_text_initialization_fail' => 'Не удалось инициализировать текст сообщения Телеграм',
'error_title' => 'Перегрузка\!',
'error_description' => 'Чат\-робот обрабатывает *длинную очередь* запросов',
'error_repeat' => 'Если не заработает *через 30 минут*, пишите оператору',
'error_account' => 'Аккаунт',
'error_button_chat_user' => 'Чат с пользователем',
'error_button_chat_operator' => 'Чат с оператором',
// Прочее // Прочее
'why_so_shroomious' => 'почему такой грибъёзный' 'why_so_shroomious' => 'почему такой грибъёзный'
]; ];

View File

@@ -10,7 +10,7 @@ use kodorvan\constructor\models\core,
kodorvan\constructor\models\settings, kodorvan\constructor\models\settings,
kodorvan\constructor\models\worker, kodorvan\constructor\models\worker,
kodorvan\constructor\models\project, kodorvan\constructor\models\project,
kodorvan\constructor\models\project\enumerations\type as project_type, kodorvan\constructor\models\project\enumerations\architecture as project_architecture,
kodorvan\constructor\models\project\enumerations\status as project_status; kodorvan\constructor\models\project\enumerations\status as project_status;
// The library for languages support // The library for languages support
@@ -150,7 +150,7 @@ final class account extends core implements record_interface
// Writing the updated record into the account object // Writing the updated record into the account object
$this->record = $updated; $this->record = $updated;
// Deserializing parameters // Deserializing the record
$this->deserialize(); $this->deserialize();
// Exit (success) // Exit (success)
@@ -166,7 +166,7 @@ final class account extends core implements record_interface
// Writing the found record into the account object // Writing the found record into the account object
$this->record = $account; $this->record = $account;
// Deserializing parameters // Deserializing the record
$this->deserialize(); $this->deserialize();
// Exit (success) // Exit (success)
@@ -374,6 +374,9 @@ final class account extends core implements record_interface
if ($authorizations instanceof authorizations) { if ($authorizations instanceof authorizations) {
// Found the account authorizations // Found the account authorizations
// Deserializing the record
$authorizations->deserialize();
// Exit (success) // Exit (success)
return $authorizations; return $authorizations;
} }
@@ -397,6 +400,9 @@ final class account extends core implements record_interface
if ($worker instanceof worker) { if ($worker instanceof worker) {
// Found the account worker // Found the account worker
// Deserializing the record
$worker->deserialize();
// Exit (success) // Exit (success)
return $worker; return $worker;
} }
@@ -420,6 +426,9 @@ final class account extends core implements record_interface
if ($settings instanceof settings) { if ($settings instanceof settings) {
// Found the account settings // Found the account settings
// Deserializing the record
$settings->deserialize();
// Exit (success) // Exit (success)
return $settings; return $settings;
} }
@@ -433,23 +442,23 @@ final class account extends core implements record_interface
* *
* Search for the account projects * Search for the account projects
* *
* @param project_type|null $type Type of the project * @param project_architecture|null $architecture architecture of the project
* @param project_status|null $status Status of the project * @param project_status|null $status Status of the project
* @param int $amount Maximum amount * @param int $amount Maximum amount
* *
* @return array The account projects * @return array The account projects
*/ */
public function projects( public function projects(
?project_type $type = null, ?project_architecture $architecture = null,
?project_status $status = null, ?project_status $status = null,
int $amount = 20 int $amount = 1000
): array { ): array {
// Search for the account projects // Search for the account projects
$projects = new project()->database->read( $projects = new project()->database->read(
filter: fn(record $record) => filter: fn(record $record) =>
$record->active === 1 $record->active === 1
&& $record->account === $this->identifier && $record->account === $this->identifier
&& ($type === null || $record->type === $type->name) && ($architecture === null || $record->architecture === $architecture->name)
&& ($status === null || $record->status === $status->name), && ($status === null || $record->status === $status->name),
amount: $amount amount: $amount
); );
@@ -475,7 +484,7 @@ final class account extends core implements record_interface
filter: fn(record $record) => filter: fn(record $record) =>
$record->active === 1 $record->active === 1
&& match (project_status::{$record->status}) { && match (project_status::{$record->status}) {
project_status::developing, project_status::developed, project_status::launched => true, project_status::developing, project_status::launched => true,
default => false default => false
}, },
amount: $amount amount: $amount

View File

@@ -75,8 +75,10 @@ final class authorizations extends core implements record_interface
new column('account', type::long_long_unsigned), new column('account', type::long_long_unsigned),
new column('system', type::char), new column('system', type::char),
new column('settings', type::char), new column('settings', type::char),
/* new column('', type::char), */
new column('system_settings', type::char), new column('system_settings', type::char),
new column('system_deals', type::char),
new column('system_invoices', type::char),
new column('system_projects', type::char),
new column('active', type::char), new column('active', type::char),
new column('updated', type::integer_unsigned), new column('updated', type::integer_unsigned),
new column('created', type::integer_unsigned) new column('created', type::integer_unsigned)
@@ -94,6 +96,9 @@ final class authorizations extends core implements record_interface
* @param bool $system * @param bool $system
* @param bool $settings * @param bool $settings
* @param bool $system_settings * @param bool $system_settings
* @param bool $system_deals
* @param bool $system_projects
* @param bool $system_invoices
* @param bool $active Is the record active? * @param bool $active Is the record active?
* *
* @return int|false The record, if created * @return int|false The record, if created
@@ -103,6 +108,9 @@ final class authorizations extends core implements record_interface
bool $system = true, bool $system = true,
bool $settings = true, bool $settings = true,
bool $system_settings = false, bool $system_settings = false,
bool $system_deals = false,
bool $system_projects = false,
bool $system_invoices = false,
bool $active = true, bool $active = true,
): record|false ): record|false
{ {
@@ -112,6 +120,9 @@ final class authorizations extends core implements record_interface
(int) $system, (int) $system,
(int) $settings, (int) $settings,
(int) $system_settings, (int) $system_settings,
(int) $system_deals,
(int) $system_projects,
(int) $system_invoices,
(int) $active, (int) $active,
svoboda::timestamp(), svoboda::timestamp(),
svoboda::timestamp() svoboda::timestamp()
@@ -142,6 +153,9 @@ final class authorizations extends core implements record_interface
$this->record->system = (int) $this->record->system; $this->record->system = (int) $this->record->system;
$this->record->settings = (int) $this->record->settings; $this->record->settings = (int) $this->record->settings;
$this->record->system_settings = (int) $this->record->system_settings; $this->record->system_settings = (int) $this->record->system_settings;
$this->record->system_deals = (int) $this->record->system_deals;
$this->record->system_projects = (int) $this->record->system_projects;
$this->record->system_invoices = (int) $this->record->system_invoices;
$this->record->active = (int) $this->record->active; $this->record->active = (int) $this->record->active;
// Writing the status of serializing // Writing the status of serializing
@@ -169,6 +183,9 @@ final class authorizations extends core implements record_interface
$this->record->system = (bool) $this->record->system; $this->record->system = (bool) $this->record->system;
$this->record->settings = (bool) $this->record->settings; $this->record->settings = (bool) $this->record->settings;
$this->record->system_settings = (bool) $this->record->system_settings; $this->record->system_settings = (bool) $this->record->system_settings;
$this->record->system_deals = (bool) $this->record->system_deals;
$this->record->system_projects = (bool) $this->record->system_projects;
$this->record->system_invoices = (bool) $this->record->system_invoices;
$this->record->active = (bool) $this->record->active; $this->record->active = (bool) $this->record->active;
// Writing the status of serializing // Writing the status of serializing

View File

@@ -0,0 +1,240 @@
<?php
declare(strict_types=1);
namespace kodorvan\constructor\models;
// Files of the project
use kodorvan\constructor\models\core,
kodorvan\constructor\models\deal\enumerations\direction as deal_direction,
kodorvan\constructor\models\project\enumerations\status as project_status,
kodorvan\constructor\models\project\enumerations\architecture as project_architecture,
kodorvan\constructor\models\project\enumerations\purpose as project_purpose,
kodorvan\constructor\models\project\enumerations\integration as project_integration,
kodorvan\constructor\models\worker\enumerations\type as worker_type;
// 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,
LogicException as exception_logic,
RuntimeException as exception_runtime;
/**
* Deal
*
* @package kodorvan\constructor\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 deal extends core implements record_interface
{
use record_trait;
/**
* File
*
* @var string $file Path to the database file
*/
protected string $file = DATABASES . DIRECTORY_SEPARATOR . 'projects' . DIRECTORY_SEPARATOR . 'deals.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('account', type::long_long_unsigned),
new column('project', type::long_long_unsigned),
new column('direction', type::char),
new column('description', type::string, ['length' => 512]),
new column('hours', type::integer_unsigned),
new column('cost', type::float),
new column('payment', type::float),
new column('prepayment', type::float),
new column('programmers', type::integer_unsigned),
new column('designers', type::integer_unsigned),
new column('boosters', type::integer_unsigned),
new column('active', type::char),
new column('confirmed', type::integer_unsigned),
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
*
* @throws exception_logic when failed to process project integration
*
* @param int $account The account identifier
* @param int $project The project identifier
* @param deal_direction $direction Direction of the deal
* @param string|null $description Description of the project
* @param int $hours Hours of the project development
* @param int|float $cost Cost per hour of the project development
* @param int|float $payment Payment of the project development
* @param int|float $prepayment Prepayment of the project development
* @param int $programmers Programmers of the project
* @param int $designers Designers of the project
* @param int $boosters Boosters of the project
* @param int $active Is the record active?
*
* @return record|false The record, if created
*/
public function write(
int $account,
int $project,
deal_direction $direction,
?string $description = null,
int $hours = PROJECT_HOURS_MINIMAL,
int|float $cost = PROJECT_COST_HOUR_DEFAULT,
int|float $payment,
int|float $prepayment,
int $programmers = 0,
int $designers = 0,
int $boosters = 0,
bool $active = true,
): record|false {
// Initializing the record
$record = $this->database->record(
$this->database->count() + 1,
$account,
$project,
$direction->value,
$description,
$hours,
(float) $cost,
(float) $payment,
(float) $prepayment,
$programmers,
$designers,
$boosters,
(int) $active,
0,
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 implementor is already serialized');
}
// Serializing the record parameters
$this->record->direction = $this->record->direction->value;
$this->record->active = (int) $this->record->active;
// Writing the status of serializing
$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 implementor is already deserialized');
}
// Deserializing the record parameters
$this->record->direction = deal_direction::from($this->record->direction);
$this->record->active = (bool) $this->record->active;
// Writing the status of serializing
$this->serialized = false;
// Exit (success)
return $this;
}
/**
* Project
*
* Search for the project
*
* @return project|null The project
*/
public function project(): ?project
{
// Search for the account project
$project = new project()->read(filter: fn(record $record) => $record->identifier === $this->project && $record->active === 1);
if ($project instanceof project) {
// Found the account project
// Deserializing the project
$project->deserialize();
// Exit (success)
return $project;
}
// Exit (fail)
return null;
}
}

View File

@@ -0,0 +1,23 @@
<?php
declare(strict_types=1);
namespace kodorvan\constructor\models\deal\enumerations;
// Built-in libraries
use InvalidArgumentException as exception_argument,
DomainException as exception_domain;
/**
* Direction
*
* @package kodorvan\neurobot\models\deal\enumerations
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
enum direction: int
{
case inbound = 0;
case outbound = 1;
}

View File

@@ -7,7 +7,10 @@ namespace kodorvan\constructor\models;
// Files of the project // Files of the project
use kodorvan\constructor\models\core, use kodorvan\constructor\models\core,
kodorvan\constructor\models\project\enumerations\status as project_status, kodorvan\constructor\models\project\enumerations\status as project_status,
kodorvan\constructor\models\project\enumerations\status as project_type; kodorvan\constructor\models\project\enumerations\architecture as project_architecture,
kodorvan\constructor\models\project\enumerations\purpose as project_purpose,
kodorvan\constructor\models\project\enumerations\integration as project_integration,
kodorvan\constructor\models\worker\enumerations\type as worker_type;
// Baza database // Baza database
use mirzaev\baza\database, use mirzaev\baza\database,
@@ -25,6 +28,7 @@ use svoboda\time\statement as svoboda;
// Built-in libraries // Built-in libraries
use Exception as exception, use Exception as exception,
LogicException as exception_logic,
RuntimeException as exception_runtime; RuntimeException as exception_runtime;
/** /**
@@ -44,7 +48,7 @@ final class project extends core implements record_interface
* *
* @var string $file Path to the database file * @var string $file Path to the database file
*/ */
protected string $file = DATABASES . DIRECTORY_SEPARATOR . 'project.baza'; protected string $file = DATABASES . DIRECTORY_SEPARATOR . 'projects.baza';
/** /**
* Database * Database
@@ -75,9 +79,15 @@ final class project extends core implements record_interface
->columns( ->columns(
new column('identifier', type::long_long_unsigned), new column('identifier', type::long_long_unsigned),
new column('account', type::long_long_unsigned), new column('account', type::long_long_unsigned),
new column('status', type::string, ['length' => 16]), new column('deal', type::long_long_unsigned),
new column('type', type::string, ['length' => 32]),
new column('name', type::string, ['length' => 64]), new column('name', type::string, ['length' => 64]),
new column('status', type::string, ['length' => 16]),
new column('architecture', type::string, ['length' => 32]),
new column('purpose', type::string, ['length' => 32]),
new column('integrations', type::integer_unsigned),
/* new column('programmers', type::integer_unsigned),
new column('designers', type::integer_unsigned),
new column('boosters', type::integer_unsigned), */
/* new column('', type::), */ /* new column('', type::), */
new column('active', type::char), new column('active', type::char),
new column('updated', type::integer_unsigned), new column('updated', type::integer_unsigned),
@@ -92,34 +102,85 @@ final class project extends core implements record_interface
/** /**
* Write * Write
* *
* @throws exception_logic when failed to process project integration
*
* @param int $account The account identifier * @param int $account The account identifier
* @param int|null $deal The deal identifier
* @param project_status $status Status of the project * @param project_status $status Status of the project
* @param project_type $status Type of the project
* @param string|null $name Name of the project * @param string|null $name Name of the project
* @param project_architecture|null $architecture Architecture of the project
* @param project_purpose|null $purpose Purpose of the project
* @param int|array $Inegrations Integrations of the project
* @param int $programmers Programmers of the project
* @param int $designers Designers of the project
* @param int $boosters Boosters of the project
* @param int $active Is the record active? * @param int $active Is the record active?
* *
* @return record|false The record, if created * @return record|false The record, if created
*/ */
public function write( public function write(
int $account, int $account,
project_status $status = project_status::creating, ?int $deal = null,
project_type $type = project_type::special,
?string $name = null, ?string $name = null,
project_status $status = project_status::creating,
?project_architecture $architecture = null,
?project_purpose $purpose = null,
int|array $integrations = 0b000000,
/* int $programmers = 0,
int $designers = 0,
int $boosters = 0, */
bool $active = true, bool $active = true,
): record|false { ): record|false {
if (empty($name)) { if (empty($name)) {
// Not received the project name // Not received the project name
// Generating the project name // Generating the project name
$name = 'Project №' . count(new account()->read(filter: fn(record $record) => $record->active === 1 && $record->account === $account)?->projects() ?? []); $name = 'Project №' . count(new account()->read(filter: fn(record $record) => $record->active === 1 && $record->identifier === $account)?->projects() ?? []);
} }
if (is_array($integrations)) {
// Received integrations in array format
// Initializing the project integrations buffer
$buffer = 0b000000;
foreach ($integrations as $integration) {
// Iterating over integrations
if ($integration instanceof project_integration) {
// Project integration
// Writing the project integration into the project integrations buffer
$buffer |= $integration->value;
} else {
// Not project integration
throw new exception_logic('Failed to process project integration');
}
}
// Reinitializing the project integrations
$integrations = $buffer;
// Deinitializing the project integrations buffer
unset($buffer);
}
// Initializing the record
$record = $this->database->record( $record = $this->database->record(
$this->database->count() + 1, $this->database->count() + 1,
$account, $account,
$status->name, (int) $deal,
$type->name,
$name, $name,
$status->name,
$architecture?->name ?? '',
$purpose?->name ?? '',
$integrations,
/* $programmers,
$designers,
$boosters, */
(int) $active, (int) $active,
svoboda::timestamp(), svoboda::timestamp(),
svoboda::timestamp() svoboda::timestamp()
@@ -147,9 +208,11 @@ final class project extends core implements record_interface
} }
// Serializing the record parameters // Serializing the record parameters
$this->record->active = (int) $this->record->active;
$this->record->status = $this->record->status->name; $this->record->status = $this->record->status->name;
$this->record->type = $this->record->type->name; $this->record->architecture = $this->record->architecture?->name ?? '';
$this->record->purpose = $this->record->purpose?->name ?? '';
$this->record->integrations = project_integration::encode($this->record->integrations);
$this->record->active = (int) $this->record->active;
// Writing the status of serializing // Writing the status of serializing
$this->serialized = true; $this->serialized = true;
@@ -173,9 +236,15 @@ final class project extends core implements record_interface
} }
// Deserializing the record parameters // Deserializing the record parameters
$this->record->active = (bool) $this->record->active;
$this->record->status = project_status::{$this->record->status}; $this->record->status = project_status::{$this->record->status};
$this->record->type = project_status::{$this->record->type}; $this->record->architecture = $this->architecture instanceof project_architecture
? project_architecture::{$this->architecture->type}
: null;
$this->record->purpose = $this->purpose instanceof project_purpose
? project_purpose::{$this->purpose->type}
: null;
$this->record->integrations = project_integration::decode($this->record->integrations);
$this->record->active = (bool) $this->record->active;
// Writing the status of serializing // Writing the status of serializing
$this->serialized = false; $this->serialized = false;
@@ -185,32 +254,205 @@ final class project extends core implements record_interface
} }
/** /**
* Parameters * Account
* *
* Search for all the project properties * Search for the account
* *
* @return self The instance from which the method was called (fluent interface) * @return account|null The account
*/ */
public function parameters(): self public function account(): ?account
{ {
// Deserializing the record parameters // Search for the account account
$this->record->active = (bool) $this->record->active; $account = new account()->read(filter: fn(record $record) => $record->identifier === $this->account && $record->active === 1);
$this->record->status = project_status::{$this->record->status};
if (!$this->serialized) { if ($account instanceof account) {
// Not serialized // Found the account account
} else {
// Serialized
// Exit (fail)
throw new exception('The project implementator is serialized');
}
/* if ($this->record->type === '') */
// Deserializing the record
$account->deserialize();
// Exit (success) // Exit (success)
return $this; return $account;
}
// Exit (fail)
return null;
}
/**
* Hours
*
* Calculate the project development hours
*
* @throws exception_runtime The record is not deserialized
*
* @param bool $absolute Summary all coefficients and then multiply?
*
* @return int|float The project development hours
*/
public function hours(bool $absolute = false): int
{
if ($this->serialized) {
// The record is serialized
throw new exception_runtime('The record is not deserialized');
}
// Initializing start hours
$start = PROJECT_START_HOURS ?? 1;
$start < 1 and $start = 1;
// Initializing additional hours
$additional = PROJECT_HOURS_ADDITIONAL ?? 0;
if ($absolute) {
// The absolute coefficient
// Declaring coefficient
$coefficient = PROJECT_START_COEFFICIENT ?? 0;
if (isset($this->architecture)) {
// Initialized the project architecture
// Adding into the coefficient
$coefficient += $this->architecture->coefficient() ?? 0;
}
if (isset($this->purpose)) {
// Initialized the project purpose
// Adding into the coefficient
$coefficient += $this->purpose->coefficient() ?? 0;
}
if (!empty($this->integrations)) {
// Initialized the project integrations
foreach ($this->integrations as $integration) {
// Iterating over the project integrations
// Adding into the coefficient
$coefficient += $integration->coefficient() ?? 0;
}
}
// Calculating the development hours
$hours = $start * $coefficient + $additional;
// Calculating and exit (success)
return (int) ceil(max($hours, PROJECT_HOURS_MINIMAL));
} else {
// The relative coefficient
// Initializing the development hours
$hours = $start;
if (isset($this->architecture)) {
// Initialized the project architecture
// Adding into the coefficient
$hours *= $this->architecture->coefficient() ?? 1;
}
if (isset($this->purpose)) {
// Initialized the project purpose
// Adding into the coefficient
$hours *= $this->purpose->coefficient() ?? 1;
}
if (!empty($this->integrations)) {
// Initialized the project integrations
foreach ($this->integrations as $integration) {
// Iterating over the project integrations
// Adding into the coefficient
$hours *= $integration->coefficient() ?? 1;
}
}
// Calculating with additional hours
$hours += $additional;
// Calculating and exit (success)
return (int) ceil(max($hours, PROJECT_HOURS_MINIMAL));
}
}
/**
* Payment
*
* Calculate the project development payment
*
* @param int|float $cost Cost per hour
* @param int $hours The project development hours
* @param int $programmers Programmers
* @param int $designers Designers
* @param int $boosters Boosters
*
* @return array ['full' => int|float, 'prepayment' => int|float]
*/
public function payment(
int|float $cost = PROJECT_COST_HOUR_DEFAULT,
int $hours = PROJECT_HOURS_MINIMAL,
int $programmers = 0,
int $designers = 0,
int $boosters = 0
): array {
// Initializing costs
$costs = [
'full' => ceil($hours * $cost),
'prepayment' => null
];
// Initializing default workers amounts
$default = [
'programmers' => $this->architecture?->workers()[worker_type::programmer->name] ?? 0,
'designers' => $this->architecture?->workers()[worker_type::designer->name] ?? 0,
'boosters' => $this->architecture?->workers()[worker_type::booster->name] ?? 0
];
if ($programmers > $default['programmers']) {
// Programmers amount more than default
// Calculating the full cost
$costs['full'] *= $programmers * PROJECT_WORKERS_PROGRAMMERS_COEFFICIENT;
} else if ($programmers < $default['programmers']) {
// Programmers amount less than default
// Calculating the full cost
$costs['full'] /= max($programmers, 1) * PROJECT_WORKERS_PROGRAMMERS_COEFFICIENT;
}
if ($designers > $default['designers']) {
// Designers amount more than default
// Calculating the full cost
$costs['full'] *= $designers * PROJECT_WORKERS_DESIGNERS_COEFFICIENT;
} else if ($designers < $default['designers']) {
// Designers amount less than default
// Calculating the full cost
$costs['full'] /= max($designers, 1) * PROJECT_WORKERS_DESIGNERS_COEFFICIENT;
}
if ($boosters > $default['boosters']) {
// Boosters amount more than default
// Calculating the full cost
$costs['full'] *= $boosters * PROJECT_WORKERS_BOOSTERS_COEFFICIENT;
} else if ($boosters < $default['boosters']) {
// Boosters amount less than default
// Calculating the full cost
$costs['full'] /= max($boosters, 1) * PROJECT_WORKERS_BOOSTERS_COEFFICIENT;
}
// Calculating the prepayment
$costs['prepayment'] = ceil($costs['full'] * (PROJECT_COST_PREPAYMENT_PERCENTS / 100));
// Exit (success)
return $costs;
} }
} }

View File

@@ -19,14 +19,14 @@ use InvalidArgumentException as exception_argument,
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/ */
enum integration enum integration: int
{ {
case one_c; case one_c = 0b000001;
case bitrix24; case bitrix24 = 0b000010;
case moy_sklad; case moy_sklad = 0b000100;
case telegram; case telegram = 0b001000;
case mail; case mail = 0b010000;
case excel; case excel = 0b100000;
/** /**
* Label * Label
@@ -96,4 +96,54 @@ enum integration
default => 2 default => 2
}; };
} }
/**
* Decode
*
* @param int $bitmask The encoded cases
*
* @return array Decoded cases
*/
public static function decode(int $bitmask): array
{
// Initializing the registry of decoded cases
$decoded = [];
foreach (static::cases() as $case) {
// Iterating over cases
if ($bitmask & $case->value) {
// Decoded the case
// Decoding the case and writing into the registry of decoded cases
$decoded[] = $case;
}
}
// Exit (success)
return $decoded;
}
/**
* Encode
*
* @param array $cases The decoded cases
*
* @return int Encoded cases bitmask
*/
public static function encode(array $cases):int
{
// Initializing the registry of encoded cases
$encoded = 0b000000;
foreach ($cases as $case) {
// Iterating over cases
// Encoding the case and writing into the registry of encoded cases
$encoded |= $case->value;
}
// Exit (success)
return $encoded;
}
} }

View File

@@ -95,10 +95,10 @@ enum purpose
language::en => 'Calculate', language::en => 'Calculate',
language::ru => 'Расчёты' language::ru => 'Расчёты'
}, },
/* static::logic => match ($language) { static::logic => match ($language) {
language::en => 'Logic', language::en => 'Logic',
language::ru => 'Логика' language::ru => 'Логика'
}, */ },
static::game => match ($language) { static::game => match ($language) {
language::en => 'Game', language::en => 'Game',
language::ru => 'Игра' language::ru => 'Игра'

View File

@@ -19,12 +19,8 @@ use InvalidArgumentException as exception_argument,
enum status enum status
{ {
case creating; case creating;
case calculated;
case requested; case requested;
case invoiced; case invoiced;
case developing; case developing;
case developed;
case launched; case launched;
} }

View File

@@ -82,10 +82,6 @@ final class start extends command
// Initializing the account // Initializing the account
$account = $robot->get('account'); $account = $robot->get('account');
// 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]) ? '' : "🔏 *$localization->menu_update:* " . unmarkdown($git[0]);
// Calculating amount of projects // Calculating amount of projects
$projects = count($account->projects()); $projects = count($account->projects());
@@ -115,14 +111,42 @@ final class start extends command
) )
); );
// Title
$title = "📋 *$localization->menu_title*";
// Declaring the message variables
$welcome = $cooperation = null;
if ($projects > 0) {
// The account have projects
// Welcome
$welcome = "🤟 *$localization->menu_description_partner*";
// Cooperation
$cooperation = $localization->menu_cooperation;
} else {
// The account have not projects
// Welcome
$welcome = $localization->menu_description_guest;
}
// Update
/* 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); // Formatted */
exec(command: 'git log --oneline $(git describe --tags --abbrev=0 @^ --always)..@ -1 --format="%at"', output: $git);
$update = empty($git[0]) ? '' : "🔏 *$localization->menu_update:* ![" . $git[0] . '](tg://time?unix=' . $git[0] . '&format=r)';
// Sending the message
$robot->sendMessage( $robot->sendMessage(
text: implode( text: implode(
"\n\n", "\n\n",
[ array_filter([
"📋 *$localization->menu_title*", $title,
$projects > 0 ? printf($localization->menu_description_partner, $partners) : $localization->menu_description_guest, $welcome,
$cooperation,
$update $update
] ])
), ),
parse_mode: mode::MARKDOWN, parse_mode: mode::MARKDOWN,
disable_notification: true, disable_notification: true,

View File

@@ -0,0 +1,48 @@
<?php
declare(strict_types=1);
namespace kodorvan\constructor\models\telegram;
// Files of the project
use kodorvan\constructor\models\core,
kodorvan\constructor\models\account,
kodorvan\constructor\models\localization,
kodorvan\constructor\models\settings as model,
kodorvan\constructor\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 SergiX44\Nutgram\Nutgram as telegram,
SergiX44\Nutgram\Telegram\Exceptions\TelegramException as telegram_exception,
SergiX44\Nutgram\Exception\ApiException as telegram_api_exception;
// Built-in libraries
use Error as error;
/**
* Telegram exceptions
*
* @package kodorvan\constructor\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*
* @deprecated
*/
final class exceptions extends telegram_api_exception
{
public static ?string $pattern = '.*';
public function __invoke(telegram $robot, telegram_exception $exception)
{
// override this method to change the default behaviour:
$robot->sendMessage('robot zdox');
throw new static($exception->getMessage(), $exception->getCode(), $exception);
}
}

View File

@@ -8,8 +8,13 @@ namespace kodorvan\constructor\models\telegram;
use kodorvan\constructor\models\core, use kodorvan\constructor\models\core,
kodorvan\constructor\models\account, kodorvan\constructor\models\account,
kodorvan\constructor\models\localization, kodorvan\constructor\models\localization,
kodorvan\constructor\models\settings as model, kodorvan\constructor\models\settings,
kodorvan\constructor\models\telegram\processes\language\select as process_language_select; kodorvan\constructor\models\deal,
kodorvan\constructor\models\project as model,
kodorvan\constructor\models\project\enumerations\status as project_status,
kodorvan\constructor\models\worker\enumerations\type as worker_type,
kodorvan\constructor\models\telegram\processes\language\select as process_language_select,
kodorvan\constructor\models\telegram\conversations\project as conversation_project;
// Library for languages support // Library for languages support
use mirzaev\languages\language; use mirzaev\languages\language;
@@ -17,6 +22,16 @@ use mirzaev\languages\language;
// The library for escaping all markdown symbols // The library for escaping all markdown symbols
use function mirzaev\unmarkdown; use function mirzaev\unmarkdown;
// Svoboda time
use svoboda\time\statement as svoboda;
// Baza database
use mirzaev\baza\database,
mirzaev\baza\column,
mirzaev\baza\record,
mirzaev\baza\enumerations\encoding,
mirzaev\baza\enumerations\type;
// Framework for Telegram // Framework for Telegram
use SergiX44\Nutgram\Nutgram as telegram, use SergiX44\Nutgram\Nutgram as telegram,
SergiX44\Nutgram\Telegram\Properties\ParseMode as mode, SergiX44\Nutgram\Telegram\Properties\ParseMode as mode,
@@ -27,7 +42,8 @@ use SergiX44\Nutgram\Nutgram as telegram,
SergiX44\Nutgram\Telegram\Types\Keyboard\InlineKeyboardButton as button; SergiX44\Nutgram\Telegram\Types\Keyboard\InlineKeyboardButton as button;
// Built-in libraries // Built-in libraries
use Error as error; use DateTime as datetime,
Error as error;
/** /**
* Telegram project * Telegram project
@@ -40,42 +56,260 @@ use Error as error;
final class project extends core final class project extends core
{ {
/** /**
* Language * Create
* *
* Write the language into the account and the robot instance * Starting the project creating process
* *
* @param telegram $robot The chat-robot instance * @param telegram $robot The chat-robot instance
* @param language $language The language
* *
* @return void * @return void
*/ */
public static function create(telegram $robot, language $language = LANGUAGE_DEFAULT): void public static function create(telegram $robot): void
{ {
/* // Initializing the account // Initializing the account
$account = $robot->get('account'); $account = $robot->get('account');
if ($account instanceof account) { // Initializing the project record
// Initialized the account $record = new model()->write(account: $account->identifier);
// Initializing the menu message localization // Initializing the project
$localization = new localization($language); $project = new model(record: $record);
if ($localization instanceof localization) { // Deserializing the record
// Initialized the localization $project->deserialize();
// Starting the project creating process
conversation_project::begin(
bot: $robot,
userId: $robot->userId(),
chatId: $robot->chatId(),
data: ['instance' => $project]
);
}
/**
* Accept
*
* Accept the project and issue an invoice
*
* @param telegram $robot The robot
*
* @return void
*/
public function accept(telegram $robot): void
{
// Sending the "typing" action
/* $robot->sendChatAction('typing'); */
// Initializing the account language
$language = $robot->get('language') ?? LANGUAGE_DEFAULT;
// Initializing the account localization
$localization = $robot->get('localization') ?? new localization($language);
// Initializing the account
$account = $robot->get('account');
// Initializing the account authorizations
$authorizations = $account->authorizations();
if ($authorizations->system_deals) {
// Authorized to deals (system)
if ($authorizations->system_projects) {
// Authorized to projects (system)
if ($authorizations->system_invoices) {
// Authorized to invoices (system)
// The message
$message = $robot->message();
// The message text
$text = $message?->text;
// The project identifier
preg_match('/^.*#(\d+)$/m', $text, $matches);
$identifier = (int) $matches[1];
unset($matches);
// The deal
$deal = new deal()->read(filter: fn(record $record) => $record->identifier === $identifier && $record->active === 1);
// Deserializing the record
$deal->deserialize();
if ($deal instanceof deal) {
// Initialized the deal
// The project
$project = $deal->project();
if ($project instanceof model) {
// Initialized the project
if ($project->status === project_status::requested && $deal->confirmed === 0) {
// The project deal is not confirmed
// Initializing the keyboard
$keyboard = keyboard::make();
// Writing the row into the keyboard
$keyboard->addRow(
button::make(
text: "🔏 $localization->project_accepted_button_prepayment",
url: 'https://t.me/' . $robot->user()->username
)
);
// Initializing the receiver account
$receiver = $project->account();
// Title
$title = "🏗 *$localization->project_accepted_title*";
// Description
$description = $localization->project_accepted_description;
// Prepayment
$prepayment = "*$localization->project_accepted_prepayment:* $deal->prepayment" . $receiver->currency->symbol();
// Documents
$documents = $localization->project_accepted_documents;
// Sending the message // Sending the message
$robot->sendMessage( $robot->sendMessage(
text: "✅ *$localization->settings_language_update_success:* " . trim($from->flag() . ' ' . $from->label($to)) . ' → *' . trim($to->flag() . ' ' . $to->label($to)) . '*', text: implode(
"\n\n",
array_filter([
$title,
$description,
$documents,
$prepayment
])
),
chat_id: $receiver->identifier_telegram,
parse_mode: mode::MARKDOWN, parse_mode: mode::MARKDOWN,
disable_notification: true disable_notification: true,
reply_markup: $keyboard
); );
// Sending the message // Ending the conversation
$robot->answerCallbackQuery( $robot->endConversation();
text: $to->label($to),
show_alert: false // Writing the confirmation date
$deal->confirmed = svoboda::timestamp();
// Serializing the record
$deal->serialize();
// Updating the deal record
$deal->update();
// Deserializing the record
$deal->deserialize();
// Initializing the keyboard
$keyboard = keyboard::make();
// Writing the row into the keyboard
$keyboard->addRow(
button::make(
text: "✉️ $localization->project_deal_button_chat",
url: 'https://t.me/' . $robot->user()->username
)
); );
// Converting the confirmation to unixtime format
$unixtime = $deal->confirmed + svoboda::datetime()->getTimestamp();
// Confirmed
/* $confirmed = "*$localization->project_accepted_confirmed*: ![" . $unixtime . '](tg://time?unix=' . $unixtime . '&format=r)'; */
$confirmed = "$localization->project_accepted_confirmed: ![" . $unixtime . '](tg://time?unix=' . $unixtime . '&format=r)';
// Updating the message buttons
$message->editText(
text: implode(
"\n\n",
[
$message->getText(),
$confirmed
]
),
/* parse_mode: mode::MARKDOWN, */
reply_markup: $keyboard
);
} else {
// The project deal is confirmed
// Sending the message
$robot->sendMessage(
text: "⚠️ *$localization->project_accepted_already_confirmed*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
}
} else {
// Not initialized the project
// Sending the message
$robot->sendMessage(
text: "⚠️ *$localization->project_accepted_project_not_found*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
}
} else {
// Not initialized the project
// Sending the message
$robot->sendMessage(
text: "⚠️ *$localization->project_accepted_deal_not_found*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
}
} else {
// Not authorized to invoices (system)
// Sending the message
$robot->sendMessage(
text: "⛔ *$localization->not_authorized_system_invoices*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
}
} else {
// Not authorized to projects (system)
// Sending the message
$robot->sendMessage(
text: "⛔ *$localization->not_authorized_system_projects*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
}
} else {
// Not authorized to deals (system)
// Sending the message
$robot->sendMessage(
text: "⛔ *$localization->not_authorized_system_deals*",
parse_mode: mode::MARKDOWN,
);
// Ending the conversation
$robot->endConversation();
} }
} */
} }
} }

View File

@@ -1,52 +0,0 @@
<?php
declare(strict_types=1);
namespace kodorvan\constructor;
// Framework for PHP
use mirzaev\minimal\core,
mirzaev\minimal\route;
// 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');
// Initializing dependencies
require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
// Initializing core
$core = new core(namespace: __NAMESPACE__);
// Initializing routes
$core->router
->write('/', new route('index', 'index'), 'GET')
;
// Handling request
$core->start();

View File

@@ -6,12 +6,14 @@ namespace kodorvan\constructor;
// Files of the project // Files of the project
use kodorvan\constructor\models\account, use kodorvan\constructor\models\account,
kodorvan\constructor\models\telegram\settings, kodorvan\constructor\models\localization,
kodorvan\constructor\models\telegram\settings as telegram_settings,
kodorvan\constructor\models\telegram\project as telegram_project,
kodorvan\constructor\models\telegram\commands\start as command_start, kodorvan\constructor\models\telegram\commands\start as command_start,
kodorvan\constructor\models\telegram\commands\account as command_account, kodorvan\constructor\models\telegram\commands\account as command_account,
kodorvan\constructor\models\telegram\commands\society as command_society, kodorvan\constructor\models\telegram\commands\society as command_society,
kodorvan\constructor\models\telegram\commands\language as command_language, kodorvan\constructor\models\telegram\commands\language as command_language,
kodorvan\constructor\models\telegram\conversations\project\create as conversation_project_create, /* kodorvan\constructor\models\telegram\conversations\project\create as conversation_project_create, */
kodorvan\constructor\models\telegram\middlewares\account as middleware_account, kodorvan\constructor\models\telegram\middlewares\account as middleware_account,
kodorvan\constructor\models\telegram\middlewares\language as middleware_language, kodorvan\constructor\models\telegram\middlewares\language as middleware_language,
kodorvan\constructor\models\telegram\middlewares\localization as middleware_localization, kodorvan\constructor\models\telegram\middlewares\localization as middleware_localization,
@@ -25,15 +27,25 @@ use mirzaev\languages\language;
use mirzaev\minimal\core, use mirzaev\minimal\core,
mirzaev\minimal\route; mirzaev\minimal\route;
// The library for escaping all markdown symbols
use function mirzaev\unmarkdown;
// Framework for Telegram // Framework for Telegram
use SergiX44\Nutgram\Nutgram as telegram, use SergiX44\Nutgram\Nutgram as framework_telegram,
SergiX44\Nutgram\Configuration as telegram_settings, SergiX44\Nutgram\Configuration as framework_telegram_settings,
SergiX44\Nutgram\RunningMode\Webhook as webhook; SergiX44\Nutgram\RunningMode\Webhook as framework_telegram_webhook,
SergiX44\Nutgram\Telegram\Properties\ParseMode as mode,
SergiX44\Nutgram\Telegram\Exceptions\TelegramException as framework_telegram_exception,
SergiX44\Nutgram\Telegram\Types\Keyboard\InlineKeyboardMarkup as keyboard,
SergiX44\Nutgram\Telegram\Types\Keyboard\InlineKeyboardButton as button;
// The symphony cache library // The symphony cache library
use Symfony\Component\Cache\Adapter\FilesystemAdapter as cache_adapter, use Symfony\Component\Cache\Adapter\FilesystemAdapter as cache_adapter,
Symfony\Component\Cache\Psr16Cache as cache; Symfony\Component\Cache\Psr16Cache as cache;
// Built-in libraries
use Exception as exception;
// Enabling debugging // Enabling debugging
/* ini_set('error_reporting', E_ALL); /* ini_set('error_reporting', E_ALL);
ini_set('display_errors', 1); ini_set('display_errors', 1);
@@ -70,18 +82,19 @@ define('TELEGRAM', require(SETTINGS . DIRECTORY_SEPARATOR . 'telegram.php'));
require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php'; require ROOT . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
// Initializing the robot // Initializing the robot
$robot = new telegram( $robot = new framework_telegram(
token: TELEGRAM['key'], token: TELEGRAM['key'],
config: new telegram_settings( config: new framework_telegram_settings(
botName: TELEGRAM['name'], botName: TELEGRAM['name'],
cache: new cache(new cache_adapter()) cache: new cache(new cache_adapter())
) )
); );
$webhook = new webhook(secretToken: TELEGRAM['password']); $webhook = new framework_telegram_webhook(secretToken: TELEGRAM['password']);
$webhook->setSafeMode(true); $webhook->setSafeMode(true);
$robot->setRunningMode($webhook); $robot->setRunningMode($webhook);
$robot->throttle(10);
$robot->middleware(middleware_account::class); $robot->middleware(middleware_account::class);
$robot->middleware(middleware_language::class); $robot->middleware(middleware_language::class);
@@ -103,13 +116,108 @@ foreach (language::cases() as $language) {
// Iterating over languages // Iterating over languages
// Select the language // Select the language
$robot->onCallbackQueryData('settings_language_$language->name', fn(telegram $robot) => settings::language(robot: $robot, language: $language)); $robot->onCallbackQueryData('settings_language_$language->name', fn(framework_telegram $robot) => telegram_settings::language(robot: $robot, language: $language));
}; };
// Society // Society
$robot->registerCommand(command_society::class); $robot->registerCommand(command_society::class);
// Project: create // Project: create
$robot->onCallbackQueryData('project_create', conversation_project_create::class); $robot->onCallbackQueryData('project_create', [telegram_project::class, 'create']);
$robot->run(); // Project: request
$robot
->onCallbackQueryData('project_deal_accept', [telegram_project::class, 'accept']);
$robot->onApiError(function (framework_telegram $robot, framework_telegram_exception $exception) {
try {
// Writing into the errors output
error_log($exception->getMessage());
// Initializing the account language
$language = $robot->get('language') ?? LANGUAGE_DEFAULT;
// Initializing the account localization
$localization = $robot->get('localization') ?? new localization($language);
// Initializing the keyboard
$keyboard = keyboard::make();
// Writing the row into the keyboard
$keyboard->addRow(
button::make(
text: "✉️ $localization->error_button_chat_operator",
url: PROJECT_OPERATOR_URL ?? PROJECT_MEDIA_URL ?? 'https://t.me/kodorvan'
)
);
// Title
$title = "🔥 *$localization->error_title*";
// Description
$description = $localization->error_description;
// Repeat
$repeat = $localization->error_repeat;
// Sending the message
$robot->sendMessage(
text: implode(
"\n\n",
[
$title,
$description,
$repeat
]
),
disable_notification: true,
parse_mode: mode::MARKDOWN,
reply_markup: $keyboard
);
// Ending the conversation
$robot->endConversation();
foreach (ERRORS_RECEIVERS as $receiver) {
// Iterating over errors receivers
// Initializing the keyboard
$keyboard = keyboard::make();
// Writing the row into the keyboard
$keyboard->addRow(
button::make(
text: "✉️ $localization->error_button_chat_user",
url: 'https://t.me/' . $robot->user()->username
)
);
// Domain
/* $domain = "*$localization->error_account:* @" . $robot->user()->username; */
// Sending the message
$robot->sendMessage(
text: implode(
"\n\n",
[
$exception->getMessage(),
/* $domain */
]
),
chat_id: $receiver,
/* parse_mode: mode::MARKDOWN, */
reply_markup: $keyboard
);
}
} catch (exception $exception) {
// Writing into the errors output
error_log($exception->getMessage());
}
});
try {
$robot->run();
} catch (exception $exception) {
// Writing into the errors output
error_log($exception->getMessage());
}