溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Laravel異常上下文解決方案是什么

發(fā)布時間:2022-01-06 19:44:35 來源:億速云 閱讀:154 作者:柒染 欄目:開發(fā)技術(shù)

Laravel異常上下文解決方案是什么,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

前言

異常時我們通常希望在用戶側(cè)給一個友好的提示,但默認使用框架的異常處理方案是不 OK 的。

最近項目遇到一個情況,我們在遇到用戶訪問某個信息沒有權(quán)限的時候,希望提示詳細的原因,比如當訪問一個團隊資源時非成員訪問的場景下會提示一個:您不是 [xxxxxx] 團隊的成員,暫時無法查看,可<申請加入>,同時需要顯示打碼后的團隊名稱,以及加入按鈕,可是接口方的邏輯是當沒有權(quán)限時直接 abort 了:

abort_if(!$user->isMember($resouce->team), 403, '您無權(quán)訪問該資源');

得到的響應結(jié)果如下:

HTTP/1.0 403 Forbidden

{
	"message": "您無權(quán)訪問該資源"
}

我們不可能將 message 用 html 來完成前端提示頁的展示,這樣耦合性太強,違背了前后端分離的原則。我們的目標是返回如下的格式即可解決:

HTTP/1.0 403 Forbidden

{
	"message": "您無權(quán)訪問該資源",
	"team": {
		"id": "abxT8sioa0Ms",
		"name": "CoDesign****"
	}
}

通過攜帶上下文的方法傳遞數(shù)據(jù),方便了前端同學自由組合。

開始改造

當然這并不是什么復雜的事情,直接修改原來的 abort_if 即可解決:

- abort_if(!$user->isMember($resouce->team), 403, '您無權(quán)訪問該資源');
+ if (!$user->isMember($resouce->team)) {
+	return response()->json([
+		'message' => '您無權(quán)訪問該資源',
+		'team' => [
+			'id' => $resouce->team_id,
+			'name'=> $resouce->team->desensitised_name,
+		]
+	], 403);
+ }

這樣看起來解決了問題,可是試想一下,如果是在閉包里面檢測到異常想要退出,上面這種 return 式的寫法就會比較難搞了,畢竟 return 只會終止最近的上下文環(huán)境,我們還是希望像 abort 一樣能終止整個應用的執(zhí)行,再進行另一番改造。

優(yōu)化實現(xiàn)

看了 abort 源碼,我發(fā)現(xiàn)它的第一個參數(shù)其實支持 \Symfony\Component\HttpFoundation\Response 實例,而上面????我們 return 的結(jié)果就是它的實例,所以我們只需要改成這樣就可以了:

 if (!$user->isMember($resouce->team)) {
	abort(response()->json([
		'message' => '您無權(quán)訪問該資源',
		'team' => [
			'id' => $resouce->team_id,
			'name'=> $resouce->team->desensitised_name,
		]
	], 403));
 }

新的問題來了,如果需要復用的時候還是比較尷尬,這段代碼將會重復出現(xiàn)在各種有此權(quán)限判斷的地方,這并不是我們想要的。

邏輯復用

為了達到邏輯復用,我認證看了 \App\Exceptions\Handler 的實現(xiàn),發(fā)現(xiàn)父類的 render 方法還有這么一個設計:

public function render($request, Throwable $e)
{
    if (method_exists($e, 'render') && $response = $e->render($request)) {
        return Router::toResponse($request, $response);
    } elseif ($e instanceof Responsable) {
        return $e->toResponse($request);
    }

    //...

所以,我們可以將這個邏輯抽離為一個獨立的異常類,實現(xiàn) render 方法即可:

$ ./artisan make:exception NotTeamMemberException

代碼如下:

<?php

namespace App\Exceptions;

use App\Team;

class NotTeamMemberException extends \Exception
{
    public Team $team;

    public function __construct(Team $team, $message = "")
    {
        $this->team = $team;
        parent::__construct($message, 403);
    }

    public function render()
    {
        return response()->json(
            [
                'message' => !empty($this->message) ? $this->message : '您無權(quán)訪問該資源',
                'team' => [
                    'id' => $this->team->id,
                    'name' => $this->team->desensitised_name,
                ],
            ],
            403
        );
    }
}

這樣一來,我們的邏輯就變成了:

if (!$user->isMember($resouce->team)) {
 	throw new NotTeamMemberException($resouce->team, '您無權(quán)訪問該資源');
}

當然也可以簡化為:

\throw_if(!$user->isMember($resouce->team), NotTeamMemberException::class, $resouce->team, '您無權(quán)訪問該資源');

問題到這里總算以一個比較完美的方式解決了,如果你有更好的方案歡迎評論探討。

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

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI