mystic.c 22 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006
  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's 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. #ifdef CPP_MADMAN_STL_CODE
  413. const char *random_phrase(const char *words, int len, int last_seen) {
  414. // ooh. a map of char *s to last_seen_events. :P
  415. static map<const char *, array<int>> tracker;
  416. map<const char *, array<int>>::iterator it;
  417. array<int, last_seen> it = tracker.find(words);
  418. if (it == tracker.end()) {
  419. // key does not exist.
  420. array<int, last_seen> last;
  421. for (int i = 0; i < last_seen; i++) {
  422. last[i] = -1;
  423. };
  424. tracker.insert(words, last);
  425. it = tracker.find(words);
  426. };
  427. }
  428. #endif
  429. void harry_event(int fd) {
  430. // Make something happen
  431. char buffer[100];
  432. int r;
  433. // This is no where near finished, BUT!
  434. const char *phrases[] = {"Hahaha", "Snicker, snicker", "Boo!",
  435. "MeOW", "I see U", "Arrooo!",
  436. "Ahh-wooo!", "Aaaooo!"};
  437. const char *cp;
  438. // Remember the last phrase used,
  439. // and don't repeat (the last two)!
  440. do {
  441. r = random() % ((sizeof(phrases) / sizeof(char *)) - 1);
  442. } while (have_seen(last_seen_harry_event, MAX_HARRY_EVENT_DUPS, r));
  443. ZF_LOGD("%d => %d %d", r, last_seen_harry_event[0], last_seen_harry_event[1]);
  444. cp = phrases[r];
  445. sprintf(buffer, "^S2%s^P2^D%02d", cp, (int)strlen(cp));
  446. ZF_LOGD("harry_event: render(%d, \"%s\")", fd, buffer);
  447. render(fd, buffer);
  448. }
  449. void init_harry() {
  450. init_have_seen(last_seen_harry_event, MAX_HARRY_EVENT_DUPS);
  451. ZF_LOGD("init => %d %d", last_seen_harry_event[0], last_seen_harry_event[1]);
  452. }
  453. /*
  454. The code to get the username and fullname is useless on telnet
  455. connections.
  456. */
  457. char *username = NULL;
  458. char *fullname = NULL;
  459. /*
  460. Pascal String Copy. Copy from pascal string, to C String.
  461. First char is pascal string length. (Max 255).
  462. */
  463. void pcopy(char *pstring, char *str) {
  464. int len = (int)*pstring;
  465. strncpy(str, pstring + 1, len);
  466. str[len] = 0;
  467. }
  468. /*
  469. This only works for those few idiots that use the
  470. horribly broken SSH crap that Mystic uses.
  471. */
  472. int locate_user(const char *alias) {
  473. FILE *user;
  474. char buffer[0x600];
  475. char temp[100];
  476. user = fopen("data/users.dat", "rb");
  477. if (user == NULL)
  478. return 0;
  479. // Carry on!
  480. while (fread(buffer, 0x600, 1, user) == 1) {
  481. pcopy(buffer + 0x6d, temp);
  482. if (strcasecmp(temp, username) == 0) {
  483. pcopy(buffer + 0x8c, temp);
  484. fullname = strdup(temp);
  485. break;
  486. }
  487. /*
  488. printf("Alias: %s\n", temp);
  489. pcopy(buffer + 0x8c, temp );
  490. printf("Full Name: %s\n", temp );
  491. */
  492. }
  493. fclose(user);
  494. return 1;
  495. }
  496. // Buffers are BSIZE + 1, so a buffer that size can strcpy safely.
  497. regex_t ANSI;
  498. regex_t WORDS;
  499. regex_t WORD;
  500. int init_regex(void) {
  501. int ret;
  502. char ansi[] = "\x1b\[[0-9]+(;[0-9]+)*?[a-zA-Z]";
  503. char words[] = "[a-zA-Z]+( [a-zA-Z]+)+";
  504. char word[] = "[a-zA-Z]+";
  505. char errorbuf[100];
  506. if (ret = regcomp(&ANSI, ansi, REG_EXTENDED | REG_NEWLINE)) {
  507. regerror(ret, &ANSI, errorbuf, sizeof(errorbuf));
  508. ZF_LOGW("Regex %s failed to compile: %s", ansi, errorbuf);
  509. return 0;
  510. };
  511. if (ret = regcomp(&WORDS, words, REG_EXTENDED | REG_NEWLINE)) {
  512. regerror(ret, &WORDS, errorbuf, sizeof(errorbuf));
  513. ZF_LOGW("Regex %s failed to compile: %s", words, errorbuf);
  514. return 0;
  515. };
  516. if (ret = regcomp(&WORD, word, REG_EXTENDED | REG_NEWLINE)) {
  517. regerror(ret, &WORD, errorbuf, sizeof(errorbuf));
  518. ZF_LOGW("Regex %s failed to compile: %s", word, errorbuf);
  519. return 0;
  520. };
  521. return 1;
  522. }
  523. int regmatch(regex_t *preg, const char *string, size_t nmatch,
  524. regmatch_t pmatch[], int eflags) {
  525. // returns number of matches found. (Max nmatch)
  526. int matches = 0;
  527. int offset = 0;
  528. int ret;
  529. while (matches < nmatch) {
  530. ret = regexec(preg, string + offset, nmatch - matches, pmatch + matches,
  531. eflags);
  532. if (!ret) {
  533. int current = offset;
  534. offset += pmatch[matches].rm_eo;
  535. pmatch[matches].rm_so += current;
  536. pmatch[matches].rm_eo += current;
  537. matches++;
  538. } else if (ret == REG_NOMATCH) {
  539. break;
  540. } else {
  541. break;
  542. }
  543. }
  544. return matches;
  545. }
  546. #define MAX_MATCH 32
  547. regmatch_t rxmatch[MAX_MATCH];
  548. int rx_match(regex_t *regex, const char *buffer) {
  549. int ret;
  550. ret = regmatch(regex, buffer, MAX_MATCH, rxmatch, 0);
  551. if (0) {
  552. for (int i = 0; i < ret; i++) {
  553. ZF_LOGI("%d : (%d-%d)", i, rxmatch[i].rm_so, rxmatch[i].rm_eo);
  554. }
  555. }
  556. return ret;
  557. }
  558. /*
  559. Terminal processing section.
  560. Monitor lines, position, color.
  561. What screen size do I want to emulate?
  562. ANSI codes.
  563. Do I need this??
  564. */
  565. int random_activate(int w) {
  566. int r = random() % 100;
  567. if (r <= (w * 10)) {
  568. return 1;
  569. };
  570. return 0;
  571. }
  572. /*
  573. * The buffer that we've been given is much larger now.
  574. *
  575. */
  576. int mangle(int fd, char *buffer) {
  577. int x, i;
  578. int need_render = 0; // changing word case around doesn't need the render
  579. int mangled = 0;
  580. char *cp;
  581. // Ok, we want to look for words to:
  582. // MaNGlE , or transpose (both?!)
  583. // or possibly transpose words
  584. char work[(BSIZE * 4) + 1];
  585. ZF_LOGI("mangle(%s)", repr(buffer));
  586. strcpy(work, buffer);
  587. /*
  588. NOTE: We copy the buffer, so we can clear out ANSI codes, etc.
  589. Otherwise we might mess some ANSI up in the manglying
  590. process.
  591. */
  592. /*
  593. (random) Look for ANSI CLS and:
  594. display random spooky texts around, with delays ... then CLS.
  595. display ANSI graphic file, with delays ... then CLS
  596. */
  597. const char *ANSI_CLS = "\x1b[2J";
  598. cp = strstr(buffer, ANSI_CLS);
  599. if (cp != NULL) {
  600. ZF_LOGI("seen: ANSI_CLS");
  601. if (random_activate(9)) {
  602. char display[100] = "";
  603. ZF_LOGI("mangle(ANSI_CLS)");
  604. // sprintf( display, "^P2...");
  605. // This string actually screws up ANSI detection (takes too long)
  606. // strcpy(display, "^P2^S501234567890^P1abcdef^P2g^P3h^P4i^S0^P2");
  607. strcpy(display, "^P2^S301234^P15^S0^P2");
  608. // Move the buffer so there's room for the display string.
  609. memmove(cp + strlen(display), cp, strlen(cp) + 1);
  610. strncpy(cp, display, strlen(display));
  611. ZF_LOGI("mangle(ANSI_CLS): (%d) %s", (int)strlen(buffer), repr(buffer));
  612. need_render = 1;
  613. /*
  614. Copy the new buffer over, but hide our "render" code
  615. from the remaining mangler steps.
  616. */
  617. strcpy(work, buffer);
  618. i = cp - buffer;
  619. // find offset into "buffer"
  620. // apply to work.
  621. memset(work + i, ' ', strlen(display));
  622. };
  623. }
  624. /* work -- clear out ANSI so we don't mangle ANSI codes. */
  625. x = rx_match(&ANSI, work);
  626. char replace_with = ' ';
  627. if (x > 0) {
  628. ZF_LOGD("found %d ANSI", x);
  629. for (i = 0; i < x; i++) {
  630. memset(work + rxmatch[i].rm_so, replace_with,
  631. rxmatch[i].rm_eo - rxmatch[i].rm_so);
  632. };
  633. ZF_LOGD("Work Now : (%d) %s", (int)strlen(work), repr(work));
  634. }
  635. // ZF_LOGI("mangle: %s", repr(work));
  636. /*
  637. (random) Locate words (in work), and possibly flip them around.
  638. Transpose words. Transpose case. Transpose letters.
  639. */
  640. x = rx_match(&WORDS, work);
  641. ZF_LOGD("found %d WORDS", x);
  642. if (x > 0) {
  643. for (i = 0; i < x; i++) {
  644. // Do things here.
  645. if (i % 3 == 0) {
  646. for (int p = rxmatch[i].rm_so; p < rxmatch[i].rm_eo; p++) {
  647. buffer[p] = tolower(buffer[p]);
  648. mangled++;
  649. }
  650. } else {
  651. if (i % 3 == 1) {
  652. for (int p = rxmatch[i].rm_so; p < rxmatch[i].rm_eo; p++) {
  653. buffer[p] = toupper(buffer[p]);
  654. mangled++;
  655. }
  656. }
  657. }
  658. }
  659. }
  660. /*
  661. (random) Locate single words, and transpose them. Transpose case.
  662. Transpose letters.
  663. */
  664. /*
  665. (random) Display up to certain point. Delay.
  666. Print some characters slowly. Delay.
  667. */
  668. if (need_render) {
  669. ZF_LOGD("HH %d : (%d) %s", need_render, (int)strlen(buffer), repr(buffer));
  670. } else {
  671. if (mangled) {
  672. ZF_LOGD("Mangled %d : %s", mangled, repr(buffer));
  673. }
  674. }
  675. if (need_render) {
  676. render(fd, buffer);
  677. } else {
  678. write(fd, buffer, strlen(buffer));
  679. };
  680. return need_render && mangled;
  681. }
  682. int harry_happens(time_t *last_event, int wakeup) {
  683. time_t now = time(NULL);
  684. int elapsed = now - *last_event;
  685. if (elapsed > wakeup) {
  686. // Ok! It's been too long since we've done something.
  687. *last_event = now;
  688. return 1;
  689. }
  690. return 0;
  691. }
  692. int main(int argc, char *argv[]) {
  693. int master;
  694. pid_t pid;
  695. int node = -1;
  696. file_output_open("horrible_harry.log");
  697. init_harry();
  698. srandom(time(NULL));
  699. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  700. // -PUWISHPASSWORD
  701. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  702. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  703. // -PUP2LAT3
  704. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  705. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  706. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML0 -SL1 -ST0 -CUnknown
  707. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz
  708. // -PUP2LAT3
  709. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML1 -SL1 -ST2 -CUnknown -Ubugz
  710. // -PUP2LAT3
  711. // SSH: -ML1 -ST2
  712. // Telnet: -ML0 -ST0
  713. // Locate username (if given) in the command line
  714. // -U<username>
  715. for (int x = 0; x < argc; x++) {
  716. if (strncmp("-U", argv[x], 2) == 0) {
  717. username = argv[x] + 2;
  718. ZF_LOGI("Username: [%s]", username);
  719. };
  720. if (strncmp("-SL", argv[x], 3) == 0) {
  721. node = argv[x][3] - '0' + 1;
  722. ZF_LOGI("Node: %d", node);
  723. }
  724. }
  725. if (username != NULL) {
  726. locate_user(username);
  727. ZF_LOGD("Username: [%s] A.K.A. [%s]", username, fullname);
  728. }
  729. if (!init_regex())
  730. return 2;
  731. // With IGNBRK I don't think I need this anymore. (Nope!)
  732. // signal(SIGINT, SIG_IGN);
  733. pid = forkpty(&master, NULL, NULL, NULL);
  734. // impossible to fork
  735. if (pid < 0) {
  736. return 1;
  737. }
  738. // child
  739. else if (pid == 0) {
  740. char *args[20]; // max 20 args
  741. int x;
  742. args[0] = TARGET;
  743. for (x = 1; x < argc; x++) {
  744. args[x] = argv[x];
  745. };
  746. args[x] = NULL;
  747. // run Mystic, run!
  748. execvp(TARGET, args);
  749. }
  750. // parent
  751. else {
  752. // remove the echo
  753. // ICANON - raw mode. No more line buffering!
  754. struct termios tios, orig1;
  755. struct timeval timeout;
  756. time_t last_event = 0; // time(NULL);
  757. ZF_LOGD("starting");
  758. tcgetattr(master, &tios);
  759. tios.c_lflag &= ~(ECHO | ECHONL | ICANON);
  760. tcsetattr(master, TCSAFLUSH, &tios);
  761. tcgetattr(1, &orig1);
  762. tios = orig1;
  763. tios.c_iflag &= ~(ICRNL | IXON); // Disable software flow control
  764. tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
  765. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  766. // ISIG should be Ctrl-C and Ctrl-Z IGNBRK +
  767. tcsetattr(1, TCSAFLUSH, &tios);
  768. for (;;) {
  769. // define estruturas para o select, que serve para verificar qual
  770. // se tornou "pronto pra uso"
  771. fd_set read_fd;
  772. fd_set write_fd;
  773. fd_set except_fd;
  774. // inicializa as estruturas
  775. FD_ZERO(&read_fd);
  776. FD_ZERO(&write_fd);
  777. FD_ZERO(&except_fd);
  778. // atribui o descritor master, obtido pelo forkpty, ao read_fd
  779. FD_SET(master, &read_fd);
  780. // atribui o stdin ao read_fd
  781. FD_SET(STDIN_FILENO, &read_fd);
  782. // o descritor tem que ser unico para o programa, a documentacao
  783. // recomenda um calculo entre os descritores sendo usados + 1
  784. /*
  785. TODO: Figure out how this would work.
  786. I'm thinking something like timeouts 30-50 seconds?
  787. And as we get closer, 15-25 seconds.
  788. */
  789. // we're in luck! The last parameter is time interval/timeout. :D
  790. timeout.tv_sec = 10;
  791. timeout.tv_usec = 0;
  792. // select(master+1, &read_fd, &write_fd, &except_fd, NULL);
  793. if (select(master + 1, &read_fd, &write_fd, &except_fd, &timeout) == 0) {
  794. // This means timeout!
  795. ZF_LOGI("TIMEOUT");
  796. harry_event(STDOUT_FILENO);
  797. }
  798. char input[BSIZE + 1];
  799. static char output[(BSIZE * 4) + 1];
  800. int total;
  801. // read_fd esta atribuido com read_fd?
  802. if (FD_ISSET(master, &read_fd)) {
  803. // leia o que bc esta mandando
  804. if ((total = read(master, &output, BSIZE)) != -1) {
  805. // e escreva isso na saida padrao
  806. output[total] = 0;
  807. // if ( harry_happens( &last_event, 5)) {
  808. if (1) {
  809. ZF_LOGI("harry_happens");
  810. if (mangle(STDOUT_FILENO, output) == 0) {
  811. // failed, so. Try again.
  812. last_event = 0;
  813. }
  814. } else {
  815. write(STDOUT_FILENO, &output, total);
  816. // This is OUTPUT from the BBS
  817. // ZF_LOGI( ">> %s", repr(output));
  818. ZF_LOGI(">> %d chars", (int)strlen(output));
  819. // I think repr is flipping out here. :(
  820. // ZF_LOGI( ">> %d", (int)strlen(repr(output)));
  821. }
  822. } else
  823. break;
  824. }
  825. // read_fd esta atribuido com a entrada padrao?
  826. if (FD_ISSET(STDIN_FILENO, &read_fd)) {
  827. // leia a entrada padrao
  828. total = read(STDIN_FILENO, &input, BSIZE);
  829. input[total] = 0;
  830. // e escreva no bc
  831. ZF_LOGI("<< %s", repr(input));
  832. write(master, &input, total);
  833. // This is INPUT from the USER
  834. // ZF_LOGI_MEM( input, strlen(input), "<< ");
  835. }
  836. }
  837. // Restore terminal
  838. tcsetattr(1, TCSAFLUSH, &orig1);
  839. ZF_LOGD("exit");
  840. }
  841. return 0;
  842. }