Skip to content

Commit 657e40b

Browse files
committed
Renamed attribute obs_dimname to obs_dim, consistent with Xarray practice
1 parent 2874f07 commit 657e40b

7 files changed

+84
-84
lines changed

examples/example_find_positions_at_obs_times.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525
def gridwaves(tds):
2626
t = tds[['lat', 'lon',
2727
'time']].traj.gridtime(tds['time_waves_imu'].squeeze())
28-
return t.traj.to_2d(obs_dimname='obs_waves_imu')
28+
return t.traj.to_2d(obs_dim='obs_waves_imu')
2929

3030

3131
dsw = ds.groupby('trajectory').map(gridwaves)

tests/test_read_sfy.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ def test_interpret_sfy(test_data):
88
ds = xr.open_dataset(test_data / 'bug32.nc')
99
print(ds)
1010

11-
assert ds.traj.obs_dimname == 'package'
11+
assert ds.traj.obs_dim == 'package'
1212
assert ds.traj.time_varname == 'position_time'
1313

1414
assert ds.traj.is_2d()

trajan/accessor.py

+20-20
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,10 @@ def detect_tx_dim(ds):
2626
raise ValueError("Could not determine x / lon variable")
2727

2828

29-
def detect_time_dim(ds, obs_dimname):
30-
logger.debug(f'Detecting time-dimension for "{obs_dimname}"..')
29+
def detect_time_dim(ds, obs_dim):
30+
logger.debug(f'Detecting time-dimension for "{obs_dim}"..')
3131
for v in ds.variables:
32-
if obs_dimname in ds[v].dims and 'time' in v:
32+
if obs_dim in ds[v].dims and 'time' in v:
3333
return v
3434

3535
raise ValueError("no time dimension detected")
@@ -47,7 +47,7 @@ def __new__(cls, ds):
4747
ds = ds.expand_dims({'trajectory': 1})
4848
ds['trajectory'].attrs['cf_role'] = 'trajectory_id'
4949

50-
obs_dimname = None
50+
obs_dim = None
5151
time_varname = None
5252

5353
tx = detect_tx_dim(ds)
@@ -62,7 +62,7 @@ def __new__(cls, ds):
6262
# NOTE: this is probably not standard; something to point to the CF conventions?
6363
# NOTE: for now, there is no discovery of the "index" dim, this is hardcorded; any way to do better?
6464
if "index" in tx.dims:
65-
obs_dimname = "index"
65+
obs_dim = "index"
6666

6767
# discover the timecoord variable name #######################
6868
# find all variables with standard_name "time"
@@ -90,30 +90,30 @@ def __new__(cls, ds):
9090
else:
9191
raise ValueError(f"cannot deduce rowsizevar; we have the following candidates: {with_dim_trajectory = }")
9292
# sanity check
93-
if not np.sum(ds[rowsizevar].to_numpy()) == len(ds[obs_dimname]):
93+
if not np.sum(ds[rowsizevar].to_numpy()) == len(ds[obs_dim]):
9494
raise ValueError("mismatch between the index length and the sum of the deduced trajectory lengths")
9595

9696
logger.debug(
97-
f"1D storage dataset; detected: {obs_dimname = }, {timecoord = }, {trajectorycoord = }, {rowsizevar}"
97+
f"1D storage dataset; detected: {obs_dim = }, {timecoord = }, {trajectorycoord = }, {rowsizevar}"
9898
)
9999

100-
return ocls(ds, obs_dimname, timecoord, trajectorycoord, rowsizevar)
100+
return ocls(ds, obs_dim, timecoord, trajectorycoord, rowsizevar)
101101

102102
else:
103103
logging.warning(f"{ds} has {tx.dims = } which is of dimension 1 but is not index; this is a bit unusual; try to parse with Traj1d or Traj2d")
104104

105105
# we have a ds where 2D arrays are used to store data, this is either Traj1d or Traj2d
106106
# there may also be some slightly unusual cases where these Traj1d and Traj2d classes will be used on data with 1D arrays
107107
if 'obs' in tx.dims:
108-
obs_dimname = 'obs'
109-
time_varname = detect_time_dim(ds, obs_dimname)
108+
obs_dim = 'obs'
109+
time_varname = detect_time_dim(ds, obs_dim)
110110

111111
elif 'index' in tx.dims:
112-
obs_dimname = 'obs'
113-
time_varname = detect_time_dim(ds, obs_dimname)
112+
obs_dim = 'obs'
113+
time_varname = detect_time_dim(ds, obs_dim)
114114

