2016-10-05 22:08:43 -04:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
2017-03-27 23:12:12 -05:00
|
|
|
namespace JKingWeb\Arsse\Db;
|
2016-10-17 16:49:39 -04:00
|
|
|
use JKingWeb\DrUUID\UUID as UUID;
|
2016-10-05 22:08:43 -04:00
|
|
|
|
2017-03-02 20:47:00 -05:00
|
|
|
abstract class AbstractDriver implements Driver {
|
2017-02-16 14:29:42 -06:00
|
|
|
protected $transDepth = 0;
|
2017-05-07 18:27:16 -04:00
|
|
|
protected $transStatus = [];
|
2017-02-19 16:02:03 -06:00
|
|
|
|
2017-06-04 18:00:18 -04:00
|
|
|
public abstract function prepareArray($query, array $paramTypes): Statement;
|
|
|
|
|
2017-02-16 14:29:42 -06:00
|
|
|
public function schemaVersion(): int {
|
|
|
|
try {
|
2017-03-27 23:12:12 -05:00
|
|
|
return (int) $this->query("SELECT value from arsse_settings where key is schema_version")->getValue();
|
2017-03-09 17:14:26 -05:00
|
|
|
} catch(Exception $e) {
|
2017-02-16 14:29:42 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2017-02-19 16:02:03 -06:00
|
|
|
|
2017-05-06 12:02:27 -04:00
|
|
|
public function begin(): Transaction {
|
|
|
|
return new Transaction($this);
|
|
|
|
}
|
|
|
|
|
2017-05-07 18:27:16 -04:00
|
|
|
public function savepointCreate(): int {
|
2017-03-27 23:12:12 -05:00
|
|
|
$this->exec("SAVEPOINT arsse_".(++$this->transDepth));
|
2017-05-07 18:27:16 -04:00
|
|
|
$this->transStatus[$this->transDepth] = self::TR_PEND;
|
|
|
|
return $this->transDepth;
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
2016-10-05 22:08:43 -04:00
|
|
|
|
2017-05-07 18:27:16 -04: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 14:29:42 -06:00
|
|
|
} else {
|
2017-05-07 18:27:16 -04:00
|
|
|
throw new ExceptionSavepoint("invalid", ['action' => "commit", 'index' => $index]);
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
|
|
|
}
|
2016-10-05 22:08:43 -04:00
|
|
|
|
2017-05-07 18:27:16 -04: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 14:29:42 -06:00
|
|
|
} else {
|
2017-05-07 18:27:16 -04:00
|
|
|
throw new ExceptionSavepoint("invalid", ['action' => "rollback", 'index' => $index]);
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
|
|
|
}
|
2016-10-05 22:08:43 -04:00
|
|
|
|
2017-02-16 14:29:42 -06:00
|
|
|
public function lock(): bool {
|
|
|
|
if($this->schemaVersion() < 1) return true;
|
|
|
|
if($this->isLocked()) return false;
|
|
|
|
$uuid = UUID::mintStr();
|
2017-03-09 17:14:26 -05:00
|
|
|
try {
|
2017-03-27 23:12:12 -05:00
|
|
|
$this->prepare("INSERT INTO arsse_settings(key,value) values(?,?)", "str", "str")->run("lock", $uuid);
|
2017-03-09 17:14:26 -05:00
|
|
|
} catch(ExceptionInput $e) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-02-16 14:29:42 -06:00
|
|
|
sleep(1);
|
2017-03-27 23:12:12 -05:00
|
|
|
return ($this->query("SELECT value from arsse_settings where key is 'lock'")->getValue() == $uuid);
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
2016-10-17 16:49:39 -04:00
|
|
|
|
2017-02-16 14:29:42 -06:00
|
|
|
public function unlock(): bool {
|
2017-03-09 21:39:42 -05:00
|
|
|
if($this->schemaVersion() < 1) return true;
|
2017-03-27 23:12:12 -05:00
|
|
|
$this->exec("DELETE from arsse_settings where key is 'lock'");
|
2017-03-09 17:14:26 -05:00
|
|
|
return true;
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
2016-10-17 16:49:39 -04:00
|
|
|
|
2017-02-16 14:29:42 -06:00
|
|
|
public function isLocked(): bool {
|
|
|
|
if($this->schemaVersion() < 1) return false;
|
2017-03-27 23:12:12 -05:00
|
|
|
return ($this->query("SELECT count(*) from arsse_settings where key is 'lock'")->getValue() > 0);
|
2017-02-16 14:29:42 -06:00
|
|
|
}
|
2016-10-17 16:49:39 -04:00
|
|
|
|
2017-06-04 18:00:18 -04:00
|
|
|
public function prepare($query, ...$paramType): Statement {
|
2017-02-16 14:29:42 -06:00
|
|
|
return $this->prepareArray($query, $paramType);
|
|
|
|
}
|
2016-10-05 22:08:43 -04:00
|
|
|
}
|