このドキュメントは現在翻訳中です。一部のページが韓国語で表示される場合があります。
メインコンテンツまでスキップ
バージョン: master

오류 처리 (Error Handling)

소개 (Introduction)

새 Laravel 프로젝트를 시작하면 오류 및 예외 처리는 이미 설정되어 있습니다. 그러나 언제든지 애플리케이션의 bootstrap/app.php에서 withExceptions 메서드를 사용하여 애플리케이션이 예외를 보고하고 렌더링하는 방식을 관리할 수 있습니다.

withExceptions 클로저에 제공되는 $exceptions 객체는 Illuminate\Foundation\Configuration\Exceptions의 인스턴스이며, 애플리케이션의 예외 처리를 관리합니다. 이 문서 전반에서 이 객체를 더 자세히 살펴보겠습니다.

설정 (Configuration)

config/app.php 설정 파일의 debug 옵션은 오류에 대한 정보가 실제로 사용자에게 얼마나 표시될지를 결정합니다. 기본적으로 이 옵션은 .env 파일에 저장된 APP_DEBUG 환경 변수의 값을 따르도록 설정되어 있습니다.

로컬 개발 중에는 APP_DEBUG 환경 변수를 true로 설정해야 합니다.

프로덕션 환경에서는 APP_DEBUG 값이 항상 false여야 합니다. 프로덕션에서 이 값이 true로 설정되면 민감한 설정 값이 애플리케이션의 최종 사용자에게 노출될 위험이 있습니다.

예외 처리 (Handling Exceptions)

예외 보고

Laravel에서 예외 보고는 예외를 로그에 기록하거나 Sentry, Flare 같은 외부 서비스로 보내는 데 사용됩니다. 기본적으로 예외는 로깅 설정에 따라 로그에 기록됩니다. 하지만 원하는 방식으로 자유롭게 예외를 기록할 수 있습니다.

서로 다른 타입의 예외를 서로 다른 방식으로 보고해야 한다면, 애플리케이션의 bootstrap/app.php에서 report 예외 메서드를 사용하여 특정 타입의 예외를 보고해야 할 때 실행될 클로저를 등록할 수 있습니다. Laravel은 클로저의 타입 힌트를 검사하여 해당 클로저가 어떤 타입의 예외를 보고하는지 결정합니다.

use App\Exceptions\InvalidOrderException;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->report(function (InvalidOrderException $e) {
// ...
});
})

report 메서드를 사용하여 사용자 정의 예외 보고 콜백을 등록하더라도, Laravel은 애플리케이션의 기본 로깅 설정을 사용하여 예외를 계속 로그에 기록합니다. 예외가 기본 로깅 스택으로 전달되는 것을 중지하려면 보고 콜백을 정의할 때 stop 메서드를 사용하거나 콜백에서 false를 반환하면 됩니다.

use App\Exceptions\InvalidOrderException;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->report(function (InvalidOrderException $e) {
// ...
})->stop();

$exceptions->report(function (InvalidOrderException $e) {
return false;
});
})

특정 예외의 예외 보고를 사용자 정의하려면 보고 가능한 예외를 활용할 수도 있습니다.

전역 로그 컨텍스트

가능한 경우 Laravel은 현재 사용자의 ID를 모든 예외 로그 메시지에 컨텍스트 데이터로 자동 추가합니다. 애플리케이션의 bootstrap/app.php 파일에서 context 예외 메서드를 사용하여 직접 전역 컨텍스트 데이터를 정의할 수 있습니다. 이 정보는 애플리케이션이 작성하는 모든 예외 로그 메시지에 포함됩니다.

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->context(fn () => [
'foo' => 'bar',
]);
})

예외 로그 컨텍스트

모든 로그 메시지에 컨텍스트를 추가하는 것은 유용할 수 있지만, 때로는 특정 예외에 로그에 포함하고 싶은 고유한 컨텍스트가 있을 수 있습니다. 애플리케이션의 예외 클래스 중 하나에 context 메서드를 정의하면, 해당 예외의 로그 항목에 추가되어야 하는 관련 데이터를 지정할 수 있습니다.

<?php

namespace App\Exceptions;

use Exception;

class InvalidOrderException extends Exception
{
// ...

/**
* Get the exception's context information.
*
* @return array<string, mixed>
*/
public function context(): array
{
return ['order_id' => $this->orderId];
}
}

report 헬퍼

때로는 예외를 보고해야 하지만 현재 요청 처리는 계속해야 할 수 있습니다. report 헬퍼 함수는 사용자에게 오류 페이지를 렌더링하지 않고 예외를 빠르게 보고할 수 있게 해줍니다.

