You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lnav/src/logfile.hh

174 lines
4.0 KiB
C++

/**
* @file logfile.hh
*/
#ifndef __logfile_hh
#define __logfile_hh
#include <stdio.h>
#include <stdint.h>
#include <sys/types.h>
#include <string>
#include <vector>
#include <algorithm>
#include "line_buffer.hh"
#include "log_format.hh"
class logfile;
/**
* Observer interface for logfile indexing progress.
*
* @see logfile
*/
class logfile_observer {
public:
virtual ~logfile_observer() { };
/**
* @param lf The logfile object that is doing the indexing.
* @param off The current offset in the file being processed.
* @param total The total size of the file.
*/
virtual void logfile_indexing(logfile &lf, off_t off, size_t total) = 0;
};
/**
* Container for the lines in a log file and some metadata.
*/
class logfile {
public:
class error {
public:
error(std::string filename, int err)
: e_filename(filename),
e_err(err) { };
std::string e_filename;
int e_err;
};
typedef std::vector<logline>::iterator iterator;
typedef std::vector<logline>::const_iterator const_iterator;
/**
* Construct a logfile with the given arguments.
*
* @param filename The name of the log file.
* @param fd The file descriptor for accessing the file or -1 if the
* constructor should open the file specified by 'filename'. The
* descriptor needs to be seekable.
*/
logfile(std::string filename, auto_fd fd = -1) throw (error);
virtual ~logfile();
/** @return The filename as given in the constructor. */
std::string &get_filename() { return this->lf_filename; };
/**
* @return The detected format, rebuild_index() must be called before this
* will return a value other than FORMAT_UNKNOWN.
*/
log_format *get_format() { return this->lf_format.get(); };
/**
* @return The last modified time of the file when the file was last
* indexed.
*/
time_t get_modified_time() { return this->lf_index_time; };
iterator begin() { return this->lf_index.begin(); }
const_iterator begin() const { return this->lf_index.begin(); }
iterator end() { return this->lf_index.end(); }
const_iterator end() const { return this->lf_index.end(); }
/** @return The number of lines in the index. */
size_t size() { return this->lf_index.size(); }
logline &operator[](int index) { return this->lf_index[index]; };
/**
* Read a line from the file.
*
* @param ll The line to read.
* @param line_out Storage to hold the line itself.
*/
void read_line(iterator ll, std::string &line_out);
/**
* Read a line from the file.
*
* @param ll The line to read.
* @return The contents of the line as a string.
*/
std::string read_line(iterator ll)
{
std::string retval;
this->read_line(ll, retval);
return retval;
};
/**
* Index any new data in the log file.
*
* @param lo The observer object that will be called regularly during
* indexing.
* @return True if any new lines were indexed.
*/
bool rebuild_index(logfile_observer *lo = NULL) throw (line_buffer::error);
bool operator<(const logfile &rhs) const
{
bool retval;
if (this->lf_index.size() == 0) {
retval = true;
}
else if (rhs.lf_index.size() == 0) {
retval = false;
}
else {
retval = this->lf_index[0].get_time() < rhs.lf_index[0].get_time();
}
return retval;
};
/** Check the invariants for this object. */
bool invariant(void)
{
assert(this->lf_filename.size() > 0);
return true;
};
protected:
/**
* Process a line from the file.
*
* @param offset The offset of the line in the file.
* @param prefix The contents of the line.
* @param len The length of the 'prefix' string.
*/
void process_prefix(off_t offset, char *prefix, int len);
std::string lf_filename;
std::auto_ptr<log_format> lf_format;
std::vector<logline> lf_index;
time_t lf_index_time;
size_t lf_index_size;
line_buffer lf_line_buffer;
};
#endif