| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  | use crate::error::{Error, Result};
 | 
					
						
							| 
									
										
										
										
											2021-03-28 17:28:49 +02:00
										 |  |  | use crate::session::{ImportSession, ImportTrack, State};
 | 
					
						
							| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  | use gstreamer::prelude::*;
 | 
					
						
							|  |  |  | use gstreamer::{ClockTime, ElementFactory, MessageType, MessageView, TocEntryType};
 | 
					
						
							|  |  |  | use gstreamer::tags::{Duration, TrackNumber};
 | 
					
						
							| 
									
										
										
										
											2021-03-28 17:28:49 +02:00
										 |  |  | use log::info;
 | 
					
						
							| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  | use sha2::{Sha256, Digest};
 | 
					
						
							|  |  |  | use std::path::PathBuf;
 | 
					
						
							| 
									
										
										
										
											2021-03-28 17:28:49 +02:00
										 |  |  | use std::sync::Mutex;
 | 
					
						
							|  |  |  | use tokio::sync::watch;
 | 
					
						
							| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /// Create a new import session for the default disc drive.
 | 
					
						
							|  |  |  | pub(super) fn new() -> Result<ImportSession> {
 | 
					
						
							| 
									
										
										
										
											2021-03-28 17:28:49 +02:00
										 |  |  |     let (state_sender, state_receiver) = watch::channel(State::Waiting);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  |     let mut tracks = Vec::new();
 | 
					
						
							|  |  |  |     let mut hasher = Sha256::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Build the GStreamer pipeline. It will contain a fakesink initially to be able to run it
 | 
					
						
							|  |  |  |     // forward to the paused state without specifying a file name before knowing the tracks.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let cdparanoiasrc = ElementFactory::make("cdparanoiasrc", None)?;
 | 
					
						
							|  |  |  |     let queue = ElementFactory::make("queue", None)?;
 | 
					
						
							|  |  |  |     let audioconvert = ElementFactory::make("audioconvert", None)?;
 | 
					
						
							|  |  |  |     let flacenc = ElementFactory::make("flacenc", None)?;
 | 
					
						
							|  |  |  |     let fakesink = gstreamer::ElementFactory::make("fakesink", None)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let pipeline = gstreamer::Pipeline::new(None);
 | 
					
						
							|  |  |  |     pipeline.add_many(&[&cdparanoiasrc, &queue, &audioconvert, &flacenc, &fakesink])?;
 | 
					
						
							|  |  |  |     gstreamer::Element::link_many(&[&cdparanoiasrc, &queue, &audioconvert, &flacenc, &fakesink])?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let bus = pipeline.get_bus().ok_or(Error::u(String::from("Failed to get bus from pipeline.")))?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Run the pipeline into the paused state and wait for the resulting TOC message on the bus.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pipeline.set_state(gstreamer::State::Paused)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let msg = bus.timed_pop_filtered(ClockTime::from_seconds(5),
 | 
					
						
							|  |  |  |         &vec![MessageType::Toc, MessageType::Error]);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let toc = match msg {
 | 
					
						
							|  |  |  |         Some(msg) => match msg.view() {
 | 
					
						
							|  |  |  |             MessageView::Error(err) => Err(Error::os(err.get_error())),
 | 
					
						
							|  |  |  |             MessageView::Toc(toc) => Ok(toc.get_toc().0),
 | 
					
						
							|  |  |  |             _ => Err(Error::u(format!("Unexpected message from GStreamer: {:?}", msg))),
 | 
					
						
							|  |  |  |         },
 | 
					
						
							|  |  |  |         None => Err(Error::Timeout(
 | 
					
						
							|  |  |  |             format!("Timeout while waiting for first message from GStreamer."))),
 | 
					
						
							|  |  |  |     }?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pipeline.set_state(gstreamer::State::Ready)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Replace the fakesink with the real filesink. This won't need to be synced to the pipeline
 | 
					
						
							|  |  |  |     // state, because we will set the whole pipeline's state to playing later.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     gstreamer::Element::unlink(&flacenc, &fakesink);
 | 
					
						
							|  |  |  |     fakesink.set_state(gstreamer::State::Null)?;
 | 
					
						
							|  |  |  |     pipeline.remove(&fakesink)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let filesink = gstreamer::ElementFactory::make("filesink", None)?;
 | 
					
						
							|  |  |  |     pipeline.add(&filesink)?;
 | 
					
						
							|  |  |  |     gstreamer::Element::link(&flacenc, &filesink)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get track data from the toc message that was received above.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let tmp_dir = create_tmp_dir()?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for entry in toc.get_entries() {
 | 
					
						
							|  |  |  |         if entry.get_entry_type() == TocEntryType::Track {
 | 
					
						
							|  |  |  |             let duration = entry.get_tags()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("No tags in TOC entry.")))?
 | 
					
						
							|  |  |  |                 .get::<Duration>()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("No duration tag found in TOC entry.")))?
 | 
					
						
							|  |  |  |                 .get()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("Failed to unwrap duration tag from TOC entry.")))?
 | 
					
						
							|  |  |  |                 .mseconds()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("Failed to unwrap track duration.")))?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let number = entry.get_tags()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("No tags in TOC entry.")))?
 | 
					
						
							|  |  |  |                 .get::<TrackNumber>()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(String::from("No track number tag found in TOC entry.")))?
 | 
					
						
							|  |  |  |                 .get()
 | 
					
						
							|  |  |  |                 .ok_or(Error::u(
 | 
					
						
							|  |  |  |                     String::from("Failed to unwrap track number tag from TOC entry.")))?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             hasher.update(duration.to_le_bytes());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let name = format!("Track {}", number);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let file_name = format!("track_{:02}.flac", number);
 | 
					
						
							|  |  |  |             let mut path = tmp_dir.clone();
 | 
					
						
							|  |  |  |             path.push(file_name);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let track = ImportTrack {
 | 
					
						
							|  |  |  |                 number,
 | 
					
						
							|  |  |  |                 name,
 | 
					
						
							|  |  |  |                 path,
 | 
					
						
							|  |  |  |                 duration,
 | 
					
						
							|  |  |  |             };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             tracks.push(track);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let source_id = base64::encode_config(hasher.finalize(), base64::URL_SAFE);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     info!("Successfully loaded audio CD with {} tracks.", tracks.len());
 | 
					
						
							|  |  |  |     info!("Source ID: {}", source_id);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let tracks_clone = tracks.clone();
 | 
					
						
							|  |  |  |     let copy = move || {
 | 
					
						
							|  |  |  |         for track in &tracks_clone {
 | 
					
						
							|  |  |  |             info!("Starting to rip track {}.", track.number);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             cdparanoiasrc.set_property("track", &track.number)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // The filesink needs to be reset to be able to change the file location.
 | 
					
						
							|  |  |  |             filesink.set_state(gstreamer::State::Null)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let path = track.path.to_str().unwrap();
 | 
					
						
							|  |  |  |             filesink.set_property("location", &path)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // This will also affect the filesink as expected.
 | 
					
						
							|  |  |  |             pipeline.set_state(gstreamer::State::Playing)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for msg in bus.iter_timed(ClockTime::none()) {
 | 
					
						
							|  |  |  |                 match msg.view() {
 | 
					
						
							|  |  |  |                     MessageView::Eos(..) => {
 | 
					
						
							|  |  |  |                         info!("Finished ripping track {}.", track.number);
 | 
					
						
							|  |  |  |                         pipeline.set_state(gstreamer::State::Ready)?;
 | 
					
						
							|  |  |  |                         break;
 | 
					
						
							|  |  |  |                     },
 | 
					
						
							|  |  |  |                     MessageView::Error(err) => {
 | 
					
						
							|  |  |  |                         pipeline.set_state(gstreamer::State::Null)?;
 | 
					
						
							|  |  |  |                         Err(Error::os(err.get_error()))?;
 | 
					
						
							|  |  |  |                     },
 | 
					
						
							|  |  |  |                     _ => (),
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pipeline.set_state(gstreamer::State::Null)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let session = ImportSession {
 | 
					
						
							|  |  |  |         source_id,
 | 
					
						
							|  |  |  |         tracks,
 | 
					
						
							|  |  |  |         copy: Some(Box::new(copy)),
 | 
					
						
							| 
									
										
										
										
											2021-03-28 17:28:49 +02:00
										 |  |  |         state_sender,
 | 
					
						
							|  |  |  |         state_receiver: Mutex::new(state_receiver),
 | 
					
						
							| 
									
										
										
										
											2021-02-20 19:03:26 +01:00
										 |  |  |     };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Ok(session)
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Create a new temporary directory and return its path.
 | 
					
						
							|  |  |  | fn create_tmp_dir() -> Result<PathBuf> {
 | 
					
						
							|  |  |  |     let mut tmp_dir = glib::get_tmp_dir().ok_or(Error::u(
 | 
					
						
							|  |  |  |         String::from("Failed to get temporary directory using glib::get_tmp_dir().")))?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let dir_name = format!("musicus-{}", rand::random::<u64>());
 | 
					
						
							|  |  |  |     tmp_dir.push(dir_name);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::fs::create_dir(&tmp_dir)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Ok(tmp_dir)
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 |