[lldb] In python tests, call dumpSessionInfo(). (#188859)
Updates the lldb python test suite to ensure we call dumpSessionInfo()
in the test result's stopTest() method. This will ensure that we get the
session info dumped for all tests, even those that don't have an
explicit call to dumpSessionInfo() in the test case.
Additionally, I updated the lldb-dap test case to mark the '-dap.log' as
a log file, which will be recorded in the test output on failure.
Here is an example test run with a failure:
```
PASS: LLDB (build/bin/clang-arm64) :: test_step (TestDAP_step.TestDAP_step)
FAIL: LLDB (build/bin/clang-arm64) :: test_step_over_inlined_function (TestDAP_step.TestDAP_step)
Log Files:
- build/lldb-test-build.noindex/tools/lldb-dap/step/TestDAP_step/Failure.log
- build/lldb-test-build.noindex/tools/lldb-dap/step/TestDAP_step/Failure-dap.log
======================================================================
FAIL: test_step_over_inlined_function (TestDAP_step.TestDAP_step)
Test stepping over when the program counter is in another file.
----------------------------------------------------------------------
Traceback (most recent call last):
File "llvm-project/lldb/test/API/tools/lldb-dap/step/TestDAP_step.py", line 113, in test_step_over_inlined_function
self.assertFalse(
AssertionError: True is not false : expect path ending with 'main.cpp'.
Config=arm64-build/bin/clang
----------------------------------------------------------------------
Ran 2 tests in 4.849s
```
This commit is contained in:
parent
34f5b80731
commit
dd59a99cbf
@ -44,6 +44,7 @@ import signal
|
||||
from subprocess import *
|
||||
import sys
|
||||
import time
|
||||
import datetime
|
||||
import traceback
|
||||
from typing import Optional, Union
|
||||
|
||||
@ -966,6 +967,9 @@ class Base(unittest.TestCase):
|
||||
self.lib_lldb = lib
|
||||
self.darwinWithFramework = self.platformIsDarwin()
|
||||
|
||||
# As the last operation, mark the setup completed for dumpSessionInfo.
|
||||
self.__setup_done__ = True
|
||||
|
||||
def setAsync(self, value):
|
||||
"""Sets async mode to True/False and ensures it is reset after the testcase completes."""
|
||||
old_async = self.dbg.GetAsync()
|
||||
@ -1247,16 +1251,18 @@ class Base(unittest.TestCase):
|
||||
Dump the debugger interactions leading to a test error/failure. This
|
||||
allows for more convenient postmortem analysis.
|
||||
|
||||
See also LLDBTestResult (dotest.py) which is a singlton class derived
|
||||
See also LLDBTestResult (dotest.py) which is a singleton class derived
|
||||
from TextTestResult and overwrites addError, addFailure, and
|
||||
addExpectedFailure methods to allow us to to mark the test instance as
|
||||
such.
|
||||
"""
|
||||
# Ensure 'setUp' has completed.
|
||||
if not getattr(self, "__setup_done__", False):
|
||||
return
|
||||
|
||||
# We are here because self.tearDown() detected that this test instance
|
||||
# either errored or failed. The lldb.test_result singleton contains
|
||||
# two lists (errors and failures) which get populated by the unittest
|
||||
# framework. Look over there for stack trace information.
|
||||
# The lldb.test_result singleton contains two lists (errors and
|
||||
# failures) which get populated by the unittest framework. Look over
|
||||
# there for stack trace information.
|
||||
#
|
||||
# The lists contain 2-tuples of TestCase instances and strings holding
|
||||
# formatted tracebacks.
|
||||
@ -1286,16 +1292,15 @@ class Base(unittest.TestCase):
|
||||
|
||||
session_file = self.getLogBasenameForCurrentTest() + ".log"
|
||||
|
||||
lldbutil.mkdir_p(os.path.dirname(session_file))
|
||||
# Python 3 doesn't support unbuffered I/O in text mode. Open buffered.
|
||||
session = encoded_file.open(session_file, "utf-8", mode="w")
|
||||
session = encoded_file.open(session_file, "utf-8", mode="a")
|
||||
|
||||
if not self.__unexpected__ and not self.__skipped__:
|
||||
for test, traceback in pairs:
|
||||
if test is self:
|
||||
print(traceback, file=session)
|
||||
|
||||
import datetime
|
||||
|
||||
print(
|
||||
"Session info generated @",
|
||||
datetime.datetime.now().ctime(),
|
||||
@ -1307,8 +1312,12 @@ class Base(unittest.TestCase):
|
||||
# process the log files
|
||||
if prefix != "Success" or lldbtest_config.log_success:
|
||||
# keep all log files, rename them to include prefix
|
||||
# e.g. .../TestDAP_module/Incomplete.log > Failure_<test-name>.log
|
||||
src_log_basename = self.getLogBasenameForCurrentTest()
|
||||
dst_log_basename = self.getLogBasenameForCurrentTest(prefix)
|
||||
dst_log_basename = (
|
||||
f"{self.getLogBasenameForCurrentTest(prefix)}_{self.testMethodName}"
|
||||
)
|
||||
files = []
|
||||
for src in self.log_files:
|
||||
if os.path.isfile(src):
|
||||
dst = src.replace(src_log_basename, dst_log_basename)
|
||||
@ -1323,6 +1332,12 @@ class Base(unittest.TestCase):
|
||||
|
||||
lldbutil.mkdir_p(os.path.dirname(dst))
|
||||
os.rename(src, dst)
|
||||
files.append(dst)
|
||||
if files:
|
||||
print(
|
||||
"Log Files:\n - %s" % ("\n - ".join(files)),
|
||||
file=sys.stderr,
|
||||
)
|
||||
else:
|
||||
# success! (and we don't want log files) delete log files
|
||||
for log_file in self.log_files:
|
||||
|
||||
@ -296,3 +296,10 @@ class LLDBTestResult(unittest.TextTestResult):
|
||||
self.stream.write(
|
||||
"XPASS: LLDB (%s) :: %s\n" % (self._config_string(test), str(test))
|
||||
)
|
||||
|
||||
def stopTest(self, test):
|
||||
"""Dump the session info for debugging."""
|
||||
dumpSessionInfo = getattr(test, "dumpSessionInfo", None)
|
||||
if dumpSessionInfo:
|
||||
dumpSessionInfo()
|
||||
super().stopTest(test)
|
||||
|
||||
@ -5,7 +5,6 @@ from __future__ import annotations
|
||||
|
||||
from typing import Protocol
|
||||
import argparse
|
||||
import binascii
|
||||
import dataclasses
|
||||
import enum
|
||||
import json
|
||||
@ -15,7 +14,6 @@ import pathlib
|
||||
import re
|
||||
import signal
|
||||
import socket
|
||||
import string
|
||||
import subprocess
|
||||
import sys
|
||||
import threading
|
||||
@ -126,16 +124,6 @@ class Breakpoint(TypedDict, total=False):
|
||||
return src.get("verified", False)
|
||||
|
||||
|
||||
def dump_dap_log(log_file: Optional[str]) -> None:
|
||||
print("========= DEBUG ADAPTER PROTOCOL LOGS =========", file=sys.stderr)
|
||||
if log_file is None:
|
||||
print("no log file available", file=sys.stderr)
|
||||
else:
|
||||
with open(log_file, "r") as file:
|
||||
print(file.read(), file=sys.stderr)
|
||||
print("========= END =========", file=sys.stderr)
|
||||
|
||||
|
||||
class NotSupportedError(KeyError):
|
||||
"""Raised if a feature is not supported due to its capabilities."""
|
||||
|
||||
@ -216,11 +204,9 @@ class DebugCommunication(object):
|
||||
recv: BinaryIO,
|
||||
send: BinaryIO,
|
||||
init_commands: Optional[List[str]] = None,
|
||||
log_file: Optional[str] = None,
|
||||
spawn_helper: Optional[SpawnHelperCallback] = None,
|
||||
):
|
||||
self._log = Log()
|
||||
self.log_file = log_file
|
||||
self.send = send
|
||||
self.recv = recv
|
||||
self.spawn_helper = spawn_helper
|
||||
@ -1656,8 +1642,6 @@ class DebugCommunication(object):
|
||||
self.send.close()
|
||||
if self._recv_thread.is_alive():
|
||||
self._recv_thread.join()
|
||||
if self.log_file:
|
||||
dump_dap_log(self.log_file)
|
||||
|
||||
def request_setInstructionBreakpoints(self, memory_reference=[]):
|
||||
breakpoints = []
|
||||
@ -1681,6 +1665,7 @@ class DebugAdapterServer(DebugCommunication):
|
||||
*,
|
||||
executable: Optional[str] = None,
|
||||
connection: Optional[str] = None,
|
||||
connection_timeout: Optional[stintr] = None,
|
||||
init_commands: Optional[list[str]] = None,
|
||||
log_file: Optional[str] = None,
|
||||
env: Optional[Dict[str, str]] = None,
|
||||
@ -1693,6 +1678,7 @@ class DebugAdapterServer(DebugCommunication):
|
||||
process, connection = DebugAdapterServer.launch(
|
||||
executable=executable,
|
||||
connection=connection,
|
||||
connection_timeout=connection_timeout,
|
||||
env=env,
|
||||
log_file=log_file,
|
||||
additional_args=additional_args,
|
||||
@ -1715,7 +1701,6 @@ class DebugAdapterServer(DebugCommunication):
|
||||
s.makefile("rb"),
|
||||
s.makefile("wb"),
|
||||
init_commands,
|
||||
log_file,
|
||||
spawn_helper,
|
||||
)
|
||||
self.connection = connection
|
||||
@ -1724,7 +1709,6 @@ class DebugAdapterServer(DebugCommunication):
|
||||
self.process.stdout,
|
||||
self.process.stdin,
|
||||
init_commands,
|
||||
log_file,
|
||||
spawn_helper,
|
||||
)
|
||||
|
||||
|
||||
@ -24,26 +24,40 @@ class DAPTestCaseBase(TestBase):
|
||||
DEFAULT_TIMEOUT: Final[float] = dap_server.DEFAULT_TIMEOUT
|
||||
NO_DEBUG_INFO_TESTCASE = True
|
||||
|
||||
def setUp(self):
|
||||
self.dap_server_count = 0
|
||||
super().setUp()
|
||||
|
||||
def create_debug_adapter(
|
||||
self,
|
||||
lldbDAPEnv: Optional[dict[str, str]] = None,
|
||||
connection: Optional[str] = None,
|
||||
connection_timeout: Optional[int] = None,
|
||||
additional_args: Optional[list[str]] = None,
|
||||
):
|
||||
"""Create the Visual Studio Code debug adapter"""
|
||||
self.assertTrue(
|
||||
is_exe(self.lldbDAPExec), "lldb-dap must exist and be executable"
|
||||
)
|
||||
log_file_path = self.getBuildArtifact("dap.log")
|
||||
if self.dap_server_count:
|
||||
log_file_path = (
|
||||
self.getLogBasenameForCurrentTest()
|
||||
+ f"-dap-{self.dap_server_count}.log"
|
||||
)
|
||||
else:
|
||||
log_file_path = self.getLogBasenameForCurrentTest() + "-dap.log"
|
||||
self.dap_server_count += 1
|
||||
self.dap_server = dap_server.DebugAdapterServer(
|
||||
executable=self.lldbDAPExec,
|
||||
connection=connection,
|
||||
connection_timeout=connection_timeout,
|
||||
init_commands=self.setUpCommands(),
|
||||
log_file=log_file_path,
|
||||
env=lldbDAPEnv,
|
||||
additional_args=additional_args or [],
|
||||
additional_args=additional_args,
|
||||
spawn_helper=self.spawnSubprocess,
|
||||
)
|
||||
self.log_files.append(log_file_path)
|
||||
|
||||
def build_and_create_debug_adapter(
|
||||
self,
|
||||
|
||||
@ -14,24 +14,15 @@ EXIT_SUCCESS = 0
|
||||
|
||||
class TestDAP_io(lldbdap_testcase.DAPTestCaseBase):
|
||||
def launch(self):
|
||||
log_file_path = self.getBuildArtifact("dap.log")
|
||||
process, _ = dap_server.DebugAdapterServer.launch(
|
||||
executable=self.lldbDAPExec, log_file=log_file_path
|
||||
)
|
||||
self.create_debug_adapter()
|
||||
self.assertIsNotNone(self.dap_server.process)
|
||||
process = self.dap_server.process
|
||||
|
||||
def cleanup():
|
||||
# If the process is still alive, terminate it.
|
||||
# If the process is still alive, kill it.
|
||||
if process.poll() is None:
|
||||
process.terminate()
|
||||
process.kill()
|
||||
process.wait()
|
||||
stdout_data = process.stdout.read().decode()
|
||||
print("========= STDOUT =========", file=sys.stderr)
|
||||
print(stdout_data, file=sys.stderr)
|
||||
print("========= END =========", file=sys.stderr)
|
||||
print("========= DEBUG ADAPTER PROTOCOL LOGS =========", file=sys.stderr)
|
||||
with open(log_file_path, "r") as file:
|
||||
print(file.read(), file=sys.stderr)
|
||||
print("========= END =========", file=sys.stderr)
|
||||
|
||||
# Execute the cleanup function during test case tear down.
|
||||
self.addTearDownHook(cleanup)
|
||||
|
||||
@ -11,31 +11,38 @@ import dap_server
|
||||
from lldbsuite.test.decorators import *
|
||||
from lldbsuite.test.lldbtest import *
|
||||
import lldbdap_testcase
|
||||
from subprocess import Popen
|
||||
from typing import Tuple
|
||||
|
||||
|
||||
class TestDAP_server(lldbdap_testcase.DAPTestCaseBase):
|
||||
def start_server(
|
||||
self, connection, connection_timeout=None, wait_seconds_for_termination=None
|
||||
):
|
||||
log_file_path = self.getBuildArtifact("dap.log")
|
||||
(process, connection) = dap_server.DebugAdapterServer.launch(
|
||||
executable=self.lldbDAPExec,
|
||||
connection=connection,
|
||||
connection_timeout=connection_timeout,
|
||||
log_file=log_file_path,
|
||||
self, connection, connection_timeout=30
|
||||
) -> Tuple[Popen[bytes], str]:
|
||||
self.create_debug_adapter(
|
||||
connection=connection, connection_timeout=connection_timeout
|
||||
)
|
||||
assert self.dap_server.process
|
||||
assert self.dap_server.connection
|
||||
|
||||
# Save the process instance for the cleanup in case a new server is
|
||||
# created.
|
||||
process: Popen[bytes] = self.dap_server.process
|
||||
|
||||
def cleanup():
|
||||
if wait_seconds_for_termination is not None:
|
||||
process.wait(wait_seconds_for_termination)
|
||||
else:
|
||||
process.terminate()
|
||||
try:
|
||||
process.stdin.close()
|
||||
process.wait(timeout=5)
|
||||
except TimeoutExpired:
|
||||
process.kill()
|
||||
|
||||
self.addTearDownHook(cleanup)
|
||||
|
||||
return (process, connection)
|
||||
return (process, self.dap_server.connection)
|
||||
|
||||
def run_debug_session(self, connection, name, sleep_seconds_in_middle=None):
|
||||
def run_debug_session(
|
||||
self, connection: str, name: str, *, sleep_seconds_in_middle: float = 0
|
||||
):
|
||||
self.dap_server = dap_server.DebugAdapterServer(
|
||||
connection=connection, spawn_helper=self.spawnSubprocess
|
||||
)
|
||||
@ -48,7 +55,7 @@ class TestDAP_server(lldbdap_testcase.DAPTestCaseBase):
|
||||
args=[name],
|
||||
disconnectAutomatically=False,
|
||||
)
|
||||
if sleep_seconds_in_middle is not None:
|
||||
if sleep_seconds_in_middle:
|
||||
time.sleep(sleep_seconds_in_middle)
|
||||
self.set_source_breakpoints(source, [breakpoint_line])
|
||||
self.continue_to_next_stop()
|
||||
@ -88,14 +95,11 @@ class TestDAP_server(lldbdap_testcase.DAPTestCaseBase):
|
||||
@skipIfWindows
|
||||
def test_server_interrupt(self):
|
||||
"""
|
||||
Test launching a binary with lldb-dap in server mode and shutting down the server while the debug session is still active.
|
||||
Test launching a binary with lldb-dap in server mode and shutting down
|
||||
the server while the debug session is still active.
|
||||
"""
|
||||
self.build()
|
||||
(process, connection) = self.start_server(connection="listen://localhost:0")
|
||||
self.dap_server = dap_server.DebugAdapterServer(
|
||||
connection=connection,
|
||||
spawn_helper=self.spawnSubprocess,
|
||||
)
|
||||
(process, _) = self.start_server(connection="listen://localhost:0")
|
||||
program = self.getBuildArtifact("a.out")
|
||||
source = "main.c"
|
||||
breakpoint_line = line_number(source, "// breakpoint")
|
||||
@ -122,39 +126,39 @@ class TestDAP_server(lldbdap_testcase.DAPTestCaseBase):
|
||||
@skipIfWindows
|
||||
def test_connection_timeout_at_server_start(self):
|
||||
"""
|
||||
Test launching lldb-dap in server mode with connection timeout and waiting for it to terminate automatically when no client connects.
|
||||
Test launching lldb-dap in server mode with connection timeout and
|
||||
waiting for it to terminate automatically when no client connects.
|
||||
"""
|
||||
self.build()
|
||||
self.start_server(
|
||||
connection="listen://localhost:0",
|
||||
connection_timeout=1,
|
||||
wait_seconds_for_termination=5,
|
||||
)
|
||||
|
||||
@skipIfWindows
|
||||
def test_connection_timeout_long_debug_session(self):
|
||||
"""
|
||||
Test launching lldb-dap in server mode with connection timeout and terminating the server after the a long debug session.
|
||||
Test launching lldb-dap in server mode with connection timeout and
|
||||
terminating the server after the a long debug session.
|
||||
"""
|
||||
self.build()
|
||||
(_, connection) = self.start_server(
|
||||
connection="listen://localhost:0",
|
||||
connection_timeout=1,
|
||||
wait_seconds_for_termination=5,
|
||||
)
|
||||
# The connection timeout should not cut off the debug session
|
||||
self.run_debug_session(connection, "Alice", 1.5)
|
||||
self.run_debug_session(connection, "Alice", sleep_seconds_in_middle=1.5)
|
||||
|
||||
@skipIfWindows
|
||||
def test_connection_timeout_multiple_sessions(self):
|
||||
"""
|
||||
Test launching lldb-dap in server mode with connection timeout and terminating the server after the last debug session.
|
||||
Test launching lldb-dap in server mode with connection timeout and
|
||||
terminating the server after the last debug session.
|
||||
"""
|
||||
self.build()
|
||||
(_, connection) = self.start_server(
|
||||
connection="listen://localhost:0",
|
||||
connection_timeout=1,
|
||||
wait_seconds_for_termination=5,
|
||||
)
|
||||
time.sleep(0.5)
|
||||
# Should be able to connect to the server.
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user