Source code for icesat2_toolkit.io.ATL03

#!/usr/bin/env python
u"""
ATL03.py (05/2024)
Read ICESat-2 ATL03 and ATL09 data files to calculate average segment surfaces
    ATL03 datasets: Global Geolocated Photons
    ATL09 datasets: Atmospheric Characteristics

PYTHON DEPENDENCIES:
    numpy: Scientific Computing Tools For Python
        https://numpy.org
        https://numpy.org/doc/stable/user/numpy-for-matlab-users.html
    scipy: Scientific Tools for Python
        https://docs.scipy.org/doc/
    h5py: Python interface for Hierarchal Data Format 5 (HDF5)
        https://www.h5py.org/

UPDATE HISTORY:
    Updated 05/2024: use wrapper to importlib for optional dependencies
        check if input filename is an open HDF5 file object
    Updated 03/2024: use pathlib to define and operate on paths
    Updated 11/2023: drop DIMENSION_LIST, CLASS and NAME attributes
    Updated 12/2022: place some imports behind try/except statements
        refactor ICESat-2 data product read programs under io
    Updated 04/2022: updated docstrings to numpy documentation format
    Updated 10/2021: using python logging for handling verbose output
    Updated 02/2021: add check if input streaming from bytes
    Updated 10/2020: add small function to find valid beam groups
    Updated 09/2020: map ATL09 to ATL03 using delta times
    Updated 07/2020: added function docstrings
    Updated 06/2020: add additional beam check within heights groups
    Updated 11/2019: create attribute dictionaries but don't fill if False
    Updated 09/2019: add functions for reading main and beam level variables
    Updated 03/2019: extract a set of ATL09 parameters for each ATL03 segment_ID
    Updated 02/2019: continued writing read program with first ATL03 release
    Written 05/2017
"""
from __future__ import print_function, division

import io
import re
import logging
import pathlib
import warnings
import numpy as np
import scipy.interpolate
from icesat2_toolkit.utilities import import_dependency

# attempt imports
h5py = import_dependency('h5py')

