mirror of
https://github.com/espocrm/espocrm.git
synced 2026-07-01 08:26:04 +00:00
409 lines
15 KiB
PHP
409 lines
15 KiB
PHP
<?php
|
|
/************************************************************************
|
|
* This file is part of EspoCRM.
|
|
*
|
|
* EspoCRM - Open Source CRM application.
|
|
* Copyright (C) 2014-2017 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
|
|
* Website: http://www.espocrm.com
|
|
*
|
|
* EspoCRM is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* EspoCRM is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of this program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU General Public License version 3.
|
|
*
|
|
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
|
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
|
************************************************************************/
|
|
|
|
use Espo\ORM\DB\Query\Mysql as Query;
|
|
use Espo\ORM\EntityFactory;
|
|
|
|
use Espo\Entities\Post;
|
|
use Espo\Entities\Comment;
|
|
use Espo\Entities\Tag;
|
|
use Espo\Entities\Note;
|
|
|
|
require_once 'tests/unit/testData/DB/Entities.php';
|
|
require_once 'tests/unit/testData/DB/MockPDO.php';
|
|
require_once 'tests/unit/testData/DB/MockDBResult.php';
|
|
|
|
class QueryTest extends PHPUnit_Framework_TestCase
|
|
{
|
|
protected $query;
|
|
|
|
protected $pdo;
|
|
|
|
protected $entityFactory;
|
|
|
|
protected function setUp()
|
|
{
|
|
$this->pdo = $this->getMock('MockPDO');
|
|
$this->pdo
|
|
->expects($this->any())
|
|
->method('quote')
|
|
->will($this->returnCallback(function() {
|
|
$args = func_get_args();
|
|
return "'" . $args[0] . "'";
|
|
}));
|
|
|
|
|
|
$this->entityFactory = $this->getMockBuilder('\\Espo\\ORM\\EntityFactory')->disableOriginalConstructor()->getMock();
|
|
$this->entityFactory->expects($this->any())
|
|
->method('create')
|
|
->will($this->returnCallback(function() {
|
|
$args = func_get_args();
|
|
$className = "\\Espo\\Entities\\" . $args[0];
|
|
return new $className();
|
|
}));
|
|
|
|
$this->query = new Query($this->pdo, $this->entityFactory);
|
|
|
|
$this->post = new \Espo\Entities\Post();
|
|
$this->comment = new \Espo\Entities\Comment();
|
|
$this->tag = new \Espo\Entities\Tag();
|
|
$this->note = new \Espo\Entities\Note();
|
|
|
|
$this->contact = new \Espo\Entities\Contact();
|
|
$this->account = new \Espo\Entities\Account();
|
|
}
|
|
|
|
protected function tearDown()
|
|
{
|
|
unset($this->query);
|
|
unset($this->pdo);
|
|
unset($this->post);
|
|
unset($this->tag);
|
|
unset($this->note);
|
|
unset($this->contact);
|
|
unset($this->account);
|
|
}
|
|
|
|
public function testSelectAllColumns()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Account', array(
|
|
'orderBy' => 'name',
|
|
'order' => 'ASC',
|
|
'offset' => 10,
|
|
'limit' => 20
|
|
));
|
|
|
|
$expectedSql =
|
|
"SELECT account.id AS `id`, account.name AS `name`, account.deleted AS `deleted` FROM `account` " .
|
|
"WHERE account.deleted = '0' ORDER BY account.name ASC LIMIT 10, 20";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testSelectSkipTextColumns()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Article', array(
|
|
'orderBy' => 'name',
|
|
'order' => 'ASC',
|
|
'offset' => 10,
|
|
'limit' => 20,
|
|
'skipTextColumns' => true
|
|
));
|
|
|
|
$expectedSql =
|
|
"SELECT article.id AS `id`, article.name AS `name`, article.deleted AS `deleted` FROM `article` " .
|
|
"WHERE article.deleted = '0' ORDER BY article.name ASC LIMIT 10, 20";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testSelectWithBelongsToJoin()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
|
|
));
|
|
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.post_id AS `postId`, post.name AS `postName`, comment.name AS `name`, comment.deleted AS `deleted` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testSelectWithSpecifiedColumns()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'name')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.name AS `name` FROM `comment` " .
|
|
"WHERE comment.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'name', 'postName')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.name AS `name`, post.name AS `postName` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'name', 'postName'),
|
|
'leftJoins' => array('post')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.name AS `name`, post.name AS `postName` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'name'),
|
|
'leftJoins' => array('post')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.name AS `name` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testWithSpecifiedFunction()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'postId', 'post.name', 'COUNT:id'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('postId', 'post.name')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, comment.post_id AS `postId`, post.name AS `post.name`, COUNT(comment.id) AS `COUNT:id` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0' " .
|
|
"GROUP BY comment.post_id, post.name";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('id', 'COUNT:id', 'MONTH:post.createdAt'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('MONTH:post.createdAt')
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id`, COUNT(comment.id) AS `COUNT:id`, DATE_FORMAT(post.created_at, '%Y-%m') AS `MONTH:post.createdAt` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0' " .
|
|
"GROUP BY DATE_FORMAT(post.created_at, '%Y-%m')";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testSelectWithJoinChildren()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Post', array(
|
|
'select' => ['id', 'name'],
|
|
'leftJoins' => [['notes', 'notesLeft']]
|
|
));
|
|
|
|
$expectedSql =
|
|
"SELECT post.id AS `id`, post.name AS `name` FROM `post` " .
|
|
"LEFT JOIN `note` AS `notesLeft` ON post.id = notesLeft.parent_id AND notesLeft.parent_type = 'Post' AND notesLeft.deleted = '0' " .
|
|
"WHERE post.deleted = '0'";
|
|
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testSelectWithSubquery()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Post', array(
|
|
'select' => ['id', 'name'],
|
|
'whereClause' => array(
|
|
'post.id=s' => array(
|
|
'entityType' => 'Post',
|
|
'selectParams' => array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'name' => 'test'
|
|
)
|
|
)
|
|
)
|
|
)
|
|
));
|
|
|
|
$expectedSql = "SELECT post.id AS `id`, post.name AS `name` FROM `post` WHERE post.id IN (SELECT post.id AS `id` FROM `post` WHERE post.name = 'test' AND post.deleted = '0') AND post.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Post', array(
|
|
'select' => ['id', 'name'],
|
|
'whereClause' => array(
|
|
'post.id!=s' => array(
|
|
'entityType' => 'Post',
|
|
'selectParams' => array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'name' => 'test'
|
|
)
|
|
)
|
|
)
|
|
)
|
|
));
|
|
|
|
$expectedSql = "SELECT post.id AS `id`, post.name AS `name` FROM `post` WHERE post.id NOT IN (SELECT post.id AS `id` FROM `post` WHERE post.name = 'test' AND post.deleted = '0') AND post.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
|
|
$sql = $this->query->createSelectQuery('Post', array(
|
|
'select' => ['id', 'name'],
|
|
'whereClause' => array(
|
|
'NOT'=> array(
|
|
'name' => 'test',
|
|
'post.createdById' => '1'
|
|
)
|
|
)
|
|
));
|
|
|
|
$expectedSql = "SELECT post.id AS `id`, post.name AS `name` FROM `post` WHERE post.id NOT IN (SELECT post.id AS `id` FROM `post` WHERE post.name = 'test' AND post.created_by_id = '1' AND post.deleted = '0') AND post.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testOrderBy()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('COUNT:id', 'YEAR:post.createdAt'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('YEAR:post.createdAt'),
|
|
'orderBy' => 2
|
|
));
|
|
$expectedSql =
|
|
"SELECT COUNT(comment.id) AS `COUNT:id`, YEAR(post.created_at) AS `YEAR:post.createdAt` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0' " .
|
|
"GROUP BY YEAR(post.created_at) ".
|
|
"ORDER BY 2 ASC";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('COUNT:id', 'post.name'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('post.name'),
|
|
'orderBy' => 'LIST:post.name:Test,Hello',
|
|
));
|
|
|
|
$expectedSql =
|
|
"SELECT COUNT(comment.id) AS `COUNT:id`, post.name AS `post.name` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0' " .
|
|
"GROUP BY post.name ".
|
|
"ORDER BY FIELD(post.name, 'Hello', 'Test') DESC";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('COUNT:id', 'YEAR:post.createdAt', 'post.name'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('YEAR:post.createdAt', 'post.name'),
|
|
'orderBy' => array(
|
|
array(2, 'DESC'),
|
|
array('LIST:post.name:Test,Hello')
|
|
)
|
|
));
|
|
$expectedSql =
|
|
"SELECT COUNT(comment.id) AS `COUNT:id`, YEAR(post.created_at) AS `YEAR:post.createdAt`, post.name AS `post.name` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE comment.deleted = '0' " .
|
|
"GROUP BY YEAR(post.created_at), post.name ".
|
|
"ORDER BY 2 DESC, FIELD(post.name, 'Hello', 'Test') DESC";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testForeign()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => array('COUNT:comment.id', 'postId', 'postName'),
|
|
'leftJoins' => array('post'),
|
|
'groupBy' => array('postId'),
|
|
'whereClause' => array(
|
|
'post.createdById' => 'id_1'
|
|
),
|
|
));
|
|
$expectedSql =
|
|
"SELECT COUNT(comment.id) AS `COUNT:comment.id`, comment.post_id AS `postId`, post.name AS `postName` FROM `comment` " .
|
|
"LEFT JOIN `post` AS `post` ON comment.post_id = post.id " .
|
|
"WHERE post.created_by_id = 'id_1' AND comment.deleted = '0' " .
|
|
"GROUP BY comment.post_id";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testInArray()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'id' => ['id_1']
|
|
),
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id` FROM `comment` " .
|
|
"WHERE comment.id IN ('id_1') AND comment.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'id!=' => ['id_1']
|
|
),
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id` FROM `comment` " .
|
|
"WHERE comment.id NOT IN ('id_1') AND comment.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'id' => []
|
|
),
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id` FROM `comment` " .
|
|
"WHERE 0 AND comment.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'name' => 'Test',
|
|
'id!=' => []
|
|
),
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id` FROM `comment` " .
|
|
"WHERE comment.name = 'Test' AND 1 AND comment.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
public function testFunction1()
|
|
{
|
|
$sql = $this->query->createSelectQuery('Comment', array(
|
|
'select' => ['id'],
|
|
'whereClause' => array(
|
|
'MONTH_NUMBER:comment.created_at' => 2
|
|
)
|
|
));
|
|
$expectedSql =
|
|
"SELECT comment.id AS `id` FROM `comment` " .
|
|
"WHERE MONTH(comment.created_at) = '2' AND comment.deleted = '0'";
|
|
$this->assertEquals($expectedSql, $sql);
|
|
}
|
|
|
|
} |