biboumi_adhoc_commands.cpp 29.5 KB
Newer Older
1
#include <xmpp/biboumi_adhoc_commands.hpp>
2
#include <xmpp/biboumi_component.hpp>
3 4
#include <utils/scopeguard.hpp>
#include <bridge/bridge.hpp>
5
#include <config/config.hpp>
6
#include <utils/string.hpp>
7
#include <utils/split.hpp>
8
#include <xmpp/jid.hpp>
louiz’'s avatar
louiz’ committed
9
#include <algorithm>
louiz’'s avatar
louiz’ committed
10
#include <sstream>
11
#include <iomanip>
12 13 14 15 16 17 18

#include <biboumi.h>

#ifdef USE_DATABASE
#include <database/database.hpp>
#endif

19 20 21 22
#ifndef HAS_PUT_TIME
#include <ctime>
#endif

23
using namespace std::string_literals;
24

25
void DisconnectUserStep1(XmppComponent& xmpp_component, AdhocSession&, XmlNode& command_node)
26
{
louiz’'s avatar
louiz’ committed
27
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
28

29
  XmlSubNode x(command_node, "jabber:x:data:x");
30
  x["type"] = "form";
31
  XmlSubNode title(x, "title");
32
  title.set_inner("Disconnect a user from the gateway");
33
  XmlSubNode instructions(x, "instructions");
34
  instructions.set_inner("Choose a user JID and a quit message");
35
  XmlSubNode jids_field(x, "field");
36 37 38
  jids_field["var"] = "jids";
  jids_field["type"] = "list-multi";
  jids_field["label"] = "The JIDs to disconnect";
39
  XmlSubNode required(jids_field, "required");
40
  for (Bridge* bridge: biboumi_component.get_bridges())
41
    {
42
      XmlSubNode option(jids_field, "option");
43
      option["label"] = bridge->get_jid();
44
      XmlSubNode value(option, "value");
45 46 47
      value.set_inner(bridge->get_jid());
    }

48
  XmlSubNode message_field(x, "field");
49 50 51
  message_field["var"] = "quit-message";
  message_field["type"] = "text-single";
  message_field["label"] = "Quit message";
52
  XmlSubNode message_value(message_field, "value");
53 54 55
  message_value.set_inner("Disconnected by admin");
}

