2023-11-13 11:44:14 +01:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/tim-beatham/wgmesh/pkg/ctrlserver"
|
|
|
|
"github.com/tim-beatham/wgmesh/pkg/ipc"
|
|
|
|
logging "github.com/tim-beatham/wgmesh/pkg/log"
|
2023-11-20 19:07:52 +01:00
|
|
|
"github.com/tim-beatham/wgmesh/pkg/what8words"
|
2023-11-13 11:44:14 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type ApiServer interface {
|
|
|
|
GetMeshes(c *gin.Context)
|
|
|
|
Run(addr string) error
|
|
|
|
}
|
|
|
|
|
|
|
|
type SmegServer struct {
|
|
|
|
router *gin.Engine
|
2023-12-31 15:25:06 +01:00
|
|
|
client *ipc.ClientIpc
|
2023-11-20 19:07:52 +01:00
|
|
|
words *what8words.What8Words
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
func (s *SmegServer) routeToApiRoute(meshNode ctrlserver.MeshNode) []Route {
|
|
|
|
routes := make([]Route, len(meshNode.Routes))
|
|
|
|
|
|
|
|
for index, route := range meshNode.Routes {
|
|
|
|
|
2023-11-27 19:55:41 +01:00
|
|
|
if route.Path == nil {
|
|
|
|
route.Path = make([]string, 0)
|
2023-11-20 19:07:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
routes[index] = Route{
|
2023-11-27 19:55:41 +01:00
|
|
|
Prefix: route.Destination,
|
|
|
|
Path: route.Path,
|
2023-11-20 19:07:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return routes
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SmegServer) meshNodeToAPIMeshNode(meshNode ctrlserver.MeshNode) *SmegNode {
|
2023-11-13 11:44:14 +01:00
|
|
|
if meshNode.Routes == nil {
|
2023-11-27 19:55:41 +01:00
|
|
|
meshNode.Routes = make([]ctrlserver.MeshRoute, 0)
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
alias := meshNode.Alias
|
|
|
|
|
|
|
|
if alias == "" {
|
|
|
|
alias, _ = s.words.ConvertIdentifier(meshNode.WgHost)
|
|
|
|
}
|
|
|
|
|
2023-11-13 11:44:14 +01:00
|
|
|
return &SmegNode{
|
|
|
|
WgHost: meshNode.WgHost,
|
|
|
|
WgEndpoint: meshNode.WgEndpoint,
|
|
|
|
Endpoint: meshNode.HostEndpoint,
|
|
|
|
Timestamp: int(meshNode.Timestamp),
|
|
|
|
Description: meshNode.Description,
|
2023-11-20 19:07:52 +01:00
|
|
|
Routes: s.routeToApiRoute(meshNode),
|
2023-11-13 11:44:14 +01:00
|
|
|
PublicKey: meshNode.PublicKey,
|
2023-11-20 19:07:52 +01:00
|
|
|
Alias: alias,
|
2023-11-17 23:13:51 +01:00
|
|
|
Services: meshNode.Services,
|
2023-12-11 10:55:25 +01:00
|
|
|
Stats: SmegStats{
|
|
|
|
TotalTransmit: meshNode.Stats.TransmitBytes,
|
|
|
|
TotalReceived: meshNode.Stats.ReceivedBytes,
|
|
|
|
KeepAliveInterval: meshNode.Stats.PersistentKeepAliveInterval,
|
|
|
|
AllowedIps: meshNode.Stats.AllowedIPs,
|
|
|
|
},
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
func (s *SmegServer) meshToAPIMesh(meshId string, nodes []ctrlserver.MeshNode) SmegMesh {
|
2023-11-13 11:44:14 +01:00
|
|
|
var smegMesh SmegMesh
|
|
|
|
smegMesh.MeshId = meshId
|
|
|
|
smegMesh.Nodes = make(map[string]SmegNode)
|
|
|
|
|
|
|
|
for _, node := range nodes {
|
2023-11-20 19:07:52 +01:00
|
|
|
smegMesh.Nodes[node.WgHost] = *s.meshNodeToAPIMeshNode(node)
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return smegMesh
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateMesh: creates a mesh network
|
|
|
|
func (s *SmegServer) CreateMesh(c *gin.Context) {
|
|
|
|
var createMesh CreateMeshRequest
|
|
|
|
|
|
|
|
if err := c.ShouldBindJSON(&createMesh); err != nil {
|
|
|
|
c.JSON(http.StatusBadRequest, &gin.H{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
2023-11-20 14:03:42 +01:00
|
|
|
|
2023-11-13 11:44:14 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ipcRequest := ipc.NewMeshArgs{
|
2023-12-12 12:58:47 +01:00
|
|
|
WgArgs: ipc.WireGuardArgs{
|
|
|
|
WgPort: createMesh.WgPort,
|
|
|
|
},
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var reply string
|
|
|
|
|
2023-12-31 15:25:06 +01:00
|
|
|
err := s.client.CreateMesh(&ipcRequest, &reply)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
c.JSON(http.StatusBadRequest, &gin.H{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, &gin.H{
|
|
|
|
"meshid": reply,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// JoinMesh: joins a mesh network
|
|
|
|
func (s *SmegServer) JoinMesh(c *gin.Context) {
|
|
|
|
var joinMesh JoinMeshRequest
|
|
|
|
|
|
|
|
if err := c.ShouldBindJSON(&joinMesh); err != nil {
|
|
|
|
c.JSON(http.StatusBadRequest, &gin.H{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ipcRequest := ipc.JoinMeshArgs{
|
2023-12-31 15:25:06 +01:00
|
|
|
MeshId: joinMesh.MeshId,
|
|
|
|
IpAddress: joinMesh.Bootstrap,
|
2023-12-12 12:58:47 +01:00
|
|
|
WgArgs: ipc.WireGuardArgs{
|
|
|
|
WgPort: joinMesh.WgPort,
|
|
|
|
},
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var reply string
|
|
|
|
|
2023-12-31 15:25:06 +01:00
|
|
|
err := s.client.JoinMesh(ipcRequest, &reply)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
c.JSON(http.StatusBadRequest, &gin.H{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, &gin.H{
|
|
|
|
"status": "success",
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMesh: given a meshId returns the corresponding mesh
|
|
|
|
// network.
|
|
|
|
func (s *SmegServer) GetMesh(c *gin.Context) {
|
|
|
|
meshidParam := c.Param("meshid")
|
|
|
|
|
|
|
|
var meshid string = meshidParam
|
|
|
|
|
|
|
|
getMeshReply := new(ipc.GetMeshReply)
|
|
|
|
|
2023-12-31 15:25:06 +01:00
|
|
|
err := s.client.GetMesh(meshid, getMeshReply)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
c.JSON(http.StatusNotFound,
|
|
|
|
&gin.H{
|
|
|
|
"error": fmt.Sprintf("could not find mesh %s", meshidParam),
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
mesh := s.meshToAPIMesh(meshidParam, getMeshReply.Nodes)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
c.JSON(http.StatusOK, mesh)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SmegServer) GetMeshes(c *gin.Context) {
|
|
|
|
listMeshesReply := new(ipc.ListMeshReply)
|
|
|
|
|
2023-12-31 15:25:06 +01:00
|
|
|
err := s.client.ListMeshes(listMeshesReply)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logging.Log.WriteErrorf(err.Error())
|
|
|
|
c.JSON(http.StatusBadRequest, nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
meshes := make([]SmegMesh, 0)
|
|
|
|
|
|
|
|
for _, mesh := range listMeshesReply.Meshes {
|
|
|
|
getMeshReply := new(ipc.GetMeshReply)
|
|
|
|
|
2023-12-31 15:25:06 +01:00
|
|
|
err := s.client.GetMesh(mesh, getMeshReply)
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logging.Log.WriteErrorf(err.Error())
|
|
|
|
c.JSON(http.StatusBadRequest, nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
meshes = append(meshes, s.meshToAPIMesh(mesh, getMeshReply.Nodes))
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, meshes)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SmegServer) Run(addr string) error {
|
|
|
|
logging.Log.WriteInfof("Running API server")
|
|
|
|
return s.router.Run(addr)
|
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
func NewSmegServer(conf ApiServerConf) (ApiServer, error) {
|
2023-12-31 15:25:06 +01:00
|
|
|
client, err := ipc.NewClientIpc()
|
2023-11-13 11:44:14 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-20 19:07:52 +01:00
|
|
|
words, err := what8words.NewWhat8Words(conf.WordsFile)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-13 11:44:14 +01:00
|
|
|
router := gin.Default()
|
|
|
|
|
|
|
|
router.Use(gin.LoggerWithConfig(gin.LoggerConfig{
|
|
|
|
Output: logging.Log.Writer(),
|
|
|
|
}))
|
|
|
|
|
|
|
|
smegServer := &SmegServer{
|
|
|
|
router: router,
|
|
|
|
client: client,
|
2023-11-20 19:07:52 +01:00
|
|
|
words: words,
|
2023-11-13 11:44:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
router.GET("/meshes", smegServer.GetMeshes)
|
|
|
|
router.GET("/mesh/:meshid", smegServer.GetMesh)
|
|
|
|
router.POST("/mesh/create", smegServer.CreateMesh)
|
|
|
|
router.POST("/mesh/join", smegServer.JoinMesh)
|
|
|
|
return smegServer, nil
|
|
|
|
}
|