use std::{
fmt,
ops::{Range, RangeInclusive},
};
use crate::descriptor::{FileDescriptorInner, FileIndex};
pub struct DescriptorError {
errors: Box<[DescriptorErrorKind]>,
#[cfg(feature = "miette")]
source: Option<miette::NamedSource<String>>,
}
#[derive(Debug)]
pub(super) enum DescriptorErrorKind {
MissingRequiredField {
label: Label,
},
UnknownSyntax {
syntax: String,
found: Label,
},
DuplicateFileName {
name: String,
},
FileNotFound {
name: String,
found: Label,
},
InvalidImportIndex,
InvalidOneofIndex,
DuplicateName {
name: String,
first: Label,
second: Label,
},
DuplicateFieldNumber {
number: u32,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
first: Label,
second: Label,
},
DuplicateFieldJsonName {
name: String,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
first: Label,
second: Label,
},
DuplicateFieldCamelCaseName {
first_name: String,
second_name: String,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
first: Label,
second: Label,
},
InvalidFieldNumber {
number: i32,
found: Label,
},
FieldNumberInReservedRange {
number: i32,
range: Range<i32>,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
defined: Label,
found: Label,
},
FieldNumberInExtensionRange {
number: i32,
range: Range<i32>,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
defined: Label,
found: Label,
},
ExtensionNumberOutOfRange {
number: i32,
message: String,
found: Label,
},
NameNotFound {
name: String,
found: Label,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
help: Option<String>,
},
NameShadowed {
name: String,
shadowed_name: String,
found: Label,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
help: Option<String>,
},
InvalidType {
name: String,
expected: String,
found: Label,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
defined: Label,
},
InvalidFieldDefault {
value: String,
kind: String,
found: Label,
},
EmptyEnum {
found: Label,
},
InvalidProto3EnumDefault {
found: Label,
},
DuplicateEnumNumber {
number: i32,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
first: Label,
second: Label,
},
EnumNumberInReservedRange {
number: i32,
range: RangeInclusive<i32>,
#[cfg_attr(not(feature = "miette"), allow(dead_code))]
defined: Label,
found: Label,
},
OptionNotFound {
name: String,
found: Label,
},
InvalidOptionType {
name: String,
ty: String,
value: String,
is_last: bool,
found: Label,
},
InvalidOptionExtendee {
name: String,
expected_extendee: String,
actual_extendee: String,
found: Label,
},
#[cfg(feature = "text-format")]
InvalidMessageOption {
name: String,
ty: String,
found: Label,
err: crate::text_format::ParseError,
},
DuplicateOption {
name: String,
found: Label,
},
DecodeFileDescriptorSet {
err: prost::DecodeError,
},
}
#[derive(Debug)]
pub(super) struct Label {
file: String,
path: Box<[i32]>,
span: Option<[i32; 4]>,
#[cfg(feature = "miette")]
message: String,
#[cfg(feature = "miette")]
resolved: Option<miette::SourceSpan>,
}
impl DescriptorError {
pub(super) fn new(errors: Vec<DescriptorErrorKind>) -> DescriptorError {
debug_assert!(!errors.is_empty());
DescriptorError {
errors: errors.into(),
#[cfg(feature = "miette")]
source: None,
}
}
pub fn file(&self) -> Option<&str> {
self.first().label().map(|l| l.file.as_str())
}
pub fn line(&self) -> Option<usize> {
self.first()
.label()
.and_then(|l| l.span)
.map(|s| s[0] as usize)
}
pub fn column(&self) -> Option<usize> {
self.first()
.label()
.and_then(|l| l.span)
.map(|s| s[1] as usize)
}
pub fn path(&self) -> Option<&[i32]> {
self.first().label().map(|l| l.path.as_ref())
}
#[cfg(feature = "miette")]
#[cfg_attr(docsrs, doc(cfg(feature = "miette")))]
pub fn with_source_code(mut self, source: &str) -> Self {
if let Some(file) = self.file() {
let file = file.to_owned();
self.source = Some(miette::NamedSource::new(&file, source.to_owned()));
for error in self.errors.as_mut() {
error.add_source_code(&file, source);
}
}
self
}
fn first(&self) -> &DescriptorErrorKind {
&self.errors[0]
}
}
impl std::error::Error for DescriptorError {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
self.first().source()
}
}
impl fmt::Display for DescriptorError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.first().fmt(f)
}
}
impl fmt::Debug for DescriptorError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if let Some(file) = self.file() {
write!(f, "{}:", file)?;
if let (Some(line), Some(column)) = (self.line(), self.column()) {
write!(f, "{}:{}:", line + 1, column + 1)?;
}
write!(f, " ")?;
}
write!(f, "{}", self)
}
}
#[cfg(feature = "miette")]
#[cfg_attr(docsrs, doc(cfg(feature = "miette")))]
impl miette::Diagnostic for DescriptorError {
fn code<'a>(&'a self) -> Option<Box<dyn fmt::Display + 'a>> {
self.first().code()
}
fn severity(&self) -> Option<miette::Severity> {
self.first().severity()
}
fn help<'a>(&'a self) -> Option<Box<dyn fmt::Display + 'a>> {
self.first().help()
}
fn url<'a>(&'a self) -> Option<Box<dyn fmt::Display + 'a>> {
self.first().url()
}
fn source_code(&self) -> Option<&dyn miette::SourceCode> {
match &self.source {
Some(source) => Some(source),
None => None,
}
}
fn labels(&self) -> Option<Box<dyn Iterator<Item = miette::LabeledSpan> + '_>> {
self.first().labels()
}
fn related<'a>(&'a self) -> Option<Box<dyn Iterator<Item = &'a dyn miette::Diagnostic> + 'a>> {
if self.errors.len() > 1 {
Some(Box::new(
self.errors
.iter()
.map(|e| e as &dyn miette::Diagnostic)
.skip(1),
))
} else {
None
}
}
fn diagnostic_source(&self) -> Option<&dyn miette::Diagnostic> {
self.first().diagnostic_source()
}
}
impl DescriptorErrorKind {
fn label(&self) -> Option<&Label> {
match self {
DescriptorErrorKind::MissingRequiredField { label } => Some(label),
DescriptorErrorKind::UnknownSyntax { found, .. } => Some(found),
DescriptorErrorKind::DuplicateFileName { .. } => None,
DescriptorErrorKind::FileNotFound { found, .. } => Some(found),
DescriptorErrorKind::InvalidImportIndex => None,
DescriptorErrorKind::InvalidOneofIndex => None,
DescriptorErrorKind::DuplicateName { second, .. } => Some(second),
DescriptorErrorKind::DuplicateFieldNumber { second, .. } => Some(second),
DescriptorErrorKind::DuplicateFieldJsonName { second, .. } => Some(second),
DescriptorErrorKind::DuplicateFieldCamelCaseName { second, .. } => Some(second),
DescriptorErrorKind::InvalidFieldNumber { found, .. } => Some(found),
DescriptorErrorKind::FieldNumberInReservedRange { found, .. } => Some(found),
DescriptorErrorKind::FieldNumberInExtensionRange { found, .. } => Some(found),
DescriptorErrorKind::ExtensionNumberOutOfRange { found, .. } => Some(found),
DescriptorErrorKind::NameNotFound { found, .. } => Some(found),
DescriptorErrorKind::NameShadowed { found, .. } => Some(found),
DescriptorErrorKind::InvalidType { found, .. } => Some(found),
DescriptorErrorKind::InvalidFieldDefault { found, .. } => Some(found),
DescriptorErrorKind::EmptyEnum { found } => Some(found),
DescriptorErrorKind::InvalidProto3EnumDefault { found } => Some(found),
DescriptorErrorKind::DuplicateEnumNumber { second, .. } => Some(second),
DescriptorErrorKind::EnumNumberInReservedRange { found, .. } => Some(found),
DescriptorErrorKind::OptionNotFound { found, .. } => Some(found),
DescriptorErrorKind::InvalidOptionType { found, .. } => Some(found),
DescriptorErrorKind::InvalidOptionExtendee { found, .. } => Some(found),
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { found, .. } => Some(found),
DescriptorErrorKind::DuplicateOption { found, .. } => Some(found),
DescriptorErrorKind::DecodeFileDescriptorSet { .. } => None,
}
}
#[cfg(feature = "miette")]
fn add_source_code(&mut self, file: &str, source: &str) {
match self {
DescriptorErrorKind::MissingRequiredField { label } => {
label.resolve_span(file, source);
}
DescriptorErrorKind::UnknownSyntax { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateFileName { .. } => {}
DescriptorErrorKind::FileNotFound { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::InvalidImportIndex => {}
DescriptorErrorKind::InvalidOneofIndex => {}
DescriptorErrorKind::DuplicateName { first, second, .. } => {
first.resolve_span(file, source);
second.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateFieldNumber { first, second, .. } => {
first.resolve_span(file, source);
second.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateFieldJsonName { first, second, .. } => {
first.resolve_span(file, source);
second.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateFieldCamelCaseName { first, second, .. } => {
first.resolve_span(file, source);
second.resolve_span(file, source);
}
DescriptorErrorKind::InvalidFieldNumber { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::FieldNumberInReservedRange { defined, found, .. } => {
defined.resolve_span(file, source);
found.resolve_span(file, source);
}
DescriptorErrorKind::FieldNumberInExtensionRange { defined, found, .. } => {
defined.resolve_span(file, source);
found.resolve_span(file, source);
}
DescriptorErrorKind::ExtensionNumberOutOfRange { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::NameNotFound { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::NameShadowed { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::InvalidType { found, defined, .. } => {
found.resolve_span(file, source);
defined.resolve_span(file, source);
}
DescriptorErrorKind::InvalidFieldDefault { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::EmptyEnum { found } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::InvalidProto3EnumDefault { found } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateEnumNumber { first, second, .. } => {
first.resolve_span(file, source);
second.resolve_span(file, source);
}
DescriptorErrorKind::EnumNumberInReservedRange { defined, found, .. } => {
found.resolve_span(file, source);
defined.resolve_span(file, source);
}
DescriptorErrorKind::OptionNotFound { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::InvalidOptionType { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::InvalidOptionExtendee { found, .. } => {
found.resolve_span(file, source);
}
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::DuplicateOption { found, .. } => {
found.resolve_span(file, source);
}
DescriptorErrorKind::DecodeFileDescriptorSet { .. } => {}
}
}
}
impl std::error::Error for DescriptorErrorKind {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
DescriptorErrorKind::DecodeFileDescriptorSet { err } => Some(err),
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { err, .. } => Some(err),
_ => None,
}
}
}
impl fmt::Display for DescriptorErrorKind {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
DescriptorErrorKind::MissingRequiredField { label } => {
write!(f, "missing required field at {:?}", label.path)
}
DescriptorErrorKind::UnknownSyntax { syntax, .. } => {
write!(f, "unknown syntax '{}'", syntax)
}
DescriptorErrorKind::DuplicateFileName { name, .. } => {
write!(
f,
"a different file named '{}' has already been added",
name
)
}
DescriptorErrorKind::FileNotFound { name, .. } => {
write!(f, "imported file '{}' has not been added", name)
}
DescriptorErrorKind::InvalidImportIndex => {
write!(f, "invalid import index")
}
DescriptorErrorKind::InvalidOneofIndex => {
write!(f, "invalid oneof index")
}
DescriptorErrorKind::DuplicateName {
name,
first,
second,
} => {
if first.file == second.file {
write!(f, "name '{}' is defined twice", name)
} else {
write!(
f,
"name '{}' is already defined in file '{}'",
name, first.file
)
}
}
DescriptorErrorKind::DuplicateFieldNumber { number, .. } => {
write!(f, "field number '{}' is already used", number)
}
DescriptorErrorKind::DuplicateFieldJsonName { name, .. } => {
write!(f, "a field with JSON name '{}' is already defined", name)
}
DescriptorErrorKind::DuplicateFieldCamelCaseName {
first_name,
second_name,
..
} => {
write!(
f,
"camel-case name of field '{first_name}' conflicts with field '{second_name}'"
)
}
DescriptorErrorKind::InvalidFieldNumber { number, .. } => {
write!(f, "invalid field number '{}'", number)
}
DescriptorErrorKind::FieldNumberInReservedRange { number, range, .. } => {
write!(
f,
"field number '{}' conflicts with reserved range '{} to {}'",
number,
range.start,
range.end - 1
)
}
DescriptorErrorKind::FieldNumberInExtensionRange { number, range, .. } => {
write!(
f,
"field number '{}' conflicts with extension range '{} to {}'",
number,
range.start,
range.end - 1
)
}
DescriptorErrorKind::ExtensionNumberOutOfRange {
number, message, ..
} => {
write!(
f,
"message '{}' does not define '{}' as an extension number",
message, number
)
}
DescriptorErrorKind::NameNotFound { name, .. } => {
write!(f, "name '{}' is not defined", name)
}
DescriptorErrorKind::NameShadowed {
name,
shadowed_name,
..
} => {
write!(
f,
"'{name}' resolves to '{shadowed_name}', which is not defined",
)
}
DescriptorErrorKind::InvalidType { name, expected, .. } => {
write!(f, "'{}' is not {}", name, expected)
}
DescriptorErrorKind::InvalidFieldDefault { value, kind, .. } => {
write!(f, "invalid default value '{}' for type '{}'", value, kind)
}
DescriptorErrorKind::EmptyEnum { .. } => {
write!(f, "enums must have at least one value")
}
DescriptorErrorKind::InvalidProto3EnumDefault { .. } => {
write!(f, "the first value for proto3 enums must be 0")
}
DescriptorErrorKind::DuplicateEnumNumber { number, .. } => {
write!(f, "enum number '{}' has already been used", number)
}
DescriptorErrorKind::EnumNumberInReservedRange { number, range, .. } => {
write!(
f,
"enum number '{}' conflicts with reserved range '{} to {}'",
number,
range.start(),
range.end()
)
}
DescriptorErrorKind::OptionNotFound { name, .. } => {
write!(f, "option field '{}' is not defined", name)
}
DescriptorErrorKind::InvalidOptionType {
name,
ty,
value,
is_last,
..
} => {
if *is_last {
write!(
f,
"expected a value of type '{}' for option '{}', but found '{}'",
ty, name, value
)
} else {
write!(
f,
"cannot set field for option '{}' value of type '{}'",
name, ty
)
}
}
DescriptorErrorKind::InvalidOptionExtendee {
name,
expected_extendee,
actual_extendee,
..
} => {
write!(
f,
"expected an extension to type '{}', but '{}' extends '{}'",
expected_extendee, name, actual_extendee
)
}
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { name, ty, .. } => {
write!(f, "invalid value of type '{}' for option '{}'", ty, name)
}
DescriptorErrorKind::DuplicateOption { name, .. } => {
write!(f, "option field '{}' has already been set", name)
}
DescriptorErrorKind::DecodeFileDescriptorSet { .. } => {
write!(f, "failed to decode file descriptor set")
}
}
}
}
#[cfg(feature = "miette")]
#[cfg_attr(docsrs, doc(cfg(feature = "miette")))]
impl miette::Diagnostic for DescriptorErrorKind {
fn help<'a>(&'a self) -> Option<Box<dyn fmt::Display + 'a>> {
use crate::descriptor::{RESERVED_MESSAGE_FIELD_NUMBERS, VALID_MESSAGE_FIELD_NUMBERS};
match self {
DescriptorErrorKind::MissingRequiredField { .. } => None,
DescriptorErrorKind::UnknownSyntax { .. } => {
Some(Box::new("valid values are 'proto2' and 'proto3'"))
}
DescriptorErrorKind::DuplicateFileName { .. } => None,
DescriptorErrorKind::FileNotFound { .. } => None,
DescriptorErrorKind::InvalidImportIndex => None,
DescriptorErrorKind::InvalidOneofIndex => None,
DescriptorErrorKind::DuplicateName { .. } => None,
DescriptorErrorKind::DuplicateFieldNumber { .. } => None,
DescriptorErrorKind::InvalidFieldNumber { number, .. } => {
if !VALID_MESSAGE_FIELD_NUMBERS.contains(number) {
Some(Box::new(format!(
"field numbers must be between {} and {}",
VALID_MESSAGE_FIELD_NUMBERS.start,
VALID_MESSAGE_FIELD_NUMBERS.end - 1
)))
} else if RESERVED_MESSAGE_FIELD_NUMBERS.contains(number) {
Some(Box::new(format!(
"field numbers {} to {} are reserved",
RESERVED_MESSAGE_FIELD_NUMBERS.start,
RESERVED_MESSAGE_FIELD_NUMBERS.end - 1
)))
} else {
None
}
}
DescriptorErrorKind::FieldNumberInReservedRange { .. } => None,
DescriptorErrorKind::FieldNumberInExtensionRange { .. } => None,
DescriptorErrorKind::DuplicateFieldJsonName { .. } => None,
DescriptorErrorKind::DuplicateFieldCamelCaseName { .. } => None,
DescriptorErrorKind::NameNotFound { help, .. }
| DescriptorErrorKind::NameShadowed { help, .. } => help
.as_ref()
.map(|h| -> Box<dyn fmt::Display> { Box::new(h.clone()) }),
DescriptorErrorKind::InvalidType { .. } => None,
DescriptorErrorKind::InvalidFieldDefault { .. } => None,
DescriptorErrorKind::EmptyEnum { .. } => None,
DescriptorErrorKind::InvalidProto3EnumDefault { .. } => None,
DescriptorErrorKind::DuplicateEnumNumber { .. } => Some(Box::new(
"set the 'allow_alias' option allow re-using enum numbers",
)),
DescriptorErrorKind::EnumNumberInReservedRange { .. } => None,
DescriptorErrorKind::OptionNotFound { .. } => None,
DescriptorErrorKind::InvalidOptionType { .. } => None,
DescriptorErrorKind::InvalidOptionExtendee { .. } => None,
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { .. } => None,
DescriptorErrorKind::DuplicateOption { .. } => None,
DescriptorErrorKind::DecodeFileDescriptorSet { .. } => None,
DescriptorErrorKind::ExtensionNumberOutOfRange { .. } => None,
}
}
fn labels(&self) -> Option<Box<dyn Iterator<Item = miette::LabeledSpan> + '_>> {
let mut spans = Vec::new();
match self {
DescriptorErrorKind::MissingRequiredField { label } => spans.extend(label.to_span()),
DescriptorErrorKind::UnknownSyntax { found: defined, .. } => {
spans.extend(defined.to_span());
}
DescriptorErrorKind::DuplicateFileName { .. } => {}
DescriptorErrorKind::FileNotFound { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidImportIndex => {}
DescriptorErrorKind::InvalidOneofIndex => {}
DescriptorErrorKind::DuplicateName { first, second, .. } => {
spans.extend(first.to_span());
spans.extend(second.to_span());
}
DescriptorErrorKind::DuplicateFieldNumber { first, second, .. } => {
spans.extend(first.to_span());
spans.extend(second.to_span());
}
DescriptorErrorKind::DuplicateFieldJsonName { first, second, .. } => {
spans.extend(first.to_span());
spans.extend(second.to_span());
}
DescriptorErrorKind::DuplicateFieldCamelCaseName { first, second, .. } => {
spans.extend(first.to_span());
spans.extend(second.to_span());
}
DescriptorErrorKind::NameNotFound { found, .. }
| DescriptorErrorKind::NameShadowed { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidFieldNumber { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::FieldNumberInReservedRange { defined, found, .. } => {
spans.extend(defined.to_span());
spans.extend(found.to_span());
}
DescriptorErrorKind::FieldNumberInExtensionRange { defined, found, .. } => {
spans.extend(defined.to_span());
spans.extend(found.to_span());
}
DescriptorErrorKind::ExtensionNumberOutOfRange { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidType { found, defined, .. } => {
spans.extend(found.to_span());
spans.extend(defined.to_span());
}
DescriptorErrorKind::InvalidFieldDefault { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::EmptyEnum { found } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidProto3EnumDefault { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::DuplicateEnumNumber { first, second, .. } => {
spans.extend(first.to_span());
spans.extend(second.to_span());
}
DescriptorErrorKind::EnumNumberInReservedRange { defined, found, .. } => {
spans.extend(found.to_span());
spans.extend(defined.to_span());
}
DescriptorErrorKind::OptionNotFound { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidOptionType { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::InvalidOptionExtendee { found, .. } => {
spans.extend(found.to_span());
}
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::DuplicateOption { found, .. } => {
spans.extend(found.to_span());
}
DescriptorErrorKind::DecodeFileDescriptorSet { .. } => {}
}
if spans.is_empty() {
None
} else {
Some(Box::new(spans.into_iter()))
}
}
fn diagnostic_source(&self) -> Option<&dyn miette::Diagnostic> {
match self {
#[cfg(feature = "text-format")]
DescriptorErrorKind::InvalidMessageOption { err, .. } => Some(err),
_ => None,
}
}
}
impl Label {
pub fn new(
files: &[FileDescriptorInner],
#[cfg_attr(not(feature = "miette"), allow(unused_variables))] message: impl ToString,
file: FileIndex,
path: Box<[i32]>,
) -> Self {
let file = &files[file as usize].raw;
let span = file
.source_code_info
.as_ref()
.and_then(|s| s.location.iter().find(|l| *l.path == *path))
.and_then(|l| match *l.span {
[start_line, start_col, end_col] => {
Some([start_line, start_col, start_line, end_col])
}
[start_line, start_col, end_line, end_col] => {
Some([start_line, start_col, end_line, end_col])
}
_ => None,
});
Label {
file: file.name().to_owned(),
span,
path,
#[cfg(feature = "miette")]
message: message.to_string(),
#[cfg(feature = "miette")]
resolved: None,
}
}
#[cfg(feature = "miette")]
pub fn resolve_span(&mut self, file: &str, source: &str) {
if file == self.file {
if let Some([start_line, start_col, end_line, end_col]) = self.span {
let start = miette::SourceOffset::from_location(
source,
start_line.saturating_add(1) as _,
start_col.saturating_add(1) as _,
)
.offset();
let end = miette::SourceOffset::from_location(
source,
end_line.saturating_add(1) as _,
end_col.saturating_add(1) as _,
)
.offset();
self.resolved = Some(miette::SourceSpan::from(start..end));
}
}
}
#[cfg(feature = "miette")]
fn to_span(&self) -> Option<miette::LabeledSpan> {
match self.resolved {
Some(span) if !span.is_empty() => Some(miette::LabeledSpan::new_with_span(
Some(self.message.clone()),
span,
)),
_ => None,
}
}
}