Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Store failure reason into assess metadata #2166

Merged
merged 7 commits into from
Feb 1, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions kani-driver/src/assess/args.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,9 @@ pub enum AssessSubcommand {
/// `cargo kani assess scan` subcommand arguments
#[derive(Debug, Parser)]
pub struct ScanArgs {
// TODO: When assess scan is invoked using `--existing-only`, it should check if the Kani version
// from the existing metadata files matches the current version. Otherwise, the results may no
// longer hold.
/// Don't run assess on found packages, just re-analyze the results from a previous run
#[arg(long, hide = true)]
pub existing_only: bool,
Expand Down
75 changes: 53 additions & 22 deletions kani-driver/src/assess/metadata.rs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,11 @@ use super::AssessArgs;
/// This is not a stable interface.
#[derive(Serialize, Deserialize)]
pub struct AssessMetadata {
/// Kani version that was used to generate the metadata.
#[serde(rename = "kani_version")]
pub version: String,
/// Contains an error message in cases where it failed the execution.
pub error: Option<SessionError>,
/// Report on the presence of `codegen_unimplemented` in the analyzed packages
pub unsupported_features: TableBuilder<UnsupportedFeaturesTableRow>,
/// Report of the reasons why tests could not be analyzed by Kani
Expand All @@ -35,32 +40,62 @@ pub struct AssessMetadata {
pub promising_tests: TableBuilder<PromisingTestsTableRow>,
}

impl AssessMetadata {
pub fn new(
unsupported_features: TableBuilder<UnsupportedFeaturesTableRow>,
failure_reasons: TableBuilder<FailureReasonsTableRow>,
promising_tests: TableBuilder<PromisingTestsTableRow>,
) -> AssessMetadata {
AssessMetadata {
version: env!("CARGO_PKG_VERSION").to_string(),
error: None,
unsupported_features,
failure_reasons,
promising_tests,
}
}

pub fn from_error(err: &dyn std::error::Error) -> AssessMetadata {
let error = Some(SessionError {
root_cause: err.source().map(|e| format!("{e:#}")),
msg: err.to_string(),
});
AssessMetadata {
version: env!("CARGO_PKG_VERSION").to_string(),
error,
unsupported_features: TableBuilder::new(),
failure_reasons: TableBuilder::new(),
promising_tests: TableBuilder::new(),
}
}
pub fn empty() -> AssessMetadata {
AssessMetadata {
version: env!("CARGO_PKG_VERSION").to_string(),
error: None,
unsupported_features: TableBuilder::new(),
failure_reasons: TableBuilder::new(),
promising_tests: TableBuilder::new(),
}
}
}

#[derive(Serialize, Deserialize, Debug)]
pub struct SessionError {
pub root_cause: Option<String>,
pub msg: String,
}

/// If given the argument to so do, write the assess metadata to the target file.
pub(crate) fn write_metadata(args: &AssessArgs, build: AssessMetadata) -> Result<()> {
pub(crate) fn write_metadata(args: &AssessArgs, metadata: AssessMetadata) -> Result<()> {
if let Some(path) = &args.emit_metadata {
let out_file = File::create(&path)?;
let writer = BufWriter::new(out_file);
// use pretty for now to keep things readable and debuggable, but this should change eventually
serde_json::to_writer_pretty(writer, &build)?;
serde_json::to_writer_pretty(writer, &metadata)?;
}
Ok(())
}

/// Write metadata with unsupported features only, supporting the `--only-codegen` option.
pub(crate) fn write_partial_metadata(
args: &AssessArgs,
unsupported_features: TableBuilder<UnsupportedFeaturesTableRow>,
) -> Result<()> {
write_metadata(
args,
AssessMetadata {
unsupported_features,
failure_reasons: TableBuilder::new(),
promising_tests: TableBuilder::new(),
},
)
}

/// Read assess metadata from a file.
pub(crate) fn read_metadata(path: &Path) -> Result<AssessMetadata> {
// this function already exists, but a proxy here helps find it :)
Expand All @@ -72,11 +107,7 @@ pub(crate) fn read_metadata(path: &Path) -> Result<AssessMetadata> {
/// This is not a complicated operation, because the assess metadata structure is meant
/// to accomodate multiple packages already, so we're just "putting it together".
pub(crate) fn aggregate_metadata(metas: Vec<AssessMetadata>) -> AssessMetadata {
let mut result = AssessMetadata {
unsupported_features: TableBuilder::new(),
failure_reasons: TableBuilder::new(),
promising_tests: TableBuilder::new(),
};
let mut result = AssessMetadata::empty();
for meta in metas {
for item in meta.unsupported_features.build() {
result.unsupported_features.add(item.clone());
Expand Down
30 changes: 21 additions & 9 deletions kani-driver/src/assess/mod.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
// Copyright Kani Contributors
// SPDX-License-Identifier: Apache-2.0 OR MIT

use self::metadata::{write_metadata, AssessMetadata};
use anyhow::Result;
use kani_metadata::KaniMetadata;

use crate::assess::table_builder::TableBuilder;
use crate::metadata::merge_kani_metadata;
use crate::project;
use crate::session::KaniSession;
Expand All @@ -21,11 +23,23 @@ mod table_unsupported_features;
/// `cargo kani assess` main entry point.
///
/// See <https://model-checking.github.io/kani/dev-assess.html>
pub(crate) fn cargokani_assess_main(mut session: KaniSession, args: AssessArgs) -> Result<()> {
pub(crate) fn run_assess(session: KaniSession, args: AssessArgs) -> Result<()> {
if let Some(args::AssessSubcommand::Scan(args)) = &args.command {
return scan::assess_scan_main(session, args);
}

let result = assess_project(session);
match result {
Ok(metadata) => write_metadata(&args, metadata),
Err(err) => {
let metadata = AssessMetadata::from_error(err.as_ref());
write_metadata(&args, metadata)?;
Err(err.context("Failed to assess project"))
}
}
}

fn assess_project(mut session: KaniSession) -> Result<AssessMetadata> {
// Fix (as in "make unchanging/unchangable") some settings.
// This is a temporary hack to make things work, until we get around to refactoring how arguments
// work generally in kani-driver. These arguments, for instance, are all prepended to the subcommand,
Expand Down Expand Up @@ -69,8 +83,11 @@ pub(crate) fn cargokani_assess_main(mut session: KaniSession, args: AssessArgs)
}

if session.args.only_codegen {
metadata::write_partial_metadata(&args, unsupported_features)?;
return Ok(());
return Ok(AssessMetadata::new(
unsupported_features,
TableBuilder::new(),
TableBuilder::new(),
));
}

// Done with the 'cargo-kani' part, now we're going to run *test* harnesses instead of proof:
Expand All @@ -95,12 +112,7 @@ pub(crate) fn cargokani_assess_main(mut session: KaniSession, args: AssessArgs)
let promising_tests = table_promising_tests::build(&results);
println!("{}", promising_tests.render());

metadata::write_metadata(
&args,
metadata::AssessMetadata { unsupported_features, failure_reasons, promising_tests },
)?;

Ok(())
Ok(AssessMetadata::new(unsupported_features, failure_reasons, promising_tests))
}

/// Merges a collection of Kani metadata by figuring out which package each belongs to, from cargo metadata.
Expand Down
74 changes: 71 additions & 3 deletions kani-driver/src/call_cargo.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,11 @@
use crate::args::KaniArgs;
use crate::session::{KaniSession, ReachabilityMode};
use anyhow::{bail, Context, Result};
use cargo_metadata::{Metadata, MetadataCommand, Package};
use cargo_metadata::diagnostic::{Diagnostic, DiagnosticLevel};
use cargo_metadata::{Message, Metadata, MetadataCommand, Package};
use std::ffi::OsString;
use std::fs;
use std::io::BufReader;
use std::path::{Path, PathBuf};
use std::process::Command;
use tracing::{debug, trace};
Expand Down Expand Up @@ -81,6 +83,10 @@ impl KaniSession {
cargo_args.push("--target-dir".into());
cargo_args.push(target_dir.into());

// Configuration needed to parse cargo compilation status.
cargo_args.push("--message-format".into());
cargo_args.push("json-diagnostic-rendered-ansi".into());

if self.args.tests {
// Use test profile in order to pull dev-dependencies and compile using `--test`.
// Initially the plan was to use `--tests` but that brings in multiple targets.
Expand Down Expand Up @@ -120,9 +126,10 @@ impl KaniSession {
.args(&pkg_args)
.env("RUSTC", &self.kani_compiler)
.env("RUSTFLAGS", "--kani-flags")
.env("KANIFLAGS", &crate::util::join_osstring(&kani_args, " "));
.env("KANIFLAGS", &crate::util::join_osstring(&kani_args, " "))
.env("CARGO_TERM_PROGRESS_WHEN", "never");

self.run_terminal(cmd)?;
self.run_cargo(cmd)?;
found_target = true;
}
}
Expand Down Expand Up @@ -167,6 +174,67 @@ impl KaniSession {

cmd.exec().context("Failed to get cargo metadata.")
}

/// Run cargo and collect any error found.
/// TODO: We should also use this to collect the artifacts generated by cargo.
fn run_cargo(&self, cargo_cmd: Command) -> Result<()> {
let support_color = atty::is(atty::Stream::Stdout);
if let Some(mut cargo_process) = self.run_piped(cargo_cmd)? {
let reader = BufReader::new(cargo_process.stdout.take().unwrap());
let mut error_count = 0;
for message in Message::parse_stream(reader) {
let message = message.unwrap();
match message {
Message::CompilerMessage(msg) => match msg.message.level {
DiagnosticLevel::FailureNote => {
print_msg(&msg.message, support_color)?;
}
DiagnosticLevel::Error => {
error_count += 1;
print_msg(&msg.message, support_color)?;
}
DiagnosticLevel::Ice => {
print_msg(&msg.message, support_color)?;
let _ = cargo_process.wait();
return Err(anyhow::Error::msg(msg.message).context(format!(
"Failed to compile `{}` due to an internal compiler error.",
msg.target.name
)));
}
_ => {
if !self.args.quiet {
print_msg(&msg.message, support_color)?;
}
}
},
Message::CompilerArtifact(_)
| Message::BuildScriptExecuted(_)
| Message::BuildFinished(_) => {
// do nothing
}
// Non-exhaustive enum.
_ => {}
}
}
let status = cargo_process.wait()?;
if !status.success() {
bail!(
"Failed to execute cargo ({status}). Found {error_count} compilation errors."
);
}
}
Ok(())
}
}

/// Print the compiler message following the coloring schema.
fn print_msg(diagnostic: &Diagnostic, use_rendered: bool) -> Result<()> {
if use_rendered {
print!("{diagnostic}");
} else {
print!("{}", console::strip_ansi_codes(diagnostic.rendered.as_ref().unwrap()));
}
Ok(())
}

/// Given a `path` with glob characters in it (e.g. `*.json`), return a vector of matching files
Expand Down
4 changes: 2 additions & 2 deletions kani-driver/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -65,9 +65,9 @@ fn cargokani_main(input_args: Vec<OsString>) -> Result<()> {
let session = session::KaniSession::new(args.common_opts)?;

if let Some(CargoKaniSubcommand::Assess(args)) = args.command {
return assess::cargokani_assess_main(session, args);
return assess::run_assess(session, args);
} else if session.args.assess {
return assess::cargokani_assess_main(session, assess::AssessArgs::default());
return assess::run_assess(session, assess::AssessArgs::default());
}

let project = project::cargo_project(&session)?;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,2 @@
error: Crate crate_with_global_asm contains global ASM, which is not supported by Kani. Rerun with `--enable-unstable --ignore-global-asm` to suppress this error (**Verification results may be impacted**).
error: could not compile `crate_with_global_asm` due to previous error
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a useful line that points out the name of the crate that had an error. Is the crate name available elsewhere?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There seems to be a bug on Cargo that it counts an extra error if you use --message-format=json. Basically, the compiler prints an error message saying how many errors were found. In regular mode, cargo excludes this message, but when emitting json, cargo skips that, and counts it as an extra error.

I.e., for this test to pass, I have three options:

  1. Remove the check from here.
  2. Update the error count:
error: could not compile `crate_with_global_asm` due to 2 previous errors
  1. Update the expected message to not include the number of errors:
error: could not compile `crate_with_global_asm`

I picked the first since it was the simplest, but I can change. Any preference?

Copy link
Contributor

@zhassan-aws zhassan-aws Jan 31, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see. In that case, the option you went with is fine. I just wanted to make sure the message still gets printed.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

FYI, I updated this one expected test to have the message that includes the name of the crate and the counter as a sanity check, but I didn't bother adding to all the other tests, because one would hope that Cargo will update that counter eventually. In that case, we would only need to update one test.

error: cargo exited with status exit status: 101
error: could not compile `crate_with_global_asm` due to 2 previous errors
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,3 @@ error: unable to resolve function/method: crate::other_crate2::mock
error: unable to resolve function/method: super::other_crate2::mock
error: unable to resolve function/method: self::other_crate2::mock
error: unable to resolve function/method: other_crate1::mock
error: could not compile `stubbing-do-not-resolve` due to 4 previous errors
12 changes: 12 additions & 0 deletions tests/cargo-ui/assess-error/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
# Copyright Kani Contributors
# SPDX-License-Identifier: Apache-2.0 OR MIT

[package]
name = "compilation-error"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[package.metadata.kani]
flags = { assess=true, enable-unstable=true }
1 change: 1 addition & 0 deletions tests/cargo-ui/assess-error/expected
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
error: Failed to assess project: Failed to execute cargo (exit status: 101). Found 3 compilation errors.
19 changes: 19 additions & 0 deletions tests/cargo-ui/assess-error/src/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
// Copyright Kani Contributors
// SPDX-License-Identifier: Apache-2.0 OR MIT
//! Check that the compilation error detection works as expected
use std::option;

pub fn add(left: usize, right: u32) -> usize {
left + right
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn it_works() {
let result = add(2, 2);
assert_eq!(result, 4);
}
}
Original file line number Diff line number Diff line change
@@ -1,3 +1,2 @@
error: `&str` doesn't implement `DoIt`. The function `foo` cannot be stubbed by `bar` due to generic bounds not being met.
error: `&str` doesn't implement `std::cmp::PartialEq`. The function `foo` cannot be stubbed by `bar` due to generic bounds not being met.
error: could not compile `function-stubbing-trait-mismatch` due to 2 previous errors