examples: appsrc: improved error handling

The program should now exit gracefully with human readable messages.
This commit is contained in:
Philippe Normand 2017-08-03 15:52:08 +01:00 committed by Sebastian Dröge
parent d6c032be7b
commit 50e6e2e108

View file

@ -3,24 +3,67 @@ use gst::*;
extern crate gstreamer_app as gst_app; extern crate gstreamer_app as gst_app;
use gst_app::*; use gst_app::*;
extern crate glib;
use std::fmt;
use std::u64; use std::u64;
use std::thread; use std::thread;
const WIDTH: usize = 320; const WIDTH: usize = 320;
const HEIGHT: usize = 240; const HEIGHT: usize = 240;
fn main() { #[derive(Debug)]
gst::init().unwrap(); enum AppSrcExError {
InitFailed(glib::Error),
ElementNotFound(&'static str),
ElementLinkFailed(&'static str, &'static str),
SetStateError(&'static str),
ElementError(std::string::String, glib::Error, std::string::String),
}
impl fmt::Display for AppSrcExError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match *self {
AppSrcExError::InitFailed(ref e) => {
write!(f, "GStreamer initialization failed: {:?}", e)
}
AppSrcExError::ElementNotFound(ref e) => write!(f, "Element {} not found", e),
AppSrcExError::ElementLinkFailed(ref e1, ref e2) => {
write!(f, "Link failed between {} and {}", e1, e2)
}
AppSrcExError::SetStateError(ref state) => {
write!(f, "Pipeline failed to switch to state {}", state)
}
AppSrcExError::ElementError(ref element, ref err, ref debug) => {
write!(f, "Error from {}: {} ({:?})", element, err, debug)
}
}
}
}
fn create_pipeline() -> Result<(Pipeline, AppSrc), AppSrcExError> {
gst::init().map_err(|e| AppSrcExError::InitFailed(e))?;
let pipeline = gst::Pipeline::new(None); let pipeline = gst::Pipeline::new(None);
let src = gst::ElementFactory::make("appsrc", None).unwrap(); let src = gst::ElementFactory::make("appsrc", None)
let videoconvert = gst::ElementFactory::make("videoconvert", None).unwrap(); .ok_or(AppSrcExError::ElementNotFound("appsrc"))?;
let sink = gst::ElementFactory::make("autovideosink", None).unwrap();
pipeline.add_many(&[&src, &videoconvert, &sink]).unwrap(); let videoconvert = gst::ElementFactory::make("videoconvert", None)
gst::Element::link_many(&[&src, &videoconvert, &sink]).unwrap(); .ok_or(AppSrcExError::ElementNotFound("videoconvert"))?;
let sink = gst::ElementFactory::make("autovideosink", None)
.ok_or(AppSrcExError::ElementNotFound("autovideosink"))?;
let appsrc = src.clone().dynamic_cast::<AppSrc>().unwrap(); pipeline
.add_many(&[&src, &videoconvert, &sink])
.expect("Unable to add elements in the pipeline");
gst::Element::link(&src, &videoconvert)
.map_err(|_| AppSrcExError::ElementLinkFailed("src", "videoconvert"))?;
gst::Element::link(&videoconvert, &sink)
.map_err(|_| AppSrcExError::ElementLinkFailed("videoconvert", "sink"))?;
let appsrc = src.clone()
.dynamic_cast::<AppSrc>()
.expect("Source element is expected to be an appsrc!");
appsrc.set_caps(&Caps::new_simple( appsrc.set_caps(&Caps::new_simple(
"video/x-raw", "video/x-raw",
&[ &[
@ -34,6 +77,12 @@ fn main() {
appsrc.set_max_bytes(1); appsrc.set_max_bytes(1);
appsrc.set_property_block(true); appsrc.set_property_block(true);
Ok((pipeline, appsrc))
}
fn main_loop() -> Result<(), AppSrcExError> {
let (pipeline, appsrc) = create_pipeline()?;
thread::spawn(move || { thread::spawn(move || {
for i in 0..100 { for i in 0..100 {
println!("Producing frame {}", i); println!("Producing frame {}", i);
@ -44,14 +93,14 @@ fn main() {
let g = if i % 3 == 0 { 0 } else { 255 }; let g = if i % 3 == 0 { 0 } else { 255 };
let b = if i % 5 == 0 { 0 } else { 255 }; let b = if i % 5 == 0 { 0 } else { 255 };
for _ in 0..(320 * 240) { for _ in 0..(WIDTH * HEIGHT) {
vec.push(b); vec.push(b);
vec.push(g); vec.push(g);
vec.push(r); vec.push(r);
vec.push(0); vec.push(0);
} }
let mut buffer = Buffer::from_vec(vec).unwrap(); let mut buffer = Buffer::from_vec(vec).expect("Unable to create a Buffer");
buffer.get_mut().unwrap().set_pts(i * 500_000_000); buffer.get_mut().unwrap().set_pts(i * 500_000_000);
if appsrc.push_buffer(buffer) != FlowReturn::Ok { if appsrc.push_buffer(buffer) != FlowReturn::Ok {
@ -62,12 +111,13 @@ fn main() {
appsrc.end_of_stream(); appsrc.end_of_stream();
}); });
assert_ne!( if let gst::StateChangeReturn::Failure = pipeline.set_state(gst::State::Playing) {
pipeline.set_state(gst::State::Playing), return Err(AppSrcExError::SetStateError("playing"));
gst::StateChangeReturn::Failure }
);
let bus = pipeline.get_bus().unwrap(); let bus = pipeline
.get_bus()
.expect("Pipeline without bus. Shouldn't happen!");
loop { loop {
let msg = match bus.timed_pop(u64::MAX) { let msg = match bus.timed_pop(u64::MAX) {
@ -78,20 +128,27 @@ fn main() {
match msg.view() { match msg.view() {
MessageView::Eos(..) => break, MessageView::Eos(..) => break,
MessageView::Error(err) => { MessageView::Error(err) => {
println!( pipeline.set_state(gst::State::Null);
"Error from {}: {} ({:?})", return Err(AppSrcExError::ElementError(
msg.get_src().get_path_string(), msg.get_src().get_path_string(),
err.get_error(), err.get_error(),
err.get_debug() err.get_debug().unwrap(),
); ));
break;
} }
_ => (), _ => (),
} }
} }
assert_ne!( if let gst::StateChangeReturn::Failure = pipeline.set_state(gst::State::Null) {
pipeline.set_state(gst::State::Null), return Err(AppSrcExError::SetStateError("null"));
gst::StateChangeReturn::Failure }
);
Ok(())
}
fn main() {
match main_loop() {
Ok(r) => r,
Err(e) => eprintln!("Error! {}", e),
}
} }