summaryrefslogtreecommitdiffstats
path: root/src/AtomWriter.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/AtomWriter.cpp')
-rw-r--r--src/AtomWriter.cpp652
1 files changed, 652 insertions, 0 deletions
diff --git a/src/AtomWriter.cpp b/src/AtomWriter.cpp
new file mode 100644
index 00000000..f235aafc
--- /dev/null
+++ b/src/AtomWriter.cpp
@@ -0,0 +1,652 @@
+/*
+ This file is part of Ingen.
+ Copyright 2007-2017 David Robillard <http://drobilla.net/>
+
+ 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 <http://www.gnu.org/licenses/>.
+*/
+
+/** @page protocol Ingen Protocol
+ * @tableofcontents
+ *
+ * @section introduction Controlling Ingen
+ *
+ * Ingen is controlled via a simple RDF-based protocol. This conceptual
+ * protocol can be used in two concrete ways:
+ *
+ * 1. When Ingen is running as a process, a socket accepts messages in
+ * (textual) Turtle syntax, and responds in the same syntax. Transfers are
+ * delimited by NULL characters in the stream, so the client knows when to
+ * finish parsing to interpret responses. By default, Ingen listens on
+ * unix:///tmp/ingen.sock and tcp://localhost:16180
+ *
+ * 2. When Ingen is running as an LV2 plugin, the control and notify ports
+ * accept and respond using (binary) LV2 atoms. Messages are read and written
+ * as events with atom:Object bodies. The standard rules for LV2 event
+ * transmission apply, but note that the graph manipulations described here are
+ * executed asynchronously and not with sample accuracy, so the response may
+ * come at a later frame or cycle.
+ *
+ * For documentation purposes, this page discusses messages in Turtle syntax,
+ * but the same protocol is used in the LV2 case, just in a more compact binary
+ * encoding.
+ *
+ * Conceptually, Ingen represents a tree of objects, each of which has a path
+ * (like /main/in or /main/osc/out) and a set of properties. A property is a
+ * URI key and some value. All changes to Ingen are represented as
+ * manipulations of this tree of dictionaries. The core of the protocol is
+ * based on the LV2 patch extension, which defines several messages for
+ * manipulating data in this model which resemble HTTP methods.
+ */
+
+#include <cassert>
+#include <cstdlib>
+#include <string>
+
+#include <boost/variant/apply_visitor.hpp>
+
+#include "ingen/AtomSink.hpp"
+#include "ingen/AtomWriter.hpp"
+#include "ingen/Node.hpp"
+#include "ingen/URIMap.hpp"
+#include "raul/Path.hpp"
+#include "serd/serd.h"
+
+namespace Ingen {
+
+AtomWriter::AtomWriter(URIMap& map, URIs& uris, AtomSink& sink)
+ : _map(map)
+ , _uris(uris)
+ , _sink(sink)
+{
+ lv2_atom_forge_init(&_forge, &map.urid_map_feature()->urid_map);
+ _out.set_forge_sink(&_forge);
+}
+
+void
+AtomWriter::finish_msg()
+{
+ assert(!_forge.stack);
+ _sink.write(_out.atom());
+ _out.clear();
+}
+
+void
+AtomWriter::message(const Message& message)
+{
+ boost::apply_visitor(*this, message);
+}
+
+/** @page protocol
+ * @subsection Bundles
+ *
+ * An [ingen:BundleStart](http://drobilla.net/ns/ingen#BundleStart) marks the
+ * start of a bundle in the operation stream. A bundle groups a series of
+ * messages for coarse-grained undo or atomic execution.
+ *
+ * @code{.ttl}
+ * [] a ingen:BundleStart .
+ * @endcode
+
+ * An [ingen:BundleEnd](http://drobilla.net/ns/ingen#BundleEnd) marks the end
+ * of a bundle in the operation stream.
+ *
+ * @code{.ttl}
+ * [] a ingen:BundleEnd .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const BundleBegin& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.ingen_BundleStart, message.seq);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+void
+AtomWriter::operator()(const BundleEnd& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.ingen_BundleEnd, message.seq);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+void
+AtomWriter::forge_uri(const URI& uri)
+{
+ if (serd_uri_string_has_scheme((const uint8_t*)uri.c_str())) {
+ lv2_atom_forge_urid(&_forge, _map.map_uri(uri.c_str()));
+ } else {
+ lv2_atom_forge_uri(&_forge, uri.c_str(), uri.length());
+ }
+}
+
+void
+AtomWriter::forge_properties(const Properties& properties)
+{
+ for (auto p : properties) {
+ lv2_atom_forge_key(&_forge, _map.map_uri(p.first.c_str()));
+ if (p.second.type() == _forge.URI) {
+ forge_uri(URI(p.second.ptr<char>()));
+ } else {
+ lv2_atom_forge_atom(&_forge, p.second.size(), p.second.type());
+ lv2_atom_forge_write(&_forge, p.second.get_body(), p.second.size());
+ }
+ }
+}
+
+void
+AtomWriter::forge_arc(const Raul::Path& tail, const Raul::Path& head)
+{
+ LV2_Atom_Forge_Frame arc;
+ lv2_atom_forge_object(&_forge, &arc, 0, _uris.ingen_Arc);
+ lv2_atom_forge_key(&_forge, _uris.ingen_tail);
+ forge_uri(path_to_uri(tail));
+ lv2_atom_forge_key(&_forge, _uris.ingen_head);
+ forge_uri(path_to_uri(head));
+ lv2_atom_forge_pop(&_forge, &arc);
+}
+
+void
+AtomWriter::forge_request(LV2_Atom_Forge_Frame* frame, LV2_URID type, int32_t id)
+{
+ lv2_atom_forge_object(&_forge, frame, 0, type);
+
+ if (id) {
+ lv2_atom_forge_key(&_forge, _uris.patch_sequenceNumber);
+ lv2_atom_forge_int(&_forge, id);
+ }
+}
+
+void
+AtomWriter::forge_context(Resource::Graph ctx)
+{
+ switch (ctx) {
+ case Resource::Graph::EXTERNAL:
+ lv2_atom_forge_key(&_forge, _uris.patch_context);
+ forge_uri(_uris.ingen_externalContext);
+ break;
+ case Resource::Graph::INTERNAL:
+ lv2_atom_forge_key(&_forge, _uris.patch_context);
+ forge_uri(_uris.ingen_internalContext);
+ break;
+ default: break;
+ }
+}
+
+/** @page protocol
+ * @section methods Methods
+ * @subsection Put
+ *
+ * Send a [Put](http://lv2plug.in/ns/ext/patch#Put) to set properties on an
+ * object, creating it if necessary.
+ *
+ * If the object already exists, all existing object properties with keys that
+ * match any property in the message are first removed. Other properties are
+ * left unchanged.
+ *
+ * If the object does not yet exist, the message must contain sufficient
+ * information to create it, including at least one rdf:type property.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Put ;
+ * patch:subject </main/osc> ;
+ * patch:body [
+ * a ingen:Block ;
+ * lv2:prototype <http: //drobilla.net/plugins/mda/Shepard>
+ * ] .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Put& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Put, message.seq);
+ forge_context(message.ctx);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.uri);
+ lv2_atom_forge_key(&_forge, _uris.patch_body);
+
+ LV2_Atom_Forge_Frame body;
+ lv2_atom_forge_object(&_forge, &body, 0, 0);
+ forge_properties(message.properties);
+ lv2_atom_forge_pop(&_forge, &body);
+
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Patch
+ *
+ * Send a [Patch](http://lv2plug.in/ns/ext/patch#Patch) to manipulate the
+ * properties of an object. A set of properties are first removed, then
+ * another is added. Analogous to WebDAV PROPPATCH.
+ *
+ * The special value [patch:wildcard](http://lv2plug.in/ns/ext/patch#wildcard)
+ * may be used to specify that any value with the given key should be removed.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Patch ;
+ * patch:subject </main/osc> ;
+ * patch:add [
+ * lv2:name "Osckillator" ;
+ * ingen:canvasX 32.0 ;
+ * ingen:canvasY 32.0 ;
+ * ] ;
+ * patch:remove [
+ * eg:name "Old name" ; # Remove specific value
+ * ingen:canvasX patch:wildcard ; # Remove all
+ * ingen:canvasY patch:wildcard ; # Remove all
+ * ] .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Delta& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Patch, message.seq);
+ forge_context(message.ctx);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.uri);
+
+ lv2_atom_forge_key(&_forge, _uris.patch_remove);
+ LV2_Atom_Forge_Frame remove_obj;
+ lv2_atom_forge_object(&_forge, &remove_obj, 0, 0);
+ forge_properties(message.remove);
+ lv2_atom_forge_pop(&_forge, &remove_obj);
+
+ lv2_atom_forge_key(&_forge, _uris.patch_add);
+ LV2_Atom_Forge_Frame add_obj;
+ lv2_atom_forge_object(&_forge, &add_obj, 0, 0);
+ forge_properties(message.add);
+ lv2_atom_forge_pop(&_forge, &add_obj);
+
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Copy
+ *
+ * Send a [Copy](http://lv2plug.in/ns/ext/copy#Copy) to copy an object from
+ * its current location (subject) to another (destination).
+ *
+ * If both the subject and destination are inside Ingen, like block paths, then the old object
+ * is copied by, for example, creating a new plugin instance.
+ *
+ * If the subject is a filename (file URI or atom:Path) and the destination is
+ * inside Ingen, then the subject must be an Ingen graph file or bundle, which
+ * is loaded to the specified destination path.
+ *
+ * If the subject is inside Ingen and the destination is a filename, then the
+ * subject is saved to an Ingen bundle at the given destination.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Copy ;
+ * patch:subject </main/osc> ;
+ * patch:destination </main/osc2> .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Copy& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Copy, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.old_uri);
+ lv2_atom_forge_key(&_forge, _uris.patch_destination);
+ forge_uri(message.new_uri);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Move
+ *
+ * Send a [Move](http://lv2plug.in/ns/ext/move#Move) to move an object from its
+ * current location (subject) to another (destination).
+ *
+ * Both subject and destination must be paths in Ingen with the same parent,
+ * moving between graphs is currently not supported.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Move ;
+ * patch:subject </main/osc> ;
+ * patch:destination </main/osc2> .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Move& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Move, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(path_to_uri(message.old_path));
+ lv2_atom_forge_key(&_forge, _uris.patch_destination);
+ forge_uri(path_to_uri(message.new_path));
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Delete
+ *
+ * Send a [Delete](http://lv2plug.in/ns/ext/delete#Delete) to remove an
+ * object from the engine and destroy it.
+ *
+ * All properties of the object are lost, as are all references to the object
+ * (e.g. any connections to it).
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Delete ;
+ * patch:subject </main/osc> .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Del& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Delete, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.uri);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Set
+ *
+ * Send a [Set](http://lv2plug.in/ns/ext/patch#Set) to set a property on an
+ * object. Any existing value for that property is removed.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Set ;
+ * patch:subject </main/osc> ;
+ * patch:property lv2:name ;
+ * patch:value "Oscwellator" .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const SetProperty& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Set, message.seq);
+ forge_context(message.ctx);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.subject);
+ lv2_atom_forge_key(&_forge, _uris.patch_property);
+ lv2_atom_forge_urid(&_forge, _map.map_uri(message.predicate.c_str()));
+ lv2_atom_forge_key(&_forge, _uris.patch_value);
+ lv2_atom_forge_atom(&_forge, message.value.size(), message.value.type());
+ lv2_atom_forge_write(&_forge, message.value.get_body(), message.value.size());
+
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Undo
+ *
+ * Use [ingen:Undo](http://drobilla.net/ns/ingen#Undo) to undo the last change
+ * to the engine. Undo transactions can be delimited using bundle markers, if
+ * the last operations(s) received were in a bundle, then an Undo will undo the
+ * effects of that entire bundle.
+ *
+ * @code{.ttl}
+ * [] a ingen:Undo .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Undo& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.ingen_Undo, message.seq);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Undo
+ *
+ * Use [ingen:Redo](http://drobilla.net/ns/ingen#Redo) to redo the last undone change.
+ *
+ * @code{.ttl}
+ * [] a ingen:Redo .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Redo& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.ingen_Redo, message.seq);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Get
+ *
+ * Send a [Get](http://lv2plug.in/ns/ext/patch#Get) to get the description
+ * of the subject.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Get ;
+ * patch:subject </main/osc> .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Get& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Get, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(message.subject);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ *
+ * @section arcs Arc Manipulation
+ */
+
+/** @page protocol
+ * @subsection Connect Connecting Ports
+ *
+ * Ports are connected by putting an arc with the desired tail (an output port)
+ * and head (an input port). The tail and head must both be within the
+ * subject, which must be a graph.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Put ;
+ * patch:subject </main/> ;
+ * patch:body [
+ * a ingen:Arc ;
+ * ingen:tail </main/osc/out> ;
+ * ingen:head </main/filt/in> ;
+ * ] .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Connect& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Put, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(path_to_uri(Raul::Path::lca(message.tail, message.head)));
+ lv2_atom_forge_key(&_forge, _uris.patch_body);
+ forge_arc(message.tail, message.head);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection Disconnect Disconnecting Ports
+ *
+ * Ports are disconnected by deleting the arc between them. The description of
+ * the arc is the same as in the put command used to create the connection.
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Delete ;
+ * patch:body [
+ * a ingen:Arc ;
+ * ingen:tail </main/osc/out> ;
+ * ingen:head </main/filt/in> ;
+ * ] .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const Disconnect& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Delete, message.seq);
+ lv2_atom_forge_key(&_forge, _uris.patch_body);
+ forge_arc(message.tail, message.head);
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @subsection DisconnectAll Fully Disconnecting Anything
+ *
+ * All arcs to or from anything can be removed by giving the special property
+ * ingen:incidentTo rather than a specific head and tail. This works for both
+ * ports and blocks (where the effect is to disconnect everything from ports on
+ * that block).
+ *
+ * @code{.ttl}
+ * []
+ * a patch:Delete ;
+ * patch:subject </main> ;
+ * patch:body [
+ * a ingen:Arc ;
+ * ingen:incidentTo </main/osc/out>
+ * ] .
+ * @endcode
+ */
+void
+AtomWriter::operator()(const DisconnectAll& message)
+{
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Delete, message.seq);
+
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ forge_uri(path_to_uri(message.graph));
+
+ lv2_atom_forge_key(&_forge, _uris.patch_body);
+ LV2_Atom_Forge_Frame arc;
+ lv2_atom_forge_object(&_forge, &arc, 0, _uris.ingen_Arc);
+ lv2_atom_forge_key(&_forge, _uris.ingen_incidentTo);
+ forge_uri(path_to_uri(message.path));
+ lv2_atom_forge_pop(&_forge, &arc);
+
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+/** @page protocol
+ * @section Responses
+ *
+ * Ingen responds to requests if the patch:sequenceNumber property is set. For
+ * example:
+ * @code{.ttl}
+ * []
+ * a patch:Get ;
+ * patch:sequenceNumber 42 ;
+ * patch:subject </main/osc> .
+ * @endcode
+ *
+ * Might receive a response like:
+ * @code{.ttl}
+ * []
+ * a patch:Response ;
+ * patch:sequenceNumber 42 ;
+ * patch:subject </main/osc> ;
+ * patch:body 0 .
+ * @endcode
+ *
+ * Where 0 is a status code, 0 meaning success and any other value being an
+ * error. Information about status codes, including error message strings,
+ * are defined in ingen.lv2/errors.ttl.
+ *
+ * Note that a response is only a status response, operations that manipulate
+ * the graph may generate new data on the stream, e.g. the above get request
+ * would also receive a put that describes /main/osc in the stream immediately
+ * following the response.
+ */
+void
+AtomWriter::operator()(const Response& response)
+{
+ const auto& subject = response.subject;
+ if (!response.id) {
+ return;
+ }
+
+ LV2_Atom_Forge_Frame msg;
+ forge_request(&msg, _uris.patch_Response, 0);
+ lv2_atom_forge_key(&_forge, _uris.patch_sequenceNumber);
+ lv2_atom_forge_int(&_forge, response.id);
+ if (!subject.empty()) {
+ lv2_atom_forge_key(&_forge, _uris.patch_subject);
+ lv2_atom_forge_uri(&_forge, subject.c_str(), subject.length());
+ }
+ lv2_atom_forge_key(&_forge, _uris.patch_body);
+ lv2_atom_forge_int(&_forge, static_cast<int>(response.status));
+ lv2_atom_forge_pop(&_forge, &msg);
+ finish_msg();
+}
+
+void
+AtomWriter::operator()(const Error&)
+{
+}
+
+/** @page protocol
+ * @section loading Loading and Unloading Bundles
+ *
+ * The property ingen:loadedBundle on the engine can be used to load
+ * or unload bundles from Ingen's world. For example:
+ *
+ * @code{.ttl}
+ * # Load /old.lv2
+ * []
+ * a patch:Put ;
+ * patch:subject </> ;
+ * patch:body [
+ * ingen:loadedBundle <file:///old.lv2/>
+ * ] .
+ *
+ * # Replace /old.lv2 with /new.lv2
+ * []
+ * a patch:Patch ;
+ * patch:subject </> ;
+ * patch:remove [
+ * ingen:loadedBundle <file:///old.lv2/>
+ * ];
+ * patch:add [
+ * ingen:loadedBundle <file:///new.lv2/>
+ * ] .
+ * @endcode
+ */
+
+} // namespace Ingen