[DDC-551] Initial support for filters in the JoinedSubclassPersister
Still some things to do.
This commit is contained in:
parent
4c94a7ccc5
commit
f6d5f0481e
4 changed files with 125 additions and 7 deletions
|
@ -1584,7 +1584,7 @@ class BasicEntityPersister
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function generateFilterConditionSQL(ClassMetadata $targetEntity, $targetTableAlias)
|
private function generateFilterConditionSQL(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
$filterSql = '';
|
$filterSql = '';
|
||||||
|
|
||||||
|
|
|
@ -328,8 +328,17 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||||
|
|
||||||
$joinSql .= $baseTableAlias . '.' . $idColumn . ' = ' . $tableAlias . '.' . $idColumn;
|
$joinSql .= $baseTableAlias . '.' . $idColumn . ' = ' . $tableAlias . '.' . $idColumn;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Filters for each parent table
|
||||||
|
$filterSql = $this->generateFilterConditionSQL($parentClass, $tableAlias, $parentClass->table['name']);
|
||||||
|
if('' !== $filterSql) {
|
||||||
|
if (!$first) $joinSql .= ' AND ';
|
||||||
|
$joinSql .= $filterSql;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//@todo: filters for the sub tables of childs
|
||||||
|
|
||||||
// OUTER JOIN sub tables
|
// OUTER JOIN sub tables
|
||||||
foreach ($this->_class->subClasses as $subClassName) {
|
foreach ($this->_class->subClasses as $subClassName) {
|
||||||
$subClass = $this->_em->getClassMetadata($subClassName);
|
$subClass = $this->_em->getClassMetadata($subClassName);
|
||||||
|
@ -389,6 +398,13 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||||
$lockSql = ' ' . $this->_platform->getWriteLockSql();
|
$lockSql = ' ' . $this->_platform->getWriteLockSql();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Filters for the base table
|
||||||
|
$filterSql = $this->generateFilterConditionSQL($this->_class, $baseTableAlias, $this->_class->table['name']);
|
||||||
|
if('' !== $filterSql) {
|
||||||
|
if($conditionSql) $conditionSql .= ' AND ';
|
||||||
|
$conditionSql .= $filterSql;
|
||||||
|
}
|
||||||
|
|
||||||
return $this->_platform->modifyLimitQuery('SELECT ' . $this->_selectColumnListSql
|
return $this->_platform->modifyLimitQuery('SELECT ' . $this->_selectColumnListSql
|
||||||
. ' FROM ' . $this->_class->getQuotedTableName($this->_platform) . ' ' . $baseTableAlias
|
. ' FROM ' . $this->_class->getQuotedTableName($this->_platform) . ' ' . $baseTableAlias
|
||||||
. $joinSql
|
. $joinSql
|
||||||
|
@ -473,4 +489,19 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||||
$value = $this->fetchVersionValue($this->_getVersionedClassMetadata(), $id);
|
$value = $this->fetchVersionValue($this->_getVersionedClassMetadata(), $id);
|
||||||
$this->_class->setFieldValue($entity, $this->_class->versionField, $value);
|
$this->_class->setFieldValue($entity, $this->_class->versionField, $value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function generateFilterConditionSQL(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
|
{
|
||||||
|
$filterSql = '';
|
||||||
|
|
||||||
|
$first = true;
|
||||||
|
foreach($this->_em->getFilters()->getEnabledFilters() as $filter) {
|
||||||
|
if("" !== $filterExpr = $filter->addFilterConstraint($targetEntity, $targetTableAlias, $targetTable)) {
|
||||||
|
if ( ! $first) $filterSql .= ' AND '; else $first = false;
|
||||||
|
$filterSql .= '(' . $filterExpr . ')';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $filterSql;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,5 +73,5 @@ abstract class SQLFilter
|
||||||
/**
|
/**
|
||||||
* @return string The constraint if there is one, empty string otherwise
|
* @return string The constraint if there is one, empty string otherwise
|
||||||
*/
|
*/
|
||||||
abstract public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias);
|
abstract public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '');
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,10 @@ use Doctrine\Tests\Models\CMS\CmsGroup;
|
||||||
use Doctrine\Tests\Models\CMS\CmsArticle;
|
use Doctrine\Tests\Models\CMS\CmsArticle;
|
||||||
use Doctrine\Tests\Models\CMS\CmsComment;
|
use Doctrine\Tests\Models\CMS\CmsComment;
|
||||||
|
|
||||||
|
use Doctrine\Tests\Models\Company\CompanyPerson;
|
||||||
|
use Doctrine\Tests\Models\Company\CompanyManager;
|
||||||
|
use Doctrine\Tests\Models\Company\CompanyEmployee;
|
||||||
|
|
||||||
require_once __DIR__ . '/../../TestInit.php';
|
require_once __DIR__ . '/../../TestInit.php';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -30,6 +34,7 @@ class SQLFilterTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
public function setUp()
|
public function setUp()
|
||||||
{
|
{
|
||||||
$this->useModelSet('cms');
|
$this->useModelSet('cms');
|
||||||
|
$this->useModelSet('company');
|
||||||
parent::setUp();
|
parent::setUp();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -444,6 +449,44 @@ class SQLFilterTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$this->assertEquals(1, count($user->groups->slice(0,10)));
|
$this->assertEquals(1, count($user->groups->slice(0,10)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testJoinSubclassPersister_FilterOnBaseTable()
|
||||||
|
{
|
||||||
|
$this->loadCompanyFixtureData();
|
||||||
|
$this->assertEquals(2, count($this->_em->getRepository('Doctrine\Tests\Models\Company\CompanyManager')->findAll()));
|
||||||
|
|
||||||
|
// Enable the filter
|
||||||
|
$conf = $this->_em->getConfiguration();
|
||||||
|
$conf->addFilter("manager_title", "\Doctrine\Tests\ORM\Functional\CompanyManagerTitleFilter");
|
||||||
|
$this->_em->getFilters()
|
||||||
|
->enable("manager_title")
|
||||||
|
->setParameter("title", "devlead", \Doctrine\DBAL\Types\Type::getType(\Doctrine\DBAL\Types\Type::STRING)->getBindingType());
|
||||||
|
|
||||||
|
$this->_em->getConnection()->getConfiguration()->setSQLLogger(new \Doctrine\DBAL\Logging\EchoSQLLogger);
|
||||||
|
|
||||||
|
$managers = $this->_em->getRepository('Doctrine\Tests\Models\Company\CompanyManager')->findAll();
|
||||||
|
$this->assertEquals(1, count($managers));
|
||||||
|
$this->assertEquals("Guilherme", $managers[0]->getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testJoinSubclassPersister_FilterOnParentTable()
|
||||||
|
{
|
||||||
|
$this->loadCompanyFixtureData();
|
||||||
|
$this->assertEquals(2, count($this->_em->getRepository('Doctrine\Tests\Models\Company\CompanyManager')->findAll()));
|
||||||
|
|
||||||
|
// Enable the filter
|
||||||
|
$conf = $this->_em->getConfiguration();
|
||||||
|
$conf->addFilter("employee_department", "\Doctrine\Tests\ORM\Functional\CompanyEmployeeDepartmentFilter");
|
||||||
|
$this->_em->getFilters()
|
||||||
|
->enable("employee_department")
|
||||||
|
->setParameter("department", "parsers", \Doctrine\DBAL\Types\Type::getType(\Doctrine\DBAL\Types\Type::STRING)->getBindingType());
|
||||||
|
|
||||||
|
$this->_em->getConnection()->getConfiguration()->setSQLLogger(new \Doctrine\DBAL\Logging\EchoSQLLogger);
|
||||||
|
|
||||||
|
$managers = $this->_em->getRepository('Doctrine\Tests\Models\Company\CompanyManager')->findAll();
|
||||||
|
$this->assertEquals(1, count($managers));
|
||||||
|
$this->assertEquals("Guilherme", $managers[0]->getName());
|
||||||
|
}
|
||||||
|
|
||||||
private function loadFixtureData()
|
private function loadFixtureData()
|
||||||
{
|
{
|
||||||
$user = new CmsUser;
|
$user = new CmsUser;
|
||||||
|
@ -507,11 +550,31 @@ class SQLFilterTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$this->groupId = $group->id;
|
$this->groupId = $group->id;
|
||||||
$this->groupId2 = $group2->id;
|
$this->groupId2 = $group2->id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function loadCompanyFixtureData()
|
||||||
|
{
|
||||||
|
$manager = new CompanyManager;
|
||||||
|
$manager->setName('Roman');
|
||||||
|
$manager->setTitle('testlead');
|
||||||
|
$manager->setSalary(42);
|
||||||
|
$manager->setDepartment('persisters');
|
||||||
|
|
||||||
|
$manager2 = new CompanyManager;
|
||||||
|
$manager2->setName('Guilherme');
|
||||||
|
$manager2->setTitle('devlead');
|
||||||
|
$manager2->setSalary(42);
|
||||||
|
$manager2->setDepartment('parsers');
|
||||||
|
|
||||||
|
$this->_em->persist($manager);
|
||||||
|
$this->_em->persist($manager2);
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MySoftDeleteFilter extends SQLFilter
|
class MySoftDeleteFilter extends SQLFilter
|
||||||
{
|
{
|
||||||
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias)
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
if ($targetEntity->name != "MyEntity\SoftDeleteNewsItem") {
|
if ($targetEntity->name != "MyEntity\SoftDeleteNewsItem") {
|
||||||
return "";
|
return "";
|
||||||
|
@ -523,7 +586,7 @@ class MySoftDeleteFilter extends SQLFilter
|
||||||
|
|
||||||
class MyLocaleFilter extends SQLFilter
|
class MyLocaleFilter extends SQLFilter
|
||||||
{
|
{
|
||||||
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias)
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
if (!in_array("LocaleAware", $targetEntity->reflClass->getInterfaceNames())) {
|
if (!in_array("LocaleAware", $targetEntity->reflClass->getInterfaceNames())) {
|
||||||
return "";
|
return "";
|
||||||
|
@ -535,7 +598,7 @@ class MyLocaleFilter extends SQLFilter
|
||||||
|
|
||||||
class CMSCountryFilter extends SQLFilter
|
class CMSCountryFilter extends SQLFilter
|
||||||
{
|
{
|
||||||
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias)
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsAddress") {
|
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsAddress") {
|
||||||
return "";
|
return "";
|
||||||
|
@ -547,7 +610,7 @@ class CMSCountryFilter extends SQLFilter
|
||||||
|
|
||||||
class CMSGroupPrefixFilter extends SQLFilter
|
class CMSGroupPrefixFilter extends SQLFilter
|
||||||
{
|
{
|
||||||
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias)
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsGroup") {
|
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsGroup") {
|
||||||
return "";
|
return "";
|
||||||
|
@ -558,7 +621,7 @@ class CMSGroupPrefixFilter extends SQLFilter
|
||||||
}
|
}
|
||||||
class CMSArticleTopicFilter extends SQLFilter
|
class CMSArticleTopicFilter extends SQLFilter
|
||||||
{
|
{
|
||||||
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias)
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
{
|
{
|
||||||
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsArticle") {
|
if ($targetEntity->name != "Doctrine\Tests\Models\CMS\CmsArticle") {
|
||||||
return "";
|
return "";
|
||||||
|
@ -567,3 +630,27 @@ class CMSArticleTopicFilter extends SQLFilter
|
||||||
return $targetTableAlias.'.topic = ' . $this->getParameter('topic'); // getParam uses connection to quote the value.
|
return $targetTableAlias.'.topic = ' . $this->getParameter('topic'); // getParam uses connection to quote the value.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class CompanyManagerTitleFilter extends SQLFilter
|
||||||
|
{
|
||||||
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
|
{
|
||||||
|
if ($targetEntity->name != "Doctrine\Tests\Models\Company\CompanyManager") {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $targetTableAlias.'.title = ' . $this->getParameter('title'); // getParam uses connection to quote the value.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CompanyEmployeeDepartmentFilter extends SQLFilter
|
||||||
|
{
|
||||||
|
public function addFilterConstraint(ClassMetadata $targetEntity, $targetTableAlias, $targetTable = '')
|
||||||
|
{
|
||||||
|
if ($targetEntity->name != "Doctrine\Tests\Models\Company\CompanyEmployee") {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $targetTableAlias.'.department = ' . $this->getParameter('department'); // getParam uses connection to quote the value.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue