package main import ( "encoding/json" "fmt" "net/http" "os" "strconv" "time" "git.highperfocused.tech/highperfocused/lumina-relay/relay/cache" "git.highperfocused.tech/highperfocused/lumina-relay/relay/trending" "github.com/fiatjaf/eventstore/postgresql" "github.com/fiatjaf/khatru" "github.com/fiatjaf/khatru/policies" ) // Cache for storing generic data like event counts var dataCache = cache.New() const eventCountCacheKey = "total_event_count" const eventCountCacheDuration = 1 * time.Minute func getEnv(key, fallback string) string { if value, ok := os.LookupEnv(key); ok { return value } return fallback } // Gets total event count, using cache if available func getTotalEventCount(db *postgresql.PostgresBackend) (int, error) { // Try getting from cache first if cachedCount, ok := dataCache.Get(eventCountCacheKey); ok { return cachedCount.(int), nil } // If not in cache, query the database count := 0 row := db.DB.QueryRow("SELECT COUNT(*) FROM event") if err := row.Scan(&count); err != nil { return 0, err } // Update the cache dataCache.Set(eventCountCacheKey, count, eventCountCacheDuration) return count, nil } // Updates event count in the background periodically func startEventCountUpdater(db *postgresql.PostgresBackend) { go func() { ticker := time.NewTicker(eventCountCacheDuration) defer ticker.Stop() for range ticker.C { count := 0 row := db.DB.QueryRow("SELECT COUNT(*) FROM event") if err := row.Scan(&count); err != nil { fmt.Printf("Error updating event count: %v\n", err) continue } dataCache.Set(eventCountCacheKey, count, eventCountCacheDuration) fmt.Printf("Updated event count cache: %d events\n", count) } }() } func main() { fmt.Print(` LUMINA RELAY `) // create the relay instance relay := khatru.NewRelay() // set up relay properties with environment variable configuration relay.Info.Name = getEnv("RELAY_NAME", "LUMINA Relay") relay.Info.PubKey = getEnv("RELAY_PUBKEY", "79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798") relay.Info.Description = getEnv("RELAY_DESCRIPTION", "LUMINA Relay") relay.Info.Icon = getEnv("RELAY_ICON", "https://external-content.duckduckgo.com/iu/?u=https%3A%2F%2Fliquipedia.net%2Fcommons%2Fimages%2F3%2F35%2FSCProbe.jpg&f=1&nofb=1&ipt=0cbbfef25bce41da63d910e86c3c343e6c3b9d63194ca9755351bb7c2efa3359&ipo=images") relay.Info.Software = "lumina-relay" relay.Info.Version = "0.1.0" // Print relay information fmt.Printf("Name: %s\n", relay.Info.Name) fmt.Printf("Public Key: %s\n", relay.Info.PubKey) fmt.Printf("Description: %s\n\n", relay.Info.Description) // Configure PostgreSQL connection with environment variable postgresURL := getEnv("POSTGRES_URL", "postgres://postgres:postgres@postgres/postgres?sslmode=disable") db := postgresql.PostgresBackend{DatabaseURL: postgresURL} if err := db.Init(); err != nil { panic(err) } // Initialize trending system to start background calculations fmt.Println("Initializing trending system...") if err := trending.Initialize(db.DB.DB); err != nil { fmt.Printf("Warning: Error initializing trending system: %v\n", err) } // Initialize event count cache and start periodic updates fmt.Println("Initializing event count cache...") _, err := getTotalEventCount(&db) if err != nil { fmt.Printf("Warning: Error initializing event count cache: %v\n", err) } startEventCountUpdater(&db) relay.StoreEvent = append(relay.StoreEvent, db.SaveEvent) relay.QueryEvents = append(relay.QueryEvents, db.QueryEvents) relay.DeleteEvent = append(relay.DeleteEvent, db.DeleteEvent) relay.ReplaceEvent = append(relay.ReplaceEvent, db.ReplaceEvent) relay.CountEvents = append(relay.CountEvents, db.CountEvents) relay.RejectEvent = append( relay.RejectEvent, policies.PreventLargeTags(120), policies.PreventTimestampsInThePast(time.Hour*2), policies.PreventTimestampsInTheFuture(time.Minute*30), ) mux := relay.Router() // set up other http handlers mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("content-type", "text/html") // Get event count from cache count, err := getTotalEventCount(&db) if err != nil { fmt.Printf("Error getting event count: %v\n", err) count = 0 // Fall back to zero if there's an error } // Improved HTML content with link to stats page fmt.Fprintf(w, ` LUMINA Relay

