/*
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 "EventWriter.hpp"
#include "events.hpp"
#define LOG(s) s << "[EventWriter] "
using namespace std;
using namespace Raul;
namespace Ingen {
namespace Server {
EventWriter::EventWriter(Engine& engine)
: _engine(engine)
, _request_id(-1)
{
}
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 */
return (_engine.driver())
? _engine.driver()->frame_time() + _engine.driver()->block_length()
: 0;
}
void
EventWriter::set_response_id(int32_t id)
{
_request_id = id;
}
void
EventWriter::put(const URI& uri,
const Resource::Properties& properties,
const Resource::Graph ctx)
{
_engine.enqueue_event(
new Events::SetMetadata(_engine, _respondee.get(), _request_id, now(),
true, ctx, uri, properties));
}
void
EventWriter::delta(const URI& uri,
const Resource::Properties& remove,
const Resource::Properties& add)
{
_engine.enqueue_event(
new Events::SetMetadata(_engine, _respondee.get(), _request_id, now(),
false, Resource::DEFAULT, uri, add, remove));
}
void
EventWriter::move(const Path& old_path,
const Path& new_path)
{
_engine.enqueue_event(
new Events::Move(_engine, _respondee.get(), _request_id, now(),
old_path, new_path));
}
void
EventWriter::del(const URI& uri)
{
if (uri == "ingen:engine") {
if (_respondee) {
_respondee->response(_request_id, SUCCESS);
}
_engine.quit();
} else {
_engine.enqueue_event(
new Events::Delete(_engine, _respondee.get(), _request_id, now(), uri));
}
}
void
EventWriter::connect(const Path& tail_path,
const Path& head_path)
{
_engine.enqueue_event(
new Events::Connect(_engine, _respondee.get(), _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;
}
_engine.enqueue_event(
new Events::Disconnect(_engine, _respondee.get(), _request_id, now(),
src, dst));
}
void
EventWriter::disconnect_all(const Path& patch_path,
const Path& path)
{
_engine.enqueue_event(
new Events::DisconnectAll(_engine, _respondee.get(), _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();
_engine.enqueue_event(
new Events::Ping(_engine, _respondee.get(), _request_id, now()));
} else {
_engine.enqueue_event(
new Events::Deactivate(_engine, _respondee.get(), _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));
_engine.enqueue_event(
new Events::SetMetadata(_engine, _respondee.get(), _request_id, now(),
false, Resource::DEFAULT, uri, add, remove));
}
}
void
EventWriter::get(const URI& uri)
{
_engine.enqueue_event(
new Events::Get(_engine, _respondee.get(), _request_id, now(), uri));
}
} // namespace Server
} // namespace Ingen