feat: [rd] AI 견적 엔진 Phase 1 구현

- 모델 3개: AiQuotationModule, AiQuotation, AiQuotationItem
- AiQuotationService: Gemini/Claude 2단계 AI 파이프라인
- RdController: R&D 대시보드 + AI 견적 Blade 화면
- AiQuotationController: AI 견적 API (생성/목록/상세/재분석)
- Blade 뷰: 대시보드, 목록, 생성, 상세, HTMX 테이블
- 라우트: /rd/* (web), /admin/rd/* (api)
This commit is contained in:
김보곤
2026-03-02 17:43:47 +09:00
parent 1299543f4d
commit a3afa1a405
13 changed files with 1707 additions and 0 deletions

View File

@@ -0,0 +1,109 @@
<?php
namespace App\Http\Controllers\Api\Admin\Rd;
use App\Http\Controllers\Controller;
use App\Http\Requests\Rd\StoreAiQuotationRequest;
use App\Services\Rd\AiQuotationService;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Illuminate\View\View;
class AiQuotationController extends Controller
{
public function __construct(
private readonly AiQuotationService $quotationService
) {}
/**
* 목록 (HTMX partial 또는 JSON)
*/
public function index(Request $request): View|JsonResponse
{
$params = $request->only(['status', 'search', 'per_page']);
$quotations = $this->quotationService->getList($params);
if ($request->header('HX-Request')) {
return view('rd.ai-quotation.partials.table', compact('quotations'));
}
return response()->json([
'success' => true,
'data' => $quotations,
]);
}
/**
* 견적 생성 + AI 분석 실행
*/
public function store(StoreAiQuotationRequest $request): JsonResponse
{
$result = $this->quotationService->createAndAnalyze($request->validated());
if ($result['ok']) {
return response()->json([
'success' => true,
'message' => 'AI 분석이 완료되었습니다.',
'data' => $result['quotation'],
]);
}
return response()->json([
'success' => false,
'message' => 'AI 분석에 실패했습니다.',
'error' => $result['error'],
'data' => $result['quotation'] ?? null,
], 422);
}
/**
* 상세 조회
*/
public function show(int $id): JsonResponse
{
$quotation = $this->quotationService->getById($id);
if (! $quotation) {
return response()->json([
'success' => false,
'message' => 'AI 견적을 찾을 수 없습니다.',
], 404);
}
return response()->json([
'success' => true,
'data' => $quotation,
]);
}
/**
* AI 재분석
*/
public function analyze(int $id): JsonResponse
{
$quotation = $this->quotationService->getById($id);
if (! $quotation) {
return response()->json([
'success' => false,
'message' => 'AI 견적을 찾을 수 없습니다.',
], 404);
}
$result = $this->quotationService->runAnalysis($quotation);
if ($result['ok']) {
return response()->json([
'success' => true,
'message' => 'AI 재분석이 완료되었습니다.',
'data' => $result['quotation'],
]);
}
return response()->json([
'success' => false,
'message' => 'AI 재분석에 실패했습니다.',
'error' => $result['error'],
], 422);
}
}

View File

@@ -0,0 +1,74 @@
<?php
namespace App\Http\Controllers;
use App\Models\Rd\AiQuotation;
use App\Services\Rd\AiQuotationService;
use Illuminate\Http\Request;
use Illuminate\View\View;
class RdController extends Controller
{
public function __construct(
private readonly AiQuotationService $quotationService
) {}
/**
* R&D 대시보드
*/
public function index(Request $request): View|\Illuminate\Http\Response
{
if ($request->header('HX-Request')) {
return response('', 200)->header('HX-Redirect', route('rd.index'));
}
$dashboard = $this->quotationService->getDashboardStats();
$statuses = AiQuotation::getStatuses();
return view('rd.index', compact('dashboard', 'statuses'));
}
/**
* AI 견적 목록
*/
public function quotations(Request $request): View|\Illuminate\Http\Response
{
if ($request->header('HX-Request')) {
return response('', 200)->header('HX-Redirect', route('rd.ai-quotation.index'));
}
$statuses = AiQuotation::getStatuses();
return view('rd.ai-quotation.index', compact('statuses'));
}
/**
* AI 견적 생성 폼
*/
public function createQuotation(Request $request): View|\Illuminate\Http\Response
{
if ($request->header('HX-Request')) {
return response('', 200)->header('HX-Redirect', route('rd.ai-quotation.create'));
}
return view('rd.ai-quotation.create');
}
/**
* AI 견적 상세
*/
public function showQuotation(Request $request, int $id): View|\Illuminate\Http\Response
{
if ($request->header('HX-Request')) {
return response('', 200)->header('HX-Redirect', route('rd.ai-quotation.show', $id));
}
$quotation = $this->quotationService->getById($id);
if (! $quotation) {
abort(404, 'AI 견적을 찾을 수 없습니다.');
}
return view('rd.ai-quotation.show', compact('quotation'));
}
}