2026-01-28 15:13:51 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"net/http"
|
2026-01-28 15:17:27 +01:00
|
|
|
"os"
|
2026-01-28 15:13:51 +01:00
|
|
|
"sync"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"envguard/internal/models"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
2026-01-28 15:17:27 +01:00
|
|
|
services = []models.Service{}
|
|
|
|
|
mu sync.Mutex
|
|
|
|
|
dataFile = "services.json"
|
2026-01-28 15:13:51 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2026-01-28 15:17:27 +01:00
|
|
|
if err := loadServices(); err != nil {
|
|
|
|
|
log.Printf("⚠️ No se pudo cargar services.json, usando defaults: %v", err)
|
|
|
|
|
services = []models.Service{
|
|
|
|
|
{Name: "auth-service"},
|
|
|
|
|
{Name: "payments-api"},
|
|
|
|
|
{Name: "user-db"},
|
|
|
|
|
{Name: "notifications"},
|
|
|
|
|
{Name: "front-web"},
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
fmt.Printf("✅ %d servicios cargados desde disco\n", len(services))
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-28 15:13:51 +01:00
|
|
|
http.HandleFunc("/services", handleServices)
|
|
|
|
|
http.HandleFunc("/lock", handleLock)
|
|
|
|
|
http.HandleFunc("/unlock", handleUnlock)
|
|
|
|
|
|
|
|
|
|
fmt.Println("🚦 Lock Server corriendo en :8080")
|
|
|
|
|
if err := http.ListenAndServe(":8080", nil); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-28 15:17:27 +01:00
|
|
|
func loadServices() error {
|
|
|
|
|
file, err := os.Open(dataFile)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
return json.NewDecoder(file).Decode(&services)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func saveServices() error {
|
|
|
|
|
file, err := os.Create(dataFile)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
|
|
encoder := json.NewEncoder(file)
|
|
|
|
|
encoder.SetIndent("", " ")
|
|
|
|
|
return encoder.Encode(services)
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-28 15:13:51 +01:00
|
|
|
func handleServices(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
mu.Lock()
|
|
|
|
|
defer mu.Unlock()
|
|
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
|
|
|
json.NewEncoder(w).Encode(services)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func handleLock(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
if r.Method != http.MethodPost {
|
|
|
|
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var req models.LockRequest
|
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mu.Lock()
|
|
|
|
|
defer mu.Unlock()
|
|
|
|
|
|
|
|
|
|
for i := range services {
|
|
|
|
|
if services[i].Name == req.ServiceName {
|
|
|
|
|
if services[i].IsLocked {
|
|
|
|
|
http.Error(w, "Service already locked", http.StatusConflict)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
services[i].IsLocked = true
|
|
|
|
|
services[i].LockedBy = req.User
|
|
|
|
|
services[i].LockedAt = time.Now()
|
|
|
|
|
|
2026-01-28 15:17:27 +01:00
|
|
|
if err := saveServices(); err != nil {
|
|
|
|
|
log.Printf("Error guardando estado: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-28 15:13:51 +01:00
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
http.Error(w, "Service not found", http.StatusNotFound)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func handleUnlock(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
if r.Method != http.MethodPost {
|
|
|
|
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var req models.UnlockRequest
|
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mu.Lock()
|
|
|
|
|
defer mu.Unlock()
|
|
|
|
|
|
|
|
|
|
for i := range services {
|
|
|
|
|
if services[i].Name == req.ServiceName {
|
|
|
|
|
if !services[i].IsLocked {
|
|
|
|
|
http.Error(w, "Service is not locked", http.StatusBadRequest)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if services[i].LockedBy != req.User {
|
|
|
|
|
http.Error(w, "You cannot unlock a service locked by someone else", http.StatusForbidden)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
services[i].IsLocked = false
|
|
|
|
|
services[i].LockedBy = ""
|
|
|
|
|
services[i].LockedAt = time.Time{}
|
|
|
|
|
|
2026-01-28 15:17:27 +01:00
|
|
|
if err := saveServices(); err != nil {
|
|
|
|
|
log.Printf("Error guardando estado: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-28 15:13:51 +01:00
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
http.Error(w, "Service not found", http.StatusNotFound)
|
|
|
|
|
}
|