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