115115
elif 'time' in tx.dims:
116-
obs_dimname = 'time'
116+
obs_dim = 'time'
117117
time_varname = 'time'
118118

119119
else:
@@ -122,18 +122,18 @@ def __new__(cls, ds):
122122
'cf_role',
123123
None) == 'trajectory_id' and not 'traj' in d:
124124

125-
obs_dimname = d
126-
time_varname = detect_time_dim(ds, obs_dimname)
125+
obs_dim = d
126+
time_varname = detect_time_dim(ds, obs_dim)
127127

128128
break
129129

130-
if obs_dimname is None:
130+
if obs_dim is None:
131131
logger.warning('No time or obs dimension detected.')
132132

133133
logger.debug(
134-
f"Detected obs-dim: {obs_dimname}, detected time-dim: {time_varname}.")
134+
f"Detected obs-dim: {obs_dim}, detected time-variable: {time_varname}.")
135135

136-
if obs_dimname is None:
136+
if obs_dim is None:
137137
ocls = Traj1d
138138

139139
elif len(ds[time_varname].shape) <= 1:
@@ -146,7 +146,7 @@ def __new__(cls, ds):
146146

147147
else:
148148
raise ValueError(
149-
f'Time dimension has shape greater than 2: {ds["time_varname"].shape}'
149+
f'Time variable has more than two dimensions: {ds[time_varname].shape}'
150150
)
151151

152-
return ocls(ds, obs_dimname, time_varname)
152+
return ocls(ds, obs_dim, time_varname)

trajan/ragged.py

+8-8
Original file line numberDiff line numberDiff line change
@@ -14,12 +14,12 @@ class ContiguousRagged(Traj):
1414
trajdim: str
1515
rowvar: str
1616

17-
def __init__(self, ds, obs_dimname, time_varname, trajectorycoord, rowsizevar):
17+
def __init__(self, ds, obs_dim, time_varname, trajectorycoord, rowsizevar):
1818
self.trajdim = trajectorycoord
1919
self.rowvar = rowsizevar
20-
super().__init__(ds, obs_dimname, time_varname)
20+
super().__init__(ds, obs_dim, time_varname)
2121

22-
def to_2d(self, obs_dimname='obs'):
22+
def to_2d(self, obs_dim='obs'):
2323
"""This actually converts a contiguous ragged xarray Dataset into an xarray Dataset that follows the Traj2d conventions."""
2424
global_attrs = self.ds.attrs
2525

@@ -70,7 +70,7 @@ def to_2d(self, obs_dimname='obs'):
7070

