For a given Entity, I manually choose which relationships to use (including inherited ones) and a give each a friendly name.
The code below works for the two sample inherited relations from LegalBody but as soon as I uncomment the one for Person, I get an exception: "Relation at index 2 doesn't contain an entity already added to the FROM clause. Bad alias?"
which I suspect is the old "entities are entities not individual tables" issue I have come across before.
Still, you get the idea - the DataTable will have columns made up of the appropriate Friendly names and I can fetch for just one entity or all so I can even find 'unused' Persons.
The idea is sound but I just get so confused converting metadata into SQL I can handwrite (this is the latest in many previous attempts).
public class DeleteChecker
{
public static DeleteChecker ForPerson(int id)
{
return new DeleteChecker(new PersonEntity(id), LegalBodyRelations, PersonRelations);
}
static void LegalBodyRelations(DeleteChecker result)
{
result.AddRelation(LegalBodyEntity.Relations.AccountsTransactionEntityUsingLegalBodyID, "Invoices");
result.AddRelation(LegalBodyEntity.Relations.AssetToOwnerEntityUsingLegalBodyID, "Assets Owned");
}
static void PersonRelations(DeleteChecker result)
{
//result.AddRelation(PersonEntity.Relations.LifeAssuredEntityUsingPersonID, "Life Assured on Policy");
}
readonly CommonEntityBase entity;
readonly List<RelationInfo> relationInfos = new List<RelationInfo>();
DeleteChecker(CommonEntityBase entity, params Action<DeleteChecker>[] sources)
{
this.entity = entity;
foreach (var source in sources)
{
source(this);
}
}
void AddRelation(IEntityRelation relation, string friendlyName)
{
relationInfos.Add(new RelationInfo { FriendlyName = friendlyName, Relation = relation });
}
class RelationInfo
{
public IEntityRelation Relation { get; set; }
public string FriendlyName { get; set; }
}
public object BuildQuery()
{
var resultSetFields = new ResultsetFields(relationInfos.Count + 1);
var bucket = new RelationPredicateBucket();
resultSetFields.DefineField(entity.Fields.PrimaryKeyFields[0], 0, "ID");
var groupByCollection = new GroupByCollection(resultSetFields[0]);
for (var i = 0; i < relationInfos.Count; i++)
{
var resultSetInfo = relationInfos[i];
var relation = resultSetInfo.Relation;
resultSetFields.DefineField(relation.GetFKEntityFieldCore(0), i + 1, resultSetInfo.FriendlyName, AggregateFunction.Count);
bucket.Relations.Add(relation, JoinHint.Left);
}
using (var adapter = new DataAccessAdapter())
{
var dataTable = new DataTable();
adapter.FetchTypedList(resultSetFields, dataTable, bucket, 0, null, false, groupByCollection);
}
return null;
}
The SQL produced is:-
SELECT DISTINCT
[LPA_L1].[ID],
COUNT([LPA_L2].[LegalBodyID]) AS [Invoices],
COUNT([TIPS].[dbo].[AssetToOwner].[LegalBodyID]) AS [Assets Owned]
FROM
((( [TIPS].[dbo].[LegalBody] [LPA_L1]
LEFT JOIN
[TIPS].[dbo].[AccountsTransaction] [LPA_L2] ON [LPA_L1].[ID] = [LPA_L2].[LegalBodyID])
LEFT JOIN
[TIPS].[dbo].[AssetToOwner] ON [LPA_L1].[ID] = [TIPS].[dbo].[AssetToOwner].[LegalBodyID])
LEFT JOIN
[TIPS].[dbo].[Person] [LPA_L3] ON [LPA_L1].[ID] = [LPA_L3].[ID])
WHERE
( ( ( [LPA_L3].[ID] IS NOT NULL)))
GROUP BY
[LPA_L1].[ID]
(a COUNT(*) would probably be better then I don't need to use a PK or FK field)