You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

182 lines
4.6 KiB
Go

package iperf
import (
"context"
"fmt"
"log"
"net"
"sync"
"time"
api "github.com/BGrewell/go-iperf/api/go"
"google.golang.org/grpc"
)
func NewController(port int) (controller *Controller, err error) {
c := &Controller{
Port: port,
clients: make(map[string]*Client),
servers: make(map[string]*Server),
clientLock: sync.Mutex{},
serverLock: sync.Mutex{},
}
err = c.startListener()
return c, err
}
// Controller is a helper in the go-iperf package that is designed to run on both the client and the server side. On the
// server side it listens for new gRPC connections, when a connection is made by a client the client can tell it to
// start a new iperf server instance. It will start a instance on an unused port and return the port number to the
// client. This allows the entire iperf setup and session to be performed from the client side.
//
// CLIENT SERVER
// connect to grpc ---> accept grpc connection
// call StartServer() ---> find unused port
// start iperf server on port
// get server port info <--- return port information to client
type Controller struct {
api.UnimplementedCommandServer
Port int
cmdClient api.CommandClient
clientLock sync.Mutex
serverLock sync.Mutex
clients map[string]*Client
servers map[string]*Server
}
// StartServer is the handler for the gRPC function StartServer()
func (c *Controller) GrpcRequestServer(context.Context, *api.StartServerRequest) (*api.StartServerResponse, error) {
srv, err := c.NewServer()
srv.SetOneOff(true)
if err != nil {
return nil, err
}
err = srv.Start()
if err != nil {
return nil, err
}
c.serverLock.Lock()
c.servers[srv.Id] = srv
c.serverLock.Unlock()
reply := &api.StartServerResponse{
Id: srv.Id,
ListenPort: int32(srv.Port()),
}
return reply, nil
}
// StartListener starts a command listener which is used to accept gRPC connections from another go-iperf controller
func (c *Controller) startListener() (err error) {
listener, err := net.Listen("tcp", fmt.Sprintf("%s:%d", "0.0.0.0", c.Port))
if err != nil {
return err
}
gs := grpc.NewServer()
api.RegisterCommandServer(gs, c)
go func() {
err := gs.Serve(listener)
if err != nil {
log.Fatal(err)
}
}()
time.Sleep(250 * time.Millisecond)
return nil
}
// NewServer gets a new instance of an iperf server on a free port
func (c *Controller) NewServer() (server *Server, err error) {
freePort, err := GetUnusedTcpPort()
s := NewServer()
s.SetPort(freePort)
c.serverLock.Lock()
c.servers[s.Id] = s
c.serverLock.Unlock()
return s, nil
}
// StopServer shuts down an iperf server and frees any actively used resources
func (c *Controller) StopServer(id string) (err error) {
c.serverLock.Lock()
delete(c.servers, id)
c.serverLock.Unlock()
return nil
}
// NewClient gets a new instance of an iperf client and also starts up a matched iperf server instance on the specified
// serverAddr. If it fails to connect to the gRPC interface of the controller on the remote side it will return an error
func (c *Controller) NewClient(serverAddr string) (client *Client, err error) {
grpc, err := GetConnectedClient(serverAddr, c.Port)
if err != nil {
return nil, err
}
ctx, cancel := context.WithTimeout(context.Background(), 2 * time.Second)
defer cancel()
reply, err := grpc.GrpcRequestServer(ctx, &api.StartServerRequest{})
srvPort := int(reply.ListenPort)
fmt.Printf("[!] server is listening on port %d\n", srvPort)
cli := NewClient(serverAddr)
cli.SetPort(srvPort)
c.clientLock.Lock()
c.clients[cli.Id] = cli
c.clientLock.Unlock()
return cli, nil
}
// StopClient will clean up the server side connection and shut down any actively used resources
func (c *Controller) StopClient(id string) (err error) {
c.clientLock.Lock()
delete(c.clients, id)
c.clientLock.Unlock()
return nil
}
func GetConnectedClient(addr string, port int) (client api.CommandClient, err error) {
conn, err := grpc.Dial(fmt.Sprintf("%s:%d", addr, port), grpc.WithInsecure(), grpc.WithBlock(), grpc.WithTimeout(time.Second*2))
if err != nil {
return nil, err
}
client = api.NewCommandClient(conn)
return client, nil
}
func GetUnusedTcpPort() (int, error) {
addr, err := net.ResolveTCPAddr("tcp", "localhost:0")
if err != nil {
return 0, err
}
l, err := net.ListenTCP("tcp", addr)
if err != nil {
return 0, err
}
defer l.Close()
return l.Addr().(*net.TCPAddr).Port, nil
}
//func GetUnusedUdpPort() (int, error) {
// addr, err := net.ResolveUDPAddr("udp", "localhost:0")
// if err != nil {
// return 0, err
// }
//
// l, err := net.ListenUDP("udp", addr)
// if err != nil {
// return 0, err
// }
//
// defer l.Close()
// return l.LocalAddr().(*net.UDPAddr).Port, nil
//}