This is somewhat complicated to explain, so please bear with me.
I have an ASP.NET MVC 2 project that is slowly killing me in which I'm trying to take form data and translate it into entities to create or update, depending on the context of the situation. The most relevant parts (pseudo-code):
Entity Game
Scalar properties
EntityCollection<Platform> Platforms
And the basic workflow is:
Form data -> model bound to a DTO -> mapping the DTO to an EF4 entity with AutoMapper.
It all works well with one exception - I need to create or update the Game entity's Platforms EntityCollection with the raw integer index data contained in the DTO. So, here's what I've been trying, which does not work:
public AdminController(IArticleRepository articleRepository, IGameRepository gameRepository, INewsRepository newsRepository)
{
_articleRepository = articleRepository;
_gameRepository = gameRepository;
_newsRepository = newsRepository;
Mapper.CreateMap<AdminGameEditModel, Game>()
.BeforeMap((s, d) =>
{
if (d.Platforms.Count > 0)
{
Platform[] existing = d.Platforms.ToArray();
foreach (var plat in existing)
{
d.Platforms.Remove(plat);
}
}
foreach (var platId in s.PlatformIDs)
{
var newPlat = _gameRepository.GetPlatform(platId);
d.Platforms.Add(newPlat); // <-- where it chokes
}
})
.ForMember(dest => dest.BoxArtPath, opt => opt.Ignore())
.ForMember(dest => dest.IndexImagePath, opt => opt.Ignore())
.ForMember(dest => dest.Cons, opt => opt.MapFrom(src => String.Join("|", src.Cons)))
.ForMember(dest => dest.Pros, opt => opt.MapFrom(src => String.Join("|", src.Pros)))
.ForMember(dest => dest.LastModified, opt => opt.UseValue(DateTime.Now))
.ForMember(dest => dest.Platforms, opt => opt.Ignore());
}
To be specific, I'm getting an exception at the line I highlighted above which says:
The relationship between the two objects cannot be defined because they are attached to different ObjectContext objects.
Some research tells me that this is to be expected, as my fresh Game object has no ObjectContext that it's associated with, and a null context is considered to be a separate context. See this brief explanation from Julie Lerman for more.
Okay, so being the intrepid person I am, I figured I would simply be able to register my game with the ObjectContext and everything would be fixed. So, I tried:
Game game = new Game();
_gameRepository.RegisterGame(game);
Where RegisterGame is simply:
public void RegisterGame(Game game)
{
_siteDB.Games.AddObject(game);
}
Unfortunately, that did not work. The same exception is being thrown at the same point.
So, it looks like I'll have to add each Platform's EntityKey to my EntityCollection. Only problem is, I'm not sure how to do it.
So, any ideas?
EDIT: Progress of a sort. I tried adding just the EntityKeys of the Platform entities, like so:
Mapper.CreateMap<AdminGameEditModel, Game>()
.BeforeMap((s, d) =>
{
if (d.Platforms.Count > 0)
{
Platform[] existing = d.Platforms.ToArray();
foreach (var plat in existing)
{
d.Platforms.Remove(plat);
}
}
foreach (var platId in s.PlatformIDs)
{
var newPlat = _gameRepository.GetPlatform(platId);
d.Platforms.Add(new Platform { EntityKey = newPlat.EntityKey });
}
})
And it removes the 'two different ObjectContexts' exception. The problem is that I'm getting the following exception if I try to add or attach the new Game entity to my context:
An object with the same key already exists in the ObjectStateManager. The ObjectStateManager cannot track multiple objects with the same key.
Unfortunately, the exception doesn't specify which object it is, but I'm betting it's one of the Platform objects, since they already exist and I'm merely trying to forge a new relationship between the two.
So, the question remains: how the hell does one make a new entity and populate its EntityCollection<> property with existing entities? I can't be the only person who ever wanted to create a new entity and create new many-to-many relationships between it and existing entities, right?