mirror of
https://github.com/ditatompel/xmr-remote-nodes.git
synced 2025-01-08 05:52:10 +07:00
Basic add node action implemented
This commit is contained in:
parent
92acb52aac
commit
7cd802e640
4 changed files with 158 additions and 28 deletions
|
@ -1,26 +1,41 @@
|
|||
<script>
|
||||
import { applyAction, enhance } from '$app/forms';
|
||||
import { slide } from 'svelte/transition';
|
||||
import { invalidateAll, goto } from '$app/navigation';
|
||||
import { apiUri } from '$lib/utils/common';
|
||||
import { ProgressBar } from '@skeletonlabs/skeleton';
|
||||
/** @type {import('./$types').PageData} */
|
||||
export let data;
|
||||
|
||||
/** @type {import('./$types').ActionData} */
|
||||
export let form;
|
||||
/**
|
||||
* @typedef formResult
|
||||
* @type {object}
|
||||
* @property {string} status
|
||||
* @property {string} message
|
||||
* @property {null | object} data
|
||||
*/
|
||||
/** @type {formResult} */
|
||||
export let formResult;
|
||||
|
||||
let isProcessing = false;
|
||||
|
||||
/** @type {import('./$types').SubmitFunction} */
|
||||
const handleForm = async () => {
|
||||
/** @param {{ currentTarget: EventTarget & HTMLFormElement}} event */
|
||||
async function handleSubmit(event) {
|
||||
isProcessing = true;
|
||||
return async ({ result }) => {
|
||||
const data = new FormData(event.currentTarget);
|
||||
|
||||
const response = await fetch(event.currentTarget.action, {
|
||||
method: 'POST',
|
||||
body: data
|
||||
});
|
||||
|
||||
formResult = await response.json();
|
||||
isProcessing = false;
|
||||
if (result.type === 'success' || result.type === 'redirect') {
|
||||
close();
|
||||
|
||||
if (formResult.status === 'ok') {
|
||||
// rerun all `load` functions, following the successful update
|
||||
await invalidateAll();
|
||||
goto('/remote-nodes');
|
||||
}
|
||||
}
|
||||
await applyAction(result);
|
||||
};
|
||||
};
|
||||
</script>
|
||||
|
||||
<header id="hero" class="hero-gradient py-7">
|
||||
|
@ -37,7 +52,12 @@
|
|||
<div class="section-container text-center">
|
||||
<p>Enter your Monero node information below (IPv4 host only):</p>
|
||||
|
||||
<form class="mx-auto w-full max-w-3xl py-2" method="POST" use:enhance={handleForm}>
|
||||
<form
|
||||
class="mx-auto w-full max-w-3xl py-2"
|
||||
action={apiUri('/api/v1/nodes')}
|
||||
method="POST"
|
||||
on:submit|preventDefault={handleSubmit}
|
||||
>
|
||||
<div class="grid grid-cols-1 gap-4 py-6 md:grid-cols-4">
|
||||
<label class="label">
|
||||
<span>Protocol *</span>
|
||||
|
@ -50,7 +70,7 @@
|
|||
<span>Host / IP *</span>
|
||||
<input
|
||||
class="input variant-form-material"
|
||||
name="host"
|
||||
name="hostname"
|
||||
type="text"
|
||||
placeholder="Eg: node.example.com or 172.16.17.18"
|
||||
disabled={isProcessing}
|
||||
|
@ -74,24 +94,23 @@
|
|||
|
||||
<div class="mx-auto w-full max-w-3xl py-2">
|
||||
{#if !isProcessing}
|
||||
{#if form?.status === 'error'}
|
||||
<div class="alert variant-ghost-error" transition:slide={{ duration: 500 }}>
|
||||
<div class="alert-message">
|
||||
<h3 class="h3">Error!</h3>
|
||||
<p>{form.message}!</p>
|
||||
</div>
|
||||
{#if formResult?.status === 'error'}
|
||||
<div class="mx-4 p-4 mb-4 text-sm rounded-lg bg-gray-700 text-red-400" role="alert">
|
||||
<span class="font-medium">Error:</span>
|
||||
{formResult.message}!
|
||||
</div>
|
||||
{/if}
|
||||
{#if form?.status === 'ok'}
|
||||
<div class="alert variant-ghost-success" transition:slide={{ duration: 500 }}>
|
||||
<div class="alert-message">
|
||||
<h3 class="h3">Success!</h3>
|
||||
<p>{form.message}!</p>
|
||||
</div>
|
||||
{#if formResult?.status === 'ok'}
|
||||
<div class="mx-4 p-4 mb-4 text-sm rounded-lg bg-gray-700 text-green-400" role="alert">
|
||||
<span class="font-medium">Success:</span>
|
||||
{formResult.message}!
|
||||
</div>
|
||||
{/if}
|
||||
{:else}
|
||||
<ProgressBar meter="bg-secondary-500" track="bg-secondary-500/30" value={undefined} />
|
||||
<div class="mx-4 p-4 mb-4 text-sm rounded-lg bg-gray-700 text-blue-400" role="alert">
|
||||
<span class="font-medium">Processing...</span>
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
|
|
|
@ -2,6 +2,7 @@ package handler
|
|||
|
||||
import (
|
||||
"fmt"
|
||||
"strconv"
|
||||
"time"
|
||||
|
||||
"github.com/ditatompel/xmr-nodes/internal/database"
|
||||
|
@ -114,6 +115,36 @@ func Prober(c *fiber.Ctx) error {
|
|||
})
|
||||
}
|
||||
|
||||
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 := repo.NewMoneroRepo(database.GetDB())
|
||||
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,
|
||||
})
|
||||
}
|
||||
|
||||
func Crons(c *fiber.Ctx) error {
|
||||
cronRepo := repo.NewCron(database.GetDB())
|
||||
|
||||
|
|
|
@ -14,5 +14,6 @@ func V1Api(app *fiber.App) {
|
|||
|
||||
v1.Get("/prober", Prober)
|
||||
v1.Post("/prober", Prober)
|
||||
v1.Post("/nodes", AddNode)
|
||||
v1.Get("/crons", Crons)
|
||||
}
|
||||
|
|
79
internal/repo/monero.go
Normal file
79
internal/repo/monero.go
Normal file
|
@ -0,0 +1,79 @@
|
|||
package repo
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"net"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/ditatompel/xmr-nodes/internal/database"
|
||||
)
|
||||
|
||||
type MoneroRepository interface {
|
||||
Add(protocol string, host string, port uint) error
|
||||
}
|
||||
|
||||
type MoneroRepo struct {
|
||||
db *database.DB
|
||||
}
|
||||
|
||||
func NewMoneroRepo(db *database.DB) MoneroRepository {
|
||||
return &MoneroRepo{db}
|
||||
}
|
||||
|
||||
func (repo *MoneroRepo) Add(protocol string, hostname string, port uint) error {
|
||||
if protocol != "http" && protocol != "https" {
|
||||
return errors.New("Invalid protocol, must one of or HTTP/HTTPS")
|
||||
}
|
||||
|
||||
if port > 65535 || port < 1 {
|
||||
return errors.New("Invalid port number")
|
||||
}
|
||||
|
||||
is_tor := false
|
||||
if strings.HasSuffix(hostname, ".onion") {
|
||||
is_tor = true
|
||||
}
|
||||
ip := ""
|
||||
|
||||
if !is_tor {
|
||||
hostIps, err := net.LookupIP(hostname)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
hostIp := hostIps[0].To4()
|
||||
if hostIp == nil {
|
||||
return errors.New("Host IP is not IPv4")
|
||||
}
|
||||
if hostIp.IsPrivate() {
|
||||
return errors.New("IP address is private")
|
||||
}
|
||||
if hostIp.IsLoopback() {
|
||||
return errors.New("IP address is loopback address")
|
||||
}
|
||||
|
||||
ip = hostIp.String()
|
||||
}
|
||||
|
||||
check := `SELECT id FROM tbl_node WHERE protocol = ? AND hostname = ? AND port = ? LIMIT 1`
|
||||
row, err := repo.db.Query(check, protocol, hostname, port)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer row.Close()
|
||||
|
||||
if row.Next() {
|
||||
return errors.New("Node already monitored")
|
||||
}
|
||||
statusDb, _ := json.Marshal([5]int{2, 2, 2, 2, 2})
|
||||
|
||||
query := `INSERT INTO tbl_node (protocol, hostname, port, is_tor, nettype, ip_addr, lat, lon, date_entered, last_checked, last_check_status) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`
|
||||
_, err = repo.db.Exec(query, protocol, hostname, port, is_tor, "", ip, 0, 0, time.Now().Unix(), 0, string(statusDb))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
Loading…
Reference in a new issue