c@5
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
c@5
|
2
|
c@18
|
3 /*
|
c@18
|
4 VamPipe
|
c@18
|
5
|
c@18
|
6 Centre for Digital Music, Queen Mary, University of London.
|
c@18
|
7 Copyright 2015-2016 QMUL.
|
c@18
|
8
|
c@18
|
9 Permission is hereby granted, free of charge, to any person
|
c@18
|
10 obtaining a copy of this software and associated documentation
|
c@18
|
11 files (the "Software"), to deal in the Software without
|
c@18
|
12 restriction, including without limitation the rights to use, copy,
|
c@18
|
13 modify, merge, publish, distribute, sublicense, and/or sell copies
|
c@18
|
14 of the Software, and to permit persons to whom the Software is
|
c@18
|
15 furnished to do so, subject to the following conditions:
|
c@18
|
16
|
c@18
|
17 The above copyright notice and this permission notice shall be
|
c@18
|
18 included in all copies or substantial portions of the Software.
|
c@18
|
19
|
c@18
|
20 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
c@18
|
21 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
c@18
|
22 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
c@18
|
23 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
c@18
|
24 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
c@18
|
25 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
c@18
|
26 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
c@18
|
27
|
c@18
|
28 Except as contained in this notice, the names of the Centre for
|
c@18
|
29 Digital Music; Queen Mary, University of London; and Chris Cannam
|
c@18
|
30 shall not be used in advertising or otherwise to promote the sale,
|
c@18
|
31 use or other dealings in this Software without prior written
|
c@18
|
32 authorization.
|
c@18
|
33 */
|
c@18
|
34
|
c@5
|
35 #ifndef VAMP_JSON_H
|
c@5
|
36 #define VAMP_JSON_H
|
c@5
|
37
|
c@5
|
38 #include <vector>
|
c@5
|
39 #include <string>
|
c@5
|
40 #include <sstream>
|
c@5
|
41
|
c@5
|
42 #include <json11/json11.hpp>
|
c@5
|
43 #include <base-n/include/basen.hpp>
|
c@5
|
44
|
c@5
|
45 #include <vamp-hostsdk/Plugin.h>
|
c@5
|
46 #include <vamp-hostsdk/PluginLoader.h>
|
c@5
|
47
|
c@10
|
48 #include "bits/PluginHandleMapper.h"
|
c@49
|
49 #include "bits/PluginOutputIdMapper.h"
|
c@25
|
50 #include "bits/RequestResponseType.h"
|
c@10
|
51
|
c@10
|
52 namespace vampipe {
|
c@10
|
53
|
c@6
|
54 /**
|
c@6
|
55 * Convert the structures laid out in the Vamp SDK classes into JSON
|
c@6
|
56 * (and back again) following the schema in the vamp-json-schema
|
c@6
|
57 * project repo.
|
c@60
|
58 *
|
c@60
|
59 * Functions with names starting "from" convert from a Vamp SDK object
|
c@60
|
60 * to JSON output. Most of them return a json11::Json object, with a
|
c@60
|
61 * few exceptions for low-level utilities that return a string. These
|
c@60
|
62 * functions succeed all of the time.
|
c@60
|
63 *
|
c@60
|
64 * Functions with names starting "to" convert to a Vamp SDK object
|
c@60
|
65 * from JSON input. These functions all accept a json11::Json object
|
c@60
|
66 * as first argument, with a few exceptions for low-level utilities
|
c@60
|
67 * that accept a string. These functions all accept a string reference
|
c@60
|
68 * as a final argument and return an error string through it if the
|
c@60
|
69 * conversion fails. If conversion fails the return value is
|
c@60
|
70 * undefined, and any returned object may be incomplete or
|
c@60
|
71 * invalid. Callers should check for an empty error string (indicating
|
c@60
|
72 * success) before using the returned value.
|
c@6
|
73 */
|
c@60
|
74
|
c@60
|
75 //!!! todo: convert pmapper to err style
|
c@60
|
76
|
c@5
|
77 class VampJson
|
c@5
|
78 {
|
c@5
|
79 public:
|
c@45
|
80 /** Serialisation format for arrays of floats (process input and
|
c@45
|
81 * feature values). Structures that can be serialised in more
|
c@45
|
82 * than one way will include either a "values" field (for Text
|
c@45
|
83 * serialisation) or a "b64values" field (for Base64) but should
|
c@45
|
84 * not include both. When parsing, if a "b64values" field is
|
c@45
|
85 * found, it will always take priority over a "values" field.
|
c@45
|
86 */
|
c@44
|
87 enum class BufferSerialisation {
|
c@45
|
88
|
c@45
|
89 /** Default JSON serialisation of values in array form. This
|
c@45
|
90 * is relatively slow to parse and serialise, and can take a
|
c@45
|
91 * lot of space.
|
c@45
|
92 */
|
c@45
|
93 Text,
|
c@45
|
94
|
c@45
|
95 /** Base64-encoded string of the raw data as packed IEEE
|
c@45
|
96 * 32-bit floats. Faster and more compact than the text
|
c@45
|
97 * encoding but more complicated to provide, especially if
|
c@45
|
98 * starting from an environment that does not use IEEE 32-bit
|
c@45
|
99 * floats! Note that Base64 serialisations produced by this
|
c@45
|
100 * library do not including padding characters and so are not
|
c@45
|
101 * necessarily multiples of 4 characters long. You will need
|
c@45
|
102 * to pad them yourself if concatenating them or supplying to
|
c@45
|
103 * a consumer that expects padding.
|
c@45
|
104 */
|
c@45
|
105 Base64
|
c@44
|
106 };
|
c@44
|
107
|
c@60
|
108 static bool failed(const std::string &err) {
|
c@60
|
109 return err != "";
|
c@60
|
110 }
|
c@5
|
111
|
c@5
|
112 template <typename T>
|
c@5
|
113 static json11::Json
|
c@5
|
114 fromBasicDescriptor(const T &t) {
|
c@5
|
115 return json11::Json::object {
|
c@5
|
116 { "identifier", t.identifier },
|
c@5
|
117 { "name", t.name },
|
c@5
|
118 { "description", t.description }
|
c@5
|
119 };
|
c@5
|
120 }
|
c@5
|
121
|
c@5
|
122 template <typename T>
|
c@5
|
123 static void
|
c@60
|
124 toBasicDescriptor(json11::Json j, T &t, std::string &err) {
|
c@5
|
125 if (!j.is_object()) {
|
c@60
|
126 err = "object expected for basic descriptor content";
|
c@60
|
127 return;
|
c@5
|
128 }
|
c@5
|
129 if (!j["identifier"].is_string()) {
|
c@60
|
130 err = "string expected for identifier";
|
c@60
|
131 return;
|
c@5
|
132 }
|
c@5
|
133 t.identifier = j["identifier"].string_value();
|
c@5
|
134 t.name = j["name"].string_value();
|
c@5
|
135 t.description = j["description"].string_value();
|
c@5
|
136 }
|
c@5
|
137
|
c@5
|
138 template <typename T>
|
c@5
|
139 static json11::Json
|
c@5
|
140 fromValueExtents(const T &t) {
|
c@5
|
141 return json11::Json::object {
|
c@5
|
142 { "min", t.minValue },
|
c@5
|
143 { "max", t.maxValue }
|
c@5
|
144 };
|
c@5
|
145 }
|
c@5
|
146
|
c@5
|
147 template <typename T>
|
c@5
|
148 static bool
|
c@60
|
149 toValueExtents(json11::Json j, T &t, std::string &err) {
|
c@5
|
150 if (j["extents"].is_null()) {
|
c@5
|
151 return false;
|
c@5
|
152 } else if (j["extents"].is_object()) {
|
c@5
|
153 if (j["extents"]["min"].is_number() &&
|
c@5
|
154 j["extents"]["max"].is_number()) {
|
c@5
|
155 t.minValue = j["extents"]["min"].number_value();
|
c@5
|
156 t.maxValue = j["extents"]["max"].number_value();
|
c@5
|
157 return true;
|
c@5
|
158 } else {
|
c@60
|
159 err = "numbers expected for min and max";
|
c@60
|
160 return false;
|
c@5
|
161 }
|
c@5
|
162 } else {
|
c@60
|
163 err = "object expected for extents (if present)";
|
c@60
|
164 return false;
|
c@5
|
165 }
|
c@5
|
166 }
|
c@5
|
167
|
c@5
|
168 static json11::Json
|
c@5
|
169 fromRealTime(const Vamp::RealTime &r) {
|
c@5
|
170 return json11::Json::object {
|
c@5
|
171 { "s", r.sec },
|
c@5
|
172 { "n", r.nsec }
|
c@5
|
173 };
|
c@5
|
174 }
|
c@5
|
175
|
c@5
|
176 static Vamp::RealTime
|
c@60
|
177 toRealTime(json11::Json j, std::string &err) {
|
c@5
|
178 json11::Json sec = j["s"];
|
c@5
|
179 json11::Json nsec = j["n"];
|
c@5
|
180 if (!sec.is_number() || !nsec.is_number()) {
|
c@60
|
181 err = "invalid Vamp::RealTime object " + j.dump();
|
c@60
|
182 return {};
|
c@5
|
183 }
|
c@5
|
184 return Vamp::RealTime(sec.int_value(), nsec.int_value());
|
c@5
|
185 }
|
c@5
|
186
|
c@5
|
187 static std::string
|
c@5
|
188 fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType type) {
|
c@5
|
189 switch (type) {
|
c@5
|
190 case Vamp::Plugin::OutputDescriptor::OneSamplePerStep:
|
c@5
|
191 return "OneSamplePerStep";
|
c@5
|
192 case Vamp::Plugin::OutputDescriptor::FixedSampleRate:
|
c@5
|
193 return "FixedSampleRate";
|
c@5
|
194 case Vamp::Plugin::OutputDescriptor::VariableSampleRate:
|
c@5
|
195 return "VariableSampleRate";
|
c@5
|
196 }
|
c@5
|
197 return "";
|
c@5
|
198 }
|
c@5
|
199
|
c@5
|
200 static Vamp::Plugin::OutputDescriptor::SampleType
|
c@60
|
201 toSampleType(std::string text, std::string &err) {
|
c@5
|
202 if (text == "OneSamplePerStep") {
|
c@5
|
203 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
|
c@5
|
204 } else if (text == "FixedSampleRate") {
|
c@5
|
205 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
|
c@5
|
206 } else if (text == "VariableSampleRate") {
|
c@5
|
207 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
|
c@5
|
208 } else {
|
c@60
|
209 err = "invalid sample type string: " + text;
|
c@60
|
210 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
|
c@5
|
211 }
|
c@5
|
212 }
|
c@5
|
213
|
c@5
|
214 static json11::Json
|
c@5
|
215 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) {
|
c@5
|
216 json11::Json::object jo {
|
c@5
|
217 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
218 { "unit", desc.unit },
|
c@5
|
219 { "sampleType", fromSampleType(desc.sampleType) },
|
c@5
|
220 { "sampleRate", desc.sampleRate },
|
c@5
|
221 { "hasDuration", desc.hasDuration }
|
c@5
|
222 };
|
c@5
|
223 if (desc.hasFixedBinCount) {
|
c@5
|
224 jo["binCount"] = int(desc.binCount);
|
c@5
|
225 jo["binNames"] = json11::Json::array
|
c@5
|
226 (desc.binNames.begin(), desc.binNames.end());
|
c@5
|
227 }
|
c@5
|
228 if (desc.hasKnownExtents) {
|
c@5
|
229 jo["extents"] = fromValueExtents(desc);
|
c@5
|
230 }
|
c@5
|
231 if (desc.isQuantized) {
|
c@5
|
232 jo["quantizeStep"] = desc.quantizeStep;
|
c@5
|
233 }
|
c@5
|
234 return json11::Json(jo);
|
c@5
|
235 }
|
c@12
|
236
|
c@5
|
237 static Vamp::Plugin::OutputDescriptor
|
c@60
|
238 toOutputDescriptor(json11::Json j, std::string &err) {
|
c@5
|
239
|
c@5
|
240 Vamp::Plugin::OutputDescriptor od;
|
c@5
|
241 if (!j.is_object()) {
|
c@60
|
242 err = "object expected for output descriptor";
|
c@60
|
243 return {};
|
c@5
|
244 }
|
c@5
|
245
|
c@60
|
246 toBasicDescriptor(j["basic"], od, err);
|
c@60
|
247 if (failed(err)) return {};
|
c@5
|
248
|
c@5
|
249 od.unit = j["unit"].string_value();
|
c@5
|
250
|
c@60
|
251 od.sampleType = toSampleType(j["sampleType"].string_value(), err);
|
c@60
|
252 if (failed(err)) return {};
|
c@5
|
253
|
c@5
|
254 if (!j["sampleRate"].is_number()) {
|
c@60
|
255 err = "number expected for sample rate";
|
c@60
|
256 return {};
|
c@5
|
257 }
|
c@5
|
258 od.sampleRate = j["sampleRate"].number_value();
|
c@5
|
259 od.hasDuration = j["hasDuration"].bool_value();
|
c@5
|
260
|
c@5
|
261 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) {
|
c@5
|
262 od.hasFixedBinCount = true;
|
c@5
|
263 od.binCount = j["binCount"].int_value();
|
c@5
|
264 for (auto &n: j["binNames"].array_items()) {
|
c@5
|
265 if (!n.is_string()) {
|
c@60
|
266 err = "string expected for bin name";
|
c@60
|
267 return {};
|
c@5
|
268 }
|
c@5
|
269 od.binNames.push_back(n.string_value());
|
c@5
|
270 }
|
c@5
|
271 } else {
|
c@5
|
272 od.hasFixedBinCount = false;
|
c@5
|
273 }
|
c@5
|
274
|
c@60
|
275 bool extentsPresent = toValueExtents(j, od, err);
|
c@60
|
276 if (failed(err)) return {};
|
c@60
|
277
|
c@5
|
278 od.hasKnownExtents = extentsPresent;
|
c@5
|
279
|
c@5
|
280 if (j["quantizeStep"].is_number()) {
|
c@5
|
281 od.isQuantized = true;
|
c@5
|
282 od.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
283 } else {
|
c@5
|
284 od.isQuantized = false;
|
c@5
|
285 }
|
c@5
|
286
|
c@5
|
287 return od;
|
c@5
|
288 }
|
c@5
|
289
|
c@5
|
290 static json11::Json
|
c@5
|
291 fromParameterDescriptor(const Vamp::PluginBase::ParameterDescriptor &desc) {
|
c@5
|
292
|
c@5
|
293 json11::Json::object jo {
|
c@5
|
294 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
295 { "unit", desc.unit },
|
c@5
|
296 { "extents", fromValueExtents(desc) },
|
c@5
|
297 { "defaultValue", desc.defaultValue },
|
c@5
|
298 { "valueNames", json11::Json::array
|
c@5
|
299 (desc.valueNames.begin(), desc.valueNames.end()) }
|
c@5
|
300 };
|
c@5
|
301 if (desc.isQuantized) {
|
c@5
|
302 jo["quantizeStep"] = desc.quantizeStep;
|
c@5
|
303 }
|
c@5
|
304 return json11::Json(jo);
|
c@5
|
305 }
|
c@5
|
306
|
c@5
|
307 static Vamp::PluginBase::ParameterDescriptor
|
c@60
|
308 toParameterDescriptor(json11::Json j, std::string &err) {
|
c@5
|
309
|
c@5
|
310 Vamp::PluginBase::ParameterDescriptor pd;
|
c@5
|
311 if (!j.is_object()) {
|
c@60
|
312 err = "object expected for parameter descriptor";
|
c@60
|
313 return {};
|
c@5
|
314 }
|
c@5
|
315
|
c@60
|
316 toBasicDescriptor(j["basic"], pd, err);
|
c@60
|
317 if (failed(err)) return {};
|
c@5
|
318
|
c@5
|
319 pd.unit = j["unit"].string_value();
|
c@5
|
320
|
c@60
|
321 bool extentsPresent = toValueExtents(j, pd, err);
|
c@60
|
322 if (failed(err)) return {};
|
c@5
|
323 if (!extentsPresent) {
|
c@60
|
324 err = "extents must be present in parameter descriptor";
|
c@60
|
325 return {};
|
c@5
|
326 }
|
c@5
|
327
|
c@5
|
328 if (!j["defaultValue"].is_number()) {
|
c@60
|
329 err = "number expected for default value";
|
c@60
|
330 return {};
|
c@5
|
331 }
|
c@5
|
332
|
c@5
|
333 pd.defaultValue = j["defaultValue"].number_value();
|
c@5
|
334
|
c@5
|
335 pd.valueNames.clear();
|
c@5
|
336 for (auto &n: j["valueNames"].array_items()) {
|
c@5
|
337 if (!n.is_string()) {
|
c@60
|
338 err = "string expected for value name";
|
c@60
|
339 return {};
|
c@5
|
340 }
|
c@5
|
341 pd.valueNames.push_back(n.string_value());
|
c@5
|
342 }
|
c@5
|
343
|
c@5
|
344 if (j["quantizeStep"].is_number()) {
|
c@5
|
345 pd.isQuantized = true;
|
c@5
|
346 pd.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
347 } else {
|
c@5
|
348 pd.isQuantized = false;
|
c@5
|
349 }
|
c@5
|
350
|
c@5
|
351 return pd;
|
c@5
|
352 }
|
c@5
|
353
|
c@5
|
354 static std::string
|
c@5
|
355 fromFloatBuffer(const float *buffer, size_t nfloats) {
|
c@5
|
356 // must use char pointers, otherwise the converter will only
|
c@5
|
357 // encode every 4th byte (as it will count up in float* steps)
|
c@5
|
358 const char *start = reinterpret_cast<const char *>(buffer);
|
c@5
|
359 const char *end = reinterpret_cast<const char *>(buffer + nfloats);
|
c@5
|
360 std::string encoded;
|
c@5
|
361 bn::encode_b64(start, end, back_inserter(encoded));
|
c@5
|
362 return encoded;
|
c@5
|
363 }
|
c@5
|
364
|
c@5
|
365 static std::vector<float>
|
c@60
|
366 toFloatBuffer(std::string encoded, std::string & /* err */) {
|
c@5
|
367 std::string decoded;
|
c@5
|
368 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded));
|
c@5
|
369 const float *buffer = reinterpret_cast<const float *>(decoded.c_str());
|
c@5
|
370 size_t n = decoded.size() / sizeof(float);
|
c@47
|
371 return std::vector<float>(buffer, buffer + n);
|
c@5
|
372 }
|
c@5
|
373
|
c@5
|
374 static json11::Json
|
c@44
|
375 fromFeature(const Vamp::Plugin::Feature &f,
|
c@44
|
376 BufferSerialisation serialisation) {
|
c@5
|
377
|
c@5
|
378 json11::Json::object jo;
|
c@5
|
379 if (f.values.size() > 0) {
|
c@44
|
380 if (serialisation == BufferSerialisation::Text) {
|
c@35
|
381 jo["values"] = json11::Json::array(f.values.begin(),
|
c@35
|
382 f.values.end());
|
c@35
|
383 } else {
|
c@35
|
384 jo["b64values"] = fromFloatBuffer(f.values.data(),
|
c@35
|
385 f.values.size());
|
c@35
|
386 }
|
c@5
|
387 }
|
c@5
|
388 if (f.label != "") {
|
c@5
|
389 jo["label"] = f.label;
|
c@5
|
390 }
|
c@5
|
391 if (f.hasTimestamp) {
|
c@5
|
392 jo["timestamp"] = fromRealTime(f.timestamp);
|
c@5
|
393 }
|
c@5
|
394 if (f.hasDuration) {
|
c@5
|
395 jo["duration"] = fromRealTime(f.duration);
|
c@5
|
396 }
|
c@5
|
397 return json11::Json(jo);
|
c@5
|
398 }
|
c@5
|
399
|
c@5
|
400 static Vamp::Plugin::Feature
|
c@60
|
401 toFeature(json11::Json j, BufferSerialisation &serialisation, std::string &err) {
|
c@5
|
402
|
c@5
|
403 Vamp::Plugin::Feature f;
|
c@5
|
404 if (!j.is_object()) {
|
c@60
|
405 err = "object expected for feature";
|
c@60
|
406 return {};
|
c@5
|
407 }
|
c@5
|
408 if (j["timestamp"].is_object()) {
|
c@60
|
409 f.timestamp = toRealTime(j["timestamp"], err);
|
c@60
|
410 if (failed(err)) return {};
|
c@5
|
411 f.hasTimestamp = true;
|
c@5
|
412 }
|
c@5
|
413 if (j["duration"].is_object()) {
|
c@60
|
414 f.duration = toRealTime(j["duration"], err);
|
c@60
|
415 if (failed(err)) return {};
|
c@5
|
416 f.hasDuration = true;
|
c@5
|
417 }
|
c@5
|
418 if (j["b64values"].is_string()) {
|
c@60
|
419 f.values = toFloatBuffer(j["b64values"].string_value(), err);
|
c@60
|
420 if (failed(err)) return {};
|
c@44
|
421 serialisation = BufferSerialisation::Base64;
|
c@5
|
422 } else if (j["values"].is_array()) {
|
c@5
|
423 for (auto v : j["values"].array_items()) {
|
c@5
|
424 f.values.push_back(v.number_value());
|
c@5
|
425 }
|
c@44
|
426 serialisation = BufferSerialisation::Text;
|
c@5
|
427 }
|
c@5
|
428 f.label = j["label"].string_value();
|
c@5
|
429 return f;
|
c@5
|
430 }
|
c@5
|
431
|
c@5
|
432 static json11::Json
|
c@44
|
433 fromFeatureSet(const Vamp::Plugin::FeatureSet &fs,
|
c@49
|
434 const PluginOutputIdMapper &omapper,
|
c@44
|
435 BufferSerialisation serialisation) {
|
c@5
|
436
|
c@5
|
437 json11::Json::object jo;
|
c@5
|
438 for (const auto &fsi : fs) {
|
c@5
|
439 std::vector<json11::Json> fj;
|
c@5
|
440 for (const Vamp::Plugin::Feature &f: fsi.second) {
|
c@44
|
441 fj.push_back(fromFeature(f, serialisation));
|
c@5
|
442 }
|
c@49
|
443 jo[omapper.indexToId(fsi.first)] = fj;
|
c@5
|
444 }
|
c@5
|
445 return json11::Json(jo);
|
c@5
|
446 }
|
c@5
|
447
|
c@5
|
448 static Vamp::Plugin::FeatureList
|
c@44
|
449 toFeatureList(json11::Json j,
|
c@60
|
450 BufferSerialisation &serialisation, std::string &err) {
|
c@5
|
451
|
c@5
|
452 Vamp::Plugin::FeatureList fl;
|
c@5
|
453 if (!j.is_array()) {
|
c@60
|
454 err = "array expected for feature list";
|
c@60
|
455 return {};
|
c@5
|
456 }
|
c@5
|
457 for (const json11::Json &fj : j.array_items()) {
|
c@60
|
458 fl.push_back(toFeature(fj, serialisation, err));
|
c@60
|
459 if (failed(err)) return {};
|
c@5
|
460 }
|
c@5
|
461 return fl;
|
c@5
|
462 }
|
c@5
|
463
|
c@5
|
464 static Vamp::Plugin::FeatureSet
|
c@49
|
465 toFeatureSet(json11::Json j,
|
c@49
|
466 const PluginOutputIdMapper &omapper,
|
c@60
|
467 BufferSerialisation &serialisation,
|
c@60
|
468 std::string &err) {
|
c@5
|
469
|
c@5
|
470 Vamp::Plugin::FeatureSet fs;
|
c@5
|
471 if (!j.is_object()) {
|
c@60
|
472 err = "object expected for feature set";
|
c@60
|
473 return {};
|
c@5
|
474 }
|
c@5
|
475 for (auto &entry : j.object_items()) {
|
c@49
|
476 int n = omapper.idToIndex(entry.first);
|
c@49
|
477 if (fs.find(n) != fs.end()) {
|
c@60
|
478 err = "duplicate numerical index for output";
|
c@60
|
479 return {};
|
c@5
|
480 }
|
c@60
|
481 fs[n] = toFeatureList(entry.second, serialisation, err);
|
c@60
|
482 if (failed(err)) return {};
|
c@5
|
483 }
|
c@5
|
484 return fs;
|
c@5
|
485 }
|
c@5
|
486
|
c@5
|
487 static std::string
|
c@5
|
488 fromInputDomain(Vamp::Plugin::InputDomain domain) {
|
c@5
|
489
|
c@5
|
490 switch (domain) {
|
c@5
|
491 case Vamp::Plugin::TimeDomain:
|
c@5
|
492 return "TimeDomain";
|
c@5
|
493 case Vamp::Plugin::FrequencyDomain:
|
c@5
|
494 return "FrequencyDomain";
|
c@5
|
495 }
|
c@5
|
496 return "";
|
c@5
|
497 }
|
c@5
|
498
|
c@5
|
499 static Vamp::Plugin::InputDomain
|
c@60
|
500 toInputDomain(std::string text, std::string &err) {
|
c@5
|
501
|
c@5
|
502 if (text == "TimeDomain") {
|
c@5
|
503 return Vamp::Plugin::TimeDomain;
|
c@5
|
504 } else if (text == "FrequencyDomain") {
|
c@5
|
505 return Vamp::Plugin::FrequencyDomain;
|
c@5
|
506 } else {
|
c@60
|
507 err = "invalid input domain string: " + text;
|
c@60
|
508 return {};
|
c@5
|
509 }
|
c@5
|
510 }
|
c@5
|
511
|
c@5
|
512 static json11::Json
|
c@5
|
513 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) {
|
c@5
|
514
|
c@5
|
515 json11::Json::object jo;
|
c@5
|
516 jo["pluginKey"] = d.pluginKey;
|
c@5
|
517 jo["basic"] = fromBasicDescriptor(d.basic);
|
c@5
|
518 jo["maker"] = d.maker;
|
c@5
|
519 jo["copyright"] = d.copyright;
|
c@5
|
520 jo["pluginVersion"] = d.pluginVersion;
|
c@5
|
521
|
c@5
|
522 json11::Json::array cat;
|
c@5
|
523 for (const std::string &c: d.category) cat.push_back(c);
|
c@5
|
524 jo["category"] = cat;
|
c@5
|
525
|
c@5
|
526 jo["minChannelCount"] = d.minChannelCount;
|
c@5
|
527 jo["maxChannelCount"] = d.maxChannelCount;
|
c@5
|
528
|
c@5
|
529 json11::Json::array params;
|
c@5
|
530 Vamp::PluginBase::ParameterList vparams = d.parameters;
|
c@5
|
531 for (auto &p: vparams) params.push_back(fromParameterDescriptor(p));
|
c@5
|
532 jo["parameters"] = params;
|
c@5
|
533
|
c@5
|
534 json11::Json::array progs;
|
c@5
|
535 Vamp::PluginBase::ProgramList vprogs = d.programs;
|
c@5
|
536 for (auto &p: vprogs) progs.push_back(p);
|
c@5
|
537 jo["programs"] = progs;
|
c@5
|
538
|
c@5
|
539 jo["inputDomain"] = fromInputDomain(d.inputDomain);
|
c@5
|
540
|
c@5
|
541 json11::Json::array outinfo;
|
c@5
|
542 auto vouts = d.basicOutputInfo;
|
c@5
|
543 for (auto &o: vouts) outinfo.push_back(fromBasicDescriptor(o));
|
c@5
|
544 jo["basicOutputInfo"] = outinfo;
|
c@5
|
545
|
c@5
|
546 return json11::Json(jo);
|
c@5
|
547 }
|
c@5
|
548
|
c@5
|
549 static Vamp::HostExt::PluginStaticData
|
c@60
|
550 toPluginStaticData(json11::Json j, std::string &err) {
|
c@5
|
551
|
c@5
|
552 if (!j.has_shape({
|
c@5
|
553 { "pluginKey", json11::Json::STRING },
|
c@5
|
554 { "pluginVersion", json11::Json::NUMBER },
|
c@5
|
555 { "minChannelCount", json11::Json::NUMBER },
|
c@5
|
556 { "maxChannelCount", json11::Json::NUMBER },
|
c@5
|
557 { "inputDomain", json11::Json::STRING }}, err)) {
|
c@60
|
558
|
c@60
|
559 err = "malformed plugin static data: " + err;
|
c@60
|
560
|
c@60
|
561 } else if (!j["basicOutputInfo"].is_array()) {
|
c@60
|
562
|
c@60
|
563 err = "array expected for basic output info";
|
c@60
|
564
|
c@60
|
565 } else if (!j["maker"].is_null() &&
|
c@60
|
566 !j["maker"].is_string()) {
|
c@60
|
567
|
c@60
|
568 err = "string expected for maker";
|
c@60
|
569
|
c@60
|
570 } else if (!j["copyright"].is_null() &&
|
c@60
|
571 !j["copyright"].is_string()) {
|
c@60
|
572 err = "string expected for copyright";
|
c@60
|
573
|
c@60
|
574 } else if (!j["category"].is_null() &&
|
c@60
|
575 !j["category"].is_array()) {
|
c@60
|
576
|
c@60
|
577 err = "array expected for category";
|
c@60
|
578
|
c@60
|
579 } else if (!j["parameters"].is_null() &&
|
c@60
|
580 !j["parameters"].is_array()) {
|
c@60
|
581
|
c@60
|
582 err = "array expected for parameters";
|
c@60
|
583
|
c@60
|
584 } else if (!j["programs"].is_null() &&
|
c@60
|
585 !j["programs"].is_array()) {
|
c@60
|
586
|
c@60
|
587 err = "array expected for programs";
|
c@60
|
588
|
c@60
|
589 } else if (!j["inputDomain"].is_null() &&
|
c@60
|
590 !j["inputDomain"].is_string()) {
|
c@60
|
591
|
c@60
|
592 err = "string expected for inputDomain";
|
c@60
|
593
|
c@60
|
594 } else if (!j["basicOutputInfo"].is_null() &&
|
c@60
|
595 !j["basicOutputInfo"].is_array()) {
|
c@60
|
596
|
c@60
|
597 err = "array expected for basicOutputInfo";
|
c@60
|
598
|
c@60
|
599 } else {
|
c@60
|
600
|
c@60
|
601 Vamp::HostExt::PluginStaticData psd;
|
c@60
|
602
|
c@60
|
603 psd.pluginKey = j["pluginKey"].string_value();
|
c@60
|
604
|
c@60
|
605 toBasicDescriptor(j["basic"], psd.basic, err);
|
c@60
|
606 if (failed(err)) return {};
|
c@60
|
607
|
c@60
|
608 psd.maker = j["maker"].string_value();
|
c@60
|
609 psd.copyright = j["copyright"].string_value();
|
c@60
|
610 psd.pluginVersion = j["pluginVersion"].int_value();
|
c@60
|
611
|
c@60
|
612 for (const auto &c : j["category"].array_items()) {
|
c@60
|
613 if (!c.is_string()) {
|
c@60
|
614 err = "strings expected in category array";
|
c@60
|
615 return {};
|
c@60
|
616 }
|
c@60
|
617 psd.category.push_back(c.string_value());
|
c@60
|
618 }
|
c@60
|
619
|
c@60
|
620 psd.minChannelCount = j["minChannelCount"].int_value();
|
c@60
|
621 psd.maxChannelCount = j["maxChannelCount"].int_value();
|
c@60
|
622
|
c@60
|
623 for (const auto &p : j["parameters"].array_items()) {
|
c@60
|
624 auto pd = toParameterDescriptor(p, err);
|
c@60
|
625 if (failed(err)) return {};
|
c@60
|
626 psd.parameters.push_back(pd);
|
c@60
|
627 }
|
c@60
|
628
|
c@60
|
629 for (const auto &p : j["programs"].array_items()) {
|
c@60
|
630 if (!p.is_string()) {
|
c@60
|
631 err = "strings expected in programs array";
|
c@60
|
632 return {};
|
c@60
|
633 }
|
c@60
|
634 psd.programs.push_back(p.string_value());
|
c@60
|
635 }
|
c@60
|
636
|
c@60
|
637 psd.inputDomain = toInputDomain(j["inputDomain"].string_value(), err);
|
c@60
|
638 if (failed(err)) return {};
|
c@60
|
639
|
c@60
|
640 for (const auto &bo : j["basicOutputInfo"].array_items()) {
|
c@60
|
641 Vamp::HostExt::PluginStaticData::Basic b;
|
c@60
|
642 toBasicDescriptor(bo, b, err);
|
c@60
|
643 if (failed(err)) return {};
|
c@60
|
644 psd.basicOutputInfo.push_back(b);
|
c@60
|
645 }
|
c@60
|
646
|
c@60
|
647 return psd;
|
c@5
|
648 }
|
c@5
|
649
|
c@60
|
650 // fallthrough error case
|
c@60
|
651 return {};
|
c@5
|
652 }
|
c@5
|
653
|
c@5
|
654 static json11::Json
|
c@5
|
655 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) {
|
c@5
|
656
|
c@5
|
657 json11::Json::object jo;
|
c@5
|
658
|
c@5
|
659 json11::Json::object paramValues;
|
c@5
|
660 for (auto &vp: c.parameterValues) {
|
c@5
|
661 paramValues[vp.first] = vp.second;
|
c@5
|
662 }
|
c@5
|
663 jo["parameterValues"] = paramValues;
|
c@5
|
664
|
c@5
|
665 if (c.currentProgram != "") {
|
c@5
|
666 jo["currentProgram"] = c.currentProgram;
|
c@5
|
667 }
|
c@5
|
668
|
c@5
|
669 jo["channelCount"] = c.channelCount;
|
c@5
|
670 jo["stepSize"] = c.stepSize;
|
c@5
|
671 jo["blockSize"] = c.blockSize;
|
c@5
|
672
|
c@5
|
673 return json11::Json(jo);
|
c@5
|
674 }
|
c@5
|
675
|
c@5
|
676 static Vamp::HostExt::PluginConfiguration
|
c@60
|
677 toPluginConfiguration(json11::Json j, std::string &err) {
|
c@5
|
678
|
c@5
|
679 if (!j.has_shape({
|
c@5
|
680 { "channelCount", json11::Json::NUMBER },
|
c@5
|
681 { "stepSize", json11::Json::NUMBER },
|
c@5
|
682 { "blockSize", json11::Json::NUMBER } }, err)) {
|
c@60
|
683 err = "malformed plugin configuration: " + err;
|
c@60
|
684 return {};
|
c@5
|
685 }
|
c@5
|
686
|
c@5
|
687 if (!j["parameterValues"].is_null() &&
|
c@5
|
688 !j["parameterValues"].is_object()) {
|
c@60
|
689 err = "object expected for parameter values";
|
c@60
|
690 return {};
|
c@5
|
691 }
|
c@5
|
692
|
c@5
|
693 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
694 if (!pv.second.is_number()) {
|
c@60
|
695 err = "number expected for parameter value";
|
c@60
|
696 return {};
|
c@5
|
697 }
|
c@5
|
698 }
|
c@5
|
699
|
c@5
|
700 if (!j["currentProgram"].is_null() &&
|
c@5
|
701 !j["currentProgram"].is_string()) {
|
c@60
|
702 err = "string expected for program name";
|
c@60
|
703 return {};
|
c@5
|
704 }
|
c@5
|
705
|
c@5
|
706 Vamp::HostExt::PluginConfiguration config;
|
c@5
|
707
|
c@5
|
708 config.channelCount = j["channelCount"].number_value();
|
c@5
|
709 config.stepSize = j["stepSize"].number_value();
|
c@5
|
710 config.blockSize = j["blockSize"].number_value();
|
c@5
|
711
|
c@5
|
712 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
713 config.parameterValues[pv.first] = pv.second.number_value();
|
c@5
|
714 }
|
c@5
|
715
|
c@5
|
716 if (j["currentProgram"].is_string()) {
|
c@5
|
717 config.currentProgram = j["currentProgram"].string_value();
|
c@5
|
718 }
|
c@5
|
719
|
c@5
|
720 return config;
|
c@5
|
721 }
|
c@5
|
722
|
c@5
|
723 static json11::Json
|
c@5
|
724 fromAdapterFlags(int flags) {
|
c@5
|
725
|
c@5
|
726 json11::Json::array arr;
|
c@5
|
727
|
c@5
|
728 if (flags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
|
c@5
|
729 arr.push_back("AdaptInputDomain");
|
c@5
|
730 }
|
c@5
|
731 if (flags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
|
c@5
|
732 arr.push_back("AdaptChannelCount");
|
c@5
|
733 }
|
c@5
|
734 if (flags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
|
c@5
|
735 arr.push_back("AdaptBufferSize");
|
c@5
|
736 }
|
c@5
|
737
|
c@5
|
738 return json11::Json(arr);
|
c@5
|
739 }
|
c@5
|
740
|
c@5
|
741 static Vamp::HostExt::PluginLoader::AdapterFlags
|
c@60
|
742 toAdapterFlags(json11::Json j, std::string &err) {
|
c@60
|
743
|
c@60
|
744 int flags = 0x0;
|
c@5
|
745
|
c@5
|
746 if (!j.is_array()) {
|
c@5
|
747
|
c@60
|
748 err = "array expected for adapter flags";
|
c@60
|
749
|
c@60
|
750 } else {
|
c@60
|
751
|
c@60
|
752 for (auto &jj: j.array_items()) {
|
c@60
|
753 if (!jj.is_string()) {
|
c@60
|
754 err = "string expected for adapter flag";
|
c@60
|
755 break;
|
c@60
|
756 }
|
c@60
|
757 std::string text = jj.string_value();
|
c@60
|
758 if (text == "AdaptInputDomain") {
|
c@60
|
759 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
|
c@60
|
760 } else if (text == "AdaptChannelCount") {
|
c@60
|
761 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
|
c@60
|
762 } else if (text == "AdaptBufferSize") {
|
c@60
|
763 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
|
c@60
|
764 } else if (text == "AdaptAllSafe") {
|
c@60
|
765 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE;
|
c@60
|
766 } else if (text == "AdaptAll") {
|
c@60
|
767 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL;
|
c@60
|
768 } else {
|
c@60
|
769 err = "invalid adapter flag string: " + text;
|
c@60
|
770 break;
|
c@60
|
771 }
|
c@5
|
772 }
|
c@5
|
773 }
|
c@5
|
774
|
c@5
|
775 return Vamp::HostExt::PluginLoader::AdapterFlags(flags);
|
c@5
|
776 }
|
c@5
|
777
|
c@5
|
778 static json11::Json
|
c@17
|
779 fromLoadRequest(const Vamp::HostExt::LoadRequest &req) {
|
c@5
|
780
|
c@5
|
781 json11::Json::object jo;
|
c@5
|
782 jo["pluginKey"] = req.pluginKey;
|
c@5
|
783 jo["inputSampleRate"] = req.inputSampleRate;
|
c@5
|
784 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags);
|
c@5
|
785 return json11::Json(jo);
|
c@5
|
786 }
|
c@5
|
787
|
c@5
|
788 static Vamp::HostExt::LoadRequest
|
c@60
|
789 toLoadRequest(json11::Json j, std::string &err) {
|
c@5
|
790
|
c@5
|
791 if (!j.has_shape({
|
c@5
|
792 { "pluginKey", json11::Json::STRING },
|
c@32
|
793 { "inputSampleRate", json11::Json::NUMBER } }, err)) {
|
c@60
|
794 err = "malformed load request: " + err;
|
c@60
|
795 return {};
|
c@5
|
796 }
|
c@5
|
797
|
c@5
|
798 Vamp::HostExt::LoadRequest req;
|
c@5
|
799 req.pluginKey = j["pluginKey"].string_value();
|
c@5
|
800 req.inputSampleRate = j["inputSampleRate"].number_value();
|
c@32
|
801 if (!j["adapterFlags"].is_null()) {
|
c@60
|
802 req.adapterFlags = toAdapterFlags(j["adapterFlags"], err);
|
c@60
|
803 if (failed(err)) return {};
|
c@32
|
804 }
|
c@5
|
805 return req;
|
c@5
|
806 }
|
c@10
|
807
|
c@10
|
808 static json11::Json
|
c@17
|
809 fromLoadResponse(const Vamp::HostExt::LoadResponse &resp,
|
c@49
|
810 const PluginHandleMapper &pmapper) {
|
c@10
|
811
|
c@10
|
812 json11::Json::object jo;
|
c@49
|
813 jo["pluginHandle"] = double(pmapper.pluginToHandle(resp.plugin));
|
c@10
|
814 jo["staticData"] = fromPluginStaticData(resp.staticData);
|
c@10
|
815 jo["defaultConfiguration"] =
|
c@10
|
816 fromPluginConfiguration(resp.defaultConfiguration);
|
c@10
|
817 return json11::Json(jo);
|
c@10
|
818 }
|
c@10
|
819
|
c@10
|
820 static Vamp::HostExt::LoadResponse
|
c@10
|
821 toLoadResponse(json11::Json j,
|
c@60
|
822 const PluginHandleMapper &pmapper, std::string &err) {
|
c@10
|
823
|
c@10
|
824 if (!j.has_shape({
|
c@10
|
825 { "pluginHandle", json11::Json::NUMBER },
|
c@10
|
826 { "staticData", json11::Json::OBJECT },
|
c@10
|
827 { "defaultConfiguration", json11::Json::OBJECT } }, err)) {
|
c@60
|
828 err = "malformed load response: " + err;
|
c@60
|
829 return {};
|
c@10
|
830 }
|
c@10
|
831
|
c@10
|
832 Vamp::HostExt::LoadResponse resp;
|
c@49
|
833 resp.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@60
|
834 resp.staticData = toPluginStaticData(j["staticData"], err);
|
c@60
|
835 if (failed(err)) return {};
|
c@60
|
836 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"],
|
c@60
|
837 err);
|
c@60
|
838 if (failed(err)) return {};
|
c@10
|
839 return resp;
|
c@10
|
840 }
|
c@12
|
841
|
c@12
|
842 static json11::Json
|
c@13
|
843 fromConfigurationRequest(const Vamp::HostExt::ConfigurationRequest &cr,
|
c@49
|
844 const PluginHandleMapper &pmapper) {
|
c@13
|
845
|
c@13
|
846 json11::Json::object jo;
|
c@13
|
847
|
c@49
|
848 jo["pluginHandle"] = pmapper.pluginToHandle(cr.plugin);
|
c@13
|
849 jo["configuration"] = fromPluginConfiguration(cr.configuration);
|
c@13
|
850
|
c@13
|
851 return json11::Json(jo);
|
c@13
|
852 }
|
c@13
|
853
|
c@13
|
854 static Vamp::HostExt::ConfigurationRequest
|
c@13
|
855 toConfigurationRequest(json11::Json j,
|
c@60
|
856 const PluginHandleMapper &pmapper, std::string &err) {
|
c@13
|
857
|
c@13
|
858 if (!j.has_shape({
|
c@13
|
859 { "pluginHandle", json11::Json::NUMBER },
|
c@13
|
860 { "configuration", json11::Json::OBJECT } }, err)) {
|
c@60
|
861 err = "malformed configuration request: " + err;
|
c@60
|
862 return {};
|
c@13
|
863 }
|
c@13
|
864
|
c@13
|
865 Vamp::HostExt::ConfigurationRequest cr;
|
c@49
|
866 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@60
|
867 cr.configuration = toPluginConfiguration(j["configuration"], err);
|
c@60
|
868 if (failed(err)) return {};
|
c@13
|
869 return cr;
|
c@13
|
870 }
|
c@13
|
871
|
c@13
|
872 static json11::Json
|
c@55
|
873 fromConfigurationResponse(const Vamp::HostExt::ConfigurationResponse &cr,
|
c@55
|
874 const PluginHandleMapper &pmapper) {
|
c@12
|
875
|
c@13
|
876 json11::Json::object jo;
|
c@55
|
877
|
c@55
|
878 jo["pluginHandle"] = pmapper.pluginToHandle(cr.plugin);
|
c@12
|
879
|
c@12
|
880 json11::Json::array outs;
|
c@12
|
881 for (auto &d: cr.outputs) {
|
c@12
|
882 outs.push_back(fromOutputDescriptor(d));
|
c@12
|
883 }
|
c@13
|
884 jo["outputList"] = outs;
|
c@12
|
885
|
c@13
|
886 return json11::Json(jo);
|
c@12
|
887 }
|
c@12
|
888
|
c@13
|
889 static Vamp::HostExt::ConfigurationResponse
|
c@55
|
890 toConfigurationResponse(json11::Json j,
|
c@60
|
891 const PluginHandleMapper &pmapper, std::string &err) {
|
c@13
|
892
|
c@12
|
893 Vamp::HostExt::ConfigurationResponse cr;
|
c@12
|
894
|
c@55
|
895 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@55
|
896
|
c@12
|
897 if (!j["outputList"].is_array()) {
|
c@60
|
898 err = "array expected for output list";
|
c@60
|
899 return {};
|
c@12
|
900 }
|
c@12
|
901
|
c@12
|
902 for (const auto &o: j["outputList"].array_items()) {
|
c@60
|
903 cr.outputs.push_back(toOutputDescriptor(o, err));
|
c@60
|
904 if (failed(err)) return {};
|
c@12
|
905 }
|
c@12
|
906
|
c@12
|
907 return cr;
|
c@12
|
908 }
|
c@16
|
909
|
c@16
|
910 static json11::Json
|
c@16
|
911 fromProcessRequest(const Vamp::HostExt::ProcessRequest &r,
|
c@49
|
912 const PluginHandleMapper &pmapper,
|
c@44
|
913 BufferSerialisation serialisation) {
|
c@16
|
914
|
c@16
|
915 json11::Json::object jo;
|
c@49
|
916 jo["pluginHandle"] = pmapper.pluginToHandle(r.plugin);
|
c@16
|
917
|
c@16
|
918 json11::Json::object io;
|
c@16
|
919 io["timestamp"] = fromRealTime(r.timestamp);
|
c@16
|
920
|
c@16
|
921 json11::Json::array chans;
|
c@16
|
922 for (size_t i = 0; i < r.inputBuffers.size(); ++i) {
|
c@16
|
923 json11::Json::object c;
|
c@44
|
924 if (serialisation == BufferSerialisation::Text) {
|
c@44
|
925 c["values"] = json11::Json::array(r.inputBuffers[i].begin(),
|
c@44
|
926 r.inputBuffers[i].end());
|
c@44
|
927 } else {
|
c@44
|
928 c["b64values"] = fromFloatBuffer(r.inputBuffers[i].data(),
|
c@44
|
929 r.inputBuffers[i].size());
|
c@44
|
930 }
|
c@16
|
931 chans.push_back(c);
|
c@16
|
932 }
|
c@16
|
933 io["inputBuffers"] = chans;
|
c@16
|
934
|
c@16
|
935 jo["processInput"] = io;
|
c@16
|
936 return json11::Json(jo);
|
c@16
|
937 }
|
c@17
|
938
|
c@17
|
939 static Vamp::HostExt::ProcessRequest
|
c@44
|
940 toProcessRequest(json11::Json j,
|
c@49
|
941 const PluginHandleMapper &pmapper,
|
c@60
|
942 BufferSerialisation &serialisation, std::string &err) {
|
c@17
|
943
|
c@17
|
944 if (!j.has_shape({
|
c@17
|
945 { "pluginHandle", json11::Json::NUMBER },
|
c@17
|
946 { "processInput", json11::Json::OBJECT } }, err)) {
|
c@60
|
947 err = "malformed process request: " + err;
|
c@60
|
948 return {};
|
c@17
|
949 }
|
c@17
|
950
|
c@17
|
951 auto input = j["processInput"];
|
c@17
|
952
|
c@17
|
953 if (!input.has_shape({
|
c@17
|
954 { "timestamp", json11::Json::OBJECT },
|
c@17
|
955 { "inputBuffers", json11::Json::ARRAY } }, err)) {
|
c@60
|
956 err = "malformed process request: " + err;
|
c@60
|
957 return {};
|
c@17
|
958 }
|
c@17
|
959
|
c@17
|
960 Vamp::HostExt::ProcessRequest r;
|
c@49
|
961 r.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@17
|
962
|
c@60
|
963 r.timestamp = toRealTime(input["timestamp"], err);
|
c@60
|
964 if (failed(err)) return {};
|
c@17
|
965
|
c@17
|
966 for (auto a: input["inputBuffers"].array_items()) {
|
c@44
|
967
|
c@17
|
968 if (a["b64values"].is_string()) {
|
c@60
|
969 std::vector<float> buf = toFloatBuffer(a["b64values"].string_value(),
|
c@60
|
970 err);
|
c@60
|
971 if (failed(err)) return {};
|
c@44
|
972 r.inputBuffers.push_back(buf);
|
c@44
|
973 serialisation = BufferSerialisation::Base64;
|
c@44
|
974
|
c@17
|
975 } else if (a["values"].is_array()) {
|
c@17
|
976 std::vector<float> buf;
|
c@17
|
977 for (auto v : a["values"].array_items()) {
|
c@17
|
978 buf.push_back(v.number_value());
|
c@17
|
979 }
|
c@17
|
980 r.inputBuffers.push_back(buf);
|
c@44
|
981 serialisation = BufferSerialisation::Text;
|
c@44
|
982
|
c@17
|
983 } else {
|
c@60
|
984 err = "expected values or b64values in inputBuffers object";
|
c@60
|
985 return {};
|
c@17
|
986 }
|
c@17
|
987 }
|
c@17
|
988
|
c@17
|
989 return r;
|
c@17
|
990 }
|
c@17
|
991
|
c@17
|
992 static json11::Json
|
c@17
|
993 fromVampRequest_List() {
|
c@17
|
994
|
c@17
|
995 json11::Json::object jo;
|
c@17
|
996 jo["type"] = "list";
|
c@17
|
997 return json11::Json(jo);
|
c@17
|
998 }
|
c@17
|
999
|
c@17
|
1000 static json11::Json
|
c@59
|
1001 fromVampResponse_List(const Vamp::HostExt::ListResponse &resp) {
|
c@17
|
1002
|
c@17
|
1003 json11::Json::object jo;
|
c@24
|
1004 jo["type"] = "list";
|
c@59
|
1005 jo["success"] = true;
|
c@17
|
1006
|
c@17
|
1007 json11::Json::array arr;
|
c@56
|
1008 for (const auto &a: resp.pluginData) {
|
c@17
|
1009 arr.push_back(fromPluginStaticData(a));
|
c@17
|
1010 }
|
c@24
|
1011 json11::Json::object po;
|
c@24
|
1012 po["plugins"] = arr;
|
c@24
|
1013
|
c@24
|
1014 jo["content"] = po;
|
c@17
|
1015 return json11::Json(jo);
|
c@17
|
1016 }
|
c@17
|
1017
|
c@17
|
1018 static json11::Json
|
c@17
|
1019 fromVampRequest_Load(const Vamp::HostExt::LoadRequest &req) {
|
c@17
|
1020
|
c@17
|
1021 json11::Json::object jo;
|
c@17
|
1022 jo["type"] = "load";
|
c@17
|
1023 jo["content"] = fromLoadRequest(req);
|
c@17
|
1024 return json11::Json(jo);
|
c@17
|
1025 }
|
c@17
|
1026
|
c@17
|
1027 static json11::Json
|
c@17
|
1028 fromVampResponse_Load(const Vamp::HostExt::LoadResponse &resp,
|
c@49
|
1029 const PluginHandleMapper &pmapper) {
|
c@17
|
1030
|
c@17
|
1031 json11::Json::object jo;
|
c@24
|
1032 jo["type"] = "load";
|
c@17
|
1033 jo["success"] = (resp.plugin != 0);
|
c@17
|
1034 jo["errorText"] = "";
|
c@49
|
1035 jo["content"] = fromLoadResponse(resp, pmapper);
|
c@17
|
1036 return json11::Json(jo);
|
c@17
|
1037 }
|
c@17
|
1038
|
c@17
|
1039 static json11::Json
|
c@17
|
1040 fromVampRequest_Configure(const Vamp::HostExt::ConfigurationRequest &req,
|
c@49
|
1041 const PluginHandleMapper &pmapper) {
|
c@17
|
1042
|
c@17
|
1043 json11::Json::object jo;
|
c@17
|
1044 jo["type"] = "configure";
|
c@49
|
1045 jo["content"] = fromConfigurationRequest(req, pmapper);
|
c@17
|
1046 return json11::Json(jo);
|
c@17
|
1047 }
|
c@17
|
1048
|
c@17
|
1049 static json11::Json
|
c@55
|
1050 fromVampResponse_Configure(const Vamp::HostExt::ConfigurationResponse &resp,
|
c@55
|
1051 const PluginHandleMapper &pmapper) {
|
c@17
|
1052
|
c@17
|
1053 json11::Json::object jo;
|
c@24
|
1054 jo["type"] = "configure";
|
c@17
|
1055 jo["success"] = (!resp.outputs.empty());
|
c@17
|
1056 jo["errorText"] = "";
|
c@55
|
1057 jo["content"] = fromConfigurationResponse(resp, pmapper);
|
c@17
|
1058 return json11::Json(jo);
|
c@17
|
1059 }
|
c@17
|
1060
|
c@17
|
1061 static json11::Json
|
c@17
|
1062 fromVampRequest_Process(const Vamp::HostExt::ProcessRequest &req,
|
c@49
|
1063 const PluginHandleMapper &pmapper,
|
c@44
|
1064 BufferSerialisation serialisation) {
|
c@17
|
1065
|
c@17
|
1066 json11::Json::object jo;
|
c@17
|
1067 jo["type"] = "process";
|
c@49
|
1068 jo["content"] = fromProcessRequest(req, pmapper, serialisation);
|
c@17
|
1069 return json11::Json(jo);
|
c@17
|
1070 }
|
c@17
|
1071
|
c@17
|
1072 static json11::Json
|
c@44
|
1073 fromVampResponse_Process(const Vamp::HostExt::ProcessResponse &resp,
|
c@51
|
1074 const PluginHandleMapper &pmapper,
|
c@44
|
1075 BufferSerialisation serialisation) {
|
c@17
|
1076
|
c@17
|
1077 json11::Json::object jo;
|
c@24
|
1078 jo["type"] = "process";
|
c@17
|
1079 jo["success"] = true;
|
c@17
|
1080 jo["errorText"] = "";
|
c@52
|
1081 json11::Json::object po;
|
c@52
|
1082 po["pluginHandle"] = pmapper.pluginToHandle(resp.plugin);
|
c@52
|
1083 po["features"] = fromFeatureSet(resp.features,
|
c@57
|
1084 *pmapper.pluginToOutputIdMapper(resp.plugin),
|
c@52
|
1085 serialisation);
|
c@52
|
1086 jo["content"] = po;
|
c@17
|
1087 return json11::Json(jo);
|
c@17
|
1088 }
|
c@17
|
1089
|
c@17
|
1090 static json11::Json
|
c@55
|
1091 fromVampRequest_Finish(const Vamp::HostExt::FinishRequest &req,
|
c@49
|
1092 const PluginHandleMapper &pmapper) {
|
c@17
|
1093
|
c@17
|
1094 json11::Json::object jo;
|
c@17
|
1095 jo["type"] = "finish";
|
c@24
|
1096 json11::Json::object fo;
|
c@55
|
1097 fo["pluginHandle"] = pmapper.pluginToHandle(req.plugin);
|
c@24
|
1098 jo["content"] = fo;
|
c@17
|
1099 return json11::Json(jo);
|
c@17
|
1100 }
|
c@17
|
1101
|
c@17
|
1102 static json11::Json
|
c@44
|
1103 fromVampResponse_Finish(const Vamp::HostExt::ProcessResponse &resp,
|
c@51
|
1104 const PluginHandleMapper &pmapper,
|
c@44
|
1105 BufferSerialisation serialisation) {
|
c@17
|
1106
|
c@17
|
1107 json11::Json::object jo;
|
c@24
|
1108 jo["type"] = "finish";
|
c@17
|
1109 jo["success"] = true;
|
c@17
|
1110 jo["errorText"] = "";
|
c@52
|
1111 json11::Json::object po;
|
c@52
|
1112 po["pluginHandle"] = pmapper.pluginToHandle(resp.plugin);
|
c@52
|
1113 po["features"] = fromFeatureSet(resp.features,
|
c@57
|
1114 *pmapper.pluginToOutputIdMapper(resp.plugin),
|
c@52
|
1115 serialisation);
|
c@52
|
1116 jo["content"] = po;
|
c@17
|
1117 return json11::Json(jo);
|
c@17
|
1118 }
|
c@24
|
1119
|
c@41
|
1120 static json11::Json
|
c@52
|
1121 fromError(std::string errorText, RRType responseType) {
|
c@41
|
1122
|
c@41
|
1123 json11::Json::object jo;
|
c@43
|
1124 std::string type;
|
c@41
|
1125
|
c@43
|
1126 if (responseType == RRType::List) type = "list";
|
c@43
|
1127 else if (responseType == RRType::Load) type = "load";
|
c@43
|
1128 else if (responseType == RRType::Configure) type = "configure";
|
c@43
|
1129 else if (responseType == RRType::Process) type = "process";
|
c@43
|
1130 else if (responseType == RRType::Finish) type = "finish";
|
c@43
|
1131 else type = "invalid";
|
c@41
|
1132
|
c@43
|
1133 jo["type"] = type;
|
c@41
|
1134 jo["success"] = false;
|
c@52
|
1135 jo["errorText"] = std::string("error in ") + type + " request: " + errorText;
|
c@41
|
1136 return json11::Json(jo);
|
c@41
|
1137 }
|
c@41
|
1138
|
c@24
|
1139 private: // go private briefly for a couple of helper functions
|
c@24
|
1140
|
c@24
|
1141 static void
|
c@60
|
1142 checkTypeField(json11::Json j, std::string expected, std::string &err) {
|
c@24
|
1143 if (!j["type"].is_string()) {
|
c@60
|
1144 err = "string expected for type";
|
c@60
|
1145 return;
|
c@24
|
1146 }
|
c@24
|
1147 if (j["type"].string_value() != expected) {
|
c@60
|
1148 err = "expected value \"" + expected + "\" for type";
|
c@60
|
1149 return;
|
c@24
|
1150 }
|
c@24
|
1151 }
|
c@24
|
1152
|
c@24
|
1153 static bool
|
c@60
|
1154 successful(json11::Json j, std::string &err) {
|
c@24
|
1155 if (!j["success"].is_bool()) {
|
c@60
|
1156 err = "bool expected for success";
|
c@60
|
1157 return false;
|
c@24
|
1158 }
|
c@24
|
1159 return j["success"].bool_value();
|
c@24
|
1160 }
|
c@24
|
1161
|
c@24
|
1162 public:
|
c@25
|
1163 static RRType
|
c@60
|
1164 getRequestResponseType(json11::Json j, std::string &err) {
|
c@25
|
1165
|
c@25
|
1166 if (!j["type"].is_string()) {
|
c@60
|
1167 err = "string expected for type";
|
c@60
|
1168 return RRType::NotValid;
|
c@25
|
1169 }
|
c@25
|
1170
|
c@25
|
1171 std::string type = j["type"].string_value();
|
c@25
|
1172
|
c@25
|
1173 if (type == "list") return RRType::List;
|
c@25
|
1174 else if (type == "load") return RRType::Load;
|
c@25
|
1175 else if (type == "configure") return RRType::Configure;
|
c@25
|
1176 else if (type == "process") return RRType::Process;
|
c@25
|
1177 else if (type == "finish") return RRType::Finish;
|
c@60
|
1178 else if (type == "invalid") return RRType::NotValid;
|
c@25
|
1179 else {
|
c@60
|
1180 err = "unknown or unexpected request/response type \"" + type + "\"";
|
c@60
|
1181 return RRType::NotValid;
|
c@25
|
1182 }
|
c@25
|
1183 }
|
c@44
|
1184
|
c@24
|
1185 static void
|
c@60
|
1186 toVampRequest_List(json11::Json j, std::string &err) {
|
c@60
|
1187 checkTypeField(j, "list", err);
|
c@24
|
1188 }
|
c@24
|
1189
|
c@56
|
1190 static Vamp::HostExt::ListResponse
|
c@60
|
1191 toVampResponse_List(json11::Json j, std::string &err) {
|
c@24
|
1192
|
c@56
|
1193 Vamp::HostExt::ListResponse resp;
|
c@60
|
1194 if (successful(j, err) && !failed(err)) {
|
c@24
|
1195 for (const auto &a: j["content"]["plugins"].array_items()) {
|
c@60
|
1196 resp.pluginData.push_back(toPluginStaticData(a, err));
|
c@60
|
1197 if (failed(err)) return {};
|
c@24
|
1198 }
|
c@24
|
1199 }
|
c@60
|
1200
|
c@56
|
1201 return resp;
|
c@24
|
1202 }
|
c@24
|
1203
|
c@24
|
1204 static Vamp::HostExt::LoadRequest
|
c@60
|
1205 toVampRequest_Load(json11::Json j, std::string &err) {
|
c@24
|
1206
|
c@60
|
1207 checkTypeField(j, "load", err);
|
c@60
|
1208 if (failed(err)) return {};
|
c@60
|
1209 return toLoadRequest(j["content"], err);
|
c@24
|
1210 }
|
c@24
|
1211
|
c@24
|
1212 static Vamp::HostExt::LoadResponse
|
c@60
|
1213 toVampResponse_Load(json11::Json j,
|
c@60
|
1214 const PluginHandleMapper &pmapper,
|
c@60
|
1215 std::string &err) {
|
c@24
|
1216
|
c@24
|
1217 Vamp::HostExt::LoadResponse resp;
|
c@60
|
1218 if (successful(j, err) && !failed(err)) {
|
c@60
|
1219 resp = toLoadResponse(j["content"], pmapper, err);
|
c@24
|
1220 }
|
c@24
|
1221 return resp;
|
c@24
|
1222 }
|
c@24
|
1223
|
c@24
|
1224 static Vamp::HostExt::ConfigurationRequest
|
c@60
|
1225 toVampRequest_Configure(json11::Json j,
|
c@60
|
1226 const PluginHandleMapper &pmapper,
|
c@60
|
1227 std::string &err) {
|
c@24
|
1228
|
c@60
|
1229 checkTypeField(j, "configure", err);
|
c@60
|
1230 if (failed(err)) return {};
|
c@60
|
1231 return toConfigurationRequest(j["content"], pmapper, err);
|
c@24
|
1232 }
|
c@24
|
1233
|
c@24
|
1234 static Vamp::HostExt::ConfigurationResponse
|
c@60
|
1235 toVampResponse_Configure(json11::Json j,
|
c@60
|
1236 const PluginHandleMapper &pmapper,
|
c@60
|
1237 std::string &err) {
|
c@24
|
1238
|
c@24
|
1239 Vamp::HostExt::ConfigurationResponse resp;
|
c@60
|
1240 if (successful(j, err) && !failed(err)) {
|
c@60
|
1241 resp = toConfigurationResponse(j["content"], pmapper, err);
|
c@24
|
1242 }
|
c@24
|
1243 return resp;
|
c@24
|
1244 }
|
c@24
|
1245
|
c@24
|
1246 static Vamp::HostExt::ProcessRequest
|
c@49
|
1247 toVampRequest_Process(json11::Json j, const PluginHandleMapper &pmapper,
|
c@60
|
1248 BufferSerialisation &serialisation, std::string &err) {
|
c@24
|
1249
|
c@60
|
1250 checkTypeField(j, "process", err);
|
c@60
|
1251 if (failed(err)) return {};
|
c@60
|
1252 return toProcessRequest(j["content"], pmapper, serialisation, err);
|
c@24
|
1253 }
|
c@24
|
1254
|
c@24
|
1255 static Vamp::HostExt::ProcessResponse
|
c@49
|
1256 toVampResponse_Process(json11::Json j,
|
c@51
|
1257 const PluginHandleMapper &pmapper,
|
c@60
|
1258 BufferSerialisation &serialisation, std::string &err) {
|
c@24
|
1259
|
c@24
|
1260 Vamp::HostExt::ProcessResponse resp;
|
c@60
|
1261 if (successful(j, err) && !failed(err)) {
|
c@51
|
1262 auto jc = j["content"];
|
c@52
|
1263 auto h = jc["pluginHandle"].int_value();
|
c@52
|
1264 resp.plugin = pmapper.handleToPlugin(h);
|
c@52
|
1265 resp.features = toFeatureSet(jc["features"],
|
c@57
|
1266 *pmapper.handleToOutputIdMapper(h),
|
c@60
|
1267 serialisation, err);
|
c@24
|
1268 }
|
c@24
|
1269 return resp;
|
c@24
|
1270 }
|
c@24
|
1271
|
c@55
|
1272 static Vamp::HostExt::FinishRequest
|
c@60
|
1273 toVampRequest_Finish(json11::Json j, const PluginHandleMapper &pmapper,
|
c@60
|
1274 std::string &err) {
|
c@24
|
1275
|
c@60
|
1276 checkTypeField(j, "finish", err);
|
c@60
|
1277 if (failed(err)) return {};
|
c@55
|
1278 Vamp::HostExt::FinishRequest req;
|
c@55
|
1279 req.plugin = pmapper.handleToPlugin
|
c@55
|
1280 (j["content"]["pluginHandle"].int_value());
|
c@55
|
1281 return req;
|
c@24
|
1282 }
|
c@24
|
1283
|
c@24
|
1284 static Vamp::HostExt::ProcessResponse
|
c@49
|
1285 toVampResponse_Finish(json11::Json j,
|
c@51
|
1286 const PluginHandleMapper &pmapper,
|
c@60
|
1287 BufferSerialisation &serialisation, std::string &err) {
|
c@24
|
1288
|
c@24
|
1289 Vamp::HostExt::ProcessResponse resp;
|
c@60
|
1290 if (successful(j, err) && !failed(err)) {
|
c@51
|
1291 auto jc = j["content"];
|
c@52
|
1292 auto h = jc["pluginHandle"].int_value();
|
c@52
|
1293 resp.plugin = pmapper.handleToPlugin(h);
|
c@52
|
1294 resp.features = toFeatureSet(jc["features"],
|
c@57
|
1295 *pmapper.handleToOutputIdMapper(h),
|
c@60
|
1296 serialisation, err);
|
c@24
|
1297 }
|
c@24
|
1298 return resp;
|
c@24
|
1299 }
|
c@5
|
1300 };
|
c@5
|
1301
|
c@10
|
1302 }
|
c@5
|
1303
|
c@5
|
1304 #endif
|