]> git.stg.codes - stg.git/blob - projects/stargazer/plugins/configuration/sgconfig/parser_users.cpp
Use std::jthread and C++17.
[stg.git] / projects / stargazer / plugins / configuration / sgconfig / parser_users.cpp
1 /*
2  *    This program is free software; you can redistribute it and/or modify
3  *    it under the terms of the GNU General Public License as published by
4  *    the Free Software Foundation; either version 2 of the License, or
5  *    (at your option) any later version.
6  *
7  *    This program is distributed in the hope that it will be useful,
8  *    but WITHOUT ANY WARRANTY; without even the implied warranty of
9  *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10  *    GNU General Public License for more details.
11  *
12  *    You should have received a copy of the GNU General Public License
13  *    along with this program; if not, write to the Free Software
14  *    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
15  */
16
17 /*
18  *    Author : Boris Mikhailenko <stg34@stargazer.dp.ua>
19  *    Author : Maxim Mamontov <faust@stargazer.dp.ua>
20  */
21
22 #include "parser_users.h"
23
24 #include "stg/users.h"
25 #include "stg/user.h"
26 #include "stg/user_conf.h"
27 #include "stg/user_stat.h"
28 #include "stg/tariffs.h"
29 #include "stg/tariff.h"
30 #include "stg/user_property.h"
31
32 #include <cstdio>
33 #include <cassert>
34
35 using STG::PARSER::GET_USERS;
36 using STG::PARSER::GET_USER;
37 using STG::PARSER::ADD_USER;
38 using STG::PARSER::DEL_USER;
39 using STG::PARSER::CHG_USER;
40 using STG::PARSER::CHECK_USER;
41
42 const char * GET_USERS::tag  = "GetUsers";
43 const char * GET_USER::tag   = "GetUser";
44 const char * ADD_USER::tag   = "AddUser";
45 const char * CHG_USER::tag   = "SetUser";
46 const char * DEL_USER::tag   = "DelUser";
47 const char * CHECK_USER::tag = "CheckUser";
48
49 using UserPtr = STG::User*;
50 using ConstUserPtr = const STG::User*;
51
52 namespace
53 {
54
55 std::string UserToXML(const STG::User & user, bool loginInStart, bool showPass, time_t lastTime = 0)
56 {
57     std::string answer;
58
59     if (loginInStart)
60         answer += "<User login=\"" + user.GetLogin() + "\" result=\"ok\">";
61     else
62         answer += "<User result=\"ok\">";
63
64     answer += "<Login value=\"" + user.GetLogin() + "\"/>";
65
66     if (user.GetProperties().password.ModificationTime() > lastTime)
67     {
68         if (showPass)
69             answer += "<Password value=\"" + user.GetProperties().password.Get() + "\" />";
70         else
71             answer += "<Password value=\"++++++\"/>";
72     }
73
74     if (user.GetProperties().cash.ModificationTime() > lastTime)
75         answer += "<Cash value=\"" + std::to_string(user.GetProperties().cash.Get()) + "\"/>";
76     if (user.GetProperties().freeMb.ModificationTime() > lastTime)
77         answer += "<FreeMb value=\"" + std::to_string(user.GetProperties().freeMb.Get()) + "\"/>";
78     if (user.GetProperties().credit.ModificationTime() > lastTime)
79         answer += "<Credit value=\"" + std::to_string(user.GetProperties().credit.Get()) + "\"/>";
80
81     if (user.GetProperties().nextTariff.Get() != "")
82     {
83         if (user.GetProperties().tariffName.ModificationTime() > lastTime ||
84             user.GetProperties().nextTariff.ModificationTime() > lastTime)
85             answer += "<Tariff value=\"" + user.GetProperties().tariffName.Get() + "/" + user.GetProperties().nextTariff.Get() + "\"/>";
86     }
87     else
88     {
89         if (user.GetProperties().tariffName.ModificationTime() > lastTime)
90             answer += "<Tariff value=\"" + user.GetProperties().tariffName.Get() + "\"/>";
91     }
92
93     if (user.GetProperties().note.ModificationTime() > lastTime)
94         answer += "<Note value=\"" + Encode12str(user.GetProperties().note) + "\"/>";
95     if (user.GetProperties().phone.ModificationTime() > lastTime)
96         answer += "<Phone value=\"" + Encode12str(user.GetProperties().phone) + "\"/>";
97     if (user.GetProperties().address.ModificationTime() > lastTime)
98         answer += "<Address value=\"" + Encode12str(user.GetProperties().address) + "\"/>";
99     if (user.GetProperties().email.ModificationTime() > lastTime)
100         answer += "<Email value=\"" + Encode12str(user.GetProperties().email) + "\"/>";
101
102     std::vector<const STG::UserPropertyLogged<std::string> *> userdata;
103     userdata.push_back(user.GetProperties().userdata0.GetPointer());
104     userdata.push_back(user.GetProperties().userdata1.GetPointer());
105     userdata.push_back(user.GetProperties().userdata2.GetPointer());
106     userdata.push_back(user.GetProperties().userdata3.GetPointer());
107     userdata.push_back(user.GetProperties().userdata4.GetPointer());
108     userdata.push_back(user.GetProperties().userdata5.GetPointer());
109     userdata.push_back(user.GetProperties().userdata6.GetPointer());
110     userdata.push_back(user.GetProperties().userdata7.GetPointer());
111     userdata.push_back(user.GetProperties().userdata8.GetPointer());
112     userdata.push_back(user.GetProperties().userdata9.GetPointer());
113
114     for (size_t i = 0; i < userdata.size(); i++)
115         if (userdata[i]->ModificationTime() > lastTime)
116             answer += "<UserData" + std::to_string(i) + " value=\"" + Encode12str(userdata[i]->Get()) + "\" />";
117
118     if (user.GetProperties().realName.ModificationTime() > lastTime)
119         answer += "<Name value=\"" + Encode12str(user.GetProperties().realName) + "\"/>";
120     if (user.GetProperties().group.ModificationTime() > lastTime)
121         answer += "<Group value=\"" + Encode12str(user.GetProperties().group) + "\"/>";
122     if (user.GetConnectedModificationTime() > lastTime)
123         answer += std::string("<Status value=\"") + (user.GetConnected() ? "1" : "0") + "\"/>";
124     if (user.GetProperties().alwaysOnline.ModificationTime() > lastTime)
125         answer += std::string("<AOnline value=\"") + (user.GetProperties().alwaysOnline.Get() ? "1" : "0") + "\"/>";
126     if (user.GetCurrIPModificationTime() > lastTime)
127         answer += "<CurrIP value=\"" + inet_ntostring(user.GetCurrIP()) + "\"/>";
128     if (user.GetPingTime() > lastTime)
129         answer += "<PingTime value=\"" + std::to_string(user.GetPingTime()) + "\"/>";
130     if (user.GetProperties().ips.ModificationTime() > lastTime)
131         answer += "<IP value=\"" + user.GetProperties().ips.Get().toString() + "\"/>";
132
133     answer += "<Traff";
134     const auto & upload(user.GetProperties().up.Get());
135     const auto & download(user.GetProperties().down.Get());
136     if (user.GetProperties().up.ModificationTime() > lastTime)
137         for (size_t j = 0; j < DIR_NUM; j++)
138             answer += " MU" + std::to_string(j) + "=\"" + std::to_string(upload[j]) + "\"";
139     if (user.GetProperties().down.ModificationTime() > lastTime)
140         for (size_t j = 0; j < DIR_NUM; j++)
141             answer += " MD" + std::to_string(j) + "=\"" + std::to_string(download[j]) + "\"";
142     if (user.GetSessionUploadModificationTime() > lastTime)
143         for (size_t j = 0; j < DIR_NUM; j++)
144             answer += " SU" + std::to_string(j) + "=\"" + std::to_string(user.GetSessionUpload()[j]) + "\"";
145     if (user.GetSessionDownloadModificationTime() > lastTime)
146         for (size_t j = 0; j < DIR_NUM; j++)
147             answer += " SD" + std::to_string(j) + "=\"" + std::to_string(user.GetSessionDownload()[j]) + "\"";
148     answer += "/>";
149
150     if (user.GetProperties().disabled.ModificationTime() > lastTime)
151         answer += std::string("<Down value=\"") + (user.GetProperties().disabled.Get() ? "1" : "0") + "\"/>";
152     if (user.GetProperties().disabledDetailStat.ModificationTime() > lastTime)
153         answer += std::string("<DisableDetailStat value=\"") + (user.GetProperties().disabledDetailStat.Get() ? "1" : "0") + "\"/>";
154     if (user.GetProperties().passive.ModificationTime() > lastTime)
155         answer += std::string("<Passive value=\"") + (user.GetProperties().passive.Get() ? "1" : "0") + "\"/>";
156     if (user.GetProperties().lastCashAdd.ModificationTime() > lastTime)
157         answer += "<LastCash value=\"" + std::to_string(user.GetProperties().lastCashAdd.Get()) + "\"/>";
158     if (user.GetProperties().lastCashAddTime.ModificationTime() > lastTime)
159         answer += "<LastTimeCash value=\"" + std::to_string(user.GetProperties().lastCashAddTime.Get()) + "\"/>";
160     if (user.GetProperties().lastActivityTime.ModificationTime() > lastTime)
161         answer += "<LastActivityTime value=\"" + std::to_string(user.GetProperties().lastActivityTime.Get()) + "\"/>";
162     if (user.GetProperties().creditExpire.ModificationTime() > lastTime)
163         answer += "<CreditExpire value=\"" + std::to_string(user.GetProperties().creditExpire.Get()) + "\"/>";
164
165     if (lastTime == 0)
166     {
167         answer += "<AuthorizedBy>";
168         std::vector<std::string> list(user.GetAuthorizers());
169         for (std::vector<std::string>::const_iterator it = list.begin(); it != list.end(); ++it)
170             answer += "<Auth name=\"" + *it + "\"/>";
171         answer += "</AuthorizedBy>";
172     }
173
174     answer += "</User>";
175
176     return answer;
177 }
178
179 } // namespace anonymous
180
181 int GET_USERS::Start(void *, const char * el, const char ** attr)
182 {
183     if (strcasecmp(el, m_tag.c_str()) != 0)
184     {
185         printfd(__FILE__, "Got wrong tag: '%s' instead of '%s'\n", el, m_tag.c_str());
186         return -1;
187     }
188
189     while (attr && *attr && *(attr + 1))
190     {
191         if (strcasecmp(*attr, "LastUpdate") == 0)
192             str2x(*(attr + 1), m_lastUserUpdateTime);
193         ++attr;
194     }
195
196     return 0;
197 }
198
199 void GET_USERS::CreateAnswer()
200 {
201     int h = m_users.OpenSearch();
202     assert(h);
203
204     if (m_lastUserUpdateTime > 0)
205         m_answer = "<Users LastUpdate=\"" + std::to_string(time(NULL)) + "\">";
206     else
207         m_answer = "<Users>";
208
209     UserPtr u;
210
211     while (m_users.SearchNext(h, &u) == 0)
212         m_answer += UserToXML(*u, true, m_currAdmin.priv().userConf || m_currAdmin.priv().userPasswd, m_lastUserUpdateTime);
213
214     m_users.CloseSearch(h);
215
216     m_answer += "</Users>";
217 }
218
219 int GET_USER::Start(void *, const char * el, const char ** attr)
220 {
221     if (strcasecmp(el, m_tag.c_str()) != 0)
222         return -1;
223
224     if (attr[1] == NULL)
225         return -1;
226
227     m_login = attr[1];
228     return 0;
229 }
230
231 void GET_USER::CreateAnswer()
232 {
233     ConstUserPtr u;
234
235     if (m_users.FindByName(m_login, &u))
236         m_answer = "<User result=\"error\" reason=\"User not found.\"/>";
237     else
238         m_answer = UserToXML(*u, false, m_currAdmin.priv().userConf || m_currAdmin.priv().userPasswd);
239 }
240
241 int ADD_USER::Start(void *, const char * el, const char ** attr)
242 {
243     m_depth++;
244
245     if (m_depth == 1)
246     {
247         if (strcasecmp(el, m_tag.c_str()) == 0)
248             return 0;
249     }
250     else
251     {
252         if (strcasecmp(el, "login") == 0)
253         {
254             m_login = attr[1];
255             return 0;
256         }
257     }
258     return -1;
259 }
260
261 void ADD_USER::CreateAnswer()
262 {
263     if (m_users.Exists(m_login))
264         m_answer = "<" + m_tag + " result=\"error\" reason=\"User '" + m_login + "' exists.\"/>";
265     else if (m_users.Add(m_login, &m_currAdmin) == 0)
266         m_answer = "<" + m_tag + " result=\"ok\"/>";
267     else
268         m_answer = "<" + m_tag + " result=\"error\" reason=\"Access denied\"/>";
269 }
270
271 int CHG_USER::Start(void *, const char * el, const char ** attr)
272 {
273     m_depth++;
274
275     if (m_depth == 1)
276     {
277         if (strcasecmp(el, m_tag.c_str()) == 0)
278             return 0;
279     }
280     else
281     {
282         if (strcasecmp(el, "login") == 0)
283         {
284             m_login = attr[1];
285             return 0;
286         }
287
288         if (strcasecmp(el, "ip") == 0)
289         {
290             m_ucr.ips = UserIPs::parse(attr[1]);
291             return 0;
292         }
293
294         if (strcasecmp(el, "password") == 0)
295         {
296             m_ucr.password = attr[1];
297             return 0;
298         }
299
300         if (strcasecmp(el, "address") == 0)
301         {
302             m_ucr.address = Decode21str(attr[1]);
303             return 0;
304         }
305
306         if (strcasecmp(el, "aonline") == 0)
307         {
308             m_ucr.alwaysOnline = (*(attr[1]) != '0');
309             return 0;
310         }
311
312         if (strcasecmp(el, "cash") == 0)
313         {
314             if (attr[2] && (strcasecmp(attr[2], "msg") == 0))
315                 m_cashMsg = Decode21str(attr[3]);
316
317             double cash = 0;
318             if (strtodouble2(attr[1], cash) == 0)
319                 m_usr.cash = cash;
320
321             m_cashMustBeAdded = (strcasecmp(attr[0], "add") == 0);
322
323             return 0;
324         }
325
326         if (strcasecmp(el, "CreditExpire") == 0)
327         {
328             long int creditExpire = 0;
329             if (str2x(attr[1], creditExpire) == 0)
330                 m_ucr.creditExpire = creditExpire;
331
332             return 0;
333         }
334
335         if (strcasecmp(el, "credit") == 0)
336         {
337             double credit = 0;
338             if (strtodouble2(attr[1], credit) == 0)
339                 m_ucr.credit = credit;
340             return 0;
341         }
342
343         if (strcasecmp(el, "freemb") == 0)
344         {
345             double freeMb = 0;
346             if (strtodouble2(attr[1], freeMb) == 0)
347                 m_usr.freeMb = freeMb;
348             return 0;
349         }
350
351         if (strcasecmp(el, "down") == 0)
352         {
353             int down = 0;
354             if (str2x(attr[1], down) == 0)
355                 m_ucr.disabled = down;
356             return 0;
357         }
358
359         if (strcasecmp(el, "DisableDetailStat") == 0)
360         {
361             int disabledDetailStat = 0;
362             if (str2x(attr[1], disabledDetailStat) == 0)
363                 m_ucr.disabledDetailStat = disabledDetailStat;
364             return 0;
365         }
366
367         if (strcasecmp(el, "email") == 0)
368         {
369             m_ucr.email = Decode21str(attr[1]);
370             return 0;
371         }
372
373         for (int i = 0; i < USERDATA_NUM; i++)
374         {
375             char name[15];
376             sprintf(name, "userdata%d", i);
377             if (strcasecmp(el, name) == 0)
378             {
379                 m_ucr.userdata[i] = Decode21str(attr[1]);
380                 return 0;
381             }
382         }
383
384         if (strcasecmp(el, "group") == 0)
385         {
386             m_ucr.group = Decode21str(attr[1]);
387             return 0;
388         }
389
390         if (strcasecmp(el, "note") == 0)
391         {
392             m_ucr.note = Decode21str(attr[1]);
393             return 0;
394         }
395
396         if (strcasecmp(el, "passive") == 0)
397         {
398             int passive = 0;
399             if (str2x(attr[1], passive) == 0)
400                 m_ucr.passive = passive;
401             return 0;
402         }
403
404         if (strcasecmp(el, "phone") == 0)
405         {
406             m_ucr.phone = Decode21str(attr[1]);
407             return 0;
408         }
409
410         if (strcasecmp(el, "Name") == 0)
411         {
412             m_ucr.realName = Decode21str(attr[1]);
413             return 0;
414         }
415
416         if (strcasecmp(el, "traff") == 0)
417         {
418             int j = 0;
419             while (attr[j])
420             {
421                 int dir = attr[j][2] - '0';
422
423                 if (strncasecmp(attr[j], "md", 2) == 0)
424                 {
425                     uint64_t t = 0;
426                     str2x(attr[j + 1], t);
427                     m_downr[dir] = t;
428                 }
429                 if (strncasecmp(attr[j], "mu", 2) == 0)
430                 {
431                     uint64_t t = 0;
432                     str2x(attr[j + 1], t);
433                     m_upr[dir] = t;
434                 }
435                 j += 2;
436             }
437             return 0;
438         }
439
440         if (strcasecmp(el, "tariff") == 0)
441         {
442             if (strcasecmp(attr[0], "now") == 0)
443                 m_ucr.tariffName = attr[1];
444
445             if (strcasecmp(attr[0], "delayed") == 0)
446                 m_ucr.nextTariff = attr[1];
447
448             return 0;
449         }
450     }
451     return -1;
452 }
453
454 void CHG_USER::CreateAnswer()
455 {
456     if (ApplyChanges() == 0)
457         m_answer = "<" + m_tag + " result=\"ok\"/>";
458     else
459         m_answer = "<" + m_tag + " result=\"error\"/>";
460 }
461
462 int CHG_USER::ApplyChanges()
463 {
464     printfd(__FILE__, "PARSER_CHG_USER::ApplyChanges()\n");
465     UserPtr u;
466
467     if (m_users.FindByName(m_login, &u))
468         return -1;
469
470     bool check = false;
471     bool alwaysOnline = u->GetProperties().alwaysOnline;
472     if (!m_ucr.alwaysOnline.empty())
473     {
474         check = true;
475         alwaysOnline = m_ucr.alwaysOnline.const_data();
476     }
477     bool onlyOneIP = u->GetProperties().ips.ConstData().onlyOneIP();
478     if (!m_ucr.ips.empty())
479     {
480         check = true;
481         onlyOneIP = m_ucr.ips.const_data().onlyOneIP();
482     }
483
484     if (check && alwaysOnline && !onlyOneIP)
485     {
486         printfd(__FILE__, "Requested change leads to a forbidden state: AlwaysOnline with multiple IP's\n");
487         PluginLogger::get("conf_sg")("%s Requested change leads to a forbidden state: AlwaysOnline with multiple IP's", m_currAdmin.logStr().c_str());
488         return -1;
489     }
490
491     for (size_t i = 0; i < m_ucr.ips.const_data().count(); ++i)
492     {
493         ConstUserPtr user;
494         uint32_t ip = m_ucr.ips.const_data().operator[](i).ip;
495         if (m_users.IsIPInUse(ip, m_login, &user))
496         {
497             printfd(__FILE__, "Trying to assign an IP %s to '%s' that is already in use by '%s'\n", inet_ntostring(ip).c_str(), m_login.c_str(), user->GetLogin().c_str());
498             PluginLogger::get("conf_sg")("%s trying to assign an IP %s to '%s' that is currently in use by '%s'", m_currAdmin.logStr().c_str(), inet_ntostring(ip).c_str(), m_login.c_str(), user->GetLogin().c_str());
499             return -1;
500         }
501     }
502
503     if (!m_ucr.ips.empty())
504         if (!u->GetProperties().ips.Set(m_ucr.ips.const_data(), m_currAdmin, m_login, m_store))
505             return -1;
506
507     if (!m_ucr.alwaysOnline.empty())
508         if (!u->GetProperties().alwaysOnline.Set(m_ucr.alwaysOnline.const_data(),
509                                                  m_currAdmin, m_login, m_store))
510             return -1;
511
512     if (!m_ucr.address.empty())
513         if (!u->GetProperties().address.Set(m_ucr.address.const_data(), m_currAdmin, m_login, m_store))
514             return -1;
515
516     if (!m_ucr.creditExpire.empty())
517         if (!u->GetProperties().creditExpire.Set(m_ucr.creditExpire.const_data(),
518                                                  m_currAdmin, m_login, m_store))
519             return -1;
520
521     if (!m_ucr.credit.empty())
522         if (!u->GetProperties().credit.Set(m_ucr.credit.const_data(), m_currAdmin, m_login, m_store))
523             return -1;
524
525     if (!m_usr.freeMb.empty())
526         if (!u->GetProperties().freeMb.Set(m_usr.freeMb.const_data(), m_currAdmin, m_login, m_store))
527             return -1;
528
529     if (!m_ucr.disabled.empty())
530         if (!u->GetProperties().disabled.Set(m_ucr.disabled.const_data(), m_currAdmin, m_login, m_store))
531             return -1;
532
533     if (!m_ucr.disabledDetailStat.empty())
534         if (!u->GetProperties().disabledDetailStat.Set(m_ucr.disabledDetailStat.const_data(), m_currAdmin, m_login, m_store))
535             return -1;
536
537     if (!m_ucr.email.empty())
538         if (!u->GetProperties().email.Set(m_ucr.email.const_data(), m_currAdmin, m_login, m_store))
539             return -1;
540
541     if (!m_ucr.group.empty())
542         if (!u->GetProperties().group.Set(m_ucr.group.const_data(), m_currAdmin, m_login, m_store))
543             return -1;
544
545     if (!m_ucr.note.empty())
546         if (!u->GetProperties().note.Set(m_ucr.note.const_data(), m_currAdmin, m_login, m_store))
547             return -1;
548
549     std::vector<STG::UserPropertyLogged<std::string> *> userdata;
550     userdata.push_back(u->GetProperties().userdata0.GetPointer());
551     userdata.push_back(u->GetProperties().userdata1.GetPointer());
552     userdata.push_back(u->GetProperties().userdata2.GetPointer());
553     userdata.push_back(u->GetProperties().userdata3.GetPointer());
554     userdata.push_back(u->GetProperties().userdata4.GetPointer());
555     userdata.push_back(u->GetProperties().userdata5.GetPointer());
556     userdata.push_back(u->GetProperties().userdata6.GetPointer());
557     userdata.push_back(u->GetProperties().userdata7.GetPointer());
558     userdata.push_back(u->GetProperties().userdata8.GetPointer());
559     userdata.push_back(u->GetProperties().userdata9.GetPointer());
560
561     for (size_t i = 0; i < userdata.size(); i++)
562         if (!m_ucr.userdata[i].empty())
563             if(!userdata[i]->Set(m_ucr.userdata[i].const_data(), m_currAdmin, m_login, m_store))
564                 return -1;
565
566     if (!m_ucr.passive.empty())
567         if (!u->GetProperties().passive.Set(m_ucr.passive.const_data(), m_currAdmin, m_login, m_store))
568             return -1;
569
570     if (!m_ucr.password.empty())
571         if (!u->GetProperties().password.Set(m_ucr.password.const_data(), m_currAdmin, m_login, m_store))
572             return -1;
573
574     if (!m_ucr.phone.empty())
575         if (!u->GetProperties().phone.Set(m_ucr.phone.const_data(), m_currAdmin, m_login, m_store))
576             return -1;
577
578     if (!m_ucr.realName.empty())
579         if (!u->GetProperties().realName.Set(m_ucr.realName.const_data(), m_currAdmin, m_login, m_store))
580             return -1;
581
582     if (!m_usr.cash.empty())
583     {
584         if (m_cashMustBeAdded)
585         {
586             if (!u->GetProperties().cash.Set(m_usr.cash.const_data() + u->GetProperties().cash,
587                                              m_currAdmin,
588                                              m_login,
589                                              m_store,
590                                              m_cashMsg))
591                 return -1;
592         }
593         else
594         {
595             if (!u->GetProperties().cash.Set(m_usr.cash.const_data(), m_currAdmin, m_login, m_store, m_cashMsg))
596                 return -1;
597         }
598     }
599
600     if (!m_ucr.tariffName.empty())
601     {
602         const auto newTariff = m_tariffs.FindByName(m_ucr.tariffName.const_data());
603         if (newTariff)
604         {
605             const auto tariff = u->GetTariff();
606             std::string message = tariff->TariffChangeIsAllowed(*newTariff, time(NULL));
607             if (message.empty())
608             {
609                 if (!u->GetProperties().tariffName.Set(m_ucr.tariffName.const_data(), m_currAdmin, m_login, m_store))
610                     return -1;
611                 u->ResetNextTariff();
612             }
613             else
614             {
615                 PluginLogger::get("conf_sg")("Tariff change is prohibited for user %s. %s", u->GetLogin().c_str(), message.c_str());
616             }
617         }
618         else
619         {
620             //WriteServLog("SetUser: Tariff %s not found", ud.conf.tariffName.c_str());
621             return -1;
622         }
623     }
624
625     if (!m_ucr.nextTariff.empty())
626     {
627         if (m_tariffs.FindByName(m_ucr.nextTariff.const_data()))
628         {
629             if (!u->GetProperties().nextTariff.Set(m_ucr.nextTariff.const_data(), m_currAdmin, m_login, m_store))
630                 return -1;
631         }
632         else
633         {
634             //WriteServLog("SetUser: Tariff %s not found", ud.conf.tariffName.c_str());
635             return -1;
636         }
637     }
638
639     auto up = u->GetProperties().up.get();
640     auto down = u->GetProperties().down.get();
641     int upCount = 0;
642     int downCount = 0;
643     for (int i = 0; i < DIR_NUM; i++)
644     {
645         if (!m_upr[i].empty())
646         {
647             up[i] = m_upr[i].data();
648             upCount++;
649         }
650         if (!m_downr[i].empty())
651         {
652             down[i] = m_downr[i].data();
653             downCount++;
654         }
655     }
656
657     if (upCount)
658         if (!u->GetProperties().up.Set(up, m_currAdmin, m_login, m_store))
659             return -1;
660
661     if (downCount)
662         if (!u->GetProperties().down.Set(down, m_currAdmin, m_login, m_store))
663             return -1;
664
665     u->WriteConf();
666     u->WriteStat();
667
668     return 0;
669 }
670
671 int DEL_USER::Start(void *, const char *el, const char **attr)
672 {
673     res = 0;
674     if (strcasecmp(el, m_tag.c_str()) == 0)
675     {
676         if (attr[0] == NULL || attr[1] == NULL)
677         {
678             //CreateAnswer("Parameters error!");
679             CreateAnswer();
680             return 0;
681         }
682
683         if (m_users.FindByName(attr[1], &u))
684         {
685             res = 1;
686             CreateAnswer();
687             return 0;
688         }
689         CreateAnswer();
690         return 0;
691     }
692     return -1;
693 }
694
695 int DEL_USER::End(void *, const char *el)
696 {
697     if (strcasecmp(el, m_tag.c_str()) == 0)
698     {
699         if (!res)
700             m_users.Del(u->GetLogin(), &m_currAdmin);
701
702         return 0;
703     }
704     return -1;
705 }
706
707 void DEL_USER::CreateAnswer()
708 {
709     if (res)
710         m_answer = "<" + m_tag + " value=\"error\" reason=\"User not found\"/>";
711     else
712         m_answer = "<" + m_tag + " value=\"ok\"/>";
713 }
714
715 int CHECK_USER::Start(void *, const char *el, const char **attr)
716 {
717     if (strcasecmp(el, m_tag.c_str()) == 0)
718     {
719         if (attr[0] == NULL || attr[1] == NULL ||
720             attr[2] == NULL || attr[3] == NULL)
721         {
722             CreateAnswer("Invalid parameters.");
723             printfd(__FILE__, "PARSER_CHECK_USER - attr err\n");
724             return 0;
725         }
726
727         ConstUserPtr user;
728         if (m_users.FindByName(attr[1], &user))
729         {
730             CreateAnswer("User not found.");
731             printfd(__FILE__, "PARSER_CHECK_USER - login err\n");
732             return 0;
733         }
734
735         if (strcmp(user->GetProperties().password.Get().c_str(), attr[3]))
736         {
737             CreateAnswer("Wrong password.");
738             printfd(__FILE__, "PARSER_CHECK_USER - passwd err\n");
739             return 0;
740         }
741
742         CreateAnswer(NULL);
743         return 0;
744     }
745     return -1;
746 }
747
748 int CHECK_USER::End(void *, const char *el)
749 {
750     if (strcasecmp(el, m_tag.c_str()) == 0)
751         return 0;
752     return -1;
753 }
754
755 void CHECK_USER::CreateAnswer(const char * error)
756 {
757     if (error)
758         m_answer = "<" + m_tag + " value=\"Err\" reason=\"" + error + "\"/>";
759     else
760         m_answer = "<" + m_tag + " value=\"Ok\"/>";
761 }