Skip to content

Commit

Permalink
Fix fahrenheit deltas
Browse files Browse the repository at this point in the history
  • Loading branch information
aulemahal committed Jun 26, 2024
1 parent be6fe11 commit 06404e2
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 4 deletions.
6 changes: 5 additions & 1 deletion CHANGES.rst
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,11 @@ Changelog

v0.51.0 (unreleased)
--------------------
Contributors to this version: Trevor James Smith (:user:`Zeitsperre`).
Contributors to this version: Trevor James Smith (:user:`Zeitsperre`), Pascal Bourgault (:user:`aulemahal`).

Bug fixes
^^^^^^^^^
* Units of degree-days computations with Fahrenheit input fixed to yield "°R d". Added a new ``core.units.ensure_absolute_temperature`` method to convert from delta to absolute temperatures (:issue:`1789`, :pull:`1804`).

Internal changes
^^^^^^^^^^^^^^^^
Expand Down
2 changes: 2 additions & 0 deletions tests/test_units.py
Original file line number Diff line number Diff line change
Expand Up @@ -332,6 +332,8 @@ def index(
("", "sum", "count", 365, "d"),
("kg m-2", "var", "var", 0, "kg2 m-4"),
("°C", "argmax", "doymax", 0, ""),
("°C", "sum", "integral", 365, "K d"),
("°F", "sum", "integral", 365, "d °R"), # not sure why the order is different
],
)
def test_to_agg_units(in_u, opfunc, op, exp, exp_u):
Expand Down
30 changes: 27 additions & 3 deletions xclim/core/units.py
Original file line number Diff line number Diff line change
Expand Up @@ -463,6 +463,25 @@ def infer_sampling_units(
return out


DELTA_ABSOLUTE_TEMP = {
units.delta_degC: units.kelvin,
units.delta_degF: units.rankine,
}


def ensure_absolute_temperature(units: str):
"""Convert temperature units to their absolute counterpart, assuming they represented a difference (delta).
Celsius becomes Kelvin, Fahrenheit becomes Rankine. Does nothing for other units.
"""
a = str2pint(units)
# ensure a delta pint unit
a = a - 0 * a
if a.units in DELTA_ABSOLUTE_TEMP:
return pint2cfunits(DELTA_ABSOLUTE_TEMP[a.units])
return units


def to_agg_units(
out: xr.DataArray, orig: xr.DataArray, op: str, dim: str = "time"
) -> xr.DataArray:
Expand Down Expand Up @@ -526,11 +545,16 @@ def to_agg_units(
>>> degdays.units
'K d'
"""
if op in ["amin", "min", "amax", "max", "mean", "std", "sum"]:
if op in ["amin", "min", "amax", "max", "mean", "sum"]:
out.attrs["units"] = orig.attrs["units"]

elif op in ["std"]:
out.attrs["units"] = ensure_absolute_temperature(orig.attrs["units"])

elif op in ["var"]:
out.attrs["units"] = pint2cfunits(str2pint(orig.units) ** 2)
out.attrs["units"] = pint2cfunits(
str2pint(ensure_absolute_temperature(orig.units)) ** 2
)

elif op in ["doymin", "doymax"]:
out.attrs.update(
Expand All @@ -539,7 +563,7 @@ def to_agg_units(

elif op in ["count", "integral"]:
m, freq_u_raw = infer_sampling_units(orig[dim])
orig_u = str2pint(orig.units)
orig_u = str2pint(ensure_absolute_temperature(orig.units))
freq_u = str2pint(freq_u_raw)
out = out * m

Expand Down

0 comments on commit 06404e2

Please sign in to comment.