# PURPOSE: read ICESat-2 ATL03 HDF5 data files
[docs]def read_granule(FILENAME, ATTRIBUTES=False, KEEP=False, **kwargs): """ Reads ICESat-2 ATL03 Global Geolocated Photons data files Parameters ---------- FILENAME: str full path to ATL03 file ATTRIBUTES: bool, default False read file, group and variable attributes KEEP: bool, default False keep file object open Returns ------- IS2_atl03_mds: dict ATL03 variables IS2_atl03_attrs: dict ATL03 attributes IS2_atl03_beams: list valid ICESat-2 beams within ATL03 file """ # Open the HDF5 file for reading if isinstance(FILENAME, io.IOBase): fileID = h5py.File(FILENAME, 'r') elif isinstance(FILENAME, h5py.File): fileID = FILENAME else: FILENAME = pathlib.Path(FILENAME).expanduser().absolute() fileID = h5py.File(FILENAME, 'r') # Output HDF5 file information logging.info(fileID.filename) logging.info(list(fileID.keys())) # allocate python dictionaries for ICESat-2 ATL03 variables and attributes IS2_atl03_mds = {} IS2_atl03_attrs = {} # read each input beam within the file IS2_atl03_beams = [] for gtx in [k for k in fileID.keys() if bool(re.match(r'gt\d[lr]',k))]: # check if subsetted beam contains data # check in both the geolocation and heights groups try: fileID[gtx]['geolocation']['segment_id'] fileID[gtx]['heights']['delta_time'] except KeyError: pass else: IS2_atl03_beams.append(gtx) # for each included beam for gtx in IS2_atl03_beams: # get each HDF5 variable IS2_atl03_mds[gtx] = {} IS2_atl03_mds[gtx]['heights'] = {} IS2_atl03_mds[gtx]['geolocation'] = {} IS2_atl03_mds[gtx]['bckgrd_atlas'] = {} IS2_atl03_mds[gtx]['geophys_corr'] = {} # ICESat-2 Measurement Group for key,val in fileID[gtx]['heights'].items(): IS2_atl03_mds[gtx]['heights'][key] = val[:] # ICESat-2 Geolocation Group for key,val in fileID[gtx]['geolocation'].items(): IS2_atl03_mds[gtx]['geolocation'][key] = val[:] # ICESat-2 Background Photon Rate Group for key,val in fileID[gtx]['bckgrd_atlas'].items(): IS2_atl03_mds[gtx]['bckgrd_atlas'][key] = val[:] # ICESat-2 Geophysical Corrections Group: Values for tides (ocean, # solid earth, pole, load, and equilibrium), inverted barometer (IB) # effects, and range corrections for tropospheric delays for key,val in fileID[gtx]['geophys_corr'].items(): IS2_atl03_mds[gtx]['geophys_corr'][key] = val[:] # Getting attributes of included variables if ATTRIBUTES: # Getting attributes of IS2_atl03_mds beam variables IS2_atl03_attrs[gtx] = {} IS2_atl03_attrs[gtx]['heights'] = {} IS2_atl03_attrs[gtx]['geolocation'] = {} IS2_atl03_attrs[gtx]['bckgrd_atlas'] = {} IS2_atl03_attrs[gtx]['geophys_corr'] = {} # Global Group Attributes for att_name,att_val in fileID[gtx].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[gtx][att_name] = att_val # ICESat-2 Measurement Group for key,val in fileID[gtx]['heights'].items(): IS2_atl03_attrs[gtx]['heights'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[gtx]['heights'][key][att_name]=att_val # ICESat-2 Geolocation Group for key,val in fileID[gtx]['geolocation'].items(): IS2_atl03_attrs[gtx]['geolocation'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[gtx]['geolocation'][key][att_name]=att_val # ICESat-2 Background Photon Rate Group for key,val in fileID[gtx]['bckgrd_atlas'].items(): IS2_atl03_attrs[gtx]['bckgrd_atlas'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[gtx]['bckgrd_atlas'][key][att_name]=att_val # ICESat-2 Geophysical Corrections Group for key,val in fileID[gtx]['geophys_corr'].items(): IS2_atl03_attrs[gtx]['geophys_corr'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[gtx]['geophys_corr'][key][att_name]=att_val # ICESat-2 spacecraft orientation at time IS2_atl03_mds['orbit_info'] = {} IS2_atl03_attrs['orbit_info'] = {} for key,val in fileID['orbit_info'].items(): IS2_atl03_mds['orbit_info'][key] = val[:] # Getting attributes of group and included variables if ATTRIBUTES: # Global Group Attributes for att_name,att_val in fileID['orbit_info'].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['orbit_info'][att_name] = att_val # Variable Attributes IS2_atl03_attrs['orbit_info'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['orbit_info'][key][att_name] = att_val # information ancillary to the data product # number of GPS seconds between the GPS epoch (1980-01-06T00:00:00Z UTC) # and ATLAS Standard Data Product (SDP) epoch (2018-01-01T00:00:00Z UTC) # Add this value to delta time parameters to compute full gps_seconds # could alternatively use the Julian day of the ATLAS SDP epoch: 2458119.5 # and add leap seconds since 2018-01-01T00:00:00Z UTC (ATLAS SDP epoch) IS2_atl03_mds['ancillary_data'] = {} IS2_atl03_attrs['ancillary_data'] = {} ancillary_keys = ['atlas_sdp_gps_epoch','data_end_utc','data_start_utc', 'end_cycle','end_geoseg','end_gpssow','end_gpsweek','end_orbit', 'end_region','end_rgt','granule_end_utc','granule_start_utc','release', 'start_cycle','start_geoseg','start_gpssow','start_gpsweek', 'start_orbit','start_region','start_rgt','version'] for key in ancillary_keys: # get each HDF5 variable IS2_atl03_mds['ancillary_data'][key] = fileID['ancillary_data'][key][:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs['ancillary_data'][key] = {} for att_name,att_val in fileID['ancillary_data'][key].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['ancillary_data'][key][att_name] = att_val # transmit-echo-path (tep) parameters IS2_atl03_mds['ancillary_data']['tep'] = {} IS2_atl03_attrs['ancillary_data']['tep'] = {} for key,val in fileID['ancillary_data']['tep'].items(): # get each HDF5 variable IS2_atl03_mds['ancillary_data']['tep'][key] = val[:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs['ancillary_data']['tep'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['ancillary_data']['tep'][key][att_name] = att_val # channel dead time and first photon bias derived from ATLAS calibration cal1,cal2 = ('ancillary_data','calibrations') for var in ['dead_time','first_photon_bias']: IS2_atl03_mds[cal1][var] = {} IS2_atl03_attrs[cal1][var] = {} for key,val in fileID[cal1][cal2][var].items(): # get each HDF5 variable if isinstance(val, h5py.Dataset): IS2_atl03_mds[cal1][var][key] = val[:] elif isinstance(val, h5py.Group): IS2_atl03_mds[cal1][var][key] = {} for k,v in val.items(): IS2_atl03_mds[cal1][var][key][k] = v[:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs[cal1][var][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[cal1][var][key][att_name] = att_val if isinstance(val, h5py.Group): for k,v in val.items(): IS2_atl03_attrs[cal1][var][key][k] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[cal1][var][key][k][att_name]=att_val # get ATLAS impulse response variables for the transmitter echo path (TEP) tep1,tep2 = ('atlas_impulse_response','tep_histogram') IS2_atl03_mds[tep1] = {} IS2_atl03_attrs[tep1] = {} for pce in ['pce1_spot1','pce2_spot3']: IS2_atl03_mds[tep1][pce] = {tep2:{}} IS2_atl03_attrs[tep1][pce] = {tep2:{}} # for each TEP variable for key,val in fileID[tep1][pce][tep2].items(): IS2_atl03_mds[tep1][pce][tep2][key] = val[:] # Getting attributes of included variables if ATTRIBUTES: # Global Group Attributes for att_name,att_val in fileID[tep1][pce][tep2].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[tep1][pce][tep2][att_name] = att_val # Variable Attributes IS2_atl03_attrs[tep1][pce][tep2][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[tep1][pce][tep2][key][att_name] = att_val # Global File Attributes if ATTRIBUTES: for att_name,att_val in fileID.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[att_name] = att_val # Closing the HDF5 file if not KEEP: fileID.close() # Return the datasets and variables return (IS2_atl03_mds, IS2_atl03_attrs, IS2_atl03_beams)
# PURPOSE: read ICESat-2 ATL09 HDF5 data file for specific variables
[docs]def interpolate_ATL09(FILENAME, pfl, dtime, ATTRIBUTES=True, KEEP=False, **kwargs): """ Reads ICESat-2 ATL09 Atmospheric Characteristics data files and interpolates a subset of variables to ATL03 segment lengths Parameters ---------- FILENAME: str full path to ATL03 file pfl: str profile for a given beam dtime: float ATL03 reference photon delta_time ATTRIBUTES: bool, default True read file, group and variable attributes KEEP: bool, default False keep file object open Returns ------- IS2_atl09_mds: dict ATL09 variables remapped to ATL03 segments IS2_atl09_attrs: dict ATL09 attributes """ # Open the HDF5 file for reading if isinstance(FILENAME, io.IOBase): fileID = h5py.File(FILENAME, 'r') elif isinstance(FILENAME, h5py.File): fileID = FILENAME else: FILENAME = pathlib.Path(FILENAME).expanduser().absolute() fileID = h5py.File(FILENAME, 'r') # allocate python dictionaries for ICESat-2 ATL09 variables and attributes IS2_atl09_mds = {} IS2_atl09_attrs = {} # read profile reported for the ATLAS strong beams within the file IS2_atl09_mds[pfl] = dict(high_rate={}) # extract delta_time for mapping ATL09 atmospheric parameters to ATL03 delta_time = fileID[pfl]['high_rate']['delta_time'][:] # Calibrated Attenuated Backscatter at 25 hz high_rate_keys = ['aclr_true','bsnow_con','bsnow_dens','bsnow_h', 'bsnow_h_dens','bsnow_od','bsnow_psc','cloud_flag_asr','cloud_flag_atm', 'cloud_fold_flag','column_od_asr','column_od_asr_qf','msw_flag', 'snow_ice','solar_azimuth','solar_elevation','surf_refl_true'] # extract variables of interest and map to ATL03 segments for key in high_rate_keys: val = np.copy(fileID[pfl]['high_rate'][key][:]) fint = scipy.interpolate.interp1d(delta_time, val, kind='nearest', fill_value='extrapolate') IS2_atl09_mds[pfl]['high_rate'][key] = fint(dtime).astype(val.dtype) # Getting attributes of included variables if ATTRIBUTES: # Getting attributes of IS2_atl09_mds profile variables IS2_atl09_attrs[pfl] = dict(high_rate={}) # Global Group Attributes for att_name,att_val in fileID[pfl].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl09_attrs[pfl][att_name] = att_val # Variable Attributes for key in high_rate_keys: IS2_atl09_attrs[pfl]['high_rate'][key] = {} for att_name,att_val in fileID[pfl]['high_rate'][key].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl09_attrs[pfl]['high_rate'][key][att_name] = att_val # Global File Attributes if ATTRIBUTES: for att_name,att_val in fileID.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl09_attrs[att_name] = att_val # Closing the HDF5 file if not KEEP: fileID.close() # Return the datasets and variables return (IS2_atl09_mds, IS2_atl09_attrs)
# PURPOSE: find valid beam groups within ICESat-2 ATL03 HDF5 data files
[docs]def find_beams(FILENAME, KEEP=False ,**kwargs): """ Find valid beam groups within ICESat-2 ATL03 Global Geolocated Photons data files Parameters ---------- FILENAME: str full path to ATL03 file KEEP: bool, default False keep file object open Returns ------- IS2_atl03_beams: list valid ICESat-2 beams within ATL03 file """ # Open the HDF5 file for reading if isinstance(FILENAME, io.IOBase): fileID = h5py.File(FILENAME, 'r') elif isinstance(FILENAME, h5py.File): fileID = FILENAME else: FILENAME = pathlib.Path(FILENAME).expanduser().absolute() fileID = h5py.File(FILENAME, 'r') # output list of beams IS2_atl03_beams = [] # read each input beam within the file for gtx in [k for k in fileID.keys() if bool(re.match(r'gt\d[lr]',k))]: # check if subsetted beam contains data # check in both the geolocation and heights groups try: fileID[gtx]['geolocation']['segment_id'] fileID[gtx]['heights']['delta_time'] except KeyError: pass else: IS2_atl03_beams.append(gtx) # Closing the HDF5 file if not KEEP: fileID.close() # return the list of beams return IS2_atl03_beams
# PURPOSE: read ICESat-2 ATL03 HDF5 data files for main level variables
[docs]def read_main(FILENAME, ATTRIBUTES=False, KEEP=False, **kwargs): """ Reads ICESat-2 ATL03 Global Geolocated Photons data files for only the main-level variables and not the beam-level data Parameters ---------- FILENAME: str full path to ATL03 file ATTRIBUTES: bool, default False read file, group and variable attributes KEEP: bool, default False keep file object open Returns ------- IS2_atl03_mds: dict ATL03 main-level variables IS2_atl03_attrs: dict ATL03 main-level attributes IS2_atl03_beams: list valid ICESat-2 beams within ATL03 file """ # Open the HDF5 file for reading if isinstance(FILENAME, io.IOBase): fileID = h5py.File(FILENAME, 'r') elif isinstance(FILENAME, h5py.File): fileID = FILENAME else: FILENAME = pathlib.Path(FILENAME).expanduser().absolute() fileID = h5py.File(FILENAME, 'r') # Output HDF5 file information logging.info(fileID.filename) logging.info(list(fileID.keys())) # allocate python dictionaries for ICESat-2 ATL03 variables and attributes IS2_atl03_mds = {} IS2_atl03_attrs = {} # read each input beam within the file IS2_atl03_beams = [] for gtx in [k for k in fileID.keys() if bool(re.match(r'gt\d[lr]',k))]: # check if subsetted beam contains data # check in both the geolocation and heights groups try: fileID[gtx]['geolocation']['segment_id'] fileID[gtx]['heights']['delta_time'] except KeyError: pass else: IS2_atl03_beams.append(gtx) # ICESat-2 spacecraft orientation at time IS2_atl03_mds['orbit_info'] = {} IS2_atl03_attrs['orbit_info'] = {} for key,val in fileID['orbit_info'].items(): IS2_atl03_mds['orbit_info'][key] = val[:] # Getting attributes of group and included variables if ATTRIBUTES: # Global Group Attributes for att_name,att_val in fileID['orbit_info'].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['orbit_info'][att_name] = att_val # Variable Attributes IS2_atl03_attrs['orbit_info'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['orbit_info'][key][att_name] = att_val # information ancillary to the data product # number of GPS seconds between the GPS epoch (1980-01-06T00:00:00Z UTC) # and ATLAS Standard Data Product (SDP) epoch (2018-01-01T00:00:00Z UTC) # Add this value to delta time parameters to compute full gps_seconds # could alternatively use the Julian day of the ATLAS SDP epoch: 2458119.5 # and add leap seconds since 2018-01-01T00:00:00Z UTC (ATLAS SDP epoch) IS2_atl03_mds['ancillary_data'] = {} IS2_atl03_attrs['ancillary_data'] = {} ancillary_keys = ['atlas_sdp_gps_epoch','data_end_utc','data_start_utc', 'end_cycle','end_geoseg','end_gpssow','end_gpsweek','end_orbit', 'end_region','end_rgt','granule_end_utc','granule_start_utc','release', 'start_cycle','start_geoseg','start_gpssow','start_gpsweek', 'start_orbit','start_region','start_rgt','version'] for key in ancillary_keys: # get each HDF5 variable IS2_atl03_mds['ancillary_data'][key] = fileID['ancillary_data'][key][:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs['ancillary_data'][key] = {} for att_name,att_val in fileID['ancillary_data'][key].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['ancillary_data'][key][att_name] = att_val # transmit-echo-path (tep) parameters IS2_atl03_mds['ancillary_data']['tep'] = {} IS2_atl03_attrs['ancillary_data']['tep'] = {} for key,val in fileID['ancillary_data']['tep'].items(): # get each HDF5 variable IS2_atl03_mds['ancillary_data']['tep'][key] = val[:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs['ancillary_data']['tep'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['ancillary_data']['tep'][key][att_name] = att_val # channel dead time and first photon bias derived from ATLAS calibration cal1,cal2 = ('ancillary_data','calibrations') for var in ['dead_time','first_photon_bias']: IS2_atl03_mds[cal1][var] = {} IS2_atl03_attrs[cal1][var] = {} for key,val in fileID[cal1][cal2][var].items(): # get each HDF5 variable if isinstance(val, h5py.Dataset): IS2_atl03_mds[cal1][var][key] = val[:] elif isinstance(val, h5py.Group): IS2_atl03_mds[cal1][var][key] = {} for k,v in val.items(): IS2_atl03_mds[cal1][var][key][k] = v[:] # Getting attributes of group and included variables if ATTRIBUTES: # Variable Attributes IS2_atl03_attrs[cal1][var][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[cal1][var][key][att_name] = att_val if isinstance(val, h5py.Group): for k,v in val.items(): IS2_atl03_attrs[cal1][var][key][k] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[cal1][var][key][k][att_name]=att_val # get ATLAS impulse response variables for the transmitter echo path (TEP) tep1,tep2 = ('atlas_impulse_response','tep_histogram') IS2_atl03_mds[tep1] = {} IS2_atl03_attrs[tep1] = {} for pce in ['pce1_spot1','pce2_spot3']: IS2_atl03_mds[tep1][pce] = {tep2:{}} IS2_atl03_attrs[tep1][pce] = {tep2:{}} # for each TEP variable for key,val in fileID[tep1][pce][tep2].items(): IS2_atl03_mds[tep1][pce][tep2][key] = val[:] # Getting attributes of included variables if ATTRIBUTES: # Global Group Attributes for att_name,att_val in fileID[tep1][pce][tep2].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[tep1][pce][tep2][att_name] = att_val # Variable Attributes IS2_atl03_attrs[tep1][pce][tep2][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[tep1][pce][tep2][key][att_name] = att_val # Global File Attributes if ATTRIBUTES: for att_name,att_val in fileID.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[att_name] = att_val # Closing the HDF5 file if not KEEP: fileID.close() # Return the datasets and variables return (IS2_atl03_mds, IS2_atl03_attrs, IS2_atl03_beams)
# PURPOSE: read ICESat-2 ATL03 HDF5 data files for beam variables
[docs]def read_beam(FILENAME, gtx, ATTRIBUTES=False, KEEP=False, **kwargs): """ Reads ICESat-2 ATL03 Global Geolocated Photons data files for a specific beam Parameters ---------- FILENAME: str full path to ATL03 file gtx: str beam name based on ground track and position - ``'gt1l'`` - ``'gt1r'`` - ``'gt2l'`` - ``'gt2r'`` - ``'gt3l'`` - ``'gt3r'`` ATTRIBUTES: bool, default False read file, group and variable attributes KEEP: bool, default False keep file object open Returns ------- IS2_atl03_mds: dict ATL03 beam-level variables IS2_atl03_attrs: dict ATL03 beam-level attributes """ # Open the HDF5 file for reading if isinstance(FILENAME, io.IOBase): fileID = h5py.File(FILENAME, 'r') elif isinstance(FILENAME, h5py.File): fileID = FILENAME else: FILENAME = pathlib.Path(FILENAME).expanduser().absolute() fileID = h5py.File(FILENAME, 'r') # Output HDF5 file information logging.info(fileID.filename) logging.info(list(fileID.keys())) # allocate python dictionaries for ICESat-2 ATL03 variables and attributes IS2_atl03_mds = {} IS2_atl03_attrs = {} # get each HDF5 variable IS2_atl03_mds['heights'] = {} IS2_atl03_mds['geolocation'] = {} IS2_atl03_mds['bckgrd_atlas'] = {} IS2_atl03_mds['geophys_corr'] = {} # ICESat-2 Measurement Group for key,val in fileID[gtx]['heights'].items(): IS2_atl03_mds['heights'][key] = val[:] # ICESat-2 Geolocation Group for key,val in fileID[gtx]['geolocation'].items(): IS2_atl03_mds['geolocation'][key] = val[:] # ICESat-2 Background Photon Rate Group for key,val in fileID[gtx]['bckgrd_atlas'].items(): IS2_atl03_mds['bckgrd_atlas'][key] = val[:] # ICESat-2 Geophysical Corrections Group: Values for tides (ocean, # solid earth, pole, load, and equilibrium), inverted barometer (IB) # effects, and range corrections for tropospheric delays for key,val in fileID[gtx]['geophys_corr'].items(): IS2_atl03_mds['geophys_corr'][key] = val[:] # Getting attributes of included variables if ATTRIBUTES: # Getting attributes of IS2_atl03_mds beam variables IS2_atl03_attrs['heights'] = {} IS2_atl03_attrs['geolocation'] = {} IS2_atl03_attrs['bckgrd_atlas'] = {} IS2_atl03_attrs['geophys_corr'] = {} IS2_atl03_attrs['Atlas_impulse_response'] = {} # Global Group Attributes for att_name,att_val in fileID[gtx].attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs[att_name] = att_val # ICESat-2 Measurement Group for key,val in fileID[gtx]['heights'].items(): IS2_atl03_attrs['heights'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['heights'][key][att_name]=att_val # ICESat-2 Geolocation Group for key,val in fileID[gtx]['geolocation'].items(): IS2_atl03_attrs['geolocation'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['geolocation'][key][att_name]=att_val # ICESat-2 Background Photon Rate Group for key,val in fileID[gtx]['bckgrd_atlas'].items(): IS2_atl03_attrs['bckgrd_atlas'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['bckgrd_atlas'][key][att_name]=att_val # ICESat-2 Geophysical Corrections Group for key,val in fileID[gtx]['geophys_corr'].items(): IS2_atl03_attrs['geophys_corr'][key] = {} for att_name,att_val in val.attrs.items(): if att_name not in ('DIMENSION_LIST','CLASS','NAME'): IS2_atl03_attrs['geophys_corr'][key][att_name]=att_val # Closing the HDF5 file if not KEEP: fileID.close() # Return the datasets and variables return (IS2_atl03_mds, IS2_atl03_attrs)