/*
Copyright (c) 2013 yvt
Portion of the code is based on Serverbrowser.cpp.
This file is part of OpenSpades.
OpenSpades 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.
OpenSpades 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 OpenSpades. If not, see .
*/
#include
#include "MainScreenHelper.h"
#include "Serverbrowser.h"
#include
#include
#include
#include
#include "MainScreen.h"
#include
#include
#include
#include "MainWindow.h" // for credits
SPADES_SETTING(cl_serverListUrl, "http://services.buildandshoot.com/serverlist.json");
namespace spades {
namespace gui {
// FIXME: mostly duplicated code with Serverbrowser.cpp
class MainScreenHelper::ServerListQuery: public Thread {
Mutex infoLock;
Handle owner;
std::string buffer;
static size_t curlWriteCallback( void *ptr, size_t size, size_t nmemb, ServerListQuery* query ) {
size_t dataSize = size * nmemb;
return query->writeCallback( ptr, dataSize );
}
size_t writeCallback( void *ptr, size_t size ) {
buffer.append(reinterpret_cast(ptr), size);
return size;
}
void ReturnResult(MainScreenServerList *list){
AutoLocker lock(&(owner->newResultArrayLock));
delete owner->newResult;
owner->newResult = list;
owner = NULL; // release owner
}
void ProcessResponse() {
Json::Reader reader;
Json::Value root;
MainScreenServerList *resp = new MainScreenServerList();
try{
if( reader.parse( buffer, root, false ) ) {
for( Json::Value::iterator it = root.begin(); it != root.end(); ++it ) {
Json::Value &obj = *it;
Serveritem* srv = Serveritem::create( obj );
if( srv ) {
resp->list.push_back(new MainScreenServerItem(srv));
delete srv;
}
}
}
ReturnResult(resp);
}catch(...){
delete resp;
throw;
}
}
public:
ServerListQuery(MainScreenHelper *owner):
owner(owner){
}
virtual void Run() {
try{
CURL* cHandle = curl_easy_init();
if( cHandle ) {
try{
curl_easy_setopt( cHandle, CURLOPT_USERAGENT, OpenSpades_VER_STR );
curl_easy_setopt( cHandle, CURLOPT_URL, cl_serverListUrl.CString() );
curl_easy_setopt( cHandle, CURLOPT_WRITEFUNCTION, &ServerListQuery::curlWriteCallback );
curl_easy_setopt( cHandle, CURLOPT_WRITEDATA, this );
if( 0 == curl_easy_perform( cHandle ) ) {
ProcessResponse();
} else {
SPRaise("HTTP request error.");
}
curl_easy_cleanup( cHandle );
}catch(...){
curl_easy_cleanup( cHandle );
throw;
}
}else{
SPRaise("Failed to create cURL object.");
}
}catch(std::exception& ex) {
MainScreenServerList *lst = new MainScreenServerList();
lst->message = ex.what();
ReturnResult(lst);
}catch(...) {
MainScreenServerList *lst = new MainScreenServerList();
lst->message = "Unknown error.";
ReturnResult(lst);
}
}
};
MainScreenHelper::MainScreenHelper(MainScreen *scr):
mainScreen(scr),
result(NULL),
newResult(NULL),
query(NULL){
SPADES_MARK_FUNCTION();
}
MainScreenHelper::~MainScreenHelper() {
SPADES_MARK_FUNCTION();
if(query) {
query->MarkForAutoDeletion();
}
delete result;
delete newResult;
}
void MainScreenHelper::MainScreenDestroyed() {
SPADES_MARK_FUNCTION();
mainScreen = NULL;
}
bool MainScreenHelper::PollServerListState() {
SPADES_MARK_FUNCTION();
AutoLocker lock(&newResultArrayLock);
if(newResult) {
delete result;
result = newResult;
newResult = NULL;
query->MarkForAutoDeletion();
query = NULL;
return true;
}
return false;
}
void MainScreenHelper::StartQuery() {
if(query) {
// ongoing
return;
}
query = new ServerListQuery(this);
query->Start();
}
static std::string TextifyHTML(const std::string& html) {
std::string out;
int blankLineLength = 0;
bool inTag = false;
bool inTagName = false;
bool hasLetter = false;
std::string tagName;
for(size_t i = 0; i < html.size(); i++){
if(html[i] == '<') {
inTag = true;
inTagName = true;
tagName.clear();
}
if(inTag) {
if(inTagName) {
if(html[i] == '/')
i++;
if(html[i] < 'a' || html[i] > 'z') {
inTagName = false;
if(html[i] == '>') {
inTag = false;
}
if(tagName == "br" ||
tagName == "h3" ||
tagName == "h2") {
if(blankLineLength < 1) {
out += "\n";
hasLetter = false;
blankLineLength++;
}
}
}else{
tagName += html[i];
}
}else if(html[i] == '>') {
inTag = false;
}
continue;
}
bool isLetter = !isspace(html[i]);
if(hasLetter == false && isLetter == false)
continue;
hasLetter = true;
blankLineLength = 0;
out += html[i];
}
return out;
}
std::string MainScreenHelper::GetCredits() {
std::string html = std::string(reinterpret_cast(aboutText), sizeof(aboutText));
html = Replace(html, "${PACKAGE_STRING}", PACKAGE_STRING);
html = Replace(html, "©", "(C)");
return TextifyHTML(html);
}
struct serverSorter {
int type;
bool order;
serverSorter( int type_, bool order_ ) : type(type_), order(order_) {;}
bool operator() ( MainScreenServerItem* x, MainScreenServerItem* y ) const
{
if( order ) {
MainScreenServerItem* t = x;
x = y;
y = t;
}
switch( type ) {
default:
case 0: return asInt( x->GetPing(), y->GetPing() );
case 1: return asInt( x->GetNumPlayers(), y->GetNumPlayers() );
case 2: return asString( x->GetName(), y->GetName() );
case 3: return asString( x->GetMapName(), y->GetMapName() );
case 4: return asString( x->GetGameMode(), y->GetGameMode() );
case 5: return asString( x->GetProtocol(), y->GetProtocol() );
case 6: return asString( x->GetCountry(), y->GetCountry() );
}
}
bool asInt( int x, int y ) const
{
if( x == y ) return false;
return (x( c1 ) ) < std::tolower( static_cast( c1 ) ));
}
};
CScriptArray *MainScreenHelper::GetServerList(std::string sortKey, bool descending) {
if(result == NULL){
return NULL;
}
std::vector lst = result->list;
if(lst.empty())
return NULL;
int sortKeyId = -1;
if(sortKey == "Ping") {
sortKeyId = 0;
}else if(sortKey == "NumPlayers") {
sortKeyId = 1;
}else if(sortKey == "Name") {
sortKeyId = 2;
}else if(sortKey == "MapName") {
sortKeyId = 3;
}else if(sortKey == "GameMode") {
sortKeyId = 4;
}else if(sortKey == "Protocol") {
sortKeyId = 5;
}else if(sortKey == "Country") {
sortKeyId = 6;
}
if(sortKeyId != -1){
std::sort( lst.begin(), lst.end(), serverSorter( sortKeyId, descending ) );
}
asIScriptEngine *eng = ScriptManager::GetInstance()->GetEngine();
asIObjectType* t = eng->GetObjectTypeById(eng->GetTypeIdByDecl("array"));
SPAssert(t != NULL);
CScriptArray *arr = new CScriptArray(lst.size(), t);
for(size_t i = 0; i < lst.size(); i++){
arr->SetValue((asUINT)i, &(lst[i]));
}
return arr;
}
std::string MainScreenHelper::ConnectServer() {
if(mainScreen == NULL){
return "mainScreen == NULL";
}
return mainScreen->Connect();
}
std::string MainScreenHelper::GetServerListQueryMessage() {
if(result == NULL)
return "";
return result->message;
}
std::string MainScreenHelper::GetPendingErrorMessage() {
std::string s = errorMessage;
errorMessage.clear();
return s;
}
MainScreenServerList::~MainScreenServerList() {
for(size_t i = 0; i < list.size(); i++)
list[i]->Release();
}
MainScreenServerItem::MainScreenServerItem(Serveritem *item) {
SPADES_MARK_FUNCTION();
name = item->Name();
address = item->Ip();
mapName = item->Map();
gameMode = item->GameMode();
country = item->Country();
protocol = item->Version();
ping = item->Ping();
numPlayers = item->Players();
maxPlayers = item->MaxPlayers();
}
MainScreenServerItem::~MainScreenServerItem() {
SPADES_MARK_FUNCTION();
}
}
}