Mercurial > hg > piper-cpp
comparison vamp-support/RdfTypes.h @ 229:d54dab075247
Implement RDF type lookup
author | Chris Cannam <cannam@all-day-breakfast.com> |
---|---|
date | Mon, 12 Jun 2017 10:20:25 +0100 |
parents | c6aed932318d |
children |
comparison
equal
deleted
inserted
replaced
228:691b0daad1c6 | 229:d54dab075247 |
---|---|
45 | 45 |
46 #include <mutex> | 46 #include <mutex> |
47 | 47 |
48 namespace piper_vamp { | 48 namespace piper_vamp { |
49 | 49 |
50 //!!! todo: better (+ optional) error reporting; check whether file | |
51 //!!! exists before parsing it to avoid spurious error messages; | |
52 //!!! refactoring | |
53 | |
50 class RdfTypes | 54 class RdfTypes |
51 { | 55 { |
52 public: | 56 public: |
53 RdfTypes() : | 57 RdfTypes() : |
54 m_world(sord_world_new()) | 58 m_world(sord_world_new()) |
96 SerdReader *reader = sord_new_reader(targetModel, env, SERD_TURTLE, 0); | 100 SerdReader *reader = sord_new_reader(targetModel, env, SERD_TURTLE, 0); |
97 SerdStatus rv = serd_reader_read_file | 101 SerdStatus rv = serd_reader_read_file |
98 (reader, (const uint8_t *)filename.c_str()); | 102 (reader, (const uint8_t *)filename.c_str()); |
99 bool success = (rv == SERD_SUCCESS); | 103 bool success = (rv == SERD_SUCCESS); |
100 if (!success) { | 104 if (!success) { |
101 std::cerr << "Failed to import RDF from " << filename << std::endl; | 105 // We are asking Serd to parse the file without having |
102 } else { | 106 // checked whether it actually exists or not (in order to |
103 std::cerr << "Imported RDF from " << filename << std::endl; | 107 // avoid duplicating ugly platform/encoding-specific stuff |
108 // in this file). So don't bleat if the file is simply not | |
109 // found, but only if there's a real parse error | |
110 if (rv != SERD_ERR_NOT_FOUND && | |
111 rv != SERD_ERR_UNKNOWN) { | |
112 std::cerr << "Failed to import RDF from " << filename | |
113 << ": " << serd_strerror(rv) << std::endl; | |
114 } | |
104 } | 115 } |
105 serd_reader_free(reader); | 116 serd_reader_free(reader); |
106 serd_env_free(env); | 117 serd_env_free(env); |
107 return success; | 118 return success; |
108 } | 119 } |
115 | 126 |
116 auto li = library.rfind('.'); | 127 auto li = library.rfind('.'); |
117 if (li == std::string::npos) return {}; | 128 if (li == std::string::npos) return {}; |
118 auto withoutSuffix = library.substr(0, li); | 129 auto withoutSuffix = library.substr(0, li); |
119 | 130 |
120 std::vector<std::string> suffixes { "ttl", "TTL", "n3", "N3" }; | 131 std::vector<std::string> suffixes { "n3", "N3", "ttl", "TTL" }; |
121 std::vector<std::string> candidates; | 132 std::vector<std::string> candidates; |
122 | 133 |
123 for (auto suffix : suffixes) { | 134 for (auto suffix : suffixes) { |
124 candidates.push_back(withoutSuffix + "." + suffix); | 135 candidates.push_back(withoutSuffix + "." + suffix); |
125 } | 136 } |
126 | 137 |
127 return candidates; | 138 return candidates; |
128 } | 139 } |
129 | 140 |
130 void | 141 void |
131 loadStaticOutputInfoFromModel(SordModel *model, std::string pluginKey, | 142 loadStaticOutputInfoFromModel(SordModel *model, |
143 std::string pluginKey, | |
132 StaticOutputInfo &info) { | 144 StaticOutputInfo &info) { |
145 | |
133 // we want to find a graph like | 146 // we want to find a graph like |
147 // | |
134 // :plugin a vamp:Plugin | 148 // :plugin a vamp:Plugin |
135 // :plugin vamp:identifier "pluginId" | 149 // :plugin vamp:identifier "pluginId" |
136 // :library vamp:available_plugin :plugin | 150 // :library vamp:available_plugin :plugin |
137 // :library vamp:identifier "libraryId" | 151 // :library vamp:identifier "libraryId" |
138 // :plugin vamp:output :output1 | 152 // :plugin vamp:output :output1 |
139 // :plugin vamp:output :output2 | 153 // :plugin vamp:output :output2 |
140 // :plugin vamp:output :output3 | 154 // :plugin vamp:output :output3 |
141 // :output1 vamp:computes_event_type :event | 155 // :output1 vamp:computes_event_type :event |
142 // :output2 vamp:computes_feature :feature | 156 // :output2 vamp:computes_feature :feature |
143 // :output3 vamp:computes_signal_type :signal | 157 // :output3 vamp:computes_signal_type :signal |
144 // and where pluginKey == libraryId + ":" + pluginId | 158 // |
145 | 159 // in which pluginKey == libraryId + ":" + pluginId |
146 // Although, since we know we just loaded an RDF file | |
147 // associated with one particular plugin library, we could in | |
148 // theory skip the library bits. We'd still have most of the | |
149 // rest though since each library can have multiple plugins | |
150 // with output names that could conflict | |
151 | 160 |
152 std::string libraryId, pluginId; | 161 std::string libraryId, pluginId; |
153 decomposePluginKey(pluginKey, libraryId, pluginId); | 162 decomposePluginKey(pluginKey, libraryId, pluginId); |
154 | 163 |
155 //!!! | 164 typedef const uint8_t *S; |
165 | |
166 SordNode *a = sord_new_uri | |
167 (m_world, S("http://www.w3.org/1999/02/22-rdf-syntax-ns#type")); | |
168 | |
169 SordNode *pluginType = sord_new_uri | |
170 (m_world, S("http://purl.org/ontology/vamp/Plugin")); | |
171 | |
172 SordNode *identProp = sord_new_uri | |
173 (m_world, S("http://purl.org/ontology/vamp/identifier")); | |
174 SordNode *availProp = sord_new_uri | |
175 (m_world, S("http://purl.org/ontology/vamp/available_plugin")); | |
176 SordNode *outputProp = sord_new_uri | |
177 (m_world, S("http://purl.org/ontology/vamp/output")); | |
178 | |
179 SordNode *computesEventProp = sord_new_uri | |
180 (m_world, S("http://purl.org/ontology/vamp/computes_event_type")); | |
181 SordNode *computesFeatureProp = sord_new_uri | |
182 (m_world, S("http://purl.org/ontology/vamp/computes_feature")); | |
183 SordNode *computesSignalProp = sord_new_uri | |
184 (m_world, S("http://purl.org/ontology/vamp/computes_signal_type")); | |
185 | |
186 SordIter *pluginItr = 0; | |
187 | |
188 for (pluginItr = sord_search(model, 0, a, pluginType, 0); | |
189 !sord_iter_end(pluginItr); | |
190 sord_iter_next(pluginItr)) { | |
191 | |
192 const SordNode *pluginNode = | |
193 sord_iter_get_node(pluginItr, SORD_SUBJECT); | |
194 | |
195 SordNode *pluginIdNode = | |
196 sord_get(model, pluginNode, identProp, 0, 0); | |
197 | |
198 if (!pluginIdNode || | |
199 sord_node_get_type(pluginIdNode) != SORD_LITERAL || | |
200 (const char *)sord_node_get_string(pluginIdNode) != pluginId) { | |
201 // This is a plugin node, but it's not the plugin node | |
202 // we're looking for. (We have to check both the type | |
203 // property, vamp:Plugin, and the identifier, | |
204 // vamp:identifier, because the identifier is just a | |
205 // string and it's possible it could be used for an | |
206 // output or parameter rather than just a plugin.) | |
207 continue; | |
208 } | |
209 | |
210 SordNode *libraryNode = | |
211 sord_get(model, 0, availProp, pluginNode, 0); | |
212 | |
213 if (!libraryNode) { | |
214 std::cerr << "Plugin is not listed as being in a library, " | |
215 << "skipping library id check" << std::endl; | |
216 } else { | |
217 SordNode *libIdNode = | |
218 sord_get(model, libraryNode, identProp, 0, 0); | |
219 if (!libIdNode || | |
220 sord_node_get_type(libIdNode) != SORD_LITERAL || | |
221 (const char *)sord_node_get_string(libIdNode) != libraryId) { | |
222 std::cerr << "Ignoring plugin in wrong library" << std::endl; | |
223 continue; | |
224 } | |
225 } | |
226 | |
227 SordIter *outputItr = 0; | |
228 | |
229 for (outputItr = sord_search(model, pluginNode, outputProp, 0, 0); | |
230 !sord_iter_end(outputItr); | |
231 sord_iter_next(outputItr)) { | |
232 | |
233 const SordNode *outputNode = | |
234 sord_iter_get_node(outputItr, SORD_OBJECT); | |
235 | |
236 SordNode *outputIdNode = | |
237 sord_get(model, outputNode, identProp, 0, 0); | |
238 | |
239 if (!outputIdNode || | |
240 sord_node_get_type(outputIdNode) != SORD_LITERAL || | |
241 !sord_node_get_string(outputIdNode)) { | |
242 std::cerr << "Ignoring output with no id" << std::endl; | |
243 continue; | |
244 } | |
245 | |
246 std::string outputId = | |
247 (const char *)sord_node_get_string(outputIdNode); | |
248 | |
249 SordIter *propItr = 0; | |
250 | |
251 for (propItr = sord_search(model, outputNode, 0, 0, 0); | |
252 !sord_iter_end(propItr); | |
253 sord_iter_next(propItr)) { | |
254 | |
255 const SordNode *propNode = | |
256 sord_iter_get_node(propItr, SORD_PREDICATE); | |
257 | |
258 if (sord_node_equals(propNode, computesEventProp) || | |
259 sord_node_equals(propNode, computesFeatureProp) || | |
260 sord_node_equals(propNode, computesSignalProp)) { | |
261 | |
262 const SordNode *computesNode = | |
263 sord_iter_get_node(propItr, SORD_OBJECT); | |
264 | |
265 if (sord_node_get_type(computesNode) != SORD_URI || | |
266 !sord_node_get_string(computesNode)) { | |
267 std::cerr << "Ignoring non-URI computes node" | |
268 << std::endl; | |
269 continue; | |
270 } | |
271 | |
272 std::string typeURI = | |
273 (const char *)sord_node_get_string(computesNode); | |
274 | |
275 std::cerr << "Found type <" << typeURI | |
276 << "> for output \"" << outputId | |
277 << "\" of plugin \"" << pluginId | |
278 << "\" in library " << libraryId | |
279 << std::endl; | |
280 | |
281 StaticOutputDescriptor desc; | |
282 desc.typeURI = typeURI; | |
283 info[outputId] = desc; | |
284 | |
285 break; // only interested in one "computes" property | |
286 } | |
287 } | |
288 | |
289 sord_iter_free(propItr); | |
290 } | |
291 | |
292 sord_iter_free(outputItr); | |
293 } | |
294 | |
295 sord_iter_free(pluginItr); | |
156 } | 296 } |
157 | 297 |
158 bool decomposePluginKey(std::string pluginKey, | 298 bool decomposePluginKey(std::string pluginKey, |
159 std::string &libraryId, | 299 std::string &libraryId, |
160 std::string &pluginId) { | 300 std::string &pluginId) { |