2026-04-06 00:20:51 -05:00

232 lines
6.1 KiB
Go

package server
import (
"AdaptixServer/core/axscript"
"AdaptixServer/core/connector"
"AdaptixServer/core/database"
"AdaptixServer/core/eventing"
"AdaptixServer/core/extender"
"AdaptixServer/core/profile"
"AdaptixServer/core/utils/logs"
"AdaptixServer/core/utils/safe"
"AdaptixServer/core/utils/token"
"net"
"os"
"time"
"github.com/Adaptix-Framework/axc2"
"github.com/goccy/go-yaml"
)
func NewTeamserver() *Teamserver {
dbms, err := database.NewDatabase(logs.RepoLogsInstance.DbPath)
if err != nil {
logs.Error("", "Failed to create a DBMS: "+err.Error())
return nil
}
broker := NewMessageBroker()
broker.Start()
ts := &Teamserver{
Profile: profile.NewProfile(),
DBMS: dbms,
Broker: broker,
EventManager: eventing.NewEventManager(),
OTPManager: token.NewOTPManager(60*time.Second, 30*time.Second),
listener_configs: safe.NewMap(),
agent_configs: safe.NewMap(),
service_configs: safe.NewMap(),
wm_agent_types: make(map[string]string),
wm_listeners: make(map[string][]string),
notifications: safe.NewSlice(),
Agents: safe.NewMap(),
listeners: safe.NewMap(),
downloads: safe.NewMap(),
tmp_uploads: safe.NewMap(),
terminals: safe.NewMap(),
pivots: safe.NewSlice(),
builders: safe.NewMap(),
}
ts.ScriptManager = axscript.NewScriptManager(ts)
ts.TaskManager = NewTaskManager(ts)
ts.TunnelManager = NewTunnelManager(ts)
ts.Extender = extender.NewExtender(ts)
return ts
}
func (ts *Teamserver) SetProfile(path string) error {
var (
err error
fileContent []byte
)
fileContent, err = os.ReadFile(path)
if err != nil {
return err
}
err = yaml.Unmarshal(fileContent, ts.Profile)
if err != nil {
return err
}
return nil
}
func (ts *Teamserver) RestoreData() {
var (
ok bool
err error
)
ok = ts.DBMS.DatabaseExists()
if !ok {
return
}
logs.Info("", "Restore data from Database...")
/// AGENTS
countAgents := 0
restoreAgents := ts.DBMS.DbAgentAll()
for _, agentData := range restoreAgents {
extenderAgent, err := ts.Extender.ExAgentGetExtender(agentData.Name)
if err != nil {
logs.Warn(" ", "Failed to get extenderAgent for agent %v (%v): %v", agentData.Id, agentData.Name, err.Error())
continue
}
agent := &Agent{
Extender: extenderAgent,
HostedTunnelData: safe.NewSafeQueue(0x1000),
HostedTasks: safe.NewSafeQueue(0x100),
HostedTunnelTasks: safe.NewSafeQueue(0x100),
RunningTasks: safe.NewMap(),
RunningJobs: safe.NewMap(),
PivotParent: nil,
PivotChilds: safe.NewSlice(),
Tick: false,
Active: true,
}
if agentData.Mark == "Terminated" {
agent.Active = false
}
if agentData.Mark == "" {
if !agentData.Async {
agentData.Mark = "Disconnect"
}
}
agent.SetData(agentData)
ts.Agents.Put(agentData.Id, agent)
packet := CreateSpAgentNew(agentData)
ts.TsSyncAllClients(packet)
ts.TsNotifyAgent(true, agentData)
countAgents++
}
logs.Success(" ", "Restored %v agents", countAgents)
/// PIVOT
countPivots := 0
restorePivots := ts.DBMS.DbPivotAll()
for _, restorePivot := range restorePivots {
_ = ts.TsPivotCreate(restorePivot.PivotId, restorePivot.ParentAgentId, restorePivot.ChildAgentId, restorePivot.PivotName, true)
countPivots++
}
logs.Success(" ", "Restored %v pivots", countPivots)
logs.Success(" ", "Restored %v listeners", ts.DBMS.DbTableCount("Listeners"))
logs.Success(" ", "Restored %v screenshots", ts.DBMS.DbTableCount("Screenshots"))
logs.Success(" ", "Restored %v downloads", ts.DBMS.DbTableCount("Downloads"))
logs.Success(" ", "Restored %v credentials", ts.DBMS.DbTableCount("Credentials"))
logs.Success(" ", "Restored %v targets", ts.DBMS.DbTableCount("Targets"))
/// LISTENERS
restoreListeners := ts.DBMS.DbListenerAll()
for _, restoreListener := range restoreListeners {
err = ts.TsListenerStart(restoreListener.ListenerName, restoreListener.ListenerRegName, restoreListener.ListenerConfig, restoreListener.CreateTime, restoreListener.Watermark, restoreListener.CustomData)
if err != nil {
logs.Error("", "Failed to restore listener %s: %s", restoreListener.ListenerName, err.Error())
} else {
value, ok := ts.listeners.Get(restoreListener.ListenerName)
if ok {
listenerData := value.(adaptix.ListenerData)
if restoreListener.ListenerStatus == "Paused" && listenerData.Status == "Listen" {
err = ts.Extender.ExListenerPause(restoreListener.ListenerName)
if err != nil {
logs.Error("", "Failed to pause restored listener %s: %s", restoreListener.ListenerName, err.Error())
} else {
listenerData.Status = "Paused"
ts.listeners.Put(restoreListener.ListenerName, listenerData)
packet := CreateSpListenerEdit(listenerData)
ts.TsSyncAllClients(packet)
}
}
}
}
}
}
func (ts *Teamserver) Start() {
var (
stopped chan bool
err error
)
interfaces, err := net.Interfaces()
if err == nil {
ts.Parameters.Interfaces = append(ts.Parameters.Interfaces, "0.0.0.0")
for _, i := range interfaces {
iAddrs, err := i.Addrs()
if err == nil {
for _, addr := range iAddrs {
ipNet, ok := addr.(*net.IPNet)
if ok {
if ipNet.IP.To4() != nil {
ts.Parameters.Interfaces = append(ts.Parameters.Interfaces, ipNet.IP.String())
}
}
}
}
}
}
if len(ts.Parameters.Interfaces) == 0 {
ts.Parameters.Interfaces = append(ts.Parameters.Interfaces, "0.0.0.0")
ts.Parameters.Interfaces = append(ts.Parameters.Interfaces, "127.0.0.1")
}
ts.AdaptixServer, err = connector.NewTsConnector(ts, *ts.Profile.Server, *ts.Profile.HttpServer)
if err != nil {
logs.Error("", "Failed to init HTTP handler: "+err.Error())
return
}
ts.Extender.LoadPlugins(ts.Profile.Server.Extenders)
ts.TsAxScriptLoadFromProfile()
go ts.AdaptixServer.Start(&stopped)
logs.Success("", "Starting server -> https://%s:%v%s", ts.Profile.Server.Interface, ts.Profile.Server.Port, ts.Profile.Server.Endpoint)
ts.RestoreData()
logs.Success("", "The AdaptixC2 server is ready")
go ts.TsAgentTickUpdate()
<-stopped
logs.Warn("", "Teamserver finished")
os.Exit(0)
}