JMESPath descend in tree for filter

2019-06-09 11:36发布

What I'm trying to achieve

I want to extract the volume ID for the root block device using describe-instances.

What I tried so far

aws ec2 describe-instances --filters "Name=tag:Backup,Values=True" --query 'Reservations[].Instances[].{Name: Tags[?Key==`Name`].Value | [0], Id: InstanceId, Block: BlockDeviceMappings[?DeviceName==RootDeviceName].Ebs.VolumeId, Test: RootDeviceName}'

What's not working

Several things:

  1. Ebs.VolumeId is not the direct descendant of DeviceName, it is descending from BlockDeviceMappings.
  2. RootDeviceName is not a descendant of BlockDeviceMappings.

So when I'm trying to pull the RootDeviceName and search the VolumeId accordingly I'm getting a blank field (Block: is for testing and irrelevant to the case).

The first 2 fields are correct.

Thanks in advance!

1条回答
Root(大扎)
2楼-- · 2019-06-09 12:17

Yes, that is quite an ask!

The closest I got working was to specify the actual value for DeviceName:

aws ec2 describe-instances --query 'Reservations[].Instances[].[InstanceId, BlockDeviceMappings[?DeviceName==`/dev/xvda`].Ebs.VolumeId]'

(This syntax worked on a Mac.)

Frankly, I'd recommend using a language to make the call (eg Python) and then apply your own logic, rather than trying to convince JMESPath to extract the correct values.

It would be something like this:

import boto3

ec2_client = boto3.client('ec2', region_name = 'ap-southeast-2')

response = ec2_client.describe_instances(
    Filters=[
        {
            'Name': 'tag:Backup',
            'Values': ['True']
        }
    ]
)

for r in response['Reservations']:
    for i in r['Instances']:
        name = [t['Value'] for t in i['Tags'] if t['Key'] == 'Name'][0]
        for b in i['BlockDeviceMappings']:
            if b['DeviceName'] == i['RootDeviceName']:
                print (i['InstanceId'], name, b['Ebs']['VolumeId'])
查看更多
登录 后发表回答