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
|