2013-05-31 15:01:31 +00:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2013, Timothy Stack
|
|
|
|
*
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright notice, this
|
|
|
|
* list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
* * Neither the name of Timothy Stack nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ''AS IS'' AND ANY
|
|
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
|
|
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* @file session_data.cc
|
|
|
|
*/
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <glob.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
|
2014-02-04 06:29:59 +00:00
|
|
|
#include "spookyhash/SpookyV2.h"
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "yajlpp.hh"
|
|
|
|
#include "lnav.hh"
|
2013-06-03 04:11:17 +00:00
|
|
|
#include "logfile.hh"
|
2013-07-14 04:31:59 +00:00
|
|
|
#include "sql_util.hh"
|
2013-05-31 15:01:31 +00:00
|
|
|
#include "lnav_util.hh"
|
2013-06-06 14:01:32 +00:00
|
|
|
#include "lnav_config.hh"
|
2013-05-31 15:01:31 +00:00
|
|
|
#include "session_data.hh"
|
2015-07-08 13:35:36 +00:00
|
|
|
#include "command_executor.hh"
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
static const char *LOG_METADATA_NAME = "log_metadata.db";
|
|
|
|
|
|
|
|
static const char *BOOKMARK_TABLE_DEF =
|
|
|
|
"CREATE TABLE IF NOT EXISTS bookmarks (\n"
|
|
|
|
" log_time datetime,\n"
|
|
|
|
" log_format varchar(64),\n"
|
|
|
|
" log_hash varchar(128),\n"
|
|
|
|
" session_time integer,\n"
|
|
|
|
" part_name text,\n"
|
|
|
|
" access_time datetime DEFAULT CURRENT_TIMESTAMP,\n"
|
|
|
|
"\n"
|
|
|
|
" PRIMARY KEY (log_time, log_format, log_hash, session_time)\n"
|
|
|
|
");\n"
|
|
|
|
"CREATE TABLE IF NOT EXISTS time_offset (\n"
|
|
|
|
" log_time datetime,\n"
|
|
|
|
" log_format varchar(64),\n"
|
|
|
|
" log_hash varchar(128),\n"
|
|
|
|
" session_time integer,\n"
|
|
|
|
" offset_sec integer,\n"
|
|
|
|
" offset_usec integer,\n"
|
|
|
|
" access_time datetime DEFAULT CURRENT_TIMESTAMP,\n"
|
|
|
|
"\n"
|
|
|
|
" PRIMARY KEY (log_time, log_format, log_hash, session_time)\n"
|
|
|
|
");\n";
|
|
|
|
|
|
|
|
static const char *BOOKMARK_LRU_STMT =
|
|
|
|
"DELETE FROM bookmarks WHERE access_time <= "
|
|
|
|
" (SELECT access_time FROM bookmarks "
|
|
|
|
" ORDER BY access_time DESC LIMIT 1 OFFSET 50000)";
|
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
static const size_t MAX_SESSIONS = 8;
|
2013-06-13 14:18:59 +00:00
|
|
|
static const size_t MAX_SESSION_FILE_COUNT = 256;
|
|
|
|
|
|
|
|
typedef std::vector<std::pair<int, string> > timestamped_list_t;
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
static std::vector<content_line_t> marked_session_lines;
|
|
|
|
static std::vector<content_line_t> offset_session_lines;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2014-03-19 05:47:36 +00:00
|
|
|
static bool bind_line(sqlite3 *db,
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_stmt *stmt,
|
|
|
|
content_line_t cl,
|
|
|
|
time_t session_time)
|
2013-06-03 04:11:17 +00:00
|
|
|
{
|
2013-07-14 04:31:59 +00:00
|
|
|
logfile_sub_source &lss = lnav_data.ld_log_source;
|
|
|
|
logfile::iterator line_iter;
|
|
|
|
logfile *lf;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
lf = lss.find(cl);
|
2014-03-19 05:47:36 +00:00
|
|
|
|
|
|
|
if (lf == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
line_iter = lf->begin() + cl;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
char timestamp[64];
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sql_strftime(timestamp, sizeof(timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt, 1,
|
|
|
|
timestamp, strlen(timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
2014-03-19 05:47:36 +00:00
|
|
|
return false;
|
2013-07-14 04:31:59 +00:00
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2015-06-03 13:36:58 +00:00
|
|
|
intern_string_t format_name = lf->get_format()->get_name();
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt, 2,
|
2015-06-03 13:36:58 +00:00
|
|
|
format_name.get(), format_name.size(),
|
2013-07-14 04:31:59 +00:00
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log format -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
2014-03-19 05:47:36 +00:00
|
|
|
return false;
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
|
2015-12-22 13:27:36 +00:00
|
|
|
shared_buffer_ref sbr;
|
|
|
|
lf->read_line(line_iter, sbr);
|
|
|
|
std::string line_hash = hash_bytes(sbr.get_data(), sbr.length(),
|
|
|
|
&cl, sizeof(cl),
|
|
|
|
NULL);
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt, 3,
|
|
|
|
line_hash.c_str(), line_hash.length(),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
2014-03-19 05:47:36 +00:00
|
|
|
return false;
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_int64(stmt, 4, session_time) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind session time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
2014-03-19 05:47:36 +00:00
|
|
|
return false;
|
2013-07-14 04:31:59 +00:00
|
|
|
}
|
2014-03-19 05:47:36 +00:00
|
|
|
|
|
|
|
return true;
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
|
2013-06-13 14:18:59 +00:00
|
|
|
struct session_file_info {
|
|
|
|
session_file_info(int timestamp,
|
|
|
|
const string &id,
|
|
|
|
const string &path)
|
2013-06-16 01:07:50 +00:00
|
|
|
: sfi_timestamp(timestamp), sfi_id(id), sfi_path(path) {};
|
2013-06-13 14:18:59 +00:00
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
bool operator<(const session_file_info &other) const
|
|
|
|
{
|
|
|
|
if (this->sfi_timestamp < other.sfi_timestamp) {
|
2013-06-13 14:18:59 +00:00
|
|
|
return true;
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
|
|
|
if (this->sfi_path < other.sfi_path) {
|
2013-06-13 14:18:59 +00:00
|
|
|
return true;
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-06-13 14:18:59 +00:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
int sfi_timestamp;
|
2013-06-13 14:18:59 +00:00
|
|
|
string sfi_id;
|
|
|
|
string sfi_path;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void cleanup_session_data(void)
|
|
|
|
{
|
2013-06-16 01:07:50 +00:00
|
|
|
static_root_mem<glob_t, globfree> session_file_list;
|
2013-06-13 14:18:59 +00:00
|
|
|
std::list<struct session_file_info> session_info_list;
|
|
|
|
map<string, int> session_count;
|
2013-06-16 01:07:50 +00:00
|
|
|
string session_file_pattern;
|
2013-06-13 14:18:59 +00:00
|
|
|
|
|
|
|
session_file_pattern = dotlnav_path("*-*.ts*.json");
|
|
|
|
|
|
|
|
if (glob(session_file_pattern.c_str(),
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
session_file_list.inout()) == 0) {
|
|
|
|
for (size_t lpc = 0; lpc < session_file_list->gl_pathc; lpc++) {
|
|
|
|
const char *path = session_file_list->gl_pathv[lpc];
|
2013-06-16 01:07:50 +00:00
|
|
|
char hash_id[64];
|
|
|
|
int timestamp;
|
2013-06-13 14:18:59 +00:00
|
|
|
const char *base;
|
|
|
|
|
|
|
|
base = strrchr(path, '/') + 1;
|
|
|
|
if (sscanf(base, "file-%63[^.].ts%d.json",
|
|
|
|
hash_id, ×tamp) == 2) {
|
|
|
|
session_count[hash_id] += 1;
|
|
|
|
session_info_list.push_back(session_file_info(
|
2013-06-16 01:07:50 +00:00
|
|
|
timestamp, hash_id, path));
|
2013-06-13 14:18:59 +00:00
|
|
|
}
|
|
|
|
if (sscanf(base,
|
|
|
|
"view-info-%63[^.].ts%d.ppid%*d.json",
|
|
|
|
hash_id,
|
|
|
|
×tamp) == 2) {
|
|
|
|
session_count[hash_id] += 1;
|
|
|
|
session_info_list.push_back(session_file_info(
|
2013-06-16 01:07:50 +00:00
|
|
|
timestamp, hash_id, path));
|
2013-06-13 14:18:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
session_info_list.sort();
|
|
|
|
|
2013-09-10 13:20:37 +00:00
|
|
|
size_t session_loops = 0;
|
2013-08-25 14:12:41 +00:00
|
|
|
|
2013-06-13 14:18:59 +00:00
|
|
|
while (session_info_list.size() > MAX_SESSION_FILE_COUNT) {
|
|
|
|
const session_file_info &front = session_info_list.front();
|
|
|
|
|
2013-08-25 14:12:41 +00:00
|
|
|
session_loops += 1;
|
|
|
|
if (session_loops < MAX_SESSION_FILE_COUNT &&
|
|
|
|
session_count[front.sfi_id] == 1) {
|
2013-06-13 14:18:59 +00:00
|
|
|
session_info_list.splice(session_info_list.end(),
|
|
|
|
session_info_list,
|
|
|
|
session_info_list.begin());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (remove(front.sfi_path.c_str()) != 0) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"Unable to remove session file: %s -- %s\n",
|
2013-06-13 14:18:59 +00:00
|
|
|
front.sfi_path.c_str(),
|
|
|
|
strerror(errno));
|
|
|
|
}
|
|
|
|
session_count[front.sfi_id] -= 1;
|
|
|
|
session_info_list.pop_front();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
session_info_list.sort();
|
|
|
|
|
|
|
|
while (session_info_list.size() > MAX_SESSION_FILE_COUNT) {
|
|
|
|
const session_file_info &front = session_info_list.front();
|
|
|
|
|
|
|
|
if (remove(front.sfi_path.c_str()) != 0) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"Unable to remove session file: %s -- %s\n",
|
2013-06-13 14:18:59 +00:00
|
|
|
front.sfi_path.c_str(),
|
|
|
|
strerror(errno));
|
|
|
|
}
|
|
|
|
session_count[front.sfi_id] -= 1;
|
|
|
|
session_info_list.pop_front();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
void init_session(void)
|
|
|
|
{
|
2014-02-04 17:26:25 +00:00
|
|
|
byte_array<2, uint64> hash;
|
2014-02-04 06:29:59 +00:00
|
|
|
SpookyHash context;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
lnav_data.ld_session_time = time(NULL);
|
|
|
|
|
2014-02-04 06:29:59 +00:00
|
|
|
context.Init(0, 0);
|
|
|
|
hash_updater updater(&context);
|
2016-03-12 22:12:23 +00:00
|
|
|
for (map<string, logfile_open_options>::iterator iter = lnav_data.ld_file_names.begin();
|
|
|
|
iter != lnav_data.ld_file_names.end();
|
|
|
|
++iter) {
|
|
|
|
updater(iter->first);
|
|
|
|
}
|
2014-02-04 17:26:25 +00:00
|
|
|
context.Final(hash.out(0), hash.out(1));
|
2013-05-31 15:01:31 +00:00
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
lnav_data.ld_session_id = hash.to_string();
|
2014-03-07 13:20:49 +00:00
|
|
|
|
|
|
|
log_info("init_session: time=%d; id=%s", lnav_data.ld_session_time,
|
|
|
|
lnav_data.ld_session_id.c_str());
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void scan_sessions(void)
|
|
|
|
{
|
2013-06-16 01:07:50 +00:00
|
|
|
std::list<session_pair_t> &session_file_names =
|
|
|
|
lnav_data.ld_session_file_names;
|
|
|
|
|
|
|
|
static_root_mem<glob_t, globfree> view_info_list;
|
2013-06-02 21:20:15 +00:00
|
|
|
std::list<session_pair_t>::iterator iter;
|
2013-06-16 01:07:50 +00:00
|
|
|
char view_info_pattern_base[128];
|
2013-05-31 15:01:31 +00:00
|
|
|
string view_info_pattern;
|
2013-06-02 21:20:15 +00:00
|
|
|
string old_session_name;
|
2013-06-16 01:07:50 +00:00
|
|
|
int index;
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2013-06-13 14:18:59 +00:00
|
|
|
cleanup_session_data();
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
if (lnav_data.ld_session_file_index >= 0 &&
|
|
|
|
lnav_data.ld_session_file_index < (int)session_file_names.size()) {
|
|
|
|
iter = session_file_names.begin();
|
|
|
|
|
|
|
|
advance(iter, lnav_data.ld_session_file_index);
|
|
|
|
old_session_name = iter->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
session_file_names.clear();
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
snprintf(view_info_pattern_base, sizeof(view_info_pattern_base),
|
|
|
|
"view-info-%s.*.json",
|
|
|
|
lnav_data.ld_session_id.c_str());
|
|
|
|
view_info_pattern = dotlnav_path(view_info_pattern_base);
|
2013-06-16 01:07:50 +00:00
|
|
|
if (glob(view_info_pattern.c_str(), 0, NULL,
|
|
|
|
view_info_list.inout()) == 0) {
|
2013-05-31 15:01:31 +00:00
|
|
|
for (size_t lpc = 0; lpc < view_info_list->gl_pathc; lpc++) {
|
2013-06-02 21:20:15 +00:00
|
|
|
const char *path = view_info_list->gl_pathv[lpc];
|
2013-06-16 01:07:50 +00:00
|
|
|
int timestamp, ppid;
|
2013-06-03 04:22:16 +00:00
|
|
|
const char *base;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
base = strrchr(path, '/') + 1;
|
2013-05-31 15:01:31 +00:00
|
|
|
if (sscanf(base,
|
|
|
|
"view-info-%*[^.].ts%d.ppid%d.json",
|
|
|
|
×tamp,
|
|
|
|
&ppid) == 2) {
|
2013-06-02 21:20:15 +00:00
|
|
|
ppid_time_pair_t ptp;
|
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
ptp.first = (ppid == getppid()) ? 1 : 0;
|
2013-06-02 21:20:15 +00:00
|
|
|
ptp.second = timestamp;
|
|
|
|
session_file_names.push_back(make_pair(ptp, path));
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
session_file_names.sort();
|
|
|
|
|
|
|
|
while (session_file_names.size() > MAX_SESSIONS) {
|
2013-06-13 14:18:59 +00:00
|
|
|
const std::string &name = session_file_names.front().second;
|
|
|
|
|
|
|
|
if (remove(name.c_str()) != 0) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"Unable to remove session: %s -- %s\n",
|
2013-06-13 14:18:59 +00:00
|
|
|
name.c_str(),
|
|
|
|
strerror(errno));
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
session_file_names.pop_front();
|
|
|
|
}
|
|
|
|
|
|
|
|
lnav_data.ld_session_file_index = ((int)session_file_names.size()) - 1;
|
|
|
|
|
|
|
|
for (index = 0, iter = session_file_names.begin();
|
|
|
|
iter != session_file_names.end();
|
|
|
|
index++, ++iter) {
|
|
|
|
if (iter->second == old_session_name) {
|
|
|
|
lnav_data.ld_session_file_index = index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
static void load_time_bookmarks(void)
|
|
|
|
{
|
|
|
|
logfile_sub_source &lss = lnav_data.ld_log_source;
|
|
|
|
std::map<content_line_t, bookmark_metadata> &bm_meta = lss.get_user_bookmark_metadata();
|
|
|
|
auto_mem<sqlite3, sqlite_close_wrapper> db;
|
|
|
|
string db_path = dotlnav_path(LOG_METADATA_NAME);
|
|
|
|
auto_mem<sqlite3_stmt> stmt(sqlite3_finalize);
|
|
|
|
logfile_sub_source::iterator file_iter;
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2014-03-07 13:20:49 +00:00
|
|
|
log_info("loading bookmark db: %s", db_path.c_str());
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_open(db_path.c_str(), db.out()) != SQLITE_OK) {
|
|
|
|
return;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"SELECT *,session_time=? as same_session FROM bookmarks WHERE "
|
|
|
|
" log_time between ? and ? and log_format = ? "
|
|
|
|
" ORDER BY same_session DESC, session_time DESC",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare bookmark select statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (file_iter = lnav_data.ld_log_source.begin();
|
|
|
|
file_iter != lnav_data.ld_log_source.end();
|
|
|
|
++file_iter) {
|
|
|
|
char low_timestamp[64], high_timestamp[64];
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-07-14 04:31:59 +00:00
|
|
|
content_line_t base_content_line;
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (lf == NULL)
|
|
|
|
continue;
|
2013-06-16 01:07:50 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
base_content_line = lss.get_file_base_content_line(file_iter);
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
logfile::iterator line_iter = lf->begin();
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sql_strftime(low_timestamp, sizeof(low_timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_int64(stmt.in(), 1, lnav_data.ld_session_load_time) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind session time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt.in(), 2,
|
|
|
|
low_timestamp, strlen(low_timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind low log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
line_iter = lf->end();
|
|
|
|
--line_iter;
|
|
|
|
sql_strftime(high_timestamp, sizeof(high_timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_text(stmt.in(), 3,
|
|
|
|
high_timestamp, strlen(high_timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind high log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2015-06-03 13:36:58 +00:00
|
|
|
intern_string_t format_name = lf->get_format()->get_name();
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt.in(), 4,
|
2015-06-03 13:36:58 +00:00
|
|
|
format_name.get(), format_name.size(),
|
2013-07-14 04:31:59 +00:00
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log format -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
date_time_scanner dts;
|
|
|
|
bool done = false;
|
|
|
|
string line;
|
|
|
|
int64_t last_mark_time = -1;
|
2013-06-19 04:56:06 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
while (!done) {
|
|
|
|
int rc = sqlite3_step(stmt.in());
|
2013-06-19 04:56:06 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
switch (rc) {
|
|
|
|
case SQLITE_OK:
|
|
|
|
case SQLITE_DONE:
|
|
|
|
done = true;
|
|
|
|
break;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
case SQLITE_ROW: {
|
|
|
|
const char *log_time = (const char *)sqlite3_column_text(stmt.in(), 0);
|
|
|
|
const char *log_hash = (const char *)sqlite3_column_text(stmt.in(), 2);
|
|
|
|
const char *part_name = (const char *)sqlite3_column_text(stmt.in(), 4);
|
|
|
|
int64_t mark_time = sqlite3_column_int64(stmt.in(), 3);
|
|
|
|
struct timeval log_tv;
|
2014-06-18 04:29:42 +00:00
|
|
|
struct exttm log_tm;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (last_mark_time == -1) {
|
|
|
|
last_mark_time = mark_time;
|
|
|
|
}
|
|
|
|
else if (last_mark_time != mark_time) {
|
|
|
|
done = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-11-06 15:29:20 +00:00
|
|
|
if (part_name == NULL) {
|
2013-07-14 04:31:59 +00:00
|
|
|
continue;
|
2013-11-06 15:29:20 +00:00
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
if (!dts.scan(log_time, strlen(log_time), NULL, &log_tm, log_tv)) {
|
2013-07-14 04:31:59 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
line_iter = lower_bound(lf->begin(),
|
|
|
|
lf->end(),
|
|
|
|
log_tv);
|
|
|
|
while (line_iter != lf->end()) {
|
|
|
|
struct timeval line_tv = line_iter->get_timeval();
|
|
|
|
|
|
|
|
if ((line_tv.tv_sec != log_tv.tv_sec) ||
|
|
|
|
(line_tv.tv_usec != log_tv.tv_usec)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-12-22 13:27:36 +00:00
|
|
|
shared_buffer_ref sbr;
|
|
|
|
content_line_t cl = content_line_t(std::distance(lf->begin(), line_iter));
|
|
|
|
lf->read_line(line_iter, sbr);
|
|
|
|
|
|
|
|
string line_hash = hash_bytes(sbr.get_data(), sbr.length(),
|
|
|
|
&cl, sizeof(cl),
|
|
|
|
NULL);
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (line_hash == log_hash) {
|
|
|
|
content_line_t line_cl = content_line_t(
|
|
|
|
base_content_line + std::distance(lf->begin(), line_iter));
|
|
|
|
|
|
|
|
if (part_name != NULL && part_name[0] != '\0') {
|
2014-03-07 13:20:49 +00:00
|
|
|
lss.set_user_mark(&textview_curses::BM_PARTITION,
|
|
|
|
line_cl);
|
2013-07-14 04:31:59 +00:00
|
|
|
bm_meta[line_cl].bm_name = part_name;
|
|
|
|
}
|
2014-03-07 13:20:49 +00:00
|
|
|
else {
|
|
|
|
marked_session_lines.push_back(line_cl);
|
|
|
|
lss.set_user_mark(&textview_curses::BM_USER,
|
|
|
|
line_cl);
|
|
|
|
|
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
++line_iter;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
const char *errmsg;
|
|
|
|
|
|
|
|
errmsg = sqlite3_errmsg(lnav_data.ld_db);
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"bookmark select error: code %d -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
rc,
|
|
|
|
errmsg);
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"SELECT *,session_time=? as same_session FROM time_offset WHERE "
|
|
|
|
" log_time between ? and ? and log_format = ? "
|
|
|
|
" ORDER BY same_session DESC, session_time DESC",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare time_offset select statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (file_iter = lnav_data.ld_log_source.begin();
|
|
|
|
file_iter != lnav_data.ld_log_source.end();
|
|
|
|
++file_iter) {
|
|
|
|
char low_timestamp[64], high_timestamp[64];
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-07-14 04:31:59 +00:00
|
|
|
content_line_t base_content_line;
|
|
|
|
|
|
|
|
if (lf == NULL)
|
2013-06-03 04:11:17 +00:00
|
|
|
continue;
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
lss.find(lf->get_filename().c_str(), base_content_line);
|
|
|
|
|
|
|
|
logfile::iterator line_iter = lf->begin();
|
|
|
|
|
|
|
|
sql_strftime(low_timestamp, sizeof(low_timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_int64(stmt.in(), 1, lnav_data.ld_session_load_time) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind session time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt.in(), 2,
|
|
|
|
low_timestamp, strlen(low_timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind low log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
line_iter = lf->end();
|
|
|
|
--line_iter;
|
|
|
|
sql_strftime(high_timestamp, sizeof(high_timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_text(stmt.in(), 3,
|
|
|
|
high_timestamp, strlen(high_timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind high log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:36:58 +00:00
|
|
|
intern_string_t format_name = lf->get_format()->get_name();
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_text(stmt.in(), 4,
|
2015-06-03 13:36:58 +00:00
|
|
|
format_name.get(), format_name.size(),
|
2013-07-14 04:31:59 +00:00
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log format -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
date_time_scanner dts;
|
|
|
|
bool done = false;
|
|
|
|
string line;
|
|
|
|
int64_t last_mark_time = -1;
|
|
|
|
|
|
|
|
while (!done) {
|
|
|
|
int rc = sqlite3_step(stmt.in());
|
|
|
|
|
|
|
|
switch (rc) {
|
|
|
|
case SQLITE_OK:
|
|
|
|
case SQLITE_DONE:
|
|
|
|
done = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SQLITE_ROW: {
|
|
|
|
const char *log_time = (const char *)sqlite3_column_text(stmt.in(), 0);
|
|
|
|
const char *log_hash = (const char *)sqlite3_column_text(stmt.in(), 2);
|
|
|
|
int64_t mark_time = sqlite3_column_int64(stmt.in(), 3);
|
|
|
|
struct timeval log_tv;
|
2014-06-18 04:29:42 +00:00
|
|
|
struct exttm log_tm;
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (last_mark_time == -1) {
|
|
|
|
last_mark_time = mark_time;
|
|
|
|
}
|
|
|
|
else if (last_mark_time != mark_time) {
|
|
|
|
done = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_column_type(stmt.in(), 4) == SQLITE_NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
if (!dts.scan(log_time, strlen(log_time), NULL, &log_tm, log_tv)) {
|
2013-07-14 04:31:59 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
line_iter = lower_bound(lf->begin(),
|
|
|
|
lf->end(),
|
|
|
|
log_tv);
|
|
|
|
while (line_iter != lf->end()) {
|
|
|
|
struct timeval line_tv = line_iter->get_timeval();
|
|
|
|
|
|
|
|
if ((line_tv.tv_sec != log_tv.tv_sec) ||
|
|
|
|
(line_tv.tv_usec != log_tv.tv_usec)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
lf->read_line(line_iter, line);
|
|
|
|
|
|
|
|
string line_hash = hash_string(line);
|
|
|
|
if (line_hash == log_hash) {
|
|
|
|
int file_line = std::distance(lf->begin(), line_iter);
|
|
|
|
content_line_t line_cl = content_line_t(
|
|
|
|
base_content_line + file_line);
|
|
|
|
struct timeval offset;
|
|
|
|
|
|
|
|
offset_session_lines.push_back(line_cl);
|
|
|
|
offset.tv_sec = sqlite3_column_int64(stmt.in(), 4);
|
|
|
|
offset.tv_usec = sqlite3_column_int64(stmt.in(), 5);
|
|
|
|
lf->adjust_content_time(file_line, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
++line_iter;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
const char *errmsg;
|
|
|
|
|
|
|
|
errmsg = sqlite3_errmsg(lnav_data.ld_db);
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"bookmark select error: code %d -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
rc,
|
|
|
|
errmsg);
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
break;
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
}
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
static int read_save_time(yajlpp_parse_context *ypc, long long value)
|
2013-06-14 13:49:00 +00:00
|
|
|
{
|
|
|
|
lnav_data.ld_session_save_time = value;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-03-08 05:53:24 +00:00
|
|
|
static int read_time_offset(yajlpp_parse_context *ypc, int value)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
static int read_files(yajlpp_parse_context *ypc, const unsigned char *str, size_t len)
|
2013-05-31 15:01:31 +00:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
static int read_last_search(yajlpp_parse_context *ypc, const unsigned char *str, size_t len)
|
2013-06-15 23:42:21 +00:00
|
|
|
{
|
2013-06-16 01:07:50 +00:00
|
|
|
string regex = std::string((const char *)str, len);
|
2013-06-15 23:42:21 +00:00
|
|
|
const char **view_name;
|
2013-06-16 01:07:50 +00:00
|
|
|
int view_index;
|
2013-06-15 23:42:21 +00:00
|
|
|
|
|
|
|
view_name = find(lnav_view_strings,
|
|
|
|
lnav_view_strings + LNV__MAX,
|
|
|
|
ypc->get_path_fragment(-2));
|
|
|
|
view_index = view_name - lnav_view_strings;
|
|
|
|
|
2013-11-06 15:29:20 +00:00
|
|
|
if (view_index < LNV__MAX && !regex.empty()) {
|
2013-06-15 23:42:21 +00:00
|
|
|
execute_search((lnav_view_t)view_index, regex);
|
|
|
|
lnav_data.ld_views[view_index].set_follow_search(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
static int read_top_line(yajlpp_parse_context *ypc, long long value)
|
2013-05-31 15:01:31 +00:00
|
|
|
{
|
2013-06-16 01:07:50 +00:00
|
|
|
const char ** view_name;
|
2013-05-31 15:01:31 +00:00
|
|
|
int view_index;
|
|
|
|
|
|
|
|
view_name = find(lnav_view_strings,
|
|
|
|
lnav_view_strings + LNV__MAX,
|
|
|
|
ypc->get_path_fragment(-2));
|
|
|
|
view_index = view_name - lnav_view_strings;
|
|
|
|
if (view_index < LNV__MAX) {
|
|
|
|
textview_curses &tc = lnav_data.ld_views[view_index];
|
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
if (value != -1 && value < tc.get_inner_height()) {
|
2013-05-31 15:01:31 +00:00
|
|
|
tc.set_top(vis_line_t(value));
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:32:39 +00:00
|
|
|
static int read_word_wrap(yajlpp_parse_context *ypc, int value)
|
|
|
|
{
|
|
|
|
const char ** view_name;
|
|
|
|
int view_index;
|
|
|
|
|
|
|
|
view_name = find(lnav_view_strings,
|
|
|
|
lnav_view_strings + LNV__MAX,
|
|
|
|
ypc->get_path_fragment(-2));
|
|
|
|
view_index = view_name - lnav_view_strings;
|
2014-03-15 17:08:50 +00:00
|
|
|
if (view_index == LNV_HELP) {
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (view_index < LNV__MAX) {
|
2013-11-08 15:32:39 +00:00
|
|
|
textview_curses &tc = lnav_data.ld_views[view_index];
|
|
|
|
|
|
|
|
tc.set_word_wrap(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
static int read_commands(yajlpp_parse_context *ypc, const unsigned char *str, size_t len)
|
2013-06-02 21:20:15 +00:00
|
|
|
{
|
|
|
|
std::string cmdline = std::string((const char *)str, len);
|
2015-03-28 13:30:30 +00:00
|
|
|
const char ** view_name;
|
|
|
|
int view_index;
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2015-03-28 13:30:30 +00:00
|
|
|
view_name = find(lnav_view_strings,
|
|
|
|
lnav_view_strings + LNV__MAX,
|
|
|
|
ypc->get_path_fragment(-3));
|
|
|
|
view_index = view_name - lnav_view_strings;
|
|
|
|
bool active = ensure_view(&lnav_data.ld_views[view_index]);
|
2013-06-02 21:20:15 +00:00
|
|
|
execute_command(cmdline);
|
2015-03-28 13:30:30 +00:00
|
|
|
if (!active) {
|
|
|
|
lnav_data.ld_view_stack.pop();
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-01-05 14:18:58 +00:00
|
|
|
static struct json_path_handler view_handlers[] = {
|
|
|
|
json_path_handler("top_line"),
|
|
|
|
|
|
|
|
json_path_handler()
|
|
|
|
};
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
static struct json_path_handler view_info_handlers[] = {
|
2016-01-05 14:18:58 +00:00
|
|
|
json_path_handler("/save-time", read_save_time),
|
|
|
|
json_path_handler("/time-offset", read_time_offset),
|
|
|
|
json_path_handler("/files#", read_files),
|
|
|
|
json_path_handler("/views/([^/]+)/top_line", read_top_line),
|
|
|
|
json_path_handler("/views/([^/]+)/search", read_last_search),
|
|
|
|
json_path_handler("/views/([^/]+)/word_wrap", read_word_wrap),
|
|
|
|
json_path_handler("/views/([^/]+)/commands#", read_commands),
|
2013-05-31 15:01:31 +00:00
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
json_path_handler()
|
2013-05-31 15:01:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void load_session(void)
|
|
|
|
{
|
2013-06-02 21:20:15 +00:00
|
|
|
std::list<session_pair_t>::iterator sess_iter;
|
2013-06-16 01:07:50 +00:00
|
|
|
yajl_handle handle;
|
2013-06-20 13:08:53 +00:00
|
|
|
auto_fd fd;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
if (lnav_data.ld_session_file_names.empty()) {
|
2013-07-14 04:31:59 +00:00
|
|
|
load_time_bookmarks();
|
2013-05-31 15:01:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
sess_iter = lnav_data.ld_session_file_names.begin();
|
|
|
|
advance(sess_iter, lnav_data.ld_session_file_index);
|
2013-06-19 13:47:36 +00:00
|
|
|
lnav_data.ld_session_load_time = sess_iter->first.second;
|
2013-06-14 13:49:00 +00:00
|
|
|
lnav_data.ld_session_save_time = sess_iter->first.second;
|
2013-06-02 21:20:15 +00:00
|
|
|
string &view_info_name = sess_iter->second;
|
|
|
|
|
2013-07-23 12:55:08 +00:00
|
|
|
yajlpp_parse_context ypc(view_info_name, view_info_handlers);
|
|
|
|
handle = yajl_alloc(&ypc.ypc_callbacks, NULL, &ypc);
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
load_time_bookmarks();
|
2013-06-19 13:47:36 +00:00
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
if ((fd = open(view_info_name.c_str(), O_RDONLY)) < 0) {
|
|
|
|
perror("cannot open session file");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
unsigned char buffer[1024];
|
2014-11-04 04:24:46 +00:00
|
|
|
ssize_t rc;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
2014-03-07 13:20:49 +00:00
|
|
|
log_info("loading session file: %s", view_info_name.c_str());
|
2013-05-31 15:01:31 +00:00
|
|
|
while ((rc = read(fd, buffer, sizeof(buffer))) > 0) {
|
|
|
|
yajl_parse(handle, buffer, rc);
|
|
|
|
}
|
|
|
|
yajl_complete_parse(handle);
|
|
|
|
}
|
|
|
|
yajl_free(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void yajl_writer(void *context, const char *str, size_t len)
|
|
|
|
{
|
|
|
|
FILE *file = (FILE *)context;
|
|
|
|
|
|
|
|
fwrite(str, len, 1, file);
|
|
|
|
}
|
|
|
|
|
2014-03-07 13:20:49 +00:00
|
|
|
static void save_user_bookmarks(
|
|
|
|
sqlite3 *db, sqlite3_stmt *stmt, bookmark_vector<content_line_t> &user_marks)
|
|
|
|
{
|
|
|
|
logfile_sub_source &lss = lnav_data.ld_log_source;
|
|
|
|
std::map<content_line_t, bookmark_metadata> &bm_meta =
|
|
|
|
lss.get_user_bookmark_metadata();
|
|
|
|
bookmark_vector<content_line_t>::iterator iter;
|
|
|
|
|
|
|
|
for (iter = user_marks.begin(); iter != user_marks.end(); ++iter) {
|
|
|
|
std::map<content_line_t, bookmark_metadata>::iterator meta_iter;
|
|
|
|
logfile::iterator line_iter;
|
|
|
|
content_line_t cl = *iter;
|
|
|
|
|
|
|
|
meta_iter = bm_meta.find(cl);
|
|
|
|
|
|
|
|
marked_session_lines.push_back(cl);
|
|
|
|
|
2014-03-19 05:47:36 +00:00
|
|
|
if (!bind_line(db, stmt, cl, lnav_data.ld_session_time)) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-03-07 13:20:49 +00:00
|
|
|
|
|
|
|
if (meta_iter == bm_meta.end()) {
|
|
|
|
if (sqlite3_bind_text(stmt, 5, "", 0, SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2014-03-07 13:20:49 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (sqlite3_bind_text(stmt, 5,
|
|
|
|
meta_iter->second.bm_name.c_str(),
|
|
|
|
meta_iter->second.bm_name.length(),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2014-03-07 13:20:49 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_step(stmt) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2014-03-07 13:20:49 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sqlite3_reset(stmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
static void save_time_bookmarks(void)
|
2013-06-02 21:20:15 +00:00
|
|
|
{
|
2013-07-14 04:31:59 +00:00
|
|
|
auto_mem<sqlite3, sqlite_close_wrapper> db;
|
|
|
|
string db_path = dotlnav_path(LOG_METADATA_NAME);
|
|
|
|
auto_mem<char, sqlite3_free> errmsg;
|
|
|
|
auto_mem<sqlite3_stmt> stmt(sqlite3_finalize);
|
2013-06-16 01:07:50 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_open(db_path.c_str(), db.out()) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("unable to open bookmark DB -- %s\n", db_path.c_str());
|
2013-07-14 04:31:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_exec(db.in(), BOOKMARK_TABLE_DEF, NULL, NULL, errmsg.out()) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("unable to make bookmark table -- %s\n", errmsg.in());
|
2013-07-14 04:31:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_exec(db.in(), "BEGIN TRANSACTION", NULL, NULL, errmsg.out()) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("unable to begin transaction -- %s\n", errmsg.in());
|
2013-07-14 04:31:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
logfile_sub_source &lss = lnav_data.ld_log_source;
|
|
|
|
bookmarks<content_line_t>::type &bm = lss.get_user_bookmarks();
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"DELETE FROM bookmarks WHERE "
|
|
|
|
" log_time = ? and log_format = ? and log_hash = ? "
|
|
|
|
" and session_time = ?",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare bookmark delete statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-19 04:56:06 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (std::vector<content_line_t>::iterator cl_iter = marked_session_lines.begin();
|
|
|
|
cl_iter != marked_session_lines.end();
|
|
|
|
++cl_iter) {
|
2014-03-19 05:47:36 +00:00
|
|
|
if (!bind_line(
|
|
|
|
db.in(), stmt.in(), *cl_iter, lnav_data.ld_session_time)) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-19 04:56:06 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_step(stmt.in()) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
marked_session_lines.clear();
|
|
|
|
|
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"REPLACE INTO bookmarks"
|
|
|
|
" (log_time, log_format, log_hash, session_time, part_name)"
|
|
|
|
" VALUES (?, ?, ?, ?, ?)",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare bookmark replace statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
{
|
|
|
|
logfile_sub_source::iterator file_iter;
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (file_iter = lnav_data.ld_log_source.begin();
|
|
|
|
file_iter != lnav_data.ld_log_source.end();
|
|
|
|
++file_iter) {
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-07-14 04:31:59 +00:00
|
|
|
content_line_t base_content_line;
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (lf == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
base_content_line = lss.get_file_base_content_line(file_iter);
|
2014-03-19 05:47:36 +00:00
|
|
|
base_content_line = content_line_t(
|
|
|
|
base_content_line + lf->size() - 1);
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2014-03-19 05:47:36 +00:00
|
|
|
if (!bind_line(db.in(), stmt.in(), base_content_line,
|
|
|
|
lnav_data.ld_session_time)) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_null(stmt.in(), 5) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-06-30 23:43:08 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_step(stmt.in()) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
sqlite3_reset(stmt.in());
|
2013-06-03 04:11:17 +00:00
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
}
|
2013-06-03 04:11:17 +00:00
|
|
|
|
2014-03-07 13:20:49 +00:00
|
|
|
save_user_bookmarks(db.in(), stmt.in(), bm[&textview_curses::BM_USER]);
|
|
|
|
save_user_bookmarks(db.in(), stmt.in(), bm[&textview_curses::BM_PARTITION]);
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"DELETE FROM time_offset WHERE "
|
|
|
|
" log_time = ? and log_format = ? and log_hash = ? "
|
|
|
|
" and session_time = ?",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare time_offset delete statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (std::vector<content_line_t>::iterator cl_iter = offset_session_lines.begin();
|
|
|
|
cl_iter != offset_session_lines.end();
|
|
|
|
++cl_iter) {
|
2014-03-19 05:47:36 +00:00
|
|
|
if (!bind_line(
|
|
|
|
db.in(), stmt.in(), *cl_iter, lnav_data.ld_session_time)) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_step(stmt.in()) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
offset_session_lines.clear();
|
|
|
|
|
|
|
|
if (sqlite3_prepare_v2(db.in(),
|
|
|
|
"REPLACE INTO time_offset"
|
|
|
|
" (log_time, log_format, log_hash, session_time, offset_sec, offset_usec)"
|
|
|
|
" VALUES (?, ?, ?, ?, ?, ?)",
|
|
|
|
-1,
|
|
|
|
stmt.out(),
|
|
|
|
NULL) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not prepare time_offset replace statemnt -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
{
|
|
|
|
logfile_sub_source::iterator file_iter;
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (file_iter = lnav_data.ld_log_source.begin();
|
|
|
|
file_iter != lnav_data.ld_log_source.end();
|
|
|
|
++file_iter) {
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-07-14 04:31:59 +00:00
|
|
|
content_line_t base_content_line;
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (lf == NULL)
|
|
|
|
continue;
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
base_content_line = lss.get_file_base_content_line(file_iter);
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-03-19 05:47:36 +00:00
|
|
|
if (!bind_line(db.in(), stmt.in(), base_content_line,
|
|
|
|
lnav_data.ld_session_time)) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_null(stmt.in(), 5) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-06-02 21:20:15 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_null(stmt.in(), 6) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_step(stmt.in()) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
|
|
|
}
|
|
|
|
}
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
for (logfile_sub_source::iterator file_iter = lss.begin();
|
|
|
|
file_iter != lss.end();
|
|
|
|
++file_iter) {
|
|
|
|
logfile::iterator line_iter;
|
2013-06-30 23:43:08 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
if ((*file_iter)->get_file() == NULL)
|
2013-07-14 04:31:59 +00:00
|
|
|
continue;
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (!lf->is_time_adjusted())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
line_iter = lf->begin() + lf->get_time_offset_line();
|
|
|
|
|
|
|
|
char timestamp[64];
|
|
|
|
|
|
|
|
sql_strftime(timestamp, sizeof(timestamp),
|
2014-02-23 18:23:17 +00:00
|
|
|
lf->original_line_time(line_iter), 'T');
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt.in(), 1,
|
|
|
|
timestamp, strlen(timestamp),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-06-02 21:20:15 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:36:58 +00:00
|
|
|
intern_string_t format_name = lf->get_format()->get_name();
|
2013-07-14 04:31:59 +00:00
|
|
|
|
|
|
|
if (sqlite3_bind_text(stmt.in(), 2,
|
2015-06-03 13:36:58 +00:00
|
|
|
format_name.get(), format_name.size(),
|
2013-07-14 04:31:59 +00:00
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log format -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-07-05 16:14:39 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
std::string line_hash = hash_string(lf->read_line(line_iter));
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_text(stmt.in(), 3,
|
|
|
|
line_hash.c_str(), line_hash.length(),
|
|
|
|
SQLITE_TRANSIENT) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind log hash -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_int64(stmt.in(), 4, lnav_data.ld_session_time) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind session time -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
struct timeval offset = lf->get_time_offset();
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_bind_int64(stmt.in(), 5, offset.tv_sec) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind offset_sec -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_bind_int64(stmt.in(), 6, offset.tv_usec) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("could not bind offset_usec -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db.in()));
|
|
|
|
return;
|
2013-07-05 16:14:39 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
if (sqlite3_step(stmt.in()) != SQLITE_DONE) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error(
|
|
|
|
"could not execute bookmark insert statement -- %s\n",
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_errmsg(db));
|
|
|
|
return;
|
|
|
|
}
|
2013-07-05 16:14:39 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
sqlite3_reset(stmt.in());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_exec(db.in(), "COMMIT", NULL, NULL, errmsg.out()) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("unable to begin transaction -- %s\n", errmsg.in());
|
2013-07-14 04:31:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sqlite3_exec(db.in(), BOOKMARK_LRU_STMT, NULL, NULL, errmsg.out()) != SQLITE_OK) {
|
2014-11-04 14:36:49 +00:00
|
|
|
log_error("unable to delete old bookmarks -- %s\n", errmsg.in());
|
2013-07-14 04:31:59 +00:00
|
|
|
return;
|
2013-06-02 21:20:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
void save_session(void)
|
|
|
|
{
|
|
|
|
string view_file_name, view_file_tmp_name;
|
2013-06-16 01:07:50 +00:00
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
auto_mem<FILE> file(fclose);
|
2013-06-16 01:07:50 +00:00
|
|
|
char view_base_name[256];
|
|
|
|
yajl_gen handle = NULL;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
2013-07-14 04:31:59 +00:00
|
|
|
save_time_bookmarks();
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2013-06-12 04:10:59 +00:00
|
|
|
/* TODO: save the last search query */
|
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
snprintf(view_base_name, sizeof(view_base_name),
|
|
|
|
"view-info-%s.ts%ld.ppid%d.json",
|
|
|
|
lnav_data.ld_session_id.c_str(),
|
|
|
|
lnav_data.ld_session_time,
|
|
|
|
getppid());
|
|
|
|
|
2013-06-16 01:07:50 +00:00
|
|
|
view_file_name = dotlnav_path(view_base_name);
|
2013-05-31 15:01:31 +00:00
|
|
|
view_file_tmp_name = view_file_name + ".tmp";
|
|
|
|
|
|
|
|
if ((file = fopen(view_file_tmp_name.c_str(), "w")) == NULL) {
|
|
|
|
perror("Unable to open session file");
|
|
|
|
}
|
|
|
|
else if ((handle = yajl_gen_alloc(NULL)) == NULL) {
|
|
|
|
perror("Unable to create yajl_gen object");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
yajl_gen_config(handle,
|
|
|
|
yajl_gen_print_callback, yajl_writer, file.in());
|
|
|
|
|
|
|
|
{
|
|
|
|
yajlpp_map root_map(handle);
|
|
|
|
|
2013-06-14 13:49:00 +00:00
|
|
|
root_map.gen("save-time");
|
2013-11-08 15:32:39 +00:00
|
|
|
root_map.gen((long long)time(NULL));
|
2013-06-14 13:49:00 +00:00
|
|
|
|
2014-03-08 05:53:24 +00:00
|
|
|
root_map.gen("time-offset");
|
|
|
|
root_map.gen(lnav_data.ld_log_source.is_time_offset_enabled());
|
|
|
|
|
2013-05-31 15:01:31 +00:00
|
|
|
root_map.gen("files");
|
|
|
|
|
|
|
|
{
|
|
|
|
yajlpp_array file_list(handle);
|
|
|
|
|
2016-03-12 22:12:23 +00:00
|
|
|
for (map<string, logfile_open_options>::iterator iter = lnav_data.ld_file_names.begin();
|
|
|
|
iter != lnav_data.ld_file_names.end();
|
|
|
|
++iter) {
|
|
|
|
file_list.gen(iter->first);
|
|
|
|
}
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
root_map.gen("views");
|
|
|
|
|
|
|
|
{
|
|
|
|
yajlpp_map top_view_map(handle);
|
|
|
|
|
|
|
|
for (int lpc = 0; lpc < LNV__MAX; lpc++) {
|
|
|
|
textview_curses &tc = lnav_data.ld_views[lpc];
|
2013-06-16 01:07:50 +00:00
|
|
|
unsigned long width;
|
|
|
|
vis_line_t height;
|
2013-05-31 15:01:31 +00:00
|
|
|
|
|
|
|
top_view_map.gen(lnav_view_strings[lpc]);
|
|
|
|
|
|
|
|
yajlpp_map view_map(handle);
|
|
|
|
|
|
|
|
view_map.gen("top_line");
|
|
|
|
|
|
|
|
tc.get_dimensions(height, width);
|
2013-11-08 15:32:39 +00:00
|
|
|
if (tc.get_top() >= tc.get_top_for_last_row()) {
|
|
|
|
view_map.gen(-1LL);
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
|
|
|
else{
|
2013-05-31 15:01:31 +00:00
|
|
|
view_map.gen((long long)tc.get_top());
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-06-13 14:18:59 +00:00
|
|
|
|
|
|
|
view_map.gen("search");
|
|
|
|
view_map.gen(lnav_data.ld_last_search[lpc]);
|
2013-11-08 15:32:39 +00:00
|
|
|
|
|
|
|
view_map.gen("word_wrap");
|
|
|
|
view_map.gen(tc.get_word_wrap());
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
text_sub_source *tss = tc.get_sub_source();
|
|
|
|
if (tss == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
filter_stack::iterator filter_iter;
|
|
|
|
filter_stack &fs = tss->get_filters();
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
view_map.gen("commands");
|
|
|
|
yajlpp_array cmd_array(handle);
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
for (filter_iter = fs.begin();
|
|
|
|
filter_iter != fs.end();
|
|
|
|
++filter_iter) {
|
|
|
|
if (!(*filter_iter)->is_enabled()) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2014-10-20 05:16:40 +00:00
|
|
|
cmd_array.gen((*filter_iter)->to_command());
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
2015-03-28 13:30:30 +00:00
|
|
|
if (lpc == LNV_LOG) {
|
|
|
|
textview_curses::highlight_map_t &hmap =
|
|
|
|
lnav_data.ld_views[LNV_LOG].get_highlights();
|
|
|
|
textview_curses::highlight_map_t::iterator hl_iter;
|
|
|
|
|
|
|
|
for (hl_iter = hmap.begin();
|
|
|
|
hl_iter != hmap.end();
|
|
|
|
++hl_iter) {
|
|
|
|
if (hl_iter->first[0] == '$') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
cmd_array.gen("highlight " + hl_iter->first);
|
2014-10-20 05:16:40 +00:00
|
|
|
}
|
2013-06-16 01:07:50 +00:00
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
}
|
|
|
|
}
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
yajl_gen_clear(handle);
|
|
|
|
yajl_gen_free(handle);
|
|
|
|
|
|
|
|
fclose(file.release());
|
|
|
|
|
2015-09-18 03:55:31 +00:00
|
|
|
log_perror(rename(view_file_tmp_name.c_str(), view_file_name.c_str()));
|
2015-03-28 13:30:30 +00:00
|
|
|
|
|
|
|
log_debug("Saved session: %s", view_file_name.c_str());
|
2013-05-31 15:01:31 +00:00
|
|
|
}
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
|
|
|
|
void reset_session(void)
|
|
|
|
{
|
2013-06-16 01:07:50 +00:00
|
|
|
textview_curses::highlight_map_t &hmap =
|
|
|
|
lnav_data.ld_views[LNV_LOG].get_highlights();
|
2013-06-02 21:20:15 +00:00
|
|
|
textview_curses::highlight_map_t::iterator hl_iter = hmap.begin();
|
|
|
|
|
2014-03-07 13:20:49 +00:00
|
|
|
log_info("reset session: time=%d", lnav_data.ld_session_time);
|
|
|
|
|
2013-06-02 21:20:15 +00:00
|
|
|
save_session();
|
|
|
|
scan_sessions();
|
|
|
|
|
|
|
|
lnav_data.ld_session_time = time(NULL);
|
|
|
|
|
|
|
|
while (hl_iter != hmap.end()) {
|
|
|
|
if (hl_iter->first[0] == '$') {
|
|
|
|
++hl_iter;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
hmap.erase(hl_iter++);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-30 23:43:08 +00:00
|
|
|
logfile_sub_source::iterator file_iter;
|
|
|
|
|
|
|
|
for (file_iter = lnav_data.ld_log_source.begin();
|
|
|
|
file_iter != lnav_data.ld_log_source.end();
|
|
|
|
++file_iter) {
|
2014-10-20 05:16:40 +00:00
|
|
|
logfile *lf = (*file_iter)->get_file();
|
2013-06-30 23:43:08 +00:00
|
|
|
|
|
|
|
lf->clear_time_offset();
|
|
|
|
}
|
2013-07-14 04:31:59 +00:00
|
|
|
|
2015-12-22 13:27:36 +00:00
|
|
|
lnav_data.ld_log_source.clear_min_max_log_times();
|
|
|
|
|
|
|
|
lnav_data.ld_log_source.get_user_bookmark_metadata().clear();
|
|
|
|
|
2015-09-14 15:56:42 +00:00
|
|
|
for (int lpc = 0; lpc < LNV__MAX; lpc++) {
|
|
|
|
textview_curses &tc = lnav_data.ld_views[lpc];
|
|
|
|
text_sub_source *tss = tc.get_sub_source();
|
2014-10-20 05:16:40 +00:00
|
|
|
|
2015-09-14 15:56:42 +00:00
|
|
|
if (tss == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
tss->get_filters().clear_filters();
|
|
|
|
tss->text_filters_changed();
|
|
|
|
tss->text_clear_marks(&textview_curses::BM_USER);
|
|
|
|
tc.get_bookmarks()[&textview_curses::BM_USER].clear();
|
|
|
|
tss->text_clear_marks(&textview_curses::BM_PARTITION);
|
|
|
|
tc.get_bookmarks()[&textview_curses::BM_PARTITION].clear();
|
|
|
|
tc.reload_data();
|
|
|
|
}
|
2013-06-02 21:20:15 +00:00
|
|
|
}
|