DroneWQ is a Python package that can be used to analyze multispectral data collected from a drone to derive ocean color radiometry and water quality properties. These scripts are specific for the MicaSense RedEdge and Altum cameras.
Project description
DroneWQ: A Python library for measuring water quality with a multispectral drone sensor
DroneWQ is a Python package that can be used to analyze multispectral data collected from a drone to derive ocean color radiometry and water quality properties. These scripts are specific for the MicaSense RedEdge and Altum cameras. Please note that since this code was originally developed, changes to proprietary sensor branding have occurred. In 2021, MicaSense was acquired by AgEagle Aerial Systems, which subsequently rebranded as EagleNXT in 2025. The MicaSense sensor product line remains available and is expected to remain compatible with DroneWQ.
For details on the processing and theory of DroneWQ, please visit our readthedocs: https://dronewq.readthedocs.io/
Additional information on the methods can be found in:
Román, A., Heredia, S., Windle, A. E., Tovar-Sánchez, A., & Navarro, G., 2024. Enhancing Georeferencing and Mosaicking Techniques over Water Surfaces with High-Resolution Unmanned Aerial Vehicle (UAV) Imagery. Remote Sensing, 16(2), 290. https://doi.org/10.3390/rs16020290
Gray, P.C., Windle, A.E., Dale, J., Savelyev, I.B., Johnson, Z.I., Silsbe, G.M., Larsen, G.D. and Johnston, D.W., 2022. Robust ocean color from drones: Viewing geometry, sky reflection removal, uncertainty analysis, and a survey of the Gulf Stream front. Limnology and Oceanography: Methods. https://doi.org/10.1002/lom3.10511
Windle, A.E. and Silsbe, G.M., 2021. Evaluation of unoccupied aircraft system (UAS) remote sensing reflectance retrievals for water quality monitoring in coastal waters. Frontiers in Environmental Science, p.182. https://doi.org/10.3389/fenvs.2021.674247
Installation
[!IMPORTANT] Our code requires a system level dependencies called
gdal,zbar,exiftool, andopencv. It is the most straightforward to install them through conda-forge.
Install from PyPI (Recommended)
The easiest way to install DroneWQ is using conda and pip:
First, create your conda environment:
conda create -n {your project name} python=3.13 exiftool gdal zbar opencv -c conda-forge
Then, activate your environemt:
conda activate {your project name}
Finally, install dronewq:
pip install dronewq
Install from Source
If you want to install from source:
git clone https://github.com/aewindle110/DroneWQ.git
cd DroneWQ
conda create -n {your project name} python=3.13 exiftool gdal zbar opencv -c conda-forge
conda activate {your project name}
pip install .
System Requirements
DroneWQ requires Python >=3.10. Some dependencies require additional system libraries which is installed through conda:
- GDAL: Required for geospatial operations
- ExifTool: Required for reading MicaSense image metadata
- ZBar: Required for QR code reading from calibration panels
On Ubuntu/Debian:
sudo apt-get update
sudo apt-get install gdal-bin libgdal-dev exiftool opencv zbar-tools python3-gdal python3-cartopy
[!IMPORTANT] Pip on macOS currently doesn't search through the place homebrew installations go to. Thus, you have to specify it using this method https://github.com/npinchot/zbar/issues/3#issuecomment-1038005495
On macOS (using Homebrew):
brew install gdal exiftool zbar opencv
Docker Installation (Alternative)
The Docker image includes all dependencies pre-configured. See https://docs.docker.com/ for installation files.
With Docker installed and running, launch the container:
docker run -it -v <local directory>:/home/jovyan --rm -p 8888:8888 clifgray/dronewq:v3
where <local directory> is where you want data to be saved.
Then launch Jupyter Lab:
jupyter lab --allow-root --ip 0.0.0.0 /home/jovyan
Copy the generated URL from the terminal into your web browser.
Note: We have included a modified version of the MicaSense imageprocessing scripts in this repo. Our modifications include:
- Radiance data type expressed as Float32 instead of Uint16
- Image radiance output in milliwatts (mW) instead of watts (W)
- Modified
capture.save_capture_as_stack()to not scale and filter data
These modifications impact the panel_ed calculation. When MicaSense releases a package with user-specified radiance data types, we will revert to their official package.
Quick Start
1. Organize Your Data
DroneWQ requires MicaSense images organized in a specific folder structure:
<main_directory>/
├── panel/ # Calibrated reflectance panel images (before/after flight)
├── raw_sky_imgs/ # Sky images (40° from zenith, ~135° azimuth)
├── raw_water_imgs/ # Water images from flight
└── align_img/ # One image capture (5 .tif files) for alignment
Directory descriptions:
- panel/: Contains image captures of the MicaSense calibrated reflectance panel
- raw_sky_imgs/: Contains sky images taken at 40° from zenith and ~135° azimuthal viewing direction
- raw_water_imgs/: Contains all water images captured during the flight
- align_img/: Contains one image capture (5 .tif files, one per band) from
raw_water_imgs/used to compute the warp matrix for aligning all images
You can find a sample dataset (Lake Erie) at Zenodo DOI. It is recommended to upload this dataset to the examples directory.
2. Configure Settings
Before processing, configure the main directory path:
import dronewq
# Configure the main directory containing your organized images
dronewq.configure(main_dir="/path/to/your/main_directory")
The configure() function automatically sets up all subdirectory paths based on the main directory.
3. Process Raw Imagery to Remote Sensing Reflectance
The main processing function converts raw imagery to calibrated remote sensing reflectance:
from dronewq import Hedley, DlsEd, ThresholdMasking
# Process raw images to Rrs
dronewq.RRSPipeline(
output_folder=output_folder,
lw_method=Hedley(save_images=True),
ed_method=DlsEd(output_folder),
masking_method=ThresholdMasking(nir_threshold=0.02),
workers=4,
)
Processing workflow:
- Raw → Lt: Converts raw pixel values to radiance (Lt)
- Lt → Lw: Removes sky reflection to get water-leaving radiance (Lw)
- Lw → Rrs: Normalizes by downwelling irradiance (Ed) to get remote sensing reflectance (Rrs)
- Masking: Optionally masks pixels containing glint, shadows, or vegetation
4. Calculate Water Quality Parameters
Apply bio-optical algorithms to retrieve water quality parameters:
# Calculate chlorophyll-a using Gitelson algorithm
dronewq.save_wq_imgs(
wq_alg=["chl_gitelson"], # Options: chl_gitelson, chl_hu, chl_ocx, chl_hu_ocx, nechad_tsm
num_workers=4
)
5. Georeference and Mosaic
Georeference individual images and create an orthomosaic:
# Load metadata
import pandas as pd
metadata = pd.read_csv("/path/to/metadata.csv")
# Compute flight lines
flight_lines = dronewq.compute_flight_lines(
captures_yaw=metadata['Yaw'].values,
altitude=metadata['Altitude'].values[0],
pitch=0,
roll=0
)
# Georeference images
dronewq.georeference(
metadata=metadata,
input_dir=dronewq.settings.rrs_dir,
output_dir="/path/to/georeferenced/",
lines=flight_lines
)
# Create mosaic
dronewq.mosaic(
input_dir="/path/to/georeferenced/",
output_path="/path/to/mosaic.tif"
)
Detailed Documentation
For detailed documentation on the processing theory and methods, please visit: https://dronewq.readthedocs.io/
Example Workflow
See the primary_demo.ipynb notebook for a complete example workflow using the Lake Erie dataset. The notebook demonstrates:
- Setting up the workspace
- Extracting and viewing metadata
- Processing raw imagery to Rrs
- Applying bio-optical algorithms
- Georeferencing and creating mosaics
Performance Tips
- Parallel Processing: Adjust
num_workersbased on your CPU cores (default: 4) - Batch Processing: Use
startandcountparameters to process large datasets in batches - save_images=False: Turn off intermediate image saving to save disk space and speed up processing
Example orthomosaic of UAS images collected over Western Lake Erie, processed to chlorophyll a concentration using DroneWQ.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file dronewq-1.0.0.tar.gz.
File metadata
- Download URL: dronewq-1.0.0.tar.gz
- Upload date:
- Size: 70.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
96eb4075a78b32f43d1d63e42db7736837cb0bd60e84b2b2ac4b6cf0e5880a3f
|
|
| MD5 |
15ec80413e22742ed8875b73e78c12ec
|
|
| BLAKE2b-256 |
1a66ce190cc4636df9e47f72d578193cdd53569887e1e064bb4b32e3be8f06bc
|
Provenance
The following attestation bundles were made for dronewq-1.0.0.tar.gz:
Publisher:
python-publish.yml on aewindle110/DroneWQ
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
dronewq-1.0.0.tar.gz -
Subject digest:
96eb4075a78b32f43d1d63e42db7736837cb0bd60e84b2b2ac4b6cf0e5880a3f - Sigstore transparency entry: 1123220619
- Sigstore integration time:
-
Permalink:
aewindle110/DroneWQ@9cb999b3d45d282765a463b0445f95d994db987f -
Branch / Tag:
refs/tags/1.0.0 - Owner: https://github.com/aewindle110
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@9cb999b3d45d282765a463b0445f95d994db987f -
Trigger Event:
release
-
Statement type:
File details
Details for the file dronewq-1.0.0-py3-none-any.whl.
File metadata
- Download URL: dronewq-1.0.0-py3-none-any.whl
- Upload date:
- Size: 89.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c3ec1b53f79e21181b2d6fcd38a8c7966ccb749fc5b278ebd5d1c7ecb3c378e1
|
|
| MD5 |
0260eb7d751673a3ee8a39acb8bcbecc
|
|
| BLAKE2b-256 |
f4833f5b3e87e49064811f25a6f3d7f2d8c7433aed1911e84630f635fa02c7c4
|
Provenance
The following attestation bundles were made for dronewq-1.0.0-py3-none-any.whl:
Publisher:
python-publish.yml on aewindle110/DroneWQ
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
dronewq-1.0.0-py3-none-any.whl -
Subject digest:
c3ec1b53f79e21181b2d6fcd38a8c7966ccb749fc5b278ebd5d1c7ecb3c378e1 - Sigstore transparency entry: 1123220624
- Sigstore integration time:
-
Permalink:
aewindle110/DroneWQ@9cb999b3d45d282765a463b0445f95d994db987f -
Branch / Tag:
refs/tags/1.0.0 - Owner: https://github.com/aewindle110
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@9cb999b3d45d282765a463b0445f95d994db987f -
Trigger Event:
release
-
Statement type: