smegmesh/pkg/query/query.go

111 lines
2.9 KiB
Go
Raw Normal View History

2023-11-01 14:17:58 +01:00
package query
import (
"encoding/json"
"fmt"
"strings"
2023-11-01 14:17:58 +01:00
"github.com/jmespath/go-jmespath"
"github.com/tim-beatham/smegmesh/pkg/conf"
"github.com/tim-beatham/smegmesh/pkg/lib"
"github.com/tim-beatham/smegmesh/pkg/mesh"
2023-11-01 14:17:58 +01:00
)
// Querier queries a data store for the given data
// and returns data in the corresponding encoding
type Querier interface {
Query(meshId string, queryParams string) ([]byte, error)
}
// JmesQuerier: queries the datstore in JMESPath syntax
2023-11-01 14:17:58 +01:00
type JmesQuerier struct {
2023-11-05 19:03:58 +01:00
manager mesh.MeshManager
2023-11-01 14:17:58 +01:00
}
// QueryError: query error if something went wrong
2023-11-01 14:17:58 +01:00
type QueryError struct {
msg string
}
// QuerRoute: represents a route in the query
type QueryRoute struct {
Destination string `json:"destination"`
HopCount int `json:"hopCount"`
Path string `json:"path"`
}
// QueryNode: represents a single node in the query
2023-11-01 14:17:58 +01:00
type QueryNode struct {
HostEndpoint string `json:"hostEndpoint"`
PublicKey string `json:"publicKey"`
WgEndpoint string `json:"wgEndpoint"`
WgHost string `json:"wgHost"`
Timestamp int64 `json:"timestamp"`
Description string `json:"description"`
Routes []QueryRoute `json:"routes"`
Alias string `json:"alias"`
Services map[string]string `json:"services"`
Type conf.NodeType `json:"type"`
2023-11-01 14:17:58 +01:00
}
func (m *QueryError) Error() string {
return m.msg
}
// Query: queries the the datastore at the given meshid
2023-11-01 14:17:58 +01:00
func (j *JmesQuerier) Query(meshId, queryParams string) ([]byte, error) {
2023-11-05 19:03:58 +01:00
mesh, ok := j.manager.GetMeshes()[meshId]
2023-11-01 14:17:58 +01:00
if !ok {
return nil, &QueryError{msg: fmt.Sprintf("%s does not exist", meshId)}
}
snapshot, err := mesh.GetMesh()
if err != nil {
return nil, err
}
nodes := lib.Map(lib.MapValues(snapshot.GetNodes()), MeshNodeToQueryNode)
2023-11-01 14:17:58 +01:00
result, err := jmespath.Search(queryParams, nodes)
if err != nil {
return nil, err
}
bytes, err := json.Marshal(result)
return bytes, err
}
// MeshNodeToQuerynode: convert the mesh node into a query abstraction
func MeshNodeToQueryNode(node mesh.MeshNode) *QueryNode {
2023-11-01 14:17:58 +01:00
queryNode := new(QueryNode)
queryNode.HostEndpoint = node.GetHostEndpoint()
pubKey, _ := node.GetPublicKey()
queryNode.PublicKey = pubKey.String()
queryNode.WgEndpoint = node.GetWgEndpoint()
queryNode.WgHost = node.GetWgHost().String()
queryNode.Timestamp = node.GetTimeStamp()
queryNode.Routes = lib.Map(node.GetRoutes(), func(r mesh.Route) QueryRoute {
return QueryRoute{
Destination: r.GetDestination().String(),
HopCount: r.GetHopCount(),
Path: strings.Join(r.GetPath(), ","),
}
})
2023-11-01 14:17:58 +01:00
queryNode.Description = node.GetDescription()
queryNode.Alias = node.GetAlias()
queryNode.Services = node.GetServices()
queryNode.Type = node.GetType()
2023-11-01 14:17:58 +01:00
return queryNode
}
2023-11-05 19:03:58 +01:00
func NewJmesQuerier(manager mesh.MeshManager) Querier {
2023-11-01 14:17:58 +01:00
return &JmesQuerier{manager: manager}
}