astro/basic/planet_transit_test.go
starainrt bec7b8a0d8
feat: 增强日月食搜索、沙罗周期与内行星凌日
- 使用压缩表加速查找日月食沙罗周期信息
- 优化日月食搜索跳步,减少非食季朔望月扫描
- 新增本地日全食、日环食、月全食搜索接口,返回 ok 区分未找到结果
- 新增水星、金星地心凌日查询及测试
2026-05-03 19:00:08 +08:00

146 lines
4.2 KiB
Go

package basic
import (
"math"
"testing"
"time"
)
func TestKnownMercuryTransits(t *testing.T) {
tests := []struct {
name string
query time.Time
greatest time.Time
}{
{
name: "2016 May",
query: time.Date(2016, 1, 1, 0, 0, 0, 0, time.UTC),
greatest: time.Date(2016, 5, 9, 14, 57, 0, 0, time.UTC),
},
{
name: "2019 Nov",
query: time.Date(2019, 1, 1, 0, 0, 0, 0, time.UTC),
greatest: time.Date(2019, 11, 11, 15, 20, 0, 0, time.UTC),
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
result := NextMercuryTransit(Date2JDE(tc.query))
if !result.Valid {
t.Fatal("expected valid transit")
}
got := JDE2DateByZone(result.Greatest, time.UTC, false)
t.Logf("start=%s greatest=%s end=%s min=%.3f sun=%.3f planet=%.3f",
JDE2DateByZone(result.ExternalIngress, time.UTC, false),
got,
JDE2DateByZone(result.ExternalEgress, time.UTC, false),
result.MinimumSeparationArcsec,
result.SunSemidiameterArcsec,
result.PlanetSemidiameterArcsec,
)
if math.Abs(got.Sub(tc.greatest).Seconds()) > 20*60 {
t.Fatalf("greatest mismatch: got %s want near %s", got, tc.greatest)
}
if !result.HasInternal {
t.Fatalf("expected internal contacts")
}
if !(result.ExternalIngress < result.InternalIngress &&
result.InternalIngress < result.Greatest &&
result.Greatest < result.InternalEgress &&
result.InternalEgress < result.ExternalEgress) {
t.Fatalf("contacts out of order: %+v", result)
}
})
}
}
func TestKnownVenusTransits(t *testing.T) {
tests := []struct {
name string
query time.Time
greatest time.Time
}{
{
name: "2004 Jun",
query: time.Date(2004, 1, 1, 0, 0, 0, 0, time.UTC),
greatest: time.Date(2004, 6, 8, 8, 20, 0, 0, time.UTC),
},
{
name: "2012 Jun",
query: time.Date(2012, 1, 1, 0, 0, 0, 0, time.UTC),
greatest: time.Date(2012, 6, 6, 1, 29, 0, 0, time.UTC),
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
result := NextVenusTransit(Date2JDE(tc.query))
if !result.Valid {
t.Fatal("expected valid transit")
}
got := JDE2DateByZone(result.Greatest, time.UTC, false)
t.Logf("start=%s greatest=%s end=%s min=%.3f sun=%.3f planet=%.3f",
JDE2DateByZone(result.ExternalIngress, time.UTC, false),
got,
JDE2DateByZone(result.ExternalEgress, time.UTC, false),
result.MinimumSeparationArcsec,
result.SunSemidiameterArcsec,
result.PlanetSemidiameterArcsec,
)
if math.Abs(got.Sub(tc.greatest).Seconds()) > 20*60 {
t.Fatalf("greatest mismatch: got %s want near %s", got, tc.greatest)
}
if !result.HasInternal {
t.Fatalf("expected internal contacts")
}
if !(result.ExternalIngress < result.InternalIngress &&
result.InternalIngress < result.Greatest &&
result.Greatest < result.InternalEgress &&
result.InternalEgress < result.ExternalEgress) {
t.Fatalf("contacts out of order: %+v", result)
}
})
}
}
func TestTransitSearchSkipsSparseEvents(t *testing.T) {
mercuryResult := NextMercuryTransit(Date2JDE(time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC)))
if !mercuryResult.Valid {
t.Fatal("expected Mercury transit")
}
mercuryGreatest := JDE2DateByZone(mercuryResult.Greatest, time.UTC, false)
if mercuryGreatest.Year() != 2032 || mercuryGreatest.Month() != time.November {
t.Fatalf("unexpected next Mercury transit: %s", mercuryGreatest)
}
venusResult := NextVenusTransit(Date2JDE(time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC)))
if !venusResult.Valid {
t.Fatal("expected Venus transit")
}
venusGreatest := JDE2DateByZone(venusResult.Greatest, time.UTC, false)
if venusGreatest.Year() != 2117 || venusGreatest.Month() != time.December {
t.Fatalf("unexpected next Venus transit: %s", venusGreatest)
}
}
func BenchmarkNextMercuryTransitFrom2026(b *testing.B) {
jd := Date2JDE(time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC))
for i := 0; i < b.N; i++ {
result := NextMercuryTransit(jd)
if !result.Valid {
b.Fatal("expected valid transit")
}
}
}
func BenchmarkNextVenusTransitFrom2026(b *testing.B) {
jd := Date2JDE(time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC))
for i := 0; i < b.N; i++ {
result := NextVenusTransit(jd)
if !result.Valid {
b.Fatal("expected valid transit")
}
}
}