187 lines
3.9 KiB
Go
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
|
|
}
|