tools_spiffs_utils.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. #define LOG_LOCAL_LEVEL ESP_LOG_INFO
  2. #include "tools_spiffs_utils.h"
  3. #include "esp_log.h"
  4. #include "tools.h"
  5. #include "PBW.h"
  6. #include <ctype.h> // For isprint()
  7. #include <dirent.h>
  8. #include <fnmatch.h>
  9. #include <iomanip> // for std::setw
  10. #include <set>
  11. #include <sstream>
  12. #include <sys/stat.h> // for file stat
  13. #include <vector>
  14. static const char* TAG = "spiffs_utils";
  15. static bool initialized = false;
  16. static esp_vfs_spiffs_conf_t* spiffs_conf = NULL;
  17. const char* spiffs_base_path = "/spiffs";
  18. // Struct to represent a file entry
  19. // list of reserved files that the system controls
  20. const std::vector<std::string> restrictedPaths = {
  21. "/spiffs/defaults/*", "/spiffs/fonts/*", "/spiffs/targets/*", "/spiffs/www/*"};
  22. void init_spiffs() {
  23. if (initialized) {
  24. ESP_LOGV(TAG, "SPIFFS already initialized. returning");
  25. return;
  26. }
  27. ESP_LOGI(TAG, "Initializing the SPI File system");
  28. spiffs_conf = (esp_vfs_spiffs_conf_t*)malloc(sizeof(esp_vfs_spiffs_conf_t));
  29. spiffs_conf->base_path = spiffs_base_path;
  30. spiffs_conf->partition_label = NULL;
  31. spiffs_conf->max_files = 5;
  32. spiffs_conf->format_if_mount_failed = true;
  33. // Use settings defined above to initialize and mount SPIFFS filesystem.
  34. // Note: esp_vfs_spiffs_register is an all-in-one convenience function.
  35. esp_err_t ret = esp_vfs_spiffs_register(spiffs_conf);
  36. if (ret != ESP_OK) {
  37. if (ret == ESP_FAIL) {
  38. ESP_LOGE(TAG, "Failed to mount or format filesystem");
  39. } else if (ret == ESP_ERR_NOT_FOUND) {
  40. ESP_LOGE(TAG, "Failed to find SPIFFS partition");
  41. } else {
  42. ESP_LOGE(TAG, "Failed to initialize SPIFFS (%s)", esp_err_to_name(ret));
  43. }
  44. return;
  45. }
  46. size_t total = 0, used = 0;
  47. ret = esp_spiffs_info(spiffs_conf->partition_label, &total, &used);
  48. if (ret != ESP_OK) {
  49. ESP_LOGW(TAG, "Failed to get SPIFFS partition information (%s). Formatting...",
  50. esp_err_to_name(ret));
  51. esp_spiffs_format(spiffs_conf->partition_label);
  52. } else {
  53. ESP_LOGI(TAG, "Partition size: total: %d, used: %d", total, used);
  54. }
  55. initialized = true;
  56. }
  57. bool write_file(const uint8_t* data, size_t sz, const char* filename) {
  58. bool result = true;
  59. FILE* file = NULL;
  60. init_spiffs();
  61. if (data == NULL) {
  62. ESP_LOGE(TAG, "Cannot write file. Data not received");
  63. return false;
  64. }
  65. if (sz == 0) {
  66. ESP_LOGE(TAG, "Cannot write file. Data length 0");
  67. return false;
  68. }
  69. file = fopen(filename, "wb");
  70. if (file == NULL) {
  71. ESP_LOGE(TAG, "Error opening %s for writing", filename);
  72. return false;
  73. }
  74. size_t written = fwrite(data, 1, sz, file);
  75. if (written != sz) {
  76. ESP_LOGE(TAG, "Write error. Wrote %d bytes of %d.", written, sz);
  77. result = false;
  78. }
  79. fclose(file);
  80. return result;
  81. }
  82. void* load_file(uint32_t memflags, size_t* sz, const char* filename) {
  83. void* data = NULL;
  84. FILE* file = NULL;
  85. init_spiffs();
  86. size_t fsz = 0;
  87. file = fopen(filename, "rb");
  88. if (file == NULL) {
  89. return data;
  90. }
  91. fseek(file, 0, SEEK_END);
  92. fsz = ftell(file);
  93. fseek(file, 0, SEEK_SET);
  94. if (fsz > 0) {
  95. ESP_LOGD(TAG, "Allocating %d bytes to load file %s content with flags: %s ", fsz, filename,
  96. get_mem_flag_desc(memflags));
  97. data = (void*)heap_caps_calloc(1, fsz, memflags);
  98. if (data == NULL) {
  99. ESP_LOGE(TAG, "Failed to allocate %d bytes to load file %s", fsz, filename);
  100. } else {
  101. fread(data, 1, fsz, file);
  102. if (sz) {
  103. *sz = fsz;
  104. }
  105. }
  106. } else {
  107. ESP_LOGW(TAG, "File is empty. Nothing to read");
  108. }
  109. fclose(file);
  110. return data;
  111. }
  112. bool get_file_info(struct stat* pfileInfo, const char* filename) {
  113. // ensure that the spiffs is initialized
  114. struct stat fileInfo;
  115. init_spiffs();
  116. if (strlen(filename) == 0) {
  117. ESP_LOGE(TAG, "Invalid file name");
  118. return false;
  119. }
  120. ESP_LOGD(TAG, "Getting file info for %s", filename);
  121. bool result = false;
  122. // Use stat to fill the fileInfo structure
  123. if (stat(filename, &fileInfo) != 0) {
  124. ESP_LOGD(TAG, "File %s not found", filename);
  125. } else {
  126. result = true;
  127. if (pfileInfo) {
  128. memcpy(pfileInfo, &fileInfo, sizeof(fileInfo));
  129. }
  130. ESP_LOGD(TAG, "File %s has %lu bytes", filename, fileInfo.st_size);
  131. }
  132. return result;
  133. }
  134. bool is_restricted_path(const char* filename) {
  135. for (const auto& pattern : restrictedPaths) {
  136. if (fnmatch(pattern.c_str(), filename, 0) == 0) {
  137. return true;
  138. }
  139. }
  140. return false;
  141. }
  142. bool erase_path(const char* filename, bool restricted) {
  143. std::string full_path_with_wildcard = std::string(filename);
  144. if (full_path_with_wildcard.empty()) {
  145. ESP_LOGE(TAG, "Error constructing full path");
  146. return false;
  147. }
  148. ESP_LOGI(TAG, "Erasing file(s) matching pattern %s", full_path_with_wildcard.c_str());
  149. // Extract directory path and wildcard pattern
  150. size_t lastSlashPos = full_path_with_wildcard.find_last_of('/');
  151. std::string dirpath = full_path_with_wildcard.substr(0, lastSlashPos);
  152. std::string wildcard = full_path_with_wildcard.substr(lastSlashPos + 1);
  153. ESP_LOGD(TAG, "Last slash pos: %d, dirpath: %s, wildcard %s ", lastSlashPos, dirpath.c_str(),
  154. wildcard.c_str());
  155. DIR* dir = opendir(dirpath.empty() ? "." : dirpath.c_str());
  156. if (!dir) {
  157. ESP_LOGE(TAG, "Error opening directory");
  158. return false;
  159. }
  160. bool result = false;
  161. struct dirent* ent;
  162. while ((ent = readdir(dir)) != NULL) {
  163. if (fnmatch(wildcard.c_str(), ent->d_name, 0) == 0) {
  164. std::string fullfilename = dirpath + "/" + ent->d_name;
  165. // Check if the file is restricted
  166. if (restricted && is_restricted_path(fullfilename.c_str())) {
  167. ESP_LOGW(TAG, "Skipping restricted file %s", fullfilename.c_str());
  168. continue;
  169. }
  170. ESP_LOGW(TAG, "Deleting file %s", fullfilename.c_str());
  171. if (remove(fullfilename.c_str()) != 0) {
  172. ESP_LOGE(TAG, "Error deleting file %s", fullfilename.c_str());
  173. } else {
  174. result = true;
  175. }
  176. } else {
  177. ESP_LOGV(
  178. TAG, "%s does not match file pattern to delete: %s", ent->d_name, wildcard.c_str());
  179. }
  180. }
  181. closedir(dir);
  182. return result;
  183. }
  184. // Function to format and print a file entry
  185. void printFileEntry(const tools_file_entry_t& entry) {
  186. const char* suffix;
  187. double size;
  188. // Format the size
  189. if (entry.type == 'F') {
  190. if (entry.size < 1024) { // less than 1 KB
  191. size = entry.size;
  192. suffix = " B";
  193. printf("%c %10.0f%s %-80s%4s\n", entry.type, size, suffix, entry.name.c_str(),
  194. entry.restricted ? "X" : "-");
  195. } else {
  196. if (entry.size < 1024 * 1024) { // 1 KB to <1 MB
  197. size = entry.size / 1024;
  198. suffix = " KB";
  199. } else { // 1 MB and above
  200. size = entry.size / (1024 * 1024);
  201. suffix = " MB";
  202. }
  203. printf("%c %10.0f%s %-80s%4s\n", entry.type, size, suffix, entry.name.c_str(),
  204. entry.restricted ? "X" : "-");
  205. }
  206. } else {
  207. printf("%c - %-80s%4s\n", entry.type, entry.name.c_str(),
  208. entry.restricted ? "X" : "-");
  209. }
  210. }
  211. void listFiles(const char* path_requested_char) {
  212. // Ensure that the SPIFFS is initialized
  213. init_spiffs();
  214. auto path_requested = std::string(path_requested_char);
  215. auto filesList = get_files_list(path_requested);
  216. printf("---------------------------------------------------------------------------------------"
  217. "---------------\n");
  218. printf("T SIZE NAME "
  219. " RSTR\n");
  220. printf("---------------------------------------------------------------------------------------"
  221. "---------------\n");
  222. uint64_t total = 0;
  223. int nfiles = 0;
  224. for (auto& e : filesList) {
  225. if (e.type == 'F') {
  226. total += e.size;
  227. nfiles++;
  228. }
  229. printFileEntry(e);
  230. }
  231. printf("---------------------------------------------------------------------------------------"
  232. "---------------\n");
  233. if (total > 0) {
  234. printf("Total : %lu bytes in %d file(s)\n", (unsigned long)total, nfiles);
  235. }
  236. uint32_t tot = 0, used = 0;
  237. esp_spiffs_info(NULL, &tot, &used);
  238. printf("SPIFFS: free %d KB of %d KB\n", (tot - used) / 1024, tot / 1024);
  239. printf("---------------------------------------------------------------------------------------"
  240. "---------------\n");
  241. }
  242. bool out_file_binding(pb_ostream_t* stream, const uint8_t* buf, size_t count) {
  243. FILE* file = (FILE*)stream->state;
  244. ESP_LOGV(TAG, "Writing %d bytes to file", count);
  245. return fwrite(buf, 1, count, file) == count;
  246. }
  247. bool in_file_binding(pb_istream_t* stream, pb_byte_t* buf, size_t count) {
  248. FILE* file = (FILE*)stream->state;
  249. ESP_LOGV(TAG, "Reading %d bytes from file", count);
  250. return fread(buf, 1, count, file) == count;
  251. }
  252. // Function to list files matching the path_requested and return a std::list of FileEntry
  253. std::list<tools_file_entry_t> get_files_list(const std::string& path_requested) {
  254. std::list<tools_file_entry_t> fileList;
  255. std::set<std::string> directoryNames;
  256. struct dirent* ent;
  257. struct stat sb;
  258. // Ensure that the SPIFFS is initialized
  259. init_spiffs();
  260. std::string prefix = (path_requested.back() != '/' ? path_requested + "/" : path_requested);
  261. DIR* dir = opendir(path_requested.c_str());
  262. if (!dir) {
  263. ESP_LOGE(TAG, "Error opening directory %s ", path_requested.c_str());
  264. return fileList;
  265. }
  266. while ((ent = readdir(dir)) != NULL) {
  267. tools_file_entry_t fileEntry;
  268. fileEntry.name = prefix + ent->d_name;
  269. fileEntry.type = (ent->d_type == DT_REG) ? 'F' : 'D';
  270. fileEntry.restricted = is_restricted_path(fileEntry.name.c_str());
  271. if (stat(fileEntry.name.c_str(), &sb) == -1) {
  272. ESP_LOGE(TAG, "Ignoring file %s ", fileEntry.name.c_str());
  273. continue;
  274. }
  275. fileEntry.size = sb.st_size;
  276. fileList.push_back(fileEntry);
  277. // Extract all parent directory names
  278. size_t pos = 0;
  279. while ((pos = fileEntry.name.find('/', pos + 1)) != std::string::npos) {
  280. directoryNames.insert(fileEntry.name.substr(0, pos));
  281. }
  282. }
  283. closedir(dir);
  284. // Add directories to the file list
  285. for (const auto& dirName : directoryNames) {
  286. tools_file_entry_t dirEntry;
  287. dirEntry.name = dirName;
  288. dirEntry.type = 'D'; // Mark as directory
  289. fileList.push_back(dirEntry);
  290. }
  291. // Sort the list by directory/file name
  292. fileList.sort(
  293. [](const tools_file_entry_t& a, const tools_file_entry_t& b) { return a.name < b.name; });
  294. // Remove duplicates
  295. fileList.unique(
  296. [](const tools_file_entry_t& a, const tools_file_entry_t& b) { return a.name == b.name; });
  297. return fileList;
  298. }
  299. bool cat_file(const char* filename) {
  300. size_t sz;
  301. uint8_t* content = (uint8_t*)load_file_psram(&sz, filename);
  302. if (content == NULL) {
  303. printf("Failed to load file\n");
  304. return false;
  305. }
  306. for (size_t i = 0; i < sz; i++) {
  307. if (isprint(content[i])) {
  308. printf("%c", content[i]); // Print as a character
  309. } else {
  310. printf("\\x%02x", content[i]); // Print as hexadecimal
  311. }
  312. }
  313. printf("\n"); // New line after printing the content
  314. free(content);
  315. return true;
  316. }
  317. bool dump_data(const uint8_t* pData, size_t length) {
  318. if (pData == NULL && length == 0) {
  319. printf("%s/%s\n", pData == nullptr ? "Invalid Data" : "Data OK",
  320. length == 0 ? "Invalid Length" : "Length OK");
  321. return false;
  322. }
  323. for (size_t i = 0; i < length; i++) {
  324. if (isprint((char)pData[i])) {
  325. printf("%c", (char)pData[i]); // Print as a character
  326. } else {
  327. printf("\\x%02x", (char)pData[i]); // Print as hexadecimal
  328. }
  329. }
  330. printf("\n"); // New line after printing the content
  331. return true;
  332. }
  333. bool dump_structure(const pb_msgdesc_t* fields, const void* src_struct) {
  334. try {
  335. std::vector<pb_byte_t> encodedData = System::PBHelper::EncodeData(fields, src_struct);
  336. return dump_data(encodedData.data(), encodedData.size());
  337. } catch (const std::runtime_error& e) {
  338. ESP_LOGE(TAG, "Error in dump_structure: %s", e.what());
  339. return false;
  340. }
  341. }