audio_key.rs 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. use byteorder::{BigEndian, ByteOrder, WriteBytesExt};
  2. use bytes::Bytes;
  3. use futures::sync::oneshot;
  4. use futures::{Async, Future, Poll};
  5. use std::collections::HashMap;
  6. use std::io::Write;
  7. use util::SeqGenerator;
  8. use util::{SpotifyId, FileId};
  9. #[derive(Debug,Hash,PartialEq,Eq,Copy,Clone)]
  10. pub struct AudioKey(pub [u8; 16]);
  11. #[derive(Debug,Hash,PartialEq,Eq,Copy,Clone)]
  12. pub struct AudioKeyError;
  13. component! {
  14. AudioKeyManager : AudioKeyManagerInner {
  15. sequence: SeqGenerator<u32> = SeqGenerator::new(0),
  16. pending: HashMap<u32, oneshot::Sender<Result<AudioKey, AudioKeyError>>> = HashMap::new(),
  17. }
  18. }
  19. impl AudioKeyManager {
  20. pub fn dispatch(&self, cmd: u8, mut data: Bytes) {
  21. let seq = BigEndian::read_u32(data.split_to(4).as_ref());
  22. let sender = self.lock(|inner| inner.pending.remove(&seq));
  23. if let Some(sender) = sender {
  24. match cmd {
  25. 0xd => {
  26. let mut key = [0u8; 16];
  27. key.copy_from_slice(data.as_ref());
  28. let _ = sender.send(Ok(AudioKey(key)));
  29. }
  30. 0xe => {
  31. warn!("error audio key {:x} {:x}", data.as_ref()[0], data.as_ref()[1]);
  32. let _ = sender.send(Err(AudioKeyError));
  33. }
  34. _ => (),
  35. }
  36. }
  37. }
  38. pub fn request(&self, track: SpotifyId, file: FileId) -> AudioKeyFuture<AudioKey> {
  39. let (tx, rx) = oneshot::channel();
  40. let seq = self.lock(move |inner| {
  41. let seq = inner.sequence.get();
  42. inner.pending.insert(seq, tx);
  43. seq
  44. });
  45. self.send_key_request(seq, track, file);
  46. AudioKeyFuture(rx)
  47. }
  48. fn send_key_request(&self, seq: u32, track: SpotifyId, file: FileId) {
  49. let mut data: Vec<u8> = Vec::new();
  50. data.write(&file.0).unwrap();
  51. data.write(&track.to_raw()).unwrap();
  52. data.write_u32::<BigEndian>(seq).unwrap();
  53. data.write_u16::<BigEndian>(0x0000).unwrap();
  54. self.session().send_packet(0xc, data)
  55. }
  56. }
  57. pub struct AudioKeyFuture<T>(oneshot::Receiver<Result<T, AudioKeyError>>);
  58. impl <T> Future for AudioKeyFuture<T> {
  59. type Item = T;
  60. type Error = AudioKeyError;
  61. fn poll(&mut self) -> Poll<T, AudioKeyError> {
  62. match self.0.poll() {
  63. Ok(Async::Ready(Ok(value))) => Ok(Async::Ready(value)),
  64. Ok(Async::Ready(Err(err))) => Err(err),
  65. Ok(Async::NotReady) => Ok(Async::NotReady),
  66. Err(oneshot::Canceled) => Err(AudioKeyError),
  67. }
  68. }
  69. }