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