modifs.mineures sans commentaires
This commit is contained in:
parent
7a3b0e456d
commit
907f17af33
|
@ -5,59 +5,44 @@ namespace nur\sery\db;
|
||||||
* Class AbstractCapacitor: implémentation de base d'un {@link ICapacitor}
|
* Class AbstractCapacitor: implémentation de base d'un {@link ICapacitor}
|
||||||
*/
|
*/
|
||||||
abstract class AbstractCapacitor implements ICapacitor {
|
abstract class AbstractCapacitor implements ICapacitor {
|
||||||
abstract protected function getChannel(?string $name=null): CapacitorChannel;
|
abstract protected function getChannel(?string $name): CapacitorChannel;
|
||||||
|
|
||||||
abstract function _exists(CapacitorChannel $channel): bool;
|
abstract function _exists(CapacitorChannel $channel): bool;
|
||||||
|
|
||||||
/** tester si le canal spécifié existe */
|
/** tester si le canal spécifié existe */
|
||||||
function exists(?string $channel=null): bool {
|
function exists(?string $channel): bool {
|
||||||
return $this->_exists($this->getChannel($channel));
|
return $this->_exists($this->getChannel($channel));
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function _reset(CapacitorChannel $channel): void;
|
abstract function _reset(CapacitorChannel $channel): void;
|
||||||
|
|
||||||
/** supprimer le canal spécifié */
|
/** supprimer le canal spécifié */
|
||||||
function reset(?string $channel=null): void {
|
function reset(?string $channel): void {
|
||||||
$this->_reset($this->getChannel($channel));
|
$this->_reset($this->getChannel($channel));
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function _charge($item, CapacitorChannel $channel): void;
|
abstract function _charge(CapacitorChannel $channel, $item, ?callable $func, ?array $args): bool;
|
||||||
|
|
||||||
/** charger une valeur dans le canal */
|
function charge(?string $channel, $item, ?callable $func=null, ?array $args=null): bool {
|
||||||
function charge($item, ?string $channel=null): void {
|
return $this->_charge($this->getChannel($channel), $item, $func, $args);
|
||||||
$this->_charge($item, $this->getChannel($channel));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function _discharge($keys=null, CapacitorChannel $channel=null, ?bool $reset=null): iterable;
|
abstract function _discharge(CapacitorChannel $channel, $filter, ?bool $reset): iterable;
|
||||||
|
|
||||||
/** décharger les données du canal spécifié */
|
function discharge(?string $channel, $filter=null, ?bool $reset=null): iterable {
|
||||||
function discharge($keys=null, ?string $channel=null, ?bool $reset=null): iterable {
|
return $this->_discharge($this->getChannel($channel), $filter, $reset);
|
||||||
return $this->_discharge($keys, $this->getChannel($channel), $reset);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function _get($keys, CapacitorChannel $channel=null);
|
abstract function _get(CapacitorChannel $channel, $filter);
|
||||||
|
|
||||||
/**
|
function get(?string $channel, $filter) {
|
||||||
* obtenir l'élément identifié par les clés spécifiées sur le canal spécifié
|
return $this->_get($this->getChannel($channel), $filter);
|
||||||
*
|
|
||||||
* si $keys n'est pas un tableau, il est transformé en ["_id" => $keys]
|
|
||||||
*/
|
|
||||||
function get($keys, ?string $channel=null) {
|
|
||||||
return $this->_get($keys, $this->getChannel($channel));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function _each($keys, callable $func, ?array $args=null, CapacitorChannel $channel=null): void;
|
abstract function _each(CapacitorChannel $channel, $filter, callable $func, ?array $args): void;
|
||||||
|
|
||||||
/**
|
function each(?string $channel, $filter, callable $func, ?array $args=null): void {
|
||||||
* appeler une fonction pour chaque élément du canal spécifié.
|
$this->_each($this->getChannel($channel), $filter, $func, $args);
|
||||||
*
|
|
||||||
* $keys permet de filtrer parmi les élements chargés
|
|
||||||
*
|
|
||||||
* si $func retourne un tableau, il est utilisé pour mettre à jour
|
|
||||||
* l'enregistrement.
|
|
||||||
*/
|
|
||||||
function each($keys, callable $func, ?array $args=null, ?string $channel=null): void {
|
|
||||||
$this->_each($keys, $func, $args, $this->getChannel($channel));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract function close(): void;
|
abstract function close(): void;
|
||||||
|
|
|
@ -25,20 +25,20 @@ class Capacitor {
|
||||||
$this->capacitor->_reset($this->channel);
|
$this->capacitor->_reset($this->channel);
|
||||||
}
|
}
|
||||||
|
|
||||||
function charge($item) {
|
function charge($item, ?callable $func=null, ?array $args=null): bool {
|
||||||
$this->capacitor->_charge($item, $this->channel);
|
return $this->capacitor->_charge($this->channel, $item, $func, $args);
|
||||||
}
|
}
|
||||||
|
|
||||||
function discharge($keys=null, ?bool $reset=null): iterable {
|
function discharge($filter=null, ?bool $reset=null): iterable {
|
||||||
return $this->capacitor->_discharge($keys, $this->channel, $reset);
|
return $this->capacitor->_discharge($this->channel, $filter, $reset);
|
||||||
}
|
}
|
||||||
|
|
||||||
function get($keys) {
|
function get($filter) {
|
||||||
return $this->capacitor->_get($keys, $this->channel);
|
return $this->capacitor->_get($this->channel, $filter);
|
||||||
}
|
}
|
||||||
|
|
||||||
function each($keys, callable $func, ?array $args=null): void {
|
function each($filter, callable $func, ?array $args=null): void {
|
||||||
$this->capacitor->_each($keys, $func, $args, $this->channel);
|
$this->capacitor->_each($this->channel, $filter, $func, $args);
|
||||||
}
|
}
|
||||||
|
|
||||||
function close(): void {
|
function close(): void {
|
||||||
|
|
|
@ -38,10 +38,24 @@ class CapacitorChannel {
|
||||||
$this->created = $created;
|
$this->created = $created;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* retourner un ensemble de définitions pour des colonnes supplémentaires à
|
||||||
|
* insérer lors du chargement d'une valeur
|
||||||
|
*
|
||||||
|
* la colonne "_id" de définition "integer primary key autoincrement" est la
|
||||||
|
* clé primaire par défaut. elle peut être redéfinie, et dans ce cas la valeur
|
||||||
|
* à utiliser doit être retournée par {@link getKeyValues()}
|
||||||
|
*/
|
||||||
function getKeyDefinitions(): ?array {
|
function getKeyDefinitions(): ?array {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* calculer les valeurs des colonnes supplémentaires à insérer pour le
|
||||||
|
* chargement de $item
|
||||||
|
*
|
||||||
|
* Si "_id" est retourné, la ligne existante est mise à jour le cas échéant.
|
||||||
|
*/
|
||||||
function getKeyValues($item): ?array {
|
function getKeyValues($item): ?array {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,33 +7,44 @@ namespace nur\sery\db;
|
||||||
*/
|
*/
|
||||||
interface ICapacitor {
|
interface ICapacitor {
|
||||||
/** tester si le canal spécifié existe */
|
/** tester si le canal spécifié existe */
|
||||||
function exists(?string $channel=null): bool;
|
function exists(?string $channel): bool;
|
||||||
|
|
||||||
/** supprimer le canal spécifié */
|
/** supprimer le canal spécifié */
|
||||||
function reset(?string $channel=null): void;
|
function reset(?string $channel): void;
|
||||||
|
|
||||||
/** charger une valeur dans le canal */
|
/**
|
||||||
function charge($item, ?string $channel=null): void;
|
* charger une valeur dans le canal
|
||||||
|
*
|
||||||
|
* Si $func!==null, après avoir calculé les valeurs des clés supplémentaires
|
||||||
|
* avec {@link CapacitorChannel::getKeyValues()}, la fonction est appelée avec
|
||||||
|
* les arguments ($item, $keyValues, $row, ...$args)
|
||||||
|
* Si la fonction retourne un tableau, il est utilisé pour modifié les valeurs
|
||||||
|
* insérées/mises à jour
|
||||||
|
*
|
||||||
|
* @return true si l'objet a été chargé ou mis à jour, false s'il existait
|
||||||
|
* déjà à l'identique dans le canal
|
||||||
|
*/
|
||||||
|
function charge(?string $channel, $item, ?callable $func=null, ?array $args=null): bool;
|
||||||
|
|
||||||
/** décharger les données du canal spécifié */
|
/** décharger les données du canal spécifié */
|
||||||
function discharge($keys=null, ?string $channel=null, ?bool $reset=null): iterable;
|
function discharge(?string $channel, $filter=null, ?bool $reset=null): iterable;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* obtenir l'élément identifié par les clés spécifiées sur le canal spécifié
|
* obtenir l'élément identifié par les clés spécifiées sur le canal spécifié
|
||||||
*
|
*
|
||||||
* si $keys n'est pas un tableau, il est transformé en ["_id" => $keys]
|
* si $filter n'est pas un tableau, il est transformé en ["_id" => $filter]
|
||||||
*/
|
*/
|
||||||
function get($keys, ?string $channel=null);
|
function get(?string $channel, $filter);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* appeler une fonction pour chaque élément du canal spécifié.
|
* appeler une fonction pour chaque élément du canal spécifié.
|
||||||
*
|
*
|
||||||
* $keys permet de filtrer parmi les élements chargés
|
* $filter permet de filtrer parmi les élements chargés
|
||||||
*
|
*
|
||||||
* si $func retourne un tableau, il est utilisé pour mettre à jour
|
* si $func retourne un tableau, il est utilisé pour mettre à jour
|
||||||
* l'enregistrement.
|
* l'enregistrement.
|
||||||
*/
|
*/
|
||||||
function each($keys, callable $func, ?array $args=null, ?string $channel=null): void;
|
function each(?string $channel, $filter, callable $func, ?array $args=null): void;
|
||||||
|
|
||||||
function close(): void;
|
function close(): void;
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,6 +27,9 @@ class SqliteCapacitor extends AbstractCapacitor{
|
||||||
$columns = cl::merge([
|
$columns = cl::merge([
|
||||||
"_id" => "integer primary key autoincrement",
|
"_id" => "integer primary key autoincrement",
|
||||||
"_item" => "text",
|
"_item" => "text",
|
||||||
|
"_sum" => "varchar(40)",
|
||||||
|
"_created" => "datetime",
|
||||||
|
"_modified" => "datetime",
|
||||||
], $channel->getKeyDefinitions());
|
], $channel->getKeyDefinitions());
|
||||||
$this->sqlite->exec([
|
$this->sqlite->exec([
|
||||||
"create table if not exists",
|
"create table if not exists",
|
||||||
|
@ -46,7 +49,7 @@ class SqliteCapacitor extends AbstractCapacitor{
|
||||||
return $channel;
|
return $channel;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getChannel(?string $name=null): CapacitorChannel {
|
protected function getChannel(?string $name): CapacitorChannel {
|
||||||
$name = CapacitorChannel::verifix_name($name);
|
$name = CapacitorChannel::verifix_name($name);
|
||||||
$channel = $this->channels[$name] ?? null;
|
$channel = $this->channels[$name] ?? null;
|
||||||
if ($channel === null) {
|
if ($channel === null) {
|
||||||
|
@ -73,25 +76,81 @@ class SqliteCapacitor extends AbstractCapacitor{
|
||||||
$channel->setCreated(false);
|
$channel->setCreated(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _charge($item, CapacitorChannel $channel): void {
|
function _charge(CapacitorChannel $channel, $item, ?callable $func, ?array $args): bool {
|
||||||
$this->_create($channel);
|
$this->_create($channel);
|
||||||
$values = cl::merge($channel->getKeyValues($item), [
|
$now = date("Y-m-d H:i:s");
|
||||||
"_item" => serialize($item),
|
$_item = serialize($item);
|
||||||
|
$_sum = sha1($_item);
|
||||||
|
$values = cl::merge([
|
||||||
|
"_item" => $_item,
|
||||||
|
"_sum" => $_sum,
|
||||||
|
], $channel->getKeyValues($item));
|
||||||
|
$row = null;
|
||||||
|
$id = $values["_id"] ?? null;
|
||||||
|
if ($id !== null) {
|
||||||
|
# modification
|
||||||
|
$row = $this->sqlite->one([
|
||||||
|
"select _item, _sum, _created, _modified",
|
||||||
|
"from" => $channel->getTableName(),
|
||||||
|
"where" => ["_id" => $id],
|
||||||
]);
|
]);
|
||||||
|
}
|
||||||
|
$insert = null;
|
||||||
|
if ($row === null) {
|
||||||
|
# création
|
||||||
|
$values = cl::merge($values, [
|
||||||
|
"_created" => $now,
|
||||||
|
"_modified" => $now,
|
||||||
|
]);
|
||||||
|
$insert = true;
|
||||||
|
} elseif ($_sum !== $row["_sum"]) {
|
||||||
|
# modification
|
||||||
|
$values = cl::merge($values, [
|
||||||
|
"_modified" => $now,
|
||||||
|
]);
|
||||||
|
$insert = false;
|
||||||
|
}
|
||||||
|
if ($func !== null) {
|
||||||
|
$context = func::_prepare($func);
|
||||||
|
$args ??= [];
|
||||||
|
$updates = func::_call($context, [$item, $values, $row, ...$args]);
|
||||||
|
if (is_array($updates)) {
|
||||||
|
if (array_key_exists("_item", $updates)) {
|
||||||
|
$updates["_item"] = serialize($updates["_item"]);
|
||||||
|
if (!array_key_exists("_modified", $updates)) {
|
||||||
|
$updates["_modified"] = $now;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$values = cl::merge($values, $updates);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($insert === null) {
|
||||||
|
# aucune modification
|
||||||
|
return false;
|
||||||
|
} elseif ($insert) {
|
||||||
$this->sqlite->exec([
|
$this->sqlite->exec([
|
||||||
"insert",
|
"insert",
|
||||||
"into" => $channel->getTableName(),
|
"into" => $channel->getTableName(),
|
||||||
"values" => $values,
|
"values" => $values,
|
||||||
]);
|
]);
|
||||||
|
} else {
|
||||||
|
$this->sqlite->exec([
|
||||||
|
"update",
|
||||||
|
"table" => $channel->getTableName(),
|
||||||
|
"values" => $values,
|
||||||
|
"where" => ["_id" => $id],
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
function _discharge($keys=null, CapacitorChannel $channel=null, ?bool $reset=null): iterable {
|
function _discharge(CapacitorChannel $channel, $filter, ?bool $reset): iterable {
|
||||||
if ($keys !== null && !is_array($keys)) $keys = ["_id" => $keys];
|
if ($filter !== null && !is_array($filter)) $filter = ["_id" => $filter];
|
||||||
if ($reset === null) $reset = $keys === null;
|
if ($reset === null) $reset = $filter === null;
|
||||||
$rows = $this->sqlite->all([
|
$rows = $this->sqlite->all([
|
||||||
"select _item",
|
"select _item",
|
||||||
"from" => $channel->getTableName(),
|
"from" => $channel->getTableName(),
|
||||||
"where" => $keys,
|
"where" => $filter,
|
||||||
]);
|
]);
|
||||||
foreach ($rows as $row) {
|
foreach ($rows as $row) {
|
||||||
$item = unserialize($row['_item']);
|
$item = unserialize($row['_item']);
|
||||||
|
@ -100,27 +159,30 @@ class SqliteCapacitor extends AbstractCapacitor{
|
||||||
if ($reset) $this->_reset($channel);
|
if ($reset) $this->_reset($channel);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _get($keys, CapacitorChannel $channel=null) {
|
function _get(CapacitorChannel $channel, $filter) {
|
||||||
if ($keys === null) throw ValueException::null("keys");
|
if ($filter === null) throw ValueException::null("keys");
|
||||||
if (!is_array($keys)) $keys = ["_id" => $keys];
|
if (!is_array($filter)) $filter = ["_id" => $filter];
|
||||||
$row = $this->sqlite->one([
|
$row = $this->sqlite->one([
|
||||||
"select _item",
|
"select _item",
|
||||||
"from" => $channel->getTableName(),
|
"from" => $channel->getTableName(),
|
||||||
"where" => $keys,
|
"where" => $filter,
|
||||||
]);
|
]);
|
||||||
if ($row === null) return null;
|
if ($row === null) return null;
|
||||||
else return unserialize($row["_item"]);
|
else return unserialize($row["_item"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _each($keys, callable $func, ?array $args=null, CapacitorChannel $channel=null): void {
|
function _each(CapacitorChannel $channel, $filter, callable $func, ?array $args): void {
|
||||||
if ($keys !== null && !is_array($keys)) $keys = ["_id" => $keys];
|
if ($filter !== null && !is_array($filter)) $filter = ["_id" => $filter];
|
||||||
$context = func::_prepare($func);
|
$context = func::_prepare($func);
|
||||||
$sqlite = $this->sqlite;
|
$sqlite = $this->sqlite;
|
||||||
$tableName = $channel->getTableName();
|
$tableName = $channel->getTableName();
|
||||||
|
$commited = false;
|
||||||
|
$sqlite->beginTransaction();
|
||||||
|
try {
|
||||||
$rows = $sqlite->all([
|
$rows = $sqlite->all([
|
||||||
"select",
|
"select",
|
||||||
"from" => $tableName,
|
"from" => $tableName,
|
||||||
"where" => $keys,
|
"where" => $filter,
|
||||||
]);
|
]);
|
||||||
$args ??= [];
|
$args ??= [];
|
||||||
foreach ($rows as $row) {
|
foreach ($rows as $row) {
|
||||||
|
@ -138,6 +200,11 @@ class SqliteCapacitor extends AbstractCapacitor{
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
$sqlite->commit();
|
||||||
|
$commited = true;
|
||||||
|
} finally {
|
||||||
|
if (!$commited) $sqlite->rollback();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function close(): void {
|
function close(): void {
|
||||||
|
|
|
@ -1,23 +1,25 @@
|
||||||
<?php
|
<?php
|
||||||
namespace nur\sery\db\sqlite;
|
namespace nur\sery\db\sqlite;
|
||||||
|
|
||||||
|
use nulib\tests\TestCase;
|
||||||
|
use nur\sery\db\Capacitor;
|
||||||
use nur\sery\db\CapacitorChannel;
|
use nur\sery\db\CapacitorChannel;
|
||||||
use PHPUnit\Framework\TestCase;
|
|
||||||
|
|
||||||
class SqliteCapacitorTest extends TestCase {
|
class SqliteCapacitorTest extends TestCase {
|
||||||
function _testChargeStrings(SqliteCapacitor $capacitor, ?string $channel) {
|
function _testChargeStrings(SqliteCapacitor $capacitor, ?string $channel) {
|
||||||
$capacitor->reset($channel);
|
$capacitor->reset($channel);
|
||||||
$capacitor->charge("first", $channel);
|
$capacitor->charge($channel, "first");
|
||||||
$capacitor->charge("second", $channel);
|
$capacitor->charge($channel, "second");
|
||||||
$capacitor->charge("third", $channel);
|
$capacitor->charge($channel, "third");
|
||||||
$items = iterator_to_array($capacitor->discharge(null, $channel, false));
|
$items = iterator_to_array($capacitor->discharge($channel, null, false));
|
||||||
self::assertSame(["first", "second", "third"], $items);
|
self::assertSame(["first", "second", "third"], $items);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _testChargeArrays(SqliteCapacitor $capacitor, ?string $channel) {
|
function _testChargeArrays(SqliteCapacitor $capacitor, ?string $channel) {
|
||||||
$capacitor->reset($channel);
|
$capacitor->reset($channel);
|
||||||
$capacitor->charge(["id" => 10, "name" => "first"], $channel);
|
$capacitor->charge($channel, ["id" => 10, "name" => "first"]);
|
||||||
$capacitor->charge(["name" => "second", "id" => 20], $channel);
|
$capacitor->charge($channel, ["name" => "second", "id" => 20]);
|
||||||
$capacitor->charge(["name" => "third", "id" => "30"], $channel);
|
$capacitor->charge($channel, ["name" => "third", "id" => "30"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function testChargeStrings() {
|
function testChargeStrings() {
|
||||||
|
@ -45,8 +47,9 @@ class SqliteCapacitorTest extends TestCase {
|
||||||
|
|
||||||
function testEach() {
|
function testEach() {
|
||||||
$capacitor = new SqliteCapacitor(__DIR__.'/capacitor.db');
|
$capacitor = new SqliteCapacitor(__DIR__.'/capacitor.db');
|
||||||
$capacitor->addChannel(new class extends CapacitorChannel {
|
$capacitor = new Capacitor($capacitor, new class extends CapacitorChannel {
|
||||||
const NAME = "each";
|
const NAME = "each";
|
||||||
|
|
||||||
function getKeyDefinitions(): ?array {
|
function getKeyDefinitions(): ?array {
|
||||||
return [
|
return [
|
||||||
"age" => "integer",
|
"age" => "integer",
|
||||||
|
@ -60,12 +63,11 @@ class SqliteCapacitorTest extends TestCase {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
$channel = "each";
|
$capacitor->reset();
|
||||||
$capacitor->reset($channel);
|
$capacitor->charge(["name" => "first", "age" => 5]);
|
||||||
$capacitor->charge(["name" => "first", "age" => 5], $channel);
|
$capacitor->charge(["name" => "second", "age" => 10]);
|
||||||
$capacitor->charge(["name" => "second", "age" => 10], $channel);
|
$capacitor->charge(["name" => "third", "age" => 15]);
|
||||||
$capacitor->charge(["name" => "third", "age" => 15], $channel);
|
$capacitor->charge(["name" => "fourth", "age" => 20]);
|
||||||
$capacitor->charge(["name" => "fourth", "age" => 20], $channel);
|
|
||||||
|
|
||||||
$setDone = function ($item, $row, $suffix=null) {
|
$setDone = function ($item, $row, $suffix=null) {
|
||||||
$updates = ["done" => 1];
|
$updates = ["done" => 1];
|
||||||
|
@ -75,9 +77,39 @@ class SqliteCapacitorTest extends TestCase {
|
||||||
}
|
}
|
||||||
return $updates;
|
return $updates;
|
||||||
};
|
};
|
||||||
$capacitor->each(["age" => [">", 10]], $setDone, ["++"], $channel);
|
$capacitor->each(["age" => [">", 10]], $setDone, ["++"]);
|
||||||
$capacitor->each(["done" => 0], $setDone, null, $channel);
|
$capacitor->each(["done" => 0], $setDone, null);
|
||||||
Txx(iterator_to_array($capacitor->discharge(null, $channel, false)));
|
Txx(iterator_to_array($capacitor->discharge(null, false)));
|
||||||
|
|
||||||
|
$capacitor->close();
|
||||||
|
self::assertTrue(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function testPrimayKey() {
|
||||||
|
$capacitor = new SqliteCapacitor(__DIR__.'/capacitor.db');
|
||||||
|
$capacitor = new Capacitor($capacitor, new class extends CapacitorChannel {
|
||||||
|
const NAME = "pk";
|
||||||
|
|
||||||
|
function getKeyDefinitions(): ?array {
|
||||||
|
return [
|
||||||
|
"_id" => "varchar primary key",
|
||||||
|
"done" => "integer default 0",
|
||||||
|
];
|
||||||
|
}
|
||||||
|
function getKeyValues($item): ?array {
|
||||||
|
return [
|
||||||
|
"_id" => $item["numero"],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$capacitor->charge(["numero" => "a", "name" => "first", "age" => 5]);
|
||||||
|
$capacitor->charge(["numero" => "b", "name" => "second", "age" => 10]);
|
||||||
|
$capacitor->charge(["numero" => "c", "name" => "third", "age" => 15]);
|
||||||
|
$capacitor->charge(["numero" => "d", "name" => "fourth", "age" => 20]);
|
||||||
|
sleep(2);
|
||||||
|
$capacitor->charge(["numero" => "b", "name" => "second", "age" => 100]);
|
||||||
|
$capacitor->charge(["numero" => "d", "name" => "fourth", "age" => 200]);
|
||||||
|
|
||||||
$capacitor->close();
|
$capacitor->close();
|
||||||
self::assertTrue(true);
|
self::assertTrue(true);
|
||||||
|
|
Loading…
Reference in New Issue