#include "keys_dict.h"
 
#include <storage/storage.h>
#include <flipper_format/flipper_format.h>
#include <toolbox/stream/file_stream.h>
#include <toolbox/stream/buffered_file_stream.h>
#include <toolbox/args.h>
 
#define TAG "KeysDict"
 
struct KeysDict {
    Stream* stream;
    size_t key_size;
    size_t key_size_symbols;
    size_t total_keys;
};
 
static inline void keys_dict_add_ending_new_line(KeysDict* instance) {
    if(stream_seek(instance->stream, -1, StreamOffsetFromEnd)) {
        uint8_t last_char = 0;
 
        // Check if the last char is new line or add a new line
        if(stream_read(instance->stream, &last_char, 1) == 1 && last_char != '\n') {
            FURI_LOG_D(TAG, "Adding new line ending");
            stream_write_char(instance->stream, '\n');
        }
 
        stream_rewind(instance->stream);
    }
}
 
static bool keys_dict_read_key_line(KeysDict* instance, FuriString* line, bool* is_endfile) {
    if(stream_read_line(instance->stream, line) == false) {
        *is_endfile = true;
    }
 
    else {
        FURI_LOG_T(
            TAG, "Read line: %s, len: %zu", furi_string_get_cstr(line), furi_string_size(line));
 
        bool is_comment = furi_string_get_char(line, 0) == '#';
 
        if(!is_comment) {
            furi_string_left(line, instance->key_size_symbols - 1);
        }
 
        bool is_correct_size = furi_string_size(line) == instance->key_size_symbols - 1;
 
        return !is_comment && is_correct_size;
    }
 
    return false;
}
 
bool keys_dict_check_presence(const char* path) {
    furi_assert(path);
 
    Storage* storage = furi_record_open(RECORD_STORAGE);
 
    bool dict_present = storage_common_stat(storage, path, NULL) == FSE_OK;
 
    furi_record_close(RECORD_STORAGE);
 
    return dict_present;
}
 
KeysDict* keys_dict_alloc(const char* path, KeysDictMode mode, size_t key_size) {
    furi_assert(path);
    furi_assert(key_size > 0);
 
    KeysDict* instance = malloc(sizeof(KeysDict));
 
    Storage* storage = furi_record_open(RECORD_STORAGE);
    furi_assert(storage);
 
    instance->stream = buffered_file_stream_alloc(storage);
    furi_assert(instance->stream);
 
    FS_OpenMode open_mode = (mode == KeysDictModeOpenAlways) ? FSOM_OPEN_ALWAYS :
                                                               FSOM_OPEN_EXISTING;
 
    // Byte = 2 symbols + 1 end of line
    instance->key_size = key_size;
    instance->key_size_symbols = key_size * 2 + 1;
 
    instance->total_keys = 0;
 
    bool file_exists =
        buffered_file_stream_open(instance->stream, path, FSAM_READ_WRITE, open_mode);
 
    if(!file_exists) {
        buffered_file_stream_close(instance->stream);
    } else {
        // Eventually add new line character in the last line to avoid skipping keys
        keys_dict_add_ending_new_line(instance);
    }
 
    FuriString* line = furi_string_alloc();
 
    bool is_endfile = false;
 
    // In this loop we only count the entries in the file
    // We prefer not to load the whole file in memory for space reasons
    while(file_exists && !is_endfile) {
        bool read_key = keys_dict_read_key_line(instance, line, &is_endfile);
        if(read_key) {
            instance->total_keys++;
        }
    }
    stream_rewind(instance->stream);
    FURI_LOG_I(TAG, "Loaded dictionary with %u keys", instance->total_keys);
 
    furi_string_free(line);
 
    return instance;
}
 
void keys_dict_free(KeysDict* instance) {
    furi_assert(instance);
    furi_assert(instance->stream);
 
    buffered_file_stream_close(instance->stream);
    stream_free(instance->stream);
    free(instance);
 
    furi_record_close(RECORD_STORAGE);
}
 
static void keys_dict_int_to_str(KeysDict* instance, const uint8_t* key_int, FuriString* key_str) {
    furi_assert(instance);
    furi_assert(key_str);
    furi_assert(key_int);
 
    furi_string_reset(key_str);
 
    for(size_t i = 0; i < instance->key_size; i++)
        furi_string_cat_printf(key_str, "%02X", key_int[i]);
}
 
static void keys_dict_str_to_int(KeysDict* instance, FuriString* key_str, uint64_t* key_int) {
    furi_assert(instance);
    furi_assert(key_str);
    furi_assert(key_int);
 
    uint8_t key_byte_tmp;
    char h, l;
 
    *key_int = 0ULL;
 
    for(size_t i = 0; i < instance->key_size_symbols - 1; i += 2) {
        h = furi_string_get_char(key_str, i);
        l = furi_string_get_char(key_str, i + 1);
 
        args_char_to_hex(h, l, &key_byte_tmp);
        *key_int |= (uint64_t)key_byte_tmp << (8 * (instance->key_size - 1 - i / 2));
    }
}
 
