diff --git a/doc/platforms_supported.rst b/doc/platforms_supported.rst index e01b5e46..b583246f 100644 --- a/doc/platforms_supported.rst +++ b/doc/platforms_supported.rst @@ -95,8 +95,17 @@ have been included in Pyspectral. - `rsr_viirs_Suomi-NPP.h5` - GSICS_ * - Landsat-8 oli - - `rsr_oli_Landsat-8.h5` - - NASA-Landsat-OLI_ + - `rsr_oli_tirs_Landsat-8.h5` + - NASA-Landsat-8-OLI_ + * - Landsat-8 tirs + - `rsr_oli_tirs_Landsat-8.h5` + - NASA-Landsat-8-TIRS_ + * - Landsat-9 oli-2 + - `rsr_oli_tirs_Landsat-9.h5` + - NASA-Landsat-9-OLI_ + * - Landsat-9 tirs-2 + - `rsr_oli_tirs_Landsat-9.h5` + - NASA-Landsat-9-TIRS_ * - FY-3D mersi-2 - `rsr_mersi-2_FY-3D.h5` - CMA_ (Acquired via personal contact) @@ -125,7 +134,10 @@ have been included in Pyspectral. .. _ESA-Sentinel-OLCI: https://sentinel.esa.int/documents/247904/322304/OLCI+SRF+%28NetCDF%29/15cfd7a6-b7bc-4051-87f8-c35d765ae43a .. _ESA-Sentinel-SLSTR: https://sentinel.esa.int/web/sentinel/technical-guides/sentinel-3-slstr/instrument/measured-spectral-response-function-data .. _ESA-Sentinel-MSI: https://earth.esa.int/documents/247904/685211/S2-SRF_COPE-GSEG-EOPG-TN-15-0007_3.0.xlsx -.. _NASA-Landsat-OLI: https://landsat.gsfc.nasa.gov/wp-content/uploads/2013/06/Ball_BA_RSR.v1.1-1.xlsx +.. _NASA-Landsat-8-OLI: https://landsat.gsfc.nasa.gov/wp-content/uploads/2014/09/Ball_BA_RSR.v1.2.xlsx +.. _NASA-Landsat-9-OLI: https://landsat.gsfc.nasa.gov/wp-content/uploads/2024/03/L9_OLI2_Ball_BA_RSR.v2-1.xlsx +.. _NASA-Landsat-8-TIRS: https://landsat.gsfc.nasa.gov/wp-content/uploads/2013/06/TIRS_Relative_Spectral_Responses.BA_.v1.xlsx +.. _NASA-Landsat-9-TIRS: https://landsat.gsfc.nasa.gov/wp-content/uploads/2021-10/L9_TIRS2_Relative_Spectral_Responses.BA.v1.0.xlsx .. _NESDIS: https://ncc.nesdis.noaa.gov/J1VIIRS/J1VIIRSSpectralResponseFunctions.php .. _CMA: http://www.cma.gov.cn/en2014/ .. _NWPSAF-MetImage: https://nwpsaf.eu/downloads/rtcoef_rttov12/ir_srf/rtcoef_metopsg_1_metimage_srf.html diff --git a/pyspectral/etc/pyspectral.yaml b/pyspectral/etc/pyspectral.yaml index 06d0aca6..df651f4e 100644 --- a/pyspectral/etc/pyspectral.yaml +++ b/pyspectral/etc/pyspectral.yaml @@ -136,6 +136,16 @@ download_from_internet: True # Envisat-aatsr: # path: /path/to/original/envisat/aatsr/data +# Landsat-8-oli_tirs: +# path: D:\sat_data\ +# oli: Ball_BA_RSR.v1.2.xlsx +# tirs: TIRS_Relative_Spectral_Responses.BA_.v1.xlsx + +# Landsat-9-oli_tirs: +# path: D:\sat_data\ +# oli: L9_OLI2_Ball_BA_RSR.v2-1.xlsx +# tirs: L9_TIRS2_Relative_Spectral_Responses.BA.v1.0.xlsx + # Sentinel-3A-slstr: # path: /path/to/original/sentinel-3a/slstr/data # ch1: SLSTR_FM02_S1_20150122.nc diff --git a/pyspectral/utils.py b/pyspectral/utils.py index e74d9d2a..b304c540 100644 --- a/pyspectral/utils.py +++ b/pyspectral/utils.py @@ -87,7 +87,8 @@ 'Electro-L-N2': 'msu-gs', 'Sentinel-3A': ['olci', 'slstr'], 'Sentinel-3B': ['olci', 'slstr'], - 'Landsat-8': 'oli', + 'Landsat-8': 'oli_tirs', + 'Landsat-9': 'oli_tirs', 'Meteosat-10': 'seviri', 'Meteosat-11': 'seviri', 'Meteosat-8': 'seviri', @@ -108,10 +109,10 @@ 'avhrr-2': 'avhrr/2', 'avhrr-3': 'avhrr/3'} -HTTP_PYSPECTRAL_RSR = "https://zenodo.org/records/13833977/files/pyspectral_rsr_data.tgz" +HTTP_PYSPECTRAL_RSR = "https://zenodo.org/records/14008148/files/pyspectral_rsr_data.tgz" RSR_DATA_VERSION_FILENAME = "PYSPECTRAL_RSR_VERSION" -RSR_DATA_VERSION = "v1.4.0" +RSR_DATA_VERSION = "v1.4.1" ATM_CORRECTION_LUT_VERSION = {} diff --git a/rsr_convert_scripts/oli_reader.py b/rsr_convert_scripts/oli_reader.py deleted file mode 100644 index b833b795..00000000 --- a/rsr_convert_scripts/oli_reader.py +++ /dev/null @@ -1,85 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# -# Copyright (c) 2017-2022 Pytroll developers -# -# -# This program is free software: you can redistribute it and/or modify -# it under the terms of the GNU General Public License as published by -# the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# This program is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU General Public License for more details. -# -# You should have received a copy of the GNU General Public License -# along with this program. If not, see . - -"""Landsat-8 OLI reader.""" - -import logging -import os - -import numpy as np -from xlrd import open_workbook - -from pyspectral.raw_reader import InstrumentRSR -from pyspectral.utils import convert2hdf5 as tohdf5 - -LOG = logging.getLogger(__name__) - - -OLI_BAND_NAMES = {'CoastalAerosol': 'B1', - 'Blue': 'B2', - 'Green': 'B3', - 'Red': 'B4', - 'NIR': 'B5', - 'Cirrus': 'B9', - 'SWIR1': 'B6', - 'SWIR2': 'B7', - 'Pan': 'B8'} - - -class OliRSR(InstrumentRSR): - """Class for Landsat OLI RSR.""" - - def __init__(self, bandname, platform_name): - """Read the Landsat OLI relative spectral responses for all channels.""" - super(OliRSR, self).__init__(bandname, platform_name) - - self.instrument = 'oli' - self._get_options_from_config() - - LOG.debug("Filename: %s", str(self.path)) - if os.path.exists(self.path): - self._load() - else: - raise IOError("Couldn't find an existing file for this band: " + - str(self.bandname)) - - def _load(self, scale=0.001): - """Load the Landsat OLI relative spectral responses.""" - with open_workbook(self.path) as wb_: - for sheet in wb_.sheets(): - if sheet.name in ['Plot of AllBands', ]: - continue - ch_name = OLI_BAND_NAMES.get(sheet.name.strip()) - - if ch_name != self.bandname: - continue - - wvl = sheet.col_values(0, 2) - resp = sheet.col_values(1, 2) - - self.rsr = {'wavelength': np.array(wvl) / 1000., - 'response': np.array(resp)} - break - - -if __name__ == "__main__": - bands = OLI_BAND_NAMES.values() - bands.sort() - for platform_name in ['Landsat-8', ]: - tohdf5(OliRSR, platform_name, bands) diff --git a/rsr_convert_scripts/oli_tirs_reader.py b/rsr_convert_scripts/oli_tirs_reader.py new file mode 100644 index 00000000..686883b3 --- /dev/null +++ b/rsr_convert_scripts/oli_tirs_reader.py @@ -0,0 +1,124 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# +# Copyright (c) 2017-2024 Pytroll developers +# +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + +"""Landsat-8/9 OLI/TIRS reader. + +This reader generates spectral responses for OLI and TIRS instruments aboard Landsat-8 and -9. +We assume that the instruments are one combined instrument from the user perspective, called `oli_tirs` rather +than generating RSRs for the two instruments separately. +The original spectral response data can be found at the links below. +Landsat-8/OLI: + = https://landsat.gsfc.nasa.gov/wp-content/uploads/2014/09/Ball_BA_RSR.v1.2.xlsx +Landsat-9/OLI-2: + = https://landsat.gsfc.nasa.gov/wp-content/uploads/2024/03/L9_OLI2_Ball_BA_RSR.v2-1.xlsx +Landsat-8/TIRS: + = https://landsat.gsfc.nasa.gov/wp-content/uploads/2013/06/TIRS_Relative_Spectral_Responses.BA_.v1.xlsx +Landsat-9/TIRS-2: + = https://landsat.gsfc.nasa.gov/wp-content/uploads/2021-10/L9_TIRS2_Relative_Spectral_Responses.BA.v1.0.xlsx +""" + +import logging +from pathlib import Path + +import numpy as np +import pandas as pd + +from pyspectral.raw_reader import InstrumentRSR +from pyspectral.utils import convert2hdf5 as tohdf5 + +LOG = logging.getLogger(__name__) + +OLI_BAND_NAMES = {"B1": "CoastalAerosol", + "B2": "Blue", + "B3": "Green", + "B4": "Red", + "B5": "NIR", + "B6": "Cirrus", + "B7": "SWIR1", + "B8": "SWIR2", + "B9": "Pan"} + +TIRS_SHEETNAMES_L8 = {"B10": "TIRS BA RSR", + "B11": "TIRS BA RSR"} +TIRS_BAND_NAMES_L8 = {"B10": "TIRS1 10.8um band average", + "B11": "TIRS2 12.0um band average"} + +TIRS_SHEETNAMES_L9 = {"B10": "TIRS Band 10 BA RSR", + "B11": "TIRS Band 11 BA RSR"} +TIRS_BAND_NAMES_L9 = {"B10": "Band 10 Band=Average RSR", + "B11": "Band 11 Band-Average RSR"} + + +class OliRSR(InstrumentRSR): + """Class for Landsat OLI RSR.""" + + def __init__(self, bandname, platform_name): + """Read the Landsat OLI relative spectral responses for all channels.""" + super(OliRSR, self).__init__(bandname, platform_name) + self.instrument = "oli_tirs" + self._get_options_from_config() + self.band = bandname + opts = self.options[f"{platform_name}-{self.instrument}"] + if bandname in OLI_BAND_NAMES: + self.path = Path(opts["path"]) / Path(opts["oli"]) + elif bandname in TIRS_BAND_NAMES_L8: + self.path = Path(opts["path"]) / Path(opts["tirs"]) + else: + raise ValueError(f"Unknown band name: {bandname}") + + LOG.debug(f"Filename: {self.path}") + if self.path.exists(): + self._load() + else: + raise IOError("Couldn't find an existing file for this band: " + + str(self.bandname)) + + def _load(self, scale=0.001): + """Load the Landsat OLI relative spectral responses.""" + if self.band in OLI_BAND_NAMES: + df = pd.read_excel(self.path, engine="openpyxl", sheet_name=OLI_BAND_NAMES[self.band]) + wvl = np.array(df["Wavelength"]) / 1000. + resp = np.array(df["BA RSR [watts]"]) + else: + if self.platform_name == "Landsat-8": + sheet_name = TIRS_SHEETNAMES_L8[self.band] + band_name = TIRS_BAND_NAMES_L8[self.band] + elif self.platform_name == "Landsat-9": + sheet_name = TIRS_SHEETNAMES_L9[self.band] + band_name = TIRS_BAND_NAMES_L9[self.band] + else: + raise ValueError(f"Unknown platform: {self.platform_name}") + df = pd.read_excel(self.path, engine="openpyxl", sheet_name=sheet_name) + + wvl = np.array(df["wavelength [um]"]) / 1000. + resp = np.array(df[band_name]) + + # Cut unneeded points + pts = np.argwhere(resp > 0.002) + wvl = np.squeeze(wvl[pts]) + resp = np.squeeze(resp[pts]) + + self.rsr = {"wavelength": wvl, + "response": resp} + + +if __name__ == "__main__": + bands = sorted(OLI_BAND_NAMES.keys()) + sorted(TIRS_BAND_NAMES_L8.keys()) + for platform_name in ["Landsat-8", "Landsat-9"]: + tohdf5(OliRSR, platform_name, bands) diff --git a/setup.py b/setup.py index b851fa2e..f04045d3 100644 --- a/setup.py +++ b/setup.py @@ -1,7 +1,7 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -# Copyright (c) 2013-2022 Pytroll +# Copyright (c) 2013-2024 Pytroll # # # This program is free software: you can redistribute it and/or modify @@ -71,6 +71,7 @@ 'matplotlib': ['matplotlib'], 'pandas': ['pandas'], 'tqdm': ['tqdm'], + 'openpyxl': ['openpyxl'], 'test': test_requires, 'dask': dask_extra}, scripts=['bin/plot_rsr.py', 'bin/composite_rsr_plot.py',