public function isValid(string $value): bool
{
try {
// Validate the value...
} catch (Throwable $e) {
report($e);

return false;
}
}

보고된 예외 중복 제거

애플리케이션 전반에서 report 함수를 사용하고 있다면, 가끔 같은 예외를 여러 번 보고하여 로그에 중복 항목이 생길 수 있습니다.

하나의 예외 인스턴스가 반드시 한 번만 보고되도록 하려면, 애플리케이션의 bootstrap/app.php 파일에서 dontReportDuplicates 예외 메서드를 호출할 수 있습니다.

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->dontReportDuplicates();
})

이제 동일한 예외 인스턴스로 report 헬퍼를 호출하면 첫 번째 호출만 보고됩니다.

$original = new RuntimeException('Whoops!');

report($original); // reported

try {
throw $original;
} catch (Throwable $caught) {
report($caught); // ignored
}

report($original); // ignored
report($caught); // ignored

예외 로그 레벨

애플리케이션의 로그에 메시지가 작성될 때, 메시지는 지정된 로그 레벨로 기록됩니다. 로그 레벨은 기록되는 메시지의 심각도나 중요도를 나타냅니다.

앞서 언급했듯이 report 메서드를 사용하여 사용자 정의 예외 보고 콜백을 등록하더라도, Laravel은 애플리케이션의 기본 로깅 설정을 사용하여 예외를 계속 로그에 기록합니다. 하지만 로그 레벨은 메시지가 기록되는 채널에 영향을 줄 수 있으므로, 특정 예외가 어떤 로그 레벨로 기록될지 설정하고 싶을 수 있습니다.

이를 위해 애플리케이션의 bootstrap/app.php 파일에서 level 예외 메서드를 사용할 수 있습니다. 이 메서드는 첫 번째 인수로 예외 타입을, 두 번째 인수로 로그 레벨을 받습니다.

use PDOException;
use Psr\Log\LogLevel;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->level(PDOException::class, LogLevel::CRITICAL);
})

타입별 예외 무시

애플리케이션을 만들다 보면 보고하고 싶지 않은 예외 타입이 있을 수 있습니다. 이러한 예외를 무시하려면 애플리케이션의 bootstrap/app.php 파일에서 dontReport 예외 메서드를 사용할 수 있습니다. 이 메서드에 제공된 클래스는 절대 보고되지 않습니다. 다만 사용자 정의 렌더링 로직은 여전히 가질 수 있습니다.

use App\Exceptions\InvalidOrderException;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->dontReport([
InvalidOrderException::class,
]);
})

또는 예외 클래스에 Illuminate\Contracts\Debug\ShouldntReport 인터페이스를 간단히 "표시"할 수도 있습니다. 예외가 이 인터페이스로 표시되면 Laravel의 예외 핸들러가 절대 보고하지 않습니다.

<?php

namespace App\Exceptions;

use Exception;
use Illuminate\Contracts\Debug\ShouldntReport;

class PodcastProcessingException extends Exception implements ShouldntReport
{
//
}

특정 타입의 예외를 언제 무시할지 더 세밀하게 제어해야 한다면, dontReportWhen 메서드에 클로저를 제공할 수 있습니다.

use App\Exceptions\InvalidOrderException;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->dontReportWhen(function (Throwable $e) {
return $e instanceof PodcastProcessingException &&
$e->reason() === 'Subscription expired';
});
})

내부적으로 Laravel은 이미 일부 오류 타입을 무시합니다. 예를 들어 404 HTTP 오류로 인한 예외, origin 불일치로 생성된 403 HTTP 응답, 유효하지 않은 CSRF 토큰으로 생성된 419 HTTP 응답 등이 있습니다. Laravel이 특정 타입의 예외를 더 이상 무시하지 않도록 하려면, 애플리케이션의 bootstrap/app.php 파일에서 stopIgnoring 예외 메서드를 사용할 수 있습니다.

use Symfony\Component\HttpKernel\Exception\HttpException;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->stopIgnoring(HttpException::class);
})

예외 렌더링

기본적으로 Laravel 예외 핸들러는 예외를 HTTP 응답으로 변환해 줍니다. 하지만 특정 타입의 예외에 대해 사용자 정의 렌더링 클로저를 자유롭게 등록할 수 있습니다. 애플리케이션의 bootstrap/app.php 파일에서 render 예외 메서드를 사용하면 됩니다.

render 메서드에 전달되는 클로저는 Illuminate\Http\Response 인스턴스를 반환해야 하며, 이 인스턴스는 response 헬퍼를 통해 생성할 수 있습니다. Laravel은 클로저의 타입 힌트를 검사하여 해당 클로저가 어떤 타입의 예외를 렌더링하는지 결정합니다.

