{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# 3. Beamforming for location\n", "\n", "This notebook computes the beamformed network response and demonstrates how to use it for the detection and location of an eathquake. The workflow consists of computing waveform features first (envelopes, kurtosis, etc.) and, then, to shift-and-stack them according to the previously computed travel times.\n", "\n", "## Contents\n", "\n", "* [Compute waveform features](#compute-waveform-features)\n", "\n", "* [Read travel times](#read-travel-times)\n", "\n", "* [Show an example waveform](#show-an-example-envelope)" ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [], "source": [ "import beampower as bp\n", "import glob\n", "import h5py as h5\n", "import numpy as np\n", "import os\n", "import pandas as pd\n", "import tqdm\n", "\n", "from matplotlib import pyplot as plt\n", "from scipy import signal, stats\n", "from obspy import read, read_inventory, UTCDateTime\n", "from obspy.geodetics.base import locations2degrees, degrees2kilometers\n", "try:\n", " from scipy.stats import median_abs_deviation as scimad\n", "except ImportError:\n", " from scipy.stats import median_absolute_deviation as scimad" ] }, { "cell_type": "code", "execution_count": 2, "metadata": {}, "outputs": [], "source": [ "%load_ext autoreload\n", "%autoreload 2" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Compute waveform features\n", "\n", "The underlying paradigm of the beamforming method is that when using the exact travel times to align the traces, they sum up to the largest amplitude beam. However, the waves emitted by a double-couple source may sum up to zero due to antisymmetries in the radiated wavefield. Furthermore, imprecisions in the velocity model and the finiteness of the grid often mean that the exact travel times are not available in the pre-computed travel time grid.\n", "\n", "\n", "To address the aforementionned drawbacks, we compute positive-valued *waveform features* to handle antisymmetries in the wavefield and we make sure that they are smooth enough in time to be correctly aligned by at least one set of pre-computed travel times (think of trying, unsuccesfully, to align Diracs with a finite resolution travel time grid!).\n", "\n", "Here, we use the waveform envelopes. The envelope is computed from the analytical signal (Hilbert transform). To avoid noisy stations from dominating the beams, we normalize each trace by its median absolute deviation. The envelopes are clipped above $10^5$ times the mean absolute deviation in order to avoid spikes or proximal transient noise to corrupt the beams." ] }, { "cell_type": "code", "execution_count": 3, "metadata": {}, "outputs": [], "source": [ "def envelope(x):\n", " \"\"\"Envelope transformation.\n", "\n", " Calculate the envelope of the input one-dimensional signal with the Hilbert\n", " transform. The data is normalized by the mean absolute deviation over the\n", " entire signal window first. The envelope is clipped at a maximum of 10^5\n", " times the mad of the envelope.\n", " \n", " Arguments\n", " ---------\n", " x: array-like\n", " The input one-dimensional signal.\n", " \n", " Returns\n", " -------\n", " array-like\n", " The envelope of x with same shape than x.\n", " \"\"\"\n", "\n", " # Envelope\n", " x = np.abs(signal.hilbert(x))\n", " \n", " # Normalization\n", " x_mad = scimad(x)\n", " x_mad = 1.0 if x_mad == 0.0 else x_mad\n", " x = (x - np.median(x)) / x_mad\n", " \n", " # Clip\n", " x_max = 10.0 ** 5.0\n", " return x.clip(None, x_max)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Read the network metadata into the `network` `pandas.DataFrame`. We will use `network` and its list of stations to order consistently the `waveform_features` and `travel_times` arrays." ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", " | code | \n", "longitude | \n", "latitude | \n", "elevation | \n", "depth | \n", "
---|---|---|---|---|---|
0 | \n", "DC06 | \n", "30.265751 | \n", "40.616718 | \n", "555.0 | \n", "-0.555 | \n", "
1 | \n", "DC07 | \n", "30.242170 | \n", "40.667080 | \n", "164.0 | \n", "-0.164 | \n", "
2 | \n", "DC08 | \n", "30.250130 | \n", "40.744438 | \n", "162.0 | \n", "-0.162 | \n", "
3 | \n", "DD06 | \n", "30.317770 | \n", "40.623539 | \n", "182.0 | \n", "-0.182 | \n", "
4 | \n", "DE07 | \n", "30.411539 | \n", "40.679661 | \n", "40.0 | \n", "-0.040 | \n", "
5 | \n", "DE08 | \n", "30.406469 | \n", "40.748562 | \n", "31.0 | \n", "-0.031 | \n", "
6 | \n", "SAUV | \n", "30.327200 | \n", "40.740200 | \n", "170.0 | \n", "-0.170 | \n", "
7 | \n", "SPNC | \n", "30.308300 | \n", "40.686001 | \n", "190.0 | \n", "-0.190 | \n", "