Skip to content

Commit 2b43e94

Browse files
KminekMatejdg
authored andcommitted
Bulk write implementation (#73)
1 parent 6b32f2a commit 2b43e94

File tree

6 files changed

+258
-1
lines changed

6 files changed

+258
-1
lines changed

src/Caching/BulkWriter.php

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
<?php
2+
3+
/**
4+
* This file is part of the Nette Framework (https://nette.org)
5+
* Copyright (c) 2004 David Grudl (https://davidgrudl.com)
6+
*/
7+
8+
declare(strict_types=1);
9+
10+
namespace Nette\Caching;
11+
12+
13+
/**
14+
* Cache storage with a bulk write support.
15+
*/
16+
interface BulkWriter
17+
{
18+
/**
19+
* Writes to cache in bulk.
20+
* @param array{string, mixed} $items
21+
*/
22+
function bulkWrite(array $items, array $dependencies): void;
23+
24+
/**
25+
* Removes multiple items from cache
26+
*/
27+
function bulkRemove(array $keys): void;
28+
}

src/Caching/Cache.php

+50
Original file line numberDiff line numberDiff line change
@@ -175,6 +175,56 @@ public function bulkLoad(array $keys, ?callable $generator = null): array
175175
}
176176

177177

178+
/**
179+
* Writes multiple items into cache
180+
*
181+
* @param array $items
182+
* @param array|null $dependencies
183+
* @return array Stored items
184+
*
185+
* @throws InvalidArgumentException
186+
*/
187+
public function bulkSave(array $items, ?array $dependencies = null): array
188+
{
189+
$storedItems = [];
190+
191+
if (!$this->storage instanceof BulkWriter) {
192+
193+
foreach ($items as $key => $data) {
194+
$storedItems[$key] = $this->save($key, $data, $dependencies);
195+
}
196+
return $storedItems;
197+
}
198+
199+
$dependencies = $this->completeDependencies($dependencies);
200+
201+
if (isset($dependencies[self::Expire]) && $dependencies[self::Expire] <= 0) {
202+
$this->storage->bulkRemove(array_map(fn($key): string => $this->generateKey($key), array_keys($items)));
203+
return [];
204+
}
205+
206+
$removals = [];
207+
$toCache = [];
208+
foreach ($items as $key => $data) {
209+
$cKey = $this->generateKey($key);
210+
211+
if ($data === null) {
212+
$removals[] = $cKey;
213+
} else {
214+
$storedItems[$key] = $toCache[$cKey] = $data;
215+
}
216+
}
217+
218+
if (!empty($removals)) {
219+
$this->storage->bulkRemove($removals);
220+
}
221+
222+
$this->storage->bulkWrite($toCache, $dependencies);
223+
224+
return $storedItems;
225+
}
226+
227+
178228
/**
179229
* Writes item into the cache.
180230
* Dependencies are:

src/Caching/Storages/MemcachedStorage.php

+43-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
/**
1717
* Memcached storage using memcached extension.
1818
*/
19-
class MemcachedStorage implements Nette\Caching\Storage, Nette\Caching\BulkReader
19+
class MemcachedStorage implements Nette\Caching\Storage, Nette\Caching\BulkReader, Nette\Caching\BulkWriter
2020
{
2121
/** @internal cache structure */
2222
private const
@@ -168,12 +168,54 @@ public function write(string $key, $data, array $dp): void
168168
}
169169

170170

171+
public function bulkWrite(array $items, array $dp): void
172+
{
173+
if (isset($dp[Cache::Items])) {
174+
throw new Nette\NotSupportedException('Dependent items are not supported by MemcachedStorage.');
175+
}
176+
177+
$meta = $records = [];
178+
$expire = 0;
179+
if (isset($dp[Cache::Expire])) {
180+
$expire = (int) $dp[Cache::Expire];
181+
if (!empty($dp[Cache::Sliding])) {
182+
$meta[self::MetaDelta] = $expire; // sliding time
183+
}
184+
}
185+
186+
if (isset($dp[Cache::Callbacks])) {
187+
$meta[self::MetaCallbacks] = $dp[Cache::Callbacks];
188+
}
189+
190+
foreach ($items as $key => $meta[self::MetaData]) {
191+
$key = urlencode($this->prefix . $key);
192+
$records[$key] = $meta;
193+
194+
if (isset($dp[Cache::Tags]) || isset($dp[Cache::Priority])) {
195+
if (!$this->journal) {
196+
throw new Nette\InvalidStateException('CacheJournal has not been provided.');
197+
}
198+
199+
$this->journal->write($key, $dp);
200+
}
201+
}
202+
203+
$this->memcached->setMulti($records, $expire);
204+
}
205+
206+
171207
public function remove(string $key): void
172208
{
173209
$this->memcached->delete(urlencode($this->prefix . $key), 0);
174210
}
175211

176212

