database.cpp 12 KB
Newer Older
1 2 3
#include "biboumi.h"
#ifdef USE_DATABASE

4 5
#include <database/select_query.hpp>
#include <database/save.hpp>
6
#include <database/database.hpp>
7
#include <utils/get_first_non_empty.hpp>
8
#include <utils/time.hpp>
9
#include <utils/uuid.hpp>
10

11
#include <config/config.hpp>
louiz’'s avatar
louiz’ committed
12 13
#include <database/sqlite3_engine.hpp>
#include <database/postgresql_engine.hpp>
14

louiz’'s avatar
louiz’ committed
15
#include <database/engine.hpp>
16 17
#include <database/index.hpp>

louiz’'s avatar
louiz’ committed
18 19 20 21 22 23 24
#include <memory>

std::unique_ptr<DatabaseEngine> Database::db;
Database::MucLogLineTable Database::muc_log_lines("muclogline_");
Database::GlobalOptionsTable Database::global_options("globaloptions_");
Database::IrcServerOptionsTable Database::irc_server_options("ircserveroptions_");
Database::IrcChannelOptionsTable Database::irc_channel_options("ircchanneloptions_");
25
Database::RosterTable Database::roster("roster");
26
Database::AfterConnectionCommandsTable Database::after_connection_commands("after_connection_commands_");
27
std::map<Database::CacheKey, Database::EncodingIn::real_type> Database::encoding_in_cache{};
28

29 30 31
Database::GlobalPersistent::GlobalPersistent():
    Column<bool>{Config::get_bool("persistent_by_default", false)}
{}
32

33
void Database::open(const std::string& filename)
34
{
35 36 37
  // Try to open the specified database.
  // Close and replace the previous database pointer if it succeeded. If it did
  // not, just leave things untouched
louiz’'s avatar
louiz’ committed
38 39
  std::unique_ptr<DatabaseEngine> new_db;
  static const auto psql_prefix = "postgresql://"s;
40 41 42
  static const auto psql_prefix2 = "postgres://"s;
  if ((filename.substr(0, psql_prefix.size()) == psql_prefix) ||
      (filename.substr(0, psql_prefix2.size()) == psql_prefix2))
43
    new_db = PostgresqlEngine::open(filename);
louiz’'s avatar
louiz’ committed
44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
  else
    new_db = Sqlite3Engine::open(filename);
  if (!new_db)
    return;
  Database::db = std::move(new_db);
  Database::muc_log_lines.create(*Database::db);
  Database::muc_log_lines.upgrade(*Database::db);
  Database::global_options.create(*Database::db);
  Database::global_options.upgrade(*Database::db);
  Database::irc_server_options.create(*Database::db);
  Database::irc_server_options.upgrade(*Database::db);
  Database::irc_channel_options.create(*Database::db);
  Database::irc_channel_options.upgrade(*Database::db);
  Database::roster.create(*Database::db);
  Database::roster.upgrade(*Database::db);
59 60
  Database::after_connection_commands.create(*Database::db);
  Database::after_connection_commands.upgrade(*Database::db);
louiz’'s avatar
louiz’ committed
61
  create_index<Database::Owner, Database::IrcChanName, Database::IrcServerName>(*Database::db, "archive_index", Database::muc_log_lines.get_name());
62 63
}

64

65
Database::GlobalOptions Database::get_global_options(const std::string& owner)
66
{
67
  auto request = select(Database::global_options);
68
  request.where() << Owner{} << "=" << owner;
69

louiz’'s avatar
louiz’ committed
70
  auto result = request.execute(*Database::db);
71
  if (result.size() == 1)
72 73 74
    return result.front();
  Database::GlobalOptions options{Database::global_options.get_name()};
  options.col<Owner>() = owner;
75
  return options;
76 77
}

78
Database::IrcServerOptions Database::get_irc_server_options(const std::string& owner, const std::string& server)
79
{
80
  auto request = select(Database::irc_server_options);
81
  request.where() << Owner{} << "=" << owner << " and " << Server{} << "=" << server;
82

louiz’'s avatar
louiz’ committed
83
  auto result = request.execute(*Database::db);
84
  if (result.size() == 1)
85 86 87 88
    return result.front();
  Database::IrcServerOptions options{Database::irc_server_options.get_name()};
  options.col<Owner>() = owner;
  options.col<Server>() = server;
89
  return options;
90 91
}

92 93 94 95 96
Database::AfterConnectionCommands Database::get_after_connection_commands(const IrcServerOptions& server_options)
{
  const auto id = server_options.col<Id>();
  if (id == Id::unset_value)
    return {};
97
  auto request = select(Database::after_connection_commands);
98 99 100 101 102 103 104 105 106
  request.where() << ForeignKey{} << "=" << id;
  return request.execute(*Database::db);
}

