mystic.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648
  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. const char * repr( const char * data) {
  43. static char buffer[1024];
  44. char * cp;
  45. strcpy( buffer, data );
  46. cp = buffer;
  47. while ( *cp != 0) {
  48. char c = *cp;
  49. if (isspace(c)) {
  50. if (c == ' ') {
  51. cp++;
  52. continue;
  53. };
  54. /* Ok, it's form-feed ('\f'), newline ('\n'), carriage return ('\r'), horizontal tab ('\t'), and vertical tab ('\v') */
  55. memmove( cp + 1, cp, strlen(cp) + 1);
  56. *cp = '\\';
  57. cp++;
  58. switch(c) {
  59. case '\f':
  60. *cp = 'f';
  61. cp++;
  62. break;
  63. case '\n':
  64. *cp = 'n';
  65. cp++;
  66. break;
  67. case '\r':
  68. *cp = 'r';
  69. cp++;
  70. break;
  71. case '\t':
  72. *cp = 't';
  73. cp++;
  74. break;
  75. case '\v':
  76. *cp = 'v';
  77. cp++;
  78. break;
  79. default:
  80. *cp = '?';
  81. cp++;
  82. break;
  83. }
  84. continue;
  85. }
  86. if (isprint(c)) {
  87. cp++;
  88. continue;
  89. };
  90. // Ok, default to \xHH output.
  91. memmove( cp + 3, cp, strlen(cp) + 1);
  92. *cp = '\\'; cp++;
  93. *cp = 'x'; cp++;
  94. char buffer[3];
  95. sprintf(buffer, "%02x", (int)c);
  96. *cp = buffer[0]; cp++;
  97. *cp = buffer[1]; cp++;
  98. continue;
  99. }
  100. return buffer;
  101. }
  102. // END LOGGING
  103. // What is the name of the actual, real Mystic that
  104. // we'll be executing and manglying?
  105. #define TARGET "./mySTIC"
  106. #define BSIZE 1024
  107. const char * it_is_now(void) {
  108. static char buffer[100];
  109. time_t timer;
  110. struct tm* tm_info;
  111. timer = time(NULL);
  112. tm_info = localtime(&timer);
  113. strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", tm_info);
  114. return buffer;
  115. }
  116. void slow_write(int fd, int speed, char * buffer, int len) {
  117. int x;
  118. for( x = 0; x < len; x++) {
  119. usleep(speed);
  120. write( fd, &buffer[x], 1);
  121. }
  122. }
  123. struct render {
  124. int speed;
  125. int effect;
  126. } current_render;
  127. int render_overlimit = 0;
  128. void reset_render(void) {
  129. current_render.speed = 0;
  130. current_render.effect = 0;
  131. render_overlimit = 0;
  132. }
  133. #define TRIGGER "^"
  134. #define SLEEP_LIMIT 30
  135. int ms_sleep(unsigned int ms) {
  136. int result = 0;
  137. struct timespec ts = {
  138. ms / 1000,
  139. (ms % 1000) * 1000000L
  140. };
  141. do {
  142. struct timespec ts_sleep = ts;
  143. result = nanosleep(&ts_sleep, &ts);
  144. } while ( (-1 == result));
  145. return result;
  146. }
  147. void render_sleep(void) {
  148. if (render_overlimit)
  149. return;
  150. if (current_render.speed) { // * 100 still too slow.
  151. ms_sleep(current_render.speed * 10);
  152. }
  153. }
  154. void write_color(int fd, int color) {
  155. char buffer[10];
  156. sprintf(buffer, "\x1b[%dm", color);
  157. write(fd, buffer, strlen(buffer));
  158. }
  159. const char * process_trigger(int fd, const char * cp) {
  160. char ch;
  161. int i, x, y;
  162. ch = toupper(*cp);
  163. cp++;
  164. switch(ch) {
  165. case 'D':
  166. i = 0;
  167. if (isdigit(*cp)) {
  168. i = (*cp) - '0';
  169. cp++;
  170. };
  171. if (isdigit(*cp)) {
  172. i *= 10;
  173. i += (*cp) - '0';
  174. cp++;
  175. };
  176. if ((i > 0) && (i < 80)) {
  177. ZF_LOGI("DEL %02d", i);
  178. for (x = 0; x < i; x++ ) {
  179. write(fd, "\b \b", 3);
  180. }
  181. };
  182. break;
  183. case 'C':
  184. i = 0;
  185. if (isdigit(*cp)) {
  186. i = (*cp) - '0';
  187. cp++;
  188. };
  189. if (isdigit(*cp)) {
  190. i *= 10;
  191. i += (*cp) - '0';
  192. cp++;
  193. };
  194. write_color(fd, i);
  195. break;
  196. case 'G':
  197. x = 0;
  198. if (isdigit(*cp)) {
  199. x = (*cp) - '0';
  200. cp++;
  201. };
  202. if (isdigit(*cp)) {
  203. x *= 10;
  204. x += (*cp) - '0';
  205. cp++;
  206. };
  207. y = 0;
  208. if (isdigit(*cp)) {
  209. y = (*cp) - '0';
  210. cp++;
  211. };
  212. if (isdigit(*cp)) {
  213. y *= 10;
  214. y += (*cp) - '0';
  215. cp++;
  216. };
  217. break;
  218. case 'R':
  219. i = 0;
  220. if (isdigit(*cp)) {
  221. i = (*cp) - '0';
  222. cp++;
  223. };
  224. if ( ( i > 0 ) && ( i < 10) ) {
  225. ZF_LOGI("RENDER %d", i);
  226. current_render.effect = i;
  227. } else {
  228. current_render.effect = 0;
  229. }
  230. break;
  231. case 'S':
  232. i = 0;
  233. if (isdigit(*cp)) {
  234. i = (*cp) - '0';
  235. cp++;
  236. };
  237. if ( ( i > 0 ) && ( i < 10) ) {
  238. ZF_LOGI( "SPEED %d", i);
  239. current_render.speed = i;
  240. } else {
  241. current_render.speed = 0;
  242. }
  243. break;
  244. case 'P':
  245. i = 0;
  246. if (isdigit(*cp)) {
  247. i = (*cp) - '0';
  248. cp++;
  249. };
  250. if ( ( i > 0 ) && ( i < 10) ) {
  251. ZF_LOGI( "PAWS %d", i);
  252. // sleep(i);
  253. if (!render_overlimit) {
  254. sleep(i);
  255. };
  256. }
  257. break;
  258. }
  259. return cp;
  260. }
  261. void render_effect(int fd, char ch) {
  262. int effect = current_render.effect;
  263. int l;
  264. char space = ' ';
  265. char bs = '\b';
  266. switch (effect) {
  267. case 1:
  268. // CHAR + SPC + BS
  269. render_sleep();
  270. write(fd, &ch, 1);
  271. render_sleep(); // Maybe extra sleep here?
  272. write(fd, &space, 1);
  273. render_sleep();
  274. write(fd, &bs, 1);
  275. break;
  276. case 2:
  277. // CHAR + 8 spaces + 8 BS
  278. render_sleep();
  279. write(fd, &ch, 1);
  280. for(l = 0; l < 8; l++) {
  281. render_sleep();
  282. write(fd, &space, 1);
  283. }
  284. for(l = 0; l < 8; l++) {
  285. render_sleep();
  286. write(fd, &bs, 1);
  287. }
  288. break;
  289. case 0:
  290. default:
  291. // NORMAL
  292. render_sleep();
  293. write(fd, &ch, 1);
  294. break;
  295. }
  296. }
  297. void render( int fd, const char * string_out ) {
  298. const char * cp = string_out;
  299. const char * trigger = cp;
  300. time_t start = time(NULL);
  301. int elapsed;
  302. int over = 0;
  303. reset_render();
  304. ZF_LOGD( "render(%d, %s)", fd, string_out);
  305. // Check our time from time to time.
  306. // If we start running long, kill sleeps.
  307. while ( (trigger = strstr(cp, TRIGGER)) != NULL) {
  308. // There is special things to handle in here.
  309. while ( cp != trigger ) {
  310. elapsed = time(NULL)-start;
  311. if (elapsed > SLEEP_LIMIT) {
  312. render_overlimit = 1;
  313. current_render.speed = 0;
  314. };
  315. // write(fd, cp, 1 );
  316. render_effect(fd, *cp);
  317. cp++;
  318. };
  319. // ZF_LOGI( "at trigger: (%s)", cp);
  320. cp += strlen(TRIGGER);
  321. // Ok, we're pointing at the trigger -- do something.
  322. cp = process_trigger(fd, cp);
  323. // ZF_LOGI( "after trigger: (%s)", cp);
  324. };
  325. // We still might be under a rendering effect.
  326. while (*cp != 0) {
  327. elapsed = time(NULL)-start;
  328. if (elapsed > SLEEP_LIMIT) {
  329. render_overlimit = 1;
  330. current_render.speed = 0;
  331. };
  332. // write(fd, cp, 1);
  333. render_effect(fd, *cp);
  334. cp++;
  335. }
  336. }
  337. void harry_event(int fd) {
  338. // Make something happen
  339. char buffer[100];
  340. int r = random() % 5;
  341. // This is no where near finished, BUT!
  342. // TO FIX: Remember the last phrase used,
  343. // and don't repeat (the last two)!
  344. const char * phrases[] =
  345. {
  346. "Hahaha",
  347. "Snicker, snicker",
  348. "Boo!",
  349. "MeOW",
  350. "I see U"
  351. };
  352. const char * cp;
  353. cp = phrases[r];
  354. sprintf(buffer, "^S2%s^P2^D%02d", cp, (int)strlen(cp));
  355. ZF_LOGD( "harry_event: render(%d, \"%s\")", fd, buffer);
  356. render(fd, buffer);
  357. }
  358. /*
  359. NOTE: Logging is single file, don't use in production!
  360. It won't handle multiple writers.
  361. */
  362. char * username = NULL;
  363. char * fullname = NULL;
  364. void pcopy(char * pstring, char * str) {
  365. int len = (int)*pstring;
  366. strncpy( str, pstring+1, len );
  367. str[len] = 0;
  368. }
  369. /*
  370. This only works for those few idiots that use SSH,
  371. the Mystic broken SSH!
  372. */
  373. int locate_user(const char *alias) {
  374. FILE * user;
  375. char buffer[0x600];
  376. char temp[100];
  377. user = fopen("data/users.dat", "rb");
  378. if (user == NULL)
  379. return 0;
  380. // Carry on!
  381. while (fread(buffer, 0x600, 1, user) == 1) {
  382. pcopy( buffer + 0x6d, temp );
  383. if ( strcasecmp( temp, username) == 0) {
  384. pcopy(buffer + 0x8c, temp );
  385. fullname = strdup(temp);
  386. break;
  387. }
  388. /*
  389. printf("Alias: %s\n", temp);
  390. pcopy(buffer + 0x8c, temp );
  391. printf("Full Name: %s\n", temp );
  392. */
  393. }
  394. fclose(user);
  395. return 1;
  396. }
  397. // Buffers are BSIZE + 1, so a buffer that size can strcpy safely.
  398. void mangle( char * buffer ) {
  399. // Ok, we want to look for words to:
  400. // MaNGlE , or transpose (both?!)
  401. // or possibly transpose words
  402. }
  403. int main(int argc, char * argv[])
  404. {
  405. int master;
  406. pid_t pid;
  407. int node = -1;
  408. file_output_open("horrible_harry.log");
  409. srandom(time(NULL));
  410. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz -PUWISHPASSWORD
  411. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  412. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz -PUP2LAT3
  413. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  414. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  415. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML0 -SL1 -ST0 -CUnknown
  416. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown -Ubugz -PUP2LAT3
  417. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML1 -SL1 -ST2 -CUnknown -Ubugz -PUP2LAT3
  418. // SSH: -ML1 -ST2
  419. // Telnet: -ML0 -ST0
  420. // Locate username (if given) in the command line
  421. // -U<username>
  422. for (int x = 0; x < argc; x++) {
  423. if (strncmp("-U", argv[x], 2) == 0) {
  424. username = argv[x] + 2;
  425. ZF_LOGI( "Username: [%s]", username);
  426. };
  427. if (strncmp("-SL", argv[x], 3) == 0) {
  428. node = argv[x][3] - '0' + 1;
  429. ZF_LOGI( "Node: %d", node);
  430. }
  431. }
  432. if (username != NULL) {
  433. locate_user(username);
  434. ZF_LOGD( "Username: [%s] A.K.A. [%s]", username, fullname);
  435. }
  436. // With IGNBRK I don't think I need this anymore. (Nope!)
  437. // signal(SIGINT, SIG_IGN);
  438. pid = forkpty(&master, NULL, NULL, NULL);
  439. // impossible to fork
  440. if (pid < 0) {
  441. return 1;
  442. }
  443. // child
  444. else if (pid == 0) {
  445. char *args[20]; // max 20 args
  446. int x;
  447. args[0] = TARGET;
  448. for ( x = 1; x < argc; x++ ) {
  449. args[x] = argv[x];
  450. };
  451. args[x] = NULL;
  452. // run Mystic, run!
  453. execvp( TARGET, args);
  454. }
  455. // parent
  456. else {
  457. // remove the echo
  458. // ICANON - raw mode. No more line buffering!
  459. struct termios tios, orig1;
  460. struct timeval timeout;
  461. int last_event;
  462. tcgetattr(master, &tios);
  463. tios.c_lflag &= ~(ECHO | ECHONL | ICANON );
  464. tcsetattr(master, TCSAFLUSH, &tios);
  465. tcgetattr(1, &orig1);
  466. tios = orig1;
  467. tios.c_iflag &= ~(ICRNL | IXON); // Disable software flow control
  468. tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN );
  469. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  470. // ISIG should be Ctrl-C and Ctrl-Z IGNBRK +
  471. tcsetattr(1, TCSAFLUSH, &tios);
  472. for (;;) {
  473. // define estruturas para o select, que serve para verificar qual
  474. // se tornou "pronto pra uso"
  475. fd_set read_fd;
  476. fd_set write_fd;
  477. fd_set except_fd;
  478. // inicializa as estruturas
  479. FD_ZERO(&read_fd);
  480. FD_ZERO(&write_fd);
  481. FD_ZERO(&except_fd);
  482. // atribui o descritor master, obtido pelo forkpty, ao read_fd
  483. FD_SET(master, &read_fd);
  484. // atribui o stdin ao read_fd
  485. FD_SET(STDIN_FILENO, &read_fd);
  486. // o descritor tem que ser unico para o programa, a documentacao
  487. // recomenda um calculo entre os descritores sendo usados + 1
  488. // we're in luck! The last parameter is time interval/timeout. :D
  489. timeout.tv_sec = 10;
  490. timeout.tv_usec = 0;
  491. // select(master+1, &read_fd, &write_fd, &except_fd, NULL);
  492. if ( select(master+1, &read_fd, &write_fd, &except_fd, &timeout) == 0 ) {
  493. // This means timeout!
  494. harry_event(STDOUT_FILENO);
  495. ZF_LOGI( "%s : TICK", it_is_now());
  496. }
  497. char input[BSIZE + 1];
  498. char output[BSIZE + 1];
  499. int total;
  500. // read_fd esta atribuido com read_fd?
  501. if (FD_ISSET(master, &read_fd))
  502. {
  503. // leia o que bc esta mandando
  504. if ((total = read(master, &output, BSIZE)) != -1) {
  505. // e escreva isso na saida padrao
  506. output[total] = 0;
  507. if (random() % 20 < 3) {
  508. mangle( output );
  509. }
  510. write(STDOUT_FILENO, &output, total);
  511. // This is OUTPUT from the BBS
  512. ZF_LOGI( ">> %s", repr(output));
  513. // ZF_LOGI_MEM( output, strlen(output), ">> ");
  514. } else
  515. break;
  516. }
  517. // read_fd esta atribuido com a entrada padrao?
  518. if (FD_ISSET(STDIN_FILENO, &read_fd))
  519. {
  520. // leia a entrada padrao
  521. total = read(STDIN_FILENO, &input, BSIZE);
  522. input[total] = 0;
  523. // e escreva no bc
  524. ZF_LOGI( "<< %s", repr(input));
  525. write(master, &input, total);
  526. // This is INPUT from the USER
  527. // ZF_LOGI_MEM( input, strlen(input), "<< ");
  528. }
  529. }
  530. // Restore terminal
  531. tcsetattr(1, TCSAFLUSH, &orig1);
  532. ZF_LOGD("exit");
  533. }
  534. return 0;
  535. }