annotate vamp-server/convert.cpp @ 118:ff3fd8d1b2dc

Boilerplate comments
author Chris Cannam <c.cannam@qmul.ac.uk>
date Thu, 27 Oct 2016 12:01:37 +0100
parents d15cb1151d76
children 6b11ca6bb0a3
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@75 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/PreservingPluginHandleMapper.h"
c@75 40
c@75 41 #include <iostream>
c@75 42 #include <sstream>
c@75 43 #include <stdexcept>
c@75 44
c@91 45 #include <capnp/serialize.h>
c@91 46
c@75 47 using namespace std;
c@75 48 using namespace json11;
c@97 49 using namespace piper_vamp;
c@75 50
c@75 51 void usage()
c@75 52 {
c@75 53 string myname = "piper-convert";
c@75 54 cerr << "\n" << myname <<
c@116 55 ": Validate and convert Piper request and response messages\n\n"
c@116 56 " Usage: " << myname << " [-i <informat>] [-o <outformat>] request\n"
c@116 57 " " << myname << " [-i <informat>] [-o <outformat>] response\n\n"
c@116 58 " where\n"
c@116 59 " <informat>: the format to read from stdin\n"
c@116 60 " (\"json\" or \"capnp\", default is \"json\")\n"
c@116 61 " <outformat>: the format to convert to and write to stdout\n"
c@116 62 " (\"json\", \"json-b64\" or \"capnp\", default is \"json\")\n"
c@116 63 " request|response: whether messages are Vamp request or response type\n\n"
c@116 64 "If <informat> and <outformat> differ, convert from <informat> to <outformat>.\n"
c@116 65 "If <informat> and <outformat> are the same, just check validity of incoming\n"
c@116 66 "messages and pass them to output.\n\n"
c@116 67 "Specifying \"json-b64\" as output format forces base64 encoding for process and\n"
c@116 68 "feature blocks, unlike the \"json\" output format which uses text encoding.\n"
c@116 69 "The \"json\" input format accepts either.\n\n";
c@75 70
c@75 71 exit(2);
c@75 72 }
c@75 73
c@75 74 Json
c@75 75 convertRequestJson(string input, string &err)
c@75 76 {
c@75 77 Json j = Json::parse(input, err);
c@75 78 if (err != "") {
c@116 79 err = "invalid json: " + err;
c@116 80 return {};
c@75 81 }
c@75 82 if (!j.is_object()) {
c@116 83 err = "object expected at top level";
c@75 84 } else if (!j["method"].is_string()) {
c@116 85 err = "string expected for method field";
c@75 86 } else if (!j["params"].is_null() && !j["params"].is_object()) {
c@116 87 err = "object expected for params field";
c@75 88 }
c@75 89 return j;
c@75 90 }
c@75 91
c@75 92 Json
c@75 93 convertResponseJson(string input, string &err)
c@75 94 {
c@75 95 Json j = Json::parse(input, err);
c@75 96 if (err != "") {
c@116 97 err = "invalid json: " + err;
c@116 98 return {};
c@75 99 }
c@75 100 if (!j.is_object()) {
c@116 101 err = "object expected at top level";
c@75 102 } else {
c@75 103 if (!j["result"].is_object()) {
c@75 104 if (!j["error"].is_object()) {
c@75 105 err = "expected either result or error object";
c@75 106 }
c@75 107 }
c@75 108 }
c@75 109 return j;
c@75 110 }
c@75 111
c@75 112 //!!! Lots of potential for refactoring the conversion classes based
c@75 113 //!!! on the common matter in the following eight functions...
c@75 114
c@75 115 PreservingPluginHandleMapper mapper;
c@75 116
c@75 117 static RequestOrResponse::RpcId
c@75 118 readJsonId(const Json &j)
c@75 119 {
c@75 120 RequestOrResponse::RpcId id;
c@75 121
c@75 122 if (j["id"].is_number()) {
c@75 123 id.type = RequestOrResponse::RpcId::Number;
c@75 124 id.number = j["id"].number_value();
c@75 125 } else if (j["id"].is_string()) {
c@75 126 id.type = RequestOrResponse::RpcId::Tag;
c@75 127 id.tag = j["id"].string_value();
c@75 128 } else {
c@75 129 id.type = RequestOrResponse::RpcId::Absent;
c@75 130 }
c@75 131
c@75 132 return id;
c@75 133 }
c@75 134
c@75 135 static Json
c@75 136 writeJsonId(const RequestOrResponse::RpcId &id)
c@75 137 {
c@75 138 if (id.type == RequestOrResponse::RpcId::Number) {
c@75 139 return id.number;
c@75 140 } else if (id.type == RequestOrResponse::RpcId::Tag) {
c@75 141 return id.tag;
c@75 142 } else {
c@75 143 return Json();
c@75 144 }
c@75 145 }
c@75 146
c@75 147 template <typename Reader>
c@75 148 static RequestOrResponse::RpcId
c@75 149 readCapnpId(const Reader &r)
c@75 150 {
c@75 151 int number;
c@75 152 string tag;
c@75 153 switch (r.getId().which()) {
c@97 154 case piper::RpcRequest::Id::Which::NUMBER:
c@75 155 number = r.getId().getNumber();
c@75 156 return { RequestOrResponse::RpcId::Number, number, "" };
c@97 157 case piper::RpcRequest::Id::Which::TAG:
c@75 158 tag = r.getId().getTag();
c@75 159 return { RequestOrResponse::RpcId::Tag, 0, tag };
c@97 160 case piper::RpcRequest::Id::Which::NONE:
c@75 161 return { RequestOrResponse::RpcId::Absent, 0, "" };
c@75 162 }
c@75 163 return {};
c@75 164 }
c@75 165
c@75 166 template <typename Builder>
c@75 167 static void
c@75 168 buildCapnpId(Builder &b, const RequestOrResponse::RpcId &id)
c@75 169 {
c@75 170 switch (id.type) {
c@75 171 case RequestOrResponse::RpcId::Number:
c@75 172 b.getId().setNumber(id.number);
c@75 173 break;
c@75 174 case RequestOrResponse::RpcId::Tag:
c@75 175 b.getId().setTag(id.tag);
c@75 176 break;
c@75 177 case RequestOrResponse::RpcId::Absent:
c@75 178 b.getId().setNone();
c@75 179 break;
c@75 180 }
c@75 181 }
c@75 182
c@75 183 RequestOrResponse
c@75 184 readRequestJson(string &err)
c@75 185 {
c@75 186 RequestOrResponse rr;
c@75 187 rr.direction = RequestOrResponse::Request;
c@75 188
c@75 189 string input;
c@75 190 if (!getline(cin, input)) {
c@116 191 // the EOF case, not actually an error
c@116 192 rr.type = RRType::NotValid;
c@116 193 return rr;
c@75 194 }
c@75 195
c@75 196 Json j = convertRequestJson(input, err);
c@75 197 if (err != "") return {};
c@75 198
c@75 199 rr.type = VampJson::getRequestResponseType(j, err);
c@75 200 if (err != "") return {};
c@75 201
c@75 202 rr.id = readJsonId(j);
c@75 203
c@75 204 VampJson::BufferSerialisation serialisation =
c@75 205 VampJson::BufferSerialisation::Array;
c@75 206
c@75 207 switch (rr.type) {
c@75 208
c@75 209 case RRType::List:
c@116 210 VampJson::toRpcRequest_List(j, err); // type check only
c@116 211 break;
c@75 212 case RRType::Load:
c@116 213 rr.loadRequest = VampJson::toRpcRequest_Load(j, err);
c@116 214 break;
c@75 215 case RRType::Configure:
c@116 216 rr.configurationRequest = VampJson::toRpcRequest_Configure(j, mapper, err);
c@116 217 break;
c@75 218 case RRType::Process:
c@116 219 rr.processRequest = VampJson::toRpcRequest_Process(j, mapper, serialisation, err);
c@116 220 break;
c@75 221 case RRType::Finish:
c@116 222 rr.finishRequest = VampJson::toRpcRequest_Finish(j, mapper, err);
c@116 223 break;
c@75 224 case RRType::NotValid:
c@116 225 break;
c@75 226 }
c@75 227
c@75 228 return rr;
c@75 229 }
c@75 230
c@75 231 void
c@75 232 writeRequestJson(RequestOrResponse &rr, bool useBase64)
c@75 233 {
c@75 234 Json j;
c@75 235
c@75 236 VampJson::BufferSerialisation serialisation =
c@75 237 (useBase64 ?
c@75 238 VampJson::BufferSerialisation::Base64 :
c@75 239 VampJson::BufferSerialisation::Array);
c@75 240
c@75 241 Json id = writeJsonId(rr.id);
c@75 242
c@75 243 switch (rr.type) {
c@75 244
c@75 245 case RRType::List:
c@116 246 j = VampJson::fromRpcRequest_List(id);
c@116 247 break;
c@75 248 case RRType::Load:
c@116 249 j = VampJson::fromRpcRequest_Load(rr.loadRequest, id);
c@116 250 break;
c@75 251 case RRType::Configure:
c@116 252 j = VampJson::fromRpcRequest_Configure(rr.configurationRequest, mapper, id);
c@116 253 break;
c@75 254 case RRType::Process:
c@116 255 j = VampJson::fromRpcRequest_Process
c@116 256 (rr.processRequest, mapper, serialisation, id);
c@116 257 break;
c@75 258 case RRType::Finish:
c@116 259 j = VampJson::fromRpcRequest_Finish(rr.finishRequest, mapper, id);
c@116 260 break;
c@75 261 case RRType::NotValid:
c@116 262 break;
c@75 263 }
c@75 264
c@75 265 cout << j.dump() << endl;
c@75 266 }
c@75 267
c@75 268 RequestOrResponse
c@75 269 readResponseJson(string &err)
c@75 270 {
c@75 271 RequestOrResponse rr;
c@75 272 rr.direction = RequestOrResponse::Response;
c@75 273
c@75 274 string input;
c@75 275 if (!getline(cin, input)) {
c@116 276 // the EOF case, not actually an error
c@116 277 rr.type = RRType::NotValid;
c@116 278 return rr;
c@75 279 }
c@75 280
c@75 281 Json j = convertResponseJson(input, err);
c@75 282 if (err != "") return {};
c@75 283
c@75 284 rr.type = VampJson::getRequestResponseType(j, err);
c@75 285 if (err != "") return {};
c@75 286
c@75 287 rr.id = readJsonId(j);
c@75 288
c@75 289 VampJson::BufferSerialisation serialisation =
c@75 290 VampJson::BufferSerialisation::Array;
c@75 291
c@75 292 rr.success = j["success"].bool_value();
c@75 293 rr.errorText = j["errorText"].string_value();
c@75 294
c@75 295 switch (rr.type) {
c@75 296
c@75 297 case RRType::List:
c@116 298 rr.listResponse = VampJson::toRpcResponse_List(j, err);
c@116 299 break;
c@75 300 case RRType::Load:
c@116 301 rr.loadResponse = VampJson::toRpcResponse_Load(j, mapper, err);
c@116 302 break;
c@75 303 case RRType::Configure:
c@116 304 rr.configurationResponse = VampJson::toRpcResponse_Configure(j, mapper, err);
c@116 305 break;
c@75 306 case RRType::Process:
c@116 307 rr.processResponse = VampJson::toRpcResponse_Process(j, mapper, serialisation, err);
c@116 308 break;
c@75 309 case RRType::Finish:
c@116 310 rr.finishResponse = VampJson::toRpcResponse_Finish(j, mapper, serialisation, err);
c@116 311 break;
c@75 312 case RRType::NotValid:
c@116 313 break;
c@75 314 }
c@75 315
c@75 316 return rr;
c@75 317 }
c@75 318
c@75 319 void
c@75 320 writeResponseJson(RequestOrResponse &rr, bool useBase64)
c@75 321 {
c@75 322 Json j;
c@75 323
c@75 324 VampJson::BufferSerialisation serialisation =
c@75 325 (useBase64 ?
c@75 326 VampJson::BufferSerialisation::Base64 :
c@75 327 VampJson::BufferSerialisation::Array);
c@75 328
c@75 329 Json id = writeJsonId(rr.id);
c@75 330
c@75 331 if (!rr.success) {
c@75 332
c@116 333 j = VampJson::fromError(rr.errorText, rr.type, id);
c@75 334
c@75 335 } else {
c@75 336
c@116 337 switch (rr.type) {
c@75 338
c@116 339 case RRType::List:
c@116 340 j = VampJson::fromRpcResponse_List(rr.listResponse, id);
c@116 341 break;
c@116 342 case RRType::Load:
c@116 343 j = VampJson::fromRpcResponse_Load(rr.loadResponse, mapper, id);
c@116 344 break;
c@116 345 case RRType::Configure:
c@116 346 j = VampJson::fromRpcResponse_Configure(rr.configurationResponse,
c@75 347 mapper, id);
c@116 348 break;
c@116 349 case RRType::Process:
c@116 350 j = VampJson::fromRpcResponse_Process
c@116 351 (rr.processResponse, mapper, serialisation, id);
c@116 352 break;
c@116 353 case RRType::Finish:
c@116 354 j = VampJson::fromRpcResponse_Finish
c@116 355 (rr.finishResponse, mapper, serialisation, id);
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 cout << j.dump() << endl;
c@75 363 }
c@75 364
c@75 365 RequestOrResponse
c@75 366 readRequestCapnp(kj::BufferedInputStreamWrapper &buffered)
c@75 367 {
c@75 368 RequestOrResponse rr;
c@75 369 rr.direction = RequestOrResponse::Request;
c@75 370
c@97 371 capnp::InputStreamMessageReader message(buffered);
c@97 372 piper::RpcRequest::Reader reader = message.getRoot<piper::RpcRequest>();
c@75 373
c@75 374 rr.type = VampnProto::getRequestResponseType(reader);
c@75 375 rr.id = readCapnpId(reader);
c@75 376
c@75 377 switch (rr.type) {
c@75 378
c@75 379 case RRType::List:
c@116 380 VampnProto::readRpcRequest_List(reader); // type check only
c@116 381 break;
c@75 382 case RRType::Load:
c@116 383 VampnProto::readRpcRequest_Load(rr.loadRequest, reader);
c@116 384 break;
c@75 385 case RRType::Configure:
c@116 386 VampnProto::readRpcRequest_Configure(rr.configurationRequest,
c@116 387 reader, mapper);
c@116 388 break;
c@75 389 case RRType::Process:
c@116 390 VampnProto::readRpcRequest_Process(rr.processRequest, reader, mapper);
c@116 391 break;
c@75 392 case RRType::Finish:
c@116 393 VampnProto::readRpcRequest_Finish(rr.finishRequest, reader, mapper);
c@116 394 break;
c@75 395 case RRType::NotValid:
c@116 396 break;
c@75 397 }
c@75 398
c@75 399 return rr;
c@75 400 }
c@75 401
c@75 402 void
c@75 403 writeRequestCapnp(RequestOrResponse &rr)
c@75 404 {
c@97 405 capnp::MallocMessageBuilder message;
c@97 406 piper::RpcRequest::Builder builder = message.initRoot<piper::RpcRequest>();
c@75 407
c@75 408 buildCapnpId(builder, rr.id);
c@75 409
c@75 410 switch (rr.type) {
c@75 411
c@75 412 case RRType::List:
c@116 413 VampnProto::buildRpcRequest_List(builder);
c@116 414 break;
c@75 415 case RRType::Load:
c@116 416 VampnProto::buildRpcRequest_Load(builder, rr.loadRequest);
c@116 417 break;
c@75 418 case RRType::Configure:
c@116 419 VampnProto::buildRpcRequest_Configure(builder,
c@75 420 rr.configurationRequest, mapper);
c@116 421 break;
c@75 422 case RRType::Process:
c@116 423 VampnProto::buildRpcRequest_Process(builder, rr.processRequest, mapper);
c@116 424 break;
c@75 425 case RRType::Finish:
c@116 426 VampnProto::buildRpcRequest_Finish(builder, rr.finishRequest, mapper);
c@116 427 break;
c@75 428 case RRType::NotValid:
c@116 429 break;
c@75 430 }
c@75 431
c@75 432 writeMessageToFd(1, message);
c@75 433 }
c@75 434
c@75 435 RequestOrResponse
c@75 436 readResponseCapnp(kj::BufferedInputStreamWrapper &buffered)
c@75 437 {
c@75 438 RequestOrResponse rr;
c@75 439 rr.direction = RequestOrResponse::Response;
c@75 440
c@97 441 capnp::InputStreamMessageReader message(buffered);
c@97 442 piper::RpcResponse::Reader reader = message.getRoot<piper::RpcResponse>();
c@75 443
c@75 444 rr.type = VampnProto::getRequestResponseType(reader);
c@75 445 rr.success = true;
c@75 446 rr.errorText = "";
c@75 447 rr.id = readCapnpId(reader);
c@75 448 int errorCode = 0;
c@75 449
c@75 450 switch (rr.type) {
c@75 451
c@75 452 case RRType::List:
c@116 453 VampnProto::readRpcResponse_List(rr.listResponse, reader);
c@116 454 break;
c@75 455 case RRType::Load:
c@116 456 VampnProto::readRpcResponse_Load(rr.loadResponse, reader, mapper);
c@116 457 break;
c@75 458 case RRType::Configure:
c@116 459 VampnProto::readRpcResponse_Configure(rr.configurationResponse,
c@116 460 reader, mapper);
c@116 461 break;
c@75 462 case RRType::Process:
c@116 463 VampnProto::readRpcResponse_Process(rr.processResponse, reader, mapper);
c@116 464 break;
c@75 465 case RRType::Finish:
c@116 466 VampnProto::readRpcResponse_Finish(rr.finishResponse, reader, mapper);
c@116 467 break;
c@75 468 case RRType::NotValid:
c@75 469 // error
c@75 470 rr.success = false;
c@75 471 VampnProto::readRpcResponse_Error(errorCode, rr.errorText, reader);
c@116 472 break;
c@75 473 }
c@75 474
c@75 475 return rr;
c@75 476 }
c@75 477
c@75 478 void
c@75 479 writeResponseCapnp(RequestOrResponse &rr)
c@75 480 {
c@97 481 capnp::MallocMessageBuilder message;
c@97 482 piper::RpcResponse::Builder builder = message.initRoot<piper::RpcResponse>();
c@75 483
c@75 484 buildCapnpId(builder, rr.id);
c@75 485
c@75 486 if (!rr.success) {
c@75 487
c@116 488 VampnProto::buildRpcResponse_Error(builder, rr.errorText, rr.type);
c@75 489
c@75 490 } else {
c@116 491
c@116 492 switch (rr.type) {
c@75 493
c@116 494 case RRType::List:
c@116 495 VampnProto::buildRpcResponse_List(builder, rr.listResponse);
c@116 496 break;
c@116 497 case RRType::Load:
c@116 498 VampnProto::buildRpcResponse_Load(builder, rr.loadResponse, mapper);
c@116 499 break;
c@116 500 case RRType::Configure:
c@116 501 VampnProto::buildRpcResponse_Configure(builder, rr.configurationResponse, mapper);
c@116 502 break;
c@116 503 case RRType::Process:
c@116 504 VampnProto::buildRpcResponse_Process(builder, rr.processResponse, mapper);
c@116 505 break;
c@116 506 case RRType::Finish:
c@116 507 VampnProto::buildRpcResponse_Finish(builder, rr.finishResponse, mapper);
c@116 508 break;
c@116 509 case RRType::NotValid:
c@116 510 break;
c@116 511 }
c@75 512 }
c@75 513
c@75 514 writeMessageToFd(1, message);
c@75 515 }
c@75 516
c@75 517 RequestOrResponse
c@75 518 readInputJson(RequestOrResponse::Direction direction, string &err)
c@75 519 {
c@75 520 if (direction == RequestOrResponse::Request) {
c@116 521 return readRequestJson(err);
c@75 522 } else {
c@116 523 return readResponseJson(err);
c@75 524 }
c@75 525 }
c@75 526
c@75 527 RequestOrResponse
c@75 528 readInputCapnp(RequestOrResponse::Direction direction)
c@75 529 {
c@75 530 static kj::FdInputStream stream(0); // stdin
c@75 531 static kj::BufferedInputStreamWrapper buffered(stream);
c@75 532
c@75 533 if (buffered.tryGetReadBuffer() == nullptr) {
c@116 534 return {};
c@75 535 }
c@75 536
c@75 537 if (direction == RequestOrResponse::Request) {
c@116 538 return readRequestCapnp(buffered);
c@75 539 } else {
c@116 540 return readResponseCapnp(buffered);
c@75 541 }
c@75 542 }
c@75 543
c@75 544 RequestOrResponse
c@75 545 readInput(string format, RequestOrResponse::Direction direction)
c@75 546 {
c@75 547 if (format == "json") {
c@116 548 string err;
c@116 549 auto result = readInputJson(direction, err);
c@116 550 if (err != "") throw runtime_error(err);
c@116 551 else return result;
c@75 552 } else if (format == "capnp") {
c@116 553 return readInputCapnp(direction);
c@75 554 } else {
c@116 555 throw runtime_error("unknown input format \"" + format + "\"");
c@75 556 }
c@75 557 }
c@75 558
c@75 559 void
c@75 560 writeOutput(string format, RequestOrResponse &rr)
c@75 561 {
c@75 562 if (format == "json") {
c@116 563 if (rr.direction == RequestOrResponse::Request) {
c@116 564 writeRequestJson(rr, false);
c@116 565 } else {
c@116 566 writeResponseJson(rr, false);
c@116 567 }
c@75 568 } else if (format == "json-b64") {
c@116 569 if (rr.direction == RequestOrResponse::Request) {
c@116 570 writeRequestJson(rr, true);
c@116 571 } else {
c@116 572 writeResponseJson(rr, true);
c@116 573 }
c@75 574 } else if (format == "capnp") {
c@116 575 if (rr.direction == RequestOrResponse::Request) {
c@116 576 writeRequestCapnp(rr);
c@116 577 } else {
c@116 578 writeResponseCapnp(rr);
c@116 579 }
c@75 580 } else {
c@116 581 throw runtime_error("unknown output format \"" + format + "\"");
c@75 582 }
c@75 583 }
c@75 584
c@75 585 int main(int argc, char **argv)
c@75 586 {
c@75 587 if (argc < 2) {
c@116 588 usage();
c@75 589 }
c@75 590
c@75 591 string informat = "json", outformat = "json";
c@75 592 RequestOrResponse::Direction direction = RequestOrResponse::Request;
c@75 593 bool haveDirection = false;
c@75 594
c@75 595 for (int i = 1; i < argc; ++i) {
c@75 596
c@116 597 string arg = argv[i];
c@116 598 bool final = (i + 1 == argc);
c@116 599
c@116 600 if (arg == "-i") {
c@116 601 if (final) usage();
c@116 602 else informat = argv[++i];
c@75 603
c@116 604 } else if (arg == "-o") {
c@116 605 if (final) usage();
c@116 606 else outformat = argv[++i];
c@75 607
c@116 608 } else if (arg == "request") {
c@116 609 direction = RequestOrResponse::Request;
c@116 610 haveDirection = true;
c@75 611
c@116 612 } else if (arg == "response") {
c@116 613 direction = RequestOrResponse::Response;
c@116 614 haveDirection = true;
c@116 615
c@116 616 } else {
c@116 617 usage();
c@116 618 }
c@75 619 }
c@75 620
c@75 621 if (informat == "" || outformat == "" || !haveDirection) {
c@116 622 usage();
c@75 623 }
c@75 624
c@75 625 while (true) {
c@75 626
c@116 627 try {
c@75 628
c@116 629 RequestOrResponse rr = readInput(informat, direction);
c@75 630
c@116 631 // NotValid without an exception indicates EOF:
c@116 632 if (rr.type == RRType::NotValid) break;
c@75 633
c@116 634 writeOutput(outformat, rr);
c@116 635
c@116 636 } catch (std::exception &e) {
c@75 637
c@116 638 cerr << "Error: " << e.what() << endl;
c@116 639 exit(1);
c@116 640 }
c@75 641 }
c@75 642
c@75 643 exit(0);
c@75 644 }
c@75 645
c@75 646