Source code for satpy.readers.msi_ec_l1c_h5

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Copyright (c) 2024.
#
# This file is part of satpy.
#
# satpy 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.
#
# satpy 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
# satpy.  If not, see <http://www.gnu.org/licenses/>.
"""A reader for Level 1C data produced by the MSI instrument aboard EarthCARE.

The MSI L1 files are described in the document "MSI L1 Product Definitions":
https://earthcarehandbook.earth.esa.int/documents/d/earthcare-data-handbook/msi-l1-product-definitions-volume-a-nominal-products

"""
import logging

import numpy as np

from satpy.readers.core.hdf5 import HDF5FileHandler
from satpy.utils import get_legacy_chunk_size

LOG = logging.getLogger(__name__)
CHUNK_SIZE = get_legacy_chunk_size()


[docs] class MSIECL1CFileHandler(HDF5FileHandler): """File handler for MSI L1c H5 files."""
[docs] def __init__(self, filename, filename_info, filetype_info): """Init the file handler.""" super(MSIECL1CFileHandler, self).__init__(filename, filename_info, filetype_info)
@property def end_time(self): """Get end time.""" return self.filename_info["end_time"] @property def start_time(self): """Get start time.""" return self.filename_info["start_time"]
[docs] def get_dataset(self, dataset_id, ds_info): """Load data variable and metadata and calibrate if needed.""" file_key = ds_info.get("file_key", dataset_id["name"]) data = self[file_key] # Band data is stored in a 3d array (Band x Along_Track x Across_Track). # This means we have to select a single 2d array for a given band, # and the correct index is given in the reader YAML. band_index = ds_info.get("band_index") if band_index is not None: data = data[band_index] # The dataset has incorrect units attribute (due to storing multiple types). Fix it here. data.attrs.update(ds_info) data.attrs.update({"units": ds_info.get("units")}) if "fill_value" in ds_info: ds_fill = ds_info["fill_value"] data = data.where(data != ds_fill) # VIS/SWIR data can have radiance or reflectance calibration. if "calibration" in ds_info: cal_type = ds_info["calibration"].name data = self._calibrate(data, band_index, cal_type) # Rename dimensions, as some have incorrect names (notably the pixel value data). if "dim_1" in data.dims: data = data.rename({"dim_1": "y", "dim_2": "x"}) return data
[docs] def _calibrate(self, chan_data, band_index, cal_type): """Calibrate the data.""" if cal_type == "reflectance": sol_irrad = self["ScienceData/solar_spectral_irradiance"] chan_data.data = chan_data.data * 100. * np.pi / sol_irrad[band_index].data.reshape((1, sol_irrad.shape[1])) return chan_data return chan_data