From 5bc89d27a2f1bb54df9a3db41db4d56f24adde4e Mon Sep 17 00:00:00 2001 From: Konstantin Date: Sat, 18 Jan 2025 10:53:43 +0300 Subject: [PATCH 1/2] hw4 --- .gitignore | 2 +- README.md | 86 +++++++++++++++++- code/composer.json | 11 +++ code/composer.lock | 18 ++++ code/entrypoint.sh | 5 ++ code/public/index.php | 8 ++ code/src/Controller/IndexController.php | 33 +++++++ code/src/Core/App.php | 27 ++++++ code/src/Service/RequestHandler.php | 28 ++++++ code/src/Service/SessionService.php | 16 ++++ code/src/Service/StringService.php | 42 +++++++++ docker-compose.yml | 113 ++++++++++++++++++++++++ docker/nginx-balancer/Dockerfile | 9 ++ docker/nginx-balancer/balancer.conf | 26 ++++++ docker/nginx/backend1.conf | 25 ++++++ docker/nginx/backend2.conf | 25 ++++++ docker/nginx/backend3.conf | 25 ++++++ docker/php/Dockerfile | 20 +++++ docker/php/php-fpm.conf | 10 +++ docker/php/php.ini | 14 +++ 20 files changed, 540 insertions(+), 3 deletions(-) create mode 100644 code/composer.json create mode 100644 code/composer.lock create mode 100644 code/entrypoint.sh create mode 100644 code/public/index.php create mode 100644 code/src/Controller/IndexController.php create mode 100644 code/src/Core/App.php create mode 100644 code/src/Service/RequestHandler.php create mode 100644 code/src/Service/SessionService.php create mode 100644 code/src/Service/StringService.php create mode 100644 docker-compose.yml create mode 100644 docker/nginx-balancer/Dockerfile create mode 100644 docker/nginx-balancer/balancer.conf create mode 100644 docker/nginx/backend1.conf create mode 100644 docker/nginx/backend2.conf create mode 100644 docker/nginx/backend3.conf create mode 100644 docker/php/Dockerfile create mode 100644 docker/php/php-fpm.conf create mode 100644 docker/php/php.ini diff --git a/.gitignore b/.gitignore index b28e598f1..d81858678 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,5 @@ .idea/ -/vendor/ +/code/vendor/ .DS_Store .AppleDouble .LSOverride diff --git a/README.md b/README.md index b44ec14b8..79411739b 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,85 @@ -# PHP_2024 +# Инструкция по установке и запуску приложения -https://otus.ru/lessons/razrabotchik-php/?utm_source=github&utm_medium=free&utm_campaign=otus +## Установка и запуск + +1. Убедитесь, что у вас установлен Docker и Docker Compose. +2. Клонируйте репозиторий на ваш компьютер: + + ```bash + git clone -b KRudenko/hw4 git@github.com:otusteamedu/PHP_2024.git + ``` + +3. Перейдите в каталог с репозиторием: + + ```bash + cd PHP_2024 + ``` + +4. Соберите и запустите контейнеры Docker с помощью Docker Compose: + + ```bash + docker-compose up -d + ``` + +# Проверка работы кластера + +Каждый раз делая запрос `localhost`, будет выводиться имя хоста: + + ```shell + Hello from hostname php-fpm2! + ``` + +# Проверка работы скобок + +## Успешный запрос + + ```shell + curl -X POST -w "\nHTTP Code: %{http_code}" localhost/string -d "string=(()()()())((((()()()))(()()()(((()))))))" + ``` + +## Неуспешный запрос + + ```shell + curl -X POST -w "\nHTTP Code: %{http_code}" localhost/string -d "string=(()()()()))((((()()()))(()()()(((()))))))" + ``` + +## Пустой запрос + + ```shell + curl -X POST -w "\nHTTP Code: %{http_code}" localhost/string -d "string" + ``` + +## Неверный метод + + ```shell + curl -X GET -w "\nHTTP Code: %{http_code}" localhost/string -d "string" + ``` + +# Проверка работы сессий + +## Из браузера или postman + +Запросите адрес `localhost/session` несколько раз подряд и вы увидите, +что в каждом запросе сессия остается неизменной, а счетчик увеличивается. + +## Из консоли + +Выполните запрос, что бы получить id сессии: + + ```shell + curl localhost/session + ``` + +Затем подставьте полученный id сессии в следующий запрос вместо `{SESSION_ID}`. +Каждый раз выполняя запрос, вы увидите как увеличивается счетчик: + + ```shell + curl localhost/session -H 'Cookie: PHPSESSID={SESSION_ID}' + ``` + +## Контейнер докера + + ```shell + docker exec -it otus-redis redis-cli + keys * + ``` diff --git a/code/composer.json b/code/composer.json new file mode 100644 index 000000000..1cd7e2d32 --- /dev/null +++ b/code/composer.json @@ -0,0 +1,11 @@ +{ + "name": "krudenko/hw4", + "description": "Otus Home Work 4", + "type": "project", + "autoload": { + "psr-4": { + "KRudenko\\Otus\\": "src/" + } + }, + "require": {} +} diff --git a/code/composer.lock b/code/composer.lock new file mode 100644 index 000000000..aaebe6169 --- /dev/null +++ b/code/composer.lock @@ -0,0 +1,18 @@ +{ + "_readme": [ + "This file locks the dependencies of your project to a known state", + "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", + "This file is @generated automatically" + ], + "content-hash": "4d78bf53c015ffc52514d013b1821772", + "packages": [], + "packages-dev": [], + "aliases": [], + "minimum-stability": "stable", + "stability-flags": [], + "prefer-stable": false, + "prefer-lowest": false, + "platform": [], + "platform-dev": [], + "plugin-api-version": "2.1.0" +} diff --git a/code/entrypoint.sh b/code/entrypoint.sh new file mode 100644 index 000000000..266e51950 --- /dev/null +++ b/code/entrypoint.sh @@ -0,0 +1,5 @@ +#!/bin/bash + +composer install + +php-fpm diff --git a/code/public/index.php b/code/public/index.php new file mode 100644 index 000000000..7c2439250 --- /dev/null +++ b/code/public/index.php @@ -0,0 +1,8 @@ +run(); diff --git a/code/src/Controller/IndexController.php b/code/src/Controller/IndexController.php new file mode 100644 index 000000000..ff84b06aa --- /dev/null +++ b/code/src/Controller/IndexController.php @@ -0,0 +1,33 @@ +stringService = new StringService; + $this->sessionService = new SessionService; + } + + public function handleStringRequest(): string + { + if ($_SERVER['REQUEST_METHOD'] !== 'POST') { + http_response_code(405); + return "Данные не были отправлены методом POST."; + } + + return $this->stringService->processString(); + } + + public function handleSessionRequest(): string + { + return $this->sessionService->processSession(); + } +} diff --git a/code/src/Core/App.php b/code/src/Core/App.php new file mode 100644 index 000000000..8f0b1c04b --- /dev/null +++ b/code/src/Core/App.php @@ -0,0 +1,27 @@ +handler = new RequestHandler(); + } + + public function run(): string + { + $this->startSession(); + return $this->handler->handle(); + } + + public function startSession(): void + { + session_start(); + } +} diff --git a/code/src/Service/RequestHandler.php b/code/src/Service/RequestHandler.php new file mode 100644 index 000000000..52e958c08 --- /dev/null +++ b/code/src/Service/RequestHandler.php @@ -0,0 +1,28 @@ +controller = new IndexController(); + } + + public function handle(): string + { + $requestUri = $_SERVER['REQUEST_URI']; + return match ($requestUri) { + '/string' => $this->controller->handleStringRequest(), + '/session' => $this->controller->handleSessionRequest(), + '/' => sprintf('Hello from hostname %s!', gethostname()), + default => 'Доступны только адреса /string и /session', + }; + } +} diff --git a/code/src/Service/SessionService.php b/code/src/Service/SessionService.php new file mode 100644 index 000000000..fe5840e0c --- /dev/null +++ b/code/src/Service/SessionService.php @@ -0,0 +1,16 @@ +calculateBrackets($string); + + if ($ans === 0) { + http_response_code(200); + return "Строка корректна"; + } else { + http_response_code(400); + return "Строка не корректна"; + } + } + + public function calculateBrackets(string $string): int + { + $ans = 0; + foreach (str_split($string) as $char) { + if ($char === '(') { + $ans++; + } else { + $ans--; + } + if ($ans === -1) { + break; + } + } + return $ans; + } +} diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 000000000..68eedb835 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,113 @@ +version: '3.8' + +services: + # Балансировщик Nginx + nginx-balancer: + image: nginx:latest + container_name: otus-balancer + ports: + - "80:80" + volumes: + - ./docker/nginx-balancer/balancer.conf:/etc/nginx/nginx.conf + depends_on: + - backend1 + - backend2 + - backend3 + networks: + - app-network + + # Nginx backend1 + backend1: + image: nginx:latest + container_name: otus-backend1 + volumes: + - ./docker/nginx/backend1.conf:/etc/nginx/nginx.conf + - ./docker/php/php-fpm.conf:/usr/local/etc/php-fpm.conf + - ./code:/var/www/html + depends_on: + - php-fpm1 + - redis + networks: + - app-network + + # Nginx backend2 + backend2: + image: nginx:latest + container_name: otus-backend2 + volumes: + - ./docker/nginx/backend2.conf:/etc/nginx/nginx.conf + - ./docker/php/php-fpm.conf:/usr/local/etc/php-fpm.conf + - ./code:/var/www/html + depends_on: + - php-fpm2 + - redis + networks: + - app-network + + # Nginx backend3 + backend3: + image: nginx:latest + container_name: otus-backend3 + volumes: + - ./docker/nginx/backend3.conf:/etc/nginx/nginx.conf + - ./docker/php/php-fpm.conf:/usr/local/etc/php-fpm.conf + - ./code:/var/www/html + depends_on: + - php-fpm3 + - redis + networks: + - app-network + + # PHP-FPM1 + php-fpm1: + build: + context: . + dockerfile: docker/php/Dockerfile + container_name: otus-php-fpm1 + hostname: php-fpm1 + volumes: + - ./code:/var/www/html + depends_on: + - redis + networks: + - app-network + + # PHP-FPM1 + php-fpm2: + build: + context: . + dockerfile: docker/php/Dockerfile + container_name: otus-php-fpm2 + hostname: php-fpm2 + volumes: + - ./code:/var/www/html + depends_on: + - redis + networks: + - app-network + + # PHP-FPM1 + php-fpm3: + build: + context: . + dockerfile: docker/php/Dockerfile + container_name: otus-php-fpm3 + hostname: php-fpm3 + volumes: + - ./code:/var/www/html + depends_on: + - redis + networks: + - app-network + + # Redis + redis: + image: redis:latest + container_name: otus-redis + networks: + - app-network + +# Сеть для соединения контейнеров +networks: + app-network: + driver: bridge diff --git a/docker/nginx-balancer/Dockerfile b/docker/nginx-balancer/Dockerfile new file mode 100644 index 000000000..babd7c2a5 --- /dev/null +++ b/docker/nginx-balancer/Dockerfile @@ -0,0 +1,9 @@ +FROM nginx:latest + +# Копируем конфигурацию Nginx +COPY balanser.conf /etc/nginx/nginx.conf + +# Копируем код приложения +COPY ./code /var/www/html + +EXPOSE 80 diff --git a/docker/nginx-balancer/balancer.conf b/docker/nginx-balancer/balancer.conf new file mode 100644 index 000000000..215038dc1 --- /dev/null +++ b/docker/nginx-balancer/balancer.conf @@ -0,0 +1,26 @@ +worker_processes 1; + +events { + worker_connections 1024; +} + +http { + upstream backend { + least_conn; + server backend1:80; + server backend2:80; + server backend3:80; + } + + server { + listen 80; + + location / { + proxy_pass http://backend; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + } + } +} diff --git a/docker/nginx/backend1.conf b/docker/nginx/backend1.conf new file mode 100644 index 000000000..5e363282b --- /dev/null +++ b/docker/nginx/backend1.conf @@ -0,0 +1,25 @@ +worker_processes 1; + +events { + worker_connections 1024; +} + +http { + server { + listen 80; + + root /var/www/html/public; + index index.php index.html; + + location / { + try_files $uri $uri/ /index.php?$query_string; + } + + location ~ \.php$ { + include fastcgi_params; + fastcgi_pass php-fpm1:9000; + fastcgi_index index.php; + fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; + } + } +} \ No newline at end of file diff --git a/docker/nginx/backend2.conf b/docker/nginx/backend2.conf new file mode 100644 index 000000000..ad648666d --- /dev/null +++ b/docker/nginx/backend2.conf @@ -0,0 +1,25 @@ +worker_processes 1; + +events { + worker_connections 1024; +} + +http { + server { + listen 80; + + root /var/www/html/public; + index index.php index.html; + + location / { + try_files $uri $uri/ /index.php?$query_string; + } + + location ~ \.php$ { + include fastcgi_params; + fastcgi_pass php-fpm2:9000; + fastcgi_index index.php; + fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; + } + } +} \ No newline at end of file diff --git a/docker/nginx/backend3.conf b/docker/nginx/backend3.conf new file mode 100644 index 000000000..1eaf08f7c --- /dev/null +++ b/docker/nginx/backend3.conf @@ -0,0 +1,25 @@ +worker_processes 1; + +events { + worker_connections 1024; +} + +http { + server { + listen 80; + + root /var/www/html/public; + index index.php index.html; + + location / { + try_files $uri $uri/ /index.php?$query_string; + } + + location ~ \.php$ { + include fastcgi_params; + fastcgi_pass php-fpm3:9000; + fastcgi_index index.php; + fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; + } + } +} \ No newline at end of file diff --git a/docker/php/Dockerfile b/docker/php/Dockerfile new file mode 100644 index 000000000..6e58504d0 --- /dev/null +++ b/docker/php/Dockerfile @@ -0,0 +1,20 @@ +FROM php:8.2-fpm + +RUN apt-get update && apt-get install -y \ + libzip-dev \ + zip \ + unzip \ + libssl-dev \ + && docker-php-ext-install zip \ + && pecl install redis \ + && docker-php-ext-enable redis + +RUN curl -sS https://getcomposer.org/installer | php && mv composer.phar /usr/local/bin/composer + +COPY /docker/php/php.ini /usr/local/etc/php/conf.d/custom.ini + +COPY ./code /var/www/html + +EXPOSE 9000 + +ENTRYPOINT ["sh", "./entrypoint.sh"] diff --git a/docker/php/php-fpm.conf b/docker/php/php-fpm.conf new file mode 100644 index 000000000..061471caf --- /dev/null +++ b/docker/php/php-fpm.conf @@ -0,0 +1,10 @@ +[global] +daemonize = no + +[www] +listen = 9000 +pm = dynamic +pm.max_children = 5 +pm.start_servers = 2 +pm.min_spare_servers = 1 +pm.max_spare_servers = 3 diff --git a/docker/php/php.ini b/docker/php/php.ini new file mode 100644 index 000000000..f0bb3b3bd --- /dev/null +++ b/docker/php/php.ini @@ -0,0 +1,14 @@ +[global] +daemonize = no + +[www] +listen = /var/run/php/php8.2-fpm.sock +pm = dynamic +pm.max_children = 5 +pm.start_servers = 2 +pm.min_spare_servers = 1 +pm.max_spare_servers = 3 + +; Настройка сессий в Redis +session.save_handler = redis +session.save_path = "tcp://redis:6379" From 9678145d95cfde2c356881f208119997f0a640ef Mon Sep 17 00:00:00 2001 From: Konstantin Date: Sat, 18 Jan 2025 17:30:30 +0300 Subject: [PATCH 2/2] action-php-codesniffer --- code/src/Controller/IndexController.php | 4 ++-- code/src/Core/App.php | 3 +-- code/src/Service/RequestHandler.php | 3 +-- 3 files changed, 4 insertions(+), 6 deletions(-) diff --git a/code/src/Controller/IndexController.php b/code/src/Controller/IndexController.php index ff84b06aa..954bc5aa1 100644 --- a/code/src/Controller/IndexController.php +++ b/code/src/Controller/IndexController.php @@ -12,8 +12,8 @@ public function __construct() { - $this->stringService = new StringService; - $this->sessionService = new SessionService; + $this->stringService = new StringService(); + $this->sessionService = new SessionService(); } public function handleStringRequest(): string diff --git a/code/src/Core/App.php b/code/src/Core/App.php index 8f0b1c04b..27640de55 100644 --- a/code/src/Core/App.php +++ b/code/src/Core/App.php @@ -6,10 +6,9 @@ readonly class App { - private RequestHandler $handler; - function __construct() + public function __construct() { $this->handler = new RequestHandler(); } diff --git a/code/src/Service/RequestHandler.php b/code/src/Service/RequestHandler.php index 52e958c08..3e4b8e63d 100644 --- a/code/src/Service/RequestHandler.php +++ b/code/src/Service/RequestHandler.php @@ -6,10 +6,9 @@ readonly class RequestHandler { - private IndexController $controller; - function __construct() + public function __construct() { $this->controller = new IndexController();