7171
# trajectory vars
7272
'time':
73-
xr.DataArray(dims=["trajectory", obs_dimname],
73+
xr.DataArray(dims=["trajectory", obs_dim],
7474
data=array_time,
7575
attrs={
7676
"standard_name": "time",
@@ -81,7 +81,7 @@ def to_2d(self, obs_dimname='obs'):
8181

8282
# now add all "normal" variables
8383
# NOTE: for now, we only consider scalar vars; if we want to consider more complex vars (e.g., spectra), this will need updated
84-
# NOTE: such an update would typically need to look at the dims of the variable, and if there are additional dims to obs_dimname, create a higer dim variable
84+
# NOTE: such an update would typically need to look at the dims of the variable, and if there are additional dims to obs_dim, create a higer dim variable
8585

8686
for crrt_data_var in self.ds.data_vars:
8787
attrs = self.ds[crrt_data_var].attrs
@@ -90,9 +90,9 @@ def to_2d(self, obs_dimname='obs'):
9090
continue
9191

9292
if len(self.ds[crrt_data_var].dims
93-
) != 1 or self.ds[crrt_data_var].dims[0] != self.obs_dimname:
93+
) != 1 or self.ds[crrt_data_var].dims[0] != self.obs_dim:
9494
raise ValueError(
95-
f"data_vars element {crrt_data_var} has dims {self.ds[crrt_data_var].dims}, expected {(self.obs_dimname,)}"
95+
f"data_vars element {crrt_data_var} has dims {self.ds[crrt_data_var].dims}, expected {(self.obs_dim,)}"
9696
)
9797

9898
crrt_var = np.full((nbr_trajectories, longest_trajectory), np.nan)
@@ -119,7 +119,7 @@ def to_2d(self, obs_dimname='obs'):
119119
crrt_data_var = "lat"
120120

121121
ds_converted_to_traj2d[crrt_data_var] = \
122-
xr.DataArray(dims=["trajectory", obs_dimname],
122+
xr.DataArray(dims=["trajectory", obs_dim],
123123
data=crrt_var,
124124
attrs=attrs)
125125

trajan/traj.py

+20-20
Original file line numberDiff line numberDiff line change
@@ -33,10 +33,10 @@ def detect_tx_dim(ds):
3333
raise ValueError("Could not determine x / lon variable")
3434

3535

36-
def detect_time_dim(ds, obs_dimname):
37-
logger.debug(f'Detecting time-dimension for "{obs_dimname}"..')
36+
def detect_time_dim(ds, obs_dim):
37+
logger.debug(f'Detecting time-dimension for "{obs_dim}"..')
3838
for v in ds.variables:
39-
if obs_dimname in ds[v].dims and 'time' in v:
39+
if obs_dim in ds[v].dims and 'time' in v:
4040
return v
4141

4242
raise ValueError("no time dimension detected")
@@ -50,12 +50,12 @@ class Traj:
5050

5151
__gcrs__: pyproj.CRS
5252

53-
def __init__(self, ds, obs_dimname, time_varname):
53+
def __init__(self, ds, obs_dim, time_varname):
5454
self.ds = ds
5555
self.__plot__ = None
5656
self.__animate__ = None
5757
self.__gcrs__ = pyproj.CRS.from_epsg(4326)
58-
self.obs_dimname = obs_dimname # dimension along which time increases
58+
self.obs_dim = obs_dim # dimension along which time increases
5959
self.time_varname = time_varname
6060

6161
def __repr__(self):
@@ -528,18 +528,18 @@ def distance_to_next(self):
528528

529529
lon = self.ds.lon
530530
lat = self.ds.lat
531-
lenobs = self.ds.sizes[self.obs_dimname]
532-
lonfrom = lon.isel({self.obs_dimname: slice(0, lenobs - 1)})
533-
latfrom = lat.isel({self.obs_dimname: slice(0, lenobs - 1)})
534-
lonto = lon.isel({self.obs_dimname: slice(1, lenobs)})
535-
latto = lat.isel({self.obs_dimname: slice(1, lenobs)})
531+
lenobs = self.ds.sizes[self.obs_dim]
532+
lonfrom = lon.isel({self.obs_dim: slice(0, lenobs - 1)})
533+
latfrom = lat.isel({self.obs_dim: slice(0, lenobs - 1)})
534+
lonto = lon.isel({self.obs_dim: slice(1, lenobs)})
535+
latto = lat.isel({self.obs_dim: slice(1, lenobs)})
536536
geod = pyproj.Geod(ellps='WGS84')
537537
azimuth_forward, a2, distance = geod.inv(lonfrom, latfrom, lonto,
538538
latto)
539539

540540
distance = xr.DataArray(distance, coords=lonfrom.coords, dims=lon.dims)
541-
distance = xr.concat((distance, distance.isel({self.obs_dimname: -1})),
542-
dim=self.obs_dimname) # repeating last time step to
541+
distance = xr.concat((distance, distance.isel({self.obs_dim: -1})),
542+
dim=self.obs_dim) # repeating last time step to
543543
return distance
544544

545545
def azimuth_to_next(self):
@@ -560,11 +560,11 @@ def azimuth_to_next(self):
560560
# TODO: method is almost duplicate of "distance_to_next" above
561561
lon = self.ds.lon
562562
lat = self.ds.lat
563-
lenobs = self.ds.dims[self.obs_dimname]
564-
lonfrom = lon.isel({self.obs_dimname: slice(0, lenobs - 1)})
565-
latfrom = lat.isel({self.obs_dimname: slice(0, lenobs - 1)})
566-
lonto = lon.isel({self.obs_dimname: slice(1, lenobs)})
567-
latto = lat.isel({self.obs_dimname: slice(1, lenobs)})
563+
lenobs = self.ds.dims[self.obs_dim]
564+
lonfrom = lon.isel({self.obs_dim: slice(0, lenobs - 1)})
565+
latfrom = lat.isel({self.obs_dim: slice(0, lenobs - 1)})
566+
lonto = lon.isel({self.obs_dim: slice(1, lenobs)})
567+
latto = lat.isel({self.obs_dim: slice(1, lenobs)})
568568
geod = pyproj.Geod(ellps='WGS84')
569569
azimuth_forward, a2, distance = geod.inv(lonfrom, latfrom, lonto,
570570
latto)
@@ -573,8 +573,8 @@ def azimuth_to_next(self):
573573
coords=lonfrom.coords,
574574
dims=lon.dims)
575575
azimuth_forward = xr.concat(
576-
(azimuth_forward, azimuth_forward.isel({self.obs_dimname: -1})),
577-
dim=self.obs_dimname) # repeating last time step to
576+
(azimuth_forward, azimuth_forward.isel({self.obs_dim: -1})),
577+
dim=self.obs_dim) # repeating last time step to
578578
return azimuth_forward
579579

580580
def velocity_components(self):
@@ -890,7 +890,7 @@ def condense_obs(self) -> xr.Dataset:
890890
"""
891891

892892
@abstractmethod
893-
def to_2d(self, obs_dimname='obs') -> xr.Dataset:
893+
def to_2d(self, obs_dim='obs') -> xr.Dataset:
894894
"""
895895
Convert dataset into a 2D dataset from.
896896
"""

trajan/traj1d.py

+14-14
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,8 @@ class Traj1d(Traj):
1414
A structured dataset, where each trajectory is always given at the same times. Typically the output from a model or from a gridded dataset.
1515
"""
1616

17-
def __init__(self, ds, obs_dimname, time_varname):
18-
super().__init__(ds, obs_dimname, time_varname)
17+
def __init__(self, ds, obs_dim, time_varname):
18+
super().__init__(ds, obs_dim, time_varname)
1919

2020
def timestep(self):
2121
"""Time step between observations in seconds."""
@@ -28,14 +28,14 @@ def is_1d(self):
2828
def is_2d(self):
2929
return False
3030

31-
def to_2d(self, obs_dimname='obs'):
31+
def to_2d(self, obs_dim='obs'):
3232
ds = self.ds.copy()
3333
time = ds[self.time_varname].rename({
34-
self.time_varname: obs_dimname
34+
self.time_varname: obs_dim
3535
}).expand_dims(dim={'trajectory': ds.sizes['trajectory']})
36-
ds = ds.rename({self.time_varname: obs_dimname})
36+
ds = ds.rename({self.time_varname: obs_dim})
3737
ds[self.time_varname] = time
38-
ds[obs_dimname] = np.arange(0, ds.sizes[obs_dimname])
38+
ds[obs_dim] = np.arange(0, ds.sizes[obs_dim])
3939

4040
return ds
4141

@@ -101,8 +101,8 @@ def skill(self, other, method='liu-weissberg', **kwargs):
101101
)
102102

