c@5
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
c@5
|
2
|
c@5
|
3 #ifndef VAMP_JSON_H
|
c@5
|
4 #define VAMP_JSON_H
|
c@5
|
5
|
c@5
|
6 #include <vector>
|
c@5
|
7 #include <string>
|
c@5
|
8 #include <sstream>
|
c@5
|
9 #include <stdexcept>
|
c@5
|
10
|
c@5
|
11 #include <json11/json11.hpp>
|
c@5
|
12 #include <base-n/include/basen.hpp>
|
c@5
|
13
|
c@5
|
14 #include <vamp-hostsdk/Plugin.h>
|
c@5
|
15 #include <vamp-hostsdk/PluginLoader.h>
|
c@5
|
16
|
c@10
|
17 #include "bits/PluginHandleMapper.h"
|
c@10
|
18
|
c@10
|
19 namespace vampipe {
|
c@10
|
20
|
c@6
|
21 /**
|
c@6
|
22 * Convert the structures laid out in the Vamp SDK classes into JSON
|
c@6
|
23 * (and back again) following the schema in the vamp-json-schema
|
c@6
|
24 * project repo.
|
c@6
|
25 */
|
c@5
|
26 class VampJson
|
c@5
|
27 {
|
c@5
|
28 public:
|
c@5
|
29 class Failure : virtual public std::runtime_error {
|
c@5
|
30 public:
|
c@5
|
31 Failure(std::string s) : runtime_error(s) { }
|
c@5
|
32 };
|
c@5
|
33
|
c@5
|
34 template <typename T>
|
c@5
|
35 static json11::Json
|
c@5
|
36 fromBasicDescriptor(const T &t) {
|
c@5
|
37 return json11::Json::object {
|
c@5
|
38 { "identifier", t.identifier },
|
c@5
|
39 { "name", t.name },
|
c@5
|
40 { "description", t.description }
|
c@5
|
41 };
|
c@5
|
42 }
|
c@5
|
43
|
c@5
|
44 template <typename T>
|
c@5
|
45 static void
|
c@5
|
46 toBasicDescriptor(json11::Json j, T &t) {
|
c@5
|
47 if (!j.is_object()) {
|
c@5
|
48 throw Failure("object expected for basic descriptor content");
|
c@5
|
49 }
|
c@5
|
50 if (!j["identifier"].is_string()) {
|
c@5
|
51 throw Failure("string expected for identifier");
|
c@5
|
52 }
|
c@5
|
53 t.identifier = j["identifier"].string_value();
|
c@5
|
54 t.name = j["name"].string_value();
|
c@5
|
55 t.description = j["description"].string_value();
|
c@5
|
56 }
|
c@5
|
57
|
c@5
|
58 template <typename T>
|
c@5
|
59 static json11::Json
|
c@5
|
60 fromValueExtents(const T &t) {
|
c@5
|
61 return json11::Json::object {
|
c@5
|
62 { "min", t.minValue },
|
c@5
|
63 { "max", t.maxValue }
|
c@5
|
64 };
|
c@5
|
65 }
|
c@5
|
66
|
c@5
|
67 template <typename T>
|
c@5
|
68 static bool
|
c@5
|
69 toValueExtents(json11::Json j, T &t) {
|
c@5
|
70 if (j["extents"].is_null()) {
|
c@5
|
71 return false;
|
c@5
|
72 } else if (j["extents"].is_object()) {
|
c@5
|
73 if (j["extents"]["min"].is_number() &&
|
c@5
|
74 j["extents"]["max"].is_number()) {
|
c@5
|
75 t.minValue = j["extents"]["min"].number_value();
|
c@5
|
76 t.maxValue = j["extents"]["max"].number_value();
|
c@5
|
77 return true;
|
c@5
|
78 } else {
|
c@5
|
79 throw Failure("numbers expected for min and max");
|
c@5
|
80 }
|
c@5
|
81 } else {
|
c@5
|
82 throw Failure("object expected for extents (if present)");
|
c@5
|
83 }
|
c@5
|
84 }
|
c@5
|
85
|
c@5
|
86 static json11::Json
|
c@5
|
87 fromRealTime(const Vamp::RealTime &r) {
|
c@5
|
88 return json11::Json::object {
|
c@5
|
89 { "s", r.sec },
|
c@5
|
90 { "n", r.nsec }
|
c@5
|
91 };
|
c@5
|
92 }
|
c@5
|
93
|
c@5
|
94 static Vamp::RealTime
|
c@5
|
95 toRealTime(json11::Json j) {
|
c@5
|
96 json11::Json sec = j["s"];
|
c@5
|
97 json11::Json nsec = j["n"];
|
c@5
|
98 if (!sec.is_number() || !nsec.is_number()) {
|
c@5
|
99 throw Failure("invalid Vamp::RealTime object " + j.dump());
|
c@5
|
100 }
|
c@5
|
101 return Vamp::RealTime(sec.int_value(), nsec.int_value());
|
c@5
|
102 }
|
c@5
|
103
|
c@5
|
104 static std::string
|
c@5
|
105 fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType type) {
|
c@5
|
106 switch (type) {
|
c@5
|
107 case Vamp::Plugin::OutputDescriptor::OneSamplePerStep:
|
c@5
|
108 return "OneSamplePerStep";
|
c@5
|
109 case Vamp::Plugin::OutputDescriptor::FixedSampleRate:
|
c@5
|
110 return "FixedSampleRate";
|
c@5
|
111 case Vamp::Plugin::OutputDescriptor::VariableSampleRate:
|
c@5
|
112 return "VariableSampleRate";
|
c@5
|
113 }
|
c@5
|
114 return "";
|
c@5
|
115 }
|
c@5
|
116
|
c@5
|
117 static Vamp::Plugin::OutputDescriptor::SampleType
|
c@5
|
118 toSampleType(std::string text) {
|
c@5
|
119 if (text == "OneSamplePerStep") {
|
c@5
|
120 return Vamp::Plugin::OutputDescriptor::OneSamplePerStep;
|
c@5
|
121 } else if (text == "FixedSampleRate") {
|
c@5
|
122 return Vamp::Plugin::OutputDescriptor::FixedSampleRate;
|
c@5
|
123 } else if (text == "VariableSampleRate") {
|
c@5
|
124 return Vamp::Plugin::OutputDescriptor::VariableSampleRate;
|
c@5
|
125 } else {
|
c@5
|
126 throw Failure("invalid sample type string: " + text);
|
c@5
|
127 }
|
c@5
|
128 }
|
c@5
|
129
|
c@5
|
130 static json11::Json
|
c@5
|
131 fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) {
|
c@5
|
132 json11::Json::object jo {
|
c@5
|
133 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
134 { "unit", desc.unit },
|
c@5
|
135 { "sampleType", fromSampleType(desc.sampleType) },
|
c@5
|
136 { "sampleRate", desc.sampleRate },
|
c@5
|
137 { "hasDuration", desc.hasDuration }
|
c@5
|
138 };
|
c@5
|
139 if (desc.hasFixedBinCount) {
|
c@5
|
140 jo["binCount"] = int(desc.binCount);
|
c@5
|
141 jo["binNames"] = json11::Json::array
|
c@5
|
142 (desc.binNames.begin(), desc.binNames.end());
|
c@5
|
143 }
|
c@5
|
144 if (desc.hasKnownExtents) {
|
c@5
|
145 jo["extents"] = fromValueExtents(desc);
|
c@5
|
146 }
|
c@5
|
147 if (desc.isQuantized) {
|
c@5
|
148 jo["quantizeStep"] = desc.quantizeStep;
|
c@5
|
149 }
|
c@5
|
150 return json11::Json(jo);
|
c@5
|
151 }
|
c@12
|
152
|
c@5
|
153 static Vamp::Plugin::OutputDescriptor
|
c@5
|
154 toOutputDescriptor(json11::Json j) {
|
c@5
|
155
|
c@5
|
156 Vamp::Plugin::OutputDescriptor od;
|
c@5
|
157 if (!j.is_object()) {
|
c@5
|
158 throw Failure("object expected for output descriptor");
|
c@5
|
159 }
|
c@5
|
160
|
c@5
|
161 toBasicDescriptor(j["basic"], od);
|
c@5
|
162
|
c@5
|
163 od.unit = j["unit"].string_value();
|
c@5
|
164
|
c@5
|
165 od.sampleType = toSampleType(j["sampleType"].string_value());
|
c@5
|
166
|
c@5
|
167 if (!j["sampleRate"].is_number()) {
|
c@5
|
168 throw Failure("number expected for sample rate");
|
c@5
|
169 }
|
c@5
|
170 od.sampleRate = j["sampleRate"].number_value();
|
c@5
|
171 od.hasDuration = j["hasDuration"].bool_value();
|
c@5
|
172
|
c@5
|
173 if (j["binCount"].is_number() && j["binCount"].int_value() > 0) {
|
c@5
|
174 od.hasFixedBinCount = true;
|
c@5
|
175 od.binCount = j["binCount"].int_value();
|
c@5
|
176 for (auto &n: j["binNames"].array_items()) {
|
c@5
|
177 if (!n.is_string()) {
|
c@5
|
178 throw Failure("string expected for bin name");
|
c@5
|
179 }
|
c@5
|
180 od.binNames.push_back(n.string_value());
|
c@5
|
181 }
|
c@5
|
182 } else {
|
c@5
|
183 od.hasFixedBinCount = false;
|
c@5
|
184 }
|
c@5
|
185
|
c@5
|
186 bool extentsPresent = toValueExtents(j, od);
|
c@5
|
187 od.hasKnownExtents = extentsPresent;
|
c@5
|
188
|
c@5
|
189 if (j["quantizeStep"].is_number()) {
|
c@5
|
190 od.isQuantized = true;
|
c@5
|
191 od.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
192 } else {
|
c@5
|
193 od.isQuantized = false;
|
c@5
|
194 }
|
c@5
|
195
|
c@5
|
196 return od;
|
c@5
|
197 }
|
c@5
|
198
|
c@5
|
199 static json11::Json
|
c@5
|
200 fromParameterDescriptor(const Vamp::PluginBase::ParameterDescriptor &desc) {
|
c@5
|
201
|
c@5
|
202 json11::Json::object jo {
|
c@5
|
203 { "basic", fromBasicDescriptor(desc) },
|
c@5
|
204 { "unit", desc.unit },
|
c@5
|
205 { "extents", fromValueExtents(desc) },
|
c@5
|
206 { "defaultValue", desc.defaultValue },
|
c@5
|
207 { "valueNames", json11::Json::array
|
c@5
|
208 (desc.valueNames.begin(), desc.valueNames.end()) }
|
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@5
|
215
|
c@5
|
216 static Vamp::PluginBase::ParameterDescriptor
|
c@5
|
217 toParameterDescriptor(json11::Json j) {
|
c@5
|
218
|
c@5
|
219 Vamp::PluginBase::ParameterDescriptor pd;
|
c@5
|
220 if (!j.is_object()) {
|
c@5
|
221 throw Failure("object expected for parameter descriptor");
|
c@5
|
222 }
|
c@5
|
223
|
c@5
|
224 toBasicDescriptor(j["basic"], pd);
|
c@5
|
225
|
c@5
|
226 pd.unit = j["unit"].string_value();
|
c@5
|
227
|
c@5
|
228 bool extentsPresent = toValueExtents(j, pd);
|
c@5
|
229 if (!extentsPresent) {
|
c@5
|
230 throw Failure("extents must be present in parameter descriptor");
|
c@5
|
231 }
|
c@5
|
232
|
c@5
|
233 if (!j["defaultValue"].is_number()) {
|
c@5
|
234 throw Failure("number expected for default value");
|
c@5
|
235 }
|
c@5
|
236
|
c@5
|
237 pd.defaultValue = j["defaultValue"].number_value();
|
c@5
|
238
|
c@5
|
239 pd.valueNames.clear();
|
c@5
|
240 for (auto &n: j["valueNames"].array_items()) {
|
c@5
|
241 if (!n.is_string()) {
|
c@5
|
242 throw Failure("string expected for value name");
|
c@5
|
243 }
|
c@5
|
244 pd.valueNames.push_back(n.string_value());
|
c@5
|
245 }
|
c@5
|
246
|
c@5
|
247 if (j["quantizeStep"].is_number()) {
|
c@5
|
248 pd.isQuantized = true;
|
c@5
|
249 pd.quantizeStep = j["quantizeStep"].number_value();
|
c@5
|
250 } else {
|
c@5
|
251 pd.isQuantized = false;
|
c@5
|
252 }
|
c@5
|
253
|
c@5
|
254 return pd;
|
c@5
|
255 }
|
c@5
|
256
|
c@5
|
257 static std::string
|
c@5
|
258 fromFloatBuffer(const float *buffer, size_t nfloats) {
|
c@5
|
259 // must use char pointers, otherwise the converter will only
|
c@5
|
260 // encode every 4th byte (as it will count up in float* steps)
|
c@5
|
261 const char *start = reinterpret_cast<const char *>(buffer);
|
c@5
|
262 const char *end = reinterpret_cast<const char *>(buffer + nfloats);
|
c@5
|
263 std::string encoded;
|
c@5
|
264 bn::encode_b64(start, end, back_inserter(encoded));
|
c@5
|
265 return encoded;
|
c@5
|
266 }
|
c@5
|
267
|
c@5
|
268 static std::vector<float>
|
c@5
|
269 toFloatBuffer(std::string encoded) {
|
c@5
|
270 std::string decoded;
|
c@5
|
271 bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded));
|
c@5
|
272 const float *buffer = reinterpret_cast<const float *>(decoded.c_str());
|
c@5
|
273 size_t n = decoded.size() / sizeof(float);
|
c@5
|
274 return std::vector<float>(buffer, buffer + n);
|
c@5
|
275 }
|
c@5
|
276
|
c@5
|
277 static json11::Json
|
c@5
|
278 fromFeature(const Vamp::Plugin::Feature &f) {
|
c@5
|
279
|
c@5
|
280 json11::Json::object jo;
|
c@5
|
281 if (f.values.size() > 0) {
|
c@5
|
282 jo["b64values"] = fromFloatBuffer(f.values.data(), f.values.size());
|
c@5
|
283 }
|
c@5
|
284 if (f.label != "") {
|
c@5
|
285 jo["label"] = f.label;
|
c@5
|
286 }
|
c@5
|
287 if (f.hasTimestamp) {
|
c@5
|
288 jo["timestamp"] = fromRealTime(f.timestamp);
|
c@5
|
289 }
|
c@5
|
290 if (f.hasDuration) {
|
c@5
|
291 jo["duration"] = fromRealTime(f.duration);
|
c@5
|
292 }
|
c@5
|
293 return json11::Json(jo);
|
c@5
|
294 }
|
c@5
|
295
|
c@5
|
296 static Vamp::Plugin::Feature
|
c@5
|
297 toFeature(json11::Json j) {
|
c@5
|
298
|
c@5
|
299 Vamp::Plugin::Feature f;
|
c@5
|
300 if (!j.is_object()) {
|
c@5
|
301 throw Failure("object expected for feature");
|
c@5
|
302 }
|
c@5
|
303 if (j["timestamp"].is_object()) {
|
c@5
|
304 f.timestamp = toRealTime(j["timestamp"]);
|
c@5
|
305 f.hasTimestamp = true;
|
c@5
|
306 }
|
c@5
|
307 if (j["duration"].is_object()) {
|
c@5
|
308 f.duration = toRealTime(j["duration"]);
|
c@5
|
309 f.hasDuration = true;
|
c@5
|
310 }
|
c@5
|
311 if (j["b64values"].is_string()) {
|
c@5
|
312 f.values = toFloatBuffer(j["b64values"].string_value());
|
c@5
|
313 } else if (j["values"].is_array()) {
|
c@5
|
314 for (auto v : j["values"].array_items()) {
|
c@5
|
315 f.values.push_back(v.number_value());
|
c@5
|
316 }
|
c@5
|
317 }
|
c@5
|
318 f.label = j["label"].string_value();
|
c@5
|
319 return f;
|
c@5
|
320 }
|
c@5
|
321
|
c@5
|
322 static json11::Json
|
c@5
|
323 fromFeatureSet(const Vamp::Plugin::FeatureSet &fs) {
|
c@5
|
324
|
c@5
|
325 json11::Json::object jo;
|
c@5
|
326 for (const auto &fsi : fs) {
|
c@5
|
327 std::vector<json11::Json> fj;
|
c@5
|
328 for (const Vamp::Plugin::Feature &f: fsi.second) {
|
c@5
|
329 fj.push_back(fromFeature(f));
|
c@5
|
330 }
|
c@5
|
331 std::stringstream sstr;
|
c@5
|
332 sstr << fsi.first;
|
c@5
|
333 std::string n = sstr.str();
|
c@5
|
334 jo[n] = fj;
|
c@5
|
335 }
|
c@5
|
336 return json11::Json(jo);
|
c@5
|
337 }
|
c@5
|
338
|
c@5
|
339 static Vamp::Plugin::FeatureList
|
c@5
|
340 toFeatureList(json11::Json j) {
|
c@5
|
341
|
c@5
|
342 Vamp::Plugin::FeatureList fl;
|
c@5
|
343 if (!j.is_array()) {
|
c@5
|
344 throw Failure("array expected for feature list");
|
c@5
|
345 }
|
c@5
|
346 for (const json11::Json &fj : j.array_items()) {
|
c@5
|
347 fl.push_back(toFeature(fj));
|
c@5
|
348 }
|
c@5
|
349 return fl;
|
c@5
|
350 }
|
c@5
|
351
|
c@5
|
352 static Vamp::Plugin::FeatureSet
|
c@5
|
353 toFeatureSet(json11::Json j) {
|
c@5
|
354
|
c@5
|
355 Vamp::Plugin::FeatureSet fs;
|
c@5
|
356 if (!j.is_object()) {
|
c@5
|
357 throw Failure("object expected for feature set");
|
c@5
|
358 }
|
c@5
|
359 for (auto &entry : j.object_items()) {
|
c@5
|
360 std::string nstr = entry.first;
|
c@5
|
361 size_t count = 0;
|
c@5
|
362 int n = stoi(nstr, &count);
|
c@5
|
363 if (n < 0 || fs.find(n) != fs.end() || count < nstr.size()) {
|
c@5
|
364 throw Failure("invalid or duplicate numerical index for output");
|
c@5
|
365 }
|
c@5
|
366 fs[n] = toFeatureList(entry.second);
|
c@5
|
367 }
|
c@5
|
368 return fs;
|
c@5
|
369 }
|
c@5
|
370
|
c@5
|
371 static std::string
|
c@5
|
372 fromInputDomain(Vamp::Plugin::InputDomain domain) {
|
c@5
|
373
|
c@5
|
374 switch (domain) {
|
c@5
|
375 case Vamp::Plugin::TimeDomain:
|
c@5
|
376 return "TimeDomain";
|
c@5
|
377 case Vamp::Plugin::FrequencyDomain:
|
c@5
|
378 return "FrequencyDomain";
|
c@5
|
379 }
|
c@5
|
380 return "";
|
c@5
|
381 }
|
c@5
|
382
|
c@5
|
383 static Vamp::Plugin::InputDomain
|
c@5
|
384 toInputDomain(std::string text) {
|
c@5
|
385
|
c@5
|
386 if (text == "TimeDomain") {
|
c@5
|
387 return Vamp::Plugin::TimeDomain;
|
c@5
|
388 } else if (text == "FrequencyDomain") {
|
c@5
|
389 return Vamp::Plugin::FrequencyDomain;
|
c@5
|
390 } else {
|
c@5
|
391 throw Failure("invalid input domain string: " + text);
|
c@5
|
392 }
|
c@5
|
393 }
|
c@5
|
394
|
c@5
|
395 static json11::Json
|
c@5
|
396 fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) {
|
c@5
|
397
|
c@5
|
398 json11::Json::object jo;
|
c@5
|
399 jo["pluginKey"] = d.pluginKey;
|
c@5
|
400 jo["basic"] = fromBasicDescriptor(d.basic);
|
c@5
|
401 jo["maker"] = d.maker;
|
c@5
|
402 jo["copyright"] = d.copyright;
|
c@5
|
403 jo["pluginVersion"] = d.pluginVersion;
|
c@5
|
404
|
c@5
|
405 json11::Json::array cat;
|
c@5
|
406 for (const std::string &c: d.category) cat.push_back(c);
|
c@5
|
407 jo["category"] = cat;
|
c@5
|
408
|
c@5
|
409 jo["minChannelCount"] = d.minChannelCount;
|
c@5
|
410 jo["maxChannelCount"] = d.maxChannelCount;
|
c@5
|
411
|
c@5
|
412 json11::Json::array params;
|
c@5
|
413 Vamp::PluginBase::ParameterList vparams = d.parameters;
|
c@5
|
414 for (auto &p: vparams) params.push_back(fromParameterDescriptor(p));
|
c@5
|
415 jo["parameters"] = params;
|
c@5
|
416
|
c@5
|
417 json11::Json::array progs;
|
c@5
|
418 Vamp::PluginBase::ProgramList vprogs = d.programs;
|
c@5
|
419 for (auto &p: vprogs) progs.push_back(p);
|
c@5
|
420 jo["programs"] = progs;
|
c@5
|
421
|
c@5
|
422 jo["inputDomain"] = fromInputDomain(d.inputDomain);
|
c@5
|
423
|
c@5
|
424 json11::Json::array outinfo;
|
c@5
|
425 auto vouts = d.basicOutputInfo;
|
c@5
|
426 for (auto &o: vouts) outinfo.push_back(fromBasicDescriptor(o));
|
c@5
|
427 jo["basicOutputInfo"] = outinfo;
|
c@5
|
428
|
c@5
|
429 return json11::Json(jo);
|
c@5
|
430 }
|
c@5
|
431
|
c@5
|
432 static Vamp::HostExt::PluginStaticData
|
c@5
|
433 toPluginStaticData(json11::Json j) {
|
c@5
|
434
|
c@5
|
435 std::string err;
|
c@5
|
436 if (!j.has_shape({
|
c@5
|
437 { "pluginKey", json11::Json::STRING },
|
c@5
|
438 { "pluginVersion", json11::Json::NUMBER },
|
c@5
|
439 { "minChannelCount", json11::Json::NUMBER },
|
c@5
|
440 { "maxChannelCount", json11::Json::NUMBER },
|
c@5
|
441 { "inputDomain", json11::Json::STRING }}, err)) {
|
c@5
|
442 throw Failure("malformed plugin static data: " + err);
|
c@5
|
443 }
|
c@5
|
444
|
c@5
|
445 if (!j["basicOutputInfo"].is_array()) {
|
c@5
|
446 throw Failure("array expected for basic output info");
|
c@5
|
447 }
|
c@5
|
448
|
c@5
|
449 if (!j["maker"].is_null() &&
|
c@5
|
450 !j["maker"].is_string()) {
|
c@5
|
451 throw Failure("string expected for maker");
|
c@5
|
452 }
|
c@5
|
453
|
c@5
|
454 if (!j["copyright"].is_null() &&
|
c@5
|
455 !j["copyright"].is_string()) {
|
c@5
|
456 throw Failure("string expected for copyright");
|
c@5
|
457 }
|
c@5
|
458
|
c@5
|
459 if (!j["category"].is_null() &&
|
c@5
|
460 !j["category"].is_array()) {
|
c@5
|
461 throw Failure("array expected for category");
|
c@5
|
462 }
|
c@5
|
463
|
c@5
|
464 if (!j["parameters"].is_null() &&
|
c@5
|
465 !j["parameters"].is_array()) {
|
c@5
|
466 throw Failure("array expected for parameters");
|
c@5
|
467 }
|
c@5
|
468
|
c@5
|
469 if (!j["programs"].is_null() &&
|
c@5
|
470 !j["programs"].is_array()) {
|
c@5
|
471 throw Failure("array expected for programs");
|
c@5
|
472 }
|
c@5
|
473
|
c@5
|
474 if (!j["inputDomain"].is_null() &&
|
c@5
|
475 !j["inputDomain"].is_string()) {
|
c@5
|
476 throw Failure("string expected for inputDomain");
|
c@5
|
477 }
|
c@5
|
478
|
c@5
|
479 if (!j["basicOutputInfo"].is_null() &&
|
c@5
|
480 !j["basicOutputInfo"].is_array()) {
|
c@5
|
481 throw Failure("array expected for basicOutputInfo");
|
c@5
|
482 }
|
c@5
|
483
|
c@5
|
484 Vamp::HostExt::PluginStaticData psd;
|
c@5
|
485
|
c@5
|
486 psd.pluginKey = j["pluginKey"].string_value();
|
c@5
|
487
|
c@5
|
488 toBasicDescriptor(j["basic"], psd.basic);
|
c@5
|
489
|
c@5
|
490 psd.maker = j["maker"].string_value();
|
c@5
|
491 psd.copyright = j["copyright"].string_value();
|
c@5
|
492 psd.pluginVersion = j["pluginVersion"].int_value();
|
c@5
|
493
|
c@5
|
494 for (const auto &c : j["category"].array_items()) {
|
c@5
|
495 if (!c.is_string()) {
|
c@5
|
496 throw Failure("strings expected in category array");
|
c@5
|
497 }
|
c@5
|
498 psd.category.push_back(c.string_value());
|
c@5
|
499 }
|
c@5
|
500
|
c@5
|
501 psd.minChannelCount = j["minChannelCount"].int_value();
|
c@5
|
502 psd.maxChannelCount = j["maxChannelCount"].int_value();
|
c@5
|
503
|
c@5
|
504 for (const auto &p : j["parameters"].array_items()) {
|
c@5
|
505 auto pd = toParameterDescriptor(p);
|
c@5
|
506 psd.parameters.push_back(pd);
|
c@5
|
507 }
|
c@5
|
508
|
c@5
|
509 for (const auto &p : j["programs"].array_items()) {
|
c@5
|
510 if (!p.is_string()) {
|
c@5
|
511 throw Failure("strings expected in programs array");
|
c@5
|
512 }
|
c@5
|
513 psd.programs.push_back(p.string_value());
|
c@5
|
514 }
|
c@5
|
515
|
c@5
|
516 psd.inputDomain = toInputDomain(j["inputDomain"].string_value());
|
c@5
|
517
|
c@5
|
518 for (const auto &bo : j["basicOutputInfo"].array_items()) {
|
c@5
|
519 Vamp::HostExt::PluginStaticData::Basic b;
|
c@5
|
520 toBasicDescriptor(bo, b);
|
c@5
|
521 psd.basicOutputInfo.push_back(b);
|
c@5
|
522 }
|
c@5
|
523
|
c@5
|
524 return psd;
|
c@5
|
525 }
|
c@5
|
526
|
c@5
|
527 static json11::Json
|
c@5
|
528 fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) {
|
c@5
|
529
|
c@5
|
530 json11::Json::object jo;
|
c@5
|
531
|
c@5
|
532 json11::Json::object paramValues;
|
c@5
|
533 for (auto &vp: c.parameterValues) {
|
c@5
|
534 paramValues[vp.first] = vp.second;
|
c@5
|
535 }
|
c@5
|
536 jo["parameterValues"] = paramValues;
|
c@5
|
537
|
c@5
|
538 if (c.currentProgram != "") {
|
c@5
|
539 jo["currentProgram"] = c.currentProgram;
|
c@5
|
540 }
|
c@5
|
541
|
c@5
|
542 jo["channelCount"] = c.channelCount;
|
c@5
|
543 jo["stepSize"] = c.stepSize;
|
c@5
|
544 jo["blockSize"] = c.blockSize;
|
c@5
|
545
|
c@5
|
546 return json11::Json(jo);
|
c@5
|
547 }
|
c@5
|
548
|
c@5
|
549 static Vamp::HostExt::PluginConfiguration
|
c@5
|
550 toPluginConfiguration(json11::Json j) {
|
c@5
|
551
|
c@5
|
552 std::string err;
|
c@5
|
553 if (!j.has_shape({
|
c@5
|
554 { "channelCount", json11::Json::NUMBER },
|
c@5
|
555 { "stepSize", json11::Json::NUMBER },
|
c@5
|
556 { "blockSize", json11::Json::NUMBER } }, err)) {
|
c@5
|
557 throw Failure("malformed plugin configuration: " + err);
|
c@5
|
558 }
|
c@5
|
559
|
c@5
|
560 if (!j["parameterValues"].is_null() &&
|
c@5
|
561 !j["parameterValues"].is_object()) {
|
c@5
|
562 throw Failure("object expected for parameter values");
|
c@5
|
563 }
|
c@5
|
564
|
c@5
|
565 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
566 if (!pv.second.is_number()) {
|
c@5
|
567 throw Failure("number expected for parameter value");
|
c@5
|
568 }
|
c@5
|
569 }
|
c@5
|
570
|
c@5
|
571 if (!j["currentProgram"].is_null() &&
|
c@5
|
572 !j["currentProgram"].is_string()) {
|
c@5
|
573 throw Failure("string expected for program name");
|
c@5
|
574 }
|
c@5
|
575
|
c@5
|
576 Vamp::HostExt::PluginConfiguration config;
|
c@5
|
577
|
c@5
|
578 config.channelCount = j["channelCount"].number_value();
|
c@5
|
579 config.stepSize = j["stepSize"].number_value();
|
c@5
|
580 config.blockSize = j["blockSize"].number_value();
|
c@5
|
581
|
c@5
|
582 for (auto &pv : j["parameterValues"].object_items()) {
|
c@5
|
583 config.parameterValues[pv.first] = pv.second.number_value();
|
c@5
|
584 }
|
c@5
|
585
|
c@5
|
586 if (j["currentProgram"].is_string()) {
|
c@5
|
587 config.currentProgram = j["currentProgram"].string_value();
|
c@5
|
588 }
|
c@5
|
589
|
c@5
|
590 return config;
|
c@5
|
591 }
|
c@5
|
592
|
c@5
|
593 static json11::Json
|
c@5
|
594 fromAdapterFlags(int flags) {
|
c@5
|
595
|
c@5
|
596 json11::Json::array arr;
|
c@5
|
597
|
c@5
|
598 if (flags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) {
|
c@5
|
599 arr.push_back("AdaptInputDomain");
|
c@5
|
600 }
|
c@5
|
601 if (flags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) {
|
c@5
|
602 arr.push_back("AdaptChannelCount");
|
c@5
|
603 }
|
c@5
|
604 if (flags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) {
|
c@5
|
605 arr.push_back("AdaptBufferSize");
|
c@5
|
606 }
|
c@5
|
607
|
c@5
|
608 return json11::Json(arr);
|
c@5
|
609 }
|
c@5
|
610
|
c@5
|
611 static Vamp::HostExt::PluginLoader::AdapterFlags
|
c@5
|
612 toAdapterFlags(json11::Json j) {
|
c@5
|
613
|
c@5
|
614 if (!j.is_array()) {
|
c@5
|
615 throw Failure("array expected for adapter flags");
|
c@5
|
616 }
|
c@5
|
617 int flags = 0x0;
|
c@5
|
618
|
c@5
|
619 for (auto &jj: j.array_items()) {
|
c@5
|
620 if (!jj.is_string()) {
|
c@5
|
621 throw Failure("string expected for adapter flag");
|
c@5
|
622 }
|
c@5
|
623 std::string text = jj.string_value();
|
c@5
|
624 if (text == "AdaptInputDomain") {
|
c@5
|
625 flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN;
|
c@5
|
626 } else if (text == "AdaptChannelCount") {
|
c@5
|
627 flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT;
|
c@5
|
628 } else if (text == "AdaptBufferSize") {
|
c@5
|
629 flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE;
|
c@5
|
630 } else if (text == "AdaptAllSafe") {
|
c@5
|
631 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE;
|
c@5
|
632 } else if (text == "AdaptAll") {
|
c@5
|
633 flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL;
|
c@5
|
634 } else {
|
c@5
|
635 throw Failure("invalid adapter flag string: " + text);
|
c@5
|
636 }
|
c@5
|
637 }
|
c@5
|
638
|
c@5
|
639 return Vamp::HostExt::PluginLoader::AdapterFlags(flags);
|
c@5
|
640 }
|
c@5
|
641
|
c@5
|
642 static json11::Json
|
c@5
|
643 fromLoadRequest(Vamp::HostExt::LoadRequest req) {
|
c@5
|
644
|
c@5
|
645 json11::Json::object jo;
|
c@5
|
646 jo["pluginKey"] = req.pluginKey;
|
c@5
|
647 jo["inputSampleRate"] = req.inputSampleRate;
|
c@5
|
648 jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags);
|
c@5
|
649 return json11::Json(jo);
|
c@5
|
650 }
|
c@5
|
651
|
c@5
|
652 static Vamp::HostExt::LoadRequest
|
c@5
|
653 toLoadRequest(json11::Json j) {
|
c@5
|
654
|
c@5
|
655 std::string err;
|
c@5
|
656
|
c@5
|
657 if (!j.has_shape({
|
c@5
|
658 { "pluginKey", json11::Json::STRING },
|
c@5
|
659 { "inputSampleRate", json11::Json::NUMBER },
|
c@5
|
660 { "adapterFlags", json11::Json::ARRAY } }, err)) {
|
c@12
|
661 throw Failure("malformed load request: " + err);
|
c@5
|
662 }
|
c@5
|
663
|
c@5
|
664 Vamp::HostExt::LoadRequest req;
|
c@5
|
665 req.pluginKey = j["pluginKey"].string_value();
|
c@5
|
666 req.inputSampleRate = j["inputSampleRate"].number_value();
|
c@5
|
667 req.adapterFlags = toAdapterFlags(j["adapterFlags"]);
|
c@5
|
668 return req;
|
c@5
|
669 }
|
c@10
|
670
|
c@10
|
671 static json11::Json
|
c@10
|
672 fromLoadResponse(Vamp::HostExt::LoadResponse resp,
|
c@10
|
673 PluginHandleMapper &mapper) {
|
c@10
|
674
|
c@10
|
675 json11::Json::object jo;
|
c@10
|
676 jo["pluginHandle"] = double(mapper.pluginToHandle(resp.plugin));
|
c@10
|
677 jo["staticData"] = fromPluginStaticData(resp.staticData);
|
c@10
|
678 jo["defaultConfiguration"] =
|
c@10
|
679 fromPluginConfiguration(resp.defaultConfiguration);
|
c@10
|
680 return json11::Json(jo);
|
c@10
|
681 }
|
c@10
|
682
|
c@10
|
683 static Vamp::HostExt::LoadResponse
|
c@10
|
684 toLoadResponse(json11::Json j,
|
c@10
|
685 PluginHandleMapper &mapper) {
|
c@10
|
686
|
c@10
|
687 std::string err;
|
c@10
|
688
|
c@10
|
689 if (!j.has_shape({
|
c@10
|
690 { "pluginHandle", json11::Json::NUMBER },
|
c@10
|
691 { "staticData", json11::Json::OBJECT },
|
c@10
|
692 { "defaultConfiguration", json11::Json::OBJECT } }, err)) {
|
c@12
|
693 throw Failure("malformed load response: " + err);
|
c@10
|
694 }
|
c@10
|
695
|
c@10
|
696 Vamp::HostExt::LoadResponse resp;
|
c@10
|
697 resp.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@10
|
698 resp.staticData = toPluginStaticData(j["staticData"]);
|
c@10
|
699 resp.defaultConfiguration = toPluginConfiguration(j["defaultConfiguration"]);
|
c@10
|
700 return resp;
|
c@10
|
701 }
|
c@12
|
702
|
c@12
|
703 static json11::Json
|
c@13
|
704 fromConfigurationRequest(const Vamp::HostExt::ConfigurationRequest &cr,
|
c@13
|
705 PluginHandleMapper &mapper) {
|
c@13
|
706
|
c@13
|
707 json11::Json::object jo;
|
c@13
|
708
|
c@13
|
709 jo["pluginHandle"] = mapper.pluginToHandle(cr.plugin);
|
c@13
|
710 jo["configuration"] = fromPluginConfiguration(cr.configuration);
|
c@13
|
711
|
c@13
|
712 return json11::Json(jo);
|
c@13
|
713 }
|
c@13
|
714
|
c@13
|
715 static Vamp::HostExt::ConfigurationRequest
|
c@13
|
716 toConfigurationRequest(json11::Json j,
|
c@13
|
717 PluginHandleMapper &mapper) {
|
c@13
|
718
|
c@13
|
719 std::string err;
|
c@13
|
720
|
c@13
|
721 if (!j.has_shape({
|
c@13
|
722 { "pluginHandle", json11::Json::NUMBER },
|
c@13
|
723 { "configuration", json11::Json::OBJECT } }, err)) {
|
c@13
|
724 throw Failure("malformed configuration request: " + err);
|
c@13
|
725 }
|
c@13
|
726
|
c@13
|
727 Vamp::HostExt::ConfigurationRequest cr;
|
c@13
|
728 cr.plugin = mapper.handleToPlugin(j["pluginHandle"].int_value());
|
c@13
|
729 cr.configuration = toPluginConfiguration(j["configuration"]);
|
c@13
|
730 return cr;
|
c@13
|
731 }
|
c@13
|
732
|
c@13
|
733 static json11::Json
|
c@12
|
734 fromConfigurationResponse(const Vamp::HostExt::ConfigurationResponse &cr) {
|
c@12
|
735
|
c@13
|
736 json11::Json::object jo;
|
c@12
|
737
|
c@12
|
738 json11::Json::array outs;
|
c@12
|
739 for (auto &d: cr.outputs) {
|
c@12
|
740 outs.push_back(fromOutputDescriptor(d));
|
c@12
|
741 }
|
c@13
|
742 jo["outputList"] = outs;
|
c@12
|
743
|
c@13
|
744 return json11::Json(jo);
|
c@12
|
745 }
|
c@12
|
746
|
c@13
|
747 static Vamp::HostExt::ConfigurationResponse
|
c@13
|
748 toConfigurationResponse(json11::Json j) {
|
c@13
|
749
|
c@12
|
750 Vamp::HostExt::ConfigurationResponse cr;
|
c@12
|
751
|
c@12
|
752 if (!j["outputList"].is_array()) {
|
c@12
|
753 throw Failure("array expected for output list");
|
c@12
|
754 }
|
c@12
|
755
|
c@12
|
756 for (const auto &o: j["outputList"].array_items()) {
|
c@12
|
757 cr.outputs.push_back(toOutputDescriptor(o));
|
c@12
|
758 }
|
c@12
|
759
|
c@12
|
760 return cr;
|
c@12
|
761 }
|
c@16
|
762
|
c@16
|
763 static json11::Json
|
c@16
|
764 fromProcessRequest(const Vamp::HostExt::ProcessRequest &r,
|
c@16
|
765 PluginHandleMapper &mapper) {
|
c@16
|
766
|
c@16
|
767 json11::Json::object jo;
|
c@16
|
768 jo["pluginHandle"] = mapper.pluginToHandle(r.plugin);
|
c@16
|
769
|
c@16
|
770 json11::Json::object io;
|
c@16
|
771 io["timestamp"] = fromRealTime(r.timestamp);
|
c@16
|
772
|
c@16
|
773 json11::Json::array chans;
|
c@16
|
774 for (size_t i = 0; i < r.inputBuffers.size(); ++i) {
|
c@16
|
775 json11::Json::object c;
|
c@16
|
776 c["b64values"] = fromFloatBuffer(r.inputBuffers[i].data(),
|
c@16
|
777 r.inputBuffers[i].size());
|
c@16
|
778 chans.push_back(c);
|
c@16
|
779 }
|
c@16
|
780 io["inputBuffers"] = chans;
|
c@16
|
781
|
c@16
|
782 jo["processInput"] = io;
|
c@16
|
783 return json11::Json(jo);
|
c@16
|
784 }
|
c@5
|
785 };
|
c@5
|
786
|
c@10
|
787 }
|
c@5
|
788
|
c@5
|
789 #endif
|