use serde::{Deserialize, Serialize};
use serde_json::Value;
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(tag = "type", rename_all = "lowercase")]
pub enum ProtocolMessage {
Request(RequestMessage),
Response(ResponseMessage),
Event(EventMessage),
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct RequestMessage {
pub seq: i64,
#[serde(rename = "type")]
pub message_type: String,
pub command: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub arguments: Option<Value>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ResponseMessage {
pub seq: i64,
#[serde(rename = "type")]
pub message_type: String,
pub request_seq: i64,
pub success: bool,
pub command: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub message: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub body: Option<Value>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct EventMessage {
pub seq: i64,
#[serde(rename = "type")]
pub message_type: String,
pub event: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub body: Option<Value>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct InitializeArguments {
#[serde(skip_serializing_if = "Option::is_none")]
pub client_id: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub client_name: Option<String>,
#[serde(rename = "adapterID")]
pub adapter_id: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub locale: Option<String>,
#[serde(default = "default_true")]
pub lines_start_at1: bool,
#[serde(default = "default_true")]
pub columns_start_at1: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub path_format: Option<String>,
#[serde(default)]
pub supports_variable_type: bool,
#[serde(default)]
pub supports_variable_paging: bool,
#[serde(default)]
pub supports_run_in_terminal_request: bool,
#[serde(default)]
pub supports_memory_references: bool,
#[serde(default)]
pub supports_progress_reporting: bool,
}
fn default_true() -> bool {
true
}
impl Default for InitializeArguments {
fn default() -> Self {
Self {
client_id: Some("debugger-cli".to_string()),
client_name: Some("LLM Debugger CLI".to_string()),
adapter_id: "lldb-dap".to_string(),
locale: None,
lines_start_at1: true,
columns_start_at1: true,
path_format: Some("path".to_string()),
supports_variable_type: true,
supports_variable_paging: true,
supports_run_in_terminal_request: false,
supports_memory_references: true,
supports_progress_reporting: false,
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct LaunchArguments {
pub program: String,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub args: Vec<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub cwd: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub env: Option<std::collections::HashMap<String, String>>,
#[serde(default)]
pub stop_on_entry: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub init_commands: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub pre_run_commands: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub request: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub console: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub python: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub just_my_code: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
pub mode: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub stop_at_entry: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
pub stop_at_beginning_of_main_subprogram: Option<bool>,
#[serde(rename = "type", skip_serializing_if = "Option::is_none")]
pub type_attr: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub source_maps: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
pub out_files: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub runtime_executable: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub runtime_args: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub skip_files: Option<Vec<String>>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct AttachArguments {
pub pid: u32,
#[serde(skip_serializing_if = "Option::is_none")]
pub wait_for: Option<bool>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SetBreakpointsArguments {
pub source: Source,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub breakpoints: Vec<SourceBreakpoint>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SetFunctionBreakpointsArguments {
pub breakpoints: Vec<FunctionBreakpoint>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ContinueArguments {
pub thread_id: i64,
#[serde(default)]
pub single_thread: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct StepArguments {
pub thread_id: i64,
#[serde(skip_serializing_if = "Option::is_none")]
pub granularity: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct PauseArguments {
pub thread_id: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct StackTraceArguments {
pub thread_id: i64,
#[serde(skip_serializing_if = "Option::is_none")]
pub start_frame: Option<i64>,
#[serde(skip_serializing_if = "Option::is_none")]
pub levels: Option<i64>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ScopesArguments {
pub frame_id: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct VariablesArguments {
pub variables_reference: i64,
#[serde(skip_serializing_if = "Option::is_none")]
pub start: Option<i64>,
#[serde(skip_serializing_if = "Option::is_none")]
pub count: Option<i64>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct EvaluateArguments {
pub expression: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub frame_id: Option<i64>,
#[serde(skip_serializing_if = "Option::is_none")]
pub context: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct DisconnectArguments {
#[serde(default)]
pub restart: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub terminate_debuggee: Option<bool>,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Capabilities {
#[serde(default)]
pub supports_configuration_done_request: bool,
#[serde(default)]
pub supports_function_breakpoints: bool,
#[serde(default)]
pub supports_conditional_breakpoints: bool,
#[serde(default)]
pub supports_hit_conditional_breakpoints: bool,
#[serde(default)]
pub supports_evaluate_for_hovers: bool,
#[serde(default)]
pub supports_step_back: bool,
#[serde(default)]
pub supports_set_variable: bool,
#[serde(default)]
pub supports_restart_frame: bool,
#[serde(default)]
pub supports_restart_request: bool,
#[serde(default)]
pub supports_goto_targets_request: bool,
#[serde(default)]
pub supports_step_in_targets_request: bool,
#[serde(default)]
pub supports_completions_request: bool,
#[serde(default)]
pub supports_modules_request: bool,
#[serde(default)]
pub supports_data_breakpoints: bool,
#[serde(default)]
pub supports_read_memory_request: bool,
#[serde(default)]
pub supports_disassemble_request: bool,
#[serde(default)]
pub supports_terminate_request: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SetBreakpointsResponseBody {
pub breakpoints: Vec<Breakpoint>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct StackTraceResponseBody {
pub stack_frames: Vec<StackFrame>,
#[serde(skip_serializing_if = "Option::is_none")]
pub total_frames: Option<i64>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ThreadsResponseBody {
pub threads: Vec<Thread>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ScopesResponseBody {
pub scopes: Vec<Scope>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VariablesResponseBody {
pub variables: Vec<Variable>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct EvaluateResponseBody {
pub result: String,
#[serde(rename = "type", skip_serializing_if = "Option::is_none")]
pub type_name: Option<String>,
#[serde(default)]
pub variables_reference: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ContinueResponseBody {
#[serde(default = "default_true")]
pub all_threads_continued: bool,
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Source {
#[serde(skip_serializing_if = "Option::is_none")]
pub name: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub path: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub source_reference: Option<i64>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SourceBreakpoint {
pub line: u32,
#[serde(skip_serializing_if = "Option::is_none")]
pub column: Option<u32>,
#[serde(skip_serializing_if = "Option::is_none")]
pub condition: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub hit_condition: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub log_message: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct FunctionBreakpoint {
pub name: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub condition: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub hit_condition: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Breakpoint {
#[serde(skip_serializing_if = "Option::is_none")]
pub id: Option<u32>,
pub verified: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub message: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub source: Option<Source>,
#[serde(skip_serializing_if = "Option::is_none")]
pub line: Option<u32>,
#[serde(skip_serializing_if = "Option::is_none")]
pub column: Option<u32>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct StackFrame {
pub id: i64,
pub name: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub source: Option<Source>,
pub line: u32,
pub column: u32,
#[serde(skip_serializing_if = "Option::is_none")]
pub module_id: Option<Value>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct Thread {
pub id: i64,
pub name: String,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Scope {
pub name: String,
pub variables_reference: i64,
#[serde(default)]
pub expensive: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub source: Option<Source>,
#[serde(skip_serializing_if = "Option::is_none")]
pub line: Option<u32>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Variable {
pub name: String,
pub value: String,
#[serde(rename = "type", skip_serializing_if = "Option::is_none")]
pub type_name: Option<String>,
#[serde(default)]
pub variables_reference: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct StoppedEventBody {
pub reason: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub thread_id: Option<i64>,
#[serde(default)]
pub all_threads_stopped: bool,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub hit_breakpoint_ids: Vec<u32>,
#[serde(skip_serializing_if = "Option::is_none")]
pub text: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OutputEventBody {
pub category: Option<String>,
pub output: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub source: Option<Source>,
#[serde(skip_serializing_if = "Option::is_none")]
pub line: Option<u32>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ThreadEventBody {
pub reason: String,
pub thread_id: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ExitedEventBody {
pub exit_code: i32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct TerminatedEventBody {
#[serde(default)]
pub restart: bool,
}
#[derive(Debug, Clone)]
pub enum Event {
Initialized,
Stopped(StoppedEventBody),
Continued { thread_id: i64, all_threads_continued: bool },
Exited(ExitedEventBody),
Terminated(Option<TerminatedEventBody>),
Thread(ThreadEventBody),
Output(OutputEventBody),
Breakpoint { reason: String, breakpoint: Breakpoint },
Unknown { event: String, body: Option<Value> },
}
impl Event {
pub fn from_message(msg: &EventMessage) -> Self {
match msg.event.as_str() {
"initialized" => Event::Initialized,
"stopped" => {
if let Some(body) = &msg.body {
if let Ok(stopped) = serde_json::from_value(body.clone()) {
return Event::Stopped(stopped);
}
}
Event::Unknown {
event: msg.event.clone(),
body: msg.body.clone(),
}
}
"continued" => {
let thread_id = msg.body.as_ref()
.and_then(|b| b.get("threadId"))
.and_then(|v| v.as_i64())
.unwrap_or(0);
let all_threads_continued = msg.body.as_ref()
.and_then(|b| b.get("allThreadsContinued"))
.and_then(|v| v.as_bool())
.unwrap_or(true);
Event::Continued { thread_id, all_threads_continued }
}
"exited" => {
if let Some(body) = &msg.body {
if let Ok(exited) = serde_json::from_value(body.clone()) {
return Event::Exited(exited);
}
}
Event::Exited(ExitedEventBody { exit_code: 0 })
}
"terminated" => {
let body = msg.body.as_ref()
.and_then(|b| serde_json::from_value(b.clone()).ok());
Event::Terminated(body)
}
"thread" => {
if let Some(body) = &msg.body {
if let Ok(thread) = serde_json::from_value(body.clone()) {
return Event::Thread(thread);
}
}
Event::Unknown {
event: msg.event.clone(),
body: msg.body.clone(),
}
}
"output" => {
if let Some(body) = &msg.body {
if let Ok(output) = serde_json::from_value(body.clone()) {
return Event::Output(output);
}
}
Event::Unknown {
event: msg.event.clone(),
body: msg.body.clone(),
}
}
"breakpoint" => {
if let Some(body) = &msg.body {
let reason = body.get("reason")
.and_then(|v| v.as_str())
.unwrap_or("unknown")
.to_string();
if let Some(bp) = body.get("breakpoint") {
if let Ok(breakpoint) = serde_json::from_value(bp.clone()) {
return Event::Breakpoint { reason, breakpoint };
}
}
}
Event::Unknown {
event: msg.event.clone(),
body: msg.body.clone(),
}
}
_ => Event::Unknown {
event: msg.event.clone(),
body: msg.body.clone(),
},
}
}
}