feat: use custom logger with timestamp in rfc3339 (#67)
This commit is contained in:
parent
ea8b9e9cce
commit
7f062b6705
8 changed files with 45 additions and 68 deletions
17
Cargo.lock
generated
17
Cargo.lock
generated
|
@ -583,7 +583,6 @@ dependencies = [
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap",
|
||||||
"diqwest",
|
"diqwest",
|
||||||
"env_logger",
|
|
||||||
"futures",
|
"futures",
|
||||||
"get_if_addrs",
|
"get_if_addrs",
|
||||||
"headers",
|
"headers",
|
||||||
|
@ -629,16 +628,6 @@ dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "env_logger"
|
|
||||||
version = "0.9.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0b2cf0344971ee6c64c31be0d530793fba457d322dfec2810c453d0ef228f9c3"
|
|
||||||
dependencies = [
|
|
||||||
"humantime",
|
|
||||||
"log",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "event-listener"
|
name = "event-listener"
|
||||||
version = "2.5.2"
|
version = "2.5.2"
|
||||||
|
@ -1033,12 +1022,6 @@ version = "1.0.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421"
|
checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "humantime"
|
|
||||||
version = "2.1.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hyper"
|
name = "hyper"
|
||||||
version = "0.14.19"
|
version = "0.14.19"
|
||||||
|
|
|
@ -34,7 +34,6 @@ lazy_static = "1.4"
|
||||||
uuid = { version = "1.1", features = ["v4", "fast-rng"] }
|
uuid = { version = "1.1", features = ["v4", "fast-rng"] }
|
||||||
urlencoding = "2.1"
|
urlencoding = "2.1"
|
||||||
xml-rs = "0.8"
|
xml-rs = "0.8"
|
||||||
env_logger = { version = "0.9", default-features = false, features = ["humantime"] }
|
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
socket2 = "0.4"
|
socket2 = "0.4"
|
||||||
async-stream = "0.3"
|
async-stream = "0.3"
|
||||||
|
|
30
src/logger.rs
Normal file
30
src/logger.rs
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
use chrono::{Local, SecondsFormat};
|
||||||
|
use log::{Level, Metadata, Record};
|
||||||
|
use log::{LevelFilter, SetLoggerError};
|
||||||
|
|
||||||
|
struct SimpleLogger;
|
||||||
|
|
||||||
|
impl log::Log for SimpleLogger {
|
||||||
|
fn enabled(&self, metadata: &Metadata) -> bool {
|
||||||
|
metadata.level() <= Level::Info
|
||||||
|
}
|
||||||
|
|
||||||
|
fn log(&self, record: &Record) {
|
||||||
|
if self.enabled(record.metadata()) {
|
||||||
|
let timestamp = Local::now().to_rfc3339_opts(SecondsFormat::Secs, true);
|
||||||
|
if record.level() < Level::Info {
|
||||||
|
eprintln!("{} {} - {}", timestamp, record.level(), record.args());
|
||||||
|
} else {
|
||||||
|
println!("{} {} - {}", timestamp, record.level(), record.args());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn flush(&self) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
static LOGGER: SimpleLogger = SimpleLogger;
|
||||||
|
|
||||||
|
pub fn init() -> Result<(), SetLoggerError> {
|
||||||
|
log::set_logger(&LOGGER).map(|()| log::set_max_level(LevelFilter::Info))
|
||||||
|
}
|
15
src/main.rs
15
src/main.rs
|
@ -1,5 +1,6 @@
|
||||||
mod args;
|
mod args;
|
||||||
mod auth;
|
mod auth;
|
||||||
|
mod logger;
|
||||||
mod server;
|
mod server;
|
||||||
mod streamer;
|
mod streamer;
|
||||||
mod tls;
|
mod tls;
|
||||||
|
@ -12,9 +13,8 @@ use crate::args::{matches, Args};
|
||||||
use crate::server::{Request, Server};
|
use crate::server::{Request, Server};
|
||||||
use crate::tls::{TlsAcceptor, TlsStream};
|
use crate::tls::{TlsAcceptor, TlsStream};
|
||||||
|
|
||||||
use std::io::Write;
|
|
||||||
use std::net::{IpAddr, SocketAddr, TcpListener as StdTcpListener};
|
use std::net::{IpAddr, SocketAddr, TcpListener as StdTcpListener};
|
||||||
use std::{env, sync::Arc};
|
use std::sync::Arc;
|
||||||
|
|
||||||
use futures::future::join_all;
|
use futures::future::join_all;
|
||||||
use tokio::net::TcpListener;
|
use tokio::net::TcpListener;
|
||||||
|
@ -32,16 +32,7 @@ async fn main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn run() -> BoxResult<()> {
|
async fn run() -> BoxResult<()> {
|
||||||
if env::var("RUST_LOG").is_err() {
|
logger::init().map_err(|e| format!("Failed to init logger, {}", e))?;
|
||||||
env::set_var("RUST_LOG", "info")
|
|
||||||
}
|
|
||||||
env_logger::builder()
|
|
||||||
.format(|buf, record| {
|
|
||||||
let timestamp = buf.timestamp_millis();
|
|
||||||
writeln!(buf, "[{} {}] {}", timestamp, record.level(), record.args())
|
|
||||||
})
|
|
||||||
.init();
|
|
||||||
|
|
||||||
let args = Args::parse(matches())?;
|
let args = Args::parse(matches())?;
|
||||||
let args = Arc::new(args);
|
let args = Arc::new(args);
|
||||||
let handles = serve(args.clone())?;
|
let handles = serve(args.clone())?;
|
||||||
|
|
|
@ -36,7 +36,6 @@ fn path_prefix_propfind(
|
||||||
#[case("index.html")]
|
#[case("index.html")]
|
||||||
fn serve_single_file(tmpdir: TempDir, port: u16, #[case] file: &str) -> Result<(), Error> {
|
fn serve_single_file(tmpdir: TempDir, port: u16, #[case] file: &str) -> Result<(), Error> {
|
||||||
let mut child = Command::cargo_bin("dufs")?
|
let mut child = Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path().join(file))
|
.arg(tmpdir.path().join(file))
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
|
|
@ -6,14 +6,13 @@ use assert_cmd::prelude::*;
|
||||||
use assert_fs::fixture::TempDir;
|
use assert_fs::fixture::TempDir;
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
use rstest::rstest;
|
use rstest::rstest;
|
||||||
use std::io::{BufRead, BufReader};
|
use std::io::Read;
|
||||||
use std::process::{Command, Stdio};
|
use std::process::{Command, Stdio};
|
||||||
|
|
||||||
#[rstest]
|
#[rstest]
|
||||||
#[case(&["-b", "20.205.243.166"])]
|
#[case(&["-b", "20.205.243.166"])]
|
||||||
fn bind_fails(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
fn bind_fails(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
||||||
Command::cargo_bin("dufs")?
|
Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
@ -51,7 +50,6 @@ fn bind_ipv4_ipv6(
|
||||||
#[case(&["--path-prefix", "/prefix"])]
|
#[case(&["--path-prefix", "/prefix"])]
|
||||||
fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
||||||
let mut child = Command::cargo_bin("dufs")?
|
let mut child = Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
@ -61,22 +59,23 @@ fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> R
|
||||||
|
|
||||||
wait_for_port(port);
|
wait_for_port(port);
|
||||||
|
|
||||||
// WARN assumes urls list is terminated by an empty line
|
let stdout = child.stdout.as_mut().expect("Failed to get stdout");
|
||||||
let url_lines = BufReader::new(child.stdout.take().unwrap())
|
let mut buf = [0; 1000];
|
||||||
|
let buf_len = stdout.read(&mut buf)?;
|
||||||
|
let output = std::str::from_utf8(&buf[0..buf_len])?;
|
||||||
|
let url_lines = output
|
||||||
.lines()
|
.lines()
|
||||||
.map(|line| line.expect("Error reading stdout"))
|
|
||||||
.take_while(|line| !line.is_empty()) /* non-empty lines */
|
.take_while(|line| !line.is_empty()) /* non-empty lines */
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>()
|
||||||
let url_lines = url_lines.join("\n");
|
.join("\n");
|
||||||
|
|
||||||
let urls = Regex::new(r"http://[a-zA-Z0-9\.\[\]:/]+")
|
let urls = Regex::new(r"http://[a-zA-Z0-9\.\[\]:/]+")
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.captures_iter(url_lines.as_str())
|
.captures_iter(url_lines.as_str())
|
||||||
.map(|caps| caps.get(0).unwrap().as_str())
|
.filter_map(|caps| caps.get(0).map(|v| v.as_str()))
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
assert!(!urls.is_empty());
|
assert!(!urls.is_empty());
|
||||||
|
|
||||||
for url in urls {
|
for url in urls {
|
||||||
reqwest::blocking::get(url)?.error_for_status()?;
|
reqwest::blocking::get(url)?.error_for_status()?;
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,21 +13,7 @@ pub type Error = Box<dyn std::error::Error>;
|
||||||
|
|
||||||
/// File names for testing purpose
|
/// File names for testing purpose
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub static FILES: &[&str] = &[
|
pub static FILES: &[&str] = &["test.txt", "test.html", "index.html", "😀.bin"];
|
||||||
"test.txt",
|
|
||||||
"test.html",
|
|
||||||
"index.html",
|
|
||||||
"test.mkv",
|
|
||||||
#[cfg(not(windows))]
|
|
||||||
"test \" \' & < >.csv",
|
|
||||||
"😀.data",
|
|
||||||
"⎙.mp4",
|
|
||||||
"#[]{}()@!$&'`+,;= %20.test",
|
|
||||||
#[cfg(unix)]
|
|
||||||
":?#[]{}<>()@!$&'`|*+,;= %20.test",
|
|
||||||
#[cfg(not(windows))]
|
|
||||||
"foo\\bar.test",
|
|
||||||
];
|
|
||||||
|
|
||||||
/// Directory names for testing diretory don't exist
|
/// Directory names for testing diretory don't exist
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
|
@ -41,10 +27,6 @@ pub static DIR_NO_INDEX: &str = "dir-no-index/";
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub static DIRECTORIES: &[&str] = &["dira/", "dirb/", "dirc/", DIR_NO_INDEX];
|
pub static DIRECTORIES: &[&str] = &["dira/", "dirb/", "dirc/", DIR_NO_INDEX];
|
||||||
|
|
||||||
/// Name of a deeply nested file
|
|
||||||
#[allow(dead_code)]
|
|
||||||
pub static DEEPLY_NESTED_FILE: &str = "very/deeply/nested/test.rs";
|
|
||||||
|
|
||||||
/// Test fixture which creates a temporary directory with a few files and directories inside.
|
/// Test fixture which creates a temporary directory with a few files and directories inside.
|
||||||
/// The directories also contain files.
|
/// The directories also contain files.
|
||||||
#[fixture]
|
#[fixture]
|
||||||
|
@ -69,10 +51,6 @@ pub fn tmpdir() -> TempDir {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tmpdir
|
|
||||||
.child(&DEEPLY_NESTED_FILE)
|
|
||||||
.write_str("File in a deeply nested directory.")
|
|
||||||
.expect("Couldn't write to file");
|
|
||||||
tmpdir
|
tmpdir
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,7 +74,6 @@ where
|
||||||
let tmpdir = tmpdir();
|
let tmpdir = tmpdir();
|
||||||
let child = Command::cargo_bin("dufs")
|
let child = Command::cargo_bin("dufs")
|
||||||
.expect("Couldn't find test binary")
|
.expect("Couldn't find test binary")
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
@ -124,7 +101,6 @@ where
|
||||||
let tmpdir = tmpdir();
|
let tmpdir = tmpdir();
|
||||||
let child = Command::cargo_bin("dufs")
|
let child = Command::cargo_bin("dufs")
|
||||||
.expect("Couldn't find test binary")
|
.expect("Couldn't find test binary")
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
|
|
@ -76,12 +76,12 @@ fn get_dir_search(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
||||||
|
|
||||||
#[rstest]
|
#[rstest]
|
||||||
fn get_dir_search2(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
fn get_dir_search2(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
||||||
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "😀.data"))?;
|
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "😀.bin"))?;
|
||||||
assert_eq!(resp.status(), 200);
|
assert_eq!(resp.status(), 200);
|
||||||
let paths = utils::retrive_index_paths(&resp.text()?);
|
let paths = utils::retrive_index_paths(&resp.text()?);
|
||||||
assert!(!paths.is_empty());
|
assert!(!paths.is_empty());
|
||||||
for p in paths {
|
for p in paths {
|
||||||
assert!(p.contains(&"😀.data"));
|
assert!(p.contains(&"😀.bin"));
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue