Source code for mhkit.dolfyn.rotate.api

from . import vector as r_vec
from . import awac as r_awac
from . import signature as r_sig
from . import rdi as r_rdi
from .base import _make_model
import numpy as np
import xarray as xr
import warnings


# The 'rotation chain'
rc = ["beam", "inst", "earth", "principal"]

rot_module_dict = {
    # Nortek instruments
    "vector": r_vec,
    "awac": r_awac,
    "signature": r_sig,
    "ad2cp": r_sig,
    # TRDI instruments
    "rdi": r_rdi,
}


[docs] def rotate2(ds, out_frame="earth", inplace=True): """ Rotate a dataset to a new coordinate system. Parameters ---------- ds : xr.Dataset The dolfyn dataset (ADV or ADCP) to rotate. out_frame : string {'beam', 'inst', 'earth', 'principal'} The coordinate system to rotate the data into. inplace : bool When True ``ds`` is modified. When False a copy is returned. Default = True Returns ------- ds : xarray.Dataset or None Returns a new rotated dataset **when ``inplace=False``**, otherwise returns None. Notes ----- - This function rotates all variables in ``ds.attrs['rotate_vars']``. - In order to rotate to the 'principal' frame, a value should exist for ``ds.attrs['principal_heading']``. The function :func:`calc_principal_heading <dolfyn.calc_principal_heading>` is recommended for this purpose, e.g.: ds.attrs['principal_heading'] = dolfyn.calc_principal_heading(ds['vel'].mean(range)) where here we are using the depth-averaged velocity to calculate the principal direction. """ # Create and return deep copy if not writing "in place" if not inplace: ds = ds.copy(deep=True) csin = ds.coord_sys.lower() if csin == "ship": csin = "inst" # Returns True/False if head2inst_rotmat has been set/not-set. r_vec._check_inst2head_rotmat(ds) if out_frame == "principal" and csin != "earth": warnings.warn( "You are attempting to rotate into the 'principal' " "coordinate system, but the dataset is in the {} " "coordinate system. Be sure that 'principal_heading' is " "defined based on the earth coordinate system.".format(csin) ) rmod = None for ky in rot_module_dict: if ky in _make_model(ds): rmod = rot_module_dict[ky] break if rmod is None: raise ValueError( "Rotations are not defined for " "instrument '{}'.".format(_make_model(ds)) ) # Get the 'indices' of the rotation chain try: iframe_in = rc.index(csin) except ValueError: raise Exception( "The coordinate system of the input " "dataset, '{}', is invalid.".format(ds.coord_sys) ) try: iframe_out = rc.index(out_frame.lower()) except ValueError: raise Exception( "The specified output coordinate system " "is invalid, please select one of: 'beam', 'inst', " "'earth', 'principal'." ) if iframe_out == iframe_in: print("Data is already in the {} coordinate system".format(out_frame)) if iframe_out > iframe_in: reverse = False else: reverse = True while ds.coord_sys.lower() != out_frame.lower(): csin = ds.coord_sys if csin == "ship": csin = "inst" inow = rc.index(csin) if reverse: func = getattr(rmod, "_" + rc[inow - 1] + "2" + rc[inow]) else: func = getattr(rmod, "_" + rc[inow] + "2" + rc[inow + 1]) ds = func(ds, reverse=reverse) if not inplace: return ds
[docs] def calc_principal_heading(vel, tidal_mode=True): """ Compute the principal angle of the horizontal velocity. Parameters ---------- vel : np.ndarray (2,...,Nt), or (3,...,Nt) The 2D or 3D velocity array (3rd-dim is ignored in this calculation) tidal_mode : bool If true, range is set from 0 to +/-180 degrees. If false, range is 0 to 360 degrees. Default = True Returns ------- p_heading : float or ndarray The principal heading in degrees clockwise from North. Notes ----- When tidal_mode=True, this tool calculates the heading that is aligned with the bidirectional flow. It does so following these steps: 1. rotates vectors with negative velocity by 180 degrees 2. then doubles those angles to make a complete circle again 3. computes a mean direction from this, and halves that angle (to undo the doubled-angles in step 2) 4. The returned angle is forced to be between 0 and 180. So, you may need to add 180 to this if you want your positive direction to be in the western-half of the plane. Otherwise, this function simply computes the average direction using a vector method. """ dt = vel[0] + vel[1] * 1j if tidal_mode: # Flip all vectors that are below the x-axis dt[dt.imag <= 0] *= -1 # Now double the angle, so that angles near pi and 0 get averaged # together correctly: dt *= np.exp(1j * np.angle(dt)) dt = np.ma.masked_invalid(dt) # Divide the angle by 2 to remove the doubling done on the previous # line. pang = np.angle(np.nanmean(dt, -1, dtype=np.complex128)) / 2 else: pang = np.angle(np.nanmean(dt, -1)) return np.round((90 - np.rad2deg(pang)), decimals=4)
[docs] def set_declination(ds, declin, inplace=True): """ Set the magnetic declination Parameters ---------- ds : xarray.Dataset or :class:`dolfyn.velocity.Velocity` The input dataset or velocity class declination : float The value of the magnetic declination in degrees (positive values specify that Magnetic North is clockwise from True North) inplace : bool When True ``ds`` is modified. When False a copy is returned. Default = True Returns ------- ds : xarray.Dataset or None Returns a new dataset with declination set **when ``inplace=False``**, otherwise returns None. Notes ----- This function modifies the data object in the following ways: - If the dataset is in the *earth* reference frame at the time of setting declination, it will be rotated into the "*True-East*, *True-North*, Up" (hereafter, ETU) coordinate system - ``dat['orientmat']`` is modified to be an ETU to instrument (XYZ) rotation matrix (rather than the magnetic-ENU to XYZ rotation matrix). Therefore, all rotations to/from the 'earth' frame will now be to/from this ETU coordinate system. - The value of the specified declination will be stored in ``dat.attrs['declination']`` - ``dat['heading']`` is adjusted for declination (i.e., it is relative to True North). - If ``dat.attrs['principal_heading']`` is set, it is adjusted to account for the orientation of the new 'True' earth coordinate system (i.e., calling set_declination on a data object in the principal coordinate system, then calling dat.rotate2('earth') will yield a data object in the new 'True' earth coordinate system) """ # Create and return deep copy if not writing "in place" if not inplace: ds = ds.copy(deep=True) if "declination" in ds.attrs: angle = declin - ds.attrs.pop("declination") else: angle = declin cd = np.cos(-np.deg2rad(angle)) sd = np.sin(-np.deg2rad(angle)) # The ordering is funny here because orientmat is the # transpose of the inst->earth rotation matrix: Rdec = np.array([[cd, -sd, 0], [sd, cd, 0], [0, 0, 1]]) if ds.coord_sys == "earth": rotate2earth = True rotate2(ds, "inst", inplace=True) else: rotate2earth = False ds["orientmat"].values = np.einsum( "kj...,ij->ki...", ds["orientmat"].values, Rdec, ) if "heading" in ds: ds["heading"] += angle if rotate2earth: rotate2(ds, "earth", inplace=True) if "principal_heading" in ds.attrs: ds.attrs["principal_heading"] += angle ds.attrs["declination"] = declin ds.attrs["declination_in_orientmat"] = 1 # logical if not inplace: return ds
[docs] def set_inst2head_rotmat(ds, rotmat, inplace=True): """ Set the instrument to head rotation matrix for the Nortek ADV if it hasn't already been set through a '.userdata.json' file. Parameters ---------- ds : xarray.Dataset The data set to assign inst2head_rotmat rotmat : float 3x3 rotation matrix inplace : bool When True ``ds`` is modified. When False a copy is returned. Default = True Returns ------- ds : xarray.Dataset or None Returns a new dataset with inst2head_rotmat set **when ``inplace=False``**, otherwise returns None. Notes ----- If the data object is in earth or principal coords, it is first rotated to 'inst' before assigning inst2head_rotmat, it is then rotated back to the coordinate system in which it was input. This way the inst2head_rotmat gets applied correctly (in inst coordinate system). """ # Create and return deep copy if not writing "in place" if not inplace: ds = ds.copy(deep=True) if not ds.inst_model.lower() == "vector": raise Exception( "Setting 'inst2head_rotmat' is only supported " "for Nortek Vector ADVs." ) if ds.get("inst2head_rotmat", None) is not None: raise Exception( "You are setting 'inst2head_rotmat' after it has already " "been set. You can only set it once." ) csin = ds.coord_sys if csin not in ["inst", "beam"]: rotate2(ds, "inst", inplace=True) ds["inst2head_rotmat"] = xr.DataArray( np.array(rotmat), dims=["x1", "x2"], coords={"x1": [1, 2, 3], "x2": [1, 2, 3]} ) ds.attrs["inst2head_rotmat_was_set"] = 1 # logical # Note that there is no validation that the user doesn't # change `ds.attrs['inst2head_rotmat']` after calling this # function. if not csin == "beam": # csin not 'beam', then we're in inst ds = r_vec._rotate_inst2head(ds) if csin not in ["inst", "beam"]: rotate2(ds, csin, inplace=True) if not inplace: return ds