Mercurial > hg > piper-cpp
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; |