diff options
-rwxr-xr-x | ciao_extract_sbp.py | 197 |
1 files changed, 197 insertions, 0 deletions
diff --git a/ciao_extract_sbp.py b/ciao_extract_sbp.py new file mode 100755 index 0000000..ee1a592 --- /dev/null +++ b/ciao_extract_sbp.py @@ -0,0 +1,197 @@ +#!/usr/bin/env python3 +# +# Extract surface brightness profile, with optional background +# subtraction and region exclusion. +# +# Aaron LI +# Created: 2016-05-17 +# +# Change log: +# + +import os +import re +import argparse +import subprocess +import tempfile + +import numpy as np +from astropy.io import fits + + +def check_acis_type(infile): + """ + Check the ACIS type of the infile: ACIS-I or ACIS-S + """ + subprocess.run(["punlearn", "dmkeypar"]) + ret = subprocess.run(["dmkeypar", infile, "DETNAM", "echo=yes"], + check=True, stdout=subprocess.PIPE) + detnam = ret.stdout.decode("utf-8") + if re.match(r"^ACIS-0123[4-9]*", detnam): + results = {"type": "ACIS-I", "ccd": "0:3"} + elif re.match(r"^ACIS-[0-6]*7", detnam): + results = {"type": "ACIS-S", "ccd": "7"} + else: + raise ValueError("unknown DETNAM: %s" % detnam) + return results + + +def make_sbp_region(regfile, exclude_regfile=None, fov=None, ccd=None): + """ + Make the regions for SBP extraction, considering the regions to be + excluded and FoV constraint. + + Return a list containing all the SBP regions. + """ + regions = map(str.strip, open(regfile).readlines()) + regions = list(filter(lambda x: re.match(r"^(circle|annulus|pie).*", + x, re.I), + regions)) + if exclude_regfile is not None: + ex_regions = map(str.strip, open(exclude_regfile).readlines()) + ex_regions = list(filter(lambda x: not re.match(r"^\s*(|#.*)\s*$", x), + ex_regions)) + ex_regions = "*!".join([""] + ex_regions) + regions = [reg+ex_regions for reg in regions] + if fov is not None: + with tempfile.NamedTemporaryFile() as fp: + subprocess.run(["punlearn", "dmmakereg"]) + subprocess.run(args=[ + "dmmakereg", + "region=region(%s[ccd_id=%s])" % (fov, ccd), + "outfile=%s" % fp.name, + "kernel=ascii", + "clobber=yes" + ]) + fov_regions = map(str.strip, open(fp.name).readlines()) + fov_regions = list(filter(lambda x: re.match(r"^physical;polygon.*$", + x, re.I), + fov_regions)) + fov_regions = [re.sub(r"^physical;\s*", "", + re.sub(r"\s*#\s*$", "", reg)) + for reg in fov_regions] + regions = [ + "+".join([ + reg + "*" + fov for fov in fov_regions + ]) + for reg in regions + ] + return regions + + +def extract_sbp(infile, expmap, regfile, outprefix, bkg=None, erange=None): + """ + Extract the surface brightness profile + + If `bkg` is provided, then background subtraction is considered. + """ + if erange is not None: + erange = "[energy=%s]" % erange + else: + erange = "" + sbpfile = outprefix + ".fits" + cmd_args = [ + "dmextract", + "infile=%s%s[bin sky=@%s]" % (infile, erange, regfile), + "outfile=%s" % sbpfile, + "exp=%s" % expmap, + "opt=generic", + "clobber=yes" + ] + if bkg is not None: + # consider background subtraction + subprocess.run(["punlearn", "dmkeypar"]) + ret = subprocess.run(args=["dmkeypar", infile, "EXPOSURE", "echo=yes"], + check=True, stdout=subprocess.PIPE) + exposure_evt = float(ret.stdout.decode("utf-8")) + ret = subprocess.run(args=["dmkeypar", bkg, "EXPOSURE", "echo=yes"], + check=True, stdout=subprocess.PIPE) + exposure_bkg = float(ret.stdout.decode("utf-8")) + bkg_norm = exposure_evt / exposure_bkg + cmd_args += [ + "bkg=%s%s[bin sky=@%s]" % (bkg, erange, regfile), + "bkgnorm=%s" % bkg_norm, + "bkgexp=)exp" + ] + subprocess.run(["punlearn", "dmextract"]) + subprocess.run(args=cmd_args) + # add `RMID` and `R_ERR` columns + sbpfile_rmid = outprefix + "_rmid.fits" + subprocess.run(["punlearn", "dmtcalc"]) + subprocess.run(args=[ + "dmtcalc", + "infile=%s" % sbpfile, + "outfile=%s" % sbpfile_rmid, + "expression=RMID=(R[0]+R[1])/2,R_ERR=(R[1]-R[0])/2", + "clobber=yes" + ]) + # dump SBP data + with fits.open(sbpfile_rmid) as sbpfits: + rmid = sbpfits["HISTOGRAM"].data["RMID"] + r_err = sbpfits["HISTOGRAM"].data["R_ERR"] + sur_flux = sbpfits["HISTOGRAM"].data["SUR_FLUX"] + sur_flux_err = sbpfits["HISTOGRAM"].data["SUR_FLUX_ERR"] + sbpdata = np.column_stack([rmid, r_err, sur_flux, sur_flux_err]) + sbp_txt = outprefix + ".txt" + np.savetxt(sbp_txt, sbpdata, header="RMID R_ERR SUR_FLUX SUR_FLUX_ERR") + # create a QDP file + sbp_qdp = map(str.strip, open(sbp_txt).readlines()) + sbp_qdp = [re.sub(r"#", "!", line) for line in sbp_qdp] + sbp_qdp = [ + "READ SERR 1 2", + 'LABEL T "Surface Brightness Profile"', + 'LABEL X "Radius (pixel)"', + 'LABEL Y "Surface Flux (photons/cm\\u2\\d/pixel\\u2\\d/s)"', + "LOG X Y ON" + ] + sbp_qdp + open(outprefix + ".qdp", "w").write("\n".join(sbp_qdp) + "\n") + + +def main(): + parser = argparse.ArgumentParser( + description="Extract surface brightness profile") + parser.add_argument("-j", "--json", dest="json", required=False, + help="the *_INFO.json file " + + "(default: find ../*_INFO.json)") + parser.add_argument("-r", "--region", dest="region", + required=False, default="sbprofile.reg", + help="surface brightness profile region file " + + "(default: sbprofile.reg)") + parser.add_argument("-R", "--exclude-region", dest="exclude_region", + default=None, + help="file containing regions to be excluded") + parser.add_argument("-i", "--infile", dest="infile", required=True, + help="input EVT2/image file") + parser.add_argument("-e", "--expmap", dest="expmap", required=True, + help="exposure map of the input file") + parser.add_argument("-b", "--bkg", dest="bkg", default=None, + help="background event/image of the input file") + parser.add_argument("-E", "--erange", dest="erange", default=None, + help="energy range of interest (event files only)") + parser.add_argument("-F", "--fov", dest="fov", default=None, + help="FoV FITS file (for applying FoV constraint " + + "to the SBP regions)") + parser.add_argument("-o", "--outprefix", dest="outprefix", + required=False, + help="prefix of output files (default: same " + + "basename as the input region file)") + args = parser.parse_args() + + # set output prefix if not specified + if not args.outprefix: + args.outprefix = os.path.splitext(args.region)[0] + + acis_type = check_acis_type(args.infile) + regions = make_sbp_region(regfile=args.region, + exclude_regfile=args.exclude_region, + fov=args.fov, ccd=acis_type["ccd"]) + regfile_out = args.outprefix + "_fix.reg" + open(regfile_out, "w").write("\n".join(regions) + "\n") + + extract_sbp(infile=args.infile, expmap=args.expmap, + regfile=regfile_out, outprefix=args.outprefix, + bkg=args.bkg, erange=args.erange) + + +if __name__ == "__main__": + main() |