comparison json/VampJson.h @ 66:6f160dee1192

Instead of using separate values and b64values entries in JSON serialisations, allow numeric arrays to be replaced by b64 variants wherever they appear (discriminating by type). Also rename values to featureValues in feature throughout, as values turns out to be a hazardous name in a JS context. Finally use Array instead of Text for array encoding (seems clearer).
author Chris Cannam <c.cannam@qmul.ac.uk>
date Tue, 27 Sep 2016 15:04:59 +0100
parents 2d866edd79d5
children db17657ac875
comparison
equal deleted inserted replaced
65:2d866edd79d5 66:6f160dee1192
76 76
77 class VampJson 77 class VampJson
78 { 78 {
79 public: 79 public:
80 /** Serialisation format for arrays of floats (process input and 80 /** Serialisation format for arrays of floats (process input and
81 * feature values). Structures that can be serialised in more 81 * feature values). Wherever such an array appears, it may
82 * than one way will include either a "values" field (for Text 82 * alternatively be replaced by a single string containing a
83 * serialisation) or a "b64values" field (for Base64) but should 83 * base-64 encoding of the IEEE float buffer. When parsing, if a
84 * not include both. When parsing, if a "b64values" field is 84 * string is found instead of an array in this case, it will be
85 * found, it will always take priority over a "values" field. 85 * interpreted as a base-64 encoded buffer. Only array or base-64
86 * encoding may be provided, not both.
86 */ 87 */
87 enum class BufferSerialisation { 88 enum class BufferSerialisation {
88 89
89 /** Default JSON serialisation of values in array form. This 90 /** Default JSON serialisation of values in array form. This
90 * is relatively slow to parse and serialise, and can take a 91 * is relatively slow to parse and serialise, and can take a
91 * lot of space. 92 * lot of space.
92 */ 93 */
93 Text, 94 Array,
94 95
95 /** Base64-encoded string of the raw data as packed IEEE 96 /** Base64-encoded string of the raw data as packed
96 * 32-bit floats. Faster and more compact than the text 97 * little-endian IEEE 32-bit floats. Faster and more compact
97 * encoding but more complicated to provide, especially if 98 * than the text encoding but more complicated to
98 * starting from an environment that does not use IEEE 32-bit 99 * provide. Note that Base64 serialisations produced by this
99 * floats! Note that Base64 serialisations produced by this
100 * library do not including padding characters and so are not 100 * library do not including padding characters and so are not
101 * necessarily multiples of 4 characters long. You will need 101 * necessarily multiples of 4 characters long. You will need
102 * to pad them yourself if concatenating them or supplying to 102 * to pad them yourself if concatenating them or supplying to
103 * a consumer that expects padding. 103 * a consumer that expects padding.
104 */ 104 */
398 fromFeature(const Vamp::Plugin::Feature &f, 398 fromFeature(const Vamp::Plugin::Feature &f,
399 BufferSerialisation serialisation) { 399 BufferSerialisation serialisation) {
400 400
401 json11::Json::object jo; 401 json11::Json::object jo;
402 if (f.values.size() > 0) { 402 if (f.values.size() > 0) {
403 if (serialisation == BufferSerialisation::Text) { 403 if (serialisation == BufferSerialisation::Array) {
404 jo["values"] = json11::Json::array(f.values.begin(), 404 jo["featureValues"] = json11::Json::array(f.values.begin(),
405 f.values.end()); 405 f.values.end());
406 } else { 406 } else {
407 jo["b64values"] = fromFloatBuffer(f.values.data(), 407 jo["featureValues"] = fromFloatBuffer(f.values.data(),
408 f.values.size()); 408 f.values.size());
409 } 409 }
410 } 410 }
411 if (f.label != "") { 411 if (f.label != "") {
412 jo["label"] = f.label; 412 jo["label"] = f.label;
413 } 413 }
436 if (j["duration"].is_object()) { 436 if (j["duration"].is_object()) {
437 f.duration = toRealTime(j["duration"], err); 437 f.duration = toRealTime(j["duration"], err);
438 if (failed(err)) return {}; 438 if (failed(err)) return {};
439 f.hasDuration = true; 439 f.hasDuration = true;
440 } 440 }
441 if (j["b64values"].is_string()) { 441 if (j["featureValues"].is_string()) {
442 f.values = toFloatBuffer(j["b64values"].string_value(), err); 442 f.values = toFloatBuffer(j["featureValues"].string_value(), err);
443 if (failed(err)) return {}; 443 if (failed(err)) return {};
444 serialisation = BufferSerialisation::Base64; 444 serialisation = BufferSerialisation::Base64;
445 } else if (j["values"].is_array()) { 445 } else if (j["featureValues"].is_array()) {
446 for (auto v : j["values"].array_items()) { 446 for (auto v : j["featureValues"].array_items()) {
447 f.values.push_back(v.number_value()); 447 f.values.push_back(v.number_value());
448 } 448 }
449 serialisation = BufferSerialisation::Text; 449 serialisation = BufferSerialisation::Array;
450 } 450 }
451 f.label = j["label"].string_value(); 451 f.label = j["label"].string_value();
452 return f; 452 return f;
453 } 453 }
454 454
941 json11::Json::object io; 941 json11::Json::object io;
942 io["timestamp"] = fromRealTime(r.timestamp); 942 io["timestamp"] = fromRealTime(r.timestamp);
943 943
944 json11::Json::array chans; 944 json11::Json::array chans;
945 for (size_t i = 0; i < r.inputBuffers.size(); ++i) { 945 for (size_t i = 0; i < r.inputBuffers.size(); ++i) {
946 json11::Json::object c; 946 if (serialisation == BufferSerialisation::Array) {
947 if (serialisation == BufferSerialisation::Text) { 947 chans.push_back(json11::Json::array(r.inputBuffers[i].begin(),
948 c["values"] = json11::Json::array(r.inputBuffers[i].begin(), 948 r.inputBuffers[i].end()));
949 r.inputBuffers[i].end());
950 } else { 949 } else {
951 c["b64values"] = fromFloatBuffer(r.inputBuffers[i].data(), 950 chans.push_back(fromFloatBuffer(r.inputBuffers[i].data(),
952 r.inputBuffers[i].size()); 951 r.inputBuffers[i].size()));
953 } 952 }
954 chans.push_back(c);
955 } 953 }
956 io["inputBuffers"] = chans; 954 io["inputBuffers"] = chans;
957 955
958 jo["processInput"] = io; 956 jo["processInput"] = io;
959 return json11::Json(jo); 957 return json11::Json(jo);
984 r.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value()); 982 r.plugin = pmapper.handleToPlugin(j["pluginHandle"].int_value());
985 983
986 r.timestamp = toRealTime(input["timestamp"], err); 984 r.timestamp = toRealTime(input["timestamp"], err);
987 if (failed(err)) return {}; 985 if (failed(err)) return {};
988 986
989 for (auto a: input["inputBuffers"].array_items()) { 987 for (const auto &a: input["inputBuffers"].array_items()) {
990 988
991 if (a["b64values"].is_string()) { 989 if (a.is_string()) {
992 std::vector<float> buf = toFloatBuffer(a["b64values"].string_value(), 990 std::vector<float> buf = toFloatBuffer(a.string_value(),
993 err); 991 err);
994 if (failed(err)) return {}; 992 if (failed(err)) return {};
995 r.inputBuffers.push_back(buf); 993 r.inputBuffers.push_back(buf);
996 serialisation = BufferSerialisation::Base64; 994 serialisation = BufferSerialisation::Base64;
997 995
998 } else if (a["values"].is_array()) { 996 } else if (a.is_array()) {
999 std::vector<float> buf; 997 std::vector<float> buf;
1000 for (auto v : a["values"].array_items()) { 998 for (auto v : a.array_items()) {
1001 buf.push_back(v.number_value()); 999 buf.push_back(v.number_value());
1002 } 1000 }
1003 r.inputBuffers.push_back(buf); 1001 r.inputBuffers.push_back(buf);
1004 serialisation = BufferSerialisation::Text; 1002 serialisation = BufferSerialisation::Array;
1005 1003
1006 } else { 1004 } else {
1007 err = "expected values or b64values in inputBuffers object"; 1005 err = "expected arrays or strings in inputBuffers array";
1008 return {}; 1006 return {};
1009 } 1007 }
1010 } 1008 }
1011 1009
1012 return r; 1010 return r;