File: System\Data\EntityModel\SchemaObjectModel\Parameter.cs
Project: ndp\fx\src\DataEntity\System.Data.Entity.csproj (System.Data.Entity)
//---------------------------------------------------------------------
// <copyright file="Parameter.cs" company="Microsoft">
//      Copyright (c) Microsoft Corporation.  All rights reserved.
// </copyright>
//
// @owner       Microsoft
// @backupOwner Microsoft
//---------------------------------------------------------------------
 
namespace System.Data.EntityModel.SchemaObjectModel
{
    using System;
    using System.Collections.Generic;
    using System.Data;
    using System.Data.Entity;
    using System.Data.Metadata.Edm;
    using System.Diagnostics;
    using System.Globalization;
    using System.Text;
    using System.Xml;
    using Som = System.Data.EntityModel.SchemaObjectModel;
 
    /// <summary>
    /// Summary description for StructuredProperty.
    /// </summary>
    internal class Parameter : FacetEnabledSchemaElement
    {
        #region Instance Fields
        private ParameterDirection _parameterDirection = ParameterDirection.Input;
        private CollectionKind _collectionKind = CollectionKind.None;
        private ModelFunctionTypeElement _typeSubElement = null;
        private bool _isRefType = false;
        #endregion
 
        #region constructor
        /// <summary>
        /// 
        /// </summary>
        /// <param name="parentElement"></param>
        internal Parameter(Function parentElement)
            : base(parentElement)
        {
            _typeUsageBuilder = new TypeUsageBuilder(this);
        }
 
        #endregion
 
        #region Public Properties
 
        internal ParameterDirection ParameterDirection
        {
            get
            {
                return _parameterDirection;
            }
        }
 
        internal CollectionKind CollectionKind
        {
            get
            {
                return _collectionKind;
            }
            set
            {
                _collectionKind = value;
            }
        }
 
        internal bool IsRefType
        {
            get { return _isRefType; }
        }
 
        internal override TypeUsage TypeUsage
        {
            get 
            {
                if (_typeSubElement != null)
                {
                    return _typeSubElement.GetTypeUsage();
                }
                else if (base.TypeUsage == null)
                {
                    return null;
                }
                else if (CollectionKind != CollectionKind.None)
                {
                    return TypeUsage.Create(new CollectionType(base.TypeUsage));
                }
                else
                {
                    return base.TypeUsage;
                }
            }
        }
 
        #endregion
 
        new internal SchemaType Type
        {
            get
            {
                return _type;
            }
        }
 
 
        internal void WriteIdentity(StringBuilder builder)
        {
            builder.Append("Parameter(");
            if (UnresolvedType != null && !UnresolvedType.Trim().Equals(String.Empty))
            {
                if (_collectionKind != CollectionKind.None)
                {
                    builder.Append("Collection(" + UnresolvedType + ")");
                }
                else if (_isRefType)
                {
                    builder.Append("Ref(" + UnresolvedType + ")");
                }
                else
                {
                    builder.Append(UnresolvedType);
                }
            }
            else if (_typeSubElement!=null)
            {
                _typeSubElement.WriteIdentity(builder);
            }
            builder.Append(")");
        }
 
        internal override SchemaElement Clone(SchemaElement parentElement)
        {
            Parameter parameter = new Parameter((Function)parentElement);
            parameter._collectionKind = _collectionKind;
            parameter._parameterDirection = _parameterDirection;
            parameter._type = _type;
            parameter.Name = this.Name;
            parameter._typeUsageBuilder = this._typeUsageBuilder;
            return parameter;
        }
 
        internal bool ResolveNestedTypeNames(Converter.ConversionCache convertedItemCache, Dictionary<Som.SchemaElement, GlobalItem> newGlobalItems)
        {
            if (_typeSubElement == null)
            {
                return false;
            }
            return _typeSubElement.ResolveNameAndSetTypeUsage(convertedItemCache, newGlobalItems);
        }
 
 
        protected override bool HandleAttribute(XmlReader reader)
        {
            if (base.HandleAttribute(reader))
            {
                return true;
            }
            else if (CanHandleAttribute(reader, XmlConstants.TypeElement))
            {
                HandleTypeAttribute(reader);
                return true;
            }
            else if (CanHandleAttribute(reader, XmlConstants.Mode))
            {
                HandleModeAttribute(reader);
                return true;
            }
            else if (_typeUsageBuilder.HandleAttribute(reader))
            {
                return true;
            }
 
            return false;
        }
 
