• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • dedecms
  • ecshop
  • z-blog
  • UcHome
  • UCenter
  • drupal
  • WordPress
  • 帝国cms
  • phpcms
  • 动易cms
  • phpwind
  • discuz
  • 科汛cms
  • 风讯cms
  • 建站教程
  • 运营技巧
您的位置:首页 > CMS教程 >建站教程 > 踩坑分享:Laravel集成phpCAS过程

踩坑分享:Laravel集成phpCAS过程

作者:站长图库 字体:[增加 减小] 来源:互联网 时间:2022-04-29

站长图库向大家介绍了Laravel,集成phpCAS等相关知识,希望对您有所帮助

下面给大家分享一个Laravel 集成 phpCAS 踩坑记,希望对需要的朋友有所帮助!


Laravel 集成 phpCAS 踩坑记

CAS 是目前比较流行的单点登录协议,官方提供了 php 版本的 client 端 phpCAS,到目前为止其编码风格还一直停留在 PEAR 时代,连命名空间都没有使用。好在 phpCAS 支持 composer 引入,做过几个 Laravel 项目引入也没有什么问题,然而这两天有一个项目需要从单机部署变成多机部署,万万没想到在这里踩了一些坑,在此记录一下。

回调坑

在跳转到 CAS Server 进行认证时发现,传入的回调地址被加上了端口8080。因为是多机部署,所以访问请求会先经过负载均衡器(阿里云 SLB),再到达 web 服务器,而这个8080是 web 服务器的监听端口。

于是追查 phpCAS 生成回调地址的逻辑,发现有这么一段代码:

if (empty($_SERVER['HTTP_X_FORWARDED_PORT'])) {    $server_port = $_SERVER['SERVER_PORT'];} else {    $ports = explode(',', $_SERVER['HTTP_X_FORWARDED_PORT']);    $server_port = $ports[0];}

而阿里云的 SLB 并不会传给后端服务器 X-FORWARDED-PORT 这个 http 头,因此 phpCAS 就会拿到 $_SERVER['SERVER_PORT'] 也就是 nginx 的端口8080。

好在 phpCAS 提供了 setFixedServiceURL 函数,可以让我们手动去设定回调地址:

phpCAS::setFixedServiceURL($request->url());

这下回调地址正常了,但是从 CAS Server 返回到 client 端时被告知 ticket 无效。

继续查日志和代码,发现这里是自己疏忽了,当 CAS Server 返回到 client 端时页面的 url 是 http://client/login?ticket=xxxxx,而 client 端使用 ticket 向 server 换取用户信息时还需要带上申请该 ticket 时的回调地址(service),server 端会校验 ticket 和 service 是否一致,而申请 ticket 时的 service 应该是 http://client/login,因此我们需要把 url 里的 ticket 参数去掉。

phpCAS::setFixedServiceURL($this->getUrlWithoutTicket($request));

getUrlWithoutTicket 函数如下:

private function getUrlWithoutTicket(Request $request){    $query = parse_query($request->getQueryString());    unset($query['ticket']);    $question = $request->getBaseUrl().$request->getPathInfo() == '/' ? '/?' : '?';     return $query ? $request->url().$question.http_build_query($query) : $request->url();}


Session 坑

这是一个 phpCAS + Laravel 的组合坑,坑得死去活来没脾气。

PHP 默认是 Session 存储方式是文件,因此单机变多机一个很重要的点就是处理 Session 共享。方案也很简单,就是把 Session 存储方式从文件改成 redis/memecache/database 等。

Laravel 默认提供了这些 driver,于是兴冲冲地改了下 .env 文件,把 SESSION_DRIVER 改成 redis。拉到线上一试,发现不行,phpCAS 对 $_SESSION 变量的变更并没有被写到 redis 里,怎么回事!

于是追了一下 Laravel 的 Session 实现,发现并不是想象中的使用 session_set_save_handler 来注册 Session 读写逻辑,也就是说 Laravel 的 Session 其实并没有修改 php 的 $_SESSION 的读写逻辑,直接操作 $_SESSION 还是走的默认行为(读写本地文件)。

那好吧,好在 Laravel 的几个 SessionDriver 都实现了 SessionHandlerInterface 接口,我们可以自己调用一下 session_set_save_handler:

session_set_save_handler(app(StartSession::class)->getSession($request)->getHandler());

万万没想到报错!

session_write_close(): Session callback expects true/false return value

