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
|