SmartWiki开发日记之Laravel缓存扩展

澳门新葡亰网址 1

昨天按照手册教程,动手写一个Auth扩展,按照包独立性的原则,我不希望将Auth::extend()这种方法写在
start.php
中,毫无疑问,我选择了在服务提供器register()方法中注册扩展驱动。然而,事与愿违……

SmartWiki简介请阅读:

澳门新葡亰网址 1

因为SmartWiki的演示站点部署在阿里云上,阿里云有一个128M免费的Memcache服务,刚开始按照Memcached的配置方式配置完后,发现Laravel报错,查看日志报错位置是addServer出错,连不上阿里云的Memcache。

发现问题

很无奈,于是安装阿里云的手册写了一个脚本放到服务器上,结果可以连接,也可以写入。

当我在 LoauthServiceProvider 中这样写的时候:

阿里云提供的脚本如下:

public function register()
{
    //
    Auth::extend('loauth',function($app){});
}
<?php$connect = new Memcached;  //声明一个新的memcached链接$connect->setOption(Memcached::OPT_COMPRESSION, false); //关闭压缩功能$connect->setOption(Memcached::OPT_BINARY_PROTOCOL, true); //使用binary二进制协议$connect->addServer('00000000.ocs.aliyuncs.com', 11211); //添加OCS实例地址及端口号//$connect->setSaslAuthData('aaaaaaaaaa, 'password'); //设置OCS帐号密码进行鉴权,如已开启免密码功能,则无需此步骤$connect->set("hello", "world");echo 'hello: ',$connect->get("hello");print_r( $connect->getVersion;$connect->quit();

报错

翻看laravel的Memcached驱动,在
/vendor/laravel/framework/src/Illuminate/Cache/MemcachedConnector.php
中创建Memcached对象的代码如下:

Call to undefined method IlluminateSupportFacadesAuth::extend()
public function connect(array $servers){    $memcached = $this->getMemcached();    // For each server in the array, we'll just extract the configuration and add    // the server to the Memcached connection. Once we have added all of these    // servers we'll verify the connection is successful and return it back.    foreach ($servers as $server) {        $memcached->addServer(            $server['host'], $server['port'], $server['weight']        );    }    $memcachedStatus = $memcached->getVersion();    if (! is_array($memcachedStatus)) {        throw new RuntimeException('No Memcached servers added.');    }    if (in_array('255.255.255', $memcachedStatus) && count(array_unique($memcachedStatus)) === 1) {        throw new RuntimeException('Could not establish Memcached connection.');    }    return $memcached;}

寻找原因

可以看到laravel的Memcached没有设置setOption方法的选项,仅仅包含最简连接建立,紧接着就调用getVersion来测试是否连通。而阿里云的演示代码是设置了关闭压缩和使用binary二进制协议的选项的。

当时就纳闷了,找原因,怀疑是Auth没注册?检查发现注册了,因为在路由中可以使用;php
artisan clear-compiled
没用;百思不得其解,甚至怀疑是我不小心修改了核心类,还重新下载了一次laravel包,问题依旧。

没办法只能自己来扩展Memcached的功能实现自定义选项。laravel中扩展缓存可以使用Cache::extend来扩展。扩展代码如下:

折腾了一晚上,最终我把目光锁定在 AuthServiceProvider 的 $defer 属性。

Cache::extend('MemcachedExtend', function ($app) {    $memcached = $this->createMemcached($app);    // 从配置文件中读取缓存前缀    $prefix = $app['config']['cache.prefix'];    // 创建 MemcachedStore 对象    $store = new MemcachedStore($memcached, $prefix);    // 创建 Repository 对象,并返回    return new Repository($store);});

/** * 创建Memcached对象 * @param $app * @return mixed */protected function createMemcached($app){    // 从配置文件中读取 Memcached 服务器配置    $servers = $app['config']['cache.stores.MemcachedExtend.servers'];    // 利用 IlluminateCacheMemcachedConnector 类来创建新的 Memcached 对象    $memcached = new Memcached;    foreach ($servers as $server) {        $memcached->addServer(            $server['host'], $server['port'], $server['weight']        );    }    // 如果服务器上的 PHP Memcached 扩展支持 SASL 认证    if (ini_get('memcached.use_sasl') && isset($app['config']['cache.storess.MemcachedExtend.memcached_user']) && isset($app['config']['cache.storess.MemcachedExtend.memcached_pass'])) {        // 从配置文件中读取 sasl 认证用户名        $user = $app['config']['cache.storess.MemcachedExtend.memcached_user'];        // 从配置文件中读取 sasl 认证密码        $pass = $app['config']['cache.storess.MemcachedExtend.memcached_pass'];        // 指定用于 sasl 认证的账号密码        $memcached->setSaslAuthData($user, $pass);    }    //扩展    if (isset($app['config']['cache.stores.MemcachedExtend.options'])) {        foreach ($app['config']['cache.stores.MemcachedExtend.options'] as $key => $option) {            $memcached->setOption($key, $option);        }    }    $memcachedStatus = $memcached->getVersion();    if (! is_array($memcachedStatus)) {        throw new RuntimeException('No Memcached servers added.');    }    if (in_array('255.255.255', $memcachedStatus) && count(array_unique($memcachedStatus)) === 1) {        throw new RuntimeException('Could not establish Memcached connection.');    }    return $memcached;}

根据手册以及注释,我们得知 $defer
属性是用来延迟加载该服务提供器,说直白点就是延迟执行 register()
方法,只需要配合provides()方法即可实现。举个例子:

网上有流传的laravel缓存扩展的文章,其中对配置读取在5.2以上版本不适用。

public function provides()
{
    return array('auth');
}

缓存扩展后的代码是需要创建一个ServiceProvider来注册服务提供者。服务提供者是Laravel应用启动的中心,你自己的应用以及所有Laravel的核心服务都是通过服务提供者启动。

这个是 AuthServiceProvider
里的方法,当框架初始化的时候,会依次加载服务提供器,如果发现这个服务提供器protected
$defer=true 那么就会调用它的 provides()
方法,其返回的数组包含需要延迟加载的服务名称,这样当我们在路由、控制器或者其他地方调用
Auth::METHOD() 的时候,才会去调用提供器的 register() 方法。

但是,我们所谓的“启动”指的是什么?通常,这意味着注册事物,包括注册服务容器绑定、事件监听器、中间件甚至路由。服务提供者是应用配置的中心。

确定症结

澳门新葡亰网址,如果你打开Laravel自带的config/app.php文件,将会看到一个providers数组,这里就是应用所要加载的所有服务提供者类,当然,其中很多是延迟加载的,也就是说不是每次请求都会被加载,只有真的用到它们的时候才会加载。

那么问题来了,既然是被动延迟加载,也就是说当我调用Auth类方法时应该会自动实例化Auth类啊,为什么我在LoauthServiceProvider中调用的时候却提示方法不存在,但是在路由中却可以呢。

所有的服务提供者都继承自IlluminateSupportServiceProvider类。大部分服务提供者都包含两个方法:
register 和 boot
。在register方法中,你唯一要做的事情就是绑事物到服务容器,不要尝试在其中注册事件监听器,路由或者任何其它功能。

发表评论

电子邮件地址不会被公开。 必填项已用*标注

相关文章

网站地图xml地图