Sobes.tech
Назад к вопросам
Senior
7

/** * == ФТ == * Сервис обработки заказа получает заказ для обработки через Rest API. * Обработка в следующем: * - Проверить наличие товара на складе (вызывается внешний сервис по REST/HTTP). Если товара нет, то обработка заказа прекращается. * - Сохранить заказ в БД заказов * - Запустить дальнейшую обработку заказа (Processing, вызывается внешний сервис по REST/HTTP) * - Инициировать уведомления о новом заказе (Notification, вызывается внешний сервис по REST/HTTP) * * == НОТ == * - обработка должна быть такой, что если сохранить в БД заказ не удалось, то запускать заказ в обработку (processing) и инициировать уведомление (notification) не надо * - использовать для корреляции логов идентификатор заказа * - поддержать метрики 1) количество принятых заказов 2) количество успешно обработанных заказов * - сервис должен обрабатывать до 100 заказов в секунду */

Компании, где спрашивали
СБЕРСБЕР

Ответ от нейросети

sobes.tech AI

Для реализации сервиса обработки заказов с указанными требованиями можно использовать следующий подход:

  • Проверка наличия товара: вызвать внешний REST-сервис, если товара нет — прекратить обработку.

  • Сохранение заказа в БД: если сохранение не удалось, не запускать дальнейшую обработку и уведомления.

  • Запуск дальнейшей обработки и уведомлений: вызвать соответствующие REST-сервисы.

  • Корреляция логов: использовать идентификатор заказа как часть контекста логирования (например, через MDC в SLF4J).

  • Метрики: использовать счетчики для количества принятых и успешно обработанных заказов (например, Micrometer).

  • Производительность: обеспечить асинхронную обработку с ограничением до 100 заказов в секунду, например, с помощью пула потоков или реактивного подхода.

Пример упрощённого кода на Java с использованием Spring Boot и RestTemplate:

@Service
public class OrderProcessingService {
    private final RestTemplate restTemplate;
    private final OrderRepository orderRepository;
    private final MeterRegistry meterRegistry;

    private final Counter acceptedOrders;
    private final Counter successfulOrders;

    public OrderProcessingService(RestTemplate restTemplate, OrderRepository orderRepository, MeterRegistry meterRegistry) {
        this.restTemplate = restTemplate;
        this.orderRepository = orderRepository;
        this.meterRegistry = meterRegistry;
        this.acceptedOrders = meterRegistry.counter("orders.accepted");
        this.successfulOrders = meterRegistry.counter("orders.successful");
    }

    public void processOrder(Order order) {
        MDC.put("orderId", order.getId());
        acceptedOrders.increment();

        // Проверка наличия товара
        Boolean inStock = restTemplate.getForObject("http://stock-service/api/stock/{productId}", Boolean.class, order.getProductId());
        if (inStock == null || !inStock) {
            // Товара нет, прекращаем обработку
            MDC.clear();
            return;
        }

        try {
            orderRepository.save(order);
        } catch (Exception e) {
            // Не удалось сохранить заказ, не запускаем processing и notification
            MDC.clear();
            return;
        }

        // Запуск дальнейшей обработки
        restTemplate.postForEntity("http://processing-service/api/process", order, Void.class);

        // Инициирование уведомлений
        restTemplate.postForEntity("http://notification-service/api/notify", order, Void.class);

        successfulOrders.increment();
        MDC.clear();
    }
}

Для обеспечения пропускной способности до 100 заказов в секунду можно использовать:

  • Асинхронную обработку с очередью и пулом потоков.
  • Реактивные библиотеки (Spring WebFlux).
  • Ограничение скорости (rate limiting) на входящих запросах.

Таким образом, сервис будет корректно обрабатывать заказы, вести метрики и логи с корреляцией по идентификатору заказа.