Commit fe261118 by Qiang Xue

Fixes #4051: Renamed `yii\caching\GroupDependency` to `TagDependency` and added…

Fixes #4051: Renamed `yii\caching\GroupDependency` to `TagDependency` and added support for associating multiple tags to a single cached data item
parent 2e71f6e9
...@@ -242,8 +242,8 @@ Aquí abajo se muestra un sumario de las dependencias disponibles: ...@@ -242,8 +242,8 @@ Aquí abajo se muestra un sumario de las dependencias disponibles:
- [[yii\caching\DbDependency]]: la dependencia cambia si el resultado de la consulta de la sentencia SQL especificada cambia. - [[yii\caching\DbDependency]]: la dependencia cambia si el resultado de la consulta de la sentencia SQL especificada cambia.
- [[yii\caching\ExpressionDependency]]: la dependencia cambia si el resultado de la expresión de PHP especificada cambia. - [[yii\caching\ExpressionDependency]]: la dependencia cambia si el resultado de la expresión de PHP especificada cambia.
- [[yii\caching\FileDependency]]: la dependencia cambia si se modifica la última fecha de modificación del archivo. - [[yii\caching\FileDependency]]: la dependencia cambia si se modifica la última fecha de modificación del archivo.
- [[yii\caching\GroupDependency]]: marca un elemento de datos en caché con un nombre de grupo. Puedes invalidar los elementos de datos almacenados en caché - [[yii\caching\TagDependency]]: marca un elemento de datos en caché con un nombre de grupo. Puedes invalidar los elementos de datos almacenados en caché
con el mismo nombre del grupo a la vez llamando a [[yii\caching\GroupDependency::invalidate()]]. con el mismo nombre del grupo a la vez llamando a [[yii\caching\TagDependency::invalidate()]].
## Consultas en Caché <a name="query-caching"></a> ## Consultas en Caché <a name="query-caching"></a>
......
...@@ -196,7 +196,8 @@ $data = $cache->get($key); ...@@ -196,7 +196,8 @@ $data = $cache->get($key);
- [[yii\caching\DbDependency]]: 如果指定 SQL 语句的查询结果发生了变化,则依赖改变。 - [[yii\caching\DbDependency]]: 如果指定 SQL 语句的查询结果发生了变化,则依赖改变。
- [[yii\caching\ExpressionDependency]]: 如果指定的 PHP 表达式执行结果发生变化,则依赖改变。 - [[yii\caching\ExpressionDependency]]: 如果指定的 PHP 表达式执行结果发生变化,则依赖改变。
- [[yii\caching\FileDependency]]: 如果文件的最后修改时间发生变化,则依赖改变。 - [[yii\caching\FileDependency]]: 如果文件的最后修改时间发生变化,则依赖改变。
- [[yii\caching\GroupDependency]]: 将一项缓存数据标记到一个组名,你可以通过调用 [[yii\caching\GroupDependency::invalidate()]] 一次性将同样组名的缓存数据全部置为失效状态。 - [[yii\caching\TagDependency]]: 为一项缓存数据添加一个或多个标签。你可以通过调用 [[yii\caching\TagDependency::invalidate()]]
一次性将具有指定标签的缓存数据全部置为失效状态。
## 查询缓存 <a name="query-caching"></a> ## 查询缓存 <a name="query-caching"></a>
......
...@@ -232,8 +232,8 @@ Below is a summary of the available cache dependencies: ...@@ -232,8 +232,8 @@ Below is a summary of the available cache dependencies:
- [[yii\caching\DbDependency]]: the dependency is changed if the query result of the specified SQL statement is changed. - [[yii\caching\DbDependency]]: the dependency is changed if the query result of the specified SQL statement is changed.
- [[yii\caching\ExpressionDependency]]: the dependency is changed if the result of the specified PHP expression is changed. - [[yii\caching\ExpressionDependency]]: the dependency is changed if the result of the specified PHP expression is changed.
- [[yii\caching\FileDependency]]: the dependency is changed if the file's last modification time is changed. - [[yii\caching\FileDependency]]: the dependency is changed if the file's last modification time is changed.
- [[yii\caching\GroupDependency]]: marks a cached data item with a group name. You may invalidate the cached data items - [[yii\caching\TagDependency]]: associates a cached data item with one or multiple tags. You may invalidate
with the same group name all at once by calling [[yii\caching\GroupDependency::invalidate()]]. the cached data items with the specified tag(s) by calling [[yii\caching\TagDependency::invalidate()]].
## Query Caching <a name="query-caching"></a> ## Query Caching <a name="query-caching"></a>
......
...@@ -10,7 +10,7 @@ namespace yii\sphinx; ...@@ -10,7 +10,7 @@ namespace yii\sphinx;
use yii\base\Object; use yii\base\Object;
use yii\caching\Cache; use yii\caching\Cache;
use Yii; use Yii;
use yii\caching\GroupDependency; use yii\caching\TagDependency;
use yii\db\Exception; use yii\db\Exception;
/** /**
...@@ -139,8 +139,8 @@ class Schema extends Object ...@@ -139,8 +139,8 @@ class Schema extends Object
if ($refresh || ($index = $cache->get($key)) === false) { if ($refresh || ($index = $cache->get($key)) === false) {
$index = $this->loadIndexSchema($realName); $index = $this->loadIndexSchema($realName);
if ($index !== null) { if ($index !== null) {
$cache->set($key, $index, $db->schemaCacheDuration, new GroupDependency([ $cache->set($key, $index, $db->schemaCacheDuration, new TagDependency([
'group' => $this->getCacheGroup(), 'tags' => $this->getCacheTag(),
])); ]));
} }
} }
...@@ -168,11 +168,11 @@ class Schema extends Object ...@@ -168,11 +168,11 @@ class Schema extends Object
} }
/** /**
* Returns the cache group name. * Returns the cache tag name.
* This allows [[refresh()]] to invalidate all cached index schemas. * This allows [[refresh()]] to invalidate all cached index schemas.
* @return string the cache group name * @return string the cache tag name
*/ */
protected function getCacheGroup() protected function getCacheTag()
{ {
return md5(serialize([ return md5(serialize([
__CLASS__, __CLASS__,
...@@ -299,7 +299,7 @@ class Schema extends Object ...@@ -299,7 +299,7 @@ class Schema extends Object
/* @var $cache Cache */ /* @var $cache Cache */
$cache = is_string($this->db->schemaCache) ? Yii::$app->get($this->db->schemaCache, false) : $this->db->schemaCache; $cache = is_string($this->db->schemaCache) ? Yii::$app->get($this->db->schemaCache, false) : $this->db->schemaCache;
if ($this->db->enableSchemaCache && $cache instanceof Cache) { if ($this->db->enableSchemaCache && $cache instanceof Cache) {
GroupDependency::invalidate($cache, $this->getCacheGroup()); TagDependency::invalidate($cache, $this->getCacheTag());
} }
$this->_indexNames = []; $this->_indexNames = [];
$this->_indexes = []; $this->_indexes = [];
......
...@@ -170,6 +170,7 @@ Yii Framework 2 Change Log ...@@ -170,6 +170,7 @@ Yii Framework 2 Change Log
- Chg #3899: Moved `MailEvent` class to `yii\mail` namespace (cebe) - Chg #3899: Moved `MailEvent` class to `yii\mail` namespace (cebe)
- Chg #3956: Flash messages set via `Yii::$app->session->setFlash()` will be removed only if they are accessed (qiangxue) - Chg #3956: Flash messages set via `Yii::$app->session->setFlash()` will be removed only if they are accessed (qiangxue)
- Chg #3989: The default value for `yii\log\FileTarget::$rotateByCopy` now defaults to true to work on windows by default (cebe) - Chg #3989: The default value for `yii\log\FileTarget::$rotateByCopy` now defaults to true to work on windows by default (cebe)
- Chg #4051: Renamed `yii\caching\GroupDependency` to `TagDependency` and added support for associating multiple tags to a single cached data item (qiangxue)
- Chg #4071: `mail` component renamed to `mailer`, `yii\log\EmailTarget::$mail` renamed to `yii\log\EmailTarget::$mailer` (samdark) - Chg #4071: `mail` component renamed to `mailer`, `yii\log\EmailTarget::$mail` renamed to `yii\log\EmailTarget::$mailer` (samdark)
- Chg #4147: `BaseMailer::compose()` will not overwrite the `message` parameter if it is explicitly provided (qiangxue) - Chg #4147: `BaseMailer::compose()` will not overwrite the `message` parameter if it is explicitly provided (qiangxue)
- Chg #4201: change default value of `SyslogTarget::facility` from LOG_SYSLOG to LOG_USER (dizews) - Chg #4201: change default value of `SyslogTarget::facility` from LOG_SYSLOG to LOG_USER (dizews)
......
...@@ -74,7 +74,11 @@ Upgrade from Yii 2.0 Beta ...@@ -74,7 +74,11 @@ Upgrade from Yii 2.0 Beta
* `mail` component was renamed to `mailer`, `yii\log\EmailTarget::$mail` was renamed to `yii\log\EmailTarget::$mailer`. * `mail` component was renamed to `mailer`, `yii\log\EmailTarget::$mail` was renamed to `yii\log\EmailTarget::$mailer`.
Please update all references in the code and config files. Please update all references in the code and config files.
* `\yii\rbac\PhpManager` now stores data in three separate files instead of one. In order to convert old file to * `yii\caching\GroupDependency` was renamed to `TagDependency`. You should create such a dependency using the code
`new \yii\caching\TagDependency(['tags' => 'TagName'])`, where `TagName` is similar to the group name that you
previously used.
* `yii\rbac\PhpManager` now stores data in three separate files instead of one. In order to convert old file to
new ones save the following code as `convert.php` that should be placed in the same directory your `rbac.php` is in: new ones save the following code as `convert.php` that should be placed in the same directory your `rbac.php` is in:
```php ```php
......
...@@ -7,71 +7,104 @@ ...@@ -7,71 +7,104 @@
namespace yii\caching; namespace yii\caching;
use yii\base\InvalidConfigException;
/** /**
* GroupDependency marks a cached data item with a group name. * TagDependency associates a cached data item with one or multiple [[tags]].
* *
* You may invalidate the cached data items with the same group name all at once * By calling [[invalidate()]], you can invalidate all cached data items that are associated with the specified tag name(s).
* by calling [[invalidate()]].
* *
* @author Qiang Xue <qiang.xue@gmail.com> * @author Qiang Xue <qiang.xue@gmail.com>
* @since 2.0 * @since 2.0
*/ */
class GroupDependency extends Dependency class TagDependency extends Dependency
{ {
/** /**
* @var string the group name. This property must be set. * @var string|array a list of tag names for this dependency. For a single tag, you may specify it as a string.
*/ */
public $group; public $tags = [];
/** /**
* Generates the data needed to determine if dependency has been changed. * Generates the data needed to determine if dependency has been changed.
* This method does nothing in this class. * This method does nothing in this class.
* @param Cache $cache the cache component that is currently evaluating this dependency * @param Cache $cache the cache component that is currently evaluating this dependency
* @return mixed the data needed to determine if dependency has been changed. * @return mixed the data needed to determine if dependency has been changed.
* @throws InvalidConfigException if [[group]] is not set.
*/ */
protected function generateDependencyData($cache) protected function generateDependencyData($cache)
{ {
if ($this->group === null) { $timestamps = $this->getTimestamps($cache, (array) $this->tags);
throw new InvalidConfigException('GroupDependency::group must be set');
$newKeys = [];
foreach ($timestamps as $key => $timestamp) {
if ($timestamp === false) {
$newKeys[] = $key;
}
} }
$version = $cache->get([__CLASS__, $this->group]); if (!empty($newKeys)) {
if ($version === false) { $timestamps = array_merge($timestamps, $this->touchKeys($cache, $newKeys));
$version = $this->invalidate($cache, $this->group);
} }
return $version; return $timestamps;
} }
/** /**
* Performs the actual dependency checking. * Performs the actual dependency checking.
* @param Cache $cache the cache component that is currently evaluating this dependency * @param Cache $cache the cache component that is currently evaluating this dependency
* @return boolean whether the dependency is changed or not. * @return boolean whether the dependency is changed or not.
* @throws InvalidConfigException if [[group]] is not set.
*/ */
public function getHasChanged($cache) public function getHasChanged($cache)
{ {
if ($this->group === null) { $timestamps = $this->getTimestamps($cache, (array) $this->tags);
throw new InvalidConfigException('GroupDependency::group must be set'); return $timestamps !== $this->data;
} }
$version = $cache->get([__CLASS__, $this->group]);
return $version === false || $version !== $this->data; /**
* Invalidates all of the cached data items that are associated with any of the specified [[tags]].
* @param Cache $cache the cache component that caches the data items
* @param string|array $tags
*/
public static function invalidate($cache, $tags)
{
$keys = [];
foreach ((array)$tags as $tag) {
$keys[] = $cache->buildKey([__CLASS__, $tag]);
}
static::touchKeys($cache, $keys);
} }
/** /**
* Invalidates all of the cached data items that have the same [[group]]. * Generates the timestamp for the specified cache keys.
* @param Cache $cache the cache component that caches the data items * @param Cache $cache
* @param string $group the group name * @param string[] $keys
* @return string the current version number * @return array the timestamp indexed by cache keys
*/ */
public static function invalidate($cache, $group) protected static function touchKeys($cache, $keys)
{ {
$version = microtime(); $items = [];
$cache->set([__CLASS__, $group], $version); $time = microtime();
foreach ($keys as $key) {
$items[$key] = $time;
}
$cache->mset($items);
return $items;
}
/**
* Returns the timestamps for the specified tags.
* @param Cache $cache
* @param string[] $tags
* @return array the timestamps indexed by the specified tags.
*/
protected function getTimestamps($cache, $tags)
{
if (empty($tags)) {
return [];
}
$keys = [];
foreach ($tags as $tag) {
$keys[] = $cache->buildKey([__CLASS__, $tag]);
}
return $version; return $cache->mget($keys);
} }
} }
...@@ -66,7 +66,7 @@ return [ ...@@ -66,7 +66,7 @@ return [
'yii\caching\ExpressionDependency' => YII_PATH . '/caching/ExpressionDependency.php', 'yii\caching\ExpressionDependency' => YII_PATH . '/caching/ExpressionDependency.php',
'yii\caching\FileCache' => YII_PATH . '/caching/FileCache.php', 'yii\caching\FileCache' => YII_PATH . '/caching/FileCache.php',
'yii\caching\FileDependency' => YII_PATH . '/caching/FileDependency.php', 'yii\caching\FileDependency' => YII_PATH . '/caching/FileDependency.php',
'yii\caching\GroupDependency' => YII_PATH . '/caching/GroupDependency.php', 'yii\caching\TagDependency' => YII_PATH . '/caching/TagDependency.php',
'yii\caching\MemCache' => YII_PATH . '/caching/MemCache.php', 'yii\caching\MemCache' => YII_PATH . '/caching/MemCache.php',
'yii\caching\MemCacheServer' => YII_PATH . '/caching/MemCacheServer.php', 'yii\caching\MemCacheServer' => YII_PATH . '/caching/MemCacheServer.php',
'yii\caching\WinCache' => YII_PATH . '/caching/WinCache.php', 'yii\caching\WinCache' => YII_PATH . '/caching/WinCache.php',
......
...@@ -12,7 +12,7 @@ use yii\base\Object; ...@@ -12,7 +12,7 @@ use yii\base\Object;
use yii\base\NotSupportedException; use yii\base\NotSupportedException;
use yii\base\InvalidCallException; use yii\base\InvalidCallException;
use yii\caching\Cache; use yii\caching\Cache;
use yii\caching\GroupDependency; use yii\caching\TagDependency;
/** /**
* Schema is the base class for concrete DBMS-specific schema classes. * Schema is the base class for concrete DBMS-specific schema classes.
...@@ -114,8 +114,8 @@ abstract class Schema extends Object ...@@ -114,8 +114,8 @@ abstract class Schema extends Object
if ($refresh || ($table = $cache->get($key)) === false) { if ($refresh || ($table = $cache->get($key)) === false) {
$this->_tables[$name] = $table = $this->loadTableSchema($realName); $this->_tables[$name] = $table = $this->loadTableSchema($realName);
if ($table !== null) { if ($table !== null) {
$cache->set($key, $table, $db->schemaCacheDuration, new GroupDependency([ $cache->set($key, $table, $db->schemaCacheDuration, new TagDependency([
'group' => $this->getCacheGroup(), 'tags' => $this->getCacheTag(),
])); ]));
} }
} else { } else {
...@@ -145,11 +145,11 @@ abstract class Schema extends Object ...@@ -145,11 +145,11 @@ abstract class Schema extends Object
} }
/** /**
* Returns the cache group name. * Returns the cache tag name.
* This allows [[refresh()]] to invalidate all cached table schemas. * This allows [[refresh()]] to invalidate all cached table schemas.
* @return string the cache group name * @return string the cache tag name
*/ */
protected function getCacheGroup() protected function getCacheTag()
{ {
return md5(serialize([ return md5(serialize([
__CLASS__, __CLASS__,
...@@ -241,7 +241,7 @@ abstract class Schema extends Object ...@@ -241,7 +241,7 @@ abstract class Schema extends Object
/* @var $cache Cache */ /* @var $cache Cache */
$cache = is_string($this->db->schemaCache) ? Yii::$app->get($this->db->schemaCache, false) : $this->db->schemaCache; $cache = is_string($this->db->schemaCache) ? Yii::$app->get($this->db->schemaCache, false) : $this->db->schemaCache;
if ($this->db->enableSchemaCache && $cache instanceof Cache) { if ($this->db->enableSchemaCache && $cache instanceof Cache) {
GroupDependency::invalidate($cache, $this->getCacheGroup()); TagDependency::invalidate($cache, $this->getCacheTag());
} }
$this->_tableNames = []; $this->_tableNames = [];
$this->_tables = []; $this->_tables = [];
......
<?php
namespace yiiunit\framework\caching;
use yiiunit\TestCase;
use yii\caching\FileCache;
use yii\caching\TagDependency;
/**
* @group caching
*/
class TagDependencyTest extends TestCase
{
public function testInvalidate()
{
$cache = new FileCache(['cachePath' => '@yiiunit/runtime/cache']);
// single tag test
$cache->set('a1', 11, 0, new TagDependency(['tags' => 't1']));
$cache->set('a2', 12, 0, new TagDependency(['tags' => 't1']));
$cache->set('b1', 21, 0, new TagDependency(['tags' => 't2']));
$cache->set('b2', 22, 0, new TagDependency(['tags' => 't2']));
$this->assertEquals(11, $cache->get('a1'));
$this->assertEquals(12, $cache->get('a2'));
$this->assertEquals(21, $cache->get('b1'));
$this->assertEquals(22, $cache->get('b2'));
TagDependency::invalidate($cache, 't1');
$this->assertFalse($cache->get('a1'));
$this->assertFalse($cache->get('a2'));
$this->assertEquals(21, $cache->get('b1'));
$this->assertEquals(22, $cache->get('b2'));
TagDependency::invalidate($cache, 't2');
$this->assertFalse($cache->get('a1'));
$this->assertFalse($cache->get('a2'));
$this->assertFalse($cache->get('b1'));
$this->assertFalse($cache->get('b2'));
// multiple tag test
$cache->set('a1', 11, 0, new TagDependency(['tags' => ['t1', 't2']]));
$cache->set('a2', 12, 0, new TagDependency(['tags' => 't1']));
$cache->set('b1', 21, 0, new TagDependency(['tags' => ['t1', 't2']]));
$cache->set('b2', 22, 0, new TagDependency(['tags' => 't2']));
$this->assertEquals(11, $cache->get('a1'));
$this->assertEquals(12, $cache->get('a2'));
$this->assertEquals(21, $cache->get('b1'));
$this->assertEquals(22, $cache->get('b2'));
TagDependency::invalidate($cache, 't1');
$this->assertFalse($cache->get('a1'));
$this->assertFalse($cache->get('a2'));
$this->assertFalse($cache->get('b1'));
$this->assertEquals(22, $cache->get('b2'));
TagDependency::invalidate($cache, 't2');
$this->assertFalse($cache->get('a1'));
$this->assertFalse($cache->get('a2'));
$this->assertFalse($cache->get('b1'));
$this->assertFalse($cache->get('b2'));
$cache->set('a1', 11, 0, new TagDependency(['tags' => ['t1', 't2']]));
$cache->set('a2', 12, 0, new TagDependency(['tags' => 't1']));
$cache->set('b1', 21, 0, new TagDependency(['tags' => ['t1', 't2']]));
$cache->set('b2', 22, 0, new TagDependency(['tags' => 't2']));
$this->assertEquals(11, $cache->get('a1'));
$this->assertEquals(12, $cache->get('a2'));
$this->assertEquals(21, $cache->get('b1'));
$this->assertEquals(22, $cache->get('b2'));
TagDependency::invalidate($cache, ['t1', 't2']);
$this->assertFalse($cache->get('a1'));
$this->assertFalse($cache->get('a2'));
$this->assertFalse($cache->get('b1'));
$this->assertFalse($cache->get('b2'));
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment