python利用subprocess执行shell命令

subprocess以及常用的封装函数

运行python的时候,我们都是在创建并运行一个进程。像Linux进程那样,一个进程可以fork一个子进程,并让这个子进程exec另外一个程序。在Python中,我们通过标准库中的subprocess包来fork一个子进程,并运行一个外部的程序。
subprocess包中定义有数个创建子进程的函数,这些函数分别以不同的方式创建子进程,所以我们可以根据需要来从中选取一个使用。另外subprocess还提供了一些管理标准流(standard stream)和管道(pipe)的工具,从而在进程间使用文本通信。

subprocess 模块可以启动一个新进程,并连接到它们的输入/输出/错误管道,获取进程执行的结果。Popen 是 subprocess的核心, 负责子进程的创建和管理, run() 方法可以便捷的获取进程的返回结果。

subprocess.Popen()

# 常用参数(入参)
# args 
	shell 命令:可以是字符串或者序列类型(list or tuple)
# bufsize 
	缓冲区大小:创建标准流的管道对象时使用, 默认值:-1
	0 : 不使用缓冲区
	1 : 表示行缓冲,仅当 universal_newlines=True时可用(也就是文本模式时)
	正数 : 表示缓冲区的大小
	负数 : 表示使用默认的缓冲区大小
# stdin, stdout, stderr
	分别表示程序的标准输入、输出、错误句柄
# preexec_fn 
	只有在Unix平台下有效,用于指定一个可执行对象,它将在子进程运行之前被调用
# shell 
	如果该参数为True, 将通过操作系统的shell执行指定命令
# cwd 
	用于设置子进程的当前目录
# env
	用于指定子进程的环境变量
	如果 env=None, 则子进程的环境变量将从父进程中继承	
# encoding 
	设置编码类型
	cn:
		mac : utf-8
		windows : gb2312

# Popen 对象的方法/属性
## poll():
	检查进程是否终止,如果终止返回returncode,否则返回None
## wait(timeout): 
	等待子进程终止,可以指定超时时间
## communicate(input, timeout):
	与子进程交互,发动和读取数据
## send_signal():
	发送信号到子进程
## terminate():
	停止子进程,即是发送SIGTERM信号到子进程
## kill():
	杀掉子进程,即是发送SIGKILL信号到子进程

# Popen 返回对象
	返回对象一般都是二进制的标准输入、输出和错误句柄

subprocess.run()

# 常用参数(入参)
# args 
	shell 命令:可以是字符串或者序列类型(list or tuple)
# stdin, stdout, stderr
	分别表示程序的标准输入、输出、错误句柄, 默认值均是None
	若要与程序交互,则需要设置stdin
	若需获取输出和错误,则需要设置stdout, stderr
		一般设置为subprocess.PIPE代表打开通向标准流的管道,创建一个新的管道
		文件对象设置为subprocess.DEVNULL
		stderr也可以设置为subprocess.STDOUT
			表示将子程序的标准错误输出重定向到了标准输出
# shell 
	默认值为False
	若为True:
		args参数可以是复杂字符串,同人为在linux中输入命令一样
	若为False:
		args命令中包含参数,则必须以命令列表的方式传入
		一般可以通过shlex.split(string)处理即可
# universal_lines
	若为True:
		stdin,stdout,stderr均已字符串方式读写,
	若为False:
		stdin,stdout,stderr以字节流方式读写
# cwd 
	用于设置子进程的当前目录(执行路径)
	运行非shell命令时,可以用来指定程序的所在路径
# timeout
	命令超时时间,当程序超过指定时间没有返回时,则会抛出异常

subprocess.call()

父进程等待子进程完成,并且返回子进程执行的结果 0/1 其实现方式
返回退出信息(returncode,相当于Linux exit code)

subprocess.check_call()

父进程等待子进程完成
返回0
检查退出信息,如果returncode不为0,则举出错误subprocess.CalledProcessError,该对象包含有returncode属性,可用try…except…来检查

subprocess.check_output()

执行args命令,返回值为命令执行的输出结果;
若执行成功,则函数返回值为命令输出结果;若执行失败,则抛出异常;
(类似subprocess.run(args, check=True, stdout=subprocess.PIPE).stdout)

测试

import subprocess


def popen(cmd):
    ret = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, encoding="gb2312")
    if not ret.returncode:
        print(ret.stdout.read())


def run(cmd):
    ret = subprocess.run(cmd, stderr=subprocess.PIPE, stdout=subprocess.PIPE, stdin=subprocess.PIPE, shell=True, encoding="gb2312")
    if not ret.returncode:
        print(ret.stdout.strip())

def call(cmd):
    ret = subprocess.call(cmd, stdout=subprocess.PIPE, stdin=subprocess.PIPE, stderr=subprocess.PIPE, shell=True)
    if ret == 0:
        print("执行成功!!")

def checkcall(cmd):
    ret = subprocess.check_output(cmd, shell=True, stdin=subprocess.PIPE, stderr=subprocess.PIPE, encoding="gb2312")
    print(ret.strip())

cmd = "python --version"
checkcall(cmd)
call(cmd)
popen(cmd)
run(cmd)

结果

Python 3.10.7
执行成功!!
Python 3.10.7

Python 3.10.7

也可以使用对ctf中一些exe文件进行爆破

posted @ 2023-01-15 23:57  gvpn  阅读(1053)  评论(0编辑  收藏  举报