annotate rdf/RDFFeatureWriter.cpp @ 585:cbc884787a08

* If we have track metadata, write it out in a separate mo:Track associated with this audio file instead of incorrectly associating it with the Signal. Also, make the audio file encode the signal instead of having the signal "available_as" the audio file (apparently available_as is not in the domain of signals).
author Chris Cannam
date Fri, 08 May 2009 10:09:47 +0000
parents bf1f6d68a142
children 4ab8e8f7d7ed
rev   line source
Chris@498 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@498 2
Chris@498 3 /*
Chris@498 4 Sonic Annotator
Chris@498 5 A utility for batch feature extraction from audio files.
Chris@498 6 Mark Levy, Chris Sutton and Chris Cannam, Queen Mary, University of London.
Chris@498 7 Copyright 2007-2008 QMUL.
Chris@498 8
Chris@498 9 This program is free software; you can redistribute it and/or
Chris@498 10 modify it under the terms of the GNU General Public License as
Chris@498 11 published by the Free Software Foundation; either version 2 of the
Chris@498 12 License, or (at your option) any later version. See the file
Chris@498 13 COPYING included with this distribution for more information.
Chris@498 14 */
Chris@498 15
Chris@498 16 #include <fstream>
Chris@498 17
Chris@498 18 #include "vamp-hostsdk/PluginHostAdapter.h"
Chris@498 19 #include "vamp-hostsdk/PluginLoader.h"
Chris@498 20
Chris@498 21 #include "RDFFeatureWriter.h"
Chris@498 22 #include "RDFTransformFactory.h"
Chris@498 23
Chris@498 24 #include <QTextStream>
Chris@498 25 #include <QUrl>
Chris@508 26 #include <QFileInfo>
Chris@498 27 #include <QRegExp>
Chris@498 28
Chris@498 29 using namespace std;
Chris@498 30 using Vamp::Plugin;
Chris@498 31 using Vamp::PluginBase;
Chris@498 32
Chris@498 33 RDFFeatureWriter::RDFFeatureWriter() :
Chris@498 34 FileFeatureWriter(SupportOneFilePerTrackTransform |
Chris@498 35 SupportOneFilePerTrack |
Chris@498 36 SupportOneFileTotal,
Chris@498 37 "n3"),
Chris@498 38 m_plain(false),
Chris@498 39 m_count(0)
Chris@498 40 {
Chris@498 41 }
Chris@498 42
Chris@498 43 RDFFeatureWriter::~RDFFeatureWriter()
Chris@498 44 {
Chris@498 45 }
Chris@498 46
Chris@498 47 RDFFeatureWriter::ParameterList
Chris@498 48 RDFFeatureWriter::getSupportedParameters() const
Chris@498 49 {
Chris@498 50 ParameterList pl = FileFeatureWriter::getSupportedParameters();
Chris@498 51 Parameter p;
Chris@498 52
Chris@498 53 p.name = "plain";
Chris@498 54 p.description = "Use \"plain\" RDF even if transform metadata is available.";
Chris@498 55 p.hasArg = false;
Chris@498 56 pl.push_back(p);
Chris@498 57
Chris@498 58 p.name = "signal-uri";
Chris@498 59 p.description = "Link the output RDF to the given signal URI.";
Chris@498 60 p.hasArg = true;
Chris@498 61 pl.push_back(p);
Chris@498 62
Chris@498 63 return pl;
Chris@498 64 }
Chris@498 65
Chris@498 66 void
Chris@498 67 RDFFeatureWriter::setParameters(map<string, string> &params)
Chris@498 68 {
Chris@498 69 FileFeatureWriter::setParameters(params);
Chris@498 70
Chris@498 71 for (map<string, string>::iterator i = params.begin();
Chris@498 72 i != params.end(); ++i) {
Chris@498 73 if (i->first == "plain") {
Chris@498 74 m_plain = true;
Chris@498 75 }
Chris@498 76 if (i->first == "signal-uri") {
Chris@498 77 m_suri = i->second.c_str();
Chris@498 78 }
Chris@498 79 }
Chris@498 80 }
Chris@498 81
Chris@504 82 void
Chris@504 83 RDFFeatureWriter::setTrackMetadata(QString trackId,
Chris@504 84 TrackMetadata metadata)
Chris@504 85 {
Chris@504 86 m_metadata[trackId] = metadata;
Chris@504 87 }
Chris@504 88
Chris@504 89 void
Chris@510 90 RDFFeatureWriter::setFixedEventTypeURI(QString uri)
Chris@510 91 {
Chris@510 92 m_fixedEventTypeURI = uri;
Chris@510 93 }
Chris@510 94
Chris@510 95 void
Chris@504 96 RDFFeatureWriter::write(QString trackId,
Chris@504 97 const Transform &transform,
Chris@504 98 const Plugin::OutputDescriptor& output,
Chris@504 99 const Plugin::FeatureList& features,
Chris@504 100 std::string summaryType)
Chris@498 101 {
Chris@498 102 QString pluginId = transform.getPluginIdentifier();
Chris@498 103
Chris@498 104 if (m_rdfDescriptions.find(pluginId) == m_rdfDescriptions.end()) {
Chris@498 105
Chris@498 106 m_rdfDescriptions[pluginId] = PluginRDFDescription(pluginId);
Chris@498 107
Chris@498 108 if (m_rdfDescriptions[pluginId].haveDescription()) {
Chris@498 109 cerr << "NOTE: Have RDF description for plugin ID \""
Chris@498 110 << pluginId.toStdString() << "\"" << endl;
Chris@498 111 } else {
Chris@498 112 cerr << "NOTE: Do not have RDF description for plugin ID \""
Chris@498 113 << pluginId.toStdString() << "\"" << endl;
Chris@498 114 }
Chris@498 115 }
Chris@498 116
Chris@498 117 // Need to select appropriate output file for our track/transform
Chris@498 118 // combination
Chris@498 119
Chris@498 120 QTextStream *stream = getOutputStream(trackId, transform.getIdentifier());
Chris@512 121 if (!stream) {
Chris@512 122 std::cerr << "RDFFeatureWriter::write: ERROR: No output stream for track id \""
Chris@512 123 << trackId.toStdString() << "\" and transform \""
Chris@512 124 << transform.getIdentifier().toStdString() << "\"" << std::endl;
Chris@513 125 exit(1);
Chris@512 126 }
Chris@498 127
Chris@498 128 if (m_startedStreamTransforms.find(stream) ==
Chris@498 129 m_startedStreamTransforms.end()) {
Chris@530 130 cerr << "This stream is new, writing prefixes" << endl;
Chris@498 131 writePrefixes(stream);
Chris@498 132 if (m_singleFileName == "" && !m_stdout) {
Chris@498 133 writeSignalDescription(stream, trackId);
Chris@498 134 }
Chris@498 135 }
Chris@498 136
Chris@498 137 if (m_startedStreamTransforms[stream].find(transform) ==
Chris@498 138 m_startedStreamTransforms[stream].end()) {
Chris@498 139 m_startedStreamTransforms[stream].insert(transform);
Chris@498 140 writeLocalFeatureTypes
Chris@498 141 (stream, transform, output, m_rdfDescriptions[pluginId]);
Chris@498 142 }
Chris@498 143
Chris@498 144 if (m_singleFileName != "" || m_stdout) {
Chris@498 145 if (m_startedTrackIds.find(trackId) == m_startedTrackIds.end()) {
Chris@498 146 writeSignalDescription(stream, trackId);
Chris@498 147 m_startedTrackIds.insert(trackId);
Chris@498 148 }
Chris@498 149 }
Chris@498 150
Chris@498 151 QString timelineURI = m_trackTimelineURIs[trackId];
Chris@498 152
Chris@498 153 if (timelineURI == "") {
Chris@498 154 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing features without having established a timeline URI!" << endl;
Chris@498 155 exit(1);
Chris@498 156 }
Chris@498 157
Chris@498 158 if (summaryType != "") {
Chris@498 159
Chris@498 160 writeSparseRDF(stream, transform, output, features,
Chris@498 161 m_rdfDescriptions[pluginId], timelineURI);
Chris@498 162
Chris@498 163 } else if (m_rdfDescriptions[pluginId].haveDescription() &&
Chris@498 164 m_rdfDescriptions[pluginId].getOutputDisposition
Chris@498 165 (output.identifier.c_str()) ==
Chris@498 166 PluginRDFDescription::OutputDense) {
Chris@498 167
Chris@498 168 QString signalURI = m_trackSignalURIs[trackId];
Chris@498 169
Chris@498 170 if (signalURI == "") {
Chris@498 171 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing dense features without having established a signal URI!" << endl;
Chris@498 172 exit(1);
Chris@498 173 }
Chris@498 174
Chris@498 175 writeDenseRDF(stream, transform, output, features,
Chris@498 176 m_rdfDescriptions[pluginId], signalURI, timelineURI);
Chris@498 177
Chris@507 178 } else if (!m_plain &&
Chris@507 179 m_rdfDescriptions[pluginId].haveDescription() &&
Chris@507 180 m_rdfDescriptions[pluginId].getOutputDisposition
Chris@507 181 (output.identifier.c_str()) ==
Chris@507 182 PluginRDFDescription::OutputTrackLevel &&
Chris@507 183 m_rdfDescriptions[pluginId].getOutputFeatureAttributeURI
Chris@507 184 (output.identifier.c_str()) != "") {
Chris@507 185
Chris@507 186 QString signalURI = m_trackSignalURIs[trackId];
Chris@507 187
Chris@507 188 if (signalURI == "") {
Chris@507 189 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing track-level features without having established a signal URI!" << endl;
Chris@507 190 exit(1);
Chris@507 191 }
Chris@507 192
Chris@507 193 writeTrackLevelRDF(stream, transform, output, features,
Chris@507 194 m_rdfDescriptions[pluginId], signalURI);
Chris@507 195
Chris@498 196 } else {
Chris@498 197
Chris@498 198 writeSparseRDF(stream, transform, output, features,
Chris@498 199 m_rdfDescriptions[pluginId], timelineURI);
Chris@498 200 }
Chris@498 201 }
Chris@498 202
Chris@498 203 void
Chris@498 204 RDFFeatureWriter::writePrefixes(QTextStream *sptr)
Chris@498 205 {
Chris@498 206 QTextStream &stream = *sptr;
Chris@498 207
Chris@498 208 stream << "@prefix dc: <http://purl.org/dc/elements/1.1/> .\n"
Chris@498 209 << "@prefix mo: <http://purl.org/ontology/mo/> .\n"
Chris@498 210 << "@prefix af: <http://purl.org/ontology/af/> .\n"
Chris@504 211 << "@prefix foaf: <http://xmlns.com/foaf/0.1/> . \n"
Chris@498 212 << "@prefix event: <http://purl.org/NET/c4dm/event.owl#> .\n"
Chris@498 213 << "@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> .\n"
Chris@498 214 << "@prefix rdfs: <http://www.w3.org/2000/01/rdf-schema#> .\n"
Chris@498 215 << "@prefix xsd: <http://www.w3.org/2001/XMLSchema#> .\n"
Chris@498 216 << "@prefix tl: <http://purl.org/NET/c4dm/timeline.owl#> .\n"
Chris@498 217 << "@prefix vamp: <http://purl.org/ontology/vamp/> .\n"
Chris@498 218 << "@prefix : <#> .\n\n";
Chris@498 219 }
Chris@498 220
Chris@498 221 void
Chris@498 222 RDFFeatureWriter::writeSignalDescription(QTextStream *sptr,
Chris@498 223 QString trackId)
Chris@498 224 {
Chris@530 225 std::cerr << "RDFFeatureWriter::writeSignalDescription" << std::endl;
Chris@530 226
Chris@498 227 QTextStream &stream = *sptr;
Chris@498 228
Chris@498 229 /*
Chris@498 230 * Describe signal we're analysing (AudioFile, Signal, TimeLine, etc.)
Chris@498 231 */
Chris@498 232
Chris@498 233 QUrl url(trackId);
Chris@498 234 QString scheme = url.scheme().toLower();
Chris@498 235 bool local = (scheme == "" || scheme == "file" || scheme.length() == 1);
Chris@498 236
Chris@498 237 if (local) {
Chris@498 238 if (scheme == "") {
Chris@498 239 url.setScheme("file");
Chris@508 240 url.setPath(QFileInfo(url.path()).absoluteFilePath());
Chris@498 241 } else if (scheme.length() == 1) { // DOS drive letter!
Chris@498 242 url.setScheme("file");
Chris@498 243 url.setPath(scheme + ":" + url.path());
Chris@498 244 }
Chris@498 245 }
Chris@498 246
Chris@498 247 //!!! FIX: If we are appending, we need to start counting after
Chris@498 248 //all of the existing counts that are already in the file!
Chris@498 249
Chris@498 250 uint64_t signalCount = m_count++;
Chris@498 251
Chris@498 252 if (m_trackSignalURIs.find(trackId) == m_trackSignalURIs.end()) {
Chris@498 253 m_trackSignalURIs[trackId] = QString(":signal_%1").arg(signalCount);
Chris@498 254 }
Chris@585 255
Chris@498 256 if (m_suri != NULL) {
Chris@498 257 m_trackSignalURIs[trackId] = "<" + m_suri + ">";
Chris@498 258 }
Chris@498 259 QString signalURI = m_trackSignalURIs[trackId];
Chris@498 260
Chris@585 261 if (m_trackTrackURIs.find(trackId) == m_trackTrackURIs.end()) {
Chris@585 262 m_trackTrackURIs[trackId] = QString(":track_%1").arg(signalCount);
Chris@585 263 }
Chris@585 264 QString trackURI = m_trackTrackURIs[trackId];
Chris@585 265
Chris@498 266 if (m_trackTimelineURIs.find(trackId) == m_trackTimelineURIs.end()) {
Chris@498 267 m_trackTimelineURIs[trackId] = QString(":signal_timeline_%1").arg(signalCount);
Chris@498 268 }
Chris@498 269 QString timelineURI = m_trackTimelineURIs[trackId];
Chris@498 270
Chris@585 271 if (m_metadata.find(trackId) != m_metadata.end()) {
Chris@585 272 TrackMetadata tm = m_metadata[trackId];
Chris@585 273 if (tm.title != "" || tm.maker != "") {
Chris@585 274 // We only write a Track at all if we have some
Chris@585 275 // title/artist metadata to put in it. Otherwise we can't
Chris@585 276 // be sure that what we have is a Track, in the
Chris@585 277 // publication sense -- it may just be a fragment, a test
Chris@585 278 // file, whatever. Since we'd have no metadata to
Chris@585 279 // associate with our Track, the only effect of including
Chris@585 280 // a Track would be to assert that this was one, which is
Chris@585 281 // the one thing we wouldn't know...
Chris@585 282 stream << trackURI << " a mo:Track ";
Chris@585 283 if (tm.title != "") {
Chris@585 284 stream << ";\n dc:title \"\"\"" << tm.title << "\"\"\" ";
Chris@585 285 }
Chris@585 286 if (tm.maker != "") {
Chris@585 287 stream << ";\n foaf:maker [ a mo:MusicArtist; foaf:name \"\"\"" << tm.maker << "\"\"\" ] ";
Chris@585 288 }
Chris@585 289 if (trackId != "") {
Chris@585 290 stream << ";\n mo:available_as <" << url.toEncoded().data() << "> ";
Chris@585 291 }
Chris@585 292 stream << ".\n\n";
Chris@585 293 }
Chris@585 294 }
Chris@585 295
Chris@500 296 if (trackId != "") {
Chris@585 297 stream << "<" << url.toEncoded().data() << "> a mo:AudioFile ;\n";
Chris@585 298 stream << " mo:encodes " << signalURI << ".\n\n";
Chris@500 299 }
Chris@500 300
Chris@500 301 stream << signalURI << " a mo:Signal ;\n";
Chris@500 302
Chris@500 303 stream << " mo:time [\n"
Chris@498 304 << " a tl:Interval ;\n"
Chris@498 305 << " tl:onTimeLine "
Chris@498 306 << timelineURI << "\n ] .\n\n";
Chris@585 307
Chris@585 308 stream << timelineURI << " a tl:Timeline .\n";
Chris@498 309 }
Chris@498 310
Chris@498 311 void
Chris@498 312 RDFFeatureWriter::writeLocalFeatureTypes(QTextStream *sptr,
Chris@498 313 const Transform &transform,
Chris@498 314 const Plugin::OutputDescriptor &od,
Chris@498 315 PluginRDFDescription &desc)
Chris@498 316 {
Chris@498 317 QString outputId = od.identifier.c_str();
Chris@498 318 QTextStream &stream = *sptr;
Chris@498 319
Chris@507 320 // There is no "needFeatureType" for track-level outputs, because
Chris@507 321 // we can't meaningfully write a feature at all if we don't know
Chris@507 322 // what property to use for it. If the output is track level but
Chris@507 323 // there is no feature type given, we have to revert to events.
Chris@507 324
Chris@498 325 bool needEventType = false;
Chris@498 326 bool needSignalType = false;
Chris@498 327
Chris@498 328 //!!! bin names, extents and so on can be written out using e.g. vamp:bin_names ( "a" "b" "c" )
Chris@498 329
Chris@498 330 if (desc.getOutputDisposition(outputId) ==
Chris@498 331 PluginRDFDescription::OutputDense) {
Chris@498 332
Chris@498 333 // no feature events, so may need signal type but won't need
Chris@498 334 // event type
Chris@498 335
Chris@498 336 if (m_plain) {
Chris@498 337
Chris@498 338 needSignalType = true;
Chris@498 339
Chris@498 340 } else if (desc.getOutputSignalTypeURI(outputId) == "") {
Chris@498 341
Chris@498 342 needSignalType = true;
Chris@498 343 }
Chris@498 344
Chris@507 345 } else if (desc.getOutputDisposition(outputId) ==
Chris@507 346 PluginRDFDescription::OutputTrackLevel) {
Chris@507 347
Chris@507 348 // see note above -- need to generate an event type if no
Chris@507 349 // feature type given, or if in plain mode
Chris@507 350
Chris@507 351 if (m_plain) {
Chris@507 352
Chris@507 353 needEventType = true;
Chris@507 354
Chris@507 355 } else if (desc.getOutputFeatureAttributeURI(outputId) == "") {
Chris@507 356
Chris@507 357 if (desc.getOutputEventTypeURI(outputId) == "") {
Chris@507 358
Chris@507 359 needEventType = true;
Chris@507 360 }
Chris@507 361 }
Chris@507 362
Chris@498 363 } else {
Chris@498 364
Chris@498 365 // may need event type but won't need signal type
Chris@498 366
Chris@498 367 if (m_plain) {
Chris@498 368
Chris@498 369 needEventType = true;
Chris@498 370
Chris@498 371 } else if (desc.getOutputEventTypeURI(outputId) == "") {
Chris@498 372
Chris@498 373 needEventType = true;
Chris@498 374 }
Chris@498 375 }
Chris@498 376
Chris@498 377 QString transformUri;
Chris@498 378 if (m_transformURIs.find(transform) != m_transformURIs.end()) {
Chris@498 379 transformUri = m_transformURIs[transform];
Chris@498 380 } else {
Chris@498 381 transformUri = QString(":transform_%1_%2").arg(m_count++).arg(outputId);
Chris@498 382 m_transformURIs[transform] = transformUri;
Chris@498 383 }
Chris@498 384
Chris@500 385 if (transform.getIdentifier() != "") {
Chris@508 386 stream << endl
Chris@508 387 << RDFTransformFactory::writeTransformToRDF(transform, transformUri)
Chris@500 388 << endl;
Chris@500 389 }
Chris@498 390
Chris@510 391 if (needEventType && m_fixedEventTypeURI == "") {
Chris@498 392
Chris@498 393 QString uri;
Chris@498 394 if (m_syntheticEventTypeURIs.find(transform) !=
Chris@498 395 m_syntheticEventTypeURIs.end()) {
Chris@498 396 uri = m_syntheticEventTypeURIs[transform];
Chris@498 397 } else {
Chris@498 398 uri = QString(":event_type_%1").arg(m_count++);
Chris@498 399 m_syntheticEventTypeURIs[transform] = uri;
Chris@498 400 }
Chris@498 401
Chris@498 402 stream << uri
Chris@498 403 << " rdfs:subClassOf event:Event ;" << endl
Chris@498 404 << " dc:title \"" << od.name.c_str() << "\" ;" << endl
Chris@498 405 << " dc:format \"" << od.unit.c_str() << "\" ;" << endl
Chris@498 406 << " dc:description \"" << od.description.c_str() << "\" ."
Chris@498 407 << endl << endl;
Chris@498 408 }
Chris@498 409
Chris@498 410 if (needSignalType) {
Chris@498 411
Chris@498 412 QString uri;
Chris@498 413 if (m_syntheticSignalTypeURIs.find(transform) !=
Chris@498 414 m_syntheticSignalTypeURIs.end()) {
Chris@498 415 uri = m_syntheticSignalTypeURIs[transform];
Chris@498 416 } else {
Chris@498 417 uri = QString(":signal_type_%1").arg(m_count++);
Chris@498 418 m_syntheticSignalTypeURIs[transform] = uri;
Chris@498 419 }
Chris@498 420
Chris@498 421 stream << uri
Chris@498 422 << " rdfs:subClassOf af:Signal ;" << endl
Chris@498 423 << " dc:title \"" << od.name.c_str() << "\" ;" << endl
Chris@498 424 << " dc:format \"" << od.unit.c_str() << "\" ;" << endl
Chris@498 425 << " dc:description \"" << od.description.c_str() << "\" ."
Chris@498 426 << endl << endl;
Chris@498 427 }
Chris@498 428 }
Chris@498 429
Chris@498 430 void
Chris@498 431 RDFFeatureWriter::writeSparseRDF(QTextStream *sptr,
Chris@498 432 const Transform &transform,
Chris@498 433 const Plugin::OutputDescriptor& od,
Chris@498 434 const Plugin::FeatureList& featureList,
Chris@498 435 PluginRDFDescription &desc,
Chris@498 436 QString timelineURI)
Chris@498 437 {
Chris@512 438 // std::cerr << "RDFFeatureWriter::writeSparseRDF: have " << featureList.size() << " features" << std::endl;
Chris@512 439
Chris@498 440 if (featureList.empty()) return;
Chris@498 441 QTextStream &stream = *sptr;
Chris@498 442
Chris@498 443 bool plain = (m_plain || !desc.haveDescription());
Chris@498 444
Chris@498 445 QString outputId = od.identifier.c_str();
Chris@498 446
Chris@498 447 // iterate through FeatureLists
Chris@498 448
Chris@498 449 for (int i = 0; i < featureList.size(); ++i) {
Chris@498 450
Chris@498 451 const Plugin::Feature &feature = featureList[i];
Chris@498 452 uint64_t featureNumber = m_count++;
Chris@498 453
Chris@498 454 stream << ":event_" << featureNumber << " a ";
Chris@498 455
Chris@510 456 if (m_fixedEventTypeURI != "") {
Chris@510 457 stream << m_fixedEventTypeURI << " ;\n";
Chris@510 458 } else {
Chris@510 459 QString eventTypeURI = desc.getOutputEventTypeURI(outputId);
Chris@510 460 if (plain || eventTypeURI == "") {
Chris@510 461 if (m_syntheticEventTypeURIs.find(transform) !=
Chris@510 462 m_syntheticEventTypeURIs.end()) {
Chris@510 463 stream << m_syntheticEventTypeURIs[transform] << " ;\n";
Chris@510 464 } else {
Chris@510 465 stream << ":event_type_" << outputId << " ;\n";
Chris@510 466 }
Chris@498 467 } else {
Chris@510 468 stream << "<" << eventTypeURI << "> ;\n";
Chris@498 469 }
Chris@498 470 }
Chris@498 471
Chris@498 472 QString timestamp = feature.timestamp.toString().c_str();
Chris@498 473 timestamp.replace(QRegExp("^ +"), "");
Chris@498 474
Chris@498 475 if (feature.hasDuration && feature.duration > Vamp::RealTime::zeroTime) {
Chris@498 476
Chris@498 477 QString duration = feature.duration.toString().c_str();
Chris@498 478 duration.replace(QRegExp("^ +"), "");
Chris@498 479
Chris@498 480 stream << " event:time [ \n"
Chris@498 481 << " a tl:Interval ;\n"
Chris@498 482 << " tl:onTimeLine " << timelineURI << " ;\n"
Chris@498 483 << " tl:beginsAt \"PT" << timestamp
Chris@498 484 << "S\"^^xsd:duration ;\n"
Chris@498 485 << " tl:duration \"PT" << duration
Chris@498 486 << "S\"^^xsd:duration ;\n"
Chris@498 487 << " ] ";
Chris@498 488
Chris@498 489 } else {
Chris@498 490
Chris@498 491 stream << " event:time [ \n"
Chris@498 492 << " a tl:Instant ;\n" //location of the event in time
Chris@498 493 << " tl:onTimeLine " << timelineURI << " ;\n"
Chris@498 494 << " tl:at \"PT" << timestamp
Chris@498 495 << "S\"^^xsd:duration ;\n ] ";
Chris@498 496 }
Chris@498 497
Chris@500 498 if (transform.getIdentifier() != "") {
Chris@500 499 stream << ";\n";
Chris@500 500 stream << " vamp:computed_by " << m_transformURIs[transform] << " ";
Chris@500 501 }
Chris@498 502
Chris@498 503 if (feature.label.length() > 0) {
Chris@498 504 stream << ";\n";
Chris@510 505 stream << " rdfs:label \"\"\"" << feature.label.c_str() << "\"\"\" ";
Chris@498 506 }
Chris@498 507
Chris@498 508 if (!feature.values.empty()) {
Chris@498 509 stream << ";\n";
Chris@498 510 //!!! named bins?
Chris@498 511 stream << " af:feature \"" << feature.values[0];
Chris@498 512 for (int j = 1; j < feature.values.size(); ++j) {
Chris@498 513 stream << " " << feature.values[j];
Chris@498 514 }
Chris@498 515 stream << "\" ";
Chris@498 516 }
Chris@498 517
Chris@498 518 stream << ".\n";
Chris@498 519 }
Chris@498 520 }
Chris@498 521
Chris@498 522 void
Chris@507 523 RDFFeatureWriter::writeTrackLevelRDF(QTextStream *sptr,
Chris@507 524 const Transform &transform,
Chris@507 525 const Plugin::OutputDescriptor& od,
Chris@507 526 const Plugin::FeatureList& featureList,
Chris@507 527 PluginRDFDescription &desc,
Chris@507 528 QString signalURI)
Chris@507 529 {
Chris@507 530 if (featureList.empty()) return;
Chris@507 531 QTextStream &stream = *sptr;
Chris@507 532
Chris@507 533 bool plain = (m_plain || !desc.haveDescription());
Chris@507 534
Chris@507 535 QString outputId = od.identifier.c_str();
Chris@507 536 QString featureUri = desc.getOutputFeatureAttributeURI(outputId);
Chris@507 537
Chris@507 538 if (featureUri == "") {
Chris@507 539 cerr << "RDFFeatureWriter::writeTrackLevelRDF: ERROR: No feature URI available -- this function should not have been called!" << endl;
Chris@507 540 return;
Chris@507 541 }
Chris@507 542
Chris@507 543 for (int i = 0; i < featureList.size(); ++i) {
Chris@507 544
Chris@507 545 const Plugin::Feature &feature = featureList[i];
Chris@507 546
Chris@507 547 if (feature.values.empty()) {
Chris@507 548
Chris@507 549 if (feature.label == "") continue;
Chris@507 550
Chris@508 551 stream << signalURI << " " << featureUri << " \"\"\""
Chris@508 552 << feature.label.c_str() << "\"\"\" .\n";
Chris@507 553
Chris@507 554 } else {
Chris@507 555
Chris@507 556 stream << signalURI << " " << featureUri << " \""
Chris@507 557 << feature.values[0] << "\"^^xsd:float .\n";
Chris@507 558 }
Chris@507 559 }
Chris@507 560 }
Chris@507 561
Chris@507 562 void
Chris@498 563 RDFFeatureWriter::writeDenseRDF(QTextStream *sptr,
Chris@498 564 const Transform &transform,
Chris@498 565 const Plugin::OutputDescriptor& od,
Chris@498 566 const Plugin::FeatureList& featureList,
Chris@498 567 PluginRDFDescription &desc,
Chris@498 568 QString signalURI,
Chris@498 569 QString timelineURI)
Chris@498 570 {
Chris@498 571 if (featureList.empty()) return;
Chris@498 572
Chris@498 573 StringTransformPair sp(signalURI, transform);
Chris@498 574
Chris@498 575 if (m_openDenseFeatures.find(sp) == m_openDenseFeatures.end()) {
Chris@498 576
Chris@498 577 StreamBuffer b(sptr, "");
Chris@498 578 m_openDenseFeatures[sp] = b;
Chris@498 579
Chris@498 580 QString &str(m_openDenseFeatures[sp].second);
Chris@498 581 QTextStream stream(&str);
Chris@498 582
Chris@498 583 bool plain = (m_plain || !desc.haveDescription());
Chris@498 584 QString outputId = od.identifier.c_str();
Chris@498 585
Chris@498 586 uint64_t featureNumber = m_count++;
Chris@498 587
Chris@498 588 // need to write out feature timeline map -- for this we need
Chris@498 589 // the sample rate, window length and hop size from the
Chris@498 590 // transform
Chris@498 591
Chris@498 592 stream << "\n:feature_timeline_" << featureNumber << " a tl:DiscreteTimeLine .\n\n";
Chris@498 593
Chris@498 594 size_t stepSize = transform.getStepSize();
Chris@498 595 if (stepSize == 0) {
Chris@498 596 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing dense features without having set the step size properly!" << endl;
Chris@498 597 return;
Chris@498 598 }
Chris@498 599
Chris@498 600 size_t blockSize = transform.getBlockSize();
Chris@498 601 if (blockSize == 0) {
Chris@498 602 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing dense features without having set the block size properly!" << endl;
Chris@498 603 return;
Chris@498 604 }
Chris@498 605
Chris@498 606 float sampleRate = transform.getSampleRate();
Chris@498 607 if (sampleRate == 0.f) {
Chris@498 608 cerr << "RDFFeatureWriter: INTERNAL ERROR: writing dense features without having set the sample rate properly!" << endl;
Chris@498 609 return;
Chris@498 610 }
Chris@498 611
Chris@498 612 stream << ":feature_timeline_map_" << featureNumber
Chris@498 613 << " a tl:UniformSamplingWindowingMap ;\n"
Chris@498 614 << " tl:rangeTimeLine :feature_timeline_" << featureNumber << " ;\n"
Chris@498 615 << " tl:domainTimeLine " << timelineURI << " ;\n"
Chris@498 616 << " tl:sampleRate \"" << int(sampleRate) << "\"^^xsd:int ;\n"
Chris@498 617 << " tl:windowLength \"" << blockSize << "\"^^xsd:int ;\n"
Chris@498 618 << " tl:hopSize \"" << stepSize << "\"^^xsd:int .\n\n";
Chris@498 619
Chris@498 620 stream << signalURI << " af:signal_feature :feature_"
Chris@498 621 << featureNumber << " ." << endl << endl;
Chris@498 622
Chris@498 623 stream << ":feature_" << featureNumber << " a ";
Chris@498 624
Chris@498 625 QString signalTypeURI = desc.getOutputSignalTypeURI(outputId);
Chris@498 626 if (plain || signalTypeURI == "") {
Chris@498 627 if (m_syntheticSignalTypeURIs.find(transform) !=
Chris@498 628 m_syntheticSignalTypeURIs.end()) {
Chris@498 629 stream << m_syntheticSignalTypeURIs[transform] << " ;\n";
Chris@498 630 } else {
Chris@498 631 stream << ":signal_type_" << outputId << " ;\n";
Chris@498 632 }
Chris@498 633 } else {
Chris@498 634 stream << signalTypeURI << " ;\n";
Chris@498 635 }
Chris@498 636
Chris@498 637 stream << " mo:time ["
Chris@498 638 << "\n a tl:Interval ;"
Chris@498 639 << "\n tl:onTimeLine :feature_timeline_" << featureNumber << " ;";
Chris@498 640
Chris@498 641 RealTime startrt = transform.getStartTime();
Chris@498 642 RealTime durationrt = transform.getDuration();
Chris@498 643
Chris@498 644 int start = RealTime::realTime2Frame(startrt, sampleRate) / stepSize;
Chris@498 645 int duration = RealTime::realTime2Frame(durationrt, sampleRate) / stepSize;
Chris@498 646
Chris@498 647 if (start != 0) {
Chris@498 648 stream << "\n tl:start \"" << start << "\"^^xsd:int ;";
Chris@498 649 }
Chris@498 650 if (duration != 0) {
Chris@498 651 stream << "\n tl:duration \"" << duration << "\"^^xsd:int ;";
Chris@498 652 }
Chris@498 653
Chris@498 654 stream << "\n ] ;\n";
Chris@498 655
Chris@584 656 if (transform.getIdentifier() != "") {
Chris@584 657 stream << " vamp:computed_by " << m_transformURIs[transform] << " ;\n";
Chris@584 658 }
Chris@584 659
Chris@498 660 if (od.hasFixedBinCount) {
Chris@498 661 // We only know the height, so write the width as zero
Chris@498 662 stream << " af:dimensions \"" << od.binCount << " 0\" ;\n";
Chris@498 663 }
Chris@498 664
Chris@498 665 stream << " af:value \"";
Chris@498 666 }
Chris@498 667
Chris@498 668 QString &str = m_openDenseFeatures[sp].second;
Chris@498 669 QTextStream stream(&str);
Chris@498 670
Chris@498 671 for (int i = 0; i < featureList.size(); ++i) {
Chris@498 672
Chris@498 673 const Plugin::Feature &feature = featureList[i];
Chris@498 674
Chris@498 675 for (int j = 0; j < feature.values.size(); ++j) {
Chris@498 676 stream << feature.values[j] << " ";
Chris@498 677 }
Chris@498 678 }
Chris@498 679 }
Chris@498 680
Chris@498 681 void RDFFeatureWriter::finish()
Chris@498 682 {
Chris@578 683 // cerr << "RDFFeatureWriter::finish()" << endl;
Chris@498 684
Chris@498 685 // close any open dense feature literals
Chris@498 686
Chris@498 687 for (map<StringTransformPair, StreamBuffer>::iterator i =
Chris@498 688 m_openDenseFeatures.begin();
Chris@498 689 i != m_openDenseFeatures.end(); ++i) {
Chris@512 690 // cerr << "closing a stream" << endl;
Chris@498 691 StreamBuffer &b = i->second;
Chris@498 692 *(b.first) << b.second << "\" ." << endl;
Chris@498 693 }
Chris@498 694
Chris@498 695 m_openDenseFeatures.clear();
Chris@530 696 m_startedStreamTransforms.clear();
Chris@530 697
Chris@530 698 FileFeatureWriter::finish();
Chris@498 699 }
Chris@498 700
Chris@498 701