- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
106 lines
3.3 KiB
Go
106 lines
3.3 KiB
Go
package jupiter
|
|
|
|
import (
|
|
"encoding/json"
|
|
"math"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
const galileanShadowToleranceArcsec = 0.2
|
|
|
|
type galileanPhenomenaSample struct {
|
|
UTC string `json:"utc"`
|
|
Phenomena map[string]galileanPhenomenonExpectation `json:"phenomena"`
|
|
}
|
|
|
|
type galileanPhenomenonExpectation struct {
|
|
Transit bool `json:"transit"`
|
|
Occultation bool `json:"occultation"`
|
|
Eclipse bool `json:"eclipse"`
|
|
ShadowTransit bool `json:"shadow_transit"`
|
|
ShadowXArcsec *float64 `json:"shadow_x_arcsec,omitempty"`
|
|
ShadowYArcsec *float64 `json:"shadow_y_arcsec,omitempty"`
|
|
}
|
|
|
|
func TestGalileanPhenomenaAgainstHorizonsBaseline(t *testing.T) {
|
|
samples := loadGalileanPhenomenaBaseline(t)
|
|
maxShadowX := 0.0
|
|
maxShadowY := 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 := SatellitePhenomena(date)
|
|
for name, want := range sample.Phenomena {
|
|
phenomenon := selectGalileanPhenomenon(got, name)
|
|
if phenomenon.Transit != want.Transit {
|
|
t.Fatalf("%s transit mismatch at %s: got %v want %v", name, sample.UTC, phenomenon.Transit, want.Transit)
|
|
}
|
|
if phenomenon.Occultation != want.Occultation {
|
|
t.Fatalf("%s occultation mismatch at %s: got %v want %v", name, sample.UTC, phenomenon.Occultation, want.Occultation)
|
|
}
|
|
if phenomenon.Eclipse != want.Eclipse {
|
|
t.Fatalf("%s eclipse mismatch at %s: got %v want %v", name, sample.UTC, phenomenon.Eclipse, want.Eclipse)
|
|
}
|
|
if phenomenon.ShadowTransit != want.ShadowTransit {
|
|
t.Fatalf("%s shadow-transit mismatch at %s: got %v want %v", name, sample.UTC, phenomenon.ShadowTransit, want.ShadowTransit)
|
|
}
|
|
if !want.ShadowTransit {
|
|
continue
|
|
}
|
|
if want.ShadowXArcsec == nil || want.ShadowYArcsec == nil {
|
|
t.Fatalf("%s shadow baseline incomplete at %s", name, sample.UTC)
|
|
}
|
|
xDiff := math.Abs(phenomenon.ShadowOffsetXArcsec - *want.ShadowXArcsec)
|
|
yDiff := math.Abs(phenomenon.ShadowOffsetYArcsec - *want.ShadowYArcsec)
|
|
if xDiff > maxShadowX {
|
|
maxShadowX = xDiff
|
|
}
|
|
if yDiff > maxShadowY {
|
|
maxShadowY = yDiff
|
|
}
|
|
if xDiff > galileanShadowToleranceArcsec {
|
|
t.Fatalf("%s shadow X mismatch at %s: got %.6f want %.6f", name, sample.UTC, phenomenon.ShadowOffsetXArcsec, *want.ShadowXArcsec)
|
|
}
|
|
if yDiff > galileanShadowToleranceArcsec {
|
|
t.Fatalf("%s shadow Y mismatch at %s: got %.6f want %.6f", name, sample.UTC, phenomenon.ShadowOffsetYArcsec, *want.ShadowYArcsec)
|
|
}
|
|
}
|
|
}
|
|
t.Logf("galilean phenomena shadow max diff: X=%.3f arcsec Y=%.3f arcsec", maxShadowX, maxShadowY)
|
|
}
|
|
|
|
func loadGalileanPhenomenaBaseline(t *testing.T) []galileanPhenomenaSample {
|
|
t.Helper()
|
|
data, err := os.ReadFile("testdata/galilean_phenomena_horizons.json")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var samples []galileanPhenomenaSample
|
|
if err := json.Unmarshal(data, &samples); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(samples) == 0 {
|
|
t.Fatal("empty phenomena baseline")
|
|
}
|
|
return samples
|
|
}
|
|
|
|
func selectGalileanPhenomenon(info GalileanPhenomenaInfo, name string) GalileanSatellitePhenomenon {
|
|
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)
|
|
}
|
|
}
|