Compare commits
4 Commits
edb17e825d
...
effe887b3b
Author | SHA1 | Date | |
---|---|---|---|
|
effe887b3b | ||
|
ef247fc843 | ||
|
b3c89a01d0 | ||
|
11cc168b3a |
@ -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,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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
6
go.mod
@ -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
|
||||||
|
@ -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
|
||||||
|
2
main.go
2
main.go
@ -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
24
templates/cp.html
Normal 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" . }}
|
@ -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 }}"
|
||||||
|
@ -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
61
templates/login.html
Normal 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" . }}
|
39
web/controlPanelHandler.go
Normal file
39
web/controlPanelHandler.go
Normal 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"
|
||||||
|
}
|
||||||
|
}
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -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
42
web/loginHandler.go
Normal 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
25
web/logoutHandler.go
Normal 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
18
web/models.go
Normal 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)
|
||||||
|
}
|
@ -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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user