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.
168 lines
6.1 KiB
168 lines
6.1 KiB
// ==========================================================================
|
|
// Squidex Headless CMS
|
|
// ==========================================================================
|
|
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|
// All rights reserved. Licensed under the MIT license.
|
|
// ==========================================================================
|
|
|
|
using System;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.Extensions.Caching.Memory;
|
|
using Microsoft.Extensions.Options;
|
|
using Squidex.Domain.Apps.Core;
|
|
using Squidex.Domain.Apps.Core.Schemas;
|
|
using Squidex.Domain.Apps.Entities.Schemas;
|
|
using Squidex.Domain.Apps.Entities.TestHelpers;
|
|
using Squidex.Infrastructure;
|
|
using Squidex.Infrastructure.Json.Objects;
|
|
using Squidex.Infrastructure.Queries;
|
|
using Squidex.Infrastructure.Validation;
|
|
using Xunit;
|
|
|
|
namespace Squidex.Domain.Apps.Entities.Contents.Queries
|
|
{
|
|
public class ContentQueryParserTests
|
|
{
|
|
private readonly ISchemaEntity schema;
|
|
private readonly NamedId<Guid> appId = NamedId.Of(Guid.NewGuid(), "my-app");
|
|
private readonly NamedId<Guid> schemaId = NamedId.Of(Guid.NewGuid(), "my-app");
|
|
private readonly Context requestContext;
|
|
private readonly ContentQueryParser sut;
|
|
|
|
public ContentQueryParserTests()
|
|
{
|
|
var options = Options.Create(new ContentOptions { DefaultPageSize = 30 });
|
|
|
|
requestContext = new Context(Mocks.FrontendUser(), Mocks.App(appId));
|
|
|
|
var schemaDef =
|
|
new Schema(schemaId.Name)
|
|
.AddString(1, "firstName", Partitioning.Invariant);
|
|
|
|
schema = Mocks.Schema(appId, schemaId, schemaDef);
|
|
|
|
var cache = new MemoryCache(Options.Create(new MemoryCacheOptions()));
|
|
|
|
sut = new ContentQueryParser(cache, JsonHelper.DefaultSerializer, options);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_use_existing_query()
|
|
{
|
|
var clrQuery = new ClrQuery();
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, Q.Empty.WithQuery(clrQuery));
|
|
|
|
Assert.Same(parsed, clrQuery);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_if_odata_query_is_invalid()
|
|
{
|
|
var query = Q.Empty.WithODataQuery("$filter=invalid");
|
|
|
|
await Assert.ThrowsAsync<ValidationException>(() => sut.ParseQueryAsync(requestContext, schema, query).AsTask());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_if_json_query_is_invalid()
|
|
{
|
|
var query = Q.Empty.WithJsonQuery("invalid");
|
|
|
|
await Assert.ThrowsAsync<ValidationException>(() => sut.ParseQueryAsync(requestContext, schema, query).AsTask());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_parse_odata_query()
|
|
{
|
|
var query = Q.Empty.WithODataQuery("$top=100&$orderby=data/firstName/iv asc&$search=Hello World");
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("FullText: 'Hello World'; Take: 100; Sort: data.firstName.iv Ascending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_parse_odata_query_and_enrich_with_defaults()
|
|
{
|
|
var query = Q.Empty.WithODataQuery("$top=200&$filter=data/firstName/iv eq 'ABC'");
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("Filter: data.firstName.iv == 'ABC'; Take: 200; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_parse_json_query_and_enrich_with_defaults()
|
|
{
|
|
var query = Q.Empty.WithJsonQuery(Json("{ 'filter': { 'path': 'data.firstName.iv', 'op': 'eq', 'value': 'ABC' } }"));
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("Filter: data.firstName.iv == 'ABC'; Take: 30; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_convert_json_query_and_enrich_with_defaults()
|
|
{
|
|
var query = Q.Empty.WithJsonQuery(
|
|
new Query<IJsonValue>
|
|
{
|
|
Filter = new CompareFilter<IJsonValue>("data.firstName.iv", CompareOperator.Equals, JsonValue.Create("ABC"))
|
|
});
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("Filter: data.firstName.iv == 'ABC'; Take: 30; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_parse_json_full_text_query_and_enrich_with_defaults()
|
|
{
|
|
var query = Q.Empty.WithJsonQuery(Json("{ 'fullText': 'Hello' }"));
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("FullText: 'Hello'; Take: 30; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_convert_json_full_text_query_and_enrich_with_defaults()
|
|
{
|
|
var query = Q.Empty.WithJsonQuery(
|
|
new Query<IJsonValue>
|
|
{
|
|
FullText = "Hello"
|
|
});
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("FullText: 'Hello'; Take: 30; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_apply_default_page_size()
|
|
{
|
|
var query = Q.Empty;
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("Take: 30; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_limit_number_of_contents()
|
|
{
|
|
var query = Q.Empty.WithODataQuery("$top=300&$skip=20");
|
|
|
|
var parsed = await sut.ParseQueryAsync(requestContext, schema, query);
|
|
|
|
Assert.Equal("Skip: 20; Take: 200; Sort: lastModified Descending", parsed.ToString());
|
|
}
|
|
|
|
private static string Json(string text)
|
|
{
|
|
return text.Replace('\'', '"');
|
|
}
|
|
}
|
|
}
|
|
|