56
void DisconnectUserStep2(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
57
{
louiz’'s avatar
louiz’ committed
58
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
59

60 61
  // Find out if the jids, and the quit message are provided in the form.
  std::string quit_message;
62
  const XmlNode* x = command_node.get_child("x", "jabber:x:data");
63 64
  if (x)
    {
65 66 67
      const XmlNode* message_field = nullptr;
      const XmlNode* jids_field = nullptr;
      for (const XmlNode* field: x->get_children("field", "jabber:x:data"))
68 69 70 71 72 73
        if (field->get_tag("var") == "jids")
          jids_field = field;
        else if (field->get_tag("var") == "quit-message")
          message_field = field;
      if (message_field)
        {
74
          const XmlNode* value = message_field->get_child("value", "jabber:x:data");
75 76 77 78 79 80
          if (value)
            quit_message = value->get_inner();
        }
      if (jids_field)
        {
          std::size_t num = 0;
81
          for (const XmlNode* value: jids_field->get_children("value", "jabber:x:data"))
82
            {
83
              Bridge* bridge = biboumi_component.find_user_bridge(value->get_inner());
84 85 86 87 88 89 90 91
              if (bridge)
                {
                  bridge->shutdown(quit_message);
                  num++;
                }
            }
          command_node.delete_all_children();

92
          XmlSubNode note(command_node, "note");
93 94 95 96 97 98 99
          note["type"] = "info";
          if (num == 0)
            note.set_inner("No user were disconnected.");
          else if (num == 1)
            note.set_inner("1 user has been disconnected.");
          else
            note.set_inner(std::to_string(num) + " users have been disconnected.");
100
          return;
101 102
        }
    }
103
  XmlSubNode error(command_node, ADHOC_NS":error");
104
  error["type"] = "modify";
105
  XmlSubNode condition(error, STANZA_NS":bad-request");
106 107
  session.terminate();
}
108 109

#ifdef USE_DATABASE
110 111 112 113 114 115 116 117

void ConfigureGlobalStep1(XmppComponent&, AdhocSession& session, XmlNode& command_node)
{
  const Jid owner(session.get_owner_jid());
  const Jid target(session.get_target_jid());

  auto options = Database::get_global_options(owner.bare());

118
  XmlSubNode x(command_node, "jabber:x:data:x");
119
  x["type"] = "form";
120
  XmlSubNode title(x, "title");
121
  title.set_inner("Configure some global default settings.");
122
  XmlSubNode instructions(x, "instructions");
123 124
  instructions.set_inner("Edit the form, to configure your global settings for the component.");

125
  XmlSubNode max_histo_length(x, "field");
126 127 128 129 130
  max_histo_length["var"] = "max_history_length";
  max_histo_length["type"] = "text-single";
  max_histo_length["label"] = "Max history length";
  max_histo_length["desc"] = "The maximum number of lines in the history that the server sends when joining a channel";

131 132
  {
    XmlSubNode value(max_histo_length, "value");
133
    value.set_inner(std::to_string(options.col<Database::MaxHistoryLength>()));
134
  }
135

136
  XmlSubNode record_history(x, "field");
137 138 139 140 141
  record_history["var"] = "record_history";
  record_history["type"] = "boolean";
  record_history["label"] = "Record history";
  record_history["desc"] = "Whether to save the messages into the database, or not";

142 143 144
  {
    XmlSubNode value(record_history, "value");
    value.set_name("value");
145
    if (options.col<Database::RecordHistory>())
146 147 148 149
      value.set_inner("true");
    else
      value.set_inner("false");
  }
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164

  XmlSubNode persistent(x, "field");
  persistent["var"] = "persistent";
  persistent["type"] = "boolean";
  persistent["label"] = "Make all channels persistent";
  persistent["desc"] = "If true, all channels will be persistent";

  {
    XmlSubNode value(persistent, "value");
    value.set_name("value");
    if (options.col<Database::Persistent>())
      value.set_inner("true");
    else
      value.set_inner("false");
  }
165 166
}

167
void ConfigureGlobalStep2(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
168
{
louiz’'s avatar
louiz’ committed
169
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
170

171 172 173 174 175 176 177 178 179 180 181
  const XmlNode* x = command_node.get_child("x", "jabber:x:data");
  if (x)
    {
      const Jid owner(session.get_owner_jid());
      auto options = Database::get_global_options(owner.bare());
      for (const XmlNode* field: x->get_children("field", "jabber:x:data"))
        {
          const XmlNode* value = field->get_child("value", "jabber:x:data");

          if (field->get_tag("var") == "max_history_length" &&
              value && !value->get_inner().empty())
182
            options.col<Database::MaxHistoryLength>() = atoi(value->get_inner().data());
183 184 185
          else if (field->get_tag("var") == "record_history" &&
                   value && !value->get_inner().empty())
            {
186
              options.col<Database::RecordHistory>() = to_bool(value->get_inner());
187 188
              Bridge* bridge = biboumi_component.find_user_bridge(owner.bare());
              if (bridge)
189
                bridge->set_record_history(options.col<Database::RecordHistory>());
190
            }
191 192 193
          else if (field->get_tag("var") == "persistent" &&
                   value)
            options.col<Database::Persistent>() = to_bool(value->get_inner());
194 195
        }

196
      options.save(Database::db);
197 198

      command_node.delete_all_children();
199
      XmlSubNode note(command_node, "note");
200 201 202 203
      note["type"] = "info";
      note.set_inner("Configuration successfully applied.");
      return;
    }
204
  XmlSubNode error(command_node, ADHOC_NS":error");
205
  error["type"] = "modify";
206
  XmlSubNode condition(error, STANZA_NS":bad-request");
207 208 209
  session.terminate();
}

210
void ConfigureIrcServerStep1(XmppComponent&, AdhocSession& session, XmlNode& command_node)
211 212 213
{
  const Jid owner(session.get_owner_jid());
  const Jid target(session.get_target_jid());
214 215 216
  std::string server_domain;
  if ((server_domain = Config::get("fixed_irc_server", "")).empty())
    server_domain = target.local;
217
  auto options = Database::get_irc_server_options(owner.local + "@" + owner.domain,
218
                                                  server_domain);
219

220
  XmlSubNode x(command_node, "jabber:x:data:x");
221
  x["type"] = "form";
222
  XmlSubNode title(x, "title");
223
  title.set_inner("Configure the IRC server "s + server_domain);
224
  XmlSubNode instructions(x, "instructions");
225
  instructions.set_inner("Edit the form, to configure the settings of the IRC server "s + server_domain);
226

227
  XmlSubNode ports(x, "field");
228 229 230 231
  ports["var"] = "ports";
  ports["type"] = "text-multi";
  ports["label"] = "Ports";
  ports["desc"] = "List of ports to try, without TLS. Defaults: 6667.";
232
  for (const auto& val: utils::split(options.col<Database::Ports>(), ';', false))
233
    {
234
      XmlSubNode ports_value(ports, "value");
235 236 237 238
      ports_value.set_inner(val);
    }

#ifdef BOTAN_FOUND
239
  XmlSubNode tls_ports(x, "field");
240 241 242 243
  tls_ports["var"] = "tls_ports";
  tls_ports["type"] = "text-multi";
  tls_ports["label"] = "TLS ports";
  tls_ports["desc"] = "List of ports to try, with TLS. Defaults: 6697, 6670.";
244
  for (const auto& val: utils::split(options.col<Database::TlsPorts>(), ';', false))
245
    {
246
      XmlSubNode tls_ports_value(tls_ports, "value");
247 248
      tls_ports_value.set_inner(val);
    }
249

250
  XmlSubNode verify_cert(x, "field");
251 252 253 254
  verify_cert["var"] = "verify_cert";
  verify_cert["type"] = "boolean";
  verify_cert["label"] = "Verify certificate";
  verify_cert["desc"] = "Whether or not to abort the connection if the server’s TLS certificate is invalid";
255
  XmlSubNode verify_cert_value(verify_cert, "value");
256
  if (options.col<Database::VerifyCert>())
257 258 259
    verify_cert_value.set_inner("true");
  else
    verify_cert_value.set_inner("false");
260

261
  XmlSubNode fingerprint(x, "field");
262 263 264
  fingerprint["var"] = "fingerprint";
  fingerprint["type"] = "text-single";
  fingerprint["label"] = "SHA-1 fingerprint of the TLS certificate to trust.";
265
  if (!options.col<Database::TrustedFingerprint>().empty())
266
    {
267
      XmlSubNode fingerprint_value(fingerprint, "value");
268
      fingerprint_value.set_inner(options.col<Database::TrustedFingerprint>());
269
    }
270
#endif
271

272
  XmlSubNode pass(x, "field");
273 274
  pass["var"] = "pass";
  pass["type"] = "text-private";
louiz’'s avatar
louiz’ committed
275 276
  pass["label"] = "Server password";
  pass["desc"] = "Will be used in a PASS command when connecting";
277
  if (!options.col<Database::Pass>().empty())
278
    {
279
      XmlSubNode pass_value(pass, "value");
280
      pass_value.set_inner(options.col<Database::Pass>());
281 282
    }

283
  XmlSubNode after_cnt_cmd(x, "field");
284 285 286 287
  after_cnt_cmd["var"] = "after_connect_command";
  after_cnt_cmd["type"] = "text-single";
  after_cnt_cmd["desc"] = "Custom IRC command sent after the connection is established with the server.";
  after_cnt_cmd["label"] = "After-connection IRC command";
288
  if (!options.col<Database::AfterConnectionCommand>().empty())
289
    {
290
      XmlSubNode after_cnt_cmd_value(after_cnt_cmd, "value");
291
      after_cnt_cmd_value.set_inner(options.col<Database::AfterConnectionCommand>());
292 293
    }

294
  if (Config::get("realname_customization", "true") == "true")
295
    {
296
      XmlSubNode username(x, "field");
297 298 299
      username["var"] = "username";
      username["type"] = "text-single";
      username["label"] = "Username";
300
      if (!options.col<Database::Username>().empty())
301
        {
302
          XmlSubNode username_value(username, "value");
303
          username_value.set_inner(options.col<Database::Username>());
304 305
        }

306
      XmlSubNode realname(x, "field");
307 308 309
      realname["var"] = "realname";
      realname["type"] = "text-single";
      realname["label"] = "Realname";
310
      if (!options.col<Database::Realname>().empty())
311
        {
312
          XmlSubNode realname_value(realname, "value");
313
          realname_value.set_inner(options.col<Database::Realname>());
314
        }
315 316
    }

317
  XmlSubNode encoding_out(x, "field");
318 319 320 321
  encoding_out["var"] = "encoding_out";
  encoding_out["type"] = "text-single";
  encoding_out["desc"] = "The encoding used when sending messages to the IRC server.";
  encoding_out["label"] = "Out encoding";
322
  if (!options.col<Database::EncodingOut>().empty())
323
    {
324
      XmlSubNode encoding_out_value(encoding_out, "value");
325
      encoding_out_value.set_inner(options.col<Database::EncodingOut>());
326 327
    }

328
  XmlSubNode encoding_in(x, "field");
329 330 331 332
  encoding_in["var"] = "encoding_in";
  encoding_in["type"] = "text-single";
  encoding_in["desc"] = "The encoding used to decode message received from the IRC server.";
  encoding_in["label"] = "In encoding";
333
  if (!options.col<Database::EncodingIn>().empty())
334
    {
335
      XmlSubNode encoding_in_value(encoding_in, "value");
336
      encoding_in_value.set_inner(options.col<Database::EncodingIn>());
337
    }
338 339
}

340
void ConfigureIrcServerStep2(XmppComponent&, AdhocSession& session, XmlNode& command_node)
341 342 343 344 345 346
{
  const XmlNode* x = command_node.get_child("x", "jabber:x:data");
  if (x)
    {
      const Jid owner(session.get_owner_jid());
      const Jid target(session.get_target_jid());
347 348 349
      std::string server_domain;
      if ((server_domain = Config::get("fixed_irc_server", "")).empty())
        server_domain = target.local;
350
      auto options = Database::get_irc_server_options(owner.local + "@" + owner.domain,
351
                                                      server_domain);
352 353 354
      for (const XmlNode* field: x->get_children("field", "jabber:x:data"))
        {
          const XmlNode* value = field->get_child("value", "jabber:x:data");
355 356 357 358 359 360
          const std::vector<const XmlNode*> values = field->get_children("value", "jabber:x:data");
          if (field->get_tag("var") == "ports")
            {
              std::string ports;
              for (const auto& val: values)
                ports += val->get_inner() + ";";
361
              options.col<Database::Ports>() = ports;
362 363 364 365 366 367 368 369
            }

#ifdef BOTAN_FOUND
          else if (field->get_tag("var") == "tls_ports")
            {
              std::string ports;
              for (const auto& val: values)
                ports += val->get_inner() + ";";
370
              options.col<Database::TlsPorts>() = ports;
371
            }
372

373
          else if (field->get_tag("var") == "verify_cert" && value
374
                   && !value->get_inner().empty())
375 376
            {
              auto val = to_bool(value->get_inner());
377
              options.col<Database::VerifyCert>() = val;
378
            }
379 380 381 382

          else if (field->get_tag("var") == "fingerprint" && value &&
                   !value->get_inner().empty())
            {
383
              options.col<Database::TrustedFingerprint>() = value->get_inner();
384 385
            }

386
#endif // BOTAN_FOUND
387 388

          else if (field->get_tag("var") == "pass" &&
389
                   value && !value->get_inner().empty())
390
            options.col<Database::Pass>() = value->get_inner();
391 392 393

          else if (field->get_tag("var") == "after_connect_command" &&
                   value && !value->get_inner().empty())
394
            options.col<Database::AfterConnectionCommand>() = value->get_inner();
395 396 397

          else if (field->get_tag("var") == "username" &&
                   value && !value->get_inner().empty())
398 399 400
            {
              auto username = value->get_inner();
              // The username must not contain spaces
401
              std::replace(username.begin(), username.end(), ' ', '_');
402
              options.col<Database::Username>() = username;
403
            }
404 405 406

          else if (field->get_tag("var") == "realname" &&
                   value && !value->get_inner().empty())
407
            options.col<Database::Realname>() = value->get_inner();
408 409 410

          else if (field->get_tag("var") == "encoding_out" &&
                   value && !value->get_inner().empty())
411
            options.col<Database::EncodingOut>() = value->get_inner();
412 413 414

          else if (field->get_tag("var") == "encoding_in" &&
                   value && !value->get_inner().empty())
415
            options.col<Database::EncodingIn>() = value->get_inner();
416

417 418
        }

419
      options.save(Database::db);
420 421

      command_node.delete_all_children();
422
      XmlSubNode note(command_node, "note");
423 424 425 426
      note["type"] = "info";
      note.set_inner("Configuration successfully applied.");
      return;
    }
427
  XmlSubNode error(command_node, ADHOC_NS":error");
428
  error["type"] = "modify";
429
  XmlSubNode condition(error, STANZA_NS":bad-request");
430 431 432 433 434 435 436
  session.terminate();
}

void ConfigureIrcChannelStep1(XmppComponent&, AdhocSession& session, XmlNode& command_node)
{
  const Jid owner(session.get_owner_jid());
  const Jid target(session.get_target_jid());
437 438 439 440 441 442

  insert_irc_channel_configuration_form(command_node, owner, target);
}

void insert_irc_channel_configuration_form(XmlNode& node, const Jid& requester, const Jid& target)
{
443
  const Iid iid(target.local, {});
444 445

  auto options = Database::get_irc_channel_options_with_server_default(requester.local + "@" + requester.domain,
446 447
                                                                       iid.get_server(), iid.get_local());

448
  XmlSubNode x(node, "jabber:x:data:x");
449
  x["type"] = "form";
450
  XmlSubNode title(x, "title");
451
  title.set_inner("Configure the IRC channel "s + iid.get_local() + " on server "s + iid.get_server());
452
  XmlSubNode instructions(x, "instructions");
453 454
  instructions.set_inner("Edit the form, to configure the settings of the IRC channel "s + iid.get_local());

455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474
  XmlSubNode record_history(x, "field");
  record_history["var"] = "record_history";
  record_history["type"] = "list-single";
  record_history["label"] = "Record history for this channel";
  record_history["desc"] = "If unset, the value is the one configured globally";

  {
    // Value selected by default
    XmlSubNode value(record_history, "value");
    value.set_inner(options.col<Database::RecordHistoryOptional>().to_string());
  }
  // All three possible values
  for (const auto& val: {"unset", "true", "false"})
  {
    XmlSubNode option(record_history, "option");
    option["label"] = val;
    XmlSubNode value(option, "value");
    value.set_inner(val);
  }

475
  XmlSubNode encoding_out(x, "field");
476 477 478 479
  encoding_out["var"] = "encoding_out";
  encoding_out["type"] = "text-single";
  encoding_out["desc"] = "The encoding used when sending messages to the IRC server. Defaults to the server's “out encoding” if unset for the channel";
  encoding_out["label"] = "Out encoding";
480
  if (!options.col<Database::EncodingOut>().empty())
481
    {
482
      XmlSubNode encoding_out_value(encoding_out, "value");
483
      encoding_out_value.set_inner(options.col<Database::EncodingOut>());
484 485
    }

486
  XmlSubNode encoding_in(x, "field");
487 488 489 490
  encoding_in["var"] = "encoding_in";
  encoding_in["type"] = "text-single";
  encoding_in["desc"] = "The encoding used to decode message received from the IRC server. Defaults to the server's “in encoding” if unset for the channel";
  encoding_in["label"] = "In encoding";
491
  if (!options.col<Database::EncodingIn>().empty())
492
    {
493
      XmlSubNode encoding_in_value(encoding_in, "value");
494
      encoding_in_value.set_inner(options.col<Database::EncodingIn>());
495
    }
louiz’'s avatar
louiz’ committed
496 497 498 499 500 501 502 503 504

  XmlSubNode persistent(x, "field");
  persistent["var"] = "persistent";
  persistent["type"] = "boolean";
  persistent["desc"] = "If set to true, when all XMPP clients have left this channel, biboumi will stay idle in it, without sending a PART command.";
  persistent["label"] = "Persistent";
  {
    XmlSubNode value(persistent, "value");
    value.set_name("value");
505
    if (options.col<Database::Persistent>())
louiz’'s avatar
louiz’ committed
506 507 508 509
      value.set_inner("true");
    else
      value.set_inner("false");
  }
510 511
}

512
void ConfigureIrcChannelStep2(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
513
{
514 515 516
  const Jid owner(session.get_owner_jid());
  const Jid target(session.get_target_jid());

517
  if (handle_irc_channel_configuration_form(xmpp_component, command_node, owner, target))
518 519 520 521 522 523 524 525 526 527 528 529 530 531 532
    {
      command_node.delete_all_children();
      XmlSubNode note(command_node, "note");
      note["type"] = "info";
      note.set_inner("Configuration successfully applied.");
    }
  else
    {
      XmlSubNode error(command_node, ADHOC_NS":error");
      error["type"] = "modify";
      XmlSubNode condition(error, STANZA_NS":bad-request");
      session.terminate();
    }
}

533
bool handle_irc_channel_configuration_form(XmppComponent& xmpp_component, const XmlNode& node, const Jid& requester, const Jid& target)
534 535
{
  const XmlNode* x = node.get_child("x", "jabber:x:data");
536 537
  if (x)
    {
538
      if (x->get_tag("type") == "submit")
539
        {
540
          const Iid iid(target.local, {});
541
          auto options = Database::get_irc_channel_options(requester.bare(),
542 543 544 545
                                                           iid.get_server(), iid.get_local());
          for (const XmlNode *field: x->get_children("field", "jabber:x:data"))
            {
              const XmlNode *value = field->get_child("value", "jabber:x:data");
546

547 548
              if (field->get_tag("var") == "encoding_out" &&
                  value && !value->get_inner().empty())
549
                options.col<Database::EncodingOut>() = value->get_inner();
louiz’'s avatar
louiz’ committed
550

551 552
              else if (field->get_tag("var") == "encoding_in" &&
                       value && !value->get_inner().empty())
553
                options.col<Database::EncodingIn>() = value->get_inner();
554

555 556
              else if (field->get_tag("var") == "persistent" &&
                       value)
557
                options.col<Database::Persistent>() = to_bool(value->get_inner());
558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582
              else if (field->get_tag("var") == "record_history" &&
                       value && !value->get_inner().empty())
                {
                  OptionalBool& database_value = options.col<Database::RecordHistoryOptional>();
                  if (value->get_inner() == "true")
                    database_value.set_value(true);
                  else if (value->get_inner() == "false")
                    database_value.set_value(false);
                  else
                    database_value.unset();
                  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
                  Bridge* bridge = biboumi_component.find_user_bridge(requester.bare());
                  if (bridge)
                    {
                      if (database_value.is_set)
                        bridge->set_record_history(database_value.value);
                      else
                        { // It is unset, we need to fetch the Global option, to
                          // know if it’s enabled or not
                          auto g_options = Database::get_global_options(requester.bare());
                          bridge->set_record_history(g_options.col<Database::RecordHistory>());
                        }
                    }
                }

583
            }
584

585
          options.save(Database::db);
586 587
        }
      return true;
588
    }
589
  return false;
590 591
}
#endif  // USE_DATABASE
592

593
void DisconnectUserFromServerStep1(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
594 595 596 597 598 599 600 601 602 603
{
  const Jid owner(session.get_owner_jid());
  if (owner.bare() != Config::get("admin", ""))
    { // A non-admin is not allowed to disconnect other users, only
      // him/herself, so we just skip this step
      auto next_step = session.get_next_step();
      next_step(xmpp_component, session, command_node);
    }
  else
    { // Send a form to select the user to disconnect
louiz’'s avatar
louiz’ committed
604
      auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
605

606
      XmlSubNode x(command_node, "jabber:x:data:x");
607
      x["type"] = "form";
608
      XmlSubNode title(x, "title");
609
      title.set_inner("Disconnect a user from selected IRC servers");
610
      XmlSubNode instructions(x, "instructions");
611
      instructions.set_inner("Choose a user JID");
612
      XmlSubNode jids_field(x, "field");
613 614 615
      jids_field["var"] = "jid";
      jids_field["type"] = "list-single";
      jids_field["label"] = "The JID to disconnect";
616
      XmlSubNode required(jids_field, "required");
617
      for (Bridge* bridge: biboumi_component.get_bridges())
618
        {
619
          XmlSubNode option(jids_field, "option");
620
          option["label"] = bridge->get_jid();
621
          XmlSubNode value(option, "value");
622 623 624 625 626
          value.set_inner(bridge->get_jid());
        }
    }
}

627
void DisconnectUserFromServerStep2(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648
{
  // If no JID is contained in the command node, it means we skipped the
  // previous stage, and the jid to disconnect is the executor's jid
  std::string jid_to_disconnect = session.get_owner_jid();

  if (const XmlNode* x = command_node.get_child("x", "jabber:x:data"))
    {
      for (const XmlNode* field: x->get_children("field", "jabber:x:data"))
        if (field->get_tag("var") == "jid")
          {
            if (const XmlNode* value = field->get_child("value", "jabber:x:data"))
              jid_to_disconnect = value->get_inner();
          }
    }

  // Save that JID for the last step
  session.vars["jid"] = jid_to_disconnect;

  // Send a data form to let the user choose which server to disconnect the
  // user from
  command_node.delete_all_children();
louiz’'s avatar
louiz’ committed
649
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
650

651
  XmlSubNode x(command_node, "jabber:x:data:x");
652
  x["type"] = "form";
653
  XmlSubNode title(x, "title");
654
  title.set_inner("Disconnect a user from selected IRC servers");
655
  XmlSubNode instructions(x, "instructions");
656
  instructions.set_inner("Choose one or more servers to disconnect this JID from");
657
  XmlSubNode jids_field(x, "field");
658 659 660
  jids_field["var"] = "irc-servers";
  jids_field["type"] = "list-multi";
  jids_field["label"] = "The servers to disconnect from";
661
  XmlSubNode required(jids_field, "required");
662
  Bridge* bridge = biboumi_component.find_user_bridge(jid_to_disconnect);
663 664 665

  if (!bridge || bridge->get_irc_clients().empty())
    {
666
      XmlSubNode note(command_node, "note");
667 668 669 670 671 672 673 674
      note["type"] = "info";
      note.set_inner("User "s + jid_to_disconnect + " is not connected to any IRC server.");
      session.terminate();
      return ;
    }

  for (const auto& pair: bridge->get_irc_clients())
    {
675
      XmlSubNode option(jids_field, "option");
676
      option["label"] = pair.first;
677
      XmlSubNode value(option, "value");
678 679 680
      value.set_inner(pair.first);
    }

681
  XmlSubNode message_field(x, "field");
682 683 684
  message_field["var"] = "quit-message";
  message_field["type"] = "text-single";
  message_field["label"] = "Quit message";
685
  XmlSubNode message_value(message_field, "value");
686 687 688
  message_value.set_inner("Killed by admin");
}

689
void DisconnectUserFromServerStep3(XmppComponent& xmpp_component, AdhocSession& session, XmlNode& command_node)
690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713
{
  const auto it = session.vars.find("jid");
  if (it == session.vars.end())
    return ;
  const auto jid_to_disconnect = it->second;

  std::vector<std::string> servers;
  std::string quit_message;

  if (const XmlNode* x = command_node.get_child("x", "jabber:x:data"))
    {
      for (const XmlNode* field: x->get_children("field", "jabber:x:data"))
        {
          if (field->get_tag("var") == "irc-servers")
            {
              for (const XmlNode* value: field->get_children("value", "jabber:x:data"))
                servers.push_back(value->get_inner());
            }
          else if (field->get_tag("var") == "quit-message")
            if (const XmlNode* value = field->get_child("value", "jabber:x:data"))
              quit_message = value->get_inner();
        }
    }

louiz’'s avatar
louiz’ committed
714
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(xmpp_component);
715
  Bridge* bridge = biboumi_component.find_user_bridge(jid_to_disconnect);
716 717 718 719 720 721 722 723 724 725 726 727 728 729 730
  auto& clients = bridge->get_irc_clients();

  std::size_t number = 0;

  for (const auto& hostname: servers)
    {
      auto it = clients.find(hostname);
      if (it != clients.end())
        {
          it->second->on_error({"ERROR", {quit_message}});
          clients.erase(it);
          number++;
        }
    }
  command_node.delete_all_children();
731
  XmlSubNode note(command_node, "note");
732 733 734 735 736 737 738
  note["type"] = "info";
  std::string msg = jid_to_disconnect + " was disconnected from " + std::to_string(number) + " IRC server";
  if (number > 1)
    msg += "s";
  msg += ".";
  note.set_inner(msg);
}
739 740 741

void GetIrcConnectionInfoStep1(XmppComponent& component, AdhocSession& session, XmlNode& command_node)
{
louiz’'s avatar
louiz’ committed
742
  auto& biboumi_component = dynamic_cast<BiboumiComponent&>(component);
743 744 745 746 747 748 749 750 751 752

  const Jid owner(session.get_owner_jid());
  const Jid target(session.get_target_jid());

  std::string message{};

  // As the function is exited, set the message in the response.
  utils::ScopeGuard sg([&message, &command_node]()
                       {
                         command_node.delete_all_children();
753
                         XmlSubNode note(command_node, "note");
754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780
                         note["type"] = "info";
                         note.set_inner(message);
                       });

  Bridge* bridge = biboumi_component.get_user_bridge(owner.bare());
  if (!bridge)
    {
      message = "You are not connected to anything.";
      return;
    }

  std::string hostname;
  if ((hostname = Config::get("fixed_irc_server", "")).empty())
    hostname = target.local;

  IrcClient* irc = bridge->find_irc_client(hostname);
  if (!irc || !irc->is_connected())
    {
      message = "You are not connected to the IRC server "s + hostname;
      return;
    }

  std::ostringstream ss;
  ss << "Connected to IRC server " << irc->get_hostname() << " on port " << irc->get_port();
  if (irc->is_using_tls())
    ss << " (using TLS)";
  const std::time_t now_c = std::chrono::system_clock::to_time_t(irc->connection_date);
781
#ifdef HAS_PUT_TIME
782
  ss << " since " << std::put_time(std::localtime(&now_c), "%F %T");
783 784 785 786 787 788 789
#else
  constexpr std::size_t timestamp_size{10 + 1 + 8 + 1};
  char buf[timestamp_size] = {};
  const auto res = std::strftime(buf, timestamp_size, "%F %T", std::localtime(&now_c));
  if (res > 0)
    ss << " since " << buf;
#endif
790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808
  ss << " (" << std::chrono::duration_cast<std::chrono::seconds>(std::chrono::system_clock::now() - irc->connection_date).count() << " seconds ago).";

  for (const auto& it: bridge->resources_in_chan)
    {
      const auto& channel_key = it.first;
      const auto& irc_hostname = std::get<1>(channel_key);
      const auto& resources = it.second;

      if (irc_hostname == irc->get_hostname() && !resources.empty())
        {
          const auto& channel_name = std::get<0>(channel_key);
          ss << "\n" << channel_name << " from " << resources.size() << " resource" << (resources.size() > 1 ? "s": "") << ": ";
          for (const auto& resource: resources)
            ss << resource << " ";
        }
    }

  message = ss.str();
}