요청 생명주기 (Request Lifecycle)
소개
실제 환경에서 어떤 도구를 사용할 때, 그 도구가 어떻게 동작하는지 이해하면 더 자신감을 느낍니다. 애플리케이션 개발도 마찬가지입니다. 개발 도구가 어떻게 작동하는지 알면 더 편안하고 자신 있게 사용할 수 있습니다.
이 문서의 목표는 Laravel 프레임워크가 어떻게 동작하는지 높은 수준에서 이해할 수 있도록 돕는 것입니다. 프레임워크를 전체적으로 잘 알게 되면 모든 것이 덜 '마법같이' 느껴지고 애플리케이션 구축에 더 자신감이 생깁니다. 모든 용어를 처음부터 전부 이해하지 못하더라도 낙심하지 마세요! 기본적인 흐름만 파악하고, 문서의 다른 부분을 탐색하면서 지식을 천천히 쌓아가면 됩니다.
생명주기 개요
초기 단계
Laravel 애플리케이션에 들어오는 모든 요청의 진입점은 public/index.php 파일입니다. 모든 요청은 웹 서버(Apache / Nginx) 설정에 의해 이 파일로 전달됩니다. index.php 파일 자체는 많은 코드를 담고 있지 않고, 나머지 프레임워크를 불러오는 시작점 역할을 합니다.
index.php는 Composer가 생성한 오토로더를 불러온 후, bootstrap/app.php에서 Laravel 애플리케이션 인스턴스를 가져옵니다. Laravel이 처음 하는 작업은 이 애플리케이션 인스턴스와 서비스 컨테이너 인스턴스를 생성하는 것입니다.
HTTP / 콘솔 커널
그 다음, 들어온 요청 유형에 따라 애플리케이션 인스턴스의 handleRequest 또는 handleCommand 메서드를 사용하여 요청이 HTTP 커널이나 콘솔 커널로 전달됩니다. 이 두 커널은 모든 요청이 흐르는 중앙 지점 역할을 합니다. 지금은 Illuminate\Foundation\Http\Kernel 인스턴스인 HTTP 커널에 집중하겠습니다.
HTTP 커널은 요청 처리 전에 실행할 bootstrappers 배열을 정의합니다. 이 부트스트래퍼들은 오류 처리, 로깅 설정, 애플리케이션 환경 감지 등 요청을 실제로 처리하기 전에 완료되어야 할 작업들을 수행합니다. 보통 이 클래스들은 내부 Laravel 설정을 담당해서 개발자가 신경 쓸 필요가 없습니다.
또한 HTTP 커널은 요청을 애플리케이션의 미들웨어 스택을 통해 전달합니다. 이 미들웨어들은 HTTP 세션 읽기/쓰기, 애플리케이션이 유지보수 모드인지 여부 판단, CSRF 토큰 검증 등을 처리합니다. 이 부분에 대해서는 이후 더 자세히 다루겠습니다.
HTTP 커널의 handle 메서드 시그니처는 매우 간단합니다: Request를 받아 Response를 반환합니다. 커널을 애플리케이션 전체를 대표하는 거대한 블랙박스라고 생각하세요. HTTP 요청을 넣으면 HTTP 응답을 돌려줍니다.
서비스 프로바이더
커널 부트스트래핑 작업 중 가장 중요한 것 중 하나는 애플리케이션의 서비스 프로바이더를 불러오는 것입니다. 서비스 프로바이더는 데이터베이스, 큐, 유효성 검증, 라우팅 등 프레임워크의 다양한 구성요소를 부트스트랩하는 역할을 합니다.
Laravel은 이 프로바이더 목록을 반복하며 각각을 인스턴스화합니다. 인스턴스화가 끝나면 모든 프로바이더의 register 메서드를 호출합니다. 그 후 모든 프로바이더가 등록된 상태에서 각각의 boot 메서드를 호출합니다. 이는 각 서비스 프로바이더의 boot 메서드가 실행될 때 모든 컨테이너 바인딩이 등록되고 준비된 상태임을 보장하기 위함입니다.
사실상 Laravel이 제공하는 거의 모든 주요 기능은 서비스 프로바이더를 통해 부트스트랩되고 구성됩니다. 프레임워크가 제공하는 다양한 기능을 부트스트랩하고 설정하기 때문에, 서비스 프로바이더는 Laravel 부트스트랩 과정에서 가장 중요한 부분입니다.
내부적으로 수십 개의 서비스 프로바이더를 사용하지만, 여러분도 직접 자신의 서비스 프로바이더를 만들 수 있습니다. 애플리케이션에서 사용하는 사용자 정의 또는 서드파티 서비스 프로바이더 목록은 bootstrap/providers.php 파일에서 확인할 수 있습니다.
라우팅
애플리케이션이 부트스트랩되고 모든 서비스 프로바이더가 등록되면, Request는 디스패치 작업을 위해 라우터로 전달됩니다. 라우터는 요청을 적절한 라우트나 컨트롤러로 전달하고, 라우트에 지정된 미들웨어를 실행합니다.
미들웨어는 애플리케이션에 들어오는 HTTP 요청을 필터링하거나 검사하는 편리한 메커니즘을 제공합니다. 예를 들어, Laravel은 애플리케이션 사용자가 인증되었는지 확인하는 미들웨어를 포함하고 있습니다. 사용자가 인증되지 않은 경우, 미들웨어는 사용자를 로그인 화면으로 리다이렉트합니다. 반면 인증되어 있다면 요청이 애플리케이션 내부로 계속 진행됩니다. 일부 미들웨어는 애플리케이션 내 모든 라우트에 할당됩니다(PreventRequestsDuringMaintenance 같은 경우), 반면 일부는 특정 라우트나 라우트 그룹에만 지정됩니다. 미들웨어에 관한 자세한 내용은 미들웨어 문서를 참고하세요.
요청이 매칭된 라우트에 할당된 모든 미들웨어를 통과하면, 라우트나 컨트롤러 메서드가 실행되고, 반환된 응답은 다시 라우트의 미들웨어 체인을 거쳐 전달됩니다.
마무리
라우트 또는 컨트롤러 메서드가 응답을 반환하면, 이 응답은 다시 라우트 미들웨어를 거쳐 애플리케이션이 나가는 응답을 수정하거나 확인할 기회를 갖게 합니다.
마지막으로 응답이 미들웨어를 모두 통과하면, HTTP 커널의 handle 메서드는 응답 객체를 애플리케이션 인스턴스의 handleRequest에 반환하고, 이 메서드는 응답의 send 메서드를 호출합니다. send 메서드는 응답 내용을 사용자 웹 브라우저로 전송합니다. 이로써 Laravel 요청 생명주기의 전체 과정을 완료하게 됩니다!
서비스 프로바이더에 집중하기
서비스 프로바이더는 Laravel 애플리케이션 부트스트랩의 핵심입니다. 애플리케이션 인스턴스를 생성하고, 서비스 프로바이더를 등록한 후 요청을 부트스트랩된 애플리케이션에 전달하는 과정이 전부입니다. 정말 이만큼 간단합니다!
서비스 프로바이더가 Laravel 애플리케이션을 어떻게 구축하고 부트스트랩하는지 확실히 이해하는 것은 매우 중요합니다. 애플리케이션에서 작성한 사용자 정의 서비스 프로바이더는 app/Providers 디렉터리에 저장됩니다.
기본적으로 AppServiceProvider는 비교적 비어 있는 상태입니다. 이 프로바이더는 애플리케이션만의 부트스트랩과 서비스 컨테이너 바인딩을 추가하기에 좋은 장소입니다. 큰 규모의 애플리케이션이라면, 여러 개의 서비스 프로바이더를 만들어서 각각 애플리케이션의 특정 서비스를 위한 보다 세분화된 부트스트래핑을 수행하는 것도 좋습니다.