Compare commits

...

4 Commits

Author SHA1 Message Date
Pijus Kamandulis
effe887b3b Implement control panel 2025-07-04 21:23:21 +03:00
Pijus Kamandulis
ef247fc843 Added auth 2025-07-04 21:05:28 +03:00
Pijus Kamandulis
b3c89a01d0 Show stats by worker in daily stats 2025-07-04 19:04:24 +03:00
Pijus Kamandulis
11cc168b3a Cache current day stats for 5m 2025-07-04 18:42:35 +03:00
18 changed files with 431 additions and 88 deletions

View File

@ -6,12 +6,14 @@ type Config struct {
Port int `json:"port"` Port int `json:"port"`
LogPath string `json:"logPath"` LogPath string `json:"logPath"`
DatabasePath string `json:"databasePath"` DatabasePath string `json:"databasePath"`
AdminPassword string `json:"adminPassword"`
} }
func ParseFlags() Config { func ParseFlags() Config {
port := flag.Int("Port", 8080, "Listen port") port := flag.Int("Port", 8080, "Listen port")
logPath := flag.String("LogPath", "logs", "Path to log files") logPath := flag.String("LogPath", "logs", "Path to log files")
databasePath := flag.String("DatabasePath", "badgerdb", "Path to the database directory") databasePath := flag.String("DatabasePath", "badgerdb", "Path to the database directory")
adminPassword := flag.String("AdminPassword", "", "Admin password for the web interface, disabled if empty")
flag.Parse() flag.Parse()
@ -19,5 +21,6 @@ func ParseFlags() Config {
Port: *port, Port: *port,
LogPath: *logPath, LogPath: *logPath,
DatabasePath: *databasePath, DatabasePath: *databasePath,
AdminPassword: *adminPassword,
} }
} }

View File

