charman.cpp 7.1 KB

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