Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

{Core} Pass JSON output between commands #6

Closed
wants to merge 2 commits into from
Closed

{Core} Pass JSON output between commands #6

wants to merge 2 commits into from

Conversation

jiasli
Copy link
Owner

@jiasli jiasli commented May 7, 2020

Pass JSON output between commands

Goal

The goal is to facilitate the interaction between commands, like how PowerShell uses variables to store objects.

For example, PowerShell CmdLet Add-AzVirtualNetworkSubnetConfig takes -VirtualNetwork <PSVirtualNetwork>. See Create a virtual network using PowerShell.

$virtualNetwork = New-AzVirtualNetwork -ResourceGroupName rg1 -Name vnet1 -Location WestUS -AddressPrefix 10.0.0.0/16
$subnetConfig = Add-AzVirtualNetworkSubnetConfig -Name default -AddressPrefix 10.0.0.0/24 -VirtualNetwork $virtualNetwork
$virtualNetwork | Set-AzVirtualNetwork

Current status

Currently, with Azure CLI users need to append --query id --output tsv to a command to extract the information from the output JSON and pass it to the next command, as documented in Tips for using Azure CLI effectively.

For example, we want to add a newly created subnet to a storage account's network rule. See Configure Azure Storage firewalls and virtual networks.

With CLI in Bash:

# Prepare resources
az group create -g rg1 -l westus
az network vnet create -g rg1 -n vnet1
az network vnet subnet create -g rg1 --vnet-name vnet1 -n subnet1 --address-prefixes 10.0.0.0/24 --service-endpoints  Microsoft.Storage
az storage account create -g rg1 -n st0507

# Add the created subnet to storage account's network rule
subnet=$(az network vnet subnet show -g rg1 --vnet-name vnet1 -n subnet1 --query id --output tsv)
az storage account network-rule add -g rg1 --account-name st0507 --subnet $subnet

With Azure PowerShell:

$subnet = Get-AzVirtualNetwork -ResourceGroupName "myresourcegroup" -Name "myvnet" | Get-AzVirtualNetworkSubnetConfig -Name "mysubnet"
Add-AzStorageAccountNetworkRule -ResourceGroupName "myresourcegroup" -Name "mystorageaccount" -VirtualNetworkResourceId $subnet.Id

We want to eliminate the usage of --query id --output tsv and achieve the same effect.

Changes

In this PR, a new factory get_json_query_type(query='id') is added for command arguments. It has one parameter as the JMESPath query string, which is used to query the input JSON string. If the input is not a valid JSON, the original value is used.

c.argument('subnet', type=get_json_query_type(), help='Name or ID of subnet. If name is supplied, `--vnet-name` must be supplied.')

Then we can pipe the output JSON to the next command with either variable style or pipeline style.

Variable style

Bash

https://www.gnu.org/software/bash/manual/html_node/Shell-Parameters.html

subnet=$(az network vnet subnet show -g rg1 --vnet-name vnet1 -n subnet1)
az storage account network-rule add -g rg1 --account-name st0507 --subnet "$subnet"

PowerShell

https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_variables

Variable style doesn't work well with PowerShell due to a bug in PowerShell. See appendix for more information.

Pipeline style

Bash

https://www.gnu.org/software/bash/manual/html_node/Pipelines.html

az network vnet subnet show -g rg1 --vnet-name vnet1 -n subnet1 | az storage account network-rule add -g rg1 --account-name st0507 --subnet @-

PowerShell

https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_pipelines

As symbol @ is interpreted by PowerShell as splatting symbol, so quote it or escape it.

az network vnet subnet show -g rg1 --vnet-name vnet1 -n subnet1 | az storage account network-rule add -g rg1 --account-name st0507 --subnet '@-'
az network vnet subnet show -g rg1 --vnet-name vnet1 -n subnet1 | az storage account network-rule add -g rg1 --account-name st0507 --subnet `@-

More information about quoting can be found at https://github.com/Azure/azure-cli/blob/dev/doc/use_cli_effectively.md#quoting-issues

Problems

Some problems can emerge with this approach:

  • The JSON can't be used to populate multiple parameters. For example, populating --resource-group and --vnet-name when a vnet JSON is provided:

    az network vnet subnet create --resource-group rg1 --vnet-name $vnet -n subnet1

    Possible solutions:

    1. Support --vnet which takes a resource ID by itself, like
      az network vnet subnet create --vnet /subscriptions/0b1f6471-1bf0-4dda-aec3-cb9272f09590/resourceGroups/rg1/providers/Microsoft.Network/virtualNetworks/vnet1 -n subnet1
      This solution is perhaps the best.
    2. Pass the JSON twice to the following command, like
      az network vnet subnet create --resource-group $vnet --vnet-name $vnet -n subnet1
      This solution is verbose and counterintuitive.
  • This can conflict with the newly introduced local context which may automatically populate --resource-group and --vnet-name

Appendix

Quoting issue with PowerShell

Due to issue PowerShell/PowerShell#1995, double quotes within the JSON string are lost when calling a native .exe file.

# Note that the double quotes are lost
> python.exe -c "import sys; print(sys.argv)" '{"key": "value"}'
['-c', '{key: value}']

# Escape double quotes (") with backward-slashes (\), and quote the string with single quotes (')
> python.exe -c "import sys; print(sys.argv)" '{\"key": \"value\"}'
['-c', '{"key: "value"}']

# First escape double quotes with backticks (`), then escape double quotes with backward-slash (\)
> python.exe -c "import sys; print(sys.argv)" "{\`"key\`": \`"value\`"}"
['-c', '{"key": "value"}']

As you can see, the workaround makes the command awkward. Of course we can do some replacement to the JSON string to workaround the PowerShell issue but it takes more efforts, thus counteracting the benefit we gain from variable style.

@jiasli jiasli changed the title Pass JSON output between commands {Core} Pass JSON output between commands May 7, 2020
@jiasli jiasli closed this Apr 9, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant