Rand Stats

Dawa

cpan:BDUGGAN

NAME

Dawa -- A runtime debugger for Raku programs

SYNPOSIS

Use from the command line with the dawa command:

image

Use from within a program with the stop statement:

image

DESCRIPTION

Dawa is a run-time debugger for Raku programs.

It exports one subroutine: stop, which will pause execution of the current thread of the program, and allow for introspecting the stack, and stepping through subsequent statements. It also supports debugging of multi-threaded programs.

Importing Dawa adds extra ops to the AST that is generated during compilation. Specifically, it adds at most one extra node per line. There is a significant perforance penalty for this, as well as a risk that your program will break. Patches welcome!

The functionality of dawa is inspired by Ruby's pry and Python's pdb.

USAGE

After stop is reached, a repl is started, which has a few commands. Type h to see them. Currently, these are the commands:

       break (b) : [N [filename] ] add a breakpoint at line N [in filename]
continue (c, ^D) : continue execution of this thread
       defer (d) : [n] Defer to thread [n], or the next waiting one
        eval (e) : [id] evaluate code in the current context [or in thread #id]
        help (h) : this help
          ls (l) : [-a] [id] show [all] lexical variables in the current scope [or in thread #id]
        next (n) : run the next statement in any thread
        quit (q) : terminate the program (exit)
        step (s) : execute the next statement in the same thread
     threads (t) : [id] show threads being tracked [or just thread #id]
       where (w) : show a stack trace and the current location in the code

Anything else is treated as a Raku statement: it will be evaluated, the result will be shown.

Breakpoints

Breakpoints can be set with b, for example:

  dawa [1]> b 13
  Added breakpoint at line 13 in eg/debug.raku

As shown above, breakpoints are indicated using and are not thread-specific. The triangle (▶) is the line of code that will be executed next. The [1] indicates that this is thread 1. The [1] in the prompt indicates that statements will currently be evaluated in the context of thread 1 by default.

Details of "next"

In Raku, there can be multiple statements per line, and multiple "statements" within a statement, which makes it tricky to debug a program by stepping through "statements" or stepping through "lines". Dawa has a number of heuristics for when to stop, but the idea is to stop at most once per line and at most once per statement. Typing "n" may sometimes stay within the same block of code, if a statement containing other statements spans several lines. The display will indicate the block of code containing the statement that will be executed.

Multiple Threads

If several threads are stopped at once, a lock is used in order to only have one repl at a time. Threads wait for this lock. After either continuing or going on to the next statement, another thread that is waiting for the lock may potentially become active in the repl. i.e. "next" advances to the next statement in any thread. To stay in the same thread, use "step".

∙ Stopping thread Thread #1 (Initial thread)

--- current stack ---
  in block <unit> at example.raku line 11

-- current location --
   3 │       │ my $i = 1;
   4 │       │
   5 │       │ start loop {
   6 │       │   $i++;
   7 │   [7] │   $i++;
   8 │       │ }
   9 │       │
  10 │  [1]▶ │ stop;
  11 │       │ say 'bye!';
  12 │       │

Type h for help
dawa [1]>

Note that in this situation thread 7 continues to run even while the repl is active. To also stop thread 7 while debugging, you can add a breakpoint (since breakpoints apply to all threads).

The eval command can be used to evaluate expression in another thread. For instance, eval 7 $i will evaluate the $i in thread 7.

The ls command can show lexical variables in another thread. Note that only variables in the innermost lexical scope will be shown.

The defer command can be used to switch to another stopped thread. Here is an example:

Switching between threads

When multiple threads are stopped, defer will switch from one to another. For instance, the example below has eg/defer.raku, with threads 7-11 stopped, as well as thread 1. After looking at the stack in thread 1, typing d 8 will change to thread 8, and subsequent commands will run in the context of that thread. (also note that if a lot of threads are stopped at the same line of code, they are shown in a footnote)

dawa [1]> w

--- current stack ---
  in block <unit> at eg/defer.raku line 16

-- current location --
   4 │       │   start {
   5 │       │     my $x = 10;
   6 │       │     loop {
   7 │       │       stop;
   8 │   (a) │       $x++;
   9 │       │     }
  10 │       │   }
  11 │       │ }
  12 │       │
  13 │       │ my $y = 100;
  14 │       │ loop {
  15 │       │   stop;
  16 │  [1]▶ │   say "y is $y";
  17 │       │   $y += 111;
  18 │       │   last if $y > 500;
  19 │       │ }
  20 │       │
────────────────────────────────────────
(a) : [8][9][7][10][11]
────────────────────────────────────────

dawa [1]> d 8
  8 ▶       $x++;
dawa [8]> w

--- current stack ---
  in block  at eg/defer.raku line 8

-- current location --
  4 │       │   start {
  5 │       │     my $x = 10;
  6 │       │     loop {
  7 │       │       stop;
  8 │  (a)▶ │       $x++;
  9 │       │     }
 10 │       │   }
 11 │       │ }
 12 │       │
 13 │       │ my $y = 100;
 14 │       │ loop {
 15 │       │   stop;
 16 │   [1] │   say "y is $y";
 17 │       │   $y += 111;
 18 │       │   last if $y > 500;
 19 │       │ }
 20 │       │
────────────────────────────────────────
(a) : [8][9][7][10][11]
────────────────────────────────────────

dawa [8]> $x
10

ABOUT THE NAME

The word dawa can refer to either medicine or poison in Swahili. In the latter sense, it would be used to describe bug spray, i.e. a debugger -- but hopefully it'll also help be a cure for any ailments in your programs.

SEE ALSO

  1. There is a built-in repl statement, which will pause execution and start a repl loop at the current location.

  2. rakudo-debugger provides a separate executable for debugging. Techniques there inspired this module.

ENVIRONMENT

The readline history is stored in DAWA_HISTORY_FILE, ~/.dawa-history by default.

BUGS

Since this relies on undocumented behavior, it could break at any time. Modifying the AST may also cause your program to behave in unexpected ways. It may be possible to improve this once the AST work in Raku is available.

MAILING LIST

https://lists.sr.ht/~bduggan/raku-dawa

AUTHOR

Brian Duggan (bduggan at matatu.org)