Skip to content

Commit

Permalink
SQLite: Add SQLite DB backend support
Browse files Browse the repository at this point in the history
  • Loading branch information
insunaa committed Jan 6, 2024
1 parent 0de1bc1 commit 21f3ec9
Show file tree
Hide file tree
Showing 23 changed files with 732 additions and 34 deletions.
4 changes: 4 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,8 @@ endif()
if(UNIX AND (BUILD_GAME_SERVER OR BUILD_LOGIN_SERVER OR BUILD_EXTRACTORS))
if(POSTGRESQL)
find_package(PostgreSQL REQUIRED)
elseif(SQLITE)
find_package(SQLite3 REQUIRED)
else()
find_package(MySQL REQUIRED)
endif()
Expand Down Expand Up @@ -359,6 +361,8 @@ set(DEFINITIONS "")

if(POSTGRESQL)
set(DEFINITIONS ${DEFINITIONS} DO_POSTGRESQL)
elseif(SQLITE)
set(DEFINITIONS ${DEFINITIONS} DO_SQLITE)
else()
set(DEFINITIONS ${DEFINITIONS} DO_MYSQL)
endif()
Expand Down
2 changes: 1 addition & 1 deletion src/game/Accounts/AccountMgr.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ AccountOpResult AccountMgr::CreateAccount(std::string username, std::string pass
const char* v_hex = srp.GetVerifier().AsHexStr();

bool update_sv = LoginDatabase.PExecute(
"INSERT INTO account(username,v,s,joindate) VALUES('%s','%s','%s',NOW())",
"INSERT INTO account(username,v,s,joindate) VALUES('%s','%s','%s'," _NOW_ ")",
username.c_str(), v_hex, s_hex);

OPENSSL_free((void*)s_hex);
Expand Down
2 changes: 1 addition & 1 deletion src/game/BattleGround/BattleGround.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -730,7 +730,7 @@ void BattleGround::EndBattleGround(Team winner)
{
static SqlStatementID insPvPstatsBattleground;

SqlStatement stmt = CharacterDatabase.CreateStatement(insPvPstatsBattleground, "INSERT INTO pvpstats_battlegrounds (id, winner_team, bracket_id, type, date) VALUES (?, ?, ?, ?, NOW())");
SqlStatement stmt = CharacterDatabase.CreateStatement(insPvPstatsBattleground, "INSERT INTO pvpstats_battlegrounds (id, winner_team, bracket_id, type, date) VALUES (?, ?, ?, ?, " _NOW_ ")");

uint8 battleground_bracket = GetMinLevel() / 10;
uint8 battleground_type = (uint8)GetTypeId();
Expand Down
6 changes: 3 additions & 3 deletions src/game/Chat/Level2.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -837,7 +837,7 @@ bool ChatHandler::HandleGameObjectTargetCommand(char* args)
uint32 id;
if (ExtractUInt32(&cId, id))
{
queryResult = WorldDatabase.PQuery("SELECT guid, id, position_x, position_y, position_z, orientation, map, (POW(position_x - '%f', 2) + POW(position_y - '%f', 2) + POW(position_z - '%f', 2)) AS order_ FROM gameobject WHERE map = '%i' AND guid = '%u' ORDER BY order_ ASC LIMIT 1",
queryResult = WorldDatabase.PQuery("SELECT guid, id, position_x, position_y, position_z, orientation, map, (POW(position_x - %f, 2) + POW(position_y - %f, 2) + POW(position_z - %f, 2)) AS order_ FROM gameobject WHERE map = '%i' AND guid = '%u' ORDER BY order_ ASC LIMIT 1",
pl->GetPositionX(), pl->GetPositionY(), pl->GetPositionZ(), pl->GetMapId(), id);
}
else
Expand Down Expand Up @@ -1215,8 +1215,8 @@ bool ChatHandler::HandleGameObjectNearCommand(char* args)

Player* pl = m_session->GetPlayer();
auto queryResult = WorldDatabase.PQuery("SELECT guid, id, position_x, position_y, position_z, map, "
"(POW(position_x - '%f', 2) + POW(position_y - '%f', 2) + POW(position_z - '%f', 2)) AS order_ "
"FROM gameobject WHERE map='%u' AND (POW(position_x - '%f', 2) + POW(position_y - '%f', 2) + POW(position_z - '%f', 2)) <= '%f' ORDER BY order_",
"(POW(position_x - %f, 2) + POW(position_y - %f, 2) + POW(position_z - %f, 2)) AS order_ "
"FROM gameobject WHERE map='%u' AND (POW(position_x - %f, 2) + POW(position_y - %f, 2) + POW(position_z - %f, 2)) <= %f ORDER BY order_",
pl->GetPositionX(), pl->GetPositionY(), pl->GetPositionZ(),
pl->GetMapId(), pl->GetPositionX(), pl->GetPositionY(), pl->GetPositionZ(), distance * distance);

Expand Down
14 changes: 7 additions & 7 deletions src/game/Chat/Level3.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -5130,7 +5130,7 @@ bool ChatHandler::HandleBanInfoCharacterCommand(char* args)

bool ChatHandler::HandleBanInfoHelper(uint32 accountid, char const* accountname)
{
auto queryResult = LoginDatabase.PQuery("SELECT FROM_UNIXTIME(banned_at),expires_at-banned_at,active,expires_at,reason,banned_by,unbanned_at,unbanned_by "
auto queryResult = LoginDatabase.PQuery("SELECT " _FROM_UNIXTIME_("banned_at") ",expires_at-banned_at,active,expires_at,reason,banned_by,unbanned_at,unbanned_by "
"FROM account_banned WHERE account_id = '%u' ORDER BY banned_at ASC", accountid);
if (!queryResult)
{
Expand Down Expand Up @@ -5181,7 +5181,7 @@ bool ChatHandler::HandleBanInfoIPCommand(char* args)
std::string IP = cIP;

LoginDatabase.escape_string(IP);
auto queryResult = LoginDatabase.PQuery("SELECT ip, FROM_UNIXTIME(banned_at), FROM_UNIXTIME(expires_at), expires_at-UNIX_TIMESTAMP(), reason,banned_by,expires_at-banned_at"
auto queryResult = LoginDatabase.PQuery("SELECT ip, " _FROM_UNIXTIME_("banned_at") ", " _FROM_UNIXTIME_("expires_at") ", expires_at-" _UNIXTIME_ ", reason,banned_by,expires_at-banned_at"
"FROM ip_banned WHERE ip = '%s'", IP.c_str());
if (!queryResult)
{
Expand All @@ -5199,7 +5199,7 @@ bool ChatHandler::HandleBanInfoIPCommand(char* args)

bool ChatHandler::HandleBanListCharacterCommand(char* args)
{
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");

char* cFilter = ExtractLiteralArg(&args);
if (!cFilter)
Expand All @@ -5219,7 +5219,7 @@ bool ChatHandler::HandleBanListCharacterCommand(char* args)

bool ChatHandler::HandleBanListAccountCommand(char* args)
{
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");

char* cFilter = ExtractLiteralArg(&args);
std::string filter = cFilter ? cFilter : "";
Expand Down Expand Up @@ -5328,7 +5328,7 @@ bool ChatHandler::HandleBanListHelper(std::unique_ptr<QueryResult> queryResult)

bool ChatHandler::HandleBanListIPCommand(char* args)
{
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");

char* cFilter = ExtractLiteralArg(&args);
std::string filter = cFilter ? cFilter : "";
Expand All @@ -5339,13 +5339,13 @@ bool ChatHandler::HandleBanListIPCommand(char* args)
if (filter.empty())
{
queryResult = LoginDatabase.Query("SELECT ip,banned_at,expires_at,banned_by,reason FROM ip_banned"
" WHERE (banned_at=expires_at OR expires_at>UNIX_TIMESTAMP())"
" WHERE (banned_at=expires_at OR expires_at>" _UNIXTIME_ ")"
" ORDER BY expires_at");
}
else
{
queryResult = LoginDatabase.PQuery("SELECT ip,banned_at,expires_at,banned_by,reason FROM ip_banned"
" WHERE (banned_at=expires_at OR expires_at>UNIX_TIMESTAMP()) AND ip " _LIKE_ " " _CONCAT3_("'%%'", "'%s'", "'%%'")
" WHERE (banned_at=expires_at OR expires_at>" _UNIXTIME_ ") AND ip " _LIKE_ " " _CONCAT3_("'%%'", "'%s'", "'%%'")
" ORDER BY expires_at", filter.c_str());
}

Expand Down
2 changes: 1 addition & 1 deletion src/game/GameEvents/GameEventMgr.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -647,7 +647,7 @@ uint32 GameEventMgr::Initialize() // return the next e
}
while (queryResult->NextRow());

CharacterDatabase.Execute("TRUNCATE game_event_status");
CharacterDatabase.Execute(_TRUNCATE_ " game_event_status");
}

uint32 delay = Update(&activeAtShutdown);
Expand Down
4 changes: 2 additions & 2 deletions src/game/Maps/MapPersistentStateMgr.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -990,7 +990,7 @@ void MapPersistentStateManager::InitWorldMaps()
void MapPersistentStateManager::LoadCreatureRespawnTimes()
{
// remove outdated data
CharacterDatabase.DirectExecute("DELETE FROM creature_respawn WHERE respawntime <= UNIX_TIMESTAMP(NOW())");
CharacterDatabase.DirectExecute("DELETE FROM creature_respawn WHERE respawntime <= " _UNIXNOW_);

uint32 count = 0;

Expand Down Expand Up @@ -1054,7 +1054,7 @@ void MapPersistentStateManager::LoadCreatureRespawnTimes()
void MapPersistentStateManager::LoadGameobjectRespawnTimes()
{
// remove outdated data
CharacterDatabase.DirectExecute("DELETE FROM gameobject_respawn WHERE respawntime <= UNIX_TIMESTAMP(NOW())");
CharacterDatabase.DirectExecute("DELETE FROM gameobject_respawn WHERE respawntime <= " _UNIXNOW_);

uint32 count = 0;

Expand Down
6 changes: 3 additions & 3 deletions src/game/Server/WorldSocket.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -411,9 +411,9 @@ bool WorldSocket::HandleAuthSession(WorldPacket& recvPacket)

// Re-check account ban (same check as in realmd)
auto banresult =
LoginDatabase.PQuery("SELECT 1 FROM account_banned WHERE account_id = %u AND active = 1 AND (expires_at > UNIX_TIMESTAMP() OR expires_at = banned_at)"
LoginDatabase.PQuery("SELECT 1 FROM account_banned WHERE account_id = %u AND active = 1 AND (expires_at > " _UNIXTIME_ " OR expires_at = banned_at)"
"UNION "
"SELECT 1 FROM ip_banned WHERE (expires_at = banned_at OR expires_at > UNIX_TIMESTAMP()) AND ip = '%s'",
"SELECT 1 FROM ip_banned WHERE (expires_at = banned_at OR expires_at > " _UNIXTIME_ ") AND ip = '%s'",
id, GetRemoteAddress().c_str());

if (banresult) // if account banned
Expand Down Expand Up @@ -485,7 +485,7 @@ bool WorldSocket::HandleAuthSession(WorldPacket& recvPacket)
// No SQL injection, username escaped.
static SqlStatementID updAccount;

SqlStatement stmt = LoginDatabase.CreateStatement(updAccount, "INSERT INTO account_logons(accountId,ip,loginTime,loginSource) VALUES(?,?,NOW(),?)");
SqlStatement stmt = LoginDatabase.CreateStatement(updAccount, "INSERT INTO account_logons(accountId,ip,loginTime,loginSource) VALUES(?,?," _NOW_ ",?)");
stmt.PExecute(id, address.c_str(), std::to_string(LOGIN_TYPE_MANGOSD).c_str());

m_crypt.Init(&K);
Expand Down
18 changes: 9 additions & 9 deletions src/game/World/World.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -857,7 +857,7 @@ void World::SetInitialWorldSettings()
LoginDatabase.PExecute("UPDATE realmlist SET icon = %u, timezone = %u WHERE id = '%u'", server_type, realm_zone, realmID);

///- Remove the bones (they should not exist in DB though) and old corpses after a restart
CharacterDatabase.PExecute("DELETE FROM corpse WHERE corpse_type = '0' OR time < (UNIX_TIMESTAMP()-'%u')", 3 * DAY);
CharacterDatabase.PExecute("DELETE FROM corpse WHERE corpse_type = '0' OR time < (" _UNIXTIME_ "-'%u')", 3 * DAY);

/// load spell_dbc first! dbc's need them
sLog.outString("Loading spell_template...");
Expand Down Expand Up @@ -1341,7 +1341,7 @@ void World::SetInitialWorldSettings()
CheckLootTemplates_Reference(ids_set);

sLog.outString("Deleting expired bans...");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");
sLog.outString();

sLog.outString("Calculate next weekly quest reset time...");
Expand Down Expand Up @@ -1844,17 +1844,17 @@ void World::WarnAccount(uint32 accountId, std::string from, std::string reason,
reason = std::string(type) + ": " + reason;
LoginDatabase.escape_string(reason);

LoginDatabase.PExecute("INSERT INTO account_banned (account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', UNIX_TIMESTAMP(), UNIX_TIMESTAMP()+1, '%s', '%s', '0')",
LoginDatabase.PExecute("INSERT INTO account_banned (account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', " _UNIXTIME_ ", " _UNIXTIME_ "+1, '%s', '%s', '0')",
accountId, from.c_str(), reason.c_str());
}

BanReturn World::BanAccount(WorldSession *session, uint32 duration_secs, const std::string& reason, const std::string& author)
{
if (duration_secs)
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', UNIX_TIMESTAMP(), UNIX_TIMESTAMP()+%u, '%s', '%s', '1')",
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', " _UNIXTIME_ ", " _UNIXTIME_ "+%u, '%s', '%s', '1')",
session->GetAccountId(), duration_secs, author.c_str(), reason.c_str());
else
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', UNIX_TIMESTAMP(), 0, '%s', '%s', '1')",
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', " _UNIXTIME_ ", 0, '%s', '%s', '1')",
session->GetAccountId(), author.c_str(), reason.c_str());

session->KickPlayer();
Expand All @@ -1877,7 +1877,7 @@ BanReturn World::BanAccount(BanMode mode, std::string nameOrIP, uint32 duration_
case BAN_IP:
// No SQL injection as strings are escaped
resultAccounts = LoginDatabase.PQuery("SELECT accountId FROM account_logons WHERE ip = '%s' ORDER BY loginTime DESC LIMIT 1", nameOrIP.c_str());
LoginDatabase.PExecute("INSERT INTO ip_banned VALUES ('%s',UNIX_TIMESTAMP(),UNIX_TIMESTAMP()+%u,'%s','%s')", nameOrIP.c_str(), duration_secs, safe_author.c_str(), reason.c_str());
LoginDatabase.PExecute("INSERT INTO ip_banned VALUES ('%s'," _UNIXTIME_ "," _UNIXTIME_ "+%u,'%s','%s')", nameOrIP.c_str(), duration_secs, safe_author.c_str(), reason.c_str());
break;
case BAN_ACCOUNT:
// No SQL injection as string is escaped
Expand Down Expand Up @@ -1908,7 +1908,7 @@ BanReturn World::BanAccount(BanMode mode, std::string nameOrIP, uint32 duration_
if (mode != BAN_IP)
{
// No SQL injection as strings are escaped
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', UNIX_TIMESTAMP(), UNIX_TIMESTAMP()+%u, '%s', '%s', '1')",
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active) VALUES ('%u', " _UNIXTIME_ ", " _UNIXTIME_ "+%u, '%s', '%s', '1')",
account, duration_secs, safe_author.c_str(), reason.c_str());
}

Expand Down Expand Up @@ -1941,7 +1941,7 @@ bool World::RemoveBanAccount(BanMode mode, const std::string& source, const std:
return false;

// NO SQL injection as account is uint32
LoginDatabase.PExecute("UPDATE account_banned SET active = '0', unbanned_at = UNIX_TIMESTAMP(), unbanned_by = '%s' WHERE account_id = '%u'", source.data(), account);
LoginDatabase.PExecute("UPDATE account_banned SET active = '0', unbanned_at = " _UNIXTIME_ ", unbanned_by = '%s' WHERE account_id = '%u'", source.data(), account);
WarnAccount(account, source, message, "UNBAN");
}
return true;
Expand Down Expand Up @@ -2233,7 +2233,7 @@ void World::LoadSpamRecords(bool reload)
void World::ResetWeeklyQuests()
{
DETAIL_LOG("Weekly quests reset for all characters.");
CharacterDatabase.Execute("TRUNCATE character_queststatus_weekly");
CharacterDatabase.Execute(_TRUNCATE_ " character_queststatus_weekly");
for (SessionMap::const_iterator itr = m_sessions.begin(); itr != m_sessions.end(); ++itr)
if (itr->second->GetPlayer())
itr->second->GetPlayer()->ResetWeeklyQuestStatus();
Expand Down
10 changes: 5 additions & 5 deletions src/realmd/AuthSocket.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -382,7 +382,7 @@ bool AuthSocket::_HandleLogonChallenge()
///- Verify that this IP is not in the ip_banned table
// No SQL injection possible (paste the IP address as passed by the socket)
std::unique_ptr<QueryResult> ip_banned_result(LoginDatabase.PQuery("SELECT expires_at FROM ip_banned "
"WHERE (expires_at = banned_at OR expires_at > UNIX_TIMESTAMP()) AND ip = '%s'", m_address.c_str()));
"WHERE (expires_at = banned_at OR expires_at > " _UNIXTIME_ ") AND ip = '%s'", m_address.c_str()));

if (ip_banned_result)
{
Expand Down Expand Up @@ -431,7 +431,7 @@ bool AuthSocket::_HandleLogonChallenge()
{
///- If the account is banned, reject the logon attempt
auto banresult = LoginDatabase.PQuery("SELECT banned_at,expires_at FROM account_banned WHERE "
"account_id = %u AND active = 1 AND (expires_at > UNIX_TIMESTAMP() OR expires_at = banned_at)", fields[0].GetUInt32());
"account_id = %u AND active = 1 AND (expires_at > " _UNIXTIME_ " OR expires_at = banned_at)", fields[0].GetUInt32());
if (banresult)
{
if ((*banresult)[0].GetUInt64() == (*banresult)[1].GetUInt64())
Expand Down Expand Up @@ -595,7 +595,7 @@ bool AuthSocket::_HandleLogonProof()
LoginDatabase.PExecute("UPDATE account SET sessionkey = '%s', locale = '%s', failed_logins = 0, os = '%s', platform = '%s' WHERE username = '%s'", K_hex, _safelocale.c_str(), m_os.c_str(), m_platform.c_str(), _safelogin.c_str());
std::unique_ptr<QueryResult> loginfail(LoginDatabase.PQuery("SELECT id FROM account WHERE username = '%s'", _safelogin.c_str()));
if (loginfail)
LoginDatabase.PExecute("INSERT INTO account_logons(accountId,ip,loginTime,loginSource) VALUES('%u','%s',NOW(),'%u')", loginfail->Fetch()[0].GetUInt32(), m_address.c_str(), LOGIN_TYPE_REALMD);
LoginDatabase.PExecute("INSERT INTO account_logons(accountId,ip,loginTime,loginSource) VALUES('%u','%s'," _NOW_ ",'%u')", loginfail->Fetch()[0].GetUInt32(), m_address.c_str(), LOGIN_TYPE_REALMD);
OPENSSL_free((void*)K_hex);

///- Finish SRP6 and send the final result to the client
Expand Down Expand Up @@ -643,7 +643,7 @@ bool AuthSocket::_HandleLogonProof()
{
uint32 acc_id = fields[0].GetUInt32();
LoginDatabase.PExecute("INSERT INTO account_banned(account_id, banned_at, expires_at, banned_by, reason, active)"
"VALUES ('%u',UNIX_TIMESTAMP(),UNIX_TIMESTAMP()+'%u','MaNGOS realmd','Failed login autoban',1)",
"VALUES ('%u'," _UNIXTIME_ "," _UNIXTIME_ "+'%u','MaNGOS realmd','Failed login autoban',1)",
acc_id, WrongPassBanTime);
BASIC_LOG("[AuthChallenge] account %s got banned for '%u' seconds because it failed to authenticate '%u' times",
_login.c_str(), WrongPassBanTime, failed_logins);
Expand All @@ -652,7 +652,7 @@ bool AuthSocket::_HandleLogonProof()
{
std::string current_ip = m_address;
LoginDatabase.escape_string(current_ip);
LoginDatabase.PExecute("INSERT INTO ip_banned VALUES ('%s',UNIX_TIMESTAMP(),UNIX_TIMESTAMP()+'%u','MaNGOS realmd','Failed login autoban')",
LoginDatabase.PExecute("INSERT INTO ip_banned VALUES ('%s'," _UNIXTIME_ "," _UNIXTIME_ "+'%u','MaNGOS realmd','Failed login autoban')",
current_ip.c_str(), WrongPassBanTime);
BASIC_LOG("[AuthChallenge] IP %s got banned for '%u' seconds because account %s failed to authenticate '%u' times",
current_ip.c_str(), WrongPassBanTime, _login.c_str(), failed_logins);
Expand Down
4 changes: 2 additions & 2 deletions src/realmd/Main.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -228,8 +228,8 @@ int main(int argc, char* argv[])
// cleanup query
// set expired bans to inactive
LoginDatabase.BeginTransaction();
LoginDatabase.Execute("UPDATE account_banned SET active = 0 WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=UNIX_TIMESTAMP() AND expires_at<>banned_at");
LoginDatabase.Execute("UPDATE account_banned SET active = 0 WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");
LoginDatabase.Execute("DELETE FROM ip_banned WHERE expires_at<=" _UNIXTIME_ " AND expires_at<>banned_at");
LoginDatabase.CommitTransaction();

// FIXME - more intelligent selection of thread count is needed here. config option?
Expand Down
Loading

0 comments on commit 21f3ec9

Please sign in to comment.