From 77bc22ffec958150b532e718a5bb27eb88bac267 Mon Sep 17 00:00:00 2001 From: groot Date: Sun, 14 Apr 2019 14:32:49 +0800 Subject: [PATCH 1/4] add cache classes Former-commit-id: 7a268e064088b6f4a759a714d6e7dd70d5c6dcfb --- cpp/src/CMakeLists.txt | 5 +- cpp/src/cache/Cache.cpp | 220 +++++++++++++++++++++++++++++ cpp/src/cache/Cache.h | 74 ++++++++++ cpp/src/cache/CacheMgr.cpp | 100 +++++++++++++ cpp/src/cache/CacheMgr.h | 50 +++++++ cpp/src/cache/DataObj.h | 35 +++++ cpp/src/cache/LRU.h | 102 +++++++++++++ cpp/unittest/CMakeLists.txt | 1 + cpp/unittest/cache/CMakeLists.txt | 19 +++ cpp/unittest/cache/cache_tests.cpp | 41 ++++++ 10 files changed, 646 insertions(+), 1 deletion(-) create mode 100644 cpp/src/cache/Cache.cpp create mode 100644 cpp/src/cache/Cache.h create mode 100644 cpp/src/cache/CacheMgr.cpp create mode 100644 cpp/src/cache/CacheMgr.h create mode 100644 cpp/src/cache/DataObj.h create mode 100644 cpp/src/cache/LRU.h create mode 100644 cpp/unittest/cache/CMakeLists.txt create mode 100644 cpp/unittest/cache/cache_tests.cpp diff --git a/cpp/src/CMakeLists.txt b/cpp/src/CMakeLists.txt index 5b1d303ada..bd314be8e6 100644 --- a/cpp/src/CMakeLists.txt +++ b/cpp/src/CMakeLists.txt @@ -4,8 +4,11 @@ # Proprietary and confidential. #------------------------------------------------------------------------------- +AUX_SOURCE_DIRECTORY(./cache cache_files) + set(vecwise_engine_src ${CMAKE_CURRENT_SOURCE_DIR}/main.cpp - ) + ${cache_files} + cache/DataObj.h) add_library(vecwise_engine SHARED ${vecwise_engine_src}) \ No newline at end of file diff --git a/cpp/src/cache/Cache.cpp b/cpp/src/cache/Cache.cpp new file mode 100644 index 0000000000..7673d1685b --- /dev/null +++ b/cpp/src/cache/Cache.cpp @@ -0,0 +1,220 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#include "Cache.h" + +#include + +namespace zilliz { +namespace vecwise { +namespace cache { + +Cache::Cache(int64_t mem_capacity, uint64_t cache_max_count) + : usage_(0), + capacity_(mem_capacity*1024*1024*1024), + lru_(cache_max_count) { +// AGENT_LOG_DEBUG << "Construct Cache with capacity " << std::to_string(mem_capacity) +} + +size_t Cache::size() const { + std::lock_guard lock(mutex_); + return lru_.size(); +} + +bool Cache::exists(const std::string& key) { + std::lock_guard lock(mutex_); + return lru_.exists(key); +} + +DataObjPtr Cache::get(const std::string& key) { + std::lock_guard lock(mutex_); + if(!lru_.exists(key)){ + return nullptr; + } + + const CacheObjPtr& cache_obj = lru_.get(key); + return cache_obj->data_; +} + +void Cache::insert(const std::string& key, const DataObjPtr& data_ptr) { + { + std::lock_guard lock(mutex_); + + /* if key already exist, over-write old data */ + if (lru_.exists(key)) { + CacheObjPtr obj_ptr = lru_.get(key); + + usage_ -= obj_ptr->data_->size(); + obj_ptr->data_ = data_ptr; + usage_ += data_ptr->size(); + } else { + CacheObjPtr obj_ptr(new CacheObj(data_ptr)); + lru_.put(key, obj_ptr); + usage_ += data_ptr->size(); + } + +// AGENT_LOG_DEBUG << "Insert into LRU(" << (capacity_ > 0 ? std::to_string(usage_ * 100 / capacity_) : "Nan") +// << "%, +" << data_ptr->size() << ", " << usage_ << ", " << lru_.size() << "):" +// << " " << key; + } + + if (usage_ > capacity_) { +// AGENT_LOG_TRACE << "Current usage " << usage_ +// << " exceeds cache capacity " << capacity_ +// << ", start free memory"; + free_memory(); + } +} + +void Cache::insert(const std::string& key, const std::shared_ptr& data, int64_t size) { + DataObjPtr ptr = std::make_shared(data, size); + insert(key, ptr); +} + +void Cache::erase(const std::string& key) { + std::lock_guard lock(mutex_); + if(!lru_.exists(key)){ + return; + } + + const CacheObjPtr& obj_ptr = lru_.get(key); + const DataObjPtr& data_ptr = obj_ptr->data_; + usage_ -= data_ptr->size(); +// AGENT_LOG_DEBUG << "Erase from LRU(" << (capacity_ > 0 ? std::to_string(usage_*100/capacity_) : "Nan") +// << "%, -" << data_ptr->size() << ", " << usage_ << ", " << lru_.size() << "): " +// << (data_ptr->flags().get_flag(DataObjAttr::kPinned) ? "Pinned " : "") +// << (data_ptr->flags().get_flag(DataObjAttr::kValid) ? "Valid " : "") +// << "(ref:" << obj_ptr->ref_ << ") " +// << key; + lru_.erase(key); +} + +void Cache::clear() { + std::lock_guard lock(mutex_); + lru_.clear(); + usage_ = 0; +// AGENT_LOG_DEBUG << "Clear LRU !"; +} + +#if 0 /* caiyd 20190221, need more testing before enable */ +void Cache::flush_to_file(const std::string& key, const CacheObjPtr& obj_ptr) { + if (!this->swap_enabled_) return; + + const DataObjPtr data_ptr = obj_ptr->data(); + + if (data_ptr == nullptr || data_ptr->size() == 0) return; + if (data_ptr->ptr() == nullptr) return; + + std::string name = std::to_string(reinterpret_cast(data_ptr.get())); + filesys::CreateDirectory(this->swap_path_); + + /* write cache data to file */ + obj_ptr->set_file_path(this->swap_path_ + "/" + name); + std::shared_ptr outfile = nullptr; + filesys::OpenWritableFile(obj_ptr->file_path(), false, &outfile); + filesys::WriteFile(outfile, data_ptr->ptr().get(), data_ptr->size()); + (void)outfile->Close(); + + AGENT_LOG_DEBUG << "Flush cache data: " << key << ", to file: " << obj_ptr->file_path(); + + /* free cache memory */ + data_ptr->ptr().reset(); + usage_ -= data_ptr->size(); +} + +void Cache::restore_from_file(const std::string& key, const CacheObjPtr& obj_ptr) { + if (!this->swap_enabled_) return; + + const DataObjPtr data_ptr = obj_ptr->data(); + if (data_ptr == nullptr || data_ptr->size() == 0) return; + + std::shared_ptr infile = nullptr; + int64_t file_size, bytes_read; + + /* load cache data from file */ + if (!filesys::FileExist(obj_ptr->file_path())) { + THROW_AGENT_UNEXPECTED_ERROR("File not exist: " + obj_ptr->file_path()); + } + filesys::OpenReadableFile(obj_ptr->file_path(), &infile); + infile->GetSize(&file_size); + if (data_ptr->size() != file_size) { + THROW_AGENT_UNEXPECTED_ERROR("File size not match: " + obj_ptr->file_path()); + } + data_ptr->set_ptr(lib::gpu::MakeShared(data_ptr->size(), lib::gpu::MallocHint::kUnifiedGlobal)); + infile->Read(file_size, &bytes_read, data_ptr->ptr().get()); + infile->Close(); + + AGENT_LOG_DEBUG << "Restore cache data: " << key << ", from file: " << obj_ptr->file_path(); + + /* clear file path */ + obj_ptr->set_file_path(""); + usage_ += data_ptr->size(); +} +#endif + +/* free memory space when CACHE occupation exceed its capacity */ +void Cache::free_memory() { + if (usage_ <= capacity_) return; + + int64_t threshhold = capacity_ * THRESHHOLD_PERCENT; + int64_t delta_size = usage_ - threshhold; + + std::set key_array; + int64_t released_size = 0; + + { + std::lock_guard lock(mutex_); + + auto it = lru_.rbegin(); + while (it != lru_.rend() && released_size < delta_size) { + auto& key = it->first; + auto& obj_ptr = it->second; + const auto& data_ptr = obj_ptr->data_; + + key_array.emplace(key); + released_size += data_ptr->size(); + ++it; + } + } + +// AGENT_LOG_DEBUG << "to be released memory size: " << released_size; + + for (auto& key : key_array) { + erase(key); + } + + print(); +} + +void Cache::print() { + int64_t still_pinned_count = 0; + int64_t total_pinned_size = 0; + int64_t total_valid_empty_size = 0; + { + std::lock_guard lock(mutex_); + + for (auto it = lru_.begin(); it != lru_.end(); ++it) { + auto& obj_ptr = it->second; + const auto& data_ptr = obj_ptr->data_; + if (data_ptr != nullptr) { + total_pinned_size += data_ptr->size(); + ++still_pinned_count; + } else { + total_valid_empty_size += data_ptr->size(); + } + } + } + +// AGENT_LOG_DEBUG << "[Still Pinned count]: " << still_pinned_count; +// AGENT_LOG_DEBUG << "[Pinned Memory total size(byte)]: " << total_pinned_size; +// AGENT_LOG_DEBUG << "[valid_empty total size(byte)]: " << total_valid_empty_size; +// AGENT_LOG_DEBUG << "[free memory size(byte)]: " << capacity_ - total_pinned_size - total_valid_empty_size; +} + +} // cache +} // vecwise +} // zilliz + diff --git a/cpp/src/cache/Cache.h b/cpp/src/cache/Cache.h new file mode 100644 index 0000000000..530f25fa04 --- /dev/null +++ b/cpp/src/cache/Cache.h @@ -0,0 +1,74 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#pragma once + +#include +#include +#include + +#include "LRU.h" +#include "DataObj.h" + +namespace zilliz { +namespace vecwise { +namespace cache { + +const std::string SWAP_DIR = ".CACHE"; +const float THRESHHOLD_PERCENT = 0.75; + +class Cache { +private: + class CacheObj { + public: + CacheObj() = delete; + + CacheObj(const DataObjPtr& data) + : data_(data) { + } + + CacheObj(const std::shared_ptr& data, int64_t size) { + data_ = std::make_shared(data, size); + } + + public: + DataObjPtr data_ = nullptr; + }; + + using CacheObjPtr = std::shared_ptr; + +public: + //mem_capacity, units:GB + Cache(int64_t mem_capacity, uint64_t cache_max_count); + ~Cache() = default; + + int64_t usage() const { return usage_; } + int64_t capacity() const { return capacity_; } + + size_t size() const; + bool exists(const std::string& key); + DataObjPtr get(const std::string& key); + void insert(const std::string& key, const DataObjPtr& data); + void insert(const std::string& key, const std::shared_ptr& data, int64_t size); + void erase(const std::string& key); + void print(); + void clear(); + void free_memory(); + +private: + int64_t usage_; + int64_t capacity_; + + LRU lru_; + mutable std::mutex mutex_; +}; + +using CachePtr = std::shared_ptr; + +} // cache +} // vecwise +} // zilliz + diff --git a/cpp/src/cache/CacheMgr.cpp b/cpp/src/cache/CacheMgr.cpp new file mode 100644 index 0000000000..d5e66cd7be --- /dev/null +++ b/cpp/src/cache/CacheMgr.cpp @@ -0,0 +1,100 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#include "CacheMgr.h" + +namespace zilliz { +namespace vecwise { +namespace cache { + +CacheMgr::CacheMgr() { + //TODO: loada config to initialize cache + cache_ = std::make_shared(16, 1UL<<32); +} + +size_t CacheMgr::ItemCount() const { + if(cache_ == nullptr) { + return 0; + } + + return cache_->size(); +} + +bool CacheMgr::IsExists(const std::string& key) { + if(cache_ == nullptr) { + return false; + } + + return cache_->exists(key); +} + +DataObjPtr CacheMgr::GetItem(const std::string& key) { + if(cache_ == nullptr) { + return nullptr; + } + + return cache_->get(key); +} + +void CacheMgr::InsertItem(const std::string& key, const DataObjPtr& data) { + if(cache_ == nullptr) { + return; + } + + cache_->insert(key, data); +} + +void CacheMgr::InsertItem(const std::string& key, const std::shared_ptr& data, int64_t size) { + if(cache_ == nullptr) { + return; + } + + cache_->insert(key, data, size); +} + +void CacheMgr::EraseItem(const std::string& key) { + if(cache_ == nullptr) { + return; + } + + cache_->erase(key); +} + +void CacheMgr::PrintInfo() { + if(cache_ == nullptr) { + return; + } + + cache_->print(); +} + +void CacheMgr::ClearCache() { + if(cache_ == nullptr) { + return; + } + + cache_->clear(); +} + +int64_t CacheMgr::CacheUsage() const { + if(cache_ == nullptr) { + return 0; + } + + return cache_->usage(); +} + +int64_t CacheMgr::CacheCapacity() const { + if(cache_ == nullptr) { + return 0; + } + + return cache_->capacity(); +} + +} +} +} diff --git a/cpp/src/cache/CacheMgr.h b/cpp/src/cache/CacheMgr.h new file mode 100644 index 0000000000..22d461734a --- /dev/null +++ b/cpp/src/cache/CacheMgr.h @@ -0,0 +1,50 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#pragma once + +#include "Cache.h" + +namespace zilliz { +namespace vecwise { +namespace cache { + +class CacheMgr { +public: + static CacheMgr& GetInstance() { + static CacheMgr mgr; + return mgr; + } + + size_t ItemCount() const; + + bool IsExists(const std::string& key); + + DataObjPtr GetItem(const std::string& key); + + void InsertItem(const std::string& key, const DataObjPtr& data); + void InsertItem(const std::string& key, const std::shared_ptr& data, int64_t size); + + void EraseItem(const std::string& key); + + void PrintInfo(); + + void ClearCache(); + + int64_t CacheUsage() const; + int64_t CacheCapacity() const; + +private: + CacheMgr(); + +private: + CachePtr cache_; +}; + + +} +} +} diff --git a/cpp/src/cache/DataObj.h b/cpp/src/cache/DataObj.h new file mode 100644 index 0000000000..89b7ad533b --- /dev/null +++ b/cpp/src/cache/DataObj.h @@ -0,0 +1,35 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#pragma once + +#include + +namespace zilliz { +namespace vecwise { +namespace cache { + +class DataObj { +public: + DataObj(const std::shared_ptr& data, int64_t size) + : data_(data), size_(size) + {} + + std::shared_ptr data() { return data_; } + const std::shared_ptr& data() const { return data_; } + + int64_t size() const { return size_; } + +private: + std::shared_ptr data_ = nullptr; + int64_t size_ = 0; +}; + +using DataObjPtr = std::shared_ptr; + +} +} +} \ No newline at end of file diff --git a/cpp/src/cache/LRU.h b/cpp/src/cache/LRU.h new file mode 100644 index 0000000000..3f5620850c --- /dev/null +++ b/cpp/src/cache/LRU.h @@ -0,0 +1,102 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// + +#pragma once + +#include +#include +#include +#include + +namespace zilliz { +namespace vecwise { +namespace cache { + +template +class LRU { +public: + typedef typename std::pair key_value_pair_t; + typedef typename std::list::iterator list_iterator_t; + typedef typename std::list::reverse_iterator reverse_list_iterator_t; + + LRU(size_t max_size) : _max_size(max_size) {} + + void put(const key_t& key, const value_t& value) { + auto it = _cache_items_map.find(key); + _cache_items_list.push_front(key_value_pair_t(key, value)); + if (it != _cache_items_map.end()) { + _cache_items_list.erase(it->second); + _cache_items_map.erase(it); + } + _cache_items_map[key] = _cache_items_list.begin(); + + if (_cache_items_map.size() > _max_size) { + auto last = _cache_items_list.end(); + last--; + _cache_items_map.erase(last->first); + _cache_items_list.pop_back(); + } + } + + const value_t& get(const key_t& key) { + auto it = _cache_items_map.find(key); + if (it == _cache_items_map.end()) { + throw std::range_error("There is no such key in cache"); + } else { + _cache_items_list.splice(_cache_items_list.begin(), _cache_items_list, it->second); + return it->second->second; + } + } + + void erase(const key_t& key) { + auto it = _cache_items_map.find(key); + if (it != _cache_items_map.end()) { + _cache_items_list.erase(it->second); + _cache_items_map.erase(it); + } + } + + bool exists(const key_t& key) const { + return _cache_items_map.find(key) != _cache_items_map.end(); + } + + size_t size() const { + return _cache_items_map.size(); + } + + list_iterator_t begin() { + _iter = _cache_items_list.begin(); + return _iter; + } + + list_iterator_t end() { + return _cache_items_list.end(); + } + + reverse_list_iterator_t rbegin() { + return _cache_items_list.rbegin(); + } + + reverse_list_iterator_t rend() { + return _cache_items_list.rend(); + } + + void clear() { + _cache_items_list.clear(); + _cache_items_map.clear(); + } + +private: + std::list _cache_items_list; + std::unordered_map _cache_items_map; + size_t _max_size; + list_iterator_t _iter; +}; + +} // cache +} // vecwise +} // zilliz + diff --git a/cpp/unittest/CMakeLists.txt b/cpp/unittest/CMakeLists.txt index ab080d92fb..baaa8afa0b 100644 --- a/cpp/unittest/CMakeLists.txt +++ b/cpp/unittest/CMakeLists.txt @@ -16,4 +16,5 @@ set(unittest_libs gmock_main pthread) +add_subdirectory(cache) add_subdirectory(log) \ No newline at end of file diff --git a/cpp/unittest/cache/CMakeLists.txt b/cpp/unittest/cache/CMakeLists.txt new file mode 100644 index 0000000000..eb1856ae1a --- /dev/null +++ b/cpp/unittest/cache/CMakeLists.txt @@ -0,0 +1,19 @@ +#------------------------------------------------------------------------------- +# Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +# Unauthorized copying of this file, via any medium is strictly prohibited. +# Proprietary and confidential. +#------------------------------------------------------------------------------- +include_directories(../../src) + +set(cache_srcs + ../../src/cache/Cache.cpp + ../../src/cache/CacheMgr.cpp) + +set(cache_test_src + ${unittest_srcs} + ${cache_srcs} + cache_tests.cpp) + +add_executable(cache_tests ${cache_test_src}) + +target_link_libraries(cache_tests ${unittest_libs}) diff --git a/cpp/unittest/cache/cache_tests.cpp b/cpp/unittest/cache/cache_tests.cpp new file mode 100644 index 0000000000..3505f5aa8f --- /dev/null +++ b/cpp/unittest/cache/cache_tests.cpp @@ -0,0 +1,41 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include +#include "cache/CacheMgr.h" + +using namespace zilliz::vecwise; + +namespace { +cache::DataObjPtr MakeData(int64_t size) { + auto data_ptr = std::shared_ptr(new char[size], std::default_delete()); + return std::make_shared(data_ptr, size); +} + +#define MAKE_1GB_DATA MakeData(1*1024*1024*1024) +#define MAKE_100MB_DATA MakeData(100*1024*1024) +#define MAKE_1MB_DATA MakeData(1*1024*1024) +} + +TEST(CacheTest, CACHE_TEST) { + cache::CacheMgr cache_mgr = cache::CacheMgr::GetInstance(); + + for(int i = 0; i < 10; i++) { + cache_mgr.InsertItem(std::to_string(i), MAKE_100MB_DATA); + } + + ASSERT_EQ(cache_mgr.ItemCount(), 10); + + std::string key = "test_data"; + cache_mgr.InsertItem(key, MAKE_100MB_DATA); + cache::DataObjPtr data = cache_mgr.GetItem(key); + ASSERT_TRUE(data != nullptr); + ASSERT_TRUE(cache_mgr.IsExists(key)); + ASSERT_EQ(data->size(), 100*1024*1024); + + cache_mgr.EraseItem(key); + data = cache_mgr.GetItem(key); + ASSERT_TRUE(data == nullptr); +} \ No newline at end of file From 86de13c86060951c8c804bbf06272a8f4d6ab606 Mon Sep 17 00:00:00 2001 From: groot Date: Sun, 14 Apr 2019 14:36:45 +0800 Subject: [PATCH 2/4] add cache classes Former-commit-id: 3f1545335159443d0b86bacbd6fdecd040f40820 --- cpp/src/CMakeLists.txt | 2 +- cpp/src/cache/CacheMgr.cpp | 6 +++--- cpp/src/cache/CacheMgr.h | 4 ++-- cpp/unittest/cache/CMakeLists.txt | 4 +--- cpp/unittest/cache/cache_tests.cpp | 2 +- 5 files changed, 8 insertions(+), 10 deletions(-) diff --git a/cpp/src/CMakeLists.txt b/cpp/src/CMakeLists.txt index bd314be8e6..0dfdf91d5a 100644 --- a/cpp/src/CMakeLists.txt +++ b/cpp/src/CMakeLists.txt @@ -4,7 +4,7 @@ # Proprietary and confidential. #------------------------------------------------------------------------------- -AUX_SOURCE_DIRECTORY(./cache cache_files) +aux_source_directory(./cache cache_files) set(vecwise_engine_src ${CMAKE_CURRENT_SOURCE_DIR}/main.cpp diff --git a/cpp/src/cache/CacheMgr.cpp b/cpp/src/cache/CacheMgr.cpp index d5e66cd7be..df53c42510 100644 --- a/cpp/src/cache/CacheMgr.cpp +++ b/cpp/src/cache/CacheMgr.cpp @@ -15,15 +15,15 @@ CacheMgr::CacheMgr() { cache_ = std::make_shared(16, 1UL<<32); } -size_t CacheMgr::ItemCount() const { +uint64_t CacheMgr::ItemCount() const { if(cache_ == nullptr) { return 0; } - return cache_->size(); + return (uint64_t)(cache_->size()); } -bool CacheMgr::IsExists(const std::string& key) { +bool CacheMgr::ItemExists(const std::string& key) { if(cache_ == nullptr) { return false; } diff --git a/cpp/src/cache/CacheMgr.h b/cpp/src/cache/CacheMgr.h index 22d461734a..759e76b9a1 100644 --- a/cpp/src/cache/CacheMgr.h +++ b/cpp/src/cache/CacheMgr.h @@ -19,9 +19,9 @@ public: return mgr; } - size_t ItemCount() const; + uint64_t ItemCount() const; - bool IsExists(const std::string& key); + bool ItemExists(const std::string& key); DataObjPtr GetItem(const std::string& key); diff --git a/cpp/unittest/cache/CMakeLists.txt b/cpp/unittest/cache/CMakeLists.txt index eb1856ae1a..9a088d7c44 100644 --- a/cpp/unittest/cache/CMakeLists.txt +++ b/cpp/unittest/cache/CMakeLists.txt @@ -5,9 +5,7 @@ #------------------------------------------------------------------------------- include_directories(../../src) -set(cache_srcs - ../../src/cache/Cache.cpp - ../../src/cache/CacheMgr.cpp) +aux_source_directory(../../src/cache cache_srcs) set(cache_test_src ${unittest_srcs} diff --git a/cpp/unittest/cache/cache_tests.cpp b/cpp/unittest/cache/cache_tests.cpp index 3505f5aa8f..7526c1f6cd 100644 --- a/cpp/unittest/cache/cache_tests.cpp +++ b/cpp/unittest/cache/cache_tests.cpp @@ -32,7 +32,7 @@ TEST(CacheTest, CACHE_TEST) { cache_mgr.InsertItem(key, MAKE_100MB_DATA); cache::DataObjPtr data = cache_mgr.GetItem(key); ASSERT_TRUE(data != nullptr); - ASSERT_TRUE(cache_mgr.IsExists(key)); + ASSERT_TRUE(cache_mgr.ItemExists(key)); ASSERT_EQ(data->size(), 100*1024*1024); cache_mgr.EraseItem(key); From aac43d2ffeb540da1f6d73e9089c690da8d1c02a Mon Sep 17 00:00:00 2001 From: groot Date: Sun, 14 Apr 2019 16:25:44 +0800 Subject: [PATCH 3/4] add server code Former-commit-id: 00da072153df8d103f6d232db6f945aff1bd2869 --- cpp/CMakeLists.txt | 5 + cpp/conf/server_config.yaml | 3 + cpp/src/CMakeLists.txt | 21 ++- cpp/src/config/ConfigNode.cpp | 221 ++++++++++++++++++++++++++++++ cpp/src/config/ConfigNode.h | 63 +++++++++ cpp/src/config/IConfigMgr.cpp | 20 +++ cpp/src/config/IConfigMgr.h | 43 ++++++ cpp/src/config/YamlConfigMgr.cpp | 109 +++++++++++++++ cpp/src/config/YamlConfigMgr.h | 52 +++++++ cpp/src/main.cpp | 97 ++++++++++++- cpp/src/server/Server.cpp | 227 +++++++++++++++++++++++++++++++ cpp/src/server/Server.h | 49 +++++++ cpp/src/server/ServerConfig.cpp | 94 +++++++++++++ cpp/src/server/ServerConfig.h | 39 ++++++ cpp/src/utils/CommonUtil.cpp | 145 ++++++++++++++++++++ cpp/src/utils/CommonUtil.h | 33 +++++ cpp/src/utils/Error.h | 35 +++++ cpp/src/utils/SignalUtil.cpp | 62 +++++++++ cpp/src/utils/SignalUtil.h | 20 +++ cpp/src/utils/ThreadPool.h | 118 ++++++++++++++++ cpp/src/utils/TimeRecorder.cpp | 144 ++++++++++++++++++++ cpp/src/utils/TimeRecorder.h | 55 ++++++++ 22 files changed, 1650 insertions(+), 5 deletions(-) create mode 100644 cpp/conf/server_config.yaml create mode 100644 cpp/src/config/ConfigNode.cpp create mode 100644 cpp/src/config/ConfigNode.h create mode 100644 cpp/src/config/IConfigMgr.cpp create mode 100644 cpp/src/config/IConfigMgr.h create mode 100644 cpp/src/config/YamlConfigMgr.cpp create mode 100644 cpp/src/config/YamlConfigMgr.h create mode 100644 cpp/src/server/Server.cpp create mode 100644 cpp/src/server/Server.h create mode 100644 cpp/src/server/ServerConfig.cpp create mode 100644 cpp/src/server/ServerConfig.h create mode 100644 cpp/src/utils/CommonUtil.cpp create mode 100755 cpp/src/utils/CommonUtil.h create mode 100644 cpp/src/utils/Error.h create mode 100644 cpp/src/utils/SignalUtil.cpp create mode 100644 cpp/src/utils/SignalUtil.h create mode 100644 cpp/src/utils/ThreadPool.h create mode 100644 cpp/src/utils/TimeRecorder.cpp create mode 100644 cpp/src/utils/TimeRecorder.h diff --git a/cpp/CMakeLists.txt b/cpp/CMakeLists.txt index c6b1eede97..f71ad4821d 100644 --- a/cpp/CMakeLists.txt +++ b/cpp/CMakeLists.txt @@ -52,6 +52,9 @@ include_directories(${VECWISE_ENGINE_INCLUDE}) include_directories(${VECWISE_ENGINE_SRC}) include_directories(${VECWISE_THIRD_PARTY_BUILD}/include) +link_directories(${CMAKE_CURRRENT_BINARY_DIR}) +link_directories(./third_party/build/lib) + execute_process(COMMAND bash build.sh WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/third_party) @@ -60,3 +63,5 @@ add_subdirectory(src) if (BUILD_UNIT_TEST) add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/unittest) endif(BUILD_UNIT_TEST) + +add_custom_target(Clean-All COMMAND ${CMAKE_BUILD_TOOL} clean) diff --git a/cpp/conf/server_config.yaml b/cpp/conf/server_config.yaml new file mode 100644 index 0000000000..6b23da0ec6 --- /dev/null +++ b/cpp/conf/server_config.yaml @@ -0,0 +1,3 @@ +server_config: + address: 127.0.0.1 + port: 21001 \ No newline at end of file diff --git a/cpp/src/CMakeLists.txt b/cpp/src/CMakeLists.txt index 0dfdf91d5a..e400fb7200 100644 --- a/cpp/src/CMakeLists.txt +++ b/cpp/src/CMakeLists.txt @@ -5,10 +5,27 @@ #------------------------------------------------------------------------------- aux_source_directory(./cache cache_files) +aux_source_directory(./config config_files) +aux_source_directory(./server server_files) +aux_source_directory(./utils utils_files) set(vecwise_engine_src ${CMAKE_CURRENT_SOURCE_DIR}/main.cpp ${cache_files} - cache/DataObj.h) + ) -add_library(vecwise_engine SHARED ${vecwise_engine_src}) \ No newline at end of file +add_library(vecwise_engine SHARED ${vecwise_engine_src}) + +add_executable(vecwise_engine_server + ${config_files} + ${server_files} + ${utils_files} + ) + +set(dependency_libs + vecwise_engine + yaml-cpp + boost_system + boost_filesystem + ) +target_link_libraries(vecwise_engine_server ${dependency_libs}) \ No newline at end of file diff --git a/cpp/src/config/ConfigNode.cpp b/cpp/src/config/ConfigNode.cpp new file mode 100644 index 0000000000..d67630db95 --- /dev/null +++ b/cpp/src/config/ConfigNode.cpp @@ -0,0 +1,221 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#include "ConfigNode.h" +#include "utils/Error.h" +#include "utils/CommonUtil.h" + +#include +#include +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +void ConfigNode::Combine(const ConfigNode& target) { + const std::map& kv = target.GetConfig(); + for(auto itr = kv.begin(); itr != kv.end(); ++itr){ + config_[itr->first] = itr->second; + } + + const std::map >& sequences = target.GetSequences(); + for(auto itr = sequences.begin(); itr != sequences.end(); ++itr){ + sequences_[itr->first] = itr->second; + } + + const std::map& children = target.GetChildren(); + for(auto itr = children.begin(); itr != children.end(); ++itr){ + children_[itr->first] = itr->second; + } +} + +//key/value pair config +void +ConfigNode::SetValue(const std::string& key, const std::string& value) { + config_[key] = value; +} + +std::string +ConfigNode::GetValue(const std::string& param_key, const std::string& default_val) const { + auto ref = config_.find(param_key); + if(ref != config_.end()) { + return ref->second; + } + + //THROW_UNEXPECTED_ERROR("Can't find parameter key: " + param_key); + return default_val; +} + +bool +ConfigNode::GetBoolValue(const std::string ¶m_key, bool default_val) const { + std::string val = GetValue(param_key); + if (!val.empty()) { + std::transform(val.begin(), val.end(), val.begin(), ::tolower); + return (val == "true" || val == "on" || val == "yes" || val == "1"); + } else { + return default_val; + } +} + +int32_t +ConfigNode::GetInt32Value(const std::string ¶m_key, int32_t default_val) const { + std::string val = GetValue(param_key); + if (!val.empty()) { + return (int32_t)std::strtol(val.c_str(), nullptr, 10); + } else { + return default_val; + } +} + +int64_t +ConfigNode::GetInt64Value(const std::string ¶m_key, int64_t default_val) const { + std::string val = GetValue(param_key); + if (!val.empty()) { + return std::strtol(val.c_str(), nullptr, 10); + } else { + return default_val; + } +} + +float +ConfigNode::GetFloatValue(const std::string ¶m_key, float default_val) const { + std::string val = GetValue(param_key); + if (!val.empty()) { + return std::strtof(val.c_str(), nullptr); + } else { + return default_val; + } +} + +double +ConfigNode::GetDoubleValue(const std::string ¶m_key, double default_val) const { + std::string val = GetValue(param_key); + if (!val.empty()) { + return std::strtold(val.c_str(), nullptr); + } else { + return default_val; + } +} + +const std::map& +ConfigNode::GetConfig() const { + return config_; +}; + +void ConfigNode::ClearConfig() { + config_.clear(); +} + +//key/object config +void +ConfigNode::AddChild(const std::string& type_name, const ConfigNode& config) { + children_[type_name] = config; +} + +ConfigNode +ConfigNode::GetChild(const std::string& type_name) const { + auto ref = children_.find(type_name); + if(ref != children_.end()) { + return ref->second; + } + + ConfigNode nc; + return nc; +} + +ConfigNode& +ConfigNode::GetChild(const std::string &type_name) { + return children_[type_name]; +} + +void +ConfigNode::GetChildren(ConfigNodeArr& arr) const { + arr.clear(); + for(auto ref : children_){ + arr.push_back(ref.second); + } +} + +const std::map& +ConfigNode::GetChildren() const { + return children_; +} + +void ConfigNode::ClearChildren() { + children_.clear(); +} + +//key/sequence config +void +ConfigNode::AddSequenceItem(const std::string &key, const std::string &item) { + sequences_[key].push_back(item); +} + +std::vector +ConfigNode::GetSequence(const std::string &key) const { + auto itr = sequences_.find(key); + if(itr != sequences_.end()) { + return itr->second; + } else { + std::vector temp; + return temp; + } +} + +const std::map >& +ConfigNode::GetSequences() const { + return sequences_; +} + +void ConfigNode::ClearSequences() { + sequences_.clear(); +} + +void +ConfigNode::PrintAll(const std::string& prefix) const { + for(auto& elem : config_) { + CommonUtil::PrintInfo(prefix + elem.first + ": " + elem.second); + } + + for(auto& elem : sequences_) { + CommonUtil::PrintInfo(prefix + elem.first + ": "); + for(auto& str : elem.second) { + CommonUtil::PrintInfo(prefix + " - " + str); + } + } + + for(auto& elem : children_) { + CommonUtil::PrintInfo(prefix + elem.first + ": "); + elem.second.PrintAll(prefix + " "); + } +} + +std::string +ConfigNode::DumpString(const std::string &prefix) const { + std::stringstream str_buffer; + const std::string endl = "\n"; + for(auto& elem : config_) { + str_buffer << prefix << elem.first << ": " << elem.second << endl; + } + + for(auto& elem : sequences_) { + str_buffer << prefix << elem.first << ": " << endl; + for(auto& str : elem.second) { + str_buffer << prefix + " - " << str << endl; + } + } + + for(auto& elem : children_) { + str_buffer << prefix << elem.first << ": " << endl; + str_buffer << elem.second.DumpString(prefix + " ") << endl; + } + + return str_buffer.str(); +} + +} +} +} diff --git a/cpp/src/config/ConfigNode.h b/cpp/src/config/ConfigNode.h new file mode 100644 index 0000000000..a0a7d1d264 --- /dev/null +++ b/cpp/src/config/ConfigNode.h @@ -0,0 +1,63 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include +#include +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +class ConfigNode; +typedef std::vector ConfigNodeArr; + +class ConfigNode { + public: + void Combine(const ConfigNode& target); + + //key/value pair config + void SetValue(const std::string &key, const std::string &value); + + std::string GetValue(const std::string ¶m_key, const std::string &default_val = "") const; + bool GetBoolValue(const std::string ¶m_key, bool default_val = false) const; + int32_t GetInt32Value(const std::string ¶m_key, int32_t default_val = 0) const; + int64_t GetInt64Value(const std::string ¶m_key, int64_t default_val = 0) const; + float GetFloatValue(const std::string ¶m_key, float default_val = 0.0) const; + double GetDoubleValue(const std::string ¶m_key, double default_val = 0.0) const; + + const std::map& GetConfig() const; + void ClearConfig(); + + //key/object config + void AddChild(const std::string &type_name, const ConfigNode &config); + ConfigNode GetChild(const std::string &type_name) const; + ConfigNode& GetChild(const std::string &type_name); + void GetChildren(ConfigNodeArr &arr) const; + + const std::map& GetChildren() const; + void ClearChildren(); + + //key/sequence config + void AddSequenceItem(const std::string &key, const std::string &item); + std::vector GetSequence(const std::string &key) const; + + const std::map >& GetSequences() const; + void ClearSequences(); + + void PrintAll(const std::string &prefix = "") const; + std::string DumpString(const std::string &prefix = "") const; + + private: + std::map config_; + std::map children_; + std::map > sequences_; +}; + +} +} +} diff --git a/cpp/src/config/IConfigMgr.cpp b/cpp/src/config/IConfigMgr.cpp new file mode 100644 index 0000000000..42b01b608b --- /dev/null +++ b/cpp/src/config/IConfigMgr.cpp @@ -0,0 +1,20 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#include "IConfigMgr.h" +#include "YamlConfigMgr.h" + +namespace zilliz { +namespace vecwise { +namespace server { + +IConfigMgr * IConfigMgr::GetInstance() { + static YamlConfigMgr mgr; + return &mgr; +} + +} +} +} diff --git a/cpp/src/config/IConfigMgr.h b/cpp/src/config/IConfigMgr.h new file mode 100644 index 0000000000..31f6ed2e96 --- /dev/null +++ b/cpp/src/config/IConfigMgr.h @@ -0,0 +1,43 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include "utils/Error.h" +#include "ConfigNode.h" + +namespace zilliz { +namespace vecwise { +namespace server { + +// this class can parse nested config file and return config item +// config file example(yaml style) +// AAA: 1 +// BBB: +// CCC: hello +// DDD: 23.5 +// +// usage +// const IConfigMgr* mgr = IConfigMgr::GetInstance(); +// const ConfigNode& node = mgr->GetRootNode(); +// std::string val = node.GetValue("AAA"); // return '1' +// const ConfigNode& child = node.GetChild("BBB"); +// val = child.GetValue("CCC"); //return 'hello' + +class IConfigMgr { + public: + static IConfigMgr* GetInstance(); + + virtual ServerError LoadConfigFile(const std::string &filename) = 0; + virtual void Print() const = 0;//will be deleted + virtual std::string DumpString() const = 0; + + virtual const ConfigNode& GetRootNode() const = 0; + virtual ConfigNode& GetRootNode() = 0; +}; + +} +} +} diff --git a/cpp/src/config/YamlConfigMgr.cpp b/cpp/src/config/YamlConfigMgr.cpp new file mode 100644 index 0000000000..afcd0c74fa --- /dev/null +++ b/cpp/src/config/YamlConfigMgr.cpp @@ -0,0 +1,109 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#include "YamlConfigMgr.h" +#include "utils/CommonUtil.h" + +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +ServerError YamlConfigMgr::LoadConfigFile(const std::string &filename) { + struct stat directoryStat; + int statOK = stat(filename.c_str(), &directoryStat); + if (statOK != 0) { + CommonUtil::PrintError("File not found: " + filename); + return SERVER_UNEXPECTED_ERROR; + } + + try { + node_ = YAML::LoadFile(filename); + LoadConfigNode(node_, config_); + } + catch (YAML::Exception& e) { + CommonUtil::PrintError("Failed to load config file: " + std::string(e.what ())); + return SERVER_UNEXPECTED_ERROR; + } + + return SERVER_SUCCESS; +} + +void YamlConfigMgr::Print() const { + CommonUtil::PrintInfo("System config content:"); + config_.PrintAll(); +} + +std::string YamlConfigMgr::DumpString() const { + return config_.DumpString(""); +} + +const ConfigNode& YamlConfigMgr::GetRootNode() const { + return config_; +} + +ConfigNode& YamlConfigMgr::GetRootNode() { + return config_; +} + +bool +YamlConfigMgr::SetConfigValue(const YAML::Node& node, + const std::string& key, + ConfigNode& config) { + if(node[key].IsDefined ()) { + config.SetValue(key, node[key].as()); + return true; + } + return false; +} + +bool +YamlConfigMgr::SetChildConfig(const YAML::Node& node, + const std::string& child_name, + ConfigNode& config) { + if(node[child_name].IsDefined ()) { + ConfigNode sub_config; + LoadConfigNode(node[child_name], sub_config); + config.AddChild(child_name, sub_config); + return true; + } + return false; +} + +bool +YamlConfigMgr::SetSequence(const YAML::Node &node, + const std::string &child_name, + ConfigNode &config) { + if(node[child_name].IsDefined ()) { + size_t cnt = node[child_name].size(); + for(size_t i = 0; i < cnt; i++){ + config.AddSequenceItem(child_name, node[child_name][i].as()); + } + return true; + } + return false; +} + +void +YamlConfigMgr::LoadConfigNode(const YAML::Node& node, ConfigNode& config) { + std::string key; + for (YAML::const_iterator it = node.begin(); it != node.end(); ++it) { + if(!it->first.IsNull()){ + key = it->first.as(); + } + if(node[key].IsScalar()) { + SetConfigValue(node, key, config); + } else if(node[key].IsMap()){ + SetChildConfig(node, key, config); + } else if(node[key].IsSequence()){ + SetSequence(node, key, config); + } + } +} + +} +} +} diff --git a/cpp/src/config/YamlConfigMgr.h b/cpp/src/config/YamlConfigMgr.h new file mode 100644 index 0000000000..d68ec668a1 --- /dev/null +++ b/cpp/src/config/YamlConfigMgr.h @@ -0,0 +1,52 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include "IConfigMgr.h" +#include "ConfigNode.h" +#include "utils/Error.h" + +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +class YamlConfigMgr : public IConfigMgr { + public: + virtual ServerError LoadConfigFile(const std::string &filename); + virtual void Print() const; + virtual std::string DumpString() const; + + virtual const ConfigNode& GetRootNode() const; + virtual ConfigNode& GetRootNode(); + + private: + bool SetConfigValue(const YAML::Node& node, + const std::string& key, + ConfigNode& config); + + bool SetChildConfig(const YAML::Node& node, + const std::string &name, + ConfigNode &config); + + bool + SetSequence(const YAML::Node &node, + const std::string &child_name, + ConfigNode &config); + + void LoadConfigNode(const YAML::Node& node, ConfigNode& config); + + private: + YAML::Node node_; + ConfigNode config_; +}; + +} +} +} + + diff --git a/cpp/src/main.cpp b/cpp/src/main.cpp index 41c259a45b..5e3b13a181 100644 --- a/cpp/src/main.cpp +++ b/cpp/src/main.cpp @@ -3,7 +3,98 @@ // Unauthorized copying of this file, via any medium is strictly prohibited. // Proprietary and confidential. //////////////////////////////////////////////////////////////////////////////// +#include "server/Server.h" -void test() { - return ; -} \ No newline at end of file +#include +#include +#include +#include +#include + +#include "utils/SignalUtil.h" +#include "utils/CommonUtil.h" + +void print_help(const std::string &app_name); + +using namespace zilliz::vecwise; + +int +main(int argc, char *argv[]) { + server::CommonUtil::PrintInfo("Vecwise engine server start"); + +// zilliz::lib::gpu::InitMemoryAllocator(); + + signal(SIGSEGV, server::SignalUtil::HandleSignal); + signal(SIGUSR1, server::SignalUtil::HandleSignal); + signal(SIGUSR2, server::SignalUtil::HandleSignal); + + std::string app_name = basename(argv[0]); + static struct option long_options[] = {{"conf_file", required_argument, 0, 'c'}, + {"help", no_argument, 0, 'h'}, + {"daemon", no_argument, 0, 'd'}, + {"pid_file", required_argument, 0, 'p'}, + {NULL, 0, 0, 0}}; + + int option_index = 0; + int64_t start_daemonized = 0; +// int pid_fd; + + std::string config_filename; + std::string pid_filename; + + app_name = argv[0]; + +// if(argc < 5) { +// print_help(app_name); +// return EXIT_FAILURE; +// } + + int value; + while ((value = getopt_long(argc, argv, "c:p:dh", long_options, &option_index)) != -1) { + switch (value) { + case 'c': { + char *config_filename_ptr = strdup(optarg); + config_filename = config_filename_ptr; + free(config_filename_ptr); + printf("Loading configuration from: %s\n", config_filename.c_str()); + break; + } + + case 'p': { + char *pid_filename_ptr = strdup(optarg); + pid_filename = pid_filename_ptr; + free(pid_filename_ptr); + printf("%s\n", pid_filename.c_str()); + break; + } + + case 'd': + start_daemonized = 1; + break; + case 'h': + print_help(app_name); + return EXIT_SUCCESS; + case '?': + print_help(app_name); + return EXIT_FAILURE; + default: + print_help(app_name); + break; + } + } + + server::Server* server_ptr = server::Server::Instance(); + server_ptr->Init(start_daemonized, pid_filename, config_filename); + return server_ptr->Start(); +} + +void +print_help(const std::string &app_name) { + printf("\n Usage: %s [OPTIONS]\n\n", app_name.c_str()); + printf(" Options:\n"); + printf(" -h --help Print this help\n"); + printf(" -c --conf_file filename Read configuration from the file\n"); + printf(" -d --daemon Daemonize this application\n"); + printf(" -p --pid_file filename PID file used by daemonized app\n"); + printf("\n"); +} diff --git a/cpp/src/server/Server.cpp b/cpp/src/server/Server.cpp new file mode 100644 index 0000000000..8e9a68c831 --- /dev/null +++ b/cpp/src/server/Server.cpp @@ -0,0 +1,227 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include "Server.h" +#include "utils/CommonUtil.h" +#include "utils/SignalUtil.h" +#include "utils/TimeRecorder.h" + +#include +#include +#include +#include +#include +#include +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +Server* +Server::Instance() { + static Server server; + return &server; +} + +Server::Server() + : opt_config_ptr_(nullptr){ + +} +Server::~Server() { + +} + +void +Server::Init(int64_t daemonized, const std::string& pid_filename, const std::string& config_filename) { + daemonized_ = daemonized; + pid_filename_ = pid_filename; + config_filename_ = config_filename; +} + +void +Server::Daemonize() { + if (daemonized_ == 0) { + return; + } + + CommonUtil::PrintInfo("Megawise server run in daemonize mode"); + +// std::string log_path(GetLogDirFullPath()); +// log_path += "zdb_server.(INFO/WARNNING/ERROR/CRITICAL)"; +// CommonUtil::PrintInfo("Log will be exported to: " + log_path); + + pid_t pid = 0; + + // Fork off the parent process + pid = fork(); + + // An error occurred + if (pid < 0) { + exit(EXIT_FAILURE); + } + + // Success: terminate parent + if (pid > 0) { + exit(EXIT_SUCCESS); + } + + // On success: The child process becomes session leader + if (setsid() < 0) { + exit(EXIT_FAILURE); + } + + // Ignore signal sent from child to parent process + signal(SIGCHLD, SIG_IGN); + + // Fork off for the second time + pid = fork(); + + // An error occurred + if (pid < 0) { + exit(EXIT_FAILURE); + } + + // Terminate the parent + if (pid > 0) { + exit(EXIT_SUCCESS); + } + + // Set new file permissions + umask(0); + + // Change the working directory to root + int ret = chdir("/"); + if(ret != 0){ + return; + } + + // Close all open fd + for (long fd = sysconf(_SC_OPEN_MAX); fd > 0; fd--) { + close(fd); + } + + CommonUtil::PrintInfo("Redirect stdin/stdout/stderr to /dev/null"); + + // Redirect stdin/stdout/stderr to /dev/null + stdin = fopen("/dev/null", "r"); + stdout = fopen("/dev/null", "w+"); + stderr = fopen("/dev/null", "w+"); + // Try to write PID of daemon to lockfile + if (!pid_filename_.empty()) { + pid_fd = open(pid_filename_.c_str(), O_RDWR | O_CREAT, 0640); + if (pid_fd < 0) { + CommonUtil::PrintInfo("Can't open filename: " + pid_filename_ + ", Error: " + strerror(errno)); + exit(EXIT_FAILURE); + } + if (lockf(pid_fd, F_TLOCK, 0) < 0) { + CommonUtil::PrintInfo("Can't lock filename: " + pid_filename_ + ", Error: " + strerror(errno)); + exit(EXIT_FAILURE); + } + + std::string pid_file_context = std::to_string(getpid()); + ssize_t res = write(pid_fd, pid_file_context.c_str(), pid_file_context.size()); + if(res != 0){ + return; + } + } +} + +int +Server::Start() { + if (daemonized_) { + Daemonize(); + } + + do { + try { + // Read config file + if(LoadConfig() != SERVER_SUCCESS) { + return 1; + } + + //log path is defined by LoadConfig, so InitLog must be called after LoadConfig + ServerConfig *config = ServerConfig::GetInstance(); + ConfigNode server_config = config->GetServerConfig(); + + //print config into console and log + opt_config_ptr_->PrintAll(); + + // Handle Signal + signal(SIGINT, SignalUtil::HandleSignal); + signal(SIGHUP, SignalUtil::HandleSignal); + signal(SIGTERM, SignalUtil::HandleSignal); + + StartService(); + + CommonUtil::PrintInfo("Megawise server is running..."); + + } catch(std::exception& ex){ + std::string info = "Megawise server encounter exception: " + std::string(ex.what()); + CommonUtil::PrintError(info); + + CommonUtil::PrintInfo("Is another server instance running?"); + break; + } + } while(false); + + Stop(); + return 0; +} + +void +Server::Stop() { + CommonUtil::PrintInfo("Megawise server will be closed"); + + // Unlock and close lockfile + if (pid_fd != -1) { + int ret = lockf(pid_fd, F_ULOCK, 0); + if(ret != 0){ + + } + ret = close(pid_fd); + if(ret != 0){ + + } + } + + // Try to delete lockfile + if (!pid_filename_.empty()) { + int ret = unlink(pid_filename_.c_str()); + if(ret != 0){ + + } + } + + running_ = 0; + + StopService(); + + + CommonUtil::PrintInfo("Megawise server closed"); +} + + +ServerError +Server::LoadConfig() { + opt_config_ptr_ = ServerConfig::GetInstance(); + opt_config_ptr_->LoadConfigFile(config_filename_); + + return SERVER_SUCCESS; +} + +void +Server::StartService() { + +} + +void +Server::StopService() { + +} + +} +} +} diff --git a/cpp/src/server/Server.h b/cpp/src/server/Server.h new file mode 100644 index 0000000000..b6ebae1e5b --- /dev/null +++ b/cpp/src/server/Server.h @@ -0,0 +1,49 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include "ServerConfig.h" +#include "utils/Error.h" + +#include +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +class Server { + public: + static Server* Instance(); + + void Init(int64_t daemonized, const std::string& pid_filename, const std::string& config_filename); + int Start(); + void Stop(); + + private: + Server(); + ~Server(); + + void Daemonize(); + + static void HandleSignal(int signal); + ServerError LoadConfig(); + + void StartService(); + void StopService(); + + private: + int64_t daemonized_ = 0; + int64_t running_ = 1; + int pid_fd = -1; + std::string pid_filename_; + std::string config_filename_; + ServerConfig* opt_config_ptr_ = nullptr; +}; // Server + +} // server +} // sql +} // zilliz diff --git a/cpp/src/server/ServerConfig.cpp b/cpp/src/server/ServerConfig.cpp new file mode 100644 index 0000000000..09f4c752dd --- /dev/null +++ b/cpp/src/server/ServerConfig.cpp @@ -0,0 +1,94 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include "ServerConfig.h" + +#include +#include +#include +#include +#include + +#include "config/IConfigMgr.h" + +namespace zilliz { +namespace vecwise { +namespace server { + +ServerConfig* +ServerConfig::GetInstance() { + static ServerConfig config; + return &config; +} + +ServerError +ServerConfig::LoadConfigFile(const std::string& config_filename) { + std::string filename = config_filename; + if(filename.empty()){ + std::cout << "ERROR: a config file is required" << std::endl; + exit(1);//directly exit program if config file not specified + } + struct stat directoryStat; + int statOK = stat(filename.c_str(), &directoryStat); + if (statOK != 0) { + std::cout << "ERROR: " << filename << " not found!" << std::endl; + exit(1);//directly exit program if config file not found + } + + try { + IConfigMgr* mgr = const_cast(IConfigMgr::GetInstance()); + ServerError err = mgr->LoadConfigFile(filename); + if(err != 0) { + std::cout << "Server failed to load config file" << std::endl; + exit(1);//directly exit program if the config file is illegal + } + } + catch (YAML::Exception& e) { + std::cout << "Server failed to load config file: " << std::endl; + return SERVER_UNEXPECTED_ERROR; + } + + return SERVER_SUCCESS; +} + +void +ServerConfig::PrintAll() const { + if(const IConfigMgr* mgr = IConfigMgr::GetInstance()) { + std::string str = mgr->DumpString(); +// SERVER_LOG_INFO << "\n" << str; + std::cout << "\n" << str << std::endl; + } +} + +ConfigNode +ServerConfig::GetServerConfig() const { + const IConfigMgr* mgr = IConfigMgr::GetInstance(); + const ConfigNode& root_node = mgr->GetRootNode(); + return root_node.GetChild(CONFIG_SERVER); +} + +ConfigNode& +ServerConfig::GetServerConfig() { + IConfigMgr* mgr = IConfigMgr::GetInstance(); + ConfigNode& root_node = mgr->GetRootNode(); + return root_node.GetChild(CONFIG_SERVER); +} + +std::string +ServerConfig::GetServerAddress() const { + ConfigNode server_config = GetServerConfig(); + return server_config.GetValue(CONFIG_ADDRESS); +} + +std::string +ServerConfig::GetServerPort() const { + ConfigNode server_config = GetServerConfig(); + return server_config.GetValue(CONFIG_PORT); +} + + +} +} +} diff --git a/cpp/src/server/ServerConfig.h b/cpp/src/server/ServerConfig.h new file mode 100644 index 0000000000..4277d66066 --- /dev/null +++ b/cpp/src/server/ServerConfig.h @@ -0,0 +1,39 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include "utils/Error.h" +#include "config/ConfigNode.h" + +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +static const std::string CONFIG_SERVER = "server_config"; +static const std::string CONFIG_ADDRESS = "address"; +static const std::string CONFIG_PORT = "port"; + + +class ServerConfig { + public: + static ServerConfig *GetInstance(); + + ServerError LoadConfigFile(const std::string& config_filename); + void PrintAll() const; + + ConfigNode GetServerConfig() const; + ConfigNode& GetServerConfig(); + + std::string GetServerAddress() const; + std::string GetServerPort() const; +}; + +} +} +} + diff --git a/cpp/src/utils/CommonUtil.cpp b/cpp/src/utils/CommonUtil.cpp new file mode 100644 index 0000000000..da5d011e68 --- /dev/null +++ b/cpp/src/utils/CommonUtil.cpp @@ -0,0 +1,145 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include "CommonUtil.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "boost/filesystem.hpp" + +#if defined(__x86_64__) +#define THREAD_MULTIPLY_CPU 1 +#elif defined(__powerpc64__) +#define THREAD_MULTIPLY_CPU 4 +#else +#define THREAD_MULTIPLY_CPU 1 +#endif + +namespace zilliz { +namespace vecwise { +namespace server { + +namespace fs = boost::filesystem; + +void CommonUtil::PrintInfo(const std::string& info){ +// SERVER_LOG_INFO << info; + std::cout << info << std::endl; +} + +void CommonUtil::PrintError(const std::string& info){ +// SERVER_LOG_ERROR << info; + std::cout << info << std::endl; +} + +bool CommonUtil::GetSystemMemInfo(unsigned long &totalMem, unsigned long &freeMem) { + struct sysinfo info; + int ret = sysinfo(&info); + totalMem = info.totalram; + freeMem = info.freeram; + + return ret == 0;//succeed 0, failed -1 +} + +bool CommonUtil::GetSystemAvailableThreads(unsigned int &threadCnt) { + //threadCnt = std::thread::hardware_concurrency(); + threadCnt = sysconf(_SC_NPROCESSORS_CONF); + threadCnt *= THREAD_MULTIPLY_CPU; + if (threadCnt == 0) + threadCnt = 8; + + return true; +} + +bool CommonUtil::IsDirectoryExit(const std::string &path) +{ + DIR *dp = nullptr; + if ((dp = opendir(path.c_str())) == nullptr) { + return false; + } + + closedir(dp); + return true; +} + +ServerError CommonUtil::CreateDirectory(const std::string &path) { + struct stat directoryStat; + int statOK = stat(path.c_str(), &directoryStat); + if (statOK == 0) { + return SERVER_SUCCESS;//already exist + } + + fs::path fs_path(path); + fs::path parent_path = fs_path.parent_path(); + ServerError err = CreateDirectory(parent_path.string()); + if(err != SERVER_SUCCESS){ + return err; + } + + statOK = stat(path.c_str(), &directoryStat); + if (statOK == 0) { + return SERVER_SUCCESS;//already exist + } + + int makeOK = mkdir(path.c_str(), S_IRWXU|S_IRGRP|S_IROTH); + if (makeOK != 0) { + return SERVER_UNEXPECTED_ERROR; + } + + return SERVER_SUCCESS; +} + +void RemoveDirectory(const std::string &path) { + DIR *pDir = NULL; + struct dirent *dmsg; + char szFileName[256]; + char szFolderName[256]; + + strcpy(szFolderName, path.c_str()); + strcat(szFolderName, "/%s"); + if ((pDir = opendir(path.c_str())) != NULL) { + while ((dmsg = readdir(pDir)) != NULL) { + if (strcmp(dmsg->d_name, ".") != 0 + && strcmp(dmsg->d_name, "..") != 0) { + sprintf(szFileName, szFolderName, dmsg->d_name); + std::string tmp = szFileName; + if (tmp.find(".") == std::string::npos) { + RemoveDirectory(szFileName); + } + remove(szFileName); + } + } + } + + if (pDir != NULL) { + closedir(pDir); + } + remove(path.c_str()); +} + +ServerError DeleteDirectory(const std::string &path) { + struct stat directoryStat; + int statOK = stat(path.c_str(), &directoryStat); + if (statOK != 0) + return SERVER_SUCCESS; + + RemoveDirectory(path); + return SERVER_SUCCESS; +} + +bool IsFileExist(const std::string &path) { + return (access(path.c_str(), F_OK) == 0); +} + + +} +} +} diff --git a/cpp/src/utils/CommonUtil.h b/cpp/src/utils/CommonUtil.h new file mode 100755 index 0000000000..9ac96fd333 --- /dev/null +++ b/cpp/src/utils/CommonUtil.h @@ -0,0 +1,33 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include + +#include "Error.h" + +namespace zilliz { +namespace vecwise { +namespace server { + +class CommonUtil { + public: + static void PrintInfo(const std::string& info); + static void PrintError(const std::string& info); + + static bool GetSystemMemInfo(unsigned long &totalMem, unsigned long &freeMem); + static bool GetSystemAvailableThreads(unsigned int &threadCnt); + + static bool IsFileExist(const std::string &path); + static bool IsDirectoryExit(const std::string &path); + static ServerError CreateDirectory(const std::string &path); + static ServerError DeleteDirectory(const std::string &path); +}; + +} +} +} + diff --git a/cpp/src/utils/Error.h b/cpp/src/utils/Error.h new file mode 100644 index 0000000000..6f3de0adaa --- /dev/null +++ b/cpp/src/utils/Error.h @@ -0,0 +1,35 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +using ServerError = int32_t; + +constexpr ServerError SERVER_SUCCESS = 0; + +constexpr ServerError SERVER_ERROR_CODE_BASE = 0x30000; + +constexpr ServerError +ToGlobalServerErrorCode(const ServerError error_code) { + return SERVER_ERROR_CODE_BASE + SERVER_ERROR_CODE_BASE; +} + +constexpr ServerError SERVER_UNEXPECTED_ERROR = ToGlobalServerErrorCode(0x001); +constexpr ServerError SERVER_UNSUPPORTED_ERROR = ToGlobalServerErrorCode(0x002); +constexpr ServerError SERVER_NULL_POINTER = ToGlobalServerErrorCode(0x003); +constexpr ServerError SERVER_INVALID_ARGUMENT = ToGlobalServerErrorCode(0x004); +constexpr ServerError SERVER_FILE_NOT_FOUND = ToGlobalServerErrorCode(0x005); +constexpr ServerError SERVER_NOT_IMPLEMENT = ToGlobalServerErrorCode(0x006); + +} // namespace server +} // namespace vecwise +} // namespace zilliz + diff --git a/cpp/src/utils/SignalUtil.cpp b/cpp/src/utils/SignalUtil.cpp new file mode 100644 index 0000000000..ef83b6c115 --- /dev/null +++ b/cpp/src/utils/SignalUtil.cpp @@ -0,0 +1,62 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include "SignalUtil.h" +#include "CommonUtil.h" +#include "server/Server.h" + +#include +#include + +namespace zilliz { +namespace vecwise { +namespace server { + +void SignalUtil::HandleSignal(int signum){ + CommonUtil::PrintInfo("Server received signal:" + std::to_string(signum)); + + switch(signum){ + case SIGUSR2:{ + server::Server* server_ptr = server::Server::Instance(); + server_ptr->Stop(); + + exit(0); + + } + default:{ + SignalUtil::PrintStacktrace(); + + std::string info = "Server encounter critical signal:"; + info += std::to_string(signum); +// SendSignalMessage(signum, info); + + CommonUtil::PrintInfo(info); + + server::Server* server_ptr = server::Server::Instance(); + server_ptr->Stop(); + + exit(1); + } + } +} + +void SignalUtil::PrintStacktrace() { + CommonUtil::PrintInfo("Call stack:"); + + const int size = 32; + void* array[size]; + int stack_num = backtrace(array, size); + char ** stacktrace = backtrace_symbols(array, stack_num); + for (int i = 0; i < stack_num; ++i) { + std::string info = stacktrace[i]; + CommonUtil::PrintInfo(info); + } + free(stacktrace); +} + + +} +} +} diff --git a/cpp/src/utils/SignalUtil.h b/cpp/src/utils/SignalUtil.h new file mode 100644 index 0000000000..d713d9caa2 --- /dev/null +++ b/cpp/src/utils/SignalUtil.h @@ -0,0 +1,20 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +namespace zilliz { +namespace vecwise { +namespace server { + +class SignalUtil { + public: + static void HandleSignal(int signum); + static void PrintStacktrace(); +}; + +} +} +} diff --git a/cpp/src/utils/ThreadPool.h b/cpp/src/utils/ThreadPool.h new file mode 100644 index 0000000000..04439ba766 --- /dev/null +++ b/cpp/src/utils/ThreadPool.h @@ -0,0 +1,118 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include +#include +#include +#include +#include +#include +#include +#include +#include + + +#define MAX_THREADS_NUM 32 + +namespace zilliz { +namespace sql { +namespace storage { + +class ThreadPool { +public: + ThreadPool(size_t threads, size_t queue_size = 1000); + + template + auto enqueue(F &&f, Args &&... args) + -> std::future::type>; + + ~ThreadPool(); + +private: + // need to keep track of threads so we can join them + std::vector workers; + + // the task queue + std::queue > tasks; + + size_t max_queue_size; + + // synchronization + std::mutex queue_mutex; + + std::condition_variable condition; + + bool stop; +}; + + +// the constructor just launches some amount of workers +inline ThreadPool::ThreadPool(size_t threads, size_t queue_size) + : max_queue_size(queue_size), stop(false) { + for (size_t i = 0; i < threads; ++i) + workers.emplace_back( + [this] { + for (;;) { + std::function task; + + { + std::unique_lock lock(this->queue_mutex); + this->condition.wait(lock, + [this] { return this->stop || !this->tasks.empty(); }); + if (this->stop && this->tasks.empty()) + return; + task = std::move(this->tasks.front()); + this->tasks.pop(); + } + this->condition.notify_all(); + + task(); + } + } + ); +} + +// add new work item to the pool +template +auto ThreadPool::enqueue(F &&f, Args &&... args) +-> std::future::type> { + using return_type = typename std::result_of::type; + + auto task = std::make_shared >( + std::bind(std::forward(f), std::forward(args)...) + ); + + std::future res = task->get_future(); + { + std::unique_lock lock(queue_mutex); + this->condition.wait(lock, + [this] { return this->tasks.size() < max_queue_size; }); + // don't allow enqueueing after stopping the pool + if (stop) + throw std::runtime_error("enqueue on stopped ThreadPool"); + + tasks.emplace([task]() { (*task)(); }); + } + condition.notify_all(); + return res; +} + +// the destructor joins all threads +inline ThreadPool::~ThreadPool() { + { + std::unique_lock lock(queue_mutex); + stop = true; + } + condition.notify_all(); + for (std::thread &worker: workers) + worker.join(); +} + +} +} +} + diff --git a/cpp/src/utils/TimeRecorder.cpp b/cpp/src/utils/TimeRecorder.cpp new file mode 100644 index 0000000000..da768a3cff --- /dev/null +++ b/cpp/src/utils/TimeRecorder.cpp @@ -0,0 +1,144 @@ +//////////////////////////////////////////////////////////////////////////////// +// Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved +// Unauthorized copying of this file, via any medium is strictly prohibited. +// Proprietary and confidential. +//////////////////////////////////////////////////////////////////////////////// +#include "TimeRecorder.h" +#include "utils/CommonUtil.h" + + +namespace zilliz { +namespace vecwise { +namespace server { + +TimeRecorder::TimeRecorder(const std::string &header, + TimeRecorder::TimeDisplayUnit unit, + int64_t log_level) : + header_(header), + time_unit_(unit), + log_level_(log_level) { + start_ = last_ = stdclock::now(); + span_ = 0.0; +} + +std::string +TimeRecorder::GetTimeSpanStr(TimeRecorder::TimeDisplayUnit &unit, double span) const { + std::string spanStr; + std::string unitStr; + + switch (unit) { + case TimeRecorder::eTimeAutoUnit: { + if (span >= 1000000) { + int64_t t = (int64_t) span; + int64_t hour, minute; + double second; + hour = t / 1000000 / 3600; + t -= hour * 3600 * 1000000; + minute = t / 1000000 / 60; + t -= minute * 60 * 1000000; + second = t * 0.000001; + spanStr += (hour < 10 ? "0" : "") + std::to_string(hour) + ":"; + spanStr += (minute < 10 ? "0" : "") + std::to_string(minute) + ":"; + spanStr += (second < 10 ? "0" : "") + std::to_string(second); + unitStr = ""; + } else if (span >= 1000) { + spanStr = std::to_string(span * 0.001); + unitStr = " ms"; + } else { + spanStr = std::to_string(span); + unitStr = " us"; + } + } + break; + case TimeRecorder::eTimeHourUnit: + spanStr = std::to_string((span * 0.000001) / 3600); + unitStr = " hour"; + break; + case TimeRecorder::eTimeMinuteUnit: + spanStr = std::to_string((span * 0.000001) / 60); + unitStr = " min"; + break; + case TimeRecorder::eTimeSecondUnit: + spanStr = std::to_string(span * 0.000001); + unitStr = " sec"; + break; + case TimeRecorder::eTimeMilliSecUnit: + spanStr = std::to_string(span * 0.001); + unitStr = " ms"; + break; + case TimeRecorder::eTimeMicroSecUnit: + default: + spanStr = std::to_string(span); + unitStr = " us"; + break; + } + + return spanStr + unitStr; +} + +void +TimeRecorder::PrintTimeRecord(const std::string &msg, double span) { + std::string strLog; + if (!header_.empty()) strLog += header_ + ": "; + strLog += msg; + strLog += " ("; + strLog += GetTimeSpanStr(time_unit_, span); + strLog += ")"; + + switch (log_level_) { + case 0: { + CommonUtil::PrintInfo(strLog); + break; + } + case 1: { + CommonUtil::PrintInfo(strLog); + break; + } + case 2: { + CommonUtil::PrintInfo(strLog); + break; + } + case 3: { + CommonUtil::PrintInfo(strLog); + break; + } + case 4: { + CommonUtil::PrintInfo(strLog); + break; + } + case 5: { + CommonUtil::PrintInfo(strLog); + break; + } + default: { + CommonUtil::PrintInfo(strLog); + break; + } + } +} + +void +TimeRecorder::Record(const std::string &msg) { + stdclock::time_point curr = stdclock::now(); + span_ = (std::chrono::duration(curr - last_)).count(); + last_ = curr; + + PrintTimeRecord(msg, span_); +} + +void +TimeRecorder::Elapse(const std::string &msg) { + stdclock::time_point curr = stdclock::now(); + span_ = (std::chrono::duration(curr - start_)).count(); + + PrintTimeRecord(msg, span_); +} + +double +TimeRecorder::Span() { + return span_; +} + +} +} +} diff --git a/cpp/src/utils/TimeRecorder.h b/cpp/src/utils/TimeRecorder.h new file mode 100644 index 0000000000..676dc185ea --- /dev/null +++ b/cpp/src/utils/TimeRecorder.h @@ -0,0 +1,55 @@ +/******************************************************************************* + * Copyright 上海赜睿信息科技有限公司(Zilliz) - All Rights Reserved + * Unauthorized copying of this file, via any medium is strictly prohibited. + * Proprietary and confidential. + ******************************************************************************/ +#pragma once + +#include +#include + + +namespace zilliz { +namespace vecwise { +namespace server { + +class TimeRecorder { + using stdclock = std::chrono::high_resolution_clock; + +public: + enum TimeDisplayUnit { + eTimeAutoUnit = 0, + eTimeHourUnit, + eTimeMinuteUnit, + eTimeSecondUnit, + eTimeMilliSecUnit, + eTimeMicroSecUnit, + }; + + TimeRecorder(const std::string &header, + TimeRecorder::TimeDisplayUnit unit = TimeRecorder::eTimeAutoUnit, + int64_t log_level = 1); //trace = 0, debug = 1, info = 2, warn = 3, error = 4, critical = 5 + + void Record(const std::string &msg); + + void Elapse(const std::string &msg); + + double Span(); + +private: + std::string GetTimeSpanStr(TimeRecorder::TimeDisplayUnit &unit, double span) const; + + void PrintTimeRecord(const std::string &msg, double span); + +private: + std::string header_; + TimeRecorder::TimeDisplayUnit time_unit_; + stdclock::time_point start_; + stdclock::time_point last_; + double span_; + int64_t log_level_; +}; + +} +} +}