190 lines
5.2 KiB
Go
190 lines
5.2 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"os/signal"
|
|
"syscall"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/rss2/backend/internal/cache"
|
|
"github.com/rss2/backend/internal/config"
|
|
"github.com/rss2/backend/internal/db"
|
|
"github.com/rss2/backend/internal/handlers"
|
|
"github.com/rss2/backend/internal/middleware"
|
|
"github.com/rss2/backend/internal/services"
|
|
)
|
|
|
|
func initDB() {
|
|
ctx := context.Background()
|
|
|
|
// Crear tabla entity_aliases si no existe
|
|
_, err := db.GetPool().Exec(ctx, `
|
|
CREATE TABLE IF NOT EXISTS entity_aliases (
|
|
id SERIAL PRIMARY KEY,
|
|
canonical_name VARCHAR(255) NOT NULL,
|
|
alias VARCHAR(255) NOT NULL,
|
|
tipo VARCHAR(50) NOT NULL CHECK (tipo IN ('persona', 'organizacion', 'lugar', 'tema')),
|
|
created_at TIMESTAMP DEFAULT NOW(),
|
|
UNIQUE(alias, tipo)
|
|
)
|
|
`)
|
|
if err != nil {
|
|
log.Printf("Warning: Could not create entity_aliases table: %v", err)
|
|
} else {
|
|
log.Println("Table entity_aliases ready")
|
|
}
|
|
|
|
// Añadir columna role a users si no existe
|
|
_, err = db.GetPool().Exec(ctx, `
|
|
ALTER TABLE users ADD COLUMN IF NOT EXISTS role VARCHAR(20) DEFAULT 'user'
|
|
`)
|
|
if err != nil {
|
|
log.Printf("Warning: Could not add role column: %v", err)
|
|
} else {
|
|
log.Println("Column role ready")
|
|
}
|
|
|
|
// Crear tabla de configuración si no existe
|
|
_, err = db.GetPool().Exec(ctx, `
|
|
CREATE TABLE IF NOT EXISTS config (
|
|
key VARCHAR(100) PRIMARY KEY,
|
|
value TEXT,
|
|
updated_at TIMESTAMP DEFAULT NOW()
|
|
)
|
|
`)
|
|
if err != nil {
|
|
log.Printf("Warning: Could not create config table: %v", err)
|
|
} else {
|
|
log.Println("Table config ready")
|
|
}
|
|
|
|
// Insertar configuración por defecto si no existe
|
|
db.GetPool().Exec(ctx, `
|
|
INSERT INTO config (key, value) VALUES ('translator_type', 'cpu')
|
|
ON CONFLICT (key) DO NOTHING
|
|
`)
|
|
db.GetPool().Exec(ctx, `
|
|
INSERT INTO config (key, value) VALUES ('translator_workers', '2')
|
|
ON CONFLICT (key) DO NOTHING
|
|
`)
|
|
db.GetPool().Exec(ctx, `
|
|
INSERT INTO config (key, value) VALUES ('translator_status', 'stopped')
|
|
ON CONFLICT (key) DO NOTHING
|
|
`)
|
|
}
|
|
|
|
func main() {
|
|
cfg := config.Load()
|
|
|
|
if err := db.Connect(cfg.DatabaseURL); err != nil {
|
|
log.Fatalf("Failed to connect to database: %v", err)
|
|
}
|
|
defer db.Close()
|
|
log.Println("Connected to PostgreSQL")
|
|
|
|
// Auto-setup DB tables
|
|
initDB()
|
|
|
|
if err := cache.Connect(cfg.RedisURL); err != nil {
|
|
log.Printf("Warning: Failed to connect to Redis: %v", err)
|
|
} else {
|
|
defer cache.Close()
|
|
log.Println("Connected to Redis")
|
|
}
|
|
|
|
services.Init(cfg)
|
|
|
|
r := gin.Default()
|
|
|
|
r.Use(middleware.CORSMiddleware())
|
|
r.Use(middleware.LoggerMiddleware())
|
|
|
|
r.GET("/health", func(c *gin.Context) {
|
|
c.JSON(200, gin.H{"status": "ok"})
|
|
})
|
|
|
|
api := r.Group("/api")
|
|
{
|
|
// Serve static images downloaded by wiki_worker
|
|
api.StaticFS("/wiki-images", gin.Dir("/app/data/wiki_images", false))
|
|
|
|
api.POST("/auth/login", handlers.Login)
|
|
api.POST("/auth/register", handlers.Register)
|
|
api.GET("/auth/check-first-user", handlers.CheckFirstUser)
|
|
|
|
news := api.Group("/news")
|
|
{
|
|
news.GET("", handlers.GetNews)
|
|
news.GET("/:id", handlers.GetNewsByID)
|
|
news.DELETE("/:id", middleware.AuthRequired(), handlers.DeleteNews)
|
|
}
|
|
|
|
feeds := api.Group("/feeds")
|
|
{
|
|
feeds.GET("", handlers.GetFeeds)
|
|
feeds.GET("/export", handlers.ExportFeeds)
|
|
feeds.GET("/:id", handlers.GetFeedByID)
|
|
feeds.POST("", middleware.AuthRequired(), handlers.CreateFeed)
|
|
feeds.POST("/import", middleware.AuthRequired(), handlers.ImportFeeds)
|
|
feeds.PUT("/:id", middleware.AuthRequired(), handlers.UpdateFeed)
|
|
feeds.DELETE("/:id", middleware.AuthRequired(), handlers.DeleteFeed)
|
|
feeds.POST("/:id/toggle", middleware.AuthRequired(), handlers.ToggleFeedActive)
|
|
feeds.POST("/:id/reactivate", middleware.AuthRequired(), handlers.ReactivateFeed)
|
|
}
|
|
|
|
api.GET("/search", handlers.SearchNews)
|
|
|
|
api.GET("/entities", handlers.GetEntities)
|
|
|
|
api.GET("/stats", handlers.GetStats)
|
|
|
|
api.GET("/categories", handlers.GetCategories)
|
|
api.GET("/countries", handlers.GetCountries)
|
|
|
|
admin := api.Group("/admin")
|
|
admin.Use(middleware.AuthRequired(), middleware.AdminRequired())
|
|
{
|
|
admin.POST("/aliases", handlers.CreateAlias)
|
|
admin.GET("/aliases/export", handlers.ExportAliases)
|
|
admin.POST("/aliases/import", handlers.ImportAliases)
|
|
admin.POST("/entities/retype", handlers.PatchEntityTipo)
|
|
admin.GET("/backup", handlers.BackupDatabase)
|
|
admin.GET("/backup/news", handlers.BackupNewsZipped)
|
|
admin.GET("/users", handlers.GetUsers)
|
|
admin.POST("/users/:id/promote", handlers.PromoteUser)
|
|
admin.POST("/users/:id/demote", handlers.DemoteUser)
|
|
admin.POST("/reset-db", handlers.ResetDatabase)
|
|
admin.GET("/workers/status", handlers.GetWorkerStatus)
|
|
admin.POST("/workers/config", handlers.SetWorkerConfig)
|
|
admin.POST("/workers/start", handlers.StartWorkers)
|
|
admin.POST("/workers/stop", handlers.StopWorkers)
|
|
}
|
|
|
|
auth := api.Group("/auth")
|
|
auth.Use(middleware.AuthRequired())
|
|
{
|
|
auth.GET("/me", handlers.GetCurrentUser)
|
|
}
|
|
}
|
|
|
|
middleware.SetJWTSecret(cfg.SecretKey)
|
|
|
|
port := cfg.ServerPort
|
|
addr := fmt.Sprintf(":%s", port)
|
|
|
|
go func() {
|
|
log.Printf("Server starting on %s", addr)
|
|
if err := r.Run(addr); err != nil {
|
|
log.Fatalf("Failed to start server: %v", err)
|
|
}
|
|
}()
|
|
|
|
quit := make(chan os.Signal, 1)
|
|
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
|
|
<-quit
|
|
|
|
log.Println("Shutting down server...")
|
|
}
|