2014-04-19 16:56:16 +04:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ownCloud - App Framework
|
|
|
|
*
|
|
|
|
* @author Bernhard Posselt
|
|
|
|
* @copyright 2012 Bernhard Posselt dev@bernhard-posselt.com
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 3 of the License, or any later version.
|
|
|
|
*
|
|
|
|
* This library 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 AFFERO GENERAL PUBLIC LICENSE for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2014-07-17 13:21:25 +04:00
|
|
|
namespace Test\AppFramework\Db;
|
2014-04-19 16:56:16 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Simple utility class for testing mappers
|
|
|
|
*/
|
2014-11-11 01:30:38 +03:00
|
|
|
abstract class MapperTestUtility extends \Test\TestCase {
|
2014-04-19 16:56:16 +04:00
|
|
|
protected $db;
|
|
|
|
private $query;
|
|
|
|
private $queryAt;
|
|
|
|
private $prepareAt;
|
|
|
|
private $fetchAt;
|
|
|
|
private $iterators;
|
2015-01-29 21:16:28 +03:00
|
|
|
|
2014-04-19 16:56:16 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Run this function before the actual test to either set or initialize the
|
|
|
|
* db. After this the db can be accessed by using $this->db
|
|
|
|
*/
|
2019-11-21 18:40:38 +03:00
|
|
|
protected function setUp(): void {
|
2014-11-11 01:30:38 +03:00
|
|
|
parent::setUp();
|
|
|
|
|
2014-04-19 16:56:16 +04:00
|
|
|
$this->db = $this->getMockBuilder(
|
2015-02-24 18:47:38 +03:00
|
|
|
'\OCP\IDBConnection')
|
2014-04-19 16:56:16 +04:00
|
|
|
->disableOriginalConstructor()
|
|
|
|
->getMock();
|
|
|
|
|
2016-09-07 20:36:22 +03:00
|
|
|
$this->query = $this->createMock('\PDOStatement');
|
2014-04-19 16:56:16 +04:00
|
|
|
$this->queryAt = 0;
|
|
|
|
$this->prepareAt = 0;
|
2015-02-24 18:47:38 +03:00
|
|
|
$this->iterators = [];
|
2014-04-19 16:56:16 +04:00
|
|
|
$this->fetchAt = 0;
|
|
|
|
}
|
|
|
|
|
2015-03-19 16:55:21 +03:00
|
|
|
/**
|
|
|
|
* Checks if an array is associative
|
|
|
|
* @param array $array
|
|
|
|
* @return bool true if associative
|
|
|
|
*/
|
|
|
|
private function isAssocArray(array $array) {
|
|
|
|
return array_values($array) !== $array;
|
|
|
|
}
|
2014-04-19 16:56:16 +04:00
|
|
|
|
2015-03-19 16:55:21 +03:00
|
|
|
/**
|
|
|
|
* Returns the correct PDO constant based on the value type
|
|
|
|
* @param $value
|
2016-02-29 11:44:40 +03:00
|
|
|
* @return int PDO constant
|
2015-03-19 16:55:21 +03:00
|
|
|
*/
|
|
|
|
private function getPDOType($value) {
|
|
|
|
switch (gettype($value)) {
|
|
|
|
case 'integer':
|
|
|
|
return \PDO::PARAM_INT;
|
|
|
|
case 'boolean':
|
|
|
|
return \PDO::PARAM_BOOL;
|
|
|
|
default:
|
|
|
|
return \PDO::PARAM_STR;
|
|
|
|
}
|
|
|
|
}
|
2015-02-24 19:03:56 +03:00
|
|
|
|
2015-02-24 19:05:04 +03:00
|
|
|
/**
|
|
|
|
* Create mocks and set expected results for database queries
|
|
|
|
* @param string $sql the sql query that you expect to receive
|
|
|
|
* @param array $arguments the expected arguments for the prepare query
|
|
|
|
* method
|
|
|
|
* @param array $returnRows the rows that should be returned for the result
|
|
|
|
* of the database query. If not provided, it wont be assumed that fetch
|
|
|
|
* will be called on the result
|
|
|
|
*/
|
2020-10-05 16:12:57 +03:00
|
|
|
protected function setMapperResult($sql, $arguments = [], $returnRows = [],
|
|
|
|
$limit = null, $offset = null, $expectClose = false) {
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($limit === null && $offset === null) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->db->expects($this->at($this->prepareAt))
|
|
|
|
->method('prepare')
|
|
|
|
->with($this->equalTo($sql))
|
|
|
|
->will(($this->returnValue($this->query)));
|
2020-04-10 15:19:56 +03:00
|
|
|
} elseif ($limit !== null && $offset === null) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->db->expects($this->at($this->prepareAt))
|
|
|
|
->method('prepare')
|
|
|
|
->with($this->equalTo($sql), $this->equalTo($limit))
|
|
|
|
->will(($this->returnValue($this->query)));
|
2020-04-10 15:19:56 +03:00
|
|
|
} elseif ($limit === null && $offset !== null) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->db->expects($this->at($this->prepareAt))
|
|
|
|
->method('prepare')
|
|
|
|
->with($this->equalTo($sql),
|
|
|
|
$this->equalTo(null),
|
|
|
|
$this->equalTo($offset))
|
|
|
|
->will(($this->returnValue($this->query)));
|
2020-04-10 15:19:56 +03:00
|
|
|
} else {
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->db->expects($this->at($this->prepareAt))
|
|
|
|
->method('prepare')
|
|
|
|
->with($this->equalTo($sql),
|
|
|
|
$this->equalTo($limit),
|
|
|
|
$this->equalTo($offset))
|
|
|
|
->will(($this->returnValue($this->query)));
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->iterators[] = new ArgumentIterator($returnRows);
|
|
|
|
|
|
|
|
$iterators = $this->iterators;
|
|
|
|
$fetchAt = $this->fetchAt;
|
|
|
|
|
|
|
|
$this->query->expects($this->any())
|
|
|
|
->method('fetch')
|
2020-03-26 00:21:27 +03:00
|
|
|
->willReturnCallback(
|
2020-04-09 14:53:40 +03:00
|
|
|
function () use ($iterators, $fetchAt) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$iterator = $iterators[$fetchAt];
|
|
|
|
$result = $iterator->next();
|
|
|
|
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($result === false) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$fetchAt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->queryAt++;
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
2020-03-26 00:21:27 +03:00
|
|
|
);
|
2015-02-24 19:05:04 +03:00
|
|
|
|
2015-03-19 16:55:21 +03:00
|
|
|
if ($this->isAssocArray($arguments)) {
|
2020-04-10 15:19:56 +03:00
|
|
|
foreach ($arguments as $key => $argument) {
|
2015-03-19 16:55:21 +03:00
|
|
|
$pdoConstant = $this->getPDOType($argument);
|
|
|
|
$this->query->expects($this->at($this->queryAt))
|
|
|
|
->method('bindValue')
|
|
|
|
->with($this->equalTo($key),
|
|
|
|
$this->equalTo($argument),
|
|
|
|
$this->equalTo($pdoConstant));
|
|
|
|
$this->queryAt++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$index = 1;
|
2020-04-10 15:19:56 +03:00
|
|
|
foreach ($arguments as $argument) {
|
2015-03-19 16:55:21 +03:00
|
|
|
$pdoConstant = $this->getPDOType($argument);
|
|
|
|
$this->query->expects($this->at($this->queryAt))
|
|
|
|
->method('bindValue')
|
|
|
|
->with($this->equalTo($index),
|
|
|
|
$this->equalTo($argument),
|
|
|
|
$this->equalTo($pdoConstant));
|
|
|
|
$index++;
|
|
|
|
$this->queryAt++;
|
2015-02-24 19:05:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->query->expects($this->at($this->queryAt))
|
|
|
|
->method('execute')
|
2020-10-05 16:12:57 +03:00
|
|
|
->willReturnCallback(function ($sql, $p = null, $o = null, $s = null) {
|
2020-03-26 00:21:27 +03:00
|
|
|
});
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->queryAt++;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if ($expectClose) {
|
|
|
|
$closing = $this->at($this->queryAt);
|
|
|
|
} else {
|
|
|
|
$closing = $this->any();
|
|
|
|
}
|
|
|
|
$this->query->expects($closing)->method('closeCursor');
|
|
|
|
$this->queryAt++;
|
|
|
|
|
|
|
|
$this->prepareAt++;
|
|
|
|
$this->fetchAt++;
|
|
|
|
}
|
2014-04-19 16:56:16 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class ArgumentIterator {
|
2015-02-24 19:05:04 +03:00
|
|
|
private $arguments;
|
2015-01-29 21:16:28 +03:00
|
|
|
|
2020-04-09 14:53:40 +03:00
|
|
|
public function __construct($arguments) {
|
2015-02-24 19:05:04 +03:00
|
|
|
$this->arguments = $arguments;
|
|
|
|
}
|
2015-01-29 21:16:28 +03:00
|
|
|
|
2020-04-09 14:53:40 +03:00
|
|
|
public function next() {
|
2015-02-24 19:05:04 +03:00
|
|
|
$result = array_shift($this->arguments);
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($result === null) {
|
2015-02-24 19:05:04 +03:00
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
}
|
2014-04-19 16:56:16 +04:00
|
|
|
}
|