hharry.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. #include <fcntl.h>
  2. #include <pty.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. #include <sys/select.h>
  6. #include <sys/wait.h>
  7. #include <termios.h>
  8. #include <unistd.h>
  9. #include <sstream>
  10. #include <string>
  11. using namespace std;
  12. // #include <signal.h> // handle Ctrl-C/SIGINT
  13. #include <strings.h> // strcasecmp
  14. #include <time.h>
  15. #include <ctype.h>
  16. #include <stdlib.h> // random()
  17. /* Log level guideline:
  18. * - ZF_LOG_FATAL - happened something impossible and absolutely unexpected.
  19. * Process can't continue and must be terminated.
  20. * Example: division by zero, unexpected modifications from other thread.
  21. * - ZF_LOG_ERROR - happened something possible, but highly unexpected. The
  22. * process is able to recover and continue execution.
  23. * Example: out of memory (could also be FATAL if not handled properly).
  24. * - ZF_LOG_WARN - happened something that *usually* should not happen and
  25. * significantly changes application behavior for some period of time.
  26. * Example: configuration file not found, auth error.
  27. * - ZF_LOG_INFO - happened significant life cycle event or major state
  28. * transition.
  29. * Example: app started, user logged in.
  30. * - ZF_LOG_DEBUG - minimal set of events that could help to reconstruct the
  31. * execution path. Usually disabled in release builds.
  32. * - ZF_LOG_VERBOSE - all other events. Usually disabled in release builds.
  33. *
  34. * *Ideally*, log file of debugged, well tested, production ready application
  35. * should be empty or very small. Choosing a right log level is as important as
  36. * providing short and self descriptive log message.
  37. */
  38. /*
  39. #define ZF_LOG_VERBOSE 1
  40. #define ZF_LOG_DEBUG 2
  41. #define ZF_LOG_INFO 3
  42. #define ZF_LOG_WARN 4
  43. #define ZF_LOG_ERROR 5
  44. #define ZF_LOG_FATAL 6
  45. */
  46. // When debugging low-level, use this:
  47. // #define ZF_LOG_LEVEL ZF_LOG_VERBOSE
  48. // Except this doesn't work. It needs to be anywere the
  49. // zf_log.h is included.
  50. // LOGGING with file output
  51. #include "zf_log.h"
  52. FILE *g_log_file;
  53. static void file_output_callback(const zf_log_message *msg, void *arg) {
  54. (void)arg;
  55. *msg->p = '\n';
  56. fwrite(msg->buf, msg->p - msg->buf + 1, 1, g_log_file);
  57. fflush(g_log_file);
  58. }
  59. static void file_output_close(void) { fclose(g_log_file); }
  60. static int file_output_open(const char *const log_path) {
  61. g_log_file = fopen(log_path, "a");
  62. if (!g_log_file) {
  63. ZF_LOGW("Failed to open log file %s", log_path);
  64. return 0;
  65. }
  66. atexit(file_output_close);
  67. zf_log_set_output_v(ZF_LOG_PUT_STD, 0, file_output_callback);
  68. return 1;
  69. }
  70. void log_flush(void) { fflush(g_log_file); }
  71. // END LOGGING
  72. #include "terminal.h"
  73. #include "utils.h"
  74. struct console_details console;
  75. #include "wordplay.h"
  76. /*
  77. What is the name of the actual, real Mystic executable
  78. that we'll be executing and mangling?
  79. */
  80. #define TARGET "./mySTIC"
  81. // Size of our input and output buffers.
  82. #define BSIZE 1024
  83. /*
  84. These are harry "timeout" events.
  85. These happen when we've been sitting around awhile.
  86. */
  87. /*
  88. The code to get the username and fullname is useless on telnet
  89. connections.
  90. */
  91. string username;
  92. string fullname;
  93. /*
  94. This only works for those few idiots that use the
  95. horribly broken SSH crap that Mystic uses.
  96. */
  97. int locate_user(const char *alias) {
  98. FILE *user;
  99. char buffer[0x600];
  100. char temp[100];
  101. user = fopen("data/users.dat", "rb");
  102. if (user == NULL)
  103. return 0;
  104. // Carry on!
  105. while (fread(buffer, 0x600, 1, user) == 1) {
  106. pcopy(buffer + 0x6d, temp);
  107. if (strcasecmp(temp, username.c_str()) == 0) {
  108. pcopy(buffer + 0x8c, temp);
  109. fullname.assign(temp);
  110. break;
  111. }
  112. /*
  113. printf("Alias: %s\n", temp);
  114. pcopy(buffer + 0x8c, temp );
  115. printf("Full Name: %s\n", temp );
  116. */
  117. }
  118. fclose(user);
  119. return 1;
  120. }
  121. /*
  122. This is done. :D My buffering system works with stack'em.
  123. TO FIX: Stop using c strings, must use char * buffer + int length.
  124. MAY CONTAIN NULL VALUES.
  125. Rework some things here.
  126. Here's the "plan":
  127. if buffer is EMPTY:
  128. time_idle = 1;
  129. // setup for "random timeout value mess"
  130. // we're in luck! The last parameter is time interval/timeout. :D
  131. timeout.tv_sec = 10; // randrange(10-25)
  132. timeout.tv_usec = 0;
  133. NOT EMPTY:
  134. // we're in luck! The last parameter is time interval/timeout. :D
  135. timeout.tv_sec = 0;
  136. timeout.tv_usec = 10; // Wild Guess Here? Maybe higher, maybe lower?
  137. time_idle = 0;
  138. ON READ:
  139. read/append to current buffer.
  140. We can't use nulls -- what if they are using ZModem, there's nulls in the
  141. file! Look for trailing / the very last "\r\n".
  142. (I could mangle/chunk it line by line. But I'm not sure I'd need to do
  143. that.)
  144. Optional "mangle" buffer up to that very point -- and send up to that point.
  145. Option #2: Maybe we send everything if program has been running for under
  146. 20 seconds. This would allow the ANSI detect to not get screwed up by this
  147. new idea.
  148. ON TIMEOUT:
  149. if time_idle:
  150. Activate funny harry timeout events.
  151. else:
  152. Ok, we *STILL* haven't received any more characters into the buffer --
  153. even after waiting. (Maybe we haven't waited long enough?)
  154. send the pending information in the buffer and clear it out.
  155. Maybe this is a prompt, and there won't be a \r\n.
  156. This allows for cleaner process of "lines" of buffer. We shouldn't break
  157. in the midDLE OF A WORD. Downside is that we sit on buffer contents a little
  158. while / some amount of time -- which will add some lag to prompts showing up.
  159. (LAG? Are you kidding?)
  160. ZModem:
  161. start: "rz^M**"...
  162. 05-12 18:12:15.916 >> rz^M**^XB00000000000000^M<8A>^Q
  163. 05-12 18:12:15.928 << **\x18B0100000023be50\r\n\x11
  164. 05-12 18:12:15.928 >> *^XC^D
  165. 05-12 18:12:15.939 << **\x18B0900000000a87c\r\n\x11
  166. 05-12 18:12:15.940 >> *^XC
  167. # Start of PK zipfile.
  168. 05-12 18:12:15.941 >> PK^C^D^T
  169. end:
  170. 05-12 18:26:38.700 << **\x18B0100000023be50\r\n\x11
  171. 05-12 18:26:38.700 >> **^XB0823a77600344c^M<8A>
  172. 05-12 18:26:38.711 << **\x18B0800000000022d\r\n
  173. 05-12 18:26:38.712 >> OO^MESC[0m
  174. */
  175. // TODO: Get everything above this -- into another file.
  176. int main(int argc, char *argv[]) {
  177. int master;
  178. pid_t pid;
  179. int node = -1;
  180. init_harry();
  181. srandom(time(NULL));
  182. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown
  183. // -Ubugz -PUWISHPASSWORD
  184. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  185. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown
  186. // -Ubugz -PUWISH
  187. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  188. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML0 -SL0 -ST0 -CUnknown
  189. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML0 -SL1 -ST0 -CUnknown
  190. // ./mystic -TID7 -IP192.168.0.1 -HOSTUnknown -ML1 -SL0 -ST2 -CUnknown
  191. // -Ubugz -PDUMBWAYTODOTHIS
  192. // ./mystic -TID9 -IP192.168.0.1 -HOSTUnknown -ML1 -SL1 -ST2 -CUnknown
  193. // -Ubugz -PIDONTUSEPASCAL
  194. // SSH: -ML1 -ST2
  195. // Telnet: -ML0 -ST0
  196. // Locate username (if given) in the command line
  197. // -U<username>
  198. for (int x = 0; x < argc; x++) {
  199. if (strncmp("-U", argv[x], 2) == 0) {
  200. username.assign(argv[x] + 2);
  201. }
  202. if (strncmp("-SL", argv[x], 3) == 0) {
  203. node = atoi(argv[x] + 3) + 1;
  204. }
  205. }
  206. if (node == -1) {
  207. // likely this is someone trying to run something
  208. printf("Try ./mySTIC (whatever it is you're trying)!\n");
  209. return 2;
  210. }
  211. string logfile;
  212. {
  213. ostringstream buffer;
  214. buffer << "horrible_harry_" << node << ".log";
  215. logfile = buffer.str();
  216. };
  217. if (!file_output_open((const char *)logfile.c_str()))
  218. return 2;
  219. ZF_LOGI("Node: %d", node);
  220. if (!username.empty()) {
  221. locate_user(username.c_str());
  222. ZF_LOGD("Username: [%s] A.K.A. [%s]", (const char *)username.c_str(),
  223. (const char *)fullname.c_str());
  224. }
  225. pid = forkpty(&master, NULL, NULL, NULL);
  226. // impossible to fork
  227. if (pid < 0) {
  228. return 1;
  229. }
  230. // child
  231. else if (pid == 0) {
  232. char *args[20]; // max 20 args
  233. int x;
  234. char new_exec[] = TARGET;
  235. // build new args list
  236. args[0] = new_exec;
  237. for (x = 1; x < argc; x++) {
  238. args[x] = argv[x];
  239. };
  240. // null term the list
  241. args[x] = NULL;
  242. // run Mystic, run!
  243. execvp(TARGET, args);
  244. }
  245. // parent
  246. else {
  247. struct termios tios, orig1;
  248. struct timeval timeout;
  249. time_t last_event = 0; // time(NULL);
  250. ZF_LOGD("starting");
  251. tcgetattr(master, &tios);
  252. tios.c_lflag &= ~(ECHO | ECHONL | ICANON);
  253. /*
  254. tios.c_iflag &= ~(ICRNL | IXON | BRKINT);
  255. tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
  256. tios.c_oflag &= ~(OPOST);
  257. */
  258. tcsetattr(master, TCSAFLUSH, &tios);
  259. tcgetattr(1, &orig1);
  260. tios = orig1;
  261. tios.c_iflag &= ~(ICRNL | IXON | BRKINT);
  262. tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
  263. tios.c_oflag &= ~(OPOST);
  264. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  265. tcsetattr(1, TCSAFLUSH, &tios);
  266. /*
  267. This doesn't need to be static -- because it is part of
  268. main. Once main ends, we're done.
  269. */
  270. string buffer;
  271. buffer.reserve(BSIZE * 2);
  272. string play;
  273. play.reserve(4096);
  274. // int size = 0; // use buffer.size() instead
  275. for (;;) {
  276. int time_idle;
  277. // define estruturas para o select, que serve para verificar qual
  278. // se tornou "pronto pra uso"
  279. fd_set read_fd;
  280. fd_set write_fd;
  281. fd_set except_fd;
  282. // inicializa as estruturas
  283. FD_ZERO(&read_fd);
  284. FD_ZERO(&write_fd);
  285. FD_ZERO(&except_fd);
  286. // atribui o descritor master, obtido pelo forkpty, ao read_fd
  287. FD_SET(master, &read_fd);
  288. // atribui o stdin ao read_fd
  289. FD_SET(STDIN_FILENO, &read_fd);
  290. // o descritor tem que ser unico para o programa, a documentacao
  291. // recomenda um calculo entre os descritores sendo usados + 1
  292. /*
  293. TODO: Figure out how this would work.
  294. I'm thinking something like timeouts 30-50 seconds?
  295. And as we get closer, 15-25 seconds.
  296. */
  297. if (buffer.size() == 0) {
  298. // buffer is empty
  299. timeout.tv_sec = randrange(10, 20);
  300. timeout.tv_usec = 0;
  301. time_idle = 1;
  302. } else {
  303. // buffer is not empty
  304. timeout.tv_sec = 0;
  305. timeout.tv_usec = 1;
  306. time_idle = 0;
  307. }
  308. if (select(master + 1, &read_fd, &write_fd, &except_fd, &timeout) == 0) {
  309. ZF_LOGI("TIMEOUT");
  310. // This means timeout!
  311. if (time_idle) {
  312. harry_idle_event(STDOUT_FILENO);
  313. } else {
  314. ZF_LOGI_MEM(buffer.data(), buffer.size(), "TIMEOUT buffer size=%lu",
  315. buffer.size());
  316. play.assign(buffer);
  317. mangle(STDOUT_FILENO, play);
  318. /*
  319. ZF_LOGI("console_receive");
  320. console_receive(&console, buffer);
  321. ZF_LOGI("write buffer");
  322. write(STDOUT_FILENO, buffer.data(), buffer.size());
  323. */
  324. ZF_LOGI("buffer clear");
  325. buffer.clear();
  326. // size = 0;
  327. // buffer is empty now
  328. }
  329. }
  330. // read_fd esta atribuido com read_fd?
  331. if (FD_ISSET(master, &read_fd)) {
  332. // leia o que bc esta mandando
  333. // ZF_LOGD("read (%d) %d bytes", size, BSIZE - size);
  334. char read_buffer[BSIZE + 1];
  335. int total;
  336. // We may adjust this later on (adjusting read length).
  337. if ((total = read(master, read_buffer, BSIZE)) != -1) {
  338. // Ok, we've read more into the buffer.
  339. ZF_LOGV("Read %d bytes", total);
  340. buffer.append(read_buffer, total);
  341. // ZF_LOGV_MEM(buffer + size, total, "Read %d bytes:", total);
  342. // size += total;
  343. // ZF_LOGV_MEM(buffer, size, "Buffer now:");
  344. int pos = buffer.rfind("\r\n");
  345. // rstrnstr(buffer, size, "\r\n");
  346. // >= 0) {
  347. if (pos != string::npos) {
  348. // found something!
  349. pos += 2;
  350. // play = buffer.substr() wipes out play's reserve.
  351. // play = buffer.substr(0, pos);
  352. play.assign(buffer, 0, pos);
  353. ZF_LOGI("play %lu size, %lu cap", play.size(), play.capacity());
  354. // play.copy(buffer.data(), pos);
  355. //) = buffer.substr(0, pos);
  356. buffer.erase(0, pos);
  357. mangle(STDOUT_FILENO, play);
  358. // ZF_LOGD_MEM(buffer, pos, "mangle buffer %d bytes:", pos);
  359. // mangle(STDOUT_FILENO, buffer, pos);
  360. // memmove(buffer, buffer + pos, size - pos);
  361. // size -= pos;
  362. // } else {
  363. // ZF_LOGV("position of /r/n not found.");
  364. }
  365. // Ok, we failed to find CR+NL. What's the buffer size at?
  366. if (buffer.size() > BSIZE) {
  367. // Ok, there's something going on, and it doesn't look good
  368. // unsure if I want to feed this into the console
  369. // my guess at this point would be zmodem xfer
  370. ZF_LOGI("Buffer %lu bytes, write only...", buffer.size());
  371. // console_receive(&console, buffer);
  372. write(STDOUT_FILENO, buffer.data(), buffer.size());
  373. buffer.clear();
  374. }
  375. } else
  376. break;
  377. }
  378. // read_fd esta atribuido com a entrada padrao?
  379. if (FD_ISSET(STDIN_FILENO, &read_fd)) {
  380. // leia a entrada padrao
  381. char input[BSIZE];
  382. int r = read(STDIN_FILENO, &input, BSIZE);
  383. input[r] = 0;
  384. // e escreva no bc
  385. if (r > 50) {
  386. ZF_LOGI("<< %d bytes", r);
  387. } else {
  388. ZF_LOGI("<< %s", repr(input));
  389. };
  390. write(master, &input, r);
  391. // This is INPUT from the USER
  392. // ZF_LOGI_MEM( input, strlen(input), "<< ");
  393. }
  394. }
  395. // Restore terminal
  396. tcsetattr(1, TCSAFLUSH, &orig1);
  397. ZF_LOGD("exit");
  398. }
  399. return 0;
  400. }