Source code for satpy.tests.reader_tests.test_nucaps

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Copyright (c) 2017-2018 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/>.
"""Module for testing the satpy.readers.nucaps module."""

import datetime
import os
import unittest
from unittest import mock

import numpy as np

from satpy.tests.reader_tests.test_netcdf_utils import FakeNetCDF4FileHandler
from satpy.tests.utils import convert_file_content_to_data_array

DEFAULT_FILE_DTYPE = np.float32
DEFAULT_FILE_SHAPE = (120,)
DEFAULT_PRES_FILE_SHAPE = (120, 100,)
DEFAULT_FILE_DATA = np.arange(DEFAULT_FILE_SHAPE[0], dtype=DEFAULT_FILE_DTYPE)
DEFAULT_PRES_FILE_DATA = np.arange(DEFAULT_PRES_FILE_SHAPE[1], dtype=DEFAULT_FILE_DTYPE)
DEFAULT_PRES_FILE_DATA = np.repeat([DEFAULT_PRES_FILE_DATA], DEFAULT_PRES_FILE_SHAPE[0], axis=0)
DEFAULT_FILE_FACTORS = np.array([2.0, 1.0], dtype=np.float32)
DEFAULT_LAT_DATA = np.linspace(45, 65, DEFAULT_FILE_SHAPE[0]).astype(DEFAULT_FILE_DTYPE)
DEFAULT_LON_DATA = np.linspace(5, 45, DEFAULT_FILE_SHAPE[0]).astype(DEFAULT_FILE_DTYPE)
ALL_PRESSURE_LEVELS = [
    0.0161, 0.0384, 0.0769, 0.137, 0.2244, 0.3454, 0.5064, 0.714,
    0.9753, 1.2972, 1.6872, 2.1526, 2.7009, 3.3398, 4.077, 4.9204,
    5.8776, 6.9567, 8.1655, 9.5119, 11.0038, 12.6492, 14.4559, 16.4318,
    18.5847, 20.9224, 23.4526, 26.1829, 29.121, 32.2744, 35.6505,
    39.2566, 43.1001, 47.1882, 51.5278, 56.126, 60.9895, 66.1253,
    71.5398, 77.2396, 83.231, 89.5204, 96.1138, 103.017, 110.237,
    117.777, 125.646, 133.846, 142.385, 151.266, 160.496, 170.078,
    180.018, 190.32, 200.989, 212.028, 223.441, 235.234, 247.408,
    259.969, 272.919, 286.262, 300, 314.137, 328.675, 343.618, 358.966,
    374.724, 390.893, 407.474, 424.47, 441.882, 459.712, 477.961,
    496.63, 515.72, 535.232, 555.167, 575.525, 596.306, 617.511, 639.14,
    661.192, 683.667, 706.565, 729.886, 753.628, 777.79, 802.371,
    827.371, 852.788, 878.62, 904.866, 931.524, 958.591, 986.067,
    1013.95, 1042.23, 1070.92, 1100
]
ALL_PRESSURE_LEVELS = np.repeat([ALL_PRESSURE_LEVELS], DEFAULT_PRES_FILE_SHAPE[0], axis=0)


