2024-05-22 22:45:38 +07:00
|
|
|
package monero
|
2024-05-04 17:24:47 +07:00
|
|
|
|
|
|
|
import (
|
2024-05-07 01:08:01 +07:00
|
|
|
"database/sql"
|
2024-05-04 17:24:47 +07:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2024-05-04 18:52:47 +07:00
|
|
|
"fmt"
|
2024-05-24 05:37:27 +07:00
|
|
|
"log/slog"
|
2024-05-05 01:42:47 +07:00
|
|
|
"math"
|
2024-05-04 17:24:47 +07:00
|
|
|
"net"
|
2024-05-04 18:52:47 +07:00
|
|
|
"slices"
|
2024-05-04 17:24:47 +07:00
|
|
|
"strings"
|
|
|
|
"time"
|
2024-05-08 21:35:04 +07:00
|
|
|
"xmr-remote-nodes/internal/database"
|
2024-05-22 22:45:38 +07:00
|
|
|
"xmr-remote-nodes/internal/geo"
|
2024-05-04 18:52:47 +07:00
|
|
|
|
|
|
|
"github.com/jmoiron/sqlx/types"
|
2024-05-04 17:24:47 +07:00
|
|
|
)
|
|
|
|
|
|
|
|
type MoneroRepository interface {
|
2024-05-22 22:45:38 +07:00
|
|
|
Node(id int) (Node, error)
|
2024-05-04 17:24:47 +07:00
|
|
|
Add(protocol string, host string, port uint) error
|
2024-05-27 05:13:03 +07:00
|
|
|
Nodes(QueryNodes) (Nodes, error)
|
2024-05-22 22:45:38 +07:00
|
|
|
GiveJob(acceptTor int) (Node, error)
|
2024-05-05 01:42:47 +07:00
|
|
|
ProcessJob(report ProbeReport, proberId int64) error
|
2024-05-06 13:19:48 +07:00
|
|
|
NetFee() []NetFee
|
2024-05-22 22:45:38 +07:00
|
|
|
Countries() ([]Countries, error)
|
2024-05-27 06:38:11 +07:00
|
|
|
Logs(QueryLogs) (FetchLogs, error)
|
2024-05-04 17:24:47 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
type MoneroRepo struct {
|
|
|
|
db *database.DB
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:15:40 +07:00
|
|
|
func New() MoneroRepository {
|
|
|
|
return &MoneroRepo{db: database.GetDB()}
|
2024-05-04 17:24:47 +07:00
|
|
|
}
|
|
|
|
|
2024-05-27 05:13:03 +07:00
|
|
|
// Node represents a single remote node
|
2024-05-22 22:45:38 +07:00
|
|
|
type Node struct {
|
|
|
|
ID uint `json:"id,omitempty" db:"id"`
|
2024-05-04 18:52:47 +07:00
|
|
|
Hostname string `json:"hostname" db:"hostname"`
|
2024-05-22 22:45:38 +07:00
|
|
|
IP string `json:"ip" db:"ip_addr"`
|
2024-05-04 18:52:47 +07:00
|
|
|
Port uint `json:"port" db:"port"`
|
|
|
|
Protocol string `json:"protocol" db:"protocol"`
|
|
|
|
IsTor bool `json:"is_tor" db:"is_tor"`
|
|
|
|
IsAvailable bool `json:"is_available" db:"is_available"`
|
2024-05-22 22:45:38 +07:00
|
|
|
Nettype string `json:"nettype" db:"nettype"`
|
2024-05-04 22:53:03 +07:00
|
|
|
Height uint `json:"height" db:"height"`
|
2024-05-04 18:52:47 +07:00
|
|
|
AdjustedTime uint `json:"adjusted_time" db:"adjusted_time"`
|
|
|
|
DatabaseSize uint `json:"database_size" db:"database_size"`
|
|
|
|
Difficulty uint `json:"difficulty" db:"difficulty"`
|
2024-05-04 22:53:03 +07:00
|
|
|
Version string `json:"version" db:"version"`
|
|
|
|
Status string `json:"status,omitempty"`
|
2024-05-05 01:42:47 +07:00
|
|
|
Uptime float64 `json:"uptime" db:"uptime"`
|
2024-05-04 18:52:47 +07:00
|
|
|
EstimateFee uint `json:"estimate_fee" db:"estimate_fee"`
|
2024-05-22 22:45:38 +07:00
|
|
|
ASN uint `json:"asn" db:"asn"`
|
|
|
|
ASNName string `json:"asn_name" db:"asn_name"`
|
2024-05-04 18:52:47 +07:00
|
|
|
CountryCode string `json:"cc" db:"country"`
|
|
|
|
CountryName string `json:"country_name" db:"country_name"`
|
|
|
|
City string `json:"city" db:"city"`
|
2024-05-22 22:45:38 +07:00
|
|
|
Latitude float64 `json:"latitude" db:"lat"`
|
|
|
|
Longitude float64 `json:"longitude" db:"lon"`
|
2024-05-04 18:52:47 +07:00
|
|
|
DateEntered uint `json:"date_entered,omitempty" db:"date_entered"`
|
|
|
|
LastChecked uint `json:"last_checked" db:"last_checked"`
|
|
|
|
FailedCount uint `json:"failed_count,omitempty" db:"failed_count"`
|
|
|
|
LastCheckStatus types.JSONText `json:"last_check_statuses" db:"last_check_status"`
|
2024-05-22 22:45:38 +07:00
|
|
|
CORSCapable bool `json:"cors" db:"cors_capable"`
|
2024-05-04 18:52:47 +07:00
|
|
|
}
|
|
|
|
|
2024-05-27 05:13:03 +07:00
|
|
|
// Get node from database by id
|
2024-05-22 22:45:38 +07:00
|
|
|
func (repo *MoneroRepo) Node(id int) (Node, error) {
|
|
|
|
var node Node
|
2024-05-07 01:08:01 +07:00
|
|
|
err := repo.db.Get(&node, `SELECT * FROM tbl_node WHERE id = ?`, id)
|
|
|
|
if err != nil && err != sql.ErrNoRows {
|
|
|
|
fmt.Println("WARN:", err)
|
|
|
|
return node, errors.New("Can't get node information")
|
|
|
|
}
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
return node, errors.New("Node not found")
|
|
|
|
}
|
|
|
|
return node, err
|
|
|
|
}
|
|
|
|
|
2024-05-27 05:13:03 +07:00
|
|
|
// Nodes represents a list of nodes
|
2024-05-22 22:45:38 +07:00
|
|
|
type Nodes struct {
|
|
|
|
TotalRows int `json:"total_rows"`
|
|
|
|
RowsPerPage int `json:"rows_per_page"`
|
|
|
|
Items []*Node `json:"items"`
|
2024-05-04 18:52:47 +07:00
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
// QueryNodes represents database query parameters
|
2024-05-27 05:13:03 +07:00
|
|
|
type QueryNodes struct {
|
|
|
|
Host string
|
|
|
|
Nettype string
|
|
|
|
Protocol string
|
|
|
|
CC string // 2 letter country code
|
|
|
|
Status int
|
|
|
|
CORS int
|
|
|
|
|
|
|
|
// pagination
|
2024-05-04 18:52:47 +07:00
|
|
|
RowsPerPage int
|
|
|
|
Page int
|
|
|
|
SortBy string
|
|
|
|
SortDirection string
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
// Get nodes from database
|
2024-05-27 05:13:03 +07:00
|
|
|
func (repo *MoneroRepo) Nodes(q QueryNodes) (Nodes, error) {
|
2024-05-04 18:52:47 +07:00
|
|
|
queryParams := []interface{}{}
|
|
|
|
whereQueries := []string{}
|
|
|
|
where := ""
|
|
|
|
|
|
|
|
if q.Host != "" {
|
|
|
|
whereQueries = append(whereQueries, "(hostname LIKE ? OR ip_addr LIKE ?)")
|
|
|
|
queryParams = append(queryParams, "%"+q.Host+"%")
|
|
|
|
queryParams = append(queryParams, "%"+q.Host+"%")
|
|
|
|
}
|
2024-05-22 22:45:38 +07:00
|
|
|
if q.Nettype != "any" {
|
|
|
|
if q.Nettype != "mainnet" && q.Nettype != "stagenet" && q.Nettype != "testnet" {
|
|
|
|
return Nodes{}, errors.New("Invalid nettype, must be one of 'mainnet', 'stagenet', 'testnet' or 'any'")
|
2024-05-09 20:44:49 +07:00
|
|
|
}
|
2024-05-06 14:33:13 +07:00
|
|
|
whereQueries = append(whereQueries, "nettype = ?")
|
2024-05-22 22:45:38 +07:00
|
|
|
queryParams = append(queryParams, q.Nettype)
|
2024-05-06 14:33:13 +07:00
|
|
|
}
|
|
|
|
if q.Protocol != "any" {
|
2024-05-09 20:44:49 +07:00
|
|
|
allowedProtocols := []string{"tor", "http", "https"}
|
|
|
|
if !slices.Contains(allowedProtocols, q.Protocol) {
|
2024-05-22 22:45:38 +07:00
|
|
|
return Nodes{}, errors.New("Invalid protocol, must be one of '" + strings.Join(allowedProtocols, "', '") + "' or 'any'")
|
2024-05-09 20:44:49 +07:00
|
|
|
}
|
2024-05-06 14:33:13 +07:00
|
|
|
if q.Protocol == "tor" {
|
|
|
|
whereQueries = append(whereQueries, "is_tor = ?")
|
|
|
|
queryParams = append(queryParams, 1)
|
|
|
|
} else {
|
|
|
|
whereQueries = append(whereQueries, "(protocol = ? AND is_tor = ?)")
|
|
|
|
queryParams = append(queryParams, q.Protocol)
|
|
|
|
queryParams = append(queryParams, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if q.CC != "any" {
|
|
|
|
whereQueries = append(whereQueries, "country = ?")
|
|
|
|
if q.CC == "UNKNOWN" {
|
|
|
|
queryParams = append(queryParams, "")
|
|
|
|
} else {
|
|
|
|
queryParams = append(queryParams, q.CC)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if q.Status != -1 {
|
|
|
|
whereQueries = append(whereQueries, "is_available = ?")
|
|
|
|
queryParams = append(queryParams, q.Status)
|
|
|
|
}
|
2024-05-22 22:45:38 +07:00
|
|
|
if q.CORS != -1 {
|
2024-05-06 14:33:13 +07:00
|
|
|
whereQueries = append(whereQueries, "cors_capable = ?")
|
|
|
|
queryParams = append(queryParams, 1)
|
|
|
|
}
|
2024-05-04 18:52:47 +07:00
|
|
|
|
|
|
|
if len(whereQueries) > 0 {
|
|
|
|
where = "WHERE " + strings.Join(whereQueries, " AND ")
|
|
|
|
}
|
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
nodes := Nodes{}
|
2024-05-04 18:52:47 +07:00
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
queryTotalRows := fmt.Sprintf(`
|
|
|
|
SELECT
|
|
|
|
COUNT(id) AS total_rows
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
%s`, where)
|
2024-05-04 18:52:47 +07:00
|
|
|
|
|
|
|
err := repo.db.QueryRow(queryTotalRows, queryParams...).Scan(&nodes.TotalRows)
|
|
|
|
if err != nil {
|
|
|
|
return nodes, err
|
|
|
|
}
|
|
|
|
queryParams = append(queryParams, q.RowsPerPage, (q.Page-1)*q.RowsPerPage)
|
|
|
|
|
|
|
|
allowedSort := []string{"last_checked", "uptime"}
|
|
|
|
sortBy := "last_checked"
|
|
|
|
if slices.Contains(allowedSort, q.SortBy) {
|
|
|
|
sortBy = q.SortBy
|
|
|
|
}
|
|
|
|
sortDirection := "DESC"
|
|
|
|
if q.SortDirection == "asc" {
|
|
|
|
sortDirection = "ASC"
|
|
|
|
}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
query := fmt.Sprintf(`
|
|
|
|
SELECT
|
|
|
|
id,
|
|
|
|
protocol,
|
|
|
|
hostname,
|
|
|
|
port,
|
|
|
|
is_tor,
|
|
|
|
is_available,
|
|
|
|
nettype,
|
|
|
|
height,
|
|
|
|
adjusted_time,
|
|
|
|
database_size,
|
|
|
|
difficulty,
|
|
|
|
version,
|
|
|
|
uptime,
|
|
|
|
estimate_fee,
|
|
|
|
ip_addr,
|
|
|
|
asn,
|
|
|
|
asn_name,
|
|
|
|
country,
|
|
|
|
country_name,
|
|
|
|
city,
|
|
|
|
lat,
|
|
|
|
lon,
|
|
|
|
date_entered,
|
|
|
|
last_checked,
|
|
|
|
last_check_status,
|
|
|
|
cors_capable
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
%s -- where query if any
|
|
|
|
ORDER BY
|
|
|
|
%s
|
|
|
|
%s
|
|
|
|
LIMIT ?
|
|
|
|
OFFSET ?`, where, sortBy, sortDirection)
|
2024-05-04 18:52:47 +07:00
|
|
|
|
|
|
|
row, err := repo.db.Query(query, queryParams...)
|
|
|
|
if err != nil {
|
|
|
|
return nodes, err
|
|
|
|
}
|
|
|
|
defer row.Close()
|
|
|
|
|
|
|
|
nodes.RowsPerPage = q.RowsPerPage
|
|
|
|
|
|
|
|
for row.Next() {
|
2024-05-22 22:45:38 +07:00
|
|
|
var node Node
|
2024-05-20 04:22:58 +07:00
|
|
|
err = row.Scan(
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.ID,
|
2024-05-20 04:22:58 +07:00
|
|
|
&node.Protocol,
|
|
|
|
&node.Hostname,
|
|
|
|
&node.Port,
|
|
|
|
&node.IsTor,
|
|
|
|
&node.IsAvailable,
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.Nettype,
|
2024-05-20 04:22:58 +07:00
|
|
|
&node.Height,
|
|
|
|
&node.AdjustedTime,
|
|
|
|
&node.DatabaseSize,
|
|
|
|
&node.Difficulty,
|
|
|
|
&node.Version,
|
|
|
|
&node.Uptime,
|
|
|
|
&node.EstimateFee,
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.IP,
|
|
|
|
&node.ASN,
|
|
|
|
&node.ASNName,
|
2024-05-20 04:22:58 +07:00
|
|
|
&node.CountryCode,
|
|
|
|
&node.CountryName,
|
|
|
|
&node.City,
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.Latitude,
|
|
|
|
&node.Longitude,
|
2024-05-20 04:22:58 +07:00
|
|
|
&node.DateEntered,
|
|
|
|
&node.LastChecked,
|
|
|
|
&node.LastCheckStatus,
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.CORSCapable)
|
2024-05-04 18:52:47 +07:00
|
|
|
if err != nil {
|
|
|
|
return nodes, err
|
|
|
|
}
|
|
|
|
nodes.Items = append(nodes.Items, &node)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nodes, nil
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
type QueryLogs struct {
|
2024-05-22 22:45:38 +07:00
|
|
|
NodeID int // 0 fpr all, >0 for specific node
|
|
|
|
WorkerID int // 0 for all, >0 for specific worker
|
2024-05-07 21:31:40 +07:00
|
|
|
Status int // -1 for all, 0 for failed, 1 for success
|
|
|
|
FailedReason string // empty for all, if not empty, will be used as search from failed_reaso
|
2024-05-06 17:19:17 +07:00
|
|
|
|
|
|
|
RowsPerPage int
|
|
|
|
Page int
|
|
|
|
SortBy string
|
|
|
|
SortDirection string
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
type FetchLog struct {
|
2024-05-22 22:45:38 +07:00
|
|
|
ID int `db:"id" json:"id,omitempty"`
|
|
|
|
NodeID int `db:"node_id" json:"node_id"`
|
|
|
|
ProberID int `db:"prober_id" json:"prober_id"`
|
2024-05-06 17:19:17 +07:00
|
|
|
Status int `db:"is_available" json:"status"`
|
|
|
|
Height int `db:"height" json:"height"`
|
|
|
|
AdjustedTime int `db:"adjusted_time" json:"adjusted_time"`
|
|
|
|
DatabaseSize int `db:"database_size" json:"database_size"`
|
|
|
|
Difficulty int `db:"difficulty" json:"difficulty"`
|
|
|
|
EstimateFee int `db:"estimate_fee" json:"estimate_fee"`
|
|
|
|
DateChecked int `db:"date_checked" json:"date_checked"`
|
|
|
|
FailedReason string `db:"failed_reason" json:"failed_reason"`
|
|
|
|
FetchRuntime float64 `db:"fetch_runtime" json:"fetch_runtime"`
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
type FetchLogs struct {
|
2024-05-06 17:19:17 +07:00
|
|
|
TotalRows int `json:"total_rows"`
|
|
|
|
RowsPerPage int `json:"rows_per_page"`
|
2024-05-27 06:38:11 +07:00
|
|
|
Items []*FetchLog `json:"items"`
|
2024-05-06 17:19:17 +07:00
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
// Logs returns list of fetched log result for given query
|
|
|
|
func (repo *MoneroRepo) Logs(q QueryLogs) (FetchLogs, error) {
|
2024-05-06 17:19:17 +07:00
|
|
|
queryParams := []interface{}{}
|
|
|
|
whereQueries := []string{}
|
|
|
|
where := ""
|
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
if q.NodeID != 0 {
|
2024-05-06 17:19:17 +07:00
|
|
|
whereQueries = append(whereQueries, "node_id = ?")
|
2024-05-22 22:45:38 +07:00
|
|
|
queryParams = append(queryParams, q.NodeID)
|
2024-05-06 17:19:17 +07:00
|
|
|
}
|
2024-05-07 21:31:40 +07:00
|
|
|
if q.Status != -1 {
|
|
|
|
whereQueries = append(whereQueries, "is_available = ?")
|
|
|
|
queryParams = append(queryParams, q.Status)
|
|
|
|
}
|
|
|
|
if q.FailedReason != "" {
|
|
|
|
whereQueries = append(whereQueries, "failed_reason LIKE ?")
|
|
|
|
queryParams = append(queryParams, "%"+q.FailedReason+"%")
|
|
|
|
}
|
2024-05-06 17:19:17 +07:00
|
|
|
|
|
|
|
if len(whereQueries) > 0 {
|
|
|
|
where = "WHERE " + strings.Join(whereQueries, " AND ")
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
var fetchLogs FetchLogs
|
2024-05-06 17:19:17 +07:00
|
|
|
|
|
|
|
queryTotalRows := fmt.Sprintf("SELECT COUNT(id) FROM tbl_probe_log %s", where)
|
|
|
|
err := repo.db.QueryRow(queryTotalRows, queryParams...).Scan(&fetchLogs.TotalRows)
|
|
|
|
if err != nil {
|
|
|
|
return fetchLogs, err
|
|
|
|
}
|
|
|
|
queryParams = append(queryParams, q.RowsPerPage, (q.Page-1)*q.RowsPerPage)
|
|
|
|
|
|
|
|
allowedSort := []string{"date_checked", "fetch_runtime"}
|
|
|
|
sortBy := "id"
|
|
|
|
if slices.Contains(allowedSort, q.SortBy) {
|
|
|
|
sortBy = q.SortBy
|
|
|
|
}
|
|
|
|
sortDirection := "DESC"
|
|
|
|
if q.SortDirection == "asc" {
|
|
|
|
sortDirection = "ASC"
|
|
|
|
}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
query := fmt.Sprintf(`
|
|
|
|
SELECT
|
|
|
|
id,
|
|
|
|
node_id,
|
|
|
|
prober_id,
|
|
|
|
is_available,
|
|
|
|
height,
|
|
|
|
adjusted_time,
|
|
|
|
database_size,
|
|
|
|
difficulty,
|
|
|
|
estimate_fee,
|
|
|
|
date_checked,
|
|
|
|
failed_reason,
|
|
|
|
fetch_runtime
|
|
|
|
FROM
|
|
|
|
tbl_probe_log
|
|
|
|
%s -- where query
|
|
|
|
ORDER BY
|
|
|
|
%s
|
|
|
|
%s
|
|
|
|
LIMIT ?
|
|
|
|
OFFSET ?`, where, sortBy, sortDirection)
|
2024-05-06 17:19:17 +07:00
|
|
|
|
|
|
|
row, err := repo.db.Query(query, queryParams...)
|
|
|
|
if err != nil {
|
|
|
|
return fetchLogs, err
|
|
|
|
}
|
|
|
|
defer row.Close()
|
|
|
|
|
|
|
|
fetchLogs.RowsPerPage = q.RowsPerPage
|
|
|
|
|
|
|
|
for row.Next() {
|
2024-05-27 06:38:11 +07:00
|
|
|
var fl FetchLog
|
2024-05-20 04:22:58 +07:00
|
|
|
err = row.Scan(
|
2024-05-27 06:38:11 +07:00
|
|
|
&fl.ID,
|
|
|
|
&fl.NodeID,
|
|
|
|
&fl.ProberID,
|
|
|
|
&fl.Status,
|
|
|
|
&fl.Height,
|
|
|
|
&fl.AdjustedTime,
|
|
|
|
&fl.DatabaseSize,
|
|
|
|
&fl.Difficulty,
|
|
|
|
&fl.EstimateFee,
|
|
|
|
&fl.DateChecked,
|
|
|
|
&fl.FailedReason,
|
|
|
|
&fl.FetchRuntime)
|
2024-05-06 17:19:17 +07:00
|
|
|
if err != nil {
|
|
|
|
return fetchLogs, err
|
|
|
|
}
|
|
|
|
|
2024-05-27 06:38:11 +07:00
|
|
|
fetchLogs.Items = append(fetchLogs.Items, &fl)
|
2024-05-06 17:19:17 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return fetchLogs, nil
|
|
|
|
}
|
|
|
|
|
2024-05-04 17:24:47 +07:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
row, err := repo.db.Query(`
|
|
|
|
SELECT
|
|
|
|
id
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
WHERE
|
|
|
|
protocol = ?
|
|
|
|
AND hostname = ?
|
|
|
|
AND port = ?
|
|
|
|
LIMIT 1`, protocol, hostname, port)
|
2024-05-04 17:24:47 +07:00
|
|
|
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})
|
2024-05-20 04:22:58 +07:00
|
|
|
_, err = repo.db.Exec(`
|
|
|
|
INSERT INTO tbl_node (
|
|
|
|
protocol,
|
|
|
|
hostname,
|
|
|
|
port,
|
|
|
|
is_tor,
|
|
|
|
nettype,
|
|
|
|
ip_addr,
|
|
|
|
lat,
|
|
|
|
lon,
|
|
|
|
date_entered,
|
|
|
|
last_checked,
|
|
|
|
last_check_status
|
|
|
|
) VALUES (
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?
|
|
|
|
)`,
|
|
|
|
protocol,
|
|
|
|
hostname,
|
|
|
|
port,
|
|
|
|
is_tor,
|
|
|
|
"",
|
|
|
|
ip,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
time.Now().Unix(),
|
|
|
|
0,
|
|
|
|
string(statusDb))
|
2024-05-04 17:24:47 +07:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2024-05-04 19:27:21 +07:00
|
|
|
|
2024-05-06 17:45:18 +07:00
|
|
|
func (repo *MoneroRepo) Delete(id uint) error {
|
|
|
|
if _, err := repo.db.Exec(`DELETE FROM tbl_node WHERE id = ?`, id); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if _, err := repo.db.Exec(`DELETE FROM tbl_probe_log WHERE node_id = ?`, id); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
func (repo *MoneroRepo) GiveJob(acceptTor int) (Node, error) {
|
2024-05-04 19:27:21 +07:00
|
|
|
queryParams := []interface{}{}
|
|
|
|
whereQueries := []string{}
|
|
|
|
where := ""
|
|
|
|
|
|
|
|
if acceptTor != 1 {
|
|
|
|
whereQueries = append(whereQueries, "is_tor = ?")
|
|
|
|
queryParams = append(queryParams, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(whereQueries) > 0 {
|
|
|
|
where = "WHERE " + strings.Join(whereQueries, " AND ")
|
|
|
|
}
|
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
var node Node
|
2024-05-04 19:27:21 +07:00
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
query := fmt.Sprintf(`
|
|
|
|
SELECT
|
|
|
|
id,
|
|
|
|
hostname,
|
|
|
|
port,
|
|
|
|
protocol,
|
|
|
|
is_tor,
|
|
|
|
last_check_status
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
%s -- where query if any
|
|
|
|
ORDER BY
|
|
|
|
last_checked ASC
|
|
|
|
LIMIT 1`, where)
|
|
|
|
err := repo.db.QueryRow(query, queryParams...).Scan(
|
2024-05-22 22:45:38 +07:00
|
|
|
&node.ID,
|
2024-05-20 04:22:58 +07:00
|
|
|
&node.Hostname,
|
|
|
|
&node.Port,
|
|
|
|
&node.Protocol,
|
|
|
|
&node.IsTor,
|
|
|
|
&node.LastCheckStatus)
|
2024-05-04 19:27:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return node, err
|
|
|
|
}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
_, err = repo.db.Exec(`
|
|
|
|
UPDATE tbl_node
|
|
|
|
SET last_checked = ?
|
2024-05-22 22:45:38 +07:00
|
|
|
WHERE id = ?`, time.Now().Unix(), node.ID)
|
2024-05-04 19:27:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return node, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return node, nil
|
|
|
|
}
|
2024-05-05 01:42:47 +07:00
|
|
|
|
|
|
|
type ProbeReport struct {
|
2024-05-22 22:45:38 +07:00
|
|
|
TookTime float64 `json:"took_time"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
NodeInfo Node `json:"node_info"`
|
2024-05-05 01:42:47 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *MoneroRepo) ProcessJob(report ProbeReport, proberId int64) error {
|
2024-05-22 22:45:38 +07:00
|
|
|
if report.NodeInfo.ID == 0 {
|
2024-05-05 01:42:47 +07:00
|
|
|
return errors.New("Invalid node")
|
|
|
|
}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
qInsertLog := `
|
|
|
|
INSERT INTO tbl_probe_log (
|
|
|
|
node_id,
|
|
|
|
prober_id,
|
|
|
|
is_available,
|
|
|
|
height,
|
|
|
|
adjusted_time,
|
|
|
|
database_size,
|
|
|
|
difficulty,
|
|
|
|
estimate_fee,
|
|
|
|
date_checked,
|
|
|
|
failed_reason,
|
|
|
|
fetch_runtime
|
|
|
|
) VALUES (
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?,
|
|
|
|
?
|
|
|
|
)`
|
|
|
|
_, err := repo.db.Exec(qInsertLog,
|
2024-05-22 22:45:38 +07:00
|
|
|
report.NodeInfo.ID,
|
2024-05-20 04:22:58 +07:00
|
|
|
proberId,
|
|
|
|
report.NodeInfo.IsAvailable,
|
|
|
|
report.NodeInfo.Height,
|
|
|
|
report.NodeInfo.AdjustedTime,
|
|
|
|
report.NodeInfo.DatabaseSize,
|
|
|
|
report.NodeInfo.Difficulty,
|
|
|
|
report.NodeInfo.EstimateFee,
|
|
|
|
time.Now().Unix(),
|
|
|
|
report.Message,
|
|
|
|
report.TookTime)
|
2024-05-05 01:42:47 +07:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
now := time.Now()
|
|
|
|
limitTs := now.AddDate(0, -1, 0).Unix()
|
|
|
|
|
|
|
|
nodeStats := struct {
|
|
|
|
OnlineCount uint `db:"online"`
|
|
|
|
OfflineCount uint `db:"offline"`
|
|
|
|
TotalFetched uint `db:"total_fetched"`
|
|
|
|
}{}
|
|
|
|
|
2024-05-20 04:22:58 +07:00
|
|
|
qstats := `
|
|
|
|
SELECT
|
|
|
|
SUM(if(is_available='1',1,0)) AS online,
|
|
|
|
SUM(if(is_available='0',1,0)) AS offline,
|
|
|
|
SUM(if(id='0',0,1)) AS total_fetched
|
|
|
|
FROM
|
|
|
|
tbl_probe_log
|
|
|
|
WHERE
|
|
|
|
node_id = ?
|
|
|
|
AND date_checked > ?`
|
2024-05-24 05:37:27 +07:00
|
|
|
if err := repo.db.Get(&nodeStats, qstats, report.NodeInfo.ID, limitTs); err != nil {
|
|
|
|
slog.Warn(err.Error())
|
|
|
|
}
|
2024-05-05 01:42:47 +07:00
|
|
|
|
|
|
|
avgUptime := (float64(nodeStats.OnlineCount) / float64(nodeStats.TotalFetched)) * 100
|
|
|
|
report.NodeInfo.Uptime = math.Ceil(avgUptime*100) / 100
|
|
|
|
|
|
|
|
var statuses [5]int
|
|
|
|
errUnmarshal := report.NodeInfo.LastCheckStatus.Unmarshal(&statuses)
|
|
|
|
if errUnmarshal != nil {
|
|
|
|
fmt.Println("Warning", errUnmarshal.Error())
|
|
|
|
statuses = [5]int{2, 2, 2, 2, 2}
|
|
|
|
}
|
|
|
|
|
|
|
|
nodeAvailable := 0
|
|
|
|
|
|
|
|
if report.NodeInfo.IsAvailable {
|
|
|
|
nodeAvailable = 1
|
|
|
|
}
|
|
|
|
newStatuses := statuses[1:]
|
|
|
|
newStatuses = append(newStatuses, nodeAvailable)
|
|
|
|
statuesValueToDb, errMarshalStatus := json.Marshal(newStatuses)
|
|
|
|
if errMarshalStatus != nil {
|
|
|
|
fmt.Println("WARN", errMarshalStatus.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// recheck IP
|
2024-05-22 22:45:38 +07:00
|
|
|
if report.NodeInfo.IP != "" {
|
2024-05-30 12:46:33 +07:00
|
|
|
if ipInfo, errGeoIp := geo.Info(report.NodeInfo.IP); errGeoIp != nil {
|
2024-05-05 02:20:54 +07:00
|
|
|
fmt.Println("WARN:", errGeoIp.Error())
|
|
|
|
} else {
|
2024-05-30 12:46:33 +07:00
|
|
|
report.NodeInfo.ASN = ipInfo.ASN
|
|
|
|
report.NodeInfo.ASNName = ipInfo.ASNOrg
|
2024-05-05 02:20:54 +07:00
|
|
|
report.NodeInfo.CountryCode = ipInfo.CountryCode
|
|
|
|
report.NodeInfo.CountryName = ipInfo.CountryName
|
|
|
|
report.NodeInfo.City = ipInfo.City
|
2024-05-22 22:45:38 +07:00
|
|
|
report.NodeInfo.Longitude = ipInfo.Longitude
|
|
|
|
report.NodeInfo.Latitude = ipInfo.Latitude
|
2024-05-05 02:20:54 +07:00
|
|
|
}
|
|
|
|
}
|
2024-05-05 01:42:47 +07:00
|
|
|
|
2024-05-08 19:03:41 +07:00
|
|
|
if report.NodeInfo.IsAvailable {
|
2024-05-20 04:22:58 +07:00
|
|
|
update := `
|
|
|
|
UPDATE tbl_node
|
|
|
|
SET
|
|
|
|
is_available = ?,
|
|
|
|
nettype = ?,
|
|
|
|
height = ?,
|
|
|
|
adjusted_time = ?,
|
|
|
|
database_size = ?,
|
|
|
|
difficulty = ?,
|
|
|
|
version = ?,
|
|
|
|
uptime = ?,
|
|
|
|
estimate_fee = ?,
|
|
|
|
ip_addr = ?,
|
|
|
|
asn = ?,
|
|
|
|
asn_name = ?,
|
|
|
|
country = ?,
|
|
|
|
country_name = ?,
|
|
|
|
city = ?,
|
|
|
|
last_checked = ?,
|
|
|
|
last_check_status = ?,
|
2024-05-08 19:29:26 +07:00
|
|
|
cors_capable = ?
|
2024-05-20 04:22:58 +07:00
|
|
|
WHERE
|
|
|
|
id = ?`
|
2024-05-24 05:37:27 +07:00
|
|
|
_, err := repo.db.Exec(update,
|
2024-05-20 04:22:58 +07:00
|
|
|
nodeAvailable,
|
2024-05-22 22:45:38 +07:00
|
|
|
report.NodeInfo.Nettype,
|
2024-05-20 04:22:58 +07:00
|
|
|
report.NodeInfo.Height,
|
|
|
|
report.NodeInfo.AdjustedTime,
|
|
|
|
report.NodeInfo.DatabaseSize,
|
|
|
|
report.NodeInfo.Difficulty,
|
|
|
|
report.NodeInfo.Version,
|
|
|
|
report.NodeInfo.Uptime,
|
|
|
|
report.NodeInfo.EstimateFee,
|
2024-05-22 22:45:38 +07:00
|
|
|
report.NodeInfo.IP,
|
|
|
|
report.NodeInfo.ASN,
|
|
|
|
report.NodeInfo.ASNName,
|
2024-05-20 04:22:58 +07:00
|
|
|
report.NodeInfo.CountryCode,
|
|
|
|
report.NodeInfo.CountryName,
|
|
|
|
report.NodeInfo.City,
|
|
|
|
now.Unix(),
|
|
|
|
string(statuesValueToDb),
|
2024-05-22 22:45:38 +07:00
|
|
|
report.NodeInfo.CORSCapable,
|
|
|
|
report.NodeInfo.ID)
|
2024-05-24 05:37:27 +07:00
|
|
|
if err != nil {
|
|
|
|
slog.Warn(err.Error())
|
|
|
|
}
|
2024-05-08 19:03:41 +07:00
|
|
|
} else {
|
2024-05-24 05:37:27 +07:00
|
|
|
u := `
|
2024-05-20 04:22:58 +07:00
|
|
|
UPDATE tbl_node
|
|
|
|
SET
|
|
|
|
is_available = ?,
|
|
|
|
uptime = ?,
|
|
|
|
last_checked = ?,
|
|
|
|
last_check_status = ?
|
|
|
|
WHERE
|
|
|
|
id = ?`
|
2024-05-24 05:37:27 +07:00
|
|
|
if _, err := repo.db.Exec(u, nodeAvailable, report.NodeInfo.Uptime, now.Unix(), string(statuesValueToDb), report.NodeInfo.ID); err != nil {
|
|
|
|
slog.Warn(err.Error())
|
|
|
|
}
|
2024-05-08 19:03:41 +07:00
|
|
|
}
|
2024-05-05 01:42:47 +07:00
|
|
|
|
2024-05-06 17:45:18 +07:00
|
|
|
if avgUptime <= 0 && nodeStats.TotalFetched > 300 {
|
|
|
|
fmt.Println("Deleting Monero node (0% uptime from > 300 records)")
|
2024-05-24 05:37:27 +07:00
|
|
|
if err := repo.Delete(report.NodeInfo.ID); err != nil {
|
|
|
|
slog.Warn(err.Error())
|
|
|
|
}
|
2024-05-06 17:45:18 +07:00
|
|
|
}
|
|
|
|
|
2024-05-24 05:37:27 +07:00
|
|
|
_, err = repo.db.Exec(`
|
2024-05-20 04:22:58 +07:00
|
|
|
UPDATE tbl_prober
|
|
|
|
SET last_submit_ts = ?
|
|
|
|
WHERE id = ?`, now.Unix(), proberId)
|
2024-05-07 22:01:21 +07:00
|
|
|
|
2024-05-05 01:42:47 +07:00
|
|
|
return err
|
|
|
|
}
|
2024-05-06 13:19:48 +07:00
|
|
|
|
|
|
|
type NetFee struct {
|
|
|
|
Nettype string `json:"nettype" db:"nettype"`
|
|
|
|
EstimateFee uint `json:"estimate_fee" db:"estimate_fee"`
|
|
|
|
NodeCount int `json:"node_count" db:"node_count"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *MoneroRepo) NetFee() []NetFee {
|
|
|
|
netTypes := [3]string{"mainnet", "stagenet", "testnet"}
|
|
|
|
netFees := []NetFee{}
|
|
|
|
|
|
|
|
for _, net := range netTypes {
|
|
|
|
fees := NetFee{}
|
2024-05-20 04:22:58 +07:00
|
|
|
err := repo.db.Get(&fees, `
|
|
|
|
SELECT
|
|
|
|
COUNT(id) AS node_count,
|
|
|
|
nettype,
|
|
|
|
estimate_fee
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
WHERE
|
|
|
|
nettype = ?
|
|
|
|
GROUP BY
|
|
|
|
estimate_fee
|
|
|
|
ORDER BY
|
|
|
|
node_count DESC
|
|
|
|
LIMIT 1`, net)
|
2024-05-06 13:19:48 +07:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println("WARN:", err.Error())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
netFees = append(netFees, fees)
|
|
|
|
}
|
|
|
|
|
|
|
|
return netFees
|
|
|
|
}
|
2024-05-06 13:35:15 +07:00
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
type Countries struct {
|
2024-05-06 13:35:15 +07:00
|
|
|
TotalNodes int `json:"total_nodes" db:"total_nodes"`
|
2024-05-22 22:45:38 +07:00
|
|
|
CC string `json:"cc" db:"country"` // country code
|
2024-05-06 13:35:15 +07:00
|
|
|
Name string `json:"name" db:"country_name"`
|
|
|
|
}
|
|
|
|
|
2024-05-22 22:45:38 +07:00
|
|
|
func (repo *MoneroRepo) Countries() ([]Countries, error) {
|
|
|
|
countries := []Countries{}
|
2024-05-20 04:22:58 +07:00
|
|
|
err := repo.db.Select(&countries, `
|
|
|
|
SELECT
|
|
|
|
COUNT(id) AS total_nodes,
|
|
|
|
country,
|
|
|
|
country_name
|
|
|
|
FROM
|
|
|
|
tbl_node
|
|
|
|
GROUP BY
|
|
|
|
country
|
|
|
|
ORDER BY
|
|
|
|
country ASC`)
|
2024-05-06 13:35:15 +07:00
|
|
|
return countries, err
|
|
|
|
}
|