cmParsePHPCoverage.cxx 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. #include "cmParsePHPCoverage.h"
  2. #include "cmSystemTools.h"
  3. #include <cmsys/Directory.hxx>
  4. #include <cmsys/FStream.hxx>
  5. /*
  6. To setup coverage for php.
  7. - edit php.ini to add auto prepend and append php files from phpunit
  8. auto_prepend_file =
  9. auto_append_file =
  10. - run the tests
  11. - run this program on all the files in c:/tmp
  12. */
  13. cmParsePHPCoverage::cmParsePHPCoverage(cmCTestCoverageHandlerContainer& cont,
  14. cmCTest* ctest)
  15. :Coverage(cont), CTest(ctest)
  16. {
  17. }
  18. bool cmParsePHPCoverage::ReadUntil(std::istream& in, char until)
  19. {
  20. char c = 0;
  21. while(in.get(c) && c != until)
  22. {
  23. }
  24. if(c != until)
  25. {
  26. return false;
  27. }
  28. return true;
  29. }
  30. bool cmParsePHPCoverage::ReadCoverageArray(std::istream& in,
  31. std::string const& fileName)
  32. {
  33. cmCTestCoverageHandlerContainer::SingleFileCoverageVector& coverageVector
  34. = this->Coverage.TotalCoverage[fileName];
  35. char c;
  36. char buf[4];
  37. in.read(buf, 3);
  38. buf[3] = 0;
  39. if(strcmp(buf, ";a:") != 0)
  40. {
  41. cmCTestLog(this->CTest, ERROR_MESSAGE,
  42. "failed to read start of coverage array, found : "
  43. << buf << "\n");
  44. return false;
  45. }
  46. int size = 0;
  47. if(!this->ReadInt(in, size))
  48. {
  49. cmCTestLog(this->CTest, ERROR_MESSAGE,
  50. "failed to read size ");
  51. return false;
  52. }
  53. if(!in.get(c) && c == '{')
  54. {
  55. cmCTestLog(this->CTest, ERROR_MESSAGE,
  56. "failed to read open {\n");
  57. return false;
  58. }
  59. for(int i =0; i < size; i++)
  60. {
  61. this->ReadUntil(in, ':');
  62. int line = 0;
  63. this->ReadInt(in, line);
  64. // ok xdebug may have a bug here
  65. // it seems to be 1 based but often times
  66. // seems to have a 0'th line.
  67. line--;
  68. if(line < 0)
  69. {
  70. line = 0;
  71. }
  72. this->ReadUntil(in, ':');
  73. int value = 0;
  74. this->ReadInt(in, value);
  75. // make sure the vector is the right size and is
  76. // initialized with -1 for each line
  77. while(coverageVector.size() <= static_cast<size_t>(line) )
  78. {
  79. coverageVector.push_back(-1);
  80. }
  81. // if value is less than 0, set it to zero
  82. // TODO figure out the difference between
  83. // -1 and -2 in xdebug coverage?? For now
  84. // assume less than 0 is just not covered
  85. // CDash expects -1 for non executable code (like comments)
  86. // and 0 for uncovered code, and a positive value
  87. // for number of times a line was executed
  88. if(value < 0)
  89. {
  90. value = 0;
  91. }
  92. // if unset then set it to value
  93. if(coverageVector[line] == -1)
  94. {
  95. coverageVector[line] = value;
  96. }
  97. // otherwise increment by value
  98. else
  99. {
  100. coverageVector[line] += value;
  101. }
  102. }
  103. return true;
  104. }
  105. bool cmParsePHPCoverage::ReadInt(std::istream& in, int& v)
  106. {
  107. std::string s;
  108. char c = 0;
  109. while(in.get(c) && c != ':' && c != ';')
  110. {
  111. s += c;
  112. }
  113. v = atoi(s.c_str());
  114. return true;
  115. }
  116. bool cmParsePHPCoverage::ReadArraySize(std::istream& in, int& size)
  117. {
  118. char c = 0;
  119. in.get(c);
  120. if(c != 'a')
  121. {
  122. return false;
  123. }
  124. if(in.get(c) && c == ':')
  125. {
  126. if(this->ReadInt(in, size))
  127. {
  128. return true;
  129. }
  130. }
  131. return false;
  132. }
  133. bool cmParsePHPCoverage::ReadFileInformation(std::istream& in)
  134. {
  135. char buf[4];
  136. in.read(buf, 2);
  137. buf[2] = 0;
  138. if(strcmp(buf, "s:") != 0)
  139. {
  140. cmCTestLog(this->CTest, ERROR_MESSAGE,
  141. "failed to read start of file info found: [" << buf << "]\n");
  142. return false;
  143. }
  144. char c;
  145. int size = 0;
  146. if(this->ReadInt(in, size))
  147. {
  148. size++; // add one for null termination
  149. char* s = new char[size+1];
  150. // read open quote
  151. if(in.get(c) && c != '"')
  152. {
  153. delete[] s;
  154. return false;
  155. }
  156. // read the string data
  157. in.read(s, size-1);
  158. s[size-1] = 0;
  159. std::string fileName = s;
  160. delete [] s;
  161. // read close quote
  162. if(in.get(c) && c != '"')
  163. {
  164. cmCTestLog(this->CTest, ERROR_MESSAGE,
  165. "failed to read close quote\n"
  166. << "read [" << c << "]\n");
  167. return false;
  168. }
  169. if(!this->ReadCoverageArray(in, fileName) )
  170. {
  171. cmCTestLog(this->CTest, ERROR_MESSAGE,
  172. "failed to read coverage array for file: "
  173. << fileName << "\n");
  174. return false;
  175. }
  176. return true;
  177. }
  178. return false;
  179. }
  180. bool cmParsePHPCoverage::ReadPHPData(const char* file)
  181. {
  182. cmsys::ifstream in(file);
  183. if(!in)
  184. {
  185. return false;
  186. }
  187. int size = 0;
  188. this->ReadArraySize(in, size);
  189. char c = 0;
  190. in.get(c);
  191. if(c != '{')
  192. {
  193. cmCTestLog(this->CTest, ERROR_MESSAGE,
  194. "failed to read open array\n");
  195. return false;
  196. }
  197. for(int i =0; i < size; i++)
  198. {
  199. if(!this->ReadFileInformation(in))
  200. {
  201. cmCTestLog(this->CTest, ERROR_MESSAGE,
  202. "Failed to read file #" << i << "\n");
  203. return false;
  204. }
  205. in.get(c);
  206. if(c != '}')
  207. {
  208. cmCTestLog(this->CTest, ERROR_MESSAGE,
  209. "failed to read close array\n");
  210. return false;
  211. }
  212. }
  213. return true;
  214. }
  215. bool cmParsePHPCoverage::ReadPHPCoverageDirectory(const char* d)
  216. {
  217. cmsys::Directory dir;
  218. if(!dir.Load(d))
  219. {
  220. return false;
  221. }
  222. size_t numf;
  223. unsigned int i;
  224. numf = dir.GetNumberOfFiles();
  225. for (i = 0; i < numf; i++)
  226. {
  227. std::string file = dir.GetFile(i);
  228. if(file != "." && file != ".."
  229. && !cmSystemTools::FileIsDirectory(file))
  230. {
  231. std::string path = d;
  232. path += "/";
  233. path += file;
  234. if(!this->ReadPHPData(path.c_str()))
  235. {
  236. return false;
  237. }
  238. }
  239. }
  240. return true;
  241. }