comparison json/VampJson.h @ 60:8a4bcb3dc3a6

Replace exceptions throughout the JSON-handling and adapter code with string-arg error handling. No longer need exception handling enabled in Emscripten (with its consequent runtime overhead - though we still need to check whether this error handling regime is actually faster).
author Chris Cannam <c.cannam@qmul.ac.uk>
date Tue, 20 Sep 2016 16:35:47 +0100
parents 77833938f0f8
children 2d866edd79d5
comparison
equal deleted inserted replaced
59:77833938f0f8 60:8a4bcb3dc3a6
36 #define VAMP_JSON_H 36 #define VAMP_JSON_H
37 37
38 #include <vector> 38 #include <vector>
39 #include <string> 39 #include <string>
40 #include <sstream> 40 #include <sstream>
41 #include <stdexcept>
42 41
43 #include <json11/json11.hpp> 42 #include <json11/json11.hpp>
44 #include <base-n/include/basen.hpp> 43 #include <base-n/include/basen.hpp>
45 44
46 #include <vamp-hostsdk/Plugin.h> 45 #include <vamp-hostsdk/Plugin.h>
54 53
55 /** 54 /**
56 * Convert the structures laid out in the Vamp SDK classes into JSON 55 * Convert the structures laid out in the Vamp SDK classes into JSON
57 * (and back again) following the schema in the vamp-json-schema 56 * (and back again) following the schema in the vamp-json-schema
58 * project repo. 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.
59 */ 73 */
74
75 //!!! todo: convert pmapper to err style
76
60 class VampJson 77 class VampJson
61 { 78 {
62 public: 79 public:
63 /** Serialisation format for arrays of floats (process input and 80 /** Serialisation format for arrays of floats (process input and
64 * feature values). Structures that can be serialised in more 81 * feature values). Structures that can be serialised in more
86 * a consumer that expects padding. 103 * a consumer that expects padding.
87 */ 104 */
88 Base64 105 Base64
89 }; 106 };
90 107
91 class Failure : virtual public std::runtime_error { 108 static bool failed(const std::string &err) {
92 public: 109 return err != "";
93 Failure(std::string s) : runtime_error(s) { } 110 }
94 };
95 111
96 template <typename T> 112 template <typename T>
97 static json11::Json 113 static json11::Json
98 fromBasicDescriptor(const T &t) { 114 fromBasicDescriptor(const T &t) {
99 return json11::Json::object { 115 return json11::Json::object {
103 }; 119 };
104 } 120 }
105 121
106 template <typename T> 122 template <typename T>
107 static void 123 static void
108 toBasicDescriptor(json11::Json j, T &t) { 124 toBasicDescriptor(json11::Json j, T &t, std::string &err) {
109 if (!j.is_object()) { 125 if (!j.is_object()) {
110 throw Failure("object expected for basic descriptor content"); 126 err = "object expected for basic descriptor content";
127 return;
111 } 128 }
112 if (!j["identifier"].is_string()) { 129 if (!j["identifier"].is_string()) {
113 throw Failure("string expected for identifier"); 130 err = "string expected for identifier";
131 return;
114 } 132 }
115 t.identifier = j["identifier"].string_value(); 133 t.identifier = j["identifier"].string_value();
116 t.name = j["name"].string_value(); 134 t.name = j["name"].string_value();
117 t.description = j["description"].string_value(); 135 t.description = j["description"].string_value();
118 } 136 }
126 }; 144 };
127 } 145 }
128 146
129 template <typename T> 147 template <typename T>
130 static bool 148 static bool
131 toValueExtents(json11::Json j, T &t) { 149 toValueExtents(json11::Json j, T &t, std::string &err) {
132 if (j["extents"].is_null()) { 150 if (j["extents"].is_null()) {
133 return false; 151 return false;
134 } else if (j["extents"].is_object()) { 152 } else if (j["extents"].is_object()) {
135 if (j["extents"]["min"].is_number() && 153 if (j["extents"]["min"].is_number() &&
136 j["extents"]["max"].is_number()) { 154 j["extents"]["max"].is_number()) {
137 t.minValue = j["extents"]["min"].number_value(); 155 t.minValue = j["extents"]["min"].number_value();
138 t.maxValue = j["extents"]["max"].number_value(); 156 t.maxValue = j["extents"]["max"].number_value();
139 return true; 157 return true;
140 } else { 158 } else {
141 throw Failure("numbers expected for min and max"); 159 err = "numbers expected for min and max";
160 return false;
142 } 161 }
143 } else { 162 } else {
144 throw Failure("object expected for extents (if present)"); 163 err = "object expected for extents (if present)";
164 return false;
145 } 165 }
146 } 166 }
147 167
148 static json11::Json 168 static json11::Json
149 fromRealTime(const Vamp::RealTime &r) { 169 fromRealTime(const Vamp::RealTime &r) {
152 { "n", r.nsec } 172 { "n", r.nsec }
153 }; 173 };
154 } 174 }
155 175
156 static Vamp::RealTime 176 static Vamp::RealTime
157 toRealTime(json11::Json j) { 177 toRealTime(json11::Json j, std::string &err) {
158 json11::Json sec = j["s"]; 178 json11::Json sec = j["s"];
159 json11::Json nsec = j["n"]; 179 json11::Json nsec = j["n"];
160 if (!sec.is_number() || !nsec.is_number()) { 180 if (!sec.is_number() || !nsec.is_number()) {
161 throw Failure("invalid Vamp::RealTime object " + j.dump()); 181 err = "invalid Vamp::RealTime object " + j.dump();
182 return {};
162 } 183 }
163 return Vamp::RealTime(sec.int_value(), nsec.int_value()); 184 return Vamp::RealTime(sec.int_value(), nsec.int_value());
164 } 185 }
165 186
166 static std::string 187 static std::string
175 } 196 }
176 return ""; 197 return "";
177 } 198 }
178 199
179 static Vamp::Plugin::OutputDescriptor::SampleType 200 static Vamp::Plugin::OutputDescriptor::SampleType
180 toSampleType(std::string text) { 201 toSampleType(std::string text, std::string &err) {
181 if (text == "OneSamplePerStep") { 202 if (text == "OneSamplePerStep") {
182 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep; 203 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
183 } else if (text == "FixedSampleRate") { 204 } else if (text == "FixedSampleRate") {
184 return Vamp::Plugin::OutputDescriptor::FixedSampleRate; 205 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
185 } else if (text == "VariableSampleRate") { 206 } else if (text == "VariableSampleRate") {
186 return Vamp::Plugin::OutputDescriptor::VariableSampleRate; 207 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
187 } else { 208 } else {
188 throw Failure("invalid sample type string: " + text); 209 err = "invalid sample type string: " + text;
210 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
189 } 211 }
190 } 212 }
191 213
192 static json11::Json 214 static json11::Json
193 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) { 215 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) {
211 } 233 }
212 return json11::Json(jo); 234 return json11::Json(jo);
213 } 235 }
214 236
215 static Vamp::Plugin::OutputDescriptor 237 static Vamp::Plugin::OutputDescriptor
216 toOutputDescriptor(json11::Json j) { 238 toOutputDescriptor(json11::Json j, std::string &err) {
217 239
218 Vamp::Plugin::OutputDescriptor od; 240 Vamp::Plugin::OutputDescriptor od;
219 if (!j.is_object()) { 241 if (!j.is_object()) {
220 throw Failure("object expected for output descriptor"); 242 err = "object expected for output descriptor";
221 } 243 return {};
222 244 }
223 toBasicDescriptor(j["basic"], od); 245
246 toBasicDescriptor(j["basic"], od, err);
247 if (failed(err)) return {};
224 248
225 od.unit = j["unit"].string_value(); 249 od.unit = j["unit"].string_value();
226 250
227 od.sampleType = toSampleType(j["sampleType"].string_value()); 251 od.sampleType = toSampleType(j["sampleType"].string_value(), err);
252 if (failed(err)) return {};
228 253
229 if (!j["sampleRate"].is_number()) { 254 if (!j["sampleRate"].is_number()) {
230 throw Failure("number expected for sample rate"); 255 err = "number expected for sample rate";
256 return {};
231 } 257 }
232 od.sampleRate = j["sampleRate"].number_value(); 258 od.sampleRate = j["sampleRate"].number_value();
233 od.hasDuration = j["hasDuration"].bool_value(); 259 od.hasDuration = j["hasDuration"].bool_value();
234 260
235 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) { 261 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) {
236 od.hasFixedBinCount = true; 262 od.hasFixedBinCount = true;
237 od.binCount = j["binCount"].int_value(); 263 od.binCount = j["binCount"].int_value();
238 for (auto &n: j["binNames"].array_items()) { 264 for (auto &n: j["binNames"].array_items()) {
239 if (!n.is_string()) { 265 if (!n.is_string()) {
240 throw Failure("string expected for bin name"); 266 err = "string expected for bin name";
267 return {};
241 } 268 }
242 od.binNames.push_back(n.string_value()); 269 od.binNames.push_back(n.string_value());
243 } 270 }
244 } else { 271 } else {
245 od.hasFixedBinCount = false; 272 od.hasFixedBinCount = false;
246 } 273 }
247 274
248 bool extentsPresent = toValueExtents(j, od); 275 bool extentsPresent = toValueExtents(j, od, err);
276 if (failed(err)) return {};
277
249 od.hasKnownExtents = extentsPresent; 278 od.hasKnownExtents = extentsPresent;
250 279
251 if (j["quantizeStep"].is_number()) { 280 if (j["quantizeStep"].is_number()) {
252 od.isQuantized = true; 281 od.isQuantized = true;
253 od.quantizeStep = j["quantizeStep"].number_value(); 282 od.quantizeStep = j["quantizeStep"].number_value();
274 } 303 }
275 return json11::Json(jo); 304 return json11::Json(jo);
276 } 305 }
277 306
278 static Vamp::PluginBase::ParameterDescriptor 307 static Vamp::PluginBase::ParameterDescriptor
279 toParameterDescriptor(json11::Json j) { 308 toParameterDescriptor(json11::Json j, std::string &err) {
280 309
281 Vamp::PluginBase::ParameterDescriptor pd; 310 Vamp::PluginBase::ParameterDescriptor pd;
282 if (!j.is_object()) { 311 if (!j.is_object()) {
283 throw Failure("object expected for parameter descriptor"); 312 err = "object expected for parameter descriptor";
284 } 313 return {};
285 314 }
286 toBasicDescriptor(j["basic"], pd); 315
316 toBasicDescriptor(j["basic"], pd, err);
317 if (failed(err)) return {};
287 318
288 pd.unit = j["unit"].string_value(); 319 pd.unit = j["unit"].string_value();
289 320
290 bool extentsPresent = toValueExtents(j, pd); 321 bool extentsPresent = toValueExtents(j, pd, err);
322 if (failed(err)) return {};
291 if (!extentsPresent) { 323 if (!extentsPresent) {
292 throw Failure("extents must be present in parameter descriptor"); 324 err = "extents must be present in parameter descriptor";
325 return {};
293 } 326 }
294 327
295 if (!j["defaultValue"].is_number()) { 328 if (!j["defaultValue"].is_number()) {
296 throw Failure("number expected for default value"); 329 err = "number expected for default value";
330 return {};
297 } 331 }
298 332
299 pd.defaultValue = j["defaultValue"].number_value(); 333 pd.defaultValue = j["defaultValue"].number_value();
300 334
301 pd.valueNames.clear(); 335 pd.valueNames.clear();
302 for (auto &n: j["valueNames"].array_items()) { 336 for (auto &n: j["valueNames"].array_items()) {
303 if (!n.is_string()) { 337 if (!n.is_string()) {
304 throw Failure("string expected for value name"); 338 err = "string expected for value name";
339 return {};
305 } 340 }
306 pd.valueNames.push_back(n.string_value()); 341 pd.valueNames.push_back(n.string_value());
307 } 342 }
308 343
309 if (j["quantizeStep"].is_number()) { 344 if (j["quantizeStep"].is_number()) {
326 bn::encode_b64(start, end, back_inserter(encoded)); 361 bn::encode_b64(start, end, back_inserter(encoded));
327 return encoded; 362 return encoded;
328 } 363 }
329 364
330 static std::vector<float> 365 static std::vector<float>
331 toFloatBuffer(std::string encoded) { 366 toFloatBuffer(std::string encoded, std::string & /* err */) {
332 std::string decoded; 367 std::string decoded;
333 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded)); 368 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded));
334 const float *buffer = reinterpret_cast<const float *>(decoded.c_str()); 369 const float *buffer = reinterpret_cast<const float *>(decoded.c_str());
335 size_t n = decoded.size() / sizeof(float); 370 size_t n = decoded.size() / sizeof(float);
336 return std::vector<float>(buffer, buffer + n); 371 return std::vector<float>(buffer, buffer + n);
361 } 396 }
362 return json11::Json(jo); 397 return json11::Json(jo);
363 } 398 }
364 399
365 static Vamp::Plugin::Feature 400 static Vamp::Plugin::Feature
366 toFeature(json11::Json j, 401 toFeature(json11::Json j, BufferSerialisation &serialisation, std::string &err) {
367 BufferSerialisation &serialisation) {
368 402
369 Vamp::Plugin::Feature f; 403 Vamp::Plugin::Feature f;
370 if (!j.is_object()) { 404 if (!j.is_object()) {
371 throw Failure("object expected for feature"); 405 err = "object expected for feature";
406 return {};
372 } 407 }
373 if (j["timestamp"].is_object()) { 408 if (j["timestamp"].is_object()) {
374 f.timestamp = toRealTime(j["timestamp"]); 409 f.timestamp = toRealTime(j["timestamp"], err);
410 if (failed(err)) return {};
375 f.hasTimestamp = true; 411 f.hasTimestamp = true;
376 } 412 }
377 if (j["duration"].is_object()) { 413 if (j["duration"].is_object()) {
378 f.duration = toRealTime(j["duration"]); 414 f.duration = toRealTime(j["duration"], err);
415 if (failed(err)) return {};
379 f.hasDuration = true; 416 f.hasDuration = true;
380 } 417 }
381 if (j["b64values"].is_string()) { 418 if (j["b64values"].is_string()) {
382 f.values = toFloatBuffer(j["b64values"].string_value()); 419 f.values = toFloatBuffer(j["b64values"].string_value(), err);
420 if (failed(err)) return {};
383 serialisation = BufferSerialisation::Base64; 421 serialisation = BufferSerialisation::Base64;
384 } else if (j["values"].is_array()) { 422 } else if (j["values"].is_array()) {
385 for (auto v : j["values"].array_items()) { 423 for (auto v : j["values"].array_items()) {
386 f.values.push_back(v.number_value()); 424 f.values.push_back(v.number_value());
387 } 425 }
407 return json11::Json(jo); 445 return json11::Json(jo);
408 } 446 }
409 447
410 static Vamp::Plugin::FeatureList 448 static Vamp::Plugin::FeatureList
411 toFeatureList(json11::Json j, 449 toFeatureList(json11::Json j,
412 BufferSerialisation &serialisation) { 450 BufferSerialisation &serialisation, std::string &err) {
413 451
414 Vamp::Plugin::FeatureList fl; 452 Vamp::Plugin::FeatureList fl;
415 if (!j.is_array()) { 453 if (!j.is_array()) {
416 throw Failure("array expected for feature list"); 454 err = "array expected for feature list";
455 return {};
417 } 456 }
418 for (const json11::Json &fj : j.array_items()) { 457 for (const json11::Json &fj : j.array_items()) {
419 fl.push_back(toFeature(fj, serialisation)); 458 fl.push_back(toFeature(fj, serialisation, err));
459 if (failed(err)) return {};
420 } 460 }
421 return fl; 461 return fl;
422 } 462 }
423 463
424 static Vamp::Plugin::FeatureSet 464 static Vamp::Plugin::FeatureSet
425 toFeatureSet(json11::Json j, 465 toFeatureSet(json11::Json j,
426 const PluginOutputIdMapper &omapper, 466 const PluginOutputIdMapper &omapper,
427 BufferSerialisation &serialisation) { 467 BufferSerialisation &serialisation,
468 std::string &err) {
428 469
429 Vamp::Plugin::FeatureSet fs; 470 Vamp::Plugin::FeatureSet fs;
430 if (!j.is_object()) { 471 if (!j.is_object()) {
431 throw Failure("object expected for feature set"); 472 err = "object expected for feature set";
473 return {};
432 } 474 }
433 for (auto &entry : j.object_items()) { 475 for (auto &entry : j.object_items()) {
434 int n = omapper.idToIndex(entry.first); 476 int n = omapper.idToIndex(entry.first);
435 if (fs.find(n) != fs.end()) { 477 if (fs.find(n) != fs.end()) {
436 throw Failure("duplicate numerical index for output"); 478 err = "duplicate numerical index for output";
437 } 479 return {};
438 fs[n] = toFeatureList(entry.second, serialisation); 480 }
481 fs[n] = toFeatureList(entry.second, serialisation, err);
482 if (failed(err)) return {};
439 } 483 }
440 return fs; 484 return fs;
441 } 485 }
442 486
443 static std::string 487 static std::string
451 } 495 }
452 return ""; 496 return "";
453 } 497 }
454 498
455 static Vamp::Plugin::InputDomain 499 static Vamp::Plugin::InputDomain
456 toInputDomain(std::string text) { 500 toInputDomain(std::string text, std::string &err) {
457 501
458 if (text == "TimeDomain") { 502 if (text == "TimeDomain") {
459 return Vamp::Plugin::TimeDomain; 503 return Vamp::Plugin::TimeDomain;
460 } else if (text == "FrequencyDomain") { 504 } else if (text == "FrequencyDomain") {
461 return Vamp::Plugin::FrequencyDomain; 505 return Vamp::Plugin::FrequencyDomain;
462 } else { 506 } else {
463 throw Failure("invalid input domain string: " + text); 507 err = "invalid input domain string: " + text;
508 return {};
464 } 509 }
465 } 510 }
466 511
467 static json11::Json 512 static json11::Json
468 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) { 513 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) {
500 545
501 return json11::Json(jo); 546 return json11::Json(jo);
502 } 547 }
503 548
504 static Vamp::HostExt::PluginStaticData 549 static Vamp::HostExt::PluginStaticData
505 toPluginStaticData(json11::Json j) { 550 toPluginStaticData(json11::Json j, std::string &err) {
506 551
507 std::string err;
508 if (!j.has_shape({ 552 if (!j.has_shape({
509 { "pluginKey", json11::Json::STRING }, 553 { "pluginKey", json11::Json::STRING },
510 { "pluginVersion", json11::Json::NUMBER }, 554 { "pluginVersion", json11::Json::NUMBER },
511 { "minChannelCount", json11::Json::NUMBER }, 555 { "minChannelCount", json11::Json::NUMBER },
512 { "maxChannelCount", json11::Json::NUMBER }, 556 { "maxChannelCount", json11::Json::NUMBER },
513 { "inputDomain", json11::Json::STRING }}, err)) { 557 { "inputDomain", json11::Json::STRING }}, err)) {
514 throw Failure("malformed plugin static data: " + err); 558
515 } 559 err = "malformed plugin static data: " + err;
516 560
517 if (!j["basicOutputInfo"].is_array()) { 561 } else if (!j["basicOutputInfo"].is_array()) {
518 throw Failure("array expected for basic output info"); 562
519 } 563 err = "array expected for basic output info";
520 564
521 if (!j["maker"].is_null() && 565 } else if (!j["maker"].is_null() &&
522 !j["maker"].is_string()) { 566 !j["maker"].is_string()) {
523 throw Failure("string expected for maker"); 567
524 } 568 err = "string expected for maker";
525 569
526 if (!j["copyright"].is_null() && 570 } else if (!j["copyright"].is_null() &&
527 !j["copyright"].is_string()) { 571 !j["copyright"].is_string()) {
528 throw Failure("string expected for copyright"); 572 err = "string expected for copyright";
529 } 573
530 574 } else if (!j["category"].is_null() &&
531 if (!j["category"].is_null() && 575 !j["category"].is_array()) {
532 !j["category"].is_array()) { 576
533 throw Failure("array expected for category"); 577 err = "array expected for category";
534 } 578
535 579 } else if (!j["parameters"].is_null() &&
536 if (!j["parameters"].is_null() && 580 !j["parameters"].is_array()) {
537 !j["parameters"].is_array()) { 581
538 throw Failure("array expected for parameters"); 582 err = "array expected for parameters";
539 } 583
540 584 } else if (!j["programs"].is_null() &&
541 if (!j["programs"].is_null() && 585 !j["programs"].is_array()) {
542 !j["programs"].is_array()) { 586
543 throw Failure("array expected for programs"); 587 err = "array expected for programs";
544 } 588
545 589 } else if (!j["inputDomain"].is_null() &&
546 if (!j["inputDomain"].is_null() && 590 !j["inputDomain"].is_string()) {
547 !j["inputDomain"].is_string()) { 591
548 throw Failure("string expected for inputDomain"); 592 err = "string expected for inputDomain";
549 } 593
550 594 } else if (!j["basicOutputInfo"].is_null() &&
551 if (!j["basicOutputInfo"].is_null() && 595 !j["basicOutputInfo"].is_array()) {
552 !j["basicOutputInfo"].is_array()) { 596
553 throw Failure("array expected for basicOutputInfo"); 597 err = "array expected for basicOutputInfo";
554 } 598
555 599 } else {
556 Vamp::HostExt::PluginStaticData psd; 600
557 601 Vamp::HostExt::PluginStaticData psd;
558 psd.pluginKey = j["pluginKey"].string_value(); 602
559 603 psd.pluginKey = j["pluginKey"].string_value();
560 toBasicDescriptor(j["basic"], psd.basic); 604
561 605 toBasicDescriptor(j["basic"], psd.basic, err);
562 psd.maker = j["maker"].string_value(); 606 if (failed(err)) return {};
563 psd.copyright = j["copyright"].string_value(); 607
564 psd.pluginVersion = j["pluginVersion"].int_value(); 608 psd.maker = j["maker"].string_value();
565 609 psd.copyright = j["copyright"].string_value();
566 for (const auto &c : j["category"].array_items()) { 610 psd.pluginVersion = j["pluginVersion"].int_value();
567 if (!c.is_string()) { 611
568 throw Failure("strings expected in category array"); 612 for (const auto &c : j["category"].array_items()) {
569 } 613 if (!c.is_string()) {
570 psd.category.push_back(c.string_value()); 614 err = "strings expected in category array";
571 } 615 return {};
572 616 }
573 psd.minChannelCount = j["minChannelCount"].int_value(); 617 psd.category.push_back(c.string_value());
574 psd.maxChannelCount = j["maxChannelCount"].int_value(); 618 }
575 619
576 for (const auto &p : j["parameters"].array_items()) { 620 psd.minChannelCount = j["minChannelCount"].int_value();
577 auto pd = toParameterDescriptor(p); 621 psd.maxChannelCount = j["maxChannelCount"].int_value();
578 psd.parameters.push_back(pd); 622
579 } 623 for (const auto &p : j["parameters"].array_items()) {
580 624 auto pd = toParameterDescriptor(p, err);
581 for (const auto &p : j["programs"].array_items()) { 625 if (failed(err)) return {};
582 if (!p.is_string()) { 626 psd.parameters.push_back(pd);
583 throw Failure("strings expected in programs array"); 627 }
584 } 628
585 psd.programs.push_back(p.string_value()); 629 for (const auto &p : j["programs"].array_items()) {
586 } 630 if (!p.is_string()) {
587 631 err = "strings expected in programs array";
588 psd.inputDomain = toInputDomain(j["inputDomain"].string_value()); 632 return {};
589 633 }
590 for (const auto &bo : j["basicOutputInfo"].array_items()) { 634 psd.programs.push_back(p.string_value());
591 Vamp::HostExt::PluginStaticData::Basic b; 635 }
592 toBasicDescriptor(bo, b); 636
593 psd.basicOutputInfo.push_back(b); 637 psd.inputDomain = toInputDomain(j["inputDomain"].string_value(), err);
594 } 638 if (failed(err)) return {};
595 639
596 return psd; 640 for (const auto &bo : j["basicOutputInfo"].array_items()) {
641 Vamp::HostExt::PluginStaticData::Basic b;
642 toBasicDescriptor(bo, b, err);
643 if (failed(err)) return {};
644 psd.basicOutputInfo.push_back(b);
645 }
646
647 return psd;
648 }
649
650 // fallthrough error case
651 return {};
597 } 652 }
598 653
599 static json11::Json 654 static json11::Json
600 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) { 655 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) {
601 656
617 672
618 return json11::Json(jo); 673 return json11::Json(jo);
619 } 674 }
620 675
621 static Vamp::HostExt::PluginConfiguration 676 static Vamp::HostExt::PluginConfiguration
622 toPluginConfiguration(json11::Json j) { 677 toPluginConfiguration(json11::Json j, std::string &err) {
623 678
624 std::string err;
625 if (!j.has_shape({ 679 if (!j.has_shape({
626 { "channelCount", json11::Json::NUMBER }, 680 { "channelCount", json11::Json::NUMBER },
627 { "stepSize", json11::Json::NUMBER }, 681 { "stepSize", json11::Json::NUMBER },
628 { "blockSize", json11::Json::NUMBER } }, err)) { 682 { "blockSize", json11::Json::NUMBER } }, err)) {
629 throw Failure("malformed plugin configuration: " + err); 683 err = "malformed plugin configuration: " + err;
684 return {};
630 } 685 }
631 686
632 if (!j["parameterValues"].is_null() && 687 if (!j["parameterValues"].is_null() &&
633 !j["parameterValues"].is_object()) { 688 !j["parameterValues"].is_object()) {
634 throw Failure("object expected for parameter values"); 689 err = "object expected for parameter values";
690 return {};
635 } 691 }
636 692
637 for (auto &pv : j["parameterValues"].object_items()) { 693 for (auto &pv : j["parameterValues"].object_items()) {
638 if (!pv.second.is_number()) { 694 if (!pv.second.is_number()) {
639 throw Failure("number expected for parameter value"); 695 err = "number expected for parameter value";
696 return {};
640 } 697 }
641 } 698 }
642 699
643 if (!j["currentProgram"].is_null() && 700 if (!j["currentProgram"].is_null() &&
644 !j["currentProgram"].is_string()) { 701 !j["currentProgram"].is_string()) {
645 throw Failure("string expected for program name"); 702 err = "string expected for program name";
703 return {};
646 } 704 }
647 705
648 Vamp::HostExt::PluginConfiguration config; 706 Vamp::HostExt::PluginConfiguration config;
649 707
650 config.channelCount = j["channelCount"].number_value(); 708 config.channelCount = j["channelCount"].number_value();
679 737
680 return json11::Json(arr); 738 return json11::Json(arr);
681 } 739 }
682 740
683 static Vamp::HostExt::PluginLoader::AdapterFlags 741 static Vamp::HostExt::PluginLoader::AdapterFlags
684 toAdapterFlags(json11::Json j) { 742 toAdapterFlags(json11::Json j, std::string &err) {
743
744 int flags = 0x0;
685 745
686 if (!j.is_array()) { 746 if (!j.is_array()) {
687 throw Failure("array expected for adapter flags"); 747
688 } 748 err = "array expected for adapter flags";
689 int flags = 0x0; 749
690 750 } else {
691 for (auto &jj: j.array_items()) { 751
692 if (!jj.is_string()) { 752 for (auto &jj: j.array_items()) {
693 throw Failure("string expected for adapter flag"); 753 if (!jj.is_string()) {
694 } 754 err = "string expected for adapter flag";
695 std::string text = jj.string_value(); 755 break;
696 if (text == "AdaptInputDomain") { 756 }
697 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN; 757 std::string text = jj.string_value();
698 } else if (text == "AdaptChannelCount") { 758 if (text == "AdaptInputDomain") {
699 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT; 759 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
700 } else if (text == "AdaptBufferSize") { 760 } else if (text == "AdaptChannelCount") {
701 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE; 761 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
702 } else if (text == "AdaptAllSafe") { 762 } else if (text == "AdaptBufferSize") {
703 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE; 763 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
704 } else if (text == "AdaptAll") { 764 } else if (text == "AdaptAllSafe") {
705 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL; 765 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE;
706 } else { 766 } else if (text == "AdaptAll") {
707 throw Failure("invalid adapter flag string: " + text); 767 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL;
768 } else {
769 err = "invalid adapter flag string: " + text;
770 break;
771 }
708 } 772 }
709 } 773 }
710 774
711 return Vamp::HostExt::PluginLoader::AdapterFlags(flags); 775 return Vamp::HostExt::PluginLoader::AdapterFlags(flags);
712 } 776 }
720 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags); 784 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags);
721 return json11::Json(jo); 785 return json11::Json(jo);
722 } 786 }
723 787
724 static Vamp::HostExt::LoadRequest 788 static Vamp::HostExt::LoadRequest
725 toLoadRequest(json11::Json j) { 789 toLoadRequest(json11::Json j, std::string &err) {
726 790
727 std::string err;
728
729 if (!j.has_shape({ 791 if (!j.has_shape({
730 { "pluginKey", json11::Json::STRING }, 792 { "pluginKey", json11::Json::STRING },
731 { "inputSampleRate", json11::Json::NUMBER } }, err)) { 793 { "inputSampleRate", json11::Json::NUMBER } }, err)) {
732 throw Failure("malformed load request: " + err); 794 err = "malformed load request: " + err;
795 return {};
733 } 796 }
734 797
735 Vamp::HostExt::LoadRequest req; 798 Vamp::HostExt::LoadRequest req;
736 req.pluginKey = j["pluginKey"].string_value(); 799 req.pluginKey = j["pluginKey"].string_value();
737 req.inputSampleRate = j["inputSampleRate"].number_value(); 800 req.inputSampleRate = j["inputSampleRate"].number_value();
738 if (!j["adapterFlags"].is_null()) { 801 if (!j["adapterFlags"].is_null()) {
739 req.adapterFlags = toAdapterFlags(j["adapterFlags"]); 802 req.adapterFlags = toAdapterFlags(j["adapterFlags"], err);
803 if (failed(err)) return {};
740 } 804 }
741 return req; 805 return req;
742 } 806 }
743 807
744 static json11::Json 808 static json11::Json
753 return json11::Json(jo); 817 return json11::Json(jo);
754 } 818 }
755 819
756 static Vamp::HostExt::LoadResponse 820 static Vamp::HostExt::LoadResponse
757 toLoadResponse(json11::Json j, 821 toLoadResponse(json11::Json j,
758 const PluginHandleMapper &pmapper) { 822 const PluginHandleMapper &pmapper, std::string &err) {
759
760 std::string err;
761 823
762 if (!j.has_shape({ 824 if (!j.has_shape({
763 { "pluginHandle", json11::Json::NUMBER }, 825 { "pluginHandle", json11::Json::NUMBER },
764 { "staticData", json11::Json::OBJECT }, 826 { "staticData", json11::Json::OBJECT },
765 { "defaultConfiguration", json11::Json::OBJECT } }, err)) { 827 { "defaultConfiguration", json11::Json::OBJECT } }, err)) {
766 throw Failure("malformed load response: " + err); 828 err = "malformed load response: " + err;
829 return {};
767 } 830 }
768 831
769 Vamp::HostExt::LoadResponse resp; 832 Vamp::HostExt::LoadResponse resp;
770 resp.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value()); 833 resp.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
771 resp.staticData = toPluginStaticData(j["staticData"]); 834 resp.staticData = toPluginStaticData(j["staticData"], err);
772 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"]); 835 if (failed(err)) return {};
836 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"],
837 err);
838 if (failed(err)) return {};
773 return resp; 839 return resp;
774 } 840 }
775 841
776 static json11::Json 842 static json11::Json
777 fromConfigurationRequest(const Vamp::HostExt::ConfigurationRequest &cr, 843 fromConfigurationRequest(const Vamp::HostExt::ConfigurationRequest &cr,
785 return json11::Json(jo); 851 return json11::Json(jo);
786 } 852 }
787 853
788 static Vamp::HostExt::ConfigurationRequest 854 static Vamp::HostExt::ConfigurationRequest
789 toConfigurationRequest(json11::Json j, 855 toConfigurationRequest(json11::Json j,
790 const PluginHandleMapper &pmapper) { 856 const PluginHandleMapper &pmapper, std::string &err) {
791
792 std::string err;
793 857
794 if (!j.has_shape({ 858 if (!j.has_shape({
795 { "pluginHandle", json11::Json::NUMBER }, 859 { "pluginHandle", json11::Json::NUMBER },
796 { "configuration", json11::Json::OBJECT } }, err)) { 860 { "configuration", json11::Json::OBJECT } }, err)) {
797 throw Failure("malformed configuration request: " + err); 861 err = "malformed configuration request: " + err;
862 return {};
798 } 863 }
799 864
800 Vamp::HostExt::ConfigurationRequest cr; 865 Vamp::HostExt::ConfigurationRequest cr;
801 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value()); 866 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
802 cr.configuration = toPluginConfiguration(j["configuration"]); 867 cr.configuration = toPluginConfiguration(j["configuration"], err);
868 if (failed(err)) return {};
803 return cr; 869 return cr;
804 } 870 }
805 871
806 static json11::Json 872 static json11::Json
807 fromConfigurationResponse(const Vamp::HostExt::ConfigurationResponse &cr, 873 fromConfigurationResponse(const Vamp::HostExt::ConfigurationResponse &cr,
820 return json11::Json(jo); 886 return json11::Json(jo);
821 } 887 }
822 888
823 static Vamp::HostExt::ConfigurationResponse 889 static Vamp::HostExt::ConfigurationResponse
824 toConfigurationResponse(json11::Json j, 890 toConfigurationResponse(json11::Json j,
825 const PluginHandleMapper &pmapper) { 891 const PluginHandleMapper &pmapper, std::string &err) {
826 892
827 Vamp::HostExt::ConfigurationResponse cr; 893 Vamp::HostExt::ConfigurationResponse cr;
828 894
829 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value()); 895 cr.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
830 896
831 if (!j["outputList"].is_array()) { 897 if (!j["outputList"].is_array()) {
832 throw Failure("array expected for output list"); 898 err = "array expected for output list";
899 return {};
833 } 900 }
834 901
835 for (const auto &o: j["outputList"].array_items()) { 902 for (const auto &o: j["outputList"].array_items()) {
836 cr.outputs.push_back(toOutputDescriptor(o)); 903 cr.outputs.push_back(toOutputDescriptor(o, err));
904 if (failed(err)) return {};
837 } 905 }
838 906
839 return cr; 907 return cr;
840 } 908 }
841 909
869 } 937 }
870 938
871 static Vamp::HostExt::ProcessRequest 939 static Vamp::HostExt::ProcessRequest
872 toProcessRequest(json11::Json j, 940 toProcessRequest(json11::Json j,
873 const PluginHandleMapper &pmapper, 941 const PluginHandleMapper &pmapper,
874 BufferSerialisation &serialisation) { 942 BufferSerialisation &serialisation, std::string &err) {
875
876 std::string err;
877 943
878 if (!j.has_shape({ 944 if (!j.has_shape({
879 { "pluginHandle", json11::Json::NUMBER }, 945 { "pluginHandle", json11::Json::NUMBER },
880 { "processInput", json11::Json::OBJECT } }, err)) { 946 { "processInput", json11::Json::OBJECT } }, err)) {
881 throw Failure("malformed process request: " + err); 947 err = "malformed process request: " + err;
948 return {};
882 } 949 }
883 950
884 auto input = j["processInput"]; 951 auto input = j["processInput"];
885 952
886 if (!input.has_shape({ 953 if (!input.has_shape({
887 { "timestamp", json11::Json::OBJECT }, 954 { "timestamp", json11::Json::OBJECT },
888 { "inputBuffers", json11::Json::ARRAY } }, err)) { 955 { "inputBuffers", json11::Json::ARRAY } }, err)) {
889 throw Failure("malformed process request: " + err); 956 err = "malformed process request: " + err;
957 return {};
890 } 958 }
891 959
892 Vamp::HostExt::ProcessRequest r; 960 Vamp::HostExt::ProcessRequest r;
893 r.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value()); 961 r.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
894 962
895 r.timestamp = toRealTime(input["timestamp"]); 963 r.timestamp = toRealTime(input["timestamp"], err);
964 if (failed(err)) return {};
896 965
897 for (auto a: input["inputBuffers"].array_items()) { 966 for (auto a: input["inputBuffers"].array_items()) {
898 967
899 if (a["b64values"].is_string()) { 968 if (a["b64values"].is_string()) {
900 std::vector<float> buf = toFloatBuffer(a["b64values"].string_value()); 969 std::vector<float> buf = toFloatBuffer(a["b64values"].string_value(),
970 err);
971 if (failed(err)) return {};
901 r.inputBuffers.push_back(buf); 972 r.inputBuffers.push_back(buf);
902 serialisation = BufferSerialisation::Base64; 973 serialisation = BufferSerialisation::Base64;
903 974
904 } else if (a["values"].is_array()) { 975 } else if (a["values"].is_array()) {
905 std::vector<float> buf; 976 std::vector<float> buf;
908 } 979 }
909 r.inputBuffers.push_back(buf); 980 r.inputBuffers.push_back(buf);
910 serialisation = BufferSerialisation::Text; 981 serialisation = BufferSerialisation::Text;
911 982
912 } else { 983 } else {
913 throw Failure("expected values or b64values in inputBuffers object"); 984 err = "expected values or b64values in inputBuffers object";
985 return {};
914 } 986 }
915 } 987 }
916 988
917 return r; 989 return r;
918 } 990 }
1061 jo["type"] = type; 1133 jo["type"] = type;
1062 jo["success"] = false; 1134 jo["success"] = false;
1063 jo["errorText"] = std::string("error in ") + type + " request: " + errorText; 1135 jo["errorText"] = std::string("error in ") + type + " request: " + errorText;
1064 return json11::Json(jo); 1136 return json11::Json(jo);
1065 } 1137 }
1066
1067 static json11::Json
1068 fromException(const std::exception &e, RRType responseType) {
1069
1070 return fromError(e.what(), responseType);
1071 }
1072 1138
1073 private: // go private briefly for a couple of helper functions 1139 private: // go private briefly for a couple of helper functions
1074 1140
1075 static void 1141 static void
1076 checkTypeField(json11::Json j, std::string expected) { 1142 checkTypeField(json11::Json j, std::string expected, std::string &err) {
1077 if (!j["type"].is_string()) { 1143 if (!j["type"].is_string()) {
1078 throw Failure("string expected for type"); 1144 err = "string expected for type";
1145 return;
1079 } 1146 }
1080 if (j["type"].string_value() != expected) { 1147 if (j["type"].string_value() != expected) {
1081 throw Failure("expected value \"" + expected + "\" for type"); 1148 err = "expected value \"" + expected + "\" for type";
1149 return;
1082 } 1150 }
1083 } 1151 }
1084 1152
1085 static bool 1153 static bool
1086 successful(json11::Json j) { 1154 successful(json11::Json j, std::string &err) {
1087 if (!j["success"].is_bool()) { 1155 if (!j["success"].is_bool()) {
1088 throw Failure("bool expected for success"); 1156 err = "bool expected for success";
1157 return false;
1089 } 1158 }
1090 return j["success"].bool_value(); 1159 return j["success"].bool_value();
1091 } 1160 }
1092 1161
1093 public: 1162 public:
1094 static RRType 1163 static RRType
1095 getRequestResponseType(json11::Json j) { 1164 getRequestResponseType(json11::Json j, std::string &err) {
1096 1165
1097 if (!j["type"].is_string()) { 1166 if (!j["type"].is_string()) {
1098 throw Failure("string expected for type"); 1167 err = "string expected for type";
1168 return RRType::NotValid;
1099 } 1169 }
1100 1170
1101 std::string type = j["type"].string_value(); 1171 std::string type = j["type"].string_value();
1102 1172
1103 if (type == "list") return RRType::List; 1173 if (type == "list") return RRType::List;
1104 else if (type == "load") return RRType::Load; 1174 else if (type == "load") return RRType::Load;
1105 else if (type == "configure") return RRType::Configure; 1175 else if (type == "configure") return RRType::Configure;
1106 else if (type == "process") return RRType::Process; 1176 else if (type == "process") return RRType::Process;
1107 else if (type == "finish") return RRType::Finish; 1177 else if (type == "finish") return RRType::Finish;
1178 else if (type == "invalid") return RRType::NotValid;
1108 else { 1179 else {
1109 throw Failure("unknown or unexpected request/response type \"" + 1180 err = "unknown or unexpected request/response type \"" + type + "\"";
1110 type + "\""); 1181 return RRType::NotValid;
1111 } 1182 }
1112 } 1183 }
1113 1184
1114 static void 1185 static void
1115 toVampRequest_List(json11::Json j) { 1186 toVampRequest_List(json11::Json j, std::string &err) {
1116 1187 checkTypeField(j, "list", err);
1117 checkTypeField(j, "list");
1118 } 1188 }
1119 1189
1120 static Vamp::HostExt::ListResponse 1190 static Vamp::HostExt::ListResponse
1121 toVampResponse_List(json11::Json j) { 1191 toVampResponse_List(json11::Json j, std::string &err) {
1122 1192
1123 Vamp::HostExt::ListResponse resp; 1193 Vamp::HostExt::ListResponse resp;
1124 if (successful(j)) { 1194 if (successful(j, err) && !failed(err)) {
1125 for (const auto &a: j["content"]["plugins"].array_items()) { 1195 for (const auto &a: j["content"]["plugins"].array_items()) {
1126 resp.pluginData.push_back(toPluginStaticData(a)); 1196 resp.pluginData.push_back(toPluginStaticData(a, err));
1127 } 1197 if (failed(err)) return {};
1128 } 1198 }
1199 }
1200
1129 return resp; 1201 return resp;
1130 } 1202 }
1131 1203
1132 static Vamp::HostExt::LoadRequest 1204 static Vamp::HostExt::LoadRequest
1133 toVampRequest_Load(json11::Json j) { 1205 toVampRequest_Load(json11::Json j, std::string &err) {
1134 1206
1135 checkTypeField(j, "load"); 1207 checkTypeField(j, "load", err);
1136 return toLoadRequest(j["content"]); 1208 if (failed(err)) return {};
1209 return toLoadRequest(j["content"], err);
1137 } 1210 }
1138 1211
1139 static Vamp::HostExt::LoadResponse 1212 static Vamp::HostExt::LoadResponse
1140 toVampResponse_Load(json11::Json j, const PluginHandleMapper &pmapper) { 1213 toVampResponse_Load(json11::Json j,
1214 const PluginHandleMapper &pmapper,
1215 std::string &err) {
1141 1216
1142 Vamp::HostExt::LoadResponse resp; 1217 Vamp::HostExt::LoadResponse resp;
1143 if (successful(j)) { 1218 if (successful(j, err) && !failed(err)) {
1144 resp = toLoadResponse(j["content"], pmapper); 1219 resp = toLoadResponse(j["content"], pmapper, err);
1145 } 1220 }
1146 return resp; 1221 return resp;
1147 } 1222 }
1148 1223
1149 static Vamp::HostExt::ConfigurationRequest 1224 static Vamp::HostExt::ConfigurationRequest
1150 toVampRequest_Configure(json11::Json j, const PluginHandleMapper &pmapper) { 1225 toVampRequest_Configure(json11::Json j,
1151 1226 const PluginHandleMapper &pmapper,
1152 checkTypeField(j, "configure"); 1227 std::string &err) {
1153 return toConfigurationRequest(j["content"], pmapper); 1228
1229 checkTypeField(j, "configure", err);
1230 if (failed(err)) return {};
1231 return toConfigurationRequest(j["content"], pmapper, err);
1154 } 1232 }
1155 1233
1156 static Vamp::HostExt::ConfigurationResponse 1234 static Vamp::HostExt::ConfigurationResponse
1157 toVampResponse_Configure(json11::Json j, const PluginHandleMapper &pmapper) { 1235 toVampResponse_Configure(json11::Json j,
1236 const PluginHandleMapper &pmapper,
1237 std::string &err) {
1158 1238
1159 Vamp::HostExt::ConfigurationResponse resp; 1239 Vamp::HostExt::ConfigurationResponse resp;
1160 if (successful(j)) { 1240 if (successful(j, err) && !failed(err)) {
1161 resp = toConfigurationResponse(j["content"], pmapper); 1241 resp = toConfigurationResponse(j["content"], pmapper, err);
1162 } 1242 }
1163 return resp; 1243 return resp;
1164 } 1244 }
1165 1245
1166 static Vamp::HostExt::ProcessRequest 1246 static Vamp::HostExt::ProcessRequest
1167 toVampRequest_Process(json11::Json j, const PluginHandleMapper &pmapper, 1247 toVampRequest_Process(json11::Json j, const PluginHandleMapper &pmapper,
1168 BufferSerialisation &serialisation) { 1248 BufferSerialisation &serialisation, std::string &err) {
1169 1249
1170 checkTypeField(j, "process"); 1250 checkTypeField(j, "process", err);
1171 return toProcessRequest(j["content"], pmapper, serialisation); 1251 if (failed(err)) return {};
1252 return toProcessRequest(j["content"], pmapper, serialisation, err);
1172 } 1253 }
1173 1254
1174 static Vamp::HostExt::ProcessResponse 1255 static Vamp::HostExt::ProcessResponse
1175 toVampResponse_Process(json11::Json j, 1256 toVampResponse_Process(json11::Json j,
1176 const PluginHandleMapper &pmapper, 1257 const PluginHandleMapper &pmapper,
1177 BufferSerialisation &serialisation) { 1258 BufferSerialisation &serialisation, std::string &err) {
1178 1259
1179 Vamp::HostExt::ProcessResponse resp; 1260 Vamp::HostExt::ProcessResponse resp;
1180 if (successful(j)) { 1261 if (successful(j, err) && !failed(err)) {
1181 auto jc = j["content"]; 1262 auto jc = j["content"];
1182 auto h = jc["pluginHandle"].int_value(); 1263 auto h = jc["pluginHandle"].int_value();
1183 resp.plugin = pmapper.handleToPlugin(h); 1264 resp.plugin = pmapper.handleToPlugin(h);
1184 resp.features = toFeatureSet(jc["features"], 1265 resp.features = toFeatureSet(jc["features"],
1185 *pmapper.handleToOutputIdMapper(h), 1266 *pmapper.handleToOutputIdMapper(h),
1186 serialisation); 1267 serialisation, err);
1187 } 1268 }
1188 return resp; 1269 return resp;
1189 } 1270 }
1190 1271
1191 static Vamp::HostExt::FinishRequest 1272 static Vamp::HostExt::FinishRequest
1192 toVampRequest_Finish(json11::Json j, const PluginHandleMapper &pmapper) { 1273 toVampRequest_Finish(json11::Json j, const PluginHandleMapper &pmapper,
1193 1274 std::string &err) {
1194 checkTypeField(j, "finish"); 1275
1276 checkTypeField(j, "finish", err);
1277 if (failed(err)) return {};
1195 Vamp::HostExt::FinishRequest req; 1278 Vamp::HostExt::FinishRequest req;
1196 req.plugin = pmapper.handleToPlugin 1279 req.plugin = pmapper.handleToPlugin
1197 (j["content"]["pluginHandle"].int_value()); 1280 (j["content"]["pluginHandle"].int_value());
1198 return req; 1281 return req;
1199 } 1282 }
1200 1283
1201 static Vamp::HostExt::ProcessResponse 1284 static Vamp::HostExt::ProcessResponse
1202 toVampResponse_Finish(json11::Json j, 1285 toVampResponse_Finish(json11::Json j,
1203 const PluginHandleMapper &pmapper, 1286 const PluginHandleMapper &pmapper,
1204 BufferSerialisation &serialisation) { 1287 BufferSerialisation &serialisation, std::string &err) {
1205 1288
1206 Vamp::HostExt::ProcessResponse resp; 1289 Vamp::HostExt::ProcessResponse resp;
1207 if (successful(j)) { 1290 if (successful(j, err) && !failed(err)) {
1208 auto jc = j["content"]; 1291 auto jc = j["content"];
1209 auto h = jc["pluginHandle"].int_value(); 1292 auto h = jc["pluginHandle"].int_value();
1210 resp.plugin = pmapper.handleToPlugin(h); 1293 resp.plugin = pmapper.handleToPlugin(h);
1211 resp.features = toFeatureSet(jc["features"], 1294 resp.features = toFeatureSet(jc["features"],
1212 *pmapper.handleToOutputIdMapper(h), 1295 *pmapper.handleToOutputIdMapper(h),
1213 serialisation); 1296 serialisation, err);
1214 } 1297 }
1215 return resp; 1298 return resp;
1216 } 1299 }
1217 }; 1300 };
1218 1301