use crate::prelude::*;
use colored::Colorize;
pub mod message;
pub mod report;
#[derive_group(Serializers)]
#[derive(Debug, Clone, JsonSchema)]
pub struct Diagnostics {
pub kind: Kind,
pub span: Vec<hax_frontend_exporter::Span>,
pub context: String,
pub owner_id: Option<hax_frontend_exporter::DefId>,
}
impl std::fmt::Display for Diagnostics {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
write!(f, "({}) ", self.context)?;
match &self.kind {
Kind::Unimplemented { issue_id, details } => write!(
f,
"something is not implemented yet.{}{}",
match issue_id {
Some(id) => format!("This is discussed in issue https://github.com/hacspec/hax/issues/{id}.\nPlease upvote or comment this issue if you see this error message."),
_ => "".to_string(),
},
match details {
Some(details) => format!("\n{}", details),
_ => "".to_string(),
}
),
Kind::UnsupportedMacro { id } => write!(
f,
"The unexpanded macro {} it is not supported by this backend. Please verify the option you passed the {} (or {}) option.",
id.bold(),
"--inline-macro-call".bold(), "-i".bold()
),
Kind::UnsafeBlock => write!(f, "Unsafe blocks are not allowed."),
Kind::AssertionFailure {details} => write!(
f,
"Fatal error: something we considered as impossible occurred! {}\nDetails: {}",
"Please report this by submitting an issue on GitHub!".bold(),
details
),
Kind::UnallowedMutRef => write!(
f,
"The mutation of this {} is not allowed here.",
"&mut".bold()
),
Kind::ExpectedMutRef => write!(
f,
"At this position, Hax was expecting an expression of the shape `&mut _`. Hax forbids `f(x)` (where `f` expects a mutable reference as input) when `x` is not a {}{} or when it is a dereference expression.
{}
",
"place expression".bold(),
"[1]".bright_black(),
"[1]: https://doc.rust-lang.org/reference/expressions.html#place-expressions-and-value-expressions"
),
Kind::ClosureMutatesParentBindings {bindings} => write!(
f,
"The bindings {:?} cannot be mutated here: they don't belong to the closure scope, and this is not allowed.",
bindings
),
Kind::ArbitraryLHS => write!(f, "Assignation of an arbitrary left-hand side is not supported. `lhs = e` is fine only when `lhs` is a combination of local identifiers, field accessors and index accessors."),
Kind::AttributeRejected {reason} => write!(f, "Here, this attribute cannot be used: {reason}."),
Kind::NonTrivialAndMutFnInput => write!(f, "The support in hax of function with one or more inputs of type `&mut _` is limited. Onlu trivial patterns are allowed there: `fn f(x: &mut (T, U)) ...` is allowed while `f((x, y): &mut (T, U))` is rejected."),
Kind::FStarParseError { fstar_snippet, details: _ } => write!(f, "The following code snippet could not be parsed as valid F*:\n```\n{fstar_snippet}\n```"),
_ => write!(f, "{:?}", self.kind),
}
}
}
#[derive_group(Serializers)]
#[derive(Debug, Clone, JsonSchema)]
#[repr(u16)]
pub enum Kind {
UnsafeBlock = 0,
Unimplemented {
issue_id: Option<u32>,
details: Option<String>,
} = 1,
AssertionFailure {
details: String,
} = 2,
UnallowedMutRef = 3,
UnsupportedMacro {
id: String,
} = 4,
ErrorParsingMacroInvocation {
macro_id: String,
details: String,
} = 5,
ClosureMutatesParentBindings {
bindings: Vec<String>,
} = 6,
ArbitraryLHS = 7,
ExplicitRejection {
reason: String,
} = 8,
UnsupportedTupleSize {
tuple_size: u32,
reason: String,
} = 9,
ExpectedMutRef = 10,
NonTrivialAndMutFnInput = 11,
AttributeRejected {
reason: String,
} = 12,
FStarParseError {
fstar_snippet: String,
details: String,
} = 13,
}
impl Kind {
pub fn discriminant(&self) -> u16 {
unsafe { *(self as *const Self as *const u16) }
}
pub fn code(&self) -> String {
format!("HAX{:0>4}", self.discriminant())
}
}