tools_spiffs_utils.h 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. /*
  2. *
  3. * Sebastien L. 2023, sle118@hotmail.com
  4. * Philippe G. 2023, philippe_44@outlook.com
  5. *
  6. * This software is released under the MIT License.
  7. * https://opensource.org/licenses/MIT
  8. *
  9. * License Overview:
  10. * ----------------
  11. * The MIT License is a permissive open source license. As a user of this software, you are free to:
  12. * - Use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of this software.
  13. * - Use the software for private, commercial, or any other purposes.
  14. *
  15. * Conditions:
  16. * - You must include the above copyright notice and this permission notice in all
  17. * copies or substantial portions of the Software.
  18. *
  19. * The MIT License offers a high degree of freedom and is well-suited for both open source and
  20. * commercial applications. It places minimal restrictions on how the software can be used,
  21. * modified, and redistributed. For more details on the MIT License, please refer to the link above.
  22. */
  23. #pragma once
  24. #include "esp_spiffs.h"
  25. #include "esp_system.h"
  26. #include "pb.h" // Nanopb header for encoding (serialization)
  27. #include "pb_decode.h" // Nanopb header for decoding (deserialization)
  28. #include "pb_encode.h" // Nanopb header for encoding (serialization)
  29. #include "sys/stat.h"
  30. #include <stdlib.h>
  31. #ifdef __cplusplus
  32. #include <cstdarg> // for va_list, va_start, va_end
  33. #include <list>
  34. #include <string>
  35. #include <vector>
  36. /**
  37. * @brief Represents a file entry.
  38. */
  39. typedef struct {
  40. char type; /**< File type ('F' for regular file, 'D' for directory). */
  41. long size; /**< File size in bytes. */
  42. std::string name; /**< File or directory name. */
  43. bool restricted; /**< Restricted (system controled)*/
  44. } tools_file_entry_t;
  45. extern const std::vector<std::string> restrictedPaths;
  46. /**
  47. * @brief Retrieve a list of file entries in the specified directory.
  48. *
  49. * This function collects information about files and directories in the specified
  50. * directory path. The caller is responsible for crafting the complete path
  51. * (including any necessary SPIFFS base path).
  52. *
  53. * @param path_requested The directory path for which to list files and directories.
  54. * @return A std::list of tools_file_entry_t representing the files and directories in the specified
  55. * path.
  56. *
  57. * @note The caller is responsible for adding the SPIFFS base path if needed.
  58. *
  59. * Example usage:
  60. * @code
  61. * std::string base_path = "/spiffs";
  62. * std::string directory = "/some_directory";
  63. * std::string full_path = base_path + directory;
  64. * std::list<tools_file_entry_t> fileList = get_files_list(full_path);
  65. * for (const auto& entry : fileList) {
  66. * // Access entry.type, entry.size, and entry.name
  67. * }
  68. * @endcode
  69. */
  70. std::list<tools_file_entry_t> get_files_list(const std::string& path_requested);
  71. extern "C" {
  72. #endif
  73. void init_spiffs();
  74. extern const char* spiffs_base_path;
  75. /**
  76. * @brief Retrieves information about a file.
  77. *
  78. * This function uses the stat system call to fill a struct stat with information about the file.
  79. *
  80. * @param pfileInfo Pointer to a struct stat where file information will be stored.
  81. * @param filename The file path to get info for
  82. * @return bool True if the file information was successfully retrieved, false otherwise.
  83. */
  84. bool get_file_info(struct stat* pfileInfo, const char* filename);
  85. /**
  86. * @brief Loads the entire content of a file into memory.
  87. *
  88. * This function opens a file in binary read mode and loads its entire
  89. * content into a memory buffer. The memory for the buffer is allocated based
  90. * on the specified memory flags. The size of the loaded data is stored in the
  91. * variable pointed to by 'sz'.
  92. *
  93. * @param memflags Flags indicating the type of memory to allocate for the buffer.
  94. * This can be a combination of memory capabilities like MALLOC_CAP_SPIRAM,
  95. * MALLOC_CAP_8BIT, MALLOC_CAP_INTERNAL, MALLOC_CAP_DMA, etc.
  96. * @param sz Pointer to a size_t variable where the size of the loaded data will be stored.
  97. * Optional if loaded size is needed.
  98. * @param filename The path of the file to load. The file should exist and be readable.
  99. * @return A pointer to the allocated memory containing the file data. Returns NULL if the
  100. * file cannot be opened, if memory allocation fails, or if the file is empty.
  101. */
  102. void* load_file(uint32_t memflags, size_t* sz, const char* filename);
  103. /**
  104. * @brief Macro to load a file into PSRAM (Pseudo-Static Random Access Memory).
  105. *
  106. * This macro is a convenience wrapper for 'load_file' to load file data into PSRAM.
  107. * It is suitable for larger data that does not fit into the internal memory.
  108. *
  109. * @param pSz Pointer to a size_t variable to store the size of the loaded data.
  110. * @param filename The path of the file to load.
  111. * @return A pointer to the allocated memory in PSRAM containing the file data, or NULL on failure.
  112. */
  113. #define load_file_psram(pSz, filename) load_file(MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT, pSz, filename)
  114. /**
  115. * @brief Macro to load a file into DMA-capable internal memory.
  116. *
  117. * This macro is a convenience wrapper for 'load_file' to load file data into
  118. * DMA-capable internal memory. It is suitable for smaller data that needs to be
  119. * accessed by DMA controllers.
  120. *
  121. * @param pSz Pointer to a size_t variable to store the size of the loaded data.
  122. * @param filename The path of the file to load.
  123. * @return A pointer to the allocated memory in internal DMA-capable memory, or NULL on failure.
  124. */
  125. #define load_file_dma(pSz, filename) load_file(MALLOC_CAP_INTERNAL | MALLOC_CAP_DMA, pSz, filename)
  126. /**
  127. * @brief Erases files matching a specified pattern within a path.
  128. *
  129. * This function deletes files that match the given wildcard pattern in the specified path.
  130. * If the 'restricted' flag is set to true, the function will not delete files that match
  131. * any of the paths specified in the restricted paths list.
  132. *
  133. * @param path A string representing the path and wildcard pattern of files to be deleted.
  134. * Example: "folder/test*.txt" will attempt to delete all test*.txt files in the
  135. * 'folder' directory.
  136. * @param restricted A boolean flag indicating whether to apply restrictions on file deletion.
  137. * If true, files matching the restricted paths will not be deleted.
  138. * @return Returns true if all files matching the pattern are successfully deleted or skipped (in
  139. * case of restrictions). Returns false if there is an error in deleting any of the files or if the
  140. * directory cannot be opened.
  141. */
  142. bool erase_path(const char* path, bool restricted);
  143. /**
  144. * @brief Checks if a given filename matches any of the restricted paths.
  145. *
  146. * This function determines whether the provided filename matches any pattern specified
  147. * in the list of restricted paths. It is typically used to prevent certain files or directories
  148. * from being modified or deleted.
  149. *
  150. * @param filename The name of the file to check against the restricted paths.
  151. * @return Returns true if the filename matches any of the restricted paths.
  152. * Returns false otherwise.
  153. */
  154. bool is_restricted_path(const char* filename);
  155. bool in_file_binding(pb_istream_t* stream, pb_byte_t* buf, size_t count);
  156. bool out_file_binding(pb_ostream_t* stream, const uint8_t* buf, size_t count);
  157. /**
  158. * @brief Writes binary data to a file.
  159. *
  160. * This function writes a given array of bytes (data) to a file. The file path
  161. * is constructed from a variable number of string arguments.
  162. *
  163. * @param data Pointer to the data array to be written.
  164. * @param sz Size of the data array in bytes.
  165. * @param filename The file path to write to
  166. * @return bool True if the file is written successfully, false otherwise.
  167. *
  168. * @note This function initializes the SPIFFS before writing and logs errors.
  169. */
  170. bool write_file(const uint8_t* data, size_t sz, const char* filename);
  171. void listFiles(const char* path_requested);
  172. /**
  173. * @brief Prints the content of a specified file.
  174. *
  175. * This function reads the content of the file specified by `filename` and prints it to the standard
  176. * output. Each byte of the file is checked to determine if it is a printable ASCII character. If a
  177. * byte is printable, it is printed as a character; otherwise, it is printed in its hexadecimal
  178. * representation.
  179. *
  180. * The function utilizes `load_file_psram` to load the file into memory. It is assumed that
  181. * `load_file_psram` handles the opening and closing of the file, as well as memory allocation and
  182. * error handling.
  183. *
  184. * @param filename The path of the file to be printed. It should be a null-terminated string.
  185. *
  186. * @return Returns `true` if the file is successfully read and printed. Returns `false` if the file
  187. * cannot be opened, read, or if any other error occurs during processing.
  188. *
  189. * @note The function prints a hexadecimal representation (prefixed with \x) for non-printable
  190. * characters. For example, a byte with value 0x1F is printed as \x1F.
  191. *
  192. * @warning The function assumes that `load_file_psram` returns a `NULL` pointer if the file cannot
  193. * be loaded or if any error occurs. Ensure that `load_file_psram` adheres to this behavior.
  194. */
  195. bool cat_file(const char* filename);
  196. /**
  197. * @brief Dumps the given data to standard output.
  198. *
  199. * This function prints the provided data array to the standard output.
  200. * If the data is printable (as per the `isprint` standard function), it is printed
  201. * as a character. Otherwise, it is printed in hexadecimal format. The function
  202. * also checks for null data or zero length and reports it before returning false.
  203. *
  204. * @param pData Pointer to the data array to be dumped.
  205. * @param length The length of the data array.
  206. *
  207. * @return Returns `true` if the data is valid (not null and non-zero length), `false` otherwise.
  208. *
  209. * @note The function prints a newline character after dumping the entire data array.
  210. */
  211. bool dump_data(const uint8_t* pData, size_t length);
  212. /**
  213. * @brief Encodes a protobuf structure and dumps the encoded data to the console.
  214. *
  215. * This method takes a protobuf message structure, encodes it using NanoPB, and then dumps
  216. * the encoded data to the console. It is useful for debugging purposes to visualize
  217. * the encoded protobuf data.
  218. *
  219. * @param fields Pointer to the field descriptions array (generated by NanoPB).
  220. * @param src_struct Pointer to the structure instance to be encoded.
  221. *
  222. * @return Returns `true` if the data was successfully encoded and dumped, `false` otherwise.
  223. */
  224. bool dump_structure(const pb_msgdesc_t* fields, const void* src_struct);
  225. #ifdef __cplusplus
  226. }
  227. #endif