aldrin_parser/error/
duplicate_service_item.rsuse super::Error;
use crate::ast::{Ident, ServiceDef};
use crate::diag::{Diagnostic, DiagnosticKind, Formatted, Formatter};
use crate::validate::Validate;
use crate::{util, Parsed, Span};
#[derive(Debug)]
pub struct DuplicateServiceItem {
schema_name: String,
duplicate: Ident,
first: Span,
service_ident: Ident,
}
impl DuplicateServiceItem {
pub(crate) fn validate(service: &ServiceDef, validate: &mut Validate) {
let mut fallback_dup = false;
util::find_duplicates(
service.items(),
|item| item.name().value(),
|duplicate, first| {
validate.add_error(Self {
schema_name: validate.schema_name().to_owned(),
duplicate: duplicate.name().clone(),
first: first.name().span(),
service_ident: service.name().clone(),
})
},
);
if let Some(fallback) = service.function_fallback() {
for item in service.items() {
if fallback.name().value() == item.name().value() {
validate.add_error(Self {
schema_name: validate.schema_name().to_owned(),
duplicate: fallback.name().clone(),
first: item.name().span(),
service_ident: service.name().clone(),
});
fallback_dup = true;
break;
}
}
}
if let Some(fallback) = service.event_fallback() {
for item in service.items() {
if fallback.name().value() == item.name().value() {
validate.add_error(Self {
schema_name: validate.schema_name().to_owned(),
duplicate: fallback.name().clone(),
first: item.name().span(),
service_ident: service.name().clone(),
});
fallback_dup = true;
break;
}
}
}
if !fallback_dup {
if let (Some(func), Some(ev)) = (service.function_fallback(), service.event_fallback())
{
if func.name().value() == ev.name().value() {
let (duplicate, first) = if func.span().from < ev.span().from {
(ev.name().clone(), func.name().span())
} else {
(func.name().clone(), ev.name().span())
};
validate.add_error(Self {
schema_name: validate.schema_name().to_owned(),
duplicate,
first,
service_ident: service.name().clone(),
});
}
}
}
}
pub fn duplicate(&self) -> &Ident {
&self.duplicate
}
pub fn first(&self) -> Span {
self.first
}
pub fn service_ident(&self) -> &Ident {
&self.service_ident
}
}
impl Diagnostic for DuplicateServiceItem {
fn kind(&self) -> DiagnosticKind {
DiagnosticKind::Error
}
fn schema_name(&self) -> &str {
&self.schema_name
}
fn format<'a>(&'a self, parsed: &'a Parsed) -> Formatted<'a> {
let mut fmt = Formatter::new(
self,
format!(
"duplicate item `{}` in service `{}`",
self.duplicate.value(),
self.service_ident.value()
),
);
if let Some(schema) = parsed.get_schema(&self.schema_name) {
fmt.main_block(
schema,
self.duplicate.span().from,
self.duplicate.span(),
"duplicate defined here",
)
.info_block(schema, self.first.from, self.first, "first defined here");
}
fmt.format()
}
}
impl From<DuplicateServiceItem> for Error {
fn from(e: DuplicateServiceItem) -> Self {
Self::DuplicateServiceItem(e)
}
}