Skip to content

Latest commit

 

History

History
396 lines (280 loc) · 11.1 KB

context.md

File metadata and controls

396 lines (280 loc) · 11.1 KB

簡介

Laravel 的「上下文」功能使您能夠在應用程式中的請求、工作和命令執行期間捕獲、檢索和共享資訊。這些捕獲的資訊也包含在您的應用程式寫入的日誌中,讓您更深入地了解在寫入日誌條目之前發生的周圍程式碼執行歷史,並允許您在分佈式系統中跟踪執行流程。

如何運作

了解 Laravel 的上下文功能的最佳方式是通過使用內建的記錄功能來實際操作。要開始,您可以使用 Context 門面將資訊添加到上下文。在此示例中,我們將使用中介層在每個傳入請求上將請求 URL 和唯一的追蹤 ID 添加到上下文中:

<?php

namespace App\Http\Middleware;

use Closure;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Context;
use Illuminate\Support\Str;
use Symfony\Component\HttpFoundation\Response;

class AddContext
{
    /**
     * Handle an incoming request.
     */
    public function handle(Request $request, Closure $next): Response
    {
        Context::add('url', $request->url());
        Context::add('trace_id', Str::uuid()->toString());

        return $next($request);
    }
}

添加到上下文的資訊會自動附加為任何在請求期間寫入的日誌條目的元資料。將上下文附加為元資料允許將資訊傳遞給個別的日誌條目與通過 Context 共享的資訊區分開來。例如,假設我們寫入以下日誌條目:

Log::info('User authenticated.', ['auth_id' => Auth::id()]);

寫入的日誌將包含傳遞給日誌條目的 auth_id,但它還將包含上下文的 urltrace_id 作為元資料:

User authenticated. {"auth_id":27} {"url":"https://example.com/login","trace_id":"e04e1a11-e75c-4db3-b5b5-cfef4ef56697"}

捕獲上下文

堆疊

檢索上下文

確定項目存在性

移除上下文

隱藏上下文

事件

脫水

水合

上下文中添加的信息也可以提供给分派到队列的作业。例如,假设我们在添加一些信息到上下文后将 ProcessPodcast 作业分派到队列:

// In our middleware...
Context::add('url', $request->url());
Context::add('trace_id', Str::uuid()->toString());

// In our controller...
ProcessPodcast::dispatch($podcast);

当作业被分派时,当前存储在上下文中的任何信息都会被捕获并与作业共享。然后,在作业执行时,捕获的信息会被重新注入到当前上下文中。因此,如果我们作业的 handle 方法是写入日志:

class ProcessPodcast implements ShouldQueue
{
    use Queueable;

    // ...

    /**
     * Execute the job.
     */
    public function handle(): void
    {
        Log::info('Processing podcast.', [
            'podcast_id' => $this->podcast->id,
        ]);

        // ...
    }
}

生成的日志条目将包含在最初分派作业的请求期间添加到上下文中的信息:

處理播客。{"podcast_id":95} {"url":"https://example.com/login","trace_id":"e04e1a11-e75c-4db3-b5b5-cfef4ef56697"}

雖然我們專注於 Laravel 上下文的內建日誌相關功能,以下文件將說明上下文如何允許您跨 HTTP 請求/排隊作業邊界共享信息,甚至如何添加隱藏上下文數據,這些數據不會與日誌條目一起寫入。

捕獲上下文

您可以使用 Context 門面的 add 方法將信息存儲在當前上下文中:

use Illuminate\Support\Facades\Context;

Context::add('key', 'value');

要一次添加多個項目,您可以將關聯數組傳遞給 add 方法:

Context::add([
    'first_key' => 'value',
    'second_key' => 'value',
]);

add 方法將覆蓋共享相同鍵的任何現有值。如果您只希望在鍵尚不存在時將信息添加到上下文中,則可以使用 addIf 方法:

Context::add('key', 'first');

Context::get('key');
// "first"

Context::addIf('key', 'second');

Context::get('key');
// "first"

條件上下文

when 方法可用於根據給定條件向上下文添加數據。提供給 when 方法的第一個閉包將在給定條件求值為 true 時被調用,而第二個閉包將在條件求值為 false 時被調用:

use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Context;

Context::when(
    Auth::user()->isAdmin(),
    fn ($context) => $context->add('permissions', Auth::user()->permissions),
    fn ($context) => $context->add('permissions', []),
);

堆疊

Context 提供了創建 "堆疊" 的能力,這些堆疊是按照添加順序存儲的數據列表。您可以通過調用 push 方法將信息添加到堆疊中:

use Illuminate\Support\Facades\Context;

Context::push('breadcrumbs', 'first_value');

Context::push('breadcrumbs', 'second_value', 'third_value');

Context::get('breadcrumbs');
// [
//     'first_value',
//     'second_value',
//     'third_value',
// ]

堆疊可用於捕獲有關請求的歷史信息,例如應用程序中正在發生的事件。例如,您可以創建一個事件監聽器,在每次執行查詢時將信息推送到堆疊中,捕獲查詢 SQL 和持續時間作為一個元組:

use Illuminate\Support\Facades\Context;
use Illuminate\Support\Facades\DB;

DB::listen(function ($event) {
    Context::push('queries', [$event->time, $event->sql]);
});

