]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_spanningtree/override_map.cpp
612df80f3ef0802b9e8ee04f1d8c83ddf084893f
[user/henk/code/inspircd.git] / src / modules / m_spanningtree / override_map.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2014 Adam <Adam@anope.org>
5  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
6  *   Copyright (C) 2007-2008 Craig Edwards <craigedwards@brainbox.cc>
7  *   Copyright (C) 2008 Robin Burchell <robin+git@viroteck.net>
8  *
9  * This file is part of InspIRCd.  InspIRCd is free software: you can
10  * redistribute it and/or modify it under the terms of the GNU General Public
11  * License as published by the Free Software Foundation, version 2.
12  *
13  * This program is distributed in the hope that it will be useful, but WITHOUT
14  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
15  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
16  * details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
20  */
21
22
23 #include "inspircd.h"
24
25 #include "main.h"
26 #include "utils.h"
27 #include "treeserver.h"
28 #include "commands.h"
29
30 CommandMap::CommandMap(Module* Creator)
31         : Command(Creator, "MAP", 0, 1)
32 {
33         Penalty = 2;
34 }
35
36 static inline bool IsHidden(User* user, TreeServer* server)
37 {
38         if (!user->IsOper())
39         {
40                 if (server->Hidden)
41                         return true;
42                 if (Utils->HideULines && server->IsULine())
43                         return true;
44         }
45
46         return false;
47 }
48
49 // Calculate the map depth the servers go, and the longest server name
50 static void GetDepthAndLen(TreeServer* current, unsigned int depth, unsigned int& max_depth, unsigned int& max_len)
51 {
52         if (depth > max_depth)
53                 max_depth = depth;
54         if (current->GetName().length() > max_len)
55                 max_len = current->GetName().length();
56
57         const TreeServer::ChildServers& servers = current->GetChildren();
58         for (TreeServer::ChildServers::const_iterator i = servers.begin(); i != servers.end(); ++i)
59         {
60                 TreeServer* child = *i;
61                 GetDepthAndLen(child, depth + 1, max_depth, max_len);
62         }
63 }
64
65 static std::vector<std::string> GetMap(User* user, TreeServer* current, unsigned int max_len, unsigned int depth)
66 {
67         float percent = 0;
68
69         const user_hash& users = ServerInstance->Users->GetUsers();
70         if (!users.empty())
71         {
72                 // If there are no users, WHO THE HELL DID THE /MAP?!?!?!
73                 percent = current->UserCount * 100.0 / users.size();
74         }
75
76         std::string buffer = current->GetName();
77         if (user->IsOper())
78         {
79                 buffer += " (" + current->GetID() + ")";
80         }
81
82         // Pad with spaces until its at max len, max_len must always be >= my names length
83         buffer.append(max_len - current->GetName().length(), ' ');
84
85         char buf[16];
86         snprintf(buf, sizeof(buf), "%5d [%5.2f%%]", current->UserCount, percent);
87         buffer += buf;
88
89         if (user->IsOper())
90         {
91                 time_t secs_up = ServerInstance->Time() - current->age;
92                 buffer += " [Up: " + ModuleSpanningTree::TimeToStr(secs_up) + (current->rtt == 0 ? "]" : " Lag: " + ConvToStr(current->rtt) + "ms]");
93         }
94
95         std::vector<std::string> map;
96         map.push_back(buffer);
97
98         const TreeServer::ChildServers& servers = current->GetChildren();
99         for (TreeServer::ChildServers::const_iterator i = servers.begin(); i != servers.end(); ++i)
100         {
101                 TreeServer* child = *i;
102
103                 if (IsHidden(user, child))
104                         continue;
105
106                 bool last = true;
107                 for (TreeServer::ChildServers::const_iterator j = i + 1; last && j != servers.end(); ++j)
108                         if (!IsHidden(user, *j))
109                                 last = false;
110
111                 unsigned int next_len;
112
113                 if (user->IsOper() || !Utils->FlatLinks)
114                 {
115                         // This child is indented by us, so remove the depth from the max length to align the users properly
116                         next_len = max_len - 2;
117                 }
118                 else
119                 {
120                         // This user can not see depth, so max_len remains constant
121                         next_len = max_len;
122                 }
123
124                 // Build the map for this child
125                 std::vector<std::string> child_map = GetMap(user, child, next_len, depth + 1);
126
127                 for (std::vector<std::string>::const_iterator j = child_map.begin(); j != child_map.end(); ++j)
128                 {
129                         const char* prefix;
130
131                         if (user->IsOper() || !Utils->FlatLinks)
132                         {
133                                 // If this server is not the root child
134                                 if (j != child_map.begin())
135                                 {
136                                         // If this child is not my last child, then add |
137                                         // to be able to "link" the next server in my list to me, and to indent this childs servers
138                                         if (!last)
139                                                 prefix = "| ";
140                                         // Otherwise this is my last child, so just use a space as theres nothing else linked to me below this
141                                         else
142                                                 prefix = "  ";
143                                 }
144                                 // If we get here, this server must be the root child
145                                 else
146                                 {
147                                         // If this is the last child, it gets a `-
148                                         if (last)
149                                                 prefix = "`-";
150                                         // Otherwise this isn't the last child, so it gets |-
151                                         else
152                                                 prefix = "|-";
153                                 }
154                         }
155                         else
156                                 // User can't see depth, so use no prefix
157                                 prefix = "";
158
159                         // Add line to the map
160                         map.push_back(prefix + *j);
161                 }
162         }
163
164         return map;
165 }
166
167 CmdResult CommandMap::Handle(const std::vector<std::string>& parameters, User* user)
168 {
169         if (parameters.size() > 0)
170         {
171                 // Remote MAP, the target server is the 1st parameter
172                 TreeServer* s = Utils->FindServerMask(parameters[0]);
173                 if (!s)
174                 {
175                         user->WriteNumeric(ERR_NOSUCHSERVER, "%s :No such server", parameters[0].c_str());
176                         return CMD_FAILURE;
177                 }
178
179                 if (!s->IsRoot())
180                         return CMD_SUCCESS;
181         }
182
183         // Max depth and max server name length
184         unsigned int max_depth = 0;
185         unsigned int max_len = 0;
186         GetDepthAndLen(Utils->TreeRoot, 0, max_depth, max_len);
187
188         unsigned int max;
189         if (user->IsOper() || !Utils->FlatLinks)
190         {
191                 // Each level of the map is indented by 2 characters, making the max possible line (max_depth * 2) + max_len
192                 max = (max_depth * 2) + max_len;
193         }
194         else
195         {
196                 // This user can't see any depth
197                 max = max_len;
198         }
199
200         std::vector<std::string> map = GetMap(user, Utils->TreeRoot, max, 0);
201         for (std::vector<std::string>::const_iterator i = map.begin(); i != map.end(); ++i)
202                 user->SendText(":%s %03d %s :%s", ServerInstance->Config->ServerName.c_str(),
203                         RPL_MAP, user->nick.c_str(), i->c_str());
204
205         size_t totusers = ServerInstance->Users->GetUsers().size();
206         float avg_users = (float) totusers / Utils->serverlist.size();
207
208         user->SendText(":%s %03d %s :%u server%s and %u user%s, average %.2f users per server",
209                 ServerInstance->Config->ServerName.c_str(), RPL_MAPUSERS, user->nick.c_str(),
210                 (unsigned int)Utils->serverlist.size(), (Utils->serverlist.size() > 1 ? "s" : ""), (unsigned int)totusers, (totusers > 1 ? "s" : ""), avg_users);
211         user->SendText(":%s %03d %s :End of /MAP", ServerInstance->Config->ServerName.c_str(),
212                 RPL_ENDMAP, user->nick.c_str());
213
214         return CMD_SUCCESS;
215 }
216
217 RouteDescriptor CommandMap::GetRouting(User* user, const std::vector<std::string>& parameters)
218 {
219         if (!parameters.empty())
220                 return ROUTE_UNICAST(parameters[0]);
221         return ROUTE_LOCALONLY;
222 }