При создании веб-приложения конечной целью является публикация его в интернете. Но это может показаться сложным, особенно если вы раньше никогда этим не занимались. Целью данного руководства как раз и будет рассказать об этапах публикации сайта на VDS-сервер (с последующим обеспечением его защиты), с применением технологий контейнеризации.
В этой статье будут рассматриваться несколько технологий, и если вы столкнетесь с незнакомыми понятиями, не стоит волноваться. По мере прочтения у вас сложится пазл в голове, и в дальнейшем вы сами сможете устанавливать сайт на VDS. Итогом станет полноценный сайт с доменным именем и SSL-сертификатом.
В этой статье примером будет выступать полноценный веб-сайт — Frontend (клиентская часть) на Next.js, Backend (серверная часть) на Django (python), контейнеризация через Docker.
Для этого мы выполним следующие шаги:
Начнем с главного:
.tw1.su
). О том, как его получить, вы можете прочитать в нашей документации.Рекомендуем прочитать данную статью полностью, перед тем как приступать к развертыванию. Это даст вам понять, что вас ждет и как лучше поступить в вашем конкретном случае.
Продолжим наше руководство по тому, как опубликовать сайт на VDS-сервере.
vds
Заказать сервер можно по этой ссылке.
Внизу вы видите самый дешевый комплект — 285 рублей в месяц за стандартный VDS с публичным IPv4. ОС желательно выбрать Ubuntu, так как именно для нее написано наше руководство.
Публичный IPv4 требуется, если ваша сеть не поддерживает IPv6-подключения.
После вам нужно будет подключиться по протоколу SSH к вашему серверу.
SSH — это сетевой протокол для удаленного управления операционной системой с помощью командной строки и передачи данных в зашифрованном виде.
Если у вас Linux, то вы можете просто ввести команду:
ssh root@<ip>
Как вы видите, по умолчанию мы подключаемся к root
. Это является брешью в безопасности (злоумышленник может получить доступ ко всей системе), поэтому позднее мы ее исправим.
А для Windows вы можете установить PuTTY — клиент SSH. Но необязательно использовать PuTTY, с версии Windows 10 1809 SSH-клиент доступен через PowerShell.
Нам нужно будет установить:
sudo apt install nginx certbot python3-certbot-nginx
sudo snap install docker
Docker мы будем устанавливать через Ubuntu Snaps, потому что благодаря ему автоматически будут установлены все нужные зависимости для работы Docker-compose. Ubuntu Snaps — это универсальный формат пакетов, который не зависит от конкретного Linux-дистрибутива.
Итак, вы решили опубликовать сайт на VDS-сервере. Но стоит подумать о защите сервера, иначе злоумышленники могут получить несанкционированный доступ к системе.
Как сказано выше, одна из проблем — через SSH мы сразу подключаемся к суперпользователю (root
), то есть удаленно администрируем сервер без ограничений. А этим может воспользоваться злоумышленник.
Поэтому мы заведем другого пользователя и отключим доступ к SSH суперпользователю.
Для этого существует команда useradd
:
useradd [options] <username>
После нужно задать пароль новому пользователю и добавить его в группу sudo
:
passwd <username>
usermod -aG sudo <username>
Группа sudo
нужна для того, чтобы мы могли работать с системой.
Чтобы мы могли подключаться к серверу через ключ, а не через пароль, нам нужно его сгенерировать. Иначе останется возможность утечки пароля администратора:
ssh-keygen -t rsa
Затем скопируйте публичный ключ (~/.ssh/id_rsa.pub
) и вставьте его в секцию SSH-ключей сервера на Timeweb Cloud.
Также давайте настроим SSH-ключ для пользователя.
Есть разные программы, реализующие подключение по SSH, такие, как lsh и Dropbear, но самой популярной является OpenSSH. Установка клиента OpenSSH на Ubuntu:
sudo apt install openssh
Запуск демона SSH (sshd
) на сервере под Ubuntu:
sudo systemctl start ssh
sudo systemctl enable ssh
После вы должны создать на сервере директорию SSH в домашней директории юзера (из-под аккаунта администратора) и добавить публичный ключ, который мы сгенерировали ранее, в файл authorized_keys
. А также изменить права:
mkdir -p /home/<username>/.ssh && touch /home/<username>/.ssh/authorized_keys
chmod 700 /home/<username>/.ssh && chmod 600 /home/<username>/.ssh/authorized_keys
chown -R <username>:<username> /home/<username>/
chown -R <username>:<username> /home/<username>/.ssh
После отредактируем файл /etc/ssh/sshd_config
:
PermitRootLogin no # изменить с yes на no
PasswordAuthentication no # изменить с yes на no
Эти параметры запретят входить через root
и аутентификацию по паролю.
Брандмауэр (Firewall), грубо говоря, блокирует часть портов для защиты от посторонних подключений. В Ubuntu идет встроенный ufw, поэтому настроим его:
sudo ufw allow ssh # разрешаем ssh
sudo ufw allow 'Nginx Full' # разрешаем Nginx
sudo ufw enable # запускаем
Fail2Ban — это утилита для защиты от брутфорса (bruteforce — грубая сила, взлом методом перебора паролей). Она анализирует логи и попытки входа и может блокировать IP-адреса в зависимости от правил. Например, после 5 неудачных попыток в течение 10 минут блокируем IP-адрес на 2 часа.
Установка:
sudo apt install fail2ban
systemctl start fail2ban # запускаем
systemctl enable fail2ban # включаем автозагрузку
И редактируем конфиг /etc/fail2ban/jail.conf
:
[DEFAULT]
ignorecommand =
bantime = 120m
findtime = 10m
maxretry = 9
Если в течение 10 минут было 9 попыток войти, мы блокируем IP на 120 минут.
Нам надо перезапустить сервис fail2ban, так как мы отредактировали конфигурацию. Сделать это можно через systemctl
(утилита для управления системой инициализации systemd
):
sudo systemctl restart fail2ban
Основная часть установки сайта на VDS-сервер — это настройка контейнеризации через Docker.
Docker — это платформа для создания контейнеров с открытым исходным кодом.
Контейнеры Docker — это стандартизированные, изолированные и портативные пакеты программного обеспечения, которые включают в себя всё необходимое для запуска приложения, включая код, среду выполнения, системные инструменты, библиотеки и настройки.
Для этого туториала был создан репозиторий, который содержит два каталога: frontend
и backend
.
Директория frontend
содержит код для Next.js-приложения, когда как backend
отвечает за серверную часть (Django).
Чтобы начать, клонируйте репозиторий-шаблон:
git clone https://github.com/alexeev-prog/website-deploy-tutorial
И измените в backend/backend/settings.py
константу ALLOWED_HOSTS
:
ALLOWED_HOSTS = [
"109.68.212.254", 'hardtobecoder.tw1.su', 'localhost', '0.0.0.0'
]
109.68.212.254
замените на IP-адрес сервера, hardtobecoder.tw1.su
— на ваш домен.
В директории frontend
есть файл Dockerfile, который отвечает за установку модулей и запуск приложения:
FROM node:20-alpine
WORKDIR /usr/app
COPY ./ /usr/app/
COPY package.json /usr/app/
RUN npm install --legacy-peer-deps
COPY . /usr/app
RUN npm run build
CMD ["npm", "run", "start"]
Он задействует Node.JS и пакетный менеджер npm
для установки и запуска.
Также есть файл frontend/.Dockerignore
, который копирует frontend/.gitignore
.
Теперь рассмотрим контейнер для серверной части на Django. Аналогично с Frontend (клиентская часть), есть Dockerfile, который выглядит следующим образом:
FROM python:3.9
ENV PYTHONBUFFERED 1
ENV PYTHONDONTWRITEBYTECODE 1
copy requirements.txt .
RUN python -m pip install -r requirements.txt
RUN mkdir app
WORKDIR /app
COPY . /app
CMD gunicorn --workers 3 -b 0.0.0.0:8000 backend.wsgi
Данный Docker-контейнер будет запускать gunicorn
— WSGI-сервер. Позднее мы подключим к нему Nginx как обратный прокси-сервер.
И точно также существует файл backend/.dockerignore
.
Для того, чтобы наши Docker-контейнеры работали вместе, есть файл Docker-compose.yml
:
version: '3.8'
services:
frontend:
build:
context: ./frontend
dockerfile: Dockerfile
ports:
— "3000:3000"
depends_on:
— backend
backend:
build:
context: ./backend
dockerfile: Dockerfile
ports:
— "8000:8000"
Docker-compose — этой скрипт на Python, который позволяет запускать несколько Docker-контейнеров и настраивать их совместную работу.
Nginx — это веб-сервер. При обслуживании Django-приложения Nginx выступает в качестве обратного прокси-сервера: отвечает за обработку входящих запросов и их переадресацию.
В директории nginx
есть два файла — default
и create.sh
. default
является файлом конфигурации нашего сервера:
server {
listen 80;
server_name hardtobecoder.tw1.su;
location / {
proxy_pass http://localhost:8000;
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;
}
}
Замените hardtobecoder.tw1.su
на ваш домен, а 8000
на нужный вам порт (в зависимости от того, какой вы указали в Dockerfile Backend’а или Frontend’а).
После этого посмотрим файл create.sh
— баш-скрипт для установки и настройки прокси-сервера Nginx и загрузки SSL-сертификата:
sudo rm -rf /etc/nginx/sites-enabled/default # удаляем старую конфигурацию
sudo cp default -r /etc/nginx/sites-available/ # копируем конфигурацию
sudo ln -s /etc/nginx/sites-available/default /etc/nginx/sites-enabled # создаем ссылку в каталог включенных сайтов
sudo certbot --nginx -d hardtobecoder.tw1.su # генерируем SSL-сертификат
sudo nginx -t # проверяем конфигурацию
sudo nginx -s reload # перезагрузка Nginx
sudo systemctl restart Nginx # перезагрузка сервиса Nginx
Не забудьте изменить hardtobecoder.tw1.su
на ваш домен.
Если вы все настроили правильно, перейдите в директорию с файлом Docker-compose.yml
и запустите следующие команды:
sudo docker-compose build
sudo docker-compose up -d
cd nginx
./create.sh
После того как вы перейдете на адрес сайта, вы увидите, что ваше приложение доступно через HTTPS благодаря домену и SSL-сертификату!
Эти команды соберут контейнеры, запустят в режиме демона, и в конце будет включен и настроен Nginx.
Выгодные тарифы на VDS/VPS в Timeweb Cloud
Сайт успешно опубликован! Вы теперь знаете, как поставить сайт на VDS: как настроить сервер, Docker-контейнеры, запускать веб-приложения на Python или JavaScript. Если вам показалось это руководство полезным, поделитесь им с друзьями или коллегами. Если у вас возникли проблемы, перепроверьте ваши действия или напишите в поддержку Timeweb Cloud.