From ebef6dba09cb436054775939be9363c61a3cf854 Mon Sep 17 00:00:00 2001 From: Eric Fang Date: Sat, 17 Jun 2023 05:56:49 -0700 Subject: [PATCH] Introduce a `log-level` flag. (#2036) * Implement `--log-level` In short: Propose the default log level for release to error Keep the --debug flag and it sets the log level to debug Introduce --log-level flag to set the log level --debug flag will be ignored if --log-level flag is also set. Keep the default log level for debug build to debug Signed-off-by: yihuaf * minor fix to CI Signed-off-by: yihuaf * move log-level to youki_extend Signed-off-by: yihuaf * remove the workflow_dispatch trigger Signed-off-by: yihuaf --------- Signed-off-by: yihuaf --- .../integration_tests_validation.yaml | 2 +- README.md | 1 + crates/liboci-cli/src/lib.rs | 7 +- crates/youki/src/main.rs | 3 + crates/youki/src/observability.rs | 100 ++++++++---------- docs/src/user/basic_usage.md | 10 ++ scripts/rust_integration_tests.sh | 2 +- .../integration_test/src/utils/test_utils.rs | 3 - .../runtimetest/src/utils.rs | 6 +- 9 files changed, 68 insertions(+), 66 deletions(-) diff --git a/.github/workflows/integration_tests_validation.yaml b/.github/workflows/integration_tests_validation.yaml index 400fdcd75..8961a6406 100644 --- a/.github/workflows/integration_tests_validation.yaml +++ b/.github/workflows/integration_tests_validation.yaml @@ -20,7 +20,7 @@ jobs: with: files: | .github/workflows/integration_tests_validation.yaml - integration_test/tests/rust-integration-tests/** + tests/rust-integration-tests/** files_ignore: | **.md - name: List all changed files diff --git a/README.md b/README.md index a618466a7..93f9a4aab 100644 --- a/README.md +++ b/README.md @@ -160,6 +160,7 @@ $ sudo apt-get install \ libelf-dev \ libseccomp-dev \ libclang-dev \ + glibc-static \ libssl-dev ``` diff --git a/crates/liboci-cli/src/lib.rs b/crates/liboci-cli/src/lib.rs index 646d017dd..6576cfdf4 100644 --- a/crates/liboci-cli/src/lib.rs +++ b/crates/liboci-cli/src/lib.rs @@ -66,13 +66,12 @@ pub enum CommonCmd { // flags, but these are commonly accepted by runtimes #[derive(Parser, Debug)] pub struct GlobalOpts { - /// change log level to debug. - // Example in future : '--debug change log level to debug. (default: "warn")' - #[clap(long)] - pub debug: bool, /// set the log file to write youki logs to (default is '/dev/stderr') #[clap(short, long, overrides_with("log"))] pub log: Option, + /// change log level to debug, but the `log-level` flag takes precedence + #[clap(long)] + pub debug: bool, /// set the log format ('text' (default), or 'json') (default: "text") #[clap(long)] pub log_format: Option, diff --git a/crates/youki/src/main.rs b/crates/youki/src/main.rs index dfb2923da..bf82cb76c 100644 --- a/crates/youki/src/main.rs +++ b/crates/youki/src/main.rs @@ -21,6 +21,9 @@ struct YoukiExtendOpts { /// Enable logging to systemd-journald #[clap(long)] pub systemd_log: bool, + /// set the log level (default is 'error') + #[clap(long)] + pub log_level: Option, } // High-level commandline option definition diff --git a/crates/youki/src/observability.rs b/crates/youki/src/observability.rs index 0c18d5875..a9bfd57dd 100644 --- a/crates/youki/src/observability.rs +++ b/crates/youki/src/observability.rs @@ -6,7 +6,6 @@ use std::str::FromStr; use tracing::Level; use tracing_subscriber::prelude::*; -const LOG_LEVEL_ENV_NAME: &str = "YOUKI_LOG_LEVEL"; const LOG_FORMAT_TEXT: &str = "text"; const LOG_FORMAT_JSON: &str = "json"; enum LogFormat { @@ -20,7 +19,7 @@ const DEFAULT_LOG_LEVEL: &str = "debug"; /// If not in debug mode, default level is warn to get important logs #[cfg(not(debug_assertions))] -const DEFAULT_LOG_LEVEL: &str = "warn"; +const DEFAULT_LOG_LEVEL: &str = "error"; fn detect_log_format(log_format: Option<&str>) -> Result { match log_format { @@ -30,20 +29,23 @@ fn detect_log_format(log_format: Option<&str>) -> Result { } } -fn detect_log_level(is_debug: bool) -> Result { - let filter: Cow = if is_debug { - "debug".into() - } else if let Ok(level) = std::env::var(LOG_LEVEL_ENV_NAME) { - level.into() - } else { - DEFAULT_LOG_LEVEL.into() +fn detect_log_level(input: Option, is_debug: bool) -> Result { + // We keep the `debug` flag for backward compatibility, but use `log-level` + // as the main way to set the log level due to the flexibility. If both are + // specified, `log-level` takes precedence. + let log_level: Cow = match input { + None if is_debug => "debug".into(), + None => DEFAULT_LOG_LEVEL.into(), + Some(level) => level.into(), }; - Ok(Level::from_str(filter.as_ref())?) + + Ok(Level::from_str(log_level.as_ref())?) } #[derive(Debug, Default)] pub struct ObservabilityConfig { pub log_debug_flag: bool, + pub log_level: Option, pub log_file: Option, pub log_format: Option, pub systemd_log: bool, @@ -53,6 +55,7 @@ impl From<&crate::Opts> for ObservabilityConfig { fn from(opts: &crate::Opts) -> Self { Self { log_debug_flag: opts.global.debug, + log_level: opts.youki_extend.log_level.to_owned(), log_file: opts.global.log.to_owned(), log_format: opts.global.log_format.to_owned(), systemd_log: opts.youki_extend.systemd_log, @@ -65,8 +68,8 @@ where T: Into, { let config = config.into(); - let level = - detect_log_level(config.log_debug_flag).with_context(|| "failed to parse log level")?; + let level = detect_log_level(config.log_level, config.log_debug_flag) + .with_context(|| "failed to parse log level")?; let log_level_filter = tracing_subscriber::filter::LevelFilter::from(level); let log_format = detect_log_format(config.log_format.as_deref()) .with_context(|| "failed to detect log format")?; @@ -149,62 +152,52 @@ where mod tests { use super::*; use libcontainer::test_utils::TestCallbackError; - use serial_test::serial; - use std::{env, path::Path}; - - struct LogLevelGuard { - original_level: Option, - } - - impl LogLevelGuard { - fn new(level: &str) -> Result { - let original_level = env::var(LOG_LEVEL_ENV_NAME).ok(); - env::set_var(LOG_LEVEL_ENV_NAME, level); - Ok(Self { original_level }) - } - } - - impl Drop for LogLevelGuard { - fn drop(self: &mut LogLevelGuard) { - if let Some(level) = self.original_level.as_ref() { - env::set_var(LOG_LEVEL_ENV_NAME, level); - } else { - env::remove_var(LOG_LEVEL_ENV_NAME); - } - } - } + use std::path::Path; #[test] - fn test_detect_log_level_is_debug() { - let _guard = LogLevelGuard::new("error").unwrap(); - assert_eq!(detect_log_level(true).unwrap(), tracing::Level::DEBUG) + fn test_detect_log_level() { + let test = vec![ + ("error", tracing::Level::ERROR), + ("warn", tracing::Level::WARN), + ("info", tracing::Level::INFO), + ("debug", tracing::Level::DEBUG), + ("trace", tracing::Level::TRACE), + ]; + for (input, expected) in test { + assert_eq!( + detect_log_level(Some(input.to_string()), false) + .expect("failed to parse log level"), + expected + ) + } + assert_eq!( + detect_log_level(None, true).expect("failed to parse log level"), + tracing::Level::DEBUG + ); + // Invalid log level should fail the parse + assert!(detect_log_level(Some("invalid".to_string()), false).is_err()); } #[test] - #[serial] fn test_detect_log_level_default() { - let _guard = LogLevelGuard::new("error").unwrap(); - env::remove_var(LOG_LEVEL_ENV_NAME); if cfg!(debug_assertions) { - assert_eq!(detect_log_level(false).unwrap(), tracing::Level::DEBUG) + assert_eq!( + detect_log_level(None, false).unwrap(), + tracing::Level::DEBUG + ) } else { - assert_eq!(detect_log_level(false).unwrap(), tracing::Level::WARN) + assert_eq!( + detect_log_level(None, false).unwrap(), + tracing::Level::ERROR + ) } } - #[test] - #[serial] - fn test_detect_log_level_from_env() { - let _guard = LogLevelGuard::new("error").unwrap(); - assert_eq!(detect_log_level(false).unwrap(), tracing::Level::ERROR) - } - #[test] fn test_init_many_times() -> Result<()> { let cb = || { let temp_dir = tempfile::tempdir().expect("failed to create temp dir"); let log_file = Path::join(temp_dir.path(), "test.log"); - let _guard = LogLevelGuard::new("error").unwrap(); let config = ObservabilityConfig { log_file: Some(log_file), ..Default::default() @@ -224,11 +217,11 @@ mod tests { libcontainer::test_utils::test_in_child_process(|| { let temp_dir = tempfile::tempdir().expect("failed to create temp dir"); let log_file = Path::join(temp_dir.path(), "test.log"); - let _guard = LogLevelGuard::new("error").unwrap(); // Note, we can only init the tracing once, so we have to test in a // single unit test. The orders are important here. let config = ObservabilityConfig { log_file: Some(log_file.clone()), + log_level: Some("error".to_string()), ..Default::default() }; init(config).map_err(|err| TestCallbackError::Other(err.into()))?; @@ -268,7 +261,6 @@ mod tests { libcontainer::test_utils::test_in_child_process(|| { let temp_dir = tempfile::tempdir().expect("failed to create temp dir"); let log_file = Path::join(temp_dir.path(), "test.log"); - let _guard = LogLevelGuard::new("error").unwrap(); // Note, we can only init the tracing once, so we have to test in a // single unit test. The orders are important here. let config = ObservabilityConfig { diff --git a/docs/src/user/basic_usage.md b/docs/src/user/basic_usage.md index b5fe0cffc..1c3c058b3 100644 --- a/docs/src/user/basic_usage.md +++ b/docs/src/user/basic_usage.md @@ -171,3 +171,13 @@ cd .. ./youki list ./youki delete rootless_container ``` + +#### Log level + +`youki` defaults the log level to `error` in the release build. In the debug +build, the log level defaults to `debug`. The `--log-level` flag can be used to +set the log-level. For least amount of log, we recommend using the `error` log +level. For the most spammy logging, we have a `trace` level. + +For compatibility with `runc` and `crun`, we have a `--debug` flag to set the +log level to `debug`. This flag is ignored if `--log-level` is also set. diff --git a/scripts/rust_integration_tests.sh b/scripts/rust_integration_tests.sh index 4d36ca270..553ea4402 100755 --- a/scripts/rust_integration_tests.sh +++ b/scripts/rust_integration_tests.sh @@ -24,7 +24,7 @@ if [ ! -f ${ROOT}/bundle.tar.gz ]; then fi touch ${LOGFILE} -sudo YOUKI_LOG_LEVEL="error" ${ROOT}/integration_test run --runtime "$RUNTIME" --runtimetest ${ROOT}/runtimetest > $LOGFILE +sudo ${ROOT}/integration_test run --runtime "$RUNTIME" --runtimetest ${ROOT}/runtimetest > $LOGFILE if [ 0 -ne $(grep "not ok" $LOGFILE | wc -l ) ]; then cat $LOGFILE diff --git a/tests/rust-integration-tests/integration_test/src/utils/test_utils.rs b/tests/rust-integration-tests/integration_test/src/utils/test_utils.rs index 207a9f0db..565c468aa 100644 --- a/tests/rust-integration-tests/integration_test/src/utils/test_utils.rs +++ b/tests/rust-integration-tests/integration_test/src/utils/test_utils.rs @@ -48,9 +48,6 @@ pub fn create_container>(id: &str, dir: P) -> Result { // in test_inside_container function .stdout(Stdio::piped()) .stderr(Stdio::piped()) - // set log level to error only, otherwise - // we get warnings in stderr - .env("YOUKI_LOG_LEVEL", "error") .arg("--root") .arg(dir.as_ref().join("runtime")) .arg("create") diff --git a/tests/rust-integration-tests/runtimetest/src/utils.rs b/tests/rust-integration-tests/runtimetest/src/utils.rs index 42784f361..ab991a079 100644 --- a/tests/rust-integration-tests/runtimetest/src/utils.rs +++ b/tests/rust-integration-tests/runtimetest/src/utils.rs @@ -217,7 +217,7 @@ pub fn test_mount_releatime_option(path: &str) -> Result<(), std::io::Error> { println!( "{:?} file three metadata atime is {:?}", test_file_path, - two_metadata.atime() + three_metadata.atime() ); if two_metadata.atime() != three_metadata.atime() { return Err(std::io::Error::new( @@ -282,8 +282,8 @@ pub fn test_mount_noreleatime_option(path: &str) -> Result<(), std::io::Error> { println!( "{:?} file three atime is {:?},mtime is {:?},current time is {:?}", test_file_path, - two_metadata.atime(), - two_metadata.mtime(), + three_metadata.atime(), + three_metadata.mtime(), std::time::SystemTime::now() );