将Windows cmd传递到Python子进程



我正在尝试使用Pyton的子流程执行aws-cli命令

windows cmd:

aws --profile some_profile --region some_region ec2 describe-instances --filters Name=tag:some_tag,Values=some_value --query "Reservations[*].Instances[*].{AvailabilityZone:Placement.AvailabilityZone,Status:State.Name,Name:Tags[?Key=='Name']|[0].Value}" --output=table

这就是我努力做到的:

profile = "some_profile"
region = "some_region"
ec2_filters = "Name=tag:some_tag,Values=some_value"
ec2_query = "Reservations[*].Instances[*].{AvailabilityZone:Placement.AvailabilityZone,Status:State.Name,Name:Tags[?Key=='Name']|[0].Value}"
ec2_output_type = "table"
proc = subprocess.Popen(["aws", "--profile", profile, "--region", region, "ec2", "describe-instances", "--filters", ec2_filters, "--query", ec2_query, "--output", ec2_output_type], stdout=subprocess.PIPE, shell=True)

这是错误消息:

'[0]。值}"未被识别为内部或外部命令,可操作程序或批处理文件。

我没有安装aws,所以我创建了一个模拟批处理文件来吐出它收到的内容。我确实试过我最初的猜测,你是对的,这通常会让事情变得困难,但我发现了。很抱歉没有测试我让你尝试的内容。

无论如何,aws.bat包含一行echo %*,它打印回批处理文件作为参数接收的任何内容,所以我们知道它正在工作。

然后,我试着使用你的命令。我得到了和你一样的错误,所以我把它修改为:

.aws.bat --profile some_profile --region some_region ec2 describe-instances --filters Name=tag:some_tag,Values=some_value --query '"Reservations[*].Instances[*].{AvailabilityZone:Placement.AvailabilityZone,Status:State.Name,Name:Tags[?Key=='Name']|[0].Value}"' --output=table

这将命令输出回来,意味着它得到了正确执行。

然后,我修改了您的代码,以确保所有查询都有引号。我使用了简单的字符串连接来实现这一点。

import subprocess
profile = "some_profile"
region = "some_region"
ec2_filters = "Name=tag:some_tag,Values=some_value"
ec2_query = (
'"Reservations[*].Instances[*].{AvailabilityZone:Placement.AvailabilityZone,Status:State.Name,Name:Tags[?Key=='
"'Name'" 
']|[0].Value}"'
)
ec2_output_type = "table"
proc = subprocess.Popen(["aws.bat", "--profile", profile, "--region", region, "ec2", "describe-instances", "--filters", ec2_filters, "--query", ec2_query, "--output", ec2_output_type])

这起到了作用。有趣的是,如果我以一种非正统的方式使用三引号,它也同样有效。

ec2_query = ' '''"Reservations[*].Instances[*].{AvailabilityZone:Placement.AvailabilityZone,Status:State.Name,Name:Tags[?Key=='Name']|[0].Value}"' '''

注意开始,' '''"。我真的不知道发生了什么。

无论如何,更简单的解决方案是分解字符串,这样引号就不会混淆。

最新更新