Files
Vasyka 1ff888131f Stage 16 — AI Layer: VIN decoder + diagnostic / parts / price helpers
VinDecoder (deterministic, no API):
- ISO 3779/3780 parsing: WMI manufacturer (~60 brands), year (cyclical with
  post-2010 disambiguation via position 7), region, plant, NA checksum
- Strip non-VIN chars, accept dashes/spaces, reject I/O/Q per spec

AiAssistantService:
- Refactored provider HTTP into postClaude/postOpenAI/postGemini so both
  chat history and one-shot calls share the same transport
- singleShot(system, userPrompt, provider?) for fire-and-forget calls
- 4 specialized helpers with tight prompts:
  - suggestDiagnosis(WO) — diagnostician based on complaint + VIN info
  - suggestParts(WO, task) — OEM parts list for an operation
  - suggestPrice(Part) — markup recommendation with justification
  - vinRecommendations(vin, mileage) — scheduled maintenance from decoded VIN
- monthlyUsage() — token spend MTD by provider

Filament:
- VehicleResource: "Decode VIN" + "AI: recomandări" actions
- WorkOrderResource Edit: "AI: sugerează diagnostic" header action
- PartResource: "AI: preț recomandat" action
- Shared views: filament.tenant.ai-reply, filament.tenant.vin-decode
- AiAssistant page shows monthly token usage banner

Tests (13 new):
- 8 VinDecoder unit tests with real VIN samples (Honda 2003, VW 1999, Audi
  2014, Dacia, unknown WMI, lowercase/dashes, forbidden chars)
- 5 AiHelpers feature tests with Http::fake covering all providers + no-key
  fallback + token usage aggregation

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-27 20:24:09 +00:00

