// // Copyright 2017 Ettus Research (National Instruments) // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // #ifndef INCLUDED_UTILS_RPC_HPP #define INCLUDED_UTILS_RPC_HPP #include #include #include #include namespace uhd { /*! Abstraction for RPC client * * Purpose of this class is to wrap the underlying RPC implementation. * This class holds a connection to an RPC server (the connection is severed on * destruction). */ class rpc_client { public: using sptr = std::shared_ptr; static sptr make(std::string const& addr, uint16_t port) { return std::make_shared(addr, port); } /*! * \param addr An IP address to connect to * \param port Port to connect to */ rpc_client(std::string const& addr, uint16_t port) : _client(addr, port) {} /*! Perform an RPC request. * * Thread safe (locked). This function blocks until it receives a valid * response from the server. * * \param func_name The function name that is called via RPC * \param args All these arguments are passed to the RPC call * * \throws uhd::runtime_error in case of failure */ template return_type request(std::string const& func_name, Args&&... args) { std::lock_guard lock(_mutex); try { return _client.call(func_name, std::forward(args)...) .template as(); } catch (const ::rpc::rpc_error &ex) { throw uhd::runtime_error(str( boost::format("Error during RPC call to `%s'. Error message: %s") % func_name % ex.what() )); } catch (const std::bad_cast& ex) { throw uhd::runtime_error(str( boost::format("Error during RPC call to `%s'. Error message: %s") % func_name % ex.what() )); } }; /*! Perform an RPC notification. * * Thread safe (locked). This function does not require a response from the * server, although the underlying implementation may provide one. * * \param func_name The function name that is called via RPC * \param args All these arguments are passed to the RPC call * * \throws uhd::runtime_error in case of failure */ template void notify(std::string const& func_name, Args&&... args) { std::lock_guard lock(_mutex); try { _client.call(func_name, std::forward(args)...); } catch (const ::rpc::rpc_error &ex) { throw uhd::runtime_error(str( boost::format("Error during RPC call to `%s'. Error message: %s") % func_name % ex.what() )); } catch (const std::bad_cast& ex) { throw uhd::runtime_error(str( boost::format("Error during RPC call to `%s'. Error message: %s") % func_name % ex.what() )); } }; /*! Like request(), also provides a token. * * This is a convenience wrapper to directly call a function that requires * a token without having to have a copy of the token. */ template return_type request_with_token(std::string const& func_name, Args&&... args) { return request(func_name, _token, std::forward(args)...); }; /*! Like notify(), also provides a token. * * This is a convenience wrapper to directly call a function that requires * a token without having to have a copy of the token. */ template void notify_with_token(std::string const& func_name, Args&&... args) { notify(func_name, _token, std::forward(args)...); }; /*! Sets the token value. This is used by the `_with_token` methods. */ void set_token(const std::string &token) { _token = token; } void set_timeout(size_t timeout_ms) { _client.set_timeout(timeout_ms); } private: std::string _token; std::mutex _mutex; ::rpc::client _client; }; } /* namespace uhd */ #endif /* INCLUDED_UTILS_RPC_HPP */