Welcome to LUMINA Relay!

Number of events stored: %d

View Event Stats

View Trending History

`, count) }) mux.HandleFunc("/stats", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("content-type", "text/html") // Query the number of events for each kind, sorted by kind rows, err := db.DB.Query("SELECT kind, COUNT(*) FROM event GROUP BY kind ORDER BY kind") if err != nil { fmt.Printf("Error querying event kinds: %v\n", err) return } defer rows.Close() stats := make(map[string]int) for rows.Next() { var kind string var count int if err := rows.Scan(&kind, &count); err != nil { fmt.Printf("Error scanning row: %v\n", err) return } stats[kind] = count } // Improved HTML content for stats fmt.Fprintf(w, ` LUMINA Relay Stats

Event Stats

`) for kind, count := range stats { fmt.Fprintf(w, ` `, kind, count) } fmt.Fprintf(w, `
Kind Count
%s %d
`) }) mux.HandleFunc("/api/stats", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") // Get total count from cache totalCount, err := getTotalEventCount(&db) if err != nil { http.Error(w, fmt.Sprintf("Error getting event count: %v", err), http.StatusInternalServerError) return } // Query the number of events for each kind, sorted by kind rows, err := db.DB.Query("SELECT kind, COUNT(*) FROM event GROUP BY kind ORDER BY kind") if err != nil { http.Error(w, fmt.Sprintf("Error querying event kinds: %v", err), http.StatusInternalServerError) return } defer rows.Close() stats := make(map[string]int) for rows.Next() { var kind string var count int if err := rows.Scan(&kind, &count); err != nil { http.Error(w, fmt.Sprintf("Error scanning row: %v", err), http.StatusInternalServerError) return } stats[kind] = count } // Add total count to the stats response := map[string]interface{}{ "total": totalCount, "kinds": stats, } // Encode stats to JSON and write to response if err := json.NewEncoder(w).Encode(response); err != nil { http.Error(w, fmt.Sprintf("Error encoding JSON: %v", err), http.StatusInternalServerError) } }) mux.HandleFunc("/api/trending/kind20", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") trendingPosts, err := trending.GetTrendingKind20(db.DB.DB) if err != nil { http.Error(w, fmt.Sprintf("Error getting trending posts: %v", err), http.StatusInternalServerError) return } response := map[string]interface{}{ "trending": trendingPosts, } if err := json.NewEncoder(w).Encode(response); err != nil { http.Error(w, fmt.Sprintf("Error encoding JSON: %v", err), http.StatusInternalServerError) } }) // Add endpoint for trending history mux.HandleFunc("/api/trending/history/kind20", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Type", "application/json") // Parse query parameters for pagination limitStr := r.URL.Query().Get("limit") offsetStr := r.URL.Query().Get("offset") limit := 10 // Default limit offset := 0 // Default offset if limitStr != "" { if val, err := strconv.Atoi(limitStr); err == nil && val > 0 { limit = val } } if offsetStr != "" { if val, err := strconv.Atoi(offsetStr); err == nil && val >= 0 { offset = val } } // Get trending history for kind 20 history, err := trending.GetTrendingHistoryForKind(db.DB.DB, 20, limit, offset) if err != nil { http.Error(w, fmt.Sprintf("Error getting trending history: %v", err), http.StatusInternalServerError) return } // Get total count for pagination info totalCount, err := trending.GetTrendingHistoryCount(db.DB.DB, 20) if err != nil { http.Error(w, fmt.Sprintf("Error getting trending history count: %v", err), http.StatusInternalServerError) return } response := map[string]interface{}{ "history": history, "pagination": map[string]interface{}{ "total": totalCount, "limit": limit, "offset": offset, }, } if err := json.NewEncoder(w).Encode(response); err != nil { http.Error(w, fmt.Sprintf("Error encoding JSON: %v", err), http.StatusInternalServerError) } }) // Add UI for trending history mux.HandleFunc("/trending/history", func(w http.ResponseWriter, r *http.Request) { w.Header().Set("content-type", "text/html") fmt.Fprintf(w, ` LUMINA Relay - Trending History

Trending History

Archive of trending posts calculations

Loading trending history data...
`) }) fmt.Println("running on :3334") http.ListenAndServe(":3334", relay) }