Handling Concurrency Failures
Printed From: IdeaBlade
Category: DevForce
Forum Name: DevForce 2009
Forum Discription: For .NET 3.5
URL: http://www.ideablade.com/forum/forum_posts.asp?TID=1515
Printed Date: 06-Apr-2025 at 4:10pm
Topic: Handling Concurrency Failures
Posted By: skingaby
Subject: Handling Concurrency Failures
Date Posted: 09-Oct-2009 at 1:48pm
So I am trying to implement some sort of concurrency management in Silverlight. It seems like I am barking up the wrong tree.
I started from the IdeaBlade DevForce\Learning Resources\040_BusObjPersistence\ConcurrencyConflicts\Samples\300WNF_WinForms\CodeCS.
Of course, this is a WinForms example, so it doesn't translate well.
On the client, we call this:
public void SaveItemList(IList<BaseEntity> itemList, Action<EntitySavedEventArgs> saveCompletedAction)
{
List<BaseEntity> saveList = new List<BaseEntity>();
foreach (BaseEntity item in itemList)
{
if (item.EntityAspect.HasChanges())
{
saveList.Add(item);
}
}
if (saveList.Count != 0)
{
EntityManager.SaveChangesAsync(saveList, args => SaveCallback(args, saveCompletedAction), userState);
}
}
protected void SaveCallback(EntitySavedEventArgs args, Action<EntitySavedEventArgs> saveCompletedAction)
{
if (args.Error != null)
{
var saveException = args.Exception as EntityManagerSaveException;
if (saveException != null)
{
switch (saveException.FailureType)
{
case PersistenceFailure.Concurrency:
ConcurrencyHandler.ConflictScenario aConflictScenario =
ConcurrencyHandler.HandleConcurrencyException(saveException,
ConcurrencyHandler.ConcurrencyResolutionStyle.DatabaseVersionWins); //try a simple handler first
break;
...
}
}
}
saveCompletedAction(args);
}
The ConcurrencyHandler looks like this:
public static ConflictScenario HandleConcurrencyException(EntityManagerSaveException saveException, ConcurrencyResolutionStyle pResolutionStyle)
{
if (saveException.EntitiesWithErrors.Count > 0)
{
_entitiesWithErrors = (IList<BaseEntity>)saveException.EntitiesWithErrors; //COLLECTION IS EMPTY
}
switch (pResolutionStyle)
{
case ConcurrencyResolutionStyle.DatabaseVersionWins:
KeepDatabaseChanges();
return ConflictScenario.NotDiscriminated;
}
}
private static IList<BaseEntity> _entitiesWithErrors;
private static EntityKeyList _keyList;
private static void KeepDatabaseChanges()
{
var keys = new List<EntityKey>();
foreach (Entity entityWithError in _entitiesWithErrors)
{
keys.Add(entityWithError.EntityAspect.EntityKey);
}
_keyList = new EntityKeyList(_entitiesWithErrors[0].GetType(), keys); //THIS IS WRONG
var em = LocalEntityManager.DefaultManager;
em.RefetchEntitiesAsync(_keyList, MergeStrategy.OverwriteChanges, RefetchCompleted, null);
} //KeepDatabaseChanges
public static void RefetchCompleted(AsyncEventArgs args)
{
if (args.Error != null)
{
foreach (BaseEntity entityWithError in _entitiesWithErrors)
{
//entityWithError.ForceRefreshInUI(); //HOW?????
}
}
}
Here are the problems I can't seem to work around. Any suggestions would be welcome.
1) saveException.EntitiesWithErrors is an empty collection, I had expected it to be populated with the entities that have a concurrency error;
2) EntityKeyList is type specific, but the EntitiesWithErrors collection is not and could have a mixture of types in it; i.e. the code above will fail because the entities being saved include Orders and OrderDetails, but the EntityKeyList will get the Type of the first one in its constructor new EntityKeyList(_entitiesWithErrors[0].GetType(), keys);
3) RefetchEntitiesAsync docs say it is a beta feature, so should I not be using it yet? Is there some other way to refetch asynchronously?
4) The entities in the UI are held in an ObservableCollection that is created in the ViewModel when the query.ExecuteAsync calls back, like so:
private void GetDealsCallback(EntityFetchedEventArgs<Deal> args)
{
var results = args.Result;
var sortedResults = from item in results orderby item.LastUpdateTms descending select item;
var obsColl = TypeHelper.CreateObservableCollection<Deal>(sortedResults);
this.DealListInternal = obsColl; //THE UI GRID IS BOUND TO DEALLISTINTERNAL
}
Once the entities are "refetched", how can I force the UI to refresh the relevant rows in the grid? I am going to have the same problem when I auto-update the entities in the cache on a push (which we have to implement somehow too, but that's a seperate post).
|
Replies:
Posted By: GregD
Date Posted: 13-Oct-2009 at 6:46am
Simon, I'm working on a Silverlight implementation of concurrency handling and will let you know something shortly.
|
Posted By: GregD
Date Posted: 13-Oct-2009 at 1:32pm
Simon:
In attempting to create a Silverlight solution, I reproduced the same issue you report, which is that the EntitiesWithErrors list is empty, when it shouldn't be. (Note that it will only ever contain a single entity, as saves are transactional and one entity at a time.)
At Kim's behest I have filed a Very High priority bug for this.
|
Posted By: skingaby
Date Posted: 14-Oct-2009 at 10:40am
Posted By: skingaby
Date Posted: 17-Oct-2009 at 1:02pm
Greg, you say the list "will only ever contain a single entity". But the EntityManager.SaveChangesAsync method is saving ALL the dirty entities in a transaction, not just one at a time. Oh, I get it. The save will fail on the first concurrency (or any other) failure and that will be the entity in the "EntitiesWithErrors" collection. Right?
|
Posted By: GregD
Date Posted: 19-Oct-2009 at 11:45am
Posted By: cjohnson84
Date Posted: 21-Dec-2009 at 7:39am
Did the problem involving the EntitiesWithErrors being empty ever get resolved? I am trying to implement concurrency handling in my DevForce Silverlight application following the same methodology described above and I'm finding the EntitieswithErrors collection is still empty. I am using DevForce Silverlight 5.2.3.1.
|
Posted By: skingaby
Date Posted: 21-Dec-2009 at 4:13pm
One thing someone at IB said was that the Domain Model namespace and the Domain Model project namespace should be the same. The bug was that when they weren't, the EntitiesWithErrors was always empty. I also think they fixed it in 5.2.4.2, but I am not sure. I have been completely rebuilding my domain projects and my app classes, so I have not been able to work on concurrency in a while.
|
Posted By: cjohnson84
Date Posted: 22-Dec-2009 at 5:36am
I verified that my Domain Model and my Domain Model project have the same namespace. Is version 5.2.4.2 available for download? I'm not seeing it on the website.
|
Posted By: skingaby
Date Posted: 22-Dec-2009 at 6:08am
Posted By: skingaby
Date Posted: 22-Dec-2009 at 6:10am
I am not sure then. Hopefully kimj will chime in and remind us what the setup needs to be to catch the concurrency failure. I will be looking at it again in the next couple of days. (I finally got my completely redone solution working last night. Yay!)
|
Posted By: cjohnson84
Date Posted: 22-Dec-2009 at 6:12am
I'm downloading 5.2.4.2 now. I didn't realize it was released. I'm going to try it and see what happens. Thank you for your help! I appreciate it!
|
Posted By: cjohnson84
Date Posted: 22-Dec-2009 at 7:10am
I verfied the problem is fixed in 5.2.4.2. Thank you skingaby for your help!
|
Posted By: kimj
Date Posted: 22-Dec-2009 at 9:01am
Thanks Simon. I just saw this question yesterday afternoon, and Simon beat me to answering it this morning.
We normally announce the availability of a new release here on the forum, but 5.2.4.x turned out to be a somewhat stealth release. You can always find the latest version and release notes on the Resources/Documentation page of our web site when in doubt.
|
Posted By: cjohnson84
Date Posted: 05-Jan-2010 at 7:07am
Was a Silverlight implementation of concurrency handling ever completed as mentioned above? If so is it available somewhere to download? The only sample in the Learning Resources is for WinForms.
|
Posted By: GregD
Date Posted: 05-Jan-2010 at 11:19am
Sorry, no Silverlight sample yet, but I'll put it on our list.
|
Posted By: cjohnson84
Date Posted: 07-Jan-2010 at 8:03am
Since there is no Silverlight example I have been examining the HandlingConcurrencyConflicts Winforms sample included in the Learning Resources.
Could someone explain to me where the "BaseEntity" class came from that the Customer class inherits from in the code-behind for the domain model?
Examining the "BaseEntity" code the comments state that the class was generated. Is this the case? If so what options in the DevForce Object Mapper were selected to create this class?
When I generate a domain model all of the objects inherit from "Entity" and I don't get a "BaseEntity" class generated.
I am assuming that the "BaseEntity" class was custom written and the author went in and manual edited the code-behind for the domain model by changing the inheritance from "Entity" to "BaseEntity" on the objects in the model.
Is this assumption correct?
|
Posted By: GregD
Date Posted: 07-Jan-2010 at 9:57am
You can inject base classes into your model using the Object Mapper, and then specify which other classes you want to inherit from them. The base classes so injected will then be generated by the Object Mapper, and all classes designated to inherit from them will be generated with that inheritance.
I haven't looked at the Concurrency solution for a while, but rather doubt that the use of a BaseEntity is in any way necessary to handle concurrency conflicts. In this as in other uses, a base class is simply a good, economical place to put any methods or properties you'd like all your entities to have.
|
Posted By: cjohnson84
Date Posted: 07-Jan-2010 at 10:01am
Thank you Greg for explaining. I'm slowly figuring all of this out!
|
Posted By: skingaby
Date Posted: 07-Jan-2010 at 6:05pm
The default will be for your entities to inherit from Entity. However, if you want to implement any common base entity logic, you will need to add a BaseEntity class (you can change the name) that inherits from Entity and tell the Ideablade Object Mapper to use that as your default Base class.
For example, all of our Oracle tables need a temporary key when they are created, so our BaseEntity class looks like this:
public abstract class BaseEntity : Entity
{
/// <summary>
/// Call to create a new instance of this type in the provided EntityManager.
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="emProvider"></param>
/// <returns></returns>
public static T Create<T>(IEntityManagerProvider emProvider) where T : BaseEntity
{
var entity = emProvider.EntityManager.CreateEntity<T>();
entity.Initialize();
return entity;
}
#region Initialize
internal void Initialize()
{
Initializing = true;
GetATemporaryKey();
InitializeDefaults();
BeforeAddToManager();
EntityAspect.AddToManager();
AfterAddToManager();
Initializing = false;
}
private void GetATemporaryKey()
{
var key = EntityAspect.EntityMetadata.KeyProperties[0];
if (key.IsAutoIncrementing)
{
try
{
EntityAspect.EntityManager.GenerateId(this, key);
}
catch (IdeaBladeException ex)
{
//eat the exception thrown if this class does not have a Generator
Logging.Logging.Log(ex);
}
catch (Exception ex)
{
Logging.Logging.Log(ex);
throw;
}
}
}
protected virtual void InitializeDefaults() { }
protected virtual void BeforeAddToManager() { }
protected virtual void AfterAddToManager() { }
public bool Initializing { get; protected set; }
#endregion
} |
|
Posted By: cjohnson84
Date Posted: 11-Jan-2010 at 6:03am
I've run across an issue in trying to implement concurrency conflict handling in my Silverlight application (DevForce 5.2.4.2). I have modeled my concurrency handler after the concurrency handler in the "HandlingConcurrencyConflicts" WinForms sample project in the Learning Resources.
I am trying to verify the case where the local user is attempting to update a record that has already been deleted by another user. I have opened the database in SQL Server management studio and I have also started my application at the same time. I delete a record from the table and then in my application I make an update to the record and then attempt to save. I get a concurrency conflict as expected but I find that because the entity has an original value I end up falling into the block of code that handles the case where the local user and remote user both made changes to the record instead of falling into the block of code that handles the scenario I'm trying to verify.
For further clarification, my code is essentially the same as that in the "ResolveMyConflicts" routine in the "ConcurrencyHandler" class in the Learning Resources sample.
Looking at that routine, the first thing that is done is RefetchEntity is called with the MergeStrategy set to "PreserveChangesUpdateOriginal". If you read the documentation on this MergeStrategy it states:
"The refetch replaces the cached entity's original version with the values from the current data source entity but it preserves the cached entity's current version values thus retaining the pending changes"
I can see where that makes sense for the case where two users make changes to the same record but how would this work in the case where the record doesn't exist in the data source because it was deleted?
What happens to the original version values of the cached entity? Is there a flag I should be looking at that tells me the record in the data source was not found and so the original version values of the cached entity were not replaced from the data source so that I can then move into the correct block of code to handle this particular type of concurrency conflict?
I tried switching to the "PreserveChangesUnlessOriginalObsolete" merge strategy but that takes me into the block of code that handles the case where the local user tries to delete a record that was modified by another user.
Could someone explain to me how this scenario works and where I am going wrong and the Learning Resource sample seems to be going wrong as well?
|
Posted By: GregD
Date Posted: 11-Jan-2010 at 11:25am
I can't swear that I tested (or even addressed) this particular thing in the current WinForm example, but at least back in pre-EntityFramework days (DevForce Classic), the Original values would get a bunch of either nulls or default values (can't remember which) when replaced based on a database record that no longer exists. You could use that to identify the situation you're targetting.
What are you seeing for Original values for your deleted record after the replacement?
|
Posted By: cjohnson84
Date Posted: 11-Jan-2010 at 11:44am
Here is my code for the refetch
_entityManager.RefetchEntitiesAsync(_entityKeyList, MergeStrategy.PreserveChangesUpdateOriginal, RefetchCallback, null);
Here is what I see in my "RefetchCallback" method for the entity in question. Note my entity in question is a "Client" comprised of ID, FirstName, LastName, TimeStamp (this is my concurrency column on the table):
_currentValues
[0] 19
[1] Chad
[2] Johnson CCJ
_originalValuesBackup
[0] 19
[1] Chad
[2] Johnson
_proposedValuesBackup
[0] 19
[1] Chad
[2] Johnson
CurrentValues
[0] 19
[1] Chad
[2] Johnson CCJ
Nothing was nulled out from what I can tell.
|
Posted By: GregD
Date Posted: 11-Jan-2010 at 4:42pm
Well, it may be that Entity Framework handles this situation differently. All I can really tell you is to comb over the conflicted entity (and drill into the exception that gets thrown) to see if you can identify anything in either that gets set to a value that enables you to identify the conflict uniquely as resulting from an attempt update a remotely deleted entity.
You're looking for anything in the state of the regurgitated entity that is characteristic of that situation only. (That's what I did when I wrote the original version.)
|
|