astro/semantics_regression_test.go
starainrt 3ffdbe0034
feat: 扩展天文计算能力
- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息
- 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算
- 新增木星伽利略卫星位置、现象与接触事件计算
- 新增恒星星表、星座判定、自行修正与观测辅助能力
- 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包
- 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算
- 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试
- 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试
- 更新中文和英文 README,补充示例、精度说明、SVG 配图
2026-05-01 22:38:44 +08:00

189 lines
6.7 KiB
Go

package astro_test
import (
"math"
"testing"
"time"
"b612.me/astro/basic"
"b612.me/astro/calendar"
"b612.me/astro/jupiter"
"b612.me/astro/mars"
"b612.me/astro/mercury"
"b612.me/astro/moon"
"b612.me/astro/neptune"
"b612.me/astro/saturn"
"b612.me/astro/star"
"b612.me/astro/sun"
"b612.me/astro/uranus"
"b612.me/astro/venus"
)
func nearlyEqual(a, b float64) bool {
return math.Abs(a-b) <= 1e-12
}
func TestPlanetAbsoluteQuantitiesIgnoreInputTimezone(t *testing.T) {
utc := time.Date(2026, 1, 2, 3, 4, 5, 123456789, time.UTC)
cst := time.FixedZone("CST", 8*3600)
local := utc.In(cst)
scalars := []struct {
name string
fn func(time.Time) float64
}{
{"mercury.ApparentLo", mercury.ApparentLo},
{"mercury.ApparentBo", mercury.ApparentBo},
{"mercury.ApparentRa", mercury.ApparentRa},
{"mercury.ApparentDec", mercury.ApparentDec},
{"mercury.ApparentMagnitude", mercury.ApparentMagnitude},
{"mercury.EarthDistance", mercury.EarthDistance},
{"mercury.SunDistance", mercury.SunDistance},
{"venus.ApparentLo", venus.ApparentLo},
{"venus.ApparentBo", venus.ApparentBo},
{"venus.ApparentRa", venus.ApparentRa},
{"venus.ApparentDec", venus.ApparentDec},
{"venus.ApparentMagnitude", venus.ApparentMagnitude},
{"venus.EarthDistance", venus.EarthDistance},
{"venus.SunDistance", venus.SunDistance},
{"mars.ApparentLo", mars.ApparentLo},
{"mars.ApparentBo", mars.ApparentBo},
{"mars.ApparentRa", mars.ApparentRa},
{"mars.ApparentDec", mars.ApparentDec},
{"mars.ApparentMagnitude", mars.ApparentMagnitude},
{"mars.EarthDistance", mars.EarthDistance},
{"mars.SunDistance", mars.SunDistance},
{"jupiter.ApparentLo", jupiter.ApparentLo},
{"jupiter.ApparentBo", jupiter.ApparentBo},
{"jupiter.ApparentRa", jupiter.ApparentRa},
{"jupiter.ApparentDec", jupiter.ApparentDec},
{"jupiter.ApparentMagnitude", jupiter.ApparentMagnitude},
{"jupiter.EarthDistance", jupiter.EarthDistance},
{"jupiter.SunDistance", jupiter.SunDistance},
{"saturn.ApparentLo", saturn.ApparentLo},
{"saturn.ApparentBo", saturn.ApparentBo},
{"saturn.ApparentRa", saturn.ApparentRa},
{"saturn.ApparentDec", saturn.ApparentDec},
{"saturn.ApparentMagnitude", saturn.ApparentMagnitude},
{"saturn.EarthDistance", saturn.EarthDistance},
{"saturn.SunDistance", saturn.SunDistance},
{"uranus.ApparentLo", uranus.ApparentLo},
{"uranus.ApparentBo", uranus.ApparentBo},
{"uranus.ApparentRa", uranus.ApparentRa},
{"uranus.ApparentDec", uranus.ApparentDec},
{"uranus.ApparentMagnitude", uranus.ApparentMagnitude},
{"uranus.EarthDistance", uranus.EarthDistance},
{"uranus.SunDistance", uranus.SunDistance},
{"neptune.ApparentLo", neptune.ApparentLo},
{"neptune.ApparentBo", neptune.ApparentBo},
{"neptune.ApparentRa", neptune.ApparentRa},
{"neptune.ApparentDec", neptune.ApparentDec},
{"neptune.ApparentMagnitude", neptune.ApparentMagnitude},
{"neptune.EarthDistance", neptune.EarthDistance},
{"neptune.SunDistance", neptune.SunDistance},
}
for _, tc := range scalars {
if !nearlyEqual(tc.fn(utc), tc.fn(local)) {
t.Fatalf("%s should depend on absolute time only", tc.name)
}
}
pairs := []struct {
name string
fn func(time.Time) (float64, float64)
}{
{"mercury.ApparentRaDec", mercury.ApparentRaDec},
{"venus.ApparentRaDec", venus.ApparentRaDec},
{"mars.ApparentRaDec", mars.ApparentRaDec},
{"jupiter.ApparentRaDec", jupiter.ApparentRaDec},
{"saturn.ApparentRaDec", saturn.ApparentRaDec},
{"uranus.ApparentRaDec", uranus.ApparentRaDec},
{"neptune.ApparentRaDec", neptune.ApparentRaDec},
}
for _, tc := range pairs {
leftA, leftB := tc.fn(utc)
rightA, rightB := tc.fn(local)
if !nearlyEqual(leftA, rightA) || !nearlyEqual(leftB, rightB) {
t.Fatalf("%s should depend on absolute time only", tc.name)
}
}
}
func TestJDECalcRejectsGregorianGap(t *testing.T) {
cases := []float64{5, 6.5, 10, 14.25}
for _, day := range cases {
got := basic.JDECalc(1582, 10, day)
if !math.IsNaN(got) {
t.Fatalf("1582-10-%v should be rejected, got %.15f", day, got)
}
}
before := basic.JDECalc(1582, 10, 4)
after := basic.JDECalc(1582, 10, 15)
if math.IsNaN(before) || math.IsNaN(after) {
t.Fatal("boundary dates around Gregorian reform should remain valid")
}
if !nearlyEqual(after-before, 1) {
t.Fatalf("1582-10-15 should remain the civil day after 1582-10-04")
}
}
func TestCalendarAddPreservesOriginalTimezone(t *testing.T) {
oldLocal := time.Local
time.Local = time.UTC
defer func() {
time.Local = oldLocal
}()
tz := time.FixedZone("CST", 8*3600)
start := time.Date(1985, 1, 21, 9, 30, 0, 0, tz)
lunar, err := calendar.SolarToLunar(start)
if err != nil {
t.Fatal(err)
}
expected, err := calendar.SolarToLunar(lunar.Time().Add(36 * time.Hour))
if err != nil {
t.Fatal(err)
}
shifted := lunar.Add(36 * time.Hour).Time()
if delta := shifted.Sub(expected.Time()); delta < -time.Millisecond || delta > time.Millisecond {
t.Fatalf("calendar.Time.Add should not depend on time.Local: got %v want %v", shifted, expected.Time())
}
}
func TestObservationZenithSemantics(t *testing.T) {
date := time.Date(2026, 4, 26, 9, 30, 45, 123456789, time.FixedZone("CST", 8*3600))
lon := 116.391
lat := 39.907
ra := 6.752477
dec := -16.716116
checks := []struct {
name string
altitude func() float64
zenith func() float64
}{
{"sun", func() float64 { return sun.Altitude(date, lon, lat) }, func() float64 { return sun.Zenith(date, lon, lat) }},
{"moon", func() float64 { return moon.Altitude(date, lon, lat) }, func() float64 { return moon.Zenith(date, lon, lat) }},
{"star", func() float64 { return star.Altitude(date, ra, dec, lon, lat) }, func() float64 { return star.Zenith(date, ra, dec, lon, lat) }},
{"mercury", func() float64 { return mercury.Altitude(date, lon, lat) }, func() float64 { return mercury.Zenith(date, lon, lat) }},
{"venus", func() float64 { return venus.Altitude(date, lon, lat) }, func() float64 { return venus.Zenith(date, lon, lat) }},
{"mars", func() float64 { return mars.Altitude(date, lon, lat) }, func() float64 { return mars.Zenith(date, lon, lat) }},
{"jupiter", func() float64 { return jupiter.Altitude(date, lon, lat) }, func() float64 { return jupiter.Zenith(date, lon, lat) }},
{"saturn", func() float64 { return saturn.Altitude(date, lon, lat) }, func() float64 { return saturn.Zenith(date, lon, lat) }},
{"uranus", func() float64 { return uranus.Altitude(date, lon, lat) }, func() float64 { return uranus.Zenith(date, lon, lat) }},
{"neptune", func() float64 { return neptune.Altitude(date, lon, lat) }, func() float64 { return neptune.Zenith(date, lon, lat) }},
}
for _, tc := range checks {
if !nearlyEqual(tc.zenith(), 90-tc.altitude()) {
t.Fatalf("%s zenith should equal 90-altitude", tc.name)
}
}
}