2019-08-11 19:34:29 +00:00
|
|
|
#ifndef FISH_HISTORY_FILE_H
|
|
|
|
#define FISH_HISTORY_FILE_H
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include <sys/mman.h>
|
2019-08-11 19:34:29 +00:00
|
|
|
|
|
|
|
#include <cassert>
|
2019-10-16 18:55:40 +00:00
|
|
|
#include <ctime>
|
2019-08-11 19:34:29 +00:00
|
|
|
#include <memory>
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "maybe.h"
|
2019-08-11 19:34:29 +00:00
|
|
|
|
2019-08-12 16:16:24 +00:00
|
|
|
class history_item_t;
|
2021-05-08 23:02:49 +00:00
|
|
|
class history_tests_t;
|
2019-08-12 16:16:24 +00:00
|
|
|
|
2019-08-11 19:34:29 +00:00
|
|
|
// History file types.
|
|
|
|
enum history_file_type_t { history_type_fish_2_0, history_type_fish_1_x };
|
|
|
|
|
|
|
|
/// history_file_contents_t holds the read-only contents of a file.
|
|
|
|
class history_file_contents_t {
|
|
|
|
public:
|
|
|
|
/// Construct a history file contents from a file descriptor. The file descriptor is not closed.
|
|
|
|
static std::unique_ptr<history_file_contents_t> create(int fd);
|
|
|
|
|
|
|
|
/// Decode an item at a given offset.
|
|
|
|
history_item_t decode_item(size_t offset) const;
|
|
|
|
|
|
|
|
/// Support for iterating item offsets.
|
|
|
|
/// The cursor should initially be 0.
|
|
|
|
/// If cutoff is nonzero, skip items whose timestamp is newer than cutoff.
|
|
|
|
/// \return the offset of the next item, or none() on end.
|
2020-04-03 02:39:29 +00:00
|
|
|
maybe_t<size_t> offset_of_next_item(size_t *cursor, time_t cutoff) const;
|
2019-08-11 19:34:29 +00:00
|
|
|
|
|
|
|
/// Get the file type.
|
|
|
|
history_file_type_t type() const { return type_; }
|
|
|
|
|
|
|
|
/// Get the size of the contents.
|
|
|
|
size_t length() const { return length_; }
|
|
|
|
|
|
|
|
/// Return a pointer to the beginning.
|
|
|
|
const char *begin() const { return address_at(0); }
|
|
|
|
|
|
|
|
/// Return a pointer to one-past-the-end.
|
|
|
|
const char *end() const { return address_at(length_); }
|
|
|
|
|
|
|
|
/// Access the address at a given offset.
|
|
|
|
const char *address_at(size_t offset) const {
|
|
|
|
assert(offset <= length_ && "Invalid offset");
|
|
|
|
return start_ + offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
~history_file_contents_t();
|
|
|
|
|
|
|
|
private:
|
2021-05-08 20:51:09 +00:00
|
|
|
// A type wrapping up the logic around mmap and munmap.
|
|
|
|
struct mmap_region_t;
|
2021-05-10 23:49:11 +00:00
|
|
|
const std::unique_ptr<mmap_region_t> region_;
|
2019-08-11 19:34:29 +00:00
|
|
|
|
2021-05-08 20:51:09 +00:00
|
|
|
// The memory mapped pointer and length.
|
2021-05-08 23:02:49 +00:00
|
|
|
// The ptr aliases our region. The length may be slightly smaller, if there is a trailing
|
|
|
|
// incomplete history item.
|
2021-05-08 20:51:09 +00:00
|
|
|
const char *const start_;
|
2019-08-11 19:34:29 +00:00
|
|
|
const size_t length_;
|
|
|
|
|
|
|
|
// The type of the mapped file.
|
2021-05-08 20:11:38 +00:00
|
|
|
// This is set at construction and not changed after.
|
|
|
|
history_file_type_t type_{};
|
2019-08-11 19:34:29 +00:00
|
|
|
|
2021-05-08 23:02:49 +00:00
|
|
|
// Private constructors; use the static create() function.
|
2021-05-08 20:51:09 +00:00
|
|
|
explicit history_file_contents_t(std::unique_ptr<mmap_region_t> region);
|
2021-05-08 23:02:49 +00:00
|
|
|
history_file_contents_t(const char *start, size_t length);
|
2021-05-08 20:11:38 +00:00
|
|
|
|
|
|
|
// Try to infer the file type to populate type_.
|
|
|
|
// \return true on success, false on error.
|
|
|
|
bool infer_file_type();
|
2019-08-11 19:34:29 +00:00
|
|
|
|
|
|
|
history_file_contents_t(history_file_contents_t &&) = delete;
|
|
|
|
void operator=(history_file_contents_t &&) = delete;
|
|
|
|
};
|
|
|
|
|
2019-08-12 16:16:24 +00:00
|
|
|
/// Append a history item to a buffer, in preparation for outputting it to the history file.
|
|
|
|
void append_history_item_to_buffer(const history_item_t &item, std::string *buffer);
|
|
|
|
|
2019-08-11 19:34:29 +00:00
|
|
|
#endif
|