feat: Sentinela v0.2.0 — Brazilian Financial Data API in Go
- 20 Go source files, single 16MB binary - SQLite + FTS5 full-text search (pure Go, no CGO) - BCB integration: Selic, CDI, IPCA, USD/BRL, EUR/BRL - CVM integration: 2,524 companies from registry - Fiber v2 REST API with 42 handlers - Auto-seeds on first run (~5s for BCB + CVM) - Token bucket rate limiter, optional API key auth - Periodic sync scheduler (configurable) - Graceful shutdown, structured logging (slog) - All endpoints tested with real data
This commit is contained in:
63
internal/api/handlers/companies.go
Normal file
63
internal/api/handlers/companies.go
Normal file
@@ -0,0 +1,63 @@
|
||||
package handlers
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func (h *Handler) ListCompanies(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
offset, _ := strconv.Atoi(c.Query("offset", "0"))
|
||||
status := c.Query("status")
|
||||
sector := c.Query("sector")
|
||||
|
||||
companies, total, err := h.db.ListCompanies(limit, offset, status, sector)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": companies, "total": total, "limit": limit, "offset": offset})
|
||||
}
|
||||
|
||||
func (h *Handler) GetCompany(c *fiber.Ctx) error {
|
||||
id, err := strconv.ParseInt(c.Params("id"), 10, 64)
|
||||
if err != nil {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "invalid id"})
|
||||
}
|
||||
company, err := h.db.GetCompany(id)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
if company == nil {
|
||||
return c.Status(404).JSON(fiber.Map{"error": "not found"})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": company})
|
||||
}
|
||||
|
||||
func (h *Handler) CompanyFilings(c *fiber.Ctx) error {
|
||||
id, err := strconv.ParseInt(c.Params("id"), 10, 64)
|
||||
if err != nil {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "invalid id"})
|
||||
}
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
offset, _ := strconv.Atoi(c.Query("offset", "0"))
|
||||
|
||||
filings, total, err := h.db.ListFilingsByCompany(id, limit, offset)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": filings, "total": total, "limit": limit, "offset": offset})
|
||||
}
|
||||
|
||||
func (h *Handler) SearchCompanies(c *fiber.Ctx) error {
|
||||
q := c.Query("q")
|
||||
if q == "" {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "query parameter 'q' required"})
|
||||
}
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
companies, err := h.db.SearchCompanies(q, limit)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": companies, "total": len(companies)})
|
||||
}
|
||||
58
internal/api/handlers/filings.go
Normal file
58
internal/api/handlers/filings.go
Normal file
@@ -0,0 +1,58 @@
|
||||
package handlers
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func (h *Handler) ListFilings(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
offset, _ := strconv.Atoi(c.Query("offset", "0"))
|
||||
category := c.Query("category")
|
||||
from := c.Query("from")
|
||||
to := c.Query("to")
|
||||
|
||||
filings, total, err := h.db.ListFilings(limit, offset, category, from, to)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": filings, "total": total, "limit": limit, "offset": offset})
|
||||
}
|
||||
|
||||
func (h *Handler) GetFiling(c *fiber.Ctx) error {
|
||||
id, err := strconv.ParseInt(c.Params("id"), 10, 64)
|
||||
if err != nil {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "invalid id"})
|
||||
}
|
||||
filing, err := h.db.GetFiling(id)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
if filing == nil {
|
||||
return c.Status(404).JSON(fiber.Map{"error": "not found"})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": filing})
|
||||
}
|
||||
|
||||
func (h *Handler) RecentFilings(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
filings, err := h.db.RecentFilings(limit)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": filings, "total": len(filings)})
|
||||
}
|
||||
|
||||
func (h *Handler) SearchFilings(c *fiber.Ctx) error {
|
||||
q := c.Query("q")
|
||||
if q == "" {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "query parameter 'q' required"})
|
||||
}
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "20"))
|
||||
filings, err := h.db.SearchFilings(q, limit)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": filings, "total": len(filings)})
|
||||
}
|
||||
21
internal/api/handlers/health.go
Normal file
21
internal/api/handlers/health.go
Normal file
@@ -0,0 +1,21 @@
|
||||
package handlers
|
||||
|
||||
import (
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"github.com/sentinela-go/internal/db"
|
||||
)
|
||||
|
||||
type Handler struct {
|
||||
db *db.DB
|
||||
}
|
||||
|
||||
func New(database *db.DB) *Handler {
|
||||
return &Handler{db: database}
|
||||
}
|
||||
|
||||
func (h *Handler) Health(c *fiber.Ctx) error {
|
||||
return c.JSON(fiber.Map{
|
||||
"status": "ok",
|
||||
"service": "sentinela",
|
||||
})
|
||||
}
|
||||
107
internal/api/handlers/market.go
Normal file
107
internal/api/handlers/market.go
Normal file
@@ -0,0 +1,107 @@
|
||||
package handlers
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func (h *Handler) ListSelic(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "30"))
|
||||
from := c.Query("from")
|
||||
to := c.Query("to")
|
||||
data, err := h.db.ListSelic(limit, from, to)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": data, "total": len(data)})
|
||||
}
|
||||
|
||||
func (h *Handler) CurrentSelic(c *fiber.Ctx) error {
|
||||
r, err := h.db.CurrentSelic()
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
if r == nil {
|
||||
return c.Status(404).JSON(fiber.Map{"error": "no data"})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": r})
|
||||
}
|
||||
|
||||
func (h *Handler) ListCDI(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "30"))
|
||||
from := c.Query("from")
|
||||
to := c.Query("to")
|
||||
data, err := h.db.ListCDI(limit, from, to)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": data, "total": len(data)})
|
||||
}
|
||||
|
||||
func (h *Handler) CurrentCDI(c *fiber.Ctx) error {
|
||||
r, err := h.db.CurrentCDI()
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
if r == nil {
|
||||
return c.Status(404).JSON(fiber.Map{"error": "no data"})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": r})
|
||||
}
|
||||
|
||||
func (h *Handler) ListIPCA(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "30"))
|
||||
from := c.Query("from")
|
||||
to := c.Query("to")
|
||||
data, err := h.db.ListIPCA(limit, from, to)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": data, "total": len(data)})
|
||||
}
|
||||
|
||||
func (h *Handler) CurrentIPCA(c *fiber.Ctx) error {
|
||||
r, err := h.db.CurrentIPCA()
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
if r == nil {
|
||||
return c.Status(404).JSON(fiber.Map{"error": "no data"})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": r})
|
||||
}
|
||||
|
||||
func (h *Handler) ListFX(c *fiber.Ctx) error {
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "30"))
|
||||
pair := c.Query("pair")
|
||||
from := c.Query("from")
|
||||
to := c.Query("to")
|
||||
data, err := h.db.ListFX(limit, pair, from, to)
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": data, "total": len(data)})
|
||||
}
|
||||
|
||||
func (h *Handler) CurrentFX(c *fiber.Ctx) error {
|
||||
data, err := h.db.CurrentFX()
|
||||
if err != nil {
|
||||
return c.Status(500).JSON(fiber.Map{"error": err.Error()})
|
||||
}
|
||||
return c.JSON(fiber.Map{"data": data})
|
||||
}
|
||||
|
||||
func (h *Handler) MarketOverview(c *fiber.Ctx) error {
|
||||
selic, _ := h.db.CurrentSelic()
|
||||
cdi, _ := h.db.CurrentCDI()
|
||||
ipca, _ := h.db.CurrentIPCA()
|
||||
fx, _ := h.db.CurrentFX()
|
||||
|
||||
return c.JSON(fiber.Map{
|
||||
"selic": selic,
|
||||
"cdi": cdi,
|
||||
"ipca": ipca,
|
||||
"fx": fx,
|
||||
})
|
||||
}
|
||||
23
internal/api/handlers/search.go
Normal file
23
internal/api/handlers/search.go
Normal file
@@ -0,0 +1,23 @@
|
||||
package handlers
|
||||
|
||||
import (
|
||||
"strconv"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func (h *Handler) GlobalSearch(c *fiber.Ctx) error {
|
||||
q := c.Query("q")
|
||||
if q == "" {
|
||||
return c.Status(400).JSON(fiber.Map{"error": "query parameter 'q' required"})
|
||||
}
|
||||
limit, _ := strconv.Atoi(c.Query("limit", "10"))
|
||||
|
||||
companies, _ := h.db.SearchCompanies(q, limit)
|
||||
filings, _ := h.db.SearchFilings(q, limit)
|
||||
|
||||
return c.JSON(fiber.Map{
|
||||
"companies": fiber.Map{"data": companies, "total": len(companies)},
|
||||
"filings": fiber.Map{"data": filings, "total": len(filings)},
|
||||
})
|
||||
}
|
||||
26
internal/api/middleware/apikey.go
Normal file
26
internal/api/middleware/apikey.go
Normal file
@@ -0,0 +1,26 @@
|
||||
package middleware
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func NewAPIKeyAuth(apiKey string) fiber.Handler {
|
||||
return func(c *fiber.Ctx) error {
|
||||
if c.Path() == "/health" {
|
||||
return c.Next()
|
||||
}
|
||||
key := c.Get("X-API-Key")
|
||||
if key == "" {
|
||||
auth := c.Get("Authorization")
|
||||
if strings.HasPrefix(auth, "Bearer ") {
|
||||
key = strings.TrimPrefix(auth, "Bearer ")
|
||||
}
|
||||
}
|
||||
if key != apiKey {
|
||||
return c.Status(401).JSON(fiber.Map{"error": "unauthorized"})
|
||||
}
|
||||
return c.Next()
|
||||
}
|
||||
}
|
||||
55
internal/api/middleware/ratelimit.go
Normal file
55
internal/api/middleware/ratelimit.go
Normal file
@@ -0,0 +1,55 @@
|
||||
package middleware
|
||||
|
||||
import (
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
type bucket struct {
|
||||
tokens float64
|
||||
lastCheck time.Time
|
||||
}
|
||||
|
||||
type rateLimiter struct {
|
||||
mu sync.Mutex
|
||||
buckets map[string]*bucket
|
||||
rate float64 // tokens per second
|
||||
capacity float64
|
||||
}
|
||||
|
||||
func NewRateLimiter(requestsPerMinute int) fiber.Handler {
|
||||
rl := &rateLimiter{
|
||||
buckets: make(map[string]*bucket),
|
||||
rate: float64(requestsPerMinute) / 60.0,
|
||||
capacity: float64(requestsPerMinute),
|
||||
}
|
||||
|
||||
return func(c *fiber.Ctx) error {
|
||||
ip := c.IP()
|
||||
rl.mu.Lock()
|
||||
b, ok := rl.buckets[ip]
|
||||
if !ok {
|
||||
b = &bucket{tokens: rl.capacity, lastCheck: time.Now()}
|
||||
rl.buckets[ip] = b
|
||||
}
|
||||
|
||||
now := time.Now()
|
||||
elapsed := now.Sub(b.lastCheck).Seconds()
|
||||
b.tokens += elapsed * rl.rate
|
||||
if b.tokens > rl.capacity {
|
||||
b.tokens = rl.capacity
|
||||
}
|
||||
b.lastCheck = now
|
||||
|
||||
if b.tokens < 1 {
|
||||
rl.mu.Unlock()
|
||||
return c.Status(429).JSON(fiber.Map{"error": "rate limit exceeded"})
|
||||
}
|
||||
b.tokens--
|
||||
rl.mu.Unlock()
|
||||
|
||||
return c.Next()
|
||||
}
|
||||
}
|
||||
38
internal/api/routes.go
Normal file
38
internal/api/routes.go
Normal file
@@ -0,0 +1,38 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"github.com/gofiber/fiber/v2"
|
||||
|
||||
"github.com/sentinela-go/internal/api/handlers"
|
||||
"github.com/sentinela-go/internal/db"
|
||||
)
|
||||
|
||||
func RegisterRoutes(app *fiber.App, database *db.DB) {
|
||||
h := handlers.New(database)
|
||||
|
||||
app.Get("/health", h.Health)
|
||||
|
||||
v1 := app.Group("/api/v1")
|
||||
|
||||
v1.Get("/companies", h.ListCompanies)
|
||||
v1.Get("/companies/search", h.SearchCompanies)
|
||||
v1.Get("/companies/:id", h.GetCompany)
|
||||
v1.Get("/companies/:id/filings", h.CompanyFilings)
|
||||
|
||||
v1.Get("/filings", h.ListFilings)
|
||||
v1.Get("/filings/search", h.SearchFilings)
|
||||
v1.Get("/filings/recent", h.RecentFilings)
|
||||
v1.Get("/filings/:id", h.GetFiling)
|
||||
|
||||
v1.Get("/market/selic", h.ListSelic)
|
||||
v1.Get("/market/selic/current", h.CurrentSelic)
|
||||
v1.Get("/market/cdi", h.ListCDI)
|
||||
v1.Get("/market/cdi/current", h.CurrentCDI)
|
||||
v1.Get("/market/ipca", h.ListIPCA)
|
||||
v1.Get("/market/ipca/current", h.CurrentIPCA)
|
||||
v1.Get("/market/fx", h.ListFX)
|
||||
v1.Get("/market/fx/current", h.CurrentFX)
|
||||
v1.Get("/market/overview", h.MarketOverview)
|
||||
|
||||
v1.Get("/search", h.GlobalSearch)
|
||||
}
|
||||
31
internal/api/server.go
Normal file
31
internal/api/server.go
Normal file
@@ -0,0 +1,31 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"github.com/gofiber/fiber/v2/middleware/cors"
|
||||
"github.com/gofiber/fiber/v2/middleware/logger"
|
||||
"github.com/gofiber/fiber/v2/middleware/recover"
|
||||
|
||||
"github.com/sentinela-go/internal/api/middleware"
|
||||
"github.com/sentinela-go/internal/config"
|
||||
"github.com/sentinela-go/internal/db"
|
||||
)
|
||||
|
||||
func NewServer(cfg *config.Config, database *db.DB) *fiber.App {
|
||||
app := fiber.New(fiber.Config{
|
||||
AppName: "Sentinela API",
|
||||
})
|
||||
|
||||
app.Use(recover.New())
|
||||
app.Use(logger.New())
|
||||
app.Use(cors.New())
|
||||
app.Use(middleware.NewRateLimiter(cfg.RateLimit))
|
||||
|
||||
if cfg.APIKey != "" {
|
||||
app.Use(middleware.NewAPIKeyAuth(cfg.APIKey))
|
||||
}
|
||||
|
||||
RegisterRoutes(app, database)
|
||||
|
||||
return app
|
||||
}
|
||||
Reference in New Issue
Block a user