mirror of
https://github.com/zoe-may/TDoG-Skin.git
synced 2025-01-19 14:57:23 +08:00
258 lines
10 KiB
PHP
258 lines
10 KiB
PHP
|
<?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);
|
||
|
}
|
||
|
}
|