jaimezac27/predis

PHP 的一个灵活且功能齐全的 Redis 客户端。

1.x-dev 2023-10-01 16:06 UTC

This package is auto-updated.

Last update: 2024-09-30 01:49:12 UTC


README

Software license Latest stable Latest development Monthly installs Build status Coverage Status

PHP 7.2 及以上版本的一个灵活且功能齐全的 Redis 客户端。

更多关于此项目的详细信息可以在 常见问题解答 中找到。

主要特性

  • 支持 Redis 从 3.07.2
  • 支持使用客户端分片和可插拔键空间分发器的集群。
  • 支持 redis-cluster (Redis >= 3.0)。
  • 支持主从复制配置和 redis-sentinel
  • 使用可自定义的前缀策略透明地键前缀。
  • 单节点和集群(仅客户端分片)上的命令流水线。
  • Redis 事务(Redis >= 2.0)和 CAS 操作(Redis >= 2.2)的抽象。
  • Lua 脚本(Redis >= 2.6)的抽象,并自动在 EVALSHAEVAL 之间切换。
  • 基于 PHP 迭代器的 SCANSSCANZSCANHSCAN(Redis >= 2.8)的抽象。
  • 客户端在第一次命令时懒惰地建立连接,并且可以持久化。
  • 可以通过 TCP/IP(也支持 TLS/SSL 加密)或 UNIX 域套接字建立连接。
  • 支持自定义连接类,以提供不同的网络或协议后端。
  • 灵活的系统用于定义自定义命令和覆盖默认命令。

如何 安装 和使用 Predis

此库可在 Packagist 上找到,以便使用 Composer 管理项目的依赖关系。每个版本的压缩存档都在 GitHub 上提供。

composer require predis/predis

加载库

Predis 依赖于 PHP 的自动加载功能,在需要时加载其文件,并符合 PSR-4 标准。当通过 Composer 管理依赖关系时,自动加载会自动处理,但在没有任何自动加载设施的项目或脚本中也可以利用其自身的自动加载器。

// Prepend a base path if Predis is not available in your "include_path".
require 'Predis/Autoloader.php';

Predis\Autoloader::register();

连接到 Redis

在创建客户端实例时不传递任何连接参数时,Predis 假设 127.0.0.16379 作为默认的主机和端口。默认的 connect() 操作超时时间为 5 秒。

$client = new Predis\Client();
$client->set('foo', 'bar');
$value = $client->get('foo');

连接参数可以以 URI 字符串或命名数组的形式提供。后者是提供参数的首选方式,但 URI 字符串在从非结构化或部分结构化来源读取参数时可能很有用。

// Parameters passed using a named array:
$client = new Predis\Client([
    'scheme' => 'tcp',
    'host'   => '10.0.0.1',
    'port'   => 6379,
]);

// Same set of parameters, passed using an URI string:
$client = new Predis\Client('tcp://10.0.0.1:6379');

可以通过在参数集中添加 password 来访问受密码保护的服务器。当 Redis >= 6.0 上启用了 ACL 时,用户认证需要 usernamepassword

还可以使用 UNIX 域套接字连接到本地 Redis 实例,在这种情况下,参数必须使用 unix 方案并指定套接字文件的路径。

$client = new Predis\Client(['scheme' => 'unix', 'path' => '/path/to/redis.sock']);
$client = new Predis\Client('unix:/path/to/redis.sock');

客户端可以利用TLS/SSL加密连接到安全的远程Redis实例,无需配置stunnel等SSL代理。当连接到运行在各种云托管提供商的节点时,这非常有用。可以使用tls方案以及通过ssl参数传递的适当options数组来启用加密。

// Named array of connection parameters:
$client = new Predis\Client([
  'scheme' => 'tls',
  'ssl'    => ['cafile' => 'private.pem', 'verify_peer' => true],
]);

// Same set of parameters, but using an URI string:
$client = new Predis\Client('tls://127.0.0.1?ssl[cafile]=private.pem&ssl[verify_peer]=1');

支持连接方案redistcp的别名)和redisstls的别名),它们的区别在于包含这些方案的URI字符串的解析遵循它们各自的IANA临时注册文档中描述的规则。

支持的实际连接参数列表可能因每个连接后端而异,因此建议参考它们的特定文档或实现以获取详细信息。

当提供连接参数数组以及适当的选项以指导客户端如何聚合它们(集群、复制或自定义聚合逻辑)时,Predis可以聚合多个连接。在为每个节点提供配置时,可以混合使用命名数组和URI字符串。

$client = new Predis\Client([
    'tcp://10.0.0.1?alias=first-node', ['host' => '10.0.0.2', 'alias' => 'second-node'],
], [
    'cluster' => 'predis',
]);

有关更多详细信息,请参阅本文件的聚合连接部分。

到Redis的连接是懒加载的,这意味着只有当需要时客户端才会连接到服务器。虽然建议让客户端在幕后自行处理,但有时仍希望控制连接何时打开或关闭:这可以通过调用$client->connect()$client->disconnect()轻松实现。请注意,这些方法对聚合连接的影响可能因每个具体实现而异。

客户端配置

可以通过将特定客户端选项传递给Predis\Client::__construct()的第二个参数来配置客户端的许多方面和行为。

$client = new Predis\Client($parameters, ['prefix' => 'sample:']);

选项使用类似DI的容器进行管理,其值仅在需要时才会懒加载。Predis默认支持的客户端选项包括:

  • prefix:应用于命令中找到的每个键的前缀字符串。
  • exceptions:客户端是否应在Redis错误时抛出或返回响应。
  • connections:连接后端列表或连接工厂实例。
  • cluster:指定集群后端(predisredis或可调用对象)。
  • replication:指定复制后端(predissentinel或可调用对象)。
  • aggregate:使用可调用对象配置客户端的自定义聚合连接。
  • parameters:聚合连接的默认连接参数列表。
  • commands:指定通过库使用的命令工厂实例。

用户还可以提供自定义选项,其值为或可调用对象(用于懒加载),它们存储在选项容器中,供库稍后使用。

聚合连接

聚合连接是Predis实现集群和复制的基础,它们用于将多个连接分组到单个Redis节点,并隐藏根据上下文正确处理它们所需的具体逻辑。聚合连接通常需要在创建新的客户端实例时提供连接参数数组以及适当的客户端选项。

集群

Predis可以配置为以传统的客户端分片方法在集群模式下工作,创建独立节点集群并在它们之间分配键空间。这种方法需要某种节点外部健康监控以及节点添加或删除时手动重新平衡键空间。

$parameters = ['tcp://10.0.0.1', 'tcp://10.0.0.2', 'tcp://10.0.0.3'];
$options    = ['cluster' => 'predis'];

$client = new Predis\Client($parameters);

随着Redis 3.0的推出,引入了一种新的监督和协调的集群类型,即redis-cluster。这种方法的算法与分布键空间不同,Redis节点通过gossip协议进行通信以处理健康状态、重新平衡、节点发现和请求重定向来协调自己。为了连接由redis-cluster管理的集群,客户端需要一个节点列表(不一定完整,因为如果需要,它将自动发现新节点)以及将cluster客户端选项设置为redis

$parameters = ['tcp://10.0.0.1', 'tcp://10.0.0.2', 'tcp://10.0.0.3'];
$options    = ['cluster' => 'redis'];

$client = new Predis\Client($parameters, $options);

复制

客户端可以配置为在单个主节点/多个从节点设置中运行,以提供更好的服务可用性。使用复制时,Predis识别只读命令并将它们发送到随机从节点以提供某种负载均衡,并在检测到任何可能修改键空间或键值的值的操作命令时切换到主节点。当从节点失败时,客户端不是引发连接错误,而是尝试在配置中提供的不同从节点之间回退到另一个从节点。

使用客户端在复制模式下所需的基本配置需要将一个Redis服务器识别为主节点(可以通过通过设置连接参数中的role参数为master来完成此操作)以及一个或多个从节点(在这种情况下,将role设置为slave对于从节点是可选的)

$parameters = ['tcp://10.0.0.1?role=master', 'tcp://10.0.0.2', 'tcp://10.0.0.3'];
$options    = ['replication' => 'predis'];

$client = new Predis\Client($parameters, $options);

上述配置具有静态的服务器列表,完全依赖于客户端的逻辑,但可以使用redis-sentinel依赖项来依赖一个更健壮的HA环境,其中sentinel服务器充当客户端的服务发现权威源。客户端与redis-sentinel一起工作所需的最小配置是一个指向多个sentinel实例的连接参数列表,将replication选项设置为sentinel,并将service选项设置为服务的名称

$sentinels = ['tcp://10.0.0.1', 'tcp://10.0.0.2', 'tcp://10.0.0.3'];
$options   = ['replication' => 'sentinel', 'service' => 'mymaster'];

$client = new Predis\Client($sentinels, $options);

如果主节点和从节点配置为要求客户端进行身份验证,则必须通过全局parameters客户端选项提供密码。此选项还可以用于指定不同的数据库索引。客户端选项数组将如下所示

$options = [
    'replication' => 'sentinel',
    'service' => 'mymaster',
    'parameters' => [
        'password' => $secretpassword,
        'database' => 10,
    ],
];

虽然Predis能够区分执行写操作和只读操作的命令,但EVALEVALSHA代表一个特殊情况,其中客户端切换到主节点,因为它无法确定何时可以在从节点上安全执行Lua脚本。虽然这是默认行为,但在某些Lua脚本不执行写操作的情况下,可以提供提示以告知客户端在执行时坚持使用从节点

$parameters = ['tcp://10.0.0.1?role=master', 'tcp://10.0.0.2', 'tcp://10.0.0.3'];
$options    = ['replication' => function () {
    // Set scripts that won't trigger a switch from a slave to the master node.
    $strategy = new Predis\Replication\ReplicationStrategy();
    $strategy->setScriptReadOnly($LUA_SCRIPT);

    return new Predis\Connection\Replication\MasterSlaveReplication($strategy);
}];

$client = new Predis\Client($parameters, $options);
$client->eval($LUA_SCRIPT, 0);             // Sticks to slave using `eval`...
$client->evalsha(sha1($LUA_SCRIPT), 0);    // ... and `evalsha`, too.

examples目录包含一些脚本,演示了如何配置和使用客户端在基本和复杂场景中利用复制。

命令管道

管道化可以帮助提高性能,当需要向服务器发送多个命令时,通过减少由网络往返时间引入的延迟。管道化也适用于聚合连接。客户端可以在可调用的块中执行管道,或返回一个具有链式命令能力的管道实例,这归功于其流畅的接口

// Executes a pipeline inside the given callable block:
$responses = $client->pipeline(function ($pipe) {
    for ($i = 0; $i < 1000; $i++) {
        $pipe->set("key:$i", str_pad($i, 4, '0', 0));
        $pipe->get("key:$i");
    }
});

// Returns a pipeline that can be chained thanks to its fluent interface:
$responses = $client->pipeline()->set('foo', 'bar')->get('foo')->execute();

事务

客户端提供了基于MULTIEXEC的Redis事务抽象,其接口与命令管道类似

// Executes a transaction inside the given callable block:
$responses = $client->transaction(function ($tx) {
    $tx->set('foo', 'bar');
    $tx->get('foo');
});

// Returns a transaction that can be chained thanks to its fluent interface:
$responses = $client->transaction()->set('foo', 'bar')->get('foo')->execute();

该抽象可以执行检查和设置操作,归功于WATCHUNWATCH,并提供在WATCH键被触摸时由Redis中止的事务的自动重试。有关使用CAS的事务示例,请参阅以下示例

添加新命令

尽管我们努力将Predis更新以保持与Redis中所有可用命令的同步,您可能更愿意坚持使用库的旧版本或提供不同的方式来过滤特定命令的参数或解析响应。为此,Predis提供了实现新命令类的能力,以定义或覆盖客户端使用的默认命令工厂中的命令。

// Define a new command by extending Predis\Command\Command:
class BrandNewRedisCommand extends Predis\Command\Command
{
    public function getId()
    {
        return 'NEWCMD';
    }
}

// Inject your command in the current command factory:
$client = new Predis\Client($parameters, [
    'commands' => [
        'newcmd' => 'BrandNewRedisCommand',
    ],
]);

$response = $client->newcmd();

还有一种方法可以发送未经过滤参数或解析响应的原始命令。用户必须以数组形式提供命令的参数列表,遵循由Redis命令文档定义的签名。

$response = $client->executeRaw(['SET', 'foo', 'bar']);

脚本命令

虽然可以在Redis 2.6+版本中使用Lua脚本直接使用EVALEVALSHA命令利用Lua脚本,但Predis提供了脚本命令作为基于这些命令的高级抽象,以简化操作。脚本命令可以注册到客户端使用的命令工厂中,并且可以像普通Redis命令一样访问,但它们定义的Lua脚本会被发送到服务器进行远程执行。内部默认使用EVALSHA,并通过SHA1哈希值识别脚本以节省带宽,但在需要时可以使用EVAL作为后备。

// Define a new script command by extending Predis\Command\ScriptCommand:
class ListPushRandomValue extends Predis\Command\ScriptCommand
{
    public function getKeysCount()
    {
        return 1;
    }

    public function getScript()
    {
        return <<<LUA
math.randomseed(ARGV[1])
local rnd = tostring(math.random())
redis.call('lpush', KEYS[1], rnd)
return rnd
LUA;
    }
}

// Inject the script command in the current command factory:
$client = new Predis\Client($parameters, [
    'commands' => [
        'lpushrand' => 'ListPushRandomValue',
    ],
]);

$response = $client->lpushrand('random_values', $seed = mt_rand());

可定制的连接后端

Predis可以使用不同的连接后端连接到Redis。内置的Relay集成利用了Relay扩展,通过在PHP共享运行时内存中缓存Redis数据集的部分副本来获得主要性能提升。

$client = new Predis\Client('tcp://127.0.0.1', [
    'connections' => 'relay',
]);

开发者可以创建自己的连接类以支持全新的网络后端、扩展现有类或提供完全不同的实现。连接类必须实现Predis\Connection\NodeConnectionInterface或扩展Predis\Connection\AbstractConnection

class MyConnectionClass implements Predis\Connection\NodeConnectionInterface
{
    // Implementation goes here...
}

// Use MyConnectionClass to handle connections for the `tcp` scheme:
$client = new Predis\Client('tcp://127.0.0.1', [
    'connections' => ['tcp' => 'MyConnectionClass'],
]);

有关如何创建新的连接后端的更深入的了解,您可以参考Predis\Connection命名空间中标准连接类的实际实现。

开发

报告错误和贡献代码

对Predis的贡献无论是以新特性、错误修复还是只是错误报告的形式,都受到高度赞赏。我们只要求您遵守问题和拉取请求模板。

测试套件

注意:请勿在任何生产环境或包含您感兴趣数据的Redis实例上运行Predis提供的测试套件!

Predis有一个全面的测试套件,涵盖库的各个方面,并且可以可选地针对运行的Redis实例执行集成测试(需要>= 2.4.0以验证每个命令实现的正确性。对于不受支持的Redis命令的集成测试将自动跳过。如果您没有Redis运行,集成测试可以禁用。有关测试此库的更多详细信息,请参阅测试README

Predis使用GitHub Actions进行持续集成,过去和当前构建的历史可以在它的操作页面找到。

许可证

Predis的代码在MIT许可证的条款下分发(请参阅LICENSE)。