English 中文(简体)
Issue with ) when using Azure CLI to set Azure Function app Configurations
原标题:

I published an app using Azure CLI with command

func azure functionapp publish [my-app-name] --nozip

Now I want to update the configuration of the app to connect to the Azure Key Vault and get secrets from the Vault. So, I am running this command

az functionapp config appsettings set --name [my-app-name] --resource-group [my-app-res-group] --settings "PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])"

this returns an error like this

request failed: Error occurred in request., RetryError: HTTPSConnectionPool(host= management.azure.com , port=443): Max retries exceeded with url: /subscriptions/[my-sub-guid]/resourceGroups/[my-app-res-group]/providers/Microsoft.Web/sites/[my-app-name]/config/appsettings?api-version=2019-08-01 (Caused by ResponseError( too many 500 error responses ,))

But the command work fine and publish the setting to the Azure except it publish not

PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])

But

PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name]

Whats the problem with closing ) and what can I do to overcome this issue.

P.S. I know I can do this both from Azure Portal, Visual Studio and VS Code but it s critical for me to do this using commandline tool!

最佳回答

You can have a look at this wiki around quoting issues with PowerShell:

To make it work you have few options:

  1. Add additional double quotes for force powershell to treat the argument as a literal:

    az functionapp config appsettings set --name [my-app-name] --resource-group [my-app-res-group] --settings `""PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])"`"
    
  2. Use --% to stop PowerShell from parsing the argument and escape double quotes

    az --% functionapp config appsettings set --name [my-app-name] --resource-group [my-app-res-group] --settings "PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])"
    
问题回答

To complement Thomas s helpful answer:

The problem comes down to two separate behaviors:

  • The az CLI is implemented as a batch file (az.cmd)

  • PowerShell - of necessity - rebuilds the command line that is passed to external programs (which includes batch files) behind the scenes.

    • In doing so, it uses "..." quoting (the only kind of quoting that external programs can be expected to understand) on demand: argument values are only enclosed in "..." if they contain spaces.

Unfortunately - and inappropriately - cmd.exe (the interpreter of batch files) evaluates the arguments passed to batch file as if they had been submitted from inside a cmd.exe session, which causes space-less arguments - which PowerShell passes unquoted - that happen to contain cmd.exe metacharacters (e.g, &) to be interpreted as such, breaking the command.


The upshot is: you need to control the quoting used on the process command line explicitly:

  • To pass verbatim arguments to az.cmd, you can use --%, the stop-parsing token - though note its fundamental limitations, summarized in the bottom section of this answer:
# Note the use of --% after --settings; only for *verbatim* arguments.
az functionapp config appsettings set --name [my-app-name] --resource-group [my-app-res-group] --settings --% "PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])"
  • If you need string expansion (interpolation), i.e. if the arguments need to be expressed in terms of PowerShell variables or expressions, call via cmd.exe /c:

    • On the PowerShell side, an expandable (double-quoted) string ("...") ensures that PowerShell variables (e.g., $foo) or expressions (e.g., $(1 + 2)) are interpolated; embedded " chars. must be escaped as `"
cmd /c "az functionapp config appsettings set --name [my-app-name] --resource-group [my-app-res-group] --settings `"PublicKey=@Microsoft.KeyVault(VaultName=[vault-name];SecretName=[secret-name])`""

Note:

  • Using direct invocation with `""..."`" or "`"...`"" to ensure "..." enclosure on the resulting process command line should never have worked, and only works due to a long-standing bug - see this answer.

    • While this bug was mostly fixed in PowerShell (Core) 7.3, the old, broken behavior is - unfortunately - selectively retained on Windows, notably when calling batch files.

  • The advantage of the cmd /c approach is that it predictably works, irrespective of what external program you re calling, and is therefore the preferable way to control the exact quoting if and when needed.





相关问题
Mutually exclusive powershell parameters

SCENARIO I m writing a cmdlet for Powershell 2.0 using Visual Studio 2008 and .NET 3.5 the cmdlet requires 3 arguments. my intended grammar of the cmdlet is something like this: cmdletname [foo|...

Run a program from PowerShell with timeout

I ll write a script that runs a program and wait for it finished. But if the program is not finished within a specified time I want that the program is killed.

How to transpose data in powershell

I have a file that looks like this: a,1 b,2 c,3 a,4 b,5 c,6 (...repeat 1,000s of lines) How can I transpose it into this? a,b,c 1,2,3 4,5,6 Thanks

Powershell v2 remoting and delegation

I have installed Powershell V2 on 2 machines and run Enable-PsRemoting on both of them. Both machines are Win 2003 R2 and are joined to the same active directory domain and I can successfully run ...

PowerShell -match operator and multiple groups

I have the following log entry that I am processing in PowerShell I m trying to extract all the activity names and durations using the -match operator but I am only getting one match group back. I m ...

热门标签