# PHP-FPM 초보자 가이드 > **작성일**: 2026-02-22 > **대상**: SAM 프로젝트에 새로 합류한 개발자 > **서버 인프라 학습 시리즈** | Part 3 of 3 > [1. 서버 동작 원리](server-how-it-works.md) → [2. Nginx & FastCGI](nginx-fastcgi-guide.md) → **3. PHP-FPM** --- ## 1. 개요 ### 1.1 PHP-FPM이란 **PHP-FPM**(FastCGI Process Manager)은 PHP 코드를 실행하는 **프로세스 관리자**다. Nginx는 PHP를 직접 실행하지 못하므로, PHP-FPM이 대신 실행하고 결과를 돌려준다. ### 1.2 이 문서의 목적 [서버 동작 원리 가이드](server-how-it-works.md)에서 PHP-FPM을 간략히 소개했다. 이 문서는 **왜 필요한지**, **어떻게 동작하는지**, **SAM에서 어떤 설정으로 쓰이는지**를 다룬다. --- ## 2. PHP가 웹에서 동작하는 방식의 역사 ### 2.1 CGI → mod_php → PHP-FPM ``` [1세대 CGI] 요청마다 PHP 프로세스 생성/종료 → 느림 [2세대 mod_php] Apache에 PHP 내장 → 빠르지만 Nginx 불가, 메모리 낭비 [3세대 PHP-FPM] Nginx와 분리, 워커 풀 재사용 → 빠르고 유연 ← SAM이 사용 ``` **비유로 이해하기**: | 세대 | 비유 | |------|------| | CGI | 손님마다 직원을 채용하고 해고하는 식당 | | mod_php | 직원이 주방장(Apache)과 한 몸 — 따로 관리 불가 | | **PHP-FPM** | 안내 데스크(Nginx)와 업무 창구(PHP-FPM) 분리된 은행 | ### 2.2 요약 | 세대 | 방식 | 장점 | 단점 | |------|------|------|------| | CGI | 매번 프로세스 생성 | 단순 | 느림, 리소스 낭비 | | mod_php | Apache에 내장 | CGI보다 빠름 | Nginx 불가, 메모리 낭비 | | **PHP-FPM** | 독립 프로세스 관리 | 빠름, 유연 | 설정 필요 | --- ## 3. PHP-FPM의 구조 ### 3.1 Master / Worker 모델 PHP-FPM은 **Master 프로세스** 1개와 **Worker 프로세스** 여러 개로 구성된다. ``` PHP-FPM ┌──────────────────────────────────┐ │ Master 프로세스 (관리자) │ │ ├── Worker 1 (대기 중) │ │ ├── Worker 2 (요청 처리 중) │ │ ├── Worker 3 (대기 중) │ │ ├── Worker 4 (요청 처리 중) │ │ └── Worker 5 (대기 중) │ └──────────────────────────────────┘ ``` **은행 창구 비유**: | PHP-FPM | 은행 | |---------|------| | Master 프로세스 | 지점장 (직원 수 조절, 감독) | | Worker 프로세스 | 창구 직원 (실제 업무 처리) | | `pm.max_children` | 최대 창구 수 | | 요청 큐 | 대기 번호표 줄 | - **Master**: 워커 생성/종료, 비정상 워커 재시작, 포트 9000 대기 - **Worker**: PHP 코드 실행, 1 Worker = 1 요청, 완료 후 다음 요청 대기 ### 3.2 포트 9000과 프로세스 관리 모드 PHP-FPM은 **TCP 포트 9000**에서 요청을 기다린다. ``` Nginx ──── TCP 9000 ────→ PHP-FPM Master ──→ 빈 Worker에 배정 ``` Master가 Worker 수를 관리하는 3가지 모드: | 모드 | 설명 | SAM | |------|------|-----| | `static` | 항상 고정 수 유지 | - | | **`dynamic`** | 트래픽에 따라 조절 | **사용 중** | | `ondemand` | 요청 올 때만 생성 | - | --- ## 4. Nginx와 PHP-FPM의 관계 ### 4.1 왜 Nginx는 PHP를 직접 못 실행하는가 Nginx는 정적 파일 서빙과 리버스 프록시 전용이다. PHP 해석 엔진이 없으므로 `.php` 파일을 실행할 수 없다. ### 4.2 FastCGI 프로토콜 Nginx와 PHP-FPM은 **FastCGI** 프로토콜로 통신한다. ``` 브라우저 ──HTTP──→ Nginx ──FastCGI──→ PHP-FPM ──→ PHP 실행 │ 브라우저 ←──HTTP── Nginx ←──FastCGI── PHP-FPM ←────┘ ``` Nginx가 FastCGI로 전달하는 주요 정보: - `SCRIPT_FILENAME` — 실행할 PHP 파일 경로 (`/var/www/mng/public/index.php`) - `REQUEST_METHOD` — GET, POST 등 - `QUERY_STRING` — URL 파라미터 ### 4.3 역할 분담 ``` ┌─────────────────────┐ ┌─────────────────────┐ │ Nginx │ │ PHP-FPM │ │ │ TCP │ │ │ • SSL 종료 │ 9000 │ • PHP 코드 실행 │ │ • 도메인 라우팅 │─────→│ • Laravel 구동 │ │ • 정적 파일 서빙 │ │ • DB 조회 │ │ • 보안 필터링 │ │ • HTML/JSON 생성 │ └─────────────────────┘ └─────────────────────┘ ``` --- ## 5. SAM에서의 PHP-FPM ### 5.1 컨테이너 구조 ``` Docker ├── sam-nginx-1 ── 외부 리버스 프록시 ├── sam-api-1 │ └── Supervisor │ ├── php-fpm ← 포트 9000 │ ├── nginx ← 컨테이너 내부 웹서버 │ ├── queue-worker ← 백그라운드 작업 │ └── scheduler ← 60초 예약 작업 ├── sam-mng-1 │ └── Supervisor │ ├── php-fpm ← 포트 9000 │ ├── nginx ← 컨테이너 내부 웹서버 │ └── queue-worker x2 └── sam-mysql-1 ``` Supervisor가 PHP-FPM과 Nginx를 함께 관리한다. 컨테이너 시작 시 Supervisor가 모든 프로세스를 기동한다. ### 5.2 PHP-FPM 설정 (`www.conf`) SAM 설정 파일 위치: `docker/mng/www.conf` ```ini [www] user = www-data group = www-data listen = 0.0.0.0:9000 pm = dynamic pm.max_children = 20 pm.start_servers = 5 pm.min_spare_servers = 5 pm.max_spare_servers = 10 pm.max_requests = 500 ``` **각 설정의 의미**: | 설정 | 값 | 의미 | |------|-----|------| | `pm` | `dynamic` | 트래픽에 따라 워커 수 조절 | | `pm.max_children` | `20` | 최대 동시 처리 수 (= 최대 창구 20개) | | `pm.start_servers` | `5` | 시작 시 워커 수 | | `pm.min_spare_servers` | `5` | 최소 대기 워커 수 | | `pm.max_spare_servers` | `10` | 최대 대기 워커 수 | | `pm.max_requests` | `500` | 500건 처리 후 워커 재시작 (메모리 누수 방지) | ### 5.3 워커 수의 동적 변화 ``` 워커 수 20 ┤ ■■■■ (피크) 10 ┤ ■■■■■ ■■■■■ 5 ┤ ■■■■■■■■■■■■■ ■■■■■■■■ 0 ┤────────────────────────────────────────────── 새벽 오전 점심 오후 ``` ### 5.4 Docker 이미지 SAM은 `php:8.4-fpm` 공식 이미지를 기반으로 한다. 이미지 이름의 `fpm`이 PHP-FPM 내장을 의미한다. ```dockerfile FROM php:8.4-fpm # Supervisor로 php-fpm + nginx 동시 기동 CMD ["/usr/bin/supervisord"] ``` --- ## 6. 자주 묻는 질문 ### 6.1 "502 Bad Gateway 에러가 뭔가요?" Nginx가 PHP-FPM에 연결 실패 시 발생한다. PHP-FPM이 죽었거나, 컨테이너가 비정상 시작된 경우다. ```bash # 컨테이너 상태 확인 docker ps # 로그 확인 docker logs sam-mng-1 ``` ### 6.2 "워커가 부족하면?" 20개 워커가 모두 처리 중이면 새 요청은 대기열에 들어간다. 너무 오래 대기하면 504 Gateway Timeout이 발생한다. `pm.max_children`을 올리거나 느린 코드를 최적화한다. ### 6.3 "PHP-FPM을 재시작하려면?" ```bash # 컨테이너 전체 재시작 docker restart sam-mng-1 # PHP-FPM만 재시작 docker exec sam-mng-1 supervisorctl restart php-fpm ``` ### 6.4 "프로세스 상태를 확인하려면?" ```bash docker exec sam-mng-1 supervisorctl status # nginx RUNNING pid 8, uptime 2:30:00 # php-fpm RUNNING pid 9, uptime 2:30:00 # queue-worker_00 RUNNING pid 10, uptime 2:30:00 ``` ### 6.5 "php artisan serve와 뭐가 다른가요?" `php artisan serve`는 개발용 간이 서버로 동시 요청 1개만 처리한다. PHP-FPM은 워커 수만큼 동시 처리가 가능한 프로덕션용이다. SAM은 PHP-FPM을 사용한다. --- ## 관련 문서 **학습 시리즈**: | 순서 | 문서 | 설명 | |------|------|------| | Part 1 | [server-how-it-works.md](server-how-it-works.md) | 서버 동작 원리 전체 흐름 | | Part 2 | [nginx-fastcgi-guide.md](nginx-fastcgi-guide.md) | Nginx와 FastCGI 프로토콜 심화 (이전) | **참고 문서**: | 문서 | 설명 | |------|------| | [docker-setup.md](../specs/docker-setup.md) | Docker 환경 설정값 상세 | --- **최종 업데이트**: 2026-02-22