10 03 2011
SharePoint 2010 Performance with Item Level Permissions
Tonight’s little project was inspired by a conversation on Twitter with @WonderLaura and @Toddklindt. Laura had asked about the implications of lots of item-level permissions in a big list in SharePoint (here), to which Todd replied with this, and followed up with this and this.
So this got me thinking: where is the performance bottleneck when you have item-level permissions in place? Todd’s comments leant towards the construction of the view at the Web Part level, but something was making me think that the checking of permissions would be done at the database-query level. Surely an SPRequest wouldn’t poll the database for all items in a request if not all of them are viewable by the user starting the request?
Time, I thought, to do an experiment. I want to use the SharePoint Foundation 2010 Developer Dashboard to monitor the performance of a large (20,000 item) list before and after item-level permissions have been applied – to help us see if we can spot the bottleneck.
Using SharePoint Foundation 2010, I created a custom list and left it with just the one column for this experiment. I used Visual Studio 2010 (with the CKS:Dev pack) to populate the list with 20,000 items using a Console Application. (Sources for this project can be found at the end of this post).
I then fired up the developer dashboard to see what the performance looks like. I created a normal View in the GUI which returned 200 items from the list. Here’s the results (click for larger version):
The figures I’m interested in looking at are the “DECLARE @DocParentIdForRF” Database Query and the PerfTest OnPreRender numbers. The latter is the time taking to execute the OnPreRender method of the List (which, in ListViewWebPart, prepares the Ribbon for use, and prepares client side ScriptLinks for DataFormWebPart). I have no idea what the first query is all about, it’s not a stored procedure nor a function in the content database and it’s definitely not constructed by any of the managed code of SharePoint. If anyone does know what this does, please let me know purely to satisfy my curiosity.
Next, I rolled up another console application to randomly apply one of the Role Definitions available in the site to every single item in the list, giving us a kind of “worse-case” scenario to look. Again, source code for this is at the end of this post.
Back to the list view, and here’re the results from the Developer Dashboard. Notice huge jump in both the figures – the database query takes around 2 seconds as well as the OnPreRender for the web part! This was after refreshing the page a total of five times once every 10 seconds to let caching, etc. do its thing.
So, for some reason OnPreRender takes an absolute age with a massive list, even though the web part is rendering the same 200 items in the first page of the view, except now we have item level permissions for each one of those items. This method, however, doesn’t deal with the items! It just registers ScriptLinks used by the DataFormWebPart.
As the database query takes less time in both before and after, this leads me to think that the query is slowing down the OnPreRender for some reason, thus leading me to suspect further that performance loss is indeed down to the querying of the database!
If anyone else has any thoughts on this, they’d be gratefully received.
And now for the source.
This first piece of code will create your 2000 items.
1 |
using System;<br>using System.Linq;<br>using Microsoft.SharePoint;<br><br>namespace ItemPerfTest<br>{<br> class Program<br> {<br> static void Main(string[] args)<br> {<br> Console.Out.Write("Preparing List...");<br> using (SPSite site = new SPSite("http://demolab-sps2010"))<br> using (SPWeb web = site.OpenWeb())<br> {<br> SPList list = web.Lists["PerfTest"];<br> for (int i = 0; i < 20001; i++)<br> {<br> Console.Out.WriteLine(string.Format("Adding item: {0}...", i));<br> SPListItem item = list.Items.Add();<br> item["Title"] = "Item" + i;<br> item.Update();<br> }<br> }<br> }<br> }<br>}<br><br> |
This second piece will assign each one a randomly selected Role Definition, after breaking the items Role Inheritance.
1 |
using System;<br>using System.Linq;<br>using Microsoft.SharePoint;<br>using System.Collections.Generic;<br><br>namespace UniquePermissions<br>{<br> class Program<br> {<br> static void Main(string[] args)<br> {<br> List<SPRoleDefinition> roles = new List<SPRoleDefinition>();<br> Random numbergenerator = new Random();<br> using (SPSite site = new SPSite("http://demolab-sps2010"))<br> using (SPWeb web = site.OpenWeb())<br> {<br> Console.Out.WriteLine("Preparing list...");<br> SPRoleDefinition contributor = web.RoleDefinitions.GetByType(SPRoleType.Contributor);<br> SPRoleDefinition guest = web.RoleDefinitions.GetByType(SPRoleType.Guest);<br> roles.Add(contributor);<br> roles.Add(guest);<br><br><br> SPList list = web.Lists["PerfTest"];<br><br> foreach(SPListItem item in list.Items)<br> {<br> item.BreakRoleInheritance(true);<br> SPRoleAssignment randomRole = web.RoleAssignments[numbergenerator.Next(0, web.RoleAssignments.Count)];<br> item.RoleAssignments.Add(randomRole);<br> item.Update();<br> Console.Out.WriteLine("Applying role assment {0} to item {1}....", randomRole.Member.LoginName, item.Title);<br> }<br><br> }<br><br> Console.WriteLine();<br> }<br> }<br>}<br> |
1 |
PS: Here’s the detail on that query as called:
1 |
SqlCommand: 'DECLARE @DocParentIdForRF uniqueidentifier SELECT TOP 1 @DocParentIdForRF = Docs.Id FROM Docs WHERE Docs.SiteId = @SITEID AND Docs.DirName = @FDN AND Docs.LeafName = @FLN; SET NOCOUNT ON; DECLARE @_scopeTbl TABLE(_listId uniqueidentifier NOT NULL,_id uniqueidentifier NOT NULL,_level bit NOT NULL,PRIMARY KEY CLUSTERED(_listId,_id,_level)); DECLARE @_count0_0 INT; SET @_count0_0 = 0; WHILE @_count0_0<@I9P BEGIN INSERT INTO @_scopeTbl VALUES (@L2, CAST(SUBSTRING(@L10IMG,@_count0_0*16 + 1, 16) AS uniqueidentifier),0); SET @_count0_0=@_count0_0 + 1; END;SELECT TOP(@NUMROWS) t1.[TimeCreated] AS c0, UserData.[tp_Modified], t1.[SortBehavior] AS c2, UserData.[tp_ContentTypeId], UserData.[tp_ID], t1.[ScopeId] AS c4, t1.[Type] AS c1, UserData.[tp_ModerationStatus], UserData.[tp_Level], UserData.[nvarchar1], CASE WHEN DATALENGTH(t1.DirName) = 0 THEN t1.LeafName WHEN DATALENGTH(t1.LeafName) = 0 THEN t1.DirName ELSE t1.DirName + N'/' + t1.LeafName END AS c3, UserData.[tp_HasAttachment], t1.[LeafName] AS c5, UserData.[nvarchar2] FROM AllUserData AS UserData WITH(INDEX=AllUserData_PK) LEFT OUTER LOOP JOIN Docs AS t1 WITH(NOLOCK) ON (UserData.[tp_CalculatedVersion] = 0 ) AND (UserData.[tp_IsCurrentVersion] = CONVERT(bit,1) ) AND (UserData.[tp_DeleteTransactionId] = 0x ) AND (UserData.[tp_RowOrdinal] = 0) AND (t1.SiteId=UserData.tp_SiteId) AND (t1.SiteId = @SITEID) AND (t1.ParentId = UserData.tp_ParentId) AND (t1.Id = UserData.tp_DocId) AND ( (UserData.tp_Level = 1) ) AND (t1.Level = UserData.tp_Level) AND (t1.IsCurrentVersion = 1) AND (t1.Level = 1 OR t1.Level = 2) WHERE ( (UserData.tp_Level = 1) ) AND (UserData.tp_SiteId=@SITEID AND (UserData.tp_ParentId=@DocParentIdForRF)) AND (UserData.tp_RowOrdinal=0) AND (t1.SiteId=@SITEID AND (t1.ParentId=@DocParentIdForRF)) AND (UserData.tp_ListID=@LISTID) AND ( (UserData.tp_Level = 1) ) AND (UserData.tp_SiteId=@SITEID AND (UserData.tp_ParentId=@DocParentIdForRF)) AND (UserData.tp_RowOrdinal=0) AND (t1.SiteId=@SITEID AND (t1.ParentId=@DocParentIdForRF)) ORDER BY UserData.[tp_ID] ASC OPTION (FORCE ORDER, MAXDOP 1)'<br> CommandType: Text CommandTimeout: 0<br> Parameter: '@LFFP' Type: UniqueIdentifier Size: 0 Direction: Input Value: '00000000-0000-0000-0000-000000000000'<br> Parameter: '@SITEID' Type: UniqueIdentifier Size: 0 Direction: Input Value: '02ea4e54-c1ae-4b8c-92fc-45f0c91af393'<br> Parameter: '@L2' Type: UniqueIdentifier Size: 0 Direction: Input Value: '4047d6c1-d522-4db6-91eb-ab257dabbac5'<br> Parameter: '@FDN' Type: NVarChar Size: 4000 Direction: Input Value: 'Lists'<br> Parameter: '@FLN' Type: NVarChar Size: 4000 Direction: Input Value: 'PerfTest'<br> Parameter: '@RequestGuid' Type: UniqueIdentifier Size: 0 Direction: Input Value: '11bf00f2-9479-4ef7-a8c2-9c3b005f7b8e'<br> Parameter: '@L6' Type: UniqueIdentifier Size: 0 Direction: Input Value: '17e4e245-cb54-4b57-bc17-ab4215ac66f2'<br> Parameter: '@LISTID' Type: UniqueIdentifier Size: 0 Direction: Input Value: '4047d6c1-d522-4db6-91eb-ab257dabbac5'<br> Parameter: '@NUMROWS' Type: BigInt Size: 0 Direction: Input Value: '11'<br> Parameter: '@I9P' Type: Int Size: 0 Direction: Input Value: '20002'<br> Parameter: '@L10IMG' Type: VarBinary Size: -1 Direction: Input<br> |
Text-only Description field in Google News RSS Feeds in XSLT SharePoint 2010 Performance with Item Level Permissions Part 2
You should also monitor the time it takes (espec on db) to set the unique permissions. A combination of BreakRoleInheritance(true) and lots of different permissions to different users should also take an exponentially growing load on the db due to the way SharePoint sets limited access from list item and up ( to allow users access to navigate to item)
@TGITM – good point and very true. This example would’ve put a horrendous load on the database for the reasons you give. My code examples are obviously a single-run setting up of the environment, and even the coding style I used (I foreach’d over SPList.Items!!) would never be done in production for other performance reasons. This gives me an idea for another blog post for another day actually 😉
You’re looking at the wrong place. It’s not OnPreRender that stalls this. The times on the right hand side is just the offset. Take a look on the left hand side and you’ll see that its doing databinding which takes time. The databinding occurs before the onprerender, hence the offset. And it’s the databinding that takes time in this case.
Maybe I’m just not following your randomizer code, but what does the first part do? I see you add Guest and Contributor roles to a list of roledefinitions to what? It doesn’t get assigned does it?
And the roleassignments are random in the parent web? How many role assignments were there? Did you intend to add these roles to the web for the listitem to make use of.
@Brian: yeah the first half just gets me a List collection of 2 roles, and I pick a random one from that list when I assign it to each item in turn. I’ve actually just noticed that the code I pasted into the post is an old version, before I realised I wasn’t doing SPListItem.Updated()! I’ll add this now.
Great post! now im thinking about this..