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/test/lnav_doctests.cc

201 lines
6.3 KiB
C++

/**
* Copyright (c) 2017, 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.
*/
#include "config.h"
#define DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN
#include "doctest.hh"
#include "lnav_config.hh"
#include "relative_time.hh"
#include "unique_path.hh"
#include "logfile.hh"
#include "base/humanize.hh"
#include "log_format.hh"
using namespace std;
std::vector<std::shared_ptr<log_format>>& log_format::get_root_formats()
{
static std::vector<std::shared_ptr<log_format>> retval;
return retval;
}
#if 0
TEST_CASE("overwritten-logfile") {
string fname = "reload_test.0";
ofstream(fname) << "test 1\n";
logfile_open_options loo;
logfile lf(fname, loo);
auto build_result = lf.rebuild_index();
CHECK(build_result == logfile::RR_NEW_LINES);
CHECK(lf.size() == 1);
sleep(1);
ofstream(fname) << "test 2\n";
auto rebuild_result = lf.rebuild_index();
CHECK(rebuild_result == logfile::RR_NO_NEW_LINES);
CHECK(lf.is_closed());
}
#endif
TEST_CASE("duration2str") {
string val;
duration2str(25 * 60 * 60 * 1000 + 123, val);
CHECK(val == "1d1h0m0s");
val.clear();
duration2str(10 * 1000 + 123, val);
CHECK(val == "10s123");
val.clear();
duration2str(10 * 1000, val);
CHECK(val == "10s000");
val.clear();
duration2str(100, val);
CHECK(val == "100");
val.clear();
duration2str(0, val);
CHECK(val == "");
val.clear();
duration2str(-10, val);
CHECK(val == "-010");
val.clear();
duration2str(-10 * 1000, val);
CHECK(val == "-10s000");
}
TEST_CASE("humanize::file_size") {
CHECK(humanize::file_size(0) == "0.0 B");
CHECK(humanize::file_size(1) == "1.0 B");
CHECK(humanize::file_size(1024) == "1.0KB");
CHECK(humanize::file_size(1500) == "1.5KB");
CHECK(humanize::file_size(55LL * 784LL * 1024LL * 1024LL) == "42.1GB");
CHECK(humanize::file_size(-1LL) == "Unknown");
CHECK(humanize::file_size(std::numeric_limits<int64_t>::max()) == "8.0EB");
}
TEST_CASE("ptime_fmt") {
const char *date_str = "2018-05-16 18:16:42";
struct exttm tm;
off_t off = 0;
bool rc = ptime_fmt("%Y-%d-%m\t%H:%M:%S", &tm, date_str, off, strlen(date_str));
CHECK(!rc);
CHECK(off == 8);
}
TEST_CASE("rgb_color from string") {
string name = "SkyBlue1";
auto color = rgb_color::from_str(name).unwrap();
CHECK(color.rc_r == 135);
CHECK(color.rc_g == 215);
CHECK(color.rc_b == 255);
}
TEST_CASE("ptime_roundtrip") {
const char *fmts[] = {
"%Y-%m-%d %l:%M:%S %p",
"%Y-%m-%d %I:%M:%S %p",
};
time_t now = time(nullptr);
for (auto fmt : fmts) {
for (time_t sec = now; sec < (now + (24 * 60 * 60)); sec++) {
char ftime_result[128];
char strftime_result[128];
struct exttm etm;
memset(&etm, 0, sizeof(etm));
gmtime_r(&sec, &etm.et_tm);
etm.et_flags = ETF_YEAR_SET | ETF_MONTH_SET | ETF_DAY_SET;
size_t ftime_size = ftime_fmt(ftime_result, sizeof(ftime_result),
fmt, etm);
size_t strftime_size = strftime(strftime_result,
sizeof(strftime_result), fmt,
&etm.et_tm);
CHECK(string(ftime_result, ftime_size) ==
string(strftime_result, strftime_size));
struct exttm etm2;
off_t off = 0;
memset(&etm2, 0, sizeof(etm2));
bool rc = ptime_fmt(fmt, &etm2, ftime_result, off, ftime_size);
CHECK(rc);
CHECK(sec == tm2sec(&etm2.et_tm));
}
}
}
class my_path_source : public unique_path_source {
public:
explicit my_path_source(ghc::filesystem::path p) : mps_path(std::move(p)) {
}
ghc::filesystem::path get_path() const override {
return this->mps_path;
}
ghc::filesystem::path mps_path;
};
TEST_CASE("unique_path") {
unique_path_generator upg;
auto bar = make_shared<my_path_source>("/foo/bar");
auto bar_dupe = make_shared<my_path_source>("/foo/bar");
auto baz = make_shared<my_path_source>("/foo/baz");
auto baz2 = make_shared<my_path_source>("/foo2/bar");
auto log1 = make_shared<my_path_source>(
"/home/bob/downloads/machine1/var/log/syslog.log");
auto log2 = make_shared<my_path_source>(
"/home/bob/downloads/machine2/var/log/syslog.log");
upg.add_source(bar);
upg.add_source(bar_dupe);
upg.add_source(baz);
upg.add_source(baz2);
upg.add_source(log1);
upg.add_source(log2);
upg.generate();
CHECK(bar->get_unique_path() == "[foo]/bar");
CHECK(bar_dupe->get_unique_path() == "[foo]/bar");
CHECK(baz->get_unique_path() == "baz");
CHECK(baz2->get_unique_path() == "[foo2]/bar");
CHECK(log1->get_unique_path() == "[machine1]/syslog.log");
CHECK(log2->get_unique_path() == "[machine2]/syslog.log");
}