Design a quick commerce grocery delivery system.

Design a quick commerce grocery delivery system.

Requirements:

a. How would you manage inventory updates in real-time across multiple store locations?

b. How would you handle order prioritization for perishable and non-perishable goods?

c. What considerations would you make for handling large order volumes during peak demand times, such as festivals or weekends?

d. How would you ensure that delivery routes are optimized based on the proximity of multiple stores, traffic conditions, and delivery time slots?


Proposed Solution:


1. Managing Real-time Inventory Updates Across Multiple Store Locations

  • Approach: Each store maintains its own inventory database, and updates (e.g., stock levels) are broadcast in real-time to a central system and related microservices. This can be achieved via event-driven architecture.

Design Patterns:

  • Observer Pattern: Used for notifying the central system whenever there's an inventory change in any store. Each store's inventory can act as a subject, and the central system as an observer.
  • Event-Driven Architecture (EDA): Each store can publish inventory events (REST API or message queue), which are processed by a central service to update the global inventory.

Tools & Technologies:

  • Kafka or RabbitMQ for asynchronous event processing.
  • Redis or Memcached for fast access to inventory data across stores.

public interface InventoryObserver {
    void update(String storeId, Product product, int quantity);
}        
public class CentralInventoryService implements InventoryObserver {
    @Override
    public void update(String storeId, Product product, int quantity) {
        // Update central inventory or notify relevant microservices. 
       //  Let's assume we can notify all the relevant stores within a location.
        System.out.println("Central System Updated: " + storeId + " updated " + product.getName() + " stock to " + quantity);
    }
}        
public class StoreInventory {
    private List<InventoryObserver> observers = new ArrayList<>();
    
    public void addObserver(InventoryObserver observer) {
        observers.add(observer);
    }
    
    public void updateProductStock(Product product, int quantity) {
        notifyObservers(product, quantity);
    }

    private void notifyObservers(Product product, int quantity) {
        for (InventoryObserver observer : observers) {
            observer.update(this.storeId, product, quantity);
        }
    }
}        
public class Product {
    private String name;
    private boolean isPerishable;

    public Product(String name, boolean isPerishable) {
        this.name = name;
        this.isPerishable = isPerishable;
    }

    public boolean isPerishable() {
        return isPerishable;
    }

    public String getName() {
        return name;
    }
}        
   StoreInventory storeInventory = new StoreInventory();
   storeInventory.addObserver(centralInventoryService);
   Product product = new Product("Apple", true);
   storeInventory.updateProductStock(product1 100); // Notify observers        
OUTPUT: Central System Updated: Store123 updated Apple stock to 100        

2. Order Prioritization for Perishable and Non-perishable Goods

  • Approach: Orders with perishable goods should have higher priority over non-perishable ones, especially in terms of delivery time.

Design Patterns:

  • Strategy Pattern: For dynamically applying different prioritization strategies based on factors like delivery time, perishability, and proximity to stores.
  • Chain of Responsibility Pattern: Use this to allow prioritization based on product types. Different handlers (perishable, non-perishable, default or mixed) will process the orders.

class Order {
    private List<Product> products;

    public Order(List<Product> products) {
        this.products = products;
    }

    public boolean hasPerishableItems() {
        return products.stream().anyMatch(Product::isPerishable);
    }

    public boolean hasNonPerishableItems() {
        return products.stream().anyMatch(product -> !product.isPerishable());
    }
}        
@Data
@AllArgsConstructor
public class OrderPriorityConfig {
    private int highPriority;
    private int mediumPriority;
    private int lowPriority;
    private int lowestPriority = 5;


    public OrderPriorityConfig() {
        this.highPriority = 1;
        this.mediumPriority = 2;
        this.lowPriority = 3;
    }
    
}        
public interface OrderPriorityStrategy {
    int calculatePriority(Order order, OrderPriorityConfig config);
}        
public class PerishablePriorityStrategy implements OrderPriorityStrategy {
    @Override
    public int calculatePriority(Order order, OrderPriorityConfig config) {
        return order.hasPerishableItems() ? config.getHighPriority() : config.getLowPriority();
    }
}        
public class NonPerishablePriorityStrategy implements OrderPriorityStrategy {
    @Override
    public int calculatePriority(Order order, OrderPriorityConfig config) {
        return order.hasNonPerishableItems() ? config.getMediumPriority() : config.getLowPriority();
    }
}        
interface OrderPriorityHandler {
    void setNextHandler(OrderPriorityHandler nextHandler);
    Integer calculatePriority(Order order);
}        
abstract class BasePriorityHandler implements OrderPriorityHandler {
    protected OrderPriorityHandler nextHandler;
    protected OrderPriorityStrategy strategy;

    public BasePriorityHandler(OrderPriorityStrategy strategy) {
        this.strategy = strategy;
    }

    @Override
    public void setNextHandler(OrderPriorityHandler nextHandler) {
        this.nextHandler = nextHandler;
    }

    protected Integer passToNext(Order order) {
        if (nextHandler != null) {
            return nextHandler.calculatePriority(order);
        }
        return null; 
    }
}        
class PerishablePriorityHandler extends BasePriorityHandler {
    public PerishablePriorityHandler(OrderPriorityStrategy strategy) {
        super(strategy);
    }

    @Override
    public Integer calculatePriority(Order order) {
        if (order.hasPerishableItems()) {
            return strategy.calculatePriority(order, new OrderPriorityConfig()); 
        } else {
            return passToNext(order); 
        }
    }
}        
class NonPerishablePriorityHandler extends BasePriorityHandler {
    public NonPerishablePriorityHandler(OrderPriorityStrategy strategy) {
        super(strategy);
    }

    @Override
    public Integer calculatePriority(Order order) {
        if (order.hasNonPerishableItems()) {
            return strategy.calculatePriority(order, new OrderPriorityConfig()); 
        } else {
            return passToNext(order); 
        }
    }
}        
class DefaultPriorityHandler extends BasePriorityHandler {
    public DefaultPriorityHandler() {
        super(null);
    }

    @Override
    public Integer calculatePriority(Order order) {
        return new OrderPriorityConfig().getLowestPriority();
    }
}        
@Service
class OrderPriorityService {
    private OrderPriorityHandler chain;

    public OrderPriorityService() {
        OrderPriorityHandler perishableHandler = new PerishablePriorityHandler(new PerishablePriorityStrategy());

        OrderPriorityHandler nonPerishableHandler = new NonPerishablePriorityHandler(new NonPerishablePriorityStrategy());

        OrderPriorityHandler defaultHandler = new DefaultPriorityHandler();

        perishableHandler.setNextHandler(nonPerishableHandler);
        nonPerishableHandler.setNextHandler(defaultHandler);

        this.chain = perishableHandler;
    }

    public int calculatePriority(Order order) {
        return chain.calculatePriority(order);
    }
}        
Order order1 = new Order(Arrays.asList(new Product("A",true)));
System.out.println("Order 1 Priority: " + orderPriorityService.calculatePriority(order1)); // Output: 1

Order order2 = new Order(Arrays.asList(new Product("B", false)));
System.out.println("Order 2 Priority: " + orderPriorityService.calculatePriority(order2)); // Output: 2

Order order3 = new Order(Arrays.asList());
System.out.println("Order 3 Priority: " + orderPriorityService.calculatePriority(order3)); // Output: 5        

3. Handling Large Order Volumes During Peak Demand (Festivals or Weekends)

  • Approach: During peak demand times, the system must balance load across multiple microservices and ensure that large order volumes don’t overwhelm any one part of the system.

Design Patterns:

  • Circuit Breaker Pattern: To handle failures gracefully during high demand (e.g., one microservice is overwhelmed and slow, so it triggers fallback logic).
  • Bulkhead Pattern: Isolate system resources to avoid cascading failures. For instance, isolate different parts of the system such as payment processing or inventory updates.
  • Queue-Based Load Leveling: Use message queues (Kafka, SQS) to handle asynchronous order processing and spark for distributed processing.
  • SNS with SQS: Use this combination for message polling across various receivers in notification systems.


public class OrderService {
    @CircuitBreaker(fallbackMethod = "fallbackProcessOrder")
    public void processOrder(Order order) {
        // Order processing logic
    }

    public void fallbackProcessOrder(Order order) {
        // Fallback logic if the circuit breaker is triggered
    }
}        

Handling large order volumes during peak demand times, such as festivals or weekends, requires careful planning and optimization of resources to ensure smooth operations. Here are the key considerations for managing high traffic during such periods:

1. Scalability of Infrastructure

  • Cloud-Based Auto Scaling: Use cloud infrastructure (e.g., AWS, Azure, GCP) with auto-scaling capabilities to dynamically adjust resources (e.g., servers, databases) based on traffic. This ensures you handle increased demand without manual intervention.
  • Horizontal Scaling: Add more servers to distribute the load across multiple nodes rather than relying on a single powerful machine.
  • Load Balancers: Implement load balancing to evenly distribute traffic across servers, preventing any one server from becoming overwhelmed.

