comparison vamp-json/VampJson.h @ 75:81e1c48e97f9

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