/*
Movie Database - OMDb/IMDb
(C) 2009-2016 NG-Team
(C) 2016 NI-Team
License: GPL
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, see .
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "mdb-imdb.h"
CIMDB *CIMDB::getInstance()
{
static CIMDB *imdb = NULL;
if (!imdb)
imdb = new CIMDB();
return imdb;
}
CIMDB::CIMDB()
{
key = g_settings.omdb_api_key;
hintbox = NULL;
search_url = "http://www.google.de/search?q=";
search_outfile = "/tmp/google.out";
search_error = "IMDb: Google download failed";
imdb_url = "http://www.omdbapi.com/?plot=full&r=json&apikey=" + key + "&i=";
imdb_outfile = "/tmp/imdb.json";
posterfile = "/tmp/imdb.jpg";
acc = 0;
}
CIMDB::~CIMDB()
{
cleanup();
}
void CIMDB::setTitle(std::string epgtitle)
{
hintbox = new CHintBox(LOCALE_MESSAGEBOX_INFO, LOCALE_IMDB_READ_DATA);
hintbox->paint();
getMovieDetails(epgtitle);
if (hintbox)
{
hintbox->hide();
delete hintbox;
hintbox = NULL;
}
}
std::string CIMDB::utf82url(std::string s)
{
std::stringstream ss;
for (size_t i = 0; i < s.length(); ++i)
{
if (unsigned(s[i]) <= ' ')
{
ss << '+';
}
else if (unsigned(s[i]) <= '\x27')
{
ss << "%" << std::hex << unsigned(s[i]);
}
else
{
ss << s[i];
}
}
return ss.str();
}
std::string CIMDB::parseString(std::string search1, std::string search2, std::string str)
{
std::string ret, search;
size_t pos_wildcard, pos_search1, pos_search2;
pos_wildcard = pos_search1 = pos_search2 = std::string::npos;
if ((pos_wildcard = search1.find('*')) != std::string::npos)
{
search = search1.substr(0, pos_wildcard);
//std::cout << "wildcard detected" << '\t' << "= " << search << "[*]" << search1.substr(pos_wildcard+1) << std::endl;
}
else
search = search1;
//std::cout << "search1" << "\t\t\t" << "= " << '"' << search << '"' << std::endl;
if ((pos_search1 = str.find(search)) != std::string::npos)
{
//std::cout << "search1 found" << "\t\t" << "= " << '"' << search << '"' << " at pos "<< (int)(pos_search1) << " => " << str << std::endl;
pos_search1 += search.length();
if (pos_wildcard != std::string::npos)
{
size_t pos_wildcard_ext;
std::string wildcard_ext = search1.substr(pos_wildcard + 1);
//std::cout << "wildcard_ext" << "\t\t" << "= " << '"' << wildcard_ext << '"' << std::endl;
if ((pos_wildcard_ext = str.find(wildcard_ext, pos_wildcard + 1)) != std::string::npos)
{
//std::cout << "wildcard_ext found" << "\t" << "= " << '"' << wildcard_ext << '"' << " at pos "<< (int)(pos_wildcard_ext) << " => " << str << std::endl;
pos_search1 = pos_wildcard_ext + wildcard_ext.length();
}
else
{
//std::cout << "wildcard_ext not found in line " << acc << " - exit" << std::endl;
return ("");
}
}
}
else
{
//std::cout << "search1 not found in line " << acc << " - exit" << std::endl;
return ("");
}
if (pos_search1 != std::string::npos)
{
//std::cout << "search2 " << "\t\t" << "= " << '"' << search2 << '"' << std::endl;
if (search2 == "\n")
{
ret = str.substr(pos_search1, str.length() - pos_search1);
return (ret);
}
if ((pos_search2 = str.find(search2, pos_search1)) != std::string::npos)
{
if (search2.empty())
pos_search2 = str.length();
//std::cout << "search2" << "\t\t\t" << "= " << '"' << search2 << '"' << " found at "<< (int)(pos_search2) << " => " << str << std::endl;
ret = str.substr(pos_search1, pos_search2 - pos_search1);
}
//else
//std::cout << "search2 not found in line " << acc << " - exit" << std::endl;
}
return (ret);
}
std::string CIMDB::parseFile(std::string search1, std::string search2, const char *file, std::string firstline, int line_offset)
{
acc = 0;
std::ifstream fh;
std::string str, ret;
size_t pos_firstline;
pos_firstline = std::string::npos;
if (firstline.empty())
pos_firstline = 0;
fh.open(file, std::ios::in);
if (fh.is_open())
{
int line = 0;
while (!fh.eof())
{
getline(fh, str);
acc++;
if (pos_firstline == std::string::npos)
{
if ((pos_firstline = str.find(firstline)) != std::string::npos)
{
//std::cout << "firstline found " << str << std::endl;
}
continue;
}
if (line_offset /*&& pos_firstline != std::string::npos*/)
{
if (line + 1 != line_offset)
{
line++;
continue;
}
}
ret = parseString(search1, search2, str);
if (!ret.empty())
break;
}
fh.close();
}
return (ret);
}
std::string CIMDB::googleIMDb(std::string s)
{
CHTTPTool httpTool;
std::string ret = search_error;
std::string search_string("title+");
char *search_char = (char *) s.c_str();
m.clear();
unlink(search_outfile.c_str());
unlink(imdb_outfile.c_str());
unlink(posterfile.c_str());
while (*search_char != 0)
{
if (*search_char == ' ')
{
search_string += '+';
}
else
{
search_string += *search_char;
}
search_char++;
}
std::string url = search_url + utf82url(search_string) + "%20www.imdb.com";
if (httpTool.downloadFile(url, search_outfile.c_str()))
{
ret = parseFile("https://www.imdb.com/title/", ">", search_outfile.c_str());
if (ret.empty())
ret = parseFile("http://www.imdb.de/title/", ">", search_outfile.c_str());
std::string delimiters = "/&;";
size_t next = ret.find_first_of(delimiters, 0);
ret = ret.substr(0, next);
}
return ret;
}
void CIMDB::initMap(std::map &my)
{
std::string errMsg = "";
Json::Value root;
std::ostringstream ss;
std::ifstream fh(imdb_outfile.c_str(), std::ifstream::in);
ss << fh.rdbuf();
std::string filedata = ss.str();
bool parsedSuccess = parseJsonFromString(filedata, &root, &errMsg);
if (!parsedSuccess)
{
std::cout << "Failed to parse JSON\n";
std::cout << errMsg << std::endl;
my["Response"] = "False"; // we fake a false response
return;
}
/*
we grab only what we need to avoid bad surprises
when api is changed
*/
my["Actors"] = root.get("Actors", "").asString();
my["Awards"] = root.get("Awards", "").asString();
my["BoxOffice"] = root.get("BoxOffice", "").asString();
my["Country"] = root.get("Country", "").asString();
my["Director"] = root.get("Director", "").asString();
my["Genre"] = root.get("Genre", "").asString();
my["imdbID"] = root.get("imdbID", "").asString();
my["imdbRating"] = root.get("imdbRating", "N/A").asString();
my["imdbVotes"] = root.get("imdbVotes", "").asString();
my["Metascore"] = root.get("Metascore", "N/A").asString();
my["Plot"] = root.get("Plot", "").asString();
my["Poster"] = root.get("Poster", "N/A").asString();
my["Production"] = root.get("Production", "").asString();
my["Released"] = root.get("Released", "").asString();
my["Response"] = root.get("Response", "False").asString();
my["Runtime"] = root.get("Runtime", "").asString();
my["Title"] = root.get("Title", "").asString();
my["Website"] = root.get("Website", "").asString();
my["Writer"] = root.get("Writer", "").asString();
my["Year"] = root.get("Year", "").asString();
// currently unused
//my["Rated"] = root.get("Rated", "").asString();
//my["Type"] = root.get("Type", "").asString();
}
int CIMDB::getMovieDetails(const std::string &epgTitle)
{
CHTTPTool httpTool;
int ret = 0;
std::string imdb_id = googleIMDb(epgTitle);
if (((imdb_id.find(search_error)) != std::string::npos))
return ret;
std::string url = imdb_url + imdb_id;
if (httpTool.downloadFile(url, imdb_outfile.c_str()))
{
initMap(m);
//std::cout << "m now contains " << m.size() << " elements.\n";
if (m.empty() || m["Response"] != "True")
return 0;
//for (std::map::iterator it=m.begin(); it!=m.end(); ++it)
// std::cout << it->first << " => " << it->second << '\n';
//download Poster
if (m["Poster"] != "N/A")
{
// if possible load bigger image
std::string origURL("300");
std::string replURL("600");
if (m["Poster"].compare(m["Poster"].size() - 7, 3, origURL) == 0)
{
//std::cout << "########## " << m["Poster"] << " contains " << origURL << '\n';
m["Poster"].replace(m["Poster"].size() - 7, 3, replURL);
//std::cout << "########## New string: " << m["Poster"] << '\n';
}
if (httpTool.downloadFile(m["Poster"], posterfile.c_str()))
return 2;
else
{
if (access(posterfile.c_str(), F_OK) == 0)
unlink(posterfile.c_str());
return 1;
}
}
ret = 2;
}
return ret;
}
bool CIMDB::checkElement(std::string element)
{
if (m[element].empty() || m[element].compare("N/A") == 0)
return false;
else
return true;
}
std::string CIMDB::getEPGText()
{
if (m["imdbID"].empty() || m["Response"] != "True")
return g_Locale->getText(LOCALE_IMDB_DATA_FAILED);
std::string epgtext("");
if (checkElement("imdbRating"))
{
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_RATING) + ": " + m["imdbRating"] + "/10; ";
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_VOTES) + ": " + m["imdbVotes"] + "\n";
}
if (checkElement("Metascore"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_METASCORE) + ": " + m["Metascore"] + "/100\n";
if (checkElement("Title"))
{
epgtext += "\n";
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_TITLE) + ": " + m["Title"] + "\n";
}
if (checkElement("Released"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_RELEASED) + ": " + m["Country"] + ", " + m["Released"] + "\n";
if (checkElement("Runtime"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_RUNTIME) + ": " + m["Runtime"] + "\n";
if (checkElement("Genre"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_GENRE) + ": " + m["Genre"] + "\n";
if (checkElement("Awards"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_AWARDS) + ": " + m["Awards"] + "\n";
if (checkElement("Director"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_DIRECTOR) + ": " + m["Director"] + "\n";
if (checkElement("Writer"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_WRITER) + ": " + m["Writer"] + "\n";
if (checkElement("Production"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_PRODUCTION) + ": " + m["Production"] + "\n";
if (checkElement("Website"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_WEBSITE) + ": " + m["Website"] + "\n";
if (checkElement("BoxOffice"))
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_BOXOFFICE) + ": " + m["BoxOffice"] + "\n";
if (checkElement("Actors"))
{
epgtext += "\n";
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_ACTORS) + ": " + m["Actors"] + "\n";
}
if (checkElement("Plot"))
{
epgtext += "\n";
epgtext += g_Locale->getString(LOCALE_IMDB_DATA_PLOT) + ": " + m["Plot"];
}
return epgtext;
}
std::string CIMDB::getMovieText()
{
std::string movietext("");
if (checkElement("Plot"))
{
movietext = m["Plot"] + "\n";
if (checkElement("Title"))
{
movietext += "\n";
movietext += g_Locale->getString(LOCALE_IMDB_DATA_TITLE) + ": ";
movietext += m["Title"];
}
if (checkElement("Country"))
{
movietext += "\n";
movietext += g_Locale->getString(LOCALE_IMDB_DATA_RELEASED) + ": ";
movietext += m["Country"];
if (checkElement("Released"))
movietext += ", " + m["Released"];
}
if (checkElement("Actors"))
{
movietext += "\n";
movietext += g_Locale->getString(LOCALE_IMDB_DATA_ACTORS) + ": ";
movietext += m["Actors"];
}
}
return movietext;
}
int CIMDB::getStars()
{
std::string imdbRating = "";
if (checkElement("imdbRating"))
imdbRating = getIMDbElement("imdbRating");
if (imdbRating.empty() || imdbRating == "N/A")
return 0;
return (int)(atof(imdbRating.c_str()) * 10);
}
void CIMDB::cleanup()
{
if (access(search_outfile.c_str(), F_OK) == 0)
unlink(search_outfile.c_str());
if (access(posterfile.c_str(), F_OK) == 0)
unlink(posterfile.c_str());
}