[docs]classDitherDetailer(BaseDetailer):""" make a uniform dither pattern. Offset by a maximum radius in a random direction. Mostly intended for DDF pointings, the BaseMarkovDF_survey class includes dithering for large areas. Parameters ---------- max_dither : `float` (0.7) The maximum dither size to use (degrees). per_night : `bool` (True) If true, us the same dither offset for an entire night nnights : `int` (7305) The number of nights to pre-generate random dithers for """def__init__(self,max_dither=0.7,seed=42,per_night=True,nnights=7305):self.survey_features={}self.current_night=-1self.max_dither=np.radians(max_dither)self.per_night=per_nightself.rng=np.random.default_rng(seed)self.angles=self.rng.random(nnights)*2*np.piself.radii=self.max_dither*np.sqrt(self.rng.random(nnights))self.offsets=(self.rng.random((nnights,2))-0.5)*2.0*self.max_ditherself.offset=Nonedef_generate_offsets(self,n_offsets,night):ifself.per_night:ifnight!=self.current_night:self.current_night=nightself.offset=self.offsets[night,:]angle=self.angles[night]radius=self.radii[night]self.offset=np.array([radius*np.cos(angle),radius*np.sin(angle)])offsets=np.tile(self.offset,(n_offsets,1))else:angle=self.rng.random(n_offsets)*2*np.piradius=self.max_dither*np.sqrt(self.rng.random(n_offsets))offsets=np.array([radius*np.cos(angle),radius*np.sin(angle)]).Treturnoffsetsdef__call__(self,obs_array,conditions):iflen(obs_array)==0:returnobs_array# Generate offsets in RA and Decoffsets=self._generate_offsets(len(obs_array),conditions.night)new_ra,new_dec=gnomonic_project_tosky(offsets[:,0],offsets[:,1],obs_array["RA"],obs_array["dec"])new_ra,new_dec=wrap_ra_dec(new_ra,new_dec)obs_array["RA"]=new_raobs_array["dec"]=new_decreturnobs_array
[docs]classDeltaCoordDitherDetailer(BaseDetailer):"""Dither pattern set by user. Each input observation is expanded to have an observation at each dither position. Parameters ---------- delta_ra : `np.array` Angular distances to move on sky in the RA-direction (degrees). delta_dec : `np.array` Angular distances to move on sky in the dec-direction (degree). delta_rotskypos : `np.array` Angular shifts to make in the rotskypos values (degrees). Default None applies no rotational shift """def__init__(self,delta_ra,delta_dec,delta_rotskypos=None):ifdelta_rotskyposisnotNone:ifnp.size(delta_ra)!=np.size(delta_rotskypos):raiseValueError("size of delta_ra (%i) is not equal to size of delta_rotskypos (%i)"%(np.size(delta_ra),np.size(delta_rotskypos)))ifnp.size(delta_ra)!=np.size(delta_dec):raiseValueError("Sizes of delta_ra (%i) and delta_dec (%i) do not match."%(np.size(delta_ra),np.size(delta_dec)))self.survey_features={}self.delta_ra=np.radians(delta_ra)self.delta_dec=np.radians(delta_dec)ifdelta_rotskyposisNone:self.delta_rotskypos=delta_rotskyposelse:self.delta_rotskypos=np.radians(delta_rotskypos)def__call__(self,obs_array,conditions):output_array_list=[]forobsinobs_array:dithered_observations=ObservationArray(self.delta_ra.size)forkeyinobs.dtype.names:dithered_observations[key]=obs[key]# Generate grid on the equator (so RA offsets are not small)new_decs=self.delta_dec.copy()# Adding 90 deg here for later rotation about x-axisnew_ras=self.delta_ra+np.pi/2# Rotate ra and dec about the x-axis# pi/2 term to convert dec to phix,y,z=thetaphi2xyz(new_ras,new_decs+np.pi/2.0)# rotate so offsets are now centered on proper decxp,yp,zp=rotx(obs["dec"],x,y,z)theta,phi=xyz2thetaphi(xp,yp,zp)new_decs=phi-np.pi/2# Remove 90 degree rot from earlier and center on proper RAnew_ras=theta-np.pi/2+obs["RA"]# Make sure coords are in proper rangenew_ras,new_decs=wrap_ra_dec(new_ras,new_decs)dithered_observations["RA"]=new_rasdithered_observations["dec"]=new_decsifself.delta_rotskyposisnotNone:dithered_observations["rotSkyPos_desired"]+=self.delta_rotskyposoutput_array_list.append(dithered_observations)result=np.concatenate(output_array_list)returnresult
[docs]classEuclidDitherDetailer(BaseDetailer):"""Directional dithering for Euclid DDFs Parameters ---------- dither_bearing_dir : `list` A list with the dither amplitude in along the axis connecting the two positions. Default [-0.25, 1] in degrees. dither_bearing_perp : `list` A list with the dither amplitude perpendicular to the axis connecting the two positions. Default [-0.25, 1] in degrees. seed : `float` Random number seed to use (42). per_night : `bool` If dither shifts should be per night (default True), or if dithers should be every pointing (False). ra_a, dec_a, ra_b, dec_b : `float` Positions for the two field centers. Default None will load the positions from rubin_scheduler.utils.ddf_locations nnights : `int` Number of nights to generate dither positions for. Default 7305 (20 years). """def__init__(self,dither_bearing_dir=[-0.25,1],dither_bearing_perp=[-0.25,0.25],seed=42,per_night=True,ra_a=None,dec_a=None,ra_b=None,dec_b=None,nnights=7305,):self.survey_features={}default_locations=ddf_locations()ifra_aisNone:self.ra_a=np.radians(default_locations["EDFS_a"][0])else:self.ra_a=np.radians(ra_a)ifra_bisNone:self.ra_b=np.radians(default_locations["EDFS_b"][0])else:self.ra_b=np.radians(ra_b)ifdec_aisNone:self.dec_a=np.radians(default_locations["EDFS_a"][1])else:self.dec_a=np.radians(dec_a)ifdec_bisNone:self.dec_b=np.radians(default_locations["EDFS_b"][1])else:self.dec_b=np.radians(dec_b)self.dither_bearing_dir=np.radians(dither_bearing_dir)self.dither_bearing_perp=np.radians(dither_bearing_perp)self.bearing_atob=bearing(self.ra_a,self.dec_a,self.ra_b,self.dec_b)self.bearing_btoa=bearing(self.ra_b,self.dec_b,self.ra_a,self.dec_a)self.current_night=-1self.per_night=per_nightself.shifted_ra_a=Noneself.shifted_dec_a=Noneself.shifted_ra_b=Noneself.shifted_dec_b=Nonerng=np.random.default_rng(seed)self.bearings_mag_1=rng.uniform(low=self.dither_bearing_dir.min(),high=self.dither_bearing_dir.max(),size=nnights,)self.perp_mag_1=rng.uniform(low=self.dither_bearing_perp.min(),high=self.dither_bearing_perp.max(),size=nnights,)self.bearings_mag_2=rng.uniform(low=self.dither_bearing_dir.min(),high=self.dither_bearing_dir.max(),size=nnights,)self.perp_mag_2=rng.uniform(low=self.dither_bearing_perp.min(),high=self.dither_bearing_perp.max(),size=nnights,)def_generate_offsets(self,n_offsets,night):ifself.per_night:ifnight!=self.current_night:self.current_night=nightbearing_mag=self.bearings_mag_1[night]perp_mag=self.perp_mag_1[night]# Move point a along the bearingsself.shifted_dec_a,self.shifted_ra_a=dest_latlon(bearing_mag,self.bearing_atob,self.dec_a,self.ra_a)self.shifted_dec_a,self.shifted_ra_a=dest_latlon(perp_mag,self.bearing_atob+np.pi/2.0,self.shifted_dec_a,self.shifted_ra_a,)# Shift the second positionbearing_mag=self.bearings_mag_2[night]perp_mag=self.perp_mag_2[night]self.shifted_dec_b,self.shifted_ra_b=dest_latlon(bearing_mag,self.bearing_btoa,self.dec_b,self.ra_b)self.shifted_dec_b,self.shifted_ra_b=dest_latlon(perp_mag,self.bearing_btoa+np.pi/2.0,self.shifted_dec_b,self.shifted_ra_b,)else:raiseValueError("not implamented")return(self.shifted_ra_a,self.shifted_dec_a,self.shifted_ra_b,self.shifted_dec_b,)def__call__(self,obs_array,conditions):# Generate offsets in RA and Decra_a,dec_a,ra_b,dec_b=self._generate_offsets(len(obs_array),conditions.night)fori,obsinenumerate(obs_array):if"DD:EDFS_a"inobs["scheduler_note"][0:9]:obs_array[i]["RA"]=ra_aobs_array[i]["dec"]=dec_aelif"DD:EDFS_b"inobs["scheduler_note"][0:9]:obs_array[i]["RA"]=ra_bobs_array[i]["dec"]=dec_belse:raiseValueError("scheduler_note does not contain EDFS_a or EDFS_b.")returnobs_array
[docs]classCameraRotDetailer(BaseDetailer):""" Randomly set the camera rotation, either for each exposure, or per night. Parameters ---------- max_rot : `float` (90.) The maximum amount to offset the camera (degrees) min_rot : `float` (90) The minimum to offset the camera (degrees) dither : `str` If "night", change positions per night. If call, change per call. If "all", randomize per visit. Default "night". telescope : `str` Telescope name. Options of "rubin" or "auxtel". Default "rubin". nnights : `int` Number of dither positions to generate. """def__init__(self,max_rot=90.0,min_rot=-90.0,dither="night",per_night=None,seed=42,nnights=7305,telescope="rubin",):self.survey_features={}ifper_nightisTrue:warnings.warn("per_night deprecated, setting dither='night'",FutureWarning)dither="night"ifper_nightisFalse:warnings.warn("per_night deprecated, setting dither='all'",FutureWarning)dither="all"ifditherisTrue:warnings.warn("dither=True deprecated, swapping to dither='night'",FutureWarning)dither="night"ifditherisFalse:warnings.warn("dither=False deprecated, swapping to dither='all'",FutureWarning)dither="all"self.current_night=-1self.max_rot=np.radians(max_rot)self.min_rot=np.radians(min_rot)self.range=self.max_rot-self.min_rotself.dither=ditherself.rng=np.random.default_rng(seed)ifdither=="call":self.offsets=self.rng.random((nnights,nnights))else:self.offsets=self.rng.random(nnights)self.offset=Noneself.rc=rotation_converter(telescope=telescope)self.call_num=0def_generate_offsets(self,n_offsets,night):ifself.dither=="night":ifnight!=self.current_night:self.current_night=nightself.offset=self.offsets[night]*self.range+self.min_rotoffsets=np.ones(n_offsets)*self.offsetelifself.dither=="call":ifnight!=self.current_night:self.current_night=nightself.call_num=0self.offset=self.offsets[night][self.call_num]*self.range+self.min_rotself.call_num+=1offsets=np.ones(n_offsets)*self.offsetelifself.dither=="all":self.rng=np.random.default_rng()offsets=self.rng.random(n_offsets)*self.range+self.min_rotelse:raiseValueError("dither kwarg must be set to 'night', 'call', or 'all'.")returnoffsetsdef__call__(self,observation_array,conditions):# Generate offsets in camamera rotatoroffsets=self._generate_offsets(len(observation_array),conditions.night)alt,az=_approx_ra_dec2_alt_az(observation_array["RA"],observation_array["dec"],conditions.site.latitude_rad,conditions.site.longitude_rad,conditions.mjd,)obs_pa=_approx_altaz2pa(alt,az,conditions.site.latitude_rad)observation_array["rotSkyPos"]=self.rc._rottelpos2rotskypos(offsets,obs_pa)observation_array["rotTelPos"]=offsetsreturnobservation_array
[docs]classCameraSmallRotPerObservationListDetailer(BaseDetailer):""" Randomly set the camera rotation for each observation list. Generates a small sequential offset for sequential visits in the same band; adds a random offset for each band change. Parameters ---------- max_rot : `float`, optional The maximum amount to offset the camera (degrees). Default of 85 allows some padding for camera rotator. min_rot : `float`, optional The minimum to offset the camera (degrees) Default of -85 allows some padding for camera rotator. seed : `int`, optional Seed for random number generation (per night). per_visit_rot : `float`, optional Sequential rotation to add per visit. telescope : `str`, optional Telescope name. Options of "rubin" or "auxtel". Default "rubin". This is used to determine conversions between rotSkyPos and rotTelPos. """def__init__(self,max_rot=85.0,min_rot=-85.0,seed=42,per_visit_rot=0.0,telescope="rubin"):self.survey_features={}self.current_night=-1self.max_rot=np.radians(max_rot)self.min_rot=np.radians(min_rot)self.rot_range=self.max_rot-self.min_rotself.seed=seedself.per_visit_rot=np.radians(per_visit_rot)self.offset=Noneself.rc=rotation_converter(telescope=telescope)def_generate_offsets_band_change(self,band_list,mjd,initial_offset):"""Generate a random camera rotation for each band change or add a small offset for each sequential observation. """mjd_hash=round(100*(np.asarray(mjd).item()%100))rng=np.random.default_rng(mjd_hash*self.seed)offsets=np.zeros(len(band_list))# Find the locations of the band changesband_changes=np.where(np.array(band_list[:-1])!=np.array(band_list[1:]))[0]band_changes=np.concatenate([np.array([-1]),band_changes])# But add one because of counting and offsets above.band_changes+=1# Count visits per band in the sequence.nvis_per_band=np.concatenate([np.diff(band_changes),np.array([len(band_list)-1-band_changes[-1]])])# Set up the random rotator offsets for each band change# This includes first rotation .. maybe not needed?forfchange_idx,nvis_finzip(band_changes,nvis_per_band):rot_range=self.rot_range-self.per_visit_rot*nvis_f# At the band change spot, update to random offsetoffsets[fchange_idx:]=rng.random()*rot_range+self.min_rot# After the band change point, add incremental rotation# (we'll wipe this when we get to next fchange_idx)offsets[fchange_idx:]+=self.per_visit_rot*np.arange(len(band_list)-fchange_idx)offsets=np.where(offsets>self.max_rot,self.max_rot,offsets)returnoffsetsdef__call__(self,observation_list,conditions):# Generate offsets in camera rotatorband_list=[np.asarray(obs["band"]).item()forobsinobservation_list]offsets=self._generate_offsets_band_change(band_list,conditions.mjd,conditions.rot_tel_pos)fori,obsinenumerate(observation_list):alt,az=_approx_ra_dec2_alt_az(obs["RA"],obs["dec"],conditions.site.latitude_rad,conditions.site.longitude_rad,conditions.mjd,)obs_pa=_approx_altaz2pa(alt,az,conditions.site.latitude_rad)obs["rotSkyPos"]=self.rc._rottelpos2rotskypos(offsets[i],obs_pa)obs["rotTelPos"]=offsets[i]returnobservation_list
[docs]classComCamGridDitherDetailer(BaseDetailer):""" Generate an offset pattern to synthesize a 2x2 grid of ComCam pointings. Parameters ---------- rotTelPosDesired : `float`, (0.) The physical rotation angle of the camera rotator (degrees) scale : `float` (0.355) Half of the offset between grid pointing centers. (degrees) dither : `float` (0.05) Dither offsets within grid to fill chip gaps. (degrees) telescope : `str`, ("comcam") Telescope name. Default "comcam". This is used to determine conversions between rotSkyPos and rotTelPos. """def__init__(self,rotTelPosDesired=0.0,scale=0.355,dither=0.05,telescope="comcam"):self.survey_features={}self.rotTelPosDesired=np.radians(rotTelPosDesired)self.scale=np.radians(scale)self.dither=np.radians(dither)self.rc=rotation_converter(telescope=telescope)def_rotate(self,x,y,angle):x_rot=x*np.cos(angle)-y*np.sin(angle)y_rot=x*np.sin(angle)+y*np.cos(angle)returnx_rot,y_rotdef_generate_offsets(self,n_offsets,band_list,rotSkyPos):# 2 x 2 pointing gridx_grid=np.array([-1.0*self.scale,-1.0*self.scale,self.scale,self.scale])y_grid=np.array([-1.0*self.scale,self.scale,self.scale,-1.0*self.scale])x_grid_rot,y_grid_rot=self._rotate(x_grid,y_grid,-1.0*rotSkyPos)offsets_grid_rot=np.array([x_grid_rot,y_grid_rot]).T# Dither pattern within grid to fill chip gaps# Psuedo-random offsetsx_dither=np.array([0.0,-0.5*self.dither,-1.25*self.dither,1.5*self.dither,0.75*self.dither,])y_dither=np.array([0.0,-0.75*self.dither,1.5*self.dither,1.25*self.dither,-0.5*self.dither,])x_dither_rot,y_dither_rot=self._rotate(x_dither,y_dither,-1.0*rotSkyPos)offsets_dither_rot=np.array([x_dither_rot,y_dither_rot]).T# Find the indices of the band changesband_changes=np.where(np.array(band_list[:-1])!=np.array(band_list[1:]))[0]band_changes=np.concatenate([np.array([-1]),band_changes])band_changes+=1offsets=[]index_band=0foriiinrange(0,n_offsets):ifiiinband_changes:# Reset the count after each band changeindex_band=0index_grid=index_band%4index_dither=np.floor(index_band/4).astype(int)%5offsets.append(offsets_grid_rot[index_grid]+offsets_dither_rot[index_dither])index_band+=1returnnp.vstack(offsets)def__call__(self,observation_list,conditions):iflen(observation_list)==0:returnobservation_listband_list=[np.asarray(obs["band"]).item()forobsinobservation_list]# Initial estimate of rotSkyPos corresponding to desired rotTelPosalt,az,pa=_approx_ra_dec2_alt_az(observation_list[0]["RA"],observation_list[0]["dec"],conditions.site.latitude_rad,conditions.site.longitude_rad,conditions.mjd,return_pa=True,)rotSkyPos=self.rc._rottelpos2rotskypos(self.rotTelPosDesired,pa)# Generate offsets in RA and Decoffsets=self._generate_offsets(len(observation_list),band_list,rotSkyPos)# Project offsets onto skyobs_array=np.concatenate(observation_list)new_ra,new_dec=gnomonic_project_tosky(offsets[:,0],offsets[:,1],obs_array["RA"],obs_array["dec"])new_ra,new_dec=wrap_ra_dec(new_ra,new_dec)# Update observationsforiiinrange(0,len(observation_list)):observation_list[ii]["RA"]=new_ra[ii]observation_list[ii]["dec"]=new_dec[ii]alt,az,pa=_approx_ra_dec2_alt_az(new_ra[ii],new_dec[ii],conditions.site.latitude_rad,conditions.site.longitude_rad,conditions.mjd,return_pa=True,)observation_list[ii]["rotSkyPos"]=rotSkyPosobservation_list[ii]["rotTelPos"]=self.rc._rotskypos2rottelpos(rotSkyPos,pa)returnobservation_list