pub use super::generated_api::api::{
action::{Action as ProtobufAction, Position as ProtobufPosition},
event::{
event::Payload as ProtobufEventPayload,
layout_parsing_error::ErrorType as ProtobufLayoutParsingErrorType,
pane_scrollback_response, ActionCompletePayload as ProtobufActionCompletePayload,
AvailableLayoutInfoPayload as ProtobufAvailableLayoutInfoPayload,
ClientInfo as ProtobufClientInfo, ClientPaneHistory as ProtobufClientPaneHistory,
ClientTabHistory as ProtobufClientTabHistory, ContextItem as ProtobufContextItem,
CopyDestination as ProtobufCopyDestination, CwdChangedPayload as ProtobufCwdChangedPayload,
Event as ProtobufEvent, EventNameList as ProtobufEventNameList,
EventType as ProtobufEventType, FileMetadata as ProtobufFileMetadata,
InputModeKeybinds as ProtobufInputModeKeybinds, KdlError as ProtobufKdlError,
KdlErrorVariant as ProtobufKdlErrorVariant, KeyBind as ProtobufKeyBind,
LayoutInfo as ProtobufLayoutInfo, LayoutMetadata as ProtobufLayoutMetadata,
LayoutParsingError as ProtobufLayoutParsingError,
LayoutWithError as ProtobufLayoutWithError, ModeUpdatePayload as ProtobufModeUpdatePayload,
PaneContents as ProtobufPaneContents, PaneContentsEntry as ProtobufPaneContentsEntry,
PaneId as ProtobufPaneId, PaneInfo as ProtobufPaneInfo,
PaneManifest as ProtobufPaneManifest, PaneMetadata as ProtobufPaneMetadata,
PaneRenderReportPayload as ProtobufPaneRenderReportPayload,
PaneScrollbackResponse as ProtobufPaneScrollbackResponse, PaneType as ProtobufPaneType,
PluginConfigurationChangedPayload as ProtobufPluginConfigurationChangedPayload,
PluginInfo as ProtobufPluginInfo, ResurrectableSession as ProtobufResurrectableSession,
SelectedText as ProtobufSelectedText, SessionManifest as ProtobufSessionManifest,
SyntaxError as ProtobufSyntaxError, TabInfo as ProtobufTabInfo,
TabMetadata as ProtobufTabMetadata, UserActionPayload as ProtobufUserActionPayload,
WebServerStatusPayload as ProtobufWebServerStatusPayload, WebSharing as ProtobufWebSharing,
*,
},
input_mode::InputMode as ProtobufInputMode,
key::Key as ProtobufKey,
style::Style as ProtobufStyle,
};
#[allow(hidden_glob_reexports)]
use crate::data::{
ClientId, ClientInfo, CopyDestination, Event, EventType, FileMetadata, InputMode,
KeyWithModifier, LayoutInfo, LayoutMetadata, ModeInfo, Mouse, PaneContents, PaneId, PaneInfo,
PaneManifest, PaneMetadata, PaneScrollbackResponse, PermissionStatus, PluginCapabilities,
PluginInfo, SelectedText, SessionInfo, Style, TabInfo, TabMetadata, WebServerStatus,
WebSharing,
};
use crate::errors::prelude::*;
use crate::input::actions::Action;
use std::collections::{BTreeMap, HashMap, HashSet};
use std::convert::TryFrom;
use std::net::IpAddr;
use std::path::PathBuf;
use std::str::FromStr;
use std::time::Duration;
impl TryFrom<ProtobufEvent> for Event {
type Error = &'static str;
fn try_from(protobuf_event: ProtobufEvent) -> Result<Self, &'static str> {
match ProtobufEventType::from_i32(protobuf_event.name) {
Some(ProtobufEventType::ModeUpdate) => match protobuf_event.payload {
Some(ProtobufEventPayload::ModeUpdatePayload(protobuf_mode_update_payload)) => {
let mode_info: ModeInfo = protobuf_mode_update_payload.try_into()?;
Ok(Event::ModeUpdate(mode_info))
},
_ => Err("Malformed payload for the ModeUpdate Event"),
},
Some(ProtobufEventType::TabUpdate) => match protobuf_event.payload {
Some(ProtobufEventPayload::TabUpdatePayload(protobuf_tab_info_payload)) => {
let mut tab_infos: Vec<TabInfo> = vec![];
for protobuf_tab_info in protobuf_tab_info_payload.tab_info {
tab_infos.push(TabInfo::try_from(protobuf_tab_info)?);
}
Ok(Event::TabUpdate(tab_infos))
},
_ => Err("Malformed payload for the TabUpdate Event"),
},
Some(ProtobufEventType::PaneUpdate) => match protobuf_event.payload {
Some(ProtobufEventPayload::PaneUpdatePayload(protobuf_pane_update_payload)) => {
let mut pane_manifest: HashMap<usize, Vec<PaneInfo>> = HashMap::new();
for protobuf_pane_manifest in protobuf_pane_update_payload.pane_manifest {
let tab_index = protobuf_pane_manifest.tab_index as usize;
let mut panes = vec![];
for protobuf_pane_info in protobuf_pane_manifest.panes {
panes.push(protobuf_pane_info.try_into()?);
}
if pane_manifest.contains_key(&tab_index) {
return Err("Duplicate tab definition in pane manifest");
}
pane_manifest.insert(tab_index, panes);
}
Ok(Event::PaneUpdate(PaneManifest {
panes: pane_manifest,
}))
},
_ => Err("Malformed payload for the PaneUpdate Event"),
},
Some(ProtobufEventType::Key) => match protobuf_event.payload {
Some(ProtobufEventPayload::KeyPayload(protobuf_key)) => {
Ok(Event::Key(protobuf_key.try_into()?))
},
_ => Err("Malformed payload for the Key Event"),
},
Some(ProtobufEventType::Mouse) => match protobuf_event.payload {
Some(ProtobufEventPayload::MouseEventPayload(protobuf_mouse)) => {
Ok(Event::Mouse(protobuf_mouse.try_into()?))
},
_ => Err("Malformed payload for the Mouse Event"),
},
Some(ProtobufEventType::Timer) => match protobuf_event.payload {
Some(ProtobufEventPayload::TimerPayload(seconds)) => {
Ok(Event::Timer(seconds as f64))
},
_ => Err("Malformed payload for the Timer Event"),
},
Some(ProtobufEventType::CopyToClipboard) => match protobuf_event.payload {
Some(ProtobufEventPayload::CopyToClipboardPayload(copy_to_clipboard)) => {
let protobuf_copy_to_clipboard =
ProtobufCopyDestination::from_i32(copy_to_clipboard)
.ok_or("Malformed copy to clipboard payload")?;
Ok(Event::CopyToClipboard(
protobuf_copy_to_clipboard.try_into()?,
))
},
_ => Err("Malformed payload for the Copy To Clipboard Event"),
},
Some(ProtobufEventType::SystemClipboardFailure) => match protobuf_event.payload {
None => Ok(Event::SystemClipboardFailure),
_ => Err("Malformed payload for the system clipboard failure Event"),
},
Some(ProtobufEventType::InputReceived) => match protobuf_event.payload {
None => Ok(Event::InputReceived),
_ => Err("Malformed payload for the input received Event"),
},
Some(ProtobufEventType::Visible) => match protobuf_event.payload {
Some(ProtobufEventPayload::VisiblePayload(is_visible)) => {
Ok(Event::Visible(is_visible))
},
_ => Err("Malformed payload for the visible Event"),
},
Some(ProtobufEventType::CustomMessage) => match protobuf_event.payload {
Some(ProtobufEventPayload::CustomMessagePayload(custom_message_payload)) => {
Ok(Event::CustomMessage(
custom_message_payload.message_name,
custom_message_payload.payload,
))
},
_ => Err("Malformed payload for the custom message Event"),
},
Some(ProtobufEventType::FileSystemCreate) => match protobuf_event.payload {
Some(ProtobufEventPayload::FileListPayload(file_list_payload)) => {
let file_paths = file_list_payload
.paths
.iter()
.zip(file_list_payload.paths_metadata.iter())
.map(|(p, m)| (PathBuf::from(p), m.into()))
.collect();
Ok(Event::FileSystemCreate(file_paths))
},
_ => Err("Malformed payload for the file system create Event"),
},
Some(ProtobufEventType::FileSystemRead) => match protobuf_event.payload {
Some(ProtobufEventPayload::FileListPayload(file_list_payload)) => {
let file_paths = file_list_payload
.paths
.iter()
.zip(file_list_payload.paths_metadata.iter())
.map(|(p, m)| (PathBuf::from(p), m.into()))
.collect();
Ok(Event::FileSystemRead(file_paths))
},
_ => Err("Malformed payload for the file system read Event"),
},
Some(ProtobufEventType::FileSystemUpdate) => match protobuf_event.payload {
Some(ProtobufEventPayload::FileListPayload(file_list_payload)) => {
let file_paths = file_list_payload
.paths
.iter()
.zip(file_list_payload.paths_metadata.iter())
.map(|(p, m)| (PathBuf::from(p), m.into()))
.collect();
Ok(Event::FileSystemUpdate(file_paths))
},
_ => Err("Malformed payload for the file system update Event"),
},
Some(ProtobufEventType::FileSystemDelete) => match protobuf_event.payload {
Some(ProtobufEventPayload::FileListPayload(file_list_payload)) => {
let file_paths = file_list_payload
.paths
.iter()
.zip(file_list_payload.paths_metadata.iter())
.map(|(p, m)| (PathBuf::from(p), m.into()))
.collect();
Ok(Event::FileSystemDelete(file_paths))
},
_ => Err("Malformed payload for the file system delete Event"),
},
Some(ProtobufEventType::PermissionRequestResult) => match protobuf_event.payload {
Some(ProtobufEventPayload::PermissionRequestResultPayload(payload)) => {
if payload.granted {
Ok(Event::PermissionRequestResult(PermissionStatus::Granted))
} else {
Ok(Event::PermissionRequestResult(PermissionStatus::Denied))
}
},
_ => Err("Malformed payload for the file system delete Event"),
},
Some(ProtobufEventType::SessionUpdate) => match protobuf_event.payload {
Some(ProtobufEventPayload::SessionUpdatePayload(
protobuf_session_update_payload,
)) => {
let mut session_infos: Vec<SessionInfo> = vec![];
let mut resurrectable_sessions: Vec<(String, Duration)> = vec![];
for protobuf_session_info in protobuf_session_update_payload.session_manifests {
session_infos.push(SessionInfo::try_from(protobuf_session_info)?);
}
for protobuf_resurrectable_session in
protobuf_session_update_payload.resurrectable_sessions
{
resurrectable_sessions.push(protobuf_resurrectable_session.into());
}
Ok(Event::SessionUpdate(
session_infos,
resurrectable_sessions.into(),
))
},
_ => Err("Malformed payload for the SessionUpdate Event"),
},
Some(ProtobufEventType::RunCommandResult) => match protobuf_event.payload {
Some(ProtobufEventPayload::RunCommandResultPayload(run_command_result_payload)) => {
Ok(Event::RunCommandResult(
run_command_result_payload.exit_code,
run_command_result_payload.stdout,
run_command_result_payload.stderr,
run_command_result_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
))
},
_ => Err("Malformed payload for the RunCommandResult Event"),
},
Some(ProtobufEventType::WebRequestResult) => match protobuf_event.payload {
Some(ProtobufEventPayload::WebRequestResultPayload(web_request_result_payload)) => {
Ok(Event::WebRequestResult(
web_request_result_payload.status as u16,
web_request_result_payload
.headers
.into_iter()
.map(|h| (h.name, h.value))
.collect(),
web_request_result_payload.body,
web_request_result_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
))
},
_ => Err("Malformed payload for the WebRequestResult Event"),
},
Some(ProtobufEventType::CommandPaneOpened) => match protobuf_event.payload {
Some(ProtobufEventPayload::CommandPaneOpenedPayload(
command_pane_opened_payload,
)) => Ok(Event::CommandPaneOpened(
command_pane_opened_payload.terminal_pane_id,
command_pane_opened_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
)),
_ => Err("Malformed payload for the CommandPaneOpened Event"),
},
Some(ProtobufEventType::CommandPaneExited) => match protobuf_event.payload {
Some(ProtobufEventPayload::CommandPaneExitedPayload(
command_pane_exited_payload,
)) => Ok(Event::CommandPaneExited(
command_pane_exited_payload.terminal_pane_id,
command_pane_exited_payload.exit_code,
command_pane_exited_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
)),
_ => Err("Malformed payload for the CommandPaneExited Event"),
},
Some(ProtobufEventType::PaneClosed) => match protobuf_event.payload {
Some(ProtobufEventPayload::PaneClosedPayload(pane_closed_payload)) => {
let pane_id = pane_closed_payload
.pane_id
.ok_or("Malformed payload for the PaneClosed Event")?;
Ok(Event::PaneClosed(PaneId::try_from(pane_id)?))
},
_ => Err("Malformed payload for the PaneClosed Event"),
},
Some(ProtobufEventType::EditPaneOpened) => match protobuf_event.payload {
Some(ProtobufEventPayload::EditPaneOpenedPayload(command_pane_opened_payload)) => {
Ok(Event::EditPaneOpened(
command_pane_opened_payload.terminal_pane_id,
command_pane_opened_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
))
},
_ => Err("Malformed payload for the EditPaneOpened Event"),
},
Some(ProtobufEventType::EditPaneExited) => match protobuf_event.payload {
Some(ProtobufEventPayload::EditPaneExitedPayload(command_pane_exited_payload)) => {
Ok(Event::EditPaneExited(
command_pane_exited_payload.terminal_pane_id,
command_pane_exited_payload.exit_code,
command_pane_exited_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
))
},
_ => Err("Malformed payload for the EditPaneExited Event"),
},
Some(ProtobufEventType::CommandPaneReRun) => match protobuf_event.payload {
Some(ProtobufEventPayload::CommandPaneRerunPayload(command_pane_rerun_payload)) => {
Ok(Event::CommandPaneReRun(
command_pane_rerun_payload.terminal_pane_id,
command_pane_rerun_payload
.context
.into_iter()
.map(|c_i| (c_i.name, c_i.value))
.collect(),
))
},
_ => Err("Malformed payload for the CommandPaneReRun Event"),
},
Some(ProtobufEventType::FailedToWriteConfigToDisk) => match protobuf_event.payload {
Some(ProtobufEventPayload::FailedToWriteConfigToDiskPayload(
failed_to_write_configuration_payload,
)) => Ok(Event::FailedToWriteConfigToDisk(
failed_to_write_configuration_payload.file_path,
)),
_ => Err("Malformed payload for the FailedToWriteConfigToDisk Event"),
},
Some(ProtobufEventType::ListClients) => match protobuf_event.payload {
Some(ProtobufEventPayload::ListClientsPayload(mut list_clients_payload)) => {
Ok(Event::ListClients(
list_clients_payload
.client_info
.drain(..)
.filter_map(|c| c.try_into().ok())
.collect(),
))
},
_ => Err("Malformed payload for the FailedToWriteConfigToDisk Event"),
},
Some(ProtobufEventType::HostFolderChanged) => match protobuf_event.payload {
Some(ProtobufEventPayload::HostFolderChangedPayload(
host_folder_changed_payload,
)) => Ok(Event::HostFolderChanged(PathBuf::from(
host_folder_changed_payload.new_host_folder_path,
))),
_ => Err("Malformed payload for the HostFolderChanged Event"),
},
Some(ProtobufEventType::FailedToChangeHostFolder) => match protobuf_event.payload {
Some(ProtobufEventPayload::FailedToChangeHostFolderPayload(
failed_to_change_host_folder_payload,
)) => Ok(Event::FailedToChangeHostFolder(
failed_to_change_host_folder_payload.error_message,
)),
_ => Err("Malformed payload for the FailedToChangeHostFolder Event"),
},
Some(ProtobufEventType::PastedText) => match protobuf_event.payload {
Some(ProtobufEventPayload::PastedTextPayload(pasted_text_payload)) => {
Ok(Event::PastedText(pasted_text_payload.pasted_text))
},
_ => Err("Malformed payload for the PastedText Event"),
},
Some(ProtobufEventType::ConfigWasWrittenToDisk) => match protobuf_event.payload {
None => Ok(Event::ConfigWasWrittenToDisk),
_ => Err("Malformed payload for the ConfigWasWrittenToDisk Event"),
},
Some(ProtobufEventType::WebServerStatus) => match protobuf_event.payload {
Some(ProtobufEventPayload::WebServerStatusPayload(web_server_status)) => {
Ok(Event::WebServerStatus(web_server_status.try_into()?))
},
_ => Err("Malformed payload for the WebServerStatus Event"),
},
Some(ProtobufEventType::BeforeClose) => match protobuf_event.payload {
None => Ok(Event::BeforeClose),
_ => Err("Malformed payload for the BeforeClose Event"),
},
Some(ProtobufEventType::FailedToStartWebServer) => match protobuf_event.payload {
Some(ProtobufEventPayload::FailedToStartWebServerPayload(
failed_to_start_web_server_payload,
)) => Ok(Event::FailedToStartWebServer(
failed_to_start_web_server_payload.error,
)),
_ => Err("Malformed payload for the FailedToStartWebServer Event"),
},
Some(ProtobufEventType::InterceptedKeyPress) => match protobuf_event.payload {
Some(ProtobufEventPayload::KeyPayload(protobuf_key)) => {
Ok(Event::InterceptedKeyPress(protobuf_key.try_into()?))
},
_ => Err("Malformed payload for the InterceptedKeyPress Event"),
},
Some(ProtobufEventType::PaneRenderReport) => match protobuf_event.payload {
Some(ProtobufEventPayload::PaneRenderReportPayload(protobuf_payload)) => {
Ok(Event::PaneRenderReport(protobuf_payload.try_into()?))
},
_ => Err("Malformed payload for the PaneRenderReport Event"),
},
Some(ProtobufEventType::PaneRenderReportWithAnsi) => match protobuf_event.payload {
Some(ProtobufEventPayload::PaneRenderReportWithAnsiPayload(protobuf_payload)) => {
Ok(Event::PaneRenderReportWithAnsi(
protobuf_payload.try_into()?,
))
},
_ => Err("Malformed payload for the PaneRenderReportWithAnsi Event"),
},
Some(ProtobufEventType::UserAction) => match protobuf_event.payload {
Some(ProtobufEventPayload::UserActionPayload(protobuf_payload)) => {
let action: Action = protobuf_payload
.action
.ok_or("Missing action in UserAction payload")?
.try_into()
.map_err(|_| "Failed to convert Action in UserAction payload")?;
let client_id = protobuf_payload.client_id as u16;
let terminal_id = protobuf_payload.terminal_id;
let cli_client_id = protobuf_payload.cli_client_id.map(|id| id as u16);
Ok(Event::UserAction(
action,
client_id,
terminal_id,
cli_client_id,
))
},
_ => Err("Malformed payload for the UserAction Event"),
},
Some(ProtobufEventType::ActionComplete) => match protobuf_event.payload {
Some(ProtobufEventPayload::ActionCompletePayload(protobuf_payload)) => {
let action: Action = protobuf_payload
.action
.ok_or("Missing action in ActionComplete payload")?
.try_into()
.map_err(|_| "Failed to convert Action in ActionComplete payload")?;
let pane_id = protobuf_payload
.pane_id
.map(|id| id.try_into())
.transpose()
.map_err(|_| "Failed to convert PaneId in ActionComplete payload")?;
let context: BTreeMap<String, String> = protobuf_payload
.context
.into_iter()
.map(|item| (item.name, item.value))
.collect();
Ok(Event::ActionComplete(action, pane_id, context))
},
_ => Err("Malformed payload for the ActionComplete Event"),
},
Some(ProtobufEventType::CwdChanged) => match protobuf_event.payload {
Some(ProtobufEventPayload::CwdChangedPayload(protobuf_payload)) => {
let pane_id: PaneId = protobuf_payload
.pane_id
.ok_or("Missing pane_id in CwdChanged payload")?
.try_into()
.map_err(|_| "Failed to convert PaneId in CwdChanged payload")?;
let new_cwd = PathBuf::from(protobuf_payload.new_cwd);
let focused_client_ids: Vec<ClientId> = protobuf_payload
.focused_client_ids
.into_iter()
.map(|id| id as u16)
.collect();
Ok(Event::CwdChanged(pane_id, new_cwd, focused_client_ids))
},
_ => Err("Malformed payload for the CwdChanged Event"),
},
Some(ProtobufEventType::AvailableLayoutInfo) => match protobuf_event.payload {
Some(ProtobufEventPayload::AvailableLayoutInfoPayload(
available_layout_info_payload,
)) => {
let mut available_layouts: Vec<LayoutInfo> = vec![];
let mut layouts_with_errors: Vec<crate::data::LayoutWithError> = vec![];
for protobuf_layout_info in available_layout_info_payload.available_layouts {
available_layouts.push(LayoutInfo::try_from(protobuf_layout_info)?);
}
for protobuf_error in available_layout_info_payload.layouts_with_errors {
layouts_with_errors
.push(crate::data::LayoutWithError::try_from(protobuf_error)?);
}
Ok(Event::AvailableLayoutInfo(
available_layouts,
layouts_with_errors,
))
},
_ => Err("Malformed payload for the AvailableLayoutInfo Event"),
},
Some(ProtobufEventType::PluginConfigurationChanged) => match protobuf_event.payload {
Some(ProtobufEventPayload::PluginConfigurationChangedPayload(payload)) => {
let configuration = payload
.configuration
.into_iter()
.map(|item| (item.name, item.value))
.collect();
Ok(Event::PluginConfigurationChanged(configuration))
},
_ => Err("Malformed payload for PluginConfigurationChanged Event"),
},
Some(ProtobufEventType::HighlightClicked) => match protobuf_event.payload {
Some(ProtobufEventPayload::HighlightClickedPayload(p)) => {
let pane_id = p
.pane_id
.ok_or("Missing pane_id in HighlightClicked")?
.try_into()?;
let context = p
.context
.into_iter()
.map(|item| (item.name, item.value))
.collect();
Ok(Event::HighlightClicked {
pane_id,
pattern: p.pattern,
matched_string: p.matched_string,
context,
})
},
_ => Err("Malformed payload for HighlightClicked Event"),
},
Some(ProtobufEventType::InitialKeybinds) => match protobuf_event.payload {
Some(ProtobufEventPayload::InitialKeybindsPayload(p)) => {
let keybinds = p
.keybinds
.into_iter()
.filter_map(|imk| {
let mode: InputMode =
ProtobufInputMode::from_i32(imk.mode)?.try_into().ok()?;
let key_binds: Vec<(KeyWithModifier, Vec<Action>)> = imk
.key_bind
.into_iter()
.filter_map(|kb| {
let key: KeyWithModifier = kb.key?.try_into().ok()?;
let actions: Vec<Action> = kb
.action
.into_iter()
.filter_map(|a| a.try_into().ok())
.collect();
Some((key, actions))
})
.collect();
Some((mode, key_binds))
})
.collect();
Ok(Event::InitialKeybinds(keybinds))
},
_ => Err("Malformed payload for InitialKeybinds Event"),
},
None => Err("Unknown Protobuf Event"),
}
}
}
impl TryFrom<ProtobufClientInfo> for ClientInfo {
type Error = &'static str;
fn try_from(protobuf_client_info: ProtobufClientInfo) -> Result<Self, &'static str> {
Ok(ClientInfo::new(
protobuf_client_info.client_id as u16,
protobuf_client_info
.pane_id
.ok_or("No pane id found")?
.try_into()?,
protobuf_client_info.running_command,
protobuf_client_info.is_current_client,
))
}
}
impl TryFrom<ClientInfo> for ProtobufClientInfo {
type Error = &'static str;
fn try_from(client_info: ClientInfo) -> Result<Self, &'static str> {
Ok(ProtobufClientInfo {
client_id: client_info.client_id as u32,
pane_id: Some(client_info.pane_id.try_into()?),
running_command: client_info.running_command,
is_current_client: client_info.is_current_client,
})
}
}
impl TryFrom<Event> for ProtobufEvent {
type Error = &'static str;
fn try_from(event: Event) -> Result<Self, &'static str> {
match event {
Event::ModeUpdate(mode_info) => {
let protobuf_mode_update_payload = mode_info.try_into()?;
Ok(ProtobufEvent {
name: ProtobufEventType::ModeUpdate as i32,
payload: Some(event::Payload::ModeUpdatePayload(
protobuf_mode_update_payload,
)),
})
},
Event::TabUpdate(tab_infos) => {
let mut protobuf_tab_infos = vec![];
for tab_info in tab_infos {
protobuf_tab_infos.push(tab_info.try_into()?);
}
let tab_update_payload = TabUpdatePayload {
tab_info: protobuf_tab_infos,
};
Ok(ProtobufEvent {
name: ProtobufEventType::TabUpdate as i32,
payload: Some(event::Payload::TabUpdatePayload(tab_update_payload)),
})
},
Event::PaneUpdate(pane_manifest) => {
let mut protobuf_pane_manifests = vec![];
for (tab_index, pane_infos) in pane_manifest.panes {
let mut protobuf_pane_infos = vec![];
for pane_info in pane_infos {
protobuf_pane_infos.push(pane_info.try_into()?);
}
protobuf_pane_manifests.push(ProtobufPaneManifest {
tab_index: tab_index as u32,
panes: protobuf_pane_infos,
});
}
Ok(ProtobufEvent {
name: ProtobufEventType::PaneUpdate as i32,
payload: Some(event::Payload::PaneUpdatePayload(PaneUpdatePayload {
pane_manifest: protobuf_pane_manifests,
})),
})
},
Event::Key(key) => Ok(ProtobufEvent {
name: ProtobufEventType::Key as i32,
payload: Some(event::Payload::KeyPayload(key.try_into()?)),
}),
Event::Mouse(mouse_event) => {
let protobuf_mouse_payload = mouse_event.try_into()?;
Ok(ProtobufEvent {
name: ProtobufEventType::Mouse as i32,
payload: Some(event::Payload::MouseEventPayload(protobuf_mouse_payload)),
})
},
Event::Timer(seconds) => Ok(ProtobufEvent {
name: ProtobufEventType::Timer as i32,
payload: Some(event::Payload::TimerPayload(seconds as f32)),
}),
Event::CopyToClipboard(clipboard_destination) => {
let protobuf_copy_destination: ProtobufCopyDestination =
clipboard_destination.try_into()?;
Ok(ProtobufEvent {
name: ProtobufEventType::CopyToClipboard as i32,
payload: Some(event::Payload::CopyToClipboardPayload(
protobuf_copy_destination as i32,
)),
})
},
Event::SystemClipboardFailure => Ok(ProtobufEvent {
name: ProtobufEventType::SystemClipboardFailure as i32,
payload: None,
}),
Event::InputReceived => Ok(ProtobufEvent {
name: ProtobufEventType::InputReceived as i32,
payload: None,
}),
Event::Visible(is_visible) => Ok(ProtobufEvent {
name: ProtobufEventType::Visible as i32,
payload: Some(event::Payload::VisiblePayload(is_visible)),
}),
Event::CustomMessage(message, payload) => Ok(ProtobufEvent {
name: ProtobufEventType::CustomMessage as i32,
payload: Some(event::Payload::CustomMessagePayload(CustomMessagePayload {
message_name: message,
payload,
})),
}),
Event::FileSystemCreate(event_paths) => {
let mut paths = vec![];
let mut paths_metadata = vec![];
for (path, path_metadata) in event_paths {
paths.push(path.display().to_string());
paths_metadata.push(path_metadata.into());
}
let file_list_payload = FileListPayload {
paths,
paths_metadata,
};
Ok(ProtobufEvent {
name: ProtobufEventType::FileSystemCreate as i32,
payload: Some(event::Payload::FileListPayload(file_list_payload)),
})
},
Event::FileSystemRead(event_paths) => {
let mut paths = vec![];
let mut paths_metadata = vec![];
for (path, path_metadata) in event_paths {
paths.push(path.display().to_string());
paths_metadata.push(path_metadata.into());
}
let file_list_payload = FileListPayload {
paths,
paths_metadata,
};
Ok(ProtobufEvent {
name: ProtobufEventType::FileSystemRead as i32,
payload: Some(event::Payload::FileListPayload(file_list_payload)),
})
},
Event::FileSystemUpdate(event_paths) => {
let mut paths = vec![];
let mut paths_metadata = vec![];
for (path, path_metadata) in event_paths {
paths.push(path.display().to_string());
paths_metadata.push(path_metadata.into());
}
let file_list_payload = FileListPayload {
paths,
paths_metadata,
};
Ok(ProtobufEvent {
name: ProtobufEventType::FileSystemUpdate as i32,
payload: Some(event::Payload::FileListPayload(file_list_payload)),
})
},
Event::FileSystemDelete(event_paths) => {
let mut paths = vec![];
let mut paths_metadata = vec![];
for (path, path_metadata) in event_paths {
paths.push(path.display().to_string());
paths_metadata.push(path_metadata.into());
}
let file_list_payload = FileListPayload {
paths,
paths_metadata,
};
Ok(ProtobufEvent {
name: ProtobufEventType::FileSystemDelete as i32,
payload: Some(event::Payload::FileListPayload(file_list_payload)),
})
},
Event::PermissionRequestResult(permission_status) => {
let granted = match permission_status {
PermissionStatus::Granted => true,
PermissionStatus::Denied => false,
};
Ok(ProtobufEvent {
name: ProtobufEventType::PermissionRequestResult as i32,
payload: Some(event::Payload::PermissionRequestResultPayload(
PermissionRequestResultPayload { granted },
)),
})
},
Event::SessionUpdate(session_infos, resurrectable_sessions) => {
let mut protobuf_session_manifests = vec![];
for session_info in session_infos {
protobuf_session_manifests.push(session_info.try_into()?);
}
let mut protobuf_resurrectable_sessions = vec![];
for resurrectable_session in resurrectable_sessions {
protobuf_resurrectable_sessions.push(resurrectable_session.into());
}
let session_update_payload = SessionUpdatePayload {
session_manifests: protobuf_session_manifests,
resurrectable_sessions: protobuf_resurrectable_sessions,
};
Ok(ProtobufEvent {
name: ProtobufEventType::SessionUpdate as i32,
payload: Some(event::Payload::SessionUpdatePayload(session_update_payload)),
})
},
Event::RunCommandResult(exit_code, stdout, stderr, context) => {
let run_command_result_payload = RunCommandResultPayload {
exit_code,
stdout,
stderr,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::RunCommandResult as i32,
payload: Some(event::Payload::RunCommandResultPayload(
run_command_result_payload,
)),
})
},
Event::WebRequestResult(status, headers, body, context) => {
let web_request_result_payload = WebRequestResultPayload {
status: status as i32,
headers: headers
.into_iter()
.map(|(name, value)| Header { name, value })
.collect(),
body,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::WebRequestResult as i32,
payload: Some(event::Payload::WebRequestResultPayload(
web_request_result_payload,
)),
})
},
Event::CommandPaneOpened(terminal_pane_id, context) => {
let command_pane_opened_payload = CommandPaneOpenedPayload {
terminal_pane_id,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::CommandPaneOpened as i32,
payload: Some(event::Payload::CommandPaneOpenedPayload(
command_pane_opened_payload,
)),
})
},
Event::CommandPaneExited(terminal_pane_id, exit_code, context) => {
let command_pane_exited_payload = CommandPaneExitedPayload {
terminal_pane_id,
exit_code,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::CommandPaneExited as i32,
payload: Some(event::Payload::CommandPaneExitedPayload(
command_pane_exited_payload,
)),
})
},
Event::PaneClosed(pane_id) => Ok(ProtobufEvent {
name: ProtobufEventType::PaneClosed as i32,
payload: Some(event::Payload::PaneClosedPayload(PaneClosedPayload {
pane_id: Some(pane_id.try_into()?),
})),
}),
Event::EditPaneOpened(terminal_pane_id, context) => {
let command_pane_opened_payload = EditPaneOpenedPayload {
terminal_pane_id,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::EditPaneOpened as i32,
payload: Some(event::Payload::EditPaneOpenedPayload(
command_pane_opened_payload,
)),
})
},
Event::EditPaneExited(terminal_pane_id, exit_code, context) => {
let command_pane_exited_payload = EditPaneExitedPayload {
terminal_pane_id,
exit_code,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::EditPaneExited as i32,
payload: Some(event::Payload::EditPaneExitedPayload(
command_pane_exited_payload,
)),
})
},
Event::CommandPaneReRun(terminal_pane_id, context) => {
let command_pane_rerun_payload = CommandPaneReRunPayload {
terminal_pane_id,
context: context
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
};
Ok(ProtobufEvent {
name: ProtobufEventType::CommandPaneReRun as i32,
payload: Some(event::Payload::CommandPaneRerunPayload(
command_pane_rerun_payload,
)),
})
},
Event::FailedToWriteConfigToDisk(file_path) => Ok(ProtobufEvent {
name: ProtobufEventType::FailedToWriteConfigToDisk as i32,
payload: Some(event::Payload::FailedToWriteConfigToDiskPayload(
FailedToWriteConfigToDiskPayload { file_path },
)),
}),
Event::ListClients(mut client_info_list) => Ok(ProtobufEvent {
name: ProtobufEventType::ListClients as i32,
payload: Some(event::Payload::ListClientsPayload(ListClientsPayload {
client_info: client_info_list
.drain(..)
.filter_map(|c| c.try_into().ok())
.collect(),
})),
}),
Event::HostFolderChanged(new_host_folder_path) => Ok(ProtobufEvent {
name: ProtobufEventType::HostFolderChanged as i32,
payload: Some(event::Payload::HostFolderChangedPayload(
HostFolderChangedPayload {
new_host_folder_path: new_host_folder_path.display().to_string(),
},
)),
}),
Event::FailedToChangeHostFolder(error_message) => Ok(ProtobufEvent {
name: ProtobufEventType::FailedToChangeHostFolder as i32,
payload: Some(event::Payload::FailedToChangeHostFolderPayload(
FailedToChangeHostFolderPayload { error_message },
)),
}),
Event::PastedText(pasted_text) => Ok(ProtobufEvent {
name: ProtobufEventType::PastedText as i32,
payload: Some(event::Payload::PastedTextPayload(PastedTextPayload {
pasted_text,
})),
}),
Event::ConfigWasWrittenToDisk => Ok(ProtobufEvent {
name: ProtobufEventType::ConfigWasWrittenToDisk as i32,
payload: None,
}),
Event::WebServerStatus(web_server_status) => Ok(ProtobufEvent {
name: ProtobufEventType::WebServerStatus as i32,
payload: Some(event::Payload::WebServerStatusPayload(
ProtobufWebServerStatusPayload::try_from(web_server_status)?,
)),
}),
Event::BeforeClose => Ok(ProtobufEvent {
name: ProtobufEventType::BeforeClose as i32,
payload: None,
}),
Event::FailedToStartWebServer(error) => Ok(ProtobufEvent {
name: ProtobufEventType::FailedToStartWebServer as i32,
payload: Some(event::Payload::FailedToStartWebServerPayload(
FailedToStartWebServerPayload { error },
)),
}),
Event::InterceptedKeyPress(key) => Ok(ProtobufEvent {
name: ProtobufEventType::InterceptedKeyPress as i32,
payload: Some(event::Payload::KeyPayload(key.try_into()?)),
}),
Event::PaneRenderReport(pane_contents_map) => Ok(ProtobufEvent {
name: ProtobufEventType::PaneRenderReport as i32,
payload: Some(event::Payload::PaneRenderReportPayload(
pane_contents_map.try_into()?,
)),
}),
Event::PaneRenderReportWithAnsi(pane_contents_map) => Ok(ProtobufEvent {
name: ProtobufEventType::PaneRenderReportWithAnsi as i32,
payload: Some(event::Payload::PaneRenderReportWithAnsiPayload(
pane_contents_map.try_into()?,
)),
}),
Event::UserAction(action, client_id, terminal_id, cli_client_id) => {
let protobuf_action: ProtobufAction = action
.try_into()
.map_err(|_| "Failed to convert Action to protobuf")?;
let protobuf_payload = ProtobufUserActionPayload {
action: Some(protobuf_action),
client_id: client_id as u32,
terminal_id,
cli_client_id: cli_client_id.map(|id| id as u32),
};
Ok(ProtobufEvent {
name: ProtobufEventType::UserAction as i32,
payload: Some(event::Payload::UserActionPayload(protobuf_payload)),
})
},
Event::ActionComplete(action, pane_id, context) => {
let protobuf_action = action.try_into()?;
let protobuf_pane_id = pane_id.map(|id| id.try_into()).transpose()?;
let context_items: Vec<ProtobufContextItem> = context
.into_iter()
.map(|(name, value)| ProtobufContextItem { name, value })
.collect();
let action_complete_payload = ProtobufActionCompletePayload {
action: Some(protobuf_action),
pane_id: protobuf_pane_id,
context: context_items,
};
Ok(ProtobufEvent {
name: ProtobufEventType::ActionComplete as i32,
payload: Some(event::Payload::ActionCompletePayload(
action_complete_payload,
)),
})
},
Event::CwdChanged(pane_id, new_cwd, focused_client_ids) => {
let protobuf_pane_id: ProtobufPaneId = pane_id.try_into()?;
let new_cwd_string = new_cwd
.to_str()
.ok_or("Failed to convert PathBuf to string")?
.to_string();
let focused_client_ids_u32: Vec<u32> =
focused_client_ids.into_iter().map(|id| id as u32).collect();
let cwd_changed_payload = ProtobufCwdChangedPayload {
pane_id: Some(protobuf_pane_id),
new_cwd: new_cwd_string,
focused_client_ids: focused_client_ids_u32,
};
Ok(ProtobufEvent {
name: ProtobufEventType::CwdChanged as i32,
payload: Some(event::Payload::CwdChangedPayload(cwd_changed_payload)),
})
},
Event::AvailableLayoutInfo(available_layouts, layouts_with_errors) => {
let mut protobuf_available_layouts = vec![];
let mut protobuf_layouts_with_errors = vec![];
for layout_info in available_layouts {
protobuf_available_layouts.push(layout_info.try_into()?);
}
for layout_error in layouts_with_errors {
protobuf_layouts_with_errors.push(layout_error.try_into()?);
}
let available_layout_info_payload = ProtobufAvailableLayoutInfoPayload {
available_layouts: protobuf_available_layouts,
layouts_with_errors: protobuf_layouts_with_errors,
};
Ok(ProtobufEvent {
name: ProtobufEventType::AvailableLayoutInfo as i32,
payload: Some(event::Payload::AvailableLayoutInfoPayload(
available_layout_info_payload,
)),
})
},
Event::PluginConfigurationChanged(configuration) => {
let configuration_items: Vec<ProtobufContextItem> = configuration
.into_iter()
.map(|(name, value)| ProtobufContextItem { name, value })
.collect();
let payload = ProtobufPluginConfigurationChangedPayload {
configuration: configuration_items,
};
Ok(ProtobufEvent {
name: ProtobufEventType::PluginConfigurationChanged as i32,
payload: Some(event::Payload::PluginConfigurationChangedPayload(payload)),
})
},
Event::HighlightClicked {
pane_id,
pattern,
matched_string,
context,
} => Ok(ProtobufEvent {
name: ProtobufEventType::HighlightClicked as i32,
payload: Some(event::Payload::HighlightClickedPayload(
HighlightClickedPayload {
pane_id: pane_id.try_into().ok(),
pattern,
matched_string,
context: context
.into_iter()
.map(|(name, value)| ProtobufContextItem { name, value })
.collect(),
},
)),
}),
Event::InitialKeybinds(keybinds) => {
let mut protobuf_keybinds: Vec<ProtobufInputModeKeybinds> = vec![];
for (input_mode, input_mode_keybinds) in keybinds {
let mode: ProtobufInputMode = input_mode.try_into()?;
let mut key_binds: Vec<ProtobufKeyBind> = vec![];
for (key, actions) in input_mode_keybinds {
let protobuf_key: ProtobufKey = key.try_into()?;
let mut protobuf_actions: Vec<ProtobufAction> = vec![];
for action in actions {
if let Ok(protobuf_action) = action.try_into() {
protobuf_actions.push(protobuf_action);
}
}
key_binds.push(ProtobufKeyBind {
key: Some(protobuf_key),
action: protobuf_actions,
});
}
protobuf_keybinds.push(ProtobufInputModeKeybinds {
mode: mode as i32,
key_bind: key_binds,
});
}
Ok(ProtobufEvent {
name: ProtobufEventType::InitialKeybinds as i32,
payload: Some(event::Payload::InitialKeybindsPayload(
InitialKeybindsPayload {
keybinds: protobuf_keybinds,
},
)),
})
},
}
}
}
impl TryFrom<SessionInfo> for ProtobufSessionManifest {
type Error = &'static str;
fn try_from(session_info: SessionInfo) -> Result<Self, &'static str> {
let mut protobuf_pane_manifests = vec![];
for (tab_index, pane_infos) in session_info.panes.panes {
let mut protobuf_pane_infos = vec![];
for pane_info in pane_infos {
protobuf_pane_infos.push(pane_info.try_into()?);
}
protobuf_pane_manifests.push(ProtobufPaneManifest {
tab_index: tab_index as u32,
panes: protobuf_pane_infos,
});
}
Ok(ProtobufSessionManifest {
name: session_info.name,
panes: protobuf_pane_manifests,
tabs: session_info
.tabs
.iter()
.filter_map(|t| t.clone().try_into().ok())
.collect(),
connected_clients: session_info.connected_clients as u32,
is_current_session: session_info.is_current_session,
available_layouts: session_info
.available_layouts
.into_iter()
.filter_map(|l| ProtobufLayoutInfo::try_from(l).ok())
.collect(),
plugins: session_info
.plugins
.into_iter()
.map(|p| ProtobufPluginInfo::from(p))
.collect(),
web_clients_allowed: session_info.web_clients_allowed,
web_client_count: session_info.web_client_count as u32,
tab_history: session_info
.tab_history
.into_iter()
.map(|t| ProtobufClientTabHistory::from(t))
.collect(),
pane_history: session_info
.pane_history
.into_iter()
.map(|p| ProtobufClientPaneHistory::from(p))
.collect(),
creation_time: session_info.creation_time.as_secs(),
})
}
}
impl From<(u16, Vec<usize>)> for ProtobufClientTabHistory {
fn from((client_id, tab_history): (u16, Vec<usize>)) -> ProtobufClientTabHistory {
ProtobufClientTabHistory {
client_id: client_id as u32,
tab_history: tab_history.into_iter().map(|t| t as u32).collect(),
}
}
}
impl From<(u16, Vec<PaneId>)> for ProtobufClientPaneHistory {
fn from((client_id, pane_history): (u16, Vec<PaneId>)) -> ProtobufClientPaneHistory {
ProtobufClientPaneHistory {
client_id: client_id as u32,
pane_history: pane_history
.into_iter()
.filter_map(|p| p.try_into().ok())
.collect(),
}
}
}
impl From<(u32, PluginInfo)> for ProtobufPluginInfo {
fn from((plugin_id, plugin_info): (u32, PluginInfo)) -> ProtobufPluginInfo {
ProtobufPluginInfo {
plugin_id,
plugin_url: plugin_info.location,
plugin_config: plugin_info
.configuration
.into_iter()
.map(|(name, value)| ContextItem { name, value })
.collect(),
}
}
}
impl TryFrom<ProtobufSessionManifest> for SessionInfo {
type Error = &'static str;
fn try_from(protobuf_session_manifest: ProtobufSessionManifest) -> Result<Self, &'static str> {
let mut pane_manifest: HashMap<usize, Vec<PaneInfo>> = HashMap::new();
for protobuf_pane_manifest in protobuf_session_manifest.panes {
let tab_index = protobuf_pane_manifest.tab_index as usize;
let mut panes = vec![];
for protobuf_pane_info in protobuf_pane_manifest.panes {
panes.push(protobuf_pane_info.try_into()?);
}
if pane_manifest.contains_key(&tab_index) {
return Err("Duplicate tab definition in pane manifest");
}
pane_manifest.insert(tab_index, panes);
}
let panes = PaneManifest {
panes: pane_manifest,
};
let mut plugins = BTreeMap::new();
for plugin_info in protobuf_session_manifest.plugins.into_iter() {
let mut configuration = BTreeMap::new();
for context_item in plugin_info.plugin_config.into_iter() {
configuration.insert(context_item.name, context_item.value);
}
plugins.insert(
plugin_info.plugin_id,
PluginInfo {
location: plugin_info.plugin_url,
configuration,
},
);
}
let mut tab_history = BTreeMap::new();
for client_tab_history in protobuf_session_manifest.tab_history.into_iter() {
let client_id = client_tab_history.client_id;
let tab_history_for_client = client_tab_history
.tab_history
.iter()
.map(|t| *t as usize)
.collect();
tab_history.insert(client_id as u16, tab_history_for_client);
}
let mut pane_history = BTreeMap::new();
for client_pane_history in protobuf_session_manifest.pane_history.into_iter() {
let client_id = client_pane_history.client_id;
let pane_history_for_client = client_pane_history
.pane_history
.into_iter()
.filter_map(|p| p.try_into().ok())
.collect();
pane_history.insert(client_id as u16, pane_history_for_client);
}
Ok(SessionInfo {
name: protobuf_session_manifest.name,
tabs: protobuf_session_manifest
.tabs
.iter()
.filter_map(|t| t.clone().try_into().ok())
.collect(),
panes,
connected_clients: protobuf_session_manifest.connected_clients as usize,
is_current_session: protobuf_session_manifest.is_current_session,
available_layouts: protobuf_session_manifest
.available_layouts
.into_iter()
.filter_map(|l| LayoutInfo::try_from(l).ok())
.collect(),
plugins,
web_clients_allowed: protobuf_session_manifest.web_clients_allowed,
web_client_count: protobuf_session_manifest.web_client_count as usize,
tab_history,
pane_history,
creation_time: Duration::from_secs(protobuf_session_manifest.creation_time),
})
}
}
impl TryFrom<LayoutInfo> for ProtobufLayoutInfo {
type Error = &'static str;
fn try_from(layout_info: LayoutInfo) -> Result<Self, &'static str> {
match layout_info {
LayoutInfo::File(name, layout_metadata) => Ok(ProtobufLayoutInfo {
source: "file".to_owned(),
name,
layout_metadata: Some(layout_metadata.try_into()?),
}),
LayoutInfo::BuiltIn(name) => Ok(ProtobufLayoutInfo {
source: "built-in".to_owned(),
name,
layout_metadata: None,
}),
LayoutInfo::Url(name) => Ok(ProtobufLayoutInfo {
source: "url".to_owned(),
name,
layout_metadata: None,
}),
LayoutInfo::Stringified(stringified_layout) => Ok(ProtobufLayoutInfo {
source: "stringified".to_owned(),
name: stringified_layout.clone(),
layout_metadata: None,
}),
}
}
}
impl TryFrom<ProtobufLayoutInfo> for LayoutInfo {
type Error = &'static str;
fn try_from(protobuf_layout_info: ProtobufLayoutInfo) -> Result<Self, &'static str> {
match protobuf_layout_info.source.as_str() {
"file" => {
let layout_metadata = protobuf_layout_info
.layout_metadata
.map(|m| m.try_into())
.transpose()?
.unwrap_or_default();
Ok(LayoutInfo::File(protobuf_layout_info.name, layout_metadata))
},
"built-in" => Ok(LayoutInfo::BuiltIn(protobuf_layout_info.name)),
"url" => Ok(LayoutInfo::Url(protobuf_layout_info.name)),
"stringified" => Ok(LayoutInfo::Stringified(protobuf_layout_info.name)),
_ => Err("Unknown source for layout"),
}
}
}
impl TryFrom<ProtobufLayoutMetadata> for LayoutMetadata {
type Error = &'static str;
fn try_from(protobuf_metadata: ProtobufLayoutMetadata) -> Result<Self, &'static str> {
let tabs = protobuf_metadata
.tabs
.into_iter()
.map(|t| t.try_into())
.collect::<Result<Vec<_>, _>>()?;
Ok(LayoutMetadata {
tabs,
creation_time: protobuf_metadata.creation_time,
update_time: protobuf_metadata.update_time,
})
}
}
impl TryFrom<LayoutMetadata> for ProtobufLayoutMetadata {
type Error = &'static str;
fn try_from(metadata: LayoutMetadata) -> Result<Self, &'static str> {
let tabs = metadata
.tabs
.into_iter()
.map(|t| t.try_into())
.collect::<Result<Vec<_>, _>>()?;
Ok(ProtobufLayoutMetadata {
tabs,
creation_time: metadata.creation_time,
update_time: metadata.update_time,
})
}
}
impl TryFrom<ProtobufTabMetadata> for TabMetadata {
type Error = &'static str;
fn try_from(protobuf_metadata: ProtobufTabMetadata) -> Result<Self, &'static str> {
let panes = protobuf_metadata
.pane_metadata
.into_iter()
.map(|p| p.try_into())
.collect::<Result<Vec<_>, _>>()?;
Ok(TabMetadata {
panes,
name: protobuf_metadata.name,
})
}
}
impl TryFrom<TabMetadata> for ProtobufTabMetadata {
type Error = &'static str;
fn try_from(metadata: TabMetadata) -> Result<Self, &'static str> {
let pane_metadata = metadata
.panes
.into_iter()
.map(|p| p.try_into())
.collect::<Result<Vec<_>, _>>()?;
Ok(ProtobufTabMetadata {
pane_metadata,
name: metadata.name,
})
}
}
impl TryFrom<ProtobufPaneMetadata> for PaneMetadata {
type Error = &'static str;
fn try_from(protobuf_metadata: ProtobufPaneMetadata) -> Result<Self, &'static str> {
Ok(PaneMetadata {
name: protobuf_metadata.name,
is_plugin: protobuf_metadata.is_plugin,
is_builtin_plugin: protobuf_metadata.is_builtin_plugin,
})
}
}
impl TryFrom<PaneMetadata> for ProtobufPaneMetadata {
type Error = &'static str;
fn try_from(metadata: PaneMetadata) -> Result<Self, &'static str> {
Ok(ProtobufPaneMetadata {
name: metadata.name,
is_plugin: metadata.is_plugin,
is_builtin_plugin: metadata.is_builtin_plugin,
})
}
}
impl TryFrom<ProtobufLayoutWithError> for crate::data::LayoutWithError {
type Error = &'static str;
fn try_from(protobuf: ProtobufLayoutWithError) -> Result<Self, Self::Error> {
Ok(crate::data::LayoutWithError {
layout_name: protobuf.layout_name,
error: protobuf.error.ok_or("Missing error field")?.try_into()?,
})
}
}
impl TryFrom<crate::data::LayoutWithError> for ProtobufLayoutWithError {
type Error = &'static str;
fn try_from(layout_error: crate::data::LayoutWithError) -> Result<Self, Self::Error> {
Ok(ProtobufLayoutWithError {
layout_name: layout_error.layout_name,
error: Some(layout_error.error.try_into()?),
})
}
}
impl TryFrom<ProtobufLayoutParsingError> for crate::data::LayoutParsingError {
type Error = &'static str;
fn try_from(protobuf: ProtobufLayoutParsingError) -> Result<Self, Self::Error> {
match protobuf.error_type.ok_or("Missing error_type")? {
ProtobufLayoutParsingErrorType::KdlError(kdl_variant) => {
Ok(crate::data::LayoutParsingError::KdlError {
kdl_error: kdl_variant
.kdl_error
.ok_or("Missing kdl_error")?
.try_into()?,
file_name: kdl_variant.file_name,
source_code: kdl_variant.source_code,
})
},
ProtobufLayoutParsingErrorType::SyntaxError(_) => {
Ok(crate::data::LayoutParsingError::SyntaxError)
},
}
}
}
impl TryFrom<crate::data::LayoutParsingError> for ProtobufLayoutParsingError {
type Error = &'static str;
fn try_from(error: crate::data::LayoutParsingError) -> Result<Self, Self::Error> {
let error_type = match error {
crate::data::LayoutParsingError::KdlError {
kdl_error,
file_name,
source_code,
} => ProtobufLayoutParsingErrorType::KdlError(ProtobufKdlErrorVariant {
kdl_error: Some(kdl_error.try_into()?),
file_name,
source_code,
}),
crate::data::LayoutParsingError::SyntaxError => {
ProtobufLayoutParsingErrorType::SyntaxError(ProtobufSyntaxError {})
},
};
Ok(ProtobufLayoutParsingError {
error_type: Some(error_type),
})
}
}
impl TryFrom<ProtobufKdlError> for crate::input::config::KdlError {
type Error = &'static str;
fn try_from(protobuf: ProtobufKdlError) -> Result<Self, Self::Error> {
Ok(crate::input::config::KdlError {
error_message: protobuf.error_message,
src: None, offset: protobuf.offset.map(|o| o as usize),
len: protobuf.len.map(|l| l as usize),
help_message: protobuf.help_message,
})
}
}
impl TryFrom<crate::input::config::KdlError> for ProtobufKdlError {
type Error = &'static str;
fn try_from(kdl: crate::input::config::KdlError) -> Result<Self, Self::Error> {
Ok(ProtobufKdlError {
error_message: kdl.error_message,
offset: kdl.offset.map(|o| o as u64),
len: kdl.len.map(|l| l as u64),
help_message: kdl.help_message,
})
}
}
impl TryFrom<CopyDestination> for ProtobufCopyDestination {
type Error = &'static str;
fn try_from(copy_destination: CopyDestination) -> Result<Self, &'static str> {
match copy_destination {
CopyDestination::Command => Ok(ProtobufCopyDestination::Command),
CopyDestination::Primary => Ok(ProtobufCopyDestination::Primary),
CopyDestination::System => Ok(ProtobufCopyDestination::System),
}
}
}
impl TryFrom<ProtobufCopyDestination> for CopyDestination {
type Error = &'static str;
fn try_from(protobuf_copy_destination: ProtobufCopyDestination) -> Result<Self, &'static str> {
match protobuf_copy_destination {
ProtobufCopyDestination::Command => Ok(CopyDestination::Command),
ProtobufCopyDestination::Primary => Ok(CopyDestination::Primary),
ProtobufCopyDestination::System => Ok(CopyDestination::System),
}
}
}
impl TryFrom<MouseEventPayload> for Mouse {
type Error = &'static str;
fn try_from(mouse_event_payload: MouseEventPayload) -> Result<Self, &'static str> {
match MouseEventName::from_i32(mouse_event_payload.mouse_event_name) {
Some(MouseEventName::MouseScrollUp) => match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::LineCount(line_count)) => {
Ok(Mouse::ScrollUp(line_count as usize))
},
_ => Err("Malformed payload for mouse scroll up"),
},
Some(MouseEventName::MouseScrollDown) => {
match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::LineCount(line_count)) => {
Ok(Mouse::ScrollDown(line_count as usize))
},
_ => Err("Malformed payload for mouse scroll down"),
}
},
Some(MouseEventName::MouseLeftClick) => match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::Position(position)) => Ok(
Mouse::LeftClick(position.line as isize, position.column as usize),
),
_ => Err("Malformed payload for mouse left click"),
},
Some(MouseEventName::MouseRightClick) => {
match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::Position(position)) => Ok(
Mouse::RightClick(position.line as isize, position.column as usize),
),
_ => Err("Malformed payload for mouse right click"),
}
},
Some(MouseEventName::MouseHold) => match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::Position(position)) => Ok(
Mouse::Hold(position.line as isize, position.column as usize),
),
_ => Err("Malformed payload for mouse hold"),
},
Some(MouseEventName::MouseRelease) => match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::Position(position)) => Ok(
Mouse::Release(position.line as isize, position.column as usize),
),
_ => Err("Malformed payload for mouse release"),
},
Some(MouseEventName::MouseHover) => match mouse_event_payload.mouse_event_payload {
Some(mouse_event_payload::MouseEventPayload::Position(position)) => Ok(
Mouse::Hover(position.line as isize, position.column as usize),
),
_ => Err("Malformed payload for mouse hover"),
},
None => Err("Malformed payload for MouseEventName"),
}
}
}
impl TryFrom<Mouse> for MouseEventPayload {
type Error = &'static str;
fn try_from(mouse: Mouse) -> Result<Self, &'static str> {
match mouse {
Mouse::ScrollUp(number_of_lines) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseScrollUp as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::LineCount(
number_of_lines as u32,
)),
}),
Mouse::ScrollDown(number_of_lines) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseScrollDown as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::LineCount(
number_of_lines as u32,
)),
}),
Mouse::LeftClick(line, column) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseLeftClick as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::Position(
ProtobufPosition {
line: line as i64,
column: column as i64,
},
)),
}),
Mouse::RightClick(line, column) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseRightClick as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::Position(
ProtobufPosition {
line: line as i64,
column: column as i64,
},
)),
}),
Mouse::Hold(line, column) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseHold as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::Position(
ProtobufPosition {
line: line as i64,
column: column as i64,
},
)),
}),
Mouse::Release(line, column) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseRelease as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::Position(
ProtobufPosition {
line: line as i64,
column: column as i64,
},
)),
}),
Mouse::Hover(line, column) => Ok(MouseEventPayload {
mouse_event_name: MouseEventName::MouseHover as i32,
mouse_event_payload: Some(mouse_event_payload::MouseEventPayload::Position(
ProtobufPosition {
line: line as i64,
column: column as i64,
},
)),
}),
}
}
}
impl TryFrom<ProtobufPaneInfo> for PaneInfo {
type Error = &'static str;
fn try_from(protobuf_pane_info: ProtobufPaneInfo) -> Result<Self, &'static str> {
Ok(PaneInfo {
id: protobuf_pane_info.id,
is_plugin: protobuf_pane_info.is_plugin,
is_focused: protobuf_pane_info.is_focused,
is_fullscreen: protobuf_pane_info.is_fullscreen,
is_floating: protobuf_pane_info.is_floating,
is_suppressed: protobuf_pane_info.is_suppressed,
title: protobuf_pane_info.title,
exited: protobuf_pane_info.exited,
exit_status: protobuf_pane_info.exit_status,
is_held: protobuf_pane_info.is_held,
pane_x: protobuf_pane_info.pane_x as usize,
pane_content_x: protobuf_pane_info.pane_content_x as usize,
pane_y: protobuf_pane_info.pane_y as usize,
pane_content_y: protobuf_pane_info.pane_content_y as usize,
pane_rows: protobuf_pane_info.pane_rows as usize,
pane_content_rows: protobuf_pane_info.pane_content_rows as usize,
pane_columns: protobuf_pane_info.pane_columns as usize,
pane_content_columns: protobuf_pane_info.pane_content_columns as usize,
cursor_coordinates_in_pane: protobuf_pane_info
.cursor_coordinates_in_pane
.map(|position| (position.column as usize, position.line as usize)),
terminal_command: protobuf_pane_info.terminal_command,
plugin_url: protobuf_pane_info.plugin_url,
is_selectable: protobuf_pane_info.is_selectable,
index_in_pane_group: protobuf_pane_info
.index_in_pane_group
.iter()
.map(|index_in_pane_group| {
(
index_in_pane_group.client_id as u16,
index_in_pane_group.index as usize,
)
})
.collect(),
default_fg: protobuf_pane_info.default_fg,
default_bg: protobuf_pane_info.default_bg,
})
}
}
impl TryFrom<PaneInfo> for ProtobufPaneInfo {
type Error = &'static str;
fn try_from(pane_info: PaneInfo) -> Result<Self, &'static str> {
Ok(ProtobufPaneInfo {
id: pane_info.id,
is_plugin: pane_info.is_plugin,
is_focused: pane_info.is_focused,
is_fullscreen: pane_info.is_fullscreen,
is_floating: pane_info.is_floating,
is_suppressed: pane_info.is_suppressed,
title: pane_info.title,
exited: pane_info.exited,
exit_status: pane_info.exit_status,
is_held: pane_info.is_held,
pane_x: pane_info.pane_x as u32,
pane_content_x: pane_info.pane_content_x as u32,
pane_y: pane_info.pane_y as u32,
pane_content_y: pane_info.pane_content_y as u32,
pane_rows: pane_info.pane_rows as u32,
pane_content_rows: pane_info.pane_content_rows as u32,
pane_columns: pane_info.pane_columns as u32,
pane_content_columns: pane_info.pane_content_columns as u32,
cursor_coordinates_in_pane: pane_info.cursor_coordinates_in_pane.map(|(x, y)| {
ProtobufPosition {
column: x as i64,
line: y as i64,
}
}),
terminal_command: pane_info.terminal_command,
plugin_url: pane_info.plugin_url,
is_selectable: pane_info.is_selectable,
index_in_pane_group: pane_info
.index_in_pane_group
.iter()
.map(|(&client_id, &index)| IndexInPaneGroup {
client_id: client_id as u32,
index: index as u32,
})
.collect(),
default_fg: pane_info.default_fg,
default_bg: pane_info.default_bg,
})
}
}
impl TryFrom<ProtobufTabInfo> for TabInfo {
type Error = &'static str;
fn try_from(protobuf_tab_info: ProtobufTabInfo) -> Result<Self, &'static str> {
Ok(TabInfo {
position: protobuf_tab_info.position as usize,
name: protobuf_tab_info.name,
active: protobuf_tab_info.active,
panes_to_hide: protobuf_tab_info.panes_to_hide as usize,
is_fullscreen_active: protobuf_tab_info.is_fullscreen_active,
is_sync_panes_active: protobuf_tab_info.is_sync_panes_active,
are_floating_panes_visible: protobuf_tab_info.are_floating_panes_visible,
other_focused_clients: protobuf_tab_info
.other_focused_clients
.iter()
.map(|c| *c as u16)
.collect(),
active_swap_layout_name: protobuf_tab_info.active_swap_layout_name,
is_swap_layout_dirty: protobuf_tab_info.is_swap_layout_dirty,
viewport_rows: protobuf_tab_info.viewport_rows as usize,
viewport_columns: protobuf_tab_info.viewport_columns as usize,
display_area_rows: protobuf_tab_info.display_area_rows as usize,
display_area_columns: protobuf_tab_info.display_area_columns as usize,
selectable_tiled_panes_count: protobuf_tab_info.selectable_tiled_panes_count as usize,
selectable_floating_panes_count: protobuf_tab_info.selectable_floating_panes_count
as usize,
tab_id: protobuf_tab_info.tab_id as usize,
has_bell_notification: protobuf_tab_info.has_bell_notification,
is_flashing_bell: protobuf_tab_info.is_flashing_bell,
})
}
}
impl TryFrom<TabInfo> for ProtobufTabInfo {
type Error = &'static str;
fn try_from(tab_info: TabInfo) -> Result<Self, &'static str> {
Ok(ProtobufTabInfo {
position: tab_info.position as u32,
name: tab_info.name,
active: tab_info.active,
panes_to_hide: tab_info.panes_to_hide as u32,
is_fullscreen_active: tab_info.is_fullscreen_active,
is_sync_panes_active: tab_info.is_sync_panes_active,
are_floating_panes_visible: tab_info.are_floating_panes_visible,
other_focused_clients: tab_info
.other_focused_clients
.iter()
.map(|c| *c as u32)
.collect(),
active_swap_layout_name: tab_info.active_swap_layout_name,
is_swap_layout_dirty: tab_info.is_swap_layout_dirty,
viewport_rows: tab_info.viewport_rows as u32,
viewport_columns: tab_info.viewport_columns as u32,
display_area_rows: tab_info.display_area_rows as u32,
display_area_columns: tab_info.display_area_columns as u32,
selectable_tiled_panes_count: tab_info.selectable_tiled_panes_count as u32,
selectable_floating_panes_count: tab_info.selectable_floating_panes_count as u32,
tab_id: tab_info.tab_id as u32,
has_bell_notification: tab_info.has_bell_notification,
is_flashing_bell: tab_info.is_flashing_bell,
})
}
}
impl TryFrom<ProtobufModeUpdatePayload> for ModeInfo {
type Error = &'static str;
fn try_from(
mut protobuf_mode_update_payload: ProtobufModeUpdatePayload,
) -> Result<Self, &'static str> {
let current_mode: InputMode =
ProtobufInputMode::from_i32(protobuf_mode_update_payload.current_mode)
.ok_or("Malformed InputMode in the ModeUpdate Event")?
.try_into()?;
let base_mode: Option<InputMode> = protobuf_mode_update_payload
.base_mode
.and_then(|b_m| ProtobufInputMode::from_i32(b_m)?.try_into().ok());
let keybinds: Vec<(InputMode, Vec<(KeyWithModifier, Vec<Action>)>)> =
protobuf_mode_update_payload
.keybinds
.iter_mut()
.filter_map(|k| {
let input_mode: InputMode = ProtobufInputMode::from_i32(k.mode)
.ok_or("Malformed InputMode in the ModeUpdate Event")
.ok()?
.try_into()
.ok()?;
let mut keybinds: Vec<(KeyWithModifier, Vec<Action>)> = vec![];
for mut protobuf_keybind in k.key_bind.drain(..) {
let key: KeyWithModifier = protobuf_keybind.key.unwrap().try_into().ok()?;
let mut actions: Vec<Action> = vec![];
for action in protobuf_keybind.action.drain(..) {
if let Ok(action) = action.try_into() {
actions.push(action);
}
}
keybinds.push((key, actions));
}
Some((input_mode, keybinds))
})
.collect();
let style: Style = protobuf_mode_update_payload
.style
.and_then(|m| m.try_into().ok())
.ok_or("malformed payload for mode_info")?;
let session_name = protobuf_mode_update_payload.session_name;
let editor = protobuf_mode_update_payload
.editor
.map(|e| PathBuf::from(e));
let shell = protobuf_mode_update_payload.shell.map(|s| PathBuf::from(s));
let web_clients_allowed = protobuf_mode_update_payload.web_clients_allowed;
let web_sharing = protobuf_mode_update_payload
.web_sharing
.and_then(|w| ProtobufWebSharing::from_i32(w))
.map(|w| w.into());
let capabilities = PluginCapabilities {
arrow_fonts: protobuf_mode_update_payload.arrow_fonts_support,
};
let currently_marking_pane_group =
protobuf_mode_update_payload.currently_marking_pane_group;
let is_web_client = protobuf_mode_update_payload.is_web_client;
let web_server_ip = protobuf_mode_update_payload
.web_server_ip
.as_ref()
.and_then(|web_server_ip| IpAddr::from_str(web_server_ip).ok());
let web_server_port = protobuf_mode_update_payload
.web_server_port
.map(|w| w as u16);
let web_server_capability = protobuf_mode_update_payload.web_server_capability;
let mode_info = ModeInfo {
mode: current_mode,
keybinds,
style,
capabilities,
session_name,
base_mode,
editor,
shell,
web_clients_allowed,
web_sharing,
currently_marking_pane_group,
is_web_client,
web_server_ip,
web_server_port,
web_server_capability,
};
Ok(mode_info)
}
}
impl TryFrom<ModeInfo> for ProtobufModeUpdatePayload {
type Error = &'static str;
fn try_from(mode_info: ModeInfo) -> Result<Self, &'static str> {
let current_mode: ProtobufInputMode = mode_info.mode.try_into()?;
let base_mode: Option<ProtobufInputMode> = mode_info
.base_mode
.and_then(|mode| ProtobufInputMode::try_from(mode).ok());
let style: ProtobufStyle = mode_info.style.try_into()?;
let arrow_fonts_support: bool = mode_info.capabilities.arrow_fonts;
let session_name = mode_info.session_name;
let editor = mode_info.editor.map(|e| e.display().to_string());
let shell = mode_info.shell.map(|s| s.display().to_string());
let web_clients_allowed = mode_info.web_clients_allowed;
let web_sharing = mode_info.web_sharing.map(|w| w as i32);
let currently_marking_pane_group = mode_info.currently_marking_pane_group;
let is_web_client = mode_info.is_web_client;
let web_server_ip = mode_info.web_server_ip.map(|i| format!("{}", i));
let web_server_port = mode_info.web_server_port.map(|p| p as u32);
let web_server_capability = mode_info.web_server_capability;
let mut protobuf_input_mode_keybinds: Vec<ProtobufInputModeKeybinds> = vec![];
for (input_mode, input_mode_keybinds) in mode_info.keybinds {
let mode: ProtobufInputMode = input_mode.try_into()?;
let mut keybinds: Vec<ProtobufKeyBind> = vec![];
for (key, actions) in input_mode_keybinds {
let protobuf_key: ProtobufKey = key.try_into()?;
let mut protobuf_actions: Vec<ProtobufAction> = vec![];
for action in actions {
if let Ok(protobuf_action) = action.try_into() {
protobuf_actions.push(protobuf_action);
}
}
let key_bind = ProtobufKeyBind {
key: Some(protobuf_key),
action: protobuf_actions,
};
keybinds.push(key_bind);
}
let input_mode_keybind = ProtobufInputModeKeybinds {
mode: mode as i32,
key_bind: keybinds,
};
protobuf_input_mode_keybinds.push(input_mode_keybind);
}
Ok(ProtobufModeUpdatePayload {
current_mode: current_mode as i32,
style: Some(style),
keybinds: protobuf_input_mode_keybinds,
arrow_fonts_support,
session_name,
base_mode: base_mode.map(|b_m| b_m as i32),
editor,
shell,
web_clients_allowed,
web_sharing,
currently_marking_pane_group,
is_web_client,
web_server_ip,
web_server_port,
web_server_capability,
})
}
}
impl TryFrom<ProtobufEventNameList> for HashSet<EventType> {
type Error = &'static str;
fn try_from(protobuf_event_name_list: ProtobufEventNameList) -> Result<Self, &'static str> {
let event_types: Vec<ProtobufEventType> = protobuf_event_name_list
.event_types
.iter()
.filter_map(|i| ProtobufEventType::from_i32(*i))
.collect();
let event_types: Vec<EventType> = event_types
.iter()
.filter_map(|e| EventType::try_from(*e).ok())
.collect();
Ok(event_types.into_iter().collect())
}
}
impl TryFrom<HashSet<EventType>> for ProtobufEventNameList {
type Error = &'static str;
fn try_from(event_types: HashSet<EventType>) -> Result<Self, &'static str> {
let protobuf_event_name_list = ProtobufEventNameList {
event_types: event_types
.iter()
.filter_map(|e| ProtobufEventType::try_from(*e).ok())
.map(|e| e as i32)
.collect(),
};
Ok(protobuf_event_name_list)
}
}
impl TryFrom<ProtobufEventType> for EventType {
type Error = &'static str;
fn try_from(protobuf_event_type: ProtobufEventType) -> Result<Self, &'static str> {
Ok(match protobuf_event_type {
ProtobufEventType::ModeUpdate => EventType::ModeUpdate,
ProtobufEventType::TabUpdate => EventType::TabUpdate,
ProtobufEventType::PaneUpdate => EventType::PaneUpdate,
ProtobufEventType::Key => EventType::Key,
ProtobufEventType::Mouse => EventType::Mouse,
ProtobufEventType::Timer => EventType::Timer,
ProtobufEventType::CopyToClipboard => EventType::CopyToClipboard,
ProtobufEventType::SystemClipboardFailure => EventType::SystemClipboardFailure,
ProtobufEventType::InputReceived => EventType::InputReceived,
ProtobufEventType::Visible => EventType::Visible,
ProtobufEventType::CustomMessage => EventType::CustomMessage,
ProtobufEventType::FileSystemCreate => EventType::FileSystemCreate,
ProtobufEventType::FileSystemRead => EventType::FileSystemRead,
ProtobufEventType::FileSystemUpdate => EventType::FileSystemUpdate,
ProtobufEventType::FileSystemDelete => EventType::FileSystemDelete,
ProtobufEventType::PermissionRequestResult => EventType::PermissionRequestResult,
ProtobufEventType::SessionUpdate => EventType::SessionUpdate,
ProtobufEventType::RunCommandResult => EventType::RunCommandResult,
ProtobufEventType::WebRequestResult => EventType::WebRequestResult,
ProtobufEventType::CommandPaneOpened => EventType::CommandPaneOpened,
ProtobufEventType::CommandPaneExited => EventType::CommandPaneExited,
ProtobufEventType::PaneClosed => EventType::PaneClosed,
ProtobufEventType::EditPaneOpened => EventType::EditPaneOpened,
ProtobufEventType::EditPaneExited => EventType::EditPaneExited,
ProtobufEventType::CommandPaneReRun => EventType::CommandPaneReRun,
ProtobufEventType::FailedToWriteConfigToDisk => EventType::FailedToWriteConfigToDisk,
ProtobufEventType::ListClients => EventType::ListClients,
ProtobufEventType::HostFolderChanged => EventType::HostFolderChanged,
ProtobufEventType::FailedToChangeHostFolder => EventType::FailedToChangeHostFolder,
ProtobufEventType::PastedText => EventType::PastedText,
ProtobufEventType::ConfigWasWrittenToDisk => EventType::ConfigWasWrittenToDisk,
ProtobufEventType::WebServerStatus => EventType::WebServerStatus,
ProtobufEventType::BeforeClose => EventType::BeforeClose,
ProtobufEventType::FailedToStartWebServer => EventType::FailedToStartWebServer,
ProtobufEventType::InterceptedKeyPress => EventType::InterceptedKeyPress,
ProtobufEventType::PaneRenderReport => EventType::PaneRenderReport,
ProtobufEventType::PaneRenderReportWithAnsi => EventType::PaneRenderReportWithAnsi,
ProtobufEventType::UserAction => EventType::UserAction,
ProtobufEventType::ActionComplete => EventType::ActionComplete,
ProtobufEventType::CwdChanged => EventType::CwdChanged,
ProtobufEventType::AvailableLayoutInfo => EventType::AvailableLayoutInfo,
ProtobufEventType::PluginConfigurationChanged => EventType::PluginConfigurationChanged,
ProtobufEventType::HighlightClicked => EventType::HighlightClicked,
ProtobufEventType::InitialKeybinds => EventType::InitialKeybinds,
})
}
}
impl TryFrom<EventType> for ProtobufEventType {
type Error = &'static str;
fn try_from(event_type: EventType) -> Result<Self, &'static str> {
Ok(match event_type {
EventType::ModeUpdate => ProtobufEventType::ModeUpdate,
EventType::TabUpdate => ProtobufEventType::TabUpdate,
EventType::PaneUpdate => ProtobufEventType::PaneUpdate,
EventType::Key => ProtobufEventType::Key,
EventType::Mouse => ProtobufEventType::Mouse,
EventType::Timer => ProtobufEventType::Timer,
EventType::CopyToClipboard => ProtobufEventType::CopyToClipboard,
EventType::SystemClipboardFailure => ProtobufEventType::SystemClipboardFailure,
EventType::InputReceived => ProtobufEventType::InputReceived,
EventType::Visible => ProtobufEventType::Visible,
EventType::CustomMessage => ProtobufEventType::CustomMessage,
EventType::FileSystemCreate => ProtobufEventType::FileSystemCreate,
EventType::FileSystemRead => ProtobufEventType::FileSystemRead,
EventType::FileSystemUpdate => ProtobufEventType::FileSystemUpdate,
EventType::FileSystemDelete => ProtobufEventType::FileSystemDelete,
EventType::PermissionRequestResult => ProtobufEventType::PermissionRequestResult,
EventType::SessionUpdate => ProtobufEventType::SessionUpdate,
EventType::RunCommandResult => ProtobufEventType::RunCommandResult,
EventType::WebRequestResult => ProtobufEventType::WebRequestResult,
EventType::CommandPaneOpened => ProtobufEventType::CommandPaneOpened,
EventType::CommandPaneExited => ProtobufEventType::CommandPaneExited,
EventType::PaneClosed => ProtobufEventType::PaneClosed,
EventType::EditPaneOpened => ProtobufEventType::EditPaneOpened,
EventType::EditPaneExited => ProtobufEventType::EditPaneExited,
EventType::CommandPaneReRun => ProtobufEventType::CommandPaneReRun,
EventType::FailedToWriteConfigToDisk => ProtobufEventType::FailedToWriteConfigToDisk,
EventType::ListClients => ProtobufEventType::ListClients,
EventType::HostFolderChanged => ProtobufEventType::HostFolderChanged,
EventType::FailedToChangeHostFolder => ProtobufEventType::FailedToChangeHostFolder,
EventType::PastedText => ProtobufEventType::PastedText,
EventType::ConfigWasWrittenToDisk => ProtobufEventType::ConfigWasWrittenToDisk,
EventType::WebServerStatus => ProtobufEventType::WebServerStatus,
EventType::BeforeClose => ProtobufEventType::BeforeClose,
EventType::FailedToStartWebServer => ProtobufEventType::FailedToStartWebServer,
EventType::InterceptedKeyPress => ProtobufEventType::InterceptedKeyPress,
EventType::PaneRenderReport => ProtobufEventType::PaneRenderReport,
EventType::PaneRenderReportWithAnsi => ProtobufEventType::PaneRenderReportWithAnsi,
EventType::UserAction => ProtobufEventType::UserAction,
EventType::ActionComplete => ProtobufEventType::ActionComplete,
EventType::CwdChanged => ProtobufEventType::CwdChanged,
EventType::AvailableLayoutInfo => ProtobufEventType::AvailableLayoutInfo,
EventType::PluginConfigurationChanged => ProtobufEventType::PluginConfigurationChanged,
EventType::HighlightClicked => ProtobufEventType::HighlightClicked,
EventType::InitialKeybinds => ProtobufEventType::InitialKeybinds,
})
}
}
impl From<ProtobufResurrectableSession> for (String, Duration) {
fn from(protobuf_resurrectable_session: ProtobufResurrectableSession) -> (String, Duration) {
(
protobuf_resurrectable_session.name,
Duration::from_secs(protobuf_resurrectable_session.creation_time),
)
}
}
impl From<(String, Duration)> for ProtobufResurrectableSession {
fn from(session_name_and_creation_time: (String, Duration)) -> ProtobufResurrectableSession {
ProtobufResurrectableSession {
name: session_name_and_creation_time.0,
creation_time: session_name_and_creation_time.1.as_secs(),
}
}
}
impl From<&ProtobufFileMetadata> for Option<FileMetadata> {
fn from(protobuf_file_metadata: &ProtobufFileMetadata) -> Option<FileMetadata> {
if protobuf_file_metadata.metadata_is_set {
Some(FileMetadata {
is_file: protobuf_file_metadata.is_file,
is_dir: protobuf_file_metadata.is_dir,
is_symlink: protobuf_file_metadata.is_symlink,
len: protobuf_file_metadata.len,
})
} else {
None
}
}
}
impl From<Option<FileMetadata>> for ProtobufFileMetadata {
fn from(file_metadata: Option<FileMetadata>) -> ProtobufFileMetadata {
match file_metadata {
Some(file_metadata) => ProtobufFileMetadata {
metadata_is_set: true,
is_file: file_metadata.is_file,
is_dir: file_metadata.is_dir,
is_symlink: file_metadata.is_symlink,
len: file_metadata.len,
},
None => ProtobufFileMetadata {
metadata_is_set: false,
..Default::default()
},
}
}
}
#[test]
fn serialize_mode_update_event() {
use prost::Message;
let mode_update_event = Event::ModeUpdate(Default::default());
let protobuf_event: ProtobufEvent = mode_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
mode_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_mode_update_event_with_non_default_values() {
use crate::data::{BareKey, Palette, PaletteColor, ThemeHue};
use prost::Message;
let mode_update_event = Event::ModeUpdate(ModeInfo {
mode: InputMode::Locked,
keybinds: vec![
(
InputMode::Locked,
vec![(
KeyWithModifier::new(BareKey::Char('b')).with_alt_modifier(),
vec![Action::SwitchToMode {
input_mode: InputMode::Normal,
}],
)],
),
(
InputMode::Tab,
vec![(
KeyWithModifier::new(BareKey::Up).with_alt_modifier(),
vec![Action::SwitchToMode {
input_mode: InputMode::Pane,
}],
)],
),
(
InputMode::Pane,
vec![
(
KeyWithModifier::new(BareKey::Char('b')).with_ctrl_modifier(),
vec![
Action::SwitchToMode {
input_mode: InputMode::Tmux,
},
Action::Write {
key_with_modifier: None,
bytes: vec![10],
is_kitty_keyboard_protocol: false,
},
],
),
(
KeyWithModifier::new(BareKey::Char('a')),
vec![Action::WriteChars {
chars: "foo".to_owned(),
}],
),
],
),
],
style: Style {
colors: Palette {
source: crate::data::PaletteSource::Default,
theme_hue: ThemeHue::Light,
fg: PaletteColor::Rgb((1, 1, 1)),
bg: PaletteColor::Rgb((200, 200, 200)),
black: PaletteColor::EightBit(1),
red: PaletteColor::EightBit(2),
green: PaletteColor::EightBit(2),
yellow: PaletteColor::EightBit(2),
blue: PaletteColor::EightBit(2),
magenta: PaletteColor::EightBit(2),
cyan: PaletteColor::EightBit(2),
white: PaletteColor::EightBit(2),
orange: PaletteColor::EightBit(2),
gray: PaletteColor::EightBit(2),
purple: PaletteColor::EightBit(2),
gold: PaletteColor::EightBit(2),
silver: PaletteColor::EightBit(2),
pink: PaletteColor::EightBit(2),
brown: PaletteColor::Rgb((222, 221, 220)),
}
.into(),
rounded_corners: true,
hide_session_name: false,
},
capabilities: PluginCapabilities { arrow_fonts: false },
session_name: Some("my awesome test session".to_owned()),
base_mode: Some(InputMode::Locked),
editor: Some(PathBuf::from("my_awesome_editor")),
shell: Some(PathBuf::from("my_awesome_shell")),
web_clients_allowed: Some(true),
web_sharing: Some(WebSharing::default()),
currently_marking_pane_group: Some(false),
is_web_client: Some(false),
web_server_ip: IpAddr::from_str("127.0.0.1").ok(),
web_server_port: Some(8082),
web_server_capability: Some(true),
});
let protobuf_event: ProtobufEvent = mode_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
mode_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_tab_update_event() {
use prost::Message;
let tab_update_event = Event::TabUpdate(Default::default());
let protobuf_event: ProtobufEvent = tab_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
tab_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_tab_update_event_with_non_default_values() {
use prost::Message;
let tab_update_event = Event::TabUpdate(vec![
TabInfo {
position: 0,
name: "First tab".to_owned(),
active: true,
panes_to_hide: 2,
is_fullscreen_active: true,
is_sync_panes_active: false,
are_floating_panes_visible: true,
other_focused_clients: vec![2, 3, 4],
active_swap_layout_name: Some("my cool swap layout".to_owned()),
is_swap_layout_dirty: false,
viewport_rows: 10,
viewport_columns: 10,
display_area_rows: 10,
display_area_columns: 10,
selectable_tiled_panes_count: 10,
selectable_floating_panes_count: 10,
tab_id: 0,
has_bell_notification: false,
is_flashing_bell: false,
},
TabInfo {
position: 1,
name: "Secondtab".to_owned(),
active: false,
panes_to_hide: 5,
is_fullscreen_active: false,
is_sync_panes_active: true,
are_floating_panes_visible: true,
other_focused_clients: vec![1, 5, 111],
active_swap_layout_name: None,
is_swap_layout_dirty: true,
viewport_rows: 10,
viewport_columns: 10,
display_area_rows: 10,
display_area_columns: 10,
selectable_tiled_panes_count: 10,
selectable_floating_panes_count: 10,
tab_id: 1,
has_bell_notification: false,
is_flashing_bell: false,
},
TabInfo::default(),
]);
let protobuf_event: ProtobufEvent = tab_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
tab_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_pane_update_event() {
use prost::Message;
let pane_update_event = Event::PaneUpdate(Default::default());
let protobuf_event: ProtobufEvent = pane_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
pane_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_key_event() {
use crate::data::BareKey;
use prost::Message;
let key_event = Event::Key(KeyWithModifier::new(BareKey::Char('a')).with_ctrl_modifier());
let protobuf_event: ProtobufEvent = key_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
key_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_mouse_event() {
use prost::Message;
let mouse_event = Event::Mouse(Mouse::LeftClick(1, 1));
let protobuf_event: ProtobufEvent = mouse_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
mouse_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_mouse_event_without_position() {
use prost::Message;
let mouse_event = Event::Mouse(Mouse::ScrollUp(17));
let protobuf_event: ProtobufEvent = mouse_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
mouse_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_timer_event() {
use prost::Message;
let timer_event = Event::Timer(1.5);
let protobuf_event: ProtobufEvent = timer_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
timer_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_copy_to_clipboard_event() {
use prost::Message;
let copy_event = Event::CopyToClipboard(CopyDestination::Primary);
let protobuf_event: ProtobufEvent = copy_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
copy_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_clipboard_failure_event() {
use prost::Message;
let copy_event = Event::SystemClipboardFailure;
let protobuf_event: ProtobufEvent = copy_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
copy_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_input_received_event() {
use prost::Message;
let input_received_event = Event::InputReceived;
let protobuf_event: ProtobufEvent = input_received_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
input_received_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_visible_event() {
use prost::Message;
let visible_event = Event::Visible(true);
let protobuf_event: ProtobufEvent = visible_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
visible_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_custom_message_event() {
use prost::Message;
let custom_message_event = Event::CustomMessage("foo".to_owned(), "bar".to_owned());
let protobuf_event: ProtobufEvent = custom_message_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
custom_message_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_file_system_create_event() {
use prost::Message;
let file_system_event = Event::FileSystemCreate(vec![
("/absolute/path".into(), None),
("./relative_path".into(), Default::default()),
]);
let protobuf_event: ProtobufEvent = file_system_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
file_system_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_file_system_read_event() {
use prost::Message;
let file_system_event = Event::FileSystemRead(vec![
("/absolute/path".into(), None),
("./relative_path".into(), Default::default()),
]);
let protobuf_event: ProtobufEvent = file_system_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
file_system_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_file_system_update_event() {
use prost::Message;
let file_system_event = Event::FileSystemUpdate(vec![
("/absolute/path".into(), None),
("./relative_path".into(), Some(Default::default())),
]);
let protobuf_event: ProtobufEvent = file_system_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
file_system_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_file_system_delete_event() {
use prost::Message;
let file_system_event = Event::FileSystemDelete(vec![
("/absolute/path".into(), None),
("./relative_path".into(), Default::default()),
]);
let protobuf_event: ProtobufEvent = file_system_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
file_system_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_session_update_event() {
use prost::Message;
let session_update_event = Event::SessionUpdate(Default::default(), Default::default());
let protobuf_event: ProtobufEvent = session_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
session_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_session_update_event_with_non_default_values() {
use prost::Message;
let tab_infos = vec![
TabInfo {
position: 0,
name: "First tab".to_owned(),
active: true,
panes_to_hide: 2,
is_fullscreen_active: true,
is_sync_panes_active: false,
are_floating_panes_visible: true,
other_focused_clients: vec![2, 3, 4],
active_swap_layout_name: Some("my cool swap layout".to_owned()),
is_swap_layout_dirty: false,
viewport_rows: 10,
viewport_columns: 10,
display_area_rows: 10,
display_area_columns: 10,
selectable_tiled_panes_count: 10,
selectable_floating_panes_count: 10,
tab_id: 0,
has_bell_notification: false,
is_flashing_bell: false,
},
TabInfo {
position: 1,
name: "Secondtab".to_owned(),
active: false,
panes_to_hide: 5,
is_fullscreen_active: false,
is_sync_panes_active: true,
are_floating_panes_visible: true,
other_focused_clients: vec![1, 5, 111],
active_swap_layout_name: None,
is_swap_layout_dirty: true,
viewport_rows: 10,
viewport_columns: 10,
display_area_rows: 10,
display_area_columns: 10,
selectable_tiled_panes_count: 10,
selectable_floating_panes_count: 10,
tab_id: 1,
has_bell_notification: false,
is_flashing_bell: false,
},
TabInfo::default(),
];
let mut panes = HashMap::new();
let mut index_in_pane_group_1 = BTreeMap::new();
index_in_pane_group_1.insert(1, 0);
index_in_pane_group_1.insert(2, 0);
index_in_pane_group_1.insert(3, 0);
let mut index_in_pane_group_2 = BTreeMap::new();
index_in_pane_group_2.insert(1, 1);
index_in_pane_group_2.insert(2, 1);
index_in_pane_group_2.insert(3, 1);
let panes_list = vec![
PaneInfo {
id: 1,
is_plugin: false,
is_focused: true,
is_fullscreen: true,
is_floating: false,
is_suppressed: false,
title: "pane 1".to_owned(),
exited: false,
exit_status: None,
is_held: false,
pane_x: 0,
pane_content_x: 1,
pane_y: 0,
pane_content_y: 1,
pane_rows: 5,
pane_content_rows: 4,
pane_columns: 22,
pane_content_columns: 21,
cursor_coordinates_in_pane: Some((0, 0)),
terminal_command: Some("foo".to_owned()),
plugin_url: None,
is_selectable: true,
index_in_pane_group: index_in_pane_group_1,
default_fg: None,
default_bg: None,
},
PaneInfo {
id: 1,
is_plugin: true,
is_focused: true,
is_fullscreen: true,
is_floating: false,
is_suppressed: false,
title: "pane 1".to_owned(),
exited: false,
exit_status: None,
is_held: false,
pane_x: 0,
pane_content_x: 1,
pane_y: 0,
pane_content_y: 1,
pane_rows: 5,
pane_content_rows: 4,
pane_columns: 22,
pane_content_columns: 21,
cursor_coordinates_in_pane: Some((0, 0)),
terminal_command: None,
plugin_url: Some("i_am_a_fake_plugin".to_owned()),
is_selectable: true,
index_in_pane_group: index_in_pane_group_2,
default_fg: None,
default_bg: None,
},
];
panes.insert(0, panes_list);
let mut plugins = BTreeMap::new();
let mut plugin_configuration = BTreeMap::new();
plugin_configuration.insert("config_key".to_owned(), "config_value".to_owned());
plugins.insert(
1,
PluginInfo {
location: "https://example.com/my-plugin.wasm".to_owned(),
configuration: plugin_configuration,
},
);
let mut tab_history = BTreeMap::new();
tab_history.insert(1, vec![1, 2, 3]);
tab_history.insert(2, vec![1, 2, 3]);
let session_info_1 = SessionInfo {
name: "session 1".to_owned(),
tabs: tab_infos,
panes: PaneManifest { panes },
connected_clients: 2,
is_current_session: true,
available_layouts: vec![
LayoutInfo::File(
"layout 1".to_owned(),
LayoutMetadata {
tabs: vec![],
creation_time: "0".to_owned(),
update_time: "0".to_owned(),
},
),
LayoutInfo::BuiltIn("layout2".to_owned()),
LayoutInfo::File(
"layout3".to_owned(),
LayoutMetadata {
tabs: vec![],
creation_time: "0".to_owned(),
update_time: "0".to_owned(),
},
),
],
plugins,
web_clients_allowed: false,
web_client_count: 1,
tab_history,
pane_history: Default::default(),
creation_time: Duration::from_secs(100),
};
let session_info_2 = SessionInfo {
name: "session 2".to_owned(),
tabs: vec![],
panes: PaneManifest {
panes: HashMap::new(),
},
connected_clients: 0,
is_current_session: false,
available_layouts: vec![
LayoutInfo::File(
"layout 1".to_owned(),
LayoutMetadata {
tabs: vec![],
creation_time: "0".to_owned(),
update_time: "0".to_owned(),
},
),
LayoutInfo::BuiltIn("layout2".to_owned()),
LayoutInfo::File(
"layout3".to_owned(),
LayoutMetadata {
tabs: vec![],
creation_time: "0".to_owned(),
update_time: "0".to_owned(),
},
),
],
plugins: Default::default(),
web_clients_allowed: false,
web_client_count: 0,
tab_history: Default::default(),
pane_history: Default::default(),
creation_time: Duration::from_secs(200),
};
let session_infos = vec![session_info_1, session_info_2];
let resurrectable_sessions = vec![];
let session_update_event = Event::SessionUpdate(session_infos, resurrectable_sessions);
let protobuf_event: ProtobufEvent = session_update_event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
session_update_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
impl TryFrom<ProtobufPaneId> for PaneId {
type Error = &'static str;
fn try_from(protobuf_pane_id: ProtobufPaneId) -> Result<Self, &'static str> {
match ProtobufPaneType::from_i32(protobuf_pane_id.pane_type) {
Some(ProtobufPaneType::Terminal) => Ok(PaneId::Terminal(protobuf_pane_id.id)),
Some(ProtobufPaneType::Plugin) => Ok(PaneId::Plugin(protobuf_pane_id.id)),
None => Err("Failed to convert PaneId"),
}
}
}
impl TryFrom<PaneId> for ProtobufPaneId {
type Error = &'static str;
fn try_from(pane_id: PaneId) -> Result<Self, &'static str> {
match pane_id {
PaneId::Terminal(id) => Ok(ProtobufPaneId {
pane_type: ProtobufPaneType::Terminal as i32,
id,
}),
PaneId::Plugin(id) => Ok(ProtobufPaneId {
pane_type: ProtobufPaneType::Plugin as i32,
id,
}),
}
}
}
impl Into<ProtobufWebSharing> for WebSharing {
fn into(self) -> ProtobufWebSharing {
match self {
WebSharing::On => ProtobufWebSharing::On,
WebSharing::Off => ProtobufWebSharing::Off,
WebSharing::Disabled => ProtobufWebSharing::Disabled,
}
}
}
impl Into<WebSharing> for ProtobufWebSharing {
fn into(self) -> WebSharing {
match self {
ProtobufWebSharing::On => WebSharing::On,
ProtobufWebSharing::Off => WebSharing::Off,
ProtobufWebSharing::Disabled => WebSharing::Disabled,
}
}
}
impl TryFrom<WebServerStatus> for ProtobufWebServerStatusPayload {
type Error = &'static str;
fn try_from(web_server_status: WebServerStatus) -> Result<Self, &'static str> {
match web_server_status {
WebServerStatus::Online(url) => Ok(ProtobufWebServerStatusPayload {
web_server_status_indication: WebServerStatusIndication::Online as i32,
payload: Some(url),
}),
WebServerStatus::DifferentVersion(version) => Ok(ProtobufWebServerStatusPayload {
web_server_status_indication: WebServerStatusIndication::DifferentVersion as i32,
payload: Some(format!("{}", version)),
}),
WebServerStatus::Offline => Ok(ProtobufWebServerStatusPayload {
web_server_status_indication: WebServerStatusIndication::Offline as i32,
payload: None,
}),
}
}
}
impl TryFrom<ProtobufWebServerStatusPayload> for WebServerStatus {
type Error = &'static str;
fn try_from(
protobuf_web_server_status: ProtobufWebServerStatusPayload,
) -> Result<Self, &'static str> {
match WebServerStatusIndication::from_i32(
protobuf_web_server_status.web_server_status_indication,
) {
Some(WebServerStatusIndication::Online) => {
let payload = protobuf_web_server_status
.payload
.ok_or("payload_not_found")?;
Ok(WebServerStatus::Online(payload))
},
Some(WebServerStatusIndication::DifferentVersion) => {
let payload = protobuf_web_server_status
.payload
.ok_or("payload_not_found")?;
Ok(WebServerStatus::DifferentVersion(payload))
},
Some(WebServerStatusIndication::Offline) => Ok(WebServerStatus::Offline),
None => Err("Unknown status"),
}
}
}
impl TryFrom<ProtobufPaneRenderReportPayload> for HashMap<PaneId, PaneContents> {
type Error = &'static str;
fn try_from(protobuf_payload: ProtobufPaneRenderReportPayload) -> Result<Self, &'static str> {
let mut pane_contents_map = HashMap::new();
for entry in protobuf_payload.pane_contents {
let pane_id = entry
.pane_id
.ok_or("Missing pane_id in PaneContentsEntry")?
.try_into()?;
let pane_contents = entry
.pane_contents
.ok_or("Missing pane_contents in PaneContentsEntry")?
.try_into()?;
pane_contents_map.insert(pane_id, pane_contents);
}
Ok(pane_contents_map)
}
}
impl TryFrom<HashMap<PaneId, PaneContents>> for ProtobufPaneRenderReportPayload {
type Error = &'static str;
fn try_from(pane_contents_map: HashMap<PaneId, PaneContents>) -> Result<Self, &'static str> {
let mut pane_contents_vec = vec![];
for (pane_id, pane_contents) in pane_contents_map {
pane_contents_vec.push(ProtobufPaneContentsEntry {
pane_id: Some(pane_id.try_into()?),
pane_contents: Some(pane_contents.try_into()?),
});
}
Ok(ProtobufPaneRenderReportPayload {
pane_contents: pane_contents_vec,
})
}
}
impl TryFrom<ProtobufPaneContents> for PaneContents {
type Error = &'static str;
fn try_from(protobuf_contents: ProtobufPaneContents) -> Result<Self, &'static str> {
let selected_text = protobuf_contents
.selected_text
.map(|st| st.try_into())
.transpose()?;
Ok(PaneContents {
viewport: protobuf_contents.viewport,
selected_text,
lines_above_viewport: protobuf_contents.lines_above_viewport,
lines_below_viewport: protobuf_contents.lines_below_viewport,
})
}
}
impl TryFrom<PaneContents> for ProtobufPaneContents {
type Error = &'static str;
fn try_from(pane_contents: PaneContents) -> Result<Self, &'static str> {
let selected_text = pane_contents
.selected_text
.map(|st| st.try_into())
.transpose()?;
Ok(ProtobufPaneContents {
viewport: pane_contents.viewport,
selected_text,
lines_above_viewport: pane_contents.lines_above_viewport,
lines_below_viewport: pane_contents.lines_below_viewport,
})
}
}
impl TryFrom<ProtobufPaneScrollbackResponse> for PaneScrollbackResponse {
type Error = &'static str;
fn try_from(protobuf_response: ProtobufPaneScrollbackResponse) -> Result<Self, &'static str> {
match protobuf_response.response {
Some(pane_scrollback_response::Response::Ok(pane_contents)) => {
Ok(PaneScrollbackResponse::Ok(pane_contents.try_into()?))
},
Some(pane_scrollback_response::Response::Err(error_msg)) => {
Ok(PaneScrollbackResponse::Err(error_msg))
},
None => Err("PaneScrollbackResponse missing response field"),
}
}
}
impl TryFrom<PaneScrollbackResponse> for ProtobufPaneScrollbackResponse {
type Error = &'static str;
fn try_from(response: PaneScrollbackResponse) -> Result<Self, &'static str> {
let response_field = match response {
PaneScrollbackResponse::Ok(pane_contents) => {
pane_scrollback_response::Response::Ok(pane_contents.try_into()?)
},
PaneScrollbackResponse::Err(error_msg) => {
pane_scrollback_response::Response::Err(error_msg)
},
};
Ok(ProtobufPaneScrollbackResponse {
response: Some(response_field),
})
}
}
impl TryFrom<ProtobufSelectedText> for SelectedText {
type Error = &'static str;
fn try_from(protobuf_selected_text: ProtobufSelectedText) -> Result<Self, &'static str> {
Ok(SelectedText {
start: protobuf_selected_text
.start
.ok_or("Missing start in SelectedText")?
.try_into()?,
end: protobuf_selected_text
.end
.ok_or("Missing end in SelectedText")?
.try_into()?,
})
}
}
impl TryFrom<SelectedText> for ProtobufSelectedText {
type Error = &'static str;
fn try_from(selected_text: SelectedText) -> Result<Self, &'static str> {
Ok(ProtobufSelectedText {
start: Some(selected_text.start.try_into()?),
end: Some(selected_text.end.try_into()?),
})
}
}
#[test]
fn serialize_pane_render_report_with_ansi_event() {
use prost::Message;
let pane_render_report_with_ansi_event = Event::PaneRenderReportWithAnsi(Default::default());
let protobuf_event: ProtobufEvent = pane_render_report_with_ansi_event
.clone()
.try_into()
.unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
pane_render_report_with_ansi_event, deserialized_event,
"Event properly serialized/deserialized without change"
);
}
#[test]
fn serialize_pane_render_report_with_ansi_event_with_data() {
use prost::Message;
use std::collections::HashMap;
let mut pane_contents_map = HashMap::new();
pane_contents_map.insert(
PaneId::Terminal(1),
PaneContents {
viewport: vec![
"\x1b[31mred text\x1b[0m".to_owned(),
"\x1b[1mbold text\x1b[0m".to_owned(),
],
selected_text: None,
lines_above_viewport: vec![],
lines_below_viewport: vec![],
},
);
let event = Event::PaneRenderReportWithAnsi(pane_contents_map);
let protobuf_event: ProtobufEvent = event.clone().try_into().unwrap();
let serialized_protobuf_event = protobuf_event.encode_to_vec();
let deserialized_protobuf_event: ProtobufEvent =
Message::decode(serialized_protobuf_event.as_slice()).unwrap();
let deserialized_event: Event = deserialized_protobuf_event.try_into().unwrap();
assert_eq!(
event, deserialized_event,
"PaneRenderReportWithAnsi event with ANSI data properly serialized/deserialized"
);
}