charman.cpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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. }
  28. }
  29. void CharMan::regular_expressions(void) {
  30. static std::regex words("[a-zA-Z'-]+([ ]{1,2}[a-zA-Z&'-]+)+");
  31. // I need position and length.
  32. for (auto it =
  33. std::sregex_iterator(this->text.begin(), this->text.end(), words);
  34. it != std::sregex_iterator(); ++it) {
  35. pos_len.push_back(std::make_pair(it->position(), it->length()));
  36. int pos = it->position(), len = it->length();
  37. ZF_LOGD("pos %d len %d (%s)", pos, len,
  38. this->text.substr(pos, len).c_str());
  39. }
  40. }
  41. char CharMan::get(int pos) { return this->text[pos]; }
  42. void CharMan::set(int pos, char ch) {
  43. this->text[pos] = ch;
  44. int idx = this->text_offsets[pos];
  45. if (idx >= 0) {
  46. this->buffer[idx] = ch;
  47. this->work[idx] = ch;
  48. }
  49. }
  50. int CharMan::word_mangler(std::pair<int, int> pos_len) {
  51. int pos = pos_len.first;
  52. int state = randrange(-1, 1);
  53. int count = 0;
  54. for (int p = 0; p < pos_len.second; ++p) {
  55. char c = this->get(pos + p);
  56. if (randint(pos_len.second) == p)
  57. break;
  58. switch (state) {
  59. case -1:
  60. if (islower(c)) {
  61. count++;
  62. this->set(pos + p, toupper(c));
  63. }
  64. break;
  65. case 0:
  66. if (islower(c)) {
  67. count++;
  68. this->set(pos + p, toupper(c));
  69. } else {
  70. if (isupper(c)) {
  71. count++;
  72. this->set(pos + p, tolower(c));
  73. }
  74. }
  75. break;
  76. case 1:
  77. if (isupper(c)) {
  78. count++;
  79. this->set(pos + p, tolower(c));
  80. }
  81. break;
  82. }
  83. }
  84. return count;
  85. }
  86. // What is the max number of characters I should wrangle?
  87. #define MAX_TRANSPOSE 2
  88. int CharMan::word_wrangler(std::pair<int, int> pos_len) {
  89. int count = 0;
  90. int p;
  91. int len;
  92. if (pos_len.second < 4)
  93. return 0;
  94. p = randint(pos_len.second - 4) + 2;
  95. for (len = 0; len < MAX_TRANSPOSE; ++len) {
  96. if (!isalpha(this->get(pos_len.first + p + len)))
  97. break;
  98. }
  99. ZF_LOGD("Wrangler: %d, %d", p, len);
  100. if (len >= 2) {
  101. for (int x = 0; x < len / 2; x++) {
  102. char ch = this->get(pos_len.first + p + x);
  103. this->set(pos_len.first + p + x,
  104. this->get(pos_len.first + p + len - 1 - x));
  105. this->set(pos_len.first + p + len - 1 - x, ch);
  106. }
  107. count++;
  108. }
  109. return count;
  110. }
  111. CharMan::CharMan(std::string &buffer, std::string &work, std::string &text,
  112. std::vector<int> &text_offsets)
  113. : buffer(buffer), work(work), text(text), text_offsets(text_offsets) {
  114. /*
  115. this->buffer = buffer;
  116. this->work = work;
  117. this->text = text;
  118. this->text_offsets = text_offsets;
  119. */
  120. this->mangle_count = 0;
  121. this->mangle_chars = 0;
  122. this->level = harry_level();
  123. if (!this->level)
  124. return;
  125. // validate();
  126. regular_expressions();
  127. ZF_LOGD("Found %d word groups", (int)pos_len.size());
  128. if (pos_len.size() > 0) {
  129. for (int i = 0; i < pos_len.size(); ++i) {
  130. if (random_activate(8)) {
  131. int c = word_mangler(pos_len[i]);
  132. if (c) {
  133. this->mangle_count++;
  134. this->mangle_chars += c;
  135. }
  136. }
  137. if (random_activate(4)) {
  138. if (word_wrangler(pos_len[i]))
  139. this->mangle_count++;
  140. }
  141. }
  142. }
  143. /*
  144. Display up to certain point.
  145. Print some characters slowly. Delay.
  146. ** This would require "need_render" **
  147. */
  148. };
  149. CharMan::~CharMan() {
  150. ZF_LOGD("~CharMan");
  151. // validate();
  152. }