void Database::set_after_connection_commands(const Database::IrcServerOptions& server_options, Database::AfterConnectionCommands& commands)
{
  const auto id = server_options.col<Id>();
  if (id == Id::unset_value)
    return ;
107 108

  Transaction transaction;
109 110 111 112 113 114 115
  auto query = Database::after_connection_commands.del();
  query.where() << ForeignKey{} << "=" << id;
  query.execute(*Database::db);

  for (auto& command: commands)
    {
      command.col<ForeignKey>() = server_options.col<Id>();
116
      save(command, *Database::db);
117 118 119
    }
}

120
Database::IrcChannelOptions Database::get_irc_channel_options(const std::string& owner, const std::string& server, const std::string& channel)
121
{
122
  auto request = select(Database::irc_channel_options);
123 124 125
  request.where() << Owner{} << "=" << owner <<\
          " and " << Server{} << "=" << server <<\
          " and " << Channel{} << "=" << channel;
louiz’'s avatar
louiz’ committed
126
  auto result = request.execute(*Database::db);
127
  if (result.size() == 1)
128 129 130 131 132
    return result.front();
  Database::IrcChannelOptions options{Database::irc_channel_options.get_name()};
  options.col<Owner>() = owner;
  options.col<Server>() = server;
  options.col<Channel>() = channel;
133
  return options;
134 135
}

136 137
Database::IrcChannelOptions Database::get_irc_channel_options_with_server_default(const std::string& owner, const std::string& server,
                                                                                  const std::string& channel)
138 139 140
{
  auto coptions = Database::get_irc_channel_options(owner, server, channel);
  auto soptions = Database::get_irc_server_options(owner, server);
141

142 143 144 145
  coptions.col<EncodingIn>() = get_first_non_empty(coptions.col<EncodingIn>(),
                                                   soptions.col<EncodingIn>());
  coptions.col<EncodingOut>() = get_first_non_empty(coptions.col<EncodingOut>(),
                                                    soptions.col<EncodingOut>());
146

147 148
  coptions.col<MaxHistoryLength>() = get_first_non_empty(coptions.col<MaxHistoryLength>(),
                                                         soptions.col<MaxHistoryLength>());
149 150 151 152

  return coptions;
}

153
Database::IrcChannelOptions Database::get_irc_channel_options_with_server_and_global_default(const std::string& owner, const std::string& server, const std::string& channel)
154 155 156 157 158
{
  auto coptions = Database::get_irc_channel_options(owner, server, channel);
  auto soptions = Database::get_irc_server_options(owner, server);
  auto goptions = Database::get_global_options(owner);

159 160
  coptions.col<EncodingIn>() = get_first_non_empty(coptions.col<EncodingIn>(),
                                                   soptions.col<EncodingIn>());
161

162 163 164 165 166 167
  coptions.col<EncodingOut>() = get_first_non_empty(coptions.col<EncodingOut>(),
                                                    soptions.col<EncodingOut>());

  coptions.col<MaxHistoryLength>() = get_first_non_empty(coptions.col<MaxHistoryLength>(),
                                                         soptions.col<MaxHistoryLength>(),
                                                         goptions.col<MaxHistoryLength>());
168

169 170 171
  return coptions;
}

172
std::string Database::store_muc_message(const std::string& owner, const std::string& chan_name,
173
                                        const std::string& server_name, Database::time_point date,
174
                                        const std::string& body, const std::string& nick)
175
{
176
  auto line = Database::muc_log_lines.row();
177

178 179
  auto uuid = Database::gen_uuid();

180 181 182 183
  line.col<Uuid>() = uuid;
  line.col<Owner>() = owner;
  line.col<IrcChanName>() = chan_name;
  line.col<IrcServerName>() = server_name;
184
  line.col<Date>() = std::chrono::duration_cast<std::chrono::seconds>(date.time_since_epoch()).count();
185 186
  line.col<Body>() = body;
  line.col<Nick>() = nick;
187

188
  save(line, *Database::db);
189 190

  return uuid;
191 192
}

193
std::vector<Database::MucLogLine> Database::get_muc_logs(const std::string& owner, const std::string& chan_name, const std::string& server,
194
                                                   int limit, const std::string& start, const std::string& end, const Id::real_type reference_record_id, Database::Paging paging)
