🚀 مُصادَق: تحديث برمجي جديد 2026-05-03 13:39
This commit is contained in:
84
app/Services/AI/OpenAIProvider.php
Normal file
84
app/Services/AI/OpenAIProvider.php
Normal file
@@ -0,0 +1,84 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Services\AI;
|
||||
|
||||
use App\Services\AI\Contracts\AIProviderInterface;
|
||||
use Exception;
|
||||
|
||||
final class OpenAIProvider implements AIProviderInterface
|
||||
{
|
||||
private string $apiKey;
|
||||
private string $model;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->apiKey = $_ENV['OPENAI_API_KEY'] ?? '';
|
||||
$this->model = $_ENV['OPENAI_MODEL'] ?? 'gpt-4o-mini';
|
||||
}
|
||||
|
||||
public function isConfigured(): bool
|
||||
{
|
||||
return !empty($this->apiKey);
|
||||
}
|
||||
|
||||
public function extractInvoiceData(string $fileContent, string $mimeType, string $prompt): array
|
||||
{
|
||||
if (!$this->isConfigured()) {
|
||||
throw new Exception("OpenAI API Key is missing. Please configure it in .env");
|
||||
}
|
||||
|
||||
$base64Data = base64_encode($fileContent);
|
||||
|
||||
$payload = [
|
||||
'model' => $this->model,
|
||||
'messages' => [
|
||||
[
|
||||
'role' => 'user',
|
||||
'content' => [
|
||||
[
|
||||
'type' => 'text',
|
||||
'text' => $prompt
|
||||
],
|
||||
[
|
||||
'type' => 'image_url',
|
||||
'image_url' => [
|
||||
'url' => "data:{$mimeType};base64,{$base64Data}"
|
||||
]
|
||||
]
|
||||
]
|
||||
]
|
||||
],
|
||||
'response_format' => ['type' => 'json_object'],
|
||||
'temperature' => 0.1
|
||||
];
|
||||
|
||||
$ch = curl_init('https://api.openai.com/v1/chat/completions');
|
||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($ch, CURLOPT_POST, true);
|
||||
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($payload));
|
||||
curl_setopt($ch, CURLOPT_HTTPHEADER, [
|
||||
'Content-Type: application/json',
|
||||
"Authorization: Bearer {$this->apiKey}"
|
||||
]);
|
||||
|
||||
$response = curl_exec($ch);
|
||||
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
curl_close($ch);
|
||||
|
||||
if ($httpCode !== 200) {
|
||||
throw new Exception("OpenAI Extraction failed. HTTP Code: {$httpCode}. Response: {$response}");
|
||||
}
|
||||
|
||||
$result = json_decode($response, true);
|
||||
$text = $result['choices'][0]['message']['content'] ?? '{}';
|
||||
|
||||
$data = json_decode($text, true);
|
||||
if (!is_array($data)) {
|
||||
throw new Exception("Failed to parse OpenAI output as JSON: {$text}");
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
}
|
||||
77
app/Services/RiskAnalysisService.php
Normal file
77
app/Services/RiskAnalysisService.php
Normal file
@@ -0,0 +1,77 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Services;
|
||||
|
||||
use App\Core\Database;
|
||||
|
||||
final class RiskAnalysisService
|
||||
{
|
||||
public function calculateCompanyRiskScore(string $companyId): array
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$score = 100;
|
||||
$factors = [];
|
||||
|
||||
// 1. Rejection Rate
|
||||
$stmt = $db->prepare("SELECT status, COUNT(*) as count FROM invoices WHERE company_id = ? GROUP BY status");
|
||||
$stmt->execute([$companyId]);
|
||||
$stats = $stmt->fetchAll();
|
||||
|
||||
$total = 0;
|
||||
$rejected = 0;
|
||||
foreach ($stats as $stat) {
|
||||
$total += $stat['count'];
|
||||
if ($stat['status'] === 'rejected' || $stat['status'] === 'validation_failed') {
|
||||
$rejected += $stat['count'];
|
||||
}
|
||||
}
|
||||
|
||||
if ($total > 0) {
|
||||
$rejectionRate = $rejected / $total;
|
||||
if ($rejectionRate > 0.10) { // More than 10% rejections
|
||||
$penalty = min(30, (int)(($rejectionRate - 0.10) * 100));
|
||||
$score -= $penalty;
|
||||
$factors[] = "نسبة رفض عالية: " . round($rejectionRate * 100, 1) . "% (خصم {$penalty} نقطة)";
|
||||
}
|
||||
}
|
||||
|
||||
// 2. High Value Cash Invoices
|
||||
$stmt = $db->prepare("SELECT COUNT(*) as count FROM invoices WHERE company_id = ? AND invoice_type = 'cash' AND grand_total > 5000");
|
||||
$stmt->execute([$companyId]);
|
||||
$highValueCash = $stmt->fetch()['count'];
|
||||
|
||||
if ($highValueCash > 0) {
|
||||
$penalty = min(20, $highValueCash * 2);
|
||||
$score -= $penalty;
|
||||
$factors[] = "وجود فواتير نقدية بقيم عالية: {$highValueCash} فاتورة (خصم {$penalty} نقطة)";
|
||||
}
|
||||
|
||||
// 3. Late submissions (invoice_date is much older than created_at)
|
||||
$stmt = $db->prepare("SELECT COUNT(*) as count FROM invoices WHERE company_id = ? AND DATEDIFF(created_at, invoice_date) > 7");
|
||||
$stmt->execute([$companyId]);
|
||||
$lateInvoices = $stmt->fetch()['count'];
|
||||
|
||||
if ($lateInvoices > 0) {
|
||||
$penalty = min(15, $lateInvoices * 1);
|
||||
$score -= $penalty;
|
||||
$factors[] = "تأخير في رفع الفواتير: {$lateInvoices} فاتورة متأخرة بأكثر من 7 أيام (خصم {$penalty} نقطة)";
|
||||
}
|
||||
|
||||
// Determine Risk Level
|
||||
$riskLevel = 'low';
|
||||
if ($score < 50) {
|
||||
$riskLevel = 'high';
|
||||
} elseif ($score < 80) {
|
||||
$riskLevel = 'medium';
|
||||
}
|
||||
|
||||
return [
|
||||
'score' => max(0, $score),
|
||||
'level' => $riskLevel,
|
||||
'factors' => $factors,
|
||||
'calculated_at' => date('Y-m-d H:i:s')
|
||||
];
|
||||
}
|
||||
}
|
||||
83
app/Services/TotpService.php
Normal file
83
app/Services/TotpService.php
Normal file
@@ -0,0 +1,83 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Services;
|
||||
|
||||
final class TotpService
|
||||
{
|
||||
private const ALPHABET = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ234567';
|
||||
|
||||
public function generateSecret(): string
|
||||
{
|
||||
$secret = '';
|
||||
for ($i = 0; $i < 16; $i++) {
|
||||
$secret .= self::ALPHABET[random_int(0, 31)];
|
||||
}
|
||||
return $secret;
|
||||
}
|
||||
|
||||
public function verify(string $secret, string $code): bool
|
||||
{
|
||||
$currentTime = floor(time() / 30);
|
||||
|
||||
// Check current, previous and next window (allow 30s clock drift)
|
||||
for ($i = -1; $i <= 1; $i++) {
|
||||
if ($this->calculateCode($secret, (int)($currentTime + $i)) === $code) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private function calculateCode(string $secret, int $time): string
|
||||
{
|
||||
$key = $this->base32Decode($secret);
|
||||
$timeHex = str_pad(dechex($time), 16, '0', STR_PAD_LEFT);
|
||||
$timeBin = pack('H*', $timeHex);
|
||||
|
||||
$hash = hash_hmac('sha1', $timeBin, $key, true);
|
||||
$offset = ord($hash[19]) & 0xf;
|
||||
|
||||
$otp = (
|
||||
((ord($hash[$offset]) & 0x7f) << 24) |
|
||||
((ord($hash[$offset + 1]) & 0xff) << 16) |
|
||||
((ord($hash[$offset + 2]) & 0xff) << 8) |
|
||||
(ord($hash[$offset + 3]) & 0xff)
|
||||
) % 1000000;
|
||||
|
||||
return str_pad((string)$otp, 6, '0', STR_PAD_LEFT);
|
||||
}
|
||||
|
||||
private function base32Decode(string $base32): string
|
||||
{
|
||||
$base32 = strtoupper($base32);
|
||||
$buffer = 0;
|
||||
$bufferSize = 0;
|
||||
$decoded = '';
|
||||
|
||||
for ($i = 0; $i < strlen($base32); $i++) {
|
||||
$char = $base32[$i];
|
||||
$pos = strpos(self::ALPHABET, $char);
|
||||
if ($pos === false) continue;
|
||||
|
||||
$buffer = ($buffer << 5) | $pos;
|
||||
$bufferSize += 5;
|
||||
|
||||
if ($bufferSize >= 8) {
|
||||
$bufferSize -= 8;
|
||||
$decoded .= chr(($buffer >> $bufferSize) & 0xff);
|
||||
}
|
||||
}
|
||||
|
||||
return $decoded;
|
||||
}
|
||||
|
||||
public function getQrCodeUrl(string $userEmail, string $secret, string $issuer = 'Musadaq'): string
|
||||
{
|
||||
$label = urlencode($issuer . ':' . $userEmail);
|
||||
$otpauth = "otpauth://totp/{$label}?secret={$secret}&issuer=" . urlencode($issuer);
|
||||
return "https://api.qrserver.com/v1/create-qr-code/?size=200x200&data=" . urlencode($otpauth);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user