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