Source code for rubin_scheduler.scheduler.surveys.plan_night_survey

__all__ = ("PlanAheadSurvey",)

from copy import copy

import healpy as hp
import numpy as np

import rubin_scheduler.scheduler.basis_functions as bfs
from rubin_scheduler.scheduler import features

from .surveys import BlobSurvey


[docs] class PlanAheadSurvey(BlobSurvey): """Have a survey object that can plan ahead if it will want to observer a blob later in the night Parameters ---------- delta_mjd_tol : float The tolerance to alow on when to execute scheduled observations (days) minimum_sky_area : float The minimum sky area to demand before making a scheduled observation (square degrees) track_bands : str The band name we want to prevent long gaps on in_season : float The distance in RA from the meridian at midnight to consider (hours). This is the half-width cadence : float Ignore gaps below this length (days) """ def __init__( self, basis_functions, basis_weights, delta_mjd_tol=0.3 / 24.0, minimum_sky_area=200.0, track_bands="g", in_season=2.5, cadence=9, **kwargs, ): super(PlanAheadSurvey, self).__init__(basis_functions, basis_weights, **kwargs) # note that self.night is already being used for tracking # tesselation. So here's an attribute for seeing if the night # has changed for cadence tracking self.night_cad = -100 self.scheduled_obs = None self.delta_mjd_tol = delta_mjd_tol self.minimum_sky_area = minimum_sky_area # sq degrees self.extra_features = {} self.extra_features["last_observed"] = features.Last_observed(bandname=track_bands) self.extra_basis_functions = {} self.extra_basis_functions["moon_mask"] = bfs.Moon_avoidance_basis_function() self.track_bands = track_bands self.in_season = in_season / 12.0 * np.pi # to radians self.pix_area = hp.nside2pixarea(self.nside, degrees=True) self.cadence = cadence def check_night(self, conditions): """""" delta_mjd = conditions.mjd - self.extra_features["last_observed"].feature moon_mask = self.extra_basis_functions["moon_mask"](conditions) pix_to_obs = np.where( (delta_mjd > self.cadence) & (np.abs(conditions.az_to_antisun) < self.in_season) & (moon_mask >= 0) )[0] area = np.size(pix_to_obs) * self.pix_area # If there are going to be some observations at a given time if area > self.minimum_sky_area: # Maybe just calculate the mean (with angles) # Via https://en.wikipedia.org/wiki/ # Mean_of_circular_quantities mean_ra = np.arctan2( np.mean(np.sin(conditions.ra[pix_to_obs])), np.mean(np.cos(conditions.ra[pix_to_obs])), ) if mean_ra < 0: mean_ra += 2.0 * np.pi hour_angle = conditions.lmst - mean_ra * 12.0 / np.pi # This should be running -12 hours to +12 hours hour_angle[np.where(hour_angle < -12)] += 24 hour_angle[np.where(hour_angle > 12)] -= 24 if hour_angle < 0: self.scheduled_obs = conditions.mjd - hour_angle / 24.0 else: self.scheduled_obs = conditions.mjd else: self.scheduled_obs = None
[docs] def calc_reward_function(self, conditions): # Only compute if we will want to observe sometime in the night self.reward = -np.inf if self._check_feasibility(conditions): if self.night_cad != conditions.night: self.check_night(conditions) self.night_cad = copy(conditions.night) if self.scheduled_obs is not None: # If there are scheduled observations, and we are in # the correct time window delta_mjd = conditions.mjd - self.scheduled_obs if (np.abs(delta_mjd) < self.delta_mjd_tol) & (self.scheduled_obs is not None): # So, we think there's a region that has had a # long gap and can be observed # call the standard reward function self.reward = super(PlanAheadSurvey, self).calc_reward_function(conditions) return self.reward
def generate_observations(self, conditions): observations = super(PlanAheadSurvey, self).generate_observations(conditions) # We are providing observations, so clear the scheduled obs # and reset the night so # self.check_night will get called again in case there's # another blob that should be done after this one completes self.scheduled_obs = None self.night_cad = conditions.night - 1 return observations