123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315 |
- // Copyright (c) rAthena Dev Teams - Licensed under GNU GPL
- // For more information, see LICENCE in the main folder
- #include "database.hpp"
- #include "showmsg.hpp"
- bool YamlDatabase::nodeExists( const YAML::Node& node, const std::string& name ){
- try{
- if( node[name] ){
- return true;
- }else{
- return false;
- }
- // TODO: catch( ... ) instead?
- }catch( const YAML::InvalidNode& ){
- return false;
- }catch( const YAML::BadSubscript& ){
- return false;
- }
- }
- bool YamlDatabase::nodesExist( const YAML::Node& node, std::initializer_list<const std::string> names ){
- bool missing = false;
- for( const std::string& name : names ){
- if( !this->nodeExists( node, name ) ){
- ShowError( "Missing mandatory node \"%s\".\n", name.c_str() );
- missing = true;
- }
- }
- if( missing ){
- this->invalidWarning( node, "At least one mandatory node did not exist.\n" );
- return false;
- }
- return true;
- }
- bool YamlDatabase::verifyCompatibility( const YAML::Node& rootNode ){
- if( !this->nodeExists( rootNode, "Header" ) ){
- ShowError( "No database \"Header\" was found.\n" );
- return false;
- }
- const YAML::Node& headerNode = rootNode["Header"];
- if( !this->nodeExists( headerNode, "Type" ) ){
- ShowError( "No database \"Type\" was found.\n" );
- return false;
- }
- const YAML::Node& typeNode = headerNode["Type"];
- const std::string& tmpType = typeNode.as<std::string>();
- if( tmpType != this->type ){
- ShowError( "Database type mismatch: %s != %s.\n", this->type.c_str(), tmpType.c_str() );
- return false;
- }
- uint16 tmpVersion;
- if( !this->asUInt16( headerNode, "Version", tmpVersion ) ){
- ShowError("Invalid header \"Version\" type for %s database.\n", this->type.c_str());
- return false;
- }
- if( tmpVersion != this->version ){
- if( tmpVersion > this->version ){
- ShowError( "Database version %hu is not supported. Maximum version is: %hu\n", tmpVersion, this->version );
- return false;
- }else if( tmpVersion >= this->minimumVersion ){
- ShowWarning( "Database version %hu is outdated and should be updated. Current version is: %hu\n", tmpVersion, this->minimumVersion );
- }else{
- ShowError( "Database version %hu is not supported anymore. Minimum version is: %hu\n", tmpVersion, this->minimumVersion );
- return false;
- }
- }
- return true;
- }
- bool YamlDatabase::load(){
- return this->load( this->getDefaultLocation() );
- }
- bool YamlDatabase::reload(){
- this->clear();
- return this->load();
- }
- bool YamlDatabase::load(const std::string& path) {
- YAML::Node rootNode;
- try {
- ShowStatus( "Loading '" CL_WHITE "%s" CL_RESET "'..." CL_CLL "\r", path.c_str() );
- rootNode = YAML::LoadFile(path);
- }
- catch(YAML::Exception &e) {
- ShowError("Failed to read %s database file from '" CL_WHITE "%s" CL_RESET "'.\n", this->type.c_str(), path.c_str());
- ShowError("%s (Line %d: Column %d)\n", e.msg.c_str(), e.mark.line, e.mark.column);
- return false;
- }
- // Required here already for header error reporting
- this->currentFile = path;
- if (!this->verifyCompatibility(rootNode)){
- ShowError("Failed to verify compatibility with %s database file from '" CL_WHITE "%s" CL_RESET "'.\n", this->type.c_str(), this->currentFile.c_str());
- return false;
- }
- const YAML::Node& header = rootNode["Header"];
- if( this->nodeExists( header, "Clear" ) ){
- bool clear;
- if( this->asBool( header, "Clear", clear ) && clear ){
- this->clear();
- }
- }
- this->parse( rootNode );
- this->parseImports( rootNode );
- this->loadingFinished();
- return true;
- }
- void YamlDatabase::loadingFinished(){
- // Does nothing by default, just for hooking
- }
- void YamlDatabase::parse( const YAML::Node& rootNode ){
- uint64 count = 0;
- if( this->nodeExists( rootNode, "Body" ) ){
- const YAML::Node& bodyNode = rootNode["Body"];
- size_t childNodesCount = bodyNode.size();
- size_t childNodesProgressed = 0;
- const char* fileName = this->currentFile.c_str();
- for( const YAML::Node &node : bodyNode ){
- count += this->parseBodyNode( node );
- ShowStatus( "Loading [%" PRIdPTR "/%" PRIdPTR "] entries from '" CL_WHITE "%s" CL_RESET "'" CL_CLL "\r", ++childNodesProgressed, childNodesCount, fileName );
- }
- ShowStatus( "Done reading '" CL_WHITE "%" PRIu64 CL_RESET "' entries in '" CL_WHITE "%s" CL_RESET "'" CL_CLL "\n", count, fileName );
- }
- }
- void YamlDatabase::parseImports( const YAML::Node& rootNode ){
- if( this->nodeExists( rootNode, "Footer" ) ){
- const YAML::Node& footerNode = rootNode["Footer"];
- if( this->nodeExists( footerNode, "Imports") ){
- for( const YAML::Node& node : footerNode["Imports"] ){
- std::string importFile;
- if( !this->asString( node, "Path", importFile ) ){
- continue;
- }
- if( this->nodeExists( node, "Mode" ) ){
- std::string mode;
- if( !this->asString( node, "Mode", mode ) ){
- continue;
- }
- #ifdef RENEWAL
- std::string compiledMode = "Renewal";
- #else
- std::string compiledMode = "Prerenewal";
- #endif
- if( compiledMode != mode ){
- // Skip this import
- continue;
- }
- }
- this->load( importFile );
- }
- }
- }
- }
- template <typename R> bool YamlDatabase::asType( const YAML::Node& node, const std::string& name, R& out ){
- if( this->nodeExists( node, name ) ){
- const YAML::Node& dataNode = node[name];
- try {
- R value = dataNode.as<R>();
- out = value;
- return true;
- }catch( const YAML::BadConversion& ){
- this->invalidWarning( dataNode, "Unable to parse \"%s\".\n", name.c_str() );
- return false;
- }
- }else{
- this->invalidWarning( node, "Missing node \"%s\".\n", name.c_str() );
- return false;
- }
- }
- bool YamlDatabase::asBool(const YAML::Node &node, const std::string &name, bool &out) {
- return asType<bool>(node, name, out);
- }
- bool YamlDatabase::asInt16( const YAML::Node& node, const std::string& name, int16& out ){
- return asType<int16>( node, name, out);
- }
- bool YamlDatabase::asUInt16(const YAML::Node& node, const std::string& name, uint16& out) {
- return asType<uint16>(node, name, out);
- }
- bool YamlDatabase::asInt32(const YAML::Node &node, const std::string &name, int32 &out) {
- return asType<int32>(node, name, out);
- }
- bool YamlDatabase::asUInt32(const YAML::Node &node, const std::string &name, uint32 &out) {
- return asType<uint32>(node, name, out);
- }
- bool YamlDatabase::asInt64(const YAML::Node &node, const std::string &name, int64 &out) {
- return asType<int64>(node, name, out);
- }
- bool YamlDatabase::asUInt64(const YAML::Node &node, const std::string &name, uint64 &out) {
- return asType<uint64>(node, name, out);
- }
- bool YamlDatabase::asFloat(const YAML::Node &node, const std::string &name, float &out) {
- return asType<float>(node, name, out);
- }
- bool YamlDatabase::asDouble(const YAML::Node &node, const std::string &name, double &out) {
- return asType<double>(node, name, out);
- }
- bool YamlDatabase::asString(const YAML::Node &node, const std::string &name, std::string &out) {
- return asType<std::string>(node, name, out);
- }
- bool YamlDatabase::asUInt16Rate( const YAML::Node& node, const std::string& name, uint16& out, uint16 maximum ){
- if( this->asUInt16( node, name, out ) ){
- if( out > maximum ){
- this->invalidWarning( node[name], "Node \"%s\" with value %" PRIu16 " exceeds maximum of %" PRIu16 ".\n", name.c_str(), out, maximum );
- return false;
- }else if( out == 0 ){
- this->invalidWarning( node[name], "Node \"%s\" needs to be at least 1.\n", name.c_str() );
- return false;
- }else{
- return true;
- }
- }else{
- return false;
- }
- }
- bool YamlDatabase::asUInt32Rate( const YAML::Node& node, const std::string& name, uint32& out, uint32 maximum ){
- if( this->asUInt32( node, name, out ) ){
- if( out > maximum ){
- this->invalidWarning( node[name], "Node \"%s\" with value %" PRIu32 " exceeds maximum of %" PRIu32 ".\n", name.c_str(), out, maximum );
- return false;
- }else if( out == 0 ){
- this->invalidWarning( node[name], "Node \"%s\" needs to be at least 1.\n", name.c_str() );
- return false;
- }else{
- return true;
- }
- }else{
- return false;
- }
- }
- void YamlDatabase::invalidWarning( const YAML::Node &node, const char* fmt, ... ){
- va_list ap;
- va_start(ap, fmt);
- // Remove any remaining garbage of a previous loading line
- ShowMessage( CL_CLL );
- // Print the actual error
- _vShowMessage( MSG_ERROR, fmt, ap );
- va_end(ap);
- ShowError( "Occurred in file '" CL_WHITE "%s" CL_RESET "' on line %d and column %d.\n", this->currentFile.c_str(), node.Mark().line + 1, node.Mark().column );
- #ifdef DEBUG
- YAML::Emitter out;
- out << node;
- ShowMessage( "%s\n", out.c_str() );
- #endif
- }
- std::string YamlDatabase::getCurrentFile(){
- return this->currentFile;
- }
|