// ========================================================================== // Squidex Headless CMS // ========================================================================== // Copyright (c) Squidex UG (haftungsbeschraenkt) // All rights reserved. Licensed under the MIT license. // ========================================================================== using FakeItEasy; using Microsoft.Extensions.Options; using Squidex.Domain.Apps.Core.Tags; using Squidex.Domain.Apps.Core.TestHelpers; using Squidex.Domain.Apps.Entities.TestHelpers; using Squidex.Infrastructure; using Squidex.Infrastructure.Queries; using Squidex.Infrastructure.Validation; using Xunit; namespace Squidex.Domain.Apps.Entities.Assets.Queries { public class AssetQueryParserTests { private readonly ITagService tagService = A.Fake(); private readonly NamedId appId = NamedId.Of(DomainId.NewGuid(), "my-app"); private readonly Context requestContext; private readonly AssetQueryParser sut; public AssetQueryParserTests() { requestContext = new Context(Mocks.FrontendUser(), Mocks.App(appId)); var options = Options.Create(new AssetOptions { DefaultPageSize = 30 }); sut = new AssetQueryParser(TestUtils.DefaultSerializer, tagService, options); } [Fact] public async Task Should_skip_total_if_set_in_context() { var q = await sut.ParseAsync(requestContext.Clone(b => b.WithoutTotal()), Q.Empty); Assert.True(q.NoTotal); } [Fact] public async Task Should_throw_if_odata_query_is_invalid() { var query = Q.Empty.WithODataQuery("$filter=invalid"); await Assert.ThrowsAsync(() => sut.ParseAsync(requestContext, query)); } [Fact] public async Task Should_throw_if_json_query_is_invalid() { var query = Q.Empty.WithJsonQuery("invalid"); await Assert.ThrowsAsync(() => sut.ParseAsync(requestContext, query)); } [Fact] public async Task Should_parse_odata_query() { var query = Q.Empty.WithODataQuery("$top=100&$orderby=fileName asc&$search=Hello World"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("FullText: 'Hello World'; Take: 100; Sort: fileName Ascending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_parse_odata_query_and_enrich_with_defaults() { var query = Q.Empty.WithODataQuery("$top=200&$filter=fileName eq 'ABC'"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Filter: fileName == 'ABC'; Take: 200; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_parse_json_query_and_enrich_with_defaults() { var query = Q.Empty.WithJsonQuery("{ \"filter\": { \"path\": \"fileName\", \"op\": \"eq\", \"value\": \"ABC\" } }"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Filter: fileName == 'ABC'; Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_parse_json_full_text_query_and_enrich_with_defaults() { var query = Q.Empty.WithJsonQuery("{ \"fullText\": \"Hello\" }"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("FullText: 'Hello'; Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Theory] [InlineData(0L)] [InlineData(-1L)] [InlineData(long.MaxValue)] [InlineData(long.MinValue)] public async Task Should_apply_default_take_size_if_not_defined(long take) { var query = Q.Empty.WithQuery(new ClrQuery { Take = take }); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_set_take_to_ids_count_if_take_not_defined() { var query = Q.Empty.WithIds("1, 2, 3"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Take: 3; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_not_set_take_to_ids_count_if_take_defined() { var query = Q.Empty.WithIds("1, 2, 3").WithQuery(new ClrQuery { Take = 20 }); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Take: 20; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_apply_default_take_limit() { var query = Q.Empty.WithODataQuery("$top=300&$skip=20"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Skip: 20; Take: 200; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_not_apply_id_ordering_twice() { var query = Q.Empty.WithODataQuery("$top=300&$skip=20&$orderby=id desc"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Skip: 20; Take: 200; Sort: id Descending", q.Query.ToString()); } [Fact] public async Task Should_denormalize_tags() { A.CallTo(() => tagService.GetTagIdsAsync(appId.Id, TagGroups.Assets, A>.That.Contains("name1"), default)) .Returns(new Dictionary { ["name1"] = "id1" }); var query = Q.Empty.WithODataQuery("$filter=tags eq 'name1'"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Filter: tags == 'id1'; Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_not_fail_if_tags_not_found() { A.CallTo(() => tagService.GetTagIdsAsync(appId.Id, TagGroups.Assets, A>.That.Contains("name1"), default)) .Returns(new Dictionary()); var query = Q.Empty.WithODataQuery("$filter=tags eq 'name1'"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Filter: tags == 'name1'; Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); } [Fact] public async Task Should_not_normalize_other_field() { var query = Q.Empty.WithODataQuery("$filter=fileSize eq 123"); var q = await sut.ParseAsync(requestContext, query); Assert.Equal("Filter: fileSize == 123; Take: 30; Sort: lastModified Descending, id Ascending", q.Query.ToString()); A.CallTo(() => tagService.GetTagIdsAsync(appId.Id, A._, A>._, A._)) .MustNotHaveHappened(); } } }