annotate vamp-server/simple-server.cpp @ 133:74a7c2a8d6b6

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