All Downloads are FREE. Search and download functionalities are using the official Maven repository.

lucuma.ags.package.scala Maven / Gradle / Ivy

// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA)
// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause

package lucuma.ags

import cats.data.NonEmptyList
import cats.syntax.all.*
import coulomb.*
import lucuma.catalog.BandsList
import lucuma.catalog.BrightnessConstraints
import lucuma.catalog.FaintnessConstraint
import lucuma.catalog.SaturationConstraint
import lucuma.core.enums.CloudExtinction
import lucuma.core.enums.GuideSpeed
import lucuma.core.enums.ImageQuality
import lucuma.core.enums.Site
import lucuma.core.enums.SkyBackground
import lucuma.core.math.Angle
import lucuma.core.math.BrightnessValue
import lucuma.core.math.Wavelength
import lucuma.core.math.skycalc.averageParallacticAngle
import lucuma.core.model.ConstraintSet
import lucuma.core.model.ObjectTracking
import lucuma.core.model.PosAngleConstraint
import lucuma.core.util.Enumerated
import lucuma.core.util.TimeSpan

import java.time.Duration
import java.time.Instant

val baseFwhm = Wavelength.fromIntNanometers(500).get

// FWHM as seen on the optical wavefront sensor (WFS)
// Operate on Double, we don't need exact precision
def wfsFwhm(sciFwhm: ImageQuality, wavelength: Wavelength): Double = {
  val coeff =
    baseFwhm.toPicometers.value.value.toDouble / wavelength.toPicometers.value.value.toDouble
  (sciFwhm.toDeciArcSeconds.value.value / 10.0) * math.pow(coeff, -0.2)

}

// Calculate the widest set of constraints, useful to cache catalog results
// We get the union off all possible constraints at the slow guide speed
// darkest sky background and 300nm wavelength
val widestConstraints: BrightnessConstraints = {
  val widest = Wavelength.fromIntNanometers(300).get

  val constraints = Enumerated[ImageQuality].all.flatMap { iq =>
    Enumerated[CloudExtinction].all.map { ce =>
      faintLimit(GuideSpeed.Slow, widest, SkyBackground.Darkest, iq, ce)
    }
  }
  // The list is never empty
  val lowest      = constraints.maximumOption.get
  BrightnessConstraints(BandsList.GaiaBandsList, lowest, none)
}

/**
 * Calculates the daintness limits for a given Guide Speed/Wavelength and conditions These are based
 * on the gaia G band and described here:
 */
def faintLimit(
  guideSpeed: GuideSpeed,
  wavelength: Wavelength,
  sb:         SkyBackground,
  iq:         ImageQuality,
  ce:         CloudExtinction
): FaintnessConstraint = {
  val limit = sb match {
    case SkyBackground.Darkest =>
      guideSpeed match {
        case GuideSpeed.Fast   =>
          16.4 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Medium =>
          16.9 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Slow   =>
          17.4 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
      }
    case SkyBackground.Dark    =>
      guideSpeed match {
        case GuideSpeed.Fast   =>
          16.3 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Medium =>
          16.8 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Slow   =>
          17.3 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
      }
    case SkyBackground.Gray    =>
      guideSpeed match {
        case GuideSpeed.Fast   =>
          16.2 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Medium =>
          16.7 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Slow   =>
          17.2 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
      }
    case SkyBackground.Bright  =>
      guideSpeed match {
        case GuideSpeed.Fast   =>
          16.1 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Medium =>
          16.6 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
        case GuideSpeed.Slow   =>
          17.1 - 0.8 * wfsFwhm(iq, wavelength) - ce.toBrightness
      }
  }
  FaintnessConstraint(BrightnessValue.unsafeFrom(BigDecimal(limit)))
}

def gaiaBrightnessConstraints(
  guideSpeed: GuideSpeed,
  wavelength: Wavelength,
  sb:         SkyBackground,
  iq:         ImageQuality,
  ce:         CloudExtinction
): BrightnessConstraints = {
  val faintness  = faintLimit(guideSpeed, wavelength, sb, iq, ce)
  val saturation = SaturationConstraint(
    BrightnessValue.unsafeFrom(faintness.brightness.value.value - 6)
  )
  BrightnessConstraints(BandsList.GaiaBandsList, faintness, saturation.some)
}

def gaiaBrightnessConstraints(
  constraints: ConstraintSet,
  guideSpeed:  GuideSpeed,
  wavelength:  Wavelength
): BrightnessConstraints =
  gaiaBrightnessConstraints(guideSpeed,
                            wavelength,
                            constraints.skyBackground,
                            constraints.imageQuality,
                            constraints.cloudExtinction
  )

private val UnconstrainedAngles =
  NonEmptyList.fromList(
    (0 until 360 by 10).map(a => Angle.fromDoubleDegrees(a.toDouble)).toList
  )

extension (posAngleConstraint: PosAngleConstraint)
  def anglesToTestAt(
    site:     Site,
    tracking: ObjectTracking,
    vizTime:  Instant,
    duration: Duration
  ): Option[NonEmptyList[Angle]] =
    TimeSpan
      .fromDuration(duration)
      .filter(_ > TimeSpan.Zero)
      .flatMap: ts =>
        posAngleConstraint match
          case PosAngleConstraint.Fixed(a)               => NonEmptyList.of(a).some
          case PosAngleConstraint.AllowFlip(a)           => NonEmptyList.of(a, a.flip).some
          case PosAngleConstraint.ParallacticOverride(a) => NonEmptyList.of(a).some
          case PosAngleConstraint.AverageParallactic     =>
            averageParallacticAngle(site.place, tracking, vizTime, ts)
              .map(a => NonEmptyList.of(a, a.flip))
          case PosAngleConstraint.Unbounded              => UnconstrainedAngles




© 2015 - 2025 Weber Informatics LLC | Privacy Policy