annotate json/VampJson.h @ 39:183be3bc9d7b

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