mirror of
https://github.com/FunKey-Project/RetroFE.git
synced 2025-12-26 16:48:51 +01:00
456 lines
16 KiB
C++
456 lines
16 KiB
C++
/* This file is part of RetroFE.
|
|
*
|
|
* RetroFE 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.
|
|
*
|
|
* RetroFE 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 RetroFE. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "MetadataDatabase.h"
|
|
#include "../Collection/CollectionInfo.h"
|
|
#include "../Collection/Item.h"
|
|
#include "../Utility/Log.h"
|
|
#include "../Utility/Utils.h"
|
|
#include "Configuration.h"
|
|
#include "DB.h"
|
|
#include <algorithm>
|
|
#include <dirent.h>
|
|
#include <fstream>
|
|
#include <list>
|
|
#include <rapidxml.hpp>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <map>
|
|
#include <sys/types.h>
|
|
#include <sqlite3.h>
|
|
#include <zlib.h>
|
|
#include <exception>
|
|
|
|
MetadataDatabase::MetadataDatabase(DB &db, Configuration &c)
|
|
: Config(c)
|
|
, DBInstance(db)
|
|
{
|
|
|
|
}
|
|
|
|
MetadataDatabase::~MetadataDatabase()
|
|
{
|
|
}
|
|
|
|
bool MetadataDatabase::ResetDatabase()
|
|
{
|
|
int rc;
|
|
char *error = NULL;
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
|
|
Logger::Write(Logger::ZONE_INFO, "Metadata", "Erasing");
|
|
|
|
std::string sql;
|
|
sql.append("DROP TABLE IF EXISTS Meta;");
|
|
|
|
rc = sqlite3_exec(handle, sql.c_str(), NULL, 0, &error);
|
|
|
|
if(rc != SQLITE_OK)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Unable to create Metadata table. Error: " << error;
|
|
Logger::Write(Logger::ZONE_ERROR, "Metadata", ss.str());
|
|
return false;
|
|
}
|
|
|
|
return Initialize();
|
|
}
|
|
|
|
bool MetadataDatabase::Initialize()
|
|
{
|
|
int rc;
|
|
char *error = NULL;
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
|
|
std::string sql;
|
|
sql.append("CREATE TABLE IF NOT EXISTS Meta(");
|
|
sql.append("collectionName TEXT KEY,");
|
|
sql.append("name TEXT NOT NULL DEFAULT '',");
|
|
sql.append("title TEXT NOT NULL DEFAULT '',");
|
|
sql.append("year TEXT NOT NULL DEFAULT '',");
|
|
sql.append("manufacturer TEXT NOT NULL DEFAULT '',");
|
|
sql.append("cloneOf TEXT NOT NULL DEFAULT '',");
|
|
sql.append("players TEXT NOT NULL DEFAULT '',");
|
|
sql.append("buttons TEXT NOT NULL DEFAULT '');");
|
|
sql.append("CREATE UNIQUE INDEX IF NOT EXISTS MetaUniqueId ON Meta(collectionName, name);");
|
|
|
|
rc = sqlite3_exec(handle, sql.c_str(), NULL, 0, &error);
|
|
|
|
if(rc != SQLITE_OK)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Unable to create Metadata table. Error: " << error;
|
|
Logger::Write(Logger::ZONE_ERROR, "Metadata", ss.str());
|
|
|
|
return false;
|
|
}
|
|
|
|
if(NeedsRefresh())
|
|
{
|
|
ImportDirectory();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool MetadataDatabase::ImportDirectory()
|
|
{
|
|
DIR *dp;
|
|
struct dirent *dirp;
|
|
std::string hyperListPath = "Meta/Hyperlist";
|
|
std::string mameListPath = "Meta/Mamelist";
|
|
|
|
dp = opendir(hyperListPath.c_str());
|
|
|
|
while((dirp = readdir(dp)) != NULL)
|
|
{
|
|
if (dirp->d_type != DT_DIR && std::string(dirp->d_name) != "." && std::string(dirp->d_name) != "..")
|
|
{
|
|
|
|
std::string basename = dirp->d_name;
|
|
|
|
std::string extension = basename.substr(basename.find_last_of("."), basename.size()-1);
|
|
basename = basename.substr(0, basename.find_last_of("."));
|
|
std::string collectionName = basename.substr(0, basename.find_first_of("."));
|
|
|
|
|
|
if(extension == ".xml")
|
|
{
|
|
std::string importFile = Configuration::GetAbsolutePath() + "/" + hyperListPath + "/" + dirp->d_name;
|
|
Logger::Write(Logger::ZONE_INFO, "Metadata", "Importing hyperlist: " + importFile);
|
|
Config.SetStatus("Scraping data from " + importFile);
|
|
ImportHyperList(importFile, collectionName);
|
|
}
|
|
}
|
|
}
|
|
|
|
dp = opendir(mameListPath.c_str());
|
|
|
|
while((dirp = readdir(dp)) != NULL)
|
|
{
|
|
if (dirp->d_type != DT_DIR && std::string(dirp->d_name) != "." && std::string(dirp->d_name) != "..")
|
|
{
|
|
|
|
std::string basename = dirp->d_name;
|
|
|
|
std::string extension = basename.substr(basename.find_last_of("."), basename.size()-1);
|
|
basename = basename.substr(0, basename.find_last_of("."));
|
|
std::string collectionName = basename.substr(0, basename.find_first_of("."));
|
|
|
|
|
|
if(extension == ".xml")
|
|
{
|
|
std::string importFile = Configuration::GetAbsolutePath() + "/" + mameListPath + "/" + dirp->d_name;
|
|
Logger::Write(Logger::ZONE_INFO, "Metadata", "Importing mamelist: " + importFile);
|
|
Config.SetStatus("Scraping data from " + importFile);
|
|
ImportMameList(importFile, collectionName);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
return true;
|
|
}
|
|
|
|
void MetadataDatabase::InjectMetadata(CollectionInfo *collection)
|
|
{
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
int rc;
|
|
sqlite3_stmt *stmt;
|
|
|
|
bool showParenthesis = true;
|
|
bool showSquareBrackets = true;
|
|
|
|
(void)Config.GetProperty("showParenthesis", showParenthesis);
|
|
(void)Config.GetProperty("showSquareBrackets", showSquareBrackets);
|
|
|
|
|
|
// items into a hash to make it easily searchable
|
|
std::vector<Item *> *items = collection->GetItems();
|
|
std::map<std::string, Item *> itemMap;
|
|
|
|
for(std::vector<Item *>::iterator it = items->begin(); it != items->end(); it++)
|
|
{
|
|
itemMap[(*it)->GetName()] = *it;
|
|
}
|
|
|
|
//todo: program crashes if this query fails
|
|
sqlite3_prepare_v2(handle,
|
|
"SELECT DISTINCT Meta.name, Meta.title, Meta.year, Meta.manufacturer, Meta.players, Meta.buttons, Meta.cloneOf "
|
|
"FROM Meta WHERE collectionName=? ORDER BY title ASC;",
|
|
-1, &stmt, 0);
|
|
|
|
sqlite3_bind_text(stmt, 1, collection->GetMetadataType().c_str(), -1, SQLITE_TRANSIENT);
|
|
|
|
rc = sqlite3_step(stmt);
|
|
|
|
while(rc == SQLITE_ROW)
|
|
{
|
|
std::string name = (char *)sqlite3_column_text(stmt, 0);
|
|
std::string fullTitle = (char *)sqlite3_column_text(stmt, 1);
|
|
std::string year = (char *)sqlite3_column_text(stmt, 2);
|
|
std::string manufacturer = (char *)sqlite3_column_text(stmt, 3);
|
|
std::string numberPlayers = (char *)sqlite3_column_text(stmt, 4);
|
|
std::string numberButtons = (char *)sqlite3_column_text(stmt, 5);
|
|
std::string cloneOf = (char *)sqlite3_column_text(stmt, 6);
|
|
std::string launcher;
|
|
std::string title = fullTitle;
|
|
|
|
//todo: this should be a helper method, peformed both in CollectionInfoBuilder
|
|
if(!showParenthesis)
|
|
{
|
|
std::string::size_type firstPos = title.find_first_of("(");
|
|
std::string::size_type secondPos = title.find_first_of(")", firstPos);
|
|
|
|
while(firstPos != std::string::npos && secondPos != std::string::npos)
|
|
{
|
|
firstPos = title.find_first_of("(");
|
|
secondPos = title.find_first_of(")", firstPos);
|
|
|
|
if (firstPos != std::string::npos)
|
|
{
|
|
title.erase(firstPos, (secondPos - firstPos) + 1);
|
|
}
|
|
}
|
|
}
|
|
if(!showSquareBrackets)
|
|
{
|
|
std::string::size_type firstPos = title.find_first_of("[");
|
|
std::string::size_type secondPos = title.find_first_of("]", firstPos);
|
|
|
|
while(firstPos != std::string::npos && secondPos != std::string::npos)
|
|
{
|
|
firstPos = title.find_first_of("[");
|
|
secondPos = title.find_first_of("]", firstPos);
|
|
|
|
if (firstPos != std::string::npos && secondPos != std::string::npos)
|
|
{
|
|
title.erase(firstPos, (secondPos - firstPos) + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
std::map<std::string, Item *>::iterator it = itemMap.find(name);
|
|
|
|
if(it != itemMap.end())
|
|
{
|
|
Item *item = it->second;
|
|
item->SetTitle(title);
|
|
item->SetFullTitle(fullTitle);
|
|
item->SetYear(year);
|
|
item->SetManufacturer(manufacturer);
|
|
item->SetNumberPlayers(numberPlayers);
|
|
item->SetNumberButtons(numberButtons);
|
|
item->SetCloneOf(cloneOf);
|
|
}
|
|
rc = sqlite3_step(stmt);
|
|
}
|
|
}
|
|
|
|
bool MetadataDatabase::NeedsRefresh()
|
|
{
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
sqlite3_stmt *stmt;
|
|
|
|
sqlite3_prepare_v2(handle,
|
|
"SELECT COUNT(*) FROM Meta;",
|
|
-1, &stmt, 0);
|
|
|
|
int rc = sqlite3_step(stmt);
|
|
|
|
if(rc == SQLITE_ROW)
|
|
{
|
|
int count = sqlite3_column_int(stmt, 0);
|
|
|
|
return (count == 0) ? true : false;
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool MetadataDatabase::ImportHyperList(std::string hyperlistFile, std::string collectionName)
|
|
{
|
|
bool retVal = false;
|
|
char *error = NULL;
|
|
rapidxml::xml_document<> doc;
|
|
std::ifstream file(hyperlistFile.c_str());
|
|
std::vector<char> buffer((std::istreambuf_iterator<char>(file)), std::istreambuf_iterator<char>());
|
|
|
|
try
|
|
{
|
|
buffer.push_back('\0');
|
|
|
|
doc.parse<0>(&buffer[0]);
|
|
|
|
rapidxml::xml_node<> *root = doc.first_node("menu");
|
|
|
|
if(!root)
|
|
{
|
|
Logger::Write(Logger::ZONE_ERROR, "Metadata", "Does not appear to be a HyperList file (missing <menu> tag)");
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
sqlite3_exec(handle, "BEGIN IMMEDIATE TRANSACTION;", NULL, NULL, &error);
|
|
for(rapidxml::xml_node<> *game = root->first_node("game"); game; game = game->next_sibling("game"))
|
|
{
|
|
rapidxml::xml_attribute<> *nameXml = game->first_attribute("name");
|
|
rapidxml::xml_node<> *descriptionXml = game->first_node("description");
|
|
rapidxml::xml_node<> *cloneofXml = game->first_node("cloneof");
|
|
rapidxml::xml_node<> *crcXml = game->first_node("crc");
|
|
rapidxml::xml_node<> *manufacturerXml = game->first_node("manufacturer");
|
|
rapidxml::xml_node<> *yearXml = game->first_node("year");
|
|
rapidxml::xml_node<> *genreXml = game->first_node("genre");
|
|
rapidxml::xml_node<> *ratingXml = game->first_node("rating");
|
|
rapidxml::xml_node<> *enabledXml = game->first_node("enabled");
|
|
std::string name = (nameXml) ? nameXml->value() : "";
|
|
std::string description = (descriptionXml) ? descriptionXml->value() : "";
|
|
std::string crc = (crcXml) ? crcXml->value() : "";
|
|
std::string cloneOf = (cloneofXml) ? cloneofXml->value() : "";
|
|
std::string manufacturer = (manufacturerXml) ? manufacturerXml->value() : "";
|
|
std::string year = (yearXml) ? yearXml->value() : "";
|
|
std::string genre = (genreXml) ? genreXml->value() : "";
|
|
std::string rating = (ratingXml) ? ratingXml->value() : "";
|
|
std::string enabled = (enabledXml) ? enabledXml->value() : "";
|
|
|
|
if(name.length() > 0)
|
|
{
|
|
sqlite3_stmt *stmt;
|
|
|
|
sqlite3_prepare_v2(handle,
|
|
"INSERT OR REPLACE INTO Meta (name, title, year, manufacturer, cloneOf, collectionName) VALUES (?,?,?,?,?,?)",
|
|
-1, &stmt, 0);
|
|
|
|
sqlite3_bind_text(stmt, 1, name.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 2, description.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 3, year.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 4, manufacturer.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 5, cloneOf.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 6, collectionName.c_str(), -1, SQLITE_TRANSIENT);
|
|
|
|
sqlite3_step(stmt);
|
|
sqlite3_finalize(stmt);
|
|
}
|
|
}
|
|
sqlite3_exec(handle, "COMMIT TRANSACTION;", NULL, NULL, &error);
|
|
}
|
|
}
|
|
catch(rapidxml::parse_error &e)
|
|
{
|
|
std::string what = e.what();
|
|
long line = static_cast<long>(std::count(&buffer.front(), e.where<char>(), char('\n')) + 1);
|
|
std::stringstream ss;
|
|
ss << "Could not parse layout file. [Line: " << line << "] Reason: " << e.what();
|
|
|
|
Logger::Write(Logger::ZONE_ERROR, "Metadata", ss.str());
|
|
}
|
|
catch(std::exception &e)
|
|
{
|
|
std::string what = e.what();
|
|
Logger::Write(Logger::ZONE_ERROR, "Metadata", "Could not parse hyperlist file. Reason: " + what);
|
|
}
|
|
|
|
|
|
return retVal;
|
|
}
|
|
|
|
bool MetadataDatabase::ImportMameList(std::string filename, std::string collectionName)
|
|
{
|
|
bool retVal = true;
|
|
rapidxml::xml_document<> doc;
|
|
rapidxml::xml_node<> * rootNode;
|
|
char *error = NULL;
|
|
sqlite3 *handle = DBInstance.GetHandle();
|
|
|
|
Logger::Write(Logger::ZONE_INFO, "Mamelist", "Importing mamelist file \"" + filename + "\" (this will take a while)");
|
|
std::ifstream file(filename.c_str());
|
|
|
|
std::vector<char> buffer((std::istreambuf_iterator<char>(file)), std::istreambuf_iterator<char>());
|
|
|
|
buffer.push_back('\0');
|
|
|
|
doc.parse<0>(&buffer[0]);
|
|
|
|
rootNode = doc.first_node("mame");
|
|
|
|
sqlite3_exec(handle, "BEGIN IMMEDIATE TRANSACTION;", NULL, NULL, &error);
|
|
for (rapidxml::xml_node<> * game = rootNode->first_node("game"); game; game = game->next_sibling())
|
|
{
|
|
rapidxml::xml_attribute<> *nameNode = game->first_attribute("name");
|
|
rapidxml::xml_attribute<> *cloneOfXml = game->first_attribute("cloneof");
|
|
|
|
if(nameNode != NULL)
|
|
{
|
|
std::string name = nameNode->value();
|
|
rapidxml::xml_node<> *descriptionNode = game->first_node("description");
|
|
rapidxml::xml_node<> *yearNode = game->first_node("year");
|
|
rapidxml::xml_node<> *manufacturerNode = game->first_node("manufacturer");
|
|
rapidxml::xml_node<> *inputNode = game->first_node("input");
|
|
|
|
std::string description = (descriptionNode == NULL) ? nameNode->value() : descriptionNode->value();
|
|
std::string year = (yearNode == NULL) ? "" : yearNode->value();
|
|
std::string manufacturer = (manufacturerNode == NULL) ? "" : manufacturerNode->value();
|
|
std::string cloneOf = (cloneOfXml == NULL) ? "" : cloneOfXml->value();
|
|
std::string players;
|
|
std::string buttons;
|
|
|
|
if(inputNode != NULL)
|
|
{
|
|
rapidxml::xml_attribute<> *playersAttribute = inputNode->first_attribute("players");
|
|
rapidxml::xml_attribute<> *buttonsAttribute = inputNode->first_attribute("buttons");
|
|
|
|
if(playersAttribute)
|
|
{
|
|
players = playersAttribute->value();
|
|
}
|
|
|
|
if(buttonsAttribute)
|
|
{
|
|
buttons = buttonsAttribute->value();
|
|
}
|
|
|
|
}
|
|
|
|
sqlite3_stmt *stmt;
|
|
|
|
sqlite3_prepare_v2(handle,
|
|
"INSERT OR REPLACE INTO Meta (name, title, year, manufacturer, players, buttons, cloneOf, collectionName) VALUES (?,?,?,?,?,?,?,?)",
|
|
-1, &stmt, 0);
|
|
|
|
|
|
sqlite3_bind_text(stmt, 1, name.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 2, description.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 3, year.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 4, manufacturer.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 5, players.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 6, buttons.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 7, cloneOf.c_str(), -1, SQLITE_TRANSIENT);
|
|
sqlite3_bind_text(stmt, 8, collectionName.c_str(), -1, SQLITE_TRANSIENT);
|
|
|
|
sqlite3_step(stmt);
|
|
sqlite3_finalize(stmt);
|
|
}
|
|
}
|
|
|
|
sqlite3_exec(handle, "COMMIT TRANSACTION;", NULL, NULL, &error);
|
|
|
|
return retVal;
|
|
}
|
|
|
|
|