Skip to content

Commit

Permalink
new: added configuration file support with customizations for predefi…
Browse files Browse the repository at this point in the history
…ned field names and value mapping for levels
  • Loading branch information
pamburus committed Jun 30, 2021
1 parent 195f7cd commit cfb2349
Show file tree
Hide file tree
Showing 9 changed files with 653 additions and 95 deletions.
211 changes: 197 additions & 14 deletions Cargo.lock

Large diffs are not rendered by default.

3 changes: 3 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -18,15 +18,18 @@ bytefmt = "0"
chrono = { version = "0", features = ["serde"] }
chrono-tz = "0"
closure = "0"
config = "0"
crossbeam-channel = "0"
crossbeam-queue = "0"
crossbeam-utils = "0"
derive_deref = "1"
error-chain = "0"
flate2 = "1"
heapless = "0"
atty = "0"
itertools = "0"
num_cpus = "1"
platform-dirs = "0"
serde = { version = "1", features = ["derive"] }
serde_json = { version = "1", features = ["raw_value"] }
shellwords = "1"
Expand Down
15 changes: 11 additions & 4 deletions src/app.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,15 @@ use serde_json as json;
use crate::datefmt::{DateTimeFormat, DateTimeFormatter};
use crate::error::*;
use crate::formatting::RecordFormatter;
use crate::model::{Filter, Record};
use crate::model::{Filter, Parser, ParserSettings, RawRecord};
use crate::scanning::{BufFactory, Scanner, Segment, SegmentBuf, SegmentBufFactory};
use crate::settings::Settings;
use crate::theme::Theme;
use crate::IncludeExcludeKeyFilter;

// TODO: merge Options to Settings and replace Options with Settings.
pub struct Options {
pub settings: Settings,
pub theme: Arc<Theme>,
pub time_format: DateTimeFormat,
pub raw_fields: bool,
Expand Down Expand Up @@ -47,6 +50,8 @@ impl App {
let n = self.options.concurrency;
let sfi = Arc::new(SegmentBufFactory::new(self.options.buffer_size));
let bfo = BufFactory::new(self.options.buffer_size);
let settings = ParserSettings::from(&self.options.settings);
let parser = Parser::new(&settings);
thread::scope(|scope| -> Result<()> {
// prepare receive/transmit channels for input data
let (txi, rxi): (Vec<_>, Vec<_>) = (0..n).map(|_| channel::bounded(1)).unzip();
Expand All @@ -69,7 +74,7 @@ impl App {
}));
// spawn processing threads
for (rxi, txo) in izip!(rxi, txo) {
scope.spawn(closure!(ref bfo, ref sfi, |_| {
scope.spawn(closure!(ref bfo, ref parser, ref sfi, |_| {
let mut formatter = RecordFormatter::new(
self.options.theme.clone(),
DateTimeFormatter::new(
Expand All @@ -84,7 +89,7 @@ impl App {
match segment {
Segment::Complete(segment) => {
let mut buf = bfo.new_buf();
self.process_segement(&segment, &mut formatter, &mut buf);
self.process_segement(&parser, &segment, &mut formatter, &mut buf);
sfi.recycle(segment);
if let Err(_) = txo.send(buf) {
break;
Expand Down Expand Up @@ -128,6 +133,7 @@ impl App {

fn process_segement(
&self,
parser: &Parser,
segment: &SegmentBuf,
formatter: &mut RecordFormatter,
buf: &mut Vec<u8>,
Expand All @@ -137,8 +143,9 @@ impl App {
if data.len() == 0 {
continue;
}
let mut stream = json::Deserializer::from_slice(data).into_iter::<Record>();
let mut stream = json::Deserializer::from_slice(data).into_iter::<RawRecord>();
while let Some(Ok(record)) = stream.next() {
let record = parser.parse(record);
if record.matches(&self.options.filter) {
formatter.format_record(buf, &record);
}
Expand Down
8 changes: 8 additions & 0 deletions src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ use std::io;
use std::num::{ParseIntError, TryFromIntError};

// third-party imports
use config::ConfigError;
use thiserror::Error;

/// Error is an error which may occur in the application.
Expand All @@ -15,6 +16,8 @@ pub enum Error {
ParseIntError(#[from] ParseIntError),
#[error(transparent)]
TryFromIntError(#[from] TryFromIntError),
#[error("failed to load configuration: {0}")]
Config(#[from] ConfigError),
#[error(transparent)]
Boxed(#[from] Box<dyn std::error::Error + std::marker::Send>),
#[error("file {filename:?} not found")]
Expand All @@ -24,6 +27,11 @@ pub enum Error {
value: String,
valid_values: Vec<String>,
},
#[error("invalid field kind {value:?}, use any of {valid_values:?}")]
InvalidFieldKind {
value: String,
valid_values: Vec<String>,
},
#[error(
"invalid size {0:?}, use {:?} or {:?} format for IEC units or {:?} format for SI units",
"64K",
Expand Down
1 change: 1 addition & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ pub mod error;
pub mod fmtx;
pub mod input;
pub mod output;
pub mod settings;
pub mod theme;
pub mod timestamp;
pub mod types;
Expand Down
5 changes: 5 additions & 0 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ use std::time::Duration;
use ansi_term::Colour;
use chrono::{FixedOffset, Local, TimeZone};
use chrono_tz::{Tz, UTC};
use platform_dirs::AppDirs;
use structopt::clap::arg_enum;
use structopt::StructOpt;

Expand All @@ -18,6 +19,7 @@ use hl::datefmt::LinuxDateFormat;
use hl::error::*;
use hl::input::{open, ConcatReader, Input, InputStream};
use hl::output::{OutputStream, Pager};
use hl::settings::Settings;
use hl::signal::SignalHandler;
use hl::theme::Theme;
use hl::Level;
Expand Down Expand Up @@ -166,6 +168,8 @@ fn parse_non_zero_size(s: &str) -> Result<usize> {
}

fn run() -> Result<()> {
let app_dirs = AppDirs::new(Some("hl"), true).unwrap();
let settings = Settings::load(&app_dirs)?;
let opt = Opt::from_args();
let stdout_is_atty = || atty::is(atty::Stream::Stdout);

Expand Down Expand Up @@ -220,6 +224,7 @@ fn run() -> Result<()> {

// Create app.
let app = hl::App::new(hl::Options {
settings: settings,
theme: Arc::new(theme),
raw_fields: opt.raw_fields,
time_format: LinuxDateFormat::new(&opt.time_format).compile(),
Expand Down
Loading

0 comments on commit cfb2349

Please sign in to comment.