// ========================================================================== // Squidex Headless CMS // ========================================================================== // Copyright (c) Squidex UG (haftungsbeschraenkt) // All rights reserved. Licensed under the MIT license. // ========================================================================== using System; using System.IO; using System.Security.Claims; using System.Text; using System.Threading; using System.Threading.Tasks; using FakeItEasy; using Microsoft.Extensions.Caching.Memory; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Options; using Squidex.Assets; using Squidex.Domain.Apps.Core.Contents; using Squidex.Domain.Apps.Core.Rules.EnrichedEvents; using Squidex.Domain.Apps.Core.Scripting; using Squidex.Domain.Apps.Core.TestHelpers; using Squidex.Domain.Apps.Entities.TestHelpers; using Squidex.Infrastructure; using Squidex.Infrastructure.Json.Objects; using Xunit; namespace Squidex.Domain.Apps.Entities.Assets { public class AssetsJintExtensionTests : IClassFixture { private readonly IAssetQueryService assetQuery = A.Fake(); private readonly IAssetFileStore assetFileStore = A.Fake(); private readonly IAppProvider appProvider = A.Fake(); private readonly NamedId appId = NamedId.Of(DomainId.NewGuid(), "my-app"); private readonly JintScriptEngine sut; public AssetsJintExtensionTests() { var services = new ServiceCollection() .AddSingleton(appProvider) .AddSingleton(assetQuery) .AddSingleton(assetFileStore) .BuildServiceProvider(); var extensions = new IJintExtension[] { new AssetsJintExtension(services) }; A.CallTo(() => appProvider.GetAppAsync(appId.Id, false)) .Returns(Mocks.App(appId)); sut = new JintScriptEngine(new MemoryCache(Options.Create(new MemoryCacheOptions())), extensions) { TimeoutScript = TimeSpan.FromSeconds(2), TimeoutExecution = TimeSpan.FromSeconds(10) }; } [Fact] public async Task Should_resolve_asset() { var (vars, asset) = SetupAssetVars(); var script = @" getAsset(data.assets.iv[0], function (assets) { var result1 = `Text: ${assets[0].fileName} ${assets[0].id}`; complete(`${result1}`); });"; var expected = $@" Text: {asset.FileName} {asset.Id} "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_assets() { var (vars, assets) = SetupAssetsVars(); var script = @" getAssets(data.assets.iv, function (assets) { var result1 = `Text: ${assets[0].fileName} ${assets[0].id}`; var result2 = `Text: ${assets[1].fileName} ${assets[1].id}`; complete(`${result1}\n${result2}`); });"; var expected = $@" Text: {assets[0].FileName} {assets[0].Id} Text: {assets[1].FileName} {assets[1].Id} "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_asset_text() { var (vars, asset) = SetupAssetVars(); SetupText(asset.Id, Encoding.UTF8.GetBytes("Hello Asset")); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }); });"; var expected = @" Text: Hello Asset "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_asset_text_with_utf8() { var (vars, asset) = SetupAssetVars(); SetupText(asset.Id, Encoding.UTF8.GetBytes("Hello Asset")); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }, 'utf8'); });"; var expected = @" Text: Hello Asset "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_asset_text_with_unicode() { var (vars, asset) = SetupAssetVars(); SetupText(asset.Id, Encoding.Unicode.GetBytes("Hello Asset")); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }, 'unicode'); });"; var expected = @" Text: Hello Asset "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_asset_text_with_ascii() { var (vars, asset) = SetupAssetVars(); SetupText(asset.Id, Encoding.ASCII.GetBytes("Hello Asset")); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }, 'ascii'); });"; var expected = @" Text: Hello Asset "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_resolve_asset_text_with_base64() { var (vars, asset) = SetupAssetVars(); SetupText(asset.Id, Encoding.UTF8.GetBytes("Hello Asset")); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }, 'base64'); });"; var expected = @" Text: SGVsbG8gQXNzZXQ= "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_not_resolve_asset_text_if_too_big() { var (vars, _) = SetupAssetVars(1_000_000); var script = @" getAssets(data.assets.iv, function (assets) { getAssetText(assets[0], function (text) { var result = `Text: ${text}`; complete(result); }); });"; var expected = @" Text: ErrorTooBig "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); A.CallTo(() => assetFileStore.DownloadAsync(A._, A._, A._, null, A._, A._, A._)) .MustNotHaveHappened(); } [Fact] public async Task Should_resolve_asset_text_from_event() { var @event = new EnrichedAssetEvent { Id = DomainId.NewGuid(), FileVersion = 0, FileSize = 100, AppId = appId }; SetupText(@event.Id, Encoding.UTF8.GetBytes("Hello Asset")); var vars = new ScriptVars { ["event"] = @event }; var script = @" getAssetText(event, function (text) { var result = `Text: ${text}`; complete(result); });"; var expected = @" Text: Hello Asset "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); } [Fact] public async Task Should_not_resolve_asset_text_from_event_if_too_big() { var @event = new EnrichedAssetEvent { Id = DomainId.NewGuid(), FileVersion = 0, FileSize = 1_000_000, AppId = appId }; var vars = new ScriptVars { ["event"] = @event }; var script = @" getAssetText(event, function (text) { var result = `Text: ${text}`; complete(result); });"; var expected = @" Text: ErrorTooBig "; var result = (await sut.ExecuteAsync(vars, script)).ToString(); Assert.Equal(Cleanup(expected), Cleanup(result)); A.CallTo(() => assetFileStore.DownloadAsync(A._, A._, A._, null, A._, A._, A._)) .MustNotHaveHappened(); } private void SetupText(DomainId id, byte[] bytes) { A.CallTo(() => assetFileStore.DownloadAsync(appId.Id, id, 0, null, A._, A._, A._)) .Invokes(x => { var stream = x.GetArgument(4)!; stream.Write(bytes); }); } private (ScriptVars, IAssetEntity) SetupAssetVars(int fileSize = 100) { var assetId = DomainId.NewGuid(); var asset = CreateAsset(assetId, 1, fileSize); var user = new ClaimsPrincipal(); var data = new ContentData() .AddField("assets", new ContentFieldData() .AddInvariant(JsonValue.Array(assetId))); A.CallTo(() => assetQuery.QueryAsync( A.That.Matches(x => x.App.Id == appId.Id && x.User == user), null, A.That.HasIds(assetId), A._)) .Returns(ResultList.CreateFrom(2, asset)); var vars = new ScriptVars { ["data"] = data, ["appId"] = appId.Id, ["appName"] = appId.Name, ["user"] = user }; return (vars, asset); } private (ScriptVars, IAssetEntity[]) SetupAssetsVars(int fileSize = 100) { var assetId1 = DomainId.NewGuid(); var asset1 = CreateAsset(assetId1, 1, fileSize); var assetId2 = DomainId.NewGuid(); var asset2 = CreateAsset(assetId1, 2, fileSize); var user = new ClaimsPrincipal(); var data = new ContentData() .AddField("assets", new ContentFieldData() .AddInvariant(JsonValue.Array(assetId1, assetId2))); A.CallTo(() => assetQuery.QueryAsync( A.That.Matches(x => x.App.Id == appId.Id && x.User == user), null, A.That.HasIds(assetId1, assetId2), A._)) .Returns(ResultList.CreateFrom(2, asset1, asset2)); var vars = new ScriptVars { ["data"] = data, ["appId"] = appId.Id, ["appName"] = appId.Name, ["user"] = user }; return (vars, new[] { asset1, asset2 }); } private IEnrichedAssetEntity CreateAsset(DomainId assetId, int index, int fileSize = 100) { return new AssetEntity { AppId = appId, Id = assetId, FileSize = fileSize, FileName = $"file{index}.jpg", }; } private static string Cleanup(string text) { return text .Replace("\r", string.Empty) .Replace("\n", string.Empty) .Replace(" ", string.Empty); } } }