您可以使用 stackContainshiddenStackContains 方法來確定值是否在堆疊中:

if (Context::stackContains('breadcrumbs', 'first_value')) {
    //
}

if (Context::hiddenStackContains('secrets', 'first_value')) {
    //
}

stackContainshiddenStackContains 方法還接受閉包作為它們的第二個參數,從而更好地控制值比較操作:

use Illuminate\Support\Facades\Context;
use Illuminate\Support\Str;

return Context::stackContains('breadcrumbs', function ($value) {
    return Str::startsWith($value, 'query_');
});

檢索上下文

您可以使用 Context 門面的 get 方法從上下文中檢索信息:

use Illuminate\Support\Facades\Context;

$value = Context::get('key');

only 方法可用於檢索上下文中信息的子集:

$data = Context::only(['first_key', 'second_key']);

pull 方法可用於從上下文中檢索信息並立即從上下文中刪除它:

$value = Context::pull('key');

如果上下文數據存儲在堆疊中,則可以使用 pop 方法從堆疊中彈出項目:

Context::push('breadcrumbs', 'first_value', 'second_value');

Context::pop('breadcrumbs')
// second_value

Context::get('breadcrumbs');
// ['first_value'] 

如果您想要檢索上下文中存儲的所有信息,可以調用 all 方法:

$data = Context::all();

確定項目存在性

您可以使用 hasmissing 方法來確定上下文是否為給定鍵存儲了任何值:

use Illuminate\Support\Facades\Context;

if (Context::has('key')) {
    // ...
}

if (Context::missing('key')) {
    // ...
}

has 方法將返回 true,無論存儲的值如何。因此,例如,具有 null 值的鍵將被認為存在:

Context::add('key', null);

Context::has('key');
// true

移除上下文

forget 方法可用於從當前上下文中刪除鍵及其值:

use Illuminate\Support\Facades\Context;

Context::add(['first_key' => 1, 'second_key' => 2]);

Context::forget('first_key');

Context::all();

// ['second_key' => 2]

您可以通過向 forget 方法提供一個陣列來一次性忘記多個鍵:

Context::forget(['first_key', 'second_key']);

隱藏上下文

上下文提供了存儲“隱藏”數據的功能。這些隱藏信息不會附加到日誌中,也無法通過上面記錄的數據檢索方法訪問。上下文提供了一組不同的方法來與隱藏上下文信息進行交互:

use Illuminate\Support\Facades\Context;

Context::addHidden('key', 'value');

Context::getHidden('key');
// 'value'

Context::get('key');
// null

“隱藏”方法與上面記錄的非隱藏方法的功能相同:

Context::addHidden(/* ... */);
Context::addHiddenIf(/* ... */);
Context::pushHidden(/* ... */);
Context::getHidden(/* ... */);
Context::pullHidden(/* ... */);
Context::popHidden(/* ... */);
Context::onlyHidden(/* ... */);
Context::allHidden(/* ... */);
Context::hasHidden(/* ... */);
Context::forgetHidden(/* ... */);

事件

上下文分發了兩個事件,允許您鉤入上下文的水合和脫水過程。

為了說明這些事件如何使用,假設在應用程序的中間件中,您根據傳入的 HTTP 請求的 Accept-Language 標頭設置 app.locale 配置值。上下文的事件允許您在請求期間捕獲此值並在隊列上恢復它,確保在隊列上發送的通知具有正確的 app.locale 值。我們可以使用上下文的事件和隱藏數據來實現這一點,以下文檔將進行說明。

脫水

每當作業被派發到隊列時,上下文中的數據會被“脫水”,並與作業的有效載荷一起捕獲。Context::dehydrating 方法允許您註冊一個在脫水過程中將被調用的閉包。在此閉包中,您可以對將與排隊作業共享的數據進行更改。

通常,您應該在應用程序的 AppServiceProvider 類的 boot 方法中註冊 dehydrating 回調:

use Illuminate\Log\Context\Repository;
use Illuminate\Support\Facades\Config;
use Illuminate\Support\Facades\Context;

/**
 * Bootstrap any application services.
 */
public function boot(): void
{
    Context::dehydrating(function (Repository $context) {
        $context->addHidden('locale', Config::get('app.locale'));
    });
}

Note

您不應在 dehydrating 回調中使用 Context 門面,因為這將改變當前進程的上下文。請確保只對回調傳遞的存儲庫進行更改。

已填充

當排隊的工作開始在佇列上執行時,與工作共享的任何上下文將被“填充”回當前上下文。Context::hydrated 方法允許您註冊一個閉包,在填充過程中將被調用。

通常,您應該在應用程式的 AppServiceProvider 類別的 boot 方法中註冊 hydrated 回調:

use Illuminate\Log\Context\Repository;
use Illuminate\Support\Facades\Config;
use Illuminate\Support\Facades\Context;

/**
 * Bootstrap any application services.
 */
public function boot(): void
{
    Context::hydrated(function (Repository $context) {
        if ($context->hasHidden('locale')) {
            Config::set('app.locale', $context->getHidden('locale'));
        }
    });
}

Note

應該避免在 hydrated 回調中使用 Context 門面,而應確保僅對傳遞給回調的存儲庫進行更改。