Files
zurg/internal/handlers/home.go
Ben Adrian Sarmiento 962845fb81 Multi-token support
2024-06-28 04:47:43 +02:00

496 lines
14 KiB
Go

package handlers
import (
"encoding/json"
"fmt"
"net/http"
"os"
"runtime"
"sort"
"strings"
"github.com/debridmediamanager/zurg/internal/config"
"github.com/debridmediamanager/zurg/internal/version"
"github.com/debridmediamanager/zurg/pkg/realdebrid"
)
type SponsorResponse struct {
Patreon string `json:"patreon"`
Github string `json:"github"`
Paypal string `json:"paypal"`
}
type RootResponse struct {
Version string `json:"version"`
BuiltAt string `json:"built_at"`
GitCommit string `json:"git_commit"`
Html string `json:"html"`
Dav string `json:"dav"`
Infuse string `json:"infuse"`
Logs string `json:"logs"`
UserInfo *realdebrid.User `json:"user_info"`
APITrafficMB uint64 `json:"traffic_from_api"`
RequestedMB uint64 `json:"requested_mb"`
ServedMB uint64 `json:"served_mb"`
LibrarySize int `json:"library_size"` // Number of torrents in the library
TorrentsToRepair string `json:"repair_queue"` // List of torrents in the repair queue
MemAlloc uint64 `json:"mem_alloc"` // Memory allocation in MB
TotalAlloc uint64 `json:"total_alloc"` // Total memory allocated in MB
Sys uint64 `json:"sys"` // System memory in MB
NumGC uint32 `json:"num_gc"` // Number of completed GC cycles
PID int `json:"pid"` // Process ID
Sponsor SponsorResponse `json:"sponsor_zurg"` // Sponsorship links
Config config.ZurgConfig `json:"config"`
Token string `json:"token"`
IDsToDelete []string `json:"ids_to_delete"`
Hosts []string `json:"hosts"`
}
func (zr *Handlers) generateResponse(resp http.ResponseWriter, req *http.Request) (*RootResponse, error) {
userInfo, err := zr.api.GetUserInformation()
if err != nil {
http.Error(resp, err.Error(), http.StatusInternalServerError)
return nil, err
}
var mem runtime.MemStats
runtime.ReadMemStats(&mem)
allTorrents, _ := zr.torMgr.DirectoryMap.Get(config.ALL_TORRENTS)
repairQueueStr := ""
if zr.torMgr.RepairQueue == nil {
repairQueueStr = "uninitialized"
} else if zr.torMgr.RepairQueue.IsEmpty() {
repairQueueStr = "empty"
} else {
var repairList []string
for _, torrent := range zr.torMgr.RepairQueue.ToSlice() {
if torrent != nil {
repairList = append(repairList, zr.torMgr.GetKey(torrent))
}
}
sort.Strings(repairList)
repairQueueStr = strings.Join(repairList, ", ")
}
sortedIDs := zr.torMgr.OnceDoneBin.ToSlice()
sort.Strings(sortedIDs)
trafficDetails, err := zr.api.GetTrafficDetails()
if err != nil {
http.Error(resp, err.Error(), http.StatusInternalServerError)
return nil, err
}
var trafficFromAPI int64
trafficFromAPI = 0
if _, ok := trafficDetails["real-debrid.com"]; ok {
trafficFromAPI = trafficDetails["real-debrid.com"]
}
userInfo.Premium = userInfo.Premium / 86400
userInfo.Expiration = strings.Replace(userInfo.Expiration, "Z", "+01:00", 1)
token := zr.cfg.GetToken()
return &RootResponse{
Version: version.GetVersion(),
BuiltAt: version.GetBuiltAt(),
GitCommit: version.GetGitCommit(),
Html: fmt.Sprintf("//%s/http/", req.Host),
Dav: fmt.Sprintf("//%s/dav/", req.Host),
Infuse: fmt.Sprintf("//%s/infuse/", req.Host),
Logs: fmt.Sprintf("//%s/logs/", req.Host),
UserInfo: userInfo,
APITrafficMB: bToMb(uint64(trafficFromAPI)),
RequestedMB: bToMb(zr.downloader.RequestedBytes.Load()),
ServedMB: bToMb(zr.downloader.TotalBytes.Load()),
LibrarySize: allTorrents.Count(),
TorrentsToRepair: repairQueueStr,
MemAlloc: bToMb(mem.Alloc),
TotalAlloc: bToMb(mem.TotalAlloc),
Sys: bToMb(mem.Sys),
NumGC: mem.NumGC,
PID: os.Getpid(),
Sponsor: SponsorResponse{
Patreon: "https://www.patreon.com/debridmediamanager",
Github: "https://github.com/sponsors/debridmediamanager",
Paypal: "https://paypal.me/yowmamasita",
},
Config: zr.cfg.GetConfig(),
Token: strings.Replace(token, token[len(token)-48:], "*****", 1),
IDsToDelete: sortedIDs,
Hosts: zr.hosts,
}, nil
}
func (zr *Handlers) handleHomeJson(resp http.ResponseWriter, req *http.Request) {
response, err := zr.generateResponse(resp, req)
if err != nil {
return
}
// Send the response as JSON by marshalling it
jsonResponse, err := json.Marshal(response)
if err != nil {
http.Error(resp, err.Error(), http.StatusInternalServerError)
return
}
// write the response
resp.Header().Set("Content-Type", "application/json")
resp.Write(jsonResponse)
}
func (zr *Handlers) handleHome(resp http.ResponseWriter, req *http.Request) {
response, err := zr.generateResponse(resp, req)
if err != nil {
return
}
out := fmt.Sprintf(`<table border="1px">
<tr>
<th colspan="3">zurg</th>
</tr>
<tr>
<td>Version</td>
<td colspan="2">%s</td>
</tr>
<tr>
<td>Built At</td>
<td colspan="2">%s</td>
</tr>
<tr>
<td>Git Commit</td>
<td colspan="2">%s</td>
</tr>
<tr>
<td>HTML</td>
<td colspan="2"><a href="%s">%s</a></td>
</tr>
<tr>
<td>DAV</td>
<td colspan="2"><a href="%s">%s</a></td>
</tr>
<tr>
<td>Infuse</td>
<td colspan="2"><a href="%s">%s</a></td>
</tr>
<tr>
<td>Logs</td>
<td colspan="2"><a href="%s">%s</a></td>
</tr>`,
response.Version,
response.BuiltAt,
response.GitCommit,
response.Html,
response.Html,
response.Dav,
response.Dav,
response.Infuse,
response.Infuse,
response.Logs,
response.Logs,
)
denominator := response.RequestedMB
if denominator == 0 {
denominator = 1
}
efficiency := response.ServedMB * 100 / denominator
if zr.trafficOnStartup.Load() > response.APITrafficMB {
// it cannot be bigger than traffic logged
// so it must be a reset back to 0
zr.trafficOnStartup.Store(response.APITrafficMB * 1024 * 1024)
}
out += fmt.Sprintf(`
<tr>
<td>Library Size</td>
<td colspan="2">%d items</td>
</tr>
<tr>
<td>Memory Allocation</td>
<td colspan="2">%d MB</td>
</tr>
<tr>
<td>Total Memory Allocated</td>
<td colspan="2">%d MB</td>
</tr>
<tr>
<td>System Memory</td>
<td colspan="2">%d MB</td>
</tr>
<tr>
<td>Number of GC Cycles</td>
<td colspan="2">%d</td>
</tr>
<tr>
<td>Process ID</td>
<td colspan="2">%d</td>
</tr>
<tr>
<td>Traffic Logged</td>
<td colspan="2">%d MB (%d MB added)</td>
</tr>
<tr>
<td>Traffic Requested</td>
<td colspan="2">%d MB</td>
</tr>
<tr>
<td>Traffic Served</td>
<td colspan="2">%d MB</td>
</tr>
<tr>
<td>Traffic Efficiency</td>
<td colspan="2">%d%% (wasted %d MB)</td>
</tr>`,
response.LibrarySize,
response.MemAlloc,
response.TotalAlloc,
response.Sys,
response.NumGC,
response.PID,
response.APITrafficMB,
response.APITrafficMB-bToMb(zr.trafficOnStartup.Load()),
response.RequestedMB,
response.ServedMB,
efficiency,
response.RequestedMB-response.ServedMB,
)
out += fmt.Sprintf(`
<tr>
<td rowspan="3">Sponsor Zurg</td>
<td>Patreon</td>
<td><a href="%s">%s</a></td>
</tr>
<tr>
<td>Github</td>
<td><a href="%s">%s</a></td>
</tr>
<tr>
<td>Paypal</td>
<td><a href="%s">%s</a></td>
</tr>`,
response.Sponsor.Patreon,
response.Sponsor.Patreon,
response.Sponsor.Github,
response.Sponsor.Github,
response.Sponsor.Paypal,
response.Sponsor.Paypal,
)
out += fmt.Sprintf(`
<tr>
<td rowspan="6">User Info</td>
<td>Username</td>
<td>%s</td>
</tr>
<tr>
<td>Points</td>
<td>%d</td>
</tr>
<tr>
<td>Locale</td>
<td>%s</td>
</tr>
<tr>
<td>Type</td>
<td>%s</td>
</tr>
<tr>
<td>Premium</td>
<td>%d days</td>
</tr>
<tr>
<td>Expiration</td>
<td>%s</td>
</tr>`,
response.UserInfo.Username,
response.UserInfo.Points,
response.UserInfo.Locale,
response.UserInfo.Type,
response.UserInfo.Premium,
response.UserInfo.Expiration,
)
out += fmt.Sprintf(`
<tr>
<td rowspan="22">Config</td>
<td>Version</td>
<td>%s</td>
</tr>
<tr>
<td>Token</td>
<td>%s</td>
</tr>
<tr>
<td>Host</td>
<td>%s</td>
</tr>
<tr>
<td>Port</td>
<td>%s</td>
</tr>
<tr>
<td>Workers</td>
<td>%d running / %d free / %d total</td>
</tr>
<tr>
<td>Refresh Every...</td>
<td>%d secs</td>
</tr>
<tr>
<td>Retain RD Torrent Name</td>
<td>%t</td>
</tr>
<tr>
<td>Retain Folder Name Extension</td>
<td>%t</td>
</tr>
<tr>
<td>Can Repair</td>
<td>%t</td>
</tr>
<tr>
<td>Action to take on RAR'ed torrents</td>
<td>%s</td>
</tr>
<tr>
<td>Repair Every...</td>
<td>%d mins</td>
</tr>
<tr>
<td>Refresh Download Mount Every...</td>
<td>%d mins</td>
</tr>
<tr>
<td>Dump Torrents Every...</td>
<td>%d mins</td>
</tr>
<tr>
<td>API Timeout</td>
<td>%d secs</td>
</tr>
<tr>
<td>Download Timeout</td>
<td>%d secs</td>
</tr>
<tr>
<td>Retries Until Failed</td>
<td>%d</td>
</tr>
<tr>
<td>Auto-Analyze New Torrents</td>
<td>%t</td>
</tr>
<tr>
<td>Cache Network Test Results</td>
<td>%t</td>
</tr>
<tr>
<td>Additional Playable Extensions</td>
<td>%s</td>
</tr>
<tr>
<td>Serve From Rclone</td>
<td>%t</td>
</tr>
<tr>
<td>Force IPv6</td>
<td>%t</td>
</tr>
<tr>
<td>On Library Update</td>
<td>%v</td>
</tr>`,
response.Config.Version,
response.Token,
response.Config.GetHost(),
response.Config.GetPort(),
zr.workerPool.Running(),
zr.workerPool.Free(),
zr.workerPool.Cap(),
response.Config.GetRefreshEverySecs(),
response.Config.EnableRetainRDTorrentName(),
response.Config.EnableRetainFolderNameExtension(),
response.Config.EnableRepair(),
response.Config.GetRarAction(),
response.Config.GetRepairEveryMins(),
response.Config.GetDownloadsEveryMins(),
response.Config.GetDumpTorrentsEveryMins(),
response.Config.GetApiTimeoutSecs(),
response.Config.GetDownloadTimeoutSecs(),
response.Config.GetRetriesUntilFailed(),
response.Config.ShouldAutoAnalyzeNewTorrents(),
response.Config.ShouldCacheNetworkTestResults(),
response.Config.GetPlayableExtensions(),
response.Config.ShouldServeFromRclone(),
response.Config.ShouldForceIPv6(),
response.Config.GetOnLibraryUpdate(),
)
out += fmt.Sprintf(`
<tr>
<td>IDs to be deleted</td>
<td colspan="2">%v</td>
</tr>
<tr>
<td>Torrents to be repaired</td>
<td colspan="2">%s</td>
</tr>
<tr>
<td>Hosts</td>
<td colspan="2">%v</td>
</tr>
<tr>
<td>Utilities</td>
<td colspan="2">
<form method="post" action="/torrents/dump">
<input type="submit" value="Dump torrents" /> Copy all your zurgtorrent files to your dump folder
</form>
<form method="post" action="/torrents/analyze">
<input type="submit" value="Analyze torrents" /> Required to use media_info_* filters
</form>
<form method="post" action="/torrents/repair">
<input type="submit" value="Repair torrents" /> Trigger repair of all torrents
</form>
<form method="post" action="/torrents/reset-repair-state">
<input type="submit" value="Reset repair state" /> Reset repair state of all torrents so they can be repaired again
</form>
</td>
</tr>
<tr>
<td>Debug</td>
<td colspan="2">
<form method="get" action="/logs/upload">
<input type="submit" value="Upload logs" /> Share the link to get help
</form>
<form method="post" action="/reboot-worker">
<input type="submit" value="Reboot worker pool" />
</form>
<form method="post" action="/reboot-refresh">
<input type="submit" value="Reboot refresh worker" />
</form>
<form method="post" action="/reboot-repair">
<input type="submit" value="Reboot repair worker" />
</form>
<form method="post" action="/downloads/remount">
<input type="submit" value="Remount downloads" />
</form>
</td>
</tr>
</table>`,
response.IDsToDelete,
response.TorrentsToRepair,
response.Hosts,
)
fmt.Fprint(resp, out)
}
func bToMb(b uint64) uint64 {
return b / 1024 / 1024
}