slicec 0.3.3

The Slice parser and other core components for Slice compilers.
Documentation
// Copyright (c) ZeroC, Inc.

use crate::diagnostics::{Diagnostic, Diagnostics, Error};
use crate::grammar::*;

use std::collections::HashMap;

pub fn validate_enum(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    backing_type_bounds(enum_def, diagnostics);
    allowed_underlying_types(enum_def, diagnostics);
    enumerator_values_are_unique(enum_def, diagnostics);
    underlying_type_cannot_be_optional(enum_def, diagnostics);
    nonempty_if_checked(enum_def, diagnostics);
    check_compact_modifier(enum_def, diagnostics);
    compact_enums_cannot_contain_tags(enum_def, diagnostics);

    // Fields in Slice1 files are already rejected by `encoding_patcher`.
    if enum_def.underlying.is_some() && !enum_def.supported_encodings().supports(Encoding::Slice1) {
        cannot_contain_fields(enum_def, diagnostics);
    }
}

/// Validate that the enumerators are within the bounds of the specified underlying type.
fn backing_type_bounds(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if enum_def.supported_encodings().supports(Encoding::Slice1) {
        // Enum was defined in a Slice1 file, so it's underlying type is int32 and its enumerators must be positive.
        for enumerator in enum_def.enumerators() {
            let value = enumerator.value();
            if value < 0 || value > i32::MAX as i128 {
                Diagnostic::new(Error::EnumeratorValueOutOfBounds {
                    enumerator_identifier: enumerator.identifier().to_owned(),
                    value,
                    min: 0,
                    max: i32::MAX as i128,
                })
                .set_span(enumerator.span())
                .push_into(diagnostics);
            }
        }
    } else {
        // Enum was defined in a Slice2 file.

        fn check_bounds(enum_def: &Enum, (min, max): (i128, i128), diagnostics: &mut Diagnostics) {
            enum_def
                .enumerators()
                .iter()
                .filter(|enumerator| enumerator.value() < min || enumerator.value() > max)
                .for_each(|enumerator| {
                    let error = Error::EnumeratorValueOutOfBounds {
                        enumerator_identifier: enumerator.identifier().to_owned(),
                        value: enumerator.value(),
                        min,
                        max,
                    };
                    Diagnostic::new(error)
                        .set_span(enumerator.span())
                        .push_into(diagnostics);
                });
        }
        match &enum_def.underlying {
            Some(underlying_type) => {
                // Non-integral underlying types are rejected by the `allowed_underlying_types` check.
                if let Some(bounds) = underlying_type.numeric_bounds() {
                    check_bounds(enum_def, bounds, diagnostics);
                }
            }
            None => {
                // For enumerators in Slice2, values must fit within varint32 and be positive.
                const VARINT32_MAX: i128 = i32::MAX as i128;
                check_bounds(enum_def, (0, VARINT32_MAX), diagnostics);
            }
        }
    }
}

/// Validate that the backing type (if present) is an integral type.
fn allowed_underlying_types(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if let Some(underlying_type) = &enum_def.underlying {
        if !underlying_type.is_integral() {
            Diagnostic::new(Error::EnumUnderlyingTypeNotSupported {
                enum_identifier: enum_def.identifier().to_owned(),
                kind: Some(underlying_type.definition().kind().to_owned()),
            })
            .set_span(enum_def.span())
            .push_into(diagnostics);
        }
    }
}

/// Validate that enumerator values aren't re-used within an enum.
fn enumerator_values_are_unique(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    let mut value_to_enumerator_map: HashMap<i128, &Enumerator> = HashMap::new();
    for enumerator in enum_def.enumerators() {
        // If the value is already in the map, another enumerator already used it. Get that enumerator from the map
        // and report an error. Otherwise add the enumerator and its value to the map.
        if let Some(alt_enum) = value_to_enumerator_map.get(&enumerator.value()) {
            Diagnostic::new(Error::DuplicateEnumeratorValue {
                enumerator_value: enumerator.value(),
            })
            .set_span(enumerator.span())
            .add_note(
                format!("the value was previously used by '{}' here:", alt_enum.identifier()),
                Some(alt_enum.span()),
            )
            .push_into(diagnostics);
        } else {
            value_to_enumerator_map.insert(enumerator.value(), enumerator);
        }
    }
}

/// Validate the underlying type of an enum is not optional.
fn underlying_type_cannot_be_optional(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if let Some(ref typeref) = enum_def.underlying {
        if typeref.is_optional {
            Diagnostic::new(Error::CannotUseOptionalUnderlyingType {
                enum_identifier: enum_def.identifier().to_owned(),
            })
            .set_span(enum_def.span())
            .push_into(diagnostics);
        }
    }
}

/// Validate that a checked enum must not be empty.
fn nonempty_if_checked(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if !enum_def.is_unchecked && enum_def.enumerators.is_empty() {
        Diagnostic::new(Error::MustContainEnumerators {
            enum_identifier: enum_def.identifier().to_owned(),
        })
        .set_span(enum_def.span())
        .push_into(diagnostics);
    }
}

/// Validate that this enum's enumerators don't specify any fields.
/// This function should only be called for enums with underlying types.
fn cannot_contain_fields(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    debug_assert!(enum_def.underlying.is_some());

    for enumerator in enum_def.enumerators() {
        if enumerator.fields.is_some() {
            Diagnostic::new(Error::EnumeratorCannotContainFields {
                enumerator_identifier: enumerator.identifier().to_owned(),
            })
            .set_span(enumerator.span())
            .add_note(
                "an underlying type was specified here:",
                Some(enum_def.underlying.as_ref().unwrap().span()),
            )
            .push_into(diagnostics);
        }
    }
}

/// Validate that compact enums do not have an underlying type and are not marked as 'unchecked'.
fn check_compact_modifier(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if enum_def.is_compact {
        if let Some(underlying) = &enum_def.underlying {
            Diagnostic::new(Error::CannotBeCompact {
                kind: enum_def.kind(),
                identifier: enum_def.identifier().to_owned(),
            })
            .set_span(enum_def.span())
            .add_note(
                "compact enums cannot also have underlying types; try removing either the 'compact' modifier, or the underlying type",
                Some(underlying.span()),
            )
            .push_into(diagnostics);
        }

        if enum_def.is_unchecked {
            Diagnostic::new(Error::CannotBeCompact {
                kind: enum_def.kind(),
                identifier: enum_def.identifier().to_owned(),
            })
            .set_span(enum_def.span())
            .add_note(
                "An enum cannot be both unchecked and compact - try removing the 'compact' modifier",
                None,
            )
            .push_into(diagnostics);
        }
    }
}

/// Validate that tags cannot be used in compact enums.
fn compact_enums_cannot_contain_tags(enum_def: &Enum, diagnostics: &mut Diagnostics) {
    if enum_def.is_compact {
        for enumerator in enum_def.enumerators() {
            for field in enumerator.fields() {
                if field.is_tagged() {
                    Diagnostic::new(Error::CompactTypeCannotContainTaggedFields { kind: enum_def.kind() })
                        .set_span(field.span())
                        .add_note(
                            format!("enum '{}' is declared compact here", enum_def.identifier()),
                            Some(enum_def.span()),
                        )
                        .push_into(diagnostics);
                }
            }
        }
    }
}