Mercurial > hg > piper-cpp
diff json/VampJson.h @ 5:6e8607ebad03
Promote the more successful experiments (todo: get them to build again)
author | Chris Cannam <c.cannam@qmul.ac.uk> |
---|---|
date | Fri, 13 May 2016 13:48:59 +0100 |
parents | |
children | d8358afe3f2c |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/json/VampJson.h Fri May 13 13:48:59 2016 +0100 @@ -0,0 +1,664 @@ +/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ + +#ifndef VAMP_JSON_H +#define VAMP_JSON_H + +#include <vector> +#include <string> +#include <sstream> +#include <stdexcept> + +#include <json11/json11.hpp> +#include <base-n/include/basen.hpp> + +#include <vamp-hostsdk/Plugin.h> +#include <vamp-hostsdk/PluginLoader.h> + +class VampJson +{ +public: + class Failure : virtual public std::runtime_error { + public: + Failure(std::string s) : runtime_error(s) { } + }; + + template <typename T> + static json11::Json + fromBasicDescriptor(const T &t) { + return json11::Json::object { + { "identifier", t.identifier }, + { "name", t.name }, + { "description", t.description } + }; + } + + template <typename T> + static void + toBasicDescriptor(json11::Json j, T &t) { + if (!j.is_object()) { + throw Failure("object expected for basic descriptor content"); + } + if (!j["identifier"].is_string()) { + throw Failure("string expected for identifier"); + } + t.identifier = j["identifier"].string_value(); + t.name = j["name"].string_value(); + t.description = j["description"].string_value(); + } + + template <typename T> + static json11::Json + fromValueExtents(const T &t) { + return json11::Json::object { + { "min", t.minValue }, + { "max", t.maxValue } + }; + } + + template <typename T> + static bool + toValueExtents(json11::Json j, T &t) { + if (j["extents"].is_null()) { + return false; + } else if (j["extents"].is_object()) { + if (j["extents"]["min"].is_number() && + j["extents"]["max"].is_number()) { + t.minValue = j["extents"]["min"].number_value(); + t.maxValue = j["extents"]["max"].number_value(); + return true; + } else { + throw Failure("numbers expected for min and max"); + } + } else { + throw Failure("object expected for extents (if present)"); + } + } + + static json11::Json + fromRealTime(const Vamp::RealTime &r) { + return json11::Json::object { + { "s", r.sec }, + { "n", r.nsec } + }; + } + + static Vamp::RealTime + toRealTime(json11::Json j) { + json11::Json sec = j["s"]; + json11::Json nsec = j["n"]; + if (!sec.is_number() || !nsec.is_number()) { + throw Failure("invalid Vamp::RealTime object " + j.dump()); + } + return Vamp::RealTime(sec.int_value(), nsec.int_value()); + } + + static std::string + fromSampleType(Vamp::Plugin::OutputDescriptor::SampleType type) { + switch (type) { + case Vamp::Plugin::OutputDescriptor::OneSamplePerStep: + return "OneSamplePerStep"; + case Vamp::Plugin::OutputDescriptor::FixedSampleRate: + return "FixedSampleRate"; + case Vamp::Plugin::OutputDescriptor::VariableSampleRate: + return "VariableSampleRate"; + } + return ""; + } + + static Vamp::Plugin::OutputDescriptor::SampleType + toSampleType(std::string text) { + if (text == "OneSamplePerStep") { + return Vamp::Plugin::OutputDescriptor::OneSamplePerStep; + } else if (text == "FixedSampleRate") { + return Vamp::Plugin::OutputDescriptor::FixedSampleRate; + } else if (text == "VariableSampleRate") { + return Vamp::Plugin::OutputDescriptor::VariableSampleRate; + } else { + throw Failure("invalid sample type string: " + text); + } + } + + static json11::Json + fromOutputDescriptor(const Vamp::Plugin::OutputDescriptor &desc) { + json11::Json::object jo { + { "basic", fromBasicDescriptor(desc) }, + { "unit", desc.unit }, + { "sampleType", fromSampleType(desc.sampleType) }, + { "sampleRate", desc.sampleRate }, + { "hasDuration", desc.hasDuration } + }; + if (desc.hasFixedBinCount) { + jo["binCount"] = int(desc.binCount); + jo["binNames"] = json11::Json::array + (desc.binNames.begin(), desc.binNames.end()); + } + if (desc.hasKnownExtents) { + jo["extents"] = fromValueExtents(desc); + } + if (desc.isQuantized) { + jo["quantizeStep"] = desc.quantizeStep; + } + return json11::Json(jo); + } + + static Vamp::Plugin::OutputDescriptor + toOutputDescriptor(json11::Json j) { + + Vamp::Plugin::OutputDescriptor od; + if (!j.is_object()) { + throw Failure("object expected for output descriptor"); + } + + toBasicDescriptor(j["basic"], od); + + od.unit = j["unit"].string_value(); + + od.sampleType = toSampleType(j["sampleType"].string_value()); + + if (!j["sampleRate"].is_number()) { + throw Failure("number expected for sample rate"); + } + od.sampleRate = j["sampleRate"].number_value(); + od.hasDuration = j["hasDuration"].bool_value(); + + if (j["binCount"].is_number() && j["binCount"].int_value() > 0) { + od.hasFixedBinCount = true; + od.binCount = j["binCount"].int_value(); + for (auto &n: j["binNames"].array_items()) { + if (!n.is_string()) { + throw Failure("string expected for bin name"); + } + od.binNames.push_back(n.string_value()); + } + } else { + od.hasFixedBinCount = false; + } + + bool extentsPresent = toValueExtents(j, od); + od.hasKnownExtents = extentsPresent; + + if (j["quantizeStep"].is_number()) { + od.isQuantized = true; + od.quantizeStep = j["quantizeStep"].number_value(); + } else { + od.isQuantized = false; + } + + return od; + } + + static json11::Json + fromParameterDescriptor(const Vamp::PluginBase::ParameterDescriptor &desc) { + + json11::Json::object jo { + { "basic", fromBasicDescriptor(desc) }, + { "unit", desc.unit }, + { "extents", fromValueExtents(desc) }, + { "defaultValue", desc.defaultValue }, + { "valueNames", json11::Json::array + (desc.valueNames.begin(), desc.valueNames.end()) } + }; + if (desc.isQuantized) { + jo["quantizeStep"] = desc.quantizeStep; + } + return json11::Json(jo); + } + + static Vamp::PluginBase::ParameterDescriptor + toParameterDescriptor(json11::Json j) { + + Vamp::PluginBase::ParameterDescriptor pd; + if (!j.is_object()) { + throw Failure("object expected for parameter descriptor"); + } + + toBasicDescriptor(j["basic"], pd); + + pd.unit = j["unit"].string_value(); + + bool extentsPresent = toValueExtents(j, pd); + if (!extentsPresent) { + throw Failure("extents must be present in parameter descriptor"); + } + + if (!j["defaultValue"].is_number()) { + throw Failure("number expected for default value"); + } + + pd.defaultValue = j["defaultValue"].number_value(); + + pd.valueNames.clear(); + for (auto &n: j["valueNames"].array_items()) { + if (!n.is_string()) { + throw Failure("string expected for value name"); + } + pd.valueNames.push_back(n.string_value()); + } + + if (j["quantizeStep"].is_number()) { + pd.isQuantized = true; + pd.quantizeStep = j["quantizeStep"].number_value(); + } else { + pd.isQuantized = false; + } + + return pd; + } + + static std::string + fromFloatBuffer(const float *buffer, size_t nfloats) { + // must use char pointers, otherwise the converter will only + // encode every 4th byte (as it will count up in float* steps) + const char *start = reinterpret_cast<const char *>(buffer); + const char *end = reinterpret_cast<const char *>(buffer + nfloats); + std::string encoded; + bn::encode_b64(start, end, back_inserter(encoded)); + return encoded; + } + + static std::vector<float> + toFloatBuffer(std::string encoded) { + std::string decoded; + bn::decode_b64(encoded.begin(), encoded.end(), back_inserter(decoded)); + const float *buffer = reinterpret_cast<const float *>(decoded.c_str()); + size_t n = decoded.size() / sizeof(float); + return std::vector<float>(buffer, buffer + n); + } + + static json11::Json + fromFeature(const Vamp::Plugin::Feature &f) { + + json11::Json::object jo; + if (f.values.size() > 0) { + jo["b64values"] = fromFloatBuffer(f.values.data(), f.values.size()); + } + if (f.label != "") { + jo["label"] = f.label; + } + if (f.hasTimestamp) { + jo["timestamp"] = fromRealTime(f.timestamp); + } + if (f.hasDuration) { + jo["duration"] = fromRealTime(f.duration); + } + return json11::Json(jo); + } + + static Vamp::Plugin::Feature + toFeature(json11::Json j) { + + Vamp::Plugin::Feature f; + if (!j.is_object()) { + throw Failure("object expected for feature"); + } + if (j["timestamp"].is_object()) { + f.timestamp = toRealTime(j["timestamp"]); + f.hasTimestamp = true; + } + if (j["duration"].is_object()) { + f.duration = toRealTime(j["duration"]); + f.hasDuration = true; + } + if (j["b64values"].is_string()) { + f.values = toFloatBuffer(j["b64values"].string_value()); + } else if (j["values"].is_array()) { + for (auto v : j["values"].array_items()) { + f.values.push_back(v.number_value()); + } + } + f.label = j["label"].string_value(); + return f; + } + + static json11::Json + fromFeatureSet(const Vamp::Plugin::FeatureSet &fs) { + + json11::Json::object jo; + for (const auto &fsi : fs) { + std::vector<json11::Json> fj; + for (const Vamp::Plugin::Feature &f: fsi.second) { + fj.push_back(fromFeature(f)); + } + std::stringstream sstr; + sstr << fsi.first; + std::string n = sstr.str(); + jo[n] = fj; + } + return json11::Json(jo); + } + + static Vamp::Plugin::FeatureList + toFeatureList(json11::Json j) { + + Vamp::Plugin::FeatureList fl; + if (!j.is_array()) { + throw Failure("array expected for feature list"); + } + for (const json11::Json &fj : j.array_items()) { + fl.push_back(toFeature(fj)); + } + return fl; + } + + static Vamp::Plugin::FeatureSet + toFeatureSet(json11::Json j) { + + Vamp::Plugin::FeatureSet fs; + if (!j.is_object()) { + throw Failure("object expected for feature set"); + } + for (auto &entry : j.object_items()) { + std::string nstr = entry.first; + size_t count = 0; + int n = stoi(nstr, &count); + if (n < 0 || fs.find(n) != fs.end() || count < nstr.size()) { + throw Failure("invalid or duplicate numerical index for output"); + } + fs[n] = toFeatureList(entry.second); + } + return fs; + } + + static std::string + fromInputDomain(Vamp::Plugin::InputDomain domain) { + + switch (domain) { + case Vamp::Plugin::TimeDomain: + return "TimeDomain"; + case Vamp::Plugin::FrequencyDomain: + return "FrequencyDomain"; + } + return ""; + } + + static Vamp::Plugin::InputDomain + toInputDomain(std::string text) { + + if (text == "TimeDomain") { + return Vamp::Plugin::TimeDomain; + } else if (text == "FrequencyDomain") { + return Vamp::Plugin::FrequencyDomain; + } else { + throw Failure("invalid input domain string: " + text); + } + } + + static json11::Json + fromPluginStaticData(const Vamp::HostExt::PluginStaticData &d) { + + json11::Json::object jo; + jo["pluginKey"] = d.pluginKey; + jo["basic"] = fromBasicDescriptor(d.basic); + jo["maker"] = d.maker; + jo["copyright"] = d.copyright; + jo["pluginVersion"] = d.pluginVersion; + + json11::Json::array cat; + for (const std::string &c: d.category) cat.push_back(c); + jo["category"] = cat; + + jo["minChannelCount"] = d.minChannelCount; + jo["maxChannelCount"] = d.maxChannelCount; + + json11::Json::array params; + Vamp::PluginBase::ParameterList vparams = d.parameters; + for (auto &p: vparams) params.push_back(fromParameterDescriptor(p)); + jo["parameters"] = params; + + json11::Json::array progs; + Vamp::PluginBase::ProgramList vprogs = d.programs; + for (auto &p: vprogs) progs.push_back(p); + jo["programs"] = progs; + + jo["inputDomain"] = fromInputDomain(d.inputDomain); + + json11::Json::array outinfo; + auto vouts = d.basicOutputInfo; + for (auto &o: vouts) outinfo.push_back(fromBasicDescriptor(o)); + jo["basicOutputInfo"] = outinfo; + + return json11::Json(jo); + } + + static Vamp::HostExt::PluginStaticData + toPluginStaticData(json11::Json j) { + + std::string err; + if (!j.has_shape({ + { "pluginKey", json11::Json::STRING }, + { "pluginVersion", json11::Json::NUMBER }, + { "minChannelCount", json11::Json::NUMBER }, + { "maxChannelCount", json11::Json::NUMBER }, + { "inputDomain", json11::Json::STRING }}, err)) { + throw Failure("malformed plugin static data: " + err); + } + + if (!j["basicOutputInfo"].is_array()) { + throw Failure("array expected for basic output info"); + } + + if (!j["maker"].is_null() && + !j["maker"].is_string()) { + throw Failure("string expected for maker"); + } + + if (!j["copyright"].is_null() && + !j["copyright"].is_string()) { + throw Failure("string expected for copyright"); + } + + if (!j["category"].is_null() && + !j["category"].is_array()) { + throw Failure("array expected for category"); + } + + if (!j["parameters"].is_null() && + !j["parameters"].is_array()) { + throw Failure("array expected for parameters"); + } + + if (!j["programs"].is_null() && + !j["programs"].is_array()) { + throw Failure("array expected for programs"); + } + + if (!j["inputDomain"].is_null() && + !j["inputDomain"].is_string()) { + throw Failure("string expected for inputDomain"); + } + + if (!j["basicOutputInfo"].is_null() && + !j["basicOutputInfo"].is_array()) { + throw Failure("array expected for basicOutputInfo"); + } + + Vamp::HostExt::PluginStaticData psd; + + psd.pluginKey = j["pluginKey"].string_value(); + + toBasicDescriptor(j["basic"], psd.basic); + + psd.maker = j["maker"].string_value(); + psd.copyright = j["copyright"].string_value(); + psd.pluginVersion = j["pluginVersion"].int_value(); + + for (const auto &c : j["category"].array_items()) { + if (!c.is_string()) { + throw Failure("strings expected in category array"); + } + psd.category.push_back(c.string_value()); + } + + psd.minChannelCount = j["minChannelCount"].int_value(); + psd.maxChannelCount = j["maxChannelCount"].int_value(); + + for (const auto &p : j["parameters"].array_items()) { + auto pd = toParameterDescriptor(p); + psd.parameters.push_back(pd); + } + + for (const auto &p : j["programs"].array_items()) { + if (!p.is_string()) { + throw Failure("strings expected in programs array"); + } + psd.programs.push_back(p.string_value()); + } + + psd.inputDomain = toInputDomain(j["inputDomain"].string_value()); + + for (const auto &bo : j["basicOutputInfo"].array_items()) { + Vamp::HostExt::PluginStaticData::Basic b; + toBasicDescriptor(bo, b); + psd.basicOutputInfo.push_back(b); + } + + return psd; + } + + static json11::Json + fromPluginConfiguration(const Vamp::HostExt::PluginConfiguration &c) { + + json11::Json::object jo; + + json11::Json::object paramValues; + for (auto &vp: c.parameterValues) { + paramValues[vp.first] = vp.second; + } + jo["parameterValues"] = paramValues; + + if (c.currentProgram != "") { + jo["currentProgram"] = c.currentProgram; + } + + jo["channelCount"] = c.channelCount; + jo["stepSize"] = c.stepSize; + jo["blockSize"] = c.blockSize; + + return json11::Json(jo); + } + + static Vamp::HostExt::PluginConfiguration + toPluginConfiguration(json11::Json j) { + + std::string err; + if (!j.has_shape({ + { "channelCount", json11::Json::NUMBER }, + { "stepSize", json11::Json::NUMBER }, + { "blockSize", json11::Json::NUMBER } }, err)) { + throw Failure("malformed plugin configuration: " + err); + } + + if (!j["parameterValues"].is_null() && + !j["parameterValues"].is_object()) { + throw Failure("object expected for parameter values"); + } + + for (auto &pv : j["parameterValues"].object_items()) { + if (!pv.second.is_number()) { + throw Failure("number expected for parameter value"); + } + } + + if (!j["currentProgram"].is_null() && + !j["currentProgram"].is_string()) { + throw Failure("string expected for program name"); + } + + Vamp::HostExt::PluginConfiguration config; + + config.channelCount = j["channelCount"].number_value(); + config.stepSize = j["stepSize"].number_value(); + config.blockSize = j["blockSize"].number_value(); + + for (auto &pv : j["parameterValues"].object_items()) { + config.parameterValues[pv.first] = pv.second.number_value(); + } + + if (j["currentProgram"].is_string()) { + config.currentProgram = j["currentProgram"].string_value(); + } + + return config; + } + + static json11::Json + fromAdapterFlags(int flags) { + + json11::Json::array arr; + + if (flags & Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN) { + arr.push_back("AdaptInputDomain"); + } + if (flags & Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT) { + arr.push_back("AdaptChannelCount"); + } + if (flags & Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE) { + arr.push_back("AdaptBufferSize"); + } + + return json11::Json(arr); + } + + static Vamp::HostExt::PluginLoader::AdapterFlags + toAdapterFlags(json11::Json j) { + + if (!j.is_array()) { + throw Failure("array expected for adapter flags"); + } + int flags = 0x0; + + for (auto &jj: j.array_items()) { + if (!jj.is_string()) { + throw Failure("string expected for adapter flag"); + } + std::string text = jj.string_value(); + if (text == "AdaptInputDomain") { + flags |= Vamp::HostExt::PluginLoader::ADAPT_INPUT_DOMAIN; + } else if (text == "AdaptChannelCount") { + flags |= Vamp::HostExt::PluginLoader::ADAPT_CHANNEL_COUNT; + } else if (text == "AdaptBufferSize") { + flags |= Vamp::HostExt::PluginLoader::ADAPT_BUFFER_SIZE; + } else if (text == "AdaptAllSafe") { + flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL_SAFE; + } else if (text == "AdaptAll") { + flags |= Vamp::HostExt::PluginLoader::ADAPT_ALL; + } else { + throw Failure("invalid adapter flag string: " + text); + } + } + + return Vamp::HostExt::PluginLoader::AdapterFlags(flags); + } + + static json11::Json + fromLoadRequest(Vamp::HostExt::LoadRequest req) { + + json11::Json::object jo; + jo["pluginKey"] = req.pluginKey; + jo["inputSampleRate"] = req.inputSampleRate; + jo["adapterFlags"] = fromAdapterFlags(req.adapterFlags); + return json11::Json(jo); + } + + static Vamp::HostExt::LoadRequest + toLoadRequest(json11::Json j) { + + std::string err; + + if (!j.has_shape({ + { "pluginKey", json11::Json::STRING }, + { "inputSampleRate", json11::Json::NUMBER }, + { "adapterFlags", json11::Json::ARRAY } }, err)) { + throw VampJson::Failure("malformed load request: " + err); + } + + Vamp::HostExt::LoadRequest req; + req.pluginKey = j["pluginKey"].string_value(); + req.inputSampleRate = j["inputSampleRate"].number_value(); + req.adapterFlags = toAdapterFlags(j["adapterFlags"]); + return req; + } +}; + + +#endif