        #region Private Methods
 
        private void HandleTypeAttribute(XmlReader reader)
        {
            Debug.Assert(reader != null);
            Debug.Assert(UnresolvedType == null);
 
            string type;
            if (!Utils.GetString(Schema, reader, out type))
                return;
 
            TypeModifier typeModifier;
 
            Function.RemoveTypeModifier(ref type, out typeModifier, out _isRefType);
 
            switch (typeModifier)
            {
                case TypeModifier.Array:
                    CollectionKind = CollectionKind.Bag;
                    break;
                default:
                    Debug.Assert(typeModifier == TypeModifier.None, string.Format(CultureInfo.CurrentCulture, "Type is not valid for property {0}: {1}. The modifier for the type cannot be used in this context.", FQName, reader.Value));
                    break;
            }
 
            if (!Utils.ValidateDottedName(Schema, reader, type))
                return;
 
            UnresolvedType = type;
        }
 
        private void HandleModeAttribute(XmlReader reader)
        {
            Debug.Assert(reader != null);
 
            string value = reader.Value;
 
            if (String.IsNullOrEmpty(value))
            {
                return;
            }
 
            value = value.Trim();
 
            if (!String.IsNullOrEmpty(value))
            {
                switch (value)
                {
                    case XmlConstants.In:
                        _parameterDirection = ParameterDirection.Input;
                        break;
                    case XmlConstants.Out:
                        _parameterDirection = ParameterDirection.Output;
                        if (this.ParentElement.IsComposable && this.ParentElement.IsFunctionImport) 
                        {
                            AddErrorBadParameterDirection(value, reader, System.Data.Entity.Strings.BadParameterDirectionForComposableFunctions);
                        }
                        break;
                    case XmlConstants.InOut:
                        _parameterDirection = ParameterDirection.InputOutput;
                        if (this.ParentElement.IsComposable && this.ParentElement.IsFunctionImport)
                        {
                            AddErrorBadParameterDirection(value, reader, System.Data.Entity.Strings.BadParameterDirectionForComposableFunctions);
                        }
                        break;
                    default:
                        {                           
                            AddErrorBadParameterDirection(value, reader, System.Data.Entity.Strings.BadParameterDirection);
                        }
                        break;
                }
            }
        }
 
        private void AddErrorBadParameterDirection(string value, XmlReader reader, Func<object, object, object, object, string> errorFunc)
        {
            // don't try to identify the parameter by any of the attributes
            // because we are still parsing attributes, and we don't know which ones
            // have been parsed yet.
            AddError(ErrorCode.BadParameterDirection, EdmSchemaErrorSeverity.Error, reader,
                       errorFunc(
                                this.ParentElement.Parameters.Count, // indexed at 0 to be similar to the old exception
                                this.ParentElement.Name,
                                this.ParentElement.ParentElement.FQName,
                                value));
        }
 
        protected override bool HandleElement(XmlReader reader)
        {
            if (base.HandleElement(reader))
            {
                return true;
            }
            else if (CanHandleElement(reader, XmlConstants.CollectionType))
            {
                HandleCollectionTypeElement(reader);
                return true;
            }
            else if (CanHandleElement(reader, XmlConstants.ReferenceType))
            {
                HandleReferenceTypeElement(reader);
                return true;
            }
            else if (CanHandleElement(reader, XmlConstants.TypeRef))
            {
                HandleTypeRefElement(reader);
                return true;
            }
            else if (CanHandleElement(reader, XmlConstants.RowType))
            {
                HandleRowTypeElement(reader);
                return true;
            }
            else if (Schema.DataModel == SchemaDataModelOption.EntityDataModel)
            {
                if (CanHandleElement(reader, XmlConstants.ValueAnnotation))
                {
                    // EF does not support this EDM 3.0 element, so ignore it.
                    SkipElement(reader);
                    return true;
                }
                else if (CanHandleElement(reader, XmlConstants.TypeAnnotation))
                {
                    // EF does not support this EDM 3.0 element, so ignore it.
                    SkipElement(reader);
                    return true;
                }
            }
 
            return false;
        }
 
 
        protected void HandleCollectionTypeElement(XmlReader reader)
        {
            Debug.Assert(reader != null);
 
            var subElement = new CollectionTypeElement(this);
            subElement.Parse(reader);
            _typeSubElement = subElement;
        }
 
