mystic.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984
  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <unistd.h> // usleep()
  4. #include <fcntl.h>
  5. #include <pty.h>
  6. #include <termios.h>
  7. #include <sys/select.h>
  8. #include <sys/wait.h>
  9. #include <signal.h> // handle Ctrl-C/SIGINT
  10. #include <strings.h> // strcasecmp
  11. #include <time.h> // usleep(), nanonsleep() ?
  12. #include <ctype.h>
  13. #include <stdlib.h> // random()
  14. #include <regex.h>
  15. // LOGGING with file output
  16. #include "zf_log.h"
  17. #include <stdio.h>
  18. #include <stdlib.h>
  19. FILE *g_log_file;
  20. static void file_output_callback(const zf_log_message *msg, void *arg) {
  21. (void)arg;
  22. *msg->p = '\n';
  23. fwrite(msg->buf, msg->p - msg->buf + 1, 1, g_log_file);
  24. fflush(g_log_file);
  25. }
  26. static void file_output_close(void) { fclose(g_log_file); }
  27. static void file_output_open(const char *const log_path) {
  28. g_log_file = fopen(log_path, "a");
  29. if (!g_log_file) {
  30. ZF_LOGW("Failed to open log file %s", log_path);
  31. return;
  32. }
  33. atexit(file_output_close);
  34. zf_log_set_output_v(ZF_LOG_PUT_STD, 0, file_output_callback);
  35. }
  36. // END LOGGING
  37. /*
  38. What is the name of the actual, real Mystic executable
  39. that we'll be executing and mangling?
  40. */
  41. #define TARGET "./mySTIC"
  42. // Size of our input and output buffers.
  43. #define BSIZE 128
  44. /*
  45. // Don't need this, zf_log does date/time stamps on output.
  46. const char * it_is_now(void) {
  47. static char buffer[100];
  48. time_t timer;
  49. struct tm* tm_info;
  50. timer = time(NULL);
  51. tm_info = localtime(&timer);
  52. strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", tm_info);
  53. return buffer;
  54. }
  55. void slow_write(int fd, int speed, char * buffer, int len) {
  56. int x;
  57. for( x = 0; x < len; x++) {
  58. usleep(speed);
  59. write( fd, &buffer[x], 1);
  60. }
  61. }
  62. */
  63. /**
  64. * Display a repr of the given string.
  65. *
  66. * This converts most \n\r\v\f\t codes,
  67. * defaults to \xHH (hex value).
  68. */
  69. const char *repr(const char *data) {
  70. static char buffer[4096];
  71. char *cp;
  72. strcpy(buffer, data);
  73. cp = buffer;
  74. while (*cp != 0) {
  75. char c = *cp;
  76. if (isspace(c)) {
  77. if (c == ' ') {
  78. cp++;
  79. continue;
  80. };
  81. /* Ok, it's form-feed ('\f'), newline ('\n'), carriage return ('\r'),
  82. * horizontal tab ('\t'), and vertical tab ('\v') */
  83. memmove(cp + 1, cp, strlen(cp) + 1);
  84. *cp = '\\';
  85. cp++;
  86. switch (c) {
  87. case '\f':
  88. *cp = 'f';
  89. cp++;
  90. break;
  91. case '\n':
  92. *cp = 'n';
  93. cp++;
  94. break;
  95. case '\r':
  96. *cp = 'r';
  97. cp++;
  98. break;
  99. case '\t':
  100. *cp = 't';
  101. cp++;
  102. break;
  103. case '\v':
  104. *cp = 'v';
  105. cp++;
  106. break;
  107. default:
  108. *cp = '?';
  109. cp++;
  110. break;
  111. }
  112. continue;
  113. }
  114. if (isprint(c)) {
  115. cp++;
  116. continue;
  117. };
  118. // Ok, default to \xHH output.
  119. memmove(cp + 3, cp, strlen(cp) + 1);
  120. *cp = '\\';
  121. cp++;
  122. *cp = 'x';
  123. cp++;
  124. char buffer[3];
  125. sprintf(buffer, "%02x", (int)c & 0xff);
  126. *cp = buffer[0];
  127. cp++;
  128. *cp = buffer[1];
  129. cp++;
  130. continue;
  131. }
  132. return buffer;
  133. }
  134. struct render {
  135. int speed;
  136. int effect;
  137. } current_render;
  138. int render_overlimit = 0;
  139. void reset_render(void) {
  140. current_render.speed = 0;
  141. current_render.effect = 0;
  142. render_overlimit = 0;
  143. }
  144. #define TRIGGER "^"
  145. // Max limit we'll sleep before ignoring effects/speed.
  146. #define SLEEP_LIMIT 30
  147. int ms_sleep(unsigned int ms) {
  148. int result = 0;
  149. struct timespec ts = {ms / 1000, (ms % 1000) * 1000000L};
  150. do {
  151. struct timespec ts_sleep = ts;
  152. result = nanosleep(&ts_sleep, &ts);
  153. } while ((-1 == result));
  154. return result;
  155. }
  156. void render_sleep(void) {
  157. if (render_overlimit)
  158. return;
  159. if (current_render.speed) { // * 100 still too slow.
  160. ms_sleep(current_render.speed * 10);
  161. }
  162. }
  163. // This needs work.
  164. void write_color(int fd, int color) {
  165. char buffer[10];
  166. sprintf(buffer, "\x1b[%dm", color);
  167. write(fd, buffer, strlen(buffer));
  168. }
  169. /**
  170. * process_trigger( fd, *cp )
  171. *
  172. * This process a command trigger.
  173. * It has seen TRIGGER, and now it is
  174. * processing whatever comes after it.
  175. * It will perform the process, and
  176. * return the char * of whatever is next.
  177. */
  178. const char *process_trigger(int fd, const char *cp) {
  179. char ch;
  180. int i, x, y;
  181. ch = toupper(*cp);
  182. cp++;
  183. switch (ch) {
  184. case 'D':
  185. i = 0;
  186. if (isdigit(*cp)) {
  187. i = (*cp) - '0';
  188. cp++;
  189. };
  190. if (isdigit(*cp)) {
  191. i *= 10;
  192. i += (*cp) - '0';
  193. cp++;
  194. };
  195. if ((i > 0) && (i < 80)) {
  196. ZF_LOGI("DEL %02d", i);
  197. for (x = 0; x < i; x++) {
  198. write(fd, "\b \b", 3);
  199. }
  200. };
  201. break;
  202. case 'C':
  203. i = 0;
  204. if (isdigit(*cp)) {
  205. i = (*cp) - '0';
  206. cp++;
  207. };
  208. if (isdigit(*cp)) {
  209. i *= 10;
  210. i += (*cp) - '0';
  211. cp++;
  212. };
  213. write_color(fd, i);
  214. break;
  215. case 'G':
  216. x = 0;
  217. if (isdigit(*cp)) {
  218. x = (*cp) - '0';
  219. cp++;
  220. };
  221. if (isdigit(*cp)) {
  222. x *= 10;
  223. x += (*cp) - '0';
  224. cp++;
  225. };
  226. y = 0;
  227. if (isdigit(*cp)) {
  228. y = (*cp) - '0';
  229. cp++;
  230. };
  231. if (isdigit(*cp)) {
  232. y *= 10;
  233. y += (*cp) - '0';
  234. cp++;
  235. };
  236. break;
  237. case 'R':
  238. i = 0;
  239. if (isdigit(*cp)) {
  240. i = (*cp) - '0';
  241. cp++;
  242. };
  243. if ((i > 0) && (i < 10)) {
  244. ZF_LOGI("RENDER %d", i);
  245. current_render.effect = i;
  246. } else {
  247. current_render.effect = 0;
  248. }
  249. break;
  250. case 'S':
  251. i = 0;
  252. if (isdigit(*cp)) {
  253. i = (*cp) - '0';
  254. cp++;
  255. };
  256. if ((i > 0) && (i < 10)) {
  257. ZF_LOGI("SPEED %d", i);
  258. current_render.speed = i;
  259. } else {
  260. current_render.speed = 0;
  261. }
  262. break;
  263. case 'P':
  264. i = 0;
  265. if (isdigit(*cp)) {
  266. i = (*cp) - '0';
  267. cp++;
  268. };
  269. if ((i > 0) && (i < 10)) {
  270. ZF_LOGI("PAWS %d", i);
  271. // sleep(i);
  272. if (!render_overlimit) {
  273. sleep(i);
  274. };
  275. }
  276. break;
  277. }
  278. return cp;
  279. }
  280. /**
  281. * render_effect( fd, ch )
  282. *
  283. * Displays the given character with whatever
  284. * rendering effect is currently active.
  285. * (If any).
  286. */
  287. void render_effect(int fd, char ch) {
  288. int effect = current_render.effect;
  289. int l;
  290. char space = ' ';
  291. char bs = '\b';
  292. switch (effect) {
  293. case 1:
  294. // CHAR + SPC + BS
  295. render_sleep();
  296. write(fd, &ch, 1);
  297. render_sleep();
  298. write(fd, &space, 1);
  299. render_sleep();
  300. render_sleep();
  301. write(fd, &bs, 1);
  302. break;
  303. case 2:
  304. // CHAR + 8 spaces + 8 BS
  305. render_sleep();
  306. write(fd, &ch, 1);
  307. for (l = 0; l < 8; l++) {
  308. render_sleep();
  309. write(fd, &space, 1);
  310. }
  311. for (l = 0; l < 8; l++) {
  312. render_sleep();
  313. write(fd, &bs, 1);
  314. }
  315. break;
  316. case 0:
  317. default:
  318. // NORMAL
  319. render_sleep();
  320. write(fd, &ch, 1);
  321. break;
  322. }
  323. }
  324. /**
  325. * render( fd, string_out )
  326. *
  327. * Render an entire string.
  328. * Handles TRIGGER.
  329. * Renders with effects.
  330. */
  331. void render(int fd, const char *string_out) {
  332. const char *cp = string_out;
  333. const char *trigger = cp;
  334. time_t start = time(NULL);
  335. int elapsed;
  336. int over = 0;
  337. reset_render();
  338. ZF_LOGD("render(%d, %s)", fd, repr(string_out));
  339. // Check our time from time to time.
  340. // If we start running long, disable sleeps.
  341. while ((trigger = strstr(cp, TRIGGER)) != NULL) {
  342. // There is special things to handle in here.
  343. while (cp != trigger) {
  344. elapsed = time(NULL) - start;
  345. if (elapsed > SLEEP_LIMIT) {
  346. render_overlimit = 1;
  347. current_render.speed = 0;
  348. };
  349. // write(fd, cp, 1 );
  350. render_effect(fd, *cp);
  351. cp++;
  352. };
  353. // ZF_LOGI( "at trigger: (%s)", cp);
  354. cp += strlen(TRIGGER);
  355. // Ok, we're pointing at the trigger -- do something.
  356. cp = process_trigger(fd, cp);
  357. // ZF_LOGI( "after trigger: (%s)", cp);
  358. };
  359. // We still might be under a rendering effect.
  360. while (*cp != 0) {
  361. elapsed = time(NULL) - start;
  362. if (elapsed > SLEEP_LIMIT) {
  363. render_overlimit = 1;
  364. current_render.speed = 0;
  365. };
  366. // write(fd, cp, 1);
  367. render_effect(fd, *cp);
  368. cp++;
  369. }
  370. }
  371. // Beanzilla'a no repeats
  372. /**
  373. * have_seen( list, len, item )
  374. *
  375. * Returns 1 (true) if item is in the list.
  376. * Rotates the list [x] = [x+1], and
  377. * list[len-1] = item, return 0 (false)
  378. */
  379. int have_seen(int *list, int len, int item) {
  380. int x;
  381. for (x = 0; x < len; x++) {
  382. if (list[x] == item) {
  383. return 1;
  384. }
  385. };
  386. // Ok, it is something different
  387. for (x = 0; x < len - 1; x++) {
  388. list[x] = list[x + 1];
  389. };
  390. list[x] = item;
  391. return 0;
  392. }
  393. /**
  394. * init_have_seen( list, len )
  395. *
  396. * Initialize the have_seen list with -1.
  397. * (-1 isn't a valid index, so we start
  398. * out with all invalid.)
  399. */
  400. void init_have_seen(int *list, int len) {
  401. int x;
  402. for (x = 0; x < len; x++) {
  403. list[x] = -1;
  404. }
  405. }
  406. /*
  407. These are harry "timeout" events.
  408. These happen when we've been sitting around awhile.
  409. */
  410. #define MAX_HARRY_EVENT_DUPS 2
  411. int last_seen_harry_event[MAX_HARRY_EVENT_DUPS];
  412. void harry_event(int fd) {
  413. // Make something happen
  414. char buffer[100];
  415. int r;
  416. // This is no where near finished, BUT!
  417. const char *phrases[] = {"Hahaha", "Snicker, snicker", "Boo!", "MeOW",
  418. "I see U"};
  419. const char *cp;
  420. // Remember the last phrase used,
  421. // and don't repeat (the last two)!
  422. do {
  423. r = random() % 5;
  424. } while (have_seen(last_seen_harry_event, MAX_HARRY_EVENT_DUPS, r));
  425. ZF_LOGD("%d => %d %d", r, last_seen_harry_event[0], last_seen_harry_event[1]);
  426. cp = phrases[r];
  427. sprintf(buffer, "^S2%s^P2^D%02d", cp, (int)strlen(cp));
  428. ZF_LOGD("harry_event: render(%d, \"%s\")", fd, buffer);
  429. render(fd, buffer);
  430. }
  431. void init_harry() {
  432. init_have_seen(last_seen_harry_event, MAX_HARRY_EVENT_DUPS);
  433. ZF_LOGD("init => %d %d", last_seen_harry_event[0], last_seen_harry_event[1]);
  434. }
  435. /*
  436. NOTE: Logging is single file, don't use in production!
  437. It won't handle multiple writers.
  438. */
  439. char *username = NULL;
  440. char *fullname = NULL;
  441. /*
  442. Pascal String Copy. Copy from pascal string, to C String.
  443. First char is pascal string length. (Max 255).
  444. */
  445. void pcopy(char *pstring, char *str) {
  446. int len = (int)*pstring;
  447. strncpy(str, pstring + 1, len);
  448. str[len] = 0;
  449. }
  450. /*
  451. This only works for those few idiots that use the
  452. horribly broken SSH crap that Mystic uses.
  453. */
  454. int locate_user(const char *alias) {
  455. FILE *user;
  456. char buffer[0x600];
  457. char temp[100];
  458. user = fopen("data/users.dat", "rb");
  459. if (user == NULL)
  460. return 0;
  461. // Carry on!
  462. while (fread(buffer, 0x600, 1, user) == 1) {
  463. pcopy(buffer + 0x6d, temp);
  464. if (strcasecmp(temp, username) == 0) {
  465. pcopy(buffer + 0x8c, temp);
  466. fullname = strdup(temp);
  467. break;
  468. }
  469. /*
  470. printf("Alias: %s\n", temp);
  471. pcopy(buffer + 0x8c, temp );
  472. printf("Full Name: %s\n", temp );
  473. */
  474. }
  475. fclose(user);
  476. return 1;
  477. }
  478. // Buffers are BSIZE + 1, so a buffer that size can strcpy safely.
  479. regex_t ANSI;
  480. regex_t WORDS;
  481. regex_t WORD;
  482. int init_regex(void) {
  483. int ret;
  484. char ansi[] = "\x1b\[[0-9]+(;[0-9]+)*?[a-zA-Z]";
  485. char words[] = "[a-zA-Z]+( [a-zA-Z]+)+";
  486. char word[] = "[a-zA-Z]+";
  487. char errorbuf[100];
  488. if (ret = regcomp(&ANSI, ansi, REG_EXTENDED | REG_NEWLINE)) {
  489. regerror(ret, &ANSI, errorbuf, sizeof(errorbuf));
  490. ZF_LOGW("Regex %s failed to compile: %s", ansi, errorbuf);
  491. return 0;
  492. };
  493. if (ret = regcomp(&WORDS, words, REG_EXTENDED | REG_NEWLINE)) {
  494. regerror(ret, &WORDS, errorbuf, sizeof(errorbuf));
  495. ZF_LOGW("Regex %s failed to compile: %s", words, errorbuf);
  496. return 0;
  497. };
  498. if (ret = regcomp(&WORD, word, REG_EXTENDED | REG_NEWLINE)) {
  499. regerror(ret, &WORD, errorbuf, sizeof(errorbuf));
  500. ZF_LOGW("Regex %s failed to compile: %s", word, errorbuf);
  501. return 0;
  502. };
  503. return 1;
  504. }
  505. int regmatch(regex_t *preg, const char *string, size_t nmatch,
  506. regmatch_t pmatch[], int eflags) {
  507. // returns number of matches found. (Max nmatch)
  508. int matches = 0;
  509. int offset = 0;
  510. int ret;
  511. while (matches < nmatch) {
  512. ret = regexec(preg, string + offset, nmatch - matches, pmatch + matches,
  513. eflags);
  514. if (!ret) {
  515. int current = offset;
  516. offset += pmatch[matches].rm_eo;
  517. pmatch[matches].rm_so += current;
  518. pmatch[matches].rm_eo += current;
  519. matches++;
  520. } else if (ret == REG_NOMATCH) {
  521. break;
  522. } else {
  523. break;
  524. }
  525. }
  526. return matches;
  527. }
  528. #define MAX_MATCH 32
  529. regmatch_t rxmatch[MAX_MATCH];
  530. int rx_match(regex_t *regex, const char *buffer) {
  531. int ret;
  532. ret = regmatch(regex, buffer, MAX_MATCH, rxmatch, 0);
  533. if (0) {
  534. for (int i = 0; i < ret; i++) {
  535. ZF_LOGI("%d : (%d-%d)", i, rxmatch[i].rm_so, rxmatch[i].rm_eo);
  536. }
  537. }
  538. return ret;
  539. }
  540. /*
  541. Terminal processing section.
  542. Monitor lines, position, color.
  543. What screen size do I want to emulate?
  544. ANSI codes.
  545. Do I need this??
  546. */
  547. int random_activate(int w) {
  548. int r = random() % 100;
  549. if (r <= (w * 10)) {
  550. return 1;
  551. };
  552. return 0;
  553. }
  554. /*
  555. * The buffer that we've been given is much larger now.
  556. *
  557. */
  558. int mangle(int fd, char *buffer) {
  559. int x, i;
  560. int need_render = 0; // changing word case around doesn't need the render
  561. int mangled = 0;
  562. char *cp;
  563. // Ok, we want to look for words to:
  564. // MaNGlE , or transpose (both?!)
  565. // or possibly transpose words
  566. char work[(BSIZE * 4) + 1];
  567. ZF_LOGI("mangle(%s)", repr(buffer));
  568. strcpy(work, buffer);
  569. /*
  570. NOTE: We copy the buffer, so we can clear out ANSI codes, etc.
  571. Otherwise we might mess some ANSI up in the manglying
  572. process.
  573. */
  574. /*
  575. (random) Look for ANSI CLS and:
  576. display random spooky texts around, with delays ... then CLS.
  577. display ANSI graphic file, with delays ... then CLS
  578. */
  579. const char *ANSI_CLS = "\x1b[2J";
  580. cp = strstr(buffer, ANSI_CLS);
  581. if (cp != NULL) {
  582. ZF_LOGI("seen: ANSI_CLS");
  583. if (random_activate(9)) {
  584. char display[100] = "";
  585. ZF_LOGI("mangle(ANSI_CLS)");
  586. // sprintf( display, "^P2...");
  587. // This string actually screws up ANSI detection (takes too long)
  588. // strcpy(display, "^P2^S501234567890^P1abcdef^P2g^P3h^P4i^S0^P2");
  589. strcpy(display, "^P2^S301234^P15^S0^P2");
  590. // Move the buffer so there's room for the display string.
  591. memmove(cp + strlen(display), cp, strlen(cp) + 1);
  592. strncpy(cp, display, strlen(display));
  593. ZF_LOGI("mangle(ANSI_CLS): (%d) %s", (int)strlen(buffer), repr(buffer));
  594. need_render = 1;
  595. /*
  596. Copy the new buffer over, but hide our "render" code
  597. from the remaining mangler steps.
  598. */
  599. strcpy(work, buffer);
  600. i = cp - buffer;
  601. // find offset into "buffer"
  602. // apply to work.
  603. memset(work + i, ' ', strlen(display));
  604. };
  605. }
  606. /* work -- clear out ANSI so we don't mangle ANSI codes. */
  607. x = rx_match(&ANSI, work);
  608. char replace_with = ' ';
  609. if (x > 0) {
  610. ZF_LOGD("found %d ANSI", x);
  611. for (i = 0; i < x; i++) {
  612. memset(work + rxmatch[i].rm_so, replace_with,
  613. rxmatch[i].rm_eo - rxmatch[i].rm_so);
  614. };
  615. ZF_LOGD("Work Now : (%d) %s", (int)strlen(work), repr(work));
  616. }
  617. // ZF_LOGI("mangle: %s", repr(work));
  618. /*
  619. (random) Locate words (in work), and possibly flip them around.
  620. Transpose words. Transpose case. Transpose letters.
  621. */
  622. x = rx_match(&WORDS, work);
  623. ZF_LOGD("found %d WORDS", x);
  624. if (x > 0) {
  625. for (i = 0; i < x; i++) {
  626. // Do things here.
  627. if (i % 3 == 0) {
  628. for (int p = rxmatch[i].rm_so; p < rxmatch[i].rm_eo; p++) {
  629. buffer[p] = tolower(buffer[p]);
  630. mangled++;
  631. }
  632. } else {
  633. if (i % 3 == 1) {
  634. for (int p = rxmatch[i].rm_so; p < rxmatch[i].rm_eo; p++) {
  635. buffer[p] = toupper(buffer[p]);
  636. mangled++;
  637. }
  638. }
  639. }
  640. }
  641. }
  642. /*
  643. (random) Locate single words, and transpose them. Transpose case.
  644. Transpose letters.
  645. */
  646. /*
  647. (random) Display up to certain point. Delay.
  648. Print some characters slowly. Delay.
  649. */
  650. if (need_render) {
  651. ZF_LOGD("HH %d : (%d) %s", need_render, (int)strlen(buffer), repr(buffer));
  652. } else {
  653. if (mangled) {
  654. ZF_LOGD("Mangled %d : %s", mangled, repr(buffer));
  655. }
  656. }
  657. if (need_render) {
  658. render(fd, buffer);
  659. } else {
  660. write(fd, buffer, strlen(buffer));
  661. };
  662. return need_render && mangled;
  663. }
  664. int harry_happens(time_t *last_event, int wakeup) {
  665. time_t now = time(NULL);
  666. int elapsed = now - *last_event;
  667. if (elapsed > wakeup) {
  668. // Ok! It's been too long since we've done something.
  669. *last_event = now;
  670. return 1;
  671. }
  672. return 0;
  673. }
  674. int main(int argc, char *argv[]) {
  675. int master;
  676. pid_t pid;
  677. int node = -1;
  678. file_output_open("horrible_harry.log");
  679. init_harry();
  680. srandom(time(NULL));
  681. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  682. // -PUWISHPASSWORD
  683. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  684. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  685. // -PUP2LAT3
  686. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  687. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  688. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML0 -SL1 -ST0 -CUnknown
  689. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  690. // -PUP2LAT3
  691. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML1 -SL1 -ST2 -CUnknown -Ubugz
  692. // -PUP2LAT3
  693. // SSH: -ML1 -ST2
  694. // Telnet: -ML0 -ST0
  695. // Locate username (if given) in the command line
  696. // -U<username>
  697. for (int x = 0; x < argc; x++) {
  698. if (strncmp("-U", argv[x], 2) == 0) {
  699. username = argv[x] + 2;
  700. ZF_LOGI("Username: [%s]", username);
  701. };
  702. if (strncmp("-SL", argv[x], 3) == 0) {
  703. node = argv[x][3] - '0' + 1;
  704. ZF_LOGI("Node: %d", node);
  705. }
  706. }
  707. if (username != NULL) {
  708. locate_user(username);
  709. ZF_LOGD("Username: [%s] A.K.A. [%s]", username, fullname);
  710. }
  711. if (!init_regex())
  712. return 2;
  713. // With IGNBRK I don't think I need this anymore. (Nope!)
  714. // signal(SIGINT, SIG_IGN);
  715. pid = forkpty(&master, NULL, NULL, NULL);
  716. // impossible to fork
  717. if (pid < 0) {
  718. return 1;
  719. }
  720. // child
  721. else if (pid == 0) {
  722. char *args[20]; // max 20 args
  723. int x;
  724. args[0] = TARGET;
  725. for (x = 1; x < argc; x++) {
  726. args[x] = argv[x];
  727. };
  728. args[x] = NULL;
  729. // run Mystic, run!
  730. execvp(TARGET, args);
  731. }
  732. // parent
  733. else {
  734. // remove the echo
  735. // ICANON - raw mode. No more line buffering!
  736. struct termios tios, orig1;
  737. struct timeval timeout;
  738. time_t last_event = 0; // time(NULL);
  739. ZF_LOGD("starting");
  740. tcgetattr(master, &tios);
  741. tios.c_lflag &= ~(ECHO | ECHONL | ICANON);
  742. tcsetattr(master, TCSAFLUSH, &tios);
  743. tcgetattr(1, &orig1);
  744. tios = orig1;
  745. tios.c_iflag &= ~(ICRNL | IXON); // Disable software flow control
  746. tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
  747. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  748. // ISIG should be Ctrl-C and Ctrl-Z IGNBRK +
  749. tcsetattr(1, TCSAFLUSH, &tios);
  750. for (;;) {
  751. // define estruturas para o select, que serve para verificar qual
  752. // se tornou "pronto pra uso"
  753. fd_set read_fd;
  754. fd_set write_fd;
  755. fd_set except_fd;
  756. // inicializa as estruturas
  757. FD_ZERO(&read_fd);
  758. FD_ZERO(&write_fd);
  759. FD_ZERO(&except_fd);
  760. // atribui o descritor master, obtido pelo forkpty, ao read_fd
  761. FD_SET(master, &read_fd);
  762. // atribui o stdin ao read_fd
  763. FD_SET(STDIN_FILENO, &read_fd);
  764. // o descritor tem que ser unico para o programa, a documentacao
  765. // recomenda um calculo entre os descritores sendo usados + 1
  766. /*
  767. TODO: Figure out how this would work.
  768. I'm thinking something like timeouts 30-50 seconds?
  769. And as we get closer, 15-25 seconds.
  770. */
  771. // we're in luck! The last parameter is time interval/timeout. :D
  772. timeout.tv_sec = 10;
  773. timeout.tv_usec = 0;
  774. // select(master+1, &read_fd, &write_fd, &except_fd, NULL);
  775. if (select(master + 1, &read_fd, &write_fd, &except_fd, &timeout) == 0) {
  776. // This means timeout!
  777. ZF_LOGI("TIMEOUT");
  778. harry_event(STDOUT_FILENO);
  779. }
  780. char input[BSIZE + 1];
  781. static char output[(BSIZE * 4) + 1];
  782. int total;
  783. // read_fd esta atribuido com read_fd?
  784. if (FD_ISSET(master, &read_fd)) {
  785. // leia o que bc esta mandando
  786. if ((total = read(master, &output, BSIZE)) != -1) {
  787. // e escreva isso na saida padrao
  788. output[total] = 0;
  789. // if ( harry_happens( &last_event, 5)) {
  790. if (1) {
  791. ZF_LOGI("harry_happens");
  792. if (mangle(STDOUT_FILENO, output) == 0) {
  793. // failed, so. Try again.
  794. last_event = 0;
  795. }
  796. } else {
  797. write(STDOUT_FILENO, &output, total);
  798. // This is OUTPUT from the BBS
  799. // ZF_LOGI( ">> %s", repr(output));
  800. ZF_LOGI(">> %d chars", (int)strlen(output));
  801. // I think repr is flipping out here. :(
  802. // ZF_LOGI( ">> %d", (int)strlen(repr(output)));
  803. }
  804. } else
  805. break;
  806. }
  807. // read_fd esta atribuido com a entrada padrao?
  808. if (FD_ISSET(STDIN_FILENO, &read_fd)) {
  809. // leia a entrada padrao
  810. total = read(STDIN_FILENO, &input, BSIZE);
  811. input[total] = 0;
  812. // e escreva no bc
  813. ZF_LOGI("<< %s", repr(input));
  814. write(master, &input, total);
  815. // This is INPUT from the USER
  816. // ZF_LOGI_MEM( input, strlen(input), "<< ");
  817. }
  818. }
  819. // Restore terminal
  820. tcsetattr(1, TCSAFLUSH, &orig1);
  821. ZF_LOGD("exit");
  822. }
  823. return 0;
  824. }