annotate vamp-support/StaticOutputRdf.h @ 296:50a0b4fea7f1 tip master

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