mystic.c 21 KB

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