mirror of https://github.com/Squidex/squidex.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
231 lines
8.7 KiB
231 lines
8.7 KiB
// ==========================================================================
|
|
// Squidex Headless CMS
|
|
// ==========================================================================
|
|
// Copyright (c) Squidex UG (haftungsbeschränkt)
|
|
// All rights reserved. Licensed under the MIT license.
|
|
// ==========================================================================
|
|
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using MongoDB.Driver;
|
|
using Squidex.Domain.Apps.Entities.Assets;
|
|
using Squidex.Domain.Apps.Entities.Assets.Repositories;
|
|
using Squidex.Domain.Apps.Entities.MongoDb.Assets.Visitors;
|
|
using Squidex.Infrastructure;
|
|
using Squidex.Infrastructure.MongoDb;
|
|
using Squidex.Infrastructure.MongoDb.Queries;
|
|
using Squidex.Infrastructure.Translations;
|
|
using Squidex.Log;
|
|
|
|
namespace Squidex.Domain.Apps.Entities.MongoDb.Assets
|
|
{
|
|
public sealed partial class MongoAssetRepository : MongoRepositoryBase<MongoAssetEntity>, IAssetRepository
|
|
{
|
|
public MongoAssetRepository(IMongoDatabase database)
|
|
: base(database)
|
|
{
|
|
}
|
|
|
|
public IMongoCollection<MongoAssetEntity> GetInternalCollection()
|
|
{
|
|
return Collection;
|
|
}
|
|
|
|
protected override string CollectionName()
|
|
{
|
|
return "States_Assets2";
|
|
}
|
|
|
|
protected override Task SetupCollectionAsync(IMongoCollection<MongoAssetEntity> collection, CancellationToken ct = default)
|
|
{
|
|
return collection.Indexes.CreateManyAsync(new[]
|
|
{
|
|
new CreateIndexModel<MongoAssetEntity>(
|
|
Index
|
|
.Ascending(x => x.IndexedAppId)
|
|
.Ascending(x => x.IsDeleted)
|
|
.Ascending(x => x.ParentId)
|
|
.Ascending(x => x.Tags)
|
|
.Descending(x => x.LastModified)),
|
|
new CreateIndexModel<MongoAssetEntity>(
|
|
Index
|
|
.Ascending(x => x.IndexedAppId)
|
|
.Ascending(x => x.IsDeleted)
|
|
.Ascending(x => x.Slug)),
|
|
new CreateIndexModel<MongoAssetEntity>(
|
|
Index
|
|
.Ascending(x => x.IndexedAppId)
|
|
.Ascending(x => x.IsDeleted)
|
|
.Ascending(x => x.FileHash)
|
|
.Ascending(x => x.FileName)
|
|
.Ascending(x => x.FileSize))
|
|
}, ct);
|
|
}
|
|
|
|
public async IAsyncEnumerable<IAssetEntity> StreamAll(DomainId appId)
|
|
{
|
|
var find = Collection.Find(x => x.IndexedAppId == appId && !x.IsDeleted);
|
|
|
|
using (var cursor = await find.ToCursorAsync())
|
|
{
|
|
while (await cursor.MoveNextAsync())
|
|
{
|
|
foreach (var entity in cursor.Current)
|
|
{
|
|
yield return entity;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task<IResultList<IAssetEntity>> QueryAsync(DomainId appId, DomainId? parentId, Q q)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>("QueryAsyncByQuery"))
|
|
{
|
|
try
|
|
{
|
|
if (q.Ids != null && q.Ids.Count > 0)
|
|
{
|
|
var filter = BuildFilter(appId, q.Ids.ToHashSet());
|
|
|
|
var assetEntities =
|
|
await Collection.Find(filter).SortByDescending(x => x.LastModified)
|
|
.QueryLimit(q.Query)
|
|
.QuerySkip(q.Query)
|
|
.ToListAsync();
|
|
long assetTotal = assetEntities.Count;
|
|
|
|
if (q.NoTotal)
|
|
{
|
|
assetTotal = -1;
|
|
}
|
|
else if (assetEntities.Count >= q.Query.Take || q.Query.Skip > 0)
|
|
{
|
|
assetTotal = await Collection.Find(filter).CountDocumentsAsync();
|
|
}
|
|
|
|
return ResultList.Create(assetTotal, assetEntities.OfType<IAssetEntity>());
|
|
}
|
|
else
|
|
{
|
|
var query = q.Query.AdjustToModel();
|
|
|
|
var filter = query.BuildFilter(appId, parentId);
|
|
|
|
var assetEntities =
|
|
await Collection.Find(filter)
|
|
.QueryLimit(query)
|
|
.QuerySkip(query)
|
|
.QuerySort(query)
|
|
.ToListAsync();
|
|
long assetTotal = assetEntities.Count;
|
|
|
|
if (q.NoTotal)
|
|
{
|
|
assetTotal = -1;
|
|
}
|
|
else if (assetEntities.Count >= q.Query.Take || q.Query.Skip > 0)
|
|
{
|
|
assetTotal = await Collection.Find(filter).CountDocumentsAsync();
|
|
}
|
|
|
|
return ResultList.Create<IAssetEntity>(assetTotal, assetEntities);
|
|
}
|
|
}
|
|
catch (MongoQueryException ex) when (ex.Message.Contains("17406"))
|
|
{
|
|
throw new DomainException(T.Get("common.resultTooLarge"));
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task<IReadOnlyList<DomainId>> QueryIdsAsync(DomainId appId, HashSet<DomainId> ids)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>("QueryAsyncByIds"))
|
|
{
|
|
var assetEntities =
|
|
await Collection.Find(BuildFilter(appId, ids)).Only(x => x.Id)
|
|
.ToListAsync();
|
|
|
|
var field = Field.Of<MongoAssetFolderEntity>(x => nameof(x.Id));
|
|
|
|
return assetEntities.Select(x => DomainId.Create(x[field].AsString)).ToList();
|
|
}
|
|
}
|
|
|
|
public async Task<IReadOnlyList<DomainId>> QueryChildIdsAsync(DomainId appId, DomainId parentId)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>())
|
|
{
|
|
var assetEntities =
|
|
await Collection.Find(x => x.IndexedAppId == appId && !x.IsDeleted && x.ParentId == parentId).Only(x => x.Id)
|
|
.ToListAsync();
|
|
|
|
var field = Field.Of<MongoAssetFolderEntity>(x => nameof(x.Id));
|
|
|
|
return assetEntities.Select(x => DomainId.Create(x[field].AsString)).ToList();
|
|
}
|
|
}
|
|
|
|
public async Task<IAssetEntity?> FindAssetAsync(DomainId appId, string hash, string fileName, long fileSize)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>())
|
|
{
|
|
var assetEntity =
|
|
await Collection.Find(x => x.IndexedAppId == appId && !x.IsDeleted && x.FileHash == hash && x.FileName == fileName && x.FileSize == fileSize)
|
|
.FirstOrDefaultAsync();
|
|
|
|
return assetEntity;
|
|
}
|
|
}
|
|
|
|
public async Task<IAssetEntity?> FindAssetBySlugAsync(DomainId appId, string slug)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>())
|
|
{
|
|
var assetEntity =
|
|
await Collection.Find(x => x.IndexedAppId == appId && !x.IsDeleted && x.Slug == slug)
|
|
.FirstOrDefaultAsync();
|
|
|
|
return assetEntity;
|
|
}
|
|
}
|
|
|
|
public async Task<IAssetEntity?> FindAssetAsync(DomainId appId, DomainId id)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>())
|
|
{
|
|
var documentId = DomainId.Combine(appId, id);
|
|
|
|
var assetEntity =
|
|
await Collection.Find(x => x.DocumentId == documentId && !x.IsDeleted)
|
|
.FirstOrDefaultAsync();
|
|
|
|
return assetEntity;
|
|
}
|
|
}
|
|
|
|
public async Task<IAssetEntity?> FindAssetAsync(DomainId id)
|
|
{
|
|
using (Profiler.TraceMethod<MongoAssetRepository>())
|
|
{
|
|
var assetEntity =
|
|
await Collection.Find(x => x.Id == id && !x.IsDeleted)
|
|
.FirstOrDefaultAsync();
|
|
|
|
return assetEntity;
|
|
}
|
|
}
|
|
|
|
private static FilterDefinition<MongoAssetEntity> BuildFilter(DomainId appId, HashSet<DomainId> ids)
|
|
{
|
|
var documentIds = ids.Select(x => DomainId.Combine(appId, x));
|
|
|
|
return Filter.And(
|
|
Filter.In(x => x.DocumentId, documentIds),
|
|
Filter.Ne(x => x.IsDeleted, true));
|
|
}
|
|
}
|
|
}
|
|
|