import ../common/[types, utils] # Define function prototypes proc executePwd(ctx: AgentCtx, task: Task): TaskResult proc executeCd(ctx: AgentCtx, task: Task): TaskResult proc executeDir(ctx: AgentCtx, task: Task): TaskResult proc executeRm(ctx: AgentCtx, task: Task): TaskResult proc executeRmdir(ctx: AgentCtx, task: Task): TaskResult proc executeMove(ctx: AgentCtx, task: Task): TaskResult proc executeCopy(ctx: AgentCtx, task: Task): TaskResult # Module definition let module* = Module( name: protect("filesystem"), description: protect("Conduct simple filesystem operations via Windows API."), moduleType: MODULE_FILESYSTEM, commands: @[ Command( name: protect("pwd"), commandType: CMD_PWD, description: protect("Retrieve current working directory."), example: protect("pwd"), arguments: @[], execute: executePwd ), Command( name: protect("cd"), commandType: CMD_CD, description: protect("Change current working directory."), example: protect("cd C:\\Windows\\Tasks"), arguments: @[ Argument(name: protect("directory"), description: protect("Relative or absolute path of the directory to change to."), argumentType: STRING, isRequired: true) ], execute: executeCd ), Command( name: protect("ls"), commandType: CMD_LS, description: protect("List files and directories."), example: protect("ls C:\\Users\\Administrator\\Desktop"), arguments: @[ Argument(name: protect("directory"), description: protect("Relative or absolute path. Default: current working directory."), argumentType: STRING, isRequired: false) ], execute: executeDir ), Command( name: protect("rm"), commandType: CMD_RM, description: protect("Remove a file."), example: protect("rm C:\\Windows\\Tasks\\payload.exe"), arguments: @[ Argument(name: protect("file"), description: protect("Relative or absolute path to the file to delete."), argumentType: STRING, isRequired: true) ], execute: executeRm ), Command( name: protect("rmdir"), commandType: CMD_RMDIR, description: protect("Remove a directory."), example: protect("rm C:\\Payloads"), arguments: @[ Argument(name: protect("directory"), description: protect("Relative or absolute path to the directory to delete."), argumentType: STRING, isRequired: true) ], execute: executeRmdir ), Command( name: protect("move"), commandType: CMD_MOVE, description: protect("Move a file or directory."), example: protect("move source.exe C:\\Windows\\Tasks\\destination.exe"), arguments: @[ Argument(name: protect("source"), description: protect("Source file path."), argumentType: STRING, isRequired: true), Argument(name: protect("destination"), description: protect("Destination file path."), argumentType: STRING, isRequired: true) ], execute: executeMove ), Command( name: protect("copy"), commandType: CMD_COPY, description: protect("Copy a file or directory."), example: protect("copy source.exe C:\\Windows\\Tasks\\destination.exe"), arguments: @[ Argument(name: protect("source"), description: protect("Source file path."), argumentType: STRING, isRequired: true), Argument(name: protect("destination"), description: protect("Destination file path."), argumentType: STRING, isRequired: true) ], execute: executeCopy ) ] ) # Implementation of the execution functions when not defined(agent): proc executePwd(ctx: AgentCtx, task: Task): TaskResult = nil proc executeCd(ctx: AgentCtx, task: Task): TaskResult = nil proc executeDir(ctx: AgentCtx, task: Task): TaskResult = nil proc executeRm(ctx: AgentCtx, task: Task): TaskResult = nil proc executeRmdir(ctx: AgentCtx, task: Task): TaskResult = nil proc executeMove(ctx: AgentCtx, task: Task): TaskResult = nil proc executeCopy(ctx: AgentCtx, task: Task): TaskResult = nil when defined(agent): import os, strutils, strformat, times, algorithm, winim import ../agent/core/io import ../agent/protocol/result import ../common/utils # Retrieve current working directory proc executePwd(ctx: AgentCtx, task: Task): TaskResult = print " [>] Retrieving current working directory." try: # Get current working directory using GetCurrentDirectory let buffer = newWString(MAX_PATH + 1) length = GetCurrentDirectoryW(MAX_PATH, &buffer) if length == 0: raise newException(CatchableError, GetLastError().getError()) let output = $buffer[0 ..< (int)length] return createTaskResult(task, STATUS_COMPLETED, RESULT_STRING, string.toBytes(output)) except CatchableError as err: return createTaskResult(task, STATUS_FAILED, RESULT_STRING, string.toBytes(err.msg)) # Change working directory proc executeCd(ctx: AgentCtx, task: Task): TaskResult = # Parse arguments let targetDirectory = Bytes.toString(task.args[0].data) print " [>] Changing current working directory to {targetDirectory}." try: # Get current working directory using GetCurrentDirectory if SetCurrentDirectoryW(targetDirectory) == FALSE: raise newException(CatchableError, GetLastError().getError()) return createTaskResult(task, STATUS_COMPLETED, RESULT_NO_OUTPUT, @[]) except CatchableError as err: return createTaskResult(task, STATUS_FAILED, RESULT_STRING, string.toBytes(err.msg)) # List files and directories at a specific or at the current path proc executeDir(ctx: AgentCtx, task: Task): TaskResult = try: var targetDirectory: string # Parse arguments case int(task.argCount): of 0: # Get current working directory using GetCurrentDirectory let cwdBuffer = newWString(MAX_PATH + 1) cwdLength = GetCurrentDirectoryW(MAX_PATH, &cwdBuffer) if cwdLength == 0: raise newException(CatchableError, GetLastError().getError()) targetDirectory = $cwdBuffer[0 ..< (int)cwdLength] of 1: targetDirectory = Bytes.toString(task.args[0].data) else: discard print fmt" [>] Listing files and directories in {targetDirectory}." # Prepare search pattern (target directory + \*) let searchPattern = targetDirectory & "\\*" let searchPatternW = newWString(searchPattern) var findData: WIN32_FIND_DATAW hFind: HANDLE output = "" entries: seq[string] = @[] totalFiles = 0 totalDirs = 0 # Find files and directories in target directory hFind = FindFirstFileW(searchPatternW, &findData) if hFind == INVALID_HANDLE_VALUE: raise newException(CatchableError, GetLastError().getError()) # Directory was found and can be listed else: output = fmt"Directory: {targetDirectory}" & "\n\n" output &= "Mode LastWriteTime Length Name" & "\n" output &= "---- ------------- ------ ----" & "\n" # Process all files and directories while true: let fileName = $cast[WideCString](addr findData.cFileName[0]) # Skip current and parent directory entries if fileName != "." and fileName != "..": # Get file attributes and size let isDir = (findData.dwFileAttributes and FILE_ATTRIBUTE_DIRECTORY) != 0 let isHidden = (findData.dwFileAttributes and FILE_ATTRIBUTE_HIDDEN) != 0 let isReadOnly = (findData.dwFileAttributes and FILE_ATTRIBUTE_READONLY) != 0 let isArchive = (findData.dwFileAttributes and FILE_ATTRIBUTE_ARCHIVE) != 0 let fileSize = (int64(findData.nFileSizeHigh) shl 32) or int64(findData.nFileSizeLow) # Handle flags var mode = "" if isDir: mode = "d" inc totalDirs else: mode = "-" inc totalFiles if isArchive: mode &= "a" else: mode &= "-" if isReadOnly: mode &= "r" else: mode &= "-" if isHidden: mode &= "h" else: mode &= "-" if (findData.dwFileAttributes and FILE_ATTRIBUTE_SYSTEM) != 0: mode &= "s" else: mode &= "-" # Convert FILETIME to local time and format var localTime: FILETIME systemTime: SYSTEMTIME dateTimeStr = protect("01/01/1970 00:00:00") if FileTimeToLocalFileTime(&findData.ftLastWriteTime, &localTime) != 0 and FileTimeToSystemTime(&localTime, &systemTime) != 0: # Format date and time in PowerShell style dateTimeStr = fmt"{systemTime.wDay:02d}/{systemTime.wMonth:02d}/{systemTime.wYear} {systemTime.wHour:02d}:{systemTime.wMinute:02d}:{systemTime.wSecond:02d}" # Format file size var sizeStr = "" if isDir: sizeStr = protect("