charman.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. #include <ctype.h>
  2. #include <regex>
  3. #include <string>
  4. #include <vector>
  5. #include "charman.h"
  6. #include "utils.h"
  7. #include "zf_log.h"
  8. void CharMan::validate(void) {
  9. bool valid = true;
  10. for (int x = 0; x < text_offsets.size(); ++x) {
  11. int offset = text_offsets[x];
  12. if (offset >= 0) {
  13. if (text[x] != work[offset]) {
  14. ZF_LOGE("validate: %d off %d [%c != %c]", x, offset, text[x],
  15. work[offset]);
  16. valid = false;
  17. }
  18. if ((text[x] != ' ') && (text[x] != buffer[offset])) {
  19. ZF_LOGE("validate: %d off %d [%c != %c]", x, offset, text[x],
  20. buffer[offset]);
  21. valid = false;
  22. }
  23. }
  24. }
  25. if (!valid) {
  26. ZF_LOGE("* NOT VALID* Somethings hosed.");
  27. diagnostics();
  28. }
  29. }
  30. void CharMan::diagnostics(void) {
  31. ZF_LOGV_MEM(buffer.data(), buffer.size(), "Buffer:");
  32. ZF_LOGV_MEM(work.data(), work.size(), "Work:");
  33. ZF_LOGV_MEM(text.data(), text.size(), "Text Buffer:");
  34. std::ostringstream oss;
  35. int comma = 0;
  36. for (auto it = std::begin(text_offsets); it != std::end(text_offsets); ++it) {
  37. if (comma) {
  38. oss << ", ";
  39. };
  40. comma++;
  41. oss << *it;
  42. if (comma == 30) {
  43. std::string temp_output = oss.str();
  44. ZF_LOGV("Vector: %s", temp_output.c_str());
  45. // reset ostringstream
  46. oss.str(std::string());
  47. oss.clear();
  48. comma = 0;
  49. }
  50. }
  51. std::string vector_output = oss.str();
  52. ZF_LOGV("Vector: %s", vector_output.c_str());
  53. // reset oss (if we need it)
  54. oss.str(std::string());
  55. oss.clear();
  56. }
  57. void CharMan::regular_expressions(void) {
  58. static std::regex words("[a-zA-Z'-]+([ ]{1,2}[a-zA-Z&'-]+)+");
  59. // I need position and length.
  60. pos_len.clear();
  61. for (auto it =
  62. std::sregex_iterator(this->text.begin(), this->text.end(), words);
  63. it != std::sregex_iterator(); ++it) {
  64. pos_len.push_back(std::make_pair(it->position(), it->length()));
  65. int pos = it->position(), len = it->length();
  66. ZF_LOGD("pos %d len %d (%s)", pos, len,
  67. this->text.substr(pos, len).c_str());
  68. }
  69. }
  70. char CharMan::get(int pos) { return this->text[pos]; }
  71. void CharMan::set(int pos, char ch) {
  72. this->text[pos] = ch;
  73. int idx = this->text_offsets[pos];
  74. if (idx >= 0) {
  75. this->buffer[idx] = ch;
  76. this->work[idx] = ch;
  77. }
  78. }
  79. void CharMan::insert(int pos, std::string str) {
  80. int len = str.size();
  81. int idx = this->text_offsets[pos];
  82. ZF_LOGE("insert( POS %d, LEN %d, IDX %d, %s)", pos, len, idx, str.c_str());
  83. diagnostics();
  84. std::string blank(len, ' ');
  85. // Don't insert into text.
  86. // Insert blank into work.
  87. // Update indexes >= idx
  88. if (idx >= 0) {
  89. this->buffer.insert(idx, str);
  90. this->work.insert(idx, blank);
  91. // UPDATE indexes!
  92. for (auto it = std::begin(this->text_offsets);
  93. it != std::end(this->text_offsets); ++it) {
  94. if (*it >= idx) {
  95. *it += len;
  96. }
  97. }
  98. ZF_LOGE("Indexes updated... check your work");
  99. diagnostics();
  100. }
  101. }
  102. int CharMan::word_mangler(std::pair<int, int> pos_len) {
  103. int pos = pos_len.first;
  104. int state = randrange(-1, 1);
  105. int count = 0;
  106. for (int p = 0; p < pos_len.second; ++p) {
  107. char c = this->get(pos + p);
  108. if (randint(pos_len.second) == p)
  109. break;
  110. switch (state) {
  111. case -1:
  112. if (islower(c)) {
  113. count++;
  114. this->set(pos + p, toupper(c));
  115. }
  116. break;
  117. case 0:
  118. if (islower(c)) {
  119. count++;
  120. this->set(pos + p, toupper(c));
  121. } else {
  122. if (isupper(c)) {
  123. count++;
  124. this->set(pos + p, tolower(c));
  125. }
  126. }
  127. break;
  128. case 1:
  129. if (isupper(c)) {
  130. count++;
  131. this->set(pos + p, tolower(c));
  132. }
  133. break;
  134. }
  135. }
  136. return count;
  137. }
  138. // What is the max number of characters I should wrangle?
  139. #define MAX_TRANSPOSE 2
  140. int CharMan::word_wrangler(std::pair<int, int> pos_len) {
  141. int count = 0;
  142. int p;
  143. int len;
  144. if (pos_len.second < 4)
  145. return 0;
  146. p = randint(pos_len.second - 4) + 2;
  147. for (len = 0; len < MAX_TRANSPOSE; ++len) {
  148. if (!isalpha(this->get(pos_len.first + p + len)))
  149. break;
  150. }
  151. ZF_LOGD("Wrangler: %d, %d", p, len);
  152. if (len >= 2) {
  153. for (int x = 0; x < len / 2; x++) {
  154. char ch = this->get(pos_len.first + p + x);
  155. this->set(pos_len.first + p + x,
  156. this->get(pos_len.first + p + len - 1 - x));
  157. this->set(pos_len.first + p + len - 1 - x, ch);
  158. }
  159. count++;
  160. }
  161. return count;
  162. }
  163. /*
  164. Display up to certain point.
  165. Print some characters slowly. Delay.
  166. */
  167. int CharMan::word_tangler(std::pair<int, int> pos_len) {
  168. int count = 0;
  169. int p;
  170. int len;
  171. std::string part = this->text.substr(pos_len.first, pos_len.second);
  172. ZF_LOGE("tangler [%s]", logrepr(part.c_str()));
  173. if (pos_len.second < 4)
  174. return 0;
  175. /* p = randint(pos_len.second - 4);
  176. len = randint(pos_len.second - p); */
  177. p = pos_len.first;
  178. len = pos_len.second;
  179. if (len >= 2) {
  180. ZF_LOGD("Tangler: %d, %d", p, len);
  181. this->validate();
  182. std::ostringstream buffer;
  183. std::string tangle;
  184. int r = 1; // randint(2) + 1;
  185. buffer << "^P1^R" << r;
  186. r = randint(4) + 1;
  187. buffer << "^S" << r;
  188. tangle = buffer.str();
  189. std::string reset = "^R0^S0";
  190. this->insert(p + len, reset);
  191. this->validate();
  192. this->insert(p, tangle);
  193. this->validate();
  194. return 1;
  195. }
  196. return 0;
  197. }
  198. CharMan::CharMan(std::string &buffer, std::string &work, std::string &text,
  199. std::vector<int> &text_offsets)
  200. : buffer(buffer), work(work), text(text), text_offsets(text_offsets) {
  201. /*
  202. this->buffer = buffer;
  203. this->work = work;
  204. this->text = text;
  205. this->text_offsets = text_offsets;
  206. */
  207. this->mangle_count = 0;
  208. this->mangle_chars = 0;
  209. this->need_render = 0;
  210. this->level = harry_level();
  211. if (!this->level)
  212. return;
  213. // validate();
  214. regular_expressions();
  215. ZF_LOGD("Found %d word groups", (int)pos_len.size());
  216. if (pos_len.size() > 0) {
  217. for (int i = 0; i < pos_len.size(); ++i) {
  218. int active = 0;
  219. if (random_activate(level * 2)) { // 8
  220. int c = word_mangler(pos_len[i]);
  221. if (c) {
  222. active = 1;
  223. this->mangle_count++;
  224. this->mangle_chars += c;
  225. }
  226. }
  227. if (random_activate(level)) { // 4
  228. if (word_wrangler(pos_len[i])) {
  229. this->mangle_count++;
  230. active = 1;
  231. }
  232. }
  233. if (!active && random_activate(level)) {
  234. if (word_tangler(pos_len[i])) {
  235. this->need_render = 1;
  236. }
  237. }
  238. }
  239. }
  240. };
  241. CharMan::~CharMan() {
  242. ZF_LOGD("~CharMan");
  243. // validate();
  244. }