silverstripe-framework/tests/php/Core/ArrayLibTest.php
Guy Sartorelli e2e32317d6
API Move various classes to more appropriate namespaces (#11370)
Also rename ViewableData to ModelData ahead of the template layer
lift-and-shift
2024-09-23 14:31:50 +12:00

540 lines
16 KiB
PHP

<?php
namespace SilverStripe\ORM\Tests;
use PHPUnit\Framework\ExpectationFailedException;
use SilverStripe\Core\ArrayLib;
use SilverStripe\Dev\SapphireTest;
use PHPUnit\Framework\Attributes\DataProvider;
class ArrayLibTest extends SapphireTest
{
public function testInvert()
{
$arr = [
'row1' => [
'col1' =>'val1',
'col2' => 'val2'
],
'row2' => [
'col1' => 'val3',
'col2' => 'val4'
]
];
$this->assertEquals(
ArrayLib::invert($arr),
[
'col1' => [
'row1' => 'val1',
'row2' => 'val3',
],
'col2' => [
'row1' => 'val2',
'row2' => 'val4',
],
]
);
}
public function testValuekey()
{
$this->assertEquals(
ArrayLib::valuekey(
[
'testkey1' => 'testvalue1',
'testkey2' => 'testvalue2'
]
),
[
'testvalue1' => 'testvalue1',
'testvalue2' => 'testvalue2'
]
);
}
public function testArrayMapRecursive()
{
$array = [
'a ',
[' b', 'c'],
];
$strtoupper = [
'A ',
[' B', 'C'],
];
$trim = [
'a',
['b', 'c'],
];
$this->assertEquals(
$strtoupper,
ArrayLib::array_map_recursive('strtoupper', $array)
);
$this->assertEquals(
$trim,
ArrayLib::array_map_recursive('trim', $array)
);
}
public function testArrayMergeRecursive()
{
$first = [
'first' => 'a',
'second' => 'b',
];
$second = [
'third' => 'c',
'fourth' => 'd',
];
$expected = [
'first' => 'a',
'second' => 'b',
'third' => 'c',
'fourth' => 'd',
];
$this->assertEquals(
$expected,
ArrayLib::array_merge_recursive($first, $second),
'First values should supplement second values'
);
$first = [
'first' => 'a',
'second' => 'b',
];
$second = [
'first' => 'c',
'third' => 'd',
];
$expected = [
'first' => 'c',
'second' => 'b',
'third' => 'd',
];
$this->assertEquals(
$expected,
ArrayLib::array_merge_recursive($first, $second),
'Second values should override first values'
);
$first = [
'first' => [
'first' => 'a',
],
'second' => [
'second' => 'b',
],
];
$second = [
'first' => [
'first' => 'c',
],
'third' => [
'third' => 'd',
],
];
$expected = [
'first' => [
'first' => 'c',
],
'second' => [
'second' => 'b',
],
'third' => [
'third' => 'd',
],
];
$this->assertEquals(
$expected,
ArrayLib::array_merge_recursive($first, $second),
'Nested second values should override first values'
);
$first = [
'first' => [
'first' => 'a',
],
'second' => [
'second' => 'b',
],
];
$second = [
'first' => [
'second' => 'c',
],
'third' => [
'third' => 'd',
],
];
$expected = [
'first' => [
'first' => 'a',
'second' => 'c',
],
'second' => [
'second' => 'b',
],
'third' => [
'third' => 'd',
],
];
$this->assertEquals(
$expected,
ArrayLib::array_merge_recursive($first, $second),
'Nested first values should supplement second values'
);
$first = [
'first' => [
0 => 'a',
],
'second' => [
1 => 'b',
],
];
$second = [
'first' => [
0 => 'c',
],
'third' => [
2 => 'd',
],
];
$expected = [
'first' => [
0 => 'c',
],
'second' => [
1 => 'b',
],
'third' => [
2 => 'd',
],
];
$this->assertEquals(
$expected,
ArrayLib::array_merge_recursive($first, $second),
'Numeric keys should behave like string keys'
);
}
public function testFlatten()
{
$options = [
'1' => 'one',
'2' => 'two'
];
$expected = $options;
$this->assertEquals($expected, ArrayLib::flatten($options));
$options = [
'1' => [
'2' => 'two',
'3' => 'three'
],
'4' => 'four'
];
$expected = [
'2' => 'two',
'3' => 'three',
'4' => 'four'
];
$this->assertEquals($expected, ArrayLib::flatten($options));
}
/**
* Test that items can be added during iteration
*/
public function testIterateVolatileAppended()
{
$initial = [
'one' => [ 'next' => 'two', 'prev' => null ],
'two' => [ 'next' => 'three', 'prev' => 'one' ],
'three' => [ 'next' => null, 'prev' => 'two' ],
];
// Test new items are iterated
$items = $initial;
$seen = [];
foreach (ArrayLib::iterateVolatile($items) as $key => $value) {
$seen[$key] = $value;
// Append four
if ($key === 'three') {
$items['three']['next'] = 'four';
$items['four'] = [ 'next' => null, 'prev' => 'three'];
}
// Prepend zero won't force it to be iterated next, but it will be iterated
if ($key === 'one') {
$items['one']['next'] = 'zero';
$items = array_merge(
['zero' => [ 'next' => 'one', 'prev' => 'three']],
$items
);
}
}
$expected = [
'one' => [ 'next' => 'two', 'prev' => null ],
'two' => [ 'next' => 'three', 'prev' => 'one' ],
'three' => [ 'next' => null, 'prev' => 'two' ],
'zero' => [ 'next' => 'one', 'prev' => 'three'],
'four' => [ 'next' => null, 'prev' => 'three']
];
// All items are iterated (order not deterministic)
$this->assertEquals(
$expected,
$seen,
'New items are iterated over'
);
}
/**
* Test that items can be modified during iteration
*/
public function testIterateVolatileModified()
{
$initial = [
'one' => [ 'next' => 'two', 'prev' => null ],
'two' => [ 'next' => 'three', 'prev' => 'one' ],
'three' => [ 'next' => 'four', 'prev' => 'two' ],
'four' => [ 'next' => null, 'prev' => 'three' ],
];
// Test new items are iterated
$items = $initial;
$seen = [];
foreach (ArrayLib::iterateVolatile($items) as $key => $value) {
$seen[$key] = $value;
// One modifies two
if ($key === 'one') {
$items['two']['modifiedby'] = 'one';
}
// Two removes three, preventing it from being iterated next
if ($key === 'two') {
unset($items['three']);
}
// Four removes two, but since it's already been iterated by this point
// it's too late.
if ($key === 'four') {
unset($items['two']);
}
}
$expected = [
'one' => [ 'next' => 'two', 'prev' => null ],
'two' => [ 'next' => 'three', 'prev' => 'one', 'modifiedby' => 'one' ],
'four' => [ 'next' => null, 'prev' => 'three' ],
];
// All items are iterated (order not deterministic)
$this->assertEquals(
ksort($expected),
ksort($seen),
'New items are iterated over'
);
}
public function testShuffleAssociative()
{
$list = ['a' => 1, 'b' => 2, 'c' => 3, 'd' => 4, 'e' => 5, 'f' => 6];
$copy = $list;
// Try shuffling 3 times - it's technically possible the result of a shuffle could be
// the exact same order as the original list.
for ($attempts = 1; $attempts <= 3; $attempts++) {
ArrayLib::shuffleAssociative($copy);
// Check value/key association is retained
foreach ($list as $key => $value) {
$this->assertEquals($value, $copy[$key]);
}
$failed = false;
try {
// Check the order is different
$this->assertNotSame($list, $copy);
} catch (ExpectationFailedException $e) {
$failed = true;
// Only fail the test if we've tried and failed 3 times.
if ($attempts === 3) {
throw $e;
}
}
// If we've passed the shuffle test, don't retry.
if (!$failed) {
break;
}
}
}
public static function provideInsertBefore(): array
{
return [
'simple insertion' => [
'insert' => 'new',
'before' => 'def',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'new', 'def', '0', null, true, 0, 'last']
],
'insert before first' => [
'insert' => 'new',
'before' => 'abc',
'strict' => true,
'splat' => false,
'expected' => ['new', 'abc', '', [1,2,3], 'def', '0', null, true, 0, 'last']
],
'insert before last' => [
'insert' => 'new',
'before' => 'last',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'new', 'last']
],
'insert before missing' => [
'insert' => 'new',
'before' => 'this value isnt there',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last', 'new']
],
'strict' => [
'insert' => 'new',
'before' => 0,
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 'new', 0, 'last']
],
'not strict' => [
'insert' => 'new',
'before' => 0,
'strict' => false,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', 'new', '0', null, true, 0, 'last']
],
'before array' => [
'insert' => 'new',
'before' => [1,2,3],
'strict' => true,
'splat' => false,
'expected' => ['abc', '', 'new', [1,2,3], 'def', '0', null, true, 0, 'last']
],
'before missing array' => [
'insert' => 'new',
'before' => ['a', 'b', 'c'],
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last', 'new']
],
'splat array' => [
'insert' => ['a', 'b', 'c'],
'before' => 'def',
'strict' => true,
'splat' => true,
'expected' => ['abc', '', [1,2,3], 'a', 'b', 'c', 'def', '0', null, true, 0, 'last']
],
'no splat array' => [
'insert' => ['a', 'b', 'c'],
'before' => 'def',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], ['a', 'b', 'c'], 'def', '0', null, true, 0, 'last']
],
];
}
#[DataProvider('provideInsertBefore')]
public function testInsertBefore(mixed $insert, mixed $before, bool $strict, bool $splat, array $expected): void
{
$array = ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last'];
$final = ArrayLib::insertBefore($array, $insert, $before, $strict, $splat);
$this->assertSame($expected, $final);
}
public static function provideInsertAfter(): array
{
return [
'simple insertion' => [
'insert' => 'new',
'after' => 'def',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', 'new', '0', null, true, 0, 'last']
],
'insert after first' => [
'insert' => 'new',
'after' => 'abc',
'strict' => true,
'splat' => false,
'expected' => ['abc', 'new', '', [1,2,3], 'def', '0', null, true, 0, 'last']
],
'insert after last' => [
'insert' => 'new',
'after' => 'last',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last', 'new']
],
'insert after missing' => [
'insert' => 'new',
'after' => 'this value isnt there',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last', 'new']
],
'strict' => [
'insert' => 'new',
'after' => 0,
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'new', 'last']
],
'not strict' => [
'insert' => 'new',
'after' => 0,
'strict' => false,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', 'new', null, true, 0, 'last']
],
'after array' => [
'insert' => 'new',
'after' => [1,2,3],
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'new', 'def', '0', null, true, 0, 'last']
],
'after missing array' => [
'insert' => 'new',
'after' => ['a', 'b', 'c'],
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last', 'new']
],
'splat array' => [
'insert' => ['a', 'b', 'c'],
'after' => 'def',
'strict' => true,
'splat' => true,
'expected' => ['abc', '', [1,2,3], 'def', 'a', 'b', 'c', '0', null, true, 0, 'last']
],
'no splat array' => [
'insert' => ['a', 'b', 'c'],
'after' => 'def',
'strict' => true,
'splat' => false,
'expected' => ['abc', '', [1,2,3], 'def', ['a', 'b', 'c'], '0', null, true, 0, 'last']
],
];
}
#[DataProvider('provideInsertAfter')]
public function testInsertAfter(mixed $insert, mixed $after, bool $strict, bool $splat, array $expected): void
{
$array = ['abc', '', [1,2,3], 'def', '0', null, true, 0, 'last'];
$final = ArrayLib::insertAfter($array, $insert, $after, $strict, $splat);
$this->assertSame($expected, $final);
}
}