from grids.test_feature_grid import feature_grid_fixture
Tutorial: Create and Export QUIC-Fire Inputs with FastFuels SDK
In this tutorial, you'll learn how to use the FastFuels SDK to create QUIC-Fire input files for a region of interest. We'll walk through each step of the process, from authentication to final export.
Prerequisites
Before starting this tutorial, make sure you have:
- FastFuels SDK installed (pip install fastfuels-sdk
)
- A valid FastFuels API key
- Basic familiarity with Python and GeoPandas
What You'll Learn
By the end of this tutorial, you'll know how to: - Set up authentication for the FastFuels SDK - Create and work with Domains - Generate grids for fuels and topography - Export the data to QUIC-Fire format - Use the convenience function for simplified workflows
Step 1: Set Up Authentication
First, let's configure the SDK with your API key. While you can use environment variables, we'll set it programmatically for this tutorial:
Step 2: Create a Region of Interest
Let's create a GeoDataFrame representing an area in the Blue Mountain Recreation Area:
import geopandas as gpd
from shapely.geometry import Polygon
# Define the polygon coordinates for Blue Mountain area
coordinates = [
[-114.09957018646286, 46.82933208815811],
[-114.10141707482919, 46.828370407248826],
[-114.10010954324228, 46.82690548814563],
[-114.09560673134018, 46.8271123684554],
[-114.09592544216444, 46.829058122675065],
[-114.09957018646286, 46.82933208815811]
]
# Create a GeoDataFrame
polygon = Polygon(coordinates)
roi = gpd.GeoDataFrame(
geometry=[polygon],
crs="EPSG:4326" # WGS 84 coordinate system
)
Step 3: Create a Domain
Now let's create a Domain from our ROI:
from fastfuels_sdk.domains import Domain
domain = Domain.from_geodataframe(
geodataframe=roi,
name="Blue Mountain ROI",
description="Test area in Blue Mountain Recreation Area",
horizontal_resolution=2.0, # 2-meter horizontal resolution
vertical_resolution=1.0 # 1-meter vertical resolution
)
print(f"Created domain with ID: {domain.id}")
Step 4: Create Road and Water Features
Let's add road and water features from OpenStreetMap:
from fastfuels_sdk.features import Features
# Initialize Features for our domain
features = Features.from_domain_id(domain.id)
# Create features from OpenStreetMap
road_feature = features.create_road_feature_from_osm()
water_feature = features.create_water_feature_from_osm()
# Wait for features to be ready
road_feature.wait_until_completed()
water_feature.wait_until_completed()
Step 5: Create Feature Grid
Next, use the road and water features we created in Step 4 to generate a feature grid:
from fastfuels_sdk.grids import Grids
# Create feature grid
feature_grid = (
Grids.from_domain_id(domain.id)
.create_feature_grid(
attributes=["road", "water"],
)
)
feature_grid.wait_until_completed()
This will be used to mask out trees and non-burnable areas when we create the tree inventory in Step 7 and the surface grid in Step 8.
Step 6: Create Topography Grid
Add elevation data from 3DEP:
from fastfuels_sdk.grids import TopographyGridBuilder
topography_grid = (
TopographyGridBuilder(domain_id=domain.id)
.with_elevation_from_3dep(interpolation_method="linear")
.build()
)
topography_grid.wait_until_completed()
Step 7: Create Tree Inventory and Grid
Create a tree inventory and generate the canopy fuel grid:
from fastfuels_sdk.inventories import Inventories
from fastfuels_sdk.grids import TreeGridBuilder
# Create tree inventory
tree_inventory = Inventories.from_domain_id(
domain.id
).create_tree_inventory_from_treemap(
feature_masks=["road", "water"]
)
tree_inventory.wait_until_completed()
# Create tree grid
tree_grid = (
TreeGridBuilder(domain_id=domain.id)
.with_bulk_density_from_tree_inventory()
.build()
)
tree_grid.wait_until_completed()
Step 8: Create Surface Grid
Generate the surface fuels grid:
from fastfuels_sdk.grids import SurfaceGridBuilder
surface_grid = (
SurfaceGridBuilder(domain_id=domain.id)
.with_fuel_load_from_landfire(
product="FBFM40",
version="2022",
interpolation_method="cubic",
curing_live_herbaceous=0.25,
curing_live_woody=0.1,
groups=["oneHour"],
feature_masks=["road", "water"],
remove_non_burnable=["NB1", "NB2"],
)
.with_fuel_depth_from_landfire(
product="FBFM40",
version="2022",
interpolation_method="cubic",
feature_masks=["road", "water"],
remove_non_burnable=["NB1", "NB2"],
)
.with_uniform_fuel_moisture(
value=0.15,
feature_masks=["road", "water"]
)
.build()
)
surface_grid.wait_until_completed()
Step 9: Export to QUIC-Fire Format
Create and download the QUIC-Fire export:
from fastfuels_sdk.grids import Grids
from pathlib import Path
# Create the export
export = Grids.from_domain_id(domain.id).create_export("QUIC-Fire")
export.wait_until_completed()
# Download to a local directory
export_path = Path("quicfire_export")
export.to_file(export_path)
Using the Convenience Function
Now that you understand the complete process, you can use the convenience function to accomplish the same result with less code:
from fastfuels_sdk.convenience import export_roi_to_quicfire
# Export using the convenience function
export = export_roi_to_quicfire(
roi=roi,
export_path=Path("quicfire_export"),
verbose=True # See progress updates
)
Verifying the Export
Check your export directory for the QUIC-Fire input files:
# List the exported files
for file in Path("quicfire_export").glob("*.dat"):
print(f"Found QUIC-Fire input file: {file.name}")
You should see:
- treesrhof.dat
: Fuel bulk density
- treesmoist.dat
: Fuel moisture content
- treesfueldepth.dat
: Surface fuel bed depth
- topo.dat
: Elevation data
Next Steps
Now that you can create QUIC-Fire inputs, you might want to: - Experiment with different regions and resolutions - Customize the grid parameters for your specific needs - Learn about more advanced features of the FastFuels SDK
Troubleshooting
If you encounter issues:
- Check that your API key is valid and properly configured
- Verify that your ROI polygon is valid and uses the correct coordinate system
- Ensure you have sufficient permissions to write to the export directory
- Check the status of resources using their status
property