ADD: Agrega modelo y migración para 'folio_tag' y 'tag_number' en 'vehicle_fakes', y actualiza el controlador para registrar vehículos robados

This commit is contained in:
Juan Felipe Zapata Moreno 2025-12-20 14:32:20 -06:00
parent 6c31941803
commit b27f360109
4 changed files with 106 additions and 94 deletions

View File

@ -3,9 +3,8 @@
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Models\Vehicle; use App\Models\Vehicle;
use App\Models\Detection; use App\Models\Detection;
use App\Models\VehicleFake;
use App\Services\VehicleService; use App\Services\VehicleService;
use App\Services\ConsultaEstatalService;
use App\Services\ReporteRoboService;
use Illuminate\Http\JsonResponse; use Illuminate\Http\JsonResponse;
use Notsoweb\ApiResponse\Enums\ApiResponse; use Notsoweb\ApiResponse\Enums\ApiResponse;
use Illuminate\Http\Request; use Illuminate\Http\Request;
@ -16,13 +15,11 @@
class VehicleController extends Controller class VehicleController extends Controller
{ {
public function __construct( public function __construct(
private VehicleService $vehicleService, private VehicleService $vehicleService
private ConsultaEstatalService $consultaEstatal,
private ReporteRoboService $reporteRobo
) {} ) {}
/** /**
* Consultar y registrar vehículo por PLACA o VIN * Consultar y registrar vehículo por PLACA o VIN (Alta de vehículo robado)
*/ */
public function consultarVehiculo(Request $request): JsonResponse public function consultarVehiculo(Request $request): JsonResponse
{ {
@ -42,91 +39,77 @@ public function consultarVehiculo(Request $request): JsonResponse
]); ]);
} }
try { $placa = $request->placa;
$placa = $request->placa; $vin = $request->vin;
$vin = $request->vin;
// Consultar padrón estatal (obtener datos completos del vehículo) // Buscar vehículo en vehicle_fakes
Log::info('Consultando padrón estatal', ['placa' => $placa, 'vin' => $vin]); $vehiculo = VehicleFake::when($vin, function($query) use ($vin) {
return $query->where('vin', $vin);
})
->when($placa, function($query) use ($placa) {
return $query->orWhere('placa', $placa);
})
->first();
// Si viene VIN, usar VIN, si no usar placa if (!$vehiculo) {
$criterio = $vin ?: $placa; return ApiResponse::NOT_FOUND->response([
$datosEstatal = $this->consultaEstatal->consultarPorEpc($criterio);
if (!$datosEstatal || (!$datosEstatal['vin'] && !$datosEstatal['placa'])) {
return ApiResponse::BAD_REQUEST->response([
'success' => false,
'message' => 'No se encontró información del vehículo en el padrón estatal'
]);
}
// Obtener VIN y Placa completos
$vinCompleto = $datosEstatal['vin'] ?: $vin;
$placaCompleta = $datosEstatal['placa'] ?: $placa;
Log::info('Datos del padrón estatal obtenidos', [
'vin' => $vinCompleto,
'placa' => $placaCompleta
]);
// Consultar REPUVE
$reporteRobo = $this->reporteRobo->consultarPorVin($vinCompleto, $placaCompleta);
$tieneReporte = $reporteRobo['tiene_reporte'] ?? false;
// Si tiene reporte de robo → Guardar en Redis
if ($tieneReporte) {
$epc = 'MANUAL_' . ($vinCompleto ?: $placaCompleta);
$this->guardarEnRedis($epc, $datosEstatal, $reporteRobo['datos']);
Log::warning('¡VEHÍCULO CON REPORTE DE ROBO REGISTRADO!', [
'vin' => $vinCompleto,
'placa' => $placaCompleta,
'autoridad' => $reporteRobo['datos']['autoridad'] ?? null
]);
return ApiResponse::OK->response([
'success' => true,
'tiene_reporte_robo' => true,
'message' => '¡ALERTA! El vehículo tiene reporte de robo',
'vehiculo' => [
'vin' => $vinCompleto,
'placa' => $placaCompleta,
],
'reporte_robo' => $reporteRobo['datos']
]);
}
// No tiene reporte de robo
Log::info('Vehículo consultado sin reporte de robo', [
'vin' => $vinCompleto,
'placa' => $placaCompleta
]);
return ApiResponse::OK->response([
'success' => true,
'tiene_reporte_robo' => false,
'message' => 'Vehículo sin reporte de robo',
'vehiculo' => [
'vin' => $vinCompleto,
'placa' => $placaCompleta,
]
]);
} catch (\Exception $e) {
Log::error('Error al consultar vehículo', [
'placa' => $request->placa,
'vin' => $request->vin,
'error' => $e->getMessage()
]);
return ApiResponse::INTERNAL_ERROR->response([
'success' => false, 'success' => false,
'message' => 'Error al consultar vehículo', 'message' => 'No se encontró el vehículo en la base de datos'
'error' => $e->getMessage()
]); ]);
} }
// Generar EPC
$epc = 'E280117000000' . $vehiculo->vin;
// Verificar si ya está en Redis
$key = "vehiculo:robado:{$placa}-{$vin}";
if (Redis::exists($key)) {
return ApiResponse::BAD_REQUEST->response([
'success' => false,
'message' => 'El vehículo ya está registrado como robado'
]);
}
// Guardar en Redis como robado
$datos = [
'epc' => $epc,
'folio_tag' => $vehiculo->folio_tag,
'tag_number' => $vehiculo->tag_number,
'vin' => $vehiculo->vin,
'placa' => $vehiculo->placa,
'marca' => $vehiculo->marca,
'modelo' => $vehiculo->modelo,
'color' => $vehiculo->color,
'fecha_robo' => now()->format('Y-m-d'),
'acta' => 'MANUAL-' . now()->format('Ymd-His'),
'denunciante' => 'SISTEMA',
'fecha_acta' => now()->format('Y-m-d'),
'primera_deteccion' => now()->toIso8601String(),
'ultima_deteccion' => now()->toIso8601String(),
];
Redis::set($key, json_encode($datos));
Log::warning('Vehículo dado de alta como robado (MANUAL)', [
'epc' => $epc,
'vin' => $vehiculo->vin,
'placa' => $vehiculo->placa
]);
return ApiResponse::OK->response([
'success' => true,
'message' => 'Vehículo registrado como robado exitosamente',
'vehiculo' => [
'epc' => $epc,
'folio_tag' => $vehiculo->folio_tag,
'tag_number' => $vehiculo->tag_number,
'vin' => $vehiculo->vin,
'placa' => $vehiculo->placa,
'marca' => $vehiculo->marca,
'modelo' => $vehiculo->modelo,
'color' => $vehiculo->color
]
]);
} }
/** /**
@ -203,12 +186,11 @@ public function recuperarVehiculo(Request $request): JsonResponse
// 2. Guardar en MySQL como recuperado // 2. Guardar en MySQL como recuperado
$vehiculoRecuperado = Vehicle::create([ $vehiculoRecuperado = Vehicle::create([
'epc' => $epc, 'epc' => $epc,
'vin' => $datosRedis['vin'], 'vin' => $datosRedis['vin'] ?? null,
'placa' => $datosRedis['placa'], 'placa' => $datosRedis['placa'] ?? null,
'fecha_robo' => $datosRedis['fecha_robo'], 'fecha_robo' => $datosRedis['fecha_robo'] ?? null,
'autoridad_robo' => $datosRedis['autoridad'], 'acta_robo' => $datosRedis['acta'] ?? null,
'acta_robo' => $datosRedis['acta'], 'denunciante' => $datosRedis['denunciante'] ?? null,
'denunciante' => $datosRedis['denunciante'],
'fecha_acta' => $datosRedis['fecha_acta'] ?? null, 'fecha_acta' => $datosRedis['fecha_acta'] ?? null,
'fecha_recuperacion' => now(), 'fecha_recuperacion' => now(),
'datos_robo_original' => $datosRedis 'datos_robo_original' => $datosRedis
@ -233,7 +215,6 @@ public function recuperarVehiculo(Request $request): JsonResponse
'placa' => $vehiculoRecuperado->placa, 'placa' => $vehiculoRecuperado->placa,
'fecha_robo' => $vehiculoRecuperado->fecha_robo, 'fecha_robo' => $vehiculoRecuperado->fecha_robo,
'fecha_recuperacion' => $vehiculoRecuperado->fecha_recuperacion, 'fecha_recuperacion' => $vehiculoRecuperado->fecha_recuperacion,
'autoridad_robo' => $vehiculoRecuperado->autoridad_robo,
'acta_robo' => $vehiculoRecuperado->acta_robo, 'acta_robo' => $vehiculoRecuperado->acta_robo,
] ]
]); ]);

View File

@ -16,6 +16,8 @@
class VehicleFake extends Model class VehicleFake extends Model
{ {
protected $fillable = [ protected $fillable = [
'folio_tag',
'tag_number',
'placa', 'placa',
'vin', 'vin',
'marca', 'marca',

View File

@ -0,0 +1,30 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::table('vehicle_fakes', function (Blueprint $table) {
$table->string('folio_tag')->unique()->after('id');
$table->string('tag_number')->unique()->after('folio_tag');
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::table('vehicle_fakes', function (Blueprint $table) {
$table->dropColumn('folio_tag');
$table->dropColumn('tag_number');
});
}
};

View File

@ -22,6 +22,5 @@ public function run(): void
$this->call(RoleSeeder::class); $this->call(RoleSeeder::class);
$this->call(UserSeeder::class); $this->call(UserSeeder::class);
$this->call(SettingSeeder::class); $this->call(SettingSeeder::class);
$this->call(VehicleTestSeeder::class);
} }
} }