//---------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//
// @owner [....]
// @backupOwner [....]
//---------------------------------------------------------------------
namespace System.Data.EntityModel.SchemaObjectModel
{
using System.Collections.Generic;
using System.Data.Metadata.Edm;
using System.Data.Objects.DataClasses;
using System.Diagnostics;
using System.Globalization;
using System.Xml;
///
/// Represents an AssociationSet element.
///
internal sealed class EntityContainerAssociationSet : EntityContainerRelationshipSet
{
// Note: If you add more fields, please make sure you handle that in the clone method
private Dictionary _relationshipEnds = new Dictionary();
private List _rolelessEnds = new List();
///
/// Constructs an EntityContainerAssociationSet
///
/// Reference to the schema element.
public EntityContainerAssociationSet( EntityContainer parentElement )
: base( parentElement )
{
}
///
/// The ends defined and infered for this AssociationSet
///
internal override IEnumerable Ends
{
get
{
foreach ( EntityContainerAssociationSetEnd end in _relationshipEnds.Values )
{
yield return end;
}
foreach ( EntityContainerAssociationSetEnd end in _rolelessEnds )
{
yield return end;
}
}
}
protected override bool HandleAttribute(XmlReader reader)
{
if (base.HandleAttribute(reader))
{
return true;
}
else if (CanHandleAttribute(reader, XmlConstants.Association))
{
HandleRelationshipTypeNameAttribute(reader);
return true;
}
return false;
}
protected override bool HandleElement(XmlReader reader)
{
if (base.HandleElement(reader))
{
return true;
}
else if (CanHandleElement(reader, XmlConstants.End))
{
HandleEndElement(reader);
return true;
}
return false;
}
///
/// The method that is called when an End element is encountered.
///
/// The XmlReader positioned at the EndElement.
private void HandleEndElement( XmlReader reader )
{
Debug.Assert( reader != null );
EntityContainerAssociationSetEnd end = new EntityContainerAssociationSetEnd( this );
end.Parse( reader );
if ( end.Role == null )
{
// we will resolve the role name later and put it in the
// normal _relationshipEnds dictionary
_rolelessEnds.Add( end );
return;
}
if ( HasEnd( end.Role ) )
{
end.AddError( ErrorCode.InvalidName, EdmSchemaErrorSeverity.Error, reader,
System.Data.Entity.Strings.DuplicateEndName(end.Name ) );
return;
}
_relationshipEnds.Add( end.Role, end );
}
internal override void ResolveTopLevelNames()
{
base.ResolveTopLevelNames();
// this just got resolved
Debug.Assert(
Relationship == null || Relationship.RelationshipKind == RelationshipKind.Association,
string.Format(CultureInfo.InvariantCulture, "The relationship referenced by the Association attribute of {0} is not an Association relationship.", FQName));
}
internal override void ResolveSecondLevelNames()
{
base.ResolveSecondLevelNames();
// the base class should have fixed up the role names on my ends
foreach (EntityContainerAssociationSetEnd end in _rolelessEnds)
{
if (end.Role != null)
{
if (HasEnd(end.Role))
{
end.AddError(ErrorCode.InvalidName, EdmSchemaErrorSeverity.Error,
System.Data.Entity.Strings.InferRelationshipEndGivesAlreadyDefinedEnd(end.EntitySet.FQName, Name));
}
else
{
_relationshipEnds.Add(end.Role, end);
}
}
// any that didn't get resolved will already have errors entered
}
_rolelessEnds.Clear();
}
///
/// Create and add a EntityContainerEnd from the IRelationshipEnd provided
///
/// The relationship end of the end to add.
/// The entitySet to associate with the relationship end.
protected override void AddEnd( IRelationshipEnd relationshipEnd, EntityContainerEntitySet entitySet )
{
Debug.Assert( relationshipEnd != null );
Debug.Assert( !_relationshipEnds.ContainsKey( relationshipEnd.Name ) );
// we expect set to be null sometimes
EntityContainerAssociationSetEnd end = new EntityContainerAssociationSetEnd( this );
end.Role = relationshipEnd.Name;
end.RelationshipEnd = relationshipEnd;
end.EntitySet = entitySet;
if ( end.EntitySet != null )
{
_relationshipEnds.Add( end.Role, end );
}
}
protected override bool HasEnd( string role )
{
return _relationshipEnds.ContainsKey( role );
}
internal override SchemaElement Clone(SchemaElement parentElement)
{
EntityContainerAssociationSet associationSet = new EntityContainerAssociationSet((EntityContainer)parentElement);
associationSet.Name = this.Name;
associationSet.Relationship = this.Relationship;
foreach (EntityContainerAssociationSetEnd end in this.Ends)
{
EntityContainerAssociationSetEnd clonedEnd = (EntityContainerAssociationSetEnd)end.Clone(associationSet);
associationSet._relationshipEnds.Add(clonedEnd.Role, clonedEnd);
}
return associationSet;
}
}
}