diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index b7bd824a..804bc82a 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -354,8 +354,8 @@ To create a new reader test, simply add a small, single-station dataset and the | ├── 📁 check_readers | ├── 📁 DISDRODB | ├── 📁 Raw -| ├── 📁 `` : e.g. GPM, ARM, EPFL, ... -| ├── 📁 `` : e.g. PARSIVEL_2007 +| ├── 📁 `` : e.g. GPM, ARM, EPFL, ... +| ├── 📁 `` : e.g. PARSIVEL_2007 | ├── 📁 data | ├── 📁 ``.\* | ├── 📁 issue diff --git a/disdrodb/api/io.py b/disdrodb/api/io.py index e64ec313..566b8006 100644 --- a/disdrodb/api/io.py +++ b/disdrodb/api/io.py @@ -157,7 +157,7 @@ def _get_campaigns_stations(base_dir, product, data_source, campaign_names): def _get_data_source_stations(base_dir, product, data_source): """Return list of available stations for a specific data source. - Returns a tuple (, , ) + Returns a tuple (, , ) """ # Get data source directory data_source_dir = get_disdrodb_path( diff --git a/disdrodb/data_transfer/upload_data.py b/disdrodb/data_transfer/upload_data.py index 0395d454..497fa476 100644 --- a/disdrodb/data_transfer/upload_data.py +++ b/disdrodb/data_transfer/upload_data.py @@ -36,8 +36,8 @@ def click_station_arguments(function: object): Function. """ function = click.argument("station_name", metavar="")(function) - function = click.argument("campaign_name", metavar="")(function) - function = click.argument("data_source", metavar="")(function) + function = click.argument("campaign_name", metavar="")(function) + function = click.argument("data_source", metavar="")(function) return function diff --git a/disdrodb/l0/io.py b/disdrodb/l0/io.py index 871bd1a0..89aaeff6 100644 --- a/disdrodb/l0/io.py +++ b/disdrodb/l0/io.py @@ -81,7 +81,7 @@ def _infer_disdrodb_tree_path(path: str) -> str: ------- str Path inside the DISDRODB archive. - Format: DISDRODB///... + Format: DISDRODB///... """ # Retrieve path elements (os-specific) p = Path(path) @@ -90,7 +90,7 @@ def _infer_disdrodb_tree_path(path: str) -> str: idx_occurrence = np.where(np.isin(list_path_elements, "DISDRODB"))[0] # If DISDRODB directory not present, raise error if len(idx_occurrence) == 0: - raise ValueError(f"The DISDRODB directory is not present in {path}") + raise ValueError(f"The DISDRODB directory is not present in the path '{path}'") # Find the rightermost occurrence right_most_occurrence = max(idx_occurrence) # Define the disdrodb path @@ -98,7 +98,7 @@ def _infer_disdrodb_tree_path(path: str) -> str: return disdrodb_fpath -def __infer_disdrodb_tree_path_components(path: str) -> list: +def _infer_disdrodb_tree_path_components(path: str) -> list: """Return a list with the component of the disdrodb_path. Parameters @@ -110,7 +110,7 @@ def __infer_disdrodb_tree_path_components(path: str) -> list: ------- list Path element inside the DISDRODB archive. - Format: ["DISDRODB", , , ...] + Format: ["DISDRODB", , , ...] """ # Retrieve disdrodb path disdrodb_fpath = _infer_disdrodb_tree_path(path) @@ -135,7 +135,7 @@ def _infer_campaign_name_from_path(path: str) -> str: str Name of the campaign. """ - list_path_elements = __infer_disdrodb_tree_path_components(path) + list_path_elements = _infer_disdrodb_tree_path_components(path) if len(list_path_elements) <= 3: raise ValueError(f"Impossible to determine campaign_name from {path}") campaign_name = list_path_elements[3] @@ -157,13 +157,33 @@ def _infer_data_source_from_path(path: str) -> str: str Name of the data source. """ - list_path_elements = __infer_disdrodb_tree_path_components(path) + list_path_elements = _infer_disdrodb_tree_path_components(path) if len(list_path_elements) <= 2: raise ValueError(f"Impossible to determine data_source from {path}") data_source = list_path_elements[2] return data_source +def _check_campaign_name_is_upper_case(campaign_dir): + """Check the campaign name of campaign_dir is upper case !""" + campaign_name = _infer_campaign_name_from_path(campaign_dir) + upper_campaign_name = campaign_name.upper() + if campaign_name != upper_campaign_name: + msg = f"The campaign directory name {campaign_name} must be uppercase: {upper_campaign_name}" + logger.error(msg) + raise ValueError(msg) + + +def _check_data_source_is_upper_case(campaign_dir): + """Check the data_source name of campaign_dir is upper case !""" + data_source = _infer_data_source_from_path(campaign_dir) + upper_data_source = data_source.upper() + if data_source != upper_data_source: + msg = f"The data_source directory name {data_source} must be defined uppercase: {upper_data_source}" + logger.error(msg) + raise ValueError(msg) + + ####--------------------------------------------------------------------------. #### Directory/Filepaths L0A and L0B products @@ -443,7 +463,7 @@ def get_raw_file_list(raw_dir, station_name, glob_patterns, verbose=False, debug ---------- raw_dir : str Directory of the campaign where to search for files. - Format <..>/DISDRODB/Raw// + Format <..>/DISDRODB/Raw// station_name : str ID of the station verbose : bool, optional @@ -483,7 +503,7 @@ def get_l0a_file_list(processed_dir, station_name, debugging_mode): ---------- processed_dir : str Directory of the campaign where to search for the L0A files. - Format <..>/DISDRODB/Processed// + Format <..>/DISDRODB/Processed// station_name : str ID of the station debugging_mode : bool, optional @@ -572,36 +592,67 @@ def _remove_if_exists(fpath: str, force: bool = False) -> None: logger.info(f"Deleted folder {fpath}") -def _parse_fpath(fpath: str) -> str: - """Ensure fpath does not end with /. +def _create_required_directory(dir_path, dir_name): + """Create directory inside the directory.""" + try: + new_dir_path = os.path.join(dir_path, dir_name) + os.makedirs(new_dir_path, exist_ok=True) + except Exception as e: + msg = f"Can not create folder {dir_name} at {new_dir_path}. Error: {e}" + logger.exception(msg) + raise FileNotFoundError(msg) + + +def _copy_file(src_fpath, dst_fpath): + """Copy a file from a location to another.""" + filename = os.path.basename(src_fpath) + dst_dir = os.path.dirname(dst_fpath) + try: + shutil.copy(src_fpath, dst_fpath) + msg = f"{filename} copied at {dst_fpath}." + logger.info(msg) + except Exception as e: + msg = f"Something went wrong when copying {filename} into {dst_dir}.\n The error is: {e}." + logger.error(msg) + raise ValueError(msg) + + +def remove_path_trailing_slash(path: str) -> str: + """ + Removes a trailing slash or backslash from a file path if it exists. + + This function ensures that the provided file path is normalized by removing + any trailing directory separator characters ('/' or '\\'). This is useful for + maintaining consistency in path strings and for preparing paths for operations + that may not expect a trailing slash. Parameters ---------- - fpath : str - Input file path + path : str + The file path to normalize. Returns ------- str - Output file path + The normalized path without a trailing slash. Raises ------ TypeError - Error il file path not compliant + If the input path is not a string. + + Examples + -------- + >>> remove_trailing_slash("some/path/") + 'some/path' + >>> remove_trailing_slash("another\\path\\") + 'another\\path' """ - - if not isinstance(fpath, str): - raise TypeError("'_parse_fpath' expects a directory/filepath string.") - if fpath[-1] == "/": - print(f"{fpath} should not end with /.") - fpath = fpath[:-1] - - elif fpath[-1] == "\\": - print(f"{fpath} should not end with /.") - fpath = fpath[:-1] - - return fpath + if not isinstance(path, str): + raise TypeError("Input must be a string representing a file path.") + # Remove trailing slash or backslash (if present) + path = path.rstrip("/\\") + return path ####--------------------------------------------------------------------------. @@ -945,7 +996,7 @@ def check_raw_dir(raw_dir: str, verbose: bool = False) -> None: 2. Check that 'raw_dir' follows the expect directory structure 3. Check that each station_name directory contains data 4. Check that for each station_name the mandatory metadata.yml is specified. - 4. Check that for each station_name the mandatory issue.yml is specified. + 5. Check that for each station_name the mandatory issue.yml is specified. Parameters ---------- @@ -960,7 +1011,11 @@ def check_raw_dir(raw_dir: str, verbose: bool = False) -> None: _check_raw_dir_is_a_directory(raw_dir) # Ensure valid path format - raw_dir = _parse_fpath(raw_dir) + raw_dir = remove_path_trailing_slash(raw_dir) + + # Check and are upper case + _check_data_source_is_upper_case(raw_dir) + _check_campaign_name_is_upper_case(raw_dir) # Check there are directories in raw_dir _check_directories_in_raw_dir(raw_dir) @@ -981,77 +1036,131 @@ def check_raw_dir(raw_dir: str, verbose: bool = False) -> None: #### PROCESSED Directory Checks -def _check_is_processed_dir(processed_dir): - if not isinstance(processed_dir, str): - raise TypeError("Provide 'processed_dir' as a string'.") - - # Parse the fpath - processed_dir = _parse_fpath(processed_dir) - +def _check_is_inside_processed_directory(processed_dir): + """Check the path is located within the DISDRODB/Processed directory.""" # Check is the processed_dir if processed_dir.find("DISDRODB/Processed") == -1 and processed_dir.find("DISDRODB\\Processed") == -1: msg = "Expecting 'processed_dir' to contain the pattern */DISDRODB/Processed/*. or *\\DISDRODB\\Processed\\*." logger.error(msg) raise ValueError(msg) - # Check processed_dir does not end with "DISDRODB/Processed" - # - It must contain also the directory - if ( - processed_dir.endswith("Processed") - or processed_dir.endswith("Processed/") - or processed_dir.endswith("Processed\\") - ): - msg = "Expecting 'processed_dir' to contain the pattern */DISDRODB/Processed/." + +def _check_valid_processed_dir(processed_dir): + """Check the validity of 'processed_dir'. + + The path must represents this path */DISDRODB/Processed// + """ + last_component = os.path.basename(processed_dir) + tree_components = _infer_disdrodb_tree_path_components(processed_dir) + tree_path = "/".join(tree_components) + # Check that is not data_source or 'Processed' directory + if len(tree_components) < 4: + msg = "Expecting 'processed_dir' to contain the pattern <...>/DISDRODB/Processed//." + msg = msg + f"It only provides {tree_path}" logger.error(msg) raise ValueError(msg) + # Check that ends with the campaign_name + campaign_name = tree_components[3] + if last_component != campaign_name: + msg = "Expecting 'processed_dir' to contain the pattern <...>/DISDRODB/Processed//." + msg = msg + f"The 'processed_dir' path {processed_dir} does not end with '{campaign_name}'!" + logger.error(msg) + raise ValueError(msg) + + +def check_processed_dir(processed_dir): + """Check input, format and validity of the 'processed_dir' directory path. + + Parameters + ---------- + processed_dir : str + Path to the campaign directory in the 'DISDRODB/Processed directory tree + + Returns + ------- + str + Path of the processed campaign directory + """ + # Check path type + if not isinstance(processed_dir, str): + raise TypeError("Provide 'processed_dir' as a string'.") + + # Ensure valid path format + processed_dir = remove_path_trailing_slash(processed_dir) + + # Check the path is inside the DISDRDB/Processed directory + _check_is_inside_processed_directory(processed_dir) + + # Check processed_dir is <...>/DISDRODB/Processed// + _check_valid_processed_dir(processed_dir) + + # Check and are upper case + _check_data_source_is_upper_case(processed_dir) + _check_campaign_name_is_upper_case(processed_dir) + return processed_dir -def _check_campaign_name(raw_dir: str, processed_dir: str) -> str: - """Check that 'raw_dir' and 'processed_dir' have same campaign_name. +####---------------------------------------------------------------------------. +#### L0A and L0B directory creation routines + + +def _check_data_source_consistency(raw_dir: str, processed_dir: str) -> str: + """Check that 'raw_dir' and 'processed_dir' have same data_source. Parameters ---------- raw_dir : str - Path of the raw directory + Path of the raw campaign directory processed_dir : str - Path of the processed directory + Path of the processed campaign directory Returns ------- str - Campaign name in capital letter + data_source in capital letter. Raises ------ ValueError - Error if both paths do not match. + Error if the data_source of the two directory paths does not match. """ - upper_campaign_name = os.path.basename(raw_dir).upper() - raw_campaign_name = os.path.basename(raw_dir) - processed_campaign_name = os.path.basename(processed_dir) - if raw_campaign_name != processed_campaign_name: - msg = f"'raw_dir' and 'processed_dir' must ends with same {upper_campaign_name}" + raw_data_source = _infer_campaign_name_from_path(raw_dir) + processed_data_source = _infer_campaign_name_from_path(processed_dir) + if raw_data_source != processed_data_source: + msg = f"'raw_dir' and 'processed_dir' must ends with same : {raw_data_source}" logger.error(msg) raise ValueError(msg) + return raw_data_source.upper() - if raw_campaign_name != upper_campaign_name: - msg = f"'raw_dir' and 'processed_dir' must ends with UPPERCASE {upper_campaign_name}" - logger.error(msg) - raise ValueError(msg) - return upper_campaign_name +def _check_campaign_name_consistency(raw_dir: str, processed_dir: str) -> str: + """Check that 'raw_dir' and 'processed_dir' have same campaign_name. + + Parameters + ---------- + raw_dir : str + Path of the raw campaign directory + processed_dir : str + Path of the processed campaign directory + Returns + ------- + str + Campaign name in capital letter. -def _create_processed_dir_folder(processed_dir, dir_name): - """Create directory inside the processed_dir directory.""" - try: - folder_path = os.path.join(processed_dir, dir_name) - os.makedirs(folder_path, exist_ok=True) - except Exception as e: - msg = f"Can not create folder {dir_name} at {folder_path}. Error: {e}" - logger.exception(msg) - raise FileNotFoundError(msg) + Raises + ------ + ValueError + Error if the campaign_name of the two directory paths does not match. + """ + raw_campaign_name = _infer_campaign_name_from_path(raw_dir) + processed_campaign_name = _infer_campaign_name_from_path(processed_dir) + if raw_campaign_name != processed_campaign_name: + msg = f"'raw_dir' and 'processed_dir' must ends with same : {raw_campaign_name}" + logger.error(msg) + raise ValueError(msg) + return raw_campaign_name.upper() def _copy_station_metadata(raw_dir: str, processed_dir: str, station_name: str) -> None: @@ -1060,9 +1169,9 @@ def _copy_station_metadata(raw_dir: str, processed_dir: str, station_name: str) Parameters ---------- raw_dir : str - Path of the raw directory + Path of the raw campaign directory processed_dir : str - Path of the processed directory + Path of the processed campaign directory Raises ------ @@ -1080,15 +1189,8 @@ def _copy_station_metadata(raw_dir: str, processed_dir: str, station_name: str) raise ValueError(f"No metadata available for {station_name} at {raw_metadata_fpath}") # Define the destination fpath processed_metadata_fpath = os.path.join(processed_metadata_dir, os.path.basename(raw_metadata_fpath)) - # Try copying the file - try: - shutil.copy(raw_metadata_fpath, processed_metadata_fpath) - msg = f"{metadata_fname} copied at {processed_metadata_fpath}." - logger.info(msg) - except Exception as e: - msg = f"Something went wrong when copying {metadata_fname} into {processed_metadata_dir}.\n The error is: {e}." - logger.error(msg) - raise ValueError(msg) + # Copy the metadata file + _copy_file(src_fpath=raw_metadata_fpath, dst_fpath=processed_metadata_fpath) return None @@ -1127,24 +1229,13 @@ def _check_pre_existing_station_data(campaign_dir, product, station_name, force= raise ValueError(msg) -def check_processed_dir(processed_dir): - """Check input, format and validity of the directory path - - Parameters - ---------- - processed_dir : str - Path of the processed directory - - Returns - ------- - str - Path of the processed directory - """ - processed_dir = _check_is_processed_dir(processed_dir) - return processed_dir - - -def create_initial_directory_structure(raw_dir, processed_dir, station_name, force, verbose=False, product="L0A"): +def create_initial_directory_structure(raw_dir, + processed_dir, + station_name, + force, + product, + verbose=False, + ): """Create directory structure for the first L0 DISDRODB product. If the input data are raw text files --> product = "L0A" (run_l0a) @@ -1156,10 +1247,9 @@ def create_initial_directory_structure(raw_dir, processed_dir, station_name, for raw_dir = check_raw_dir(raw_dir=raw_dir, verbose=verbose) processed_dir = check_processed_dir(processed_dir=processed_dir) - # Check valid campaign name - # - The campaign_name concides between raw and processed dir - # - The campaign_name is all upper case - _ = _check_campaign_name(raw_dir=raw_dir, processed_dir=processed_dir) + # Check consistent data_source and campaign name + _ = _check_data_source_consistency(raw_dir=raw_dir, processed_dir=processed_dir) + _ = _check_campaign_name_consistency(raw_dir=raw_dir, processed_dir=processed_dir) # Get list of available stations (at raw level) list_stations = _get_list_stations_with_data(product="RAW", campaign_dir=raw_dir) @@ -1169,9 +1259,9 @@ def create_initial_directory_structure(raw_dir, processed_dir, station_name, for raise ValueError(f"No data available for station {station_name}. Available stations: {list_stations}.") # Create required directory (if they don't exists) - _create_processed_dir_folder(processed_dir, dir_name="metadata") - _create_processed_dir_folder(processed_dir, dir_name="info") - _create_processed_dir_folder(processed_dir, dir_name=product) + _create_required_directory(processed_dir, dir_name="metadata") + _create_required_directory(processed_dir, dir_name="info") + _create_required_directory(processed_dir, dir_name=product) # Copy the station metadata _copy_station_metadata(raw_dir=raw_dir, processed_dir=processed_dir, station_name=station_name) @@ -1185,7 +1275,11 @@ def create_initial_directory_structure(raw_dir, processed_dir, station_name, for ) -def create_directory_structure(processed_dir, product, station_name, force, verbose=False): +def create_directory_structure(processed_dir, + product, + station_name, + force, + verbose=False): """Create directory structure for L0B and higher DISDRODB products.""" from disdrodb.api.checks import check_product from disdrodb.api.io import _get_list_stations_with_data @@ -1207,7 +1301,7 @@ def create_directory_structure(processed_dir, product, station_name, force, verb ) # Create required directory (if they don't exists) - _create_processed_dir_folder(processed_dir, dir_name=product) + _create_required_directory(processed_dir, dir_name=product) # Remove / directory if force=True _check_pre_existing_station_data( @@ -1220,6 +1314,9 @@ def create_directory_structure(processed_dir, product, station_name, force, verb ####--------------------------------------------------------------------------. #### DISDRODB L0A Readers + + +# --> TODO: in L0A processing ! def _read_l0a(fpath: str, verbose: bool = False, debugging_mode: bool = False) -> pd.DataFrame: # Log msg = f" - Reading L0 Apache Parquet file at {fpath} started." @@ -1267,6 +1364,7 @@ def read_l0a_dataframe( # Check filepaths validity if not isinstance(fpaths, (list, str)): raise TypeError("Expecting fpaths to be a string or a list of strings.") + # ---------------------------------------- # If fpath is a string, convert to list if isinstance(fpaths, str): diff --git a/disdrodb/l0/l0_processing.py b/disdrodb/l0/l0_processing.py index 93b9625b..80995f78 100644 --- a/disdrodb/l0/l0_processing.py +++ b/disdrodb/l0/l0_processing.py @@ -352,7 +352,7 @@ def run_l0a( ---------- raw_dir : str The directory path where all the raw content of a specific campaign is stored. - The path must have the following structure: ``<...>/DISDRODB/Raw//``. + The path must have the following structure: ``<...>/DISDRODB/Raw//``. Inside the ``raw_dir`` directory, it is required to adopt the following structure:: - ``/data//`` @@ -362,15 +362,15 @@ def run_l0a( - For each ````, there must be a corresponding YAML file in the metadata subfolder. - The ``campaign_name`` are expected to be UPPER CASE. - - The ```` must semantically match between: + - The ```` must semantically match between: - the ``raw_dir`` and ``processed_dir`` directory paths; - with the key ``campaign_name`` within the metadata YAML files. processed_dir : str The desired directory path for the processed DISDRODB L0A and L0B products. - The path should have the following structure: ``<...>/DISDRODB/Processed//``. + The path should have the following structure: ``<...>/DISDRODB/Processed//``. For testing purposes, this function exceptionally accepts also a directory path simply ending - with ```` (e.g., ``/tmp/``). + with ```` (e.g., ``/tmp/``). station_name : str The name of the station. @@ -498,7 +498,7 @@ def run_l0b( ---------- raw_dir : str The directory path where all the raw content of a specific campaign is stored. - The path must have the following structure: ``<...>/DISDRODB/Raw//``. + The path must have the following structure: ``<...>/DISDRODB/Raw//``. Inside the ``raw_dir`` directory, it is required to adopt the following structure:: - ``/data//`` @@ -508,15 +508,15 @@ def run_l0b( - For each ````, there must be a corresponding YAML file in the metadata subfolder. - The ``campaign_name`` are expected to be UPPER CASE. - - The ```` must semantically match between: + - The ```` must semantically match between: - the ``raw_dir`` and ``processed_dir`` directory paths; - with the key ``campaign_name`` within the metadata YAML files. processed_dir : str The desired directory path for the processed DISDRODB L0A and L0B products. - The path should have the following structure: ``<...>/DISDRODB/Processed//``. + The path should have the following structure: ``<...>/DISDRODB/Processed//``. For testing purposes, this function exceptionally accepts also a directory path simply ending - with ```` (e.g., ``/tmp/``). + with ```` (e.g., ``/tmp/``). station_name : str The name of the station. @@ -644,7 +644,7 @@ def run_l0b_from_nc( ---------- raw_dir : str The directory path where all the raw content of a specific campaign is stored. - The path must have the following structure: ``<...>/DISDRODB/Raw//``. + The path must have the following structure: ``<...>/DISDRODB/Raw//``. Inside the ``raw_dir`` directory, it is required to adopt the following structure:: - ``/data//`` @@ -654,15 +654,15 @@ def run_l0b_from_nc( - For each ````, there must be a corresponding YAML file in the metadata subfolder. - The ``campaign_name`` are expected to be UPPER CASE. - - The ```` must semantically match between: + - The ```` must semantically match between: - the ``raw_dir`` and ``processed_dir`` directory paths; - with the key ``campaign_name`` within the metadata YAML files. processed_dir : str The desired directory path for the processed DISDRODB L0A and L0B products. - The path should have the following structure: ``<...>/DISDRODB/Processed//``. + The path should have the following structure: ``<...>/DISDRODB/Processed//``. For testing purposes, this function exceptionally accepts also a directory path simply ending - with ```` (e.g., ``/tmp/``). + with ```` (e.g., ``/tmp/``). station_name : str The name of the station. diff --git a/disdrodb/l0/l0_reader.py b/disdrodb/l0/l0_reader.py index 8eb574f2..87973266 100644 --- a/disdrodb/l0/l0_reader.py +++ b/disdrodb/l0/l0_reader.py @@ -275,19 +275,19 @@ def _check_metadata_reader(metadata): if "reader" not in metadata: raise ValueError("The reader is not specified in the metadata.") # If the reader name is specified, test it is valid. - # - Convention: reader: "/" in disdrodb.l0.readers + # - Convention: reader: "/" in disdrodb.l0.readers reader_reference = metadata.get("reader") # - Check it contains / if "/" not in reader_reference: raise ValueError( f"The reader '{reader_reference}' reported in the metadata is not valid. Must have" - " '/' pattern." + " '/' pattern." ) # - Get the reader_reference component list reader_components = reader_reference.split("/") # - Check composed by two elements if len(reader_components) != 2: - raise ValueError("Expecting the reader reference to be composed of /.") + raise ValueError("Expecting the reader reference to be composed of /.") # - Retrieve reader data source and reader name reader_data_source = reader_components[0] reader_name = reader_components[1] @@ -380,7 +380,7 @@ def reader_generic_docstring(): ---------- raw_dir : str The directory path where all the raw content of a specific campaign is stored. - The path must have the following structure ``<...>/DISDRODB/Raw//``. + The path must have the following structure ``<...>/DISDRODB/Raw//``. Inside the ``raw_dir`` directory, it is required to adopt the following structure:: - ``/data//`` @@ -389,17 +389,17 @@ def reader_generic_docstring(): **Important points:** - For each ````, there must be a corresponding YAML file in the metadata subfolder. - - The ```` are expected to be UPPER CASE. - - The ```` must semantically match between: + - The ```` are expected to be UPPER CASE. + - The ```` must semantically match between: - the ``raw_dir`` and ``processed_dir`` directory paths; - with the key ``campaign_name`` within the metadata YAML files. processed_dir : str The desired directory path for the processed DISDRODB L0A and L0B products. - The path should have the following structure ``<...>/DISDRODB/Processed//`` + The path should have the following structure ``<...>/DISDRODB/Processed//`` For testing purposes, this function exceptionally accepts also a directory path simply ending - with ```` (e.g., ``/tmp/``). + with ```` (e.g., ``/tmp/``). station_name : str The name of the station. diff --git a/disdrodb/l0/routines.py b/disdrodb/l0/routines.py index d8b95489..2de461ac 100644 --- a/disdrodb/l0/routines.py +++ b/disdrodb/l0/routines.py @@ -48,8 +48,8 @@ def click_l0_station_arguments(function: object): Function. """ function = click.argument("station_name", metavar="")(function) - function = click.argument("campaign_name", metavar="")(function) - function = click.argument("data_source", metavar="")(function) + function = click.argument("campaign_name", metavar="")(function) + function = click.argument("data_source", metavar="")(function) return function diff --git a/disdrodb/tests/conftest.py b/disdrodb/tests/conftest.py index 8245079f..b68e96a9 100644 --- a/disdrodb/tests/conftest.py +++ b/disdrodb/tests/conftest.py @@ -19,8 +19,8 @@ def create_fake_metadata_file( base_dir, metadata_dict={}, - data_source="data_source", - campaign_name="campaign_name", + data_source="DATA_SOURCE", + campaign_name="CAMPAIGN_NAME", station_name="station_name", ): # Define metadata directory diff --git a/disdrodb/tests/test_data_transfer/test_upload_data.py b/disdrodb/tests/test_data_transfer/test_upload_data.py index ee62010c..535159f9 100644 --- a/disdrodb/tests/test_data_transfer/test_upload_data.py +++ b/disdrodb/tests/test_data_transfer/test_upload_data.py @@ -32,7 +32,7 @@ def create_fake_data_dir( - base_dir, data_source="data_source", campaign_name="campaign_name", station_name="station_name" + base_dir, data_source="DATA_SOURCE", campaign_name="CAMPAIGN_NAME", station_name="station_name" ): data_dir = base_dir / "Raw" / data_source / campaign_name / "data" / station_name if not data_dir.exists(): diff --git a/disdrodb/tests/test_l0/test_io.py b/disdrodb/tests/test_l0/test_io.py index 74ab1a8d..85000c12 100644 --- a/disdrodb/tests/test_l0/test_io.py +++ b/disdrodb/tests/test_l0/test_io.py @@ -34,25 +34,70 @@ TEST_DATA_DIR = os.path.join(__root_path__, "disdrodb", "tests", "data") -def test_check_is_processed_dir(tmp_path): - subfolder_path = tmp_path / "DISDRODB" / "Processed" / "data_source" / "campaign_name" - subfolder_path.mkdir(parents=True) +def test_check_processed_dir(tmp_path): + # Check correct path + processed_dir = tmp_path / "DISDRODB" / "Processed" / "DATA_SOURCE" / "CAMPAIGN_NAME" + processed_dir.mkdir(parents=True, exist_ok=True) - assert io._check_is_processed_dir(str(subfolder_path)) == str(subfolder_path) + assert io.check_processed_dir(str(processed_dir)) == str(processed_dir) + # Check wrong type raises error + with pytest.raises(TypeError): + io.check_processed_dir(1) -def test_check_processed_dir(tmp_path): - subfolder_path = tmp_path / "DISDRODB" / "Processed" / "data_source" / "campaign_name" - subfolder_path.mkdir(parents=True) + # Check wrong path (Raw) + processed_dir = tmp_path / "DISDRODB" / "Raw" / "DATA_SOURCE" / "CAMPAIGN_NAME" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) - assert io.check_processed_dir(str(subfolder_path)) == str(subfolder_path) + # Check wrong path (only data_source) + processed_dir = tmp_path / "DISDRODB" / "Processed" / "DATA_SOURCE" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) + + # Check wrong path (only Processed) + processed_dir = tmp_path / "DISDRODB" / "Processed" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) + + # Check wrong path (station_dir) + processed_dir = tmp_path / "DISDRODB" / "Processed" / "DATA_SOURCE" / "CAMPAIGN_NAME" / "data" / "station_name" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) + + # Check wrong path (lowercase data_source) + processed_dir = tmp_path / "DISDRODB" / "Processed" / "data_source" / "CAMPAIGN_NAME" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) + + # Check wrong path (lowercase data_source) + processed_dir = tmp_path / "DISDRODB" / "Processed" / "DATA_SOURCE" / "campaign_name" + processed_dir.mkdir(parents=True, exist_ok=True) + with pytest.raises(ValueError): + io.check_processed_dir(str(processed_dir)) def test_create_initial_directory_structure(tmp_path, mocker): + force = False + product = "LOA" + + # Define station info base_dir = tmp_path / "DISDRODB" - station_name = "station_1" - data_source = "data_source" + data_source = "DATA_SOURCE" campaign_name = "CAMPAIGN_NAME" + station_name = "station_1" + + # Define Raw campaign directory structure + raw_dir = tmp_path / "DISDRODB" / "Raw" / "DATA_SOURCE" / campaign_name + raw_station_dir = raw_dir / "data" / station_name + raw_station_dir.mkdir(parents=True) + + # - Add metadata metadata_dict = {} _ = create_fake_metadata_file( base_dir=base_dir, @@ -61,34 +106,52 @@ def test_create_initial_directory_structure(tmp_path, mocker): campaign_name=campaign_name, station_name=station_name, ) - - raw_dir = os.path.join(tmp_path, "DISDRODB", "Raw", "data_source", campaign_name) - force = False - subfolder_path = tmp_path / "DISDRODB" / "Raw" / "data_source" / campaign_name / "data" / station_name - subfolder_path.mkdir(parents=True) - - fake_csv_file_path = os.path.join(subfolder_path, f"{station_name}.csv") + # - Add fake file + fake_csv_file_path = os.path.join(raw_station_dir, f"{station_name}.csv") with open(fake_csv_file_path, "w") as f: f.write("fake csv file") - processed_dir = os.path.join(tmp_path, "DISDRODB", "Processed", campaign_name) - subfolder_path = tmp_path / "DISDRODB" / "Processed" / campaign_name - subfolder_path.mkdir(parents=True) + # Define Processed campaign directory + processed_dir = tmp_path / "DISDRODB" / "Processed" / data_source / campaign_name + processed_dir.mkdir(parents=True) + + # Mock to pass metadata checks mocker.patch("disdrodb.metadata.check_metadata.check_metadata_compliance", return_value=None) + # Execute create_initial_directory_structure io.create_initial_directory_structure( - raw_dir=raw_dir, processed_dir=processed_dir, station_name=station_name, force=force + raw_dir=str(raw_dir), + processed_dir=str(processed_dir), + station_name=station_name, + force=force, + product=product, ) - l0a_folder_path = os.path.join(processed_dir, "L0A") - assert os.path.exists(l0a_folder_path) + # Test product directory has been created + expected_folder_path = os.path.join(processed_dir, product) + assert os.path.exists(expected_folder_path) def test_create_directory_structure(tmp_path, mocker): + # from pathlib import Path + # tmp_path = Path("/tmp/test12") + # tmp_path.mkdir() + + force = False + product = "L0B" + + # Define station info base_dir = tmp_path / "DISDRODB" - station_name = "station_1" - data_source = "data_source" + data_source = "DATA_SOURCE" campaign_name = "CAMPAIGN_NAME" + station_name = "station_1" + + # Define Raw campaign directory structure + raw_dir = tmp_path / "DISDRODB" / "Raw" / "DATA_SOURCE" / campaign_name + raw_station_dir = raw_dir / "data" / station_name + raw_station_dir.mkdir(parents=True) + + # - Add metadata metadata_dict = {} _ = create_fake_metadata_file( base_dir=base_dir, @@ -97,40 +160,37 @@ def test_create_directory_structure(tmp_path, mocker): campaign_name=campaign_name, station_name=station_name, ) - - os.path.join(tmp_path, "DISDRODB", "Raw", "data_source", campaign_name) - force = False - subfolder_path = tmp_path / "DISDRODB" / "Raw" / "data_source" / campaign_name / "data" / station_name - subfolder_path.mkdir(parents=True) - - fake_csv_file_path = os.path.join(subfolder_path, f"{station_name}.csv") + # - Add fake file + fake_csv_file_path = os.path.join(raw_station_dir, f"{station_name}.csv") with open(fake_csv_file_path, "w") as f: f.write("fake csv file") - processed_dir = os.path.join(tmp_path, "DISDRODB", "Processed", campaign_name) - subfolder_path = tmp_path / "DISDRODB" / "Processed" / campaign_name / "L0B" - subfolder_path.mkdir(parents=True) + # Define Processed campaign directory + processed_dir = tmp_path / "DISDRODB" / "Processed" / data_source / campaign_name + processed_dir.mkdir(parents=True) - product = "L0B" - station_name = "station_1" - force = False + # subfolder_path = tmp_path / "DISDRODB" / "Processed" / campaign_name / "L0B" + # subfolder_path.mkdir(parents=True) + # Mock to pass some internal checks mocker.patch("disdrodb.api.io._get_list_stations_with_data", return_value=[station_name]) mocker.patch("disdrodb.l0.io._check_pre_existing_station_data", return_value=None) + # Execute create_directory_structure io.create_directory_structure( - processed_dir=processed_dir, product=product, station_name=station_name, force=force, verbose=False + processed_dir=str(processed_dir), product=product, station_name=station_name, force=force, verbose=False ) - l0a_folder_path = os.path.join(processed_dir, "L0B") + # Test product directory has been created + l0a_folder_path = os.path.join(processed_dir, product) assert os.path.exists(l0a_folder_path) def test_check_raw_dir_is_a_directory(tmp_path): base_dir = tmp_path / "DISDRODB" station_name = "station_1" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" metadata_dict = {} _ = create_fake_metadata_file( base_dir=base_dir, @@ -193,12 +253,12 @@ def test__infer_base_dir_from_fpath(): assert io._infer_base_dir_from_fpath(base_dir) == base_dir -def test__infer_disdrodb_tree_path_components(): +def test_infer_disdrodb_tree_path_components(): # Assert retrieve correct disdrodb path path_components = ["DISDRODB", "Raw", "DATA_SOURCE", "CAMPAIGN_NAME"] disdrodb_path = os.path.join(*path_components) path = os.path.join("whatever_path", disdrodb_path) - assert io.__infer_disdrodb_tree_path_components(path) == path_components + assert io._infer_disdrodb_tree_path_components(path) == path_components def test__infer_data_source_from_path(): @@ -432,14 +492,14 @@ def test__remove_directory(tmp_path): assert res -def test_parse_fpath(): +def testremove_path_trailing_slash(): path_dir_windows_in = "\\DISDRODB\\Processed\\DATA_SOURCE\\CAMPAIGN_NAME\\" path_dir_windows_out = "\\DISDRODB\\Processed\\DATA_SOURCE\\CAMPAIGN_NAME" - assert io._parse_fpath(path_dir_windows_in) == path_dir_windows_out + assert io.remove_path_trailing_slash(path_dir_windows_in) == path_dir_windows_out path_dir_linux_in = "/DISDRODB/Processed/DATA_SOURCE/CAMPAIGN_NAME/" path_dir_linux_out = "/DISDRODB/Processed/DATA_SOURCE/CAMPAIGN_NAME" - assert io._parse_fpath(path_dir_linux_in) == path_dir_linux_out + assert io.remove_path_trailing_slash(path_dir_linux_in) == path_dir_linux_out def test_check_raw_dir(): @@ -460,7 +520,7 @@ def test_check_raw_dir(): assert io.check_raw_dir(raw_dir) == raw_dir -def test_check_campaign_name(): +def test_check_campaign_name_consistency(): campaign_name = "CAMPAIGN_NAME" data_source = "DATA_SOURCE" path_raw = os.path.join( @@ -480,7 +540,7 @@ def test_check_campaign_name(): campaign_name, ) - assert io._check_campaign_name(path_raw, path_process) == campaign_name + assert io._check_campaign_name_consistency(path_raw, path_process) == campaign_name def test_copy_station_metadata(): diff --git a/disdrodb/tests/test_l0/test_l0_reader.py b/disdrodb/tests/test_l0/test_l0_reader.py index a2eb027f..311cd7bf 100644 --- a/disdrodb/tests/test_l0/test_l0_reader.py +++ b/disdrodb/tests/test_l0/test_l0_reader.py @@ -64,8 +64,8 @@ def test_check_metadata_reader(): def test_get_station_reader_function(tmp_path): base_dir = tmp_path / "DISDRODB" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAGIN_NAME" station_name = "station_name" metadata_dict = {"reader": f"{DATA_SOURCE}/{CAMPAIGN_NAME}"} diff --git a/disdrodb/tests/test_l0/test_l0b_concat.py b/disdrodb/tests/test_l0/test_l0b_concat.py index e592671e..1012a3d2 100644 --- a/disdrodb/tests/test_l0/test_l0b_concat.py +++ b/disdrodb/tests/test_l0/test_l0b_concat.py @@ -179,8 +179,8 @@ def create_fake_data_file(tmp_path, data_source, campaign_name, station_name="", def test_run_l0b_concat_station(tmp_path): # Define station info - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "test_station" root_dir_path = os.path.join(tmp_path, "DISDRODB", "Processed", data_source, campaign_name) @@ -222,13 +222,13 @@ def mock_write_l0b(ds: xr.Dataset, fpath: str, force=False) -> None: def test_run_disdrodb_l0b_concat(tmp_path): # from pathlib import Path - # tmp_path = Path("/tmp/test10") + # tmp_path = Path("/tmp/test11") # tmp_path.mkdir() # Define stations info base_dir = os.path.join(tmp_path, "DISDRODB") - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name1 = "test_station_1" station_name2 = "test_station_2" @@ -260,7 +260,7 @@ def test_run_disdrodb_l0b_concat(tmp_path): # Run concatenation command run_disdrodb_l0b_concat( - base_dir=base_dir, + base_dir=str(base_dir), data_sources=data_source, campaign_names=campaign_name, station_names=[station_name1, station_name2], @@ -268,9 +268,10 @@ def test_run_disdrodb_l0b_concat(tmp_path): verbose=False, ) - # Assert the presence of 2 concatenated netcdf files (one for each station) - list_files = glob.glob(os.path.join(base_dir, "Processed", data_source, campaign_name, "L0B", "*.nc")) - assert len(list_files) == 2 + # # Assert the presence of 2 concatenated netcdf files (one for each station) + # expected_dst_dir = os.path.join(base_dir, "Processed", data_source, campaign_name, "L0B") + # list_files = glob.glob(os.path.join(expected_dst_dir, "*.nc")) + # assert len(list_files) == 2 # Check that if L0B files are removed, raise error if no stations available with pytest.raises(ValueError): diff --git a/disdrodb/tests/test_l0/test_metadata.py b/disdrodb/tests/test_l0/test_metadata.py index e24cf8e0..23b69f06 100644 --- a/disdrodb/tests/test_l0/test_metadata.py +++ b/disdrodb/tests/test_l0/test_metadata.py @@ -33,7 +33,7 @@ def create_fake_station_file( - base_dir, data_source="data_source", campaign_name="campaign_name", station_name="station_name" + base_dir, data_source="DATA_SOURCE", campaign_name="CAMPAIGN_NAME", station_name="station_name" ): subfolder_path = base_dir / "Raw" / data_source / campaign_name / "data" / station_name if not os.path.exists(subfolder_path): @@ -72,7 +72,7 @@ def test_get_default_metadata(): assert isinstance(_get_default_metadata_dict(), dict) -def create_fake_metadata_folder(tmp_path, data_source="data_source", campaign_name="campaign_name"): +def create_fake_metadata_folder(tmp_path, data_source="DATA_SOURCE", campaign_name="CAMPAIGN_NAME"): subfolder_path = tmp_path / "DISDRODB" / "Raw" / data_source / campaign_name / "metadata" if not os.path.exists(subfolder_path): subfolder_path.mkdir(parents=True) @@ -83,8 +83,8 @@ def create_fake_metadata_folder(tmp_path, data_source="data_source", campaign_na def test_write_default_metadata(tmp_path): - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_name" fpath = os.path.join(create_fake_metadata_folder(tmp_path, data_source, campaign_name), f"{station_name}.yml") diff --git a/disdrodb/tests/test_metadata/test_check_metadata.py b/disdrodb/tests/test_metadata/test_check_metadata.py index 5ac934a4..8e7a8091 100644 --- a/disdrodb/tests/test_metadata/test_check_metadata.py +++ b/disdrodb/tests/test_metadata/test_check_metadata.py @@ -126,14 +126,14 @@ def test_check_archive_metadata_campaign_name(tmp_path): base_dir = tmp_path / "DISDRODB" # Test 1 : Correct campaign_name metadata key - campaign_name = "campaign_name" + campaign_name = "CAMPAIGN_NAME" metadata_dict = {"campaign_name": campaign_name} _ = create_fake_metadata_file(base_dir=base_dir, campaign_name=campaign_name, metadata_dict=metadata_dict) is_valid = check_archive_metadata_campaign_name(str(base_dir)) assert is_valid # Test 2 : Wrong campaign_name metadata key - campaign_name = "campaign_name" + campaign_name = "CAMPAIGN_NAME" metadata_dict = {"campaign_name": ""} _ = create_fake_metadata_file(base_dir=base_dir, campaign_name=campaign_name, metadata_dict=metadata_dict) is_valid = check_archive_metadata_campaign_name(str(base_dir)) @@ -144,14 +144,14 @@ def test_check_archive_metadata_data_source(tmp_path): base_dir = tmp_path / "DISDRODB" # Test 1 : Correct data_source metadata key - data_source = "data_source" + data_source = "DATA_SOURCE" metadata_dict = {"data_source": data_source} _ = create_fake_metadata_file(base_dir=base_dir, data_source=data_source, metadata_dict=metadata_dict) is_valid = check_archive_metadata_data_source(str(base_dir)) assert is_valid # Test 2 : Wrong data_source metadata key - data_source = "data_source" + data_source = "DATA_SOURCE" metadata_dict = {"data_source": ""} _ = create_fake_metadata_file(base_dir=base_dir, data_source=data_source, metadata_dict=metadata_dict) is_valid = check_archive_metadata_data_source(str(base_dir)) diff --git a/disdrodb/tests/test_metadata/test_metadata_info.py b/disdrodb/tests/test_metadata/test_metadata_info.py index 56d91a03..6f6f551e 100644 --- a/disdrodb/tests/test_metadata/test_metadata_info.py +++ b/disdrodb/tests/test_metadata/test_metadata_info.py @@ -31,8 +31,8 @@ def test_get_archive_metadata_key_value(tmp_path): # Test 1 : one config file expected_key = "key1" expected_value = "value1" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_name1" metadata_dict = {expected_key: expected_value} @@ -51,8 +51,8 @@ def test_get_archive_metadata_key_value(tmp_path): # Test 2 : two config files expected_key = "key1" expected_value = "value1" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_name2" metadata_dict = {expected_key: expected_value} _ = create_fake_metadata_file( @@ -71,8 +71,8 @@ def test_get_archive_metadata_key_value(tmp_path): # Test 3: test tuple expected_key = "key1" expected_value = "value1" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_name3" metadata_dict = {expected_key: expected_value} diff --git a/disdrodb/tests/test_metadata/test_metadata_io.py b/disdrodb/tests/test_metadata/test_metadata_io.py index 349cd5e9..a52d1e77 100644 --- a/disdrodb/tests/test_metadata/test_metadata_io.py +++ b/disdrodb/tests/test_metadata/test_metadata_io.py @@ -26,7 +26,7 @@ from disdrodb.tests.conftest import create_fake_metadata_file -def create_fake_data_file(tmp_path, data_source="data_source", campaign_name="campaign_name", station_name=""): +def create_fake_data_file(tmp_path, data_source="DATA_SOURCE", campaign_name="CAMPAIGN_NAME", station_name=""): subfolder_path = tmp_path / "DISDRODB" / "Raw" / data_source / campaign_name / "data" / station_name if not os.path.exists(subfolder_path): subfolder_path.mkdir(parents=True) @@ -48,8 +48,8 @@ def test__get_list_all_metadata(tmp_path): # Test 1 : one metadata file key_name = "key1" metadata_dict = {key_name: "value1"} - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_1" metadata_filepath = create_fake_metadata_file( @@ -94,8 +94,8 @@ def test__get_list_metadata_with_data(tmp_path): base_dir = tmp_path / "DISDRODB" # Test 1 : one metadata file + one data file - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_1" key_name = "key1" @@ -161,8 +161,8 @@ def test_get_list_metadata_file(tmp_path): base_dir = tmp_path / "DISDRODB" - data_source = "data_source" - campaign_name = "campaign_name" + data_source = "DATA_SOURCE" + campaign_name = "CAMPAIGN_NAME" station_name = "station_name" metadata_filepath = create_fake_metadata_file( base_dir=base_dir, diff --git a/docs/source/readers.rst b/docs/source/readers.rst index 128bfcc2..bb10fe9b 100644 --- a/docs/source/readers.rst +++ b/docs/source/readers.rst @@ -85,7 +85,7 @@ A reader is a function defined by the following input arguments: * ``raw_dir`` : str - The directory path where all the raw data of a specific campaign/network are stored. - * The path must have the following structure: ``<...>/DISDRODB/Raw///DISDRODB/Raw///data//`` @@ -94,9 +94,9 @@ A reader is a function defined by the following input arguments: * ``processed_dir`` : str - The desired directory path where to save the DISDRODB L0A and L0B products. - * The path should have the following structure: ``<...>/DISDRODB/Processed//`` - * The ```` must match with the one specified in the ``raw_dir``. - * For reader testing purposes, you can define i.e. ``/tmp/DISDRODB/Processed//`` + * The path should have the following structure: ``<...>/DISDRODB/Processed//`` + * The ```` must match with the one specified in the ``raw_dir``. + * For reader testing purposes, you can define i.e. ``/tmp/DISDRODB/Processed//`` * ``station_name`` : str - Name of the station to be processed. diff --git a/docs/source/software_structure.rst b/docs/source/software_structure.rst index 47eccfee..13f8d1da 100644 --- a/docs/source/software_structure.rst +++ b/docs/source/software_structure.rst @@ -33,8 +33,8 @@ The current software structure is described below: | ├── 📁 manuals | ├── 📜 \*.pdf | ├── 📁 readers -| ├── 📁 `` -| ├── 📜 \.py +| ├── 📁 `` +| ├── 📜 \.py | ├── 📁 scripts | ├── 📜 disdrodb_run_l0_station.py | ├── 📜 disdrodb_run_l0 diff --git a/tutorials/reader_preparation.ipynb b/tutorials/reader_preparation.ipynb index 0af0a4f6..f9811e0f 100644 --- a/tutorials/reader_preparation.ipynb +++ b/tutorials/reader_preparation.ipynb @@ -256,6 +256,7 @@ " station_name=station_name,\n", " force=force,\n", " verbose=False,\n", + " product=\"L0A\",\n", ")" ] },