One of the people on our team wanted to have the ability to truncate tables on the staging database while this person was doing some testing.
Here is what Books On Line has about permissions for the TRUNCATED statement
The minimum permission required is ALTER on table_name. TRUNCATE TABLE permissions default to the table owner, members of the sysadmin fixed server role, and the db_owner and db_ddladmin fixed database roles, and are not transferable.
Before I answer why someone would need ALTER TABLE permissions when the person already has DELETE permissions, let’s run some code that will show the ‘problem’.
CREATE DATABASE Test go USE Test GO CREATE TABLE TestTruncate(Id int) GO INSERT TestTruncate values(1) GO
Now create a new user and give the user datareader and datawriter permissions
USE master GO CREATE LOGIN TestLogin WITH PASSWORD=N'Test', DEFAULT_DATABASE=master, CHECK_EXPIRATION=OFF, CHECK_POLICY=OFF GO USE Test GO CREATE USER TestLogin FOR LOGIN TestLogin GO USE Test GO ALTER ROLE db_datareader ADD MEMBER TestLogin GO USE Test GO ALTER ROLE db_datawriter ADD MEMBER TestLogin GO
Now that the user is created, login as that user and run the TRUNCATE TABLE command
TRUNCATE TABLE TestTruncate
Msg 1088, Level 16, State 7, Line 1
Cannot find the object “TestTruncate” because it does not exist or you do not have permissions.
Cannot find the object “TestTruncate” because it does not exist or you do not have permissions.
As you can see, you don’t have permission. Executing a delete will work just fine
DELETE TestTruncate
(1 row(s) affected)
Before I give you a workaround, let’s try to figure out why the minimum requirement is ALTER TABLE.
What is the difference between a DELETE and a TRUNCATE in terms of logging? When a TRUNCATE occurs, the operation does not log individual row deletions, a DELETE operation does. The reason this is important is because if you have a trigger on the table, in needs to be disabled before the TRUNCATE occurs. Now you know why ALTER TABLE is required, triggers need to be disabled.
ALTER TABLE SomeTable DISABLE TRIGGER SomeTrigger
And in order to disable the trigger, ALTER TABLE permissions are required as a minimum.
But I don’t want people altering tables on our staging and QA servers, so here is one way of giving the person the ability to TRUNCATE a table without giving them permissions explicitly. Create a stored procedure and use WITH EXECUTE AS, this will define the execution context of the stored procedure. In the example below, I picked a user that has sufficient privileges to perform the TRUNCATE.
CREATE PROCEDURE prTruncate WITH EXECUTE AS 'SuperUser' AS TRUNCATE TABLE TestTruncate GO
All you have to do is give your user execute permissions to the stored procedure you just created
GRANT EXECUTE ON prTruncate TO TestLogin GO
Now if you execute the stored procedure as the TestLogin user, you will see it will run just fine
EXEC prTruncate