2016-12-30 06:21:17 +01:00
|
|
|
<?php
|
2016-09-14 16:27:39 +02:00
|
|
|
|
|
|
|
namespace Amp\Postgres;
|
|
|
|
|
2017-06-21 05:17:53 +02:00
|
|
|
use Amp\CallableMaker;
|
|
|
|
use Amp\CancellationToken;
|
2017-07-27 07:32:34 +02:00
|
|
|
use Amp\Coroutine;
|
|
|
|
use Amp\Deferred;
|
2017-06-21 05:17:53 +02:00
|
|
|
use Amp\Promise;
|
|
|
|
use function Amp\call;
|
2016-09-14 16:27:39 +02:00
|
|
|
|
2017-08-02 06:01:55 +02:00
|
|
|
abstract class AbstractConnection implements Connection {
|
2016-09-14 16:27:39 +02:00
|
|
|
use CallableMaker;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-21 07:18:24 +02:00
|
|
|
/** @var \Amp\Postgres\Executor */
|
2017-08-01 07:38:12 +02:00
|
|
|
private $handle;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-21 07:18:24 +02:00
|
|
|
/** @var \Amp\Deferred|null Used to only allow one transaction at a time. */
|
2016-09-14 16:27:39 +02:00
|
|
|
private $busy;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-14 16:27:39 +02:00
|
|
|
/** @var callable */
|
|
|
|
private $release;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-14 16:27:39 +02:00
|
|
|
/**
|
|
|
|
* @param string $connectionString
|
2017-06-05 06:42:18 +02:00
|
|
|
* @param \Amp\CancellationToken $token
|
2016-09-14 16:27:39 +02:00
|
|
|
*
|
2017-03-17 16:17:24 +01:00
|
|
|
* @return \Amp\Promise<\Amp\Postgres\Connection>
|
2016-09-14 16:27:39 +02:00
|
|
|
*/
|
2017-06-05 06:42:18 +02:00
|
|
|
abstract public static function connect(string $connectionString, CancellationToken $token = null): Promise;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-14 16:27:39 +02:00
|
|
|
/**
|
2017-08-01 07:38:12 +02:00
|
|
|
* @param \Amp\Postgres\Handle $handle
|
2016-09-14 16:27:39 +02:00
|
|
|
*/
|
2017-08-01 07:38:12 +02:00
|
|
|
public function __construct(Handle $handle) {
|
|
|
|
$this->handle = $handle;
|
2016-09-14 16:27:39 +02:00
|
|
|
$this->release = $this->callableFromInstanceMethod("release");
|
|
|
|
}
|
|
|
|
|
2017-11-05 22:38:17 +01:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function isAlive(): bool {
|
|
|
|
return $this->handle->isAlive();
|
|
|
|
}
|
|
|
|
|
2016-09-14 16:27:39 +02:00
|
|
|
/**
|
2017-06-05 05:15:50 +02:00
|
|
|
* @param string $methodName Method to execute.
|
2016-09-14 16:27:39 +02:00
|
|
|
* @param mixed ...$args Arguments to pass to function.
|
|
|
|
*
|
2017-05-26 20:14:04 +02:00
|
|
|
* @return \Amp\Promise
|
2016-09-14 16:27:39 +02:00
|
|
|
*
|
|
|
|
* @throws \Amp\Postgres\FailureException
|
|
|
|
*/
|
2017-07-27 07:32:34 +02:00
|
|
|
private function send(string $methodName, ...$args): \Generator {
|
|
|
|
while ($this->busy) {
|
|
|
|
yield $this->busy->promise();
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2017-11-05 18:24:38 +01:00
|
|
|
return $this->handle->{$methodName}(...$args);
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-21 07:18:24 +02:00
|
|
|
/**
|
|
|
|
* Releases the transaction lock.
|
|
|
|
*/
|
2016-09-14 16:27:39 +02:00
|
|
|
private function release() {
|
2017-11-18 04:33:49 +01:00
|
|
|
\assert($this->busy !== null);
|
|
|
|
|
2017-07-27 07:32:34 +02:00
|
|
|
$deferred = $this->busy;
|
|
|
|
$this->busy = null;
|
|
|
|
$deferred->resolve();
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-14 16:27:39 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 18:06:21 +01:00
|
|
|
public function query(string $sql): Promise {
|
2017-07-27 07:32:34 +02:00
|
|
|
return new Coroutine($this->send("query", $sql));
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2017-11-18 04:33:49 +01:00
|
|
|
public function execute(string $sql, array $params = []): Promise {
|
|
|
|
return new Coroutine($this->send("execute", $sql, $params));
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 18:06:21 +01:00
|
|
|
public function prepare(string $sql): Promise {
|
2017-07-27 07:32:34 +02:00
|
|
|
return new Coroutine($this->send("prepare", $sql));
|
2016-09-19 18:12:32 +02:00
|
|
|
}
|
2017-05-16 06:28:37 +02:00
|
|
|
|
|
|
|
|
2016-09-21 07:18:24 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 18:06:21 +01:00
|
|
|
public function notify(string $channel, string $payload = ""): Promise {
|
2017-07-27 07:32:34 +02:00
|
|
|
return new Coroutine($this->send("notify", $channel, $payload));
|
2016-09-21 07:18:24 +02:00
|
|
|
}
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2016-09-19 18:12:32 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 18:06:21 +01:00
|
|
|
public function listen(string $channel): Promise {
|
2017-07-27 07:32:34 +02:00
|
|
|
return new Coroutine($this->send("listen", $channel));
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
2016-11-15 18:06:21 +01:00
|
|
|
public function transaction(int $isolation = Transaction::COMMITTED): Promise {
|
2017-07-27 07:32:34 +02:00
|
|
|
return call(function () use ($isolation) {
|
|
|
|
switch ($isolation) {
|
|
|
|
case Transaction::UNCOMMITTED:
|
2017-08-01 07:38:12 +02:00
|
|
|
yield $this->handle->query("BEGIN TRANSACTION ISOLATION LEVEL READ UNCOMMITTED");
|
2017-07-27 07:32:34 +02:00
|
|
|
break;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2017-07-27 07:32:34 +02:00
|
|
|
case Transaction::COMMITTED:
|
2017-08-01 07:38:12 +02:00
|
|
|
yield $this->handle->query("BEGIN TRANSACTION ISOLATION LEVEL READ COMMITTED");
|
2017-07-27 07:32:34 +02:00
|
|
|
break;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2017-07-27 07:32:34 +02:00
|
|
|
case Transaction::REPEATABLE:
|
2017-08-01 07:38:12 +02:00
|
|
|
yield $this->handle->query("BEGIN TRANSACTION ISOLATION LEVEL REPEATABLE READ");
|
2017-07-27 07:32:34 +02:00
|
|
|
break;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2017-07-27 07:32:34 +02:00
|
|
|
case Transaction::SERIALIZABLE:
|
2017-08-01 07:38:12 +02:00
|
|
|
yield $this->handle->query("BEGIN TRANSACTION ISOLATION LEVEL SERIALIZABLE");
|
2017-07-27 07:32:34 +02:00
|
|
|
break;
|
2017-05-16 06:28:37 +02:00
|
|
|
|
2017-07-27 07:32:34 +02:00
|
|
|
default:
|
|
|
|
throw new \Error("Invalid transaction type");
|
|
|
|
}
|
2017-05-16 06:14:02 +02:00
|
|
|
|
2017-11-05 18:24:38 +01:00
|
|
|
$this->busy = new Deferred;
|
|
|
|
|
2017-08-01 07:38:12 +02:00
|
|
|
$transaction = new Transaction($this->handle, $isolation);
|
2017-11-18 05:00:52 +01:00
|
|
|
$transaction->onDestruct($this->release);
|
2016-09-14 16:27:39 +02:00
|
|
|
return $transaction;
|
|
|
|
});
|
|
|
|
}
|
2017-08-01 07:38:12 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function quoteString(string $data): string {
|
|
|
|
return $this->handle->quoteString($data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function quoteName(string $name): string {
|
|
|
|
return $this->handle->quoteName($name);
|
|
|
|
}
|
2016-09-14 16:27:39 +02:00
|
|
|
}
|