From 9481d53adccdd02284b7fe9e5c34d64ca67bbd49 Mon Sep 17 00:00:00 2001 From: Tangent 128 Date: Fri, 29 Sep 2017 01:31:06 -0400 Subject: [PATCH] Parse, chunk, & reconstruct served WebM file. --- src/bin/loop_server.rs | 75 ++++++++++++++++++++++++++++++++++++++---- 1 file changed, 69 insertions(+), 6 deletions(-) diff --git a/src/bin/loop_server.rs b/src/bin/loop_server.rs index 20083b5..01ad4f1 100644 --- a/src/bin/loop_server.rs +++ b/src/bin/loop_server.rs @@ -3,30 +3,39 @@ extern crate hyper; extern crate lab_ebml; use futures::future::FutureResult; -use futures::stream::{once, Stream}; +use futures::stream::{once, iter, Stream}; use lab_ebml::chunk::Chunk; +use lab_ebml::Schema; +use lab_ebml::webm::*; +use lab_ebml::webm::WebmElement::*; use hyper::{Get, StatusCode}; use hyper::header::ContentType; use hyper::server::{Http, Request, Response, Service}; use std::env::args; +use std::io::Cursor; use std::net::ToSocketAddrs; use std::sync::Arc; const SRC_FILE: &'static [u8] = include_bytes!("../data/test1.webm"); -struct WebmServer; +#[derive(Clone)] +struct WebmServer(Chunk, Vec); -type BodyStream = Box, Error = hyper::Error>>; +type BodyStream = Box, Error = hyper::Error>>; impl Service for WebmServer { type Request = Request; - type Response = Response; + type Response = Response>>; type Error = hyper::Error; type Future = FutureResult; fn call(&self, req: Request) -> Self::Future { let response = match (req.method(), req.path()) { (&Get, "/loop") => { - let stream: BodyStream = Box::new(once(Ok(Chunk::Headers {bytes: Arc::new(SRC_FILE)}))); + let results: Vec> = self.1.iter().map(|x| Ok(x.clone())).collect(); + let stream: BodyStream> = Box::new( + once(Ok(self.0.clone())) + .chain(iter(results)) + ); Response::new() .with_header(ContentType("video/webm".parse().unwrap())) .with_body(stream) @@ -42,5 +51,59 @@ impl Service for WebmServer { pub fn main() { let addr = args().nth(1).unwrap().to_socket_addrs().unwrap().next().unwrap(); - Http::new().bind(&addr, || Ok(WebmServer)).unwrap().run().unwrap(); + let webm_service = create_loop(); + Http::new().bind(&addr, move || Ok(webm_service.clone())).unwrap().run().unwrap(); +} + +fn create_loop() -> WebmServer { + let mut header = None; + let mut reading_head = true; + + let mut cluster_header = None; + let mut cluster_timecode = 0; + let mut chunks = Vec::new(); + + let mut buffer = Cursor::new(Vec::new()); + + for element in Webm.parse(SRC_FILE) { + match element { + Cluster => { + if reading_head { + header = Some(Chunk::Headers {bytes: Arc::new(buffer.into_inner())}); + } else { + if let Some(chunk) = cluster_header.take() { + chunks.push(chunk); + } + chunks.push(Chunk::ClusterBody {bytes: Arc::new(buffer.into_inner())}); + } + buffer = Cursor::new(Vec::new()); + reading_head = false; + }, + Timecode(timecode) => { + cluster_timecode = timecode; + cluster_header = Some(Chunk::>::new_cluster_head(timecode)); + }, + SimpleBlock(ref block) => { + if let Some(ref mut chunk) = cluster_header { + if (block.flags & 0b10000000) != 0 { + // TODO: this is incorrect, condition needs to also affirm we're the first video block of the cluster + chunk.mark_keyframe(true); + } + chunk.extend_timespan(cluster_timecode + (block.timecode as u64)); + } + encode_webm_element(&SimpleBlock(*block), &mut buffer).unwrap(); + }, + Info => continue, + Void => continue, + Unknown(_) => continue, + ref other => { + encode_webm_element(other, &mut buffer).unwrap(); + }, + } + } + + // finish last cluster + chunks.push(Chunk::ClusterBody {bytes: Arc::new(buffer.into_inner())}); + + WebmServer(header.unwrap(), chunks) }