//------------------------------------------------------------------------------ /* This file is part of clio: https://github.com/XRPLF/clio Copyright (c) 2024, the clio developers. Permission to use, copy, modify, and distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ //============================================================================== #pragma once #include "util/Mutex.hpp" #include #include #include #include #include #include #include namespace util { /** * @brief Cache of requests' responses with TTL support and configurable cacheable commands */ class ResponseExpirationCache { /** * @brief A class to store a cache entry. */ class Entry { std::chrono::steady_clock::time_point lastUpdated_; std::optional response_; public: /** * @brief Put a response into the cache * * @param response The response to store */ void put(boost::json::object response); /** * @brief Get the response from the cache * * @return The response */ std::optional get() const; /** * @brief Get the last time the cache was updated * * @return The last time the cache was updated */ std::chrono::steady_clock::time_point lastUpdated() const; /** * @brief Invalidate the cache entry */ void invalidate(); }; std::chrono::steady_clock::duration cacheTimeout_; std::unordered_map> cache_; bool shouldCache(std::string const& cmd); public: /** * @brief Construct a new Cache object * * @param cacheTimeout The time for cache entries to expire * @param cmds The commands that should be cached */ ResponseExpirationCache( std::chrono::steady_clock::duration cacheTimeout, std::unordered_set const& cmds ) : cacheTimeout_(cacheTimeout) { for (auto const& command : cmds) { cache_.emplace(command, Entry{}); } } /** * @brief Get a response from the cache * * @param cmd The command to get the response for * @return The response if it exists or std::nullopt otherwise */ [[nodiscard]] std::optional get(std::string const& cmd) const; /** * @brief Put a response into the cache if the request should be cached * * @param cmd The command to store the response for * @param response The response to store */ void put(std::string const& cmd, boost::json::object const& response); /** * @brief Invalidate all entries in the cache */ void invalidate(); }; } // namespace util