Source code for rubin_scheduler.scheduler.detailers.short_survey

__all__ = ("ShortExptDetailer",)

import warnings

import numpy as np

import rubin_scheduler.scheduler.features as features
from rubin_scheduler.scheduler.detailers import BaseDetailer
from rubin_scheduler.scheduler.utils import HpInLsstFov
from rubin_scheduler.utils import DEFAULT_NSIDE, SURVEY_START_MJD


[docs] class ShortExptDetailer(BaseDetailer): """Check if the area has been observed with a short exposure time this year. If not, add some short exposures. Parameters ---------- exp_time : `float` (1.) The short exposure time to use. nobs : `float` (2) The number of observations to try and take per year night_max : `float` (None) Do not apply any changes to the observation list if the current night is greater than night_max. n_repeat : `int` (1) How many short observations to do in a row. time_scale : `bool` (False) Should the short observations be scaled throughout the year (True), or taken as fast as possible (False). """ def __init__( self, exp_time=1.0, bandname="r", nside=DEFAULT_NSIDE, footprint=None, nobs=2, mjd0=SURVEY_START_MJD, survey_name="short", read_approx=2.0, night_max=None, n_repeat=1, time_scale=False, filtername=None, ): if filtername is not None: warnings.warn("filtername deprecated in favor of bandname", FutureWarning) bandname = filtername self.read_approx = read_approx self.exp_time = exp_time self.bandname = bandname self.nside = nside self.footprint = footprint self.nobs = nobs self.survey_name = survey_name self.mjd0 = mjd0 self.night_max = night_max self.n_repeat = n_repeat self.time_scale = time_scale self.survey_features = {} # XXX--need a feature that tracks short exposures in the band self.survey_features["nobs"] = features.N_observations( bandname=bandname, nside=nside, survey_name=self.survey_name ) # Need to be able to look up hpids for each observation self.obs2hpid = HpInLsstFov(nside=nside) def __call__(self, observation_list, conditions): # XXX--this logic would probably make more sense as a feasability # basis function. Might consider expanding the detailer base class # to include basis functions. if self.night_max is not None: if conditions.night > self.night_max: return observation_list out_observations = [] # Compute how many observations we should have taken by now if self.time_scale: n_goal = self.nobs * np.ceil(conditions.night / self.night_max) else: n_goal = self.nobs time_to_add = 0.0 for observation in observation_list: out_observations.append(observation) if observation["band"] == self.bandname: hpids = self.obs2hpid(observation["RA"], observation["dec"]) # Crop off anything outside the target footprint hpids = hpids[np.where(self.footprint[hpids] > 0)] # Crop off things where we already have enough observation hpids = hpids[np.where(self.survey_features["nobs"].feature[hpids] < n_goal)] if np.size(hpids) > 0: for i in range(0, self.n_repeat): new_obs = observation.copy() new_obs["exptime"] = self.exp_time new_obs["nexp"] = 1 new_obs["scheduler_note"] = self.survey_name out_observations.append(new_obs) time_to_add += new_obs["exptime"] + self.read_approx # pump up the flush time for observation in observation_list: observation["flush_by_mjd"] += time_to_add / 3600.0 / 24.0 return out_observations