2021-11-04 10:06:13 +00:00
|
|
|
#include "flipper_file_helper.h"
|
|
|
|
|
|
|
|
const char* flipper_file_filetype_key = "Filetype";
|
|
|
|
const char* flipper_file_version_key = "Version";
|
|
|
|
const char flipper_file_delimiter = ':';
|
|
|
|
const char flipper_file_comment = '#';
|
|
|
|
|
|
|
|
#ifdef __linux__
|
|
|
|
const char* flipper_file_scratchpad = ".scratch.pad";
|
|
|
|
#else
|
|
|
|
const char* flipper_file_scratchpad = "/any/.scratch.pad";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bool flipper_file_read_valid_key(File* file, string_t key) {
|
2021-11-15 17:09:40 +00:00
|
|
|
string_reset(key);
|
2021-11-04 10:06:13 +00:00
|
|
|
bool found = false;
|
|
|
|
bool error = false;
|
|
|
|
const uint8_t buffer_size = 32;
|
|
|
|
uint8_t buffer[buffer_size];
|
|
|
|
bool accumulate = true;
|
|
|
|
bool new_line = true;
|
|
|
|
|
|
|
|
while(true) {
|
|
|
|
uint16_t bytes_were_read = storage_file_read(file, buffer, buffer_size);
|
|
|
|
if(bytes_were_read == 0) break;
|
|
|
|
|
|
|
|
for(uint16_t i = 0; i < bytes_were_read; i++) {
|
|
|
|
if(buffer[i] == flipper_file_eoln) {
|
|
|
|
// EOL found, clean data, start accumulating data and set the new_line flag
|
2021-11-15 17:09:40 +00:00
|
|
|
string_reset(key);
|
2021-11-04 10:06:13 +00:00
|
|
|
accumulate = true;
|
|
|
|
new_line = true;
|
|
|
|
} else if(buffer[i] == flipper_file_eolr) {
|
|
|
|
// Ignore
|
|
|
|
} else if(buffer[i] == flipper_file_comment && new_line) {
|
|
|
|
// if there is a comment character and we are at the beginning of a new line
|
|
|
|
// do not accumulate comment data and reset the new_line flag
|
|
|
|
accumulate = false;
|
|
|
|
new_line = false;
|
|
|
|
} else if(buffer[i] == flipper_file_delimiter) {
|
|
|
|
if(new_line) {
|
|
|
|
// we are on a "new line" and found the delimiter
|
|
|
|
// this can only be if we have previously found some kind of key, so
|
|
|
|
// clear the data, set the flag that we no longer want to accumulate data
|
|
|
|
// and reset the new_line flag
|
2021-11-15 17:09:40 +00:00
|
|
|
string_reset(key);
|
2021-11-04 10:06:13 +00:00
|
|
|
accumulate = false;
|
|
|
|
new_line = false;
|
|
|
|
} else {
|
|
|
|
// parse the delimiter only if we are accumulating data
|
|
|
|
if(accumulate) {
|
|
|
|
// we found the delimiter, move the rw pointer to the correct location
|
|
|
|
// and signal that we have found something
|
|
|
|
if(!file_helper_seek(file, i - bytes_were_read)) {
|
|
|
|
error = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// just new symbol, reset the new_line flag
|
|
|
|
new_line = false;
|
|
|
|
if(accumulate) {
|
|
|
|
// and accumulate data if we want
|
|
|
|
string_push_back(key, buffer[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(found || error) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool flipper_file_seek_to_key(File* file, const char* key) {
|
|
|
|
bool found = false;
|
|
|
|
string_t readed_key;
|
|
|
|
|
|
|
|
string_init(readed_key);
|
|
|
|
|
|
|
|
while(!storage_file_eof(file)) {
|
|
|
|
if(flipper_file_read_valid_key(file, readed_key)) {
|
|
|
|
if(string_cmp_str(readed_key, key) == 0) {
|
|
|
|
if(!file_helper_seek(file, 2)) break;
|
|
|
|
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
string_clear(readed_key);
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool flipper_file_write_key(File* file, const char* key) {
|
|
|
|
bool result = false;
|
|
|
|
|
|
|
|
do {
|
|
|
|
result = file_helper_write(file, key, strlen(key));
|
|
|
|
if(!result) break;
|
|
|
|
|
|
|
|
const char delimiter_buffer[2] = {flipper_file_delimiter, ' '};
|
|
|
|
result = file_helper_write(file, delimiter_buffer, sizeof(delimiter_buffer));
|
|
|
|
} while(false);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool flipper_file_get_scratchpad_name(const char** name) {
|
|
|
|
// TODO do not rewrite existing file
|
|
|
|
*name = flipper_file_scratchpad;
|
|
|
|
return true;
|
|
|
|
}
|