Source code for instruments.galileo

"""Module for reading raw Galileo cloud radar data."""

import os
from tempfile import NamedTemporaryFile, TemporaryDirectory

import numpy as np

from cloudnetpy import concat_lib, output, utils
from cloudnetpy.instruments.instruments import GALILEO
from cloudnetpy.instruments.nc_radar import ChilboltonRadar
from cloudnetpy.metadata import MetaData


[docs] def galileo2nc( raw_files: str, output_file: str, site_meta: dict, uuid: str | None = None, date: str | None = None, ) -> str: """Converts 'Galileo' cloud radar data into Cloudnet Level 1b netCDF file. Args: raw_files: Input file name or folder containing multiple input files. output_file: Output filename. site_meta: Dictionary containing information about the site. Required key value pair is `name`. Optional are `latitude`, `longitude`, `altitude` and `snr_limit` (default = 3). uuid: Set specific UUID for the file. date: Expected date as YYYY-MM-DD of all profiles in the file. Returns: UUID of the generated file. Raises: ValidTimeStampError: No valid timestamps found. Examples: >>> from cloudnetpy.instruments import galileo2nc >>> site_meta = {'name': 'Chilbolton'} >>> galileo2nc('raw_radar.nc', 'radar.nc', site_meta) >>> galileo2nc('/one/day/of/galileo/files/', 'radar.nc', site_meta) """ keymap = { "ZED_HC": "Zh", "VEL_HC": "v", "SPW_HC": "width", "LDR_HC": "ldr", "SNR_HC": "SNR", "elevation": "elevation", "azimuth": "azimuth_angle", "height": "altitude", "antenna_diameter": "antenna_diameter", "beamwidthV": "beamwidthV", "beamwidthH": "beamwidthH", } with TemporaryDirectory() as temp_dir: if os.path.isdir(raw_files): with NamedTemporaryFile( dir=temp_dir, suffix=".nc", delete=False, ) as temp_file: nc_filename = temp_file.name valid_filenames = utils.get_sorted_filenames(raw_files, ".nc") valid_filenames = utils.get_files_with_variables( valid_filenames, ["time", "ZED_HC"] ) valid_filenames = utils.get_files_with_common_range(valid_filenames) variables = list(keymap.keys()) concat_lib.concatenate_files( valid_filenames, nc_filename, variables=variables, ) else: nc_filename = raw_files with Galileo(nc_filename, site_meta) as galileo: galileo.init_data(keymap) galileo.add_time_and_range() if date is not None: galileo.check_date(date) galileo.sort_timestamps() galileo.remove_duplicate_timestamps() snr_limit = site_meta.get("snr_limit", 3) galileo.screen_by_snr(snr_limit=snr_limit) galileo.mask_clutter() galileo.mask_invalid_data() galileo.add_radar_specific_variables() galileo.add_nyquist_velocity(keymap) galileo.add_site_geolocation() valid_indices = galileo.add_zenith_and_azimuth_angles( elevation_threshold=1, elevation_diff_threshold=0.1, azimuth_diff_threshold=0.1, ) galileo.screen_time_indices(valid_indices) galileo.add_height() galileo.test_if_all_masked() attributes = output.add_time_attribute(ATTRIBUTES, galileo.date) output.update_attributes(galileo.data, attributes) return output.save_level1b(galileo, output_file, uuid)
class Galileo(ChilboltonRadar): """Class for Galileo raw radar data. Child of ChilboltonRadar(). Args: full_path: Filename of a daily Galileo .nc NetCDF file. site_meta: Site properties in a dictionary. Required keys are: `name`. """ def __init__(self, full_path: str, site_meta: dict): super().__init__(full_path, site_meta) self.date = self._init_date() self.instrument = GALILEO def mask_clutter(self) -> None: """Masks clutter.""" # Only strong Z values are valid n_low_gates = 15 ind = np.where(self.data["Zh"][:, :n_low_gates] < -15) and np.where( self.data["ldr"][:, :n_low_gates] > -5, ) self.data["v"].mask_indices(ind) ATTRIBUTES = { "antenna_diameter": MetaData(long_name="Antenna diameter", units="m"), "beamwidthV": MetaData(long_name="Vertical angular beamwidth", units="degree"), "beamwidthH": MetaData(long_name="Horizontal angular beamwidth", units="degree"), }