2017-09-05 09:36:10 +02:00
|
|
|
<?php
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
use PHPUnit\DbUnit\Database\Connection;
|
|
|
|
use PHPUnit\DbUnit\DataSet\IDataSet;
|
|
|
|
|
2019-05-15 12:03:55 +02:00
|
|
|
if (!defined("ROOT_DIR")) { define("ROOT_DIR", __DIR__ . "/../.."); }
|
2020-02-20 17:08:47 +01:00
|
|
|
require_once(__DIR__ . "/../../helpers/config.php");
|
2019-05-15 12:03:55 +02:00
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
abstract class BaseDatabaseTestCase extends PHPUnit\DbUnit\TestCase {
|
2017-09-05 09:36:10 +02:00
|
|
|
|
2019-05-15 12:03:55 +02:00
|
|
|
/**
|
|
|
|
* @var PDO $pdo
|
|
|
|
*/
|
2020-02-17 18:51:27 +01:00
|
|
|
static private $pdo;
|
2019-05-15 12:03:55 +02:00
|
|
|
/**
|
2021-04-22 19:47:52 +02:00
|
|
|
* @var PHPUnit\DbUnit\Database\Connection $conn
|
2019-05-15 12:03:55 +02:00
|
|
|
*/
|
2020-02-17 18:51:27 +01:00
|
|
|
private $conn;
|
2019-01-24 19:07:41 +01:00
|
|
|
static private $driver = "mysql";
|
2017-09-05 09:36:10 +02:00
|
|
|
|
2020-02-20 17:08:47 +01:00
|
|
|
protected $mockConfig;
|
|
|
|
|
2017-09-05 09:36:10 +02:00
|
|
|
protected $testUser = "testUser";
|
|
|
|
protected $testUser2 = "testUser2";
|
|
|
|
protected $testAdminUser = "admin";
|
|
|
|
protected $testAdminPass = "admin";
|
|
|
|
protected $testPass = "testPass1234567890-;";
|
|
|
|
protected $testUserId = 1;
|
|
|
|
protected $testUserId2 = 2;
|
|
|
|
protected $testUserId3 = 3;
|
|
|
|
protected $testTrackId = 1;
|
|
|
|
protected $testTrackId2 = 2;
|
|
|
|
protected $testTrackName = "test track";
|
|
|
|
protected $testTrackComment = "test track comment";
|
|
|
|
protected $testTimestamp = 1502974402;
|
2020-02-17 18:51:27 +01:00
|
|
|
protected $testLat = 0.0;
|
2017-09-19 22:04:40 +02:00
|
|
|
protected $testLon = 10.604001083;
|
2017-09-05 09:36:10 +02:00
|
|
|
protected $testAltitude = 10.01;
|
|
|
|
protected $testSpeed = 10.01;
|
|
|
|
protected $testBearing = 10.01;
|
|
|
|
protected $testAccuracy = 10;
|
|
|
|
protected $testProvider = "gps";
|
|
|
|
protected $testComment = "test comment";
|
2019-07-12 21:50:21 +02:00
|
|
|
protected $testImage = "1234_1502974402_5d1a1960335cf.jpg";
|
2017-09-05 09:36:10 +02:00
|
|
|
|
2019-01-24 19:07:41 +01:00
|
|
|
// Fixes PostgreSQL: "cannot truncate a table referenced in a foreign key constraint"
|
|
|
|
protected function getSetUpOperation() {
|
2021-04-22 19:47:52 +02:00
|
|
|
return PHPUnit\DbUnit\Operation\Factory::CLEAN_INSERT(true);
|
2019-01-24 19:07:41 +01:00
|
|
|
}
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
public function setUp(): void {
|
2019-01-24 19:07:41 +01:00
|
|
|
parent::setUp();
|
2020-02-20 17:08:47 +01:00
|
|
|
$this->mockConfig = new uConfig(false);
|
2019-01-24 19:07:41 +01:00
|
|
|
}
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
public static function setUpBeforeClass(): void {
|
2017-09-05 09:36:10 +02:00
|
|
|
if (file_exists(__DIR__ . '/../.env')) {
|
2021-04-22 21:44:03 +02:00
|
|
|
$dotenv = Dotenv\Dotenv::createUnsafeImmutable(__DIR__ . '/..');
|
2017-09-05 09:36:10 +02:00
|
|
|
$dotenv->load();
|
2019-01-24 19:07:41 +01:00
|
|
|
$dotenv->required(['DB_DSN', 'DB_USER', 'DB_PASS']);
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
|
2021-04-22 21:44:03 +02:00
|
|
|
$db_dsn = getenv('DB_DSN');
|
|
|
|
$db_user = getenv('DB_USER');
|
|
|
|
$db_pass = getenv('DB_PASS');
|
2017-09-05 09:36:10 +02:00
|
|
|
|
|
|
|
// pdo connection
|
|
|
|
if (self::$pdo == null) {
|
2019-01-24 19:07:41 +01:00
|
|
|
self::$pdo = new PDO($db_dsn, $db_user, $db_pass);
|
|
|
|
self::$driver = self::$pdo->getAttribute(PDO::ATTR_DRIVER_NAME);
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
public static function tearDownAfterClass(): void {
|
2017-09-05 09:36:10 +02:00
|
|
|
self::$pdo = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set up database connection
|
|
|
|
* This will also override uDb class connection
|
|
|
|
*
|
2021-04-22 19:47:52 +02:00
|
|
|
* @return Connection
|
2017-09-05 09:36:10 +02:00
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
public function getConnection(): Connection {
|
2017-09-05 09:36:10 +02:00
|
|
|
if ($this->conn === null) {
|
|
|
|
$this->conn = $this->createDefaultDBConnection(self::$pdo, getenv('DB_NAME'));
|
|
|
|
}
|
|
|
|
return $this->conn;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create data set from xml fixture
|
|
|
|
*
|
2021-04-22 19:47:52 +02:00
|
|
|
* @return PHPUnit\DbUnit\DataSet\IDataSet
|
2017-09-05 09:36:10 +02:00
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function getDataSet(): IDataSet {
|
2019-01-28 23:03:03 +01:00
|
|
|
$this->resetAutoincrement();
|
|
|
|
return $this->createFlatXMLDataSet(__DIR__ . '/../fixtures/fixture_empty.xml');
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function resetAutoincrement($users = 1, $tracks = 1, $positions = 1, $layers = 1): void {
|
2020-02-17 18:51:27 +01:00
|
|
|
if (self::$driver === "pgsql") {
|
2020-05-20 17:12:07 +02:00
|
|
|
self::$pdo->exec("ALTER SEQUENCE IF EXISTS users_id_seq RESTART WITH $users");
|
|
|
|
self::$pdo->exec("ALTER SEQUENCE IF EXISTS tracks_id_seq RESTART WITH $tracks");
|
|
|
|
self::$pdo->exec("ALTER SEQUENCE IF EXISTS positions_id_seq RESTART WITH $positions");
|
|
|
|
self::$pdo->exec("ALTER SEQUENCE IF EXISTS ol_layers_id_seq RESTART WITH $layers");
|
2020-02-17 18:51:27 +01:00
|
|
|
} else if (self::$driver === "sqlite") {
|
2019-05-15 12:03:55 +02:00
|
|
|
$retry = 1;
|
|
|
|
do {
|
|
|
|
try {
|
2020-02-17 18:51:27 +01:00
|
|
|
self::$pdo->exec("DELETE FROM sqlite_sequence WHERE NAME = 'users'");
|
|
|
|
self::$pdo->exec("DELETE FROM sqlite_sequence WHERE NAME = 'tracks'");
|
|
|
|
self::$pdo->exec("DELETE FROM sqlite_sequence WHERE NAME = 'positions'");
|
2020-02-19 18:42:44 +01:00
|
|
|
self::$pdo->exec("DELETE FROM sqlite_sequence WHERE NAME = 'ol_layers'");
|
2019-05-15 12:03:55 +02:00
|
|
|
$retry = 0;
|
|
|
|
} catch (Exception $e) {
|
|
|
|
// sqlite raises error when db schema changes in another connection.
|
|
|
|
if (strpos($e->getMessage(), 'database schema has changed') !== false) {
|
|
|
|
self::$pdo = null;
|
|
|
|
self::setUpBeforeClass();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while ($retry--);
|
2019-01-24 19:07:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-20 17:12:07 +02:00
|
|
|
/**
|
|
|
|
* Reset connection
|
|
|
|
* Fixes sqlite error when db schema changes in another connection.
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function resetConnection(): void {
|
2020-05-20 17:12:07 +02:00
|
|
|
$this->closeConnection($this->conn);
|
|
|
|
$this->conn = null;
|
|
|
|
self::tearDownAfterClass();
|
|
|
|
self::setUpBeforeClass();
|
|
|
|
}
|
|
|
|
|
2017-09-05 09:36:10 +02:00
|
|
|
/**
|
|
|
|
* Insert to database from array
|
|
|
|
*
|
|
|
|
* @param string $table Table name
|
|
|
|
* @param array $rowsArr Array of rows
|
|
|
|
* @return int|null Last insert id if available, NULL otherwise
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
private function pdoInsert(string $table, array $rowsArr = []): ?int {
|
2017-09-05 09:36:10 +02:00
|
|
|
$ret = NULL;
|
|
|
|
if (!empty($rowsArr)) {
|
|
|
|
$values = ':' . implode(', :', array_keys($rowsArr));
|
|
|
|
$columns = implode(', ', array_keys($rowsArr));
|
|
|
|
$query = "INSERT INTO $table ($columns) VALUES ($values)";
|
|
|
|
$stmt = self::$pdo->prepare($query);
|
|
|
|
if ($stmt !== false) {
|
|
|
|
$stmt->execute(array_combine(explode(', ', $values), array_values($rowsArr)));
|
|
|
|
}
|
|
|
|
$ret = self::$pdo->lastInsertId();
|
|
|
|
}
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
2017-09-19 22:04:40 +02:00
|
|
|
/**
|
|
|
|
* Execute raw insert query on database
|
|
|
|
*
|
|
|
|
* @param string $query Insert query
|
|
|
|
* @return int|null Last insert id if available, NULL otherwise
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
private function pdoInsertRaw(string $query): ?int {
|
|
|
|
$ret = null;
|
2017-09-19 22:04:40 +02:00
|
|
|
if (self::$pdo->exec($query) !== false) {
|
|
|
|
$ret = self::$pdo->lastInsertId();
|
|
|
|
}
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
2017-09-05 09:36:10 +02:00
|
|
|
/**
|
|
|
|
* Get single column from first row of query result
|
|
|
|
*
|
|
|
|
* @param string $query SQL query
|
|
|
|
* @param int $columnNumber Optional column number (default is first column)
|
|
|
|
* @return string|bool Column or false if no data
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function pdoGetColumn(string $query, int $columnNumber = 0) {
|
2017-09-05 09:36:10 +02:00
|
|
|
$column = false;
|
|
|
|
$stmt = self::$pdo->query($query);
|
|
|
|
if ($stmt !== false) {
|
|
|
|
$column = $stmt->fetchColumn($columnNumber);
|
|
|
|
$stmt->closeCursor();
|
|
|
|
}
|
|
|
|
return $column;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert user data to database
|
|
|
|
* If parameters are omitted they default test values are used
|
|
|
|
*
|
2021-04-22 19:47:52 +02:00
|
|
|
* @param string|null $user User login
|
|
|
|
* @param string|null $pass User password
|
2020-02-17 18:51:27 +01:00
|
|
|
* @param bool $isAdmin User is admin
|
2017-09-05 09:36:10 +02:00
|
|
|
* @return int|bool User id or false on error
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function addTestUser(?string $user = null, ?string $pass = null, bool $isAdmin = false) {
|
2017-09-05 09:36:10 +02:00
|
|
|
if (is_null($user)) { $user = $this->testUser; }
|
|
|
|
if (is_null($pass)) { $pass = $this->testPass; }
|
2020-02-17 18:51:27 +01:00
|
|
|
$id = $this->pdoInsert('users', [ 'login' => $user, 'password' => $pass, 'admin' => (int) $isAdmin ]);
|
2020-01-07 17:21:40 +01:00
|
|
|
if ($id !== false) {
|
|
|
|
return (int) $id;
|
|
|
|
}
|
|
|
|
return false;
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert track data to database.
|
|
|
|
* If parameters are omitted they default test values are used
|
|
|
|
*
|
2021-04-22 19:47:52 +02:00
|
|
|
* @param int|null $userId Optional track id
|
|
|
|
* @param string|null $trackName Optional track name
|
|
|
|
* @param string|null $comment Optional comment
|
2017-09-05 09:36:10 +02:00
|
|
|
* @return int|bool Track id or false on error
|
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function addTestTrack(?int $userId = null, ?string $trackName = null, ?string $comment = null) {
|
2017-09-05 09:36:10 +02:00
|
|
|
if (is_null($userId)) { $userId = $this->testUserId; }
|
|
|
|
if (is_null($trackName)) { $trackName = $this->testTrackName; }
|
2017-09-19 22:04:40 +02:00
|
|
|
if (is_null($comment)) { $comment = $this->testTrackComment; }
|
2020-01-07 17:21:40 +01:00
|
|
|
$id = $this->pdoInsert('tracks', [ 'user_id' => $userId, 'name' => $trackName, 'comment' => $comment ]);
|
|
|
|
if ($id !== false) {
|
|
|
|
return (int) $id;
|
|
|
|
}
|
|
|
|
return false;
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert position data to database
|
|
|
|
* If parameters are omitted they default test values are used
|
|
|
|
*
|
2021-04-22 19:47:52 +02:00
|
|
|
* @param int|null $userId
|
|
|
|
* @param int|null $trackId
|
|
|
|
* @param int|null $timeStamp
|
|
|
|
* @param float|null $latitude
|
|
|
|
* @param float|null $longitude
|
|
|
|
* @return int|null Position id or false on error
|
2017-09-05 09:36:10 +02:00
|
|
|
*/
|
2021-04-22 19:47:52 +02:00
|
|
|
protected function addTestPosition(?int $userId = null, ?int $trackId = null, ?int $timeStamp = null, ?float $latitude = null, ?float $longitude = null): ?int {
|
2017-09-05 09:36:10 +02:00
|
|
|
if (is_null($userId)) { $userId = $this->testUserId; }
|
|
|
|
if (is_null($trackId)) { $trackId = $this->testTrackId; }
|
|
|
|
if (is_null($timeStamp)) { $timeStamp = $this->testTimestamp; }
|
|
|
|
if (is_null($latitude)) { $latitude = $this->testLat; }
|
|
|
|
if (is_null($longitude)) { $longitude = $this->testLon; }
|
|
|
|
|
2017-09-19 22:04:40 +02:00
|
|
|
$query = "INSERT INTO positions (user_id, track_id, time, latitude, longitude)
|
2019-01-24 19:07:41 +01:00
|
|
|
VALUES ('$userId', '$trackId', " . $this->from_unixtime($timeStamp) . ", '$latitude', '$longitude')";
|
2017-09-19 22:04:40 +02:00
|
|
|
return $this->pdoInsertRaw($query);
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
2019-01-24 19:07:41 +01:00
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
public function unix_timestamp(string $column): string {
|
2019-01-24 19:07:41 +01:00
|
|
|
switch (self::$driver) {
|
|
|
|
default:
|
|
|
|
case "mysql":
|
|
|
|
return "UNIX_TIMESTAMP($column)";
|
|
|
|
case "pgsql":
|
|
|
|
return "EXTRACT(EPOCH FROM $column)";
|
|
|
|
case "sqlite":
|
|
|
|
return "STRFTIME('%s', $column)";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-22 19:47:52 +02:00
|
|
|
public function from_unixtime(string $column): string {
|
2019-01-24 19:07:41 +01:00
|
|
|
switch (self::$driver) {
|
|
|
|
default:
|
|
|
|
case "mysql":
|
|
|
|
return "FROM_UNIXTIME($column)";
|
|
|
|
case "pgsql":
|
|
|
|
return "TO_TIMESTAMP($column)";
|
|
|
|
case "sqlite":
|
2019-01-28 23:03:03 +01:00
|
|
|
return "DATETIME($column, 'unixepoch')";
|
2019-01-24 19:07:41 +01:00
|
|
|
}
|
|
|
|
}
|
2017-09-05 09:36:10 +02:00
|
|
|
}
|
|
|
|
?>
|