/* This file is part of Ingen. Copyright 2007-2012 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 . */ #include #include "raul/log.hpp" #include "ingen/shared/URIs.hpp" #include "ClientBroadcaster.hpp" #include "Driver.hpp" #include "Engine.hpp" #include "EventQueue.hpp" #include "EventWriter.hpp" #include "events.hpp" #define LOG(s) s << "[EventWriter] " using namespace std; using namespace Raul; namespace Ingen { namespace Server { EventWriter::EventWriter(Engine& engine, EventSink& sink) : _request_client(NULL) , _sink(sink) , _request_id(-1) , _engine(engine) , _in_bundle(false) { } EventWriter::~EventWriter() { } SampleCount EventWriter::now() const { // Exactly one cycle latency (some could run ASAP if we get lucky, but not always, and a slight // constant latency is far better than jittery lower (average) latency if (_engine.driver()) return _engine.driver()->frame_time() + _engine.driver()->block_length(); else return 0; } void EventWriter::set_response_id(int32_t id) { if (!_request_client) { // Kludge _request_client = _engine.broadcaster()->client( "http://drobilla.net/ns/ingen#internal"); } _request_id = id; } /* *** ServerInterface implementation below here *** */ // Bundle commands void EventWriter::bundle_begin() { _in_bundle = true; } void EventWriter::bundle_end() { _in_bundle = false; } // Object commands void EventWriter::put(const URI& uri, const Resource::Properties& properties, const Resource::Graph ctx) { _sink.event(new Events::SetMetadata(_engine, _request_client, _request_id, now(), true, ctx, uri, properties)); } void EventWriter::delta(const URI& uri, const Resource::Properties& remove, const Resource::Properties& add) { _sink.event(new Events::SetMetadata(_engine, _request_client, _request_id, now(), false, Resource::DEFAULT, uri, add, remove)); } void EventWriter::move(const Path& old_path, const Path& new_path) { _sink.event(new Events::Move(_engine, _request_client, _request_id, now(), old_path, new_path)); } void EventWriter::del(const URI& uri) { if (uri == "ingen:engine") { if (_request_client) { _request_client->response(_request_id, SUCCESS); } _engine.quit(); } else { _sink.event(new Events::Delete(_engine, _request_client, _request_id, now(), uri)); } } void EventWriter::connect(const Path& tail_path, const Path& head_path) { _sink.event(new Events::Connect(_engine, _request_client, _request_id, now(), tail_path, head_path)); } void EventWriter::disconnect(const Path& src, const Path& dst) { if (!Path::is_path(src) && !Path::is_path(dst)) { LOG(Raul::error) << "Bad disconnect request " << src << " => " << dst << std::endl; return; } _sink.event(new Events::Disconnect(_engine, _request_client, _request_id, now(), Path(src.str()), Path(dst.str()))); } void EventWriter::disconnect_all(const Path& patch_path, const Path& path) { _sink.event(new Events::DisconnectAll(_engine, _request_client, _request_id, now(), patch_path, path)); } void EventWriter::set_property(const URI& uri, const URI& predicate, const Atom& value) { if (uri == "ingen:engine" && predicate == "ingen:enabled" && value.type() == _engine.world()->forge().Bool) { if (value.get_bool()) { _engine.activate(); _sink.event(new Events::Ping(_engine, _request_client, _request_id, now())); } else { _sink.event(new Events::Deactivate(_engine, _request_client, _request_id, now())); } } else { Resource::Properties remove; remove.insert(make_pair(predicate, _engine.world()->uris()->wildcard)); Resource::Properties add; add.insert(make_pair(predicate, value)); _sink.event(new Events::SetMetadata( _engine, _request_client, _request_id, now(), false, Resource::DEFAULT, uri, add, remove)); } } // Requests // void EventWriter::get(const URI& uri) { _sink.event(new Events::Get(_engine, _request_client, _request_id, now(), uri)); } } // namespace Server } // namespace Ingen