IFU Cube Fitting#
Use case: continuum and emission-line modeling of AGN; 1.47-1.87um.
Data: NIFS on Gemini; NGC 4151.
Tools: specutils, jdaviz/cubeviz, astropy, matplotlib, bottleneck.
Cross-intrument: NIRSpec; potentially MIRI
Documentation: This notebook is part of a STScI’s larger post-pipeline Data Analysis Tools Ecosystem and can be downloaded directly from the JDAT Notebook Github directory.
Introduction#
This notebook uses an example 3-D IFU Datacube of the Active Galactic Nucleii NGC 4151 Storchi-Bergmann et al. 2009, MNRAS, V 394, pp. 1148-1166. This is a ground-based AO-fed H-band dataset (1.47-1.87um) from the Near-Infrared Integral Field Spectrograph (NIFS) instrument at the Gemini Observatory. NIFS is a very similar image slicing IFU to JWST NIRSpec.
This notebook performs some simple spectral investigations. The notebook utilizes jdaviz/cubviz to inspect the dataset and extract 1-D spectra. The continuum is fit in a region near to the 1.644um [Fe II] emission from the AGN outflow and subtracted. The centrally compact atomic Hydrogen Brackett 12 feature, which is nearby in wavelength and contaminates the [Fe II] outflow emission, is also fit and removed. The data sub-cubes of the continuum model and the isolated and continuum subtracted [Fe II] emission are then saved. These saved data sets will ultimately serve as starting points for the future notebooks in this set.
Note: This notebook is designed to analyze the 1.6440 [Fe II] emission but the wavelengths can be altered to fit and remove continuum around any emission line of interest.
Import Packages#
time for timing
numpy for array processing and math
matplotlib.pyplot for plotting images and spectra
astropy.io for reading and writing FITS cubes and images
astropy.modeling for modeling spectral curves
astropy.utils.data for accessing the data
specutils.fitting for spectral data fitting
specutils Spectrum1D for modeling emission lines
jdaviz to use cubeviz in the notebook
# load important packages
import os
import time
from IPython.display import YouTubeVideo
import warnings
import numpy as np
from astropy.io import fits
from astropy import units as u
from astropy.modeling.functional_models import Gaussian1D
from astropy.utils.data import download_file
from specutils import Spectrum1D
from jdaviz import Cubeviz
from specutils.manipulation import extract_region
from specutils.spectra import SpectralRegion
from regions import PixCoord, CirclePixelRegion
from glue.core.roi import XRangeROI
import jdaviz
print(jdaviz.__version__)
4.2.dev15+gd019ae49
# load and configure matplotlib
%matplotlib inline
import matplotlib.pyplot as plt
plt.rcParams.update({'figure.max_open_warning': 0})
# Save and Load Objects Using Pickle (needed to get the parameter file at the end)
import pickle
def save_obj(obj, name):
with open(name, 'wb') as f:
pickle.dump(obj, f, pickle.HIGHEST_PROTOCOL)
def load_obj(name):
with open(name, 'rb') as f:
return pickle.load(f)
# This cell accesses the datacube file, defines the wavelength grid from header information and then plots a simple
# 1-D collapsed spectrum of the IFU data.
# Read in a 3-D IFU datacube of interest, and header.
cube_file = 'https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/NGC4151_Hband.fits'
fn = download_file('https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/NGC4151_Hband.fits', cache=True)
cube = fits.getdata(cube_file)
header_cube = fits.getheader(cube_file)
# grab data information and wavelength definitions.
nz, ny, nx = cube.shape
crdelt3 = header_cube['CDELT3']
crval3 = header_cube['CRVAL3']
# define the wavelength grid (microns) from the header (Angstroms)
# and the AGN redshift and the emission line of interest.
wave = ((crdelt3 * (np.arange(0, nz, 1))) + crval3)/10000.0
redshift = 0.00332
emission_line = 1.64400*(1 + redshift)
emission_line_index = (np.abs(wave-emission_line)).argmin()
# make a simple summed 1d spectrum of the full cube
flux1 = np.sum(cube, axis=(1, 2))
# plot the full 1-D spectrum.
plt.figure(0)
plt.plot(wave, flux1)
plt.xlabel('wavelength (um)')
plt.ylabel('flux')
plt.show()
We see that the spectral edges of the summed 1D are ‘ratty’. The 1D spectral array goes beyond the nominal useable data range of the instrument. We’ll ignore the poor spectral regions and focus on the AGN flux.
The [Fe II] feature that we are interested in is the bright, strong emission just shortward of 1.65um. The contaminating H I Br 12 emission is just blueward of the [Fe II].
We can use this plot window to read wavelength values of interest to define our analysis spectral ranges (see wavelength/flux grid data to the lower right of the plot window).
Special Note - in this particular dataset, a portion of the spectrum on the red side of the [FeII] emission provides a clean measure of the continuum. The blue-ward side of the [Fe II] and HI Brackett 12 emission has other emission and absorption features that make clear continuum ID very difficult. As a result, it is more accurate to do a simple linear fit to the red side of the spectrum rather than a more expanded spectral region that encompasses the emission..
# This cell defines the wavelength regions of interest: around the emission line, and the location
# where you want to fit and remove the continuum very accurately. Make a plot that shows the regions.
# Here we select a region that includes the emission line
# wavelength plus a small range of continuum around it.
# Determine these limits by investigating the flux in the above plot. Read
# the wavelength values off of the plot information at the lower right.
wave_emission_limit1 = 1.630
wave_emission_limit2 = 1.665
# Here we define a spectral range where we will use the
# flux to generate a continuum model. The flux shape in this
# AGN is quite linear around the redward emission, so we will use only a
# short segment of the spectrum on the red side of the emission
# feature.
# We again determine these values by investigating the wavelengths in the
# above plot window.
continuum_limit1 = 1.656
continuum_limit2 = 1.673
# Define the wavelength region around the emission - indices
wavemin = (np.abs(wave-wave_emission_limit1)).argmin()
wavemax = (np.abs(wave-wave_emission_limit2)).argmin()
# Define the wavelength region used to fit the continuum flux level - indices.
continuummin = (np.abs(wave-continuum_limit1)).argmin()
continuummax = (np.abs(wave-continuum_limit2)).argmin()
# Show the region used for the emission line and continuum fit. Alter the wavelengths
# above if this doesn't look good.
plt.figure(1)
plt.plot(wave, flux1)
plt.plot(wave[wavemin:wavemax], flux1[wavemin:wavemax])
plt.plot(wave[continuummin:continuummax], flux1[continuummin:continuummax], color='r')
plt.xlabel('wavelength (um)')
plt.ylabel('flux')
plt.show()
Cubeviz Visualization#
You can also visualize images inside a Jupyter notebook using Cubeviz
Video:#
This Cubeviz Demo is from the official JWST Observer YouTube channel. It shows an example of how to use Cubeviz for a specific science case (not this notebook’s science case).
vid = YouTubeVideo("ayb6OkmZUwU")
display(vid)
cubeviz = Cubeviz()
cubeviz.show()
# Here, we load the data into the Cubeviz app.
with warnings.catch_warnings():
warnings.simplefilter('ignore')
cubeviz.load_data(fn)
WARNING:root:Invalid BUNIT, using count as data unit
A video is shown below illustrating the procedure. The following steps are applied:
When you load your cube, you will see a collapsed spectrum of all the spaxels in the spectral viewer at the bottom.
If you draw a region (circle or square) in the flux viewer, you will see a collapsed spectrum of that particular region in the spectral viewer, too. For this example, we want to first define a circular region at the central position of the bright AGN flux, which is at approximately the cube center position.
Now, use the flux viewer and again use the ‘define circular region of interest’ icon to make spectra at two positions associated with the outflow emission in [Fe II]. The redshifted outflow is at approximate x position = 12, y position = 36. This will be ‘Subset 2’ and will show up in green in the display. The blueshifted outflow is at approximately x position = 48, y position = 24 in pixel index units. This will be ‘Subset 3’ and will show up in blue in the display. Hint: the coordinates of the cursor are reported at the top of the tool
Defining your Spectral Regions#
Next, you will want to define the wavelengths of interest in your spectral viewer for both your line and continuum analysis. To do this, you will similarly click the ‘define region of interest’ icon in your spectral viewer and drag a box over the wavelengths you desire. The line emission (‘Subset 4’ ) should span approximately 1.630 - 1.665 um, and the continuum emission (‘Subset 5’) should span approximately 1.656 - 1.673 um.
Hint: subsets can be modified in the Subset Tools plugin
Some Notes:#
If your cell window requires you to scroll to see the different displays in cubeviz, you can toggle the scroll window in the main menu of the notebook: Cell -> Current Outputs -> Toggle Scrolling
To better visualize the cube, you can change the display options in the Plot Option plugin.
Extract Subset Spectrum in Cubeviz Spectrum Viewer#
Retrieve the spectra of the user-defined regions from the Spectrum Viewer as a Spectrum1D object. First, we create the regions from the API in case the notebook is not run interactively.
# Create spatial regions
spatial_regions = cubeviz.get_interactive_regions()
if 'Subset 1' not in spatial_regions.keys():
agn_region = CirclePixelRegion(center=PixCoord(x=29, y=29), radius=6)
cubeviz.load_regions(agn_region)
if 'Subset 2' not in spatial_regions.keys():
redshifted_outflow = CirclePixelRegion(center=PixCoord(x=12, y=36), radius=6)
cubeviz.load_regions(redshifted_outflow)
if 'Subset 3' not in spatial_regions.keys():
blueshifted_outflow = CirclePixelRegion(center=PixCoord(x=48, y=24), radius=6)
cubeviz.load_regions(blueshifted_outflow)
spatial_regions = cubeviz.get_interactive_regions()
spatial_regions
WARNING: AstropyDeprecationWarning: The get_interactive_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead. [warnings]
WARNING:astroquery:AstropyDeprecationWarning: The get_interactive_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead.
WARNING: AstropyDeprecationWarning: The get_interactive_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead. [warnings]
WARNING:astroquery:AstropyDeprecationWarning: The get_interactive_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead.
{'Subset 1': <CirclePixelRegion(center=PixCoord(x=29, y=29), radius=6)>,
'Subset 2': <CirclePixelRegion(center=PixCoord(x=12, y=36), radius=6)>,
'Subset 3': <CirclePixelRegion(center=PixCoord(x=48, y=24), radius=6)>}
By default, the spectra have been extracted with the sum
function and default options in the spectral extraction plugin. For the purpose of this notebook, we would like to have spectra extracted with the mean
function instead. We can do that from the interface like in the screenshot here below or running the following cells.
spec_ext = cubeviz.plugins['Spectral Extraction']
spec_ext.function = 'Mean'
spec_ext.aperture = 'Subset 1'
spec_ext.add_results = 'Spectrum (Subset 1, mean)'
spec_ext.extract()
spec_ext.aperture = 'Subset 2'
spec_ext.add_results = 'Spectrum (Subset 2, mean)'
spec_ext.extract()
spec_ext.aperture = 'Subset 3'
spec_ext.add_results = 'Spectrum (Subset 3, mean)'
spec_ext.extract()
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
<Spectrum1D(flux=[0.0 ... 0.0] ct / pix2 (shape=(2040,), mean=14.82522 ct / pix2); spectral_axis=<SpectralAxis
(observer to target:
radial_velocity=0.0 km / s
redshift=0.0)
[14766.40039062 14768.00057662 14769.60076261 ... 18025.97925293
18027.57943892 18029.17962492] Angstrom> (length=2040))>
# Extract spectra corresponding to the colored regions in cubeviz
spectrum1 = cubeviz.get_data("Spectrum (Subset 1, mean)") # AGN Center
spectrum2 = cubeviz.get_data("Spectrum (Subset 2, mean)") # Red shifted component
spectrum3 = cubeviz.get_data("Spectrum (Subset 3, mean)") # Blue shifted component
spectrum1
<Spectrum1D(flux=[0.0 ... 0.0] ct / pix2 (shape=(2040,), mean=231.18710 ct / pix2); spectral_axis=<SpectralAxis
(observer to target:
radial_velocity=0.0 km / s
redshift=0.0)
[14766.40039062 14768.00057662 14769.60076261 ... 18025.97925293
18027.57943892 18029.17962492] Angstrom> (length=2040))>
# Extract the line region defined in the spectral viewer
regions = cubeviz.specviz.get_spectral_regions()
if "Subset 4" in regions.keys():
line_region = regions["Subset 4"]
else:
line_region = SpectralRegion(1.630*u.um, 1.665*u.um)
sv = cubeviz.app.get_viewer('spectrum-viewer')
sv.toolbar_active_subset.selected = []
sv.apply_roi(XRangeROI(16300, 16650))
WARNING: AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead. [warnings]
WARNING:astroquery:AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead.
# Extract the continuum region defined in the spectral viewer
if "Subset 5" in regions.keys():
continuum_region = regions["Subset 5"]
else:
continuum_region = SpectralRegion(1.656*u.um, 1.673*u.um)
sv = cubeviz.app.get_viewer('spectrum-viewer')
sv.toolbar_active_subset.selected = []
sv.apply_roi(XRangeROI(16560, 16730))
regions = cubeviz.specviz.get_spectral_regions()
regions
WARNING: AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead. [warnings]
WARNING:astroquery:AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead.
{'Subset 4': Spectral Region, 1 sub-regions:
(16300.0 Angstrom, 16650.0 Angstrom) ,
'Subset 5': Spectral Region, 1 sub-regions:
(16560.0 Angstrom, 16730.0 Angstrom) }
# Apply the spectral region
# (creates new collapsed spectra if user did not in jdaviz)
if not spectrum1:
flux_agn = np.sum(cube[:, (ny//2)-3:(ny//2)+3, (nx//2)-3:(nx//2)+3], axis=(1, 2))
tmpspec = Spectrum1D(flux=flux_agn*u.Unit('count'), spectral_axis=wave*u.micron)
spec_agn = extract_region(tmpspec, line_region)
spec_agn_continuum = extract_region(tmpspec, continuum_region)
else:
spec_agn = extract_region(spectrum1, line_region)
spec_agn_continuum = extract_region(spectrum1, continuum_region)
if not spectrum2:
flux_feii_red = np.sum(cube[:, (36)-3:(36)+3, (12)-3:(12)+3], axis=(1, 2))
tmpspec = Spectrum1D(flux=flux_feii_red*u.Unit('count'), spectral_axis=wave*u.micron)
spec_feii_red = extract_region(tmpspec, line_region)
spec_feii_red_continuum = extract_region(tmpspec, continuum_region)
else:
spec_feii_red = extract_region(spectrum2, line_region)
spec_feii_red_continuum = extract_region(spectrum2, continuum_region)
if not spectrum3:
flux_feii_blue = np.sum(cube[:, (28)-3:(28)+3, (50)-3:(50)+3], axis=(1, 2))
tmpspec = Spectrum1D(flux=flux_feii_blue*u.Unit('count'), spectral_axis=wave*u.micron)
spec_feii_blue = extract_region(tmpspec, line_region)
spec_feii_blue_continuum = extract_region(tmpspec, continuum_region)
else:
spec_feii_blue = extract_region(spectrum3, line_region)
spec_feii_blue_continuum = extract_region(spectrum3, continuum_region)
# Visualize new subsets
plt.figure()
plt.plot(spec_agn.spectral_axis, spec_agn.flux, color='black')
plt.title('Spectrum subset 1')
plt.xlabel('wavelength')
plt.ylabel('flux')
Text(0, 0.5, 'flux')
# Visualize new subsets
plt.figure()
plt.plot(spec_feii_blue.spectral_axis, spec_feii_blue.flux, color='b')
plt.plot(spec_feii_red.spectral_axis, spec_feii_red.flux, color='r')
plt.title('Spectra subset 2 and 3')
plt.xlabel('wavelength')
plt.ylabel('flux')
Text(0, 0.5, 'flux')
Fit the Continuum at the Spectral Region Location#
Open up Model Fitting Plugin. There are a number of fields to fill in and drop down menus to select from. It is important to keep in mind that the Data menu will provide only spectra to model, while the Spectral Region menu will provide only spectral region subsets to choose. In other words, you can fit the spectra in specific spectral regions. If no spectral region is selected, the entire wavelength array will be fit by the mode.
We will first fit an individual spectrum to test if our fitting parameters are good, then we will switch to fitting the full cube.
Single spectrum
Select Data: Spectrum Subset 1 (mean)
Select Spectral Region: Subset 5
Model: Linear1D
ModelID: L
Click “Add component”
Model Parameters: Leave Default
Model Equation Editor: L
Model Label: LinFitCont
Click “Fit model”, which fits the collapsed spectrum.
View the fit in the spectral viewer and confirm you are happy with it.
Full cube Toggle “Cube fit” at the top.
Change the name to LinFitCont_cube and click again “Fit model”.
The fitted cube can be accessed in the data dropdown of the 2D viewers.
Use the API instead#
models = cubeviz.get_models()
if 'LinFitCont' in models.keys():
singlemodel = models['LinFitCont']
else:
# Open model fitting plugin
plugin_mf = cubeviz.plugins['Model Fitting']
plugin_mf.open_in_tray()
# Input the appropriate datasets
plugin_mf.dataset = 'Spectrum (sum)'
plugin_mf.spectral_subset = 'Subset 5'
# Input model component
plugin_mf.create_model_component(model_component='Linear1D',
model_component_label='L1')
# Model equation gets populated automatically
plugin_mf.equation = 'L1'
# After we run this, we go to the GUI and check that the fit makes sense
plugin_mf.add_results.label = 'LinFitCont'
plugin_mf.cube_fit = False
plugin_mf.calculate_fit()
if 'LinFitCont_cube (30, 30)' in models.keys():
cubemodel = models['LinFitCont_cube']
else:
# Open model fitting plugin
plugin_mf = cubeviz.plugins['Model Fitting']
plugin_mf.open_in_tray()
# Set the fitting to cube
plugin_mf.cube_fit = True
# Input the appropriate datasets
plugin_mf.dataset = 'contents[SCI]'
plugin_mf.spectral_subset = 'Subset 5'
# Input model component
plugin_mf.create_model_component(model_component='Linear1D',
model_component_label='L2')
# Model equation gets populated automatically
plugin_mf.equation = 'L2'
# After we run this, we go to the GUI and check that the fit makes sense
plugin_mf.add_results.label = 'LinFitCont_cube'
plugin_mf.calculate_fit()
WARNING: Model is linear in parameters; consider using linear fitting methods. [astropy.modeling.fitting]
WARNING:astroquery:Model is linear in parameters; consider using linear fitting methods.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING
: Model is linear in parameters; consider using linear fitting methods. [astropy.modeling.fitting]
WARNING:astroquery:Model is linear in parameters; consider using linear fitting methods.
WARNING
: Model is linear in parameters; consider using linear fitting methods. [astropy.modeling.fitting]
WARNING:astroquery:Model is linear in parameters; consider using linear fitting methods.
WARNING
: Model is linear in parameters; consider using linear fitting methods. [astropy.modeling.fitting]
WARNING:astroquery:Model is linear in parameters; consider using linear fitting methods.
models = cubeviz.get_models()
#models
models['LinFitCont_cube (30, 30)']
<Linear1D(slope=0.07431289 ct / (Angstrom pix2), intercept=-502.91068201 ct / pix2, name='L2')>
Pulling other data#
Note, in cubeviz, you can either return the collapsed spectra as we did above by using the function
keyword argument along with (optionally) a spatial_subset
, or return the entire data cube by omitting
these keywords as below.
# List available data
print(cubeviz.data_labels)
# Get full original cube out
scidata = cubeviz.get_data("contents[SCI]")
scidata
['contents[SCI]', 'Spectrum (sum)', 'Spectrum (Subset 1, sum)', 'Spectrum (Subset 2, sum)', 'Spectrum (Subset 3, sum)', 'Spectrum (Subset 1, mean)', 'Spectrum (Subset 2, mean)', 'Spectrum (Subset 3, mean)', 'LinFitCont', 'LinFitCont_cube']
<Spectrum1D(flux=[[[0.0 ... 0.0]]] ct / pix2 (shape=(59, 59, 2040), mean=23.85399 ct / pix2); spectral_axis=<SpectralAxis
(observer to target:
radial_velocity=0.0 km / s
redshift=0.0)
[14766.40039062 14768.00057662 14769.60076261 ... 18025.97925293
18027.57943892 18029.17962492] Angstrom> (length=2040))>
# Extract SCI cube and continuum model from Cubeviz above and make a continuum subtracted cube
if 'LinFitCont_cube' in cubeviz.app.data_collection:
cont_psf_cube = cubeviz.get_data("LinFitCont_cube")
print('Check shape of the objects')
print(scidata.shape)
print(cont_psf_cube.shape)
# Obtain continuum subtracted cube
sci_contsub = scidata-cont_psf_cube
# Save to file
# sci_contsub.write('NGC4151_Hband_ContinuumSubtract.fits', format='wcs1d-fits', overwrite=True)
# cont_psf_cube.write('NGC4151_Hband_ContinuumPSF.fits', format='wcs1d-fits', overwrite=True)
Check shape of the objects
(59, 59, 2040)
(59, 59, 2040)
Developer Note:
I hit a traceback if I try to save the cubes to file, because they do not have a proper header
# Look at the continuum subtracted cube in Cubeviz
if sci_contsub:
with warnings.catch_warnings():
warnings.simplefilter('ignore')
cubeviz2 = Cubeviz()
cubeviz2.load_data(sci_contsub, data_label='Continuum Subtracted')
cubeviz2.show()
Alternative way to do continuum subtraction using numpy#
# Re-read in original IFU cube for manipulation
cube_file = 'https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/NGC4151_Hband.fits'
newfn = download_file(cube_file, cache=True)
newheader_cube = fits.getheader(cube_file)
start_time = time.time()
sci_contsub_np = np.zeros([nx, ny, nz])
cont_psf_cube_np = np.zeros([nx, ny, nz])
for i in range(1, nx-2):
for j in range(1, ny-2):
flux1 = cube[:, j, i]
cont_fit = np.polyfit(wave[continuummin:continuummax], flux1[continuummin:continuummax], 1)
fitval = np.poly1d(cont_fit)
continuum = fitval(wave)
sci_contsub_np[i, j, :] = flux1 - continuum
cont_psf_cube_np[i, j, :] = continuum
print(sci_contsub_np.shape)
del newheader_cube['MODE']
fits.writeto('NGC4151_Hband_ContinuumSubtract_numpy.fits', sci_contsub_np, newheader_cube, overwrite=True)
fits.writeto('NGC4151_Hband_ContinuumPSF_numpy.fits', cont_psf_cube_np, newheader_cube, overwrite=True)
print('Continuum subtracted cube saved. PSF continuum cube saved.')
(59, 59, 2040)
Continuum subtracted cube saved. PSF continuum cube saved.
Developer Note:
the new created file cannot be open in Cubeviz
Fitting your multiple component Gaussian Model#
Now we want to investigate an initial fit to the Br 12 emission feature, which is a pesky contaminant nearby in wavelength to our target [Fe II] emission. The Br 12 is centrally compact and arises only from the nucleus of the AGN, not from the outflow. Make a plot of the fit results.
First, select the wavelength region of interest following a similar procedure as performed at the top. There is no option to set the spectral regions to a user input, so we recommend zooming in and drawing by eye. The line emission (‘Subset 1’ ) should again span approximately 1.630 - 1.665 um.
For this example, we recommend setting up a 3 component gaussian model with the following inputs
Open up Model Fitting Plugin. There are a number of fields to fill in and drop down menus to select from. It is important to keep in mind that the Data menu will provide only spectra to model, while the Spectral Region menu will provide only spectral region subsets to choose. In other words, you can fit the spectra in specific spectral regions. If no spectral region is selected, the entire wavelength array will be fit by the mode.
Data: Spectrum (sum)
Spectral region: Subset 1
Model: Three different Gaussians with ModelID’s set to G1, G2, and G3
Model Parameters:
G1: stdev=8, mean=16410
G2: stdev=7, mean=16480
G3: stdev=50, mean=16460
You can turn on the ‘Fixed’ option if you need to, but these numbers should provide a good starting guess for the fit.
Model Equation Editor: G1+G2+g3
Model Label: GaussAll
Hit Fit, which fits the collapsed spectrum.
View the fit in the spectral viewer and confirm you are happy with it. Modify if necessary.
Then remove the ‘Fixed’ options, toggle Cube Fit, change the name to GaussAll_cube, and run again.
This will again create two models that can now be accessed within the Data Dropdown menus:
A 1D linear fit of the lines in the collapsed cube.
A 3D linear fit of the lines for each spaxel in the cube.
Wow, that multi-component fit looks great. Good deal.
Now we’re going to use the continuum psf cube from a prior cell with the Brackett model created in the above cell to create a full 3-D model of the central emission that isn’t caused by the outflow [Fe II].
Developer Note:
the fit to the full cube gets stuck and never ends
Exercise#
Now you can try to adapt the code shown above to run the model fitting from the API!
Hint:
plugin_mf.create_model_component(model_component='Gaussian1D', model_component_label='G1')
plugin_mf.set_model_component('G1', 'mean', value=16410)
# Your code
Extract what we need from Cubeviz#
# Extract the spectral regions defined in the spectral viewer
regions = cubeviz2.specviz.get_spectral_regions()
print(regions)
if "Subset 1" in regions.keys():
line_region = regions["Subset 1"]
else:
line_region = SpectralRegion(1.630*u.um, 1.665*u.um)
WARNING: AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead. [warnings]
WARNING:astroquery:AstropyDeprecationWarning: The get_spectral_regions function is deprecated and may be removed in a future version.
Use subset_tools.get_regions instead.
{}
# List available data
alldata = cubeviz2.app.data_collection
print(alldata)
print()
# List spectra available in spectrum-viewer
spec = cubeviz2.specviz.get_spectra()
print(spec)
DataCollection (2 data sets)
0: Continuum Subtracted[FLUX]
1: Spectrum (sum)
{'Spectrum (sum)': <Spectrum1D(flux=[-49521.146981022495 ... -105655.8681098123] ct (shape=(2040,), mean=5176.47242 ct); spectral_axis=<SpectralAxis
(observer to target:
radial_velocity=0.0 km / s
redshift=0.0)
[14766.40039062 14768.00057662 14769.60076261 ... 18025.97925293
18027.57943892 18029.17962492] Angstrom> (length=2040))>}
/opt/hostedtoolcache/Python/3.11.11/x64/lib/python3.11/site-packages/jdaviz/configs/specviz/helper.py:141: UserWarning: Applying the value from the redshift slider to the output spectra. To avoid seeing this warning, explicitly set the apply_slider_redshift keyword option to True or False.
warnings.warn("Applying the value from the redshift "
Developer note: currently there is no way to use the spectral extraction plugin on a created model cube. The 1D model should be calculated on the ‘mean’ extracted spectrum if the ‘mean’ model is desired.
# Get gauss model spectrum and model cube
spec_ext2 = cubeviz2.plugins['Spectral Extraction']
spec_ext2.function = 'Mean'
spec_ext2.aperture = 'Entire Cube'
spec_ext2.add_results = 'Spectrum entire (mean)'
spec_ext2.extract()
# This is used only as spectral axis for the models afterwards
all_spec = cubeviz2.get_data('Spectrum entire (mean)')
if 'GaussAll' in alldata:
gauss_spec = cubeviz2.get_data('GaussAll') # AGN Center Model Spec. This is 'sum', not 'mean'
print('Model spectrum 1D available')
print(gauss_spec)
else:
gauss_spec = False
print('No GaussAll model created')
print()
if 'GaussAll_cube' in alldata:
gauss_cube = cubeviz2.get_data('GaussAll_cube') # AGN Center Model Cube
params = cubeviz2.get_model_parameters()
print('Model spectrum 3D available')
else:
gauss_cube = False
params = False
print('No GaussAll_cube model created')
print()
# Check to see if user used Cubeviz (above), and, if not, read in premade data
if gauss_cube is False:
# Get both the model cube and the continuum cube not created with Cubeviz
fn = download_file('https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/gauss_model_cube.fits', cache=False)
tgauss_cube = fits.getdata(fn)
gauss_cube = tgauss_cube.transpose(2, 1, 0)
print('Shape of downloaded model cube: ', gauss_cube.shape)
fn_continuum = 'NGC4151_Hband_ContinuumPSF_numpy.fits'
continuum_cube = fits.open(fn_continuum, memmap=False)
newfull_header = fits.getheader(fn_continuum)
continuum_data = continuum_cube[0].data
print('Shape of downloaded continuum cube: ', continuum_data.shape)
else:
print('Shape of created model cube: ', gauss_cube.shape)
continuum_data = sci # Created in Cubeviz1
print('Shape of created continuum cube: ', continuum_data.shape)
print()
if params is False:
fn = download_file('https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/gauss_params.pkl', cache=True)
params = load_obj(fn)
print('Keys of downloaded model parameters: ', params.keys())
else:
print('Keys of created model parameters: ', params.keys())
print()
if not all_spec:
fn = download_file('https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/all_spec.fits', cache=False)
all_spec = Spectrum1D.read(fn)
print('Shape of downloaded continuum subtracted spectrum: ', all_spec.shape)
else:
print('Shape of created continuum subtracted spectrum: ', all_spec.shape)
print()
if gauss_spec is False:
fn = download_file('https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/gauss_spec.fits', cache=False)
gauss_spec = Spectrum1D.read(fn)
print('Shape of downloaded model spectrum: ', gauss_spec.shape)
else:
print('Shape of created model spectrum: ', gauss_spec.shape)
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
WARNING: UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm. [astropy.units.format.generic]
WARNING:astroquery:UnitsWarning: The unit 'Angstrom' has been deprecated in the VOUnit standard. Suggested: 0.1nm.
No GaussAll model created
No GaussAll_cube model created
Shape of downloaded model cube: (59, 59, 2040)
Shape of downloaded continuum cube: (59, 59, 2040)
Keys of downloaded model parameters: dict_keys(['GaussAll', 'GaussAll_3d'])
Shape of created continuum subtracted spectrum: (2040,)
Shape of downloaded model spectrum: (2040,)
# Overwrite gauss model with only 2 of the components of interest
if 'GaussAll_cube' in alldata:
gauss_cube_2component = gauss_cube.flux * 0.
model_label = "GaussAll_cube"
specunit = 1.
ampunit = 1.
else:
gauss_cube_2component = gauss_cube * 0.
model_label = "GaussAll_3d"
specunit = u.Angstrom
ampunit = u.Unit('count')
print(gauss_cube_2component.shape)
nx, ny, nz = gauss_cube_2component.shape
for i in range(0, nx-1):
for j in range(0, ny-1):
amp1 = params[model_label]['amplitude_0'][i][j]
amp2 = params[model_label]['amplitude_2'][i][j]
m1 = params[model_label]['mean_0'][i][j]*1E10 # Original model was in meters
m2 = params[model_label]['mean_2'][i][j]*1E10
stdev1 = params[model_label]['stddev_0'][i][j]*1E10
stdev2 = params[model_label]['stddev_2'][i][j]*1E10
g1 = Gaussian1D(amplitude=amp1*ampunit, mean=m1*specunit, stddev=stdev1*specunit)
g2 = Gaussian1D(amplitude=amp2*ampunit, mean=m2*specunit, stddev=stdev2*specunit)
gauss_cube_2component[i, j, :] = g1(all_spec.spectral_axis)+g2(all_spec.spectral_axis)
gauss_cube_2component_spec = Spectrum1D(spectral_axis=all_spec.spectral_axis,
flux=gauss_cube_2component*ampunit)
(59, 59, 2040)
Note: Why is the fit not done with all components at once instead of adding the continuum component here?
# Add the continuum cube to the new model cube
full_model = gauss_cube_2component_spec + continuum_data
print(full_model.shape)
(59, 59, 2040)
# Subtract the model to create the final cube where the [Fe II] emission is isolated.
# Re-read in original IFU cube for manipulation
cube_file = 'https://data.science.stsci.edu/redirect/JWST/jwst-data_analysis_tools/IFU_cube_continuum_fit/NGC4151_Hband.fits'
newfinalsub_header = fits.getheader(cube_file)
# Cube from cubeviz is scidata
final_sub_cube = scidata.flux.value - full_model.flux.value
final_sub_cube_units = Spectrum1D(spectral_axis=scidata.spectral_axis,
flux=final_sub_cube*ampunit)
print(final_sub_cube_units.shape)
print(scidata.shape)
(59, 59, 2040)
(59, 59, 2040)
# Delete any existing output in current directory
if os.path.exists("NGC4151_Hband_FinalSubtract.fits"):
os.remove("NGC4151_Hband_FinalSubtract.fits")
else:
print("The file does not exist")
if os.path.exists("NGC4151_Hband_ContinuumandBrackettModel.fits"):
os.remove("NGC4151_Hband_ContinuumandBrackettModel.fits")
else:
print("The file does not exist")
The file does not exist
The file does not exist
Developer note: Fits writeto gives an error. Making cell raw for now.
# Make the final plots to illustrate the original spectrum, model fits, and final continuum+gassian subtracted cube
plt.figure()
plt.xlim([16200, 16650])
plt.ylim([600, 900])
plt.plot(all_spec.spectral_axis, continuum_data[30, 30, :], label='Continuum')
plt.plot(all_spec.spectral_axis, scidata.flux[30, 30, :], label='Original Data')
plt.plot(all_spec.spectral_axis, full_model.flux[30, 30, :], label='2 Component Model')
plt.plot(all_spec.spectral_axis, final_sub_cube_units.flux[30, 30, :]+700*ampunit, label='Model Subtraction+Offset')
plt.legend()
plt.xlabel('wavelength')
plt.ylabel('flux')
plt.show()