본문으로 건너뛰기
버전: 9.x

HTTP 응답 (HTTP Responses)

응답 생성하기

문자열 및 배열

모든 라우트와 컨트롤러는 최종적으로 사용자 브라우저로 보낼 응답을 반환해야 합니다. 라라벨에서는 다양한 방식으로 응답을 반환할 수 있습니다. 가장 기본적인 방법은 라우트나 컨트롤러에서 문자열을 반환하는 것인데, 프레임워크가 이 문자열을 자동으로 완전한 HTTP 응답으로 만들어 줍니다.

Route::get('/', function () {
return 'Hello World';
});

라우트나 컨트롤러에서 문자열뿐만 아니라 배열도 반환할 수 있습니다. 배열을 반환하면 프레임워크가 이를 자동으로 JSON 응답으로 변환합니다.

Route::get('/', function () {
return [1, 2, 3];
});

[!NOTE] 라우트나 컨트롤러에서 Eloquent 컬렉션도 반환할 수 있다는 것을 알고 계셨나요? 자동으로 JSON으로 변환되어 응답됩니다. 한번 시도해 보세요!

응답 객체

실무에서는 단순한 문자열이나 배열만 반환하는 경우는 드뭅니다. 일반적으로는 Illuminate\Http\Response 인스턴스나 뷰(View) 를 반환하게 됩니다.

Response 인스턴스를 반환하면 HTTP 상태 코드나 헤더 값을 자유롭게 지정할 수 있습니다. Response 클래스는 Symfony\Component\HttpFoundation\Response를 상속하고 있으므로, HTTP 응답을 만들기 위한 다양한 메서드를 제공합니다.

Route::get('/home', function () {
return response('Hello World', 200)
->header('Content-Type', 'text/plain');
});

Eloquent 모델 및 컬렉션

Eloquent ORM의 모델이나 컬렉션도 라우트나 컨트롤러에서 직접 반환할 수 있습니다. 이 경우 라라벨이 해당 모델이나 컬렉션을 자동으로 JSON 응답으로 변환하며, 모델의 hidden 속성 역시 존중하여 처리합니다.

use App\Models\User;

Route::get('/user/{user}', function (User $user) {
return $user;
});

응답에 헤더 추가하기

대부분의 응답 메서드는 메서드 체이닝을 지원하므로, 여러 개의 헤더를 연속적으로 지정하여 응답 객체를 손쉽게 생성할 수 있습니다. 예를 들어, header 메서드를 여러 번 호출해 다양한 헤더를 응답에 추가할 수 있습니다.

return response($content)
->header('Content-Type', $type)
->header('X-Header-One', 'Header Value')
->header('X-Header-Two', 'Header Value');

또는, withHeaders 메서드를 사용해 헤더들을 배열 형태로 한 번에 추가할 수도 있습니다.

return response($content)
->withHeaders([
'Content-Type' => $type,
'X-Header-One' => 'Header Value',
'X-Header-Two' => 'Header Value',
]);

캐시 제어 미들웨어

라라벨에는 cache.headers 미들웨어가 포함되어 있어, 여러 라우트에 대해 Cache-Control 헤더를 손쉽게 설정할 수 있습니다. 각 디렉티브는 해당 cache-control 옵션의 snake case 형태로, 세미콜론(;)으로 구분하여 지정합니다. 만약 etag를 포함시키면, 응답 컨텐츠의 MD5 해시값이 자동으로 ETag로 설정됩니다.

Route::middleware('cache.headers:public;max_age=2628000;etag')->group(function () {
Route::get('/privacy', function () {
// ...
});

Route::get('/terms', function () {
// ...
});
});

응답에 쿠키 추가하기

Illuminate\Http\Response 인스턴스의 cookie 메서드를 통해 응답에 쿠키를 추가할 수 있습니다. 이 메서드에는 쿠키의 이름, 값, 유효기간(분 단위)을 전달하면 됩니다.

return response('Hello World')->cookie(
'name', 'value', $minutes
);

cookie 메서드는 종종 잘 사용되지 않는 몇 가지 추가 인수도 받을 수 있습니다. 이 인수들은 PHP의 기본 setcookie 함수에 전달하는 인수와 동일하게 동작합니다.

return response('Hello World')->cookie(
'name', 'value', $minutes, $path, $domain, $secure, $httpOnly
);

아직 응답 인스턴스가 없는 상황에서도, Cookie 파사드를 이용하여 쿠키를 "큐"에 등록하고, 실제 응답이 생성될 때 쿠키를 붙일 수도 있습니다. queue 메서드는 쿠키 인스턴스를 만들 때 필요한 인수들을 받으며, 등록된 쿠키들은 실제 응답이 브라우저로 전송될 때 자동으로 추가됩니다.

use Illuminate\Support\Facades\Cookie;

Cookie::queue('name', 'value', $minutes);

쿠키 인스턴스 생성하기

