Files
StockRadar/internal/etoro/poller.go
T
2026-04-20 22:51:41 +02:00

187 lines
3.9 KiB
Go

package etoro
import (
"fmt"
"log"
"sync"
"time"
"git.rouggy.com/rouggy/stockradar/internal/db"
)
type SyncStatus struct {
Syncing bool `json:"syncing"`
Progress int `json:"progress"`
Total int `json:"total"`
Count int `json:"count"`
LastSync time.Time `json:"last_sync"`
LastError string `json:"last_error,omitempty"`
}
type Poller struct {
db *db.DB
client *Client
getKeys func() (apiKey, userKey string, err error)
ticker *time.Ticker
done chan struct{}
mu sync.Mutex
syncing bool
progress int
total int
lastSync time.Time
lastError string
}
func NewPoller(database *db.DB, getKeys func() (string, string, error)) *Poller {
return &Poller{
db: database,
client: New(),
getKeys: getKeys,
done: make(chan struct{}),
}
}
func (p *Poller) Start() {
p.ticker = time.NewTicker(24 * time.Hour)
go func() {
// Sync uniquement si la DB est vide
if p.dbCount() == 0 {
if err := p.Sync(); err != nil {
log.Printf("etoro poller: initial sync: %v", err)
}
} else {
log.Printf("etoro: %d instruments déjà en DB, sync ignorée au démarrage", p.dbCount())
}
for {
select {
case <-p.ticker.C:
if err := p.Sync(); err != nil {
log.Printf("etoro poller: sync: %v", err)
}
case <-p.done:
return
}
}
}()
}
func (p *Poller) Stop() {
if p.ticker != nil {
p.ticker.Stop()
}
close(p.done)
}
func (p *Poller) Status() SyncStatus {
p.mu.Lock()
defer p.mu.Unlock()
return SyncStatus{
Syncing: p.syncing,
Progress: p.progress,
Total: p.total,
Count: p.dbCount(),
LastSync: p.lastSync,
LastError: p.lastError,
}
}
func (p *Poller) Sync() error {
p.mu.Lock()
if p.syncing {
p.mu.Unlock()
return nil // déjà en cours
}
p.syncing = true
p.progress = 0
p.total = 0
p.lastError = ""
p.mu.Unlock()
defer func() {
p.mu.Lock()
p.syncing = false
p.lastSync = time.Now()
p.mu.Unlock()
}()
apiKey, userKey, err := p.getKeys()
if err != nil || apiKey == "" || userKey == "" {
p.mu.Lock()
p.lastError = "clés API eToro non configurées (Settings)"
p.mu.Unlock()
return fmt.Errorf("etoro: clés manquantes")
}
p.client.SetKeys(apiKey, userKey)
log.Println("etoro: fetching instruments…")
const pageSize = 500
inserted := 0
fetched := 0
for page := 1; ; page++ {
sr, err := p.client.fetchPage(5, pageSize, page)
if err != nil {
p.mu.Lock()
p.lastError = err.Error()
p.mu.Unlock()
log.Printf("etoro: fetch error page %d: %v", page, err)
return err
}
// On connaît le total dès la première page
if page == 1 {
p.mu.Lock()
p.total = sr.TotalItems
p.mu.Unlock()
log.Printf("etoro: %d stocks à synchroniser", sr.TotalItems)
}
for _, s := range sr.Items {
if s.IsHidden || s.IsDelisted || !s.IsBuyEnabled {
continue
}
_, err := p.db.Exec(`
INSERT INTO instruments (instrument_id, ticker, name, exchange_id, asset_class_id, synced_at)
VALUES (?, ?, ?, ?, ?, CURRENT_TIMESTAMP)
ON CONFLICT(instrument_id) DO UPDATE SET
ticker = excluded.ticker,
name = excluded.name,
exchange_id = excluded.exchange_id,
synced_at = CURRENT_TIMESTAMP
`, s.InstrumentID, s.SymbolFull, s.InstrumentDisplayName,
s.ExchangeID, s.AssetClassID)
if err == nil {
inserted++
}
fetched++
}
p.mu.Lock()
p.progress = fetched
p.mu.Unlock()
log.Printf("etoro: page %d — %d/%d traités", page, fetched, sr.TotalItems)
if page*pageSize >= sr.TotalItems {
break
}
}
log.Printf("etoro: sync terminée — %d instruments en DB", inserted)
return nil
}
func (p *Poller) dbCount() int {
var n int
p.db.QueryRow(`SELECT COUNT(*) FROM instruments`).Scan(&n)
return n
}
// IsEtoro vérifie si un ticker est dans l'univers eToro.
func IsEtoro(database *db.DB, ticker string) bool {
var count int
database.QueryRow(`SELECT COUNT(*) FROM instruments WHERE ticker = ?`, ticker).Scan(&count)
return count > 0
}