I have a multi-tenant database in SQL Server 2012 where each tenant's rows are identified by a tenant_id
column (aka the Shared Database, Shared Schema approach). Some tenants, particularly the newer ones, have very few rows, while others have many.
SQL Server's query optimizer normally builds a query plan based on the parameters provided during its first execution, then re-uses this plan for all future queries even if different parameters are provided. This is known as parameter sniffing.
The problem we have with our database is that SQL Server sometimes builds these plans based on parameters that point to a smaller tenant, which works fine for that tenant, but then when it reapplies the cached plan to a larger tenant it fails catastrophically (usually timing out, in fact). Typically we find out about this situation only when one of our larger tenants contacts us about experiencing time-out errors, then we have to get into the system and manually flush all the query plans to correct it.
There is a query hint you can use to prevent SQL Server from caching query plans (OPTIMIZE FOR UNKNOWN
) but this results in some extra overhead since the query plan is being regenerated every time the query is called. An additional problem is that we're using Entity Framework which offers no ability to specify the OPTIMIZE FOR UNKNOWN
hint on queries.
So the question is -- what is the best practice for multi-tenant databases with regard to parameter sniffing? Is there a way to disable parameter sniffing database-wide without having to specify it on every query? If so, is that even the best approach? Should I be partitioning the data in some other way? Is there some other approach I'm not thinking of?
OPTIMIZE FOR UNKNOWN
does not have the behavior you describe. You are confusing it withOPTION (RECOMPILE)
. If you are on Enterprise Edition you can look at plan guides to provide the hint. Or there is a trace flag (4136) to disable parameter sniffing entirely but that applies to the instance not database. – Laroy