annotate vamp-json/VampJson.h @ 218:ea8994465322

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