Chris@145
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@145
|
2
|
Chris@145
|
3 /*
|
Chris@145
|
4 Sonic Annotator
|
Chris@145
|
5 A utility for batch feature extraction from audio files.
|
Chris@145
|
6 Mark Levy, Chris Sutton and Chris Cannam, Queen Mary, University of London.
|
Chris@145
|
7 Copyright 2007-2014 QMUL.
|
Chris@145
|
8
|
Chris@145
|
9 This program is free software; you can redistribute it and/or
|
Chris@145
|
10 modify it under the terms of the GNU General Public License as
|
Chris@145
|
11 published by the Free Software Foundation; either version 2 of the
|
Chris@145
|
12 License, or (at your option) any later version. See the file
|
Chris@145
|
13 COPYING included with this distribution for more information.
|
Chris@145
|
14 */
|
Chris@145
|
15
|
Chris@145
|
16 #include "JAMSFeatureWriter.h"
|
Chris@145
|
17
|
Chris@145
|
18 using namespace std;
|
Chris@145
|
19 using Vamp::Plugin;
|
Chris@145
|
20 using Vamp::PluginBase;
|
Chris@145
|
21
|
Chris@145
|
22 #include "base/Exceptions.h"
|
Chris@145
|
23 #include "rdf/PluginRDFIndexer.h"
|
Chris@145
|
24
|
Chris@145
|
25 JAMSFeatureWriter::JAMSFeatureWriter() :
|
Chris@145
|
26 FileFeatureWriter(SupportOneFilePerTrackTransform |
|
Chris@145
|
27 SupportOneFilePerTrack |
|
Chris@152
|
28 SupportOneFileTotal |
|
Chris@145
|
29 SupportStdOut,
|
Chris@145
|
30 "json"),
|
Chris@145
|
31 m_network(false),
|
Chris@145
|
32 m_networkRetrieved(false)
|
Chris@145
|
33 {
|
Chris@145
|
34 }
|
Chris@145
|
35
|
Chris@145
|
36 JAMSFeatureWriter::~JAMSFeatureWriter()
|
Chris@145
|
37 {
|
Chris@145
|
38 }
|
Chris@145
|
39
|
Chris@145
|
40 string
|
Chris@145
|
41 JAMSFeatureWriter::getDescription() const
|
Chris@145
|
42 {
|
Chris@145
|
43 return "Write features to JSON files in JAMS (JSON Annotated Music Specification) format.";
|
Chris@145
|
44 }
|
Chris@145
|
45
|
Chris@145
|
46 JAMSFeatureWriter::ParameterList
|
Chris@145
|
47 JAMSFeatureWriter::getSupportedParameters() const
|
Chris@145
|
48 {
|
Chris@145
|
49 ParameterList pl = FileFeatureWriter::getSupportedParameters();
|
Chris@145
|
50 Parameter p;
|
Chris@145
|
51
|
Chris@145
|
52 p.name = "network";
|
Chris@145
|
53 p.description = "Attempt to retrieve RDF descriptions of plugins from network, if not available locally";
|
Chris@145
|
54 p.hasArg = false;
|
Chris@145
|
55 pl.push_back(p);
|
Chris@145
|
56
|
Chris@145
|
57 return pl;
|
Chris@145
|
58 }
|
Chris@145
|
59
|
Chris@145
|
60 void
|
Chris@145
|
61 JAMSFeatureWriter::setParameters(map<string, string> ¶ms)
|
Chris@145
|
62 {
|
Chris@145
|
63 FileFeatureWriter::setParameters(params);
|
Chris@145
|
64
|
Chris@145
|
65 for (map<string, string>::iterator i = params.begin();
|
Chris@145
|
66 i != params.end(); ++i) {
|
Chris@145
|
67 if (i->first == "network") {
|
Chris@145
|
68 m_network = true;
|
Chris@145
|
69 }
|
Chris@145
|
70 }
|
Chris@145
|
71 }
|
Chris@145
|
72
|
Chris@145
|
73 void
|
Chris@145
|
74 JAMSFeatureWriter::setTrackMetadata(QString trackId, TrackMetadata metadata)
|
Chris@145
|
75 {
|
Chris@145
|
76 QString json
|
Chris@153
|
77 ("\n\"file_metadata\":\n"
|
Chris@152
|
78 " { \"artist\": \"%1\",\n"
|
Chris@152
|
79 " \"title\": \"%2\" },\n");
|
Chris@145
|
80 m_metadata[trackId] = json.arg(metadata.maker).arg(metadata.title);
|
Chris@152
|
81 cerr << "setTrackMetadata: metadata is: " << m_metadata[trackId] << endl;
|
Chris@145
|
82 }
|
Chris@145
|
83
|
Chris@153
|
84 static double
|
Chris@153
|
85 realTime2Sec(const Vamp::RealTime &r)
|
Chris@153
|
86 {
|
Chris@153
|
87 return r / Vamp::RealTime(1, 0);
|
Chris@153
|
88 }
|
Chris@153
|
89
|
Chris@145
|
90 void
|
Chris@145
|
91 JAMSFeatureWriter::write(QString trackId,
|
Chris@145
|
92 const Transform &transform,
|
Chris@145
|
93 const Plugin::OutputDescriptor& ,
|
Chris@145
|
94 const Plugin::FeatureList& features,
|
Chris@145
|
95 std::string /* summaryType */)
|
Chris@145
|
96 {
|
Chris@145
|
97 QString transformId = transform.getIdentifier();
|
Chris@145
|
98
|
Chris@145
|
99 QTextStream *sptr = getOutputStream(trackId, transformId);
|
Chris@145
|
100 if (!sptr) {
|
Chris@145
|
101 throw FailedToOpenOutputStream(trackId, transformId);
|
Chris@145
|
102 }
|
Chris@145
|
103
|
Chris@145
|
104 QTextStream &stream = *sptr;
|
Chris@145
|
105
|
Chris@152
|
106 TrackTransformPair tt(trackId, transformId);
|
Chris@152
|
107 TrackTransformPair targetKey = getFilenameKey(trackId, transformId);
|
Chris@152
|
108
|
Chris@152
|
109 if (m_startedTargets.find(targetKey) == m_startedTargets.end()) {
|
Chris@152
|
110 // Need to write track-level preamble
|
Chris@152
|
111 stream << "{" << m_metadata[trackId] << endl;
|
Chris@152
|
112 m_startedTargets.insert(targetKey);
|
Chris@152
|
113 }
|
Chris@152
|
114
|
Chris@153
|
115 bool justBegun = false;
|
Chris@153
|
116
|
Chris@152
|
117 if (m_data.find(tt) == m_data.end()) {
|
Chris@145
|
118
|
Chris@145
|
119 identifyTask(transform);
|
Chris@145
|
120
|
Chris@152
|
121 QString json("\"%1\": [ ");
|
Chris@152
|
122 m_data[tt] = json.arg(getTaskKey(m_tasks[transformId]));
|
Chris@153
|
123 justBegun = true;
|
Chris@145
|
124 }
|
Chris@145
|
125
|
Chris@153
|
126 QString d = m_data[tt];
|
Chris@153
|
127
|
Chris@145
|
128 for (int i = 0; i < int(features.size()); ++i) {
|
Chris@153
|
129
|
Chris@153
|
130 if (i > 0 || !justBegun) {
|
Chris@153
|
131 d += ",\n";
|
Chris@153
|
132 } else {
|
Chris@153
|
133 d += "\n";
|
Chris@153
|
134 }
|
Chris@153
|
135
|
Chris@153
|
136 d += " { ";
|
Chris@145
|
137
|
Chris@153
|
138 Plugin::Feature f(features[i]);
|
Chris@153
|
139
|
Chris@153
|
140 switch (m_tasks[transformId]) {
|
Chris@153
|
141
|
Chris@153
|
142 case ChordTask:
|
Chris@153
|
143 case SegmentTask:
|
Chris@153
|
144 case NoteTask:
|
Chris@153
|
145 case UnknownTask:
|
Chris@153
|
146 if (f.hasDuration) {
|
Chris@153
|
147 d += QString
|
Chris@153
|
148 ("\"start\": { \"value\": %1 }, "
|
Chris@153
|
149 "\"end\": { \"value\": %2 }")
|
Chris@153
|
150 .arg(realTime2Sec(f.timestamp))
|
Chris@153
|
151 .arg(realTime2Sec
|
Chris@153
|
152 (f.timestamp +
|
Chris@153
|
153 (f.hasDuration ? f.duration : Vamp::RealTime::zeroTime)));
|
Chris@153
|
154 break;
|
Chris@153
|
155 } else {
|
Chris@153
|
156 // don't break; fall through to simpler no-duration case
|
Chris@153
|
157 }
|
Chris@153
|
158
|
Chris@153
|
159 case BeatTask:
|
Chris@153
|
160 case KeyTask:
|
Chris@153
|
161 case OnsetTask:
|
Chris@153
|
162 d += QString("\"time\": { \"value\": %1 }")
|
Chris@153
|
163 .arg(realTime2Sec(f.timestamp));
|
Chris@153
|
164 break;
|
Chris@161
|
165
|
Chris@161
|
166 case MelodyTask:
|
Chris@161
|
167 case PitchTask:
|
Chris@161
|
168 //!!!
|
Chris@161
|
169 break;
|
Chris@153
|
170 }
|
Chris@153
|
171
|
Chris@153
|
172 if (f.label != "") {
|
Chris@153
|
173 d += QString(", \"label\": { \"value\": \"%2\" }")
|
Chris@153
|
174 .arg(f.label.c_str());
|
Chris@153
|
175 } else if (f.values.size() > 0) {
|
Chris@153
|
176 d += QString(", \"label\": { \"value\": \"%2\" }")
|
Chris@153
|
177 .arg(f.values[0]);
|
Chris@153
|
178 }
|
Chris@153
|
179
|
Chris@153
|
180 d += " }";
|
Chris@145
|
181 }
|
Chris@153
|
182
|
Chris@153
|
183 m_data[tt] = d;
|
Chris@145
|
184 }
|
Chris@145
|
185
|
Chris@145
|
186 void
|
Chris@152
|
187 JAMSFeatureWriter::finish()
|
Chris@152
|
188 {
|
Chris@152
|
189 cerr << "Finish called on " << this << endl;
|
Chris@152
|
190
|
Chris@152
|
191 set<QTextStream *> startedStreams;
|
Chris@152
|
192
|
Chris@152
|
193 for (DataMap::const_iterator i = m_data.begin();
|
Chris@152
|
194 i != m_data.end(); ++i) {
|
Chris@152
|
195
|
Chris@152
|
196 TrackTransformPair tt = i->first;
|
Chris@152
|
197 QString data = i->second;
|
Chris@152
|
198
|
Chris@152
|
199 QTextStream *sptr = getOutputStream(tt.first, tt.second);
|
Chris@152
|
200 if (!sptr) {
|
Chris@152
|
201 throw FailedToOpenOutputStream(tt.first, tt.second);
|
Chris@152
|
202 }
|
Chris@152
|
203
|
Chris@152
|
204 if (startedStreams.find(sptr) != startedStreams.end()) {
|
Chris@152
|
205 *sptr << "," << endl;
|
Chris@152
|
206 }
|
Chris@152
|
207 startedStreams.insert(sptr);
|
Chris@152
|
208
|
Chris@153
|
209 *sptr << data << "\n ]";
|
Chris@152
|
210 }
|
Chris@152
|
211
|
Chris@152
|
212 for (FileStreamMap::const_iterator i = m_streams.begin();
|
Chris@152
|
213 i != m_streams.end(); ++i) {
|
Chris@152
|
214 *(i->second) << endl << "}" << endl;
|
Chris@152
|
215 }
|
Chris@152
|
216
|
Chris@152
|
217 m_data.clear();
|
Chris@152
|
218 m_startedTargets.clear();
|
Chris@152
|
219
|
Chris@152
|
220 FileFeatureWriter::finish();
|
Chris@152
|
221 }
|
Chris@152
|
222
|
Chris@152
|
223 void
|
Chris@145
|
224 JAMSFeatureWriter::loadRDFDescription(const Transform &transform)
|
Chris@145
|
225 {
|
Chris@145
|
226 QString pluginId = transform.getPluginIdentifier();
|
Chris@145
|
227 if (m_rdfDescriptions.find(pluginId) != m_rdfDescriptions.end()) return;
|
Chris@145
|
228
|
Chris@145
|
229 if (m_network && !m_networkRetrieved) {
|
Chris@145
|
230 PluginRDFIndexer::getInstance()->indexConfiguredURLs();
|
Chris@145
|
231 m_networkRetrieved = true;
|
Chris@145
|
232 }
|
Chris@145
|
233
|
Chris@145
|
234 m_rdfDescriptions[pluginId] = PluginRDFDescription(pluginId);
|
Chris@145
|
235
|
Chris@145
|
236 if (m_rdfDescriptions[pluginId].haveDescription()) {
|
Chris@145
|
237 cerr << "NOTE: Have RDF description for plugin ID \""
|
Chris@145
|
238 << pluginId << "\"" << endl;
|
Chris@145
|
239 } else {
|
Chris@145
|
240 cerr << "NOTE: No RDF description for plugin ID \""
|
Chris@145
|
241 << pluginId << "\"" << endl;
|
Chris@145
|
242 if (!m_network) {
|
Chris@145
|
243 cerr << " Consider using the --json-network option to retrieve plugin descriptions" << endl;
|
Chris@145
|
244 cerr << " from the network where possible." << endl;
|
Chris@145
|
245 }
|
Chris@145
|
246 }
|
Chris@145
|
247 }
|
Chris@145
|
248
|
Chris@145
|
249 void
|
Chris@145
|
250 JAMSFeatureWriter::identifyTask(const Transform &transform)
|
Chris@145
|
251 {
|
Chris@145
|
252 QString transformId = transform.getIdentifier();
|
Chris@145
|
253 if (m_tasks.find(transformId) != m_tasks.end()) return;
|
Chris@145
|
254
|
Chris@145
|
255 loadRDFDescription(transform);
|
Chris@145
|
256
|
Chris@145
|
257 Task task = UnknownTask;
|
Chris@145
|
258
|
Chris@145
|
259 QString pluginId = transform.getPluginIdentifier();
|
Chris@145
|
260 QString outputId = transform.getOutput();
|
Chris@145
|
261
|
Chris@145
|
262 const PluginRDFDescription &desc = m_rdfDescriptions[pluginId];
|
Chris@145
|
263
|
Chris@145
|
264 if (desc.haveDescription()) {
|
Chris@145
|
265
|
Chris@145
|
266 PluginRDFDescription::OutputDisposition disp =
|
Chris@145
|
267 desc.getOutputDisposition(outputId);
|
Chris@145
|
268
|
Chris@145
|
269 QString af = "http://purl.org/ontology/af/";
|
Chris@145
|
270
|
Chris@145
|
271 if (disp == PluginRDFDescription::OutputSparse) {
|
Chris@145
|
272
|
Chris@145
|
273 QString eventUri = desc.getOutputEventTypeURI(outputId);
|
Chris@145
|
274
|
Chris@145
|
275 //!!! todo: allow user to prod writer for task type
|
Chris@145
|
276
|
Chris@145
|
277 if (eventUri == af + "Note") {
|
Chris@145
|
278 task = NoteTask;
|
Chris@145
|
279 } else if (eventUri == af + "Beat") {
|
Chris@145
|
280 task = BeatTask;
|
Chris@145
|
281 } else if (eventUri == af + "ChordSegment") {
|
Chris@145
|
282 task = ChordTask;
|
Chris@145
|
283 } else if (eventUri == af + "KeyChange") {
|
Chris@145
|
284 task = KeyTask;
|
Chris@145
|
285 } else if (eventUri == af + "KeySegment") {
|
Chris@145
|
286 task = KeyTask;
|
Chris@145
|
287 } else if (eventUri == af + "Onset") {
|
Chris@145
|
288 task = OnsetTask;
|
Chris@145
|
289 } else if (eventUri == af + "NonTonalOnset") {
|
Chris@145
|
290 task = OnsetTask;
|
Chris@145
|
291 } else if (eventUri == af + "Segment") {
|
Chris@145
|
292 task = SegmentTask;
|
Chris@145
|
293 } else if (eventUri == af + "SpeechSegment") {
|
Chris@145
|
294 task = SegmentTask;
|
Chris@145
|
295 } else if (eventUri == af + "StructuralSegment") {
|
Chris@145
|
296 task = SegmentTask;
|
Chris@145
|
297 } else {
|
Chris@145
|
298 cerr << "WARNING: Unsupported event type URI <"
|
Chris@145
|
299 << eventUri << ">, proceeding with UnknownTask type"
|
Chris@145
|
300 << endl;
|
Chris@145
|
301 }
|
Chris@145
|
302
|
Chris@145
|
303 } else {
|
Chris@145
|
304
|
Chris@145
|
305 cerr << "WARNING: Cannot currently write dense or track-level outputs to JSON format (only sparse ones). Will proceed using UnknownTask type, but this probably isn't going to work" << endl;
|
Chris@145
|
306 }
|
Chris@145
|
307 }
|
Chris@145
|
308
|
Chris@145
|
309 m_tasks[transformId] = task;
|
Chris@145
|
310 }
|
Chris@145
|
311
|
Chris@145
|
312 QString
|
Chris@145
|
313 JAMSFeatureWriter::getTaskKey(Task task)
|
Chris@145
|
314 {
|
Chris@145
|
315 switch (task) {
|
Chris@145
|
316 case UnknownTask: return "unknown";
|
Chris@145
|
317 case BeatTask: return "beat";
|
Chris@145
|
318 case OnsetTask: return "onset";
|
Chris@145
|
319 case ChordTask: return "chord";
|
Chris@145
|
320 case SegmentTask: return "segment";
|
Chris@145
|
321 case KeyTask: return "key";
|
Chris@145
|
322 case NoteTask: return "note";
|
Chris@145
|
323 case MelodyTask: return "melody";
|
Chris@145
|
324 case PitchTask: return "pitch";
|
Chris@145
|
325 }
|
Chris@145
|
326 return "unknown";
|
Chris@145
|
327 }
|