2009-11-04 4 views
9

Existe-t-il un moyen de déterminer par programme si un paramètre de procédure stockée SQL Server a une valeur par défaut? (Points bonus si vous pouvez déterminer ce que le défaut est.) SqlCommandBuilder.DeriveParameters() n'essaie même pas.Comment puis-je déterminer si un paramètre de procédure stockée SQL Server a un paramètre par défaut?

Merci d'avance pour votre aide!

EDIT: Honnêtement, je ne me soucie pas si c'est une requête SQL, un objet SMO, etc.

Répondre

14

J'ai trouvé un moyen en utilisant SMO:

Server srv; 
srv = new Server("ServerName"); 

Database db; 
db = srv.Databases["MyDatabase"]; 

var Params = db.StoredProcedures["MyStoredProc"].Parameters; 

foreach(StoredProcedureParameter param in Params) { 
    Console.WriteLine(param.Name + "-" + param.DefaultValue); 
} 
+0

+1 excellent! Bon à savoir –

+0

Genius, merci :) –

1

Exécutez la procédure stockée sp_help builtin?

+2

Où dans le resultset pour sp_help-t-il indiquer si un paramètre a une défaut? – GuyBehindtheGuy

+1

liste le paramètre - mais pas si oui ou non ils ont une valeur par défaut, et ce que ce serait :-( –

6

Pas un gros problème dans SQL Server 2005 et plus:

SELECT 
    pa.NAME, 
    t.name 'Type', 
    pa.max_length, 
    pa.has_default_value, 
    pa.default_value 
FROM 
    sys.parameters pa 
INNER JOIN 
    sys.procedures pr ON pa.object_id = pr.object_id 
INNER JOIN 
    sys.types t ON pa.system_type_id = t.system_type_id 
WHERE 
     pr.Name = 'YourStoredProcName' 

Malheureusement, même si cela semblait être un morceau de gâteau - il ne fonctionne pas :-(

De Technet:

SQL Server uniquement maintient les valeurs par défaut valeurs pour les objets CLR dans ce catalogue view; par conséquent, cette colonne a une valeur de 0 pour les objets Transact-SQL. Pour afficher la valeur par défaut d'un paramètre dans un objet Transact-SQL, interrogez la colonne de définition de la vue de catalogue sys.sql_modules ou utilisez la fonction système OBJECT_DEFINITION.

Donc, tout ce que vous pouvez faire est soit requête sys.sql_modules ou appelez SELECT object_definition(object_id) pour obtenir essentiellement la définition SQL (le code source T-SQL) pour votre proc stocké et alors vous aurez besoin d'analyser que (succions !! big time .....)

On dirait qu'il n'y a vraiment pas d'autre façon de le faire ... Je suis étonné et appaled .....

Peut-être que dans SQL Server 2008 R2? :-) Marc

+0

Hmm ... avez-vous réellement exécuté ce? Sur SQL 2008 avec la compatibilité db mis à "100", 'has_default_value 0 est 0 pour chaque ligne, même si le paramètre a définitivement un défaut – GuyBehindtheGuy

+0

Oui, je l'ai couru - malheureusement, tous mes sprocs n'ont jamais de valeurs par défaut, donc je ne pouvais pas vraiment vérifier - laissez-moi vérifier ... –

+3

Wow. méthode recommandée pour le faire est de _parse le corps de la procédure stockée_?! That sucks – GuyBehindtheGuy

1

Pour les procédures stockées, je crois que vous auriez à écrire quelque chose qui parse T-SQL, ou use the T-SQL parser que Microsoft fournit. L'analyseur et le générateur de script vivent dans deux assemblages. Le Microsoft.Data.Schema.ScriptDom contient des classes fournisseur agnostique et l'assemblée Microsoft.Data.Schema.ScriptDom.Sql contiennent des classes pour le générateur d'analyseur et script qui sont spécifiques SQL Server.Comment utiliser spécifiquement ceci pour identifier les paramètres et s'ils sont par défaut n'est pas couvert et serait quelque chose que vous auriez à travailler dessus (probablement avec beaucoup d'efforts) en utilisant l'exemple de code.

+0

Idée intéressante. J'aurais besoin d'une licence pour VSTS Database Edition, cependant. :-) – GuyBehindtheGuy

1

Ce genre est un hack, mais vous pouvez toujours juste donner des paramètres facultatifs un nom spécial comme:

@AgeOptional = 15

