Variable Substitution in a PowerShell Script Block

Doctor Scripto

Dr Scripto

Summary: Microsoft Scripting Guy, Ed Wilson, talks about performing variable substitution inside a Windows PowerShell script block.
Hey, Scripting Guy! Question Hey, Scripting Guy!  I am trying to create a command. The command contains variables that I would like to assign prior to creating the command. However, when I get to the script block portion of my code, it does not do the variable substitution for the value the way an expanding string normally works. Can you help me?
—SW
Hey, Scripting Guy! Answer Hello SW,
Microsoft Scripting Guy, Ed Wilson, is here. Last week at the first ever Northern Virginia PowerShell User Group, the Scripting Wife and I found a great little tea shop. I can’t help but thinking about the scone I got there—it was pretty good.

Expanding variable values

One of the really cool things about Windows PowerShell is the expanding strings—I absolutely love them. It surely beats having to do lots of string concatenation like I had to do back in the VBScript days. To illustrate this technique, I assign a string to the value of a variable—in this case, the $a variable. I look at the value contained in the variable. This code is shown here.

PS C:> $a = “This is a string”

PS C:> $a

This is a string
Now, by using the expanding string, I can see the value that is contained inside the $a variable. To suppress the variable expansion, I escape it with the grave accent character as shown here.

PS C:> “The value of `$a is $a”

The value of $a is This is a string

The problem with a script block

If, on the other hand, I want to expand the value of a variable inside a script block, it does not work. This is shown in the code that follows.

PS C:> $a = “This is a string”

PS C:> $a

This is a string

PS C:> $b = {“The value of `$a is $a”}

PS C:> $b

“The value of `$a is $a”

Solving the problem with variable expansion in a script block

The solution to expanding a variable inside a script block is to do two things. First create the script block as an expanding string. This is shown here:

PS C:> $a = “This is a string”

PS C:> $a

This is a string

PS C:> $b = “The value of `$a is $a”

PS C:> $b

The value of $a is This is a string
Now, I use the static Create method from the [scriptblock] class. This will create a script block. To do this, I use the [scriptblock] class and then call the Create method while passing the string contained in the $b variable. This is shown here.

PS C:> [scriptblock]::Create($b)

The value of $a is This is a string
I can confirm that it is in fact a script block by piping the results to the Get-Member cmdlet as shown here:

PS C:> [scriptblock]::Create($b) | gm

 

   TypeName: System.Management.Automation.ScriptBlock

 

Name                    MemberType Definition

—-                    ———- ———-

CheckRestrictedLanguage Method     void CheckRestrictedLanguage(System.Collection…

Equals                  Method     bool Equals(System.Object obj)

GetHashCode             Method     int GetHashCode()

GetNewClosure           Method     scriptblock GetNewClosure()

GetObjectData           Method     void GetObjectData(System.Runtime.Serializatio…

GetPowerShell           Method     powershell GetPowerShell(Params System.Object[…

GetSteppablePipeline    Method     System.Management.Automation.SteppablePipeline…

GetType                 Method     type GetType()

Invoke                  Method     System.Collections.ObjectModel.Collection[psob…

InvokeReturnAsIs        Method     System.Object InvokeReturnAsIs(Params System.O…

ToString                Method     string ToString()

Ast                     Property   System.Management.Automation.Language.Ast Ast …

Attributes              Property   System.Collections.Generic.List[System.Attribu…

File                    Property   string File {get;}

IsFilter                Property   bool IsFilter {get;set;}

Module                  Property   psmoduleinfo Module {get;}

StartPosition           Property   System.Management.Automation.PSToken StartPosi…
Now, the cool thing about this is that I can also store the script block into another variable. This is shown here:

$sb = [scriptblock]::Create($b)
After I have stored the script block into the variable, I can also call any of the methods or properties of the script block. For example, here is the AST:

PS C:> $sb.Ast

 

ParamBlock         :

BeginBlock         :

ProcessBlock       :

EndBlock           : The value of $a is This is a string

DynamicParamBlock  :

ScriptRequirements :

Extent             : The value of $a is This is a string

Parent             :
It is not horribly exciting in this example, but for more complex code, it is definitely exciting stuff. We have a great Hey, Scripting Guy! Blog post written by Bartek Bielawski, which offers several good ideas for further exploration: Learn How IT Pros Can Use the PowerShell AST.
SW, that is all there is to using variable expansion and substitution in a Windows PowerShell script block.  Join me tomorrow when I will talk about way cool Windows PowerShell stuff.
I invite you to follow me on Twitter and Facebook. If you have any questions, send email to me at scripter@microsoft.com, or post your questions on the Official Scripting Guys Forum. See you tomorrow. Until then, peace.
Ed Wilson, Microsoft Scripting Guy 

Doctor Scripto
Dr Scripto

Scripter, PowerShell, vbScript, BAT, CMD

Follow Dr Scripto   

2 comments

  • Avatar
    Veldink, Mike (CT RDA FOA ART-US)

    Hello Ed, thank you for this insight. But isn’t this method a bit cumbersome when you work interactively in PS? Isn’t there a more convenient way to e.g. do this bash one-liner in PS if I explicitly wanted to Start-Job a script block with a substituted f in it: $ f=*.txt; find . -name $f&

  • Yawar Murtaza
    Yawar Murtaza

    Thank you so much for this post!!!!
    I have been looing for this and so gald to find this article. In my case I had a string variable that was being concotinated based upon some conditions. That string is a method call with some mandator and non-mandatory params. The past param was PSCredential that I was not sure how to pass that until i used ` thing.

    $functionCallDef = [string]::Format(“{0} -param1 {1} -param2 {2} -Credential `$Credential”, $functionCall, $p1, $p2);
    then using [ScriptBlock]::Create($functionCallDef) created the script block that is executed using & $sp.

    Many thanks
    Yawar

Leave a comment