Add Docker support with Nginx, PHP-FPM and MySQL
This commit is contained in:
52
.env.example
52
.env.example
@@ -1,44 +1,34 @@
|
|||||||
# Nomina Pegaso - Configuración
|
# Application - GENERAR CON: php artisan key:generate
|
||||||
APP_NAME="Nomina Pegaso"
|
APP_NAME="Nomina Ventas"
|
||||||
APP_ENV=local
|
APP_ENV=local
|
||||||
APP_KEY=base64:OKIa4LjLBL6WSZ7HZXKDVpFwJ/KMI6bljtmxWQEcHKY=
|
APP_KEY=
|
||||||
APP_DEBUG=true
|
APP_DEBUG=true
|
||||||
APP_URL=http://nomina-pegaso.casa
|
APP_URL=http://localhost:80
|
||||||
|
|
||||||
APP_LOCALE=es
|
|
||||||
APP_FALLBACK_LOCALE=es
|
|
||||||
APP_FAKER_LOCALE=es_MX
|
|
||||||
|
|
||||||
APP_MAINTENANCE_DRIVER=file
|
|
||||||
|
|
||||||
BCRYPT_ROUNDS=12
|
|
||||||
|
|
||||||
LOG_CHANNEL=stack
|
LOG_CHANNEL=stack
|
||||||
LOG_STACK=single
|
|
||||||
LOG_DEPRECATIONS_CHANNEL=null
|
|
||||||
LOG_LEVEL=debug
|
LOG_LEVEL=debug
|
||||||
|
|
||||||
|
# Database - Docker (OBLIGATORIOS - sin valores por defecto)
|
||||||
DB_CONNECTION=mysql
|
DB_CONNECTION=mysql
|
||||||
DB_HOST=10.10.4.17
|
DB_HOST=db
|
||||||
DB_PORT=3391
|
DB_PORT=3306
|
||||||
DB_DATABASE=nomina_pegaso
|
DB_DATABASE=nomina_ventas
|
||||||
DB_USERNAME=nickpons666
|
DB_USERNAME=
|
||||||
DB_PASSWORD=MiPo6425@@
|
DB_PASSWORD=
|
||||||
|
|
||||||
|
# MySQL Root (OBLIGATORIO)
|
||||||
|
MYSQL_ROOT_PASSWORD=
|
||||||
|
|
||||||
|
# Session & Cache
|
||||||
SESSION_DRIVER=database
|
SESSION_DRIVER=database
|
||||||
SESSION_LIFETIME=120
|
SESSION_LIFETIME=120
|
||||||
SESSION_ENCRYPT=true
|
|
||||||
SESSION_PATH=/
|
|
||||||
SESSION_DOMAIN=null
|
|
||||||
|
|
||||||
BROADCAST_CONNECTION=log
|
|
||||||
FILESYSTEM_DISK=local
|
|
||||||
QUEUE_CONNECTION=database
|
|
||||||
|
|
||||||
CACHE_STORE=database
|
CACHE_STORE=database
|
||||||
|
|
||||||
VITE_APP_NAME="${APP_NAME}"
|
# Telegram Bot ( OPCIONAL )
|
||||||
|
TELEGRAM_BOT_TOKEN=
|
||||||
# Telegram Bot
|
|
||||||
TELEGRAM_BOT_TOKEN=8324407449:AAF2awMeZ9pgSIp0MvV1r5owu8lO7SEK70E
|
|
||||||
TELEGRAM_WEBHOOK_URL=
|
TELEGRAM_WEBHOOK_URL=
|
||||||
|
|
||||||
|
# Docker Build
|
||||||
|
APP_PORT=80
|
||||||
|
PUID=1000
|
||||||
|
PGID=1000
|
||||||
106
Dockerfile
Normal file
106
Dockerfile
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
# ============================================
|
||||||
|
# Laravel PHP-FPM 8.3 - Production Optimized
|
||||||
|
# ============================================
|
||||||
|
|
||||||
|
# Stage 1: Builder with Composer
|
||||||
|
FROM composer:2 AS composer-builder
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Copy composer files first for better layer caching
|
||||||
|
COPY composer.json composer.lock* ./
|
||||||
|
RUN composer install --no-dev --optimize-autoloader --no-interaction --prefer-dist
|
||||||
|
|
||||||
|
# Stage 2: PHP-FPM Production Image
|
||||||
|
FROM php:8.3-fpm-bookworm
|
||||||
|
|
||||||
|
# Build arguments for dynamic user UID/GID
|
||||||
|
ARG PUID=1000
|
||||||
|
ARG PGID=1000
|
||||||
|
|
||||||
|
# Labels
|
||||||
|
LABEL maintainer="dev@local.dev" \
|
||||||
|
description="Laravel PHP-FPM 8.3 production container"
|
||||||
|
|
||||||
|
# Install system dependencies
|
||||||
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
|
curl \
|
||||||
|
libzip-dev \
|
||||||
|
libxml2-dev \
|
||||||
|
libcurl4-openssl-dev \
|
||||||
|
libgd-dev \
|
||||||
|
libfreetype6-dev \
|
||||||
|
libjpeg-dev \
|
||||||
|
libpng-dev \
|
||||||
|
libonig-dev \
|
||||||
|
libmcrypt-dev \
|
||||||
|
libsqlite3-dev \
|
||||||
|
unzip \
|
||||||
|
git \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# Install PHP extensions
|
||||||
|
RUN docker-php-ext-configure gd --with-freetype --with-jpeg \
|
||||||
|
&& docker-php-ext-install -j$(nproc) \
|
||||||
|
pdo \
|
||||||
|
pdo_mysql \
|
||||||
|
mbstring \
|
||||||
|
xml \
|
||||||
|
curl \
|
||||||
|
zip \
|
||||||
|
gd \
|
||||||
|
opcache
|
||||||
|
|
||||||
|
# Configure PHP for production
|
||||||
|
RUN echo "opcache.enable=1" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "opcache.memory_consumption=128" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "opcache.interned_strings_buffer=8" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "opcache.max_accelerated_files=10000" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "opcache.revalidate_freq=2" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "opcache.fast_shutdown=1" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "upload_max_filesize=100M" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini \
|
||||||
|
&& echo "post_max_size=100M" >> /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini
|
||||||
|
|
||||||
|
# Copy Composer from builder stage
|
||||||
|
COPY --from=composer-builder /usr/bin/composer /usr/bin/composer
|
||||||
|
COPY --from=composer-builder /root/.composer /root/.composer
|
||||||
|
|
||||||
|
# Create Laravel storage directories with proper permissions
|
||||||
|
RUN mkdir -p /var/www/html/storage/framework/{cache,sessions,views} \
|
||||||
|
&& mkdir -p /var/www/html/storage/logs \
|
||||||
|
&& mkdir -p /var/www/html/bootstrap/cache \
|
||||||
|
&& chmod -R 775 /var/www/html/storage \
|
||||||
|
&& chmod -R 775 /var/www/html/bootstrap/cache
|
||||||
|
|
||||||
|
# Create system user for Laravel (dynamic UID/GID)
|
||||||
|
RUN groupadd -g ${PGID} laravel && \
|
||||||
|
useradd -u ${PUID} -g laravel -m -s /bin/bash laravel
|
||||||
|
|
||||||
|
# Set working directory
|
||||||
|
WORKDIR /var/www/html
|
||||||
|
|
||||||
|
# Copy application files
|
||||||
|
COPY --chown=laravel:laravel . /var/www/html
|
||||||
|
|
||||||
|
# Fix storage permissions
|
||||||
|
RUN chown -R laravel:laravel /var/www/html/storage \
|
||||||
|
&& chown -R laravel:laravel /var/www/html/bootstrap/cache \
|
||||||
|
&& chmod -R 775 /var/www/html/storage \
|
||||||
|
&& chmod -R 775 /var/www/html/bootstrap/cache
|
||||||
|
|
||||||
|
# Switch to non-root user
|
||||||
|
USER laravel
|
||||||
|
|
||||||
|
# Expose port 9000 (PHP-FPM)
|
||||||
|
EXPOSE 9000
|
||||||
|
|
||||||
|
# Health check
|
||||||
|
HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \
|
||||||
|
CMD php-fpm-healthcheck || exit 1
|
||||||
|
|
||||||
|
# Create healthcheck script
|
||||||
|
RUN echo '#!/bin/sh' > /usr/local/bin/php-fpm-healthcheck \
|
||||||
|
&& echo 'SCRIPT_NAME=/health.php SCRIPT_FILENAME=/health.php REQUEST_METHOD=GET cgi-fcgi -bind -connect 127.0.0.1:9000 > /dev/null 2>&1' >> /usr/local/bin/php-fpm-healthcheck \
|
||||||
|
&& chmod +x /usr/local/bin/php-fpm-healthcheck
|
||||||
|
|
||||||
|
CMD ["php-fpm"]
|
||||||
95
docker-compose.yml
Normal file
95
docker-compose.yml
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
services:
|
||||||
|
nginx:
|
||||||
|
image: nginx:alpine
|
||||||
|
container_name: nomina_nginx
|
||||||
|
ports:
|
||||||
|
- "${APP_PORT:-80}:80"
|
||||||
|
volumes:
|
||||||
|
- ./public:/var/www/html/public:ro
|
||||||
|
- ./docker/nginx/nginx.conf:/etc/nginx/conf.d/default.conf:ro
|
||||||
|
depends_on:
|
||||||
|
- app
|
||||||
|
networks:
|
||||||
|
- app_network
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "wget", "-q", "--spider", "http://localhost"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: '0.5'
|
||||||
|
memory: 256M
|
||||||
|
|
||||||
|
app:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
args:
|
||||||
|
PUID: ${PUID:-1000}
|
||||||
|
PGID: ${PGID:-1000}
|
||||||
|
container_name: nomina_app
|
||||||
|
environment:
|
||||||
|
- APP_ENV=${APP_ENV:-production}
|
||||||
|
- APP_DEBUG=${APP_DEBUG:-false}
|
||||||
|
- APP_KEY=${APP_KEY}
|
||||||
|
- APP_URL=${APP_URL:-http://localhost}
|
||||||
|
- DB_CONNECTION=${DB_CONNECTION:-mysql}
|
||||||
|
- DB_HOST=${DB_HOST:-db}
|
||||||
|
- DB_PORT=${DB_PORT:-3306}
|
||||||
|
- DB_DATABASE=${DB_DATABASE:-nomina_ventas}
|
||||||
|
- DB_USERNAME=${DB_USERNAME}
|
||||||
|
- DB_PASSWORD=${DB_PASSWORD}
|
||||||
|
- SESSION_DRIVER=${SESSION_DRIVER:-database}
|
||||||
|
- CACHE_STORE=${CACHE_STORE:-database}
|
||||||
|
- TELEGRAM_BOT_TOKEN=${TELEGRAM_BOT_TOKEN}
|
||||||
|
- TELEGRAM_WEBHOOK_URL=${TELEGRAM_WEBHOOK_URL}
|
||||||
|
volumes:
|
||||||
|
- ./:/var/www/html:delegated
|
||||||
|
- storage_data:/var/www/html/storage
|
||||||
|
depends_on:
|
||||||
|
db:
|
||||||
|
condition: service_healthy
|
||||||
|
networks:
|
||||||
|
- app_network
|
||||||
|
restart: unless-stopped
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: '1.0'
|
||||||
|
memory: 512M
|
||||||
|
|
||||||
|
db:
|
||||||
|
image: mysql:8.0
|
||||||
|
container_name: nomina_mysql
|
||||||
|
environment:
|
||||||
|
- MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD}
|
||||||
|
- MYSQL_DATABASE=${DB_DATABASE:-nomina_ventas}
|
||||||
|
- MYSQL_USER=${DB_USERNAME}
|
||||||
|
- MYSQL_PASSWORD=${DB_PASSWORD}
|
||||||
|
volumes:
|
||||||
|
- mysql_data:/var/lib/mysql
|
||||||
|
networks:
|
||||||
|
- app_network
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "mysqladmin", "ping", "-h", "localhost"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
start_period: 30s
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
cpus: '1.0'
|
||||||
|
memory: 1G
|
||||||
|
|
||||||
|
networks:
|
||||||
|
app_network:
|
||||||
|
driver: bridge
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
mysql_data:
|
||||||
|
storage_data:
|
||||||
57
docker/nginx/nginx.conf
Normal file
57
docker/nginx/nginx.conf
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
server_name localhost;
|
||||||
|
root /var/www/html/public;
|
||||||
|
index index.php index.html;
|
||||||
|
|
||||||
|
client_max_body_size 100M;
|
||||||
|
client_body_timeout 300s;
|
||||||
|
proxy_read_timeout 300s;
|
||||||
|
proxy_connect_timeout 300s;
|
||||||
|
send_timeout 300s;
|
||||||
|
|
||||||
|
access_log /var/log/nginx/access.log;
|
||||||
|
error_log /var/log/nginx/error.log;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
try_files $uri $uri/ /index.php?$query_string;
|
||||||
|
}
|
||||||
|
|
||||||
|
location ~* \.(jpg|jpeg|png|gif|ico|css|js|svg|woff|woff2|ttf|eot)$ {
|
||||||
|
expires 1y;
|
||||||
|
add_header Cache-Control "public, immutable";
|
||||||
|
access_log off;
|
||||||
|
}
|
||||||
|
|
||||||
|
location ~ \.php$ {
|
||||||
|
fastcgi_pass app:9000;
|
||||||
|
fastcgi_index index.php;
|
||||||
|
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||||
|
fastcgi_param PATH_INFO $fastcgi_path_info;
|
||||||
|
include fastcgi_params;
|
||||||
|
}
|
||||||
|
|
||||||
|
add_header X-Frame-Options "SAMEORIGIN" always;
|
||||||
|
add_header X-Content-Type-Options "nosniff" always;
|
||||||
|
add_header X-XSS-Protection "1; mode=block" always;
|
||||||
|
add_header Referrer-Policy "strict-origin-when-cross-origin" always;
|
||||||
|
add_header Permissions-Policy "camera=(), microphone=(), geolocation=()" always;
|
||||||
|
|
||||||
|
location ~ /\.(?!well-known).* {
|
||||||
|
deny all;
|
||||||
|
return 404;
|
||||||
|
}
|
||||||
|
|
||||||
|
location ~ /(\.env|composer\.json|composer\.lock|storage|\.git) {
|
||||||
|
deny all;
|
||||||
|
return 404;
|
||||||
|
}
|
||||||
|
|
||||||
|
gzip on;
|
||||||
|
gzip_vary on;
|
||||||
|
gzip_proxied any;
|
||||||
|
gzip_comp_level 6;
|
||||||
|
gzip_types text/plain text/css text/xml application/json application/javascript application/xml+rss application/atom+xml image/svg+xml;
|
||||||
|
|
||||||
|
autoindex off;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user