annotate json/VampJson.h @ 10:c8451896c40e

Some conversions for plugin handles
author Chris Cannam <c.cannam@qmul.ac.uk>
date Tue, 17 May 2016 09:58:15 +0100
parents d8358afe3f2c
children 828930f9a65d
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@5 3 #ifndef VAMP_JSON_H
c@5 4 #define VAMP_JSON_H
c@5 5
c@5 6 #include <vector>
c@5 7 #include <string>
c@5 8 #include <sstream>
c@5 9 #include <stdexcept>
c@5 10
c@5 11 #include <json11/json11.hpp>
c@5 12 #include <base-n/include/basen.hpp>
c@5 13
c@5 14 #include <vamp-hostsdk/Plugin.h>
c@5 15 #include <vamp-hostsdk/PluginLoader.h>
c@5 16
c@10 17 #include "bits/PluginHandleMapper.h"
c@10 18
c@10 19 namespace vampipe {
c@10 20
c@6 21 /**
c@6 22 * Convert the structures laid out in the Vamp SDK classes into JSON
c@6 23 * (and back again) following the schema in the vamp-json-schema
c@6 24 * project repo.
c@6 25 */
c@5 26 class VampJson
c@5 27 {
c@5 28 public:
c@5 29 class Failure : virtual public std::runtime_error {
c@5 30 public:
c@5 31 Failure(std::string s) : runtime_error(s) { }
c@5 32 };
c@5 33
c@5 34 template <typename T>
c@5 35 static json11::Json
c@5 36 fromBasicDescriptor(const T &t) {
c@5 37 return json11::Json::object {
c@5 38 { "identifier", t.identifier },
c@5 39 { "name", t.name },
c@5 40 { "description", t.description }
c@5 41 };
c@5 42 }
c@5 43
c@5 44 template <typename T>
c@5 45 static void
c@5 46 toBasicDescriptor(json11::Json j, T &t) {
c@5 47 if (!j.is_object()) {
c@5 48 throw Failure("object expected for basic descriptor content");
c@5 49 }
c@5 50 if (!j["identifier"].is_string()) {
c@5 51 throw Failure("string expected for identifier");
c@5 52 }
c@5 53 t.identifier = j["identifier"].string_value();
c@5 54 t.name = j["name"].string_value();
c@5 55 t.description = j["description"].string_value();
c@5 56 }
c@5 57
c@5 58 template <typename T>
c@5 59 static json11::Json
c@5 60 fromValueExtents(const T &t) {
c@5 61 return json11::Json::object {
c@5 62 { "min", t.minValue },
c@5 63 { "max", t.maxValue }
c@5 64 };
c@5 65 }
c@5 66
c@5 67 template <typename T>
c@5 68 static bool
c@5 69 toValueExtents(json11::Json j, T &t) {
c@5 70 if (j["extents"].is_null()) {
c@5 71 return false;
c@5 72 } else if (j["extents"].is_object()) {
c@5 73 if (j["extents"]["min"].is_number() &&
c@5 74 j["extents"]["max"].is_number()) {
c@5 75 t.minValue = j["extents"]["min"].number_value();
c@5 76 t.maxValue = j["extents"]["max"].number_value();
c@5 77 return true;
c@5 78 } else {
c@5 79 throw Failure("numbers expected for min and max");
c@5 80 }
c@5 81 } else {
c@5 82 throw Failure("object expected for extents (if present)");
c@5 83 }
c@5 84 }
c@5 85
c@5 86 static json11::Json
c@5 87 fromRealTime(const Vamp::RealTime &r) {
c@5 88 return json11::Json::object {
c@5 89 { "s", r.sec },
c@5 90 { "n", r.nsec }
c@5 91 };
c@5 92 }
c@5 93
c@5 94 static Vamp::RealTime
c@5 95 toRealTime(json11::Json j) {
c@5 96 json11::Json sec = j["s"];
c@5 97 json11::Json nsec = j["n"];
c@5 98 if (!sec.is_number() || !nsec.is_number()) {
c@5 99 throw Failure("invalid Vamp::RealTime object " + j.dump());
c@5 100 }
c@5 101 return Vamp::RealTime(sec.int_value(), nsec.int_value());
c@5 102 }
c@5 103
c@5 104 static std::string
c@5 105 fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType type) {
c@5 106 switch (type) {
c@5 107 case Vamp::Plugin::OutputDescriptor::OneSamplePerStep:
c@5 108 return "OneSamplePerStep";
c@5 109 case Vamp::Plugin::OutputDescriptor::FixedSampleRate:
c@5 110 return "FixedSampleRate";
c@5 111 case Vamp::Plugin::OutputDescriptor::VariableSampleRate:
c@5 112 return "VariableSampleRate";
c@5 113 }
c@5 114 return "";
c@5 115 }
c@5 116
c@5 117 static Vamp::Plugin::OutputDescriptor::SampleType
c@5 118 toSampleType(std::string text) {
c@5 119 if (text == "OneSamplePerStep") {
c@5 120 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
c@5 121 } else if (text == "FixedSampleRate") {
c@5 122 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
c@5 123 } else if (text == "VariableSampleRate") {
c@5 124 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
c@5 125 } else {
c@5 126 throw Failure("invalid sample type string: " + text);
c@5 127 }
c@5 128 }
c@5 129
c@5 130 static json11::Json
c@5 131 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) {
c@5 132 json11::Json::object jo {
c@5 133 { "basic", fromBasicDescriptor(desc) },
c@5 134 { "unit", desc.unit },
c@5 135 { "sampleType", fromSampleType(desc.sampleType) },
c@5 136 { "sampleRate", desc.sampleRate },
c@5 137 { "hasDuration", desc.hasDuration }
c@5 138 };
c@5 139 if (desc.hasFixedBinCount) {
c@5 140 jo["binCount"] = int(desc.binCount);
c@5 141 jo["binNames"] = json11::Json::array
c@5 142 (desc.binNames.begin(), desc.binNames.end());
c@5 143 }
c@5 144 if (desc.hasKnownExtents) {
c@5 145 jo["extents"] = fromValueExtents(desc);
c@5 146 }
c@5 147 if (desc.isQuantized) {
c@5 148 jo["quantizeStep"] = desc.quantizeStep;
c@5 149 }
c@5 150 return json11::Json(jo);
c@5 151 }
c@5 152
c@5 153 static Vamp::Plugin::OutputDescriptor
c@5 154 toOutputDescriptor(json11::Json j) {
c@5 155
c@5 156 Vamp::Plugin::OutputDescriptor od;
c@5 157 if (!j.is_object()) {
c@5 158 throw Failure("object expected for output descriptor");
c@5 159 }
c@5 160
c@5 161 toBasicDescriptor(j["basic"], od);
c@5 162
c@5 163 od.unit = j["unit"].string_value();
c@5 164
c@5 165 od.sampleType = toSampleType(j["sampleType"].string_value());
c@5 166
c@5 167 if (!j["sampleRate"].is_number()) {
c@5 168 throw Failure("number expected for sample rate");
c@5 169 }
c@5 170 od.sampleRate = j["sampleRate"].number_value();
c@5 171 od.hasDuration = j["hasDuration"].bool_value();
c@5 172
c@5 173 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) {
c@5 174 od.hasFixedBinCount = true;
c@5 175 od.binCount = j["binCount"].int_value();
c@5 176 for (auto &n: j["binNames"].array_items()) {
c@5 177 if (!n.is_string()) {
c@5 178 throw Failure("string expected for bin name");
c@5 179 }
c@5 180 od.binNames.push_back(n.string_value());
c@5 181 }
c@5 182 } else {
c@5 183 od.hasFixedBinCount = false;
c@5 184 }
c@5 185
c@5 186 bool extentsPresent = toValueExtents(j, od);
c@5 187 od.hasKnownExtents = extentsPresent;
c@5 188
c@5 189 if (j["quantizeStep"].is_number()) {
c@5 190 od.isQuantized = true;
c@5 191 od.quantizeStep = j["quantizeStep"].number_value();
c@5 192 } else {
c@5 193 od.isQuantized = false;
c@5 194 }
c@5 195
c@5 196 return od;
c@5 197 }
c@5 198
c@5 199 static json11::Json
c@5 200 fromParameterDescriptor(const Vamp::PluginBase::ParameterDescriptor &desc) {
c@5 201
c@5 202 json11::Json::object jo {
c@5 203 { "basic", fromBasicDescriptor(desc) },
c@5 204 { "unit", desc.unit },
c@5 205 { "extents", fromValueExtents(desc) },
c@5 206 { "defaultValue", desc.defaultValue },
c@5 207 { "valueNames", json11::Json::array
c@5 208 (desc.valueNames.begin(), desc.valueNames.end()) }
c@5 209 };
c@5 210 if (desc.isQuantized) {
c@5 211 jo["quantizeStep"] = desc.quantizeStep;
c@5 212 }
c@5 213 return json11::Json(jo);
c@5 214 }
c@5 215
c@5 216 static Vamp::PluginBase::ParameterDescriptor
c@5 217 toParameterDescriptor(json11::Json j) {
c@5 218
c@5 219 Vamp::PluginBase::ParameterDescriptor pd;
c@5 220 if (!j.is_object()) {
c@5 221 throw Failure("object expected for parameter descriptor");
c@5 222 }
c@5 223
c@5 224 toBasicDescriptor(j["basic"], pd);
c@5 225
c@5 226 pd.unit = j["unit"].string_value();
c@5 227
c@5 228 bool extentsPresent = toValueExtents(j, pd);
c@5 229 if (!extentsPresent) {
c@5 230 throw Failure("extents must be present in parameter descriptor");
c@5 231 }
c@5 232
c@5 233 if (!j["defaultValue"].is_number()) {
c@5 234 throw Failure("number expected for default value");
c@5 235 }
c@5 236
c@5 237 pd.defaultValue = j["defaultValue"].number_value();
c@5 238
c@5 239 pd.valueNames.clear();
c@5 240 for (auto &n: j["valueNames"].array_items()) {
c@5 241 if (!n.is_string()) {
c@5 242 throw Failure("string expected for value name");
c@5 243 }
c@5 244 pd.valueNames.push_back(n.string_value());
c@5 245 }
c@5 246
c@5 247 if (j["quantizeStep"].is_number()) {
c@5 248 pd.isQuantized = true;
c@5 249 pd.quantizeStep = j["quantizeStep"].number_value();
c@5 250 } else {
c@5 251 pd.isQuantized = false;
c@5 252 }
c@5 253
c@5 254 return pd;
c@5 255 }
c@5 256
c@5 257 static std::string
c@5 258 fromFloatBuffer(const float *buffer, size_t nfloats) {
c@5 259 // must use char pointers, otherwise the converter will only
c@5 260 // encode every 4th byte (as it will count up in float* steps)
c@5 261 const char *start = reinterpret_cast<const char *>(buffer);
c@5 262 const char *end = reinterpret_cast<const char *>(buffer + nfloats);
c@5 263 std::string encoded;
c@5 264 bn::encode_b64(start, end, back_inserter(encoded));
c@5 265 return encoded;
c@5 266 }
c@5 267
c@5 268 static std::vector<float>
c@5 269 toFloatBuffer(std::string encoded) {
c@5 270 std::string decoded;
c@5 271 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded));
c@5 272 const float *buffer = reinterpret_cast<const float *>(decoded.c_str());
c@5 273 size_t n = decoded.size() / sizeof(float);
c@5 274 return std::vector<float>(buffer, buffer + n);
c@5 275 }
c@5 276
c@5 277 static json11::Json
c@5 278 fromFeature(const Vamp::Plugin::Feature &f) {
c@5 279
c@5 280 json11::Json::object jo;
c@5 281 if (f.values.size() > 0) {
c@5 282 jo["b64values"] = fromFloatBuffer(f.values.data(), f.values.size());
c@5 283 }
c@5 284 if (f.label != "") {
c@5 285 jo["label"] = f.label;
c@5 286 }
c@5 287 if (f.hasTimestamp) {
c@5 288 jo["timestamp"] = fromRealTime(f.timestamp);
c@5 289 }
c@5 290 if (f.hasDuration) {
c@5 291 jo["duration"] = fromRealTime(f.duration);
c@5 292 }
c@5 293 return json11::Json(jo);
c@5 294 }
c@5 295
c@5 296 static Vamp::Plugin::Feature
c@5 297 toFeature(json11::Json j) {
c@5 298
c@5 299 Vamp::Plugin::Feature f;
c@5 300 if (!j.is_object()) {
c@5 301 throw Failure("object expected for feature");
c@5 302 }
c@5 303 if (j["timestamp"].is_object()) {
c@5 304 f.timestamp = toRealTime(j["timestamp"]);
c@5 305 f.hasTimestamp = true;
c@5 306 }
c@5 307 if (j["duration"].is_object()) {
c@5 308 f.duration = toRealTime(j["duration"]);
c@5 309 f.hasDuration = true;
c@5 310 }
c@5 311 if (j["b64values"].is_string()) {
c@5 312 f.values = toFloatBuffer(j["b64values"].string_value());
c@5 313 } else if (j["values"].is_array()) {
c@5 314 for (auto v : j["values"].array_items()) {
c@5 315 f.values.push_back(v.number_value());
c@5 316 }
c@5 317 }
c@5 318 f.label = j["label"].string_value();
c@5 319 return f;
c@5 320 }
c@5 321
c@5 322 static json11::Json
c@5 323 fromFeatureSet(const Vamp::Plugin::FeatureSet &fs) {
c@5 324
c@5 325 json11::Json::object jo;
c@5 326 for (const auto &fsi : fs) {
c@5 327 std::vector<json11::Json> fj;
c@5 328 for (const Vamp::Plugin::Feature &f: fsi.second) {
c@5 329 fj.push_back(fromFeature(f));
c@5 330 }
c@5 331 std::stringstream sstr;
c@5 332 sstr << fsi.first;
c@5 333 std::string n = sstr.str();
c@5 334 jo[n] = fj;
c@5 335 }
c@5 336 return json11::Json(jo);
c@5 337 }
c@5 338
c@5 339 static Vamp::Plugin::FeatureList
c@5 340 toFeatureList(json11::Json j) {
c@5 341
c@5 342 Vamp::Plugin::FeatureList fl;
c@5 343 if (!j.is_array()) {
c@5 344 throw Failure("array expected for feature list");
c@5 345 }
c@5 346 for (const json11::Json &fj : j.array_items()) {
c@5 347 fl.push_back(toFeature(fj));
c@5 348 }
c@5 349 return fl;
c@5 350 }
c@5 351
c@5 352 static Vamp::Plugin::FeatureSet
c@5 353 toFeatureSet(json11::Json j) {
c@5 354
c@5 355 Vamp::Plugin::FeatureSet fs;
c@5 356 if (!j.is_object()) {
c@5 357 throw Failure("object expected for feature set");
c@5 358 }
c@5 359 for (auto &entry : j.object_items()) {
c@5 360 std::string nstr = entry.first;
c@5 361 size_t count = 0;
c@5 362 int n = stoi(nstr, &count);
c@5 363 if (n < 0 || fs.find(n) != fs.end() || count < nstr.size()) {
c@5 364 throw Failure("invalid or duplicate numerical index for output");
c@5 365 }
c@5 366 fs[n] = toFeatureList(entry.second);
c@5 367 }
c@5 368 return fs;
c@5 369 }
c@5 370
c@5 371 static std::string
c@5 372 fromInputDomain(Vamp::Plugin::InputDomain domain) {
c@5 373
c@5 374 switch (domain) {
c@5 375 case Vamp::Plugin::TimeDomain:
c@5 376 return "TimeDomain";
c@5 377 case Vamp::Plugin::FrequencyDomain:
c@5 378 return "FrequencyDomain";
c@5 379 }
c@5 380 return "";
c@5 381 }
c@5 382
c@5 383 static Vamp::Plugin::InputDomain
c@5 384 toInputDomain(std::string text) {
c@5 385
c@5 386 if (text == "TimeDomain") {
c@5 387 return Vamp::Plugin::TimeDomain;
c@5 388 } else if (text == "FrequencyDomain") {
c@5 389 return Vamp::Plugin::FrequencyDomain;
c@5 390 } else {
c@5 391 throw Failure("invalid input domain string: " + text);
c@5 392 }
c@5 393 }
c@5 394
c@5 395 static json11::Json
c@5 396 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) {
c@5 397
c@5 398 json11::Json::object jo;
c@5 399 jo["pluginKey"] = d.pluginKey;
c@5 400 jo["basic"] = fromBasicDescriptor(d.basic);
c@5 401 jo["maker"] = d.maker;
c@5 402 jo["copyright"] = d.copyright;
c@5 403 jo["pluginVersion"] = d.pluginVersion;
c@5 404
c@5 405 json11::Json::array cat;
c@5 406 for (const std::string &c: d.category) cat.push_back(c);
c@5 407 jo["category"] = cat;
c@5 408
c@5 409 jo["minChannelCount"] = d.minChannelCount;
c@5 410 jo["maxChannelCount"] = d.maxChannelCount;
c@5 411
c@5 412 json11::Json::array params;
c@5 413 Vamp::PluginBase::ParameterList vparams = d.parameters;
c@5 414 for (auto &p: vparams) params.push_back(fromParameterDescriptor(p));
c@5 415 jo["parameters"] = params;
c@5 416
c@5 417 json11::Json::array progs;
c@5 418 Vamp::PluginBase::ProgramList vprogs = d.programs;
c@5 419 for (auto &p: vprogs) progs.push_back(p);
c@5 420 jo["programs"] = progs;
c@5 421
c@5 422 jo["inputDomain"] = fromInputDomain(d.inputDomain);
c@5 423
c@5 424 json11::Json::array outinfo;
c@5 425 auto vouts = d.basicOutputInfo;
c@5 426 for (auto &o: vouts) outinfo.push_back(fromBasicDescriptor(o));
c@5 427 jo["basicOutputInfo"] = outinfo;
c@5 428
c@5 429 return json11::Json(jo);
c@5 430 }
c@5 431
c@5 432 static Vamp::HostExt::PluginStaticData
c@5 433 toPluginStaticData(json11::Json j) {
c@5 434
c@5 435 std::string err;
c@5 436 if (!j.has_shape({
c@5 437 { "pluginKey", json11::Json::STRING },
c@5 438 { "pluginVersion", json11::Json::NUMBER },
c@5 439 { "minChannelCount", json11::Json::NUMBER },
c@5 440 { "maxChannelCount", json11::Json::NUMBER },
c@5 441 { "inputDomain", json11::Json::STRING }}, err)) {
c@5 442 throw Failure("malformed plugin static data: " + err);
c@5 443 }
c@5 444
c@5 445 if (!j["basicOutputInfo"].is_array()) {
c@5 446 throw Failure("array expected for basic output info");
c@5 447 }
c@5 448
c@5 449 if (!j["maker"].is_null() &&
c@5 450 !j["maker"].is_string()) {
c@5 451 throw Failure("string expected for maker");
c@5 452 }
c@5 453
c@5 454 if (!j["copyright"].is_null() &&
c@5 455 !j["copyright"].is_string()) {
c@5 456 throw Failure("string expected for copyright");
c@5 457 }
c@5 458
c@5 459 if (!j["category"].is_null() &&
c@5 460 !j["category"].is_array()) {
c@5 461 throw Failure("array expected for category");
c@5 462 }
c@5 463
c@5 464 if (!j["parameters"].is_null() &&
c@5 465 !j["parameters"].is_array()) {
c@5 466 throw Failure("array expected for parameters");
c@5 467 }
c@5 468
c@5 469 if (!j["programs"].is_null() &&
c@5 470 !j["programs"].is_array()) {
c@5 471 throw Failure("array expected for programs");
c@5 472 }
c@5 473
c@5 474 if (!j["inputDomain"].is_null() &&
c@5 475 !j["inputDomain"].is_string()) {
c@5 476 throw Failure("string expected for inputDomain");
c@5 477 }
c@5 478
c@5 479 if (!j["basicOutputInfo"].is_null() &&
c@5 480 !j["basicOutputInfo"].is_array()) {
c@5 481 throw Failure("array expected for basicOutputInfo");
c@5 482 }
c@5 483
c@5 484 Vamp::HostExt::PluginStaticData psd;
c@5 485
c@5 486 psd.pluginKey = j["pluginKey"].string_value();
c@5 487
c@5 488 toBasicDescriptor(j["basic"], psd.basic);
c@5 489
c@5 490 psd.maker = j["maker"].string_value();
c@5 491 psd.copyright = j["copyright"].string_value();
c@5 492 psd.pluginVersion = j["pluginVersion"].int_value();
c@5 493
c@5 494 for (const auto &c : j["category"].array_items()) {
c@5 495 if (!c.is_string()) {
c@5 496 throw Failure("strings expected in category array");
c@5 497 }
c@5 498 psd.category.push_back(c.string_value());
c@5 499 }
c@5 500
c@5 501 psd.minChannelCount = j["minChannelCount"].int_value();
c@5 502 psd.maxChannelCount = j["maxChannelCount"].int_value();
c@5 503
c@5 504 for (const auto &p : j["parameters"].array_items()) {
c@5 505 auto pd = toParameterDescriptor(p);
c@5 506 psd.parameters.push_back(pd);
c@5 507 }
c@5 508
c@5 509 for (const auto &p : j["programs"].array_items()) {
c@5 510 if (!p.is_string()) {
c@5 511 throw Failure("strings expected in programs array");
c@5 512 }
c@5 513 psd.programs.push_back(p.string_value());
c@5 514 }
c@5 515
c@5 516 psd.inputDomain = toInputDomain(j["inputDomain"].string_value());
c@5 517
c@5 518 for (const auto &bo : j["basicOutputInfo"].array_items()) {
c@5 519 Vamp::HostExt::PluginStaticData::Basic b;
c@5 520 toBasicDescriptor(bo, b);
c@5 521 psd.basicOutputInfo.push_back(b);
c@5 522 }
c@5 523
c@5 524 return psd;
c@5 525 }
c@5 526
c@5 527 static json11::Json
c@5 528 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) {
c@5 529
c@5 530 json11::Json::object jo;
c@5 531
c@5 532 json11::Json::object paramValues;
c@5 533 for (auto &vp: c.parameterValues) {
c@5 534 paramValues[vp.first] = vp.second;
c@5 535 }
c@5 536 jo["parameterValues"] = paramValues;
c@5 537
c@5 538 if (c.currentProgram != "") {
c@5 539 jo["currentProgram"] = c.currentProgram;
c@5 540 }
c@5 541
c@5 542 jo["channelCount"] = c.channelCount;
c@5 543 jo["stepSize"] = c.stepSize;
c@5 544 jo["blockSize"] = c.blockSize;
c@5 545
c@5 546 return json11::Json(jo);
c@5 547 }
c@5 548
c@5 549 static Vamp::HostExt::PluginConfiguration
c@5 550 toPluginConfiguration(json11::Json j) {
c@5 551
c@5 552 std::string err;
c@5 553 if (!j.has_shape({
c@5 554 { "channelCount", json11::Json::NUMBER },
c@5 555 { "stepSize", json11::Json::NUMBER },
c@5 556 { "blockSize", json11::Json::NUMBER } }, err)) {
c@5 557 throw Failure("malformed plugin configuration: " + err);
c@5 558 }
c@5 559
c@5 560 if (!j["parameterValues"].is_null() &&
c@5 561 !j["parameterValues"].is_object()) {
c@5 562 throw Failure("object expected for parameter values");
c@5 563 }
c@5 564
c@5 565 for (auto &pv : j["parameterValues"].object_items()) {
c@5 566 if (!pv.second.is_number()) {
c@5 567 throw Failure("number expected for parameter value");
c@5 568 }
c@5 569 }
c@5 570
c@5 571 if (!j["currentProgram"].is_null() &&
c@5 572 !j["currentProgram"].is_string()) {
c@5 573 throw Failure("string expected for program name");
c@5 574 }
c@5 575
c@5 576 Vamp::HostExt::PluginConfiguration config;
c@5 577
c@5 578 config.channelCount = j["channelCount"].number_value();
c@5 579 config.stepSize = j["stepSize"].number_value();
c@5 580 config.blockSize = j["blockSize"].number_value();
c@5 581
c@5 582 for (auto &pv : j["parameterValues"].object_items()) {
c@5 583 config.parameterValues[pv.first] = pv.second.number_value();
c@5 584 }
c@5 585
c@5 586 if (j["currentProgram"].is_string()) {
c@5 587 config.currentProgram = j["currentProgram"].string_value();
c@5 588 }
c@5 589
c@5 590 return config;
c@5 591 }
c@5 592
c@5 593 static json11::Json
c@5 594 fromAdapterFlags(int flags) {
c@5 595
c@5 596 json11::Json::array arr;
c@5 597
c@5 598 if (flags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
c@5 599 arr.push_back("AdaptInputDomain");
c@5 600 }
c@5 601 if (flags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
c@5 602 arr.push_back("AdaptChannelCount");
c@5 603 }
c@5 604 if (flags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
c@5 605 arr.push_back("AdaptBufferSize");
c@5 606 }
c@5 607
c@5 608 return json11::Json(arr);
c@5 609 }
c@5 610
c@5 611 static Vamp::HostExt::PluginLoader::AdapterFlags
c@5 612 toAdapterFlags(json11::Json j) {
c@5 613
c@5 614 if (!j.is_array()) {
c@5 615 throw Failure("array expected for adapter flags");
c@5 616 }
c@5 617 int flags = 0x0;
c@5 618
c@5 619 for (auto &jj: j.array_items()) {
c@5 620 if (!jj.is_string()) {
c@5 621 throw Failure("string expected for adapter flag");
c@5 622 }
c@5 623 std::string text = jj.string_value();
c@5 624 if (text == "AdaptInputDomain") {
c@5 625 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
c@5 626 } else if (text == "AdaptChannelCount") {
c@5 627 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
c@5 628 } else if (text == "AdaptBufferSize") {
c@5 629 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
c@5 630 } else if (text == "AdaptAllSafe") {
c@5 631 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE;
c@5 632 } else if (text == "AdaptAll") {
c@5 633 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL;
c@5 634 } else {
c@5 635 throw Failure("invalid adapter flag string: " + text);
c@5 636 }
c@5 637 }
c@5 638
c@5 639 return Vamp::HostExt::PluginLoader::AdapterFlags(flags);
c@5 640 }
c@5 641
c@5 642 static json11::Json
c@5 643 fromLoadRequest(Vamp::HostExt::LoadRequest req) {
c@5 644
c@5 645 json11::Json::object jo;
c@5 646 jo["pluginKey"] = req.pluginKey;
c@5 647 jo["inputSampleRate"] = req.inputSampleRate;
c@5 648 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags);
c@5 649 return json11::Json(jo);
c@5 650 }
c@5 651
c@5 652 static Vamp::HostExt::LoadRequest
c@5 653 toLoadRequest(json11::Json j) {
c@5 654
c@5 655 std::string err;
c@5 656
c@5 657 if (!j.has_shape({
c@5 658 { "pluginKey", json11::Json::STRING },
c@5 659 { "inputSampleRate", json11::Json::NUMBER },
c@5 660 { "adapterFlags", json11::Json::ARRAY } }, err)) {
c@5 661 throw VampJson::Failure("malformed load request: " + err);
c@5 662 }
c@5 663
c@5 664 Vamp::HostExt::LoadRequest req;
c@5 665 req.pluginKey = j["pluginKey"].string_value();
c@5 666 req.inputSampleRate = j["inputSampleRate"].number_value();
c@5 667 req.adapterFlags = toAdapterFlags(j["adapterFlags"]);
c@5 668 return req;
c@5 669 }
c@10 670
c@10 671 static json11::Json
c@10 672 fromLoadResponse(Vamp::HostExt::LoadResponse resp,
c@10 673 PluginHandleMapper &mapper) {
c@10 674
c@10 675 json11::Json::object jo;
c@10 676 jo["pluginHandle"] = double(mapper.pluginToHandle(resp.plugin));
c@10 677 jo["staticData"] = fromPluginStaticData(resp.staticData);
c@10 678 jo["defaultConfiguration"] =
c@10 679 fromPluginConfiguration(resp.defaultConfiguration);
c@10 680 return json11::Json(jo);
c@10 681 }
c@10 682
c@10 683 static Vamp::HostExt::LoadResponse
c@10 684 toLoadResponse(json11::Json j,
c@10 685 PluginHandleMapper &mapper) {
c@10 686
c@10 687 std::string err;
c@10 688
c@10 689 if (!j.has_shape({
c@10 690 { "pluginHandle", json11::Json::NUMBER },
c@10 691 { "staticData", json11::Json::OBJECT },
c@10 692 { "defaultConfiguration", json11::Json::OBJECT } }, err)) {
c@10 693 throw VampJson::Failure("malformed load response: " + err);
c@10 694 }
c@10 695
c@10 696 Vamp::HostExt::LoadResponse resp;
c@10 697 resp.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
c@10 698 resp.staticData = toPluginStaticData(j["staticData"]);
c@10 699 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"]);
c@10 700 return resp;
c@10 701 }
c@5 702 };
c@5 703
c@10 704 }
c@5 705
c@5 706 #endif