Coverage for gwcelery/tasks/gwskynet.py: 97%
78 statements
« prev ^ index » next coverage.py v7.4.4, created at 2025-01-17 06:48 +0000
« prev ^ index » next coverage.py v7.4.4, created at 2025-01-17 06:48 +0000
1"""GWSkyNet annotation with GWSkyNet model"""
2import json
3import re
4from functools import cache
6import numpy as np
8from .. import app
9from ..util.tempfile import NamedTemporaryFile
10from . import gracedb, igwn_alert, superevents
12manual_pref_event_change_regexp = re.compile(
13 app.conf['views_manual_preferred_event_log_message'].replace('.', '\\.')
14 .replace('{}', '.+')
15)
18@cache
19def GWSkyNet_model():
20 # FIXME Remove import from function scope once importing GWSkyNet is not a
21 # slow operation
22 from GWSkyNet import GWSkyNet
24 return GWSkyNet.load_GWSkyNet_model()
27# FIXME: run GWSkyNet on general-purpose workers
28# once https://git.ligo.org/manleong.chan/gwskynet/-/issues/6 is fixed.
29@app.task(queue='openmp', shared=False)
30def gwskynet_annotation(input_list, SNRs, superevent_id):
31 """Perform the series of tasks necessary for GWSkyNet to
33 Parameters
34 ----------
35 input_list : list
36 The output of _download_and_keep_file_name that includes the
37 downloaded the skymap and the versioned file name of the skymap.
38 This list is in the form [skymap, skymap_filename].
39 snr : numpy array of floats
40 detector SNRs.
41 superevent_id : str
42 superevent uid
43 skymap_filename : str
44 versioned filename for skymap
45 """
46 # FIXME Remove import from function scope once importing GWSkyNet is not a
47 # slow operation
48 from GWSkyNet import GWSkyNet
50 filecontents, skymap_filename = input_list
51 with NamedTemporaryFile(content=filecontents) as fitsfile:
52 GWSkyNet_input = GWSkyNet.prepare_data(fitsfile.name)
53 # One of the inputs from BAYESTAR to GWSkyNet is the list of instruments,
54 # i.e., metadata['instruments'], which is converted to a binary array with
55 # three elements, i.e. GWSkyNet_input[2], for H1, L1 and V1.
56 # GWSkyNet 2.4.0 uses this array to indicate detector with SNR >= 4.5
57 GWSkyNet_input[2][0] = np.where(SNRs >= app.conf['gwskynet_snr_threshold'],
58 1, 0)
59 gwskynet_score = GWSkyNet.predict(GWSkyNet_model(), GWSkyNet_input)
60 FAP, FNP = GWSkyNet.get_rates(gwskynet_score)
61 fap = FAP[0]
62 fnp = FNP[0]
63 gs = gwskynet_score[0]
64 gwskynet_output = {'superevent_id': superevent_id,
65 'file': skymap_filename,
66 'GWSkyNet_score': gs,
67 'GWSkyNet_FAP': fap,
68 'GWSkyNet_FNP': fnp}
69 return json.dumps(gwskynet_output)
72def get_cbc_event_snr(event):
73 """Get detector SNRs from the LVAlert packet.
75 Parameters
76 ----------
77 event : dict
78 Event dictionary (e.g., the return value from
79 :meth:`gwcelery.tasks.gracedb.get_event`, or
80 ``preferred_event_data`` in igwn-alert packet.)
82 Returns
83 -------
84 snr : numpy array of floats
85 detector SNRs.
87 """
88 # GWSkyNet 2.4.0 uses this SNR array to modify one of the inputs, so
89 # snr needs to be formatted such that index 0, 1 and 2 points to H1,
90 # L1 and V1 respectively
91 snr = np.zeros(3)
92 attribs = event['extra_attributes']['SingleInspiral']
93 for det in attribs:
94 if det['ifo'] == 'H1':
95 snr[0] = det['snr']
96 if det['ifo'] == 'L1':
97 snr[1] = det['snr']
98 if det['ifo'] == 'V1':
99 snr[2] = det['snr']
100 return snr
103@gracedb.task(shared=False)
104def _download_and_return_file_name(filename, graceid):
105 """Wrapper around gracedb.download that returns the file name."""
106 filecontents = gracedb.download(filename, graceid)
107 return [filecontents, filename]
110@gracedb.task(shared=False)
111def _unpack_gwskynet_annotation_and_upload(gwskynet_output, graceid):
112 filename = 'gwskynet.json'
113 gwskynet_output_dict = json.loads(gwskynet_output)
114 message = ('GWSkyNet annotation from <a href='
115 '"/api/events/{graceid}/files/'
116 '{skymap_filename}">'
117 '{skymap_filename}</a>.'
118 ' GWSkyNet score: {cs},'
119 ' GWSkyNet FAP: {GWSkyNet_FAP},'
120 ' GWSkyNet FNP: {GWSkyNet_FNP}.').format(
121 graceid=graceid,
122 skymap_filename=gwskynet_output_dict['file'],
123 cs=np.round(gwskynet_output_dict['GWSkyNet_score'], 3),
124 GWSkyNet_FAP=np.round(gwskynet_output_dict['GWSkyNet_FAP'],
125 3),
126 GWSkyNet_FNP=np.round(gwskynet_output_dict['GWSkyNet_FNP'],
127 3)
128 )
129 return gracedb.upload(gwskynet_output, filename, graceid, message=message,
130 tags=['em_follow', 'public'])
133def _should_annotate(preferred_event, new_label, new_log_comment, labels,
134 alert_type):
135 # First check if the event passes all of GWSkyNet's annotation criteria
136 SNRs = get_cbc_event_snr(preferred_event)
138 if not (preferred_event['search'].lower() == 'allsky' and
139 preferred_event['far'] <= app.conf['gwskynet_upper_far_threshold']
140 and (SNRs >= app.conf['gwskynet_snr_threshold']).sum() >= 2 and
141 np.sqrt(sum(SNRs**2)) >=
142 app.conf['gwskynet_network_snr_threshold']):
143 return False
145 annotate = False
146 # Check if the GWSkyNet should annotate in response to this IGWN-Alert
147 if alert_type == 'label_added':
148 if superevents.should_publish(preferred_event, significant=False) is \
149 False and new_label == 'SKYMAP_READY':
150 # if the superevent is with FAR higher than the preliminary alert
151 # threshold, GWSkyNet will anotate the superevent directly.
152 annotate = True
153 elif new_label == 'GCN_PRELIM_SENT' or \
154 new_label == 'LOW_SIGNIF_PRELIM_SENT':
155 # if the FAR is lower than the preliminary alert threshold then
156 # GWSkyNet annotates the superevent if the preliminary alert has
157 # been sent.
158 annotate = True
159 elif 'GCN_PRELIM_SENT' not in labels and 'LOW_SIGNIF_PRELIM_SENT' not in \
160 labels:
161 # GWSkyNet annotations not applied until after initial prelim sent when
162 # FAR passes alert threshold
163 pass
164 elif new_log_comment.startswith('Localization copied from '):
165 # GWSkyNet will also annotate the superevent if the sky map
166 # has been changed (i.e. a sky map from a new g-event has been copied)
167 annotate = True
168 elif manual_pref_event_change_regexp.match(new_log_comment):
169 # Need to check for a different log comment if the preferred event has
170 # been changed manually
171 annotate = True
173 return annotate
176@igwn_alert.handler('superevent',
177 shared=False)
178def handle_cbc_superevent(alert):
179 """"Annotate the CBC preferred events of superevents using GWSkyNet
180 """
181 if alert['object']['preferred_event_data']['group'] != 'CBC':
182 return
184 if alert['alert_type'] != 'label_added' and \
185 alert['alert_type'] != 'log':
186 return
188 superevent_id = alert['uid']
189 preferred_event = alert['object']['preferred_event_data']
190 new_label = alert['data'].get('name', '')
191 new_log_comment = alert['data'].get('comment', '')
192 labels = alert['object'].get('labels', [])
193 SNRs = get_cbc_event_snr(preferred_event)
195 if _should_annotate(preferred_event, new_label, new_log_comment, labels,
196 alert['alert_type']):
197 (
198 gracedb.get_latest_file.s(superevent_id,
199 'bayestar.multiorder.fits')
200 |
201 _download_and_return_file_name.s(superevent_id)
202 |
203 gwskynet_annotation.s(SNRs, superevent_id)
204 |
205 _unpack_gwskynet_annotation_and_upload.s(superevent_id)
206 ).apply_async()