From 46b2e1598ed75319735893ed039ac669aca2d8a7 Mon Sep 17 00:00:00 2001 From: David Robillard Date: Thu, 28 May 2009 14:33:33 +0000 Subject: Make SetMetadataEvent do multiple properties at once. git-svn-id: http://svn.drobilla.net/lad/trunk/ingen@2019 a436a847-0d15-0410-975c-d299462d15a1 --- src/client/ClientStore.cpp | 22 ++++- src/engine/ClientBroadcaster.cpp | 8 ++ src/engine/ClientBroadcaster.hpp | 4 +- src/engine/QueuedEngineInterface.cpp | 3 + src/engine/events/SetMetadataEvent.cpp | 165 ++++++++++++++++++++------------- src/engine/events/SetMetadataEvent.hpp | 44 +++++---- src/serialisation/Parser.cpp | 70 +++++++------- src/shared/ResourceImpl.cpp | 17 +++- src/shared/ResourceImpl.hpp | 3 +- 9 files changed, 211 insertions(+), 125 deletions(-) diff --git a/src/client/ClientStore.cpp b/src/client/ClientStore.cpp index 7774e9c3..12bbdaf9 100644 --- a/src/client/ClientStore.cpp +++ b/src/client/ClientStore.cpp @@ -104,6 +104,14 @@ ClientStore::add_object(SharedPtr object) } + for (Resource::Properties::const_iterator i = object->meta().properties().begin(); + i != object->meta().properties().end(); ++i) + object->signal_property(i->first, i->second); + + for (Resource::Properties::const_iterator i = object->properties().begin(); + i != object->properties().end(); ++i) + object->signal_property(i->first, i->second); + /*cout << "[Store] Added " << object->path() << " {" << endl; for (iterator i = begin(); i != end(); ++i) { cout << "\t" << i->first << endl; @@ -279,6 +287,12 @@ ClientStore::put(const URI& uri, const Resource::Properties& properties) cerr << "\t" << i->first << " = " << i->second << " :: " << i->second.type() << endl; cerr << "}" << endl; + SharedPtr obj = PtrCast(object(path)); + if (obj) { + cerr << "OBJECT EXISTS " << path << endl; + obj->set_properties(properties); + } + bool is_patch, is_node, is_port, is_output; DataType data_type(DataType::UNKNOWN); ResourceImpl::type(properties, is_patch, is_node, is_port, is_output, data_type); @@ -289,7 +303,7 @@ ClientStore::put(const URI& uri, const Resource::Properties& properties) if (p != properties.end() && p->second.is_valid() && p->second.type() == Atom::INT) poly = p->second.get_int32(); SharedPtr model(new PatchModel(path, poly)); - model->merge_properties(properties); + model->set_properties(properties); add_object(model); } else if (is_node) { const Resource::Properties::const_iterator p = properties.find("rdf:instanceOf"); @@ -297,11 +311,11 @@ ClientStore::put(const URI& uri, const Resource::Properties& properties) if (p->second.is_valid() && p->second.type() == Atom::URI) { if ((plug = plugin(p->second.get_uri()))) { SharedPtr n(new NodeModel(plug, path)); - n->merge_properties(properties); + n->set_properties(properties); add_object(n); } else { SharedPtr n(new NodeModel(p->second.get_uri(), path)); - n->merge_properties(properties); + n->set_properties(properties); //add_plugin_orphan(n); add_object(n); } @@ -312,7 +326,7 @@ ClientStore::put(const URI& uri, const Resource::Properties& properties) if (data_type != DataType::UNKNOWN) { PortModel::Direction pdir = is_output ? PortModel::OUTPUT : PortModel::INPUT; SharedPtr p(new PortModel(path, 0, data_type, pdir)); - p->merge_properties(properties); + p->set_properties(properties); add_object(p); } else { cerr << "WARNING: Illegal port " << path << endl; diff --git a/src/engine/ClientBroadcaster.cpp b/src/engine/ClientBroadcaster.cpp index 8aaffe94..50a1112b 100644 --- a/src/engine/ClientBroadcaster.cpp +++ b/src/engine/ClientBroadcaster.cpp @@ -162,6 +162,14 @@ ClientBroadcaster::send_disconnection(const Path& src_port_path, const Path& dst } +void +ClientBroadcaster::send_put(const Raul::URI& subject, const Shared::Resource::Properties& properties) +{ + for (Clients::const_iterator i = _clients.begin(); i != _clients.end(); ++i) + (*i).second->put(subject, properties); +} + + /** Send notification of a property update. * * Like control changes, does not send update to client that set the property, if applicable. diff --git a/src/engine/ClientBroadcaster.hpp b/src/engine/ClientBroadcaster.hpp index 2aea8eb0..8d9d5c4d 100644 --- a/src/engine/ClientBroadcaster.hpp +++ b/src/engine/ClientBroadcaster.hpp @@ -69,8 +69,8 @@ public: void send_connection(const SharedPtr connection); void send_disconnection(const Raul::Path& src_port_path, const Raul::Path& dst_port_path); void send_move(const Raul::Path& old_path, const Raul::Path& new_path); - void send_variable_change(const Raul::URI& node_path, const Raul::URI& key, const Raul::Atom& value); - void send_property_change(const Raul::URI& node_path, const Raul::URI& key, const Raul::Atom& value); + void send_put(const Raul::URI& subject, const Shared::Resource::Properties& properties); + void send_property_change(const Raul::URI& subject, const Raul::URI& key, const Raul::Atom& value); void send_port_value(const Raul::Path& port_path, const Raul::Atom& value); void send_activity(const Raul::Path& path); diff --git a/src/engine/QueuedEngineInterface.cpp b/src/engine/QueuedEngineInterface.cpp index 4b2926e7..7d9a8dcc 100644 --- a/src/engine/QueuedEngineInterface.cpp +++ b/src/engine/QueuedEngineInterface.cpp @@ -155,6 +155,7 @@ void QueuedEngineInterface::put(const URI& uri, const Resource::Properties& properties) { + cerr << "PUT " << uri << endl; size_t hash = uri.find("#"); bool meta = (hash != string::npos); Path path(meta ? (string("/") + uri.chop_start("#")) : uri.str()); @@ -169,6 +170,8 @@ QueuedEngineInterface::put(const URI& uri, DataType data_type(DataType::UNKNOWN); ResourceImpl::type(properties, is_patch, is_node, is_port, is_output, data_type); + // PutEvent + if (is_patch) { uint32_t poly = 1; iterator p = properties.find("ingen:polyphony"); diff --git a/src/engine/events/SetMetadataEvent.cpp b/src/engine/events/SetMetadataEvent.cpp index 00db9161..a88490d6 100644 --- a/src/engine/events/SetMetadataEvent.cpp +++ b/src/engine/events/SetMetadataEvent.cpp @@ -43,10 +43,7 @@ SetMetadataEvent::SetMetadataEvent( const Atom& value) : QueuedEvent(engine, responder, timestamp) , _error(NO_ERROR) - , _special_type(NONE) , _subject(subject) - , _key(key) - , _value(value) , _object(NULL) , _patch(NULL) , _compiled_patch(NULL) @@ -55,6 +52,27 @@ SetMetadataEvent::SetMetadataEvent( { cerr << "SET " << subject << " : " << key << " = " << value << endl; assert(value.type() != Atom::URI || strcmp(value.get_uri(), "lv2:ControlPort")); + _properties.insert(make_pair(key, value)); +} + + +SetMetadataEvent::SetMetadataEvent( + Engine& engine, + SharedPtr responder, + SampleCount timestamp, + bool meta, + const URI& subject, + const Shared::Resource::Properties& properties) + : QueuedEvent(engine, responder, timestamp) + , _error(NO_ERROR) + , _subject(subject) + , _properties(properties) + , _object(NULL) + , _patch(NULL) + , _compiled_patch(NULL) + , _is_meta(meta) + , _success(false) +{ } @@ -75,43 +93,55 @@ SetMetadataEvent::pre_process() /*cerr << "SET " << _object->path() << (_property ? " PROP " : " VAR ") << _key << " :: " << _value.type() << endl;*/ - GraphObjectImpl* obj = dynamic_cast(_object); - if (obj) { - if (_is_meta) - obj->meta().set_property(_key, _value); - else - obj->set_property(_key, _value); - - _patch = dynamic_cast(_object); - - if (_key.str() == "ingen:broadcast") { - _special_type = ENABLE_BROADCAST; - } else if (_patch) { - if (_key.str() == "ingen:enabled") { - if (_value.type() == Atom::BOOL) { - _special_type = ENABLE; - if (_value.get_bool() && !_patch->compiled_patch()) - _compiled_patch = _patch->compile(); + _types.reserve(_properties.size()); + typedef Shared::Resource::Properties Properties; + for (Properties::iterator p = _properties.begin(); p != _properties.end(); ++p) { + const Raul::URI& key = p->first; + const Raul::Atom& value = p->second; + GraphObjectImpl* obj = dynamic_cast(_object); + if (obj) { + if (_is_meta) + obj->meta().set_property(key, value); + else + obj->set_property(key, value); + + _patch = dynamic_cast(_object); + + if (key.str() == "ingen:broadcast") { + _types.push_back(ENABLE_BROADCAST); + } else if (_patch) { + if (key.str() == "ingen:enabled") { + if (value.type() == Atom::BOOL) { + _types.push_back(ENABLE); + if (value.get_bool() && !_patch->compiled_patch()) + _compiled_patch = _patch->compile(); + } else { + _error = BAD_TYPE; + } + } else if (key.str() == "ingen:polyphonic") { + if (value.type() == Atom::BOOL) { + _types.push_back(POLYPHONIC); + } else { + _error = BAD_TYPE; + } + } else if (key.str() == "ingen:polyphony") { + if (value.type() == Atom::INT) { + _types.push_back(POLYPHONY); + _patch->prepare_internal_poly(value.get_int32()); + } else { + _error = BAD_TYPE; + } } else { - _error = BAD_TYPE; - } - } else if (_key.str() == "ingen:polyphonic") { - if (_value.type() == Atom::BOOL) { - _special_type = POLYPHONIC; - } else { - _error = BAD_TYPE; - } - } else if (_key.str() == "ingen:polyphony") { - if (_value.type() == Atom::INT) { - _special_type = POLYPHONY; - _patch->prepare_internal_poly(_value.get_int32()); - } else { - _error = BAD_TYPE; + _types.push_back(NONE); } + if (_error != NO_ERROR) + break; } + } else { + _types.push_back(NONE); } - } else { - _object->set_property(_key, _value); + + _object->set_property(key, value); } QueuedEvent::pre_process(); @@ -124,33 +154,38 @@ SetMetadataEvent::execute(ProcessContext& context) if (_error != NO_ERROR) return; - PortImpl* port = 0; - GraphObjectImpl* object = 0; - switch (_special_type) { - case ENABLE_BROADCAST: - if ((port = dynamic_cast(_object))) - port->broadcast(_value.get_bool()); - break; - case ENABLE: - if (_value.get_bool()) { - if (!_patch->compiled_patch()) - _patch->compiled_patch(_compiled_patch); - _patch->enable(); - } else { - _patch->disable(); + typedef Shared::Resource::Properties Properties; + std::vector::const_iterator t = _types.begin(); + for (Properties::iterator p = _properties.begin(); p != _properties.end(); ++p, ++t) { + const Raul::Atom& value = p->second; + PortImpl* port = 0; + GraphObjectImpl* object = 0; + switch (*t) { + case ENABLE_BROADCAST: + if ((port = dynamic_cast(_object))) + port->broadcast(value.get_bool()); + break; + case ENABLE: + if (value.get_bool()) { + if (!_patch->compiled_patch()) + _patch->compiled_patch(_compiled_patch); + _patch->enable(); + } else { + _patch->disable(); + } + break; + case POLYPHONIC: + if ((object = dynamic_cast(_object))) + if (!object->set_polyphonic(*_engine.maid(), value.get_bool())) + _error = INTERNAL; + break; + case POLYPHONY: + if (!_patch->apply_internal_poly(*_engine.maid(), value.get_int32())) + _error = INTERNAL; + break; + default: + _success = true; } - break; - case POLYPHONIC: - if ((object = dynamic_cast(_object))) - if (!object->set_polyphonic(*_engine.maid(), _value.get_bool())) - _error = INTERNAL; - break; - case POLYPHONY: - if (!_patch->apply_internal_poly(*_engine.maid(), _value.get_int32())) - _error = INTERNAL; - break; - default: - _success = true; } QueuedEvent::execute(context); @@ -163,16 +198,16 @@ SetMetadataEvent::post_process() switch (_error) { case NO_ERROR: _responder->respond_ok(); - _engine.broadcaster()->send_property_change(_subject, _key, _value); + _engine.broadcaster()->send_put(_subject, _properties); break; case NOT_FOUND: _responder->respond_error((boost::format( - "Unable to find object '%1%' to set '%2%'") % _subject % _key).str()); + "Unable to find object '%1%'") % _subject).str()); case INTERNAL: _responder->respond_error("Internal error"); break; case BAD_TYPE: - _responder->respond_error((boost::format("Bad type for '%1%'") % _key).str()); + _responder->respond_error("Bad type for predicate"); break; } } diff --git a/src/engine/events/SetMetadataEvent.hpp b/src/engine/events/SetMetadataEvent.hpp index b33cc9ca..e43b1e9f 100644 --- a/src/engine/events/SetMetadataEvent.hpp +++ b/src/engine/events/SetMetadataEvent.hpp @@ -18,6 +18,7 @@ #ifndef SETMETADATAEVENT_H #define SETMETADATAEVENT_H +#include #include "raul/URI.hpp" #include "raul/Atom.hpp" #include "shared/ResourceImpl.hpp" @@ -37,13 +38,22 @@ class CompiledPatch; class SetMetadataEvent : public QueuedEvent { public: - SetMetadataEvent(Engine& engine, - SharedPtr responder, - SampleCount timestamp, - bool meta, - const Raul::URI& subject, - const Raul::URI& key, - const Raul::Atom& value); + SetMetadataEvent( + Engine& engine, + SharedPtr responder, + SampleCount timestamp, + bool meta, + const Raul::URI& subject, + const Raul::URI& key, + const Raul::Atom& value); + + SetMetadataEvent( + Engine& engine, + SharedPtr responder, + SampleCount timestamp, + bool meta, + const Raul::URI& subject, + const Shared::Resource::Properties& properties); void pre_process(); void execute(ProcessContext& context); @@ -51,22 +61,22 @@ public: private: enum { NO_ERROR, NOT_FOUND, INTERNAL, BAD_TYPE } _error; - enum { + enum SpecialType { NONE, ENABLE, ENABLE_BROADCAST, POLYPHONY, POLYPHONIC - } _special_type; + }; - Raul::URI _subject; - Raul::URI _key; - Raul::Atom _value; - Shared::ResourceImpl* _object; - PatchImpl* _patch; - CompiledPatch* _compiled_patch; - bool _is_meta; - bool _success; + std::vector _types; + Raul::URI _subject; + Shared::Resource::Properties _properties; + Shared::ResourceImpl* _object; + PatchImpl* _patch; + CompiledPatch* _compiled_patch; + bool _is_meta; + bool _success; }; diff --git a/src/serialisation/Parser.cpp b/src/serialisation/Parser.cpp index 781278a4..41b777a3 100644 --- a/src/serialisation/Parser.cpp +++ b/src/serialisation/Parser.cpp @@ -224,7 +224,7 @@ Parser::parse_update( for (Redland::Query::Results::iterator i = results.begin(); i != results.end(); ++i) { Glib::Mutex::Lock lock(world->rdf_world->mutex()); const string obj_uri((*i)["s"].to_string()); - const string key(world->rdf_world->prefixes().qualify((*i)["p"].to_string())); + const string key(world->rdf_world->qualify((*i)["p"].to_string())); const Redland::Node& val_node((*i)["o"]); const Atom a(AtomRDF::node_to_atom(val_node)); if (key != "") @@ -437,10 +437,10 @@ Parser::parse_patch( patches.insert((*i)["patch"].to_string()); } - typedef multimap Properties; - typedef map Resources; - typedef map Objects; - typedef map Types; + typedef multimap Properties; + typedef map Resources; + typedef map Objects; + typedef map Types; /* Find nodes on this patch */ @@ -486,10 +486,11 @@ Parser::parse_patch( Objects::iterator plug_i = plugin_nodes.find(node.to_string()); Types::iterator type_i = types.find(node.to_string()); if (node.type() == Redland::Node::RESOURCE && type_i != types.end()) { + const string key = world->rdf_world->qualify(predicate.to_string()); if (patch_i != patch_nodes.end()) { - patch_i->second.insert(make_pair(predicate.to_string(), object)); + patch_i->second.insert(make_pair(key, object)); } else if (plug_i != plugin_nodes.end()) { - plug_i->second.insert(make_pair(predicate.to_string(), object)); + plug_i->second.insert(make_pair(key, object)); } else { cerr << "WARNING: Unrecognized node: " << node.to_string() << endl; } @@ -507,10 +508,7 @@ Parser::parse_patch( continue; parse_patch(world, target, model, res_i->second, patch_path, Symbol(node_path.name())); Glib::Mutex::Lock lock(world->rdf_world->mutex()); - for (Properties::iterator j = i->second.begin(); j != i->second.end(); ++j) { - const string key = world->rdf_world->prefixes().qualify(j->first); - target->set_property(node_path, key, AtomRDF::node_to_atom(j->second)); - } + target->put(node_path, i->second); } /* Create plugin nodes */ @@ -522,12 +520,8 @@ Parser::parse_patch( Resource::Properties props; props.insert(make_pair("rdf:type", Raul::Atom(Raul::Atom::URI, "ingen:Node"))); props.insert(make_pair("rdf:instanceOf", Raul::Atom(Raul::Atom::URI, type_i->second))); + props.insert(i->second.begin(), i->second.end()); target->put(node_path, props); - Glib::Mutex::Lock lock(world->rdf_world->mutex()); - for (Properties::iterator j = i->second.begin(); j != i->second.end(); ++j) { - const string key = world->rdf_world->prefixes().qualify(j->first); - target->set_property(node_path, key, AtomRDF::node_to_atom(j->second)); - } } @@ -538,6 +532,7 @@ Parser::parse_patch( "?node lv2:port ?port .\n" "?port ?key ?val .\n" "}"); + Objects node_ports; results = query.run(*world->rdf_world, model, base_uri); for (Results::iterator i = results.begin(); i != results.end(); ++i) { Glib::Mutex::Lock lock(world->rdf_world->mutex()); @@ -548,15 +543,19 @@ Parser::parse_patch( continue; } + Objects::iterator p = node_ports.find(port_uri); + if (p == node_ports.end()) + p = node_ports.insert(make_pair(port_uri, Properties())).first; + const Path node_path(relative_uri(base_uri, node_uri, true)); const Symbol port_sym = port_uri.substr(node_uri.length() + 1); const Path port_path = node_path.base() + port_sym; - const string key = world->rdf_world->prefixes().qualify((*i)["key"].to_string()); - if (key == "ingen:value") { - target->set_port_value(port_path, AtomRDF::node_to_atom((*i)["val"])); - } else { - target->set_property(port_path, key, AtomRDF::node_to_atom((*i)["val"])); - } + const string key = world->rdf_world->qualify((*i)["key"].to_string()); + p->second.insert(make_pair(key, AtomRDF::node_to_atom((*i)["val"]))); + } + + for (Objects::iterator i = node_ports.begin(); i != node_ports.end(); ++i) { + target->put(Raul::Path(relative_uri(base_uri, i->first, true)), i->second); } @@ -574,7 +573,7 @@ Parser::parse_patch( Redland::Node& type = (*i)["type"]; if (port.type() == Redland::Node::RESOURCE && type.type() == Redland::Node::RESOURCE) { types.insert(make_pair(port.to_string(), - world->rdf_world->prefixes().qualify(type.to_string()))); + world->rdf_world->qualify(type.to_string()))); patch_ports.insert(make_pair(port.to_string(), Properties())); } } @@ -591,7 +590,7 @@ Parser::parse_patch( Glib::Mutex::Lock lock(world->rdf_world->mutex()); const string port_uri = (*i)["port"].to_string(); const Path port_path(Path(relative_uri(base_uri, port_uri, true))); - const string key = world->rdf_world->prefixes().qualify((*i)["key"].to_string()); + const string key = world->rdf_world->qualify((*i)["key"].to_string()); Objects::iterator ports_i = patch_ports.find(port_uri); if (ports_i == patch_ports.end()) ports_i = patch_ports.insert(make_pair(port_uri, Properties())).first; @@ -609,13 +608,15 @@ Parser::parse_patch( } bool is_input = false; bool is_output = false; - Redland::Node* type = 0; + Raul::Atom* type = 0; for (Properties::iterator t = types_range.first; t != types_range.second; ++t) { - if (t->second.to_string() == "lv2:InputPort") { + if (t->second.type() != Atom::URI) { + continue; + } else if (!strcmp(t->second.get_uri(), "lv2:InputPort")) { is_input = true; - } else if (t->second.to_string() == "lv2:OutputPort") { + } else if (!strcmp(t->second.get_uri(), "lv2:OutputPort")) { is_output = true; - } else if (!type || type->to_string() == t->second.to_string()) { + } else if (!type) { type = &t->second; } else { cerr << "ERROR: Port has several data types" << endl; @@ -790,20 +791,23 @@ Parser::parse_properties( subject + " ?key ?val .\n" "}"); + Resource::Properties properties; Redland::Query::Results results = query.run(*world->rdf_world, model); for (Redland::Query::Results::iterator i = results.begin(); i != results.end(); ++i) { Glib::Mutex::Lock lock(world->rdf_world->mutex()); - const string key = world->rdf_world->prefixes().qualify(string((*i)["key"])); + const string key = world->rdf_world->qualify(string((*i)["key"])); const Redland::Node& val = (*i)["val"]; if (key != "") - target->set_property(path, key, AtomRDF::node_to_atom(val)); + properties.insert(make_pair(key, AtomRDF::node_to_atom(val))); } + cout << "PROPERTIES: " << path << endl; + + target->put(path, properties); + // Set passed properties last to override any loaded values if (data) - for (GraphObject::Properties::const_iterator i = data.get().begin(); - i != data.get().end(); ++i) - target->set_property(path, i->first, i->second); + target->put(path, data.get()); return true; } diff --git a/src/shared/ResourceImpl.cpp b/src/shared/ResourceImpl.cpp index 9e403be3..ca3eaacd 100644 --- a/src/shared/ResourceImpl.cpp +++ b/src/shared/ResourceImpl.cpp @@ -50,8 +50,6 @@ ResourceImpl::add_property(const Raul::URI& uri, const Raul::Atom& value) void ResourceImpl::set_property(const Raul::URI& uri, const Raul::Atom& value) { - cerr << "SET PROPERTY " << uri << " = " << value << endl; - assert(value.type() != Raul::Atom::URI || strcmp(value.get_uri(), "lv2:ControlPort")); _properties.erase(uri); _properties.insert(make_pair(uri, value)); } @@ -126,9 +124,22 @@ ResourceImpl::type( void -ResourceImpl::merge_properties(const Properties& p) +ResourceImpl::set_properties(const Properties& p) { typedef Resource::Properties::const_iterator iterator; + for (iterator i = p.begin(); i != p.end(); ++i) + _properties.erase(i->first); + for (iterator i = p.begin(); i != p.end(); ++i) + set_property(i->first, i->second); +} + + +void +ResourceImpl::add_properties(const Properties& p) +{ + typedef Resource::Properties::const_iterator iterator; + for (iterator i = p.begin(); i != p.end(); ++i) + _properties.erase(i->first); for (iterator i = p.begin(); i != p.end(); ++i) add_property(i->first, i->second); } diff --git a/src/shared/ResourceImpl.hpp b/src/shared/ResourceImpl.hpp index 1f8dbedd..118d4dbe 100644 --- a/src/shared/ResourceImpl.hpp +++ b/src/shared/ResourceImpl.hpp @@ -40,7 +40,8 @@ public: const Raul::Atom& get_property(const Raul::URI& uri) const; void set_property(const Raul::URI& uri, const Raul::Atom& value); void add_property(const Raul::URI& uri, const Raul::Atom& value); - void merge_properties(const Properties& p); + void set_properties(const Properties& p); + void add_properties(const Properties& p); sigc::signal signal_property; -- cgit v1.2.1