174 lines
8.5 KiB
PHP
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<x-filament-panels::page>
<style>
.ai-wrap { display: grid; grid-template-columns: 240px 1fr; gap: 16px; min-height: 70vh; }
@media (max-width: 768px) { .ai-wrap { grid-template-columns: 1fr; } }
.ai-side {
background: #f9fafb; border: 1px solid #e5e7eb; border-radius: 10px;
padding: 12px; overflow-y: auto;
}
.dark .ai-side { background: #1f2937; border-color: #374151; }
.ai-new-btn {
width: 100%; padding: 8px 12px; border-radius: 6px;
background: #3b82f6; color: white; font-size: 13px; font-weight: 500;
border: none; cursor: pointer; margin-bottom: 12px;
}
.ai-new-btn:hover { background: #2563eb; }
.ai-chat-item {
padding: 8px 10px; border-radius: 6px; margin-bottom: 4px;
font-size: 12px; cursor: pointer; display: flex; justify-content: space-between; align-items: center; gap: 6px;
}
.ai-chat-item:hover { background: #fff; }
.dark .ai-chat-item:hover { background: #374151; }
.ai-chat-item.active { background: #fff; border: 1px solid #3b82f6; }
.dark .ai-chat-item.active { background: #111827; }
.ai-chat-title { flex: 1; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.ai-del { color: #9ca3af; cursor: pointer; padding: 2px 4px; border-radius: 4px; font-size: 14px; border: none; background: none; }
.ai-del:hover { color: #dc2626; background: #fef2f2; }
.ai-main {
background: #fff; border: 1px solid #e5e7eb; border-radius: 10px;
display: flex; flex-direction: column;
}
.dark .ai-main { background: #1f2937; border-color: #374151; }
.ai-messages {
flex: 1; overflow-y: auto; padding: 16px;
display: flex; flex-direction: column; gap: 12px;
min-height: 400px;
}
.ai-msg { max-width: 85%; padding: 10px 14px; border-radius: 12px; font-size: 13px; line-height: 1.5; }
.ai-msg-user {
align-self: flex-end;
background: #3b82f6; color: white;
border-bottom-right-radius: 4px;
}
.ai-msg-assistant {
align-self: flex-start;
background: #f3f4f6; color: #1f2937;
border-bottom-left-radius: 4px;
}
.dark .ai-msg-assistant { background: #374151; color: #f9fafb; }
.ai-msg pre { white-space: pre-wrap; word-wrap: break-word; margin: 0; font-family: inherit; }
.ai-msg-meta { font-size: 10px; color: #9ca3af; margin-top: 4px; }
.ai-empty { text-align: center; color: #9ca3af; padding: 60px 20px; }
.ai-form {
border-top: 1px solid #e5e7eb;
padding: 12px; display: flex; gap: 8px;
}
.dark .ai-form { border-color: #374151; }
.ai-input {
flex: 1; padding: 10px 12px; border: 1px solid #e5e7eb; border-radius: 8px;
font-size: 13px; font-family: inherit; resize: none; min-height: 40px;
}
.dark .ai-input { background: #111827; border-color: #374151; color: #f9fafb; }
.ai-send {
padding: 10px 20px; background: #3b82f6; color: white;
border: none; border-radius: 8px; cursor: pointer; font-size: 13px; font-weight: 500;
}
.ai-send:hover { background: #2563eb; }
.ai-send:disabled { opacity: 0.5; cursor: not-allowed; }
.ai-loading {
align-self: flex-start;
color: #9ca3af; font-size: 13px;
padding: 10px 14px;
display: flex; align-items: center; gap: 6px;
}
.ai-loading .dot { width: 6px; height: 6px; background: #9ca3af; border-radius: 50%; animation: ai-bounce 1.4s infinite; }
.ai-loading .dot:nth-child(2) { animation-delay: 0.2s; }
.ai-loading .dot:nth-child(3) { animation-delay: 0.4s; }
@keyframes ai-bounce { 0%, 80%, 100% { opacity: 0.3; } 40% { opacity: 1; } }
</style>
@php
$chat = $this->getChat();
$chats = $this->getChats();
$usage = $this->getUsage();
$totalTokens = collect($usage)->sum(fn ($u) => $u['tokens_in'] + $u['tokens_out']);
$totalCalls = collect($usage)->sum('calls');
@endphp
@if ($totalCalls > 0)
<div style="margin-bottom:12px;padding:10px 14px;background:#eff6ff;border:1px solid #bfdbfe;border-radius:8px;font-size:12px;color:#1e3a8a;">
<strong>Tokens consumate luna aceasta:</strong>
{{ number_format($totalTokens) }} tokens · {{ $totalCalls }} cereri
@foreach ($usage as $provider => $u)
· {{ strtoupper($provider) }}: {{ number_format($u['tokens_in'] + $u['tokens_out']) }}
@endforeach
</div>
@endif
<div class="ai-wrap" x-data x-init="$nextTick(() => { const m = $refs.msgs; if (m) m.scrollTop = m.scrollHeight; })">
{{-- Sidebar: chat history --}}
<div class="ai-side">
<button type="button" class="ai-new-btn" wire:click="newChat">+ Conversație nouă</button>
@forelse ($chats as $c)
<div class="ai-chat-item {{ $c->id === $chatId ? 'active' : '' }}">
<span class="ai-chat-title" wire:click="selectChat({{ $c->id }})" style="cursor:pointer;">
{{ $c->title }}
</span>
<button type="button" class="ai-del" wire:click="deleteChat({{ $c->id }})"
wire:confirm="Șterge conversația?">×</button>
</div>
@empty
<div style="font-size:11px;color:#9ca3af;padding:8px;">Nicio conversație</div>
@endforelse
</div>
{{-- Main chat area --}}
<div class="ai-main">
<div class="ai-messages" x-ref="msgs"
x-init="$watch('$wire.loading', () => $nextTick(() => $el.scrollTop = $el.scrollHeight))">
@if (! $chat || $chat->messages->isEmpty())
<div class="ai-empty">
<div style="font-size:32px;margin-bottom:8px;"></div>
<div style="font-size:14px;font-weight:600;margin-bottom:4px;">Asistent AI pentru autoservice</div>
<div style="font-size:12px;">Întreabă- orice diagnostic, sumar, sugestii, redactare mesaje către clienți...</div>
<div style="font-size:11px;margin-top:16px;color:#9ca3af;">
<b>Configurare:</b> mergi la Setări Asistent AI și adaugă cheia API (Claude / GPT / Gemini).
</div>
</div>
@else
@foreach ($chat->messages as $m)
@if ($m->role === 'user')
<div class="ai-msg ai-msg-user"><pre>{{ $m->content }}</pre></div>
@elseif ($m->role === 'assistant')
<div class="ai-msg ai-msg-assistant">
<pre>{{ $m->content }}</pre>
@if (! empty($m->meta['provider']))
<div class="ai-msg-meta">
{{ strtoupper($m->meta['provider']) }}
@if (! empty($m->meta['latency_ms'])) · {{ $m->meta['latency_ms'] }}ms @endif
@if (! empty($m->meta['tokens_out'])) · {{ $m->meta['tokens_out'] }} tok @endif
</div>
@endif
</div>
@endif
@endforeach
@endif
@if ($loading)
<div class="ai-loading">
<span class="dot"></span><span class="dot"></span><span class="dot"></span>
<span>asistent gândește...</span>
</div>
@endif
</div>
<form wire:submit="send" class="ai-form">
<textarea
class="ai-input"
wire:model="newMessage"
placeholder="Scrie mesajul tău..."
@keydown.enter.exact.prevent="$wire.send()"
@keydown.shift.enter="$el.value += '\n'"
rows="1"
></textarea>
<button type="submit" class="ai-send" wire:loading.attr="disabled">Trimite</button>
</form>
</div>
</div>
</x-filament-panels::page>