追了一下 Laravel 的代码,发现 redis driver 的父类 Illuminate\Session\CacheBasedSessionHandler 的 write 方法返回的是 void。于是提了一个 PR 打算修一下,没想到被拒绝,原来是之前有人修过又被 revert 了,说是会导致服务器卡住,然而我并没有找到具体的 issue。

那好吧,memcache 和 redis 都是继承的这个父类,那我就换只好 database 试试看。

这回 session_write_close 不报错了,但是 CAS 登录还是有问题,不断在 CAS server 和回调 url 之间跳转。于是又追了一路 log 和代码,发现 database driver 类 Illuminate\Session\DatabaseSessionHandler 的 destroy 方法在销毁 Session 之后没有将 $this->exists 属性标记为 false,而 phpCAS 有一处逻辑是 renameSession

$old_session = $_SESSION;session_destroy();$session_id = preg_replace('/[^a-zA-Z0-9\-]/', '', $ticket);session_id($session_id);session_start();$_SESSION = $old_session;

后果就是 $_SESSION = $old_session; 所对应操作 session 表的 sql 执行的是 update 而不是 insert,也就是没能将 session 数据写入 session 表!

实在没有办法了,只能自己写一个 Session Wrapper 来处理。

从上面两个情况来看,redis driver 比较好处理,只要能在调用 write 方法时返回 true 就可以了。所以代码如下

namespace App\Services; use SessionHandlerInterface; class MySession implements SessionHandlerInterface{    /**     * @var SessionHandlerInterface     */    protected $realHdl;     /**     * Session constructor.     * @param SessionHandlerInterface $realHdl     */    public function __construct(SessionHandlerInterface $realHdl)    {        $this->realHdl = $realHdl;    }     public function close()    {        return $this->realHdl->close();    }     public function destroy($session_id)    {        return $this->realHdl->destroy($session_id);    }     public function gc($maxlifetime)    {        return $this->realHdl->gc($maxlifetime);    }     public function open($save_path, $name)    {        return $this->realHdl->open($save_path, $name);    }     public function read($session_id)    {        return $this->realHdl->read($session_id) ?: '';    }     public function write($session_id, $session_data)    {        $this->realHdl->write($session_id, $session_data);         return true; // 这里    }}

然后调用 session_set_save_handler 变成

session_set_save_handler(new MySession(app(StartSession::class)->getSession($request)->getHandler()));

Done !


分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • laravel与thinkphp之间的区别与优缺点
  • 说说在Laravel中怎么执行Shell命令 ?
  • Laravel中用Observer事件致Redis队列异常问题
  • 手把手带你使用Vue + Laravel开发一个简单的 CRUD 应用
  • 分享个人推荐的Laravel或其它框架的编程规范
  • Laravel中三种中间件的作用
  • 解决Laravel使用laravel-excel扩展包(maatwebsite/excel)导入报错问题
  • 分享一个顺丰同城配送的扩展包并在laravel中使用
  • 介绍Laravel8路由模块新增missing方法
  • 解析如何进行Laravel表单验证分层设计和验证场景应用

相关文章

  • 2022-04-29百度编辑器上传word文件转为html
  • 2022-04-29详解使用PHP编写爬虫的方法
  • 2022-04-29帝国CMS灵动标签去除重复标题信息
  • 2022-04-29Photoshop制作地裂火焰效果艺术字教程
  • 2022-04-29Photoshop创建有光泽的塑料3D文字教程
  • 2022-04-29利用CSS如何实现全兼容的毛玻璃效果?
  • 2022-04-29Centos7 yum安装php的方法
  • 2022-04-29帝国CMS二次开发会员登陆赠送积分
  • 2022-04-29JavaScript中Number()方法的两种用法
  • 2022-04-29如何解决php json_encode乱码的问题

文章分类

  • dedecms
  • ecshop
  • z-blog
  • UcHome
  • UCenter
  • drupal
  • WordPress
  • 帝国cms
  • phpcms
  • 动易cms
  • phpwind
  • discuz
  • 科汛cms
  • 风讯cms
  • 建站教程
  • 运营技巧

最近更新的内容

    • javascript如何删除数组里的某个元素
    • TP6+vue-element-admin实现后台登录验证码
    • 分享个人推荐的Laravel或其它框架的编程规范
    • 聊聊怎么在docker上部署运行workerman
    • ThinkPHP5框架中Redis是如何使用和封装?
    • Photoshop制作可爱的卡通剪纸字
    • Ajax实现登录案例
    • 详解Angular中的Observable(可观察对象)
    • 详解Node.js Buffer的使用
    • Photoshop制作透明大气的导航按钮

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有