Loading…

SharePoint Event Receiver ItemUpdated method fires twice

I ran into an issue with an Event Receiver that was deployed to a Document Center Site library.  The requirement is pretty simple, when a document is added to the Documents library or an existing document is updated, create a task for someone to review the document’s metadata.

Problem

The problem is that the ItemUpdated event in this custom Event Receiver will fire twice under certain circumstances–such as having require document checkout set on the Documents library.  In my case, users do not load documents directly to the site.  Documents are only added or updated in this library from the Content Organizer via a custom Event Receiver/OOTB Approval Workflow combo on multiple Document Center Workspaces–which exist in their own Site Collections.

I’ve done my due diligence and followed recommendations here and here.  I made sure that the event receiver was not registered twice using the SharePoint 2010 Manager tool.  I made sure to DisableEventFiring() before creating the task item.  Regardless of what I tried, I really couldn’t track down exactly why it was getting fired twice.

Solution

To work around this issue, I implemented the following code:

SPListItem item = properties.ListItem;

if (item != null)

{

SPPropertyBag currentBag = item.Web.Properties;

string docID = item[“_dlc_DocId”].ToString();

if (currentBag.ContainsKey(docID) && currentBag[docID] != null)

{

currentBag[docID] = null;

currentBag.Update();

}

else

{

EventFiringEnabled = false;

CreateTask(properties);

currentBag.Add(docID, “added”);

currentBag.Update();

this.EventFiringEnabled = true;

}

The code looks for the key (the Doc ID) in the property bag and if it is present, skip the creation of the task and remove the property from the bag.  The document’s Doc ID is being used as the key–it will be unique.  It really doesn’t matter what the value of the property is–it just matters that it is there.  This could have been implemented using Session State, but this is not available in events ending with “ed” in Event Receivers and probably not the best option anyway for several reasons.

It is not enough to call the remove method like so:

currentBag.Remove(docID);

currentBag.Update();

This call will not necessarily remove the item from the bag–in my case it did not, however setting it to null will.  It is important to null out the item–if not, the size of the bag will keep growing.

References

How to Access Account Information with SPWeb.EnsureUser

The SPWeb.EnsureUser method is a handy method made available by the SPWeb object.  This method will check whether the logon name is a member of the SPWeb and if not, it will add it.  For example, this can be used in an Event Receiver that handles an ItemAdded event and needs to ensure that the Approver or the original Author of a document has access to it in the new SPWeb location.  The user object returned is an instance of SPUser and can be used to retrieve information as necessary.

SPUser user = web.EnsureUser(account);

This code could throw an exception if the context running doesn’t have the appropriate permissions.  The best way to implement this method is by using SPSecurity.RunWithElevatedPrivileges.  Here is a good writeup by Dan Parker on how best to implement this.

References

Mac OS X SSD Tweaks

I put a Crucial M4 SSD into my early 2011 Macbook Pro last month; I’ve been very happy so far.

m4_ssd

Crucial M4 Reviews I read before purchasing:

There are a few things that you can do to increase the performance of your computer after putting in the SSD.  For example, the sudden motion sensor (SMS) doesn’t need to be on for a solid state drive, and it does not need to be put to sleep.

This article has some very good tips and explains the need for them pretty well:

http://poller.se/2010/08/optimizing-mac-os-x-for-ssd-drives/

I did the following according to the above article:

  1. I did disable my sleep image.  I gained 16GB of space back (the size of your RAM), and the machine sleeps in a few seconds
  2. Disabled my Hard Drive Sleep
  3. No need for the SMS
  4. I did not enable noatime.
  5. I did not disable Spotlight, but I did restrict the indexing to only my applications.  I really only use it for launching apps with the Command + Spacebar shortcut combo anyway

Additionally I did enable TRIM support for the SSD using this tool and have noticed better reads and writes:

http://www.groths.org/?page_id=322

I have been very happy with the SSD upgrade and the applied tweaks.  Keep in mind though that these changes may have been seamless for me, but that doesn’t mean they will be for you.  Make sure to do your own research, exercise caution, and then execute at your own risk.  If your computer blows up, fails, or turns into a PC–I will not be held responsible.

Discovering, Activating, and Deactivating Features In A SharePoint Site Collection Using PowerShell

Recently I had to deactivate/activate a hidden feature that was deployed to a SharePoint 2010 Site Collection.   It was deployed as hidden to prevent a critical capability from being turned off accidentally.  Hidden features do not show up on the Manage Site Features page, but they can be displayed using PowerShell.

I needed to perform the following:

  • List the active features
  • Deactivate/Activate the feature

List The Active Features

Getting the Feature list is pretty simple.  Run the following PowerShell Command from the SharePoint 2010 Management Shell (Replace http://SharePointSite/ with the site you would like to use):

Get-SPFeature -Site http://SharePointSite/

A list similar to the following will be returned displaying both the feature name and corresponding ID:

Active Features.img_assist_custom-549x89

If you are not sure what Web the feature is activated in, run the following:

Get-SPSite http://SharePointSite/ | Get-SPWeb -Limit ALL | %{ Get-SPFeature -Web $_ } | Sort DisplayName -Unique | FT DisplayName,Id

This will return a distinct list of ALL features that are active in the Site Collection.

Deactivating the Feature

A feature can be disabled by name or ID.  In my case, I chose to deactivate and activate the feature by ID rather than name.  The name of the feature was too long and did not fully display in the results from above.

Disable-SPFeature –Identity FeatureNameHere –url http://SharePointSite/

Or

Disable-SPFeature –Identity FeatureIDHere  –url http://SharePointSite/

Activating the Feature

To Activate the feature, do the same as above, but use Enable-SPFeature instead of Disable-SPFeature.

Enable-SPFeature –Identity FeatureNameHere –url http://SharePointSite/

Or

Enable-SPFeature –Identity FeatureIDHere  –url http://SharePointSite/

References