Mercurial > hg > piper-cpp
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 |