/index-calculator

A python program which calculates a chosen index for Sentinel-2 and Landsat 8 images and gives multiple output options.

Primary LanguagePythonGNU General Public License v3.0GPL-3.0

Index Calculator

Calculate different indices of Sentinel-2 and Landsat 8 raster images and, if desired, cut them to a specific area of interest (AOI, recommended since they are clipped beforehand and can therefore strongly decrease execution times!).

Test picture

Following indices are available so far:

Index Sentinel-2 Landsat 8
Atmospherically Resistant Vegetation Index (ARVI) x x
Green Chlorophyll Vegetation Index (GCI) x x
Green Normalized Difference Vegetation Index (GNDVI) x
Normalized Burn Ratio (NBR) x x
Normalized Burn Ratio 2 (NBR2) x x
Normalized Difference Build-up Index (NDBI) x x
Normalized Difference Moisture Index (NDMI) x x
Normalized Difference Red-Edge Vegetation Index (NDRE) x
Normalized Difference Snow Index (NDSI) x x
Normalized Difference Vegetation Index (NDVI) x x
Normalized Difference Water Index (NDWI) x x
Red-Edge Inflection Point (REIP) x
Soil-Adjusted Vegetation Index (SAVI) x x
Structure Intensive Pigment Vegetation Index (SIPI) x x

If you want any indices to be implemented as well, please don't hesitate to write an issue.

Output options

The following outputs can be automatically generated and saved to ./results/:

  • Plot the resulting array with the most suitable ranges (as found in literature)
  • Save the plot as figure
  • Save the resulting array as txt-file
  • Export the resulting array as GIS-ready tif-file (raster)
  • Generate a histogram and include descriptive statistics (min, max, mean, std.dev)

As before, if there is any type of output which you would want to get as well, please write an issue.

Data

Required data to calculate indices are multispectral raster images with specific bands needed for specific indices. So far both the Sentinel-2 and Landsat 8 satellite platforms with their respective multispectral sensoring systems are implemented and can be used as input datasets.
Note: indices for Landsat 8 datasets can only be calculated with a spatial resolution of 30 meters, Sentinel-2 offers the possibility to calculate with a spatial resolution of 10, 20 and 60 meters.

The datasets can be acquired through different ways, the following two are only exemplarily shown:

How to acquire Sentinel-2 raster data
  1. Navigate to Copernicus Open Access Hub by ESA registration form and set up an account (skip if already registered)
  2. Log in on Copernicus Open Access Hub. Without logging in you cannot download the required data
  3. Specify the search area in the map with right-click (move map with left-click and zoom in with mouse wheel)
  4. Click on the three stripes left of the search box to open the advanced search (upper left corner of screen)
  5. Select Sentinel-2 and put following statement in the box for the cloud cover: [0 TO 2]
  6. If you want to search for data in a specific time period, put the required dates in "sensing period"
  7. Click on the search button (upper right of search box) and wait until the results are displayed
  8. Search for an image with full extent (no black parts) and minimal cloud cover
  9. Hover over the entry and click on the eye icon ("View product details") which appears along with other icons on the lower right side of the entry
  10. Check in the quick look window if the data seems suitable

    If the images you are looking for are offline, take a look at troubleshooting - Sentinel-2 data offline for some help on that problem.

  11. In the Inspector, click on the download-arrow in the lower right corner to download the complete ZIP-file
  12. When downloaded, extract the ZIP-file and put the new folder in the ./data/raster/ folder (./data/raster/S*)
