2016-12-29 19:16:04 -06:00
|
|
|
<?php
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2016-08-23 16:47:40 -05:00
|
|
|
namespace Amp\Parallel\Worker;
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2017-03-16 17:03:59 -05:00
|
|
|
use Amp\Loop;
|
2017-07-27 23:47:36 -05:00
|
|
|
use Amp\Struct;
|
2016-08-18 11:04:48 -05:00
|
|
|
|
2018-10-21 10:54:46 -05:00
|
|
|
final class BasicEnvironment implements Environment
|
2018-10-07 09:50:45 -05:00
|
|
|
{
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var array */
|
2016-01-14 17:44:43 -06:00
|
|
|
private $data = [];
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var \SplPriorityQueue */
|
2016-01-14 17:44:43 -06:00
|
|
|
private $queue;
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var string */
|
2016-01-14 17:44:43 -06:00
|
|
|
private $timer;
|
|
|
|
|
2018-10-07 09:50:45 -05:00
|
|
|
public function __construct()
|
|
|
|
{
|
2018-10-07 09:22:56 -05:00
|
|
|
$this->queue = $queue = new \SplPriorityQueue;
|
|
|
|
$data = &$this->data;
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
$this->timer = Loop::repeat(1000, static function ($watcherId) use ($queue, &$data) {
|
2016-08-18 11:04:48 -05:00
|
|
|
$time = \time();
|
2018-10-07 09:22:56 -05:00
|
|
|
while (!$queue->isEmpty()) {
|
|
|
|
list($key, $expiration) = $queue->top();
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
if (!isset($data[$key])) {
|
2017-07-27 23:47:36 -05:00
|
|
|
// Item removed.
|
2018-10-07 09:22:56 -05:00
|
|
|
$queue->extract();
|
2017-07-27 23:47:36 -05:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
$struct = $data[$key];
|
2017-07-27 23:47:36 -05:00
|
|
|
|
|
|
|
if ($struct->expire === 0) {
|
|
|
|
// Item was set again without a TTL.
|
2018-10-07 09:22:56 -05:00
|
|
|
$queue->extract();
|
2017-07-27 23:47:36 -05:00
|
|
|
continue;
|
|
|
|
}
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2017-07-29 10:29:16 -05:00
|
|
|
if ($struct->expire !== $expiration) {
|
|
|
|
// Expiration changed or TTL updated.
|
2018-10-07 09:22:56 -05:00
|
|
|
$queue->extract();
|
2017-07-29 10:29:16 -05:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-07-27 23:47:36 -05:00
|
|
|
if ($time < $struct->expire) {
|
|
|
|
// Item at top has not expired, break out of loop.
|
|
|
|
break;
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
unset($data[$key]);
|
2017-07-27 23:47:36 -05:00
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
$queue->extract();
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
2018-10-07 09:22:56 -05:00
|
|
|
if ($queue->isEmpty()) {
|
|
|
|
Loop::disable($watcherId);
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
});
|
2017-05-18 09:51:31 +02:00
|
|
|
|
2016-08-18 11:04:48 -05:00
|
|
|
Loop::disable($this->timer);
|
|
|
|
Loop::unreference($this->timer);
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function exists(string $key): bool
|
|
|
|
{
|
2016-01-23 00:00:56 -06:00
|
|
|
return isset($this->data[$key]);
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
*
|
|
|
|
* @return mixed|null Returns null if the key does not exist.
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function get(string $key)
|
|
|
|
{
|
2017-07-27 23:47:36 -05:00
|
|
|
if (!isset($this->data[$key])) {
|
|
|
|
return null;
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
2017-07-27 23:47:36 -05:00
|
|
|
$struct = $this->data[$key];
|
|
|
|
|
|
|
|
if ($struct->ttl !== null) {
|
2017-07-29 10:29:16 -05:00
|
|
|
$expire = \time() + $struct->ttl;
|
|
|
|
if ($struct->expire < $expire) {
|
|
|
|
$struct->expire = $expire;
|
|
|
|
$this->queue->insert([$key, $struct->expire], -$struct->expire);
|
|
|
|
}
|
2017-07-27 23:47:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return $struct->data;
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
2017-05-18 09:51:31 +02:00
|
|
|
|
2016-01-14 17:44:43 -06:00
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
* @param mixed $value Using null for the value deletes the key.
|
2017-07-27 23:47:36 -05:00
|
|
|
* @param int $ttl Number of seconds until data is automatically deleted. Use null for unlimited TTL.
|
|
|
|
*
|
|
|
|
* @throws \Error If the time-to-live is not a positive integer.
|
2016-01-14 17:44:43 -06:00
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function set(string $key, $value, int $ttl = null)
|
|
|
|
{
|
2017-07-27 23:47:36 -05:00
|
|
|
if ($value === null) {
|
2016-01-14 17:44:43 -06:00
|
|
|
$this->delete($key);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-27 23:47:36 -05:00
|
|
|
if ($ttl !== null && $ttl <= 0) {
|
|
|
|
throw new \Error("The time-to-live must be a positive integer or null");
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
2017-07-27 23:47:36 -05:00
|
|
|
$struct = new class {
|
|
|
|
use Struct;
|
|
|
|
public $data;
|
|
|
|
public $expire = 0;
|
|
|
|
public $ttl;
|
|
|
|
};
|
|
|
|
|
|
|
|
$struct->data = $value;
|
|
|
|
|
|
|
|
if ($ttl !== null) {
|
|
|
|
$struct->ttl = $ttl;
|
|
|
|
$struct->expire = \time() + $ttl;
|
2017-07-29 10:29:16 -05:00
|
|
|
$this->queue->insert([$key, $struct->expire], -$struct->expire);
|
2016-01-14 17:44:43 -06:00
|
|
|
|
2016-08-18 11:04:48 -05:00
|
|
|
Loop::enable($this->timer);
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
2017-07-27 23:47:36 -05:00
|
|
|
$this->data[$key] = $struct;
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $key
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function delete(string $key)
|
|
|
|
{
|
2017-07-27 23:47:36 -05:00
|
|
|
unset($this->data[$key]);
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Alias of exists().
|
|
|
|
*
|
|
|
|
* @param $key
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function offsetExists($key)
|
|
|
|
{
|
2016-01-14 17:44:43 -06:00
|
|
|
return $this->exists($key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Alias of get().
|
|
|
|
*
|
|
|
|
* @param string $key
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function offsetGet($key)
|
|
|
|
{
|
2016-01-14 17:44:43 -06:00
|
|
|
return $this->get($key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-07-27 23:47:36 -05:00
|
|
|
* Alias of set() with $ttl = null.
|
2016-01-14 17:44:43 -06:00
|
|
|
*
|
|
|
|
* @param string $key
|
|
|
|
* @param mixed $value
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function offsetSet($key, $value)
|
|
|
|
{
|
2016-01-14 17:44:43 -06:00
|
|
|
$this->set($key, $value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Alias of delete().
|
|
|
|
*
|
|
|
|
* @param string $key
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function offsetUnset($key)
|
|
|
|
{
|
2016-01-14 17:44:43 -06:00
|
|
|
$this->delete($key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes all values.
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function clear()
|
|
|
|
{
|
2016-01-14 17:44:43 -06:00
|
|
|
$this->data = [];
|
|
|
|
|
2016-08-18 11:04:48 -05:00
|
|
|
Loop::disable($this->timer);
|
2016-08-22 18:25:19 -05:00
|
|
|
$this->queue = new \SplPriorityQueue;
|
2016-01-14 17:44:43 -06:00
|
|
|
}
|
|
|
|
}
|