나중에 응답 인스턴스에 붙이기 위해, Symfony\Component\HttpFoundation\Cookie 인스턴스를 미리 생성해 둘 수도 있습니다. 이 때는 전역 cookie 헬퍼 함수를 사용하면 됩니다. 이렇게 생성한 쿠키는 반드시 응답 인스턴스에 추가해야만 브라우저로 전달됩니다.

$cookie = cookie('name', 'value', $minutes);

return response('Hello World')->cookie($cookie);

쿠키 미리 만료시키기

응답에서 withoutCookie 메서드를 사용하면 특정 쿠키를 미리 만료시켜 제거할 수 있습니다.

return response('Hello World')->withoutCookie('name');

아직 응답 인스턴스가 없는 경우라면, Cookie 파사드의 expire 메서드를 사용해도 쿠키를 만료시킬 수 있습니다.

Cookie::expire('name');

쿠키와 암호화 처리

기본적으로 라라벨에서 발급되는 모든 쿠키는 암호화되고 서명되어, 클라이언트가 내용을 읽거나 변조할 수 없습니다. 만약 특정 쿠키에 한해 암호화를 사용하지 않으려면, app/Http/Middleware 디렉터리에 있는 App\Http\Middleware\EncryptCookies 미들웨어의 $except 속성에 해당 쿠키의 이름을 추가하면 됩니다.

/**
* The names of the cookies that should not be encrypted.
*
* @var array
*/
protected $except = [
'cookie_name',
];

리다이렉트

리다이렉트 응답은 Illuminate\Http\RedirectResponse 클래스의 인스턴스로, 사용자를 다른 URL로 이동할 때 필요한 적절한 헤더를 포함합니다. 다양한 방식으로 RedirectResponse 인스턴스를 생성할 수 있지만, 가장 간단한 방법은 전역 redirect 헬퍼를 사용하는 것입니다.

Route::get('/dashboard', function () {
return redirect('home/dashboard');
});

종종 사용자가 이전 페이지로 돌아가게 하고 싶을 때가 있습니다. 예를 들어, 입력 폼이 올바르지 않을 때 이전 위치로 돌려보낼 수 있는데, 이럴 때는 전역 back 헬퍼 함수를 사용하세요. 이 기능은 세션을 활용하므로, 반드시 back 함수를 호출하는 라우트가 web 미들웨어 그룹을 사용하는지 확인해야 합니다.

Route::post('/user/profile', function () {
// 요청을 유효성 검증...

return back()->withInput();
});

이름이 지정된 라우트로 리다이렉트

redirect 헬퍼를 아무 인수 없이 호출하면 Illuminate\Routing\Redirector 인스턴스가 리턴되고, 이 인스턴스에서 다양한 메서드를 사용할 수 있습니다. 예를 들어, route 메서드를 이용해 이름이 지정된 라우트로 리다이렉트할 수 있습니다.

return redirect()->route('login');

라우트에 파라미터가 필요한 경우, 두 번째 인수로 파라미터 배열을 전달합니다.

// 예: 라우트 URI가 /profile/{id} 인 경우

return redirect()->route('profile', ['id' => 1]);

Eloquent 모델을 통한 파라미터 자동 채움

라우트 파라미터(ID 등)가 Eloquent 모델에서 채워지는 경우, 모델 인스턴스 자체를 전달할 수도 있습니다. 그러면 ID가 자동으로 추출되어 적용됩니다.

// 예: 라우트 URI가 /profile/{id} 인 경우

return redirect()->route('profile', [$user]);

특정 컬럼 값을 라우트 파라미터로 사용하고 싶다면, 라우트 파라미터 정의에서 해당 컬럼을 지정하거나(/profile/{id:slug}) Eloquent 모델에서 getRouteKey 메서드를 오버라이드할 수 있습니다.

/**
* 모델의 라우트 key 값을 반환합니다.
*
* @return mixed
*/
public function getRouteKey()
{
return $this->slug;
}

컨트롤러 액션으로 리다이렉트

컨트롤러 액션으로도 리다이렉트할 수 있습니다. 이때는 action 메서드에 컨트롤러 이름과 액션 메서드명을 전달하면 됩니다.

use App\Http\Controllers\UserController;

return redirect()->action([UserController::class, 'index']);

컨트롤러 라우트에 파라미터가 필요하다면 두 번째 인수로 파라미터 배열을 전달하세요.

return redirect()->action(
[UserController::class, 'profile'], ['id' => 1]
);

외부 도메인으로 리다이렉트

가끔 애플리케이션 외부의 도메인으로 리다이렉트할 필요가 있을 때, away 메서드를 사용하면 추가적인 URL 인코딩, 검증, 확인 없이 해당 주소로 바로 리다이렉트할 수 있습니다.

return redirect()->away('https://www.google.com');

세션 데이터와 함께 리다이렉트

