|
2 | 2 | This is the code to make the Level 3 data product.
|
3 | 3 | """
|
4 | 4 | import argparse
|
| 5 | +from pathlib import Path |
| 6 | +from pprint import pprint |
5 | 7 |
|
| 8 | +import h5netcdf |
| 9 | +import numpy as np |
| 10 | +from termcolor import cprint |
6 | 11 |
|
| 12 | +from . import config_parser |
| 13 | +from . import metadata_mgr |
7 | 14 |
|
8 |
| -class Level3: |
9 |
| - """Class for applying the Level2 -> Level3 processing stage. |
10 | 15 |
|
11 |
| - The key method is `run()`, which acts lik a main() method for |
12 |
| - this stage. |
13 |
| - """ |
14 |
| - def __init__(self, config): |
| 16 | +class Level3: |
| 17 | + """Class for applying the Level2 -> Level3 processing stage.""" |
| 18 | + def __init__(self, run_name, config): |
15 | 19 | """
|
16 |
| - Parameters |
17 |
| - ---------- |
18 |
| - level2_data : dict, str -> array |
19 |
| - Level 2 data, mapping internal variable names to their values |
20 |
| - (generally numpy arrays) |
21 |
| - config : config_parser.Config |
22 |
| - SunCET Data Processing Pipeline configration object |
| 20 | + Args |
| 21 | + run_name: string, name of run we are processing |
| 22 | + config, config_parser.Config, SunCET Data Processing Pipeline |
| 23 | + configration object |
23 | 24 | """
|
| 25 | + self.run_name = run_name |
| 26 | + self.run_dir = Path('processing_runs') / run_name |
24 | 27 | self.config = config
|
25 | 28 |
|
| 29 | + if not self.run_dir.exists(): |
| 30 | + raise RuntimeError(f'Could not find directory {self.run_dir}') |
| 31 | + |
26 | 32 | def run(self):
|
27 | 33 | """Main method to process the level2 -> level3 stage."""
|
28 |
| - # Parse command line arguments |
29 |
| - parser = self._get_parser() |
30 |
| - args = parser.parse_args() |
| 34 | + # Load metadata |
| 35 | + metadata = metadata_mgr.MetadataManager(self.run_dir) |
31 | 36 |
|
32 |
| - def _get_parser(self): |
33 |
| - """Get command line ArgumentParser object with options defined. |
34 |
| - |
35 |
| - Returns |
36 |
| - ------- |
37 |
| - parser : argparse.ArgumentParser |
38 |
| - object which can be used to parse command line objects |
| 37 | + # Start NetCDF File |
| 38 | + nc_output_path = self.run_dir / 'level3' / 'suncet_level3.nc' |
| 39 | + nc = Level3NetCDFWriter(nc_output_path, metadata) |
| 40 | + |
| 41 | + # Write some blank values |
| 42 | + nc.write_variable('carring_lat', np.zeros(100)) |
| 43 | + nc.write_variable('carring_long', np.ones(100)) |
| 44 | + nc.close() |
| 45 | + |
| 46 | + |
| 47 | +class Level3NetCDFWriter: |
| 48 | + """Class for writing Level3 NetCDF Output.""" |
| 49 | + def __init__(self, output_path, metadata): |
| 50 | + self._output_path = output_path |
| 51 | + self._metadata = metadata |
| 52 | + self._nc_file = h5netcdf.File(self._output_path, 'w') |
| 53 | + |
| 54 | + def write_variable(self, internal_name, variable_value): |
| 55 | + """Write a variable and its associated metadata to the file. |
| 56 | +
|
| 57 | + This function is passed the internal name of the variable, and uses |
| 58 | + the metadata manager to look up the NetCDF4 name and associated |
| 59 | + attrbutes. |
| 60 | +
|
| 61 | + Args |
| 62 | + internal_name: Internal name of variable (within code) |
| 63 | + variable_value: Value for the variable in the file |
39 | 64 | """
|
40 |
| - parser = argparse.ArgumentParser() |
41 |
| - parser.add_argument('-v', '--verbose', help='Print more debugging output') |
| 65 | + variable_name = self._metadata.get_netcdf4_variable_name(internal_name) |
| 66 | + |
| 67 | + # Wrote variable data |
| 68 | + print(f'Writing internal variable ', end='') |
| 69 | + cprint(internal_name, 'yellow', end='') |
| 70 | + print(f' NetCDF variable ', end='') |
| 71 | + cprint(variable_name, 'yellow') |
42 | 72 |
|
43 |
| - return parser |
| 73 | + # TODO: this is broken |
| 74 | + self._nc_file.dimensions[variable_name + '_dim'] = variable_value.shape |
| 75 | + |
| 76 | + nc_variable = self._nc_file.create_variable( |
| 77 | + name=variable_name, |
| 78 | + dimensions=(variable_name + '_dim',), |
| 79 | + dtype=variable_value.dtype |
| 80 | + ) |
| 81 | + |
| 82 | + nc_variable[:] = variable_value |
44 | 83 |
|
| 84 | + # Write variable attributes |
| 85 | + attrs = self._metadata.get_netcdf4_attrs(internal_name) |
45 | 86 |
|
| 87 | + print('attributes:') |
| 88 | + pprint(attrs) |
| 89 | + |
| 90 | + for key, value in attrs.items(): |
| 91 | + nc_variable.attrs[key] = value |
| 92 | + |
| 93 | + print() |
| 94 | + |
| 95 | + def close(self): |
| 96 | + """Close the NetCDF file, commiting all changes.""" |
| 97 | + self._nc_file.close() |
| 98 | + |
| 99 | + |
46 | 100 | def final_shdr_compositing_fix(level2_data, config):
|
47 | 101 | """Fix any lingaring SHDR Compositing Issues.
|
48 | 102 |
|
49 |
| - Parameters |
50 |
| - ---------- |
51 |
| - level2_data : dict, str -> array |
52 |
| - Level 2 data, mapping internal variable names to their values |
53 |
| - (generally numpy arrays) |
54 |
| - config : config_parser.Config |
55 |
| - SunCET Data Processing Pipeline configration object |
| 103 | + Args |
| 104 | + level2_data : dict, str -> array |
| 105 | + Level 2 data, mapping internal variable names to their values |
| 106 | + (generally numpy arrays) |
| 107 | + config : config_parser.Config |
| 108 | + SunCET Data Processing Pipeline configration object |
56 | 109 |
|
57 |
| - Returns |
58 |
| - ------- |
59 |
| - level2_data_fixed : dict, str -> array |
60 |
| - Copy of level2 data with the fix applied. |
| 110 | + Returns |
| 111 | + level2_data_fixed : dict, str -> array |
| 112 | + Copy of level2 data with the fix applied. |
61 | 113 | """
|
62 | 114 | raise NotImplementedError()
|
| 115 | + |
| 116 | + |
| 117 | +def _get_parser(): |
| 118 | + """Get command line ArgumentParser object with options defined. |
| 119 | + |
| 120 | + Returns |
| 121 | + object which can be used to parse command line objects |
| 122 | + """ |
| 123 | + parser = argparse.ArgumentParser() |
| 124 | + parser.add_argument('-n', '--run-name', type=str, required=True, |
| 125 | + help='String name of the run') |
| 126 | + parser.add_argument('-v', '--verbose', help='Print more debugging output') |
63 | 127 |
|
| 128 | + return parser |
64 | 129 |
|
65 |
| -if __name__ == '__main__': |
66 |
| - level3 = Level3() |
| 130 | + |
| 131 | +def main(): |
| 132 | + """Main method when running this script directly.""" |
| 133 | + args = _get_parser().parse_args() |
| 134 | + |
| 135 | + # Load config |
| 136 | + config_filename = Path('processing_runs') / args.run_name / 'config.ini' |
| 137 | + config = config_parser.ConfigParser() |
| 138 | + config.read(config_filename) |
| 139 | + |
| 140 | + # Call run() method on Level3 class |
| 141 | + level3 = Level3(args.run_name, config) |
67 | 142 | level3.run()
|
| 143 | + |
| 144 | + |
| 145 | +if __name__ == '__main__': |
| 146 | + main() |
0 commit comments