195
{
196 197 198
  if (limit == 0)
    return {};

199
  auto request = select(Database::muc_log_lines);
200 201 202
  request.where() << Database::Owner{} << "=" << owner << \
          " and " << Database::IrcChanName{} << "=" << chan_name << \
          " and " << Database::IrcServerName{} << "=" << server;
203 204 205 206 207

  if (!start.empty())
    {
      const auto start_time = utils::parse_datetime(start);
      if (start_time != -1)
208
        request << " and " << Database::Date{} << ">=" << start_time;
209 210 211 212 213
    }
  if (!end.empty())
    {
      const auto end_time = utils::parse_datetime(end);
      if (end_time != -1)
214
        request << " and " << Database::Date{} << "<=" << end_time;
215
    }
216
  if (reference_record_id != Id::unset_value)
217
    {
218
      request << " and " << Id{};
219 220 221 222
      if (paging == Database::Paging::first)
        request << ">";
      else
        request << "<";
223
      request << reference_record_id;
224
    }
225

226
  if (paging == Database::Paging::first)
227
    request.order_by() << Database::Date{} << " ASC, " << Id{} << " ASC ";
228
  else
229
    request.order_by() << Database::Date{} << " DESC, " << Id{} << " DESC ";
230 231 232 233

  if (limit >= 0)
    request.limit() << limit;

louiz’'s avatar
louiz’ committed
234
  auto result = request.execute(*Database::db);
235 236 237 238 239

  if (paging == Database::Paging::first)
    return result;
  else
    return {result.crbegin(), result.crend()};
240 241
}

242 243 244
Database::MucLogLine Database::get_muc_log(const std::string& owner, const std::string& chan_name, const std::string& server,
                                           const std::string& uuid, const std::string& start, const std::string& end)
{
245
  auto request = select(Database::muc_log_lines);
246 247 248 249 250 251 252 253 254
  request.where() << Database::Owner{} << "=" << owner << \
          " and " << Database::IrcChanName{} << "=" << chan_name << \
          " and " << Database::IrcServerName{} << "=" << server << \
          " and " << Database::Uuid{} << "=" << uuid;

  if (!start.empty())
    {
      const auto start_time = utils::parse_datetime(start);
      if (start_time != -1)
255
        request << " and " << Database::Date{} << ">=" << start_time;
256 257 258 259 260
    }
  if (!end.empty())
    {
      const auto end_time = utils::parse_datetime(end);
      if (end_time != -1)
261
        request << " and " << Database::Date{} << "<=" << end_time;
262 263 264 265 266 267 268 269 270
    }

  auto result = request.execute(*Database::db);

  if (result.empty())
    throw Database::RecordNotFound{};
  return result.front();
}

271 272 273 274 275 276 277
void Database::add_roster_item(const std::string& local, const std::string& remote)
{
  auto roster_item = Database::roster.row();

  roster_item.col<Database::LocalJid>() = local;
  roster_item.col<Database::RemoteJid>() = remote;

278
  save(roster_item, *Database::db);
279 280 281 282 283 284 285 286
}

void Database::delete_roster_item(const std::string& local, const std::string& remote)
{
  Query query("DELETE FROM "s + Database::roster.get_name());
  query << " WHERE " << Database::RemoteJid{} << "=" << remote << \
           " AND " << Database::LocalJid{} << "=" << local;

louiz’'s avatar
louiz’ committed
287
//  query.execute(*Database::db);
288 289 290 291
}

bool Database::has_roster_item(const std::string& local, const std::string& remote)
{
292
  auto query = select(Database::roster);
293 294 295
  query.where() << Database::LocalJid{} << "=" << local << \
        " and " << Database::RemoteJid{} << "=" << remote;

louiz’'s avatar
louiz’ committed
296
  auto res = query.execute(*Database::db);
297 298 299 300 301 302

  return !res.empty();
}

std::vector<Database::RosterItem> Database::get_contact_list(const std::string& local)
{
303
  auto query = select(Database::roster);
304 305
  query.where() << Database::LocalJid{} << "=" << local;

louiz’'s avatar
louiz’ committed
306
  return query.execute(*Database::db);
307 308 309 310
}

std::vector<Database::RosterItem> Database::get_full_roster()
{
311
  auto query = select(Database::roster);
312

louiz’'s avatar
louiz’ committed
313
  return query.execute(*Database::db);
314 315
}

316 317
void Database::close()
{
318
  Database::db = nullptr;
319
}
320

321 322
std::string Database::gen_uuid()
{
323
  return utils::gen_uuid();
324 325
}

326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344
Transaction::Transaction()
{
  const auto result = Database::raw_exec("BEGIN");
  if (std::get<bool>(result) == false)
    log_error("Failed to create SQL transaction: ", std::get<std::string>(result));
  else
    this->success = true;

}

Transaction::~Transaction()
{
  if (this->success)
    {
      const auto result = Database::raw_exec("END");
      if (std::get<bool>(result) == false)
        log_error("Failed to end SQL transaction: ", std::get<std::string>(result));
    }
}
345
#endif