Files
hamview/server.go
maze fb898bb058
Some checks failed
Run tests / test (1.25) (push) Has been cancelled
Run tests / test (stable) (push) Has been cancelled
Initial import
2026-02-22 20:27:07 +01:00

424 lines
9.7 KiB
Go

package hamview
import (
"database/sql"
"encoding/base64"
"encoding/hex"
"errors"
"fmt"
"net"
"net/http"
"os"
"slices"
"strconv"
"strings"
"time"
echologrus "github.com/cemkiy/echo-logrus"
"github.com/labstack/echo/v4"
"github.com/labstack/echo/v4/middleware"
"git.maze.io/go/ham/protocol/meshcore"
"git.maze.io/go/ham/protocol/meshcore/crypto"
)
const DefaultServerListen = ":8073"
type ServerConfig struct {
Listen string `yaml:"listen"`
}
type Server struct {
listen string
listenAddr *net.TCPAddr
db *sql.DB
}
func NewServer(serverConfig *ServerConfig, databaseConfig *DatabaseConfig) (*Server, error) {
if serverConfig.Listen == "" {
serverConfig.Listen = DefaultServerListen
}
listenAddr, err := net.ResolveTCPAddr("tcp", serverConfig.Listen)
if err != nil {
return nil, fmt.Errorf("hamview: invalid listen address %q: %v", serverConfig.Listen, err)
}
db, err := sql.Open(databaseConfig.Type, databaseConfig.Conf)
if err != nil {
return nil, err
}
return &Server{
listen: serverConfig.Listen,
listenAddr: listenAddr,
db: db,
}, nil
}
func (server *Server) Run() error {
echologrus.Logger = Logger
e := echo.New()
e.Logger = echologrus.GetEchoLogger()
e.Use(echologrus.Hook())
e.Use(middleware.RequestLogger())
e.Use(middleware.CORSWithConfig(middleware.CORSConfig{
AllowOrigins: []string{"*"},
AllowHeaders: []string{echo.HeaderOrigin, echo.HeaderContentType, echo.HeaderAccept},
}))
e.GET("/api/v1/meshcore/nodes", server.apiGetMeshCoreNodes)
e.GET("/api/v1/meshcore/packets", server.apiGetMeshCorePackets)
e.GET("/api/v1/meshcore/path/:origin/:path", server.apiGetMeshCorePath)
e.GET("/api/v1/meshcore/sources", server.apiGetMeshCoreSources)
if server.listenAddr.IP == nil || server.listenAddr.IP.Equal(net.ParseIP("0.0.0.0")) || server.listenAddr.IP.Equal(net.ParseIP("::")) {
Logger.Infof("server: listening on http://127.0.0.1:%d", server.listenAddr.Port)
} else {
Logger.Infof("server: listening on http://%s:%d", server.listenAddr.IP, server.listenAddr.Port)
}
return e.Start(server.listen)
}
func (server *Server) apiError(ctx echo.Context, err error, status ...int) error {
Logger.Warnf("server: error serving %s %s: %v", ctx.Request().Method, ctx.Request().URL.Path, err)
if len(status) > 0 {
return ctx.JSON(status[0], map[string]any{
"error": err.Error(),
})
}
switch {
case os.IsNotExist(err):
return ctx.JSON(http.StatusNotFound, nil)
case os.IsPermission(err):
return ctx.JSON(http.StatusUnauthorized, nil)
default:
return ctx.JSON(http.StatusInternalServerError, map[string]any{
"error": err.Error(),
})
}
}
type meshCoreNodeResponse struct {
SNR float64 `json:"snr"`
RSSI int8 `json:"rssi"`
Name string `json:"name"`
PublicKey []byte `json:"public_key"`
Prefix byte `json:"prefix"`
NodeType meshcore.NodeType `json:"node_type"`
FirstHeard time.Time `json:"first_heard"`
LastHeard time.Time `json:"last_heard"`
Position *meshcore.Position `json:"position"`
}
func (server *Server) apiGetMeshCoreNodes(ctx echo.Context) error {
/*
nodeTypes := getQueryInts(ctx, "type")
if len(nodeTypes) == 0 {
nodeTypes = []int{
int(meshcore.Repeater),
}
}
*/
nodeType := meshcore.Repeater
if ctx.QueryParam("type") != "" {
nodeType = meshcore.NodeType(getQueryInt(ctx, "type"))
}
rows, err := server.db.Query(sqlSelectMeshCoreNodesLastPosition, nodeType, 25)
if err != nil {
return server.apiError(ctx, err)
}
var (
response []meshCoreNodeResponse
prefix []byte
)
for rows.Next() {
var (
row meshCoreNodeResponse
lat, lng *float64
)
if err := rows.Scan(
&row.SNR,
&row.RSSI,
&row.Name,
&row.PublicKey,
&prefix,
&row.NodeType,
&row.FirstHeard,
&row.LastHeard,
&lat,
&lng,
); err != nil {
return server.apiError(ctx, err)
}
if lat != nil && lng != nil {
row.Position = &meshcore.Position{
Latitude: *lat,
Longitude: *lng,
}
}
if len(prefix) > 0 {
row.Prefix = prefix[0]
}
response = append(response, row)
}
return ctx.JSON(http.StatusOK, response)
}
type meshCorePacketResponse struct {
SNR float64 `json:"snr"`
RSSI int8 `json:"rssi"`
Hash []byte `json:"hash"`
RouteType byte `json:"route_type"`
PayloadType byte `json:"payload_type"`
Path []byte `json:"path"`
ReceivedAt time.Time `json:"received_at"`
Raw []byte `json:"raw"`
Parsed []byte `json:"parsed"`
}
func (server *Server) apiGetMeshCorePackets(ctx echo.Context) error {
var (
query string
limit = 25
args []any
)
if hashParam := ctx.QueryParam("hash"); hashParam != "" {
var (
hash []byte
err error
)
switch len(hashParam) {
case base64.URLEncoding.EncodedLen(8):
hash, err = base64.URLEncoding.DecodeString(hashParam)
case hex.EncodedLen(8):
hash, err = hex.DecodeString(hashParam)
default:
err = errors.New("invalid encoding")
}
if err != nil {
return server.apiError(ctx, err, http.StatusBadRequest)
}
query = sqlSelectMeshCorePacketsByHash
args = []any{hash}
} else {
query = sqlSelectMeshCorePackets
args = []any{limit}
}
rows, err := server.db.Query(query, args...)
if err != nil {
return server.apiError(ctx, err)
}
var response []meshCorePacketResponse
for rows.Next() {
var row meshCorePacketResponse
if err := rows.Scan(
&row.SNR,
&row.RSSI,
&row.Hash,
&row.RouteType,
&row.PayloadType,
&row.Path,
&row.ReceivedAt,
&row.Raw,
&row.Parsed,
); err != nil {
return server.apiError(ctx, err)
}
response = append(response, row)
}
return ctx.JSON(http.StatusOK, response)
}
type meshCorePathResponse struct {
Origin *meshCoreNode `json:"origin"`
Path []*meshCoreNodeDistance `json:"path"`
}
func (server *Server) apiGetMeshCorePath(ctx echo.Context) error {
origin, err := hex.DecodeString(ctx.Param("origin"))
if err != nil || len(origin) != crypto.PublicKeySize {
return ctx.JSON(http.StatusBadRequest, map[string]any{
"error": "invalid origin",
})
}
path, err := hex.DecodeString(ctx.Param("path"))
if err != nil || len(path) == 0 {
return ctx.JSON(http.StatusBadRequest, map[string]any{
"error": "invalid path",
})
}
var (
node meshCoreNodeDistance
prefix []byte
latitude, longitude *float64
)
if err := server.db.QueryRow(`
SELECT
n.name,
n.public_key,
n.prefix,
n.first_heard,
n.last_heard,
n.last_latitude,
n.last_longitude
FROM
meshcore_node n
WHERE
n.node_type = 2 AND
n.public_key = $1
`,
origin,
).Scan(
&node.Name,
&node.PublicKey,
&prefix,
&node.FirstHeard,
&node.LastHeard,
&latitude,
&longitude,
); err != nil {
return server.apiError(ctx, err)
}
node.Prefix = MeshCorePrefix(prefix[0])
if latitude == nil || longitude == nil {
return ctx.JSON(http.StatusNotFound, map[string]any{
"error": "origin has no known position",
})
}
node.Position = &meshcore.Position{
Latitude: *latitude,
Longitude: *longitude,
}
var (
current = &node
trace []*meshCoreNodeDistance
)
slices.Reverse(path)
for _, prefix := range path {
if prefix != byte(current.Prefix) {
var hop *meshCoreNodeDistance
if hop, err = meshCoreRepeaterWithPrefixCloseTo(server.db, MeshCorePrefix(prefix), current.Position); err != nil {
if !os.IsNotExist(err) {
return server.apiError(ctx, err)
}
current = &meshCoreNodeDistance{
meshCoreNode: meshCoreNode{
Prefix: MeshCorePrefix(prefix),
Position: current.Position,
},
}
} else {
current = hop
}
}
trace = append(trace, current)
}
/*
if path[len(path)-1] == node.Prefix {
path = path[:len(path)-2]
}
*/
var response = meshCorePathResponse{
Origin: &node.meshCoreNode,
Path: trace,
}
return ctx.JSON(http.StatusOK, response)
}
type meshCoreSourcesResponse struct {
Window time.Time `json:"time"`
Packets map[string]int `json:"packets"`
}
func (server *Server) apiGetMeshCoreSources(ctx echo.Context) error {
var (
now = time.Now().UTC()
windows = map[string]struct {
Interval int
Since time.Duration
}{
"24h": {900, time.Hour * 24},
"1w": {3600, time.Hour * 24 * 7},
}
window = ctx.QueryParam("window")
)
if window == "" {
window = "24h"
}
params, ok := windows[window]
if !ok {
return server.apiError(ctx, os.ErrNotExist)
}
rows, err := server.db.Query(sqlSelectMeshCorePacketsByRepeaterWindowed, params.Interval, now.Add(-params.Since))
if err != nil {
return server.apiError(ctx, err)
}
var (
response []*meshCoreSourcesResponse
buckets = make(map[int64]*meshCoreSourcesResponse)
)
for rows.Next() {
var result struct {
Window time.Time
Repeater string
Packets int
}
if err := rows.Scan(&result.Window, &result.Repeater, &result.Packets); err != nil {
return server.apiError(ctx, err)
}
if result.Packets <= 10 {
continue // ignore
}
if bucket, ok := buckets[result.Window.Unix()]; ok {
bucket.Packets[result.Repeater] = result.Packets
} else {
bucket = &meshCoreSourcesResponse{
Window: result.Window,
Packets: map[string]int{
result.Repeater: result.Packets,
},
}
response = append(response, bucket)
buckets[result.Window.Unix()] = bucket
}
}
return ctx.JSON(http.StatusOK, response)
}
func getQueryInt(ctx echo.Context, param string) int {
v, _ := strconv.Atoi(ctx.QueryParam(param))
return v
}
func getQueryInts(ctx echo.Context, param string) []int {
var values []int
if keys := strings.Split(ctx.QueryParam(param), ","); len(keys) > 0 {
for _, value := range keys {
if v, err := strconv.Atoi(value); err == nil {
values = append(values, v)
}
}
}
return values
}