size_t keys_dict_get_total_keys(KeysDict* instance) {
    furi_assert(instance);
 
    return instance->total_keys;
}
 
bool keys_dict_rewind(KeysDict* instance) {
    furi_assert(instance);
    furi_assert(instance->stream);
 
    return stream_rewind(instance->stream);
}
 
static bool keys_dict_get_next_key_str(KeysDict* instance, FuriString* key) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(key);
 
    bool key_read = false;
    bool is_endfile = false;
 
    furi_string_reset(key);
 
    while(!key_read && !is_endfile) key_read = keys_dict_read_key_line(instance, key, &is_endfile);
 
    return key_read;
}
 
bool keys_dict_get_next_key(KeysDict* instance, uint8_t* key, size_t key_size) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(instance->key_size == key_size);
    furi_assert(key);
 
    FuriString* temp_key = furi_string_alloc();
 
    bool key_read = keys_dict_get_next_key_str(instance, temp_key);
 
    if(key_read) {
        size_t tmp_len = key_size;
        uint64_t key_int = 0;
 
        keys_dict_str_to_int(instance, temp_key, &key_int);
 
        while(tmp_len--) {
            key[tmp_len] = (uint8_t)key_int;
            key_int >>= 8;
        }
    }
 
    furi_string_free(temp_key);
    return key_read;
}
 
static bool keys_dict_is_key_present_str(KeysDict* instance, FuriString* key) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(key);
 
    FuriString* line = furi_string_alloc();
 
    bool is_endfile = false;
    bool line_found = false;
 
    uint32_t actual_pos = stream_tell(instance->stream);
    stream_rewind(instance->stream);
 
    while(!line_found && !is_endfile)
        line_found = // The line is found if the line was read and the key is equal to the line
            (keys_dict_read_key_line(instance, line, &is_endfile)) &&
            (furi_string_equal(key, line));
 
    furi_string_free(line);
 
    // Restore the position of the stream
    stream_seek(instance->stream, actual_pos, StreamOffsetFromStart);
 
    return line_found;
}
 
bool keys_dict_is_key_present(KeysDict* instance, const uint8_t* key, size_t key_size) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(instance->key_size == key_size);
    furi_assert(key);
 
    FuriString* temp_key = furi_string_alloc();
 
    keys_dict_int_to_str(instance, key, temp_key);
    bool key_found = keys_dict_is_key_present_str(instance, temp_key);
    furi_string_free(temp_key);
 
    return key_found;
}
 
static bool keys_dict_add_key_str(KeysDict* instance, FuriString* key) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(key);
 
    furi_string_cat_str(key, "\n");
 
    bool key_added = false;
 
    uint32_t actual_pos = stream_tell(instance->stream);
 
    if(stream_seek(instance->stream, 0, StreamOffsetFromEnd) &&
       stream_insert_string(instance->stream, key)) {
        instance->total_keys++;
        key_added = true;
    }
 
    stream_seek(instance->stream, actual_pos, StreamOffsetFromStart);
 
    return key_added;
}
 
bool keys_dict_add_key(KeysDict* instance, const uint8_t* key, size_t key_size) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(instance->key_size == key_size);
    furi_assert(key);
 
    FuriString* temp_key = furi_string_alloc();
    furi_assert(temp_key);
 
    keys_dict_int_to_str(instance, key, temp_key);
    bool key_added = keys_dict_add_key_str(instance, temp_key);
 
    FURI_LOG_I(TAG, "Added key %s", furi_string_get_cstr(temp_key));
 
    furi_string_free(temp_key);
 
    return key_added;
}
 
bool keys_dict_delete_key(KeysDict* instance, const uint8_t* key, size_t key_size) {
    furi_assert(instance);
    furi_assert(instance->stream);
    furi_assert(instance->key_size == key_size);
    furi_assert(key);
 
    bool key_removed = false;
    bool is_endfile = false;
 
    uint8_t* temp_key = malloc(key_size);
 
    stream_rewind(instance->stream);
 
    while(!key_removed && !is_endfile) {
        if(!keys_dict_get_next_key(instance, temp_key, key_size)) {
            break;
        }
 
        if(memcmp(temp_key, key, key_size) == 0) {
            stream_seek(instance->stream, -instance->key_size_symbols, StreamOffsetFromCurrent);
            if(stream_delete(instance->stream, instance->key_size_symbols) == false) {
                break;
            }
            instance->total_keys--;
            key_removed = true;
        }
    }
 
    FuriString* tmp = furi_string_alloc();
 
    keys_dict_int_to_str(instance, key, tmp);
 
    FURI_LOG_I(TAG, "Removed key %s", furi_string_get_cstr(tmp));
 
    furi_string_free(tmp);
 
    stream_rewind(instance->stream);
    free(temp_key);
 
    return key_removed;
}

V576 Incorrect format. Consider checking the fourth actual argument of the 'furi_log_print_format' function. The integer argument of 32-bit size is expected.

V560 A part of conditional expression is always true: !is_endfile.