Make chunker handle additional headers (such as from concatenated files)
This commit is contained in:
parent
6eca0b923d
commit
7d4a26dad5
1 changed files with 32 additions and 1 deletions
33
src/chunk.rs
33
src/chunk.rs
|
@ -75,9 +75,13 @@ impl<B: AsRef<[u8]>> AsRef<[u8]> for Chunk<B> {
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
enum ChunkerState {
|
enum ChunkerState {
|
||||||
BuildingHeader(Cursor<Vec<u8>>),
|
BuildingHeader(Cursor<Vec<u8>>),
|
||||||
// WIP ClusterHead & body buffer
|
// ClusterHead & body buffer
|
||||||
BuildingCluster(ClusterHead, Cursor<Vec<u8>>),
|
BuildingCluster(ClusterHead, Cursor<Vec<u8>>),
|
||||||
EmittingClusterBody(Vec<u8>),
|
EmittingClusterBody(Vec<u8>),
|
||||||
|
EmittingClusterBodyBeforeNewHeader {
|
||||||
|
body: Vec<u8>,
|
||||||
|
new_header: Cursor<Vec<u8>>
|
||||||
|
},
|
||||||
EmittingFinalClusterBody(Vec<u8>),
|
EmittingFinalClusterBody(Vec<u8>),
|
||||||
End
|
End
|
||||||
}
|
}
|
||||||
|
@ -136,6 +140,26 @@ impl<S: EbmlEventSource> Stream for WebmChunker<S>
|
||||||
match self.source.poll_event() {
|
match self.source.poll_event() {
|
||||||
Err(passthru) => return Err(ChunkingError::OtherError(passthru)),
|
Err(passthru) => return Err(ChunkingError::OtherError(passthru)),
|
||||||
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
||||||
|
Ok(Async::Ready(Some(element @ WebmElement::EbmlHead)))
|
||||||
|
| Ok(Async::Ready(Some(element @ WebmElement::Segment))) => {
|
||||||
|
let liberated_cluster_head = mem::replace(cluster_head, ClusterHead::new(0));
|
||||||
|
let liberated_buffer = mem::replace(buffer, Cursor::new(Vec::new()));
|
||||||
|
|
||||||
|
let mut new_header_cursor = Cursor::new(Vec::new());
|
||||||
|
match encode_webm_element(element, &mut new_header_cursor) {
|
||||||
|
Ok(_) => {
|
||||||
|
return_value = Some(Ok(Async::Ready(Some(Chunk::ClusterHead(liberated_cluster_head)))));
|
||||||
|
new_state = Some(ChunkerState::EmittingClusterBodyBeforeNewHeader{
|
||||||
|
body: liberated_buffer.into_inner(),
|
||||||
|
new_header: new_header_cursor
|
||||||
|
});
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
return_value = Some(Err(ChunkingError::IoError(err)));
|
||||||
|
new_state = Some(ChunkerState::End);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Ok(Async::Ready(Some(WebmElement::Cluster))) => {
|
Ok(Async::Ready(Some(WebmElement::Cluster))) => {
|
||||||
let liberated_cluster_head = mem::replace(cluster_head, ClusterHead::new(0));
|
let liberated_cluster_head = mem::replace(cluster_head, ClusterHead::new(0));
|
||||||
let liberated_buffer = mem::replace(buffer, Cursor::new(Vec::new()));
|
let liberated_buffer = mem::replace(buffer, Cursor::new(Vec::new()));
|
||||||
|
@ -191,6 +215,13 @@ impl<S: EbmlEventSource> Stream for WebmChunker<S>
|
||||||
Cursor::new(Vec::new())
|
Cursor::new(Vec::new())
|
||||||
));
|
));
|
||||||
},
|
},
|
||||||
|
ChunkerState::EmittingClusterBodyBeforeNewHeader { ref mut body, ref mut new_header } => {
|
||||||
|
let liberated_body = mem::replace(body, Vec::new());
|
||||||
|
let liberated_header_cursor = mem::replace(new_header, Cursor::new(Vec::new()));
|
||||||
|
|
||||||
|
return_value = Some(Ok(Async::Ready(Some(Chunk::ClusterBody {bytes: Arc::new(liberated_body)}))));
|
||||||
|
new_state = Some(ChunkerState::BuildingHeader(liberated_header_cursor));
|
||||||
|
},
|
||||||
ChunkerState::EmittingFinalClusterBody(ref mut buffer) => {
|
ChunkerState::EmittingFinalClusterBody(ref mut buffer) => {
|
||||||
// flush final Cluster on end of stream
|
// flush final Cluster on end of stream
|
||||||
let liberated_buffer = mem::replace(buffer, Vec::new());
|
let liberated_buffer = mem::replace(buffer, Vec::new());
|
||||||
|
|
Loading…
Reference in a new issue