from exactextract import exact_extract
import ibis
from ibis import _
Having generated a Cloud-Optimized GeoTiff (COG) file containing our NDVI calculations, we are now ready to return to our redlining data and determine if indeed we still see the legacy of those racist practices more than 50 years after they were outlawed.
The aggregation of raster pixels grouped by their containing polygons is something like the spatial equivalent of our .group_by()
+ .aggregate()
operations in ibis, where the raster defines the values and the polygons define the groups.
To compute the average NDVI in each polygon, we will introduce yet another module – exactextract
. This package can efficiently compute summary statistics for each polygon given a raster image. While several different modules can perform this kind of aggregation, exactextract
is quite fast and scalable, using an efficient algorithm in C++, and gets it’s name for an ability to account for fractional coverage of a raster cell by a polygon, like so.
We begin as before by grabbing our remote city data. Here we have added an additional filter to focus only on residental data, as non-residental sites are not graded even though they are still marked on the map.
= ibis.duckdb.connect(extensions=["spatial"])
con
= (
redlines
con"/vsicurl/https://dsl.richmond.edu/panorama/redlining/static/mappinginequality.gpkg")
.read_geo(filter(_.city == "New Haven", _.residential)
.
)
Spatial operations such as overlaying polygons on the raster requires that the two datasets have matching coordinate reference systems. While we have already transformed our raster data to the ‘longitude, latitude’ coordinates, EPSG:4326, we need to manually add this annotation to declare that our parquet data is already using this projection as well. We do this with .set_crs()
option on geopandas
object:
= redlines.execute().set_crs("EPSG:4326") city
The exact_extract
documentation shows this is a flexible method able to take a raster and vector input and compute a range of summary statistics. The method is also somewhat flexible in how it formats the output – here we indicate we want the output formulated as a ‘geopandas’ object, keeping the geometry column and other useful columns from the original input data.
= exact_extract("ndvi.tif",
city_stats
city, "mean"],
[= True,
include_geom =["label", "grade", "city", "fill"],
include_cols="pandas")
output
city_stats.head()
/opt/miniforge3/lib/python3.12/site-packages/exactextract/exact_extract.py:330: RuntimeWarning: Spatial reference system of input features does not exactly match raster.
warnings.warn(
label | grade | city | fill | mean | geometry | |
---|---|---|---|---|---|---|
0 | A1 | A | New Haven | #76a865 | 0.709624 | MULTIPOLYGON (((-72.9 41.36885, -72.90325 41.3... |
1 | A2 | A | New Haven | #76a865 | 0.761685 | MULTIPOLYGON (((-72.89401 41.36331, -72.89256 ... |
2 | A3 | A | New Haven | #76a865 | 0.665130 | MULTIPOLYGON (((-72.90981 41.35949, -72.90284 ... |
3 | A4 | A | New Haven | #76a865 | 0.751388 | MULTIPOLYGON (((-72.90284 41.35795, -72.8999 4... |
4 | A5 | A | New Haven | #76a865 | 0.742981 | MULTIPOLYGON (((-72.89801 41.3493, -72.89883 4... |
We will write out this updated file as a parquet dataset. This provides both a mechanism to store our completed calculation and an easy way to go back from geopandas
into ibis
.
"new_haven_stats.parquet") city_stats.to_parquet(
We can bring our data back into ibis, where we can do our familiar manipulations.
= con.read_parquet("new_haven_stats.parquet") city
and we are finally ready to summarize the data to determine if the spectre of redlining remains: on average, are A grades still greener than D grades? Construct the rest of the ibis
code to compute the average NDVI by grade.