From ce6dc3d549bc58515d5d55db1d0a009a92887f3c Mon Sep 17 00:00:00 2001 From: David Robillard Date: Sun, 18 Sep 2016 04:42:31 -0400 Subject: Add fancy communication logging --- ingen/Tee.hpp | 137 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 137 insertions(+) create mode 100644 ingen/Tee.hpp (limited to 'ingen/Tee.hpp') diff --git a/ingen/Tee.hpp b/ingen/Tee.hpp new file mode 100644 index 00000000..1577f6dd --- /dev/null +++ b/ingen/Tee.hpp @@ -0,0 +1,137 @@ +/* + This file is part of Ingen. + Copyright 2007-2016 David Robillard + + Ingen is free software: you can redistribute it and/or modify it under the + terms of the GNU Affero General Public License as published by the Free + Software Foundation, either version 3 of the License, or any later version. + + Ingen 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 Affero General Public License for details. + + You should have received a copy of the GNU Affero General Public License + along with Ingen. If not, see . +*/ + +#ifndef INGEN_ENGINE_TEE_HPP +#define INGEN_ENGINE_TEE_HPP + +#include +#include +#include + +#include "ingen/Interface.hpp" +#include "ingen/types.hpp" + +namespace Ingen { + +/** Interface that forwards all calls to several sinks. */ +class Tee : public Interface +{ +public: + typedef std::set< SPtr > Sinks; + + Tee(const Sinks& sinks = {}) + : _sinks(sinks) + {} + + void add_sink(SPtr sink) { + std::lock_guard lock(_sinks_mutex); + _sinks.insert(sink); + } + + bool remove_sink(SPtr sink) { + std::lock_guard lock(_sinks_mutex); + return (_sinks.erase(sink) > 0); + } + + virtual SPtr respondee() const { + return (*_sinks.begin())->respondee(); + } + + virtual void set_respondee(SPtr respondee) { + (*_sinks.begin())->set_respondee(respondee); + } + +#define BROADCAST(msg, ...) \ + std::lock_guard lock(_sinks_mutex); \ + for (const auto& s : _sinks) { \ + s->msg(__VA_ARGS__); \ + } + + void bundle_begin() { BROADCAST(bundle_begin); } + void bundle_end() { BROADCAST(bundle_end); } + + void put(const Raul::URI& uri, + const Resource::Properties& properties, + Resource::Graph ctx=Resource::Graph::DEFAULT) { + BROADCAST(put, uri, properties); + } + + void delta(const Raul::URI& uri, + const Resource::Properties& remove, + const Resource::Properties& add) { + BROADCAST(delta, uri, remove, add); + } + + void copy(const Raul::URI& old_uri, + const Raul::URI& new_uri) { + BROADCAST(copy, old_uri, new_uri); + } + + void move(const Raul::Path& old_path, + const Raul::Path& new_path) { + BROADCAST(move, old_path, new_path); + } + + void del(const Raul::URI& uri) { BROADCAST(del, uri); } + + void connect(const Raul::Path& tail, + const Raul::Path& head) { + BROADCAST(connect, tail, head); + } + + void disconnect(const Raul::Path& tail, + const Raul::Path& head) { + BROADCAST(disconnect, tail, head); + } + + void disconnect_all(const Raul::Path& graph, + const Raul::Path& path) { + BROADCAST(disconnect_all, graph, path); + } + + void set_property(const Raul::URI& subject, + const Raul::URI& predicate, + const Atom& value) { + BROADCAST(set_property, subject, predicate, value); + } + + void undo() { BROADCAST(undo); } + void redo() { BROADCAST(redo); } + + void set_response_id(int32_t id) { BROADCAST(set_response_id, id); } + + void get(const Raul::URI& uri) { BROADCAST(get, uri); } + + void response(int32_t id, Status status, const std::string& subject) { + BROADCAST(response, id, status, subject); + } + + void error(const std::string& msg) { BROADCAST(error, msg); } + +#undef BROADCAST + + Raul::URI uri() const { return Raul::URI("ingen:/tee"); } + const Sinks& sinks() const { return _sinks; } + Sinks& sinks() { return _sinks; } + +private: + std::mutex _sinks_mutex; + Sinks _sinks; +}; + +} // namespace Ingen + +#endif // INGEN_ENGINE_TEE_HPP -- cgit v1.2.1