annotate utilities/json-to-capnp.cpp @ 20:7ca64f74add2

Add and use PreservingPluginHandleMapper -- removing the constant variant, which was a misstep
author Chris Cannam <c.cannam@qmul.ac.uk>
date Fri, 20 May 2016 11:08:53 +0100
parents f379b0e9a8e1
children dff96610ac05
rev   line source
c@5 1
c@5 2 #include "VampJson.h"
c@6 3 #include "VampnProto.h"
c@5 4
c@5 5 #include <iostream>
c@5 6 #include <sstream>
c@5 7 #include <stdexcept>
c@5 8
c@5 9 using namespace std;
c@5 10 using namespace json11;
c@5 11 using namespace vampipe;
c@5 12
c@5 13 // Accepting JSON objects with two fields, "type" and "payload". The
c@5 14 // "type" string corresponds to the JSON schema filename
c@5 15 // (e.g. "outputdescriptor") and the "payload" is the JSON object
c@5 16 // encoded with that schema.
c@5 17
c@5 18 Json
c@5 19 json_input(string input)
c@5 20 {
c@5 21 string err;
c@5 22 Json j = Json::parse(input, err);
c@5 23 if (err != "") {
c@5 24 throw VampJson::Failure("invalid json: " + err);
c@5 25 }
c@5 26 if (!j.is_object()) {
c@5 27 throw VampJson::Failure("object expected at top level");
c@5 28 }
c@5 29 if (!j["type"].is_string()) {
c@5 30 throw VampJson::Failure("string expected for type field");
c@5 31 }
c@5 32 if (!j["payload"].is_object()) {
c@5 33 throw VampJson::Failure("object expected for payload field");
c@5 34 }
c@5 35 return j;
c@5 36 }
c@5 37
c@20 38 class PreservingPluginHandleMapper : public PluginHandleMapper
c@20 39 {
c@20 40 public:
c@20 41 PreservingPluginHandleMapper() : m_handle(0), m_plugin(0) { }
c@20 42
c@20 43 virtual int32_t pluginToHandle(Vamp::Plugin *p) {
c@20 44 if (p == m_plugin) return m_handle;
c@20 45 else throw NotFound();
c@20 46 }
c@20 47
c@20 48 virtual Vamp::Plugin *handleToPlugin(int32_t h) {
c@20 49 m_handle = h;
c@20 50 m_plugin = reinterpret_cast<Vamp::Plugin *>(1);
c@20 51 return m_plugin;
c@20 52 }
c@20 53
c@20 54 private:
c@20 55 int32_t m_handle;
c@20 56 Vamp::Plugin *m_plugin;
c@20 57 };
c@20 58
c@5 59 void
c@5 60 handle_input(::capnp::MallocMessageBuilder &message, string input)
c@5 61 {
c@5 62 string err;
c@5 63
c@5 64 Json j = json_input(input);
c@5 65 string type = j["type"].string_value();
c@5 66 Json payload = j["payload"];
c@5 67
c@5 68 if (type == "basic") {
c@5 69 throw VampJson::Failure("can't convert Basic block on its own");
c@5 70
c@5 71 } else if (type == "configurationrequest") {
c@20 72
c@20 73 auto req = message.initRoot<ConfigurationRequest>();
c@20 74 PreservingPluginHandleMapper mapper;
c@20 75 VampnProto::buildConfigurationRequest
c@20 76 (req, VampJson::toConfigurationRequest(payload, mapper), mapper);
c@5 77
c@5 78 } else if (type == "configurationresponse") {
c@5 79 throw VampJson::Failure("not implemented yet"); ///!!!
c@5 80
c@5 81 } else if (type == "feature") {
c@5 82 auto f = message.initRoot<Feature>();
c@6 83 VampnProto::buildFeature
c@5 84 (f, VampJson::toFeature(payload));
c@5 85
c@5 86 } else if (type == "featureset") {
c@5 87 auto fs = message.initRoot<FeatureSet>();
c@6 88 VampnProto::buildFeatureSet
c@5 89 (fs, VampJson::toFeatureSet(payload));
c@5 90
c@5 91 } else if (type == "loadrequest") {
c@5 92 auto req = message.initRoot<LoadRequest>();
c@6 93 VampnProto::buildLoadRequest
c@5 94 (req, VampJson::toLoadRequest(payload));
c@5 95
c@5 96 } else if (type == "loadresponse") {
c@5 97 //!!! response types & configure call for plugin handles, but
c@5 98 //!!! we don't have any context in which a plugin handle can
c@5 99 //!!! be persistent here
c@5 100 throw VampJson::Failure("not implemented yet"); ///!!!
c@5 101
c@5 102 } else if (type == "outputdescriptor") {
c@5 103 auto od = message.initRoot<OutputDescriptor>();
c@6 104 VampnProto::buildOutputDescriptor
c@5 105 (od, VampJson::toOutputDescriptor(payload));
c@5 106
c@5 107 } else if (type == "parameterdescriptor") {
c@5 108 auto pd = message.initRoot<ParameterDescriptor>();
c@6 109 VampnProto::buildParameterDescriptor
c@5 110 (pd, VampJson::toParameterDescriptor(payload));
c@5 111
c@5 112 } else if (type == "pluginconfiguration") {
c@5 113 auto pc = message.initRoot<PluginConfiguration>();
c@5 114 auto config = VampJson::toPluginConfiguration(payload);
c@6 115 VampnProto::buildPluginConfiguration(pc, config);
c@5 116
c@5 117 } else if (type == "pluginstaticdata") {
c@5 118 auto pc = message.initRoot<PluginStaticData>();
c@5 119 auto sd = VampJson::toPluginStaticData(payload);
c@6 120 VampnProto::buildPluginStaticData(pc, sd);
c@5 121
c@5 122 } else if (type == "processblock") {
c@5 123 throw VampJson::Failure("not implemented yet"); ///!!!
c@5 124
c@5 125 } else if (type == "realtime") {
c@5 126 auto b = message.initRoot<RealTime>();
c@6 127 VampnProto::buildRealTime
c@5 128 (b, VampJson::toRealTime(payload));
c@5 129
c@5 130 } else if (type == "valueextents") {
c@5 131 throw VampJson::Failure("no ValueExtents struct in Cap'n Proto mapping");
c@5 132
c@5 133 } else {
c@5 134 throw VampJson::Failure("unknown or unsupported JSON schema type " +
c@5 135 type);
c@5 136 }
c@5 137 }
c@5 138
c@5 139 int main(int, char **)
c@5 140 {
c@5 141 string input;
c@5 142
c@5 143 while (getline(cin, input)) {
c@5 144 try {
c@5 145 ::capnp::MallocMessageBuilder message;
c@5 146 handle_input(message, input);
c@5 147 writePackedMessageToFd(1, message); // stdout
c@5 148 return 0;
c@5 149 } catch (const VampJson::Failure &e) {
c@5 150 cerr << "Failed to convert JSON to Cap'n Proto message: "
c@5 151 << e.what() << endl;
c@5 152 return 1;
c@5 153 }
c@5 154 }
c@5 155 }
c@5 156
c@5 157