Налаштування захисту від шахрайських замовлень 1С-Бітрікс
Шахрайські замовлення мають характерні патерни: один IP — десятки замовлень за годину, тимчасові email, невідповідність міста доставки та IP-геолокації, замовлення на велику суму без історії покупок. Набір перевірок у 1С-Бітрікс закриває 80–90% таких випадків без зовнішніх сервісів.
Багаторівнева перевірка
Захист вбудовується в подію OnBeforeOrderFinalAction. Три рівні дій: пропустити, позначити для ручної перевірки, заблокувати.
// /local/php_interface/init.php
AddEventHandler('sale', 'OnBeforeOrderFinalAction', ['\Local\Fraud\OrderGuard', 'check']);
namespace Local\Fraud;
class OrderGuard
{
public static function check(\Bitrix\Sale\Order $order): \Bitrix\Main\EventResult
{
if ($order->getId() > 0) {
return new \Bitrix\Main\EventResult(\Bitrix\Main\EventResult::SUCCESS);
}
$violations = self::runChecks($order);
if (in_array('block', array_column($violations, 'action'), true)) {
return new \Bitrix\Main\EventResult(
\Bitrix\Main\EventResult::ERROR,
new \Bitrix\Main\Error('Замовлення заблоковано системою безпеки. Зверніться до підтримки.')
);
}
if (!empty($violations)) {
$comment = implode('; ', array_column($violations, 'reason'));
$order->setField('COMMENTS', '[REVIEW] ' . $comment);
}
return new \Bitrix\Main\EventResult(\Bitrix\Main\EventResult::SUCCESS);
}
private static function runChecks(\Bitrix\Sale\Order $order): array
{
$violations = [];
$ip = $_SERVER['REMOTE_ADDR'] ?? '';
$props = $order->getPropertyCollection();
$email = $props->getItemByOrderPropertyCode('EMAIL')?->getValue() ?? '';
$phone = $props->getItemByOrderPropertyCode('PHONE')?->getValue() ?? '';
// 1. Перевищення ліміту замовлень з IP
$ipOrders = self::countOrdersByIp($ip, 1); // за 1 годину
if ($ipOrders >= 10) {
$violations[] = ['action' => 'block', 'reason' => "IP {$ip}: {$ipOrders} orders/hour"];
} elseif ($ipOrders >= 3) {
$violations[] = ['action' => 'review', 'reason' => "IP: {$ipOrders} orders/hour"];
}
// 2. Тимчасовий email
if (self::isDisposableEmail($email)) {
$violations[] = ['action' => 'review', 'reason' => 'Disposable email'];
}
// 3. Невідповідність міста доставки та IP
$deliveryCity = $props->getItemByOrderPropertyCode('CITY')?->getValue() ?? '';
if ($deliveryCity && self::isCityMismatch($ip, $deliveryCity)) {
$violations[] = ['action' => 'review', 'reason' => 'City/IP mismatch'];
}
// 4. Велике замовлення від нового користувача
$userId = (int)$order->getUserId();
if ($order->getPrice() > 50000 && $userId > 0 && self::getPreviousOrderCount($userId) === 0) {
$violations[] = ['action' => 'review', 'reason' => 'High amount + new user'];
}
return $violations;
}
private static function countOrdersByIp(string $ip, int $hours): int
{
$ip = \Bitrix\Main\Application::getConnection()->getSqlHelper()->forSql($ip);
$from = date('Y-m-d H:i:s', time() - $hours * 3600);
return (int)\Bitrix\Main\Application::getConnection()->query(
"SELECT COUNT(*) cnt FROM b_sale_order
WHERE CREATED_BY_IP = '{$ip}' AND DATE_INSERT >= '{$from}'"
)->fetch()['cnt'];
}
private static function isDisposableEmail(string $email): bool
{
$domain = strtolower(substr(strrchr($email, '@'), 1));
$domains = ['mailinator.com', 'guerrillamail.com', 'tempmail.com', 'throwam.com',
'yopmail.com', '10minutemail.com', 'trashmail.com', 'dispostable.com'];
return in_array($domain, $domains, true);
}
private static function isCityMismatch(string $ip, string $deliveryCity): bool
{
// Проста перевірка через geoip: якщо розбіжність > 1000 км
if (!function_exists('geoip_record_by_name')) return false;
$geo = @geoip_record_by_name($ip);
if (!$geo || empty($geo['city'])) return false;
// Нормалізуємо і порівнюємо
$geoCity = mb_strtolower(trim($geo['city']));
$deliveryNorm = mb_strtolower(trim($deliveryCity));
return $geoCity !== '' && !str_contains($deliveryNorm, $geoCity)
&& !str_contains($geoCity, $deliveryNorm);
}
private static function getPreviousOrderCount(int $userId): int
{
return (int)\Bitrix\Main\Application::getConnection()->query(
"SELECT COUNT(*) cnt FROM b_sale_order
WHERE USER_ID = {$userId} AND STATUS_ID NOT IN ('C')"
)->fetch()['cnt'];
}
}
Повідомлення менеджера про підозрілі замовлення
При статусі [REVIEW] агент кожні 15 хвилин перевіряє нові замовлення з цією міткою і надсилає повідомлення менеджеру:
$suspiciousOrders = \Bitrix\Sale\OrderTable::getList([
'filter' => [
'STATUS_ID' => 'N',
'%COMMENTS' => '[REVIEW]',
'>=DATE_INSERT' => new \Bitrix\Main\Type\DateTime(date('Y-m-d H:i:s', time() - 900)),
],
'select' => ['ID', 'PRICE', 'COMMENTS', 'DATE_INSERT'],
])->fetchAll();
Терміни реалізації
| Конфігурація | Термін |
|---|---|
| Базові перевірки (IP, email, ліміти) | 2–3 дні |
| + геолокація, повідомлення менеджеру | +1–2 дні |
| + адміністративний інтерфейс управління | +2 дні |







