/Current/ODX.Core/datasetbuilder.cs

# · C# · 200 lines · 163 code · 35 blank · 2 comment · 36 complexity · f8e14d9900165a4360c2fd9eba04ac14 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using System.Reflection;
  5. namespace ODX.Core
  6. {
  7. internal static class DataSetBuilder
  8. {
  9. internal static DataSet BuildDataSet(Polymorpher pm)
  10. {
  11. DataSet ds = new DataSet();
  12. foreach (string table in pm.GetTables())
  13. {
  14. DataTable dt = ds.Tables.Add(table);
  15. DataColumn id = dt.Columns.Add("ID", typeof (string));
  16. AddProperties(new ColumnInfo(id.ColumnName, true, 22), id);
  17. dt.Constraints.Add("PK_" + dt.TableName, id, true);
  18. if (pm.DoesTableVersioned(table))
  19. {
  20. DataColumn dc = dt.Columns.Add(SpecialColumns.RowVersion, typeof (string));
  21. AddProperties(new ColumnInfo(dc.ColumnName, false, 22), dc);
  22. }
  23. }
  24. foreach (string table in pm.GetTables())
  25. {
  26. string parent = pm.GetTypeTable(pm.GetRootTypeForTable(table).BaseType);
  27. if ( parent != null )
  28. {
  29. DataTable baseTable = ds.Tables[parent];
  30. DataTable inheritedTable = ds.Tables[table];
  31. CreateRelation(ds, baseTable.TableName, inheritedTable.TableName, new ColumnInfo("ID", false, 22));
  32. }
  33. }
  34. foreach ( Type type in pm.GetTypes() )
  35. {
  36. string table = pm.GetTypeTable(type);
  37. if ( table == null )
  38. continue;
  39. DataTable dt = ds.Tables[table];
  40. if (BuilderUtils.IsHierarchyRoot(type))
  41. {
  42. DataColumn dc = dt.Columns.Add(Session.TypeDefField, typeof (string));
  43. AddProperties(new ColumnInfo(dc.ColumnName,false,22), dc);
  44. }
  45. foreach (PropertyInfo pi in type.GetProperties(BuilderUtils.PropertyFilter))
  46. {
  47. if ( !BuilderUtils.IsAutoProperty(pi) )
  48. continue;
  49. if (pi.PropertyType.IsSubclassOf(typeof(Entity)))
  50. CreateParent(pi, dt, pm);
  51. else if (new List<Type>(pi.PropertyType.GetInterfaces()).Contains(typeof(IEntityListBase)))
  52. CreateList(pi, dt, pm);
  53. else
  54. CreateAtom(pi, dt);
  55. }
  56. }
  57. foreach ( DataTable dt in ds.Tables )
  58. {
  59. foreach ( Constraint c in dt.Constraints )
  60. {
  61. // Single-column unique constraints are created by assigning DataColumn.Unique property
  62. // So it's much easier to rename constraints. Otherwise they is named "constraint1", ...
  63. UniqueConstraint uc = c as UniqueConstraint;
  64. if (uc != null && !uc.IsPrimaryKey && uc.Columns.Length == 1)
  65. uc.ConstraintName = "UC_" + uc.Columns[0].ColumnName;
  66. }
  67. Type t = pm.GetRootTypeForTable(dt.TableName);
  68. if ( t == null )
  69. continue;
  70. UniqueConstraintAttribute[] attrs = (UniqueConstraintAttribute[])t.GetCustomAttributes(typeof(UniqueConstraintAttribute), false);
  71. foreach (UniqueConstraintAttribute a in attrs)
  72. {
  73. List<DataColumn> cols = new List<DataColumn>();
  74. foreach ( string col in a.Columns )
  75. cols.Add(dt.Columns[col]);
  76. dt.Constraints.Add(
  77. "UC_" + string.Join("_", a.Columns),
  78. cols.ToArray(),
  79. false);
  80. }
  81. }
  82. return ds;
  83. }
  84. private static void CreateList(PropertyInfo pi, DataTable dt, Polymorpher pm)
  85. {
  86. LinkTableAttribute[] linkAttrs = (LinkTableAttribute[])pi.GetCustomAttributes(typeof(LinkTableAttribute), false);
  87. if ( linkAttrs.Length > 0 )
  88. {
  89. CreateLink(dt, linkAttrs[0].Name, linkAttrs[0].ParentColumn, linkAttrs[0].ChildColumn, pi, pm);
  90. }
  91. else
  92. {
  93. ColumnInfo ci = BuilderUtils.GetColumnInfo(pi);
  94. if ( ci == null ) // this is many-To-Manu relation with autogenerated link table name
  95. CreateLink(dt, null, null, null, pi, pm);
  96. else
  97. CreateRelation(dt.DataSet, dt.TableName, BuilderUtils.GetRelatedTableName(pi, pm), ci);
  98. }
  99. }
  100. private static void CreateLink(DataTable dt, string linkTableName, string parentColName, string childColName, PropertyInfo pi, Polymorpher pm)
  101. {
  102. string thisTable = dt.TableName;
  103. string childTable = BuilderUtils.GetRelatedTableName(pi, pm);
  104. string thisFK = parentColName ?? thisTable + "ID";
  105. string childFK = childColName ?? childTable + "ID";
  106. if ( (childFK == thisFK) || (linkTableName == null && thisTable == childTable) )
  107. throw new OdxException(
  108. "When self-associating table you should use explicit link table name & FK names in link table attribute.");
  109. if (linkTableName == null)
  110. linkTableName = thisTable.CompareTo(childTable) > 0 ? thisTable + childTable : childTable + thisTable;
  111. if (dt.DataSet.Tables.Contains(linkTableName))
  112. return;
  113. DataTable link = dt.DataSet.Tables.Add(linkTableName);
  114. DataColumn linkID = link.Columns.Add("ID", typeof (string));
  115. link.PrimaryKey = new DataColumn[] { linkID };
  116. AddProperties(new ColumnInfo(linkID.ColumnName, true, 22), linkID);
  117. CreateRelation(dt.DataSet, thisTable, link.TableName, new ColumnInfo(thisFK, false, 22));
  118. CreateRelation(dt.DataSet, childTable, link.TableName, new ColumnInfo(childFK, false, 22));
  119. link.Constraints.Add(
  120. "UC_" + thisFK + "_" + childFK,
  121. new DataColumn[] {link.Columns[thisFK], link.Columns[childFK]},
  122. false);
  123. }
  124. private static void CreateParent(PropertyInfo pi, DataTable dt, Polymorpher pm)
  125. {
  126. ColumnInfo ci = BuilderUtils.GetColumnInfo(pi);
  127. CreateRelation(dt.DataSet, BuilderUtils.GetRelatedTableName(pi, pm), dt.TableName, ci);
  128. }
  129. private static void CreateRelation(DataSet ds, string parentTableName, string tableName, ColumnInfo ci)
  130. {
  131. string name = "FK_" + parentTableName + "_" + tableName + "_" + ci.Name;
  132. if ( ds.Relations.Contains(name) )
  133. return;
  134. DataTable parent = ds.Tables[parentTableName];
  135. DataTable child = ds.Tables[tableName];
  136. DataColumn dc;
  137. if (child.Columns.Contains(ci.Name))
  138. dc = child.Columns[ci.Name];
  139. else
  140. dc = child.Columns.Add(ci.Name, typeof (string));
  141. dc.Unique |= ci.IsUnique;
  142. ds.Relations.Add(name, parent.PrimaryKey[0], dc);
  143. AddProperties(ci,dc);
  144. }
  145. private static void CreateAtom(PropertyInfo pi, DataTable dt)
  146. {
  147. Type type = pi.PropertyType;
  148. if (type.IsEnum)
  149. type = typeof (int);
  150. if (type == typeof(Guid))
  151. type = typeof (string);
  152. ColumnInfo ci = BuilderUtils.GetColumnInfo(pi);
  153. DataColumn dc = dt.Columns.Add(ci.Name, type);
  154. dc.Unique = ci.IsUnique;
  155. AddProperties(ci, dc);
  156. }
  157. private static void AddProperties(ColumnInfo ci, DataColumn dc)
  158. {
  159. if ( ci.DbType != null )
  160. dc.ExtendedProperties["dbType"] = ci.DbType;
  161. dc.ExtendedProperties["dbMaxLength"] = ci.MaxLength;
  162. dc.ExtendedProperties["dbScale"] = ci.Scale;
  163. }
  164. }
  165. }