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() | 
