//--------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner Microsoft // @backupOwner Microsoft //--------------------------------------------------------------------- using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Data.EntityModel.SchemaObjectModel; using System.Data.Metadata.Edm; using System.Data.Entity; using System.Diagnostics; namespace System.Data.EntityModel.SchemaObjectModel { /// /// Helper methods used for Schema Object Model (validation) validation. /// internal static class ValidationHelper { /// /// Validates whether facets are declared correctly. /// /// Schema element being validated. Must not be null. /// Resolved type (from declaration on the element). Possibly null. /// TypeUsageBuilder for the current element. Must not be null. internal static void ValidateFacets(SchemaElement element, SchemaType type, TypeUsageBuilder typeUsageBuilder) { Debug.Assert(element != null); Debug.Assert(typeUsageBuilder != null); if (type != null) { var schemaEnumType = type as SchemaEnumType; if (schemaEnumType != null) { typeUsageBuilder.ValidateEnumFacets(schemaEnumType); } else if(!(type is ScalarType) && typeUsageBuilder.HasUserDefinedFacets) { Debug.Assert(!(type is SchemaEnumType), "Note that enums should have already been handled."); // Non-scalar type should not have Facets. element.AddError(ErrorCode.FacetOnNonScalarType, EdmSchemaErrorSeverity.Error, Strings.FacetsOnNonScalarType(type.FQName)); } } else { if(typeUsageBuilder.HasUserDefinedFacets) { // Type attribute not specified but facets exist. element.AddError(ErrorCode.IncorrectlyPlacedFacet, EdmSchemaErrorSeverity.Error, Strings.FacetDeclarationRequiresTypeAttribute); } } } /// /// Validated whether a type is declared correctly. /// /// Schema element being validated. Must not be null. /// Resolved type (from declaration on the element). Possibly null. /// Child schema element. Possibly null. /// /// For some elements (e.g. ReturnType) we allow the type to be defined inline in an attribute on the element itself or /// by using nested elements. These definitions are mutually exclusive. /// internal static void ValidateTypeDeclaration(SchemaElement element, SchemaType type, SchemaElement typeSubElement) { Debug.Assert(element != null); if (type == null && typeSubElement == null) { //Type not declared as either attribute or subelement element.AddError(ErrorCode.TypeNotDeclared, EdmSchemaErrorSeverity.Error, Strings.TypeMustBeDeclared); } if (type != null && typeSubElement != null) { //Both attribute and sub-element declarations exist element.AddError(ErrorCode.TypeDeclaredAsAttributeAndElement, EdmSchemaErrorSeverity.Error, Strings.TypeDeclaredAsAttributeAndElement); } } /// /// Validate that reference type is an entity type. /// /// Schema element being validated. Must not be null. /// Resolved type (from declaration on the element). Possibly null. internal static void ValidateRefType(SchemaElement element, SchemaType type) { Debug.Assert(element != null); if (type != null && !(type is SchemaEntityType)) { // Ref type refers to non entity type. element.AddError(ErrorCode.ReferenceToNonEntityType, EdmSchemaErrorSeverity.Error, Strings.ReferenceToNonEntityType(type.FQName)); } } } }