This came up a couple of days ago, a person created a User-Defined Table Type and then used the regular db_datawriter/db_datareader account to use this User-Defined Table Type and he would get the following error
**Msg 229, Level 14, State 5, Line 9
The EXECUTE permission was denied on the object ‘SysObjectsCount’, database ’testTVP’, schema ‘dbo’.**
Let’s take a look at how to give permissions because if you do the following
GRANT EXECUTE ON SysObjectsCount TO TestLogin
You will get this error
**Msg 15151, Level 16, State 1, Line 1
Cannot find the object ‘SysObjectsCount’, because it does not exist or you do not have permission.**
Let’s fix this, I will show you some code so that you can reproduce this
First create the following login
USE [master]
GO
CREATE LOGIN [TestLogin] WITH PASSWORD=N'test', DEFAULT_DATABASE=[master], CHECK_EXPIRATION=OFF, CHECK_POLICY=OFF
GO
Now create the following database
CREATE DATABASE testTVP
GO
Create a new user in the database that we just created and give the user db_datareader and db_datawriter roles
USE testTVP
GO
CREATE USER TestLogin FOR LOGIN TestLogin
GO
EXEC sp_addrolemember N'db_datareader', N'TestLogin'
GO
USE [testTVP]
GO
EXEC sp_addrolemember N'db_datawriter', N'TestLogin'
GO
Now create the following type
CREATE TYPE SysObjectsCount AS TABLE(quantity INT, xtype CHAR(2))
GO
Now run the following piece of code
DECLARE @mySystableCount AS SysObjectsCount
INSERT @mySystableCount
SELECT COUNT(*),xtype FROM sysobjects
GROUP BY xtype
SELECT * FROM @mySystableCount
That runs fine right? Open another connection but this time login as TestLogin
Try to run that code again
DECLARE @mySystableCount AS SysObjectsCount
INSERT @mySystableCount
SELECT COUNT(*),xtype FROM sysobjects
GROUP BY xtype
SELECT * FROM @mySystableCount
You get the following error
**Msg 229, Level 14, State 5, Line 9
The EXECUTE permission was denied on the object ‘SysObjectsCount’, database ’testTVP’, schema ‘dbo’.**
In order to give the permissions to testLogin, you need to execute the following code, yes the :: is correct.
GRANT EXECUTE ON TYPE::SysObjectsCount to TestLogin
That reminds me of how you would use fn_helpcollations()
SELECT * FROM ::fn_helpcollations()
Just keep in mind that you need to have those double colons there. It would be nicer if you could just do something like this instead
GRANT EXECUTE ON TYPE SysObjectsCount to TestLogin
Anyway, execute this
GRANT EXECUTE ON TYPE::SysObjectsCount to TestLogin
Now you should be able to run this code connected as TestLogin
DECLARE @mySystableCount AS SysObjectsCount
INSERT @mySystableCount
SELECT COUNT(*),xtype FROM sysobjects
GROUP BY xtype
SELECT * FROM @mySystableCount
And if you look now in SSMS, you will be able to see the User-Defined Table Type in the User-Defined Table Types folder
That’s it for this post…hopefully this will help someone else in the future
See also my post SQL Advent 2011 Day 18: Table-valued Parameters for some more info about User-Defined Table Types