        protected void HandleReferenceTypeElement(XmlReader reader)
        {
            Debug.Assert(reader != null);
 
            var subElement = new ReferenceTypeElement(this);
            subElement.Parse(reader);
            _typeSubElement = subElement;
        }
 
        protected void HandleTypeRefElement(XmlReader reader)
        {
            Debug.Assert(reader != null);
 
            var subElement = new TypeRefElement(this);
            subElement.Parse(reader);
            _typeSubElement = subElement;
        }
 
        protected void HandleRowTypeElement(XmlReader reader)
        {
            Debug.Assert(reader != null);
 
            var subElement = new RowTypeElement(this);
            subElement.Parse(reader);
            _typeSubElement = subElement;
        }
 
        #endregion
 
        internal override void ResolveTopLevelNames()
        {
            // If type was defined as an attribute: <ReturnType Type="int"/>
            if (_unresolvedType != null)
            {
                base.ResolveTopLevelNames();
            }
 
            // If type was defined as a subelement: <ReturnType><CollectionType>...</CollectionType></ReturnType>
            if (_typeSubElement != null)
            {
                _typeSubElement.ResolveTopLevelNames();
            }
        }
 
        internal override void Validate()
        {
            base.Validate();
 
            ValidationHelper.ValidateTypeDeclaration(this, _type, _typeSubElement);
 
            if (Schema.DataModel != SchemaDataModelOption.EntityDataModel)
            {
                Debug.Assert(Schema.DataModel == SchemaDataModelOption.ProviderDataModel ||
                             Schema.DataModel == SchemaDataModelOption.ProviderManifestModel, "Unexpected data model");
 
                bool collectionAllowed = this.ParentElement.IsAggregate;
 
                // Only scalar parameters are allowed for functions in s-space.
                Debug.Assert(_typeSubElement == null, "Unexpected type subelement inside <Parameter> element.");
                if (_type != null && (_type is ScalarType == false || (!collectionAllowed && _collectionKind != CollectionKind.None)))
                {
                    string typeName = "";
                    if (_type != null)
                    {
                        typeName = Function.GetTypeNameForErrorMessage(_type, _collectionKind, _isRefType);
                    }
                    else if (_typeSubElement != null)
                    {
                        typeName = _typeSubElement.FQName;
                    }
                    if (Schema.DataModel == SchemaDataModelOption.ProviderManifestModel)
                    {
                        AddError(ErrorCode.FunctionWithNonEdmTypeNotSupported,
                                 EdmSchemaErrorSeverity.Error,
                                 this,
                                 System.Data.Entity.Strings.FunctionWithNonEdmPrimitiveTypeNotSupported(typeName, this.ParentElement.FQName));
                    }
                    else
                    {
                        AddError(ErrorCode.FunctionWithNonPrimitiveTypeNotSupported,
                                 EdmSchemaErrorSeverity.Error,
                                 this,
                                 System.Data.Entity.Strings.FunctionWithNonPrimitiveTypeNotSupported(typeName, this.ParentElement.FQName));
                    }
                    return;
                }
            }
 
            ValidationHelper.ValidateFacets(this, _type, _typeUsageBuilder);
            
            if (_isRefType)
            {
                ValidationHelper.ValidateRefType(this, _type);
            }
            
 
            if (_typeSubElement != null)
            {
                _typeSubElement.Validate();
            }
        }
    }
}