openethereum/util/src/panics.rs

174 lines
4.5 KiB
Rust
Raw Normal View History

2016-02-09 16:47:21 +01:00
// Copyright 2015, 2016 Ethcore (UK) Ltd.
// This file is part of Parity.
// Parity is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// Parity is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
//! Panic utilities
use std::thread;
use std::ops::DerefMut;
use std::any::Any;
use std::sync::{Arc, Mutex};
2016-02-09 16:47:21 +01:00
2016-02-10 15:28:43 +01:00
/// Thread-safe closure for handling possible panics
pub trait OnPanicListener: Send + Sync + 'static {
/// Invoke listener
fn call(&mut self, arg: &str);
2016-02-09 16:47:21 +01:00
}
2016-02-10 15:28:43 +01:00
/// Trait indicating that the structure catches some of the panics (most probably from spawned threads)
/// and it's possbile to be notified when one of the threads panics.
pub trait MayPanic {
/// `closure` will be invoked whenever panic in thread is caught
fn on_panic<F>(&self, closure: F)
2016-02-10 15:28:43 +01:00
where F: OnPanicListener;
2016-02-09 16:47:21 +01:00
}
2016-02-10 15:28:43 +01:00
/// Structure that allows to catch panics and notify listeners
pub struct PanicHandler {
listeners: Mutex<Vec<Box<OnPanicListener>>>
}
2016-02-09 16:47:21 +01:00
2016-02-10 15:28:43 +01:00
impl PanicHandler {
/// Creates new `PanicHandler` wrapped in `Arc`
pub fn new_arc() -> Arc<PanicHandler> {
Arc::new(Self::new())
2016-02-09 16:47:21 +01:00
}
2016-02-10 15:28:43 +01:00
/// Creates new `PanicHandler`
pub fn new() -> PanicHandler {
PanicHandler {
2016-02-09 16:47:21 +01:00
listeners: Mutex::new(vec![])
}
}
2016-02-10 15:28:43 +01:00
/// Invoke closure and catch any possible panics.
/// In case of panic notifies all listeners about it.
#[allow(deprecated)]
// TODO [todr] catch_panic is deprecated but panic::recover has different bounds (not allowing mutex)
2016-02-10 15:28:43 +01:00
pub fn catch_panic<G, R>(&self, g: G) -> thread::Result<R> where G: FnOnce() -> R + Send + 'static {
let result = thread::catch_panic(g);
if let Err(ref e) = result {
2016-02-10 15:28:43 +01:00
let res = convert_to_string(e);
if let Some(r) = res {
2016-02-10 15:28:43 +01:00
self.notify_all(r);
2016-02-09 16:47:21 +01:00
}
}
2016-02-09 16:47:21 +01:00
result
}
2016-02-10 15:28:43 +01:00
/// Notify listeners about panic
pub fn notify_all(&self, r: String) {
let mut listeners = self.listeners.lock().unwrap();
for listener in listeners.deref_mut() {
2016-02-10 15:28:43 +01:00
listener.call(&r);
2016-02-09 16:47:21 +01:00
}
}
2016-02-09 16:47:21 +01:00
}
2016-02-10 15:28:43 +01:00
impl MayPanic for PanicHandler {
fn on_panic<F>(&self, closure: F)
2016-02-10 15:28:43 +01:00
where F: OnPanicListener {
2016-02-09 16:47:21 +01:00
self.listeners.lock().unwrap().push(Box::new(closure));
}
}
2016-02-10 15:28:43 +01:00
impl<F> OnPanicListener for F
where F: FnMut(String) + Send + Sync + 'static {
fn call(&mut self, arg: &str) {
self(arg.to_owned())
}
}
2016-02-10 15:28:43 +01:00
fn convert_to_string(t: &Box<Any + Send>) -> Option<String> {
let as_str = t.downcast_ref::<&'static str>().map(|t| t.clone().to_owned());
let as_string = t.downcast_ref::<String>().cloned();
2016-02-10 15:28:43 +01:00
as_str.or(as_string)
}
2016-02-09 16:47:21 +01:00
#[test]
fn should_notify_listeners_about_panic () {
use std::sync::RwLock;
2016-02-09 16:47:21 +01:00
// given
let invocations = Arc::new(RwLock::new(vec![]));
let i = invocations.clone();
2016-02-10 15:28:43 +01:00
let p = PanicHandler::new();
p.on_panic(move |t| i.write().unwrap().push(t));
2016-02-09 16:47:21 +01:00
// when
p.catch_panic(|| panic!("Panic!")).unwrap_err();
2016-02-09 16:47:21 +01:00
// then
assert!(invocations.read().unwrap()[0] == "Panic!");
}
#[test]
fn should_notify_listeners_about_panic_when_string_is_dynamic () {
use std::sync::RwLock;
// given
let invocations = Arc::new(RwLock::new(vec![]));
let i = invocations.clone();
2016-02-10 15:28:43 +01:00
let p = PanicHandler::new();
p.on_panic(move |t| i.write().unwrap().push(t));
// when
p.catch_panic(|| panic!("Panic: {}", 1)).unwrap_err();
// then
assert!(invocations.read().unwrap()[0] == "Panic: 1");
}
2016-02-09 16:47:21 +01:00
#[test]
fn should_notify_listeners_about_panic_in_other_thread () {
use std::thread;
use std::sync::RwLock;
2016-02-09 16:47:21 +01:00
// given
let invocations = Arc::new(RwLock::new(vec![]));
let i = invocations.clone();
2016-02-10 15:28:43 +01:00
let p = PanicHandler::new();
p.on_panic(move |t| i.write().unwrap().push(t));
2016-02-09 16:47:21 +01:00
// when
let t = thread::spawn(move ||
p.catch_panic(|| panic!("Panic!")).unwrap()
2016-02-09 16:47:21 +01:00
);
t.join().unwrap_err();
// then
assert!(invocations.read().unwrap()[0] == "Panic!");
}
#[test]
fn should_forward_panics () {
use std::sync::RwLock;
// given
let invocations = Arc::new(RwLock::new(vec![]));
let i = invocations.clone();
2016-02-10 15:28:43 +01:00
let p = PanicHandler::new();
p.on_panic(move |t| i.write().unwrap().push(t));
2016-02-10 15:28:43 +01:00
let p2 = PanicHandler::new();
p2.on_panic(move |t| p.notify_all(t));
// when
p2.catch_panic(|| panic!("Panic!")).unwrap_err();
2016-02-09 16:47:21 +01:00
// then
assert!(invocations.read().unwrap()[0] == "Panic!");
}