use crate::errors::SolidHunterError;
use colored::Colorize;
use serde::{Deserialize, Serialize};
use std::fmt;
pub type LintResult = Result<Vec<LintDiag>, SolidHunterError>;
#[derive(Clone, Serialize, Deserialize, Debug)]
pub struct LintDiag {
pub range: Range,
pub severity: Severity,
#[serde(skip_serializing_if = "Option::is_none")]
pub code: Option<NumberOrString>,
#[serde(skip_serializing_if = "Option::is_none")]
pub source: Option<String>,
pub message: String,
pub id: String,
pub uri: Uri,
#[serde(rename = "sourceFileContent")]
pub source_file_content: String,
}
fn compute_format_line_padding(line: usize) -> String {
let padding: String;
if line > 99 {
padding = " ".repeat(0);
} else if line > 9 {
padding = " ".to_string();
} else {
padding = " ".repeat(2);
}
padding
}
fn try_trim_max_offset(line: &str, max_offset: usize) -> (&str, usize) {
let mut offset: usize = 0;
for (i, c) in line.chars().enumerate() {
if i >= max_offset {
break;
}
if c.is_whitespace() {
offset += 1;
}
}
(&line[offset..], offset)
}
impl LintDiag {
fn format_highlighted_lines(&self) -> String {
let mut formatted = " |\n".to_string();
let first_line = self
.source_file_content
.lines()
.nth(self.range.start.line - 1)
.unwrap();
let trimmed_first_line = first_line.trim_start();
let max_offset = first_line.len() - trimmed_first_line.len();
for line_nb in self.range.start.line..self.range.end.line + 1 {
let line = self.source_file_content.lines().nth(line_nb - 1).unwrap();
let (trimmed_line, offset) = try_trim_max_offset(line, max_offset);
let mut higlight_length = trimmed_line.len();
if self.range.start.line == self.range.end.line {
higlight_length = self.range.end.character - self.range.start.character;
} else if line_nb == self.range.start.line {
higlight_length = trimmed_line.len() - (self.range.start.character - offset);
} else if line_nb == self.range.end.line {
higlight_length = trimmed_line.len() - (self.range.end.character - offset) + 1;
}
formatted = format!(
"{}{}{}| {}\n | {}{}\n",
formatted,
line_nb,
compute_format_line_padding(line_nb),
trimmed_line,
" ".repeat(if line_nb == self.range.start.line {
self.range.start.character - offset
} else {
0
}),
"^".repeat(higlight_length)
);
}
formatted
}
}
impl fmt::Display for LintDiag {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"\n{}: {}\n --> {}:{}:{}\n |\n{}",
self.severity,
self.message,
self.uri,
self.range.start.line,
self.range.start.character,
self.format_highlighted_lines()
)
}
}
impl fmt::Display for Severity {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let severity = match self {
Severity::ERROR => "error".to_string().red(),
Severity::WARNING => "warning".to_string().yellow(),
Severity::INFO => "info".to_string().blue(),
Severity::HINT => "hint".to_string().green(),
};
write!(f, "{}", severity)
}
}
impl PartialEq for Position {
fn eq(&self, other: &Self) -> bool {
self.line == other.line && self.character == other.character
}
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub struct Position {
pub line: usize,
pub character: usize,
}
#[derive(PartialEq, Eq, Clone, Copy, Serialize, Deserialize, Debug)]
pub enum Severity {
ERROR = 1,
WARNING = 2,
INFO = 3,
HINT = 4,
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub struct Range {
pub start: Position,
pub end: Position,
}
impl Range {
pub fn compute_length(&self, content: &str) -> usize {
if self.start.line == self.end.line {
self.end.character - self.start.character
} else {
let mut length = 0;
let mut line = self.start.line;
let mut character = self.start.character;
while line < self.end.line {
let line_content = content.lines().nth(line - 1).unwrap();
length += line_content.len() + 1 - character;
line += 1;
character = 0;
}
length += self.end.character - character;
length
}
}
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub enum NumberOrString {
Number(i32),
String(String),
}
type Uri = String;