351 lines
9.0 KiB
C++
351 lines
9.0 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2008-2009 by Andrzej Rybczak *
|
|
* electricityispower@gmail.com *
|
|
* *
|
|
* This program 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 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This program 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 this program; if not, write to the *
|
|
* Free Software Foundation, Inc., *
|
|
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
***************************************************************************/
|
|
|
|
#include <sys/stat.h>
|
|
#include <fstream>
|
|
|
|
#include "charset.h"
|
|
#include "helpers.h"
|
|
#include "lyrics.h"
|
|
#include "settings.h"
|
|
#include "song.h"
|
|
|
|
extern Window *wCurrent;
|
|
extern Scrollpad *sLyrics;
|
|
extern Scrollpad *sInfo;
|
|
|
|
const string artists_folder = home_folder + "/" + ".ncmpcpp/artists";
|
|
const string lyrics_folder = home_folder + "/" + ".lyrics";
|
|
|
|
#ifdef HAVE_CURL_CURL_H
|
|
extern pthread_t lyrics_downloader;
|
|
extern pthread_t artist_info_downloader;
|
|
extern bool lyrics_ready;
|
|
extern bool artist_info_ready;
|
|
pthread_mutex_t curl = PTHREAD_MUTEX_INITIALIZER;
|
|
#endif
|
|
|
|
namespace
|
|
{
|
|
size_t write_data(char *buffer, size_t size, size_t nmemb, string data)
|
|
{
|
|
size_t result = size * nmemb;
|
|
data.append(buffer, result);
|
|
return result;
|
|
}
|
|
|
|
void EscapeHtml(string &str)
|
|
{
|
|
for (size_t i = str.find("<"); i != string::npos; i = str.find("<"))
|
|
{
|
|
int j = str.find(">")+1;
|
|
str.replace(i, j-i, "");
|
|
}
|
|
for (size_t i = str.find("'"); i != string::npos; i = str.find("'"))
|
|
str.replace(i, 6, "'");
|
|
for (size_t i = str.find("""); i != string::npos; i = str.find("""))
|
|
str.replace(i, 6, "\"");
|
|
for (size_t i = str.find("&"); i != string::npos; i = str.find("&"))
|
|
str.replace(i, 5, "&");
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_CURL_CURL_H
|
|
void * GetArtistInfo(void *ptr)
|
|
{
|
|
string *strptr = static_cast<string *>(ptr);
|
|
string artist = *strptr;
|
|
delete strptr;
|
|
|
|
locale_to_utf(artist);
|
|
|
|
string filename = artist + ".txt";
|
|
ToLower(filename);
|
|
EscapeUnallowedChars(filename);
|
|
|
|
const string fullpath = artists_folder + "/" + filename;
|
|
mkdir(artists_folder.c_str(), 0755);
|
|
|
|
string result;
|
|
std::ifstream input(fullpath.c_str());
|
|
|
|
if (input.is_open())
|
|
{
|
|
bool first = 1;
|
|
string line;
|
|
while (getline(input, line))
|
|
{
|
|
if (!first)
|
|
*sInfo << "\n";
|
|
utf_to_locale(line);
|
|
*sInfo << line;
|
|
first = 0;
|
|
}
|
|
input.close();
|
|
sInfo->SetFormatting(fmtBold, "\n\nSimilar artists:\n", fmtBoldEnd, 0);
|
|
sInfo->SetFormatting(Config.color2, "\n * ", clEnd);
|
|
artist_info_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
for (string::iterator it = artist.begin(); it != artist.end(); it++)
|
|
if (*it == ' ')
|
|
*it = '+';
|
|
|
|
CURLcode code;
|
|
|
|
string url = "http://ws.audioscrobbler.com/2.0/?method=artist.getinfo&artist=" + artist + "&api_key=d94e5b6e26469a2d1ffae8ef20131b79";
|
|
|
|
pthread_mutex_lock(&curl);
|
|
CURL *info = curl_easy_init();
|
|
curl_easy_setopt(info, CURLOPT_URL, url.c_str());
|
|
curl_easy_setopt(info, CURLOPT_WRITEFUNCTION, write_data);
|
|
curl_easy_setopt(info, CURLOPT_WRITEDATA, &result);
|
|
curl_easy_setopt(info, CURLOPT_CONNECTTIMEOUT, 10);
|
|
curl_easy_setopt(info, CURLOPT_NOSIGNAL, 1);
|
|
code = curl_easy_perform(info);
|
|
curl_easy_cleanup(info);
|
|
pthread_mutex_unlock(&curl);
|
|
|
|
if (code != CURLE_OK)
|
|
{
|
|
*sInfo << "Error while fetching artist's info: " << curl_easy_strerror(code);
|
|
artist_info_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
size_t a, b;
|
|
bool erase = 0;
|
|
bool save = 1;
|
|
|
|
a = result.find("status=\"failed\"");
|
|
|
|
if (a != string::npos)
|
|
{
|
|
EscapeHtml(result);
|
|
*sInfo << "Last.fm returned an error message: " << result;
|
|
artist_info_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
vector<string> similar;
|
|
for (size_t i = result.find("<name>"); i != string::npos; i = result.find("<name>"))
|
|
{
|
|
result[i] = '.';
|
|
size_t j = result.find("</name>");
|
|
result[j] = '.';
|
|
i += 6;
|
|
similar.push_back(result.substr(i, j-i));
|
|
EscapeHtml(similar.back());
|
|
}
|
|
vector<string> urls;
|
|
for (size_t i = result.find("<url>"); i != string::npos; i = result.find("<url>"))
|
|
{
|
|
result[i] = '.';
|
|
size_t j = result.find("</url>");
|
|
result[j] = '.';
|
|
i += 5;
|
|
urls.push_back(result.substr(i, j-i));
|
|
}
|
|
|
|
a = result.find("<content>")+9;
|
|
b = result.find("</content>");
|
|
|
|
if (a == b)
|
|
{
|
|
result = "No description available for this artist.";
|
|
save = 0;
|
|
}
|
|
else
|
|
{
|
|
a += 9; // for <![CDATA[
|
|
b -= 3; // for ]]>
|
|
result = result.substr(a, b-a);
|
|
}
|
|
|
|
EscapeHtml(result);
|
|
for (size_t i = 0; i < result.length(); i++)
|
|
{
|
|
if (erase)
|
|
{
|
|
result.erase(result.begin()+i);
|
|
erase = 0;
|
|
}
|
|
if (result[i] == 13)
|
|
{
|
|
result[i] = '\n';
|
|
erase = 1;
|
|
}
|
|
else if (result[i] == '\t')
|
|
result[i] = ' ';
|
|
}
|
|
|
|
int i = result.length();
|
|
if (!isgraph(result[i-1]))
|
|
{
|
|
while (!isgraph(result[--i])) { }
|
|
result = result.substr(0, i+1);
|
|
}
|
|
|
|
Buffer filebuffer;
|
|
if (save)
|
|
filebuffer << result;
|
|
utf_to_locale(result);
|
|
*sInfo << result;
|
|
|
|
if (save)
|
|
filebuffer << "\n\nSimilar artists:\n";
|
|
*sInfo << fmtBold << "\n\nSimilar artists:\n" << fmtBoldEnd;
|
|
for (size_t i = 1; i < similar.size(); i++)
|
|
{
|
|
if (save)
|
|
filebuffer << "\n * " << similar[i] << " (" << urls[i] << ")";
|
|
utf_to_locale(similar[i]);
|
|
utf_to_locale(urls[i]);
|
|
*sInfo << "\n" << Config.color2 << " * " << clEnd << similar[i] << " (" << urls[i] << ")";
|
|
}
|
|
|
|
if (save)
|
|
filebuffer << "\n\n" << urls.front();
|
|
utf_to_locale(urls.front());
|
|
*sInfo << "\n\n" << urls.front();
|
|
|
|
if (save)
|
|
{
|
|
std::ofstream output(fullpath.c_str());
|
|
if (output.is_open())
|
|
{
|
|
output << filebuffer.Str();
|
|
output.close();
|
|
}
|
|
}
|
|
artist_info_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
#endif // HAVE_CURL_CURL_H
|
|
|
|
void *GetLyrics(void *song)
|
|
{
|
|
string artist = static_cast<Song *>(song)->GetArtist();
|
|
string title = static_cast<Song *>(song)->GetTitle();
|
|
|
|
locale_to_utf(artist);
|
|
locale_to_utf(title);
|
|
|
|
string filename = artist + " - " + title + ".txt";
|
|
const string fullpath = lyrics_folder + "/" + filename;
|
|
mkdir(lyrics_folder.c_str(), 0755);
|
|
|
|
std::ifstream input(fullpath.c_str());
|
|
|
|
if (input.is_open())
|
|
{
|
|
bool first = 1;
|
|
string line;
|
|
while (getline(input, line))
|
|
{
|
|
if (!first)
|
|
*sLyrics << "\n";
|
|
utf_to_locale(line);
|
|
*sLyrics << line;
|
|
first = 0;
|
|
}
|
|
# ifdef HAVE_CURL_CURL_H
|
|
lyrics_ready = 1;
|
|
pthread_exit(NULL);
|
|
# endif
|
|
}
|
|
# ifdef HAVE_CURL_CURL_H
|
|
for (string::iterator it = artist.begin(); it != artist.end(); it++)
|
|
if (*it == ' ')
|
|
*it = '+';
|
|
|
|
for (string::iterator it = title.begin(); it != title.end(); it++)
|
|
if (*it == ' ')
|
|
*it = '+';
|
|
|
|
CURLcode code;
|
|
|
|
string result;
|
|
|
|
string url = "http://lyricwiki.org/api.php?artist=";
|
|
url += artist;
|
|
url += "&song=";
|
|
url += title;
|
|
url += "&fmt=xml";
|
|
|
|
pthread_mutex_lock(&curl);
|
|
CURL *lyrics = curl_easy_init();
|
|
curl_easy_setopt(lyrics, CURLOPT_URL, url.c_str());
|
|
curl_easy_setopt(lyrics, CURLOPT_WRITEFUNCTION, write_data);
|
|
curl_easy_setopt(lyrics, CURLOPT_WRITEDATA, &result);
|
|
curl_easy_setopt(lyrics, CURLOPT_CONNECTTIMEOUT, 10);
|
|
curl_easy_setopt(lyrics, CURLOPT_NOSIGNAL, 1);
|
|
code = curl_easy_perform(lyrics);
|
|
curl_easy_cleanup(lyrics);
|
|
pthread_mutex_unlock(&curl);
|
|
|
|
if (code != CURLE_OK)
|
|
{
|
|
*sLyrics << "Error while fetching lyrics: " << curl_easy_strerror(code);
|
|
lyrics_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
int a, b;
|
|
a = result.find("<lyrics>")+8;
|
|
b = result.find("</lyrics>");
|
|
|
|
result = result.substr(a, b-a);
|
|
|
|
if (result == "Not found")
|
|
{
|
|
*sLyrics << result;
|
|
lyrics_ready = 1;
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
for (size_t i = result.find("<"); i != string::npos; i = result.find("<"))
|
|
result.replace(i, 4, "<");
|
|
for (size_t i = result.find(">"); i != string::npos; i = result.find(">"))
|
|
result.replace(i, 4, ">");
|
|
|
|
EscapeHtml(result);
|
|
|
|
string localized_result = result;
|
|
utf_to_locale(localized_result);
|
|
*sLyrics << localized_result;
|
|
|
|
std::ofstream output(fullpath.c_str());
|
|
if (output.is_open())
|
|
{
|
|
output << result;
|
|
output.close();
|
|
}
|
|
lyrics_ready = 1;
|
|
pthread_exit(NULL);
|
|
# else
|
|
else
|
|
*sLyrics << "Local lyrics not found. As ncmpcpp has been compiled without curl support, you can put appropriate lyrics into ~/.lyrics directory (file syntax is \"ARTIST - TITLE.txt\") or recompile ncmpcpp with curl support.";
|
|
return NULL;
|
|
# endif
|
|
}
|
|
|