Coverage for gwcelery/tasks/p_astro.py: 51%

68 statements  

« prev     ^ index     » next       coverage.py v7.4.4, created at 2025-01-17 06:48 +0000

1"""Computation of ``p_astro`` by source category and utilities 

2related to ``p_astro.json`` source classification files. 

3See Kapadia et al (2019), :doi:`10.1088/1361-6382/ab5f2d`, for details. 

4""" 

5import io 

6import json 

7 

8from celery.utils.log import get_task_logger 

9 

10try: 

11 from ligo.p_astro import computation as pastrocomp 

12except ImportError: # p_astro older than lscsoft/p_astro!42 

13 from ligo import p_astro_computation as pastrocomp 

14 

15import numpy as np 

16from matplotlib import pyplot as plt 

17 

18from .. import app 

19from ..util import PromiseProxy, closing_figures, read_json 

20from . import gracedb, igwn_alert 

21 

22MEAN_VALUES_DICT = PromiseProxy( 

23 read_json, ('ligo.data', 'H1L1V1-mean_counts-1126051217-61603201.json')) 

24 

25THRESHOLDS_DICT = PromiseProxy( 

26 read_json, ('ligo.data', 'H1L1V1-pipeline-far_snr-thresholds.json')) 

27 

28P_ASTRO_LIVETIME = PromiseProxy( 

29 read_json, ('ligo.data', 'p_astro_livetime.json')) 

30 

31 

32log = get_task_logger(__name__) 

33 

34 

35@app.task(shared=False) 

36def compute_p_astro(snr, far, mass1, mass2, pipeline, instruments): 

37 """ 

38 Task to compute `p_astro` by source category. 

39 

40 Parameters 

41 ---------- 

42 snr : float 

43 event's SNR 

44 far : float 

45 event's cfar 

46 mass1 : float 

47 event's mass1 

48 mass2 : float 

49 event's mass2 

50 instruments : set 

51 set of instruments that detected the event 

52 

53 Returns 

54 ------- 

55 p_astros : str 

56 JSON dump of the p_astro by source category 

57 

58 Example 

59 ------- 

60 >>> p_astros = json.loads(compute_p_astro(files)) 

61 >>> p_astros 

62 {'BNS': 0.999, 'BBH': 0.0, 'NSBH': 0.0, 'Terrestrial': 0.001} 

63 

64 """ 

65 # Ensure SNR does not increase indefinitely beyond limiting FAR 

66 # for MBTA and PyCBC events 

67 snr_choice = pastrocomp.choose_snr(far, 

68 snr, 

69 pipeline, 

70 instruments, 

71 THRESHOLDS_DICT) 

72 

73 # Define constants to compute bayesfactors 

74 snr_star = 8.5 

75 far_star = 1 / (30 * 86400) 

76 

77 # Compute astrophysical bayesfactor for 

78 # GraceDB event 

79 fground = 3 * snr_star**3 / (snr_choice**4) 

80 bground = far / far_star 

81 astro_bayesfac = fground / bground 

82 

83 # Update terrestrial count based on far threshold 

84 lam_0 = far_star * P_ASTRO_LIVETIME['p_astro_livetime'] 

85 mean_values_dict = dict(MEAN_VALUES_DICT) 

86 mean_values_dict["counts_Terrestrial"] = lam_0 

87 

88 # Compute categorical p_astro values 

89 p_astro_values = \ 

90 pastrocomp.evaluate_p_astro_from_bayesfac(astro_bayesfac, 

91 mean_values_dict, 

92 mass1, 

93 mass2) 

94 # Dump mean values in json file 

95 return json.dumps(p_astro_values) 

96 

97 

98def _format_prob(prob): 

99 if prob >= 1: 

100 return '100%' 

101 elif prob <= 0: 

102 return '0%' 

103 elif prob > 0.99: 

104 return '>99%' 

105 elif prob < 0.01: 

106 return '<1%' 

107 else: 

108 return '{}%'.format(int(np.round(100 * prob))) 

109 

110 

111@app.task(shared=False) 

112@closing_figures() 

113def plot(contents): 

114 """Make a visualization of the source classification. 

115 

116 Parameters 

117 ---------- 

118 contents : str, bytes 

119 The contents of the ``p_astro.json`` file. 

120 

121 Returns 

122 ------- 

123 png : bytes 

124 The contents of a PNG file. 

125 

126 Notes 

127 ----- 

128 The unusually small size of the plot (2.5 x 2 inches) is optimized for 

129 viewing in GraceDB's image display widget. 

130 

131 Examples 

132 -------- 

133 .. plot:: 

134 :include-source: 

135 

136 >>> from gwcelery.tasks import p_astro 

137 >>> contents = ''' 

138 ... {"Terrestrial": 0.001, "BNS": 0.65, "NSBH": 0.20, 

139 ... "BBH": 0.059} 

140 ... ''' 

141 >>> p_astro.plot(contents) 

142 

143 """ 

144 # Explicitly use a non-interactive Matplotlib backend. 

145 plt.switch_backend('agg') 

146 

147 classification = json.loads(contents) 

148 outfile = io.BytesIO() 

149 

150 probs, names = zip( 

151 *sorted(zip(classification.values(), classification.keys()))) 

152 

153 with plt.style.context('seaborn-v0_8-white'): 

154 fig, ax = plt.subplots(figsize=(2.5, 2)) 

155 ax.barh(names, probs) 

156 for i, prob in enumerate(probs): 

157 ax.annotate(_format_prob(prob), (0, i), (4, 0), 

158 textcoords='offset points', ha='left', va='center') 

159 ax.set_xlim(0, 1) 

160 ax.set_xticks([]) 

161 ax.tick_params(left=False) 

162 for side in ['top', 'bottom', 'right']: 

163 ax.spines[side].set_visible(False) 

164 fig.tight_layout() 

165 fig.savefig(outfile, format='png') 

166 return outfile.getvalue() 

167 

168 

169@igwn_alert.handler('superevent', 

170 'mdc_superevent', 

171 shared=False) 

172def handle(alert): 

173 """LVAlert handler to plot and upload a visualization of every 

174 ``*.p_astro.json`` file that is added to a superevent. 

175 """ 

176 if alert['alert_type'] != 'log': 

177 return 

178 

179 graceid = alert['uid'] 

180 filename = alert['data'].get('filename') 

181 p_astro_filenames = {f'{pipeline}.p_astro.json' for pipeline in 

182 ['cwb', 'gstlal', 'mbta', 'pycbc', 'spiir', 

183 'RapidPE_RIFT']} 

184 

185 if filename in p_astro_filenames: 

186 ( 

187 gracedb.download.s(filename, graceid) 

188 | 

189 plot.s() 

190 | 

191 gracedb.upload.s( 

192 filename.replace('.json', '.png'), graceid, 

193 message=( 

194 'Source classification visualization from ' 

195 '<a href="/api/superevents/{graceid}/files/{filename}">' 

196 '{filename}</a>').format( 

197 graceid=graceid, filename=filename), 

198 tags=['em_follow', 'p_astro', 'public'] 

199 ) 

200 ).delay()