diff --git a/Scheduler/feature_scheduler/maintel/fbs_config_sit_survey_block_t345.py b/Scheduler/feature_scheduler/maintel/fbs_config_sit_survey_block_t345.py new file mode 100644 index 00000000..1ebe6383 --- /dev/null +++ b/Scheduler/feature_scheduler/maintel/fbs_config_sit_survey_block_t345.py @@ -0,0 +1,194 @@ +# This file is part of ts_config_ocs. +# +# Developed for the Vera Rubin Observatory Telescope and Site System. +# This product includes software developed by the LSST Project +# (https://www.lsst.org). +# See the COPYRIGHT file at the top-level directory of this distribution +# for details of code ownership. +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + +import numpy as np +from rubin_scheduler.scheduler import basis_functions, detailers, example, features +from rubin_scheduler.scheduler.schedulers import CoreScheduler +from rubin_scheduler.scheduler.surveys import BlobSurvey +from rubin_scheduler.scheduler.utils import CurrentAreaMap, Footprint +from rubin_scheduler.site_models import Almanac + + +def get_scheduler(): + nside = 32 + science_program = "BLOCK-T345" + survey_start = 60653.5 + camera_rot_limits = [-80, 80] + + map_band_to_filtername = { + "u": "u_02", + "g": "g_01", + "r": "r_03", + "i": "i_06", + "z": "z_03", + "y": "y_04", + } + + filtername = map_band_to_filtername["r"] + filtername2 = map_band_to_filtername["g"] + + # Masks are fine - no band specific information. + mask_basis_functions = example.standard_masks( + nside=nside, + # Let's avoid the moon by this many degrees + moon_distance=30.0, + # Erik says to make wind speed minor so set limit high + wind_speed_maximum=50.0, + # I think these are the values appropriate for alt limits now? + min_alt=40, + max_alt=70, + min_az=0, + max_az=360, + # Avoid going into avoidance regions 30 minutes into the future + shadow_minutes=30, + ) + # Mask basis functions have zero weights. + mask_basis_functions_weights = [0 for mask in mask_basis_functions] + + # Set up footprint stuff here instead of in rubin_scheduler. + # Use the Almanac to find the position of the sun at the start of survey. + almanac = Almanac(mjd_start=survey_start) + sun_moon_info = almanac.get_sun_moon_positions(survey_start) + sun_ra_start = sun_moon_info["sun_RA"].copy() + # So this is a dictionary of filternames : int. + filterdict = {} + for i, f in enumerate(map_band_to_filtername.values()): + filterdict[f] = i + footprints = Footprint( + filters=filterdict, + mjd_start=survey_start, + sun_ra_start=sun_ra_start, + nside=nside, + ) + # need to remap this to filtername not band + footprint = CurrentAreaMap(nside=nside) + footprint_hp, labels = footprint.return_maps() + new_dtype = np.dtype([(map_band_to_filtername[f], " 0] + [ + (i[0], i[1]) for i in rf2 if i[1] > 0 + ] + # Remove the M5Diff basis functions as unusable at present + reward_functions = [ + (bf, weight) + for (bf, weight) in reward_functions + if not isinstance(bf, basis_functions.M5DiffBasisFunction) + ] + + # unpack the basis functions and weights + reward_basis_functions_weights = [val[1] for val in reward_functions] + reward_basis_functions = [val[0] for val in reward_functions] + + # Set up blob surveys. + pair_time = 20 + if filtername2 is None: + survey_name = "simple pair %i, %s" % (pair_time, filtername) + else: + survey_name = "simple pair %i, %s%s" % (pair_time, filtername, filtername2) + + # Set up detailers for each requested observation. + detailer_list = [] + # Avoid camera rotator limits. + detailer_list.append( + detailers.CameraRotDetailer( + min_rot=np.min(camera_rot_limits), max_rot=np.max(camera_rot_limits) + ) + ) + # Reorder visits in a blob so that closest to current altitude is first. + detailer_list.append(detailers.CloseAltDetailer()) + # Add a detailer to label visits as either first or second of the pair. + if filtername2 is not None: + detailer_list.append(detailers.TakeAsPairsDetailer(filtername=filtername2)) + + # Set up the survey. + ignore_obs = ["DD"] + + BlobSurvey_params = { + "slew_approx": 7.5, + "filter_change_approx": 140.0, + "read_approx": 2.4, + "flush_time": pair_time * 3, + "smoothing_kernel": None, + "nside": nside, + "seed": 42, + "dither": True, + "twilight_scale": False, + } + + pair_survey = BlobSurvey( + reward_basis_functions + mask_basis_functions, + reward_basis_functions_weights + mask_basis_functions_weights, + filtername1=filtername, + filtername2=filtername2, + exptime=150, + ideal_pair_time=pair_time, + survey_name=survey_name, + ignore_obs=ignore_obs, + nexp=1, + detailers=detailer_list, + science_program=science_program, + **BlobSurvey_params, + ) + + # Tucking this here so we can look at how many observations + # recorded for this survey and what was the last one. + pair_survey.extra_features["ObsRecorded"] = features.NObsCount() + pair_survey.extra_features["LastObs"] = features.LastObservation() + + scheduler = CoreScheduler([pair_survey], nside=nside) + return nside, scheduler + + +if __name__ == "config": + nside, scheduler = get_scheduler() diff --git a/Scheduler/observing_blocks_maintel/AOS/BLOCK-T345.json b/Scheduler/observing_blocks_maintel/AOS/BLOCK-T345.json new file mode 100644 index 00000000..9ad106ff --- /dev/null +++ b/Scheduler/observing_blocks_maintel/AOS/BLOCK-T345.json @@ -0,0 +1,79 @@ +{ + "name": "AOSSurveyMode", + "program": "BLOCK-T345", + "constraints": [], + "scripts": [ + { + "name": "maintel/track_target.py", + "standard": true, + "parameters": { + "target_name": "$name", + "slew_icrs": { + "ra": "$ra", + "dec": "$dec" + }, + "rot_value": "$rot", + "rot_type": "PhysicalSky", + "az_wrap_strategy": "NOUNWRAP" + } + }, + { + "name": "maintel/close_loop_comcam.py", + "standard": true, + "parameters": { + "exposure_time": 15, + "max_iter": 1, + "gain_sequence": [ + 0.75 + ], + "mode": "FAM", + "program": "$program", + "note": "closed_loop_aos_survey_mode", + "reason": "aos_survey_mode", + "filter": "$band_filter", + "used_dofs": [ + 0, + 1, + 2, + 3, + 4, + 5, + 6, + 7, + 8, + 9, + 10, + 11, + 12, + 13, + 14, + 17, + 18, + 19, + 20, + 21, + 22, + 23, + 28, + 29, + 30, + 31, + 32, + 33, + 34, + 37, + 38, + 39, + 40, + 41, + 42, + 45, + 46 + ], + "apply_corrections": true, + "use_ocps": true + } + } + ], + "configuration_schema": "" +} \ No newline at end of file diff --git a/Scheduler/v7/maintel_fbs_sit_block_t345.yaml b/Scheduler/v7/maintel_fbs_sit_block_t345.yaml new file mode 100644 index 00000000..df2b096e --- /dev/null +++ b/Scheduler/v7/maintel_fbs_sit_block_t345.yaml @@ -0,0 +1,47 @@ +maintel: + driver_type: feature_scheduler + mode: ADVANCE + startup_type: COLD + startup_database: /home/saluser/rubin_sim_data/fbs_observation_database_maintel.sql + instrument_name: CCCamera + models: + observatory_model: + camera: + filter_max_changes_burst_num: 1000000 + filter_max_changes_avg_num: 30000 + optics_loop_corr: + tel_optics_cl_alt_limit: + - 0 + - 60 + - 90 + park: + filter_position: r_03 + driver_configuration: + parameters: + night_boundary: -10.0 + stop_tracking_observing_script_name: maintel/stop_tracking.py + feature_scheduler_driver_configuration: + observation_database_name: /home/saluser/rubin_sim_data/fbs_observation_database_maintel.sql + scheduler_config: /net/obs-env/auto_base_packages/ts_config_ocs/Scheduler/feature_scheduler/maintel/fbs_config_sit_survey_block_t345.py + telemetry: + streams: + - name: seeing + efd_table: lsst.sal.DIMM.logevent_dimmMeasurement + efd_columns: + - fwhm + efd_delta_time: 300.0 + fill_value: 1.0 + - name: wind_speed + efd_table: lsst.sal.ESS.airFlow + efd_columns: + - speed + efd_delta_time: 300.0 + fill_value: 0.0 + csc_index: 301 + - name: wind_direction + efd_table: lsst.sal.ESS.airFlow + efd_columns: + - direction + efd_delta_time: 300.0 + fill_value: 0.0 + csc_index: 301