mirror of
https://github.com/dotnetcore/FreeSql.git
synced 2026-02-06 00:10:55 +08:00
326 lines
14 KiB
C#
326 lines
14 KiB
C#
using FreeSql.Internal;
|
|
using FreeSql.Internal.Model;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Data;
|
|
using System.Data.Common;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using FreeSql.DataAnnotations;
|
|
using System.Reflection;
|
|
using FreeSql.Internal.ObjectPool;
|
|
using FreeSql.Provider.TDengine.Attributes;
|
|
|
|
namespace FreeSql.TDengine
|
|
{
|
|
internal class TDengineCodeFirst : Internal.CommonProvider.CodeFirstProvider
|
|
{
|
|
public TDengineCodeFirst(IFreeSql orm, CommonUtils commonUtils, CommonExpression commonExpression) : base(orm,
|
|
commonUtils, commonExpression)
|
|
{
|
|
}
|
|
|
|
static object DicCsToDbLock = new object();
|
|
static readonly Dictionary<string, CsToDb<DbType>> DicCsToDb = new Dictionary<string, CsToDb<DbType>>()
|
|
{
|
|
{ typeof(bool).FullName, CsToDb.New(DbType.Boolean, "BOOL", "BOOL", null, false, null) },
|
|
{ typeof(bool?).FullName, CsToDb.New(DbType.Boolean, "BOOL", "BOOL", null, true, null) },
|
|
{ typeof(DateTime).FullName, CsToDb.New(DbType.DateTime, "TIMESTAMP", "TIMESTAMP", null, false, null) },
|
|
{ typeof(DateTime?).FullName, CsToDb.New(DbType.DateTime, "TIMESTAMP", "TIMESTAMP", null, true, null) },
|
|
{ typeof(TimeSpan).FullName, CsToDb.New(DbType.DateTime, "TIMESTAMP", "TIMESTAMP", null, false, null) },
|
|
{ typeof(TimeSpan?).FullName, CsToDb.New(DbType.DateTime, "TIMESTAMP", "TIMESTAMP", null, true, null) },
|
|
{ typeof(short).FullName, CsToDb.New(DbType.Int16, "SMALLINT", "SMALLINT", null, false, 0) },
|
|
{ typeof(short?).FullName, CsToDb.New(DbType.Int16, "SMALLINT", "SMALLINT", null, true, null) },
|
|
{ typeof(int).FullName, CsToDb.New(DbType.Int32, "INT", "INT", null, false, 0) },
|
|
{ typeof(int?).FullName, CsToDb.New(DbType.Int32, "INT", "INT", null, true, null) },
|
|
{ typeof(sbyte).FullName, CsToDb.New(DbType.SByte, "TINYINT", "TINYINT", null, false, 0) },
|
|
{ typeof(sbyte?).FullName, CsToDb.New(DbType.SByte, "TINYINT", "TINYINT", null, true, null) },
|
|
{ typeof(long).FullName, CsToDb.New(DbType.Int64, "BIGINT", "BIGINT", null, false, 0) },
|
|
{ typeof(long?).FullName, CsToDb.New(DbType.Int64, "BIGINT", "BIGINT", null, true, null) },
|
|
{ typeof(byte).FullName, CsToDb.New(DbType.Byte, "TINYINT UNSIGNED", "TINYINT UNSIGNED", null, false, 0) },
|
|
{
|
|
typeof(byte?).FullName,
|
|
CsToDb.New(DbType.Byte, "TINYINT UNSIGNED", "TINYINT UNSIGNED", null, true, null)
|
|
},
|
|
{
|
|
typeof(ushort).FullName,
|
|
CsToDb.New(DbType.UInt16, "SMALLINT UNSIGNED", "SMALLINT UNSIGNED", null, false, 0)
|
|
},
|
|
{
|
|
typeof(ushort?).FullName,
|
|
CsToDb.New(DbType.UInt16, "SMALLINT UNSIGNED", "SMALLINT UNSIGNED", null, true, null)
|
|
},
|
|
{ typeof(uint).FullName, CsToDb.New(DbType.UInt32, "INT UNSIGNED", "INT UNSIGNED", null, false, 0) },
|
|
{ typeof(uint?).FullName, CsToDb.New(DbType.UInt32, "INT UNSIGNED", "INT UNSIGNED", null, true, null) },
|
|
{ typeof(ulong).FullName, CsToDb.New(DbType.UInt64, "BIGINT UNSIGNED", "BIGINT UNSIGNED", null, false, 0) },
|
|
{
|
|
typeof(ulong?).FullName,
|
|
CsToDb.New(DbType.UInt64, "BIGINT UNSIGNED", "BIGINT UNSIGNED", null, true, null)
|
|
},
|
|
{ typeof(float).FullName, CsToDb.New(DbType.Single, "FLOAT", "FLOAT", null, false, 0) },
|
|
{ typeof(float?).FullName, CsToDb.New(DbType.Single, "FLOAT", "FLOAT", null, true, null) },
|
|
{ typeof(double).FullName, CsToDb.New(DbType.Double, "DOUBLE", "DOUBLE", null, false, 0) },
|
|
{ typeof(double?).FullName, CsToDb.New(DbType.Double, "DOUBLE", "DOUBLE", null, true, null) },
|
|
{ typeof(string).FullName, CsToDb.New(DbType.String, "NCHAR", "NCHAR(255)", null, false, 0) },
|
|
};
|
|
|
|
public override DbInfoResult GetDbInfo(Type type)
|
|
{
|
|
if (DicCsToDb.TryGetValue(type.FullName, out var trydc))
|
|
return new DbInfoResult((int)trydc.type, trydc.dbtype, trydc.dbtypeFull, trydc.isnullable,
|
|
trydc.defaultValue);
|
|
if (type.IsArray) return null;
|
|
var enumType = type.IsEnum ? type : null;
|
|
if (enumType == null && type.IsNullableType() && type.GenericTypeArguments.Length == 1 && type.GenericTypeArguments.First().IsEnum) enumType = type.GenericTypeArguments.First();
|
|
if (enumType != null)
|
|
{
|
|
var newItem = enumType.GetCustomAttributes(typeof(FlagsAttribute), false).Any() ?
|
|
CsToDb.New(DbType.Int32, "int", $"int{(type.IsEnum ? " NOT NULL" : "")}", false, type.IsEnum ? false : true, enumType.CreateInstanceGetDefaultValue()) :
|
|
CsToDb.New(DbType.Int64, "long", $"long{(type.IsEnum ? " NOT NULL" : "")}", false, type.IsEnum ? false : true, enumType.CreateInstanceGetDefaultValue());
|
|
if (DicCsToDb.ContainsKey(type.FullName) == false)
|
|
{
|
|
lock (DicCsToDbLock)
|
|
{
|
|
if (DicCsToDb.ContainsKey(type.FullName) == false)
|
|
DicCsToDb.Add(type.FullName, newItem);
|
|
}
|
|
}
|
|
return new DbInfoResult((int)newItem.type, newItem.dbtype, newItem.dbtypeFull, newItem.isnullable, newItem.defaultValue);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
protected override string GetComparisonDDLStatements(params TypeSchemaAndName[] objects)
|
|
{
|
|
Object<DbConnection> conn = null;
|
|
string database = null;
|
|
var sb = new StringBuilder();
|
|
try
|
|
{
|
|
conn = _orm.Ado.MasterPool.Get(TimeSpan.FromSeconds(5));
|
|
database = conn.Value.Database;
|
|
foreach (var obj in objects)
|
|
{
|
|
if (sb.Length > 0) sb.Append(Environment.NewLine);
|
|
var tb = obj.tableSchema;
|
|
if (tb == null)
|
|
throw new Exception(CoreErrorStrings.S_Type_IsNot_Migrable(obj.tableSchema.Type.FullName));
|
|
if (tb.Columns.Any() == false)
|
|
throw new Exception(
|
|
CoreErrorStrings.S_Type_IsNot_Migrable_0Attributes(obj.tableSchema.Type.FullName));
|
|
|
|
var tbName = _commonUtils.SplitTableName(tb.DbName).First();
|
|
|
|
tbName = _commonUtils.QuoteSqlName(database, tbName);
|
|
|
|
if (!TryTableExists(tbName))
|
|
{
|
|
TableHandle(ref tb, ref database, tb.Type, ref sb, tbName);
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
try
|
|
{
|
|
if (string.IsNullOrEmpty(database) == false)
|
|
conn.Value.ChangeDatabase(database);
|
|
_orm.Ado.MasterPool.Return(conn);
|
|
}
|
|
catch
|
|
{
|
|
_orm.Ado.MasterPool.Return(conn, true);
|
|
}
|
|
}
|
|
|
|
|
|
var ddl = sb.Length == 0 ? null : sb.ToString();
|
|
return ddl;
|
|
}
|
|
|
|
private void CreateColumns(ref TableInfo tb, ref StringBuilder sb)
|
|
{
|
|
//创建表
|
|
foreach (var columnInfo in tb.ColumnsByPosition.Where(c =>
|
|
!c.Table.Properties[c.CsName].IsDefined(typeof(TDengineTagAttribute))))
|
|
{
|
|
sb.Append($" {Environment.NewLine} ").Append(_commonUtils.QuoteSqlName(columnInfo.Attribute.Name))
|
|
.Append(" ")
|
|
.Append(columnInfo.Attribute.DbType);
|
|
sb.Append(",");
|
|
}
|
|
|
|
sb.Remove(sb.Length - 1, 1).Append($"{Environment.NewLine})");
|
|
}
|
|
|
|
private void TableHandle(ref TableInfo tb, ref string database, Type type, ref StringBuilder sb, string tbName)
|
|
{
|
|
//判断是否超级表
|
|
var subTableAttribute = type.GetCustomAttribute<TDengineSubTableAttribute>();
|
|
|
|
//要创建表的为子表
|
|
if (subTableAttribute != null)
|
|
{
|
|
if (_commonUtils is TDengineUtils utils)
|
|
{
|
|
var superTableDescribe = utils.GetSuperTableDescribe(type);
|
|
|
|
if (superTableDescribe == null) return;
|
|
|
|
var superTableName = _commonUtils.QuoteSqlName(database, superTableDescribe.SuperTableName);
|
|
var superTableInfo = GetTableByEntity(superTableDescribe.SuperTableType);
|
|
|
|
//判断超表是否存在
|
|
if (!TryTableExists(superTableName))
|
|
{
|
|
//先创建超级表
|
|
CreateSuperTable(ref superTableInfo, ref sb, superTableName);
|
|
_orm.Ado.ExecuteNonQuery(sb.ToString());
|
|
sb = sb.Clear();
|
|
}
|
|
|
|
var subTableName = _commonUtils.QuoteSqlName(database, subTableAttribute.Name);
|
|
|
|
//创建子表
|
|
CreateSubTable(ref tb, ref sb, superTableName, subTableName, ref superTableInfo);
|
|
}
|
|
}
|
|
//要创建的为超级表
|
|
else if (type.IsDefined(typeof(TDengineSuperTableAttribute)))
|
|
{
|
|
var superTableAttribute = type.GetCustomAttribute<TDengineSuperTableAttribute>();
|
|
if (superTableAttribute == null) return;
|
|
tbName = _commonUtils.QuoteSqlName(database, superTableAttribute.Name);
|
|
CreateSuperTable(ref tb, ref sb, tbName);
|
|
}
|
|
//创建普通表
|
|
else
|
|
{
|
|
CreateNormalTable(ref tb, ref sb, tbName);
|
|
}
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// 创建子表
|
|
/// </summary>
|
|
/// <param name="childTableInfo"></param>
|
|
/// <param name="sb"></param>
|
|
/// <param name="superTableName"></param>
|
|
private void CreateSubTable(ref TableInfo childTableInfo, ref StringBuilder sb, string superTableName,
|
|
string subTableName, ref TableInfo
|
|
superTableInfo)
|
|
{
|
|
sb.Append($"CREATE TABLE {subTableName}{Environment.NewLine}");
|
|
sb.Append($"USING {superTableName} (");
|
|
|
|
var tagCols = superTableInfo.ColumnsByPosition.Where(c =>
|
|
c.Table.Properties[c.CsName].IsDefined(typeof(TDengineTagAttribute))).ToArray();
|
|
|
|
var tagValues = new List<object>(tagCols.Count());
|
|
|
|
var tableInstance = Activator.CreateInstance(childTableInfo.Type);
|
|
|
|
foreach (var columnInfo in tagCols)
|
|
{
|
|
var tagValue = childTableInfo.Properties[columnInfo.CsName].GetValue(tableInstance);
|
|
tagValues.Add(tagValue);
|
|
sb.Append($" {Environment.NewLine} ").Append(_commonUtils.QuoteSqlName(columnInfo.Attribute.Name))
|
|
.Append(",");
|
|
}
|
|
|
|
sb.Remove(sb.Length - 1, 1).Append($"{Environment.NewLine}) TAGS (");
|
|
|
|
foreach (var tagValue in tagValues)
|
|
{
|
|
sb.Append($" {Environment.NewLine} ").Append(HandleTagValue(tagValue)).Append(",");
|
|
}
|
|
|
|
sb.Remove(sb.Length - 1, 1).Append($"{Environment.NewLine});");
|
|
}
|
|
|
|
/// <summary>
|
|
/// 创建超级表
|
|
/// </summary>
|
|
/// <param name="tb"></param>
|
|
/// <param name="sb"></param>
|
|
/// <param name="superTableName"></param>
|
|
private void CreateSuperTable(ref TableInfo tb, ref StringBuilder sb, string superTableName)
|
|
{
|
|
sb.Append($"CREATE STABLE {superTableName} (");
|
|
CreateColumns(ref tb, ref sb);
|
|
sb.Append($" TAGS (");
|
|
|
|
var columInfos = tb.ColumnsByPosition.Where(c =>
|
|
c.Table.Properties[c.CsName].IsDefined(typeof(TDengineTagAttribute)));
|
|
|
|
foreach (var columnInfo in columInfos)
|
|
{
|
|
sb.Append($" {Environment.NewLine} ").Append(_commonUtils.QuoteSqlName(columnInfo.Attribute.Name))
|
|
.Append(" ")
|
|
.Append(columnInfo.Attribute.DbType);
|
|
sb.Append(",");
|
|
}
|
|
|
|
sb.Remove(sb.Length - 1, 1).Append($"{Environment.NewLine});");
|
|
}
|
|
|
|
/// <summary>
|
|
/// 创建普通表
|
|
/// </summary>
|
|
/// <param name="tb"></param>
|
|
/// <param name="sb"></param>
|
|
/// <param name="normalTableName"></param>
|
|
private void CreateNormalTable(ref TableInfo tb, ref StringBuilder sb, string normalTableName)
|
|
{
|
|
sb.Append($"CREATE TABLE {normalTableName} (");
|
|
CreateColumns(ref tb, ref sb);
|
|
foreach (var columnInfo in tb.ColumnsByPosition.Where(c =>
|
|
c.Table.Properties[c.CsName].IsDefined(typeof(TDengineTagAttribute))))
|
|
{
|
|
sb.Append($" {Environment.NewLine} ").Append(_commonUtils.QuoteSqlName(columnInfo.Attribute.Name))
|
|
.Append(" ")
|
|
.Append(columnInfo.Attribute.DbType);
|
|
sb.Append(",");
|
|
}
|
|
|
|
sb.Remove(sb.Length - 1, 1).Append(");");
|
|
}
|
|
|
|
private bool TryTableExists(string tbName)
|
|
{
|
|
var flag = true;
|
|
try
|
|
{
|
|
var executeScalar = _orm.Ado.ExecuteScalar(CommandType.Text,
|
|
$"DESCRIBE {tbName}");
|
|
|
|
if (executeScalar == null)
|
|
{
|
|
flag = false;
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (e.Message.Contains("Table does not exist"))
|
|
{
|
|
flag = false;
|
|
}
|
|
}
|
|
|
|
return flag;
|
|
}
|
|
|
|
private object HandleTagValue(object tagValue)
|
|
{
|
|
if (tagValue is DateTime || tagValue is string)
|
|
{
|
|
return $"\"{tagValue}\"";
|
|
}
|
|
else
|
|
{
|
|
return tagValue;
|
|
}
|
|
}
|
|
}
|
|
} |