새 URL로 리다이렉트하면서 세션에 데이터를 플래시하는 경우가 자주 있습니다. 주로 특정 작업이 성공적으로 처리된 뒤에, 성공 메시지를 세션에 저장해 전달할 때 사용됩니다. 이런 상황에서, 리다이렉트 응답을 반환할 때 with 메서드를 체이닝해 세션에 데이터를 플래시할 수 있습니다.

Route::post('/user/profile', function () {
// ...

return redirect('dashboard')->with('status', 'Profile updated!');
});

사용자가 리다이렉트된 이후에는 세션에 저장된 메시지를 화면에 표시할 수 있습니다. 예를 들어 Blade 문법에서는 아래와 같이 할 수 있습니다.

@if (session('status'))
<div class="alert alert-success">
{{ session('status') }}
</div>
@endif

입력값과 함께 리다이렉트

RedirectResponse 인스턴스에서 제공하는 withInput 메서드를 활용하면, 현재 요청의 입력값을 세션에 플래시하고 사용자를 새로운 위치로 리다이렉트할 수 있습니다. 주로 유효성 검사에 실패한 경우에 활용하며, 플래시된 입력값은 다음 요청에서 손쉽게 재사용하여 폼을 다시 채울 수 있습니다.

return back()->withInput();

기타 응답 타입

response 헬퍼는 다양한 유형의 응답 인스턴스를 쉽게 생성할 때 사용할 수 있습니다. 인수 없이 response 헬퍼를 호출하면 Illuminate\Contracts\Routing\ResponseFactory 컨트랙트의 구현 인스턴스가 반환됩니다. 이 컨트랙트에는 여러 유용한 응답 생성 메서드가 포함되어 있습니다.

뷰(View) 응답

응답의 상태 코드와 헤더 값을 직접 지정할 필요가 있으면서도, 콘텐츠로 응답을 반환하고 싶다면 view 메서드를 사용하세요.

return response()
->view('hello', $data, 200)
->header('Content-Type', $type);

별도의 HTTP 상태 코드나 헤더를 지정할 필요가 없다면 전역 view 헬퍼 함수를 그대로 사용할 수도 있습니다.

JSON 응답

json 메서드는 Content-Type 헤더를 자동으로 application/json으로 지정하고, 전달된 배열을 PHP의 json_encode 함수를 이용해 JSON 문자열로 변환합니다.

return response()->json([
'name' => 'Abigail',
'state' => 'CA',
]);

JSONP 응답을 만들려면 json 메서드와 함께 withCallback 메서드도 사용할 수 있습니다.

return response()
->json(['name' => 'Abigail', 'state' => 'CA'])
->withCallback($request->input('callback'));

파일 다운로드

download 메서드는 특정 경로의 파일을 사용자의 브라우저가 다운로드하도록 유도하는 응답을 생성합니다. 두 번째 인수로는 다운로드 시 사용자에게 보여질 파일명을 지정할 수 있으며, 세 번째 인수로는 추가적인 HTTP 헤더의 배열을 전달할 수 있습니다.

return response()->download($pathToFile);

return response()->download($pathToFile, $name, $headers);

[!WARNING] 파일 다운로드를 담당하는 Symfony HttpFoundation은 다운로드 파일의 이름이 반드시 ASCII 문자여야 합니다.

스트리밍 다운로드

작업 결과 문자열을 파일로 먼저 저장하지 않고 바로 다운로드 형태로 제공하고 싶을 때는 streamDownload 메서드를 사용할 수 있습니다. 이 메서드는 콜백, 파일명, 그리고 선택적으로 헤더 배열을 인수로 받습니다.

use App\Services\GitHub;

return response()->streamDownload(function () {
echo GitHub::api('repo')
->contents()
->readme('laravel', 'laravel')['contents'];
}, 'laravel-readme.md');

파일 응답

file 메서드는 이미지나 PDF 같은 파일을 다운로드가 아닌, 브라우저 내에서 바로 보여주는 용도로 사용할 수 있습니다. 첫 번째 인수로 파일 경로를, 두 번째 인수로 헤더 배열을 전달합니다.

return response()->file($pathToFile);

return response()->file($pathToFile, $headers);

응답 매크로

여러 라우트와 컨트롤러에서 반복적으로 사용할 맞춤 응답을 정의하고 싶다면, Response 파사드의 macro 메서드를 사용해 매크로를 등록할 수 있습니다. 보통 이 작업은 애플리케이션의 서비스 프로바이더, 예를 들면 App\Providers\AppServiceProviderboot 메서드에서 수행합니다.

<?php

namespace App\Providers;

use Illuminate\Support\Facades\Response;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
{
/**
* Bootstrap any application services.
*
* @return void
*/
public function boot()
{
Response::macro('caps', function ($value) {
return Response::make(strtoupper($value));
});
}
}

macro 함수는 첫 번째 인수로 매크로명을, 두 번째 인수로 클로저(익명 함수)를 받습니다. 이렇게 등록한 매크로는 ResponseFactory 구현체나 response 헬퍼에서 매크로 이름을 호출하는 것으로 실행할 수 있습니다.

return response()->caps('foo');