16 releases
new 0.3.10 | Jan 6, 2025 |
---|---|
0.3.9 | Jan 4, 2025 |
0.3.8 | Dec 28, 2024 |
0.2.10 | Nov 30, 2024 |
0.1.1 | Aug 15, 2024 |
#1487 in Command line utilities
1,732 downloads per month
Used in 2 crates
20KB
323 lines
lla
- plugin interface
This crate provides a plugin interface for the lla
command line tool.
Plugin Architecture
The plugin system in lla
is designed to be robust and version-independent, using a message-passing architecture that ensures ABI compatibility across different Rust versions. Here's how it works:
Core Components
-
Protocol Buffer Interface
- All communication between the main application and plugins uses Protocol Buffers
- Messages are defined in
plugin.proto
, providing a language-agnostic contract - Supports various operations like decoration, field formatting, and custom actions
-
FFI Boundary
- Plugins are loaded dynamically using
libloading
- Communication crosses the FFI boundary using only C-compatible types
- Raw bytes are used for data transfer, avoiding Rust-specific ABI details
- Plugins are loaded dynamically using
ABI Compatibility
The plugin system solves the ABI compatibility problem through several mechanisms:
-
Message-Based Communication
- Instead of direct function calls, all interaction happens through serialized Protocol Buffer messages
- This eliminates dependency on Rust's internal ABI, which can change between versions
- Plugins and the main application can be compiled with different Rust versions
-
Version Control
- Each plugin declares its API version
- The system performs version checking during plugin loading
- Incompatible plugins are rejected with clear error messages
-
Stable Interface
- The FFI layer uses only C-compatible types, ensuring ABI stability
- Complex Rust types are serialized before crossing plugin boundaries
- The Protocol Buffer schema acts as a stable contract between components
Plugin Development
To create a plugin:
- Implement the plugin interface defined in the Protocol Buffer schema
- Use the provided macros and traits for proper FFI setup
- Compile as a dynamic library (
.so
,.dll
, or.dylib
)
The main application will handle loading, version verification, and communication with your plugin automatically.
Example Plugin
Here's a simple example of a file type categorizer plugin that demonstrates the key concepts:
use lla_plugin_interface::{DecoratedEntry, Plugin};
use prost::Message as ProstMessage;
/// A simple plugin that categorizes files based on their extensions
pub struct SimpleCategorizerPlugin {
categories: Vec<(String, Vec<String>)>, // (category_name, extensions)
}
impl SimpleCategorizerPlugin {
pub fn new() -> Self {
Self {
categories: vec![
("Document".to_string(), vec!["txt", "pdf", "doc"].into_iter().map(String::from).collect()),
("Image".to_string(), vec!["jpg", "png", "gif"].into_iter().map(String::from).collect()),
("Code".to_string(), vec!["rs", "py", "js"].into_iter().map(String::from).collect()),
]
}
}
fn get_category(&self, entry: &DecoratedEntry) -> Option<String> {
let extension = entry.path.extension()?.to_str()?.to_lowercase();
self.categories.iter()
.find(|(_, exts)| exts.contains(&extension))
.map(|(category, _)| category.clone())
}
}
impl Plugin for SimpleCategorizerPlugin {
fn handle_raw_request(&mut self, request: &[u8]) -> Vec<u8> {
use lla_plugin_interface::proto::{self, plugin_message};
// Decode the incoming protobuf message
let proto_msg = match proto::PluginMessage::decode(request) {
Ok(msg) => msg,
Err(e) => return self.encode_error(&format!("Failed to decode request: {}", e)),
};
// Handle different message types
let response_msg = match proto_msg.message {
// Return plugin metadata
Some(plugin_message::Message::GetName(_)) => {
plugin_message::Message::NameResponse("simple-categorizer".to_string())
}
Some(plugin_message::Message::GetVersion(_)) => {
plugin_message::Message::VersionResponse("0.1.0".to_string())
}
Some(plugin_message::Message::GetDescription(_)) => {
plugin_message::Message::DescriptionResponse(
"A simple file categorizer plugin".to_string(),
)
}
// Handle file decoration request
Some(plugin_message::Message::Decorate(entry)) => {
let mut decorated_entry = match DecoratedEntry::try_from(entry.clone()) {
Ok(e) => e,
Err(e) => return self.encode_error(&format!("Failed to convert entry: {}", e)),
};
// Add category to the entry's custom fields
if let Some(category) = self.get_category(&decorated_entry) {
decorated_entry.custom_fields.insert("category".to_string(), category);
}
plugin_message::Message::DecoratedResponse(decorated_entry.into())
}
_ => plugin_message::Message::ErrorResponse("Invalid request type".to_string()),
};
// Encode and return the response
let response = proto::PluginMessage {
message: Some(response_msg),
};
let mut buf = bytes::BytesMut::with_capacity(response.encoded_len());
response.encode(&mut buf).unwrap();
buf.to_vec()
}
}
// Register the plugin with the main application
lla_plugin_interface::declare_plugin!(SimpleCategorizerPlugin);
This example demonstrates:
- Using Protocol Buffers for communication
- Implementing the
Plugin
trait - Handling different message types
- Processing file metadata
- Adding custom fields to entries
- Proper error handling
- Using the plugin declaration macro
The plugin can be compiled as a dynamic library and loaded by the main application at runtime, with full ABI compatibility regardless of the Rust version used to compile either component.
Dependencies
~0.6–2.9MB
~49K SLoC