using NPoco;
using Umbraco.Cms.Core;
using Umbraco.Cms.Core.Cache;
using Umbraco.Cms.Core.Models;
using Umbraco.Cms.Core.Models.Entities;
using Umbraco.Cms.Core.Persistence.Querying;
using Umbraco.Cms.Core.Services;
using Umbraco.Cms.Infrastructure.Persistence.Dtos;
using Umbraco.Cms.Infrastructure.Persistence.Querying;
using Umbraco.Cms.Infrastructure.Persistence.SqlSyntax;
using Umbraco.Cms.Infrastructure.Scoping;
using Umbraco.Extensions;
using static Umbraco.Cms.Core.Persistence.SqlExtensionsStatics;
namespace Umbraco.Cms.Infrastructure.Persistence.Repositories.Implement;
///
/// Represents the EntityRepository used to query entity objects.
///
///
/// Limited to objects that have a corresponding node (in umbracoNode table).
/// Returns objects, i.e. lightweight representation of entities.
///
internal class EntityRepository : RepositoryBase, IEntityRepositoryExtended
{
public EntityRepository(IScopeAccessor scopeAccessor, AppCaches appCaches)
: base(scopeAccessor, appCaches)
{
}
#region Repository
public int CountByQuery(IQuery query, IEnumerable objectTypes, IQuery? filter)
{
Sql sql = Sql();
sql.SelectCount();
sql
.From();
sql.WhereIn(x => x.NodeObjectType, objectTypes );
foreach (Tuple queryClause in query.GetWhereClauses())
{
sql.Where(queryClause.Item1, queryClause.Item2);
}
if (filter is not null)
{
foreach (Tuple filterClause in filter.GetWhereClauses())
{
sql.Where(filterClause.Item1, filterClause.Item2);
}
}
return Database.ExecuteScalar(sql);
}
public IEnumerable GetPagedResultsByQuery(IQuery query, ISet objectTypes,
long pageIndex, int pageSize, out long totalRecords,
IQuery? filter, Ordering? ordering) =>
GetPagedResultsByQuery(query, objectTypes.ToArray(), pageIndex, pageSize, out totalRecords, filter, ordering);
// get a page of entities
public IEnumerable GetPagedResultsByQuery(IQuery query, Guid[] objectTypes,
long pageIndex, int pageSize, out long totalRecords,
IQuery? filter, Ordering? ordering, Action>? sqlCustomization = null)
{
var isContent = objectTypes.Any(objectType =>
objectType == Constants.ObjectTypes.Document || objectType == Constants.ObjectTypes.DocumentBlueprint);
var isMedia = objectTypes.Any(objectType => objectType == Constants.ObjectTypes.Media);
var isMember = objectTypes.Any(objectType => objectType == Constants.ObjectTypes.Member);
Sql sql = GetBaseWhere(isContent, isMedia, isMember, false, s =>
{
sqlCustomization?.Invoke(s);
if (filter != null)
{
foreach (Tuple filterClause in filter.GetWhereClauses())
{
s.Where(filterClause.Item1, filterClause.Item2);
}
}
}, objectTypes);
ordering = ordering ?? Ordering.ByDefault();
var translator = new SqlTranslator(sql, query);
sql = translator.Translate();
sql = AddGroupBy(isContent, isMedia, isMember, sql, ordering.IsEmpty);
if (!ordering.IsEmpty)
{
// apply ordering
ApplyOrdering(ref sql, ordering);
}
// TODO: we should be able to do sql = sql.OrderBy(x => Alias(x.NodeId, "NodeId")); but we can't because the OrderBy extension don't support Alias currently
// no matter what we always must have node id ordered at the end
sql = ordering.Direction == Direction.Ascending ? sql.OrderBy("NodeId") : sql.OrderByDescending("NodeId");
// for content we must query for ContentEntityDto entities to produce the correct culture variant entity names
var pageIndexToFetch = pageIndex + 1;
IEnumerable dtos;
Page? page = Database.Page(pageIndexToFetch, pageSize, sql);
dtos = page.Items;
totalRecords = page.TotalItems;
EntitySlim[] entities = dtos.Select(BuildEntity).ToArray();
BuildVariants(entities.OfType());
return entities;
}
public IEntitySlim? Get(Guid key)
{
Sql sql = GetBaseWhere(false, false, false, false, key);
BaseDto? dto = Database.FirstOrDefault(sql);
return dto == null ? null : BuildEntity(dto);
}
private IEntitySlim? GetEntity(Sql sql, bool isContent, bool isMedia, bool isMember)
{
// isContent is going to return a 1:M result now with the variants so we need to do different things
if (isContent)
{
List? cdtos = Database.Fetch(sql);
return cdtos.Count == 0 ? null : BuildVariants(BuildDocumentEntity(cdtos[0]));
}
BaseDto? dto = isMedia
? Database.FirstOrDefault(sql)
: Database.FirstOrDefault(sql);
if (dto == null)
{
return null;
}
EntitySlim entity = BuildEntity(dto);
return entity;
}
public IEntitySlim? Get(Guid key, Guid objectTypeId)
{
var isContent = objectTypeId == Constants.ObjectTypes.Document ||
objectTypeId == Constants.ObjectTypes.DocumentBlueprint;
var isMedia = objectTypeId == Constants.ObjectTypes.Media;
var isMember = objectTypeId == Constants.ObjectTypes.Member;
Sql sql = GetFullSqlForEntityType(isContent, isMedia, isMember, objectTypeId, key);
return GetEntity(sql, isContent, isMedia, isMember);
}
public IEntitySlim? Get(int id)
{
Sql sql = GetBaseWhere(false, false, false, false, id);
BaseDto? dto = Database.FirstOrDefault(sql);
return dto == null ? null : BuildEntity(dto);
}
public IEntitySlim? Get(int id, Guid objectTypeId)
{
var isContent = objectTypeId == Constants.ObjectTypes.Document ||
objectTypeId == Constants.ObjectTypes.DocumentBlueprint;
var isMedia = objectTypeId == Constants.ObjectTypes.Media;
var isMember = objectTypeId == Constants.ObjectTypes.Member;
Sql sql = GetFullSqlForEntityType(isContent, isMedia, isMember, objectTypeId, id);
return GetEntity(sql, isContent, isMedia, isMember);
}
public IEnumerable GetAll(Guid objectType, params int[] ids) =>
ids.Length > 0
? PerformGetAll(objectType, sql => sql.WhereIn(x => x.NodeId, ids.Distinct()))
: PerformGetAll(objectType);
public IEnumerable GetAll(Guid objectType, params Guid[] keys) =>
keys.Length > 0
? PerformGetAll(objectType, sql => sql.WhereIn(x => x.UniqueId, keys.Distinct()))
: PerformGetAll(objectType);
private IEnumerable GetEntities(Sql sql, bool isContent, bool isMedia, bool isMember)
{
// isContent is going to return a 1:M result now with the variants so we need to do different things
if (isContent)
{
List? cdtos = Database.Fetch(sql);
return cdtos.Count == 0
? Enumerable.Empty()
: BuildVariants(cdtos.Select(BuildDocumentEntity)).ToList();
}
IEnumerable? dtos = isMedia
? (IEnumerable)Database.Fetch(sql)
: Database.Fetch(sql);
EntitySlim[] entities = dtos.Select(BuildEntity).ToArray();
return entities;
}
private IEnumerable PerformGetAll(Guid objectType, Action>? filter = null)
{
var isContent = objectType == Constants.ObjectTypes.Document ||
objectType == Constants.ObjectTypes.DocumentBlueprint;
var isMedia = objectType == Constants.ObjectTypes.Media;
var isMember = objectType == Constants.ObjectTypes.Member;
Sql sql = GetFullSqlForEntityType(isContent, isMedia, isMember, objectType, filter);
return GetEntities(sql, isContent, isMedia, isMember);
}
public IEnumerable GetAllPaths(Guid objectType, params int[]? ids) =>
ids?.Any() ?? false
? PerformGetAllPaths(objectType, sql => sql.WhereIn(x => x.NodeId, ids.Distinct()))
: PerformGetAllPaths(objectType);
public IEnumerable GetAllPaths(Guid objectType, params Guid[] keys) =>
keys.Any()
? PerformGetAllPaths(objectType, sql => sql.WhereIn(x => x.UniqueId, keys.Distinct()))
: PerformGetAllPaths(objectType);
private IEnumerable PerformGetAllPaths(Guid objectType, Action>? filter = null)
{
// NodeId is named Id on TreeEntityPath = use an alias
Sql sql = Sql().Select(x => Alias(x.NodeId, nameof(TreeEntityPath.Id)), x => x.Path)
.From().Where(x => x.NodeObjectType == objectType);
filter?.Invoke(sql);
return Database.Fetch(sql);
}
public IEnumerable GetByQuery(IQuery query)
{
Sql sqlClause = GetBase(false, false, false, null);
var translator = new SqlTranslator(sqlClause, query);
Sql sql = translator.Translate();
sql = AddGroupBy(false, false, false, sql, true);
List? dtos = Database.Fetch(sql);
return dtos.Select(BuildEntity).ToList();
}
public IEnumerable GetByQuery(IQuery query, Guid objectType)
{
var isContent = objectType == Constants.ObjectTypes.Document ||
objectType == Constants.ObjectTypes.DocumentBlueprint;
var isMedia = objectType == Constants.ObjectTypes.Media;
var isMember = objectType == Constants.ObjectTypes.Member;
Sql sql = GetBaseWhere(isContent, isMedia, isMember, false, null, new[] {objectType});
var translator = new SqlTranslator(sql, query);
sql = translator.Translate();
sql = AddGroupBy(isContent, isMedia, isMember, sql, true);
return GetEntities(sql, isContent, isMedia, isMember);
}
public UmbracoObjectTypes GetObjectType(int id)
{
Sql sql = Sql().Select(x => x.NodeObjectType).From()
.Where(x => x.NodeId == id);
return ObjectTypes.GetUmbracoObjectType(Database.ExecuteScalar(sql));
}
public UmbracoObjectTypes GetObjectType(Guid key)
{
Sql sql = Sql().Select(x => x.NodeObjectType).From()
.Where(x => x.UniqueId == key);
return ObjectTypes.GetUmbracoObjectType(Database.ExecuteScalar(sql));
}
public int ReserveId(Guid key)
{
NodeDto node;
Sql sql = SqlContext.Sql()
.Select()
.From()
.Where(x => x.UniqueId == key && x.NodeObjectType == Constants.ObjectTypes.IdReservation);
node = Database.SingleOrDefault(sql);
if (node != null)
{
throw new InvalidOperationException("An identifier has already been reserved for this Udi.");
}
node = new NodeDto
{
UniqueId = key,
Text = "RESERVED.ID",
NodeObjectType = Constants.ObjectTypes.IdReservation,
CreateDate = DateTime.Now,
UserId = null,
ParentId = -1,
Level = 1,
Path = "-1",
SortOrder = 0,
Trashed = false
};
Database.Insert(node);
return node.NodeId;
}
public bool Exists(Guid key)
{
Sql sql = Sql().SelectCount().From().Where(x => x.UniqueId == key);
return Database.ExecuteScalar(sql) > 0;
}
public bool Exists(IEnumerable keys)
{
var distictKeys = keys.Distinct();
Sql sql = Sql().SelectCount().From().Where(x => distictKeys.Contains(x.UniqueId));
return Database.ExecuteScalar(sql) == distictKeys.Count();
}
///
public bool Exists(Guid key, Guid objectType)
{
Sql sql = Sql()
.SelectCount()
.From()
.Where(x => x.UniqueId == key && x.NodeObjectType == objectType);
return Database.ExecuteScalar(sql) > 0;
}
public bool Exists(int id, Guid objectType)
{
Sql sql = Sql()
.SelectCount()
.From()
.Where(x => x.NodeId == id && x.NodeObjectType == objectType);
return Database.ExecuteScalar(sql) > 0;
}
public bool Exists(int id)
{
Sql sql = Sql().SelectCount().From().Where(x => x.NodeId == id);
return Database.ExecuteScalar(sql) > 0;
}
private DocumentEntitySlim BuildVariants(DocumentEntitySlim entity)
=> BuildVariants(new[] {entity}).First();
private IEnumerable BuildVariants(IEnumerable entities)
{
List? v = null;
var entitiesList = entities.ToList();
foreach (DocumentEntitySlim e in entitiesList)
{
if (e.Variations.VariesByCulture())
{
(v ?? (v = new List())).Add(e);
}
}
if (v == null)
{
return entitiesList;
}
// fetch all variant info dtos
IEnumerable dtos = Database.FetchByGroups(v.Select(x => x.Id),
Constants.Sql.MaxParameterCount, GetVariantInfos);
// group by node id (each group contains all languages)
var xdtos = dtos.GroupBy(x => x.NodeId).ToDictionary(x => x.Key, x => x);
foreach (DocumentEntitySlim e in v)
{
// since we're only iterating on entities that vary, we must have something
IGrouping edtos = xdtos[e.Id];
e.CultureNames = edtos.Where(x => x.CultureAvailable).ToDictionary(x => x.IsoCode, x => x.Name);
e.PublishedCultures = edtos.Where(x => x.CulturePublished).Select(x => x.IsoCode);
e.EditedCultures = edtos.Where(x => x.CultureAvailable && x.CultureEdited).Select(x => x.IsoCode);
}
return entitiesList;
}
#endregion
#region Sql
protected Sql GetVariantInfos(IEnumerable ids) =>
Sql()
.Select(x => x.NodeId)
.AndSelect(x => x.IsoCode)
.AndSelect("doc", x => Alias(x.Published, "DocumentPublished"),
x => Alias(x.Edited, "DocumentEdited"))
.AndSelect("dcv",
x => Alias(x.Available, "CultureAvailable"), x => Alias(x.Published, "CulturePublished"),
x => Alias(x.Edited, "CultureEdited"),
x => Alias(x.Name, "Name"))
// from node x language
.From()
.CrossJoin()
// join to document - always exists - indicates global document published/edited status
.InnerJoin("doc")
.On((node, doc) => node.NodeId == doc.NodeId, aliasRight: "doc")
// left-join do document variation - matches cultures that are *available* + indicates when *edited*
.LeftJoin("dcv")
.On(
(node, dcv, lang) => node.NodeId == dcv.NodeId && lang.Id == dcv.LanguageId, aliasRight: "dcv")
// for selected nodes
.WhereIn(x => x.NodeId, ids)
.OrderBy(x => x.Id);
// gets the full sql for a given object type and a given unique id
protected Sql GetFullSqlForEntityType(bool isContent, bool isMedia, bool isMember, Guid objectType,
Guid uniqueId)
{
Sql sql = GetBaseWhere(isContent, isMedia, isMember, false, objectType, uniqueId);
return AddGroupBy(isContent, isMedia, isMember, sql, true);
}
// gets the full sql for a given object type and a given node id
protected Sql GetFullSqlForEntityType(bool isContent, bool isMedia, bool isMember, Guid objectType,
int nodeId)
{
Sql sql = GetBaseWhere(isContent, isMedia, isMember, false, objectType, nodeId);
return AddGroupBy(isContent, isMedia, isMember, sql, true);
}
// gets the full sql for a given object type, with a given filter
protected Sql GetFullSqlForEntityType(bool isContent, bool isMedia, bool isMember, Guid objectType,
Action>? filter)
{
Sql sql = GetBaseWhere(isContent, isMedia, isMember, false, filter, new[] {objectType});
return AddGroupBy(isContent, isMedia, isMember, sql, true);
}
// gets the base SELECT + FROM [+ filter] sql
// always from the 'current' content version
protected Sql GetBase(bool isContent, bool isMedia, bool isMember, Action>? filter,
bool isCount = false)
{
Sql sql = Sql();
if (isCount)
{
sql.SelectCount();
}
else
{
sql
.Select(x => x.NodeId, x => x.Trashed, x => x.ParentId, x => x.UserId, x => x.Level,
x => x.Path)
.AndSelect(x => x.SortOrder, x => x.UniqueId, x => x.Text, x => x.NodeObjectType,
x => x.CreateDate)
.Append(", COUNT(child.id) AS children");
if (isContent || isMedia || isMember)
{
sql
.AndSelect(x => Alias(x.Id, "versionId"), x => x.VersionDate)
.AndSelect(
x => x.Alias,
x => x.Icon,
x => x.Thumbnail,
x => x.ListView,
x => x.Variations)
.AndSelect("ContentTypeNode", x => Alias(x.UniqueId, "ContentTypeKey"));
}
if (isContent)
{
sql
.AndSelect(x => x.Published, x => x.Edited);
}
if (isMedia)
{
sql
.AndSelect(x => Alias(x.Path, "MediaPath"));
}
}
sql
.From();
if (isContent || isMedia || isMember)
{
sql
.LeftJoin()
.On((left, right) => left.NodeId == right.NodeId && right.Current)
.LeftJoin().On((left, right) => left.NodeId == right.NodeId)
.LeftJoin()
.On((left, right) => left.ContentTypeId == right.NodeId)
.LeftJoin("ContentTypeNode")
.On((left, right) => left.NodeId == right.NodeId, aliasLeft: "ContentTypeNode");
}
if (isContent)
{
sql
.LeftJoin().On((left, right) => left.NodeId == right.NodeId);
}
if (isMedia)
{
sql
.LeftJoin()
.On((left, right) => left.Id == right.Id);
}
//Any LeftJoin statements need to come last
if (isCount == false)
{
sql
.LeftJoin("child")
.On((left, right) => left.NodeId == right.ParentId, aliasRight: "child");
}
filter?.Invoke(sql);
return sql;
}
// gets the base SELECT + FROM [+ filter] + WHERE sql
// for a given object type, with a given filter
protected Sql GetBaseWhere(bool isContent, bool isMedia, bool isMember, bool isCount,
Action>? filter, Guid[] objectTypes)
{
Sql sql = GetBase(isContent, isMedia, isMember, filter, isCount);
if (objectTypes.Length > 0)
{
sql.WhereIn(x => x.NodeObjectType, objectTypes);
}
return sql;
}
// gets the base SELECT + FROM + WHERE sql
// for a given node id
protected Sql GetBaseWhere(bool isContent, bool isMedia, bool isMember, bool isCount, int id)
{
Sql sql = GetBase(isContent, isMedia, isMember, null, isCount)
.Where(x => x.NodeId == id);
return AddGroupBy(isContent, isMedia, isMember, sql, true);
}
// gets the base SELECT + FROM + WHERE sql
// for a given unique id
protected Sql GetBaseWhere(bool isContent, bool isMedia, bool isMember, bool isCount, Guid uniqueId)
{
Sql sql = GetBase(isContent, isMedia, isMember, null, isCount)
.Where(x => x.UniqueId == uniqueId);
return AddGroupBy(isContent, isMedia, isMember, sql, true);
}
// gets the base SELECT + FROM + WHERE sql
// for a given object type and node id
protected Sql GetBaseWhere(bool isContent, bool isMedia, bool isMember, bool isCount, Guid objectType,
int nodeId) =>
GetBase(isContent, isMedia, isMember, null, isCount)
.Where(x => x.NodeId == nodeId && x.NodeObjectType == objectType);
// gets the base SELECT + FROM + WHERE sql
// for a given object type and unique id
protected Sql GetBaseWhere(bool isContent, bool isMedia, bool isMember, bool isCount, Guid objectType,
Guid uniqueId) =>
GetBase(isContent, isMedia, isMember, null, isCount)
.Where(x => x.UniqueId == uniqueId && x.NodeObjectType == objectType);
// gets the GROUP BY / ORDER BY sql
// required in order to count children
protected Sql AddGroupBy(bool isContent, bool isMedia, bool isMember, Sql sql,
bool defaultSort)
{
sql
.GroupBy(x => x.NodeId, x => x.Trashed, x => x.ParentId, x => x.UserId, x => x.Level, x => x.Path)
.AndBy(x => x.SortOrder, x => x.UniqueId, x => x.Text, x => x.NodeObjectType, x => x.CreateDate);
if (isContent)
{
sql
.AndBy(x => x.Published, x => x.Edited);
}
if (isMedia)
{
sql
.AndBy(x => Alias(x.Path, "MediaPath"));
}
if (isContent || isMedia || isMember)
{
sql
.AndBy(x => x.Id, x => x.VersionDate)
.AndBy(
x => x.Alias,
x => x.Icon,
x => x.Thumbnail,
x => x.ListView,
x => x.Variations)
.AndBy("ContentTypeNode", x => x.UniqueId);
}
if (defaultSort)
{
sql.OrderBy(x => x.SortOrder);
}
return sql;
}
private void ApplyOrdering(ref Sql sql, Ordering ordering)
{
if (sql == null)
{
throw new ArgumentNullException(nameof(sql));
}
if (ordering == null)
{
throw new ArgumentNullException(nameof(ordering));
}
// TODO: although the default ordering string works for name, it wont work for others without a table or an alias of some sort
// As more things are attempted to be sorted we'll prob have to add more expressions here
string orderBy;
Ordering? runner = ordering;
do
{
switch (runner.OrderBy?.ToUpperInvariant())
{
case "NODEOBJECTTYPE":
orderBy = $"UPPER({SqlSyntax.GetQuotedColumn(NodeDto.TableName, "nodeObjectType")})";
break;
case "PATH":
orderBy = SqlSyntax.GetQuotedColumn(NodeDto.TableName, "path");
break;
default:
orderBy = runner.OrderBy ?? string.Empty;
break;
}
if (runner.Direction == Direction.Ascending)
{
sql.OrderBy(orderBy);
}
else
{
sql.OrderByDescending(orderBy);
}
runner = runner.Next;
}
while (runner is not null);
}
#endregion
#region Classes
///
/// The DTO used to fetch results for a generic content item which could be either a document, media or a member
///
private class GenericContentEntityDto : DocumentEntityDto
{
public string? MediaPath { get; set; }
}
///
/// The DTO used to fetch results for a document item with its variation info
///
private class DocumentEntityDto : BaseDto
{
public ContentVariation Variations { get; set; }
public bool Published { get; set; }
public bool Edited { get; set; }
}
///
/// The DTO used to fetch results for a media item with its media path info
///
private class MediaEntityDto : BaseDto
{
public string? MediaPath { get; set; }
}
///
/// The DTO used to fetch results for a member item
///
private class MemberEntityDto : BaseDto
{
}
public class VariantInfoDto
{
public int NodeId { get; set; }
public string IsoCode { get; set; } = null!;
public string Name { get; set; } = null!;
public bool DocumentPublished { get; set; }
public bool DocumentEdited { get; set; }
public bool CultureAvailable { get; set; }
public bool CulturePublished { get; set; }
public bool CultureEdited { get; set; }
}
// ReSharper disable once ClassNeverInstantiated.Local
///
/// the DTO corresponding to fields selected by GetBase
///
private class BaseDto
{
// ReSharper disable UnusedAutoPropertyAccessor.Local
// ReSharper disable UnusedMember.Local
public int NodeId { get; set; }
public bool Trashed { get; set; }
public int ParentId { get; set; }
public int? UserId { get; set; }
public int Level { get; set; }
public string Path { get; } = null!;
public int SortOrder { get; set; }
public Guid UniqueId { get; set; }
public string? Text { get; set; }
public Guid NodeObjectType { get; set; }
public DateTime CreateDate { get; set; }
public DateTime VersionDate { get; set; }
public int Children { get; set; }
public int VersionId { get; set; }
public string Alias { get; } = null!;
public string? Icon { get; set; }
public string? Thumbnail { get; set; }
public bool IsContainer { get; set; }
public Guid ContentTypeKey { get; set; }
public Guid? ListView { get; set; }
// ReSharper restore UnusedAutoPropertyAccessor.Local
// ReSharper restore UnusedMember.Local
}
#endregion
#region Factory
private EntitySlim BuildEntity(BaseDto dto)
{
if (dto.NodeObjectType == Constants.ObjectTypes.Document
|| dto.NodeObjectType == Constants.ObjectTypes.DocumentBlueprint)
{
return BuildDocumentEntity(dto);
}
if (dto.NodeObjectType == Constants.ObjectTypes.Media)
{
return BuildMediaEntity(dto);
}
if (dto.NodeObjectType == Constants.ObjectTypes.Member)
{
return BuildMemberEntity(dto);
}
// EntitySlim does not track changes
var entity = new EntitySlim();
BuildEntity(entity, dto);
return entity;
}
private static void BuildEntity(EntitySlim entity, BaseDto dto)
{
entity.Trashed = dto.Trashed;
entity.CreateDate = dto.CreateDate;
entity.UpdateDate = dto.VersionDate;
entity.CreatorId = dto.UserId ?? Constants.Security.UnknownUserId;
entity.Id = dto.NodeId;
entity.Key = dto.UniqueId;
entity.Level = dto.Level;
entity.Name = dto.Text;
entity.NodeObjectType = dto.NodeObjectType;
entity.ParentId = dto.ParentId;
entity.Path = dto.Path;
entity.SortOrder = dto.SortOrder;
entity.HasChildren = dto.Children > 0;
entity.IsContainer = dto.IsContainer;
}
private static void BuildContentEntity(ContentEntitySlim entity, BaseDto dto)
{
BuildEntity(entity, dto);
entity.ContentTypeAlias = dto.Alias;
entity.ContentTypeIcon = dto.Icon;
entity.ContentTypeThumbnail = dto.Thumbnail;
entity.ContentTypeKey = dto.ContentTypeKey;
entity.ListViewKey = dto.ListView;
}
private MediaEntitySlim BuildMediaEntity(BaseDto dto)
{
// EntitySlim does not track changes
var entity = new MediaEntitySlim();
BuildContentEntity(entity, dto);
// fill in the media info
if (dto is MediaEntityDto mediaEntityDto)
{
entity.MediaPath = mediaEntityDto.MediaPath;
}
else if (dto is GenericContentEntityDto genericContentEntityDto)
{
entity.MediaPath = genericContentEntityDto.MediaPath;
}
return entity;
}
private DocumentEntitySlim BuildDocumentEntity(BaseDto dto)
{
// EntitySlim does not track changes
var entity = new DocumentEntitySlim();
BuildContentEntity(entity, dto);
if (dto is DocumentEntityDto contentDto)
{
// fill in the invariant info
entity.Edited = contentDto.Edited;
entity.Published = contentDto.Published;
entity.Variations = contentDto.Variations;
}
return entity;
}
private MemberEntitySlim BuildMemberEntity(BaseDto dto)
{
// EntitySlim does not track changes
var entity = new MemberEntitySlim();
BuildEntity(entity, dto);
entity.ContentTypeAlias = dto.Alias;
entity.ContentTypeIcon = dto.Icon;
entity.ContentTypeThumbnail = dto.Thumbnail;
return entity;
}
#endregion
}