[docs] class FakeNetCDF4FileHandler2(FakeNetCDF4FileHandler): """Swap-in NetCDF4 File Handler."""
[docs] def get_test_content(self, filename, filename_info, filetype_info): """Mimic reader input file content.""" file_content = { "/attr/time_coverage_start": "2020-10-20T12:00:00.5Z", "/attr/time_coverage_end": "2020-10-20T12:00:36Z", "/attr/start_orbit_number": 1, "/attr/end_orbit_number": 2, "/attr/platform_name": "NPP", "/attr/instrument_name": "CrIS, ATMS, VIIRS", } for k, units, standard_name in [ ("Solar_Zenith", "degrees", "solar_zenith_angle"), ("Topography", "meters", ""), ("Land_Fraction", "1", ""), ("Surface_Pressure", "mb", ""), ("Skin_Temperature", "Kelvin", "surface_temperature"), ]: file_content[k] = DEFAULT_FILE_DATA file_content[k + "/shape"] = DEFAULT_FILE_SHAPE file_content[k + "/attr/units"] = units file_content[k + "/attr/valid_range"] = (0., 120.) file_content[k + "/attr/_FillValue"] = -9999. if standard_name: file_content[k + "/attr/standard_name"] = standard_name for k, units, standard_name in [ ("Temperature", "Kelvin", "air_temperature"), ("Effective_Pressure", "mb", ""), ("H2O", "1", ""), ("H2O_MR", "g/g", ""), ("O3", "1", ""), ("O3_MR", "1", ""), ("Liquid_H2O", "1", ""), ("Liquid_H2O_MR", "g/g", "cloud_liquid_water_mixing_ratio"), ("CO", "1", ""), ("CO_MR", "1", ""), ("CH4", "1", ""), ("CH4_MR", "1", ""), ("CO2", "1", ""), ("HNO3", "1", ""), ("HNO3_MR", "1", ""), ("N2O", "1", ""), ("N2O_MR", "1", ""), ("SO2", "1", ""), ("SO2_MR", "1", ""), ]: file_content[k] = DEFAULT_PRES_FILE_DATA file_content[k + "/shape"] = DEFAULT_PRES_FILE_SHAPE file_content[k + "/attr/units"] = units file_content[k + "/attr/valid_range"] = (0., 120.) file_content[k + "/attr/_FillValue"] = -9999. if standard_name: file_content[k + "/attr/standard_name"] = standard_name k = "Pressure" file_content[k] = ALL_PRESSURE_LEVELS file_content[k + "/shape"] = DEFAULT_PRES_FILE_SHAPE file_content[k + "/attr/units"] = "mb" file_content[k + "/attr/valid_range"] = (0., 2000.) file_content[k + "/attr/_FillValue"] = -9999. k = "Quality_Flag" file_content[k] = DEFAULT_FILE_DATA.astype(np.int32) file_content[k + "/shape"] = DEFAULT_FILE_SHAPE file_content[k + "/attr/valid_range"] = (0, 31) file_content[k + "/attr/_FillValue"] = -9999. k = "Longitude" file_content[k] = DEFAULT_LON_DATA file_content[k + "/shape"] = DEFAULT_FILE_SHAPE file_content[k + "/attr/units"] = "degrees_east" file_content[k + "/attr/valid_range"] = (-180., 180.) file_content[k + "/attr/standard_name"] = "longitude" file_content[k + "/attr/_FillValue"] = -9999. k = "Latitude" file_content[k] = DEFAULT_LAT_DATA file_content[k + "/shape"] = DEFAULT_FILE_SHAPE file_content[k + "/attr/units"] = "degrees_north" file_content[k + "/attr/valid_range"] = (-90., 90.) file_content[k + "/attr/standard_name"] = "latitude" file_content[k + "/attr/_FillValue"] = -9999. attrs = ("_FillValue", "flag_meanings", "flag_values", "units") cris_fors_dim_name = "Number_of_CrIS_FORs" pressure_levels_dim_name = "Number_of_P_Levels" if ("_v1" in filename): cris_fors_dim_name = "number_of_FORs" pressure_levels_dim_name = "number_of_p_levels" convert_file_content_to_data_array( file_content, attrs=attrs, dims=("z", cris_fors_dim_name, pressure_levels_dim_name)) return file_content
[docs] class TestNUCAPSReader(unittest.TestCase): """Test NUCAPS Reader.""" yaml_file = "nucaps.yaml"
[docs] def setUp(self): """Wrap NetCDF4 file handler with our own fake handler.""" from satpy._config import config_search_paths from satpy.readers.nucaps import NUCAPSFileHandler self.reader_configs = config_search_paths(os.path.join("readers", self.yaml_file)) # http://stackoverflow.com/questions/12219967/how-to-mock-a-base-class-with-python-mock-library self.p = mock.patch.object(NUCAPSFileHandler, "__bases__", (FakeNetCDF4FileHandler2,)) self.fake_handler = self.p.start() self.p.is_local = True
[docs] def tearDown(self): """Stop wrapping the NetCDF4 file handler.""" self.p.stop()
[docs] def test_init(self): """Test basic init with no extra parameters.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) assert len(loadables) == 1 r.create_filehandlers(loadables) # make sure we have some files assert r.file_handlers
[docs] def test_init_with_kwargs(self): """Test basic init with extra parameters.""" from satpy.readers import load_reader r = load_reader(self.reader_configs, mask_surface=False) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) assert len(loadables) == 1 r.create_filehandlers(loadables, fh_kwargs={"mask_surface": False}) # make sure we have some files assert r.file_handlers
[docs] def test_load_nonpressure_based(self): """Test loading all channels that aren't based on pressure.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Solar_Zenith", "Topography", "Land_Fraction", "Surface_Pressure", "Skin_Temperature", "Quality_Flag", ]) assert len(datasets) == 6 for v in datasets.values(): # self.assertNotEqual(v.info['resolution'], 0) # self.assertEqual(v.info['units'], 'degrees') assert v.ndim == 1 assert v.attrs["sensor"] == set(["cris", "atms", "viirs"]) assert type(v.attrs["start_time"]) == datetime.datetime assert type(v.attrs["end_time"]) == datetime.datetime
[docs] def test_load_pressure_based(self): """Test loading all channels based on pressure.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature", "Effective_Pressure", "H2O", "H2O_MR", "O3", "O3_MR", "Liquid_H2O", "Liquid_H2O_MR", "CO", "CO_MR", "CH4", "CH4_MR", "CO2", "HNO3", "HNO3_MR", "N2O", "N2O_MR", "SO2", "SO2_MR", ]) assert len(datasets) == 19 for v in datasets.values(): # self.assertNotEqual(v.info['resolution'], 0) assert v.ndim == 2 if np.issubdtype(v.dtype, np.floating): assert "_FillValue" not in v.attrs
[docs] def test_load_multiple_files_pressure(self): """Test loading Temperature from multiple input files.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", "NUCAPS-EDR_v1r0_npp_s201603011159009_e201603011159307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=True) assert len(datasets) == 100 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_individual_pressure_levels_true(self): """Test loading Temperature with individual pressure datasets.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=True) assert len(datasets) == 100 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_individual_pressure_levels_min_max(self): """Test loading individual Temperature with min/max level specified.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=(100., 150.)) assert len(datasets) == 6 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_individual_pressure_levels_single(self): """Test loading individual Temperature with specific levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=(103.017,)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_pressure_levels_true(self): """Test loading Temperature with all pressure levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=True) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == DEFAULT_PRES_FILE_SHAPE
[docs] def test_load_pressure_levels_min_max(self): """Test loading Temperature with min/max level specified.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=(100., 150.)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == (DEFAULT_PRES_FILE_SHAPE[0], 6)
[docs] def test_load_pressure_levels_single(self): """Test loading a specific Temperature level.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=(103.017,)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == (DEFAULT_PRES_FILE_SHAPE[0], 1)
[docs] def test_load_pressure_levels_single_and_pressure_levels(self): """Test loading a specific Temperature level and pressure levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-EDR_v1r0_npp_s201603011158009_e201603011158307_c201603011222270.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature", "Pressure_Levels"], pressure_levels=(103.017,)) assert len(datasets) == 2 t_ds = datasets["Temperature"] assert t_ds.ndim == 2 assert t_ds.shape == (DEFAULT_PRES_FILE_SHAPE[0], 1) pl_ds = datasets["Pressure_Levels"] assert pl_ds.shape == (1,)
[docs] class TestNUCAPSScienceEDRReader(unittest.TestCase): """Test NUCAPS Science EDR Reader.""" yaml_file = "nucaps.yaml"
[docs] def setUp(self): """Wrap NetCDF4 file handler with our own fake handler.""" from satpy._config import config_search_paths from satpy.readers.nucaps import NUCAPSFileHandler self.reader_configs = config_search_paths(os.path.join("readers", self.yaml_file)) # http://stackoverflow.com/questions/12219967/how-to-mock-a-base-class-with-python-mock-library self.p = mock.patch.object(NUCAPSFileHandler, "__bases__", (FakeNetCDF4FileHandler2,)) self.fake_handler = self.p.start() self.p.is_local = True
[docs] def tearDown(self): """Stop wrapping the NetCDF4 file handler.""" self.p.stop()
[docs] def test_init(self): """Test basic init with no extra parameters.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) assert len(loadables) == 1 r.create_filehandlers(loadables) # make sure we have some files assert r.file_handlers
[docs] def test_load_nonpressure_based(self): """Test loading all channels that aren't based on pressure.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Topography", "Land_Fraction", "Surface_Pressure", "Skin_Temperature", "Quality_Flag", ]) assert len(datasets) == 5 for v in datasets.values(): assert v.ndim == 1 assert v.attrs["sensor"] == set(["cris", "atms", "viirs"]) assert type(v.attrs["start_time"]) == datetime.datetime assert type(v.attrs["end_time"]) == datetime.datetime
[docs] def test_load_pressure_based(self): """Test loading all channels based on pressure.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature", "H2O", "H2O_MR", "O3", "O3_MR", "CO", "CO_MR", "CH4", "CH4_MR", "CO2", "HNO3", "HNO3_MR", "N2O", "N2O_MR", "SO2", "SO2_MR", ]) assert len(datasets) == 16 for v in datasets.values(): # self.assertNotEqual(v.info['resolution'], 0) assert v.ndim == 2
[docs] def test_load_individual_pressure_levels_true(self): """Test loading Temperature with individual pressure datasets.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=True) assert len(datasets) == 100 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_individual_pressure_levels_min_max(self): """Test loading individual Temperature with min/max level specified.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=(100., 150.)) assert len(datasets) == 6 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_individual_pressure_levels_single(self): """Test loading individual Temperature with specific levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(r.pressure_dataset_names["Temperature"], pressure_levels=(103.017,)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 1
[docs] def test_load_pressure_levels_true(self): """Test loading Temperature with all pressure levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=True) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == DEFAULT_PRES_FILE_SHAPE
[docs] def test_load_pressure_levels_min_max(self): """Test loading Temperature with min/max level specified.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=(100., 150.)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == (DEFAULT_PRES_FILE_SHAPE[0], 6)
[docs] def test_load_pressure_levels_single(self): """Test loading a specific Temperature level.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature"], pressure_levels=(103.017,)) assert len(datasets) == 1 for v in datasets.values(): assert v.ndim == 2 assert v.shape == (DEFAULT_PRES_FILE_SHAPE[0], 1)
[docs] def test_load_pressure_levels_single_and_pressure_levels(self): """Test loading a specific Temperature level and pressure levels.""" from satpy.readers import load_reader r = load_reader(self.reader_configs) loadables = r.select_files_from_pathnames([ "NUCAPS-sciEDR_am_npp_s20190703223319_e20190703223349_STC_fsr.nc", ]) r.create_filehandlers(loadables) datasets = r.load(["Temperature", "Pressure_Levels"], pressure_levels=(103.017,)) assert len(datasets) == 2 t_ds = datasets["Temperature"] assert t_ds.ndim == 2 assert t_ds.shape == (DEFAULT_PRES_FILE_SHAPE[0], 1) pl_ds = datasets["Pressure_Levels"] assert pl_ds.shape == (1,)