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:
2026-02-10 11:15:54 -03:00
commit f7c8b446bf
28 changed files with 1763 additions and 0 deletions

View 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)})
}

View 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)})
}

View 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",
})
}

View 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,
})
}

View 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)},
})
}

View 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()
}
}

View 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
View 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
View 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
}