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.
147 lines
5.6 KiB
147 lines
5.6 KiB
// ==========================================================================
|
|
// Squidex Headless CMS
|
|
// ==========================================================================
|
|
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|
// All rights reserved. Licensed under the MIT license.
|
|
// ==========================================================================
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using MongoDB.Driver;
|
|
using NodaTime;
|
|
using Squidex.Domain.Apps.Core.ConvertContent;
|
|
using Squidex.Domain.Apps.Entities.Apps;
|
|
using Squidex.Domain.Apps.Entities.Contents;
|
|
using Squidex.Domain.Apps.Entities.Contents.State;
|
|
using Squidex.Domain.Apps.Entities.MongoDb.Contents.Visitors;
|
|
using Squidex.Domain.Apps.Entities.Schemas;
|
|
using Squidex.Infrastructure;
|
|
using Squidex.Infrastructure.Json;
|
|
using Squidex.Infrastructure.MongoDb;
|
|
using Squidex.Infrastructure.Queries;
|
|
using Squidex.Infrastructure.Reflection;
|
|
using Squidex.Infrastructure.States;
|
|
|
|
namespace Squidex.Domain.Apps.Entities.MongoDb.Contents
|
|
{
|
|
internal sealed class MongoContentDraftCollection : MongoContentCollection
|
|
{
|
|
public MongoContentDraftCollection(IMongoDatabase database, IJsonSerializer serializer)
|
|
: base(database, serializer, "State_Content_Draft")
|
|
{
|
|
}
|
|
|
|
protected override async Task SetupCollectionAsync(IMongoCollection<MongoContentEntity> collection, CancellationToken ct = default(CancellationToken))
|
|
{
|
|
await collection.Indexes.CreateManyAsync(
|
|
new[]
|
|
{
|
|
new CreateIndexModel<MongoContentEntity>(
|
|
Index
|
|
.Ascending(x => x.IndexedSchemaId)
|
|
.Ascending(x => x.Id)
|
|
.Ascending(x => x.IsDeleted)),
|
|
new CreateIndexModel<MongoContentEntity>(
|
|
Index
|
|
.Text(x => x.DataText)
|
|
.Ascending(x => x.IndexedSchemaId)
|
|
.Ascending(x => x.IsDeleted)
|
|
.Ascending(x => x.Status)),
|
|
}, ct);
|
|
|
|
await base.SetupCollectionAsync(collection, ct);
|
|
}
|
|
|
|
public async Task<IReadOnlyList<Guid>> QueryIdsAsync(Guid appId, ISchemaEntity schema, FilterNode filterNode)
|
|
{
|
|
var filter = filterNode.AdjustToModel(schema.SchemaDef, true).ToFilter(schema.Id);
|
|
|
|
var contentEntities =
|
|
await Collection.Find(filter).Only(x => x.Id)
|
|
.ToListAsync();
|
|
|
|
return contentEntities.Select(x => Guid.Parse(x["_id"].AsString)).ToList();
|
|
}
|
|
|
|
public async Task<IReadOnlyList<Guid>> QueryIdsAsync(Guid appId)
|
|
{
|
|
var contentEntities =
|
|
await Collection.Find(x => x.IndexedAppId == appId).Only(x => x.Id)
|
|
.ToListAsync();
|
|
|
|
return contentEntities.Select(x => Guid.Parse(x["_id"].AsString)).ToList();
|
|
}
|
|
|
|
public Task QueryScheduledWithoutDataAsync(Instant now, Func<IContentEntity, Task> callback)
|
|
{
|
|
return Collection.Find(x => x.ScheduledAt < now && x.IsDeleted != true)
|
|
.Not(x => x.DataByIds)
|
|
.Not(x => x.DataDraftByIds)
|
|
.Not(x => x.DataText)
|
|
.ForEachAsync(c =>
|
|
{
|
|
callback(c);
|
|
});
|
|
}
|
|
|
|
public async Task<IContentEntity> FindContentAsync(IAppEntity app, ISchemaEntity schema, Guid id)
|
|
{
|
|
var contentEntity =
|
|
await Collection.Find(x => x.IndexedSchemaId == schema.Id && x.Id == id && x.IsDeleted != true).Not(x => x.DataText)
|
|
.FirstOrDefaultAsync();
|
|
|
|
contentEntity?.ParseData(schema.SchemaDef, Serializer);
|
|
|
|
return contentEntity;
|
|
}
|
|
|
|
public async Task<(ContentState Value, long Version)> ReadAsync(Guid key, Func<Guid, Guid, Task<ISchemaEntity>> getSchema)
|
|
{
|
|
var contentEntity =
|
|
await Collection.Find(x => x.Id == key).Not(x => x.DataText)
|
|
.FirstOrDefaultAsync();
|
|
|
|
if (contentEntity != null)
|
|
{
|
|
var schema = await getSchema(contentEntity.IndexedAppId, contentEntity.IndexedSchemaId);
|
|
|
|
contentEntity.ParseData(schema.SchemaDef, Serializer);
|
|
|
|
return (SimpleMapper.Map(contentEntity, new ContentState()), contentEntity.Version);
|
|
}
|
|
|
|
return (null, EtagVersion.NotFound);
|
|
}
|
|
|
|
public async Task UpsertAsync(MongoContentEntity content, long oldVersion)
|
|
{
|
|
try
|
|
{
|
|
content.DataText = content.DataDraftByIds.ToFullText();
|
|
|
|
await Collection.ReplaceOneAsync(x => x.Id == content.Id && x.Version == oldVersion, content, Upsert);
|
|
}
|
|
catch (MongoWriteException ex)
|
|
{
|
|
if (ex.WriteError.Category == ServerErrorCategory.DuplicateKey)
|
|
{
|
|
var existingVersion =
|
|
await Collection.Find(x => x.Id == content.Id).Only(x => x.Id, x => x.Version)
|
|
.FirstOrDefaultAsync();
|
|
|
|
if (existingVersion != null)
|
|
{
|
|
throw new InconsistentStateException(existingVersion["vs"].AsInt64, oldVersion, ex);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|