TDoG-Skin/vendor/lorisleiva/laravel-search-string/tests/VisitorBuildColumnsTest.php

258 lines
10 KiB
PHP
Raw Normal View History

2024-08-17 18:43:48 +08:00
<?php
namespace Lorisleiva\LaravelSearchString\Tests;
use Lorisleiva\LaravelSearchString\Tests\Concerns\DumpsWhereClauses;
use Lorisleiva\LaravelSearchString\Tests\Concerns\GeneratesEloquentBuilder;
use Lorisleiva\LaravelSearchString\Visitors\AttachRulesVisitor;
use Lorisleiva\LaravelSearchString\Visitors\BuildColumnsVisitor;
use Lorisleiva\LaravelSearchString\Visitors\RemoveNotSymbolVisitor;
class VisitorBuildColumnsTest extends VisitorTest
{
use DumpsWhereClauses;
public function visitors($manager, $builder, $model)
{
return [
new RemoveNotSymbolVisitor(),
new AttachRulesVisitor($manager),
new BuildColumnsVisitor($manager, $builder),
];
}
/** @test */
public function it_generates_basic_where_clauses_that_match_the_query_operator()
{
$this->assertWhereClauses('name:1', ['Basic[and][0]' => 'products.name = 1']);
$this->assertWhereClauses('name=1', ['Basic[and][0]' => 'products.name = 1']);
$this->assertWhereClauses('name=Hello', ['Basic[and][0]' => 'products.name = Hello']);
$this->assertWhereClauses('name="Hello World"', ['Basic[and][0]' => 'products.name = Hello World']);
$this->assertWhereClauses('not name:1', ['Basic[and][0]' => 'products.name != 1']);
$this->assertWhereClauses('not name=Hello', ['Basic[and][0]' => 'products.name != Hello']);
$this->assertWhereClauses('name<0', ['Basic[and][0]' => 'products.name < 0']);
$this->assertWhereClauses('name<=0', ['Basic[and][0]' => 'products.name <= 0']);
$this->assertWhereClauses('name>0', ['Basic[and][0]' => 'products.name > 0']);
$this->assertWhereClauses('name>=0', ['Basic[and][0]' => 'products.name >= 0']);
$this->assertWhereClauses('not name<0', ['Basic[and][0]' => 'products.name >= 0']);
$this->assertWhereClauses('not name<=0', ['Basic[and][0]' => 'products.name > 0']);
$this->assertWhereClauses('not name>0', ['Basic[and][0]' => 'products.name <= 0']);
$this->assertWhereClauses('not name>=0', ['Basic[and][0]' => 'products.name < 0']);
// boolean_variable is defined in the `columns.boolean` option.
$this->assertWhereClauses('boolean_variable', ['Basic[and][0]' => 'products.boolean_variable = true']);
$this->assertWhereClauses('not boolean_variable', ['Basic[and][0]' => 'products.boolean_variable = false']);
}
/** @test */
public function it_can_generate_in_and_not_in_where_clauses()
{
$this->assertWhereClauses('name in (1,2,3)', ['In[and][0]' => 'products.name [1, 2, 3]']);
$this->assertWhereClauses('not name in (1,2,3)', ['NotIn[and][0]' => 'products.name [1, 2, 3]']);
$this->assertWhereClauses('name:1,2,3', ['In[and][0]' => 'products.name [1, 2, 3]']);
$this->assertWhereClauses('not name:1,2,3', ['NotIn[and][0]' => 'products.name [1, 2, 3]']);
}
/** @test */
public function it_generates_where_clauses_from_aliased_columned_using_the_real_column_name()
{
$model = $this->getModelWithColumns([
'zipcode' => 'postcode',
'created_at' => ['key' => '/^(created|date)$/', 'date' => true, 'boolean' => true],
'activated' => ['key' => 'active', 'boolean' => true],
]);
$this->assertWhereClauses('postcode:1028', ['Basic[and][0]' => 'models.zipcode = 1028'], $model);
$this->assertWhereClauses('postcode>10', ['Basic[and][0]' => 'models.zipcode > 10'], $model);
$this->assertWhereClauses('not postcode in (1000, 1002)', ['NotIn[and][0]' => 'models.zipcode [1000, 1002]'], $model);
$this->assertWhereClauses('created>2019-01-01', ['Basic[and][0]' => 'models.created_at > 2019-01-01 23:59:59'], $model);
$this->assertWhereClauses('created', ['NotNull[and][0]' => 'models.created_at'], $model);
$this->assertWhereClauses('date', ['NotNull[and][0]' => 'models.created_at'], $model);
$this->assertWhereClauses('not created', ['Null[and][0]' => 'models.created_at'], $model);
$this->assertWhereClauses('not date', ['Null[and][0]' => 'models.created_at'], $model);
$this->assertWhereClauses('active', ['Basic[and][0]' => 'models.activated = true'], $model);
$this->assertWhereClauses('not active', ['Basic[and][0]' => 'models.activated = false'], $model);
}
/** @test */
public function it_searches_using_like_where_clauses()
{
$this->assertWhereClauses('foobar', [
'Nested[and][0]' => [
'Basic[or][0]' => 'products.name like %foobar%',
'Basic[or][1]' => 'products.description like %foobar%',
]
]);
$this->assertWhereClauses('not foobar', [
'Nested[and][0]' => [
'Basic[and][0]' => 'products.name not like %foobar%',
'Basic[and][1]' => 'products.description not like %foobar%',
]
]);
}
/** @test */
public function it_can_searches_using_case_insensitive_like_where_clauses()
{
$model = $this->getModelWithOptions([
'case_insensitive' => true,
'columns' => [
'name' => ['searchable' => true],
'description' => ['searchable' => true],
]
]);
$this->assertWhereClauses('FooBar', [
'Nested[and][0]' => [
'Basic[or][0]' => 'LOWER(models.name) like %foobar%',
'Basic[or][1]' => 'LOWER(models.description) like %foobar%',
]
], $model);
$this->assertWhereClauses('not FooBar', [
'Nested[and][0]' => [
'Basic[and][0]' => 'LOWER(models.name) not like %foobar%',
'Basic[and][1]' => 'LOWER(models.description) not like %foobar%',
]
], $model);
}
/** @test */
public function it_does_not_add_where_clause_if_no_searchable_columns_were_given()
{
$model = $this->getModelWithOptions([]);
$this->assertWhereClauses('foobar', [], $model);
$this->assertWhereClauses('not foobar', [], $model);
}
/** @test */
public function it_does_not_nest_where_clauses_if_only_one_searchable_columns_is_given()
{
$model = $this->getModelWithColumns([
'name' => [ 'searchable' => true ]
]);
$this->assertWhereClauses('foobar', [
'Basic[and][0]' => 'models.name like %foobar%'
], $model);
$this->assertWhereClauses('not foobar', [
'Basic[and][0]' => 'models.name not like %foobar%'
], $model);
}
/** @test */
public function it_wraps_basic_queries_in_nested_and_or_where_clauses()
{
$this->assertWhereClauses('name:1 and price>1', [
'Nested[and][0]' => [
'Basic[and][0]' => 'products.name = 1',
'Basic[and][1]' => 'products.price > 1',
]
]);
$this->assertWhereClauses('name:1 or price>1', [
'Nested[and][0]' => [
'Basic[or][0]' => 'products.name = 1',
'Basic[or][1]' => 'products.price > 1',
]
]);
}
/** @test */
public function it_wraps_search_queries_in_nested_and_or_where_clauses()
{
$this->assertWhereClauses('foo and bar', [
'Nested[and][0]' => [
'Nested[and][0]' => [
'Basic[or][0]' => 'products.name like %foo%',
'Basic[or][1]' => 'products.description like %foo%',
],
'Nested[and][1]' => [
'Basic[or][0]' => 'products.name like %bar%',
'Basic[or][1]' => 'products.description like %bar%',
],
]
]);
$this->assertWhereClauses('foo or bar', [
'Nested[and][0]' => [
'Nested[or][0]' => [
'Basic[or][0]' => 'products.name like %foo%',
'Basic[or][1]' => 'products.description like %foo%',
],
'Nested[or][1]' => [
'Basic[or][0]' => 'products.name like %bar%',
'Basic[or][1]' => 'products.description like %bar%',
],
]
]);
$this->assertWhereClauses('not foo or not bar', [
'Nested[and][0]' => [
'Nested[or][0]' => [
'Basic[and][0]' => 'products.name not like %foo%',
'Basic[and][1]' => 'products.description not like %foo%',
],
'Nested[or][1]' => [
'Basic[and][0]' => 'products.name not like %bar%',
'Basic[and][1]' => 'products.description not like %bar%',
],
]
]);
}
/** @test */
public function it_wraps_complex_and_or_operators_in_nested_where_clauses()
{
$this->assertWhereClauses('name:4 or (name:1 or name:2) and price>1 or name:3', [
'Nested[and][0]' => [
'Basic[or][0]' => 'products.name = 4',
'Nested[or][1]' => [
'Nested[and][0]' => [
'Basic[or][0]' => 'products.name = 1',
'Basic[or][1]' => 'products.name = 2',
],
'Basic[and][1]' => 'products.price > 1',
],
'Basic[or][2]' => 'products.name = 3',
]
]);
}
/** @test */
public function it_updates_query_values_according_to_the_rules_map()
{
$model = $this->getModelWithColumns([
'support_level_id' => [
'key' => 'support_level',
'map' => [
'testing' => 1,
'community' => 2,
'official' => 3,
],
]
]);
$this->assertWhereClauses('support_level:testing', ['Basic[and][0]' => 'models.support_level_id = 1'], $model);
$this->assertWhereClauses('support_level:community', ['Basic[and][0]' => 'models.support_level_id = 2'], $model);
$this->assertWhereClauses('support_level:official', ['Basic[and][0]' => 'models.support_level_id = 3'], $model);
}
/** @test */
public function it_does_not_update_query_values_if_the_rule_mapping_is_missing()
{
$model = $this->getModelWithColumns([
'support_level_id' => [
'key' => 'support_level',
'map' => ['testing' => 1],
]
]);
$this->assertWhereClauses('support_level:missing_value', ['Basic[and][0]' => 'models.support_level_id = missing_value'], $model);
}
}