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
|