July 18th, 2015

Weekend Scripter: Break in to a Script via a Command

Doctor Scripto
Scripter

Summary: Ed Wilson, Microsoft Scripting Guy, talks about using the debugger to break in to a script via a specific command. Microsoft Scripting Guy, Ed Wilson, is here. One of the cool things, when troubleshooting a specific function in Windows PowerShell, is to set a breakpoint to occur each time the command is called. Remember, Windows PowerShell functions are also Windows PowerShell commands, just as language statements and cmdlets are.

To set the breakpoint in a command, you use the -command parameter. You can break on a call to a Windows PowerShell cmdlet, function, or external script. You can use aliases when setting breakpoints. When you create a breakpoint on an alias for a cmdlet, the debugger will only stop on the use of the alias—not on the actual command name.

In addition, you do not have to specify a script for the debugger to break. If you do not type a path to a script, the debugger will be active for everything within the Windows PowerShell console session. Every occurrence of the Foreach command will cause the debugger to break. The command to set such a breakpoint is shown here:

Set-PSBreakpoint -Command Foreach

Because Foreach is a language statement in addition to an alias for the Foreach-Object cmdlet, you might wonder whether the Windows PowerShell debugger will break on the language statement and on the use of the alias for the cmdlet. The answer is no.

You can set breakpoints on language statements, but the debugger will not break on a language statement. As shown here, the debugger breaks on the use of the Foreach alias, but not on the use of the Foreach-Object cmdlet or the % alias:

PS C:\> 1..3 | Foreach-Object {$_}
1
2
3
PS C:\> 1..3 | % {$_}
1
2
3
PS C:\> 1..3 | foreach {$_}
Hit Command breakpoint on ‘foreach’

At line:1 char:1
+ 1..3 | foreach {$_}
+ ~~~~~~~~~~~~~~~~~~~
[DBG]: PS C:\>> c
1
2
3
PS C:\>

Note You can use the shortcut technique of creating the breakpoint for the Windows PowerShell session and not specifically for the script. By leaving out the -script parameter when creating a breakpoint, you cause the debugger to break in to any running script that uses the named function. This allows you to use the same breakpoints when debugging scripts that use the same function.

When creating a breakpoint for the DivideNum function that is used in the C:\FSO\BadScript.ps1 script, you can leave off the path to the script, because only this script uses the DivideNum function.

This makes the command easier to type, but it could become confusing if you’re looking through a collection of breakpoints. If you are debugging multiple scripts in a single Windows PowerShell console session, it could become confusing if you do not specify the script to which the breakpoint applies—unless, of course, you are specifically debugging the function as it is used in multiple scripts.

Creating a command breakpoint for the DivideNum function is shown here:

PS C:\> Set-PSBreakpoint -Command DivideNum

  ID Script            Line Command           Variable         Action
  — ——            —-       ——-                ——–            ——
   7                        DivideNum When you run the script, it hits a breakpoint when the DivideNum function is called. When BadScript.ps1 hits the DivideNum function, the value of $num is 0. As you step through the DivideNum function, you assign a value of 2 to the $num variable, a result of 6 is displayed, and then the 12/$num operation is carried out. Next, the AddOne function is called and the value of $num once again becomes 0. When the AddTwo function is called, the value of $num also becomes 0. This is shown here:

PS C:\> C:\fso\BadScript.ps1
Hit Command breakpoint on ‘DivideNum’

At C:\fso\BadScript.ps1:37 char:1
+ {
+ ~
[DBG]: PS C:\>> s
At C:\fso\BadScript.ps1:38 char:2
+  12/$num
+  ~~~~~~~
[DBG]: PS C:\>> $num
0
[DBG]: PS C:\>> $num=2
[DBG]: PS C:\>> s
6
At C:\fso\BadScript.ps1:39 char:1
+ } #end function DivideNum
+ ~
[DBG]: PS C:\>> s
At C:\fso\BadScript.ps1:45 char:1
+ AddOne($num) | AddTwo($num)
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~
[DBG]: PS C:\>> $num
0
[DBG]: PS C:\>> s
At C:\fso\BadScript.ps1:12 char:1
+ {
+ ~
[DBG]: PS C:\>> q
PS C:\>

Responding to breakpoints

When the script reaches a breakpoint, control of the Windows PowerShell console is turned over to you. Inside the debugger, you can type any legal Windows PowerShell command, and even run cmdlets such as Get-Process or Get-Service.

In addition, there are several new debugging commands that can be typed into the Windows PowerShell console when a breakpoint has been reached. The available debug commands are shown in the following table.

Windows PowerShell debugging commands

Keyboard shortcut

Command name

Command meaning

S

Step-Into

Executes the next statement and then stops.

V

Step-Over

Executes the next statement, but skips functions and invocations. The skipped statements are executed, but not stepped through.

O

Step-Out

Steps up one level in the current function, if nested. If in the main body, it continues to the end or the next breakpoint. The skipped statements are executed, but not stepped through.

C

Continue

Continues to run until the script is complete or until the next breakpoint is reached. The skipped statements are executed, but not stepped through.

L

List

Displays the part of the script that is executing. By default, it displays the current line, 5 previous lines, and 10 subsequent lines. To continue listing the script, press ENTER.

L <M>

List

Displays 16 lines of the script, beginning with the line number specified by M.

L <M> <N>

List

Displays the number of lines of the script specified by N, beginning with the line number specified by M.

Q

Stop (Quit)

Stops executing the script and exits the debugger.

K

Get-PsCallStack

Displays the current call stack.

ENTER

Repeat

Repeats the last command if it was Step-Into, Step-Over, or List. Otherwise, represents a submit action.

H or ?

Help

Displays the debugger command Help.

That wraps up Debugging Week. 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 

Author

The "Scripting Guys" is a historical title passed from scripter to scripter. The current revision has morphed into our good friend Doctor Scripto who has been with us since the very beginning.

0 comments

Discussion are closed.