annotate json/VampJson.h @ 24:533ca5ca3404

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