diff options
Diffstat (limited to 'src/lib.rs')
-rw-r--r-- | src/lib.rs | 175 |
1 files changed, 170 insertions, 5 deletions
diff --git a/src/lib.rs b/src/lib.rs index 8519a25..2235e5b 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -15,7 +15,46 @@ * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. */ -//! <!-- TODO (placeholder) --> +//! Write hexchat plugins in Rust! +//! +//! This library provides safe API bindings for hexchat, but doesn't attempt to fix hexchat's own +//! bugs. It makes no effort to stop you from unloading your own code while it's still running, for +//! example. +//! +//! When using this library, it's strongly recommended to avoid heap-allocated statics (static +//! mutexes, lazy_static, etc). This is because it's currently impossible to deallocate those on +//! plugin unload. This can be worked around by placing those statics as fields in your plugin +//! struct. +//! +//! This caveat does not apply to static assets (`static FOO: &'static str`, for example), but it +//! does apply to thread-local storage. +//! +//! # Examples +//! +//! ``` +//! #[macro_use] +//! extern crate hexchat_plugin; +//! +//! use hexchat_plugin::{Plugin, PluginHandle}; +//! +//! #[derive(Default)] +//! struct MyPlugin {} +//! +//! impl Plugin for MyPlugin { +//! fn init(&self, ph: &mut PluginHandle, arg: Option<&str>) -> bool { +//! ph.register("myplugin", "0.1.0", "my simple plugin"); +//! ph.hook_command("hello-world", |ph, arg, arg_eol| { +//! ph.print("Hello, World!"); +//! hexchat_plugin::EAT_ALL +//! }, 0, Some("prints 'Hello, World!'")); +//! true +//! } +//! } +//! +//! hexchat_plugin!(MyPlugin); +//! +//! # fn main() { } // satisfy the compiler, we can't actually run the code +//! ``` /* * Some info about how HexChat does things: @@ -176,7 +215,17 @@ pub trait Plugin { // Structs -/// A hexchat plugin handle. +/// A hexchat plugin handle, used to register hooks and interact with hexchat. +/// +/// # Examples +/// +/// ``` +/// use hexchat_plugin::{PluginHandle}; +/// +/// fn init(ph: &mut PluginHandle) { +/// ph.register("myplug", "0.1.0", "my awesome plug"); +/// } +/// ``` pub struct PluginHandle { ph: *mut internals::Ph, // Used for registration @@ -184,16 +233,62 @@ pub struct PluginHandle { } /// Arguments passed to a hook, until the next argument. +/// +/// # Examples +/// +/// ``` +/// use hexchat_plugin::{PluginHandle, Word, WordEol, Eat}; +/// +/// fn cmd_foo(ph: &mut PluginHandle, arg: Word, arg_eol: WordEol) -> Eat { +/// if arg.len() < 3 { +/// ph.print("Not enough arguments"); +/// } else { +/// ph.print(&format!("{} {} {}", arg[0], arg[1], arg[2])); +/// } +/// hexchat_plugin::EAT_ALL +/// } +/// +/// fn on_privmsg(ph: &mut PluginHandle, word: Word, word_eol: WordEol) -> Eat { +/// if word.len() > 0 && word[0].starts_with('@') { +/// ph.print("We have message tags!?"); +/// } +/// hexchat_plugin::EAT_NONE +/// } +/// ``` pub struct Word<'a> { word: Vec<&'a str> } /// Arguments passed to a hook, until the end of the line. +/// +/// # Examples +/// +/// ``` +/// use hexchat_plugin::{PluginHandle, Word, WordEol, Eat}; +/// +/// fn cmd_foo(ph: &mut PluginHandle, arg: Word, arg_eol: WordEol) -> Eat { +/// if arg.len() < 3 { +/// ph.print("Not enough arguments"); +/// } else { +/// ph.print(&format!("{} {} {}", arg[0], arg[1], arg_eol[2])); +/// } +/// hexchat_plugin::EAT_ALL +/// } +/// +/// fn on_privmsg(ph: &mut PluginHandle, word: Word, word_eol: WordEol) -> Eat { +/// if word_eol.len() > 0 && word[0].starts_with('@') { +/// ph.print("We have message tags!?"); +/// } +/// hexchat_plugin::EAT_NONE +/// } +/// ``` pub struct WordEol<'a> { word_eol: Vec<&'a str> } /// A safety wrapper to ensure you're working with a valid context. +/// +/// This mechanism attempts to reduce the likelihood of segfaults. pub struct EnsureValidContext<'a> { ph: &'a mut PluginHandle, } @@ -206,8 +301,10 @@ pub struct EventAttrs<'a> { _dummy: PhantomData<&'a ()>, } -/// An status indicator for event callbacks. Indicates whether to continue processing, eat hexchat, +/// A status indicator for event callbacks. Indicates whether to continue processing, eat hexchat, /// eat plugin, or eat all. +/// +/// Returned by most hooks. #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)] pub struct Eat { do_eat: i32, @@ -436,11 +533,21 @@ impl PluginHandle { } } - /// Registers this hexchat plugin. + /// Registers this hexchat plugin. This must be called exactly once when the plugin is loaded. /// /// # Panics /// /// This function panics if this plugin is already registered. + /// + /// # Examples + /// + /// ``` + /// use hexchat_plugin::PluginHandle; + /// + /// fn init(ph: &mut PluginHandle) { + /// ph.register("foo", "0.1.0", "my foo plugin"); + /// } + /// ``` pub fn register(&mut self, name: &str, desc: &str, ver: &str) { unsafe { let info = self.info; @@ -574,7 +681,20 @@ impl PluginHandle { } } - /// Sets a command hook + /// Sets a command hook. + /// + /// # Examples + /// + /// ``` + /// use hexchat_plugin::PluginHandle; + /// + /// fn register_commands(ph: &mut PluginHandle) { + /// ph.hook_command("hello-world", |ph, arg, arg_eol| { + /// ph.print("Hello, World!"); + /// hexchat_plugin::EAT_ALL + /// }, 0, Some("prints 'Hello, World!'")); + /// } + /// ``` pub fn hook_command<F>(&mut self, cmd: &str, cb: F, pri: i32, help: Option<&str>) -> CommandHookHandle where F: Fn(&mut PluginHandle, Word, WordEol) -> Eat + 'static + ::std::panic::RefUnwindSafe { unsafe extern "C" fn callback(word: *const *const libc::c_char, word_eol: *const *const libc::c_char, ud: *mut libc::c_void) -> libc::c_int { // hook may unhook itself. @@ -609,6 +729,21 @@ impl PluginHandle { } } /// Sets a server hook. + /// + /// # Examples + /// + /// ``` + /// use hexchat_plugin::PluginHandle; + /// + /// fn register_server_hooks(ph: &mut PluginHandle) { + /// ph.hook_server("PRIVMSG", |ph, word, word_eol| { + /// if word.len() > 0 && word[0].starts_with('@') { + /// ph.print("We have message tags!?"); + /// } + /// hexchat_plugin::EAT_NONE + /// }, 0); + /// } + /// ``` pub fn hook_server<F>(&mut self, cmd: &str, cb: F, pri: i32) -> ServerHookHandle where F: Fn(&mut PluginHandle, Word, WordEol) -> Eat + 'static + ::std::panic::RefUnwindSafe { self.hook_server_attrs(cmd, move |ph, w, we, _| cb(ph, w, we), pri) } @@ -646,6 +781,23 @@ impl PluginHandle { } } /// Sets a print hook. + /// + /// # Examples + /// + /// ``` + /// use hexchat_plugin::PluginHandle; + /// + /// fn register_print_hooks(ph: &mut PluginHandle) { + /// ph.hook_print("Channel Message", |ph, arg| { + /// if let Some(nick) = word.get(0) { + /// if nick == "KnOwN_SpAmMeR" { + /// return hexchat_plugin::EAT_ALL + /// } + /// } + /// hexchat_plugin::EAT_NONE + /// }, 0); + /// } + /// ``` pub fn hook_print<F>(&mut self, name: &str, cb: F, pri: i32) -> PrintHookHandle where F: Fn(&mut PluginHandle, Word) -> Eat + 'static + ::std::panic::RefUnwindSafe { self.hook_print_attrs(name, move |ph, w, _| cb(ph, w), pri) } @@ -682,6 +834,19 @@ impl PluginHandle { } } /// Sets a timer hook + /// + /// # Examples + /// + /// ``` + /// use hexchat_plugin::PluginHandle; + /// + /// fn register_timers(ph: &mut PluginHandle) { + /// ph.hook_timer(|ph| { + /// ph.print("timer up!"); + /// false + /// }, 2000); + /// } + /// ``` pub fn hook_timer<F>(&mut self, timeout: i32, cb: F) -> TimerHookHandle where F: Fn(&mut PluginHandle) -> bool + 'static + ::std::panic::RefUnwindSafe { unsafe extern "C" fn callback(ud: *mut libc::c_void) -> libc::c_int { // hook may unhook itself. |