2. Caching Mechanisms

  • Use Distributed Caching: Implement caching solutions like Redis, Memcached, or CDN to reduce load on your database. Frequently accessed data, such as product catalogs or user profiles, should be cached to speed up response times.
  • Inventory Caching: Cache inventory status locally for each store location, with regular syncs to the central inventory system. This prevents constant database hits for inventory checks during peak times.

3. Optimized Inventory Management

  • Real-Time Inventory Updates: Use real-time inventory management with strong consistency to avoid stockouts or overselling. Systems should be capable of quick updates as inventory is consumed.
  • Distributed Inventory: Spread inventory across multiple warehouses and stores to ensure proximity to delivery locations and faster fulfillment. This also helps balance order volumes across regions.
  • Buffer Stock: Maintain a buffer stock of fast-moving items, especially perishable goods, during high-demand periods to prevent stockouts.

4. Order Prioritization and Batch Processing

  • Order Priority Based on Type and Urgency: Implement a priority system for perishable vs. non-perishable goods. Perishable items should be fulfilled first to avoid spoilage, while high-priority customers (e.g., express deliveries) should also be handled sooner.
  • Batch Processing: Group orders based on location or product category to optimize picking, packing, and delivery. This can reduce the load on the logistics team and improve efficiency during peak times.

5. Efficient Delivery Logistics

  • Route Optimization: Use route optimization algorithms (potentially enhanced by AI) to select the best delivery paths based on traffic, delivery location proximity, and available drivers. Solutions like Google Maps API, or custom-built tools with Spark and machine learning, can help optimize routes.
  • Dynamic Assignment of Drivers: During high demand, use dynamic driver assignment by considering proximity to stores and current traffic conditions. This ensures faster delivery and reduces idle time for drivers.
  • Partnerships with 3rd Party Delivery Services: If your fleet can't handle the volume, partner with third-party delivery services like Uber, DoorDash, or local logistics firms to expand capacity on demand.

6. Queue and Throttling Mechanisms

  • Queue-Based Order Processing: Use message queues (e.g., Kafka, RabbitMQ) to decouple services and handle order bursts smoothly. Orders are placed in a queue and processed asynchronously, allowing for smoother operation during peak times.
  • Throttling: Implement API request throttling to prevent system overload. This ensures that services don’t crash under sudden spikes in traffic, and important transactions can still be processed.

7. Peak-Time Capacity Planning

  • Predictive Analytics: Use historical data to forecast order volumes during peak times. This allows you to prepare well in advance with adequate stock, delivery capacity, and customer support.
  • Elastic Workforce: Hire temporary workers or have on-call staff for critical operations like packing, sorting, and customer service during peak times. Use workforce scheduling tools to ensure enough manpower is available.
  • Stock Pre-Allocation: Pre-allocate stock to stores based on demand predictions. Use AI/ML models to predict product demand and ensure that popular items are stocked up.

8. System Resilience and Failover

  • Database Sharding: Implement database sharding to distribute the load on the database across multiple instances. Each shard handles a subset of data, reducing the impact of heavy read/write operations.
  • Backup and Disaster Recovery: Ensure that backup systems and disaster recovery plans are in place. If one system fails, another should take over without downtime.
  • Failover Systems: If your main infrastructure experiences a bottleneck or failure, have redundant systems that can take over operations seamlessly to avoid disruption.

9. Customer Communication

  • Real-Time Updates: Provide real-time status updates to customers about their order status, expected delivery times, and delays (if any). This can reduce customer inquiries and frustration during busy periods.
  • Surge Pricing and Expected Delays: During peak demand, communicate the possibility of delays upfront or implement surge pricing for express services, much like Uber does, to manage customer expectations and load.

10. Optimized Checkout and Payment Systems

  • Quick and Efficient Checkout: Optimize the checkout process by using cached user profiles, one-click payments, and seamless integration with multiple payment gateways.
  • Payment Load Balancing: Use multiple payment gateways to handle large volumes of transactions without causing delays or downtime.

11. Monitoring and Alerting

  • Real-Time System Monitoring: Implement robust monitoring tools (e.g., Datadog, New Relic, Grafana) to track system performance, database health, and order fulfillment metrics in real-time. Set up alerts to be notified of potential bottlenecks early.
  • Order Fulfillment Tracking: Monitor order processing times, warehouse bottlenecks, and delivery delays during peak times. Implement contingency plans if certain thresholds are exceeded.

class CacheService {
    public void updateInventory(String storeId, String productId, int quantity) {
        System.out.println("Updated inventory in Redis cache for Store: " + storeId + ", Product: " + productId + ", Quantity: " + quantity);
    }
}        
public class CentralInventoryService implements InventoryObserver {
    private KafkaService kafkaService;
    private CacheService cacheService; // redis

    public CentralInventoryService(KafkaService kafkaService, CacheService cacheService) {
        this.kafkaService = kafkaService;
        this.cacheService = cacheService;
    }

    @Override
    public void update(String storeId, Product product, int quantity) {
        cacheService.updateInventory(storeId, product.getName(), quantity);
        kafkaService.publish("inventory_updates", new InventoryUpdateEvent(storeId, product.getName(), quantity));
        // other microservices will consume
        System.out.println("Central inventory updated and notification sent via Kafka.");
    }
}        
class InventoryUpdateEvent {
    private String storeId;
    private String productId;
    private int quantity;

    public InventoryUpdateEvent(String storeId, String productId, int quantity) {
        this.storeId = storeId;
        this.productId = productId;
        this.quantity = quantity;
    }

    @Override
    public String toString() {
        return "InventoryUpdateEvent{" +
                "storeId='" + storeId + '\'' +
                ", productId='" + productId + '\'' +
                ", quantity=" + quantity +
                '}';
    }
}        
class KafkaService<T> {
    public void publish(String topic, T event) {
        System.out.println("Published event to topic: " + topic + " -> " + event);
    }
}        
public class StoreInventory {
    private String storeId;
    private List<InventoryObserver> observers = new ArrayList<>();
    private CacheService cacheService;

    public StoreInventory(String storeId, CacheService cacheService) {
        this.storeId = storeId;
        this.cacheService = cacheService;
    }

    public void addObserver(InventoryObserver observer) {
        observers.add(observer);
    }

    public void updateProductStock(Product product, int quantity) {
        cacheService.updateInventory(storeId, product.getName(), quantity);
        System.out.println("Stock updated in local cache for store: " + storeId);
        notifyObservers(product, quantity);
    }

    private void notifyObservers(Product product, int quantity) {
        for (InventoryObserver observer : observers) {
            observer.update(this.storeId, product, quantity);
        }
    }
}        
public class Driver {

    public static void main(String[] args) {
        KafkaService<InventoryUpdateEvent> kafkaService = new KafkaService<>();
        CacheService cacheService = new CacheService();
        CentralInventoryService centralInventoryService = new CentralInventoryService(kafkaService, cacheService);
        StoreInventory storeInventory = new StoreInventory("Store1", cacheService);
        storeInventory.addObserver(centralInventoryService);

        Product milk = new Product( "Milk", true);
        Product cereal = new Product( "Cereal", false);
        System.out.println("Peak demand updates:");
        storeInventory.updateProductStock(milk, 100);
        storeInventory.updateProductStock(cereal, 200);
        System.out.println("\nRegular updates:");
        storeInventory.updateProductStock(milk, 50);
    }

}        
OUTPUT:

Peak demand updates:
Updated inventory in Redis cache for Store: Store1, Product: Milk, Quantity: 100
Stock updated in local cache for store: Store1
Updated inventory in Redis cache for Store: Store1, Product: Milk, Quantity: 100
Published event to topic: inventory_updates -> InventoryUpdateEvent{storeId='Store1', productId='Milk', quantity=100}
Central inventory updated and notification sent via Kafka.
Updated inventory in Redis cache for Store: Store1, Product: Cereal, Quantity: 200
Stock updated in local cache for store: Store1
Updated inventory in Redis cache for Store: Store1, Product: Cereal, Quantity: 200
Published event to topic: inventory_updates -> InventoryUpdateEvent{storeId='Store1', productId='Cereal', quantity=200}
Central inventory updated and notification sent via Kafka.

Regular updates:
Updated inventory in Redis cache for Store: Store1, Product: Milk, Quantity: 50
Stock updated in local cache for store: Store1
Updated inventory in Redis cache for Store: Store1, Product: Milk, Quantity: 50
Published event to topic: inventory_updates -> InventoryUpdateEvent{storeId='Store1', productId='Milk', quantity=50}
Central inventory updated and notification sent via Kafka.        

Refer to my other articles mentioned below to learn more about how to use kafka along with redis and spark for handling large scale:

https://www.dhirubhai.net/pulse/design-system-based-internal-app-swiggyzomato-notify-all-ashish-uv2fc

https://www.dhirubhai.net/pulse/designing-leaderboard-internal-app-used-swiggy-zomato-sai-ashish-kep4c


4. Optimizing Delivery Routes Based on Proximity, Traffic, and Delivery Time Slots

  • Approach: A dynamic routing algorithm that optimizes delivery based on multiple factors such as store proximity, traffic data, and time slots using spark, kafka, min heap, batch processing and redis cache and influence factor computation based on related driver filter data from the database based on route, shift, location, and performance metrics such as delivery times and customer ratings.

@Service
public class CacheService {
    private Map<String, Driver> driverCache;
    private TrafficData trafficDataCache;

    public CacheService() {
        driverCache = new HashMap<>();
        trafficDataCache = new TrafficData();
    }

    public Map<String, Driver> getAvailableDrivers() {
        return driverCache;
    }

    public TrafficData getTrafficData() {
        return trafficDataCache;
    }

    public void updateDriverLocation(String driverId, Location location) {
        Driver driver = driverCache.get(driverId);
        if (driver != null) {
            driver.setLocation(location);
        }
    }
}        
@Data
public class Delivery {
    private String customerId;
    private String storeId;
    private String deliveryId;
    private Location deliveryLocation;


    public Delivery(String customerId, String storeId, String deliveryId, Location deliveryLocation) {
        this.customerId = customerId;
        this.storeId = storeId;
        this.deliveryId = deliveryId;
        this.deliveryLocation = deliveryLocation;
    }

}        
public class DeliveryRequestEvent extends KafkaEvent {
    private List<Store> stores;
    private Delivery delivery;

    public DeliveryRequestEvent(List<Store> stores, Delivery delivery) {
        this.stores = stores;
        this.delivery = delivery;
    }

    public List<Store> getStores() {
        return stores;
    }

    public Delivery getDelivery() {
        return delivery;
    }
}        
@NoArgsConstructor
@AllArgsConstructor
@Data
public class Driver {
    private String id = "1234";
    private Location location;
    private int estimatedTime = 23456789;
}        
public class DeliveryRequestEvent extends KafkaEvent {
    private List<Store> stores;
    private Delivery delivery;

    public DeliveryRequestEvent(List<Store> stores, Delivery delivery) {
        this.stores = stores;
        this.delivery = delivery;
    }

    public List<Store> getStores() {
        return stores;
    }

    public Delivery getDelivery() {
        return delivery;
    }
}        
public class DriverLocationUpdateEvent extends KafkaEvent {
    private String driverId;
    private Location location;

    public DriverLocationUpdateEvent(String driverId, Location location) {
        this.driverId = driverId;
        this.location = location;
    }

    public String getDriverId() {
        return driverId;
    }

    public Location getLocation() {
        return location;
    }
}        
import java.util.UUID;
import java.time.Instant;

public abstract class KafkaEvent {
    private final String eventId;
    private final Instant timestamp;

    public KafkaEvent() {
        this.eventId = UUID.randomUUID().toString(); 
        this.timestamp = Instant.now();          
    }

    public String getEventId() {
        return eventId;
    }

    public Instant getTimestamp() {
        return timestamp;
    }

    @Override
    public String toString() {
        return "KafkaEvent{" +
                "eventId='" + eventId + '\'' +
                ", timestamp=" + timestamp +
                '}';
    }
}        
@Service
public class KafkaService<T extends KafkaEvent> {
    public void subscribe(String topic, Consumer<T> consumer) {
        System.out.println("Subscribed to topic: " + topic);
    }

    public void publish(String topic, KafkaEvent event) {
        System.out.println("Published event to topic: " + topic + ", event: " + event);
    }
}        
@Data
@AllArgsConstructor
public class Location {
    private double latitude;
    private double longitude;
    private String address; 
    private String city;    
    
    public Location(double latitude, double longitude) {
        this.latitude = latitude;
        this.longitude = longitude;
    }

    @Override
    public String toString() {
        return "Location{" +
                "latitude=" + latitude +
                ", longitude=" + longitude +
                (address != null ? ", address='" + address + '\'' : "") +
                (city != null ? ", city='" + city + '\'' : "") +
                '}';
    }
}        
public class OptimizedRouteEvent extends KafkaEvent {
    private Route route;

    public OptimizedRouteEvent(Route route) {
        this.route = route;
    }
}        
@Data
@AllArgsConstructor
public class Route {
    private String id;
    private Location deliveryLocation;
    private int estimatedTime;
}        
@Service
@RequiredArgsConstructor
public class RouteOptimizationService {
    private final CacheService cacheService;
    private final KafkaService<DeliveryRequestEvent> deliveryRequestService;
    private final KafkaService<DriverLocationUpdateEvent> driverLocationService;
    private final SparkService sparkService;

    public Route optimizeRoute(List<Store> stores, Delivery delivery) {
        Map<String, Driver> availableDrivers = cacheService.getAvailableDrivers();
        TrafficData trafficData = cacheService.getTrafficData();
        Location deliveryLocation = delivery.getDeliveryLocation();
        List<Location> storeLocations = stores.stream().map(Store::getLocation).toList();
        List<Driver> optimizedDrivers = sparkService.optimizeDrivers(availableDrivers, storeLocations, deliveryLocation, trafficData);
        Driver bestDriver = selectBestDriver(optimizedDrivers, delivery);
        return new Route(bestDriver.getId(), deliveryLocation, bestDriver.getEstimatedTime());
    }

    private Driver selectBestDriver(List<Driver> drivers, Delivery delivery) {
        PriorityQueue<Driver> driversPQ = new PriorityQueue<>(drivers);
        // use min-heap to find the best driver and handle corner cases
        return driversPQ.poll();
    }

    public void listenForEvents() {
        deliveryRequestService.subscribe("delivery_requests", this::onDeliveryRequest);
        driverLocationService.subscribe("driver_locations", this::onDriverLocationUpdate);
    }

    private void onDeliveryRequest(DeliveryRequestEvent event) {
        List<Store> stores = event.getStores();
        Delivery delivery = event.getDelivery();
        Route route = optimizeRoute(stores, delivery);
        deliveryRequestService.publish("optimized_routes", new OptimizedRouteEvent(route));
    }

    private void onDriverLocationUpdate(DriverLocationUpdateEvent event) {
        cacheService.updateDriverLocation(event.getDriverId(), event.getLocation());
    }
}        
@Service
public class SparkService {

    public List<Driver> optimizeDrivers(Map<String, Driver> availableDrivers, List<Location> storeLocations, Location deliveryLocation, TrafficData trafficData) {
        // Spark job to analyze historical data, traffic conditions, and optimize drivers
        // Mock optimization logic

        // Implement batch processing with a Kafka executor to
        // filter data from the database based on route, shift, location,
        // and performance metrics such as delivery times and customer ratings.
        // Utilize an influence factor computation to rank drivers.
        // Cache the relevant drivers and use a priority queue or
        // min-heap to identify the optimal driver for each delivery.

        System.out.println("Running Spark job to optimize drivers...");

        availableDrivers.put("1234", new Driver());

        return availableDrivers.values().stream().toList(); 
    }
}        
public class TrafficData {
    private Map<String, Integer> trafficMap;  // Map of location to traffic delay in minutes

    public TrafficData() {
        trafficMap = new HashMap<>();
    }

    public int getTrafficDelay(String location) {
        return trafficMap.getOrDefault(location, 0);
    }

    public void updateTraffic(String location, int delay) {
        trafficMap.put(location, delay);
    }
}        
KafkaService<DeliveryRequestEvent> deliveryRequestService = new KafkaService<>();

KafkaService<DriverLocationUpdateEvent> driverLocationService = new KafkaService<>();

RouteOptimizationService routeService = new RouteOptimizationService(cacheService, deliveryRequestService, driverLocationService, sparkService);

Delivery delivery = new Delivery("123", "456", "789", new Location(12.9716, 77.5946));
List<Store> stores = Arrays.asList(
        new Store("Store1", new Location(12.9716, 77.5946)),
        new Store("Store2", new Location(12.9721, 77.5955))
);

Route route = routeService.optimizeRoute(stores, delivery);
System.out.println("Optimized Route: " + route);        
OUTPUT:

Running Spark job to optimize drivers...
Optimized Route: Route(id=1234, deliveryLocation=Location{latitude=12.9716, longitude=77.5946}, estimatedTime=23456789)        

Related articles to understand how to compute related drivers and rate them at scale:

https://www.dhirubhai.net/pulse/design-system-based-internal-app-swiggyzomato-notify-all-ashish-uv2fc

https://www.dhirubhai.net/pulse/designing-leaderboard-internal-app-used-swiggy-zomato-sai-ashish-kep4c


#softwaredevelopment #softwareengineering #applicationdevelopment #systemdesign



要查看或添加评论,请登录

社区洞察

其他会员也浏览了