hoping someone might be able to help me with a bit of an issue. Essentially i'm trying to get a rough size of all of the fields in my database as i'd like to do some math on it to guesstimate what the size will be with a compression technique applied to it.
I can do this for most fields by looking at the datatype and using the number of rows to get the number of bytes it's taking up. However on something like a varchar(max) field this is not as easy and so i decided to approach this by getting the average length within the column and multiplying by number of rows. But i've hit a snag which i'll describe below.
I have the following stored proc (i tried a function too but you can't call dynamic SQL from a function).
CREATE PROCEDURE dbo.getFieldSize(@column varchar(255), @table varchar(255), @ret decimal(15,7) OUTPUT)
AS
BEGIN
DECLARE @lengthSQL varchar(50)
/SET @lengthSQL = 'SELECT @ret = AVG(DATALENGTH(' + @column + ')) FROM [' + @table +']'/
SET @lengthSQL = 'SELECT @ret = AVG(DATALENGTH(' + @column + ')) FROM ' + @table
exec sp_executesql @lengthSQL
RETURN @ret
END
GO
And then i call it using...
SELECT b.TABLE_SCHEMA as 'Schema',
CASE WHEN DATA_TYPE IN ('nvarchar') AND CHARACTER_MAXIMUM_LENGTH <> -1 AND c.distinctItems <> 0 AND c.totalCount <> 0 THEN exec('select max(len(' + b.TABLE_CATALOG + ' + '.' + ' + b.COLUMN_NAME + '))')
FROM ....
The above is basically just checking to make sure it is a varchar(max) field and contains some values within the column. I then try and execute the SP and pass the column name and table name for which i need the avg length but i get the following error.
Msg 156, Level 15, State 1, Line 57 Incorrect syntax near the keyword 'exec'.
I learned you cannot call a dynamic SQL from a function and you cannot call a SP from a CASE statement. So at this point it seems like it's a catch 22 and i cannot do what i need using SQL. Can anyone think of any workarounds or i'm I out of luck on this?
Actually, you can do Dynamic SQL in a scalar UDF, it just needs to be a SQLCLR UDF ;-). But this is fairly simple to do using the in-process / internal connection (i.e.
SqlConnection("Context Connection = true;");
). Meaning, the assembly can be set to SAFE.Also, object / column / index names are all NVARCHAR. And objects (if not also the others) are declared as
sysname
which is an alias forNVARCHAR(128)
. Just FYI.So, something like the following (which I have tested and it does work):