Mercurial > hg > piper-cpp
diff utilities/vampipe-convert.cpp @ 24:533ca5ca3404
More on reading and writing json messages
author | Chris Cannam <c.cannam@qmul.ac.uk> |
---|---|
date | Mon, 23 May 2016 16:09:25 +0100 |
parents | d678cd00e593 |
children | 5b9690d18241 |
line wrap: on
line diff
--- a/utilities/vampipe-convert.cpp Fri May 20 18:05:02 2016 +0100 +++ b/utilities/vampipe-convert.cpp Mon May 23 16:09:25 2016 +0100 @@ -10,31 +10,11 @@ using namespace json11; using namespace vampipe; -// Accepting JSON objects with two fields, "type" and "payload". The +// Accepting JSON objects with two fields, "type" and "content". The // "type" string corresponds to the JSON schema filename -// (e.g. "outputdescriptor") and the "payload" is the JSON object +// (e.g. "outputdescriptor") and the "content" is the JSON object // encoded with that schema. -Json -json_input(string input) -{ - string err; - Json j = Json::parse(input, err); - if (err != "") { - throw VampJson::Failure("invalid json: " + err); - } - if (!j.is_object()) { - throw VampJson::Failure("object expected at top level"); - } - if (!j["type"].is_string()) { - throw VampJson::Failure("string expected for type field"); - } - if (!j["payload"].is_object()) { - throw VampJson::Failure("object expected for payload field"); - } - return j; -} - class PreservingPluginHandleMapper : public PluginHandleMapper { public: @@ -56,112 +36,63 @@ Vamp::Plugin *m_plugin; }; -void -handle_input(::capnp::MallocMessageBuilder &message, string input) -{ - string err; - - Json j = json_input(input); - string type = j["type"].string_value(); - Json payload = j["payload"]; - - if (type == "configurationrequest") { - auto req = message.initRoot<ConfigurationRequest>(); - PreservingPluginHandleMapper mapper; - VampnProto::buildConfigurationRequest - (req, VampJson::toConfigurationRequest(payload, mapper), mapper); - - } else if (type == "configurationresponse") { - auto resp = message.initRoot<ConfigurationResponse>(); - VampnProto::buildConfigurationResponse - (resp, VampJson::toConfigurationResponse(payload)); - - } else if (type == "feature") { - auto f = message.initRoot<Feature>(); - VampnProto::buildFeature - (f, VampJson::toFeature(payload)); - - } else if (type == "featureset") { - auto fs = message.initRoot<FeatureSet>(); - VampnProto::buildFeatureSet - (fs, VampJson::toFeatureSet(payload)); - - } else if (type == "loadrequest") { - auto req = message.initRoot<LoadRequest>(); - VampnProto::buildLoadRequest - (req, VampJson::toLoadRequest(payload)); - - } else if (type == "loadresponse") { - auto resp = message.initRoot<LoadResponse>(); - PreservingPluginHandleMapper mapper; - VampnProto::buildLoadResponse - (resp, VampJson::toLoadResponse(payload, mapper), mapper); - - } else if (type == "outputdescriptor") { - auto od = message.initRoot<OutputDescriptor>(); - VampnProto::buildOutputDescriptor - (od, VampJson::toOutputDescriptor(payload)); - - } else if (type == "parameterdescriptor") { - auto pd = message.initRoot<ParameterDescriptor>(); - VampnProto::buildParameterDescriptor - (pd, VampJson::toParameterDescriptor(payload)); - - } else if (type == "pluginconfiguration") { - auto pc = message.initRoot<PluginConfiguration>(); - auto config = VampJson::toPluginConfiguration(payload); - VampnProto::buildPluginConfiguration(pc, config); - - } else if (type == "pluginstaticdata") { - auto pc = message.initRoot<PluginStaticData>(); - auto sd = VampJson::toPluginStaticData(payload); - VampnProto::buildPluginStaticData(pc, sd); - - } else if (type == "processrequest") { - auto p = message.initRoot<ProcessRequest>(); - PreservingPluginHandleMapper mapper; - VampnProto::buildProcessRequest - (p, VampJson::toProcessRequest(payload, mapper), mapper); - - } else if (type == "realtime") { - auto b = message.initRoot<RealTime>(); - VampnProto::buildRealTime - (b, VampJson::toRealTime(payload)); - - } else { - throw VampJson::Failure("unknown or unsupported JSON schema type " + - type); - } -} - void usage() { string myname = "vampipe-convert"; cerr << "\n" << myname << - ": Convert Vamp request and response messages between formats\n\n" - " Usage: " << myname << " -i <informat> -o <outformat>\n\n" - "Where <informat> and <outformat> may be \"json\" or \"capnp\".\n" - "Messages are read from stdin and written to stdout.\n" << endl; + ": Validate and convert Vamp request and response messages\n\n" + " Usage: " << myname << " [-i <informat>] [-o <outformat>] request\n" + " " << myname << " [-i <informat>] [-o <outformat>] response\n\n" + " where\n" + " <informat>: the format to read from stdin\n" + " (\"json\" or \"capnp\", default is \"json\")\n" + " <outformat>: the format to convert to and write to stdout\n" + " (\"json\" or \"capnp\", default is \"json\")\n" + " request|response: whether to expect Vamp request or response messages\n\n" + "If <informat> and <outformat> differ, convert from <informat> to <outformat>.\n" + "If <informat> and <outformat> are the same, just check validity of incoming\n" + "messages and pass them to output.\n\n"; + exit(2); } class RequestOrResponse { public: + enum Direction { + Request, Response + }; enum Type { List, Load, Configure, Process, Finish, Eof }; + + static Type typeForName(string name) { + if (name == "list") return List; + else if (name == "load") return Load; + else if (name == "configure") return Configure; + else if (name == "process") return Process; + else if (name == "finish") return Finish; + else if (name == "eof") return Eof; + else { + throw runtime_error("unknown or unexpected request/response type \"" + + name + "\""); + } + } + RequestOrResponse() : // nothing by default + direction(Request), type(Eof), success(false), finishPlugin(0) { } + Direction direction; Type type; bool success; string errorText; PreservingPluginHandleMapper mapper; - + + vector<Vamp::HostExt::PluginStaticData> listResponse; Vamp::HostExt::LoadRequest loadRequest; Vamp::HostExt::LoadResponse loadResponse; Vamp::HostExt::ConfigurationRequest configurationRequest; @@ -170,96 +101,219 @@ Vamp::HostExt::ProcessResponse processResponse; Vamp::Plugin *finishPlugin; Vamp::HostExt::ProcessResponse finishResponse; - }; +Json +convertRequestJson(string input) +{ + string err; + Json j = Json::parse(input, err); + if (err != "") { + throw VampJson::Failure("invalid json: " + err); + } + if (!j.is_object()) { + throw VampJson::Failure("object expected at top level"); + } + if (!j["type"].is_string()) { + throw VampJson::Failure("string expected for type field"); + } + if (!j["content"].is_object()) { + throw VampJson::Failure("object expected for content field"); + } + return j; +} + +Json +convertResponseJson(string input) +{ + string err; + Json j = Json::parse(input, err); + if (err != "") { + throw VampJson::Failure("invalid json: " + err); + } + if (!j.is_object()) { + throw VampJson::Failure("object expected at top level"); + } + if (!j["success"].is_bool()) { + throw VampJson::Failure("bool expected for success field"); + } + if (!j["content"].is_object()) { + throw VampJson::Failure("object expected for content field"); + } + return j; +} + RequestOrResponse -readInputJson() +readRequestJson() { RequestOrResponse rr; + rr.direction = RequestOrResponse::Request; + + string input; + if (!getline(cin, input)) { + rr.type = RequestOrResponse::Eof; + return rr; + } + + Json j = convertRequestJson(input); + string type = j["type"].string_value(); + rr.type = RequestOrResponse::typeForName(type); + + if (rr.type == RequestOrResponse::Load) { + rr.loadRequest = VampJson::toVampRequest_Load(j); + + } else if (rr.type == RequestOrResponse::Configure) { + rr.configurationRequest = VampJson::toVampRequest_Configure(j, rr.mapper); + + } else if (rr.type == RequestOrResponse::Process) { + rr.processRequest = VampJson::toVampRequest_Process(j, rr.mapper); + + } else if (rr.type == RequestOrResponse::Finish) { + rr.finishPlugin = VampJson::toVampRequest_Finish(j, rr.mapper); + } + + return rr; +} + +void +writeRequestJson(RequestOrResponse &rr) +{ + Json j; + + if (rr.type == RequestOrResponse::List) { + j = VampJson::fromVampRequest_List(); + + } else if (rr.type == RequestOrResponse::Load) { + j = VampJson::fromVampRequest_Load(rr.loadRequest); + + } else if (rr.type == RequestOrResponse::Configure) { + j = VampJson::fromVampRequest_Configure(rr.configurationRequest, rr.mapper); + + } else if (rr.type == RequestOrResponse::Process) { + j = VampJson::fromVampRequest_Process(rr.processRequest, rr.mapper); + + } else if (rr.type == RequestOrResponse::Finish) { + j = VampJson::fromVampRequest_Finish(rr.finishPlugin, rr.mapper); + } + + cout << j.dump() << endl; +} + +RequestOrResponse +readResponseJson() +{ + RequestOrResponse rr; + rr.direction = RequestOrResponse::Response; + string input; if (!getline(cin, input)) { rr.type = RequestOrResponse::Eof; return rr; } - Json j = json_input(input); + Json j = convertResponseJson(input); string type = j["type"].string_value(); + rr.type = RequestOrResponse::typeForName(type); - if (type == "list") { - rr.type = RequestOrResponse::List; + if (rr.type == RequestOrResponse::List) { + rr.listResponse = VampJson::toVampResponse_List(j); + + } else if (rr.type == RequestOrResponse::Load) { + rr.loadResponse = VampJson::toVampResponse_Load(j, rr.mapper); - } else if (type == "load") { - //!!! ah, we need a way to know whether we're dealing with a request or response here - rr.type = RequestOrResponse::Load; - rr.loadRequest = VampJson::toLoadRequest(j["content"]); + } else if (rr.type == RequestOrResponse::Configure) { + rr.configurationResponse = VampJson::toVampResponse_Configure(j); - } else if (type == "configure") { - rr.type = RequestOrResponse::Configure; - rr.configurationRequest = - VampJson::toConfigurationRequest(j["content"], rr.mapper); + } else if (rr.type == RequestOrResponse::Process) { + rr.processResponse = VampJson::toVampResponse_Process(j); - } else if (type == "process") { - rr.type = RequestOrResponse::Process; - rr.processRequest = - VampJson::toProcessRequest(j["content"], rr.mapper); - - } else if (type == "finish") { - rr.type = RequestOrResponse::Finish; - //!!! VampJsonify - rr.finishPlugin = rr.mapper.handleToPlugin(j["content"]["pluginHandle"].int_value()); - - } else { - throw runtime_error("unknown or unexpected request/response type \"" + - type + "\""); + } else if (rr.type == RequestOrResponse::Finish) { + rr.finishResponse = VampJson::toVampResponse_Finish(j); } return rr; } +void +writeResponseJson(RequestOrResponse &rr) +{ + Json j; + + if (rr.type == RequestOrResponse::List) { + j = VampJson::fromVampResponse_List("", rr.listResponse); + + } else if (rr.type == RequestOrResponse::Load) { + j = VampJson::fromVampResponse_Load(rr.loadResponse, rr.mapper); + } + + cout << j.dump() << endl; +} + RequestOrResponse -readInput(string format) +readInput(string format, RequestOrResponse::Direction direction) { if (format == "json") { - return readInputJson(); + if (direction == RequestOrResponse::Request) { + return readRequestJson(); + } else { + return readResponseJson(); + } } else { throw runtime_error("unknown or unimplemented format \"" + format + "\""); } } void -writeOutput(string format, const RequestOrResponse &rr) +writeOutput(string format, RequestOrResponse &rr) { - throw runtime_error("writeOutput not implemented yet"); - + if (format == "json") { + if (rr.direction == RequestOrResponse::Request) { + writeRequestJson(rr); + } else { + writeResponseJson(rr); + } + } else { + throw runtime_error("unknown or unimplemented format \"" + format + "\""); + } } int main(int argc, char **argv) { - if (argc != 5) { + if (argc < 2) { usage(); } - string informat, outformat; + string informat = "json", outformat = "json"; + RequestOrResponse::Direction direction; + bool haveDirection = false; - for (int i = 1; i + 1 < argc; ++i) { + for (int i = 1; i < argc; ++i) { string arg = argv[i]; + bool final = (i + 1 == argc); if (arg == "-i") { - if (informat != "") usage(); + if (informat != "" || final) usage(); else informat = argv[++i]; } else if (arg == "-o") { - if (outformat != "") usage(); + if (outformat != "" || final) usage(); else outformat = argv[++i]; + } else if (arg == "request") { + direction = RequestOrResponse::Request; + haveDirection = true; + + } else if (arg == "response") { + direction = RequestOrResponse::Response; + haveDirection = true; + } else { usage(); } } - if (informat == "" || outformat == "") { + if (informat == "" || outformat == "" || !haveDirection) { usage(); } @@ -267,7 +321,7 @@ try { - RequestOrResponse rr = readInput(informat); + RequestOrResponse rr = readInput(informat, direction); if (rr.type == RequestOrResponse::Eof) break; writeOutput(outformat, rr);