use App\Exceptions\InvalidOrderException;
use Illuminate\Http\Request;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->render(function (InvalidOrderException $e, Request $request) {
return response()->view('errors.invalid-order', status: 500);
});
})

render 메서드를 사용하여 NotFoundHttpException 같은 Laravel 또는 Symfony 내장 예외의 렌더링 동작을 재정의할 수도 있습니다. render 메서드에 전달된 클로저가 값을 반환하지 않으면 Laravel의 기본 예외 렌더링이 사용됩니다.

use Illuminate\Http\Request;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->render(function (NotFoundHttpException $e, Request $request) {
if ($request->is('api/*')) {
return response()->json([
'message' => 'Record not found.'
], 404);
}
});
})

예외를 JSON으로 렌더링

예외를 렌더링할 때 Laravel은 요청의 Accept 헤더를 기준으로 예외를 HTML 응답으로 렌더링할지 JSON 응답으로 렌더링할지 자동으로 결정합니다. Laravel이 HTML 또는 JSON 예외 응답 중 무엇을 렌더링할지 결정하는 방식을 사용자 정의하고 싶다면 shouldRenderJsonWhen 메서드를 활용할 수 있습니다.

use Illuminate\Http\Request;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->shouldRenderJsonWhen(function (Request $request, Throwable $e) {
if ($request->is('admin/*')) {
return true;
}

return $request->expectsJson();
});
})

예외 응답 사용자 정의

드물게 Laravel 예외 핸들러가 렌더링하는 전체 HTTP 응답을 사용자 정의해야 할 수 있습니다. 이를 위해 respond 메서드를 사용하여 응답 사용자 정의 클로저를 등록할 수 있습니다.

use Symfony\Component\HttpFoundation\Response;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->respond(function (Response $response) {
if ($response->getStatusCode() === 419) {
return back()->with([
'message' => 'The page expired, please try again.',
]);
}

return $response;
});
})

보고 가능 및 렌더링 가능 예외

애플리케이션의 bootstrap/app.php 파일에서 사용자 정의 보고 및 렌더링 동작을 정의하는 대신, 애플리케이션의 예외 클래스에 직접 reportrender 메서드를 정의할 수 있습니다. 이러한 메서드가 존재하면 프레임워크가 자동으로 호출합니다.

<?php

namespace App\Exceptions;

use Exception;
use Illuminate\Http\Request;
use Illuminate\Http\Response;

class InvalidOrderException extends Exception
{
/**
* Report the exception.
*/
public function report(): void
{
// ...
}

/**
* Render the exception as an HTTP response.
*/
public function render(Request $request): Response
{
return response(/* ... */);
}
}

예외가 Laravel 또는 Symfony 내장 예외처럼 이미 렌더링 가능한 예외를 확장하는 경우, 예외의 기본 HTTP 응답을 렌더링하려면 예외의 render 메서드에서 false를 반환할 수 있습니다.

/**
* Render the exception as an HTTP response.
*/
public function render(Request $request): Response|bool
{
if (/** Determine if the exception needs custom rendering */) {

return response(/* ... */);
}

return false;
}

예외에 특정 조건이 충족될 때만 필요한 사용자 정의 보고 로직이 포함되어 있다면, Laravel이 때로는 기본 예외 처리 설정을 사용하여 예외를 보고하도록 지시해야 할 수 있습니다. 이를 위해 예외의 report 메서드에서 false를 반환할 수 있습니다.

/**
* Report the exception.
*/
public function report(): bool
{
if (/** Determine if the exception needs custom reporting */) {

// ...

return true;
}

return false;
}

report 메서드에 필요한 의존성을 타입 힌트로 지정할 수 있으며, Laravel의 서비스 컨테이너가 해당 의존성을 메서드에 자동으로 주입합니다.

보고된 예외 스로틀링

애플리케이션이 매우 많은 수의 예외를 보고한다면, 실제로 로그에 기록되거나 애플리케이션의 외부 오류 추적 서비스로 전송되는 예외 수를 제한하고 싶을 수 있습니다.

예외를 무작위 샘플 비율로 처리하려면 애플리케이션의 bootstrap/app.php 파일에서 throttle 예외 메서드를 사용할 수 있습니다. throttle 메서드는 Lottery 인스턴스를 반환해야 하는 클로저를 받습니다.

use Illuminate\Support\Lottery;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->throttle(function (Throwable $e) {
return Lottery::odds(1, 1000);
});
})

