2024-05-03 17:11:56 +00:00
|
|
|
package handler
|
|
|
|
|
|
|
|
import (
|
2024-10-29 13:41:22 +00:00
|
|
|
"fmt"
|
2024-05-04 10:24:47 +00:00
|
|
|
"strconv"
|
2024-07-06 18:28:44 +00:00
|
|
|
|
2024-10-29 13:41:22 +00:00
|
|
|
"github.com/a-h/templ"
|
|
|
|
"github.com/ditatompel/xmr-remote-nodes/internal/handler/views"
|
2024-07-06 18:28:44 +00:00
|
|
|
"github.com/ditatompel/xmr-remote-nodes/internal/monero"
|
2024-05-03 17:11:56 +00:00
|
|
|
|
|
|
|
"github.com/gofiber/fiber/v2"
|
2024-10-29 13:41:22 +00:00
|
|
|
"github.com/gofiber/fiber/v2/middleware/adaptor"
|
2024-05-03 17:11:56 +00:00
|
|
|
)
|
|
|
|
|
2024-10-29 13:41:22 +00:00
|
|
|
// Render Home Page
|
2024-10-29 14:30:04 +00:00
|
|
|
func (s *fiberServer) homeHandler(c *fiber.Ctx) error {
|
2024-10-29 13:41:22 +00:00
|
|
|
p := views.Meta{
|
|
|
|
Title: "Monero Remote Node",
|
|
|
|
Description: "A website that helps you monitor your favourite Monero remote nodes, but YOU BETTER RUN AND USE YOUR OWN NODE.",
|
|
|
|
Keywords: "monero,monero,xmr,monero node,xmrnode,cryptocurrency,monero remote node,monero testnet,monero stagenet",
|
|
|
|
Robots: "INDEX,FOLLOW",
|
|
|
|
Permalink: "https://xmr.ditatompel.com",
|
|
|
|
Identifier: "/",
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Set("Link", fmt.Sprintf(`<%s>; rel="canonical"`, p.Permalink))
|
|
|
|
home := views.BaseLayout(p, views.Home())
|
|
|
|
handler := adaptor.HTTPHandler(templ.Handler(home))
|
|
|
|
|
|
|
|
return handler(c)
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:17:51 +00:00
|
|
|
// Returns a single node information based on `id` query param
|
|
|
|
func Node(c *fiber.Ctx) error {
|
2024-05-06 18:08:01 +00:00
|
|
|
nodeId, err := c.ParamsInt("id", 0)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(fiber.StatusUnprocessableEntity).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if nodeId == 0 {
|
|
|
|
return c.Status(fiber.StatusUnprocessableEntity).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": "Invalid node id",
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-06 18:08:01 +00:00
|
|
|
node, err := moneroRepo.Node(nodeId)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": node,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:17:51 +00:00
|
|
|
// Returns a list of nodes
|
|
|
|
func Nodes(c *fiber.Ctx) error {
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-26 22:13:03 +00:00
|
|
|
query := monero.QueryNodes{
|
2024-05-04 11:52:47 +00:00
|
|
|
RowsPerPage: c.QueryInt("limit", 10),
|
|
|
|
Page: c.QueryInt("page", 1),
|
|
|
|
SortBy: c.Query("sort_by", "id"),
|
|
|
|
SortDirection: c.Query("sort_direction", "desc"),
|
|
|
|
Host: c.Query("host"),
|
2024-05-22 15:45:38 +00:00
|
|
|
Nettype: c.Query("nettype", "any"),
|
2024-05-06 07:33:13 +00:00
|
|
|
Protocol: c.Query("protocol", "any"),
|
|
|
|
CC: c.Query("cc", "any"),
|
|
|
|
Status: c.QueryInt("status", -1),
|
2024-05-22 15:45:38 +00:00
|
|
|
CORS: c.QueryInt("cors", -1),
|
2024-05-04 11:52:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nodes, err := moneroRepo.Nodes(query)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": nodes,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Returns probe logs reported by nodes
|
|
|
|
//
|
|
|
|
// The embadded web UI use `node_id` query param to filter logs
|
2024-05-06 10:19:17 +00:00
|
|
|
func ProbeLogs(c *fiber.Ctx) error {
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-26 23:38:11 +00:00
|
|
|
query := monero.QueryLogs{
|
2024-05-06 10:19:17 +00:00
|
|
|
RowsPerPage: c.QueryInt("limit", 10),
|
|
|
|
Page: c.QueryInt("page", 1),
|
|
|
|
SortBy: c.Query("sort_by", "id"),
|
|
|
|
SortDirection: c.Query("sort_direction", "desc"),
|
2024-05-22 15:45:38 +00:00
|
|
|
NodeID: c.QueryInt("node_id", 0),
|
2024-05-07 14:31:40 +00:00
|
|
|
Status: c.QueryInt("status", -1),
|
|
|
|
FailedReason: c.Query("failed_reason"),
|
2024-05-06 10:19:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
logs, err := moneroRepo.Logs(query)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": logs,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Handles `POST /nodes` request to add a new node
|
2024-05-04 10:24:47 +00:00
|
|
|
func AddNode(c *fiber.Ctx) error {
|
|
|
|
formPort := c.FormValue("port")
|
|
|
|
port, err := strconv.Atoi(formPort)
|
|
|
|
if err != nil {
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": "Invalid port number",
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
protocol := c.FormValue("protocol")
|
|
|
|
hostname := c.FormValue("hostname")
|
|
|
|
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-04 10:24:47 +00:00
|
|
|
if err := moneroRepo.Add(protocol, hostname, uint(port)); err != nil {
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Query Ok",
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Returns majority network fees
|
2024-05-31 06:30:56 +00:00
|
|
|
func NetFees(c *fiber.Ctx) error {
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-06 06:19:48 +00:00
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
2024-05-31 06:30:56 +00:00
|
|
|
"data": moneroRepo.NetFees(),
|
2024-05-06 06:19:48 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Returns list of countries (count by nodes)
|
2024-05-06 06:35:15 +00:00
|
|
|
func Countries(c *fiber.Ctx) error {
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-05-06 06:35:15 +00:00
|
|
|
countries, err := moneroRepo.Countries()
|
|
|
|
if err != nil {
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": countries,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Returns node to be probed by the client (prober)
|
|
|
|
//
|
|
|
|
// This handler should protected by `CheckProber` middleware.
|
2024-05-04 12:27:21 +00:00
|
|
|
func GiveJob(c *fiber.Ctx) error {
|
|
|
|
acceptTor := c.QueryInt("accept_tor", 0)
|
2024-09-05 17:08:59 +00:00
|
|
|
acceptIPv6 := c.QueryInt("accept_ipv6", 0)
|
2024-05-04 12:27:21 +00:00
|
|
|
|
2024-05-26 23:15:40 +00:00
|
|
|
moneroRepo := monero.New()
|
2024-09-05 17:08:59 +00:00
|
|
|
node, err := moneroRepo.GiveJob(acceptTor, acceptIPv6)
|
2024-05-04 12:27:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": node,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
// Handles probe report submission by the prober
|
|
|
|
//
|
|
|
|
// This handler should protected by `CheckProber` middleware.
|
2024-05-04 18:42:47 +00:00
|
|
|
func ProcessJob(c *fiber.Ctx) error {
|
2024-06-03 07:20:10 +00:00
|
|
|
var report monero.ProbeReport
|
2024-05-04 18:42:47 +00:00
|
|
|
|
|
|
|
if err := c.BodyParser(&report); err != nil {
|
|
|
|
return c.Status(fiber.StatusUnprocessableEntity).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-06-03 07:20:10 +00:00
|
|
|
moneroRepo := monero.New()
|
|
|
|
|
2024-05-04 18:42:47 +00:00
|
|
|
if err := moneroRepo.ProcessJob(report, c.Locals("prober_id").(int64)); err != nil {
|
|
|
|
return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{
|
|
|
|
"status": "error",
|
|
|
|
"message": err.Error(),
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Success",
|
|
|
|
"data": nil,
|
|
|
|
})
|
|
|
|
}
|