feat: encrypt OTP and phone in verification table

This commit is contained in:
Hamza-Ayed
2026-05-01 01:12:08 +03:00
parent 6753d015a1
commit 031b1c6d64
2 changed files with 45 additions and 35 deletions

View File

@@ -4,53 +4,53 @@
* الخطوة الثانية من تسجيل الدخول: التحقق من الـ OTP وإصدار التوكن النهائي
*/
require_once __DIR__ . '/../../core/bootstrap.php';
require_once __DIR__ . '/../../functions.php';
$phone = filterRequest('phone');
$otp = filterRequest('otp');
$fingerprint = filterRequest('fingerprint'); // مطلوب لربط التوكن بالجهاز
$audience = filterRequest('aud') ?? 'admin';
if (empty($phone) || empty($otp) || empty($fingerprint)) {
jsonError("Phone, OTP and fingerprint are required.");
if (empty($otp) || empty($fingerprint)) {
jsonError("OTP and fingerprint are required.");
exit;
}
try {
$con = Database::get('main');
// 1. التحقق من الـ OTP
// 1. جلب بيانات المسؤول عبر البصمة (مصدر موثوق وغير مشفر)
$fpHash = hash('sha256', $fingerprint);
$stmt = $con->prepare("SELECT * FROM adminUser WHERE fingerprint_hash = :fp LIMIT 1");
$stmt->execute([':fp' => $fpHash]);
$admin = $stmt->fetch(PDO::FETCH_ASSOC);
if (!$admin) {
jsonError("المسؤول غير موجود أو البصمة غير مطابقة.");
exit;
}
// 2. رقم الهاتف المشفر (للاستخدام في جدول OTP)
$encryptedPhone = $admin['phone'] ?? '';
// فك تشفيره لو احتجنا إرساله أو عرضه، لكن هنا نحن نحتاج المشفر للبحث
// $phone = $encryptionHelper->decryptData($encryptedPhone);
// تشفير الرمز (OTP) القادم من التطبيق للمقارنة
$encryptedOtp = $encryptionHelper->encryptData((string)$otp);
// 3. التحقق من الـ OTP (باستخدام القيم المشفرة)
$stmt = $con->prepare("SELECT * FROM token_verification_admin
WHERE phone_number = ? AND token = ?
AND expiration_time >= NOW()");
$stmt->execute([$phone, $otp]);
$stmt->execute([$encryptedPhone, $encryptedOtp]);
if ($stmt->rowCount() === 0) {
jsonError("رمز التحقق غير صالح أو منتهي الصلاحية.");
exit;
}
// حذف الرمز بعد استخدامه لمرة واحدة
$con->prepare("DELETE FROM token_verification_admin WHERE phone_number = ?")->execute([$phone]);
// 2. جلب بيانات المسؤول
$stmt = $con->prepare("SELECT * FROM adminUser WHERE phone = ? LIMIT 1");
$stmt->execute([$phone]);
$admin = $stmt->fetch(PDO::FETCH_ASSOC);
if (!$admin) {
jsonError("المسؤول غير موجود.");
exit;
}
// 3. التحقق من البصمة (اختياري لكن مفضل لزيادة الأمان)
$fpHash = hash('sha256', $fingerprint);
if ($admin['fingerprint_hash'] !== $fpHash) {
// إذا كانت البصمة مختلفة، ربما يحاول تسجيل الدخول من جهاز آخر
// يمكننا إما المنع أو السماح وتحديث البصمة (حسب السياسة)
// هنا سنقوم بالمنع لزيادة الأمان
jsonError("عذراً، لا يمكن تسجيل الدخول من هذا الجهاز.");
exit;
}
// حذف الرمز بعد استخدامه لمرة واحدة (باستخدام الرقم المشفر)
$con->prepare("DELETE FROM token_verification_admin WHERE phone_number = ?")->execute([$encryptedPhone]);
// 4. إصدار التوكن النهائي
$jwtService = new JwtService($redis);
@@ -66,7 +66,7 @@ $stmt->execute([$phone, $otp]);
$jwt = $jwtService->generateAccessToken($admin['id'], $role, $audience, $fingerprint);
// فك تشفير الاسم للعرض
// فك تشفير البيانات للعرض
$admin['name'] = $encryptionHelper->decryptData($admin['name']) ?: $admin['name'];
unset($admin['password']);