From b6d66041dfc36ee5903c69ec220b6f6c76f8fe02 Mon Sep 17 00:00:00 2001 From: Mark Neubauer Date: Thu, 25 Apr 2024 12:00:03 -0500 Subject: [PATCH] updates --- _sources/Project_02.md | 4 +- .../Project_02_GravitationalWaves.ipynb | 563 ++++++++++++++++++ _toc.yml | 2 + 3 files changed, 568 insertions(+), 1 deletion(-) create mode 100644 _sources/projects/Project_02_GravitationalWaves.ipynb diff --git a/_sources/Project_02.md b/_sources/Project_02.md index 35dd250..52e7ed4 100644 --- a/_sources/Project_02.md +++ b/_sources/Project_02.md @@ -3,4 +3,6 @@ ## Select from one of the following projects: * {doc}`projects/Project_02_NuclearGeometryQGP` -* {doc}`projects/Project_02_AnisotropyQGP` \ No newline at end of file +* {doc}`projects/Project_02_AnisotropyQGP` +* {doc}`projects/Project_02_AberratedImages` +* {doc}`projects/Project_02_GravitationalWaves` \ No newline at end of file diff --git a/_sources/projects/Project_02_GravitationalWaves.ipynb b/_sources/projects/Project_02_GravitationalWaves.ipynb new file mode 100644 index 0000000..7f8deaa --- /dev/null +++ b/_sources/projects/Project_02_GravitationalWaves.ipynb @@ -0,0 +1,563 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": { + "id": "V5ppSGgISDAy" + }, + "source": [ + "# Detection of Gravitational Waves" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "
" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "%matplotlib inline\n", + "import matplotlib.pyplot as plt\n", + "import matplotlib.mlab as mlab\n", + "# Standard python numerical analysis imports:\n", + "import numpy as np\n", + "from scipy import signal\n", + "from scipy.interpolate import interp1d\n", + "from scipy.signal import butter, filtfilt, iirdesign, zpk2tf, freqz\n", + "import warnings\n", + "warnings.filterwarnings('ignore')" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "1jeD_lEeqLBh" + }, + "source": [ + "## Overview\n", + "\n", + "Gravitational waves are predicted by Einstein's general theory of relativity in 1916, which posits that gravity warps space and time, causing objects traveling through it to follow a curved path. By analogy to electromagnetism, time variation of the mass quadrupole moment of the source is expected to lead to transverse waves of spatial strain. The existence of GW was first demonstrated in 1974 by the discovery of a binary system composed of a pulsar in orbit around a neutron star by Hulse and Taylor [[1]](https://ui.adsabs.harvard.edu/abs/1975ApJ...195L..51H/abstract). However, direct detections of GW did not arrive until 2016. In that year, The LIGO (The Laser Interferometer Gravitational-Wave Observatory) collaboration reported the first direct detection of GW from a binary black hole system merging to form a single black hole [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102). The observations reported in this paper and futher GW detections wprovide new tests of generay relativity in its strong-field regime, and GW observations have become an important new means to learn about the Universe.\n", + "\n", + "In this project, you will reproduce the results reported in [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102) with LIGO open data. You will analyze a particular GW event GW150914 (the first GW ever detected). \n", + "\n", + "You will also explore machine learning methods applied to simulated LIGO data for the detection of GWs. " + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "SprcSrHttoNZ" + }, + "source": [ + "## Data Sources\n", + "\n", + "This project is based on the open data from the LIGO Scientific Collaboration. The data is hosted by the Gravitational Wave Open Science Center (GWOSC), formerly known as the LIGO Open Science Center, was created to provide public access to gravitational-wave data products. The collaborations running LIGO, Virgo, GEO600, and KAGRA have all agreed to use GWOSC services as the primary access points for public data products. This collaborative approach benefits users by creating a uniform interface to access data from multiple observatories, and provides cost savings to the various observatories by sharing the tools, services, and human resources.\n", + "\n", + "GWOSC Data Repository: https://gwosc.org/data\n", + "\n", + "You will also used simulated GW challenge data from https://www.kaggle.com/competitions/g2net-gravitational-wave-detection/data to explore ML methods for GW detection. " + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "bVGeqweYvvh3" + }, + "source": [ + "## Questions" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "rYEE9PEHw772" + }, + "source": [ + "### Question 01\n", + "\n", + " What are gravity waves? How are they generated? How fast do they propogate? What can they tell us about the early universe?" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "-F8WubbKzlP6" + }, + "source": [ + "### Question 02\n", + "\n", + "How are they detected? Describe the LIGO experiment and the three detector sites (). " + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "a9VgzmClywsu" + }, + "source": [ + "## Filtering a TimeSeries to detect gravitational waves\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Importing and Setup" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You will need to install and import the `gwpy` module for data acquisition and helper functions\n", + "\n", + "Please note, you will have to restart runtime to run your code after pip installing the `gwpy` module." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "!pip install --ignore-installed --no-cache-dir gwpy" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The raw ‘strain’ output of the LIGO detectors is recorded as a `TimeSeries` with contributions from a large number of known and unknown noise sources, as well as possible gravitational wave signals.\n", + "\n", + "In order to uncover a real signal we need to filter out noises that otherwise hide the signal in the data. We can do this by using the `gwpy.signal` module to design a digital filter to cut out low and high frequency noise, as well as notch out fixed frequencies polluted by known artefacts.\n", + "\n", + "First we download the raw LIGO-Hanford (H1) strain data from the GWOSC public archive:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "from gwpy.timeseries import TimeSeries\n", + "hdata = TimeSeries.fetch_open_data('H1', 1126259446, 1126259478)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Next we can design a zero-pole-gain (ZPK) filter to remove the extranious noise.\n", + "\n", + "First we import the `gwpy.signal.filter_design` module and create a bandpass() filter to remove both low and high frequency content" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "from gwpy.signal import filter_design\n", + "bp = filter_design.bandpass(50, 250, hdata.sample_rate)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Now we want to combine the bandpass with a series of `notch()` filters, so we create those for the first three harmonics of the 60 Hz AC mains power:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "notches = [filter_design.notch(line, hdata.sample_rate) for\n", + " line in (60, 120, 180)]" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "and concatenate each of our filters together to create a single ZPK model:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "zpk = filter_design.concatenate_zpks(bp, *notches)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Now, we can apply our combined filter to the data, using `filtfilt=True` to filter both backwards and forwards to preserve the correct phase at all frequencies" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "hfilt = hdata.filter(zpk, filtfilt=True)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Note: The filter_design methods return digital filters by default, so we apply them using TimeSeries.filter. If we had analogue filters (perhaps by passing `analog=True` to the filter design method), the easiest application would be `TimeSeries.zpk`" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The filter_design methods return infinite impulse response filters by default, which, when applied, corrupt a small amount of data at the beginning and the end of our original `TimeSeries`. We can discard those data using the `crop()` method (for consistency we apply this to both data series):" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "hdata = hdata.crop(*hdata.span.contract(1))\n", + "hfilt = hfilt.crop(*hfilt.span.contract(1))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Finally, we can `plot()` the original and filtered data, adding some code to prettify the figure:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "from gwpy.plot import Plot\n", + "plot = Plot(hdata, hfilt, figsize=[12, 6], separate=True, sharex=True,\n", + " color='gwpy:ligo-hanford')\n", + "ax1, ax2 = plot.axes\n", + "ax1.set_title('LIGO-Hanford strain data around GW150914')\n", + "ax1.text(1.0, 1.01, 'Unfiltered data', transform=ax1.transAxes, ha='right')\n", + "ax1.set_ylabel('Amplitude [strain]', y=-0.2)\n", + "ax2.set_ylabel('')\n", + "ax2.text(1.0, 1.01, r'50-250\\,Hz bandpass, notches at 60, 120, 180 Hz',\n", + " transform=ax2.transAxes, ha='right')\n", + "plot.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We see now a spike around 16 seconds into the data, so let’s zoom into that time (and prettify):" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "plot = hfilt.plot(color='gwpy:ligo-hanford')\n", + "ax = plot.gca()\n", + "ax.set_title('LIGO-Hanford strain data around GW150914')\n", + "ax.set_ylabel('Amplitude [strain]')\n", + "ax.set_xlim(1126259462, 1126259462.6)\n", + "ax.set_xscale('seconds', epoch=1126259462)\n", + "plot.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Question 03" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Congratulations, you have succesfully filtered LIGO data to uncover the first ever directly-detected gravitational wave signal, GW150914! \n", + "\n", + "But wait, what about LIGO-Livingston (L1)? Your task is to add data from LIGO-Livingston to our figure by following the same procedure. You can load the data with:\n", + "\n", + "`ldata = TimeSeries.fetch_open_data('L1', 1126259446, 1126259478)`\n", + "\n", + "___Hint___: The article announcing the detection told us that the signals were separated by a certain time between the detectors, and that the relative orientation of those detectors means that we need to invert the data from one before comparing them. If you do it right, you should be able to generate a comparison plot resembling the upper-right plot in Figure 1 in [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102)." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Data in the Frequency Domain" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Plotting these data in the Fourier domain gives us an idea of the frequency content of the data. A way to visualize the frequency content of the data is to plot the amplitude spectral density, ASD.\n", + "\n", + "The ASDs are the square root of the power spectral densities (PSDs), which are averages of the square of the fast fourier transforms (FFTs) of the data.\n", + "\n", + "They are an estimate of the \"strain-equivalent noise\" of the detectors versus frequency, which limit the ability of the detectors to identify GW signals.\n", + "\n", + "They are in units of strain/rt(Hz). So, if you want to know the root-mean-square (rms) strain noise in a frequency band, integrate (sum) the squares of the ASD over that band, then take the square-root.\n", + "\n", + "You will see that the unfiltered LIGO data are dominated by low frequency noise; there is no way to see a signal here, without the signal processing you have done previously." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "# sampling rate:\n", + "fs = 4096\n", + "\n", + "# number of sample for the fast fourier transform:\n", + "NFFT = 1*fs\n", + "fmin = 10\n", + "fmax = 2000\n", + "Pxx_H1data, freqs = mlab.psd(hdata, Fs = fs, NFFT = NFFT)\n", + "Pxx_H1filt, freqs = mlab.psd(hfilt, Fs = fs, NFFT = NFFT)\n", + "\n", + "# We will use interpolations of the ASDs computed above for whitening:\n", + "psd_H1data = interp1d(freqs, Pxx_H1data)\n", + "psd_H1filt = interp1d(freqs, Pxx_H1filt)\n", + "\n", + "# plot the ASDs:\n", + "plt.figure()\n", + "plt.loglog(freqs, np.sqrt(Pxx_H1data),'r',label='H1 strain')\n", + "plt.loglog(freqs, np.sqrt(Pxx_H1filt),'g',label='H1 strain (filtered)')\n", + "plt.axis([fmin, fmax, 1e-24, 1e-19])\n", + "plt.grid('on')\n", + "plt.ylabel('ASD (strain/rtHz)')\n", + "plt.xlabel('Freq (Hz)')\n", + "plt.legend(loc='upper center')\n", + "plt.title('Advanced LIGO strain data near GW150914')" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "NOTE that we only plot the data between fmin = 10 Hz and fmax = 2000 Hz.\n", + "\n", + "Below fmin, the data are not properly calibrated. That's OK, because the noise is so high below fmin that LIGO cannot sense gravitational wave strain from astrophysical sources in that band.\n", + "\n", + "The sample rate is fs = 4096 Hz (2^12 Hz), so the data cannot capture frequency content above the Nyquist frequency = fs/2 = 2048 Hz. That's OK, because GW150914 only has detectable frequency content in the range 20 Hz - 300 Hz.\n", + "\n", + "You can see strong spectral lines in the data; they are all of instrumental origin. Some are engineered into the detectors (mirror suspension resonances at ~500 Hz and harmonics, calibration lines, control dither lines, etc) and some (60 Hz and harmonics) are unwanted. We'll return to these, later." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Question 04" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Generate the analogous ASD vs Frequency plot to the one above for the LIGO-Livingston (L1) data and generate a plot that directly compares the H1 and L1 data (analogous to Figure 3 in [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102))." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Evolution of the Frequency over Time: The Spectrogram" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "One of the most useful methods of visualising gravitational-wave data is to use a spectrogram, highlighting the frequency-domain content of some data over a number of time steps. We can calculate a Spectrogram using the spectrogram() method of the TimeSeries over a 2-second stride with a 1-second FFT and # .5-second overlap (50%). Note that `TimeSeries.spectrogram()` returns a Power Spectral Density (PSD) Spectrogram by default, so we use the ** (1/2.) to convert this into a (more familiar) Amplitude Spectral Density." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "specgram = hfilt.spectrogram2(fftlength=1/16., overlap=15/256.) ** (1/2.)\n", + "specgram = specgram.crop_frequencies(20) # drop everything below highpass\n", + "\n", + "plot = specgram.plot(norm='log', cmap='viridis', yscale='log')\n", + "ax = plot.gca()\n", + "ax.set_title('LIGO-Hanford strain data around GW150914')\n", + "ax.set_xlim(1126259462, 1126259463)\n", + "ax.colorbar(label=r'Strain ASD [1/$\\sqrt{\\mathrm{Hz}}$]')\n", + "plot.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "There is a hint of the famous \"chirp\" signal (around $t=0.4$ in the plot) representing the spin-down and binary BH merger for the `GW150914` discovery event. With some additional filtering that you will do in the next problem, we can make this much more visible. " + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Q-transform" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "One of the most useful tools for filtering and visualising short-duration features in a TimeSeries is the Q-transform. This is regularly used by the Detector Characterization working groups of the LIGO Scientific Collaboration and the Virgo Collaboration to produce high-resolution time-frequency maps of transient noise (glitches) and potential gravitational-wave signals.\n", + "\n", + "This algorithm was used to visualise the first ever gravitational-wave detection `GW150914`, so we can reproduce that result (bottom panel of Figure 1 in n [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102)) here." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Question 05" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "With guidance from https://gwpy.github.io/docs/2.1.3/examples/timeseries/qscan, generate the Q-filtered spectrogram for both the LIGO-Hanford and LIGO-Livingston data. Use the same time and frequency ranges as in [[2]](https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102). Specifically, perform the Q-transform for $t=0.2$ seconds around the GW event `GW150914` and set the time-axis limits to be `-0.17 seconds` before the event to `0.03 seconds` after the event. " + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "dt = 0.2 #-- Set width of q-transform plot, in seconds\n", + "hq = hfilt.q_transform(outseg=(tevent-dt, tevent+dt))\n", + "\n", + "plt.clf()\n", + "fig = hq.plot()\n", + "ax = fig.gca()\n", + "fig.colorbar(label=\"Normalised energy\")\n", + "ax.grid(False)\n", + "plt.xlim(tevent-0.17, tevent+0.03)\n", + "plt.ylim(0, 512)\n", + "plt.ylabel('Frequency (Hz)')\n", + "plt.title('Q-Transform')\n", + "plt.show()\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Machine Learning Approaches to Gravitational Wave Detection" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Given the subtle nature of the GW signals, machine learning methods can be powerful to find gravitational wave signals from binary black hole collisions. Next you will develop and explore one or more ML methods for GW detection using simulated waveforms from the [G2Net Gravitational Wave Detection Challenge](https://www.kaggle.com/competitions/g2net-gravitational-wave-detection/overview)." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Question 06" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Develop, train and evaluate an ML model used to detect GW signals from the mergers of binary black holes. Specifically, you'll build a model to analyze simulated GW time-series data from a network of Earth-based detectors.\n", + "\n", + "Some examples could be found [here](https://diposit.ub.edu/dspace/bitstream/2445/201012/1/DANA%20RU%C3%8DZ%20ABEL_7999646.pdf) which use a `Conv1D` network and compare with a pre-trained generative Transformer model. Other challenge participant solutions can be found (here)[https://www.kaggle.com/competitions/g2net-gravitational-wave-detection/code]. You have a lot of freedom in answer this question. A CNN seems like a good start, but feel free to explore ML methods here. Its more important to get one working ML model to classify signal and background waveforms then to thinly \"play\" with several models. \n", + "\n", + "You can take inspiration and code from publically-available challenge solutions, but you might cite all sources for full credit!" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "jBUTSqcNN4ee" + }, + "source": [ + "## References\n", + "\n", + "* __[1]__ Hulse, R. A. and Taylor, J. H., \"Discovery of a pulsar in a binary system\" Astrophysical Journal, Vol. 195, p. L51-L53 (1975), https://ui.adsabs.harvard.edu/abs/1975ApJ...195L..51H\n", + "\n", + "* __[2]__ B. P. Abbott et al. (LIGO Scientific Collaboration and Virgo Collaboration), \"Observation of Gravitational Waves from a Binary Black Hole Merger\", Phys. Rev. Lett. 116, 061102 (2016), https://journals.aps.org/prl/abstract/10.1103/PhysRevLett.116.061102" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Acknowledgements\n", + "\n", + "* Initial version: Mark Neubauer\n", + "\n", + "© Copyright 2024" + ] + } + ], + "metadata": { + "colab": { + "provenance": [] + }, + "kernelspec": { + "display_name": "Python 3", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.2" + } + }, + "nbformat": 4, + "nbformat_minor": 0 +} diff --git a/_toc.yml b/_toc.yml index c8a9469..d534aa2 100644 --- a/_toc.yml +++ b/_toc.yml @@ -109,6 +109,8 @@ chapters: sections: - file: _sources/projects/Project_02_NuclearGeometryQGP - file: _sources/projects/Project_02_AnisotropyQGP + - file: _sources/projects/Project_02_AberratedImages + - file: _sources/projects/Project_02_GravitationalWaves - file: _sources/Week_14 # Learning from the Machines sections: - url: https://docs.google.com/presentation/d/1hkfaU7JVy1f5S8jURZvTY67KRzP7I8zGRf4Zku_bpM4/edit?usp=sharing