Skip to content

Commit

Permalink
Added near meridian constraint
Browse files Browse the repository at this point in the history
  • Loading branch information
ramonefoster committed Aug 9, 2024
1 parent 7880daa commit fd05906
Show file tree
Hide file tree
Showing 2 changed files with 62 additions and 3 deletions.
46 changes: 45 additions & 1 deletion astroplan/constraints.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@
"LocalTimeConstraint", "PrimaryEclipseConstraint",
"SecondaryEclipseConstraint", "Constraint", "TimeConstraint",
"observability_table", "months_observable", "max_best_rescale",
"min_best_rescale", "PhaseConstraint", "is_event_observable"]
"min_best_rescale", "PhaseConstraint", "is_event_observable", "NearMeridianContraint"]

_current_year = time.localtime().tm_year # needed for backward compatibility
_current_year_time_range = Time( # needed for backward compatibility
Expand Down Expand Up @@ -938,6 +938,50 @@ def compute_constraint(self, times, observer=None, targets=None):
return mask



class NearMeridianConstraint(Constraint):
"""
Constraint near the Meridian.
"""
def __init__(self, min=None, boolean_constraint=True):
"""
min : `~astropy.units.Quantity` or `None` (optional)
Minimum acceptable distance to meridian.
`None`indicates no limit.
Examples
--------
Constrain observations to targets that are 3 degrees away from the meridian.
>>> import astropy.units as u
>>> constraint = NearMeridianConstraint(min=3*u.deg)
This can be useful for observations using German-Equatorial Mounts, to avoid
flipping the side of pier during exposures.
"""
self.min = min if min is not None else 0*u.hourangle
self.boolean_constraint = boolean_constraint

def compute_constraint(self, times, observer, targets):
lst = observer.local_sidereal_time(times)
meridian = SkyCoord(ra=lst, dec=targets.dec)

# Calculate separation from meridian
# Targets are automatically converted to SkyCoord objects
# by __call__ before compute_constraint is called.
meridian_separation = meridian.separation(targets)

if self.boolean_constraint:
mask = (self.min < meridian_separation)
return mask

# if we want to return a non-boolean score
else:
# rescale the meridian_separation values so that they become
# scores between zero and one
rescale = min_best_rescale(meridian_separation, self.min, less_than_min=0)
return rescale


def is_always_observable(constraints, observer, targets, times=None,
time_range=None, time_grid_resolution=0.5*u.hour):
"""
Expand Down
19 changes: 17 additions & 2 deletions astroplan/tests/test_constraints.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
import numpy as np
import astropy.units as u
from astropy.time import Time
from astropy.coordinates import Galactic, SkyCoord, get_sun, get_body
from astropy.coordinates import Galactic, SkyCoord, EarthLocation, get_sun, get_body
from astropy.utils import minversion
import pytest

Expand All @@ -20,7 +20,7 @@
TimeConstraint, LocalTimeConstraint, months_observable,
max_best_rescale, min_best_rescale, PhaseConstraint,
PrimaryEclipseConstraint, SecondaryEclipseConstraint,
is_event_observable)
is_event_observable, NearMeridianConstraint)
from ..periodic import EclipsingSystem
from ..exceptions import MissingConstraintWarning

Expand Down Expand Up @@ -200,6 +200,20 @@ def test_sun_separation():
assert np.all(is_constraint_met == [False, True, True])


def test_near_meridian():
time_range = Time(["2024-10-08 18:28", "2024-10-08 22:30"])
target = FixedTarget(coord=SkyCoord(ra=19.75*u.hour, dec=-22.05*u.deg), name="name")

# Pico dos Dias Observatory (Brazil)
opd = Observer(location=EarthLocation(lat=-22.53, lon=-45.58, height=1864))
constraint = NearMeridianConstraint(min=3*u.deg)
is_constraint_met = constraint(opd, opd, target,
time_range=time_range)

results = constraint(opd, target, times=time_grid_from_range(time_range))
assert np.all(results == [True, True, False, True, True])


def test_moon_separation():
time = Time('2003-04-05 06:07:08')
apo = Observer.at_site("APO")
Expand Down Expand Up @@ -419,6 +433,7 @@ def test_rescale_minmax():
AtNightConstraint(),
SunSeparationConstraint(min=90*u.deg),
MoonSeparationConstraint(min=20*u.deg),
NearMeridianConstraint(min=3*u.deg),
LocalTimeConstraint(min=dt.time(23, 50), max=dt.time(4, 8)),
TimeConstraint(*Time(["2015-08-28 03:30", "2015-09-05 10:30"]))
]
Expand Down

0 comments on commit fd05906

Please sign in to comment.