package handler import ( "fmt" "strconv" "github.com/a-h/templ" "github.com/ditatompel/xmr-remote-nodes/internal/handler/views" "github.com/ditatompel/xmr-remote-nodes/internal/monero" "github.com/ditatompel/xmr-remote-nodes/internal/paging" "github.com/gofiber/fiber/v2" "github.com/gofiber/fiber/v2/middleware/adaptor" ) // Render Home Page func (s *fiberServer) homeHandler(c *fiber.Ctx) error { 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) } // Render Add Node Page func (s *fiberServer) addNodeHandler(c *fiber.Ctx) error { p := views.Meta{ Title: "Add Monero Node", Description: "You can use this page to add known remote node to the system so my bots can monitor it.", Keywords: "monero,monero node,monero public node,monero wallet,list monero node,monero node monitoring", Robots: "INDEX,FOLLOW", Permalink: "https://xmr.ditatompel.com/add-node", Identifier: "/add-node", } c.Set("Link", fmt.Sprintf(`<%s>; rel="canonical"`, p.Permalink)) home := views.BaseLayout(p, views.AddNode()) handler := adaptor.HTTPHandler(templ.Handler(home)) return handler(c) } // Returns a single node information based on `id` query param func Node(c *fiber.Ctx) error { 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, }) } moneroRepo := monero.New() 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, }) } // Render Remote Nodes Page func (s *fiberServer) remoteNodesHandler(c *fiber.Ctx) error { p := views.Meta{ Title: "Public Monero Remote Nodes List", Description: "Although it's possible to use these existing public Monero nodes, you're MUST RUN AND USE YOUR OWN NODE!", Keywords: "monero remote nodes,public monero nodes,monero public nodes,monero wallet,tor monero node,monero cors rpc", Robots: "INDEX,FOLLOW", Permalink: "https://xmr.ditatompel.com/remote-nodes", Identifier: "/remote-nodes", } moneroRepo := monero.New() query := monero.QueryNodes{ Paging: paging.Paging{ Limit: c.QueryInt("limit", 10), // rows per page Page: c.QueryInt("page", 1), SortBy: c.Query("sort_by", "id"), SortDirection: c.Query("sort_direction", "desc"), Refresh: c.QueryInt("refresh", 0), }, Host: c.Query("host"), Nettype: c.Query("nettype", "any"), Protocol: c.Query("protocol", "any"), CC: c.Query("cc", "any"), Status: c.QueryInt("status", -1), CORS: c.Query("cors"), } nodes, err := moneroRepo.Nodes(query) if err != nil { return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ "status": "error", "message": err.Error(), "data": nil, }) } countries, err := moneroRepo.Countries() if err != nil { return c.JSON(fiber.Map{ "status": "error", "message": err.Error(), "data": nil, }) } pagination := paging.NewPagination(query.Page, nodes.TotalPages) // handle request from HTMX if c.Get("HX-Target") == "tbl_nodes" { cmp := views.BlankLayout(views.TableNodes(nodes, countries, query, pagination)) handler := adaptor.HTTPHandler(templ.Handler(cmp)) return handler(c) } c.Set("Link", fmt.Sprintf(`<%s>; rel="canonical"`, p.Permalink)) home := views.BaseLayout(p, views.RemoteNodes(nodes, countries, query, pagination)) handler := adaptor.HTTPHandler(templ.Handler(home)) return handler(c) } // Returns a list of nodes (API) func Nodes(c *fiber.Ctx) error { moneroRepo := monero.New() query := monero.QueryNodes{ Paging: paging.Paging{ Limit: c.QueryInt("limit", 10), // rows per page Page: c.QueryInt("page", 1), SortBy: c.Query("sort_by", "id"), SortDirection: c.Query("sort_direction", "desc"), Refresh: c.QueryInt("refresh", 0), }, Host: c.Query("host"), Nettype: c.Query("nettype", "any"), Protocol: c.Query("protocol", "any"), CC: c.Query("cc", "any"), Status: c.QueryInt("status", -1), CORS: c.Query("cors"), } 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, }) } // Returns probe logs reported by nodes // // The embadded web UI use `node_id` query param to filter logs func ProbeLogs(c *fiber.Ctx) error { moneroRepo := monero.New() query := monero.QueryLogs{ RowsPerPage: c.QueryInt("limit", 10), Page: c.QueryInt("page", 1), SortBy: c.Query("sort_by", "id"), SortDirection: c.Query("sort_direction", "desc"), NodeID: c.QueryInt("node_id", 0), Status: c.QueryInt("status", -1), FailedReason: c.Query("failed_reason"), } 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, }) } // Handles `POST /nodes` request to add a new node 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") moneroRepo := monero.New() 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, }) } // Returns majority network fees func NetFees(c *fiber.Ctx) error { moneroRepo := monero.New() return c.JSON(fiber.Map{ "status": "ok", "message": "Success", "data": moneroRepo.NetFees(), }) } // Returns list of countries (count by nodes) func Countries(c *fiber.Ctx) error { moneroRepo := monero.New() 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, }) } // Returns node to be probed by the client (prober) // // This handler should protected by `CheckProber` middleware. func GiveJob(c *fiber.Ctx) error { acceptTor := c.QueryInt("accept_tor", 0) acceptIPv6 := c.QueryInt("accept_ipv6", 0) moneroRepo := monero.New() node, err := moneroRepo.GiveJob(acceptTor, acceptIPv6) 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, }) } // Handles probe report submission by the prober // // This handler should protected by `CheckProber` middleware. func ProcessJob(c *fiber.Ctx) error { var report monero.ProbeReport if err := c.BodyParser(&report); err != nil { return c.Status(fiber.StatusUnprocessableEntity).JSON(fiber.Map{ "status": "error", "message": err.Error(), "data": nil, }) } moneroRepo := monero.New() 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, }) }