1
1
Fork 0
mirror of https://code.mensbeam.com/MensBeam/Arsse.git synced 2025-01-07 08:22:41 +00:00
Arsse/lib/Db/AbstractDriver.php

136 lines
5.4 KiB
PHP
Raw Normal View History

<?php
declare(strict_types=1);
2017-03-28 04:12:12 +00:00
namespace JKingWeb\Arsse\Db;
2016-10-17 20:49:39 +00:00
use JKingWeb\DrUUID\UUID as UUID;
abstract class AbstractDriver implements Driver {
2017-02-16 20:29:42 +00:00
protected $transDepth = 0;
protected $transStatus = [];
public abstract function prepareArray($query, array $paramTypes): Statement;
2017-02-16 20:29:42 +00:00
public function schemaVersion(): int {
try {
2017-03-28 04:12:12 +00:00
return (int) $this->query("SELECT value from arsse_settings where key is schema_version")->getValue();
2017-03-09 22:14:26 +00:00
} catch(Exception $e) {
2017-02-16 20:29:42 +00:00
return 0;
}
}
public function begin(): Transaction {
return new Transaction($this);
}
public function savepointCreate(): int {
2017-03-28 04:12:12 +00:00
$this->exec("SAVEPOINT arsse_".(++$this->transDepth));
$this->transStatus[$this->transDepth] = self::TR_PEND;
return $this->transDepth;
2017-02-16 20:29:42 +00:00
}
public function savepointRelease(int $index = null): bool {
if(is_null($index)) $index = $this->transDepth;
if(array_key_exists($index, $this->transStatus)) {
switch($this->transStatus[$index]) {
case self::TR_PEND:
$this->exec("RELEASE SAVEPOINT arsse_".$index);
$this->transStatus[$index] = self::TR_COMMIT;
$a = $index;
while(++$a && $a <= $this->transDepth) {
if($this->transStatus[$a] <= self::TR_PEND) $this->transStatus[$a] = self::TR_PEND_COMMIT;
}
$out = true;
break;
case self::TR_PEND_COMMIT:
$this->transStatus[$index] = self::TR_COMMIT;
$out = true;
break;
case self::TR_PEND_ROLLBACK:
$this->transStatus[$index] = self::TR_COMMIT;
$out = false;
break;
case self::TR_COMMIT:
case self::TR_ROLLBACK:
throw new ExceptionSavepoint("stale", ['action' => "commit", 'index' => $index]);
default:
throw new Exception("unknownSavepointStatus", $this->transStatus[$index]);
}
if($index==$this->transDepth) {
while($this->transDepth > 0 && $this->transStatus[$this->transDepth] > self::TR_PEND) {
array_pop($this->transStatus);
$this->transDepth--;
}
}
return $out;
2017-02-16 20:29:42 +00:00
} else {
throw new ExceptionSavepoint("invalid", ['action' => "commit", 'index' => $index]);
2017-02-16 20:29:42 +00:00
}
}
public function savepointUndo(int $index = null): bool {
if(is_null($index)) $index = $this->transDepth;
if(array_key_exists($index, $this->transStatus)) {
switch($this->transStatus[$index]) {
case self::TR_PEND:
$this->exec("ROLLBACK TRANSACTION TO SAVEPOINT arsse_".$index);
$this->exec("RELEASE SAVEPOINT arsse_".$index);
$this->transStatus[$index] = self::TR_ROLLBACK;
$a = $index;
while(++$a && $a <= $this->transDepth) {
if($this->transStatus[$a] <= self::TR_PEND) $this->transStatus[$a] = self::TR_PEND_ROLLBACK;
}
$out = true;
break;
case self::TR_PEND_COMMIT:
$this->transStatus[$index] = self::TR_ROLLBACK;
$out = false;
break;
case self::TR_PEND_ROLLBACK:
$this->transStatus[$index] = self::TR_ROLLBACK;
$out = true;
break;
case self::TR_COMMIT:
case self::TR_ROLLBACK:
throw new ExceptionSavepoint("stale", ['action' => "rollback", 'index' => $index]);
default:
throw new Exception("unknownSavepointStatus", $this->transStatus[$index]);
}
if($index==$this->transDepth) {
while($this->transDepth > 0 && $this->transStatus[$this->transDepth] > self::TR_PEND) {
array_pop($this->transStatus);
$this->transDepth--;
}
}
return $out;
2017-02-16 20:29:42 +00:00
} else {
throw new ExceptionSavepoint("invalid", ['action' => "rollback", 'index' => $index]);
2017-02-16 20:29:42 +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();
2017-03-09 22:14:26 +00:00
try {
2017-03-28 04:12:12 +00:00
$this->prepare("INSERT INTO arsse_settings(key,value) values(?,?)", "str", "str")->run("lock", $uuid);
2017-03-09 22:14:26 +00:00
} catch(ExceptionInput $e) {
return false;
}
2017-02-16 20:29:42 +00:00
sleep(1);
2017-03-28 04:12:12 +00:00
return ($this->query("SELECT value from arsse_settings where key is 'lock'")->getValue() == $uuid);
2017-02-16 20:29:42 +00:00
}
2016-10-17 20:49:39 +00:00
2017-02-16 20:29:42 +00:00
public function unlock(): bool {
2017-03-10 02:39:42 +00:00
if($this->schemaVersion() < 1) return true;
2017-03-28 04:12:12 +00:00
$this->exec("DELETE from arsse_settings where key is 'lock'");
2017-03-09 22:14:26 +00:00
return true;
2017-02-16 20:29:42 +00:00
}
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;
2017-03-28 04:12:12 +00:00
return ($this->query("SELECT count(*) from arsse_settings where key is 'lock'")->getValue() > 0);
2017-02-16 20:29:42 +00:00
}
2016-10-17 20:49:39 +00:00
public function prepare($query, ...$paramType): Statement {
2017-02-16 20:29:42 +00:00
return $this->prepareArray($query, $paramType);
}
}