Files
espocrm/tests/unit/Espo/ORM/DB/QueryTest.php
2017-04-26 16:32:24 +03:00

398 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);
}
}