2018-06-29 19:52:40 +02:00
using System ;
using System.Collections.Generic ;
2018-09-19 18:51:27 +02:00
using System.Data.Common ;
2018-06-29 19:52:40 +02:00
using System.Linq ;
using NPoco ;
using Umbraco.Core.Persistence.DatabaseModelDefinitions ;
using Umbraco.Core.Scoping ;
namespace Umbraco.Core.Persistence.SqlSyntax
{
/// <summary>
/// Represents an SqlSyntaxProvider for Sql Server.
/// </summary>
[SqlSyntaxProvider(Constants.DbProviderNames.SqlServer)]
public class SqlServerSyntaxProvider : MicrosoftSqlSyntaxProviderBase < SqlServerSyntaxProvider >
{
// IUmbracoDatabaseFactory to be lazily injected
public SqlServerSyntaxProvider ( Lazy < IScopeProvider > lazyScopeProvider )
{
_serverVersion = new Lazy < ServerVersionInfo > ( ( ) = >
{
var scopeProvider = lazyScopeProvider . Value ;
if ( scopeProvider = = null )
throw new InvalidOperationException ( "Failed to determine Sql Server version (no scope provider)." ) ;
using ( var scope = scopeProvider . CreateScope ( ) )
{
var version = DetermineVersion ( scope . Database ) ;
scope . Complete ( ) ;
return version ;
}
} ) ;
}
private readonly Lazy < ServerVersionInfo > _serverVersion ;
internal ServerVersionInfo ServerVersion = > _serverVersion . Value ;
internal enum VersionName
{
Invalid = - 1 ,
Unknown = 0 ,
V7 = 1 ,
V2000 = 2 ,
V2005 = 3 ,
V2008 = 4 ,
V2012 = 5 ,
V2014 = 6 ,
V2016 = 7 ,
2018-09-19 18:51:27 +02:00
V2017 = 8 ,
2018-06-29 19:52:40 +02:00
Other = 99
}
internal enum EngineEdition
{
Unknown = 0 ,
Desktop = 1 ,
Standard = 2 ,
Enterprise = 3 ,
Express = 4 ,
Azure = 5
}
internal class ServerVersionInfo
{
public string Edition { get ; set ; }
public string InstanceName { get ; set ; }
public string ProductVersion { get ; set ; }
public VersionName ProductVersionName { get ; private set ; }
public EngineEdition EngineEdition { get ; set ; }
public bool IsAzure = > EngineEdition = = EngineEdition . Azure ;
public string MachineName { get ; set ; }
public string ProductLevel { get ; set ; }
public void Initialize ( )
{
2018-09-19 18:51:27 +02:00
ProductVersionName = MapProductVersion ( ProductVersion ) ;
}
}
private static VersionName MapProductVersion ( string productVersion )
{
var firstPart = string . IsNullOrWhiteSpace ( productVersion ) ? "??" : productVersion . Split ( '.' ) [ 0 ] ;
switch ( firstPart )
{
case "??" :
return VersionName . Invalid ;
case "14" :
return VersionName . V2017 ;
case "13" :
return VersionName . V2016 ;
case "12" :
return VersionName . V2014 ;
case "11" :
return VersionName . V2012 ;
case "10" :
return VersionName . V2008 ;
case "9" :
return VersionName . V2005 ;
case "8" :
return VersionName . V2000 ;
case "7" :
return VersionName . V7 ;
default :
return VersionName . Other ;
2018-06-29 19:52:40 +02:00
}
}
private static ServerVersionInfo DetermineVersion ( IUmbracoDatabase database )
{
// Edition: "Express Edition", "Windows Azure SQL Database..."
// EngineEdition: 1/Desktop 2/Standard 3/Enterprise 4/Express 5/Azure
// ProductLevel: RTM, SPx, CTP...
const string sql = @ "select
SERVERPROPERTY ( ' Edition ' ) Edition ,
SERVERPROPERTY ( ' EditionID ' ) EditionId ,
SERVERPROPERTY ( ' InstanceName ' ) InstanceName ,
SERVERPROPERTY ( ' ProductVersion ' ) ProductVersion ,
SERVERPROPERTY ( ' BuildClrVersion ' ) BuildClrVersion ,
SERVERPROPERTY ( ' EngineEdition ' ) EngineEdition ,
SERVERPROPERTY ( ' IsClustered ' ) IsClustered ,
SERVERPROPERTY ( ' MachineName ' ) MachineName ,
SERVERPROPERTY ( ' ResourceLastUpdateDateTime ' ) ResourceLastUpdateDateTime ,
SERVERPROPERTY ( ' ProductLevel ' ) ProductLevel ; ";
try
{
var version = database . Fetch < ServerVersionInfo > ( sql ) . First ( ) ;
version . Initialize ( ) ;
return version ;
}
catch ( Exception e )
{
// can't ignore, really
throw new Exception ( "Failed to determine Sql Server version (see inner exception)." , e ) ;
}
}
2018-09-19 18:51:27 +02:00
internal static VersionName GetVersionName ( string connectionString , string providerName )
{
var factory = DbProviderFactories . GetFactory ( providerName ) ;
var connection = factory . CreateConnection ( ) ;
if ( connection = = null )
throw new InvalidOperationException ( $"Could not create a connection for provider \" { providerName } \ "." ) ;
connection . ConnectionString = connectionString ;
using ( connection )
{
try
{
connection . Open ( ) ;
var command = connection . CreateCommand ( ) ;
command . CommandText = "SELECT SERVERPROPERTY('ProductVersion');" ;
var productVersion = command . ExecuteScalar ( ) . ToString ( ) ;
connection . Close ( ) ;
return MapProductVersion ( productVersion ) ;
}
catch
{
return VersionName . Unknown ;
}
}
}
2018-06-29 19:52:40 +02:00
/// <summary>
/// SQL Server stores default values assigned to columns as constraints, it also stores them with named values, this is the only
/// server type that does this, therefore this method doesn't exist on any other syntax provider
/// </summary>
/// <returns></returns>
public IEnumerable < Tuple < string , string , string , string > > GetDefaultConstraintsPerColumn ( IDatabase db )
{
2018-08-26 23:13:08 +02:00
var items = db . Fetch < dynamic > ( "SELECT TableName = t.Name, ColumnName = c.Name, dc.Name, dc.[Definition] FROM sys.tables t INNER JOIN sys.default_constraints dc ON t.object_id = dc.parent_object_id INNER JOIN sys.columns c ON dc.parent_object_id = c.object_id AND c.column_id = dc.parent_column_id INNER JOIN sys.schemas as s on t.[schema_id] = s.[schema_id] WHERE s.name = (SELECT SCHEMA_NAME())" ) ;
2018-06-29 19:52:40 +02:00
return items . Select ( x = > new Tuple < string , string , string , string > ( x . TableName , x . ColumnName , x . Name , x . Definition ) ) ;
}
public override IEnumerable < string > GetTablesInSchema ( IDatabase db )
{
2018-08-26 23:13:08 +02:00
var items = db . Fetch < dynamic > ( "SELECT TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = (SELECT SCHEMA_NAME())" ) ;
2018-06-29 19:52:40 +02:00
return items . Select ( x = > x . TABLE_NAME ) . Cast < string > ( ) . ToList ( ) ;
}
public override IEnumerable < ColumnInfo > GetColumnsInSchema ( IDatabase db )
{
2018-08-26 23:13:08 +02:00
var items = db . Fetch < dynamic > ( "SELECT TABLE_NAME, COLUMN_NAME, ORDINAL_POSITION, COLUMN_DEFAULT, IS_NULLABLE, DATA_TYPE FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA = (SELECT SCHEMA_NAME())" ) ;
2018-06-29 19:52:40 +02:00
return
items . Select (
item = >
new ColumnInfo ( item . TABLE_NAME , item . COLUMN_NAME , item . ORDINAL_POSITION , item . COLUMN_DEFAULT ,
item . IS_NULLABLE , item . DATA_TYPE ) ) . ToList ( ) ;
}
2018-08-20 15:16:58 +10:00
/// <inheritdoc />
2018-06-29 19:52:40 +02:00
public override IEnumerable < Tuple < string , string > > GetConstraintsPerTable ( IDatabase db )
{
var items =
db . Fetch < dynamic > (
2018-08-26 23:13:08 +02:00
"SELECT TABLE_NAME, CONSTRAINT_NAME FROM INFORMATION_SCHEMA.CONSTRAINT_TABLE_USAGE WHERE TABLE_SCHEMA = (SELECT SCHEMA_NAME())" ) ;
2018-06-29 19:52:40 +02:00
return items . Select ( item = > new Tuple < string , string > ( item . TABLE_NAME , item . CONSTRAINT_NAME ) ) . ToList ( ) ;
}
2018-08-20 15:16:58 +10:00
/// <inheritdoc />
2018-06-29 19:52:40 +02:00
public override IEnumerable < Tuple < string , string , string > > GetConstraintsPerColumn ( IDatabase db )
{
var items =
db . Fetch < dynamic > (
2018-08-26 23:13:08 +02:00
"SELECT TABLE_NAME, COLUMN_NAME, CONSTRAINT_NAME FROM INFORMATION_SCHEMA.CONSTRAINT_COLUMN_USAGE WHERE TABLE_SCHEMA = (SELECT SCHEMA_NAME())" ) ;
2018-06-29 19:52:40 +02:00
return items . Select ( item = > new Tuple < string , string , string > ( item . TABLE_NAME , item . COLUMN_NAME , item . CONSTRAINT_NAME ) ) . ToList ( ) ;
}
2018-08-20 15:16:58 +10:00
/// <inheritdoc />
2018-06-29 19:52:40 +02:00
public override IEnumerable < Tuple < string , string , string , bool > > GetDefinedIndexes ( IDatabase db )
{
var items =
db . Fetch < dynamic > (
@ "select T.name as TABLE_NAME, I.name as INDEX_NAME, AC.Name as COLUMN_NAME,
CASE WHEN I . is_unique_constraint = 1 OR I . is_unique = 1 THEN 1 ELSE 0 END AS [ UNIQUE ]
from sys . tables as T inner join sys . indexes as I on T . [ object_id ] = I . [ object_id ]
inner join sys . index_columns as IC on IC . [ object_id ] = I . [ object_id ] and IC . [ index_id ] = I . [ index_id ]
inner join sys . all_columns as AC on IC . [ object_id ] = AC . [ object_id ] and IC . [ column_id ] = AC . [ column_id ]
2018-08-26 23:13:08 +02:00
inner join sys . schemas as S on T . [ schema_id ] = S . [ schema_id ]
WHERE S . name = ( SELECT SCHEMA_NAME ( ) ) AND I . is_primary_key = 0
2018-06-29 19:52:40 +02:00
order by T . name , I . name ");
return items . Select ( item = > new Tuple < string , string , string , bool > ( item . TABLE_NAME , item . INDEX_NAME , item . COLUMN_NAME ,
item . UNIQUE = = 1 ) ) . ToList ( ) ;
}
public override bool DoesTableExist ( IDatabase db , string tableName )
{
var result =
2018-08-26 23:13:08 +02:00
db . ExecuteScalar < long > ( "SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_NAME = @TableName AND TABLE_SCHEMA = (SELECT SCHEMA_NAME())" ,
2018-06-29 19:52:40 +02:00
new { TableName = tableName } ) ;
return result > 0 ;
}
public override string FormatColumnRename ( string tableName , string oldName , string newName )
{
return string . Format ( RenameColumn , tableName , oldName , newName ) ;
}
public override string FormatTableRename ( string oldName , string newName )
{
return string . Format ( RenameTable , oldName , newName ) ;
}
protected override string FormatIdentity ( ColumnDefinition column )
{
return column . IsIdentity ? GetIdentityString ( column ) : string . Empty ;
}
public override Sql < ISqlContext > SelectTop ( Sql < ISqlContext > sql , int top )
{
return new Sql < ISqlContext > ( sql . SqlContext , sql . SQL . Insert ( sql . SQL . IndexOf ( ' ' ) , " TOP " + top ) , sql . Arguments ) ;
}
private static string GetIdentityString ( ColumnDefinition column )
{
return "IDENTITY(1,1)" ;
}
protected override string FormatSystemMethods ( SystemMethods systemMethod )
{
switch ( systemMethod )
{
case SystemMethods . NewGuid :
return "NEWID()" ;
case SystemMethods . CurrentDateTime :
return "GETDATE()" ;
//case SystemMethods.NewSequentialId:
// return "NEWSEQUENTIALID()";
//case SystemMethods.CurrentUTCDateTime:
// return "GETUTCDATE()";
}
return null ;
}
public override string DeleteDefaultConstraint = > "ALTER TABLE [{0}] DROP CONSTRAINT [DF_{0}_{1}]" ;
public override string DropIndex = > "DROP INDEX {0} ON {1}" ;
public override string RenameColumn = > "sp_rename '{0}.{1}', '{2}', 'COLUMN'" ;
}
}