예외 타입에 따라 조건부로 샘플링하는 것도 가능합니다. 특정 예외 클래스의 인스턴스만 샘플링하고 싶다면, 해당 클래스에 대해서만 Lottery 인스턴스를 반환하면 됩니다.

use App\Exceptions\ApiMonitoringException;
use Illuminate\Support\Lottery;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->throttle(function (Throwable $e) {
if ($e instanceof ApiMonitoringException) {
return Lottery::odds(1, 1000);
}
});
})

Lottery 대신 Limit 인스턴스를 반환하여 로그에 기록되거나 외부 오류 추적 서비스로 전송되는 예외에 속도 제한을 적용할 수도 있습니다. 예를 들어, 애플리케이션에서 사용하는 서드파티 서비스가 중단되어 예외가 갑자기 많이 발생하는 상황에서 로그가 폭주하지 않도록 보호하고 싶을 때 유용합니다.

use Illuminate\Broadcasting\BroadcastException;
use Illuminate\Cache\RateLimiting\Limit;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->throttle(function (Throwable $e) {
if ($e instanceof BroadcastException) {
return Limit::perMinute(300);
}
});
})

기본적으로 제한은 예외의 클래스를 속도 제한 키로 사용합니다. Limitby 메서드를 사용해 직접 키를 지정하여 이를 사용자 정의할 수 있습니다.

use Illuminate\Broadcasting\BroadcastException;
use Illuminate\Cache\RateLimiting\Limit;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->throttle(function (Throwable $e) {
if ($e instanceof BroadcastException) {
return Limit::perMinute(300)->by($e->getMessage());
}
});
})

물론 서로 다른 예외에 대해 LotteryLimit 인스턴스를 함께 반환할 수도 있습니다.

use App\Exceptions\ApiMonitoringException;
use Illuminate\Broadcasting\BroadcastException;
use Illuminate\Cache\RateLimiting\Limit;
use Illuminate\Support\Lottery;
use Throwable;

->withExceptions(function (Exceptions $exceptions): void {
$exceptions->throttle(function (Throwable $e) {
return match (true) {
$e instanceof BroadcastException => Limit::perMinute(300),
$e instanceof ApiMonitoringException => Lottery::odds(1, 1000),
default => Limit::none(),
};
});
})

HTTP 예외 (HTTP Exceptions)

일부 예외는 서버의 HTTP 오류 코드를 설명합니다. 예를 들어 "페이지를 찾을 수 없음" 오류(404), "인가되지 않음" 오류(401), 또는 개발자가 발생시킨 500 오류일 수 있습니다. 애플리케이션 어디에서든 이러한 응답을 생성하려면 abort 헬퍼를 사용할 수 있습니다.

abort(404);

사용자 정의 HTTP 오류 페이지

Laravel은 다양한 HTTP 상태 코드에 대한 사용자 정의 오류 페이지를 쉽게 표시할 수 있게 해줍니다. 예를 들어 404 HTTP 상태 코드의 오류 페이지를 사용자 정의하려면 resources/views/errors/404.blade.php view 템플릿을 생성합니다. 이 view는 애플리케이션에서 생성된 모든 404 오류에 대해 렌더링됩니다. 이 디렉터리 안의 view는 해당하는 HTTP 상태 코드와 일치하도록 이름을 지정해야 합니다. abort 함수에서 발생한 Symfony\Component\HttpKernel\Exception\HttpException 인스턴스는 $exception 변수로 view에 전달됩니다.

<h2>{{ $exception->getMessage() }}</h2>

vendor:publish Artisan 명령어를 사용하여 Laravel의 기본 오류 페이지 템플릿을 게시할 수 있습니다. 템플릿을 게시한 뒤에는 원하는 대로 사용자 정의할 수 있습니다.

php artisan vendor:publish --tag=laravel-errors

대체 HTTP 오류 페이지

특정 HTTP 상태 코드 계열에 대한 "대체" 오류 페이지를 정의할 수도 있습니다. 발생한 특정 HTTP 상태 코드에 해당하는 페이지가 없으면 이 페이지가 렌더링됩니다. 이를 위해 애플리케이션의 resources/views/errors 디렉터리에 4xx.blade.php 템플릿과 5xx.blade.php 템플릿을 정의합니다.

대체 오류 페이지를 정의할 때, Laravel은 404, 500, 503 상태 코드에 대해 내부 전용 페이지를 가지고 있으므로 대체 페이지는 이러한 오류 응답에 영향을 주지 않습니다. 이러한 상태 코드에 대해 렌더링되는 페이지를 사용자 정의하려면 각각에 대해 별도의 사용자 정의 오류 페이지를 정의해야 합니다.