c@116
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
c@118
|
2 /*
|
c@118
|
3 Piper C++
|
c@118
|
4
|
c@118
|
5 An API for audio analysis and feature extraction plugins.
|
c@118
|
6
|
c@118
|
7 Centre for Digital Music, Queen Mary, University of London.
|
c@118
|
8 Copyright 2006-2016 Chris Cannam and QMUL.
|
c@118
|
9
|
c@118
|
10 Permission is hereby granted, free of charge, to any person
|
c@118
|
11 obtaining a copy of this software and associated documentation
|
c@118
|
12 files (the "Software"), to deal in the Software without
|
c@118
|
13 restriction, including without limitation the rights to use, copy,
|
c@118
|
14 modify, merge, publish, distribute, sublicense, and/or sell copies
|
c@118
|
15 of the Software, and to permit persons to whom the Software is
|
c@118
|
16 furnished to do so, subject to the following conditions:
|
c@118
|
17
|
c@118
|
18 The above copyright notice and this permission notice shall be
|
c@118
|
19 included in all copies or substantial portions of the Software.
|
c@118
|
20
|
c@118
|
21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
c@118
|
22 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
c@118
|
23 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
c@118
|
24 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
|
c@118
|
25 ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
c@118
|
26 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
c@118
|
27 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
c@118
|
28
|
c@118
|
29 Except as contained in this notice, the names of the Centre for
|
c@118
|
30 Digital Music; Queen Mary, University of London; and Chris Cannam
|
c@118
|
31 shall not be used in advertising or otherwise to promote the sale,
|
c@118
|
32 use or other dealings in this Software without prior written
|
c@118
|
33 authorization.
|
c@118
|
34 */
|
c@75
|
35
|
c@116
|
36 #include "vamp-json/VampJson.h"
|
c@75
|
37 #include "vamp-capnp/VampnProto.h"
|
c@75
|
38 #include "vamp-support/RequestOrResponse.h"
|
c@75
|
39 #include "vamp-support/CountingPluginHandleMapper.h"
|
c@97
|
40 #include "vamp-support/LoaderRequests.h"
|
c@75
|
41
|
c@75
|
42 #include <iostream>
|
c@75
|
43 #include <sstream>
|
c@75
|
44 #include <stdexcept>
|
c@75
|
45
|
c@91
|
46 #include <capnp/serialize.h>
|
c@91
|
47
|
c@75
|
48 #include <map>
|
c@75
|
49 #include <set>
|
c@75
|
50
|
c@109
|
51 // pid for logging
|
c@109
|
52 #ifdef _WIN32
|
c@109
|
53 #include <process.h>
|
c@109
|
54 static int pid = _getpid();
|
c@109
|
55 #else
|
c@109
|
56 #include <unistd.h>
|
c@109
|
57 static int pid = getpid();
|
c@109
|
58 #endif
|
c@103
|
59
|
c@75
|
60 using namespace std;
|
c@116
|
61 using namespace json11;
|
c@97
|
62 using namespace piper_vamp;
|
c@75
|
63 using namespace Vamp;
|
c@75
|
64
|
c@102
|
65 //!!! This could be faster and lighter:
|
c@102
|
66 // - Use Capnp structures directly rather than converting to vamp-support ones
|
c@102
|
67 // - Use Vamp C API (vamp.h) directly rather than converting to C++
|
c@102
|
68 //!!! Doing the above for process() and finish() alone would be a good start
|
c@102
|
69
|
c@116
|
70 static string myname = "piper-vamp-server";
|
c@116
|
71
|
c@116
|
72 static void version()
|
c@75
|
73 {
|
c@116
|
74 cout << "1.0" << endl;
|
c@116
|
75 exit(0);
|
c@116
|
76 }
|
c@116
|
77
|
c@116
|
78 static void usage(bool successful = false)
|
c@116
|
79 {
|
c@75
|
80 cerr << "\n" << myname <<
|
c@116
|
81 ": Load and run Vamp plugins in response to Piper messages\n\n"
|
c@116
|
82 " Usage: " << myname << " [-d] <format>\n"
|
c@116
|
83 " " << myname << " -v\n"
|
c@116
|
84 " " << myname << " -h\n\n"
|
c@116
|
85 " where\n"
|
c@116
|
86 " <format>: the format to read and write messages in (\"json\" or \"capnp\")\n"
|
c@116
|
87 " -d: also print debug information to stderr\n"
|
c@116
|
88 " -v: print version number to stdout and exit\n"
|
c@116
|
89 " -h: print this text to stderr and exit\n\n"
|
c@116
|
90 "Expects Piper request messages in either Cap'n Proto or JSON format on stdin,\n"
|
c@116
|
91 "and writes response messages in the same format to stdout.\n\n";
|
c@116
|
92 if (successful) exit(0);
|
c@116
|
93 else exit(2);
|
c@75
|
94 }
|
c@75
|
95
|
c@75
|
96 static CountingPluginHandleMapper mapper;
|
c@75
|
97
|
c@116
|
98 static RequestOrResponse::RpcId
|
c@116
|
99 readId(const piper::RpcRequest::Reader &r)
|
c@75
|
100 {
|
c@75
|
101 int number;
|
c@75
|
102 string tag;
|
c@75
|
103 switch (r.getId().which()) {
|
c@97
|
104 case piper::RpcRequest::Id::Which::NUMBER:
|
c@75
|
105 number = r.getId().getNumber();
|
c@75
|
106 return { RequestOrResponse::RpcId::Number, number, "" };
|
c@97
|
107 case piper::RpcRequest::Id::Which::TAG:
|
c@75
|
108 tag = r.getId().getTag();
|
c@75
|
109 return { RequestOrResponse::RpcId::Tag, 0, tag };
|
c@97
|
110 case piper::RpcRequest::Id::Which::NONE:
|
c@75
|
111 return { RequestOrResponse::RpcId::Absent, 0, "" };
|
c@75
|
112 }
|
c@75
|
113 return {};
|
c@75
|
114 }
|
c@75
|
115
|
c@116
|
116 static void
|
c@116
|
117 buildId(piper::RpcResponse::Builder &b, const RequestOrResponse::RpcId &id)
|
c@75
|
118 {
|
c@75
|
119 switch (id.type) {
|
c@75
|
120 case RequestOrResponse::RpcId::Number:
|
c@75
|
121 b.getId().setNumber(id.number);
|
c@75
|
122 break;
|
c@75
|
123 case RequestOrResponse::RpcId::Tag:
|
c@75
|
124 b.getId().setTag(id.tag);
|
c@75
|
125 break;
|
c@75
|
126 case RequestOrResponse::RpcId::Absent:
|
c@75
|
127 b.getId().setNone();
|
c@75
|
128 break;
|
c@75
|
129 }
|
c@75
|
130 }
|
c@75
|
131
|
c@116
|
132 static RequestOrResponse::RpcId
|
c@116
|
133 readJsonId(const Json &j)
|
c@116
|
134 {
|
c@116
|
135 RequestOrResponse::RpcId id;
|
c@116
|
136
|
c@116
|
137 if (j["id"].is_number()) {
|
c@116
|
138 id.type = RequestOrResponse::RpcId::Number;
|
c@116
|
139 id.number = j["id"].number_value();
|
c@116
|
140 } else if (j["id"].is_string()) {
|
c@116
|
141 id.type = RequestOrResponse::RpcId::Tag;
|
c@116
|
142 id.tag = j["id"].string_value();
|
c@116
|
143 } else {
|
c@116
|
144 id.type = RequestOrResponse::RpcId::Absent;
|
c@116
|
145 }
|
c@116
|
146
|
c@116
|
147 return id;
|
c@116
|
148 }
|
c@116
|
149
|
c@116
|
150 static Json
|
c@116
|
151 writeJsonId(const RequestOrResponse::RpcId &id)
|
c@116
|
152 {
|
c@116
|
153 if (id.type == RequestOrResponse::RpcId::Number) {
|
c@116
|
154 return id.number;
|
c@116
|
155 } else if (id.type == RequestOrResponse::RpcId::Tag) {
|
c@116
|
156 return id.tag;
|
c@116
|
157 } else {
|
c@116
|
158 return Json();
|
c@116
|
159 }
|
c@116
|
160 }
|
c@116
|
161
|
c@116
|
162 static Json
|
c@116
|
163 convertRequestJson(string input, string &err)
|
c@116
|
164 {
|
c@116
|
165 Json j = Json::parse(input, err);
|
c@116
|
166 if (err != "") {
|
c@116
|
167 err = "invalid json: " + err;
|
c@116
|
168 return {};
|
c@116
|
169 }
|
c@116
|
170 if (!j.is_object()) {
|
c@116
|
171 err = "object expected at top level";
|
c@116
|
172 } else if (!j["method"].is_string()) {
|
c@116
|
173 err = "string expected for method field";
|
c@116
|
174 } else if (!j["params"].is_null() && !j["params"].is_object()) {
|
c@116
|
175 err = "object expected for params field";
|
c@116
|
176 }
|
c@116
|
177 return j;
|
c@116
|
178 }
|
c@116
|
179
|
c@116
|
180 RequestOrResponse
|
c@116
|
181 readRequestJson(string &err)
|
c@116
|
182 {
|
c@116
|
183 RequestOrResponse rr;
|
c@116
|
184 rr.direction = RequestOrResponse::Request;
|
c@116
|
185
|
c@116
|
186 string input;
|
c@116
|
187 if (!getline(cin, input)) {
|
c@116
|
188 // the EOF case, not actually an error
|
c@116
|
189 rr.type = RRType::NotValid;
|
c@116
|
190 return rr;
|
c@116
|
191 }
|
c@116
|
192
|
c@116
|
193 Json j = convertRequestJson(input, err);
|
c@116
|
194 if (err != "") return {};
|
c@116
|
195
|
c@116
|
196 rr.type = VampJson::getRequestResponseType(j, err);
|
c@116
|
197 if (err != "") return {};
|
c@116
|
198
|
c@116
|
199 rr.id = readJsonId(j);
|
c@116
|
200
|
c@116
|
201 VampJson::BufferSerialisation serialisation =
|
c@116
|
202 VampJson::BufferSerialisation::Array;
|
c@116
|
203
|
c@116
|
204 switch (rr.type) {
|
c@116
|
205
|
c@116
|
206 case RRType::List:
|
c@116
|
207 VampJson::toRpcRequest_List(j, err); // type check only
|
c@116
|
208 break;
|
c@116
|
209 case RRType::Load:
|
c@116
|
210 rr.loadRequest = VampJson::toRpcRequest_Load(j, err);
|
c@116
|
211 break;
|
c@116
|
212 case RRType::Configure:
|
c@116
|
213 rr.configurationRequest = VampJson::toRpcRequest_Configure(j, mapper, err);
|
c@116
|
214 break;
|
c@116
|
215 case RRType::Process:
|
c@116
|
216 rr.processRequest = VampJson::toRpcRequest_Process(j, mapper, serialisation, err);
|
c@116
|
217 break;
|
c@116
|
218 case RRType::Finish:
|
c@116
|
219 rr.finishRequest = VampJson::toRpcRequest_Finish(j, mapper, err);
|
c@116
|
220 break;
|
c@116
|
221 case RRType::NotValid:
|
c@116
|
222 break;
|
c@116
|
223 }
|
c@116
|
224
|
c@116
|
225 return rr;
|
c@116
|
226 }
|
c@116
|
227
|
c@116
|
228 void
|
c@116
|
229 writeResponseJson(RequestOrResponse &rr, bool useBase64)
|
c@116
|
230 {
|
c@116
|
231 Json j;
|
c@116
|
232
|
c@116
|
233 VampJson::BufferSerialisation serialisation =
|
c@116
|
234 (useBase64 ?
|
c@116
|
235 VampJson::BufferSerialisation::Base64 :
|
c@116
|
236 VampJson::BufferSerialisation::Array);
|
c@116
|
237
|
c@116
|
238 Json id = writeJsonId(rr.id);
|
c@116
|
239
|
c@116
|
240 if (!rr.success) {
|
c@116
|
241
|
c@116
|
242 j = VampJson::fromError(rr.errorText, rr.type, id);
|
c@116
|
243
|
c@116
|
244 } else {
|
c@116
|
245
|
c@116
|
246 switch (rr.type) {
|
c@116
|
247
|
c@116
|
248 case RRType::List:
|
c@116
|
249 j = VampJson::fromRpcResponse_List(rr.listResponse, id);
|
c@116
|
250 break;
|
c@116
|
251 case RRType::Load:
|
c@116
|
252 j = VampJson::fromRpcResponse_Load(rr.loadResponse, mapper, id);
|
c@116
|
253 break;
|
c@116
|
254 case RRType::Configure:
|
c@116
|
255 j = VampJson::fromRpcResponse_Configure(rr.configurationResponse,
|
c@116
|
256 mapper, id);
|
c@116
|
257 break;
|
c@116
|
258 case RRType::Process:
|
c@116
|
259 j = VampJson::fromRpcResponse_Process
|
c@116
|
260 (rr.processResponse, mapper, serialisation, id);
|
c@116
|
261 break;
|
c@116
|
262 case RRType::Finish:
|
c@116
|
263 j = VampJson::fromRpcResponse_Finish
|
c@116
|
264 (rr.finishResponse, mapper, serialisation, id);
|
c@116
|
265 break;
|
c@116
|
266 case RRType::NotValid:
|
c@116
|
267 break;
|
c@116
|
268 }
|
c@116
|
269 }
|
c@116
|
270
|
c@116
|
271 cout << j.dump() << endl;
|
c@116
|
272 }
|
c@116
|
273
|
c@116
|
274 void
|
c@116
|
275 writeExceptionJson(const std::exception &e, RRType type)
|
c@116
|
276 {
|
c@116
|
277 Json j = VampJson::fromError(e.what(), type, Json());
|
c@116
|
278 cout << j.dump() << endl;
|
c@116
|
279 }
|
c@116
|
280
|
c@75
|
281 RequestOrResponse
|
c@75
|
282 readRequestCapnp()
|
c@75
|
283 {
|
c@75
|
284 RequestOrResponse rr;
|
c@75
|
285 rr.direction = RequestOrResponse::Request;
|
c@75
|
286
|
c@75
|
287 static kj::FdInputStream stream(0); // stdin
|
c@75
|
288 static kj::BufferedInputStreamWrapper buffered(stream);
|
c@75
|
289
|
c@75
|
290 if (buffered.tryGetReadBuffer() == nullptr) {
|
c@116
|
291 rr.type = RRType::NotValid;
|
c@116
|
292 return rr;
|
c@75
|
293 }
|
c@75
|
294
|
c@97
|
295 capnp::InputStreamMessageReader message(buffered);
|
c@97
|
296 piper::RpcRequest::Reader reader = message.getRoot<piper::RpcRequest>();
|
c@75
|
297
|
c@75
|
298 rr.type = VampnProto::getRequestResponseType(reader);
|
c@75
|
299 rr.id = readId(reader);
|
c@75
|
300
|
c@75
|
301 switch (rr.type) {
|
c@75
|
302
|
c@75
|
303 case RRType::List:
|
c@116
|
304 VampnProto::readRpcRequest_List(reader); // type check only
|
c@116
|
305 break;
|
c@75
|
306 case RRType::Load:
|
c@116
|
307 VampnProto::readRpcRequest_Load(rr.loadRequest, reader);
|
c@116
|
308 break;
|
c@75
|
309 case RRType::Configure:
|
c@116
|
310 VampnProto::readRpcRequest_Configure(rr.configurationRequest,
|
c@116
|
311 reader, mapper);
|
c@116
|
312 break;
|
c@75
|
313 case RRType::Process:
|
c@116
|
314 VampnProto::readRpcRequest_Process(rr.processRequest, reader, mapper);
|
c@116
|
315 break;
|
c@75
|
316 case RRType::Finish:
|
c@116
|
317 VampnProto::readRpcRequest_Finish(rr.finishRequest, reader, mapper);
|
c@116
|
318 break;
|
c@75
|
319 case RRType::NotValid:
|
c@116
|
320 break;
|
c@75
|
321 }
|
c@75
|
322
|
c@75
|
323 return rr;
|
c@75
|
324 }
|
c@75
|
325
|
c@75
|
326 void
|
c@75
|
327 writeResponseCapnp(RequestOrResponse &rr)
|
c@75
|
328 {
|
c@97
|
329 capnp::MallocMessageBuilder message;
|
c@97
|
330 piper::RpcResponse::Builder builder = message.initRoot<piper::RpcResponse>();
|
c@75
|
331
|
c@75
|
332 buildId(builder, rr.id);
|
c@75
|
333
|
c@75
|
334 if (!rr.success) {
|
c@75
|
335
|
c@116
|
336 VampnProto::buildRpcResponse_Error(builder, rr.errorText, rr.type);
|
c@75
|
337
|
c@75
|
338 } else {
|
c@116
|
339
|
c@116
|
340 switch (rr.type) {
|
c@75
|
341
|
c@116
|
342 case RRType::List:
|
c@116
|
343 VampnProto::buildRpcResponse_List(builder, rr.listResponse);
|
c@116
|
344 break;
|
c@116
|
345 case RRType::Load:
|
c@116
|
346 VampnProto::buildRpcResponse_Load(builder, rr.loadResponse, mapper);
|
c@116
|
347 break;
|
c@116
|
348 case RRType::Configure:
|
c@116
|
349 VampnProto::buildRpcResponse_Configure(builder, rr.configurationResponse, mapper);
|
c@116
|
350 break;
|
c@116
|
351 case RRType::Process:
|
c@116
|
352 VampnProto::buildRpcResponse_Process(builder, rr.processResponse, mapper);
|
c@116
|
353 break;
|
c@116
|
354 case RRType::Finish:
|
c@116
|
355 VampnProto::buildRpcResponse_Finish(builder, rr.finishResponse, mapper);
|
c@116
|
356 break;
|
c@116
|
357 case RRType::NotValid:
|
c@116
|
358 break;
|
c@116
|
359 }
|
c@75
|
360 }
|
c@75
|
361
|
c@75
|
362 writeMessageToFd(1, message);
|
c@75
|
363 }
|
c@75
|
364
|
c@75
|
365 void
|
c@75
|
366 writeExceptionCapnp(const std::exception &e, RRType type)
|
c@75
|
367 {
|
c@97
|
368 capnp::MallocMessageBuilder message;
|
c@97
|
369 piper::RpcResponse::Builder builder = message.initRoot<piper::RpcResponse>();
|
c@75
|
370 VampnProto::buildRpcResponse_Exception(builder, e, type);
|
c@75
|
371
|
c@75
|
372 writeMessageToFd(1, message);
|
c@75
|
373 }
|
c@75
|
374
|
c@75
|
375 RequestOrResponse
|
c@116
|
376 handleRequest(const RequestOrResponse &request, bool debug)
|
c@75
|
377 {
|
c@75
|
378 RequestOrResponse response;
|
c@75
|
379 response.direction = RequestOrResponse::Response;
|
c@75
|
380 response.type = request.type;
|
c@75
|
381
|
c@75
|
382 switch (request.type) {
|
c@75
|
383
|
c@75
|
384 case RRType::List:
|
c@116
|
385 response.listResponse = LoaderRequests().listPluginData();
|
c@116
|
386 response.success = true;
|
c@116
|
387 break;
|
c@75
|
388
|
c@75
|
389 case RRType::Load:
|
c@116
|
390 response.loadResponse = LoaderRequests().loadPlugin(request.loadRequest);
|
c@116
|
391 if (response.loadResponse.plugin != nullptr) {
|
c@116
|
392 mapper.addPlugin(response.loadResponse.plugin);
|
c@116
|
393 if (debug) {
|
c@116
|
394 cerr << "piper-vamp-server " << pid << ": loaded plugin, handle = " << mapper.pluginToHandle(response.loadResponse.plugin) << endl;
|
c@116
|
395 }
|
c@116
|
396 response.success = true;
|
c@116
|
397 }
|
c@116
|
398 break;
|
c@116
|
399
|
c@75
|
400 case RRType::Configure:
|
c@75
|
401 {
|
c@116
|
402 auto &creq = request.configurationRequest;
|
c@116
|
403 auto h = mapper.pluginToHandle(creq.plugin);
|
c@116
|
404 if (mapper.isConfigured(h)) {
|
c@116
|
405 throw runtime_error("plugin has already been configured");
|
c@116
|
406 }
|
c@75
|
407
|
c@116
|
408 response.configurationResponse = LoaderRequests().configurePlugin(creq);
|
c@116
|
409
|
c@116
|
410 if (!response.configurationResponse.outputs.empty()) {
|
c@116
|
411 mapper.markConfigured
|
c@116
|
412 (h, creq.configuration.channelCount, creq.configuration.blockSize);
|
c@116
|
413 response.success = true;
|
c@116
|
414 }
|
c@116
|
415 break;
|
c@75
|
416 }
|
c@75
|
417
|
c@75
|
418 case RRType::Process:
|
c@75
|
419 {
|
c@116
|
420 auto &preq = request.processRequest;
|
c@116
|
421 auto h = mapper.pluginToHandle(preq.plugin);
|
c@116
|
422 if (!mapper.isConfigured(h)) {
|
c@116
|
423 throw runtime_error("plugin has not been configured");
|
c@116
|
424 }
|
c@75
|
425
|
c@116
|
426 int channels = int(preq.inputBuffers.size());
|
c@116
|
427 if (channels != mapper.getChannelCount(h)) {
|
c@116
|
428 throw runtime_error("wrong number of channels supplied to process");
|
c@116
|
429 }
|
c@116
|
430
|
c@116
|
431 const float **fbuffers = new const float *[channels];
|
c@116
|
432 for (int i = 0; i < channels; ++i) {
|
c@116
|
433 if (int(preq.inputBuffers[i].size()) != mapper.getBlockSize(h)) {
|
c@116
|
434 delete[] fbuffers;
|
c@116
|
435 throw runtime_error("wrong block size supplied to process");
|
c@116
|
436 }
|
c@116
|
437 fbuffers[i] = preq.inputBuffers[i].data();
|
c@116
|
438 }
|
c@75
|
439
|
c@116
|
440 response.processResponse.plugin = preq.plugin;
|
c@116
|
441 response.processResponse.features =
|
c@116
|
442 preq.plugin->process(fbuffers, preq.timestamp);
|
c@116
|
443 response.success = true;
|
c@75
|
444
|
c@116
|
445 delete[] fbuffers;
|
c@116
|
446 break;
|
c@75
|
447 }
|
c@75
|
448
|
c@75
|
449 case RRType::Finish:
|
c@75
|
450 {
|
c@116
|
451 auto &freq = request.finishRequest;
|
c@116
|
452 response.finishResponse.plugin = freq.plugin;
|
c@77
|
453
|
c@116
|
454 auto h = mapper.pluginToHandle(freq.plugin);
|
c@77
|
455 // Finish can be called (to unload the plugin) even if the
|
c@77
|
456 // plugin has never been configured or used. But we want to
|
c@77
|
457 // make sure we call getRemainingFeatures only if we have
|
c@77
|
458 // actually configured the plugin.
|
c@116
|
459 if (mapper.isConfigured(h)) {
|
c@77
|
460 response.finishResponse.features = freq.plugin->getRemainingFeatures();
|
c@116
|
461 }
|
c@75
|
462
|
c@116
|
463 // We do not delete the plugin here -- we need it in the
|
c@116
|
464 // mapper when converting the features. It gets deleted in the
|
c@116
|
465 // calling function.
|
c@116
|
466 response.success = true;
|
c@116
|
467 break;
|
c@75
|
468 }
|
c@75
|
469
|
c@75
|
470 case RRType::NotValid:
|
c@116
|
471 break;
|
c@75
|
472 }
|
c@75
|
473
|
c@75
|
474 return response;
|
c@75
|
475 }
|
c@75
|
476
|
c@116
|
477 RequestOrResponse
|
c@116
|
478 readRequest(string format)
|
c@75
|
479 {
|
c@116
|
480 if (format == "capnp") {
|
c@116
|
481 return readRequestCapnp();
|
c@116
|
482 } else if (format == "json") {
|
c@116
|
483 string err;
|
c@116
|
484 auto result = readRequestJson(err);
|
c@116
|
485 if (err != "") throw runtime_error(err);
|
c@116
|
486 else return result;
|
c@116
|
487 } else {
|
c@116
|
488 throw runtime_error("unknown input format \"" + format + "\"");
|
c@116
|
489 }
|
c@116
|
490 }
|
c@116
|
491
|
c@116
|
492 void
|
c@116
|
493 writeResponse(string format, RequestOrResponse &rr)
|
c@116
|
494 {
|
c@116
|
495 if (format == "capnp") {
|
c@116
|
496 writeResponseCapnp(rr);
|
c@116
|
497 } else if (format == "json") {
|
c@116
|
498 writeResponseJson(rr, false);
|
c@116
|
499 } else {
|
c@116
|
500 throw runtime_error("unknown output format \"" + format + "\"");
|
c@116
|
501 }
|
c@116
|
502 }
|
c@116
|
503
|
c@116
|
504 void
|
c@116
|
505 writeException(string format, const std::exception &e, RRType type)
|
c@116
|
506 {
|
c@116
|
507 if (format == "capnp") {
|
c@116
|
508 writeExceptionCapnp(e, type);
|
c@116
|
509 } else if (format == "json") {
|
c@116
|
510 writeExceptionJson(e, type);
|
c@116
|
511 } else {
|
c@116
|
512 throw runtime_error("unknown output format \"" + format + "\"");
|
c@116
|
513 }
|
c@116
|
514 }
|
c@116
|
515
|
c@116
|
516 int main(int argc, char **argv)
|
c@116
|
517 {
|
c@116
|
518 if (argc != 2 && argc != 3) {
|
c@116
|
519 usage();
|
c@75
|
520 }
|
c@75
|
521
|
c@116
|
522 bool debug = false;
|
c@112
|
523
|
c@116
|
524 string arg = argv[1];
|
c@116
|
525 if (arg == "-h") {
|
c@116
|
526 if (argc == 2) {
|
c@116
|
527 usage(true);
|
c@116
|
528 } else {
|
c@116
|
529 usage();
|
c@116
|
530 }
|
c@116
|
531 } else if (arg == "-v") {
|
c@116
|
532 if (argc == 2) {
|
c@116
|
533 version();
|
c@116
|
534 } else {
|
c@116
|
535 usage();
|
c@116
|
536 }
|
c@116
|
537 } else if (arg == "-d") {
|
c@116
|
538 if (argc == 2) {
|
c@116
|
539 usage();
|
c@116
|
540 } else {
|
c@116
|
541 debug = true;
|
c@116
|
542 arg = argv[2];
|
c@116
|
543 }
|
c@116
|
544 }
|
c@116
|
545
|
c@116
|
546 string format = arg;
|
c@116
|
547
|
c@116
|
548 if (format != "capnp" && format != "json") {
|
c@116
|
549 usage();
|
c@116
|
550 }
|
c@116
|
551
|
c@116
|
552 if (debug) {
|
c@116
|
553 cerr << myname << " " << pid << ": waiting for format: " << format << endl;
|
c@116
|
554 }
|
c@116
|
555
|
c@75
|
556 while (true) {
|
c@75
|
557
|
c@116
|
558 RequestOrResponse request;
|
c@116
|
559
|
c@116
|
560 try {
|
c@75
|
561
|
c@116
|
562 request = readRequest(format);
|
c@116
|
563
|
c@116
|
564 // NotValid without an exception indicates EOF:
|
c@116
|
565 if (request.type == RRType::NotValid) {
|
c@116
|
566 if (debug) {
|
c@116
|
567 cerr << myname << " " << pid << ": eof reached, exiting" << endl;
|
c@116
|
568 }
|
c@116
|
569 break;
|
c@116
|
570 }
|
c@75
|
571
|
c@116
|
572 if (debug) {
|
c@116
|
573 cerr << myname << " " << pid << ": request received, of type "
|
c@116
|
574 << int(request.type)
|
c@116
|
575 << endl;
|
c@116
|
576 }
|
c@75
|
577
|
c@116
|
578 RequestOrResponse response = handleRequest(request, debug);
|
c@75
|
579 response.id = request.id;
|
c@75
|
580
|
c@116
|
581 if (debug) {
|
c@116
|
582 cerr << myname << " " << pid << ": request handled, writing response"
|
c@116
|
583 << endl;
|
c@116
|
584 }
|
c@116
|
585
|
c@116
|
586 writeResponse(format, response);
|
c@75
|
587
|
c@116
|
588 if (debug) {
|
c@116
|
589 cerr << myname << " " << pid << ": response written" << endl;
|
c@116
|
590 }
|
c@75
|
591
|
c@116
|
592 if (request.type == RRType::Finish) {
|
c@116
|
593 auto h = mapper.pluginToHandle(request.finishRequest.plugin);
|
c@116
|
594 if (debug) {
|
c@116
|
595 cerr << myname << " " << pid << ": deleting the plugin with handle " << h << endl;
|
c@116
|
596 }
|
c@116
|
597 mapper.removePlugin(h);
|
c@116
|
598 delete request.finishRequest.plugin;
|
c@116
|
599 }
|
c@116
|
600
|
c@116
|
601 } catch (std::exception &e) {
|
c@75
|
602
|
c@116
|
603 if (debug) {
|
c@116
|
604 cerr << myname << " " << pid << ": error: " << e.what() << endl;
|
c@116
|
605 }
|
c@75
|
606
|
c@116
|
607 writeException(format, e, request.type);
|
c@116
|
608
|
c@116
|
609 exit(1);
|
c@116
|
610 }
|
c@75
|
611 }
|
c@75
|
612
|
c@75
|
613 exit(0);
|
c@75
|
614 }
|