astro/baseline_regression_test.go

124 lines
3.8 KiB
Go
Raw Normal View History

package astro_test
import (
"encoding/json"
"math"
"os"
"testing"
"time"
"b612.me/astro/basic"
"b612.me/astro/moon"
"b612.me/astro/planet"
"b612.me/astro/sun"
)
type baselinePlanetSnapshot struct {
Name string `json:"name"`
XT int `json:"xt"`
LonBits uint64 `json:"lon_bits"`
LatBits uint64 `json:"lat_bits"`
RadBits uint64 `json:"rad_bits"`
}
type baselineMoonSnapshot struct {
LonBits uint64 `json:"lon_bits"`
LatBits uint64 `json:"lat_bits"`
DisBits uint64 `json:"dis_bits"`
}
type baselineSample struct {
UTC string `json:"utc"`
TTJD float64 `json:"tt_jd"`
Planets []baselinePlanetSnapshot `json:"planets"`
Moon baselineMoonSnapshot `json:"moon"`
}
func loadBaselineSamples(t *testing.T) []baselineSample {
t.Helper()
data, err := os.ReadFile("testdata/planet_moon_baseline.json")
if err != nil {
t.Fatal(err)
}
var samples []baselineSample
if err := json.Unmarshal(data, &samples); err != nil {
t.Fatal(err)
}
if len(samples) == 0 {
t.Fatal("empty baseline samples")
}
return samples
}
func TestPlanetMoonBaselineRegression(t *testing.T) {
samples := loadBaselineSamples(t)
for _, sample := range samples {
for _, body := range sample.Planets {
gotLon := planet.WherePlanet(body.XT, 0, sample.TTJD)
if math.Float64bits(gotLon) != body.LonBits {
t.Fatalf("%s lon regression at %s", body.Name, sample.UTC)
}
gotLonN := planet.WherePlanetN(body.XT, 0, sample.TTJD, -1)
if math.Float64bits(gotLonN) != body.LonBits {
t.Fatalf("%s lon full-n regression at %s", body.Name, sample.UTC)
}
gotLat := planet.WherePlanet(body.XT, 1, sample.TTJD)
if math.Float64bits(gotLat) != body.LatBits {
t.Fatalf("%s lat regression at %s", body.Name, sample.UTC)
}
gotLatN := planet.WherePlanetN(body.XT, 1, sample.TTJD, -1)
if math.Float64bits(gotLatN) != body.LatBits {
t.Fatalf("%s lat full-n regression at %s", body.Name, sample.UTC)
}
gotRad := planet.WherePlanet(body.XT, 2, sample.TTJD)
if math.Float64bits(gotRad) != body.RadBits {
t.Fatalf("%s rad regression at %s", body.Name, sample.UTC)
}
gotRadN := planet.WherePlanetN(body.XT, 2, sample.TTJD, -1)
if math.Float64bits(gotRadN) != body.RadBits {
t.Fatalf("%s rad full-n regression at %s", body.Name, sample.UTC)
}
}
if math.Float64bits(basic.HMoonTrueLo(sample.TTJD)) != sample.Moon.LonBits {
t.Fatalf("moon lon regression at %s", sample.UTC)
}
if math.Float64bits(basic.HMoonTrueLoN(sample.TTJD, -1)) != sample.Moon.LonBits {
t.Fatalf("moon lon full-n regression at %s", sample.UTC)
}
if math.Float64bits(basic.HMoonTrueBo(sample.TTJD)) != sample.Moon.LatBits {
t.Fatalf("moon lat regression at %s", sample.UTC)
}
if math.Float64bits(basic.HMoonTrueBoN(sample.TTJD, -1)) != sample.Moon.LatBits {
t.Fatalf("moon lat full-n regression at %s", sample.UTC)
}
if math.Float64bits(basic.HMoonAway(sample.TTJD)) != sample.Moon.DisBits {
t.Fatalf("moon distance regression at %s", sample.UTC)
}
if math.Float64bits(basic.HMoonAwayN(sample.TTJD, -1)) != sample.Moon.DisBits {
t.Fatalf("moon distance full-n regression at %s", sample.UTC)
}
}
}
func TestPublicTruncationFullMatchesDefault(t *testing.T) {
date := time.Date(2026, 1, 2, 3, 4, 5, 123456789, time.UTC)
if math.Float64bits(sun.TrueLo(date)) != math.Float64bits(sun.TrueLoN(date, -1)) {
t.Fatal("sun.TrueLoN(-1) should match default")
}
if math.Float64bits(sun.TrueBo(date)) != math.Float64bits(sun.TrueBoN(date, -1)) {
t.Fatal("sun.TrueBoN(-1) should match default")
}
if math.Float64bits(moon.TrueLo(date)) != math.Float64bits(moon.TrueLoN(date, -1)) {
t.Fatal("moon.TrueLoN(-1) should match default")
}
if math.Float64bits(moon.TrueBo(date)) != math.Float64bits(moon.TrueBoN(date, -1)) {
t.Fatal("moon.TrueBoN(-1) should match default")
}
}