2016-10-06 02:08:43 +00:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace JKingWeb\NewsSync\Db;
|
2016-10-17 20:49:39 +00:00
|
|
|
use JKingWeb\DrUUID\UUID as UUID;
|
2016-10-06 02:08:43 +00:00
|
|
|
|
|
|
|
Trait Common {
|
2017-02-16 20:29:42 +00:00
|
|
|
protected $transDepth = 0;
|
2017-02-19 22:02:03 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function schemaVersion(): int {
|
|
|
|
try {
|
|
|
|
return $this->data->db->settingGet("schema_version");
|
|
|
|
} catch(\Throwable $e) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2017-02-19 22:02:03 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function begin(): bool {
|
|
|
|
$this->exec("SAVEPOINT newssync_".($this->transDepth));
|
|
|
|
$this->transDepth += 1;
|
|
|
|
return true;
|
|
|
|
}
|
2016-10-06 02:08:43 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function commit(bool $all = false): bool {
|
|
|
|
if($this->transDepth==0) return false;
|
|
|
|
if(!$all) {
|
|
|
|
$this->exec("RELEASE SAVEPOINT newssync_".($this->transDepth - 1));
|
|
|
|
$this->transDepth -= 1;
|
|
|
|
} else {
|
|
|
|
$this->exec("COMMIT TRANSACTION");
|
2017-02-19 22:02:03 +00:00
|
|
|
$this->transDepth = 0;
|
2017-02-16 20:29:42 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2016-10-06 02:08:43 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function rollback(bool $all = false): bool {
|
|
|
|
if($this->transDepth==0) return false;
|
|
|
|
if(!$all) {
|
|
|
|
$this->exec("ROLLBACK TRANSACTION TO SAVEPOINT newssync_".($this->transDepth - 1));
|
|
|
|
// rollback to savepoint does not collpase the savepoint
|
|
|
|
$this->commit();
|
|
|
|
$this->transDepth -= 1;
|
|
|
|
if($this->transDepth==0) $this->exec("ROLLBACK TRANSACTION");
|
|
|
|
} else {
|
|
|
|
$this->exec("ROLLBACK TRANSACTION");
|
|
|
|
$this->transDepth = 0;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2016-10-06 02:08:43 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function lock(): bool {
|
|
|
|
if($this->schemaVersion() < 1) return true;
|
|
|
|
if($this->isLocked()) return false;
|
|
|
|
$uuid = UUID::mintStr();
|
|
|
|
if(!$this->data->db->settingSet("lock", $uuid)) return false;
|
|
|
|
sleep(1);
|
|
|
|
if($this->data->db->settingGet("lock") != $uuid) return false;
|
|
|
|
return true;
|
|
|
|
}
|
2016-10-17 20:49:39 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function unlock(): bool {
|
|
|
|
return $this->data->db->settingRemove("lock");
|
|
|
|
}
|
2016-10-17 20:49:39 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function isLocked(): bool {
|
|
|
|
if($this->schemaVersion() < 1) return false;
|
|
|
|
return ($this->query("SELECT count(*) from newssync_settings where key = 'lock'")->getSingle() > 0);
|
|
|
|
}
|
2016-10-17 20:49:39 +00:00
|
|
|
|
2017-02-16 20:29:42 +00:00
|
|
|
public function prepare(string $query, string ...$paramType): Statement {
|
|
|
|
return $this->prepareArray($query, $paramType);
|
|
|
|
}
|
2016-10-06 02:08:43 +00:00
|
|
|
|
2017-03-02 19:19:16 +00:00
|
|
|
public static function formatDate($date, int $precision = self::TS_BOTH): string {
|
2017-02-19 22:02:03 +00:00
|
|
|
// Force UTC.
|
|
|
|
$timezone = date_default_timezone_get();
|
|
|
|
date_default_timezone_set('UTC');
|
2017-03-02 19:19:16 +00:00
|
|
|
// convert input to a Unix timestamp
|
|
|
|
// FIXME: there are more kinds of date representations
|
|
|
|
if(is_int($date)) {
|
|
|
|
$time = $date;
|
|
|
|
} else if($date===null) {
|
|
|
|
$time = 0;
|
|
|
|
} else {
|
|
|
|
$time = strtotime($date);
|
|
|
|
}
|
2017-02-19 22:02:03 +00:00
|
|
|
// ISO 8601 with space in the middle instead of T.
|
2017-03-02 19:19:16 +00:00
|
|
|
$date = date(self::TS_FORMAT[$precision], $time);
|
2017-02-19 22:02:03 +00:00
|
|
|
date_default_timezone_set($timezone);
|
|
|
|
return $date;
|
|
|
|
}
|
2016-10-06 02:08:43 +00:00
|
|
|
}
|