中文字幕日韩精品一区二区免费_精品一区二区三区国产精品无卡在_国精品无码专区一区二区三区_国产αv三级中文在线

Laravel集成phpCAS的過程是怎樣的

Laravel集成phpCAS的過程是怎樣的,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

創(chuàng)新互聯(lián)專注于井研網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗。 熱誠為您提供井研營銷型網(wǎng)站建設(shè),井研網(wǎng)站制作、井研網(wǎng)頁設(shè)計、井研網(wǎng)站官網(wǎng)定制、微信小程序開發(fā)服務(wù),打造井研網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供井研網(wǎng)站排名全網(wǎng)營銷落地服務(wù)。

Laravel 集成 phpCAS 踩坑記

CAS 是目前比較流行的單點登錄協(xié)議,官方提供了 php 版本的 client 端 phpCAS,到目前為止其編碼風格還一直停留在 PEAR 時代,連命名空間都沒有使用。好在 phpCAS 支持 composer 引入,做過幾個 Laravel 項目引入也沒有什么問題,然而這兩天有一個項目需要從單機部署變成多機部署,萬萬沒想到在這里踩了一些坑,在此記錄一下。

回調(diào)坑

在跳轉(zhuǎn)到 CAS Server 進行認證時發(fā)現(xiàn),傳入的回調(diào)地址被加上了端口8080。因為是多機部署,所以訪問請求會先經(jīng)過負載均衡器(阿里云 SLB),再到達 web 服務(wù)器,而這個8080是 web 服務(wù)器的監(jiān)聽端口。

于是追查 phpCAS 生成回調(diào)地址的邏輯,發(fā)現(xiàn)有這么一段代碼:

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 并不會傳給后端服務(wù)器 X-FORWARDED-PORT 這個 http 頭,因此 phpCAS 就會拿到 $_SERVER['SERVER_PORT'] 也就是 nginx 的端口8080。

好在 phpCAS 提供了 setFixedServiceURL 函數(shù),可以讓我們手動去設(shè)定回調(diào)地址:

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

這下回調(diào)地址正常了,但是從 CAS Server 返回到 client 端時被告知 ticket 無效。

繼續(xù)查日志和代碼,發(fā)現(xiàn)這里是自己疏忽了,當 CAS Server 返回到 client 端時頁面的 url 是 http://client/login?ticket=xxxxx,而 client 端使用 ticket 向 server 換取用戶信息時還需要帶上申請該 ticket 時的回調(diào)地址(service),server 端會校驗 ticket 和 service 是否一致,而申請 ticket 時的 service 應(yīng)該是 http://client/login,因此我們需要把 url 里的 ticket 參數(shù)去掉。

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

getUrlWithoutTicket 函數(shù)如下:

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。拉到線上一試,發(fā)現(xiàn)不行,phpCAS 對 $_SESSION 變量的變更并沒有被寫到 redis 里,怎么回事!

于是追了一下 Laravel 的 Session 實現(xiàn),發(fā)現(xiàn)并不是想象中的使用 session_set_save_handler 來注冊 Session 讀寫邏輯,也就是說 Laravel 的 Session 其實并沒有修改 php 的 $_SESSION 的讀寫邏輯,直接操作 $_SESSION 還是走的默認行為(讀寫本地文件)。

那好吧,好在 Laravel 的幾個 SessionDriver 都實現(xiàn)了 SessionHandlerInterface 接口,我們可以自己調(diào)用一下 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 的代碼,發(fā)現(xiàn) redis driver 的父類 Illuminate\Session\CacheBasedSessionHandlerwrite 方法返回的是 void。于是提了一個 PR 打算修一下,沒想到被拒絕,原來是之前有人修過又被 revert 了,說是會導致服務(wù)器卡住,然而我并沒有找到具體的 issue。

那好吧,memcache 和 redis 都是繼承的這個父類,那我就換只好 database 試試看。

這回 session_write_close 不報錯了,但是 CAS 登錄還是有問題,不斷在 CAS server 和回調(diào) url 之間跳轉(zhuǎn)。于是又追了一路 log 和代碼,發(fā)現(xiàn) database driver 類 Illuminate\Session\DatabaseSessionHandlerdestroy 方法在銷毀 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;  所對應(yīng)操作 session 表的 sql 執(zhí)行的是 update 而不是 insert,也就是沒能將 session 數(shù)據(jù)寫入 session 表!

實在沒有辦法了,只能自己寫一個 Session Wrapper 來處理。

從上面兩個情況來看,redis driver 比較好處理,只要能在調(diào)用 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; // 這里
    }
}

然后調(diào)用 session_set_save_handler 變成

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

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對創(chuàng)新互聯(lián)的支持。

網(wǎng)站欄目:Laravel集成phpCAS的過程是怎樣的
網(wǎng)頁路徑:http://www.rwnh.cn/article10/pcoigo.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供響應(yīng)式網(wǎng)站、ChatGPT、自適應(yīng)網(wǎng)站、網(wǎng)站建設(shè)、微信公眾號、云服務(wù)器

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

手機網(wǎng)站建設(shè)
隆安县| 泗阳县| 应城市| 开阳县| 钟山县| 滕州市| 海丰县| 会同县| 肃宁县| 普陀区| 开化县| 九龙城区| 棋牌| 武山县| 体育| 民和| 白玉县| 从化市| 团风县| 醴陵市| 万全县| 威海市| 万源市| 仪征市| 文山县| 介休市| 南溪县| 太康县| 清远市| 尚志市| 西充县| 井陉县| 南昌市| 海口市| 青冈县| 德钦县| 勐海县| 桐柏县| 玉门市| 壶关县| 商都县|