//--------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner Microsoft // @backupOwner Microsoft //--------------------------------------------------------------------- using System; using System.Collections.Generic; using System.Data.Common; using System.Data.Metadata.Edm; using System.Globalization; using System.Diagnostics; namespace System.Data.Query.InternalTrees { /// /// Describes metadata about a table /// internal class TableMD { private List m_columns; private List m_keys; private EntitySetBase m_extent; // null for transient tables private bool m_flattened; /// /// private initializer /// /// the entity set corresponding to this table (if any) private TableMD(EntitySetBase extent) { m_columns = new List(); m_keys = new List(); m_extent = extent; } /// /// Create a typed-table definition corresponding to an entityset (if specified) /// /// The table has exactly one column - the type of the column is specified by /// the "type" parameter. This table is considered to be un-"flattened" /// /// type of each element (row) of the table /// entityset corresponding to the table (if any) internal TableMD(TypeUsage type, EntitySetBase extent) : this(extent) { m_columns.Add(new ColumnMD(this, "element", type)); m_flattened = !PlanCompiler.TypeUtils.IsStructuredType(type); } /// /// Creates a "flattened" table definition. /// /// The table has one column for each specified property in the "properties" parameter. /// The name and datatype of each table column are taken from the corresponding property. /// /// The keys of the table (if any) are those specified in the "keyProperties" parameter /// /// The table may correspond to an entity set (if the entityset parameter was non-null) /// /// prperties corresponding to columns of the table /// /// entityset corresponding to the table (if any) internal TableMD(IEnumerable properties, IEnumerable keyProperties, EntitySetBase extent) : this(extent) { Dictionary columnMap = new Dictionary(); m_flattened = true; foreach (EdmProperty p in properties) { ColumnMD newColumn = new ColumnMD(this, p); m_columns.Add(newColumn); columnMap[p.Name] = newColumn; } foreach (EdmMember p in keyProperties) { ColumnMD keyColumn; if (!columnMap.TryGetValue(p.Name, out keyColumn)) { Debug.Assert(false, "keyMember not in columns?"); } else { m_keys.Add(keyColumn); } } } /// /// The extent metadata (if any) /// internal EntitySetBase Extent { get { return m_extent; } } /// /// List of columns of this table /// internal List Columns { get { return m_columns; } } /// /// Keys for this table /// internal List Keys { get { return m_keys; } } /// /// Is this table a "flat" table? /// internal bool Flattened { get { return m_flattened; } } /// /// String form - for debugging /// /// public override string ToString() { return (m_extent != null ? m_extent.Name : "Transient"); } } /// /// Describes information about each column /// internal class ColumnMD { private string m_name; private TypeUsage m_type; private EdmMember m_property; /// /// Default constructor /// /// Table containing this column /// Column name /// Datatype of the column internal ColumnMD(TableMD table, string name, TypeUsage type) { m_name = name; m_type = type; } /// /// More useful default constructor /// /// table containing this column /// property describing this column internal ColumnMD(TableMD table, EdmMember property) : this(table, property.Name, property.TypeUsage) { m_property = property; } /// /// Column Name /// internal string Name { get { return m_name; } } /// /// Datatype of the column /// internal TypeUsage Type { get { return m_type; } } /// /// Is this column nullable ? /// internal bool IsNullable { get { return (m_property == null) || TypeSemantics.IsNullable(m_property); } } /// /// debugging help /// /// public override string ToString() { return m_name; } } /// /// Represents one instance of a table. Contains the table metadata /// internal class Table { private TableMD m_tableMetadata; private VarList m_columns; private VarVec m_referencedColumns; private VarVec m_keys; private VarVec m_nonnullableColumns; private int m_tableId; internal Table(Command command, TableMD tableMetadata, int tableId) { m_tableMetadata = tableMetadata; m_columns = Command.CreateVarList(); m_keys = command.CreateVarVec(); m_nonnullableColumns = command.CreateVarVec(); m_tableId = tableId; Dictionary columnVarMap = new Dictionary(); foreach (ColumnMD c in tableMetadata.Columns) { ColumnVar v = command.CreateColumnVar(this, c); columnVarMap[c.Name] = v; if (!c.IsNullable) { m_nonnullableColumns.Set(v); } } foreach (ColumnMD c in tableMetadata.Keys) { ColumnVar v = columnVarMap[c.Name]; m_keys.Set(v); } m_referencedColumns = command.CreateVarVec(m_columns); } /// /// Metadata for the table instance /// internal TableMD TableMetadata { get { return m_tableMetadata; } } /// /// List of column references /// internal VarList Columns { get { return m_columns; } } /// /// Get the list of all referenced columns. /// internal VarVec ReferencedColumns { get { return m_referencedColumns; } } /// /// /// internal VarVec NonNullableColumns { get { return m_nonnullableColumns; } } /// /// List of keys /// internal VarVec Keys { get { return m_keys; } } /// /// (internal) id for this table instance /// internal int TableId { get { return m_tableId; } } /// /// String form - for debugging /// /// public override string ToString() { return String.Format(CultureInfo.InvariantCulture, "{0}::{1}", m_tableMetadata.ToString(), this.TableId); ; } } }