/*
This file is part of Ingen.
Copyright 2007-2017 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 "ingen/client/ClientStore.hpp"
#include "ingen/Log.hpp"
#include "ingen/client/ArcModel.hpp"
#include "ingen/client/BlockModel.hpp"
#include "ingen/client/GraphModel.hpp"
#include "ingen/client/ObjectModel.hpp"
#include "ingen/client/PluginModel.hpp"
#include "ingen/client/PortModel.hpp"
#include "ingen/client/SigClientInterface.hpp"
#include
#include
#include
#include
#include
#include
namespace ingen {
namespace client {
ClientStore::ClientStore(URIs& uris,
Log& log,
const std::shared_ptr& emitter)
: _uris(uris)
, _log(log)
, _emitter(emitter)
, _plugins(new Plugins())
{
if (emitter) {
emitter->signal_message().connect(
sigc::mem_fun(this, &ClientStore::message));
}
}
void
ClientStore::clear()
{
Store::clear();
_plugins->clear();
}
void
ClientStore::add_object(const std::shared_ptr& object)
{
// If we already have "this" object, merge the existing one into the new
// one (with precedence to the new values).
auto existing = find(object->path());
if (existing != end()) {
std::dynamic_pointer_cast(existing->second)->set(object);
} else {
if (!object->path().is_root()) {
std::shared_ptr parent = _object(object->path().parent());
if (parent) {
assert(object->path().is_child_of(parent->path()));
object->set_parent(parent);
parent->add_child(object);
assert(parent && (object->parent() == parent));
(*this)[object->path()] = object;
_signal_new_object.emit(object);
} else {
_log.error("Object %1% with no parent\n", object->path());
}
} else {
(*this)[object->path()] = object;
_signal_new_object.emit(object);
}
}
for (const auto& p : object->properties()) {
object->signal_property().emit(p.first, p.second);
}
}
std::shared_ptr
ClientStore::remove_object(const Raul::Path& path)
{
// Find the object, the "top" of the tree to remove
const iterator top = find(path);
if (top == end()) {
return nullptr;
}
auto object = std::dynamic_pointer_cast(top->second);
// Remove object and any adjacent arcs from parent if applicable
if (object && object->parent()) {
auto port = std::dynamic_pointer_cast(object);
if (port && std::dynamic_pointer_cast(port->parent())) {
disconnect_all(port->parent()->path(), path);
if (port->parent()->parent()) {
disconnect_all(port->parent()->parent()->path(), path);
}
} else if (port && port->parent()->parent()) {
disconnect_all(port->parent()->parent()->path(), path);
} else {
disconnect_all(object->parent()->path(), path);
}
object->parent()->remove_child(object);
}
// Remove the object and all its descendants
Objects removed;
remove(top, removed);
// Notify everything that needs to know this object has been removed
if (object) {
object->signal_destroyed().emit();
}
return object;
}
std::shared_ptr
ClientStore::_plugin(const URI& uri)
{
const Plugins::iterator i = _plugins->find(uri);
return (i == _plugins->end()) ? std::shared_ptr() : (*i).second;
}
std::shared_ptr
ClientStore::_plugin(const Atom& uri)
{
/* FIXME: Should probably be stored with URIs rather than strings, to make
this a fast case. */
const Plugins::iterator i = _plugins->find(URI(_uris.forge.str(uri, false)));
return (i == _plugins->end()) ? std::shared_ptr() : (*i).second;
}
std::shared_ptr
ClientStore::plugin(const URI& uri) const
{
return const_cast(this)->_plugin(uri);
}
std::shared_ptr
ClientStore::_object(const Raul::Path& path)
{
const iterator i = find(path);
if (i == end()) {
return nullptr;
} else {
auto model = std::dynamic_pointer_cast(i->second);
assert(model);
assert(model->path().is_root() || model->parent());
return model;
}
}
std::shared_ptr
ClientStore::object(const Raul::Path& path) const
{
return const_cast(this)->_object(path);
}
std::shared_ptr
ClientStore::_resource(const URI& uri)
{
if (uri_is_path(uri)) {
return _object(uri_to_path(uri));
} else {
return _plugin(uri);
}
}
std::shared_ptr
ClientStore::resource(const URI& uri) const
{
return const_cast(this)->_resource(uri);
}
void
ClientStore::add_plugin(const std::shared_ptr& pm)
{
std::shared_ptr existing = _plugin(pm->uri());
if (existing) {
existing->set(pm);
} else {
_plugins->emplace(pm->uri(), pm);
_signal_new_plugin.emit(pm);
}
}
/* ****** Signal Handlers ******** */
void
ClientStore::operator()(const Del& del)
{
if (uri_is_path(del.uri)) {
remove_object(uri_to_path(del.uri));
} else {
auto p = _plugins->find(del.uri);
if (p != _plugins->end()) {
_plugins->erase(p);
_signal_plugin_deleted.emit(del.uri);
}
}
}
void
ClientStore::operator()(const Copy&)
{
_log.error("Client store copy unsupported\n");
}
void
ClientStore::operator()(const Move& msg)
{
const iterator top = find(msg.old_path);
if (top != end()) {
rename(top, msg.new_path);
}
}
void
ClientStore::message(const Message& msg)
{
boost::apply_visitor(*this, msg);
}
void
ClientStore::operator()(const Put& msg)
{
using Iterator = Properties::const_iterator;
const auto& uri = msg.uri;
const auto& properties = msg.properties;
bool is_block = false;
bool is_graph = false;
bool is_output = false;
bool is_port = false;
Resource::type(uris(), properties,
is_graph, is_block, is_port, is_output);
// Check for specially handled types
const Iterator t = properties.find(_uris.rdf_type);
if (t != properties.end()) {
const Atom& type(t->second);
if (_uris.pset_Preset == type) {
const Iterator p = properties.find(_uris.lv2_appliesTo);
const Iterator l = properties.find(_uris.rdfs_label);
std::shared_ptr plug;
if (p == properties.end()) {
_log.error("Preset <%1%> with no plugin\n", uri.c_str());
} else if (l == properties.end()) {
_log.error("Preset <%1%> with no label\n", uri.c_str());
} else if (l->second.type() != _uris.forge.String) {
_log.error("Preset <%1%> label is not a string\n", uri.c_str());
} else if (!(plug = _plugin(p->second))) {
_log.error("Preset <%1%> for unknown plugin %2%\n",
uri.c_str(), _uris.forge.str(p->second, true));
} else {
plug->add_preset(uri, l->second.ptr());
}
return;
} else if (_uris.ingen_Graph == type) {
is_graph = true;
} else if (_uris.ingen_Internal == type || _uris.lv2_Plugin == type) {
std::shared_ptr p(new PluginModel(uris(), uri, type, properties));
add_plugin(p);
return;
}
}
if (!uri_is_path(uri)) {
_log.error("Put for unknown subject <%1%>\n", uri.c_str());
return;
}
const Raul::Path path(uri_to_path(uri));
auto obj = std::dynamic_pointer_cast(_object(path));
if (obj) {
obj->set_properties(properties);
return;
}
if (path.is_root()) {
is_graph = true;
}
if (is_graph) {
std::shared_ptr model(new GraphModel(uris(), path));
model->set_properties(properties);
add_object(model);
} else if (is_block) {
auto p = properties.find(_uris.lv2_prototype);
if (p == properties.end()) {
p = properties.find(_uris.ingen_prototype);
}
std::shared_ptr plug;
if (p->second.is_valid() && (p->second.type() == _uris.forge.URI ||
p->second.type() == _uris.forge.URID)) {
const URI plugin_uri(_uris.forge.str(p->second, false));
if (!(plug = _plugin(plugin_uri))) {
plug = std::make_shared(uris(),
plugin_uri,
Atom(),
Properties());
add_plugin(plug);
}
std::shared_ptr bm(new BlockModel(uris(), plug, path));
bm->set_properties(properties);
add_object(bm);
} else {
_log.warn("Block %1% has no prototype\n", path.c_str());
}
} else if (is_port) {
PortModel::Direction pdir = (is_output)
? PortModel::Direction::OUTPUT
: PortModel::Direction::INPUT;
uint32_t index = 0;
const Iterator i = properties.find(_uris.lv2_index);
if (i != properties.end() && i->second.type() == _uris.forge.Int) {
index = i->second.get();
}
std::shared_ptr p(new PortModel(uris(), path, index, pdir));
p->set_properties(properties);
add_object(p);
} else {
_log.warn("Ignoring %1% of unknown type\n", path.c_str());
}
}
void
ClientStore::operator()(const Delta& msg)
{
const auto& uri = msg.uri;
if (uri == URI("ingen:/clients/this")) {
// Client property, which we don't store (yet?)
return;
}
if (!uri_is_path(uri)) {
_log.error("Delta for unknown subject <%1%>\n", uri.c_str());
return;
}
const Raul::Path path(uri_to_path(uri));
std::shared_ptr obj = _object(path);
if (obj) {
obj->remove_properties(msg.remove);
obj->add_properties(msg.add);
} else {
_log.warn("Failed to find object `%1%'\n", path.c_str());
}
}
void
ClientStore::operator()(const SetProperty& msg)
{
const auto& subject_uri = msg.subject;
const auto& predicate = msg.predicate;
const auto& value = msg.value;
if (subject_uri == URI("ingen:/engine")) {
_log.info("Engine property <%1%> = %2%\n",
predicate.c_str(), _uris.forge.str(value, false));
return;
}
std::shared_ptr subject = _resource(subject_uri);
if (subject) {
if (predicate == _uris.ingen_activity) {
/* Activity is transient, trigger any live actions (like GUI
blinkenlights) but do not store the property. */
subject->on_property(predicate, value);
} else {
subject->set_property(predicate, value, msg.ctx);
}
} else {
std::shared_ptr plugin = _plugin(subject_uri);
if (plugin) {
plugin->set_property(predicate, value);
} else if (predicate != _uris.ingen_activity) {
_log.warn("Property <%1%> for unknown object %2%\n",
predicate.c_str(), subject_uri.c_str());
}
}
}
std::shared_ptr
ClientStore::connection_graph(const Raul::Path& tail_path,
const Raul::Path& head_path)
{
std::shared_ptr graph;
if (tail_path.parent() == head_path.parent()) {
graph = std::dynamic_pointer_cast(_object(tail_path.parent()));
}
if (!graph && tail_path.parent() == head_path.parent().parent()) {
graph = std::dynamic_pointer_cast(_object(tail_path.parent()));
}
if (!graph && tail_path.parent().parent() == head_path.parent()) {
graph = std::dynamic_pointer_cast(_object(head_path.parent()));
}
if (!graph) {
graph = std::dynamic_pointer_cast(_object(tail_path.parent().parent()));
}
if (!graph) {
_log.error("Unable to find graph for arc %1% => %2%\n",
tail_path, head_path);
}
return graph;
}
bool
ClientStore::attempt_connection(const Raul::Path& tail_path,
const Raul::Path& head_path)
{
auto tail = std::dynamic_pointer_cast(_object(tail_path));
auto head = std::dynamic_pointer_cast(_object(head_path));
if (tail && head) {
std::shared_ptr graph = connection_graph(tail_path, head_path);
std::shared_ptr arc(new ArcModel(tail, head));
graph->add_arc(arc);
return true;
} else {
_log.warn("Failed to connect %1% => %2%\n", tail_path, head_path);
return false;
}
}
void
ClientStore::operator()(const Connect& msg)
{
attempt_connection(msg.tail, msg.head);
}
void
ClientStore::operator()(const Disconnect& msg)
{
auto tail = std::dynamic_pointer_cast(_object(msg.tail));
auto head = std::dynamic_pointer_cast(_object(msg.head));
auto graph = connection_graph(msg.tail, msg.head);
if (graph) {
graph->remove_arc(tail.get(), head.get());
}
}
void
ClientStore::operator()(const DisconnectAll& msg)
{
auto graph = std::dynamic_pointer_cast(_object(msg.graph));
auto object = _object(msg.path);
if (!graph || !object) {
_log.error("Bad disconnect all notification %1% in %2%\n",
msg.path, msg.graph);
return;
}
const GraphModel::Arcs arcs = graph->arcs();
for (const auto& a : arcs) {
auto arc = std::dynamic_pointer_cast(a.second);
if (arc->tail()->parent() == object
|| arc->head()->parent() == object
|| arc->tail()->path() == msg.path
|| arc->head()->path() == msg.path) {
graph->remove_arc(arc->tail().get(), arc->head().get());
}
}
}
} // namespace client
} // namespace ingen