How to acquire Landsat 8 raster data
  1. Navigate to USGS EROS registration system and set up an account (skip if already registered)
  2. Log in on USGS login form. Without logging in you cannot download the required data
  3. Go to USGS Earth Explorer
  4. Specify the search area in the map by zooming in to the area of interest
  5. Click on the "Use Map" button on the left side. This bounds the area of interest in a rectangle which can be used as search criteria
  6. Click on "Data Sets" on top of the left box and select Landsat -> Landsat Collection 2 (Level 1 or 2) -> Landsat 8/9 OLI/TIRS...
  7. If you want to search for data in a specific time period or with specific cloud coverage, click on "Additional criteria" on top of the left boy and put the required dates in "Date Product Generated"
  8. Click on "Results" and wait until the results are displayed
  9. Search for an image with minimal cloud cover and with the right footprint (leftmost icon on each image)
  10. If the data seems suitable, you can click on the download icon of the according image. A popup pops up and enables you to choose the data you want to download
  11. Click on "Product options" and on the download button on top to download the whole data set
  12. When downloaded, extract the ZIP-file and put the new folder in the ./data/raster/ folder (./data/raster/L*)

Exemplary multispectral raster data for both satellites (S* is Sentinel-2, L* is Landsat 8) can be found here and should be put into ./data/raster/ (unzip so as the name of the file is the first folder like ./data/raster/L*/tif-files).

If you want to clip the raster data, you will need a shapefile with your AOI. It has to be manually put into ./data/shapes/.
An example AOI to test the program can be found in the HeiBOX folder mentioned above as well (unzip without creating a new folder like ./data/shapes/aoi.shp).

Getting Started

To run the program every machine having at least Python 3.9 installed is suitable. Your Python environment should additionally include following libraries:

  • numpy
  • matplotlib
  • fiona
  • rasterio

The rest should be included in the python installation. If you are using Linux or Mac, you should check the relative paths beforehand.

Installing

Create your virtual python environment and clone the repository. Open the terminal and navigate to the cloned/downloaded folder index-calculator. If you don't have the required packages installed already, call this first:

$ pip install -r ./requirements.txt

You should then be ready to execute the program.

Usage

If ready, call the program without any arguments to access the help within the terminal with information on how to use the arguments:

$ python src/main.py

usage: main.py [-h] -i Index name [-c Clip] [-sat Satellite] [-r Resolution] [-ov Optional value]
[-tif Save raster] [-gp Generate plot] [-sp Save plot] [-txt Save as txt] [-stat Statistics]

Calculate an index with Sentinel-2 satellite imagery.
You can use the following options to adapt the calculation to your needs. Have fun!

required arguments:
  -i Index name       String | Choose which index gets calculated.
                      Check the README for a list of possible indices.

optional arguments:
  -c Clip             String | Clip raster to shapefile with shapefile. Use the name and file-type only
                      (like aoi.shp). Default value: None
  -sat Satellite      String | You can use different satellite datasets (sentinel2/s2 or landsat8/l8).
                      Default value: s2
  -r Resolution       Integer | When using Sentinel-2 datasets, the indices can be calculated with
                      different resolutions (10, 20, 60 (m)). Default value: highest resolution possible
  -ov Optional value  Float | Some indices need additional values like the L-value in SAVI (0.5).
                      Default value: as in literature
  -tif Save raster    Boolean | Do you want to export the results/ndarray as tif-file locally
                      to ./results/? Use true/false. Default: false
  -gp Generate plot   Boolean | Do you want to generate a plot? Use true/false. Default: true
  -sp Save plot       Boolean | Do you want to save the plot locally to ./results/? Use true/false.
                      Default: false
  -txt Save as txt    Boolean | Do you want to save the results/ndarray as txt-file locally
                      to ./results/? Use true/false. Default: false
  -stat Statistics    Boolean | Do you want to generate statistics (histogram & descriptive) for
                      the results and save them locally to ./results/? Use true/false. Default: false

Exiting program, call again with arguments to run.

Cleaning up

If finished with multiple analyses, you can empty ./results/ and delete temporary used files from ./data/.
Make sure to save any results you want to keep to another location BEFORE executing the following command!

To do a cleanup, call:

$ python src/cleanup.py

Footnote:

If you have any questions, wishes or ideas, feel free to ask me in the issues section, I'm looking forward to it. Have fun!