c@5
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
c@5
|
2
|
c@18
|
3 /*
|
c@18
|
4 VamPipe
|
c@18
|
5
|
c@18
|
6 Centre for Digital Music, Queen Mary, University of London.
|
c@18
|
7 Copyright 2015-2016 QMUL.
|
c@18
|
8
|
c@18
|
9 Permission is hereby granted, free of charge, to any person
|
c@18
|
10 obtaining a copy of this software and associated documentation
|
c@18
|
11 files (the "Software"), to deal in the Software without
|
c@18
|
12 restriction, including without limitation the rights to use, copy,
|
c@18
|
13 modify, merge, publish, distribute, sublicense, and/or sell copies
|
c@18
|
14 of the Software, and to permit persons to whom the Software is
|
c@18
|
15 furnished to do so, subject to the following conditions:
|
c@18
|
16
|
c@18
|
17 The above copyright notice and this permission notice shall be
|
c@18
|
18 included in all copies or substantial portions of the Software.
|
c@18
|
19
|
c@18
|
20 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
c@18
|
21 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
c@18
|
22 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
c@18
|
23 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
c@18
|
24 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
c@18
|
25 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
c@18
|
26 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
c@18
|
27
|
c@18
|
28 Except as contained in this notice, the names of the Centre for
|
c@18
|
29 Digital Music; Queen Mary, University of London; and Chris Cannam
|
c@18
|
30 shall not be used in advertising or otherwise to promote the sale,
|
c@18
|
31 use or other dealings in this Software without prior written
|
c@18
|
32 authorization.
|
c@18
|
33 */
|
c@18
|
34
|
c@5
|
35 #ifndef VAMP_JSON_H
|
c@5
|
36 #define VAMP_JSON_H
|
c@5
|
37
|
c@5
|
38 #include <vector>
|
c@5
|
39 #include <string>
|
c@5
|
40 #include <sstream>
|
c@5
|
41 #include <stdexcept>
|
c@5
|
42
|
c@5
|
43 #include <json11/json11.hpp>
|
c@5
|
44 #include <base-n/include/basen.hpp>
|
c@5
|
45
|
c@5
|
46 #include <vamp-hostsdk/Plugin.h>
|
c@5
|
47 #include <vamp-hostsdk/PluginLoader.h>
|
c@5
|
48
|
c@10
|
49 #include "bits/PluginHandleMapper.h"
|
c@25
|
50 #include "bits/RequestResponseType.h"
|
c@10
|
51
|
c@46
|
52 #include <chrono>
|
c@46
|
53
|
c@10
|
54 namespace vampipe {
|
c@10
|
55
|
c@6
|
56 /**
|
c@6
|
57 * Convert the structures laid out in the Vamp SDK classes into JSON
|
c@6
|
58 * (and back again) following the schema in the vamp-json-schema
|
c@6
|
59 * project repo.
|
c@6
|
60 */
|
c@5
|
61 class VampJson
|
c@5
|
62 {
|
c@5
|
63 public:
|
c@45
|
64 /** Serialisation format for arrays of floats (process input and
|
c@45
|
65 * feature values). Structures that can be serialised in more
|
c@45
|
66 * than one way will include either a "values" field (for Text
|
c@45
|
67 * serialisation) or a "b64values" field (for Base64) but should
|
c@45
|
68 * not include both. When parsing, if a "b64values" field is
|
c@45
|
69 * found, it will always take priority over a "values" field.
|
c@45
|
70 */
|
c@44
|
71 enum class BufferSerialisation {
|
c@45
|
72
|
c@45
|
73 /** Default JSON serialisation of values in array form. This
|
c@45
|
74 * is relatively slow to parse and serialise, and can take a
|
c@45
|
75 * lot of space.
|
c@45
|
76 */
|
c@45
|
77 Text,
|
c@45
|
78
|
c@45
|
79 /** Base64-encoded string of the raw data as packed IEEE
|
c@45
|
80 * 32-bit floats. Faster and more compact than the text
|
c@45
|
81 * encoding but more complicated to provide, especially if
|
c@45
|
82 * starting from an environment that does not use IEEE 32-bit
|
c@45
|
83 * floats! Note that Base64 serialisations produced by this
|
c@45
|
84 * library do not including padding characters and so are not
|
c@45
|
85 * necessarily multiples of 4 characters long. You will need
|
c@45
|
86 * to pad them yourself if concatenating them or supplying to
|
c@45
|
87 * a consumer that expects padding.
|
c@45
|
88 */
|
c@45
|
89 Base64
|
c@44
|
90 };
|
c@44
|
91
|
c@5
|
92 class Failure : virtual public std::runtime_error {
|
c@5
|
93 public:
|
c@5
|
94 Failure(std::string s) : runtime_error(s) { }
|
c@5
|
95 };
|
c@5
|
96
|
c@5
|
97 template <typename T>
|
c@5
|
98 static json11::Json
|
c@5
|
99 fromBasicDescriptor(const T &t) {
|
c@5
|
100 return json11::Json::object {
|
c@5
|
101 { "identifier", t.identifier },
|
c@5
|
102 { "name", t.name },
|
c@5
|
103 { "description", t.description }
|
c@5
|
104 };
|
c@5
|
105 }
|
c@5
|
106
|
c@5
|
107 template <typename T>
|
c@5
|
108 static void
|
c@5
|
109 toBasicDescriptor(json11::Json j, T &t) {
|
c@5
|
110 if (!j.is_object()) {
|
c@5
|
111 throw Failure("object expected for basic descriptor content");
|
c@5
|
112 }
|
c@5
|
113 if (!j["identifier"].is_string()) {
|
c@5
|
114 throw Failure("string expected for identifier");
|
c@5
|
115 }
|
c@5
|
116 t.identifier = j["identifier"].string_value();
|
c@5
|
117 t.name = j["name"].string_value();
|
c@5
|
118 t.description = j["description"].string_value();
|
c@5
|
119 }
|
c@5
|
120
|
c@5
|
121 template <typename T>
|
c@5
|
122 static json11::Json
|
c@5
|
123 fromValueExtents(const T &t) {
|
c@5
|
124 return json11::Json::object {
|
c@5
|
125 { "min", t.minValue },
|
c@5
|
126 { "max", t.maxValue }
|
c@5
|
127 };
|
c@5
|
128 }
|
c@5
|
129
|
c@5
|
130 template <typename T>
|
c@5
|
131 static bool
|
c@5
|
132 toValueExtents(json11::Json j, T &t) {
|
c@5
|
133 if (j["extents"].is_null()) {
|
c@5
|
134 return false;
|
c@5
|
135 } else if (j["extents"].is_object()) {
|
c@5
|
136 if (j["extents"]["min"].is_number() &&
|
c@5
|
137 j["extents"]["max"].is_number()) {
|
c@5
|
138 t.minValue = j["extents"]["min"].number_value();
|
c@5
|
139 t.maxValue = j["extents"]["max"].number_value();
|
c@5
|
140 return true;
|
c@5
|
141 } else {
|
c@5
|
142 throw Failure("numbers expected for min and max");
|
c@5
|
143 }
|
c@5
|
144 } else {
|
c@5
|
145 throw Failure("object expected for extents (if present)");
|
c@5
|
146 }
|
c@5
|
147 }
|
c@5
|
148
|
c@5
|
149 static json11::Json
|
c@5
|
150 fromRealTime(const Vamp::RealTime &r) {
|
c@5
|
151 return json11::Json::object {
|
c@5
|
152 { "s", r.sec },
|
c@5
|
153 { "n", r.nsec }
|
c@5
|
154 };
|
c@5
|
155 }
|
c@5
|
156
|
c@5
|
157 static Vamp::RealTime
|
c@5
|
158 toRealTime(json11::Json j) {
|
c@5
|
159 json11::Json sec = j["s"];
|
c@5
|
160 json11::Json nsec = j["n"];
|
c@5
|
161 if (!sec.is_number() || !nsec.is_number()) {
|
c@5
|
162 throw Failure("invalid Vamp::RealTime object " + j.dump());
|
c@5
|
163 }
|
c@5
|
164 return Vamp::RealTime(sec.int_value(), nsec.int_value());
|
c@5
|
165 }
|
c@5
|
166
|
c@5
|
167 static std::string
|
c@5
|
168 fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType type) {
|
c@5
|
169 switch (type) {
|
c@5
|
170 case Vamp::Plugin::OutputDescriptor::OneSamplePerStep:
|
c@5
|
171 return "OneSamplePerStep";
|
c@5
|
172 case Vamp::Plugin::OutputDescriptor::FixedSampleRate:
|
c@5
|
173 return "FixedSampleRate";
|
c@5
|
174 case Vamp::Plugin::OutputDescriptor::VariableSampleRate:
|
c@5
|
175 return "VariableSampleRate";
|
c@5
|
176 }
|
c@5
|
177 return "";
|
c@5
|
178 }
|
c@5
|
179
|
c@5
|
180 static Vamp::Plugin::OutputDescriptor::SampleType
|
c@5
|
181 toSampleType(std::string text) {
|
c@5
|
182 if (text == "OneSamplePerStep") {
|
c@5
|
183 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
|
c@5
|
184 } else if (text == "FixedSampleRate") {
|
c@5
|
185 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
|
c@5
|
186 } else if (text == "VariableSampleRate") {
|
c@5
|
187 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
|
c@5
|
188 } else {
|
c@5
|
189 throw Failure("invalid sample type string: " + text);
|
c@5
|
190 }
|
c@5
|
191 }
|
c@5
|
192
|
c@5
|
193 static json11::Json
|
c@5
|
194 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) {
|
c@5
|
195 json11::Json::object jo {
|
c@5
|
196 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
197 { "unit", desc.unit },
|
c@5
|
198 { "sampleType", fromSampleType(desc.sampleType) },
|
c@5
|
199 { "sampleRate", desc.sampleRate },
|
c@5
|
200 { "hasDuration", desc.hasDuration }
|
c@5
|
201 };
|
c@5
|
202 if (desc.hasFixedBinCount) {
|
c@5
|
203 jo["binCount"] = int(desc.binCount);
|
c@5
|
204 jo["binNames"] = json11::Json::array
|
c@5
|
205 (desc.binNames.begin(), desc.binNames.end());
|
c@5
|
206 }
|
c@5
|
207 if (desc.hasKnownExtents) {
|
c@5
|
208 jo["extents"] = fromValueExtents(desc);
|
c@5
|
209 }
|
c@5
|
210 if (desc.isQuantized) {
|
c@5
|
211 jo["quantizeStep"] = desc.quantizeStep;
|
c@5
|
212 }
|
c@5
|
213 return json11::Json(jo);
|
c@5
|
214 }
|
c@12
|
215
|
c@5
|
216 static Vamp::Plugin::OutputDescriptor
|
c@5
|
217 toOutputDescriptor(json11::Json j) {
|
c@5
|
218
|
c@5
|
219 Vamp::Plugin::OutputDescriptor od;
|
c@5
|
220 if (!j.is_object()) {
|
c@5
|
221 throw Failure("object expected for output descriptor");
|
c@5
|
222 }
|
c@5
|
223
|
c@5
|
224 toBasicDescriptor(j["basic"], od);
|
c@5
|
225
|
c@5
|
226 od.unit = j["unit"].string_value();
|
c@5
|
227
|
c@5
|
228 od.sampleType = toSampleType(j["sampleType"].string_value());
|
c@5
|
229
|
c@5
|
230 if (!j["sampleRate"].is_number()) {
|
c@5
|
231 throw Failure("number expected for sample rate");
|
c@5
|
232 }
|
c@5
|
233 od.sampleRate = j["sampleRate"].number_value();
|
c@5
|
234 od.hasDuration = j["hasDuration"].bool_value();
|
c@5
|
235
|
c@5
|
236 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) {
|
c@5
|
237 od.hasFixedBinCount = true;
|
c@5
|
238 od.binCount = j["binCount"].int_value();
|
c@5
|
239 for (auto &n: j["binNames"].array_items()) {
|
c@5
|
240 if (!n.is_string()) {
|
c@5
|
241 throw Failure("string expected for bin name");
|
c@5
|
242 }
|
c@5
|
243 od.binNames.push_back(n.string_value());
|
c@5
|
244 }
|
c@5
|
245 } else {
|
c@5
|
246 od.hasFixedBinCount = false;
|
c@5
|
247 }
|
c@5
|
248
|
c@5
|
249 bool extentsPresent = toValueExtents(j, od);
|
c@5
|
250 od.hasKnownExtents = extentsPresent;
|
c@5
|
251
|
c@5
|
252 if (j["quantizeStep"].is_number()) {
|
c@5
|
253 od.isQuantized = true;
|
c@5
|
254 od.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
255 } else {
|
c@5
|
256 od.isQuantized = false;
|
c@5
|
257 }
|
c@5
|
258
|
c@5
|
259 return od;
|
c@5
|
260 }
|
c@5
|
261
|
c@5
|
262 static json11::Json
|
c@5
|
263 fromParameterDescriptor(const Vamp::PluginBase::ParameterDescriptor &desc) {
|
c@5
|
264
|
c@5
|
265 json11::Json::object jo {
|
c@5
|
266 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
267 { "unit", desc.unit },
|
c@5
|
268 { "extents", fromValueExtents(desc) },
|
c@5
|
269 { "defaultValue", desc.defaultValue },
|
c@5
|
270 { "valueNames", json11::Json::array
|
c@5
|
271 (desc.valueNames.begin(), desc.valueNames.end()) }
|
c@5
|
272 };
|
c@5
|
273 if (desc.isQuantized) {
|
c@5
|
274 jo["quantizeStep"] = desc.quantizeStep;
|
c@5
|
275 }
|
c@5
|
276 return json11::Json(jo);
|
c@5
|
277 }
|
c@5
|
278
|
c@5
|
279 static Vamp::PluginBase::ParameterDescriptor
|
c@5
|
280 toParameterDescriptor(json11::Json j) {
|
c@5
|
281
|
c@5
|
282 Vamp::PluginBase::ParameterDescriptor pd;
|
c@5
|
283 if (!j.is_object()) {
|
c@5
|
284 throw Failure("object expected for parameter descriptor");
|
c@5
|
285 }
|
c@5
|
286
|
c@5
|
287 toBasicDescriptor(j["basic"], pd);
|
c@5
|
288
|
c@5
|
289 pd.unit = j["unit"].string_value();
|
c@5
|
290
|
c@5
|
291 bool extentsPresent = toValueExtents(j, pd);
|
c@5
|
292 if (!extentsPresent) {
|
c@5
|
293 throw Failure("extents must be present in parameter descriptor");
|
c@5
|
294 }
|
c@5
|
295
|
c@5
|
296 if (!j["defaultValue"].is_number()) {
|
c@5
|
297 throw Failure("number expected for default value");
|
c@5
|
298 }
|
c@5
|
299
|
c@5
|
300 pd.defaultValue = j["defaultValue"].number_value();
|
c@5
|
301
|
c@5
|
302 pd.valueNames.clear();
|
c@5
|
303 for (auto &n: j["valueNames"].array_items()) {
|
c@5
|
304 if (!n.is_string()) {
|
c@5
|
305 throw Failure("string expected for value name");
|
c@5
|
306 }
|
c@5
|
307 pd.valueNames.push_back(n.string_value());
|
c@5
|
308 }
|
c@5
|
309
|
c@5
|
310 if (j["quantizeStep"].is_number()) {
|
c@5
|
311 pd.isQuantized = true;
|
c@5
|
312 pd.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
313 } else {
|
c@5
|
314 pd.isQuantized = false;
|
c@5
|
315 }
|
c@5
|
316
|
c@5
|
317 return pd;
|
c@5
|
318 }
|
c@5
|
319
|
c@5
|
320 static std::string
|
c@5
|
321 fromFloatBuffer(const float *buffer, size_t nfloats) {
|
c@46
|
322 static std::chrono::nanoseconds total;
|
c@46
|
323 auto t0 = std::chrono::high_resolution_clock::now();
|
c@5
|
324 // must use char pointers, otherwise the converter will only
|
c@5
|
325 // encode every 4th byte (as it will count up in float* steps)
|
c@5
|
326 const char *start = reinterpret_cast<const char *>(buffer);
|
c@5
|
327 const char *end = reinterpret_cast<const char *>(buffer + nfloats);
|
c@5
|
328 std::string encoded;
|
c@46
|
329 encoded.reserve(16 * nfloats / 3);
|
c@5
|
330 bn::encode_b64(start, end, back_inserter(encoded));
|
c@46
|
331 auto t1 = std::chrono::high_resolution_clock::now();
|
c@46
|
332 total += std::chrono::duration_cast<std::chrono::nanoseconds>(t1 - t0);
|
c@46
|
333 // std::cout << "encode: " << total.count()/1000000 << std::endl;
|
c@5
|
334 return encoded;
|
c@5
|
335 }
|
c@5
|
336
|
c@5
|
337 static std::vector<float>
|
c@5
|
338 toFloatBuffer(std::string encoded) {
|
c@46
|
339 static std::chrono::nanoseconds total;
|
c@46
|
340 auto t0 = std::chrono::high_resolution_clock::now();
|
c@5
|
341 std::string decoded;
|
c@46
|
342 decoded.reserve(3 * encoded.size() / 4);
|
c@5
|
343 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded));
|
c@5
|
344 const float *buffer = reinterpret_cast<const float *>(decoded.c_str());
|
c@5
|
345 size_t n = decoded.size() / sizeof(float);
|
c@46
|
346 std::vector<float> result(buffer, buffer + n);
|
c@46
|
347 auto t1 = std::chrono::high_resolution_clock::now();
|
c@46
|
348 total += std::chrono::duration_cast<std::chrono::nanoseconds>(t1 - t0);
|
c@46
|
349 // std::cout << "decode: " << total.count()/1000000 << std::endl;
|
c@46
|
350 return result;
|
c@5
|
351 }
|
c@5
|
352
|
c@5
|
353 static json11::Json
|
c@44
|
354 fromFeature(const Vamp::Plugin::Feature &f,
|
c@44
|
355 BufferSerialisation serialisation) {
|
c@5
|
356
|
c@5
|
357 json11::Json::object jo;
|
c@5
|
358 if (f.values.size() > 0) {
|
c@44
|
359 if (serialisation == BufferSerialisation::Text) {
|
c@35
|
360 jo["values"] = json11::Json::array(f.values.begin(),
|
c@35
|
361 f.values.end());
|
c@35
|
362 } else {
|
c@35
|
363 jo["b64values"] = fromFloatBuffer(f.values.data(),
|
c@35
|
364 f.values.size());
|
c@35
|
365 }
|
c@5
|
366 }
|
c@5
|
367 if (f.label != "") {
|
c@5
|
368 jo["label"] = f.label;
|
c@5
|
369 }
|
c@5
|
370 if (f.hasTimestamp) {
|
c@5
|
371 jo["timestamp"] = fromRealTime(f.timestamp);
|
c@5
|
372 }
|
c@5
|
373 if (f.hasDuration) {
|
c@5
|
374 jo["duration"] = fromRealTime(f.duration);
|
c@5
|
375 }
|
c@5
|
376 return json11::Json(jo);
|
c@5
|
377 }
|
c@5
|
378
|
c@5
|
379 static Vamp::Plugin::Feature
|
c@44
|
380 toFeature(json11::Json j,
|
c@44
|
381 BufferSerialisation &serialisation) {
|
c@5
|
382
|
c@5
|
383 Vamp::Plugin::Feature f;
|
c@5
|
384 if (!j.is_object()) {
|
c@5
|
385 throw Failure("object expected for feature");
|
c@5
|
386 }
|
c@5
|
387 if (j["timestamp"].is_object()) {
|
c@5
|
388 f.timestamp = toRealTime(j["timestamp"]);
|
c@5
|
389 f.hasTimestamp = true;
|
c@5
|
390 }
|
c@5
|
391 if (j["duration"].is_object()) {
|
c@5
|
392 f.duration = toRealTime(j["duration"]);
|
c@5
|
393 f.hasDuration = true;
|
c@5
|
394 }
|
c@5
|
395 if (j["b64values"].is_string()) {
|
c@5
|
396 f.values = toFloatBuffer(j["b64values"].string_value());
|
c@44
|
397 serialisation = BufferSerialisation::Base64;
|
c@5
|
398 } else if (j["values"].is_array()) {
|
c@5
|
399 for (auto v : j["values"].array_items()) {
|
c@5
|
400 f.values.push_back(v.number_value());
|
c@5
|
401 }
|
c@44
|
402 serialisation = BufferSerialisation::Text;
|
c@5
|
403 }
|
c@5
|
404 f.label = j["label"].string_value();
|
c@5
|
405 return f;
|
c@5
|
406 }
|
c@5
|
407
|
c@5
|
408 static json11::Json
|
c@44
|
409 fromFeatureSet(const Vamp::Plugin::FeatureSet &fs,
|
c@44
|
410 BufferSerialisation serialisation) {
|
c@5
|
411
|
c@5
|
412 json11::Json::object jo;
|
c@5
|
413 for (const auto &fsi : fs) {
|
c@5
|
414 std::vector<json11::Json> fj;
|
c@5
|
415 for (const Vamp::Plugin::Feature &f: fsi.second) {
|
c@44
|
416 fj.push_back(fromFeature(f, serialisation));
|
c@5
|
417 }
|
c@5
|
418 std::stringstream sstr;
|
c@5
|
419 sstr << fsi.first;
|
c@5
|
420 std::string n = sstr.str();
|
c@5
|
421 jo[n] = fj;
|
c@5
|
422 }
|
c@5
|
423 return json11::Json(jo);
|
c@5
|
424 }
|
c@5
|
425
|
c@5
|
426 static Vamp::Plugin::FeatureList
|
c@44
|
427 toFeatureList(json11::Json j,
|
c@44
|
428 BufferSerialisation &serialisation) {
|
c@5
|
429
|
c@5
|
430 Vamp::Plugin::FeatureList fl;
|
c@5
|
431 if (!j.is_array()) {
|
c@5
|
432 throw Failure("array expected for feature list");
|
c@5
|
433 }
|
c@5
|
434 for (const json11::Json &fj : j.array_items()) {
|
c@44
|
435 fl.push_back(toFeature(fj, serialisation));
|
c@5
|
436 }
|
c@5
|
437 return fl;
|
c@5
|
438 }
|
c@5
|
439
|
c@5
|
440 static Vamp::Plugin::FeatureSet
|
c@44
|
441 toFeatureSet(json11::Json j, BufferSerialisation &serialisation) {
|
c@5
|
442
|
c@5
|
443 Vamp::Plugin::FeatureSet fs;
|
c@5
|
444 if (!j.is_object()) {
|
c@5
|
445 throw Failure("object expected for feature set");
|
c@5
|
446 }
|
c@5
|
447 for (auto &entry : j.object_items()) {
|
c@5
|
448 std::string nstr = entry.first;
|
c@5
|
449 size_t count = 0;
|
c@5
|
450 int n = stoi(nstr, &count);
|
c@5
|
451 if (n < 0 || fs.find(n) != fs.end() || count < nstr.size()) {
|
c@5
|
452 throw Failure("invalid or duplicate numerical index for output");
|
c@5
|
453 }
|
c@44
|
454 fs[n] = toFeatureList(entry.second, serialisation);
|
c@5
|
455 }
|
c@5
|
456 return fs;
|
c@5
|
457 }
|
c@5
|
458
|
c@5
|
459 static std::string
|
c@5
|
460 fromInputDomain(Vamp::Plugin::InputDomain domain) {
|
c@5
|
461
|
c@5
|
462 switch (domain) {
|
c@5
|
463 case Vamp::Plugin::TimeDomain:
|
c@5
|
464 return "TimeDomain";
|
c@5
|
465 case Vamp::Plugin::FrequencyDomain:
|
c@5
|
466 return "FrequencyDomain";
|
c@5
|
467 }
|
c@5
|
468 return "";
|
c@5
|
469 }
|
c@5
|
470
|
c@5
|
471 static Vamp::Plugin::InputDomain
|
c@5
|
472 toInputDomain(std::string text) {
|
c@5
|
473
|
c@5
|
474 if (text == "TimeDomain") {
|
c@5
|
475 return Vamp::Plugin::TimeDomain;
|
c@5
|
476 } else if (text == "FrequencyDomain") {
|
c@5
|
477 return Vamp::Plugin::FrequencyDomain;
|
c@5
|
478 } else {
|
c@5
|
479 throw Failure("invalid input domain string: " + text);
|
c@5
|
480 }
|
c@5
|
481 }
|
c@5
|
482
|
c@5
|
483 static json11::Json
|
c@5
|
484 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) {
|
c@5
|
485
|
c@5
|
486 json11::Json::object jo;
|
c@5
|
487 jo["pluginKey"] = d.pluginKey;
|
c@5
|
488 jo["basic"] = fromBasicDescriptor(d.basic);
|
c@5
|
489 jo["maker"] = d.maker;
|
c@5
|
490 jo["copyright"] = d.copyright;
|
c@5
|
491 jo["pluginVersion"] = d.pluginVersion;
|
c@5
|
492
|
c@5
|
493 json11::Json::array cat;
|
c@5
|
494 for (const std::string &c: d.category) cat.push_back(c);
|
c@5
|
495 jo["category"] = cat;
|
c@5
|
496
|
c@5
|
497 jo["minChannelCount"] = d.minChannelCount;
|
c@5
|
498 jo["maxChannelCount"] = d.maxChannelCount;
|
c@5
|
499
|
c@5
|
500 json11::Json::array params;
|
c@5
|
501 Vamp::PluginBase::ParameterList vparams = d.parameters;
|
c@5
|
502 for (auto &p: vparams) params.push_back(fromParameterDescriptor(p));
|
c@5
|
503 jo["parameters"] = params;
|
c@5
|
504
|
c@5
|
505 json11::Json::array progs;
|
c@5
|
506 Vamp::PluginBase::ProgramList vprogs = d.programs;
|
c@5
|
507 for (auto &p: vprogs) progs.push_back(p);
|
c@5
|
508 jo["programs"] = progs;
|
c@5
|
509
|
c@5
|
510 jo["inputDomain"] = fromInputDomain(d.inputDomain);
|
c@5
|
511
|
c@5
|
512 json11::Json::array outinfo;
|
c@5
|
513 auto vouts = d.basicOutputInfo;
|
c@5
|
514 for (auto &o: vouts) outinfo.push_back(fromBasicDescriptor(o));
|
c@5
|
515 jo["basicOutputInfo"] = outinfo;
|
c@5
|
516
|
c@5
|
517 return json11::Json(jo);
|
c@5
|
518 }
|
c@5
|
519
|
c@5
|
520 static Vamp::HostExt::PluginStaticData
|
c@5
|
521 toPluginStaticData(json11::Json j) {
|
c@5
|
522
|
c@5
|
523 std::string err;
|
c@5
|
524 if (!j.has_shape({
|
c@5
|
525 { "pluginKey", json11::Json::STRING },
|
c@5
|
526 { "pluginVersion", json11::Json::NUMBER },
|
c@5
|
527 { "minChannelCount", json11::Json::NUMBER },
|
c@5
|
528 { "maxChannelCount", json11::Json::NUMBER },
|
c@5
|
529 { "inputDomain", json11::Json::STRING }}, err)) {
|
c@5
|
530 throw Failure("malformed plugin static data: " + err);
|
c@5
|
531 }
|
c@5
|
532
|
c@5
|
533 if (!j["basicOutputInfo"].is_array()) {
|
c@5
|
534 throw Failure("array expected for basic output info");
|
c@5
|
535 }
|
c@5
|
536
|
c@5
|
537 if (!j["maker"].is_null() &&
|
c@5
|
538 !j["maker"].is_string()) {
|
c@5
|
539 throw Failure("string expected for maker");
|
c@5
|
540 }
|
c@5
|
541
|
c@5
|
542 if (!j["copyright"].is_null() &&
|
c@5
|
543 !j["copyright"].is_string()) {
|
c@5
|
544 throw Failure("string expected for copyright");
|
c@5
|
545 }
|
c@5
|
546
|
c@5
|
547 if (!j["category"].is_null() &&
|
c@5
|
548 !j["category"].is_array()) {
|
c@5
|
549 throw Failure("array expected for category");
|
c@5
|
550 }
|
c@5
|
551
|
c@5
|
552 if (!j["parameters"].is_null() &&
|
c@5
|
553 !j["parameters"].is_array()) {
|
c@5
|
554 throw Failure("array expected for parameters");
|
c@5
|
555 }
|
c@5
|
556
|
c@5
|
557 if (!j["programs"].is_null() &&
|
c@5
|
558 !j["programs"].is_array()) {
|
c@5
|
559 throw Failure("array expected for programs");
|
c@5
|
560 }
|
c@5
|
561
|
c@5
|
562 if (!j["inputDomain"].is_null() &&
|
c@5
|
563 !j["inputDomain"].is_string()) {
|
c@5
|
564 throw Failure("string expected for inputDomain");
|
c@5
|
565 }
|
c@5
|
566
|
c@5
|
567 if (!j["basicOutputInfo"].is_null() &&
|
c@5
|
568 !j["basicOutputInfo"].is_array()) {
|
c@5
|
569 throw Failure("array expected for basicOutputInfo");
|
c@5
|
570 }
|
c@5
|
571
|
c@5
|
572 Vamp::HostExt::PluginStaticData psd;
|
c@5
|
573
|
c@5
|
574 psd.pluginKey = j["pluginKey"].string_value();
|
c@5
|
575
|
c@5
|
576 toBasicDescriptor(j["basic"], psd.basic);
|
c@5
|
577
|
c@5
|
578 psd.maker = j["maker"].string_value();
|
c@5
|
579 psd.copyright = j["copyright"].string_value();
|
c@5
|
580 psd.pluginVersion = j["pluginVersion"].int_value();
|
c@5
|
581
|
c@5
|
582 for (const auto &c : j["category"].array_items()) {
|
c@5
|
583 if (!c.is_string()) {
|
c@5
|
584 throw Failure("strings expected in category array");
|
c@5
|
585 }
|
c@5
|
586 psd.category.push_back(c.string_value());
|
c@5
|
587 }
|
c@5
|
588
|
c@5
|
589 psd.minChannelCount = j["minChannelCount"].int_value();
|
c@5
|
590 psd.maxChannelCount = j["maxChannelCount"].int_value();
|
c@5
|
591
|
c@5
|
592 for (const auto &p : j["parameters"].array_items()) {
|
c@5
|
593 auto pd = toParameterDescriptor(p);
|
c@5
|
594 psd.parameters.push_back(pd);
|
c@5
|
595 }
|
c@5
|
596
|
c@5
|
597 for (const auto &p : j["programs"].array_items()) {
|
c@5
|
598 if (!p.is_string()) {
|
c@5
|
599 throw Failure("strings expected in programs array");
|
c@5
|
600 }
|
c@5
|
601 psd.programs.push_back(p.string_value());
|
c@5
|
602 }
|
c@5
|
603
|
c@5
|
604 psd.inputDomain = toInputDomain(j["inputDomain"].string_value());
|
c@5
|
605
|
c@5
|
606 for (const auto &bo : j["basicOutputInfo"].array_items()) {
|
c@5
|
607 Vamp::HostExt::PluginStaticData::Basic b;
|
c@5
|
608 toBasicDescriptor(bo, b);
|
c@5
|
609 psd.basicOutputInfo.push_back(b);
|
c@5
|
610 }
|
c@5
|
611
|
c@5
|
612 return psd;
|
c@5
|
613 }
|
c@5
|
614
|
c@5
|
615 static json11::Json
|
c@5
|
616 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) {
|
c@5
|
617
|
c@5
|
618 json11::Json::object jo;
|
c@5
|
619
|
c@5
|
620 json11::Json::object paramValues;
|
c@5
|
621 for (auto &vp: c.parameterValues) {
|
c@5
|
622 paramValues[vp.first] = vp.second;
|
c@5
|
623 }
|
c@5
|
624 jo["parameterValues"] = paramValues;
|
c@5
|
625
|
c@5
|
626 if (c.currentProgram != "") {
|
c@5
|
627 jo["currentProgram"] = c.currentProgram;
|
c@5
|
628 }
|
c@5
|
629
|
c@5
|
630 jo["channelCount"] = c.channelCount;
|
c@5
|
631 jo["stepSize"] = c.stepSize;
|
c@5
|
632 jo["blockSize"] = c.blockSize;
|
c@5
|
633
|
c@5
|
634 return json11::Json(jo);
|
c@5
|
635 }
|
c@5
|
636
|
c@5
|
637 static Vamp::HostExt::PluginConfiguration
|
c@5
|
638 toPluginConfiguration(json11::Json j) {
|
c@5
|
639
|
c@5
|
640 std::string err;
|
c@5
|
641 if (!j.has_shape({
|
c@5
|
642 { "channelCount", json11::Json::NUMBER },
|
c@5
|
643 { "stepSize", json11::Json::NUMBER },
|
c@5
|
644 { "blockSize", json11::Json::NUMBER } }, err)) {
|
c@5
|
645 throw Failure("malformed plugin configuration: " + err);
|
c@5
|
646 }
|
c@5
|
647
|
c@5
|
648 if (!j["parameterValues"].is_null() &&
|
c@5
|
649 !j["parameterValues"].is_object()) {
|
c@5
|
650 throw Failure("object expected for parameter values");
|
c@5
|
651 }
|
c@5
|
652
|
c@5
|
653 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
654 if (!pv.second.is_number()) {
|
c@5
|
655 throw Failure("number expected for parameter value");
|
c@5
|
656 }
|
c@5
|
657 }
|
c@5
|
658
|
c@5
|
659 if (!j["currentProgram"].is_null() &&
|
c@5
|
660 !j["currentProgram"].is_string()) {
|
c@5
|
661 throw Failure("string expected for program name");
|
c@5
|
662 }
|
c@5
|
663
|
c@5
|
664 Vamp::HostExt::PluginConfiguration config;
|
c@5
|
665
|
c@5
|
666 config.channelCount = j["channelCount"].number_value();
|
c@5
|
667 config.stepSize = j["stepSize"].number_value();
|
c@5
|
668 config.blockSize = j["blockSize"].number_value();
|
c@5
|
669
|
c@5
|
670 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
671 config.parameterValues[pv.first] = pv.second.number_value();
|
c@5
|
672 }
|
c@5
|
673
|
c@5
|
674 if (j["currentProgram"].is_string()) {
|
c@5
|
675 config.currentProgram = j["currentProgram"].string_value();
|
c@5
|
676 }
|
c@5
|
677
|
c@5
|
678 return config;
|
c@5
|
679 }
|
c@5
|
680
|
c@5
|
681 static json11::Json
|
c@5
|
682 fromAdapterFlags(int flags) {
|
c@5
|
683
|
c@5
|
684 json11::Json::array arr;
|
c@5
|
685
|
c@5
|
686 if (flags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
|
c@5
|
687 arr.push_back("AdaptInputDomain");
|
c@5
|
688 }
|
c@5
|
689 if (flags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
|
c@5
|
690 arr.push_back("AdaptChannelCount");
|
c@5
|
691 }
|
c@5
|
692 if (flags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
|
c@5
|
693 arr.push_back("AdaptBufferSize");
|
c@5
|
694 }
|
c@5
|
695
|
c@5
|
696 return json11::Json(arr);
|
c@5
|
697 }
|
c@5
|
698
|
c@5
|
699 static Vamp::HostExt::PluginLoader::AdapterFlags
|
c@5
|
700 toAdapterFlags(json11::Json j) {
|
c@5
|
701
|
c@5
|
702 if (!j.is_array()) {
|
c@5
|
703 throw Failure("array expected for adapter flags");
|
c@5
|
704 }
|
c@5
|
705 int flags = 0x0;
|
c@5
|
706
|
c@5
|
707 for (auto &jj: j.array_items()) {
|
c@5
|
708 if (!jj.is_string()) {
|
c@5
|
709 throw Failure("string expected for adapter flag");
|
c@5
|
710 }
|
c@5
|
711 std::string text = jj.string_value();
|
c@5
|
712 if (text == "AdaptInputDomain") {
|
c@5
|
713 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
|
c@5
|
714 } else if (text == "AdaptChannelCount") {
|
c@5
|
715 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
|
c@5
|
716 } else if (text == "AdaptBufferSize") {
|
c@5
|
717 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
|
c@5
|
718 } else if (text == "AdaptAllSafe") {
|
c@5
|
719 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE;
|
c@5
|
720 } else if (text == "AdaptAll") {
|
c@5
|
721 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL;
|
c@5
|
722 } else {
|
c@5
|
723 throw Failure("invalid adapter flag string: " + text);
|
c@5
|
724 }
|
c@5
|
725 }
|
c@5
|
726
|
c@5
|
727 return Vamp::HostExt::PluginLoader::AdapterFlags(flags);
|
c@5
|
728 }
|
c@5
|
729
|
c@5
|
730 static json11::Json
|
c@17
|
731 fromLoadRequest(const Vamp::HostExt::LoadRequest &req) {
|
c@5
|
732
|
c@5
|
733 json11::Json::object jo;
|
c@5
|
734 jo["pluginKey"] = req.pluginKey;
|
c@5
|
735 jo["inputSampleRate"] = req.inputSampleRate;
|
c@5
|
736 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags);
|
c@5
|
737 return json11::Json(jo);
|
c@5
|
738 }
|
c@5
|
739
|
c@5
|
740 static Vamp::HostExt::LoadRequest
|
c@5
|
741 toLoadRequest(json11::Json j) {
|
c@5
|
742
|
c@5
|
743 std::string err;
|
c@5
|
744
|
c@5
|
745 if (!j.has_shape({
|
c@5
|
746 { "pluginKey", json11::Json::STRING },
|
c@32
|
747 { "inputSampleRate", json11::Json::NUMBER } }, err)) {
|
c@12
|
748 throw Failure("malformed load request: " + err);
|
c@5
|
749 }
|
c@5
|
750
|
c@5
|
751 Vamp::HostExt::LoadRequest req;
|
c@5
|
752 req.pluginKey = j["pluginKey"].string_value();
|
c@5
|
753 req.inputSampleRate = j["inputSampleRate"].number_value();
|
c@32
|
754 if (!j["adapterFlags"].is_null()) {
|
c@32
|
755 req.adapterFlags = toAdapterFlags(j["adapterFlags"]);
|
c@32
|
756 }
|
c@5
|
757 return req;
|
c@5
|
758 }
|
c@10
|
759
|
c@10
|
760 static json11::Json
|
c@17
|
761 fromLoadResponse(const Vamp::HostExt::LoadResponse &resp,
|
c@39
|
762 const PluginHandleMapper &mapper) {
|
c@10
|
763
|
c@10
|
764 json11::Json::object jo;
|
c@10
|
765 jo["pluginHandle"] = double(mapper.pluginToHandle(resp.plugin));
|
c@10
|
766 jo["staticData"] = fromPluginStaticData(resp.staticData);
|
c@10
|
767 jo["defaultConfiguration"] =
|
c@10
|
768 fromPluginConfiguration(resp.defaultConfiguration);
|
c@10
|
769 return json11::Json(jo);
|
c@10
|
770 }
|
c@10
|
771
|
c@10
|
772 static Vamp::HostExt::LoadResponse
|
c@10
|
773 toLoadResponse(json11::Json j,
|
c@39
|
774 const PluginHandleMapper &mapper) {
|
c@10
|
775
|
c@10
|
776 std::string err;
|
c@10
|
777
|
c@10
|
778 if (!j.has_shape({
|
c@10
|
779 { "pluginHandle", json11::Json::NUMBER },
|
c@10
|
780 { "staticData", json11::Json::OBJECT },
|
c@10
|
781 { "defaultConfiguration", json11::Json::OBJECT } }, err)) {
|
c@12
|
782 throw Failure("malformed load response: " + err);
|
c@10
|
783 }
|
c@10
|
784
|
c@10
|
785 Vamp::HostExt::LoadResponse resp;
|
c@10
|
786 resp.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@10
|
787 resp.staticData = toPluginStaticData(j["staticData"]);
|
c@10
|
788 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"]);
|
c@10
|
789 return resp;
|
c@10
|
790 }
|
c@12
|
791
|
c@12
|
792 static json11::Json
|
c@13
|
793 fromConfigurationRequest(const Vamp::HostExt::ConfigurationRequest &cr,
|
c@39
|
794 const PluginHandleMapper &mapper) {
|
c@13
|
795
|
c@13
|
796 json11::Json::object jo;
|
c@13
|
797
|
c@13
|
798 jo["pluginHandle"] = mapper.pluginToHandle(cr.plugin);
|
c@13
|
799 jo["configuration"] = fromPluginConfiguration(cr.configuration);
|
c@13
|
800
|
c@13
|
801 return json11::Json(jo);
|
c@13
|
802 }
|
c@13
|
803
|
c@13
|
804 static Vamp::HostExt::ConfigurationRequest
|
c@13
|
805 toConfigurationRequest(json11::Json j,
|
c@39
|
806 const PluginHandleMapper &mapper) {
|
c@13
|
807
|
c@13
|
808 std::string err;
|
c@13
|
809
|
c@13
|
810 if (!j.has_shape({
|
c@13
|
811 { "pluginHandle", json11::Json::NUMBER },
|
c@13
|
812 { "configuration", json11::Json::OBJECT } }, err)) {
|
c@13
|
813 throw Failure("malformed configuration request: " + err);
|
c@13
|
814 }
|
c@13
|
815
|
c@13
|
816 Vamp::HostExt::ConfigurationRequest cr;
|
c@13
|
817 cr.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@13
|
818 cr.configuration = toPluginConfiguration(j["configuration"]);
|
c@13
|
819 return cr;
|
c@13
|
820 }
|
c@13
|
821
|
c@13
|
822 static json11::Json
|
c@12
|
823 fromConfigurationResponse(const Vamp::HostExt::ConfigurationResponse &cr) {
|
c@12
|
824
|
c@13
|
825 json11::Json::object jo;
|
c@12
|
826
|
c@12
|
827 json11::Json::array outs;
|
c@12
|
828 for (auto &d: cr.outputs) {
|
c@12
|
829 outs.push_back(fromOutputDescriptor(d));
|
c@12
|
830 }
|
c@13
|
831 jo["outputList"] = outs;
|
c@12
|
832
|
c@13
|
833 return json11::Json(jo);
|
c@12
|
834 }
|
c@12
|
835
|
c@13
|
836 static Vamp::HostExt::ConfigurationResponse
|
c@13
|
837 toConfigurationResponse(json11::Json j) {
|
c@13
|
838
|
c@12
|
839 Vamp::HostExt::ConfigurationResponse cr;
|
c@12
|
840
|
c@12
|
841 if (!j["outputList"].is_array()) {
|
c@12
|
842 throw Failure("array expected for output list");
|
c@12
|
843 }
|
c@12
|
844
|
c@12
|
845 for (const auto &o: j["outputList"].array_items()) {
|
c@12
|
846 cr.outputs.push_back(toOutputDescriptor(o));
|
c@12
|
847 }
|
c@12
|
848
|
c@12
|
849 return cr;
|
c@12
|
850 }
|
c@16
|
851
|
c@16
|
852 static json11::Json
|
c@16
|
853 fromProcessRequest(const Vamp::HostExt::ProcessRequest &r,
|
c@44
|
854 const PluginHandleMapper &mapper,
|
c@44
|
855 BufferSerialisation serialisation) {
|
c@16
|
856
|
c@16
|
857 json11::Json::object jo;
|
c@16
|
858 jo["pluginHandle"] = mapper.pluginToHandle(r.plugin);
|
c@16
|
859
|
c@16
|
860 json11::Json::object io;
|
c@16
|
861 io["timestamp"] = fromRealTime(r.timestamp);
|
c@16
|
862
|
c@16
|
863 json11::Json::array chans;
|
c@16
|
864 for (size_t i = 0; i < r.inputBuffers.size(); ++i) {
|
c@16
|
865 json11::Json::object c;
|
c@44
|
866 if (serialisation == BufferSerialisation::Text) {
|
c@44
|
867 c["values"] = json11::Json::array(r.inputBuffers[i].begin(),
|
c@44
|
868 r.inputBuffers[i].end());
|
c@44
|
869 } else {
|
c@44
|
870 c["b64values"] = fromFloatBuffer(r.inputBuffers[i].data(),
|
c@44
|
871 r.inputBuffers[i].size());
|
c@44
|
872 }
|
c@16
|
873 chans.push_back(c);
|
c@16
|
874 }
|
c@16
|
875 io["inputBuffers"] = chans;
|
c@16
|
876
|
c@16
|
877 jo["processInput"] = io;
|
c@16
|
878 return json11::Json(jo);
|
c@16
|
879 }
|
c@17
|
880
|
c@17
|
881 static Vamp::HostExt::ProcessRequest
|
c@44
|
882 toProcessRequest(json11::Json j,
|
c@44
|
883 const PluginHandleMapper &mapper,
|
c@44
|
884 BufferSerialisation &serialisation) {
|
c@17
|
885
|
c@46
|
886 static std::chrono::nanoseconds total;
|
c@46
|
887 auto t0 = std::chrono::high_resolution_clock::now();
|
c@46
|
888
|
c@17
|
889 std::string err;
|
c@17
|
890
|
c@17
|
891 if (!j.has_shape({
|
c@17
|
892 { "pluginHandle", json11::Json::NUMBER },
|
c@17
|
893 { "processInput", json11::Json::OBJECT } }, err)) {
|
c@17
|
894 throw Failure("malformed process request: " + err);
|
c@17
|
895 }
|
c@17
|
896
|
c@17
|
897 auto input = j["processInput"];
|
c@17
|
898
|
c@17
|
899 if (!input.has_shape({
|
c@17
|
900 { "timestamp", json11::Json::OBJECT },
|
c@17
|
901 { "inputBuffers", json11::Json::ARRAY } }, err)) {
|
c@17
|
902 throw Failure("malformed process request: " + err);
|
c@17
|
903 }
|
c@17
|
904
|
c@17
|
905 Vamp::HostExt::ProcessRequest r;
|
c@17
|
906 r.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@17
|
907
|
c@17
|
908 r.timestamp = toRealTime(input["timestamp"]);
|
c@17
|
909
|
c@17
|
910 for (auto a: input["inputBuffers"].array_items()) {
|
c@44
|
911
|
c@17
|
912 if (a["b64values"].is_string()) {
|
c@44
|
913 std::vector<float> buf = toFloatBuffer(a["b64values"].string_value());
|
c@44
|
914 r.inputBuffers.push_back(buf);
|
c@44
|
915 serialisation = BufferSerialisation::Base64;
|
c@44
|
916
|
c@17
|
917 } else if (a["values"].is_array()) {
|
c@17
|
918 std::vector<float> buf;
|
c@17
|
919 for (auto v : a["values"].array_items()) {
|
c@17
|
920 buf.push_back(v.number_value());
|
c@17
|
921 }
|
c@17
|
922 r.inputBuffers.push_back(buf);
|
c@44
|
923 serialisation = BufferSerialisation::Text;
|
c@44
|
924
|
c@17
|
925 } else {
|
c@17
|
926 throw Failure("expected values or b64values in inputBuffers object");
|
c@17
|
927 }
|
c@17
|
928 }
|
c@17
|
929
|
c@46
|
930 auto t1 = std::chrono::high_resolution_clock::now();
|
c@46
|
931 total += std::chrono::duration_cast<std::chrono::nanoseconds>(t1 - t0);
|
c@46
|
932 // std::cout << "parse: " << total.count()/1000000 << std::endl;
|
c@46
|
933
|
c@17
|
934 return r;
|
c@17
|
935 }
|
c@17
|
936
|
c@17
|
937 static json11::Json
|
c@17
|
938 fromVampRequest_List() {
|
c@17
|
939
|
c@17
|
940 json11::Json::object jo;
|
c@17
|
941 jo["type"] = "list";
|
c@17
|
942 return json11::Json(jo);
|
c@17
|
943 }
|
c@17
|
944
|
c@17
|
945 static json11::Json
|
c@17
|
946 fromVampResponse_List(std::string errorText,
|
c@17
|
947 const std::vector<Vamp::HostExt::PluginStaticData> &d) {
|
c@17
|
948
|
c@17
|
949 json11::Json::object jo;
|
c@24
|
950 jo["type"] = "list";
|
c@17
|
951 jo["success"] = (errorText == "");
|
c@17
|
952 jo["errorText"] = errorText;
|
c@17
|
953
|
c@17
|
954 json11::Json::array arr;
|
c@17
|
955 for (const auto &a: d) {
|
c@17
|
956 arr.push_back(fromPluginStaticData(a));
|
c@17
|
957 }
|
c@24
|
958 json11::Json::object po;
|
c@24
|
959 po["plugins"] = arr;
|
c@24
|
960
|
c@24
|
961 jo["content"] = po;
|
c@17
|
962 return json11::Json(jo);
|
c@17
|
963 }
|
c@17
|
964
|
c@17
|
965 static json11::Json
|
c@17
|
966 fromVampRequest_Load(const Vamp::HostExt::LoadRequest &req) {
|
c@17
|
967
|
c@17
|
968 json11::Json::object jo;
|
c@17
|
969 jo["type"] = "load";
|
c@17
|
970 jo["content"] = fromLoadRequest(req);
|
c@17
|
971 return json11::Json(jo);
|
c@17
|
972 }
|
c@17
|
973
|
c@17
|
974 static json11::Json
|
c@17
|
975 fromVampResponse_Load(const Vamp::HostExt::LoadResponse &resp,
|
c@39
|
976 const PluginHandleMapper &mapper) {
|
c@17
|
977
|
c@17
|
978 json11::Json::object jo;
|
c@24
|
979 jo["type"] = "load";
|
c@17
|
980 jo["success"] = (resp.plugin != 0);
|
c@17
|
981 jo["errorText"] = "";
|
c@24
|
982 jo["content"] = fromLoadResponse(resp, mapper);
|
c@17
|
983 return json11::Json(jo);
|
c@17
|
984 }
|
c@17
|
985
|
c@17
|
986 static json11::Json
|
c@17
|
987 fromVampRequest_Configure(const Vamp::HostExt::ConfigurationRequest &req,
|
c@39
|
988 const PluginHandleMapper &mapper) {
|
c@17
|
989
|
c@17
|
990 json11::Json::object jo;
|
c@17
|
991 jo["type"] = "configure";
|
c@17
|
992 jo["content"] = fromConfigurationRequest(req, mapper);
|
c@17
|
993 return json11::Json(jo);
|
c@17
|
994 }
|
c@17
|
995
|
c@17
|
996 static json11::Json
|
c@17
|
997 fromVampResponse_Configure(const Vamp::HostExt::ConfigurationResponse &resp) {
|
c@17
|
998
|
c@17
|
999 json11::Json::object jo;
|
c@24
|
1000 jo["type"] = "configure";
|
c@17
|
1001 jo["success"] = (!resp.outputs.empty());
|
c@17
|
1002 jo["errorText"] = "";
|
c@24
|
1003 jo["content"] = fromConfigurationResponse(resp);
|
c@17
|
1004 return json11::Json(jo);
|
c@17
|
1005 }
|
c@17
|
1006
|
c@17
|
1007 static json11::Json
|
c@17
|
1008 fromVampRequest_Process(const Vamp::HostExt::ProcessRequest &req,
|
c@44
|
1009 const PluginHandleMapper &mapper,
|
c@44
|
1010 BufferSerialisation serialisation) {
|
c@17
|
1011
|
c@17
|
1012 json11::Json::object jo;
|
c@17
|
1013 jo["type"] = "process";
|
c@44
|
1014 jo["content"] = fromProcessRequest(req, mapper, serialisation);
|
c@17
|
1015 return json11::Json(jo);
|
c@17
|
1016 }
|
c@17
|
1017
|
c@17
|
1018 static json11::Json
|
c@44
|
1019 fromVampResponse_Process(const Vamp::HostExt::ProcessResponse &resp,
|
c@44
|
1020 BufferSerialisation serialisation) {
|
c@17
|
1021
|
c@17
|
1022 json11::Json::object jo;
|
c@24
|
1023 jo["type"] = "process";
|
c@17
|
1024 jo["success"] = true;
|
c@17
|
1025 jo["errorText"] = "";
|
c@44
|
1026 jo["content"] = fromFeatureSet(resp.features, serialisation);
|
c@17
|
1027 return json11::Json(jo);
|
c@17
|
1028 }
|
c@17
|
1029
|
c@17
|
1030 static json11::Json
|
c@24
|
1031 fromVampRequest_Finish(Vamp::Plugin *p,
|
c@39
|
1032 const PluginHandleMapper &mapper) {
|
c@17
|
1033
|
c@17
|
1034 json11::Json::object jo;
|
c@17
|
1035 jo["type"] = "finish";
|
c@24
|
1036 json11::Json::object fo;
|
c@24
|
1037 fo["pluginHandle"] = mapper.pluginToHandle(p);
|
c@24
|
1038 jo["content"] = fo;
|
c@17
|
1039 return json11::Json(jo);
|
c@17
|
1040 }
|
c@17
|
1041
|
c@17
|
1042 static json11::Json
|
c@44
|
1043 fromVampResponse_Finish(const Vamp::HostExt::ProcessResponse &resp,
|
c@44
|
1044 BufferSerialisation serialisation) {
|
c@17
|
1045
|
c@17
|
1046 json11::Json::object jo;
|
c@24
|
1047 jo["type"] = "finish";
|
c@17
|
1048 jo["success"] = true;
|
c@17
|
1049 jo["errorText"] = "";
|
c@44
|
1050 jo["content"] = fromFeatureSet(resp.features, serialisation);
|
c@17
|
1051 return json11::Json(jo);
|
c@17
|
1052 }
|
c@24
|
1053
|
c@41
|
1054 static json11::Json
|
c@41
|
1055 fromException(const std::exception &e, RRType responseType) {
|
c@41
|
1056
|
c@41
|
1057 json11::Json::object jo;
|
c@43
|
1058 std::string type;
|
c@41
|
1059
|
c@43
|
1060 if (responseType == RRType::List) type = "list";
|
c@43
|
1061 else if (responseType == RRType::Load) type = "load";
|
c@43
|
1062 else if (responseType == RRType::Configure) type = "configure";
|
c@43
|
1063 else if (responseType == RRType::Process) type = "process";
|
c@43
|
1064 else if (responseType == RRType::Finish) type = "finish";
|
c@43
|
1065 else type = "invalid";
|
c@41
|
1066
|
c@43
|
1067 jo["type"] = type;
|
c@41
|
1068 jo["success"] = false;
|
c@43
|
1069 jo["errorText"] = std::string("exception caught: ") +
|
c@43
|
1070 type + " request: " + e.what();
|
c@41
|
1071 return json11::Json(jo);
|
c@41
|
1072 }
|
c@41
|
1073
|
c@24
|
1074 private: // go private briefly for a couple of helper functions
|
c@24
|
1075
|
c@24
|
1076 static void
|
c@24
|
1077 checkTypeField(json11::Json j, std::string expected) {
|
c@24
|
1078 if (!j["type"].is_string()) {
|
c@24
|
1079 throw Failure("string expected for type");
|
c@24
|
1080 }
|
c@24
|
1081 if (j["type"].string_value() != expected) {
|
c@24
|
1082 throw Failure("expected value \"" + expected + "\" for type");
|
c@24
|
1083 }
|
c@24
|
1084 }
|
c@24
|
1085
|
c@24
|
1086 static bool
|
c@24
|
1087 successful(json11::Json j) {
|
c@24
|
1088 if (!j["success"].is_bool()) {
|
c@24
|
1089 throw Failure("bool expected for success");
|
c@24
|
1090 }
|
c@24
|
1091 return j["success"].bool_value();
|
c@24
|
1092 }
|
c@24
|
1093
|
c@24
|
1094 public:
|
c@25
|
1095 static RRType
|
c@25
|
1096 getRequestResponseType(json11::Json j) {
|
c@25
|
1097
|
c@25
|
1098 if (!j["type"].is_string()) {
|
c@25
|
1099 throw Failure("string expected for type");
|
c@25
|
1100 }
|
c@25
|
1101
|
c@25
|
1102 std::string type = j["type"].string_value();
|
c@25
|
1103
|
c@25
|
1104 if (type == "list") return RRType::List;
|
c@25
|
1105 else if (type == "load") return RRType::Load;
|
c@25
|
1106 else if (type == "configure") return RRType::Configure;
|
c@25
|
1107 else if (type == "process") return RRType::Process;
|
c@25
|
1108 else if (type == "finish") return RRType::Finish;
|
c@25
|
1109 else {
|
c@25
|
1110 throw Failure("unknown or unexpected request/response type \"" +
|
c@25
|
1111 type + "\"");
|
c@25
|
1112 }
|
c@25
|
1113 }
|
c@44
|
1114
|
c@24
|
1115 static void
|
c@24
|
1116 toVampRequest_List(json11::Json j) {
|
c@24
|
1117
|
c@24
|
1118 checkTypeField(j, "list");
|
c@24
|
1119 }
|
c@24
|
1120
|
c@24
|
1121 static std::vector<Vamp::HostExt::PluginStaticData>
|
c@24
|
1122 toVampResponse_List(json11::Json j) {
|
c@24
|
1123
|
c@24
|
1124 std::vector<Vamp::HostExt::PluginStaticData> arr;
|
c@24
|
1125 if (successful(j)) {
|
c@24
|
1126 for (const auto &a: j["content"]["plugins"].array_items()) {
|
c@24
|
1127 arr.push_back(toPluginStaticData(a));
|
c@24
|
1128 }
|
c@24
|
1129 }
|
c@24
|
1130 return arr;
|
c@24
|
1131 }
|
c@24
|
1132
|
c@24
|
1133 static Vamp::HostExt::LoadRequest
|
c@24
|
1134 toVampRequest_Load(json11::Json j) {
|
c@24
|
1135
|
c@24
|
1136 checkTypeField(j, "load");
|
c@24
|
1137 return toLoadRequest(j["content"]);
|
c@24
|
1138 }
|
c@24
|
1139
|
c@24
|
1140 static Vamp::HostExt::LoadResponse
|
c@39
|
1141 toVampResponse_Load(json11::Json j, const PluginHandleMapper &mapper) {
|
c@24
|
1142
|
c@24
|
1143 Vamp::HostExt::LoadResponse resp;
|
c@24
|
1144 if (successful(j)) {
|
c@24
|
1145 resp = toLoadResponse(j["content"], mapper);
|
c@24
|
1146 }
|
c@24
|
1147 return resp;
|
c@24
|
1148 }
|
c@24
|
1149
|
c@24
|
1150 static Vamp::HostExt::ConfigurationRequest
|
c@39
|
1151 toVampRequest_Configure(json11::Json j, const PluginHandleMapper &mapper) {
|
c@24
|
1152
|
c@24
|
1153 checkTypeField(j, "configure");
|
c@24
|
1154 return toConfigurationRequest(j["content"], mapper);
|
c@24
|
1155 }
|
c@24
|
1156
|
c@24
|
1157 static Vamp::HostExt::ConfigurationResponse
|
c@24
|
1158 toVampResponse_Configure(json11::Json j) {
|
c@24
|
1159
|
c@24
|
1160 Vamp::HostExt::ConfigurationResponse resp;
|
c@24
|
1161 if (successful(j)) {
|
c@24
|
1162 resp = toConfigurationResponse(j["content"]);
|
c@24
|
1163 }
|
c@24
|
1164 return resp;
|
c@24
|
1165 }
|
c@24
|
1166
|
c@24
|
1167 static Vamp::HostExt::ProcessRequest
|
c@44
|
1168 toVampRequest_Process(json11::Json j, const PluginHandleMapper &mapper,
|
c@44
|
1169 BufferSerialisation &serialisation) {
|
c@24
|
1170
|
c@24
|
1171 checkTypeField(j, "process");
|
c@44
|
1172 return toProcessRequest(j["content"], mapper, serialisation);
|
c@24
|
1173 }
|
c@24
|
1174
|
c@24
|
1175 static Vamp::HostExt::ProcessResponse
|
c@44
|
1176 toVampResponse_Process(json11::Json j, BufferSerialisation &serialisation) {
|
c@24
|
1177
|
c@24
|
1178 Vamp::HostExt::ProcessResponse resp;
|
c@24
|
1179 if (successful(j)) {
|
c@44
|
1180 resp.features = toFeatureSet(j["content"], serialisation);
|
c@24
|
1181 }
|
c@24
|
1182 return resp;
|
c@24
|
1183 }
|
c@24
|
1184
|
c@24
|
1185 static Vamp::Plugin *
|
c@39
|
1186 toVampRequest_Finish(json11::Json j, const PluginHandleMapper &mapper) {
|
c@24
|
1187
|
c@24
|
1188 checkTypeField(j, "finish");
|
c@24
|
1189 return mapper.handleToPlugin(j["content"]["pluginHandle"].int_value());
|
c@24
|
1190 }
|
c@24
|
1191
|
c@24
|
1192 static Vamp::HostExt::ProcessResponse
|
c@44
|
1193 toVampResponse_Finish(json11::Json j, BufferSerialisation &serialisation) {
|
c@24
|
1194
|
c@24
|
1195 Vamp::HostExt::ProcessResponse resp;
|
c@24
|
1196 if (successful(j)) {
|
c@44
|
1197 resp.features = toFeatureSet(j["content"], serialisation);
|
c@24
|
1198 }
|
c@24
|
1199 return resp;
|
c@24
|
1200 }
|
c@5
|
1201 };
|
c@5
|
1202
|
c@10
|
1203 }
|
c@5
|
1204
|
c@5
|
1205 #endif
|