2
0

test_tools.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. #define LOG_LOCAL_LEVEL ESP_LOG_DEBUG
  2. #include "esp_log.h"
  3. #include "test_common_init.h"
  4. #include "tools.h"
  5. #include "unity.h"
  6. #include <sstream>
  7. #include <string>
  8. #include <vector>
  9. #include "tools_spiffs_utils.h"
  10. static const char* TAG = "test_tools";
  11. static const char* data = "SomeTestData\0";
  12. #define UINT8T_DATA (const uint8_t*)data
  13. static size_t len = strlen(data);
  14. TEST_CASE("Test Write/Read File", "[tools]") {
  15. common_test_init();
  16. const char* testfilename = "/spiffs/test_file.bin";
  17. size_t loaded = 0;
  18. // allow +1 to the length below for null termination of the string
  19. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len + 1, testfilename));
  20. char* loaded_data = (char*)load_file_psram(&loaded, testfilename);
  21. TEST_ASSERT_EQUAL(loaded, len + 1);
  22. TEST_ASSERT_NOT_EQUAL(loaded_data, NULL);
  23. TEST_ASSERT_EQUAL_STRING((const char*)data, (char*)loaded_data);
  24. free(loaded_data);
  25. }
  26. TEST_CASE("Test Simple Erase File ", "[tools]") {
  27. std::ostringstream fullfilename;
  28. const char* prefix = "test_file";
  29. struct stat fileInfo;
  30. // test for simple file names
  31. fullfilename << "/spiffs/" << prefix << ".bin";
  32. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len, fullfilename.str().c_str()));
  33. TEST_ASSERT_TRUE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  34. TEST_ASSERT_EQUAL(fileInfo.st_size, len);
  35. TEST_ASSERT_TRUE(erase_path(fullfilename.str().c_str(), false));
  36. TEST_ASSERT_FALSE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  37. fullfilename.clear();
  38. fullfilename.str("");
  39. }
  40. TEST_CASE("Test Wildcard Erase File ", "[tools]") {
  41. std::ostringstream fullfilename;
  42. std::ostringstream fullfilename2;
  43. std::ostringstream fullnamewc;
  44. const char* prefix = "/spiffs/test_file";
  45. struct stat fileInfo;
  46. // Test for wildcard
  47. fullfilename << prefix << ".bin";
  48. fullfilename2 << prefix << ".bin2";
  49. fullnamewc << prefix << "*.bin";
  50. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len, fullfilename.str().c_str()));
  51. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len, fullfilename2.str().c_str()));
  52. TEST_ASSERT_TRUE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  53. TEST_ASSERT_EQUAL(fileInfo.st_size, len);
  54. TEST_ASSERT_TRUE(erase_path(fullnamewc.str().c_str(), false));
  55. TEST_ASSERT_FALSE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  56. TEST_ASSERT_TRUE(get_file_info(&fileInfo, fullfilename2.str().c_str()));
  57. fullnamewc.str("");
  58. fullnamewc << prefix << ".bin*";
  59. TEST_ASSERT_TRUE(erase_path(fullnamewc.str().c_str(), false));
  60. TEST_ASSERT_FALSE(get_file_info(&fileInfo, fullfilename2.str().c_str()));
  61. }
  62. TEST_CASE("Test Folder Erase File ", "[tools]") {
  63. std::ostringstream fullfilename;
  64. const char* prefix = "test_file";
  65. struct stat fileInfo;
  66. fullfilename << "/spiffs/somefolder/" << prefix << ".bin";
  67. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len, fullfilename.str().c_str()));
  68. TEST_ASSERT_TRUE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  69. TEST_ASSERT_EQUAL(fileInfo.st_size, len);
  70. TEST_ASSERT_TRUE(erase_path(fullfilename.str().c_str(), false));
  71. TEST_ASSERT_FALSE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  72. }
  73. TEST_CASE("Test Folder Wildcard Erase File ", "[tools]") {
  74. std::ostringstream fullfilename;
  75. std::ostringstream fullnamewc;
  76. const char* prefix = "test_file";
  77. struct stat fileInfo;
  78. fullfilename << "/spiffs/somefolder/" << prefix << ".bin";
  79. fullnamewc << prefix << "*.bin";
  80. TEST_ASSERT_TRUE(write_file(UINT8T_DATA, len, fullfilename.str().c_str()));
  81. TEST_ASSERT_TRUE(get_file_info(&fileInfo, fullfilename.str().c_str()));
  82. TEST_ASSERT_EQUAL(fileInfo.st_size, len);
  83. TEST_ASSERT_TRUE(erase_path(fullfilename.str().c_str(), false));
  84. TEST_ASSERT_FALSE(get_file_info(&fileInfo, fullnamewc.str().c_str()));
  85. }
  86. const std::vector<std::string> testRestrictedFiles = {"defaults/config.bin",
  87. "fonts/droid_sans_fb_11x13.bin", "targets/bureau-oled/config.bin", "www/favicon-32x32.png"};
  88. TEST_CASE("Test _write_file Function", "[tools]") {
  89. init_spiffs();
  90. const uint8_t testData[] = {0x01, 0x02, 0x03, 0x04};
  91. size_t testDataSize = sizeof(testData);
  92. const char* testFileName = "/spiffs/testfile.bin";
  93. // Test writing valid data
  94. TEST_ASSERT_TRUE(write_file(testData, testDataSize, testFileName));
  95. // Verify file size
  96. struct stat fileInfo;
  97. TEST_ASSERT_TRUE(get_file_info(&fileInfo, testFileName));
  98. TEST_ASSERT_EQUAL_UINT32(testDataSize, fileInfo.st_size);
  99. // Verify file content
  100. size_t loadedSize;
  101. uint8_t* loadedData = (uint8_t*)load_file_psram(&loadedSize, testFileName);
  102. TEST_ASSERT_NOT_NULL(loadedData);
  103. TEST_ASSERT_EQUAL_UINT32(testDataSize, loadedSize);
  104. TEST_ASSERT_EQUAL_UINT8_ARRAY(testData, loadedData, testDataSize);
  105. free(loadedData);
  106. ESP_LOGI(TAG, "Erasing test file");
  107. TEST_ASSERT_TRUE(erase_path(testFileName, false));
  108. // Test writing with null data
  109. ESP_LOGI(TAG, "Test Invalid write_file with NULL pointer");
  110. TEST_ASSERT_FALSE(write_file(NULL, testDataSize, testFileName));
  111. // Test writing with zero size
  112. ESP_LOGI(TAG, "Test Invalid write_file with data length 0");
  113. TEST_ASSERT_FALSE(write_file(testData, 0, testFileName));
  114. }
  115. TEST_CASE("Test _open_file Function", "[tools]") {
  116. init_spiffs();
  117. const char* testFileName = "/spiffs/test_open_file.bin";
  118. const char* testFileContent = "Hello, world!";
  119. size_t contentLength = strlen(testFileContent);
  120. // Test opening a file for writing
  121. FILE* writeFile = fopen(testFileName, "w");
  122. TEST_ASSERT_NOT_NULL(writeFile);
  123. fwrite(testFileContent, 1, contentLength, writeFile);
  124. fclose(writeFile);
  125. // Test opening the same file for reading
  126. FILE* readFile = fopen(testFileName, "r");
  127. TEST_ASSERT_NOT_NULL(readFile);
  128. char buffer[100];
  129. size_t bytesRead = fread(buffer, 1, contentLength, readFile);
  130. TEST_ASSERT_EQUAL_UINT32(contentLength, bytesRead);
  131. buffer[bytesRead] = '\0'; // Null-terminate the string
  132. TEST_ASSERT_EQUAL_STRING(testFileContent, buffer);
  133. fclose(readFile);
  134. // Test opening a file with an invalid mode
  135. FILE* invalidFile = fopen(testFileName, "invalid_mode");
  136. TEST_ASSERT_NULL(invalidFile);
  137. TEST_ASSERT_TRUE(erase_path(testFileName, false));
  138. // Test opening a non-existent file for reading
  139. FILE* nonExistentFile = fopen("/spiffs/non_existent_file.bin", "r");
  140. TEST_ASSERT_NULL(nonExistentFile);
  141. }
  142. TEST_CASE("Test _load_file and _get_file_info Functions", "[tools]") {
  143. init_spiffs();
  144. const char* testFileName = "/spiffs/test_load_file.bin";
  145. const char* testFileContent = "Hello, world!";
  146. size_t contentLength = strlen(testFileContent);
  147. // Write test data to a file
  148. FILE* writeFile = fopen(testFileName, "w");
  149. TEST_ASSERT_NOT_NULL(writeFile);
  150. fwrite(testFileContent, 1, contentLength, writeFile);
  151. fclose(writeFile);
  152. // Test loading file content
  153. size_t loadedSize;
  154. uint8_t* loadedData = (uint8_t*)load_file_psram(&loadedSize, testFileName);
  155. TEST_ASSERT_NOT_NULL(loadedData);
  156. TEST_ASSERT_EQUAL_UINT32(contentLength, loadedSize);
  157. TEST_ASSERT_EQUAL_UINT8_ARRAY(testFileContent, loadedData, contentLength);
  158. free(loadedData);
  159. // Test getting file information
  160. struct stat fileInfo;
  161. TEST_ASSERT_TRUE(get_file_info(&fileInfo, testFileName));
  162. TEST_ASSERT_EQUAL_UINT32(contentLength, fileInfo.st_size);
  163. TEST_ASSERT_TRUE(erase_path(testFileName, false));
  164. // Test loading a non-existent file
  165. uint8_t* nonExistentData =
  166. (uint8_t*)load_file_psram(&loadedSize, "/spiffs/non_existent_file.bin");
  167. TEST_ASSERT_NULL(nonExistentData);
  168. // Test getting information for a non-existent file
  169. TEST_ASSERT_FALSE(get_file_info(&fileInfo, "/spiffs/non_existent_file.bin"));
  170. }
  171. TEST_CASE("Test trim Function", "[tools]") {
  172. // Test trimming a string with leading and trailing spaces
  173. std::string str1 = " Hello World ";
  174. TEST_ASSERT_EQUAL_STRING("Hello World", trim(str1).c_str());
  175. // Test trimming a string with no leading or trailing spaces
  176. std::string str2 = "Hello World";
  177. TEST_ASSERT_EQUAL_STRING("Hello World", trim(str2).c_str());
  178. // Test trimming an empty string
  179. std::string str3 = "";
  180. TEST_ASSERT_EQUAL_STRING("", trim(str3).c_str());
  181. // Test trimming a string with only spaces
  182. std::string str4 = " ";
  183. TEST_ASSERT_EQUAL_STRING("", trim(str4).c_str());
  184. }
  185. TEST_CASE("Test toLowerStr Function", "[tools]") {
  186. // Test converting a mixed case string to lowercase
  187. std::string str1 = "Hello World";
  188. toLowerStr(str1);
  189. TEST_ASSERT_EQUAL_STRING("hello world", str1.c_str());
  190. // Test converting an already lowercase string
  191. std::string str2 = "hello world";
  192. toLowerStr(str2);
  193. TEST_ASSERT_EQUAL_STRING("hello world", str2.c_str());
  194. // Test converting an uppercase string
  195. std::string str3 = "HELLO WORLD";
  196. toLowerStr(str3);
  197. TEST_ASSERT_EQUAL_STRING("hello world", str3.c_str());
  198. // Test converting an empty string
  199. std::string str4 = "";
  200. toLowerStr(str4);
  201. TEST_ASSERT_EQUAL_STRING("", str4.c_str());
  202. // Test converting a string with special characters
  203. std::string str5 = "Hello-World_123";
  204. toLowerStr(str5);
  205. TEST_ASSERT_EQUAL_STRING("hello-world_123", str5.c_str());
  206. }
  207. static const std::vector<std::string> testRestrictedPaths = {
  208. "/spiffs/defaults/file.txt", // A restricted path
  209. "/spiffs/fonts/otherfile.txt", // A restricted path
  210. "/spiffs/targets/somefile.txt", // A restricted path
  211. "/spiffs/targets/*.txt", // A restricted path
  212. "/spiffs/www/index.html" // A restricted path
  213. };
  214. TEST_CASE("Test is_restricted_path with restricted paths", "[tools]") {
  215. for (const std::string& path : testRestrictedPaths) {
  216. bool result = is_restricted_path(path.c_str());
  217. TEST_ASSERT_TRUE(result);
  218. }
  219. }
  220. // Negative Testing
  221. TEST_CASE("Test is_restricted_path with non-restricted paths", "[tools]") {
  222. const std::vector<std::string> nonRestrictedPaths = {
  223. "/spiffs/allowed/file.txt", // Not a restricted path
  224. "/spiffs/allowed/otherfile.txt" // Not a restricted path
  225. };
  226. for (const std::string& path : nonRestrictedPaths) {
  227. bool result = is_restricted_path(path.c_str());
  228. TEST_ASSERT_FALSE(result);
  229. }
  230. }
  231. static const std::vector<std::string> testWildcardPaths = {
  232. "/spiffs/defaults/file.txt", // A restricted path
  233. "/spiffs/fonts/otherfile.txt", // A restricted path
  234. "/spiffs/targets/somefile.txt", // A restricted path
  235. "/spiffs/www/index.html", // A restricted path
  236. "/spiffs/defaults/*", // A path with wildcard
  237. "/spiffs/fonts/*" // A path with wildcard
  238. };
  239. TEST_CASE("Test is_restricted_path with wildcards (positive)", "[tools]") {
  240. for (const std::string& path : testWildcardPaths) {
  241. bool result = is_restricted_path(path.c_str());
  242. if (!result) {
  243. ESP_LOGE(TAG, "Unexpected result. File should be restricted: %s", path.c_str());
  244. }
  245. TEST_ASSERT_TRUE(result);
  246. }
  247. }
  248. TEST_CASE("Test Erase Restricted Path", "[erase_path]") {
  249. for (const std::string& path : testRestrictedFiles) {
  250. std::ostringstream fullfilename;
  251. fullfilename << "/spiffs/" << path;
  252. // Check if the file exists before erasing
  253. struct stat fileInfoBefore;
  254. TEST_ASSERT_TRUE(get_file_info(&fileInfoBefore, fullfilename.str().c_str()));
  255. // Attempt to erase the restricted path
  256. TEST_ASSERT_FALSE(erase_path(fullfilename.str().c_str(), true));
  257. // Check if the file still exists after the erase attempt
  258. struct stat fileInfoAfter;
  259. TEST_ASSERT_TRUE(get_file_info(&fileInfoAfter, fullfilename.str().c_str()));
  260. TEST_ASSERT_EQUAL(fileInfoBefore.st_ino, fileInfoAfter.st_ino);
  261. }
  262. }
  263. // Test case to attempt erasing restricted files with wildcards
  264. TEST_CASE("Test Erase Restricted Files with Wildcards", "[erase_path]") {
  265. // Get a list of restricted files based on restrictedPaths
  266. std::list<tools_file_entry_t> restrictedFiles = get_files_list(std::string("/"));
  267. for (const tools_file_entry_t& restrictedFile : restrictedFiles) {
  268. std::string fullfilename = "/" + restrictedFile.name;
  269. // Check if the file exists before erasing
  270. struct stat fileInfoBefore;
  271. TEST_ASSERT_TRUE(get_file_info(&fileInfoBefore, fullfilename.c_str()));
  272. // Attempt to erase the file with wildcard pattern
  273. TEST_ASSERT_FALSE(erase_path(fullfilename.c_str(), true));
  274. // Check if the file still exists after the erase attempt
  275. struct stat fileInfoAfter;
  276. TEST_ASSERT_TRUE(get_file_info(&fileInfoAfter, fullfilename.c_str()));
  277. TEST_ASSERT_EQUAL(fileInfoBefore.st_ino, fileInfoAfter.st_ino);
  278. }
  279. }
  280. // Test case to create a file and delete it bypassing restrictions
  281. TEST_CASE("Test Create and Delete File Bypassing Restrictions", "[erase_path]") {
  282. const uint8_t testData[] = {0x01, 0x02, 0x03, 0x04};
  283. size_t testDataSize = sizeof(testData);
  284. const char* testFileName = "/spiffs/defaults/test_file.bin";
  285. // Test writing valid data to create the file
  286. TEST_ASSERT_TRUE(write_file(testData, testDataSize, testFileName));
  287. // Verify file size
  288. struct stat fileInfo;
  289. TEST_ASSERT_TRUE(get_file_info(&fileInfo, testFileName));
  290. TEST_ASSERT_EQUAL_UINT32(testDataSize, fileInfo.st_size);
  291. // Attempt to erase the file with bypassing restrictions (false)
  292. TEST_ASSERT_TRUE(erase_path(testFileName, false));
  293. // Verify that the file no longer exists
  294. TEST_ASSERT_FALSE(get_file_info(&fileInfo, testFileName));
  295. }
  296. // Test function to create a file, check its presence and unrestricted status, and delete it
  297. TEST_CASE("Create, Check, and Delete File in SPIFFS", "[spiffs]") {
  298. const char* testFileName = "/spiffs/somerandomfile.bin";
  299. const uint8_t testData[] = {0x01, 0x02, 0x03, 0x04};
  300. size_t testDataSize = sizeof(testData);
  301. // Create a file
  302. TEST_ASSERT_TRUE(write_file(testData, testDataSize, testFileName));
  303. // Get the list of files
  304. std::list<tools_file_entry_t> fileList = get_files_list("/spiffs");
  305. // Check if the new file is in the list and is unrestricted
  306. bool fileFound = false;
  307. bool fileUnrestricted = false;
  308. for (const auto& fileEntry : fileList) {
  309. if (fileEntry.name == testFileName) {
  310. fileFound = true;
  311. fileUnrestricted = !fileEntry.restricted;
  312. break;
  313. }
  314. }
  315. TEST_ASSERT_TRUE(fileFound);
  316. TEST_ASSERT_TRUE(fileUnrestricted);
  317. // Delete the file
  318. TEST_ASSERT_TRUE(erase_path(testFileName, false)); // Assuming false bypasses restrictions
  319. fileFound = false;
  320. // Verify that the file no longer exists
  321. fileList = get_files_list("/spiffs");
  322. for (const auto& fileEntry : fileList) {
  323. if (std::string(testFileName) == fileEntry.name) {
  324. fileFound = true;
  325. break;
  326. }
  327. }
  328. TEST_ASSERT_FALSE(fileFound);
  329. }