annotate vamp-server/server.cpp @ 123:6b11ca6bb0a3

Binary modes
author Chris Cannam <c.cannam@qmul.ac.uk>
date Thu, 27 Oct 2016 16:13:57 +0100
parents 88ecaf8b163a
children
rev   line source
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@123 60 // for _setmode stuff
c@123 61 #ifdef _WIN32
c@123 62 #include <io.h>
c@123 63 #include <fcntl.h>
c@123 64 #endif
c@123 65
c@75 66 using namespace std;
c@116 67 using namespace json11;
c@97 68 using namespace piper_vamp;
c@75 69 using namespace Vamp;
c@75 70
c@102 71 //!!! This could be faster and lighter:
c@102 72 // - Use Capnp structures directly rather than converting to vamp-support ones
c@102 73 // - Use Vamp C API (vamp.h) directly rather than converting to C++
c@102 74 //!!! Doing the above for process() and finish() alone would be a good start
c@102 75
c@116 76 static string myname = "piper-vamp-server";
c@116 77
c@116 78 static void version()
c@75 79 {
c@116 80 cout << "1.0" << endl;
c@116 81 exit(0);
c@116 82 }
c@116 83
c@116 84 static void usage(bool successful = false)
c@116 85 {
c@75 86 cerr << "\n" << myname <<
c@116 87 ": Load and run Vamp plugins in response to Piper messages\n\n"
c@116 88 " Usage: " << myname << " [-d] <format>\n"
c@116 89 " " << myname << " -v\n"
c@116 90 " " << myname << " -h\n\n"
c@116 91 " where\n"
c@116 92 " <format>: the format to read and write messages in (\"json\" or \"capnp\")\n"
c@116 93 " -d: also print debug information to stderr\n"
c@116 94 " -v: print version number to stdout and exit\n"
c@116 95 " -h: print this text to stderr and exit\n\n"
c@116 96 "Expects Piper request messages in either Cap'n Proto or JSON format on stdin,\n"
c@116 97 "and writes response messages in the same format to stdout.\n\n";
c@116 98 if (successful) exit(0);
c@116 99 else exit(2);
c@75 100 }
c@75 101
c@75 102 static CountingPluginHandleMapper mapper;
c@75 103
c@116 104 static RequestOrResponse::RpcId
c@116 105 readId(const piper::RpcRequest::Reader &r)
c@75 106 {
c@75 107 int number;
c@75 108 string tag;
c@75 109 switch (r.getId().which()) {
c@97 110 case piper::RpcRequest::Id::Which::NUMBER:
c@75 111 number = r.getId().getNumber();
c@75 112 return { RequestOrResponse::RpcId::Number, number, "" };
c@97 113 case piper::RpcRequest::Id::Which::TAG:
c@75 114 tag = r.getId().getTag();
c@75 115 return { RequestOrResponse::RpcId::Tag, 0, tag };
c@97 116 case piper::RpcRequest::Id::Which::NONE:
c@75 117 return { RequestOrResponse::RpcId::Absent, 0, "" };
c@75 118 }
c@75 119 return {};
c@75 120 }
c@75 121
c@116 122 static void
c@116 123 buildId(piper::RpcResponse::Builder &b, const RequestOrResponse::RpcId &id)
c@75 124 {
c@75 125 switch (id.type) {
c@75 126 case RequestOrResponse::RpcId::Number:
c@75 127 b.getId().setNumber(id.number);
c@75 128 break;
c@75 129 case RequestOrResponse::RpcId::Tag:
c@75 130 b.getId().setTag(id.tag);
c@75 131 break;
c@75 132 case RequestOrResponse::RpcId::Absent:
c@75 133 b.getId().setNone();
c@75 134 break;
c@75 135 }
c@75 136 }
c@75 137
c@116 138 static RequestOrResponse::RpcId
c@116 139 readJsonId(const Json &j)
c@116 140 {
c@116 141 RequestOrResponse::RpcId id;
c@116 142
c@116 143 if (j["id"].is_number()) {
c@116 144 id.type = RequestOrResponse::RpcId::Number;
c@116 145 id.number = j["id"].number_value();
c@116 146 } else if (j["id"].is_string()) {
c@116 147 id.type = RequestOrResponse::RpcId::Tag;
c@116 148 id.tag = j["id"].string_value();
c@116 149 } else {
c@116 150 id.type = RequestOrResponse::RpcId::Absent;
c@116 151 }
c@116 152
c@116 153 return id;
c@116 154 }
c@116 155
c@116 156 static Json
c@116 157 writeJsonId(const RequestOrResponse::RpcId &id)
c@116 158 {
c@116 159 if (id.type == RequestOrResponse::RpcId::Number) {
c@116 160 return id.number;
c@116 161 } else if (id.type == RequestOrResponse::RpcId::Tag) {
c@116 162 return id.tag;
c@116 163 } else {
c@116 164 return Json();
c@116 165 }
c@116 166 }
c@116 167
c@116 168 static Json
c@116 169 convertRequestJson(string input, string &err)
c@116 170 {
c@116 171 Json j = Json::parse(input, err);
c@116 172 if (err != "") {
c@116 173 err = "invalid json: " + err;
c@116 174 return {};
c@116 175 }
c@116 176 if (!j.is_object()) {
c@116 177 err = "object expected at top level";
c@116 178 } else if (!j["method"].is_string()) {
c@116 179 err = "string expected for method field";
c@116 180 } else if (!j["params"].is_null() && !j["params"].is_object()) {
c@116 181 err = "object expected for params field";
c@116 182 }
c@116 183 return j;
c@116 184 }
c@116 185
c@116 186 RequestOrResponse
c@116 187 readRequestJson(string &err)
c@116 188 {
c@116 189 RequestOrResponse rr;
c@116 190 rr.direction = RequestOrResponse::Request;
c@116 191
c@116 192 string input;
c@116 193 if (!getline(cin, input)) {
c@116 194 // the EOF case, not actually an error
c@116 195 rr.type = RRType::NotValid;
c@116 196 return rr;
c@116 197 }
c@116 198
c@116 199 Json j = convertRequestJson(input, err);
c@116 200 if (err != "") return {};
c@116 201
c@116 202 rr.type = VampJson::getRequestResponseType(j, err);
c@116 203 if (err != "") return {};
c@116 204
c@116 205 rr.id = readJsonId(j);
c@116 206
c@116 207 VampJson::BufferSerialisation serialisation =
c@116 208 VampJson::BufferSerialisation::Array;
c@116 209
c@116 210 switch (rr.type) {
c@116 211
c@116 212 case RRType::List:
c@116 213 VampJson::toRpcRequest_List(j, err); // type check only
c@116 214 break;
c@116 215 case RRType::Load:
c@116 216 rr.loadRequest = VampJson::toRpcRequest_Load(j, err);
c@116 217 break;
c@116 218 case RRType::Configure:
c@116 219 rr.configurationRequest = VampJson::toRpcRequest_Configure(j, mapper, err);
c@116 220 break;
c@116 221 case RRType::Process:
c@116 222 rr.processRequest = VampJson::toRpcRequest_Process(j, mapper, serialisation, err);
c@116 223 break;
c@116 224 case RRType::Finish:
c@116 225 rr.finishRequest = VampJson::toRpcRequest_Finish(j, mapper, err);
c@116 226 break;
c@116 227 case RRType::NotValid:
c@116 228 break;
c@116 229 }
c@116 230
c@116 231 return rr;
c@116 232 }
c@116 233
c@116 234 void
c@116 235 writeResponseJson(RequestOrResponse &rr, bool useBase64)
c@116 236 {
c@116 237 Json j;
c@116 238
c@116 239 VampJson::BufferSerialisation serialisation =
c@116 240 (useBase64 ?
c@116 241 VampJson::BufferSerialisation::Base64 :
c@116 242 VampJson::BufferSerialisation::Array);
c@116 243
c@116 244 Json id = writeJsonId(rr.id);
c@116 245
c@116 246 if (!rr.success) {
c@116 247
c@116 248 j = VampJson::fromError(rr.errorText, rr.type, id);
c@116 249
c@116 250 } else {
c@116 251
c@116 252 switch (rr.type) {
c@116 253
c@116 254 case RRType::List:
c@116 255 j = VampJson::fromRpcResponse_List(rr.listResponse, id);
c@116 256 break;
c@116 257 case RRType::Load:
c@116 258 j = VampJson::fromRpcResponse_Load(rr.loadResponse, mapper, id);
c@116 259 break;
c@116 260 case RRType::Configure:
c@116 261 j = VampJson::fromRpcResponse_Configure(rr.configurationResponse,
c@116 262 mapper, id);
c@116 263 break;
c@116 264 case RRType::Process:
c@116 265 j = VampJson::fromRpcResponse_Process
c@116 266 (rr.processResponse, mapper, serialisation, id);
c@116 267 break;
c@116 268 case RRType::Finish:
c@116 269 j = VampJson::fromRpcResponse_Finish
c@116 270 (rr.finishResponse, mapper, serialisation, id);
c@116 271 break;
c@116 272 case RRType::NotValid:
c@116 273 break;
c@116 274 }
c@116 275 }
c@116 276
c@116 277 cout << j.dump() << endl;
c@116 278 }
c@116 279
c@116 280 void
c@116 281 writeExceptionJson(const std::exception &e, RRType type)
c@116 282 {
c@116 283 Json j = VampJson::fromError(e.what(), type, Json());
c@116 284 cout << j.dump() << endl;
c@116 285 }
c@116 286
c@75 287 RequestOrResponse
c@75 288 readRequestCapnp()
c@75 289 {
c@75 290 RequestOrResponse rr;
c@75 291 rr.direction = RequestOrResponse::Request;
c@75 292
c@75 293 static kj::FdInputStream stream(0); // stdin
c@75 294 static kj::BufferedInputStreamWrapper buffered(stream);
c@75 295
c@75 296 if (buffered.tryGetReadBuffer() == nullptr) {
c@116 297 rr.type = RRType::NotValid;
c@116 298 return rr;
c@75 299 }
c@75 300
c@97 301 capnp::InputStreamMessageReader message(buffered);
c@97 302 piper::RpcRequest::Reader reader = message.getRoot<piper::RpcRequest>();
c@75 303
c@75 304 rr.type = VampnProto::getRequestResponseType(reader);
c@75 305 rr.id = readId(reader);
c@75 306
c@75 307 switch (rr.type) {
c@75 308
c@75 309 case RRType::List:
c@116 310 VampnProto::readRpcRequest_List(reader); // type check only
c@116 311 break;
c@75 312 case RRType::Load:
c@116 313 VampnProto::readRpcRequest_Load(rr.loadRequest, reader);
c@116 314 break;
c@75 315 case RRType::Configure:
c@116 316 VampnProto::readRpcRequest_Configure(rr.configurationRequest,
c@116 317 reader, mapper);
c@116 318 break;
c@75 319 case RRType::Process:
c@116 320 VampnProto::readRpcRequest_Process(rr.processRequest, reader, mapper);
c@116 321 break;
c@75 322 case RRType::Finish:
c@116 323 VampnProto::readRpcRequest_Finish(rr.finishRequest, reader, mapper);
c@116 324 break;
c@75 325 case RRType::NotValid:
c@116 326 break;
c@75 327 }
c@75 328
c@75 329 return rr;
c@75 330 }
c@75 331
c@75 332 void
c@75 333 writeResponseCapnp(RequestOrResponse &rr)
c@75 334 {
c@97 335 capnp::MallocMessageBuilder message;
c@97 336 piper::RpcResponse::Builder builder = message.initRoot<piper::RpcResponse>();
c@75 337
c@75 338 buildId(builder, rr.id);
c@75 339
c@75 340 if (!rr.success) {
c@75 341
c@116 342 VampnProto::buildRpcResponse_Error(builder, rr.errorText, rr.type);
c@75 343
c@75 344 } else {
c@116 345
c@116 346 switch (rr.type) {
c@75 347
c@116 348 case RRType::List:
c@116 349 VampnProto::buildRpcResponse_List(builder, rr.listResponse);
c@116 350 break;
c@116 351 case RRType::Load:
c@116 352 VampnProto::buildRpcResponse_Load(builder, rr.loadResponse, mapper);
c@116 353 break;
c@116 354 case RRType::Configure:
c@116 355 VampnProto::buildRpcResponse_Configure(builder, rr.configurationResponse, mapper);
c@116 356 break;
c@116 357 case RRType::Process:
c@116 358 VampnProto::buildRpcResponse_Process(builder, rr.processResponse, mapper);
c@116 359 break;
c@116 360 case RRType::Finish:
c@116 361 VampnProto::buildRpcResponse_Finish(builder, rr.finishResponse, mapper);
c@116 362 break;
c@116 363 case RRType::NotValid:
c@116 364 break;
c@116 365 }
c@75 366 }
c@75 367
c@75 368 writeMessageToFd(1, message);
c@75 369 }
c@75 370
c@75 371 void
c@75 372 writeExceptionCapnp(const std::exception &e, RRType type)
c@75 373 {
c@97 374 capnp::MallocMessageBuilder message;
c@97 375 piper::RpcResponse::Builder builder = message.initRoot<piper::RpcResponse>();
c@75 376 VampnProto::buildRpcResponse_Exception(builder, e, type);
c@75 377
c@75 378 writeMessageToFd(1, message);
c@75 379 }
c@75 380
c@75 381 RequestOrResponse
c@116 382 handleRequest(const RequestOrResponse &request, bool debug)
c@75 383 {
c@75 384 RequestOrResponse response;
c@75 385 response.direction = RequestOrResponse::Response;
c@75 386 response.type = request.type;
c@75 387
c@75 388 switch (request.type) {
c@75 389
c@75 390 case RRType::List:
c@116 391 response.listResponse = LoaderRequests().listPluginData();
c@116 392 response.success = true;
c@116 393 break;
c@75 394
c@75 395 case RRType::Load:
c@116 396 response.loadResponse = LoaderRequests().loadPlugin(request.loadRequest);
c@116 397 if (response.loadResponse.plugin != nullptr) {
c@116 398 mapper.addPlugin(response.loadResponse.plugin);
c@116 399 if (debug) {
c@116 400 cerr << "piper-vamp-server " << pid << ": loaded plugin, handle = " << mapper.pluginToHandle(response.loadResponse.plugin) << endl;
c@116 401 }
c@116 402 response.success = true;
c@116 403 }
c@116 404 break;
c@116 405
c@75 406 case RRType::Configure:
c@75 407 {
c@116 408 auto &creq = request.configurationRequest;
c@116 409 auto h = mapper.pluginToHandle(creq.plugin);
c@116 410 if (mapper.isConfigured(h)) {
c@116 411 throw runtime_error("plugin has already been configured");
c@116 412 }
c@75 413
c@116 414 response.configurationResponse = LoaderRequests().configurePlugin(creq);
c@116 415
c@116 416 if (!response.configurationResponse.outputs.empty()) {
c@116 417 mapper.markConfigured
c@116 418 (h, creq.configuration.channelCount, creq.configuration.blockSize);
c@116 419 response.success = true;
c@116 420 }
c@116 421 break;
c@75 422 }
c@75 423
c@75 424 case RRType::Process:
c@75 425 {
c@116 426 auto &preq = request.processRequest;
c@116 427 auto h = mapper.pluginToHandle(preq.plugin);
c@116 428 if (!mapper.isConfigured(h)) {
c@116 429 throw runtime_error("plugin has not been configured");
c@116 430 }
c@75 431
c@116 432 int channels = int(preq.inputBuffers.size());
c@116 433 if (channels != mapper.getChannelCount(h)) {
c@116 434 throw runtime_error("wrong number of channels supplied to process");
c@116 435 }
c@116 436
c@116 437 const float **fbuffers = new const float *[channels];
c@116 438 for (int i = 0; i < channels; ++i) {
c@116 439 if (int(preq.inputBuffers[i].size()) != mapper.getBlockSize(h)) {
c@116 440 delete[] fbuffers;
c@116 441 throw runtime_error("wrong block size supplied to process");
c@116 442 }
c@116 443 fbuffers[i] = preq.inputBuffers[i].data();
c@116 444 }
c@75 445
c@116 446 response.processResponse.plugin = preq.plugin;
c@116 447 response.processResponse.features =
c@116 448 preq.plugin->process(fbuffers, preq.timestamp);
c@116 449 response.success = true;
c@75 450
c@116 451 delete[] fbuffers;
c@116 452 break;
c@75 453 }
c@75 454
c@75 455 case RRType::Finish:
c@75 456 {
c@116 457 auto &freq = request.finishRequest;
c@116 458 response.finishResponse.plugin = freq.plugin;
c@77 459
c@116 460 auto h = mapper.pluginToHandle(freq.plugin);
c@77 461 // Finish can be called (to unload the plugin) even if the
c@77 462 // plugin has never been configured or used. But we want to
c@77 463 // make sure we call getRemainingFeatures only if we have
c@77 464 // actually configured the plugin.
c@116 465 if (mapper.isConfigured(h)) {
c@77 466 response.finishResponse.features = freq.plugin->getRemainingFeatures();
c@116 467 }
c@75 468
c@116 469 // We do not delete the plugin here -- we need it in the
c@116 470 // mapper when converting the features. It gets deleted in the
c@116 471 // calling function.
c@116 472 response.success = true;
c@116 473 break;
c@75 474 }
c@75 475
c@75 476 case RRType::NotValid:
c@116 477 break;
c@75 478 }
c@75 479
c@75 480 return response;
c@75 481 }
c@75 482
c@116 483 RequestOrResponse
c@116 484 readRequest(string format)
c@75 485 {
c@116 486 if (format == "capnp") {
c@116 487 return readRequestCapnp();
c@116 488 } else if (format == "json") {
c@116 489 string err;
c@116 490 auto result = readRequestJson(err);
c@116 491 if (err != "") throw runtime_error(err);
c@116 492 else return result;
c@116 493 } else {
c@116 494 throw runtime_error("unknown input format \"" + format + "\"");
c@116 495 }
c@116 496 }
c@116 497
c@116 498 void
c@116 499 writeResponse(string format, RequestOrResponse &rr)
c@116 500 {
c@116 501 if (format == "capnp") {
c@116 502 writeResponseCapnp(rr);
c@116 503 } else if (format == "json") {
c@116 504 writeResponseJson(rr, false);
c@116 505 } else {
c@116 506 throw runtime_error("unknown output format \"" + format + "\"");
c@116 507 }
c@116 508 }
c@116 509
c@116 510 void
c@116 511 writeException(string format, const std::exception &e, RRType type)
c@116 512 {
c@116 513 if (format == "capnp") {
c@116 514 writeExceptionCapnp(e, type);
c@116 515 } else if (format == "json") {
c@116 516 writeExceptionJson(e, type);
c@116 517 } else {
c@116 518 throw runtime_error("unknown output format \"" + format + "\"");
c@116 519 }
c@116 520 }
c@116 521
c@116 522 int main(int argc, char **argv)
c@116 523 {
c@116 524 if (argc != 2 && argc != 3) {
c@116 525 usage();
c@75 526 }
c@75 527
c@116 528 bool debug = false;
c@112 529
c@116 530 string arg = argv[1];
c@116 531 if (arg == "-h") {
c@116 532 if (argc == 2) {
c@116 533 usage(true);
c@116 534 } else {
c@116 535 usage();
c@116 536 }
c@116 537 } else if (arg == "-v") {
c@116 538 if (argc == 2) {
c@116 539 version();
c@116 540 } else {
c@116 541 usage();
c@116 542 }
c@116 543 } else if (arg == "-d") {
c@116 544 if (argc == 2) {
c@116 545 usage();
c@116 546 } else {
c@116 547 debug = true;
c@116 548 arg = argv[2];
c@116 549 }
c@116 550 }
c@116 551
c@116 552 string format = arg;
c@116 553
c@116 554 if (format != "capnp" && format != "json") {
c@116 555 usage();
c@116 556 }
c@116 557
c@123 558 #ifdef _WIN32
c@123 559 if (format == "capnp") {
c@123 560 int result = _setmode(_fileno(stdin), _O_BINARY);
c@123 561 if (result == -1) {
c@123 562 cerr << "Failed to set binary mode on stdin, necessary for capnp format" << endl;
c@123 563 exit(1);
c@123 564 }
c@123 565 result = _setmode(_fileno(stdout), _O_BINARY);
c@123 566 if (result == -1) {
c@123 567 cerr << "Failed to set binary mode on stdout, necessary for capnp format" << endl;
c@123 568 exit(1);
c@123 569 }
c@123 570 }
c@123 571 #endif
c@123 572
c@116 573 if (debug) {
c@116 574 cerr << myname << " " << pid << ": waiting for format: " << format << endl;
c@116 575 }
c@116 576
c@75 577 while (true) {
c@75 578
c@116 579 RequestOrResponse request;
c@116 580
c@116 581 try {
c@75 582
c@116 583 request = readRequest(format);
c@116 584
c@116 585 // NotValid without an exception indicates EOF:
c@116 586 if (request.type == RRType::NotValid) {
c@116 587 if (debug) {
c@116 588 cerr << myname << " " << pid << ": eof reached, exiting" << endl;
c@116 589 }
c@116 590 break;
c@116 591 }
c@75 592
c@116 593 if (debug) {
c@116 594 cerr << myname << " " << pid << ": request received, of type "
c@116 595 << int(request.type)
c@116 596 << endl;
c@116 597 }
c@75 598
c@116 599 RequestOrResponse response = handleRequest(request, debug);
c@75 600 response.id = request.id;
c@75 601
c@116 602 if (debug) {
c@116 603 cerr << myname << " " << pid << ": request handled, writing response"
c@116 604 << endl;
c@116 605 }
c@116 606
c@116 607 writeResponse(format, response);
c@75 608
c@116 609 if (debug) {
c@116 610 cerr << myname << " " << pid << ": response written" << endl;
c@116 611 }
c@75 612
c@116 613 if (request.type == RRType::Finish) {
c@116 614 auto h = mapper.pluginToHandle(request.finishRequest.plugin);
c@116 615 if (debug) {
c@116 616 cerr << myname << " " << pid << ": deleting the plugin with handle " << h << endl;
c@116 617 }
c@116 618 mapper.removePlugin(h);
c@116 619 delete request.finishRequest.plugin;
c@116 620 }
c@116 621
c@116 622 } catch (std::exception &e) {
c@75 623
c@116 624 if (debug) {
c@116 625 cerr << myname << " " << pid << ": error: " << e.what() << endl;
c@116 626 }
c@75 627
c@116 628 writeException(format, e, request.type);
c@120 629
c@120 630 //!!! some exceptions should not be continued after,
c@120 631 //! but json/capnp parser ones should
c@120 632 //exit(1);
c@116 633 }
c@75 634 }
c@75 635
c@75 636 exit(0);
c@75 637 }