MXS-3892: Limit concurrent mapping of databases

As there are no practical benefits to multiple sessions for the same user
mapping the databases at the same time, limiting them to one update per
user is sensible. This is especially true now that we know the
information_schema tables aren't the most efficient things in the world.

The current code implements this rate limiting by closing any extra
sessions that would start a second update. The final implementation should
suspend them for the duration of the update as it is far more
user-friendly.

The limits are currently global as the shard caches are also global. This
is a performance bottleneck and it could be solved by storing the shard
cache inside of a mxs::WorkerGlobal instead of having it as a global
cache.
This commit is contained in:
Markus Mäkelä
2021-12-23 08:44:29 +02:00
parent d57de28199
commit 41c2a6ee8e
4 changed files with 92 additions and 5 deletions

View File

@ -45,7 +45,8 @@ SchemaRouterSession::SchemaRouterSession(MXS_SESSION* session,
, m_backends(backends) , m_backends(backends)
, m_config(router->m_config) , m_config(router->m_config)
, m_router(router) , m_router(router)
, m_shard(m_router->m_shard_manager.get_shard(get_cache_key(), m_config->refresh_min_interval)) , m_key(get_cache_key())
, m_shard(m_router->m_shard_manager.get_shard(m_key, m_config->refresh_min_interval))
, m_state(0) , m_state(0)
, m_sent_sescmd(0) , m_sent_sescmd(0)
, m_replied_sescmd(0) , m_replied_sescmd(0)
@ -111,6 +112,11 @@ void SchemaRouterSession::close()
} }
} }
if (m_state & INIT_MAPPING)
{
m_router->m_shard_manager.cancel_update(m_key);
}
std::lock_guard<std::mutex> guard(m_router->m_lock); std::lock_guard<std::mutex> guard(m_router->m_lock);
if (m_router->m_stats.longest_sescmd < m_stats.longest_sescmd) if (m_router->m_stats.longest_sescmd < m_stats.longest_sescmd)
@ -283,8 +289,26 @@ int32_t SchemaRouterSession::routeQuery(GWBUF* pPacket)
if (m_shard.empty()) if (m_shard.empty())
{ {
/* Generate database list */ // Check if another session has managed to update the shard cache
query_databases(); m_shard = m_router->m_shard_manager.get_shard(m_key, m_config->refresh_min_interval);
if (m_shard.empty())
{
// No entries in the cache, try to start an update
if (m_router->m_shard_manager.start_update(m_key))
{
// No other sessions are doing an update for this user, start one
query_databases();
}
else
{
// Too many concurrent updates
// TODO: Queue this session instead of killing it
m_pSession->kill(modutil_create_mysql_err_msg(1, 0, 1096, "HY000", "Too many updates"));
return 0;
}
}
} }
int ret = 0; int ret = 0;
@ -753,7 +777,7 @@ void SchemaRouterSession::handleError(GWBUF* pMessage,
void SchemaRouterSession::synchronize_shards() void SchemaRouterSession::synchronize_shards()
{ {
m_router->m_stats.shmap_cache_miss++; m_router->m_stats.shmap_cache_miss++;
m_router->m_shard_manager.update_shard(m_shard, get_cache_key()); m_router->m_shard_manager.update_shard(m_shard, m_key);
} }
/** /**

View File

@ -160,6 +160,7 @@ private:
SSRBackendList m_backends; /**< Backend references */ SSRBackendList m_backends; /**< Backend references */
SConfig m_config; /**< Session specific configuration */ SConfig m_config; /**< Session specific configuration */
SchemaRouter* m_router; /**< The router instance */ SchemaRouter* m_router; /**< The router instance */
std::string m_key; /**< Shard cache key */
Shard m_shard; /**< Database to server mapping */ Shard m_shard; /**< Database to server mapping */
std::string m_connect_db; /**< Database the user was trying to connect to */ std::string m_connect_db; /**< Database the user was trying to connect to */
std::string m_current_db; /**< Current active database */ std::string m_current_db; /**< Current active database */

View File

@ -209,4 +209,34 @@ void ShardManager::update_shard(Shard& shard, std::string user)
{ {
m_maps[user] = shard; m_maps[user] = shard;
} }
mxb_assert(m_limits[user] > 0);
--m_limits[user];
}
void ShardManager::set_update_limit(int64_t limit)
{
std::lock_guard<std::mutex> guard(m_lock);
m_update_limit = limit;
}
bool ShardManager::start_update(const std::string& user)
{
bool rval = false;
std::lock_guard<std::mutex> guard(m_lock);
if (m_limits[user] < m_update_limit)
{
++m_limits[user];
rval = true;
}
return rval;
}
void ShardManager::cancel_update(const std::string& user)
{
std::lock_guard<std::mutex> guard(m_lock);
mxb_assert(m_limits[user] > 0);
--m_limits[user];
} }

View File

@ -111,7 +111,8 @@ private:
time_t m_last_updated; time_t m_last_updated;
}; };
typedef std::unordered_map<std::string, Shard> ShardMap; typedef std::unordered_map<std::string, Shard> ShardMap;
typedef std::unordered_map<std::string, int64_t> MapLimits;
class ShardManager class ShardManager
{ {
@ -141,7 +142,38 @@ public:
*/ */
void update_shard(Shard& shard, std::string user); void update_shard(Shard& shard, std::string user);
/**
* Set how many concurrent shard updates are allowed per user
*
* By default only one update per user is allowed.
*
* @param limit Number of concurrent users to allow
*/
void set_update_limit(int64_t limit);
/**
* Start a shard update
*
* The update is considered finished when either update_shard() or cancel_update() is called. One of these
* two must be called by the session once start_update() has returned true.
*
* @param user The user whose shard is about to be updated
*
* @return True if an update can be done. False if there are too many concurrent
* updates being done by this user.
*/
bool start_update(const std::string& user);
/**
* Cancels a started shard update
*
* @param user The user whose shard was being updated
*/
void cancel_update(const std::string& user);
private: private:
mutable std::mutex m_lock; mutable std::mutex m_lock;
ShardMap m_maps; ShardMap m_maps;
MapLimits m_limits;
int64_t m_update_limit {1};
}; };