213+
public function bulkRemove(array $keys): void
214+
{
215+
$this->memcached->deleteMulti(array_map(fn($key) => urlencode($this->prefix . $key), $keys), 0);
216+
}
217+
218+
177219
public function clean(array $conditions): void
178220
{
179221
if (!empty($conditions[Cache::All])) {

tests/Caching/Cache.bulkSave.phpt

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
<?php
2+
3+
/**
4+
* Test: Nette\Caching\Cache save().
5+
*/
6+
7+
declare(strict_types=1);
8+
9+
use Nette\Caching\Cache;
10+
use Tester\Assert;
11+
12+
require __DIR__ . '/../bootstrap.php';
13+
require __DIR__ . '/Cache.php';
14+
15+
16+
test('storage without bulk write support', function () {
17+
$storage = new TestStorage;
18+
$cache = new Cache($storage, 'ns');
19+
Assert::same([1, 2], $cache->bulkSave([1, 2]), 'data');
20+
Assert::same([1 => 'value1', 2 => 'value2'], $cache->bulkSave([1 => 'value1', 2 => 'value2']), 'data');
21+
22+
$data = $cache->bulkLoad([1, 2]);
23+
Assert::same('value1', $data[1]['data']);
24+
Assert::same('value2', $data[2]['data']);
25+
});
26+
27+
test('storage with bulk write support', function () {
28+
$storage = new BulkWriteTestStorage;
29+
$cache = new Cache($storage, 'ns');
30+
Assert::same([1, 2], $cache->bulkSave([1, 2]), 'data');
31+
Assert::same([1 => 'value1', 2 => 'value2'], $cache->bulkSave([1 => 'value1', 2 => 'value2']), 'data');
32+
33+
$data = $cache->bulkLoad([1, 2]);
34+
Assert::same('value1', $data[1]['data']);
35+
Assert::same('value2', $data[2]['data']);
36+
});
37+
38+
test('dependencies', function () {
39+
$storage = new BulkWriteTestStorage;
40+
$cache = new Cache($storage, 'ns');
41+
$dependencies = [Cache::Tags => ['tag']];
42+
$cache->bulkSave([1 => 'value1', 2 => 'value2'], $dependencies);
43+
44+
$data = $cache->bulkLoad([1, 2]);
45+
Assert::same($dependencies, $data[1]['dependencies']);
46+
Assert::same($dependencies, $data[2]['dependencies']);
47+
48+
$cache->clean($dependencies);
49+
50+
Assert::same([1 => null, 2 => null], $cache->bulkLoad([1, 2]));
51+
});

tests/Caching/Cache.php

+50
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
declare(strict_types=1);
44

5+
use Nette\Caching\BulkWriter;
56
use Nette\Caching\BulkReader;
67
use Nette\Caching\Storage;
78

@@ -37,6 +38,25 @@ public function remove(string $key): void
3738

3839
public function clean(array $conditions): void
3940
{
41+
if (!empty($conditions[Nette\Caching\Cache::All])) {
42+
$this->data = [];
43+
return;
44+
}
45+
46+
//unset based by tags
47+
if (!empty($conditions[Nette\Caching\Cache::Tags])) {
48+
$unsets = [];
49+
foreach ($this->data as $key => $data) {
50+
$tags = $data['dependencies'][Nette\Caching\Cache::Tags] ?? null;
51+
if (array_intersect($conditions[Nette\Caching\Cache::Tags], $tags)) {
52+
$unsets[$key] = $key;
53+
}
54+
}
55+
56+
foreach ($unsets as $unsetKey) {
57+
unset($this->data[$unsetKey]);
58+
}
59+
}
4060
}
4161
}
4262

@@ -55,3 +75,33 @@ public function bulkRead(array $keys): array
5575
return $result;
5676
}
5777
}
78+
79+
class BulkWriteTestStorage extends TestStorage implements BulkWriter
80+
{
81+
public function bulkRead(array $keys): array
82+
{
83+
$result = [];
84+
foreach ($keys as $key) {
85+
$data = $this->read($key);
86+
if ($data !== null) {
87+
$result[$key] = $data;
88+
}
89+
}
90+
91+
return $result;
92+
}
93+
94+
95+
public function bulkRemove(array $keys): void
96+
{
97+
98+
}
99+
100+
101+
public function bulkWrite($items, array $dp): void
102+
{
103+
foreach ($items as $key => $data) {
104+
$this->write($key, $data, $dp);
105+
}
106+
}
107+
}
+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
<?php
2+
3+
/**
4+
* Test: Nette\Caching\Storages\MemcachedStorage and bulkWrite
5+
*/
6+
7+
declare(strict_types=1);
8+
9+
use Nette\Caching\Cache;
10+
use Nette\Caching\Storages\MemcachedStorage;
11+
use Nette\Caching\Storages\SQLiteJournal;
12+
use Tester\Assert;
13+
14+
require __DIR__ . '/../bootstrap.php';
15+
16+
17+
if (!MemcachedStorage::isAvailable()) {
18+
Tester\Environment::skip('Requires PHP extension Memcached.');
19+
}
20+
21+
Tester\Environment::lock('memcached-files', getTempDir());
22+
23+
24+
$storage = new MemcachedStorage('localhost', 11211, '', new SQLiteJournal(getTempDir() . '/journal-memcached.s3db'));
25+
$cache = new Cache($storage);
26+
27+
//standard
28+
$cache->bulkSave(['foo' => 'bar']);
29+
Assert::same(['foo' => 'bar', 'lorem' => null], $cache->bulkLoad(['foo', 'lorem']));
30+
31+
//tags
32+
$dependencies = [Cache::Tags => ['tag']];
33+
$cache->bulkSave(['foo' => 'bar'], $dependencies);
34+
Assert::same(['foo' => 'bar'], $cache->bulkLoad(['foo']));
35+
$cache->clean($dependencies);
36+
Assert::same(['foo' => null], $cache->bulkLoad(['foo']));

0 commit comments

Comments
 (0)