The Drone Delivery Problem#
Optimization Model for Drone Delivery
This notebooks outlines the optimization model designed to determine the minimum distance for drone delivery routing problems. The model employs a Mixed Integer Linear Programming (MILP) approach to minimize the total travel distance required to deliver packages from a warehouse to various delivery points.
Libraries
To begin, we import the necessary libraries.
pip install quantagonia
pip install folium
API_KEY="Your-API-KEY" # if you don't have one, head over to https://platform.quantagonia.com/ (free account available)
import numpy as np
import pandas as pd
import time as ttt
import random
import math
import folium
from pulp import *
from quantagonia import HybridSolverParameters
import quantagonia.mip.pulp_adapter as pulp_adapter
# Define problem parameters
NUM_LOCATIONS = 12 # Number of Location Points Including Warehouse
DRONE_CAPACITY = 3
Generate coordinates for the selected area, which in our case is Munich. To gain an initial perspective on our problem, we will visualize the data using the Folium.
random.seed(2)
# Latitude and Longitude ranges for Munich
lat_range = (48.114972911995785, 48.17622969284977) # Approximate latitude range for Munich
lon_range = (11.536111191947134, 11.611750294370422) # Approximate longitude range for Munich
# Generate random coordinates within the specified range
coordinates=np.zeros(NUM_LOCATIONS,dtype=object)
for i in range(NUM_LOCATIONS):
coordinates[i] = (round(random.uniform(lat_range[0], lat_range[1]), 17), round(random.uniform(lon_range[0], lon_range[1]), 8))
# Create a map centered around the average coordinates
avg_lat = sum(lat for lat, lon in coordinates) / len(coordinates)
avg_lon = sum(lon for lat, lon in coordinates) / len(coordinates)
map_munich = folium.Map(location=[avg_lat, avg_lon], zoom_start=10)
# Add markers for each coordinate
for i, coord in enumerate(coordinates):
if i == 0:
folium.Marker(location=coord, popup=f"Name: Warehouse, Lat: {coord[0]}, Lon: {coord[1]}", icon=folium.Icon(color='green')).add_to(map_munich)
else:
folium.Marker(location=coord, popup=f"Name: Delivery Point {i}, Lat: {coord[0]}, Lon: {coord[1]}").add_to(map_munich)
# Save the map to an HTML file
map_munich.save("Munich_coordinates_map.html")
# If running in a Jupyter notebook, display the map
map_munich
Distance Calculation
We use the Haversine formula to calculate the distance between two geographic coordinates and then construct the distance matrix for the model.
# Haversine Distance Function
def haversine_distance(coord1, coord2):
R = 6371 # Radius of the Earth in kilometers
lat1, lon1 = math.radians(coord1[0]), math.radians(coord1[1])
lat2, lon2 = math.radians(coord2[0]), math.radians(coord2[1])
dlat = lat2 - lat1
dlon = lon2 - lon1
a = math.sin(dlat / 2)**2 + math.cos(lat1) * math.cos(lat2) * math.sin(dlon / 2)**2
c = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))
return R * c
# Construct the distance matrix
distance_matrix = np.zeros((len(coordinates),len(coordinates)))
for i in range(len(coordinates)):
for j in range(len(coordinates)):
distance_matrix[i,j] = haversine_distance(coordinates[i], coordinates[j])
Sets and Indices:
Now, we define the sets:
\(i,j \in \text{locationset} = \{0,1,\ldots,(n-1)\}\), the set of locations (including warehouse, \(idx=0\))
\(i,j \in \text{locationset_without_wh} = \{1,2,\ldots,(n-1)\}\), the set of locations (excluding warehouse, \(idx=0\))
\(t \in \text{tours} = \{0,1,\ldots,\text{Maximum_Tours}\}\), the set of tours
# SETS
locationset = list(range(NUM_LOCATIONS)) ## Location set includinng warehouse (idx=0)
locationset_without_wh = list(range(1,NUM_LOCATIONS)) ## Location set excluding warehouse (idx=0)
tours = list(range(math.ceil((NUM_LOCATIONS-1)/DRONE_CAPACITY))) ## Tour set
Decision Variables
We define the decision variables for the model to be:
\(x_{ijt}\): binary Variable, 1 if location \(j\) is visited after location \(i\) within tour \(t\), otherwise 0.
\(y_{it}\): binary Variable, 1 if location \(i\) is visited in tour \(t\), otherwise \(0\).
\(u_i\): auxiliary integer variable to eliminate subtours, representing the order in which delivery point \(i\) is visited.
#DECISION VARIABLES
### x[i][j][t] - Binary: if location j is visited after location i within tour t, then 1; otherwise 0
x = []
for i in locationset:
x.append([])
for j in locationset:
x[i].append([])
for t in tours:
x[i][j].append(LpVariable(name = 'x(%d,%d,%d)' % (i,j,t), cat='Binary'))
### y[i][t] - Binary: if location i is visited in tour t, then 1; otherwise 0
y = []
for i in locationset:
y.append([])
for t in tours:
y[i].append(LpVariable(name = 'y(%d,%d)' % (i,t), cat='Binary'))
### u[i] - Auxiliary variable to eliminate subtours, representing the order in which delivery point i is visited
u = []
for i in locationset:
u.append(LpVariable(name = 'u(%d)' % (i),lowBound=0, cat='Integer'))
Objective Function
We define the objective function to minimize the total travel distance:
\begin{equation} \text{Minimize} \quad Z = \sum_{i \in \text{locationset}} \sum_{j \in \text{locationset}} \sum_{t \in \text{tours}} x_{ijt} \cdot \text{distance_matrix}_{ij} \tag{0} \end{equation}
Z = LpProblem("Deliver_Packages", LpMinimize)
### OBJECTIVE FUNCTION
Z += lpSum(x[i][j][t]*distance_matrix[i][j] for i in locationset for j in locationset for t in tours)
Constraints
Constraint 1:
The number of entries and exits for each location must be the same.
Constraint 2a:
The number of entries to each location \(j\) must be equal to 1 if it is visited in tour \(t\).
Constraint 2b:
The number of departures from each location \(i\) must be equal to 1 if it is visited in tour \(t\).
Constraint 3:
Each delivery location must be visited.
Constraint 4:
Capacity constraint: The maximum number of customers visited is limited by the drone’s capacity. Additionally, the warehouse node must be visited if any tour is activated.
Constraint 5a:
Subtour elimination and order constraints (Miller-Tucker-Zemlin (MTZ) formulation).
Constraint 5b:
Subtour elimination and order constraints (MTZ formulation).
### CONSTRAINTS
# Constraint 1: The number of entries and exits for each location must be the same.
for j in locationset:
for t in tours:
Z += lpSum(x[i][j][t] for i in locationset) == lpSum(x[j][i][t] for i in locationset)
# Constraint 2a: The number of entries to each location j must be equal to 1 if it is visited in tour t.
for j in locationset:
for t in tours:
Z +=lpSum(x[i][j][t] for i in locationset if i!=j) == y[j][t]
# Constraint 2b: The number of departures from each location i must be equal to 1 if it is visited in tour t.
for i in locationset:
for t in tours:
Z +=lpSum(x[i][j][t] for j in locationset if j!=i) == y[i][t]
# Constraint 3: Each delivery location must be visited.
for i in locationset_without_wh:
Z += lpSum(y[i][t] for t in tours) == 1
# Constraint 4: Capacity constraint: The maximum number of customers visited is limited by the drone's capacity.
# Additionally, the warehouse node must be visited if any tour is activated.
for t in tours:
Z += lpSum(y[i][t] for i in locationset_without_wh) <= DRONE_CAPACITY*y[0][t]
# Constraint 5a: Subtour Elimination and Order Constraints (Miller-Tucker-Zemlin (MTZ) formulation).
for t in tours:
for i in locationset_without_wh:
for j in locationset_without_wh:
if i==j:
continue
Z += (u[i]-u[j]+DRONE_CAPACITY*x[i][j][t] <= DRONE_CAPACITY-1)
# Constraint 5b: Subtour Elimination and Order Constraints (MTZ formulation).
for t in tours:
for i in locationset_without_wh:
Z += (u[i] <= DRONE_CAPACITY)
Z += (u[i] >= 1)
Solving the Model
We define the parameters (e.g. Time, Gap) and solve the model using a the HybridSolver from Quantagonia.
params = HybridSolverParameters()
params.set_time_limit(600)
params.set_seed(0)
q_solver = pulp_adapter.HybridSolver_CMD(api_key=API_KEY, params=params)
status = Z.solve(solver=q_solver)
print("Status:",LpStatus[status],"\nObjective Function Value: %.2f"%Z.objective.value(),"km")
✔ Queued job with jobid 0028b7fb-9985-4f49-a3dc-f18e118b3b0a...
✔ Job 0028b7fb-9985-4f49-a3dc-f18e118b3b0a unqueued, processing...
Quantagonia HybridSolver version 1.1.1781
Copyright (c) 2024 Quantagonia GmbH.
User-specified parameters:
Set parameter 'time_limit' to value '600.0'.
Set parameter 'seed' to value '0'.
Read b145cc629f4d4258ae4d044a766029be-pulp.mps in 0.59s.
Minimize a MILP with 687 constraints and 635 variables (624 binary, 11 integer, 0 implied integer,0 continuous).
Presolving model. Presolved model in 0.0s.
Reduced model has 551 constraints and 576 variables (565 binary, 11 integer, 0 continuous).
------------------------------------------------------------------------
Nodes | Incumbent | Bound | Gap (%) | Time (s) |
------------------------------------------------------------------------
1 | inf | 0.00000000 | inf | 0.00 |
* 1 | 46.7744772 | 19.8227293 | 57.62 | 0.00 |
* 3 | 43.9381887 | 27.2419397 | 38.00 | 0.02 |
* 59 | 43.3015915 | 27.2488165 | 37.07 | 0.43 |
* 82 | 43.0262083 | 27.2488165 | 36.67 | 0.68 |
* 162 | 42.3020208 | 27.2496758 | 35.58 | 1.27 |
* 229 | 42.1412607 | 27.2605539 | 35.31 | 1.80 |
386 | 42.1412607 | 27.2622454 | 35.31 | 2.93 |
* 539 | 42.1061606 | 27.2622454 | 35.25 | 4.23 |
* 880 | 41.9052291 | 28.0060363 | 33.17 | 4.76 |
* 884 | 41.8789535 | 28.0060363 | 33.13 | 4.86 |
* 1131 | 41.7559333 | 29.8798681 | 28.44 | 5.69 |
2017 | 41.7559333 | 31.0858794 | 25.55 | 11.48 |
2142 | 41.7559333 | 31.4118209 | 24.77 | 12.80 |
2338 | 41.7559333 | 31.6075000 | 24.30 | 14.04 |
2521 | 41.7559333 | 31.6556399 | 24.19 | 15.06 |
2770 | 41.7559333 | 32.1571995 | 22.99 | 16.10 |
2878 | 41.7559333 | 32.4013208 | 22.40 | 17.15 |
3036 | 41.7559333 | 32.6622455 | 21.78 | 18.62 |
3249 | 41.7559333 | 32.7038099 | 21.68 | 19.90 |
3465 | 41.7559333 | 32.8898708 | 21.23 | 21.54 |
3682 | 41.7559333 | 32.9232617 | 21.15 | 22.86 |
3908 | 41.7559333 | 33.0918351 | 20.75 | 24.07 |
4073 | 41.7559333 | 33.2434330 | 20.39 | 25.97 |
4385 | 41.7559333 | 33.3867754 | 20.04 | 27.48 |
------------------------------------------------------------------------
Nodes | Incumbent | Bound | Gap (%) | Time (s) |
------------------------------------------------------------------------
5280 | 41.7559333 | 33.5461010 | 19.66 | 28.54 |
5425 | 41.7559333 | 33.7102887 | 19.27 | 29.89 |
5811 | 41.7559333 | 33.7967712 | 19.06 | 31.01 |
6093 | 41.7559333 | 33.8393989 | 18.96 | 32.40 |
6491 | 41.7559333 | 34.0827942 | 18.38 | 33.42 |
6590 | 41.7559333 | 34.2933256 | 17.87 | 34.68 |
6883 | 41.7559333 | 34.4214993 | 17.57 | 35.74 |
7370 | 41.7559333 | 34.6465789 | 17.03 | 37.63 |
7620 | 41.7559333 | 34.8405817 | 16.56 | 38.85 |
8069 | 41.7559333 | 34.9895971 | 16.20 | 40.07 |
8380 | 41.7559333 | 35.1596993 | 15.80 | 41.13 |
8557 | 41.7559333 | 35.5214184 | 14.93 | 42.23 |
8926 | 41.7559333 | 35.6900359 | 14.53 | 43.50 |
9366 | 41.7559333 | 35.8195714 | 14.22 | 44.66 |
9814 | 41.7559333 | 35.8797123 | 14.07 | 45.72 |
10041 | 41.7559333 | 36.0815336 | 13.59 | 46.83 |
10391 | 41.7559333 | 36.3871303 | 12.86 | 48.00 |
10626 | 41.7559333 | 36.5415914 | 12.49 | 49.02 |
10924 | 41.7559333 | 36.7241490 | 12.05 | 50.25 |
11593 | 41.7559333 | 36.9130140 | 11.60 | 51.30 |
11826 | 41.7559333 | 37.0363145 | 11.30 | 52.47 |
12032 | 41.7559333 | 37.1233716 | 11.09 | 53.69 |
12624 | 41.7559333 | 37.4367840 | 10.34 | 55.43 |
13483 | 41.7559333 | 37.6922509 | 9.73 | 56.55 |
13823 | 41.7559333 | 37.9103551 | 9.21 | 57.75 |
------------------------------------------------------------------------
Nodes | Incumbent | Bound | Gap (%) | Time (s) |
------------------------------------------------------------------------
14282 | 41.7559333 | 38.2179958 | 8.47 | 58.94 |
14597 | 41.7559333 | 38.4545453 | 7.91 | 59.95 |
14974 | 41.7559333 | 38.6770731 | 7.37 | 60.99 |
15903 | 41.7559333 | 38.9883507 | 6.63 | 62.13 |
16327 | 41.7559333 | 39.2174342 | 6.08 | 63.20 |
16788 | 41.7559333 | 39.4652921 | 5.49 | 64.27 |
17336 | 41.7559333 | 39.7677276 | 4.76 | 65.38 |
18056 | 41.7559333 | 40.2870297 | 3.52 | 66.44 |
18918 | 41.7559333 | 41.7534278 | 0.01 | 67.44 |
------------------------------------------------------------------------
Optimal solution found (within relative tolerance 0.01%).
Solver Results:
- Solution Status: Optimal
- Wall Time: 67.4439 seconds
- Objective: 41.7559333
- Bound: 41.7534278
- Absolute Gap: 0.0025
- Relative Gap: 0.006%
- Nodes: 18918
- Best solution found at node 1131 after 6.2713 seconds
Finished processing job 0028b7fb-9985-4f49-a3dc-f18e118b3b0a...
Optimal
Status: Optimal
Objective Function Value: 41.76 km
Results
We extract the results:
# GET THE END RESULTS OF THE VARIABLES
resultx = np.zeros((len(locationset),len(locationset),len(tours)))
for i in locationset:
for j in locationset:
for t in tours:
resultx[i,j,t] = x[i][j][t].varValue
resulty = np.zeros((len(locationset),len(tours)))
for i in locationset:
for t in tours:
resulty[i,t] = y[i][t].varValue
## Extract the sequences
result_u = np.zeros(len(locationset))
for i in locationset_without_wh:
result_u[i]=u[i].varValue
## Extract the squences of the locations and which in tour they are visited
sequence_tour = np.zeros((NUM_LOCATIONS,2))
sequence_tour[:,1] = -1
for i in locationset_without_wh:
sequence_tour[i,0] = int(round(result_u[i]))
for t in tours:
if int(round(resulty[i,t]))==1:
sequence_tour[i,1] = t
# Construct the Output DataFrame
output_df = pd.DataFrame(sequence_tour[1:], columns=['Sequence','Tour'])
output_df['Location_Index'] = range(1,len(sequence_tour))
output_df['Coordinates'] = coordinates[1:]
output_df.sort_values(['Tour', 'Sequence'], ascending=[True, True],inplace=True)
for i in range(1,len(output_df)):
if output_df.iloc[i,1] == output_df.iloc[i-1,1]:
output_df.iloc[i,0] = output_df.iloc[i-1,0] + 1
else:
output_df.iloc[i,0] = 1
Visualizing the Results
We visualize the results using Folium:
### MAP VIEW OF THE SOLUTION
# Define the list of colors for the tours (currently max tours is limited by 15)
colors = ['purple', 'red', 'blue', 'black', 'darkred', 'lightred', 'beige',
'darkblue', 'cadetblue', 'darkpurple', 'pink', 'lightblue',
'gray', 'orange', 'lightgray']
# Calculate the average coordinates for centering the map
avg_lat = sum(lat for lat, lon in coordinates) / len(coordinates)
avg_lon = sum(lon for lat, lon in coordinates) / len(coordinates)
# Create a map centered around the average coordinates
map_munich = folium.Map(location=[avg_lat, avg_lon], zoom_start=10)
# Add the first marker as the warehouse with a green icon
folium.Marker(
location=coordinates[0],
popup=f"Name: Warehouse, Lat: {coordinates[0][0]}, Lon: {coordinates[0][1]}",
icon=folium.Icon(color='green')).add_to(map_munich)
# Add markers for each coordinate
for i in range(len(output_df)):
# Add markers for other locations with DivIcon displaying the sequence number
folium.Marker(
location=output_df['Coordinates'].iloc[i],
popup=f"Location {output_df['Location_Index'].iloc[i]}: Lat: {output_df['Coordinates'].iloc[i][0]}, Lon: {output_df['Coordinates'].iloc[i][1]}",
icon=folium.DivIcon(html=f'<div style="font-size: 16pt; color : black">{int(output_df["Sequence"].iloc[i])}</div>')
).add_to(map_munich)
# Add circle markers for other locations with specified colors
folium.CircleMarker(
location=output_df['Coordinates'].iloc[i],
radius=5,
color=colors[int(output_df['Tour'].iloc[i])%len(colors)],
fill=True,
fill_color=colors[int(output_df['Tour'].iloc[i])%len(colors)]
).add_to(map_munich)
# Create a dictionary to store coordinates by tour
tour_coordinates = {}
for i in range(len(output_df)):
tour = output_df['Tour'].iloc[i]
if tour not in tour_coordinates:
tour_coordinates[tour] = []
tour_coordinates[tour].append(output_df['Coordinates'].iloc[i])
# Draw dashed lines for each tour
for tour, coords in tour_coordinates.items():
# Connect warehouse to the first location in the tour
folium.PolyLine(
locations=[coordinates[0], coords[0]],
color=colors[int(tour)%len(colors)],
weight=2,
opacity=0.5,
dash_array='1, 1'
).add_to(map_munich)
# Connect each location in the tour by its sequence
folium.PolyLine(
locations=coords,
color=colors[int(tour)%len(colors)],
weight=2,
opacity=0.5,
dash_array='1, 1'
).add_to(map_munich)
# Connect the last location in a tour back to the warehouse
folium.PolyLine(
locations=[coords[-1], coordinates[0]],
color=colors[int(tour)%len(colors)],
weight=2,
opacity=0.5,
dash_array='1, 1'
).add_to(map_munich)
map_munich.save("map_munich.html")
map_munich
Need help with modeling? We are happy to coach you through your model formulation. Reach out to us at help@quantagonia.com or https://www.quantagonia.com/contact.