use std::{ num::NonZeroU32, path::{Path, PathBuf}, sync::LazyLock, }; use governor::{DefaultDirectRateLimiter, Quota, RateLimiter}; use path_macro::path; use rstest::{fixture, rstest}; use time::macros::{date, datetime}; use musixmatch_inofficial::{ models::{AlbumId, ArtistId, TrackId}, Error, Musixmatch, }; fn testfile>(name: P) -> PathBuf { path!(env!("CARGO_MANIFEST_DIR") / "testfiles" / name) } #[fixture] async fn mxm() -> Musixmatch { static LOGIN_LOCK: tokio::sync::OnceCell<()> = tokio::sync::OnceCell::const_new(); static MXM_LIMITER: LazyLock = LazyLock::new(|| { RateLimiter::direct(if std::env::var("CI").is_ok() { Quota::with_period(std::time::Duration::from_millis(1500)).unwrap() } else { Quota::per_second(NonZeroU32::new(4).unwrap()) }) }); MXM_LIMITER.until_ready().await; let mut mxm = Musixmatch::builder(); if let (Ok(email), Ok(password)) = ( std::env::var("MUSIXMATCH_EMAIL"), std::env::var("MUSIXMATCH_PASSWORD"), ) { mxm = mxm.credentials(email, password); } let mxm = mxm.build().unwrap(); LOGIN_LOCK.get_or_try_init(|| mxm.login()).await.unwrap(); mxm } mod album { use super::*; use musixmatch_inofficial::models::AlbumType; #[rstest] #[case::id(AlbumId::AlbumId(14248253))] #[case::musicbrainz(AlbumId::Musicbrainz("6c3cf9d8-88a8-43ed-850b-55813f01e451"))] #[tokio::test] async fn by_id(#[case] album_id: AlbumId<'_>, #[future] mxm: Musixmatch) { let album = mxm.await.album(album_id).await.unwrap(); assert_eq!(album.album_id, 14248253); assert_eq!( album.album_mbid.unwrap(), "6c3cf9d8-88a8-43ed-850b-55813f01e451" ); assert_eq!(album.album_name, "Gangnam Style (강남스타일)"); assert!(album.album_rating > 20); assert_eq!(album.album_track_count, 0); assert_eq!(album.album_release_date.unwrap(), date!(2012 - 01 - 01)); assert_eq!(album.album_release_type, AlbumType::Single); assert_eq!(album.artist_id, 410698); assert_eq!(album.artist_name, "PSY"); let first_pri_genre = &album.primary_genres.music_genre_list[0].music_genre; assert_eq!(first_pri_genre.music_genre_id, 14); assert_eq!(first_pri_genre.music_genre_parent_id, 34); assert_eq!(first_pri_genre.music_genre_name, "Pop"); assert_eq!(first_pri_genre.music_genre_name_extended, "Pop"); assert_eq!(first_pri_genre.music_genre_vanity.as_ref().unwrap(), "Pop"); let first_sec_genre = &album.secondary_genres.music_genre_list[0].music_genre; assert_eq!(first_sec_genre.music_genre_id, 17); assert_eq!(first_sec_genre.music_genre_parent_id, 34); assert_eq!(first_sec_genre.music_genre_name, "Dance"); assert_eq!(first_sec_genre.music_genre_name_extended, "Dance"); assert_eq!( first_sec_genre.music_genre_vanity.as_ref().unwrap(), "Dance" ); assert_eq!( album.album_copyright.unwrap(), "© 2012 Schoolboy/Universal Republic Records, a division of UMG Recordings, Inc." ); assert_eq!( album.album_label.unwrap(), "Silent Records/Universal Republic Records" ); assert_eq!(album.album_vanity_id, "410698/Gangnam-Style-Single"); assert!(album.updated_time > datetime!(2022-6-3 0:00 UTC)); assert_imgurl(&album.album_coverart_100x100, "/26544045.jpg"); assert_imgurl(&album.album_coverart_350x350, "/26544045_350_350.jpg"); assert_imgurl(&album.album_coverart_500x500, "/26544045_500_500.jpg"); } #[rstest] #[tokio::test] async fn album_ep(#[future] mxm: Musixmatch) { let album = mxm.await.album(AlbumId::AlbumId(23976123)).await.unwrap(); assert_eq!(album.album_name, "Waldbrand EP"); // assert_eq!(album.album_release_type, AlbumType::Ep); assert_eq!(album.album_release_date, Some(date!(2016 - 09 - 30))); } #[rstest] #[tokio::test] async fn by_id_missing(#[future] mxm: Musixmatch) { let err = mxm .await .album(AlbumId::AlbumId(999999999999)) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] #[ignore] async fn artist_albums(#[future] mxm: Musixmatch) { let albums = mxm .await .artist_albums(ArtistId::ArtistId(1039), None, 10, 1) .await .unwrap(); assert_eq!(albums.len(), 10); } #[rstest] #[tokio::test] async fn artist_albums_missing(#[future] mxm: Musixmatch) { let err = mxm .await .artist_albums(ArtistId::ArtistId(999999999999), None, 10, 1) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn charts(#[future] mxm: Musixmatch) { let albums = mxm.await.chart_albums("US", 10, 1).await.unwrap(); assert_eq!(albums.len(), 10); } } mod artist { use super::*; #[rstest] #[case::id(ArtistId::ArtistId(410698))] #[case::musicbrainz(ArtistId::Musicbrainz("f99b7d67-4e63-4678-aa66-4c6ac0f7d24a"))] #[tokio::test] async fn by_id(#[case] artist_id: ArtistId<'_>, #[future] mxm: Musixmatch) { let artist = mxm.await.artist(artist_id).await.unwrap(); // dbg!(&artist); assert_eq!(artist.artist_id, 410698); assert_eq!( artist.artist_mbid.unwrap(), "f99b7d67-4e63-4678-aa66-4c6ac0f7d24a" ); assert_eq!(artist.artist_name, "PSY"); assert!( artist.artist_name_translation_list.iter().any(|tl| { tl.artist_name_translation.language == "KO" && tl.artist_name_translation.translation == "싸이" }), "missing Korean translation in: {:?}", artist.artist_name_translation_list ); assert_eq!(artist.artist_country.unwrap(), "KR"); assert!(artist.artist_rating > 50); let first_genre = &artist.primary_genres.music_genre_list[0].music_genre; assert_eq!(first_genre.music_genre_id, 14); assert_eq!(first_genre.music_genre_parent_id, 34); assert_eq!(first_genre.music_genre_name, "Pop"); assert_eq!(first_genre.music_genre_name_extended, "Pop"); assert_eq!(first_genre.music_genre_vanity.as_ref().unwrap(), "Pop"); assert_eq!( artist.artist_twitter_url.unwrap(), "https://twitter.com/psy_oppa" ); assert_eq!( artist.artist_facebook_url.unwrap(), "https://www.facebook.com/officialpsy" ); assert_eq!(artist.artist_vanity_id, "410698"); assert!(artist.updated_time > datetime!(2016-6-30 0:00 UTC)); assert_eq!(artist.begin_date_year.unwrap(), 1977); assert_eq!(artist.begin_date.unwrap(), date!(1977 - 12 - 31)); assert_eq!(artist.end_date_year, None); assert_eq!(artist.end_date, None); let image = artist.artist_image.first().expect("artist image"); assert_eq!(image.image_id, 20511); let image_format = &image .image_format_list .iter() .find(|img| img.image_format.height == 250 && img.image_format.width == 250) .expect("image format 250px") .image_format; assert!( image_format.image_url.starts_with( "https://static.musixmatch.com/images-storage/mxmimages/1/1/5/0/2/20511_14.jpg?" ), "url: {}", image_format.image_url ); } #[rstest] #[tokio::test] async fn by_id_missing(#[future] mxm: Musixmatch) { let err = mxm .await .artist(ArtistId::ArtistId(999999999999)) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn related(#[future] mxm: Musixmatch) { let artists = mxm .await .artist_related(ArtistId::ArtistId(26485840), 10, 1) .await .unwrap(); assert_eq!(artists.len(), 10); } #[rstest] #[tokio::test] async fn related_missing(#[future] mxm: Musixmatch) { let err = mxm .await .artist_related(ArtistId::ArtistId(999999999999), 10, 1) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn search(#[future] mxm: Musixmatch) { let artists = mxm .await .artist_search("Snollebollekes", 5, 1) .await .unwrap(); let artist = &artists[0]; assert_eq!(artist.artist_id, 25344078); assert_eq!(artist.artist_name, "Snollebollekes"); } #[rstest] #[tokio::test] async fn search_empty(#[future] mxm: Musixmatch) { let artists = mxm .await .artist_search( "Rindfleischettikettierungsüberwachungsaufgabenübertragungsgesetz", 5, 1, ) .await .unwrap(); assert_eq!(artists.len(), 0); } #[rstest] #[tokio::test] async fn charts(#[future] mxm: Musixmatch) { let artists = mxm.await.chart_artists("US", 10, 1).await.unwrap(); assert_eq!(artists.len(), 10); } #[rstest] #[tokio::test] async fn charts_no_country(#[future] mxm: Musixmatch) { let artists = mxm.await.chart_artists("XY", 10, 1).await.unwrap(); assert_eq!(artists.len(), 10); } } mod track { use std::collections::HashMap; use super::*; use musixmatch_inofficial::models::{ChartName, MxmEntityType, SortOrder}; #[rstest] #[case::no_translation(false, false)] #[case::translation_2c(true, false)] #[case::translation_3c(true, true)] #[tokio::test] async fn from_match( #[case] translation_status: bool, #[case] lang_3c: bool, #[future] mxm: Musixmatch, ) { let track = mxm .await .matcher_track( "Poker Face", "Lady Gaga", "The Fame", translation_status, lang_3c, false, ) .await .unwrap(); // dbg!(&track); assert_eq!(track.track_id, 85213841); // assert_eq!( // track.track_mbid.unwrap(), // "080975b0-39b1-493c-ae64-5cb3292409bb" // ); // assert_eq!(track.track_isrc.unwrap(), "USUM70824409"); assert!( track.commontrack_isrcs[0] .iter() .any(|isrc| isrc == "USUM70824409"), "commontrack_isrcs: {:?}", &track.commontrack_isrcs[0], ); assert_eq!(track.track_spotify_id.unwrap(), "1QV6tiMFM6fSOKOGLMHYYg"); assert!( track .commontrack_spotify_ids .iter() .any(|spid| spid == "1QV6tiMFM6fSOKOGLMHYYg"), "commontrack_spotify_ids: {:?}", track.commontrack_spotify_ids, ); assert_eq!(track.track_name, "Poker Face"); assert!(track.track_rating > 50); assert_eq!(track.commontrack_id, 47672612); assert!(!track.instrumental); assert!(track.explicit); assert!(track.has_subtitles); assert!(track.has_richsync); assert!(track.has_track_structure); assert!(track.num_favourite > 50); assert!(track.lyrics_id.is_some()); assert_eq!(track.subtitle_id.unwrap(), 36450705); assert_eq!(track.album_id, 20960801); assert_eq!(track.album_name, "The Fame"); assert_eq!(track.artist_id, 378462); assert_eq!( track.artist_mbid.unwrap(), "650e7db6-b795-4eb5-a702-5ea2fc46c848" ); assert_eq!(track.artist_name, "Lady Gaga"); assert_imgurl(&track.album_coverart_100x100, "/32133892.jpg"); assert_imgurl(&track.album_coverart_350x350, "/32133892_350_350.jpg"); assert_imgurl(&track.album_coverart_500x500, "/32133892_500_500.jpg"); assert_imgurl(&track.album_coverart_800x800, "/32133892_800_800.jpg"); assert_eq!(track.commontrack_vanity_id, "Lady-Gaga/poker-face-1"); let first_release = track.first_release_date.unwrap(); assert_eq!(first_release.date(), date!(2008 - 1 - 1)); assert!(track.updated_time > datetime!(2023-1-17 0:00 UTC)); let first_pri_genre = &track.primary_genres.music_genre_list[0].music_genre; assert_eq!(first_pri_genre.music_genre_id, 14); assert_eq!(first_pri_genre.music_genre_parent_id, 34); assert_eq!(first_pri_genre.music_genre_name, "Pop"); assert_eq!(first_pri_genre.music_genre_name_extended, "Pop"); assert_eq!(first_pri_genre.music_genre_vanity.as_ref().unwrap(), "Pop"); let first_sec_genre = &track.secondary_genres.music_genre_list[0].music_genre; assert_eq!(first_sec_genre.music_genre_id, 17); assert_eq!(first_sec_genre.music_genre_parent_id, 34); assert_eq!(first_sec_genre.music_genre_name, "Dance"); assert_eq!(first_sec_genre.music_genre_name_extended, "Dance"); assert_eq!( first_sec_genre.music_genre_vanity.as_ref().unwrap(), "Dance" ); if translation_status { assert!( track.track_lyrics_translation_status.iter().all(|tl| { (if lang_3c { tl.from.as_deref() == Some("eng") } else { tl.from.as_deref() == Some("en") }) && tl.perc >= 0.0 && tl.perc <= 1.0 }), "translation: {:?}", track.track_lyrics_translation_status ); } else { assert!(track.track_lyrics_translation_status.is_empty()) } } #[rstest] #[case::trackid(TrackId::TrackId(167254015))] #[case::commontrack(TrackId::Commontrack(93933821))] #[case::vanity(TrackId::CommontrackVanity("Nightbirde-2/Girl-in-a-Bubble".into()))] #[case::isrc(TrackId::Isrc("QZDA41918667".into()))] #[case::spotify(TrackId::Spotify("2roGy5AYlaJpmL9CuXj6tT".into()))] #[tokio::test] async fn from_id(#[case] track_id: TrackId<'_>, #[future] mxm: Musixmatch) { let track = mxm.await.track(track_id, true, false, false).await.unwrap(); // dbg!(&track); assert_eq!(track.track_id, 167254015); assert_eq!(track.track_isrc.unwrap(), "QZDA41918667"); assert_eq!(track.track_spotify_id.unwrap(), "2roGy5AYlaJpmL9CuXj6tT"); assert_eq!(track.track_name, "Girl in a Bubble"); assert!(track.track_rating > 40); assert_eq!(track.track_length, 238); assert!(!track.explicit); assert!(track.has_lyrics); assert!(track.has_subtitles); assert!(track.num_favourite > 1); assert_eq!(track.lyrics_id.unwrap(), 25830616); assert_eq!(track.subtitle_id.unwrap(), 34307878); assert_eq!(track.album_id, 31842378); assert_eq!(track.album_name, "Girl in a Bubble"); assert_eq!(track.artist_id, 38035381); assert_eq!(track.artist_name, "Nightbirde"); assert_imgurl(&track.album_coverart_100x100, "/43015335.jpg"); assert_imgurl(&track.album_coverart_350x350, "/43015335_350_350.jpg"); assert_imgurl(&track.album_coverart_500x500, "/43015335_500_500.jpg"); assert_eq!(track.commontrack_vanity_id, "Nightbirde-2/Girl-in-a-Bubble"); let release_date = track.first_release_date.unwrap(); assert_eq!(release_date.date(), date!(2019 - 03 - 22)); assert!(track.updated_time > datetime!(2022-8-27 0:00 UTC)); let first_tstatus = &track.track_lyrics_translation_status[0]; assert_eq!(first_tstatus.from.as_deref(), Some("en")); assert!(first_tstatus.perc >= 0.0 && first_tstatus.perc <= 1.0); } #[rstest] #[tokio::test] #[test_log::test] async fn performer(#[future] mxm: Musixmatch) { let track = mxm .await .track(TrackId::TrackId(206591653), false, false, true) .await .unwrap(); let perf = track.performer_tagging.expect("performer tagging"); assert!(perf.completed); assert!(!perf.has_unknown); assert!(!perf.has_fan_chant); let artists = perf .resources .artists .into_iter() .map(|a| (a.artist_id, a)) .collect::>(); assert_eq!(artists.len(), 2); let jhayco = &artists[&53077263]; let bad_bunny = &artists[&33491954]; assert_eq!(jhayco.artist_name, "Jhayco"); assert_eq!(bad_bunny.artist_name, "Bad Bunny"); assert_eq!(bad_bunny.artist_image.len(), 1); for part in perf.content { assert!(!part.snippet.trim().is_empty(), "empty snippet"); assert_gte(part.performers.len(), 1, "part performers"); for performer in &part.performers { let pid = performer.fqid.expect("performer id"); assert_eq!(pid.typ, MxmEntityType::Artist); assert!(artists.contains_key(&pid.id)) } } } #[rstest] #[case::no_translation(false, false)] #[case::translation_2c(true, false)] #[case::translation_3c(true, true)] #[tokio::test] async fn from_id_translations( #[case] translation_status: bool, #[case] lang_3c: bool, #[future] mxm: Musixmatch, ) { let track = mxm .await .track( TrackId::Commontrack(47672612), translation_status, lang_3c, false, ) .await .unwrap(); // dbg!(&track); assert_eq!(track.track_id, 85213841); // assert_eq!( // track.track_mbid.unwrap(), // "080975b0-39b1-493c-ae64-5cb3292409bb" // ); // assert_eq!(track.track_isrc.unwrap(), "USUM70824409"); assert!( track.commontrack_isrcs[0] .iter() .any(|isrc| isrc == "USUM70824409"), "commontrack_isrcs: {:?}", &track.commontrack_isrcs[0], ); assert_eq!(track.track_spotify_id.unwrap(), "1QV6tiMFM6fSOKOGLMHYYg"); assert!( track .commontrack_spotify_ids .iter() .any(|spid| spid == "1QV6tiMFM6fSOKOGLMHYYg"), "commontrack_spotify_ids: {:?}", track.commontrack_spotify_ids, ); assert_eq!(track.track_name, "Poker Face"); assert!(track.track_rating > 50); assert_eq!(track.commontrack_id, 47672612); assert!(!track.instrumental); assert!(track.explicit); assert!(track.has_subtitles); assert!(track.has_richsync); assert!(track.has_track_structure); assert!(track.num_favourite > 50); assert!(track.lyrics_id.is_some()); assert_eq!(track.subtitle_id.unwrap(), 36450705); assert_eq!(track.album_id, 20960801); assert_eq!(track.album_name, "The Fame"); assert_eq!(track.artist_id, 378462); assert_eq!( track.artist_mbid.unwrap(), "650e7db6-b795-4eb5-a702-5ea2fc46c848" ); assert_eq!(track.artist_name, "Lady Gaga"); assert_imgurl(&track.album_coverart_100x100, "/32133892.jpg"); assert_imgurl(&track.album_coverart_350x350, "/32133892_350_350.jpg"); assert_imgurl(&track.album_coverart_500x500, "/32133892_500_500.jpg"); assert_imgurl(&track.album_coverart_800x800, "/32133892_800_800.jpg"); assert_eq!(track.commontrack_vanity_id, "Lady-Gaga/poker-face-1"); let first_release = track.first_release_date.unwrap(); assert_eq!(first_release.date(), date!(2008 - 1 - 1)); assert!(track.updated_time > datetime!(2023-1-17 0:00 UTC)); let first_pri_genre = &track.primary_genres.music_genre_list[0].music_genre; assert_eq!(first_pri_genre.music_genre_id, 14); assert_eq!(first_pri_genre.music_genre_parent_id, 34); assert_eq!(first_pri_genre.music_genre_name, "Pop"); assert_eq!(first_pri_genre.music_genre_name_extended, "Pop"); assert_eq!(first_pri_genre.music_genre_vanity.as_ref().unwrap(), "Pop"); let first_sec_genre = &track.secondary_genres.music_genre_list[0].music_genre; assert_eq!(first_sec_genre.music_genre_id, 17); assert_eq!(first_sec_genre.music_genre_parent_id, 34); assert_eq!(first_sec_genre.music_genre_name, "Dance"); assert_eq!(first_sec_genre.music_genre_name_extended, "Dance"); assert_eq!( first_sec_genre.music_genre_vanity.as_ref().unwrap(), "Dance" ); if translation_status { assert!( track.track_lyrics_translation_status.iter().all(|tl| { (if lang_3c { tl.from.as_deref() == Some("eng") } else { tl.from.as_deref() == Some("en") }) && tl.perc >= 0.0 && tl.perc <= 1.0 }), "translation: {:?}", track.track_lyrics_translation_status ); } else { assert!(track.track_lyrics_translation_status.is_empty()) } } #[rstest] #[tokio::test] async fn from_id_missing(#[future] mxm: Musixmatch) { let err = mxm .await .track(TrackId::TrackId(999999999999), false, false, false) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn album_tracks(#[future] mxm: Musixmatch) { let tracks = mxm .await .album_tracks(AlbumId::AlbumId(17118624), true, 20, 1) .await .unwrap(); // dbg!(&tracks); let track_names = tracks .iter() .map(|t| t.track_name.to_owned()) .collect::>(); assert_eq!( track_names, vec![ "Gäa", "Vergiss mein nicht", "Orome", "Falke flieg", "Minne", "Das Lied der Ahnen", "Hörst du den Wind", "Nan Úye", "Faolan", "Hymne der Nacht", "Avalon", "Tolo Nan", "Oonagh", ] ); tracks.iter().for_each(|t| { assert!(t.has_lyrics); assert!(t.has_subtitles); }); } #[rstest] #[tokio::test] async fn album_missing(#[future] mxm: Musixmatch) { let err = mxm .await .album_tracks(AlbumId::AlbumId(999999999999), false, 20, 1) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[case::top(ChartName::Top)] #[case::hot(ChartName::Hot)] #[tokio::test] async fn charts(#[case] chart_name: ChartName, #[future] mxm: Musixmatch) { let tracks = mxm .await .chart_tracks("US", chart_name, true, 20, 1) .await .unwrap(); assert_eq!(tracks.len(), 20); } #[rstest] #[tokio::test] async fn search(#[future] mxm: Musixmatch) { let tracks = mxm .await .track_search() .q_artist("Lena") .q_track("Satellite") .s_track_rating(SortOrder::Desc) .send(1, 0) .await .unwrap(); // dbg!(&tracks); assert_eq!(tracks.len(), 1); let track = &tracks[0]; assert_eq!(track.commontrack_id, 72643758); assert_eq!(track.track_name, "Satellite"); assert_eq!(track.artist_name, "Lena"); } #[rstest] #[tokio::test] async fn search_lyrics(#[future] mxm: Musixmatch) { let tracks = mxm .await .track_search() .q_lyrics("the whole world stops and stares for a while") .s_track_rating(SortOrder::Desc) .send(10, 1) .await .unwrap(); assert_gte(tracks.len(), 8, "tracks"); let track = &tracks[0]; assert_eq!(track.track_name, "Just the Way You Are"); assert_eq!(track.artist_name, "Bruno Mars"); } #[rstest] #[tokio::test] async fn search_empty(#[future] mxm: Musixmatch) { let artists = mxm .await .track_search() .q("Rindfleischettikettierungsüberwachungsaufgabenübertragungsgesetz") .send(10, 1) .await .unwrap(); assert_eq!(artists.len(), 0); } #[rstest] #[tokio::test] async fn genres(#[future] mxm: Musixmatch) { let genres = mxm.await.genres().await.unwrap(); assert!(genres.len() > 360); dbg!(&genres); } #[rstest] #[tokio::test] async fn snippet(#[future] mxm: Musixmatch) { let snippet = mxm .await .track_snippet(TrackId::Commontrack(8874280)) .await .unwrap(); assert_eq!(snippet.snippet_id, 23036767); assert_eq!(snippet.snippet_language.unwrap(), "en"); assert!(!snippet.instrumental); assert!(snippet.updated_time > datetime!(2022-8-29 0:00 UTC)); assert_eq!( snippet.snippet_body, "There's not a thing that I would change" ); } } mod lyrics { use futures::stream::{self, StreamExt}; use std::{fs::File, io::BufWriter}; use super::*; #[rstest] #[tokio::test] async fn from_match(#[future] mxm: Musixmatch) { let lyrics = mxm.await.matcher_lyrics("Shine", "Spektrem").await.unwrap(); // dbg!(&lyrics); assert_eq!(lyrics.lyrics_id, 34583240); assert!(!lyrics.instrumental); assert!(!lyrics.explicit); assert!( lyrics .lyrics_body .starts_with("Eyes in the sky gazing far into the night\n"), "got: {}", lyrics.lyrics_body ); assert_eq!(lyrics.lyrics_language.unwrap(), "en"); assert_eq!(lyrics.lyrics_language_description.unwrap(), "English"); let copyright = lyrics.lyrics_copyright.unwrap(); assert!(copyright.contains("Jesse Warren"), "copyright: {copyright}",); assert!(lyrics.updated_time > datetime!(2021-6-3 0:00 UTC)); } #[rstest] #[case::trackid(TrackId::TrackId(205688271))] #[case::commontrack(TrackId::Commontrack(118480583))] #[case::vanity(TrackId::CommontrackVanity("aespa/Black-Mamba".into()))] #[case::isrc(TrackId::Isrc("KRA302000590".into()))] #[case::spotify(TrackId::Spotify("1t2qYCAjUAoGfeFeoBlK51".into()))] #[tokio::test] async fn from_id(#[case] track_id: TrackId<'_>, #[future] mxm: Musixmatch) { let lyrics = mxm.await.track_lyrics(track_id).await.unwrap(); // dbg!(&lyrics); assert_eq!(lyrics.lyrics_id, 36846057); assert_eq!(lyrics.lyrics_language.unwrap(), "ko"); assert_eq!(lyrics.lyrics_language_description.unwrap(), "Korean"); let copyright = lyrics.lyrics_copyright.unwrap(); assert!( copyright.contains("Kenneth Scott Chesak"), "copyright: {copyright}", ); assert!(lyrics.updated_time > datetime!(2022-8-27 0:00 UTC)); } /// This track has no lyrics #[rstest] #[tokio::test] async fn instrumental(#[future] mxm: Musixmatch) { let lyrics = mxm .await .matcher_lyrics("drivers license", "Bobby G") .await .unwrap(); assert_eq!(lyrics.lyrics_id, 25891609); assert!(lyrics.instrumental); assert!(!lyrics.explicit); assert_eq!(lyrics.lyrics_body, ""); assert_eq!(lyrics.lyrics_language, None); assert_eq!(lyrics.lyrics_language_description, None); assert_eq!(lyrics.lyrics_copyright, None); assert!(lyrics.updated_time > datetime!(2021-6-21 0:00 UTC)); } /// This track does not exist #[rstest] #[tokio::test] async fn missing(#[future] mxm: Musixmatch) { let err = mxm .await .track_lyrics(TrackId::Spotify("2gwMMr1a4aXXN5L6KC80Pu".into())) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn download_testdata(#[future] mxm: Musixmatch) { let mxm = mxm.await; let json_path = testfile("lyrics.json"); if json_path.exists() { return; } let lyrics = mxm .track_lyrics(TrackId::Commontrack(18576954)) .await .unwrap(); let json_file = File::create(json_path).unwrap(); serde_json::to_writer_pretty(BufWriter::new(json_file), &lyrics).unwrap(); } #[rstest] #[tokio::test] async fn download_testdata_translation(#[future] mxm: Musixmatch) { let mxm = mxm.await; let json_path = testfile("translation.json"); if json_path.exists() { return; } let translations = mxm .track_lyrics_translation(TrackId::Commontrack(18576954), "de") .await .unwrap(); let json_file = File::create(json_path).unwrap(); serde_json::to_writer_pretty(BufWriter::new(json_file), &translations).unwrap(); } #[rstest] #[tokio::test] async fn concurrency(#[future] mxm: Musixmatch) { let mxm = mxm.await; let album = mxm .album_tracks( musixmatch_inofficial::models::AlbumId::AlbumId(17118624), true, 20, 1, ) .await .unwrap(); let x = stream::iter(album) .map(|track| { mxm.track_lyrics(musixmatch_inofficial::models::TrackId::TrackId( track.track_id, )) }) .buffered(8) .collect::>() .await .into_iter() .map(Result::unwrap) .collect::>(); dbg!(x); } } mod subtitles { use std::{fs::File, io::BufWriter}; use super::*; use musixmatch_inofficial::models::SubtitleFormat; #[rstest] #[tokio::test] async fn from_match(#[future] mxm: Musixmatch) { let subtitle = mxm .await .matcher_subtitle( "Shine", "Spektrem", SubtitleFormat::Json, Some(315.0), Some(1.0), ) .await .unwrap(); // dbg!(&subtitle); assert_eq!(subtitle.subtitle_id, 35340319); assert_eq!(subtitle.subtitle_language.unwrap(), "en"); assert_eq!(subtitle.subtitle_language_description.unwrap(), "English"); let copyright = subtitle.lyrics_copyright.unwrap(); assert!(copyright.contains("Jesse Warren"), "copyright: {copyright}",); assert_eq!(subtitle.subtitle_length, 316); assert!(subtitle.updated_time > datetime!(2021-6-30 0:00 UTC)); } #[rstest] #[case::trackid(TrackId::TrackId(205688271))] #[case::commontrack(TrackId::Commontrack(118480583))] #[case::vanity(TrackId::CommontrackVanity("aespa/Black-Mamba".into()))] #[case::isrc(TrackId::Isrc("KRA302000590".into()))] #[case::spotify(TrackId::Spotify("1t2qYCAjUAoGfeFeoBlK51".into()))] #[tokio::test] async fn from_id(#[case] track_id: TrackId<'_>, #[future] mxm: Musixmatch) { let subtitle = mxm .await .track_subtitle(track_id, SubtitleFormat::Json, Some(175.0), Some(1.0)) .await .unwrap(); // dbg!(&subtitle); assert_eq!(subtitle.subtitle_id, 36476905); assert_eq!(subtitle.subtitle_language.unwrap(), "ko"); assert_eq!(subtitle.subtitle_language_description.unwrap(), "Korean"); let copyright = subtitle.lyrics_copyright.unwrap(); assert!( copyright.contains("Kenneth Scott Chesak"), "copyright: {copyright}", ); assert_eq!(subtitle.subtitle_length, 175); assert!(subtitle.updated_time > datetime!(2022-8-27 0:00 UTC)); } /// This track has no lyrics #[rstest] #[tokio::test] async fn instrumental(#[future] mxm: Musixmatch) { let err = mxm .await .matcher_subtitle( "drivers license", "Bobby G", SubtitleFormat::Json, Some(246.0), Some(1.0), ) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } /// This track has not been synced #[rstest] #[tokio::test] async fn unsynced(#[future] mxm: Musixmatch) { let err = mxm .await .track_subtitle( TrackId::Spotify("6oaWIABGL7eeiMILEDyGX1".into()), SubtitleFormat::Json, Some(213.0), Some(1.0), ) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } /// Try to get subtitles with wrong length parameter #[rstest] #[tokio::test] async fn wrong_length(#[future] mxm: Musixmatch) { let err = mxm .await .track_subtitle( TrackId::Commontrack(118480583), SubtitleFormat::Json, Some(200.0), Some(1.0), ) .await .unwrap_err(); assert!(matches!(err, Error::NotFound)); } #[rstest] #[tokio::test] async fn download_testdata(#[future] mxm: Musixmatch) { let json_path = testfile("subtitles.json"); if json_path.exists() { return; } let subtitle = mxm .await .track_subtitle( TrackId::Commontrack(18576954), SubtitleFormat::Json, Some(259.0), Some(1.0), ) .await .unwrap(); let json_file = File::create(json_path).unwrap(); serde_json::to_writer_pretty(BufWriter::new(json_file), &subtitle).unwrap(); } } mod translation { use std::{fs::File, io::BufReader}; use musixmatch_inofficial::models::{Lyrics, Subtitle, TranslationList, TranslationMap}; use crate::testfile; #[test] fn translation_test() { let lyrics_path = testfile("lyrics.json"); let subtitles_path = testfile("subtitles.json"); let translation_path = testfile("translation.json"); let lyrics: Lyrics = serde_json::from_reader(BufReader::new(File::open(lyrics_path).unwrap())).unwrap(); let subtitle: Subtitle = serde_json::from_reader(BufReader::new(File::open(subtitles_path).unwrap())).unwrap(); let translations: TranslationList = serde_json::from_reader(BufReader::new(File::open(translation_path).unwrap())).unwrap(); let t_map = TranslationMap::from(translations); let lyrics_trans = t_map.translate_lyrics(&lyrics.lyrics_body); let expected_lyrics = std::fs::read_to_string(testfile("translated_lyrics.txt")).unwrap(); assert_eq!(lyrics_trans.trim(), expected_lyrics.trim()); let subtitles_trans = t_map.translate_subtitles(&subtitle.to_lines().unwrap()); let expected_lrc = std::fs::read_to_string(testfile("translated_subtitles.lrc")).unwrap(); let expected_ttml = std::fs::read_to_string(testfile("translated_subtitles.xml")).unwrap(); assert_eq!(subtitles_trans.to_lrc().trim(), expected_lrc.trim()); assert_eq!(subtitles_trans.to_ttml().trim(), expected_ttml.trim()); } } #[track_caller] fn assert_imgurl(url: &Option, ends_with: &str) { assert!( url.as_deref().is_some_and( |url| url.starts_with("https://s.mxmcdn.net/images-storage/") && url.ends_with(ends_with) ), "expected url ending with {ends_with}\ngot {:?}", url ); } /// Assert that number A is greater than or equal to number B #[track_caller] fn assert_gte(a: T, b: T, msg: &str) { assert!(a >= b, "expected >= {b} {msg}, got {a}"); }