добавляю все изменения проекта на текущий момент

This commit is contained in:
vasya
2026-02-27 18:49:27 +03:00
parent e927910fda
commit 9c35f4e35e
303 changed files with 79434 additions and 2558 deletions
@@ -0,0 +1,350 @@
<?php
namespace Modules\Taxi\App\Http\Controllers;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
use Modules\Taxi\App\Models as TaxiModels;
use App\Models\OldMagicModels;
#Гаврилов
//Rule и Validatir НЕ НУЖЕН? ЕСЛИ МЫ ВАЛИДИРУЕМ В ОТДЕЛЬОМ FORM_REQUEST
use Illuminate\Validation\Rule;
use Illuminate\Support\Facades\Validator;
use Modules\Taxi\App\Http\Requests\TaxiOrderRequest;
#Гаврилов
//КЛАСС НИЖЕ НЕ НУЖЕН, ТАК КАК ИСПОЛЬЗУЕМ СЕРВИС?
use Illuminate\Support\Facades\Mail;
use Modules\Taxi\App\Services\TaxiOrderService;
use Modules\Taxi\App\Services\TaxiMailerService;
use Modules\Taxi\App\DTO\TaxiOrderDTO;
use Modules\Taxi\App\Jobs\SendMailNewOrder;
//use Illuminate\Support\Facades\Config;
use App\Mail\Mailer;
use DateTime;
use Error;
use Exception;
use Illuminate\Support\Facades\DB;
use Modules\Taxi\App\Mail\NewOrderMail;
use Illuminate\Support\Facades\Redis;
use App\Facades\RedisNotifications;
class TaxiController extends Controller
{
/**
* Конструктор
*
* @param TaxiMailerService $sendMailService сервис отправки писем
* @param TaxiOrderService $orderService сервис работы с заказами
*/
public function __construct(protected TaxiMailerService $sendMailService, protected TaxiOrderService $orderService) {
}
/**
* Получаем информацию по существующему заказу
*
* @param integer $orderId идентификатор заказа
*/
public function getOrderById(int $orderId)
{
$taxiOrder = TaxiModels\TaxiMain::where('id', $orderId)->first();
return $taxiOrder;
}
/**
* Получение адресов офисов для подстановки в запросы такси
*
* @return array
*/
public function getOfficeAddress()
{
$taxiAddress = TaxiModels\OfficeAddress::all();
return $taxiAddress;
}
/**
* Получение всех активных заказов
*
* @return array
*/
public function getActiveOrders()
{
$todaydate = new DateTime();
$todaydate->setTime(00, 00, 00);
$todaydate = $todaydate->format('Y-m-d');
$orders = TaxiModels\TaxiMain::where('taxi_date', '>=', $todaydate)
->get();
return $orders;
}
/**
* Метод получения заказов по условиям фильтра
*
* @param Request $rqst запрос с параметрами поиска
* @return string
*/
public function getFilterOrders(Request $rqst)
{
$rqstCond = [];
foreach ($rqst->all() as $fieldName => $fieldVal) {
if (!$fieldVal) continue;
$rqstCond[] = [$fieldName, '=', $fieldVal];
}
$filterRqst = TaxiModels\TaxiMain::where($rqstCond)->get();
return json_encode($filterRqst->toArray());
}
//ГАВРИЛОВ используй транзакцию, а то создается запрос, не отправляется письмо и сообщение об ошибке
/**
* Show the form for creating a new resource.
*/
public function createTaxiOrder(TaxiOrderRequest $rqst)
{
$validateData = TaxiOrderDTO::setTaxiOrderDTO($rqst, $rqst->method());
//Пытаемся найти уже существующий запрос на этот же логин и эту же дату. Если запрос найден, возвращаем ошибку.
$searchExistsOrder = TaxiModels\TaxiMain::where([
['emp_login', '=', $validateData->emp_login],
['taxi_date', '=', $validateData->taxi_date]
])->take(1)->get();
if (count($searchExistsOrder)) {
return response()->json(['message' => "Ошибка! Заявка для сотрудника $validateData->emp_login на дату $validateData->taxi_date уже существует"], 400);
}
// $createRqstMsg;
try {
$newOrderId = (new TaxiModels\TaxiMain())->createOrder($validateData);
//$newOrderId = $this->orderService->createOrder($validateData);
$createRqstMsg = "Заявка успешно создана. Номер $newOrderId";
#Гаврилов
//ПРОВЕРЬ, ЧТОБЫ ПИСЬМО НЕ ОТПРАВЛЯЛОСЬ, ЕСЛИ ВОЗНИКАЕТ ОШИБКА НА ЭТАПЕ СОЗДАНИЯ ЗАЯВКИ
//Один метод отправки письма для всех методов работы с заказом: создание, редактирование, отмена
$this->sendMailService->sendMail(new NewOrderMail($validateData, $newOrderId));
// SendMailNewOrder::dispatch(json_decode(json_encode($validateData), true), $newOrderId)->onQueue('emails')->delay(10);
// Mail::send(new Mailer(
// [$validateData->emp_login],
// 'Создание заказа на такси',
// "Для вас создана заявка на заказ такси на дату <b>{$validateData->taxi_date}</b> и время <b>{$validateData->taxi_time}</b>. Номер заказа <b>{$newOrderId}</b>",
// #Гаврилов
// //ЗДЕСЬ НЕПОНЯТНО ОТКУДА ВЗЯТЬ ЗНАЧЕНИЕ. ОБРАТИТЬСЯ К ЛОКАЛЬНО МОДУЛЬНОМУ ENV?
// config('taxi.name_ru'),
// ));
return response()->json( ['message' => "Заявка успешно создана. Номер заявки $newOrderId"] );
} catch (\Throwable $t) {
#Гаврилов
//ОТПРАВКА СООБЩЕНИЯ С ОШИБКОЙ. ЗАТЕСТИ и отправляй response
//ОШИБКИ $T->GETmESSAGE НУЖНО ЛОГИРОВАТЬ, НО ПОЛЬЗОВАТЕЛЯМ ВЫВОДИТЬ ПРОСТО СООБЩЕНИЕ оШИБКА, ИНАЧЕ ОНИ ВИДЯТ ТЕХН ЧЕСКИЙ ТЕКСТ
//СНАЧАЛА ПЕРЕХВАТЫЙ EXCEPTION, ПОТОМ УЖЕ THOWBLE. EXCEPTION ВЫВОДИ ПОЛЬЗОВАТЕЛЮ, THORWBLE ЛОГИРУЙ ОТДЕЛЬНО
echo '<pre>'; var_dump($t->getMessage()); echo'</pre>';
echo '<pre>'; var_dump($t->getTrace()); echo'</pre>';
$createRqstMsg = "Ошибка!" . $t->getMessage() . ". Заявка не создана";
return response()->json(['message' => $createRqstMsg], 400);
}
//ГАВРИЛОВ
//ДОБАВЬ ФОРМИРОВАНИЕ ОТВЕТА ЧЕРЕЗ ГЕНЕРАЦИЮ РЕСУРСА, а не через response()
}
#Гаврилов
//ИСПОЛЬЗУЙ TaxiOrderRequest ПРИ ОБРАБОТКЕ ЗАЯВОК НА ТАКСИ
/**
* Метод отмены запроса
*
* @param integer $rqstId идентификатор запроса
* @return string
*/
public function cancelRqst(int $rqstId): string
{
#Гаврилов
//ПОПРОБУЙ ОТПРАВИТЬ ЗАПРОС К API ЕНДПОИНТУ (ОТМЕНА ЗАПРОСА НА ТАКСИ) ЧЕРЕЗ POSTMAN С ПЕРЕДАЧЕЙ ЗАГОЛОВКА АУТЕНТИФИКАЦИИ С SANCTUM ТОКЕНОМ. ДОЛЖЕН ПРОПУСТИТЬ, А БЕЗ ЗАГОЛОВКА НЕ ДОЛЖЕН
$rqstData = $this->getOrderById($rqstId);
try {
#Гаврилов
//ПЕРЕНЕСИ ЭТУ ПРОВЕРКУ В РОУТИНГ? FINDORFAIL С РЕДИРЕКТОМ И ОШИБКОЙ
if (!$rqstData) {
throw new \Exception("Запрос $rqstId не существует");
}
if (!$this->checkRqstDate($rqstData->taxi_date)) {
throw new \Exception("Дата запроса $rqstId уже прошла");
}
} catch (\Throwable $t) {
return response()->json([
'errorMsg' => "Ошибка: $t->getMessage(). Заявка не отменена",
], 400);
}
$rqstData['cancel_rqst'] = 1;
$rqstData->save();
$this->sendMailService->sendCancelOrderMail($validateData, $newOrderId);
// Mail::send(new Mailer(
// [$rqstData->emp_login],
// 'Отмена заказа на такси',
// "Заказ номер $rqstId на дату $rqstData->taxi_date и время $rqstData->taxi_time отменен",
// $this->appName));
return redirect('/taxi/home')->with([
'notification' => "Запрос $rqstId отменен",
'notification_err' => false
]);
}
/**
* Проверяем находится ли дата запроса на такси в диапазоне сегодняшнего или завтрашнего дня.
*
* @param string $rqstDate дата запроса на такси
* @return boolean
*/
private function checkRqstDate($rqstDate): bool
{
$todaydate = new DateTime();
$todaydate->setTime(00, 00, 00);
return new DateTime($rqstDate) >= $todaydate;
}
/**
* Получаем временные промежутки
*
* @return array
*/
public function getTimePeriods()
{
$empAllData = TaxiModels\TaxiTimePeriod::select('time_period', 'is_morning_time')
->where('archive', 0)
->get();
return $empAllData;
}
/**
* Метод получает информацию по всем активным пользователям
*
* @return array
*/
public function getActiveUsersInfo()
{
$empAllData = OldMagicModels\CcEmp::whereIn('emp_area', ['Курск', 'Пенза'])
->whereNotIn('emp_state', ['Уволен', 'Декрет', 'Перевод на позицию', 'Перевод в ГРАН'])
->get();
foreach ($empAllData as &$empData) {
$empData['full_name'] = $empData['emp_surname'] . " " . $empData['emp_name'] . " " . $empData['emp_surname'];
$empData['extension_number'] = $empData['emp_phone'];
$empData['emp_phone'] = $empData['emp_mob_phone'];
}
return $empAllData;
}
/**
* Метод получает информацию по конкретному пользователю
*
* @return array
*/
public function getUserInfoByLogin(string $userLogin)
{
$empLoginData = OldMagicModels\CcEmp::where('emp_login', $userLogin)
->get();
// dd($empLoginData);
$empLoginData[0]['full_name'] = $empLoginData[0]['emp_surname'] . " " . $empLoginData[0]['emp_name'] . " " . $empLoginData[0]['emp_surname'];
$empLoginData[0]['extension_number'] = $empLoginData[0]['emp_phone'];
$empLoginData[0]['emp_phone'] = $empLoginData[0]['emp_mob_phone'];
return $empLoginData;
}
#Гаврилов
//НЕ ЗАБУДЬ УДАЛИТЬ
public function testRedisMethod()
{
RedisNotifications::module('taxi')
->notifications(['текст1', 'tеуые2'])
->types(['error', 'success'])
->put();
}
/**
* Обновление заявки на такси
*
* @param TaxiOrderRequest $rqst информация по измененному запросу
* @return response
*/
public function updateTaxiOrder(TaxiOrderRequest $rqst)
{
try {
DB::beginTransaction();
#Гаврилов
//ЕЩЕ РАЗ, ЧТО ЗА LOCKFORUPDATE
$orderData = TaxiModels\TaxiMain::lockForUpdate()->find($rqst->id);
$oldOrderData = TaxiModels\TaxiMain::find($rqst->id);
if (!$orderData) {
throw new Exception("Заказ номер " . $rqst->id . " не найден");
//return response()->json(['message' => 'Заказ номер ' . $rqst->id . ' не найден'], 400);
}
$editOrderData = TaxiOrderDTO::setTaxiOrderDTO($rqst, $rqst->method());
#Гаврилов
//НА ДОМАШНЕЙ СТРАНИЦЕ ПОКАЗЫВАТЬ ТОЛЬКО ТЕ ЗАПРОСЫ, ДАТА КОТОРЫХ СЕГОДНЯ ИЛИ ЗАВТРА
$dateInterval = (new DateTime())->diff(new DateTime($orderData->taxi_date));
if ($dateInterval->format('%R') == '-' && $dateInterval->format('%a') >= 1) {
throw new Exception("Нельзя менять заявки, дата которых уже прошла");
// return response()->json(['message' => 'Нельзя менять заявки, дата которых уже прошла'], 400);
}
foreach($editOrderData as $fieldName => $fieldVal) {
if (isset($orderData->$fieldName)) {
$orderData[$fieldName] = $fieldVal;
}
}
$orderDataDiff = $orderData->getDirty();
if (empty($orderDataDiff)) {
throw new Exception("Вы не внесли никаких изменений");
//return response()->json(['message' => 'Вы не внесли никаких изменений'], 400);
}
$orderData->save();
$this->sendMailService->sendEditOrderMail($oldOrderData->toArray(), $orderDataDiff, $orderData['id']);
DB::commit();
return response()->json(['message' => 'Заявка успешно изменена'], 200);
} catch (\Exception $e) {
DB::rollBack();
return response()->json(['message' => "Ошибка." . $e->getMessage() . ". Не удалось обновить заявку"], 400);
}
//ГАВРИЛОВ. ДОБАВЬ БЛОКИРОВКУ В МОМЕНТ ПОЛУЧЕНИЯ ДАННЫХ ПО РЕДАКТИРУЕМОМУ ЗАПРОСУ И ДО МОМЕНТА ЕГО ОБНОВЛЕНИЯ. ЛИБО ЧЕРЕЗ УСТАНОВКУ УРОВНЯ ИЗОЛЯЦИИ
//ГАВРИЛОВ. ДОБАВЬ В DTO НЕ ОБНУЛЕНИЕ НЕНУЖНЫХД ДАННЫХ, А ПРИНУДИТЕЬНАЯ ИХ УСТАНОВКА. НАПРИМЕР ПРИ ОБНОВЛЕНИИ ЗАПРОСА НА ТАКСИ, УСТАНАВЛИВАТЬ EMP_LOGIN = ПЕРЕДАННОМУ ЗНАЧЕНИЮ, ВЗЯТОМУ ИЗ СУЩЕСТВУЮЩЕГО ЗАПРОСА.
//Обновляем текущую модель, добавляя туда измененные значения, затем проверяем есть ли измененные значения через getDirty()
//ГАВРИЛОВ. убрать поле Запрос актуален? и при отмене заявки убирать ее из таблицы Активные заявки
//гаврилов. СДЕЛАТЬ ВОЗМОЖНОСТЬ ВЫГРУЗИТЬ В ОТЧЕТНОСТЬ ВСЕ ИМЕЮЩИЕСЯ ЗАЯВКИ В CSV ФАЙЛ.
}
}
@@ -0,0 +1,52 @@
<?php
namespace Modules\Taxi\App\Http\Middleware;
use Closure;
use Illuminate\Http\Request;
/**
* Блокировщик работы с заявками на такси в случае слишком ранееЙ, слишком поздней работы с ресурсом
*
* Решил блочить рабоут приложения в части работы с заявками в очень ранее или очень позднее время. Это связано с нежеланием валидировать при создании или редактировании заявки передаваемый временной промежуток. Например, если текущее время 22:15, а заявка созается на 22:00.
*/
class CheckTimeRqstAvailability
{
public function handle(Request $request, Closure $next)
{
#Гаврилов
//ПОМЕНЯЙ НА 21 НА РЕЛИЗЕ
$maxRqstTime = 23;
$minRqstTime = 7;
$timezone = new \DateTimeZone('Europe/Moscow');
$timeNow = (new \DateTime())->setTimezone($timezone);
$timeMax = (new \DateTime())->setTimezone($timezone);
$timeMin = (new \DateTime())->setTimezone($timezone);
$timeMax->setTime($maxRqstTime, 00, 00);
$timeMin->setTime($minRqstTime, 00, 00);
#Гаврилов
//ПОМЕНЯЙ
if (!($timeNow > $timeMin && $timeNow < $timeMax)) {
//if (!($timeNow > $timeMin && $timeNow > $timeMax)) {
$errMsg = "Работа с заявками на такси заблокирована с $maxRqstTime:00 по 0$minRqstTime:00";
#Гаврилов
//ПОДУМАТЬ КАКОЙ КОД ЛУЧШЕ ВЕРНУТЬ
//Метод expectsJson позволяет определить вызывается ли api роут через ajax/fetch, либо web route.
if ($request->expectsJson()) {
return response()->json([
'errorMsg' => $errMsg
], 400);
} else {
#Гаврилов
//ПРОВЕРЬ КОГДА РЕАЛИЗУЕШЬ ОПОВЕЩЕНИЯ
return redirect('/taxi/home')->with([
'notification' => $errMsg,
'notification_err' => true
]);
}
} else {
return $next($request);
}
}
}
@@ -0,0 +1,109 @@
<?php
namespace Modules\Taxi\App\Http\Requests;
use Illuminate\Foundation\Http\FormRequest;
use App\Models\OldMagicModels;
use Illuminate\Validation\Rule;
use DateTime;
class TaxiOrderRequest extends FormRequest
{
/**
* Действия перед валидацией.
*
* @return void
*/
protected function prepareForValidation()
{
//echo '<pre>'; var_dump($this->all()); echo'</pre>';
}
/**
* Get the validation rules that apply to the request.
*/
public function rules(): array
{
//Получаем список активных пользователей
$empActiveLogins = OldMagicModels\CcEmp::select('emp_login')
->whereNotIn('emp_state', ['Уволен', 'Декрет'])
->get()
->toArray();
//Правила валидации
$validationRules = [
'id' => [
'int'
],
'taxi_date' => [
'date_format:Y-m-d',
//Заказ можно создать только на сегодняшний или завтрашний день
Rule::in([
(new DateTime())->format('Y-m-d'),
(new DateTime())->modify('+1 day')->format('Y-m-d')
]),
],
'taxi_time' => [
'required',
'string',
'regex:/^[0-9]{2}:[0-9]{2}$/', //время в формате 00:00
],
'cancel_rqst' => 'boolean',
'emp_phone' => [
'required',
'int',
'regex:/^(7|8)?[0-9]{10}$/'
],
'emp_login' => [
$validationRules['emp_login'] = [
'required',
'string',
'regex:/^[a-z\-]+[0-9]*$/',
//Логины ищем среди активных логинов на текущий момент
Rule::in(
array_map(fn($el) => $el['emp_login'], $empActiveLogins)
)
],
],
'taxi_address_from' => [
'required',
'string',
'max: 500'
],
'taxi_address_to' => [
'required',
'string',
'max: 500'
]
];
//Во время создания запроса поле ID не валидируется
if ($this->isMethod('POST')){
$validationRules['id'] = [
'prohibited'
];
}
return $validationRules;
}
/**
* Determine if the user is authorized to make this request.
*/
public function authorize(): bool
{
return true;
}
/**
* Переопределение сообщений об ошибках валидации
*
* @return array
*/
public function messages(): array
{
return [
'taxi_date.in' => 'Заявку на такси можно завести только на сегодняшний или завтрашний день'
];
}
}