annotate json/VampJson.h @ 73:7bfc07576830

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