door.cpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203
  1. #include "door.h"
  2. #include <algorithm>
  3. #include <chrono>
  4. #include <ctype.h>
  5. #include <string.h>
  6. #include <string>
  7. #include <thread>
  8. #include <libgen.h> // basename
  9. // time/date output std::put_time()
  10. // https://en.cppreference.com/w/cpp/io/manip/put_time
  11. #include <ctime>
  12. #include <iomanip>
  13. // alarm signal
  14. #include <signal.h>
  15. #include <unistd.h>
  16. #include <iconv.h>
  17. #include <algorithm>
  18. #include <iostream>
  19. /*
  20. My strategy here has failed.
  21. If I set enigma to cp437, then it handles everything but the cp437
  22. symbols (diamonds/hearts/spades/clubs) correctly on the unicode side.
  23. [And my door thinks it's all cp437 always]
  24. If I set enigma to utf8, then it works right on the ssh terminal side.
  25. But cp437 turns to puke because it's trying to convert cp437 from
  26. utf8 to cp437. The symbols get '?'.
  27. I can't detect unicode (when set to utf8), but I can detect cp437
  28. (by sending the diamonds/hearts characters).
  29. But I can't get through the enigma translation system. If only iconv worked
  30. correctly with hearts/clubs symbols! Then I wouldn't need this broken
  31. work-around code.
  32. */
  33. namespace door {
  34. void to_lower(std::string &text) {
  35. transform(text.begin(), text.end(), text.begin(), ::tolower);
  36. }
  37. bool replace(std::string &str, const std::string &from, const std::string &to) {
  38. size_t start_pos = str.find(from);
  39. if (start_pos == std::string::npos)
  40. return false;
  41. str.replace(start_pos, from.length(), to);
  42. return true;
  43. }
  44. static bool hangup = false;
  45. void sig_handler(int signal) {
  46. hangup = true;
  47. /*
  48. ofstream sigf;
  49. sigf.open("signal.log", std::ofstream::out | std::ofstream::app);
  50. sigf << "SNAP! GOT: " << signal << std::endl;
  51. sigf.close();
  52. */
  53. // 13 SIGPIPE -- ok, what do I do with this, eh?
  54. }
  55. class IConv {
  56. iconv_t ic;
  57. public:
  58. IConv(const char *to, const char *from);
  59. ~IConv();
  60. int convert(char *input, char *output, size_t outbufsize);
  61. };
  62. IConv::IConv(const char *to, const char *from) : ic(iconv_open(to, from)) {}
  63. IConv::~IConv() { iconv_close(ic); }
  64. int IConv::convert(char *input, char *output, size_t outbufsize) {
  65. size_t inbufsize = strlen(input);
  66. // size_t orig_size = outbufsize;
  67. // memset(output, 0, outbufsize);
  68. // https://www.gnu.org/savannah-checkouts/gnu/libiconv/documentation/libiconv-1.15/iconv.3.html
  69. int r = iconv(ic, &input, &inbufsize, &output, &outbufsize);
  70. *output = 0;
  71. return r;
  72. }
  73. static IConv converter("UTF-8", "CP437");
  74. void cp437toUnicode(std::string input, std::string &out) {
  75. char buffer[10240];
  76. char output[16384];
  77. strcpy(buffer, input.c_str());
  78. converter.convert(buffer, output, sizeof(output));
  79. out.assign(output);
  80. }
  81. void cp437toUnicode(const char *input, std::string &out) {
  82. char buffer[10240];
  83. char output[16384];
  84. strcpy(buffer, input);
  85. converter.convert(buffer, output, sizeof(output));
  86. out.assign(output);
  87. }
  88. bool unicode = false;
  89. bool full_cp437 = false;
  90. bool debug_capture = false;
  91. /**
  92. * Construct a new Door object using the commandline parameters
  93. * given to the main function.
  94. *
  95. * @example door_example.cpp
  96. */
  97. Door::Door(std::string dname, int argc, char *argv[])
  98. : std::ostream(this), doorname{dname},
  99. has_dropfile{false}, debugging{false}, seconds_elapsed{0},
  100. previous(COLOR::WHITE), track{true}, cx{1}, cy{1},
  101. inactivity{120}, node{1} {
  102. // Setup commandline options
  103. opt.addUsage("Door++ library by BUGZ (C) 2021");
  104. opt.addUsage("");
  105. opt.addUsage(" -h --help Displays this help");
  106. opt.addUsage(" -l --local Local Mode");
  107. opt.addUsage(" -d --dropfile [FILENAME] Load Dropfile");
  108. opt.addUsage(" -n --node N Set node number");
  109. // opt.addUsage(" -c --cp437 Force CP437");
  110. // opt.addUsage(" -b --bbsname NAME Set BBS Name");
  111. opt.addUsage(" -u --username NAME Set Username");
  112. opt.addUsage(" -t --timeleft N Set time left");
  113. opt.addUsage(" --maxtime N Set max time");
  114. opt.addUsage("");
  115. opt.setFlag("help", 'h');
  116. opt.setFlag("local", 'l');
  117. opt.setFlag("cp437", 'c');
  118. opt.setFlag("unicode");
  119. opt.setFlag("debuggering");
  120. opt.setOption("dropfile", 'd');
  121. // opt.setOption("bbsname", 'b');
  122. opt.setOption("username", 'u');
  123. opt.setOption("timeleft", 't');
  124. opt.setOption("maxtime");
  125. opt.processCommandArgs(argc, argv);
  126. if (!opt.hasOptions()) {
  127. opt.printUsage();
  128. exit(1);
  129. }
  130. if (opt.getFlag("help") || opt.getFlag('h')) {
  131. opt.printUsage();
  132. exit(1);
  133. }
  134. if (opt.getValue("username") != nullptr) {
  135. username = opt.getValue("username");
  136. }
  137. if (opt.getFlag("debuggering")) {
  138. debugging = true;
  139. }
  140. if (opt.getValue("node") != nullptr) {
  141. node = atoi(opt.getValue("node"));
  142. }
  143. if (opt.getValue("timeleft") != nullptr) {
  144. time_left = atoi(opt.getValue("timeleft"));
  145. } else {
  146. // sensible default
  147. time_left = 25;
  148. }
  149. /*
  150. if (opt.getValue("bbsname") != nullptr) {
  151. bbsname = opt.getValue("bbsname");
  152. }
  153. */
  154. std::string logFileName = dname + ".log";
  155. logf.open(logFileName.c_str(), std::ofstream::out | std::ofstream::app);
  156. parse_dropfile(opt.getValue("dropfile"));
  157. /*
  158. If the dropfile has time_left, we'll use it.
  159. Adjust time_left by maxtime value (if given).
  160. */
  161. if (opt.getValue("maxtime") != nullptr) {
  162. int maxtime = atoi(opt.getValue("maxtime"));
  163. if (time_left > maxtime) {
  164. logf << "Adjusting time from " << time_left << " to " << maxtime
  165. << std::endl;
  166. time_left = maxtime;
  167. }
  168. }
  169. if (opt.getFlag("local")) {
  170. if (username.empty()) {
  171. std::cout << "Local mode requires username to be set." << std::endl;
  172. opt.printUsage();
  173. exit(1);
  174. }
  175. } else {
  176. // we must have a dropfile, or else!
  177. if (!has_dropfile) {
  178. std::cout << "I require a dropfile. And a shrubbery." << std::endl;
  179. opt.printUsage();
  180. exit(1);
  181. }
  182. }
  183. // Set program name
  184. log() << "Door init" << std::endl;
  185. init();
  186. // door.sys doesn't give BBS name. system_name
  187. if (!debugging) {
  188. detect_unicode_and_screen();
  189. logf << "Screen " << width << " X " << height << " unicode " << unicode
  190. << " full_cp437 " << full_cp437 << std::endl;
  191. }
  192. if (opt.getFlag("cp437")) {
  193. unicode = false;
  194. }
  195. if (opt.getFlag("unicode")) {
  196. unicode = true;
  197. }
  198. }
  199. Door::~Door() {
  200. // restore default mode
  201. // tcsetattr(STDIN_FILENO, TCSANOW, &tio_default);
  202. log() << "dtor" << std::endl;
  203. tcsetattr(STDIN_FILENO, TCOFLUSH, &tio_default);
  204. signal(SIGHUP, SIG_DFL);
  205. signal(SIGPIPE, SIG_DFL);
  206. // time thread
  207. stop_thread.set_value();
  208. time_thread.join();
  209. log() << "done" << std::endl;
  210. logf.close();
  211. }
  212. // https://www.tutorialspoint.com/how-do-i-terminate-a-thread-in-cplusplus11
  213. void Door::time_thread_run(std::future<void> future) {
  214. while (future.wait_for(std::chrono::milliseconds(1)) ==
  215. std::future_status::timeout) {
  216. // std::cout << "Executing the thread....." << std::endl;
  217. std::this_thread::sleep_for(std::chrono::seconds(1));
  218. seconds_elapsed++;
  219. // log("TICK");
  220. // logf << "TICK " << seconds_elapsed << std::endl;
  221. if (seconds_elapsed % 60 == 0) {
  222. if (time_left > 0)
  223. time_left--;
  224. }
  225. }
  226. }
  227. void Door::init(void) {
  228. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  229. // https://viewsourcecode.org/snaptoken/kilo/03.rawInputAndOutput.html
  230. // ok! I didn't know about VTIME ! That's something different!
  231. // enable terminal RAW mode
  232. struct termios tio_raw;
  233. tcgetattr(STDIN_FILENO, &tio_default);
  234. tio_raw = tio_default;
  235. cfmakeraw(&tio_raw);
  236. // local terminal magic
  237. tio_raw.c_cc[VMIN] = 0;
  238. tio_raw.c_cc[VTIME] = 1;
  239. bpos = 0;
  240. tcsetattr(STDIN_FILENO, TCSANOW, &tio_raw);
  241. startup = std::time(nullptr);
  242. signal(SIGHUP, sig_handler);
  243. signal(SIGPIPE, sig_handler);
  244. // time thread
  245. std::future<void> stop_future = stop_thread.get_future();
  246. time_thread =
  247. std::thread(&Door::time_thread_run, this, std::move(stop_future));
  248. }
  249. void Door::detect_unicode_and_screen(void) {
  250. unicode = false;
  251. full_cp437 = false;
  252. width = 0;
  253. height = 0;
  254. if (!isatty(STDIN_FILENO)) {
  255. // https://stackoverflow.com/questions/273261/force-telnet-client-into-character-mode
  256. *this << "\377\375\042\377\373\001"; // fix telnet client
  257. }
  258. // maybe I need to be trying to detect cp437 instead of trying to detect
  259. // unicde!
  260. *this << "\x1b[0;30;40m\x1b[2J\x1b[H"; // black on black, clrscr, go home
  261. // *this << "\u2615"
  262. *this << "\x03\x04" // hearts and diamonds
  263. << "\x1b[6n"; // cursor pos
  264. *this << door::nl << "\u2615"
  265. << "\x1b[6n"; // hot beverage + cursor pos
  266. *this << "\x1b[999C\x1b[999B\x1b[6n"; // goto end of screen + cursor pos
  267. *this << "\x1b[H"; // go home
  268. this->flush();
  269. usleep(1000 * 1000);
  270. if (haskey()) {
  271. char buffer[101];
  272. int len;
  273. len = read(STDIN_FILENO, &buffer, sizeof(buffer) - 1);
  274. // logf << "read " << len << std::endl;
  275. if (len > 0) {
  276. buffer[len] = 0;
  277. int x;
  278. /*
  279. for (x = 0; x < len; x++) {
  280. if (buffer[x] < 0x20)
  281. logf << std::hex << (int)buffer[x] << " ";
  282. else
  283. logf << buffer[x] << " ";
  284. }
  285. */
  286. for (x = 0; x < len; x++) {
  287. if (buffer[x] == 0)
  288. buffer[x] = ' ';
  289. }
  290. /*
  291. logf << std::endl;
  292. logf << "BUFFER [" << (char *)buffer << "]" << std::endl;
  293. */
  294. if (1) {
  295. std::string cleanbuffer = buffer;
  296. std::string esc = "\x1b";
  297. std::string esc_text = "^[";
  298. while (replace(cleanbuffer, esc, esc_text)) {
  299. };
  300. logf << "BUFFER [" << cleanbuffer << "]" << std::endl;
  301. }
  302. // this did not work -- because of the null characters in the buffer.
  303. // 1;3R required on David's machine. I'm not sure why.
  304. // 1;3R also happens under VSCodium.
  305. // 1;4R is what I get from syncterm.
  306. if ((strstr(buffer, "1;1R") != nullptr) and
  307. ((strstr(buffer, "2;2R") != nullptr) or
  308. (strstr(buffer, "2;3R") != nullptr))) {
  309. // if ((strstr(buffer, "1;2R") != nullptr) or
  310. // (strstr(buffer, "1;3R") != nullptr)) {
  311. unicode = true;
  312. log() << "unicode enabled \u2615" << std::endl; // "U0001f926");
  313. } else {
  314. if (strstr(buffer, "1;3R") != nullptr) {
  315. full_cp437 = true;
  316. }
  317. }
  318. // Get the terminal screen size
  319. // \x1b[1;2R\x1b[41;173R
  320. // log(buffer);
  321. char *cp;
  322. /*
  323. cp = strchr(buffer, '\x1b');
  324. if (cp != nullptr) {
  325. cp = strchr(cp + 1, '\x1b');
  326. } else {
  327. log() << "Failed terminal size detection. See buffer:" << std::endl;
  328. log() << buffer << std::endl;
  329. return;
  330. }
  331. */
  332. cp = strrchr(buffer, '\x1b');
  333. if (cp != nullptr) {
  334. cp++;
  335. if (*cp == '[') {
  336. cp++;
  337. height = atoi(cp);
  338. cp = strchr(cp, ';');
  339. if (cp != nullptr) {
  340. cp++;
  341. width = atoi(cp);
  342. } else {
  343. height = 0;
  344. }
  345. }
  346. }
  347. }
  348. } else {
  349. logf << "FAIL-WHALE, no response to terminal getposition." << std::endl;
  350. }
  351. }
  352. void Door::parse_dropfile(const char *filepath) {
  353. if (filepath == nullptr)
  354. return;
  355. // Ok, parse file here...
  356. std::ifstream file(filepath);
  357. std::string line;
  358. while (std::getline(file, line)) {
  359. // These are "DOS" files. Remove trailing \r.
  360. if (!line.empty() && line[line.size() - 1] == '\r')
  361. line.erase(line.size() - 1);
  362. dropfilelines.push_back(line);
  363. }
  364. file.close();
  365. std::string filename;
  366. {
  367. // converting const char * to char * for basename.
  368. char *temp = strdup(filepath);
  369. filename = basename(temp);
  370. free(temp);
  371. }
  372. to_lower(filename);
  373. // for now, just door.sys.
  374. if (filename == "door.sys") {
  375. // Ok, parse away!
  376. node = atoi(dropfilelines[3].c_str());
  377. username = dropfilelines[9];
  378. location = dropfilelines[10];
  379. time_left = atoi(dropfilelines[18].c_str());
  380. sysop = dropfilelines[34];
  381. handle = dropfilelines[35];
  382. } else {
  383. if (filename == "door32.sys") {
  384. // https://raw.githubusercontent.com/NuSkooler/ansi-bbs/master/docs/dropfile_formats/door32_sys.txt
  385. // dropfilelines[0] = Comm type (0=local, 1=serial, 2=telnet)
  386. // dropfilelines[1] = Comm or Socket handle
  387. // dropfilelines[2] = BaudRate
  388. // dropfilelines[3] = BBS Software Version
  389. username = dropfilelines[4];
  390. handle = dropfilelines[5];
  391. time_left = atoi(dropfilelines[6].c_str());
  392. // dropfilelines[7] = Emulation (0=Ascii, 1=ANSI, .. or above = ANSI)
  393. node = atoi(dropfilelines[8].c_str());
  394. } else {
  395. std::string msg = "Unknown dropfile: ";
  396. msg += filename;
  397. log() << msg << std::endl;
  398. *this << msg << std::endl;
  399. exit(2);
  400. }
  401. }
  402. log() << "node:" << node << " username: " << username << " handle: " << handle
  403. << " time: " << time_left << std::endl;
  404. has_dropfile = true;
  405. }
  406. ofstream &Door::log(void) {
  407. // todo: have logging
  408. std::time_t t = std::time(nullptr);
  409. std::tm tm = *std::localtime(&t);
  410. logf << std::put_time(&tm, "%c ");
  411. return logf; // << output << std::endl;
  412. }
  413. bool Door::haskey(void) {
  414. fd_set socket_set;
  415. struct timeval tv;
  416. int select_ret = -1;
  417. if (hangup)
  418. return -2;
  419. if (time_left < 2)
  420. return -3;
  421. while (select_ret == -1) {
  422. FD_ZERO(&socket_set);
  423. FD_SET(STDIN_FILENO, &socket_set);
  424. tv.tv_sec = 0;
  425. tv.tv_usec = 1;
  426. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  427. if (select_ret == -1) {
  428. if (errno == EINTR)
  429. continue;
  430. log() << "hangup detected" << std::endl;
  431. hangup = true;
  432. return (-2);
  433. }
  434. if (select_ret == 0)
  435. return false;
  436. }
  437. return true;
  438. }
  439. /*
  440. low-lever read a key from terminal or stdin.
  441. Returns key, or
  442. -1 (no key available)
  443. -2 (read error)
  444. */
  445. signed int Door::getch(void) {
  446. fd_set socket_set;
  447. struct timeval tv;
  448. int select_ret = -1;
  449. int recv_ret;
  450. char key;
  451. if (door::hangup)
  452. return -2;
  453. if (time_left < 2)
  454. return -3;
  455. while (select_ret == -1) {
  456. FD_ZERO(&socket_set);
  457. FD_SET(STDIN_FILENO, &socket_set);
  458. tv.tv_sec = 0;
  459. tv.tv_usec = 100;
  460. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  461. // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
  462. if (select_ret == -1) {
  463. if (errno == EINTR)
  464. continue;
  465. log() << "hangup detected" << std::endl;
  466. door::hangup = true;
  467. return (-2);
  468. }
  469. if (select_ret == 0)
  470. return (-1);
  471. }
  472. recv_ret = read(STDIN_FILENO, &key, 1);
  473. if (recv_ret != 1) {
  474. // possibly log this.
  475. log() << "hangup" << std::endl;
  476. hangup = true;
  477. return -2;
  478. }
  479. return key;
  480. }
  481. void Door::unget(char c) {
  482. if (bpos < sizeof(buffer) - 1) {
  483. buffer[bpos] = c;
  484. bpos++;
  485. }
  486. }
  487. char Door::get(void) {
  488. if (bpos == 0)
  489. return 0;
  490. bpos--;
  491. return buffer[bpos];
  492. }
  493. signed int Door::getkey(void) {
  494. signed int c, c2;
  495. if (bpos != 0) {
  496. c = get();
  497. } else {
  498. c = getch();
  499. }
  500. if (c < 0)
  501. return c;
  502. /*
  503. We get 0x0d 0x00 for [Enter] (Syncterm)
  504. This strips out the null.
  505. */
  506. if (c == 0x0d) {
  507. c2 = getch();
  508. if ((c2 != 0) and (c2 >= 0))
  509. unget(c2);
  510. return c;
  511. }
  512. if (c == 0x1b) {
  513. // possible extended key
  514. c2 = getch();
  515. if (c2 < 0) {
  516. // nope, just plain ESC key
  517. return c;
  518. }
  519. // consume extended key values int extended buffer
  520. char extended[16];
  521. unsigned int pos = 0;
  522. extended[pos] = (char)c2;
  523. extended[pos + 1] = 0;
  524. pos++;
  525. while ((pos < sizeof(extended) - 1) and ((c2 = getch()) >= 0)) {
  526. // special case here where I'm sending out cursor location requests
  527. // and the \x1b[X;YR strings are getting buffered.
  528. if (c2 == 0x1b) {
  529. unget(c2);
  530. break;
  531. }
  532. extended[pos] = (char)c2;
  533. extended[pos + 1] = 0;
  534. pos++;
  535. }
  536. // convert extended buffer to special key
  537. if (extended[0] == '[') {
  538. switch (extended[1]) {
  539. case 'A':
  540. return XKEY_UP_ARROW;
  541. case 'B':
  542. return XKEY_DOWN_ARROW;
  543. case 'C':
  544. return XKEY_RIGHT_ARROW;
  545. case 'D':
  546. return XKEY_LEFT_ARROW;
  547. case 'H':
  548. return XKEY_HOME;
  549. case 'F':
  550. return XKEY_END; // terminal
  551. case 'K':
  552. return XKEY_END;
  553. case 'U':
  554. return XKEY_PGUP;
  555. case 'V':
  556. return XKEY_PGDN;
  557. case '@':
  558. return XKEY_INSERT;
  559. }
  560. if (extended[pos - 1] == '~') {
  561. // \x1b[digits~
  562. int number = atoi(extended + 1);
  563. switch (number) {
  564. case 2:
  565. return XKEY_INSERT; // terminal
  566. case 3:
  567. return XKEY_DELETE; // terminal
  568. case 5:
  569. return XKEY_PGUP; // terminal
  570. case 6:
  571. return XKEY_PGDN; // terminal
  572. case 15:
  573. return XKEY_F5; // terminal
  574. case 17:
  575. return XKEY_F6; // terminal
  576. case 18:
  577. return XKEY_F7; // terminal
  578. case 19:
  579. return XKEY_F8; // terminal
  580. case 20:
  581. return XKEY_F9; // terminal
  582. case 21:
  583. return XKEY_F10; // terminal
  584. case 23:
  585. return XKEY_F11;
  586. case 24:
  587. return XKEY_F12; // terminal
  588. }
  589. }
  590. }
  591. if (extended[0] == 'O') {
  592. switch (extended[1]) {
  593. case 'P':
  594. return XKEY_F1;
  595. case 'Q':
  596. return XKEY_F2;
  597. case 'R':
  598. return XKEY_F3;
  599. case 'S':
  600. return XKEY_F4;
  601. case 't':
  602. return XKEY_F5; // syncterm
  603. }
  604. }
  605. // unknown -- This needs to be logged
  606. logf << "\r\nDEBUG:\r\nESC + ";
  607. for (unsigned int x = 0; x < pos; x++) {
  608. char z = extended[x];
  609. if (iscntrl(z)) {
  610. logf << (int)z << " ";
  611. } else {
  612. logf << "'" << (char)z << "'"
  613. << " ";
  614. };
  615. }
  616. logf << "\r\n";
  617. logf.flush();
  618. return XKEY_UNKNOWN;
  619. }
  620. return c;
  621. }
  622. int Door::get_input(void) {
  623. signed int c;
  624. c = getkey();
  625. if (c < 0)
  626. return 0;
  627. return c;
  628. /*
  629. tODInputEvent event;
  630. od_get_input(&event, OD_NO_TIMEOUT, GETIN_NORMAL);
  631. */
  632. }
  633. /*
  634. The following code will wait for 1.5 second:
  635. #include <sys/select.h>
  636. #include <sys/time.h>
  637. #include <unistd.h>`
  638. int main() {
  639. struct timeval t;
  640. t.tv_sec = 1;
  641. t.tv_usec = 500000;
  642. select(0, NULL, NULL, NULL, &t);
  643. }
  644. */
  645. signed int Door::sleep_key(int secs) {
  646. fd_set socket_set;
  647. struct timeval tv;
  648. int select_ret = -1;
  649. /*
  650. int recv_ret;
  651. char key;
  652. */
  653. if (hangup)
  654. return -2;
  655. if (time_left < 2)
  656. return -3;
  657. while (select_ret == -1) {
  658. FD_ZERO(&socket_set);
  659. FD_SET(STDIN_FILENO, &socket_set);
  660. tv.tv_sec = secs;
  661. tv.tv_usec = 0;
  662. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  663. // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
  664. if (select_ret == -1) {
  665. if (errno == EINTR)
  666. continue;
  667. hangup = true;
  668. log() << "hangup detected" << std::endl;
  669. return (-2);
  670. }
  671. if (select_ret == 0)
  672. return (-1);
  673. }
  674. return getkey();
  675. }
  676. std::string Door::input_string(int max) {
  677. std::string input;
  678. // draw the input area.
  679. *this << std::string(max, ' ');
  680. *this << std::string(max, '\x08');
  681. int c;
  682. while (true) {
  683. c = sleep_key(inactivity);
  684. if (c < 0) {
  685. input.clear();
  686. return input;
  687. }
  688. if (c > 0x1000)
  689. continue;
  690. if (isprint(c)) {
  691. if (int(input.length()) < max) {
  692. *this << char(c);
  693. input.append(1, c);
  694. } else {
  695. // bell
  696. *this << '\x07';
  697. }
  698. } else {
  699. switch (c) {
  700. case 0x08:
  701. case 0x7f:
  702. if (input.length() > 0) {
  703. *this << "\x08 \x08";
  704. // this->flush();
  705. input.erase(input.length() - 1);
  706. };
  707. break;
  708. case 0x0d:
  709. return input;
  710. }
  711. }
  712. }
  713. }
  714. /**
  715. * Take given buffer and output it.
  716. *
  717. * This originally stored output in the buffer. We now directly output
  718. * via the OpenDoor od_disp_emu.
  719. *
  720. * @param s const char *
  721. * @param n std::streamsize
  722. * @return std::streamsize
  723. */
  724. std::streamsize Door::xsputn(const char *s, std::streamsize n) {
  725. if (debug_capture) {
  726. debug_buffer.append(s, n);
  727. } else {
  728. static std::string buffer;
  729. buffer.append(s, n);
  730. // setp(&(*buffer.begin()), &(*buffer.end()));
  731. if (!hangup) {
  732. std::cout << buffer;
  733. std::cout.flush();
  734. }
  735. // od_disp_emu(buffer.c_str(), TRUE);
  736. // Tracking character position could be a problem / local terminal unicode.
  737. if (track)
  738. cx += n;
  739. buffer.clear();
  740. }
  741. return n;
  742. }
  743. /**
  744. * Stores a character into the buffer.
  745. * This does still use the buffer.
  746. * @todo Replace this also with a direct call to od_disp_emu.
  747. *
  748. * @param c char
  749. * @return int
  750. */
  751. int Door::overflow(char c) {
  752. /*
  753. char temp[2];
  754. temp[0] = c;
  755. temp[1] = 0;
  756. */
  757. // buffer.push_back(c);
  758. // od_disp_emu(temp, TRUE);
  759. if (debug_capture) {
  760. debug_buffer.append(1, c);
  761. } else {
  762. if (!hangup) {
  763. std::cout << c;
  764. std::cout.flush();
  765. }
  766. }
  767. if (track)
  768. cx++;
  769. // setp(&(*buffer.begin()), &(*buffer.end()));
  770. return c;
  771. }
  772. /**
  773. * Construct a new Color Output:: Color Output object
  774. * We default to BLACK/BLACK (not really a valid color),
  775. * pos=0 and len=0.
  776. */
  777. ColorOutput::ColorOutput() : c(COLOR::BLACK, COLOR::BLACK) {
  778. pos = 0;
  779. len = 0;
  780. }
  781. /**
  782. * Reset pos and len to 0.
  783. */
  784. void ColorOutput::reset(void) {
  785. pos = 0;
  786. len = 0;
  787. }
  788. /**
  789. * Construct a new Render:: Render object
  790. *
  791. * Render consists of constant text,
  792. * and vector of ColorOutput
  793. *
  794. * @param txt Text
  795. */
  796. Render::Render(const std::string txt) : text{txt} {}
  797. /**
  798. * Output the Render.
  799. *
  800. * This consists of going through the vector, getting
  801. * the fragment (from pos and len), and outputting the
  802. * color and fragment.
  803. *
  804. * @param os
  805. */
  806. void Render::output(std::ostream &os) {
  807. for (auto out : outputs) {
  808. std::string fragment = text.substr(out.pos, out.len);
  809. os << out.c << fragment;
  810. }
  811. }
  812. /**
  813. * Construct a new Clrscr:: Clrscr object
  814. *
  815. * This is used to clear the screen.
  816. */
  817. Clrscr::Clrscr() {}
  818. /**
  819. * Clear the screen using ANSI codes.
  820. *
  821. * Not all systems home the cursor after clearing the screen.
  822. * We automatically home the cursor as well.
  823. *
  824. * @param os std::ostream&
  825. * @param clr const Clrscr&
  826. * @return std::ostream&
  827. */
  828. std::ostream &operator<<(std::ostream &os, const Clrscr &clr) {
  829. Door *d = dynamic_cast<Door *>(&os);
  830. if (d != nullptr) {
  831. d->track = false;
  832. *d << "\x1b[2J"
  833. "\x1b[H";
  834. d->cx = 1;
  835. d->cy = 1;
  836. d->track = true;
  837. } else {
  838. os << "\x1b[2J"
  839. "\x1b[H";
  840. }
  841. return os;
  842. }
  843. Clrscr cls;
  844. /**
  845. * This is used to issue NL+CR
  846. *
  847. */
  848. NewLine::NewLine() {}
  849. /**
  850. * Output Newline + CarriageReturn
  851. * @param os std::ostream
  852. * @param nl const NewLine
  853. * @return std::ostream&
  854. */
  855. std::ostream &operator<<(std::ostream &os, const NewLine &nl) {
  856. Door *d = dynamic_cast<Door *>(&os);
  857. if (d != nullptr) {
  858. d->track = false;
  859. *d << "\r\n";
  860. d->cx = 1;
  861. d->cy++;
  862. d->track = true;
  863. } else {
  864. os << "\r\n";
  865. };
  866. return os;
  867. }
  868. NewLine nl;
  869. /**
  870. * Construct a new Goto:: Goto object
  871. *
  872. * @param xpos
  873. * @param ypos
  874. */
  875. Goto::Goto(int xpos, int ypos) {
  876. x = xpos;
  877. y = ypos;
  878. }
  879. void Goto::set(int xpos, int ypos) {
  880. x = xpos;
  881. y = ypos;
  882. }
  883. /**
  884. * Output the ANSI codes to position the cursor to the given y,x position.
  885. *
  886. * @todo Optimize the ANSI goto string output.
  887. * @todo Update the Door object so it know where the cursor
  888. * is positioned.
  889. *
  890. * @param os std::ostream
  891. * @param g const Goto
  892. * @return std::ostream&
  893. */
  894. std::ostream &operator<<(std::ostream &os, const Goto &g) {
  895. Door *d = dynamic_cast<Door *>(&os);
  896. if (d != nullptr) {
  897. d->track = false;
  898. *d << "\x1b[";
  899. if (g.y > 1)
  900. *d << std::to_string(g.y);
  901. if (g.x > 1) {
  902. os << ";";
  903. *d << std::to_string(g.x);
  904. }
  905. *d << "H";
  906. d->cx = g.x;
  907. d->cy = g.y;
  908. d->track = true;
  909. } else {
  910. os << "\x1b[" << std::to_string(g.y) << ";" << std::to_string(g.x) << "H";
  911. };
  912. return os;
  913. }
  914. const char SaveCursor[] = "\x1b[s";
  915. const char RestoreCursor[] = "\x1b[u";
  916. // EXAMPLES
  917. /// BlueYellow Render example function
  918. renderFunction rBlueYellow = [](const std::string &txt) -> Render {
  919. Render r(txt);
  920. ColorOutput co;
  921. bool uc = true;
  922. ANSIColor blue(COLOR::BLUE, ATTR::BOLD);
  923. ANSIColor cyan(COLOR::YELLOW, ATTR::BOLD);
  924. co.pos = 0;
  925. co.len = 0;
  926. co.c = blue;
  927. // d << blue;
  928. int tpos = 0;
  929. for (char const &c : txt) {
  930. if (uc) {
  931. if (!isupper(c)) {
  932. // possible color change
  933. if (co.len != 0) {
  934. r.outputs.push_back(co);
  935. co.reset();
  936. co.pos = tpos;
  937. }
  938. co.c = cyan;
  939. // d << cyan;
  940. uc = false;
  941. }
  942. } else {
  943. if (isupper(c)) {
  944. if (co.len != 0) {
  945. r.outputs.push_back(co);
  946. co.reset();
  947. co.pos = tpos;
  948. }
  949. co.c = blue;
  950. // d << blue;
  951. uc = true;
  952. }
  953. }
  954. co.len++;
  955. tpos++;
  956. // d << c;
  957. }
  958. if (co.len != 0) {
  959. r.outputs.push_back(co);
  960. }
  961. return r;
  962. };
  963. door::renderFunction renderStatusValue(door::ANSIColor status,
  964. door::ANSIColor value) {
  965. door::renderFunction rf = [status,
  966. value](const std::string &txt) -> door::Render {
  967. door::Render r(txt);
  968. door::ColorOutput co;
  969. co.pos = 0;
  970. co.len = 0;
  971. co.c = status;
  972. size_t pos = txt.find(':');
  973. if (pos == std::string::npos) {
  974. // failed to find - use entire string as status color.
  975. co.len = txt.length();
  976. r.outputs.push_back(co);
  977. } else {
  978. pos++; // Have : in status color
  979. co.len = pos;
  980. r.outputs.push_back(co);
  981. co.reset();
  982. co.pos = pos;
  983. co.c = value;
  984. co.len = txt.length() - pos;
  985. r.outputs.push_back(co);
  986. }
  987. return r;
  988. };
  989. return rf;
  990. }
  991. door::renderFunction rStatusValue = [](const std::string &txt) -> door::Render {
  992. door::Render r(txt);
  993. door::ColorOutput co;
  994. // default colors STATUS: value
  995. door::ANSIColor status(door::COLOR::BLUE, door::ATTR::BOLD);
  996. door::ANSIColor value(door::COLOR::YELLOW, door::ATTR::BOLD);
  997. co.pos = 0;
  998. co.len = 0;
  999. co.c = status;
  1000. size_t pos = txt.find(':');
  1001. if (pos == std::string::npos) {
  1002. // failed to find - use entire string as status color.
  1003. co.len = txt.length();
  1004. r.outputs.push_back(co);
  1005. } else {
  1006. pos++; // Have : in status color
  1007. co.len = pos;
  1008. r.outputs.push_back(co);
  1009. co.reset();
  1010. co.pos = pos;
  1011. co.c = value;
  1012. co.len = txt.length() - pos;
  1013. r.outputs.push_back(co);
  1014. }
  1015. return r;
  1016. };
  1017. /*
  1018. std::function<void(Door &d, std::string &txt)> BlueYellow2 =
  1019. [](Door &d, std::string &txt) -> void {
  1020. bool uc = true;
  1021. ANSIColor blue(COLOR::BLACK, COLOR::CYAN);
  1022. ANSIColor cyan(COLOR::YELLOW, COLOR::BLUE, ATTR::BOLD);
  1023. d << blue;
  1024. for (char const &c : txt) {
  1025. if (uc) {
  1026. if (c == ':') {
  1027. d << cyan;
  1028. uc = false;
  1029. }
  1030. }
  1031. d << c;
  1032. }
  1033. };
  1034. std::function<void(Door &d, std::string &txt)> Aweful =
  1035. [](Door &d, std::string &txt) -> void {
  1036. for (char const &c : txt) {
  1037. // Color clr((Colors)((c % 14) + 1), Colors::BLACK, 0);
  1038. // Use only BRIGHT/LIGHT colors.
  1039. ANSIColor clr((COLOR)(c % 8), ATTR::BOLD);
  1040. d << clr << c;
  1041. }
  1042. };
  1043. */
  1044. } // namespace door