use {MultiFileDirectAccessor, InMemoryFileSystem};
use bip_disk::{DiskManagerBuilder, IDiskMessage, ODiskMessage};
use bip_metainfo::{MetainfoBuilder, PieceLength, Metainfo};
use tokio_core::reactor::{Core};
use futures::future::{Loop};
use futures::stream::Stream;
use futures::sink::{Sink};
#[test]
fn positive_complete_torrent() {
let data_a = (::random_buffer(1023), "/path/to/file/a".into());
let data_b = (::random_buffer(2000), "/path/to/file/b".into());
let files_accessor = MultiFileDirectAccessor::new("/my/downloads/".into(),
vec![data_a.clone(), data_b.clone()]);
let metainfo_bytes = MetainfoBuilder::new()
.set_piece_length(PieceLength::Custom(1024))
.build(1, files_accessor, |_| ()).unwrap();
let metainfo_file = Metainfo::from_bytes(metainfo_bytes).unwrap();
let filesystem = InMemoryFileSystem::new();
let disk_manager = DiskManagerBuilder::new()
.build(filesystem.clone());
let (send, recv) = disk_manager.split();
let mut blocking_send = send.wait();
blocking_send.send(IDiskMessage::AddTorrent(metainfo_file.clone())).unwrap();
let mut core = Core::new().unwrap();
let (good_pieces, recv) = ::core_loop_with_timeout(&mut core, 500, (0, recv), |good_pieces, recv, msg| {
match msg {
ODiskMessage::TorrentAdded(_) => Loop::Break((good_pieces, recv)),
ODiskMessage::FoundGoodPiece(_, _) => Loop::Continue((good_pieces + 1, recv)),
unexpected @ _ => panic!("Unexpected Message: {:?}", unexpected)
}
});
assert_eq!(0, good_pieces);
let mut files_bytes = Vec::new();
files_bytes.extend_from_slice(&data_a.0);
files_bytes.extend_from_slice(&data_b.0);
::send_block(&mut blocking_send, &files_bytes[0..500], metainfo_file.info().info_hash(), 0, 0, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[500..1000], metainfo_file.info().info_hash(), 0, 500, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[1000..1024], metainfo_file.info().info_hash(), 0, 1000, 24, |bytes| { bytes[0] = !bytes[0]; });
::send_block(&mut blocking_send, &files_bytes[(1024 + 0)..(1024 + 500)], metainfo_file.info().info_hash(), 1, 0, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[(1024 + 500)..(1024 + 1000)], metainfo_file.info().info_hash(), 1, 500, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[(1024 + 1000)..(1024 + 1024)], metainfo_file.info().info_hash(), 1, 1000, 24, |_| ());
::send_block(&mut blocking_send, &files_bytes[(2048 + 0)..(2048 + 500)], metainfo_file.info().info_hash(), 2, 0, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[(2048 + 500)..(2048 + 975)], metainfo_file.info().info_hash(), 2, 500, 475, |_| ());
let (recv, piece_zero_good, piece_one_good, piece_two_good) = ::core_loop_with_timeout(&mut core, 500, ((false, false, false, 0), recv),
|(piece_zero_good, piece_one_good, piece_two_good, messages_recvd), recv, msg| {
let messages_recvd = messages_recvd + 1;
let (opt_piece_index, new_value) = match msg {
ODiskMessage::FoundGoodPiece(_, index) => (Some(index), true),
ODiskMessage::FoundBadPiece(_, index) => (Some(index), false),
ODiskMessage::BlockProcessed(_) => (None, false),
unexpected @ _ => panic!("Unexpected Message: {:?}", unexpected)
};
let (piece_zero_good, piece_one_good, piece_two_good) = match opt_piece_index {
None => (piece_zero_good, piece_one_good, piece_two_good),
Some(0) => (new_value, piece_one_good, piece_two_good),
Some(1) => (piece_zero_good, new_value, piece_two_good),
Some(2) => (piece_zero_good, piece_one_good, new_value),
Some(x) => panic!("Unexpected Index {:?}", x)
};
if messages_recvd == (8 + 3) {
Loop::Break((recv, piece_zero_good, piece_one_good, piece_two_good))
} else {
Loop::Continue(((piece_zero_good, piece_one_good, piece_two_good, messages_recvd), recv))
}
}
);
assert_eq!(false, piece_zero_good);
assert_eq!(true, piece_one_good);
assert_eq!(true, piece_two_good);
::send_block(&mut blocking_send, &files_bytes[0..500], metainfo_file.info().info_hash(), 0, 0, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[500..1000], metainfo_file.info().info_hash(), 0, 500, 500, |_| ());
::send_block(&mut blocking_send, &files_bytes[1000..1024], metainfo_file.info().info_hash(), 0, 1000, 24, |_| ());
let piece_zero_good = ::core_loop_with_timeout(&mut core, 500, ((false, 0), recv),
|(piece_zero_good, messages_recvd), recv, msg| {
let messages_recvd = messages_recvd + 1;
let (opt_piece_index, new_value) = match msg {
ODiskMessage::FoundGoodPiece(_, index) => (Some(index), true),
ODiskMessage::FoundBadPiece(_, index) => (Some(index), false),
ODiskMessage::BlockProcessed(_) => (None, false),
unexpected @ _ => panic!("Unexpected Message: {:?}", unexpected)
};
let piece_zero_good = match opt_piece_index {
None => piece_zero_good,
Some(0) => new_value,
Some(x) => panic!("Unexpected Index {:?}", x)
};
if messages_recvd == (3 + 1) {
Loop::Break(piece_zero_good)
} else {
Loop::Continue(((piece_zero_good, messages_recvd), recv))
}
}
);
assert_eq!(true, piece_zero_good);
}