... puis d'écrire une méthode simple qui vérifie un paramètre pour voir si c'est optionnel Pas idéal, mais étant donné la situation, cela pourrait être une solution décente.

1

Voici ce que j'ai fait pour l'obtenir. saisir la section de la procédure stockée en commençant par le premier paramètre jusqu'à l'instruction AS. Créer une procédure stockée temporaire avec les instructions declare et renvoyer union tous les ID de paramètre, les noms, les types de colonne, s'ils ont par défaut, et leur valeur. Et puis exécuté la procédure stockée avec l'hypothèse de s'il y a un signe égal entre les paramètres qu'ils ont par défaut, et s'ils n'ont pas par défaut j'ai passé null au paramètre pendant l'exécution, et soit lire le resultset ou si existe la procédure stockée remplie une table temporaire pour que je puisse l'interroger plus tard. J'ai vérifié s'il y a des signes égaux entre les paramètres et si oui, j'ai supposé qu'ils avaient des défauts. S'il y a un commentaire etc avec signe égal la procédure qui signifie qu'ils n'ont pas par défaut et pendant l'exécution je n'ai passé aucun paramètre, l'exécution a échoué, j'ai attrapé le message d'erreur, lu le nom du paramètre et exécuté la procédure null au paramètre. Dans la procédure j'ai utilisé une fonction de concat de chaîne CLR, pour cette raison, elle ne sera pas compilée si vous exécutez directement, mais vous pouvez probablement remplacer par un chemin XML ou par mail, ou par mail je peux vous guider à travers le clr si vous voulez . Depuis que je l'ai fait syndicat tous les paramètres que je les casté comme varchar (max)

USE Util 
GO 
CREATE AGGREGATE [dbo].[StringConcat] 
(@Value nvarchar(MAX), @Delimiter nvarchar(100)) 
RETURNS nvarchar(MAX) 
EXTERNAL NAME [UtilClr].[UtilClr.Concat] 
GO 
CREATE FUNCTION dbo.GetColumnType (@TypeName SYSNAME, 
            @MaxLength SMALLINT, 
            @Precision TINYINT, 
            @Scale TINYINT, 
            @Collation SYSNAME, 
            @DBCollation SYSNAME) 
RETURNS TABLE 
    AS 
RETURN 
    SELECT CAST(CASE WHEN @TypeName IN ('char', 'varchar') 
         THEN @TypeName + '(' + CASE WHEN @MaxLength = -1 THEN 'MAX' 
                ELSE CAST(@MaxLength AS VARCHAR) 
              END + ')' + CASE WHEN @Collation <> @DBCollation THEN ' COLLATE ' + @Collation 
                   ELSE '' 
                 END 
         WHEN @TypeName IN ('nchar', 'nvarchar') 
         THEN @TypeName + '(' + CASE WHEN @MaxLength = -1 THEN 'MAX' 
                ELSE CAST(@MaxLength/2 AS VARCHAR) 
              END + ')' + CASE WHEN @Collation <> @DBCollation THEN ' COLLATE ' + @Collation 
                   ELSE '' 
                 END 
         WHEN @TypeName IN ('binary', 'varbinary') THEN @TypeName + '(' + CASE WHEN @MaxLength = -1 THEN 'MAX' 
                          ELSE CAST(@MaxLength AS VARCHAR) 
                         END + ')' 
         WHEN @TypeName IN ('bigint', 'int', 'smallint', 'tinyint') THEN @TypeName 
         WHEN @TypeName IN ('datetime2', 'time', 'datetimeoffset') THEN @TypeName + '(' + CAST (@Scale AS VARCHAR) + ')' 
         WHEN @TypeName IN ('numeric', 'decimal') THEN @TypeName + '(' + CAST(@Precision AS VARCHAR) + ', ' + CAST(@Scale AS VARCHAR) + ')' 
         ELSE @TypeName 
       END AS VARCHAR(256)) AS ColumnType 
GO 
go 
USE [master] 
GO 
IF OBJECT_ID('dbo.sp_ParamDefault') IS NULL 
    EXEC('CREATE PROCEDURE dbo.sp_ParamDefault AS SELECT 1 AS ID') 
GO 
SET ANSI_NULLS ON 
GO 
SET QUOTED_IDENTIFIER ON 
GO 
ALTER PROCEDURE dbo.sp_ParamDefault 
    @ProcName SYSNAME = NULL OUTPUT 
AS 
SET NOCOUNT ON 
SET ANSI_WARNINGS OFF 
SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED 

DECLARE @SQL VARCHAR(MAX), 
    @ObjectId INT = OBJECT_ID(LTRIM(RTRIM(@ProcName))), 
    @FirstParam VARCHAR(256), 
    @LastParam VARCHAR(256), 
    @SelValues VARCHAR(MAX), 
    @ExecString VARCHAR(MAX), 
    @WhiteSpace VARCHAR(10) = '[' + CHAR(10) + CHAR(13) + CHAR(9) + CHAR(32) + ']', 
    @TableExists BIT = ABS(SIGN(ISNULL(OBJECT_ID('tempdb..#sp_ParamDefault'), 0))), 
    @DeclareSQL VARCHAR(MAX), 
    @ErrorId INT, 
    @ErrorStr VARCHAR(MAX) 

IF @ObjectId IS NULL 
    BEGIN 
     SET @ProcName = NULL 
     PRINT '/* -- SILENCE OPERATION -- 
IF OBJECT_ID(''tempdb..#sp_ParamDefault'') IS NOT NULL DROP TABLE #sp_ParamDefault 
CREATE TABLE #sp_ParamDefault (Id INT, NAME VARCHAR(256), TYPE VARCHAR(256), HasDefault BIT, IsOutput BIT, VALUE VARCHAR(MAX)) 
*/ 

EXEC dbo.sp_ParamDefault 
    @ProcName = NULL 
' 
RETURN 
    END 

SELECT @SQL = definition, 
     @ProcName = QUOTENAME(OBJECT_SCHEMA_NAME(@ObjectId)) + '.' + QUOTENAME(OBJECT_NAME(@ObjectId)), 
     @FirstParam = FirstParam, 
     @LastParam = LastParam 
FROM sys.all_sql_modules m (NOLOCK) 
CROSS APPLY (SELECT MAX(CASE WHEN p.parameter_id = 1 THEN p.name 
         END) AS FirstParam, 
        Util.dbo.StringConcat(p.name, '%') AS Params 
      FROM sys.parameters p (NOLOCK) 
      WHERE p.object_id = m.OBJECT_ID) p 
CROSS APPLY (SELECT TOP 1 
        p.NAME AS LastParam 
      FROM sys.parameters p (NOLOCK) 
      WHERE p.object_id = m.OBJECT_ID 
      ORDER BY parameter_id DESC) l 
WHERE m.object_id = @ObjectId 
IF @FirstParam IS NULL 
    BEGIN 
     IF @TableExists = 0 
      SELECT CAST(NULL AS INT) AS Id, 
        CAST(NULL AS VARCHAR(256)) AS Name, 
        CAST(NULL AS VARCHAR(256)) AS Type, 
        CAST(NULL AS BIT) AS HasDefault, 
        CAST(NULL AS VARCHAR(MAX)) AS VALUE 
      WHERE 1 = 2 
     RETURN 
    END 

SELECT @DeclareSQL = SUBSTRING(@SQL, 1, lst + AsFnd + 2) + ' 
' 
FROM (SELECT PATINDEX ('%' + @WhiteSpace + @LastParam + @WhiteSpace + '%', @SQL) AS Lst) l 
CROSS APPLY (SELECT SUBSTRING (@SQL, lst, LEN (@SQL)) AS SQL2) s2 
CROSS APPLY (SELECT PATINDEX ('%' + @WhiteSpace + 'AS' + @WhiteSpace + '%', SQL2) AS AsFnd) af 


DECLARE @ParamTable TABLE (Id INT NOT NULL, 
          NAME SYSNAME NULL, 
          TYPE VARCHAR(256) NULL, 
          HasDefault BIGINT NULL, 
          IsOutput BIT NOT NULL, 
          TypeName SYSNAME NOT NULL) ; 
WITH pr 
      AS (SELECT p.NAME COLLATE SQL_Latin1_General_CP1_CI_AS AS ParameterName, 
         p.Parameter_id, 
         t.NAME COLLATE SQL_Latin1_General_CP1_CI_AS AS TypeName, 
         ct.ColumnType, 
         MAX(Parameter_id) OVER (PARTITION BY (SELECT 0)) AS MaxParam, 
         p.is_output 
       FROM  sys.parameters p (NOLOCK) 
       INNER JOIN sys.types t (NOLOCK) ON t.user_type_id = p.user_type_id 
       INNER JOIN sys.databases AS db (NOLOCK) ON db.database_id = DB_ID() 
       CROSS APPLY Util.dbo.GetColumnType(t.name, p.max_length, p.precision, p.scale, db.collation_name, db.collation_name) ct 
       WHERE  OBJECT_ID = @ObjectId) 
    INSERT @ParamTable 
      (Id, 
      NAME, 
      TYPE, 
      HasDefault, 
      IsOutput, 
      TypeName) 
      SELECT Parameter_id AS Id, 
        ParameterName AS NAME, 
        ColumnType AS TYPE, 
        HasDefault, 
        is_output AS IsOutput, 
        TypeName 
      FROM pr a 
      CROSS APPLY (SELECT ISNULL('%' + (SELECT Util.dbo.StringConcat (ParameterName, '%') FROM pr b WHERE b.parameter_id < a.parameter_id), '') + '%' 
           + ParameterName + '%=' + '%' + CASE WHEN parameter_id = MaxParam THEN @WhiteSpace + 'AS' + @WhiteSpace + '%' 
                    ELSE (SELECT Util.dbo.StringConcat (ParameterName, '%') FROM pr b 
                        WHERE b.parameter_id > a.parameter_id) + '%' 
                   END AS ptt) b 
      CROSS APPLY (SELECT SIGN (PATINDEX (ptt, @DeclareSQL)) AS HasDefault) hd 

AGAIN: 
SELECT @SelValues = CASE WHEN @TableExists = 1 THEN 'INSERT #sp_ParamDefault(Id, Name, Type, HasDefault, IsOutput, Value) 
'       ELSE '' 
        END + 'SELECT * FROM (VALUES' + Util.dbo.StringConcat('(' + CAST(Id AS VARCHAR) + ', ''' + Name + ''', ''' + Type + ''', ' 
                      + CAST(HasDefault AS VARCHAR) + ', ' + CAST(IsOutput AS VARCHAR) + ', ' 
                      + CASE WHEN TypeName NOT LIKE '%char%' THEN 'CAST(' + name + ' AS VARCHAR(MAX))' 
                        ELSE name 
                      END + ')', ', 
') + ' 
) d(Id, Name, Type, HasDefault, IsOutput, Value)', 
     @ExecString = 'EXEC#sp_ParamDefaultProc 
' + ISNULL(Util.dbo.StringConcat(CASE WHEN HasDefault = 0 THEN Name + ' = NULL' 
           END, ', 
'), '') 
FROM @ParamTable 

SET @SQL = 'CREATE PROCEDURE #sp_ParamDefaultProc 
' + SUBSTRING(@DeclareSQL, CHARINDEX(@FirstParam, @DeclareSQL), LEN(@DeclareSQL)) + ' 
' + @SelValues 

IF OBJECT_ID('TEMPDB..#sp_ParamDefaultProc') IS NOT NULL 
    DROP PROCEDURE #sp_ParamDefaultProc 
EXEC(@SQL) 

BEGIN TRY 
    EXEC(@ExecString) 
END TRY 
BEGIN CATCH 
    SELECT @ErrorStr = ERROR_MESSAGE(), 
      @ErrorId = ERROR_NUMBER() 
-- there must have been a comment containing equal sign between parameters 
    UPDATE p 
    SET  HasDefault = 0 
    FROM (SELECT PATINDEX ('%expects parameter ''@%', @ErrorStr) AS ii) i 
    CROSS APPLY (SELECT CHARINDEX ('''', @ErrorStr, ii + 20) AS uu) u 
    INNER JOIN @ParamTable p ON p.name = SUBSTRING(@ErrorStr, ii + 19, uu - ii - 19) 
    WHERE ii > 0 

    IF @@ROWCOUNT > 0 
     GOTO AGAIN 

    RAISERROR(@ErrorStr, 16, 1) 
    RETURN 30 
END CATCH 
GO 
EXEC sys.sp_MS_marksystemobject 
    sp_ParamDefault 
GO 
1

Ceci est la réponse de SMO dans PowerShell:

[System.Reflection.Assembly]::LoadWithPartialName("Microsoft.SqlServer.Smo") | out-null 

$srv = New-Object "Microsoft.SqlServer.Management.Smo.Server" "MyServer\MyInstance" 
$db = $srv.Databases["MyDatabase"]; 
$proc = $db.StoredProcedures["MyStoredProcedure"] 

foreach($parameter in $proc.Parameters) { 
    if ($parameter.DefaultValue){ 
    Write-Host "$proc , $parameter , $($parameter.DefaultValue)" 
    } 
    else{ 
    Write-Host "$proc , $parameter , No Default Value" 
    } 
} 
Questions connexes