Source code for pyrfu.pyrf.filt

#!/usr/bin/env python
# -*- coding: utf-8 -*-

# 3rd party imports
import numpy as np
import xarray as xr
from scipy import signal

# local imports
from pyrfu.pyrf.calc_fs import calc_fs

__author__ = "Louis Richard"
__email__ = "louisr@irfu.se"
__copyright__ = "Copyright 2020-2026"
__license__ = "MIT"
__version__ = "2.4.2"
__status__ = "Prototype"


# noinspection PyTupleAssignmentBalance
def _ellip_coefficients(f_min, f_max, order):
    num1, den1, num2, den2 = [None] * 4

    # fact defines the width between stopband and passband
    r_p, r_s, fact = 0.5, 60, 1.1

    if f_min == 0:
        if order == -1:
            order, f_max = signal.ellipord(
                f_max,
                np.min([f_max * fact, 0.9999]),
                r_p,
                r_s,
            )

        num1, den1 = signal.ellip(order, r_p, r_s, f_max, btype="lowpass")
    elif f_max == 0:
        if order == -1:
            order, f_min = signal.ellipord(
                f_min,
                np.min([f_min * fact, 0.9999]),
                r_p,
                r_s,
            )

        num1, den1 = signal.ellip(order, r_p, r_s, f_min, btype="highpass")
    else:
        if order == -1:
            order1, f_max = signal.ellipord(
                f_max,
                np.min([f_max * 1.3, 0.9999]),
                0.5,
                60,
            )
            order2, f_min = signal.ellipord(f_min, f_min * 0.75, 0.5, 60)
        else:
            order1, order2 = [order, order]

        num1, den1 = signal.ellip(order1, 0.5, 60, f_max)
        num2, den2 = signal.ellip(order2, 0.5, 60, f_min, btype="highpass")

    return num1, den1, num2, den2


[docs]def filt(inp, f_min: float = 0.0, f_max: float = 1.0, order: int = -1): r"""Filters input quantity. Parameters ---------- inp : xarray.DataArray Time series of the variable to filter. f_min : float, Optional Lower limit of the frequency range. Default is 0. (Highpass filter). f_max : float, Optional Upper limit of the frequency range. Default is 1. (Highpass filter). order : int, Optional Order of the elliptic filter. Default is -1. Returns ------- out : xarray.DataArray Time series of the filtered signal. Examples -------- >>> from pyrfu import mms, pyrf Time interval >>> tint = ["2017-07-18T13:03:34.000", "2017-07-18T13:07:00.000"] Spacecraft index >>> mms_id = 1 Load magnetic and electric fields >>> b_xyz = mms.get_data("B_gse_fgm_brst_l2", tint, mms_id) >>> e_xyz = mms.get_data("E_gse_edp_brst_l2", tint, mms_id) Convert E to field aligned coordinates >>> e_xyzfac = pyrf.convert_fac(e_xyz, b_xyz, [1,0,0]) Bandpass filter E waveform >>> e_xyzfac_hf = pyrf.filt(e_xyzfac, 4, 0, 3) >>> e_xyzfac_lf = pyrf.filt(e_xyzfac, 0, 4, 3) """ assert isinstance(inp, xr.DataArray), "inp must be a xarray.DataArray" # Data of the input inp_data = inp.data.astype(np.float64) assert isinstance(f_min, (int, float)), "f_min must be int or float" assert isinstance(f_max, (int, float)), "f_max must be int or float" assert isinstance(order, (int, float)), "order must be int or float" # Calculate the sampling frequency and normalize the cutoff frequencies # to the Nyquist frequency f_samp = calc_fs(inp) f_min, f_max = [f_min / (f_samp / 2.0), f_max / (f_samp / 2.0)] f_max = np.min([f_max, 1.0]) # Parameters of the elliptic filter. fact defines the width between # stopband and passband # r_pass, r_stop, fact = [0.5, 60, 1.1] num1, den1, num2, den2 = _ellip_coefficients(f_min, f_max, order) if len(inp_data.shape) == 1: inp_data = inp_data[:, np.newaxis] n_cols = 1 elif len(inp_data.shape) == 2: n_cols = inp_data.shape[1] elif len(inp_data.shape) == 3: inp_data = inp_data.reshape(inp_data.shape[0], -1) n_cols = inp_data.shape[1] else: raise ValueError("inp must be 1D, 2D or 3D") out_data = np.zeros(inp_data.shape, dtype=inp_data.dtype) for i_col in range(n_cols): # use different padtype and padlen for consistency with MATLAB # (and to not drive me crazy trying to figure out the differences) padtype = "odd" padlen = 3 * (max(len(num1), len(den1)) - 1) out_data[:, i_col] = signal.filtfilt( num1, den1, inp_data[:, i_col], padtype=padtype, padlen=padlen ) if num2 is not None and den2 is not None: padtype = "odd" padlen = 3 * (max(len(num2), len(den2)) - 1) out_data[:, i_col] = signal.filtfilt( num2, den2, out_data[:, i_col], padtype=padtype, padlen=padlen, ) if inp_data.shape[1] == 1: out_data = out_data[:, 0] elif len(inp.shape) == 3: out_data = out_data.reshape(inp.shape) out = xr.DataArray( out_data, coords=inp.coords, dims=inp.dims, attrs=inp.attrs, ) return out