Introduction to Fitting PHA Spectra
![[CXC Logo]](/ciao/imgs/cxc-logo.gif)
Sherpa Threads (CIAO 4.4 Sherpa v1)
Overview
Synopsis:
The basic steps used in fitting spectral data are illustrated in this thread. The data used herein were created by running the Creating ACIS Spectra for Pointlike Sources thread.
There are many options and variables that may affect how this process is applied to your data; for a more detailed explanation of the steps, see the following threads:
- Fitting Spectral Data: Fitting PHA Data with Multi-Component Source Models
- Fitting Spectral Data: Simultaneously Fitting Source and Background Spectra
For a detailed explanation of the fitting concepts behind X-ray spectral analysis in Sherpa, see the document Spectral Fitting on the Sherpa References page.
Before fitting ACIS data sets with restricted pulse-height ranges, please read the CIAO caveat page "Spectral analyses of ACIS data with a limited pulse-height range."
Last Update: 15 Dec 2011 - reviewed for CIAO 4.4 (no changes)
Contents
- Load the Spectrum & Instrument Responses
- Filter the Data & Subtract the Background
- Defining the Source Model
- Fitting
- Examining Fit Results
- Scripting It
- History
- Images
Load the Spectrum & Instrument Responses
First, load the spectrum file:
sherpa> load_pha("3c273.pi") WARNING: systematic errors were not found in file '3c273.pi' statistical errors were found in file '3c273.pi' but not used; to use them, re-read with use_errors=True read ARF file 3c273.arf read RMF file 3c273.rmf WARNING: systematic errors were not found in file '3c273_bg.pi' statistical errors were found in file '3c273_bg.pi' but not used; to use them, re-read with use_errors=True read background file 3c273_bg.pi
Since the RESPFILE, ANCRFILE, and BACKFILE header keywords were updated in the spectrum file, the response files (RMF and ARF) and background file are automatically read in as well. If the default dataset id of "1" is to be used, it does not need to be explicitly included in the load function; only the data filenames are required in this case.
If Sherpa does not automatically read in the background and response files, read them manually:
sherpa> load_arf("3c273.arf") sherpa> load_rmf("3c273.rmf") sherpa> load_bkg("3c273_bg.pi")
Use show_all and show_bkg to get the status of the Sherpa session. Some additional commands are used to get the total number of counts and counts rates calculated from the data.
sherpa> show_all() Data Set: 1 Filter: 0.0080-14.9431 Energy (keV) Noticed Channels: 1-1024 name = 3c273.pi channel = Float64[1024] counts = Float64[1024] staterror = None syserror = None bin_lo = None bin_hi = None grouping = Int16[1024] quality = Int16[1024] exposure = 38564.6089269 backscal = 1.87253514146e-05 areascal = 1.0 grouped = True subtracted = False units = energy response_ids = [1] background_ids = [1] RMF Data Set: 1:1 name = 3c273.rmf detchans = 1024 energ_lo = Float64[1090] energ_hi = Float64[1090] n_grp = Int16[1090] f_chan = UInt32[2002] n_chan = UInt32[2002] matrix = Float64[61834] offset = 1 e_min = Float64[1024] e_max = Float64[1024] ARF Data Set: 1:1 name = 3c273.arf energ_lo = Float64[1090] energ_hi = Float64[1090] specresp = Float64[1090] bin_lo = None bin_hi = None exposure = 38564.1414549 sherpa> show_bkg() Background Data Set: 1:1 Filter: 0.0080-14.9431 Energy (keV) Noticed Channels: 1-1024 name = 3c273_bg.pi channel = Float64[1024] counts = Float64[1024] staterror = None syserror = None bin_lo = None bin_hi = None grouping = None quality = None exposure = 38564.6089269 backscal = 1.87253514146e-05 areascal = 1.0 grouped = False subtracted = False units = energy rate = True plot_fac = 0 response_ids = [1] background_ids = [] sherpa> data_sum = calc_data_sum(id=1) # total counts (or values) in the data sherpa> print(data_sum) 216.0 sherpa> data_cnt_rate = calc_data_sum()/get_exposure(id=1) # calculating count rate in cts/sec sherpa> print(data_cnt_rate) 0.00560099028644 sherpa> bkg_sum = calc_data_sum(bkg_id=1) # total counts (or values) in the background data sherpa> print(bkg_sum) 216.0 sherpa> bkg_cnt_rate = calc_data_sum(bkg_id=1)/get_exposure(bkg_id=1) # calculating background count rate in cts/sec sherpa> print(bkg_cnt_rate) 0.00560099028644
Plot the data:
sherpa> plot_data()
The data are plotted in energy space - as seen in Figure 1 - since the instrument model provides the information necessary to compute the predicted counts for each bin. In general, the units of the x-axis are determined by the value in the units field of the data, which may be accessed with 'print(get_data().units)' or show_filter, and modified with set_analysis.
Filter the Data & Subtract the Background
Looking at the plot, we decide to evaluate the fit on the energy range between 0.1 and 6.0 keV; the CIAO 'Why' topic on Choosing an Energy Filter has more information on this process. To apply this filter to the dataset we can use the Sherpa ignore or notice functions. The notice_id()/ignore_id() functions require explicit input of the data set id; the first argument is the data set id, the second defines the lower energy of the range, and the third the higher energy of the range. The data between 0.1 and 6.0 keV will be noticed with use of either of the following commands:
sherpa> notice_id(1, 0.1, 6.0) OR sherpa> ignore_id(1, ":0.1, 6.0:")
At this point, we also opt to subtract the background data:
sherpa> subtract() sherpa> plot_data()
Figure 2 shows the resulting plot.
The axis scaling for all plots created in the current Sherpa session may be changed to log by calling set_xlog and set_ylog with no arguments (and changed back to linear with set_xlinear/set_ylinear):
sherpa> set_xlog() sherpa> set_ylog()
(To set the plot axis scaling for a specific type of plot, e.g., model, data, or fit plots, the set_xlog/set_ylog or set_xlinear/set_ylinear commands should be called with the appropriate argument, either "data", "model", "source", "fit", or "delchi" - similar to those accepted by the generic Sherpa plot function.)
To change the default settings for plot_data so that both the X and Y axes will be drawn using a log scale each time the function is called in the session, use the get_data_plot_prefs function:
sherpa> p = get_data_plot_prefs() sherpa> p["xlog"] = True sherpa> p["ylog"] = True
To learn how to change the default axis scale from linear to logarithmic so that these commands do not have to be run in each Sherpa session, see this Sherpa FAQ.
Defining the Source Model
Before fitting the data, it is necessary to define a model that characterizes the source. All models available in Sherpa, or only models belonging to a specific category, may be returned at the Sherpa prompt by calling the list_models function accordingly:
sherpa> list_models() # all models, same as 'list_models("all")'
sherpa> list_models("xspec") # all xspec models
sherpa> list_models("2d") # Sherpa 2D analytic models
Here, we use a source model composed of two model components:
- powlaw1d - a one-dimensional power law.
- xsphabs - an XSpec photoelectric absorption model.
We define an expression that is the product of these two components. The hydrogen column density (nH) is set to the known Galactic value for the source and the parameter is frozen so that it will not be allowed to vary in the fit.
sherpa> set_source(xsphabs.abs1 * powlaw1d.p1) sherpa> abs1.nH = 0.07 sherpa> freeze(abs1.nH)
The current source model definition may be displayed:
sherpa> show_model() Model: 1 apply_rmf(apply_arf((38564.6089269 * (xsphabs.abs1 * powlaw1d.p1)))) Param Type Value Min Max Units ----- ---- ----- --- --- ----- abs1.nh frozen 0.07 0 100000 10^22 atoms / cm^2 p1.gamma thawed 1 -10 10 p1.ref frozen 1 -3.40282e+38 3.40282e+38 p1.ampl thawed 1 0 3.40282e+38
Note that Sherpa and XSPEC absorption models have to be multiplied by a model which has normalization and amplitude parameters, such as powlaw1d; they cannot be used as single models in the source expression. It may be necessary to modify the parameter values, since the Sherpa guess functionality does not apply to absorption models. However, we can use this command to guess the initial parameter values and ranges for the power-law model component (as model parameter values are not automatically guessed in Sherpa 4.2 as they were in previous versions of the software). To have Sherpa automatically query for the initial parameter values when a model is established, set 'paramprompt(True)' (it is 'False' by default).
sherpa> guess(p1) sherpa> show_model() Model: 1 apply_rmf(apply_arf((38564.6089269 * (xsphabs.abs1 * powlaw1d.p1)))) Param Type Value Min Max Units ----- ---- ----- --- --- ----- abs1.nh frozen 0.07 0 100000 10^22 atoms / cm^2 p1.gamma thawed 1 -10 10 p1.ref frozen 1 -3.40282e+38 3.40282e+38 p1.ampl thawed 0.000148802 1.48802e-06 0.0148802
The guess command makes an initial guess at parameter values to ensure convergence, but it is always a good idea to check that the initial range of values (soft limits) is sensible for the data being fit.
Fitting
Now we are ready to run the fit, using the Sherpa default fit statistic (chi2gehrels) and optimization method (levmar). (The available fit statistics and optimization methods may be returned with the list_stats and list_methods commands, and they may be changed with set_method and set_stat.)
sherpa> fit() Dataset = 1 Method = levmar Statistic = chi2gehrels Initial fit statistic = 720.951 Final fit statistic = 37.9079 at function evaluation 19 Data points = 44 Degrees of freedom = 42 Probability [Q-value] = 0.651155 Reduced statistic = 0.902569 Change in statistic = 683.043 p1.gamma 2.15852 p1.ampl 0.000224841
The fit information returned by the fit command includes the statistic value for chi2gehrels, goodness-of-fit and reduced chi-square, along with the best-fit parameter values of the photon index and amplitude. The function calc_stat_info - the Sherpa 4.3 version of the Sherpa 3.4 GOODNESS command - and its associated get_stat_info command, may be used to return the goodness-of-fit statistics without having to re-run the fit:
sherpa> calc_stat_info() Dataset = 1 Statistic = chi2gehrels Fit statistic value = 37.9079 Data points = 44 Degrees of freedom = 42 Probability [Q-value] = 0.651155 Reduced statistic = 0.902569 sherpa> goodness = get_stat_info() sherpa> print(goodness[0]) name = Dataset [1] ids = [1] bkg_ids = None statname = chi2gehrels statval = 37.9079139472 numpoints = 44 dof = 42 qval = 0.65115526728 rstat = 0.902569379696
The calc_stat_info command is appropriate for accessing the fit statistics at the Sherpa prompt, where the information is printed to the screen, whereas get_stat_info is more useful for parsing this information within a script.
The best fit model with the data and residuals may be plotted in the same window:
sherpa> plot_fit_delchi()
which creates Figure 3. The errors are plotted as "sigma", the sigma residuals of the fit [(data - model)/error].
Examining Fit Results
Goodness of fit
In CIAO 3.4, the goodness command was used to get the chi-squared goodness-of-fit. This information is now reported with the best-fit values after a fit; it is no longer necessary to run a separate command. The show_fit and get_fit_results commands allow access to this information after the fit has been performed:
sherpa> show_fit() Optimization Method: LevMar name = levmar ftol = 1.19209289551e-07 xtol = 1.19209289551e-07 gtol = 1.19209289551e-07 maxfev = None epsfcn = 1.19209289551e-07 factor = 100.0 verbose = 0 Statistic: Chi2Gehrels Fit:Dataset = 1 Method = levmar Statistic = chi2gehrels Initial fit statistic = 6.83386e+10 Final fit statistic = 37.9079 at function evaluation 22 Data points = 44 Degrees of freedom = 42 Probability [Q-value] = 0.651155 Reduced statistic = 0.902569 Change in statistic = 6.83386e+10 p1.gamma 2.15852 p1.ampl 0.00022484 # retrieve a single value with get_fit_results(): sherpa> print(get_fit_results().qval) 0.651155274054 sherpa> print(get_fit_results().rstat) 0.90256937596
The number of bins in the fit (Data points), the number of degrees of freedom, i.e. the number of bins minus the number of free parameters, and the final fit statistic value are reported. If the chosen statistic is one of the chi-square statistics, as in this example, the reduced statistic (i.e. the statistic value divided by the number of degrees of freedom) and the probability (Q-value) are included as well.
The calc_chisqr command calculates the statistic contribution per bin:
sherpa> calc_chisqr()
array([ 8.18672858e+00, 5.97520920e+00, 1.12656172e+00,
2.70996397e-01, 1.52222572e-01, 6.41849665e-03,
7.45752882e-01, 3.65878489e-01, 7.64892123e-01,
7.54582959e-02, 6.15857733e-01, 2.33666307e+00,
2.01512590e-01, 3.57967845e-02, 9.88219709e-01,
3.98382524e-01, 1.35736379e-02, 1.26937320e+00,
1.71508420e+00, 2.12311366e-01, 8.44121608e-02,
1.43547663e-01, 5.85917784e-01, 4.80676459e-02,
1.97969717e+00, 1.00172041e-01, 1.05894104e+00,
4.39172444e-01, 3.28729745e-01, 6.84238569e-02,
1.50504771e-01, 1.24953231e+00, 6.00567768e-01,
5.35535361e-04, 8.38929628e-01, 1.14986971e+00,
1.77134437e-01, 6.99634183e-02, 2.85814374e-01,
2.05568964e+00, 2.35952178e-01, 2.13674054e-01,
1.51009824e-01, 4.34761024e-01])
Confidence intervals
The covariance() command - which may be shortened to covar() - computes covariance matrices and provides an estimate of confidence intervals for the thawed parameters; also see the related command conf():
sherpa> covar() covariance 1-sigma bounds: Param Best-Fit Lower Bound Upper Bound ----- -------- ----------- ----------- p1.gamma 2.15852 -0.0827856 0.0827856 p1.ampl 0.00022484 -1.48256e-05 1.48256e-05 sherpa> conf() p1.gamma lower bound: -0.0827856 p1.ampl lower bound: -1.48256e-05 p1.ampl upper bound: 1.48256e-05 p1.gamma upper bound: 0.0834106 Dataset = 1 Confidence Method = confidence Fitting Method = levmar Statistic = chi2gehrels confidence 1-sigma (68.2689%) bounds: Param Best-Fit Lower Bound Upper Bound ----- -------- ----------- ----------- p1.gamma 2.15852 -0.0827856 0.0834106 p1.ampl 0.00022484 -1.48256e-05 1.48256e-05
The output is the best-fit parameter value with positive and negative error estimates.
Flux and Counts
To calculate the flux of a PHA data set, use the calc_photon_flux and calc_energy_flux commands. The flux may be calculated over the entire data set or over a specific range:
sherpa> calc_photon_flux() 0.00046995137298 sherpa> calc_photon_flux(2., 10.) 7.26548742347e-05 sherpa> calc_energy_flux() 9.61260829653e-13 sherpa> calc_energy_flux(2., 10.) 4.55133191273e-13
To calculate the counts of a PHA data set, use the calc_data_sum, calc_model_sum, or calc_source_sum commands. As with the flux calculations, these commands may be given a range:
sherpa> calc_data_sum() 706.85714092 sherpa> calc_data_sum(2., 10.) 306.230157017 sherpa> calc_model_sum() 638.45693375313488 sherpa> calc_model_sum(2., 10.) 270.73187526053857 sherpa> calc_source_sum() 0.0469964088469405 sherpa> calc_source_sum(2., 10.) 0.0072654873922453405
Scripting It
The file fit.py is a Python script which performs the primary commands used above; it can be executed by typing execfile("fit.py") on the Sherpa command line.
The Sherpa script command may be used to save everything typed on the command line in a Sherpa session:
sherpa> script(filename="sherpa.log", clobber=False)
(Note that restoring a Sherpa session from such a file could be problematic since it may include syntax errors, unwanted fitting trials, etcetera.)
The CXC is committed to helping Sherpa users transition to new syntax as smoothly as possible. If you have existing Sherpa scripts or save files, submit them to us via the CXC Helpdesk and we will provide the CIAO/Sherpa 4.4 syntax to you.
History
| 14 Nov 2007 | rewritten for CIAO 4.0 Beta 3 |
| 29 Apr 2008 | show_all command is available in CIAO 4.0 |
| 09 Dec 2008 | figures moved inline with text |
| 09 Dec 2008 | updated for Sherpa 4.1 |
| 16 Feb 2009 | example of guess functionality added |
| 29 Apr 2009 | new script command is available with CIAO 4.1.2 |
| 15 Dec 2009 | updated for CIAO 4.2 |
| 09 Jul 2010 | updated for CIAO 4.2 Sherpa v2: S-Lang version of thread removed |
| 15 Dec 2010 | updated for Sherpa in CIAO 4.3: use of log_scale replaced with set_xlog/set_ylog; list_models is available with new argument options; new functions calc_stat_info and get_stat_info return goodness-of-fit information |
| 15 Dec 2011 | reviewed for CIAO 4.4 (no changes) |

![[Plot of source spectrum]](1.jpg)
![[Plot of source spectrum, filtered and background-subtracted]](2.jpg)
![[Plot of fit and sigma residuals]](3.png)