<div dir="ltr">Thanks for the context.</div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Sep 19, 2017 at 11:58 AM, Jim Ingham <span dir="ltr"><<a href="mailto:jingham@apple.com" target="_blank">jingham@apple.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span class=""><br>
> On Sep 19, 2017, at 11:25 AM, Leonard Mosescu <<a href="mailto:mosescu@google.com">mosescu@google.com</a>> wrote:<br>
><br>
> These are all great suggestions, thanks everyone!<br>
><br>
> > We should have a test. The test would need to use pexpect or something similar. If anyone has any pointer on how to make a test for this, please chime in. I was thinking just a pexpect based test.<br>
> I love tests but what exactly do we want to test here? Sorry if I'm missing something obvious - are you talking about splitting a string into chunks? The actual interruption? ...<br>
><br>
> Also, I like the idea of exposing this though the SB APIs, but that's a significant expansion of the original goal which is to improve the interactive LLDB debugger.<br>
<br>
</span>This is sort of a side point, but one of the goals of lldb is to make the command set extensible by adding commands implemented with the SB API's either from Python or from C++ (Python is more common but both are possible.)  And there are a bunch of groups here at Apple that have all sorts of special purpose LLDB commands implemented this way.  The xnu.py and Co. that gets distributed with the kernel development kit has a wide array of examples of this that are publicly available if you want to have a look.<br>
<br>
So the SB API's are in fact an intrinsic part of the interactive LLDB debugger.  We aren't there yet (there isn't an SB API to define command options & arguments so SB added commands don't parse like lldb commands do) but the end goal is that it you can't tell whether commands are from SB API's or are builtin.<br>
<br>
BTW, I think the fact that we use lldb_private API's is an unfortunate artifact of how lldb was implemented.  Having to use our vended API's to implement the commands would make them better implemented - since we would really be on the hook for them, and would reduce a lot of code duplication where we do things one way in the commands and slightly differently in the equivalent SB API's, and reduce the testing surface considerably.  I'm not sure this is a project we will ever actually get to, but it is a good aspirational goal to keep in mind.<br>
<br>
We didn't do it that way originally because we needed something we could poke at early on in the development of lldb, at a point where it would have been premature to design the SB API's.  Then lldb went from skunkworks project to mission critical debugger too quickly for us to be able to revise the decision.<br>
<span class="HOEnZb"><font color="#888888"><br>
Jim<br>
</font></span><div class="HOEnZb"><div class="h5"><br>
<br>
<br>
> It may be nice looking at SB APIs later on, but I'm afraid that trying to pile up everything in one change is just going to spiral the complexity out of control.<br>
><br>
> On Tue, Sep 19, 2017 at 11:15 AM, Zachary Turner <<a href="mailto:zturner@google.com">zturner@google.com</a>> wrote:<br>
> That would work, but I think it's adding many more pieces to the test.  Now there's threads, a Python layer, and multiprocess dotest infrastructure in the equation.  Each providing an additional source of flakiness and instability.<br>
><br>
> If all it is is a pass-through, then just calling the function it passes through to is a strictly better test, since it's focusing the test on the underlying piece of functionality and removing additional sources of failure and instability from the test.<br>
><br>
> On Tue, Sep 19, 2017 at 11:02 AM Jim Ingham <<a href="mailto:jingham@apple.com">jingham@apple.com</a>> wrote:<br>
> I'd really prefer to do this as/along with an SB API test since we also need commands made through the SB API to be interruptible, and we should test that that also works.  So this kills two birds with one stone.<br>
><br>
> In general, when developing any high-level features in lldb one of the questions you should ask yourself is whether this is useful at the SB API layer.  In this case it obviously is (actually I'm a little embarrassed I didn't think of this requirement).  If so, then it's simplest to test it at that layer.  If the SB API is anything more than a pass-through, then you might also want to write a unit test for the underlying C++ API's.  But in this case the SB function will just call the underlying CommandInterpreter one, so testing at the SB layer is sufficient.<br>
><br>
> Jim<br>
><br>
> > On Sep 19, 2017, at 10:45 AM, Zachary Turner via Phabricator <<a href="mailto:reviews@reviews.llvm.org">reviews@reviews.llvm.org</a>> wrote:<br>
> ><br>
> > zturner added a comment.<br>
> ><br>
> > In <a href="https://reviews.llvm.org/D37923#875322" rel="noreferrer" target="_blank">https://reviews.llvm.org/<wbr>D37923#875322</a>, @clayborg wrote:<br>
> ><br>
> >> We should have a test. The test would need to use pexpect or something similar. If anyone has any pointer on how to make a test for this, please chime in. I was thinking just a pexpect based test.<br>
> ><br>
> ><br>
> > This kind of thing is perfect for a unit test, but I'm not sure how easy that would be with the current design.  You'd basically do something like:<br>
> ><br>
> >  struct MockStream {<br>
> >    explicit MockStream(CommandInterpreter &Interpreter, int InterruptAfter)<br>
> >      : CommandInterpreter(<wbr>Interpreter), InterruptAfter(InterruptAfter) {}<br>
> ><br>
> >    CommandInterpreter &Interpreter;<br>
> >    const int InterruptAfter;<br>
> >    int Lines = 0;<br>
> >    std::string Buffer;<br>
> ><br>
> >    void Write(StringRef S) {<br>
> >      ++Lines;<br>
> >      if (Lines >= InterruptAfter) {<br>
> >        Interpreter.Interrupt();<br>
> >        return;<br>
> >      }<br>
> >      Buffer += S;<br>
> >    }<br>
> >  };<br>
> ><br>
> >  TEST_F(CommandInterruption) {<br>
> >    CommandInterpreter Interpreter;<br>
> >    MockStream Stream(Interpreter, 3);<br>
> >    Interpreter.<wbr>PrintCommandOutput(Stream, "a\nb\nc\nd\ne\nf\n");<br>
> >    EXPECT_EQ(Stream.Lines == 3);<br>
> >    EXPECT_EQ(Stream.Buffer == "a\nb\nc\n");<br>
> >  }<br>
> ><br>
> ><br>
> > <a href="https://reviews.llvm.org/D37923" rel="noreferrer" target="_blank">https://reviews.llvm.org/<wbr>D37923</a><br>
> ><br>
> ><br>
> ><br>
><br>
><br>
<br>
</div></div></blockquote></div><br></div>