feat(cosmobiologia): rectificador per-segundo + direcciones primarias reales

El rectificador deja la aproximación y pasa a la trigonometría exacta,
con precisión de segundo — el "microajuste argentino".

LA MATEMÁTICA. El rectificador ya NO usa el modelo simplificado
(directed_longitude, rotación uniforme de RA + convergencia GR). Ahora
usa `eternal_astrology::primary_direction::all_directions` — el método
Placidus-mundano: semi-arcos diurnos/nocturnos bajo el polo de cada
cuerpo, la trigonometría esférica de la escuela ascensional. No se
reimplementó nada: la matemática, ya probada, vive en eternal; el
engine sólo aporta la capa de optimización.

- error_de_carta: por cada evento, la distancia en años a la dirección
  primaria que perfecciona más cerca; el error total es la suma. Es la
  función de coste del microajuste — el valle es la hora real.

PRECISIÓN DE SEGUNDO. compute_natal_chart / build_eternal_inputs /
natal_cache pasan a trabajar en SEGUNDOS (compose convierte ×60). El
rectificador barre en dos pasadas: gruesa minuto a minuto sobre la
ventana (el perfil que dibuja la curva), fina segundo a segundo en
±60 s alrededor del mejor minuto.

- Rectificacion: mejor_offset_segundos; el perfil va en segundos.
- UI: panel y curva muestran «±Xm Ys · error N.NNa». Las barras siguen
  siendo clicables (scrub a esa hora candidata).

Tests verdes (engine 12, render 28). Limitación conocida: all_directions
es sólo directo — converso necesita crecer en eternal (upstream).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
sergio
2026-05-22 16:56:06 +00:00
parent 5fdae159f0
commit 36d6645e7f
6 changed files with 165 additions and 167 deletions
@@ -9,7 +9,7 @@
//! Este cache de 8 entradas es suficiente: el usuario rara vez tiene
//! más de 2 cartas activas a la vez (natal + partner) y el LRU bota la
//! más vieja cuando se llena. La clave es el **contenido** de
//! `StoredBirthData + StoredChartConfig + offset_minutes`, así que
//! `StoredBirthData + StoredChartConfig + offset_seconds`, así que
//! editar una carta invalida automáticamente su entrada.
use std::collections::hash_map::DefaultHasher;
@@ -70,7 +70,7 @@ fn cache() -> &'static Mutex<Cache> {
pub fn key_for(
birth: &StoredBirthData,
config: &StoredChartConfig,
offset_minutes: i64,
offset_seconds: i64,
) -> u64 {
let mut h = DefaultHasher::new();
// Birth data — fecha/hora/lugar.
@@ -95,8 +95,8 @@ pub fn key_for(
config.include_lilith.hash(&mut h);
config.include_main_belt_asteroids.hash(&mut h);
config.include_fixed_stars.hash(&mut h);
// Offset temporal (rectificación rápida).
offset_minutes.hash(&mut h);
// Offset temporal en segundos (microajuste de rectificación).
offset_seconds.hash(&mut h);
h.finish()
}