annotate json/VampJson.h @ 64:85ec33975434

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