- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
98 lines
2.7 KiB
Go
98 lines
2.7 KiB
Go
package jupiter
|
|
|
|
import (
|
|
"encoding/json"
|
|
"math"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
const galileanHorizonsToleranceArcsec = 1.0
|
|
|
|
type galileanHorizonsEquatorialRecord struct {
|
|
RA float64 `json:"ra_deg"`
|
|
Dec float64 `json:"dec_deg"`
|
|
Delta float64 `json:"delta_au"`
|
|
}
|
|
|
|
type galileanHorizonsOffsetRecord struct {
|
|
RA float64 `json:"ra_deg"`
|
|
Dec float64 `json:"dec_deg"`
|
|
Delta float64 `json:"delta_au"`
|
|
OffsetXArcsec float64 `json:"offset_x_arcsec"`
|
|
OffsetYArcsec float64 `json:"offset_y_arcsec"`
|
|
}
|
|
|
|
type galileanHorizonsSample struct {
|
|
UTC string `json:"utc"`
|
|
Jupiter galileanHorizonsEquatorialRecord `json:"jupiter"`
|
|
Satellites map[string]galileanHorizonsOffsetRecord `json:"satellites"`
|
|
}
|
|
|
|
func TestGalileanSatellitesAgainstHorizonsRelativeOffsets(t *testing.T) {
|
|
samples := loadGalileanHorizonsBaseline(t)
|
|
maxXDiff := 0.0
|
|
maxYDiff := 0.0
|
|
for _, sample := range samples {
|
|
date, err := time.Parse(time.RFC3339, sample.UTC)
|
|
if err != nil {
|
|
t.Fatalf("parse %s: %v", sample.UTC, err)
|
|
}
|
|
got := Satellites(date)
|
|
for name, want := range sample.Satellites {
|
|
position := selectGalileanSatellite(got, name)
|
|
xDiff := math.Abs(position.OffsetXArcsec - want.OffsetXArcsec)
|
|
yDiff := math.Abs(position.OffsetYArcsec - want.OffsetYArcsec)
|
|
if xDiff > maxXDiff {
|
|
maxXDiff = xDiff
|
|
}
|
|
if yDiff > maxYDiff {
|
|
maxYDiff = yDiff
|
|
}
|
|
if xDiff > galileanHorizonsToleranceArcsec {
|
|
t.Fatalf("%s X mismatch at %s: got %.6f want %.6f", name, sample.UTC, position.OffsetXArcsec, want.OffsetXArcsec)
|
|
}
|
|
if yDiff > galileanHorizonsToleranceArcsec {
|
|
t.Fatalf("%s Y mismatch at %s: got %.6f want %.6f", name, sample.UTC, position.OffsetYArcsec, want.OffsetYArcsec)
|
|
}
|
|
wantFront := want.Delta < sample.Jupiter.Delta
|
|
if position.InFrontOfJupiter != wantFront {
|
|
t.Fatalf("%s front/back mismatch at %s: got %v want %v", name, sample.UTC, position.InFrontOfJupiter, wantFront)
|
|
}
|
|
}
|
|
}
|
|
t.Logf("galilean Horizons max diff: X=%.3f arcsec Y=%.3f arcsec", maxXDiff, maxYDiff)
|
|
}
|
|
|
|
func loadGalileanHorizonsBaseline(t *testing.T) []galileanHorizonsSample {
|
|
t.Helper()
|
|
data, err := os.ReadFile("testdata/galilean_satellites_horizons.json")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var samples []galileanHorizonsSample
|
|
if err := json.Unmarshal(data, &samples); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(samples) == 0 {
|
|
t.Fatal("empty Galilean baseline")
|
|
}
|
|
return samples
|
|
}
|
|
|
|
func selectGalileanSatellite(info GalileanSatellitesInfo, name string) GalileanSatellitePosition {
|
|
switch name {
|
|
case "io":
|
|
return info.Io
|
|
case "europa":
|
|
return info.Europa
|
|
case "ganymede":
|
|
return info.Ganymede
|
|
case "callisto":
|
|
return info.Callisto
|
|
default:
|
|
panic("unknown satellite: " + name)
|
|
}
|
|
}
|