File: ImmutableArrayExtensionsTests.cs
Web Access
Project: src\src\Razor\src\Shared\Microsoft.AspNetCore.Razor.Utilities.Shared.UnitTests\Microsoft.AspNetCore.Razor.Utilities.Shared.UnitTests.csproj (Microsoft.AspNetCore.Razor.Utilities.Shared.UnitTests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Runtime.InteropServices;
using Xunit;
 
namespace Microsoft.AspNetCore.Razor.Utilities.Shared.Test;
 
public class ImmutableArrayExtensionsTests
{
    [Fact]
    public void GetMostRecentUniqueItems()
    {
        ImmutableArray<string> items =
        [
            "Hello",
            "HELLO",
            "HeLlO",
            new string([',', ' ']),
            new string([',', ' ']),
            "World",
            "WORLD",
            "WoRlD"
        ];
 
        var mostRecent = items.GetMostRecentUniqueItems(StringComparer.OrdinalIgnoreCase);
 
        Assert.Collection(mostRecent,
            s => Assert.Equal("HeLlO", s),
            s =>
            {
                // make sure it's the most recent ", "
                Assert.NotSame(items[3], s);
                Assert.Same(items[4], s);
            },
            s => Assert.Equal("WoRlD", s));
    }
 
    [Fact]
    public void SelectAsArray()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [2, 4, 6, 8, 10, 12, 14, 16, 18, 20];
 
        var actual = data.SelectAsArray(static x => x * 2);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void SelectAsArray_ReadOnlyList()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [2, 4, 6, 8, 10, 12, 14, 16, 18, 20];
 
        var list = (IReadOnlyList<int>)data;
 
        var actual = list.SelectAsArray(static x => x * 2);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void SelectAsArray_Enumerable()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [2, 4, 6, 8, 10, 12, 14, 16, 18, 20];
 
        var enumerable = (IEnumerable<int>)data;
 
        var actual = enumerable.SelectAsArray(static x => x * 2);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void SelectAsArray_Index()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [1, 3, 5, 7, 9, 11, 13, 15, 17, 19];
 
        var actual = data.SelectAsArray(static (x, index) => x + index);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void SelectAsArray_Index_ReadOnlyList()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [1, 3, 5, 7, 9, 11, 13, 15, 17, 19];
 
        var list = (IReadOnlyList<int>)data;
 
        var actual = list.SelectAsArray(static (x, index) => x + index);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void SelectAsArray_Index_Enumerable()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [1, 3, 5, 7, 9, 11, 13, 15, 17, 19];
 
        var enumerable = (IEnumerable<int>)data;
 
        var actual = enumerable.SelectAsArray(static (x, index) => x + index);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void InsertRange_EmptySpan_DoesNotModifyBuilder()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        builder.Add(2);
        var originalCount = builder.Count;
        
        // Act
        builder.InsertRange(1, ReadOnlySpan<int>.Empty);
        
        // Assert
        Assert.Equal(originalCount, builder.Count);
        Assert.Equal(1, builder[0]);
        Assert.Equal(2, builder[1]);
    }
    
    [Fact]
    public void InsertRange_AtEndOfBuilder_AppendsItems()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        builder.Add(2);
        var itemsToInsert = new[] { 3, 4, 5 };
        
        // Act
        builder.InsertRange(builder.Count, itemsToInsert.AsSpan());
        
        // Assert
        Assert.Equal(5, builder.Count);
        Assert.Equal([1, 2, 3, 4, 5], builder.ToArray());
    }
    
    [Fact]
    public void InsertRange_SingleItem_InsertsCorrectly()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        builder.Add(3);
        var itemToInsert = new[] { 2 };
        
        // Act
        builder.InsertRange(1, itemToInsert.AsSpan());
        
        // Assert
        Assert.Equal(3, builder.Count);
        Assert.Equal([1, 2, 3], builder.ToArray());
    }
    
    [Fact]
    public void InsertRange_MultipleItems_InsertsCorrectly()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        builder.Add(6);
        var itemsToInsert = new[] { 2, 3, 4, 5 };
        
        // Act
        builder.InsertRange(1, itemsToInsert.AsSpan());
        
        // Assert
        Assert.Equal(6, builder.Count);
        Assert.Equal([1, 2, 3, 4, 5, 6], builder.ToArray());
    }
    
    [Fact]
    public void InsertRange_AtBeginning_InsertsCorrectly()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(3);
        builder.Add(4);
        var itemsToInsert = new[] { 1, 2 };
        
        // Act
        builder.InsertRange(0, itemsToInsert.AsSpan());
        
        // Assert
        Assert.Equal(4, builder.Count);
        Assert.Equal([1, 2, 3, 4], builder.ToArray());
    }
    
    [Fact]
    public void InsertRange_NegativeIndex_ThrowsArgumentException()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        var itemsToInsert = new[] { 2, 3 };
        
        // Act & Assert
        Assert.Throws<ArgumentOutOfRangeException>(() => builder.InsertRange(-1, itemsToInsert.AsSpan()));
    }
    
    [Fact]
    public void InsertRange_IndexGreaterThanCount_ThrowsArgumentException()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<int>();
        builder.Add(1);
        var itemsToInsert = new[] { 2, 3 };
        
        // Act & Assert
        Assert.Throws<ArgumentOutOfRangeException>(() => builder.InsertRange(builder.Count + 1, itemsToInsert.AsSpan()));
    }
 
    [Fact]
    public void InsertRange_WithReferenceTypes_InsertsCorrectly()
    {
        // Arrange
        var builder = ImmutableArray.CreateBuilder<string>();
        builder.Add("apple");
        builder.Add("banana");
        var itemsToInsert = new[] { "cherry", "date" };
        
        // Act
        builder.InsertRange(1, itemsToInsert.AsSpan());
        
        // Assert
        Assert.Equal(4, builder.Count);
        Assert.Equal(["apple", "cherry", "date", "banana"], builder.ToArray());
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [2, 4, 6, 8, 10];
 
        var actual = data.WhereAsArray(static x => x % 2 == 0);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_None()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [];
 
        var actual = data.WhereAsArray(static x => false);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_All()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        var expected = data;
 
        var actual = data.WhereAsArray(static x => true);
        Assert.Equal<int>(expected, actual);
        Assert.Same(ImmutableCollectionsMarshal.AsArray(expected), ImmutableCollectionsMarshal.AsArray(actual));
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_Empty()
    {
        ImmutableArray<int> data = [];
        ImmutableArray<int> expected = [];
 
        var actual = data.WhereAsArray(static x => x > 0);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithIndex()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [1, 3, 5, 7, 9]; // Even indices (0, 2, 4, 6, 8)
 
        var actual = data.WhereAsArray(static (x, index) => index % 2 == 0);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithIndex_None()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [];
 
        var actual = data.WhereAsArray(static (x, index) => false);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithIndex_All()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        var expected = data;
 
        var actual = data.WhereAsArray(static (x, index) => true);
        Assert.Equal<int>(expected, actual);
        Assert.Same(ImmutableCollectionsMarshal.AsArray(expected), ImmutableCollectionsMarshal.AsArray(actual));
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArg()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [5, 6, 7, 8, 9, 10];
        var threshold = 5;
 
        var actual = data.WhereAsArray(threshold, static (x, arg) => x >= arg);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArg_None()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [];
        var threshold = 15;
 
        var actual = data.WhereAsArray(threshold, static (x, arg) => x >= arg);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArg_All()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        var expected = data;
        var threshold = 0;
 
        var actual = data.WhereAsArray(threshold, static (x, arg) => x >= arg);
        Assert.Equal<int>(expected, actual);
        Assert.Same(ImmutableCollectionsMarshal.AsArray(expected), ImmutableCollectionsMarshal.AsArray(actual));
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArgAndIndex()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [4, 6, 8, 10]; // Values at odd indices (1, 3, 5, 7, 9) where value >= 3
        var threshold = 3;
 
        var actual = data.WhereAsArray(threshold, static (x, arg, index) => index % 2 == 1 && x >= arg);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArgAndIndex_None()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        ImmutableArray<int> expected = [];
        var threshold = 0;
 
        var actual = data.WhereAsArray(threshold, static (x, arg, index) => false);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithArgAndIndex_All()
    {
        ImmutableArray<int> data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        var expected = data;
        var threshold = 0;
 
        var actual = data.WhereAsArray(threshold, static (x, arg, index) => true);
        Assert.Equal<int>(expected, actual);
        Assert.Same(ImmutableCollectionsMarshal.AsArray(expected), ImmutableCollectionsMarshal.AsArray(actual));
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_WithReferenceTypes()
    {
        ImmutableArray<string> data = ["apple", "banana", "cherry", "date", "elderberry"];
        ImmutableArray<string> expected = ["banana", "cherry", "elderberry"];
 
        var actual = data.WhereAsArray(static x => x.Length > 5);
        Assert.Equal<string>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_OptimizationTest()
    {
        // Test that the optimization works correctly when some items match and some don't
        ImmutableArray<int> data = [1, 3, 2, 5, 4, 7, 6, 9, 8, 10];
        ImmutableArray<int> expected = [2, 4, 6, 8, 10];
 
        var actual = data.WhereAsArray(static x => x % 2 == 0);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_FirstItemDoesNotMatch()
    {
        // Test optimization when the first item doesn't match the predicate
        ImmutableArray<int> data = [1, 2, 4, 6, 8];
        ImmutableArray<int> expected = [2, 4, 6, 8];
 
        var actual = data.WhereAsArray(static x => x % 2 == 0);
        Assert.Equal<int>(expected, actual);
    }
 
    [Fact]
    public void WhereAsArray_ImmutableArray_LastItemDoesNotMatch()
    {
        // Test optimization when the last item doesn't match the predicate
        ImmutableArray<int> data = [2, 4, 6, 8, 9];
        ImmutableArray<int> expected = [2, 4, 6, 8];
 
        var actual = data.WhereAsArray(static x => x % 2 == 0);
        Assert.Equal<int>(expected, actual);
    }
}