diff --git a/.gitignore b/.gitignore index 4b0c703..8c9b4dd 100644 --- a/.gitignore +++ b/.gitignore @@ -70,6 +70,6 @@ Thumbs.db *.dll *.exe -bin/ build/ - +bin/akashi +bin/config/ diff --git a/README.md b/README.md index 0414d3d..92a2d24 100644 --- a/README.md +++ b/README.md @@ -3,6 +3,4 @@ A C++ server for Attorney Online 2 Copyright © scatterflower 2020 -UI Design and Code - Copyright © windrammer 2020 - If you want to support the development of akashi, feel free to give a one-time gift at [liberapay](https://liberapay.com/scatterflower/) diff --git a/akashi.pro b/akashi.pro index e9b5a5d..e987da8 100644 --- a/akashi.pro +++ b/akashi.pro @@ -1,6 +1,6 @@ -QT += core gui network websockets - -greaterThan(QT_MAJOR_VERSION, 4): QT += widgets +QT += network websockets core +QT -= gui +TEMPLATE = app CONFIG += c++11 @@ -24,16 +24,3 @@ RC_ICONS = resource/icon/akashi.ico SOURCES += $$files($$PWD/src/*.cpp) HEADERS += $$files($$PWD/include/*.h) - -FORMS += $$files($$PWD/resource/forms/*.ui) - -TRANSLATIONS += \ - resource/translation/akashi_es.ts - -# Default rules for deployment. -qnx: target.path = /tmp/$${TARGET}/bin -else: unix:!android: target.path = /opt/$${TARGET}/bin -!isEmpty(target.path): INSTALLS += target - -RESOURCES += \ - resources.qrc diff --git a/bin/config_sample/areas.ini b/bin/config_sample/areas.ini new file mode 100644 index 0000000..14f79ca --- /dev/null +++ b/bin/config_sample/areas.ini @@ -0,0 +1,5 @@ +[Basement] +background=gs4 + +[Courtroom 1] +background=gs4 diff --git a/bin/config_sample/characters.txt b/bin/config_sample/characters.txt new file mode 100644 index 0000000..a7648cc --- /dev/null +++ b/bin/config_sample/characters.txt @@ -0,0 +1,51 @@ +Adrian +Apollo +April +Armstrong +Atmey +Butz +Diego +Edgeworth +Edgeworthw +Ema +EmaSkye +Franny +Franziska +Gant +Gavin +Gavin K +Godot +Gregory +Grossberg +Gumshoe +Gumshoey +Hawk +Hobo_Phoenix +Ini +Judge +Judge's Bro +Klav +Klavier +Kristoph +Lana +Layton +Lotta +Luis +Maggey +Manfred +Marshall +Matt +Maya +Mia +Miles +Oldbag +Payne +Pearl +Phoenix +Valant +Vasquez +Wellington +Winston +WinstonPayne +Young Mia +Zak diff --git a/bin/config_sample/config.ini b/bin/config_sample/config.ini new file mode 100644 index 0000000..c99be8d --- /dev/null +++ b/bin/config_sample/config.ini @@ -0,0 +1,18 @@ +[Info] +version=1 + +[Options] +advertise=true +hostname=$H +language=en +max_message_size=256 +max_players=100 +modpass=password +ms_ip=master.aceattorneyonline.com +ms_port=27016 +multiclient_limit=16 +port=27016 +server_description=This is my flashy new server +server_name=My First Server +webao_enable=true +webao_port=27017 diff --git a/bin/config_sample/music.txt b/bin/config_sample/music.txt new file mode 100644 index 0000000..585b439 --- /dev/null +++ b/bin/config_sample/music.txt @@ -0,0 +1,49 @@ +Announce The Truth (AA).opus +Announce The Truth (AJ).opus +Announce The Truth (JFA).opus +Announce The Truth (Miles).opus +Announce The Truth (T&T).opus +Confrontation ~ Presto 2009.opus +Crises of Fate.opus +Forgotten Legend.opus +Godot - The Fragrance of Dark Coffee.opus +Great Revival ~ Franziska von Karma.opus +Great Revival ~ Miles Edgeworth.opus +Hotline of Fate.opus +Interesting People.opus +Logic and Trick.opus +Luke Atmey ~ I Just Want Love.opus +Noisy People.opus +OBJECTION (AA).opus +Objection (AJ).opus +OBJECTION (JFA).opus +Objection (Miles).opus +OBJECTION (T&T).opus +Others ~ Guilty love.opus +Prelude (AA).opus +Prelude (AJ).opus +Prologue (AA).opus +Pursuit (AA) - variation.opus +Pursuit (AA).opus +Pursuit (AJ).opus +Pursuit (DS).opus +Pursuit (JFA) - variation.opus +Pursuit (JFA).opus +Pursuit (Miles).opus +Pursuit (T&T) - variation.opus +Pursuit (T&T).opus +Pursuit ~ I Want to Find the Truth (Orchestra).opus +Questioning AA (Allegro).opus +Questioning AA (Moderato).opus +Questioning AJ (Allegro).opus +Questioning AJ (Moderato).opus +Questioning JFA (Allegro).opus +Questioning JFA (Moderato).opus +Questioning T&T (Allegro).opus +Questioning T&T (Moderato).opus +Speak up Pup.opus +Suspense (AA).opus +The Great Truth Burglar.opus +Trial (AA).opus +Trial (AJ).opus +Trial (Miles).opus diff --git a/include/advertiser.h b/include/advertiser.h index 3d1f81f..b2c5752 100644 --- a/include/advertiser.h +++ b/include/advertiser.h @@ -20,7 +20,7 @@ #include "include/aopacket.h" -#include +#include #include #include #include diff --git a/include/akashimain.h b/include/akashimain.h deleted file mode 100644 index 8f18aec..0000000 --- a/include/akashimain.h +++ /dev/null @@ -1,52 +0,0 @@ -////////////////////////////////////////////////////////////////////////////////////// -// akashi - a server for Attorney Online 2 // -// Copyright (C) 2020 scatterflower // -// // -// This program is free software: you can redistribute it and/or modify // -// it under the terms of the GNU Affero General Public License as // -// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. // -// // -// You should have received a copy of the GNU Affero General Public License // -// along with this program. If not, see . // -////////////////////////////////////////////////////////////////////////////////////// -#ifndef AKASHIMAIN_H -#define AKASHIMAIN_H - -#include "include/advertiser.h" -#include "include/config_manager.h" -#include "include/server.h" - -#include -#include -#include - -QT_BEGIN_NAMESPACE -namespace Ui { -class AkashiMain; -} -QT_END_NAMESPACE - -class AkashiMain : public QMainWindow { - Q_OBJECT - - public: - AkashiMain(QWidget* parent = nullptr); - ~AkashiMain(); - - ConfigManager config_manager; - - void generateDefaultConfig(bool backup_old); - void updateConfig(int current_version); - - private: - Ui::AkashiMain* ui; - Advertiser* advertiser; - Server* server; -}; -#endif // AKASHIMAIN_H diff --git a/include/config_manager.h b/include/config_manager.h index 223f87f..cd5b8ae 100644 --- a/include/config_manager.h +++ b/include/config_manager.h @@ -30,8 +30,7 @@ class ConfigManager { public: ConfigManager(); bool initConfig(); - void generateDefaultConfig(bool backup_old); - void updateConfig(int current_version); + bool updateConfig(int current_version); struct server_settings { QString ms_ip; @@ -47,6 +46,7 @@ class ConfigManager { private: QSettings* config; + bool fileExists(QFileInfo *file); }; #endif // CONFIG_MANAGER_H diff --git a/include/server.h b/include/server.h index f1b8bbf..ff5be5a 100644 --- a/include/server.h +++ b/include/server.h @@ -23,7 +23,7 @@ #include "include/area_data.h" #include "include/ws_proxy.h" -#include +#include #include #include #include diff --git a/resource/forms/akashimain.ui b/resource/forms/akashimain.ui deleted file mode 100644 index 1f4f34d..0000000 --- a/resource/forms/akashimain.ui +++ /dev/null @@ -1,615 +0,0 @@ - - - AkashiMain - - - true - - - - 0 - 0 - 640 - 483 - - - - - Ubuntu - - - - AkashiMain - - - Qt::LeftToRight - - - /*background-color: rgb(240, 240, 240);*/ -background-color: rgb(53, 53, 53); - - - QTabWidget::Rounded - - - - - true - - - - - 0 - - - - - false - - - color: rgb(243, 243, 243); - - - 0 - - - - - 0 - 0 - - - - Qt::LeftToRight - - - - - - Server - - - - 0 - - - 9 - - - 9 - - - 9 - - - 9 - - - - - true - - - - 0 - 0 - - - - - 22 - - - - Qt::LeftToRight - - - false - - - /*background-color: rgb(45, 57, 67);*/ -background-color: rgba(191, 64, 64, 0); -color: rgb(240, 240, 240); -padding: 16px 3px 16px 3px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(11, 24, 35, 255), stop:1 rgba(61, 73, 82, 255));*/ - - - Server name - - - Qt::AlignCenter - - - - - - - Qt::Vertical - - - QSizePolicy::Fixed - - - - 2 - 2 - - - - - - - - - 0 - 0 - - - - - 18 - - - - color: rgb(240, 240, 240); -background-color: rgb(246, 36, 81); -padding: 12px 18px 12px 18px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(202, 0, 43, 255), stop:1 rgba(255, 43, 89, 255));*/ - - - Offline - - - Qt::AlignCenter - - - - - - - Qt::Vertical - - - QSizePolicy::Fixed - - - - 0 - 24 - - - - - - - - 0 - - - QLayout::SetDefaultConstraint - - - 0 - - - 0 - - - 0 - - - 0 - - - - - true - - - - 0 - 0 - - - - - 20 - - - - Qt::LeftToRight - - - false - - - QPushButton { - border: none; - border-radius: 4px; - background-color: rgb(31, 215, 141); - /*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(0, 193, 115, 255), stop:1 rgba(33, 251, 163, 255));*/ - color: rgb(240, 240, 240); - padding: 20px 30px 20px 30px; -} - -QPushButton:pressed { - background-color: rgb(72, 240, 164); -} - -QPushButton:flat { - border: none; /* no border for a flat push button */ -} - - - Start - - - false - - - - - - - - - Qt::Vertical - - - QSizePolicy::MinimumExpanding - - - - 20 - 40 - - - - - - - - 2 - - - - - Qt::Horizontal - - - - 0 - 0 - - - - - - - - - 0 - 0 - - - - - 11 - - - - background-color: rgb(45, 57, 67); -background-color: rgb(45, 57, 67, 0); -color: rgb(240, 240, 240); -/*padding: 16px 3px 16px 3px;*/ -padding: 3px 8px 3px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(11, 24, 35, 255), stop:1 rgba(61, 73, 82, 255));*/ - - - Master Server connection status: - - - - - - - - 0 - 0 - - - - - 12 - - - - color: rgb(240, 240, 240); -background-color: rgb(246, 36, 81); -border-radius: 4px; -padding: 3px 8px 3px 8px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(202, 0, 43, 255), stop:1 rgba(255, 43, 89, 255));*/ - - - Not connected - - - - - - - Qt::Horizontal - - - QSizePolicy::MinimumExpanding - - - - 0 - 0 - - - - - - - - - - Qt::Vertical - - - QSizePolicy::Fixed - - - - 0 - 24 - - - - - - - - 6 - - - QLayout::SetDefaultConstraint - - - 0 - - - 0 - - - - - 2 - - - QLayout::SetDefaultConstraint - - - - - - 0 - 0 - - - - - 12 - - - - Qt::LeftToRight - - - color: rgb(240, 240, 240); -background-color: rgb(43, 135, 211); -padding: 3px 8px 3px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(36, 128, 204, 255), stop:1 rgba(78, 175, 255, 255));*/ - - - QFrame::NoFrame - - - QFrame::Plain - - - Players - - - Qt::AlignCenter - - - false - - - - - - - true - - - - 0 - 0 - - - - - 14 - - - - color: rgb(240, 240, 240); -/*background-color: rgb(45, 57, 67);*/ -background-color: rgb(45, 57, 67, 0); -padding: 4px 8px 4px 8px; -border-top-right-radius: 2px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(11, 24, 35, 255), stop:1 rgba(61, 73, 82, 255));*/ - - - 0/100 - - - Qt::AlignCenter - - - - - - - - - 2 - - - - - - 0 - 0 - - - - - 12 - - - - color: rgb(240, 240, 240); -background-color: rgb(43, 135, 211); -padding: 3px 8px 3px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(36, 128, 204, 255), stop:1 rgba(78, 175, 255, 255));*/ - - - Software - - - Qt::AlignCenter - - - - - - - - 14 - - - - color: rgb(240, 240, 240); -/*background-color: rgb(45, 57, 67);*/ -background-color: rgb(45, 57, 67, 0); -padding: 4px 8px 4px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(11, 24, 35, 255), stop:1 rgba(61, 73, 82, 255));*/ - - - Akashi - - - Qt::AlignCenter - - - - - - - - - 2 - - - - - - 0 - 0 - - - - - 12 - - - - color: rgb(240, 240, 240); -background-color: rgb(43, 135, 211); -padding: 3px 8px 3px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(36, 128, 204, 255), stop:1 rgba(78, 175, 255, 255));*/ - - - Version - - - Qt::AlignCenter - - - - - - - - 14 - - - - color: rgb(240, 240, 240); -/*background-color: rgb(45, 57, 67);*/ -background-color: rgb(45, 57, 67, 0); -padding: 4px 8px 4px 8px; -border-radius: 4px; -/*background-color: qlineargradient(spread:pad, x1:0, y1:1, x2:0, y2:0, stop:0 rgba(11, 24, 35, 255), stop:1 rgba(61, 73, 82, 255));*/ - - - 0.0.1 - - - Qt::AlignCenter - - - - - - - - - - - - Options - - - - - - - - - - 0 - 0 - 640 - 22 - - - - - - - - diff --git a/resource/translation/akashi_es.qm b/resource/translation/akashi_es.qm deleted file mode 100644 index 2ccbd6b..0000000 Binary files a/resource/translation/akashi_es.qm and /dev/null differ diff --git a/resource/translation/akashi_es.ts b/resource/translation/akashi_es.ts deleted file mode 100644 index eaeeff6..0000000 --- a/resource/translation/akashi_es.ts +++ /dev/null @@ -1,25 +0,0 @@ - - - - - AkashiMain - - - AkashiMain - - - - - main - - - A server for Attorney Online 2 - Un servidor para Attorney Online 2 - - - - Run the server without a GUI - Empezar el servidor sin interfaz - - - diff --git a/resources.qrc b/resources.qrc deleted file mode 100644 index 4117cd0..0000000 --- a/resources.qrc +++ /dev/null @@ -1,5 +0,0 @@ - - - resource/translation/akashi_es.qm - - diff --git a/src/advertiser.cpp b/src/advertiser.cpp index 67e1f35..79af152 100644 --- a/src/advertiser.cpp +++ b/src/advertiser.cpp @@ -62,7 +62,7 @@ void Advertiser::socketConnected() AOPacket ao_packet("SCC", {concat_ports, name, description, - "akashi v" + QApplication::applicationVersion()}); + "akashi v" + QCoreApplication::applicationVersion()}); QByteArray data = ao_packet.toUtf8(); socket->write(data); diff --git a/src/akashimain.cpp b/src/akashimain.cpp deleted file mode 100644 index 155336d..0000000 --- a/src/akashimain.cpp +++ /dev/null @@ -1,60 +0,0 @@ -////////////////////////////////////////////////////////////////////////////////////// -// akashi - a server for Attorney Online 2 // -// Copyright (C) 2020 scatterflower // -// // -// This program is free software: you can redistribute it and/or modify // -// it under the terms of the GNU Affero General Public License as // -// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. // -// // -// You should have received a copy of the GNU Affero General Public License // -// along with this program. If not, see . // -////////////////////////////////////////////////////////////////////////////////////// -#include "include/akashimain.h" -#include "ui_akashimain.h" - -AkashiMain::AkashiMain(QWidget* parent) - : QMainWindow(parent), config_manager(), ui(new Ui::AkashiMain) -{ - ui->setupUi(this); - qDebug("Main application started"); - - if (config_manager.initConfig()) { - // Config is sound, so proceed with starting the server - // Validate some of the config before passing it on - ConfigManager::server_settings settings; - bool config_valid = config_manager.loadServerSettings(&settings); - - if (!config_valid) { - // TODO: send signal config invalid - config_manager.generateDefaultConfig(true); - } - else { - if (settings.advertise_server) { - // TODO: send signal advertiser started - advertiser = - new Advertiser(settings.ms_ip, settings.port, - settings.ws_port, settings.local_port, - settings.name, settings.description, this); - advertiser->contactMasterServer(); - } - - // TODO: start the server here - // TODO: send signal server starting. - server = new Server(settings.port, settings.ws_port); - server->start(); - } - } -} - -AkashiMain::~AkashiMain() -{ - delete ui; - delete advertiser; - delete server; -} diff --git a/src/aoclient.cpp b/src/aoclient.cpp index d2a58a8..08522a3 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -73,10 +73,10 @@ void AOClient::handlePacket(AOPacket packet) // Lord forgive me if (packet.header == "HI") { setHwid(packet.contents[0]); - sendPacket("ID", {"271828", "akashi", QApplication::applicationVersion()}); + sendPacket("ID", {"271828", "akashi", QCoreApplication::applicationVersion()}); } else if (packet.header == "ID") { - QSettings config("config.ini", QSettings::IniFormat); + QSettings config("config/config.ini", QSettings::IniFormat); config.beginGroup("Options"); QString max_players = config.value("max_players").toString(); config.endGroup(); @@ -114,10 +114,7 @@ void AOClient::handlePacket(AOPacket packet) fullArup(); // Give client all the area data arup(ARUPType::PLAYER_COUNT, true); // Tell everyone there is a new player - QSettings areas_ini("areas.ini", QSettings::IniFormat); - QStringList areas = areas_ini.childGroups(); - - sendPacket("FA", areas); + sendPacket("FA", server->area_names); sendPacket("OPPASS", {"DEADBEEF"}); sendPacket("DONE"); } diff --git a/src/config_manager.cpp b/src/config_manager.cpp index 1d2f06f..086ec90 100644 --- a/src/config_manager.cpp +++ b/src/config_manager.cpp @@ -19,166 +19,68 @@ ConfigManager::ConfigManager() { - config = new QSettings("config.ini", QSettings::IniFormat); + config = new QSettings("config/config.ini", QSettings::IniFormat); } // Validate and set up the config bool ConfigManager::initConfig() { - QFileInfo char_list_info("characters.txt"); - if (!(char_list_info.exists() && char_list_info.isFile())) { - // TODO: signals go here - QFile char_list("characters.txt"); - if (!char_list.open(QIODevice::WriteOnly | QIODevice::Text)) - qDebug() << "Couldn't create character list"; - QTextStream file_stream(&char_list); - - qDebug() << "Creating vanilla character list"; - - file_stream << "Adrian\n"; - file_stream << "Apollo\n"; - file_stream << "April\n"; - file_stream << "Armstrong\n"; - file_stream << "Atmey\n"; - file_stream << "Butz\n"; - file_stream << "Diego\n"; - file_stream << "Edgeworth\n"; - file_stream << "Edgeworthw\n"; - file_stream << "Ema\n"; - file_stream << "EmaSkye\n"; - file_stream << "Franny\n"; - file_stream << "Franziska\n"; - file_stream << "Gant\n"; - file_stream << "Gavin\n"; - file_stream << "Gavin K\n"; - file_stream << "Godot\n"; - file_stream << "Gregory\n"; - file_stream << "Grossberg\n"; - file_stream << "Gumshoe\n"; - file_stream << "Gumshoey\n"; - file_stream << "Hawk\n"; - file_stream << "Hobo_Phoenix\n"; - file_stream << "Ini\n"; - file_stream << "Judge\n"; - file_stream << "Judge's Bro\n"; - file_stream << "Klav\n"; - file_stream << "Klavier\n"; - file_stream << "Kristoph\n"; - file_stream << "Lana\n"; - file_stream << "Layton\n"; - file_stream << "Lotta\n"; - file_stream << "Luis\n"; - file_stream << "Maggey\n"; - file_stream << "Manfred\n"; - file_stream << "Marshall\n"; - file_stream << "Matt\n"; - file_stream << "Maya\n"; - file_stream << "Mia\n"; - file_stream << "Miles\n"; - file_stream << "Oldbag\n"; - file_stream << "Payne\n"; - file_stream << "Pearl\n"; - file_stream << "Phoenix\n"; - file_stream << "Valant\n"; - file_stream << "Vasquez\n"; - file_stream << "Wellington\n"; - file_stream << "Winston\n"; - file_stream << "WinstonPayne\n"; - file_stream << "Young Mia\n"; - file_stream << "Zak\n"; - - char_list.flush(); - char_list.close(); + QFileInfo config_dir_info("config/"); + if (!config_dir_info.exists() || !config_dir_info.isDir()) { + qCritical() << "Config directory doesn't exist!"; + return false; } - QFileInfo music_list_info("music.txt"); - if (!(music_list_info.exists() && music_list_info.isFile())) { - QFile music_list("music.txt"); - if (!music_list.open(QIODevice::WriteOnly | QIODevice::Text)) - qDebug() << "Couldn't create music list"; - QTextStream file_stream(&music_list); + // Check that areas, characters, and music lists all exist + QFileInfo areas_info("config/areas.ini"); + QFileInfo characters_info("config/characters.txt"); + QFileInfo music_info("config/music.txt"); - qDebug() << "Creating vanilla music list"; - - file_stream << "Announce The Truth (AA).opus\n"; - file_stream << "Announce The Truth (AJ).opus\n"; - file_stream << "Announce The Truth (JFA).opus\n"; - file_stream << "Announce The Truth (Miles).opus\n"; - file_stream << "Announce The Truth (T&T).opus\n"; - file_stream << "Confrontation ~ Presto 2009.opus\n"; - file_stream << "Crises of Fate.opus\n"; - file_stream << "Forgotten Legend.opus\n"; - file_stream << "Godot - The Fragrance of Dark Coffee.opus\n"; - file_stream << "Great Revival ~ Franziska von Karma.opus\n"; - file_stream << "Great Revival ~ Miles Edgeworth.opus\n"; - file_stream << "Hotline of Fate.opus\n"; - file_stream << "Interesting People.opus\n"; - file_stream << "Logic and Trick.opus\n"; - file_stream << "Luke Atmey ~ I Just Want Love.opus\n"; - file_stream << "Noisy People.opus\n"; - file_stream << "OBJECTION (AA).opus\n"; - file_stream << "Objection (AJ).opus\n"; - file_stream << "OBJECTION (JFA).opus\n"; - file_stream << "Objection (Miles).opus\n"; - file_stream << "OBJECTION (T&T).opus\n"; - file_stream << "Others ~ Guilty love.opus\n"; - file_stream << "Prelude (AA).opus\n"; - file_stream << "Prelude (AJ).opus\n"; - file_stream << "Prologue (AA).opus\n"; - file_stream << "Pursuit (AA) - variation.opus\n"; - file_stream << "Pursuit (AA).opus\n"; - file_stream << "Pursuit (AJ).opus\n"; - file_stream << "Pursuit (DS).opus\n"; - file_stream << "Pursuit (JFA) - variation.opus\n"; - file_stream << "Pursuit (JFA).opus\n"; - file_stream << "Pursuit (Miles).opus\n"; - file_stream << "Pursuit (T&T) - variation.opus\n"; - file_stream << "Pursuit (T&T).opus\n"; - file_stream << "Pursuit ~ I Want to Find the Truth (Orchestra).opus\n"; - file_stream << "Questioning AA (Allegro).opus\n"; - file_stream << "Questioning AA (Moderato).opus\n"; - file_stream << "Questioning AJ (Allegro).opus\n"; - file_stream << "Questioning AJ (Moderato).opus\n"; - file_stream << "Questioning JFA (Allegro).opus\n"; - file_stream << "Questioning JFA (Moderato).opus\n"; - file_stream << "Questioning T&T (Allegro).opus\n"; - file_stream << "Questioning T&T (Moderato).opus\n"; - file_stream << "Speak up Pup.opus\n"; - file_stream << "Suspense (AA).opus\n"; - file_stream << "The Great Truth Burglar.opus\n"; - file_stream << "Trial (AA).opus\n"; - file_stream << "Trial (AJ).opus\n"; - file_stream << "Trial (Miles).opus\n"; - - music_list.flush(); - music_list.close(); + if (!fileExists(&areas_info)) { + qCritical() << "areas.ini doesn't exist!"; + return false; + } + else { + QSettings areas_ini("config/areas.ini", QSettings::IniFormat); + if (areas_ini.childGroups().length() < 1) { + qCritical() << "areas.ini is invalid!"; + return false; + } + } + if (!fileExists(&characters_info)) { + qCritical() << "characters.txt doesn't exist!"; + return false; + } + if (!fileExists(&music_info)) { + qCritical() << "music.txt doesn't exist!"; + return false; } config->beginGroup("Info"); QString config_version = config->value("version", "none").toString(); config->endGroup(); if (config_version == "none") { - QFileInfo check_file("config.ini"); - // TODO: put proper translatable warnings here - if (!(check_file.exists() && check_file.isFile())) { - // TODO: send signal config doesn't exist - generateDefaultConfig(false); + QFileInfo check_file("config/config.ini"); + if (!fileExists(&check_file)) { + qCritical() << "config.ini doesn't exist!"; } else { - // TODO: send signal config is invalid - generateDefaultConfig(true); + qCritical() << "config.ini is invalid!"; } return false; } else if (config_version != QString::number(CONFIG_VERSION)) { bool version_number_is_valid; int current_version = config_version.toInt(&version_number_is_valid); - if (version_number_is_valid) - updateConfig(current_version); + if (version_number_is_valid) { + if (updateConfig(current_version)) + qWarning() << "config.ini was out of date, and has been updated. Please review the changes, and restart the server."; + else + qCritical() << "config.ini is invalid!"; + } else - generateDefaultConfig(true); // Version number isn't a number at all - // This means the config is invalid - // TODO: send invalid config signal + qCritical() << "config.ini is invalid!"; // Version number isn't a number at all + // This means the config is invalid return false; } else { @@ -187,57 +89,19 @@ bool ConfigManager::initConfig() } } -// Setting backup_old to true will move the existing config.ini to -// config_old.ini -void ConfigManager::generateDefaultConfig(bool backup_old) -{ - qDebug() << "Config is invalid or missing, making a new one..."; - QDir dir = QDir::current(); - if (backup_old) { - // TODO: failsafe if config_old.ini already exists - dir.rename("config.ini", "config_old.ini"); - } - - // Group: Info - // This contains basic metadata about the config - config->beginGroup("Info"); - config->setValue("version", CONFIG_VERSION); - config->endGroup(); - - // Group: Options - // This contains general configuration - config->beginGroup("Options"); - config->setValue("language", "en"); - config->setValue("hostname", "$H"); - config->setValue("max_players", "100"); - config->setValue("port", "27016"); - config->setValue("webao_enable", "true"); - config->setValue("webao_port", "27017"); - config->setValue("modpass", "password"); - config->setValue("advertise", "true"); - config->setValue("ms_ip", "master.aceattorneyonline.com"); - config->setValue("ms_port", "27016"); - config->setValue("server_name", "My First Server"); - config->setValue("server_description", "This is my flashy new server"); - config->setValue("multiclient_limit", "16"); - config->setValue("max_message_size", "256"); - config->endGroup(); -} - // Ensure version continuity with config versions -void ConfigManager::updateConfig(int current_version) +bool ConfigManager::updateConfig(int current_version) { if (current_version > CONFIG_VERSION) { // Config version is newer than the latest version, and the config is // invalid This could also mean the server is out of date, and the user // should be shown a relevant message Regardless, regen the config // anyways - // TODO: send signal config is invalid - generateDefaultConfig(true); + return false; } else if (current_version < 0) { // Negative version number? Invalid! - generateDefaultConfig(true); + return false; } else { // TODO: send signal config is out of date, and is being updated @@ -253,6 +117,7 @@ void ConfigManager::updateConfig(int current_version) break; // This is the newest version, and nothing more needs to be // done } + return true; } } @@ -293,3 +158,8 @@ bool ConfigManager::loadServerSettings(server_settings* settings) return true; } } + +bool ConfigManager::fileExists(QFileInfo* file) +{ + return (file->exists() && file->isFile()); +} diff --git a/src/main.cpp b/src/main.cpp index c21be04..1854a77 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -15,78 +15,55 @@ // You should have received a copy of the GNU Affero General Public License // // along with this program. If not, see . // ////////////////////////////////////////////////////////////////////////////////////// -#include "include/akashimain.h" +#include "include/advertiser.h" +#include "include/server.h" +#include "include/config_manager.h" -#ifdef _WIN32 -#include -#endif - -#include -#include -#include +#include #include -#include -#include -#include + +Advertiser* advertiser; +Server* server; int main(int argc, char* argv[]) { -#ifdef _WIN32 - if (AttachConsole(ATTACH_PARENT_PROCESS)) { - freopen("CONOUT$", "w", stdout); - freopen("CONOUT$", "w", stderr); - } -#endif -#ifdef __linux__ - // We have to do this before the QApplication is instantiated - // As a result, we can't use QCommandLineParser - for(int i = 0; i < argc; i++) { - if(strcmp("-l", argv[i]) == 0 || strcmp("--headless", argv[i]) == 0){ - setenv("QT_QPA_PLATFORM", "minimal", 1); + QCoreApplication app(argc, argv); + QCoreApplication::setApplicationName("akashi"); + QCoreApplication::setApplicationVersion("0.0.1"); + + qDebug("Main application started"); + + ConfigManager config_manager; + if (config_manager.initConfig()) { + // Config is sound, so proceed with starting the server + // Validate some of the config before passing it on + ConfigManager::server_settings settings; + bool config_valid = config_manager.loadServerSettings(&settings); + if (!config_valid) { + qCritical() << "config.ini is invalid!"; + qCritical() << "Exiting server due to configuration issue."; + return EXIT_FAILURE; } + + else { + if (settings.advertise_server) { + // TODO: send signal advertiser started + advertiser = + new Advertiser(settings.ms_ip, settings.port, + settings.ws_port, settings.local_port, + settings.name, settings.description); + advertiser->contactMasterServer(); + } + + // TODO: start the server here + // TODO: send signal server starting. + server = new Server(settings.port, settings.ws_port); + server->start(); + } + } else { + qCritical() << "Exiting server due to configuration issue."; + return EXIT_FAILURE; } -#endif - QApplication app(argc, argv); - QApplication::setApplicationName("akashi"); - QApplication::setApplicationVersion("0.0.1"); - - QSettings config("config.ini", QSettings::IniFormat); - config.beginGroup("Options"); - QString language = - config.value("language", QLocale().bcp47Name()).toString(); - - QTranslator qt_translator; - qt_translator.load("qt_" + language, - QLibraryInfo::location(QLibraryInfo::TranslationsPath)); - app.installTranslator(&qt_translator); - - QTranslator translator; - translator.load("akashi_" + language, ":/resource/translation/"); - app.installTranslator(&translator); - - QCommandLineParser parser; - parser.setApplicationDescription( - app.translate("main", "A server for Attorney Online 2")); - parser.addHelpOption(); - parser.addVersionOption(); - - QCommandLineOption headlessOption( - QStringList() << "l" - << "headless", - app.translate("main", "Run the server without a GUI.")); - QCommandLineOption verboseNetworkOption( - QStringList() << "nv" - << "verbose-network", - app.translate("main", "Write all network traffic to the console.")); - parser.addOption(headlessOption); - parser.addOption(verboseNetworkOption); - - parser.process(app); - bool headless = parser.isSet(headlessOption); - - AkashiMain w; - if (!headless) - w.show(); return app.exec(); } diff --git a/src/server.cpp b/src/server.cpp index c21c89e..51ef6ea 100644 --- a/src/server.cpp +++ b/src/server.cpp @@ -43,28 +43,32 @@ void Server::start() if(ws_port != -1) proxy->start(); - QFile char_list("characters.txt"); + QFile char_list("config/characters.txt"); char_list.open(QIODevice::ReadOnly | QIODevice::Text); while (!char_list.atEnd()) { characters.append(char_list.readLine().trimmed()); } char_list.close(); - QFile music_file("music.txt"); + QFile music_file("config/music.txt"); music_file.open(QIODevice::ReadOnly | QIODevice::Text); while (!music_file.atEnd()) { music_list.append(music_file.readLine().trimmed()); } music_file.close(); if(music_list[0].contains(".")) // Add a default category if none exists - music_list.insert(0, "Music"); + music_list.insert(0, "==Music=="); // TODO: add verification that this exists - QSettings areas_ini("areas.ini", QSettings::IniFormat); + QSettings areas_ini("config/areas.ini", QSettings::IniFormat); area_names = areas_ini.childGroups(); for (int i = 0; i < area_names.length(); i++) { QString area_name = area_names[i]; areas.insert(i, new AreaData(characters, area_name, i)); + areas_ini.beginGroup(area_name); + // TODO: more area config + areas[i]->background = areas_ini.value("background", "gs4").toString(); + areas_ini.endGroup(); qDebug() << "Added area" << area_name; } }