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