gus/main.go

255 lines
5.5 KiB
Go

// Copyright 2017-2025 WIT.COM Inc. All rights reserved.
// Use of this source code is governed by the GPL 3.0
package main
import (
"bufio"
"embed"
"fmt"
"io"
"net"
"os"
"strings"
sync "sync"
"time"
"github.com/google/uuid"
"go.wit.com/dev/alexflint/arg"
"go.wit.com/gui"
"go.wit.com/log"
timestamppb "google.golang.org/protobuf/types/known/timestamppb"
)
var VERSION string
var BUILDTIME string
//go:embed resources/*
var resources embed.FS
func main() {
var pp *arg.Parser
gui.InitArg()
pp = arg.MustParse(&argv)
if pp == nil {
pp.WriteHelp(os.Stdout)
os.Exit(0)
}
log.Info("tmp hack", uuid.New().String())
me = new(gusconf)
me.pollDelay = 10 * time.Second
me.portmaps = ConfigLoad()
me.events = EventLoad()
if me.portmaps == nil {
me.portmaps = NewPortmaps()
p := new(Portmap)
p.Dest = "testing:323"
me.portmaps.Append(p)
}
if argv.Daemon {
// turn off timestamps for STDOUT (systemd adds them)
log.DaemonMode(true)
startGus()
startHTTP()
os.Exit(0)
}
if gui.NoGui() {
startGus()
startHTTP()
os.Exit(0)
}
startGus()
// go NewWatchdog()
go startHTTP()
doGui()
}
func startGus() {
all := me.portmaps.All()
for all.Scan() {
pm := all.Next()
if !pm.Enabled {
continue
}
log.Info("portmap enabled for port", pm.Localport, "to", pm.Dest)
go gus3000(pm)
}
}
// func doME(pm *Portmap, gus listener.Accept) {
func doME(pm *Portmap, gus net.Listener) {
localport := int(pm.Localport)
where := pm.Dest
/*
// Listen on local port 3000
s := fmt.Sprintf("0.0.0.0:%d", port)
listener, err := net.Listen("tcp", s)
if err != nil {
log.Fatalf("Failed to listen on %s: %v", s, err)
}
defer listener.Close()
log.Info("Listening on ", s)
*/
for {
// Accept incoming connection
src, err := gus.Accept()
if err != nil {
log.Printf("Failed to accept client connection: %v\n", err)
return
}
// make a new event from this new connection
log.Printf("/me Connected on port %d from client: %s to where = %s\n", localport, src.RemoteAddr(), where)
fmt.Fprintln(src, "/me hello")
reader := bufio.NewReader(src)
// Read one line
line, err := reader.ReadString('\n')
if err != nil {
log.Info("Error reading line:", err)
return
}
log.Info("gus got Received:", line)
parts := strings.Fields(line)
if len(parts) != 3 {
return
}
if parts[0] != "/me" {
return
}
if parts[1] != "hostname" {
return
}
hostname := parts[2]
msg := fmt.Sprintf("got hostname %s for %s", hostname, src.RemoteAddr())
log.Info("gus:", msg)
fmt.Fprintln(src, msg)
if hostname == "framebook.wit.com" {
// Handle the connection in a separate goroutine
go handleConnection(src, pm.Dest, int(pm.Localport))
}
}
}
func gus3000(pm *Portmap) {
port := int(pm.Localport)
connect := pm.Dest
// Listen on local port 3000
s := fmt.Sprintf("0.0.0.0:%d", port)
listener, err := net.Listen("tcp", s)
if err != nil {
log.Fatalf("Failed to listen on %s: %v", s, err)
}
defer listener.Close()
log.Info("Listening on ", s)
if pm.UseME {
doME(pm, listener)
return
}
for {
// Accept incoming connection
clientConn, err := listener.Accept()
if err != nil {
log.Printf("Failed to accept client connection: %v", err)
continue
}
// log.Printf("Client connected: %s", clientConn.RemoteAddr())
// Handle the connection in a separate goroutine
go handleConnection(clientConn, connect, port)
}
}
func handleConnection(clientConn net.Conn, where string, localport int) {
defer clientConn.Close()
// Connect to the target server
// targetConn, err := net.Dial("tcp", "go.wit.com:80")
targetConn, err := net.Dial("tcp", where)
if err != nil {
log.Printf("Failed to connect to %s %v", where, err)
return
}
defer targetConn.Close()
// make a new event from this new connection
log.Printf("Connected on port %d from client: %s to where = %s\n", localport, clientConn.RemoteAddr(), where)
e := new(Event)
e.Etype = GusEventType_Connect
e.LocalPort = int64(localport)
e.Sock = new(GusSocket)
e.Sock.SrcIp = fmt.Sprintf("%s", clientConn.RemoteAddr())
e.Sock.DestIp = where
e.Ctime = timestamppb.New(time.Now())
me.events.Append(e)
me.eventsChanged = true
// Bidirectional copy of data
// go io.Copy(targetConn, clientConn) // Client -> Target
// io.Copy(clientConn, targetConn) // Target -> Client
ioCopy(clientConn, targetConn)
// if the socket closes, record the close time
e.Etime = timestamppb.New(time.Now())
me.eventsChanged = true
log.Printf("Connection closed on port %d from client: %s to where = %s\n", localport, clientConn.RemoteAddr(), where)
}
func enablePort(port int, dest string) {
var found bool
all := me.portmaps.All()
for all.Scan() {
pm := all.Next()
if int(pm.Localport) == port {
found = true
log.Info("Found port!", port)
if pm.Enabled {
log.Info("portmap already enabled for", pm.Localport, "to", pm.Dest)
} else {
log.Info("portmap not enabled for", pm.Localport, "to", pm.Dest)
}
}
// go gus3000(pm)
}
if !found {
log.Info("Did not find port =", port)
}
}
// should cleanly close both
func ioCopy(clientConn, targetConn net.Conn) {
defer clientConn.Close()
defer targetConn.Close()
var wg sync.WaitGroup
wg.Add(2)
// Copy data in both directions
go func() {
defer wg.Done()
io.Copy(targetConn, clientConn) // Client -> Target
targetConn.Close() // Ensure closure on EOF/error
}()
go func() {
defer wg.Done()
io.Copy(clientConn, targetConn) // Target -> Client
clientConn.Close() // Ensure closure on EOF/error
}()
wg.Wait() // Wait for both copies to complete
}