- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
214 lines
7.9 KiB
Go
214 lines
7.9 KiB
Go
package basic
|
|
|
|
import (
|
|
"encoding/json"
|
|
"math"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
type moonMaxDeclinationSample struct {
|
|
Kind string `json:"kind"`
|
|
Year int `json:"year"`
|
|
Month int `json:"month"`
|
|
TimeUTC string `json:"time_utc"`
|
|
DeclinationDeg float64 `json:"declination_deg"`
|
|
}
|
|
|
|
type moonMaxDeclinationMonthState struct {
|
|
north []DeclinationEvent
|
|
south []DeclinationEvent
|
|
northI int
|
|
southI int
|
|
}
|
|
|
|
func TestMoonMaximumDeclinationsMatchHorizonsBaseline(t *testing.T) {
|
|
// Baseline is generated from JPL Horizons by scripts/generate_moon_max_declination_baseline.sh.
|
|
data, err := os.ReadFile("testdata/moon_max_declination_baseline.json")
|
|
if err != nil {
|
|
t.Fatalf("read baseline: %v", err)
|
|
}
|
|
|
|
var samples []moonMaxDeclinationSample
|
|
if err := json.Unmarshal(data, &samples); err != nil {
|
|
t.Fatalf("decode baseline: %v", err)
|
|
}
|
|
if len(samples) == 0 {
|
|
t.Fatal("empty moon maximum declination baseline")
|
|
}
|
|
|
|
const timeTolerance = 15 * time.Second
|
|
const declinationToleranceDeg = 0.0002
|
|
|
|
states := make(map[int]*moonMaxDeclinationMonthState)
|
|
var maxTimeDiff time.Duration
|
|
var maxDeclinationDiff float64
|
|
|
|
for _, sample := range samples {
|
|
wantTime, err := time.Parse(time.RFC3339Nano, sample.TimeUTC)
|
|
if err != nil {
|
|
t.Fatalf("parse sample time %q: %v", sample.TimeUTC, err)
|
|
}
|
|
|
|
key := sample.Year*100 + sample.Month
|
|
state := states[key]
|
|
if state == nil {
|
|
state = &moonMaxDeclinationMonthState{
|
|
north: MoonMaximumNorthDeclinations(sample.Year, time.Month(sample.Month)),
|
|
south: MoonMaximumSouthDeclinations(sample.Year, time.Month(sample.Month)),
|
|
}
|
|
states[key] = state
|
|
}
|
|
|
|
var got DeclinationEvent
|
|
switch sample.Kind {
|
|
case "north":
|
|
if state.northI >= len(state.north) {
|
|
t.Fatalf("%04d-%02d missing north declination event #%d", sample.Year, sample.Month, state.northI+1)
|
|
}
|
|
got = state.north[state.northI]
|
|
state.northI++
|
|
case "south":
|
|
if state.southI >= len(state.south) {
|
|
t.Fatalf("%04d-%02d missing south declination event #%d", sample.Year, sample.Month, state.southI+1)
|
|
}
|
|
got = state.south[state.southI]
|
|
state.southI++
|
|
default:
|
|
t.Fatalf("unknown declination kind %q", sample.Kind)
|
|
}
|
|
|
|
gotTime := JDE2DateByZone(got.JDE, time.UTC, false)
|
|
timeDiff := gotTime.Sub(wantTime)
|
|
if timeDiff < 0 {
|
|
timeDiff = -timeDiff
|
|
}
|
|
if timeDiff > maxTimeDiff {
|
|
maxTimeDiff = timeDiff
|
|
}
|
|
if timeDiff > timeTolerance {
|
|
t.Fatalf("%s %04d-%02d time mismatch: got %s want %s tolerance %v", sample.Kind, sample.Year, sample.Month, gotTime.Format(time.RFC3339Nano), sample.TimeUTC, timeTolerance)
|
|
}
|
|
|
|
declinationDiff := math.Abs(got.Declination - sample.DeclinationDeg)
|
|
if declinationDiff > maxDeclinationDiff {
|
|
maxDeclinationDiff = declinationDiff
|
|
}
|
|
if declinationDiff > declinationToleranceDeg {
|
|
t.Fatalf("%s %04d-%02d declination mismatch: got %.8f want %.8f tolerance %.8f", sample.Kind, sample.Year, sample.Month, got.Declination, sample.DeclinationDeg, declinationToleranceDeg)
|
|
}
|
|
}
|
|
|
|
for key, state := range states {
|
|
year := key / 100
|
|
month := key % 100
|
|
if state.northI != len(state.north) {
|
|
t.Fatalf("%04d-%02d unconsumed north events: got %d of %d", year, month, state.northI, len(state.north))
|
|
}
|
|
if state.southI != len(state.south) {
|
|
t.Fatalf("%04d-%02d unconsumed south events: got %d of %d", year, month, state.southI, len(state.south))
|
|
}
|
|
}
|
|
|
|
t.Logf("moon maximum declination max diff: time=%v declination=%.8f deg", maxTimeDiff, maxDeclinationDiff)
|
|
}
|
|
|
|
func TestMoonMaximumDeclinationSignsAndOrder(t *testing.T) {
|
|
north := MoonMaximumNorthDeclinations(2026, time.January)
|
|
south := MoonMaximumSouthDeclinations(2026, time.January)
|
|
if len(north) == 0 || len(south) == 0 {
|
|
t.Fatalf("expected both north and south events in 2026-01, got north=%d south=%d", len(north), len(south))
|
|
}
|
|
for i, event := range north {
|
|
if event.Declination <= 0 {
|
|
t.Fatalf("north event #%d should be positive, got %.8f", i+1, event.Declination)
|
|
}
|
|
if i > 0 && !(north[i-1].JDE < event.JDE) {
|
|
t.Fatalf("north events not strictly increasing: %.12f then %.12f", north[i-1].JDE, event.JDE)
|
|
}
|
|
}
|
|
for i, event := range south {
|
|
if event.Declination >= 0 {
|
|
t.Fatalf("south event #%d should be negative, got %.8f", i+1, event.Declination)
|
|
}
|
|
if i > 0 && !(south[i-1].JDE < event.JDE) {
|
|
t.Fatalf("south events not strictly increasing: %.12f then %.12f", south[i-1].JDE, event.JDE)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestMoonMaximumDeclinationSearchMatchesMonthlyEvents(t *testing.T) {
|
|
query := time.Date(2026, time.January, 10, 0, 0, 0, 0, time.UTC)
|
|
queryJDE := Date2JDE(query)
|
|
|
|
northEvents := append([]DeclinationEvent{}, MoonMaximumNorthDeclinations(2025, time.December)...)
|
|
northEvents = append(northEvents, MoonMaximumNorthDeclinations(2026, time.January)...)
|
|
northEvents = append(northEvents, MoonMaximumNorthDeclinations(2026, time.February)...)
|
|
|
|
southEvents := append([]DeclinationEvent{}, MoonMaximumSouthDeclinations(2025, time.December)...)
|
|
southEvents = append(southEvents, MoonMaximumSouthDeclinations(2026, time.January)...)
|
|
southEvents = append(southEvents, MoonMaximumSouthDeclinations(2026, time.February)...)
|
|
|
|
assertSameDeclinationEvent(t, "last north", LastMoonMaximumNorthDeclination(queryJDE), expectedDirectionalDeclinationEvent(northEvents, queryJDE, -1, true))
|
|
assertSameDeclinationEvent(t, "next north", NextMoonMaximumNorthDeclination(queryJDE), expectedDirectionalDeclinationEvent(northEvents, queryJDE, 1, false))
|
|
assertSameDeclinationEvent(t, "closest north", ClosestMoonMaximumNorthDeclination(queryJDE), expectedClosestDeclinationEvent(northEvents, queryJDE))
|
|
|
|
assertSameDeclinationEvent(t, "last south", LastMoonMaximumSouthDeclination(queryJDE), expectedDirectionalDeclinationEvent(southEvents, queryJDE, -1, true))
|
|
assertSameDeclinationEvent(t, "next south", NextMoonMaximumSouthDeclination(queryJDE), expectedDirectionalDeclinationEvent(southEvents, queryJDE, 1, false))
|
|
assertSameDeclinationEvent(t, "closest south", ClosestMoonMaximumSouthDeclination(queryJDE), expectedClosestDeclinationEvent(southEvents, queryJDE))
|
|
}
|
|
|
|
func TestMoonMaximumDeclinationSearchAtExactEventTime(t *testing.T) {
|
|
north := MoonMaximumNorthDeclinations(2026, time.January)
|
|
if len(north) < 2 {
|
|
t.Fatalf("expected at least two north events spanning Jan 2026 search window, got %d", len(north))
|
|
}
|
|
|
|
exactJDE := north[0].JDE
|
|
assertSameDeclinationEvent(t, "exact last north", LastMoonMaximumNorthDeclination(exactJDE), north[0])
|
|
assertSameDeclinationEvent(t, "exact closest north", ClosestMoonMaximumNorthDeclination(exactJDE), north[0])
|
|
assertSameDeclinationEvent(t, "exact next north", NextMoonMaximumNorthDeclination(exactJDE), north[1])
|
|
}
|
|
|
|
func assertSameDeclinationEvent(t *testing.T, name string, got, want DeclinationEvent) {
|
|
t.Helper()
|
|
if math.Abs(got.JDE-want.JDE) > 1e-12 {
|
|
t.Fatalf("%s JDE mismatch: got %.12f want %.12f", name, got.JDE, want.JDE)
|
|
}
|
|
if math.Float64bits(got.Declination) != math.Float64bits(want.Declination) {
|
|
t.Fatalf("%s declination mismatch: got %.12f want %.12f", name, got.Declination, want.Declination)
|
|
}
|
|
}
|
|
|
|
func expectedDirectionalDeclinationEvent(events []DeclinationEvent, queryJDE float64, direction int, includeCurrent bool) DeclinationEvent {
|
|
var (
|
|
found bool
|
|
best DeclinationEvent
|
|
)
|
|
for _, event := range events {
|
|
delta := event.JDE - queryJDE
|
|
if !moonMaximumDeclinationMatchesDirection(delta, direction, includeCurrent) {
|
|
continue
|
|
}
|
|
if !found {
|
|
best = event
|
|
found = true
|
|
continue
|
|
}
|
|
if math.Abs(delta) < math.Abs(best.JDE-queryJDE) || (math.Abs(delta) == math.Abs(best.JDE-queryJDE) && event.JDE < best.JDE) {
|
|
best = event
|
|
}
|
|
}
|
|
return best
|
|
}
|
|
|
|
func expectedClosestDeclinationEvent(events []DeclinationEvent, queryJDE float64) DeclinationEvent {
|
|
last := expectedDirectionalDeclinationEvent(events, queryJDE, -1, true)
|
|
next := expectedDirectionalDeclinationEvent(events, queryJDE, 1, false)
|
|
if math.Abs(queryJDE-last.JDE) <= math.Abs(next.JDE-queryJDE) {
|
|
return last
|
|
}
|
|
return next
|
|
}
|