annotate vamp-support/RdfTypes.h @ 240:d00199f3b010

More helpful comments etc
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 13 Jun 2017 16:07:58 +0100
parents d54dab075247
children
rev   line source
cannam@222 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
cannam@222 2
cannam@222 3 /*
cannam@222 4 Piper C++
cannam@222 5
cannam@222 6 An API for audio analysis and feature extraction plugins.
cannam@222 7
cannam@222 8 Centre for Digital Music, Queen Mary, University of London.
cannam@222 9 Copyright 2006-2017 Chris Cannam and QMUL.
cannam@222 10
cannam@222 11 Permission is hereby granted, free of charge, to any person
cannam@222 12 obtaining a copy of this software and associated documentation
cannam@222 13 files (the "Software"), to deal in the Software without
cannam@222 14 restriction, including without limitation the rights to use, copy,
cannam@222 15 modify, merge, publish, distribute, sublicense, and/or sell copies
cannam@222 16 of the Software, and to permit persons to whom the Software is
cannam@222 17 furnished to do so, subject to the following conditions:
cannam@222 18
cannam@222 19 The above copyright notice and this permission notice shall be
cannam@222 20 included in all copies or substantial portions of the Software.
cannam@222 21
cannam@222 22 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
cannam@222 23 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
cannam@222 24 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
cannam@222 25 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
cannam@222 26 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
cannam@222 27 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
cannam@222 28 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
cannam@222 29
cannam@222 30 Except as contained in this notice, the names of the Centre for
cannam@222 31 Digital Music; Queen Mary, University of London; and Chris Cannam
cannam@222 32 shall not be used in advertising or otherwise to promote the sale,
cannam@222 33 use or other dealings in this Software without prior written
cannam@222 34 authorization.
cannam@222 35 */
cannam@222 36
cannam@222 37 #ifndef PIPER_RDF_TYPES_H
cannam@222 38 #define PIPER_RDF_TYPES_H
cannam@222 39
cannam@222 40 #include "StaticOutputDescriptor.h"
cannam@222 41
cannam@222 42 #include <vamp-hostsdk/PluginLoader.h>
cannam@222 43
cannam@222 44 #include <sord/sord.h>
cannam@222 45
cannam@223 46 #include <mutex>
cannam@223 47
cannam@222 48 namespace piper_vamp {
cannam@222 49
cannam@229 50 //!!! todo: better (+ optional) error reporting; check whether file
cannam@229 51 //!!! exists before parsing it to avoid spurious error messages;
cannam@229 52 //!!! refactoring
cannam@229 53
cannam@222 54 class RdfTypes
cannam@222 55 {
cannam@222 56 public:
cannam@223 57 RdfTypes() :
cannam@223 58 m_world(sord_world_new())
cannam@223 59 {}
cannam@223 60
cannam@223 61 ~RdfTypes() {
cannam@223 62 sord_world_free(m_world);
cannam@223 63 }
cannam@223 64
cannam@223 65 StaticOutputInfo loadStaticOutputInfo(Vamp::HostExt::PluginLoader::PluginKey
cannam@223 66 pluginKey) {
cannam@223 67
cannam@223 68 StaticOutputInfo info;
cannam@223 69 SordModel *model = sord_new(m_world, SORD_SPO|SORD_OPS|SORD_POS, false);
cannam@223 70 if (loadRdf(model, candidateRdfFilesFor(pluginKey))) {
cannam@225 71 loadStaticOutputInfoFromModel(model, pluginKey, info);
cannam@223 72 }
cannam@223 73 sord_free(model);
cannam@223 74 return info;
cannam@223 75 }
cannam@223 76
cannam@223 77 private:
cannam@223 78 SordWorld *m_world;
cannam@223 79
cannam@223 80 bool loadRdf(SordModel *targetModel, std::vector<std::string> filenames) {
cannam@223 81 for (auto f: filenames) {
cannam@223 82 if (loadRdfFile(targetModel, f)) {
cannam@223 83 return true;
cannam@223 84 }
cannam@223 85 }
cannam@223 86 return false;
cannam@223 87 }
cannam@223 88
cannam@223 89 bool loadRdfFile(SordModel *targetModel, std::string filename) {
cannam@223 90 std::string base = "file://" + filename;
cannam@223 91 SerdURI bu;
cannam@223 92 if (serd_uri_parse((const uint8_t *)base.c_str(), &bu) !=
cannam@223 93 SERD_SUCCESS) {
cannam@223 94 std::cerr << "Failed to parse base URI " << base << std::endl;
cannam@223 95 return false;
cannam@223 96 }
cannam@223 97 SerdNode bn = serd_node_from_string(SERD_URI,
cannam@223 98 (const uint8_t *)base.c_str());
cannam@223 99 SerdEnv *env = serd_env_new(&bn);
cannam@223 100 SerdReader *reader = sord_new_reader(targetModel, env, SERD_TURTLE, 0);
cannam@223 101 SerdStatus rv = serd_reader_read_file
cannam@223 102 (reader, (const uint8_t *)filename.c_str());
cannam@223 103 bool success = (rv == SERD_SUCCESS);
cannam@223 104 if (!success) {
cannam@229 105 // We are asking Serd to parse the file without having
cannam@229 106 // checked whether it actually exists or not (in order to
cannam@229 107 // avoid duplicating ugly platform/encoding-specific stuff
cannam@229 108 // in this file). So don't bleat if the file is simply not
cannam@229 109 // found, but only if there's a real parse error
cannam@229 110 if (rv != SERD_ERR_NOT_FOUND &&
cannam@229 111 rv != SERD_ERR_UNKNOWN) {
cannam@229 112 std::cerr << "Failed to import RDF from " << filename
cannam@229 113 << ": " << serd_strerror(rv) << std::endl;
cannam@229 114 }
cannam@223 115 }
cannam@223 116 serd_reader_free(reader);
cannam@223 117 serd_env_free(env);
cannam@223 118 return success;
cannam@223 119 }
cannam@223 120
cannam@223 121 std::vector<std::string> candidateRdfFilesFor(Vamp::HostExt::
cannam@223 122 PluginLoader::PluginKey key) {
cannam@223 123
cannam@223 124 std::string library = Vamp::HostExt::PluginLoader::getInstance()->
cannam@223 125 getLibraryPathForPlugin(key);
cannam@223 126
cannam@223 127 auto li = library.rfind('.');
cannam@223 128 if (li == std::string::npos) return {};
cannam@223 129 auto withoutSuffix = library.substr(0, li);
cannam@223 130
cannam@229 131 std::vector<std::string> suffixes { "n3", "N3", "ttl", "TTL" };
cannam@223 132 std::vector<std::string> candidates;
cannam@223 133
cannam@223 134 for (auto suffix : suffixes) {
cannam@223 135 candidates.push_back(withoutSuffix + "." + suffix);
cannam@223 136 }
cannam@223 137
cannam@223 138 return candidates;
cannam@222 139 }
cannam@225 140
cannam@225 141 void
cannam@229 142 loadStaticOutputInfoFromModel(SordModel *model,
cannam@229 143 std::string pluginKey,
cannam@225 144 StaticOutputInfo &info) {
cannam@229 145
cannam@225 146 // we want to find a graph like
cannam@229 147 //
cannam@225 148 // :plugin a vamp:Plugin
cannam@225 149 // :plugin vamp:identifier "pluginId"
cannam@225 150 // :library vamp:available_plugin :plugin
cannam@225 151 // :library vamp:identifier "libraryId"
cannam@225 152 // :plugin vamp:output :output1
cannam@225 153 // :plugin vamp:output :output2
cannam@225 154 // :plugin vamp:output :output3
cannam@225 155 // :output1 vamp:computes_event_type :event
cannam@225 156 // :output2 vamp:computes_feature :feature
cannam@225 157 // :output3 vamp:computes_signal_type :signal
cannam@229 158 //
cannam@229 159 // in which pluginKey == libraryId + ":" + pluginId
cannam@227 160
cannam@225 161 std::string libraryId, pluginId;
cannam@225 162 decomposePluginKey(pluginKey, libraryId, pluginId);
cannam@225 163
cannam@229 164 typedef const uint8_t *S;
cannam@229 165
cannam@229 166 SordNode *a = sord_new_uri
cannam@229 167 (m_world, S("http://www.w3.org/1999/02/22-rdf-syntax-ns#type"));
cannam@229 168
cannam@229 169 SordNode *pluginType = sord_new_uri
cannam@229 170 (m_world, S("http://purl.org/ontology/vamp/Plugin"));
cannam@229 171
cannam@229 172 SordNode *identProp = sord_new_uri
cannam@229 173 (m_world, S("http://purl.org/ontology/vamp/identifier"));
cannam@229 174 SordNode *availProp = sord_new_uri
cannam@229 175 (m_world, S("http://purl.org/ontology/vamp/available_plugin"));
cannam@229 176 SordNode *outputProp = sord_new_uri
cannam@229 177 (m_world, S("http://purl.org/ontology/vamp/output"));
cannam@229 178
cannam@229 179 SordNode *computesEventProp = sord_new_uri
cannam@229 180 (m_world, S("http://purl.org/ontology/vamp/computes_event_type"));
cannam@229 181 SordNode *computesFeatureProp = sord_new_uri
cannam@229 182 (m_world, S("http://purl.org/ontology/vamp/computes_feature"));
cannam@229 183 SordNode *computesSignalProp = sord_new_uri
cannam@229 184 (m_world, S("http://purl.org/ontology/vamp/computes_signal_type"));
cannam@229 185
cannam@229 186 SordIter *pluginItr = 0;
cannam@229 187
cannam@229 188 for (pluginItr = sord_search(model, 0, a, pluginType, 0);
cannam@229 189 !sord_iter_end(pluginItr);
cannam@229 190 sord_iter_next(pluginItr)) {
cannam@229 191
cannam@229 192 const SordNode *pluginNode =
cannam@229 193 sord_iter_get_node(pluginItr, SORD_SUBJECT);
cannam@229 194
cannam@229 195 SordNode *pluginIdNode =
cannam@229 196 sord_get(model, pluginNode, identProp, 0, 0);
cannam@229 197
cannam@229 198 if (!pluginIdNode ||
cannam@229 199 sord_node_get_type(pluginIdNode) != SORD_LITERAL ||
cannam@229 200 (const char *)sord_node_get_string(pluginIdNode) != pluginId) {
cannam@229 201 // This is a plugin node, but it's not the plugin node
cannam@229 202 // we're looking for. (We have to check both the type
cannam@229 203 // property, vamp:Plugin, and the identifier,
cannam@229 204 // vamp:identifier, because the identifier is just a
cannam@229 205 // string and it's possible it could be used for an
cannam@229 206 // output or parameter rather than just a plugin.)
cannam@229 207 continue;
cannam@229 208 }
cannam@229 209
cannam@229 210 SordNode *libraryNode =
cannam@229 211 sord_get(model, 0, availProp, pluginNode, 0);
cannam@229 212
cannam@229 213 if (!libraryNode) {
cannam@229 214 std::cerr << "Plugin is not listed as being in a library, "
cannam@229 215 << "skipping library id check" << std::endl;
cannam@229 216 } else {
cannam@229 217 SordNode *libIdNode =
cannam@229 218 sord_get(model, libraryNode, identProp, 0, 0);
cannam@229 219 if (!libIdNode ||
cannam@229 220 sord_node_get_type(libIdNode) != SORD_LITERAL ||
cannam@229 221 (const char *)sord_node_get_string(libIdNode) != libraryId) {
cannam@229 222 std::cerr << "Ignoring plugin in wrong library" << std::endl;
cannam@229 223 continue;
cannam@229 224 }
cannam@229 225 }
cannam@229 226
cannam@229 227 SordIter *outputItr = 0;
cannam@229 228
cannam@229 229 for (outputItr = sord_search(model, pluginNode, outputProp, 0, 0);
cannam@229 230 !sord_iter_end(outputItr);
cannam@229 231 sord_iter_next(outputItr)) {
cannam@229 232
cannam@229 233 const SordNode *outputNode =
cannam@229 234 sord_iter_get_node(outputItr, SORD_OBJECT);
cannam@229 235
cannam@229 236 SordNode *outputIdNode =
cannam@229 237 sord_get(model, outputNode, identProp, 0, 0);
cannam@229 238
cannam@229 239 if (!outputIdNode ||
cannam@229 240 sord_node_get_type(outputIdNode) != SORD_LITERAL ||
cannam@229 241 !sord_node_get_string(outputIdNode)) {
cannam@229 242 std::cerr << "Ignoring output with no id" << std::endl;
cannam@229 243 continue;
cannam@229 244 }
cannam@229 245
cannam@229 246 std::string outputId =
cannam@229 247 (const char *)sord_node_get_string(outputIdNode);
cannam@229 248
cannam@229 249 SordIter *propItr = 0;
cannam@229 250
cannam@229 251 for (propItr = sord_search(model, outputNode, 0, 0, 0);
cannam@229 252 !sord_iter_end(propItr);
cannam@229 253 sord_iter_next(propItr)) {
cannam@229 254
cannam@229 255 const SordNode *propNode =
cannam@229 256 sord_iter_get_node(propItr, SORD_PREDICATE);
cannam@229 257
cannam@229 258 if (sord_node_equals(propNode, computesEventProp) ||
cannam@229 259 sord_node_equals(propNode, computesFeatureProp) ||
cannam@229 260 sord_node_equals(propNode, computesSignalProp)) {
cannam@229 261
cannam@229 262 const SordNode *computesNode =
cannam@229 263 sord_iter_get_node(propItr, SORD_OBJECT);
cannam@229 264
cannam@229 265 if (sord_node_get_type(computesNode) != SORD_URI ||
cannam@229 266 !sord_node_get_string(computesNode)) {
cannam@229 267 std::cerr << "Ignoring non-URI computes node"
cannam@229 268 << std::endl;
cannam@229 269 continue;
cannam@229 270 }
cannam@229 271
cannam@229 272 std::string typeURI =
cannam@229 273 (const char *)sord_node_get_string(computesNode);
cannam@229 274
cannam@229 275 std::cerr << "Found type <" << typeURI
cannam@229 276 << "> for output \"" << outputId
cannam@229 277 << "\" of plugin \"" << pluginId
cannam@229 278 << "\" in library " << libraryId
cannam@229 279 << std::endl;
cannam@229 280
cannam@229 281 StaticOutputDescriptor desc;
cannam@229 282 desc.typeURI = typeURI;
cannam@229 283 info[outputId] = desc;
cannam@229 284
cannam@229 285 break; // only interested in one "computes" property
cannam@229 286 }
cannam@229 287 }
cannam@229 288
cannam@229 289 sord_iter_free(propItr);
cannam@229 290 }
cannam@229 291
cannam@229 292 sord_iter_free(outputItr);
cannam@229 293 }
cannam@229 294
cannam@229 295 sord_iter_free(pluginItr);
cannam@225 296 }
cannam@225 297
cannam@225 298 bool decomposePluginKey(std::string pluginKey,
cannam@225 299 std::string &libraryId,
cannam@225 300 std::string &pluginId) {
cannam@225 301 auto i = pluginKey.find(':');
cannam@225 302 if (i == std::string::npos || i == 0 || i + 1 == pluginKey.length()) {
cannam@225 303 return false;
cannam@225 304 }
cannam@225 305 libraryId = pluginKey.substr(0, i);
cannam@225 306 pluginId = pluginKey.substr(i + 1);
cannam@225 307 return true;
cannam@225 308 }
cannam@225 309
cannam@222 310 };
cannam@222 311
cannam@222 312 }
cannam@222 313
cannam@222 314 #endif