package main import ( "encoding/json" "fmt" "net/http" "os" "time" "git.highperfocused.tech/highperfocused/lumina-relay/relay/trending" "github.com/fiatjaf/eventstore/postgresql" "github.com/fiatjaf/khatru" "github.com/fiatjaf/khatru/policies" ) func getEnv(key, fallback string) string { if value, ok := os.LookupEnv(key); ok { return value } return fallback } 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) } 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") // Query the total number of events count := 0 row := db.DB.QueryRow("SELECT COUNT(*) FROM event") if err := row.Scan(&count); err != nil { fmt.Printf("Error counting events: %v\n", err) } // 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

`, 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") // 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) totalCount := 0 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 totalCount += 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) } }) fmt.Println("running on :3334") http.ListenAndServe(":3334", relay) }