forked from gwastro/pycbc-live-utils
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpycbclive_plot_far.py
executable file
·118 lines (97 loc) · 3.45 KB
/
pycbclive_plot_far.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
#!/usr/bin/env python
"""Make a plot to verify the rate of false alarms from PyCBC Live."""
import argparse
import glob
import h5py
import tqdm
import lal
import numpy as np
import matplotlib
matplotlib.use('agg')
import pylab as pl
from scipy.stats import poisson
parser = argparse.ArgumentParser(description=__doc__)
parser.add_argument('--input-files', type=str, required=True,
help='Glob pattern for getting trigger files')
parser.add_argument('--output-plot', type=str, required=True,
help='Output path for the plot')
parser.add_argument('--detection-times', type=float, nargs='+',
help='GPS times of detections to remove '
'from the trigger set')
args = parser.parse_args()
ifars = []
stats = []
upload_thresholds = set()
pvalue_livetimes = set()
dfuts = set()
time = 0.
ifos = {'H1', 'L1', 'V1', 'K1'}
detection_times = None
if args.detection_times:
detection_times = np.array(args.detection_times)
for fn in tqdm.tqdm(glob.glob(args.input_files)):
with h5py.File(fn, 'r') as f:
# skip legacy results, don't crash
if 'num_live_detectors' not in f.attrs:
continue
# count effective live time
if f.attrs['num_live_detectors'] > 1:
time += 8
# see if there is a candidate
try:
fgg = f['foreground']
ifar = fgg['ifar'][()]
stat = fgg['stat'][0]
end_time = None
for ifo in ifos & set(fgg.keys()):
if 'end_time' in fgg[ifo]:
end_time = fgg[ifo + '/end_time']
except KeyError:
continue
#if 'foreground/NO_FOLLOWUP' in f:
# continue
# don't count actual detections
if end_time is not None and detection_times is not None \
and np.any(np.abs(detection_times - end_time) < 2):
continue
ifars.append(ifar)
stats.append(stat)
# pick up FAR-relevant settings
cl = f.attrs['command_line']
for i, arg in enumerate(cl):
if i == 0:
continue
if cl[i-1] == '--ifar-upload-threshold':
upload_thresholds.add(float(arg))
elif cl[i-1] == '--pvalue-combination-livetime':
pvalue_livetimes.add(float(arg))
elif cl[i-1] == '--ifar-double-followup-threshold':
dfuts.add(float(arg))
ifars = np.sort(np.array(ifars))
count = np.arange(len(ifars))[::-1] + 1
time = time / lal.YRJUL_SI
rate = count / time
pl.step(ifars, rate, label='Observation')
ifars2 = np.logspace(np.log10(ifars.min()), np.log10(ifars.max()), 1000)
label = 'Expectation'
for prob in [0.6827, 0.9545, 0.9973]:
a, b = poisson.interval(prob, time / ifars2)
pl.fill_between(ifars2, a / time,
b / time, alpha=0.3,
edgecolor='none', facecolor='C1', label=label)
label = None
for ut in upload_thresholds:
pl.axvline(ut, color='r', ls='--', label='--ifar-upload-threshold')
for pvlt in pvalue_livetimes:
pl.axvline(pvlt, color='b', ls=':', label='--pvalue-combination-livetime')
for dfut in dfuts:
pl.axvline(dfut, color='g', ls='-.',
label='--ifar-double-followup-threshold')
pl.xscale('log')
pl.yscale('log')
pl.xlabel('Inverse FAR [yr]')
pl.ylabel('Cumulative rate [yr$^{-1}$]')
pl.title(args.input_files, fontsize=10)
pl.legend(fontsize=10)
pl.tight_layout()
pl.savefig(args.output_plot, dpi=200)