-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathutils_fileio.py
227 lines (185 loc) · 6.48 KB
/
utils_fileio.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
"""Utilities for file I/O and metadata handling"""
import logging
import git
import numpy as np
import xarray as xr
import xclim as xc
import icclim
import cmdline_provenance as cmdprov
import utils_spatial
def get_new_log(infile_log=None):
"""Generate command log for output file."""
try:
repo = git.Repo()
repo_url = repo.remotes[0].url.split(".git")[0]
except (git.exc.InvalidGitRepositoryError, NameError):
repo_url = None
new_log = cmdprov.new_log(
infile_logs=infile_log,
code_url=repo_url
)
return new_log
def fix_input_metadata(ds, variable, sub_daily_agg):
"""Ensure CF-compliance of input data file (which icclim requires)"""
if 'latitude' in ds.dims:
ds = ds.rename({'latitude': 'lat'})
if 'longitude' in ds.dims:
ds = ds.rename({'longitude': 'lon'})
if variable in ['pr', 'precip', 'mtpr', 'tp']:
cf_var = 'pr'
elif variable in ['tasmax', 'tmax', 'mx2t']:
cf_var = 'tasmax'
elif variable in ['tasmin', 'tmin', 'mn2t']:
cf_var = 'tasmin'
elif variable in ['tas', 't2m', '2t', 'tasmean']:
if sub_daily_agg == 'max':
cf_var = 'tasmax'
elif sub_daily_agg == 'min':
cf_var = 'tasmin'
else:
cf_var = 'tas'
elif variable in ['sfcWind', 'si10']:
cf_var = 'sfcWind'
elif variable in ['sfcWindmax', 'fg10', 'wsgsmax']:
cf_var = 'wsgsmax'
else:
ValueError(f'No metadata fixes defined for {variable}')
ds = ds.rename({variable: cf_var})
standard_names = {
'pr': 'precipitation_flux',
'tas': 'air_temperature',
'tasmin': 'air_temperature',
'tasmax': 'air_temperature',
'sfcWind': '10m_wind_speed',
'wsgsmax': 'wind_speed_of_gust',
}
long_names = {
'pr': 'Precipitation',
'tas': 'Near-Surface Air Temperature',
'tasmin': 'Daily Minimum Near-Surface Air Temperature',
'tasmax': 'Daily Maximum Near-Surface Air Temperature',
'sfcWind': 'Near-Surface (10m) Wind Speed',
'wsgsmax': 'Daily Maximum Near-Surface Wind Speed of Gust',
}
ds[cf_var].attrs['standard_name'] = standard_names[cf_var]
ds[cf_var].attrs['long_name'] = long_names[cf_var]
units = ds[cf_var].attrs['units']
if units in ['degrees_Celsius']:
ds[cf_var].attrs['units'] = 'degC'
elif units in ['mm']:
ds[cf_var].attrs['units'] = 'mm d-1'
return ds, cf_var
def fix_output_metadata(
index_ds,
index_name,
input_global_attrs,
infile_log,
software,
drop_time_bounds=False,
):
"""Make edits to output icclim or xclim metadata"""
new_global_attrs = input_global_attrs
if software == 'icclim':
new_global_attrs['icclim_version'] = icclim.__version__
new_global_attrs['references'] = index_ds.attrs['references']
elif software == 'xclim':
new_global_attrs['xclim_version'] = xc.__version__
else:
raise ValueError('unrecognised software package')
new_global_attrs['history'] = get_new_log(infile_log=infile_log)
index_ds.attrs = new_global_attrs
del index_ds[index_name].attrs['history']
del index_ds[index_name].attrs['cell_methods']
try:
del index_ds[index_name].attrs['cell methods']
except KeyError:
pass
if index_ds[index_name].attrs['units'] == '°C':
index_ds[index_name].attrs['units'] = 'degC'
if drop_time_bounds:
index_ds = index_ds.drop('time_bounds').drop('bounds')
del index_ds['time'].attrs['bounds']
return index_ds
def read_data(
infiles,
variable_name,
start_date=None,
end_date=None,
lat_bnds=None,
lon_bnds=None,
sub_daily_agg=None,
hshift=False
):
"""Read the input data file/s."""
if len(infiles) == 1:
try:
ds = xr.open_dataset(infiles[0], chunks='auto', mask_and_scale=True)
except NotImplementedError:
ds = xr.open_dataset(infiles[0], mask_and_scale=True)
else:
try:
ds = xr.open_mfdataset(infiles, chunks='auto', mask_and_scale=True)
except NotImplementedError:
ds = xr.open_mfdataset(infiles, mask_and_scale=True)
if hshift:
ds['time'] = ds['time'] - np.timedelta64(1, 'h')
if lat_bnds:
ds = utils_spatial.subset_lat(ds, lat_bnds)
if lon_bnds:
ds = utils_spatial.subset_lon(ds, lon_bnds)
if start_date or end_date:
ds = subset_time(ds, start_date=start_date, end_date=end_date)
time_freq = xr.infer_freq(ds['time'])
if sub_daily_agg and (time_freq != 'D'):
logging.info(f'resampling from {time_freq} to daily {sub_daily_agg}')
if sub_daily_agg == 'mean':
ds = ds.resample(time='1D').mean(dim='time')
elif sub_daily_agg == 'min':
ds = ds.resample(time='1D').min(dim='time')
elif sub_daily_agg == 'max':
ds = ds.resample(time='1D').max(dim='time')
time_freq = xr.infer_freq(ds['time'])
ds, cf_var = fix_input_metadata(ds, variable_name, sub_daily_agg)
try:
ds = ds.drop('height')
except ValueError:
pass
try:
ds = ds.drop('time_bnds')
except ValueError:
pass
try:
ds = ds.drop('lat_bnds')
except ValueError:
pass
try:
ds = ds.drop('lon_bnds')
except ValueError:
pass
return ds, cf_var
def subset_time(ds, start_date=None, end_date=None):
"""Subset the time axis.
Parameters
----------
ds : Union[xarray.DataArray, xarray.Dataset]
Input data.
start_date : Optional[str]
Start date of the subset.
Date string format -- can be year ("%Y"), year-month ("%Y-%m") or year-month-day("%Y-%m-%d").
Defaults to first day of input data-array.
end_date : Optional[str]
End date of the subset.
Date string format -- can be year ("%Y"), year-month ("%Y-%m") or year-month-day("%Y-%m-%d").
Defaults to last day of input data-array.
Returns
-------
Union[xarray.DataArray, xarray.Dataset]
Subsetted xarray.DataArray or xarray.Dataset
"""
return ds.sel({'time': slice(start_date, end_date)})
def profiling_stats(rprof):
"""Record profiling information"""
max_memory = np.max([result.mem for result in rprof.results])
max_cpus = np.max([result.cpu for result in rprof.results])
logging.info(f'Peak memory usage: {max_memory}MB')
logging.info(f'Peak CPU usage: {max_cpus}%')