213 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2016 MariaDB Corporation Ab
 | |
|  *
 | |
|  * Use of this software is governed by the Business Source License included
 | |
|  * in the LICENSE.TXT file and at www.mariadb.com/bsl11.
 | |
|  *
 | |
|  * Change Date: 2024-06-02
 | |
|  *
 | |
|  * On the date above, in accordance with the Business Source License, use
 | |
|  * of this software will be governed by version 2 or later of the General
 | |
|  * Public License.
 | |
|  */
 | |
| 
 | |
| #include "shard_map.hh"
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| #include <maxscale/alloc.h>
 | |
| 
 | |
| Shard::Shard()
 | |
|     : m_last_updated(time(NULL))
 | |
| {
 | |
| }
 | |
| 
 | |
| Shard::~Shard()
 | |
| {
 | |
| }
 | |
| 
 | |
| bool Shard::add_location(std::string db, SERVER* target)
 | |
| {
 | |
|     return m_map.insert(std::make_pair(db, target)).second;
 | |
| }
 | |
| 
 | |
| void Shard::add_statement(std::string stmt, SERVER* target)
 | |
| {
 | |
|     stmt_map[stmt] = target;
 | |
| }
 | |
| 
 | |
| void Shard::add_statement(uint32_t id, SERVER* target)
 | |
| {
 | |
|     MXS_DEBUG("ADDING ID: [%u] server: [%s]", id, target->name);
 | |
|     m_binary_map[id] = target;
 | |
| }
 | |
| 
 | |
| void Shard::add_ps_handle(uint32_t id, uint32_t handle)
 | |
| {
 | |
|     MXS_DEBUG("ID: [%u] HANDLE: [%u]", id, handle);
 | |
|     m_ps_handles[id] = handle;
 | |
| }
 | |
| 
 | |
| bool Shard::remove_ps_handle(uint32_t id)
 | |
| {
 | |
|     return m_ps_handles.erase(id);
 | |
| }
 | |
| 
 | |
| uint32_t Shard::get_ps_handle(uint32_t id)
 | |
| {
 | |
|     PSHandleMap::iterator it = m_ps_handles.find(id);
 | |
|     if (it != m_ps_handles.end())
 | |
|     {
 | |
|         return it->second;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void Shard::replace_location(std::string db, SERVER* target)
 | |
| {
 | |
|     m_map[db] = target;
 | |
| }
 | |
| 
 | |
| SERVER* Shard::get_location(std::string table)
 | |
| {
 | |
|     SERVER* rval = NULL;
 | |
|     if (table.find(".") == std::string::npos)
 | |
|     {
 | |
|         for (ServerMap::iterator it = m_map.begin(); it != m_map.end(); it++)
 | |
|         {
 | |
|             std::transform(table.begin(), table.end(), table.begin(), ::tolower);
 | |
|             std::string db = it->first.substr(0, it->first.find("."));
 | |
|             std::transform(db.begin(), db.end(), db.begin(), ::tolower);
 | |
|             if (db.compare(table) == 0)
 | |
|             {
 | |
|                 if ((rval && rval != it->second))
 | |
|                 {
 | |
|                     MXS_DEBUG("There are 2 databases with same name on a different servers: '%s' and '%s'. Connecting to '%s'"
 | |
|                               ,
 | |
|                               rval->name,
 | |
|                               it->second->name,
 | |
|                               rval->name);
 | |
|                     break;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     rval = it->second;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         for (ServerMap::iterator it = m_map.begin(); it != m_map.end(); it++)
 | |
|         {
 | |
|             std::transform(table.begin(), table.end(), table.begin(), ::tolower);
 | |
|             std::string db = it->first;
 | |
|             std::transform(db.begin(), db.end(), db.begin(), ::tolower);
 | |
|             if (db.compare(table) == 0)
 | |
|             {
 | |
|                 rval = it->second;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| SERVER* Shard::get_statement(std::string stmt)
 | |
| {
 | |
|     SERVER* rval = NULL;
 | |
|     ServerMap::iterator iter = stmt_map.find(stmt);
 | |
|     if (iter != stmt_map.end())
 | |
|     {
 | |
|         rval = iter->second;
 | |
|     }
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| SERVER* Shard::get_statement(uint32_t id)
 | |
| {
 | |
|     SERVER* rval = NULL;
 | |
|     BinaryPSMap::iterator iter = m_binary_map.find(id);
 | |
|     if (iter != m_binary_map.end())
 | |
|     {
 | |
|         rval = iter->second;
 | |
|     }
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| bool Shard::remove_statement(std::string stmt)
 | |
| {
 | |
|     return stmt_map.erase(stmt);
 | |
| }
 | |
| 
 | |
| bool Shard::remove_statement(uint32_t id)
 | |
| {
 | |
|     return m_binary_map.erase(id);
 | |
| }
 | |
| 
 | |
| bool Shard::stale(double max_interval) const
 | |
| {
 | |
|     time_t now = time(NULL);
 | |
| 
 | |
|     return difftime(now, m_last_updated) > max_interval;
 | |
| }
 | |
| 
 | |
| bool Shard::empty() const
 | |
| {
 | |
|     return m_map.size() == 0;
 | |
| }
 | |
| 
 | |
| void Shard::get_content(ServerMap& dest)
 | |
| {
 | |
|     for (ServerMap::iterator it = m_map.begin(); it != m_map.end(); it++)
 | |
|     {
 | |
|         dest.insert(*it);
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool Shard::newer_than(const Shard& shard) const
 | |
| {
 | |
|     return m_last_updated > shard.m_last_updated;
 | |
| }
 | |
| 
 | |
| ShardManager::ShardManager()
 | |
| {
 | |
| }
 | |
| 
 | |
| ShardManager::~ShardManager()
 | |
| {
 | |
| }
 | |
| 
 | |
| Shard ShardManager::get_shard(std::string user, double max_interval)
 | |
| {
 | |
|     std::lock_guard<std::mutex> guard(m_lock);
 | |
| 
 | |
|     ShardMap::iterator iter = m_maps.find(user);
 | |
| 
 | |
|     if (iter == m_maps.end() || iter->second.stale(max_interval))
 | |
|     {
 | |
|         // No previous shard or a stale shard, construct a new one
 | |
| 
 | |
|         if (iter != m_maps.end())
 | |
|         {
 | |
|             m_maps.erase(iter);
 | |
|         }
 | |
| 
 | |
|         return Shard();
 | |
|     }
 | |
| 
 | |
|     // Found valid shard
 | |
|     return iter->second;
 | |
| }
 | |
| 
 | |
| void ShardManager::update_shard(Shard& shard, std::string user)
 | |
| {
 | |
|     std::lock_guard<std::mutex> guard(m_lock);
 | |
|     ShardMap::iterator iter = m_maps.find(user);
 | |
| 
 | |
|     if (iter == m_maps.end() || shard.newer_than(iter->second))
 | |
|     {
 | |
|         m_maps[user] = shard;
 | |
|     }
 | |
| }
 | 
