forked from pytroll/satpy
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request pytroll#2905 from pdebuyl/mcd12q1_draft
Mcd12q1 draft
- Loading branch information
Showing
6 changed files
with
329 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
reader: | ||
name: mcd12q1 | ||
short_name: MCD12Q1 | ||
long_name: MODIS Level 3 (mcd12Q1) data in HDF-EOS format | ||
description: MODIS HDF-EOS MCD12Q1 L3 Reader | ||
status: Beta | ||
supports_fsspec: false | ||
reader: !!python/name:satpy.readers.yaml_reader.FileYAMLReader | ||
sensors: [modis] | ||
|
||
file_types: | ||
modis_mcd12q1_hdf_eos: | ||
file_patterns: ['MCD12Q1.A{start_time:%Y%j}.{tile_id}.{collection:03d}.{production_time:%Y%j%H%M%S}.hdf'] | ||
file_reader: !!python/name:satpy.readers.mcd12q1.MCD12Q1HDFFileHandler | ||
|
||
datasets: | ||
LC_Type1: | ||
name: LC_Type1 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
|
||
LC_Type2: | ||
name: LC_Type2 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Type3: | ||
name: LC_Type3 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Type4: | ||
name: LC_Type4 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Type5: | ||
name: LC_Type5 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Prop1: | ||
name: LC_Prop1 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Prop2: | ||
name: LC_Prop2 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos | ||
LC_Prop3: | ||
name: LC_Prop3 | ||
resolution: 500 | ||
file_type: modis_mcd12q1_hdf_eos |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,144 @@ | ||
#!/usr/bin/env python | ||
# -*- coding: utf-8 -*- | ||
# Copyright (c) 2024 Satpy developers | ||
# | ||
# 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/>. | ||
"""MCD12Q1 hdf-eos format reader. | ||
Introduction | ||
------------ | ||
The ``mcd12q1`` reader reads MCD12Q1 products in HDF-EOS format. | ||
The 500m product is provided on a sinusoidal grid. | ||
Reference documents and links: | ||
- MODIS land products grid: https://modis-land.gsfc.nasa.gov/MODLAND_grid.html | ||
- User guide: https://lpdaac.usgs.gov/documents/101/MCD12_User_Guide_V6.pdf | ||
- MCD12Q1 v061: MODIS/Terra+Aqua Land Cover Type Yearly L3 Global 500 m SIN Grid | ||
The reader has been tested with: | ||
- MCD12Q1: Land cover data. | ||
To get a list of the available datasets for a given file refer to the "Load data" section in :doc:`../reading`. | ||
""" | ||
import logging | ||
from typing import Iterable | ||
|
||
from pyresample import geometry | ||
|
||
from satpy.readers.hdfeos_base import HDFEOSBaseFileReader | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
class MCD12Q1HDFFileHandler(HDFEOSBaseFileReader): | ||
"""File handler for MCD12Q1 HDF-EOS 500m granules.""" | ||
def available_datasets(self, configured_datasets=None): | ||
"""Automatically determine datasets provided by this file.""" | ||
# Initialise set of variable names to carry through code | ||
handled_var_names = set() | ||
|
||
ds_dict = self.sd.datasets() | ||
|
||
for is_avail, ds_info in (configured_datasets or []): | ||
file_key = ds_info.get("file_key", ds_info["name"]) | ||
# we must add all variables here even if another file handler has | ||
# claimed the variable. It could be another instance of this file | ||
# type, and we don't want to add that variable dynamically if the | ||
# other file handler defined it by the YAML definition. | ||
handled_var_names.add(file_key) | ||
if is_avail is not None: | ||
# some other file handler said it has this dataset | ||
# we don't know any more information than the previous | ||
# file handler so let's yield early | ||
yield is_avail, ds_info | ||
continue | ||
if self.file_type_matches(ds_info["file_type"]) is None: | ||
# this is not the file type for this dataset | ||
yield None, ds_info | ||
continue | ||
yield file_key in ds_dict.keys(), ds_info | ||
|
||
yield from self._dynamic_variables_from_file(handled_var_names) | ||
|
||
def _dynamic_variables_from_file(self, handled_var_names: set) -> Iterable[tuple[bool, dict]]: | ||
res = self._get_res() | ||
for var_name in self.sd.datasets().keys(): | ||
if var_name in handled_var_names: | ||
# skip variables that YAML had configured | ||
continue | ||
common = {"file_type": "mcd12q1_500m_hdf", | ||
"resolution": res, | ||
"name": var_name} | ||
yield True, common | ||
|
||
|
||
def _get_res(self): | ||
"""Compute the resolution from the file metadata.""" | ||
gridname = self.metadata["GridStructure"]["GRID_1"]["GridName"] | ||
if "MCD12Q1" not in gridname: | ||
raise ValueError("Only MCD12Q1 grids are supported") | ||
|
||
resolution_string = self.metadata["ARCHIVEDMETADATA"]["NADIRDATARESOLUTION"]["VALUE"] | ||
if resolution_string[-1] == "m": | ||
return int(resolution_string.removesuffix("m")) | ||
else: | ||
raise ValueError("Cannot parse resolution of MCD12Q1 grid") | ||
|
||
|
||
def get_dataset(self, dataset_id, dataset_info): | ||
"""Get DataArray for specified dataset.""" | ||
dataset_name = dataset_id["name"] | ||
# xxx | ||
dataset = self.load_dataset(dataset_name, dataset_info.pop("category", False)) | ||
|
||
self._add_satpy_metadata(dataset_id, dataset) | ||
|
||
return dataset | ||
|
||
def _get_area_extent(self): | ||
"""Get the grid properties.""" | ||
# Now compute the data extent | ||
upperleft = self.metadata["GridStructure"]["GRID_1"]["UpperLeftPointMtrs"] | ||
lowerright = self.metadata["GridStructure"]["GRID_1"]["LowerRightMtrs"] | ||
|
||
return upperleft[0], lowerright[1], lowerright[0], upperleft[1] | ||
|
||
def get_area_def(self, dsid): | ||
"""Get the area definition. | ||
This is fixed, but not defined in the file. So we must | ||
generate it ourselves with some assumptions. | ||
The proj_param string comes from https://lpdaac.usgs.gov/documents/101/MCD12_User_Guide_V6.pdf | ||
""" | ||
proj_param = "proj=sinu +a=6371007.181 +b=6371007.181 +units=m" | ||
|
||
# Get the size of the dataset | ||
nrows = self.metadata["GridStructure"]["GRID_1"]["YDim"] | ||
ncols = self.metadata["GridStructure"]["GRID_1"]["XDim"] | ||
|
||
# Construct the area definition | ||
area = geometry.AreaDefinition("sinusoidal_modis", | ||
"Tiled sinusoidal L3 MODIS area", | ||
"sinusoidal", | ||
proj_param, | ||
ncols, | ||
nrows, | ||
self._get_area_extent()) | ||
|
||
return area |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.