17.1 隐藏执行CMD命令
本章内容涉及使用Socket API
和CMD
命令行工具实现本地CMD
命令执行、无管道正向CMD
和无管道反向CMD
三种功能。执行本地CMD
实现使用CreateProcess
函数创建一个新的CMD
进程,并将标准输入、输出和错误输出重定向到当前进程的标准输入、输出和错误输出。无管道正向CMD
和无管道反向CMD
使用WSASocket
函数创建TCP
套接字,并将CMD
进程的标准输入、输出和错误输出重定向到套接字的句柄上,通过网络连接实现远程命令执行功能。
首先来实现一个CMD
命令行运行功能,通过使用CreatePipe
创建匿名管道,并使用CreateProcess
函数创建一个新的CMD
进程,然后将标准输入、输出和错误输出重定向到当前进程的标准输入、输出和错误输出。这样就可以通过当前进程的输入输出来执行CMD
命令并获取命令输出结果。
CreatePipe 函数,用于创建一个匿名管道。匿名管道是一种用于进程间通信的机制,允许一个进程将输出数据传输给另一个进程。CreatePipe函数的原型如下:
BOOL CreatePipe( |
参数hReadPipe
和hWritePipe
是指向HANDLE
类型的指针,用于接收创建的管道的读端和写端的句柄。参数lpPipeAttributes
是一个指向SECURITY_ATTRIBUTES
结构体的指针,用于设置管道的安全性。参数nSize
是一个DWORD
类型的值,用于指定管道的缓冲区大小,通常可以设置为0表示使用系统默认值。
创建匿名管道后,可以使用ReadFile
函数从管道的读端读取数据,使用WriteFile
函数将数据写入管道的写端。在使用完管道后,应使用CloseHandle
函数关闭管道的句柄,以释放资源。
CreateProcess 函数可以创建一个新的进程,并为该进程分配内存空间、初始化环境变量、创建主线程等。其中,参数lpApplicationName
用于指定需要执行的可执行文件名,参数lpCommandLine
用于指定命令行参数。如果lpApplicationName
参数为NULL,则系统会自动使用lpCommandLine
参数指定的命令行来创建进程。
该函数原型如下:
BOOL CreateProcess( |
该函数可以创建包括控制台窗口的进程。如果需要使用CreateProcess()
函数创建不带控制台窗口的进程,则需要在dwCreationFlags
参数中指定CREATE_NO_WINDOW
标志位。
在创建进程时,可以通过STARTUPINFO
结构体设置进程的一些属性,例如标准输入、标准输出和标准错误输出的重定向,启动窗口的显示方式等。同时,CreateProcess()
函数会返回一个PROCESS_INFORMATION
结构体,其中包含新进程的句柄和ID等信息。
如下RunCommand
函数所示,该函数传入一个字符串类型的命令参数,并返回一个字符串执行结果,在函数内部,使用 CreatePipe()
函数创建了一个匿名管道,并使用 CreateProcess()
函数启动了一个新的 CMD
进程并将其标准输出和错误输出重定向到管道的写入端。接着使用 ReadFile()
函数从管道的读取端读取输出数据,并将读取到的数据存储到一个缓冲区中。最后,它将缓冲区的内容拼接成一个完整的输出结果返回给调用者。
// 以隐藏方式执行CMD命令 |
上述函数的调用非常容易,我们以执行ipconfig
函数为例,调用案例为RunCommand((char*)"ipconfig", szBuffer)
,函数执行命令ipconfig
参数,并将返回值存储值szBuffer
变量内,输出效果图如下所示;