wordplay.cpp 28 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010
  1. #include "images.h"
  2. #include "lastseen.h"
  3. #include "render.h"
  4. #include "terminal.h"
  5. #include "utils.h"
  6. #include <iomanip>
  7. #include <regex>
  8. #include <sstream>
  9. // #include <string.h>
  10. #include <string>
  11. #include <vector>
  12. #include "zf_log.h"
  13. #include <unistd.h> // write
  14. extern struct console_details console;
  15. #define BSIZE 512
  16. /*
  17. * harry_idle_event(fd)
  18. *
  19. * User is idle, let's let them know we're here. HAHAHA!
  20. *
  21. */
  22. void harry_idle_event(int fd) {
  23. // Make something happen
  24. std::ostringstream buffer;
  25. int r;
  26. // This is no where near finished, BUT!
  27. // Do not put any ^ codes in these -- the strlen() would be wrong.
  28. const char *phrases[] = {"Hahaha", "Snicker, snicker", "Boo!",
  29. "MeOW", "I see U", "Arrooo!",
  30. "Ahh-wooo!", "Aaaooo!"};
  31. static LastSeen last_seen_harry_event(2);
  32. do {
  33. r = randint((sizeof(phrases) / sizeof(char *)));
  34. } while (last_seen_harry_event.seen_before(r));
  35. int color = random() % 15 + 1;
  36. // %02d = std::setfill('0') << std::setw(2) << (int)
  37. buffer << "^S2^C" << std::setfill('0') << std::setw(2) << color << phrases[r]
  38. << "^P2^CR^D" << std::setw(2) << strlen(phrases[r]);
  39. /*
  40. slen = snprintf(buffer, sizeof(buffer), "^S2^C%02d%s^P2^CR^D%02d", color, cp,
  41. (int)strlen(cp));
  42. if (slen >= sizeof(buffer)) {
  43. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  44. buffer[0] = 0;
  45. }
  46. */
  47. std::string str = buffer.str();
  48. ZF_LOGD("harry_event: render(%d, \"%s\")", fd, str.c_str());
  49. render(fd, str);
  50. }
  51. void init_harry() {
  52. // init_have_seen(last_seen_harry_event, MAX_HARRY_EVENT_DUPS);
  53. // ZF_LOGD("init => %d %d", last_seen_harry_event[0],
  54. // last_seen_harry_event[1]);
  55. console_init(&console);
  56. }
  57. #ifdef UNWANTED
  58. regex_t ANSI;
  59. regex_t WORDS;
  60. regex_t WORD;
  61. int init_regex(void) {
  62. int ret;
  63. char ansi[] = "\x1b\[[0-9]+(;[0-9]+)*?[a-zA-Z]";
  64. char words[] = "[a-zA-Z]+( [a-zA-Z]+)+";
  65. char word[] = "[a-zA-Z]+";
  66. char errorbuf[100];
  67. if (ret = regcomp(&ANSI, ansi, REG_EXTENDED | REG_NEWLINE)) {
  68. regerror(ret, &ANSI, errorbuf, sizeof(errorbuf));
  69. ZF_LOGW("Regex %s failed to compile: %s", ansi, errorbuf);
  70. return 0;
  71. };
  72. if (ret = regcomp(&WORDS, words, REG_EXTENDED | REG_NEWLINE)) {
  73. regerror(ret, &WORDS, errorbuf, sizeof(errorbuf));
  74. ZF_LOGW("Regex %s failed to compile: %s", words, errorbuf);
  75. return 0;
  76. };
  77. if (ret = regcomp(&WORD, word, REG_EXTENDED | REG_NEWLINE)) {
  78. regerror(ret, &WORD, errorbuf, sizeof(errorbuf));
  79. ZF_LOGW("Regex %s failed to compile: %s", word, errorbuf);
  80. return 0;
  81. };
  82. return 1;
  83. }
  84. int regmatch(regex_t *preg, const char *string, size_t nmatch,
  85. regmatch_t pmatch[], int eflags) {
  86. // returns number of matches found. (Max nmatch)
  87. int matches = 0;
  88. int offset = 0;
  89. while (matches < nmatch) {
  90. int ret = regexec(preg, string + offset, nmatch - matches, pmatch + matches,
  91. eflags);
  92. if (!ret) {
  93. int current = offset;
  94. offset += pmatch[matches].rm_eo;
  95. pmatch[matches].rm_so += current;
  96. pmatch[matches].rm_eo += current;
  97. matches++;
  98. } else if (ret == REG_NOMATCH) {
  99. break;
  100. } else {
  101. break;
  102. }
  103. }
  104. return matches;
  105. }
  106. #define MAX_MATCH 32
  107. regmatch_t rxmatch[MAX_MATCH];
  108. int rx_match(regex_t *regex, const char *buffer) {
  109. int ret;
  110. ret = regmatch(regex, buffer, MAX_MATCH, rxmatch, 0);
  111. if (0) {
  112. for (int i = 0; i < ret; i++) {
  113. ZF_LOGI("%d : (%d-%d)", i, rxmatch[i].rm_so, rxmatch[i].rm_eo);
  114. }
  115. }
  116. return ret;
  117. }
  118. #endif
  119. /**
  120. * random_activate()
  121. *
  122. * Is a weight (1-10),
  123. * tests if random number is < weight * 10.
  124. *
  125. * So random_activate(9) happens more frequently
  126. * then random_activate(8) or lower.
  127. *
  128. * This probably needs to be fixed.
  129. * We need a better randint(RANGE) code.
  130. */
  131. int random_activate(int w) {
  132. int r = randint(100);
  133. if (r <= (w * 10)) {
  134. return 1;
  135. };
  136. return 0;
  137. }
  138. /*
  139. word_state(): // deprecated
  140. -1 only lower
  141. +1 only upper
  142. 0 mixed
  143. */
  144. int word_state(const char *buffer, int len) {
  145. int p;
  146. int upper = 0;
  147. int lower = 0;
  148. int ret;
  149. float pct;
  150. for (p = 0; p < len; p++) {
  151. char c = buffer[p];
  152. if (isalpha(c)) {
  153. if (isupper(c)) {
  154. upper++;
  155. };
  156. if (islower(c)) {
  157. lower++;
  158. };
  159. }
  160. }
  161. if (upper == lower) {
  162. return 0;
  163. }
  164. if (upper > lower) {
  165. ret = 1;
  166. pct = ((float)lower / (float)upper) * 100.0;
  167. } else {
  168. ret = -1;
  169. pct = ((float)upper / (float)lower) * 100.0;
  170. }
  171. // ZF_LOGD("So far %d with %f %%", ret, pct);
  172. if (pct < 40.0) {
  173. return ret;
  174. }
  175. return 0;
  176. }
  177. /*
  178. Given a buffer and length, mangle away.
  179. toupper, tolower, flipper
  180. */
  181. int word_mangler(std::string &buffer, std::string& word, std::string& text, std::vector<int> &text_offsets, std::pair<int,int> pos_len) {
  182. }
  183. #ifdef UNWANTED
  184. int word_mangler(char *buffer, int len) {
  185. int p;
  186. int count = 0;
  187. int state;
  188. // state = word_state(buffer, len);
  189. // ZF_LOGD("word_state(%.*s) %d", len, buffer, state);
  190. state = randrange(-1, 1);
  191. // TODO: Transposer
  192. for (p = 0; p < len; p++) {
  193. char c = buffer[p];
  194. if (randint(len) == p) {
  195. break;
  196. }
  197. switch (state) {
  198. case -1:
  199. // upper
  200. if (islower(c)) {
  201. count++;
  202. buffer[p] = toupper(c);
  203. }
  204. break;
  205. case 1:
  206. // lower
  207. if (isupper(c)) {
  208. count++;
  209. buffer[p] = tolower(c);
  210. }
  211. break;
  212. case 0:
  213. // flipper
  214. if (islower(c)) {
  215. count++;
  216. buffer[p] = toupper(c);
  217. } else {
  218. if (isupper(c)) {
  219. count++;
  220. buffer[p] = tolower(c);
  221. }
  222. }
  223. break;
  224. }
  225. }
  226. return count;
  227. }
  228. int word_wrangler(char *buffer, int len) {
  229. int p;
  230. int count;
  231. int state;
  232. // state = word_state(buffer, len);
  233. // ZF_LOGD("word_state(%.*s) %d", len, buffer, state);
  234. if (len < 5) {
  235. return 0;
  236. }
  237. p = randint(len - 4) + 2;
  238. for (count = 0; count < 4; count++) {
  239. if (!isalpha(buffer[p + count]))
  240. break;
  241. }
  242. ZF_LOGV_MEM(buffer, len, "wrangler %d len %d:", p, count);
  243. if (count >= 2) {
  244. for (int x = 0; x < count / 2; x++) {
  245. char ch = buffer[p + x];
  246. buffer[p + x] = buffer[p + count - 1 - x];
  247. buffer[p + count - 1 - x] = ch;
  248. }
  249. ZF_LOGV_MEM(buffer, len, "word now:");
  250. return 1;
  251. } else
  252. return 0;
  253. }
  254. int buffer_insert(char *buffer, int len, int max_length, int pos,
  255. const char *insert) {
  256. if (len + strlen(insert) > max_length) {
  257. ZF_LOGD("buffer_insert failed [%s]", repr(insert));
  258. return 0;
  259. }
  260. memmove(buffer + pos + strlen(insert), buffer + pos, len - pos);
  261. strncpy(buffer + pos, insert, strlen(insert));
  262. return 1;
  263. }
  264. #endif
  265. #ifdef UNWANTED
  266. /*
  267. * The buffer that we've been given is much larger now.
  268. *
  269. * We can no longer mangle or insert into the given buffer.
  270. * Why? Because we don't know what is behind it now!
  271. */
  272. int mangle(int fd, const char *buffer, int len) {
  273. int x, i;
  274. int need_render = 0; // changing word case around doesn't need the render
  275. int mangled = 0;
  276. int mangled_chars = 0;
  277. char play[BSIZE * 2]; // The main buffer to send.
  278. const char *cp;
  279. // Make a copy of buffer, since it can no longer be changed
  280. // inserted into.
  281. memcpy(play, buffer, len);
  282. // NEVER reference buffer from this point on!
  283. char work[BSIZE * 2]; // The mess with buffer.
  284. /*
  285. We use the work buffer to blank out the ANSI
  286. before trying to locate words. (So we don't
  287. grab part of the ANSI codes and mangle those!)
  288. */
  289. // Use terminal - clean out ANSI
  290. // ZF_LOGI("mangle:");
  291. ZF_LOGI_MEM(play, len, "Mangle (%u bytes):", len);
  292. // strcpy(work, buffer);
  293. /*
  294. NOTE: We copy the buffer, so we can clear out ANSI codes, etc.
  295. Otherwise we might mess some ANSI up in the manglying
  296. process.
  297. */
  298. /*
  299. Is there a way to track -- what I've inserted, and make it exempt from
  300. other modifications / mangler, wrangler?
  301. */
  302. /*
  303. (random) Look for ANSI CLS and:
  304. display random spooky texts around, with delays ... then CLS.
  305. display ANSI graphic file, with delays ... then CLS
  306. */
  307. const char *ANSI_CLS = "\x1b[2J";
  308. cp = strnstr(play, len, ANSI_CLS); // strstr(buffer, ANSI_CLS);
  309. if (cp != NULL) {
  310. static int ANSI_CLS_count = 0; // count the number we've seen
  311. ZF_LOGI("seen: ANSI_CLS");
  312. ANSI_CLS_count++;
  313. // Don't activate on the very first CLS. Too soon, don't screw up the ANSI
  314. // detection.
  315. if (ANSI_CLS_count > 1) {
  316. // Ok, figure out the restore color, just in case
  317. struct console_details temp_console;
  318. // Make exact copy of our current console state.
  319. memcpy(&temp_console, &console, sizeof(console));
  320. // Play the buffer into the console
  321. console_receive(&temp_console, play, cp - play);
  322. char restore_color[30]; // ansi color
  323. strcpy(restore_color, color_restore(&temp_console));
  324. if (random_activate(3)) {
  325. char display[100] = "";
  326. int slen;
  327. int needs_cls = 0;
  328. struct image {
  329. const char **lines;
  330. int size;
  331. int cls;
  332. int width; // height = size
  333. } images[] = {{ghost, sizeof(ghost) / sizeof(char *), 1, 0},
  334. {ghead, sizeof(ghead) / sizeof(char *), 1, 0},
  335. {wolf, sizeof(wolf) / sizeof(char *), 1, 0},
  336. {panther, sizeof(panther) / sizeof(char *), 1, 0},
  337. {bat, sizeof(bat) / sizeof(char *), 1, 0},
  338. {icu, sizeof(icu) / sizeof(char *), 0, 20},
  339. {skull, sizeof(skull) / sizeof(char *), 0, 19},
  340. {skullblink, sizeof(skullblink) / sizeof(char *), 0, 19}};
  341. static LastSeen last_files(2);
  342. int r;
  343. do {
  344. r = randint((sizeof(images) / sizeof(image)));
  345. } while (last_files.seen_before(r));
  346. char fgoto[32];
  347. if (!images[r].cls) {
  348. int x = 0, y = 0;
  349. x = randint(79 - images[r].width) + 1;
  350. y = randint(24 - images[r].size) + 1;
  351. int slen;
  352. // render image, home cursor
  353. slen = snprintf(fgoto, sizeof(fgoto), "^f%02d%02d\x1b[1;1H", x, y);
  354. if (slen >= sizeof(fgoto)) {
  355. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(fgoto));
  356. fgoto[0] = 0;
  357. }
  358. } else {
  359. strcpy(fgoto, "^F");
  360. }
  361. // (2); // (sizeof(possibles) / sizeof(file_need)) - 1);
  362. needs_cls = images[r].cls;
  363. // I get what's happening. Mystic moves cursor to home, CLS, cursor
  364. // home. When we get here, we're ALWAYS at the top of the screen...
  365. // Hence our bat isn't displayed at the end of the screen.
  366. // This is before the actual CLS, so we CLS before displaying our files.
  367. // I tried a ^P2 before doing this .. but I'd rather have the picture up
  368. // right away I think.
  369. // Ok, yes, there's no filename being sent. :P
  370. render_image(images[r].lines, images[r].size);
  371. slen = snprintf(display, sizeof(display), "%s%s%s^P3",
  372. needs_cls ? "\x1b[2J" : "", fgoto, restore_color);
  373. if (slen >= sizeof(display)) {
  374. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(display));
  375. display[0] = 0;
  376. }
  377. ZF_LOGI("mangle(ANSI_CLS): %d file inserted %s", r, repr(display));
  378. // Move the buffer so there's room for the display string.
  379. if (buffer_insert(play, len, sizeof(play), cp - play, display)) {
  380. len += strlen(display);
  381. // if (string_insert(buffer, 1024, cp - buffer, display)) {
  382. ZF_LOGI_MEM(play, len, "mangle(ANSI_CLS) (%u bytes):", len);
  383. // ZF_LOGI("mangle(ANSI_CLS):");
  384. // ZF_LOGI_REPR(buffer);
  385. // ZF_LOGI("mangle(ANSI_CLS): [%s]", repr(buffer));
  386. need_render = 1;
  387. /*
  388. Copy the new buffer over, but hide our "render" code
  389. from the remaining mangler steps.
  390. */
  391. memcpy(work, play, len);
  392. // strcpy(work, buffer);
  393. i = cp - play;
  394. // find offset into "buffer"
  395. // apply to work.
  396. memset(work + i, ' ', strlen(display));
  397. } else {
  398. ZF_LOGD("insert failed [%s].", repr(display));
  399. }
  400. } else {
  401. if (random_activate(4)) {
  402. int r;
  403. char display[100] = "";
  404. int slen;
  405. /*
  406. Interesting note here:
  407. "Anyone there" qualifies as a valid WORDS regex match.
  408. It is possible that it can get mangled/wrangled!
  409. */
  410. const char *phrasing[] = {
  411. "^R1Haha^P1ha^P1ha", "Poof!", "Got U", "Anyone there?",
  412. "^R1Knock, ^P1Knock",
  413. /*
  414. This picks random color and position -- then
  415. homes cursor and changes to another color. (This can be seen.)
  416. */
  417. "^G0101^C07^S9Segmentation fault (core dumped)^P2"};
  418. static LastSeen last_phrasing(2);
  419. ZF_LOGI("mangle(ANSI_CLS)");
  420. // sprintf( display, "^P2...");
  421. // This string actually screws up ANSI detection (takes too long)
  422. // strcpy(display, "^P2^S501234567890^P1abcdef^P2g^P3h^P4i^S0^P2");
  423. // strcpy(display, "^P2^S301234^P15^S0^P2");
  424. // Add in random text, plus color!
  425. do {
  426. r = randint(sizeof(phrasing) / sizeof(char *));
  427. } while (last_phrasing.seen_before(r));
  428. int color = random() % 15 + 1;
  429. int x = random() % 30 + 1;
  430. int y = random() % 15 + 1;
  431. /*
  432. Don't have it pause there before moving the cursor.
  433. Move the cursor, get the color changed, THEN pause.
  434. Then act all crazy.
  435. NOTE: Make sure if you use any ^R Render effects, turn them off
  436. before trying to display the restore_color. :P ^R0 Also, make
  437. sure you re-home the cursor ^G0101 because that's where they are
  438. expecting the cursor to be! (At least it's how Mystic does it.)
  439. HOME, CLS, HOME, ... Not sure what others do there. We'll see.
  440. */
  441. slen = snprintf(display, sizeof(display),
  442. "^G%02d%02d^S3^C%02d^P1%s^S0^R0%s^P1^G0101", x, y,
  443. color, phrasing[r], restore_color);
  444. if (slen >= sizeof(display)) {
  445. ZF_LOGE("snprintf %d > size %d (Phrase: %d, %s)", slen,
  446. (int)sizeof(display), r, phrasing[r]);
  447. display[0] = 0;
  448. }
  449. // sprintf(display, "^P1^S3^C%02d%s^S0^R0%s^P1", color, phrasing[r],
  450. // restore_color);
  451. // Added debug statement so we can identify what was sent... color,
  452. // number picked and what that is
  453. ZF_LOGD("mangle(ANSI_CLS): Inserted color=%02d r=%d phrase='%s'",
  454. color, r, phrasing[r]);
  455. ZF_LOGI_MEM(play, len, "mangle(ANSI_CLS) :");
  456. // Move the buffer so there's room for the display string.
  457. if (buffer_insert(play, len, sizeof(play), cp - play, display)) {
  458. len += strlen(display);
  459. // if (string_insert(buffer, BSIZE * 4, cp - buffer, display)) {
  460. ZF_LOGI_MEM(play, len, "mangle(ANSI_CLS) + :");
  461. // ZF_LOGI("mangle(ANSI_CLS):");
  462. // ZF_LOGI_REPR(buffer);
  463. need_render = 1;
  464. /*
  465. Copy the new buffer over, but hide our "render" code
  466. from the remaining mangler steps.
  467. */
  468. memcpy(work, play, len);
  469. // strcpy(work, buffer);
  470. i = cp - play;
  471. // find offset into "buffer"
  472. // apply to work.
  473. memset(work + i, ' ', strlen(display));
  474. } else {
  475. ZF_LOGD("insert failed [%s].", repr(display));
  476. }
  477. }
  478. }
  479. }
  480. }
  481. memcpy(work, play, len);
  482. // strcpy(work, buffer); // sure.
  483. // NOTE: This is NOT aware of my ^TRIGGERS, so they will show up
  484. // as valid things to mangle in work. (Keep this in mind).
  485. const char replace_with = ' ';
  486. for (x = 0; x < len; x++) {
  487. termchar tc = console_char(&console, play[x]);
  488. int ansi = tc.in_ansi;
  489. if (ansi) {
  490. work[x] = replace_with;
  491. if (tc.ansi != START) {
  492. ZF_LOGD("ANSI type %d at %d", tc.ansi, x);
  493. }
  494. }
  495. // fixup "work" so it's a valid C string
  496. if (buffer[x] == 0) {
  497. work[x] = replace_with;
  498. }
  499. }
  500. // fixup "work" buffer so it's a valid c string
  501. // (required for regex to work.)
  502. work[len] = 0;
  503. ZF_LOGV_MEM(work, len, "Work now:");
  504. /*
  505. (random) Locate words (in work), and possibly flip them around.
  506. Transpose words. Transpose case. Transpose letters.
  507. Ok, what would be interesting, is if we could find
  508. W\x1[0;34mORDS with color changes in them, and work with them.
  509. without screwing up the color changes, of course. :P
  510. Example:
  511. Y\x1b[0;1mes \x1b[0;1;34m\x1b[0;1;34;44m N\x1b[0;1;44mo
  512. Yes No
  513. ^ This would be a job for a crazy regex.
  514. I'd have to map the characters to positions in the buffer. :S
  515. I'd want mangle and wrangle to work.
  516. The Message menu -- doesn't hardly get mangled at all (at least on
  517. my test site). Because all of the color changes break up the
  518. words in the menu.
  519. */
  520. x = rx_match(&WORDS, work);
  521. ZF_LOGD("found %d word groups", x);
  522. if (x > 0) {
  523. for (i = 0; i < x; i++) {
  524. // Yes! Be random!
  525. if (random_activate(8)) {
  526. int c = word_mangler(play + rxmatch[i].rm_so,
  527. rxmatch[i].rm_eo - rxmatch[i].rm_so);
  528. if (c) {
  529. mangled++;
  530. mangled_chars += c;
  531. }
  532. }
  533. if (random_activate(4)) {
  534. word_wrangler(play + rxmatch[i].rm_so,
  535. rxmatch[i].rm_eo - rxmatch[i].rm_so);
  536. }
  537. }
  538. }
  539. /*
  540. (random) Locate single words, and transpose words.
  541. Transpose letters.
  542. */
  543. /*
  544. (random) Display up to certain point. Delay.
  545. Print some characters slowly. Delay.
  546. */
  547. if (mangled)
  548. ZF_LOGI("Mangled %d word, %d chars (render %d)", mangled, mangled_chars,
  549. need_render);
  550. if (need_render) {
  551. ZF_LOGD_MEM(play, len, "Ready to render:");
  552. // ZF_LOGD("HH %d : (%d) %s", need_render, (int)strlen(buffer),
  553. // repr(buffer));
  554. }
  555. if (need_render) {
  556. render(fd, play, len);
  557. } else {
  558. write(fd, play, len);
  559. };
  560. return need_render && mangled;
  561. }
  562. #endif
  563. /*
  564. int harry_happens(time_t *last_event, int wakeup) {
  565. time_t now = time(NULL);
  566. int elapsed = now - *last_event;
  567. if (elapsed > wakeup) {
  568. // Ok! It's been too long since we've done something.
  569. *last_event = now;
  570. return 1;
  571. }
  572. return 0;
  573. }
  574. */
  575. /*
  576. * find_words()
  577. *
  578. * This uses regex to look for words where a word is defined as one or more letter.
  579. *
  580. * We do throw out words with a len of 3. (That would be "A B". We don't want that.)
  581. * We return a vector of pairs (position, length), and the number of matches.
  582. */
  583. int find_words(std::string &text, std::vector<std::pair<int, int>> &words) {
  584. words.clear();
  585. // Yes I want & in there so BZ&BZ BBS matches.
  586. std::regex words_re("[a-zA-Z&]+( [a-zA-Z&]+)+");
  587. int count = 0;
  588. for (auto it = std::sregex_iterator(text.begin(), text.end(), words_re);
  589. it != std::sregex_iterator(); ++it) {
  590. if (it->length() > 3) {
  591. words.push_back(std::make_pair(it->position(), it->length()));
  592. count++;
  593. }
  594. }
  595. return count;
  596. }
  597. int mangle_clrscr(std::string &buffer, std::string &work, size_t pos) {
  598. static int ANSI_CLS_count = 0;
  599. ZF_LOGI("seen: ANSI_CLS");
  600. ANSI_CLS_count++;
  601. if (ANSI_CLS_count > 1) {
  602. // get the restore color value
  603. struct console_details temp_console;
  604. memcpy(&temp_console, &console, sizeof(console));
  605. console_receive(&temp_console, buffer.substr(0, pos));
  606. std::string restore_color;
  607. restore_color.assign(color_restore(&temp_console));
  608. if (random_activate(3)) {
  609. std::ostringstream display;
  610. int needs_cls = 0;
  611. struct image {
  612. const char **lines;
  613. int size;
  614. int cls;
  615. int width; // height = size
  616. } images[] = {{ghost, sizeof(ghost) / sizeof(char *), 1, 0},
  617. {ghead, sizeof(ghead) / sizeof(char *), 1, 0},
  618. {wolf, sizeof(wolf) / sizeof(char *), 1, 0},
  619. {panther, sizeof(panther) / sizeof(char *), 1, 0},
  620. {bat, sizeof(bat) / sizeof(char *), 1, 0},
  621. {icu, sizeof(icu) / sizeof(char *), 0, 20},
  622. {skull, sizeof(skull) / sizeof(char *), 0, 19},
  623. {skullblink, sizeof(skullblink) / sizeof(char *), 0, 19}};
  624. static LastSeen last_files(2);
  625. int r;
  626. do {
  627. r = randint((sizeof(images) / sizeof(image)));
  628. } while (last_files.seen_before(r));
  629. std::string fgoto;
  630. if (!images[r].cls) {
  631. int x = 0, y = 0;
  632. x = randint(79 - images[r].width) + 1;
  633. y = randint(24 - images[r].size) + 1;
  634. display << "^f" << std::setfill('0') << std::setw(2) << x
  635. << std::setw(2) << y << "\x1b[1;1H";
  636. fgoto = display.str();
  637. // reset display
  638. display.str(std::string());
  639. display.clear();
  640. // render image, home cursor
  641. // slen = snprintf(fgoto, sizeof(fgoto), "^f%02d%02d\x1b[1;1H", x, y);
  642. } else {
  643. fgoto.assign("^F");
  644. }
  645. needs_cls = images[r].cls;
  646. // I get what's happening. Mystic moves cursor to home, CLS, cursor
  647. // home. When we get here, we're ALWAYS at the top of the screen...
  648. // Hence our bat isn't displayed at the end of the screen.
  649. // This is before the actual CLS, so we CLS before displaying our files.
  650. // I tried a ^P2 before doing this .. but I'd rather have the picture up
  651. // right away I think.
  652. // Ok, yes, there's no filename being sent. :P
  653. render_image(images[r].lines, images[r].size);
  654. display << (needs_cls ? "\x1b[2J" : "") << fgoto << restore_color
  655. << "^P3";
  656. // slen = snprintf(display, sizeof(display), "%s%s%s^P3",
  657. // needs_cls ? "\x1b[2J" : "", fgoto, restore_color);
  658. std::string display_output = display.str();
  659. ZF_LOGI("mangle(ANSI_CLS): %d file inserted %s", r,
  660. repr(display_output.c_str()));
  661. // Move the buffer so there's room for the display string.
  662. buffer.insert(pos, display_output);
  663. work.insert(pos, std::string(display_output.size(), ' '));
  664. return 1; // need_render = 1;
  665. } else {
  666. if (random_activate(4)) {
  667. int r;
  668. std::ostringstream display;
  669. const char *phrasing[] = {
  670. "^R1Haha^P1ha^P1ha", "Poof!", "Got U", "Anyone there?",
  671. "^R1Knock, ^P1Knock",
  672. /*
  673. This picks random color and position -- then
  674. homes cursor and changes to another color. (This can be seen.)
  675. */
  676. "^G0101^C07^S9Segmentation fault (core dumped)^P2"};
  677. static LastSeen last_phrasing(2);
  678. ZF_LOGI("mangle(ANSI_CLS)");
  679. do {
  680. r = randint(sizeof(phrasing) / sizeof(char *));
  681. } while (last_phrasing.seen_before(r));
  682. int color = randint(14) + 1;
  683. int x = randint(30) + 1;
  684. int y = randint(15) + 1;
  685. /*
  686. Don't have it pause there before moving the cursor.
  687. Move the cursor, get the color changed, THEN pause.
  688. Then act all crazy.
  689. NOTE: Make sure if you use any ^R Render effects, turn them off
  690. before trying to display the restore_color. :P ^R0 Also, make
  691. sure you re-home the cursor ^G0101 because that's where they are
  692. expecting the cursor to be! (At least it's how Mystic does it.)
  693. HOME, CLS, HOME, ... Not sure what others do there. We'll see.
  694. */
  695. if (strncmp(phrasing[r], "^G", 2) == 0) {
  696. display << "^S3^P1" << phrasing[r] << "^S0^R0" << restore_color
  697. << "^P1^G0101";
  698. // This starts with a GOTO, so don't use our random position
  699. // slen = snprintf(display, sizeof(display),
  700. // "^S3^P1%s^S0^R0%s^P1^G0101",
  701. // phrasing[r], restore_color);
  702. } else {
  703. display << "^G" << std::setw(2) << std::setfill('0') << x
  704. << std::setw(2) << y << "^S3^C" << std::setw(2) << color
  705. << "^P1" << phrasing[r] << "^S0^R0" << restore_color
  706. << "^P1^G0101";
  707. // slen = snprintf(display, sizeof(display),
  708. // "^G%02d%02d^S3^C%02d^P1%s^S0^R0%s^P1^G0101", x, y,
  709. // color, phrasing[r], restore_color);
  710. };
  711. std::string display_output = display.str();
  712. // sprintf(display, "^P1^S3^C%02d%s^S0^R0%s^P1", color, phrasing[r],
  713. // restore_color);
  714. // Added debug statement so we can identify what was sent... color,
  715. // number picked and what that is
  716. ZF_LOGD("mangle(ANSI_CLS): Inserted color=%02d r=%d phrase='%s'", color,
  717. r, phrasing[r]);
  718. ZF_LOGI("mangle(ANSI_CLS): %d %s", r, repr(display_output.c_str()));
  719. // Move the buffer so there's room for the display string.
  720. buffer.insert(pos, display_output);
  721. work.insert(pos, std::string(display_output.size(), ' '));
  722. return 1; // need_render = 1;
  723. }
  724. }
  725. }
  726. }
  727. int mangle(int fd, std::string &buffer) {
  728. // a simple default for now.
  729. ZF_LOGV_MEM(buffer.data(), buffer.size(), "mangle(%d): %lu bytes", fd,
  730. buffer.size());
  731. int need_render = 0;
  732. int mangled = 0;
  733. int mangled_chars = 0;
  734. static std::string work;
  735. static size_t work_size = 0;
  736. work.assign(buffer);
  737. // This should allow us to monitor any memory allocations
  738. if (work.capacity() != work_size) {
  739. ZF_LOGD("work cap %lu -> %lu", work_size, work.capacity());
  740. work_size = work.capacity();
  741. }
  742. const char *ANSI_CLS = "\x1b[2J";
  743. size_t pos = buffer.find(ANSI_CLS);
  744. if (pos != std::string::npos) {
  745. if (mangle_clrscr(buffer, work, pos)) {
  746. need_render = 1;
  747. }
  748. }
  749. // Ok, maybe the work string was a bad idea?
  750. static std::string text;
  751. static std::vector<int> text_offsets;
  752. size_t stri;
  753. text.clear();
  754. text_offsets.clear();
  755. for (stri = 0; stri < buffer.size(); ++stri) {
  756. termchar tc = console_char(&console, work[stri]);
  757. if (tc.in_ansi) {
  758. if (tc.ansi != START) {
  759. // Ok, this is something. What is it?
  760. // ZF_LOGV("ANSI type %d at %lu", tc.ansi, stri);
  761. switch (tc.ansi) {
  762. case CURSOR:
  763. case CLEAR:
  764. case OTHER:
  765. text.append(1, '.');
  766. text_offsets.push_back(-1);
  767. break;
  768. case COLOR:
  769. // text.append(1, ' ');
  770. // text_offsets.push_back(-1);
  771. break;
  772. }
  773. }
  774. } else {
  775. // These should never get out of sync ...
  776. if (text.size() != text_offsets.size()) {
  777. ZF_LOGE("Error: text != text_offsets %lu != %lu", text.size(),
  778. text_offsets.size());
  779. }
  780. text.append(1, work[stri]);
  781. text_offsets.push_back(stri);
  782. }
  783. }
  784. // The current situation:
  785. ZF_LOGD_MEM(buffer.data(), buffer.size(), "Buffer:");
  786. ZF_LOGD_MEM(work.data(), work.size(), "Work:");
  787. ZF_LOGD_MEM(text.data(), text.size(), "Text Buffer:");
  788. // output vector information
  789. std::ostringstream oss;
  790. int comma = 0;
  791. for (auto it = std::begin(text_offsets); it != std::end(text_offsets); ++it) {
  792. if (comma) {
  793. oss << ", ";
  794. };
  795. comma++;
  796. oss << *it;
  797. if (comma == 30) {
  798. std::string temp_output = oss.str();
  799. ZF_LOGD("Vector: %s", temp_output.c_str());
  800. // reset ostringstream
  801. oss.str(std::string());
  802. oss.clear();
  803. comma = 0;
  804. }
  805. }
  806. std::string vector_output = oss.str();
  807. ZF_LOGD("Vector: %s", vector_output.c_str());
  808. // find matches
  809. std::vector<std::pair<int, int>> words;
  810. int found = find_words(text, words);
  811. ZF_LOGD("Found %d word groups in text.", found);
  812. if (found > 0) {
  813. for (int i = 0; i < found; i++) {
  814. std::pair<int, int> pos_len = words[i];
  815. // Yes! Be random!
  816. if (random_activate(8)) {
  817. int c = word_mangler(buffer, work, text, text_offsets, pos_len);
  818. /*
  819. int c = word_mangler(play + rxmatch[i].rm_so,
  820. rxmatch[i].rm_eo - rxmatch[i].rm_so);*/
  821. if (c) {
  822. mangled++;
  823. mangled_chars += c;
  824. }
  825. }
  826. if (random_activate(4)) {
  827. /*
  828. word_wrangler(play + rxmatch[i].rm_so,
  829. rxmatch[i].rm_eo - rxmatch[i].rm_so);
  830. */
  831. }
  832. }
  833. }
  834. if (need_render) {
  835. render(fd, buffer);
  836. } else {
  837. write(fd, buffer.data(), buffer.size());
  838. }
  839. return need_render;
  840. }