@ -258,10 +258,9 @@ func GetDailyStats(db *clover.DB, date time.Time) (*models.DailyStats, error) {
dateStr := date.Format(time.DateOnly) dateStr := date.Format(time.DateOnly)
// Check if stats already exist // Check if stats already exist
isToday := dateStr == time.Now().UTC().Format(time.DateOnly)
existingDoc, err := db.FindFirst(c.NewQuery(DailyStatsCollectionName). existingDoc, err := db.FindFirst(c.NewQuery(DailyStatsCollectionName).
Where(c.Field("Date").Eq(dateStr))) Where(c.Field("Date").Eq(dateStr)))
if !isToday && err == nil && existingDoc != nil { if err == nil && existingDoc != nil {
var stats models.DailyStats var stats models.DailyStats
if err := existingDoc.Unmarshal(&stats); err != nil { if err := existingDoc.Unmarshal(&stats); err != nil {
return nil, fmt.Errorf("failed to unmarshal daily stats: %v", err) return nil, fmt.Errorf("failed to unmarshal daily stats: %v", err)
@ -296,6 +295,9 @@ func GetDailyStats(db *clover.DB, date time.Time) (*models.DailyStats, error) {
} }
for workerName, workerShares := range sharesByWorker { for workerName, workerShares := range sharesByWorker {
workerHashrate := helpers.CalculateAverageHashrate(workerShares) workerHashrate := helpers.CalculateAverageHashrate(workerShares)
sort.Slice(workerShares, func(i, j int) bool {
return workerShares[i].SDiff > workerShares[j].SDiff
})
workerTopShare := workerShares[0] // Already sorted by SDiff workerTopShare := workerShares[0] // Already sorted by SDiff
stats.Workers[workerName] = models.WorkerDailyStats{ stats.Workers[workerName] = models.WorkerDailyStats{
@ -307,6 +309,12 @@ func GetDailyStats(db *clover.DB, date time.Time) (*models.DailyStats, error) {
// Insert or update the daily stats in the collection // Insert or update the daily stats in the collection
doc := document.NewDocumentOf(stats) doc := document.NewDocumentOf(stats)
isToday := dateStr == time.Now().UTC().Format(time.DateOnly)
if isToday {
doc.SetExpiresAt(time.Now().Add(5 * time.Minute))
}
if _, err := db.InsertOne(DailyStatsCollectionName, doc); err != nil { if _, err := db.InsertOne(DailyStatsCollectionName, doc); err != nil {
return nil, fmt.Errorf("failed to insert daily stats: %v", err) return nil, fmt.Errorf("failed to insert daily stats: %v", err)
} }

6
go.mod
View File

@ -2,14 +2,16 @@ module pool-stats
go 1.24.3 go 1.24.3
require github.com/ostafen/clover/v2 v2.0.0-alpha.3.0.20250212110647-35f6fd38bde2 require (
github.com/gofrs/uuid/v5 v5.3.1
github.com/ostafen/clover/v2 v2.0.0-alpha.3.0.20250212110647-35f6fd38bde2
)
require ( require (
github.com/cespare/xxhash/v2 v2.3.0 // indirect github.com/cespare/xxhash/v2 v2.3.0 // indirect
github.com/dgraph-io/badger/v4 v4.5.1 // indirect github.com/dgraph-io/badger/v4 v4.5.1 // indirect
github.com/dgraph-io/ristretto/v2 v2.1.0 // indirect github.com/dgraph-io/ristretto/v2 v2.1.0 // indirect
github.com/dustin/go-humanize v1.0.1 // indirect github.com/dustin/go-humanize v1.0.1 // indirect
github.com/gofrs/uuid/v5 v5.3.1 // indirect
github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8 // indirect github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8 // indirect
github.com/google/flatbuffers v25.2.10+incompatible // indirect github.com/google/flatbuffers v25.2.10+incompatible // indirect
github.com/google/orderedcode v0.0.1 // indirect github.com/google/orderedcode v0.0.1 // indirect

View File

@ -32,6 +32,7 @@ func (job *RecalculateCurrentDayStatsJob) recalculateCurrentDayStats() {
today := time.Now().Truncate(24 * time.Hour) today := time.Now().Truncate(24 * time.Hour)
yesterday := today.Add(-24 * time.Hour) yesterday := today.Add(-24 * time.Hour)
database.DeleteDailyStatsForDay(job.DB, today)
database.GetDailyStats(job.DB, today) database.GetDailyStats(job.DB, today)
// Need to keep yesterday's stats cache updated // Need to keep yesterday's stats cache updated

View File

@ -34,7 +34,7 @@ func main() {
currentDayStatsRecalcJob := jobs.NewRecalculateCurrentDayStatsJob(db) currentDayStatsRecalcJob := jobs.NewRecalculateCurrentDayStatsJob(db)
go currentDayStatsRecalcJob.Run() go currentDayStatsRecalcJob.Run()
webServer := web.NewWebServer(db, config.Port) webServer := web.NewWebServer(db, config.Port, config.AdminPassword)
if err := webServer.Start(); err != nil { if err := webServer.Start(); err != nil {
log.Fatalf("Failed to start web server: %v", err) log.Fatalf("Failed to start web server: %v", err)
} }

24
templates/cp.html Normal file
View File

@ -0,0 +1,24 @@
{{ define "title" }}Control Panel{{ end }} {{ define "header" }}⚙️ Control
Panel{{ end }} {{ define "content" }}
<div>
{{ if .Message }}
<strong>Info:</strong> {{ .Message }} {{ end }}
<table>
<thead>
<tr>
<th>Action</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td><a href="/cp?action=ClearDailyStats">Clear Daily Stats</a></td>
<td>Clear Daily Stats cache</td>
</tr>
</tbody>
</table>
</div>
{{ end }} {{ template "layout" . }}

View File

@ -1,8 +1,25 @@
{{ define "title" }}Daily Stats{{ end }} {{ define "header" }}📊 Pool Daily {{ define "title" }}Daily Stats{{ end }} {{ define "header" }}📊 Pool Daily
Stats{{ end }} {{ define "content" }} Stats{{ end }} {{ define "content" }}
<style>
.worker-rows {
display: none;
}
.expand-btn {
cursor: pointer;
}
.worker-table {
width: 100%;
}
.worker-table th,
.worker-table td {
font-size: 0.9em;
}
</style>
<table> <table>
<thead> <thead>
<tr> <tr>
<th></th>
<th>Date (UTC)</th> <th>Date (UTC)</th>
<th>Share Count</th> <th>Share Count</th>
<th>Top Share Diff</th> <th>Top Share Diff</th>
@ -10,21 +27,57 @@ Stats{{ end }} {{ define "content" }}
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{{ range .DailyStats }} {{ range $i, $ds := .DailyStats }}
<tr> <tr>
<td>{{ .Date }}</td> <td>
<td>{{ .ShareCount }}</td> <span class="expand-btn" onclick="toggleWorkers({{ $i }})"></span>
<td>{{ humanDiff .TopShare.SDiff }}</td> </td>
<td>{{ formatHashrate .PoolHashrate }}</td> <td>{{ $ds.Date }}</td>
<td>{{ $ds.ShareCount }}</td>
<td>{{ humanDiff $ds.TopShare.SDiff }}</td>
<td>{{ formatHashrate $ds.PoolHashrate }}</td>
</tr>
<tr class="worker-rows" id="workers-{{ $i }}">
<td colspan="5">
<table class="worker-table">
<thead>
<tr>
<th>Shares</th>
<th>Top Share Diff</th>
<th>Hashrate</th>
</tr>
</thead>
<tbody>
{{ range $name, $w := $ds.Workers }}
<tr>
<td>{{ $w.Shares }}</td>
<td>{{ humanDiff $w.TopShare.SDiff }}</td>
<td>{{ formatHashrate $w.Hashrate }}</td>
</tr>
{{ end }}
</tbody>
</table>
</td>
</tr> </tr>
{{ else }} {{ else }}
<tr> <tr>
<td colspan="4">No stats found for this date range.</td> <td colspan="5">No stats found for this date range.</td>
</tr> </tr>
{{ end }} {{ end }}
</tbody> </tbody>
</table> </table>
<script>
function toggleWorkers(index) {
const row = document.getElementById(`workers-${index}`);
if (row.style.display === "none" || row.style.display === "") {
row.style.display = "table-row";
} else {
row.style.display = "none";
}
}
</script>
<div> <div>
{{ if .PrevPageAvailable }} {{ if .PrevPageAvailable }}
<a class="page-link" href="?start={{ .PrevPageStart }}&end={{ .PrevPageEnd }}" <a class="page-link" href="?start={{ .PrevPageStart }}&end={{ .PrevPageEnd }}"

View File

@ -6,12 +6,50 @@
<title>{{ template "title" . }}</title> <title>{{ template "title" . }}</title>
<style> <style>
body { body {
font-family: sans-serif; font-family: "Courier New", Courier, monospace;
background: #111; background: #111;
color: #eee; color: #eee;
padding: 20px; padding: 0;
margin: 0;
text-align: center; text-align: center;
} }
nav {
background-color: #222;
border-bottom: 1px solid #444;
padding: 5px 0;
margin-bottom: 20px;
display: flex;
justify-content: space-between;
}
nav ul {
margin: 0;
padding: 0;
list-style: none;
display: inline-flex;
gap: 15px;
}
nav li {
display: inline;
}
nav a {
color: #0af;
text-decoration: none;
padding: 4px 8px;
border: 1px solid transparent;
font-weight: bold;
font-size: 14px;
}
nav a:hover {
background-color: #0af;
color: #111;
border-color: #0af;
border-radius: 3px;
}
h1 {
margin-top: 0;
}
table { table {
border-collapse: collapse; border-collapse: collapse;
margin: auto; margin: auto;
@ -46,23 +84,32 @@
color: #0af; color: #0af;
text-decoration: none; text-decoration: none;
} }
li {
display: inline;
margin: 0 10px;
}
</style> </style>
</head> </head>
<body> <body>
<h1>{{ template "header" . }}</h1> <nav>
<ul>
{{ template "content" . }} {{ template "navigation" . }}
</body>
</html>
{{ end }} {{ define "navigation" }}
<ul>
<li><a href="/">Home</a></li> <li><a href="/">Home</a></li>
<li><a href="/shares">View Shares</a></li> <li><a href="/shares">View Shares</a></li>
<li><a href="/top-shares">Top Shares</a></li> <li><a href="/top-shares">Top Shares</a></li>
<li><a href="/daily-stats">Daily Stats</a></li> <li><a href="/daily-stats">Daily Stats</a></li>
</ul> </ul>
{{ if .UserName }}
<ul>
<li>{{ .UserName }}</li>
<li><a href="/logout">Logout</a></li>
</ul>
{{ else }}
<ul>
<li><a href="/login">Login</a></li>
</ul>
{{ end }}
</nav>
<h1>{{ template "header" . }}</h1>
{{ template "content" . }}
</body>
</html>
{{ end }} {{ end }}

61
templates/login.html Normal file
View File

@ -0,0 +1,61 @@
{{ define "title" }}Login{{ end }} {{ define "header" }}🔒 Login{{ end }} {{
define "content" }}
<style>
.login-form {
display: inline-block;
text-align: left;
padding: 20px 25px;
background-color: #1a1a1a;
border: 1px solid #444;
color: #eee;
margin: 0 auto;
}
.login-form label {
display: block;
margin-bottom: 8px;
color: #ccc;
font-weight: bold;
}
.login-form input[type="password"],
.login-form input[type="submit"] {
width: 100%;
padding: 10px;
margin-bottom: 12px;
border: 1px solid #444;
background-color: #222;
color: #eee;
border-radius: 4px;
box-sizing: border-box;
}
.login-form input[type="submit"] {
background-color: #0af;
color: #111;
font-weight: bold;
cursor: pointer;
transition: background-color 0.3s ease;
}
.login-form input[type="submit"]:hover {
background-color: #08c;
}
.error {
color: #ff6b6b;
font-size: 0.9em;
margin-top: -8px;
}
</style>
<form method="POST" action="/login" class="login-form">
<label for="password">Password</label><br />
<input type="password" id="password" name="password" required /><br /><br />
<input type="submit" value="Login" />
{{ if .ErrorMessage }}
<p class="error">{{ .ErrorMessage }}</p>
{{ end }}
</form>
{{ end }} {{ template "layout" . }}

View File

@ -0,0 +1,39 @@
package web
import (
"net/http"
"pool-stats/database"
)
type ControlPanelPageData struct {
PageDataBase
Message string
}
func (ws *WebServer) ControlPanelHandler(w http.ResponseWriter, r *http.Request) {
username := ws.getUser(r)
if username == "" {
http.Error(w, "Unauthorized", http.StatusUnauthorized)
return
}
data := ControlPanelPageData{}
action := r.URL.Query().Get("action")
if action != "" {
data.Message = ws.executeAction(action)
}
ws.renderTemplate(w, r, "templates/cp.html", &data)
}
func (ws *WebServer) executeAction(action string) string {
switch action {
case "ClearDailyStats":
database.ClearDailyStats(ws.db)
return "Daily stats cleared successfully"
default:
return "Unknown action"
}
}

View File

@ -1,7 +1,6 @@
package web package web
import ( import (
"html/template"
"net/http" "net/http"
"pool-stats/constants" "pool-stats/constants"
"pool-stats/database" "pool-stats/database"
@ -10,6 +9,8 @@ import (
) )
type DailyStatsPageData struct { type DailyStatsPageData struct {
PageDataBase
DailyStats []models.DailyStats DailyStats []models.DailyStats
Start string Start string
@ -25,16 +26,10 @@ type DailyStatsPageData struct {
} }
func (ws *WebServer) DailyStatsHandler(w http.ResponseWriter, r *http.Request) { func (ws *WebServer) DailyStatsHandler(w http.ResponseWriter, r *http.Request) {
tmpl, err := template.Must(ws.templates.Clone()).ParseFiles("templates/daily_stats.html")
if err != nil {
http.Error(w, "Failed to parse template", http.StatusInternalServerError)
println("Error parsing template:", err.Error())
return
}
startParam := r.URL.Query().Get("start") startParam := r.URL.Query().Get("start")
endParam := r.URL.Query().Get("end") endParam := r.URL.Query().Get("end")
var startTime, endTime time.Time var startTime, endTime time.Time
var err error
if startParam == "" || endParam == "" { if startParam == "" || endParam == "" {
endTime = time.Now().Truncate(24 * time.Hour) endTime = time.Now().Truncate(24 * time.Hour)
@ -91,9 +86,6 @@ func (ws *WebServer) DailyStatsHandler(w http.ResponseWriter, r *http.Request) {
PrevPageStart: prevPageStart.Format(time.DateOnly), PrevPageStart: prevPageStart.Format(time.DateOnly),
PrevPageEnd: prevPageEnd.Format(time.DateOnly), PrevPageEnd: prevPageEnd.Format(time.DateOnly),
} }
if err := tmpl.ExecuteTemplate(w, "daily_stats.html", data); err != nil {
http.Error(w, "Failed to render template", http.StatusInternalServerError) ws.renderTemplate(w, r, "templates/daily_stats.html", &data)
println("Error rendering template:", err.Error())
return
}
} }

View File

@ -1,7 +1,6 @@
package web package web
import ( import (
"html/template"
"net/http" "net/http"
"pool-stats/database" "pool-stats/database"
@ -9,17 +8,12 @@ import (
) )
type IndexPageData struct { type IndexPageData struct {
PageDataBase
Stats []models.TimeWindowHighShare Stats []models.TimeWindowHighShare
} }
func (ws *WebServer) IndexHandler(w http.ResponseWriter, r *http.Request) { func (ws *WebServer) IndexHandler(w http.ResponseWriter, r *http.Request) {
tmpl, err := template.Must(ws.templates.Clone()).ParseFiles("templates/index.html")
if err != nil {
http.Error(w, "Failed to parse template", 500)
println("Error parsing template:", err.Error())
return
}
tws := database.GetTimeWindowHighShares(ws.db) tws := database.GetTimeWindowHighShares(ws.db)
if tws == nil { if tws == nil {
http.Error(w, "Failed to load time window high shares", 500) http.Error(w, "Failed to load time window high shares", 500)
@ -30,9 +24,5 @@ func (ws *WebServer) IndexHandler(w http.ResponseWriter, r *http.Request) {
Stats: tws, Stats: tws,
} }
if err := tmpl.ExecuteTemplate(w, "index.html", indexData); err != nil { ws.renderTemplate(w, r, "templates/index.html", &indexData)
http.Error(w, "Failed to render template", 500)
println("Error rendering template:", err.Error())
return
}
} }

42
web/loginHandler.go Normal file
View File

@ -0,0 +1,42 @@
package web
import (
"net/http"
)
type LoginPageData struct {
PageDataBase
ErrorMessage string
}
func (ws *WebServer) LoginHandler(w http.ResponseWriter, r *http.Request) {
loginData := LoginPageData{}
if r.Method == http.MethodPost {
if err := r.ParseForm(); err != nil {
loginData.ErrorMessage = "Failed to parse form data"
} else {
password := r.FormValue("password")
if password != ws.adminPassword || ws.adminPassword == "" {
loginData.ErrorMessage = "Invalid password"
} else {
sessionID := generateSessionID()
if sessionID == "" {
loginData.ErrorMessage = "Failed to generate session ID"
}
ws.sessions[sessionID] = "admin"
http.SetCookie(w, &http.Cookie{
Name: "session_id",
Value: sessionID,
Path: "/",
HttpOnly: true,
})
http.Redirect(w, r, "/", http.StatusSeeOther)
}
}
}
ws.renderTemplate(w, r, "templates/login.html", &loginData)
}

25
web/logoutHandler.go Normal file
View File

@ -0,0 +1,25 @@
package web
import (
"net/http"
"time"
)
func (ws *WebServer) LogoutHandler(w http.ResponseWriter, r *http.Request) {
cookie, err := r.Cookie("session_id")
if err != nil {
http.Error(w, "No session found", http.StatusUnauthorized)
return
}
delete(ws.sessions, cookie.Value)
http.SetCookie(w, &http.Cookie{
Name: "session_id",
Value: "",
Path: "/",
HttpOnly: true,
Expires: <-time.After(time.Second * 1),
})
http.Redirect(w, r, "/", http.StatusSeeOther)
}

18
web/models.go Normal file
View File

@ -0,0 +1,18 @@
package web
type PageDataBase struct {
UserName string
}
func (pageData PageDataBase) GetUserName() string {
return pageData.UserName
}
func (pageData *PageDataBase) SetUserName(userName string) {
pageData.UserName = userName
}
type IPageDataBase interface {
GetUserName() string
SetUserName(userName string)
}

View File

@ -3,10 +3,12 @@ package web
import ( import (
"html/template" "html/template"
"net/http" "net/http"
"path/filepath"
"pool-stats/helpers" "pool-stats/helpers"
"fmt" "fmt"
"github.com/gofrs/uuid/v5"
"github.com/ostafen/clover/v2" "github.com/ostafen/clover/v2"
) )
@ -14,9 +16,11 @@ type WebServer struct {
db *clover.DB db *clover.DB
port int port int
templates *template.Template templates *template.Template
sessions map[string]string
adminPassword string
} }
func NewWebServer(db *clover.DB, port int) *WebServer { func NewWebServer(db *clover.DB, port int, adminPassword string) *WebServer {
templates := template.New("base").Funcs(template.FuncMap{ templates := template.New("base").Funcs(template.FuncMap{
"add": func(a, b int) int { return a + b }, "add": func(a, b int) int { return a + b },
"sub": func(a, b int) int { return a - b }, "sub": func(a, b int) int { return a - b },
@ -33,11 +37,16 @@ func NewWebServer(db *clover.DB, port int) *WebServer {
db: db, db: db,
port: port, port: port,
templates: templates, templates: templates,
sessions: make(map[string]string),
adminPassword: adminPassword,
} }
} }
func (ws *WebServer) Start() error { func (ws *WebServer) Start() error {
http.HandleFunc("/", ws.IndexHandler) http.HandleFunc("/", ws.IndexHandler)
http.HandleFunc("/cp", ws.ControlPanelHandler)
http.HandleFunc("/login", ws.LoginHandler)
http.HandleFunc("/logout", ws.LogoutHandler)
http.HandleFunc("/shares", ws.SharesHandler) http.HandleFunc("/shares", ws.SharesHandler)
http.HandleFunc("/top-shares", ws.TopSharesHandler) http.HandleFunc("/top-shares", ws.TopSharesHandler)
http.HandleFunc("/daily-stats", ws.DailyStatsHandler) http.HandleFunc("/daily-stats", ws.DailyStatsHandler)
@ -46,3 +55,48 @@ func (ws *WebServer) Start() error {
println("Listening on", address) println("Listening on", address)
return http.ListenAndServe(address, nil) return http.ListenAndServe(address, nil)
} }
func generateSessionID() string {
uuid, err := uuid.NewV4()
if err != nil {
fmt.Println("Error generating session ID:", err)
return ""
}
return uuid.String()
}
func (ws *WebServer) getUser(r *http.Request) string {
cookie, err := r.Cookie("session_id")
if err != nil {
return ""
}
if user, ok := ws.sessions[cookie.Value]; ok {
return user
}
return ""
}
func (ws *WebServer) renderTemplate(
w http.ResponseWriter,
r *http.Request,
templateFile string,
data IPageDataBase) {
data.SetUserName(ws.getUser(r))
tmpl, err := template.Must(ws.templates.Clone()).ParseFiles(templateFile)
if err != nil {
http.Error(w, "Failed to parse template", 500)
println("Error parsing template:", err.Error())
return
}
templateName := filepath.Base(templateFile)
if err := tmpl.ExecuteTemplate(w, templateName, data); err != nil {
http.Error(w, "Failed to render template", 500)
println("Error rendering template:", err.Error())
return
}
}

View File

@ -1,7 +1,6 @@
package web package web
import ( import (
"html/template"
"net/http" "net/http"
"pool-stats/database" "pool-stats/database"
"pool-stats/models" "pool-stats/models"
@ -9,19 +8,14 @@ import (
) )
type SharePageData struct { type SharePageData struct {
PageDataBase
Shares []models.ShareLog Shares []models.ShareLog
Page int Page int
HasMore bool HasMore bool
} }
func (ws *WebServer) SharesHandler(w http.ResponseWriter, r *http.Request) { func (ws *WebServer) SharesHandler(w http.ResponseWriter, r *http.Request) {
tmpl, err := template.Must(ws.templates.Clone()).ParseFiles("templates/shares.html")
if err != nil {
http.Error(w, "Failed to parse template", 500)
println("Error parsing template:", err.Error())
return
}
entriesPerPage := 10 entriesPerPage := 10
page := r.URL.Query().Get("page") page := r.URL.Query().Get("page")
if page == "" { if page == "" {
@ -46,8 +40,6 @@ func (ws *WebServer) SharesHandler(w http.ResponseWriter, r *http.Request) {
Page: offset/entriesPerPage + 1, Page: offset/entriesPerPage + 1,
HasMore: len(shareLogs) == entriesPerPage, HasMore: len(shareLogs) == entriesPerPage,
} }
if err := tmpl.ExecuteTemplate(w, "shares.html", data); err != nil {
http.Error(w, "Failed to render template", 500) ws.renderTemplate(w, r, "templates/shares.html", &data)
return
}
} }

View File

@ -1,24 +1,18 @@
package web package web
import ( import (
"html/template"
"net/http" "net/http"
"pool-stats/database" "pool-stats/database"
"pool-stats/models" "pool-stats/models"
) )
type TopSharesPageData struct { type TopSharesPageData struct {
PageDataBase
Shares []models.ShareLog Shares []models.ShareLog
} }
func (ws *WebServer) TopSharesHandler(w http.ResponseWriter, r *http.Request) { func (ws *WebServer) TopSharesHandler(w http.ResponseWriter, r *http.Request) {
tmpl, err := template.Must(ws.templates.Clone()).ParseFiles("templates/top_shares.html")
if err != nil {
http.Error(w, "Failed to parse template", 500)
println("Error parsing template:", err.Error())
return
}
topShares := database.ListTopShares(ws.db) topShares := database.ListTopShares(ws.db)
if topShares == nil { if topShares == nil {
http.Error(w, "Failed to load top shares", 500) http.Error(w, "Failed to load top shares", 500)
@ -28,8 +22,6 @@ func (ws *WebServer) TopSharesHandler(w http.ResponseWriter, r *http.Request) {
data := TopSharesPageData{ data := TopSharesPageData{
Shares: topShares, Shares: topShares,
} }
if err := tmpl.ExecuteTemplate(w, "top_shares.html", data); err != nil {
http.Error(w, "Failed to render template", 500) ws.renderTemplate(w, r, "templates/top_shares.html", &data)
return
}
} }