3b84f8a04cc8dbafa7b999e6dfca7bee885fa8ba
[oweals/minetest.git] / src / serverlist.cpp
1 /*
2 Minetest
3 Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
4
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU Lesser General Public License as published by
7 the Free Software Foundation; either version 2.1 of the License, or
8 (at your option) any later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 GNU Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public License along
16 with this program; if not, write to the Free Software Foundation, Inc.,
17 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 */
19
20 #include <iostream>
21 #include <sstream>
22 #include <algorithm>
23
24 #include "version.h"
25 #include "main.h" // for g_settings
26 #include "settings.h"
27 #include "serverlist.h"
28 #include "filesys.h"
29 #include "porting.h"
30 #include "log.h"
31 #include "json/json.h"
32 #include "convert_json.h"
33 #include "httpfetch.h"
34 #include "util/string.h"
35
36 namespace ServerList
37 {
38 std::string getFilePath()
39 {
40         std::string serverlist_file = g_settings->get("serverlist_file");
41
42         std::string dir_path = std::string("client") + DIR_DELIM
43                 + "serverlist" + DIR_DELIM;
44         fs::CreateDir(porting::path_user + DIR_DELIM + "client");
45         fs::CreateDir(porting::path_user + DIR_DELIM + dir_path);
46         std::string rel_path = dir_path + serverlist_file;
47         std::string path = porting::path_user + DIR_DELIM + rel_path;
48         return path;
49 }
50
51 std::vector<ServerListSpec> getLocal()
52 {
53         std::string path = ServerList::getFilePath();
54         std::string liststring;
55         if(fs::PathExists(path))
56         {
57                 std::ifstream istream(path.c_str());
58                 if(istream.is_open())
59                 {
60                         std::ostringstream ostream;
61                         ostream << istream.rdbuf();
62                         liststring = ostream.str();
63                         istream.close();
64                 }
65         }
66
67         return ServerList::deSerialize(liststring);
68 }
69
70
71 std::vector<ServerListSpec> getOnline()
72 {
73         Json::Value root = fetchJsonValue((g_settings->get("serverlist_url")+"/list").c_str(), NULL);
74
75         std::vector<ServerListSpec> serverlist;
76
77         if (root.isArray()) {
78                 for (unsigned int i = 0; i < root.size(); i++)
79                 {
80                         if (root[i].isObject()) {
81                                 serverlist.push_back(root[i]);
82                         }
83                 }
84         }
85
86         return serverlist;
87 }
88
89 /*
90         Delete a server fromt he local favorites list
91 */
92 bool deleteEntry (ServerListSpec server)
93 {
94         std::vector<ServerListSpec> serverlist = ServerList::getLocal();
95         for(unsigned i = 0; i < serverlist.size(); i++)
96         {
97                 if  (serverlist[i]["address"] == server["address"]
98                 &&   serverlist[i]["port"]    == server["port"])
99                 {
100                         serverlist.erase(serverlist.begin() + i);
101                 }
102         }
103
104         std::string path = ServerList::getFilePath();
105         std::ostringstream ss(std::ios_base::binary);
106         ss << ServerList::serialize(serverlist);
107         if (!fs::safeWriteToFile(path, ss.str()))
108                 return false;
109         return true;
110 }
111
112 /*
113         Insert a server to the local favorites list
114 */
115 bool insert (ServerListSpec server)
116 {
117         // Remove duplicates
118         ServerList::deleteEntry(server);
119
120         std::vector<ServerListSpec> serverlist = ServerList::getLocal();
121
122         // Insert new server at the top of the list
123         serverlist.insert(serverlist.begin(), server);
124
125         std::string path = ServerList::getFilePath();
126         std::ostringstream ss(std::ios_base::binary);
127         ss << ServerList::serialize(serverlist);
128         fs::safeWriteToFile(path, ss.str());
129
130         return false;
131 }
132
133 std::vector<ServerListSpec> deSerialize(std::string liststring)
134 {
135         std::vector<ServerListSpec> serverlist;
136         std::istringstream stream(liststring);
137         std::string line, tmp;
138         while (std::getline(stream, line))
139         {
140                 std::transform(line.begin(), line.end(),line.begin(), ::toupper);
141                 if (line == "[SERVER]")
142                 {
143                         ServerListSpec thisserver;
144                         std::getline(stream, tmp);
145                         thisserver["name"] = tmp;
146                         std::getline(stream, tmp);
147                         thisserver["address"] = tmp;
148                         std::getline(stream, tmp);
149                         thisserver["port"] = tmp;
150                         std::getline(stream, tmp);
151                         thisserver["description"] = tmp;
152                         serverlist.push_back(thisserver);
153                 }
154         }
155         return serverlist;
156 }
157
158 std::string serialize(std::vector<ServerListSpec> serverlist)
159 {
160         std::string liststring;
161         for(std::vector<ServerListSpec>::iterator i = serverlist.begin(); i != serverlist.end(); i++)
162         {
163                 liststring += "[server]\n";
164                 liststring += (*i)["name"].asString() + "\n";
165                 liststring += (*i)["address"].asString() + "\n";
166                 liststring += (*i)["port"].asString() + "\n";
167                 liststring += (*i)["description"].asString() + "\n";
168                 liststring += "\n";
169         }
170         return liststring;
171 }
172
173 std::string serializeJson(std::vector<ServerListSpec> serverlist)
174 {
175         Json::Value root;
176         Json::Value list(Json::arrayValue);
177         for(std::vector<ServerListSpec>::iterator i = serverlist.begin(); i != serverlist.end(); i++)
178         {
179                 list.append(*i);
180         }
181         root["list"] = list;
182         Json::StyledWriter writer;
183         return writer.write( root );
184 }
185
186
187 #if USE_CURL
188 void sendAnnounce(std::string action,
189                 const std::vector<std::string> & clients_names,
190                 double uptime,
191                 u32 game_time,
192                 float lag,
193                 std::string gameid,
194                 std::vector<ModSpec> mods)
195 {
196         Json::Value server;
197         if (action.size())
198                 server["action"] = action;
199         server["port"]    = g_settings->getU16("port");
200         server["address"] = g_settings->get("server_address");
201         if (action != "delete") {
202                 server["name"]         = g_settings->get("server_name");
203                 server["description"]  = g_settings->get("server_description");
204                 server["version"]      = minetest_version_simple;
205                 server["url"]          = g_settings->get("server_url");
206                 server["creative"]     = g_settings->getBool("creative_mode");
207                 server["damage"]       = g_settings->getBool("enable_damage");
208                 server["password"]     = g_settings->getBool("disallow_empty_password");
209                 server["pvp"]          = g_settings->getBool("enable_pvp");
210                 server["uptime"]       = (int) uptime;
211                 server["game_time"]    = game_time;
212                 server["clients"]      = (int) clients_names.size();
213                 server["clients_max"]  = g_settings->get("max_users");
214                 server["clients_list"] = Json::Value(Json::arrayValue);
215                 for (std::vector<std::string>::const_iterator it = clients_names.begin();
216                                 it != clients_names.end();
217                                 ++it) {
218                         server["clients_list"].append(*it);
219                 }
220                 if (gameid != "") server["gameid"] = gameid;
221         }
222
223         if (server["action"] == "start") {
224                 server["dedicated"]         = g_settings->getBool("server_dedicated");
225                 server["rollback"]          = g_settings->getBool("enable_rollback_recording");
226                 server["mapgen"]            = g_settings->get("mg_name");
227                 server["privs"]             = g_settings->get("default_privs");
228                 server["can_see_far_names"] = g_settings->getBool("unlimited_player_transfer_distance");
229                 server["mods"]              = Json::Value(Json::arrayValue);
230                 for (std::vector<ModSpec>::iterator m = mods.begin();
231                                 m != mods.end(); ++m) {
232                         server["mods"].append(m->name);
233                 }
234                 actionstream << "Announcing to " << g_settings->get("serverlist_url") << std::endl;
235         } else {
236                 if (lag)
237                         server["lag"] = lag;
238         }
239
240         Json::FastWriter writer;
241         HTTPFetchRequest fetchrequest;
242         fetchrequest.url = g_settings->get("serverlist_url") + std::string("/announce");
243         fetchrequest.post_fields["json"] = writer.write(server);
244         fetchrequest.multipart = true;
245         httpfetch_async(fetchrequest);
246 }
247 #endif
248
249 } //namespace ServerList