LAL  7.5.0.1-bede9b2
antenna.py
Go to the documentation of this file.
1 # Copyright (C) 2018 Matthew Pitkin
2 #
3 # This program is free software; you can redistribute it and/or modify it
4 # under the terms of the GNU General Public License as published by the
5 # Free Software Foundation; either version 2 of the License, or (at your
6 # option) any later version.
7 #
8 # This program is distributed in the hope that it will be useful, but
9 # WITHOUT ANY WARRANTY; without even the implied warranty of
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
11 # Public License for more details.
12 #
13 # You should have received a copy of the GNU General Public License along
14 # with this program; if not, write to the Free Software Foundation, Inc.,
15 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 
17 
18 ## \defgroup lal_py_antenna Antenna
19 ## \ingroup lal_python
20 """This module provides a Python class for generating antenna response
21 functions. By default the class uses its own implementation of the antenna
22 response functions, but it also provides a wrapper to the equivalent
23 functions within LAL.
24 """
25 # \author Matthew Pitkin (<matthew.pitkin@ligo.org>)
26 #
27 # ### Synopsis ###
28 #
29 # ~~~
30 # from lal import antenna
31 # ~~~
32 #
33 # ### Examples ###
34 #
35 # A simple example of just getting the antenna response for LHO at a single
36 # time for a single sky position and polarization angle would be:
37 #
38 # \code
39 # from lal import antenna
40 # ra = 1.2 # right ascension in radians
41 # dec = -0.3 # declination in radians
42 # psi = 1.5 # polarization angle in radians
43 # times = 1000000000. # time (GPS seconds)
44 # resp = antenna.AntennaResponse('H1', ra, dec, psi=psi, times=times)
45 # print(resp.plus)
46 # \endcode
47 #
48 # To produce plots of the antenna responses over the sky for the different
49 # polarizations, assuming a detector at the North Pole, one could use:
50 #
51 # \code
52 # import numpy as np
53 # import lal
54 # from lal.antenna import AntennaResponse
55 # from mpl_toolkits.mplot3d import Axes3D
56 # import matplotlib.pyplot as pl
57 #
58 # # set a LALFrDetector object for the North Pole
59 # frdt = lal.FrDetector()
60 #
61 # frdt.name = 'NORTHPOLE'
62 # frdt.prefix = 'N1'
63 # frdt.vertexLongitudeRadians = 0.
64 # frdt.vertexLatitudeRadians = np.pi/2.
65 # frdt.vertexElevation = 0.
66 # frdt.xArmAltitudeRadians = 0.
67 # frdt.xArmAzimuthRadians = 0.
68 # frdt.yArmAltitudeRadians = 0.
69 # frdt.yArmAzimuthRadians = np.pi/2. # y-arm 90 degs from x-arm
70 # frdt.xArmMidpoint = 2000. # 4km long arms
71 # frdt.yArmMidpoint = 2000.
72 #
73 # # create lal.Detector object
74 # det = lal.Detector()
75 # lal.CreateDetector(det, frdt, lal.LALDETECTORTYPE_IFODIFF)
76 #
77 # # create RA/dec grids uniform on sphere
78 # ras = np.linspace(0, 2.*np.pi, 150)
79 # decs = np.arcsin(np.linspace(-1, 1, 150))
80 #
81 # # create antenna response on the sky grid
82 # resp = AntennaResponse(det, ras, decs, psi=0.0, times=900000000.0, vector=True,
83 # scalar=True)
84 #
85 # # get equatorial to cartesian factors
86 # xfac = np.cos(resp.dec_mesh)*np.cos(resp.ra_mesh)
87 # yfac = np.cos(resp.dec_mesh)*np.sin(resp.ra_mesh)
88 # zfac = np.sin(resp.dec_mesh)
89 #
90 # fig = pl.figure(figsize=(12, 7))
91 #
92 # for i, mode in enumerate(['plus', 'x', 'b', 'cross', 'y', 'l']):
93 # r = np.abs(resp.response[mode]) # get absolute values of response
94 #
95 # # convert from equatorial coordinates to cartesian
96 # X, Y, Z = r*xfac, r*yfac, r*zfac
97 #
98 # ax = fig.add_subplot(2, 3, i+1, projection='3d')
99 # ax.plot_surface(X, Y, Z, cmap=pl.cm.viridis, linewidth=0, rstride=1, cstride=1)
100 # ax.set_aspect('equal')
101 # ax.set_title(mode)
102 #
103 # fig.tight_layout()
104 #
105 # pl.show()
106 # \endcode
107 #
108 # which will display:
109 # <img src=""/>
110 #
111 # <!--
112 # To produce the above jpeg byte string for the image I have used:
113 # from io import BytesIO
114 # import base64
115 # figfile = BytesIO()
116 # fig.savefig(figfile, dpi=50, format='jpg')
117 # figfile.seek(0) # rewind to beginning of file
118 # figdata_jpg = base64.b64encode(figfile.getvalue()).decode('utf8')
119 # -->
120 ## @{
121 
122 
123 import numpy as np
124 
125 
126 # import SWIG-wrapped LAL
127 from lal import (LALDetectorIndexLHODIFF, LALDetectorIndexLLODIFF,
128  LALDetectorIndexGEO600DIFF, LALDetectorIndexVIRGODIFF,
129  LALDetectorIndexTAMA300DIFF, LALDetectorIndexKAGRADIFF,
130  LALDetectorIndexLIODIFF, LALDetectorIndexE1DIFF,
131  LALDetectorIndexE2DIFF, LALDetectorIndexE3DIFF,
132  CachedDetectors, LIGOTimeGPS, GreenwichMeanSiderealTime,
133  ComputeDetAMResponse, ComputeDetAMResponseExtraModes,
134  DAYSID_SI, Detector, TranslateHMStoRAD, TranslateDMStoRAD)
135 
136 from . import git_version
137 
138 __author__ = "Matthew Pitkin <matthew.pitkin@ligo.org>"
139 __version__ = git_version.verbose_msg
140 __date__ = git_version.date
141 
142 
143 ## mapping between detector names and LALCachedDetectors
144 DETMAP = {'H1': LALDetectorIndexLHODIFF,
145  'H2': LALDetectorIndexLHODIFF,
146  'LHO': LALDetectorIndexLHODIFF,
147  'L1': LALDetectorIndexLLODIFF,
148  'LLO': LALDetectorIndexLLODIFF,
149  'G1': LALDetectorIndexGEO600DIFF,
150  'GEO': LALDetectorIndexGEO600DIFF,
151  'GEO600': LALDetectorIndexGEO600DIFF,
152  'V1': LALDetectorIndexVIRGODIFF,
153  'VIRGO': LALDetectorIndexVIRGODIFF,
154  'T1': LALDetectorIndexTAMA300DIFF,
155  'TAMA': LALDetectorIndexTAMA300DIFF,
156  'TAMA300': LALDetectorIndexTAMA300DIFF,
157  'K1': LALDetectorIndexKAGRADIFF,
158  'KAGRA': LALDetectorIndexKAGRADIFF,
159  'LCGT': LALDetectorIndexKAGRADIFF,
160  'I1': LALDetectorIndexLIODIFF,
161  'LIO': LALDetectorIndexLIODIFF,
162  'E1': LALDetectorIndexE1DIFF,
163  'E2': LALDetectorIndexE2DIFF,
164  'E3': LALDetectorIndexE3DIFF}
165 
166 
167 ## The ranges of variables required for the antenna response (for look-up table generation)
168 VAR_RANGES = {'psi': [0., 2.*np.pi],
169  'time': [0., DAYSID_SI],
170  'ra': [0., 2.*np.pi],
171  'dec': [-1., 1.]} # this range is actually in sin(dec)
172 
173 
174 class AntennaResponse(object):
175  def __init__(self, detector, ra, dec, psi=0., times=None, tensor=True,
176  vector=False, scalar=False, use_lal=False, lookup=False,
177  lookuppair=None, bins1=100, bins2=100):
178  """
179  Calculate the long-wavelength limit antenna response functions for a given
180  ground-based gravitational wave detector. The response can include tensor,
181  vector, and scalar modes.
182 
183  @param detector: (str) a valid detector name (e.g., 'H1') or
184  lal.Detector object.
185  @param ra: (array_like) the right ascension of the source in radians,
186  or a string of the format 'hh:mm:ss.s'.
187  @param dec: (array_like) the declination of the source in radians, or a
188  string of the format 'dd:mm:ss.s'.
189  @param psi: (array_like) the polarization angle in radians. Defaults to
190  zero.
191  @param times: (array_like) an array of GPS time values at which to
192  calculate the response function.
193  @param tensor: (bool) set to calculate and store the tensor
194  polarization components (plus and cross). Defaults to True.
195  @param vector: (bool) set to calculate and store the vector
196  polarization components ("x" and "y"). Defaults to False.
197  @param scalar: (bool) set to calculate and store the scalar
198  polarization components (longitudinal and breathing). Defaults to
199  False.
200  @param use_lal: (bool) set to internally use the
201  XLALComputeDetAMResponse() and XLALComputeDetAMResponseExtraModes()
202  functions. Defaults to False.
203  @param lookup: (bool) set to generate and use a look-up table in a pair
204  of parameters for computing the antenna responses. Defaults to
205  False. If using the look-up table, the arrays of values being
206  "looked-up" must be in ascending order.
207  @param lookuppair: (list, tuple) a list of the two parameters that will
208  be used in the look-up table interpolation. Defaults to
209  <tt>['psi', 'time']</tt> (allowed values are <tt>'ra'</tt>,
210  <tt>'dec'</tt>, <tt>'psi'</tt>, or <tt>'time'</tt>)
211  @param bins1: (int) the number of bins in the grid in the first look-up
212  table parameter. Defaults to 100.
213  @param bins2: (int) the number of bins in the grid in the second
214  look-up table parameter. Defaults to 100.
215 
216  Example usage for tensor polarizations:
217  ~~~
218  >>> from lal.antenna import AntennaResponse
219  >>> # compute tensor response for a single time
220  >>> resp = AntennaResponse('H1', ra=1.2, dec=-0.3, psi=2.9,
221  ...times=1000000000)
222  >>> print('Fplus: {}'.format(resp.plus))
223  Fplus: [0.32427018]
224  >>> print('Fcross: {}'.format(resp.cross))
225  Fcross: [-0.79809163]
226  >>> # re-use class to get response at multiple new times
227  >>> resp.compute_response([1010101010., 1234567890.])
228  >>> print('Fplus: {}'.format(resp.plus))
229  Fplus: [ 0.09498567 -0.45495654]
230  >>> print('Fcross: {}'.format(resp.cross))
231  Fcross: [0.1706959 0.21690418]
232  ~~~
233 
234  Example usage for tensor, vector and scalar polarizations (at a series
235  of times):
236  ~~~
237  >>> import numpy as np
238  >>> times = np.linspace(1000000000.0, 1000086340.0, 1440)
239  >>> resp = AntennaResponse('H1', ra=1.2, dec=-0.3, psi=2.9,
240  ...scalar=True, vector=True, times=times)
241  >>> resp.plus
242  array([0.32427018, 0.32805983, 0.3318344 , ..., 0.32780195, 0.33157755,
243  0.33533786])
244  >>> resp.cross
245  array([-0.79809163, -0.79607858, -0.79404097, ..., -0.7962166 ,
246  -0.79418066, -0.79212028])
247  >>> resp.x # vector "x" polarization
248  array([-0.46915186, -0.46773594, -0.46627224, ..., -0.46783399,
249  -0.46637354, -0.46486538])
250  >>> resp.y # vector "y" polarization
251  array([-0.17075718, -0.17475991, -0.17875012, ..., -0.17448742,
252  -0.17847849, -0.18245689])
253  >>> resp.b # scalar "breathing" mode
254  array([0.05365678, 0.05573073, 0.05780282, ..., 0.05558939, 0.05766162,
255  0.05973181])
256  >>> resp.l # scalar "longitudinal mode"
257  array([-0.05365678, -0.05573073, -0.05780282, ..., -0.05558939,
258  -0.05766162, -0.05973181])
259  ~~~
260  """
261 
262  # response function dictionary
263  self.responseresponse = {'plus': None, # tensor plus polarization
264  'cross': None, # tensor cross polarization
265  'x': None, # vector "x" polarization
266  'y': None, # vector "y" polarization
267  'b': None, # scalar breathing mode polarizarion
268  'l': None} # scalar longitudinal mode polarization
269 
270  # parameter names and order in arrays
271  self.parametersparameters = ['ra', 'dec', 'psi', 'time']
272 
273  # set values
274  self.detectordetectordetectordetector = detector
275  self.rararara = ra
276  self.decdecdecdec = dec
277  self.psipsipsipsi = psi
278  self.timestimestimestimes = times
279 
280  # polarization modes
281  self.tensortensortensortensor = tensor
282  self.vectorvectorvectorvector = vector
283  self.scalarscalarscalarscalar = scalar
284 
285  # set whether to use internal LAL functions
286  self.use_laluse_laluse_laluse_lal = use_lal
287 
288  # set whether to allocate and use a look-up table
289  if lookup:
290  if lookuppair is None: # use default pair
291  self.set_lookup_pairset_lookup_pair(bins1=bins1, bins2=bins2)
292  else:
293  self.set_lookup_pairset_lookup_pair(pair=lookuppair, bins1=bins1, bins2=bins2)
294  self.lookuplookuplookuplookup = lookup
295 
296  # calculate antenna responses
297  self.compute_responsecompute_response()
298 
299  @property
300  def detector(self):
301  return self._detector_detector
302 
303  @detector.setter
304  def detector(self, det):
305  """
306  Set the detector for which to calculate the antenna response.
307 
308  @param det: (str) a valid detector name.
309  """
310 
311  if isinstance(det, Detector):
312  self._detector_detector = det.frDetector.prefix
313  self.laldetectorlaldetectorlaldetectorlaldetector = det
314  elif isinstance(det, str):
315  if det.upper() not in DETMAP.keys():
316  raise ValueError("Detector is not a valid detector name")
317 
318  self._detector_detector = det.upper()
319 
320  # set the LAL detector object
321  self.laldetectorlaldetectorlaldetectorlaldetector = self.detectordetectordetectordetector
322  else:
323  raise TypeError("Detector must be a string or lal.Detector object")
324 
325  @property
326  def laldetector(self):
327  return self._laldetector_laldetector
328 
329  @laldetector.setter
330  def laldetector(self, det):
331  """
332  Set the lal.Detector.
333 
334  @param det: (str) a valid detector name.
335  """
336 
337  if isinstance(det, Detector):
338  self._laldetector_laldetector = det.response
339  elif isinstance(det, str):
340  try:
341  detector = DETMAP[det.upper()]
342  except KeyError:
343  raise KeyError("Key {} is not a valid detector name.".format(det))
344 
345  self._laldetector_laldetector = CachedDetectors[detector].response
346  else:
347  raise TypeError("Detector must be a string or lal.Detector object")
348 
349  @property
350  def ra(self):
351  return self._ra_ra
352 
353  @ra.setter
354  def ra(self, raval):
355  """
356  Set the right ascension.
357  """
358 
359  if raval is None:
360  self._ra_ra = None
361  return
362  elif isinstance(raval, float) or isinstance(raval, int):
363  self._ra_ra = np.array([raval], dtype='float64')
364  elif isinstance(raval, list) or isinstance(raval, np.ndarray):
365  self._ra_ra = np.copy(raval).astype('float64')
366  elif isinstance(raval, str):
367  try:
368  rarad = TranslateHMStoRAD(raval)
369  self._ra_ra = np.array([rarad], dtype='float64')
370  except RuntimeError:
371  raise ValueError("Could not convert '{}' to a right "
372  "ascension".format(raval))
373  else:
374  raise TypeError("Right ascension must be an array")
375 
376  @property
377  def dec(self):
378  return self._dec_dec
379 
380  @property
381  def costheta(self):
382  return self._costheta_costheta
383 
384  @property
385  def sintheta(self):
386  return self._sintheta_sintheta
387 
388  @dec.setter
389  def dec(self, decval):
390  """
391  Set the declination.
392  """
393 
394  if decval is None:
395  self._dec_dec = None
396  self._costheta_costheta = None
397  self._sintheta_sintheta = None
398  return
399  elif isinstance(decval, float) or isinstance(decval, int):
400  self._dec_dec = np.array([decval], dtype='float64')
401  elif isinstance(decval, list) or isinstance(decval, np.ndarray):
402  self._dec_dec = np.copy(decval).astype('float64')
403  elif isinstance(decval, str):
404  try:
405  decrad = TranslateDMStoRAD(decval)
406  self._dec_dec = np.array([decrad], dtype='float64')
407  except RuntimeError:
408  raise ValueError("Could not convert '{}' to a "
409  "declination".format(decval))
410  else:
411  raise TypeError("Declination must be an array")
412 
413  self._costheta_costheta = np.cos(0.5*np.pi - self._dec_dec)
414  self._sintheta_sintheta = np.sin(0.5*np.pi - self._dec_dec)
415 
416  @property
417  def psi(self):
418  return self._psi_psi
419 
420  @property
421  def cospsi(self):
422  return self._cospsi_cospsi
423 
424  @property
425  def sinpsi(self):
426  return self._sinpsi_sinpsi
427 
428  @psi.setter
429  def psi(self, psival):
430  """
431  Set the value of the gravitational wave polarization angle psi.
432 
433  @param psival: (float) the polarization angle (radians)
434  """
435 
436  if psival is None:
437  self._psi_psi = None
438  self._cospsi_cospsi = None
439  self._sinpsi_sinpsi = None
440  return
441  elif isinstance(psival, float) or isinstance(psival, int):
442  self._psi_psi = np.array([psival], dtype='float64')
443  elif isinstance(psival, list) or isinstance(psival, np.ndarray):
444  self._psi_psi = np.copy(psival).astype('float64')
445  else:
446  raise TypeError("Polarization must be an array")
447 
448  self._psi_psi = np.mod(self._psi_psi, 2.*np.pi) # wrap at 0 and 2pi
449  self._cospsi_cospsi = np.cos(self._psi_psi)
450  self._sinpsi_sinpsi = np.sin(self._psi_psi)
451 
452  @property
453  def times(self):
454  return self._times_times
455 
456  @property
457  def gmsttimes(self):
458  try:
459  return self._gmsttimes_gmsttimes
460  except AttributeError:
461  return None
462 
463  @times.setter
464  def times(self, timearr):
465  """
466  Set array of times and GPS times.
467  """
468 
469  # check if times is just a float or int, and if so convert into an array
470  if timearr is None:
471  self._times_times = None
472  self._gmsttimes_gmsttimes = None
473  return
474  elif isinstance(timearr, float) or isinstance(timearr, int):
475  self._times_times = np.array([timearr], dtype='float64')
476  elif isinstance(timearr, list) or isinstance(timearr, np.ndarray):
477  self._times_times = np.copy(timearr).astype('float64')
478  else:
479  raise TypeError("Times must be an array")
480 
481  # get an array of GMSTs
482  self._gmsttimes_gmsttimes = np.ones_like(self._times_times) * np.nan # Greenwich Mean Siderial time (radians)
483  for i, time in enumerate(self._times_times):
484  gps = LIGOTimeGPS(time) # GPS time
485  gmstrad = GreenwichMeanSiderealTime(gps)
486  self._gmsttimes_gmsttimes[i] = gmstrad
487 
488  @property
489  def shape(self):
490  try:
491  return self._ra_mesh_ra_mesh.shape
492  except Exception:
493  if (self.rararara is not None and self.decdecdecdec is not None and
494  self.psipsipsipsi is not None and self.timestimestimestimes is not None):
495  return (len(self.rararara), len(self.decdecdecdec), len(self.psipsipsipsi),
496  len(self.timestimestimestimes))
497  else:
498  return ()
499 
500  @property
501  def cosphi(self):
502  return self._cosphi
503 
504  @property
505  def sinphi(self):
506  return self._sinphi
507 
508  @property
509  def tensor(self):
510  return self._tensor_tensor
511 
512  @tensor.setter
513  def tensor(self, tensorval):
514  """
515  Set whether to include tensor polarizations.
516  """
517 
518  if not isinstance(tensorval, bool):
519  raise TypeError("Must be boolean value")
520 
521  self._tensor_tensor = tensorval
522 
523  @property
524  def vector(self):
525  return self._vector_vector
526 
527  @vector.setter
528  def vector(self, vectorval):
529  """
530  Set whether to include vector polarizations.
531  """
532 
533  if not isinstance(vectorval, bool):
534  raise TypeError("Must be boolean value")
535 
536  self._vector_vector = vectorval
537 
538  @property
539  def scalar(self):
540  return self._scalar_scalar
541 
542  @scalar.setter
543  def scalar(self, scalarval):
544  """
545  Set whether to include scalar polarizations.
546  """
547 
548  if not isinstance(scalarval, bool):
549  raise TypeError("Must be boolean value")
550 
551  self._scalar_scalar = scalarval
552 
553  @property
554  def use_lal(self):
555  return self._use_lal_use_lal
556 
557  @use_lal.setter
558  def use_lal(self, val):
559  """
560  Set whether to use LAL antenna response functions.
561  """
562 
563  if not isinstance(val, bool):
564  raise TypeError("Must be a boolean value")
565 
566  self._use_lal_use_lal = val
567 
568  @property
569  def lookup_pair(self):
570  return self._lookup_pair_lookup_pair
571 
572  @lookup_pair.setter
573  def lookup_pair(self, pair):
574  if not isinstance(pair, (list, tuple)):
575  raise TypeError('Pair must be a list or tuple')
576 
577  if len(pair) != 2:
578  raise ValueError('Pair must only contain two values')
579 
580  for val in pair:
581  if val.lower() not in self.parametersparameters:
582  raise ValueError('Parameter {} in pair not '
583  'recognized'.format(val))
584 
585  self._lookup_pair_lookup_pair = [val.lower() for val in pair]
586 
587  # make sure pair is in same order as parameters
588  self._lookup_pair_lookup_pair = [val for val in self.parametersparameters
589  if val in self._lookup_pair_lookup_pair]
590 
591  # set parameters that are not in the look-up pair
592  self._not_lookup_pair_not_lookup_pair = [val.lower() for val in self.parametersparameters
593  if val not in self._lookup_pair_lookup_pair]
594  self._not_lookup_pair_idx_not_lookup_pair_idx = [self.parametersparameters.index(val)
595  for val in self._not_lookup_pair_not_lookup_pair]
596 
597  def set_lookup_pair(self, pair=['psi', 'time'], bins1=100, bins2=100):
598  """
599  Set the pair of parameters to use for the look-up table.
600  """
601 
602  self.lookup_pairlookup_pairlookup_pairlookup_pair = pair
603  self._lookup_pair_dict_lookup_pair_dict = {}
604 
605  for val, nbins in zip(pair, [bins1, bins2]):
606  if not isinstance(nbins, int):
607  raise TypeError("Value must be an integer")
608 
609  if nbins < 2:
610  raise ValueError("There must be at least 2 bins")
611 
612  self._lookup_pair_dict_lookup_pair_dict[val] = {}
613 
614  vrange = VAR_RANGES[val]
615 
616  # set array of bins
617  if val == 'dec':
618  # range is in sin(dec) to give uniform grid on sky sphere, so
619  # convert into dec
620  vararray = np.arcsin(np.linspace(vrange[0], vrange[1], nbins))
621  else:
622  vararray = np.linspace(vrange[0], vrange[1], nbins)
623 
624  self._lookup_pair_dict_lookup_pair_dict[val]['array'] = vararray
625  self._lookup_pair_dict_lookup_pair_dict[val]['nbins'] = nbins
626 
627  @property
628  def bins1(self):
629  return self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['nbins']
630 
631  @property
632  def array1(self):
633  return self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array']
634 
635  @property
636  def bins2(self):
637  return self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['nbins']
638 
639  @property
640  def array2(self):
641  return self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array']
642 
643  @property
644  def lookup(self):
645  return self._lookup_lookup
646 
647  @lookup.setter
648  def lookup(self, val):
649  """
650  Set the 2d look-up table.
651  """
652 
653  self._timeepoch_timeepoch = 1000000000. # a GPS epoch
654 
655  if not isinstance(val, bool):
656  raise TypeError("Value must be a boolean")
657 
658  self._lookup_lookup = False # set to False so look-up table can be calculated
659 
660  # set the look-up table
661  if val:
662  try:
663  from scipy.interpolate import RectBivariateSpline
664  except ImportError:
665  raise ImportError("Cannot import scipy")
666 
667  curra = self.rararara # save current RA value
668  curdec = self.decdecdecdec # save current dec value
669  curpsi = self.psipsipsipsi # save current psi value
670  try:
671  curtimes = self.timestimestimestimes # save current times
672  except AttributeError:
673  raise AttributeError("A time must be set for the look-up table")
674 
675  if 'psi' in self.lookup_pairlookup_pairlookup_pairlookup_pair:
676  self.psipsipsipsi = self._lookup_pair_dict_lookup_pair_dict['psi']['array']
677  if 'time' in self.lookup_pairlookup_pairlookup_pairlookup_pair:
678  self.timestimestimestimes = (self._timeepoch_timeepoch +
679  self._lookup_pair_dict_lookup_pair_dict['time']['array'])
680  if 'ra' in self.lookup_pairlookup_pairlookup_pairlookup_pair:
681  self.rararara = self._lookup_pair_dict_lookup_pair_dict['ra']['array']
682  if 'dec' in self.lookup_pairlookup_pairlookup_pairlookup_pair:
683  self.decdecdecdec = self._lookup_pair_dict_lookup_pair_dict['dec']['array']
684 
685  self._not_lookup_pair_lens_not_lookup_pair_lens = [] # lengths of the non-look-up table values
686  for val in self._not_lookup_pair_not_lookup_pair:
687  if val == 'ra':
688  self._not_lookup_pair_lens_not_lookup_pair_lens.append(len(curra))
689  if val == 'dec':
690  self._not_lookup_pair_lens_not_lookup_pair_lens.append(len(curdec))
691  if val == 'psi':
692  self._not_lookup_pair_lens_not_lookup_pair_lens.append(len(curpsi))
693  if val in 'time':
694  self._not_lookup_pair_lens_not_lookup_pair_lens.append(len(curtimes))
695 
696  # set look-up table functions
697  self._lookup_func_lookup_func = {}
698  if self.tensortensortensortensor:
699  self._lookup_func_lookup_func['plus'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
700  self._lookup_func_lookup_func['cross'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
701 
702  if self.vectorvectorvectorvector:
703  self._lookup_func_lookup_func['x'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
704  self._lookup_func_lookup_func['y'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
705 
706  if self.scalarscalarscalarscalar:
707  self._lookup_func_lookup_func['b'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
708  self._lookup_func_lookup_func['l'] = np.empty(self._not_lookup_pair_lens_not_lookup_pair_lens, dtype=object)
709 
710  for i in range(self._not_lookup_pair_lens_not_lookup_pair_lens[0]):
711  if self._not_lookup_pair_not_lookup_pair[0] == 'ra':
712  self.rararara = curra[i]
713  if self._not_lookup_pair_not_lookup_pair[0] == 'dec':
714  self.decdecdecdec = curdec[i]
715  if self._not_lookup_pair_not_lookup_pair[0] == 'psi':
716  self.psipsipsipsi = curpsi[i]
717  if self._not_lookup_pair_not_lookup_pair[0] == 'time':
718  self.timestimestimestimes = curtimes[i]
719 
720  for j in range(self._not_lookup_pair_lens_not_lookup_pair_lens[1]):
721  if self._not_lookup_pair_not_lookup_pair[1] == 'ra':
722  self.rararara = curra[j]
723  if self._not_lookup_pair_not_lookup_pair[1] == 'dec':
724  self.decdecdecdec = curdec[j]
725  if self._not_lookup_pair_not_lookup_pair[1] == 'psi':
726  self.psipsipsipsi = curpsi[j]
727  if self._not_lookup_pair_not_lookup_pair[1] == 'time':
728  self.timestimestimestimes = curtimes[j]
729 
730  self.compute_responsecompute_response()
731  if self.tensortensortensortensor:
732  self._lookup_func_lookup_func['plus'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
733  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
734  self.plusplusplusplus)
735  self._lookup_func_lookup_func['cross'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
736  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
737  self.crosscrosscrosscross)
738 
739  if self.vectorvectorvectorvector:
740  self._lookup_func_lookup_func['x'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
741  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
742  self.xxxx)
743  self._lookup_func_lookup_func['y'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
744  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
745  self.yyyy)
746 
747  if self.scalarscalarscalarscalar:
748  self._lookup_func_lookup_func['b'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
749  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
750  self.bbbb)
751  self._lookup_func_lookup_func['l'][i, j] = RectBivariateSpline(self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[0]]['array'],
752  self._lookup_pair_dict_lookup_pair_dict[self.lookup_pairlookup_pairlookup_pairlookup_pair[1]]['array'],
753  self.llll)
754 
755  # reset values
756  self.psipsipsipsi = curpsi
757  self.timestimestimestimes = curtimes
758  self.rararara = curra
759  self.decdecdecdec = curdec
760  self._lookup_lookup = True
761 
762  @property
763  def plus(self):
764  return self.responseresponse['plus']
765 
766  @property
767  def tensor_plus(self):
768  return self.plusplusplusplus
769 
770  @plus.setter
771  def plus(self, resp):
772  if self.tensortensortensortensor:
773  self.responseresponse['plus'] = resp
774 
775  @property
776  def cross(self):
777  return self.responseresponse['cross']
778 
779  @property
780  def tensor_cross(self):
781  return self.crosscrosscrosscross
782 
783  @cross.setter
784  def cross(self, resp):
785  if self.tensortensortensortensor:
786  self.responseresponse['cross'] = resp
787 
788  @property
789  def x(self):
790  return self.responseresponse['x']
791 
792  @property
793  def vector_x(self):
794  return self.xxxx
795 
796  @x.setter
797  def x(self, resp):
798  if self.vectorvectorvectorvector:
799  self.responseresponse['x'] = resp
800 
801  @property
802  def y(self):
803  return self.responseresponse['y']
804 
805  @property
806  def vector_y(self):
807  return self.yyyy
808 
809  @y.setter
810  def y(self, resp):
811  if self.vectorvectorvectorvector:
812  self.responseresponse['y'] = resp
813 
814  @property
815  def b(self):
816  return self.responseresponse['b']
817 
818  @property
819  def scalar_b(self):
820  return self.bbbb
821 
822  @b.setter
823  def b(self, resp):
824  if self.scalarscalarscalarscalar:
825  self.responseresponse['b'] = resp
826 
827  @property
828  def l(self):
829  return self.responseresponse['l']
830 
831  @property
832  def scalar_l(self):
833  return self.llll
834 
835  @l.setter
836  def l(self, resp):
837  if self.scalarscalarscalarscalar:
838  self.responseresponse['l'] = resp
839 
840  def _set_mesh(self):
841  """
842  Convert one-dimensional arrays into a mesh. The meshes dimensions are
843  ordered as right ascension, declination, psi, and time.
844  """
845 
846  # mesh order RA, dec, psi, time
847  if (self.rararara is None or self.decdecdecdec is None or
848  self.psipsipsipsi is None or self.timestimestimestimes is None):
849  return
850 
851  ramesh, decmesh, psimesh, timemesh = np.meshgrid(self.rararara, self.decdecdecdec,
852  self.psipsipsipsi,
853  self.gmsttimesgmsttimes,
854  indexing='ij')
855 
856  self.ra_meshra_meshra_meshra_mesh = ramesh
857  self.time_meshtime_meshtime_meshtime_mesh = timemesh
858  self.dec_meshdec_meshdec_meshdec_mesh = decmesh
859  self.psi_meshpsi_meshpsi_meshpsi_mesh = psimesh
860 
861  @property
862  def ra_mesh(self):
863  return self._ra_mesh_ra_mesh
864 
865  @ra_mesh.setter
866  def ra_mesh(self, val):
867  if isinstance(val, np.ndarray):
868  # remove singleton dimemsions
869  self._ra_mesh_ra_mesh = val.squeeze()
870  else:
871  self._ra_mesh_ra_mesh = val
872 
873  try:
874  self._phi_mesh_phi_mesh = np.mod(self._ra_mesh_ra_mesh, 2.*np.pi) - self._time_mesh_time_mesh
875  except Exception:
876  return
877 
878  self._cosphi_mesh_cosphi_mesh = np.cos(self._phi_mesh_phi_mesh)
879  self._sinphi_mesh_sinphi_mesh = np.sin(self._phi_mesh_phi_mesh)
880 
881  @property
882  def time_mesh(self):
883  return self._time_mesh_time_mesh
884 
885  @time_mesh.setter
886  def time_mesh(self, val):
887  if isinstance(val, np.ndarray):
888  # remove singleton dimemsions
889  self._time_mesh_time_mesh = val.squeeze()
890  else:
891  self._time_mesh_time_mesh = val
892 
893  try:
894  self._phi_mesh_phi_mesh = self._ra_mesh_ra_mesh - self._time_mesh_time_mesh
895  except Exception:
896  return
897 
898  self._cosphi_mesh_cosphi_mesh = np.cos(self._phi_mesh_phi_mesh)
899  self._sinphi_mesh_sinphi_mesh = np.sin(self._phi_mesh_phi_mesh)
900 
901  @property
902  def dec_mesh(self):
903  return self._dec_mesh_dec_mesh
904 
905  @dec_mesh.setter
906  def dec_mesh(self, val):
907  if isinstance(val, np.ndarray):
908  # remove singleton dimemsions
909  self._dec_mesh_dec_mesh = val.squeeze()
910  else:
911  self._dec_mesh_dec_mesh = val
912 
913  self._costheta_mesh_costheta_mesh = np.cos(0.5*np.pi - self.dec_meshdec_meshdec_meshdec_mesh)
914  self._sintheta_mesh_sintheta_mesh = np.sin(0.5*np.pi - self.dec_meshdec_meshdec_meshdec_mesh)
915 
916  @property
917  def psi_mesh(self):
918  return self._psi_mesh_psi_mesh
919 
920  @psi_mesh.setter
921  def psi_mesh(self, val):
922  if isinstance(val, np.ndarray):
923  # remove singleton dimemsions
924  self._psi_mesh_psi_mesh = val.squeeze()
925  else:
926  self._psi_mesh_psi_mesh = val
927 
928  self._cospsi_mesh_cospsi_mesh = np.cos(self._psi_mesh_psi_mesh)
929  self._sinpsi_mesh_sinpsi_mesh = np.sin(self._psi_mesh_psi_mesh)
930 
931  def compute_response(self, times=None):
932  """
933  Compute the detector response.
934 
935  @param times: (array_like) an array of GPS times at which to compute
936  the response function. If not set the times set at initialization
937  of the class, or using the <tt>times</tt> property.
938  """
939 
940  if times is not None:
941  self.timestimestimestimes = times
942 
943  if (self.rararara is None or self.decdecdecdec is None or self.psipsipsipsi is None or
944  self.timestimestimestimes is None):
945  return
946 
947  if self.use_laluse_laluse_laluse_lal: # use the internal LAL functions
948  self._compute_response_lal_compute_response_lal()
949  elif self.lookuplookuplookuplookup: # use look-up table
950  self._compute_response_lookup_compute_response_lookup()
951  else: # use default response
952  self._compute_response_compute_response()
953 
954  def _compute_response(self):
955  """
956  Compute antenna pattern.
957  """
958 
959  self._set_mesh_set_mesh()
960 
961  # set any required additional einsum indices
962  einsum_indices = ''
963  eindices = 'klmn' # up to four indices
964  for i in range(len(self.shapeshape)):
965  einsum_indices += eindices[i]
966 
967  # numpy einsum string inputs
968  einstr1 = 'i{},j{}->ij{}'.format(*3*[einsum_indices])
969  einstr2 = 'ij{},ij->{}'.format(*2*[einsum_indices])
970 
971  M = np.array([self._sinphi_mesh_sinphi_mesh*self._cospsi_mesh_cospsi_mesh - self._cosphi_mesh_cosphi_mesh*self._costheta_mesh_costheta_mesh*self._sinpsi_mesh_sinpsi_mesh,
972  -self._cosphi_mesh_cosphi_mesh*self._cospsi_mesh_cospsi_mesh - self._sinphi_mesh_sinphi_mesh*self._costheta_mesh_costheta_mesh*self._sinpsi_mesh_sinpsi_mesh,
973  self._sintheta_mesh_sintheta_mesh*self._sinpsi_mesh_sinpsi_mesh])
974  N = np.array([-self._sinphi_mesh_sinphi_mesh*self._sinpsi_mesh_sinpsi_mesh - self._cosphi_mesh_cosphi_mesh*self._costheta_mesh_costheta_mesh*self._cospsi_mesh_cospsi_mesh,
975  self._cosphi_mesh_cosphi_mesh*self._sinpsi_mesh_sinpsi_mesh - self._sinphi_mesh_sinphi_mesh*self._costheta_mesh_costheta_mesh*self._cospsi_mesh_cospsi_mesh,
976  self._sintheta_mesh_sintheta_mesh*self._cospsi_mesh_cospsi_mesh])
977 
978  mm = np.einsum(einstr1, M, M)
979  mn = np.einsum(einstr1, M, N)
980  nm = np.einsum(einstr1, N, M)
981  nn = np.einsum(einstr1, N, N)
982 
983  if self.tensortensortensortensor:
984  # set tensor polarization components
985  self.plusplusplusplus = np.einsum(einstr2, mm - nn, self.laldetectorlaldetectorlaldetectorlaldetector)
986  self.crosscrosscrosscross = np.einsum(einstr2, mn + nm, self.laldetectorlaldetectorlaldetectorlaldetector)
987 
988  if self.vectorvectorvectorvector or self.scalarscalarscalarscalar:
989  # set scalar and/or vector polarization components
990  Q = np.array([-self._sintheta_mesh_sintheta_mesh*self._cosphi_mesh_cosphi_mesh,
991  -self._sintheta_mesh_sintheta_mesh*self._sinphi_mesh_sinphi_mesh,
992  -self._costheta_mesh_costheta_mesh])
993 
994  if self.vectorvectorvectorvector:
995  mq = np.einsum(einstr1, M, Q)
996  qm = np.einsum(einstr1, Q, M)
997  nq = np.einsum(einstr1, N, Q)
998  qn = np.einsum(einstr1, Q, N)
999 
1000  self.xxxx = np.einsum(einstr2, mq + qm, self.laldetectorlaldetectorlaldetectorlaldetector)
1001  self.yyyy = np.einsum(einstr2, nq + qn, self.laldetectorlaldetectorlaldetectorlaldetector)
1002 
1003  if self.scalarscalarscalarscalar:
1004  qq = np.einsum(einstr1, Q, Q)
1005 
1006  self.bbbb = np.einsum(einstr2, mm + nn, self.laldetectorlaldetectorlaldetectorlaldetector)
1007  self.llll = np.einsum(einstr2, qq, self.laldetectorlaldetectorlaldetectorlaldetector)
1008 
1009  def _compute_response_lal(self):
1010  """
1011  Compute antenna pattern using LAL functions.
1012  """
1013 
1014  self._set_mesh_set_mesh()
1015 
1016  slen = np.prod(self.shapeshape) if len(self.shapeshape) != 0 else 1
1017 
1018  # allocate memory
1019  if slen != 1:
1020  fp = np.zeros(self.shapeshape)
1021  fc = np.zeros(self.shapeshape)
1022 
1023  if self.tensortensortensortensor and not self.vectorvectorvectorvector and not self.scalarscalarscalarscalar:
1024  # only requiring tensor mode
1025  antenna_func = ComputeDetAMResponse
1026 
1027  if slen == 1:
1028  fp, fc = antenna_func(self.laldetectorlaldetectorlaldetectorlaldetector,
1029  self._ra_mesh_ra_mesh.item(),
1030  self._dec_mesh_dec_mesh.item(),
1031  self._psi_mesh_psi_mesh.item(),
1032  self._time_mesh_time_mesh.item())
1033  else:
1034  for i in range(slen):
1035  idxs = np.unravel_index(i, self.shapeshape)
1036 
1037  fp[idxs], fc[idxs] = antenna_func(self.laldetectorlaldetectorlaldetectorlaldetector,
1038  self._ra_mesh_ra_mesh[idxs],
1039  self._dec_mesh_dec_mesh[idxs],
1040  self._psi_mesh_psi_mesh[idxs],
1041  self._time_mesh_time_mesh[idxs])
1042 
1043  self.plusplusplusplus = fp
1044  self.crosscrosscrosscross = fc
1045  else:
1046  antenna_func = ComputeDetAMResponseExtraModes
1047 
1048  if slen == 1:
1049  fp, fc, fb, fl, fx, fy = antenna_func(self.laldetectorlaldetectorlaldetectorlaldetector,
1050  self._ra_mesh_ra_mesh.item(),
1051  self._dec_mesh_dec_mesh.item(),
1052  self._psi_mesh_psi_mesh.item(),
1053  self._time_mesh_time_mesh.item())
1054  else:
1055  fb = np.zeros(self.shapeshape)
1056  fl = np.zeros(self.shapeshape)
1057  fx = np.zeros(self.shapeshape)
1058  fy = np.zeros(self.shapeshape)
1059 
1060  for i in range(slen):
1061  idxs = np.unravel_index(i, self.shapeshape)
1062 
1063  F = antenna_func(self.laldetectorlaldetectorlaldetectorlaldetector,
1064  self._ra_mesh_ra_mesh[idxs],
1065  self._dec_mesh_dec_mesh[idxs],
1066  self._psi_mesh_psi_mesh[idxs],
1067  self._time_mesh_time_mesh[idxs])
1068 
1069  fp[idxs] = F[0]
1070  fc[idxs] = F[1]
1071  fb[idxs] = F[2]
1072  fl[idxs] = F[3]
1073  fx[idxs] = F[4]
1074  fy[idxs] = F[5]
1075 
1076  if self.tensortensortensortensor:
1077  self.plusplusplusplus = fp
1078  self.crosscrosscrosscross = fc
1079 
1080  if self.vectorvectorvectorvector:
1081  self.xxxx = fx
1082  self.yyyy = fy
1083 
1084  if self.scalarscalarscalarscalar:
1085  self.bbbb = fb
1086  self.llll = fl
1087 
1088  def _compute_response_lookup(self):
1089  # allocate memory
1090  lushape = (len(self.rararara), len(self.decdecdecdec), len(self.psipsipsipsi), len(self.timestimestimestimes))
1091 
1092  if self.tensortensortensortensor:
1093  fp = np.zeros(lushape)
1094  fc = np.zeros(lushape)
1095 
1096  if self.scalarscalarscalarscalar:
1097  fb = np.zeros(lushape)
1098  fl = np.zeros(lushape)
1099 
1100  if self.vectorvectorvectorvector:
1101  fx = np.zeros(lushape)
1102  fy = np.zeros(lushape)
1103 
1104  # set pair of parameters in look-up table
1105  pairs = []
1106  unsorted_idxs = None
1107  for i, val in enumerate(self.lookup_pairlookup_pairlookup_pairlookup_pair):
1108  if val == 'ra':
1109  pairs.append(self.rararara)
1110  if val == 'dec':
1111  pairs.append(self.decdecdecdec)
1112  if val == 'psi':
1113  pairs.append(self.psipsipsipsi)
1114  if val == 'time':
1115  # times mod-ed by a sidereal day
1116  mtimes = np.mod(self.timestimestimestimes - self._timeepoch_timeepoch, DAYSID_SI)
1117  unsorted_idxs = np.argsort(np.argsort(mtimes)) # unsorted indices
1118  mtimes.sort() # sort times for interpolation
1119  pairs.append(mtimes)
1120 
1121  # get reshape tuple
1122  sshape = [len(self.rararara), len(self.decdecdecdec), len(self.psipsipsipsi), len(self.timestimestimestimes)]
1123  sshape[self._not_lookup_pair_idx_not_lookup_pair_idx[0]] = 1
1124  sshape[self._not_lookup_pair_idx_not_lookup_pair_idx[1]] = 1
1125  sshape = tuple(sshape)
1126 
1127  # create slice
1128  pos = 4*[slice(None)]
1129  for i in range(self._not_lookup_pair_lens_not_lookup_pair_lens[0]):
1130  # set slice
1131  pos[self._not_lookup_pair_idx_not_lookup_pair_idx[0]] = slice(i, i+1)
1132  for j in range(self._not_lookup_pair_lens_not_lookup_pair_lens[1]):
1133  # set slice
1134  pos[self._not_lookup_pair_idx_not_lookup_pair_idx[1]] = slice(j, j+1)
1135 
1136  if self.tensortensortensortensor:
1137  fp[tuple(pos)] = self._lookup_func_lookup_func['plus'][i, j](*pairs).reshape(sshape)
1138  fc[tuple(pos)] = self._lookup_func_lookup_func['cross'][i, j](*pairs).reshape(sshape)
1139 
1140  if self.scalarscalarscalarscalar:
1141  fb[tuple(pos)] = self._lookup_func_lookup_func['b'][i, j](*pairs).reshape(sshape)
1142  fl[tuple(pos)] = self._lookup_func_lookup_func['l'][i, j](*pairs).reshape(sshape)
1143 
1144  if self.vectorvectorvectorvector:
1145  fx[tuple(pos)] = self._lookup_func_lookup_func['x'][i, j](*pairs).reshape(sshape)
1146  fy[tuple(pos)] = self._lookup_func_lookup_func['y'][i, j](*pairs).reshape(sshape)
1147 
1148  if unsorted_idxs is not None:
1149  if self.tensortensortensortensor:
1150  self.plusplusplusplus = fp[:, :, :, unsorted_idxs].squeeze()
1151  self.crosscrosscrosscross = fc[:, :, :, unsorted_idxs].squeeze()
1152 
1153  if self.scalarscalarscalarscalar:
1154  self.bbbb = fb[:, :, :, unsorted_idxs].squeeze()
1155  self.llll = fl[:, :, :, unsorted_idxs].squeeze()
1156 
1157  if self.vectorvectorvectorvector:
1158  self.xxxx = fx[:, :, :, unsorted_idxs].squeeze()
1159  self.yyyy = fy[:, :, :, unsorted_idxs].squeeze()
1160  else:
1161  if self.tensortensortensortensor:
1162  self.plusplusplusplus = fp.squeeze()
1163  self.crosscrosscrosscross = fc.squeeze()
1164 
1165  if self.scalarscalarscalarscalar:
1166  self.bbbb = fb.squeeze()
1167  self.llll = fl.squeeze()
1168 
1169  if self.vectorvectorvectorvector:
1170  self.xxxx = fx.squeeze()
1171  self.yyyy = fy.squeeze()
1172 
1173  def __len__(self):
1174  return len(self.timestimestimestimes)
1175 
1176  def __call__(self, times, ra=None, dec=None, psi=None, detector=None):
1177  """
1178  Return the antenna response function as a dictionary.
1179  """
1180 
1181  if detector is not None:
1182  self.detectordetectordetectordetector = detector
1183 
1184  if ra is not None:
1185  self.rararara = ra
1186 
1187  if dec is not None:
1188  self.decdecdecdec = dec
1189 
1190  if psi is not None:
1191  self.psipsipsipsi = psi
1192 
1193  self.timestimestimestimes = times
1194 
1195  # compute response
1196  self.compute_responsecompute_response()
1197 
1198  return self.responseresponse
1199 
1200 ## @}
def vector(self, vectorval)
Set whether to include vector polarizations.
Definition: antenna.py:531
def __init__(self, detector, ra, dec, psi=0., times=None, tensor=True, vector=False, scalar=False, use_lal=False, lookup=False, lookuppair=None, bins1=100, bins2=100)
Calculate the long-wavelength limit antenna response functions for a given ground-based gravitational...
Definition: antenna.py:260
def psi(self, psival)
Set the value of the gravitational wave polarization angle psi.
Definition: antenna.py:434
def ra(self, raval)
Set the right ascension.
Definition: antenna.py:357
def compute_response(self, times=None)
Compute the detector response.
Definition: antenna.py:938
def ra_mesh(self, val)
Definition: antenna.py:866
def x(self, resp)
Definition: antenna.py:797
def l(self, resp)
Definition: antenna.py:836
def time_mesh(self, val)
Definition: antenna.py:886
def tensor(self, tensorval)
Set whether to include tensor polarizations.
Definition: antenna.py:516
def detector(self, det)
Set the detector for which to calculate the antenna response.
Definition: antenna.py:309
def dec_mesh(self, val)
Definition: antenna.py:906
def lookup(self, val)
Set the 2d look-up table.
Definition: antenna.py:651
def _compute_response_lookup(self)
Definition: antenna.py:1088
def plus(self, resp)
Definition: antenna.py:771
def __call__(self, times, ra=None, dec=None, psi=None, detector=None)
Return the antenna response function as a dictionary.
Definition: antenna.py:1179
def scalar(self, scalarval)
Set whether to include scalar polarizations.
Definition: antenna.py:546
def cross(self, resp)
Definition: antenna.py:784
def b(self, resp)
Definition: antenna.py:823
def y(self, resp)
Definition: antenna.py:810
def times(self, timearr)
Set array of times and GPS times.
Definition: antenna.py:467
def dec(self, decval)
Set the declination.
Definition: antenna.py:392
def use_lal(self, val)
Set whether to use LAL antenna response functions.
Definition: antenna.py:561
def laldetector(self, det)
Set the lal.Detector.
Definition: antenna.py:335
def psi_mesh(self, val)
Definition: antenna.py:921
def set_lookup_pair(self, pair=['psi', 'time'], bins1=100, bins2=100)
Set the pair of parameters to use for the look-up table.
Definition: antenna.py:600
def lookup_pair(self, pair)
Definition: antenna.py:573
def _compute_response_lal(self)
Definition: antenna.py:1012