mirror of https://github.com/lidarr/Lidarr
121 lines
3.6 KiB
C#
121 lines
3.6 KiB
C#
using System.Text;
|
|
|
|
namespace Marr.Data.QGen
|
|
{
|
|
public class RowCountQueryDecorator : IQuery
|
|
{
|
|
private SelectQuery _innerQuery;
|
|
|
|
public RowCountQueryDecorator(SelectQuery innerQuery)
|
|
{
|
|
_innerQuery = innerQuery;
|
|
}
|
|
|
|
public string Generate()
|
|
{
|
|
// Decide which type of paging query to create
|
|
if (_innerQuery.IsView || _innerQuery.IsJoin)
|
|
{
|
|
return ComplexRowCount();
|
|
}
|
|
return SimpleRowCount();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates a row count query for a multiple table joined query (groups by the parent entity).
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
private string ComplexRowCount()
|
|
{
|
|
// Create paged query
|
|
StringBuilder sql = new StringBuilder();
|
|
|
|
sql.AppendLine("WITH GroupCTE AS (");
|
|
sql.Append("SELECT ").AppendLine(BuildBaseTablePKColumns());
|
|
BuildGroupColumn(sql);
|
|
_innerQuery.BuildFromClause(sql);
|
|
_innerQuery.BuildJoinClauses(sql);
|
|
_innerQuery.BuildWhereClause(sql);
|
|
sql.AppendLine(")");
|
|
BuildSelectCountClause(sql);
|
|
sql.AppendLine("FROM GroupCTE");
|
|
sql.AppendLine("WHERE GroupRow = 1");
|
|
|
|
return sql.ToString();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates a row count query for a single table query (no joins).
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
private string SimpleRowCount()
|
|
{
|
|
StringBuilder sql = new StringBuilder();
|
|
|
|
BuildSelectCountClause(sql);
|
|
_innerQuery.BuildFromClause(sql);
|
|
_innerQuery.BuildJoinClauses(sql);
|
|
_innerQuery.BuildWhereClause(sql);
|
|
|
|
return sql.ToString();
|
|
}
|
|
|
|
private void BuildGroupColumn(StringBuilder sql)
|
|
{
|
|
string baseTablePKColumns = BuildBaseTablePKColumns();
|
|
sql.AppendFormat(", ROW_NUMBER() OVER (PARTITION BY {0} ORDER BY {1}) As GroupRow ", baseTablePKColumns, baseTablePKColumns);
|
|
}
|
|
|
|
private string BuildBaseTablePKColumns()
|
|
{
|
|
Table baseTable = GetBaseTable();
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
foreach (var col in baseTable.Columns.PrimaryKeys)
|
|
{
|
|
if (sb.Length > 0)
|
|
sb.AppendLine(", ");
|
|
|
|
string colName = _innerQuery.IsView ?
|
|
_innerQuery.NameOrAltName(col.ColumnInfo) :
|
|
col.ColumnInfo.Name;
|
|
|
|
sb.AppendFormat(_innerQuery.Dialect.CreateToken(string.Concat(baseTable.Alias, ".", colName)));
|
|
}
|
|
|
|
return sb.ToString();
|
|
}
|
|
|
|
private void BuildSelectCountClause(StringBuilder sql)
|
|
{
|
|
sql.AppendLine("SELECT COUNT(*)");
|
|
}
|
|
|
|
private Table GetBaseTable()
|
|
{
|
|
Table baseTable = null;
|
|
if (_innerQuery.Tables[0] is View)
|
|
{
|
|
baseTable = (_innerQuery.Tables[0] as View).Tables[0];
|
|
}
|
|
else
|
|
{
|
|
baseTable = _innerQuery.Tables[0];
|
|
}
|
|
return baseTable;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
WITH GroupCTE AS
|
|
(
|
|
SELECT [t0].[ID],[t0].[OrderName],[t1].[ID] AS OrderItemID,[t1].[OrderID],[t1].[ItemDescription],[t1].[Price],
|
|
ROW_NUMBER() OVER (PARTITION BY [t0].[ID] ORDER BY [t0].[OrderName]) As GroupRow
|
|
FROM [Order] [t0]
|
|
LEFT JOIN [OrderItem] [t1] ON (([t0].[ID] = [t1].[OrderID]))
|
|
--WHERE (([t0].[OrderName] = @P0))
|
|
)
|
|
SELECT * FROM GroupCTE
|
|
WHERE GroupRow = 1
|
|
*/ |