musixmatch-inofficial/tests/tests.rs
2025-01-13 05:20:49 +01:00

1121 lines
36 KiB
Rust

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<P: AsRef<Path>>(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<DefaultDirectRateLimiter> = 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::<HashMap<_, _>>();
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::<Vec<_>>();
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::<Vec<_>>()
.await
.into_iter()
.map(Result::unwrap)
.collect::<Vec<_>>();
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<String>, 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<T: PartialOrd + std::fmt::Display>(a: T, b: T, msg: &str) {
assert!(a >= b, "expected >= {b} {msg}, got {a}");
}