103103
diff = np.max(
104-
np.abs((self.ds[self.obs_dimname] -
105-
other[other.traj.obs_dimname]).astype('timedelta64[s]').astype(
104+
np.abs((self.ds[self.obs_dim] -
105+
other[other.traj.obs_dim]).astype('timedelta64[s]').astype(
106106
np.float64)))
107107

108108
if not np.isclose(diff, 0):
@@ -112,11 +112,11 @@ def skill(self, other, method='liu-weissberg', **kwargs):
112112

113113
s = np.zeros((self.ds.sizes['trajectory']), dtype=np.float32)
114114

115-
# ds = self.ds.dropna(dim=self.obs_dimname)
116-
# other = other.dropna(dim=other.traj.obs_dimname)
115+
# ds = self.ds.dropna(dim=self.obs_dim)
116+
# other = other.dropna(dim=other.traj.obs_dim)
117117

118-
ds = self.ds.transpose('trajectory', self.obs_dimname, ...)
119-
other = other.transpose('trajectory', other.traj.obs_dimname, ...)
118+
ds = self.ds.transpose('trajectory', self.obs_dim, ...)
119+
other = other.transpose('trajectory', other.traj.obs_dim, ...)
120120

121121
lon0 = ds.traj.tlon
122122
lat0 = ds.traj.tlat
@@ -165,9 +165,9 @@ def gridtime(self, times, time_varname=None, round=True):
165165

166166
ds = self.ds
167167

168-
if self.obs_dimname != time_varname:
168+
if self.obs_dim != time_varname:
169169
ds = ds.rename({
170-
self.obs_dimname: time_varname
170+
self.obs_dim: time_varname
171171
}).set_index({time_varname: time_varname})
172172

173173
_, ui = np.unique(ds[time_varname], return_index=True)

0 commit comments

Comments
 (0)