picodata_plugin/internal/
mod.rs1pub mod cas;
3pub(crate) mod ffi;
4pub mod types;
5
6use crate::internal::ffi::{
7 pico_ffi_authenticate, pico_ffi_instance_info, pico_ffi_raft_info, pico_ffi_rpc_version,
8 pico_ffi_version,
9};
10use crate::internal::types::InstanceInfo;
11use abi_stable::derive_macro_reexports::RResult;
12use std::{env, fs, io, process};
13use tarantool::error::BoxError;
14
15pub fn picodata_version() -> &'static str {
17 let ptr_and_len = unsafe { pico_ffi_version() };
18 let slice = unsafe { std::slice::from_raw_parts(ptr_and_len.0, ptr_and_len.1) };
20 std::str::from_utf8(slice).expect("should be valid utf8")
21}
22
23pub fn rpc_version() -> &'static str {
25 let ptr_and_len = unsafe { pico_ffi_rpc_version() };
26 let slice = unsafe { std::slice::from_raw_parts(ptr_and_len.0, ptr_and_len.1) };
28 std::str::from_utf8(slice).expect("should be valid utf8")
29}
30
31pub fn instance_info() -> Result<InstanceInfo, BoxError> {
33 match unsafe { pico_ffi_instance_info() } {
34 RResult::ROk(info) => Ok(info),
35 RResult::RErr(_) => {
36 let error = BoxError::last();
37 Err(error)
38 }
39 }
40}
41
42pub fn raft_info() -> types::RaftInfo {
44 unsafe { pico_ffi_raft_info() }
45}
46
47pub fn authenticate(username: &str, password: impl AsRef<[u8]>) -> Result<(), BoxError> {
73 match unsafe { pico_ffi_authenticate(username.into(), password.as_ref().into()) } {
74 0 => Ok(()),
75 _ => {
76 let error = BoxError::last();
77 Err(error)
78 }
79 }
80}
81
82fn dump_backtrace(msg: &str) -> Result<(), io::Error> {
85 let should_dump = env::var("PICODATA_INTERNAL_BACKTRACE_DUMP")
86 .map(|v| !v.is_empty())
87 .unwrap_or(false);
88
89 if !should_dump {
90 return Ok(());
91 }
92
93 let name = format!("picodata-{}.backtrace", process::id());
94 let path = env::current_dir()?.join(&name);
95
96 fs::write(&name, msg)
97 .map(|_| tarantool::say_info!("dumped panic backtrace to `{}`", path.display()))
98 .inspect_err(|e| tarantool::say_info!("{}", e))?;
99
100 Ok(())
101}
102
103#[inline]
104pub fn set_panic_hook() {
105 std::panic::set_hook(Box::new(|info| {
109 let version = crate::internal::picodata_version();
110
111 let backtrace = std::backtrace::Backtrace::force_capture();
113 let message = format!(
114 "Picodata {version}\n\n{info}\n\nbacktrace:\n{backtrace}\naborting due to panic"
115 );
116
117 tarantool::say_crit!("\n\n{message}");
119 dump_backtrace(&message)
120 .unwrap_or_else(|e| tarantool::say_info!("Failed to dump panic backtrace: {}", e));
121
122 std::process::abort();
123 }));
124}
125
126#[derive(thiserror::Error, Debug)]
127pub enum InternalError {
128 #[error("timeout")]
129 Timeout,
130 #[error("internal error: {0}")]
131 Any(BoxError),
132}