2010-03-24 00:11:46 +01:00
|
|
|
|
/* This file is part of Clementine.
|
2010-11-20 14:27:10 +01:00
|
|
|
|
Copyright 2010, David Sansome <me@davidsansome.com>
|
2010-03-24 00:11:46 +01:00
|
|
|
|
|
|
|
|
|
Clementine is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
Clementine is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
2011-03-13 21:01:24 +01:00
|
|
|
|
#include "config.h"
|
2010-08-03 21:29:49 +02:00
|
|
|
|
#include "core/encoding.h"
|
2010-05-10 23:50:31 +02:00
|
|
|
|
#include "core/song.h"
|
|
|
|
|
#include "radio/fixlastfm.h"
|
2010-03-01 02:47:50 +01:00
|
|
|
|
#include <lastfm/Track>
|
|
|
|
|
|
2010-03-06 21:08:01 +01:00
|
|
|
|
#include "gmock/gmock.h"
|
2010-03-01 02:47:50 +01:00
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
2010-03-01 16:40:12 +01:00
|
|
|
|
#include "test_utils.h"
|
2010-03-06 21:08:01 +01:00
|
|
|
|
#include "mock_taglib.h"
|
2010-03-01 02:47:50 +01:00
|
|
|
|
|
2010-03-07 16:26:54 +01:00
|
|
|
|
#include <QTemporaryFile>
|
2010-06-02 14:31:40 +02:00
|
|
|
|
#include <QTextCodec>
|
2010-03-07 16:26:54 +01:00
|
|
|
|
|
2011-02-04 14:29:49 +01:00
|
|
|
|
#include <id3v2tag.h>
|
2010-06-03 14:36:43 +02:00
|
|
|
|
|
2010-03-01 02:47:50 +01:00
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
class SongTest : public ::testing::Test {
|
2010-03-06 21:08:01 +01:00
|
|
|
|
protected:
|
|
|
|
|
static void SetUpTestCase() {
|
|
|
|
|
// Return something from uninteresting mock functions.
|
|
|
|
|
testing::DefaultValue<TagLib::String>::Set("foobarbaz");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MockFileRefFactory mock_factory_;
|
2010-03-01 02:47:50 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2011-03-13 21:01:24 +01:00
|
|
|
|
#ifdef HAVE_LIBLASTFM
|
2010-03-01 02:47:50 +01:00
|
|
|
|
TEST_F(SongTest, InitsFromLastFM) {
|
|
|
|
|
Song song;
|
|
|
|
|
lastfm::MutableTrack track;
|
|
|
|
|
track.setTitle("Foo");
|
|
|
|
|
lastfm::Artist artist("Bar");
|
|
|
|
|
track.setArtist(artist);
|
|
|
|
|
lastfm::Album album(artist, "Baz");
|
|
|
|
|
track.setAlbum(album);
|
|
|
|
|
|
|
|
|
|
song.InitFromLastFM(track);
|
|
|
|
|
EXPECT_EQ("Foo", song.title());
|
|
|
|
|
EXPECT_EQ("Baz", song.album());
|
|
|
|
|
EXPECT_EQ("Bar", song.artist());
|
|
|
|
|
}
|
2011-03-13 21:01:24 +01:00
|
|
|
|
#endif // HAVE_LIBLASTFM
|
2010-03-01 02:47:50 +01:00
|
|
|
|
|
2010-03-06 21:08:01 +01:00
|
|
|
|
TEST_F(SongTest, InitsFromFile) {
|
2010-03-07 16:26:54 +01:00
|
|
|
|
QTemporaryFile temp;
|
|
|
|
|
temp.open();
|
|
|
|
|
mock_factory_.ExpectCall(temp.fileName(), "Foo", "Bar", "Baz");
|
2010-03-06 21:08:01 +01:00
|
|
|
|
Song song(&mock_factory_);
|
2010-03-07 16:26:54 +01:00
|
|
|
|
song.InitFromFile(temp.fileName(), 42);
|
2010-03-06 21:08:01 +01:00
|
|
|
|
EXPECT_EQ("Foo", song.title());
|
|
|
|
|
EXPECT_EQ("Bar", song.artist());
|
|
|
|
|
EXPECT_EQ("Baz", song.album());
|
|
|
|
|
}
|
2010-03-01 02:47:50 +01:00
|
|
|
|
|
2010-05-11 14:03:55 +02:00
|
|
|
|
TEST_F(SongTest, DetectsWindows1251) {
|
|
|
|
|
char cp1251[] = { 0xc2, 0xfb, 0xe4, 0xfb, 0xf5, 0xe0, 0xe9, 0x00 }; // Выдыхай
|
|
|
|
|
UniversalEncodingHandler handler;
|
|
|
|
|
TagLib::ByteVector bytes(cp1251);
|
|
|
|
|
TagLib::String str = handler.parse(bytes);
|
|
|
|
|
EXPECT_FALSE(str.isAscii());
|
|
|
|
|
EXPECT_FALSE(str.isLatin1());
|
|
|
|
|
EXPECT_STREQ("Выдыхай", str.to8Bit(true).c_str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, LeavesASCIIAlone) {
|
2010-05-26 00:19:30 +02:00
|
|
|
|
const char* ascii = "foobar";
|
2010-05-11 14:03:55 +02:00
|
|
|
|
UniversalEncodingHandler handler;
|
|
|
|
|
TagLib::ByteVector bytes(ascii);
|
|
|
|
|
TagLib::String str = handler.parse(bytes);
|
|
|
|
|
EXPECT_TRUE(str.isAscii());
|
|
|
|
|
EXPECT_TRUE(str.isLatin1());
|
|
|
|
|
EXPECT_STREQ("foobar", str.to8Bit(false).c_str());
|
|
|
|
|
}
|
|
|
|
|
|
2010-06-02 14:31:40 +02:00
|
|
|
|
TEST_F(SongTest, FixesCP866) {
|
|
|
|
|
const char cp866[] = { 0x8a, 0xa8, 0xad, 0xae, '\0' }; // Кино
|
|
|
|
|
TagLib::ByteVector bytes(cp866);
|
|
|
|
|
TagLib::String str(bytes);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 14:31:40 +02:00
|
|
|
|
EXPECT_EQ(4, fixed.length());
|
|
|
|
|
EXPECT_STREQ("Кино", fixed.toUtf8().constData());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FixesWindows1251) {
|
|
|
|
|
const char w1251[] = { 0xca, 0xe8, 0xed, 0xee, '\0' }; // Кино
|
|
|
|
|
TagLib::ByteVector bytes(w1251);
|
|
|
|
|
TagLib::String str(bytes);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 14:31:40 +02:00
|
|
|
|
EXPECT_EQ(4, fixed.length());
|
|
|
|
|
EXPECT_STREQ("Кино", fixed.toUtf8().constData());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, DoesNotFixAscii) {
|
|
|
|
|
TagLib::ByteVector bytes("foobar");
|
|
|
|
|
TagLib::String str(bytes);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 14:31:40 +02:00
|
|
|
|
EXPECT_EQ(6, fixed.length());
|
|
|
|
|
EXPECT_STREQ("foobar", fixed.toUtf8().constData());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, DoesNotFixUtf8) {
|
|
|
|
|
TagLib::ByteVector bytes("Кино");
|
|
|
|
|
TagLib::String str(bytes, TagLib::String::UTF8);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 14:31:40 +02:00
|
|
|
|
EXPECT_EQ(4, fixed.length());
|
|
|
|
|
EXPECT_STREQ("Кино", fixed.toUtf8().constData());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, DoesNotFixExtendedAscii) {
|
|
|
|
|
char latin1[] = { 'R', 0xf6, 'y', 'k', 's', 'o', 'p', 'p', 0x00 };
|
|
|
|
|
QTextCodec* codec = QTextCodec::codecForName("latin1");
|
|
|
|
|
QString unicode = codec->toUnicode(latin1);
|
|
|
|
|
TagLib::String str(latin1);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 14:31:40 +02:00
|
|
|
|
EXPECT_EQ(fixed, unicode);
|
|
|
|
|
}
|
|
|
|
|
|
2010-06-02 15:11:51 +02:00
|
|
|
|
TEST_F(SongTest, FixesUtf8MungedIntoLatin1) {
|
|
|
|
|
char latin1[] = { 'E', 's', 't', 'h', 'e', 'r', 0xe2, 0x80, 0x99, 's', 0x00 };
|
|
|
|
|
TagLib::String str(latin1, TagLib::String::Latin1);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
2010-06-02 15:11:51 +02:00
|
|
|
|
EXPECT_EQ(8, fixed.length());
|
|
|
|
|
EXPECT_EQ(QString::fromUtf8("Esther’s"), fixed);
|
|
|
|
|
}
|
|
|
|
|
|
2010-06-03 14:36:43 +02:00
|
|
|
|
TEST_F(SongTest, TakesMajorityVote) {
|
|
|
|
|
const char w1251[] = { 0xca, 0xe8, 0xed, 0xee, '\0' }; // Кино
|
|
|
|
|
// Actually windows-1251 but gets detected as windows-1252.
|
|
|
|
|
const char w1252[] = { 0xcf, '.', 0xc7, '.', '\0' }; // П.Э.
|
2010-06-03 15:26:46 +02:00
|
|
|
|
TagLib::ID3v2::Tag* tag = new TagLib::ID3v2::Tag;
|
|
|
|
|
tag->setTitle(w1251);
|
|
|
|
|
tag->setArtist(w1251);
|
|
|
|
|
tag->setAlbum(w1252);
|
2010-06-03 14:36:43 +02:00
|
|
|
|
|
|
|
|
|
UniversalEncodingHandler handler(NS_FILTER_NON_CJK);
|
2010-06-03 15:26:46 +02:00
|
|
|
|
QTemporaryFile temp;
|
|
|
|
|
temp.open();
|
|
|
|
|
MockFile* file = new MockFile(tag, temp.fileName());
|
|
|
|
|
TagLib::FileRef ref(file);
|
|
|
|
|
EXPECT_EQ(QTextCodec::codecForName("windows-1251"), handler.Guess(ref));
|
2010-06-03 14:36:43 +02:00
|
|
|
|
}
|
|
|
|
|
|
2010-06-03 17:16:15 +02:00
|
|
|
|
TEST_F(SongTest, DecodesLatin1AsUtf8) {
|
|
|
|
|
const char utf8[] = { 0xe2, 0x80, 0x99, 0x00 };
|
|
|
|
|
TagLib::String str(utf8, TagLib::String::Latin1);
|
|
|
|
|
|
|
|
|
|
QTextCodec* codec = QTextCodec::codecForName("UTF-8");
|
|
|
|
|
|
|
|
|
|
QString fixed = Song::Decode(str, codec);
|
|
|
|
|
ASSERT_EQ(1, fixed.length());
|
|
|
|
|
EXPECT_EQ(QString::fromUtf8("’"), fixed);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, DecodesUtf8AsUtf8) {
|
|
|
|
|
const char utf8[] = { 0xe2, 0x80, 0x99, 0x00 };
|
|
|
|
|
TagLib::String str(utf8, TagLib::String::UTF8);
|
|
|
|
|
|
|
|
|
|
QTextCodec* codec = QTextCodec::codecForName("UTF-8");
|
|
|
|
|
|
|
|
|
|
QString fixed = Song::Decode(str, codec);
|
|
|
|
|
ASSERT_EQ(1, fixed.length());
|
|
|
|
|
EXPECT_EQ(QString::fromUtf8("’"), fixed);
|
|
|
|
|
}
|
|
|
|
|
|
2010-07-15 14:46:08 +02:00
|
|
|
|
TEST_F(SongTest, DecodesAmbiguousLatin1AndWindows1252) {
|
|
|
|
|
const char latin1[] = { 0x53, 0x75, 0x64, 0xe1, 0x66, 0x72, 0x69, 0x63, 0x61, 0x00 };
|
|
|
|
|
TagLib::String str(latin1, TagLib::String::Latin1);
|
|
|
|
|
QString fixed = UniversalEncodingHandler(NS_FILTER_NON_CJK).FixEncoding(str);
|
|
|
|
|
EXPECT_EQ(9, fixed.length());
|
|
|
|
|
EXPECT_EQ(QString::fromUtf8("Sudáfrica"), fixed);
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-17 18:03:49 +02:00
|
|
|
|
TEST_F(SongTest, FMPSRating) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsrating.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_FLOAT_EQ(0.42, song.rating());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FMPSRatingUser) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsratinguser.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_FLOAT_EQ(0.10, song.rating());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FMPSRatingBoth) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsratingboth.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_FLOAT_EQ(0.42, song.rating());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FMPSPlayCount) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsplaycount.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_EQ(123, song.playcount());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FMPSPlayCountUser) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsplaycountuser.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_EQ(42, song.playcount());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(SongTest, FMPSPlayCountBoth) {
|
|
|
|
|
TemporaryResource r(":/testdata/fmpsplaycountboth.mp3");
|
|
|
|
|
Song song;
|
|
|
|
|
song.InitFromFile(r.fileName(), -1);
|
|
|
|
|
EXPECT_EQ(123, song.playcount());
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-01 02:47:50 +01:00
|
|
|
|
} // namespace
|